package cmd import ( "github.com/mattn/go-isatty" "github.com/mitchellh/mapstructure" "github.com/pkg/errors" "github.com/zrepl/zrepl/logger" "os" "time" "crypto/tls" "crypto/x509" "github.com/zrepl/zrepl/cmd/tlsconf" ) type LoggingConfig struct { Outlets *logger.Outlets } type MetadataFlags int64 const ( MetadataTime MetadataFlags = 1 << iota MetadataLevel MetadataNone MetadataFlags = 0 MetadataAll MetadataFlags = ^0 ) func parseLogging(i interface{}) (c *LoggingConfig, err error) { c = &LoggingConfig{} c.Outlets = logger.NewOutlets() var asList []interface{} if err = mapstructure.Decode(i, &asList); err != nil { return nil, errors.Wrap(err, "mapstructure error") } if len(asList) == 0 { // Default config out := WriterOutlet{&HumanFormatter{}, os.Stdout} c.Outlets.Add(out, logger.Warn) return } var syslogOutlets, stdoutOutlets int for lei, le := range asList { outlet, minLevel, err := parseOutlet(le) if err != nil { return nil, errors.Wrapf(err, "cannot parse outlet #%d", lei) } var _ logger.Outlet = WriterOutlet{} var _ logger.Outlet = &SyslogOutlet{} switch outlet.(type) { case *SyslogOutlet: syslogOutlets++ case WriterOutlet: stdoutOutlets++ } c.Outlets.Add(outlet, minLevel) } if syslogOutlets > 1 { return nil, errors.Errorf("can only define one 'syslog' outlet") } if stdoutOutlets > 1 { return nil, errors.Errorf("can only define one 'stdout' outlet") } return c, nil } func parseLogFormat(i interface{}) (f EntryFormatter, err error) { var is string switch j := i.(type) { case string: is = j default: return nil, errors.Errorf("invalid log format: wrong type: %T", i) } switch is { case "human": return &HumanFormatter{}, nil case "logfmt": return &LogfmtFormatter{}, nil case "json": return &JSONFormatter{}, nil default: return nil, errors.Errorf("invalid log format: '%s'", is) } } func parseOutlet(i interface{}) (o logger.Outlet, level logger.Level, err error) { var in struct { Outlet string Level string Format string } if err = mapstructure.Decode(i, &in); err != nil { err = errors.Wrap(err, "mapstructure error") return } if in.Outlet == "" || in.Level == "" || in.Format == "" { err = errors.Errorf("must specify 'outlet', 'level' and 'format' field") return } minLevel, err := logger.ParseLevel(in.Level) if err != nil { err = errors.Wrap(err, "cannot parse 'level' field") return } formatter, err := parseLogFormat(in.Format) if err != nil { err = errors.Wrap(err, "cannot parse") return } switch in.Outlet { case "stdout": o, err = parseStdoutOutlet(i, formatter) case "tcp": o, err = parseTCPOutlet(i, formatter) case "syslog": o, err = parseSyslogOutlet(i, formatter) default: err = errors.Errorf("unknown outlet type '%s'", in.Outlet) } return o, minLevel, err } func parseStdoutOutlet(i interface{}, formatter EntryFormatter) (WriterOutlet, error) { var in struct { Time bool } if err := mapstructure.Decode(i, &in); err != nil { return WriterOutlet{}, errors.Wrap(err, "invalid structure for stdout outlet") } flags := MetadataAll writer := os.Stdout if !isatty.IsTerminal(writer.Fd()) && !in.Time { flags &= ^MetadataTime } formatter.SetMetadataFlags(flags) return WriterOutlet{ formatter, os.Stdout, }, nil } func parseTCPOutlet(i interface{}, formatter EntryFormatter) (out *TCPOutlet, err error) { var in struct { Net string Address string RetryInterval string `mapstructure:"retry_interval"` TLS map[string]interface{} } if err = mapstructure.Decode(i, &in); err != nil { return nil, errors.Wrap(err, "mapstructure error") } retryInterval, err := time.ParseDuration(in.RetryInterval) if err != nil { return nil, errors.Wrap(err, "cannot parse 'retry_interval'") } if len(in.Net) == 0 { return nil, errors.New("field 'net' must not be empty") } if len(in.Address) == 0 { return nil, errors.New("field 'address' must not be empty") } var tlsConfig *tls.Config if in.TLS != nil { tlsConfig, err = func(m map[string]interface{}, host string) (*tls.Config, error) { var in struct { CA string Cert string Key string } if err := mapstructure.Decode(m, &in); err != nil { return nil, errors.Wrap(err, "mapstructure error") } clientCert, err := tls.LoadX509KeyPair(in.Cert, in.Key) if err != nil { return nil, errors.Wrap(err, "cannot load client cert") } var rootCAs *x509.CertPool if in.CA == "" { if rootCAs, err = x509.SystemCertPool(); err != nil { return nil, errors.Wrap(err, "cannot open system cert pool") } } else { rootCAs, err = tlsconf.ParseCAFile(in.CA) if err != nil { return nil, errors.Wrap(err, "cannot parse CA cert") } } if rootCAs == nil { panic("invariant violated") } return tlsconf.ClientAuthClient(host, rootCAs, clientCert) }(in.TLS, in.Address) if err != nil { return nil, errors.New("cannot not parse TLS config in field 'tls'") } } formatter.SetMetadataFlags(MetadataAll) return NewTCPOutlet(formatter, in.Net, in.Address, tlsConfig, retryInterval), nil } func parseSyslogOutlet(i interface{}, formatter EntryFormatter) (out *SyslogOutlet, err error) { var in struct { RetryInterval string `mapstructure:"retry_interval"` } if err = mapstructure.Decode(i, &in); err != nil { return nil, errors.Wrap(err, "mapstructure error") } out = &SyslogOutlet{} out.Formatter = formatter out.Formatter.SetMetadataFlags(MetadataNone) out.RetryInterval = 0 // default to 0 as we assume local syslog will just work if in.RetryInterval != "" { out.RetryInterval, err = time.ParseDuration(in.RetryInterval) if err != nil { return nil, errors.Wrap(err, "cannot parse 'retry_interval'") } } return }