from __future__ import annotations

import argparse, datetime, json, subprocess, sys

from pathlib import Path

def to_csv(x):
    if isinstance(x, (list, tuple)):
        return ",".join(str(v) for v in x)
    return str(x)




def newest_dir(root: Path, contains: str) -> Path:

    cands = [p for p in root.iterdir() if p.is_dir() and contains in p.name]

    if not cands:

        cands = [p for p in root.iterdir() if p.is_dir()]

    cands.sort(key=lambda p: p.stat().st_mtime, reverse=True)

    return cands[0]



def stage_artifacts(run_dir: Path, out_art: Path) -> None:

    out_art.mkdir(parents=True, exist_ok=True)

    pa = run_dir / "paper_artifacts"

    if pa.is_dir():

        for f in pa.iterdir():

            if f.is_file():

                (out_art / f.name).write_bytes(f.read_bytes())

    for ext in (".png",".csv",".tex",".json"):

        for f in run_dir.glob(f"*{ext}"):

            if f.is_file():

                (out_art / f.name).write_bytes(f.read_bytes())



def main() -> None:

    ap = argparse.ArgumentParser()

    ap.add_argument("--device", default="cpu")

    ap.add_argument("--config", default="paper_configs/alignment.json")

    args = ap.parse_args()



    repo = Path(__file__).resolve().parent

    cfg = json.loads((repo / args.config).read_text())



    for tag, c in cfg["datasets"].items():

        out_root = repo / "results" / "paper_runs" / f"{tag}_covariance_alignment_nocoral"

        stamp = datetime.datetime.now().strftime("%Y%m%d_%H%M%S")

        raw_root = out_root / "raw_runs" / f"paper_{stamp}"

        raw_root.mkdir(parents=True, exist_ok=True)



        cmd = [

            sys.executable, "-u", "runners/run_experiments_icml.py",

            "--dataset", c["dataset"],

            "--seeds", to_csv(c["seeds"]),

            "--out", str(raw_root),

            "--K", str(c["K"]),

            "--rank", str(c["rank"]),

            "--methods", to_csv(c["methods"]),

            "--device", args.device,

            "--dtype", str(c.get("dtype","float64")),

            "--itspace-lambda", str(c["itspace_lambda"]),

            "--itspace-polar", str(c["itspace_polar"]),

            "--itspace-polar-ns-iters", str(c["itspace_polar_ns_iters"]),

            "--bw-gd-eta", str(c["bw_gd_eta"]),

            "--bw-gd-max-backtracks", str(c["bw_gd_max_backtracks"]),

            "--bw-gd-bt-shrink", str(c["bw_gd_bt_shrink"]),

            "--sinkhorn-eps", str(c["sinkhorn_eps"]),

            "--sinkhorn-n", str(c["sinkhorn_n"]),

            "--sinkhorn-iters-per-step", str(c["sinkhorn_iters_per_step"]),

        ]

        subprocess.run(cmd, cwd=str(repo), check=True)



        run_dir = newest_dir(raw_root, "results_icml")



        subprocess.run([sys.executable, "-u", "runners/icml_finalize_threshold_table.py",

                        "--run-dir", str(run_dir)], cwd=str(repo), check=True)

        subprocess.run([sys.executable, "-u", "runners/icml_finalize_v2.py",

                        "--run-dir", str(run_dir),

                        "--seed-plot", "0",

                        "--taus", "1e-6,1e-9",

                        "--focus-tau", "1e-6",

                        "--xmult-update", "5.0",

                        "--xmult-total", "8.0",

                        "--max-decades", "10"], cwd=str(repo), check=True)



        stage_artifacts(run_dir, out_root / "paper_artifacts")

        (out_root / "last_run_dir.txt").write_text(str(run_dir) + "\n")



if __name__ == "__main__":

    main()
