from typing import Dict, Any, List, Optional, Callable, Union
import json
import logging
from abc import ABC, abstractmethod
import asyncio
import os
import re

import sys
sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.abspath(__file__))))
from tools.base import Server, Configuration, Tool
from utils.llm_service import async_client, client

# Configure logging
logging.basicConfig(
    level=logging.INFO,
    format='%(asctime)s - %(name)s - %(levelname)s - %(funcName)s:%(lineno)d - %(message)s',
    datefmt='%Y-%m-%d %H:%M:%S'
)

# Create logger for this module
logger = logging.getLogger(__name__)


class Agent(ABC):
    """
    Base Asynchronous Agent class that provides core functionality for LLM-based agents.
    All specialized agents should inherit from this class.
    """
    def __init__(self, model: str = "gpt-4o-mini"):
        """
        Initialize the agent with a specific LLM model.
        
        Args:
            model (str): The LLM model to use for generating responses
        """
        self.model = model
        self.messages: List[Dict[str, str]] = []
        self.llm = None  # Will be initialized asynchronously
        logger.info(f"Initialized {self.__class__.__name__} with model: {model}")
        
    async def init_llm(self):
        """Initialize the LLM client asynchronously"""
        if self.llm is None:
            logger.debug("Initializing LLM client...")
            try:
                self.llm = await async_client()
                logger.info("LLM client initialized successfully")
            except Exception as e:
                logger.error(f"Failed to initialize LLM client: {str(e)}")
                raise
        
    def add_message(self, role: str, content: str) -> None:
        """
        Add a message to the conversation history.
        
        Args:
            role (str): The role of the message sender (user, assistant, system)
            content (str): The content of the message
        """
        self.messages.append({"role": role, "content": content})
        logger.debug(f"Added {role} message to conversation history (length: {len(content)} chars)")
        
    async def get_response(self, prompt: str, response_format: str = "json_object") -> str:
        """
        Get a response from the LLM based on the prompt and conversation history.
        
        Args:
            prompt (str): The prompt to send to the LLM
            response_format (str): The format of the response (json_object or text)
            
        Returns:
            str: The LLM's response
        """
        logger.debug(f"Getting LLM response with format: {response_format}")
        await self.init_llm()
        self.add_message("user", prompt)
        try:
            logger.debug(f"Sending request to LLM model: {self.model}")
            response = await self.llm.chat.completions.create(
                model=self.model,
                messages=self.messages,
                response_format={"type": response_format}
            )
            assistant_message = response.choices[0].message.content
            self.add_message("assistant", assistant_message)
            logger.info(f"Successfully received LLM response (length: {len(assistant_message)} chars)")
            return assistant_message
        except Exception as e:
            error_msg = f"Error getting LLM response: {str(e)}"
            logger.error(error_msg)
            self.add_message("system", error_msg)
            raise RuntimeError(error_msg)
    
    def parse_json_response(self, response: str) -> Dict[str, Any]:
        """
        Parse a JSON response from the LLM, handling common errors.
        
        Args:
            response (str): The JSON string response from the LLM
            
        Returns:
            Dict[str, Any]: The parsed JSON object
        """
        logger.debug("Parsing JSON response from LLM")
        try:
            # Remove any potential "json" prefix that might be in the response
            cleaned_response = response.lstrip('`json').rstrip('`')
            if cleaned_response.startswith('{'):
                result = json.loads(cleaned_response)
                logger.debug("Successfully parsed JSON response (direct format)")
                return result
            else:
                # Try to find JSON content within the response
                start_idx = cleaned_response.find('{')
                end_idx = cleaned_response.rfind('}')
                if start_idx != -1 and end_idx != -1:
                    json_str = cleaned_response[start_idx:end_idx+1]
                    result = json.loads(json_str)
                    logger.debug("Successfully parsed JSON response (extracted format)")
                    return result
                logger.warning("Could not extract valid JSON from response")
                raise ValueError("Could not extract valid JSON from response")
        except json.JSONDecodeError as e:
            logger.error(f"Failed to parse JSON response: {str(e)}")
            raise ValueError(f"Failed to parse JSON response: {str(e)}")
    
    @abstractmethod
    async def process(self, input_data: Any) -> Any:
        """
        Process input data and return a result. Must be implemented by subclasses.
        
        Args:
            input_data (Any): The input data to process
            
        Returns:
            Any: The processed result
        """
        pass


