from __future__ import annotations
import os, json, hashlib, tempfile, shutil, time
from pathlib import Path
from typing import Optional

def atomic_write(path: Path | str, data: str | bytes, *, encoding="utf-8"):
    path = Path(path)
    path.parent.mkdir(parents=True, exist_ok=True)
    if isinstance(data, str):
        b = data.encode(encoding)
    else:
        b = data
    tmp = path.with_suffix(path.suffix + f".tmp.{int(time.time()*1000)}")
    with open(tmp, "wb") as f:
        f.write(b)
        f.flush()
        os.fsync(f.fileno())
    os.replace(tmp, path)

def _sha256_of(path: Path) -> str:
    h = hashlib.sha256()
    with open(path, "rb") as f:
        for chunk in iter(lambda: f.read(8192), b""):
            h.update(chunk)
    return h.hexdigest()

def append_manifest(run_dir: Path, rel_path: Path, *, role:str, case_id:str, shot:int | None = None):
    """Append one line to manifest.jsonl located at run_dir/manifest.jsonl"""
    abs_path = (run_dir / rel_path).resolve()
    rec = {
        "ts": int(time.time()),
        "path": str(rel_path).replace("\\","/"),
        "size": abs_path.stat().st_size if abs_path.exists() else None,
        "sha256": _sha256_of(abs_path) if abs_path.exists() else None,
        "role": role,
        "case_id": case_id,
        "shot": shot,
    }
    mf = run_dir / "manifest.jsonl"
    mf.parent.mkdir(parents=True, exist_ok=True)
    with open(mf, "a", encoding="utf-8") as f:
        f.write(json.dumps(rec, ensure_ascii=False) + "\n")

def write_provenance(run_dir: Path, *, cli_argv:list[str], env:dict, toolchain:dict):
    prov = {
        "argv": cli_argv,
        "env": {k:env[k] for k in sorted(env) if k.startswith("CIRBENCH_")},
        "toolchain": toolchain,
        "ts": int(time.time())
    }
    atomic_write(run_dir/"provenance.json", json.dumps(prov, indent=2, ensure_ascii=False))
