#!/usr/bin/env python3
"""
Test Single Agent - Run individual agents for debugging and testing.

Usage:
    python orchestrator/test_single_agent.py --agent analyzer --cve CVE-2025-13287
    python orchestrator/test_single_agent.py --agent generator --cve CVE-2025-13287
    python orchestrator/test_single_agent.py --agent builder --cve CVE-2025-13287
    python orchestrator/test_single_agent.py --agent validator --cve CVE-2025-13287
    python orchestrator/test_single_agent.py --agent solver --cve CVE-2025-13287

    # Run check_phase (check_cve_ready.py loop with checker agent)
    python orchestrator/test_single_agent.py --agent check_phase --cve CVE-2025-13287
    python orchestrator/test_single_agent.py --agent check_phase --cve CVE-2025-13287 --max-retries 5

    # With custom message
    python orchestrator/test_single_agent.py --agent validator --cve CVE-2025-13287 \
        --message "Please fix the failing test_func.py"

    # Verbose mode (shows agent conversation)
    python orchestrator/test_single_agent.py --agent analyzer --cve CVE-2025-13287 -v
"""

import argparse
import asyncio
import logging
import sys
from pathlib import Path
from datetime import datetime

import yaml

from orchestrator.script_executor import ScriptExecutor

# Add parent directory to path for imports
sys.path.insert(0, str(Path(__file__).parent.parent))

from orchestrator.agent_runner import AgentRunner
from orchestrator.tool_controller import AgentType


# Default messages for each agent type
DEFAULT_MESSAGES = {
    AgentType.ANALYZER: """
You are analyzing {cve_id}.

Below is the CVE information:

---
{cve_content}
---

Begin your comprehensive analysis and create the required output files.
""",

    AgentType.GENERATOR: """
You are generating reproduction materials for {cve_id}.

The analyzer has completed their work. Review their output in .agent_state/analyzer_output/ and create:
1. A user-facing task description (task.yaml)
2. Comprehensive test suites (tests/)
3. A solution script (solution.sh)

Remember to frame the task as a bug report, not a security issue.
""",

    AgentType.BUILDER: """
You are building the Docker environment for {cve_id}.

Review the provided specifications in docker_requirements.md and task-deps/ directory.
Create a working Docker environment (Dockerfile, docker-compose.yaml).
Focus on authenticity and reproducibility.

IMPORTANT: You cannot see tests/ or solution.sh - build based only on task.yaml and docker_requirements.md.
""",

    AgentType.VALIDATOR: """
You are validating the vulnerability reproduction for {cve_id}.

Run the test suites and verify that:
1. The vulnerability is demonstrable
2. Functional tests (test_func.py) PASS
3. Vulnerability tests (test_vuln.py) FAIL (vulnerability exists)
4. Environment is stable

You may make minor adjustments if needed.
""",

    AgentType.SOLVER: """
You are validating the solution for {cve_id}.

Apply solution.sh and verify that:
1. The solution fixes the vulnerability
2. Functional tests (test_func.py) still PASS
3. Vulnerability tests (test_vuln.py) now PASS (vulnerability fixed)
4. No functionality is broken

Make adjustments to the solution if needed.
""",

    AgentType.CHECKER: """
You are the Checker agent for {cve_id}.

The check_cve_ready.py script has failed. Review the check results in .agent_state/checker_output/check_results.md, analyze the failures, fix the issues, and ensure the check passes.

Start by reading:
1. .agent_state/analyzer_output/public.md - to understand the CVE
2. .agent_state/checker_output/check_results.md - to see what failed

Then debug and fix the issues. Run ../../.venv/bin/python ../../scripts/check_cve_ready.py to verify your fixes.
""",
}


def load_config() -> dict:
    """Load configuration from config.yaml"""
    config_path = Path(__file__).parent.parent / "config.yaml"
    if not config_path.exists():
        raise FileNotFoundError(f"Config file not found: {config_path}")

    with open(config_path) as f:
        return yaml.safe_load(f)


def setup_logging(verbose: bool = False):
    """Setup logging configuration"""
    level = logging.DEBUG if verbose else logging.INFO

    logging.basicConfig(
        level=level,
        format='%(asctime)s - %(name)s - %(levelname)s - %(message)s',
        datefmt='%Y-%m-%d %H:%M:%S'
    )

    # Reduce noise from other loggers
    logging.getLogger('httpx').setLevel(logging.WARNING)
    logging.getLogger('httpcore').setLevel(logging.WARNING)


def get_cve_content(cve_id: str, cve_tasks_dir: Path) -> str:
    """Try to load CVE content from various sources"""
    # Try reproduce_cves directory first
    reproduce_cves_dir = Path(__file__).parent.parent / "reproduce_cves"
    for suffix in ['.md', '.txt', '']:
        cve_file = reproduce_cves_dir / f"{cve_id}{suffix}"
        if cve_file.exists():
            return cve_file.read_text()

    # Try analyzer output in working directory
    analyzer_output = cve_tasks_dir / cve_id / ".agent_state" / "analyzer_output" / "public.md"
    if analyzer_output.exists():
        return analyzer_output.read_text()

    return f"[CVE content for {cve_id} not found - please provide via --message or create reproduce_cves/{cve_id}.md]"


