from __future__ import annotations
import json, csv
from pathlib import Path

def _collect_metrics(task_dir: Path):
    rows = []
    for case_dir in sorted(task_dir.glob("*")):
        if not case_dir.is_dir(): continue
        m = case_dir / "metrics.json"
        if m.exists():
            try:
                obj = json.loads(m.read_text(encoding="utf-8"))
                rows.append(obj)
            except Exception:
                pass
    return rows

def write_csvs(run_dir: Path):
    out_summary = []
    for task in ("analysis","repair","refactor","transform"):
        tdir = run_dir/"raw"/task
        if not tdir.exists(): continue
        rows = _collect_metrics(tdir)
        if not rows: continue
        # stable flat keys
        def flatten(d, prefix=""):
            out={}
            for k,v in (d or {}).items():
                key = f"{prefix}{k}" if not prefix else f"{prefix}.{k}"
                if isinstance(v, dict):
                    out.update(flatten(v, key))
                else:
                    out[key]=v
            return out
        flat_rows = [flatten(r) for r in rows]
        # header
        hdr=set()
        for r in flat_rows: hdr.update(r.keys())
        hdr = ["id"] + sorted([h for h in hdr if h!="id"])
        outp = run_dir / f"{task}.csv"
        with open(outp, "w", newline="", encoding="utf-8") as f:
            w = csv.DictWriter(f, fieldnames=hdr)
            w.writeheader()
            for r in flat_rows:
                w.writerow(r)
        out_summary.append({"task": task, "n_cases": len(rows), "csv": str(outp)})
    # write overview
    import json
    (run_dir/"summary_overview.json").write_text(json.dumps(out_summary, indent=2), encoding="utf-8")
