mirror of
https://github.com/zrepl/zrepl.git
synced 2025-01-22 14:18:38 +01:00
ebf209427a
should be refactored to logger one day so the implementation of ignoring is not duplicated to each outlet. refs #10
218 lines
5.0 KiB
Go
218 lines
5.0 KiB
Go
package cmd
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/json"
|
|
"fmt"
|
|
"github.com/go-logfmt/logfmt"
|
|
"github.com/pkg/errors"
|
|
"github.com/zrepl/zrepl/logger"
|
|
"time"
|
|
)
|
|
|
|
type EntryFormatter interface {
|
|
SetMetadataFlags(flags MetadataFlags)
|
|
Format(e *logger.Entry) ([]byte, error)
|
|
}
|
|
|
|
const (
|
|
FieldLevel = "level"
|
|
FieldMessage = "msg"
|
|
FieldTime = "time"
|
|
)
|
|
|
|
const (
|
|
logJobField string = "job"
|
|
logTaskField string = "task"
|
|
logFSField string = "filesystem"
|
|
logMapFromField string = "map_from"
|
|
logMapToField string = "map_to"
|
|
logIncFromField string = "inc_from"
|
|
logIncToField string = "inc_to"
|
|
)
|
|
|
|
type NoFormatter struct{}
|
|
|
|
func (f NoFormatter) SetMetadataFlags(flags MetadataFlags) {}
|
|
|
|
func (f NoFormatter) Format(e *logger.Entry) ([]byte, error) {
|
|
return []byte(e.Message), nil
|
|
}
|
|
|
|
type HumanFormatter struct {
|
|
metadataFlags MetadataFlags
|
|
ignoreFields map[string]bool
|
|
}
|
|
|
|
func (f *HumanFormatter) SetMetadataFlags(flags MetadataFlags) {
|
|
f.metadataFlags = flags
|
|
}
|
|
|
|
func (f *HumanFormatter) SetIgnoreFields(ignore []string) {
|
|
if ignore == nil {
|
|
f.ignoreFields = nil
|
|
return
|
|
}
|
|
f.ignoreFields = make(map[string]bool, len(ignore))
|
|
|
|
for _, field := range ignore {
|
|
f.ignoreFields[field] = true
|
|
}
|
|
}
|
|
|
|
func (f *HumanFormatter) ignored(field string) bool {
|
|
return f.ignoreFields != nil && f.ignoreFields[field]
|
|
}
|
|
|
|
func (f *HumanFormatter) Format(e *logger.Entry) (out []byte, err error) {
|
|
|
|
var line bytes.Buffer
|
|
|
|
if f.metadataFlags&MetadataTime != 0 {
|
|
fmt.Fprintf(&line, "%s ", e.Time.Format(time.RFC3339))
|
|
}
|
|
if f.metadataFlags&MetadataLevel != 0 {
|
|
fmt.Fprintf(&line, "[%s]", e.Level.Short())
|
|
}
|
|
|
|
prefixFields := []string{logJobField, logTaskField, logFSField}
|
|
prefixed := make(map[string]bool, len(prefixFields)+2)
|
|
for _, field := range prefixFields {
|
|
val, ok := e.Fields[field].(string)
|
|
if ok {
|
|
if !f.ignored(field) {
|
|
fmt.Fprintf(&line, "[%s]", val)
|
|
prefixed[field] = true
|
|
}
|
|
} else {
|
|
break
|
|
}
|
|
}
|
|
// even more prefix fields
|
|
mapFrom, mapFromOk := e.Fields[logMapFromField].(string)
|
|
mapTo, mapToOk := e.Fields[logMapToField].(string)
|
|
if mapFromOk && mapToOk && !f.ignored(logMapFromField) && !f.ignored(logMapToField) {
|
|
fmt.Fprintf(&line, "[%s => %s]", mapFrom, mapTo)
|
|
prefixed[logMapFromField], prefixed[logMapToField] = true, true
|
|
}
|
|
incFrom, incFromOk := e.Fields[logIncFromField].(string)
|
|
incTo, incToOk := e.Fields[logIncToField].(string)
|
|
if incFromOk && incToOk && !f.ignored(logIncFromField) && !f.ignored(logMapToField) {
|
|
fmt.Fprintf(&line, "[%s => %s]", incFrom, incTo)
|
|
prefixed[logIncFromField], prefixed[logIncToField] = true, true
|
|
}
|
|
|
|
if line.Len() > 0 {
|
|
fmt.Fprint(&line, ": ")
|
|
}
|
|
fmt.Fprint(&line, e.Message)
|
|
|
|
if len(e.Fields)-len(prefixed) > 0 {
|
|
fmt.Fprint(&line, " ")
|
|
enc := logfmt.NewEncoder(&line)
|
|
for field, value := range e.Fields {
|
|
if prefixed[field] || f.ignored(field) {
|
|
continue
|
|
}
|
|
if err := logfmtTryEncodeKeyval(enc, field, value); err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
}
|
|
|
|
return line.Bytes(), nil
|
|
}
|
|
|
|
type JSONFormatter struct {
|
|
metadataFlags MetadataFlags
|
|
}
|
|
|
|
func (f *JSONFormatter) SetMetadataFlags(flags MetadataFlags) {
|
|
f.metadataFlags = flags
|
|
}
|
|
|
|
func (f *JSONFormatter) Format(e *logger.Entry) ([]byte, error) {
|
|
data := make(logger.Fields, len(e.Fields)+3)
|
|
for k, v := range e.Fields {
|
|
switch v := v.(type) {
|
|
case error:
|
|
// Otherwise errors are ignored by `encoding/json`
|
|
// https://github.com/sirupsen/logrus/issues/137
|
|
data[k] = v.Error()
|
|
default:
|
|
_, err := json.Marshal(v)
|
|
if err != nil {
|
|
return nil, errors.Errorf("field is not JSON encodable: %s", k)
|
|
}
|
|
data[k] = v
|
|
}
|
|
}
|
|
|
|
data[FieldMessage] = e.Message
|
|
data[FieldTime] = e.Time.Format(time.RFC3339)
|
|
data[FieldLevel] = e.Level
|
|
|
|
return json.Marshal(data)
|
|
|
|
}
|
|
|
|
type LogfmtFormatter struct {
|
|
metadataFlags MetadataFlags
|
|
}
|
|
|
|
func (f *LogfmtFormatter) SetMetadataFlags(flags MetadataFlags) {
|
|
f.metadataFlags = flags
|
|
}
|
|
|
|
func (f *LogfmtFormatter) Format(e *logger.Entry) ([]byte, error) {
|
|
var buf bytes.Buffer
|
|
enc := logfmt.NewEncoder(&buf)
|
|
|
|
if f.metadataFlags&MetadataTime != 0 {
|
|
enc.EncodeKeyval(FieldTime, e.Time)
|
|
}
|
|
if f.metadataFlags&MetadataLevel != 0 {
|
|
enc.EncodeKeyval(FieldLevel, e.Level)
|
|
}
|
|
|
|
// at least try and put job and task in front
|
|
prefixed := make(map[string]bool, 2)
|
|
prefix := []string{logJobField, logTaskField}
|
|
for _, pf := range prefix {
|
|
v, ok := e.Fields[pf]
|
|
if !ok {
|
|
break
|
|
}
|
|
if err := logfmtTryEncodeKeyval(enc, pf, v); err != nil {
|
|
return nil, err // unlikely
|
|
}
|
|
prefixed[pf] = true
|
|
}
|
|
|
|
enc.EncodeKeyval(FieldMessage, e.Message)
|
|
|
|
for k, v := range e.Fields {
|
|
if !prefixed[k] {
|
|
if err := logfmtTryEncodeKeyval(enc, k, v); err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
}
|
|
|
|
return buf.Bytes(), nil
|
|
}
|
|
|
|
func logfmtTryEncodeKeyval(enc *logfmt.Encoder, field, value interface{}) error {
|
|
|
|
err := enc.EncodeKeyval(field, value)
|
|
switch err {
|
|
case nil: // ok
|
|
return nil
|
|
case logfmt.ErrUnsupportedValueType:
|
|
enc.EncodeKeyval(field, fmt.Sprintf("<%T>", value))
|
|
return nil
|
|
}
|
|
return errors.Wrapf(err, "cannot encode field '%s'", field)
|
|
|
|
}
|