forked from influxdata/influxdb
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathhandler.go
189 lines (161 loc) · 4.67 KB
/
handler.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
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
package http
import (
"context"
"encoding/json"
"net/http"
_ "net/http/pprof" // used for debug pprof at the default path.
"github.com/go-chi/chi"
"github.com/influxdata/influxdb/v2/kit/prom"
kithttp "github.com/influxdata/influxdb/v2/kit/transport/http"
"github.com/prometheus/client_golang/prometheus"
"go.uber.org/zap"
)
const (
// MetricsPath exposes the prometheus metrics over /metrics.
MetricsPath = "/metrics"
// ReadyPath exposes the readiness of the service over /ready.
ReadyPath = "/ready"
// HealthPath exposes the health of the service over /health.
HealthPath = "/health"
// DebugPath exposes /debug/pprof for go debugging.
DebugPath = "/debug"
)
// Handler provides basic handling of metrics, health and debug endpoints.
// All other requests are passed down to the sub handler.
type Handler struct {
name string
r chi.Router
requests *prometheus.CounterVec
requestDur *prometheus.HistogramVec
// log logs all HTTP requests as they are served
log *zap.Logger
}
type (
handlerOpts struct {
log *zap.Logger
apiHandler http.Handler
debugHandler http.Handler
healthHandler http.Handler
metricsHandler http.Handler
readyHandler http.Handler
}
HandlerOptFn func(opts *handlerOpts)
)
func WithLog(l *zap.Logger) HandlerOptFn {
return func(opts *handlerOpts) {
opts.log = l
}
}
func WithAPIHandler(h http.Handler) HandlerOptFn {
return func(opts *handlerOpts) {
opts.apiHandler = h
}
}
func WithDebugHandler(h http.Handler) HandlerOptFn {
return func(opts *handlerOpts) {
opts.debugHandler = h
}
}
func WithHealthHandler(h http.Handler) HandlerOptFn {
return func(opts *handlerOpts) {
opts.healthHandler = h
}
}
func WithMetricsHandler(h http.Handler) HandlerOptFn {
return func(opts *handlerOpts) {
opts.metricsHandler = h
}
}
func WithReadyHandler(h http.Handler) HandlerOptFn {
return func(opts *handlerOpts) {
opts.readyHandler = h
}
}
// NewHandlerFromRegistry creates a new handler with the given name,
// and sets the /metrics endpoint to use the metrics from the given registry,
// after self-registering h's metrics.
func NewHandlerFromRegistry(name string, reg *prom.Registry, opts ...HandlerOptFn) *Handler {
opt := handlerOpts{
log: zap.NewNop(),
debugHandler: http.DefaultServeMux,
healthHandler: http.HandlerFunc(HealthHandler),
metricsHandler: reg.HTTPHandler(),
readyHandler: ReadyHandler(),
}
for _, o := range opts {
o(&opt)
}
h := &Handler{
name: name,
log: opt.log,
}
h.initMetrics()
r := chi.NewRouter()
// only gather metrics for system handlers
r.Group(func(r chi.Router) {
r.Use(
kithttp.Metrics(name, h.requests, h.requestDur),
)
{
r.Mount(MetricsPath, opt.metricsHandler)
r.Mount(ReadyPath, opt.readyHandler)
r.Mount(HealthPath, opt.healthHandler)
r.Mount(DebugPath, opt.debugHandler)
}
})
// gather metrics and traces for everything else
r.Group(func(r chi.Router) {
r.Use(
kithttp.Trace(name),
kithttp.Metrics(name, h.requests, h.requestDur),
)
{
r.Mount("/", opt.apiHandler)
}
})
h.r = r
reg.MustRegister(h.PrometheusCollectors()...)
return h
}
// ServeHTTP delegates a request to the appropriate subhandler.
func (h *Handler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
h.r.ServeHTTP(w, r)
}
// PrometheusCollectors satisifies prom.PrometheusCollector.
func (h *Handler) PrometheusCollectors() []prometheus.Collector {
return []prometheus.Collector{
h.requests,
h.requestDur,
}
}
func (h *Handler) initMetrics() {
const namespace = "http"
const handlerSubsystem = "api"
labelNames := []string{"handler", "method", "path", "status", "user_agent"}
h.requests = prometheus.NewCounterVec(prometheus.CounterOpts{
Namespace: namespace,
Subsystem: handlerSubsystem,
Name: "requests_total",
Help: "Number of http requests received",
}, labelNames)
h.requestDur = prometheus.NewHistogramVec(prometheus.HistogramOpts{
Namespace: namespace,
Subsystem: handlerSubsystem,
Name: "request_duration_seconds",
Help: "Time taken to respond to HTTP request",
}, labelNames)
}
func encodeResponse(ctx context.Context, w http.ResponseWriter, code int, res interface{}) error {
w.Header().Set("Content-Type", "application/json; charset=utf-8")
w.WriteHeader(code)
return json.NewEncoder(w).Encode(res)
}
func logEncodingError(log *zap.Logger, r *http.Request, err error) {
// If we encounter an error while encoding the response to an http request
// the best thing we can do is log that error, as we may have already written
// the headers for the http request in question.
log.Info("Error encoding response",
zap.String("path", r.URL.Path),
zap.String("method", r.Method),
zap.Error(err))
}