/aws-xray-java-agent

The official AWS X-Ray Auto Instrumentation Agent for Java.

Primary LanguageJavaApache License 2.0Apache-2.0

Build Status

Screenshot of the AWS X-Ray console

AWS X-Ray Java Agent

The AWS X-Ray Java Agent is a drop-in solution to enable X-ray traces on a web application, including automatic tracing of X-Ray SDK supported frameworks and libraries. The Java Agent provides use of the X-Ray SDK out of box, but requires no code changes to enable basic propagated traces. Please see the compatibility chart below for current feature parity between the X-Ray SDK and this Java Agent.

Please take a look at the Sample App for an example of using this agent.

The Java Agent is implemented using the new DiSCo library, an all purpose toolkit for building Java Agents.

Versioning

Each version of the agent corresponds to the supported version of the SDK in order to ensure version compatibility between the SDK dependency brought in by the agent consumers and the version of the agent itself. This agent is released under beta as there are a few known issues that are being ironed out. You may track these issues through the issues tab of this repository.

The agent provides auto instrumentation support for customers using version 2.4.0 of the X-Ray SDK, with an artifact version number of 2.4.0-beta.1.

Compatibility Chart

Feature X-Ray SDK X-Ray Agent
AWS SDK V1 Instrumentation (Confirmed on 1.11.x)
AWS SDK V2 Instrumentation
Centralized Sampling
Automatic Multi-threaded Support
Generate Custom Subsegments
SQL Queries
Plugins
Apache HTTP Client
HttpServlet
Lambda Layers
Log Injection
Spring Framework

Installing

The AWS X-Ray Java Agent is compatible with Java 8.

The first step is to bring in the agent JAR files into your environment. Insert the following dependencies into your project’s pom.xml file:

  <dependencies>
      <dependency>
          <groupId>com.amazonaws</groupId>
          <artifactId>aws-xray-auto-instrumentation-agent-bootstrap</artifactId>
          <version>2.4.0-beta.1</version>
          <scope>runtime</scope>
      </dependency>
      <dependency>
          <groupId>com.amazonaws</groupId>
          <artifactId>aws-xray-auto-instrumentation-agent-runtime</artifactId>
          <version>2.4.0-beta.1</version>
      </dependency>
  </dependencies>

Add the following plugin to your project:

  <plugin>
      <groupId>org.apache.maven.plugins</groupId>
      <artifactId>maven-dependency-plugin</artifactId>
      <version>3.1.1</version>
      <executions>
          <execution>
              <id>unpack-xray-agent</id>
              <phase>package</phase>
              <goals>
                  <goal>copy</goal>
              </goals>
              <configuration>
                  <overWriteReleases>false</overWriteReleases>
                  <overWriteSnapshots>false</overWriteSnapshots>
                  <overWriteIfNewer>true</overWriteIfNewer>
                  <artifactItems>
                      <artifactItem>
                          <!--Obtain the runtime agent artifact-->
                          <groupId>com.amazonaws</groupId>
                          <artifactId>aws-xray-auto-instrumentation-agent-runtime</artifactId>
                          <outputDirectory>${project.build.directory}/xray-agent</outputDirectory>
                      </artifactItem>
                      <artifactItem>
                          <!--Obtain the bootstrap agent artifact-->
                          <groupId>com.amazonaws</groupId>
                          <artifactId>aws-xray-auto-instrumentation-agent-bootstrap</artifactId>
                          <outputDirectory>${project.build.directory}/xray-agent</outputDirectory>
                      </artifactItem>
                  </artifactItems>
              </configuration>
          </execution>
      </executions>
  </plugin>

The agent will be built in your project’s target/xray-agent folder. There will be two JAR files, a JAR file that has a “runtime” classifier on it and one without a classifier.

Prior to running your application, add the following Java arguments, making sure to modify the service name with your service’s name:

-javaagent:/path-to-project/target/xray-agent/aws-xray-auto-instrumentation-agent-bootstrap-2.4.0-beta.1.jar=servicename=TheServiceName

-cp:/path-to-project/target/xray-agent/aws-xray-auto-instrumentation-agent-runtime-2.4.0-beta.1.jar

Please make sure to have the following dependencies in your target application before adding the Agent into your environment.

  <dependencies>
      <dependency>
          <groupId>com.amazonaws</groupId>
          <artifactId>aws-xray-auto-instrumentation-agent-runtime</artifactId>
          <version>2.4.0-beta.1</version>
      </dependency>
  </dependencies>

This runtime agent artifact will bring in all the necessary transitive dependencies into your environment.

Lambda Layers

