|
35 | 35 | typing.Tuple[Literal["mapping"], None],
|
36 | 36 | typing.Tuple[Literal["typed-dict"], type],
|
37 | 37 | typing.Tuple[Literal["typevar"], TypeVar],
|
38 |
| - typing.Tuple[Literal["union"], int], |
| 38 | + typing.Tuple[Literal["union"], tuple[int, bool]], |
39 | 39 | typing.Tuple[Literal["tuple"], Optional[int]],
|
40 | 40 | typing.Tuple[Literal["user-class"], Optional[int]],
|
41 | 41 | typing.Tuple[Literal["alias"], str],
|
|
44 | 44 | else:
|
45 | 45 | TypeConstructorArgs = typing.Tuple[str, Any]
|
46 | 46 |
|
| 47 | +if sys.version_info[1] >= 10: |
| 48 | + from types import UnionType |
| 49 | +else: |
| 50 | + UnionType = None |
| 51 | + |
47 | 52 | if sys.version_info[1] >= 11:
|
48 | 53 | from typing import Self
|
49 | 54 | else:
|
@@ -186,12 +191,19 @@ def _recorded_type(self, idx: int) -> typing.Tuple[Any, int]:
|
186 | 191 | idx,
|
187 | 192 | ) # pylint: disable = unnecessary-dunder-call
|
188 | 193 | if tag == "union":
|
189 |
| - assert isinstance(param, int) |
| 194 | + assert isinstance(param, tuple) |
| 195 | + num_members, use_UnionType = param |
| 196 | + assert isinstance(num_members, int) |
190 | 197 | member_ts: typing.List[Any] = []
|
191 |
| - for _ in range(param): |
| 198 | + for _ in range(num_members): |
192 | 199 | member_t, idx = self._recorded_type(idx + 1)
|
193 | 200 | member_ts.append(member_t)
|
194 |
| - return typing.Union.__getitem__(tuple(member_ts)), idx |
| 201 | + if not use_UnionType: |
| 202 | + return typing.Union.__getitem__(tuple(member_ts)), idx |
| 203 | + union_type = member_ts[0] |
| 204 | + for t in member_ts[1:]: |
| 205 | + union_type |= t |
| 206 | + return union_type, idx |
195 | 207 | if tag == "typed-dict":
|
196 | 208 | for _ in get_type_hints(param):
|
197 | 209 | _, idx = self._recorded_type(idx + 1)
|
@@ -302,8 +314,11 @@ def _record_collection(self, item_t: Any) -> None:
|
302 | 314 | def _record_mapping(self, key_t: Any, value_t: Any) -> None:
|
303 | 315 | self._append_constructor_args(("mapping", None))
|
304 | 316 |
|
305 |
| - def _record_union(self, *member_ts: Any) -> None: |
306 |
| - self._append_constructor_args(("union", len(member_ts))) |
| 317 | + def _record_union(self, *member_ts: Any, use_UnionType: bool = False) -> None: |
| 318 | + if use_UnionType: |
| 319 | + assert member_ts, "Cannot use UnionType with empty members." |
| 320 | + assert UnionType is not None, "Cannot use UnionType, version <= 3.9" |
| 321 | + self._append_constructor_args(("union", (len(member_ts), use_UnionType))) |
307 | 322 |
|
308 | 323 | def _record_variadic_tuple(self, item_t: Any) -> None:
|
309 | 324 | self._append_constructor_args(("tuple", None))
|
@@ -385,14 +400,22 @@ def _repr(
|
385 | 400 | ]
|
386 | 401 | return lines, idx
|
387 | 402 | if tag == "union":
|
388 |
| - assert isinstance(param, int) |
389 |
| - lines = [indent + "Union["] |
390 |
| - for _ in range(param): |
| 403 | + assert isinstance(param, tuple) |
| 404 | + num_members, use_UnionType = param |
| 405 | + assert isinstance(num_members, int) |
| 406 | + lines = [] |
| 407 | + if not use_UnionType: |
| 408 | + lines.append(indent + "Union[") |
| 409 | + for _ in range(num_members): |
391 | 410 | member_lines, idx = self._repr(idx + 1, level + 1)
|
392 |
| - member_lines[-1] += "," |
| 411 | + if use_UnionType: |
| 412 | + member_lines[-1] += "|" |
| 413 | + else: |
| 414 | + member_lines[-1] += "," |
393 | 415 | lines.extend(member_lines)
|
394 | 416 | assert len(lines) > 1, "Cannot take a union of no types."
|
395 |
| - lines.append(indent + "]") |
| 417 | + if not use_UnionType: |
| 418 | + lines.append(indent + "]") |
396 | 419 | return lines, idx
|
397 | 420 | if tag == "typed-dict":
|
398 | 421 | t = param
|
|
0 commit comments