-
Notifications
You must be signed in to change notification settings - Fork 21
/
Copy pathmetrics.go
142 lines (121 loc) · 3.86 KB
/
metrics.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
package transport
import (
"fmt"
"net/http"
"strconv"
"github.com/ankorstore/yokai/httpclient/normalization"
"github.com/prometheus/client_golang/prometheus"
)
const (
HttpClientMetricsRequestsCount = "http_client_requests_total"
HttpClientMetricsRequestsDuration = "http_client_requests_duration_seconds"
)
// MetricsTransport is a wrapper around [http.RoundTripper] with some [MetricsTransportConfig] configuration.
type MetricsTransport struct {
transport http.RoundTripper
config *MetricsTransportConfig
requestsCounter *prometheus.CounterVec
requestsDuration *prometheus.HistogramVec
}
// MetricsTransportConfig is the configuration of the [MetricsTransport].
type MetricsTransportConfig struct {
Registry prometheus.Registerer
Namespace string
Subsystem string
Buckets []float64
NormalizeRequestPath bool
NormalizeRequestPathMasks map[string]string
NormalizeResponseStatus bool
}
// NewMetricsTransport returns a [MetricsTransport] instance with default [MetricsTransportConfig] configuration.
func NewMetricsTransport(base http.RoundTripper) *MetricsTransport {
return NewMetricsTransportWithConfig(
base,
&MetricsTransportConfig{
Registry: prometheus.DefaultRegisterer,
Namespace: "",
Subsystem: "",
Buckets: prometheus.DefBuckets,
NormalizeRequestPath: false,
NormalizeRequestPathMasks: map[string]string{},
NormalizeResponseStatus: true,
},
)
}
// NewMetricsTransportWithConfig returns a [MetricsTransport] instance for a provided [MetricsTransportConfig] configuration.
func NewMetricsTransportWithConfig(base http.RoundTripper, config *MetricsTransportConfig) *MetricsTransport {
if base == nil {
base = NewBaseTransport()
}
if config.Registry == nil {
config.Registry = prometheus.DefaultRegisterer
}
requestsCounter := prometheus.NewCounterVec(
prometheus.CounterOpts{
Namespace: config.Namespace,
Subsystem: config.Subsystem,
Name: HttpClientMetricsRequestsCount,
Help: "Number of performed HTTP requests",
},
[]string{
"status",
"method",
"host",
"path",
},
)
requestsDuration := prometheus.NewHistogramVec(
prometheus.HistogramOpts{
Namespace: config.Namespace,
Subsystem: config.Subsystem,
Name: HttpClientMetricsRequestsDuration,
Help: "Time spent performing HTTP requests",
Buckets: config.Buckets,
},
[]string{
"method",
"host",
"path",
},
)
config.Registry.MustRegister(requestsCounter, requestsDuration)
return &MetricsTransport{
transport: base,
config: config,
requestsCounter: requestsCounter,
requestsDuration: requestsDuration,
}
}
// Base returns the wrapped [http.RoundTripper].
func (t *MetricsTransport) Base() http.RoundTripper {
return t.transport
}
// RoundTrip performs a request / response round trip, based on the wrapped [http.RoundTripper].
func (t *MetricsTransport) RoundTrip(req *http.Request) (*http.Response, error) {
host := req.URL.Host
if req.URL.Scheme != "" {
host = fmt.Sprintf("%s://%s", req.URL.Scheme, host)
}
path := req.URL.Path
if req.URL.RawQuery != "" {
path = fmt.Sprintf("%s?%s", path, req.URL.RawQuery)
}
if t.config.NormalizeRequestPath {
path = normalization.NormalizePath(t.config.NormalizeRequestPathMasks, path)
}
timer := prometheus.NewTimer(t.requestsDuration.WithLabelValues(req.Method, host, path))
resp, err := t.transport.RoundTrip(req)
timer.ObserveDuration()
respStatus := ""
if err != nil {
respStatus = "error"
} else {
if t.config.NormalizeResponseStatus {
respStatus = normalization.NormalizeStatus(resp.StatusCode)
} else {
respStatus = strconv.Itoa(resp.StatusCode)
}
}
t.requestsCounter.WithLabelValues(respStatus, req.Method, host, path).Inc()
return resp, err
}