class Planner(Agent):
    """
    Asynchronous Planner agent that creates execution plans for tasks.
    Generates a structured plan and code for a MultiAgentSystem implementation.
    """
    def __init__(self, model: str = "gpt-4o-mini"):
        """
        Initialize the Planner agent.
        
        Args:
            model (str): The LLM model to use
        """
        super().__init__(model)
        
    async def process(self, prompt: str = "", tools: Any = None) -> Dict[str, Any]:
        """
        Process a task by creating a plan and generating code for a MultiAgentSystem.
        
        Args:
            prompt (str): The task description
            tools (Any): Available tools for the agents
            
        Returns:
            Dict[str, Any]: The plan and generated code
        """
        logger.info(f"Planner processing task: {prompt[:100]}...")
        logger.debug(f"Available tools: {tools}")
        
        # Construct the prompt for the planner
        planning_prompt = f"""
        You are an expert AI system architect. Create a detailed execution plan for the following task:
        
        TASK: {prompt}
        
        Available tools: {tools if tools else 'No specific tools provided'}
        
        Your response should include:
        1. A high-level overview of the solution approach
        2. A detailed step-by-step execution plan
        3. The agent types needed and their responsibilities
        4. The communication flow between agents
        5. Any potential challenges and how to address them
        
        Format your response as a JSON object with the following structure:
        {{'steps': List[str], 'roles': List[List[str]]}}. steps is a list of plans for each step, and roles is a list of execution roles (Actor or Verifier) for each step."
        """
        
        try:
            response = await self.get_response(planning_prompt)
            result = self.parse_json_response(response)
            logger.info("Planner successfully generated execution plan")
            return result
        except Exception as e:
            logger.error(f"Planner failed to process task: {str(e)}")
            raise


class Verifier(Agent):
    """
    Asynchronous Verifier agent that validates results from other agents.
    """
    def __init__(self, model: str = "gpt-4o-mini"):
        """
        Initialize the Verifier agent.
        
        Args:
            model (str): The LLM model to use
        """
        super().__init__(model)
        
    async def process(self, prompt: str = "", result: Any = None) -> Dict[str, Any]:
        """
        Process the verification task based on the prompt and result.
        
        Args:
            prompt (str): The original task description
            result (Any): The result to verify
            
        Returns:
            Dict[str, Any]: The verification result
        """
        logger.info(f"Verifier processing verification for task: {prompt[:100]}...")
        logger.debug(f"Result to verify: {str(result)[:200]}...")
        
        verification_prompt = f"""
        You are an expert verification agent. Evaluate the following result for the given task:
        
        TASK: {prompt}
        
        RESULT: {result}
        
        Verify the result for:
        1. Correctness - Does it solve the task correctly?
        2. Completeness - Does it address all aspects of the task?
        3. Efficiency - Is the solution efficient?
        4. Robustness - Will it handle edge cases?
        
        Format your response as a JSON object with the following structure:
        {{
            "is_valid": bool,
            "score": A score from 0-10,
            "feedback": "Detailed feedback on the result",
            "improvement": "Improvement suggestions",
            "evaluation": "Overall assessment of the result"
        }}
        """
        
        try:
            response = await self.get_response(verification_prompt)
            result = self.parse_json_response(response)
            logger.info(f"Verifier completed verification with score: {result.get('score', 'N/A')}")
            return result
        except Exception as e:
            logger.error(f"Verifier failed to process verification: {str(e)}")
            raise


class Actor(Agent):
    """
    Asynchronous Actor MCP that orchestrates the interaction between user, LLM, and tools.
    Only works with UnifiedTool objects.
    """
    def __init__(self, tools=None, prompt: str = "", model: str = "gpt-4o-mini") -> None:
        super().__init__(model)
        self.tools = tools if isinstance(tools, list) else []
        
        # Create tool configs for LLM function calling
        self.tool_configs = []
        if self.tools:
            self.tool_configs = [tool.get_tool_config() for tool in self.tools]
            logger.info(f"Actor initialized with {len(self.tools)} tools: {[tool.name for tool in self.tools]}")
        else:
            logger.info("Actor initialized without tools")
        
    async def cleanup_tools(self) -> None:
        """Clean up all tools properly."""
        logger.info("Starting tool cleanup process")
        cleanup_tasks = []
        
        # Clean up tools if they have cleanup methods
        for tool in self.tools:
            if hasattr(tool, 'cleanup') and callable(tool.cleanup):
                logger.debug(f"Cleaning up tool: {tool.name}")
                if asyncio.iscoroutinefunction(tool.cleanup):
                    cleanup_tasks.append(tool.cleanup())
                else:
                    cleanup_tasks.append(asyncio.to_thread(tool.cleanup))
        
        if cleanup_tasks:
            await asyncio.gather(*cleanup_tasks)
            logger.info(f"Successfully cleaned up {len(cleanup_tasks)} tools")
        else:
            logger.debug("No tools require cleanup")
            
    async def process_llm_response(self, llm_response: str) -> Any:
        """
        Process the LLM response and execute tools if needed.
        
        Args:
            llm_response: The response from the LLM.
        
        Returns:
            The result of tool execution or the original response.
        """
        logger.debug("Processing LLM response for potential tool calls")
        try:
            # Parse the response as JSON
            response_data = self.parse_json_response(llm_response)
            
            # Check if the response contains a tool call
            if "tool" in response_data and "args" in response_data:
                tool_name = response_data["tool"]
                tool_args = response_data["args"]
                logger.info(f"Detected tool call: {tool_name} with args: {tool_args}")
                
                # Find and execute the tool
                for tool in self.tools:
                    if tool.name == tool_name:
                        logger.debug(f"Executing tool: {tool_name}")
                        result = await tool.execute(**tool_args)
                        logger.info(f"Tool {tool_name} executed successfully")
                        return result
                
                # If we get here, the tool wasn't found
                logger.warning(f"Tool '{tool_name}' not found in available tools")
                return {"error": f"Tool '{tool_name}' not found"}
            
            # If no tool call is detected, return the original response
            logger.debug("No tool call detected in LLM response")
            return llm_response
            
        except Exception as e:
            logger.error(f"Error processing LLM response: {str(e)}")   
            return {"error": f"Error processing LLM response: {str(e)}"}
    
    async def start(self, question: str) -> Any:
        """
        Main chat session handler.
        
        Args:
            question: The user's question or request.
            
        Returns:
            The final response after processing.
        """
        logger.info(f"Actor starting chat session for question: {question[:100]}...")
        try:
            # Construct the system prompt with available tools
            tools_description = "No tools available"
            if self.tools:
                tools_description = "\n".join([tool.format_for_llm() for tool in self.tools])
                logger.debug(f"Available tools description generated (length: {len(tools_description)} chars)")
            
            system_prompt = f"""
            You are an AI assistant that can use various tools to help answer questions.
            
            Available tools:
            {tools_description}
            
            When you need to use a tool, respond with a JSON object in this format:
            {{
                "tool": "tool_name",
                "args": {{
                    "arg1": "value1",
                    "arg2": "value2"
                }}
            }}
            
            If you don't need to use a tool, respond directly with helpful information.
            """
            
            # Add the system prompt to the conversation
            self.add_message("system", system_prompt)
            
            # Get the initial response from the LLM
            prompt = f"User question: {question}\n\nPlease help answer this question. Use tools if necessary."
            logger.debug("Getting initial LLM response")
            response = await self.get_response(prompt)
            
            # Process the response and execute any tools
            result = await self.process_llm_response(response)

            # If the result contains tool output, send a follow-up to the LLM
            if result != response:
                logger.info("Tool was executed, generating final response")
                follow_up_prompt = f"""
                The result is gotten from the tool:
                
                {result}
                
                Please provide a final answer to the user's question based on this information.
                output format must be json: {{'result': str, 'observation': str}}. result is the final answer and does not include the process and other details; observation is the process of the answer.
                """
                
                final_response = await self.get_response(follow_up_prompt)
                final_response = self.parse_json_response(final_response)
                logger.info("Actor successfully completed chat session with tool usage")
                return {"result": final_response['result'], "observation": final_response['observation'], "tool_output": result}
        
            # If there was an error or no tool was used, return the result directly
            logger.info("Actor completed chat session without tool usage")
            return self.parse_json_response(result)
            
        except Exception as e:
            error_msg = f"Error in actor processing: {str(e)}"
            logger.error(error_msg)
            return {"error": error_msg}
        
    async def react(self, input_data: Any, prompt: str = "") -> Any:
        """
        Process input data using ReAct pattern: Think->Act->Observe loop until final answer
        
        Args:
            input_data (Any): Input question or data
            
        Returns:
            Any: Final answer
        """
        question = str(input_data)
        logger.info(f"Actor starting ReAct pattern for question: {question[:100]}...")
        
        tool_descriptions = "\n".join([f"{tool.name}: {tool.description}" for tool in self.tools])
        tool_names = ", ".join([tool.name for tool in self.tools])
        logger.debug(f"ReAct pattern initialized with tools: {tool_names}")
        
        react_prompt = f"""
                        {prompt}

                        Answer the following question as best you can. You have access to the following tools:
                        {tool_descriptions}

                        Use the following format:

                        Question: the input question you must answer
                        Thought: you should always think about what to do
                        Action: the action to take, should be one of [{tool_names}]
                        Action Input: the input to the action, json format: {{"arg1": "value1", "arg2": "value2", ...}}   
                        Observation: the result of the action
                        ... (this Thought/Action/Action Input/Observation can repeat N times)
                        Thought: I now know the final answer
                        Final Answer: the final answer to the original input question

                        Begin!

                        Question: {question}
                        """
        
        # Clear message history and start new ReAct session
        self.messages = []
        self.add_message("user", react_prompt)
        
        max_iterations = 10  # Prevent infinite loops
        iteration = 0
        logger.debug(f"Starting ReAct loop with max iterations: {max_iterations}")
        
        try:
            await self.init_llm()
            
            while iteration < max_iterations:
                logger.debug(f"ReAct iteration {iteration + 1}/{max_iterations}")
                # Get LLM response
                response = await self.llm.chat.completions.create(
                    model=self.model,
                    messages=self.messages,
                    response_format={"type": "text"},
                    stop=["Observation:"]  # Stop before observation
                )
                
                assistant_response = response.choices[0].message.content

                self.add_message("assistant", assistant_response)
                
                # Check if contains final answer
                if "Final Answer:" in assistant_response:
                    logger.info(f"ReAct pattern found final answer in iteration {iteration + 1}")
                    # Extract final answer
                    final_pattern = r"Final Answer:(.+?)(?:\n|$)"
                    final_match = re.search(final_pattern, assistant_response, re.DOTALL)
                    if final_match:
                        final_answer = final_match.group(1).strip()
                        logger.info("ReAct pattern successfully completed")
                        return {
                            "result": final_answer,
                            "observation": f"Used ReAct pattern, reached answer after {iteration + 1} rounds of thinking and action"
                        }
                
                # Parse action
                action_pattern = r"Action:\s*([^\n]+)\s*Action Input:\s*(.+?)(?:\n|$)"
                action_match = re.search(action_pattern, assistant_response, re.DOTALL)

                if action_match:
                    action_name = action_match.group(1).strip()
                    action_input = action_match.group(2).strip()
                    logger.debug(f"ReAct detected action: {action_name} with input: {action_input[:100]}...")
                    

                    # Find corresponding tool
                    target_tool = None
                    for tool in self.tools:
                        if tool.name == action_name:
                            target_tool = tool
                            break
                    
                    if target_tool:
                        try:
                            logger.debug(f"Executing ReAct tool: {action_name}")
                            # Execute tool
                            if hasattr(target_tool, 'execute'):
                                # Use new UnifiedTool interface
                                observation = await target_tool.execute(**json.loads(action_input))
                            else:
                                # Compatible with old tool interface
                                observation = target_tool(action_input)
                            
                            logger.info(f"ReAct tool {action_name} executed successfully")
                            # Add observation result to conversation
                            observation_text = f"\nObservation: {observation}\nThought: "
                            logger.debug(f"observation: {observation}")

                            self.add_message("user", observation_text)
                            
                        except Exception as e:
                            logger.error(f"ReAct tool execution error for {action_name}: {str(e)}")
                            error_observation = f"\nObservation: Tool execution error: {str(e)}\nThought: "
                            self.add_message("user", error_observation)
                    else:
                        # Tool not found
                        logger.warning(f"ReAct tool '{action_name}' not found")
                        error_observation = f"\nObservation: Tool '{action_name}' not found, available tools: {tool_names}\nThought: "
                        self.add_message("user", error_observation)
                else:
                    # No action found, might be pure thinking
                    if "Thought:" in assistant_response:
                        # Continue thinking, no need to add extra content
                        continue
                    else:
                        # Incorrect format, prompt to continue
                        self.add_message("user", "\nPlease continue using the correct format: Thought -> Action -> Action Input\n")
                
                iteration += 1
            
            # If maximum iterations reached without answer
            logger.warning(f"ReAct pattern failed to converge after {max_iterations} iterations")
            return {
                "result": "Sorry, could not reach a final answer within the specified steps. Please try a simpler question or provide more information.",
                "observation": f"Failed to converge to final answer after {max_iterations} ReAct cycles"
            }
            
        except Exception as e:
            error_msg = f"Error occurred during ReAct processing: {str(e)}"
            logger.error(error_msg)
            return {"error": error_msg}



    async def process(self, input_data: Any, prompt: str = "You are an AI assistant.") -> Any:
        """
        Process input data and return a result.
        
        Args:
            input_data (Any): The input data to process
            
        Returns:
            Any: The processed result
        """
        logger.info(f"Actor processing input: {str(input_data)[:100]}...")
        if self.tools:
            logger.debug("Processing with tools available")
            #return await self.start(input_data)
            return await self.react(input_data, prompt)
        else:
            logger.debug("Processing without tools")
            prompt = f"{prompt}\n\nHere is the task: {input_data}.\n\noutput format must be json: {{'result': str, 'observation': str}}. result is the final answer; observation is the process of the answer."
            execution_result = await self.get_response(prompt) 
            result = self.parse_json_response(execution_result)
            logger.info("Actor completed processing without tools")
            return result
        
            
            
