Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add backport of evaluate_forward_ref #497

Merged
merged 41 commits into from
Jan 17, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
41 commits
Select commit Hold shift + click to select a range
8386b24
first draft
Daraan Oct 22, 2024
21cd9da
1st working draft
Daraan Oct 29, 2024
ade80c9
add tests
Daraan Oct 30, 2024
4521d21
Complete tests and code
Daraan Oct 30, 2024
f65e885
Merge remote-tracking branch 'upstream/main' into 3.14/evaluate_forwa…
Daraan Oct 30, 2024
f4cf6b8
Add missing failure cases
Daraan Oct 30, 2024
aebc55c
more compact code
Daraan Oct 30, 2024
d594cbc
complete Final, Generic cases
Daraan Oct 30, 2024
fb992e8
Solve global variable error
Daraan Oct 30, 2024
a195879
Better test cases for __type_params__ & owner
Daraan Oct 30, 2024
a54d7a9
Added failing test case for 3.10
Daraan Oct 30, 2024
46d3efa
add lint exception
Daraan Oct 30, 2024
2261fd1
Test was backported
Daraan Oct 30, 2024
881f926
Update changelog
Daraan Oct 30, 2024
73fb856
Add doc entry
Daraan Oct 30, 2024
01349f3
removed unnecessary import
Daraan Oct 30, 2024
c6a32ce
Merge branch 'main' into 3.14/evaluate_forward_ref
Daraan Nov 25, 2024
3e68adc
use unittest assert
Daraan Nov 26, 2024
b55d419
Use explicit namespaces
Daraan Nov 26, 2024
7380492
Changed Format.SOURCE to STRING
Daraan Nov 26, 2024
cb45bfd
Update comments
Daraan Nov 26, 2024
2bad4c1
Implement 3.11 like _type_check
Daraan Nov 26, 2024
e7b3014
Merge remote-tracking branch 'refs/remotes/origin/3.14/evaluate_forwa…
Daraan Nov 26, 2024
2615d75
Merge remote-tracking branch 'upstream/main' into 3.14/evaluate_forwa…
Daraan Nov 26, 2024
b054c81
Add lint exception
Daraan Nov 26, 2024
0f400b8
support module keyword for some python versions
Daraan Nov 26, 2024
8b44550
Reorder test
Daraan Nov 27, 2024
0f0f20d
comment update
Daraan Nov 27, 2024
8fdba74
Merge branch 'main' into 3.14/evaluate_forward_ref
Daraan Dec 13, 2024
b093a80
formating
Daraan Dec 14, 2024
c982e82
change variable usage
Daraan Dec 14, 2024
8cae202
Restructure test coverage
Daraan Dec 14, 2024
67059fd
Rework tests of nested strings
Daraan Dec 14, 2024
145e833
Merge remote-tracking branch 'upstream/main' into 3.14/evaluate_forwa…
Daraan Dec 14, 2024
a770495
Fix not using type_params in < 3.12.5
Daraan Dec 14, 2024
520b5cd
rework special cases and forward ref
Daraan Dec 14, 2024
ed4ec2d
Add ClassVar and constant check
Daraan Dec 14, 2024
85f1d3f
Explicit regex capture message
Daraan Dec 17, 2024
f995420
Add comment and and corrected regex
Daraan Dec 17, 2024
21e97c3
Merge branch 'main' into 3.14/evaluate_forward_ref
Daraan Jan 6, 2025
fb130c2
Merge branch 'main' into 3.14/evaluate_forward_ref
Daraan Jan 8, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 3 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,9 @@ aliases that have a `Concatenate` special form as their argument.
- Backport CPython PR [#124795](https://github.com/python/cpython/pull/124795):
fix `TypeAliasType` not raising an error on non-tuple inputs for `type_params`.
Patch by [Daraan](https://github.com/Daraan).
- Backport `evaluate_forward_ref` from CPython PR
[#119891](https://github.com/python/cpython/pull/119891) to evaluate `ForwardRef`s.
Patch by [Daraan](https://github.com/Daraan), backporting a CPython PR by Jelle Zijlstra.
- Fix that lists and ... could not be used for parameter expressions for `TypeAliasType`
instances before Python 3.11.
Patch by [Daraan](https://github.com/Daraan).
Expand Down
35 changes: 33 additions & 2 deletions doc/index.rst
Original file line number Diff line number Diff line change
Expand Up @@ -753,6 +753,37 @@ Functions

.. versionadded:: 4.2.0

.. function:: evaluate_forward_ref(forward_ref, *, owner=None, globals=None, locals=None, type_params=None, format=Format.VALUE)

Evaluate an :py:class:`typing.ForwardRef` as a :py:term:`type hint`.

This is similar to calling :py:meth:`annotationlib.ForwardRef.evaluate`,
but unlike that method, :func:`!evaluate_forward_ref` also:

* Recursively evaluates forward references nested within the type hint.
However, the amount of recursion is limited in Python 3.8 and 3.10.
* Raises :exc:`TypeError` when it encounters certain objects that are
not valid type hints.
* Replaces type hints that evaluate to :const:`!None` with
:class:`types.NoneType`.
* Supports the :attr:`Format.FORWARDREF` and
:attr:`Format.STRING` formats.

*forward_ref* must be an instance of :py:class:`typing.ForwardRef`.
*owner*, if given, should be the object that holds the annotations that
the forward reference derived from, such as a module, class object, or function.
It is used to infer the namespaces to use for looking up names.
*globals* and *locals* can also be explicitly given to provide
the global and local namespaces.
*type_params* is a tuple of :py:ref:`type parameters <type-params>` that
are in scope when evaluating the forward reference.
This parameter must be provided (though it may be an empty tuple) if *owner*
is not given and the forward reference does not already have an owner set.
*format* specifies the format of the annotation and is a member of
the :class:`Format` enum.

.. versionadded:: 4.13.0

.. function:: get_annotations(obj, *, globals=None, locals=None, eval_str=False, format=Format.VALUE)

See :py:func:`inspect.get_annotations`. In the standard library since Python 3.10.
Expand All @@ -764,7 +795,7 @@ Functions
of the :pep:`649` behavior on versions of Python that do not support it.

The purpose of this backport is to allow users who would like to use
:attr:`Format.FORWARDREF` or :attr:`Format.SOURCE` semantics once
:attr:`Format.FORWARDREF` or :attr:`Format.STRING` semantics once
:pep:`649` is implemented, but who also
want to support earlier Python versions, to simply write::

Expand Down Expand Up @@ -911,7 +942,7 @@ Enums
``typing_extensions`` emulates this value on versions of Python which do
not support :pep:`649` by returning the same value as for ``VALUE`` semantics.

.. attribute:: SOURCE
.. attribute:: STRING

Equal to 3. When :pep:`649` is implemented, this format will produce an annotation
dictionary where the values have been replaced by strings containing
Expand Down
228 changes: 214 additions & 14 deletions src/test_typing_extensions.py
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,7 @@
import typing_extensions
from _typed_dict_test_helper import Foo, FooGeneric, VeryAnnotated
from typing_extensions import (
_FORWARD_REF_HAS_CLASS,
_PEP_649_OR_749_IMPLEMENTED,
Annotated,
Any,
Expand Down Expand Up @@ -82,6 +83,7 @@
clear_overloads,
dataclass_transform,
deprecated,
evaluate_forward_ref,
final,
get_annotations,
get_args,
Expand Down Expand Up @@ -7948,7 +7950,7 @@ def f2(a: "undefined"): # noqa: F821
self.assertEqual(get_annotations(f2, format=2), {"a": "undefined"})

self.assertEqual(
get_annotations(f1, format=Format.SOURCE),
get_annotations(f1, format=Format.STRING),
{"a": "int"},
)
self.assertEqual(get_annotations(f1, format=3), {"a": "int"})
Expand All @@ -7975,7 +7977,7 @@ def foo():
foo, format=Format.FORWARDREF, eval_str=True
)
get_annotations(
foo, format=Format.SOURCE, eval_str=True
foo, format=Format.STRING, eval_str=True
)

def test_stock_annotations(self):
Expand All @@ -7989,7 +7991,7 @@ def foo(a: int, b: str):
{"a": int, "b": str},
)
self.assertEqual(
get_annotations(foo, format=Format.SOURCE),
get_annotations(foo, format=Format.STRING),
{"a": "int", "b": "str"},
)

Expand Down Expand Up @@ -8084,43 +8086,43 @@ def test_stock_annotations_in_module(self):
)

self.assertEqual(
get_annotations(isa, format=Format.SOURCE),
get_annotations(isa, format=Format.STRING),
{"a": "int", "b": "str"},
)
self.assertEqual(
get_annotations(isa.MyClass, format=Format.SOURCE),
get_annotations(isa.MyClass, format=Format.STRING),
{"a": "int", "b": "str"},
)
mycls = "MyClass" if _PEP_649_OR_749_IMPLEMENTED else "inspect_stock_annotations.MyClass"
self.assertEqual(
get_annotations(isa.function, format=Format.SOURCE),
get_annotations(isa.function, format=Format.STRING),
{"a": "int", "b": "str", "return": mycls},
)
self.assertEqual(
get_annotations(
isa.function2, format=Format.SOURCE
isa.function2, format=Format.STRING
),
{"a": "int", "b": "str", "c": mycls, "return": mycls},
)
self.assertEqual(
get_annotations(
isa.function3, format=Format.SOURCE
isa.function3, format=Format.STRING
),
{"a": "int", "b": "str", "c": "MyClass"},
)
self.assertEqual(
get_annotations(inspect, format=Format.SOURCE),
get_annotations(inspect, format=Format.STRING),
{},
)
self.assertEqual(
get_annotations(
isa.UnannotatedClass, format=Format.SOURCE
isa.UnannotatedClass, format=Format.STRING
),
{},
)
self.assertEqual(
get_annotations(
isa.unannotated_function, format=Format.SOURCE
isa.unannotated_function, format=Format.STRING
),
{},
)
Expand All @@ -8141,7 +8143,7 @@ def test_stock_annotations_on_wrapper(self):
)
mycls = "MyClass" if _PEP_649_OR_749_IMPLEMENTED else "inspect_stock_annotations.MyClass"
self.assertEqual(
get_annotations(wrapped, format=Format.SOURCE),
get_annotations(wrapped, format=Format.STRING),
{"a": "int", "b": "str", "return": mycls},
)
self.assertEqual(
Expand All @@ -8160,10 +8162,10 @@ def test_stringized_annotations_in_module(self):
{"eval_str": False},
{"format": Format.VALUE},
{"format": Format.FORWARDREF},
{"format": Format.SOURCE},
{"format": Format.STRING},
{"format": Format.VALUE, "eval_str": False},
{"format": Format.FORWARDREF, "eval_str": False},
{"format": Format.SOURCE, "eval_str": False},
{"format": Format.STRING, "eval_str": False},
]:
with self.subTest(**kwargs):
self.assertEqual(
Expand Down Expand Up @@ -8466,6 +8468,204 @@ def test_pep_695_generics_with_future_annotations_nested_in_function(self):
set(results.generic_func.__type_params__)
)

class TestEvaluateForwardRefs(BaseTestCase):
def test_global_constant(self):
if sys.version_info[:3] > (3, 10, 0):
self.assertTrue(_FORWARD_REF_HAS_CLASS)

def test_forward_ref_fallback(self):
with self.assertRaises(NameError):
evaluate_forward_ref(typing.ForwardRef("doesntexist"))
ref = typing.ForwardRef("doesntexist")
self.assertIs(evaluate_forward_ref(ref, format=Format.FORWARDREF), ref)

class X:
unresolvable = "doesnotexist2"

evaluated_ref = evaluate_forward_ref(
typing.ForwardRef("X.unresolvable"),
locals={"X": X},
type_params=None,
format=Format.FORWARDREF,
)
self.assertEqual(evaluated_ref, typing.ForwardRef("doesnotexist2"))

def test_evaluate_with_type_params(self):
# Use a T name that is not in globals
self.assertNotIn("Tx", globals())
if not TYPING_3_12_0:
Tx = TypeVar("Tx")
class Gen(Generic[Tx]):
alias = int
if not hasattr(Gen, "__type_params__"):
Gen.__type_params__ = (Tx,)
self.assertEqual(Gen.__type_params__, (Tx,))
del Tx
else:
ns = {}
exec(textwrap.dedent("""
class Gen[Tx]:
alias = int
"""), None, ns)
Gen = ns["Gen"]

# owner=None, type_params=None
# NOTE: The behavior of owner=None might change in the future when ForwardRef.__owner__ is available
with self.assertRaises(NameError):
evaluate_forward_ref(typing.ForwardRef("Tx"))
with self.assertRaises(NameError):
evaluate_forward_ref(typing.ForwardRef("Tx"), type_params=())
with self.assertRaises(NameError):
evaluate_forward_ref(typing.ForwardRef("Tx"), owner=int)

(Tx,) = Gen.__type_params__
self.assertIs(evaluate_forward_ref(typing.ForwardRef("Tx"), type_params=Gen.__type_params__), Tx)

# For this test its important that Tx is not a global variable, i.e. do not use "T" here
self.assertNotIn("Tx", globals())
self.assertIs(evaluate_forward_ref(typing.ForwardRef("Tx"), owner=Gen), Tx)

# Different type_params take precedence
not_Tx = TypeVar("Tx") # different TypeVar with same name
self.assertIs(evaluate_forward_ref(typing.ForwardRef("Tx"), type_params=(not_Tx,), owner=Gen), not_Tx)

# globals can take higher precedence
if _FORWARD_REF_HAS_CLASS:
self.assertIs(evaluate_forward_ref(typing.ForwardRef("Tx", is_class=True), owner=Gen, globals={"Tx": str}), str)
self.assertIs(evaluate_forward_ref(typing.ForwardRef("Tx", is_class=True), owner=Gen, type_params=(not_Tx,), globals={"Tx": str}), str)

with self.assertRaises(NameError):
evaluate_forward_ref(typing.ForwardRef("alias"), type_params=Gen.__type_params__)
self.assertIs(evaluate_forward_ref(typing.ForwardRef("alias"), owner=Gen), int)
# If you pass custom locals, we don't look at the owner's locals
with self.assertRaises(NameError):
evaluate_forward_ref(typing.ForwardRef("alias"), owner=Gen, locals={})
# But if the name exists in the locals, it works
self.assertIs(
evaluate_forward_ref(typing.ForwardRef("alias"), owner=Gen, locals={"alias": str}), str
)

@skipUnless(
HAS_FORWARD_MODULE, "Needs module 'forward' to test forward references"
)
def test_fwdref_with_module(self):
self.assertIs(
evaluate_forward_ref(typing.ForwardRef("Counter", module="collections")), collections.Counter
)
self.assertEqual(
evaluate_forward_ref(typing.ForwardRef("Counter[int]", module="collections")),
collections.Counter[int],
)

with self.assertRaises(NameError):
# If globals are passed explicitly, we don't look at the module dict
evaluate_forward_ref(typing.ForwardRef("Format", module="annotationlib"), globals={})

def test_fwdref_to_builtin(self):
self.assertIs(evaluate_forward_ref(typing.ForwardRef("int")), int)
if HAS_FORWARD_MODULE:
self.assertIs(evaluate_forward_ref(typing.ForwardRef("int", module="collections")), int)
self.assertIs(evaluate_forward_ref(typing.ForwardRef("int"), owner=str), int)

# builtins are still searched with explicit globals
self.assertIs(evaluate_forward_ref(typing.ForwardRef("int"), globals={}), int)

def test_fwdref_with_globals(self):
# explicit values in globals have precedence
obj = object()
self.assertIs(evaluate_forward_ref(typing.ForwardRef("int"), globals={"int": obj}), obj)

def test_fwdref_value_is_cached(self):
fr = typing.ForwardRef("hello")
with self.assertRaises(NameError):
evaluate_forward_ref(fr)
self.assertIs(evaluate_forward_ref(fr, globals={"hello": str}), str)
self.assertIs(evaluate_forward_ref(fr), str)

@skipUnless(TYPING_3_9_0, "Needs PEP 585 support")
def test_fwdref_with_owner(self):
self.assertEqual(
evaluate_forward_ref(typing.ForwardRef("Counter[int]"), owner=collections),
collections.Counter[int],
)

def test_name_lookup_without_eval(self):
# test the codepath where we look up simple names directly in the
# namespaces without going through eval()
self.assertIs(evaluate_forward_ref(typing.ForwardRef("int")), int)
self.assertIs(evaluate_forward_ref(typing.ForwardRef("int"), locals={"int": str}), str)
self.assertIs(
evaluate_forward_ref(typing.ForwardRef("int"), locals={"int": float}, globals={"int": str}),
float,
)
self.assertIs(evaluate_forward_ref(typing.ForwardRef("int"), globals={"int": str}), str)
import builtins

from test import support
with support.swap_attr(builtins, "int", dict):
self.assertIs(evaluate_forward_ref(typing.ForwardRef("int")), dict)

def test_nested_strings(self):
# This variable must have a different name TypeVar
Tx = TypeVar("Tx")

class Y(Generic[Tx]):
a = "X"
bT = "Y[T_nonlocal]"

Z = TypeAliasType("Z", Y[Tx], type_params=(Tx,))

evaluated_ref1a = evaluate_forward_ref(typing.ForwardRef("Y[Y['Tx']]"), locals={"Y": Y, "Tx": Tx})
self.assertEqual(get_origin(evaluated_ref1a), Y)
self.assertEqual(get_args(evaluated_ref1a), (Y[Tx],))

evaluated_ref1b = evaluate_forward_ref(
typing.ForwardRef("Y[Y['Tx']]"), locals={"Y": Y}, type_params=(Tx,)
)
self.assertEqual(get_origin(evaluated_ref1b), Y)
self.assertEqual(get_args(evaluated_ref1b), (Y[Tx],))

with self.subTest("nested string of TypeVar"):
evaluated_ref2 = evaluate_forward_ref(typing.ForwardRef("""Y["Y['Tx']"]"""), locals={"Y": Y})
self.assertEqual(get_origin(evaluated_ref2), Y)
if not TYPING_3_9_0:
self.skipTest("Nested string 'Tx' stays ForwardRef in 3.8")
self.assertEqual(get_args(evaluated_ref2), (Y[Tx],))

with self.subTest("nested string of TypeAliasType and alias"):
# NOTE: Using Y here works for 3.10
evaluated_ref3 = evaluate_forward_ref(typing.ForwardRef("""Y['Z["StrAlias"]']"""), locals={"Y": Y, "Z": Z, "StrAlias": str})
self.assertEqual(get_origin(evaluated_ref3), Y)
if sys.version_info[:2] in ((3,8), (3, 10)):
self.skipTest("Nested string 'StrAlias' is not resolved in 3.8 and 3.10")
self.assertEqual(get_args(evaluated_ref3), (Z[str],))

def test_invalid_special_forms(self):
# tests _lax_type_check to raise errors the same way as the typing module.
# Regex capture "< class 'module.name'> and "module.name"
with self.assertRaisesRegex(
TypeError, r"Plain .*Protocol('>)? is not valid as type argument"
):
evaluate_forward_ref(typing.ForwardRef("Protocol"), globals=vars(typing))
with self.assertRaisesRegex(
TypeError, r"Plain .*Generic('>)? is not valid as type argument"
):
evaluate_forward_ref(typing.ForwardRef("Generic"), globals=vars(typing))
with self.assertRaisesRegex(TypeError, r"Plain typing(_extensions)?\.Final is not valid as type argument"):
evaluate_forward_ref(typing.ForwardRef("Final"), globals=vars(typing))
with self.assertRaisesRegex(TypeError, r"Plain typing(_extensions)?\.ClassVar is not valid as type argument"):
evaluate_forward_ref(typing.ForwardRef("ClassVar"), globals=vars(typing))
if _FORWARD_REF_HAS_CLASS:
self.assertIs(evaluate_forward_ref(typing.ForwardRef("Final", is_class=True), globals=vars(typing)), Final)
self.assertIs(evaluate_forward_ref(typing.ForwardRef("ClassVar", is_class=True), globals=vars(typing)), ClassVar)
with self.assertRaisesRegex(TypeError, r"Plain typing(_extensions)?\.Final is not valid as type argument"):
evaluate_forward_ref(typing.ForwardRef("Final", is_argument=False), globals=vars(typing))
with self.assertRaisesRegex(TypeError, r"Plain typing(_extensions)?\.ClassVar is not valid as type argument"):
evaluate_forward_ref(typing.ForwardRef("ClassVar", is_argument=False), globals=vars(typing))
else:
self.assertIs(evaluate_forward_ref(typing.ForwardRef("Final", is_argument=False), globals=vars(typing)), Final)
self.assertIs(evaluate_forward_ref(typing.ForwardRef("ClassVar", is_argument=False), globals=vars(typing)), ClassVar)


if __name__ == '__main__':
main()
Loading
Loading