"""Semantic query analyzer using structured LLM output"""

from typing import List, Dict, Optional
from enum import Enum
from pydantic import BaseModel, Field
from langchain_core.prompts import ChatPromptTemplate
from langchain_core.language_models import BaseLanguageModel


class QueryIntent(str, Enum):
    """Types of medical query intents"""
    DIAGNOSTIC = "diagnostic"  # Is there X? Any signs of Y?
    MEASUREMENT = "measurement"  # What is the CTR? Measure X
    COMPARISON = "comparison"  # Compare with prior
    LOCALIZATION = "localization"  # Where is X located?
    COMPREHENSIVE = "comprehensive"  # Full assessment
    CHARACTERIZATION = "characterization"  # Describe the appearance of X


class ExecutionMode(str, Enum):
    """Execution modes for agents based on query complexity"""
    FUNCTION_CALLING = "function_calling"  # Simple, direct queries - fast execution
    REACT = "react"  # Medium complexity - step-by-step reasoning
    PLAN_EXECUTE = "plan_execute"  # Complex queries requiring structured planning


class AgentRequirement(BaseModel):
    """Requirement for a specific agent"""
    agent_name: str = Field(description="Name of the agent (airway, breathing, cardiac, diaphragm, everything)")
    priority: float = Field(description="Priority score 0-1, where 1 is highest priority")
    specific_tasks: List[str] = Field(description="Specific tasks this agent should perform")
    execution_mode: ExecutionMode = Field(description="Recommended execution mode for this agent based on task complexity")


class SemanticQueryAnalysis(BaseModel):
    """Structured query analysis result"""
    query_intent: QueryIntent = Field(description="Primary intent of the query")
    agent_requirements: List[AgentRequirement] = Field(description="Required agents with priorities, tasks, and execution modes")
    analysis_focus: List[str] = Field(description="Key areas to analyze based on the query (not expected results)")
    requires_comparison: bool = Field(description="Whether prior image comparison is needed")
    comparison_focus: Optional[List[str]] = Field(default=None, description="Specific findings to compare if comparison needed")
    confidence_threshold: float = Field(description="Minimum confidence for reliable answer (0.5-0.95)")
    clinical_context: str = Field(description="Brief clinical context extracted from query")
    overall_complexity: str = Field(description="Overall query complexity: simple, medium, complex")
    default_execution_mode: ExecutionMode = Field(description="Default execution mode for agents based on overall query complexity")


