from __future__ import annotations

import re
from dataclasses import dataclass
from pathlib import Path

from .config import LEAN_ROOT


_CHAPTER_DIR_RE = re.compile(r"^Chap\d{2}$")
_CHAPTER_FILE_RE = re.compile(r"^Chap\d{2}\.lean$")
_SECTION_AGG_FILE_RE = re.compile(r"^section\d{2}\.lean$")


@dataclass(frozen=True, slots=True)
class BookUpdateResult:
    book_rel: Path
    updated: bool
    added_imports: list[str]


@dataclass(frozen=True, slots=True)
class ChapterUpdateResult:
    chapter_rel: Path
    updated: bool
    added_imports: list[str]


def _module_name_from_rel_path(rel_lean: Path) -> str:
    rel_no_suffix = rel_lean.with_suffix("")
    return ".".join(rel_no_suffix.parts)


def _is_section_part_file(rel_lean: Path) -> bool:
    # <project>/Chapters/ChapXX/sectionYY_partK.lean
    if rel_lean.suffix != ".lean":
        return False
    parts = rel_lean.parts
    if len(parts) != 4:
        return False
    _, mid, chap, fname = parts
    return (
        mid == "Chapters"
        and bool(_CHAPTER_DIR_RE.match(chap))
        and fname.startswith("section")
        and "_part" in Path(fname).stem
    )


def _section_aggregate_for_part(rel_part: Path) -> Path:
    # section01_part3.lean -> section01.lean
    stem = rel_part.stem
    base = stem.split("_part", 1)[0]
    return rel_part.with_name(base + ".lean")


def _is_bench_part_file(rel_lean: Path) -> bool:
    # Question_bench/<bank>/<id>_partXX.lean
    if rel_lean.suffix != ".lean":
        return False
    parts = rel_lean.parts
    if len(parts) < 3:
        return False
    if parts[0] != "Question_bench":
        return False
    return "_part" in rel_lean.stem


def compile_entry_for(rel_lean: Path) -> Path:
    """
    Map an edited file to the "compilation entry" file:
    - Section part files compile via their aggregate `sectionYY.lean`.
    - Bench part files compile via their aggregate `<id>.lean`.
    - Otherwise compile the file itself.
    """
    rel_lean = Path(rel_lean)
    if _is_section_part_file(rel_lean):
        return _section_aggregate_for_part(rel_lean)
    if _is_bench_part_file(rel_lean):
        base = rel_lean.stem.split("_part", 1)[0]
        return rel_lean.with_name(base + ".lean")
    return rel_lean


def ensure_section_aggregate_exists(rel_any: Path) -> Path:
    """
    If `rel_any` is a `sectionYY_part*.lean`, ensure `sectionYY.lean` exists and imports all parts.
    Returns the aggregate relative path (which may equal `rel_any` if not a part file).
    """
    rel_any = Path(rel_any)
    if not _is_section_part_file(rel_any):
        return rel_any

    rel_agg = _section_aggregate_for_part(rel_any)
    abs_agg = LEAN_ROOT / rel_agg
    abs_parts_dir = abs_agg.parent
    prefix = rel_agg.stem + "_part"
    part_paths = list(abs_parts_dir.glob(prefix + "*.lean"))

    def _part_index(p: Path) -> int:
        m = re.search(r"_part(\d+)$", p.stem)
        if not m:
            return 10**9
        try:
            return int(m.group(1))
        except ValueError:
            return 10**9

    part_paths.sort(key=lambda p: (_part_index(p), p.name))
    imports: list[str] = []
    for p in part_paths:
        rel_p = p.relative_to(LEAN_ROOT)
        imports.append("import " + _module_name_from_rel_path(rel_p))

    abs_agg.parent.mkdir(parents=True, exist_ok=True)
    abs_agg.write_text("\n".join(imports) + ("\n" if imports else ""), encoding="utf-8")
    return rel_agg


def _chapter_aggregate_for_section(section_aggregate_rel: Path) -> Path | None:
    """
    Map `<project>/Chapters/ChapXX/sectionYY.lean` to the chapter aggregation module
    `<project>/Chapters/ChapXX.lean`.
    """
    section_aggregate_rel = Path(section_aggregate_rel)
    if section_aggregate_rel.suffix != ".lean":
        return None
    parts = section_aggregate_rel.parts
    if len(parts) != 4:
        return None
    project, mid, chap_dir, fname = parts
    if mid != "Chapters":
        return None
    if not _CHAPTER_DIR_RE.match(chap_dir):
        return None
    if not _SECTION_AGG_FILE_RE.match(fname):
        return None
    return Path(project) / "Chapters" / f"{chap_dir}.lean"


