Skip to content

Commit

Permalink
Handle special values like Inf and NaN gracefuly
Browse files Browse the repository at this point in the history
  • Loading branch information
rs committed Jul 25, 2017
1 parent 2aa3c3a commit 87aceba
Show file tree
Hide file tree
Showing 3 changed files with 93 additions and 6 deletions.
35 changes: 35 additions & 0 deletions benchmark_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -71,3 +71,38 @@ func BenchmarkLogFields(b *testing.B) {
}
})
}

func BenchmarkLogFieldType(b *testing.B) {
types := map[string]func(e *Event) *Event{
"Int": func(e *Event) *Event {
return e.Int("int", 1)
},
"Float32": func(e *Event) *Event {
return e.Float32("float", 1)
},
"Str": func(e *Event) *Event {
return e.Str("str", "foo")
},
"Err": func(e *Event) *Event {
return e.Err(errExample)
},
"Time": func(e *Event) *Event {
return e.Time("time", time.Time{})
},
"Dur": func(e *Event) *Event {
return e.Dur("dur", 1*time.Millisecond)
},
}
logger := New(ioutil.Discard)
b.ResetTimer()
for name := range types {
f := types[name]
b.Run(name, func(b *testing.B) {
b.RunParallel(func(pb *testing.PB) {
for pb.Next() {
f(logger.Info()).Msg("")
}
})
})
}
}
31 changes: 25 additions & 6 deletions field.go
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@ package zerolog
import (
"encoding/json"
"fmt"
"math"
"sort"
"strconv"
"time"
Expand Down Expand Up @@ -157,6 +158,9 @@ func appendErrorsKey(dst []byte, key string, errs []error) []byte {
}

func appendError(dst []byte, err error) []byte {
if err == nil {
return dst
}
return appendErrorKey(dst, ErrorFieldName, err)
}

Expand Down Expand Up @@ -369,38 +373,53 @@ func appendUints64(dst []byte, key string, vals []uint64) []byte {
return dst
}

func appendFloat(dst []byte, val float64, bitSize int) []byte {
// JSON does not permit NaN or Infinity. A typical JSON encoder would fail
// with an error, but a logging library wants the data to get thru so we
// make a tradeoff and store those types as string.
switch {
case math.IsNaN(val):
return append(dst, `"NaN"`...)
case math.IsInf(val, 1):
return append(dst, `"+Inf"`...)
case math.IsInf(val, -1):
return append(dst, `"-Inf"`...)
}
return strconv.AppendFloat(dst, val, 'f', -1, bitSize)
}

func appendFloat32(dst []byte, key string, val float32) []byte {
return strconv.AppendFloat(appendKey(dst, key), float64(val), 'f', -1, 32)
return appendFloat(appendKey(dst, key), float64(val), 32)
}

func appendFloats32(dst []byte, key string, vals []float32) []byte {
if len(vals) == 0 {
return append(appendKey(dst, key), '[', ']')
}
dst = append(appendKey(dst, key), '[')
dst = strconv.AppendFloat(dst, float64(vals[0]), 'f', -1, 32)
dst = appendFloat(dst, float64(vals[0]), 32)
if len(vals) > 1 {
for _, val := range vals[1:] {
dst = strconv.AppendFloat(append(dst, ','), float64(val), 'f', -1, 32)
dst = appendFloat(append(dst, ','), float64(val), 32)
}
}
dst = append(dst, ']')
return dst
}

func appendFloat64(dst []byte, key string, val float64) []byte {
return strconv.AppendFloat(appendKey(dst, key), val, 'f', -1, 32)
return appendFloat(appendKey(dst, key), val, 64)
}

func appendFloats64(dst []byte, key string, vals []float64) []byte {
if len(vals) == 0 {
return append(appendKey(dst, key), '[', ']')
}
dst = append(appendKey(dst, key), '[')
dst = strconv.AppendFloat(dst, vals[0], 'f', -1, 32)
dst = appendFloat(dst, vals[0], 32)
if len(vals) > 1 {
for _, val := range vals[1:] {
dst = strconv.AppendFloat(append(dst, ','), val, 'f', -1, 32)
dst = appendFloat(append(dst, ','), val, 64)
}
}
dst = append(dst, ']')
Expand Down
33 changes: 33 additions & 0 deletions field_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
package zerolog

import (
"math"
"reflect"
"testing"
)

func Test_appendFloat64(t *testing.T) {
tests := []struct {
name string
input float64
want []byte
}{
{"-Inf", math.Inf(-1), []byte(`"foo":"-Inf"`)},
{"+Inf", math.Inf(1), []byte(`"foo":"+Inf"`)},
{"NaN", math.NaN(), []byte(`"foo":"NaN"`)},
{"0", 0, []byte(`"foo":0`)},
{"-1.1", -1.1, []byte(`"foo":-1.1`)},
{"1e20", 1e20, []byte(`"foo":100000000000000000000`)},
{"1e21", 1e21, []byte(`"foo":1000000000000000000000`)},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
if got := appendFloat32([]byte{}, "foo", float32(tt.input)); !reflect.DeepEqual(got, tt.want) {
t.Errorf("appendFloat32() = %s, want %s", got, tt.want)
}
if got := appendFloat64([]byte{}, "foo", tt.input); !reflect.DeepEqual(got, tt.want) {
t.Errorf("appendFloat32() = %s, want %s", got, tt.want)
}
})
}
}

0 comments on commit 87aceba

Please sign in to comment.