是否可以在运行时更新 zap 记录器的日志级别?

use*_*697 8 logging go go-zap

我创建了一个记录器kubebuilder,它基于 zap 记录器:

import (
    "flag"
    "github.com/gin-gonic/gin"
    "net/http"
    "os"
    "go.uber.org/zap/zapcore"
    uzap "go.uber.org/zap"
    // Import all Kubernetes client auth plugins (e.g. Azure, GCP, OIDC, etc.)
    // to ensure that exec-entrypoint and run can make use of them.
    _ "k8s.io/client-go/plugin/pkg/client/auth"

    "k8s.io/apimachinery/pkg/runtime"
    utilruntime "k8s.io/apimachinery/pkg/util/runtime"
    clientgoscheme "k8s.io/client-go/kubernetes/scheme"
    ctrl "sigs.k8s.io/controller-runtime"
    "sigs.k8s.io/controller-runtime/pkg/healthz"
    "sigs.k8s.io/controller-runtime/pkg/log/zap"

)

var (
    scheme   = runtime.NewScheme()
    setupLog = ctrl.Log.WithName("setup")
)

var zapOpts []uzap.Option
    zapOpts = append(zapOpts, uzap.AddCaller())
    zapOpts = append(zapOpts, uzap.AddCallerSkip(1))
    zapOpts = append(zapOpts, uzap.AddStacktrace(uzap.DebugLevel))

    opts := zap.Options{
        Development:     developmentFlag,
        StacktraceLevel: stacktraceLevel,
        Level:           level,
        Encoder:         encoder,
        ZapOpts:  zapOpts,
    }

    opts.BindFlags(flag.CommandLine)
    flag.Parse()

    ctrl.SetLogger(zap.New(zap.UseFlagOptions(&opts)))
Run Code Online (Sandbox Code Playgroud)

现在我想将日志级别更改为zapcore.InfoLevel运行时。我没有找到任何SetLogLevel或类似的 API。

我需要创建新选项然后设置新级别吗?

我还需要使用sigs.k8s.io/controller-runtime/pkg/log/zap库设置记录器。记录器的接口来自go-logr并实现logr.Logger接口。如果我尝试将其更改为,zapcore.NewCore我将无法再设置记录器ctrl.SetLogger

我想保留更新所有选项的选项zap.Options 以及更改日志级别的选项,并且仍然使用来自 的 zap sigs.k8s.io/controller-runtime/pkg/log/zap

可以用 sigs.k8s.io/controller-runtime/pkg/log/zap and来做吗sigs.k8s.io/controller-runtime

Oli*_*ain 9

是的,可以使用AtomicLevel. 来自文档:

atom := zap.NewAtomicLevel()

// To keep the example deterministic, disable timestamps in the output.
encoderCfg := zap.NewProductionEncoderConfig()
encoderCfg.TimeKey = ""

logger := zap.New(zapcore.NewCore(
    zapcore.NewJSONEncoder(encoderCfg),
    zapcore.Lock(os.Stdout),
    atom,
))
defer logger.Sync()

logger.Info("info logging enabled")
atom.SetLevel(zap.ErrorLevel)
logger.Info("info logging disabled")
Run Code Online (Sandbox Code Playgroud)


bla*_*een 6

更好的答案:按照@Oliver Dain 的建议,使用zap.AtomicLevel. 详情请参阅他们的回答。

另一种选择是创建具有自定义LevelEnabler功能的核心。您可以使用zap.LevelEnablerFunc将闭包转换为zapcore.LevelEnabler.

相关文档:

LevelEnabler 决定在记录消息时是否启用给定的日志记录级别。

LevelEnablerFunc 是使用匿名函数实现 zapcore.LevelEnabler 的便捷方法。

然后该函数可能会返回truefalse基于在运行时更改的其他一些变量:

    // will be actually initialized and changed at run time 
    // based on your business logic
    var infoEnabled bool 

    errorUnlessEnabled := zap.LevelEnablerFunc(func(level zapcore.Level) bool {
        // true: log message at this level
        // false: skip message at this level
        return level >= zapcore.ErrorLevel || infoEnabled
    })

    core := zapcore.NewCore(
        zapcore.NewJSONEncoder(zap.NewProductionEncoderConfig()),
        os.Stdout,
        errorUnlessEnabled,
    )
    logger := zap.New(core)

    logger.Info("foo") // not logged
    
    infoEnabled = true

    logger.Info("foo again") // logged
Run Code Online (Sandbox Code Playgroud)

PS:这段代码是人为的。您的程序必须实现初始化、运行时值更改以及对infoEnabled变量的正确同步(如果需要)。

您可以在操场上运行此示例:https ://play.golang.org/p/oT3nvnP1Bwc