jsonlogbytes.go 2.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115
  1. package jsonlog
  2. import (
  3. "bytes"
  4. "unicode/utf8"
  5. )
  6. // JSONLogBytes is based on JSONLog.
  7. // It allows marshalling JSONLog from Log as []byte
  8. // and an already marshalled Created timestamp.
  9. type JSONLogBytes struct {
  10. Log []byte `json:"log,omitempty"`
  11. Stream string `json:"stream,omitempty"`
  12. Created string `json:"time"`
  13. }
  14. // MarshalJSONBuf is based on the same method from JSONLog
  15. // It has been modified to take into account the necessary changes.
  16. func (mj *JSONLogBytes) MarshalJSONBuf(buf *bytes.Buffer) error {
  17. var first = true
  18. buf.WriteString(`{`)
  19. if len(mj.Log) != 0 {
  20. if first == true {
  21. first = false
  22. } else {
  23. buf.WriteString(`,`)
  24. }
  25. buf.WriteString(`"log":`)
  26. ffjson_WriteJsonBytesAsString(buf, mj.Log)
  27. }
  28. if len(mj.Stream) != 0 {
  29. if first == true {
  30. first = false
  31. } else {
  32. buf.WriteString(`,`)
  33. }
  34. buf.WriteString(`"stream":`)
  35. ffjson_WriteJsonString(buf, mj.Stream)
  36. }
  37. if first == true {
  38. first = false
  39. } else {
  40. buf.WriteString(`,`)
  41. }
  42. buf.WriteString(`"time":`)
  43. buf.WriteString(mj.Created)
  44. buf.WriteString(`}`)
  45. return nil
  46. }
  47. // This is based on ffjson_WriteJsonString. It has been changed
  48. // to accept a string passed as a slice of bytes.
  49. func ffjson_WriteJsonBytesAsString(buf *bytes.Buffer, s []byte) {
  50. const hex = "0123456789abcdef"
  51. buf.WriteByte('"')
  52. start := 0
  53. for i := 0; i < len(s); {
  54. if b := s[i]; b < utf8.RuneSelf {
  55. if 0x20 <= b && b != '\\' && b != '"' && b != '<' && b != '>' && b != '&' {
  56. i++
  57. continue
  58. }
  59. if start < i {
  60. buf.Write(s[start:i])
  61. }
  62. switch b {
  63. case '\\', '"':
  64. buf.WriteByte('\\')
  65. buf.WriteByte(b)
  66. case '\n':
  67. buf.WriteByte('\\')
  68. buf.WriteByte('n')
  69. case '\r':
  70. buf.WriteByte('\\')
  71. buf.WriteByte('r')
  72. default:
  73. buf.WriteString(`\u00`)
  74. buf.WriteByte(hex[b>>4])
  75. buf.WriteByte(hex[b&0xF])
  76. }
  77. i++
  78. start = i
  79. continue
  80. }
  81. c, size := utf8.DecodeRune(s[i:])
  82. if c == utf8.RuneError && size == 1 {
  83. if start < i {
  84. buf.Write(s[start:i])
  85. }
  86. buf.WriteString(`\ufffd`)
  87. i += size
  88. start = i
  89. continue
  90. }
  91. if c == '\u2028' || c == '\u2029' {
  92. if start < i {
  93. buf.Write(s[start:i])
  94. }
  95. buf.WriteString(`\u202`)
  96. buf.WriteByte(hex[c&0xF])
  97. i += size
  98. start = i
  99. continue
  100. }
  101. i += size
  102. }
  103. if start < len(s) {
  104. buf.Write(s[start:])
  105. }
  106. buf.WriteByte('"')
  107. }