* refactor(jobs): extraheer job-mapper naar lib/jobs-mapper.ts + voeg breadcrumb-velden toe Verplaatst JobWithRelations, JOB_INCLUDE, RawJob, PriceRow, pickDescription, computeCost en mapJob naar lib/jobs-mapper.ts (zonder 'use server'). Voegt buildPriceMap helper toe en breidt de types uit met productCode, storyCode en pbiCode via task->story->pbi en product.code includes. Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com> * feat(jobs): voeg GET /api/jobs/[id] route toe + tests * feat(jobs): useJobsRealtime fetch-on-unknown met dedup-Set Wanneer een SSE-event een onbekend job_id bevat, haalt de hook de volledige job op via GET /api/jobs/[id] en upsert die in de store. Een inFlight-Set voorkomt gelijktijdige dubbele fetches voor hetzelfde job_id. Bekende jobs blijven de bestaande partial-upsert gebruiken. Zelfde logica in jobs_initial. Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com> * feat(jobs): JobCard breadcrumb + datum-fallback per kind Voeg productCode/pbiCode/storyCode/startedAt/finishedAt toe aan JobCardProps; bouw breadcrumb per job-kind en toon finishedAt → startedAt → createdAt als datum. JobsColumn geeft de nieuwe velden door. Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com> * test(jobs): JobCard breadcrumb + datum-fallback tests --------- Co-authored-by: Claude Sonnet 4.6 <noreply@anthropic.com>
35 lines
1.2 KiB
TypeScript
35 lines
1.2 KiB
TypeScript
'use server'
|
|
|
|
import { prisma } from '@/lib/prisma'
|
|
import { getSession } from '@/lib/auth'
|
|
import { JOB_INCLUDE, mapJob, buildPriceMap } from '@/lib/jobs-mapper'
|
|
import type { RawJob, JobWithRelations, PriceRow } from '@/lib/jobs-mapper'
|
|
|
|
export type { JobWithRelations } from '@/lib/jobs-mapper'
|
|
|
|
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: 'desc' },
|
|
}),
|
|
prisma.claudeJob.findMany({
|
|
where: { user_id: session.userId, status: 'DONE' },
|
|
include: JOB_INCLUDE,
|
|
orderBy: { created_at: 'desc' },
|
|
take: 100,
|
|
}),
|
|
prisma.modelPrice.findMany(),
|
|
])
|
|
|
|
const priceMap = buildPriceMap(prices as unknown as PriceRow[])
|
|
|
|
return {
|
|
activeJobs: active.map((j) => mapJob(j as unknown as RawJob, priceMap)),
|
|
doneJobs: done.map((j) => mapJob(j as unknown as RawJob, priceMap)),
|
|
}
|
|
}
|