async def run_single_agent(
    agent_type: AgentType,
    cve_id: str,
    working_dir: Path,
    config: dict,
    message: str,
    timeout: int = None
) -> dict:
    """
    Run a single agent and return the result.

    Args:
        agent_type: Type of agent to run
        cve_id: CVE identifier
        working_dir: Working directory for the agent
        config: Configuration dictionary
        message: Message to send to the agent
        timeout: Optional timeout in seconds

    Returns:
        Result dictionary from agent execution
    """
    logger = logging.getLogger(__name__)

    # Create agent runner
    agent_runner = AgentRunner(config)

    try:
        # Create session
        logger.info(f"Creating {agent_type.value} session for {cve_id}")
        session_id = await agent_runner.create_session(
            agent_type=agent_type,
            cve_id=cve_id,
            working_dir=working_dir,
            metadata={"test_mode": True}
        )

        # Get timeout from config if not specified
        if timeout is None:
            timeout = config.get('agents', {}).get('timeouts', {}).get(agent_type.value, 600)

        # Run message
        logger.info(f"Running {agent_type.value} agent (timeout: {timeout}s)")
        start_time = datetime.now()

        result = await agent_runner.run_message(session_id, message, timeout)

        duration = (datetime.now() - start_time).total_seconds()
        logger.info(f"Agent completed in {duration:.2f}s with status: {result.get('status')}")

        return result

    finally:
        # Cleanup
        await agent_runner.cleanup_all_sessions()


def _format_check_results(check_result: dict) -> str:
    """Format check results as markdown for checker agent."""
    lines = ["# CVE Ready Check Results\n"]

    status = "READY" if check_result.get('ready') else "NOT READY"
    lines.append(f"## Overall Status: {status}\n")
    lines.append(f"**CVE**: {check_result.get('cve_id', 'Unknown')}\n")

    lines.append("## Check Details\n")
    checks = check_result.get('checks', {})

    for check_name, check_data in checks.items():
        if check_data.get('skipped'):
            status = "SKIPPED"
        else:
            status = "PASS" if check_data.get('success') else "FAIL"

        lines.append(f"### {check_name}\n- Status: {status}")

        if check_data.get('missing'):
            lines.append(f"- Missing: {', '.join(check_data['missing'])}")

        if check_data.get('details', {}).get('raw_output'):
            output = check_data['details']['raw_output'][-2000:]
            lines.append(f"- Output:\n```\n{output}\n```")

        lines.append("")

    return '\n'.join(lines)


async def run_check_phase(
    cve_id: str,
    working_dir: Path,
    config: dict,
    max_retries: int = 3,
    timeout: int = None,
    verbose: bool = False
):
    """Run the full check phase: check_cve_ready.py loop with checker agent."""
    logger = logging.getLogger(__name__)
    script_executor = ScriptExecutor(config)
    agent_runner = AgentRunner(config)

    checker_output_dir = working_dir / ".agent_state" / "checker_output"
    checker_output_dir.mkdir(parents=True, exist_ok=True)

    try:
        for attempt in range(max_retries + 1):
            print(f"\n{'='*60}")
            print(f"CHECK PHASE - Attempt {attempt + 1}/{max_retries + 1}")
            print(f"{'='*60}\n")

            # Step 1: Run check_cve_ready.py
            print("Running check_cve_ready.py...")
            check_result = await script_executor.run_cve_check(
                working_dir=working_dir,
                cve_id=cve_id,
                skip_docker=False,
                skip_vuln=False,
                verbose=verbose
            )

            if check_result['ready']:
                print(f"\n✓ CHECK PASSED on attempt {attempt + 1}")
                return

            if attempt >= max_retries:
                print(f"\n✗ CHECK FAILED after {max_retries + 1} attempts")
                sys.exit(1)

            # Step 2: Save check results for checker agent
            check_results_content = _format_check_results(check_result)
            (checker_output_dir / "check_results.md").write_text(check_results_content)
            print(f"\n✗ Check failed. Starting checker agent to fix issues...")

            # Step 3: Run checker agent
            session_id = await agent_runner.create_session(
                agent_type=AgentType.CHECKER,
                cve_id=cve_id,
                working_dir=working_dir,
                metadata={"check_attempt": attempt + 1}
            )

            checker_message = DEFAULT_MESSAGES[AgentType.CHECKER].format(cve_id=cve_id)
            agent_timeout = timeout or config.get('agents', {}).get('timeouts', {}).get('checker', 600)

            result = await agent_runner.run_message(session_id, checker_message, agent_timeout)

            if result.get('status') != 'completed':
                print(f"Checker agent failed: {result.get('error', 'Unknown error')}")
                sys.exit(1)

            print(f"Checker agent completed. Re-running check...")

    finally:
        await agent_runner.cleanup_all_sessions()


