#!/usr/bin/env python3
"""Fail CI unless showcase theorem axiom sets are allowed.

Typical CI usage from the FormalSLT source root:

    python scripts/axiom_audit.py \
      --manifest examples/CheckShowcaseTheorems.lean \
      --allow propext Classical.choice Quot.sound

The script can also audit a saved Lean log:

    python scripts/axiom_audit.py \
      --manifest examples/CheckShowcaseTheorems.lean \
      --log SHOWCASE_AXIOMS.txt
"""

from __future__ import annotations

import argparse
import re
import subprocess
import sys
from pathlib import Path


PRINT_RE = re.compile(r"^\s*#print\s+axioms\s+(.+?)\s*$")
AXIOMS_RE = re.compile(r"[`']([^`']+)[`']\s+depends on axioms:\s*\[(.*?)\]", re.DOTALL)
NO_AXIOMS_RE = re.compile(r"[`']([^`']+)[`']\s+does not depend on any axioms")


def manifest_names(path: Path) -> list[str]:
    names: list[str] = []
    for line in path.read_text(encoding="utf-8").splitlines():
        match = PRINT_RE.match(line)
        if match:
            names.append(match.group(1).strip())
    if not names:
        raise SystemExit(f"no #print axioms commands found in {path}")
    return names


def run_lean(manifest: Path, root: Path) -> str:
    command = ["lake", "env", "lean", str(manifest)]
    try:
        proc = subprocess.run(
            command,
            cwd=root,
            text=True,
            stdout=subprocess.PIPE,
            stderr=subprocess.STDOUT,
            check=False,
        )
    except FileNotFoundError as exc:
        raise SystemExit("lake not found on PATH; install elan/lake or pass --log") from exc
    if proc.returncode != 0:
        sys.stderr.write(proc.stdout)
        raise SystemExit(f"{' '.join(command)} failed with exit code {proc.returncode}")
    return proc.stdout


def parse_axiom_output(output: str) -> dict[str, set[str]]:
    results: dict[str, set[str]] = {}
    for match in AXIOMS_RE.finditer(output):
        name = match.group(1).strip()
        axioms = {part.strip() for part in match.group(2).split(",") if part.strip()}
        results[name] = axioms
    for match in NO_AXIOMS_RE.finditer(output):
        results[match.group(1).strip()] = set()
    return results


def main() -> None:
    parser = argparse.ArgumentParser()
    parser.add_argument("--root", type=Path, default=Path("."))
    parser.add_argument("--manifest", type=Path, required=True)
    parser.add_argument("--allow", nargs="+", default=["propext", "Classical.choice", "Quot.sound"])
    parser.add_argument("--log", type=Path, help="Saved output from lake env lean manifest")
    parser.add_argument("--write-log", type=Path, help="Write Lean output to this file")
    args = parser.parse_args()

    root = args.root.resolve()
    manifest = args.manifest if args.manifest.is_absolute() else root / args.manifest
    names = manifest_names(manifest)

    if args.log:
        output = args.log.read_text(encoding="utf-8")
    else:
        output = run_lean(manifest, root)
    if args.write_log:
        args.write_log.write_text(output, encoding="utf-8")

    allowed = set(args.allow)
    results = parse_axiom_output(output)
    missing = [name for name in names if name not in results]
    violations = {
        name: sorted(axioms.difference(allowed))
        for name, axioms in results.items()
        if axioms.difference(allowed)
    }

    if missing:
        print("Missing #print axioms output for:")
        for name in missing:
            print(f"  - {name}")
    if violations:
        print("Disallowed axioms found:")
        for name, bad_axioms in sorted(violations.items()):
            print(f"  - {name}: {', '.join(bad_axioms)}")
    if missing or violations:
        raise SystemExit(1)

    print(f"Checked {len(names)} showcase theorem axiom traces.")
    print(f"Allowed axiom set: {', '.join(sorted(allowed))}")


if __name__ == "__main__":
    main()