class SemanticQueryAnalyzer:
    """Semantic query analyzer that uses LLM for all understanding"""
    
    def __init__(self, llm: BaseLanguageModel):
        self.llm = llm
        self.analyzer = self._create_analyzer()
    
    def _create_analyzer(self):
        """Create the semantic analyzer with structured output"""
        
        system_prompt = """You are an expert medical imaging query analyzer. Analyze the user's query about chest X-rays and determine:

1. The primary intent of the query
2. Which ABCDE agents are needed and their priorities
3. Specific tasks each agent should perform
4. Appropriate execution mode for each agent based on task complexity
5. Overall query complexity and default execution mode

Available agents and their capabilities:

**AIRWAY Agent** - Analyzes central thoracic structures:
- Tracheal position and deviation (PRIMARY responsibility)
- Carina and main bronchi patency
- Mediastinal masses or lymphadenopathy  
- Paratracheal abnormalities
- Note: Tracheal position assessment is AIRWAY's sole responsibility

**BREATHING Agent** - Evaluates lungs and pleura:
- Lung opacities (consolidation, infiltrates, nodules)
- Pleural effusions
- Pneumothorax
- Pulmonary vascular patterns
- Interstitial patterns

**CARDIAC Agent** - Assesses heart and great vessels:
- Heart size and CTR calculation
- Chamber enlargement
- Cardiac devices (pacemakers, valves)
- Mediastinal contours (only cardiac-related mediastinal assessment)
- Vascular calcifications
- Note: Do NOT activate for tracheal position queries

**DIAPHRAGM Agent** - Examines diaphragm and subdiaphragmatic region:
- Hemidiaphragm position and contour
- Free air (pneumoperitoneum)
- Costophrenic angles
- Subdiaphragmatic pathology

**EVERYTHING Agent** - Evaluates remaining structures:
- Bones (ribs, clavicles, spine)
- Soft tissues
- Medical devices (tubes, lines)
- Foreign bodies
- Surgical changes

EXECUTION MODES - Choose based on task complexity:

**FUNCTION_CALLING Mode** - For simple, direct queries:
- Single pathology detection ("Is there pneumothorax?")
- Specific measurements ("What is the CTR?")
- Yes/no questions ("Is the trachea midline?")
- Clear, focused diagnostic questions
- Fast execution, immediate tool usage

**REACT Mode** - For medium complexity queries:
- Multiple related findings ("Check for signs of pneumonia")
- Reasoning through differential diagnosis
- Queries requiring step-by-step analysis
- When multiple tools need coordination
- Iterative refinement of findings

**PLAN_EXECUTE Mode** - For complex, comprehensive queries:
- Full radiological reports ("Write a CXR report")
- Comprehensive assessments ("What abnormalities are shown?")
- Multiple system evaluation ("Complete ABCDE assessment")
- Structured reporting requirements
- Complex multi-step workflows

COMPLEXITY ASSESSMENT GUIDELINES:
- **Simple**: Single question, one pathology, yes/no, specific measurement
- **Medium**: Multiple related findings, diagnostic reasoning, comparison tasks
- **Complex**: Comprehensive reports, multiple systems, detailed characterization

IMPORTANT GUIDELINES:
1. For tracheal position queries (e.g., "Is the trachea midline?"), ONLY activate AIRWAY agent
2. For mediastinal queries, determine context:
   - If about masses/lymph nodes → AIRWAY agent
   - If about cardiac silhouette/contours → CARDIAC agent
3. Be precise - don't activate agents unless they directly contribute to answering the query
4. For measurement queries, only activate the agent that can perform that specific measurement
5. Match execution mode to task complexity - don't use complex modes for simple queries
6. Consider query intent when determining execution mode

COMPARISON QUERIES:
- Set requires_comparison=true if query mentions:
  - "compare", "comparison", "changed", "progression", "improved", "worsened", "stable"
  - "prior", "previous", "last", "earlier", "before"
  - Temporal references suggesting comparison
- When comparison is needed, specify comparison_focus with specific findings to compare
- For comparison, agents should analyze BOTH current and prior images
- Use REACT or PLAN_EXECUTE mode for comparison tasks (higher complexity)

Analyze the query semantically - understand the medical intent, not just keywords.
Assign priorities based on how critical each agent is for answering the query.
Choose execution modes that match the complexity and requirements of each task."""
        
        prompt = ChatPromptTemplate.from_messages([
            ("system", system_prompt),
            ("human", "Query: {query}\n\nProvide a comprehensive semantic analysis of this medical imaging query.")
        ])
        
        return prompt | self.llm.with_structured_output(SemanticQueryAnalysis)
    
    def analyze(self, query: str) -> SemanticQueryAnalysis:
        """Perform semantic analysis of the query"""
        try:
            analysis = self.analyzer.invoke({"query": query})
            
            # Sort agents by priority
            analysis.agent_requirements.sort(key=lambda x: x.priority, reverse=True)
            
            return analysis
            
        except Exception as e:
            print(f"Error in semantic analysis: {e}")
            # Return a safe default that will trigger all agents
            return SemanticQueryAnalysis(
                query_intent=QueryIntent.COMPREHENSIVE,
                agent_requirements=[
                    AgentRequirement(agent_name="cardiac", priority=1.0, specific_tasks=["general assessment"], execution_mode=ExecutionMode.FUNCTION_CALLING),
                    AgentRequirement(agent_name="breathing", priority=1.0, specific_tasks=["general assessment"], execution_mode=ExecutionMode.FUNCTION_CALLING)
                ],
                analysis_focus=[],
                requires_comparison=False,
                confidence_threshold=0.7,
                clinical_context="General chest X-ray assessment",
                overall_complexity="medium",
                default_execution_mode=ExecutionMode.FUNCTION_CALLING
            )
    
    def get_active_agents(self, analysis: SemanticQueryAnalysis, threshold: float = 0.3) -> List[str]:
        """Get list of agents to activate based on priority threshold"""
        return [
            req.agent_name 
            for req in analysis.agent_requirements 
            if req.priority >= threshold
        ]
    
    def get_agent_tasks(self, analysis: SemanticQueryAnalysis, agent_name: str) -> List[str]:
        """Get specific tasks for an agent"""
        for req in analysis.agent_requirements:
            if req.agent_name == agent_name:
                return req.specific_tasks
        return []
    
    def get_agent_execution_mode(self, analysis: SemanticQueryAnalysis, agent_name: str) -> ExecutionMode:
        """Get execution mode for a specific agent"""
        for req in analysis.agent_requirements:
            if req.agent_name == agent_name:
                return req.execution_mode
        # Return default mode if agent not found
        return analysis.default_execution_mode 