mirror of
				https://github.com/python/cpython.git
				synced 2025-10-25 02:43:41 +00:00 
			
		
		
		
	gh-114053: Fix bad interaction of PEP-695, PEP-563 and `get_type_hints` (#118009)
				
					
				
			Co-authored-by: Jelle Zijlstra <jelle.zijlstra@gmail.com>
This commit is contained in:
		
							parent
							
								
									15b3555e4a
								
							
						
					
					
						commit
						1e3e7ce11e
					
				
					 5 changed files with 81 additions and 10 deletions
				
			
		|  | @ -3024,7 +3024,9 @@ Introspection helpers | ||||||
| 
 | 
 | ||||||
|    This is often the same as ``obj.__annotations__``. In addition, |    This is often the same as ``obj.__annotations__``. In addition, | ||||||
|    forward references encoded as string literals are handled by evaluating |    forward references encoded as string literals are handled by evaluating | ||||||
|    them in ``globals`` and ``locals`` namespaces. For a class ``C``, return |    them in ``globals``, ``locals`` and (where applicable) | ||||||
|  |    :ref:`type parameter <type-params>` namespaces. | ||||||
|  |    For a class ``C``, return | ||||||
|    a dictionary constructed by merging all the ``__annotations__`` along |    a dictionary constructed by merging all the ``__annotations__`` along | ||||||
|    ``C.__mro__`` in reverse order. |    ``C.__mro__`` in reverse order. | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -46,7 +46,7 @@ | ||||||
| import types | import types | ||||||
| 
 | 
 | ||||||
| from test.support import captured_stderr, cpython_only, infinite_recursion | from test.support import captured_stderr, cpython_only, infinite_recursion | ||||||
| from test.typinganndata import mod_generics_cache, _typed_dict_helper | from test.typinganndata import ann_module695, mod_generics_cache, _typed_dict_helper | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| CANNOT_SUBCLASS_TYPE = 'Cannot subclass special typing classes' | CANNOT_SUBCLASS_TYPE = 'Cannot subclass special typing classes' | ||||||
|  | @ -4641,6 +4641,30 @@ def f(x: X): ... | ||||||
|             {'x': list[list[ForwardRef('X')]]} |             {'x': list[list[ForwardRef('X')]]} | ||||||
|         ) |         ) | ||||||
| 
 | 
 | ||||||
|  |     def test_pep695_generic_with_future_annotations(self): | ||||||
|  |         hints_for_A = get_type_hints(ann_module695.A) | ||||||
|  |         A_type_params = ann_module695.A.__type_params__ | ||||||
|  |         self.assertIs(hints_for_A["x"], A_type_params[0]) | ||||||
|  |         self.assertEqual(hints_for_A["y"].__args__[0], Unpack[A_type_params[1]]) | ||||||
|  |         self.assertIs(hints_for_A["z"].__args__[0], A_type_params[2]) | ||||||
|  | 
 | ||||||
|  |         hints_for_B = get_type_hints(ann_module695.B) | ||||||
|  |         self.assertEqual(hints_for_B.keys(), {"x", "y", "z"}) | ||||||
|  |         self.assertEqual( | ||||||
|  |             set(hints_for_B.values()) ^ set(ann_module695.B.__type_params__), | ||||||
|  |             set() | ||||||
|  |         ) | ||||||
|  | 
 | ||||||
|  |         hints_for_generic_function = get_type_hints(ann_module695.generic_function) | ||||||
|  |         func_t_params = ann_module695.generic_function.__type_params__ | ||||||
|  |         self.assertEqual( | ||||||
|  |             hints_for_generic_function.keys(), {"x", "y", "z", "zz", "return"} | ||||||
|  |         ) | ||||||
|  |         self.assertIs(hints_for_generic_function["x"], func_t_params[0]) | ||||||
|  |         self.assertEqual(hints_for_generic_function["y"], Unpack[func_t_params[1]]) | ||||||
|  |         self.assertIs(hints_for_generic_function["z"].__origin__, func_t_params[2]) | ||||||
|  |         self.assertIs(hints_for_generic_function["zz"].__origin__, func_t_params[2]) | ||||||
|  | 
 | ||||||
