"use client"; /** * JobStatusPoller — client island for /jobs/[id]. * * Polls /api/jobs/[id] every 3s while job.status is non-terminal. Renders: * - Phase tracker bar (queued → claimed → running → complete | failed) * - Hypothesis cards (prior/posterior bars + Tetlock confidence_band) * - Evidence cards (grade A/B/C + verbatim excerpt + bbox preview) * - Error panel when status='failed' */ import { useEffect, useState } from "react"; import Link from "next/link"; interface JobPayloadOutput { evidence_id?: string; hypothesis_id?: string; case_file?: string; chunk_id?: string; error?: string; skipped?: boolean; reason?: string; kind?: string; } interface InitialJob { job_id: string; kind: string; payload: Record | null; status: string; worker_id: string | null; started_at: string | null; finished_at: string | null; outputs: JobPayloadOutput[]; error: string | null; created_at: string; } interface EvidenceItem { evidence_id: string; grade: string | null; source_page_id: string; doc_id: string | null; page: number | null; chunk_id: string | null; verbatim_excerpt: string | null; custody_steps: Array> | null; bbox: { x: number; y: number; w: number; h: number } | null; confidence_band: string | null; } interface HypothesisItem { hypothesis_id: string; question: string | null; position: string | null; argument_for: string | null; argument_against: string | null; prior: number | string | null; posterior: number | string | null; confidence_band: string | null; status: string | null; } interface FetchedJob extends InitialJob { evidence: EvidenceItem[]; hypotheses: HypothesisItem[]; duration_ms: number | null; } const BAND_COLOR: Record = { high: "text-[#06d6a0] border-[#06d6a0]", medium: "text-[#3fde6a] border-[#3fde6a]", low: "text-[#ffa500] border-[#ffa500]", speculation: "text-[#ff6ec7] border-[#ff6ec7]", }; const GRADE_COLOR: Record = { A: "text-[#06d6a0] border-[#06d6a0]", B: "text-[#3fde6a] border-[#3fde6a]", C: "text-[#ffa500] border-[#ffa500]", }; const STATUS_LABEL: Record = { queued: "Aguardando worker", running: "Investigação em curso", complete: "Concluído", failed: "Falhou", }; const PHASES = ["queued", "running", "complete"] as const; function isTerminal(status: string): boolean { return status === "complete" || status === "failed"; } function asNumber(n: number | string | null): number | null { if (n === null || n === undefined) return null; const v = typeof n === "string" ? parseFloat(n) : n; return Number.isFinite(v) ? v : null; } function formatDuration(ms: number | null): string { if (!ms || ms < 0) return "—"; if (ms < 1000) return `${ms} ms`; if (ms < 60_000) return `${(ms / 1000).toFixed(1)} s`; return `${Math.floor(ms / 60_000)} min ${Math.floor((ms % 60_000) / 1000)} s`; } export function JobStatusPoller(props: { jobId: string; initialJob: InitialJob }) { const [job, setJob] = useState({ ...props.initialJob, evidence: [], hypotheses: [], duration_ms: null, }); const [error, setError] = useState(null); useEffect(() => { let cancelled = false; let timer: ReturnType | null = null; async function tick() { try { const r = await fetch(`/api/jobs/${props.jobId}`, { cache: "no-store" }); if (!r.ok) { if (!cancelled) setError(`HTTP ${r.status}`); } else { const data = (await r.json()) as FetchedJob; if (!cancelled) { setJob(data); setError(null); } } } catch (e) { if (!cancelled) setError((e as Error).message); } if (!cancelled) { const next = isTerminal(job.status) ? null : setTimeout(tick, 3000); timer = next; } } if (!isTerminal(job.status)) { timer = setTimeout(tick, 1500); } else { // Even when terminal, do a single hydrate to fetch evidence/hypotheses tick(); } return () => { cancelled = true; if (timer) clearTimeout(timer); }; // eslint-disable-next-line react-hooks/exhaustive-deps }, [props.jobId, job.status]); const currentPhaseIdx = (() => { if (job.status === "complete") return 2; if (job.status === "failed") return -1; if (job.status === "running") return 1; return 0; })(); return (
{/* Phase tracker */}
Status
{STATUS_LABEL[job.status] ?? job.status} {job.duration_ms !== null && ( · {formatDuration(job.duration_ms)} )}
{PHASES.map((phase, i) => { const done = i <= currentPhaseIdx && job.status !== "failed"; const active = i === currentPhaseIdx && !isTerminal(job.status); return (
{i < PHASES.length - 1 && }
); })}
{PHASES.map((p) => {p})}
{job.worker_id && (
worker: {job.worker_id}
)} {error && (
polling error: {error}
)}
{/* Failure panel */} {job.status === "failed" && (
Job failed
{job.error || "(no error message)"}
          
{job.outputs.length > 0 && (
{job.outputs.length} partial output(s)
{JSON.stringify(job.outputs, null, 2)}
              
)}
)} {/* Hypothesis cards */} {job.hypotheses.length > 0 && (
Hipóteses rivais ({job.hypotheses.length})
{job.hypotheses.map((h) => )}
)} {/* Evidence cards */} {job.evidence.length > 0 && (
Cadeia de evidência ({job.evidence.length})
{job.evidence.map((e) => )}
)} {/* Empty / in-flight state */} {!isTerminal(job.status) && job.hypotheses.length === 0 && job.evidence.length === 0 && (
🔎 Os detetives estão lendo o corpus…
Holmes constrói hipóteses rivais com priors + posteriors em ~60 s.
Locard documenta evidências verbatim com cadeia de custódia em ~30 s por chunk.
)} {/* Outputs raw */} {job.outputs.length > 0 && job.status === "complete" && (
Raw audit outputs ({job.outputs.length})
{JSON.stringify(job.outputs, null, 2)}
          
)}
); } function HypothesisCard({ h }: { h: HypothesisItem }) { const prior = asNumber(h.prior); const posterior = asNumber(h.posterior); const delta = prior !== null && posterior !== null ? posterior - prior : null; const bandTone = (h.confidence_band && BAND_COLOR[h.confidence_band]) || "text-[#9aa6b8] border-[#9aa6b8]"; return (
{h.hypothesis_id}
{h.confidence_band && ( {h.confidence_band} )}
{h.position}
{(prior !== null || posterior !== null) && (
)} {delta !== null && (
Δ {delta >= 0 ? "+" : ""}{delta.toFixed(3)} ·{" "} {delta > 0.05 ? evidência reforçou : delta < -0.05 ? evidência reduziu : evidência ambígua}
)} {h.argument_for && (
Argumento a favor
)} {h.argument_against && (
Argumento contra
)}
); } function EvidenceCard({ e }: { e: EvidenceItem }) { const gradeTone = (e.grade && GRADE_COLOR[e.grade]) || "text-[#9aa6b8] border-[#9aa6b8]"; const bandTone = (e.confidence_band && BAND_COLOR[e.confidence_band]) || "text-[#9aa6b8] border-[#9aa6b8]"; const stepsCount = Array.isArray(e.custody_steps) ? e.custody_steps.length : 0; const cropUrl = e.doc_id && e.page && e.bbox && e.bbox.w > 0 && e.bbox.h > 0 ? `/api/crop?doc=${encodeURIComponent(e.doc_id)}&page=${e.page}&x=${e.bbox.x}&y=${e.bbox.y}&w=${e.bbox.w}&h=${e.bbox.h}&w_px=480` : null; return (
{e.evidence_id}
{e.grade && ( Grade {e.grade} )} {e.confidence_band && ( {e.confidence_band} )}
{e.verbatim_excerpt && (
“{e.verbatim_excerpt}”
)} {cropUrl && ( // eslint-disable-next-line @next/next/no-img-element {`${e.doc_id} )}
{e.doc_id && e.page && ( {e.doc_id}/p{String(e.page).padStart(3, "0")}{e.chunk_id ? `#${e.chunk_id}` : ""} )} · {stepsCount} custody step{stepsCount === 1 ? "" : "s"}
{stepsCount > 0 && (
Cadeia de custódia
    {(e.custody_steps as Array>).map((s, i) => (
  1. {String(s.actor ?? "?")} {s.action ? ` — ${String(s.action)}` : ""} {s.timestamp ? ` (${String(s.timestamp)})` : ""}
  2. ))}
)}
); } function ProbabilityBar({ label, value, color }: { label: string; value: number | null; color: string }) { const pct = value !== null ? Math.round(value * 100) : 0; return (
{label} {value !== null ? value.toFixed(3) : "—"}
); } // Linkify [[doc-id/pNNN#cNNNN]] wiki-links in argument prose into tags. function ArgumentBody({ text }: { text: string }) { const parts: Array<{ kind: "text" | "link"; raw: string; href?: string; label?: string }> = []; const re = /\[\[([^\]|]+?)(?:\|([^\]]+))?\]\]/g; let lastIdx = 0; let m: RegExpExecArray | null; while ((m = re.exec(text)) !== null) { if (m.index > lastIdx) parts.push({ kind: "text", raw: text.slice(lastIdx, m.index) }); const target = m[1]; const label = m[2] ?? m[1]; let href: string | undefined; // doc-id/pNNN#cNNNN const chunkMatch = target.match(/^([a-z0-9][a-z0-9-]*)\/p(\d{3})#(c\d{4})$/); const pageMatch = target.match(/^([a-z0-9][a-z0-9-]*)\/p(\d{3})$/); if (chunkMatch) href = `/d/${chunkMatch[1]}/p${chunkMatch[2]}#${chunkMatch[3]}`; else if (pageMatch) href = `/d/${pageMatch[1]}/p${pageMatch[2]}`; parts.push({ kind: "link", raw: m[0], href, label }); lastIdx = m.index + m[0].length; } if (lastIdx < text.length) parts.push({ kind: "text", raw: text.slice(lastIdx) }); return (
{parts.map((p, i) => p.kind === "text" ? {p.raw} : p.href ? {p.label} : {p.label} )}
); }