@@ -45,7 +45,10 @@ func InstrumentHandlerInFlight(g prometheus.Gauge, next http.Handler) http.Handl
45
45
// http.Handler to observe the request duration with the provided ObserverVec.
46
46
// The ObserverVec must have valid metric and label names and must have zero,
47
47
// one, or two non-const non-curried labels. For those, the only allowed label
48
- // names are "code" and "method". The function panics otherwise. The Observe
48
+ // names are "code" and "method". The function panics otherwise. For the "method"
49
+ // label a predefined default label value set is used to filter given values.
50
+ // Values besides predefined values will count as `unknown` method.
51
+ //`WithExtraMethods` can be used to add more methods to the set. The Observe
49
52
// method of the Observer in the ObserverVec is called with the request duration
50
53
// in seconds. Partitioning happens by HTTP status code and/or HTTP method if
51
54
// the respective instance label names are present in the ObserverVec. For
@@ -58,7 +61,12 @@ func InstrumentHandlerInFlight(g prometheus.Gauge, next http.Handler) http.Handl
58
61
//
59
62
// Note that this method is only guaranteed to never observe negative durations
60
63
// if used with Go1.9+.
61
- func InstrumentHandlerDuration (obs prometheus.ObserverVec , next http.Handler ) http.HandlerFunc {
64
+ func InstrumentHandlerDuration (obs prometheus.ObserverVec , next http.Handler , opts ... Option ) http.HandlerFunc {
65
+ mwOpts := & option {}
66
+ for _ , o := range opts {
67
+ o (mwOpts )
68
+ }
69
+
62
70
code , method := checkLabels (obs )
63
71
64
72
if code {
@@ -67,22 +75,25 @@ func InstrumentHandlerDuration(obs prometheus.ObserverVec, next http.Handler) ht
67
75
d := newDelegator (w , nil )
68
76
next .ServeHTTP (d , r )
69
77
70
- obs .With (labels (code , method , r .Method , d .Status ())).Observe (time .Since (now ).Seconds ())
78
+ obs .With (labels (code , method , r .Method , d .Status (), mwOpts . extraMethods ... )).Observe (time .Since (now ).Seconds ())
71
79
})
72
80
}
73
81
74
82
return http .HandlerFunc (func (w http.ResponseWriter , r * http.Request ) {
75
83
now := time .Now ()
76
84
next .ServeHTTP (w , r )
77
- obs .With (labels (code , method , r .Method , 0 )).Observe (time .Since (now ).Seconds ())
85
+ obs .With (labels (code , method , r .Method , 0 , mwOpts . extraMethods ... )).Observe (time .Since (now ).Seconds ())
78
86
})
79
87
}
80
88
81
89
// InstrumentHandlerCounter is a middleware that wraps the provided http.Handler
82
90
// to observe the request result with the provided CounterVec. The CounterVec
83
91
// must have valid metric and label names and must have zero, one, or two
84
92
// non-const non-curried labels. For those, the only allowed label names are
85
- // "code" and "method". The function panics otherwise. Partitioning of the
93
+ // "code" and "method". The function panics otherwise. For the "method"
94
+ // label a predefined default label value set is used to filter given values.
95
+ // Values besides predefined values will count as `unknown` method.
96
+ // `WithExtraMethods` can be used to add more methods to the set. Partitioning of the
86
97
// CounterVec happens by HTTP status code and/or HTTP method if the respective
87
98
// instance label names are present in the CounterVec. For unpartitioned
88
99
// counting, use a CounterVec with zero labels.
@@ -92,20 +103,25 @@ func InstrumentHandlerDuration(obs prometheus.ObserverVec, next http.Handler) ht
92
103
// If the wrapped Handler panics, the Counter is not incremented.
93
104
//
94
105
// See the example for InstrumentHandlerDuration for example usage.
95
- func InstrumentHandlerCounter (counter * prometheus.CounterVec , next http.Handler ) http.HandlerFunc {
106
+ func InstrumentHandlerCounter (counter * prometheus.CounterVec , next http.Handler , opts ... Option ) http.HandlerFunc {
107
+ mwOpts := & option {}
108
+ for _ , o := range opts {
109
+ o (mwOpts )
110
+ }
111
+
96
112
code , method := checkLabels (counter )
97
113
98
114
if code {
99
115
return http .HandlerFunc (func (w http.ResponseWriter , r * http.Request ) {
100
116
d := newDelegator (w , nil )
101
117
next .ServeHTTP (d , r )
102
- counter .With (labels (code , method , r .Method , d .Status ())).Inc ()
118
+ counter .With (labels (code , method , r .Method , d .Status (), mwOpts . extraMethods ... )).Inc ()
103
119
})
104
120
}
105
121
106
122
return http .HandlerFunc (func (w http.ResponseWriter , r * http.Request ) {
107
123
next .ServeHTTP (w , r )
108
- counter .With (labels (code , method , r .Method , 0 )).Inc ()
124
+ counter .With (labels (code , method , r .Method , 0 , mwOpts . extraMethods ... )).Inc ()
109
125
})
110
126
}
111
127
@@ -114,7 +130,10 @@ func InstrumentHandlerCounter(counter *prometheus.CounterVec, next http.Handler)
114
130
// until the response headers are written. The ObserverVec must have valid
115
131
// metric and label names and must have zero, one, or two non-const non-curried
116
132
// labels. For those, the only allowed label names are "code" and "method". The
117
- // function panics otherwise. The Observe method of the Observer in the
133
+ // function panics otherwise. For the "method" label a predefined default label
134
+ // value set is used to filter given values. Values besides predefined values
135
+ // will count as `unknown` method.`WithExtraMethods` can be used to add more
136
+ // methods to the set. The Observe method of the Observer in the
118
137
// ObserverVec is called with the request duration in seconds. Partitioning
119
138
// happens by HTTP status code and/or HTTP method if the respective instance
120
139
// label names are present in the ObserverVec. For unpartitioned observations,
@@ -128,13 +147,18 @@ func InstrumentHandlerCounter(counter *prometheus.CounterVec, next http.Handler)
128
147
// if used with Go1.9+.
129
148
//
130
149
// See the example for InstrumentHandlerDuration for example usage.
131
- func InstrumentHandlerTimeToWriteHeader (obs prometheus.ObserverVec , next http.Handler ) http.HandlerFunc {
150
+ func InstrumentHandlerTimeToWriteHeader (obs prometheus.ObserverVec , next http.Handler , opts ... Option ) http.HandlerFunc {
151
+ mwOpts := & option {}
152
+ for _ , o := range opts {
153
+ o (mwOpts )
154
+ }
155
+
132
156
code , method := checkLabels (obs )
133
157
134
158
return http .HandlerFunc (func (w http.ResponseWriter , r * http.Request ) {
135
159
now := time .Now ()
136
160
d := newDelegator (w , func (status int ) {
137
- obs .With (labels (code , method , r .Method , status )).Observe (time .Since (now ).Seconds ())
161
+ obs .With (labels (code , method , r .Method , status , mwOpts . extraMethods ... )).Observe (time .Since (now ).Seconds ())
138
162
})
139
163
next .ServeHTTP (d , r )
140
164
})
@@ -144,8 +168,11 @@ func InstrumentHandlerTimeToWriteHeader(obs prometheus.ObserverVec, next http.Ha
144
168
// http.Handler to observe the request size with the provided ObserverVec. The
145
169
// ObserverVec must have valid metric and label names and must have zero, one,
146
170
// or two non-const non-curried labels. For those, the only allowed label names
147
- // are "code" and "method". The function panics otherwise. The Observe method of
148
- // the Observer in the ObserverVec is called with the request size in
171
+ // are "code" and "method". The function panics otherwise. For the "method"
172
+ // label a predefined default label value set is used to filter given values.
173
+ // Values besides predefined values will count as `unknown` method.
174
+ // `WithExtraMethods` can be used to add more methods to the set. The Observe
175
+ // method of the Observer in the ObserverVec is called with the request size in
149
176
// bytes. Partitioning happens by HTTP status code and/or HTTP method if the
150
177
// respective instance label names are present in the ObserverVec. For
151
178
// unpartitioned observations, use an ObserverVec with zero labels. Note that
@@ -156,31 +183,39 @@ func InstrumentHandlerTimeToWriteHeader(obs prometheus.ObserverVec, next http.Ha
156
183
// If the wrapped Handler panics, no values are reported.
157
184
//
158
185
// See the example for InstrumentHandlerDuration for example usage.
159
- func InstrumentHandlerRequestSize (obs prometheus.ObserverVec , next http.Handler ) http.HandlerFunc {
186
+ func InstrumentHandlerRequestSize (obs prometheus.ObserverVec , next http.Handler , opts ... Option ) http.HandlerFunc {
187
+ mwOpts := & option {}
188
+ for _ , o := range opts {
189
+ o (mwOpts )
190
+ }
191
+
160
192
code , method := checkLabels (obs )
161
193
162
194
if code {
163
195
return http .HandlerFunc (func (w http.ResponseWriter , r * http.Request ) {
164
196
d := newDelegator (w , nil )
165
197
next .ServeHTTP (d , r )
166
198
size := computeApproximateRequestSize (r )
167
- obs .With (labels (code , method , r .Method , d .Status ())).Observe (float64 (size ))
199
+ obs .With (labels (code , method , r .Method , d .Status (), mwOpts . extraMethods ... )).Observe (float64 (size ))
168
200
})
169
201
}
170
202
171
203
return http .HandlerFunc (func (w http.ResponseWriter , r * http.Request ) {
172
204
next .ServeHTTP (w , r )
173
205
size := computeApproximateRequestSize (r )
174
- obs .With (labels (code , method , r .Method , 0 )).Observe (float64 (size ))
206
+ obs .With (labels (code , method , r .Method , 0 , mwOpts . extraMethods ... )).Observe (float64 (size ))
175
207
})
176
208
}
177
209
178
210
// InstrumentHandlerResponseSize is a middleware that wraps the provided
179
211
// http.Handler to observe the response size with the provided ObserverVec. The
180
212
// ObserverVec must have valid metric and label names and must have zero, one,
181
213
// or two non-const non-curried labels. For those, the only allowed label names
182
- // are "code" and "method". The function panics otherwise. The Observe method of
183
- // the Observer in the ObserverVec is called with the response size in
214
+ // are "code" and "method". The function panics otherwise. For the "method"
215
+ // label a predefined default label value set is used to filter given values.
216
+ // Values besides predefined values will count as `unknown` method.
217
+ // `WithExtraMethods` can be used to add more methods to the set. The Observe
218
+ // method of the Observer in the ObserverVec is called with the response size in
184
219
// bytes. Partitioning happens by HTTP status code and/or HTTP method if the
185
220
// respective instance label names are present in the ObserverVec. For
186
221
// unpartitioned observations, use an ObserverVec with zero labels. Note that
@@ -191,12 +226,18 @@ func InstrumentHandlerRequestSize(obs prometheus.ObserverVec, next http.Handler)
191
226
// If the wrapped Handler panics, no values are reported.
192
227
//
193
228
// See the example for InstrumentHandlerDuration for example usage.
194
- func InstrumentHandlerResponseSize (obs prometheus.ObserverVec , next http.Handler ) http.Handler {
229
+ func InstrumentHandlerResponseSize (obs prometheus.ObserverVec , next http.Handler , opts ... Option ) http.Handler {
230
+ mwOpts := & option {}
231
+ for _ , o := range opts {
232
+ o (mwOpts )
233
+ }
234
+
195
235
code , method := checkLabels (obs )
236
+
196
237
return http .HandlerFunc (func (w http.ResponseWriter , r * http.Request ) {
197
238
d := newDelegator (w , nil )
198
239
next .ServeHTTP (d , r )
199
- obs .With (labels (code , method , r .Method , d .Status ())).Observe (float64 (d .Written ()))
240
+ obs .With (labels (code , method , r .Method , d .Status (), mwOpts . extraMethods ... )).Observe (float64 (d .Written ()))
200
241
})
201
242
}
202
243
@@ -290,7 +331,7 @@ func isLabelCurried(c prometheus.Collector, label string) bool {
290
331
// unnecessary allocations on each request.
291
332
var emptyLabels = prometheus.Labels {}
292
333
293
- func labels (code , method bool , reqMethod string , status int ) prometheus.Labels {
334
+ func labels (code , method bool , reqMethod string , status int , extraMethods ... string ) prometheus.Labels {
294
335
if ! (code || method ) {
295
336
return emptyLabels
296
337
}
@@ -300,7 +341,7 @@ func labels(code, method bool, reqMethod string, status int) prometheus.Labels {
300
341
labels ["code" ] = sanitizeCode (status )
301
342
}
302
343
if method {
303
- labels ["method" ] = sanitizeMethod (reqMethod )
344
+ labels ["method" ] = sanitizeMethod (reqMethod , extraMethods ... )
304
345
}
305
346
306
347
return labels
@@ -330,7 +371,12 @@ func computeApproximateRequestSize(r *http.Request) int {
330
371
return s
331
372
}
332
373
333
- func sanitizeMethod (m string ) string {
374
+ // If the wrapped http.Handler has a known method, it will be sanitized and returned.
375
+ // Otherwise, "unknown" will be returned. The known method list can be extended
376
+ // as needed by using extraMethods parameter.
377
+ func sanitizeMethod (m string , extraMethods ... string ) string {
378
+ // See https://developer.mozilla.org/en-US/docs/Web/HTTP/Methods for
379
+ // the methods chosen as default.
334
380
switch m {
335
381
case "GET" , "get" :
336
382
return "get"
@@ -348,15 +394,25 @@ func sanitizeMethod(m string) string {
348
394
return "options"
349
395
case "NOTIFY" , "notify" :
350
396
return "notify"
397
+ case "TRACE" , "trace" :
398
+ return "trace"
399
+ case "PATCH" , "patch" :
400
+ return "patch"
351
401
default :
352
- return strings .ToLower (m )
402
+ for _ , method := range extraMethods {
403
+ if strings .EqualFold (m , method ) {
404
+ return strings .ToLower (m )
405
+ }
406
+ }
407
+ return "unknown"
353
408
}
354
409
}
355
410
356
411
// If the wrapped http.Handler has not set a status code, i.e. the value is
357
- // currently 0, santizeCode will return 200, for consistency with behavior in
412
+ // currently 0, sanitizeCode will return 200, for consistency with behavior in
358
413
// the stdlib.
359
414
func sanitizeCode (s int ) string {
415
+ // See for accepted codes https://www.iana.org/assignments/http-status-codes/http-status-codes.xhtml
360
416
switch s {
361
417
case 100 :
362
418
return "100"
@@ -453,6 +509,9 @@ func sanitizeCode(s int) string {
453
509
return "511"
454
510
455
511
default :
456
- return strconv .Itoa (s )
512
+ if s >= 100 && s <= 599 {
513
+ return strconv .Itoa (s )
514
+ }
515
+ return "unknown"
457
516
}
458
517
}
0 commit comments