zrepl/cmd/config_logging.go

252 lines
5.7 KiB
Go
Raw Normal View History

2017-09-22 14:13:58 +02:00
package cmd
import (
2018-08-25 21:30:25 +02:00
"crypto/tls"
"crypto/x509"
"github.com/mattn/go-isatty"
2017-09-22 14:13:58 +02:00
"github.com/mitchellh/mapstructure"
"github.com/pkg/errors"
2018-08-25 21:30:25 +02:00
"github.com/zrepl/zrepl/cmd/tlsconf"
"github.com/zrepl/zrepl/logger"
"os"
2017-09-23 12:58:13 +02:00
"time"
2017-09-22 14:13:58 +02:00
)
type LoggingConfig struct {
Outlets *logger.Outlets
2017-09-22 14:13:58 +02:00
}
type MetadataFlags int64
2017-09-24 02:05:41 +02:00
const (
MetadataTime MetadataFlags = 1 << iota
MetadataLevel
MetadataNone MetadataFlags = 0
MetadataAll MetadataFlags = ^0
)
2017-09-22 14:13:58 +02:00
func parseLogging(i interface{}) (c *LoggingConfig, err error) {
c = &LoggingConfig{}
c.Outlets = logger.NewOutlets()
2017-09-22 14:13:58 +02:00
var asList []interface{}
if err = mapstructure.Decode(i, &asList); err != nil {
2017-09-22 14:13:58 +02:00
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
}
2017-09-22 14:13:58 +02:00
var syslogOutlets, stdoutOutlets int
for lei, le := range asList {
outlet, minLevel, err := parseOutlet(le)
2017-09-22 14:13:58 +02:00
if err != nil {
return nil, errors.Wrapf(err, "cannot parse outlet #%d", lei)
2017-09-22 14:13:58 +02:00
}
var _ logger.Outlet = WriterOutlet{}
var _ logger.Outlet = &SyslogOutlet{}
switch outlet.(type) {
case *SyslogOutlet:
syslogOutlets++
case WriterOutlet:
stdoutOutlets++
}
c.Outlets.Add(outlet, minLevel)
}
2017-09-22 14:13:58 +02:00
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)
}
2017-09-23 12:58:13 +02:00
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)
}
2017-09-22 14:13:58 +02:00
}
2017-09-22 14:13:58 +02:00
func parseOutlet(i interface{}) (o logger.Outlet, level logger.Level, err error) {
2017-09-24 14:34:50 +02:00
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
}
2017-09-24 14:34:50 +02:00
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
}
2017-09-24 14:34:50 +02:00
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
}
2017-09-24 14:34:50 +02:00
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) {
2017-09-24 02:05:41 +02:00
var in struct {
Net string
Address string
RetryInterval string `mapstructure:"retry_interval"`
2018-08-25 12:58:17 +02:00
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")
}
2017-09-24 02:05:41 +02:00
var tlsConfig *tls.Config
if in.TLS != nil {
2018-08-25 12:58:17 +02:00
tlsConfig, err = func(m map[string]interface{}, host string) (*tls.Config, error) {
var in struct {
CA string
Cert string
Key string
}
2018-08-25 12:58:17 +02:00
if err := mapstructure.Decode(m, &in); err != nil {
return nil, errors.Wrap(err, "mapstructure error")
}
clientCert, err := tls.LoadX509KeyPair(in.Cert, in.Key)
2017-09-24 02:05:41 +02:00
if err != nil {
2018-08-25 12:58:17 +02:00
return nil, errors.Wrap(err, "cannot load client cert")
2017-09-24 02:05:41 +02:00
}
2018-08-25 12:58:17 +02:00
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")
}
2017-09-24 02:05:41 +02:00
2018-08-25 12:58:17 +02:00
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'")
}
2017-09-24 02:05:41 +02:00
}
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'")
}
}
2017-09-22 14:13:58 +02:00
return
2017-09-22 14:13:58 +02:00
}