import copy
import logging
import uuid
from dataclasses import dataclass
from typing import Dict, Generic, Optional, TypeVar

T = TypeVar("T")


@dataclass
class EnvStates(Generic[T]):
    curr_hash: str
    init_hash: str
    seed: int
    states: Dict[str, T]

    def set_state(self, state: T, hash: Optional[str] = None) -> str:
        while True:
            n_hash = str(uuid.uuid4()) if hash is None else hash
            if n_hash not in self.states:
                break
            logging.warning("hash collision -> regenerate")

        self.states[n_hash] = copy.deepcopy(state)
        self.curr_hash = n_hash
        return n_hash

    def get_state(self, hash: str) -> Optional[T]:
        return self.states.get(hash, None)

    def to_dict(self) -> Dict:
        return {"states": {k: v.to_dict() for k, v in self.states.items()}}

    def from_dict(self, data: Dict):
        self.states = {k: T.from_dict(v) for k, v in data["states"].items()}


class MotionSimulator(Generic[T]):

    def init_state(self, seed: int) -> str:
        """set simulator to seed `seed`

        can be subclassed by actual simulator
        """
        init_hash = str(uuid.uuid4())
        self.env_states = EnvStates(seed=seed, init_hash=init_hash, curr_hash=init_hash, states={})
        self.add_state()

    def get_hash(self) -> str:
        return self.env_states.curr_hash

    def get_seed(self) -> int:
        return self.env_states.seed

    def run_motion(self, motion: str):
        self._run_motion(motion)
        self.add_state()

    def set_hash(self, hash: str):
        if hash == self.env_states.curr_hash:
            return  # nothing to do

        if hash not in self.env_states.states:
            raise ValueError(f"hash {hash} not found in env states")

        state = self.env_states.get_state(hash=hash)
        assert state is not None, "state not present"

        self._set_state(new_state=copy.deepcopy(state))
        self.env_states.curr_hash = hash

    def add_state(self, *, hash: Optional[str] = None) -> str:
        """add a new state transition to the env states."""
        new_hash = self.env_states.set_state(state=self._get_state(), hash=hash)
        return new_hash

    def _run_motion(self, motion: str) -> bool:
        """run motion `motion` in simulator"""
        raise NotImplementedError("run_motion() must be implemented in subclasses")

    def _get_state(self) -> T:
        """get the current state of the simulator

        must be subclassed by the actual simulator
        """
        raise NotImplementedError("get_state() must be implemented in subclasses")

    def _set_state(self, new_state: T):
        """set the simulator to state `new_state`

        must be subclassed by the actual simulator
        """
        raise NotImplementedError("set_state() must be implemented in subclasses")
