2026-04-16 15:53:46 +02:00
|
|
|
|
#!/usr/bin/env python3
|
|
|
|
|
|
"""
|
2026-04-17 07:47:56 +02:00
|
|
|
|
conversione/validate.py — Validazione qualità Markdown
|
2026-04-16 15:53:46 +02:00
|
|
|
|
|
2026-04-16 16:05:03 +02:00
|
|
|
|
Legge i report.json prodotti da pipeline.py, stampa una tabella di stato
|
2026-04-17 07:47:56 +02:00
|
|
|
|
e assegna un voto (0-100) a ogni documento.
|
2026-04-16 15:53:46 +02:00
|
|
|
|
|
2026-04-16 16:05:03 +02:00
|
|
|
|
90-100 A — ottimo, pronto per il chunker
|
|
|
|
|
|
75-89 B — buono, qualche sezione lunga ma accettabile
|
|
|
|
|
|
60-74 C — accettabile, anomalie minori da verificare
|
|
|
|
|
|
40-59 D — da rivedere, problemi strutturali o residui evidenti
|
|
|
|
|
|
0-39 F — da riprocessare, struttura assente o testo corrotto
|
2026-04-16 15:53:46 +02:00
|
|
|
|
|
|
|
|
|
|
Uso:
|
|
|
|
|
|
python conversione/validate.py # tutti gli stem
|
|
|
|
|
|
python conversione/validate.py analisi1 # stem specifico
|
2026-04-17 07:47:56 +02:00
|
|
|
|
python conversione/validate.py a b c # stem multipli
|
2026-04-17 09:20:15 +02:00
|
|
|
|
python conversione/validate.py --detail analisi1 # mostra dettaglio penalità
|
2026-04-16 15:53:46 +02:00
|
|
|
|
"""
|
|
|
|
|
|
|
2026-04-16 16:05:03 +02:00
|
|
|
|
import argparse
|
2026-04-17 07:47:56 +02:00
|
|
|
|
import json
|
2026-04-16 15:53:46 +02:00
|
|
|
|
import sys
|
|
|
|
|
|
from pathlib import Path
|
|
|
|
|
|
|
|
|
|
|
|
|
2026-04-16 16:05:03 +02:00
|
|
|
|
# ─── Punteggio ───────────────────────────────────────────────────────────────
|
2026-04-16 15:53:46 +02:00
|
|
|
|
|
2026-04-17 07:47:56 +02:00
|
|
|
|
_GRADES = [(90, "A"), (75, "B"), (60, "C"), (40, "D"), (0, "F")]
|
|
|
|
|
|
|
|
|
|
|
|
|
2026-04-17 09:20:15 +02:00
|
|
|
|
def _score(r: dict) -> tuple[int, list[str]]:
|
2026-04-16 16:05:03 +02:00
|
|
|
|
"""
|
2026-04-17 09:20:15 +02:00
|
|
|
|
Calcola un punteggio 0-100 sulla qualità del clean.md ai fini della
|
|
|
|
|
|
suddivisione in chunk e vettorizzazione.
|
|
|
|
|
|
Restituisce (score, lista_penalità_applicate).
|
|
|
|
|
|
|
|
|
|
|
|
Penalità struttura (il chunker non può operare senza header):
|
|
|
|
|
|
struttura assente (livello 0) → −40
|
|
|
|
|
|
struttura piatta (livello 1) → −15
|
|
|
|
|
|
|
|
|
|
|
|
Penalità residui (finiscono nei vettori e degradano il retrieval):
|
|
|
|
|
|
backtick → −2/cad (max −20)
|
|
|
|
|
|
dot-leader → −5/cad (max −10)
|
|
|
|
|
|
URL / watermark → −5/cad (max −15)
|
|
|
|
|
|
immagini residue → −5/cad (max −10)
|
|
|
|
|
|
<br> inline (artefatti tabelle) → −2/cad (max −15)
|
|
|
|
|
|
simboli encoding (!/" residui) → −1/cad (max −10)
|
|
|
|
|
|
formule inline [N.M] → −1/cad (max −8)
|
|
|
|
|
|
|
|
|
|
|
|
Penalità anomalie:
|
|
|
|
|
|
bare headers → −3/cad (max −15)
|
|
|
|
|
|
|
|
|
|
|
|
Non penalizzate (il chunker le normalizza):
|
|
|
|
|
|
sezioni corte, sezioni lunghe, mediana, p25
|
2026-04-16 16:05:03 +02:00
|
|
|
|
"""
|
2026-04-17 09:20:15 +02:00
|
|
|
|
score = 100
|
|
|
|
|
|
detail = []
|
2026-04-16 15:53:46 +02:00
|
|
|
|
structure = r.get("structure", {})
|
2026-04-17 07:47:56 +02:00
|
|
|
|
anomalie = r.get("anomalie", {})
|
|
|
|
|
|
residui = r.get("residui", {})
|
2026-04-16 15:53:46 +02:00
|
|
|
|
|
2026-04-17 07:47:56 +02:00
|
|
|
|
livello = structure.get("livello_struttura", 0)
|
2026-04-16 16:05:03 +02:00
|
|
|
|
|
2026-04-17 09:20:15 +02:00
|
|
|
|
# ── Struttura ─────────────────────────────────────────────────────────
|
2026-04-16 16:05:03 +02:00
|
|
|
|
if livello == 0:
|
|
|
|
|
|
score -= 40
|
2026-04-17 09:20:15 +02:00
|
|
|
|
detail.append("struttura assente −40")
|
2026-04-16 16:05:03 +02:00
|
|
|
|
elif livello == 1:
|
|
|
|
|
|
score -= 15
|
2026-04-17 09:20:15 +02:00
|
|
|
|
detail.append("struttura piatta −15")
|
|
|
|
|
|
|
|
|
|
|
|
# ── Residui ───────────────────────────────────────────────────────────
|
|
|
|
|
|
def _pen(key: str, per_item: int, cap: int, label: str) -> None:
|
|
|
|
|
|
n = residui.get(key, 0)
|
|
|
|
|
|
if n:
|
|
|
|
|
|
p = min(cap, n * per_item)
|
|
|
|
|
|
nonlocal score
|
|
|
|
|
|
score -= p
|
|
|
|
|
|
detail.append(f"{label} ×{n} −{p}")
|
|
|
|
|
|
|
|
|
|
|
|
_pen("backtick", 2, 20, "backtick")
|
|
|
|
|
|
_pen("dotleader", 5, 10, "dot-leader")
|
|
|
|
|
|
_pen("url", 5, 15, "url")
|
|
|
|
|
|
_pen("immagini", 5, 10, "immagini")
|
|
|
|
|
|
_pen("br_inline", 2, 15, "<br> inline")
|
|
|
|
|
|
_pen("simboli_encoding", 1, 10, "simboli encoding")
|
|
|
|
|
|
_pen("formule_inline", 1, 8, "formule inline")
|
2026-04-17 13:44:30 +02:00
|
|
|
|
_pen("footnote_markers", 1, 8, "footnote residui")
|
|
|
|
|
|
_pen("pua_markers", 2, 20, "caratteri PUA font Symbol")
|
2026-04-17 09:20:15 +02:00
|
|
|
|
|
|
|
|
|
|
# ── Anomalie ──────────────────────────────────────────────────────────
|
|
|
|
|
|
n_bare = anomalie.get("bare_headers", 0)
|
|
|
|
|
|
if n_bare:
|
|
|
|
|
|
p = min(15, n_bare * 3)
|
|
|
|
|
|
score -= p
|
|
|
|
|
|
detail.append(f"bare headers ×{n_bare} −{p}")
|
|
|
|
|
|
|
|
|
|
|
|
return max(0, score), detail
|
2026-04-16 16:05:03 +02:00
|
|
|
|
|
2026-04-16 15:53:46 +02:00
|
|
|
|
|
2026-04-16 16:05:03 +02:00
|
|
|
|
def _grade(score: int) -> str:
|
2026-04-17 07:47:56 +02:00
|
|
|
|
return next(g for threshold, g in _GRADES if score >= threshold)
|
2026-04-16 16:05:03 +02:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# ─── Validazione ─────────────────────────────────────────────────────────────
|
2026-04-16 15:53:46 +02:00
|
|
|
|
|
2026-04-17 09:20:15 +02:00
|
|
|
|
def validate(stems: list[str], project_root: Path, detail: bool = False) -> None:
|
2026-04-16 15:53:46 +02:00
|
|
|
|
conv_dir = project_root / "conversione"
|
|
|
|
|
|
|
2026-04-17 07:47:56 +02:00
|
|
|
|
paths = (
|
|
|
|
|
|
[conv_dir / s / "report.json" for s in stems]
|
|
|
|
|
|
if stems
|
|
|
|
|
|
else sorted(conv_dir.glob("*/report.json"))
|
|
|
|
|
|
)
|
2026-04-16 15:53:46 +02:00
|
|
|
|
|
|
|
|
|
|
if not paths:
|
|
|
|
|
|
print("Nessun report.json trovato in conversione/*/")
|
|
|
|
|
|
sys.exit(0)
|
|
|
|
|
|
|
2026-04-17 07:47:56 +02:00
|
|
|
|
rows = [
|
|
|
|
|
|
json.loads(p.read_text(encoding="utf-8")) if p.exists()
|
|
|
|
|
|
else {"stem": p.parent.name, "_missing": True}
|
|
|
|
|
|
for p in paths
|
|
|
|
|
|
]
|
2026-04-16 15:53:46 +02:00
|
|
|
|
|
|
|
|
|
|
# ── Intestazione ─────────────────────────────────────────────────────
|
2026-04-17 07:47:56 +02:00
|
|
|
|
col = max(len(r.get("stem", "stem")) for r in rows) + 2
|
2026-04-16 15:53:46 +02:00
|
|
|
|
header = (
|
2026-04-17 07:47:56 +02:00
|
|
|
|
f"{'stem':<{col}}"
|
2026-04-16 15:53:46 +02:00
|
|
|
|
f"{'h2':>4}{'h3':>5} "
|
2026-04-17 09:20:15 +02:00
|
|
|
|
f"{'strategia':<18}"
|
2026-04-16 15:53:46 +02:00
|
|
|
|
f"{'bare':>5}{'corte':>6}{'lunghe':>7}"
|
2026-04-17 09:20:15 +02:00
|
|
|
|
f"{'btk':>5}{'br':>4}{'enc':>4}{'url':>4}"
|
|
|
|
|
|
f"{'med':>6}"
|
2026-04-17 07:47:56 +02:00
|
|
|
|
f" {'voto':>4} grade"
|
2026-04-16 15:53:46 +02:00
|
|
|
|
)
|
|
|
|
|
|
sep = "─" * len(header)
|
2026-04-17 07:47:56 +02:00
|
|
|
|
print(f"\n{header}\n{sep}")
|
2026-04-16 15:53:46 +02:00
|
|
|
|
|
2026-04-16 16:05:03 +02:00
|
|
|
|
scores = []
|
|
|
|
|
|
|
2026-04-16 15:53:46 +02:00
|
|
|
|
# ── Righe ─────────────────────────────────────────────────────────────
|
|
|
|
|
|
for r in rows:
|
|
|
|
|
|
if r.get("_missing"):
|
2026-04-17 07:47:56 +02:00
|
|
|
|
print(f"{r['stem']:<{col}} (report.json non trovato)")
|
2026-04-16 15:53:46 +02:00
|
|
|
|
continue
|
|
|
|
|
|
|
2026-04-17 09:20:15 +02:00
|
|
|
|
st = r.get("structure", {})
|
|
|
|
|
|
an = r.get("anomalie", {})
|
|
|
|
|
|
res = r.get("residui", {})
|
|
|
|
|
|
dist = r.get("distribution", {})
|
|
|
|
|
|
s, pen = _score(r)
|
2026-04-16 16:05:03 +02:00
|
|
|
|
scores.append(s)
|
2026-04-16 15:53:46 +02:00
|
|
|
|
|
|
|
|
|
|
print(
|
2026-04-17 07:47:56 +02:00
|
|
|
|
f"{r['stem']:<{col}}"
|
|
|
|
|
|
f"{st.get('n_h2', 0):>4}"
|
|
|
|
|
|
f"{st.get('n_h3', 0):>5} "
|
2026-04-17 09:20:15 +02:00
|
|
|
|
f"{st.get('strategia_chunking','?'):<18}"
|
2026-04-17 07:47:56 +02:00
|
|
|
|
f"{an.get('bare_headers', 0):>5}"
|
|
|
|
|
|
f"{an.get('short_sections', 0):>6}"
|
|
|
|
|
|
f"{an.get('long_sections', 0):>7}"
|
2026-04-17 09:20:15 +02:00
|
|
|
|
f"{res.get('backtick', 0):>5}"
|
|
|
|
|
|
f"{res.get('br_inline', 0):>4}"
|
|
|
|
|
|
f"{res.get('simboli_encoding', 0):>4}"
|
|
|
|
|
|
f"{res.get('url', 0):>4}"
|
|
|
|
|
|
f"{dist.get('mediana', 0):>6}"
|
2026-04-17 07:47:56 +02:00
|
|
|
|
f" {s:>4} {_grade(s)}"
|
2026-04-16 15:53:46 +02:00
|
|
|
|
)
|
|
|
|
|
|
|
2026-04-17 09:20:15 +02:00
|
|
|
|
if detail and pen:
|
|
|
|
|
|
for p in pen:
|
|
|
|
|
|
print(f" {'':>{col}} ↳ {p}")
|
|
|
|
|
|
|
2026-04-16 16:05:03 +02:00
|
|
|
|
# ── Riepilogo ─────────────────────────────────────────────────────────
|
2026-04-16 15:53:46 +02:00
|
|
|
|
print(sep)
|
2026-04-16 16:05:03 +02:00
|
|
|
|
if scores:
|
|
|
|
|
|
media = sum(scores) / len(scores)
|
2026-04-17 07:47:56 +02:00
|
|
|
|
print(
|
|
|
|
|
|
f"Documenti: {len(scores)} "
|
|
|
|
|
|
f"Media: {media:.0f}/100 {_grade(int(media))} "
|
|
|
|
|
|
f"(A≥90 B≥75 C≥60 D≥40 F<40)"
|
|
|
|
|
|
)
|
|
|
|
|
|
print(
|
2026-04-17 09:20:15 +02:00
|
|
|
|
"\nColonne: bare=header vuoti corte=sez<150ch lunghe=sez>1500ch "
|
|
|
|
|
|
"btk=backtick br=<br>inline enc=simboli encoding med=mediana chars\n"
|
2026-04-17 07:47:56 +02:00
|
|
|
|
)
|
|
|
|
|
|
|
2026-04-16 15:53:46 +02:00
|
|
|
|
|
2026-04-17 07:47:56 +02:00
|
|
|
|
# ─── Entry point ─────────────────────────────────────────────────────────────
|
2026-04-16 15:53:46 +02:00
|
|
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
2026-04-17 07:47:56 +02:00
|
|
|
|
parser = argparse.ArgumentParser(
|
|
|
|
|
|
description="Valida i report Markdown prodotti da pipeline.py",
|
|
|
|
|
|
epilog="Senza argomenti valida tutti gli stem in conversione/*/",
|
|
|
|
|
|
)
|
|
|
|
|
|
parser.add_argument(
|
|
|
|
|
|
"stems",
|
|
|
|
|
|
nargs="*",
|
|
|
|
|
|
metavar="STEM",
|
|
|
|
|
|
help="stem da validare (es: analisi1). Ometti per tutti.",
|
|
|
|
|
|
)
|
2026-04-17 09:20:15 +02:00
|
|
|
|
parser.add_argument(
|
|
|
|
|
|
"--detail", "-d",
|
|
|
|
|
|
action="store_true",
|
|
|
|
|
|
help="mostra dettaglio penalità per ogni documento",
|
|
|
|
|
|
)
|
2026-04-17 07:47:56 +02:00
|
|
|
|
args = parser.parse_args()
|
2026-04-17 09:20:15 +02:00
|
|
|
|
validate(args.stems, Path(__file__).parent.parent, detail=args.detail)
|