import numpy as np


class Space:

    def __init__(self, dtype, shape=(), low=None, high=None):
        # For integer types, high is the excluside upper bound.
        shape = (shape,) if isinstance(shape, int) else shape
        self._dtype = np.dtype(dtype)
        assert self._dtype is not object, self._dtype
        assert isinstance(shape, tuple), shape
        self._low = self._infer_low(dtype, shape, low, high)
        self._high = self._infer_high(dtype, shape, low, high)
        self._shape = self._infer_shape(dtype, shape, low, high)
        self._discrete = np.issubdtype(self.dtype, np.integer) or self.dtype == bool
        self._random = np.random.RandomState()

    @property
    def dtype(self):
        return self._dtype

    @property
    def shape(self):
        return self._shape

    @property
    def low(self):
        return self._low

    @property
    def high(self):
        return self._high

    @property
    def discrete(self):
        return self._discrete

    def __repr__(self):
        return (
            f"Space(dtype={self.dtype.name}, "
            f"shape={self.shape}, "
            f"low={self.low.min()}, "
            f"high={self.high.max()})"
        )

    def __contains__(self, value):
        value = np.asarray(value)
        if value.shape != self.shape:
            return False
        if (value > self.high).any():
            return False
        if (value < self.low).any():
            return False
        if (value.astype(self.dtype).astype(value.dtype) != value).any():
            return False
        return True

    def sample(self):
        low, high = self.low, self.high
        if np.issubdtype(self.dtype, np.floating):
            low = np.maximum(np.ones(self.shape) * np.finfo(self.dtype).min, low)
            high = np.minimum(np.ones(self.shape) * np.finfo(self.dtype).max, high)
        return self._random.uniform(low, high, self.shape).astype(self.dtype)

    def _infer_low(self, dtype, shape, low, high):
        if low is not None:
            try:
                return np.broadcast_to(low, shape)
            except ValueError:
                raise ValueError(f"Cannot broadcast {low} to shape {shape}")
        elif np.issubdtype(dtype, np.floating):
            return -np.inf * np.ones(shape)
        elif np.issubdtype(dtype, np.integer):
            return np.iinfo(dtype).min * np.ones(shape, dtype)
        elif np.issubdtype(dtype, bool):
            return np.zeros(shape, bool)
        else:
            raise ValueError("Cannot infer low bound from shape and dtype.")

    def _infer_high(self, dtype, shape, low, high):
        if high is not None:
            try:
                return np.broadcast_to(high, shape)
            except ValueError:
                raise ValueError(f"Cannot broadcast {high} to shape {shape}")
        elif np.issubdtype(dtype, np.floating):
            return np.inf * np.ones(shape)
        elif np.issubdtype(dtype, np.integer):
            return np.iinfo(dtype).max * np.ones(shape, dtype)
        elif np.issubdtype(dtype, bool):
            return np.ones(shape, bool)
        else:
            raise ValueError("Cannot infer high bound from shape and dtype.")

    def _infer_shape(self, dtype, shape, low, high):
        if shape is None and low is not None:
            shape = low.shape
        if shape is None and high is not None:
            shape = high.shape
        if not hasattr(shape, "__len__"):
            shape = (shape,)
        assert all(dim and dim > 0 for dim in shape), shape
        return tuple(shape)
