mirror of
				https://github.com/traefik/traefik.git
				synced 2025-11-04 10:21:15 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			369 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			369 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
package tracing
 | 
						|
 | 
						|
import (
 | 
						|
	"context"
 | 
						|
	"fmt"
 | 
						|
	"io"
 | 
						|
	"net"
 | 
						|
	"net/http"
 | 
						|
	"net/url"
 | 
						|
	"slices"
 | 
						|
	"strconv"
 | 
						|
	"strings"
 | 
						|
 | 
						|
	"github.com/rs/zerolog/log"
 | 
						|
	"github.com/traefik/traefik/v3/pkg/config/static"
 | 
						|
	"github.com/traefik/traefik/v3/pkg/observability"
 | 
						|
	otypes "github.com/traefik/traefik/v3/pkg/observability/types"
 | 
						|
	"go.opentelemetry.io/contrib/propagators/autoprop"
 | 
						|
	"go.opentelemetry.io/otel"
 | 
						|
	"go.opentelemetry.io/otel/attribute"
 | 
						|
	"go.opentelemetry.io/otel/codes"
 | 
						|
	"go.opentelemetry.io/otel/propagation"
 | 
						|
	semconv "go.opentelemetry.io/otel/semconv/v1.37.0"
 | 
						|
	"go.opentelemetry.io/otel/trace"
 | 
						|
)
 | 
						|
 | 
						|
// Backend is an abstraction for tracking backend (OpenTelemetry, ...).
 | 
						|
type Backend interface {
 | 
						|
	Setup(ctx context.Context, serviceName string, sampleRate float64, resourceAttributes map[string]string) (trace.Tracer, io.Closer, error)
 | 
						|
}
 | 
						|
 | 
						|
// NewTracing Creates a Tracing.
 | 
						|
func NewTracing(ctx context.Context, conf *static.Tracing) (*Tracer, io.Closer, error) {
 | 
						|
	var backend Backend
 | 
						|
 | 
						|
	if conf.OTLP != nil {
 | 
						|
		backend = conf.OTLP
 | 
						|
	}
 | 
						|
 | 
						|
	if backend == nil {
 | 
						|
		log.Debug().Msg("Could not initialize tracing, using OpenTelemetry by default")
 | 
						|
		defaultBackend := &otypes.OTelTracing{}
 | 
						|
		backend = defaultBackend
 | 
						|
	}
 | 
						|
 | 
						|
	if err := observability.EnsureUserEnvVar(); err != nil {
 | 
						|
		return nil, nil, err
 | 
						|
	}
 | 
						|
 | 
						|
	otel.SetTextMapPropagator(autoprop.NewTextMapPropagator())
 | 
						|
 | 
						|
	tr, closer, err := backend.Setup(ctx, conf.ServiceName, conf.SampleRate, conf.ResourceAttributes)
 | 
						|
	if err != nil {
 | 
						|
		return nil, nil, err
 | 
						|
	}
 | 
						|
 | 
						|
	return NewTracer(tr, conf.CapturedRequestHeaders, conf.CapturedResponseHeaders, conf.SafeQueryParams), closer, nil
 | 
						|
}
 | 
						|
 | 
						|
// TracerFromContext extracts the trace.Tracer from the given context.
 | 
						|
