instrument
Add Opik tracing to an existing codebase. Detects language (Python/TypeScript), identifies LLM frameworks, adds appropriate decorators and integrations, marks entrypoints, and wires up environment config. Use for "instrument my code", "add opik tracing", "add observability", or "trace my agent".
Skill body
Instrument — Add Opik Tracing to a Codebase
You are instrumenting an existing codebase with Opik observability. Follow these steps precisely.
Step 1 — Scope
If $ARGUMENTS is provided, scope your work to those files or directories. Otherwise, discover the project root and instrument the main application code.
Step 2 — Detect Language & Frameworks
Scan the codebase to determine:
- Language: Python (look for
*.py,pyproject.toml,requirements.txt) or TypeScript (look for*.ts,*.tsx,package.json) - LLM frameworks in use — search imports for these patterns:
| Import pattern | Framework | Integration |
|---|---|---|
from openai / import OpenAI |
OpenAI | track_openai |
import anthropic |
Anthropic | track_anthropic |
from langchain / @langchain |
LangChain | OpikTracer callback |
from langgraph |
LangGraph | OpikTracer with graph= |
from crewai |
CrewAI | track_crewai |
import dspy |
DSPy | OpikCallback |
from google … genai |
Google Gemini | track_genai |
import boto3 … bedrock |
AWS Bedrock | track_bedrock |
from llama_index |
LlamaIndex | LlamaIndexCallbackHandler |
import litellm |
LiteLLM | OpikLogger callback |
from pydantic_ai |
Pydantic AI | Logfire OTLP bridge |
from opik.integrations.adk / from google.adk |
Google ADK | track_adk_agent_recursive |
import ollama |
Ollama | track_openai with localhost base_url or manual @opik.track |
from agents import / from openai.agents |
OpenAI Agents SDK | OpikTracingProcessor |
from haystack |
Haystack | OpikConnector |
opik-openai / trackOpenAI (TS) |
OpenAI (TS) | trackOpenAI |
opik-vercel / OpikExporter (TS) |
Vercel AI SDK | OpikExporter |
opik-langchain / OpikCallbackHandler (TS) |
LangChain.js | OpikCallbackHandler |
opik-gemini / trackGemini (TS) |
Gemini (TS) | trackGemini |
- Existing Opik usage — check if
opikor@opik.trackis already imported. If so, audit rather than re-instrument.
Step 3 — Identify the Call Graph
Find:
- Entrypoint: the top-level function that kicks off the agent (e.g.,
main,run,agent,handle_message, a route handler, or whatever the user’s main orchestration function is) - LLM call sites: functions that call an LLM provider directly
- Tool functions: retrieval, search, API calls, or other tool-like operations
- Prompts and prompt-related config: hardcoded prompt strings, system messages, message templates, and any associated model/temperature values — note these as candidates for the Prompt Library (
client.get_prompt/client.get_chat_promptwithmetadatafor model config)
Entrypoint Parameter Rules
The function marked with entrypoint=True must only accept primitive-typed parameters: str, int, float, bool, and list/dict of primitives. This is because:
- Opik reads the function’s type hints to build an input form in the UI
- Users will type these values manually in a text field via the Local Runner
- Complex types (Pydantic models, dataclasses, request objects, custom classes) cannot be entered in a UI input field
If the candidate entrypoint accepts complex types (e.g., a request model, a config object, a dataclass):
- Look higher in the call chain for a function that already accepts primitives
- If none exists, create a thin wrapper function that accepts only primitives, unpacks them, and calls the original function. Move the
entrypoint=Truedecorator to this wrapper.
Example — bad entrypoint (complex parameter):
# ❌ DO NOT mark this as entrypoint — RecommendRequest is a Pydantic model
@app.post("/recommend")
async def recommend(request: RecommendRequest):
summary, tool_results = await run_agent(user_message=build_user_message(request))
return RecommendResponse(city=request.city, recommendations=_extract_recommendations(tool_results), summary=summary)
Example — good entrypoint (primitives only):
@opik.track(name="recommend-agent", entrypoint=True)
async def _run_entrypoint(user_message: str) -> tuple[str, list[dict]]:
"""Opik entrypoint — receives only the user message for Local Runner schema."""
return await run_agent(user_message=user_message)
@app.post("/recommend")
async def recommend(request: RecommendRequest):
summary, tool_results = await _run_entrypoint(user_message=build_user_message(request))
return RecommendResponse(city=request.city, recommendations=_extract_recommendations(tool_results), summary=summary)
The wrapper extracts the primitive values from the complex object and delegates to the existing logic. The HTTP handler calls the wrapper instead of the inner function directly, so the trace captures the full execution.
Step 4 — Add Framework Integrations
For each detected framework, add the appropriate integration at the module level. See the integration table above and references/integrations.md for the exact patterns.
Python examples:
# OpenAI
from opik.integrations.openai import track_openai
client = track_openai(OpenAI()) # wrap existing client
# Anthropic
from opik.integrations.anthropic import track_anthropic
client = track_anthropic(anthropic.Anthropic())
# LangChain / LangGraph
from opik.integrations.langchain import OpikTracer
tracer = OpikTracer()
# pass config={"callbacks": [tracer]} to invoke()
# LiteLLM inside @opik.track — CRITICAL: pass span context
from opik.opik_context import get_current_span_data
# in every litellm.completion() call, add:
# metadata={"opik": {"current_span_data": get_current_span_data()}}
TypeScript examples:
// OpenAI
import { trackOpenAI } from "opik-openai";
const trackedClient = trackOpenAI(openai);
// Vercel AI SDK
import { OpikExporter } from "opik-vercel";
// set up NodeSDK with OpikExporter
Step 5 — Add @opik.track Decorators (Python) or Client Tracing (TypeScript)
This step adds the tracing scaffolding that the prompt migration in Step 6 relies on. Add decorators first so that the get_prompt / get_chat_prompt calls introduced next will land inside @opik.track-decorated functions.
Python
Add import opik at the top of each file you instrument.
| Function role | Decorator |
|---|---|
| Entrypoint (top-level agent) | @opik.track(entrypoint=True, name="<agent-name>") |
| LLM call | @opik.track(type="llm") |
| Tool / retrieval | @opik.track(type="tool") |
| Guardrail / validation | @opik.track(type="guardrail") |
| Other helper in the call chain | @opik.track |
- Entrypoint parameters must be primitives only (
str,int,float,bool,list,dict). If the natural entrypoint takes a complex type, create a wrapper — see Step 3 “Entrypoint Parameter Rules”. - Place the decorator above any existing decorators (e.g., above
@app.route) - For async functions,
@opik.trackworks the same way — no changes needed - If the function is a script entrypoint (not a long-running server), add
opik.flush_tracker()after the top-level call client.get_prompt()/client.get_chat_prompt()must be called inside a@opik.track-decorated function — this links the fetched prompt version to the trace so it appears in the Traces view. Fetching at module level works but the prompt won’t be visible in traces.
TypeScript
Use the client-based approach:
import { Opik } from "opik";
const client = new Opik({ projectName: "<project-name>" });
// In the entrypoint function:
const trace = client.trace({ name: "<agent-name>", input: { ... } });
const span = trace.span({ name: "<operation>", type: "tool", input: { ... } });
// ... logic
span.end({ output: { ... } });
trace.end({ output: { ... } });
await client.flush();
For entrypoints that should be discoverable by opik connect — note that params must only use primitive types (string, number, boolean) since users enter these values in a UI text field:
import { track } from "opik";
const myAgent = track(
{ name: "<agent-name>", entrypoint: true, params: [{ name: "query", type: "string" }] },
async (query: string) => { /* ... */ }
);
Step 6 — Migrate Prompts to the Prompt Library
For every prompt found in Step 3, replace the hardcoded value with a get_prompt / get_chat_prompt call inside the enclosing @opik.track-decorated function added in Step 5.
Classify each prompt:
- Single string (system prompt, instruction, template) →
create_prompt/get_prompt - List of
{"role", "content"}messages →create_chat_prompt/get_chat_prompt
Include model name, temperature, and any other call-level parameters in metadata so they version together with the prompt template and can be updated from the Opik UI without a code change.
get_prompt / get_chat_prompt returns None if the prompt doesn’t exist yet — check for None and create on first run so the same code handles both initial setup and subsequent runs.
Python:
opik_client = opik.Opik()
@opik.track(entrypoint=True, project_name="<project-name>")
def run_agent(question: str) -> str:
prompt = opik_client.get_prompt(name="<prompt-name>")
if prompt is None:
prompt = opik_client.create_prompt(
name="<prompt-name>",
prompt="<original hardcoded prompt text>",
metadata={"model": "<model>", "temperature": <value>},
)
system_message = prompt.format() # pass template vars if any: prompt.format(var=value)
return llm_call(
model=prompt.metadata["model"],
temperature=prompt.metadata["temperature"],
system_prompt=system_message,
question=question,
)
For multi-turn message lists:
chat_prompt = opik_client.get_chat_prompt(name="<prompt-name>")
if chat_prompt is None:
chat_prompt = opik_client.create_chat_prompt(
name="<prompt-name>",
messages=[...], # original hardcoded messages list
metadata={"model": "<model>", "temperature": <value>},
)
messages = chat_prompt.format() # pass template vars if any
return llm_call(
model=chat_prompt.metadata["model"],
temperature=chat_prompt.metadata["temperature"],
messages=messages,
)
TypeScript:
const opikClient = new Opik({ projectName: "<project-name>" });
const runAgent = track({ entrypoint: true, projectName: "<project-name>" }, async (question: string) => {
let prompt = await opikClient.getPrompt({ name: "<prompt-name>" });
if (prompt === null) {
prompt = await opikClient.createPrompt({
name: "<prompt-name>",
prompt: "<original hardcoded prompt text>",
metadata: { model: "<model>", temperature: <value> },
});
}
const systemMessage = prompt.format(); // pass template vars if any
const { model, temperature } = prompt.metadata as { model: string; temperature: number };
return llmCall({ model, temperature, systemMessage, question });
});
Step 7 — Conversational Agents: Add thread_id
If the agent handles multi-turn conversations (chat bots, support agents, multi-step assistants), wire thread_id:
@opik.track(entrypoint=True)
def handle_message(session_id: str, message: str) -> str:
opik.update_current_trace(thread_id=session_id)
return generate_response(session_id, message)
Skip this for single-shot agents or batch processing.
Step 8 — Environment Config
Follow the setup decision tree from the main opik skill:
- If the project has
.env/.env.local→ appendOPIK_API_KEY,OPIK_WORKSPACE,OPIK_URL_OVERRIDE(if missing) - If no
.envexists → Python: create/update~/.opik.config; TypeScript: create.envor.env.local - Never introduce a second config mechanism
- Never overwrite existing values
- Update
.env.example/.env.sampleif one exists - Set
project_namein code, not in env files
OPIK_URL_OVERRIDE path rules
The URL suffix depends on where Opik is hosted:
| Deployment | URL format | Example |
|---|---|---|
| Opik Cloud / managed | <base>/opik/api |
https://www.comet.com/opik/api |
| Self-hosted (local) | <base>/api |
http://localhost:5173/api |
- Cloud/managed: always append
/opik/api - Self-hosted (typically
localhostor an internal hostname): append only/api— no/opikprefix - When writing or suggesting an
OPIK_URL_OVERRIDEvalue, apply this rule so users don’t have to remember it
Step 9 — Install Dependencies
Print the install command but do NOT run it automatically. Let the user decide.
Python:
pip install opik
Plus any integration packages if needed (most are included in opik).
TypeScript:
npm install opik
Plus framework-specific packages: opik-openai, opik-vercel, opik-langchain, opik-gemini as needed.
Step 10 — Verify
After instrumentation, do a quick audit:
- Every LLM call site is traced (via integration wrapper or
@opik.track) - Exactly one function has
entrypoint=True - The entrypoint function accepts only primitive parameters (
str,int,float,bool,list,dict) — no Pydantic models, dataclasses, or custom classes - Script entrypoints call
opik.flush_tracker()(Python) orawait client.flush()(TypeScript) - LiteLLM calls inside
@opik.trackpasscurrent_span_datavia metadata - No hardcoded API keys were introduced
- Existing tests still import correctly (no circular imports introduced)
- No deprecated
opik.Prompt/opik.ChatPrompt/opik.Configusage introduced — use the Prompt library instead - All
client.get_prompt()/client.get_chat_prompt()calls are inside@opik.track-decorated functions — prompt version will not appear in traces otherwise
Anti-Patterns to Avoid
- Double-wrapping: Don’t add
@opik.track(type="llm")to a function that already uses a framework integration (e.g.,track_openai). The integration handles tracing. - Orphaned LiteLLM traces: Always pass
current_span_datawhenOpikLoggeris used inside@opik.trackcode. - Complex entrypoint parameters: The entrypoint function must only accept primitives (
str,int,float,bool,list,dict). Pydantic models, dataclasses, or custom classes can’t be typed into a UI input field. If the natural entrypoint takes a complex type, create a thin wrapper that accepts primitives. - Using deprecated
opik.Prompt/opik.ChatPrompt/opik.Config: These have been retired. Useclient.get_prompt()/client.get_chat_prompt()from the Prompt library instead. - Fetching prompts outside
@opik.track:client.get_prompt()/client.get_chat_prompt()must be called inside a@opik.track-decorated function. Fetching at module level works functionally but the prompt version won’t be linked to the trace and won’t appear in the Traces view. - Missing entrypoint: Without
entrypoint=True, Local Runner (opik connect) won’t discover the agent. - Missing flush: Scripts that exit without flushing lose trace data.
- Overwriting config: Check before writing to
.envor~/.opik.config.
References
For detailed API signatures and advanced patterns, see:
../opik/references/tracing-python.md— Python SDK reference../opik/references/tracing-typescript.md— TypeScript SDK reference../opik/references/integrations.md— All framework integrations../opik/references/observability.md— Core concepts (traces, spans, threads)