from typing import Dict, Any, List, Optional, Callable, Union
import json
from abc import ABC, abstractmethod
import asyncio
import os
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 client

class Agent(ABC):
    """
    Base 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 = client()
        
    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})
        
    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
        """
        self.add_message("user", prompt)
        try:
            response = 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)
            return assistant_message
        except Exception as e:
            error_msg = f"Error getting LLM response: {str(e)}"
            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
        """
        try:
            # Remove any potential "json" prefix that might be in the response
            cleaned_response = response.lstrip('`json').rstrip('`')
            if cleaned_response.startswith('{'):
                return json.loads(cleaned_response)
            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]
                    return json.loads(json_str)
                raise ValueError("Could not extract valid JSON from response")
        except json.JSONDecodeError as e:
            raise ValueError(f"Failed to parse JSON response: {str(e)}")
    
    @abstractmethod
    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):
    """
    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:
            prompt (str): Custom prompt template for the planner
            model (str): The LLM model to use
        """
        super().__init__(model)
    
    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:
            task (str): The task description
            tools (Any): Available tools for the agents
            
        Returns:
            Dict[str, Any]: The plan and generated code
        """
        prompt = f"{prompt}, output format must be json: {{'steps': List[str], 'roles': 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 = self.get_response(prompt)
            res = self.parse_json_response(response)
            return res

        except Exception as e:
            error_msg = f"Error in planning process: {str(e)}"
            print(error_msg)
            return {"error": error_msg}


    

class Verifier(Agent):
    """
    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)
        
    def process(self, prompt: str = "") -> Dict[str, Any]:
        """
        Process the verification task based on the prompt and result.
        
        Returns:
            Dict[str, Any]: The verification result
        """
        try:
            prompt = f"{prompt}, output format must be json: {{'is_valid': bool, 'evaluation': str}}"
            verification = self.get_response(prompt)
            return self.parse_json_response(verification)
        except Exception as e:
            error_msg = f"Error in verification process: {str(e)}"
            print(error_msg)
            return {"evaluation": error_msg}

class Actor(Agent):
    """
    Actor agent that performs specific tasks using tools.
    """
    def __init__(self, tools: Any = None, model: str = "gpt-4o-mini"):
        """
        Initialize the Actor agent.
        
        Args:
            tools (Any): The tools to use for execution
            model (str): The LLM model to use
        """
        super().__init__(model)
        self.tools = tools

    def process(self, prompt: str = "") -> Dict[str, Any]:
        """
        Process the actor's task based on its prompt.
        
        Returns:
            Dict[str, Any]: The result of the actor's processing
        """
        try:
            if self.tools == None:
                prompt = f"{prompt}, output format must be json: {{'result': str, 'observation': str}}. result is the final answer; observation is the process of the answer."
                execution_result = self.get_response(prompt) 
                return self.parse_json_response(execution_result)
            else:
                return self.act(prompt)

                
        except Exception as e:
            error_msg = f"Error in actor processing: {str(e)}"
            print(error_msg)
            return {"error": error_msg}

    def use_tool(self, tool_name: str = None, *args, **kwargs) -> Any:
        """
        Use a tool with the provided arguments.
        
        Args:
            tool_name (str, optional): Name of the tool to use (if tool is a dict of tools)
            *args: Positional arguments for the tool
            **kwargs: Keyword arguments for the tool
            
        Returns:
            Any: The result of the tool execution
            
        Raises:
            ValueError: If the tool cannot be used
        """
        try:
            if tool_name and isinstance(self.tools, dict):
                if tool_name not in self.tool:
                    raise ValueError(f"Tool '{tool_name}' not found")
                return self.tool[tool_name](*args, **kwargs)
            elif callable(self.tool):
                return self.tool(*args, **kwargs)
            elif self.tool is None:
                raise ValueError("No tool available")
            else:
                raise ValueError(f"Invalid tool type: {type(self.tool)}")
        except Exception as e:
            raise ValueError(f"Failed to use tool: {str(e)}")

    def act(self, question: str):
        sys_prompt = f"""
        You are an AI assistant that is very good at using tools.  
        Please answer the following questions as well as possible. You can use the following tools:  

        {self.tools}  

        Use the following format:  

        Question: The input question you must answer  
        "Choose the appropriate tool based on the user's question. "
        "If no tool is needed, reply directly.\n\n"
        "IMPORTANT: When you need to use a tool, you must ONLY respond with "
        If tools are needed, please be sure to provide the tool name and arguments.

        "Please use only the tools that are explicitly defined above."

        Question: {question}
        """

        messages = [{"role": "system", "content": sys_prompt}]

        response = client().chat.completions.create(
            model=self.model,
            messages=messages,
            tools=self.tools
            )
        content = response.choices[0]

        if content.finish_reason == "tool_calls":
            tool_call = content.message.tool_calls[0]
            tool_name = tool_call.function.name
            tool_args = json.loads(tool_call.function.arguments)

            result =self.use_tool(tool_name, tool_args)
            print(f"\n\n[Calling tool {tool_name} with args {tool_args}]\n\n")
            
            messages.append(content.message.model_dump())
            messages.append({
                "role": "tool",
                "content": result.content[0].text,
                "tool_call_id": tool_call.id,
            })
            user_prompt = f"""
            Please answer the user's question concisely based on the context and the results of the previous tool call.
            "Tool execution result: {result}"
            output format must be json: {{'result': str, 'observation': str}}. result is the final answer; observation is the process of the answer.
            """
            messages.append({"role": "user", "content": user_prompt})

            response = client().chat.completions.create(
                model=self.model,
                messages=messages,
                output_format={"type": "json_object"}
            )
            return self.parse_json_response(response.choices[0].message.content)
        return {"result": content.message.content, "observation": "Directly answer the user's question."}