def main():
    parser = argparse.ArgumentParser(
        description="Run a single agent for testing/debugging",
        formatter_class=argparse.RawDescriptionHelpFormatter,
        epilog="""
Examples:
  # Run analyzer for a CVE
  python orchestrator/test_single_agent.py --agent analyzer --cve CVE-2025-13287

  # Run check_phase (check_cve_ready.py loop with checker agent)
  python orchestrator/test_single_agent.py --agent check_phase --cve CVE-2025-13287

  # Run check_phase with more retries
  python orchestrator/test_single_agent.py --agent check_phase --cve CVE-2025-13287 --max-retries 5

  # Run validator with custom message
  python orchestrator/test_single_agent.py --agent validator --cve CVE-2025-13287 \\
      --message "The test_func.py is failing, please investigate"
"""
    )

    parser.add_argument(
        '--agent', '-a',
        required=True,
        choices=['analyzer', 'generator', 'builder', 'validator', 'solver', 'check_phase'],
        help='Agent type to run (check_phase runs check_cve_ready.py loop with checker agent)'
    )

    parser.add_argument(
        '--cve', '-c',
        required=True,
        help='CVE ID (e.g., CVE-2025-13287)'
    )

    parser.add_argument(
        '--message', '-m',
        help='Custom message to send to the agent (overrides default)'
    )

    parser.add_argument(
        '--timeout', '-t',
        type=int,
        help='Timeout in seconds (default: from config.yaml)'
    )

    parser.add_argument(
        '--verbose', '-v',
        action='store_true',
        help='Enable verbose output (shows agent conversation)'
    )

    parser.add_argument(
        '--working-dir', '-w',
        help='Custom working directory (default: cve_tasks_dir/CVE-ID)'
    )

    parser.add_argument(
        '--max-retries',
        type=int,
        default=3,
        help='Max checker agent retries for check_phase (default: 3)'
    )

    args = parser.parse_args()

    # Setup logging
    setup_logging(args.verbose)
    logger = logging.getLogger(__name__)

    # Load config
    try:
        config = load_config()
    except Exception as e:
        logger.error(f"Failed to load config: {e}")
        sys.exit(1)

    # Enable verbose logging if requested
    if args.verbose:
        config.setdefault('logging', {})['verbose_claude_code'] = True

    # Get agent type (check_phase is special)
    is_check_phase = args.agent == 'check_phase'
    agent_type = AgentType.CHECKER if is_check_phase else AgentType(args.agent)
    cve_id = args.cve

    # Determine working directory
    cve_tasks_dir = Path(config.get('paths', {}).get('cve_tasks_dir', './cve_tasks'))
    if args.working_dir:
        working_dir = Path(args.working_dir)
    else:
        working_dir = cve_tasks_dir / cve_id

    # Create working directory if it doesn't exist
    working_dir.mkdir(parents=True, exist_ok=True)

    # Create necessary subdirectories
    (working_dir / ".agent_state" / f"{agent_type.value}_output").mkdir(parents=True, exist_ok=True)
    (working_dir / ".logs").mkdir(parents=True, exist_ok=True)

    # Handle check_phase specially - run the full check loop
    if is_check_phase:
        asyncio.run(run_check_phase(
            cve_id=cve_id,
            working_dir=working_dir,
            config=config,
            max_retries=args.max_retries,
            timeout=args.timeout,
            verbose=args.verbose
        ))
        sys.exit(0)

    # Prepare message
    if args.message:
        message = args.message
    else:
        # Use default message template
        cve_content = get_cve_content(cve_id, cve_tasks_dir) if agent_type == AgentType.ANALYZER else ""
        message = DEFAULT_MESSAGES[agent_type].format(
            cve_id=cve_id,
            cve_content=cve_content
        ).strip()

    # Print info
    print(f"\n{'='*60}")
    print(f"Running {agent_type.value.upper()} agent for {cve_id}")
    print(f"Working directory: {working_dir}")
    print(f"{'='*60}\n")

    if not args.verbose:
        print("Message:")
        print("-" * 40)
        print(message[:500] + "..." if len(message) > 500 else message)
        print("-" * 40)
        print()

    # Run agent
    try:
        result = asyncio.run(run_single_agent(
            agent_type=agent_type,
            cve_id=cve_id,
            working_dir=working_dir,
            config=config,
            message=message,
            timeout=args.timeout
        ))

        # Print result summary
        print(f"\n{'='*60}")
        print("RESULT SUMMARY")
        print(f"{'='*60}")
        print(f"Status: {result.get('status')}")
        print(f"Duration: {result.get('duration', 0):.2f}s")
        print(f"Responses: {len(result.get('responses', []))}")

        if result.get('error'):
            print(f"Error: {result.get('error')}")

        print(f"{'='*60}\n")

        # Exit with appropriate code
        if result.get('status') == 'completed':
            sys.exit(0)
        elif result.get('status') == 'timeout':
            sys.exit(2)
        else:
            sys.exit(1)

    except KeyboardInterrupt:
        print("\n\nInterrupted by user")
        sys.exit(130)
    except Exception as e:
        logger.exception(f"Error running agent: {e}")
        sys.exit(1)


if __name__ == "__main__":
    main()
