Replace raw fmt/log calls with structured slog logger (Go) and console-based logger (TypeScript). Add request logging middleware. Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
59 lines
1.5 KiB
Go
59 lines
1.5 KiB
Go
package logger
|
|
|
|
import (
|
|
"log/slog"
|
|
"net/http"
|
|
"os"
|
|
"strings"
|
|
|
|
chimw "github.com/go-chi/chi/v5/middleware"
|
|
"github.com/lmittmann/tint"
|
|
)
|
|
|
|
// Init initializes the global slog logger with colored terminal output.
|
|
// Reads LOG_LEVEL env var (debug, info, warn, error). Default: debug.
|
|
func Init() {
|
|
level := parseLevel(os.Getenv("LOG_LEVEL"))
|
|
handler := tint.NewHandler(os.Stderr, &tint.Options{
|
|
Level: level,
|
|
TimeFormat: "15:04:05.000",
|
|
})
|
|
slog.SetDefault(slog.New(handler))
|
|
}
|
|
|
|
// NewLogger creates a named slog logger with colored terminal output.
|
|
// Useful for standalone processes (daemon, migrate) that want a component prefix.
|
|
func NewLogger(component string) *slog.Logger {
|
|
level := parseLevel(os.Getenv("LOG_LEVEL"))
|
|
handler := tint.NewHandler(os.Stderr, &tint.Options{
|
|
Level: level,
|
|
TimeFormat: "15:04:05.000",
|
|
})
|
|
return slog.New(handler).With("component", component)
|
|
}
|
|
|
|
// RequestAttrs extracts request_id and user_id from an HTTP request
|
|
// for use in handler-level structured logging.
|
|
func RequestAttrs(r *http.Request) []any {
|
|
attrs := make([]any, 0, 4)
|
|
if rid := chimw.GetReqID(r.Context()); rid != "" {
|
|
attrs = append(attrs, "request_id", rid)
|
|
}
|
|
if uid := r.Header.Get("X-User-ID"); uid != "" {
|
|
attrs = append(attrs, "user_id", uid)
|
|
}
|
|
return attrs
|
|
}
|
|
|
|
func parseLevel(s string) slog.Level {
|
|
switch strings.ToLower(strings.TrimSpace(s)) {
|
|
case "info":
|
|
return slog.LevelInfo
|
|
case "warn", "warning":
|
|
return slog.LevelWarn
|
|
case "error":
|
|
return slog.LevelError
|
|
default:
|
|
return slog.LevelDebug
|
|
}
|
|
}
|