Organizing log key/value pairs by group is a nice way to organize your logging data, but what about organizing the way your application groups data?
Maybe you want all logs created by a particular code path to be grouped together. How can you accomplish this?
Enter WithGroup…
Groups
…
Use Logger.WithGroup to qualify all of a Logger’s output with a group name. Calling WithGroup on a Logger results in a new Logger with the same Handler as the original, but with all its attributes qualified by the group name.
This can help prevent duplicate attribute keys in large systems, where subsystems might use the same keys. Pass each subsystem a different Logger with its own group name so that potential duplicates are qualified:
logger := slog.Default().With("id", systemID) parserLogger := logger.WithGroup("parser") parseInput(input, parserLogger)When parseInput logs with parserLogger, its keys will be qualified with “parser”, so even if it uses the common key “id”, the log line will have distinct keys.
So we’ve already seen how you can use With to add attributes to all logs in a logger:
logger = logger.With("request_id", req.Get("ID"))
WithGroup works the same way, but any additional keys added to the returned logger are added to the group:
func FooMiddleware(logger *slog.Logger) http.Handler {
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
logger = logger.With("foo_middleware")
/* ... */
logger.Info("Handling request", "method", r.Method)
/* ... */
logger.Debug("Checking credentials")
/* ... */
if err != nil {
logger.Error("foo failed", "error", err")
}
}
}
In this above example, all three logs add their keys inside the “foo_middleware” group, without having to specify the group more than once.