mirror of
https://github.com/caddyserver/caddy.git
synced 2025-01-13 22:36:27 +03:00
logging: Add zap.Option
support (#5944)
This commit is contained in:
parent
da7d8cb26d
commit
3248e4c89f
2 changed files with 89 additions and 28 deletions
115
logging.go
115
logging.go
|
@ -150,12 +150,18 @@ func (logging *Logging) setupNewDefault(ctx Context) error {
|
||||||
logging.Logs[DefaultLoggerName] = newDefault.CustomLog
|
logging.Logs[DefaultLoggerName] = newDefault.CustomLog
|
||||||
}
|
}
|
||||||
|
|
||||||
// set up this new log
|
// options for the default logger
|
||||||
err := newDefault.CustomLog.provision(ctx, logging)
|
options, err := newDefault.CustomLog.buildOptions()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("setting up default log: %v", err)
|
return fmt.Errorf("setting up default log: %v", err)
|
||||||
}
|
}
|
||||||
newDefault.logger = zap.New(newDefault.CustomLog.core)
|
|
||||||
|
// set up this new log
|
||||||
|
err = newDefault.CustomLog.provision(ctx, logging)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("setting up default log: %v", err)
|
||||||
|
}
|
||||||
|
newDefault.logger = zap.New(newDefault.CustomLog.core, options...)
|
||||||
|
|
||||||
// redirect the default caddy logs
|
// redirect the default caddy logs
|
||||||
defaultLoggerMu.Lock()
|
defaultLoggerMu.Lock()
|
||||||
|
@ -201,6 +207,7 @@ func (logging *Logging) closeLogs() error {
|
||||||
func (logging *Logging) Logger(mod Module) *zap.Logger {
|
func (logging *Logging) Logger(mod Module) *zap.Logger {
|
||||||
modID := string(mod.CaddyModule().ID)
|
modID := string(mod.CaddyModule().ID)
|
||||||
var cores []zapcore.Core
|
var cores []zapcore.Core
|
||||||
|
var options []zap.Option
|
||||||
|
|
||||||
if logging != nil {
|
if logging != nil {
|
||||||
for _, l := range logging.Logs {
|
for _, l := range logging.Logs {
|
||||||
|
@ -209,6 +216,13 @@ func (logging *Logging) Logger(mod Module) *zap.Logger {
|
||||||
cores = append(cores, l.core)
|
cores = append(cores, l.core)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
if len(options) == 0 {
|
||||||
|
newOptions, err := l.buildOptions()
|
||||||
|
if err != nil {
|
||||||
|
Log().Error("building options for logger", zap.String("module", modID), zap.Error(err))
|
||||||
|
}
|
||||||
|
options = newOptions
|
||||||
|
}
|
||||||
cores = append(cores, &filteringCore{Core: l.core, cl: l})
|
cores = append(cores, &filteringCore{Core: l.core, cl: l})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -216,7 +230,7 @@ func (logging *Logging) Logger(mod Module) *zap.Logger {
|
||||||
|
|
||||||
multiCore := zapcore.NewTee(cores...)
|
multiCore := zapcore.NewTee(cores...)
|
||||||
|
|
||||||
return zap.New(multiCore).Named(modID)
|
return zap.New(multiCore, options...).Named(modID)
|
||||||
}
|
}
|
||||||
|
|
||||||
// openWriter opens a writer using opener, and returns true if
|
// openWriter opens a writer using opener, and returns true if
|
||||||
|
@ -277,6 +291,20 @@ type BaseLog struct {
|
||||||
// servers.
|
// servers.
|
||||||
Sampling *LogSampling `json:"sampling,omitempty"`
|
Sampling *LogSampling `json:"sampling,omitempty"`
|
||||||
|
|
||||||
|
// If true, the log entry will include the caller's
|
||||||
|
// file name and line number. Default off.
|
||||||
|
WithCaller bool `json:"with_caller,omitempty"`
|
||||||
|
|
||||||
|
// If non-zero, and `with_caller` is true, this many
|
||||||
|
// stack frames will be skipped when determining the
|
||||||
|
// caller. Default 0.
|
||||||
|
WithCallerSkip int `json:"with_caller_skip,omitempty"`
|
||||||
|
|
||||||
|
// If not empty, the log entry will include a stack trace
|
||||||
|
// for all logs at the given level or higher. See `level`
|
||||||
|
// for possible values. Default off.
|
||||||
|
WithStacktrace string `json:"with_stacktrace,omitempty"`
|
||||||
|
|
||||||
writerOpener WriterOpener
|
writerOpener WriterOpener
|
||||||
writer io.WriteCloser
|
writer io.WriteCloser
|
||||||
encoder zapcore.Encoder
|
encoder zapcore.Encoder
|
||||||
|
@ -301,29 +329,10 @@ func (cl *BaseLog) provisionCommon(ctx Context, logging *Logging) error {
|
||||||
return fmt.Errorf("opening log writer using %#v: %v", cl.writerOpener, err)
|
return fmt.Errorf("opening log writer using %#v: %v", cl.writerOpener, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
repl := NewReplacer()
|
|
||||||
level, err := repl.ReplaceOrErr(cl.Level, true, true)
|
|
||||||
if err != nil {
|
|
||||||
return fmt.Errorf("invalid log level: %v", err)
|
|
||||||
}
|
|
||||||
level = strings.ToLower(level)
|
|
||||||
|
|
||||||
// set up the log level
|
// set up the log level
|
||||||
switch level {
|
cl.levelEnabler, err = parseLevel(cl.Level)
|
||||||
case "debug":
|
if err != nil {
|
||||||
cl.levelEnabler = zapcore.DebugLevel
|
return err
|
||||||
case "", "info":
|
|
||||||
cl.levelEnabler = zapcore.InfoLevel
|
|
||||||
case "warn":
|
|
||||||
cl.levelEnabler = zapcore.WarnLevel
|
|
||||||
case "error":
|
|
||||||
cl.levelEnabler = zapcore.ErrorLevel
|
|
||||||
case "panic":
|
|
||||||
cl.levelEnabler = zapcore.PanicLevel
|
|
||||||
case "fatal":
|
|
||||||
cl.levelEnabler = zapcore.FatalLevel
|
|
||||||
default:
|
|
||||||
return fmt.Errorf("unrecognized log level: %s", cl.Level)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if cl.EncoderRaw != nil {
|
if cl.EncoderRaw != nil {
|
||||||
|
@ -376,6 +385,24 @@ func (cl *BaseLog) buildCore() {
|
||||||
cl.core = c
|
cl.core = c
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (cl *BaseLog) buildOptions() ([]zap.Option, error) {
|
||||||
|
var options []zap.Option
|
||||||
|
if cl.WithCaller {
|
||||||
|
options = append(options, zap.AddCaller())
|
||||||
|
if cl.WithCallerSkip != 0 {
|
||||||
|
options = append(options, zap.AddCallerSkip(cl.WithCallerSkip))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if cl.WithStacktrace != "" {
|
||||||
|
levelEnabler, err := parseLevel(cl.WithStacktrace)
|
||||||
|
if err != nil {
|
||||||
|
return options, fmt.Errorf("setting up default Caddy log: %v", err)
|
||||||
|
}
|
||||||
|
options = append(options, zap.AddStacktrace(levelEnabler))
|
||||||
|
}
|
||||||
|
return options, nil
|
||||||
|
}
|
||||||
|
|
||||||
// SinkLog configures the default Go standard library
|
// SinkLog configures the default Go standard library
|
||||||
// global logger in the log package. This is necessary because
|
// global logger in the log package. This is necessary because
|
||||||
// module dependencies which are not built specifically for
|
// module dependencies which are not built specifically for
|
||||||
|
@ -389,7 +416,14 @@ func (sll *SinkLog) provision(ctx Context, logging *Logging) error {
|
||||||
if err := sll.provisionCommon(ctx, logging); err != nil {
|
if err := sll.provisionCommon(ctx, logging); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
ctx.cleanupFuncs = append(ctx.cleanupFuncs, zap.RedirectStdLog(zap.New(sll.core)))
|
|
||||||
|
options, err := sll.buildOptions()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
logger := zap.New(sll.core, options...)
|
||||||
|
ctx.cleanupFuncs = append(ctx.cleanupFuncs, zap.RedirectStdLog(logger))
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -678,6 +712,33 @@ func newDefaultProductionLogEncoder(colorize bool) zapcore.Encoder {
|
||||||
return zapcore.NewJSONEncoder(encCfg)
|
return zapcore.NewJSONEncoder(encCfg)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func parseLevel(levelInput string) (zapcore.LevelEnabler, error) {
|
||||||
|
repl := NewReplacer()
|
||||||
|
level, err := repl.ReplaceOrErr(levelInput, true, true)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("invalid log level: %v", err)
|
||||||
|
}
|
||||||
|
level = strings.ToLower(level)
|
||||||
|
|
||||||
|
// set up the log level
|
||||||
|
switch level {
|
||||||
|
case "debug":
|
||||||
|
return zapcore.DebugLevel, nil
|
||||||
|
case "", "info":
|
||||||
|
return zapcore.InfoLevel, nil
|
||||||
|
case "warn":
|
||||||
|
return zapcore.WarnLevel, nil
|
||||||
|
case "error":
|
||||||
|
return zapcore.ErrorLevel, nil
|
||||||
|
case "panic":
|
||||||
|
return zapcore.PanicLevel, nil
|
||||||
|
case "fatal":
|
||||||
|
return zapcore.FatalLevel, nil
|
||||||
|
default:
|
||||||
|
return nil, fmt.Errorf("unrecognized log level: %s", level)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Log returns the current default logger.
|
// Log returns the current default logger.
|
||||||
func Log() *zap.Logger {
|
func Log() *zap.Logger {
|
||||||
defaultLoggerMu.RLock()
|
defaultLoggerMu.RLock()
|
||||||
|
|
|
@ -68,7 +68,7 @@ func (h *Handler) handleUpgradeResponse(logger *zap.Logger, rw http.ResponseWrit
|
||||||
//nolint:bodyclose
|
//nolint:bodyclose
|
||||||
conn, brw, hijackErr := http.NewResponseController(rw).Hijack()
|
conn, brw, hijackErr := http.NewResponseController(rw).Hijack()
|
||||||
if errors.Is(hijackErr, http.ErrNotSupported) {
|
if errors.Is(hijackErr, http.ErrNotSupported) {
|
||||||
h.logger.Sugar().Errorf("can't switch protocols using non-Hijacker ResponseWriter type %T", rw)
|
h.logger.Error("can't switch protocols using non-Hijacker ResponseWriter", zap.String("type", fmt.Sprintf("%T", rw)))
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue