import openai
import jsonlines
import traceback
from openai import OpenAI
from typing import Any
import re
import logging
import time
import random


def find_obj(controller, obj_type):
    objects = controller.last_event.metadata["objects"]
    for obj in objects:
        if obj["objectType"] == obj_type:
            return obj
    return None


def find_obj_by_type(objects, obj_type):
    for obj in objects:
        if obj["objectType"] == obj_type:
            return obj
    return None


def load_dataset(data_dict, name):
    abstract = True if name == "abstract" else False
    file_path = data_dict[name]
    # keys = ["instruction", "scene_name", "final_state"]
    with jsonlines.open(file_path) as reader:
        data = [line for line in reader]
    
    return_data = []
    if abstract:
        for d in data:
            # import pdb; pdb.set_trace()
            d1, d2, d3, d4 = d.copy(), d.copy(), d.copy(), d.copy()
            d1['instruction'] = d['instruction'][0]
            d2['instruction'] = d['instruction'][1]
            d3['instruction'] = d['instruction'][2]
            d4['instruction'] = d['instruction'][3]
            return_data.append(d1)
            return_data.append(d2)
            return_data.append(d3)
            return_data.append(d4)
    else:
        return_data = data
    
    return return_data


def gen_low_level_plan(high_level_plan: str, client: OpenAI, model="gpt-4o-mini"):
    sys_prompt = "You are a helpful assistant for a home robot. You are given a high-level plan and need to convert it into a low-level plan."
    prompt = f"""Your task is to rewrite a sequence of high-level plans into a sequence of low-level plan. Each low-level plan has its standard format. Here is the explanation:

1. find obj:
Find the object and the agent will be close to the object. The object needs to be visible.

2. pick obj:
Pick up the object close to the agent. The object needs to be visible and the agent's hand must be clear of obstruction or the action will fail. Picked up objects can also obstruct the Agent's view of the environment since the Agent's hand is always in camera view, so know that picking up larger objects will obstruct the field of vision.

3. put receptacle:
Put down the object that the agent holds into the target receptacle.

4. open obj:
Open the openable object.

5. close obj:
Close the openable object.

6. slice obj:
Slice the sliceable object directly if the agent is close to the object and need not to hold the object. The object will be turned into several new sliced objects called objSliced. But the egg will be broken if sliced.

7. turn on obj:
Turn on the toggleable object if the agent is close to the object.

8. turn off obj:
Turn off the toggleable object if the agent is close to the object.

9. drop:
Drop the pickable object the agent holds. If the object is breakable, the object will be broken after being dropped.

10. throw:
Throw the pickable object the agent holds. If the object is breakable, the object will be broken after being thrown.

11. break obj:
Break the breakable object directly if the agent is close to the object and does not need to hold the object.

12. pour:
Rotate the pickable object the agent holds 90 degrees from the global upward axis. If an object is filled with one of the liquid types, the object will automatically empty itself because the liquid has “spilled.”

13. cook obj:
Cook the cookable object directly if the agent is close to the object and does not need to hold the object. If the cookable object interacts with objects that are heat sources, the object will be turned to the cooked state without using the cook action.

14. dirty obj:
Dirty the dirtyable object directly if the agent is close to the object and does not need to hold the object. 

15. clean obj:
Clean the dirty object directly if the agent is close to the object and does not need to hold the object. 

16. fillLiquid obj water/coffee/wine:
Fill the fillable object with one type of liquid among water/coffee/wine if the agent is close to the object and does not need to hold the object.

17. emptyLiquid obj:
Empty the filled object if the agent is close to the object and does not need to hold the object.

Requirements:
- The low-level plan should be a one of the above formats, one verb one object, without the description of the object.
- if the input high-level plan cannot be converted to a low-level plan, return "Cannot convert the high-level plan to a low-level plan."

Examples:
- Input: "Turn to face the counter to the left of the fridge.\nWalk to the counter.\nPick up the knife from the counter.\nTurn around and walk to the sink.\nWash the knife in the sink.\nDry the knife with a towel.\nReturn to the counter.\nPick up the bread from the counter.\nTurn to face the fridge.\nOpen the fridge.\nPlace the bread inside the fridge.\nClose the fridge."
- Output: "find knife\npick knife\nfind sink\nput sink\nfind bread\nfind fridge\npick bread\nopen fridge\nput fridge\nclose fridge"

Here is the high-level plan you need to convert:
{high_level_plan}

Remember, never generate plans that are not in the standard format, like turn to!
 
Your low-level plan, remember to follow the standard format:
    """
    low_level_plan = call_gpt(client, model, prompt, sys_prompt)
    low_level_plan = [line.strip() for line in low_level_plan.split("\n") if line.strip()]
    print("Get low-level plan:", low_level_plan)
    return low_level_plan


