#!/usr/bin/env python3

import argparse
import json
import logging
import sys
from pathlib import Path

import modal

from sorrydb.strategies.cloud_llm_strategy import CloudLLMStrategy
from sorrydb.runners.json_runner import SorryStrategy
from sorrydb.strategies.llm_proof_utils import NO_CONTEXT_PROMPT
from sorrydb.runners.modal_app import app
from sorrydb.runners.modal_hugging_face_provider import (
    ModalDeepseekProverLLMProvider,
    ModalKiminaLLMProvider,
)
from sorrydb.runners.strategy_comparison_runner import StrategyComparisonRunner
from sorrydb.strategies.rfl_strategy import NormNumStrategy, RflStrategy, SimpStrategy, SingleTacticStrategy

DEFAULT_TACTICS = [
    "rfl",
    "simp",
    "simp_all",
    "exact?",
    "grind",
    "ring",
    "norm_num",
    "omega",
    "linarith",
    "nlinarith",
    "aesop",
]

FAST_STRATEGIES: list[SorryStrategy] = [
    SingleTacticStrategy(t) for t in DEFAULT_TACTICS
]


def main():
    parser = argparse.ArgumentParser(description="Reproduce a sorry with REPL.")
    parser.add_argument(
        "--sorry-file",
        type=str,
        required=True,
        help="Path to the sorry JSON file",
    )
    parser.add_argument(
        "--output-file",
        type=str,
        required=True,
        help="Path to the output JSON file",
    )
    parser.add_argument(
        "--lean-data",
        type=str,
        default=None,
        help="Directory to store Lean data (default: use temporary directory)",
    )

    parser.add_argument(
        "--log-level",
        type=str,
        default="INFO",
        choices=["DEBUG", "INFO", "WARNING", "ERROR", "CRITICAL"],
        help="Set the logging level (default: INFO)",
    )
    parser.add_argument(
        "--log-file", type=str, help="Log file path (default: output to stdout)"
    )

    parser.add_argument(
        "--no-verify",
        action="store_true",
        help="Do not build the Lean package or verify the sorry results",
    )
    parser.add_argument(
        "--use-llm-strategies",
        action="store_true",
        help="Use LLM-based strategies instead of the default test strategies.",
    )
    parser.add_argument(
        "--llm-debug-info",
        type=str,
        default="modal_debug_info.json",
        help="Path to save llm debug info JSON (default: ./modal_debug_info.json)",
    )
    args = parser.parse_args()

    # Configure logging
    log_kwargs = {
        "level": getattr(logging, args.log_level),
        "format": "%(asctime)s - %(name)s - %(levelname)s - %(message)s",
    }
    if args.log_file:
        log_kwargs["filename"] = args.log_file
    logging.basicConfig(**log_kwargs)

    logger = logging.getLogger(__name__)
    # Convert file names arguments to Path
    sorry_file = Path(args.sorry_file)
    output_file = Path(args.output_file)

    if args.lean_data:
        lean_data_path = Path(args.lean_data)
        # If lean_data is provided, make sure it exists
        lean_data_path.mkdir(exist_ok=True)
    else:
        lean_data_path = None

    # Process the sorry JSON file
    try:
        logger.info(f"Solving sorries from: {sorry_file} using compare runners")
        runner = StrategyComparisonRunner(lean_data_path)

        runner.load_sorries(sorry_file, build_lean_projects=not args.no_verify)

        if args.use_llm_strategies:
            with modal.enable_output():  # this context manager enables modals logging
                with app.run():
                    deepseek_provider = ModalDeepseekProverLLMProvider()
                    deepseek_strategy = CloudLLMStrategy(
                        deepseek_provider,
                        prompt=NO_CONTEXT_PROMPT,
                        debug_info_path=args.llm_debug_info,
                    )
                    runner.attempt_sorries(deepseek_strategy)

                    kimina_modal_provider = ModalKiminaLLMProvider()
                    kimina_strategy = CloudLLMStrategy(
                        kimina_modal_provider,
                        prompt=NO_CONTEXT_PROMPT,
                        debug_info_path=args.llm_debug_info,
                    )
                    runner.attempt_sorries(kimina_strategy)
        else:
            for strategy in FAST_STRATEGIES:
                runner.attempt_sorries(strategy)
                runner.write_report(output_file)

        if not args.no_verify:
            runner.verify_proofs()
            runner.write_report(output_file)

        return 0

    except FileNotFoundError as e:
        logger.error(f"File not found: {e}")
        return 1
    except json.JSONDecodeError as e:
        logger.error(f"Invalid JSON: {e}")
        return 1
    except Exception as e:
        logger.error(f"Unexpected error: {e}")
        logger.exception(e)
        return 1


if __name__ == "__main__":
    sys.exit(main())
