import re
from openai import OpenAI
import json
from SCOPE.utils.symbolic_utils import getEnvRooms,merge_action_reasoning,merge_perceptual
from SCOPE.assets.pddl_asset import *
from SCOPE.assets.agent_prompt import PlanAgent_prompt,Feedback_Summary_prompt

class PlanAgent:
    def __init__(self,task,initRoom,envID,agent,api_key,save_dir,use_memory):
        self.message = []
        self.explore_Times = 0
        self.agent = agent
        if agent=="gpt-4o":
            self.client = OpenAI(api_key=api_key)
        else:
            self.client = OpenAI()
        self.goal=task["task_name"]
        self.description=task["prompt"]
        self.charAtRoom=initRoom
        self.plan_Times=0
        self.PDDL_problem = {}

        full_path = os.path.normpath(task["filepath"])
        parts = full_path.split(os.sep)
        self.taskFile = os.path.join(parts[1], parts[2])

        self.envID = envID
        self.NotExploredRoom = getEnvRooms(envID)
        self.develop_prompt = PlanAgent_prompt["prompt"]
        self.action_prompt = PlanAgent_prompt["action_prompt"]

        self.message.append({
            "role": "system",
            "content": self.develop_prompt
        })
        self.save_dir=save_dir
        self.getMemory(use_memory)


    def updateCharAtRoom(self,room):
        self.charAtRoom = room

    def jsonformat(self,answer):
        while True:
            try:
                answer_json_str = re.search(r'```json\n(.*?)\n```', answer, re.DOTALL).group(1)
                # 尝试解析 JSON 字符串
                data = json.loads(answer_json_str)

                return data
            except Exception as e:
                self.message.append({
                    "role": "user",
                    "content": f"Your previous answer contains an issue: {str(e)}"
                               "please answer again according to the format of the 'plan task example' with json. Do not include any comments or explanations in the code."
                })
                self.llmPrint()
                response = self.client.chat.completions.create(
                    model=self.agent,
                    messages=self.message
                )
                self.message.append({"role": "assistant", "content": response.choices[0].message.content})
                self.llmPrint()
                answer = response.choices[0].message.content



    def llmPrint(self):
        print(f"\033[32mPlanAgent:{self.message[-1]['role']}:\033[0m")
        print(self.message[-1]['content'])


    def plan(self,PDDL_problem,feedbackInfo="",ratio=1):
        if self.charAtRoom in self.NotExploredRoom:
            self.NotExploredRoom.remove(self.charAtRoom)
        supplementContent = f"{PDDL_problem}. It is your perception of the explored environment."

        if feedbackInfo:
            # 构造有反馈信息的再规划提示
            feedback_prompt = (
                f"\"{feedbackInfo}\" It is the feedback given by the PDDL validator to your answer. "
                "Please follow the previous example and combine it with the feedback from the PDDL validator "
                "to regenerate in JSON format!"
            )
            prompt_content = (
                f"{feedback_prompt}"+json.dumps(self.memory,indent=4)+
                f"'text_goal': {self.goal}\n"
                f"'text_instruction': 'I am currently in the {self.charAtRoom}. {self.description}'. "
                f"Not explored room: {self.NotExploredRoom}"
            )
        else:
            # 构造首次规划提示
            prompt_content = (
                    self.action_prompt  + json.dumps(self.memory,indent=4) +
                    f"'text_goal': {self.goal}\n"
                    f"'text_instruction': 'I am currently in the {self.charAtRoom}. {self.description}'. "
                    f"Not explored room: {self.NotExploredRoom}"
            )
        self.message.append({
            "role": "user",
            "content": supplementContent
        })
        self.message.append({
            "role": "user",
            "content": prompt_content
        })
        response = self.client.chat.completions.create(
            model=self.agent,
            messages=self.message
        )
        llm_response = response.choices[0].message.content
        self.message.pop(-2)
        self.message.append({"role": "assistant", "content": llm_response})
        self.llmPrint()

        # 转为结构化格式
        self.action_answer = self.jsonformat(llm_response)

        # 判断是否为“go to”指令 → 执行探索
        if re.search(r"go to (.*)", self.action_answer["choice"]):
            self.charAtRoom = re.search(r"go to (.*)", self.action_answer["choice"]).group(1)
            self.explore_Times += 1
            # 日志记录
            log_information.append({
                "Room":self.charAtRoom,
                "step": len(log_information),
                "feedbackInfo": feedbackInfo,
                "content": self.action_answer
            })
            # 写入日志文件
            if ratio==1:
                log_path = f"{self.save_dir}/task/{self.taskFile}/Env_{self.envID}/log_{self.agent}.json"
            else:
                log_path = f"{self.save_dir}/task/{self.taskFile}/Env_{self.envID}/log_{self.agent}_{ratio}.json"
            with open(log_path, "w") as f:
                json.dump(log_information, f, indent=4)
            return False, self.action_answer, self.charAtRoom

        # 非 go to 指令 → 执行实际计划
        self.plan_Times += 1
        # 日志记录
        log_information.append({
            "step": len(log_information),
            "feedbackInfo": feedbackInfo,
            "content": self.action_answer
        })
        # 写入日志文件
        if ratio == 1:
            log_path = f"{self.save_dir}/task/{self.taskFile}/Env_{self.envID}/log_{self.agent}.json"
        else:
            log_path = f"{self.save_dir}/task/{self.taskFile}/Env_{self.envID}/log_{self.agent}_{ratio}.json"
        with open(log_path, "w") as f:
            json.dump(log_information, f, indent=4)
        return True, self.action_answer, self.charAtRoom

    def getTimes(self):
        return self.explore_Times,self.plan_Times

    def clearMessage(self):
        self.message.clear()

    def isExploredFull(self):
        if self.NotExploredRoom:
            return False
        else:
            return True
    def getMemory(self,use_memory):
        if use_memory:
            with open("assets/symbolic_memory.json","r") as f:
                memory = json.load(f)
            self.memory = memory
        else:self.memory={}

    def evolveMemory(self):
        self.clearMessage()
        self.message.append({
            "role": "system",
            "content": Feedback_Summary_prompt["assets"]+Feedback_Summary_prompt["prompt"]
        })
        self.message.append({
            "role": "user",
            "content": "before:"+json.dumps(self.memory,indent=4)+"current trajectory: "+json.dumps(log_information,indent=4)
        })
        response = self.client.chat.completions.create(
            model=self.agent,
            messages=self.message
        )
        answer = response.choices[0].message.content
        self.message.append({"role": "assistant", "content": answer})
        self.llmPrint()

        while True:
            try:
                answer_json_str = re.search(r'```json\n(.*?)\n```', answer, re.DOTALL).group(1)
                # 尝试解析 JSON 字符串
                data = json.loads(answer_json_str)
                break
            except Exception as e:
                self.message.append({
                    "role": "user",
                    "content": f"Your previous answer contains an issue: {str(e)}"
                })
                self.llmPrint()
                response = self.client.chat.completions.create(
                    model=self.agent,
                    messages=self.message
                )
                self.message.append({"role": "assistant", "content": response.choices[0].message.content})
                self.llmPrint()
                answer = response.choices[0].message.content

        merged_memory = {
            "Action_Reasoning_Abstraction": merge_action_reasoning(
                self.memory["Action_Reasoning_Abstraction"],
                data["Action_Reasoning_Abstraction"]
            ),
            "Perceptual_World_Abstraction": merge_perceptual(
                self.memory["Perceptual_World_Abstraction"],
                data["Perceptual_World_Abstraction"]
            )
        }
        with open("assets/symbolic_memory.json", "w") as f:
            json.dump(merged_memory, f, indent=4)
