Skip to content

Commit 23d6cc7

Browse files
authored
Add Stringers field constructor for slices of Stringer-compatible objects (#1155)
Add a new Zap field constructor that encodes a slice of zero or more objects which implement fmt.Stringer interface into a Zap array. Usage: ```go // type Request struct{ ... } // func (a Request) String() string // // var requests []Request = ... // logger.Info("sending requests", zap.Stringers("requests", requests)) ``` This API uses a type parameter so that users don't have to build a `[]fmt.Stringer` out of a `[]Request`, and can pass the `[]Request` directly to the API. We may need a StringerValues constructor to mirror ObjectValues in the future. Credits: @zmanji, @abhinav for the suggestions
1 parent 1e46f5e commit 23d6cc7

File tree

2 files changed

+92
-1
lines changed

2 files changed

+92
-1
lines changed

array_go118.go

+33-1
Original file line numberDiff line numberDiff line change
@@ -23,7 +23,11 @@
2323

2424
package zap
2525

26-
import "go.uber.org/zap/zapcore"
26+
import (
27+
"fmt"
28+
29+
"go.uber.org/zap/zapcore"
30+
)
2731

2832
// Objects constructs a field with the given key, holding a list of the
2933
// provided objects that can be marshaled by Zap.
@@ -122,3 +126,31 @@ func (os objectValues[T, P]) MarshalLogArray(arr zapcore.ArrayEncoder) error {
122126
}
123127
return nil
124128
}
129+
130+
// Stringers constructs a field with the given key, holding a list of the
131+
// output provided by the value's String method
132+
//
133+
// Given an object that implements String on the value receiver, you
134+
// can log a slice of those objects with Objects like so:
135+
//
136+
// type Request struct{ ... }
137+
// func (a Request) String() string
138+
//
139+
// var requests []Request = ...
140+
// logger.Info("sending requests", zap.Stringers("requests", requests))
141+
//
142+
// Note that these objects must implement fmt.Stringer directly.
143+
// That is, if you're trying to marshal a []Request, the String method
144+
// must be declared on the Request type, not its pointer (*Request).
145+
func Stringers[T fmt.Stringer](key string, values []T) Field {
146+
return Array(key, stringers[T](values))
147+
}
148+
149+
type stringers[T fmt.Stringer] []T
150+
151+
func (os stringers[T]) MarshalLogArray(arr zapcore.ArrayEncoder) error {
152+
for _, o := range os {
153+
arr.AppendString(o.String())
154+
}
155+
return nil
156+
}

array_go118_test.go

+59
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,7 @@ package zap
2525

2626
import (
2727
"errors"
28+
"fmt"
2829
"testing"
2930

3031
"github.com/stretchr/testify/assert"
@@ -179,3 +180,61 @@ func TestObjectsAndObjectValues_marshalError(t *testing.T) {
179180
})
180181
}
181182
}
183+
184+
type stringerObject struct {
185+
value string
186+
}
187+
188+
func (s stringerObject) String() string {
189+
return s.value
190+
}
191+
192+
func TestStringers(t *testing.T) {
193+
t.Parallel()
194+
195+
tests := []struct {
196+
desc string
197+
give Field
198+
want []any
199+
}{
200+
{
201+
desc: "Stringers",
202+
give: Stringers("", []stringerObject{
203+
{value: "foo"},
204+
{value: "bar"},
205+
{value: "baz"},
206+
}),
207+
want: []any{
208+
"foo",
209+
"bar",
210+
"baz",
211+
},
212+
},
213+
{
214+
desc: "Stringers with []fmt.Stringer",
215+
give: Stringers("", []fmt.Stringer{
216+
stringerObject{value: "foo"},
217+
stringerObject{value: "bar"},
218+
stringerObject{value: "baz"},
219+
}),
220+
want: []any{
221+
"foo",
222+
"bar",
223+
"baz",
224+
},
225+
},
226+
}
227+
228+
for _, tt := range tests {
229+
tt := tt
230+
t.Run(tt.desc, func(t *testing.T) {
231+
t.Parallel()
232+
233+
tt.give.Key = "k"
234+
235+
enc := zapcore.NewMapObjectEncoder()
236+
tt.give.AddTo(enc)
237+
assert.Equal(t, tt.want, enc.Fields["k"])
238+
})
239+
}
240+
}

0 commit comments

Comments
 (0)