Module pearl.utils.instantiations.spaces.box
Expand source code
from __future__ import annotations
import logging
from typing import Optional, Union
import numpy as np
import torch
from pearl.api.space import Space
from pearl.utils.instantiations.spaces.utils import reshape_to_1d_tensor
from torch import Tensor
try:
    import gymnasium as gym
    from gymnasium.spaces import Box
    logging.info("Using 'gymnasium' package.")
except ModuleNotFoundError:
    import gym
    from gym.spaces import Box
    logging.warning("Using deprecated 'gym' package.")
class BoxSpace(Space):
    """A continuous, box space. This class is a wrapper around Gymnasium's
    `Box` space, but uses PyTorch tensors instead of NumPy arrays."""
    def __init__(
        self,
        low: Union[float, Tensor],
        high: Union[float, Tensor],
        seed: Optional[Union[int, np.random.Generator]] = None,
    ) -> None:
        """Constructs a `BoxSpace`.
        Args:
            low: The lower bound on each dimension of the space.
            high: The upper bound on each dimension of the space.
            seed: Random seed used to initialize the random number generator of the
                underlying Gymnasium `Box` space.
        """
        low = low.numpy(force=True) if isinstance(low, Tensor) else np.array([low])
        high = high.numpy(force=True) if isinstance(high, Tensor) else np.array([high])
        self._gym_space = Box(low=low, high=high, seed=seed)
    @property
    def is_continuous(self) -> bool:
        """Checks whether this is a continuous space."""
        return True
    def sample(self, mask: Optional[Tensor] = None) -> Tensor:
        """Sample an element uniformly at random from the space.
        Args:
            mask: An unused argument for the case of a `BoxSpace`, which
                does not support masking.
        Returns:
            A randomly sampled element.
        """
        if mask is not None:
            logging.warning("Masked sampling is not supported in `BoxSpace`. Ignoring.")
        return torch.from_numpy(self._gym_space.sample())
    @property
    def low(self) -> Tensor:
        """Returns the lower bound of the space."""
        return reshape_to_1d_tensor(torch.from_numpy(self._gym_space.low))
    @property
    def high(self) -> Tensor:
        """Returns the upper bound of the space."""
        return reshape_to_1d_tensor(torch.from_numpy(self._gym_space.high))
    @property
    def shape(self) -> torch.Size:
        """Returns the shape of an element of the space."""
        return self.low.shape
    @staticmethod
    def from_gym(gym_space: gym.Space) -> BoxSpace:
        """Constructs a `BoxSpace` given a Gymnasium `Box` space.
        Args:
            gym_space: A Gymnasium `Box` space.
        Returns:
            A `BoxSpace` with the same bounds and seed as `gym_space`.
        """
        assert isinstance(gym_space, Box)
        return BoxSpace(
            low=torch.from_numpy(gym_space.low),
            high=torch.from_numpy(gym_space.high),
            seed=gym_space._np_random,
        )Classes
- class BoxSpace (low: Union[float, Tensor], high: Union[float, Tensor], seed: Optional[Union[int, np.random.Generator]] = None)
- 
A continuous, box space. This class is a wrapper around Gymnasium's Boxspace, but uses PyTorch tensors instead of NumPy arrays.Constructs a BoxSpace.Args- low
- The lower bound on each dimension of the space.
- high
- The upper bound on each dimension of the space.
- seed
- Random seed used to initialize the random number generator of the
underlying Gymnasium Boxspace.
 Expand source codeclass BoxSpace(Space): """A continuous, box space. This class is a wrapper around Gymnasium's `Box` space, but uses PyTorch tensors instead of NumPy arrays.""" def __init__( self, low: Union[float, Tensor], high: Union[float, Tensor], seed: Optional[Union[int, np.random.Generator]] = None, ) -> None: """Constructs a `BoxSpace`. Args: low: The lower bound on each dimension of the space. high: The upper bound on each dimension of the space. seed: Random seed used to initialize the random number generator of the underlying Gymnasium `Box` space. """ low = low.numpy(force=True) if isinstance(low, Tensor) else np.array([low]) high = high.numpy(force=True) if isinstance(high, Tensor) else np.array([high]) self._gym_space = Box(low=low, high=high, seed=seed) @property def is_continuous(self) -> bool: """Checks whether this is a continuous space.""" return True def sample(self, mask: Optional[Tensor] = None) -> Tensor: """Sample an element uniformly at random from the space. Args: mask: An unused argument for the case of a `BoxSpace`, which does not support masking. Returns: A randomly sampled element. """ if mask is not None: logging.warning("Masked sampling is not supported in `BoxSpace`. Ignoring.") return torch.from_numpy(self._gym_space.sample()) @property def low(self) -> Tensor: """Returns the lower bound of the space.""" return reshape_to_1d_tensor(torch.from_numpy(self._gym_space.low)) @property def high(self) -> Tensor: """Returns the upper bound of the space.""" return reshape_to_1d_tensor(torch.from_numpy(self._gym_space.high)) @property def shape(self) -> torch.Size: """Returns the shape of an element of the space.""" return self.low.shape @staticmethod def from_gym(gym_space: gym.Space) -> BoxSpace: """Constructs a `BoxSpace` given a Gymnasium `Box` space. Args: gym_space: A Gymnasium `Box` space. Returns: A `BoxSpace` with the same bounds and seed as `gym_space`. """ assert isinstance(gym_space, Box) return BoxSpace( low=torch.from_numpy(gym_space.low), high=torch.from_numpy(gym_space.high), seed=gym_space._np_random, )Ancestors- Space
- abc.ABC
 SubclassesStatic methods- def from_gym(gym_space: gym.Space) ‑> BoxSpace
- 
Constructs a BoxSpacegiven a GymnasiumBoxspace.Args- gym_space
- A Gymnasium Boxspace.
 ReturnsA BoxSpacewith the same bounds and seed asgym_space.Expand source code@staticmethod def from_gym(gym_space: gym.Space) -> BoxSpace: """Constructs a `BoxSpace` given a Gymnasium `Box` space. Args: gym_space: A Gymnasium `Box` space. Returns: A `BoxSpace` with the same bounds and seed as `gym_space`. """ assert isinstance(gym_space, Box) return BoxSpace( low=torch.from_numpy(gym_space.low), high=torch.from_numpy(gym_space.high), seed=gym_space._np_random, )
 Instance variables- var high : torch.Tensor
- 
Returns the upper bound of the space. Expand source code@property def high(self) -> Tensor: """Returns the upper bound of the space.""" return reshape_to_1d_tensor(torch.from_numpy(self._gym_space.high))
- var low : torch.Tensor
- 
Returns the lower bound of the space. Expand source code@property def low(self) -> Tensor: """Returns the lower bound of the space.""" return reshape_to_1d_tensor(torch.from_numpy(self._gym_space.low))
 Methods- def sample(self, mask: Optional[Tensor] = None) ‑> torch.Tensor
- 
Sample an element uniformly at random from the space. Args- mask
- An unused argument for the case of a BoxSpace, which does not support masking.
 ReturnsA randomly sampled element. Expand source codedef sample(self, mask: Optional[Tensor] = None) -> Tensor: """Sample an element uniformly at random from the space. Args: mask: An unused argument for the case of a `BoxSpace`, which does not support masking. Returns: A randomly sampled element. """ if mask is not None: logging.warning("Masked sampling is not supported in `BoxSpace`. Ignoring.") return torch.from_numpy(self._gym_space.sample())
 Inherited members