import copy
from typing import Optional, Tuple
from demos.ipc.src.household.household_environment_state import HouseholdEnvironmentState


class HouseholdEnvironment:
    state: HouseholdEnvironmentState

    def _furniture_reachable(self, furniture_obj_id: str):
        target_furniture = self.state.furniture_appliances.get(furniture_obj_id)
        if target_furniture is None:
            raise ValueError(f"Furniture {furniture_obj_id} does not exist")

        if target_furniture["properties"]["openable"]:
            if not target_furniture["state"]["door_open"]:
                raise ValueError(f"Door of {furniture_obj_id} is not open")
        # else:
        # assert target_furniture["properties"]["has_flat_surface"]

    def _receptacle_reachable(self, receptacle_obj_id: str):
        receptacle = self.state.household_objects[receptacle_obj_id]
        self._furniture_reachable(receptacle["location"])

        # receptacle must be opened
        if receptacle["properties"]["openable"]:
            if not receptacle["state"]["is_open"]:
                raise ValueError(f"Receptacle {receptacle_obj_id} is not opened")

        if receptacle["properties"]["stackable"]:
            if receptacle["state"]["stacked_on"] is not None:
                # receptacle must not be stacked on top of other objects
                raise ValueError(f"Receptacle {receptacle_obj_id} is stacked on top of another object")

    def _object_stacked_on(self, obj_id: str) -> Optional[str]:
        for other_obj_id, other_obj in self.state.household_objects.items():
            if other_obj["properties"]["stackable"] and other_obj["state"]["stacked_on"] == obj_id:
                return other_obj_id
        return None

    def _object_reachable(self, obj_id: str):
        obj = self.state.household_objects[obj_id]
        if obj["location"] is None:
            raise ValueError(f"Object {obj_id} is not located in any furniture or appliance")
        if obj["location"] in self.state.household_objects:
            self._receptacle_reachable(obj["location"])
        else:
            self._furniture_reachable(obj["location"])

        # Check if the object is pickupable
        if not obj["properties"]["pickupable"]:
            raise ValueError(
                f"Object {obj_id} is not pickupable. You can pick up the receptacle or transfer the object to another receptacle."
            )

        if obj["properties"]["stackable"]:
            object_on_top = self._object_stacked_on(obj_id)
            if object_on_top is not None:
                raise ValueError(f"Object {object_on_top} is stacked on top of {obj_id}")

    def _robot_at(self, obj_id: str):
        furniture_id = self.get_furniture_applicant_of_object(obj_id)

        if furniture_id != self.state.robot_location:
            raise ValueError(
                f"Robot is at location `{self.state.robot_location}`, but object {obj_id} is at `{furniture_id}`"
            )

    def get_furniture_applicant_of_object(self, obj_id: str) -> str:
        if obj_id in self.state.furniture_appliances:
            return obj_id

        # Get the object's location
        if obj_id not in self.state.household_objects:
            raise ValueError(f"Object {obj_id} is not on/in a furniture or appliance")
        obj_location = self.state.household_objects[obj_id]["location"]

        if obj_location is None:
            raise ValueError(f"Object {obj_id} is not on/in a furniture or appliance")

        return self.get_furniture_applicant_of_object(obj_location)

    def go_to_furniture(self, target_furniture: str):
        """Move the robot to a furniture piece or appliance"""
        # Check if the target furniture exists
        if target_furniture not in self.state.furniture_appliances:
            raise ValueError(f"Furniture {target_furniture} does not exist")

        # Move the robot
        self.state.robot_location = target_furniture

    def pick_up_object_from_receptacle(self, obj_id: str, receptacle_obj_id: str):
        if obj_id not in self.state.household_objects:
            raise ValueError(f"Object {obj_id} is not a household object")
        if receptacle_obj_id not in self.state.household_objects:
            raise ValueError(f"Receptacle {receptacle_obj_id} is not a household object")
        if not self.state.gripper_empty:
            raise ValueError("Robot's gripper is not empty")

        obj_data = self.state.household_objects[obj_id]

        if obj_data["location"] != receptacle_obj_id:
            raise ValueError(f"Object {obj_id} is not located in receptacle {receptacle_obj_id}")

        self._robot_at(obj_id)
        self._object_reachable(obj_id)

        if not self.state.household_objects[receptacle_obj_id]["properties"]["is_receptacle"]:
            raise ValueError(f"{receptacle_obj_id} is not a receptacle")

        if obj_data["state"]["stacked_on"] is not None:
            raise ValueError(f"Cannot pick up {obj_id} because it is stacked on top of another object")

        if self._object_stacked_on(obj_id) is not None:
            raise ValueError(f"Cannot pick up {obj_id} because another object is stacked on top of it")

        # Update the state
        obj_data["location"] = None
        self.state.held_object = obj_id

    def put_object_in_receptacle(self, obj_id: str, receptacle_obj_id: str):
        assert obj_id in self.state.household_objects
        assert receptacle_obj_id in self.state.household_objects

        if obj_id != self.state.held_object:
            raise ValueError(f"Robot is not holding {obj_id}")

        if not self.state.household_objects[receptacle_obj_id]["properties"]["is_receptacle"]:
            raise ValueError(f"{receptacle_obj_id} is not a receptacle")

        self._robot_at(receptacle_obj_id)
        self._receptacle_reachable(receptacle_obj_id)

        # state update
        self.state.held_object = None
        self.state.household_objects[obj_id]["location"] = receptacle_obj_id

    def put_object(self, obj_id: str, target_furniture_id: str):
        if target_furniture_id not in self.state.furniture_appliances:
            raise ValueError(f"{target_furniture_id} is not a furniture or appliance")

        if obj_id not in self.state.household_objects:
            raise ValueError(f"{obj_id} is not a household object")

        # Check if the robot is holding an object
        if self.state.held_object != obj_id:
            raise ValueError(f"Robot is not holding {obj_id}")

        self._robot_at(target_furniture_id)

        # check that we can put sth on/in target_furniture
        self._furniture_reachable(target_furniture_id)

        # Update the state
        self.state.held_object = None
        self.state.household_objects[obj_id]["location"] = target_furniture_id

    def pick_up_object(self, obj_id: str, furniture_id: str):
        if not self.state.gripper_empty:
            raise ValueError(f"Robot is holding an object")

        self._object_reachable(obj_id)

        obj = self.state.household_objects[obj_id]
        if obj["location"] != furniture_id:
            raise ValueError(f"Object {obj_id} is not on/in {furniture_id}")

        obj_furniture_id = self.get_furniture_applicant_of_object(obj_id)
        assert obj_furniture_id == furniture_id, f"Object {obj_id} is not on/in {furniture_id}"
        self._robot_at(obj_id)

        if obj["state"]["stacked_on"] is not None:
            raise ValueError(f"Cannot pick up {obj_id} because it is stacked on top of another object")

        if self._object_stacked_on(obj_id) is not None:
            raise ValueError(f"Cannot pick up {obj_id} because another object is stacked on top of it")

        # state change
        self.state.held_object = obj_id
        self.state.household_objects[obj_id]["location"] = None

    def move_robot(self, target_furniture: str):
        """Move the robot to a furniture piece or appliance"""

        if target_furniture not in self.state.furniture_appliances:
            raise ValueError(f"{target_furniture} is not a furniture or appliance")

        # Move the robot
        self.state.robot_location = target_furniture

    def set_furniture_door_state(self, furniture_obj_id: str, door_open: bool):
        if furniture_obj_id not in self.state.furniture_appliances:
            raise ValueError(f"Furniture {furniture_obj_id} does not exist")

        self._robot_at(furniture_obj_id)

        furniture_appliance = self.state.furniture_appliances[furniture_obj_id]

        if not furniture_appliance["properties"]["openable"]:
            raise ValueError(f"Furniture {furniture_obj_id} does not have a door")

        if furniture_appliance["state"]["door_open"] == door_open:
            raise ValueError(f"Door of {furniture_obj_id} is already {'open' if door_open else 'closed'}")

        self.state.furniture_appliances[furniture_obj_id]["state"]["door_open"] = door_open

    def set_receptacle_open_state(self, receptacle_obj_id: str, receptacle_open: bool):
        if receptacle_obj_id not in self.state.household_objects:
            raise ValueError(f"Receptacle {receptacle_obj_id} does not exist")

        receptacle = self.state.household_objects[receptacle_obj_id]
        if receptacle["location"] is None:
            raise ValueError(f"Receptacle {receptacle_obj_id} cannot be opened while holding it")

        self._robot_at(receptacle_obj_id)

        receptacle_location = receptacle["location"]
        if receptacle_location not in self.state.furniture_appliances:
            raise ValueError(f"Furniture {receptacle_location} does not exist")

        self._furniture_reachable(receptacle_location)

        if not receptacle["properties"]["openable"]:
            raise ValueError(f"Receptacle {receptacle_obj_id} is not openable")

        if receptacle["state"]["is_open"] == receptacle_open:
            raise ValueError(f"Receptacle {receptacle_obj_id} is already {'open' if receptacle_open else 'closed'}")

        self.state.household_objects[receptacle_obj_id]["state"]["is_open"] = receptacle_open

    def mash_food(self, obj_id: str, blender_id: str):
        self._robot_at(blender_id)

        if blender_id not in self.state.household_objects:
            raise ValueError(f"{blender_id} is not a household object")

        blender = self.state.household_objects[blender_id]
        if blender["type"] != "blender":
            raise ValueError(f"Furniture {blender_id} is not a blender")

        object = self.state.household_objects[obj_id]
        if object["location"] != blender_id:
            raise ValueError(f"Object {obj_id} is not in the blender {blender_id}")

        if not object["state"]["sliced"]:
            raise ValueError(f"Object {obj_id} is not sliced")

        # state change
        object["state"]["mashed"] = True
        object["state"]["sliced"] = False

    def heat_food_with_a_microwave(self, obj_id: str, microwave_id: str):
        self._robot_at(microwave_id)

        if microwave_id not in self.state.furniture_appliances:
            raise ValueError(f"{microwave_id} is not a microwave")

        furniture = self.state.furniture_appliances[microwave_id]

        if obj_id not in self.state.household_objects:
            raise ValueError(f"{obj_id} is not a household object")

        obj = self.state.household_objects[obj_id]

        if furniture["type"] != "microwave":
            raise ValueError(f"Furniture {microwave_id} is not a microwave")

        obj_location = self.get_furniture_applicant_of_object(obj_id)
        if obj_location != microwave_id:
            raise ValueError(f"Object {obj_id} is not in the microwave {microwave_id}")

        if obj["location"] == microwave_id:
            raise ValueError(f"Object {obj_id} is not in a receptacle")

        if furniture["state"]["door_open"]:
            raise ValueError(f"Microwave {microwave_id} is not closed")

        if not obj["properties"]["heatable"]:
            raise ValueError(f"Object {obj_id} is not heatable")

        # state change
        obj["state"]["heated"] = True
        obj["properties"]["pickupable"] = False

    def heat_food_with_pan(self, obj_id: str, pan_id: str):
        self._robot_at(pan_id)

        if pan_id not in self.state.household_objects:
            raise ValueError(f"{pan_id} is not a pan")

        pan = self.state.household_objects[pan_id]
        if pan["type"] != "pan":
            raise ValueError(f"Furniture {pan_id} is not a pan")

        obj = self.state.household_objects[obj_id]
        if obj["location"] != pan_id:
            raise ValueError(f"Object {obj_id} is not in the pan {pan_id}")

        obj_location = self.state.furniture_appliances[pan["location"]]
        if obj_location["type"] != "stove_burner":
            raise ValueError(f"Pan {pan_id} is not on a stove")

        if not obj["properties"]["heatable"]:
            raise ValueError(f"Object {obj_id} is not heatable")

        # state change
        obj["state"]["heated"] = True
        obj["properties"]["pickupable"] = False

    def transfer_food(self, obj_id: str, source_receptacle_id: str, target_receptacle_id: str):
        if obj_id not in self.state.household_objects:
            raise ValueError(f"Object {obj_id} does not exist")
        if source_receptacle_id not in self.state.household_objects:
            raise ValueError(f"Source receptacle {source_receptacle_id} does not exist")
        if target_receptacle_id not in self.state.household_objects:
            raise ValueError(f"Target receptacle {target_receptacle_id} does not exist")

        if not self.state.gripper_empty:
            raise ValueError(f"Robot is holding an object")

        self._robot_at(source_receptacle_id)
        self._receptacle_reachable(source_receptacle_id)
        self._receptacle_reachable(target_receptacle_id)

        obj = self.state.household_objects[obj_id]
        source_receptacle = self.state.household_objects[source_receptacle_id]
        target_receptacle = self.state.household_objects[target_receptacle_id]

        if source_receptacle["location"] != target_receptacle["location"]:
            raise ValueError(
                f"Receptacles {source_receptacle_id} and {target_receptacle_id} are not on the same furniture"
            )

        if obj["location"] != source_receptacle_id:
            raise ValueError(f"Object {obj_id} is not in the source receptacle {source_receptacle_id}")

        # state change
        obj["location"] = target_receptacle_id

    def slice_object(self, obj_id: str, knife_id: str):
        if not self.state.held_object == knife_id:
            raise ValueError(f"Robot is not holding the knife {knife_id}")

        if self.state.held_object == obj_id:
            raise ValueError(f"Robot is holding the object {obj_id}, cannot slice it")

        self._robot_at(obj_id)

        obj = self.state.household_objects[obj_id]
        knife = self.state.household_objects[knife_id]
        if not obj["properties"]["sliceable"]:
            raise ValueError(f"Object {obj_id} is not sliceable")

        if knife["type"] != "knife":
            raise ValueError(f"Object {knife_id} is not a knife")

        if obj["location"] not in self.state.household_objects:
            raise ValueError(f"Object {obj_id} is not on a cutting board")
        cutting_board = self.state.household_objects[obj["location"]]
        if cutting_board["type"] != "cutting_board":
            raise ValueError(f"Object {obj['location']} is not a cutting board")

        if not obj["properties"]["sliceable"]:
            raise ValueError(f"Object {obj_id} is not sliceable")

        if obj["state"]["sliced"]:
            raise ValueError(f"Object {obj_id} is already sliced")

        # state change
        obj["state"]["sliced"] = True

    def wash_object(self, obj_id: str):
        if not self.state.held_object == obj_id:
            raise ValueError(f"Robot is not holding the object {obj_id}")

        robot_location = self.state.furniture_appliances[self.state.robot_location]

        if robot_location["type"] != "sink_basin":
            raise ValueError(f"Robot is not at a sink or basin")

        obj = self.state.household_objects[obj_id]

        if not obj["properties"]["washable"]:
            raise ValueError(f"Object {obj_id} is not washable")

        if obj["state"]["clean"]:
            raise ValueError(f"Object {obj_id} is already washed")

        # state change
        obj["state"]["clean"] = True

    def wipe_surface(self, surface_id: str, cloth_id: str):
        if not self.state.held_object == cloth_id:
            raise ValueError(f"Robot is not holding the cloth {cloth_id}")

        self._robot_at(surface_id)

        surface = self.state.furniture_appliances[surface_id]
        cloth = self.state.household_objects[cloth_id]

        if not surface["properties"]["wipeable"]:
            raise ValueError(f"Surface {surface_id} is not wipeable")

        if cloth["type"] != "cloth":
            raise ValueError(f"Object {cloth_id} is not a cloth")

        if not cloth["state"]["clean"]:
            raise ValueError(f"Cloth {cloth_id} is not clean")

        if surface["state"]["clean"]:
            raise ValueError(f"Surface {surface_id} is already clean")

        # state change
        surface["state"]["clean"] = True
        cloth["state"]["clean"] = False

    def vacuum_carpet(self, vacuum_id: str, carpet_id: str):
        if not self.state.held_object == vacuum_id:
            raise ValueError(f"Robot is not holding the vacuum {vacuum_id}")

        self._robot_at(carpet_id)

        carpet = self.state.furniture_appliances[carpet_id]
        vacuum = self.state.household_objects[vacuum_id]

        if vacuum["type"] != "handheld_vacuum":
            raise ValueError(f"Object {vacuum_id} is not a vacuum")
        if not vacuum["state"]["empty_dust_bin"]:
            raise ValueError(f"Vacuum {vacuum_id} has a full dust bin")

        if carpet["type"] != "carpet":
            raise ValueError(f"Object {carpet_id} is not a carpet")

        if carpet["state"]["clean"]:
            raise ValueError(f"Carpet {carpet_id} is already clean")

        # state change
        carpet["state"]["clean"] = True
        vacuum["state"]["empty_dust_bin"] = False

    def empty_vacuum_cleaner(self, vacuum_id: str, garbage_can_id: str):
        if not self.state.held_object == vacuum_id:
            raise ValueError(f"Robot is not holding the vacuum {vacuum_id}")

        self._robot_at(garbage_can_id)

        garbage_can = self.state.furniture_appliances[garbage_can_id]
        vacuum = self.state.household_objects[vacuum_id]

        if garbage_can["type"] != "garbage_can":
            raise ValueError(f"Object {garbage_can_id} is not a garbage can")
        if vacuum["type"] != "handheld_vacuum":
            raise ValueError(f"Object {vacuum_id} is not a vacuum")

        if not garbage_can["state"]["door_open"]:
            raise ValueError(f"Garbage can {garbage_can_id} is not open")

        # state change
        vacuum["state"]["empty_dust_bin"] = True

    def stack_objects(self, object_id: str, base_object_id: str):
        if object_id not in self.state.household_objects:
            raise ValueError(f"Object {object_id} does not exist")
        if base_object_id not in self.state.household_objects:
            raise ValueError(f"Base object {base_object_id} does not exist")
        if object_id == base_object_id:
            raise ValueError(f"Object {object_id} cannot be stacked on itself")

        if self.state.held_object != object_id:
            raise ValueError(f"Robot is not holding {object_id}")

        self._object_reachable(base_object_id)
        self._robot_at(base_object_id)

        base_obj = self.state.household_objects[base_object_id]
        base_obj_location = base_obj["location"]
        obj = self.state.household_objects[object_id]

        if not base_obj["properties"]["stackable"]:
            raise ValueError(f"Base object {base_object_id} is not stackable")
        if not obj["properties"]["stackable"]:
            raise ValueError(f"Object {object_id} is not stackable")

        object_on_top = self._object_stacked_on(base_object_id)
        if object_on_top is not None:
            raise ValueError(f"Object {base_object_id} is already stacked on top of {object_on_top}")

        # state change
        self.state.held_object = None
        obj["location"] = base_obj_location
        obj["state"]["stacked_on"] = base_object_id

    def unstack_objects(self, object_id: str, base_object_id: str):
        if object_id not in self.state.household_objects:
            raise ValueError(f"Object {object_id} does not exist")
        if base_object_id not in self.state.household_objects:
            raise ValueError(f"Base object {base_object_id} does not exist")
        if object_id == base_object_id:
            raise ValueError(f"Object {object_id} cannot be unstacked from itself")

        if not self.state.gripper_empty:
            raise ValueError(f"Robot is holding an object")

        self._robot_at(base_object_id)
        self._furniture_reachable(self.get_furniture_applicant_of_object(base_object_id))

        base_obj = self.state.household_objects[base_object_id]

        obj = self.state.household_objects[object_id]

        if not base_obj["properties"]["stackable"]:
            raise ValueError(f"Base object {base_object_id} is not stackable")
        if not obj["properties"]["stackable"]:
            raise ValueError(f"Object {object_id} is not stackable")

        if obj["state"]["stacked_on"] != base_object_id:
            raise ValueError(f"Object {object_id} is not stacked on {base_object_id}")

        object_on_top = self._object_stacked_on(object_id)
        if object_on_top is not None:
            raise ValueError(f"Object {object_on_top} is stacked on top of {object_id}, cannot unstack")

        # state change
        obj["state"]["stacked_on"] = None
        obj["location"] = None
        self.state.held_object = object_id

    def toggle_appliance(self, appliance_id: str, appliance_on: bool):
        if appliance_id not in self.state.household_objects:
            raise ValueError(f"Appliance {appliance_id} does not exist")

        if not self.state.gripper_empty:
            raise ValueError(f"Robot is holding an object")

        appliance = self.state.household_objects[appliance_id]

        if not appliance["properties"]["toggleable"]:
            raise ValueError(f"Appliance {appliance_id} is not toggleable. Some objects must not be toggled on but can directly be used.")

        self._robot_at(appliance_id)

        if appliance["state"]["turned_on"] == appliance_on:
            raise ValueError(f"Appliance {appliance_id} is already {'on' if appliance_on else 'off'}")

        # state change
        appliance["state"]["turned_on"] = appliance_on
