Skip to content

Commit

Permalink
change the way accumulated context is built
Browse files Browse the repository at this point in the history
  • Loading branch information
francoispqt committed Dec 28, 2018
1 parent 65e47b1 commit 7507fd9
Show file tree
Hide file tree
Showing 5 changed files with 75 additions and 59 deletions.
2 changes: 1 addition & 1 deletion benchmarks/Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ benchtrace:

.PHONY: benchonelog
benchonelog:
go test -benchmem -run=^BenchmarkOnelog -bench=^BenchmarkOnelog -benchtime=30ms
go test -benchmem -run=^BenchmarkOnelog.* -bench=^BenchmarkOnelog.* -benchtime=30ms

.PHONY: benchonelograce
benchonelograce:
Expand Down
19 changes: 12 additions & 7 deletions benchmarks/onelog_bench_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -94,12 +94,17 @@ func BenchmarkOnelog(b *testing.B) {
}
})
})
}

func BenchmarkOneLogTrace(b *testing.B) {
logger := onelog.New(ioutil.Discard, onelog.ALL)
b.ReportAllocs()
for i := 0; i < b.N; i++ {
logger.Info("message")
}
b.Run("accumulated context", func(b *testing.B) {
logger := onelog.New(ioutil.Discard, onelog.ALL).
With(func(e onelog.Entry) {
e.Int("int", 1)
})
b.ResetTimer()
b.RunParallel(func(pb *testing.PB) {
for pb.Next() {
logger.Info("message")
}
})
})
}
2 changes: 1 addition & 1 deletion entry.go
Original file line number Diff line number Diff line change
Expand Up @@ -84,7 +84,7 @@ func (e ChainEntry) Write() {
}
// first find writer for level
// if none, stop
e.Entry.l.closeEntry(e.enc)
e.Entry.l.closeEntry(e.Entry)
e.Entry.l.finalizeIfContext(e.Entry)
e.Entry.enc.Release()
}
Expand Down
101 changes: 51 additions & 50 deletions logger.go
Original file line number Diff line number Diff line change
Expand Up @@ -45,7 +45,7 @@ type Logger struct {
hook func(Entry)
w io.Writer
levels uint8
ctx []byte
ctx []func(Entry)
contextName string
}

Expand Down Expand Up @@ -82,32 +82,29 @@ func (l *Logger) Hook(h func(Entry)) *Logger {
}

