Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions src/docs.json
Original file line number Diff line number Diff line change
Expand Up @@ -1789,6 +1789,7 @@
"langsmith/trace-claude-code",
"langsmith/trace-with-codex",
"langsmith/trace-with-opencode",
"langsmith/trace-with-cursor",
"langsmith/trace-with-pi",
"langsmith/trace-with-instructor",
"langsmith/trace-with-n8n",
Expand Down
3 changes: 3 additions & 0 deletions src/images/providers/dark/cursor.svg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
3 changes: 3 additions & 0 deletions src/images/providers/light/cursor.svg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
6 changes: 6 additions & 0 deletions src/langsmith/integrations.mdx
Original file line number Diff line number Diff line change
Expand Up @@ -203,6 +203,12 @@ mode: wide
<span className="font-semibold">OpenCode</span>
</a>

<a href="/langsmith/trace-with-cursor" className="flex items-center justify-center gap-1.5 p-2 rounded-lg border border-gray-200 dark:border-gray-700 hover:border-gray-300 dark:hover:border-gray-600 no-underline ">
<img className="block dark:hidden w-5 h-5" src="/images/providers/light/cursor.svg" alt="" />
<img className="hidden dark:block w-5 h-5" src="/images/providers/dark/cursor.svg" alt="" />
<span className="font-semibold">Cursor</span>
</a>

<a href="/langsmith/trace-with-instructor" className="flex items-center justify-center gap-1.5 p-2 rounded-lg border border-gray-200 dark:border-gray-700 hover:border-gray-300 dark:hover:border-gray-600 no-underline ">
<img className="block dark:hidden w-5 h-5" src="/images/providers/light/instructor.svg" alt="" />
<img className="hidden dark:block w-5 h-5" src="/images/providers/dark/instructor.svg" alt="" />
Expand Down
179 changes: 179 additions & 0 deletions src/langsmith/trace-with-cursor.mdx
Original file line number Diff line number Diff line change
@@ -0,0 +1,179 @@
---
title: Trace Cursor sessions
sidebarTitle: Cursor
description: Capture Cursor agent turns, tool calls, model metadata, token usage, attachments, and subagents in LangSmith.
---

