diff --git a/web/web.go b/web/web.go index 848bf82e6..05b227cfb 100644 --- a/web/web.go +++ b/web/web.go @@ -69,6 +69,25 @@ import ( var localhostRepresentations = []string{"127.0.0.1", "localhost"} +// withStackTrace logs the stack trace in case the request panics. The function +// will re-raise the error which will then be handled by the net/http package. +// It is needed because the go-kit log package doesn't manage properly the +// panics from net/http (see https://github.com/go-kit/kit/issues/233). +func withStackTracer(h http.Handler, l log.Logger) http.Handler { + return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + defer func() { + if err := recover(); err != nil { + const size = 64 << 10 + buf := make([]byte, size) + buf = buf[:runtime.Stack(buf, false)] + level.Error(l).Log("msg", "panic while serving request", "client", r.RemoteAddr, "url", r.URL, "err", err, "stack", buf) + panic(err) + } + }() + h.ServeHTTP(w, r) + }) +} + var ( requestDuration = prometheus.NewHistogramVec( prometheus.HistogramOpts{ @@ -459,7 +478,7 @@ func (h *Handler) Run(ctx context.Context) error { errlog := stdlog.New(log.NewStdlibAdapter(level.Error(h.logger)), "", 0) httpSrv := &http.Server{ - Handler: nethttp.Middleware(opentracing.GlobalTracer(), mux, operationName), + Handler: withStackTracer(nethttp.Middleware(opentracing.GlobalTracer(), mux, operationName), h.logger), ErrorLog: errlog, ReadTimeout: h.options.ReadTimeout, }