#!/usr/bin/env python3
"""Inventory a FormalSLT source snapshot.

This is a lightweight inventory script, not a proof checker. It reports file,
line, declaration, and showcase-manifest counts from the submitted source tree.
Use Lean and `axiom_audit.py` for the proof/axiom certificate.
"""

from __future__ import annotations

import argparse
import re
from pathlib import Path


DECL_RE = re.compile(r"^\s*(theorem|lemma)\s+")
ATTR_DECL_RE = re.compile(r"^\s*(?:@\[[^\]]+\]\s*)?(theorem|lemma)\s+")


def line_count(path: Path) -> int:
    text = path.read_text(encoding="utf-8")
    return text.count("\n") + (0 if text.endswith("\n") or not text else 1)


def uncommented_lines(path: Path) -> list[str]:
    """Return physical lines outside Lean block comments and line comments.

    This is an inventory helper, not a full Lean parser. It prevents prose such
    as "theorem for finite classes" inside module docstrings from being counted
    as declarations.
    """
    lines: list[str] = []
    in_block = False
    for line in path.read_text(encoding="utf-8").splitlines():
        if in_block:
            if "-/" not in line:
                continue
            line = line.split("-/", 1)[1]
            in_block = False
            if not line.strip():
                continue

        if line.lstrip().startswith("--"):
            continue

        while "/-" in line:
            before, after = line.split("/-", 1)
            if "-/" in after:
                after = after.split("-/", 1)[1]
                line = before + after
                continue
            line = before
            in_block = True
            break

        if line.strip():
            lines.append(line)
    return lines


def count_decls(files: list[Path], regex: re.Pattern[str]) -> dict[str, int]:
    counts = {"theorem": 0, "lemma": 0}
    for path in files:
        for line in uncommented_lines(path):
            match = regex.match(line)
            if match:
                counts[match.group(1)] += 1
    return counts


def main() -> None:
    parser = argparse.ArgumentParser()
    parser.add_argument("root", type=Path, help="Path to expanded formalslt root")
    args = parser.parse_args()

    root = args.root
    formal_dir = root / "FormalSLT"
    root_file = root / "FormalSLT.lean"
    examples_dir = root / "examples"
    showcase = examples_dir / "CheckShowcaseTheorems.lean"

    if not formal_dir.is_dir() or not root_file.exists():
        raise SystemExit(f"{root} does not look like a FormalSLT source root")

    library_modules = sorted(formal_dir.rglob("*.lean"))
    source_files = library_modules + [root_file]
    example_files = sorted(examples_dir.rglob("*.lean")) if examples_dir.is_dir() else []

    strict_decls = count_decls(source_files, DECL_RE)
    attributed_decls = count_decls(source_files, ATTR_DECL_RE)

    showcase_checks = 0
    showcase_prints = 0
    if showcase.exists():
        for line in showcase.read_text(encoding="utf-8").splitlines():
            stripped = line.strip()
            showcase_checks += stripped.startswith("#check ")
            showcase_prints += stripped.startswith("#print axioms ")

    print("# FormalSLT Snapshot Inventory")
    print("")
    print(f"- Library modules under FormalSLT/: {len(library_modules)}")
    print(f"- Source files including FormalSLT.lean: {len(source_files)}")
    print(f"- Source lines including FormalSLT.lean: {sum(line_count(p) for p in source_files)}")
    print(f"- Example files: {len(example_files)}")
    print(f"- Example lines: {sum(line_count(p) for p in example_files)}")
    print(
        "- Comment-skipped line-leading declarations: "
        f"{strict_decls['theorem']} theorem + {strict_decls['lemma']} lemma = "
        f"{sum(strict_decls.values())}"
    )
    print(
        "- Including same-line attributed declarations: "
        f"{attributed_decls['theorem']} theorem + {attributed_decls['lemma']} lemma = "
        f"{sum(attributed_decls.values())}"
    )
    print(f"- Showcase #check commands: {showcase_checks}")
    print(f"- Showcase #print axioms commands: {showcase_prints}")
    print("")
    print(
        "Use these counts only if this exact source snapshot is the artifact submitted "
        "with the paper. If the manuscript keeps older 19,521-line / 375-theorem "
        "numbers, include the exact older snapshot or theorem inventory that produces them."
    )


if __name__ == "__main__":
    main()
