feat(digests): Phase 2 — API endpoints + /digests UI (X12)

משטחי-משתמש לקורפוס היומונים: endpoints ב-FastAPI + דף UI נפרד /digests
(לדפדוף, חיפוש, העלאה, וקישור לפסק המקורי). היומון נשאר מקור-משני המצביע
על הפסק — אינו מצוטט בהחלטה (INV-DIG1) ואינו מחלץ הלכות (INV-DIG2).

Backend (container-safe + local split):
- digest_library: פוצל ל-create_pending_digest (CONTAINER-SAFE: stage+
  extract_text+create row 'pending', בלי LLM) ↔ enrich_digest/
  process_pending_digests (local: LLM+embed+autolink). ingest_digest מאחד.
- db.list_pending_digests; MCP digest_process_pending (tool+server) — חלופה
  ל-batch script לריקון התור.
- web/app.py: 10 endpoints /api/digests/* (upload/list/search/queue-pending/
  get/patch/delete/link/relink/unlink). upload=INSERT-only pending (ה-LLM רץ
  מקומית — claude_session local-only). כולם מחזירים dict בדפוס precedent.

Frontend (Next 16, ללא api:types — hooks עם טיפוסים hand-written כמו
precedent-library.ts):
- lib/api/digests.ts — hooks (useDigests/useDigestSearch/useDigestPending/
  useUploadDigest/useLink/Relink/Unlink/Delete/Update).
- דף /digests נפרד (לא כרטיסייה ב-/precedents — לשמור גבול סמכותי/משני,
  INV-DIG1): טאבים יומונים/חיפוש + DigestCard (badge קישור-לפסק) +
  DigestUploadDialog + pending badge. nav + header-context.

אומת: backend round-trip מלא (create_pending→list_pending→process_pending→
search→restore); web-ui מתקמפל (webpack/tsc נקי, route /digests נוצר).
הערה: build דיפולטי (turbopack) נכשל ב-worktree עקב symlink ל-node_modules —
ב-CI/Docker (node_modules אמיתי) עובד; אומת עם --webpack.

Invariants: מקיים INV-DIG1/2 (upload לא מחלץ הלכות, UI מציג "מצביע לא
מצוטט"), INV-DIG3 (link/relink/queue). G4 (אין בליעה — שגיאות→toast/HTTP),
G2 (מסלול נפרד, לא מקביל). X6 (חוזה UI↔API — endpoints בדפוס precedent;
hooks hand-written כמו שאר ה-domain modules).

Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
This commit is contained in:
2026-06-07 18:11:05 +00:00
parent 955675eb1f
commit 06281996ca
13 changed files with 1305 additions and 120 deletions

View File

@@ -0,0 +1,136 @@
"use client";
import { useState } from "react";
import { Link2, Trash2 } from "lucide-react";
import { toast } from "sonner";
import { Button } from "@/components/ui/button";
import { Skeleton } from "@/components/ui/skeleton";
import {
Select, SelectContent, SelectItem, SelectTrigger, SelectValue,
} from "@/components/ui/select";
import {
useDigests, useDeleteDigest, useRelinkDigest, type DigestListFilters,
} from "@/lib/api/digests";
import type { PracticeArea } from "@/lib/api/precedent-library";
import { PRACTICE_AREAS } from "@/components/precedents/practice-area";
import { DigestCard } from "./digest-card";
import { DigestUploadDialog } from "./digest-upload-dialog";
type LinkedFilter = "all" | "linked" | "unlinked";
export function DigestListPanel() {
const [practiceArea, setPracticeArea] = useState<PracticeArea>("");
const [linked, setLinked] = useState<LinkedFilter>("all");
const filters: DigestListFilters = {
practiceArea: practiceArea || undefined,
linked: linked === "all" ? undefined : linked === "linked",
limit: 200,
};
const { data, isLoading, error } = useDigests(filters);
const del = useDeleteDigest();
const relink = useRelinkDigest();
const onRelink = (id: string) =>
relink.mutate(id, {
onSuccess: (r) =>
r.linked
? toast.success("קושר לפסק המקורי")
: toast.info("הפסק המקורי עדיין לא בקורפוס — העלה אותו לספריית הפסיקה"),
onError: (e) => toast.error(e.message),
});
const onDelete = (id: string) => {
if (!confirm("למחוק את היומון מקורפוס-הגילוי?")) return;
del.mutate(id, {
onSuccess: () => toast.success("נמחק"),
onError: (e) => toast.error(e.message),
});
};
return (
<div className="space-y-4">
<div className="flex items-end gap-3 flex-wrap">
<div className="min-w-[180px]">
<label className="text-[0.78rem] text-ink-muted">תחום</label>
<Select
value={practiceArea || "_all"}
onValueChange={(v) => setPracticeArea(v === "_all" ? "" : (v as PracticeArea))}
>
<SelectTrigger><SelectValue /></SelectTrigger>
<SelectContent>
<SelectItem value="_all">הכל</SelectItem>
{PRACTICE_AREAS.map((a) => (
<SelectItem key={a.value} value={a.value}>{a.label}</SelectItem>
))}
</SelectContent>
</Select>
</div>
<div className="min-w-[170px]">
<label className="text-[0.78rem] text-ink-muted">קישור לפסק</label>
<Select value={linked} onValueChange={(v) => setLinked(v as LinkedFilter)}>
<SelectTrigger><SelectValue /></SelectTrigger>
<SelectContent>
<SelectItem value="all">הכל</SelectItem>
<SelectItem value="linked">מקושרים</SelectItem>
<SelectItem value="unlinked">לא מקושרים (פער)</SelectItem>
</SelectContent>
</Select>
</div>
<div className="ms-auto">
<DigestUploadDialog />
</div>
</div>
{error ? (
<div className="rounded bg-danger-bg border border-danger/40 px-6 py-5 text-danger text-center">
{error.message}
</div>
) : isLoading ? (
<div className="space-y-3">
{[...Array(3)].map((_, i) => <Skeleton key={i} className="h-32 w-full" />)}
</div>
) : !data?.items.length ? (
<div className="text-center text-ink-muted py-12">
אין יומונים עדיין. העלה יומון &quot;כל יום&quot;, או הרץ
<span className="font-mono mx-1" dir="ltr">scripts/ingest_digests_batch.py</span>.
</div>
) : (
<div className="space-y-3">
<p className="text-[0.78rem] text-ink-muted">{data.count} יומונים</p>
{data.items.map((d) => (
<DigestCard
key={d.id}
digest={d}
actions={
<>
{!d.linked_case_law_id && (
<Button
variant="outline"
size="sm"
onClick={() => onRelink(d.id)}
disabled={relink.isPending}
>
<Link2 className="w-3.5 h-3.5 me-1" />
נסה לקשר
</Button>
)}
<Button
variant="outline"
size="sm"
className="text-danger hover:bg-danger-bg"
onClick={() => onDelete(d.id)}
disabled={del.isPending}
>
<Trash2 className="w-3.5 h-3.5 me-1" />
מחק
</Button>
</>
}
/>
))}
</div>
)}
</div>
);
}