Skip to content

Commit db53770

Browse files
[3.11] [3.12] gh-112618: Make Annotated cache typed (GH-112619) (GH-112628) (#112633)
[3.12] gh-112618: Make Annotated cache typed (GH-112619) (GH-112628) (cherry picked from commit 2a378ca) Co-authored-by: Alex Waygood <Alex.Waygood@Gmail.com>
1 parent 8d8efe7 commit db53770

File tree

3 files changed

+43
-2
lines changed

3 files changed

+43
-2
lines changed

Lib/test/test_typing.py

+34
Original file line numberDiff line numberDiff line change
@@ -7627,6 +7627,40 @@ class X(Annotated[int, (1, 10)]): ...
76277627
self.assertEqual(X.__mro__, (X, int, object),
76287628
"Annotated should be transparent.")
76297629

7630+
def test_annotated_cached_with_types(self):
7631+
class A(str): ...
7632+
class B(str): ...
7633+
7634+
field_a1 = Annotated[str, A("X")]
7635+
field_a2 = Annotated[str, B("X")]
7636+
a1_metadata = field_a1.__metadata__[0]
7637+
a2_metadata = field_a2.__metadata__[0]
7638+
7639+
self.assertIs(type(a1_metadata), A)
7640+
self.assertEqual(a1_metadata, A("X"))
7641+
self.assertIs(type(a2_metadata), B)
7642+
self.assertEqual(a2_metadata, B("X"))
7643+
self.assertIsNot(type(a1_metadata), type(a2_metadata))
7644+
7645+
field_b1 = Annotated[str, A("Y")]
7646+
field_b2 = Annotated[str, B("Y")]
7647+
b1_metadata = field_b1.__metadata__[0]
7648+
b2_metadata = field_b2.__metadata__[0]
7649+
7650+
self.assertIs(type(b1_metadata), A)
7651+
self.assertEqual(b1_metadata, A("Y"))
7652+
self.assertIs(type(b2_metadata), B)
7653+
self.assertEqual(b2_metadata, B("Y"))
7654+
self.assertIsNot(type(b1_metadata), type(b2_metadata))
7655+
7656+
field_c1 = Annotated[int, 1]
7657+
field_c2 = Annotated[int, 1.0]
7658+
field_c3 = Annotated[int, True]
7659+
7660+
self.assertIs(type(field_c1.__metadata__[0]), int)
7661+
self.assertIs(type(field_c2.__metadata__[0]), float)
7662+
self.assertIs(type(field_c3.__metadata__[0]), bool)
7663+
76307664

76317665
class TypeAliasTests(BaseTestCase):
76327666
def test_canonical_usage_with_variable_annotation(self):

Lib/typing.py

+7-2
Original file line numberDiff line numberDiff line change
@@ -2214,9 +2214,14 @@ class Annotated:
22142214
def __new__(cls, *args, **kwargs):
22152215
raise TypeError("Type Annotated cannot be instantiated.")
22162216

2217-
@_tp_cache
22182217
def __class_getitem__(cls, params):
2219-
if not isinstance(params, tuple) or len(params) < 2:
2218+
if not isinstance(params, tuple):
2219+
params = (params,)
2220+
return cls._class_getitem_inner(cls, *params)
2221+
2222+
@_tp_cache(typed=True)
2223+
def _class_getitem_inner(cls, *params):
2224+
if len(params) < 2:
22202225
raise TypeError("Annotated[...] should be used "
22212226
"with at least two arguments (a type and an "
22222227
"annotation).")
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
Fix a caching bug relating to :data:`typing.Annotated`.
2+
``Annotated[str, True]`` is no longer identical to ``Annotated[str, 1]``.

0 commit comments

Comments
 (0)