Golang 结构化日志包 log/slog 详解(五):LogValuer 和函数包装

上一篇文章讲解了 log/slog 包中的分组、上下文和属性值类型,本文讲解下 LogValuer 和日志记录函数的正确包装方法。

slog.LogValuer

如果想改变或者自定义一个类型的日志记录行为,可以通过实现 slog.LogValuer 接口来实现,slog.LogValuer 接口的定义如下:

type LogValuer interface {
	LogValue() Value
}

定义了一个 LogValue 方法,返回一个 Value 类型的对象。如果一个类型实现了 LogValuer 接口,那么从它的 LogValue 方法返回的 Value 将被用于日志记录,可以用来控制该类型的值在日志中的显示方式。看个简单示例:

package main

import (
	"log/slog"
	"os"
)

type Token string

// 实现 slog.LogValuer 接口
// 避免泄露 token
func (Token) LogValue() slog.Value {
	return slog.StringValue("******")
}

func main() {
	t := Token("shhhh!")
	logger := slog.New(slog.NewTextHandler(os.Stdout, nil))
	logger.Info("生成 token", "用户", "路多辛的博客", "token", t)
}

输出内容如下:

time=2023-10-15T15:06:58.253+08:00 level=INFO msg="生成 token" 用户=路多辛的博客 token=******

从安全角度看,密码或者token 等敏感信息是不能被记录在日志里面的,可以使用自定义的并且实现了 LogValue 的类型来避免这种情况产生。在这个例子中,当记录 token 日志时,token 会被转换为"******"后记录在日志里面。再看一个结合字段分组使用的示例:

package main

import (
	"log/slog"
)

type Name struct {
	First, Last string
}

func (n Name) LogValue() slog.Value {
	return slog.GroupValue(
		slog.String("first", n.First),
		slog.String("last", n.Last))
}

func main() {
	n := Name{"路多辛的博客", "路多辛的所思所想"}
	slog.Info("任务结束", "agent", n)
}

输出内容如下:

2023/10/15 15:06:09 INFO 任务结束 agent.first=路多辛的博客 agent.last=路多辛的所思所想

包装输出函数

日志记录函数使用调用堆栈上的反射来查找应用程序中日志记录调用的文件名和行号,这可能会导致包装 slog 的函数记录错误的的源信息。举个例子,如果在 mylog.go 中定义了一个日志记录函数 Infof,然后在 main.go 中调用了此函数,这种情况下,日志会将把源文件记录为 mylog.go 而不是 main.go。正确实现 Infof 函数的方式是将获取的源信息传递给 NewRecord 函数,示例代码如下:

package main

import (
	"context"
	"fmt"
	"log/slog"
	"os"
	"path/filepath"
	"runtime"
	"time"
)

func Infof(logger *slog.Logger, format string, args ...any) {
	if !logger.Enabled(context.Background(), slog.LevelInfo) {
		return
	}
	var pcs [1]uintptr
	runtime.Callers(2, pcs[:]) // skip [Callers, Infof]
	r := slog.NewRecord(time.Now(), slog.LevelInfo, fmt.Sprintf(format, args...), pcs[0])
	_ = logger.Handler().Handle(context.Background(), r)
}

func main() {
	replace := func(groups []string, a slog.Attr) slog.Attr {
		// Remove time.
		if a.Key == slog.TimeKey && len(groups) == 0 {
			return slog.Attr{}
		}
		// Remove the directory from the source's filename.
		if a.Key == slog.SourceKey {
			source := a.Value.Any().(*slog.Source)
			source.File = filepath.Base(source.File)
		}
		return a
	}
	logger := slog.New(slog.NewTextHandler(os.Stdout, &slog.HandlerOptions{AddSource: true, ReplaceAttr: replace}))
	Infof(logger, "message, %s", "formatted")
}
相关推荐
昙鱼4 分钟前
springboot创建web项目
java·前端·spring boot·后端·spring·maven
天之涯上上8 分钟前
JAVA开发 在 Spring Boot 中集成 Swagger
java·开发语言·spring boot
2402_857583499 分钟前
“协同过滤技术实战”:网上书城系统的设计与实现
java·开发语言·vue.js·科技·mfc
白宇横流学长10 分钟前
基于SpringBoot的停车场管理系统设计与实现【源码+文档+部署讲解】
java·spring boot·后端
kirito学长-Java15 分钟前
springboot/ssm太原学院商铺管理系统Java代码编写web在线购物商城
java·spring boot·后端
爱学习的白杨树16 分钟前
MyBatis的一级、二级缓存
java·开发语言·spring
OTWOL21 分钟前
两道数组有关的OJ练习题
c语言·开发语言·数据结构·c++·算法
问道飞鱼25 分钟前
【前端知识】强大的js动画组件anime.js
开发语言·前端·javascript·anime.js
拓端研究室25 分钟前
R基于贝叶斯加法回归树BART、MCMC的DLNM分布滞后非线性模型分析母婴PM2.5暴露与出生体重数据及GAM模型对比、关键窗口识别
android·开发语言·kotlin
Code成立26 分钟前
《Java核心技术I》Swing的网格包布局
java·开发语言·swing