# Lambda OTel Lite
The `lambda-otel-lite` library provides a lightweight, efficient OpenTelemetry implementation specifically designed for AWS Lambda environments. It features a custom span processor and internal extension mechanism that optimizes telemetry collection for Lambda's unique execution model.
By leveraging Lambda's execution lifecycle and providing multiple processing modes, this library enables efficient telemetry collection with minimal impact on function latency. By default, it uses the [otlp-stdout-span-exporter](https://pypi.org/project/otlp-stdout-span-exporter) to export spans to stdout for the [serverless-otlp-forwarder](https://github.com/dev7a/serverless-otlp-forwarder) project.
>[!IMPORTANT]
>This package is highly experimental and should not be used in production. Contributions are welcome.
## Features
- Lambda-optimized span processor with queue-based buffering
- Three processing modes for different use cases:
- Synchronous: Immediate span export
- Asynchronous: Background processing via internal extension
- Finalize: Compatible with standard BatchSpanProcessor
- Internal extension thread for asynchronous mode
- Sigterm handler for asynchronous and finalize mode
- Automatic Lambda resource detection
- Automatic FAAS attributes from Lambda context and events (HTTP only)
- Cold start detection and tracking
- Configurable through environment variables
- Optimized for cold start performance
## Installation
We recommend using `uv` for faster and more reliable package installation:
```bash
uv venv && source .venv/bin/activate && uv pip install lambda-otel-lite
```
## Usage
### Basic Usage
```python
from lambda_otel_lite import init_telemetry, traced_handler
from opentelemetry.trace import SpanKind
# Initialize telemetry once at module load
tracer, provider = init_telemetry(name="my-service")
def handler(event, context):
with traced_handler(
tracer=tracer,
tracer_provider=provider,
name="my-handler",
kind=SpanKind.SERVER,
event=event, # Optional: Enables automatic FAAS attributes for HTTP events
context=context, # Optional: Enables automatic FAAS attributes from context
):
# Your handler code here
process_event(event)
return {"statusCode": 200}
def process_event(event):
with tracer.start_as_current_span("process_event") as span:
span.set_attribute("event.type", event.get("type"))
# Process the event
```
### Automatic FAAS Attributes
The library automatically sets relevant FAAS attributes based on the Lambda context and event. Both `event` and `context` parameters must be passed to `traced_handler` to enable all automatic attributes:
- Resource Attributes (set at initialization):
- `cloud.provider`: "aws"
- `cloud.region`: from AWS_REGION
- `faas.name`: from AWS_LAMBDA_FUNCTION_NAME
- `faas.version`: from AWS_LAMBDA_FUNCTION_VERSION
- `faas.instance`: from AWS_LAMBDA_LOG_STREAM_NAME
- `faas.max_memory`: from AWS_LAMBDA_FUNCTION_MEMORY_SIZE
- `service.name`: from OTEL_SERVICE_NAME (defaults to function name)
- Additional attributes from OTEL_RESOURCE_ATTRIBUTES (URL-decoded)
- Span Attributes (set per invocation when passing context):
- `faas.cold_start`: true on first invocation
- `cloud.account.id`: extracted from context's invoked_function_arn
- `faas.invocation_id`: from context's aws_request_id
- `cloud.resource_id`: from context's invoked_function_arn
- HTTP Attributes (set for API Gateway events):
- `faas.trigger`: "http"
- `http.status_code`: from handler response
- `http.route`: from routeKey (v2) or resource (v1)
- `http.method`: from requestContext (v2) or httpMethod (v1)
- `http.target`: from path
- `http.scheme`: from protocol
The library automatically detects API Gateway v1 and v2 events and sets the appropriate HTTP attributes. For HTTP responses, the status code is automatically extracted from the handler's response and set as `http.status_code`. For 5xx responses, the span status is set to ERROR.
### Distributed Tracing
The library supports distributed tracing across service boundaries. Context propagation is handled automatically when you pass the `event` parameter and it contains a `headers` property. You can also provide a custom carrier extraction function for more complex scenarios:
```python
from lambda_otel_lite import init_telemetry, traced_handler
from opentelemetry.trace import SpanKind
# Initialize telemetry once at module load
tracer, provider = init_telemetry(name="my-service")
def handler(event, context):
# Context propagation is handled automatically if event has 'headers'
with traced_handler(
tracer=tracer,
tracer_provider=provider,
name="my-handler",
kind=SpanKind.SERVER,
event=event, # Will automatically extract context from event['headers'] if present
context=context,
attributes={"custom.attribute": "value"}
):
# Your handler code here
return {"statusCode": 200}
# For custom carrier extraction:
def extract_from_sqs(event: dict) -> dict:
"""Extract carrier from SQS event."""
# Extract tracing headers from the first record's message attributes
if "Records" in event and event["Records"]:
return event["Records"][0].get("messageAttributes", {})
return {}
def handler_with_custom_extraction(event, context):
with traced_handler(
tracer=tracer,
tracer_provider=provider,
name="my-handler",
kind=SpanKind.SERVER,
event=event,
context=context,
get_carrier=extract_from_sqs # Custom function to extract carrier from event
):
# Your handler code here
return {"statusCode": 200}
```
Please note that in a real world scenario, for SQS events you should probably use span links instead.
### Custom Telemetry Configuration
You can customize the telemetry setup by providing your own span processors and chaining them:
```python
from opentelemetry.sdk.trace import SpanProcessor
from opentelemetry.sdk.trace.export import BatchSpanProcessor, ConsoleSpanExporter
from lambda_otel_lite import init_telemetry
# Simple processor example
class SimpleProcessor(SpanProcessor):
def on_start(self, span, parent_context=None):
# Add attribute when span starts
span.set_attribute("example.timestamp", time.time())
tracer, provider = init_telemetry(
"my-lambda-function",
span_processors=[
SimpleProcessor(), # First add attributes
BatchSpanProcessor( # Then export spans
ConsoleSpanExporter()
)
]
)
```
If no processors are provided, the library defaults to using a `LambdaSpanProcessor` with `OTLPStdoutSpanExporter` for integration with the serverless-otlp-forwarder.
## Processing Modes
The library supports three processing modes, controlled by the `LAMBDA_EXTENSION_SPAN_PROCESSOR_MODE` environment variable:
1. **Synchronous Mode** (`sync`, default)
- Spans are exported immediately in the handler thread
- Best for development and debugging
- Highest latency but immediate span visibility
- Does not install the internal extension thread and the sigterm handler
2. **Asynchronous Mode** (`async`)
- Spans are queued and processed by the internal extension thread
- Export occurs after handler completion
- Best for production use
- Minimal impact on handler latency
- Installs the sigterm handler to flush remaining spans on termination
3. **Finalize Mode** (`finalize`)
- Installs only the sigterm handler to flush remaining spans on termination
- Typically used with the BatchSpanProcessor from the OpenTelemetry SDK for periodic flushes
### Async Mode Architecture
The async mode leverages Lambda's extension API to optimize perceived latency by deferring span export until after the response is sent to the user. Here's how it works:
```mermaid
sequenceDiagram
participant Lambda Runtime
participant Extension Thread
participant Handler
participant Span Queue
participant OTLP Exporter
Note over Extension Thread: Started by init_extension()
Extension Thread->>Lambda Runtime: Register extension (POST /register)
Lambda Runtime-->>Extension Thread: Extension ID
loop For each invocation
Extension Thread->>Lambda Runtime: Get next event (GET /next)
Lambda Runtime-->>Extension Thread: INVOKE event
Note over Extension Thread: Wait for handler_complete
Handler->>Span Queue: Add spans during execution
Handler->>Extension Thread: Signal completion (handler_complete.set())
Extension Thread->>Span Queue: Flush spans
Extension Thread->>OTLP Exporter: Export spans
Note over Extension Thread: Lock handler_complete for next invocation
end
Note over Extension Thread: On SIGTERM
Lambda Runtime->>Extension Thread: SHUTDOWN event
Extension Thread->>Span Queue: Final flush
Extension Thread->>OTLP Exporter: Export remaining spans
```
The internal extension thread coordinates with the handler using a single threading.Event:
1. Extension thread starts in a waiting state (event is locked)
2. Handler executes and adds spans to the queue
3. When handler completes, it signals the extension (unlocks the event)
4. Extension processes spans and locks the event for the next invocation
5. On shutdown, any remaining spans are flushed and exported
This architecture ensures that span export doesn't impact the handler's response time while maintaining reliable telemetry delivery.
## Environment Variables
The library can be configured using the following environment variables:
- `LAMBDA_EXTENSION_SPAN_PROCESSOR_MODE`: Processing mode (`sync`, `async`, or `finalize`, defaults to `sync`)
- `LAMBDA_SPAN_PROCESSOR_QUEUE_SIZE`: Maximum number of spans to queue in the Lambda span processor (default: 2048)
- `OTEL_SERVICE_NAME`: Override the service name (defaults to function name)
- `OTEL_RESOURCE_ATTRIBUTES`: Additional resource attributes in key=value,key2=value2 format (URL-decoded values supported)
- `OTLP_STDOUT_SPAN_EXPORTER_COMPRESSION_LEVEL`: Gzip compression level for stdout exporter (0-9, default: 6)
- 0: No compression
- 1: Best speed
- 6: Good balance between size and speed (default)
- 9: Best compression
## License
This project is licensed under the MIT License - see the [LICENSE](LICENSE) file for details.
Raw data
{
"_id": null,
"home_page": null,
"name": "lambda-otel-lite",
"maintainer": null,
"docs_url": null,
"requires_python": ">=3.12",
"maintainer_email": null,
"keywords": "aws, lambda, opentelemetry, otel, tracing",
"author": null,
"author_email": "Alessandro Bologna <alessandro.bologna@gmail.com>",
"download_url": "https://files.pythonhosted.org/packages/84/9e/04af92dd2e1eed8fe7b9a5e9b89dd1c6b1914b1685eefaafb9fdb5b7b58f/lambda_otel_lite-0.5.1.tar.gz",
"platform": null,
"description": "# Lambda OTel Lite\n\nThe `lambda-otel-lite` library provides a lightweight, efficient OpenTelemetry implementation specifically designed for AWS Lambda environments. It features a custom span processor and internal extension mechanism that optimizes telemetry collection for Lambda's unique execution model.\n\nBy leveraging Lambda's execution lifecycle and providing multiple processing modes, this library enables efficient telemetry collection with minimal impact on function latency. By default, it uses the [otlp-stdout-span-exporter](https://pypi.org/project/otlp-stdout-span-exporter) to export spans to stdout for the [serverless-otlp-forwarder](https://github.com/dev7a/serverless-otlp-forwarder) project.\n\n>[!IMPORTANT]\n>This package is highly experimental and should not be used in production. Contributions are welcome.\n\n## Features\n\n- Lambda-optimized span processor with queue-based buffering\n- Three processing modes for different use cases:\n - Synchronous: Immediate span export\n - Asynchronous: Background processing via internal extension\n - Finalize: Compatible with standard BatchSpanProcessor\n- Internal extension thread for asynchronous mode\n- Sigterm handler for asynchronous and finalize mode\n- Automatic Lambda resource detection\n- Automatic FAAS attributes from Lambda context and events (HTTP only)\n- Cold start detection and tracking\n- Configurable through environment variables\n- Optimized for cold start performance\n\n\n## Installation\n\nWe recommend using `uv` for faster and more reliable package installation:\n\n```bash\nuv venv && source .venv/bin/activate && uv pip install lambda-otel-lite\n```\n\n## Usage\n\n### Basic Usage\n\n```python\nfrom lambda_otel_lite import init_telemetry, traced_handler\nfrom opentelemetry.trace import SpanKind\n\n# Initialize telemetry once at module load\ntracer, provider = init_telemetry(name=\"my-service\")\n\ndef handler(event, context):\n with traced_handler(\n tracer=tracer,\n tracer_provider=provider,\n name=\"my-handler\",\n kind=SpanKind.SERVER,\n event=event, # Optional: Enables automatic FAAS attributes for HTTP events\n context=context, # Optional: Enables automatic FAAS attributes from context\n ):\n # Your handler code here\n process_event(event)\n return {\"statusCode\": 200}\n\ndef process_event(event):\n with tracer.start_as_current_span(\"process_event\") as span:\n span.set_attribute(\"event.type\", event.get(\"type\"))\n # Process the event\n```\n\n### Automatic FAAS Attributes\n\nThe library automatically sets relevant FAAS attributes based on the Lambda context and event. Both `event` and `context` parameters must be passed to `traced_handler` to enable all automatic attributes:\n\n- Resource Attributes (set at initialization):\n - `cloud.provider`: \"aws\"\n - `cloud.region`: from AWS_REGION\n - `faas.name`: from AWS_LAMBDA_FUNCTION_NAME\n - `faas.version`: from AWS_LAMBDA_FUNCTION_VERSION\n - `faas.instance`: from AWS_LAMBDA_LOG_STREAM_NAME\n - `faas.max_memory`: from AWS_LAMBDA_FUNCTION_MEMORY_SIZE\n - `service.name`: from OTEL_SERVICE_NAME (defaults to function name)\n - Additional attributes from OTEL_RESOURCE_ATTRIBUTES (URL-decoded)\n\n- Span Attributes (set per invocation when passing context):\n - `faas.cold_start`: true on first invocation\n - `cloud.account.id`: extracted from context's invoked_function_arn\n - `faas.invocation_id`: from context's aws_request_id\n - `cloud.resource_id`: from context's invoked_function_arn\n\n- HTTP Attributes (set for API Gateway events):\n - `faas.trigger`: \"http\"\n - `http.status_code`: from handler response\n - `http.route`: from routeKey (v2) or resource (v1)\n - `http.method`: from requestContext (v2) or httpMethod (v1)\n - `http.target`: from path\n - `http.scheme`: from protocol\n\nThe library automatically detects API Gateway v1 and v2 events and sets the appropriate HTTP attributes. For HTTP responses, the status code is automatically extracted from the handler's response and set as `http.status_code`. For 5xx responses, the span status is set to ERROR.\n\n### Distributed Tracing\n\nThe library supports distributed tracing across service boundaries. Context propagation is handled automatically when you pass the `event` parameter and it contains a `headers` property. You can also provide a custom carrier extraction function for more complex scenarios:\n\n```python\nfrom lambda_otel_lite import init_telemetry, traced_handler\nfrom opentelemetry.trace import SpanKind\n\n# Initialize telemetry once at module load\ntracer, provider = init_telemetry(name=\"my-service\")\n\ndef handler(event, context):\n # Context propagation is handled automatically if event has 'headers'\n with traced_handler(\n tracer=tracer,\n tracer_provider=provider,\n name=\"my-handler\",\n kind=SpanKind.SERVER,\n event=event, # Will automatically extract context from event['headers'] if present\n context=context,\n attributes={\"custom.attribute\": \"value\"}\n ):\n # Your handler code here\n return {\"statusCode\": 200}\n\n# For custom carrier extraction:\ndef extract_from_sqs(event: dict) -> dict:\n \"\"\"Extract carrier from SQS event.\"\"\"\n # Extract tracing headers from the first record's message attributes\n if \"Records\" in event and event[\"Records\"]:\n return event[\"Records\"][0].get(\"messageAttributes\", {})\n return {}\n\ndef handler_with_custom_extraction(event, context):\n with traced_handler(\n tracer=tracer,\n tracer_provider=provider,\n name=\"my-handler\",\n kind=SpanKind.SERVER,\n event=event,\n context=context,\n get_carrier=extract_from_sqs # Custom function to extract carrier from event\n ):\n # Your handler code here\n return {\"statusCode\": 200}\n```\n\nPlease note that in a real world scenario, for SQS events you should probably use span links instead.\n\n\n### Custom Telemetry Configuration\n\nYou can customize the telemetry setup by providing your own span processors and chaining them:\n\n```python\nfrom opentelemetry.sdk.trace import SpanProcessor\nfrom opentelemetry.sdk.trace.export import BatchSpanProcessor, ConsoleSpanExporter\nfrom lambda_otel_lite import init_telemetry\n\n# Simple processor example\nclass SimpleProcessor(SpanProcessor):\n def on_start(self, span, parent_context=None):\n # Add attribute when span starts\n span.set_attribute(\"example.timestamp\", time.time())\n\ntracer, provider = init_telemetry(\n \"my-lambda-function\",\n span_processors=[\n SimpleProcessor(), # First add attributes\n BatchSpanProcessor( # Then export spans\n ConsoleSpanExporter()\n )\n ]\n)\n```\n\nIf no processors are provided, the library defaults to using a `LambdaSpanProcessor` with `OTLPStdoutSpanExporter` for integration with the serverless-otlp-forwarder.\n\n## Processing Modes\n\nThe library supports three processing modes, controlled by the `LAMBDA_EXTENSION_SPAN_PROCESSOR_MODE` environment variable:\n\n1. **Synchronous Mode** (`sync`, default)\n - Spans are exported immediately in the handler thread\n - Best for development and debugging\n - Highest latency but immediate span visibility\n - Does not install the internal extension thread and the sigterm handler\n\n2. **Asynchronous Mode** (`async`)\n - Spans are queued and processed by the internal extension thread\n - Export occurs after handler completion\n - Best for production use\n - Minimal impact on handler latency\n - Installs the sigterm handler to flush remaining spans on termination\n\n3. **Finalize Mode** (`finalize`)\n - Installs only the sigterm handler to flush remaining spans on termination\n - Typically used with the BatchSpanProcessor from the OpenTelemetry SDK for periodic flushes\n\n### Async Mode Architecture\n\nThe async mode leverages Lambda's extension API to optimize perceived latency by deferring span export until after the response is sent to the user. Here's how it works:\n\n```mermaid\nsequenceDiagram\n participant Lambda Runtime\n participant Extension Thread\n participant Handler\n participant Span Queue\n participant OTLP Exporter\n\n Note over Extension Thread: Started by init_extension()\n Extension Thread->>Lambda Runtime: Register extension (POST /register)\n Lambda Runtime-->>Extension Thread: Extension ID\n\n loop For each invocation\n Extension Thread->>Lambda Runtime: Get next event (GET /next)\n Lambda Runtime-->>Extension Thread: INVOKE event\n Note over Extension Thread: Wait for handler_complete\n Handler->>Span Queue: Add spans during execution\n Handler->>Extension Thread: Signal completion (handler_complete.set())\n Extension Thread->>Span Queue: Flush spans\n Extension Thread->>OTLP Exporter: Export spans\n Note over Extension Thread: Lock handler_complete for next invocation\n end\n\n Note over Extension Thread: On SIGTERM\n Lambda Runtime->>Extension Thread: SHUTDOWN event\n Extension Thread->>Span Queue: Final flush\n Extension Thread->>OTLP Exporter: Export remaining spans\n```\n\nThe internal extension thread coordinates with the handler using a single threading.Event:\n1. Extension thread starts in a waiting state (event is locked)\n2. Handler executes and adds spans to the queue\n3. When handler completes, it signals the extension (unlocks the event)\n4. Extension processes spans and locks the event for the next invocation\n5. On shutdown, any remaining spans are flushed and exported\n\nThis architecture ensures that span export doesn't impact the handler's response time while maintaining reliable telemetry delivery.\n\n## Environment Variables\n\nThe library can be configured using the following environment variables:\n\n- `LAMBDA_EXTENSION_SPAN_PROCESSOR_MODE`: Processing mode (`sync`, `async`, or `finalize`, defaults to `sync`)\n- `LAMBDA_SPAN_PROCESSOR_QUEUE_SIZE`: Maximum number of spans to queue in the Lambda span processor (default: 2048)\n- `OTEL_SERVICE_NAME`: Override the service name (defaults to function name)\n- `OTEL_RESOURCE_ATTRIBUTES`: Additional resource attributes in key=value,key2=value2 format (URL-decoded values supported)\n- `OTLP_STDOUT_SPAN_EXPORTER_COMPRESSION_LEVEL`: Gzip compression level for stdout exporter (0-9, default: 6)\n - 0: No compression\n - 1: Best speed\n - 6: Good balance between size and speed (default)\n - 9: Best compression\n\n## License\n\nThis project is licensed under the MIT License - see the [LICENSE](LICENSE) file for details. ",
"bugtrack_url": null,
"license": "MIT",
"summary": "Lightweight OpenTelemetry instrumentation for AWS Lambda",
"version": "0.5.1",
"project_urls": {
"Homepage": "https://github.com/dev7a/serverless-otlp-forwarder/tree/main/packages/python/lambda_otel_lite",
"Repository": "https://github.com/dev7a/serverless-otlp-forwarder/tree/main/packages/python/lambda_otel_lite"
},
"split_keywords": [
"aws",
" lambda",
" opentelemetry",
" otel",
" tracing"
],
"urls": [
{
"comment_text": "",
"digests": {
"blake2b_256": "91af30d7b748f479ba75eb7417276de7779068bf4c84b5b2d969dcf18dccfe73",
"md5": "f47411606bb41a0b667c7920324763be",
"sha256": "5899d0d589d73d213fea0e4ecbef328050660c3d84e7c3c16042aaf77661aaa5"
},
"downloads": -1,
"filename": "lambda_otel_lite-0.5.1-py3-none-any.whl",
"has_sig": false,
"md5_digest": "f47411606bb41a0b667c7920324763be",
"packagetype": "bdist_wheel",
"python_version": "py3",
"requires_python": ">=3.12",
"size": 14602,
"upload_time": "2025-01-19T02:51:19",
"upload_time_iso_8601": "2025-01-19T02:51:19.166641Z",
"url": "https://files.pythonhosted.org/packages/91/af/30d7b748f479ba75eb7417276de7779068bf4c84b5b2d969dcf18dccfe73/lambda_otel_lite-0.5.1-py3-none-any.whl",
"yanked": false,
"yanked_reason": null
},
{
"comment_text": "",
"digests": {
"blake2b_256": "849e04af92dd2e1eed8fe7b9a5e9b89dd1c6b1914b1685eefaafb9fdb5b7b58f",
"md5": "6f473719763a683a6cbc9c2192ebd15c",
"sha256": "b79b2220bf1d25ffe05a83780edc240fb6bc6a290557f8464641fed0f784acb0"
},
"downloads": -1,
"filename": "lambda_otel_lite-0.5.1.tar.gz",
"has_sig": false,
"md5_digest": "6f473719763a683a6cbc9c2192ebd15c",
"packagetype": "sdist",
"python_version": "source",
"requires_python": ">=3.12",
"size": 11846,
"upload_time": "2025-01-19T02:51:20",
"upload_time_iso_8601": "2025-01-19T02:51:20.238057Z",
"url": "https://files.pythonhosted.org/packages/84/9e/04af92dd2e1eed8fe7b9a5e9b89dd1c6b1914b1685eefaafb9fdb5b7b58f/lambda_otel_lite-0.5.1.tar.gz",
"yanked": false,
"yanked_reason": null
}
],
"upload_time": "2025-01-19 02:51:20",
"github": true,
"gitlab": false,
"bitbucket": false,
"codeberg": false,
"github_user": "dev7a",
"github_project": "serverless-otlp-forwarder",
"travis_ci": false,
"coveralls": false,
"github_actions": true,
"lcname": "lambda-otel-lite"
}