Using Lambda Layers, you can configure a reusable Agent to include in your Lambda functions without needing to upload the source every time. To configure a Lambda Layer, see https://docs.aws.amazon.com/lambda/latest/dg/configuration-layers.html#configuration-layers-manage.

Build the agent as you would in the Installing section but add an additional artifact dependency:

  <dependency>
      <groupId>com.amazonaws</groupId>
      <artifactId>aws-xray-auto-instrumentation-agent-installer</artifactId>
      <version>2.4.0-beta.1</version>
  </dependency>

Add the following artifact item into the unpack-xray-agent execution id of the maven-dependency-plugin:

  <artifactItem>
      <!--Obtain the Agent installer artifact-->
      <groupId>com.amazonaws</groupId>
      <artifactId>aws-xray-auto-instrumentation-agent-installer</artifactId>
      <outputDirectory>${project.build.directory}/xray-agent</outputDirectory>
  </artifactItem>

This artifact is used for installing the agent during runtime. This is necessary for consumption in a Lambda environment. We do not recommend using this installer in a regular runtime environment as running the agent using the java agent argument is more reliable in ensuring all the frameworks are properly instrumented.

Build the agent and it should contain three additional jars in the ./target/xray-agent directory. Next, we need to obtain the tools.jar file from the JDK. Lambda executes the Java code in a Linux environment. In order to get this jar file, we recommend downloading the latest Correto JDK 1.8 version from the website and extracting it from the distribution. Download the tar.gz bundled version of the Correto JDK under Linux x64 and unzip it into a directory. The tools.jar should be located in lib/tools.jar. With all the jar files ready, upload them as a Lambda Layer. For your convenience, you may also retrieve the tools.jar file from the sample app resource here.

The next step is to add all the transitive dependencies of the agent as a layer. The dependencies it requires are the runtime agent (which is built in the step above), the X-Ray SDK Core package, and the X-Ray AWS SDK instrumentor package. You may use the following plugin to do so:

  <plugin>
    <groupId>org.apache.maven.plugins</groupId>
    <artifactId>maven-dependency-plugin</artifactId>
    <version>3.0.0</version>
    <executions>
      <execution>
        <id>copy-dependencies</id>
        <phase>package</phase>
        <goals>
          <goal>copy-dependencies</goal>
        </goals>
        <configuration>
          <outputDirectory>${project.build.directory}/alternateLocation</outputDirectory>
          <overWriteReleases>false</overWriteReleases>
          <overWriteSnapshots>false</overWriteSnapshots>
          <overWriteIfNewer>true</overWriteIfNewer>
        </configuration>
      </execution>
    </executions>
  </plugin>

With the transitive dependencies and agent built, add them as a lambda layer.

To use the agent, import the following class, making sure that the aws-xray-auto-instrumentation-agent-installer artifact is consumed as a dependency:

import com.amazonaws.xray.agent.XRayAgentInstaller;

Add the following line to the top most section within the class definition:

static {
    XRayAgentInstaller.installInLambda("servicename=YourServiceNameHere");
}

Your lambda function should now be instrumented.

Getting Help

Please use these community resources for getting help.

Troubleshooting

When troubleshooting the agent, one of the first steps of troubleshooting is to enable logging in the Agent. In this case, logging would mean publishing the internal log outputs to Standard Out. Logging to a standardized file is still in the works. To enable logging, please append the following line to your JVM arguments, with an equal sign as a delimiter: loggerfactory=com.amazon.disco.agent.reflect.logging.StandardOutputLoggerFactory:verbose

For example, if your JFM args looks like servicename=YourServiceNameHere, then the log-enabled argument would look like servicename=YourServiceNameHere=loggerfactory=com.amazon.disco.agent.reflect.logging.StandardOutputLoggerFactory:verbose

Here is a list of the most common issues and how to resolve them.

I've built the agent and published it as a Lambda Layer but it's still not instrumenting any of my downstream calls.

One of the most common reasons why this might be happening is that your Lambda Function does not contain all the necessary transitive dependencies the agent requires. As a sanity check, we'd recommend downloading the lambda layer and manually seeing if any dependencies are missing. You may cross reference it with the compile scoped dependencies seen here

Documentation

The developer guide provides guidance on using the AWS X-Ray Java Agent. Please refer to the Sample App for an example.

Building from Source

Once you check out the code from GitHub, you can build it using Maven. As a prerequisite, you may need to build DiSCo into Maven Local first. Follow the instructions here.

Once DiSCo has been built, you can build the package locally by running the following command:

./mvnw clean package -Dgpg.skip=true

License

The AWS X-Ray SDK Java Agent is licensed under the Apache 2.0 License. See LICENSE and NOTICE.txt for more information.