X-Ray integration - Documentation

This package provides an advanced integration between Bref applications and AWS X-Ray monitoring (opens in a new tab).

Check out the documentation below for screenshots and more details.

Installation

1. Enable Packagist.com

The bref/xray-integration package is distributed via Packagist.com (opens in a new tab), the alternative to Packagist.org (opens in a new tab) for private packages (provided by the Composer team). To install the package, you will need:

  • a Packagist.com token
  • a Packagist.com sub-repository URL

You will receive both after purchasing a license on the Bref website.

Once you have your license, enable Packagist.com in your composer.json by adding the sub-repository URL:

{
    "repositories": [
        {"type": "composer", "url": "https://bref.repo.packagist.com/<your-org-name>/"}
    ]
}
💡

Make sure to replace the URL with the one provided with the license you purchased.

Next, configure the Packagist.com token (provided with the license you purchased).

  • On developer machines, you can run the following command:

    composer config --global --auth http-basic.bref.repo.packagist.com token <your-token>
  • In CI/CD environments, you can set the COMPOSER_AUTH environment variable:

    COMPOSER_AUTH='{"http-basic": {"bref.repo.packagist.com": {"username": "token", "password": "<token-here>"}}}'
💡

The token is for your whole organization, you can share it with your team.

2. Install the package

You can now install the package via Composer:

composer require bref/xray-integration -W

Enable X-Ray tracing in your serverless.yml file:

provider:
    # ...
    tracing:
        lambda: true
        apiGateway: true # enable if you are using API Gateway

If you use Laravel, the Laravel integration will be automatically set up via a service provider.

If you use Symfony, you will need to manually register the bundle in config/bundles.php:

return [
    // ...
    Bref\Apm\XRay\Symfony\BrefXrayBundle::class => ['all' => true],
];

Troubleshooting

Send me a Slack message or an email (matthieu@bref.sh) if you have any trouble, I can help you set it up!

Usage

Laravel

If you are using Laravel, the package automatically integrates with the Laravel service provider. You can see the traces in the AWS X-Ray console.

Request tracing

The package automatically traces all requests to your application. You can see the traces in the AWS X-Ray console.

Database queries

Eloquent queries are automatically traced and the SQL queries are added to metadata.

Jobs

The package automatically traces all jobs dispatched via the Laravel queue. You can see the traces in the AWS X-Ray console, and filter them by job name.

The integration will also forward X-Ray trace IDs correctly so that distributed tracing works correctly. For example, if a job is dispatched from a Lambda function that is already traced, the job will be traced as well.

Symfony

If you are using Symfony, the package automatically integrates with the Symfony service provider. You can see the traces in the AWS X-Ray console.

Cold starts tracing

Cold starts are automatically traced when using Laravel or Symfony. There is nothing additional to set up.

In the example below, we can see the "Initialization" segment corresponding to the whole time it takes for AWS Lambda to initialize the instance. Part of that is the "Runtime startup" segment, which is the time it takes for the Bref runtime to start up.

On top of tracing cold starts, Bref will report (via annotations) whether an invocation included a user-facing cold start:

Sometimes, Lambda will start AWS Lambda instances proactively, this is called proactive initiation (opens in a new tab). This is an AWS Lambda cold start, but it is not visible to users because the cold start happens asynchronously, before the Lambda instance is used at all. This is why Bref separates:

  • user-facing cold starts (cold starts that happened during an invocation)
  • proactive initializations (cold starts that happened outside an invocation)

Proactive initializations are also annotated by Bref:

X-Ray does not handle proactive invocations well: it includes the initialization time and the duration when Lambda was not used in a trace. As such, it is possible to see Lambda invocations with a duration of several minutes. This in an AWS X-Ray bug/limitation, Bref unfortunately cannot do anything about it. This is why Bref annotates proactive invocations, so that you can spot them in X-Ray.

Here is an example: the Lambda instance was proactively started by AWS Lambda 2 minutes before it is actually used. The real invocation duration is very short, but X-Ray wrongly shows the duration as 2 minutes.

Tracing code

The simplest way to trace custom code sections is using:

use Bref\Apm\XRay\XRay;
 
XRay::subSegment('doSomething', function () {
    // Your code here
});

Tracing AWS SDK calls

Trace AWS SDK calls with the AwsSdkTracer::trace($client) method.

use Bref\Apm\XRay\AwsSdk\AwsSdkTracer;
 
$client = new EventBridgeClient([
    // ...
]);
AwsSdkTracer::trace($client);

All calls will be traced and appear in the X-Ray console. However, only the first 10 requests will be sampled (i.e. will result in propagating tracing to the target AWS service). This is to avoid oversampling and hitting the X-Ray limit of 100 linked traces. For example, if a Lambda function sends 100 messages to SQS, only the first 10 Lambda invocations triggered by the SQS messages will create sub-traces. The other 90 invocations will appear as segments in the original trace, but will not create sub-traces.

Tracing Async-AWS SDK calls

Trace Async-AWS SDK (opens in a new tab) calls by wrapping its HTTP client with TracedAsyncAwsHttpClient. For example:

use AsyncAws\Sqs\SqsClient;
use Bref\Apm\XRay\AsyncAws\TracedAsyncAwsHttpClient;
 
$sqs = new SqsClient([ /* config */ ], null, new TracedAsyncAwsHttpClient());
// or
$sqs = new SqsClient(httpClient: new TracedAsyncAwsHttpClient());

If you are configuring the HTTP client, you can pass it to the constructor:

use AsyncAws\Sqs\SqsClient;
use Bref\Apm\XRay\AsyncAws\TracedAsyncAwsHttpClient;
use Symfony\Component\HttpClient\HttpClient;
 
$httpClient = new TracedAsyncAwsHttpClient(HttpClient::create([
    // options
]));
 
$sqs = new SqsClient([ /* config */ ], null, $httpClient);

All calls will be traced and appear in the X-Ray console. However, only the first 10 requests will be sampled (i.e. will result in propagating tracing to the target AWS service). This is to avoid oversampling and hitting the X-Ray limit of 100 linked traces. For example, if a Lambda function sends 100 messages to SQS, only the first 10 Lambda invocations triggered by the SQS messages will create sub-traces. The other 90 invocations will appear as segments in the original trace, but will not create sub-traces.

Tracing Guzzle calls

Trace Guzzle calls with the GuzzleTracer::trace() middleware:

use GuzzleHttp\HandlerStack;
use GuzzleHttp\Client;
 
$stack = new HandlerStack();
$stack->push(\Bref\Apm\XRay\Guzzle\GuzzleTracer::trace());
 
$client = new Client(['handler' => $stack]);

If you are using Symfony, you can set that up in the service container:

    client:
        class: GuzzleHttp\Client
        arguments:
            - handler: '@xray.guzzle.handler_stack'
    xray.guzzle.handler_stack:
        shared: false
        class: GuzzleHttp\HandlerStack
        factory: [ GuzzleHttp\HandlerStack, create ]
        calls:
            - [ push, [ '@Bref\Apm\XRay\Guzzle\GuzzleTracer' ] ]

The Guzzle integration correctly handles requests made in parallel using Guzzle's async API and Guzzle pools.

Latency overhead

Based on our measurements, X-Ray tracing adds between 1ms to 2ms of latency per AWS Lambda invocation.