Skip to content

Commit 56e2263

Browse files
committedMay 20, 2023
feat: adding flatten formatter middleware
1 parent 1917a66 commit 56e2263

5 files changed

+205
-1
lines changed
 

‎README.md

+40
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,7 @@ Common formatters for [slog](https://pkg.go.dev/golang.org/x/exp/slog) library +
2525
- [HTTPResponseFormatter](#HTTPRequestFormatter-and-HTTPResponseFormatter): transforms a *http.Response into a readable object
2626
- [PIIFormatter](#PIIFormatter): hide private Personal Identifiable Information (PII)
2727
- [IPAddressFormatter](#IPAddressFormatter): hide ip address from logs
28+
- [FlattenFormatterMiddlewareOptions](#FlattenFormatterMiddlewareOptions): returns a formatter middleware that flatten attributes recursively
2829

2930
**Custom formatter:**
3031
- [Format](#Format): pass any attribute into a formatter
@@ -370,6 +371,45 @@ logger.
370371
// }
371372
```
372373

374+
### FlattenFormatterMiddlewareOptions
375+
376+
A formatter middleware that flatten attributes recursively.
377+
378+
```go
379+
import (
380+
slogformatter "github.com/samber/slog-formatter"
381+
slogmulti "github.com/samber/slog-multi"
382+
"golang.org/x/exp/slog"
383+
)
384+
385+
logger := slog.New(
386+
slogmulti.
387+
Pipe(slogformatter.FlattenFormatterMiddlewareOptions{Separator: ".", Prefix: "attrs", IgnorePath: false}.NewFlattenFormatterMiddlewareOptions()).
388+
Handler(slog.NewJSONHandler(os.Stdout)),
389+
)
390+
391+
logger.
392+
With("email", "samuel@acme.org").
393+
With("environment", "dev").
394+
WithGroup("group1").
395+
With("hello", "world").
396+
WithGroup("group2").
397+
With("hello", "world").
398+
Error("A message", "foo", "bar")
399+
400+
// outputs:
401+
// {
402+
// "time": "2023-05-20T22:14:55.857065+02:00",
403+
// "level": "ERROR",
404+
// "msg": "A message",
405+
// "attrs.email": "samuel@acme.org",
406+
// "attrs.environment": "dev",
407+
// "attrs.group1.hello": "world",
408+
// "attrs.group1.group2.hello": "world",
409+
// "foo": "bar"
410+
// }
411+
```
412+
373413
### Format
374414

375415
Pass every attributes into a formatter.

‎example/example.go

+19
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@ import (
55
"os"
66

77
slogformatter "github.com/samber/slog-formatter"
8+
slogmulti "github.com/samber/slog-multi"
89
"golang.org/x/exp/slog"
910
)
1011

@@ -104,6 +105,24 @@ func example() {
104105
Error("A message")
105106
}
106107

108+
func exampleFlatten() {
109+
logger := slog.New(
110+
slogmulti.
111+
Pipe(slogformatter.FlattenFormatterMiddlewareOptions{Separator: ".", Prefix: "attrs", IgnorePath: false}.NewFlattenFormatterMiddlewareOptions()).
112+
Handler(slog.NewJSONHandler(os.Stdout)),
113+
)
114+
115+
logger.
116+
With("email", "samuel@acme.org").
117+
With("environment", "dev").
118+
WithGroup("group1").
119+
With("hello", "world").
120+
WithGroup("group2").
121+
With("hello", "world").
122+
Error("A message", "foo", "bar")
123+
}
124+
107125
func main() {
108126
example()
127+
// exampleFlatten()
109128
}

‎formatter.go

-1
Original file line numberDiff line numberDiff line change
@@ -55,7 +55,6 @@ func FormatByKey(key string, formatter func(slog.Value) slog.Value) Formatter {
5555
}
5656

5757
return formatter(value), true
58-
5958
}
6059
}
6160

File renamed without changes.

‎middleware_formatter_flatten.go

+146
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,146 @@
1+
package slogformatter
2+
3+
import (
4+
"context"
5+
6+
"github.com/samber/lo"
7+
slogmulti "github.com/samber/slog-multi"
8+
"golang.org/x/exp/slog"
9+
)
10+
11+
type FlattenFormatterMiddlewareOptions struct {
12+
// Ignore attribute path and therefore ignore attribute key prefix.
13+
// Some attribute keys may collide.
14+
IgnorePath bool
15+
// Separator between prefix and key.
16+
Separator string
17+
// Attribute key prefix.
18+
Prefix string
19+
}
20+
21+
// NewFlattenFormatterMiddlewareOptions returns a formatter middleware that flatten attributes recursively.
22+
func (o FlattenFormatterMiddlewareOptions) NewFlattenFormatterMiddlewareOptions() slogmulti.Middleware {
23+
return func(next slog.Handler) slog.Handler {
24+
return &FlattenFormatterMiddleware{
25+
next: next,
26+
option: FlattenFormatterMiddlewareOptions{
27+
IgnorePath: o.IgnorePath,
28+
Separator: lo.Ternary(o.Separator == "", ".", o.Separator),
29+
Prefix: o.Prefix,
30+
},
31+
}
32+
}
33+
}
34+
35+
type FlattenFormatterMiddleware struct {
36+
next slog.Handler
37+
option FlattenFormatterMiddlewareOptions
38+
}
39+
40+
// Implements slog.Handler
41+
func (h *FlattenFormatterMiddleware) Enabled(ctx context.Context, level slog.Level) bool {
42+
return h.next.Enabled(ctx, level)
43+
}
44+
45+
// Implements slog.Handler
46+
func (h *FlattenFormatterMiddleware) Handle(ctx context.Context, record slog.Record) error {
47+
// newRecord := slog.NewRecord(record.Time, record.Level, record.Message, record.PC)
48+
49+
// record.Attrs(func(attr slog.Attr) bool {
50+
// if !h.option.IgnorePath {
51+
// newRecord.AddAttrs(
52+
// PrefixAttrKeys(
53+
// lo.Ternary(h.option.Prefix == "" || h.option.IgnorePath, "", h.option.Prefix+h.option.Separator),
54+
// FlattenAttrsWithPrefix(h.option.Separator, h.option.Prefix, []slog.Attr{attr}),
55+
// )...,
56+
// )
57+
// } else {
58+
// newRecord.AddAttrs(FlattenAttrs([]slog.Attr{attr})...)
59+
// }
60+
// return true
61+
// })
62+
63+
// return h.next.Handle(ctx, newRecord)
64+
return h.next.Handle(ctx, record)
65+
}
66+
67+
// Implements slog.Handler
68+
func (h *FlattenFormatterMiddleware) WithAttrs(attrs []slog.Attr) slog.Handler {
69+
if !h.option.IgnorePath {
70+
attrs = PrefixAttrKeys(
71+
lo.Ternary(h.option.Prefix == "" || h.option.IgnorePath, "", h.option.Prefix+h.option.Separator),
72+
FlattenAttrsWithPrefix(h.option.Separator, h.option.Prefix, attrs),
73+
)
74+
} else {
75+
attrs = FlattenAttrs(attrs)
76+
}
77+
78+
return &FlattenFormatterMiddleware{
79+
next: h.next.WithAttrs(attrs),
80+
option: h.option,
81+
}
82+
}
83+
84+
// Implements slog.Handler
85+
func (h *FlattenFormatterMiddleware) WithGroup(name string) slog.Handler {
86+
prefix := h.option.Prefix + h.option.Separator
87+
if h.option.IgnorePath || h.option.Prefix == "" {
88+
prefix = ""
89+
}
90+
91+
return &FlattenFormatterMiddleware{
92+
next: h.next,
93+
option: FlattenFormatterMiddlewareOptions{
94+
IgnorePath: h.option.IgnorePath,
95+
Separator: h.option.Separator,
96+
Prefix: prefix + name,
97+
},
98+
}
99+
}
100+
101+
// PrefixAttrKeys prefix attribute keys.
102+
func PrefixAttrKeys(prefix string, attrs []slog.Attr) []slog.Attr {
103+
return lo.Map(attrs, func(item slog.Attr, _ int) slog.Attr {
104+
return slog.Attr{
105+
Key: prefix + item.Key,
106+
Value: item.Value,
107+
}
108+
})
109+
}
110+
111+
// FlattenAttrs flatten attributes recursively.
112+
func FlattenAttrs(attrs []slog.Attr) []slog.Attr {
113+
output := []slog.Attr{}
114+
115+
for _, attr := range attrs {
116+
switch attr.Value.Kind() {
117+
case slog.KindAny, slog.KindBool, slog.KindDuration, slog.KindFloat64, slog.KindInt64, slog.KindUint64, slog.KindString, slog.KindTime:
118+
output = append(output, attr)
119+
case slog.KindGroup:
120+
output = append(output, attr.Value.Group()...)
121+
case slog.KindLogValuer:
122+
output = append(output, slog.Any(attr.Key, attr.Value.Resolve().Any()))
123+
}
124+
}
125+
126+
return output
127+
}
128+
129+
// FlattenAttrsWithPrefix flatten attributes recursively, with prefix.
130+
func FlattenAttrsWithPrefix(separator string, prefix string, attrs []slog.Attr) []slog.Attr {
131+
output := []slog.Attr{}
132+
133+
for _, attr := range attrs {
134+
switch attr.Value.Kind() {
135+
case slog.KindAny, slog.KindBool, slog.KindDuration, slog.KindFloat64, slog.KindInt64, slog.KindUint64, slog.KindString, slog.KindTime:
136+
output = append(output, attr)
137+
case slog.KindGroup:
138+
output = append(output, PrefixAttrKeys(prefix+separator+attr.Key+separator, FlattenAttrsWithPrefix(separator, "", attr.Value.Group()))...)
139+
case slog.KindLogValuer:
140+
attr := slog.Any(attr.Key, attr.Value.Resolve().Any())
141+
output = append(output, FlattenAttrsWithPrefix(separator, prefix, []slog.Attr{attr})...)
142+
}
143+
}
144+
145+
return output
146+
}

0 commit comments

Comments
 (0)
Please sign in to comment.