module/apmchiv5/middleware.go (77 lines of code) (raw):
// Licensed to Elasticsearch B.V. under one or more contributor
// license agreements. See the NOTICE file distributed with
// this work for additional information regarding copyright
// ownership. Elasticsearch B.V. licenses this file to you under
// the Apache License, Version 2.0 (the "License"); you may
// not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing,
// software distributed under the License is distributed on an
// "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
// KIND, either express or implied. See the License for the
// specific language governing permissions and limitations
// under the License.
package apmchiv5 // import "go.elastic.co/apm/module/apmchiv5/v2"
import (
"net/http"
"github.com/go-chi/chi/v5"
"go.elastic.co/apm/module/apmhttp/v2"
"go.elastic.co/apm/v2"
)
// Middleware returns a new chi middleware handler
// for tracing requests and reporting errors.
//
// The server request name will use the fully matched,
// parametrized route.
//
// By default, the middleware will use apm.DefaultTracer().
// Use WithTracer to specify an alternative tracer.
func Middleware(o ...Option) func(http.Handler) http.Handler {
opts := options{
tracer: apm.DefaultTracer(),
}
for _, o := range o {
o(&opts)
}
if opts.requestIgnorer == nil {
opts.requestIgnorer = apmhttp.NewDynamicServerRequestIgnorer(opts.tracer)
}
return func(h http.Handler) http.Handler {
serverOpts := []apmhttp.ServerOption{
apmhttp.WithTracer(opts.tracer),
apmhttp.WithServerRequestName(routeRequestName),
apmhttp.WithServerRequestIgnorer(opts.requestIgnorer),
}
if opts.panicPropagation {
serverOpts = append(serverOpts, apmhttp.WithPanicPropagation())
}
return apmhttp.Wrap(
h,
serverOpts...,
)
}
}
func routeRequestName(r *http.Request) string {
if routePattern, ok := getRoutePattern(r); ok {
return r.Method + " " + routePattern
}
return apmhttp.UnknownRouteRequestName(r)
}
func getRoutePattern(r *http.Request) (string, bool) {
routePath := r.URL.Path
if r.URL.RawPath != "" {
routePath = r.URL.RawPath
}
rctx := chi.RouteContext(r.Context())
tctx := chi.NewRouteContext()
if rctx.Routes.Match(tctx, r.Method, routePath) {
return tctx.RoutePattern(), true
}
return "", false
}
type options struct {
tracer *apm.Tracer
requestIgnorer apmhttp.RequestIgnorerFunc
panicPropagation bool
}
// Option sets options for tracing.
type Option func(*options)
// WithTracer returns an Option which sets t as the tracer
// to use for tracing server requests.
func WithTracer(t *apm.Tracer) Option {
if t == nil {
panic("t == nil")
}
return func(o *options) {
o.tracer = t
}
}
// WithPanicPropagation returns an Option which enable panic propagation.
// Any panic will be recovered and recorded as an error in a transaction,
// then the panic will fire again.
func WithPanicPropagation() Option {
return func(o *options) {
o.panicPropagation = true
}
}
// WithRequestIgnorer returns a Option which sets r as the
// function to use to determine whether or not a request should
// be ignored. If r is nil, all requests will be reported.
func WithRequestIgnorer(r apmhttp.RequestIgnorerFunc) Option {
if r == nil {
r = apmhttp.IgnoreNone
}
return func(o *options) {
o.requestIgnorer = r
}
}