The integration of reverse proxies with OpenTelemetry (OTel) has become a critical requirement for modern observability pipelines. This report evaluates four open-source reverse proxies—NGINX, HAProxy, Caddy, and Envoy—and their compatibility with OpenTelemetry for collecting logs, metrics, and traces. Each solution is analyzed for configuration workflows, data export capabilities, and architectural considerations.
NGINX: Enhanced Tracing via the ngx_otel_module
Architecture and Use Cases
NGINX, a high-performance web server and reverse proxy, supports distributed tracing through the ngx_otel_module. This module enables W3C trace context propagation and exports telemetry data via the OpenTelemetry Protocol (OTLP/gRPC). The setup is ideal for environments where NGINX acts as a reverse proxy for microservices, providing end-to-end visibility into request flows169.
Configuration Steps
-
Module Installation:
Install thenginx-plus-module-otel
package for supported Linux distributions (e.g., CentOS, Ubuntu). Dynamic loading is enabled via thenginx.conf
file:text
load_module modules/ngx_otel_module.so;
-
Directive Setup:
Enable tracing inhttp
,server
, orlocation
contexts:text
http { otel_exporter { endpoint otel-collector:4317; } server { location / { otel_trace on; otel_trace_context propagate; proxy_pass http://backend; } } }
-
otel_trace
enables tracing for specific routes. -
otel_trace_context
handles trace propagation (extract, inject, or ignore headers)9.
-
-
Span Customization:
Attributes likeotel_span_name
andotel_span_attr
add contextual metadata:text
otel_span_name "nginx_proxy"; otel_span_attr "http.method" "$request_method";
Metrics and Logs
-
Traces: Full support for distributed tracing with parent-child span relationships.
-
Metrics: Requires integration with Prometheus or OTel Collector’s NGINX receiver (not covered by the native module).
-
Logs: Standard NGINX access/error logs, which can be ingested into OTel via filelog receivers613.
HAProxy: Metrics and Logs via OTel Collector
Integration Workflow
HAProxy, a performant TCP/HTTP load balancer, relies on the OpenTelemetry Collector’s HAProxy receiver for metrics and filelog receiver for logs213.
-
Metrics Collection:
-
Expose HAProxy’s built-in statistics endpoint (e.g.,
http://haproxy:8000/haproxy?stats
). -
Configure the OTel Collector’s
haproxyreceiver
to scrape metrics:text
receivers: haproxy: endpoint: "http://haproxy:8000/haproxy?stats" exporters: otlp: endpoint: "otel-backend:4317"
Key metrics include
haproxy_connections_rate
(Gauge) andhaproxy_requests_denied
(Sum)213.
-
-
Log Collection:
-
HAProxy writes logs to
/var/log/haproxy.log
. -
Use the Collector’s
filelog
receiver:text
receivers: filelog: include: [/var/log/haproxy.log]
-
Limitations
-
Tracing: No native support for distributed tracing. Requires manual instrumentation or sidecar proxies.
-
Authentication: HAProxy’s stats endpoint lacks OAuth/MTLS support, necessitating network-level security27.
Caddy: Native OpenTelemetry Tracing Directives
Built-in OTel Integration
Caddy’s tracing directive enables automatic trace propagation and export via OTLP/gRPC. It supports W3C tracecontext and baggage headers, making it suitable for API gateways and edge proxies14516.
Configuration Example
text
example.com { tracing { span_name "caddy_proxy" } reverse_proxy localhost:8080 }
-
Environment Variables: Configure exporters using OTel standards (e.g.,
OTEL_EXPORTER_OTLP_ENDPOINT
)14. -
Logs: Access logs include
traceID
andspanID
fields, correlatable with OTel backends.
Layer 4 Proxying
For non-HTTP protocols (e.g., MQTT), Caddy’s Layer 4 proxy routes raw TCP/UDP traffic while still exporting spans314:
text
tcp://mqtt.example.com { tracing { span_name "mqtt_proxy" } reverse_proxy 10.0.0.5:1883 }
Limitations
-
Metrics: No native metric export; requires third-party plugins or sidecar collectors.
-
Advanced Tracing: Custom spans (e.g., database calls) need manual instrumentation1114.
Envoy: Cloud-Native Observability with OTel
Tracing Configuration
Envoy, a cloud-native proxy, supports OpenTelemetry through its http_connection_manager filter. The setup is ideal for service meshes and Kubernetes ingress4.
YAML Snippet
text
static_resources: listeners: - filters: - name: envoy.filters.network.http_connection_manager typed_config: "@type": type.googleapis.com/envoy.extensions.filters.network.http_connection_manager.v3.HttpConnectionManager tracing: provider: name: envoy.tracers.opentelemetry typed_config: "@type": type.googleapis.com/envoy.config.trace.v3.OpenTelemetryConfig grpc_service: envoy_grpc: cluster_name: otel-collector
-
Exporter: OTLP/gRPC to a Collector cluster.
-
Propagation: B3 and W3C headers are supported.
Metrics and Logs
-
Metrics: Envoy emits built-in metrics (e.g.,
http.downstream_rq_total
) scrapable via Prometheus or OTel412. -
Logs: Access logs are configurable in JSON format, ingestible via filelog receivers.
Comparative Analysis
Proxy | Tracing Support | Metrics Collection | Log Collection | Use Cases |
---|---|---|---|---|
NGINX | Native (ngx_otel_module) | Via Prometheus/OTel | Filelog Receiver | Legacy apps, high-traffic proxies |
HAProxy | Limited (Stats API) | OTel Collector Receiver | Filelog Receiver | TCP load balancing, minimal setup |
Caddy | Native (Directive) | Third-party plugins | Built-in Access Logs | Edge routing, TLS automation |
Envoy | Native (OTel Tracer) | Built-in Metrics | JSON Access Logs | Service meshes, Kubernetes ingress |
Conclusion and Recommendations
-
NGINX is optimal for teams requiring deep request-level tracing in heterogeneous environments.
-
HAProxy suits high-throughput TCP workloads where metrics are prioritized over traces.
-
Caddy simplifies TLS and Layer 4 proxying with zero-config OTel tracing.
-
Envoy excels in cloud-native ecosystems with native integration into service meshes.
For full observability, combine proxies with the OpenTelemetry Collector to unify logs, metrics, and traces. Future work should explore Traefik’s OTel compatibility and eBPF-based tracing for kernel-level insights.
Citations:
- https://signoz.io/blog/opentelemetry-nginx/
- https://signoz.io/blog/opentelemetry-haproxy-metrics-and-logs-monitoring/
- https://www.reddit.com/r/opnsense/comments/1fw6891/mqtt_reverse_proxy_with_caddy_gui_of_opnsense/
- https://opentelemetry.io/docs/demo/services/frontend-proxy/
- https://www.reddit.com/r/selfhosted/comments/1coa0qr/looking_to_reverse_proxy_services_within_a_local/
- https://www.alibabacloud.com/help/doc-detail/2859213.html
- https://discourse.haproxy.org/t/installation-steps-ha-proxy-as-reverse-proxy/2169
- https://caddy.community/t/tracing-configuration-for-insecure-grpc-opentelemetry-backend-like-jaeger-all-in-one/16764
- https://docs.nginx.com/nginx/admin-guide/dynamic-modules/opentelemetry/
- https://www.haproxy.com/blog/fundamentals-high-availability-and-the-role-of-a-reverse-proxy
- https://caddy.community/t/add-opentelemetry-tracing-support/13663
- https://traefik.io/traefik/
- https://help.sumologic.com/docs/integrations/web-servers/opentelemetry/haproxy-opentelemetry/
- https://caddyserver.com/docs/caddyfile/directives/tracing
- https://opentelemetry.io/docs/languages/js/exporters/
- https://forum.opnsense.org/index.php?topic=38714.0
- https://opentelemetry.io/blog/2022/instrument-nginx/
- https://www.f5.com/company/blog/nginx/nginx-tutorial-opentelemetry-tracing-understand-microservices
- https://help.sumologic.com/docs/integrations/web-servers/opentelemetry/nginx-opentelemetry/
- https://microsoft.github.io/reverse-proxy/articles/distributed-tracing.html
- https://github.com/haproxy/haproxy/issues/1640
- https://www.loadbalancer.org/blog/haproxy-reverse-proxy-the-what-when-and-how/
- https://www.honeycomb.io/blog/demo-app-opentelemetry-instrumentation-honeycomb
- https://opentelemetry.io/ecosystem/registry/
- https://github.com/open-telemetry/opentelemetry-collector-contrib/issues/24921
- https://coralogix.com/blog/ship-opentelemetry-data-to-coralogix-via-reverse-proxy-caddy-2/
- https://www.linkedin.com/posts/sonika-singh-a990a314b_ship-opentelemetry-data-to-coralogix-via-activity-7084045353502543873-Mcg1
- https://caddyserver.com/docs/modules/http.handlers.tracing
- https://github.com/caddyserver/caddy/issues/6483/linked_closing_reference