2020-09-12 11:34:59 +02:00
|
|
|
package accesslog
|
|
|
|
|
|
|
|
import (
|
|
|
|
"io"
|
2020-09-13 01:56:22 +02:00
|
|
|
"strings"
|
|
|
|
|
|
|
|
jsoniter "github.com/json-iterator/go"
|
2020-09-12 11:34:59 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
// JSON is a Formatter type for JSON logs.
|
|
|
|
type JSON struct {
|
2020-09-13 01:56:22 +02:00
|
|
|
// Indent in spaces.
|
|
|
|
// Note that, if set to > 0 then jsoniter is used instead of easyjson.
|
|
|
|
Indent string
|
|
|
|
EscapeHTML bool
|
2020-09-12 11:34:59 +02:00
|
|
|
|
2020-09-13 01:56:22 +02:00
|
|
|
jsoniter jsoniter.API
|
|
|
|
ac *AccessLog
|
2020-09-12 11:34:59 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// SetOutput creates the json encoder writes to the "dest".
|
|
|
|
// It's called automatically by the middleware when this Formatter is used.
|
|
|
|
func (f *JSON) SetOutput(dest io.Writer) {
|
2020-09-13 01:56:22 +02:00
|
|
|
f.ac, _ = dest.(*AccessLog)
|
|
|
|
if indentStep := strings.Count(f.Indent, " "); indentStep > 0 {
|
|
|
|
f.jsoniter = jsoniter.Config{
|
|
|
|
TagKey: "json",
|
|
|
|
IndentionStep: indentStep,
|
|
|
|
EscapeHTML: f.EscapeHTML,
|
|
|
|
SortMapKeys: true,
|
|
|
|
}.Froze()
|
2020-09-12 11:34:59 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Format prints the logs in JSON format.
|
|
|
|
// Writes to the destination directly,
|
|
|
|
// locks on each Format call.
|
|
|
|
func (f *JSON) Format(log *Log) (bool, error) {
|
2020-09-13 01:56:22 +02:00
|
|
|
if f.jsoniter != nil {
|
|
|
|
b, err := f.jsoniter.Marshal(log)
|
|
|
|
if err != nil {
|
|
|
|
return true, err
|
|
|
|
}
|
|
|
|
f.ac.Write(append(b, newLine))
|
|
|
|
return true, nil
|
2020-09-12 11:34:59 +02:00
|
|
|
}
|
|
|
|
|
2020-09-13 01:56:22 +02:00
|
|
|
err := f.writeEasyJSON(log)
|
2020-09-12 11:34:59 +02:00
|
|
|
return true, err
|
|
|
|
}
|