mirror of https://github.com/usememos/memos
feat: add configurable `--log-level` flag (#5934)
Signed-off-by: boojack <stevenlgtm@gmail.com> Co-authored-by: boojack <stevenlgtm@gmail.com>pull/5918/merge
parent
a6024eebf1
commit
f1e2a06b46
@ -0,0 +1,28 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"io"
|
||||
"log/slog"
|
||||
"strings"
|
||||
|
||||
"github.com/pkg/errors"
|
||||
)
|
||||
|
||||
func parseSlogLevel(s string) (slog.Level, error) {
|
||||
switch strings.ToLower(s) {
|
||||
case "debug":
|
||||
return slog.LevelDebug, nil
|
||||
case "info":
|
||||
return slog.LevelInfo, nil
|
||||
case "warn":
|
||||
return slog.LevelWarn, nil
|
||||
case "error":
|
||||
return slog.LevelError, nil
|
||||
default:
|
||||
return slog.LevelInfo, errors.Errorf("unknown log level %q: must be debug, info, warn, or error", s)
|
||||
}
|
||||
}
|
||||
|
||||
func newLogger(level slog.Level, w io.Writer) *slog.Logger {
|
||||
return slog.New(slog.NewTextHandler(w, &slog.HandlerOptions{Level: level}))
|
||||
}
|
||||
@ -0,0 +1,107 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"context"
|
||||
"log/slog"
|
||||
"strings"
|
||||
"testing"
|
||||
)
|
||||
|
||||
func TestParseSlogLevel(t *testing.T) {
|
||||
tests := []struct {
|
||||
input string
|
||||
wantLevel slog.Level
|
||||
wantErr bool
|
||||
}{
|
||||
{"debug", slog.LevelDebug, false},
|
||||
{"info", slog.LevelInfo, false},
|
||||
{"warn", slog.LevelWarn, false},
|
||||
{"error", slog.LevelError, false},
|
||||
{"DEBUG", slog.LevelDebug, false},
|
||||
{"INFO", slog.LevelInfo, false},
|
||||
{"WARN", slog.LevelWarn, false},
|
||||
{"ERROR", slog.LevelError, false},
|
||||
{"invalid", slog.LevelInfo, true},
|
||||
{"", slog.LevelInfo, true},
|
||||
}
|
||||
|
||||
for _, tt := range tests {
|
||||
t.Run(tt.input, func(t *testing.T) {
|
||||
got, err := parseSlogLevel(tt.input)
|
||||
if (err != nil) != tt.wantErr {
|
||||
t.Errorf("parseSlogLevel(%q) error = %v, wantErr %v", tt.input, err, tt.wantErr)
|
||||
}
|
||||
if got != tt.wantLevel {
|
||||
t.Errorf("parseSlogLevel(%q) = %v, want %v", tt.input, got, tt.wantLevel)
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
func TestNewLoggerLevelFiltering(t *testing.T) {
|
||||
tests := []struct {
|
||||
level slog.Level
|
||||
logAt slog.Level
|
||||
msg string
|
||||
shouldAppear bool
|
||||
}{
|
||||
// debug passes all
|
||||
{slog.LevelDebug, slog.LevelDebug, "debug-msg", true},
|
||||
{slog.LevelDebug, slog.LevelInfo, "info-msg", true},
|
||||
{slog.LevelDebug, slog.LevelWarn, "warn-msg", true},
|
||||
{slog.LevelDebug, slog.LevelError, "error-msg", true},
|
||||
// info suppresses debug
|
||||
{slog.LevelInfo, slog.LevelDebug, "debug-suppressed", false},
|
||||
{slog.LevelInfo, slog.LevelInfo, "info-visible", true},
|
||||
{slog.LevelInfo, slog.LevelWarn, "warn-visible", true},
|
||||
// warn suppresses debug+info
|
||||
{slog.LevelWarn, slog.LevelDebug, "debug-suppressed", false},
|
||||
{slog.LevelWarn, slog.LevelInfo, "info-suppressed", false},
|
||||
{slog.LevelWarn, slog.LevelWarn, "warn-visible", true},
|
||||
{slog.LevelWarn, slog.LevelError, "error-visible", true},
|
||||
// error suppresses everything below
|
||||
{slog.LevelError, slog.LevelDebug, "debug-suppressed", false},
|
||||
{slog.LevelError, slog.LevelInfo, "info-suppressed", false},
|
||||
{slog.LevelError, slog.LevelWarn, "warn-suppressed", false},
|
||||
{slog.LevelError, slog.LevelError, "error-visible", true},
|
||||
}
|
||||
|
||||
for _, tt := range tests {
|
||||
var buf bytes.Buffer
|
||||
logger := newLogger(tt.level, &buf)
|
||||
logger.Log(context.TODO(), tt.logAt, tt.msg)
|
||||
|
||||
appeared := strings.Contains(buf.String(), tt.msg)
|
||||
if appeared != tt.shouldAppear {
|
||||
t.Errorf("level=%s logAt=%s msg=%q: appeared=%v want=%v",
|
||||
tt.level, tt.logAt, tt.msg, appeared, tt.shouldAppear)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func TestNewLoggerOutputFormat(t *testing.T) {
|
||||
var buf bytes.Buffer
|
||||
logger := newLogger(slog.LevelDebug, &buf)
|
||||
logger.Info("hello-world", "key", "value")
|
||||
|
||||
out := buf.String()
|
||||
if !strings.Contains(out, "hello-world") {
|
||||
t.Errorf("expected message in output, got: %s", out)
|
||||
}
|
||||
if !strings.Contains(out, "key=value") {
|
||||
t.Errorf("expected key=value attr in output, got: %s", out)
|
||||
}
|
||||
if !strings.Contains(out, "INFO") {
|
||||
t.Errorf("expected level in output, got: %s", out)
|
||||
}
|
||||
}
|
||||
|
||||
func TestNewLoggerDoesNotMutateGlobalDefault(t *testing.T) {
|
||||
original := slog.Default()
|
||||
var buf bytes.Buffer
|
||||
_ = newLogger(slog.LevelError, &buf)
|
||||
if slog.Default() != original {
|
||||
t.Error("newLogger must not change slog.Default()")
|
||||
}
|
||||
}
|
||||
Loading…
Reference in New Issue