"""Universal interface for RadAgents Multi-Agent System"""

from typing import Dict, Optional, List, Any, Literal
from pathlib import Path
import os

from langchain_core.language_models import BaseLanguageModel
from langchain_openai import ChatOpenAI, AzureChatOpenAI

from .orchestrator import (
    LangGraphOrchestrator,
    OrchestratorConfig,
    OrchestratorResult
)
from .agents.base_abcde import ABCDEAgent
from .agents.cardiac_agent import CardiacAgent
from .agents.synthesis_agent import SynthesisReportAgent
from .reasoning.visual_cot import BiasResistantVisualCoT
from ..tools import (
    ChestXRayClassifierTool,
    ChestXRaySegmentationTool,
    XRayPhraseGroundingTool,
    DicomProcessorTool
)


def initialize_multi_agent_system(
    model: str = "gpt-4o",
    temperature: float = 0.7,
    tools_to_use: Optional[List[str]] = None,
    model_dir: str = "./models",
    temp_dir: str = "temp",
    device: str = "cuda",
    execution_mode: Literal["parallel", "sequential"] = "parallel",
    max_parallel_agents: int = 3,
    agent_timeout: int = 60,
    priority_threshold: float = 0.3,
    openai_kwargs: Optional[Dict] = None,
    smart_tool_loading: bool = True,
    required_agents: Optional[List[str]] = None,
) -> LangGraphOrchestrator:
    """
    Initialize the RadAgents multi-agent system with specified configuration.
    
    Args:
        model: LLM model to use (e.g., "gpt-4o", "gpt-4o-mini")
        temperature: Temperature for LLM
        tools_to_use: List of tool names to initialize (None = auto-detect)
        model_dir: Directory containing model weights
        temp_dir: Directory for temporary files
        device: Device to run models on ("cuda" or "cpu")
        execution_mode: "parallel" or "sequential" agent execution
        max_parallel_agents: Maximum agents to run in parallel
        agent_timeout: Timeout for each agent in seconds
        priority_threshold: Minimum priority for agent activation (0-1)
        openai_kwargs: Additional OpenAI/Azure configuration
        smart_tool_loading: If True, only load tools that might be needed
        required_agents: List of agents that must be initialized (default: ["cardiac", "synthesis"])
        
    Returns:
        LangGraphOrchestrator: Configured orchestrator ready for use
    """
    
    # Initialize LLM
    if openai_kwargs is None:
        openai_kwargs = {}
    
    # Check for Azure OpenAI
    azure_endpoint = os.environ.get("AZURE_OPENAI_ENDPOINT")
    if azure_endpoint:
        # Get all Azure OpenAI environment variables
        azure_api_key = os.environ.get("AZURE_OPENAI_API_KEY")
        azure_deployment_name = os.environ.get("AZURE_OPENAI_DEPLOYMENT_NAME", model)
        azure_api_version = os.environ.get("AZURE_OPENAI_API_VERSION", "2024-08-01-preview")
        
        # Build Azure OpenAI configuration
        azure_config = {
            "azure_endpoint": azure_endpoint,
            "deployment_name": azure_deployment_name,
            "api_version": azure_api_version,
            "temperature": temperature,
            **openai_kwargs
        }
        
        # Add API key if available
        if azure_api_key:
            azure_config["api_key"] = azure_api_key
        
        llm = AzureChatOpenAI(**azure_config)
        print(f"Using Azure OpenAI:")
        print(f"  Endpoint: {azure_endpoint}")
        print(f"  Deployment: {azure_deployment_name}")
        print(f"  API Version: {azure_api_version}")
    else:
        llm = ChatOpenAI(
            model=model,
            temperature=temperature,
            **openai_kwargs
        )
        print(f"Using OpenAI with model: {model}")
    
    # Create temp directory
    Path(temp_dir).mkdir(exist_ok=True)
    
    # Determine which tools to initialize based on required agents
    if smart_tool_loading and tools_to_use is None:
        # Determine tools based on which agents will be used
        smart_tools = set()
        
        # Default required agents if not specified
        agents_to_check = required_agents or ["cardiac", "airway", "synthesis"]
        
        # Map agents to their required tools
        agent_tool_requirements = {
            "cardiac": ["classification", "segmentation"],  # grounding is optional
            "airway": ["classification", "segmentation"],   # grounding is optional
            "breathing": ["classification", "segmentation", "grounding"],  # grounding required
            "diaphragm": ["classification", "segmentation"], # grounding is optional
            "everything": ["classification", "segmentation"], # grounding is optional
            "synthesis": []  # no tools required
        }
        
        # Collect required tools for selected agents
        for agent in agents_to_check:
            if agent in agent_tool_requirements:
                smart_tools.update(agent_tool_requirements[agent])
        
        tools_to_use = list(smart_tools)
        print(f"Smart tool loading: Selected tools {tools_to_use} for agents {agents_to_check}")
        
        # Only include grounding if breathing agent is specifically requested
        if "grounding" in tools_to_use:
            print("  Note: Grounding tool (Maira-2) will be loaded - this may take time")
    elif tools_to_use is None:
        # Load all tools if explicitly requested
        tools_to_use = ["classification", "segmentation", "grounding", "dicom"]
    
    # Define available tools
    available_tools = {
        "classification": lambda: ChestXRayClassifierTool(device=device),
        "segmentation": lambda: ChestXRaySegmentationTool(device=device),
        "grounding": lambda: XRayPhraseGroundingTool(
            cache_dir=model_dir,
            temp_dir=temp_dir,
            load_in_8bit=True,
            device=device
        ),
        "dicom": lambda: DicomProcessorTool(temp_dir=temp_dir)
    }
    
    # Initialize selected tools
    tools = {}
    
    print("Initializing tools...")
    for tool_name in tools_to_use:
        if tool_name in available_tools:
            try:
                tools[tool_name] = available_tools[tool_name]()
                print(f"✓ Initialized {tool_name} tool")
            except Exception as e:
                print(f"✗ Failed to initialize {tool_name} tool: {e}")
                # Don't fail completely, just skip this tool
    
    # Initialize V-CoT module
    vcot_module = BiasResistantVisualCoT(llm, temp_dir=temp_dir)
    
    # Initialize agents (Phase 1: CardiacAgent + SynthesisAgent)
    print("\nInitializing agents...")
    agents = {}
    
    # Default required agents
    if required_agents is None:
        required_agents = ["cardiac", "airway", "synthesis"]
    
    # Cardiac Agent - only if we have the minimal required tools
    if "cardiac" in required_agents and all(tool in tools for tool in ["classification", "segmentation"]):
        agents["cardiac"] = CardiacAgent(
            llm=llm,
            classification_tool=tools["classification"],
            segmentation_tool=tools["segmentation"],
            grounding_tool=tools.get("grounding"),  # Optional
            vcot_module=vcot_module
        )
        print("✓ Initialized CardiacAgent")
    
    # Airway Agent - add to Phase 1 testing
    if "airway" in required_agents and all(tool in tools for tool in ["classification", "segmentation"]):
        from .agents.airway_agent import AirwayAgent
        agents["airway"] = AirwayAgent(
            llm=llm,
            classification_tool=tools["classification"],
            segmentation_tool=tools["segmentation"],
            grounding_tool=tools.get("grounding"),  # Optional
            vcot_module=vcot_module
        )
        print("✓ Initialized AirwayAgent")
    
    # Synthesis Agent (always include if requested)
    if "synthesis" in required_agents:
        agents["synthesis"] = SynthesisReportAgent(llm=llm)
        print("✓ Initialized SynthesisReportAgent")
    
    # Breathing Agent - now available in Phase 1
    if "breathing" in required_agents and all(tool in tools for tool in ["classification", "segmentation"]):
        from .agents.breathing_agent import BreathingAgent
        # Breathing agent requires grounding tool
        if "grounding" in tools:
            agents["breathing"] = BreathingAgent(
                llm=llm,
                classification_tool=tools["classification"],
                segmentation_tool=tools["segmentation"],
                grounding_tool=tools["grounding"],  # Required for breathing agent
                vcot_module=vcot_module
            )
            print("✓ Initialized BreathingAgent")
        else:
            print("✗ Cannot initialize BreathingAgent: grounding tool required but not available")
    
    # Placeholder for other agents (Phase 2)
    # agents["diaphragm"] = DiaphragmAgent(...)
    # agents["everything"] = EverythingElseAgent(...)
    
    # Create orchestrator configuration
    config = OrchestratorConfig(
        parallel_execution=(execution_mode == "parallel"),
        max_parallel_agents=max_parallel_agents,
        agent_timeout=agent_timeout,
        priority_threshold=priority_threshold
    )
    
    # Create orchestrator
    orchestrator = LangGraphOrchestrator(
        llm=llm,
        agents=agents,
        config=config
    )
    
    print(f"\nMulti-agent system initialized successfully!")
    print(f"  Model: {model}")
    print(f"  Execution Mode: {execution_mode}")
    print(f"  Active Agents: {list(agents.keys())}")
    print(f"  Tools: {list(tools.keys())}")
    
    return orchestrator


