jsonlogbytes.go 2.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135
  1. package jsonlog
  2. import (
  3. "bytes"
  4. "encoding/json"
  5. "time"
  6. "unicode/utf8"
  7. )
  8. // JSONLogs marshals encoded JSONLog objects
  9. type JSONLogs struct {
  10. Log []byte `json:"log,omitempty"`
  11. Stream string `json:"stream,omitempty"`
  12. Created time.Time `json:"time"`
  13. Tag string `json:"tag,omitempty"`
  14. // json-encoded bytes
  15. RawAttrs json.RawMessage `json:"attrs,omitempty"`
  16. }
  17. // MarshalJSONBuf is an optimized JSON marshaller that avoids reflection
  18. // and unnecessary allocation.
  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. ffjsonWriteJSONBytesAsString(buf, []byte(mj.Stream))
  35. }
  36. if len(mj.Tag) > 0 {
  37. if first {
  38. first = false
  39. } else {
  40. buf.WriteString(`,`)
  41. }
  42. buf.WriteString(`"tag":`)
  43. ffjsonWriteJSONBytesAsString(buf, []byte(mj.Tag))
  44. }
  45. if len(mj.RawAttrs) > 0 {
  46. if first {
  47. first = false
  48. } else {
  49. buf.WriteString(`,`)
  50. }
  51. buf.WriteString(`"attrs":`)
  52. buf.Write(mj.RawAttrs)
  53. }
  54. if !first {
  55. buf.WriteString(`,`)
  56. }
  57. created, err := fastTimeMarshalJSON(mj.Created)
  58. if err != nil {
  59. return err
  60. }
  61. buf.WriteString(`"time":`)
  62. buf.WriteString(created)
  63. buf.WriteString(`}`)
  64. return nil
  65. }
  66. func ffjsonWriteJSONBytesAsString(buf *bytes.Buffer, s []byte) {
  67. const hex = "0123456789abcdef"
  68. buf.WriteByte('"')
  69. start := 0
  70. for i := 0; i < len(s); {
  71. if b := s[i]; b < utf8.RuneSelf {
  72. if 0x20 <= b && b != '\\' && b != '"' && b != '<' && b != '>' && b != '&' {
  73. i++
  74. continue
  75. }
  76. if start < i {
  77. buf.Write(s[start:i])
  78. }
  79. switch b {
  80. case '\\', '"':
  81. buf.WriteByte('\\')
  82. buf.WriteByte(b)
  83. case '\n':
  84. buf.WriteByte('\\')
  85. buf.WriteByte('n')
  86. case '\r':
  87. buf.WriteByte('\\')
  88. buf.WriteByte('r')
  89. default:
  90. buf.WriteString(`\u00`)
  91. buf.WriteByte(hex[b>>4])
  92. buf.WriteByte(hex[b&0xF])
  93. }
  94. i++
  95. start = i
  96. continue
  97. }
  98. c, size := utf8.DecodeRune(s[i:])
  99. if c == utf8.RuneError && size == 1 {
  100. if start < i {
  101. buf.Write(s[start:i])
  102. }
  103. buf.WriteString(`\ufffd`)
  104. i += size
  105. start = i
  106. continue
  107. }
  108. if c == '\u2028' || c == '\u2029' {
  109. if start < i {
  110. buf.Write(s[start:i])
  111. }
  112. buf.WriteString(`\u202`)
  113. buf.WriteByte(hex[c&0xF])
  114. i += size
  115. start = i
  116. continue
  117. }
  118. i += size
  119. }
  120. if start < len(s) {
  121. buf.Write(s[start:])
  122. }
  123. buf.WriteByte('"')
  124. }