Skip to content

Commit 34812d5

Browse files
committed
chore(internal): fix typing util function (openai#1083)
1 parent 9767136 commit 34812d5

File tree

2 files changed

+107
-2
lines changed

2 files changed

+107
-2
lines changed

src/openai/_utils/_typing.py

+29-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
from __future__ import annotations
22

3-
from typing import Any, cast
3+
from typing import Any, TypeVar, cast
44
from typing_extensions import Required, Annotated, get_args, get_origin
55

66
from .._types import InheritsGeneric
@@ -23,6 +23,12 @@ def is_required_type(typ: type) -> bool:
2323
return get_origin(typ) == Required
2424

2525

26+
def is_typevar(typ: type) -> bool:
27+
# type ignore is required because type checkers
28+
# think this expression will always return False
29+
return type(typ) == TypeVar # type: ignore
30+
31+
2632
# Extracts T from Annotated[T, ...] or from Required[Annotated[T, ...]]
2733
def strip_annotated_type(typ: type) -> type:
2834
if is_required_type(typ) or is_annotated_type(typ):
@@ -49,6 +55,15 @@ class MyResponse(Foo[bytes]):
4955
5056
extract_type_var(MyResponse, bases=(Foo,), index=0) -> bytes
5157
```
58+
59+
And where a generic subclass is given:
60+
```py
61+
_T = TypeVar('_T')
62+
class MyResponse(Foo[_T]):
63+
...
64+
65+
extract_type_var(MyResponse[bytes], bases=(Foo,), index=0) -> bytes
66+
```
5267
"""
5368
cls = cast(object, get_origin(typ) or typ)
5469
if cls in generic_bases:
@@ -75,6 +90,18 @@ class MyResponse(Foo[bytes]):
7590
f"Does {cls} inherit from one of {generic_bases} ?"
7691
)
7792

78-
return extract_type_arg(target_base_class, index)
93+
extracted = extract_type_arg(target_base_class, index)
94+
if is_typevar(extracted):
95+
# If the extracted type argument is itself a type variable
96+
# then that means the subclass itself is generic, so we have
97+
# to resolve the type argument from the class itself, not
98+
# the base class.
99+
#
100+
# Note: if there is more than 1 type argument, the subclass could
101+
# change the ordering of the type arguments, this is not currently
102+
# supported.
103+
return extract_type_arg(typ, index)
104+
105+
return extracted
79106

80107
raise RuntimeError(f"Could not resolve inner type variable at index {index} for {typ}")

tests/test_utils/test_typing.py

+78
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,78 @@
1+
from __future__ import annotations
2+
3+
from typing import Generic, TypeVar, cast
4+
5+
from openai._utils import extract_type_var_from_base
6+
7+
_T = TypeVar("_T")
8+
_T2 = TypeVar("_T2")
9+
_T3 = TypeVar("_T3")
10+
11+
12+
class BaseGeneric(Generic[_T]):
13+
...
14+
15+
16+
class SubclassGeneric(BaseGeneric[_T]):
17+
...
18+
19+
20+
class BaseGenericMultipleTypeArgs(Generic[_T, _T2, _T3]):
21+
...
22+
23+
24+
class SubclassGenericMultipleTypeArgs(BaseGenericMultipleTypeArgs[_T, _T2, _T3]):
25+
...
26+
27+
28+
class SubclassDifferentOrderGenericMultipleTypeArgs(BaseGenericMultipleTypeArgs[_T2, _T, _T3]):
29+
...
30+
31+
32+
def test_extract_type_var() -> None:
33+
assert (
34+
extract_type_var_from_base(
35+
BaseGeneric[int],
36+
index=0,
37+
generic_bases=cast("tuple[type, ...]", (BaseGeneric,)),
38+
)
39+
== int
40+
)
41+
42+
43+
def test_extract_type_var_generic_subclass() -> None:
44+
assert (
45+
extract_type_var_from_base(
46+
SubclassGeneric[int],
47+
index=0,
48+
generic_bases=cast("tuple[type, ...]", (BaseGeneric,)),
49+
)
50+
== int
51+
)
52+
53+
54+
def test_extract_type_var_multiple() -> None:
55+
typ = BaseGenericMultipleTypeArgs[int, str, None]
56+
57+
generic_bases = cast("tuple[type, ...]", (BaseGenericMultipleTypeArgs,))
58+
assert extract_type_var_from_base(typ, index=0, generic_bases=generic_bases) == int
59+
assert extract_type_var_from_base(typ, index=1, generic_bases=generic_bases) == str
60+
assert extract_type_var_from_base(typ, index=2, generic_bases=generic_bases) == type(None)
61+
62+
63+
def test_extract_type_var_generic_subclass_multiple() -> None:
64+
typ = SubclassGenericMultipleTypeArgs[int, str, None]
65+
66+
generic_bases = cast("tuple[type, ...]", (BaseGenericMultipleTypeArgs,))
67+
assert extract_type_var_from_base(typ, index=0, generic_bases=generic_bases) == int
68+
assert extract_type_var_from_base(typ, index=1, generic_bases=generic_bases) == str
69+
assert extract_type_var_from_base(typ, index=2, generic_bases=generic_bases) == type(None)
70+
71+
72+
def test_extract_type_var_generic_subclass_different_ordering_multiple() -> None:
73+
typ = SubclassDifferentOrderGenericMultipleTypeArgs[int, str, None]
74+
75+
generic_bases = cast("tuple[type, ...]", (BaseGenericMultipleTypeArgs,))
76+
assert extract_type_var_from_base(typ, index=0, generic_bases=generic_bases) == int
77+
assert extract_type_var_from_base(typ, index=1, generic_bases=generic_bases) == str
78+
assert extract_type_var_from_base(typ, index=2, generic_bases=generic_bases) == type(None)

0 commit comments

Comments
 (0)