- Operators debugging an issue
- Building a non-SDK client (Go, Rust, etc.)
- Wiring custom integrations
Base URL
/api/. Static dashboard files live at /, /setup, etc. — those aren’t part of the REST API.
Authentication
Two auth modes: Admin bearer token — for agents / automation:POST /api/auth/login):
/api/auth/*, /api/setup/*, /api/health, channel webhook receivers (/api/channels/slack/interactions, /api/channels/email/action/*, etc.) — these self-authenticate via signed payloads.
Content types
- Request bodies:
application/json(except channel webhooks, which useapplication/x-www-form-urlencodedper their respective vendor specs) - Response bodies:
application/json
Error format
Every error response has the shape:error_code is stable across versions; the message and docs URL may change.
| Status | Meaning |
|---|---|
| 400 | Malformed request body |
| 401 | Auth required or session invalid |
| 403 | Authenticated but not authorized (e.g. non-operator hitting admin endpoint) |
| 404 | Task / user / identity not found |
| 409 | Idempotency conflict, terminal state, or already-completed |
| 410 | Magic-link token already consumed |
| 422 | Validation error (schema mismatch, malformed config) |
| 429 | Rate limit hit |
| 500 | Server-side error (verifier provider unavailable, etc.) |
| 502 | Vendor outage (Slack, OpenAI, etc.) |
| 503 | Service not configured (e.g. Slack OAuth without credentials) |
OpenAPI spec
Interactive OpenAPI docs at/api/docs (FastAPI’s Swagger UI). The raw spec is at /api/openapi.json.
For SDK code-gen against your own clients:
Rate limits
| Endpoint | Limit | Window | Key |
|---|---|---|---|
POST /api/auth/login | 10 / 5min / IP | 5 min | IP |
POST /api/auth/login | 20 / 5min / email | 5 min | |
POST /api/setup/operator | 30 / 5min / IP | 5 min | IP |
Idempotency
EveryPOST /api/tasks carries a required idempotency_key. The contract is Stripe-style — same key returns the same task, regardless of status, for the lifetime of the row.
- First call with a fresh key: the task is inserted and the response is
201 Created. - Subsequent calls with the same key: the existing task is returned with
201 Createdplus anIdempotent-Replayed: trueresponse header. Notifications (email / Slack) do NOT re-fire on replay — that would re-page the human for already-in-flight work.
201 (instead of flipping to 200 on replay): strict HTTP semantics reserve 201 for newly-created resources, but Stripe — the model this contract is named after — keeps the original status code on replay and signals replay-vs-fresh via header. That convention breaks zero clients that check status == 201 for success while still being explicit for clients that care.
To trigger a fresh task for the same logical event (e.g. retry a refund review after an earlier one was rejected), pass a distinct key — convention is to suffix: "refund-A-4721:retry-1".
Endpoints by resource
Tasks (admin / operator / assignee)
| Method | Path | Description |
|---|---|---|
POST | /api/tasks | Create a task. Admin / operator only. |
GET | /api/tasks | List tasks. Operator: all. Assignee: scoped to own. |
GET | /api/tasks/{id} | Get one task. Operator / admin / assignee. |
POST | /api/tasks/{id}/complete | Submit response. Operator / admin / assignee. |
POST | /api/tasks/{id}/cancel | Cancel. Operator / admin only. |
GET | /api/tasks/{id}/poll | Long-poll until terminal. Operator / admin / assignee. |
GET | /api/tasks/{id}/audit | Audit trail. Operator / admin only. |
DELETE | /api/tasks/{id} | Hard delete. Operator / admin only. |
Auth
| Method | Path | Description |
|---|---|---|
POST | /api/auth/login | Email + password → session cookie. Public. |
POST | /api/auth/logout | Clear session cookie. |
GET | /api/auth/me | Session introspection. |
Setup (first-run bootstrap)
| Method | Path | Description |
|---|---|---|
GET | /api/setup/status | Is first-run setup needed? |
POST | /api/setup/operator | Create first operator. One-shot. |
Users (admin / operator)
| Method | Path | Description |
|---|---|---|
GET | /api/users | List directory users. |
POST | /api/users | Create. |
GET | /api/users/{id} | Get one. |
PATCH | /api/users/{id} | Update. |
DELETE | /api/users/{id} | Soft-delete (sets active=false). |
Channels
| Method | Path | Description |
|---|---|---|
POST | /api/channels/slack/interactions | Slack webhook (signed). |
GET/POST | /api/channels/slack/oauth/* | OAuth install flow. |
GET/POST | /api/channels/slack/installations* | OAuth install CRUD (admin). |
GET/POST | /api/channels/email/identities* | Sender identity CRUD (admin). |
GET/POST | /api/channels/email/action/{token} | Magic-link action page (signed). |
Stats
| Method | Path | Description |
|---|---|---|
GET | /api/stats | Task counts by status, completion rate, last 30d. Operator / admin. |
Health
| Method | Path | Description |
|---|---|---|
GET | /api/health | Readiness probe. Public. |
Cross-version stability
- The
/api/tasks/*shape is stable for v1.x — breaking changes are a v2 event. - Error codes (
error_codefield) are stable across all v0.x. - The dashboard and channel webhook routes are internal — third-party clients should NOT depend on their shapes.