2019-11-25 14:33:58 -08:00
|
|
|
package log
|
2019-06-20 16:36:40 -07:00
|
|
|
|
|
|
|
import (
|
|
|
|
"os"
|
2021-09-18 00:00:59 +00:00
|
|
|
"runtime"
|
|
|
|
"strconv"
|
2021-05-25 11:38:21 +03:00
|
|
|
"strings"
|
2022-09-26 18:18:28 +03:00
|
|
|
"sync"
|
2019-06-20 16:36:40 -07:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/rs/zerolog"
|
|
|
|
)
|
|
|
|
|
2021-12-13 19:23:31 +00:00
|
|
|
const defaultPerms = 0o0600
|
|
|
|
|
2022-10-05 13:21:14 +03:00
|
|
|
//nolint:gochecknoglobals
|
2022-09-26 18:18:28 +03:00
|
|
|
var loggerSetTimeFormat sync.Once
|
|
|
|
|
2020-05-11 15:13:24 -07:00
|
|
|
// Logger extends zerolog's Logger.
|
2019-11-25 14:33:58 -08:00
|
|
|
type Logger struct {
|
|
|
|
zerolog.Logger
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l Logger) Println(v ...interface{}) {
|
2023-12-08 00:05:02 -08:00
|
|
|
l.Logger.Error().Msg("panic recovered") //nolint: check-logs
|
2019-11-25 14:33:58 -08:00
|
|
|
}
|
|
|
|
|
2022-03-21 17:37:23 +00:00
|
|
|
func NewLogger(level, output string) Logger {
|
2022-09-26 18:18:28 +03:00
|
|
|
loggerSetTimeFormat.Do(func() {
|
|
|
|
zerolog.TimeFieldFormat = time.RFC3339Nano
|
|
|
|
})
|
2019-12-13 00:53:18 -05:00
|
|
|
|
2021-12-13 19:23:31 +00:00
|
|
|
lvl, err := zerolog.ParseLevel(level)
|
2019-06-20 16:36:40 -07:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2019-12-13 00:53:18 -05:00
|
|
|
|
2019-06-20 16:36:40 -07:00
|
|
|
zerolog.SetGlobalLevel(lvl)
|
2019-12-13 00:53:18 -05:00
|
|
|
|
2019-06-20 16:36:40 -07:00
|
|
|
var log zerolog.Logger
|
2019-12-13 00:53:18 -05:00
|
|
|
|
2019-11-25 14:33:58 -08:00
|
|
|
if output == "" {
|
2019-06-20 16:36:40 -07:00
|
|
|
log = zerolog.New(os.Stdout)
|
|
|
|
} else {
|
2021-12-13 19:23:31 +00:00
|
|
|
file, err := os.OpenFile(output, os.O_APPEND|os.O_WRONLY|os.O_CREATE, defaultPerms)
|
2019-06-20 16:36:40 -07:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2024-07-29 19:32:51 +02:00
|
|
|
|
2019-06-20 16:36:40 -07:00
|
|
|
log = zerolog.New(file)
|
|
|
|
}
|
2019-12-13 00:53:18 -05:00
|
|
|
|
2021-09-18 00:00:59 +00:00
|
|
|
return Logger{Logger: log.Hook(goroutineHook{}).With().Caller().Timestamp().Logger()}
|
2019-06-20 16:36:40 -07:00
|
|
|
}
|
|
|
|
|
2023-11-01 18:16:18 +02:00
|
|
|
func NewAuditLogger(level, output string) *Logger {
|
2022-09-26 18:18:28 +03:00
|
|
|
loggerSetTimeFormat.Do(func() {
|
|
|
|
zerolog.TimeFieldFormat = time.RFC3339Nano
|
|
|
|
})
|
2021-05-25 11:38:21 +03:00
|
|
|
|
2021-12-13 19:23:31 +00:00
|
|
|
lvl, err := zerolog.ParseLevel(level)
|
2021-05-25 11:38:21 +03:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
zerolog.SetGlobalLevel(lvl)
|
|
|
|
|
|
|
|
var auditLog zerolog.Logger
|
|
|
|
|
2023-11-01 18:16:18 +02:00
|
|
|
if output == "" {
|
|
|
|
auditLog = zerolog.New(os.Stdout)
|
|
|
|
} else {
|
|
|
|
auditFile, err := os.OpenFile(output, os.O_APPEND|os.O_WRONLY|os.O_CREATE, defaultPerms)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2021-05-25 11:38:21 +03:00
|
|
|
|
2023-11-01 18:16:18 +02:00
|
|
|
auditLog = zerolog.New(auditFile)
|
|
|
|
}
|
2021-05-25 11:38:21 +03:00
|
|
|
|
|
|
|
return &Logger{Logger: auditLog.With().Timestamp().Logger()}
|
|
|
|
}
|
|
|
|
|
2022-01-21 04:11:44 +00:00
|
|
|
// GoroutineID adds goroutine-id to logs to help debug concurrency issues.
|
|
|
|
func GoroutineID() int {
|
2021-09-18 00:00:59 +00:00
|
|
|
var buf [64]byte
|
|
|
|
n := runtime.Stack(buf[:], false)
|
|
|
|
idField := strings.Fields(strings.TrimPrefix(string(buf[:n]), "goroutine "))[0]
|
|
|
|
|
|
|
|
id, err := strconv.Atoi(idField)
|
|
|
|
if err != nil {
|
|
|
|
return -1
|
|
|
|
}
|
|
|
|
|
|
|
|
return id
|
|
|
|
}
|
|
|
|
|
|
|
|
type goroutineHook struct{}
|
|
|
|
|
2023-05-10 20:15:33 +03:00
|
|
|
func (h goroutineHook) Run(e *zerolog.Event, level zerolog.Level, _ string) {
|
2021-09-18 00:00:59 +00:00
|
|
|
if level != zerolog.NoLevel {
|
2022-01-21 04:11:44 +00:00
|
|
|
e.Int("goroutine", GoroutineID())
|
2021-09-18 00:00:59 +00:00
|
|
|
}
|
|
|
|
}
|