autodoc: Fix warnings with dataclasses in `Annotated` metadata (#12622)

This commit is contained in:
Adam Turner
2024-07-20 11:28:03 +01:00
committed by GitHub
parent dd77f85149
commit 2bd973e719
7 changed files with 134 additions and 16 deletions

View File

@@ -11,6 +11,10 @@ Bugs fixed
* #12601, #12625: Support callable objects in :py:class:`~typing.Annotated` type * #12601, #12625: Support callable objects in :py:class:`~typing.Annotated` type
metadata in the Python domain. metadata in the Python domain.
Patch by Adam Turner. Patch by Adam Turner.
* #12601, #12622: Resolve :py:class:`~typing.Annotated` warnings with
``sphinx.ext.autodoc``,
especially when using :mod:`dataclasses` as type metadata.
Patch by Adam Turner.
Release 7.4.6 (released Jul 18, 2024) Release 7.4.6 (released Jul 18, 2024)
===================================== =====================================

View File

@@ -2008,7 +2008,8 @@ class UninitializedGlobalVariableMixin(DataDocumenterMixinBase):
with mock(self.config.autodoc_mock_imports): with mock(self.config.autodoc_mock_imports):
parent = import_module(self.modname, self.config.autodoc_warningiserror) parent = import_module(self.modname, self.config.autodoc_warningiserror)
annotations = get_type_hints(parent, None, annotations = get_type_hints(parent, None,
self.config.autodoc_type_aliases) self.config.autodoc_type_aliases,
include_extras=True)
if self.objpath[-1] in annotations: if self.objpath[-1] in annotations:
self.object = UNINITIALIZED_ATTR self.object = UNINITIALIZED_ATTR
self.parent = parent self.parent = parent
@@ -2097,7 +2098,8 @@ class DataDocumenter(GenericAliasMixin,
if self.config.autodoc_typehints != 'none': if self.config.autodoc_typehints != 'none':
# obtain annotation for this data # obtain annotation for this data
annotations = get_type_hints(self.parent, None, annotations = get_type_hints(self.parent, None,
self.config.autodoc_type_aliases) self.config.autodoc_type_aliases,
include_extras=True)
if self.objpath[-1] in annotations: if self.objpath[-1] in annotations:
if self.config.autodoc_typehints_format == "short": if self.config.autodoc_typehints_format == "short":
objrepr = stringify_annotation(annotations.get(self.objpath[-1]), objrepr = stringify_annotation(annotations.get(self.objpath[-1]),
@@ -2541,7 +2543,8 @@ class UninitializedInstanceAttributeMixin(DataDocumenterMixinBase):
def is_uninitialized_instance_attribute(self, parent: Any) -> bool: def is_uninitialized_instance_attribute(self, parent: Any) -> bool:
"""Check the subject is an annotation only attribute.""" """Check the subject is an annotation only attribute."""
annotations = get_type_hints(parent, None, self.config.autodoc_type_aliases) annotations = get_type_hints(parent, None, self.config.autodoc_type_aliases,
include_extras=True)
return self.objpath[-1] in annotations return self.objpath[-1] in annotations
def import_object(self, raiseerror: bool = False) -> bool: def import_object(self, raiseerror: bool = False) -> bool:
@@ -2673,7 +2676,8 @@ class AttributeDocumenter(GenericAliasMixin, SlotsMixin, # type: ignore[misc]
if self.config.autodoc_typehints != 'none': if self.config.autodoc_typehints != 'none':
# obtain type annotation for this attribute # obtain type annotation for this attribute
annotations = get_type_hints(self.parent, None, annotations = get_type_hints(self.parent, None,
self.config.autodoc_type_aliases) self.config.autodoc_type_aliases,
include_extras=True)
if self.objpath[-1] in annotations: if self.objpath[-1] in annotations:
if self.config.autodoc_typehints_format == "short": if self.config.autodoc_typehints_format == "short":
objrepr = stringify_annotation(annotations.get(self.objpath[-1]), objrepr = stringify_annotation(annotations.get(self.objpath[-1]),

View File

@@ -652,7 +652,7 @@ def signature(
try: try:
# Resolve annotations using ``get_type_hints()`` and type_aliases. # Resolve annotations using ``get_type_hints()`` and type_aliases.
localns = TypeAliasNamespace(type_aliases) localns = TypeAliasNamespace(type_aliases)
annotations = typing.get_type_hints(subject, None, localns) annotations = typing.get_type_hints(subject, None, localns, include_extras=True)
for i, param in enumerate(parameters): for i, param in enumerate(parameters):
if param.name in annotations: if param.name in annotations:
annotation = annotations[param.name] annotation = annotations[param.name]

View File

@@ -2,6 +2,7 @@
from __future__ import annotations from __future__ import annotations
import dataclasses
import sys import sys
import types import types
import typing import typing
@@ -157,6 +158,7 @@ def get_type_hints(
obj: Any, obj: Any,
globalns: dict[str, Any] | None = None, globalns: dict[str, Any] | None = None,
localns: dict[str, Any] | None = None, localns: dict[str, Any] | None = None,
include_extras: bool = False,
) -> dict[str, Any]: ) -> dict[str, Any]:
"""Return a dictionary containing type hints for a function, method, module or class """Return a dictionary containing type hints for a function, method, module or class
object. object.
@@ -167,7 +169,7 @@ def get_type_hints(
from sphinx.util.inspect import safe_getattr # lazy loading from sphinx.util.inspect import safe_getattr # lazy loading
try: try:
return typing.get_type_hints(obj, globalns, localns) return typing.get_type_hints(obj, globalns, localns, include_extras=include_extras)
except NameError: except NameError:
# Failed to evaluate ForwardRef (maybe TYPE_CHECKING) # Failed to evaluate ForwardRef (maybe TYPE_CHECKING)
return safe_getattr(obj, '__annotations__', {}) return safe_getattr(obj, '__annotations__', {})
@@ -267,7 +269,20 @@ def restify(cls: Any, mode: _RestifyMode = 'fully-qualified-except-typing') -> s
return f':py:class:`{module_prefix}{_INVALID_BUILTIN_CLASSES[cls]}`' return f':py:class:`{module_prefix}{_INVALID_BUILTIN_CLASSES[cls]}`'
elif _is_annotated_form(cls): elif _is_annotated_form(cls):
args = restify(cls.__args__[0], mode) args = restify(cls.__args__[0], mode)
meta = ', '.join(map(repr, cls.__metadata__)) meta_args = []
for m in cls.__metadata__:
if isinstance(m, type):
meta_args.append(restify(m, mode))
elif dataclasses.is_dataclass(m):
# use restify for the repr of field values rather than repr
d_fields = ', '.join([
fr"{f.name}=\ {restify(getattr(m, f.name), mode)}"
for f in dataclasses.fields(m) if f.repr
])
meta_args.append(fr'{restify(type(m), mode)}\ ({d_fields})')
else:
meta_args.append(repr(m))
meta = ', '.join(meta_args)
if sys.version_info[:2] <= (3, 11): if sys.version_info[:2] <= (3, 11):
# Hardcoded to fix errors on Python 3.11 and earlier. # Hardcoded to fix errors on Python 3.11 and earlier.
return fr':py:class:`~typing.Annotated`\ [{args}, {meta}]' return fr':py:class:`~typing.Annotated`\ [{args}, {meta}]'
@@ -510,7 +525,25 @@ def stringify_annotation(
return f'{module_prefix}Literal[{args}]' return f'{module_prefix}Literal[{args}]'
elif _is_annotated_form(annotation): # for py39+ elif _is_annotated_form(annotation): # for py39+
args = stringify_annotation(annotation_args[0], mode) args = stringify_annotation(annotation_args[0], mode)
meta = ', '.join(map(repr, annotation.__metadata__)) meta_args = []
for m in annotation.__metadata__:
if isinstance(m, type):
meta_args.append(stringify_annotation(m, mode))
elif dataclasses.is_dataclass(m):
# use stringify_annotation for the repr of field values rather than repr
d_fields = ', '.join([
f"{f.name}={stringify_annotation(getattr(m, f.name), mode)}"
for f in dataclasses.fields(m) if f.repr
])
meta_args.append(f'{stringify_annotation(type(m), mode)}({d_fields})')
else:
meta_args.append(repr(m))
meta = ', '.join(meta_args)
if sys.version_info[:2] <= (3, 9):
if mode == 'smart':
return f'~typing.Annotated[{args}, {meta}]'
if mode == 'fully-qualified':
return f'typing.Annotated[{args}, {meta}]'
if sys.version_info[:2] <= (3, 11): if sys.version_info[:2] <= (3, 11):
if mode == 'fully-qualified-except-typing': if mode == 'fully-qualified-except-typing':
return f'Annotated[{args}, {meta}]' return f'Annotated[{args}, {meta}]'

View File

@@ -1,8 +1,42 @@
from __future__ import annotations # from __future__ import annotations
import dataclasses
import types
from typing import Annotated from typing import Annotated
@dataclasses.dataclass(frozen=True)
class FuncValidator:
func: types.FunctionType
@dataclasses.dataclass(frozen=True)
class MaxLen:
max_length: int
whitelisted_words: list[str]
def validate(value: str) -> str:
return value
#: Type alias for a validated string.
ValidatedString = Annotated[str, FuncValidator(validate)]
def hello(name: Annotated[str, "attribute"]) -> None: def hello(name: Annotated[str, "attribute"]) -> None:
"""docstring""" """docstring"""
pass pass
class AnnotatedAttributes:
"""docstring"""
#: Docstring about the ``name`` attribute.
name: Annotated[str, "attribute"]
#: Docstring about the ``max_len`` attribute.
max_len: list[Annotated[str, MaxLen(10, ['word_one', 'word_two'])]]
#: Docstring about the ``validated`` attribute.
validated: ValidatedString

View File

@@ -2321,18 +2321,62 @@ def test_autodoc_TypeVar(app):
@pytest.mark.sphinx('html', testroot='ext-autodoc') @pytest.mark.sphinx('html', testroot='ext-autodoc')
def test_autodoc_Annotated(app): def test_autodoc_Annotated(app):
options = {"members": None} options = {'members': None, 'member-order': 'bysource'}
actual = do_autodoc(app, 'module', 'target.annotated', options) actual = do_autodoc(app, 'module', 'target.annotated', options)
assert list(actual) == [ assert list(actual) == [
'', '',
'.. py:module:: target.annotated', '.. py:module:: target.annotated',
'', '',
'', '',
'.. py:function:: hello(name: str) -> None', '.. py:class:: FuncValidator(func: function)',
' :module: target.annotated',
'',
'',
'.. py:class:: MaxLen(max_length: int, whitelisted_words: list[str])',
' :module: target.annotated',
'',
'',
'.. py:data:: ValidatedString',
' :module: target.annotated',
'',
' Type alias for a validated string.',
'',
' alias of :py:class:`~typing.Annotated`\\ [:py:class:`str`, '
':py:class:`~target.annotated.FuncValidator`\\ (func=\\ :py:class:`~target.annotated.validate`)]',
'',
'',
".. py:function:: hello(name: ~typing.Annotated[str, 'attribute']) -> None",
' :module: target.annotated', ' :module: target.annotated',
'', '',
' docstring', ' docstring',
'', '',
'',
'.. py:class:: AnnotatedAttributes()',
' :module: target.annotated',
'',
' docstring',
'',
'',
' .. py:attribute:: AnnotatedAttributes.name',
' :module: target.annotated',
" :type: ~typing.Annotated[str, 'attribute']",
'',
' Docstring about the ``name`` attribute.',
'',
'',
' .. py:attribute:: AnnotatedAttributes.max_len',
' :module: target.annotated',
" :type: list[~typing.Annotated[str, ~target.annotated.MaxLen(max_length=10, whitelisted_words=['word_one', 'word_two'])]]",
'',
' Docstring about the ``max_len`` attribute.',
'',
'',
' .. py:attribute:: AnnotatedAttributes.validated',
' :module: target.annotated',
' :type: ~typing.Annotated[str, ~target.annotated.FuncValidator(func=~target.annotated.validate)]',
'',
' Docstring about the ``validated`` attribute.',
'',
] ]

View File

@@ -196,8 +196,8 @@ def test_restify_type_hints_containers():
def test_restify_Annotated(): def test_restify_Annotated():
assert restify(Annotated[str, "foo", "bar"]) == ":py:class:`~typing.Annotated`\\ [:py:class:`str`, 'foo', 'bar']" assert restify(Annotated[str, "foo", "bar"]) == ":py:class:`~typing.Annotated`\\ [:py:class:`str`, 'foo', 'bar']"
assert restify(Annotated[str, "foo", "bar"], 'smart') == ":py:class:`~typing.Annotated`\\ [:py:class:`str`, 'foo', 'bar']" assert restify(Annotated[str, "foo", "bar"], 'smart') == ":py:class:`~typing.Annotated`\\ [:py:class:`str`, 'foo', 'bar']"
assert restify(Annotated[float, Gt(-10.0)]) == ':py:class:`~typing.Annotated`\\ [:py:class:`float`, Gt(gt=-10.0)]' assert restify(Annotated[float, Gt(-10.0)]) == ':py:class:`~typing.Annotated`\\ [:py:class:`float`, :py:class:`tests.test_util.test_util_typing.Gt`\\ (gt=\\ -10.0)]'
assert restify(Annotated[float, Gt(-10.0)], 'smart') == ':py:class:`~typing.Annotated`\\ [:py:class:`float`, Gt(gt=-10.0)]' assert restify(Annotated[float, Gt(-10.0)], 'smart') == ':py:class:`~typing.Annotated`\\ [:py:class:`float`, :py:class:`~tests.test_util.test_util_typing.Gt`\\ (gt=\\ -10.0)]'
def test_restify_type_hints_Callable(): def test_restify_type_hints_Callable():
@@ -521,12 +521,11 @@ def test_stringify_type_hints_pep_585():
assert stringify_annotation(tuple[List[dict[int, str]], str, ...], "smart") == "tuple[~typing.List[dict[int, str]], str, ...]" assert stringify_annotation(tuple[List[dict[int, str]], str, ...], "smart") == "tuple[~typing.List[dict[int, str]], str, ...]"
@pytest.mark.xfail(sys.version_info[:2] <= (3, 9), reason='Needs fixing.')
def test_stringify_Annotated(): def test_stringify_Annotated():
assert stringify_annotation(Annotated[str, "foo", "bar"], 'fully-qualified-except-typing') == "Annotated[str, 'foo', 'bar']" assert stringify_annotation(Annotated[str, "foo", "bar"], 'fully-qualified-except-typing') == "Annotated[str, 'foo', 'bar']"
assert stringify_annotation(Annotated[str, "foo", "bar"], 'smart') == "~typing.Annotated[str, 'foo', 'bar']" assert stringify_annotation(Annotated[str, "foo", "bar"], 'smart') == "~typing.Annotated[str, 'foo', 'bar']"
assert stringify_annotation(Annotated[float, Gt(-10.0)], 'fully-qualified-except-typing') == "Annotated[float, Gt(gt=-10.0)]" assert stringify_annotation(Annotated[float, Gt(-10.0)], 'fully-qualified-except-typing') == "Annotated[float, tests.test_util.test_util_typing.Gt(gt=-10.0)]"
assert stringify_annotation(Annotated[float, Gt(-10.0)], 'smart') == "~typing.Annotated[float, Gt(gt=-10.0)]" assert stringify_annotation(Annotated[float, Gt(-10.0)], 'smart') == "~typing.Annotated[float, ~tests.test_util.test_util_typing.Gt(gt=-10.0)]"
def test_stringify_Unpack(): def test_stringify_Unpack():