diff options
Diffstat (limited to 'Doc/howto')
| -rw-r--r-- | Doc/howto/annotations.rst | 226 | ||||
| -rw-r--r-- | Doc/howto/index.rst | 1 | 
2 files changed, 227 insertions, 0 deletions
| diff --git a/Doc/howto/annotations.rst b/Doc/howto/annotations.rst new file mode 100644 index 0000000000..3e61103e99 --- /dev/null +++ b/Doc/howto/annotations.rst @@ -0,0 +1,226 @@ +.. _annotations-howto: + +************************** +Annotations Best Practices +************************** + +:author: Larry Hastings + +.. topic:: Abstract + +  This document is designed to encapsulate the best practices +  for working with annotations dicts.  If you write Python code +  that examines ``__annotations__`` on Python objects, we +  encourage you to follow the guidelines described below. + +  The document is organized into four sections: +  best practices for accessing the annotations of an object +  in Python versions 3.10 and newer, +  best practices for accessing the annotations of an object +  in Python versions 3.9 and older, +  other best practices +  for ``__annotations__`` that apply to any Python version, +  and +  quirks of ``__annotations__``. + +  Note that this document is specifically about working with +  ``__annotations__``, not uses *for* annotations. +  If you're looking for information on how to use "type hints" +  in your code, please see the :mod:`typing` module. + + +Accessing The Annotations Dict Of An Object In Python 3.10 And Newer +==================================================================== + +  Python 3.10 adds a new function to the standard library: +  :func:`inspect.get_annotations`.  In Python versions 3.10 +  and newer, calling this function is the best practice for +  accessing the annotations dict of any object that supports +  annotations.  This function can also "un-stringize" +  stringized annotations for you. + +  If for some reason :func:`inspect.get_annotations` isn't +  viable for your use case, you may access the +  ``__annotations__`` data member manually.  Best practice +  for this changed in Python 3.10 as well: as of Python 3.10, +  ``o.__annotations__`` is guaranteed to *always* work +  on Python functions, classes, and modules.  If you're +  certain the object you're examining is one of these three +  *specific* objects, you may simply use ``o.__annotations__`` +  to get at the object's annotations dict. + +  However, other types of callables--for example, +  callables created by :func:`functools.partial`--may +  not have an ``__annotations__`` attribute defined.  When +  accessing the ``__annotations__`` of a possibly unknown +  object,  best practice in Python versions 3.10 and +  newer is to call :func:`getattr` with three arguments, +  for example ``getattr(o, '__annotations__', None)``. + + +Accessing The Annotations Dict Of An Object In Python 3.9 And Older +=================================================================== + +  In Python 3.9 and older, accessing the annotations dict +  of an object is much more complicated than in newer versions. +  The problem is a design flaw in these older versions of Python, +  specifically to do with class annotations. + +  Best practice for accessing the annotations dict of other +  objects--functions, other callables, and modules--is the same +  as best practice for 3.10, assuming you aren't calling +  :func:`inspect.get_annotations`: you should use three-argument +  :func:`getattr` to access the object's ``__annotations__`` +  attribute. + +  Unfortunately, this isn't best practice for classes.  The problem +  is that, since ``__annotations__`` is optional on classes, and +  because classes can inherit attributes from their base classes, +  accessing the ``__annotations__`` attribute of a class may +  inadvertently return the annotations dict of a *base class.* +  As an example:: + +      class Base: +          a: int = 3 +          b: str = 'abc' + +      class Derived(Base): +          pass + +      print(Derived.__annotations__) + +  This will print the annotations dict from ``Base``, not +  ``Derived``. + +  Your code will have to have a separate code path if the object +  you're examining is a class (``isinstance(o, type)``). +  In that case, best practice relies on an implementation detail +  of Python 3.9 and before: if a class has annotations defined, +  they are stored in the class's ``__dict__`` dictionary.  Since +  the class may or may not have annotations defined, best practice +  is to call the ``get`` method on the class dict. + +  To put it all together, here is some sample code that safely +  accesses the ``__annotations__`` attribute on an arbitrary +  object in Python 3.9 and before:: + +      if isinstance(o, type): +          ann = o.__dict__.get('__annotations__', None) +      else: +          ann = getattr(o, '__annotations__', None) + +  After running this code, ``ann`` should be either a +  dictionary or ``None``.  You're encouraged to double-check +  the type of ``ann`` using :func:`isinstance` before further +  examination. + +  Note that some exotic or malformed type objects may not have +  a ``__dict__`` attribute, so for extra safety you may also wish +  to use :func:`getattr` to access ``__dict__``. + + +Manually Un-Stringizing Stringized Annotations +============================================== + +  In situations where some annotations may be "stringized", +  and you wish to evaluate those strings to produce the +  Python values they represent, it really is best to +  call :func:`inspect.get_annotations` to do this work +  for you. + +  If you're using Python 3.9 or older, or if for some reason +  you can't use :func:`inspect.get_annotations`, you'll need +  to duplicate its logic.  You're encouraged to examine the +  implementation of :func:`inspect.get_annotations` in the +  current Python version and follow a similar approach. + +  In a nutshell, if you wish to evaluate a stringized annotation +  on an arbitrary object ``o``: + +  * If ``o`` is a module, use ``o.__dict__`` as the +    ``globals`` when calling :func:`eval`. +  * If ``o`` is a class, use ``sys.modules[o.__module__].__dict__`` +    as the ``globals``, and ``dict(vars(o))`` as the ``locals``, +    when calling :func:`eval`. +  * If ``o`` is a wrapped callable using :func:`functools.update_wrapper`, +    :func:`functools.wraps`, or :func:`functools.partial`, iteratively +    unwrap it by accessing either ``o.__wrapped__`` or ``o.func`` as +    appropriate, until you have found the root unwrapped function. +  * If ``o`` is a callable (but not a class), use +    ``o.__globals__`` as the globals when calling :func:`eval`. + +  However, not all string values used as annotations can +  be successfully turned into Python values by :func:`eval`. +  String values could theoretically contain any valid string, +  and in practice there are valid use cases for type hints that +  require annotating with string values that specifically +  *can't* be evaluated.  For example: + +  * :pep:`604` union types using `|`, before support for this +    was added to Python 3.10. +  * Definitions that aren't needed at runtime, only imported +    when :const:`typing.TYPE_CHECKING` is true. + +  If :func:`eval` attempts to evaluate such values, it will +  fail and raise an exception.  So, when designing a library +  API that works with annotations, it's recommended to only +  attempt to evaluate string values when explicitly requested +  to by the caller. + + +Best Practices For ``__annotations__`` In Any Python Version +============================================================ + +  * You should avoid assigning to the ``__annotations__`` member +    of objects directly.  Let Python manage setting ``__annotations__``. + +  * If you do assign directly to the ``__annotations__`` member +    of an object, you should always set it to a ``dict`` object. + +  * If you directly access the ``__annotations__`` member +    of an object, you should ensure that it's a +    dictionary before attempting to examine its contents. + +  * You should avoid modifying ``__annotations__`` dicts. + +  * You should avoid deleting the ``__annotations__`` attribute +    of an object. + + +``__annotations__`` Quirks +========================== + +  In all versions of Python 3, function +  objects lazy-create an annotations dict if no annotations +  are defined on that object.  You can delete the ``__annotations__`` +  attribute using ``del fn.__annotations__``, but if you then +  access ``fn.__annotations__`` the object will create a new empty dict +  that it will store and return as its annotations.  Deleting the +  annotations on a function before it has lazily created its annotations +  dict will throw an ``AttributeError``; using ``del fn.__annotations__`` +  twice in a row is guaranteed to always throw an ``AttributeError``. + +  Everything in the above paragraph also applies to class and module +  objects in Python 3.10 and newer. + +  In all versions of Python 3, you can set ``__annotations__`` +  on a function object to ``None``.  However, subsequently +  accessing the annotations on that object using ``fn.__annotations__`` +  will lazy-create an empty dictionary as per the first paragraph of +  this section.  This is *not* true of modules and classes, in any Python +  version; those objects permit setting ``__annotations__`` to any +  Python value, and will retain whatever value is set. + +  If Python stringizes your annotations for you +  (using ``from __future__ import annotations``), and you +  specify a string as an annotation, the string will +  itself be quoted.  In effect the annotation is quoted +  *twice.*  For example:: + +       from __future__ import annotations +       def foo(a: "str"): pass + +       print(foo.__annotations__) + +  This prints ``{'a': "'str'"}``.  This shouldn't really be considered +  a "quirk"; it's mentioned here simply because it might be surprising. diff --git a/Doc/howto/index.rst b/Doc/howto/index.rst index e0dacd224d..eae8f143ee 100644 --- a/Doc/howto/index.rst +++ b/Doc/howto/index.rst @@ -30,4 +30,5 @@ Currently, the HOWTOs are:     ipaddress.rst     clinic.rst     instrumentation.rst +   annotations.rst | 
