mirror of
https://github.com/mjl-/mox.git
synced 2025-01-17 02:45:37 +03:00
5b20cba50a
we don't want external software to include internal details like mlog. slog.Logger is/will be the standard. we still have mlog for its helper functions, and its handler that logs in concise logfmt used by mox. packages that are not meant for reuse still pass around mlog.Log for convenience. we use golang.org/x/exp/slog because we also support the previous Go toolchain version. with the next Go release, we'll switch to the builtin slog.
456 lines
11 KiB
Go
456 lines
11 KiB
Go
// Copyright 2022 The Go Authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style
|
|
// license that can be found in the LICENSE file.
|
|
|
|
package slog
|
|
|
|
import (
|
|
"fmt"
|
|
"math"
|
|
"runtime"
|
|
"strconv"
|
|
"strings"
|
|
"time"
|
|
"unsafe"
|
|
|
|
"golang.org/x/exp/slices"
|
|
)
|
|
|
|
// A Value can represent any Go value, but unlike type any,
|
|
// it can represent most small values without an allocation.
|
|
// The zero Value corresponds to nil.
|
|
type Value struct {
|
|
_ [0]func() // disallow ==
|
|
// num holds the value for Kinds Int64, Uint64, Float64, Bool and Duration,
|
|
// the string length for KindString, and nanoseconds since the epoch for KindTime.
|
|
num uint64
|
|
// If any is of type Kind, then the value is in num as described above.
|
|
// If any is of type *time.Location, then the Kind is Time and time.Time value
|
|
// can be constructed from the Unix nanos in num and the location (monotonic time
|
|
// is not preserved).
|
|
// If any is of type stringptr, then the Kind is String and the string value
|
|
// consists of the length in num and the pointer in any.
|
|
// Otherwise, the Kind is Any and any is the value.
|
|
// (This implies that Attrs cannot store values of type Kind, *time.Location
|
|
// or stringptr.)
|
|
any any
|
|
}
|
|
|
|
// Kind is the kind of a Value.
|
|
type Kind int
|
|
|
|
// The following list is sorted alphabetically, but it's also important that
|
|
// KindAny is 0 so that a zero Value represents nil.
|
|
|
|
const (
|
|
KindAny Kind = iota
|
|
KindBool
|
|
KindDuration
|
|
KindFloat64
|
|
KindInt64
|
|
KindString
|
|
KindTime
|
|
KindUint64
|
|
KindGroup
|
|
KindLogValuer
|
|
)
|
|
|
|
var kindStrings = []string{
|
|
"Any",
|
|
"Bool",
|
|
"Duration",
|
|
"Float64",
|
|
"Int64",
|
|
"String",
|
|
"Time",
|
|
"Uint64",
|
|
"Group",
|
|
"LogValuer",
|
|
}
|
|
|
|
func (k Kind) String() string {
|
|
if k >= 0 && int(k) < len(kindStrings) {
|
|
return kindStrings[k]
|
|
}
|
|
return "<unknown slog.Kind>"
|
|
}
|
|
|
|
// Unexported version of Kind, just so we can store Kinds in Values.
|
|
// (No user-provided value has this type.)
|
|
type kind Kind
|
|
|
|
// Kind returns v's Kind.
|
|
func (v Value) Kind() Kind {
|
|
switch x := v.any.(type) {
|
|
case Kind:
|
|
return x
|
|
case stringptr:
|
|
return KindString
|
|
case timeLocation:
|
|
return KindTime
|
|
case groupptr:
|
|
return KindGroup
|
|
case LogValuer:
|
|
return KindLogValuer
|
|
case kind: // a kind is just a wrapper for a Kind
|
|
return KindAny
|
|
default:
|
|
return KindAny
|
|
}
|
|
}
|
|
|
|
//////////////// Constructors
|
|
|
|
// IntValue returns a Value for an int.
|
|
func IntValue(v int) Value {
|
|
return Int64Value(int64(v))
|
|
}
|
|
|
|
// Int64Value returns a Value for an int64.
|
|
func Int64Value(v int64) Value {
|
|
return Value{num: uint64(v), any: KindInt64}
|
|
}
|
|
|
|
// Uint64Value returns a Value for a uint64.
|
|
func Uint64Value(v uint64) Value {
|
|
return Value{num: v, any: KindUint64}
|
|
}
|
|
|
|
// Float64Value returns a Value for a floating-point number.
|
|
func Float64Value(v float64) Value {
|
|
return Value{num: math.Float64bits(v), any: KindFloat64}
|
|
}
|
|
|
|
// BoolValue returns a Value for a bool.
|
|
func BoolValue(v bool) Value {
|
|
u := uint64(0)
|
|
if v {
|
|
u = 1
|
|
}
|
|
return Value{num: u, any: KindBool}
|
|
}
|
|
|
|
// Unexported version of *time.Location, just so we can store *time.Locations in
|
|
// Values. (No user-provided value has this type.)
|
|
type timeLocation *time.Location
|
|
|
|
// TimeValue returns a Value for a time.Time.
|
|
// It discards the monotonic portion.
|
|
func TimeValue(v time.Time) Value {
|
|
if v.IsZero() {
|
|
// UnixNano on the zero time is undefined, so represent the zero time
|
|
// with a nil *time.Location instead. time.Time.Location method never
|
|
// returns nil, so a Value with any == timeLocation(nil) cannot be
|
|
// mistaken for any other Value, time.Time or otherwise.
|
|
return Value{any: timeLocation(nil)}
|
|
}
|
|
return Value{num: uint64(v.UnixNano()), any: timeLocation(v.Location())}
|
|
}
|
|
|
|
// DurationValue returns a Value for a time.Duration.
|
|
func DurationValue(v time.Duration) Value {
|
|
return Value{num: uint64(v.Nanoseconds()), any: KindDuration}
|
|
}
|
|
|
|
// AnyValue returns a Value for the supplied value.
|
|
//
|
|
// If the supplied value is of type Value, it is returned
|
|
// unmodified.
|
|
//
|
|
// Given a value of one of Go's predeclared string, bool, or
|
|
// (non-complex) numeric types, AnyValue returns a Value of kind
|
|
// String, Bool, Uint64, Int64, or Float64. The width of the
|
|
// original numeric type is not preserved.
|
|
//
|
|
// Given a time.Time or time.Duration value, AnyValue returns a Value of kind
|
|
// KindTime or KindDuration. The monotonic time is not preserved.
|
|
//
|
|
// For nil, or values of all other types, including named types whose
|
|
// underlying type is numeric, AnyValue returns a value of kind KindAny.
|
|
func AnyValue(v any) Value {
|
|
switch v := v.(type) {
|
|
case string:
|
|
return StringValue(v)
|
|
case int:
|
|
return Int64Value(int64(v))
|
|
case uint:
|
|
return Uint64Value(uint64(v))
|
|
case int64:
|
|
return Int64Value(v)
|
|
case uint64:
|
|
return Uint64Value(v)
|
|
case bool:
|
|
return BoolValue(v)
|
|
case time.Duration:
|
|
return DurationValue(v)
|
|
case time.Time:
|
|
return TimeValue(v)
|
|
case uint8:
|
|
return Uint64Value(uint64(v))
|
|
case uint16:
|
|
return Uint64Value(uint64(v))
|
|
case uint32:
|
|
return Uint64Value(uint64(v))
|
|
case uintptr:
|
|
return Uint64Value(uint64(v))
|
|
case int8:
|
|
return Int64Value(int64(v))
|
|
case int16:
|
|
return Int64Value(int64(v))
|
|
case int32:
|
|
return Int64Value(int64(v))
|
|
case float64:
|
|
return Float64Value(v)
|
|
case float32:
|
|
return Float64Value(float64(v))
|
|
case []Attr:
|
|
return GroupValue(v...)
|
|
case Kind:
|
|
return Value{any: kind(v)}
|
|
case Value:
|
|
return v
|
|
default:
|
|
return Value{any: v}
|
|
}
|
|
}
|
|
|
|
//////////////// Accessors
|
|
|
|
// Any returns v's value as an any.
|
|
func (v Value) Any() any {
|
|
switch v.Kind() {
|
|
case KindAny:
|
|
if k, ok := v.any.(kind); ok {
|
|
return Kind(k)
|
|
}
|
|
return v.any
|
|
case KindLogValuer:
|
|
return v.any
|
|
case KindGroup:
|
|
return v.group()
|
|
case KindInt64:
|
|
return int64(v.num)
|
|
case KindUint64:
|
|
return v.num
|
|
case KindFloat64:
|
|
return v.float()
|
|
case KindString:
|
|
return v.str()
|
|
case KindBool:
|
|
return v.bool()
|
|
case KindDuration:
|
|
return v.duration()
|
|
case KindTime:
|
|
return v.time()
|
|
default:
|
|
panic(fmt.Sprintf("bad kind: %s", v.Kind()))
|
|
}
|
|
}
|
|
|
|
// Int64 returns v's value as an int64. It panics
|
|
// if v is not a signed integer.
|
|
func (v Value) Int64() int64 {
|
|
if g, w := v.Kind(), KindInt64; g != w {
|
|
panic(fmt.Sprintf("Value kind is %s, not %s", g, w))
|
|
}
|
|
return int64(v.num)
|
|
}
|
|
|
|
// Uint64 returns v's value as a uint64. It panics
|
|
// if v is not an unsigned integer.
|
|
func (v Value) Uint64() uint64 {
|
|
if g, w := v.Kind(), KindUint64; g != w {
|
|
panic(fmt.Sprintf("Value kind is %s, not %s", g, w))
|
|
}
|
|
return v.num
|
|
}
|
|
|
|
// Bool returns v's value as a bool. It panics
|
|
// if v is not a bool.
|
|
func (v Value) Bool() bool {
|
|
if g, w := v.Kind(), KindBool; g != w {
|
|
panic(fmt.Sprintf("Value kind is %s, not %s", g, w))
|
|
}
|
|
return v.bool()
|
|
}
|
|
|
|
func (v Value) bool() bool {
|
|
return v.num == 1
|
|
}
|
|
|
|
// Duration returns v's value as a time.Duration. It panics
|
|
// if v is not a time.Duration.
|
|
func (v Value) Duration() time.Duration {
|
|
if g, w := v.Kind(), KindDuration; g != w {
|
|
panic(fmt.Sprintf("Value kind is %s, not %s", g, w))
|
|
}
|
|
|
|
return v.duration()
|
|
}
|
|
|
|
func (v Value) duration() time.Duration {
|
|
return time.Duration(int64(v.num))
|
|
}
|
|
|
|
// Float64 returns v's value as a float64. It panics
|
|
// if v is not a float64.
|
|
func (v Value) Float64() float64 {
|
|
if g, w := v.Kind(), KindFloat64; g != w {
|
|
panic(fmt.Sprintf("Value kind is %s, not %s", g, w))
|
|
}
|
|
|
|
return v.float()
|
|
}
|
|
|
|
func (v Value) float() float64 {
|
|
return math.Float64frombits(v.num)
|
|
}
|
|
|
|
// Time returns v's value as a time.Time. It panics
|
|
// if v is not a time.Time.
|
|
func (v Value) Time() time.Time {
|
|
if g, w := v.Kind(), KindTime; g != w {
|
|
panic(fmt.Sprintf("Value kind is %s, not %s", g, w))
|
|
}
|
|
return v.time()
|
|
}
|
|
|
|
func (v Value) time() time.Time {
|
|
loc := v.any.(timeLocation)
|
|
if loc == nil {
|
|
return time.Time{}
|
|
}
|
|
return time.Unix(0, int64(v.num)).In(loc)
|
|
}
|
|
|
|
// LogValuer returns v's value as a LogValuer. It panics
|
|
// if v is not a LogValuer.
|
|
func (v Value) LogValuer() LogValuer {
|
|
return v.any.(LogValuer)
|
|
}
|
|
|
|
// Group returns v's value as a []Attr.
|
|
// It panics if v's Kind is not KindGroup.
|
|
func (v Value) Group() []Attr {
|
|
if sp, ok := v.any.(groupptr); ok {
|
|
return unsafe.Slice((*Attr)(sp), v.num)
|
|
}
|
|
panic("Group: bad kind")
|
|
}
|
|
|
|
func (v Value) group() []Attr {
|
|
return unsafe.Slice((*Attr)(v.any.(groupptr)), v.num)
|
|
}
|
|
|
|
//////////////// Other
|
|
|
|
// Equal reports whether v and w represent the same Go value.
|
|
func (v Value) Equal(w Value) bool {
|
|
k1 := v.Kind()
|
|
k2 := w.Kind()
|
|
if k1 != k2 {
|
|
return false
|
|
}
|
|
switch k1 {
|
|
case KindInt64, KindUint64, KindBool, KindDuration:
|
|
return v.num == w.num
|
|
case KindString:
|
|
return v.str() == w.str()
|
|
case KindFloat64:
|
|
return v.float() == w.float()
|
|
case KindTime:
|
|
return v.time().Equal(w.time())
|
|
case KindAny, KindLogValuer:
|
|
return v.any == w.any // may panic if non-comparable
|
|
case KindGroup:
|
|
return slices.EqualFunc(v.group(), w.group(), Attr.Equal)
|
|
default:
|
|
panic(fmt.Sprintf("bad kind: %s", k1))
|
|
}
|
|
}
|
|
|
|
// append appends a text representation of v to dst.
|
|
// v is formatted as with fmt.Sprint.
|
|
func (v Value) append(dst []byte) []byte {
|
|
switch v.Kind() {
|
|
case KindString:
|
|
return append(dst, v.str()...)
|
|
case KindInt64:
|
|
return strconv.AppendInt(dst, int64(v.num), 10)
|
|
case KindUint64:
|
|
return strconv.AppendUint(dst, v.num, 10)
|
|
case KindFloat64:
|
|
return strconv.AppendFloat(dst, v.float(), 'g', -1, 64)
|
|
case KindBool:
|
|
return strconv.AppendBool(dst, v.bool())
|
|
case KindDuration:
|
|
return append(dst, v.duration().String()...)
|
|
case KindTime:
|
|
return append(dst, v.time().String()...)
|
|
case KindGroup:
|
|
return fmt.Append(dst, v.group())
|
|
case KindAny, KindLogValuer:
|
|
return fmt.Append(dst, v.any)
|
|
default:
|
|
panic(fmt.Sprintf("bad kind: %s", v.Kind()))
|
|
}
|
|
}
|
|
|
|
// A LogValuer is any Go value that can convert itself into a Value for logging.
|
|
//
|
|
// This mechanism may be used to defer expensive operations until they are
|
|
// needed, or to expand a single value into a sequence of components.
|
|
type LogValuer interface {
|
|
LogValue() Value
|
|
}
|
|
|
|
const maxLogValues = 100
|
|
|
|
// Resolve repeatedly calls LogValue on v while it implements LogValuer,
|
|
// and returns the result.
|
|
// If v resolves to a group, the group's attributes' values are not recursively
|
|
// resolved.
|
|
// If the number of LogValue calls exceeds a threshold, a Value containing an
|
|
// error is returned.
|
|
// Resolve's return value is guaranteed not to be of Kind KindLogValuer.
|
|
func (v Value) Resolve() (rv Value) {
|
|
orig := v
|
|
defer func() {
|
|
if r := recover(); r != nil {
|
|
rv = AnyValue(fmt.Errorf("LogValue panicked\n%s", stack(3, 5)))
|
|
}
|
|
}()
|
|
|
|
for i := 0; i < maxLogValues; i++ {
|
|
if v.Kind() != KindLogValuer {
|
|
return v
|
|
}
|
|
v = v.LogValuer().LogValue()
|
|
}
|
|
err := fmt.Errorf("LogValue called too many times on Value of type %T", orig.Any())
|
|
return AnyValue(err)
|
|
}
|
|
|
|
func stack(skip, nFrames int) string {
|
|
pcs := make([]uintptr, nFrames+1)
|
|
n := runtime.Callers(skip+1, pcs)
|
|
if n == 0 {
|
|
return "(no stack)"
|
|
}
|
|
frames := runtime.CallersFrames(pcs[:n])
|
|
var b strings.Builder
|
|
i := 0
|
|
for {
|
|
frame, more := frames.Next()
|
|
fmt.Fprintf(&b, "called from %s (%s:%d)\n", frame.Function, frame.File, frame.Line)
|
|
if !more {
|
|
break
|
|
}
|
|
i++
|
|
if i >= nFrames {
|
|
fmt.Fprintf(&b, "(rest of stack elided)\n")
|
|
break
|
|
}
|
|
}
|
|
return b.String()
|
|
}
|