Schema sync vanaf upstream Scrum4Me (v77617e8): FAILED toegevoegd aan Task/Story/Pbi/SprintStatus, nieuw SprintRunStatus + PrStrategy enums, SprintRun model, ClaudeJob.sprint_run_id, Product.pr_strategy. T-18 — propagateStatusUpwards in src/lib/tasks-status-update.ts. Real-time cascade Task → Story → PBI → Sprint → SprintRun bij elke task-statuswijziging. Bij FAILED cancelt sibling-jobs in dezelfde SprintRun. PBI-status BLOCKED blijft handmatig. Houd deze helper bit- voor-bit synchroon met Scrum4Me/lib/tasks-status-update.ts. updateTaskStatusWithStoryPromotion blijft als BC-wrapper. T-19 — wait-for-job.ts claim-filter. Task-jobs worden alleen geclaimd als hun SprintRun status QUEUED of RUNNING heeft. Idea-jobs blijven ongefilterd. Bij eerste claim van een QUEUED SprintRun → RUNNING binnen dezelfde tx (race-safe). T-20 — update-job-status.ts roept propagateStatusUpwards aan na elke task DONE/FAILED. Bestaande cancelPbiOnFailure-aanroep blijft voor PR-cleanup; sibling-cancellation overlap is harmless (idempotent). T-21 — classify.ts (verifier) leest nu ook "--- a/<path>" zodat delete-only commits niet meer als EMPTY worden geclassificeerd. Bug had eerder geleid tot ten onrechte FAILED-status op cmotto5h en cmotto5i (06-05-2026); zou met cascade-flow een hele sprint laten falen. Cleanup: create-todo.ts en open_todos in get-claude-context.ts verwijderd (Todo-model is op main gedropt). Endpoint geeft nu open_ideas terug — ideeën die niet PLANNED zijn. Status-mappers (src/status.ts) uitgebreid met failed. Tests: 184/184 groen (180 → 184; vier nieuwe delete-only classify-tests en herwerkte propagate-status tests). Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
53 lines
1.4 KiB
TypeScript
53 lines
1.4 KiB
TypeScript
import type { TaskStatus, StoryStatus } from '@prisma/client'
|
|
|
|
const TASK_DB_TO_API = {
|
|
TO_DO: 'todo',
|
|
IN_PROGRESS: 'in_progress',
|
|
REVIEW: 'review',
|
|
DONE: 'done',
|
|
FAILED: 'failed',
|
|
} 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',
|
|
failed: 'FAILED',
|
|
}
|
|
|
|
const STORY_DB_TO_API = {
|
|
OPEN: 'open',
|
|
IN_SPRINT: 'in_sprint',
|
|
DONE: 'done',
|
|
FAILED: 'failed',
|
|
} as const satisfies Record<StoryStatus, string>
|
|
|
|
const STORY_API_TO_DB: Record<string, StoryStatus> = {
|
|
open: 'OPEN',
|
|
in_sprint: 'IN_SPRINT',
|
|
done: 'DONE',
|
|
failed: 'FAILED',
|
|
}
|
|
|
|
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)
|