def run_multi_agent_analysis(
    orchestrator: LangGraphOrchestrator,
    query: str,
    image_path: str,
    prior_image_path: Optional[str] = None
) -> Dict[str, Any]:
    """
    Run multi-agent analysis on an image with a query.
    
    Args:
        orchestrator: Initialized orchestrator
        query: Medical query about the image
        image_path: Path to the chest X-ray image
        prior_image_path: Optional path to prior image for comparison
        
    Returns:
        Dictionary containing:
        - answer: Direct answer to the query
        - confidence: Overall confidence (0-1)
        - findings: List of all findings from agents
        - agent_results: Detailed results from each agent
        - needs_review: Whether clinical review is recommended
        - execution_time: Time taken for analysis
    """
    
    # Execute analysis
    result = orchestrator.execute(
        query=query,
        image_path=image_path,
        prior_image_path=prior_image_path
    )
    
    # Extract synthesis result
    synthesis_result = result.synthesis_result
    
    # Format response for easy consumption
    response = {
        "answer": synthesis_result.answer if synthesis_result else "No synthesis available",
        "confidence": synthesis_result.confidence if synthesis_result else 0.5,
        "needs_review": synthesis_result.requires_review if synthesis_result else True,
        "execution_time": result.execution_time,
        "query_intent": result.query_analysis.query_intent.value,
        "activated_agents": result.activated_agents,
        "findings": [],
        "agent_results": {}
    }
    
    # Extract findings from all agents
    for agent_name, analysis in result.agent_results.items():
        if analysis and isinstance(analysis, dict) and 'findings' in analysis:
            for finding in analysis['findings']:
                if isinstance(finding, dict):
                    response["findings"].append({
                        "agent": agent_name,
                        "pathology": finding.get("pathology"),
                        "confidence": finding.get("confidence"),
                        "evidence": finding.get("evidence")
                    })
        
        # Store detailed agent results
        response["agent_results"][agent_name] = analysis
    
    # Add supporting evidence and caveats if available
    if synthesis_result and hasattr(synthesis_result, 'supporting_evidence'):
        response["supporting_evidence"] = synthesis_result.supporting_evidence
    
    if synthesis_result and hasattr(synthesis_result, 'caveats'):
        response["caveats"] = synthesis_result.caveats
    
    return response


# Convenience function for one-shot analysis
def analyze_chest_xray(
    image_path: str,
    query: str,
    prior_image_path: Optional[str] = None,
    model: str = "gpt-4o",
    temperature: float = 0.7,
    device: str = "cuda",
    **kwargs
) -> Dict[str, Any]:
    """
    Convenience function for one-shot chest X-ray analysis.
    
    Args:
        image_path: Path to chest X-ray image
        query: Medical query about the image
        prior_image_path: Optional path to prior image
        model: LLM model to use
        temperature: LLM temperature
        device: Device for model execution
        **kwargs: Additional arguments for system initialization
        
    Returns:
        Analysis results dictionary
    """
    
    # Initialize system
    orchestrator = initialize_multi_agent_system(
        model=model,
        temperature=temperature,
        device=device,
        **kwargs
    )
    
    # Run analysis
    return run_multi_agent_analysis(
        orchestrator=orchestrator,
        query=query,
        image_path=image_path,
        prior_image_path=prior_image_path
    ) 