#!/usr/bin/env python3
"""
Multi-Agent Experiment Runner for RadAgents

Usage:
    # Simple usage
    python run_multi_agent_experiment.py --image data/cardic_example/mild_enlarge_cardi.png --query "describe any cardiac findings on this CXR."
    
    # Batch experiments
    python run_multi_agent_experiment.py --config experiments/cardiac_experiments.json
"""

import os
import sys
import json
import argparse
import time
from datetime import datetime
from pathlib import Path
from typing import Dict, Any, Optional, List
import warnings

# Add project root to path
sys.path.append(os.path.dirname(os.path.abspath(__file__)))

# Load environment variables (gracefully handle if dotenv not available)
try:
    from dotenv import load_dotenv
    load_dotenv()  # Load .env file if it exists
except ImportError:
    print("Note: python-dotenv not available. Using system environment variables only.")

# Suppress warnings for cleaner output
warnings.filterwarnings("ignore")

from radagents.multi_agent.interface import (
    initialize_multi_agent_system,
    run_multi_agent_analysis,
    analyze_chest_xray
)


def setup_logging(log_dir: str = "log") -> str:
    """Setup logging directory and return log filename."""
    Path(log_dir).mkdir(exist_ok=True)
    timestamp = datetime.now().strftime("%Y%m%d_%H%M%S")
    return f"{log_dir}/multi_agent_experiment_{timestamp}"


def save_results(results: Dict[str, Any], log_prefix: str, image_path: str) -> None:
    """Save experiment results to files."""
    # Save JSON results
    results_file = f"{log_prefix}_results.json"
    with open(results_file, 'w') as f:
        json.dump(results, f, indent=2, default=str)
    
    print(f"Results saved to: {results_file}")
    # Note: Input image path is already stored in experiment metadata


def print_results(results: Dict[str, Any], query: str, image_path: str) -> None:
    """Print formatted results to console."""
    print("\n" + "="*80)
    print("MULTI-AGENT ANALYSIS RESULTS")
    print("="*80)
    
    print(f"\nQuery: {query}")
    print(f"Image: {image_path}")
    print(f"Query Intent: {results.get('query_intent', 'Unknown')}")
    print(f"Activated Agents: {', '.join(results.get('activated_agents', []))}")
    print(f"Execution Time: {results.get('execution_time', 0):.2f} seconds")
    
    print(f"\n{'ANSWER':<20}: {results.get('answer', 'No answer available')}")
    print(f"{'CONFIDENCE':<20}: {results.get('confidence', 0):.3f}")
    print(f"{'NEEDS REVIEW':<20}: {'Yes' if results.get('needs_review', True) else 'No'}")
    
    # Print findings
    findings = results.get('findings', [])
    if findings:
        print(f"\n{'FINDINGS':<20}:")
        for i, finding in enumerate(findings, 1):
            print(f"  {i}. Agent: {finding.get('agent', 'Unknown')}")
            print(f"     Pathology: {finding.get('pathology', 'Unknown')}")
            print(f"     Confidence: {finding.get('confidence', 0):.3f}")
            print(f"     Evidence: {finding.get('evidence', 'None')}")
    
    # Print supporting evidence and caveats if available
    if 'supporting_evidence' in results:
        print(f"\n{'SUPPORTING EVIDENCE':<20}:")
        for evidence in results['supporting_evidence']:
            print(f"  - {evidence}")
    
    if 'caveats' in results:
        print(f"\n{'CAVEATS':<20}:")
        for caveat in results['caveats']:
            print(f"  - {caveat}")
    
    # Print detailed agent results (only if there are agent results)
    agent_results = results.get('agent_results', {})
    if agent_results:
        print(f"\n{'DETAILED AGENT RESULTS':<20}:")
        for agent_name, agent_result in agent_results.items():
            print(f"  {agent_name.upper()}:")
            if isinstance(agent_result, dict):
                # Filter out internal keys and only show meaningful results
                meaningful_keys = [k for k in agent_result.keys() 
                                 if k not in ['findings', 'plan_executed', 'react_steps']]
                if meaningful_keys:
                    for key in meaningful_keys:
                        value = agent_result[key]
                        print(f"    {key}: {value}")
                else:
                    print(f"    Analysis completed - {len(agent_result.get('findings', []))} findings generated")
            else:
                print(f"    Result: {agent_result}")
    
    print("="*80)


