import asyncio
import os
import copy

import logging

from agentverse.environments.tasksolving_env.basic import BasicEnvironment
from agentverse.initialization import load_agent, load_environment, prepare_task_config
from agentverse.utils import AGENT_TYPES
from tqdm import tqdm

openai_logger = logging.getLogger("openai")
openai_logger.setLevel(logging.WARNING)


class TaskSolvingIter:
    environment: BasicEnvironment
    task: str = ""
    logs: list = []
    data_iter = None
    data_max_len = -1

    def __init__(self, data_iter, environment: BasicEnvironment, task: str = "", data_max_len: int = -1):
        self.environment = environment
        self.task = task
        self.data_iter = data_iter
        self.data_max_len = data_max_len if data_max_len else len(data_iter.examples)

    @classmethod
    def from_task(cls, data_iter, task: str, tasks_dir: str, postfix='', data_max_len: int = -1):
        """Build an AgentVerse from a task name.
        The task name should correspond to a directory in `tasks` directory.
        Then this method will load the configuration from the yaml file in that directory.
        """
        # Prepare the config of the task
        task_config = prepare_task_config(task, tasks_dir, postfix)

        # Build the environment
        env_config = task_config["environment"]

        # Build agents for all pipeline (task)
        agents = {}
        for i, agent_config in enumerate(task_config["agents"]):
            if agent_config.get("agent_type", "") == "critic" and 0:
                agent = load_agent(agent_config)
                agents[AGENT_TYPES.CRITIC] = [
                    copy.deepcopy(agent)
                    for _ in range(task_config.get("cnt_agents", 1) - 1)
                ]
                # if agent_config.get("llm_type").startswith('[') and agent_config.get("llm_type").endswith(']'):
                #     for j, agent in enumerate(agents[AGENT_TYPES.CRITIC]):
                #         agent.
                for j, agent in enumerate(agents[AGENT_TYPES.CRITIC]):
                    agent.set_name(f"Agent {j+1}")
            else:
                agent_type = AGENT_TYPES.from_string(agent_config.get("agent_type", ""))
                if agent_type not in agents.keys():
                    agents[agent_type] = load_agent(agent_config)
                else:
                    if not isinstance(agents[agent_type], list):
                        agents[agent_type] = [agents[agent_type]]
                    agents[agent_type].append(load_agent(agent_config))
        # print('!!!!!!!!!!!!!!!The agents:!!!!!!!!!!!!!!!!!')
        # print(agents)
        # print('!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!')
        env_config["agents"] = agents

        env_config["task_description"] = task_config.get("task_description", "")
        env_config["max_rounds"] = task_config.get("max_rounds", 3)

        environment: BasicEnvironment = load_environment(env_config)

        return cls(environment=environment, task=task, data_iter=data_iter)

    def run(self):
        """Run the environment from scratch until it is done."""
        
        self.logs = []
        total_decision_making_process = []
        advice = "No advice yet."
        previous_plan = "No solution yet."
        for i, example in enumerate(tqdm(self.data_iter, total=len(self.data_iter.examples))):
            if i >= self.data_max_len and self.data_max_len > 0:
                break
            self.environment.reset()
            advice = "No advice yet."
            previous_plan = "No solution yet."
            self.environment.set_task_description(example["input"])
            while not self.environment.is_done():
                result, advice, previous_plan, logs, success, decision_making_process = asyncio.run(
                    self.environment.step(advice, previous_plan)
                )
                self.logs += logs
                total_decision_making_process += decision_making_process
        self.environment.report_metrics()
        self.save_result(previous_plan, result, self.environment.get_spend())
        return previous_plan, result, self.logs, total_decision_making_process

    def singleagent_thinking(self, preliminary_solution, advice) -> str:
        preliminary_solution = self.environment.solve(
            former_solution=preliminary_solution,
            critic_opinions=[(self.environment.evaluator, advice)],
        )
        return preliminary_solution

    def reset(self):
        self.environment.reset()

    def save_result(self, plan: str, result: str, spend: float):
        """Save the result to the result file"""
        result_file_path = "./results/" + self.task + ".txt"
        os.makedirs(os.path.dirname(result_file_path), exist_ok=True)
        with open(result_file_path, "w") as f:
            f.write("[Final Plan]\n" + plan + "\n\n")
            f.write("[Result]\n" + result)
            f.write(f"[Spent]\n${spend}")
