A full setup example is available below.
Overview
The OpenHands SDK provides built-in OpenTelemetry (OTEL) tracing support, allowing you to monitor and debug your agent’s execution in real time. You can send traces to any OTLP-compatible observability platform including:- Laminar - AI-focused observability with trace inspection, signals, and browser session replay
- MLflow - Open-source AI platform with tracing, evaluation, and LLM governance
- Honeycomb - High-performance distributed tracing
- Any OTLP-compatible backend - Including Jaeger, Datadog, New Relic, and more
- Agent execution steps
- Tool calls and executions
- LLM API calls (via LiteLLM integration)
- Browser automation sessions (when using browser-use)
- Conversation lifecycle events
Quick Start
Tracing is automatically enabled when you set the appropriate environment variables. The SDK detects the configuration on startup and initializes tracing without requiring code changes.Using Laminar
Laminar provides specialized AI observability features for OpenHands, including full conversation traces, browser session replay, and higher-level analysis features like signals.For Laminar-specific walkthroughs, see the official docs for OpenHands SDK tracing, session replay for browser agents, viewing traces, and signals.
Why use Laminar with OpenHands?
Laminar is especially useful when you want to understand how an agent behaved across one run or across many runs:- Inspect a single run in transcript, tree, or timeline views to see prompts, tool calls, outputs, and nested agent activity. See Laminar’s guide to viewing traces.
- Watch browser automation alongside trace spans with session replay for browser agents.
- Define signals to classify failures, user friction, or success patterns across many traces.
- Keep each OpenHands conversation grouped under a single session ID so multi-turn debugging is easier.
Using OpenTelemetry (OTLP) Backends
For OpenTelemetry (OTLP) compatible backends, set the following environment variables:- MLflow - Open-source AI platform with tracing, evaluation, and governance
- Honeycomb - High-performance distributed tracing
- Jaeger - Open-source distributed tracing
- Generic OTLP Collector - For other backends
Alternative Configuration Methods
You can also use these alternative environment variable formats:How It Works
The OpenHands SDK uses Laminar as its OpenTelemetry instrumentation layer for built-in tracing support. When you set the environment variables, the SDK:- Detects configuration: Checks for OTEL environment variables on startup
- Initializes tracing: Configures OpenTelemetry with the appropriate exporter
- Instruments code: Automatically wraps key functions with tracing decorators
- Captures context: Associates traces with conversation IDs for session grouping
- Exports spans: Sends trace data to your configured backend
What Gets Traced
The SDK automatically instruments these components:agent.step- Each iteration of the agent’s execution loop- Tool executions - Individual tool calls with input/output capture
- LLM calls - API requests to language models via LiteLLM
- Conversation lifecycle - Message sending, conversation runs, and title generation
- Browser sessions - When using browser-use, captures session replays (Laminar only)
Trace Hierarchy
Traces are organized hierarchically:conversation
conversation.run
agent.step
llm.completion
tool.execute
agent.step
llm.completion
tool.execute, the tool calls are traced individually, such as bash, file_editor, or task_tracker.
Configuration Reference
Environment Variables
The SDK checks for these environment variables (in order of precedence):| Variable | Description | Example |
|---|---|---|
LMNR_PROJECT_API_KEY | Laminar project API key | your-laminar-api-key |
LMNR_HTTP_PORT | HTTP port for self-hosted Laminar | 8000 |
LMNR_GRPC_PORT | gRPC port for self-hosted Laminar | 8001 |
OTEL_EXPORTER_OTLP_TRACES_ENDPOINT | Full OTLP traces endpoint URL | https://api.honeycomb.io:443/v1/traces |
OTEL_EXPORTER_OTLP_ENDPOINT | Base OTLP endpoint (traces path appended) | http://localhost:4317 |
OTEL_ENDPOINT | Short form endpoint | http://localhost:4317 |
OTEL_EXPORTER_OTLP_TRACES_HEADERS | Authentication headers for traces | x-honeycomb-team=YOUR_API_KEY |
OTEL_EXPORTER_OTLP_HEADERS | General authentication headers | Authorization=Bearer%20TOKEN |
OTEL_EXPORTER_OTLP_TRACES_PROTOCOL | Protocol for traces endpoint | http/protobuf, grpc |
OTEL_EXPORTER | Short form protocol | otlp_http, otlp_grpc |
Header Format
Headers should be comma-separatedkey=value pairs with URL encoding for special characters:
Protocol Options
The SDK supports both HTTP and gRPC protocols:http/protobuforotlp_http- HTTP with protobuf encoding (recommended for most backends)grpcorotlp_grpc- gRPC with protobuf encoding (use only if your backend supports gRPC)
Platform-Specific Configuration
Laminar Setup
- Sign up at laminar.sh
- Create a project and copy your API key
- Set the environment variable:
OpenHands Enterprise Setup
If you are running OpenHands Enterprise (OHE), you can use the same Laminar integration without changing application code:- Complete the OpenHands Enterprise quick start.
- Enable analytics in the Admin Console.
- Deploy OHE and wait for the analytics service to become ready.
- Open the Laminar UI at
https://analytics.app.<your-base-domain>. - Create a Laminar project and an ingest-only API key.
- Save that key as the Laminar Project API Key in the Admin Console.
- Redeploy, then start a conversation in OpenHands.
LMNR_ and LLM_ prefixes are automatically forwarded to the SDK runtime. That makes it possible to configure Laminar endpoint settings such as LMNR_BASE_URL, LMNR_PROJECT_API_KEY, and LMNR_FORCE_HTTP, as well as the LLM that powers Laminar’s own AI features (chat-with-trace, SQL-with-AI, and signals) via LLM_PROVIDER, LLM_BASE_URL, and LLM_MODEL_SMALL|MEDIUM|LARGE.
LLM_PROVIDER accepts gemini (Laminar’s default), openai, or bedrock. Set it to openai whenever you point LLM_BASE_URL at an OpenAI-compatible gateway (for example LiteLLM, OpenRouter, or vLLM), not just the public OpenAI API. For the full list of supported values, see Laminar’s official self-hosting configuration reference.
For the full OHE flow with screenshots and configuration examples, see Analytics in OpenHands Enterprise.
MLflow Setup
MLflow is an open-source AI platform that accepts OpenTelemetry traces out of the box, alongside evaluation and LLM governance capabilities.- Start your MLflow tracking server:
For other deployment options (pip, Docker Compose, etc.), see Set Up MLflow Server.
- Configure the environment variables:
http://localhost:5000), select the experiment, and open the Traces tab to view the recorded traces.
Honeycomb Setup
- Sign up at honeycomb.io
- Get your API key from the account settings
- Configure the environment:
Jaeger Setup
For local development with Jaeger:http://localhost:16686.
Generic OTLP Collector
For other backends, use their OTLP endpoint:Advanced Usage
Disabling Observability
To disable tracing, simply unset all OTEL environment variables:Custom Span Attributes
The SDK automatically adds these attributes to spans:conversation_id- UUID of the conversationtool_name- Name of the tool being executedaction.kind- Type of action being performedsession_id- Groups all traces from one conversation
Debugging Tracing Issues
If traces are not appearing in your observability platform:-
Verify environment variables:
-
Check SDK logs: The SDK logs observability initialization at debug level:
-
Test connectivity: Ensure your application can reach the OTLP endpoint:
- Validate headers: Check that authentication headers are properly URL-encoded.
Troubleshooting
Traces Not Appearing
Problem: No traces showing up in your observability platform. Solutions:- Verify environment variables are set correctly
- Check network connectivity to the OTLP endpoint
- Ensure authentication headers are valid
- Look for SDK initialization logs at debug level
High Trace Volume
Problem: Too many spans being generated. Solutions:- Configure sampling at the collector level
- For Laminar with non-browser tools, browser instrumentation is automatically disabled
- Use backend-specific filtering rules
Performance Impact
Problem: Concerned about tracing overhead. Solutions:- Tracing has minimal overhead when properly configured
- Disable tracing in development by unsetting environment variables
- Use asynchronous exporters (default in most OTLP configurations)
Example: Full Setup
This example is available on GitHub: examples/01_standalone_sdk/27_observability_laminar.py
examples/01_standalone_sdk/27_observability_laminar.py
Running the Example
Next Steps
- Analytics in OpenHands Enterprise - Deploy Laminar inside OHE and send conversation traces automatically
- Metrics Tracking - Monitor token usage and costs alongside traces
- LLM Registry - Track multiple LLMs used in your application
- Security - Add security validation to your traced agent executions

