434 lines
12 KiB
Go
434 lines
12 KiB
Go
package slog
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"log"
|
|
"log/slog"
|
|
"strings"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/google/uuid"
|
|
"go.unistack.org/micro/v3/logger"
|
|
"go.unistack.org/micro/v3/metadata"
|
|
)
|
|
|
|
// always first to have proper check
|
|
func TestStacktrace(t *testing.T) {
|
|
ctx := context.TODO()
|
|
buf := bytes.NewBuffer(nil)
|
|
l := NewLogger(logger.WithLevel(logger.ErrorLevel), logger.WithOutput(buf),
|
|
WithHandlerFunc(slog.NewTextHandler),
|
|
logger.WithAddStacktrace(true),
|
|
)
|
|
if err := l.Init(logger.WithFields("key1", "val1")); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
l.Error(ctx, "msg1", errors.New("err"))
|
|
|
|
if !bytes.Contains(buf.Bytes(), []byte(`slog_test.go:31`)) {
|
|
t.Fatalf("logger error not works, buf contains: %s", buf.Bytes())
|
|
}
|
|
}
|
|
|
|
func TestTime(t *testing.T) {
|
|
ctx := context.TODO()
|
|
buf := bytes.NewBuffer(nil)
|
|
l := NewLogger(logger.WithLevel(logger.ErrorLevel), logger.WithOutput(buf),
|
|
WithHandlerFunc(slog.NewTextHandler),
|
|
logger.WithAddStacktrace(true),
|
|
logger.WithTimeFunc(func() time.Time {
|
|
return time.Unix(0, 0)
|
|
}),
|
|
)
|
|
if err := l.Init(logger.WithFields("key1", "val1")); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
l.Error(ctx, "msg1", errors.New("err"))
|
|
|
|
if !bytes.Contains(buf.Bytes(), []byte(`timestamp=1970-01-01T03:00:00.000000000+03:00`)) &&
|
|
!bytes.Contains(buf.Bytes(), []byte(`timestamp=1970-01-01T00:00:00.000000000Z`)) {
|
|
t.Fatalf("logger error not works, buf contains: %s", buf.Bytes())
|
|
}
|
|
}
|
|
|
|
func TestWithFields(t *testing.T) {
|
|
ctx := context.TODO()
|
|
buf := bytes.NewBuffer(nil)
|
|
l := NewLogger(logger.WithLevel(logger.InfoLevel), logger.WithOutput(buf),
|
|
WithHandlerFunc(slog.NewTextHandler),
|
|
logger.WithDedupKeys(true),
|
|
)
|
|
if err := l.Init(logger.WithFields("key1", "val1")); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
l.Info(ctx, "msg1")
|
|
|
|
l = l.Fields("key1", "val2")
|
|
|
|
l.Info(ctx, "msg2")
|
|
|
|
if !bytes.Contains(buf.Bytes(), []byte(`msg=msg2 key1=val1`)) {
|
|
t.Fatalf("logger error not works, buf contains: %s", buf.Bytes())
|
|
}
|
|
}
|
|
|
|
func TestWithDedupKeysWithAddFields(t *testing.T) {
|
|
ctx := context.TODO()
|
|
buf := bytes.NewBuffer(nil)
|
|
l := NewLogger(logger.WithLevel(logger.InfoLevel), logger.WithOutput(buf),
|
|
WithHandlerFunc(slog.NewTextHandler),
|
|
logger.WithDedupKeys(true),
|
|
)
|
|
if err := l.Init(logger.WithFields("key1", "val1")); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
l.Info(ctx, "msg1")
|
|
|
|
if err := l.Init(logger.WithAddFields("key2", "val2")); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
l.Info(ctx, "msg2")
|
|
|
|
if err := l.Init(logger.WithAddFields("key2", "val3", "key1", "val4")); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
l.Info(ctx, "msg3")
|
|
|
|
if !bytes.Contains(buf.Bytes(), []byte(`msg=msg3 key1=val1 key2=val2`)) {
|
|
t.Fatalf("logger error not works, buf contains: %s", buf.Bytes())
|
|
}
|
|
}
|
|
|
|
func TestWithHandlerFunc(t *testing.T) {
|
|
ctx := context.TODO()
|
|
buf := bytes.NewBuffer(nil)
|
|
l := NewLogger(logger.WithLevel(logger.InfoLevel), logger.WithOutput(buf),
|
|
WithHandlerFunc(slog.NewTextHandler),
|
|
)
|
|
if err := l.Init(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
l.Info(ctx, "msg1")
|
|
|
|
if !bytes.Contains(buf.Bytes(), []byte(`msg=msg1`)) {
|
|
t.Fatalf("logger error not works, buf contains: %s", buf.Bytes())
|
|
}
|
|
}
|
|
|
|
func TestWithAddFields(t *testing.T) {
|
|
ctx := context.TODO()
|
|
buf := bytes.NewBuffer(nil)
|
|
l := NewLogger(logger.WithLevel(logger.InfoLevel), logger.WithOutput(buf))
|
|
if err := l.Init(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
l.Info(ctx, "msg1")
|
|
|
|
if err := l.Init(logger.WithAddFields("key1", "val1")); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
l.Info(ctx, "msg2")
|
|
|
|
if err := l.Init(logger.WithAddFields("key2", "val2")); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
l.Info(ctx, "msg3")
|
|
|
|
if !bytes.Contains(buf.Bytes(), []byte(`"key1"`)) {
|
|
t.Fatalf("logger error not works, buf contains: %s", buf.Bytes())
|
|
}
|
|
if !bytes.Contains(buf.Bytes(), []byte(`"key2"`)) {
|
|
t.Fatalf("logger error not works, buf contains: %s", buf.Bytes())
|
|
}
|
|
}
|
|
|
|
func TestMultipleFieldsWithLevel(t *testing.T) {
|
|
ctx := context.TODO()
|
|
buf := bytes.NewBuffer(nil)
|
|
l := NewLogger(logger.WithLevel(logger.InfoLevel), logger.WithOutput(buf))
|
|
if err := l.Init(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
l = l.Fields("key", "val")
|
|
|
|
l.Info(ctx, "msg1")
|
|
nl := l.Clone(logger.WithLevel(logger.DebugLevel))
|
|
nl.Debug(ctx, "msg2")
|
|
l.Debug(ctx, "msg3")
|
|
if !bytes.Contains(buf.Bytes(), []byte(`"key":"val"`)) {
|
|
t.Fatalf("logger error not works, buf contains: %s", buf.Bytes())
|
|
}
|
|
if !bytes.Contains(buf.Bytes(), []byte(`"msg1"`)) {
|
|
t.Fatalf("logger error not works, buf contains: %s", buf.Bytes())
|
|
}
|
|
if !bytes.Contains(buf.Bytes(), []byte(`"msg2"`)) {
|
|
t.Fatalf("logger error not works, buf contains: %s", buf.Bytes())
|
|
}
|
|
if bytes.Contains(buf.Bytes(), []byte(`"msg3"`)) {
|
|
t.Fatalf("logger error not works, buf contains: %s", buf.Bytes())
|
|
}
|
|
}
|
|
|
|
func TestMultipleFields(t *testing.T) {
|
|
ctx := context.TODO()
|
|
buf := bytes.NewBuffer(nil)
|
|
l := NewLogger(logger.WithLevel(logger.InfoLevel), logger.WithOutput(buf))
|
|
if err := l.Init(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
l = l.Fields("key", "val")
|
|
|
|
l = l.Fields("key1", "val1")
|
|
|
|
l.Info(ctx, "msg")
|
|
|
|
if !bytes.Contains(buf.Bytes(), []byte(`"key":"val"`)) {
|
|
t.Fatalf("logger error not works, buf contains: %s", buf.Bytes())
|
|
}
|
|
if !bytes.Contains(buf.Bytes(), []byte(`"key1":"val1"`)) {
|
|
t.Fatalf("logger error not works, buf contains: %s", buf.Bytes())
|
|
}
|
|
}
|
|
|
|
func TestError(t *testing.T) {
|
|
ctx := context.TODO()
|
|
buf := bytes.NewBuffer(nil)
|
|
l := NewLogger(logger.WithLevel(logger.ErrorLevel), logger.WithOutput(buf), logger.WithAddStacktrace(true))
|
|
if err := l.Init(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
l.Error(ctx, "message", fmt.Errorf("error message"))
|
|
if !bytes.Contains(buf.Bytes(), []byte(`"stacktrace":"`)) {
|
|
t.Fatalf("logger stacktrace not works, buf contains: %s", buf.Bytes())
|
|
}
|
|
if !bytes.Contains(buf.Bytes(), []byte(`"error":"`)) {
|
|
t.Fatalf("logger error not works, buf contains: %s", buf.Bytes())
|
|
}
|
|
}
|
|
|
|
func TestErrorf(t *testing.T) {
|
|
ctx := context.TODO()
|
|
|
|
buf := bytes.NewBuffer(nil)
|
|
l := NewLogger(logger.WithLevel(logger.ErrorLevel), logger.WithOutput(buf), logger.WithAddStacktrace(true))
|
|
if err := l.Init(logger.WithContextAttrFuncs(func(_ context.Context) []interface{} {
|
|
return nil
|
|
})); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
l.Log(ctx, logger.ErrorLevel, "message", errors.New("error msg"))
|
|
|
|
l.Log(ctx, logger.ErrorLevel, "", errors.New("error msg"))
|
|
if !bytes.Contains(buf.Bytes(), []byte(`"error":"error msg"`)) {
|
|
t.Fatalf("logger error not works, buf contains: %s", buf.Bytes())
|
|
}
|
|
|
|
if !bytes.Contains(buf.Bytes(), []byte(`"stacktrace":"`)) {
|
|
t.Fatalf("logger stacktrace not works, buf contains: %s", buf.Bytes())
|
|
}
|
|
if !bytes.Contains(buf.Bytes(), []byte(`"error":"`)) {
|
|
t.Fatalf("logger error not works, buf contains: %s", buf.Bytes())
|
|
}
|
|
}
|
|
|
|
func TestContext(t *testing.T) {
|
|
ctx := context.TODO()
|
|
buf := bytes.NewBuffer(nil)
|
|
l := NewLogger(logger.WithLevel(logger.TraceLevel), logger.WithOutput(buf))
|
|
if err := l.Init(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
nl, ok := logger.FromContext(logger.NewContext(ctx, l.Fields("key", "val")))
|
|
if !ok {
|
|
t.Fatal("context without logger")
|
|
}
|
|
nl.Info(ctx, "message")
|
|
if !bytes.Contains(buf.Bytes(), []byte(`"key":"val"`)) {
|
|
t.Fatalf("logger fields not works, buf contains: %s", buf.Bytes())
|
|
}
|
|
}
|
|
|
|
func TestFields(t *testing.T) {
|
|
ctx := context.TODO()
|
|
buf := bytes.NewBuffer(nil)
|
|
l := NewLogger(logger.WithLevel(logger.TraceLevel), logger.WithOutput(buf))
|
|
if err := l.Init(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
nl := l.Fields("key", "val")
|
|
|
|
nl.Info(ctx, "message")
|
|
if !bytes.Contains(buf.Bytes(), []byte(`"key":"val"`)) {
|
|
t.Fatalf("logger fields not works, buf contains: %s", buf.Bytes())
|
|
}
|
|
}
|
|
|
|
func TestFromContextWithFields(t *testing.T) {
|
|
ctx := context.TODO()
|
|
buf := bytes.NewBuffer(nil)
|
|
var ok bool
|
|
l := NewLogger(logger.WithLevel(logger.TraceLevel), logger.WithOutput(buf))
|
|
if err := l.Init(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
nl := l.Fields("key", "val")
|
|
|
|
ctx = logger.NewContext(ctx, nl)
|
|
|
|
l, ok = logger.FromContext(ctx)
|
|
if !ok {
|
|
t.Fatalf("context does not have logger")
|
|
}
|
|
|
|
l.Info(ctx, "message")
|
|
if !bytes.Contains(buf.Bytes(), []byte(`"key":"val"`)) {
|
|
t.Fatalf("logger fields not works, buf contains: %s", buf.Bytes())
|
|
}
|
|
|
|
l.Info(ctx, "test", "uncorrected number attributes")
|
|
if !bytes.Contains(buf.Bytes(), []byte(`"!BADKEY":"uncorrected number attributes"`)) {
|
|
t.Fatalf("logger fields not works, buf contains: %s", buf.Bytes())
|
|
}
|
|
}
|
|
|
|
func TestClone(t *testing.T) {
|
|
ctx := context.TODO()
|
|
buf := bytes.NewBuffer(nil)
|
|
l := NewLogger(logger.WithLevel(logger.TraceLevel), logger.WithOutput(buf))
|
|
if err := l.Init(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
nl := l.Clone(logger.WithLevel(logger.ErrorLevel))
|
|
if err := nl.Init(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
nl.Info(ctx, "info message")
|
|
if len(buf.Bytes()) != 0 {
|
|
t.Fatal("message must not be logged")
|
|
}
|
|
l.Info(ctx, "info message")
|
|
if len(buf.Bytes()) == 0 {
|
|
t.Fatal("message must be logged")
|
|
}
|
|
}
|
|
|
|
func TestRedirectStdLogger(t *testing.T) {
|
|
buf := bytes.NewBuffer(nil)
|
|
l := NewLogger(logger.WithLevel(logger.ErrorLevel), logger.WithOutput(buf))
|
|
if err := l.Init(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
fn := logger.RedirectStdLogger(l, logger.ErrorLevel)
|
|
defer fn()
|
|
log.Print("test")
|
|
if !(bytes.Contains(buf.Bytes(), []byte(`"level":"error"`)) && bytes.Contains(buf.Bytes(), []byte(`"msg":"test"`))) {
|
|
t.Fatalf("logger error, buf %s", buf.Bytes())
|
|
}
|
|
}
|
|
|
|
func TestStdLogger(t *testing.T) {
|
|
buf := bytes.NewBuffer(nil)
|
|
l := NewLogger(logger.WithLevel(logger.TraceLevel), logger.WithOutput(buf))
|
|
if err := l.Init(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
lg := logger.NewStdLogger(l, logger.ErrorLevel)
|
|
lg.Print("test")
|
|
if !(bytes.Contains(buf.Bytes(), []byte(`"level":"error"`)) && bytes.Contains(buf.Bytes(), []byte(`"msg":"test"`))) {
|
|
t.Fatalf("logger error, buf %s", buf.Bytes())
|
|
}
|
|
}
|
|
|
|
func TestLogger(t *testing.T) {
|
|
ctx := context.TODO()
|
|
buf := bytes.NewBuffer(nil)
|
|
l := NewLogger(logger.WithLevel(logger.TraceLevel), logger.WithOutput(buf))
|
|
if err := l.Init(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
l.Trace(ctx, "trace_msg1")
|
|
l.Warn(ctx, "warn_msg1")
|
|
l.Fields("error", "test").Info(ctx, "error message")
|
|
l.Warn(ctx, "first second")
|
|
|
|
if !(bytes.Contains(buf.Bytes(), []byte(`"level":"trace"`)) && bytes.Contains(buf.Bytes(), []byte(`"msg":"trace_msg1"`))) {
|
|
t.Fatalf("logger tracer, buf %s", buf.Bytes())
|
|
}
|
|
if !(bytes.Contains(buf.Bytes(), []byte(`"level":"warn"`)) && bytes.Contains(buf.Bytes(), []byte(`"msg":"warn_msg1"`))) {
|
|
t.Fatalf("logger warn, buf %s", buf.Bytes())
|
|
}
|
|
if !(bytes.Contains(buf.Bytes(), []byte(`"level":"info"`)) && bytes.Contains(buf.Bytes(), []byte(`"msg":"error message","error":"test"`))) {
|
|
t.Fatalf("logger info, buf %s", buf.Bytes())
|
|
}
|
|
if !(bytes.Contains(buf.Bytes(), []byte(`"level":"warn"`)) && bytes.Contains(buf.Bytes(), []byte(`"msg":"first second"`))) {
|
|
t.Fatalf("logger warn, buf %s", buf.Bytes())
|
|
}
|
|
}
|
|
|
|
func Test_WithContextAttrFunc(t *testing.T) {
|
|
loggerContextAttrFuncs := []logger.ContextAttrFunc{
|
|
func(ctx context.Context) []interface{} {
|
|
md, ok := metadata.FromIncomingContext(ctx)
|
|
if !ok {
|
|
return nil
|
|
}
|
|
attrs := make([]interface{}, 0, 10)
|
|
for k, v := range md {
|
|
switch k {
|
|
case "X-Request-Id", "Phone", "External-Id", "Source-Service", "X-App-Install-Id", "Client-Id", "Client-Ip":
|
|
attrs = append(attrs, strings.ToLower(k), v)
|
|
}
|
|
}
|
|
return attrs
|
|
},
|
|
}
|
|
|
|
logger.DefaultContextAttrFuncs = append(logger.DefaultContextAttrFuncs, loggerContextAttrFuncs...)
|
|
|
|
ctx := context.TODO()
|
|
ctx = metadata.AppendIncomingContext(ctx, "X-Request-Id", uuid.New().String(),
|
|
"Source-Service", "Test-System")
|
|
|
|
buf := bytes.NewBuffer(nil)
|
|
l := NewLogger(logger.WithLevel(logger.TraceLevel), logger.WithOutput(buf))
|
|
if err := l.Init(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
l.Info(ctx, "test message")
|
|
if !(bytes.Contains(buf.Bytes(), []byte(`"level":"info"`)) && bytes.Contains(buf.Bytes(), []byte(`"msg":"test message"`))) {
|
|
t.Fatalf("logger info, buf %s", buf.Bytes())
|
|
}
|
|
if !(bytes.Contains(buf.Bytes(), []byte(`"x-request-id":"`))) {
|
|
t.Fatalf("logger info, buf %s", buf.Bytes())
|
|
}
|
|
if !(bytes.Contains(buf.Bytes(), []byte(`"source-service":"Test-System"`))) {
|
|
t.Fatalf("logger info, buf %s", buf.Bytes())
|
|
}
|
|
buf.Reset()
|
|
imd, _ := metadata.FromIncomingContext(ctx)
|
|
l.Info(ctx, "test message1")
|
|
imd.Set("Source-Service", "Test-System2")
|
|
l.Info(ctx, "test message2")
|
|
|
|
// t.Logf("xxx %s", buf.Bytes())
|
|
}
|