All checks were successful
G12 Leak-Guard / leak-guard (pull_request) Successful in 13s
גל-1 מבקלוג #127 (docs/ia-audit-redesign.md §4) — תיקון מקומי, ללא הגירת-IA. מקיים G2 בשכבת-ה-UI דרך INV-IA1/IA2/IA5/IA6 (docs/spec/X17). א) פערי-סנכרון (INV-IA2 — mutation מבטל כל קורא): - CAS-1/2: העלאת-DOCX/export מבטלים ['decision-blocks'] (מחוון source_of_truth) - APR-1/4: פתרון/יצירת-הערה מבטלים ['chair','pending'] (תיבה+תג-סרגל) - APR-5/ADM-2: אישור/batch הלכות מבטלים ['chair','pending']+['operations'] - APR-6/ADM-3: create/update/delete/upload פסיקה-חסרה מבטלים שניהם - LRN-6: ComparePanel גוזר בחירה מהקורפוס המרוענן (אין POST ל-id מחוק → 404) - LRN-8: מחיקת-קורפוס מבטלת רשימת-צ'אטים (chat שהתייתם לא נשאר עם קישור-קורפוס תקוע) - LRN-10/MET-1/MET-8: promote מבטל גם lessons וגם methodology (LessonsTab+/methodology) ב) נתונים-שגויים (INV-IA5 — סטטוס מגובה-צרכן): - LRN-4: KPI "דפוסי סגנון" — הוסר היחס-השקרי "מתוך total_patterns" (שאילתות עצמאיות) - LRN-5: findings_applied (דגל אינפורמטיבי-בלבד) → findings_approved (שער INV-LRN1 האמיתי) - ADM-1: halacha_backlog שהוחזר ונזרק → מרונדר ב-/diagnostics, מצביע ל-/approvals (INV-IA1) - ADM-6: מוני-סוכנים מסמנים "חלקי+" כשחברת-Paperclip לא נטענה - APR-3: מכוסה ע"י APR-1 (count+sample מאותה שאילתה; הבעיה היתה staleness-cache) - MET-6: עורך-צ'קליסטים מציג איזה case בוחר כל צ'קליסט (explainer-תחולה) - ADM-5: ערך-Container מסומן "ממתין ל-redeploy" כש-Coolify≠Container ג) מתים/jargon: - PRE-2: הוסר GET /api/precedent-library/queue/pending (אפס צרכני-frontend) - PRE-3/5: AuthorityBadge (binding/persuasive) מרונדר גם בחיפוש, לא רק בתור-הביקורת - MET-5: הוסר ז'רגון T7/T15 מטקסט-העזר ב-/methodology (INV-IA6) Invariants: מקיים INV-IA1/IA2/IA5/IA6 (X17), G2 (מקור-אמת יחיד בשכבת-UI), G10 (לא הוסר שום שער-אנושי — רק סנכרון/נתון/קוד-מת). שומר INV-LRN1. בדיקות: py_compile web/app.py ✓ · tsc --noEmit ✓ · eslint ✓ (לבד מ-learning-panel:109 unescaped-quote — קיים-מראש ב-main, מחוץ לסט-הממצאים). next build נכשל רק בגלל symlink node_modules ב-worktree (Turbopack) — ה-build ב-Docker/CI תקין. Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
199 lines
6.8 KiB
TypeScript
199 lines
6.8 KiB
TypeScript
"use client";
|
||
|
||
import { Card, CardContent } from "@/components/ui/card";
|
||
import { Skeleton } from "@/components/ui/skeleton";
|
||
import { SubjectDonut } from "@/components/training/subject-donut";
|
||
import { useStyleReport } from "@/lib/api/training";
|
||
|
||
function KPICard({
|
||
label,
|
||
value,
|
||
caption,
|
||
}: {
|
||
label: string;
|
||
value: string;
|
||
caption?: string;
|
||
}) {
|
||
return (
|
||
<Card className="bg-surface border-rule shadow-sm">
|
||
<CardContent className="px-5 py-4 flex flex-col gap-0.5">
|
||
<span className="text-[0.72rem] uppercase tracking-[0.08em] text-ink-muted">
|
||
{label}
|
||
</span>
|
||
<span className="font-display text-[2rem] font-black leading-none text-navy">
|
||
{value}
|
||
</span>
|
||
{caption && (
|
||
<span className="text-[0.78rem] text-ink-muted mt-1">{caption}</span>
|
||
)}
|
||
</CardContent>
|
||
</Card>
|
||
);
|
||
}
|
||
|
||
export function StyleReportPanel() {
|
||
const { data, isPending, error } = useStyleReport();
|
||
|
||
if (error) {
|
||
return (
|
||
<Card className="bg-danger-bg border-danger/40">
|
||
<CardContent className="px-6 py-5 text-center text-danger">
|
||
{error.message}
|
||
</CardContent>
|
||
</Card>
|
||
);
|
||
}
|
||
|
||
if (isPending || !data) {
|
||
return (
|
||
<div className="space-y-4">
|
||
<Skeleton className="h-24 w-full" />
|
||
<Skeleton className="h-40 w-full" />
|
||
</div>
|
||
);
|
||
}
|
||
|
||
const c = data.corpus;
|
||
const dateRange =
|
||
c.date_range[0] && c.date_range[1]
|
||
? `${c.date_range[0]} – ${c.date_range[1]}`
|
||
: undefined;
|
||
const total = c.decision_count;
|
||
const totalSubjects = c.subject_distribution.reduce((a, b) => a + b.count, 0);
|
||
|
||
return (
|
||
<div className="space-y-6">
|
||
{/* Headline */}
|
||
<Card className="bg-gold-wash border-gold/40 shadow-sm">
|
||
<CardContent className="px-6 py-4">
|
||
<p className="font-display text-gold-deep text-lg font-semibold leading-snug">
|
||
★ {c.headline}
|
||
</p>
|
||
</CardContent>
|
||
</Card>
|
||
|
||
{/* KPIs */}
|
||
<div className="grid gap-4 grid-cols-2 lg:grid-cols-4">
|
||
<KPICard label="החלטות בקורפוס" value={String(c.decision_count)} />
|
||
<KPICard
|
||
label="סך תווים"
|
||
value={`${(c.total_chars / 1000).toFixed(0)}K`}
|
||
/>
|
||
<KPICard
|
||
label="ממוצע להחלטה"
|
||
value={`${(c.avg_chars / 1000).toFixed(1)}K`}
|
||
/>
|
||
{/* LRN-4 (INV-IA5): items.length (style_patterns freq>0) and
|
||
contribution.total_patterns are independent queries — the old
|
||
"X מתוך Y" framed a false subset. Show the surfaced count alone. */}
|
||
<KPICard
|
||
label="דפוסי סגנון"
|
||
value={String(data.signature_phrases.items.length)}
|
||
caption="דפוסים חוזרים שזוהו"
|
||
/>
|
||
</div>
|
||
|
||
{/* Subjects + anatomy */}
|
||
<div className="grid gap-6 lg:grid-cols-2">
|
||
<Card className="bg-surface border-rule shadow-sm">
|
||
<CardContent className="px-6 py-5">
|
||
<h3 className="text-navy text-lg mb-4">פיזור נושאים</h3>
|
||
<SubjectDonut
|
||
segments={c.subject_distribution}
|
||
total={totalSubjects}
|
||
/>
|
||
{dateRange && (
|
||
<p className="text-[0.72rem] text-ink-muted mt-4">
|
||
טווח תאריכים: {dateRange}
|
||
</p>
|
||
)}
|
||
</CardContent>
|
||
</Card>
|
||
|
||
<Card className="bg-surface border-rule shadow-sm">
|
||
<CardContent className="px-6 py-5">
|
||
<h3 className="text-navy text-lg mb-1">אנטומיה של החלטה ממוצעת</h3>
|
||
{data.anatomy.headline && (
|
||
<p className="text-[0.78rem] text-gold-deep mb-4">
|
||
{data.anatomy.headline}
|
||
</p>
|
||
)}
|
||
{data.anatomy.sections.length === 0 ? (
|
||
<p className="text-ink-muted text-sm">אין נתונים על מבנה</p>
|
||
) : (
|
||
<ul className="space-y-2.5">
|
||
{data.anatomy.sections.map((s) => {
|
||
const pct = Math.round(s.pct * 100);
|
||
return (
|
||
<li key={s.type} className="space-y-1">
|
||
<div className="flex items-center justify-between text-[0.78rem]">
|
||
<span className="text-ink-soft font-medium">
|
||
{s.label}
|
||
</span>
|
||
<span className="text-ink-muted tabular-nums">
|
||
{pct}% · {s.avg_chars.toLocaleString()} תווים
|
||
</span>
|
||
</div>
|
||
<div className="h-2 rounded bg-rule-soft overflow-hidden">
|
||
<div
|
||
className="h-full bg-gradient-to-l from-gold to-gold-deep"
|
||
style={{ width: `${pct}%` }}
|
||
/>
|
||
</div>
|
||
</li>
|
||
);
|
||
})}
|
||
</ul>
|
||
)}
|
||
</CardContent>
|
||
</Card>
|
||
</div>
|
||
|
||
{/* Signature phrases */}
|
||
<Card className="bg-surface border-rule shadow-sm">
|
||
<CardContent className="px-6 py-5">
|
||
<h3 className="text-navy text-lg mb-1">ביטויי חתימה</h3>
|
||
{data.signature_phrases.headline && (
|
||
<p className="text-[0.78rem] text-gold-deep mb-4">
|
||
{data.signature_phrases.headline}
|
||
</p>
|
||
)}
|
||
{data.signature_phrases.items.length === 0 ? (
|
||
<p className="text-ink-muted text-sm">אין ביטויים שחולצו עדיין</p>
|
||
) : (
|
||
<ol className="space-y-2">
|
||
{data.signature_phrases.items.slice(0, 12).map((p, i) => (
|
||
<li
|
||
key={`${p.type}-${i}`}
|
||
className="flex items-start gap-3 rounded border border-rule bg-parchment/40 px-3 py-2"
|
||
>
|
||
<span className="text-[0.7rem] text-ink-muted tabular-nums shrink-0 mt-0.5">
|
||
#{i + 1}
|
||
</span>
|
||
<div className="flex-1 min-w-0">
|
||
<p className="text-ink leading-relaxed text-sm">{p.text}</p>
|
||
{p.context && (
|
||
<p className="text-[0.7rem] text-ink-muted mt-0.5">
|
||
{p.context}
|
||
</p>
|
||
)}
|
||
</div>
|
||
<span
|
||
className="
|
||
shrink-0 text-[0.72rem] rounded-full
|
||
bg-gold-wash text-gold-deep border border-gold/40
|
||
px-2 py-0.5 tabular-nums
|
||
"
|
||
>
|
||
×{p.frequency}
|
||
</span>
|
||
</li>
|
||
))}
|
||
</ol>
|
||
)}
|
||
</CardContent>
|
||
</Card>
|
||
</div>
|
||
);
|
||
}
|