Thanks to visit codestin.com
Credit goes to github.com

Skip to content

Commit 2a378ca

Browse files
authored
[3.12] gh-112618: Make Annotated cache typed (#112619) (#112628)
1 parent e3c7947 commit 2a378ca

File tree

3 files changed

+43
-2
lines changed

3 files changed

+43
-2
lines changed

Lib/test/test_typing.py

Lines changed: 34 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -8403,6 +8403,40 @@ class X(Annotated[int, (1, 10)]): ...
84038403
self.assertEqual(X.__mro__, (X, int, object),
84048404
"Annotated should be transparent.")
84058405

8406+
def test_annotated_cached_with_types(self):
8407+
class A(str): ...
8408+
class B(str): ...
8409+
8410+
field_a1 = Annotated[str, A("X")]
8411+
field_a2 = Annotated[str, B("X")]
8412+
a1_metadata = field_a1.__metadata__[0]
8413+
a2_metadata = field_a2.__metadata__[0]
8414+
8415+
self.assertIs(type(a1_metadata), A)
8416+
self.assertEqual(a1_metadata, A("X"))
8417+
self.assertIs(type(a2_metadata), B)
8418+
self.assertEqual(a2_metadata, B("X"))
8419+
self.assertIsNot(type(a1_metadata), type(a2_metadata))
8420+
8421+
field_b1 = Annotated[str, A("Y")]
8422+
field_b2 = Annotated[str, B("Y")]
8423+
b1_metadata = field_b1.__metadata__[0]
8424+
b2_metadata = field_b2.__metadata__[0]
8425+
8426+
self.assertIs(type(b1_metadata), A)
8427+
self.assertEqual(b1_metadata, A("Y"))
8428+
self.assertIs(type(b2_metadata), B)
8429+
self.assertEqual(b2_metadata, B("Y"))
8430+
self.assertIsNot(type(b1_metadata), type(b2_metadata))
8431+
8432+
field_c1 = Annotated[int, 1]
8433+
field_c2 = Annotated[int, 1.0]
8434+
field_c3 = Annotated[int, True]
8435+
8436+
self.assertIs(type(field_c1.__metadata__[0]), int)
8437+
self.assertIs(type(field_c2.__metadata__[0]), float)
8438+
self.assertIs(type(field_c3.__metadata__[0]), bool)
8439+
84068440

84078441
class TypeAliasTests(BaseTestCase):
84088442
def test_canonical_usage_with_variable_annotation(self):

Lib/typing.py

Lines changed: 7 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -2065,9 +2065,14 @@ class Annotated:
20652065
def __new__(cls, *args, **kwargs):
20662066
raise TypeError("Type Annotated cannot be instantiated.")
20672067

2068-
@_tp_cache
20692068
def __class_getitem__(cls, params):
2070-
if not isinstance(params, tuple) or len(params) < 2:
2069+
if not isinstance(params, tuple):
2070+
params = (params,)
2071+
return cls._class_getitem_inner(cls, *params)
2072+
2073+
@_tp_cache(typed=True)
2074+
def _class_getitem_inner(cls, *params):
2075+
if len(params) < 2:
20712076
raise TypeError("Annotated[...] should be used "
20722077
"with at least two arguments (a type and an "
20732078
"annotation).")
Lines changed: 2 additions & 0 deletions
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)