jsonlogbytes.go 2.4 KB

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