import re
from openai import OpenAI
import json

from SCOPE.assets.pddl_asset import *
import base64
from SCOPE.utils.symbolic_utils import getEnvObject
from SCOPE.assets.agent_prompt import ExploreAgent_prompt

class PerceiveAgent:
    def __init__(self,task,env_id,agent,api_key,save_dir):
        self.caption_message=[]
        self.pddl_message=[]
        self.exploreTimes = 0
        self.agent = agent
        if agent == "gpt-4o":
            self.client = OpenAI(api_key=api_key)
        else:
            self.client = OpenAI()
        self.envID = env_id
        self.exploredRoom=[]
        self.PDDL_problem={}
        self.charAtRoom=""
        self.roomPDDLlist=[]
        self.refineRoom=[]
        self.exp_mode=0
        self.extraInfo=[]
        self.Caption_prompt = ExploreAgent_prompt["Caption_prompt"]
        self.PDDL_prompt = ExploreAgent_prompt["PDDL_prompt"]
        self.save_dir=save_dir

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

    def encode_image(self,image_path):
        with open(image_path, "rb") as image_file:
            return base64.b64encode(image_file.read()).decode("utf-8")

    def jsonformat(self,answer):
        while True:
            try:
                answer = re.search(r'```json\n(.*?)\n```', answer, re.DOTALL).group(1)
                # 尝试解析 JSON 字符串
                data = json.loads(answer)
                return data
            except:
                self.pddl_message.append({
                    "role": "user",
                    "content": "your answer has a formatting error, please answer again strictly follow the JSON format!"
                })
                response = self.client.chat.completions.create(
                    model=self.agent,
                    messages=self.pddl_message
                )
                self.llmPrint(self.pddl_message[-1])
                self.pddl_message.append({"role": "assistant", "content": response.choices[0].message.content})
                self.llmPrint(self.pddl_message[-1])
                answer = response.choices[0].message.content

    def llmPrint(self,data):
        print(f"\033[32mPerceiveAgent:{data['role']}:\033[0m")
        print(data['content'])

    def logPrint(self,content):
        print(f"\033[34msystemlog: {content}\033[0m")

    def getRoomImage(self,roomName,batch_size=2):
        folder_path = f'{self.save_dir}/Env/Env_{self.envID}/{roomName}'
        # 获取文件夹中的所有文件
        image_files = [f for f in os.listdir(folder_path) if f.endswith(("png"))]
        # 读取所有图片
        batched_messages = []
        batch = {
            "role": "user",
            "content": []
        }
        for i, image_file in enumerate(image_files):
            i+=1
            image_path = os.path.join(folder_path, image_file)
            batch["content"].append({
                "type": "image_url",
                "image_url": {"url": f"data:image/jpeg;base64,{self.encode_image(image_path)}"},
            })
            if (i + 1) % batch_size == 0 or i == len(image_files) - 1:
                batched_messages.append(batch)
                batch = {
                    "role": "user",
                    "content": []
                }

        self.logPrint(f"{roomName} finished loading!")
        return batched_messages

    def check_TypesFormat(self,PDDL_problem_FromLLM):
        self.charAtRoom = PDDL_problem_FromLLM['room']
        state_ = PDDL_problem_FromLLM["states"]
        actionable_ = PDDL_problem_FromLLM["properties"]
        relations_ = PDDL_problem_FromLLM["relations"]
        error_answer = {
            "object": "",
            "states": "",
            "properties": "",
            "relations": ""
        }
        self.logPrint("checking for type and format")
        for i in PDDL_problem_FromLLM["objects"]:
            match = re.search('(.*?)_', i, re.DOTALL).group(1)
            if match not in self.recog_objects:
                error_answer["object"] += f"{match} not in 'Allowed objects name': {self.recog_objects}\n"

        for i in state_:
            match = re.search('\(not\((.*?)\)\)', i, re.DOTALL)
            obj=re.search("\s(.*?)\)",i,re.DOTALL).group(1)

            if obj not in PDDL_problem_FromLLM["objects"]:
                error_answer["states"] += f"""{obj} is declared in the "states", but not declared in the "objects"\n"""
            if match:
                pred = match.group(1).split(" ")[0]
            else:
                pred = re.search('\((.*?)\)', i, re.DOTALL).group(1).split(" ")[0]
            if pred not in states_predicates:
                error_answer["states"] += f"The '{pred}' predicate in the '{i}' is not a valid 'states predicate',[switchable] belong to 'properties predicate' .\n"

        for i in actionable_:
            match = re.search('\(not\((.*?)\)\)', i, re.DOTALL)
            if not re.search("\s(.*?)\)",i,re.DOTALL):
                error_answer["properties"] += f"""{i}exist format problem in "properties"\n"""
                continue
            else:
                obj=re.search("\s(.*?)\)",i,re.DOTALL).group(1)
            if obj not in PDDL_problem_FromLLM["objects"]:
                error_answer["properties"] += f"""{obj} is declared in the "properties", but must not be declared in the "objects"\n"""
            if match:
                error_answer[
                    "properties"] += f"{i} has error. Prohibit the use of 'not' to enrich 'properties field' descriptions.\n"
            else:
                pred = re.search('\((.*?)\)', i, re.DOTALL).group(1).split(" ")[0]
                if pred not in properties_predicates:
                    error_answer[
                        "properties"] += f"The '{pred}' predicate in the '{i}' is not a valid 'properties predicate'.\n"

        for i in relations_:
            match = re.search('\(not\((.*?)\)\)', i, re.DOTALL)
            if match:
                error_answer[
                    "relations"] += f"{i} has error. Prohibit the use of 'not' to enrich 'relations field' descriptions.\n"
            else:
                objlist = re.search("\((.*?)\)", i, re.DOTALL).group(1).split(" ")
                if (objlist[1] not in PDDL_problem_FromLLM["objects"]) and objlist[1]!=PDDL_problem_FromLLM["room"]:
                    error_answer["relations"] += f"""{objlist[1]} is not declared in the "objects" or "room"\n"""
                if (objlist[2] not in PDDL_problem_FromLLM["objects"]) and objlist[2]!=PDDL_problem_FromLLM["room"]:
                    error_answer["relations"] += f"""{objlist[2]} is not declared in the "objects" or "room"\n"""

                pred = re.search('\((.*?)\)', i, re.DOTALL).group(1).split(" ")[0]
                if pred not in relations_predicates:
                    error_answer[
                        "relations"] += f"The '{pred}' predicate in the '{i}' is not a valid 'relations predicate'.\n"

        if error_answer["object"] or error_answer["states"] or error_answer["properties"] or error_answer["relations"]:
            error_answer_return = {
                "role": "user",
                "content": error_answer["object"]+error_answer["states"] + error_answer["properties"] + error_answer[
                    "relations"] + "\n" + "Please according to caption regenerate again strictly according to the format of the 'perceive example' with json. "
            }
            return False, error_answer_return
        else:
            room_object = PDDL_problem_FromLLM["objects"]
            self.PDDL_problem = {
                "objects": {
                    "room": [self.charAtRoom],
                    "character": ["char_0"],
                    "object": room_object
                },
                "states": PDDL_problem_FromLLM["states"],
                "properties": PDDL_problem_FromLLM["properties"],
                "relations": list(set(PDDL_problem_FromLLM["relations"] + [f"(objAtRoom {x} {self.charAtRoom})" for x in room_object])),
            }
            return True,{}

    def convertRoomPDDLlist(self):
        self.wholePDDL_problem={
            "objects":{"room":[],"character":["char_0"],"object":[]},
            "states":[],"properties":[],"relations":[]
        }
        for PDDL_problem in self.roomPDDLlist:
            self.wholePDDL_problem["objects"]["room"]+=PDDL_problem["objects"]["room"]
            self.wholePDDL_problem["objects"]["object"] += PDDL_problem["objects"]["object"]
            self.wholePDDL_problem["states"] += PDDL_problem["states"]
            self.wholePDDL_problem["properties"] += PDDL_problem["properties"]
            self.wholePDDL_problem["relations"] += PDDL_problem["relations"] + [
                f"(objAtRoom {x} {self.charAtRoom})" for x in PDDL_problem["objects"]["object"]]
            self.wholePDDL_problem["relations"] = list(set(self.wholePDDL_problem["relations"]))
        if self.extraInfo:
            self.wholePDDL_problem["properties"]+=self.extraInfo
        # print(self.wholePDDL_problem)

    def updateWholePDDL(self,ratio=1):
        with open(self.pddl_problem_path,"r") as f:
            self.PDDL_problem = json.load(f)
        if ratio!=1:
            if self.charAtRoom in self.exploredRoom:
                for i,t in enumerate(self.roomPDDLlist):
                    if self.charAtRoom ==t["objects"]["room"]:
                        self.roomPDDLlist[i] = self.PDDL_problem
                        break
            else:
                self.roomPDDLlist.append(self.PDDL_problem)
            self.convertRoomPDDLlist()
            target_dir = f'{self.save_dir}/task/{self.taskFile}/Env_{self.envID}/ratio{ratio}'
            os.makedirs(target_dir, exist_ok=True)  # 确保目录存在
            if self.exp_mode == 0:
                with open(os.path.join(target_dir, f'raw_whole_pddl_{self.agent}.json'), "w") as f:
                    json.dump(self.wholePDDL_problem, f, indent=4)
            elif self.exp_mode == 1:
                with open(os.path.join(target_dir, f'whole_pddl_{self.agent}.json'), "w") as f:
                    json.dump(self.wholePDDL_problem, f, indent=4)
        else:
            self.roomPDDLlist.append(self.PDDL_problem)
            self.convertRoomPDDLlist()
            target_dir = f'{self.save_dir}/task/{self.taskFile}/Env_{self.envID}'
            os.makedirs(target_dir, exist_ok=True)  # 确保目录存在
            if self.exp_mode == 0:
                with open(os.path.join(target_dir, f'raw_whole_pddl_{self.agent}.json'), "w") as f:
                    json.dump(self.wholePDDL_problem, f, indent=4)

            elif self.exp_mode == 1:
                with open(os.path.join(target_dir, f'whole_pddl_{self.agent}.json'), "w") as f:
                    json.dump(self.wholePDDL_problem, f, indent=4)


    def gen_caption(self):
        group_image_info=self.getRoomImage(self.charAtRoom)
        caption=""
        for t,image_info in enumerate(group_image_info):
            self.caption_message.append({
                "role": "user",
                "content": self.Caption_prompt + f"you are now at {self.charAtRoom}. You should focus on describing these objects, 'Allowed objects name': {self.recog_objects}. Ignore floor and wall. Answer me strictly according to the example"
            })
            self.llmPrint(self.caption_message[-1])
            self.caption_message.append(image_info)
            response = self.client.chat.completions.create(
                model=self.agent,
                messages=self.caption_message
            )
            self.caption_message.pop(-1)
            self.caption_message.append({"role": "assistant", "content": response.choices[0].message.content})
            caption += response.choices[0].message.content+"\n"
            self.llmPrint(self.caption_message[-1])
            self.caption_message.pop(-1)
        with open(self.caption_path,"w") as text_file:
            print(caption)
            text_file.write(caption)

    def processPDDL(self,prompt):
        self.pddl_message.append({
            "role": "user",
            "content": prompt
        })
        self.llmPrint(self.pddl_message[-1])
        response = self.client.chat.completions.create(
            model=self.agent,
            messages=self.pddl_message
        )
        llm_reply = response.choices[0].message.content
        self.pddl_message.append({"role": "assistant", "content": llm_reply})
        self.llmPrint(self.pddl_message[-1])

        # 转换 LLM 输出为结构化 PDDL 描述
        pddl_problem = self.jsonformat(llm_reply)

        return pddl_problem

    def refineEnv(self,feedback="",tasktext="",ratio=1):
        self.caption_path = f"{self.save_dir}/Env/Env_{self.envID}/{self.charAtRoom}/caption_{self.agent}.txt"
        self.pddl_problem_path = f'{self.save_dir}/Env/Env_{self.envID}/{self.charAtRoom}/raw_pddl_{self.agent}_{ratio}.json'
        with open(self.caption_path, "r") as f:
            currentRoomCaption = f.read()
        with open(self.pddl_problem_path, "r") as f:
            currentRoomPDDL = json.load(f)
        prompt_content = "It is the current room PDDL:" + json.dumps(currentRoomPDDL, ensure_ascii=False)+ "\n"+f"It is the current room caption: {currentRoomCaption}\n"+f"It is the task description: {tasktext}\n"+f"It is the current feedback: {feedback}\n"+f"The following items are available for selection: {self.recog_objects}\n"+f"Focus on the feedback and prioritize relevant items from the task description to reduce the current perception gaps. "
        prompt_content+="""Respond strictly in JSON format.\nOnly include information that was not present in the previous PDDL description."
Example:
{
   'room': 'livingroom_12',
   'objects': ['tvstand_49','waterglass_33'],
   'states': ['(open_state box_45)'],
   'properties': ['(switchable tv_46)','(grabbable bananas_49)'],
   'relations': ['(onTop dishbowl_45 coffeetable_31)']
 }The fields 'Room', 'objects', 'states', 'properties', and 'relationships' must be included in the JSON output. Probit to describe [char_0]. The field "objects" should be a list of object ID strings, such as "sofa_123"\n
"""+f"You are currently in: {self.charAtRoom}."
        pddl_problem = self.processPDDL(prompt_content)
        pddl_problem["objects"] += currentRoomPDDL["objects"]["object"]
        pddl_problem["states"] += currentRoomPDDL["states"]
        pddl_problem["properties"] += currentRoomPDDL["properties"]
        pddl_problem["relations"] += currentRoomPDDL["relations"]
        pddl_problem["relations"] = list(set(pddl_problem["relations"]))
        self.pddl_problem=pddl_problem
        # 类型与格式检查
        self.logPrint("Start checking for type, format and duplicates")
        is_valid, feedback_msg = self.check_TypesFormat(self.pddl_problem)
        # 若检查不通过，进入反馈修复循环
        while not is_valid:
            pddl_problem = self.processPDDL(json.dumps(feedback_msg))
            pddl_problem["objects"] += currentRoomPDDL["objects"]["object"]
            pddl_problem["states"] += currentRoomPDDL["states"]
            pddl_problem["properties"] += currentRoomPDDL["properties"]
            pddl_problem["relations"] += currentRoomPDDL["relations"]
            pddl_problem["relations"] = list(set(pddl_problem["relations"]))
            self.pddl_problem = pddl_problem
            is_valid, feedback_msg = self.check_TypesFormat(self.pddl_problem)

        pddl_problem["relations"] += [f"(objAtRoom {x} {self.charAtRoom})" for x in pddl_problem["objects"]]
        dir_path = f'{self.save_dir}/task/{self.taskFile}/Env_{self.envID}/ratio{ratio}'
        os.makedirs(dir_path, exist_ok=True)  # 若目录不存在则创建
        self.pddl_problem_path = f'{self.save_dir}/task/{self.taskFile}/Env_{self.envID}/ratio{ratio}/raw_pddl_{self.agent}_{self.charAtRoom}.json'
        with open(self.pddl_problem_path, "w") as f:
            json.dump(self.PDDL_problem, f, indent=4)
        self.logPrint("Check SCOPEed!")
        self.updateWholePDDL(ratio)

    def gen_PDDL(self):
        """调用LLM生成并校验PDDL问题描述，确保其格式和语义正确。"""
        # 读取caption内容
        with open(self.caption_path, "r") as f:
            caption = f.read()
        prompt_content = (
                self.PDDL_prompt +
                f"you are now at {self.charAtRoom}. You must focus on describing these objects: {self.recog_objects}\n{self.charAtRoom} must only appear in ['room']"+caption
        )
        self.pddl_problem=self.processPDDL(prompt_content)
        self.logPrint("Start checking for type, format and duplicates")
        is_valid, feedback_msg = self.check_TypesFormat(self.pddl_problem)
        # 若检查不通过，进入反馈修复循环
        while not is_valid:
            self.pddl_problem=self.processPDDL(json.dumps(feedback_msg, ensure_ascii=False)+f"You must focus on describing these objects: {self.recog_objects} {self.charAtRoom} must only appear in ['room']")
            is_valid, feedback_msg = self.check_TypesFormat(self.pddl_problem)
        with open(self.pddl_problem_path, "w") as f:
            json.dump(self.PDDL_problem, f, indent=4)
        self.logPrint("Check SCOPEed!")

    def explore(self,room,ratio=1):
        self.exp_mode=1
        self.exploreTimes += 1
        self.charAtRoom =room
        self.recog_objects=getEnvObject(self.envID,self.charAtRoom)
        self.caption_path=f"{self.save_dir}/Env/Env_{self.envID}/{self.charAtRoom}/caption_{self.agent}.txt"
        if ratio==1:
            self.pddl_problem_path=f'{self.save_dir}/Env/Env_{self.envID}/{self.charAtRoom}/raw_pddl_{self.agent}.json'
        else:
            self.pddl_problem_path=f'{self.save_dir}/Env/Env_{self.envID}/{self.charAtRoom}/raw_pddl_{self.agent}_{ratio}.json'
        # caption stage:
        if not os.path.exists(self.caption_path):
            print("\n*****************gen caption stage:********************\n")
            self.gen_caption()
        # PDDL stage：
        if not os.path.exists(self.pddl_problem_path):
            print("\n*****************gen pddl stage:********************\n")
            self.gen_PDDL()

        if room in self.exploredRoom:
            if ratio!=1:
                if self.charAtRoom not in self.refineRoom:
                    self.refineRoom.append(self.charAtRoom)
                self.refineEnv(feedback={}, tasktext="", ratio=ratio)
        else:
            self.updateWholePDDL(ratio=ratio)
            self.exploredRoom.append(room)
        self.clear_Message()
        return self.wholePDDL_problem, self.exploreTimes

    def genRoomPDDL(self,room):
        self.exp_mode = 0
        self.exploreTimes += 1
        self.charAtRoom = room
        self.recog_objects=getEnvObject(self.envID,self.charAtRoom)
        self.caption_path=f"{self.save_dir}/Env/Env_{self.envID}/{self.charAtRoom}/caption_{self.agent}.txt"
        self.pddl_problem_path = f'{self.save_dir}/Env/Env_{self.envID}/{self.charAtRoom}/raw_pddl_{self.agent}.json'
        # caption stage:
        if not os.path.exists(self.caption_path):
            print("\n*****************gen caption stage:********************\n")
            self.gen_caption()
        # PDDL stage：
        if not os.path.exists(self.pddl_problem_path):
            print("\n*****************gen pddl stage:********************\n")
            self.gen_PDDL()
        self.clear_Message()

    def realTimeUpdate(self,feedback,tasktext="",ratio=1):
        if ratio!=1:
            if self.charAtRoom not in self.refineRoom:
                self.refineRoom.append(self.charAtRoom)
                if not os.path.exists(self.pddl_problem_path):
                    self.refineEnv(feedback=feedback,tasktext=tasktext,ratio=ratio)
        predicates = re.findall("\(.*\)", feedback["predicate"])
        flag=False
        if predicates:
            for predicate in predicates:
                predi = re.search("\((.*)\s", predicate).group(1)
                if predi in properties_predicates:
                    flag=True
                    print(predicate)
                    self.wholePDDL_problem["properties"].append(predicate)
                    self.extraInfo.append(predicate)
                elif predi in object_states_predicates:
                    flag=True
                    print(predicate)
                    self.wholePDDL_problem["states"].append(predicate)
                    self.extraInfo.append(predicate)
        if flag:
            # 有更改Env
            if ratio==1:
                target_dir = f'{self.save_dir}/task/{self.taskFile}/Env_{self.envID}'
            else:
                target_dir = f'{self.save_dir}/task/{self.taskFile}/Env_{self.envID}/ratio{ratio}'
            os.makedirs(target_dir, exist_ok=True)  # 确保目录存在
            with open(os.path.join(target_dir, f'whole_pddl_{self.agent}.json'), "w") as f:
                json.dump(self.wholePDDL_problem, f, indent=4)
            return True,self.wholePDDL_problem
        else:
            # 没有更改Env
            return False,self.wholePDDL_problem

    def preparePDDLRawWhole(self,ratio=1):
        folderPath = f"{self.save_dir}/Env/Env_{self.envID}"
        folderlist = os.listdir(folderPath)
        for folder in folderlist:
            full_folder_path = os.path.join(folderPath, folder)
            if not os.path.isdir(full_folder_path):
                continue  # 跳过不是目录的项
            if ratio == 1:
                path = os.path.join(full_folder_path, f"raw_pddl_{self.agent}.json")
            else:
                path = os.path.join(full_folder_path, f"raw_pddl_{self.agent}_{ratio}.json")
            with open(path, "r") as f:
                data = json.load(f)
                self.roomPDDLlist.append(data)

        self.convertRoomPDDLlist()
        if ratio==1:
            path=f"{self.save_dir}/Env/Env_{self.envID}/raw_whole_pddl_{self.agent}.json"
            if not os.path.exists(path):
                with open(path, "w") as f:
                    json.dump(self.wholePDDL_problem, f, indent=4)
        else:
            path=f"{self.save_dir}/Env/Env_{self.envID}/raw_whole_pddl_{self.agent}_{ratio}.json"
            if not os.path.exists(path):
                with open(path, "w") as f:
                    json.dump(self.wholePDDL_problem, f, indent=4)

    def clear_Message(self):
        self.pddl_message.clear()
        self.caption_message.clear()