- package log
-
- import (
- "fmt"
-
- "github.com/pkg/errors"
- )
-
- // NewTracingLogger enables tracing by wrapping all errors (if they
- // implement stackTracer interface) in tracedError.
- //
- // All errors returned by https://github.com/pkg/errors implement stackTracer
- // interface.
- //
- // For debugging purposes only as it doubles the amount of allocations.
- func NewTracingLogger(next Logger) Logger {
- return &tracingLogger{
- next: next,
- }
- }
-
- type stackTracer interface {
- error
- StackTrace() errors.StackTrace
- }
-
- type tracingLogger struct {
- next Logger
- }
-
- func (l *tracingLogger) Info(msg string, keyvals ...interface{}) {
- l.next.Info(msg, formatErrors(keyvals)...)
- }
-
- func (l *tracingLogger) Debug(msg string, keyvals ...interface{}) {
- l.next.Debug(msg, formatErrors(keyvals)...)
- }
-
- func (l *tracingLogger) Error(msg string, keyvals ...interface{}) {
- l.next.Error(msg, formatErrors(keyvals)...)
- }
-
- func (l *tracingLogger) With(keyvals ...interface{}) Logger {
- return &tracingLogger{next: l.next.With(formatErrors(keyvals)...)}
- }
-
- func formatErrors(keyvals []interface{}) []interface{} {
- newKeyvals := make([]interface{}, len(keyvals))
- copy(newKeyvals, keyvals)
- for i := 0; i < len(newKeyvals)-1; i += 2 {
- if err, ok := newKeyvals[i+1].(stackTracer); ok {
- newKeyvals[i+1] = tracedError{err}
- }
- }
- return newKeyvals
- }
-
- // tracedError wraps a stackTracer and just makes the Error() result
- // always return a full stack trace.
- type tracedError struct {
- wrapped stackTracer
- }
-
- var _ stackTracer = tracedError{}
-
- func (t tracedError) StackTrace() errors.StackTrace {
- return t.wrapped.StackTrace()
- }
-
- func (t tracedError) Cause() error {
- return t.wrapped
- }
-
- func (t tracedError) Error() string {
- return fmt.Sprintf("%+v", t.wrapped)
- }
|