-
-
Notifications
You must be signed in to change notification settings - Fork 20
/
Copy pathrouter.go
80 lines (66 loc) · 1.66 KB
/
router.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
package slogmulti
import (
"context"
"log/slog"
)
type router struct {
handlers []slog.Handler
}
// Router forward record to all matching slog.Handler.
func Router() *router {
return &router{
handlers: []slog.Handler{},
}
}
// Add a new handler to the router. The handler will be called if all matchers return true.
func (h *router) Add(handler slog.Handler, matchers ...func(ctx context.Context, r slog.Record) bool) *router {
return &router{
handlers: append(
h.handlers,
&RoutableHandler{
matchers: matchers,
handler: handler,
},
),
}
}
func (h *router) Handler() slog.Handler {
return Fanout(h.handlers...)
}
var _ slog.Handler = (*RoutableHandler)(nil)
// @TODO: implement round robin strategy ?
type RoutableHandler struct {
matchers []func(ctx context.Context, r slog.Record) bool
handler slog.Handler
}
// Implements slog.Handler
func (h *RoutableHandler) Enabled(ctx context.Context, l slog.Level) bool {
return h.handler.Enabled(ctx, l)
}
// Implements slog.Handler
func (h *RoutableHandler) Handle(ctx context.Context, r slog.Record) error {
for _, matcher := range h.matchers {
if !matcher(ctx, r) {
return nil
}
}
return h.handler.Handle(ctx, r)
}
// Implements slog.Handler
func (h *RoutableHandler) WithAttrs(attrs []slog.Attr) slog.Handler {
return &RoutableHandler{
matchers: h.matchers,
handler: h.handler.WithAttrs(attrs),
}
}
// Implements slog.Handler
func (h *RoutableHandler) WithGroup(name string) slog.Handler {
// https://cs.opensource.google/go/x/exp/+/46b07846:slog/handler.go;l=247
if name == "" {
return h
}
return &RoutableHandler{
matchers: h.matchers,
handler: h.handler.WithGroup(name),
}
}