jsonlogbytes.go 2.4 KB

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