"""
Star Multi-Agent Communication Architecture System

This module implements a star topology where a central coordinator agent
facilitates communication with peripheral agents. Peripheral agents only
communicate through the central hub, enabling centralized coordination.
"""

import openai
import json
import time
from typing import List, Dict, Any
from dataclasses import dataclass
from datetime import datetime
from config import get_openai_config


@dataclass
class AgentResponse:
    """
    Data structure for agent responses in multi-agent conversations.
    
    Attributes:
        agent_name: Unique identifier for the responding agent
        content: The textual response content
        round_number: Conversation round when response was generated
        timestamp: Time of response generation
        referenced_agents: List of agents whose responses were referenced
    """
    agent_name: str
    content: str
    round_number: int
    timestamp: datetime
    referenced_agents: List[str] = None


class Agent:
    """
    Individual agent implementation for star topology multi-agent systems.
    
    Each agent maintains its own conversation history and can generate responses
    based on user queries and references to the central coordinator's responses.
    """
    
    def __init__(self, name: str, system_prompt: str = None):
        """
        Initialize an agent with a unique name and system prompt.
        
        Args:
            name: Unique identifier for the agent
            system_prompt: System-level instructions for the agent's behavior
        """
        self.name = name
        self.system_prompt = system_prompt or (
            "You are an intelligent assistant. Please provide helpful and "
            "informative responses to user questions based on your knowledge "
            "and any provided context."
        )
        self.conversation_history = []
        
    def generate_response(self, user_query: str, reference_responses: List[AgentResponse] = None, 
                         round_number: int = 1) -> AgentResponse:
        """
        Generate agent response based on user query and central coordinator's response.
        
        Args:
            user_query: The original user question or prompt
            reference_responses: List of responses from central coordinator for reference
            round_number: Current conversation round number
            
        Returns:
            AgentResponse: The agent's generated response with metadata
        """
        try:
            # Build message list for API call
            messages = [{"role": "system", "content": self.system_prompt}]
            
            if round_number == 1:
                # First round: Direct response to user query
                messages.append({
                    "role": "user", 
                    "content": f"User Question: {user_query}\n\nPlease provide your response."
                })
            else:
                # Subsequent rounds: Reference central coordinator's response
                reference_text = ""
                if reference_responses:
                    reference_text = "\n\nReference response from central coordinator:\n"
                    for resp in reference_responses:
                        if resp.agent_name != self.name:  # Exclude own response
                            reference_text += f"\n{resp.agent_name}'s response:\n{resp.content}\n"
                
                messages.append({
                    "role": "user",
                    "content": f"Original user question: {user_query}\n{reference_text}\n\nPlease refine and optimize your response based on the above information."
                })
            
            # Call OpenAI API
            config = get_openai_config()
            try:
                # Try new OpenAI API client
                from openai import OpenAI
                client = OpenAI(api_key=config['api_key'], base_url=config['api_base'])
                response = client.chat.completions.create(
                    model=config['model'],
                    messages=messages,
                    temperature=config['temperature'],
                    max_tokens=config['max_tokens']
                )
                content = response.choices[0].message.content
            except ImportError:
                # Fallback to legacy API
                response = openai.ChatCompletion.create(
                    model=config['model'],
                    messages=messages,
                    temperature=config['temperature'],
                    max_tokens=config['max_tokens']
                )
                content = response.choices[0].message.content
            except Exception as e:
                # Final fallback for compatibility
                response = openai.ChatCompletion.create(
                    model=config['model'],
                    messages=messages,
                    temperature=config['temperature'],
                    max_tokens=config['max_tokens']
                )
                content = response['choices'][0]['message']['content']
            
            # Create response object
            agent_response = AgentResponse(
                agent_name=self.name,
                content=content,
                round_number=round_number,
                timestamp=datetime.now(),
                referenced_agents=[resp.agent_name for resp in reference_responses] if reference_responses else []
            )
            
            # Save to conversation history
            self.conversation_history.append(agent_response)
            
            return agent_response
            
        except Exception as e:
            print(f"Error generating response for agent {self.name}: {str(e)}")
            return AgentResponse(
                agent_name=self.name,
                content=f"Technical error occurred: {str(e)}",
                round_number=round_number,
                timestamp=datetime.now()
            )
    
    def get_history(self) -> List[AgentResponse]:
        """
        Retrieve the agent's conversation history.
        
        Returns:
            List of AgentResponse objects representing the conversation history
        """
        return self.conversation_history
    
    def clear_history(self):
        """Clear the agent's conversation history."""
        self.conversation_history = []


