Voegt een verplicht code-veld toe aan Sprint, sequentieel per product (consistent met PBI-N, ST-NNN, T-N). - **Schema** — `Sprint.code String @db.VarChar(30)` + `@@unique([product_id, code])` - **Migratie** — voegt kolom toe als nullable, backfillt bestaande sprints via `ROW_NUMBER() OVER (PARTITION BY product_id ORDER BY created_at)` als `SP-N`, en zet daarna NOT NULL + UNIQUE. - **Generator** — `generateNextSprintCode(productId)` in lib/code-server.ts volgt het patroon van story/pbi/task; createSprintAction gebruikt `createWithCodeRetry` voor race-bescherming. - **Seed** — sprint-counter per product (`SP-1`, `SP-2`, ...). Zichtbaar in: - Sprint-header (`Product › Sprint actief · SP-3`) - JobCard + JobDetailPane voor SPRINT_IMPLEMENTATION jobs - Insights: VelocityChart x-axis (compacter dan goal-truncated), AlignmentTrend tooltip, SprintInfoStrip - actions/jobs-page.ts: `sprintCode` is weer een echte code i.p.v. null Co-authored-by: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
171 lines
5.1 KiB
TypeScript
171 lines
5.1 KiB
TypeScript
'use server'
|
|
|
|
import { prisma } from '@/lib/prisma'
|
|
import { getSession } from '@/lib/auth'
|
|
import type { ClaudeJobKind, ClaudeJobStatus, VerifyResult } from '@prisma/client'
|
|
|
|
export type JobWithRelations = {
|
|
id: string
|
|
kind: ClaudeJobKind
|
|
status: ClaudeJobStatus
|
|
taskCode: string | null
|
|
taskTitle: string | null
|
|
ideaCode: string | null
|
|
ideaTitle: string | null
|
|
sprintGoal: string | null
|
|
sprintCode: string | null
|
|
productName: string
|
|
modelId: string | null
|
|
inputTokens: number | null
|
|
outputTokens: number | null
|
|
cacheReadTokens: number | null
|
|
cacheWriteTokens: number | null
|
|
costUsd: number | null
|
|
branch: string | null
|
|
prUrl: string | null
|
|
error: string | null
|
|
summary: string | null
|
|
description: string | null
|
|
verifyResult: VerifyResult | null
|
|
startedAt: Date | null
|
|
finishedAt: Date | null
|
|
createdAt: Date
|
|
sprintRunId: string | null
|
|
}
|
|
|
|
const JOB_INCLUDE = {
|
|
task: { select: { code: true, title: true, description: true, implementation_plan: true } },
|
|
idea: { select: { code: true, title: true, description: true, grill_md: true, plan_md: true } },
|
|
product: { select: { name: true } },
|
|
sprint_run: { include: { sprint: { select: { sprint_goal: true, code: true } } } },
|
|
} as const
|
|
|
|
type RawJob = {
|
|
id: string
|
|
kind: ClaudeJobKind
|
|
status: ClaudeJobStatus
|
|
model_id: string | null
|
|
input_tokens: number | null
|
|
output_tokens: number | null
|
|
cache_read_tokens: number | null
|
|
cache_write_tokens: number | null
|
|
branch: string | null
|
|
pr_url: string | null
|
|
error: string | null
|
|
summary: string | null
|
|
verify_result: VerifyResult | null
|
|
started_at: Date | null
|
|
finished_at: Date | null
|
|
created_at: Date
|
|
sprint_run_id: string | null
|
|
task: {
|
|
code: string | null
|
|
title: string
|
|
description: string | null
|
|
implementation_plan: string | null
|
|
} | null
|
|
idea: {
|
|
code: string | null
|
|
title: string
|
|
description: string | null
|
|
grill_md: string | null
|
|
plan_md: string | null
|
|
} | null
|
|
product: { name: string }
|
|
sprint_run: { sprint: { sprint_goal: string; code: string } } | null
|
|
}
|
|
|
|
type PriceRow = {
|
|
model_id: string
|
|
input_price_per_1m: { toString: () => string }
|
|
output_price_per_1m: { toString: () => string }
|
|
cache_read_price_per_1m: { toString: () => string }
|
|
cache_write_price_per_1m: { toString: () => string }
|
|
}
|
|
|
|
function pickDescription(j: RawJob): string | null {
|
|
switch (j.kind) {
|
|
case 'TASK_IMPLEMENTATION':
|
|
return j.task?.implementation_plan ?? j.task?.description ?? null
|
|
case 'IDEA_GRILL':
|
|
return j.idea?.grill_md ?? j.idea?.description ?? null
|
|
case 'IDEA_MAKE_PLAN':
|
|
return j.idea?.plan_md ?? j.idea?.description ?? null
|
|
case 'PLAN_CHAT':
|
|
return j.idea?.description ?? null
|
|
case 'SPRINT_IMPLEMENTATION':
|
|
return null
|
|
default:
|
|
return null
|
|
}
|
|
}
|
|
|
|
function computeCost(j: RawJob, priceMap: Map<string, PriceRow>): number | null {
|
|
if (!j.model_id) return null
|
|
const p = priceMap.get(j.model_id)
|
|
if (!p || j.input_tokens == null) return null
|
|
return (
|
|
((j.input_tokens ?? 0) * Number(p.input_price_per_1m.toString())) / 1_000_000 +
|
|
((j.output_tokens ?? 0) * Number(p.output_price_per_1m.toString())) / 1_000_000 +
|
|
((j.cache_read_tokens ?? 0) * Number(p.cache_read_price_per_1m.toString())) / 1_000_000 +
|
|
((j.cache_write_tokens ?? 0) * Number(p.cache_write_price_per_1m.toString())) / 1_000_000
|
|
)
|
|
}
|
|
|
|
function mapJob(j: RawJob, priceMap: Map<string, PriceRow>): JobWithRelations {
|
|
return {
|
|
id: j.id,
|
|
kind: j.kind,
|
|
status: j.status,
|
|
taskCode: j.task?.code ?? null,
|
|
taskTitle: j.task?.title ?? null,
|
|
ideaCode: j.idea?.code ?? null,
|
|
ideaTitle: j.idea?.title ?? null,
|
|
sprintGoal: j.sprint_run?.sprint.sprint_goal ?? null,
|
|
sprintCode: j.sprint_run?.sprint.code ?? null,
|
|
productName: j.product.name,
|
|
modelId: j.model_id,
|
|
inputTokens: j.input_tokens,
|
|
outputTokens: j.output_tokens,
|
|
cacheReadTokens: j.cache_read_tokens,
|
|
cacheWriteTokens: j.cache_write_tokens,
|
|
costUsd: computeCost(j, priceMap),
|
|
branch: j.branch,
|
|
prUrl: j.pr_url,
|
|
error: j.error,
|
|
summary: j.summary,
|
|
description: pickDescription(j),
|
|
verifyResult: j.verify_result,
|
|
startedAt: j.started_at,
|
|
finishedAt: j.finished_at,
|
|
createdAt: j.created_at,
|
|
sprintRunId: j.sprint_run_id,
|
|
}
|
|
}
|
|
|
|
export async function fetchJobsPageData(): Promise<{ activeJobs: JobWithRelations[]; doneJobs: JobWithRelations[] } | null> {
|
|
const session = await getSession()
|
|
if (!session.userId) return null
|
|
|
|
const [active, done, prices] = await Promise.all([
|
|
prisma.claudeJob.findMany({
|
|
where: { user_id: session.userId, status: { notIn: ['DONE'] } },
|
|
include: JOB_INCLUDE,
|
|
orderBy: { created_at: 'asc' },
|
|
}),
|
|
prisma.claudeJob.findMany({
|
|
where: { user_id: session.userId, status: 'DONE' },
|
|
include: JOB_INCLUDE,
|
|
orderBy: { finished_at: 'desc' },
|
|
take: 100,
|
|
}),
|
|
prisma.modelPrice.findMany(),
|
|
])
|
|
|
|
const priceMap = new Map<string, PriceRow>(prices.map((p) => [p.model_id, p as unknown as PriceRow]))
|
|
|
|
return {
|
|
activeJobs: active.map((j) => mapJob(j as RawJob, priceMap)),
|
|
doneJobs: done.map((j) => mapJob(j as RawJob, priceMap)),
|
|
}
|
|
}
|