This example demonstrates how to instrument an AWS Lambda function written in Python using OpenTelemetry, and then export the data to Splunk Observability Cloud. We'll use Python 3.9 for this example, but the steps for earlier Python versions are similar. The example also uses the AWS Serverless Application Model (SAM) CLI to deploy the Lambda function and an associated API Gateway to access it.
The following tools are required to deploy Python functions into AWS Lambda using SAM:
- An AWS account with permissions to create and execute Lambda functions
- Python 3.9
- Download and install AWS SAM
If you just want to build and deploy the example, feel free to skip this section.
The application used here is based on the "Hello World" example application that's part of the AWS Quick Start templates.
The provided template.yaml.base file was then updated to set the AWS_LAMBDA_EXEC_WRAPPER
environment
variable to /opt/otel-instrument
, since our example lambda function is wrapped with API Gateway:
Environment: # More info about Env Vars: https://github.com/awslabs/serverless-application-model/blob/master/versions/2016-10-31.md#environment-object
Variables:
...
AWS_LAMBDA_EXEC_WRAPPER: /opt/otel-instrument
Please refer to Configure the OpenTelemetry Lambda Layer
which includes alternate options for AWS_LAMBDA_EXEC_WRAPPER
.
We also added the SPLUNK_ACCESS_TOKEN and SPLUNK_REALM environment variables to the template.yaml.base file, as well as multiple layers which provide the instrumentation, as discussed below.
Open a command line terminal and navigate to the root of the directory.
For example:
cd ~/splunk-opentelemetry-examples/instrumentation/python/aws-lambda
export AWS_ACCESS_KEY_ID="<put the access key ID here>"
export AWS_SECRET_ACCESS_KEY="<put the secret access key here>"
export AWS_SESSION_TOKEN="<put the session token here>"
Let's first make a copy of the template.yaml.base file:
cp template.yaml.base template.yaml
Lookup the ARN for your region, language, and CPU architecture in Step 3 in this document.
Add the ARN to the layers section of the template.yaml file. For example, here's the ARN for us-west-1 and Python x86_64:
Layers:
- arn:aws:lambda:us-west-1:254067382080:layer:splunk-apm-python:10
Our example deploys the Splunk distribution of the OpenTelemetry collector to a separate layer within the lambda function. Lookup the ARN for your region in Step 6 in this document.
and add the ARN there. For example, here's the ARN for us-west-1:
Layers:
- arn:aws:lambda:us-west-1:254067382080:layer:splunk-apm-python:10
- arn:aws:lambda:us-west-1:254067382080:layer:splunk-apm-collector:10
Optionally, we can also add the Splunk Metrics Extension Layer to the template.yaml file. Lookup the ARN for your region in Step 7 in this document.
Layers:
- arn:aws:lambda:us-west-1:254067382080:layer:splunk-apm-python:10
- arn:aws:lambda:us-west-1:254067382080:layer:splunk-apm-collector:10
- arn:aws:lambda:us-west-1:254067382080:layer:splunk-lambda-metrics:10
We'll also need to specify the realm and access token for the target Splunk Observability Cloud environment. This goes in the template.yaml file as well:
Environment:
Variables:
SPLUNK_ACCESS_TOKEN: <access token>
SPLUNK_REALM: us1
OTEL_RESOURCE_ATTRIBUTES: deployment.environment=test
Next, we'll build the function using SAM:
sam build
Then deploy it:
sam deploy --guided
You'll be asked a number of questions along the way. Here are sample responses, but you should provide the desired stack name and AWS region for your lambda function.
Setting default arguments for 'sam deploy'
=========================================
Stack Name [sam-app]: aws-lambda-python-opentelemetry-example
AWS Region [us-west-1]: us-west-1
#Shows you resources changes to be deployed and require a 'Y' to initiate deploy
Confirm changes before deploy [y/N]: y
#SAM needs permission to be able to create roles to connect to the resources in your template
Allow SAM CLI IAM role creation [Y/n]: y
#Preserves the state of previously provisioned resources when an operation fails
Disable rollback [y/N]: n
HelloWorldFunction has no authentication. Is this okay? [y/N]: y
Save arguments to configuration file [Y/n]: y
SAM configuration file [samconfig.toml]:
SAM configuration environment [default]:
It will take a few moments for SAM to create all of the objects necessary to support your lambda function. Once it's ready, it will provide you with an API Gateway Endpoint URL that uses the following format:
https://${ServerlessRestApi}.execute-api.${AWS::Region}.amazonaws.com/Prod/hello/
Use the API Gateway Endpoint URL provided in the previous step to test the SAM function. You should see a response such as the following:
{"message": "hello world", "location": "54.219.240.80"}
After a minute or so, you should start to see traces for the lambda function appearing in Splunk Observability Cloud:
If you added the Splunk Metrics Extension Layer, you'll also see metrics for your lambda function by navigating to Infrastructure -> Lambda functions (OTel) and then selecting your lambda function:
Logs generated by an AWS Lambda function get sent to AWS CloudWatch. Various methods exist for ingesting logs into Splunk platform from AWS CloudWatch, such as the solution described in Stream Amazon CloudWatch Logs to Splunk Using AWS Lambda.
Once the logs are in Splunk platform, they can be made available to Splunk Observability Cloud using Log Observer Connect.
To ensure full correlation between traces generated by AWS Lambda instrumentation with metrics and logs, the Splunk Distribution of OpenTelemetry Python automatically adds trace context to log entries, such as in the following example:
2024-10-24 18:35:22,856 WARNING [app] [app.py:18] [trace_id=e360e2e4502da1977091485e1587dfcd span_id=f233ae269ef43749 resource.service.name=aws-lambda-python-opentelemetry-example trace_sampled=True] - Successfully got the IP address, returning a response.
To ensure trace context is injected into log entries, we had to set the following environment variable in the template.yaml file:
OTEL_PYTHON_LOG_CORRELATION: true
We also had to include the following package in the requirements.txt file:
opentelemetry.instrumentation.logging