// Copyright (c) 2016 Uber Technologies, Inc. // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to deal // in the Software without restriction, including without limitation the rights // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell // copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in // all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN // THE SOFTWARE. package zap import ( "testing" "time" "go.uber.org/zap/zapcore" "github.com/stretchr/testify/assert" ) func BenchmarkBoolsArrayMarshaler(b *testing.B) { // Keep this benchmark here to capture the overhead of the ArrayMarshaler // wrapper. bs := make([]bool, 50) enc := zapcore.NewJSONEncoder(zapcore.EncoderConfig{}) b.ResetTimer() for i := 0; i < b.N; i++ { Bools("array", bs).AddTo(enc.Clone()) } } func BenchmarkBoolsReflect(b *testing.B) { bs := make([]bool, 50) enc := zapcore.NewJSONEncoder(zapcore.EncoderConfig{}) b.ResetTimer() for i := 0; i < b.N; i++ { Reflect("array", bs).AddTo(enc.Clone()) } } func TestArrayWrappers(t *testing.T) { tests := []struct { desc string field zapcore.Field expected []interface{} }{ {"empty bools", Bools("", []bool{}), []interface{}(nil)}, {"empty byte strings", ByteStrings("", [][]byte{}), []interface{}(nil)}, {"empty complex128s", Complex128s("", []complex128{}), []interface{}(nil)}, {"empty complex64s", Complex64s("", []complex64{}), []interface{}(nil)}, {"empty durations", Durations("", []time.Duration{}), []interface{}(nil)}, {"empty float64s", Float64s("", []float64{}), []interface{}(nil)}, {"empty float32s", Float32s("", []float32{}), []interface{}(nil)}, {"empty ints", Ints("", []int{}), []interface{}(nil)}, {"empty int64s", Int64s("", []int64{}), []interface{}(nil)}, {"empty int32s", Int32s("", []int32{}), []interface{}(nil)}, {"empty int16s", Int16s("", []int16{}), []interface{}(nil)}, {"empty int8s", Int8s("", []int8{}), []interface{}(nil)}, {"empty strings", Strings("", []string{}), []interface{}(nil)}, {"empty times", Times("", []time.Time{}), []interface{}(nil)}, {"empty uints", Uints("", []uint{}), []interface{}(nil)}, {"empty uint64s", Uint64s("", []uint64{}), []interface{}(nil)}, {"empty uint32s", Uint32s("", []uint32{}), []interface{}(nil)}, {"empty uint16s", Uint16s("", []uint16{}), []interface{}(nil)}, {"empty uint8s", Uint8s("", []uint8{}), []interface{}(nil)}, {"empty uintptrs", Uintptrs("", []uintptr{}), []interface{}(nil)}, {"bools", Bools("", []bool{true, false}), []interface{}{true, false}}, {"byte strings", ByteStrings("", [][]byte{{1, 2}, {3, 4}}), []interface{}{[]byte{1, 2}, []byte{3, 4}}}, {"complex128s", Complex128s("", []complex128{1 + 2i, 3 + 4i}), []interface{}{1 + 2i, 3 + 4i}}, {"complex64s", Complex64s("", []complex64{1 + 2i, 3 + 4i}), []interface{}{complex64(1 + 2i), complex64(3 + 4i)}}, {"durations", Durations("", []time.Duration{1, 2}), []interface{}{time.Nanosecond, 2 * time.Nanosecond}}, {"float64s", Float64s("", []float64{1.2, 3.4}), []interface{}{1.2, 3.4}}, {"float32s", Float32s("", []float32{1.2, 3.4}), []interface{}{float32(1.2), float32(3.4)}}, {"ints", Ints("", []int{1, 2}), []interface{}{1, 2}}, {"int64s", Int64s("", []int64{1, 2}), []interface{}{int64(1), int64(2)}}, {"int32s", Int32s("", []int32{1, 2}), []interface{}{int32(1), int32(2)}}, {"int16s", Int16s("", []int16{1, 2}), []interface{}{int16(1), int16(2)}}, {"int8s", Int8s("", []int8{1, 2}), []interface{}{int8(1), int8(2)}}, {"strings", Strings("", []string{"foo", "bar"}), []interface{}{"foo", "bar"}}, {"times", Times("", []time.Time{time.Unix(0, 0), time.Unix(0, 0)}), []interface{}{time.Unix(0, 0), time.Unix(0, 0)}}, {"uints", Uints("", []uint{1, 2}), []interface{}{uint(1), uint(2)}}, {"uint64s", Uint64s("", []uint64{1, 2}), []interface{}{uint64(1), uint64(2)}}, {"uint32s", Uint32s("", []uint32{1, 2}), []interface{}{uint32(1), uint32(2)}}, {"uint16s", Uint16s("", []uint16{1, 2}), []interface{}{uint16(1), uint16(2)}}, {"uint8s", Uint8s("", []uint8{1, 2}), []interface{}{uint8(1), uint8(2)}}, {"uintptrs", Uintptrs("", []uintptr{1, 2}), []interface{}{uintptr(1), uintptr(2)}}, } for _, tt := range tests { enc := zapcore.NewMapObjectEncoder() tt.field.Key = "k" tt.field.AddTo(enc) assert.Equal(t, tt.expected, enc.Fields["k"], "%s: unexpected map contents.", tt.desc) assert.Equal(t, 1, len(enc.Fields), "%s: found extra keys in map: %v", tt.desc, enc.Fields) } }