Skip to main content

Documentation Index

Fetch the complete documentation index at: https://laminar.sh/docs/llms.txt

Use this file to discover all available pages before exploring further.

Overview

Laminar is an open-source, OpenTelemetry-native observability platform for the OpenAI Agents SDK (Python, TypeScript). Trace, debug, and monitor every agent turn, tool call, handoff, and sub-agent with a single Laminar.initialize() call. Self-host via Helm or use managed cloud. The OpenAI Agents SDK lets you build agent workflows with Agent, Runner.run / run, function tools, and handoffs between specialists. Laminar hooks into the SDK’s built-in trace processor to mirror every agent span into Laminar, plus adds the system instructions so the full prompt is visible in the trace. What Laminar captures:
  • The root agent workflow, each agents.task, and every agents.turn with the model, prompt, response, token counts, latency, and cost.
  • Every function_tool invocation, with arguments and return value.
  • Handoffs between agents, with the destination agent’s turns nested under the handoff span.
  • Agents invoked as tools (Agent.as_tool / Agent.asTool), with the sub-agent’s full run nested under the tool call.
  • Sandbox sessions (Python only) started via SandboxRunConfig, including manifest materialization and shell/file tool calls.
  • Agent instructions prepended to the input messages on every LLM span.

Getting started

1

Install

Ensure you have lmnr version 0.7.49 or higher:
pip install -U lmnr openai-agents
2

Set environment variables

export LMNR_PROJECT_API_KEY=your-laminar-project-api-key
export OPENAI_API_KEY=your-openai-api-key
3

Initialize Laminar

Laminar.initialize() auto-instruments the OpenAI Agents SDK when openai-agents is importable. No wrapping call is needed.
import asyncio

from agents import Agent, Runner
from lmnr import Laminar, observe

Laminar.initialize()

@observe(name="math-homework")
async def main():
    agent = Agent(
        name="MathHelper",
        instructions="You are a patient math tutor. Explain each step clearly.",
        model="gpt-5-mini",
    )
    result = await Runner.run(agent, "A train leaves Boston at 9am at 60 mph...")
    print(result.final_output)

if __name__ == "__main__":
    asyncio.run(main())
Wrapping your entry point in @observe() is optional but recommended: it creates a root span that captures inputs and outputs and makes the trace easy to find in the UI.

See what happened in a trace

Open the trace in Laminar and you land on the transcript view: each turn reads as a conversation, with the prompt, the model’s response, and any tool calls inline with their inputs and outputs. A tree of span names tells you the shape of the run; the transcript tells you what actually happened.
OpenAI Agents SDK trace in Laminar, transcript view
More on the trace UX: Viewing traces.

Multi-agent runs with handoffs

The OpenAI Agents SDK models specialization through handoffs: a triage agent routes the user to a specialist via handoff(other_agent) / handoff(otherAgent), and the SDK exposes each handoff as a transfer_to_<agent> tool call. In Laminar, the destination agent’s turns and tool calls nest under the handoff span, so you can follow the full multi-agent conversation in one trace.
import asyncio

from agents import Agent, Runner, function_tool, handoff
from lmnr import Laminar, observe

Laminar.initialize()


@function_tool
def cancel_booking(confirmation_code: str) -> str:
    return f"Booking {confirmation_code} cancelled. Refund in 5-7 business days."


@function_tool
def lookup_loyalty_balance(member_id: str) -> str:
    return f"Member {member_id}: 48,200 points, Gold tier."


booking_agent = Agent(
    name="BookingAgent",
    handoff_description="Handles cancellations and loyalty balance lookups.",
    instructions=(
        "You handle cancellations and loyalty questions. Use cancel_booking for "
        "cancellations and lookup_loyalty_balance for points and tier."
    ),
    tools=[cancel_booking, lookup_loyalty_balance],
    model="gpt-5-mini",
)

triage_agent = Agent(
    name="TriageAgent",
    instructions=(
        "Route the user to the correct specialist. For cancellations or loyalty, "
        "hand off to BookingAgent. Do not answer specialist topics yourself."
    ),
    handoffs=[handoff(booking_agent)],
    model="gpt-5-mini",
)


@observe(name="airline-support")
async def main():
    result = await Runner.run(
        triage_agent,
        "Cancel my booking Z9X7K2 and look up loyalty for member M-88421.",
    )
    print(result.final_output)