def ensure_chapter_imports(
    *,
    project: str,
    section_aggregate_rel: Path,
) -> ChapterUpdateResult:
    """
    Ensure `<project>/Chapters/ChapXX.lean` imports the given *aggregate* `sectionYY.lean`.

    We maintain an auto-managed block to avoid clobbering manual edits.
    """
    project = (project or "").strip()
    if not project:
        raise ValueError("project must be non-empty")

    section_aggregate_rel = Path(section_aggregate_rel)
    if section_aggregate_rel.suffix != ".lean":
        raise ValueError(f"expected a .lean file, got: {section_aggregate_rel}")
    if section_aggregate_rel.parts[:1] != (project,):
        raise ValueError(
            f"expected section file under project {project}, got: {section_aggregate_rel}"
        )

    chapter_rel = _chapter_aggregate_for_section(section_aggregate_rel)
    if chapter_rel is None:
        raise ValueError(
            f"expected an aggregate section path like {project}/Chapters/ChapXX/sectionYY.lean, got: {section_aggregate_rel}"
        )

    chapter_abs = LEAN_ROOT / chapter_rel
    chapter_abs.parent.mkdir(parents=True, exist_ok=True)
    if not chapter_abs.exists():
        _ensure_chapter_file_exists(project=project, chapter_abs=chapter_abs, chapter_rel=chapter_rel)

    begin = "-- BEGIN AUTO-IMPORTS (managed by orchestrator)"
    end = "-- END AUTO-IMPORTS"

    import_line = "import " + _module_name_from_rel_path(section_aggregate_rel)

    text = chapter_abs.read_text(encoding="utf-8")
    lines = text.splitlines()

    def _find_marker(lines: list[str], marker: str) -> int | None:
        for i, ln in enumerate(lines):
            if ln.strip() == marker:
                return i
        return None

    def _extract_block(lines: list[str]) -> tuple[list[str], list[str], list[str]]:
        """
        Return (head_with_begin, block_lines, tail_with_end_and_after).

        Robustness:
        - Accept markers with leading/trailing whitespace.
        - If multiple managed blocks exist (from prior buggy runs), collapse them into one
          by taking the first BEGIN and the last END after it.
        """
        b = _find_marker(lines, begin)
        if b is None:
            raise ValueError
        ends = [i for i, ln in enumerate(lines) if ln.strip() == end and i > b]
        if not ends:
            raise ValueError
        e = ends[-1]
        head = lines[:b] + [begin]
        block = lines[b + 1 : e]
        tail = [end] + lines[e + 1 :]
        return head, block, tail

    try:
        head, block, tail = _extract_block(lines)
    except ValueError:
        insert_at = 0
        for i, line in enumerate(lines):
            if line.startswith("import "):
                insert_at = i + 1
                continue
            if i > 0 and not line.strip():
                insert_at = i + 1
            break
        head = lines[:insert_at] + ["", begin]
        block = []
        tail = [end, ""] + lines[insert_at:]

    existing = sorted({ln.strip() for ln in block if ln.strip().startswith("import ")})
    updated_block = sorted(set(existing + [import_line]))
    added = [import_line] if import_line not in existing else []

    updated_lines = head + updated_block + tail
    updated_text = "\n".join(updated_lines).rstrip() + "\n"
    updated = updated_text != text
    if updated:
        chapter_abs.write_text(updated_text, encoding="utf-8")

    return ChapterUpdateResult(chapter_rel=chapter_rel, updated=updated, added_imports=added)


