2019-07-24 09:58:01 +02:00
|
|
|
package logutils
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"math"
|
2019-07-18 00:25:42 +02:00
|
|
|
"sync"
|
2019-07-24 09:58:01 +02:00
|
|
|
"time"
|
2019-07-18 00:25:42 +02:00
|
|
|
|
2020-01-02 10:10:19 +01:00
|
|
|
"go.uber.org/zap"
|
|
|
|
"go.uber.org/zap/zapcore"
|
|
|
|
|
|
|
|
"github.com/ethereum/go-ethereum/log"
|
|
|
|
|
2019-11-21 17:19:22 +01:00
|
|
|
"github.com/status-im/status-go/protocol/zaputil"
|
2019-07-24 09:58:01 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
type gethLoggerCore struct {
|
|
|
|
zapcore.LevelEnabler
|
|
|
|
fields []zapcore.Field
|
|
|
|
logger log.Logger
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c gethLoggerCore) With(fields []zapcore.Field) zapcore.Core {
|
|
|
|
clone := c.clone()
|
|
|
|
clone.fields = append(clone.fields, fields...)
|
|
|
|
return clone
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c gethLoggerCore) Check(ent zapcore.Entry, ce *zapcore.CheckedEntry) *zapcore.CheckedEntry {
|
|
|
|
if c.Enabled(ent.Level) {
|
|
|
|
return ce.AddCore(ent, c)
|
|
|
|
}
|
|
|
|
return ce
|
|
|
|
}
|
2020-02-10 12:22:37 +01:00
|
|
|
func (c gethLoggerCore) Write(ent zapcore.Entry, fields []zapcore.Field) error {
|
2019-07-24 09:58:01 +02:00
|
|
|
fields = append(c.fields[:], fields...)
|
|
|
|
|
|
|
|
var args []interface{}
|
|
|
|
for _, f := range fields {
|
|
|
|
switch f.Type {
|
|
|
|
case zapcore.ArrayMarshalerType,
|
|
|
|
zapcore.ObjectMarshalerType,
|
|
|
|
zapcore.BinaryType,
|
|
|
|
zapcore.ByteStringType,
|
|
|
|
zapcore.Complex128Type,
|
|
|
|
zapcore.ReflectType,
|
|
|
|
zapcore.StringerType,
|
|
|
|
zapcore.ErrorType:
|
|
|
|
args = append(args, f.Key, f.Interface)
|
|
|
|
case zapcore.BoolType:
|
|
|
|
args = append(args, f.Key, f.Integer == 1)
|
|
|
|
case zapcore.DurationType:
|
|
|
|
args = append(args, f.Key, time.Duration(f.Integer))
|
|
|
|
case zapcore.Float64Type:
|
|
|
|
args = append(args, f.Key, math.Float64frombits(uint64(f.Integer)))
|
|
|
|
case zapcore.Float32Type:
|
|
|
|
args = append(args, f.Key, math.Float32frombits(uint32(f.Integer)))
|
|
|
|
case zapcore.Int64Type,
|
|
|
|
zapcore.Int32Type,
|
|
|
|
zapcore.Int16Type,
|
|
|
|
zapcore.Int8Type,
|
|
|
|
zapcore.Uint64Type,
|
|
|
|
zapcore.Uint32Type,
|
|
|
|
zapcore.Uint16Type,
|
|
|
|
zapcore.Uint8Type:
|
|
|
|
args = append(args, f.Key, f.Integer)
|
|
|
|
case zapcore.UintptrType:
|
|
|
|
args = append(args, f.Key, uintptr(f.Integer))
|
|
|
|
case zapcore.StringType:
|
|
|
|
args = append(args, f.Key, f.String)
|
|
|
|
case zapcore.TimeType:
|
|
|
|
if f.Interface != nil {
|
|
|
|
args = append(args, f.Key, time.Unix(0, f.Integer).In(f.Interface.(*time.Location)))
|
|
|
|
} else {
|
|
|
|
// Fall back to UTC if location is nil.
|
|
|
|
args = append(args, f.Key, time.Unix(0, f.Integer))
|
|
|
|
}
|
|
|
|
case zapcore.NamespaceType:
|
|
|
|
args = append(args, "namespace", f.Key)
|
|
|
|
case zapcore.SkipType:
|
|
|
|
break
|
|
|
|
default:
|
|
|
|
panic(fmt.Sprintf("unknown field type: %v", f))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-11-09 14:35:59 +08:00
|
|
|
// set callDepth to 3 for `Output` to skip the calls to zap.Logger
|
|
|
|
// and get the correct caller in the log
|
|
|
|
callDepth := 3
|
2019-07-24 09:58:01 +02:00
|
|
|
switch ent.Level {
|
|
|
|
case zapcore.DebugLevel:
|
2023-11-09 14:35:59 +08:00
|
|
|
c.logger.Output(ent.Message, log.LvlDebug, callDepth, args...)
|
2019-07-24 09:58:01 +02:00
|
|
|
case zapcore.InfoLevel:
|
2023-11-09 14:35:59 +08:00
|
|
|
c.logger.Output(ent.Message, log.LvlInfo, callDepth, args...)
|
2019-07-24 09:58:01 +02:00
|
|
|
case zapcore.WarnLevel:
|
2023-11-09 14:35:59 +08:00
|
|
|
c.logger.Output(ent.Message, log.LvlWarn, callDepth, args...)
|
2019-07-24 09:58:01 +02:00
|
|
|
case zapcore.ErrorLevel:
|
2023-11-09 14:35:59 +08:00
|
|
|
c.logger.Output(ent.Message, log.LvlError, callDepth, args...)
|
2019-07-24 09:58:01 +02:00
|
|
|
case zapcore.DPanicLevel, zapcore.PanicLevel, zapcore.FatalLevel:
|
2023-11-09 14:35:59 +08:00
|
|
|
c.logger.Output(ent.Message, log.LvlCrit, callDepth, args...)
|
2019-07-24 09:58:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (gethLoggerCore) Sync() error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *gethLoggerCore) clone() *gethLoggerCore {
|
|
|
|
return &gethLoggerCore{
|
|
|
|
LevelEnabler: c.LevelEnabler,
|
|
|
|
fields: c.fields,
|
|
|
|
logger: c.logger,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewZapAdapter returns a new zapcore.Core interface which forwards logs to log.Logger.
|
|
|
|
func NewZapAdapter(logger log.Logger, enab zapcore.LevelEnabler) zapcore.Core {
|
|
|
|
return &gethLoggerCore{
|
|
|
|
LevelEnabler: enab,
|
|
|
|
logger: logger,
|
|
|
|
}
|
|
|
|
}
|
2019-07-18 00:25:42 +02:00
|
|
|
|
|
|
|
var registerOnce sync.Once
|
|
|
|
|
|
|
|
// NewZapLoggerWithAdapter returns a logger forwarding all logs with level info and above.
|
|
|
|
func NewZapLoggerWithAdapter(logger log.Logger) (*zap.Logger, error) {
|
|
|
|
registerOnce.Do(func() {
|
2019-07-30 20:39:16 +02:00
|
|
|
if err := zaputil.RegisterJSONHexEncoder(); err != nil {
|
2019-07-18 00:25:42 +02:00
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
cfg := zap.Config{
|
|
|
|
Level: zap.NewAtomicLevelAt(zapcore.DebugLevel),
|
|
|
|
Development: false,
|
|
|
|
Sampling: nil,
|
|
|
|
Encoding: "json-hex",
|
|
|
|
EncoderConfig: zap.NewProductionEncoderConfig(),
|
|
|
|
OutputPaths: []string{"stderr"},
|
|
|
|
ErrorOutputPaths: []string{"stderr"},
|
|
|
|
}
|
|
|
|
adapter := zap.WrapCore(
|
|
|
|
func(zapcore.Core) zapcore.Core {
|
|
|
|
return NewZapAdapter(logger, cfg.Level)
|
|
|
|
},
|
|
|
|
)
|
2023-11-09 14:35:59 +08:00
|
|
|
log.PrintOrigins(true)
|
2019-07-18 00:25:42 +02:00
|
|
|
return cfg.Build(adapter)
|
|
|
|
}
|