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 } const HumanFormatterDateFormat = time.RFC3339 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(HumanFormatterDateFormat)) } 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) }