"""
Multi-Agent Communication Architecture System

This module implements a fully connected multi-agent system that supports
dynamic agent populations, multi-round conversations, and iterative optimization.
The system enables collaborative problem-solving through unrestricted agent-to-agent
communication patterns.
"""

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 multi-agent systems.
    
    Each agent maintains its own conversation history and can generate responses
    based on user queries and references to other agents' 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 peer agent responses.
        
        Args:
            user_query: The original user question or prompt
            reference_responses: List of responses from other agents 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 other agents' responses
                reference_text = ""
                if reference_responses:
                    reference_text = "\n\nReference responses from other agents:\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 MultiAgentSystem:
    """
    Fully connected multi-agent system implementation.
    
    This class manages a collection of agents that can communicate with each other
    in a fully connected topology, where every agent can reference responses from
    all other agents in the system.
    """
    
    def __init__(self, openai_api_key: str, openai_api_base: str = None, num_agents: int = 3):
        """
        Initialize the 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: Number of agents to create in the system
        """
        # Configure OpenAI API
        openai.api_key = openai_api_key
        openai.api_base = openai_api_base
        
        # Create agents dynamically
        self.agents = {}
        for i in range(num_agents):
            agent_name = f"Agent_{i + 1}"  # Agent_1, Agent_2, Agent_3, ...
            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 among all agents.
        
        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"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 = {}
            
            # Get all responses from previous round as reference
            previous_responses = []
            if round_num > 1 and all_rounds:
                previous_responses = list(all_rounds[-1].values())
            
            # Generate responses from each agent
            for agent_name, agent in self.agents.items():
                print(f"\n{agent_name} is thinking...")
                
                response = agent.generate_response(
                    user_query=user_query,
                    reference_responses=previous_responses,
                    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 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()),
            "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"conversation_{datetime.now().strftime('%Y%m%d_%H%M%S')}.json"
        
        export_data = {
            "user_query": self.current_user_query,
            "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")