forked from farhoodlabs/paperclip
534aee66ae
## Thinking Path
> - Paperclip orchestrates AI agents for zero-human companies
> - There are many adapter types, one per agent-runtime product (Claude,
Codex, OpenCode, Cursor local CLI, etc.)
> - Cursor shipped a public TypeScript SDK on 2026-04-29 that exposes
Cursor's full hosted-agent platform (cloud VMs, harness, MCP, skills,
hooks)
> - Paperclip had no first-class adapter for this — agents that wanted
to use Cursor's managed cloud runtime had to fall back to the local CLI
adapter, which loses the cloud session, streaming, and durable run model
> - This PR adds a new `cursor_cloud` adapter built directly on
`@cursor/sdk`, with Paperclip's heartbeat mapped to Cursor's
durable-agent + per-run model
> - The benefit is that any Paperclip agent can now drive a Cursor cloud
agent across heartbeats with native session reuse, streaming, and
cancellation, while Paperclip remains the source of truth for issue/task
state
## What Changed
- New built-in adapter package `packages/adapters/cursor-cloud` (15
files, ~1.7k LOC) backed by `@cursor/sdk` ^1.0.12
- `src/server/execute.ts` — SDK-first lifecycle: `Agent.create` /
`Agent.resume` / `Agent.getRun` / `agent.send` / `run.stream` /
`run.wait`, with session reuse keyed on the (runtime env type, env name,
repo set) tuple
- `src/server/session.ts` — codec for `cursorAgentId` + `latestRunId` +
repo metadata, persisted in `runtime.sessionParams`
- `src/server/test.ts` — environment probe via `Cursor.me()` and
optional model validation via `Cursor.models.list()`
- `src/ui/parse-stdout.ts` + `src/cli/format-event.ts` — normalize
Cursor SDK message types (`status`, `thinking`, `assistant`, `user`,
`tool_call`, `tool_result`, `result`) into Paperclip transcript events
for the UI and CLI
- Registrations: `packages/shared/src/constants.ts`,
`packages/adapter-utils/src/session-compaction.ts`,
`server/src/adapters/{registry,builtin-adapter-types}.ts`,
`ui/src/adapters/{registry,adapter-display-registry}.ts` +
`ui/src/adapters/cursor-cloud/index.ts`, `cli/src/adapters/registry.ts`,
plus workspace deps in `cli`/`server`/`ui` `package.json`
- `ui/src/components/AgentConfigForm.tsx` — hide local-Cursor
`mode`/thinking-effort field for `cursor_cloud` (different config
surface)
- 11 vitest tests covering execute paths (fresh create, matching-resume,
active-run reattach, non-finished result), session codec round-trip,
transcript parsing, and config building
## Verification
Reviewer steps:
```bash
pnpm install
pnpm --filter @paperclipai/adapter-cursor-cloud typecheck # → clean
pnpm vitest run packages/adapters/cursor-cloud # → 11/11 passing
```
End-to-end check against a real Cursor cloud agent (requires
`CURSOR_API_KEY` and Cursor GitHub-app install on the target repo):
1. Create a `cursor_cloud` agent in Paperclip with `repoUrl` set to the
test repo, `repoStartingRef: main`, and `env.CURSOR_API_KEY` set
2. Trigger a heartbeat → adapter calls `Agent.create({ cloud: { env: {
type: "cloud" }, repos: [...] } })`, streams events, terminates on
`finished`
3. Trigger a second heartbeat → adapter calls `Agent.resume` or
`agent.send` follow-up depending on prior-run state, reusing
`cursorAgentId`
4. The Paperclip UI/CLI transcript reflects Cursor `status` / `thinking`
/ `assistant` events as they stream
5. Cancellation from Paperclip maps to `run.cancel()` or Cloud API v1
`cancelRun` for cross-heartbeat cancellation
A direct-SDK smoke run against a real repo (devinfoley/my_test_project @
main) confirmed: `Cursor.me()` ok → `Agent.create` → `agent.send` →
`run.stream()` (30 events) → terminal status `finished` in ~11s.
## Risks
- **New adapter, additive only.** No existing adapter or registry is
replaced; current `cursor` local-CLI adapter is untouched. Default
behavior of any existing agent is unchanged.
- **External dependency on `@cursor/sdk`.** Cursor's SDK is v1.0.x and
may evolve. Mocked unit tests cover the public surface used here; if the
SDK breaks compatibility we update the adapter independently.
- **Cost/budget.** `cursor_cloud` runs on Cursor's billed cloud VMs;
operators must understand they are spending money outside Paperclip's
budget controls when they enable this adapter. Same shape as other
API-billed adapters.
- **No webhook support in V1.** The SDK already provides
stream/wait/cancel/reattach, so V1 does not require a public callback
URL. If a future use case needs out-of-band wakes, we add a Cloud API v1
webhook bridge as a separate change. This is called out in the issue
plan document.
- **Lockfile.** Per repo policy, `pnpm-lock.yaml` is intentionally not
in this PR — CI's lockfile workflow will update it on merge given the
manifest changes.
## Model Used
- Provider: Anthropic Claude (via Claude Code / Paperclip `claude_local`
adapter)
- Model: `claude-opus-4-7` (Claude Opus 4.7), knowledge cutoff January
2026
- Mode: standard tool-use with extended reasoning
- Context: ~200k token window
- Capabilities used: code generation, multi-file edits, shell/test
execution, GitHub PR workflow
## Checklist
- [x] I have included a thinking path that traces from project context
to this change
- [x] I have specified the model used (with version and capability
details)
- [x] I have checked ROADMAP.md and confirmed this PR does not duplicate
planned core work
- [x] I have run tests locally and they pass (11/11 in
`packages/adapters/cursor-cloud`)
- [x] I have added or updated tests where applicable (4 new test files,
11 cases)
- [ ] If this change affects the UI, I have included before/after
screenshots (the only UI change is hiding the local-Cursor mode field on
the `cursor_cloud` adapter — happy to attach a screenshot if the
reviewer wants one)
- [x] I have updated relevant documentation to reflect my changes (issue
plan document supersedes the pre-SDK design; tracked in PAPA-203)
- [x] I have considered and documented any risks above
- [x] I will address all Greptile and reviewer comments before
requesting merge
---------
Co-authored-by: Paperclip <noreply@paperclip.ing>
608 lines
20 KiB
TypeScript
608 lines
20 KiB
TypeScript
import fs from "node:fs/promises";
|
|
import path from "node:path";
|
|
import {
|
|
Agent,
|
|
type AgentOptions,
|
|
type ModelSelection,
|
|
type Run,
|
|
type RunResult,
|
|
type SDKAgent,
|
|
type SDKMessage,
|
|
} from "@cursor/sdk";
|
|
import type { AdapterExecutionContext, AdapterExecutionResult, AdapterInvocationMeta } from "@paperclipai/adapter-utils";
|
|
import {
|
|
DEFAULT_PAPERCLIP_AGENT_PROMPT_TEMPLATE,
|
|
asBoolean,
|
|
asString,
|
|
buildPaperclipEnv,
|
|
joinPromptSections,
|
|
parseObject,
|
|
readPaperclipIssueWorkModeFromContext,
|
|
renderPaperclipWakePrompt,
|
|
renderTemplate,
|
|
stringifyPaperclipWakePayload,
|
|
} from "@paperclipai/adapter-utils/server-utils";
|
|
|
|
type CursorCloudSession = {
|
|
cursorAgentId: string;
|
|
latestRunId?: string;
|
|
runtime: "cloud";
|
|
envType?: "cloud" | "pool" | "machine";
|
|
envName?: string;
|
|
repos: Array<{ url: string; startingRef?: string; prUrl?: string }>;
|
|
};
|
|
|
|
type CursorCloudEvent =
|
|
| { type: "cursor_cloud.init"; sessionId: string; agentId: string; runId?: string; model?: string }
|
|
| { type: "cursor_cloud.status"; status: string; message?: string }
|
|
| { type: "cursor_cloud.message"; message: SDKMessage }
|
|
| {
|
|
type: "cursor_cloud.result";
|
|
status: string;
|
|
result?: string;
|
|
model?: string;
|
|
durationMs?: number;
|
|
git?: unknown;
|
|
error?: string;
|
|
};
|
|
|
|
function asRecord(value: unknown): Record<string, unknown> | null {
|
|
if (typeof value !== "object" || value === null || Array.isArray(value)) return null;
|
|
return value as Record<string, unknown>;
|
|
}
|
|
|
|
function asStringEnvMap(value: unknown): Record<string, string> {
|
|
const parsed = parseObject(value);
|
|
const env: Record<string, string> = {};
|
|
for (const [key, entry] of Object.entries(parsed)) {
|
|
if (typeof entry === "string") {
|
|
env[key] = entry;
|
|
} else if (typeof entry === "object" && entry !== null && !Array.isArray(entry)) {
|
|
const rec = entry as Record<string, unknown>;
|
|
if (rec.type === "plain" && typeof rec.value === "string") env[key] = rec.value;
|
|
}
|
|
}
|
|
return env;
|
|
}
|
|
|
|
function normalizeEnvType(raw: string): "cloud" | "pool" | "machine" {
|
|
const value = raw.trim().toLowerCase();
|
|
if (value === "pool" || value === "machine") return value;
|
|
return "cloud";
|
|
}
|
|
|
|
function trimNullable(value: unknown): string | null {
|
|
return typeof value === "string" && value.trim().length > 0 ? value.trim() : null;
|
|
}
|
|
|
|
function firstNonEmptyLine(text: string): string {
|
|
return (
|
|
text
|
|
.split(/\r?\n/)
|
|
.map((line) => line.trim())
|
|
.find(Boolean) ?? ""
|
|
);
|
|
}
|
|
|
|
function toModelSelection(rawModel: string): ModelSelection | undefined {
|
|
const model = rawModel.trim();
|
|
return model ? { id: model } : undefined;
|
|
}
|
|
|
|
function toSummary(result: RunResult): string | null {
|
|
const direct = trimNullable(result.result);
|
|
if (direct) return firstNonEmptyLine(direct);
|
|
return null;
|
|
}
|
|
|
|
function formatRunError(err: unknown): string {
|
|
if (err instanceof Error && err.message.trim().length > 0) return err.message.trim();
|
|
return String(err);
|
|
}
|
|
|
|
function buildWakeEnv(ctx: AdapterExecutionContext, configEnv: Record<string, string>): Record<string, string> {
|
|
const { runId, agent, context, authToken } = ctx;
|
|
const env: Record<string, string> = {
|
|
...configEnv,
|
|
...buildPaperclipEnv(agent),
|
|
PAPERCLIP_RUN_ID: runId,
|
|
};
|
|
|
|
const wakeTaskId = trimNullable(context.taskId) ?? trimNullable(context.issueId);
|
|
const wakeReason = trimNullable(context.wakeReason);
|
|
const wakeCommentId = trimNullable(context.wakeCommentId) ?? trimNullable(context.commentId);
|
|
const approvalId = trimNullable(context.approvalId);
|
|
const approvalStatus = trimNullable(context.approvalStatus);
|
|
const linkedIssueIds = Array.isArray(context.issueIds)
|
|
? context.issueIds.filter((value): value is string => typeof value === "string" && value.trim().length > 0)
|
|
: [];
|
|
const wakePayloadJson = stringifyPaperclipWakePayload(context.paperclipWake);
|
|
const issueWorkMode = readPaperclipIssueWorkModeFromContext(context);
|
|
|
|
if (wakeTaskId) env.PAPERCLIP_TASK_ID = wakeTaskId;
|
|
if (wakeReason) env.PAPERCLIP_WAKE_REASON = wakeReason;
|
|
if (wakeCommentId) env.PAPERCLIP_WAKE_COMMENT_ID = wakeCommentId;
|
|
if (approvalId) env.PAPERCLIP_APPROVAL_ID = approvalId;
|
|
if (approvalStatus) env.PAPERCLIP_APPROVAL_STATUS = approvalStatus;
|
|
if (linkedIssueIds.length > 0) env.PAPERCLIP_LINKED_ISSUE_IDS = linkedIssueIds.join(",");
|
|
if (wakePayloadJson) env.PAPERCLIP_WAKE_PAYLOAD_JSON = wakePayloadJson;
|
|
if (issueWorkMode) env.PAPERCLIP_ISSUE_WORK_MODE = issueWorkMode;
|
|
if (!trimNullable(env.PAPERCLIP_API_KEY) && authToken) {
|
|
env.PAPERCLIP_API_KEY = authToken;
|
|
}
|
|
|
|
const workspace = parseObject(context.paperclipWorkspace);
|
|
const workspaceMappings: Array<[string, unknown]> = [
|
|
["PAPERCLIP_WORKSPACE_CWD", workspace.cwd],
|
|
["PAPERCLIP_WORKSPACE_SOURCE", workspace.source],
|
|
["PAPERCLIP_WORKSPACE_ID", workspace.workspaceId],
|
|
["PAPERCLIP_WORKSPACE_REPO_URL", workspace.repoUrl],
|
|
["PAPERCLIP_WORKSPACE_REPO_REF", workspace.repoRef],
|
|
["PAPERCLIP_WORKSPACE_BRANCH", workspace.branch],
|
|
["PAPERCLIP_WORKSPACE_WORKTREE_PATH", workspace.worktreePath],
|
|
["AGENT_HOME", workspace.agentHome],
|
|
];
|
|
for (const [key, value] of workspaceMappings) {
|
|
const normalized = trimNullable(value);
|
|
if (normalized) env[key] = normalized;
|
|
}
|
|
|
|
delete env.CURSOR_API_KEY;
|
|
return env;
|
|
}
|
|
|
|
async function buildInstructionsPrefix(
|
|
config: Record<string, unknown>,
|
|
onLog: AdapterExecutionContext["onLog"],
|
|
): Promise<{ prefix: string; notes: string[]; chars: number }> {
|
|
const instructionsFilePath = asString(config.instructionsFilePath, "").trim();
|
|
if (!instructionsFilePath) {
|
|
return { prefix: "", notes: [], chars: 0 };
|
|
}
|
|
|
|
try {
|
|
const contents = await fs.readFile(instructionsFilePath, "utf8");
|
|
const instructionsDir = `${path.dirname(instructionsFilePath)}/`;
|
|
const prefix = `${contents.trim()}\n\nThe above agent instructions were loaded from ${instructionsFilePath}. Resolve any relative file references from ${instructionsDir}.\n`;
|
|
return {
|
|
prefix,
|
|
chars: prefix.length,
|
|
notes: [
|
|
`Loaded agent instructions from ${instructionsFilePath}`,
|
|
`Prepended instructions + path directive to prompt (relative references from ${instructionsDir}).`,
|
|
],
|
|
};
|
|
} catch (err) {
|
|
const reason = err instanceof Error ? err.message : String(err);
|
|
await onLog(
|
|
"stderr",
|
|
`[paperclip] Warning: could not read agent instructions file "${instructionsFilePath}": ${reason}\n`,
|
|
);
|
|
return {
|
|
prefix: "",
|
|
chars: 0,
|
|
notes: [
|
|
`Configured instructionsFilePath ${instructionsFilePath}, but file could not be read; continuing without injected instructions.`,
|
|
],
|
|
};
|
|
}
|
|
}
|
|
|
|
function renderPaperclipEnvNote(env: Record<string, string>): string {
|
|
const keys = Object.keys(env)
|
|
.filter((key) => key.startsWith("PAPERCLIP_"))
|
|
.sort();
|
|
if (keys.length === 0) return "";
|
|
return [
|
|
"Paperclip runtime note:",
|
|
`The following PAPERCLIP_* environment variables are available in the cloud agent shell: ${keys.join(", ")}`,
|
|
"Use them directly instead of assuming they are absent.",
|
|
].join("\n");
|
|
}
|
|
|
|
function readSession(params: Record<string, unknown> | null): CursorCloudSession | null {
|
|
if (!params) return null;
|
|
const record = asRecord(params);
|
|
if (!record) return null;
|
|
const cursorAgentId =
|
|
trimNullable(record.cursorAgentId) ??
|
|
trimNullable(record.agentId) ??
|
|
trimNullable(record.sessionId);
|
|
if (!cursorAgentId) return null;
|
|
const latestRunId = trimNullable(record.latestRunId) ?? trimNullable(record.runId) ?? undefined;
|
|
const envType = trimNullable(record.envType);
|
|
const envName = trimNullable(record.envName);
|
|
const reposValue = Array.isArray(record.repos) ? record.repos : [];
|
|
const repos = reposValue
|
|
.map((entry) => asRecord(entry))
|
|
.filter((entry): entry is Record<string, unknown> => Boolean(entry))
|
|
.map((entry) => ({
|
|
url: asString(entry.url, "").trim(),
|
|
startingRef: trimNullable(entry.startingRef) ?? undefined,
|
|
prUrl: trimNullable(entry.prUrl) ?? undefined,
|
|
}))
|
|
.filter((entry) => entry.url.length > 0);
|
|
return {
|
|
cursorAgentId,
|
|
...(latestRunId ? { latestRunId } : {}),
|
|
runtime: "cloud",
|
|
...(envType ? { envType: normalizeEnvType(envType) } : {}),
|
|
...(envName ? { envName } : {}),
|
|
repos,
|
|
};
|
|
}
|
|
|
|
function sessionMatches(
|
|
session: CursorCloudSession | null,
|
|
envType: "cloud" | "pool" | "machine",
|
|
envName: string | null,
|
|
repos: Array<{ url: string; startingRef?: string; prUrl?: string }>,
|
|
): boolean {
|
|
if (!session) return false;
|
|
if ((session.envType ?? "cloud") !== envType) return false;
|
|
if ((session.envName ?? null) !== envName) return false;
|
|
if (session.repos.length !== repos.length) return false;
|
|
return session.repos.every((repo, index) => {
|
|
const next = repos[index];
|
|
return repo.url === next.url
|
|
&& (repo.startingRef ?? null) === (next.startingRef ?? null)
|
|
&& (repo.prUrl ?? null) === (next.prUrl ?? null);
|
|
});
|
|
}
|
|
|
|
function buildAgentOptions(input: {
|
|
apiKey: string;
|
|
name: string;
|
|
model?: ModelSelection;
|
|
envType: "cloud" | "pool" | "machine";
|
|
envName: string | null;
|
|
repos: Array<{ url: string; startingRef?: string; prUrl?: string }>;
|
|
workOnCurrentBranch: boolean;
|
|
autoCreatePR: boolean;
|
|
skipReviewerRequest: boolean;
|
|
envVars: Record<string, string>;
|
|
}): AgentOptions {
|
|
return {
|
|
apiKey: input.apiKey,
|
|
name: input.name,
|
|
...(input.model ? { model: input.model } : {}),
|
|
cloud: {
|
|
env: {
|
|
type: input.envType,
|
|
...(input.envName ? { name: input.envName } : {}),
|
|
},
|
|
repos: input.repos,
|
|
workOnCurrentBranch: input.workOnCurrentBranch,
|
|
autoCreatePR: input.autoCreatePR,
|
|
skipReviewerRequest: input.skipReviewerRequest,
|
|
envVars: input.envVars,
|
|
},
|
|
};
|
|
}
|
|
|
|
function eventLine(event: CursorCloudEvent): string {
|
|
return `${JSON.stringify(event)}\n`;
|
|
}
|
|
|
|
async function emitMessage(onLog: AdapterExecutionContext["onLog"], message: SDKMessage) {
|
|
await onLog("stdout", eventLine({ type: "cursor_cloud.message", message }));
|
|
}
|
|
|
|
async function emitStatus(onLog: AdapterExecutionContext["onLog"], status: string, message?: string) {
|
|
await onLog("stdout", eventLine({ type: "cursor_cloud.status", status, ...(message ? { message } : {}) }));
|
|
}
|
|
|
|
async function streamRun(run: Run, onLog: AdapterExecutionContext["onLog"]) {
|
|
if (!run.supports("stream")) return;
|
|
for await (const message of run.stream()) {
|
|
await emitMessage(onLog, message);
|
|
}
|
|
}
|
|
|
|
async function getAttachedRun(input: {
|
|
apiKey: string;
|
|
session: CursorCloudSession | null;
|
|
}): Promise<Run | null> {
|
|
const latestRunId = input.session?.latestRunId;
|
|
const cursorAgentId = input.session?.cursorAgentId;
|
|
if (!latestRunId || !cursorAgentId) return null;
|
|
try {
|
|
const run = await Agent.getRun(latestRunId, {
|
|
runtime: "cloud",
|
|
agentId: cursorAgentId,
|
|
apiKey: input.apiKey,
|
|
});
|
|
return run.status === "running" ? run : null;
|
|
} catch {
|
|
return null;
|
|
}
|
|
}
|
|
|
|
export async function execute(ctx: AdapterExecutionContext): Promise<AdapterExecutionResult> {
|
|
const { runId, agent, runtime, config, context, onLog, onMeta } = ctx;
|
|
const envConfig = asStringEnvMap(config.env);
|
|
const apiKey = asString(envConfig.CURSOR_API_KEY, "").trim();
|
|
if (!apiKey) {
|
|
return {
|
|
exitCode: 1,
|
|
signal: null,
|
|
timedOut: false,
|
|
errorMessage: "CURSOR_API_KEY is required for cursor_cloud.",
|
|
provider: "cursor",
|
|
biller: "cursor",
|
|
billingType: "api",
|
|
clearSession: false,
|
|
};
|
|
}
|
|
|
|
const workspace = parseObject(context.paperclipWorkspace);
|
|
const repoUrl =
|
|
asString(config.repoUrl, "").trim() ||
|
|
asString(workspace.repoUrl, "").trim();
|
|
if (!repoUrl) {
|
|
return {
|
|
exitCode: 1,
|
|
signal: null,
|
|
timedOut: false,
|
|
errorMessage: "cursor_cloud requires repoUrl in adapterConfig or workspace context.",
|
|
provider: "cursor",
|
|
biller: "cursor",
|
|
billingType: "api",
|
|
clearSession: false,
|
|
};
|
|
}
|
|
|
|
const repoStartingRef =
|
|
trimNullable(config.repoStartingRef) ??
|
|
trimNullable(workspace.repoRef) ??
|
|
undefined;
|
|
const repoPullRequestUrl = trimNullable(config.repoPullRequestUrl) ?? undefined;
|
|
const envType = normalizeEnvType(asString(config.runtimeEnvType, "cloud"));
|
|
const envName = trimNullable(config.runtimeEnvName);
|
|
const workOnCurrentBranch = asBoolean(config.workOnCurrentBranch, false);
|
|
const autoCreatePR = asBoolean(config.autoCreatePR, false);
|
|
const skipReviewerRequest = asBoolean(config.skipReviewerRequest, false);
|
|
const model = toModelSelection(asString(config.model, ""));
|
|
const repos = [{
|
|
url: repoUrl,
|
|
...(repoStartingRef ? { startingRef: repoStartingRef } : {}),
|
|
...(repoPullRequestUrl ? { prUrl: repoPullRequestUrl } : {}),
|
|
}];
|
|
const remoteEnv = buildWakeEnv(ctx, envConfig);
|
|
const session = readSession(runtime.sessionParams) ?? (runtime.sessionId
|
|
? {
|
|
cursorAgentId: runtime.sessionId,
|
|
runtime: "cloud" as const,
|
|
repos,
|
|
}
|
|
: null);
|
|
const canReuseSession = sessionMatches(session, envType, envName, repos);
|
|
const promptTemplate = asString(config.promptTemplate, DEFAULT_PAPERCLIP_AGENT_PROMPT_TEMPLATE);
|
|
const bootstrapPromptTemplate = asString(config.bootstrapPromptTemplate, "");
|
|
const templateData = {
|
|
agentId: agent.id,
|
|
companyId: agent.companyId,
|
|
runId,
|
|
company: { id: agent.companyId },
|
|
agent,
|
|
run: { id: runId, source: "on_demand" },
|
|
context,
|
|
};
|
|
const instructions = await buildInstructionsPrefix(config, onLog);
|
|
const wakePrompt = renderPaperclipWakePrompt(context.paperclipWake, { resumedSession: canReuseSession });
|
|
const renderedBootstrapPrompt =
|
|
!canReuseSession && bootstrapPromptTemplate.trim().length > 0
|
|
? renderTemplate(bootstrapPromptTemplate, templateData).trim()
|
|
: "";
|
|
const renderedPrompt =
|
|
canReuseSession && wakePrompt.length > 0
|
|
? ""
|
|
: renderTemplate(promptTemplate, templateData).trim();
|
|
const paperclipEnvNote = renderPaperclipEnvNote(remoteEnv);
|
|
const prompt = joinPromptSections([
|
|
instructions.prefix,
|
|
renderedBootstrapPrompt,
|
|
wakePrompt,
|
|
paperclipEnvNote,
|
|
renderedPrompt,
|
|
]);
|
|
const sessionHandoffNote = asString(context.paperclipSessionHandoffMarkdown, "").trim();
|
|
const finalPrompt = joinPromptSections([prompt, sessionHandoffNote]);
|
|
|
|
const agentOptions = buildAgentOptions({
|
|
apiKey,
|
|
name: `Paperclip ${agent.name}`,
|
|
model,
|
|
envType,
|
|
envName,
|
|
repos,
|
|
workOnCurrentBranch,
|
|
autoCreatePR,
|
|
skipReviewerRequest,
|
|
envVars: remoteEnv,
|
|
});
|
|
|
|
const commandNotes = [
|
|
...instructions.notes,
|
|
canReuseSession
|
|
? `Reusing Cursor cloud agent session ${session?.cursorAgentId ?? "unknown"}`
|
|
: "Creating a new Cursor cloud agent session",
|
|
`Repository: ${repoUrl}${repoStartingRef ? ` @ ${repoStartingRef}` : ""}`,
|
|
`Runtime target: ${envType}${envName ? ` (${envName})` : ""}`,
|
|
];
|
|
|
|
if (onMeta) {
|
|
const meta: AdapterInvocationMeta = {
|
|
adapterType: "cursor_cloud",
|
|
command: "@cursor/sdk",
|
|
commandNotes,
|
|
prompt: finalPrompt,
|
|
promptMetrics: {
|
|
promptChars: finalPrompt.length,
|
|
instructionsChars: instructions.chars,
|
|
bootstrapPromptChars: renderedBootstrapPrompt.length,
|
|
wakePromptChars: wakePrompt.length,
|
|
heartbeatPromptChars: renderedPrompt.length,
|
|
},
|
|
context: {
|
|
cursorCloud: {
|
|
envType,
|
|
envName,
|
|
repoUrl,
|
|
repoStartingRef,
|
|
repoPullRequestUrl,
|
|
canReuseSession,
|
|
},
|
|
},
|
|
};
|
|
await onMeta(meta);
|
|
}
|
|
|
|
let sdkAgent: SDKAgent | null = null;
|
|
let run: Run | null = null;
|
|
let streamError: string | null = null;
|
|
try {
|
|
const attachedRun = canReuseSession
|
|
? await getAttachedRun({ apiKey, session })
|
|
: null;
|
|
|
|
if (attachedRun) {
|
|
await emitStatus(onLog, "running", `Reattached to existing Cursor run ${attachedRun.id}.`);
|
|
await onLog("stdout", eventLine({
|
|
type: "cursor_cloud.init",
|
|
sessionId: attachedRun.agentId,
|
|
agentId: attachedRun.agentId,
|
|
runId: attachedRun.id,
|
|
...(model?.id ? { model: model.id } : {}),
|
|
}));
|
|
const priorStreamPromise = streamRun(attachedRun, onLog).catch((err) => {
|
|
streamError = formatRunError(err);
|
|
});
|
|
if (attachedRun.supports("wait")) await attachedRun.wait();
|
|
await priorStreamPromise;
|
|
streamError = null;
|
|
await emitStatus(
|
|
onLog,
|
|
"running",
|
|
`Prior Cursor run ${attachedRun.id} finished; sending heartbeat follow-up so this wake's context is not dropped.`,
|
|
);
|
|
}
|
|
|
|
sdkAgent = canReuseSession && session
|
|
? await Agent.resume(session.cursorAgentId, agentOptions)
|
|
: await Agent.create(agentOptions);
|
|
run = await sdkAgent.send(finalPrompt, {
|
|
...(model ? { model } : {}),
|
|
});
|
|
await onLog("stdout", eventLine({
|
|
type: "cursor_cloud.init",
|
|
sessionId: sdkAgent.agentId,
|
|
agentId: sdkAgent.agentId,
|
|
runId: run.id,
|
|
...(model?.id ? { model: model.id } : {}),
|
|
}));
|
|
await emitStatus(onLog, "running", `Started Cursor run ${run.id}.`);
|
|
|
|
const streamPromise = streamRun(run, onLog).catch((err) => {
|
|
streamError = formatRunError(err);
|
|
});
|
|
const result = run.supports("wait")
|
|
? await run.wait()
|
|
: {
|
|
id: run.id,
|
|
status: run.status === "running" ? "error" : run.status,
|
|
result: run.result,
|
|
model: run.model,
|
|
durationMs: run.durationMs,
|
|
git: run.git,
|
|
};
|
|
await streamPromise;
|
|
|
|
const modelId = result.model?.id ?? model?.id ?? null;
|
|
await onLog("stdout", eventLine({
|
|
type: "cursor_cloud.result",
|
|
status: result.status,
|
|
...(result.result ? { result: result.result } : {}),
|
|
...(modelId ? { model: modelId } : {}),
|
|
...(typeof result.durationMs === "number" ? { durationMs: result.durationMs } : {}),
|
|
...(result.git ? { git: result.git } : {}),
|
|
...(streamError ? { error: streamError } : {}),
|
|
}));
|
|
|
|
const nextSession: CursorCloudSession = {
|
|
cursorAgentId: run.agentId,
|
|
latestRunId: result.id,
|
|
runtime: "cloud",
|
|
envType,
|
|
...(envName ? { envName } : {}),
|
|
repos,
|
|
};
|
|
const isError = result.status !== "finished";
|
|
return {
|
|
exitCode: isError ? 1 : 0,
|
|
signal: null,
|
|
timedOut: false,
|
|
errorMessage: isError ? (trimNullable(result.result) ?? streamError ?? `Cursor run ${result.status}`) : null,
|
|
sessionId: run.agentId,
|
|
sessionDisplayId: run.agentId,
|
|
sessionParams: nextSession,
|
|
provider: "cursor",
|
|
biller: "cursor",
|
|
billingType: "api",
|
|
model: modelId,
|
|
costUsd: null,
|
|
summary: toSummary(result),
|
|
resultJson: {
|
|
status: result.status,
|
|
cursorAgentId: run.agentId,
|
|
cursorRunId: result.id,
|
|
envType,
|
|
envName,
|
|
repos,
|
|
...(result.result ? { result: result.result } : {}),
|
|
...(result.git ? { git: result.git } : {}),
|
|
...(typeof result.durationMs === "number" ? { durationMs: result.durationMs } : {}),
|
|
...(streamError ? { streamError } : {}),
|
|
},
|
|
clearSession: false,
|
|
};
|
|
} catch (err) {
|
|
const reason = formatRunError(err);
|
|
if (run) {
|
|
await onLog("stdout", eventLine({
|
|
type: "cursor_cloud.result",
|
|
status: "error",
|
|
error: reason,
|
|
}));
|
|
}
|
|
return {
|
|
exitCode: 1,
|
|
signal: null,
|
|
timedOut: false,
|
|
errorMessage: reason,
|
|
sessionId: session?.cursorAgentId ?? null,
|
|
sessionDisplayId: session?.cursorAgentId ?? null,
|
|
sessionParams: session,
|
|
provider: "cursor",
|
|
biller: "cursor",
|
|
billingType: "api",
|
|
costUsd: null,
|
|
clearSession: false,
|
|
resultJson: {
|
|
status: "error",
|
|
...(run ? { cursorRunId: run.id } : {}),
|
|
...(session?.cursorAgentId ? { cursorAgentId: session.cursorAgentId } : {}),
|
|
error: reason,
|
|
},
|
|
};
|
|
} finally {
|
|
if (sdkAgent) {
|
|
try {
|
|
await sdkAgent[Symbol.asyncDispose]();
|
|
} catch {
|
|
// Best effort only.
|
|
}
|
|
}
|
|
}
|
|
}
|