Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: Improve OpenTelemetry trace instrumentation #6417

Open
wants to merge 18 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 11 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -33,4 +33,4 @@
}
}
}
}
}
6 changes: 5 additions & 1 deletion modules/caddyhttp/reverseproxy/reverseproxy.go
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,8 @@ import (
"sync"
"time"

"go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp"

"go.uber.org/zap"
"golang.org/x/net/http/httpguts"
cedricziel marked this conversation as resolved.
Show resolved Hide resolved

Expand Down Expand Up @@ -784,8 +786,10 @@ func (h *Handler) reverseProxy(rw http.ResponseWriter, req *http.Request, origRe

// do the round-trip; emit debug log with values we know are
// safe, or if there is no error, emit fuller log entry
// we are wrapping the RoundTripper to make it observable
start := time.Now()
res, err := h.Transport.RoundTrip(req)
otelTransport := otelhttp.NewTransport(h.Transport)
res, err := otelTransport.RoundTrip(req)
duration := time.Since(start)
logger := h.logger.With(
zap.String("upstream", di.Upstream.String()),
Expand Down
23 changes: 16 additions & 7 deletions modules/caddyhttp/tracing/module.go
Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,8 @@ func init() {
type Tracing struct {
// SpanName is a span name. It should follow the naming guidelines here:
// https://github.com/open-telemetry/opentelemetry-specification/blob/main/specification/trace/api.md#span
SpanName string `json:"span"`
SpanName string `json:"span"`
InjectServerTimingHeader bool `json:"server_timing,omitempty"`

// otel implements opentelemetry related logic.
otel openTelemetryWrapper
Expand All @@ -46,7 +47,7 @@ func (ot *Tracing) Provision(ctx caddy.Context) error {
ot.logger = ctx.Logger()

var err error
ot.otel, err = newOpenTelemetryWrapper(ctx, ot.SpanName)
ot.otel, err = newOpenTelemetryWrapper(ctx, ot.SpanName, ot.InjectServerTimingHeader)

return err
}
Expand All @@ -68,6 +69,7 @@ func (ot *Tracing) ServeHTTP(w http.ResponseWriter, r *http.Request, next caddyh
// UnmarshalCaddyfile sets up the module from Caddyfile tokens. Syntax:
//
// tracing {
// [injectServerTimingHeader <bool>]
// [span <span_name>]
// }
func (ot *Tracing) UnmarshalCaddyfile(d *caddyfile.Dispenser) error {
Expand All @@ -94,12 +96,19 @@ func (ot *Tracing) UnmarshalCaddyfile(d *caddyfile.Dispenser) error {
}

for d.NextBlock(0) {
if dst, ok := paramsMap[d.Val()]; ok {
if err := setParameter(d, dst); err != nil {
return err
switch d.Val() {
case "server_timing":
if d.NextArg() {
ot.InjectServerTimingHeader = true
}
default:
if dst, ok := paramsMap[d.Val()]; ok {
if err := setParameter(d, dst); err != nil {
return err
}
} else {
return d.ArgErr()
}
} else {
return d.ArgErr()
}
}
return nil
Expand Down
21 changes: 15 additions & 6 deletions modules/caddyhttp/tracing/tracer.go
Original file line number Diff line number Diff line change
Expand Up @@ -37,20 +37,19 @@ type openTelemetryWrapper struct {

handler http.Handler

spanName string
spanName string
injectServerTimingHeader bool
}

// newOpenTelemetryWrapper is responsible for the openTelemetryWrapper initialization using provided configuration.
func newOpenTelemetryWrapper(
ctx context.Context,
spanName string,
) (openTelemetryWrapper, error) {
func newOpenTelemetryWrapper(ctx context.Context, spanName string, injectServerTimingHeader bool) (openTelemetryWrapper, error) {
if spanName == "" {
spanName = defaultSpanName
}

ot := openTelemetryWrapper{
spanName: spanName,
injectServerTimingHeader: injectServerTimingHeader,
spanName: spanName,
}

version, _ := caddy.Version()
Expand Down Expand Up @@ -84,15 +83,25 @@ func newOpenTelemetryWrapper(
// serveHTTP injects a tracing context and call the next handler.
func (ot *openTelemetryWrapper) serveHTTP(w http.ResponseWriter, r *http.Request) {
ctx := r.Context()

ot.propagators.Inject(ctx, propagation.HeaderCarrier(r.Header))
spanCtx := trace.SpanContextFromContext(ctx)
if spanCtx.IsValid() {
traceID := spanCtx.TraceID().String()
spanID := spanCtx.SpanID().String()
// Add a trace_id placeholder, accessible via `{http.vars.trace_id}`.
caddyhttp.SetVar(ctx, "trace_id", traceID)
// Add a span_id placeholder, accessible via `{http.vars.span_id}`.
caddyhttp.SetVar(ctx, "span_id", spanID)
// Add the trace id to the log fields for the request.
if extra, ok := ctx.Value(caddyhttp.ExtraLogFieldsCtxKey).(*caddyhttp.ExtraLogFields); ok {
extra.Add(zap.String("traceID", traceID))
extra.Add(zap.String("spanID", spanID))
}

// Add the server-timing header so clients can make the connection
if ot.injectServerTimingHeader {
w.Header().Set("server-timing", fmt.Sprintf("traceparent;desc=\"00-%s-%s-%s\"", traceID, spanID, spanCtx.TraceFlags().String()))
hairyhenderson marked this conversation as resolved.
Show resolved Hide resolved
}
}
next := ctx.Value(nextCallCtxKey).(*nextCall)
Expand Down
4 changes: 1 addition & 3 deletions modules/caddyhttp/tracing/tracer_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -14,9 +14,7 @@ func TestOpenTelemetryWrapper_newOpenTelemetryWrapper(t *testing.T) {
var otw openTelemetryWrapper
var err error

if otw, err = newOpenTelemetryWrapper(ctx,
"",
); err != nil {
if otw, err = newOpenTelemetryWrapper(ctx, "", false); err != nil {
t.Errorf("newOpenTelemetryWrapper() error = %v", err)
t.FailNow()
}
Expand Down
Loading