import copy
import inspect
from demos.ipc.src.household.household_environment import HouseholdEnvironment
from demos.ipc.src.household.household_environment_state import HouseholdEnvironmentState
from tp_lodge.motion_planning.motion_simulator import MotionSimulator
import ast

from tp_lodge.motion_planning.motion_validator import (
    MotionSimulationErrorReason,
    MotionSimulationException,
    MotionSimulationResponseCode,
)


class HouseholdMotionSimulator(MotionSimulator[HouseholdEnvironmentState]):

    def __init__(self, env: HouseholdEnvironment):
        super().__init__()
        # self._init_env_state = copy.deepcopy(env.state)
        self._init_env_state = None
        self.env = env

    def init_state(self, seed: int):
        # only one environment
        self.env.state = copy.deepcopy(self._init_env_state)  # force reset env
        return super().init_state(seed)  # save as new start

    def _run_motion(self, motion: str) -> bool:
        func_call = ast.parse(motion, mode="eval").body
        if not isinstance(func_call, ast.Call):
            raise ValueError(f"Motion is not a function call: {motion}")
        func_name = func_call.func.id
        f_args = [ast.literal_eval(arg) for arg in func_call.args]
        # just ignore kwargs
        f_args += [ast.literal_eval(kw.value) for kw in func_call.keywords]
        f_kwargs = {}

        if func_name == "put_an_object_on_or_in_a_furniture_piece_or_an_appliance":
            f = self.env.put_object
        elif func_name == "pick_up_an_object_on_or_in_a_furniture_piece_or_an_appliance":
            f = self.env.pick_up_object
        elif func_name == "pick_up_an_object_on_or_in_a_small_receptacle":
            f = self.env.pick_up_object_from_receptacle
        elif func_name == "put_an_object_onto_or_into_a_small_receptacle":
            f = self.env.put_object_in_receptacle
        elif func_name == "go_to_a_furniture_piece_or_an_appliance":
            f = self.env.move_robot
        elif func_name == "open_a_furniture_piece_or_an_appliance":
            f_kwargs["door_open"] = True
            f = self.env.set_furniture_door_state
        elif func_name == "close_a_furniture_piece_or_an_appliance":
            f_kwargs["door_open"] = False
            f = self.env.set_furniture_door_state
        elif func_name == "open_a_small_receptacle":
            f_kwargs["receptacle_open"] = True
            f = self.env.set_receptacle_open_state
        elif func_name == "close_a_small_receptacle":
            f_kwargs["receptacle_open"] = False
            f = self.env.set_receptacle_open_state
        elif func_name == "mash_food_with_a_blender":
            f = self.env.mash_food
        elif func_name == "heat_food_with_a_microwave":
            f = self.env.heat_food_with_a_microwave
        elif func_name == "heat_food_with_pan":
            f = self.env.heat_food_with_pan
        elif func_name == "transfer_food_from_one_small_receptacle_to_another":
            f = self.env.transfer_food
        elif func_name == "slice_objects":
            f = self.env.slice_object
        elif func_name == "wash_an_object":
            f = self.env.wash_object
        elif func_name == "wipe_a_surface":
            f = self.env.wipe_surface
        elif func_name == "vacuum_a_carpet":
            f = self.env.vacuum_carpet
        elif func_name == "empty_a_vacuum_cleaner":
            f = self.env.empty_vacuum_cleaner
        elif func_name == "stack_objects":
            f = self.env.stack_objects
        elif func_name == "unstack_objects":
            f = self.env.unstack_objects
        elif func_name == "toggle_a_small_appliance_on":
            f_kwargs["appliance_on"] = True
            f = self.env.toggle_appliance
        elif func_name == "toggle_a_small_appliance_off":
            f_kwargs["appliance_on"] = False
            f = self.env.toggle_appliance
        else:
            assert False, f"Unsupported function: {func_name}"
            raise ValueError(f"Unsupported function: {func_name}")

        params = dict(inspect.signature(f).parameters)
        for f_kwarg in f_kwargs.keys():
            if f_kwarg not in params:
                raise ValueError(f"Function {func_name} does not accept keyword argument {f_kwarg}")
            else:
                del params[f_kwarg]

        if len(f_args) != len(params):
            raise MotionSimulationException(
                expected="",
                ground_truth="",
                message=f"Function {func_name} expects {len(params)} arguments, but got {len(f_args)}",
                code=MotionSimulationResponseCode.PDDL_PY_TRANSLATION,
            )

        f(*f_args, **f_kwargs)

        # check if the function exists

    def _get_state(self) -> HouseholdEnvironmentState:
        return self.env.state

    def _set_state(self, new_state: HouseholdEnvironmentState):
        self.env.state = new_state
