From 98c4692bcdd6506d46c81866d9bebeef7d462b96 Mon Sep 17 00:00:00 2001 From: junya koyama Date: Wed, 7 Feb 2024 23:58:20 +0900 Subject: [PATCH 1/3] zapslog: Support for a custom mapping of slog.Level to zap.Level Signed-off-by: junya koyama --- exp/zapslog/handler.go | 22 +++----------- exp/zapslog/leveler.go | 65 ++++++++++++++++++++++++++++++++++++++++++ 2 files changed, 69 insertions(+), 18 deletions(-) create mode 100644 exp/zapslog/leveler.go diff --git a/exp/zapslog/handler.go b/exp/zapslog/handler.go index 7951d456a..617c41879 100644 --- a/exp/zapslog/handler.go +++ b/exp/zapslog/handler.go @@ -39,6 +39,7 @@ type Handler struct { addCaller bool addStackAt slog.Level callerSkip int + leveler ConvertLeveler // List of unapplied groups. // @@ -54,6 +55,7 @@ func NewHandler(core zapcore.Core, opts ...HandlerOption) *Handler { h := &Handler{ core: core, addStackAt: slog.LevelError, + leveler: &DefaultConvertLeveler{}, } for _, v := range opts { v.apply(h) @@ -113,31 +115,15 @@ func convertAttrToField(attr slog.Attr) zapcore.Field { } } -// convertSlogLevel maps slog Levels to zap Levels. -// Note that there is some room between slog levels while zap levels are continuous, so we can't 1:1 map them. -// See also https://go.googlesource.com/proposal/+/master/design/56345-structured-logging.md?pli=1#levels -func convertSlogLevel(l slog.Level) zapcore.Level { - switch { - case l >= slog.LevelError: - return zapcore.ErrorLevel - case l >= slog.LevelWarn: - return zapcore.WarnLevel - case l >= slog.LevelInfo: - return zapcore.InfoLevel - default: - return zapcore.DebugLevel - } -} - // Enabled reports whether the handler handles records at the given level. func (h *Handler) Enabled(ctx context.Context, level slog.Level) bool { - return h.core.Enabled(convertSlogLevel(level)) + return h.core.Enabled(h.leveler.ConvertLevel(level)) } // Handle handles the Record. func (h *Handler) Handle(ctx context.Context, record slog.Record) error { ent := zapcore.Entry{ - Level: convertSlogLevel(record.Level), + Level: h.leveler.ConvertLevel(record.Level), Time: record.Time, Message: record.Message, LoggerName: h.name, diff --git a/exp/zapslog/leveler.go b/exp/zapslog/leveler.go new file mode 100644 index 000000000..de61651f5 --- /dev/null +++ b/exp/zapslog/leveler.go @@ -0,0 +1,65 @@ +// Copyright (c) 2023 Uber Technologies, Inc. +// +// Permission is hereby granted, free of charge, to any person obtaining a copy +// of this software and associated documentation files (the "Software"), to deal +// in the Software without restriction, including without limitation the rights +// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +// copies of the Software, and to permit persons to whom the Software is +// furnished to do so, subject to the following conditions: +// +// The above copyright notice and this permission notice shall be included in +// all copies or substantial portions of the Software. +// +// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN +// THE SOFTWARE. + +//go:build go1.21 + +package zapslog + +import ( + "log/slog" + + "go.uber.org/zap/zapcore" +) + +// ConvertLeveler maps from [log/slog.Level] to [go.uber.org/zap/zapcore.Level]. +// Note that there is some room between slog levels while zap levels are continuous, so we can't 1:1 map them. +// See also [structured logging proposal] +// +// [structured logging proposal]: https://go.googlesource.com/proposal/+/master/design/56345-structured-logging.md?pli=1#levels +type ConvertLeveler interface { + ConvertLevel(l slog.Level) zapcore.Level +} + +// DefaultConvertLeveler maps. +// implements: [go.uber.org/zap/exp/zapslog.ConvertLeveler] +type DefaultConvertLeveler struct{} + +// ConvertLevel static maps from [log/slog.Level] to [go.uber.org/zap/zapcore.Level]. +// - [log/slog.LevelError] to [go.uber.org/zap/zapcore.ErrorLevel] +// - [log/slog.LevelWarn] to [go.uber.org/zap/zapcore.WarnLevel] +// - [log/slog.LevelInfo] to [go.uber.org/zap/zapcore.InfoLevel] +// - [log/slog.LevelDebug] or default to [go.uber.org/zap/zapcore.DebugLevel] +// +// Note that there is some room between slog levels while zap levels are continuous, so we can't 1:1 map them. +// See also [structured logging proposal] +// +// [structured logging proposal]: https://go.googlesource.com/proposal/+/master/design/56345-structured-logging.md?pli=1#levels +func (c *DefaultConvertLeveler) ConvertLevel(l slog.Level) zapcore.Level { + switch { + case l >= slog.LevelError: + return zapcore.ErrorLevel + case l >= slog.LevelWarn: + return zapcore.WarnLevel + case l >= slog.LevelInfo: + return zapcore.InfoLevel + default: + return zapcore.DebugLevel + } +} From e4c7a434284a88b18480b7962d7b963fdeb8be0b Mon Sep 17 00:00:00 2001 From: junya koyama Date: Thu, 8 Feb 2024 19:15:09 +0900 Subject: [PATCH 2/3] zapslog: Add custom mapping of ConvertLeveler example. Signed-off-by: junya koyama --- exp/zapslog/example_test.go | 52 +++++++++++++++++++++++++++++++++++++ exp/zapslog/leveler.go | 2 +- exp/zapslog/options.go | 8 ++++++ 3 files changed, 61 insertions(+), 1 deletion(-) diff --git a/exp/zapslog/example_test.go b/exp/zapslog/example_test.go index da3c63458..896edbda4 100644 --- a/exp/zapslog/example_test.go +++ b/exp/zapslog/example_test.go @@ -30,6 +30,7 @@ import ( "go.uber.org/zap" "go.uber.org/zap/exp/zapslog" + "go.uber.org/zap/zapcore" ) type Password string @@ -69,6 +70,7 @@ func Example_slog() { "foo": "bar", })) sl.LogAttrs(ctx, slog.LevelDebug, "not show up") + sl.LogAttrs(ctx, slog.LevelDebug, "not show up") // Output: // {"level":"info","msg":"user","name":"Al","secret":"REDACTED"} @@ -77,3 +79,53 @@ func Example_slog() { // {"level":"info","msg":"message","group":{"pi":3.14,"1min":"1m0s"}} // {"level":"warn","msg":"warn msg","s":{"u":1,"m":{"foo":"bar"}}} } + +type exampleDpanicLeveler struct{} + +// exampleDpanicLeveler +func (c *exampleDpanicLeveler) ConvertLevel(l slog.Level) zapcore.Level { + switch { + case l >= slog.LevelError: + return zapcore.ErrorLevel + case l >= slog.LevelWarn: + return zapcore.WarnLevel + case l >= slog.LevelInfo: + return zapcore.InfoLevel + case l == -3: + return zapcore.DPanicLevel + case l == -5: + return zapcore.FatalLevel + default: + return zapcore.DebugLevel + } +} + +func ExampleConvertLeveler() { + logger := zap.NewExample(zap.IncreaseLevel(zap.InfoLevel)) + defer logger.Sync() + + sl := slog.New(zapslog.NewHandler(logger.Core(), zapslog.WithConvertLeveler(&exampleDpanicLeveler{}))) + ctx := context.Background() + + sl.Info("user", "name", "Al", "secret", Password("secret")) + sl.Error("oops", "err", net.ErrClosed, "status", 500) + sl.LogAttrs( + ctx, + -3, + "oops", + slog.Any("err", net.ErrClosed), + slog.Int("status", 500), + ) + sl.LogAttrs( + ctx, + -5, + "oops", + slog.Any("err", net.ErrClosed), + slog.Int("status", 500), + ) + // Output: + // {"level":"info","msg":"user","name":"Al","secret":"REDACTED"} + // {"level":"error","msg":"oops","err":"use of closed network connection","status":500} + // {"level":"dpanic","msg":"oops","err":"use of closed network connection","status":500} + // {"level":"fatal","msg":"oops","err":"use of closed network connection","status":500} +} diff --git a/exp/zapslog/leveler.go b/exp/zapslog/leveler.go index de61651f5..b4e87526c 100644 --- a/exp/zapslog/leveler.go +++ b/exp/zapslog/leveler.go @@ -37,7 +37,7 @@ type ConvertLeveler interface { ConvertLevel(l slog.Level) zapcore.Level } -// DefaultConvertLeveler maps. +// DefaultConvertLeveler static maps from [log/slog.Level] to [go.uber.org/zap/zapcore.Level]. // implements: [go.uber.org/zap/exp/zapslog.ConvertLeveler] type DefaultConvertLeveler struct{} diff --git a/exp/zapslog/options.go b/exp/zapslog/options.go index cab4d0f19..3bca36138 100644 --- a/exp/zapslog/options.go +++ b/exp/zapslog/options.go @@ -70,3 +70,11 @@ func AddStacktraceAt(lvl slog.Level) HandlerOption { log.addStackAt = lvl }) } + +// WithConvertLeveler configures the ConvertLeveler to convert log levels +// from [log/slog.Level] to [go.uber.org/zap/zapcore.Level]. +func WithConvertLeveler(leveler ConvertLeveler) HandlerOption { + return handlerOptionFunc(func(handler *Handler) { + handler.leveler = leveler + }) +} From 6b841ca7c24d7c0e6a038593ffd76583388ab77e Mon Sep 17 00:00:00 2001 From: junya koyama Date: Thu, 8 Feb 2024 19:18:16 +0900 Subject: [PATCH 3/3] zapslog: remove comment Signed-off-by: junya koyama --- exp/zapslog/example_test.go | 1 - 1 file changed, 1 deletion(-) diff --git a/exp/zapslog/example_test.go b/exp/zapslog/example_test.go index 896edbda4..8b8db4435 100644 --- a/exp/zapslog/example_test.go +++ b/exp/zapslog/example_test.go @@ -82,7 +82,6 @@ func Example_slog() { type exampleDpanicLeveler struct{} -// exampleDpanicLeveler func (c *exampleDpanicLeveler) ConvertLevel(l slog.Level) zapcore.Level { switch { case l >= slog.LevelError: