"""
Ring Multi-Agent Communication Architecture System

This module implements a ring topology where each agent communicates only
with its immediate neighbors (predecessor and successor), forming a circular
communication pattern that enables distributed information flow.
"""

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 ring topology multi-agent systems.
    
    Each agent maintains its own conversation history and can generate responses
    based on user queries and references to neighbor 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 neighbor agent responses.
        
        Args:
            user_query: The original user question or prompt
            reference_responses: List of responses from neighbor 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 neighbor agents' responses
                reference_text = ""
                if reference_responses:
                    reference_text = "\n\nReference responses from neighbor 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 RingAgentSystem:
    """
    Ring topology multi-agent system implementation.
    
    This class manages a collection of agents arranged in a circular topology
    where each agent can only communicate with its immediate neighbors,
    enabling distributed information processing.
    """
    
    def __init__(self, openai_api_key: str, openai_api_base: str = None, num_agents: int = 3):
        """
        Initialize the ring 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 ring (default: 3)
        """
        # Configure OpenAI API
        openai.api_key = openai_api_key
        openai.api_base = openai_api_base
        
        # Create agents
        self.agents = {}
        self.agent_names = []
        
        for i in range(num_agents):
            agent_name = f"Agent_{i + 1}"
            self.agents[agent_name] = Agent(agent_name)
            self.agent_names.append(agent_name)
        
        self.conversation_rounds = []
        self.current_user_query = None
        
    def get_neighbors(self, agent_name: str) -> List[str]:
        """
        Get the neighbors (predecessor and successor) of a specified agent.
        
        Args:
            agent_name: Name of the agent
            
        Returns:
            List[str]: List of neighbor agent names
        """
        if agent_name not in self.agent_names:
            return []
        
        current_index = self.agent_names.index(agent_name)
        num_agents = len(self.agent_names)
        
        # Calculate predecessor and successor indices (circular)
        prev_index = (current_index - 1) % num_agents
        next_index = (current_index + 1) % num_agents
        
        neighbors = []
        if prev_index != current_index:  # Avoid single agent case
            neighbors.append(self.agent_names[prev_index])
        if next_index != current_index and next_index != prev_index:  # Avoid duplicates and two-agent case
            neighbors.append(self.agent_names[next_index])
            
        return neighbors
        
    def run_conversation(self, user_query: str, num_rounds: int = 3) -> List[Dict[str, AgentResponse]]:
        """
        Execute a multi-round conversation in ring 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 = {}
            
            # Ring architecture special handling:
            # Each agent can only see responses from its neighbors (predecessor and successor)
            
            # Get all responses from previous round
            previous_responses = {}
            if round_num > 1 and all_rounds:
                previous_responses = all_rounds[-1]
            
            # Generate responses from each agent
            for agent_name, agent in self.agents.items():
                print(f"\n{agent_name} is thinking...")
                
                # Get this agent's neighbors
                neighbors = self.get_neighbors(agent_name)
                
                # Collect neighbor responses as reference
                reference_responses = []
                if round_num > 1:
                    for neighbor_name in neighbors:
                        if neighbor_name in previous_responses:
                            reference_responses.append(previous_responses[neighbor_name])
                
                response = agent.generate_response(
                    user_query=user_query,
                    reference_responses=reference_responses,
                    round_number=round_num
                )
                
                current_round_responses[agent_name] = response
                
                print(f"{agent_name}'s response:")
                print(response.content)
                
                # Display referenced neighbors
                if neighbors:
                    neighbor_str = ", ".join(neighbors)
                    print(f"(Referenced neighbors: {neighbor_str})")
                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 ring topology metadata
        """
        if not self.conversation_rounds:
            return {"error": "No conversation has been conducted"}
        
        # Build neighbor relationship map
        neighbor_map = {}
        for agent_name in self.agent_names:
            neighbor_map[agent_name] = self.get_neighbors(agent_name)
        
        summary = {
            "total_rounds": len(self.conversation_rounds),
            "agents": list(self.agents.keys()),
            "ring_topology": neighbor_map,
            "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"ring_conversation_{datetime.now().strftime('%Y%m%d_%H%M%S')}.json"
        
        # Build neighbor relationship map
        neighbor_map = {}
        for agent_name in self.agent_names:
            neighbor_map[agent_name] = self.get_neighbors(agent_name)
        
        export_data = {
            "system_type": "ring_architecture",
            "user_query": self.current_user_query,
            "ring_topology": neighbor_map,
            "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")
