GitRoot

craft your forge, build your project, grow your community freely
  1// SPDX-FileCopyrightText: 2025 Romain Maneschi <romain@gitroot.dev>
  2//
  3// SPDX-License-Identifier: EUPL-1.2
  4
  5package logger
  6
  7import (
  8	"context"
  9	"crypto/rand"
 10	"encoding/hex"
 11	"fmt"
 12	"log/slog"
 13	"os"
 14)
 15
 16const (
 17	FS_LOGGER_NAME          LoggerName = "GitRootFs"
 18	SSH_SERVER_LOGGER_NAME  LoggerName = "ServerSsh"
 19	HTTP_SERVER_LOGGER_NAME LoggerName = "ServerHttp"
 20	FS_STORER_LOGGER_NAME   LoggerName = "GitRootFsStorer"
 21	FS_PLUGIN               LoggerName = "FsPlugin"
 22	REPOSITORY_MANAGER      LoggerName = "RepositoryManager"
 23	PLUGIN_MANAGER          LoggerName = "PluginManager"
 24	BACKGROUND_MANAGER      LoggerName = "BackgroundManager"
 25	WASM                    LoggerName = "Wasm"
 26	EXEC                    LoggerName = "Exec"
 27)
 28
 29var levelByName = map[LoggerName]slog.Level{
 30	FS_LOGGER_NAME:          slog.LevelWarn,
 31	SSH_SERVER_LOGGER_NAME:  slog.LevelWarn,
 32	HTTP_SERVER_LOGGER_NAME: slog.LevelWarn,
 33	FS_STORER_LOGGER_NAME:   slog.LevelWarn,
 34	FS_PLUGIN:               slog.LevelWarn,
 35	REPOSITORY_MANAGER:      slog.LevelWarn,
 36	PLUGIN_MANAGER:          slog.LevelWarn,
 37	BACKGROUND_MANAGER:      slog.LevelWarn,
 38	WASM:                    slog.LevelWarn,
 39	EXEC:                    slog.LevelWarn,
 40}
 41
 42type LoggerName string
 43
 44type Logger struct {
 45	c     context.Context
 46	l     *slog.Logger
 47	level slog.Level
 48	name  LoggerName
 49}
 50
 51type LoggerPair struct {
 52	key   string
 53	value any
 54}
 55
 56func NewLogger(name LoggerName) *Logger {
 57	return NewLoggerCtx(name, context.Background())
 58}
 59
 60func NewLoggerCtx(name LoggerName, ctx context.Context) *Logger {
 61	var level = slog.LevelWarn
 62	if l, ok := levelByName[name]; ok {
 63		level = l
 64	}
 65	log := slog.New(slog.NewTextHandler(os.Stdout, &slog.HandlerOptions{Level: level}))
 66	l := &Logger{
 67		c:     ctx,
 68		l:     log,
 69		level: level,
 70		name:  name,
 71	}
 72	return l.With("name", string(name))
 73}
 74
 75func GenerateRandomID() string {
 76	bytes := make([]byte, 16)
 77	if _, err := rand.Read(bytes); err != nil {
 78		panic(err)
 79	}
 80	return hex.EncodeToString(bytes)
 81}
 82
 83func NewLoggerPair(key string, value any) LoggerPair {
 84	return LoggerPair{
 85		key:   key,
 86		value: value,
 87	}
 88}
 89
 90func (l *Logger) NewSubLogger(name string) *Logger {
 91	return l.NewSubLoggerCtx(name, l.c)
 92}
 93
 94func (l *Logger) NewSubLoggerCtx(name string, ctx context.Context) *Logger {
 95	subName := fmt.Sprintf("%s::%s", l.name, name)
 96	log := slog.New(slog.NewTextHandler(os.Stdout, &slog.HandlerOptions{Level: l.level}))
 97	sub := &Logger{
 98		c:     ctx,
 99		l:     log,
100		level: l.level,
101		name:  LoggerName(subName),
102	}
103	return sub.With("name", subName)
104}
105
106func (l *Logger) With(key string, value string) *Logger {
107	l.l = slog.New(l.l.Handler().WithAttrs([]slog.Attr{{Key: key, Value: slog.StringValue(value)}}))
108	return l
109}
110
111func (l *Logger) Debug(msg string, attr ...LoggerPair) *Logger {
112	attrs := make([]slog.Attr, len(attr))
113	for i, a := range attr {
114		attrs[i] = slog.Attr{Key: a.key, Value: slog.AnyValue(a.value)}
115	}
116	if caller, ok := extractCaller(l.c); ok {
117		attrs = append(attrs, slog.Attr{Key: "caller", Value: slog.StringValue(caller)})
118	}
119	l.l.LogAttrs(l.c, slog.LevelDebug, msg, attrs...)
120	return l
121}
122
123func (l *Logger) Info(msg string, attr ...LoggerPair) *Logger {
124	attrs := make([]slog.Attr, len(attr))
125	for i, a := range attr {
126		attrs[i] = slog.Attr{Key: a.key, Value: slog.AnyValue(a.value)}
127	}
128	if caller, ok := extractCaller(l.c); ok {
129		attrs = append(attrs, slog.Attr{Key: "caller", Value: slog.StringValue(caller)})
130	}
131	l.l.LogAttrs(l.c, slog.LevelInfo, msg, attrs...)
132	return l
133}
134
135func (l *Logger) Warn(msg string, attr ...LoggerPair) *Logger {
136	attrs := make([]slog.Attr, len(attr))
137	for i, a := range attr {
138		attrs[i] = slog.Attr{Key: a.key, Value: slog.AnyValue(a.value)}
139	}
140	l.l.LogAttrs(l.c, slog.LevelWarn, msg, attrs...)
141	return l
142}
143
144func (l *Logger) Error(msg string, err error, attr ...LoggerPair) *Logger {
145	attrs := make([]slog.Attr, len(attr)+1)
146	attrs[0] = slog.Attr{Key: "err", Value: slog.AnyValue(err)}
147	for i, a := range attr {
148		attrs[i+1] = slog.Attr{Key: a.key, Value: slog.AnyValue(a.value)}
149	}
150	l.l.LogAttrs(l.c, slog.LevelError, msg, attrs...)
151	return l
152}
153
154func (l *Logger) LogCaller(ctx context.Context, msg string) *Logger {
155	attr := slog.Attr{Key: "caller", Value: slog.StringValue("NO")}
156	if caller, ok := extractCaller(ctx); ok {
157		attr = slog.Attr{Key: "caller", Value: slog.StringValue(caller)}
158	}
159	l.l.LogAttrs(l.c, slog.LevelInfo, msg, attr)
160	return l
161}