if __name__ == "__main__":
    asyncio.run(main())
The resulting trace shows the triage turn, the handoff, and the booking agent’s tool calls in a single conversation:
OpenAI Agents SDK multi-agent trace in Laminar, transcript view
Switch to tree view when you want the full hierarchy at a glance:
OpenAI Agents SDK multi-agent trace in Laminar, tree view

Sub-agents as tools

Handoffs transfer the conversation to another agent. Sub-agents as tools keep the orchestrator in control: Agent.as_tool (Python) and Agent.asTool (TypeScript) wrap a specialist as a FunctionTool that the parent agent can call like any other tool, then synthesize the sub-agent’s output back into the parent’s turn. Laminar nests the sub-agent’s full run (its own turns and tool calls) directly under the parent’s tool-call span, so you can read the orchestrator-plus-specialist flow in a single trace.
import asyncio

from agents import Agent, Runner, function_tool
from lmnr import Laminar, observe

Laminar.initialize()


@function_tool
def search_flights(origin: str, destination: str) -> str:
    return f"Flights from {origin} to {destination}: UA123 08:00, DL456 14:30."


flight_agent = Agent(
    name="FlightAgent",
    instructions="Find flights matching the user's origin and destination.",
    tools=[search_flights],
    model="gpt-5-mini",
)

summary_agent = Agent(
    name="SummaryAgent",
    instructions="Summarize the user's itinerary in a single short paragraph.",
    model="gpt-5-mini",
)

concierge = Agent(
    name="Concierge",
    instructions=(
        "You are a travel concierge. Call book_flight to look up flights, "
        "then call write_summary to draft a one-paragraph itinerary."
    ),
    tools=[
        flight_agent.as_tool(
            tool_name="book_flight",
            tool_description="Search for flights between two cities.",
        ),
        summary_agent.as_tool(
            tool_name="write_summary",
            tool_description="Write a one-paragraph summary of an itinerary.",
        ),
    ],
    model="gpt-5-mini",
)


@observe(name="travel-concierge")
async def main():
    result = await Runner.run(
        concierge,
        "I'm flying SFO to JFK next Tuesday. Book something in the morning and summarize it.",
    )
    print(result.final_output)


if __name__ == "__main__":
    asyncio.run(main())
In tree view, the Concierge turn calls book_flight and write_summary; each call expands into the sub-agent’s own turns, tool calls, and LLM spans, rather than being flattened to a single tool-return value.
Choose between handoffs and sub-agents-as-tools based on who owns the reply: handoffs hand the conversation to the specialist and its answer is returned to the user; sub-agents-as-tools return a structured result back to the orchestrator, which then keeps writing the user-facing response.

Running agents in sandboxes (Python)

The Python SDK can execute an agent against a sandboxed workspace (locally on your machine or on a remote provider like E2B, Modal, Daytona, Runloop, Cloudflare, Vercel, or Blaxel) via SandboxAgent and SandboxRunConfig. Laminar instruments sandbox sessions and surfaces each manifest materialization step, shell exec, and file read as a span, so you can see exactly what the agent did inside the sandbox alongside the LLM turns that drove it.
Sandboxes are Python-only. The TypeScript SDK does not expose SandboxAgent / SandboxRunConfig at the time of writing.

Local sandbox (UnixLocal)

UnixLocalSandboxClient runs the agent against your host filesystem with no extra infrastructure: a good starting point for development and tests.
import asyncio

from agents import Runner
from agents.run_config import RunConfig
from agents.sandbox import Manifest, SandboxAgent, SandboxRunConfig
from agents.sandbox.entries import GitRepo
from agents.sandbox.sandboxes import UnixLocalSandboxClient
from lmnr import Laminar, observe

Laminar.initialize()


@observe(name="sandbox-repo-summary")
async def main():
    agent = SandboxAgent(
        name="WorkspaceAssistant",
        instructions="Inspect the sandbox workspace before answering.",
        default_manifest=Manifest(
            entries={
                "repo": GitRepo(repo="openai/openai-agents-python", ref="main"),
            }
        ),
        model="gpt-5-mini",
    )

    result = await Runner.run(
        agent,
        "Inspect the repo README and summarize what this project does.",
        run_config=RunConfig(
            sandbox=SandboxRunConfig(client=UnixLocalSandboxClient()),
        ),
    )
    print(result.final_output)


