OpenTelemetry integration for AI coding agents (Cursor IDE/CLI, GitHub Copilot, Claude Code, Antigravity, OpenCode-compatible runners)
Project description
OpenTelemetry Hook for AI Coding Agents
Observability for your AI pair-programmer — know what your agent is doing, one trace at a time.
An open-source OpenTelemetry integration that captures all AI coding agent activity as structured traces and logs and exports them to any OTLP-compliant backend. Works with Cursor IDE / Cursor CLI, GitHub Copilot, Claude Code, Antigravity, and compatible hook runners such as OpenCode using OpenTelemetry GenAI semantic conventions.
Every hook event — prompt submissions, tool calls, shell commands, MCP interactions, file edits, subagent orchestration — becomes an OpenTelemetry span you can query, alert on, and visualize in Jaeger, Grafana, Datadog, Honeycomb, Coralogix, or any OTLP-compatible backend.
Note: Claude Code has native OpenTelemetry support, but this repo can also be used as a hook target when you want the same hook-based pipeline across IDEs.
How It Works
The hook is a lightweight Python command that your IDE invokes on every agent event. The IDE pipes a JSON payload to stdin, the hook processes it, emits OpenTelemetry spans and logs, and returns {"continue": true} on stdout so the IDE proceeds normally. No sidecar, no daemon — just a command your IDE calls.
IDE Event → stdin (JSON) → otel-hook → OpenTelemetry SDK → OTLP Backend
↓
stdout: {"continue": true}
Features
-
Multi-IDE Support: One script, multiple hook providers —
setup.shnow creates native hook configs for Cursor, GitHub Copilot, and Claude Code without extra IDE override env vars, while the bundled examples cover Copilot/Claude/Cursor for manual installs. The runtime prefers parent-process discovery first, then explicit overrides, then self-reported payload fields, and finally heuristics when needed. -
Session-level Traces: Groups all events within a session into a single trace with a 3-tier hierarchy:
gen_ai.client.session (root)
├── gen_ai.client.generation (gen-1)
│ ├── gen_ai.client.hook.UserPromptSubmit
│ ├── gen_ai.client.hook.PreToolUse
│ ├── gen_ai.client.hook.PostToolUse
│ └── gen_ai.client.hook.Stop
├── gen_ai.client.generation (gen-2)
│ ├── gen_ai.client.hook.UserPromptSubmit
│ ├── gen_ai.client.hook.PreToolUse
│ ├── gen_ai.client.hook.PostToolUse
│ └── gen_ai.client.hook.Stop
└── gen_ai.client.hook.SessionEnd
-
GenAI Semantic Conventions: Emits OpenTelemetry GenAI attributes aligned with v1.37+ (
gen_ai.provider.name,gen_ai.operation.name,gen_ai.request.model,gen_ai.usage.*, etc.) while preserving legacygen_ai.systemfor backward compatibility. -
All Hook Events: Captures the full lifecycle — sessions, prompts, tool usage, shell commands, MCP calls, file operations, subagents, errors, and more.
-
Structured OTel Logs: Emits trace-correlated log records for MCP calls, shell executions, and tool usage — with full I/O payloads, server output, and duration. Logs are exported via OTLP alongside spans.
-
Zero Setup: Auto-provisions a Python virtual environment on first run. No manual install needed.
-
Privacy Controls: Built-in masking of emails, tokens, and usernames. Text capture is opt-in.
-
JSON Config File: All settings in
otel_config.json— no environment variable exports needed.
Supported Events
| Canonical Name | Cursor IDE / CLI | Copilot | Claude Code / Antigravity | OpenCode (plugin) |
|---|---|---|---|---|
SessionStart |
sessionStart |
sessionStart |
SessionStart |
session.created |
SessionEnd |
sessionEnd |
sessionEnd |
SessionEnd |
session.deleted, session.error |
UserPromptSubmit |
beforeSubmitPrompt |
userPromptSubmitted |
UserPromptSubmit |
message.updated (role=user) |
PreToolUse |
preToolUse |
preToolUse |
PreToolUse |
tool.execute.before ¹ |
PostToolUse |
postToolUse |
postToolUse |
PostToolUse |
tool.execute.after (exit=0) |
PostToolUseFailure |
postToolUseFailure |
— | PostToolUseFailure |
tool.execute.after (exit≠0) |
Stop |
stop |
— | Stop |
session.idle |
SubagentStart |
subagentStart |
— | SubagentStart |
— ² |
SubagentStop |
subagentStop |
— | SubagentStop |
— ² |
ErrorOccurred |
— | errorOccurred |
— | — |
BeforeShellExecution |
beforeShellExecution |
— | — | — ¹ |
AfterShellExecution |
afterShellExecution |
— | — | — ¹ |
BeforeMCPExecution |
beforeMCPExecution |
— | — | — ¹ |
AfterMCPExecution |
afterMCPExecution |
— | — | — ¹ |
BeforeReadFile |
beforeReadFile |
— | — | — ¹ |
AfterFileEdit |
afterFileEdit |
— | — | file.edited |
¹ OpenCode routes bash, read, write, MCP, and all other tools through the universal tool.execute.before/after hooks, so these events are observable as PreToolUse/PostToolUse with the appropriate tool_name.
² Subagent invocations surface as PreToolUse/PostToolUse with tool_name=task — there are no dedicated subagent hook events in OpenCode.
Installation
# Recommended: pipx keeps otel-hook on PATH in an isolated venv
pipx install opentelemetry-hooks
# Or with pip
pip install opentelemetry-hooks
To pin a specific version or install directly from a tag:
pipx install git+https://github.com/o11y-dev/opentelemetry-hooks.git@v0.11.0
Or install from a pre-built wheel from the Releases page:
pipx install opentelemetry_hooks-*.whl
Once installed, run otel-hook setup to wire your agents.
Quick Start
One-Command Setup (pip/pipx install)
After installing the package, configure your agents with the built-in CLI:
# Auto-detect all installed agents and configure globally
otel-hook setup
# Configure a specific agent
otel-hook setup --agent claude
otel-hook setup --agent cursor
otel-hook setup --agent copilot --no-global # project-scoped (run from repo root)
otel-hook setup --agent gemini
# Project-scoped instead of global
otel-hook setup --agent cursor --no-global
# Check registration status
otel-hook diagnose
# Remove hooks
otel-hook uninstall --agent claude
Setup is idempotent — safe to re-run. Then configure your OTLP endpoint:
vim ~/.local/share/opentelemetry-hooks/otel_config.json
Python API (importable)
The setup functions are importable for programmatic use:
from otel_hook import setup_agent, setup_claude, setup_cursor
setup_claude(global_=True) # ~/.claude/settings.json
setup_cursor(global_=True) # ~/.cursor/hooks.json
setup_agent("gemini", global_=True)
Source Checkout / Cursor Project Setup
If you're working from a source checkout rather than a pip install, use the
bundled setup.sh:
# Project-level — hooks.json in the current repo (.cursor/hooks.json)
bash .cursor/hooks/opentelemetry-hook/setup.sh
# Global — applies to every Cursor project (~/.cursor/hooks.json)
bash .cursor/hooks/opentelemetry-hook/setup.sh --cursor --global
Then edit your endpoint config and restart Cursor:
vim .cursor/hooks/opentelemetry-hook/otel_config.json
Clone Into an Existing Project
If your project doesn't have the hook yet, copy the entire hook directory and run setup:
# Clone the hook repo and copy the essential files into your project
git clone https://github.com/o11y-dev/opentelemetry-hooks.git /tmp/otel-hook-source
mkdir -p .cursor/hooks/opentelemetry-hook
cp /tmp/otel-hook-source/otel_hook.py .cursor/hooks/opentelemetry-hook/
cp /tmp/otel-hook-source/setup.sh .cursor/hooks/opentelemetry-hook/
cp /tmp/otel-hook-source/otel_config.example.json .cursor/hooks/opentelemetry-hook/
cp /tmp/otel-hook-source/.gitignore .cursor/hooks/opentelemetry-hook/
cp -r /tmp/otel-hook-source/examples .cursor/hooks/opentelemetry-hook/
# Run setup — creates/merges hooks.json automatically
bash .cursor/hooks/opentelemetry-hook/setup.sh
rm -rf /tmp/otel-hook-source
Prerequisites
- Python 3.12+ (the setup script checks for this)
- An OTLP-compatible backend (Jaeger, Coralogix, Datadog, Grafana, Honeycomb, etc.)
Other IDEs
Cursor CLI
Cursor CLI uses the same .cursor/hooks.json configuration and hook payload shape as Cursor IDE, so the Cursor IDE setup above in Quick Start also covers Cursor CLI. Its spans are recorded with the canonical gen_ai.client.name=cursor.
GitHub Copilot
# Repo-scoped hooks file (.github/hooks/otel-hooks.json)
bash .cursor/hooks/opentelemetry-hook/setup.sh --copilot
setup.sh --copilot creates or merges .github/hooks/otel-hooks.json and points each event directly at otel-hook (or the local otel_hook.py fallback). Copilot is then detected from the process tree first, with session_id-based heuristics as a fallback.
GitHub Copilot hooks are repository-scoped, so --copilot --global is intentionally unsupported. Commit .github/hooks/otel-hooks.json to your default branch for the coding agent to pick it up.
If you prefer a manual install, copy the bundled example instead:
mkdir -p .github/hooks
cp .cursor/hooks/opentelemetry-hook/examples/copilot-hooks.example.json .github/hooks/otel-hooks.json
Then replace {{SCRIPT_PATH}} with the hook command. For a copied-source checkout the default is python3 .cursor/hooks/opentelemetry-hook/otel_hook.py; use otel-hook only when the package is installed via pipx or pip.
See GitHub Copilot hooks docs.
Claude Code
mkdir -p .claude
cp .cursor/hooks/opentelemetry-hook/examples/claude-hooks.example.json .claude/settings.json
Replace {{SCRIPT_PATH}} with the hook command, for example:
# source checkout / copied-source
python3 .cursor/hooks/opentelemetry-hook/otel_hook.py
# pip-installed package
otel-hook
The bundled Claude example and setup.sh --claude both invoke otel-hook directly without an IDE override env var.
Claude Code is auto-detected from the parent process tree first; hook metadata such as session_id, transcript_path, permission_mode, and notification_type is used as a fallback. The camelCase alias handling is mainly for compatible third-party hook runners and mixed payload formats.
Antigravity
Antigravity workflow and hook formats can vary, so the simplest integration is to invoke the hook command directly from your workflow/rule and pin the IDE name explicitly:
# source checkout / copied-source
env IDE_OTEL_IDE_NAME=antigravity python3 .cursor/hooks/opentelemetry-hook/otel_hook.py
# pip-installed package
env IDE_OTEL_IDE_NAME=antigravity otel-hook
An example Antigravity workflow is included in examples/antigravity-workflow.example.md:
mkdir -p .agent/workflows
cp .cursor/hooks/opentelemetry-hook/examples/antigravity-workflow.example.md .agent/workflows/opentelemetry-hook.md
Replace {{SCRIPT_PATH}} in the copied workflow with the hook command you want Antigravity to invoke. For a copied-source checkout use python3 .cursor/hooks/opentelemetry-hook/otel_hook.py; use otel-hook for a pip-installed package.
OpenCode
A native TypeScript plugin is included at plugin/opencode.ts. It hooks into OpenCode's session and tool lifecycle events and pipes JSON payloads to otel-hook on stdin — the same pattern used by rtk.
Quick setup (recommended):
# Global — available in every OpenCode session
bash setup.sh --opencode --global
# Project-level — only active for this project
bash setup.sh --opencode
Manual install:
# Global
mkdir -p ~/.config/opencode/plugins
cp plugin/opencode.ts ~/.config/opencode/plugins/otel-hook.ts
# Project-level
mkdir -p .opencode/plugins
cp plugin/opencode.ts .opencode/plugins/otel-hook.ts
Restart OpenCode after installing. The bundled plugin — including the copy installed by setup.sh --opencode — invokes otel-hook directly. The runtime prefers parent-process discovery, while the plugin's source_app: "OpenCode" payload field remains a compatibility fallback. OPENCODE_CONFIG_DIR is respected if set.
Events captured: SessionStart, SessionEnd, UserPromptSubmit, PreToolUse, PostToolUse, PostToolUseFailure (detected via metadata.exit), Stop, AfterFileEdit. Bash, read, write, MCP, and subagent (task) tool calls all flow through the universal tool.execute.before/after hooks and appear as PreToolUse/PostToolUse with the appropriate tool_name.
Other compatible runners
For any hook runner not listed above, invoke otel-hook (or python3 .../otel_hook.py) and forward compatible hook JSON on stdin. Pass a self-reported client field such as ide_name, client, or source_app with the value matching your tool, or set IDE_OTEL_IDE_NAME in the environment. When your runner uses camelCase payload keys such as sessionId, toolName, toolInput, or hookEventType, the hook normalizes them automatically before exporting spans.
GitHub Copilot — Recommended Repositories
To make this hook automatically available to the GitHub Copilot coding agent across your organization's repositories, add it as a recommended repository:
- Go to your organization settings → Copilot → Coding agent → Recommended repositories
- Add
o11y-dev/opentelemetry-hooksto the list - The Copilot coding agent will now be able to reference this repo for hook setup and configuration
Configuration
Edit .cursor/hooks/opentelemetry-hook/otel_config.json:
{
"OTEL_EXPORTER_OTLP_ENDPOINT": "http://localhost:4317",
"OTEL_EXPORTER_OTLP_PROTOCOL": "grpc",
"OTEL_SERVICE_NAME": "ide-agent",
"IDE_OTEL_BATCH_ON_STOP": "true"
}
Then restart your IDE.
Configuration Reference
OTLP Exporter
| Variable | Description | Default |
|---|---|---|
OTEL_EXPORTER_OTLP_ENDPOINT |
OTLP collector endpoint | http://localhost:4317 |
OTEL_EXPORTER_OTLP_PROTOCOL |
grpc, http/protobuf, or http/json |
grpc |
OTEL_EXPORTER_OTLP_HEADERS |
Auth headers (URL-encoded key=value pairs) |
— |
OTEL_SERVICE_NAME |
Service name in traces | ide-agent |
Note:
OTEL_EXPORTER_OTLP_INSECUREis only used by the OTLP gRPC exporter (OTEL_EXPORTER_OTLP_PROTOCOL=grpc). It defaults totrue(plaintext); set tofalsefor TLS-secured gRPC endpoints. Forhttp/protobufandhttp/jsonexporters, TLS is determined by the endpoint scheme (https://vshttp://).
Hook Behavior
| Variable | Description | Default |
|---|---|---|
IDE_OTEL_BATCH_ON_STOP |
Enable session-level batching (recommended) | false |
IDE_OTEL_IDE_NAME |
Force the detected IDE name (cursor, copilot, claude, antigravity, opencode) for generic hook runners; common labels like GitHub Copilot, Claude Code, Cursor IDE / Cursor CLI, Anti Gravity, OpenCode, and their ... CLI / ... IDE variants normalize automatically |
auto-detect |
IDE_OTEL_LOCAL_SPANS |
Save hook spans locally as JSONL files for agent analysis (.state/local_spans/*.jsonl) |
unset |
IDE_OTEL_CAPTURE_TEXT |
Include prompt/response text in spans | false |
IDE_OTEL_MASK_PROMPTS |
Redact emails, tokens, usernames from text | false |
IDE_OTEL_TEXT_MAX_CHARS |
Max characters for captured text | 4000 |
IDE_OTEL_CAPTURE_TOOL_INPUT_CONTENT |
Include tool input content in logs | false |
IDE_OTEL_CAPTURE_TOOL_DEFINITIONS |
Include tool definitions in spans | false |
OTel Logs
| Variable | Description | Default |
|---|---|---|
IDE_OTEL_ENABLE_LOGS |
Enable OTel Logs signal export (OTLP) | true |
IDE_OTEL_MCP_LOG_PAYLOAD |
Include full MCP input/output payloads in logs | true |
IDE_OTEL_LOG_ALL_EVENTS |
Emit OTel log records for all hook events (not just MCP/shell/tool) | false |
OTEL_EXPORTER_OTLP_LOGS_ENDPOINT |
Override OTLP logs endpoint (auto-derived from traces endpoint if not set) | — |
Resource Attributes
| Variable | Description | Default |
|---|---|---|
OTEL_RESOURCE_ATTRIBUTES |
Comma-separated key=value pairs |
— |
IDE_OTEL_APP_NAME |
Application name | ide-agent |
IDE_OTEL_SUBSYSTEM_NAME |
Subsystem name (Coralogix) | ide-hooks |
Logging & Debug
| Variable | Description | Default |
|---|---|---|
IDE_OTEL_LOG_LEVEL |
Log level (DEBUG, INFO, WARNING, ERROR) |
WARNING |
IDE_OTEL_LOG_FILE |
Log file path | <hook-home>/otel_hook.log |
IDE_OTEL_LOG_EVENTS |
Log each hook event to file | false |
IDE_OTEL_DEBUG_CONSOLE |
Print spans to stdout (for debugging) | false |
Advanced (Rarely Needed)
These settings have sensible defaults and typically don't need to be changed:
| Variable | Description | Default |
|---|---|---|
OTEL_EXPORTER_OTLP_INSECURE |
gRPC only: true for plaintext, false for TLS |
true |
IDE_OTEL_DISABLE_BATCH |
Disable OpenTelemetry batch span processor | false |
IDE_OTEL_STATE_TTL_SECONDS |
TTL for state files before cleanup | 86400 |
IDE_OTEL_STATE_CLEANUP_INTERVAL_SECONDS |
Minimum interval between cleanup runs | 3600 |
IDE_OTEL_STATE_LOCK_TIMEOUT_SECONDS |
Max time to wait for state file locks | 2 |
IDE_OTEL_HOOK_HOME |
Override the hook's writable home directory (config, state, venv, log) | See below |
IDE_OTEL_HOOK_HOME: Whenotel-hookruns from an installed package (i.e. the module lives inside site-packages), the hook automatically uses$XDG_DATA_HOME/opentelemetry-hooks(defaulting to~/.local/share/opentelemetry-hooks) instead of the package directory, so all writable files are placed in a user-owned location. SetIDE_OTEL_HOOK_HOMEto an absolute path to override this location explicitly (useful for project-local or shared deployments). When running from a source checkout or a directly-copied script, the directory that containsotel_hook.pyis used as before.
Hook Stdout Response
The hook writes a JSON response to stdout for the IDE/client.
- Default (backward compatible):
{"continue": true}
- If
IDE_OTEL_LOCAL_SPANSis explicitly set (trueorfalse), the response includes:
{"continue": true, "local_spans": true}
For the stdout response field, local_spans uses IDE_OTEL_LOCAL_SPANS when set; otherwise internal behavior falls back to IDE_OTEL_BATCH_ON_STOP.
Local Trace Files (Agent-Friendly)
When local trace saving is enabled, each hook event is also written to JSONL in:
.cursor/hooks/opentelemetry-hook/.state/local_spans/<session_key>.jsonl.cursor/hooks/opentelemetry-hook/.state/local_spans/unscoped.jsonl(if no session key exists)
Each line is a single JSON object, for example:
{
"timestamp_ns": 1771976482308258082,
"event": "UserPromptSubmit",
"ide": "copilot",
"session_key": "agent-s1",
"generation_key": null,
"data": {
"hook_event_name": "beforeSubmitPrompt",
"session_id": "agent-s1",
"prompt": "hello"
}
}
MDM / Managed Configuration
For enterprise deployments, configuration can be pushed to developer machines via MDM (Mobile Device Management) systems such as Jamf, Intune, or Group Policy. MDM-managed settings override otel_config.json values but can still be overridden by environment variables.
Precedence (highest to lowest):
- Environment variables
- MDM-managed configuration (macOS plist / Windows registry)
otel_config.jsonfile- Built-in defaults
macOS (Configuration Profile)
The hook reads managed preferences from the domain dev.o11y.opentelemetry-hook. Deploy a .mobileconfig profile via Jamf, Mosyle, or Apple Business Manager with the following payload:
<dict>
<key>PayloadType</key>
<string>dev.o11y.opentelemetry-hook</string>
<key>OTEL_EXPORTER_OTLP_ENDPOINT</key>
<string>https://otel-collector.corp.example.com:4317</string>
<key>OTEL_EXPORTER_OTLP_PROTOCOL</key>
<string>grpc</string>
<key>OTEL_SERVICE_NAME</key>
<string>corp-ide-agent</string>
<key>IDE_OTEL_CAPTURE_TEXT</key>
<string>false</string>
</dict>
The managed plist is read from:
/Library/Managed Preferences/dev.o11y.opentelemetry-hook.plist(device-level)~/Library/Managed Preferences/dev.o11y.opentelemetry-hook.plist(user-level fallback)
Windows (Registry / Group Policy)
The hook reads string values from the Windows registry under:
HKEY_LOCAL_MACHINE\SOFTWARE\Policies\OpenTelemetryHook
with a fallback to HKEY_CURRENT_USER. Deploy via Intune, Group Policy (ADMX), or any MDM that manages registry keys:
| Registry Value Name | Type | Example |
|---|---|---|
OTEL_EXPORTER_OTLP_ENDPOINT |
REG_SZ |
https://otel-collector.corp.example.com:4317 |
OTEL_SERVICE_NAME |
REG_SZ |
corp-ide-agent |
IDE_OTEL_CAPTURE_TEXT |
REG_SZ |
false |
Any key from the Configuration Reference can be set via MDM.
Backend Examples
Jaeger (Local Development)
docker run -d --name jaeger \
-p 4317:4317 -p 4318:4318 -p 16686:16686 \
jaegertracing/all-in-one:latest
{
"OTEL_EXPORTER_OTLP_ENDPOINT": "http://localhost:4317",
"OTEL_EXPORTER_OTLP_PROTOCOL": "grpc",
"OTEL_SERVICE_NAME": "ide-agent",
"IDE_OTEL_BATCH_ON_STOP": "true"
}
View traces at http://localhost:16686
Jaeger + Local File Export
Send traces to Jaeger and save them as local JSONL files for agent analysis or offline inspection:
docker run -d --name jaeger \
-p 4317:4317 -p 4318:4318 -p 16686:16686 \
jaegertracing/all-in-one:latest
{
"OTEL_EXPORTER_OTLP_ENDPOINT": "http://localhost:4317",
"OTEL_EXPORTER_OTLP_PROTOCOL": "grpc",
"OTEL_SERVICE_NAME": "ide-agent",
"IDE_OTEL_BATCH_ON_STOP": "true",
"IDE_OTEL_LOCAL_SPANS": "true"
}
Traces are exported to Jaeger at http://localhost:16686 and simultaneously written to .state/local_spans/<session>.jsonl.
Local Files Only (No Backend)
Save spans as local JSONL files without sending to any remote backend. Useful for offline debugging, CI environments, or feeding traces back to an agent:
{
"OTEL_SERVICE_NAME": "ide-agent",
"IDE_OTEL_BATCH_ON_STOP": "true",
"IDE_OTEL_LOCAL_SPANS": "true"
}
Omit OTEL_EXPORTER_OTLP_ENDPOINT to skip remote export. Spans are written to .state/local_spans/<session>.jsonl. Each line is a JSON object with trace/span IDs, attributes, and timing — see Local Trace Files for the format.
Coralogix
{
"OTEL_EXPORTER_OTLP_ENDPOINT": "https://ingress.<region>.coralogix.com:443/v1/traces",
"OTEL_EXPORTER_OTLP_PROTOCOL": "http/protobuf",
"OTEL_EXPORTER_OTLP_HEADERS": "authorization=Bearer%20<YOUR_API_KEY>",
"OTEL_SERVICE_NAME": "ide-agent",
"IDE_OTEL_BATCH_ON_STOP": "true"
}
Replace <region> with your Coralogix domain (e.g., us1, eu1, ap1).
If Coralogix requires cx.application.name, add it via OTEL_RESOURCE_ATTRIBUTES:
{
"OTEL_RESOURCE_ATTRIBUTES": "cx.application.name=ide-agent"
}
Datadog
{
"OTEL_EXPORTER_OTLP_ENDPOINT": "http://localhost:4317",
"OTEL_EXPORTER_OTLP_PROTOCOL": "grpc",
"OTEL_SERVICE_NAME": "ide-agent",
"IDE_OTEL_BATCH_ON_STOP": "true"
}
Requires the Datadog Agent with OTLP ingestion enabled.
Grafana / Tempo
{
"OTEL_EXPORTER_OTLP_ENDPOINT": "https://otlp-gateway-<zone>.grafana.net/otlp",
"OTEL_EXPORTER_OTLP_PROTOCOL": "http/protobuf",
"OTEL_EXPORTER_OTLP_HEADERS": "authorization=Basic%20<BASE64_CREDENTIALS>",
"OTEL_SERVICE_NAME": "ide-agent",
"IDE_OTEL_BATCH_ON_STOP": "true"
}
Honeycomb
{
"OTEL_EXPORTER_OTLP_ENDPOINT": "https://api.honeycomb.io",
"OTEL_EXPORTER_OTLP_PROTOCOL": "http/protobuf",
"OTEL_EXPORTER_OTLP_HEADERS": "x-honeycomb-team=<YOUR_API_KEY>",
"OTEL_SERVICE_NAME": "ide-agent",
"IDE_OTEL_BATCH_ON_STOP": "true"
}
Span Attributes
Common (All Spans)
| Attribute | Description |
|---|---|
gen_ai.client.hook.event |
Canonical event name (PascalCase) |
gen_ai.client.name |
Outer IDE or hook host (cursor, copilot, claude, opencode, etc.) |
gen_ai.client.agent_engine |
Inner agent engine when it differs from the outer IDE (for example Cursor running Claude Code) |
gen_ai.client.session_id |
Session identifier |
gen_ai.client.generation_id |
Generation identifier (Cursor) |
gen_ai.client.workspace |
Workspace / working directory |
gen_ai.client.timestamp |
Event timestamp (ISO 8601) |
gen_ai.system |
Deprecated legacy GenAI system/provider attribute retained for backward compatibility |
gen_ai.operation.name |
chat, execute_tool, or invoke_agent |
GenAI (When Available)
| Attribute | Description |
|---|---|
gen_ai.provider.name |
Canonical GenAI provider when inferred from payload/model metadata |
gen_ai.request.model |
Requested model name |
gen_ai.response.model |
Response model name |
gen_ai.conversation.id |
Session / conversation ID |
gen_ai.usage.input_tokens |
Input token count |
gen_ai.usage.output_tokens |
Output token count |
gen_ai.usage.cache_creation.input_tokens |
Cache-write input token count when provided |
gen_ai.usage.cache_read.input_tokens |
Cache-read input token count when provided |
gen_ai.request.temperature |
Temperature setting |
gen_ai.request.max_tokens |
Max tokens setting |
gen_ai.request.choice.count |
Requested number of choices/candidates |
gen_ai.output.type |
Requested output modality (text, json, image, speech) |
gen_ai.agent.id / gen_ai.agent.name |
Agent identity when the hook payload includes agent metadata |
gen_ai.response.finish_reasons |
Finish reasons array |
gen_ai.system_instructions |
System instructions (opt-in text capture) |
gen_ai.input.messages |
Input messages (opt-in) |
gen_ai.output.messages |
Output messages (opt-in) |
Event-Specific
| Event | Key Attributes |
|---|---|
UserPromptSubmit |
gen_ai.client.composer_mode, gen_ai.request.model |
PreToolUse / PostToolUse |
gen_ai.client.tool_name, gen_ai.client.tool_id, gen_ai.client.duration_ms |
PostToolUseFailure |
gen_ai.client.tool_name, gen_ai.client.error |
BeforeShellExecution / AfterShellExecution |
gen_ai.client.command, gen_ai.client.cwd, gen_ai.client.exit_code |
BeforeMCPExecution / AfterMCPExecution |
gen_ai.client.mcp_server, gen_ai.client.mcp_tool |
BeforeReadFile / AfterFileEdit |
gen_ai.client.file_path, gen_ai.client.edits |
SubagentStart / SubagentStop |
gen_ai.client.subagent_type, gen_ai.client.agent_id |
Stop |
gen_ai.client.status, gen_ai.client.loop_count |
ErrorOccurred |
gen_ai.client.error, gen_ai.client.is_interrupt |
OTel Logs (MCP, Shell, Tool Events)
When IDE_OTEL_ENABLE_LOGS=true (default), the hook emits structured OpenTelemetry log records alongside traces. Log records are automatically correlated with the active span's trace context, so you can jump between traces and logs in your backend.
What gets logged
| Event Type | Log Records | Payload Control |
|---|---|---|
MCP calls (BeforeMCPExecution, AfterMCPExecution) |
Always when logs enabled | IDE_OTEL_MCP_LOG_PAYLOAD |
Shell execution (BeforeShellExecution, AfterShellExecution) |
Always when logs enabled | IDE_OTEL_MCP_LOG_PAYLOAD |
Tool usage (PreToolUse, PostToolUse, PostToolUseFailure) |
Always when logs enabled | IDE_OTEL_CAPTURE_TOOL_INPUT_CONTENT |
| All other events | Only when IDE_OTEL_LOG_ALL_EVENTS=true |
— |
MCP Log Attributes
| Attribute | Description |
|---|---|
gen_ai.client.mcp_server |
MCP server name |
gen_ai.client.mcp_tool |
MCP tool name |
gen_ai.client.mcp.input |
Full input payload (opt-in) |
gen_ai.client.mcp.input.length |
Input payload size |
gen_ai.client.mcp.input.sha256 |
Input payload hash |
gen_ai.client.mcp.output |
Full output payload (opt-in) |
gen_ai.client.mcp.output.length |
Output payload size |
gen_ai.client.mcp.output.sha256 |
Output payload hash |
gen_ai.client.mcp.duration_ms |
MCP call duration |
gen_ai.client.mcp.stdout |
Server stdout (if available) |
gen_ai.client.mcp.stderr |
Server stderr (if available) |
Endpoint Derivation
The logs endpoint is derived automatically:
- If
OTEL_EXPORTER_OTLP_LOGS_ENDPOINTis set, it's used directly - Otherwise,
/v1/tracesis replaced with/v1/logsinOTEL_EXPORTER_OTLP_ENDPOINT - For gRPC, the same endpoint serves all signals
Example: https://ingress.us1.coralogix.com:443/v1/traces → https://ingress.us1.coralogix.com:443/v1/logs
Session-level Batching
When IDE_OTEL_BATCH_ON_STOP=true (recommended):
- SessionStart: Pre-generates a
trace_idshared by all spans in the session. Stored in.state/sessions/. - Generation events: Buffered to
.state/batches/<generation_id>.jsonl. - Stop: Flushes the generation's events as a
gen_ai.client.generationspan with child event spans. All share the session'strace_id. Exported immediately to avoid data loss. - SessionEnd: Emits the root
gen_ai.client.sessionspan covering the full session duration. Cleans up state files.
For IDEs without a generation_id (Copilot), the hook auto-derives generation boundaries from UserPromptSubmit → Stop cycles using an internal counter.
IDE Detection
The hook auto-detects which IDE is calling it:
| Signal | IDE |
|---|---|
Parent process tree (ps parent-chain walk) |
Preferred detection for supported IDEs such as Cursor, Copilot / VS Code, Claude Code, and OpenCode |
IDE_OTEL_IDE_NAME env var |
Explicit override for generic hook runners or manual debugging |
Self-reported ide_name, client, or source_app values such as GitHub Copilot, GitHub Copilot CLI, GitHub Copilot Chat, Claude Code, Claude Code CLI, Anthropic Claude Code, Cursor IDE, Cursor CLI, Anti Gravity, Anti Gravity CLI, or OpenCode / OpenCode CLI (case-insensitive, hyphen/space-insensitive) |
Normalized to the canonical gen_ai.client.name |
conversation_id or generation_id in input |
Cursor |
transcript_path, permission_mode, or notification_type |
Claude Code |
session_id only (no Cursor-specific fields) |
GitHub Copilot |
Detection order is: (1) parent process tree, (2) explicit IDE_OTEL_IDE_NAME, (3) self-reported payload fields, then (4) heuristics. setup.sh now relies on process discovery for generated Cursor, Copilot, and Claude configs, while the env var remains available as an escape hatch for generic runners and debugging.
The detected outer IDE is recorded on spans as gen_ai.client.name and is also exported as the gen_ai.system resource attribute via OTEL_RESOURCE_ATTRIBUTES for backward compatibility. When nested signals indicate a different inner engine (for example Cursor hosting Claude Code), the hook additionally records gen_ai.client.agent_engine. When the hook can infer a provider from the payload, it also sets gen_ai.provider.name as the canonical provider attribute (v1.37+).
File Structure
.cursor/
├── hooks.json # Active Cursor hooks config (created by setup.sh)
└── hooks/
└── opentelemetry-hook/
├── setup.sh # One-command setup (creates/merges hooks.json)
├── otel_hook.py # Main hook implementation (exposed as `otel-hook` when installed)
├── otel_config.json # Your config (gitignored, auto-created)
├── otel_config.example.json # Config template
├── README.md # This file
├── examples/
│ ├── hooks.example.json # Full Cursor hooks template
│ ├── cursor-hooks.example.json # Minimal Cursor hooks template
│ ├── copilot-hooks.example.json # GitHub Copilot hooks template
│ ├── claude-hooks.example.json # Claude Code hooks template
│ └── antigravity-workflow.example.md # Antigravity workflow template
├── .gitignore # Excludes secrets, venv, state
├── .venv/ # Python venv (auto-provisioned)
└── .state/ # Runtime state
├── sessions/ # Session trace context
└── batches/ # Generation event buffers
Privacy & Security
What Gets Sent (by default)
- Event names and timing
- Tool/command names
- File paths
- Prompt/response length and SHA-256 hash (not content)
Opt-in Content Capture
Set IDE_OTEL_CAPTURE_TEXT=true to include prompt/response text. Combine with IDE_OTEL_MASK_PROMPTS=true to redact:
- Email addresses
- Long tokens / API keys
- macOS usernames from paths
Never Sent
- API keys or credentials (automatically filtered)
- File contents (unless tool_response capture is enabled)
- Raw code
Troubleshooting
Check the log
tail -f .cursor/hooks/opentelemetry-hook/otel_hook.log
Enable debug output
{
"IDE_OTEL_LOG_LEVEL": "DEBUG",
"IDE_OTEL_DEBUG_CONSOLE": "true",
"IDE_OTEL_LOG_EVENTS": "true"
}
Test manually
echo '{"hook_event_name":"SessionStart","session_id":"test-123"}' | otel-hook
Common issues
| Problem | Fix |
|---|---|
opentelemetry-sdk not installed |
Auto-provisioning may still be in progress; wait ~30s and retry, or run .venv/bin/pip install opentelemetry-sdk opentelemetry-exporter-otlp |
Missing API key |
Set OTEL_EXPORTER_OTLP_HEADERS with your auth token in config |
cx.application.name required |
Coralogix needs this — set automatically, or add to OTEL_RESOURCE_ATTRIBUTES |
| Orphan spans | Enable IDE_OTEL_BATCH_ON_STOP=true for session-level traces |
| No traces appearing | Check endpoint, protocol, and auth headers in config. Verify the backend is running and reachable. |
| Wrong IDE detected | Check the parent process chain and input payload first; for generic runners or debugging, set IDE_OTEL_IDE_NAME explicitly in the hook command |
| Traces going to the wrong backend | Verify OTEL_EXPORTER_OTLP_ENDPOINT points to the intended backend |
Contributing
Contributions are welcome. To get started:
git clone https://github.com/o11y-dev/opentelemetry-hooks.git
cd opentelemetry-hooks
pip install -r requirements-dev.txt
python -m pytest tests/ -v
Please open an issue first if you plan a large change.
Credits
- Built on pure OpenTelemetry Python SDK
- Uses OpenTelemetry GenAI Semantic Conventions
- Supports GitHub Copilot hooks, Cursor IDE / CLI hook payloads, Claude Code hook payloads, and compatible runners such as OpenCode
License
MIT
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file opentelemetry_hooks-0.11.0.tar.gz.
File metadata
- Download URL: opentelemetry_hooks-0.11.0.tar.gz
- Upload date:
- Size: 97.8 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
514c420275f5b6ccd219e2058c46c3590f701c1220e733ecbc23364803dbacd7
|
|
| MD5 |
752daf63f57e9b67a16f0b0b75e1f2a1
|
|
| BLAKE2b-256 |
2b0547d4c4270d76898037f9dc45586e83277a972676271655322d7029757823
|
File details
Details for the file opentelemetry_hooks-0.11.0-py3-none-any.whl.
File metadata
- Download URL: opentelemetry_hooks-0.11.0-py3-none-any.whl
- Upload date:
- Size: 43.5 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
83c6520e2d42d478221486b437f4afd83d22266965b6abd92ecbad3c59bcfbcb
|
|
| MD5 |
759ba6697fddf7833d0452487f9b4625
|
|
| BLAKE2b-256 |
610e7fb88ab9f51a3c25e0906693618694f65e58faf81323ceaf1010cdef434c
|