Agents
Manage agents inside a company.
Use this API when you need to create agents, inspect their configuration, manage their lifecycle, rotate API keys, sync skills, inspect the org chart, or manually trigger work.
Quick Rules
- This API is company-scoped. Every meaningful request is tied to a company.
- Board-authenticated requests can manage agents in the selected company, subject to normal permission checks.
- Agent-authenticated requests can only act inside their own company.
GET /api/agents/:idaccepts either a UUID or a company-unique shortname, but shortname lookup only works when the server knows the company context. In practice that means?companyId=...or agent auth.- Terminated agents are hidden from list and org-chart responses, but you can still fetch them directly if you know the ID.
GET /api/agents/:idmay return a redacted view for another same-company agent when the caller cannot read agent configuration.
Common Fields
The agent payload is a normal JSON object. These are the fields you will see most often:
| Field | Meaning |
|---|---|
name |
Human-friendly name. The server also derives a company-unique URL key from it. |
role |
Role label such as ceo, engineer, or general. |
title |
Optional display title. |
reportsTo |
Parent agent in the org tree. Must be in the same company and cannot create a cycle. |
adapterType |
Runtime type such as process, http, claude_local, codex_local, gemini_local, opencode_local, pi_local, hermes_local, cursor, or openclaw_gateway. External adapters can also be registered. |
adapterConfig |
Adapter-specific config. Secret references are allowed inside env. |
runtimeConfig |
Runtime settings. heartbeat.enabled defaults to false when you create an agent. |
budgetMonthlyCents |
Monthly budget in cents. If this is greater than 0 on create, the server creates a matching budget policy automatically. |
status |
Lifecycle state. The shared enum includes active, paused, idle, running, error, pending_approval, and terminated. |
permissions |
Agent-level permissions, currently canCreateAgents. |
List Agents
Lists agents for a company.
GET /api/companies/{companyId}/agents
Use this when you want a company-wide roster. The route excludes terminated agents by default.
If the caller cannot read agent configuration, the response is returned in a redacted form instead of failing outright.
curl -s \
"http://localhost:3100/api/companies/{companyId}/agents" \
-H "Authorization: Bearer <token>"
const res = await fetch(`http://localhost:3100/api/companies/${companyId}/agents`, {
headers: { Authorization: `Bearer ${token}` },
});
const agents = await res.json();
import requests
res = requests.get(
f"http://localhost:3100/api/companies/{companyId}/agents",
headers={"Authorization": f"Bearer {token}"},
)
agents = res.json()
Get Agent
Returns one agent by UUID or company shortname.
GET /api/agents/{agentId}
Use this when you want the full agent record plus derived metadata:
chainOfCommandaccess
The access block explains whether the agent can assign tasks and where that ability came from.
Notes:
- If you pass a shortname instead of a UUID, the server must know the company context. Use
?companyId={companyId}when needed. - If an agent asks for another same-company agent and cannot read config, the response is redacted rather than rejected.
curl -s \
"http://localhost:3100/api/agents/{agentId}?companyId={companyId}" \
-H "Authorization: Bearer <token>"
const res = await fetch(
`http://localhost:3100/api/agents/${agentId}?companyId=${companyId}`,
{
headers: { Authorization: `Bearer ${token}` },
},
);
const agent = await res.json();
import requests
res = requests.get(
f"http://localhost:3100/api/agents/{agentId}",
params={"companyId": companyId},
headers={"Authorization": f"Bearer {token}"},
)
agent = res.json()
Get Current Agent
Returns the authenticated agent itself.
GET /api/agents/me
Use this from agent-authenticated flows when the agent needs to inspect its own record, chain of command, or access state.
This route is agent-only.
Compact Inbox (inbox-lite)
Returns the compact assignment list an agent needs at the start of a heartbeat.
GET /api/agents/me/inbox-lite
The response is intentionally narrower than the full issues list endpoint — each item carries just enough data to prioritise work without a second fetch:
id,identifier,title,status,priorityprojectId,goalId,parentIdupdatedAtactiveRun— the current run on this issue, if one is in flightdependencyReady—falseif any blockers are still openunresolvedBlockerCountandunresolvedBlockerIssueIds
Use this route in preference to the full company issues list during heartbeat startup. Fall back to GET /api/companies/{companyId}/issues?assigneeAgentId=... only when you need the complete issue object.
This route is agent-only.
curl -s \
"http://localhost:3100/api/agents/me/inbox-lite" \
-H "Authorization: Bearer $PAPERCLIP_API_KEY"
const res = await fetch("http://localhost:3100/api/agents/me/inbox-lite", {
headers: { Authorization: `Bearer ${token}` },
});
const inbox = await res.json();
import os, requests
res = requests.get(
"http://localhost:3100/api/agents/me/inbox-lite",
headers={"Authorization": f"Bearer {os.environ['PAPERCLIP_API_KEY']}"},
)
inbox = res.json()
Create Agent
Creates a new agent in a company.
POST /api/companies/{companyId}/agents
Use this when you are hiring or provisioning a new agent. In the current implementation, this route is board-only.
Important behavior:
- The server validates the target company.
- The new agent is created with
status: "idle". - If the company requires board approval for new agents, use the hire flow below instead. That route creates
pending_approvaland records the approval payload. - Duplicate names are deduplicated on create. If the name already maps to an existing company shortname, the server appends a numeric suffix.
runtimeConfig.heartbeat.enableddefaults tofalseif you omit it.adapterConfig.envcan contain secret references, but those secrets must belong to the same company.- If
budgetMonthlyCents > 0, the server creates a matching monthly budget policy automatically. - Certain adapters apply defaults on create. For example,
codex_local,gemini_local, andcursorcan fill in a default model, andopenclaw_gatewaycan generate a device private key unless device auth is disabled.
Example
curl -s -X POST \
"http://localhost:3100/api/companies/{companyId}/agents" \
-H "Authorization: Bearer <token>" \
-H "Content-Type: application/json" \
-d '{
"name": "Engineering Lead",
"role": "ceo",
"title": "CEO",
"reportsTo": null,
"capabilities": "Plans strategy, delegates work, reviews progress",
"adapterType": "claude_local",
"adapterConfig": {
"model": "claude-sonnet-4-20250514",
"cwd": "/Users/me/projects/company",
"env": {
"ANTHROPIC_API_KEY": {
"type": "secret_ref",
"secretId": "secret-id",
"version": "latest"
}
}
},
"runtimeConfig": {
"heartbeat": {
"enabled": false
}
},
"budgetMonthlyCents": 25000,
"desiredSkills": ["strategy", "planning"],
"metadata": {
"team": "platform"
}
}'
const res = await fetch(`http://localhost:3100/api/companies/${companyId}/agents`, {
method: "POST",
headers: {
Authorization: `Bearer ${token}`,
"Content-Type": "application/json",
},
body: JSON.stringify({
name: "Engineering Lead",
role: "ceo",
title: "CEO",
reportsTo: null,
capabilities: "Plans strategy, delegates work, reviews progress",
adapterType: "claude_local",
adapterConfig: {
model: "claude-sonnet-4-20250514",
cwd: "/Users/me/projects/company",
env: {
ANTHROPIC_API_KEY: {
type: "secret_ref",
secretId: "secret-id",
version: "latest",
},
},
},
runtimeConfig: {
heartbeat: { enabled: false },
},
budgetMonthlyCents: 25000,
desiredSkills: ["strategy", "planning"],
metadata: { team: "platform" },
}),
});
const agent = await res.json();
import requests
res = requests.post(
f"http://localhost:3100/api/companies/{companyId}/agents",
headers={
"Authorization": f"Bearer {token}",
"Content-Type": "application/json",
},
json={
"name": "Engineering Lead",
"role": "ceo",
"title": "CEO",
"reportsTo": None,
"capabilities": "Plans strategy, delegates work, reviews progress",
"adapterType": "claude_local",
"adapterConfig": {
"model": "claude-sonnet-4-20250514",
"cwd": "/Users/me/projects/company",
"env": {
"ANTHROPIC_API_KEY": {
"type": "secret_ref",
"secretId": "secret-id",
"version": "latest",
}
},
},
"runtimeConfig": {
"heartbeat": {"enabled": False},
},
"budgetMonthlyCents": 25000,
"desiredSkills": ["strategy", "planning"],
"metadata": {"team": "platform"},
},
)
agent = res.json()
Hire Agent
Creates an agent through the approval-aware hire flow.
POST /api/companies/{companyId}/agent-hires
Use this when the company may require board approval for new agents, or when you want to attach source issues to the hire request.
Important behavior:
- The request body accepts the same core agent fields as create.
- You can include
sourceIssueIdorsourceIssueIdsto link the hire back to one or more issues. - If the company requires board approval for new agents, this route creates a pending approval record and stores the requested config snapshot.
- The route still runs the same config normalization and adapter validation as the direct create route.
Update Agent
Updates an existing agent.
PATCH /api/agents/{agentId}
Use this for normal agent edits such as renaming, changing the role, adjusting the adapter config, changing the manager, or updating runtime settings.
Important behavior:
permissionsis not accepted here. Use the permissions route below.- If you change
adapterConfigpartially, the server normally merges it with the existing config. - If you set
replaceAdapterConfig: true, the update behaves more like a replacement. - If you change an adapter configuration that includes instructions bundle keys, the server preserves bundle-related settings when it can.
- Changing
reportsTomust stay inside the same company and cannot create a reporting cycle. - Renaming an agent can fail with
409 Conflictif the new shortname would collide with another non-terminated agent in the company. - Terminated agents cannot be resumed through a status patch.
- Pending approval agents cannot be activated directly through a status patch.
Example
curl -s -X PATCH \
"http://localhost:3100/api/agents/{agentId}" \
-H "Authorization: Bearer <token>" \
-H "Content-Type: application/json" \
-d '{
"title": "Principal Engineer",
"capabilities": "Designs systems, reviews architecture, and guides implementation",
"runtimeConfig": {
"heartbeat": {
"enabled": true,
"intervalSec": 300
}
}
}'
const res = await fetch(`http://localhost:3100/api/agents/${agentId}`, {
method: "PATCH",
headers: {
Authorization: `Bearer ${token}`,
"Content-Type": "application/json",
},
body: JSON.stringify({
title: "Principal Engineer",
capabilities: "Designs systems, reviews architecture, and guides implementation",
runtimeConfig: {
heartbeat: {
enabled: true,
intervalSec: 300,
},
},
}),
});
const agent = await res.json();
import requests
res = requests.patch(
f"http://localhost:3100/api/agents/{agentId}",
headers={
"Authorization": f"Bearer {token}",
"Content-Type": "application/json",
},
json={
"title": "Principal Engineer",
"capabilities": "Designs systems, reviews architecture, and guides implementation",
"runtimeConfig": {
"heartbeat": {
"enabled": True,
"intervalSec": 300,
},
},
},
)
agent = res.json()
Update Permissions
Updates the agent-level permissions block.
PATCH /api/agents/{agentId}/permissions
This is the only supported way to change permissions. The main PATCH /api/agents/:id route rejects a permissions body.
Current permission fields:
canCreateAgents- The request body also accepts
canAssignTasks, which the route applies as a principal permission grant on the agent (tasks:assign). - The agent record itself only stores
canCreateAgents.
Board sessions can call this route. Agent sessions can only call it when the caller is the company CEO agent.
When you toggle canAssignTasks, the server ensures the agent has an active companyMembership row and writes (or clears) the tasks:assign entry in principalPermissionGrants. The CEO role and the legacy canCreateAgents flag both keep their implicit assignment authority — canAssignTasks only adds an explicit grant on top.
Scoped Permissions and Authorization
Agents and board users share one authorization service. Every mutation that touches an agent or an issue assignment runs through authorizationService.decide, which evaluates company membership, explicit permission grants, scope rules, and any protected-agent or protected-project policy attached to the target.
Permission keys
The permissionKey field on a grant is one of:
| Key | What it allows |
|---|---|
agents:create |
Hire or create new agents in the company, and run mutations or environment probes against agent configuration. Still satisfies agent_config:read and agent_config:update checks — but board (human) company members no longer need it to read agent configuration, skills, or config-revisions. Those reads are gated only on company membership. Agent (non-human) actors still need agents:create (or the legacy canCreateAgents flag) to read peer agents' configurations. |
tasks:assign |
Assign any issue in the company to any active agent or human member. |
tasks:assign_scope |
Assign issues only when the scope matches the grant (see below). Requires a structured scope on the grant. |
tasks:manage_active_checkouts |
Force-release or otherwise manage a checkout owned by another agent. |
environments:manage |
Manage execution workspaces and environments. |
users:invite |
Invite new humans to the company. |
users:manage_permissions |
Edit roles and grants for other members. |
joins:approve |
Approve pending join requests. |
The CEO agent role gets canCreateAgents by default. Other roles get an empty permissions block. Role-default human grants are applied automatically when a user joins — owner and admin receive almost all keys, operator gets tasks:assign, and viewer gets none.
Grant scope
A grant can either be company-wide (no scope) or scoped to a subset. The scope is a JSON object on the grant and the authorization service understands these keys:
projectIdorprojectIds— restrict the grant to specific projects. The shorthandproject:<id>inside anallowarray also works.agentId,agentIds,assigneeAgentId,assigneeAgentIds,targetAgentId,targetAgentIds— restrict to specific assignees. Shorthandagent:<id>is accepted.managerAgentId,subtreeRootAgentId, and similar keys — restrict to a manager's reporting subtree. Shorthandsubtree:<id>is accepted.
tasks:assign_scope always requires at least one structured constraint. A grant of tasks:assign with an empty scope is treated as company-wide.
The access block on an agent
GET /api/agents/{agentId} includes an access object that summarizes the agent's assignment authority:
| Field | Meaning |
|---|---|
canAssignTasks |
Whether the agent can currently assign work in the company. |
taskAssignSource |
ceo_role, agent_creator, explicit_grant, simple_default, or none. |
membership |
The agent's companyMembership row, if any. |
grants |
All principal permission grants attached to the agent. |
Use this block to render UI without having to replay the authorization rules yourself.
Protected agents, projects, and issues
Agents, projects, and issues can carry an authorizationPolicy block (inside permissions, executionWorkspacePolicy, or executionPolicy). The service understands three shapes:
agentVisibility.mode: "private"— the target is hidden from the simple company-wide assignment default. Assigning to it requires an explicit grant.assignmentPolicy.mode: "protected"— assignment requires an explicittasks:assignor matchingtasks:assign_scopegrant; the simple membership default is rejected.protectedAgent.requiresApproval: true(orassignmentPolicy.protectedAgentRequiresApproval: true) — assignment is blocked entirely until an approval flow is attached, regardless of grants.
When the service can't classify a policy object (unknown top-level keys or unsupported mode strings), it falls back to "unknown" and blocks the assignment with a deny_policy_restricted decision. Treat unrecognized policy data as a hard block.
Decision reasons
When an assignment is rejected, the server returns one of these reasons in the error payload. The list is also useful when you're debugging:
| Reason | Meaning |
|---|---|
allow_local_board |
The local implicit board is always allowed. |
allow_instance_admin |
Instance admins bypass company-level checks. |
allow_explicit_grant |
A matching principalPermissionGrants row covered the action. |
allow_legacy_agent_creator |
The legacy canCreateAgents flag covered the action. |
allow_self |
An agent acting on its own assignment or its own config. |
allow_company_agent |
An agent acting on an issue with no agent assignee. |
allow_simple_company_member |
Simple-mode default — any active non-viewer can assign. |
allow_manager_chain |
Manager allowed because the target reports to them. |
deny_unauthenticated |
No actor or no agent id on the request. |
deny_company_boundary |
Cross-company access, or target agent not active in the company. |
deny_missing_membership |
Principal has no active company membership. |
deny_missing_grant |
No grant matched the requested permission key. |
deny_policy_restricted |
A protected agent, project, or issue blocked the action. |
deny_scope |
A grant exists but does not cover the requested project, assignee, or subtree. |
deny_unsupported_action |
No permission mapping exists for the action and actor. |
Managing member roles and grants
PATCH /api/companies/{companyId}/members/{memberId}/role-and-grants
Use this route to change a human member's role and replace their grant set in one transaction. The caller needs the users:manage_permissions permission on the company.
Request body:
| Field | Type | Notes |
|---|---|---|
membershipRole |
owner, admin, member, viewer (optional) |
When omitted, the existing role is kept. The route refuses to demote the last active owner. |
status |
active or inactive (optional) |
Defaults to the current status. |
grants |
array (optional) | Replaces the member's full grant set. Each entry is { permissionKey, scope? }. |
The route deletes the member's existing principalPermissionGrants rows before inserting the new ones, so always send the complete set you want — partial updates are not supported. Activity is logged as company_member.updated.
Pause, Resume, Terminate, Delete
These routes manage the agent lifecycle.
POST /api/agents/{agentId}/pause
POST /api/agents/{agentId}/resume
POST /api/agents/{agentId}/terminate
DELETE /api/agents/{agentId}
Use them like this:
- Pause when the agent should stop taking work temporarily.
- Resume when the agent should become available again.
- Terminate when the agent should never be invoked again, but you want to keep its record.
- Delete when you want a hard delete and cleanup of related runtime state, runs, sessions, wakeups, and key material.
Important edge cases:
- Pausing a terminated agent fails.
- Resuming a terminated agent fails.
- Resuming a pending approval agent fails.
- Terminating an agent revokes all of its API keys.
- Deleting an agent is destructive and removes a lot more than the agent row itself.
API Keys
Agents can have API keys for runtime access.
GET /api/agents/{agentId}/keys
POST /api/agents/{agentId}/keys
DELETE /api/agents/{agentId}/keys/{keyId}
Use keys when an agent or worker needs long-lived authentication that belongs to a specific agent.
Important edge cases:
- A key token is shown only once at create time.
- You cannot create keys for pending approval agents.
- You cannot create keys for terminated agents.
- Revoking a key marks it revoked; it is not hard-deleted.
Example
curl -s -X POST \
"http://localhost:3100/api/agents/{agentId}/keys" \
-H "Authorization: Bearer <token>" \
-H "Content-Type: application/json" \
-d '{"name":"deploy-bot"}'
const res = await fetch(`http://localhost:3100/api/agents/${agentId}/keys`, {
method: "POST",
headers: {
Authorization: `Bearer ${token}`,
"Content-Type": "application/json",
},
body: JSON.stringify({ name: "deploy-bot" }),
});
const key = await res.json();
import requests
res = requests.post(
f"http://localhost:3100/api/agents/{agentId}/keys",
headers={
"Authorization": f"Bearer {token}",
"Content-Type": "application/json",
},
json={"name": "deploy-bot"},
)
key = res.json()
Wake an Agent
Queues or coalesces a wakeup with context.
POST /api/agents/{agentId}/wakeup
Use this when you want to trigger work manually and include extra context such as an issue ID, reason, comment ID, or a payload object.
The request body supports:
source:timer,assignment,on_demand, orautomationtriggerDetail:manual,ping,callback, orsystemreasonpayloadidempotencyKeyforceFreshSession
Important behavior:
- The route is company-scoped.
- Agent auth can only invoke the same agent.
- The endpoint returns
202 Accepted. - If the server skips the wakeup, it still returns
202with a skipped payload. - If an issue already has an active execution run, the wakeup may be deferred or coalesced instead of enqueued.
- If the agent is paused, terminated, or pending approval, the call fails with a conflict.
- Timer wakes can be disabled by heartbeat policy.
- Non-timer wakes can also be disabled by policy.
Example
curl -s -X POST \
"http://localhost:3100/api/agents/{agentId}/wakeup" \
-H "Authorization: Bearer <token>" \
-H "Content-Type: application/json" \
-d '{
"source": "assignment",
"triggerDetail": "manual",
"reason": "Issue assigned",
"payload": {
"issueId": "issue-123",
"commentId": "comment-456"
},
"idempotencyKey": "issue-123-assignment",
"forceFreshSession": false
}'
const res = await fetch(`http://localhost:3100/api/agents/${agentId}/wakeup`, {
method: "POST",
headers: {
Authorization: `Bearer ${token}`,
"Content-Type": "application/json",
},
body: JSON.stringify({
source: "assignment",
triggerDetail: "manual",
reason: "Issue assigned",
payload: {
issueId: "issue-123",
commentId: "comment-456",
},
idempotencyKey: "issue-123-assignment",
forceFreshSession: false,
}),
});
const result = await res.json();
import requests
res = requests.post(
f"http://localhost:3100/api/agents/{agentId}/wakeup",
headers={
"Authorization": f"Bearer {token}",
"Content-Type": "application/json",
},
json={
"source": "assignment",
"triggerDetail": "manual",
"reason": "Issue assigned",
"payload": {
"issueId": "issue-123",
"commentId": "comment-456",
},
"idempotencyKey": "issue-123-assignment",
"forceFreshSession": False,
},
)
result = res.json()
Common skipped responses
The exact skip reason depends on why the wakeup could not be queued. Common examples include:
wakeup_skippedissue_execution_deferredheartbeat.disabledheartbeat.wakeOnDemand.disabledbudget.blocked
Invoke Heartbeat
Triggers a simple on-demand heartbeat.
POST /api/agents/{agentId}/heartbeat/invoke
Use this when you want a straightforward manual tick without the richer wakeup payload.
Behavior is similar to wakeup, but the response is simpler:
202 Accepted- either a run object, or
{ "status": "skipped" } - agent auth can only invoke itself
Org Chart
Returns the company org structure.
GET /api/companies/{companyId}/org
GET /api/companies/{companyId}/org.svg
GET /api/companies/{companyId}/org.png
Use the JSON route when you want to build your own visualizations. Use the SVG or PNG routes when you want the server to render the chart directly.
Notes:
- These routes omit terminated agents.
styleis optional for the SVG and PNG variants.
Adapter Helpers
These routes help you inspect and validate adapter environments.
GET /api/companies/{companyId}/adapters/{type}/models
GET /api/companies/{companyId}/adapters/{type}/detect-model
POST /api/companies/{companyId}/adapters/{type}/test-environment
Use them when you are choosing a model, auto-detecting a recommended model, or checking whether an adapter config is valid before you create or update an agent.
Important notes:
- The adapter type must be known to the server.
- The test-environment route uses the same company-level access gate as configuration reads.
- The server resolves secrets before running the test.
Skills
These routes inspect and sync the skills attached to an agent.
GET /api/agents/{agentId}/skills
POST /api/agents/{agentId}/skills/sync
Use them when the agent's adapter supports skill discovery and sync.
Sync request
Send the desired skill set as the request body. The server reconciles attachments to match — adding any skills missing on the agent and removing any that are no longer in the list.
| Field | Required | Notes |
|---|---|---|
desiredSkills |
yes | Array of references. Each entry is a company-skill UUID, canonical key, or unique slug. Mix and match is fine. |
The same field is accepted at hire time on POST /api/companies/{companyId}/agents and POST /api/companies/{companyId}/agent-hires, so the agent comes online with skills already assigned.
curl -s -X POST \
"http://localhost:3100/api/agents/{agentId}/skills/sync" \
-H "Authorization: Bearer $PAPERCLIP_API_KEY" \
-H "Content-Type: application/json" \
-d '{ "desiredSkills": ["paperclip", "improve-skill"] }'
await fetch(`http://localhost:3100/api/agents/${agentId}/skills/sync`, {
method: "POST",
headers: {
Authorization: `Bearer ${token}`,
"Content-Type": "application/json",
},
body: JSON.stringify({ desiredSkills: ["paperclip", "improve-skill"] }),
});
import os, requests
requests.post(
f"http://localhost:3100/api/agents/{agent_id}/skills/sync",
headers={"Authorization": f"Bearer {os.environ['PAPERCLIP_API_KEY']}",
"Content-Type": "application/json"},
json={"desiredSkills": ["paperclip", "improve-skill"]},
)
Important notes:
- Some adapters do not implement skill sync yet.
- Unsupported adapters still return a useful snapshot with warnings.
- Sync requires update permission on the target agent.
- Skills must already be installed at the company level. The full company-skill lifecycle (file shape, import, scoping, versioning) is documented in the Skills reference.
Configuration
These routes are for reading and managing agent configuration history.
GET /api/agents/{agentId}/configuration
GET /api/agents/{agentId}/config-revisions
GET /api/agents/{agentId}/config-revisions/{revisionId}
POST /api/agents/{agentId}/config-revisions/{revisionId}/rollback
GET /api/companies/{companyId}/agent-configurations
Use them when you need to:
- inspect the effective agent configuration
- audit prior config changes
- roll back to a previous revision
- review all company agent configs in one place
Important notes:
- If you are a board (human) member of the company, configuration and config-revision reads need only your company membership — no
agents:createpermission is required. Agent (non-human) actors still needagents:create(granted, or the legacycanCreateAgentsflag on their own record) to read these. Mutations and environment probes always requireagents:create. - Revisions redact secret-bearing fields before returning them, so reads never expose secrets regardless of who you are.
- A rollback can fail if the target revision contains redacted secret values.
Instructions Bundle
These routes are for file-based instructions management:
PATCH /api/agents/{agentId}/instructions-path
GET /api/agents/{agentId}/instructions-bundle
PATCH /api/agents/{agentId}/instructions-bundle
GET /api/agents/{agentId}/instructions-bundle/file
PUT /api/agents/{agentId}/instructions-bundle/file
DELETE /api/agents/{agentId}/instructions-bundle/file
Use them when the agent’s prompt instructions are stored as files instead of only inline config.
Notes:
- The target agent or an ancestor manager can manage the instructions path.
- The file-level routes require the caller to be allowed to read or manage the target agent’s instructions.
- Relative instructions paths require
adapterConfig.cwd.
Claude Login
POST /api/agents/{agentId}/claude-login
This is board-only and only works for claude_local agents.
Common Edge Cases
- Agent shortname lookup without company context returns
422 Unprocessable Entity. - A shortname collision returns
409 Conflict. - Updating
permissionsthrough the main update route returns422. - Creating a key for a terminated or pending approval agent returns
409. - Pause and resume are not available for terminated agents.
- Wakeup can be skipped even when the request is accepted, especially when policy or execution state blocks it.
Notes
This page was rewritten to match the current agent routes, auth rules, lifecycle behavior, and docs-site markdown conventions.
Changed file:
docs/api/agents.md