if __name__ == "__main__":
    asyncio.run(main())

Remote sandbox (E2B)

Swap the client for any remote provider to run the same agent on managed infrastructure. The example below uses E2B; the pattern is identical for ModalSandboxClient, DaytonaSandboxClient, RunloopSandboxClient, CloudflareSandboxClient, VercelSandboxClient, and BlaxelSandboxClient under agents.extensions.sandbox.<provider>.
pip install -U 'openai-agents[e2b]'
export E2B_API_KEY=your-e2b-api-key
import asyncio

from agents import Runner
from agents.extensions.sandbox.e2b import E2BSandboxClient
from agents.run_config import RunConfig
from agents.sandbox import Manifest, SandboxAgent, SandboxRunConfig
from agents.sandbox.entries import GitRepo
from lmnr import Laminar, observe

Laminar.initialize()


@observe(name="sandbox-repo-summary-e2b")
async def main():
    agent = SandboxAgent(
        name="WorkspaceAssistant",
        instructions="Inspect the sandbox workspace before answering.",
        default_manifest=Manifest(
            entries={
                "repo": GitRepo(repo="openai/openai-agents-python", ref="main"),
            }
        ),
        model="gpt-5-mini",
    )

    result = await Runner.run(
        agent,
        "Inspect the repo README and summarize what this project does.",
        run_config=RunConfig(
            sandbox=SandboxRunConfig(client=E2BSandboxClient()),
        ),
    )
    print(result.final_output)


if __name__ == "__main__":
    asyncio.run(main())
In either case the trace includes a sandbox.start / sandbox.cleanup span pair around the agent’s work, the manifest materialization spans that pulled repo into the workspace, and every shell or file operation the agent ran via its sandbox tools.

Track outcomes with Signals

Traces answer what happened on this run. Signals answer the cross-trace question: how often does the triage agent answer a cancellation itself instead of handing off, when do tool calls return errors, how many runs exceed three turns without a final answer. A Signal pairs a plain-language prompt with a JSON output schema. Laminar runs it live on new traces (Triggers) or backfills it across history (Jobs) and records a structured event every time it matches. From there you query, cluster, and alert on events across every trace.
Every new project ships with a Failure Detector Signal that categorizes issues on any trace over 1000 tokens. Open it from the Signals sidebar to see events as soon as your agent traces arrive.

Query across traces

  • SQL editor for ad-hoc queries across traces, spans, signals, and evals.
  • SQL API for programmatic access from scripts and pipelines.
  • CLI (lmnr-cli sql query) for terminal-driven queries and piping JSON into shell tools or coding agents.
  • MCP server to query Laminar directly from Claude Code, Cursor, Codex, or any MCP-aware client.

Troubleshooting

  • Confirm LMNR_PROJECT_API_KEY is set in the same process that runs the SDK.
  • Python: openai-agents must be importable when Laminar.initialize() runs. Install with pip install openai-agents. The integration requires openai-agents >= 0.7.0 and lmnr >= 0.7.49.
  • TypeScript: @openai/agents must be installed in the same workspace. The integration requires @lmnr-ai/lmnr >= 0.8.21. If you import @openai/agents before Laminar.initialize() in ESM, pass the module via instrumentModules: { openAIAgents: agents }.
The SDK emits a transfer_to_<agent> tool call followed by an agents.handoff span, and the destination agent’s work lands as a sibling under the same parent task. Open the trace in tree view to see the full structure.
Make sure the sub-agent is registered in the orchestrator’s tools array via .as_tool(...) / .asTool(...), not re-exported as a plain function. Each as_tool call produces a FunctionTool whose execute spins up a nested Runner run; Laminar nests the inner run under the parent tool-call span only when the wrapper is the one invoked.
Set base_url and the ports of your instance when initializing. For a local OSS deployment:
Laminar.initialize(
    base_url="http://localhost",
    http_port=8000,
    grpc_port=8001,
)
Laminar.initialize({
  baseUrl: 'http://localhost',
  httpPort: 8000,
  port: 8001,
});

What’s next

Viewing traces

Read the transcript view, filter, and search across traces.

Signals

Detect behaviors and failures across every run, then query, cluster, and alert on them.

SQL editor and MCP server

Query traces programmatically from the UI, API, or your IDE.

Tracing structure

Sessions, metadata, and tags for deeper control.