|     def test_extended_generic_rules_subclassing(self): |     def test_extended_generic_rules_subclassing(self): | ||||||
|         class T1(Tuple[T, KT]): ... |         class T1(Tuple[T, KT]): ... | ||||||
|         class T2(Tuple[T, ...]): ... |         class T2(Tuple[T, ...]): ... | ||||||
|  |  | ||||||
							
								
								
									
										22
									
								
								Lib/test/typinganndata/ann_module695.py
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								Lib/test/typinganndata/ann_module695.py
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,22 @@ | ||||||
|  | from __future__ import annotations | ||||||
|  | from typing import Callable | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | class A[T, *Ts, **P]: | ||||||
|  |     x: T | ||||||
|  |     y: tuple[*Ts] | ||||||
|  |     z: Callable[P, str] | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | class B[T, *Ts, **P]: | ||||||
|  |     T = int | ||||||
|  |     Ts = str | ||||||
|  |     P = bytes | ||||||
|  |     x: T | ||||||
|  |     y: Ts | ||||||
|  |     z: P | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def generic_function[T, *Ts, **P]( | ||||||
|  |     x: T, *y: *Ts, z: P.args, zz: P.kwargs | ||||||
|  | ) -> None: ... | ||||||
|  | @ -399,7 +399,8 @@ def inner(*args, **kwds): | ||||||
| 
 | 
 | ||||||
|     return decorator |     return decorator | ||||||
| 
 | 
 | ||||||
| def _eval_type(t, globalns, localns, recursive_guard=frozenset()): | 
 | ||||||
|  | def _eval_type(t, globalns, localns, type_params, *, recursive_guard=frozenset()): | ||||||
|     """Evaluate all forward references in the given type t. |     """Evaluate all forward references in the given type t. | ||||||
| 
 | 
 | ||||||