class StarAgentSystem:
    """
    Star topology multi-agent system implementation.
    
    This class manages a collection of agents arranged in a star topology
    with a central coordinator that facilitates communication between
    peripheral agents.
    """
    
    def __init__(self, openai_api_key: str, openai_api_base: str = None, num_agents: int = 3):
        """
        Initialize the star multi-agent system.
        
        Args:
            openai_api_key: OpenAI API key for language model access
            openai_api_base: Base URL for OpenAI API (optional)
            num_agents: Total number of agents (default: 3, includes 1 central agent)
        """
        # Configure OpenAI API
        openai.api_key = openai_api_key
        openai.api_base = openai_api_base
        
        # Create agents: 1 central agent + (num_agents-1) peripheral agents
        self.agents = {}
        
        # Central coordinator agent
        self.agents["Central_Agent"] = Agent("Central_Agent")
        self.central_agent_name = "Central_Agent"
        
        # Peripheral agents
        for i in range(num_agents - 1):
            agent_name = f"Agent_{i + 1}"
            self.agents[agent_name] = Agent(agent_name)
        
        self.conversation_rounds = []
        self.current_user_query = None
        
    def run_conversation(self, user_query: str, num_rounds: int = 3) -> List[Dict[str, AgentResponse]]:
        """
        Execute a multi-round conversation in star architecture.
        
        Args:
            user_query: The initial question or prompt for the agents
            num_rounds: Number of conversation rounds to execute
            
        Returns:
            List of dictionaries containing agent responses for each round
        """
        self.current_user_query = user_query  # Store user query
        print(f"Starting conversation with {num_rounds} rounds")
        print(f"User Query: {user_query}")
        print("=" * 50)
        
        all_rounds = []
        
        for round_num in range(1, num_rounds + 1):
            print(f"\nRound {round_num}:")
            print("-" * 30)
            
            current_round_responses = {}
            
            # Star architecture special handling:
            # 1. Central agent responds first (round 1) or based on all previous responses (subsequent rounds)
            # 2. Peripheral agents can only see the central agent's response
            
            # Get previous round responses as reference
            previous_responses = []
            if round_num > 1 and all_rounds:
                previous_responses = list(all_rounds[-1].values())
            
            # Central agent generates response
            central_agent = self.agents[self.central_agent_name]
            print(f"\n{self.central_agent_name} is thinking...")
            
            central_response = central_agent.generate_response(
                user_query=user_query,
                reference_responses=previous_responses,
                round_number=round_num
            )
            
            current_round_responses[self.central_agent_name] = central_response
            
            print(f"{self.central_agent_name}'s response:")
            print(central_response.content)
            print()
            
            # Peripheral agents generate responses based on central agent's response
            for agent_name, agent in self.agents.items():
                if agent_name == self.central_agent_name:
                    continue  # Skip central agent
                
                print(f"{agent_name} is thinking...")
                
                # Peripheral agents can only reference central agent's response
                reference_for_peripheral = [central_response]
                
                response = agent.generate_response(
                    user_query=user_query,
                    reference_responses=reference_for_peripheral,
                    round_number=round_num
                )
                
                current_round_responses[agent_name] = response
                
                print(f"{agent_name}'s response:")
                print(response.content)
                print()
            
            all_rounds.append(current_round_responses)
            self.conversation_rounds = all_rounds
            
            # Pause between rounds
            if round_num < num_rounds:
                print(f"Round {round_num} completed, preparing next round...")
                time.sleep(1)
        
        print("\nConversation completed")
        return all_rounds
    
    def get_conversation_summary(self) -> Dict[str, Any]:
        """
        Generate a summary of the conversation.
        
        Returns:
            Dictionary containing conversation statistics and star topology metadata
        """
        if not self.conversation_rounds:
            return {"error": "No conversation has been conducted"}
        
        summary = {
            "total_rounds": len(self.conversation_rounds),
            "agents": list(self.agents.keys()),
            "central_agent": self.central_agent_name,
            "rounds_detail": []
        }
        
        for i, round_responses in enumerate(self.conversation_rounds, 1):
            round_detail = {
                "round_number": i,
                "responses": {}
            }
            
            for agent_name, response in round_responses.items():
                round_detail["responses"][agent_name] = {
                    "content_length": len(response.content),
                    "timestamp": response.timestamp.isoformat(),
                    "referenced_agents": response.referenced_agents or []
                }
            
            summary["rounds_detail"].append(round_detail)
        
        return summary
    
    def export_conversation(self, filename: str = None) -> str:
        """
        Export conversation history to JSON file.
        
        Args:
            filename: Output filename (auto-generated if not provided)
            
        Returns:
            String path to the exported file
        """
        if filename is None:
            filename = f"star_conversation_{datetime.now().strftime('%Y%m%d_%H%M%S')}.json"
        
        export_data = {
            "system_type": "star_architecture",
            "user_query": self.current_user_query,
            "central_agent": self.central_agent_name,
            "conversation_rounds": []
        }
        
        for i, round_responses in enumerate(self.conversation_rounds, 1):
            round_data = {
                "round_number": i,
                "responses": {}
            }
            
            for agent_name, response in round_responses.items():
                round_data["responses"][agent_name] = {
                    "agent_name": response.agent_name,
                    "content": response.content,
                    "round_number": response.round_number,
                    "timestamp": response.timestamp.isoformat(),
                    "referenced_agents": response.referenced_agents or []
                }
            
            export_data["conversation_rounds"].append(round_data)
        
        with open(filename, 'w', encoding='utf-8') as f:
            json.dump(export_data, f, ensure_ascii=False, indent=2)
        
        print(f"Conversation exported to: {filename}")
        return filename
    
    def clear_all_history(self):
        """Clear conversation history for all agents and the system."""
        for agent in self.agents.values():
            agent.clear_history()
        self.conversation_rounds = []
        self.current_user_query = None
        print("All conversation history cleared")