2017-01-12 14:34:54 +01:00
|
|
|
package middlewares
|
|
|
|
|
|
|
|
import (
|
|
|
|
"net/http"
|
|
|
|
"strconv"
|
|
|
|
"time"
|
2017-04-18 08:22:06 +02:00
|
|
|
|
|
|
|
"github.com/go-kit/kit/metrics"
|
2017-01-12 14:34:54 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
// Metrics is an Interface that must be satisfied by any system that
|
2017-04-18 08:22:06 +02:00
|
|
|
// wants to expose and monitor Metrics.
|
2017-01-12 14:34:54 +01:00
|
|
|
type Metrics interface {
|
|
|
|
getReqsCounter() metrics.Counter
|
2017-04-18 08:22:06 +02:00
|
|
|
getReqDurationHistogram() metrics.Histogram
|
|
|
|
RetryMetrics
|
|
|
|
}
|
|
|
|
|
|
|
|
// RetryMetrics must be satisfied by any system that wants to collect and
|
|
|
|
// expose retry specific Metrics.
|
|
|
|
type RetryMetrics interface {
|
|
|
|
getRetryCounter() metrics.Counter
|
2017-01-12 14:34:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// MetricsWrapper is a Negroni compatible Handler which relies on a
|
2017-04-18 08:22:06 +02:00
|
|
|
// given Metrics implementation to expose and monitor Traefik Metrics.
|
2017-01-12 14:34:54 +01:00
|
|
|
type MetricsWrapper struct {
|
|
|
|
Impl Metrics
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewMetricsWrapper return a MetricsWrapper struct with
|
|
|
|
// a given Metrics implementation e.g Prometheuss
|
|
|
|
func NewMetricsWrapper(impl Metrics) *MetricsWrapper {
|
|
|
|
var metricsWrapper = MetricsWrapper{
|
|
|
|
Impl: impl,
|
|
|
|
}
|
|
|
|
|
|
|
|
return &metricsWrapper
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *MetricsWrapper) ServeHTTP(rw http.ResponseWriter, r *http.Request, next http.HandlerFunc) {
|
|
|
|
start := time.Now()
|
|
|
|
prw := &responseRecorder{rw, http.StatusOK}
|
|
|
|
next(prw, r)
|
2017-04-18 08:22:06 +02:00
|
|
|
labels := []string{"code", strconv.Itoa(prw.statusCode), "method", r.Method}
|
2017-01-12 14:34:54 +01:00
|
|
|
m.Impl.getReqsCounter().With(labels...).Add(1)
|
2017-04-18 08:22:06 +02:00
|
|
|
m.Impl.getReqDurationHistogram().Observe(float64(time.Since(start).Seconds()))
|
|
|
|
}
|
|
|
|
|
|
|
|
// MetricsRetryListener is an implementation of the RetryListener interface to
|
|
|
|
// record Metrics about retry attempts.
|
|
|
|
type MetricsRetryListener struct {
|
|
|
|
retryMetrics RetryMetrics
|
2017-01-12 14:34:54 +01:00
|
|
|
}
|
|
|
|
|
2017-04-18 08:22:06 +02:00
|
|
|
// Retried tracks the retry in the Metrics implementation.
|
|
|
|
func (m *MetricsRetryListener) Retried(attempt int) {
|
|
|
|
if m.retryMetrics != nil {
|
|
|
|
m.retryMetrics.getRetryCounter().Add(1)
|
|
|
|
}
|
2017-01-12 14:34:54 +01:00
|
|
|
}
|
|
|
|
|
2017-04-18 08:22:06 +02:00
|
|
|
// NewMetricsRetryListener instantiates a MetricsRetryListener with the given RetryMetrics.
|
|
|
|
func NewMetricsRetryListener(retryMetrics RetryMetrics) RetryListener {
|
|
|
|
return &MetricsRetryListener{retryMetrics: retryMetrics}
|
2017-01-12 14:34:54 +01:00
|
|
|
}
|