import logging
from pathlib import Path
from typing import Dict, List
from pddl.core import Predicate
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,
    MotionSimulationException,
    MotionSimulationResponseCode,
)
from state_estimation import PredicateGrounder, SEMotionValidator, SEVariable

from demos.ipc.src.household.household_predicates_object_api import (
    Agent,
    FurnitureAppliance,
    FurnitureApplianceState,
    HouseholdObject,
    HouseholdObjectState,
    ObjectProperties,
    SmallItems,
    SmallReceptacle,
)


logger = logging.getLogger(__name__)


class SEHouseholdMotionValidator(SEMotionValidator):

    def __init__(self, grounder: PredicateGrounder):
        super().__init__(grounder=grounder)
        self.env = HouseholdMotionSimulator(env=HouseholdEnvironment())

    def _run_motion(self, motion: str):
        try:
            self.env.run_motion(motion)
        except TypeError as e:
            raise MotionSimulationException(
                message=str(e),
                code=MotionSimulationResponseCode.PDDL_PY_TRANSLATION,
                expected="",
                ground_truth="",
            )
        except ValueError as e:
            raise MotionSimulationException(
                message=str(e),
                code=MotionSimulationResponseCode.EFFECT_FAILED,
                expected="",
                ground_truth="",
            )

    # def reset(self, seed: Optional[int], instance_config: Optional[Dict] = None):
    #     self._env.reset(seed=seed, instance_config=instance_config)

    #     # Save initial state after reset
    #     initial_state_uuid = str(uuid.uuid4())
    #     self._save_env_state(initial_state_uuid)
    #     self._current_state_uuid = initial_state_uuid

    def get_variables(self) -> List[SEVariable]:
        objs = []
        state = self.env.env.state
        assert state is not None
        for name, object in state.household_objects.items():
            object["state"] = {k: v for k, v in object["state"].items() if k not in ["turned_on"]}
            cls = SmallReceptacle if object["family"] == "small_receptacle" else SmallItems
            obj = SEVariable(
                name=name,
                # type="object",
                value=cls(
                    properties=ObjectProperties(**object["properties"]),
                    id=name,
                    location=object["location"],
                    state=HouseholdObjectState(**object["state"]),
                ),
            )
            objs.append(obj)
        for name, appliance in state.furniture_appliances.items():
            obj = SEVariable(
                name=name,
                # type="furniture_appliance",
                value=FurnitureAppliance(
                    properties=ObjectProperties(**appliance["properties"]),
                    id=name,
                    state=FurnitureApplianceState(**appliance["state"]),
                ),
            )
            objs.append(obj)

        objs.append(
            SEVariable(
                name="robot",
                value=Agent(location=state.robot_location, grasps=state.held_object),
            )
        )
        return objs

    def get_env_hash(self):
        return self.env.get_hash()

    def set_env_hash(self, hash):
        return self.env.set_hash(hash=hash)

    def init_state(self, seed: int):
        return self.env.init_state(seed=seed)
