1
0
mirror of https://github.com/go-kratos/kratos.git synced 2025-03-27 21:49:08 +02:00
kratos/log/filter_test.go
ionling ff105d5bca
fix(log): WithContext() changed the ctx field of the parent log.Filter (#3069)
* fix(log): `WithContext()` changed the ctx field of the parent log.Filter

* test(log): add concurrence test for `WithContext()`

* fix(log): concurrence problem of `Filter`

---------

Co-authored-by: 包子 <baozhecheng@foxmail.com>
2023-11-22 11:35:58 +08:00

226 lines
5.4 KiB
Go

package log
import (
"bytes"
"context"
"io"
"strings"
"sync"
"testing"
"time"
)
func TestFilterAll(_ *testing.T) {
logger := With(DefaultLogger, "ts", DefaultTimestamp, "caller", DefaultCaller)
log := NewHelper(NewFilter(logger,
FilterLevel(LevelDebug),
FilterKey("username"),
FilterValue("hello"),
FilterFunc(testFilterFunc),
))
log.Log(LevelDebug, "msg", "test debug")
log.Info("hello")
log.Infow("password", "123456")
log.Infow("username", "kratos")
log.Warn("warn log")
}
func TestFilterLevel(_ *testing.T) {
logger := With(DefaultLogger, "ts", DefaultTimestamp, "caller", DefaultCaller)
log := NewHelper(NewFilter(NewFilter(logger, FilterLevel(LevelWarn))))
log.Log(LevelDebug, "msg1", "te1st debug")
log.Debug("test debug")
log.Debugf("test %s", "debug")
log.Debugw("log", "test debug")
log.Warn("warn log")
}
func TestFilterCaller(_ *testing.T) {
logger := With(DefaultLogger, "ts", DefaultTimestamp, "caller", DefaultCaller)
log := NewFilter(logger)
_ = log.Log(LevelDebug, "msg1", "te1st debug")
logHelper := NewHelper(NewFilter(logger))
logHelper.Log(LevelDebug, "msg1", "te1st debug")
}
func TestFilterKey(_ *testing.T) {
logger := With(DefaultLogger, "ts", DefaultTimestamp, "caller", DefaultCaller)
log := NewHelper(NewFilter(logger, FilterKey("password")))
log.Debugw("password", "123456")
}
func TestFilterValue(_ *testing.T) {
logger := With(DefaultLogger, "ts", DefaultTimestamp, "caller", DefaultCaller)
log := NewHelper(NewFilter(logger, FilterValue("debug")))
log.Debugf("test %s", "debug")
}
func TestFilterFunc(_ *testing.T) {
logger := With(DefaultLogger, "ts", DefaultTimestamp, "caller", DefaultCaller)
log := NewHelper(NewFilter(logger, FilterFunc(testFilterFunc)))
log.Debug("debug level")
log.Infow("password", "123456")
}
func BenchmarkFilterKey(b *testing.B) {
log := NewHelper(NewFilter(NewStdLogger(io.Discard), FilterKey("password")))
for i := 0; i < b.N; i++ {
log.Infow("password", "123456")
}
}
func BenchmarkFilterValue(b *testing.B) {
log := NewHelper(NewFilter(NewStdLogger(io.Discard), FilterValue("password")))
for i := 0; i < b.N; i++ {
log.Infow("password")
}
}
func BenchmarkFilterFunc(b *testing.B) {
log := NewHelper(NewFilter(NewStdLogger(io.Discard), FilterFunc(testFilterFunc)))
for i := 0; i < b.N; i++ {
log.Info("password", "123456")
}
}
func testFilterFunc(level Level, keyvals ...interface{}) bool {
if level == LevelWarn {
return true
}
for i := 0; i < len(keyvals); i++ {
if keyvals[i] == "password" {
keyvals[i+1] = fuzzyStr
}
}
return false
}
func TestFilterFuncWitchLoggerPrefix(t *testing.T) {
buf := new(bytes.Buffer)
tests := []struct {
logger Logger
want string
}{
{
logger: NewFilter(With(NewStdLogger(buf), "caller", "caller", "prefix", "whaterver"), FilterFunc(testFilterFuncWithLoggerPrefix)),
want: "",
},
{
// Filtered value
logger: NewFilter(With(NewStdLogger(buf), "caller", "caller"), FilterFunc(testFilterFuncWithLoggerPrefix)),
want: "INFO caller=caller msg=msg filtered=***\n",
},
{
// NO prefix
logger: NewFilter(With(NewStdLogger(buf)), FilterFunc(testFilterFuncWithLoggerPrefix)),
want: "INFO msg=msg filtered=***\n",
},
}
for _, tt := range tests {
err := tt.logger.Log(LevelInfo, "msg", "msg", "filtered", "true")
if err != nil {
t.Fatal("err should be nil")
}
got := buf.String()
if got != tt.want {
t.Fatalf("filter should catch prefix, want %s, got %s.", tt.want, got)
}
buf.Reset()
}
}
func testFilterFuncWithLoggerPrefix(level Level, keyvals ...interface{}) bool {
if level == LevelWarn {
return true
}
for i := 0; i < len(keyvals); i += 2 {
if keyvals[i] == "prefix" {
return true
}
if keyvals[i] == "filtered" {
keyvals[i+1] = fuzzyStr
}
}
return false
}
func TestFilterWithContext(t *testing.T) {
ctxKey := struct{}{}
ctxValue := "filter test value"
v1 := func() Valuer {
return func(ctx context.Context) interface{} {
return ctx.Value(ctxKey)
}
}
info := &bytes.Buffer{}
logger := With(NewStdLogger(info), "request_id", v1())
filter := NewFilter(logger, FilterLevel(LevelError))
ctx := context.WithValue(context.Background(), ctxKey, ctxValue)
_ = WithContext(ctx, filter).Log(LevelInfo, "kind", "test")
if info.String() != "" {
t.Error("filter is not woring")
return
}
_ = WithContext(ctx, filter).Log(LevelError, "kind", "test")
if !strings.Contains(info.String(), ctxValue) {
t.Error("don't read ctx value")
}
}
type traceIDKey struct{}
func setTraceID(ctx context.Context, tid string) context.Context {
return context.WithValue(ctx, traceIDKey{}, tid)
}
func traceIDValuer() Valuer {
return func(ctx context.Context) any {
if ctx == nil {
return ""
}
if tid := ctx.Value(traceIDKey{}); tid != nil {
return tid
}
return ""
}
}
func TestFilterWithContextConcurrent(t *testing.T) {
var buf bytes.Buffer
pctx := context.Background()
l := NewFilter(
With(NewStdLogger(&buf), "trace-id", traceIDValuer()),
FilterLevel(LevelInfo),
)
var wg sync.WaitGroup
wg.Add(1)
go func() {
defer wg.Done()
time.Sleep(time.Second)
NewHelper(l).Info("done1")
}()
wg.Add(1)
go func() {
defer wg.Done()
tid := "world"
ctx := setTraceID(pctx, tid)
NewHelper((WithContext(ctx, l))).Info("done2")
}()
wg.Wait()
expected := "INFO trace-id=world msg=done2\nINFO trace-id= msg=done1\n"
if got := buf.String(); got != expected {
t.Errorf("got: %#v", got)
}
}