|     For use of globalns and localns see the docstring for get_type_hints(). |     For use of globalns and localns see the docstring for get_type_hints(). | ||||||
|  | @ -407,7 +408,7 @@ def _eval_type(t, globalns, localns, recursive_guard=frozenset()): | ||||||
|     ForwardRef. |     ForwardRef. | ||||||
|     """ |     """ | ||||||
|     if isinstance(t, ForwardRef): |     if isinstance(t, ForwardRef): | ||||||
|         return t._evaluate(globalns, localns, recursive_guard) |         return t._evaluate(globalns, localns, type_params, recursive_guard=recursive_guard) | ||||||
|     if isinstance(t, (_GenericAlias, GenericAlias, types.UnionType)): |     if isinstance(t, (_GenericAlias, GenericAlias, types.UnionType)): | ||||||
|         if isinstance(t, GenericAlias): |         if isinstance(t, GenericAlias): | ||||||
|             args = tuple( |             args = tuple( | ||||||
|  | @ -421,7 +422,13 @@ def _eval_type(t, globalns, localns, recursive_guard=frozenset()): | ||||||
|                 t = t.__origin__[args] |                 t = t.__origin__[args] | ||||||
|             if is_unpacked: |             if is_unpacked: | ||||||
|                 t = Unpack[t] |                 t = Unpack[t] | ||||||
|         ev_args = tuple(_eval_type(a, globalns, localns, recursive_guard) for a in t.__args__) | 
 | ||||||
|  |         ev_args = tuple( | ||||||
|  |             _eval_type( | ||||||
|  |                 a, globalns, localns, type_params, recursive_guard=recursive_guard | ||||||
|  |             ) | ||||||
|  |             for a in t.__args__ | ||||||
|  |         ) | ||||||
|         if ev_args == t.__args__: |         if ev_args == t.__args__: | ||||||
|             return t |             return t | ||||||
|         if isinstance(t, GenericAlias): |         if isinstance(t, GenericAlias): | ||||||
|  | @ -974,7 +981,7 @@ def __init__(self, arg, is_argument=True, module=None, *, is_class=False): | ||||||
|         self.__forward_is_class__ = is_class |         self.__forward_is_class__ = is_class | ||||||
|         self.__forward_module__ = module |         self.__forward_module__ = module | ||||||
| 
 | 
 | ||||||
|     def _evaluate(self, globalns, localns, recursive_guard): |     def _evaluate(self, globalns, localns, type_params, *, recursive_guard): | ||||||
|         if self.__forward_arg__ in recursive_guard: |         if self.__forward_arg__ in recursive_guard: | ||||||
|             return self |             return self | ||||||
|         if not self.__forward_evaluated__ or localns is not globalns: |         if not self.__forward_evaluated__ or localns is not globalns: | ||||||
|  | @ -988,14 +995,25 @@ def _evaluate(self, globalns, localns, recursive_guard): | ||||||
|                 globalns = getattr( |                 globalns = getattr( | ||||||
|                     sys.modules.get(self.__forward_module__, None), '__dict__', globalns |                     sys.modules.get(self.__forward_module__, None), '__dict__', globalns | ||||||
|                 ) |                 ) | ||||||
|  |             if type_params: | ||||||
|  |                 # "Inject" type parameters into the local namespace | ||||||
|  |                 # (unless they are shadowed by assignments *in* the local namespace), | ||||||
|  |                 # as a way of emulating annotation scopes when calling `eval()` | ||||||
|  |                 locals_to_pass = {param.__name__: param for param in type_params} | localns | ||||||
|  |             else: | ||||||
|  |                 locals_to_pass = localns | ||||||
|             type_ = _type_check( |             type_ = _type_check( | ||||||
|                 eval(self.__forward_code__, globalns, localns), |                 eval(self.__forward_code__, globalns, locals_to_pass), | ||||||
|                 "Forward references must evaluate to types.", |                 "Forward references must evaluate to types.", | ||||||
|                 is_argument=self.__forward_is_argument__, |                 is_argument=self.__forward_is_argument__, | ||||||
|                 allow_special_forms=self.__forward_is_class__, |                 allow_special_forms=self.__forward_is_class__, | ||||||
|             ) |             ) | ||||||
|             self.__forward_value__ = _eval_type( |             self.__forward_value__ = _eval_type( | ||||||
|                 type_, globalns, localns, recursive_guard | {self.__forward_arg__} |                 type_, | ||||||
|  |                 globalns, | ||||||
|  |                 localns, | ||||||
|  |                 type_params, | ||||||
|  |                 recursive_guard=(recursive_guard | {self.__forward_arg__}), | ||||||
|             ) |             ) | ||||||
|             self.__forward_evaluated__ = True |             self.__forward_evaluated__ = True | ||||||
|         return self.__forward_value__ |         return self.__forward_value__ | ||||||
|  | @ -2334,7 +2352,7 @@ def get_type_hints(obj, globalns=None, localns=None, include_extras=False): | ||||||
|                     value = type(None) |                     value = type(None) | ||||||
|                 if isinstance(value, str): |                 if isinstance(value, str): | ||||||
|                     value = ForwardRef(value, is_argument=False, is_class=True) |                     value = ForwardRef(value, is_argument=False, is_class=True) | ||||||
|                 value = _eval_type(value, base_globals, base_locals) |                 value = _eval_type(value, base_globals, base_locals, base.__type_params__) | ||||||
|                 hints[name] = value |                 hints[name] = value | ||||||
|         return hints if include_extras else {k: _strip_annotations(t) for k, t in hints.items()} |         return hints if include_extras else {k: _strip_annotations(t) for k, t in hints.items()} | ||||||
| 
 | 
 | ||||||
|  | @ -2360,6 +2378,7 @@ def get_type_hints(obj, globalns=None, localns=None, include_extras=False): | ||||||
|             raise TypeError('{!r} is not a module, class, method, ' |             raise TypeError('{!r} is not a module, class, method, ' | ||||||
|                             'or function.'.format(obj)) |                             'or function.'.format(obj)) | ||||||
|     hints = dict(hints) |     hints = dict(hints) | ||||||
|  |     type_params = getattr(obj, "__type_params__", ()) | ||||||
|     for name, value in hints.items(): |     for name, value in hints.items(): | ||||||
|         if value is None: |         if value is None: | ||||||
|             value = type(None) |             value = type(None) | ||||||
|  | @ -2371,7 +2390,7 @@ def get_type_hints(obj, globalns=None, localns=None, include_extras=False): | ||||||
|                 is_argument=not isinstance(obj, types.ModuleType), |                 is_argument=not isinstance(obj, types.ModuleType), | ||||||
|                 is_class=False, |                 is_class=False, | ||||||
|             ) |             ) | ||||||
|         hints[name] = _eval_type(value, globalns, localns) |         hints[name] = _eval_type(value, globalns, localns, type_params) | ||||||
|     return hints if include_extras else {k: _strip_annotations(t) for k, t in hints.items()} |     return hints if include_extras else {k: _strip_annotations(t) for k, t in hints.items()} | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -0,0 +1,4 @@ | ||||||
|  | Fix erroneous :exc:`NameError` when calling :func:`typing.get_type_hints` on | ||||||
|  | a class that made use of :pep:`695` type parameters in a module that had | ||||||
|  | ``from __future__ import annotations`` at the top of the file. Patch by Alex | ||||||
|  | Waygood. | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Alex Waygood
						Alex Waygood