influxdb/http/handler.go

209 lines
6.1 KiB
Go

package http
import (
"context"
"encoding/json"
"errors"
"fmt"
"net/http"
_ "net/http/pprof" // used for debug pprof at the default path.
"strings"
"time"
"github.com/influxdata/platform/kit/prom"
opentracing "github.com/opentracing/opentracing-go"
"github.com/opentracing/opentracing-go/ext"
"github.com/opentracing/opentracing-go/log"
"github.com/prometheus/client_golang/prometheus"
"github.com/prometheus/client_golang/prometheus/promhttp"
"go.uber.org/zap"
)
const (
// MetricsPath exposes the prometheus metrics over /metrics.
MetricsPath = "/metrics"
// HealthPath exposes the health of the service over /health.
HealthPath = "/health"
// DebugPath exposes /debug/pprof for go debugging.
DebugPath = "/debug"
)
// Handler provides basic handling of metrics, health and debug endpoints.
// All other requests are passed down to the sub handler.
type Handler struct {
name string
// HealthHandler handles health requests
HealthHandler http.Handler
// MetricsHandler handles metrics requests
MetricsHandler http.Handler
// DebugHandler handles debug requests
DebugHandler http.Handler
// Handler handles all other requests
Handler http.Handler
requests *prometheus.CounterVec
requestDur *prometheus.HistogramVec
// Logger if set will log all HTTP requests as they are served
Logger *zap.Logger
// Tracer if set will be used to propagate traces through HTTP requests
Tracer opentracing.Tracer
}
// NewHandler creates a new handler with the given name.
// The name is used to tag the metrics produced by this handler.
//
// 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.
func NewHandler(name string) *Handler {
h := &Handler{
name: name,
MetricsHandler: promhttp.Handler(),
DebugHandler: http.DefaultServeMux,
}
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(),
DebugHandler: http.DefaultServeMux,
HealthHandler: http.HandlerFunc(HealthHandler),
}
h.initMetrics()
reg.MustRegister(h.PrometheusCollectors()...)
return h
}
// ServeHTTP delegates a request to the appropriate subhandler.
func (h *Handler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
// TODO: better way to do this?
statusW := newStatusResponseWriter(w)
w = statusW
if h.Tracer != nil {
// Extract opentracing Span
var serverSpan opentracing.Span
opName := fmt.Sprintf("%s:%s", h.name, r.URL.Path)
wireContext, _ := h.Tracer.Extract(
opentracing.HTTPHeaders,
opentracing.HTTPHeadersCarrier(r.Header),
)
// Create the span referring to the RPC client if available.
// If wireContext == nil, a root span will be created.
serverSpan = h.Tracer.StartSpan(
opName,
ext.RPCServerOption(wireContext),
)
serverSpan.LogFields(log.String("handler", h.name))
defer serverSpan.Finish()
r = r.WithContext(opentracing.ContextWithSpan(r.Context(), serverSpan))
}
// TODO: This could be problematic eventually. But for now it should be fine.
defer func(start time.Time) {
duration := time.Since(start)
statusClass := statusW.statusCodeClass()
statusCode := statusW.code()
h.requests.With(prometheus.Labels{
"handler": h.name,
"method": r.Method,
"path": r.URL.Path,
"status": statusClass,
}).Inc()
h.requestDur.With(prometheus.Labels{
"handler": h.name,
"method": r.Method,
"path": r.URL.Path,
"status": statusClass,
}).Observe(duration.Seconds())
if h.Logger != nil {
errField := zap.Skip()
if errStr := w.Header().Get(ErrorHeader); errStr != "" {
errField = zap.Error(errors.New(errStr))
}
errReferenceField := zap.Skip()
if errReference := w.Header().Get(ReferenceHeader); errReference != "" {
errReferenceField = zap.String("reference", errReference)
}
h.Logger.Info("Served http request",
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)),
errField,
errReferenceField,
)
}
}(time.Now())
switch {
case r.URL.Path == MetricsPath:
h.MetricsHandler.ServeHTTP(w, r)
case r.URL.Path == HealthPath:
h.HealthHandler.ServeHTTP(w, r)
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)
}
// PrometheusCollectors satisifies prom.PrometheusCollector.
func (h *Handler) PrometheusCollectors() []prometheus.Collector {
return []prometheus.Collector{
h.requests,
h.requestDur,
}
}
func (h *Handler) initMetrics() {
const namespace = "http"
const handlerSubsystem = "api"
h.requests = prometheus.NewCounterVec(prometheus.CounterOpts{
Namespace: namespace,
Subsystem: handlerSubsystem,
Name: "requests_total",
Help: "Number of http requests received",
}, []string{"handler", "method", "path", "status"})
h.requestDur = prometheus.NewHistogramVec(prometheus.HistogramOpts{
Namespace: namespace,
Subsystem: handlerSubsystem,
Name: "request_duration_seconds",
Help: "Time taken to respond to HTTP request",
// TODO(desa): determine what spacing these buckets should have.
Buckets: prometheus.ExponentialBuckets(0.001, 1.5, 25),
}, []string{"handler", "method", "path", "status"})
}
// InjectTrace writes any span from the request's context into the request headers.
func InjectTrace(r *http.Request) {
if span := opentracing.SpanFromContext(r.Context()); span != nil {
span.Tracer().Inject(
span.Context(),
opentracing.HTTPHeaders,
opentracing.HTTPHeadersCarrier(r.Header),
)
}
}