import os 
import json
import yaml
import argparse
from typing import List, Optional
from pydantic import BaseModel, Field, ValidationError
from pathlib import Path
from cresearcher.main import CodeResearcher
from rich.console import Console
from rich.syntax import Syntax

class AgentConfig(BaseModel):
    repo_url: str = Field(..., description="Git URL for the repository")
    repo_name: str = Field(..., description="Name of the repository")
    patch_gen_llm: str = Field(choices=["o1", "gpt-4o"], default="gpt-4o", description="LLM model used for patch generation")
    num_patches: int = Field(default=5, ge=1, description="Number of patches to attempt per bug")
    max_analysis_steps: int = Field(default=10, ge=1, description="Maximum steps allowed during analysis")
    logs_dir: str = Field(default="logs", description="Directory path to logs")
    work_dir: str = Field(default="workdir", description="Base directory for playgrounds")
    langs: str = Field(default="C,C++,Python,Asm,Cuda", description="Comma seperated list of languages to be supported. Languages should correspond to the list obtained by running ctags --list-languages")
    backport_commits_json: Optional[str] = Field(default=None, description="Path to backport commits JSON file needed for kBenchSyz benchmark")

class PromptsConfig(BaseModel):
    preamble: str = Field(..., description="Setup and role description for the LLM agent")
    analysis_examples: str = Field(..., description="Instructive examples for analysis phase")
    patch_gen_examples: str = Field(..., description="Instructive examples for patch generation phase")

class RootConfig(BaseModel):
    agent: AgentConfig
    prompts: PromptsConfig

class RunConfig(BaseModel):
    config: RootConfig
    target_json: str
    patch_gen_only: bool = Field(default=False, description="Flag to indicate if only patch generation is needed")
    init_run_data_path: Optional[str] = Field(default=None, description="Path to initial run data")

def load_config_from_yaml(path: str) -> RootConfig:
    try:
        with open(path, "r") as f:
            data = yaml.safe_load(f)
        return RootConfig(**data)
    except FileNotFoundError:
        raise RuntimeError(f"Config file not found: {path}")
    except ValidationError as e:
        print("\n❌ Configuration is invalid. Please fix the following issues:\n")
        for err in e.errors():
            loc = ".".join(str(x) for x in err['loc'])
            print(f"  - Field '{loc}': {err['msg']}")
        raise SystemExit(1)

def print_patches(patches: List[str]) -> None:
    console = Console()

    console.print("\n:rocket: [bold green]Patch Generation Complete![/bold green]", style="bold green")
    console.print(f"[cyan]Generated {len(patches)} patch{'es' if len(patches) != 1 else ''}.[/cyan]\n")

    for i, patch in enumerate(patches, 1):
        console.rule(f"[bold blue]Patch {i}[/bold blue]")
        syntax = Syntax(patch, "diff", theme="monokai", line_numbers=True, word_wrap=True)
        console.print(syntax)

    console.print("\n[bold green]:white_check_mark: All patches displayed successfully![/bold green]\n")


if __name__ == "__main__":
    parser = argparse.ArgumentParser(description="Run the code researcher.")
    parser.add_argument("--config", type=str, required=True, help="Path to the configuration file.")
    parser.add_argument("--target_json", type=str, required=True, help="Target bug JSON file.")
    parser.add_argument("--patch_gen_only", action="store_true", help="Flag to indicate if only patch generation is needed.")
    parser.add_argument("--init_run_data_path", type=str, help="Path to initial run data.")
    args = parser.parse_args()
    try:
        config_obj = load_config_from_yaml(args.config)
        print("✅ Configuration loaded successfully.")
    except SystemExit:
        print("❌ Failed to load configuration. Please check the file and try again.")
        exit(1)

    run_config = RunConfig(
        config=config_obj,
        target_json=args.target_json,
        patch_gen_only=args.patch_gen_only,
        init_run_data_path=args.init_run_data_path
    )
    print("✅ Configuration and CLI arguments successfully merged.")

    bugJson = run_config.target_json
    with open(bugJson, 'r') as f:
        bugData = json.load(f)
    bugData['crash_report_data'] = bugData['crashes'][0]['crash-report-data']

    workdirPath = Path(run_config.config.agent.work_dir).resolve()
    os.makedirs(workdirPath, exist_ok=True)
    bugLogsDir = Path(run_config.config.agent.logs_dir) / bugData['id']
    os.makedirs(bugLogsDir, exist_ok=True)

    cresearcher = CodeResearcher(
        repoUrl = run_config.config.agent.repo_url,
        repoName = run_config.config.agent.repo_name, 
        prompt_preamble=run_config.config.prompts.preamble,
        prompt_analysis_examples=run_config.config.prompts.analysis_examples,
        prompt_patch_gen_examples=run_config.config.prompts.patch_gen_examples,
        bugDict=bugData,
        langs=run_config.config.agent.langs,
        maxGlobalCtxSteps=run_config.config.agent.max_analysis_steps,
        workdirPath=workdirPath,
        logDir=bugLogsDir,
        backportCommitsJsonPath=Path(run_config.config.agent.backport_commits_json) if run_config.config.agent.backport_commits_json else None,
        patchGenOnly= run_config.patch_gen_only,
        initRunDataPath=Path(run_config.init_run_data_path) if run_config.init_run_data_path else None,
        patchGenllm=run_config.config.agent.patch_gen_llm,
        numPatches=run_config.config.agent.num_patches,
    )
    patches = cresearcher.run()
    print_patches(patches)  
    # # Print the patches
    # print("Final Patches:")
    # print("\n\n===\n\n".join(patches))

# those two places languages should propagate
# ctags.py
# gitdiff.py

