from __future__ import annotations

import subprocess
from pathlib import Path

from .config import LEAN_ROOT, lake_env_lean_cmd


def lake_env_lean(lean_file: Path) -> tuple[int, str, str]:
    """
    Run `lake env lean <lean_file>` inside the Lean project root.
    """
    cmd = lake_env_lean_cmd(str(lean_file))
    print(f"[lean check] running: {' '.join(cmd)} (cwd={LEAN_ROOT})")
    result = subprocess.run(
        cmd,
        cwd=LEAN_ROOT,
        text=True,
        stdout=subprocess.PIPE,
        stderr=subprocess.PIPE,
    )
    return result.returncode, result.stdout, result.stderr


def get_tail_snippet(lean_file: Path, max_chars: int = 2000) -> str:
    """
    Return up to the last `max_chars` characters of the file.
    """
    content = lean_file.read_text(encoding="utf-8")
    return content[-max_chars:]


def get_line_snippet(
    lean_file: Path,
    *,
    line: int,
    context_lines: int = 40,
    max_chars: int = 4000,
) -> str:
    """
    Return a snippet centered around a 1-based `line` number.
    """
    content_lines = lean_file.read_text(encoding="utf-8").splitlines()
    if not content_lines:
        return ""

    line0 = max(1, line)
    start = max(1, line0 - context_lines)
    end = min(len(content_lines), line0 + context_lines)
    snippet = "\n".join(content_lines[start - 1 : end])
    if len(snippet) > max_chars:
        return snippet[:max_chars]
    return snippet


def get_declaration_snippet(
    lean_file: Path,
    *,
    line: int,
    max_chars: int = 8000,
) -> str:
    """
    Return the block of text around a line, bounded by blank lines (best-effort to capture the whole declaration).
    """
    content_lines = lean_file.read_text(encoding="utf-8").splitlines()
    if not content_lines:
        return ""
    idx = max(0, min(len(content_lines) - 1, line - 1))
    start = idx
    while start > 0 and content_lines[start].strip() != "":
        start -= 1
    if content_lines[start].strip() == "" and start < idx:
        start += 1
    end = idx
    while end + 1 < len(content_lines) and content_lines[end + 1].strip() != "":
        end += 1
    snippet = "\n".join(content_lines[start : end + 1])
    if len(snippet) > max_chars:
        return snippet[:max_chars]
    return snippet
