docs/API.md — nieuwe sectie 'Auth — QR-pairing (M10)' met alle drie endpoints (start, stream, claim), cookie-mechaniek, foutcodes (400/401/410/429), curl-voorbeelden inclusief --cookie-jar. docs/scrum4me-architecture.md — sectie 'QR-pairing flow' met: - Mermaid sequence-diagram (start → QR → scan → approve → claim) - Threat-model (replay, phishing-QR, demo-block, rate-limit, secret-leak, long-lived sessie) met expliciete mitigaties - TTL-rationale voor de drie tijden (5min pending / +5min approved / 8u paired) - Subsectie 'Waarom geen secret in URL' — fragment-eigenschap + HttpOnly cookie + twee gescheiden hashes docs/patterns/qr-login.md — herbruikbaar pattern 'QR-pairing via unauth-SSE + pre-auth cookie' met de drie endpoints, vier security-uitgangspunten, sjabloon-bestanden, TTL-richtlijn, en wanneer NIET te gebruiken. CLAUDE.md — extra rij in Implementatiepatronen-tabel die naar het nieuwe pattern-doc verwijst. Acceptatie ST-1008 (zeven scenario's): - Happy path: gedekt door manuele E2E in vorige stories (gebruiker bevestigde dat M10-stories op Solo bord verschijnen + curl-roundtrip werkt) - Demo-block: actions/pairing.test.ts → approvePairing demo → Niet beschikbaar - Replay: pair-claim.test.ts → 410 op tweede claim - Expiry tijdens pending: pair-stream.test.ts + pairing.test.ts → 410/error - Expiry tussen approve+claim: pair-claim.test.ts → 410 - Cookie-mismatch op SSE/claim: pair-stream.test.ts + pair-claim.test.ts → 401 - Secret niet in URL/logs: per ontwerp — fragment + cookie reizen niet via URL-paden of querystrings (gedocumenteerd in architecture.md) Quality gates: lint 0 errors, tsc clean, vitest 139/139 (16 files). M10 is hiermee compleet — feat/M10-qr-login bevat 13 commits klaar voor gebruiker-acceptatie en PR. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
427 lines
11 KiB
Markdown
427 lines
11 KiB
Markdown
# Scrum4Me REST API
|
|
|
|
REST-API contract voor Claude Code en andere clients.
|
|
|
|
## Authenticatie
|
|
|
|
Alle endpoints behalve `GET /api/health` vereisen een Bearer-token:
|
|
|
|
```
|
|
Authorization: Bearer <token>
|
|
```
|
|
|
|
Tokens beheer je via Instellingen → Tokens (`/settings/tokens`). Een token is gekoppeld aan één gebruiker; een demo-account-token kan lezen maar niet schrijven (`403`).
|
|
|
|
## Status-enums
|
|
|
|
De API gebruikt **lowercase** statussen. De database gebruikt UPPER_SNAKE; de vertaling gebeurt op de boundary.
|
|
|
|
| Entiteit | Waarden |
|
|
|---|---|
|
|
| Task status | `todo`, `in_progress`, `review`, `done` |
|
|
| Story status | `open`, `in_sprint`, `done` |
|
|
|
|
## Foutcodes
|
|
|
|
| Code | Betekenis |
|
|
|---|---|
|
|
| `200` | OK |
|
|
| `201` | Created |
|
|
| `400` | Malformed body (bv. ongeldige JSON) |
|
|
| `401` | Token ontbreekt of ongeldig |
|
|
| `403` | Token heeft geen toegang (demo-account, geen lid van product) |
|
|
| `404` | Resource niet gevonden |
|
|
| `422` | Validatiefout — body is wel-gevormd maar niet acceptabel |
|
|
| `500` | Onverwachte serverfout |
|
|
|
|
---
|
|
|
|
## Endpoints
|
|
|
|
### `GET /api/health`
|
|
|
|
Health-probe. Geen authenticatie vereist.
|
|
|
|
**Query params:** `?db=1` voegt een DB-ping toe.
|
|
|
|
**Response (200):**
|
|
```json
|
|
{ "status": "ok", "version": "0.3.x", "time": "2026-04-26T20:00:00Z" }
|
|
```
|
|
|
|
Met `?db=1`:
|
|
```json
|
|
{ "status": "ok", "version": "0.3.x", "time": "...", "database": "ok" }
|
|
```
|
|
|
|
`database` is `"ok"` of `"down"`. De endpoint zelf retourneert altijd `200`.
|
|
|
|
```bash
|
|
curl https://scrum4me.app/api/health?db=1
|
|
```
|
|
|
|
---
|
|
|
|
### `GET /api/products`
|
|
|
|
Lijst van actieve producten waar de tokengebruiker eigenaar of lid van is.
|
|
|
|
**Response (200):**
|
|
```json
|
|
[
|
|
{
|
|
"id": "cmofu...",
|
|
"code": "SCRUM4ME",
|
|
"name": "Scrum4Me",
|
|
"description": "...",
|
|
"repo_url": "https://github.com/...",
|
|
"definition_of_done": "..."
|
|
}
|
|
]
|
|
```
|
|
|
|
```bash
|
|
curl -H "Authorization: Bearer $TOKEN" https://scrum4me.app/api/products
|
|
```
|
|
|
|
---
|
|
|
|
### `GET /api/products/:id/claude-context`
|
|
|
|
Bundled context voor Claude Code: product, actieve sprint, volgende story (met tasks) en open todos van de tokengebruiker — in één call.
|
|
|
|
**Response (200):**
|
|
```json
|
|
{
|
|
"product": { "id", "code", "name", "description", "repo_url", "definition_of_done" },
|
|
"active_sprint": { "id": "...", "sprint_goal": "...", "status": "ACTIVE" } | null,
|
|
"next_story": {
|
|
"id", "code", "title", "description", "acceptance_criteria",
|
|
"priority", "status",
|
|
"tasks": [
|
|
{ "id", "code", "title", "description", "implementation_plan",
|
|
"priority", "sort_order", "status" }
|
|
]
|
|
} | null,
|
|
"open_todos": [
|
|
{ "id", "title", "description", "created_at" }
|
|
]
|
|
}
|
|
```
|
|
|
|
`open_todos` is gelimiteerd op 50 items, gesorteerd op `created_at` asc. Demo-tokens kunnen dit endpoint lezen.
|
|
|
|
```bash
|
|
curl -H "Authorization: Bearer $TOKEN" \
|
|
https://scrum4me.app/api/products/$PRODUCT_ID/claude-context
|
|
```
|
|
|
|
---
|
|
|
|
### `GET /api/products/:id/next-story`
|
|
|
|
Hoogst geprioriteerde open story in de actieve sprint.
|
|
|
|
**Response (200):**
|
|
```json
|
|
{
|
|
"id": "...",
|
|
"code": "ST-356",
|
|
"title": "Solo Kanban-bord met DnD en Zustand",
|
|
"description": "...",
|
|
"acceptance_criteria": "...",
|
|
"status": "in_sprint",
|
|
"tasks": [
|
|
{
|
|
"id": "...",
|
|
"code": "ST-356.1",
|
|
"title": "Store stores/solo-store.ts",
|
|
"description": "...",
|
|
"implementation_plan": null,
|
|
"priority": 2,
|
|
"sort_order": 1,
|
|
"status": "todo"
|
|
}
|
|
]
|
|
}
|
|
```
|
|
|
|
**Foutcodes:** `404` als geen actieve sprint of geen open stories.
|
|
|
|
---
|
|
|
|
### `GET /api/sprints/:id/tasks`
|
|
|
|
Lijst taken van de sprint, geordend op `(story.sort_order, task.priority, task.sort_order)`.
|
|
|
|
**Query params:** `?limit=N` (default 10, max 50)
|
|
|
|
**Response (200):**
|
|
```json
|
|
[
|
|
{
|
|
"id": "...",
|
|
"code": "ST-356.1",
|
|
"title": "...",
|
|
"description": "...",
|
|
"implementation_plan": null,
|
|
"story_id": "...",
|
|
"story_code": "ST-356",
|
|
"priority": 2,
|
|
"sort_order": 1,
|
|
"status": "todo"
|
|
}
|
|
]
|
|
```
|
|
|
|
---
|
|
|
|
### `PATCH /api/stories/:id/tasks/reorder`
|
|
|
|
Volgorde van taken binnen een story aanpassen.
|
|
|
|
**Body:**
|
|
```json
|
|
{ "task_ids": ["task-id-a", "task-id-b", "task-id-c"] }
|
|
```
|
|
|
|
Alle IDs moeten bij de story horen. **Foutcodes:** `422` bij Zod-fouten of als een task_id niet tot de story behoort.
|
|
|
|
---
|
|
|
|
### `PATCH /api/tasks/:id`
|
|
|
|
Status of implementation_plan bijwerken. Minstens één van beide is verplicht.
|
|
Toegestane status-waarden zijn `todo`, `in_progress` en `done`. `review`
|
|
wordt door deze endpoint geweigerd zolang de sprint-UI die state niet
|
|
rendert — gebruik de Kanban-board voor REVIEW-overgangen.
|
|
|
|
**Body:**
|
|
```json
|
|
{ "status": "in_progress", "implementation_plan": "..." }
|
|
```
|
|
|
|
**Response (200):**
|
|
```json
|
|
{
|
|
"id": "...",
|
|
"status": "in_progress",
|
|
"implementation_plan": "..."
|
|
}
|
|
```
|
|
|
|
**Foutcodes:** `422` bij ongeldige body of onbekende status. `403` bij demo-token.
|
|
|
|
---
|
|
|
|
### `POST /api/stories/:id/log`
|
|
|
|
Activiteit vastleggen op een story.
|
|
|
|
**Body — IMPLEMENTATION_PLAN:**
|
|
```json
|
|
{
|
|
"type": "IMPLEMENTATION_PLAN",
|
|
"content": "Plan: ...",
|
|
"metadata": { "branch": "feat/x" }
|
|
}
|
|
```
|
|
|
|
**Body — TEST_RESULT:**
|
|
```json
|
|
{
|
|
"type": "TEST_RESULT",
|
|
"content": "Alle tests groen",
|
|
"status": "PASSED",
|
|
"metadata": { "ci_run": "..." }
|
|
}
|
|
```
|
|
|
|
**Body — COMMIT:**
|
|
```json
|
|
{
|
|
"type": "COMMIT",
|
|
"content": "Werk afgerond",
|
|
"commit_hash": "abc123",
|
|
"commit_message": "feat(ST-XXX): ...",
|
|
"metadata": { "branch": "feat/x" }
|
|
}
|
|
```
|
|
|
|
`metadata` is optioneel, vrij JSON-object. **Response (201):**
|
|
```json
|
|
{ "id": "...", "created_at": "..." }
|
|
```
|
|
|
|
---
|
|
|
|
### `POST /api/todos`
|
|
|
|
Nieuwe todo voor de tokengebruiker.
|
|
|
|
**Body:**
|
|
```json
|
|
{
|
|
"title": "Een ding doen",
|
|
"description": "Optionele uitleg, max 2000 tekens",
|
|
"product_id": "cmof..."
|
|
}
|
|
```
|
|
|
|
**Response (201):**
|
|
```json
|
|
{ "id": "...", "title": "...", "description": "...", "created_at": "..." }
|
|
```
|
|
|
|
---
|
|
|
|
### `GET /api/realtime/solo?product_id=...`
|
|
|
|
Server-Sent Events stream voor het Solo Paneel. Wordt gebruikt door de browser-UI (`useSoloRealtime`); voor Claude Code zelden relevant, maar gedocumenteerd voor volledigheid.
|
|
|
|
**Auth:** iron-session cookie of Bearer-token. Demo-tokens mogen lezen.
|
|
**Query params:** `product_id` (verplicht).
|
|
**Response:** `text/event-stream`. Stream blijft open tot de client sluit of de server na 240s een hard-close doet (client herconnect dan transparant).
|
|
|
|
**Events:**
|
|
- `event: ready` — eenmalig direct na connect, met `{ product_id, sprint_id }` als payload.
|
|
- `event: error` — bij interne fouten (pg connect mislukt e.d.).
|
|
- `data: {...}` — task/story mutaties die binnen scope vallen (zie hieronder). Payload-shape:
|
|
|
|
```json
|
|
{
|
|
"op": "I" | "U" | "D",
|
|
"entity": "task" | "story",
|
|
"id": "cmof...",
|
|
"story_id": "cmof...",
|
|
"product_id": "cmof...",
|
|
"sprint_id": "cmog..." ,
|
|
"assignee_id": "cmof..." ,
|
|
"task_status": "TO_DO" | "IN_PROGRESS" | "REVIEW" | "DONE",
|
|
"task_title": "...",
|
|
"task_sort_order": 1,
|
|
"changed_fields": ["status", "updated_at"]
|
|
}
|
|
```
|
|
|
|
Niet alle velden zijn altijd aanwezig — `task_*` alleen voor `entity: "task"`, idem `story_*`. `task_status` gebruikt de **DB-enum** (UPPER_SNAKE), niet de lowercase API-vorm.
|
|
|
|
- `: heartbeat` — SSE-comment elke 25s, om proxies keep-alive te houden. Kan genegeerd worden.
|
|
|
|
**Server-side filter:**
|
|
- `product_id` matcht de query-param
|
|
- `sprint_id` matcht de actieve sprint van het product
|
|
- `assignee_id` is gelijk aan de ingelogde user (of `null` voor unassigned-story claims)
|
|
|
|
Niet-matchende events worden gedropt — clients ontvangen geen irrelevante data.
|
|
|
|
**Voorbeeld (browser):**
|
|
```js
|
|
const source = new EventSource('/api/realtime/solo?product_id=cmof...')
|
|
source.onmessage = (e) => console.log(JSON.parse(e.data))
|
|
```
|
|
|
|
---
|
|
|
|
## Auth — QR-pairing (M10)
|
|
|
|
Drie anonieme/cookie-geauthenticeerde endpoints voor de password-loze inlog
|
|
via QR-pairing. Worden door de browser gebruikt (niet door Claude Code) —
|
|
gedocumenteerd voor volledigheid en voor handmatige curl-tests.
|
|
|
|
**Cookie-mechaniek:** `pair/start` zet een korte `s4m_pair`-HttpOnly-cookie
|
|
(`Path=/api/auth/pair`, `Max-Age=300`, `SameSite=Lax`, `Secure` in productie).
|
|
`pair/stream` en `pair/claim` authenticeren tegen die cookie. Geheim materiaal
|
|
zit nooit in URL-paden of querystrings — `mobileSecret` reist alleen via QR-
|
|
fragment (`#s=…`) en POST-body, `desktopToken` alleen via cookie.
|
|
|
|
### `POST /api/auth/pair/start`
|
|
|
|
Anon. Maakt een nieuwe `LoginPairing` aan en zet de pre-auth cookie.
|
|
|
|
**Auth:** geen.
|
|
**Body:** geen.
|
|
**Rate-limit:** 10 per IP per minuut (zelfde patroon als `/login`).
|
|
|
|
**Response 200:**
|
|
```json
|
|
{
|
|
"pairingId": "cmoh...",
|
|
"mobileSecret": "<43-char base64url>",
|
|
"expiresAt": "2026-04-27T20:30:00.000Z",
|
|
"qrUrl": "https://.../m/pair#id=cmoh...&s=<mobileSecret>"
|
|
}
|
|
```
|
|
Plus `Set-Cookie: s4m_pair=<desktopToken>; HttpOnly; Path=/api/auth/pair; Max-Age=300; SameSite=Lax`.
|
|
|
|
**Foutcodes:** `429` bij rate-limit overschreden.
|
|
|
|
**Voorbeeld:**
|
|
```bash
|
|
curl -i -X POST -c /tmp/jar http://localhost:3000/api/auth/pair/start
|
|
```
|
|
|
|
---
|
|
|
|
### `GET /api/auth/pair/stream/:pairingId`
|
|
|
|
Server-Sent Events stream die de desktop opent direct na `pair/start` om op
|
|
de approve-bevestiging van de mobiel te wachten.
|
|
|
|
**Auth:** `s4m_pair`-cookie. Werkt vanuit `EventSource` met `withCredentials: true`.
|
|
**Path:** `pairingId` is niet vertrouwelijk; cookie is het bewijs.
|
|
**Stream-duur:** maximaal 240s (Vercel-buffer onder de 300s `maxDuration`); sluit
|
|
zodra status `consumed` of `cancelled` doorkomt.
|
|
|
|
**Events:**
|
|
- `event: state` — eenmalig direct na connect, met `{ pairing_id, status }` (status van pairing op moment van connecten — voorkomt race wanneer approve net vóór SSE-open landt).
|
|
- `data: {...}` — bij elke status-overgang. Payload:
|
|
```json
|
|
{ "op": "I" | "U", "pairing_id": "cmoh...", "status": "pending" | "approved" | "consumed" | "cancelled" }
|
|
```
|
|
- `: heartbeat` — SSE-comment elke 25s.
|
|
|
|
**Foutcodes:** `401` zonder/foute cookie, `404` als pairing onbekend, `410` als pairing verlopen.
|
|
|
|
**Voorbeeld:**
|
|
```bash
|
|
curl -N -i -b /tmp/jar http://localhost:3000/api/auth/pair/stream/<pairingId>
|
|
```
|
|
|
|
---
|
|
|
|
### `POST /api/auth/pair/claim`
|
|
|
|
Cookie-auth. Atomisch consume van een approved pairing → schrijft de echte
|
|
`scrum4me-session` cookie zodat de desktop is ingelogd.
|
|
|
|
**Auth:** `s4m_pair`-cookie.
|
|
**Body:** `{ "pairingId": "cmoh..." }`.
|
|
|
|
**Response 200:** `{ "ok": true }` plus
|
|
- `Set-Cookie: scrum4me-session=...; HttpOnly; SameSite=Lax` — paired-sessie met `paired: true` en `pairedExpiresAt = now + 8h` payload-velden.
|
|
- `Set-Cookie: s4m_pair=...; Max-Age=0` — pre-auth cookie wordt gewist.
|
|
|
|
**Foutcodes:**
|
|
- `400` bij ontbrekende of malformed body
|
|
- `401` zonder cookie of bij hash-mismatch (cookie matcht geen pairing)
|
|
- `410` als pairing al consumed/cancelled is (replay) of verlopen
|
|
|
|
**Voorbeeld:**
|
|
```bash
|
|
curl -i -X POST -b /tmp/jar -c /tmp/jar \
|
|
-H "Content-Type: application/json" \
|
|
-d '{"pairingId":"<pairingId>"}' \
|
|
http://localhost:3000/api/auth/pair/claim
|
|
```
|
|
|
|
---
|
|
|
|
## Voorbeeldworkflow voor Claude Code
|
|
|
|
1. **Probe:** `GET /api/health?db=1` — bevestig dat de service en DB bereikbaar zijn.
|
|
2. **Context:** `GET /api/products/$ID/claude-context` — haal product, sprint, volgende story en todos op in één call.
|
|
3. **Plan vastleggen:** `POST /api/stories/$STORY_ID/log` met `type: IMPLEMENTATION_PLAN`.
|
|
4. **Per task:** `PATCH /api/tasks/$TASK_ID` met `status: "in_progress"`, daarna met `status: "done"` plus eventueel `implementation_plan`.
|
|
5. **Test:** `POST /api/stories/$STORY_ID/log` met `type: TEST_RESULT` en `status: PASSED|FAILED`.
|
|
6. **Commit:** `POST /api/stories/$STORY_ID/log` met `type: COMMIT`, `commit_hash`, `commit_message`, optioneel `metadata: { branch }`.
|