feat(learning): מסלול נקי להעלאת החלטה סופית + פאנל-סגנון דו-סוכני (DeepSeek+Gemini)

מוסיף מסלול ייעודי לקליטת ההחלטה החתומה של היו"ר, ומפעיל אותו דרך שני
שלבים אוטומטיים מדורגים עם פאנלי-סוכנים (אוטו-אישור + אסקלציה ליו"ר).

Backend (web/):
- POST /api/cases/{case}/final/upload — קליטת final חיצוני: שמירה קנונית
  (סופי-{case}.docx + עותק קורפוס-סגנון תחת case_number מלא כדי שבל"מ לא
  יתנגש עם ערר באותו מספר), פתיחת draft_final_pairs (final_received). לא נוגע
  ב-active_draft ולא מריץ retrofit (נבדל מ-exports/upload ו-mark-final → לא G2).
- POST .../final/run-learning + .../final/run-halacha — שלבים מדורגים שמעירים
  worker מקומי (claude/DeepSeek/Gemini מקומיים בלבד) דרך הרחבת
  wake_curator_for_final עם param task=learning|halacha.

פאנל-סגנון חדש (scripts/style_lesson_panel.py): שני שופטים (DeepSeek+Gemini)
על-גבי דיסטילציית-ה-Opus; הסכמה 2/2-keep → decision_lesson
(source=panel:deepseek+gemini); substance מדולג (INV-LRN5); הפיך + גיבוי CSV.
פאנל-הלכות: docstring/SCRIPTS.md עודכנו (--apply מחווט).

Frontend (web-ui/): כפתור "העלאת החלטה סופית של היו"ר" + שני כפתורים מדורגים
"הרץ למידת-קול"/"הרץ אימות-הלכות" ב-drafts-panel; כל התוויות בעברית
(badge מקור-לקח: "פאנל: דיפסיק+גמיני", "הרמס (סקירה)"...).

Spec: docs/spec/07-learning.md §0.6. Invariants: INV-LRN1/LRN4/LRN5, G10
(שער-יו"ר ידני להטמעה ל-SKILL.md/lessons.md — הפאנלים יוצרים הצעות בלבד);
G2 (מסלול-סופי הוא יכולת חסרה, לא מסלול-מקביל).

Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
This commit is contained in:
2026-06-08 09:03:26 +00:00
parent 6647aa92e6
commit f79c46a352
10 changed files with 726 additions and 25 deletions

View File

@@ -181,6 +181,73 @@ export function useDeleteDraft(caseNumber: string) {
});
}
/* ── Chair's signed final decision — clean upload path + staged pipeline ── */
export type FinalUploadResult = {
final_filename: string;
training_copy: string;
pair_id: string | null;
draft_words: number;
final_words: number;
status: string;
};
export type FinalTaskResult = {
status: string;
sub_issue_id?: string;
curator_id?: string;
reason?: string;
error?: string;
};
/** Upload Dafna's signed final decision (distinct from "upload a revised draft"). */
export function useUploadFinalDecision(caseNumber: string) {
const qc = useQueryClient();
return useMutation({
mutationFn: async (file: File): Promise<FinalUploadResult> => {
const form = new FormData();
form.append("file", file);
const res = await fetch(`/api/cases/${caseNumber}/final/upload`, {
method: "POST",
body: form,
});
if (!res.ok) {
const err = await res
.json()
.catch(() => ({ detail: "שגיאה בהעלאת ההחלטה הסופית" }));
throw new Error(err.detail ?? "שגיאה בהעלאת ההחלטה הסופית");
}
return res.json() as Promise<FinalUploadResult>;
},
onSuccess: () => {
qc.invalidateQueries({ queryKey: exportsKeys.list(caseNumber) });
qc.invalidateQueries({ queryKey: casesKeys.detail(caseNumber) });
},
});
}
/** Staged step 1 — voice learning (Opus distillation + DeepSeek+Gemini style panel). */
export function useRunFinalLearning(caseNumber: string) {
return useMutation({
mutationFn: () =>
apiRequest<FinalTaskResult>(
`/api/cases/${caseNumber}/final/run-learning`,
{ method: "POST" },
),
});
}
/** Staged step 2 — halacha validation (cited-halacha extraction + 3-judge panel). */
export function useRunFinalHalacha(caseNumber: string) {
return useMutation({
mutationFn: () =>
apiRequest<FinalTaskResult>(
`/api/cases/${caseNumber}/final/run-halacha`,
{ method: "POST" },
),
});
}
export function useMarkFinal(caseNumber: string) {
const qc = useQueryClient();
return useMutation({