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 } }