-
Notifications
You must be signed in to change notification settings - Fork 4
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
log/slog/internal/benchmarks: slog benchmarks
Add a suite of benchmarks for the LogAttrs method, which is intended to be fast. Updates golang#56345. Change-Id: If43f9f250bd588247c539bed87f81be7f5428c6d Reviewed-on: https://go-review.googlesource.com/c/go/+/478200 TryBot-Result: Gopher Robot <[email protected]> Run-TryBot: Jonathan Amsterdam <[email protected]> Reviewed-by: Alan Donovan <[email protected]>
- Loading branch information
Showing
5 changed files
with
392 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,50 @@ | ||
// Copyright 2023 The Go Authors. All rights reserved. | ||
// Use of this source code is governed by a BSD-style | ||
// license that can be found in the LICENSE file. | ||
|
||
// Package benchmarks contains benchmarks for slog. | ||
// | ||
// These benchmarks are loosely based on github.com/uber-go/zap/benchmarks. | ||
// They have the following desirable properties: | ||
// | ||
// - They test a complete log event, from the user's call to its return. | ||
// | ||
// - The benchmarked code is run concurrently in multiple goroutines, to | ||
// better simulate a real server (the most common environment for structured | ||
// logs). | ||
// | ||
// - Some handlers are optimistic versions of real handlers, doing real-world | ||
// tasks as fast as possible (and sometimes faster, in that an | ||
// implementation may not be concurrency-safe). This gives us an upper bound | ||
// on handler performance, so we can evaluate the (handler-independent) core | ||
// activity of the package in an end-to-end context without concern that a | ||
// slow handler implementation is skewing the results. | ||
// | ||
// - We also test the built-in handlers, for comparison. | ||
package benchmarks | ||
|
||
import ( | ||
"errors" | ||
"log/slog" | ||
"time" | ||
) | ||
|
||
const testMessage = "Test logging, but use a somewhat realistic message length." | ||
|
||
var ( | ||
testTime = time.Date(2022, time.May, 1, 0, 0, 0, 0, time.UTC) | ||
testString = "7e3b3b2aaeff56a7108fe11e154200dd/7819479873059528190" | ||
testInt = 32768 | ||
testDuration = 23 * time.Second | ||
testError = errors.New("fail") | ||
) | ||
|
||
var testAttrs = []slog.Attr{ | ||
slog.String("string", testString), | ||
slog.Int("status", testInt), | ||
slog.Duration("duration", testDuration), | ||
slog.Time("time", testTime), | ||
slog.Any("error", testError), | ||
} | ||
|
||
const wantText = "time=1651363200 level=0 msg=Test logging, but use a somewhat realistic message length. string=7e3b3b2aaeff56a7108fe11e154200dd/7819479873059528190 status=32768 duration=23000000000 time=1651363200 error=fail\n" |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,152 @@ | ||
// Copyright 2022 The Go Authors. All rights reserved. | ||
// Use of this source code is governed by a BSD-style | ||
// license that can be found in the LICENSE file. | ||
|
||
package benchmarks | ||
|
||
import ( | ||
"context" | ||
"flag" | ||
"internal/race" | ||
"io" | ||
"log/slog" | ||
"log/slog/internal" | ||
"testing" | ||
) | ||
|
||
func init() { | ||
flag.BoolVar(&internal.IgnorePC, "nopc", false, "do not invoke runtime.Callers") | ||
} | ||
|
||
// We pass Attrs inline because it affects allocations: building | ||
// up a list outside of the benchmarked code and passing it in with "..." | ||
// reduces measured allocations. | ||
|
||
func BenchmarkAttrs(b *testing.B) { | ||
ctx := context.Background() | ||
for _, handler := range []struct { | ||
name string | ||
h slog.Handler | ||
skipRace bool | ||
}{ | ||
{"disabled", disabledHandler{}, false}, | ||
{"async discard", newAsyncHandler(), true}, | ||
{"fastText discard", newFastTextHandler(io.Discard), false}, | ||
{"Text discard", slog.NewTextHandler(io.Discard), false}, | ||
{"JSON discard", slog.NewJSONHandler(io.Discard), false}, | ||
} { | ||
logger := slog.New(handler.h) | ||
b.Run(handler.name, func(b *testing.B) { | ||
if handler.skipRace && race.Enabled { | ||
b.Skip("skipping benchmark in race mode") | ||
} | ||
for _, call := range []struct { | ||
name string | ||
f func() | ||
}{ | ||
{ | ||
// The number should match nAttrsInline in slog/record.go. | ||
// This should exercise the code path where no allocations | ||
// happen in Record or Attr. If there are allocations, they | ||
// should only be from Duration.String and Time.String. | ||
"5 args", | ||
func() { | ||
logger.LogAttrs(nil, slog.LevelInfo, testMessage, | ||
slog.String("string", testString), | ||
slog.Int("status", testInt), | ||
slog.Duration("duration", testDuration), | ||
slog.Time("time", testTime), | ||
slog.Any("error", testError), | ||
) | ||
}, | ||
}, | ||
{ | ||
"5 args ctx", | ||
func() { | ||
logger.LogAttrs(ctx, slog.LevelInfo, testMessage, | ||
slog.String("string", testString), | ||
slog.Int("status", testInt), | ||
slog.Duration("duration", testDuration), | ||
slog.Time("time", testTime), | ||
slog.Any("error", testError), | ||
) | ||
}, | ||
}, | ||
{ | ||
"10 args", | ||
func() { | ||
logger.LogAttrs(nil, slog.LevelInfo, testMessage, | ||
slog.String("string", testString), | ||
slog.Int("status", testInt), | ||
slog.Duration("duration", testDuration), | ||
slog.Time("time", testTime), | ||
slog.Any("error", testError), | ||
slog.String("string", testString), | ||
slog.Int("status", testInt), | ||
slog.Duration("duration", testDuration), | ||
slog.Time("time", testTime), | ||
slog.Any("error", testError), | ||
) | ||
}, | ||
}, | ||
{ | ||
// Try an extreme value to see if the results are reasonable. | ||
"40 args", | ||
func() { | ||
logger.LogAttrs(nil, slog.LevelInfo, testMessage, | ||
slog.String("string", testString), | ||
slog.Int("status", testInt), | ||
slog.Duration("duration", testDuration), | ||
slog.Time("time", testTime), | ||
slog.Any("error", testError), | ||
slog.String("string", testString), | ||
slog.Int("status", testInt), | ||
slog.Duration("duration", testDuration), | ||
slog.Time("time", testTime), | ||
slog.Any("error", testError), | ||
slog.String("string", testString), | ||
slog.Int("status", testInt), | ||
slog.Duration("duration", testDuration), | ||
slog.Time("time", testTime), | ||
slog.Any("error", testError), | ||
slog.String("string", testString), | ||
slog.Int("status", testInt), | ||
slog.Duration("duration", testDuration), | ||
slog.Time("time", testTime), | ||
slog.Any("error", testError), | ||
slog.String("string", testString), | ||
slog.Int("status", testInt), | ||
slog.Duration("duration", testDuration), | ||
slog.Time("time", testTime), | ||
slog.Any("error", testError), | ||
slog.String("string", testString), | ||
slog.Int("status", testInt), | ||
slog.Duration("duration", testDuration), | ||
slog.Time("time", testTime), | ||
slog.Any("error", testError), | ||
slog.String("string", testString), | ||
slog.Int("status", testInt), | ||
slog.Duration("duration", testDuration), | ||
slog.Time("time", testTime), | ||
slog.Any("error", testError), | ||
slog.String("string", testString), | ||
slog.Int("status", testInt), | ||
slog.Duration("duration", testDuration), | ||
slog.Time("time", testTime), | ||
slog.Any("error", testError), | ||
) | ||
}, | ||
}, | ||
} { | ||
b.Run(call.name, func(b *testing.B) { | ||
b.ReportAllocs() | ||
b.RunParallel(func(pb *testing.PB) { | ||
for pb.Next() { | ||
call.f() | ||
} | ||
}) | ||
}) | ||
} | ||
}) | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,147 @@ | ||
// Copyright 2022 The Go Authors. All rights reserved. | ||
// Use of this source code is governed by a BSD-style | ||
// license that can be found in the LICENSE file. | ||
|
||
package benchmarks | ||
|
||
// Handlers for benchmarking. | ||
|
||
import ( | ||
"context" | ||
"fmt" | ||
"io" | ||
"log/slog" | ||
"log/slog/internal/buffer" | ||
"strconv" | ||
"time" | ||
) | ||
|
||
// A fastTextHandler writes a Record to an io.Writer in a format similar to | ||
// slog.TextHandler, but without quoting or locking. It has a few other | ||
// performance-motivated shortcuts, like writing times as seconds since the | ||
// epoch instead of strings. | ||
// | ||
// It is intended to represent a high-performance Handler that synchronously | ||
// writes text (as opposed to binary). | ||
type fastTextHandler struct { | ||
w io.Writer | ||
} | ||
|
||
func newFastTextHandler(w io.Writer) slog.Handler { | ||
return &fastTextHandler{w: w} | ||
} | ||
|
||
func (h *fastTextHandler) Enabled(context.Context, slog.Level) bool { return true } | ||
|
||
func (h *fastTextHandler) Handle(_ context.Context, r slog.Record) error { | ||
buf := buffer.New() | ||
defer buf.Free() | ||
|
||
if !r.Time.IsZero() { | ||
buf.WriteString("time=") | ||
h.appendTime(buf, r.Time) | ||
buf.WriteByte(' ') | ||
} | ||
buf.WriteString("level=") | ||
*buf = strconv.AppendInt(*buf, int64(r.Level), 10) | ||
buf.WriteByte(' ') | ||
buf.WriteString("msg=") | ||
buf.WriteString(r.Message) | ||
r.Attrs(func(a slog.Attr) { | ||
buf.WriteByte(' ') | ||
buf.WriteString(a.Key) | ||
buf.WriteByte('=') | ||
h.appendValue(buf, a.Value) | ||
}) | ||
buf.WriteByte('\n') | ||
_, err := h.w.Write(*buf) | ||
return err | ||
} | ||
|
||
func (h *fastTextHandler) appendValue(buf *buffer.Buffer, v slog.Value) { | ||
switch v.Kind() { | ||
case slog.KindString: | ||
buf.WriteString(v.String()) | ||
case slog.KindInt64: | ||
*buf = strconv.AppendInt(*buf, v.Int64(), 10) | ||
case slog.KindUint64: | ||
*buf = strconv.AppendUint(*buf, v.Uint64(), 10) | ||
case slog.KindFloat64: | ||
*buf = strconv.AppendFloat(*buf, v.Float64(), 'g', -1, 64) | ||
case slog.KindBool: | ||
*buf = strconv.AppendBool(*buf, v.Bool()) | ||
case slog.KindDuration: | ||
*buf = strconv.AppendInt(*buf, v.Duration().Nanoseconds(), 10) | ||
case slog.KindTime: | ||
h.appendTime(buf, v.Time()) | ||
case slog.KindAny: | ||
a := v.Any() | ||
switch a := a.(type) { | ||
case error: | ||
buf.WriteString(a.Error()) | ||
default: | ||
fmt.Fprint(buf, a) | ||
} | ||
default: | ||
panic(fmt.Sprintf("bad kind: %s", v.Kind())) | ||
} | ||
} | ||
|
||
func (h *fastTextHandler) appendTime(buf *buffer.Buffer, t time.Time) { | ||
*buf = strconv.AppendInt(*buf, t.Unix(), 10) | ||
} | ||
|
||
func (h *fastTextHandler) WithAttrs([]slog.Attr) slog.Handler { | ||
panic("fastTextHandler: With unimplemented") | ||
} | ||
|
||
func (*fastTextHandler) WithGroup(string) slog.Handler { | ||
panic("fastTextHandler: WithGroup unimplemented") | ||
} | ||
|
||
// An asyncHandler simulates a Handler that passes Records to a | ||
// background goroutine for processing. | ||
// Because sending to a channel can be expensive due to locking, | ||
// we simulate a lock-free queue by adding the Record to a ring buffer. | ||
// Omitting the locking makes this little more than a copy of the Record, | ||
// but that is a worthwhile thing to measure because Records are on the large | ||
// side. Since nothing actually reads from the ring buffer, it can handle an | ||
// arbitrary number of Records without either blocking or allocation. | ||
type asyncHandler struct { | ||
ringBuffer [100]slog.Record | ||
next int | ||
} | ||
|
||
func newAsyncHandler() *asyncHandler { | ||
return &asyncHandler{} | ||
} | ||
|
||
func (*asyncHandler) Enabled(context.Context, slog.Level) bool { return true } | ||
|
||
func (h *asyncHandler) Handle(_ context.Context, r slog.Record) error { | ||
h.ringBuffer[h.next] = r.Clone() | ||
h.next = (h.next + 1) % len(h.ringBuffer) | ||
return nil | ||
} | ||
|
||
func (*asyncHandler) WithAttrs([]slog.Attr) slog.Handler { | ||
panic("asyncHandler: With unimplemented") | ||
} | ||
|
||
func (*asyncHandler) WithGroup(string) slog.Handler { | ||
panic("asyncHandler: WithGroup unimplemented") | ||
} | ||
|
||
// A disabledHandler's Enabled method always returns false. | ||
type disabledHandler struct{} | ||
|
||
func (disabledHandler) Enabled(context.Context, slog.Level) bool { return false } | ||
func (disabledHandler) Handle(context.Context, slog.Record) error { panic("should not be called") } | ||
|
||
func (disabledHandler) WithAttrs([]slog.Attr) slog.Handler { | ||
panic("disabledHandler: With unimplemented") | ||
} | ||
|
||
func (disabledHandler) WithGroup(string) slog.Handler { | ||
panic("disabledHandler: WithGroup unimplemented") | ||
} |
Oops, something went wrong.