-
-
Notifications
You must be signed in to change notification settings - Fork 32.8k
gh-119180: Documentation for PEP 649 and 749 #122235
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
Changes from 6 commits
20dcb62
206560c
edd12a2
79ce456
3f70c21
e297349
d52fb08
4e7a2a5
35312e7
b30abbd
bcb6a9f
c82fb66
b4f6385
6ebdce9
52dc99e
61eedbd
27668ce
6254e18
c42cb60
1873912
86001c1
c1943fa
889f5aa
6dd39fb
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -64,8 +64,10 @@ language using this mechanism: | |
| generator_stop | 3.5.0b1 | 3.7 | :pep:`479`: | | ||
| | | | *StopIteration handling inside generators* | | ||
+------------------+-------------+--------------+---------------------------------------------+ | ||
| annotations | 3.7.0b1 | TBD [1]_ | :pep:`563`: | | ||
| | | | *Postponed evaluation of annotations* | | ||
| annotations | 3.7.0b1 | Never [1]_ | :pep:`563`: | | ||
| | | | *Postponed evaluation of annotations*, | | ||
| | | | :pep:`649`: *Deferred evalutation of | | ||
| | | | annotations using descriptors* | | ||
+------------------+-------------+--------------+---------------------------------------------+ | ||
|
||
.. XXX Adding a new entry? Remember to update simple_stmts.rst, too. | ||
|
@@ -115,11 +117,9 @@ language using this mechanism: | |
|
||
.. [1] | ||
``from __future__ import annotations`` was previously scheduled to | ||
become mandatory in Python 3.10, but the Python Steering Council | ||
twice decided to delay the change | ||
(`announcement for Python 3.10 <https://mail.python.org/archives/list/[email protected]/message/CLVXXPQ2T2LQ5MP2Y53VVQFCXYWQJHKZ/>`__; | ||
`announcement for Python 3.11 <https://mail.python.org/archives/list/[email protected]/message/VIZEBX5EYMSYIJNDBF6DMUMZOCWHARSO/>`__). | ||
No final decision has been made yet. See also :pep:`563` and :pep:`649`. | ||
become mandatory in Python 3.10, but the change was delayed and ultimately | ||
canceled. This feature will eventually be deprecated and removed. See | ||
:pep:`649` and :pep:`749`. | ||
Comment on lines
119
to
+122
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Should we provide or link to migration guidance here, especially given that this is the first time a |
||
|
||
|
||
.. seealso:: | ||
|
JelleZijlstra marked this conversation as resolved.
Show resolved
Hide resolved
|
Original file line number | Diff line number | Diff line change | ||||||||||||||
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
@@ -0,0 +1,178 @@ | ||||||||||||||||
:mod:`!annotationlib` --- Functionality for introspecting annotations | ||||||||||||||||
===================================================================== | ||||||||||||||||
|
||||||||||||||||
.. module:: annotationlib | ||||||||||||||||
:synopsis: Functionality for introspecting annotations | ||||||||||||||||
|
||||||||||||||||
|
||||||||||||||||
**Source code:** :source:`Lib/annotationlib.py` | ||||||||||||||||
|
||||||||||||||||
.. testsetup:: default | ||||||||||||||||
|
||||||||||||||||
import annotationlib | ||||||||||||||||
from annotationlib import * | ||||||||||||||||
|
||||||||||||||||
-------------- | ||||||||||||||||
|
||||||||||||||||
The :mod:`!annotationlib` module provides tools for introspecting :term:`annotations <annotation>` | ||||||||||||||||
on modules, classes, and functions. | ||||||||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Potentially this could be added later, but I think a longer narrative summary of I wonder if there are words from 749 that we could incorporate? Effectively, something that briefly talks to where this module fits into the landscape, perhaps aimed at the fairly experienced user of There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Yes, I was thinking of adding that too, but ran out of energy while working on the initial version. :) I'll probably work on this later. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I wrote such a summary now, along with a historical account of annotation semantics, since that seems like something the docs should cover. |
||||||||||||||||
|
||||||||||||||||
.. function:: call_annotate_function(annotate, format, *, owner=None) | ||||||||||||||||
|
||||||||||||||||
Call the :term:`annotate function` *annotate* with the given *format*, a member of the :class:`Format` | ||||||||||||||||
enum, and return the annotations dictionary produced by the function. | ||||||||||||||||
|
||||||||||||||||
Annotate functions generated by the compiler for functions, classes, and modules support only | ||||||||||||||||
the :attr:`~Format.VALUE` format when called directly. This function calls the annotate function | ||||||||||||||||
in a special environment that allows it to produce annotations in the other formats. | ||||||||||||||||
AA-Turner marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||||||||
|
||||||||||||||||
*owner* is the object that owns the annotation function, usually a function, class, or module. | ||||||||||||||||
If provided, it is used in the :attr:`~Format.FORWARDREF` format to produce :class:`ForwardRef` | ||||||||||||||||
JelleZijlstra marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||||||||
object that carry more information. | ||||||||||||||||
|
||||||||||||||||
.. versionadded:: 3.14 | ||||||||||||||||
|
||||||||||||||||
.. class:: Format | ||||||||||||||||
|
||||||||||||||||
A :class:`enum.IntEnum` describing the formats in which annotations can be returned. | ||||||||||||||||
JelleZijlstra marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||||||||
Members of the enum, or their equivalent integer values, can be passed to | ||||||||||||||||
:func:`get_annotations` and other functions in this module, as well as to | ||||||||||||||||
:attr:`~object.__annotate__` functions. | ||||||||||||||||
|
||||||||||||||||
.. attribute:: VALUE | ||||||||||||||||
|
||||||||||||||||
JelleZijlstra marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||||||||
Values are the result of evaluating the annotation expressions. | ||||||||||||||||
|
||||||||||||||||
This enum member's value is equal to 1. | ||||||||||||||||
|
||||||||||||||||
.. attribute:: FORWARDREF | ||||||||||||||||
|
||||||||||||||||
Values are real annotation values (as per :attr:`Format.VALUE` format) for defined values, | ||||||||||||||||
and :class:`ForwardRef` proxies for undefined values. Real objects may be exposed to, | ||||||||||||||||
or contain references to, :class:`ForwardRef` proxy objects. | ||||||||||||||||
JelleZijlstra marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||||||||
|
||||||||||||||||
This enum member's value is equal to 2. | ||||||||||||||||
|
||||||||||||||||
.. attribute:: SOURCE | ||||||||||||||||
|
||||||||||||||||
Values are the text string of the annotation as it appears in the source code. | ||||||||||||||||
May only be approximate; whitespace may be normalized, and constant values | ||||||||||||||||
may be optimized. It is possible the exact values of these strings could | ||||||||||||||||
change in future version of Python. | ||||||||||||||||
JelleZijlstra marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||||||||
|
||||||||||||||||
.. versionadded:: 3.14 | ||||||||||||||||
|
||||||||||||||||
.. class:: ForwardRef | ||||||||||||||||
|
||||||||||||||||
A proxy object for forward references in annotations. | ||||||||||||||||
|
||||||||||||||||
Instances of this class are returned when the :attr:`~Format.FORWARDREF` format is | ||||||||||||||||
used and annotations contain a name that cannot be resolved. This can happen | ||||||||||||||||
JelleZijlstra marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||||||||
when a forward reference is used in an annotation, such as when a class is | ||||||||||||||||
referenced before it is defined. | ||||||||||||||||
|
||||||||||||||||
.. attribute:: __forward_arg__ | ||||||||||||||||
|
||||||||||||||||
A string containing the code that was evaluated to produce the :class:`~ForwardRef`. | ||||||||||||||||
The string may not be exactly equivalent to the original source. | ||||||||||||||||
|
||||||||||||||||
.. method:: evaluate(*, globals=None, locals=None, type_params=None, owner=None) | ||||||||||||||||
|
||||||||||||||||
Evaluate the forward reference, returning its value. | ||||||||||||||||
|
||||||||||||||||
This may throw an exception such as :exc:`NameError` if the forward reference | ||||||||||||||||
JelleZijlstra marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||||||||
refers to names that do not exist. The parameters to the function can be used to | ||||||||||||||||
provide bindings for names that would otherwise be undefined. | ||||||||||||||||
|
||||||||||||||||
*globals* and *locals* are passed to :func:`eval()`, representing the global and | ||||||||||||||||
local namespaces in which the name is evaluated. *type_params*, if given, must be | ||||||||||||||||
a tuple of :ref:`type parameters <type-params>` that are in scope while the forward | ||||||||||||||||
reference is being evaluated. *owner* is the object that owns the annotation from | ||||||||||||||||
which the forward reference derives, usually a function, class, or module. | ||||||||||||||||
:class:`~ForwardRef` instances returned by :func:`get_annotations` retain | ||||||||||||||||
JelleZijlstra marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||||||||
a reference to their owner, so it is not necessary to pass it in explicitly. | ||||||||||||||||
|
||||||||||||||||
Once a :class:`~ForwardRef` instance has been evaluated, it caches the evaluated | ||||||||||||||||
value, and future calls to :meth:`evaluate` will return the cached value, regardless | ||||||||||||||||
of the parameters passed in. | ||||||||||||||||
JelleZijlstra marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||||||||
|
||||||||||||||||
.. versionadded:: 3.14 | ||||||||||||||||
|
||||||||||||||||
.. function:: get_annotate_function(obj) | ||||||||||||||||
|
||||||||||||||||
Retrieve the :term:`annotate function` for *obj*. Return ``None`` if *obj* does not have an | ||||||||||||||||
annotate function. | ||||||||||||||||
|
||||||||||||||||
This is usually equivalent to accessing the :attr:`~object.__annotate__` attribute of *obj*, | ||||||||||||||||
but direct access to the attribute may return the wrong object in certain situations involving | ||||||||||||||||
metaclasses. It is recommended to use this function instead of accessing the attribute directly. | ||||||||||||||||
JelleZijlstra marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||||||||
|
||||||||||||||||
.. versionadded:: 3.14 | ||||||||||||||||
|
||||||||||||||||
.. function:: get_annotations(obj, *, globals=None, locals=None, eval_str=False, format=Format.VALUE) | ||||||||||||||||
|
||||||||||||||||
Compute the annotations dict for an object. | ||||||||||||||||
JelleZijlstra marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||||||||
|
||||||||||||||||
``obj`` may be a callable, class, module, or other object with | ||||||||||||||||
JelleZijlstra marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||||||||
:attr:`~object.__annotate__` and :attr:`~object.__annotations__` attributes. | ||||||||||||||||
Passing in an object of any other type raises :exc:`TypeError`. | ||||||||||||||||
|
||||||||||||||||
The *format* parameter controls the format in which annotations are returned: | ||||||||||||||||
JelleZijlstra marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||||||||
|
||||||||||||||||
Returns a dict. ``get_annotations()`` returns a new dict every time | ||||||||||||||||
it's called; calling it twice on the same object will return two | ||||||||||||||||
different but equivalent dicts. | ||||||||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
Suggested change
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Why should this be indented into a note? I feel that makes the text noisier. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. You'll have a nice box around this (important) behaviour. Since your bullet list contained "Always, always, always creates a new dict", I feel this behaviour should be highlighted. I used "note" but you can also use "important" or "tip". |
||||||||||||||||
|
||||||||||||||||
This function handles several details for you: | ||||||||||||||||
|
||||||||||||||||
* If ``eval_str`` is true, values of type ``str`` will | ||||||||||||||||
JelleZijlstra marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||||||||
be un-stringized using :func:`eval()`. This is intended | ||||||||||||||||
JelleZijlstra marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||||||||
for use with stringized annotations | ||||||||||||||||
(``from __future__ import annotations``). It is an error | ||||||||||||||||
to set ``eval_str`` to true with formats other than :attr:`Format.VALUE`. | ||||||||||||||||
* If ``obj`` doesn't have an annotations dict, returns an | ||||||||||||||||
empty dict. (Functions and methods always have an | ||||||||||||||||
JelleZijlstra marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||||||||
annotations dict; classes, modules, and other types of | ||||||||||||||||
callables may not.) | ||||||||||||||||
* Ignores inherited annotations on classes, as well as annotations | ||||||||||||||||
on metaclasses. If a class | ||||||||||||||||
JelleZijlstra marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||||||||
doesn't have its own annotations dict, returns an empty dict. | ||||||||||||||||
* All accesses to object members and dict values are done | ||||||||||||||||
using ``getattr()`` and ``dict.get()`` for safety. | ||||||||||||||||
* Always, always, always returns a freshly created dict. | ||||||||||||||||
JelleZijlstra marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||||||||
|
||||||||||||||||
``eval_str`` controls whether or not values of type ``str`` are replaced | ||||||||||||||||
with the result of calling :func:`eval()` on those values: | ||||||||||||||||
|
||||||||||||||||
* If eval_str is true, :func:`eval()` is called on values of type ``str``. | ||||||||||||||||
(Note that ``get_annotations`` doesn't catch exceptions; if :func:`eval()` | ||||||||||||||||
JelleZijlstra marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||||||||
raises an exception, it will unwind the stack past the ``get_annotations`` | ||||||||||||||||
call.) | ||||||||||||||||
* If eval_str is false (the default), values of type ``str`` are unchanged. | ||||||||||||||||
JelleZijlstra marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||||||||
|
||||||||||||||||
``globals`` and ``locals`` are passed in to :func:`eval()`; see the documentation | ||||||||||||||||
for :func:`eval()` for more information. If ``globals`` or ``locals`` | ||||||||||||||||
is ``None``, this function may replace that value with a context-specific | ||||||||||||||||
default, contingent on ``type(obj)``: | ||||||||||||||||
|
||||||||||||||||
* If ``obj`` is a module, ``globals`` defaults to ``obj.__dict__``. | ||||||||||||||||
* If ``obj`` is a class, ``globals`` defaults to | ||||||||||||||||
``sys.modules[obj.__module__].__dict__`` and ``locals`` defaults | ||||||||||||||||
to the ``obj`` class namespace. | ||||||||||||||||
* If ``obj`` is a callable, ``globals`` defaults to | ||||||||||||||||
:attr:`obj.__globals__ <function.__globals__>`, | ||||||||||||||||
although if ``obj`` is a wrapped function (using | ||||||||||||||||
:func:`functools.update_wrapper`) it is first unwrapped. | ||||||||||||||||
picnixz marked this conversation as resolved.
Show resolved
Hide resolved
JelleZijlstra marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||||||||
|
||||||||||||||||
Calling :func:`!get_annotations` is best practice for accessing the | ||||||||||||||||
annotations dict of any object. See :ref:`annotations-howto` for | ||||||||||||||||
more information on annotations best practices. | ||||||||||||||||
|
||||||||||||||||
.. doctest:: | ||||||||||||||||
|
||||||||||||||||
>>> def f(a: int, b: str) -> float: | ||||||||||||||||
... pass | ||||||||||||||||
>>> get_annotations(f) | ||||||||||||||||
{'a': <class 'int'>, 'b': <class 'str'>, 'return': <class 'float'>} | ||||||||||||||||
|
||||||||||||||||
.. versionadded:: 3.14 |
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -25,4 +25,5 @@ overview: | |
__future__.rst | ||
gc.rst | ||
inspect.rst | ||
annotationlib.rst | ||
site.rst |
Uh oh!
There was an error while loading. Please reload this page.