def run_experiment(
    image_path: str,
    query: str,
    model: str = "gpt-4o",
    temperature: float = 0.7,
    required_agents: Optional[List[str]] = None,
    execution_mode: str = "parallel",
    smart_tool_loading: bool = True,
    device: str = "cuda",
    model_dir: str = "./models",
    temp_dir: str = "temp",
    save_logs: bool = True,
    **kwargs
) -> Dict[str, Any]:
    """
    Run a single multi-agent experiment.
    
    Args:
        image_path: Path to the chest X-ray image
        query: Medical query about the image
        model: LLM model to use
        temperature: LLM temperature
        required_agents: List of agents to use (None = default)
        execution_mode: "parallel" or "sequential"
        smart_tool_loading: Whether to use smart tool loading
        device: Device for model execution
        model_dir: Directory containing model weights
        temp_dir: Directory for temporary files
        save_logs: Whether to save logs
        **kwargs: Additional arguments
        
    Returns:
        Dictionary containing analysis results
    """
    
    # Verify image exists
    if not os.path.exists(image_path):
        raise FileNotFoundError(f"Image not found: {image_path}")
    
    print(f"Starting multi-agent experiment...")
    print(f"Image: {image_path}")
    print(f"Query: {query}")
    print(f"Model: {model}")
    print(f"Execution Mode: {execution_mode}")
    print(f"Smart Tool Loading: {smart_tool_loading}")
    
    # Setup logging
    log_prefix = None
    if save_logs:
        log_prefix = setup_logging()
        print(f"Log prefix: {log_prefix}")
    
    start_time = time.time()
    
    try:
        # Initialize the multi-agent system
        print("\nInitializing multi-agent system...")
        orchestrator = initialize_multi_agent_system(
            model=model,
            temperature=temperature,
            model_dir=model_dir,
            temp_dir=temp_dir,
            device=device,
            execution_mode=execution_mode,
            smart_tool_loading=smart_tool_loading,
            required_agents=required_agents,
            **kwargs
        )
        
        # Run analysis
        print("\nRunning multi-agent analysis...")
        results = run_multi_agent_analysis(
            orchestrator=orchestrator,
            query=query,
            image_path=image_path
        )
        
        # Add experiment metadata
        results['experiment_metadata'] = {
            'image_path': image_path,
            'query': query,
            'model': model,
            'temperature': temperature,
            'execution_mode': execution_mode,
            'smart_tool_loading': smart_tool_loading,
            'required_agents': required_agents,
            'timestamp': datetime.now().isoformat(),
            'total_experiment_time': time.time() - start_time
        }
        
        # Save results if logging is enabled
        if save_logs and log_prefix:
            save_results(results, log_prefix, image_path)
        
        return results
        
    except Exception as e:
        print(f"Error during experiment: {e}")
        raise


def load_experiment_config(config_path: str) -> Dict[str, Any]:
    """Load experiment configuration from JSON file."""
    with open(config_path, 'r') as f:
        return json.load(f)


def run_multiple_experiments(experiments: List[Dict[str, Any]]) -> List[Dict[str, Any]]:
    """Run multiple experiments from configuration."""
    results = []
    
    for i, exp_config in enumerate(experiments, 1):
        print(f"\n{'='*60}")
        print(f"RUNNING EXPERIMENT {i}/{len(experiments)}")
        print(f"{'='*60}")
        
        try:
            result = run_experiment(**exp_config)
            results.append(result)
            
            # Print results for this experiment
            print_results(result, exp_config['query'], exp_config['image_path'])
            
        except Exception as e:
            print(f"Experiment {i} failed: {e}")
            results.append({
                'error': str(e),
                'experiment_config': exp_config
            })
    
    return results