def ensure_book_imports(
    *,
    project: str,
    chapter_aggregate_rel: Path,
) -> BookUpdateResult:
    """
    Ensure `<project>/Book.lean` imports the given chapter aggregate file (e.g. `Chapters/Chap01.lean`).

    We maintain an auto-managed block to avoid clobbering manual edits.
    """
    project = (project or "").strip()
    if not project:
        raise ValueError("project must be non-empty")

    chapter_aggregate_rel = Path(chapter_aggregate_rel)
    if chapter_aggregate_rel.suffix != ".lean":
        raise ValueError(f"expected a .lean file, got: {chapter_aggregate_rel}")
    if chapter_aggregate_rel.parts[:1] != (project,):
        raise ValueError(
            f"expected chapter file under project {project}, got: {chapter_aggregate_rel}"
        )
    if len(chapter_aggregate_rel.parts) != 3 or chapter_aggregate_rel.parts[1] != "Chapters":
        raise ValueError(
            f"expected a chapter aggregate like {project}/Chapters/ChapXX.lean, got: {chapter_aggregate_rel}"
        )
    if not _CHAPTER_FILE_RE.match(chapter_aggregate_rel.name):
        raise ValueError(
            f"expected a chapter aggregate filename like ChapXX.lean, got: {chapter_aggregate_rel.name}"
        )

    book_rel = Path(project) / "Book.lean"
    book_abs = LEAN_ROOT / book_rel
    book_abs.parent.mkdir(parents=True, exist_ok=True)
    if not book_abs.exists():
        _ensure_book_file_exists(project=project, book_abs=book_abs)

    begin = "-- BEGIN AUTO-IMPORTS (managed by orchestrator)"
    end = "-- END AUTO-IMPORTS"

    import_line = "import " + _module_name_from_rel_path(chapter_aggregate_rel)

    text = book_abs.read_text(encoding="utf-8")
    lines = text.splitlines()

    def _find_marker(lines: list[str], marker: str) -> int | None:
        for i, ln in enumerate(lines):
            if ln.strip() == marker:
                return i
        return None

    def _extract_block(lines: list[str]) -> tuple[list[str], list[str], list[str]]:
        """
        Return (head_with_begin, block_lines, tail_with_end_and_after).

        Robustness:
        - Accept markers with leading/trailing whitespace.
        - If multiple managed blocks exist (from prior buggy runs), collapse them into one
          by taking the first BEGIN and the last END after it.
        """
        b = _find_marker(lines, begin)
        if b is None:
            raise ValueError
        ends = [i for i, ln in enumerate(lines) if ln.strip() == end and i > b]
        if not ends:
            raise ValueError
        e = ends[-1]
        head = lines[:b] + [begin]
        block = lines[b + 1 : e]
        tail = [end] + lines[e + 1 :]
        return head, block, tail

    # If Book.lean has no imports at all, ensure it is at least a valid module by adding Mathlib.
    if not any(ln.startswith("import ") for ln in lines):
        lines = ["import Mathlib", ""] + lines
        text = "\n".join(lines).rstrip() + "\n"

    try:
        head, block, tail = _extract_block(lines)
    except ValueError:
        # Insert a new managed block right after the first import block.
        insert_at = 0
        for i, line in enumerate(lines):
            if line.startswith("import "):
                insert_at = i + 1
                continue
            if i > 0 and not line.strip():
                # allow a single blank line after imports
                insert_at = i + 1
            break
        head = lines[:insert_at] + ["", begin]
        block = []
        tail = [end, ""] + lines[insert_at:]

    existing = sorted({ln.strip() for ln in block if ln.strip().startswith("import ")})
    updated_block = sorted(set(existing + [import_line]))
    added = [import_line] if import_line not in existing else []

    updated_lines = head + updated_block + tail
    updated_text = "\n".join(updated_lines).rstrip() + "\n"
    updated = updated_text != text
    if updated:
        book_abs.write_text(updated_text, encoding="utf-8")

    return BookUpdateResult(book_rel=book_rel, updated=updated, added_imports=added)


def ensure_book_exists(*, project: str) -> Path:
    """
    Ensure `<project>/Book.lean` exists (without adding any section imports).
    Returns the relative Book.lean path.
    """
    project = (project or "").strip()
    if not project:
        raise ValueError("project must be non-empty")
    book_rel = Path(project) / "Book.lean"
    book_abs = LEAN_ROOT / book_rel
    book_abs.parent.mkdir(parents=True, exist_ok=True)
    if not book_abs.exists():
        _ensure_book_file_exists(project=project, book_abs=book_abs)
    return book_rel


def _ensure_book_file_exists(*, project: str, book_abs: Path) -> None:
    book_abs.write_text(
        "\n".join(
            [
                "import Mathlib",
                "",
                "/-!",
                f"# {project}",
                "",
                "Auto-managed imports live below.",
                "-/",
                "",
            ]
        )
        + "\n",
        encoding="utf-8",
    )


def _ensure_chapter_file_exists(*, project: str, chapter_abs: Path, chapter_rel: Path) -> None:
    chapter_abs.write_text(
        "\n".join(
            [
                "import Mathlib",
                "",
                "/-!",
                f"# {project}: {chapter_rel.stem}",
                "",
                "Auto-managed imports live below.",
                "-/",
                "",
            ]
        )
        + "\n",
        encoding="utf-8",
    )
