Skip to content

Commit 9b86a50

Browse files
author
Craig Pastro
authored
SugaredLogger: Turn error into zap.Error (#1185)
1 parent 9137e0e commit 9b86a50

File tree

2 files changed

+43
-8
lines changed

2 files changed

+43
-8
lines changed

sugar.go

+20-4
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,7 @@ import (
3131
const (
3232
_oddNumberErrMsg = "Ignored key without a value."
3333
_nonStringKeyErrMsg = "Ignored key-value pairs with non-string keys."
34+
_multipleErrMsg = "Multiple errors without a key."
3435
)
3536

3637
// A SugaredLogger wraps the base Logger functionality in a slower, but less
@@ -336,10 +337,13 @@ func (s *SugaredLogger) sweetenFields(args []interface{}) []Field {
336337
return nil
337338
}
338339

339-
// Allocate enough space for the worst case; if users pass only structured
340-
// fields, we shouldn't penalize them with extra allocations.
341-
fields := make([]Field, 0, len(args))
342-
var invalid invalidPairs
340+
var (
341+
// Allocate enough space for the worst case; if users pass only structured
342+
// fields, we shouldn't penalize them with extra allocations.
343+
fields = make([]Field, 0, len(args))
344+
invalid invalidPairs
345+
seenError bool
346+
)
343347

344348
for i := 0; i < len(args); {
345349
// This is a strongly-typed field. Consume it and move on.
@@ -349,6 +353,18 @@ func (s *SugaredLogger) sweetenFields(args []interface{}) []Field {
349353
continue
350354
}
351355

356+
// If it is an error, consume it and move on.
357+
if err, ok := args[i].(error); ok {
358+
if !seenError {
359+
seenError = true
360+
fields = append(fields, Error(err))
361+
} else {
362+
s.base.Error(_multipleErrMsg, Error(err))
363+
}
364+
i++
365+
continue
366+
}
367+
352368
// Make sure this element isn't a dangling key.
353369
if i == len(args)-1 {
354370
s.base.Error(_oddNumberErrMsg, Any("ignored", args[i]))

sugar_test.go

+23-4
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,7 @@
2121
package zap
2222

2323
import (
24+
"errors"
2425
"testing"
2526

2627
"go.uber.org/zap/internal/exit"
@@ -46,6 +47,12 @@ func TestSugarWith(t *testing.T) {
4647
Context: []Field{Array("invalid", invalidPairs(pairs))},
4748
}
4849
}
50+
ignoredError := func(err error) observer.LoggedEntry {
51+
return observer.LoggedEntry{
52+
Entry: zapcore.Entry{Level: ErrorLevel, Message: _multipleErrMsg},
53+
Context: []Field{Error(err)},
54+
}
55+
}
4956

5057
tests := []struct {
5158
desc string
@@ -122,6 +129,15 @@ func TestSugarWith(t *testing.T) {
122129
nonString(invalidPair{2, true, "bar"}, invalidPair{5, 42, "reversed"}),
123130
},
124131
},
132+
{
133+
desc: "multiple errors",
134+
args: []interface{}{errors.New("first"), errors.New("second"), errors.New("third")},
135+
expected: []Field{Error(errors.New("first"))},
136+
errLogs: []observer.LoggedEntry{
137+
ignoredError(errors.New("second")),
138+
ignoredError(errors.New("third")),
139+
},
140+
},
125141
}
126142

127143
for _, tt := range tests {
@@ -198,10 +214,13 @@ func TestSugarStructuredLogging(t *testing.T) {
198214
}
199215

200216
// Common to all test cases.
201-
context := []interface{}{"foo", "bar"}
202-
extra := []interface{}{"baz", false}
203-
expectedFields := []Field{String("foo", "bar"), Bool("baz", false)}
204-
217+
var (
218+
err = errors.New("qux")
219+
context = []interface{}{"foo", "bar"}
220+
extra = []interface{}{err, "baz", false}
221+
expectedFields = []Field{String("foo", "bar"), Error(err), Bool("baz", false)}
222+
)
223+
205224
for _, tt := range tests {
206225
withSugar(t, DebugLevel, nil, func(logger *SugaredLogger, logs *observer.ObservedLogs) {
207226
logger.With(context...).Debugw(tt.msg, extra...)

0 commit comments

Comments
 (0)