@dev7a/lambda-otel-lite v0.15.0
Lambda OTel Lite
The @dev7a/lambda-otel-lite
package provides a lightweight, efficient OpenTelemetry implementation specifically designed for AWS Lambda environments in Node.js. 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 package enables efficient telemetry collection with minimal impact on function latency. By default, it uses the @dev7a/otlp-stdout-span-exporter
to export spans to CloudWatch Logs, where they can be collected and forwarded by the serverless-otlp-forwarder.
!IMPORTANT This package is highly experimental and should not be used in production. Contributions are welcome.
Table of Contents
- Requirements
- Features
- Architecture and Modules
- Installation
- Quick Start
- Processing Modes
- Telemetry Configuration
- Event Extractors
- Environment Variables
- License
- See Also
Requirements
- Node.js >= 18.0.0
Features
- Flexible Processing Modes: Support for synchronous, asynchronous, and custom export strategies
- Automatic Resource Detection: Automatic extraction of Lambda environment attributes
- Lambda Extension Integration: Built-in extension for efficient telemetry export
- Efficient Memory Usage: Fixed-size queue to prevent memory growth
- AWS Event Support: Automatic extraction of attributes from common AWS event types (API Gateway v1/v2, ALB)
- Flexible Context Propagation: Support for W3C Trace Context, AWS X-Ray, and custom propagators. Now supports configuration via the
OTEL_PROPAGATORS
environment variable (comma-separated list:tracecontext
,xray
,xray-lambda
,none
).
Architecture and Modules
The package follows a modular architecture where each component has a specific responsibility while working together efficiently:
graph TD
A[telemetry] --> B[processor]
A --> C[extension]
B --> C
E[extractors] --> F[handler]
F --> B
A --> F
H[resource attributes] --> B
A --> H
telemetry
: Core initialization and configuration- Main entry point via
initTelemetry
- Configures global tracer and span processors
- Returns a
TelemetryCompletionHandler
for span lifecycle management
- Main entry point via
processor
: Lambda-optimized span processor- Fixed-size queue implementation
- Multiple processing modes
- Coordinates with extension for async export
extension
: Lambda Extension implementation- Manages extension lifecycle and registration
- Handles span export coordination
- Implements graceful shutdown
extractors
: Event processing- Built-in support for API Gateway and ALB events
- Extensible interface for custom events
- W3C Trace Context propagation
Installation
# Install the base package
npm install --save @dev7a/lambda-otel-lite
# Recommended for type safety with event handlers:
npm install --save-dev @types/aws-lambda
# Optional: For OTLP HTTP export support
npm install --save @opentelemetry/exporter-trace-otlp-http
Quick Start
import { trace, StatusCode } from '@opentelemetry/api';
import { initTelemetry, createTracedHandler } from '@dev7a/lambda-otel-lite';
import { apiGatewayV2Extractor } from '@dev7a/lambda-otel-lite/extractors';
import type { APIGatewayProxyEventV2, Context } from 'aws-lambda'; // Import types
// Initialize telemetry once, outside the handler
const { tracer, completionHandler } = initTelemetry();
// Create traced handler with specific extractor
// You can explicitly specify the event and result types for better type safety
const traced = createTracedHandler<APIGatewayProxyEventV2, APIGatewayProxyResultV2>(
'my-api-handler',
completionHandler,
apiGatewayV2Extractor
);
// Define business logic separately
async function processUser(userId) {
// Your business logic here
return { name: 'User Name', id: userId };
}
// Create the Lambda handler with types
export const handler = traced(async (event: APIGatewayProxyEventV2, context: Context) => {
try {
// Get current span to add custom attributes
const currentSpan = trace.getActiveSpan();
currentSpan?.setAttribute('handler.version', '1.0');
// Extract userId from event path parameters
const userId = event.pathParameters?.userId || 'unknown';
currentSpan?.setAttribute('user.id', userId);
// Process business logic
const user = await processUser(userId);
// Return formatted HTTP response
return {
statusCode: 200,
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ success: true, data: user })
};
} catch (error) {
// Simple error handling
return {
statusCode: 500,
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({
success: false,
error: 'Internal server error'
})
};
}
});
Processing Modes
The package supports three processing modes for span export:
Sync Mode (default):
- Direct, synchronous export in handler thread
- Recommended for:
- low-volume telemetry
- limited resources (memory, cpu)
- when latency is not critical
- Set via
LAMBDA_EXTENSION_SPAN_PROCESSOR_MODE=sync
Async Mode:
- Export via Lambda extension using AWS Lambda Extensions API
- Spans are queued and exported after handler completion
- Uses event-based communication between handler and extension
- Registers specifically for Lambda INVOKE events
- Implements graceful shutdown with SIGTERM handling
- Error handling for:
- Event communication failures
- Export failures
- Extension registration issues
- Best for production use with high telemetry volume
- Set via
LAMBDA_EXTENSION_SPAN_PROCESSOR_MODE=async
- Requires extension initialization:
NODE_OPTIONS=--require @dev7a/lambda-otel-lite/extension
Finalize Mode:
- Registers extension with no events
- Maintains SIGTERM handler for graceful shutdown
- Ensures all spans are flushed during shutdown
- Compatible with BatchSpanProcessor for custom export strategies
- Best for specialized export requirements where you need full control
- Set via
LAMBDA_EXTENSION_SPAN_PROCESSOR_MODE=finalize
Async Processing Mode Architecture
sequenceDiagram
participant Lambda Runtime
participant Extension
participant Event Emitter
participant Handler
participant LambdaSpanProcessor
participant OTLPStdoutSpanExporter
Note over Extension: Initialization
Extension->>Lambda Runtime: Register extension (POST /register)
Lambda Runtime-->>Extension: Extension ID
Extension->>Lambda Runtime: Request next event (GET /next)
Note over Handler: Function execution starts
Handler->>LambdaSpanProcessor: Add spans during execution
Note over LambdaSpanProcessor: Spans stored in fixed-size queue
Handler->>Event Emitter: Emit 'handlerComplete'
Note over Handler: Handler returns response
Event Emitter->>Extension: Handle completion
Extension->>LambdaSpanProcessor: Force flush spans
LambdaSpanProcessor->>OTLPStdoutSpanExporter: Export spans
Extension->>Lambda Runtime: Request next event
Note over Extension: On SIGTERM
Lambda Runtime->>Extension: SHUTDOWN event
Extension->>LambdaSpanProcessor: Force flush spans
LambdaSpanProcessor->>OTLPStdoutSpanExporter: Export remaining spans
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. The extension is loaded via Node.js's --require
flag and uses an event-driven architecture:
- The extension is loaded at startup via
--require
flag - Uses Node.js's built-in
http
module for Lambda API communication - Leverages event emitters for handler completion notification
- Single event loop handles both extension and handler code
- SIGTERM handler ensures graceful shutdown with span flushing
Telemetry Configuration
The package provides several ways to configure the OpenTelemetry tracing pipeline, which is a required first step to instrument your Lambda function:
Custom configuration with custom resource attributes
import { Resource } from '@opentelemetry/resources';
import { initTelemetry, createTracedHandler } from '@dev7a/lambda-otel-lite';
// Create a custom resource with additional attributes
const resource = new Resource({
'service.version': '1.0.0',
'deployment.environment': 'production',
'custom.attribute': 'value'
});
// Initialize with custom resource
const { tracer, completionHandler } = initTelemetry({
resource
});
// Use the tracer and completion handler as usual
Custom configuration with custom span processors
import { initTelemetry, createTracedHandler } from '@dev7a/lambda-otel-lite';
import { BatchSpanProcessor } from '@opentelemetry/sdk-trace-base';
import { OTLPTraceExporter } from '@opentelemetry/exporter-trace-otlp-http';
// Create a custom processor with OTLP HTTP exporter
const processor = new BatchSpanProcessor(
new OTLPTraceExporter({
url: 'https://your-otlp-endpoint/v1/traces'
})
);
// Initialize with custom processor
const { tracer, completionHandler } = initTelemetry({
spanProcessors: [processor]
});
You can provide multiple span processors, and they will all be used to process spans. This allows you to send telemetry to multiple destinations or use different processing strategies for different types of spans.
Custom configuration with context propagators
You can now also configure context propagation using the OTEL_PROPAGATORS
environment variable, which takes precedence over the propagators
option. Supported values: tracecontext
, xray
, xray-lambda
, none
(comma-separated for multiple). For example:
export OTEL_PROPAGATORS="xray,tracecontext"
If neither the environment variable nor the option is set, the default is [LambdaXRayPropagator(), W3CTraceContextPropagator()]
.
import { initTelemetry } from '@dev7a/lambda-otel-lite';
import { W3CTraceContextPropagator } from '@opentelemetry/core';
import { B3Propagator } from '@opentelemetry/propagator-b3';
// Initialize with custom propagators
const { tracer, completionHandler } = initTelemetry({
propagators: [
new W3CTraceContextPropagator(), // W3C Trace Context
new B3Propagator(), // B3 format for Zipkin compatibility
]
});
By default, OpenTelemetry Node.js uses W3C Trace Context and W3C Baggage propagators. The propagators
parameter allows you to customize which propagators are used for context extraction and injection. This is useful when you need to integrate with systems that use different context propagation formats.
You can provide multiple propagators, and they will be combined into a composite propagator. The order matters - propagators are applied in the order they are provided.
Note: The OpenTelemetry SDK also supports configuring propagators via the
OTEL_PROPAGATORS
environment variable. If set, this environment variable takes precedence over programmatic configuration. See the OpenTelemetry JavaScript documentation for more details.
Custom configuration with ID generator
import { initTelemetry } from '@dev7a/lambda-otel-lite';
import { AWSXRayIdGenerator } from '@opentelemetry/id-generator-aws-xray';
// Initialize with X-Ray compatible ID generator
const { tracer, completionHandler } = initTelemetry({
idGenerator: new AWSXRayIdGenerator()
});
By default, OpenTelemetry uses a random ID generator that creates W3C-compatible trace and span IDs. The idGenerator
parameter allows you to customize the ID generation strategy. This is particularly useful when you need to integrate with AWS X-Ray, which requires a specific ID format.
To use the X-Ray ID generator, you'll need to install the OpenTelemetry X-Ray ID generator package:
npm install --save @opentelemetry/id-generator-aws-xray
Library specific Resource Attributes
The package adds several resource attributes under the lambda_otel_lite
namespace to provide configuration visibility when the corresponding environment variables are explicitly set:
lambda_otel_lite.extension.span_processor_mode
: Current processing mode (sync
,async
, orfinalize
)lambda_otel_lite.lambda_span_processor.queue_size
: Maximum number of spans that can be queuedlambda_otel_lite.otlp_stdout_span_exporter.compression_level
: GZIP compression level used for span export
These attributes are only added to the resource when the corresponding environment variables are explicitly set. This ensures that the recorded resource attributes accurately reflect the actual configuration values used by the components.
For example, if you set LAMBDA_SPAN_PROCESSOR_QUEUE_SIZE=4096
, then the resource attribute lambda_otel_lite.lambda_span_processor.queue_size
will be set to 4096
. If you don't set this environment variable, the attribute won't be present in the resource, even though the component may be using a default or constructor-provided value.
Event Extractors
Event extractors are responsible for extracting span attributes and context from Lambda event and context objects. The library provides built-in extractors for common Lambda triggers.
Automatic Attributes extraction
The library automatically sets relevant FAAS attributes based on the Lambda context and event. Both event
and context
parameters must be passed to tracedHandler
to enable all automatic attributes:
Resource Attributes (set at initialization):
cloud.provider
: "aws"cloud.region
: from AWS_REGIONfaas.name
: from AWS_LAMBDA_FUNCTION_NAMEfaas.version
: from AWS_LAMBDA_FUNCTION_VERSIONfaas.instance
: from AWS_LAMBDA_LOG_STREAM_NAMEfaas.max_memory
: from AWS_LAMBDA_FUNCTION_MEMORY_SIZEservice.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 invocationcloud.account.id
: extracted from context's invokedFunctionArnfaas.invocation_id
: from awsRequestIdcloud.resource_id
: from context's invokedFunctionArn
HTTP Attributes (set for API Gateway events):
faas.trigger
: "http"http.status_code
: from handler responsehttp.route
: from routeKey (v2) or resource (v1)http.method
: from requestContext (v2) or httpMethod (v1)http.target
: from pathhttp.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. For improved type safety when using these extractors, it's recommended to install @types/aws-lambda
as a dev dependency and type your handler's event
and context
parameters accordingly.
Built-in Extractors
You can import extractors in two ways:
// Method 1: Import from the dedicated extractors subpath (recommended)
import {
apiGatewayV1Extractor, // API Gateway REST API
apiGatewayV2Extractor, // API Gateway HTTP API
albExtractor, // Application Load Balancer
defaultExtractor // Basic Lambda attributes
} from '@dev7a/lambda-otel-lite/extractors';
// Method 2: Import from the main package
import {
apiGatewayV1Extractor,
apiGatewayV2Extractor,
albExtractor
} from '@dev7a/lambda-otel-lite';
Each extractor is designed to handle a specific event type and extract relevant attributes:
apiGatewayV1Extractor
: Extracts HTTP attributes from API Gateway REST API eventsapiGatewayV2Extractor
: Extracts HTTP attributes from API Gateway HTTP API eventsalbExtractor
: Extracts HTTP attributes from Application Load Balancer eventsdefaultExtractor
: Extracts basic Lambda attributes from any event type
import { apiGatewayV1Extractor } from '@dev7a/lambda-otel-lite/extractors';
// Initialize telemetry with default configuration
const { tracer, completionHandler } = initTelemetry();
export const traced = createTracedHandler(
'api-v1-handler',
completionHandler,
apiGatewayV1Extractor
);
import type { APIGatewayProxyEvent, Context } from 'aws-lambda'; // Import types
export const handler = traced(async (event: APIGatewayProxyEvent, context: Context) => {
// Your handler code using typed event and context
return {
statusCode: 200,
body: JSON.stringify({ message: 'Hello, world!' })
};
});
Custom Extractors
You can create custom extractors for event types not directly supported by the library by implementing the extractor interface:
// Import types from the main package or extractors path
import { type AttributesExtractor } from '@dev7a/lambda-otel-lite'; // Import from main package
import { type SpanAttributes, TriggerType } from '@dev7a/lambda-otel-lite/extractors';
import type { SQSEvent, Context } from 'aws-lambda';
// Example for an SQS event with proper typing
const customSqsExtractor: AttributesExtractor<SQSEvent> = (event, context) => {
const attributes = {
'messaging.system': 'AmazonSQS',
'messaging.operation': 'process',
'messaging.source.kind': 'queue',
// Extract message count or other relevant attributes
'messaging.message.count': event.Records?.length || 0,
};
// Example: Extract trace context from the first message's attributes if available
const firstRecordAttributes = event.Records?.[0]?.messageAttributes;
const carrier = firstRecordAttributes ?
Object.entries(firstRecordAttributes).reduce((acc, [key, value]) => {
if (value.stringValue) {
acc[key.toLowerCase()] = value.stringValue; // Normalize keys
}
return acc;
}, {} as Record<string, string>)
: undefined;
return {
trigger: TriggerType.Messaging, // Use appropriate trigger type
attributes,
spanName: 'process-sqs-message', // Optional custom span name
carrier, // Optional carrier for context propagation
};
};
// Create a traced handler with explicit type parameters
export const traced = createTracedHandler<SQSEvent, any>(
'custom-handler',
completionHandler,
customSqsExtractor
);
import type { SQSEvent, Context } from 'aws-lambda'; // Import types
// The handler function with proper typing
export const handler = traced(async (event: SQSEvent, context: Context) => {
// Your handler code using typed event and context
return {
statusCode: 200,
body: JSON.stringify({ message: 'Hello, world!' })
};
});
The SpanAttributes
object returned by the extractor contains:
trigger
: The type of trigger (e.g.,TriggerType.Http
,TriggerType.Messaging
,TriggerType.Other
or a custom string) - affects how spans are named. UseTriggerType
enum imported from@dev7a/lambda-otel-lite/extractors
.attributes
: An object containing key-value pairs to add as attributes to the main handler span.spanName
: Optional custom name for the main handler span (defaults to the handler name provided tocreateTracedHandler
).carrier
: Optional object containing trace context headers (e.g.,traceparent
,tracestate
,x-amzn-trace-id
). Keys should be lowercase. This is used for extracting the parent span context.kind
: OptionalSpanKind
(defaults toSpanKind.SERVER
). ImportSpanKind
from@opentelemetry/api
.
Environment Variables
The package can be configured using the following environment variables. All configuration follows a strict precedence order:
- Environment variables (highest precedence)
- Constructor parameters
- Default values (lowest precedence)
Processing Configuration
LAMBDA_EXTENSION_SPAN_PROCESSOR_MODE
: Controls span processing strategysync
: Direct export in handler thread (default)async
: Deferred export via extensionfinalize
: Custom export strategy
LAMBDA_SPAN_PROCESSOR_QUEUE_SIZE
: Maximum number of spans to queue (default: 2048)
Resource Configuration
OTEL_SERVICE_NAME
: Override the service name (defaults to function name)OTEL_RESOURCE_ATTRIBUTES
: Additional resource attributes in key=value,key2=value2 format
Export Configuration
OTLP_STDOUT_SPAN_EXPORTER_COMPRESSION_LEVEL
: Gzip compression level for stdout exporter- 0: No compression
- 1: Best speed
- 6: Good balance between size and speed (default)
- 9: Best compression
Note: If you set any of the environment variables for processing configuration or export configuration, the corresponding resource attribute will be set to reflect this configuration value. If you don't set an environment variable, the package will use the constructor parameter if provided, or fall back to the default value, but the resource attribute will not be set.
Logging
AWS_LAMBDA_LOG_LEVEL
orLOG_LEVEL
: Configure log level (debug, info, warn, error, none)
AWS Lambda Environment
The following AWS Lambda environment variables are automatically used for resource attributes:
AWS_REGION
: Region where function runsAWS_LAMBDA_FUNCTION_NAME
: Function nameAWS_LAMBDA_FUNCTION_VERSION
: Function versionAWS_LAMBDA_LOG_STREAM_NAME
: Log stream nameAWS_LAMBDA_FUNCTION_MEMORY_SIZE
: Function memory sizeOTEL_PROPAGATORS
: Comma-separated list of propagators to use for context propagation. Supported:tracecontext
,xray
,xray-lambda
,none
. Takes precedence over programmatic configuration.
License
See Also
6 months ago
6 months ago
7 months ago
7 months ago
7 months ago
7 months ago
7 months ago
7 months ago
8 months ago
8 months ago
8 months ago
8 months ago
8 months ago
8 months ago
8 months ago
8 months ago
8 months ago
9 months ago
9 months ago
9 months ago
10 months ago