"""
Flexible workflow for the multi-agent system using LangGraph
"""

from typing import Dict, Literal
from langgraph.graph import StateGraph, END
from concurrent.futures import ThreadPoolExecutor, as_completed

from ..state import MultiAgentState
from .query_analyzer import QueryAnalyzer
from ...agents.cardiac_agent import CardiacAgent # Assuming this is the main agent for Phase 1
from ...agents.synthesis_agent import SynthesisReportAgent

def create_flexible_workflow(
    agents: Dict[str, CardiacAgent], # Type hint to specific agent for now
    query_analyzer: QueryAnalyzer,
    synthesis_agent: SynthesisReportAgent,
    execution_mode: Literal["parallel", "sequential"] = "sequential"
):
    """Create workflow with query-driven agent selection and execution modes"""
    
    workflow = StateGraph(MultiAgentState)
    
    # 1. Node to analyze the query and select agents
    def analyze_query_node(state: MultiAgentState) -> MultiAgentState:
        """Analyze query and determine execution plan"""
        print("---Analyzing Query---")
        query = state["query"]
        
        analysis_result = query_analyzer.analyze_query(query)
        
        state["active_agents"] = analysis_result.primary_agents
        state["need_comparison"] = analysis_result.requires_comparison
        print(f"Active agents determined: {state['active_agents']}")
        return state

    workflow.add_node("analyze_query", analyze_query_node)
    
    # 2. Node to execute the selected ABCDE agents
    def execute_abcde_node(state: MultiAgentState) -> MultiAgentState:
        """Execute ABCDE agents based on mode and selection"""
        print(f"---Executing Agents in {execution_mode} mode---")
        active_agents = state.get("active_agents", [])
        
        # For phase 1, we only have the cardiac agent, so we simplify this
        # The full parallel/sequential logic can be added in Phase 2
        for agent_name in active_agents:
            if agent_name in agents:
                try:
                    print(f"Running agent: {agent_name}")
                    # Each agent gets a copy of the state to avoid race conditions
                    agent = agents[agent_name]
                    updated_state = agent.analyze(state.copy())
                    
                    # Merge the specific agent's analysis back into the main state
                    analysis_key = f"{agent_name}_analysis"
                    if analysis_key in updated_state:
                        state[analysis_key] = updated_state[analysis_key]

                except Exception as e:
                    print(f"Error in {agent_name} agent: {e}")
                    state[f"{agent_name}_analysis"] = {"error": str(e), "findings": []}

        return state

    workflow.add_node("execute_abcde", execute_abcde_node)
    
    # 3. Node for Synthesis/Reporting, now using the dedicated agent
    def synthesize_report_node(state: MultiAgentState) -> MultiAgentState:
        return synthesis_agent.analyze(state)

    workflow.add_node("synthesis_report", synthesize_report_node)

    # 4. Define the graph's control flow
    workflow.set_entry_point("analyze_query")
    workflow.add_edge("analyze_query", "execute_abcde")
    
    # For now, we go directly to synthesis. Comparison will be added later.
    workflow.add_edge("execute_abcde", "synthesis_report")
    workflow.add_edge("synthesis_report", END)
    
    return workflow.compile() 