2018-05-14 16:26:38 +00:00
|
|
|
package http
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"encoding/json"
|
2018-07-31 20:34:30 +00:00
|
|
|
"errors"
|
2018-05-14 16:26:38 +00:00
|
|
|
"net/http"
|
2018-09-04 21:58:37 +00:00
|
|
|
_ "net/http/pprof" // used for debug pprof at the default path.
|
2018-05-14 16:26:38 +00:00
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
|
2019-01-08 00:37:16 +00:00
|
|
|
"github.com/influxdata/influxdb/kit/prom"
|
2019-04-11 02:28:21 +00:00
|
|
|
"github.com/influxdata/influxdb/kit/tracing"
|
|
|
|
"github.com/opentracing/opentracing-go"
|
2018-05-14 16:26:38 +00:00
|
|
|
"github.com/prometheus/client_golang/prometheus"
|
|
|
|
"github.com/prometheus/client_golang/prometheus/promhttp"
|
2018-07-30 22:16:37 +00:00
|
|
|
"go.uber.org/zap"
|
2018-05-14 16:26:38 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
2018-09-04 21:58:37 +00:00
|
|
|
// MetricsPath exposes the prometheus metrics over /metrics.
|
2018-05-14 16:26:38 +00:00
|
|
|
MetricsPath = "/metrics"
|
2018-11-27 22:29:59 +00:00
|
|
|
// ReadyPath exposes the readiness of the service over /ready.
|
|
|
|
ReadyPath = "/ready"
|
2018-10-04 18:21:53 +00:00
|
|
|
// HealthPath exposes the health of the service over /health.
|
|
|
|
HealthPath = "/health"
|
2018-09-04 21:58:37 +00:00
|
|
|
// DebugPath exposes /debug/pprof for go debugging.
|
|
|
|
DebugPath = "/debug"
|
2018-05-14 16:26:38 +00:00
|
|
|
)
|
|
|
|
|
2018-10-04 18:21:53 +00:00
|
|
|
// Handler provides basic handling of metrics, health and debug endpoints.
|
2018-05-14 16:26:38 +00:00
|
|
|
// All other requests are passed down to the sub handler.
|
|
|
|
type Handler struct {
|
|
|
|
name string
|
|
|
|
// MetricsHandler handles metrics requests
|
|
|
|
MetricsHandler http.Handler
|
2018-11-27 22:29:59 +00:00
|
|
|
// ReadyHandler handles readiness checks
|
|
|
|
ReadyHandler http.Handler
|
|
|
|
// HealthHandler handles health requests
|
|
|
|
HealthHandler http.Handler
|
2018-05-14 16:26:38 +00:00
|
|
|
// DebugHandler handles debug requests
|
|
|
|
DebugHandler http.Handler
|
|
|
|
// Handler handles all other requests
|
|
|
|
Handler http.Handler
|
2018-07-11 23:24:16 +00:00
|
|
|
|
|
|
|
requests *prometheus.CounterVec
|
|
|
|
requestDur *prometheus.HistogramVec
|
2018-07-30 22:16:37 +00:00
|
|
|
|
|
|
|
// Logger if set will log all HTTP requests as they are served
|
|
|
|
Logger *zap.Logger
|
2018-05-14 16:26:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewHandler creates a new handler with the given name.
|
|
|
|
// The name is used to tag the metrics produced by this handler.
|
2018-07-11 23:24:16 +00:00
|
|
|
//
|
|
|
|
// The MetricsHandler is set to the default prometheus handler.
|
|
|
|
// It is the caller's responsibility to call prometheus.MustRegister(h.PrometheusCollectors()...).
|
|
|
|
// In most cases, you want to use NewHandlerFromRegistry instead.
|
2018-05-14 16:26:38 +00:00
|
|
|
func NewHandler(name string) *Handler {
|
2018-07-11 23:24:16 +00:00
|
|
|
h := &Handler{
|
2018-05-14 16:26:38 +00:00
|
|
|
name: name,
|
|
|
|
MetricsHandler: promhttp.Handler(),
|
|
|
|
DebugHandler: http.DefaultServeMux,
|
|
|
|
}
|
2018-07-11 23:24:16 +00:00
|
|
|
h.initMetrics()
|
|
|
|
return 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) *Handler {
|
|
|
|
h := &Handler{
|
|
|
|
name: name,
|
|
|
|
MetricsHandler: reg.HTTPHandler(),
|
2018-11-27 22:29:59 +00:00
|
|
|
ReadyHandler: http.HandlerFunc(ReadyHandler),
|
2018-10-04 18:21:53 +00:00
|
|
|
HealthHandler: http.HandlerFunc(HealthHandler),
|
2018-11-27 22:29:59 +00:00
|
|
|
DebugHandler: http.DefaultServeMux,
|
2018-07-11 23:24:16 +00:00
|
|
|
}
|
|
|
|
h.initMetrics()
|
|
|
|
reg.MustRegister(h.PrometheusCollectors()...)
|
|
|
|
return h
|
2018-05-14 16:26:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// ServeHTTP delegates a request to the appropriate subhandler.
|
|
|
|
func (h *Handler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
2019-04-11 02:28:21 +00:00
|
|
|
var span opentracing.Span
|
|
|
|
span, r = tracing.ExtractFromHTTPRequest(r, h.name)
|
2019-05-02 18:58:13 +00:00
|
|
|
userAgent := r.Header.Get("User-Agent")
|
|
|
|
if userAgent == "" {
|
|
|
|
userAgent = "unknown"
|
|
|
|
}
|
|
|
|
|
2019-04-11 02:28:21 +00:00
|
|
|
defer span.Finish()
|
|
|
|
|
2018-05-14 16:26:38 +00:00
|
|
|
// TODO: better way to do this?
|
|
|
|
statusW := newStatusResponseWriter(w)
|
|
|
|
w = statusW
|
|
|
|
|
|
|
|
// TODO: This could be problematic eventually. But for now it should be fine.
|
2018-07-30 22:16:37 +00:00
|
|
|
defer func(start time.Time) {
|
|
|
|
duration := time.Since(start)
|
|
|
|
statusClass := statusW.statusCodeClass()
|
|
|
|
statusCode := statusW.code()
|
2018-07-11 23:24:16 +00:00
|
|
|
h.requests.With(prometheus.Labels{
|
2019-05-02 18:58:13 +00:00
|
|
|
"handler": h.name,
|
|
|
|
"method": r.Method,
|
|
|
|
"path": r.URL.Path,
|
|
|
|
"status": statusClass,
|
|
|
|
"user_agent": userAgent,
|
2018-05-14 16:26:38 +00:00
|
|
|
}).Inc()
|
2018-07-11 23:24:16 +00:00
|
|
|
h.requestDur.With(prometheus.Labels{
|
2019-05-02 18:58:13 +00:00
|
|
|
"handler": h.name,
|
|
|
|
"method": r.Method,
|
|
|
|
"path": r.URL.Path,
|
|
|
|
"status": statusClass,
|
|
|
|
"user_agent": userAgent,
|
2018-07-30 22:16:37 +00:00
|
|
|
}).Observe(duration.Seconds())
|
|
|
|
if h.Logger != nil {
|
2018-08-01 14:47:58 +00:00
|
|
|
errField := zap.Skip()
|
2019-01-24 00:15:42 +00:00
|
|
|
if errStr := w.Header().Get(PlatformErrorCodeHeader); errStr != "" {
|
2018-08-01 14:47:58 +00:00
|
|
|
errField = zap.Error(errors.New(errStr))
|
|
|
|
}
|
|
|
|
errReferenceField := zap.Skip()
|
2019-01-24 00:15:42 +00:00
|
|
|
if errReference := w.Header().Get(PlatformErrorCodeHeader); errReference != "" {
|
|
|
|
errReferenceField = zap.String("error_code", PlatformErrorCodeHeader)
|
2018-08-01 14:47:58 +00:00
|
|
|
}
|
|
|
|
|
2018-10-30 07:27:51 +00:00
|
|
|
h.Logger.Debug("Request",
|
2018-07-30 22:16:37 +00:00
|
|
|
zap.String("handler", h.name),
|
|
|
|
zap.String("method", r.Method),
|
|
|
|
zap.String("path", r.URL.Path),
|
|
|
|
zap.Int("status", statusCode),
|
|
|
|
zap.Int("duration_ns", int(duration)),
|
2018-08-01 14:47:58 +00:00
|
|
|
errField,
|
|
|
|
errReferenceField,
|
2018-07-30 22:16:37 +00:00
|
|
|
)
|
|
|
|
}
|
2018-05-14 16:26:38 +00:00
|
|
|
}(time.Now())
|
|
|
|
|
|
|
|
switch {
|
|
|
|
case r.URL.Path == MetricsPath:
|
|
|
|
h.MetricsHandler.ServeHTTP(w, r)
|
2018-11-27 22:29:59 +00:00
|
|
|
case r.URL.Path == ReadyPath:
|
|
|
|
h.ReadyHandler.ServeHTTP(w, r)
|
2018-10-04 18:21:53 +00:00
|
|
|
case r.URL.Path == HealthPath:
|
|
|
|
h.HealthHandler.ServeHTTP(w, r)
|
2018-05-14 16:26:38 +00:00
|
|
|
case strings.HasPrefix(r.URL.Path, DebugPath):
|
|
|
|
h.DebugHandler.ServeHTTP(w, r)
|
|
|
|
default:
|
|
|
|
h.Handler.ServeHTTP(w, r)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
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)
|
|
|
|
}
|
|
|
|
|
2018-07-11 23:24:16 +00:00
|
|
|
// PrometheusCollectors satisifies prom.PrometheusCollector.
|
|
|
|
func (h *Handler) PrometheusCollectors() []prometheus.Collector {
|
|
|
|
return []prometheus.Collector{
|
|
|
|
h.requests,
|
|
|
|
h.requestDur,
|
|
|
|
}
|
2018-05-14 16:26:38 +00:00
|
|
|
}
|
|
|
|
|
2018-07-11 23:24:16 +00:00
|
|
|
func (h *Handler) initMetrics() {
|
|
|
|
const namespace = "http"
|
|
|
|
const handlerSubsystem = "api"
|
2018-05-14 16:26:38 +00:00
|
|
|
|
2018-07-11 23:24:16 +00:00
|
|
|
h.requests = prometheus.NewCounterVec(prometheus.CounterOpts{
|
2018-05-14 16:26:38 +00:00
|
|
|
Namespace: namespace,
|
|
|
|
Subsystem: handlerSubsystem,
|
|
|
|
Name: "requests_total",
|
|
|
|
Help: "Number of http requests received",
|
2019-05-02 18:58:13 +00:00
|
|
|
}, []string{"handler", "method", "path", "status", "user_agent"})
|
2018-05-14 16:26:38 +00:00
|
|
|
|
2018-07-11 23:24:16 +00:00
|
|
|
h.requestDur = prometheus.NewHistogramVec(prometheus.HistogramOpts{
|
2018-05-14 16:26:38 +00:00
|
|
|
Namespace: namespace,
|
|
|
|
Subsystem: handlerSubsystem,
|
|
|
|
Name: "request_duration_seconds",
|
|
|
|
Help: "Time taken to respond to HTTP request",
|
2019-05-02 18:58:13 +00:00
|
|
|
}, []string{"handler", "method", "path", "status", "user_agent"})
|
2018-07-11 23:24:16 +00:00
|
|
|
}
|
2018-08-15 20:14:51 +00:00
|
|
|
|
2018-12-20 16:07:46 +00:00
|
|
|
func logEncodingError(logger *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.
|
|
|
|
logger.Info("error encoding response",
|
|
|
|
zap.String("path", r.URL.Path),
|
|
|
|
zap.String("method", r.Method),
|
|
|
|
zap.Error(err))
|
|
|
|
}
|