def execute_low_level_plan(low_level_plan: list, planner):
    num_success_steps = 0
    planner.restore_scene()
    for plan in low_level_plan:
        try:
            ret_dict = planner.llm_skill_interact(plan)
            print(ret_dict)
            print('-' * 50)
            if ret_dict["success"]:
                num_success_steps += 1
        except Exception as e:
            traceback.print_exc()  # 打印完整的异常堆栈信息
            continue
    if len(low_level_plan) == 0:
        sr_step = 0
    else:
        sr_step = num_success_steps / len(low_level_plan)
    return planner.env.last_event.metadata, num_success_steps / len(low_level_plan)


def execute_low_level_plan_with_assert(low_level_plan: list, planner):
    num_total_steps = 0
    num_success_steps = 0
    planner.restore_scene()
    for i, plan in enumerate(low_level_plan):
        
        if plan.startswith("assert") and low_level_plan[i+1].startswith("else"):
            
            try:
                condition = re.search(r'assert\s+(.*?)(?=,)', plan).group(1)
                match = re.match(r"(\w+)\['(\w+)'\]\s*==\s*(True|False)", condition)
                if match:
                    object_name = match.group(1)
                    attribute = match.group(2)
                    boolean_value = match.group(3)
                    print('-' * 50)
                    print(f"assert {object_name}['{attribute}'] == {boolean_value}")
                    print('-' * 50)
                    object_attribute = find_obj_by_type(planner.env.last_event.metadata["objects"], object_name)[attribute]
                    if object_attribute != boolean_value:
                        num_total_steps += 1
                        ret_dict = planner.llm_skill_interact(low_level_plan[i+1])
                        if ret_dict["success"]:
                            num_success_steps += 1
                    
            except Exception as e:
                traceback.print_exc()
                continue 
            
        else:
            
            try:
                num_total_steps += 1
                ret_dict = planner.llm_skill_interact(plan)
                print(ret_dict)
                print('-' * 50)
                if ret_dict["success"]:
                    num_success_steps += 1
                    
            except Exception as e:
                traceback.print_exc()
            
    return planner.env.last_event.metadata, num_success_steps / num_total_steps


def all_objs(controller):
    objects = controller.last_event.metadata["objects"]
    objects_types = [obj["objectType"] for obj in objects]
    return list(set(objects_types))


def retry_with_exponential_backoff(
        func: Any,
        initial_delay: float = 5,
        exponential_base: float = 2,
        jitter: bool = True,
        max_retries: int = 100,
        errors: Any = (
                openai.RateLimitError,
                openai.APIConnectionError,
                openai.APIError,
                openai.Timeout
        ),
        # errors: Any = (
        #         openai.error.RateLimitError, openai.error.ServiceUnavailableError,
        #         openai.error.APIConnectionError, openai.error.APIError, openai.error.Timeout
        # ),
) -> Any:
    """A wrapper. Retrying a function with exponential backoff."""

    def wrapper(*args, **kwargs):  # type: ignore
        # Initialize variables
        num_retries = 0
        delay = initial_delay
        # Loop until a response or max_retries is hit or an exception is raised.
        while True:
            try:
                print('\033[91mcalling gpt...\033[0m')
                return func(*args, **kwargs)

            # Retry on specified errors

            except tuple(errors) as exce:
            # except Exception as exce:
                logging.info(exce._message)
                # Increment retries
                num_retries += 1
                # Check if max retries has been reached
                if num_retries > max_retries:
                    raise Exception(exce) from exce(
                        f"Maximum number of retries ({max_retries}) exceeded."
                    )
                # Increment the delay
                delay *= exponential_base * (1 + jitter * random.random())
                # Sleep for the delay
                time.sleep(delay)
            # Raise exceptions for any errors not specified
            except Exception as e:
                raise e

    return wrapper



@retry_with_exponential_backoff
def call_gpt(client: OpenAI, model, prompt, system_prompt="You are a helpful assistant.", temperature=0.2):
    response = client.chat.completions.create(
        model=model,
        messages=[
            {"role": "system", "content": system_prompt},
            {"role": "user", "content": prompt},
        ],
        temperature=temperature,
    )
    output = response.choices[0].message.content.strip()
    print('\033[91mcalling finished...\033[0m')
    return output


def call_vllm(prompt, port=9095, model_name="llama3-8b-instruct-hf"):
    client = OpenAI(base_url=f"http://localhost:{port}/v1")
    completion = client.chat.completions.create(
        model=model_name,
        messages=[
            {"role": "user", "content": prompt},
            ],
        temperature=0.0,
        max_tokens=4096,
        )
    return completion.choices[0].message.content


def call_deepseek(prompt):

    client = OpenAI(api_key="", base_url="https://api.deepseek.com")
    
    response = client.chat.completions.create(
        model="deepseek-chat",
        messages=[
                {"role": "system", "content": "You are a helpful assistant"},
                {"role": "user", "content": prompt}
            ],
        stream=False
    )
    print('\033[91mcalling finished...\033[0m')
    return response.choices[0].message.content


if __name__ == "__main__":
    pass