import logging
from typing import List
from pddl.core import Predicate
from tp_lodge.task_planning.models.pddl.pddl_domain import PDDLDomain
from demos.ipc.src.household.household_environment import HouseholdEnvironment
from demos.ipc.src.household.household_environment_state import HouseholdEnvironmentState
from demos.ipc.src.household.household_motion_simulator import HouseholdMotionSimulator
from tp_lodge.motion_planning.local_motion_validator import LocalMotionValidator


logger = logging.getLogger(__name__)


class HouseholdMotionValidator(LocalMotionValidator[HouseholdEnvironmentState]):

    def __init__(self, hide_env_feedback: bool):
        super().__init__(env=HouseholdMotionSimulator(env=HouseholdEnvironment()), hide_env_feedback=hide_env_feedback)
        # self.init_state(seed=0)

    def get_supported_predicates(self, domain: PDDLDomain) -> List[str]:
        return [
            "at-agent",
            "holding",
            "agent-hand-empty",
            "at-object",
            "at-receptacle",
            "is-open",
            "openable",
            "is-receptacle-open",
            "receptacle-openable",
            "dirty-surface",
            "sliced",
            "heated",
            "is-empty-dust-bin",
            "stacked-on",
            "dirty",
            "is-switched-on",
            # new
            "pickupable",
            # "stackable",
            "object-clear",
            "flat-surface",
            "toggleable",
            "sliceable",
            "washable",
        ]

    def validate_predicate(self, pred: Predicate) -> bool:
        args = [arg.name for arg in pred.terms]
        state = self.env.env.state  # type: HouseholdEnvironmentState

        if pred.name == "at-agent":
            assert len(args) == 2
            location = args[1]
            return state.robot_location == location

        elif pred.name == "holding":
            assert len(args) == 2
            obj = args[1]
            return state.held_object == obj

        elif pred.name == "agent-hand-empty":
            assert len(args) == 1
            return state.held_object is None

        elif pred.name == "at-object":
            assert len(args) == 2
            obj, location = args
            return state.household_objects[obj]["location"] == location

        elif pred.name == "at-receptacle":
            assert len(args) == 2
            obj, location = args
            return state.household_objects[obj]["location"] == location

        elif pred.name == "is-open":
            assert len(args) == 1
            obj_id = args[0]
            f_state = state.furniture_appliances[obj_id]["state"]
            if "door_open" not in f_state:
                raise ValueError(f"Predicate 'is-open' is not supported for object {obj_id}.")
            return f_state["door_open"]

        elif pred.name == "openable":
            assert len(args) == 1
            obj_id = args[0]
            return state.furniture_appliances[obj_id]["properties"]["openable"]

        elif pred.name == "is-receptacle-open":
            assert len(args) == 1
            obj_id = args[0]
            obj = state.household_objects[obj_id]
            if not obj["properties"]["openable"]:
                raise ValueError(f"Predicate 'is-open' is not supported for object {obj_id}.")
            return obj["state"]["is_open"]

        elif pred.name == "receptacle-openable":
            assert len(args) == 1
            obj_id = args[0]
            return state.household_objects[obj_id]["properties"]["openable"]

        elif pred.name == "dirty":
            assert len(args) == 1
            obj_id = args[0]
            obj_state = state.household_objects[obj_id]["state"]
            if not state.household_objects[obj_id]["properties"]["washable"]:
                raise ValueError(f"Predicate 'dirty' is not supported for object {obj_id}.")
            return not obj_state["clean"]

        elif pred.name == "dirty-surface":
            assert len(args) == 1
            obj_id = args[0]
            obj_state = state.furniture_appliances[obj_id]["state"]
            if not state.furniture_appliances[obj_id]["properties"]["wipeable"]:
                raise ValueError(f"Predicate 'dirty-surface' is not supported for object {obj_id}.")
            return not obj_state["clean"]

        elif pred.name == "sliced":
            assert len(args) == 1
            obj_id = args[0]
            obj_state = state.household_objects[obj_id]["state"]
            if not state.household_objects[obj_id]["properties"]["sliceable"]:
                raise ValueError(f"Predicate 'sliced' is not supported for object {obj_id}.")
            return obj_state["sliced"]

        elif pred.name == "heated":
            assert len(args) == 1
            obj_id = args[0]
            obj_state = state.household_objects[obj_id]["state"]
            if not state.household_objects[obj_id]["properties"]["heatable"]:
                raise ValueError(f"Predicate 'heated' is not supported for object {obj_id}.")
            return obj_state["heated"]

        elif pred.name == "is-empty-dust-bin":
            assert len(args) == 1
            obj_id = args[0]
            obj_state = state.household_objects[obj_id]["state"]
            if "empty_dust_bin" not in obj_state:
                raise ValueError(f"Predicate 'is-empty-dust-bin' is not supported for object {obj_id}.")
            return obj_state["empty_dust_bin"]

        elif pred.name == "stacked-on":
            assert len(args) == 2
            obj1_id, obj2_id = args
            obj1 = state.household_objects[obj1_id]
            obj2 = state.household_objects[obj2_id]

            if not obj1["properties"]["stackable"]:
                raise ValueError(f"Predicate 'stacked-on' is not supported for object {obj1_id}.")
            if not obj2["properties"]["stackable"]:
                raise ValueError(f"Predicate 'stacked-on' is not supported for object {obj2_id}.")

            return obj1["state"]["stacked_on"] == obj2_id

        elif pred.name == "is-switched-on":
            assert len(args) == 1
            obj_id = args[0]
            obj_state = state.household_objects[obj_id]["state"]
            if "turned_on" not in obj_state:
                raise ValueError(f"Predicate 'is-switched-on' is not supported for object {obj_id}.")
            return obj_state["turned_on"]

        elif pred.name == "pickupable":
            assert len(args) == 1
            obj_id = args[0]
            return state.household_objects[obj_id]["properties"]["pickupable"]

        # elif pred.name == "stackable":
        #     assert len(args) == 1
        #     obj_id = args[0]
        #     return state.household_objects[obj_id]["properties"]["stackable"]

        elif pred.name == "object-clear":
            assert len(args) == 1
            obj_id = args[0]
            for other_obj_id, other_obj in state.household_objects.items():
                if other_obj_id != obj_id and other_obj["state"].get("stacked_on", None) == obj_id:
                    return False
            return True
        elif pred.name == "flat-surface":
            assert len(args) == 1
            obj_id = args[0]
            return state.furniture_appliances[obj_id]["properties"]["has_flat_surface"]
        elif pred.name == "toggleable":
            assert len(args) == 1
            obj_id = args[0]
            return state.household_objects[obj_id]["properties"]["toggleable"]
        elif pred.name == "sliceable":
            assert len(args) == 1
            obj_id = args[0]
            return state.household_objects[obj_id]["properties"]["sliceable"]
        elif pred.name == "washable":
            assert len(args) == 1
            obj_id = args[0]
            return state.household_objects[obj_id]["properties"]["washable"]
        elif pred.name == "is-receptacle":
            assert len(args) == 1
            obj_id = args[0]
            return state.household_objects[obj_id]["properties"]["is_receptacle"]

        else:
            raise ValueError(
                f"Predicate '{pred.name}' with arguments {args} is not supported by the household environment validator."
            )
