mirror of
https://github.com/weaveworks/scope.git
synced 2026-03-06 03:31:00 +00:00
46 lines
1.1 KiB
Go
46 lines
1.1 KiB
Go
package middleware
|
|
|
|
import (
|
|
"net/http"
|
|
"strconv"
|
|
"time"
|
|
|
|
"github.com/gorilla/mux"
|
|
"github.com/prometheus/client_golang/prometheus"
|
|
)
|
|
|
|
// Instrument is a Middleware which records timings for every HTTP request
|
|
type Instrument struct {
|
|
RouteMatcher interface {
|
|
Match(*http.Request, *mux.RouteMatch) bool
|
|
}
|
|
Duration *prometheus.SummaryVec
|
|
}
|
|
|
|
// Wrap implements middleware.Interface
|
|
func (i Instrument) Wrap(next http.Handler) http.Handler {
|
|
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
begin := time.Now()
|
|
interceptor := &interceptor{ResponseWriter: w, statusCode: http.StatusOK}
|
|
next.ServeHTTP(interceptor, r)
|
|
var (
|
|
route = i.getRouteName(r)
|
|
status = strconv.Itoa(interceptor.statusCode)
|
|
took = time.Since(begin)
|
|
)
|
|
i.Duration.WithLabelValues(r.Method, route, status).Observe(float64(took.Nanoseconds()))
|
|
})
|
|
}
|
|
|
|
func (i Instrument) getRouteName(r *http.Request) string {
|
|
var routeMatch mux.RouteMatch
|
|
if !i.RouteMatcher.Match(r, &routeMatch) {
|
|
return "unmatched_path"
|
|
}
|
|
name := routeMatch.Route.GetName()
|
|
if name == "" {
|
|
return "unnamed_path"
|
|
}
|
|
return name
|
|
}
|