func (l *Logger) copy(ctxName string) *Logger {
nL := Logger{
nL := &Logger{
levels: l.levels,
w: l.w,
hook: l.hook,
contextName: ctxName,
}
return &nL
if len(l.ctx) > 0 {
var ctx = make([]func(e Entry), len(l.ctx))
copy(ctx, l.ctx)
nL.ctx = ctx
}
return nL
}

// With copies the current Logger and adds it a given context by running func f.
func (l *Logger) With(f func(Entry)) *Logger {
nL := l.copy(l.contextName)

enc := gojay.BorrowEncoder(nL.w)

e := Entry{}
e.enc = enc
enc.AppendByte(' ')
f(e)

b := enc.Buf()
nL.ctx = make([]byte, len(b[1:]))
copy(nL.ctx, b[1:])

enc.Release()
if len(nL.ctx) == 0 {
nL.ctx = make([]func(Entry), 0, 1)
}

nL.ctx = append(nL.ctx, f)
return nL
}

Expand Down Expand Up @@ -136,13 +133,13 @@ func (l *Logger) Info(msg string) {
// if we do not require a context then we
// format with formatter and return.
if l.contextName == "" {
l.beginEntry(e.Level, msg, enc)
l.beginEntry(e.Level, msg, e)
l.runHook(e)
} else {
l.openEntry(enc)
}

l.closeEntry(enc)
l.closeEntry(e)
l.finalizeIfContext(e)

enc.Release()
Expand All @@ -169,7 +166,7 @@ func (l *Logger) InfoWith(msg string) ChainEntry {
// if we do not require a context then we
// format with formatter and return.
if l.contextName == "" {
l.beginEntry(e.Level, msg, e.Entry.enc)
l.beginEntry(e.Level, msg, e.Entry)
l.runHook(e.Entry)
return e
}
Expand All @@ -192,14 +189,14 @@ func (l *Logger) InfoWithFields(msg string, fields func(Entry)) {
// if we do not require a context then we
// format with formatter and return.
if l.contextName == "" {
l.beginEntry(e.Level, msg, e.enc)
l.beginEntry(e.Level, msg, e)
l.runHook(e)
} else {
l.openEntry(e.enc)
}

fields(e)
l.closeEntry(e.enc)
l.closeEntry(e)
l.finalizeIfContext(e)

e.enc.Release()
Expand All @@ -219,13 +216,13 @@ func (l *Logger) Debug(msg string) {
// if we do not require a context then we
// format with formatter and return.
if l.contextName == "" {
l.beginEntry(e.Level, msg, e.enc)
l.beginEntry(e.Level, msg, e)
l.runHook(e)
} else {
l.openEntry(e.enc)
}

l.closeEntry(e.enc)
l.closeEntry(e)
l.finalizeIfContext(e)

e.enc.Release()
Expand All @@ -252,7 +249,7 @@ func (l *Logger) DebugWith(msg string) ChainEntry {
// if we do not require a context then we
// format with formatter and return.
if l.contextName == "" {
l.beginEntry(e.Level, msg, e.Entry.enc)
l.beginEntry(e.Level, msg, e.Entry)
l.runHook(e.Entry)
return e
}
Expand All @@ -275,14 +272,14 @@ func (l *Logger) DebugWithFields(msg string, fields func(Entry)) {
// if we do not require a context then we
// format with formatter and return.
if l.contextName == "" {
l.beginEntry(e.Level, msg, e.enc)
l.beginEntry(e.Level, msg, e)
l.runHook(e)
} else {
l.openEntry(e.enc)
}

fields(e)
l.closeEntry(e.enc)
l.closeEntry(e)
l.finalizeIfContext(e)

e.enc.Release()
Expand All @@ -302,13 +299,13 @@ func (l *Logger) Warn(msg string) {
// if we do not require a context then we
// format with formatter and return.
if l.contextName == "" {
l.beginEntry(e.Level, msg, e.enc)
l.beginEntry(e.Level, msg, e)
l.runHook(e)
} else {
l.openEntry(e.enc)
}

l.closeEntry(e.enc)
l.closeEntry(e)
l.finalizeIfContext(e)

e.enc.Release()
Expand All @@ -335,7 +332,7 @@ func (l *Logger) WarnWith(msg string) ChainEntry {
// if we do not require a context then we
// format with formatter and return.
if l.contextName == "" {
l.beginEntry(e.Level, msg, e.Entry.enc)
l.beginEntry(e.Level, msg, e.Entry)
l.runHook(e.Entry)
return e
}
Expand All @@ -359,14 +356,14 @@ func (l *Logger) WarnWithFields(msg string, fields func(Entry)) {
// if we do not require a context then we
// format with formatter and return.
if l.contextName == "" {
l.beginEntry(e.Level, msg, e.enc)
l.beginEntry(e.Level, msg, e)
l.runHook(e)
} else {
l.openEntry(e.enc)
}

fields(e)
l.closeEntry(e.enc)
l.closeEntry(e)
l.finalizeIfContext(e)

e.enc.Release()
Expand All @@ -387,13 +384,13 @@ func (l *Logger) Error(msg string) {
// if we do not require a context then we
// format with formatter and return.
if l.contextName == "" {
l.beginEntry(e.Level, msg, e.enc)
l.beginEntry(e.Level, msg, e)
l.runHook(e)
} else {
l.openEntry(e.enc)
}

l.closeEntry(e.enc)
l.closeEntry(e)
l.finalizeIfContext(e)

e.enc.Release()
Expand All @@ -420,7 +417,7 @@ func (l *Logger) ErrorWith(msg string) ChainEntry {
// if we do not require a context then we
// format with formatter and return.
if l.contextName == "" {
l.beginEntry(e.Level, msg, e.Entry.enc)
l.beginEntry(e.Level, msg, e.Entry)
l.runHook(e.Entry)
return e
}
Expand All @@ -444,14 +441,14 @@ func (l *Logger) ErrorWithFields(msg string, fields func(Entry)) {
// if we do not require a context then we
// format with formatter and return.
if l.contextName == "" {
l.beginEntry(e.Level, msg, e.enc)
l.beginEntry(e.Level, msg, e)
l.runHook(e)
} else {
l.openEntry(e.enc)
}

fields(e)
l.closeEntry(e.enc)
l.closeEntry(e)
l.finalizeIfContext(e)
e.enc.Release()
}
Expand All @@ -471,13 +468,13 @@ func (l *Logger) Fatal(msg string) {
// if we do not require a context then we
// format with formatter and return.
if l.contextName == "" {
l.beginEntry(e.Level, msg, e.enc)
l.beginEntry(e.Level, msg, e)
l.runHook(e)
} else {
l.openEntry(e.enc)
}

l.closeEntry(e.enc)
l.closeEntry(e)
l.finalizeIfContext(e)

e.enc.Release()
Expand All @@ -504,7 +501,7 @@ func (l *Logger) FatalWith(msg string) ChainEntry {
// if we do not require a context then we
// format with formatter and return.
if l.contextName == "" {
l.beginEntry(e.Level, msg, e.Entry.enc)
l.beginEntry(e.Level, msg, e.Entry)
l.runHook(e.Entry)
return e
}
Expand All @@ -529,14 +526,14 @@ func (l *Logger) FatalWithFields(msg string, fields func(Entry)) {
// if we do not require a context then we
// format with formatter and return.
if l.contextName == "" {
l.beginEntry(e.Level, msg, e.enc)
l.beginEntry(e.Level, msg, e)
l.runHook(e)
} else {
l.openEntry(e.enc)
}

fields(e)
l.closeEntry(e.enc)
l.closeEntry(e)
l.finalizeIfContext(e)

e.enc.Release()
Expand All @@ -546,12 +543,14 @@ func (l *Logger) openEntry(enc *Encoder) {
enc.AppendBytes(logOpen)
}

func (l *Logger) beginEntry(level uint8, msg string, enc *Encoder) {
enc.AppendBytes(levelsJSON[level])
enc.AppendString(msg)
func (l *Logger) beginEntry(level uint8, msg string, e Entry) {
e.enc.AppendBytes(levelsJSON[level])
e.enc.AppendString(msg)

if l.ctx != nil && l.contextName == "" {
enc.AppendBytes(l.ctx)
for _, c := range l.ctx {
c(e)
}
}
}

Expand All @@ -576,7 +575,7 @@ func (l *Logger) finalizeIfContext(entry Entry) {
entry.enc = entryEnc

// create dummy entry for applying hooks.
l.beginEntry(entry.Level, entry.Message, entryEnc)
l.beginEntry(entry.Level, entry.Message, entry)
l.runHook(entry)

// Add entry's encoded data into new encoder.
Expand All @@ -591,18 +590,20 @@ func (l *Logger) finalizeIfContext(entry Entry) {
entryEnc.Write()
}

func (l *Logger) closeEntry(enc *Encoder) {
func (l *Logger) closeEntry(e Entry) {
if l.contextName == "" {
enc.AppendBytes(logClose)
e.enc.AppendBytes(logClose)
} else {
if l.ctx != nil {
enc.AppendBytes(l.ctx)
for _, c := range l.ctx {
c(e)
}
}
enc.AppendBytes(logCloseOnly)
e.enc.AppendBytes(logCloseOnly)
}

if l.contextName == "" {
enc.Write()
e.enc.Write()
}
}

Expand Down
10 changes: 10 additions & 0 deletions logger_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -1092,4 +1092,14 @@ func TestOnelogFieldsChainAndContext(t *testing.T) {
json := `{"level":"fatal","message":"message","params":{"userID":"123456","action":"login","result":"success","int64":120,"thunder_frequency":1000}}` + "\n"
assert.Equal(t, json, string(w.b), "bytes written to the writer dont equal expected result")
})

t.Run("multi-augmented-logger", func(t *testing.T) {
w := newWriter()
parent := NewContext(w, DEBUG|INFO|WARN|ERROR|FATAL, "params")
logger := parent.With(func(e Entry) { e.Int("thunder_frequency", 1000) })
logger = logger.With(func(e Entry) { e.String("foo", "bar") })
logger.Fatal("message")
json := `{"level":"fatal","message":"message","params":{"thunder_frequency":1000,"foo":"bar"}}` + "\n"
assert.Equal(t, json, string(w.b), "bytes written to the writer dont equal expected result")
})
}

0 comments on commit 7507fd9

Please sign in to comment.