2023-06-21 10:10:45 +02:00
|
|
|
package logjson
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"sync"
|
|
|
|
|
|
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/lib/bytesutil"
|
|
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/lib/logger"
|
|
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/lib/logstorage"
|
|
|
|
"github.com/valyala/fastjson"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Parser parses a single JSON log message into Fields.
|
|
|
|
//
|
|
|
|
// See https://docs.victoriametrics.com/VictoriaLogs/keyConcepts.html#data-model
|
|
|
|
//
|
|
|
|
// Use GetParser() for obtaining the parser.
|
|
|
|
type Parser struct {
|
|
|
|
// Fields contains the parsed JSON line after Parse() call
|
|
|
|
//
|
|
|
|
// The Fields are valid until the next call to ParseLogMessage()
|
|
|
|
// or until the parser is returned to the pool with PutParser() call.
|
|
|
|
Fields []logstorage.Field
|
|
|
|
|
|
|
|
// p is used for fast JSON parsing
|
|
|
|
p fastjson.Parser
|
|
|
|
|
|
|
|
// buf is used for holding the backing data for Fields
|
|
|
|
buf []byte
|
|
|
|
|
|
|
|
// prefixBuf is used for holding the current key prefix
|
|
|
|
// when it is composed from multiple keys.
|
|
|
|
prefixBuf []byte
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Parser) reset() {
|
|
|
|
fields := p.Fields
|
|
|
|
for i := range fields {
|
|
|
|
lf := &fields[i]
|
|
|
|
lf.Name = ""
|
|
|
|
lf.Value = ""
|
|
|
|
}
|
|
|
|
p.Fields = fields[:0]
|
|
|
|
|
|
|
|
p.buf = p.buf[:0]
|
|
|
|
p.prefixBuf = p.prefixBuf[:0]
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetParser returns Parser ready to parse JSON lines.
|
|
|
|
//
|
|
|
|
// Return the parser to the pool when it is no longer needed by calling PutParser().
|
|
|
|
func GetParser() *Parser {
|
|
|
|
v := parserPool.Get()
|
|
|
|
if v == nil {
|
|
|
|
return &Parser{}
|
|
|
|
}
|
|
|
|
return v.(*Parser)
|
|
|
|
}
|
|
|
|
|
|
|
|
// PutParser returns the parser to the pool.
|
|
|
|
//
|
|
|
|
// The parser cannot be used after returning to the pool.
|
|
|
|
func PutParser(p *Parser) {
|
|
|
|
p.reset()
|
|
|
|
parserPool.Put(p)
|
|
|
|
}
|
|
|
|
|
|
|
|
var parserPool sync.Pool
|
|
|
|
|
|
|
|
// ParseLogMessage parses the given JSON log message msg into p.Fields.
|
|
|
|
//
|
|
|
|
// The p.Fields remains valid until the next call to ParseLogMessage() or PutParser().
|
|
|
|
func (p *Parser) ParseLogMessage(msg []byte) error {
|
|
|
|
s := bytesutil.ToUnsafeString(msg)
|
|
|
|
v, err := p.p.Parse(s)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("cannot parse json: %w", err)
|
|
|
|
}
|
|
|
|
if t := v.Type(); t != fastjson.TypeObject {
|
|
|
|
return fmt.Errorf("expecting json dictionary; got %s", t)
|
|
|
|
}
|
|
|
|
p.reset()
|
|
|
|
p.Fields, p.buf, p.prefixBuf = appendLogFields(p.Fields, p.buf, p.prefixBuf, v)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-06-22 04:39:22 +02:00
|
|
|
// RenameField renames field with the oldName to newName in p.Fields
|
|
|
|
func (p *Parser) RenameField(oldName, newName string) {
|
|
|
|
if oldName == "" {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
fields := p.Fields
|
|
|
|
for i := range fields {
|
|
|
|
f := &fields[i]
|
|
|
|
if f.Name == oldName {
|
|
|
|
f.Name = newName
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-21 10:10:45 +02:00
|
|
|
func appendLogFields(dst []logstorage.Field, dstBuf, prefixBuf []byte, v *fastjson.Value) ([]logstorage.Field, []byte, []byte) {
|
|
|
|
o := v.GetObject()
|
|
|
|
o.Visit(func(k []byte, v *fastjson.Value) {
|
|
|
|
t := v.Type()
|
|
|
|
switch t {
|
|
|
|
case fastjson.TypeNull:
|
|
|
|
// Skip nulls
|
|
|
|
case fastjson.TypeObject:
|
|
|
|
// Flatten nested JSON objects.
|
|
|
|
// For example, {"foo":{"bar":"baz"}} is converted to {"foo.bar":"baz"}
|
|
|
|
prefixLen := len(prefixBuf)
|
|
|
|
prefixBuf = append(prefixBuf, k...)
|
|
|
|
prefixBuf = append(prefixBuf, '.')
|
|
|
|
dst, dstBuf, prefixBuf = appendLogFields(dst, dstBuf, prefixBuf, v)
|
|
|
|
prefixBuf = prefixBuf[:prefixLen]
|
|
|
|
case fastjson.TypeArray, fastjson.TypeNumber, fastjson.TypeTrue, fastjson.TypeFalse:
|
|
|
|
// Convert JSON arrays, numbers, true and false values to their string representation
|
|
|
|
dstBufLen := len(dstBuf)
|
|
|
|
dstBuf = v.MarshalTo(dstBuf)
|
|
|
|
value := dstBuf[dstBufLen:]
|
|
|
|
dst, dstBuf = appendLogField(dst, dstBuf, prefixBuf, k, value)
|
|
|
|
case fastjson.TypeString:
|
|
|
|
// Decode JSON strings
|
|
|
|
dstBufLen := len(dstBuf)
|
|
|
|
dstBuf = append(dstBuf, v.GetStringBytes()...)
|
|
|
|
value := dstBuf[dstBufLen:]
|
|
|
|
dst, dstBuf = appendLogField(dst, dstBuf, prefixBuf, k, value)
|
|
|
|
default:
|
|
|
|
logger.Panicf("BUG: unexpected JSON type: %s", t)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
return dst, dstBuf, prefixBuf
|
|
|
|
}
|
|
|
|
|
|
|
|
func appendLogField(dst []logstorage.Field, dstBuf, prefixBuf, k, value []byte) ([]logstorage.Field, []byte) {
|
|
|
|
dstBufLen := len(dstBuf)
|
|
|
|
dstBuf = append(dstBuf, prefixBuf...)
|
|
|
|
dstBuf = append(dstBuf, k...)
|
|
|
|
name := dstBuf[dstBufLen:]
|
|
|
|
|
|
|
|
dst = append(dst, logstorage.Field{
|
|
|
|
Name: bytesutil.ToUnsafeString(name),
|
|
|
|
Value: bytesutil.ToUnsafeString(value),
|
|
|
|
})
|
|
|
|
return dst, dstBuf
|
|
|
|
}
|