Trace [Cursor](https://cursor.com/) agent sessions in LangSmith using [Cursor hooks](https://cursor.com/docs/agent/hooks). Each turn becomes one trace, grouped into a thread per Cursor conversation, which captures prompts, model responses, tool calls, token usage, and subagent activity. Use the traces to debug agent behavior, audit tool calls, track token spend per turn, and compare Cursor runs.

The plugin source is available at [`langchain-ai/langsmith-cursor-plugins`](https://github.com/langchain-ai/langsmith-cursor-plugins).

## Prerequisites

Before setting up tracing, ensure you have:

- [Cursor](https://cursor.com/) installed.
- [Node.js](https://nodejs.org/) 22.13 or later. The hooks use the built-in `node:sqlite` module to recover attachments from Cursor's local database.
- A [LangSmith API key](/langsmith/create-account-api-key).

## Install and enable the plugin
Comment thread
harisaiharish marked this conversation as resolved.

Install the plugin directly from the GitHub repository in Cursor's settings:

1. Open **Cursor → Settings → Plugins**.
2. Paste `https://github.com/langchain-ai/langsmith-cursor-plugins` into the plugin link field.
3. Confirm to add **LangSmith Tracing for Cursor**.

This is the recommended path: it installs the hooks with no clone or build step, since the repository ships a precompiled bundle.

<Warning>
Fully restart Cursor after installing so it reloads `hooks.json`.
</Warning>

### Alternative: install from a local clone

Install the hooks from a local clone of [`langsmith-cursor-plugins`](https://github.com/langchain-ai/langsmith-cursor-plugins). This writes `~/.cursor/hooks.json` for all Cursor projects:

```bash
node scripts/install.mjs
```

To scope the hooks to a single project instead, run the installer from the project directory:

```bash
node scripts/install.mjs --project
```

To preview the hook configuration without writing it, run:

```bash
node scripts/install.mjs --print
```

The installer merges its entries with any existing `hooks.json`. The committed `bundle/` means these commands run without `pnpm install` or `pnpm build` — those are only needed when contributing changes to the TypeScript source.

Restart Cursor after installing so it reloads `hooks.json`.

## Configure tracing

Tracing is disabled until both `enabled` (or `TRACE_TO_LANGSMITH=true`) and an API key are set. Configure credentials with [environment variables](#environment-variables), a [JSON config file](#config-file), or both.

### Environment variables

Every `LANGSMITH_CURSOR_*` variable also accepts the shorter `LANGSMITH_*` form. When both are set, `LANGSMITH_CURSOR_*` takes precedence.

| Variable | Default | Description |
| --- | --- | --- |
| `TRACE_TO_LANGSMITH` | `false` | Set to `"true"` to enable tracing. |
| `LANGSMITH_CURSOR_API_KEY` | - | LangSmith API key. Falls back to `LANGSMITH_API_KEY`. |
| `LANGSMITH_CURSOR_ENDPOINT` | `https://api.smith.langchain.com` | LangSmith API URL. Falls back to `LANGSMITH_ENDPOINT`. |
| `LANGSMITH_CURSOR_PROJECT` | `cursor` | LangSmith project name. Falls back to `LANGSMITH_PROJECT`. |
| `LANGSMITH_CURSOR_METADATA` | - | JSON object merged into root trace metadata. |
| `LANGSMITH_CURSOR_RUNS_ENDPOINTS` | - | JSON array of replica destinations. |
| `LANGSMITH_CURSOR_ATTACHMENTS` | `true` | Set to `"false"` to disable attachment enrichment. |
| `LANGSMITH_CURSOR_DB_PATH` | platform default | Override the Cursor `state.vscdb` path used for attachments. |
| `LANGSMITH_CURSOR_STATE_FILE` | `~/.cursor/langsmith-state.json` | Override the on-disk event-buffer state file. |
| `LANGSMITH_CURSOR_LOG_FILE` | `~/.cursor/langsmith-hook.log` | Override the hook log file. |
| `LANGSMITH_CURSOR_DEBUG` | `false` | Set to `"true"` to enable verbose hook logging. |

Add the variables to your shell configuration file (`~/.zshrc`, `~/.bashrc`, or `~/.bash_profile`):

```bash
export TRACE_TO_LANGSMITH="true"
export LANGSMITH_CURSOR_API_KEY="<your-langsmith-api-key>"
export LANGSMITH_CURSOR_PROJECT="cursor"
```

To verify hook activity, tail the log file:

```bash
tail -f ~/.cursor/langsmith-hook.log
```

### Config file

Use `~/.cursor/langsmith.json` for global defaults or `./.cursor/langsmith.json` for project-level settings. Settings resolve in this order, with later sources overriding earlier ones: defaults, global config, project config, environment variables.

```json
{
"enabled": true,
"api_key": "<your-langsmith-api-key>",
"api_url": "https://api.smith.langchain.com",
"project": "cursor"
}
```

| Field | Environment variable | Default | Description |
| --- | --- | --- | --- |
| `enabled` | `TRACE_TO_LANGSMITH` | `false` | Set to `true` to enable tracing. |
| `api_key` | `LANGSMITH_CURSOR_API_KEY`, `LANGSMITH_API_KEY` | - | LangSmith API key. |
| `api_url` | `LANGSMITH_CURSOR_ENDPOINT`, `LANGSMITH_ENDPOINT` | `https://api.smith.langchain.com` | LangSmith API URL. |
| `project` | `LANGSMITH_CURSOR_PROJECT`, `LANGSMITH_PROJECT` | `cursor` | LangSmith project name. |
| `metadata` | `LANGSMITH_CURSOR_METADATA`, `LANGSMITH_METADATA` | - | Object merged into root trace metadata. |
| `replicas` | `LANGSMITH_CURSOR_RUNS_ENDPOINTS`, `LANGSMITH_RUNS_ENDPOINTS` | - | Additional LangSmith destinations to replicate traces to. |
| `attachments` | `LANGSMITH_CURSOR_ATTACHMENTS` | `true` | Set to `false` to skip enriching turns with image and file attachment bytes from Cursor's local database. |
| `cursor_db_path` | `LANGSMITH_CURSOR_DB_PATH` | platform default | Override the Cursor `state.vscdb` path used for attachments. |

Keep config files that include API keys out of version control.

## What gets traced

The plugin listens to Cursor hooks and assembles one trace per agent turn:

- **Turns**: each turn becomes its own trace, grouped into a thread using `thread_id` = Cursor's `conversation_id`. The trace nests the model run and any tool or subagent runs underneath the turn.
- **Token usage**: per-turn `usage_metadata` on the model run.
- **Model and provider**: `ls_model_name` and `ls_provider`, normalized from Cursor's model label to a canonical provider id (for example, `claude-4.6-sonnet` becomes `claude-sonnet-4-6`). Auto mode reports `default` with provider `cursor`.
- **Tool calls**: tool runs for both successful calls and failures, with inputs and outputs.
- **Attachments**: image and file attachments recovered from Cursor's local database and rendered inline on the user message. Set `attachments` to `false` to skip this step.
- **Subagents**: each subagent appears as a nested chain run with its own tool calls underneath, linked to the parent turn.

<Note>
The plugin does not compute cost locally. Because `ls_model_name` is normalized to a canonical id and `usage_metadata` carries the token breakdown, LangSmith's server-side model price table renders cost in the UI. Auto mode reports `default`, which LangSmith cannot price.
</Note>

<Warning>
The plugin uploads Cursor conversation data, including prompts, model responses, tool inputs and outputs, and recovered attachments. Do not enable tracing for sessions that contain data you do not want stored in LangSmith.
</Warning>

### Trace metadata

Every run carries the shared `coding-agent-v1` metadata contract on `run.extra.metadata`, which lets traces from any coding agent (Claude Code, Codex, Cursor) be identified and grouped with the same stable keys.

| Scope | Keys |
| --- | --- |
| Always present | `ls_agent_kind` (`"coding_agent"`), `ls_integration` (`"cursor"`), `ls_agent_runtime` (`"Cursor"`), `ls_trace_schema_version` (`"coding-agent-v1"`), `thread_id` (= Cursor's `conversation_id`). |
| Present where known | `ls_integration_version`, `ls_agent_runtime_version` (Cursor's `cursor_version`), `turn_id` (= Cursor's `generation_id`), `turn_number`, `repository_url`, `repository_provider`, `repository_name`, `git_branch`, `git_commit_sha`, `cwd`. |
| Contextual | `local_username`, `user_email` (provisional). |
| Subagent runs only | `ls_subagent_id`, `ls_subagent_type`. |
| Tool runs only | `ls_tool_name` (only emitted when the run name differs from the native tool name). |
| Model and tool runs only | `ls_provider`, `ls_model_name`, `ls_invocation_params`, `usage_metadata`. |

Cursor's hooks do not expose stable sources for `user_id`, `sandbox_type`, or `approval_policy`, so those keys are omitted.

## View traces in LangSmith

Open the configured LangSmith project (default `cursor`) and complete a Cursor turn. The plugin uploads one trace per turn with this structure:

```
Cursor Turn N (chain)
├── <provider> (llm) model and provider, token usage, assistant text
├── Read / Shell / ... (tool)
└── Task (tool) subagent (type and task)
```

To group related turns, filter on `thread_id` in the **Threads** tab of the project.

## Known limitations

- **Subagent token usage**: Cursor does not expose a per-subagent usage breakdown via hooks or its local database, so a subagent's `Task` run carries its tool calls but no token counts.

## Troubleshooting

If traces do not appear in LangSmith:

- Confirm `TRACE_TO_LANGSMITH=true` or `"enabled": true` is set in a config file the Cursor process can see.
- Confirm `LANGSMITH_CURSOR_API_KEY` or `LANGSMITH_API_KEY` is set and valid.
- Confirm Cursor was fully restarted after the hooks were installed.
- Tail the hook log for errors: `tail -f ~/.cursor/langsmith-hook.log`.
- Enable verbose logging with `LANGSMITH_CURSOR_DEBUG=true` and re-check the log.
- If runs land in the wrong project, set `LANGSMITH_CURSOR_PROJECT` or the `project` config key.
Loading