* docs(ST-511): add backlog entry for entity codes feature Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com> * fix(ST-511): add createWithCodeRetry helper to handle P2002 race on auto codes Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com> * fix(ST-511): retry on auto-code unique conflict in story and pbi create Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com> * fix(ST-511): surface field errors for code and title in PBI dialog Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com> * fix(ST-511): read create-state errors in Story dialog fieldError Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com> * docs(ST-512): add backlog entry for REST API code/description/implementation_plan extensions; mark ST-511 done Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com> * feat(ST-512): extend REST API with code, description and implementation_plan - GET /api/products returns code, description and definition_of_done - GET /api/products/:id/next-story returns story.code and per-task code + implementation_plan - GET /api/sprints/:id/tasks returns description, implementation_plan, story_code and derived per-task code - POST /api/todos accepts and returns optional description (max 2000) All changes are additive — existing clients ignore unknown keys. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com> * docs(ST-512): mark ST-512 as done Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com> * docs(ST-513): add backlog entry for API hardening for Claude Code Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com> * feat(ST-513): add task and story status mappers for API boundary Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com> * feat(ST-513): expose lowercase status on API and accept lowercase in PATCH /api/tasks Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com> * feat(ST-513): add metadata JSONB column to StoryLog Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com> * feat(ST-513): accept optional metadata in story log and switch validation errors to 422 Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com> * feat(ST-513): add GET /api/health endpoint with optional db ping Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com> * feat(ST-513): add GET /api/products/:id/claude-context bundled endpoint Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com> * docs(ST-513): add docs/API.md and link from CLAUDE.md specs table Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com> * docs(ST-513): mark ST-513 as done Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com> * fix(ST-513): split 400 (malformed JSON) from 422 (validation), reject 'review' Codex review on PR #2: - P2.1: routes treated JSON parse failures as 422 instead of 400, breaking the contract in docs/API.md. Replace `request.json().catch(() => null)` with try/catch in 4 routes (tasks, reorder, todos, story-log) so a malformed body returns 400 and only well-formed-but-invalid bodies return 422. - P2.2: PATCH /api/tasks/:id accepted `status: "review"`, but the sprint task list UI does not render REVIEW (no label/color, the cycle helper falls back to TO_DO). Reject `review` at the API until the sprint UI is extended; document the subset in docs/API.md. Tests in __tests__/api updated for the new contract (29 assertions: zod-failures now expect 422, status payloads use lowercase API values, sprint-tasks mocks include the new story relation). Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com> --------- Co-authored-by: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
52 lines
1.4 KiB
TypeScript
52 lines
1.4 KiB
TypeScript
// Bidirectionele case-mappers voor de REST API-boundary.
|
|
// DB houdt UPPER_SNAKE; API exposeert lowercase.
|
|
|
|
import type { TaskStatus, StoryStatus } from '@prisma/client'
|
|
|
|
const TASK_DB_TO_API = {
|
|
TO_DO: 'todo',
|
|
IN_PROGRESS: 'in_progress',
|
|
REVIEW: 'review',
|
|
DONE: 'done',
|
|
} as const satisfies Record<TaskStatus, string>
|
|
|
|
const TASK_API_TO_DB: Record<string, TaskStatus> = {
|
|
todo: 'TO_DO',
|
|
in_progress: 'IN_PROGRESS',
|
|
review: 'REVIEW',
|
|
done: 'DONE',
|
|
}
|
|
|
|
const STORY_DB_TO_API = {
|
|
OPEN: 'open',
|
|
IN_SPRINT: 'in_sprint',
|
|
DONE: 'done',
|
|
} as const satisfies Record<StoryStatus, string>
|
|
|
|
const STORY_API_TO_DB: Record<string, StoryStatus> = {
|
|
open: 'OPEN',
|
|
in_sprint: 'IN_SPRINT',
|
|
done: 'DONE',
|
|
}
|
|
|
|
export type TaskStatusApi = typeof TASK_DB_TO_API[TaskStatus]
|
|
export type StoryStatusApi = typeof STORY_DB_TO_API[StoryStatus]
|
|
|
|
export function taskStatusToApi(s: TaskStatus): TaskStatusApi {
|
|
return TASK_DB_TO_API[s]
|
|
}
|
|
|
|
export function taskStatusFromApi(s: string): TaskStatus | null {
|
|
return TASK_API_TO_DB[s.toLowerCase()] ?? null
|
|
}
|
|
|
|
export function storyStatusToApi(s: StoryStatus): StoryStatusApi {
|
|
return STORY_DB_TO_API[s]
|
|
}
|
|
|
|
export function storyStatusFromApi(s: string): StoryStatus | null {
|
|
return STORY_API_TO_DB[s.toLowerCase()] ?? null
|
|
}
|
|
|
|
export const TASK_STATUS_API_VALUES = Object.values(TASK_DB_TO_API)
|
|
export const STORY_STATUS_API_VALUES = Object.values(STORY_DB_TO_API)
|