import json
import os
from Agent_block import AGENT_BLOCKS
from typing import Dict, Any, Optional, Callable
from utils.llm_service import client

class FlowGen:
    def __init__(self, task_info, tools=None, llm="gpt-4o-mini"):
        """
        Initialize the FlowGen class.

        Args:
            task_info (str): Information about the task for which the agent flow is being generated.
            tools (dict): Optional tools that might be used in generating the agent flow.
        """
        self.task_info = task_info
        self.tools = tools if tools else {}
    
    def generation(self, llm="gpt-4o-mini", temperature=1):
        prompt = f"""
        As a master of agent flow design, you need to create a reasonable agent flow based on the task information to achieve the correct result.  
        Task: {self.task_info}  
        Available tools: {self.tools}  
        Choose the appropriate tool; if none is suitable, you may choose not to select one. When no tool is selected, the agent can directly generate results based on the information.  

        You can only redefine this class.
        class MultiAgentSystem:
            def __init__(self, name: str, tools=None) -> None:
                self.name = name
                self.tools = tools
            def run(self, task: str):
                raise NotImplementedError("This method should be implemented by the subclass")
        
        You need to implement it by instantiating the two existing basic Agent classes Actor(tool) and Verifier(). 
        You can obtain results by calling the process(prompt) method; Actor will return {{'result': str, 'observation': str}}; Verifier will return {{'is_valid': bool, 'evaluation': str}}. 
        Please use correct variable names and keys to ensure correct communication between Agents.
        Please use f-string formatting for the prompt to avoid key errors.
        
        For example:
        class MultiAgentSystem:
            def __init__(self, name: str, tools=None) -> None:
                self.name = name
                self.tools = tools
            def run(self, task: str):  #Must include task.
                # import necessary packages
                from agent.Agent import Planner, Actor, Verifier

                # Initialize agents
                actor1 = Actor(None)
                ...

                # Run agent flow
                prompt1 = f"..."
                res = actor1.process(prompt1)
                ...
                answer = ...

                # Return the final answer to the task; key name must be answer
                return {{"answer": str, ...}}

        Here are some principles for design:
        1. Design the Agent flow based on the difficulty of the task; a simple task requires only one Agent.
        2. Do not introduce new agents unless necessary (balance between accuracy and cost)
        3. You can use identity setting in the prompt to improve accuracy, such as being a mathematician
        4. It is also helpful to introduce a small number of examples in the prompt.
        5. After Verifier, there must be an Actor to fix the result according to the evaluation.
        6. You can use the Agent Block Library to help you design the Agent flow.
        
        here is the Agent Block Library:
        {AGENT_BLOCKS}

        Please return a brief implementation plan and the Python implementation of the MultiAgentSystem class
        The output format is json
        {{"plan": plan, "code": code, "num_agent": num_agent}}
        """
        try:
            response = client().chat.completions.create(
                model=llm,
                messages=[{"role": "system", "content": prompt}],
                response_format={"type": "json_object"},
                temperature=temperature,
            )
            res = json.loads(response.choices[0].message.content)

            return {"plan": res["plan"], "code": res["code"], "num_agent": res["num_agent"]}
            
        except Exception as e:
            error_msg = f"Error in planning process: {str(e)}"
            print(error_msg)
            return {"error": error_msg}
