mirror of
https://github.com/mjl-/mox.git
synced 2025-01-28 07:15:55 +03:00
5336032088
so users can easily take their email out of somewhere else, and import it into mox. this goes a little way to give feedback as the import progresses: upload progress is shown (surprisingly, browsers aren't doing this...), imported mailboxes/messages are counted (batched) and import issues/warnings are displayed, all sent over an SSE connection. an import token is stored in sessionstorage. if you reload the page (e.g. after a connection error), the browser will reconnect to the running import and show its progress again. and you can just abort the import before it is finished and committed, and nothing will have changed. this also imports flags/keywords from mbox files.
410 lines
10 KiB
Go
410 lines
10 KiB
Go
// Package mlog provides logging with log levels and fields.
|
|
//
|
|
// Each log level has a function to log with and without error.
|
|
// Each such function takes a varargs list of fields (key value pairs) to log.
|
|
// Variable data should be in fields. Logging strings themselves should be
|
|
// constant, for easier log processing (e.g. building metrics based on log
|
|
// messages).
|
|
//
|
|
// The log levels can be configured per originating package, e.g. smtpclient,
|
|
// imapserver. The configuration is application-global, so each Log instance
|
|
// uses the same log levels.
|
|
//
|
|
// Print* should be used for lines that always should be printed, regardless of
|
|
// configured log levels. Useful for startup logging and subcommands.
|
|
//
|
|
// Fatal* stops the program. Its log text is always printed.
|
|
package mlog
|
|
|
|
// todo: log with source=path:linenumber? and/or stacktrace (perhaps optional)
|
|
// todo: should we turn errors logged with an context.Canceled from a level error into level info?
|
|
// todo: rethink format. perhaps simply using %#v is more useful for many types?
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"encoding/base64"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"os"
|
|
"reflect"
|
|
"strconv"
|
|
"strings"
|
|
"sync/atomic"
|
|
)
|
|
|
|
var Logfmt bool
|
|
|
|
type Level int
|
|
|
|
func (l Level) String() string {
|
|
return LevelStrings[l]
|
|
}
|
|
|
|
var LevelStrings = map[Level]string{
|
|
LevelPrint: "print",
|
|
LevelFatal: "fatal",
|
|
LevelError: "error",
|
|
LevelInfo: "info",
|
|
LevelDebug: "debug",
|
|
LevelTrace: "trace",
|
|
LevelTraceauth: "traceauth",
|
|
LevelTracedata: "tracedata",
|
|
}
|
|
|
|
var Levels = map[string]Level{
|
|
"print": LevelPrint,
|
|
"fatal": LevelFatal,
|
|
"error": LevelError,
|
|
"info": LevelInfo,
|
|
"debug": LevelDebug,
|
|
"trace": LevelTrace,
|
|
"traceauth": LevelTraceauth,
|
|
"tracedata": LevelTracedata,
|
|
}
|
|
|
|
const (
|
|
LevelPrint Level = 0 // Printed regardless of configured log level.
|
|
LevelFatal Level = 1 // Printed regardless of configured log level.
|
|
LevelError Level = 2
|
|
LevelInfo Level = 3
|
|
LevelDebug Level = 4
|
|
LevelTrace Level = 5
|
|
LevelTraceauth Level = 6
|
|
LevelTracedata Level = 7
|
|
)
|
|
|
|
// Holds a map[string]Level, mapping a package (field pkg in logs) to a log level.
|
|
// The empty string is the default/fallback log level.
|
|
var config atomic.Value
|
|
|
|
func init() {
|
|
config.Store(map[string]Level{"": LevelError})
|
|
}
|
|
|
|
// SetConfig atomically sets the new log levels used by all Log instances.
|
|
func SetConfig(c map[string]Level) {
|
|
config.Store(c)
|
|
}
|
|
|
|
// Pair is a field/value pair, for use in logged lines.
|
|
type Pair struct {
|
|
key string
|
|
value any
|
|
}
|
|
|
|
// Field is a shorthand for making a Pair.
|
|
func Field(k string, v any) Pair {
|
|
return Pair{k, v}
|
|
}
|
|
|
|
// Log is an instance potentially with its own field/value pair added to any
|
|
// logging output.
|
|
type Log struct {
|
|
fields []Pair
|
|
moreFields func() []Pair
|
|
}
|
|
|
|
// New returns a new Log instance. Each log invocation adds field "pkg".
|
|
func New(pkg string) *Log {
|
|
return &Log{
|
|
fields: []Pair{{"pkg", pkg}},
|
|
}
|
|
}
|
|
|
|
type key string
|
|
|
|
// CidKey can be used with context.WithValue to store a "cid" in a context, for logging.
|
|
var CidKey key = "cid"
|
|
|
|
// WithCid adds a field "cid".
|
|
// Also see WithContext.
|
|
func (l *Log) WithCid(cid int64) *Log {
|
|
return l.Fields(Pair{"cid", cid})
|
|
}
|
|
|
|
// WithContext adds cid from context, if present. Context are often passed to
|
|
// functions, especially between packages, to pass a "cid" for an operation. At the
|
|
// start of a function (especially if exported) a variable "log" is often
|
|
// instantiated from a package-level variable "xlog", with WithContext for its cid.
|
|
// A *Log could be passed instead, but contexts are more pervasive. For the same
|
|
// reason WithContext is more common than WithCid.
|
|
func (l *Log) WithContext(ctx context.Context) *Log {
|
|
cidv := ctx.Value(CidKey)
|
|
if cidv == nil {
|
|
return l
|
|
}
|
|
cid := cidv.(int64)
|
|
return l.WithCid(cid)
|
|
}
|
|
|
|
// Field adds fields to the logger. Each logged line adds these fields.
|
|
func (l *Log) Fields(fields ...Pair) *Log {
|
|
nl := *l
|
|
nl.fields = append(fields, nl.fields...)
|
|
return &nl
|
|
}
|
|
|
|
// MoreFields sets a function on the logger that is called just before logging,
|
|
// to retrieve additional fields to log.
|
|
func (l *Log) MoreFields(fn func() []Pair) *Log {
|
|
nl := *l
|
|
nl.moreFields = fn
|
|
return &nl
|
|
}
|
|
|
|
// Check logs an error if err is not nil. Intended for logging errors that are good
|
|
// to know, but would not influence program flow.
|
|
func (l *Log) Check(err error, text string, fields ...Pair) {
|
|
if err != nil {
|
|
l.Errorx(text, err, fields...)
|
|
}
|
|
}
|
|
|
|
func (l *Log) Trace(traceLevel Level, text string) bool {
|
|
return l.logx(traceLevel, nil, text)
|
|
}
|
|
|
|
func (l *Log) Fatal(text string, fields ...Pair) { l.Fatalx(text, nil, fields...) }
|
|
func (l *Log) Fatalx(text string, err error, fields ...Pair) {
|
|
l.plog(LevelFatal, err, text, fields...)
|
|
os.Exit(1)
|
|
}
|
|
|
|
func (l *Log) Print(text string, fields ...Pair) bool {
|
|
return l.logx(LevelPrint, nil, text, fields...)
|
|
}
|
|
func (l *Log) Printx(text string, err error, fields ...Pair) bool {
|
|
return l.logx(LevelPrint, err, text, fields...)
|
|
}
|
|
|
|
func (l *Log) Debug(text string, fields ...Pair) bool {
|
|
return l.logx(LevelDebug, nil, text, fields...)
|
|
}
|
|
func (l *Log) Debugx(text string, err error, fields ...Pair) bool {
|
|
return l.logx(LevelDebug, err, text, fields...)
|
|
}
|
|
|
|
func (l *Log) Info(text string, fields ...Pair) bool { return l.logx(LevelInfo, nil, text, fields...) }
|
|
func (l *Log) Infox(text string, err error, fields ...Pair) bool {
|
|
return l.logx(LevelInfo, err, text, fields...)
|
|
}
|
|
|
|
func (l *Log) Error(text string, fields ...Pair) bool {
|
|
return l.logx(LevelError, nil, text, fields...)
|
|
}
|
|
func (l *Log) Errorx(text string, err error, fields ...Pair) bool {
|
|
return l.logx(LevelError, err, text, fields...)
|
|
}
|
|
|
|
func (l *Log) logx(level Level, err error, text string, fields ...Pair) bool {
|
|
if ok, high := l.match(level); ok {
|
|
// Nothing.
|
|
} else if high >= LevelTrace && level == LevelTraceauth {
|
|
text = "***"
|
|
} else if high >= LevelTrace && level == LevelTracedata {
|
|
text = "..."
|
|
} else {
|
|
return false
|
|
}
|
|
if level > LevelTrace {
|
|
level = LevelTrace
|
|
}
|
|
l.plog(level, err, text, fields...)
|
|
return true
|
|
}
|
|
|
|
// escape logfmt string if required, otherwise return original string.
|
|
func logfmtValue(s string) string {
|
|
for _, c := range s {
|
|
if c == '"' || c == '\\' || c <= ' ' || c == '=' || c >= 0x7f {
|
|
return fmt.Sprintf("%q", s)
|
|
}
|
|
}
|
|
return s
|
|
}
|
|
|
|
func stringValue(iscid, nested bool, v any) string {
|
|
// Handle some common types first.
|
|
if v == nil {
|
|
return ""
|
|
}
|
|
switch r := v.(type) {
|
|
case string:
|
|
return r
|
|
case int:
|
|
return strconv.Itoa(r)
|
|
case int64:
|
|
if iscid {
|
|
return fmt.Sprintf("%x", v)
|
|
}
|
|
return strconv.FormatInt(r, 10)
|
|
case bool:
|
|
if r {
|
|
return "true"
|
|
}
|
|
return "false"
|
|
case float64:
|
|
return fmt.Sprintf("%v", v)
|
|
case []byte:
|
|
return base64.RawURLEncoding.EncodeToString(r)
|
|
case []string:
|
|
if nested && len(r) == 0 {
|
|
// Drop field from logging.
|
|
return ""
|
|
}
|
|
return "[" + strings.Join(r, ",") + "]"
|
|
}
|
|
|
|
rv := reflect.ValueOf(v)
|
|
if rv.Kind() == reflect.Ptr && rv.IsNil() {
|
|
return ""
|
|
}
|
|
|
|
if r, ok := v.(fmt.Stringer); ok {
|
|
return r.String()
|
|
}
|
|
|
|
if rv.Kind() == reflect.Ptr {
|
|
rv = rv.Elem()
|
|
return stringValue(iscid, nested, rv.Interface())
|
|
}
|
|
if rv.Kind() == reflect.Slice {
|
|
n := rv.Len()
|
|
if nested && n == 0 {
|
|
// Drop field.
|
|
return ""
|
|
}
|
|
b := &strings.Builder{}
|
|
b.WriteString("[")
|
|
for i := 0; i < n; i++ {
|
|
if i > 0 {
|
|
b.WriteString(";")
|
|
}
|
|
b.WriteString(stringValue(false, true, rv.Index(i).Interface()))
|
|
}
|
|
b.WriteString("]")
|
|
return b.String()
|
|
} else if rv.Kind() != reflect.Struct {
|
|
return fmt.Sprintf("%v", v)
|
|
}
|
|
n := rv.NumField()
|
|
t := rv.Type()
|
|
b := &strings.Builder{}
|
|
first := true
|
|
for i := 0; i < n; i++ {
|
|
fv := rv.Field(i)
|
|
if !t.Field(i).IsExported() {
|
|
continue
|
|
}
|
|
if fv.Kind() == reflect.Struct || fv.Kind() == reflect.Ptr || fv.Kind() == reflect.Interface {
|
|
// Don't recurse.
|
|
continue
|
|
}
|
|
vs := stringValue(false, true, fv.Interface())
|
|
if vs == "" {
|
|
continue
|
|
}
|
|
if !first {
|
|
b.WriteByte(' ')
|
|
}
|
|
first = false
|
|
k := strings.ToLower(t.Field(i).Name)
|
|
b.WriteString(k + "=" + logfmtValue(vs))
|
|
}
|
|
return b.String()
|
|
}
|
|
|
|
func (l *Log) plog(level Level, err error, text string, fields ...Pair) {
|
|
fields = append(l.fields, fields...)
|
|
if l.moreFields != nil {
|
|
fields = append(fields, l.moreFields()...)
|
|
}
|
|
// We build up a buffer so we can do a single atomic write of the data. Otherwise partial log lines may interleaf.
|
|
b := &bytes.Buffer{}
|
|
if Logfmt {
|
|
fmt.Fprintf(b, "l=%s m=%s", LevelStrings[level], logfmtValue(text))
|
|
if err != nil {
|
|
fmt.Fprintf(b, " err=%s", logfmtValue(err.Error()))
|
|
}
|
|
for i := 0; i < len(fields); i++ {
|
|
kv := fields[i]
|
|
fmt.Fprintf(b, " %s=%s", kv.key, logfmtValue(stringValue(kv.key == "cid", false, kv.value)))
|
|
}
|
|
b.WriteString("\n")
|
|
} else {
|
|
fmt.Fprintf(b, "%s: %s", LevelStrings[level], logfmtValue(text))
|
|
if err != nil {
|
|
fmt.Fprintf(b, ": %s", logfmtValue(err.Error()))
|
|
}
|
|
if len(fields) > 0 {
|
|
fmt.Fprint(b, " (")
|
|
for i := 0; i < len(fields); i++ {
|
|
if i > 0 {
|
|
fmt.Fprint(b, "; ")
|
|
}
|
|
kv := fields[i]
|
|
fmt.Fprintf(b, "%s: %s", kv.key, logfmtValue(stringValue(kv.key == "cid", false, kv.value)))
|
|
}
|
|
fmt.Fprint(b, ")")
|
|
}
|
|
b.WriteString("\n")
|
|
}
|
|
os.Stderr.Write(b.Bytes())
|
|
}
|
|
|
|
func (l *Log) match(level Level) (bool, Level) {
|
|
if level == LevelPrint || level == LevelFatal {
|
|
return true, level
|
|
}
|
|
|
|
cl := config.Load().(map[string]Level)
|
|
|
|
seen := false
|
|
var high Level
|
|
for _, kv := range l.fields {
|
|
if kv.key != "pkg" {
|
|
continue
|
|
}
|
|
pkg, ok := kv.value.(string)
|
|
if !ok {
|
|
continue
|
|
}
|
|
v, ok := cl[pkg]
|
|
if v > high {
|
|
high = v
|
|
}
|
|
if ok && v >= level {
|
|
return true, high
|
|
}
|
|
seen = seen || ok
|
|
}
|
|
if seen {
|
|
return false, high
|
|
}
|
|
v, ok := cl[""]
|
|
if v > high {
|
|
high = v
|
|
}
|
|
return ok && v >= level, v
|
|
}
|
|
|
|
type errWriter struct {
|
|
log *Log
|
|
level Level
|
|
msg string
|
|
}
|
|
|
|
func (w *errWriter) Write(buf []byte) (int, error) {
|
|
err := errors.New(strings.TrimSpace(string(buf)))
|
|
w.log.logx(w.level, err, w.msg)
|
|
return len(buf), nil
|
|
}
|
|
|
|
// ErrWriter returns a writer that turns each write into a logging call on "log"
|
|
// with given "level" and "msg" and the written content as an error.
|
|
// Can be used for making a Go log.Logger for use in http.Server.ErrorLog.
|
|
func ErrWriter(log *Log, level Level, msg string) io.Writer {
|
|
return &errWriter{log, level, msg}
|
|
}
|