def main():
    parser = argparse.ArgumentParser(
        description="Run RadAgents Multi-Agent Experiments",
        formatter_class=argparse.RawDescriptionHelpFormatter,
        epilog="""
Examples:
  # Single experiment
  python run_multi_agent_experiment.py \\
    --image data/cardic_example/mild_enlarge_cardi.png \\
    --query "describe any cardiac findings on this CXR."
  
  # Experiment with specific agents
  python run_multi_agent_experiment.py \\
    --image data/cardic_example/mild_enlarge_cardi.png \\
    --query "describe any cardiac findings on this CXR." \\
    --agents cardiac synthesis \\
    --model gpt-4o-mini
  
  # Load from config file
  python run_multi_agent_experiment.py --config experiments/cardiac_experiments.json
  
  # Sequential execution with breathing agent in react mode
  python run_multi_agent_experiment.py \\
    --image data/cardic_example/mild_enlarge_cardi.png \\
    --query "describe any breathing abnormalities." \\
    --agents breathing synthesis \\
    --execution-mode sequential \\
    --breathing-mode react
        """
    )
    
    # Main experiment arguments
    parser.add_argument('--image', type=str, help='Path to chest X-ray image')
    parser.add_argument('--query', type=str, help='Medical query about the image')
    parser.add_argument('--config', type=str, help='Path to experiment configuration JSON file')
    
    # Model configuration
    parser.add_argument('--model', type=str, default='gpt-4o', 
                       help='LLM model to use (default: gpt-4o)')
    parser.add_argument('--temperature', type=float, default=0.7,
                       help='LLM temperature (default: 0.7)')
    
    # System configuration
    parser.add_argument('--agents', nargs='*', default=None,
                       help='Required agents (default: cardiac airway synthesis)')
    parser.add_argument('--execution-mode', choices=['parallel', 'sequential'], 
                       default='parallel', help='Agent execution mode')
    parser.add_argument('--device', type=str, default='cuda', 
                       help='Device for model execution')
    parser.add_argument('--model-dir', type=str, default='./models',
                       help='Directory containing model weights')
    parser.add_argument('--temp-dir', type=str, default='temp',
                       help='Directory for temporary files')
    
    # Utility options
    parser.add_argument('--no-smart-loading', action='store_true',
                       help='Disable smart tool loading')
    parser.add_argument('--no-logs', action='store_true',
                       help='Disable log saving')
    parser.add_argument('--quiet', action='store_true',
                       help='Suppress detailed output')
    
    args = parser.parse_args()
    
    # Validate arguments
    if not args.config and not (args.image and args.query):
        parser.error("Either --config or both --image and --query must be provided")
    
    if args.config and (args.image or args.query):
        parser.error("Cannot use --config with --image/--query")
    
    try:
        if args.config:
            # Load and run multiple experiments from config
            print(f"Loading experiments from: {args.config}")
            config = load_experiment_config(args.config)
            
            if 'experiments' in config:
                experiments = config['experiments']
            else:
                experiments = [config]  # Single experiment config
            
            results = run_multiple_experiments(experiments)
            
        else:
            # Run single experiment
            experiment_args = {
                'image_path': args.image,
                'query': args.query,
                'model': args.model,
                'temperature': args.temperature,
                'required_agents': args.agents,
                'execution_mode': args.execution_mode,
                'smart_tool_loading': not args.no_smart_loading,
                'device': args.device,
                'model_dir': args.model_dir,
                'temp_dir': args.temp_dir,
                'save_logs': not args.no_logs
            }
            
            result = run_experiment(**experiment_args)
            
            if not args.quiet:
                print_results(result, args.query, args.image)
    
    except Exception as e:
        print(f"Experiment failed: {e}")
        sys.exit(1)
    
    print("\nExperiment completed successfully!")


if __name__ == "__main__":
    main()

# example
# run_multi_agent_experiment_cardiac.py --image data/cardic_example/mild_enlarge_cardi.png --query "describe any cardiac findings on this CXR." --no-smart-loading
