使用uber-go/zap库如何根据日志级别将日志记录到stdout或stderr?

huangapple go评论94阅读模式
英文:

How to log to stdout or stderr based on log level using uber-go/zap?

问题

我正在尝试使用这个包github.com/uber-go/zap来设置日志记录。
我需要将以下内容写入日志:

  • 将信息日志写入标准输出(stdout)
  • 将错误和致命错误日志写入标准错误(stderr)

我尝试通过设置和构建zap.Config来实现:

  1. cfg = &zap.Config{
  2. Encoding: "json",
  3. Level: zap.NewAtomicLevelAt(zapcore.DebugLevel),
  4. OutputPaths: []string{"stdout"},
  5. ErrorOutputPaths: []string{"stderr"},
  6. EncoderConfig: zapcore.EncoderConfig{
  7. MessageKey: "message",
  8. LevelKey: "level",
  9. EncodeLevel: zapcore.CapitalLevelEncoder,
  10. TimeKey: "time",
  11. EncodeTime: zapcore.ISO8601TimeEncoder,
  12. CallerKey: "caller",
  13. EncodeCaller: zapcore.ShortCallerEncoder,
  14. EncodeDuration: zapcore.MillisDurationEncoder,
  15. },
  16. }

我还尝试了这种方式:

  1. cfg = zap.NewProductionConfig()
  2. cfg.OutputPaths = []string{"stdout"}
  3. logger, err = cfg.Build(zap.AddCaller(), zap.AddCallerSkip(1))

但是所有的日志都被写入了stdout 或者 stderr。我该如何分开写入日志呢?

英文:

I am trying to set up logging using this package github.com/uber-go/zap.
I need to write:

  • Info logs to stdout
  • Error and Fatal logs to stderr

I tried to do this by setting up and building zap.Config like this:

  1. cfg = &zap.Config{
  2. Encoding: "json",
  3. Level: zap.NewAtomicLevelAt(zapcore.DebugLevel),
  4. OutputPaths: []string{"stdout"},
  5. ErrorOutputPaths: []string{"stderr"},
  6. EncoderConfig: zapcore.EncoderConfig{
  7. MessageKey: "message",
  8. LevelKey: "level",
  9. EncodeLevel: zapcore.CapitalLevelEncoder,
  10. TimeKey: "time",
  11. EncodeTime: zapcore.ISO8601TimeEncoder,
  12. CallerKey: "caller",
  13. EncodeCaller: zapcore.ShortCallerEncoder,
  14. EncodeDuration: zapcore.MillisDurationEncoder,
  15. },
  16. }

Also I tried this way:

  1. cfg = zap.NewProductionConfig()
  2. cfg.OutputPaths = []string{"stdout"}
  3. logger, err = cfg.Build(zap.AddCaller(), zap.AddCallerSkip(1))

But all logs are written to either stdout or stderr. How can I separate it?

答案1

得分: 6

使用zapcore.NewTee和两个核心:

  • 一个将zapcore.InfoLevel锁定到stdout
  • 一个将zapcore.ErrorLevelzapcore.FatalLevel锁定到stderr

这是一个演示用法的最小程序:

  1. package main
  2. import (
  3. "go.uber.org/zap"
  4. "go.uber.org/zap/zapcore"
  5. "os"
  6. )
  7. func main() {
  8. // info level enabler
  9. infoLevel := zap.LevelEnablerFunc(func(level zapcore.Level) bool {
  10. return level == zapcore.InfoLevel
  11. })
  12. // error and fatal level enabler
  13. errorFatalLevel := zap.LevelEnablerFunc(func(level zapcore.Level) bool {
  14. return level == zapcore.ErrorLevel || level == zapcore.FatalLevel
  15. })
  16. // write syncers
  17. stdoutSyncer := zapcore.Lock(os.Stdout)
  18. stderrSyncer := zapcore.Lock(os.Stderr)
  19. // tee core
  20. core := zapcore.NewTee(
  21. zapcore.NewCore(
  22. zapcore.NewJSONEncoder(zap.NewProductionEncoderConfig()),
  23. stdoutSyncer,
  24. infoLevel,
  25. ),
  26. zapcore.NewCore(
  27. zapcore.NewJSONEncoder(zap.NewProductionEncoderConfig()),
  28. stderrSyncer,
  29. errorFatalLevel,
  30. ),
  31. )
  32. // finally construct the logger with the tee core
  33. logger := zap.New(core)
  34. logger.Info("info log")
  35. logger.Error("error log")
  36. }

请注意,上面的程序记录了两条消息,一条是info级别的,一条是error级别的。要查看它们是否正确地打印到了相应的设备上,请运行程序并将stdout或stderr重定向到/dev/null,这样只会在控制台上打印另一条消息:

  1. $ go build main.go
  2. $ ./main 2>/dev/null # 仅显示stdout
  3. {"level":"info","ts":1626900981.520349,"msg":"info log"}
  4. $ ./main 1>/dev/null # 仅显示stderr
  5. {"level":"error","ts":1626901025.056065,"msg":"error log"}
英文:

Use zapcore.NewTee with two cores:

  • one that locks zapcore.InfoLevel to stdout
  • one that locks zapcore.ErrorLevel and zapcore.FatalLevel to stderr

This is a minimal program to demonstrate the usage:

  1. package main
  2. import (
  3. "go.uber.org/zap"
  4. "go.uber.org/zap/zapcore"
  5. "os"
  6. )
  7. func main() {
  8. // info level enabler
  9. infoLevel := zap.LevelEnablerFunc(func(level zapcore.Level) bool {
  10. return level == zapcore.InfoLevel
  11. })
  12. // error and fatal level enabler
  13. errorFatalLevel := zap.LevelEnablerFunc(func(level zapcore.Level) bool {
  14. return level == zapcore.ErrorLevel || level == zapcore.FatalLevel
  15. })
  16. // write syncers
  17. stdoutSyncer := zapcore.Lock(os.Stdout)
  18. stderrSyncer := zapcore.Lock(os.Stderr)
  19. // tee core
  20. core := zapcore.NewTee(
  21. zapcore.NewCore(
  22. zapcore.NewJSONEncoder(zap.NewProductionEncoderConfig()),
  23. stdoutSyncer,
  24. infoLevel,
  25. ),
  26. zapcore.NewCore(
  27. zapcore.NewJSONEncoder(zap.NewProductionEncoderConfig()),
  28. stderrSyncer,
  29. errorFatalLevel,
  30. ),
  31. )
  32. // finally construct the logger with the tee core
  33. logger := zap.New(core)
  34. logger.Info("info log")
  35. logger.Error("error log")
  36. }

Note that the program above logs two messages, one at info level and one at error level. To see that it prints both to the correct device, run it and redirect stdout or stderr to /dev/null — thus printing to the console only the other one:

  1. $ go build main.go
  2. $ ./main 2>/dev/null # shows only stdout
  3. {"level":"info","ts":1626900981.520349,"msg":"info log"}
  4. $ ./main 1>/dev/null # shows only stderr
  5. {"level":"error","ts":1626901025.056065,"msg":"error log"}

huangapple
  • 本文由 发表于 2021年7月21日 23:49:26
  • 转载请务必保留本文链接:https://go.coder-hub.com/68472667.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定