import gymnasium as gym
import numpy as np

class FixedDimActionWrapper(gym.ActionWrapper):
    """
    A Gym ActionWrapper that removes fixed (non-varying) action dimensions
    from the environment's action space.

    This wrapper identifies dimensions of the action space where the upper
    and lower bounds are equal (i.e., fixed actions) and removes them from
    the agent's controllable actions. The agent interacts only with the
    remaining, variable dimensions, while the fixed dimensions are
    automatically filled with their constant values during each step.

    Parameters
    ----------
    env : gym.Env
        The Gym environment to be wrapped.

    Attributes
    ----------
    keep : np.ndarray
        Indices of the variable action dimensions to be retained.
    fixed : np.ndarray
        Indices of the fixed action dimensions to be removed.
    fixed_values : np.ndarray
        Fixed values corresponding to the removed action dimensions.
    action_space : gym.spaces.Box
        The modified action space containing only variable dimensions.

    Methods
    -------
    action(action)
        Expands the reduced action vector by inserting the fixed values
        at their respective indices to form a full action compatible with
        the original environment.
    """
    def __init__(self, env):
        super().__init__(env)
        low, high = env.action_space.low, env.action_space.high
        self.keep = np.where((high - low) > 0)[0]
        self.fixed = np.where((high - low) == 0)[0]
        self.fixed_values = low[self.fixed]

        self.action_space = gym.spaces.Box(
            low=low[self.keep],
            high=high[self.keep],
            dtype=np.float32
        )

    def action(self, action):
        full = np.empty_like(self.env.action_space.low, dtype=np.float32)
        full[self.keep] = action
        full[self.fixed] = self.fixed_values
        return full
