[docs]classTuple(gymnasium.spaces.Tuple,Space):"""A tuple (i.e., product) of simpler spaces Example usage: self.observation_space = spaces.Tuple(Discrete(2), Discrete(3)) """_TUPLE_RE=re.compile(r"\A\s*?Tuple\((.+)\)\s*\Z")_INNER_PIECE_RE=re.compile(r"(?P<inner_rest>,\s*"r"(?P<piece>[A-Za-z]+\(.*\)))\s*\Z")def__init__(self,spaces,seed:int|typing.Sequence[int]|np.random.Generator|None=None,):gymnasium.spaces.Tuple.__init__(self,spaces,seed)Space.__init__(self)
[docs]defto_vector(self,data:np.ndarray,**kwargs)->np.ndarray:"""Flatten data using the contained spaces"""returnnp.array([s.to_vector(data[idx])foridx,sinenumerate(self.spaces)]# type: ignore[attr-defined])
[docs]defreshape_to_space(self,value:Iterable,**kwargs)->np.ndarray:"""Reshape value using the contained spaces"""as_list=np.array(value)returnnp.array([s.reshape_to_space(as_list[idx])# type: ignore[attr-defined]foridx,sinenumerate(self.spaces)])
[docs]@classmethod@functools.cachedeffrom_string(cls,s):complete_match=Tuple._TUPLE_RE.match(s)ifnotcomplete_match:raiseRuntimeError("String '%s' does not match '%s'"%(s,Tuple._TUPLE_RE))inner_str=complete_match[1]spaces=[]whilelen(inner_str)>0:match=Tuple._INNER_PIECE_RE.search(inner_str)ifmatchisNone:try:spaces.append(Space.from_string(inner_str))except:pass# We simply ignore garbage.breakelse:head,_,tail=inner_str.rpartition(match["inner_rest"])inner_str=head+tailspaces.append(Space.from_string(match["piece"]))spaces.reverse()returnTuple(*spaces)