func TracerFromContext(ctx context.Context) *Tracer {
 | 
						|
	// Prevent picking trace.noopSpan tracer.
 | 
						|
	if !trace.SpanContextFromContext(ctx).IsValid() {
 | 
						|
		return nil
 | 
						|
	}
 | 
						|
 | 
						|
	span := trace.SpanFromContext(ctx)
 | 
						|
	if span != nil && span.TracerProvider() != nil {
 | 
						|
		tracer := span.TracerProvider().Tracer("github.com/traefik/traefik")
 | 
						|
		if tracer, ok := tracer.(*Tracer); ok {
 | 
						|
			return tracer
 | 
						|
		}
 | 
						|
 | 
						|
		return nil
 | 
						|
	}
 | 
						|
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
// ExtractCarrierIntoContext reads cross-cutting concerns from the carrier into a Context.
 | 
						|
func ExtractCarrierIntoContext(ctx context.Context, headers http.Header) context.Context {
 | 
						|
	propagator := otel.GetTextMapPropagator()
 | 
						|
	return propagator.Extract(ctx, propagation.HeaderCarrier(headers))
 | 
						|
}
 | 
						|
 | 
						|
// InjectContextIntoCarrier sets cross-cutting concerns from the request context into the request headers.
 | 
						|
func InjectContextIntoCarrier(req *http.Request) {
 | 
						|
	propagator := otel.GetTextMapPropagator()
 | 
						|
	propagator.Inject(req.Context(), propagation.HeaderCarrier(req.Header))
 | 
						|
}
 | 
						|
 | 
						|
// Span is trace.Span wrapping the Traefik TracerProvider.
 | 
						|
type Span struct {
 | 
						|
	trace.Span
 | 
						|
 | 
						|
	tracerProvider *TracerProvider
 | 
						|
}
 | 
						|
 | 
						|
// TracerProvider returns the span's TraceProvider.
 | 
						|
func (s Span) TracerProvider() trace.TracerProvider {
 | 
						|
	return s.tracerProvider
 | 
						|
}
 | 
						|
 | 
						|
// TracerProvider is trace.TracerProvider wrapping the Traefik Tracer implementation.
 | 
						|
type TracerProvider struct {
 | 
						|
	trace.TracerProvider
 | 
						|
 | 
						|
	tracer *Tracer
 | 
						|
}
 | 
						|
 | 
						|
// Tracer returns the trace.Tracer for the given options.
 | 
						|
// It returns specifically the Traefik Tracer when requested.
 | 
						|
func (t TracerProvider) Tracer(name string, options ...trace.TracerOption) trace.Tracer {
 | 
						|
	if name == "github.com/traefik/traefik" {
 | 
						|
		return t.tracer
 | 
						|
	}
 | 
						|
 | 
						|
	return t.TracerProvider.Tracer(name, options...)
 | 
						|
}
 | 
						|
 | 
						|
// Tracer is trace.Tracer with additional properties.
 | 
						|
type Tracer struct {
 | 
						|
	trace.Tracer
 | 
						|
 | 
						|
	safeQueryParams         []string
 | 
						|
	capturedRequestHeaders  []string
 | 
						|
	capturedResponseHeaders []string
 | 
						|
}
 | 
						|
 | 
						|
// NewTracer builds and configures a new Tracer.
 | 
						|
func NewTracer(tracer trace.Tracer, capturedRequestHeaders, capturedResponseHeaders, safeQueryParams []string) *Tracer {
 | 
						|
	return &Tracer{
 | 
						|
		Tracer:                  tracer,
 | 
						|
		safeQueryParams:         safeQueryParams,
 | 
						|
		capturedRequestHeaders:  canonicalizeHeaders(capturedRequestHeaders),
 | 
						|
		capturedResponseHeaders: canonicalizeHeaders(capturedResponseHeaders),
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// Start starts a new span.
 | 
						|
// spancheck linter complains about span.End not being called, but this is expected here,
 | 
						|
// hence its deactivation.
 | 
						|
//
 | 
						|
//nolint:spancheck
 | 
						|
func (t *Tracer) Start(ctx context.Context, spanName string, opts ...trace.SpanStartOption) (context.Context, trace.Span) {
 | 
						|
	if t == nil {
 | 
						|
		return ctx, nil
 | 
						|
	}
 | 
						|
 | 
						|
	spanCtx, span := t.Tracer.Start(ctx, spanName, opts...)
 | 
						|
 | 
						|
	wrappedSpan := &Span{Span: span, tracerProvider: &TracerProvider{TracerProvider: span.TracerProvider(), tracer: t}}
 | 
						|
 | 
						|
	return trace.ContextWithSpan(spanCtx, wrappedSpan), wrappedSpan
 | 
						|
}
 | 
						|
 | 
						|
// CaptureClientRequest used to add span attributes from the request as a Client.
 | 
						|
func (t *Tracer) CaptureClientRequest(span trace.Span, r *http.Request) {
 | 
						|
	if t == nil || span == nil || r == nil {
 | 
						|
		return
 | 
						|
	}
 | 
						|
 | 
						|
	// Common attributes https://github.com/open-telemetry/semantic-conventions/blob/v1.26.0/docs/http/http-spans.md#common-attributes
 | 
						|
	span.SetAttributes(semconv.HTTPRequestMethodKey.String(r.Method))
 | 
						|
	span.SetAttributes(semconv.NetworkProtocolVersion(proto(r.Proto)))
 | 
						|
 | 
						|
	// Client attributes https://github.com/open-telemetry/semantic-conventions/blob/v1.26.0/docs/http/http-spans.md#http-client
 | 
						|
	sURL := t.safeURL(r.URL)
 | 
						|
	span.SetAttributes(semconv.URLFull(sURL.String()))
 | 
						|
	span.SetAttributes(semconv.URLScheme(sURL.Scheme))
 | 
						|
	span.SetAttributes(semconv.UserAgentOriginal(r.UserAgent()))
 | 
						|
 | 
						|
	host, port, err := net.SplitHostPort(sURL.Host)
 | 
						|
	if err != nil {
 | 
						|
		span.SetAttributes(semconv.NetworkPeerAddress(host))
 | 
						|
		span.SetAttributes(semconv.ServerAddress(sURL.Host))
 | 
						|
		switch sURL.Scheme {
 | 
						|
		case "http":
 | 
						|
			span.SetAttributes(semconv.NetworkPeerPort(80))
 | 
						|
			span.SetAttributes(semconv.ServerPort(80))
 | 
						|
		case "https":
 | 
						|
			span.SetAttributes(semconv.NetworkPeerPort(443))
 | 
						|
			span.SetAttributes(semconv.ServerPort(443))
 | 
						|
		}
 | 
						|
	} else {
 | 
						|
		span.SetAttributes(semconv.NetworkPeerAddress(host))
 | 
						|
		intPort, _ := strconv.Atoi(port)
 | 
						|
		span.SetAttributes(semconv.NetworkPeerPort(intPort))
 | 
						|
		span.SetAttributes(semconv.ServerAddress(host))
 | 
						|
		span.SetAttributes(semconv.ServerPort(intPort))
 | 
						|
	}
 | 
						|
 | 
						|
	for _, header := range t.capturedRequestHeaders {
 | 
						|
		// User-agent is already part of the semantic convention as a recommended attribute.
 | 
						|
		if strings.EqualFold(header, "User-Agent") {
 | 
						|
			continue
 | 
						|
		}
 | 
						|
 | 
						|
		if value := r.Header[header]; value != nil {
 | 
						|
			span.SetAttributes(attribute.StringSlice(fmt.Sprintf("http.request.header.%s", strings.ToLower(header)), value))
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// CaptureServerRequest used to add span attributes from the request as a Server.
 | 
						|
func (t *Tracer) CaptureServerRequest(span trace.Span, r *http.Request) {
 | 
						|
	if t == nil || span == nil || r == nil {
 | 
						|
		return
 | 
						|
	}
 | 
						|
 | 
						|
	// Common attributes https://github.com/open-telemetry/semantic-conventions/blob/v1.26.0/docs/http/http-spans.md#common-attributes
 | 
						|
	span.SetAttributes(semconv.HTTPRequestMethodKey.String(r.Method))
 | 
						|
	span.SetAttributes(semconv.NetworkProtocolVersion(proto(r.Proto)))
 | 
						|
 | 
						|
	sURL := t.safeURL(r.URL)
 | 
						|
	// Server attributes https://github.com/open-telemetry/semantic-conventions/blob/v1.26.0/docs/http/http-spans.md#http-server-semantic-conventions
 | 
						|
	span.SetAttributes(semconv.HTTPRequestBodySize(int(r.ContentLength)))
 | 
						|
	span.SetAttributes(semconv.URLPath(sURL.Path))
 | 
						|
	span.SetAttributes(semconv.URLQuery(sURL.RawQuery))
 | 
						|
	span.SetAttributes(semconv.URLScheme(r.Header.Get("X-Forwarded-Proto")))
 | 
						|
	span.SetAttributes(semconv.UserAgentOriginal(r.UserAgent()))
 | 
						|
	span.SetAttributes(semconv.ServerAddress(r.Host))
 | 
						|
 | 
						|
	host, port, err := net.SplitHostPort(r.RemoteAddr)
 | 
						|
	if err != nil {
 | 
						|
		span.SetAttributes(semconv.ClientAddress(r.RemoteAddr))
 | 
						|
		span.SetAttributes(semconv.NetworkPeerAddress(r.Host))
 | 
						|
	} else {
 | 
						|
		span.SetAttributes(semconv.NetworkPeerAddress(host))
 | 
						|
		span.SetAttributes(semconv.ClientAddress(host))
 | 
						|
		intPort, _ := strconv.Atoi(port)
 | 
						|
		span.SetAttributes(semconv.ClientPort(intPort))
 | 
						|
		span.SetAttributes(semconv.NetworkPeerPort(intPort))
 | 
						|
	}
 | 
						|
 | 
						|
	for _, header := range t.capturedRequestHeaders {
 | 
						|
		// User-agent is already part of the semantic convention as a recommended attribute.
 | 
						|
		if strings.EqualFold(header, "User-Agent") {
 | 
						|
			continue
 | 
						|
		}
 | 
						|
 | 
						|
		if value := r.Header[header]; value != nil {
 | 
						|
			span.SetAttributes(attribute.StringSlice(fmt.Sprintf("http.request.header.%s", strings.ToLower(header)), value))
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// CaptureResponse captures the response attributes to the span.
 | 
						|
func (t *Tracer) CaptureResponse(span trace.Span, responseHeaders http.Header, code int, spanKind trace.SpanKind) {
 | 
						|
	if t == nil || span == nil {
 | 
						|
		return
 | 
						|
	}
 | 
						|
 | 
						|
	var status codes.Code
 | 
						|
	var desc string
 | 
						|
	switch spanKind {
 | 
						|
	case trace.SpanKindServer:
 | 
						|
		status, desc = serverStatus(code)
 | 
						|
	case trace.SpanKindClient:
 | 
						|
		status, desc = clientStatus(code)
 | 
						|
	default:
 | 
						|
		status, desc = defaultStatus(code)
 | 
						|
	}
 | 
						|
	span.SetStatus(status, desc)
 | 
						|
	if code > 0 {
 | 
						|
		span.SetAttributes(semconv.HTTPResponseStatusCode(code))
 | 
						|
	}
 | 
						|
 | 
						|
	for _, header := range t.capturedResponseHeaders {
 | 
						|
		if value := responseHeaders[header]; value != nil {
 | 
						|
			span.SetAttributes(attribute.StringSlice(fmt.Sprintf("http.response.header.%s", strings.ToLower(header)), value))
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func (t *Tracer) safeURL(originalURL *url.URL) *url.URL {
 | 
						|
	if originalURL == nil {
 | 
						|
		return nil
 | 
						|
	}
 | 
						|
 | 
						|
	redactedURL := *originalURL
 | 
						|
 | 
						|
	// Redact password if exists.
 | 
						|
	if redactedURL.User != nil {
 | 
						|
		redactedURL.User = url.UserPassword("REDACTED", "REDACTED")
 | 
						|
	}
 | 
						|
 | 
						|
	// Redact query parameters.
 | 
						|
	query := redactedURL.Query()
 | 
						|
	for k := range query {
 | 
						|
		if slices.Contains(t.safeQueryParams, k) {
 | 
						|
			continue
 | 
						|
		}
 | 
						|
 | 
						|
		query.Set(k, "REDACTED")
 | 
						|
	}
 | 
						|
	redactedURL.RawQuery = query.Encode()
 | 
						|
 | 
						|
	return &redactedURL
 | 
						|
}
 | 
						|
 | 
						|
func proto(proto string) string {
 | 
						|
	switch proto {
 | 
						|
	case "HTTP/1.0":
 | 
						|
		return "1.0"
 | 
						|
	case "HTTP/1.1":
 | 
						|
		return "1.1"
 | 
						|
	case "HTTP/2":
 | 
						|
		return "2"
 | 
						|
	case "HTTP/3":
 | 
						|
		return "3"
 | 
						|
	default:
 | 
						|
		return proto
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// serverStatus returns a span status code and message for an HTTP status code
 | 
						|
// value returned by a server. Status codes in the 400-499 range are not
 | 
						|
// returned as errors.
 | 
						|
func serverStatus(code int) (codes.Code, string) {
 | 
						|
	if code < 100 || code >= 600 {
 | 
						|
		return codes.Error, fmt.Sprintf("Invalid HTTP status code %d", code)
 | 
						|
	}
 | 
						|
	if code >= 500 {
 | 
						|
		return codes.Error, ""
 | 
						|
	}
 | 
						|
	return codes.Unset, ""
 | 
						|
}
 | 
						|
 | 
						|
// clientStatus returns a span status code and message for an HTTP status code
 | 
						|
// value returned by a server. Status codes in the 400-499 range are not
 | 
						|
// returned as errors.
 | 
						|
func clientStatus(code int) (codes.Code, string) {
 | 
						|
	if code < 100 || code >= 600 {
 | 
						|
		return codes.Error, fmt.Sprintf("Invalid HTTP status code %d", code)
 | 
						|
	}
 | 
						|
	if code >= 400 {
 | 
						|
		return codes.Error, ""
 | 
						|
	}
 | 
						|
	return codes.Unset, ""
 | 
						|
}
 | 
						|
 | 
						|
// defaultStatus returns a span status code and message for an HTTP status code
 | 
						|
// value generated internally.
 | 
						|
func defaultStatus(code int) (codes.Code, string) {
 | 
						|
	if code < 100 || code >= 600 {
 | 
						|
		return codes.Error, fmt.Sprintf("Invalid HTTP status code %d", code)
 | 
						|
	}
 | 
						|
	if code >= 500 {
 | 
						|
		return codes.Error, ""
 | 
						|
	}
 | 
						|
	return codes.Unset, ""
 | 
						|
}
 | 
						|
 | 
						|
// canonicalizeHeaders converts a slice of header keys to their canonical form.
 | 
						|
// It uses http.CanonicalHeaderKey to ensure that the headers are in a consistent format.
 | 
						|
func canonicalizeHeaders(headers []string) []string {
 | 
						|
	if headers == nil {
 | 
						|
		return nil
 | 
						|
	}
 | 
						|
 | 
						|
	canonicalHeaders := make([]string, len(headers))
 | 
						|
	for i, header := range headers {
 | 
						|
		canonicalHeaders[i] = http.CanonicalHeaderKey(header)
 | 
						|
	}
 | 
						|
 | 
						|
	return canonicalHeaders
 | 
						|
}
 |