traceway/opentelemetry-symfony

Pure-PHP OpenTelemetry instrumentation for Symfony — automatic HTTP, Console, HttpClient, Messenger, Doctrine DBAL, Cache, Twig tracing and Monolog log-trace correlation with response propagation, a lightweight Tracing helper, route templates, and semantic conventions. No C extension required (ext-p

Maintainers

Package info

github.com/tracewayapp/opentelemetry-symfony-bundle

Type:symfony-bundle

pkg:composer/traceway/opentelemetry-symfony

Statistics

Installs: 10 809

Dependents: 0

Suggesters: 0

Stars: 77

Open Issues: 0

v3.0.0 2026-06-15 14:52 UTC

README

Traceway Logo

OpenTelemetry Symfony Bundle

CI codecov Packagist Version Packagist Downloads PHP Version Symfony Version License Discord

Pure-PHP OpenTelemetry instrumentation for Symfony. Automatic tracing for HTTP, Console, HttpClient, Messenger, Mailer, Scheduler, Doctrine DBAL, Cache, and Twig — plus Monolog log-trace correlation, OTel log export, and opt-in metrics. No C extension required.

Works with any OpenTelemetry-compatible backend: Traceway, Jaeger, Zipkin, Datadog, Grafana Tempo, Honeycomb, AWS X-Ray, and more.

  • Pure PHP — installs on every managed Symfony host
  • Production-ready — stable since v1.0, PHPStan level 10 with no baseline, Symfony 6.4 LTS through 8.x
  • Correct under load — Messenger context propagates across async boundaries, DBAL 3 and 4 CI-tested, re-entrance guards on HttpClient and the log handler

Installation

composer require traceway/opentelemetry-symfony

Symfony Flex registers the bundle automatically. Without Flex, add it to config/bundles.php:

return [
    // ...
    Traceway\OpenTelemetryBundle\OpenTelemetryBundle::class => ['all' => true],
];

Quick Start

OTEL_PHP_AUTOLOAD_ENABLED=true
OTEL_SERVICE_NAME=my-symfony-app
OTEL_TRACES_EXPORTER=otlp
OTEL_EXPORTER_OTLP_ENDPOINT=http://localhost:4318
OTEL_EXPORTER_OTLP_PROTOCOL=http/json

That's it. Every HTTP request, console command, outgoing call, Messenger job, DB query, cache operation, and Twig render is now traced.

It is also possible to use Symfony's DotEnv component in combination with the open_telemetry.sdk bundle configuration section for OTEL_PHP_AUTOLOAD_ENABLED, OTEL_RESOURCE_ATTRIBUTES and OTEL_EXPORTER_OTLP_HEADERS to set the relevant SDK configuration, see docs/configuration.md for more details. Using configuration setting open_telemetry.sdk.autoload_enabled registers the OpenTelemetry SDK without needing to set OTEL_PHP_AUTOLOAD_ENABLED before Composer's autoload is loaded. Furthermore, using open_telemetry.sdk.resource_attributes allows OTEL_RESOURCE_ATTRIBUTES to be set more conveniently and also supports the use of Symfony Secrets and the same applies with open_telemetry.sdk.exporter_otlp_headers for OTEL_EXPORTER_OTLP_HEADERS.

Verify the wiring:

bin/console traceway:doctor

What Gets Traced

Component Span Kind What's captured
HTTP requests SERVER Route templates (GET /api/items/{id}), status codes, body sizes, client IP, exceptions, sub-requests
Console commands INTERNAL Command name, argv, pid, exit code, exceptions
HttpClient CLIENT Outgoing requests with W3C context propagation, OTLP endpoint auto-excluded, re-entrance guard
Messenger PRODUCER/CONSUMER Message class, transport, W3C context propagation across async boundaries
Scheduler CONSUMER Schedule name, trigger, next-run, cancellation marker. Requires symfony/scheduler
Mailer PRODUCER + CLIENT create span on MailerInterface::send, send span on the transport. Recipient count, message-id, X-Transport routing
Doctrine DBAL CLIENT Parameterised SQL, transactions, db system/namespace auto-detection. DBAL 3.6+ and 4.x CI-tested
Cache INTERNAL get (hit/miss), delete, invalidateTags with pool name. Requires symfony/cache
Twig INTERNAL Template name, nested includes. Requires twig/twig
Monolog Inject trace_id, span_id + trace_flags into every log record. Opt-in OTel Logs API export with per-channel scope

Also: Server-Timing response headers, full OTel semantic conventions.

Semantic Conventions

This bundle tracks the current OTel semantic conventions, including details most instrumentations skip: _OTHER method normalization, url.full credential/query redaction, default-port inference, error.type on every failure path, stable db.system.name values, SQLSTATE as db.response.status_code, and the per-signal histogram bucket advisories.

Deliberate deviations, chosen so task-oriented backends group telemetry usefully:

Where Spec says We do Why
Messenger span name send {transport} send {MessageClass} Tasks group per message type, not per queue
Console span name {process.executable.name} the command name app:import beats php (allowed low-cardinality alternative)
Consumer parenting span links by default parent-child (links with root_spans: true) end-to-end traces out of the box
db.system, db.statement, … deprecated dual-emitted alongside the stable keys migration aid for older backends, removal in v3.0
db.query.text sanitize by default recorded verbatim (prepared statements are placeholder-safe) disable with traces.doctrine.record_statements: false

Custom attributes (console.command, cache.*, twig.*, scheduler.*, messaging.message.class) cover areas with no registered convention yet.

Configuration

All options are optional — the bundle works out of the box. A minimal config/packages/open_telemetry.yaml:

open_telemetry:
    traces:
        excluded_paths: [/health, /_profiler, /_wdt]
    metrics:
        enabled: true
    logs:
        correlation:
            enabled: true

See docs/configuration.md for the full reference and environment variables.

Manual Instrumentation

Inject TracingInterface for one-liner span creation:

use Traceway\OpenTelemetryBundle\TracingInterface;

class OrderService
{
    public function __construct(private readonly TracingInterface $tracing) {}

    public function process(int $orderId): void
    {
        $this->tracing->trace('order.validate', fn () => $this->validate($orderId));
        $this->tracing->trace('order.fulfill', function () {
            $this->tracing->trace('inventory.reserve', fn () => $this->reserve());
            $this->tracing->trace('payment.charge', fn () => $this->charge());
        });
    }
}

Mock in tests with $this->createStub(TracingInterface::class) and have trace() invoke the callback directly.

Metrics

Opt-in OpenTelemetry metrics — Messenger, Doctrine DBAL, HTTP server/client, and Mailer — alongside a MeterRegistryInterface for custom counters/histograms. See docs/metrics.md.

Doctor

bin/console traceway:doctor runs diagnostic checks against the bundle's wiring, SDK environment variables, and OTLP endpoint reachability — text or JSON, scriptable in CI.

Runtime
  ✓ ext-opentelemetry not loaded (no conflict risk)
SDK configuration
  ✓ OTEL_SERVICE_NAME = "my-symfony-app"
  ✓ OTEL_TRACES_EXPORTER = otlp
Connectivity
  ✓ OTLP endpoint reachable (HTTP 404, 7ms)
Results: 9 ok, 0 warning, 0 error, 3 skipped, 0 info

See docs/doctor.md for flags, JSON envelope schema, and custom checks.

Documentation

Configuration Full config reference and environment variables
Metrics Instrument list, manual metrics, exemplars
AWS X-Ray Native propagator / id_generator keys
Doctor traceway:doctor flags, JSON output, custom checks
Performance Overhead, sampling, exporter choice
Upgrade from v1.x Flat → nested config migration
Changelog Release history

Contributing

git clone https://github.com/tracewayapp/opentelemetry-symfony-bundle.git
cd opentelemetry-symfony-bundle
composer install
vendor/bin/phpunit
vendor/bin/phpstan analyse

See CONTRIBUTING.md. Join the conversation on Discord.

License

MIT