From 3df618fa6be023319f223ff6b151be824fc98020 Mon Sep 17 00:00:00 2001 From: dmathieu <42@dmathieu.com> Date: Wed, 29 May 2024 10:36:18 +0200 Subject: [PATCH] add otelhttp Handler.ServeHTTP and Transport.RoundTrip benchmarks --- .../net/http/otelhttp/test/handler_test.go | 41 ++++++++++++++ .../net/http/otelhttp/test/transport_test.go | 56 ++++++++++++++++--- 2 files changed, 90 insertions(+), 7 deletions(-) diff --git a/instrumentation/net/http/otelhttp/test/handler_test.go b/instrumentation/net/http/otelhttp/test/handler_test.go index 2fe39555243..1148a103286 100644 --- a/instrumentation/net/http/otelhttp/test/handler_test.go +++ b/instrumentation/net/http/otelhttp/test/handler_test.go @@ -22,6 +22,7 @@ import ( "go.opentelemetry.io/otel/propagation" "go.opentelemetry.io/otel/sdk/instrumentation" "go.opentelemetry.io/otel/sdk/metric" + sdkmetric "go.opentelemetry.io/otel/sdk/metric" "go.opentelemetry.io/otel/sdk/metric/metricdata" "go.opentelemetry.io/otel/sdk/metric/metricdata/metricdatatest" sdktrace "go.opentelemetry.io/otel/sdk/trace" @@ -528,3 +529,43 @@ func TestWithRouteTag(t *testing.T) { } } } + +func BenchmarkHandlerServeHTTP(b *testing.B) { + tp := sdktrace.NewTracerProvider() + mp := sdkmetric.NewMeterProvider() + + r, err := http.NewRequest(http.MethodGet, "http://localhost/", nil) + require.NoError(b, err) + + for _, bb := range []struct { + name string + handler http.Handler + }{ + { + name: "without the otelhttp handler", + handler: http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + fmt.Fprint(w, "Hello World") + }), + }, + { + name: "with the otelhttp handler", + handler: otelhttp.NewHandler(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + fmt.Fprint(w, "Hello World") + }), + "test_handler", + otelhttp.WithTracerProvider(tp), + otelhttp.WithMeterProvider(mp), + ), + }, + } { + b.Run(bb.name, func(b *testing.B) { + rr := httptest.NewRecorder() + + b.ReportAllocs() + b.ResetTimer() + for i := 0; i < b.N; i++ { + bb.handler.ServeHTTP(rr, r) + } + }) + } +} diff --git a/instrumentation/net/http/otelhttp/test/transport_test.go b/instrumentation/net/http/otelhttp/test/transport_test.go index b303f10c57f..fa84e5598e7 100644 --- a/instrumentation/net/http/otelhttp/test/transport_test.go +++ b/instrumentation/net/http/otelhttp/test/transport_test.go @@ -6,6 +6,7 @@ package test import ( "bytes" "context" + "fmt" "io" "net" "net/http" @@ -16,7 +17,6 @@ import ( "strings" "testing" - "go.opentelemetry.io/otel/sdk/metric" semconv "go.opentelemetry.io/otel/semconv/v1.20.0" "github.com/stretchr/testify/assert" @@ -24,6 +24,7 @@ import ( "go.opentelemetry.io/otel/attribute" "go.opentelemetry.io/otel/sdk/instrumentation" + sdkmetric "go.opentelemetry.io/otel/sdk/metric" "go.opentelemetry.io/otel/sdk/metric/metricdata" "go.opentelemetry.io/otel/sdk/metric/metricdata/metricdatatest" @@ -244,8 +245,8 @@ func TestTransportMetrics(t *testing.T) { responseBody := []byte("Hello, world!") t.Run("make http request and read entire response at once", func(t *testing.T) { - reader := metric.NewManualReader() - meterProvider := metric.NewMeterProvider(metric.WithReader(reader)) + reader := sdkmetric.NewManualReader() + meterProvider := sdkmetric.NewMeterProvider(sdkmetric.WithReader(reader)) ts := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { w.WriteHeader(http.StatusOK) @@ -302,8 +303,8 @@ func TestTransportMetrics(t *testing.T) { }) t.Run("make http request and buffer response", func(t *testing.T) { - reader := metric.NewManualReader() - meterProvider := metric.NewMeterProvider(metric.WithReader(reader)) + reader := sdkmetric.NewManualReader() + meterProvider := sdkmetric.NewMeterProvider(sdkmetric.WithReader(reader)) ts := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { w.WriteHeader(http.StatusOK) @@ -370,8 +371,8 @@ func TestTransportMetrics(t *testing.T) { }) t.Run("make http request and close body before reading completely", func(t *testing.T) { - reader := metric.NewManualReader() - meterProvider := metric.NewMeterProvider(metric.WithReader(reader)) + reader := sdkmetric.NewManualReader() + meterProvider := sdkmetric.NewMeterProvider(sdkmetric.WithReader(reader)) ts := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { w.WriteHeader(http.StatusOK) @@ -476,3 +477,44 @@ func assertClientScopeMetrics(t *testing.T, sm metricdata.ScopeMetrics, attrs at } metricdatatest.AssertEqual(t, want, sm.Metrics[2], metricdatatest.IgnoreTimestamp(), metricdatatest.IgnoreValue()) } + +func BenchmarkTransportRoundTrip(b *testing.B) { + ts := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + fmt.Fprint(w, "Hello World") + })) + defer ts.Close() + + tp := sdktrace.NewTracerProvider() + mp := sdkmetric.NewMeterProvider() + + r, err := http.NewRequest(http.MethodGet, ts.URL, nil) + require.NoError(b, err) + + for _, bb := range []struct { + name string + transport http.RoundTripper + }{ + { + name: "without the otelhttp transport", + transport: http.DefaultTransport, + }, + { + name: "with the otelhttp transport", + transport: otelhttp.NewTransport( + http.DefaultTransport, + otelhttp.WithTracerProvider(tp), + otelhttp.WithMeterProvider(mp), + ), + }, + } { + b.Run(bb.name, func(b *testing.B) { + c := http.Client{Transport: bb.transport} + + b.ReportAllocs() + b.ResetTimer() + for i := 0; i < b.N; i++ { + _, _ = c.Do(r) + } + }) + } +}