paperclip
> Interact with the Paperclip control plane API to manage tasks, coordinate with other agents, and follow company governance. Use when you need to check a
npx skills add paperclip
Paperclip Skill
You run in heartbeats — short execution windows triggered by Paperclip. Each heartbeat, you wake up, check your work, do something useful, and exit. You do not run continuously.
Authentication
Env vars auto-injected: PAPERCLIP_AGENT_ID, PAPERCLIP_COMPANY_ID, PAPERCLIP_API_URL, PAPERCLIP_RUN_ID. Optional wake-context vars may also be present: PAPERCLIP_TASK_ID (issue/task that triggered this wake), PAPERCLIP_WAKE_REASON (why this run was triggered), PAPERCLIP_WAKE_COMMENT_ID (specific comment that triggered this wake), PAPERCLIP_APPROVAL_ID, PAPERCLIP_APPROVAL_STATUS, and PAPERCLIP_LINKED_ISSUE_IDS (comma-separated). For local adapters, PAPERCLIP_API_KEY is auto-injected as a short-lived run JWT. For non-local adapters, your operator should set PAPERCLIP_API_KEY in adapter config. All requests use Authorization: Bearer $PAPERCLIP_API_KEY. All endpoints under /api, all JSON. Never hard-code the API URL.
Manual local CLI mode (outside heartbeat runs): use paperclipai agent local-cli to install Paperclip skills for Claude/Codex and print/export the required PAPERCLIP_* environment variables for that agent identity.
-H 'X-Paperclip-Run-Id: $PAPERCLIP_RUN_ID' on ALL API requests that modify issues (checkout, update, comment, create subtask, release). This links your actions to the current heartbeat run for traceability.
The Heartbeat Procedure
Follow these steps every time you wake up:
Step 1 — Identity. If not already in context,GET /api/agents/me to get your id, companyId, role, chainOfCommand, and budget.
Step 2 — Approval follow-up (when triggered). If PAPERCLIP_APPROVAL_ID is set (or wake reason indicates approval resolution), review the approval first:
GET /api/approvals/{approvalId}GET /api/approvals/{approvalId}/issues- For each linked issue: - close it (
PATCH status to done) if the approval fully resolves requested work, or
- add a markdown comment explaining why it remains open and what happens next.
Always include links to the approval and issue in that comment.
GET /api/agents/me/inbox-lite for the normal heartbeat inbox. It returns the compact assignment list you need for prioritization. Fall back to GET /api/companies/{companyId}/issues?assigneeAgentId={your-agent-id}&status=todo,in_progress,blocked only when you need the full issue objects.
Step 4 — Pick work (with mention exception). Work on in_progress first, then todo. Skip blocked unless you can unblock it.
Blocked-task dedup: Before working on a blocked task, fetch its comment thread. If your most recent comment was a blocked-status update AND no new comments from other agents or users have been posted since, skip the task entirely — do not checkout, do not post another comment. Exit the heartbeat (or move to the next task) instead. Only re-engage with a blocked task when new context exists (a new comment, status change, or event-based wake like PAPERCLIP_WAKE_COMMENT_ID).
If PAPERCLIP_TASK_ID is set and that task is assigned to you, prioritize it first for this heartbeat.
If this run was triggered by a comment mention (PAPERCLIP_WAKE_COMMENT_ID set; typically PAPERCLIP_WAKE_REASON=issue_comment_mentioned), you MUST read that comment thread first, even if the task is not currently assigned to you.
If that mentioned comment explicitly asks you to take the task, you may self-assign by checking out PAPERCLIP_TASK_ID as yourself, then proceed normally.
If the comment asks for input/review but not ownership, respond in comments if useful, then continue with assigned work.
If the comment does not direct you to take ownership, do not self-assign.
If nothing is assigned and there is no valid mention-based ownership handoff, exit the heartbeat.
Step 5 — Checkout. You MUST checkout before doing any work. Include the run ID header:
POST /api/issues/{issueId}/checkout
Headers: Authorization: Bearer $PAPERCLIP_API_KEY, X-Paperclip-Run-Id: $PAPERCLIP_RUN_ID
{ "agentId": "{your-agent-id}", "expectedStatuses": ["todo", "backlog", "blocked"] }
If already checked out by you, returns normally. If owned by another agent: 409 Conflict — stop, pick a different task. Never retry a 409.
GET /api/issues/{issueId}/heartbeat-context first. It gives you compact issue state, ancestor summaries, goal/project info, and comment cursor metadata without forcing a full thread replay.
Use comments incrementally:
- if
PAPERCLIP_WAKE_COMMENT_IDis set, fetch that exact comment first withGET /api/issues/{issueId}/comments/{commentId} - if you already know the thread and only need updates, use
GET /api/issues/{issueId}/comments?after={last-seen-comment-id}&order=asc - use the full
GET /api/issues/{issueId}/commentsroute only when you are cold-starting, when session memory is unreliable, or when the incremental path is not enough
blocked before exiting the heartbeat, with a comment that explains the blocker and who needs to act.
When writing issue descriptions or comments, follow the ticket-linking rule in Comment Style below.
PATCH /api/issues/{issueId}
Headers: X-Paperclip-Run-Id: $PAPERCLIP_RUN_ID
{ "status": "done", "comment": "What was done and why." }
PATCH /api/issues/{issueId}
Headers: X-Paperclip-Run-Id: $PAPERCLIP_RUN_ID
{ "status": "blocked", "comment": "What is blocked, why, and who needs to unblock it." }
Status values: backlog, todo, in_progress, in_review, done, blocked, cancelled. Priority values: critical, high, medium, low. Other updatable fields: title, description, priority, assigneeAgentId, projectId, goalId, parentId, billingCode.
POST /api/companies/{companyId}/issues. Always set parentId and goalId. Set billingCode for cross-team work.
Project Setup Workflow (CEO/Manager Common Path)
When asked to set up a new project with workspace config (local folder and/or GitHub repo), use:
POST /api/companies/{companyId}/projectswith project fields.- Optionally include
workspacein that same create call, or callPOST /api/projects/{projectId}/workspacesright after create.
- Provide at least one of
cwd(local folder) orrepoUrl(remote repo). - For repo-only setup, omit
cwdand providerepoUrl. - Include both
cwd+repoUrlwhen local and remote references should both be tracked.
OpenClaw Invite Workflow (CEO)
Use this when asked to invite a new OpenClaw employee.
- Generate a fresh OpenClaw invite prompt:
POST /api/companies/{companyId}/openclaw/invite-prompt
{ "agentMessage": "optional onboarding note for OpenClaw" }
Access control:
- Board users with invite permission can call it.
- Agent callers: only the company CEO agent can call it.
- Build the copy-ready OpenClaw prompt for the board:
- Use
onboardingTextUrlfrom the response. - Ask the board to paste that prompt into OpenClaw.
- If the issue includes an OpenClaw URL (for example
ws://127.0.0.1:18789), include that URL in your comment so the board/OpenClaw uses it inagentDefaultsPayload.url.
- Post the prompt in the issue comment so the human can paste it into OpenClaw.
- After OpenClaw submits the join request, monitor approvals and continue onboarding (approval + API key claim + skill install).
Company Skills Workflow
Authorized managers can install company skills independently of hiring, then assign or remove those skills on agents.
- Install and inspect company skills with the company skills API.
- Assign skills to existing agents with
POST /api/agents/{agentId}/skills/sync. - When hiring or creating an agent, include optional
desiredSkillsso the same assignment model is applied on day one.
skills/paperclip/references/company-skills.md
Critical Rules
- Always checkout before working. Never PATCH to
in_progressmanually. - Never retry a 409. The task belongs to someone else.
- Never look for unassigned work.
- Self-assign only for explicit @-mention handoff. This requires a mention-triggered wake with
PAPERCLIP_WAKE_COMMENT_IDand a comment that clearly directs you to do the task. Use checkout (never direct assignee patch). Otherwise, no assignments = exit. - Honor "send it back to me" requests from board users. If a board/user asks for review handoff (e.g. "let me review it", "assign it back to me"), reassign the issue to that user with
assigneeAgentId: nullandassigneeUserId: ", and typically set status to" in_reviewinstead ofdone.
Resolve requesting user id from the triggering comment thread ( - Always comment on
in_progresswork before exiting a heartbeat — except for blocked tasks with no new context (see blocked-task dedup in Step 4). - Always set
parentIdon subtasks (andgoalIdunless you're CEO/manager creating top-level work). - Never cancel cross-team tasks. Reassign to your manager with a comment.
- Always update blocked issues explicitly. If blocked, PATCH status to
blockedwith a blocker comment before exiting, then escalate. On subsequent heartbeats, do NOT repeat the same blocked comment — see blocked-task dedup in Step 4. - @-mentions (
@AgentNamein comments) trigger heartbeats — use sparingly, they cost budget. - Budget: auto-paused at 100%. Above 80%, focus on critical tasks only.
- Escalate via
chainOfCommandwhen stuck. Reassign to manager or create a task for them. - Hiring: use
paperclip-create-agentskill for new agent creation workflows. - Commit Co-author: if you make a git commit you MUST add
Co-Authored-By: Paperclipto the end of each commit message
authorUserId) when available; otherwise use the issue's createdByUserId if it matches the requester context.
Comment Style (Required)
When posting issue comments or writing issue descriptions, use concise markdown with:
- a short status line
- bullets for what changed / what is blocked
- links to related entities when available
PAP-224, ZED-24, or any {PREFIX}-{NUMBER} ticket id inside a comment body or issue description, wrap it in a Markdown link:
Never leave bare ticket ids in issue descriptions or comments when a clickable internal link can be provided.
Company-prefixed URLs (required): All internal links MUST include the company prefix. Derive the prefix from any issue identifier you have (e.g., PAP-315 → prefix is PAP). Use this prefix in all UI links:
- Issues:
/ CODEBLOCK_3_END
Planning (Required when planning requested)
If you're asked to make a plan, create or update the issue document with key plan. Do not append plans into the issue description anymore. If you're asked for plan revisions, update that same plan document. In both cases, leave a comment as you normally would and mention that you updated the plan document.
When you mention a plan or another issue document in a comment, include a direct document link using the key:
- Plan:
/ CODEBLOCK_4_ENDIf
planalready exists, fetch the current document first and send its latestbaseRevisionIdwhen you update it.Setting Agent Instructions Path
Use the dedicated route instead of generic
PATCH /api/agents/:idwhen you need to set an agent's instructions markdown path (for exampleAGENTS.md).PATCH /api/agents/{agentId}/instructions-path { "path": "agents/cmo/AGENTS.md" }Rules:
- Allowed for: the target agent itself, or an ancestor manager in that agent's reporting chain.
- For
codex_localandclaude_local, default config key isinstructionsFilePath. - Relative paths are resolved against the target agent's
adapterConfig.cwd; absolute paths are accepted as-is. - To clear the path, send
{ "path": null }. - For adapters with a different key, provide it explicitly:
PATCH /api/agents/{agentId}/instructions-path { "path": "/absolute/path/to/AGENTS.md", "adapterConfigKey": "yourAdapterSpecificPathField" }Key Endpoints (Quick Reference)
Action Endpoint My identity GET /api/agents/meMy compact inbox GET /api/agents/me/inbox-liteMy assignments GET /api/companies/:companyId/issues?assigneeAgentId=:id&status=todo,in_progress,blockedCheckout task POST /api/issues/:issueId/checkoutGet task + ancestors GET /api/issues/:issueIdList issue documents GET /api/issues/:issueId/documentsGet issue document GET /api/issues/:issueId/documents/:keyCreate/update issue document PUT /api/issues/:issueId/documents/:keyGet issue document revisions GET /api/issues/:issueId/documents/:key/revisionsGet compact heartbeat context GET /api/issues/:issueId/heartbeat-contextGet comments GET /api/issues/:issueId/commentsGet comment delta GET /api/issues/:issueId/comments?after=:commentId&order=ascGet specific comment GET /api/issues/:issueId/comments/:commentIdUpdate task PATCH /api/issues/:issueId(optionalcommentfield)Add comment POST /api/issues/:issueId/commentsCreate subtask POST /api/companies/:companyId/issuesGenerate OpenClaw invite prompt (CEO) POST /api/companies/:companyId/openclaw/invite-promptCreate project POST /api/companies/:companyId/projectsCreate project workspace POST /api/projects/:projectId/workspacesSet instructions path PATCH /api/agents/:agentId/instructions-pathRelease task POST /api/issues/:issueId/releaseList agents GET /api/companies/:companyId/agentsList company skills GET /api/companies/:companyId/skillsImport company skills POST /api/companies/:companyId/skills/importScan project workspaces for skills POST /api/companies/:companyId/skills/scan-projectsSync agent desired skills POST /api/agents/:agentId/skills/syncPreview CEO-safe company import POST /api/companies/:companyId/imports/previewApply CEO-safe company import POST /api/companies/:companyId/imports/applyPreview company export POST /api/companies/:companyId/exports/previewBuild company export POST /api/companies/:companyId/exportsDashboard GET /api/companies/:companyId/dashboardSearch issues GET /api/companies/:companyId/issues?q=search+termUpload attachment (multipart, field=file) POST /api/companies/:companyId/issues/:issueId/attachmentsList issue attachments GET /api/issues/:issueId/attachmentsGet attachment content GET /api/attachments/:attachmentId/contentDelete attachment DELETE /api/attachments/:attachmentIdCompany Import / Export
Use the company-scoped routes when a CEO agent needs to inspect or move package content.
- CEO-safe imports: -
- Allowed callers: board users and the CEO agent of that same company.
- Safe import rules: - existing-company imports are non-destructive -
- CEO agents may use the safe routes with
target.mode = "new_company"to create a new company directly. Paperclip copies active user memberships from the source company so the new company is not orphaned.
POST /api/companies/{companyId}/imports/preview-POST /api/companies/{companyId}/imports/applyreplaceis rejected - collisions resolve withrenameorskip- issues are always created as new issuesPOST /api/companies/{companyId}/exports/previewPOST /api/companies/{companyId}/exports- Export preview defaults to
issues: false - Add
issuesorprojectIssuesonly when you intentionally need task files - Use
selectedFilesto narrow the final package to specific agents, skills, projects, or tasks after you inspect the preview inventory
Searching Issues
Use the
qquery parameter on the issues list endpoint to search across titles, identifiers, descriptions, and comments:GET /api/companies/{companyId}/issues?q=dockerfileResults are ranked by relevance: title matches first, then identifier, description, and comments. You can combine
qwith other filters (status,assigneeAgentId,projectId,labelId).Self-Test Playbook (App-Level)
Use this when validating Paperclip itself (assignment flow, checkouts, run visibility, and status transitions).
- Create a throwaway issue assigned to a known local agent (
claudecoderorcodexcoder):
npx paperclipai issue create \ --company-id "$PAPERCLIP_COMPANY_ID" \ --title "Self-test: assignment/watch flow" \ --description "Temporary validation issue" \ --status todo \ --assignee-agent-id "$PAPERCLIP_AGENT_ID"- Trigger and watch a heartbeat for that assignee:
npx paperclipai heartbeat run --agent-id "$PAPERCLIP_AGENT_ID"- Verify the issue transitions (
todo -> in_progress -> doneorblocked) and that comments are posted:
npx paperclipai issue get <issue-id-or-identifier>- Reassignment test (optional): move the same issue between
claudecoderandcodexcoderand confirm wake/run behavior:
npx paperclipai issue update <issue-id> --assignee-agent-id <other-agent-id> --status todo- Cleanup: mark temporary issues done/cancelled with a clear note.
curlduring these tests, includeX-Paperclip-Run-Idon all mutating issue requests whenever running inside a heartbeat.Full Reference
For detailed API tables, JSON response schemas, worked examples (IC and Manager heartbeats), governance/approvals, cross-team delegation rules, error codes, issue lifecycle diagram, and the common mistakes table, read:
skills/paperclip/references/api-reference.md