2020-02-14 22:56:46 -08:00
|
|
|
package zerolog
|
2020-02-11 02:49:33 -08:00
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"os"
|
2020-02-13 21:04:50 -08:00
|
|
|
"runtime/debug"
|
2020-02-11 02:49:33 -08:00
|
|
|
|
|
|
|
"github.com/micro/go-micro/v2/logger"
|
|
|
|
"github.com/rs/zerolog"
|
2020-02-13 21:04:50 -08:00
|
|
|
zlog "github.com/rs/zerolog/log"
|
|
|
|
"github.com/rs/zerolog/pkgerrors"
|
|
|
|
)
|
|
|
|
|
|
|
|
type Mode uint8
|
|
|
|
|
|
|
|
const (
|
|
|
|
Production Mode = iota + 1
|
|
|
|
Development
|
2020-02-11 02:49:33 -08:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
2020-02-13 21:04:50 -08:00
|
|
|
// It's common to set this to a file, or leave it default which is `os.Stderr`
|
|
|
|
out io.Writer = os.Stderr
|
|
|
|
// Function to exit the application, defaults to `os.Exit()`
|
|
|
|
exitFunc = os.Exit
|
|
|
|
// Flag for whether to log caller info (off by default)
|
|
|
|
reportCaller = false
|
|
|
|
// use this logger as system wide default logger
|
|
|
|
useAsDefault = false
|
|
|
|
// The logging level the logger should log at.
|
|
|
|
// This defaults to 100 means not explicitly set by user
|
|
|
|
level logger.Level = 100
|
2020-02-14 22:56:46 -08:00
|
|
|
fields map[string]interface{}
|
2020-02-13 21:04:50 -08:00
|
|
|
hooks []zerolog.Hook
|
|
|
|
timeFormat string
|
|
|
|
// default Production (1)
|
|
|
|
mode Mode = Production
|
2020-02-11 02:49:33 -08:00
|
|
|
)
|
|
|
|
|
|
|
|
type zeroLogger struct {
|
|
|
|
nativelogger zerolog.Logger
|
|
|
|
}
|
|
|
|
|
2020-02-14 22:56:46 -08:00
|
|
|
func (l *zeroLogger) Fields(fields map[string]interface{}) logger.Logger {
|
|
|
|
return &zeroLogger{l.nativelogger.With().Fields(fields).Logger()}
|
2020-02-11 02:49:33 -08:00
|
|
|
}
|
|
|
|
|
2020-02-13 21:04:50 -08:00
|
|
|
func (l *zeroLogger) Error(err error) logger.Logger {
|
|
|
|
return &zeroLogger{
|
|
|
|
l.nativelogger.With().Fields(map[string]interface{}{zerolog.ErrorFieldName: err}).Logger(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-11 02:49:33 -08:00
|
|
|
func (l *zeroLogger) Init(opts ...logger.Option) error {
|
|
|
|
|
|
|
|
options := &Options{logger.Options{Context: context.Background()}}
|
|
|
|
for _, o := range opts {
|
|
|
|
o(&options.Options)
|
|
|
|
}
|
|
|
|
|
|
|
|
if o, ok := options.Context.Value(outKey{}).(io.Writer); ok {
|
|
|
|
out = o
|
|
|
|
}
|
2020-02-13 21:04:50 -08:00
|
|
|
if hs, ok := options.Context.Value(hooksKey{}).([]zerolog.Hook); ok {
|
|
|
|
hooks = hs
|
|
|
|
}
|
2020-02-14 22:56:46 -08:00
|
|
|
if flds, ok := options.Context.Value(fieldsKey{}).(map[string]interface{}); ok {
|
2020-02-13 21:04:50 -08:00
|
|
|
fields = flds
|
|
|
|
}
|
|
|
|
if lvl, ok := options.Context.Value(levelKey{}).(logger.Level); ok {
|
|
|
|
level = lvl
|
|
|
|
}
|
|
|
|
if tf, ok := options.Context.Value(timeFormatKey{}).(string); ok {
|
|
|
|
timeFormat = tf
|
|
|
|
}
|
|
|
|
if exitFunction, ok := options.Context.Value(exitKey{}).(func(int)); ok {
|
|
|
|
exitFunc = exitFunction
|
|
|
|
}
|
|
|
|
if caller, ok := options.Context.Value(reportCallerKey{}).(bool); ok && caller {
|
|
|
|
reportCaller = caller
|
|
|
|
}
|
|
|
|
if useDefault, ok := options.Context.Value(useAsDefaultKey{}).(bool); ok && useDefault {
|
|
|
|
useAsDefault = useDefault
|
|
|
|
}
|
|
|
|
if devMode, ok := options.Context.Value(developmentModeKey{}).(bool); ok && devMode {
|
|
|
|
mode = Development
|
|
|
|
}
|
|
|
|
if prodMode, ok := options.Context.Value(productionModeKey{}).(bool); ok && prodMode {
|
|
|
|
mode = Production
|
|
|
|
}
|
|
|
|
|
|
|
|
switch mode {
|
|
|
|
case Development:
|
|
|
|
zerolog.ErrorStackMarshaler = func(err error) interface{} {
|
|
|
|
fmt.Println(string(debug.Stack()))
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
consOut := zerolog.NewConsoleWriter(
|
2020-02-11 02:49:33 -08:00
|
|
|
func(w *zerolog.ConsoleWriter) {
|
2020-02-14 22:56:46 -08:00
|
|
|
if len(timeFormat) > 0 {
|
|
|
|
w.TimeFormat = timeFormat
|
|
|
|
}
|
2020-02-11 02:49:33 -08:00
|
|
|
w.Out = out
|
2020-02-13 21:04:50 -08:00
|
|
|
w.NoColor = false
|
2020-02-11 02:49:33 -08:00
|
|
|
},
|
|
|
|
)
|
2020-02-13 21:04:50 -08:00
|
|
|
level = logger.DebugLevel
|
|
|
|
l.nativelogger = zerolog.New(consOut).
|
|
|
|
Level(zerolog.DebugLevel).
|
|
|
|
With().Timestamp().Stack().Logger()
|
|
|
|
default: // Production
|
|
|
|
zerolog.ErrorStackMarshaler = pkgerrors.MarshalStack
|
|
|
|
l.nativelogger = zerolog.New(out).
|
|
|
|
Level(zerolog.InfoLevel).
|
|
|
|
With().Timestamp().Stack().Logger()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Change Writer if not default
|
|
|
|
if out != os.Stderr {
|
|
|
|
l.nativelogger = l.nativelogger.Output(out)
|
2020-02-11 02:49:33 -08:00
|
|
|
}
|
|
|
|
|
2020-02-13 21:04:50 -08:00
|
|
|
// Set log Level if not default
|
|
|
|
if level != 100 {
|
2020-02-11 02:49:33 -08:00
|
|
|
//zerolog.SetGlobalLevel(loggerToZerologLevel(level))
|
|
|
|
l.nativelogger = l.nativelogger.Level(loggerToZerologLevel(level))
|
|
|
|
}
|
|
|
|
|
2020-02-13 21:04:50 -08:00
|
|
|
// Adding hooks if exist
|
|
|
|
if reportCaller {
|
2020-02-11 02:49:33 -08:00
|
|
|
l.nativelogger = l.nativelogger.With().Caller().Logger()
|
|
|
|
}
|
2020-02-13 21:04:50 -08:00
|
|
|
for _, hook := range hooks {
|
|
|
|
l.nativelogger = l.nativelogger.Hook(hook)
|
|
|
|
}
|
2020-02-11 02:49:33 -08:00
|
|
|
|
2020-02-13 21:04:50 -08:00
|
|
|
// Setting timeFormat
|
|
|
|
if len(timeFormat) > 0 {
|
|
|
|
zerolog.TimeFieldFormat = timeFormat
|
2020-02-11 02:49:33 -08:00
|
|
|
}
|
|
|
|
|
2020-02-13 21:04:50 -08:00
|
|
|
// Adding seed fields if exist
|
|
|
|
if fields != nil {
|
2020-02-14 22:56:46 -08:00
|
|
|
l.nativelogger = l.nativelogger.With().Fields(fields).Logger()
|
2020-02-11 02:49:33 -08:00
|
|
|
}
|
|
|
|
|
2020-02-14 22:56:46 -08:00
|
|
|
// Also set it as zerolog's Default logger
|
2020-02-13 21:04:50 -08:00
|
|
|
if useAsDefault {
|
|
|
|
zlog.Logger = l.nativelogger
|
2020-02-11 02:49:33 -08:00
|
|
|
}
|
2020-02-13 21:04:50 -08:00
|
|
|
|
2020-02-11 02:49:33 -08:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l *zeroLogger) SetLevel(level logger.Level) {
|
|
|
|
//zerolog.SetGlobalLevel(loggerToZerologLevel(level))
|
|
|
|
l.nativelogger = l.nativelogger.Level(loggerToZerologLevel(level))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l *zeroLogger) Level() logger.Level {
|
|
|
|
return ZerologToLoggerLevel(l.nativelogger.GetLevel())
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l *zeroLogger) Log(level logger.Level, args ...interface{}) {
|
|
|
|
msg := fmt.Sprintf("%s", args)
|
2020-02-13 21:04:50 -08:00
|
|
|
l.nativelogger.WithLevel(loggerToZerologLevel(level)).Msg(msg[1 : len(msg)-1])
|
2020-02-11 02:49:33 -08:00
|
|
|
// Invoke os.Exit because unlike zerolog.Logger.Fatal zerolog.Logger.WithLevel won't stop the execution.
|
|
|
|
if level == logger.FatalLevel {
|
2020-02-13 21:04:50 -08:00
|
|
|
exitFunc(1)
|
2020-02-11 02:49:33 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l *zeroLogger) Logf(level logger.Level, format string, args ...interface{}) {
|
|
|
|
l.nativelogger.WithLevel(loggerToZerologLevel(level)).Msgf(format, args...)
|
|
|
|
// Invoke os.Exit because unlike zerolog.Logger.Fatal zerolog.Logger.WithLevel won't stop the execution.
|
|
|
|
if level == logger.FatalLevel {
|
2020-02-13 21:04:50 -08:00
|
|
|
exitFunc(1)
|
2020-02-11 02:49:33 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l *zeroLogger) String() string {
|
|
|
|
return "zerolog"
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewLogger builds a new logger based on options
|
|
|
|
func NewLogger(opts ...logger.Option) logger.Logger {
|
|
|
|
l := &zeroLogger{}
|
|
|
|
_ = l.Init(opts...)
|
|
|
|
return l
|
|
|
|
}
|
|
|
|
|
2020-02-13 21:04:50 -08:00
|
|
|
// ParseLevel converts a level string into a logger Level value.
|
|
|
|
// returns an error if the input string does not match known values.
|
|
|
|
func ParseLevel(levelStr string) (lvl logger.Level, err error) {
|
|
|
|
if zLevel, err := zerolog.ParseLevel(levelStr); err == nil {
|
|
|
|
return ZerologToLoggerLevel(zLevel), err
|
|
|
|
} else {
|
|
|
|
return lvl, fmt.Errorf("Unknown Level String: '%s' %w", levelStr, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-11 02:49:33 -08:00
|
|
|
func loggerToZerologLevel(level logger.Level) zerolog.Level {
|
|
|
|
switch level {
|
|
|
|
case logger.TraceLevel:
|
|
|
|
return zerolog.TraceLevel
|
|
|
|
case logger.DebugLevel:
|
|
|
|
return zerolog.DebugLevel
|
|
|
|
case logger.InfoLevel:
|
|
|
|
return zerolog.InfoLevel
|
|
|
|
case logger.WarnLevel:
|
|
|
|
return zerolog.WarnLevel
|
|
|
|
case logger.ErrorLevel:
|
|
|
|
return zerolog.ErrorLevel
|
|
|
|
case logger.PanicLevel:
|
|
|
|
return zerolog.PanicLevel
|
|
|
|
case logger.FatalLevel:
|
|
|
|
return zerolog.FatalLevel
|
|
|
|
default:
|
|
|
|
return zerolog.InfoLevel
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func ZerologToLoggerLevel(level zerolog.Level) logger.Level {
|
|
|
|
switch level {
|
|
|
|
case zerolog.TraceLevel:
|
|
|
|
return logger.TraceLevel
|
|
|
|
case zerolog.DebugLevel:
|
|
|
|
return logger.DebugLevel
|
|
|
|
case zerolog.InfoLevel:
|
|
|
|
return logger.InfoLevel
|
|
|
|
case zerolog.WarnLevel:
|
|
|
|
return logger.WarnLevel
|
|
|
|
case zerolog.ErrorLevel:
|
|
|
|
return logger.ErrorLevel
|
|
|
|
case zerolog.PanicLevel:
|
|
|
|
return logger.PanicLevel
|
|
|
|
case zerolog.FatalLevel:
|
|
|
|
return logger.FatalLevel
|
|
|
|
default:
|
|
|
|
return logger.InfoLevel
|
|
|
|
}
|
|
|
|
}
|