From fc503d466e90d3c0e3b63ff5185ccad0d4624ba0 Mon Sep 17 00:00:00 2001 From: Adam Turner <9087854+aa-turner@users.noreply.github.com> Date: Wed, 20 Nov 2024 10:17:25 +0000 Subject: [PATCH 01/29] Implement ``_stringify()`` for ``ASTIdentifier`` --- sphinx/domains/c/_ast.py | 3 +++ 1 file changed, 3 insertions(+) diff --git a/sphinx/domains/c/_ast.py b/sphinx/domains/c/_ast.py index 9fedbed5a8c..09f7388253e 100644 --- a/sphinx/domains/c/_ast.py +++ b/sphinx/domains/c/_ast.py @@ -68,6 +68,9 @@ def __str__(self) -> str: def get_display_string(self) -> str: return "[anonymous]" if self.is_anonymous else self.name + def _stringify(self, transform: StringifyTransform) -> str: + return transform(self.get_display_string()) + def describe_signature(self, signode: TextElement, mode: str, env: BuildEnvironment, prefix: str, symbol: Symbol) -> None: # note: slightly different signature of describe_signature due to the prefix From 697da6e4bc58b1cf967154a3ae8092833b0731ff Mon Sep 17 00:00:00 2001 From: Adam Turner <9087854+aa-turner@users.noreply.github.com> Date: Thu, 21 Nov 2024 19:08:04 +0000 Subject: [PATCH 02/29] Remove possible RecursionError in ``ASTBaseBase.__repr__()`` --- sphinx/util/cfamily.py | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/sphinx/util/cfamily.py b/sphinx/util/cfamily.py index 127b1b247b5..05b5b89222f 100644 --- a/sphinx/util/cfamily.py +++ b/sphinx/util/cfamily.py @@ -115,7 +115,7 @@ def clone(self) -> Any: return deepcopy(self) def _stringify(self, transform: StringifyTransform) -> str: - raise NotImplementedError(repr(self)) + raise NotImplementedError def __str__(self) -> str: return self._stringify(str) @@ -124,7 +124,9 @@ def get_display_string(self) -> str: return self._stringify(lambda ast: ast.get_display_string()) def __repr__(self) -> str: - return f'<{self.__class__.__name__}: {self._stringify(repr)}>' + if repr_string := self._stringify(repr): + return f'<{self.__class__.__name__}: {repr_string}>' + return f'<{self.__class__.__name__}>' ################################################################################ From ddd5aede237c4c7968cf79f5a8ba84ced1a5690c Mon Sep 17 00:00:00 2001 From: Adam Turner <9087854+AA-Turner@users.noreply.github.com> Date: Fri, 22 Nov 2024 14:54:50 +0000 Subject: [PATCH 03/29] Ensure `changeset.descname` is a string when not None (#13150) --- sphinx/builders/changes.py | 5 +---- sphinx/directives/__init__.py | 6 +++++- 2 files changed, 6 insertions(+), 5 deletions(-) diff --git a/sphinx/builders/changes.py b/sphinx/builders/changes.py index 918ed3bcaee..49cab2c7a8c 100644 --- a/sphinx/builders/changes.py +++ b/sphinx/builders/changes.py @@ -61,10 +61,7 @@ def write_documents(self, _docnames: Set[str]) -> None: return logger.info(bold(__('writing summary file...'))) for changeset in changesets: - if isinstance(changeset.descname, tuple): - descname = changeset.descname[0] - else: - descname = changeset.descname + descname = changeset.descname ttext = self.typemap[changeset.type] context = changeset.content.replace('\n', ' ') if descname and changeset.docname.startswith('c-api'): diff --git a/sphinx/directives/__init__.py b/sphinx/directives/__init__.py index 181c6f81a07..c45232c92e7 100644 --- a/sphinx/directives/__init__.py +++ b/sphinx/directives/__init__.py @@ -282,7 +282,11 @@ def run(self) -> list[Node]: if self.names: # needed for association of version{added,changed} directives - self.env.temp_data['object'] = self.names[0] + object_name: ObjDescT = self.names[0] + if isinstance(object_name, tuple): + self.env.temp_data['object'] = str(object_name[0]) + else: + self.env.temp_data['object'] = str(object_name) self.before_content() content_children = self.parse_content_to_nodes(allow_section_headings=True) content_node = addnodes.desc_content('', *content_children) From 6a8d70a2fdbbc29784b83a9e4643fffd0ff8d17e Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Rub=C3=A9n=20de=20Celis=20Hern=C3=A1ndez?= Date: Fri, 22 Nov 2024 18:03:18 +0100 Subject: [PATCH 04/29] Fix incorrect reference to ``--config-dir`` in the documentation (#13148) The correct argument name is ``--conf-dir``. --- doc/man/sphinx-build.rst | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/doc/man/sphinx-build.rst b/doc/man/sphinx-build.rst index 4c8fde48e96..63af7e49b4c 100644 --- a/doc/man/sphinx-build.rst +++ b/doc/man/sphinx-build.rst @@ -141,7 +141,7 @@ Options .. versionchanged:: 6.2 Add ``--jobs`` long option. -.. option:: -c path, --config-dir path +.. option:: -c path, --conf-dir path Don't look for the :file:`conf.py` in the source directory, but use the given configuration directory instead. Note that various other files and paths @@ -152,7 +152,7 @@ Options .. versionadded:: 0.3 .. versionchanged:: 7.3 - Add ``--config-dir`` long option. + Add ``--conf-dir`` long option. .. option:: -C, --isolated From f49f6eb656cb2398d1219d7bbb016d21e921c15a Mon Sep 17 00:00:00 2001 From: Rafael Fontenelle Date: Fri, 22 Nov 2024 14:04:28 -0300 Subject: [PATCH 05/29] Remove unnecessary parentheses in meth role (#13143) --- sphinx/builders/__init__.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/sphinx/builders/__init__.py b/sphinx/builders/__init__.py index 1420417380d..620ff82abe5 100644 --- a/sphinx/builders/__init__.py +++ b/sphinx/builders/__init__.py @@ -81,7 +81,7 @@ class Builder: # doctree versioning method versioning_method = 'none' versioning_compare = False - #: Whether it is safe to make parallel :meth:`~.Builder.write_doc()` calls. + #: Whether it is safe to make parallel :meth:`~.Builder.write_doc` calls. allow_parallel: bool = False # support translation use_message_catalog = True From 1deaf8572067c3b4766d80093de1976541b9a001 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Fri, 22 Nov 2024 17:05:57 +0000 Subject: [PATCH 06/29] Bump codecov-action to v5 (#13135) --- .github/workflows/main.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/main.yml b/.github/workflows/main.yml index c2591d82a58..caf633d3785 100644 --- a/.github/workflows/main.yml +++ b/.github/workflows/main.yml @@ -342,4 +342,4 @@ jobs: env: VIRTUALENV_SYSTEM_SITE_PACKAGES: "1" - name: codecov - uses: codecov/codecov-action@v4 + uses: codecov/codecov-action@v5 From 92210f54d939a3b4bbdcc5550c8eebd43f008549 Mon Sep 17 00:00:00 2001 From: Jens Hedegaard Nielsen Date: Fri, 22 Nov 2024 18:08:42 +0100 Subject: [PATCH 07/29] Use debug log level for AttributeErrors in viewcode (#13016) --- sphinx/ext/viewcode.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/sphinx/ext/viewcode.py b/sphinx/ext/viewcode.py index b64d67bdad3..da1713aa6fa 100644 --- a/sphinx/ext/viewcode.py +++ b/sphinx/ext/viewcode.py @@ -65,8 +65,8 @@ def _get_full_modname(modname: str, attribute: str) -> str | None: return getattr(value, '__module__', None) except AttributeError: # sphinx.ext.viewcode can't follow class instance attribute - # then AttributeError logging output only verbose mode. - logger.verbose("Didn't find %s in %s", attribute, modname) + # then AttributeError logging output only debug mode. + logger.debug("Didn't find %s in %s", attribute, modname) return None except Exception as e: # sphinx.ext.viewcode follow python domain directives. From f6f59b48715f9b272223222b5ab26bb60a83510b Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Fri, 22 Nov 2024 17:11:13 +0000 Subject: [PATCH 08/29] Bump ruff to 0.7.4 (#13141) Co-authored-by: Adam Turner <9087854+AA-Turner@users.noreply.github.com> --- pyproject.toml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pyproject.toml b/pyproject.toml index 11ed6a00b9d..e7adb48e64c 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -82,7 +82,7 @@ docs = [ ] lint = [ "flake8>=6.0", - "ruff==0.7.3", + "ruff==0.7.4", "mypy==1.13.0", "sphinx-lint>=0.9", "types-colorama==0.4.15.20240311", From 3cfccf3f1d669454b9ee4eab72391ccbc957ed34 Mon Sep 17 00:00:00 2001 From: Adam Turner <9087854+aa-turner@users.noreply.github.com> Date: Fri, 22 Nov 2024 17:14:49 +0000 Subject: [PATCH 09/29] Check exception message in test_env_relfn2path --- tests/test_environment/test_environment.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/test_environment/test_environment.py b/tests/test_environment/test_environment.py index 10e98584342..05427faf56f 100644 --- a/tests/test_environment/test_environment.py +++ b/tests/test_environment/test_environment.py @@ -190,7 +190,7 @@ def test_env_relfn2path(app): # omit docname (w/o current docname) app.env.temp_data.clear() - with pytest.raises(KeyError): + with pytest.raises(KeyError, match=r"^'docname'$"): app.env.relfn2path('images/logo.jpg') From f5d4ebdb2ef15aead2af1285e32c1e7ec624ff7b Mon Sep 17 00:00:00 2001 From: Adam Turner <9087854+aa-turner@users.noreply.github.com> Date: Fri, 22 Nov 2024 17:15:18 +0000 Subject: [PATCH 10/29] Narrow the type of `ChangeSet.descname` to `str` --- sphinx/directives/__init__.py | 2 +- sphinx/domains/changeset.py | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/sphinx/directives/__init__.py b/sphinx/directives/__init__.py index c45232c92e7..4c92af67d9d 100644 --- a/sphinx/directives/__init__.py +++ b/sphinx/directives/__init__.py @@ -296,7 +296,7 @@ def run(self) -> list[Node]: 'object-description-transform', self.domain, self.objtype, content_node ) DocFieldTransformer(self).transform_all(content_node) - self.env.temp_data['object'] = None + self.env.temp_data['object'] = '' self.after_content() if node['no-typesetting']: diff --git a/sphinx/domains/changeset.py b/sphinx/domains/changeset.py index a27c5f571a8..abf622d5bb8 100644 --- a/sphinx/domains/changeset.py +++ b/sphinx/domains/changeset.py @@ -41,7 +41,7 @@ class ChangeSet(NamedTuple): docname: str lineno: int module: str | None - descname: str | None + descname: str content: str @@ -123,7 +123,7 @@ def changesets(self) -> dict[str, list[ChangeSet]]: def note_changeset(self, node: addnodes.versionmodified) -> None: version = node['version'] module = self.env.ref_context.get('py:module') - objname = self.env.temp_data.get('object') + objname = self.env.temp_data.get('object', '') changeset = ChangeSet(node['type'], self.env.docname, node.line, # type: ignore[arg-type] module, objname, node.astext()) self.changesets.setdefault(version, []).append(changeset) From 0e15ad7f3381ab30d47761c7cb7c7cf29909ed2c Mon Sep 17 00:00:00 2001 From: Adam Turner <9087854+aa-turner@users.noreply.github.com> Date: Fri, 22 Nov 2024 17:24:07 +0000 Subject: [PATCH 11/29] Bump Ruff to 0.8.0 --- .ruff.toml | 2 +- pyproject.toml | 2 +- sphinx/_cli/util/colour.py | 2 +- sphinx/directives/__init__.py | 4 ++-- sphinx/util/docutils.py | 6 +++--- 5 files changed, 8 insertions(+), 8 deletions(-) diff --git a/.ruff.toml b/.ruff.toml index bb1503095de..e8c94eba2c8 100644 --- a/.ruff.toml +++ b/.ruff.toml @@ -292,7 +292,7 @@ select = [ [lint.per-file-ignores] "doc/*" = [ "ANN", # documentation doesn't need annotations - "TCH001", # documentation doesn't need type-checking blocks + "TC001", # documentation doesn't need type-checking blocks ] "doc/conf.py" = ["INP001", "W605"] "doc/development/tutorials/examples/*" = ["INP001"] diff --git a/pyproject.toml b/pyproject.toml index e7adb48e64c..f479619fee4 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -82,7 +82,7 @@ docs = [ ] lint = [ "flake8>=6.0", - "ruff==0.7.4", + "ruff==0.8.0", "mypy==1.13.0", "sphinx-lint>=0.9", "types-colorama==0.4.15.20240311", diff --git a/sphinx/_cli/util/colour.py b/sphinx/_cli/util/colour.py index bc1a610ba0e..dd01c56cd58 100644 --- a/sphinx/_cli/util/colour.py +++ b/sphinx/_cli/util/colour.py @@ -4,7 +4,7 @@ import os import sys -from collections.abc import Callable # NoQA: TCH003 +from collections.abc import Callable # NoQA: TC003 if sys.platform == 'win32': import colorama diff --git a/sphinx/directives/__init__.py b/sphinx/directives/__init__.py index 4c92af67d9d..d92678ab8f2 100644 --- a/sphinx/directives/__init__.py +++ b/sphinx/directives/__init__.py @@ -9,11 +9,11 @@ from docutils.parsers.rst import directives, roles from sphinx import addnodes -from sphinx.addnodes import desc_signature # NoQA: TCH001 +from sphinx.addnodes import desc_signature # NoQA: TC001 from sphinx.util import docutils from sphinx.util.docfields import DocFieldTransformer, Field, TypedField from sphinx.util.docutils import SphinxDirective -from sphinx.util.typing import ExtensionMetadata, OptionSpec # NoQA: TCH001 +from sphinx.util.typing import ExtensionMetadata, OptionSpec # NoQA: TC001 if TYPE_CHECKING: from docutils.nodes import Node diff --git a/sphinx/util/docutils.py b/sphinx/util/docutils.py index 87823bd69fd..f8fa00f4316 100644 --- a/sphinx/util/docutils.py +++ b/sphinx/util/docutils.py @@ -4,7 +4,7 @@ import os import re -from collections.abc import Sequence # NoQA: TCH003 +from collections.abc import Sequence # NoQA: TC003 from contextlib import contextmanager from copy import copy from pathlib import Path @@ -14,7 +14,7 @@ from docutils import nodes from docutils.io import FileOutput from docutils.parsers.rst import Directive, directives, roles -from docutils.parsers.rst.states import Inliner # NoQA: TCH002 +from docutils.parsers.rst.states import Inliner # NoQA: TC002 from docutils.statemachine import State, StateMachine, StringList from docutils.utils import Reporter, unescape @@ -29,7 +29,7 @@ ) if TYPE_CHECKING: - from collections.abc import Callable, Iterator # NoQA: TCH003 + from collections.abc import Callable, Iterator # NoQA: TC003 from types import ModuleType, TracebackType from docutils.frontend import Values From 3af48520d27e15a1c74c0d994ba8865d20894667 Mon Sep 17 00:00:00 2001 From: Adam Turner <9087854+aa-turner@users.noreply.github.com> Date: Fri, 22 Nov 2024 21:54:26 +0000 Subject: [PATCH 12/29] Require the PEP 563 'annotations' future import --- .ruff.toml | 7 +++++-- sphinx/__init__.py | 2 ++ sphinx/__main__.py | 2 ++ sphinx/builders/latex/nodes.py | 2 ++ sphinx/environment/adapters/asset.py | 8 +++++++- sphinx/ext/intersphinx/__main__.py | 2 ++ sphinx/pygments_styles.py | 2 ++ sphinx/testing/restructuredtext.py | 11 +++++++++-- sphinx/util/_lines.py | 3 +++ sphinx/util/build_phase.py | 2 ++ sphinx/util/http_date.py | 2 ++ tests/test_builders/test_build.py | 2 ++ tests/test_builders/test_build_changes.py | 2 ++ tests/test_builders/test_build_dirhtml.py | 2 ++ tests/test_builders/test_build_epub.py | 2 ++ tests/test_builders/test_build_gettext.py | 2 ++ tests/test_builders/test_build_html_assets.py | 2 ++ tests/test_builders/test_build_html_code.py | 2 ++ tests/test_builders/test_build_html_copyright.py | 2 ++ tests/test_builders/test_build_html_download.py | 2 ++ tests/test_builders/test_build_html_highlight.py | 2 ++ tests/test_builders/test_build_html_image.py | 2 ++ tests/test_builders/test_build_html_maths.py | 2 ++ tests/test_builders/test_build_html_numfig.py | 2 ++ tests/test_builders/test_build_html_tocdepth.py | 2 ++ tests/test_builders/test_build_html_toctree.py | 2 ++ tests/test_builders/test_build_latex.py | 2 ++ tests/test_builders/test_build_manpage.py | 2 ++ tests/test_builders/test_build_texinfo.py | 2 ++ tests/test_builders/test_build_warnings.py | 2 ++ tests/test_builders/test_incremental_reading.py | 2 ++ tests/test_config/test_copyright.py | 2 ++ tests/test_directives/test_directive_code.py | 2 ++ tests/test_directives/test_directive_only.py | 2 ++ tests/test_directives/test_directive_option.py | 2 ++ tests/test_directives/test_directive_other.py | 2 ++ tests/test_directives/test_directive_patch.py | 2 ++ .../test_directives/test_directives_no_typesetting.py | 2 ++ tests/test_domains/test_domain_c.py | 7 ++++++- tests/test_domains/test_domain_cpp.py | 7 ++++++- tests/test_domains/test_domain_js.py | 2 ++ tests/test_domains/test_domain_rst.py | 2 ++ tests/test_domains/test_domain_std.py | 2 ++ tests/test_environment/test_environment.py | 2 ++ .../test_environment/test_environment_indexentries.py | 2 ++ .../test_environment_record_dependencies.py | 2 ++ tests/test_environment/test_environment_toctree.py | 2 ++ tests/test_errors.py | 2 ++ tests/test_events.py | 2 ++ .../ext_napoleon_pep526_data_google.py | 2 ++ .../test_extensions/ext_napoleon_pep526_data_numpy.py | 2 ++ tests/test_extensions/test_ext_apidoc.py | 7 ++++++- .../test_extensions/test_ext_autodoc_autoattribute.py | 2 ++ tests/test_extensions/test_ext_autodoc_autodata.py | 2 ++ .../test_extensions/test_ext_autodoc_autofunction.py | 2 ++ tests/test_extensions/test_ext_autodoc_automodule.py | 2 ++ .../test_extensions/test_ext_autodoc_autoproperty.py | 2 ++ tests/test_extensions/test_ext_autodoc_configs.py | 9 +++++++-- tests/test_extensions/test_ext_autodoc_events.py | 2 ++ .../test_ext_autodoc_preserve_defaults.py | 2 ++ .../test_ext_autodoc_private_members.py | 2 ++ tests/test_extensions/test_ext_autosectionlabel.py | 2 ++ tests/test_extensions/test_ext_autosummary.py | 7 ++++++- tests/test_extensions/test_ext_autosummary_imports.py | 2 ++ tests/test_extensions/test_ext_coverage.py | 2 ++ tests/test_extensions/test_ext_doctest.py | 2 ++ tests/test_extensions/test_ext_duration.py | 2 ++ tests/test_extensions/test_ext_extlinks.py | 2 ++ tests/test_extensions/test_ext_githubpages.py | 2 ++ tests/test_extensions/test_ext_graphviz.py | 2 ++ tests/test_extensions/test_ext_ifconfig.py | 2 ++ tests/test_extensions/test_ext_imgconverter.py | 2 ++ tests/test_extensions/test_ext_imgmockconverter.py | 2 ++ tests/test_extensions/test_ext_inheritance_diagram.py | 2 ++ tests/test_extensions/test_ext_math.py | 2 ++ tests/test_extensions/test_ext_napoleon.py | 2 ++ tests/test_extensions/test_ext_napoleon_docstring.py | 2 ++ tests/test_extensions/test_ext_todo.py | 2 ++ tests/test_extensions/test_extension.py | 2 ++ tests/test_highlighting.py | 2 ++ tests/test_intl/test_catalogs.py | 2 ++ tests/test_intl/test_intl.py | 7 ++++++- tests/test_markup/test_markup.py | 2 ++ tests/test_markup/test_metadata.py | 1 + tests/test_markup/test_parser.py | 2 ++ tests/test_markup/test_smartquotes.py | 2 ++ tests/test_project.py | 2 ++ tests/test_pycode/test_pycode.py | 2 ++ tests/test_pycode/test_pycode_ast.py | 2 ++ tests/test_pycode/test_pycode_parser.py | 2 ++ tests/test_quickstart.py | 10 +++++++--- tests/test_roles.py | 2 ++ tests/test_theming/test_html_theme.py | 2 ++ tests/test_theming/test_templating.py | 2 ++ tests/test_theming/test_theming.py | 2 ++ .../test_transforms_move_module_targets.py | 2 ++ .../test_transforms_post_transforms_code.py | 2 ++ .../test_transforms_post_transforms_images.py | 2 ++ .../test_transforms/test_transforms_reorder_nodes.py | 2 ++ tests/test_transforms/test_unreferenced_footnotes.py | 10 ++++++++-- tests/test_util/test_util.py | 2 ++ tests/test_util/test_util_display.py | 2 ++ tests/test_util/test_util_docstrings.py | 2 ++ tests/test_util/test_util_fileutil.py | 2 ++ tests/test_util/test_util_i18n.py | 2 ++ tests/test_util/test_util_images.py | 2 ++ tests/test_util/test_util_importer.py | 2 ++ tests/test_util/test_util_lines.py | 2 ++ tests/test_util/test_util_logging.py | 2 ++ tests/test_util/test_util_matching.py | 2 ++ tests/test_util/test_util_rst.py | 2 ++ tests/test_util/test_util_template.py | 2 ++ tests/test_util/test_util_typing.py | 2 ++ tests/test_util/test_util_uri.py | 2 ++ tests/test_versioning.py | 2 ++ tests/test_writers/test_api_translator.py | 2 ++ tests/test_writers/test_docutilsconf.py | 2 ++ tests/test_writers/test_writer_latex.py | 2 ++ utils/babel_runner.py | 2 ++ utils/bump_docker.py | 2 ++ utils/convert_attestations.py | 2 ++ utils/generate_js_fixtures.py | 1 + 122 files changed, 294 insertions(+), 17 deletions(-) diff --git a/.ruff.toml b/.ruff.toml index e8c94eba2c8..be9ac015a33 100644 --- a/.ruff.toml +++ b/.ruff.toml @@ -295,7 +295,7 @@ select = [ "TC001", # documentation doesn't need type-checking blocks ] "doc/conf.py" = ["INP001", "W605"] -"doc/development/tutorials/examples/*" = ["INP001"] +"doc/development/tutorials/examples/*" = ["I002", "INP001"] # allow print() in the tutorial "doc/development/tutorials/examples/recipe.py" = [ "FURB118", @@ -360,7 +360,7 @@ select = [ # these tests need old ``typing`` generic aliases "tests/test_util/test_util_typing.py" = ["UP006", "UP007", "UP035"] -"tests/test_util/typing_test_data.py" = ["FA100", "PYI030", "UP006", "UP007", "UP035"] +"tests/test_util/typing_test_data.py" = ["FA100", "I002", "PYI030", "UP006", "UP007", "UP035"] "utils/*" = [ "T201", # whitelist ``print`` for stdout messages @@ -377,6 +377,9 @@ inline-quotes = "single" forced-separate = [ "tests", ] +required-imports = [ + "from __future__ import annotations", +] [format] preview = true diff --git a/sphinx/__init__.py b/sphinx/__init__.py index 6b610e28899..27e613dc3c9 100644 --- a/sphinx/__init__.py +++ b/sphinx/__init__.py @@ -1,5 +1,7 @@ """The Sphinx documentation toolchain.""" +from __future__ import annotations + __version__ = '8.2.0' __display_version__ = __version__ # used for command line version diff --git a/sphinx/__main__.py b/sphinx/__main__.py index dd0df10ecbb..13d626301a9 100644 --- a/sphinx/__main__.py +++ b/sphinx/__main__.py @@ -1,5 +1,7 @@ """The Sphinx documentation toolchain.""" +from __future__ import annotations + import sys from sphinx.cmd.build import main diff --git a/sphinx/builders/latex/nodes.py b/sphinx/builders/latex/nodes.py index f13c29cf6ea..0e7ebb6a7b0 100644 --- a/sphinx/builders/latex/nodes.py +++ b/sphinx/builders/latex/nodes.py @@ -1,5 +1,7 @@ """Additional nodes for LaTeX writer.""" +from __future__ import annotations + from docutils import nodes diff --git a/sphinx/environment/adapters/asset.py b/sphinx/environment/adapters/asset.py index dc0cf766957..9e6875eeea9 100644 --- a/sphinx/environment/adapters/asset.py +++ b/sphinx/environment/adapters/asset.py @@ -1,8 +1,14 @@ """Assets adapter for sphinx.environment.""" -from sphinx.environment import BuildEnvironment +from __future__ import annotations + +from typing import TYPE_CHECKING + from sphinx.util._pathlib import _StrPath +if TYPE_CHECKING: + from sphinx.environment import BuildEnvironment + class ImageAdapter: def __init__(self, env: BuildEnvironment) -> None: diff --git a/sphinx/ext/intersphinx/__main__.py b/sphinx/ext/intersphinx/__main__.py index 9b788d2362c..aa132f9c6d8 100644 --- a/sphinx/ext/intersphinx/__main__.py +++ b/sphinx/ext/intersphinx/__main__.py @@ -1,5 +1,7 @@ """Command line interface for the intersphinx extension.""" +from __future__ import annotations + import logging as _logging import sys diff --git a/sphinx/pygments_styles.py b/sphinx/pygments_styles.py index 55ca71b19d2..57924664f81 100644 --- a/sphinx/pygments_styles.py +++ b/sphinx/pygments_styles.py @@ -1,5 +1,7 @@ """Sphinx theme specific highlighting styles.""" +from __future__ import annotations + from pygments.style import Style from pygments.styles.friendly import FriendlyStyle from pygments.token import ( diff --git a/sphinx/testing/restructuredtext.py b/sphinx/testing/restructuredtext.py index 620e8483492..7231c464c72 100644 --- a/sphinx/testing/restructuredtext.py +++ b/sphinx/testing/restructuredtext.py @@ -1,11 +1,18 @@ -from docutils import nodes +from __future__ import annotations + +from typing import TYPE_CHECKING + from docutils.core import publish_doctree -from sphinx.application import Sphinx from sphinx.io import SphinxStandaloneReader from sphinx.parsers import RSTParser from sphinx.util.docutils import sphinx_domains +if TYPE_CHECKING: + from docutils import nodes + + from sphinx.application import Sphinx + def parse(app: Sphinx, text: str, docname: str = 'index') -> nodes.document: """Parse a string as reStructuredText with Sphinx application.""" diff --git a/sphinx/util/_lines.py b/sphinx/util/_lines.py index e9f8899cd29..ab0e2e032bf 100644 --- a/sphinx/util/_lines.py +++ b/sphinx/util/_lines.py @@ -1,3 +1,6 @@ +from __future__ import annotations + + def parse_line_num_spec(spec: str, total: int) -> list[int]: """Parse a line number spec (such as "1,2,4-6") and return a list of wanted line numbers. diff --git a/sphinx/util/build_phase.py b/sphinx/util/build_phase.py index 76e94a9b0d2..382fc722ee8 100644 --- a/sphinx/util/build_phase.py +++ b/sphinx/util/build_phase.py @@ -1,5 +1,7 @@ """Build phase of Sphinx application.""" +from __future__ import annotations + from enum import IntEnum diff --git a/sphinx/util/http_date.py b/sphinx/util/http_date.py index d8c51ae9b58..db0ae306b54 100644 --- a/sphinx/util/http_date.py +++ b/sphinx/util/http_date.py @@ -3,6 +3,8 @@ Reference: https://www.rfc-editor.org/rfc/rfc7231#section-7.1.1.1 """ +from __future__ import annotations + import time import warnings from email.utils import parsedate_tz diff --git a/tests/test_builders/test_build.py b/tests/test_builders/test_build.py index 5f396cc48af..2b922eaca76 100644 --- a/tests/test_builders/test_build.py +++ b/tests/test_builders/test_build.py @@ -1,5 +1,7 @@ """Test all builders.""" +from __future__ import annotations + import os import shutil from contextlib import contextmanager diff --git a/tests/test_builders/test_build_changes.py b/tests/test_builders/test_build_changes.py index 69db01ba3e5..f252b3ce3a6 100644 --- a/tests/test_builders/test_build_changes.py +++ b/tests/test_builders/test_build_changes.py @@ -1,5 +1,7 @@ """Test the ChangesBuilder class.""" +from __future__ import annotations + import pytest diff --git a/tests/test_builders/test_build_dirhtml.py b/tests/test_builders/test_build_dirhtml.py index 3e78f1ed0c8..566fb1c20e3 100644 --- a/tests/test_builders/test_build_dirhtml.py +++ b/tests/test_builders/test_build_dirhtml.py @@ -1,5 +1,7 @@ """Test dirhtml builder.""" +from __future__ import annotations + import posixpath import pytest diff --git a/tests/test_builders/test_build_epub.py b/tests/test_builders/test_build_epub.py index 3c2be6323d4..c025b2f8fc0 100644 --- a/tests/test_builders/test_build_epub.py +++ b/tests/test_builders/test_build_epub.py @@ -1,5 +1,7 @@ """Test the HTML builder and check output against XPath.""" +from __future__ import annotations + import os import subprocess import xml.etree.ElementTree as ET diff --git a/tests/test_builders/test_build_gettext.py b/tests/test_builders/test_build_gettext.py index 00d7f826ecc..d78f1e71b00 100644 --- a/tests/test_builders/test_build_gettext.py +++ b/tests/test_builders/test_build_gettext.py @@ -1,5 +1,7 @@ """Test the build process with gettext builder with the test root.""" +from __future__ import annotations + import gettext import re import subprocess diff --git a/tests/test_builders/test_build_html_assets.py b/tests/test_builders/test_build_html_assets.py index f7d21fe61b6..064c8c85ab5 100644 --- a/tests/test_builders/test_build_html_assets.py +++ b/tests/test_builders/test_build_html_assets.py @@ -1,5 +1,7 @@ """Test the HTML builder and check output against XPath.""" +from __future__ import annotations + import re from pathlib import Path diff --git a/tests/test_builders/test_build_html_code.py b/tests/test_builders/test_build_html_code.py index d32c0b3e2b9..349e1d1641a 100644 --- a/tests/test_builders/test_build_html_code.py +++ b/tests/test_builders/test_build_html_code.py @@ -1,3 +1,5 @@ +from __future__ import annotations + import pytest diff --git a/tests/test_builders/test_build_html_copyright.py b/tests/test_builders/test_build_html_copyright.py index 8e017ede4d7..7a18bceb603 100644 --- a/tests/test_builders/test_build_html_copyright.py +++ b/tests/test_builders/test_build_html_copyright.py @@ -1,3 +1,5 @@ +from __future__ import annotations + import time import pytest diff --git a/tests/test_builders/test_build_html_download.py b/tests/test_builders/test_build_html_download.py index 4855949cd60..f3f5a1fd43b 100644 --- a/tests/test_builders/test_build_html_download.py +++ b/tests/test_builders/test_build_html_download.py @@ -1,3 +1,5 @@ +from __future__ import annotations + import hashlib import re diff --git a/tests/test_builders/test_build_html_highlight.py b/tests/test_builders/test_build_html_highlight.py index 07d5e3e9887..0d713c7e871 100644 --- a/tests/test_builders/test_build_html_highlight.py +++ b/tests/test_builders/test_build_html_highlight.py @@ -1,3 +1,5 @@ +from __future__ import annotations + from unittest.mock import ANY, call, patch import pytest diff --git a/tests/test_builders/test_build_html_image.py b/tests/test_builders/test_build_html_image.py index 5a061ed4dec..bb4baa5b643 100644 --- a/tests/test_builders/test_build_html_image.py +++ b/tests/test_builders/test_build_html_image.py @@ -1,3 +1,5 @@ +from __future__ import annotations + import re from pathlib import Path diff --git a/tests/test_builders/test_build_html_maths.py b/tests/test_builders/test_build_html_maths.py index 0f776917e48..9ab04922607 100644 --- a/tests/test_builders/test_build_html_maths.py +++ b/tests/test_builders/test_build_html_maths.py @@ -1,3 +1,5 @@ +from __future__ import annotations + import pytest from sphinx.errors import ConfigError diff --git a/tests/test_builders/test_build_html_numfig.py b/tests/test_builders/test_build_html_numfig.py index 833918fccdd..95279b37728 100644 --- a/tests/test_builders/test_build_html_numfig.py +++ b/tests/test_builders/test_build_html_numfig.py @@ -1,5 +1,7 @@ """Test the HTML builder and check output against XPath.""" +from __future__ import annotations + import os import re diff --git a/tests/test_builders/test_build_html_tocdepth.py b/tests/test_builders/test_build_html_tocdepth.py index 7d6afc4c1c3..702aab8c69f 100644 --- a/tests/test_builders/test_build_html_tocdepth.py +++ b/tests/test_builders/test_build_html_tocdepth.py @@ -1,5 +1,7 @@ """Test the HTML builder and check output against XPath.""" +from __future__ import annotations + import pytest from tests.test_builders.xpath_html_util import _intradocument_hyperlink_check diff --git a/tests/test_builders/test_build_html_toctree.py b/tests/test_builders/test_build_html_toctree.py index a59de6328e6..08a6268cf4c 100644 --- a/tests/test_builders/test_build_html_toctree.py +++ b/tests/test_builders/test_build_html_toctree.py @@ -1,5 +1,7 @@ """Test the HTML builder and check output against XPath.""" +from __future__ import annotations + import re from unittest.mock import patch diff --git a/tests/test_builders/test_build_latex.py b/tests/test_builders/test_build_latex.py index d95c5e4c286..71141d573d7 100644 --- a/tests/test_builders/test_build_latex.py +++ b/tests/test_builders/test_build_latex.py @@ -1,5 +1,7 @@ """Test the build process with LaTeX builder with the test root.""" +from __future__ import annotations + import http.server import os import re diff --git a/tests/test_builders/test_build_manpage.py b/tests/test_builders/test_build_manpage.py index 17ef00e612d..d80a067504f 100644 --- a/tests/test_builders/test_build_manpage.py +++ b/tests/test_builders/test_build_manpage.py @@ -1,5 +1,7 @@ """Test the build process with manpage builder with the test root.""" +from __future__ import annotations + import docutils import pytest diff --git a/tests/test_builders/test_build_texinfo.py b/tests/test_builders/test_build_texinfo.py index e77306c3c52..5a8ce2d5e02 100644 --- a/tests/test_builders/test_build_texinfo.py +++ b/tests/test_builders/test_build_texinfo.py @@ -1,5 +1,7 @@ """Test the build process with Texinfo builder with the test root.""" +from __future__ import annotations + import re import subprocess from pathlib import Path diff --git a/tests/test_builders/test_build_warnings.py b/tests/test_builders/test_build_warnings.py index c89e33b93d5..cf307cbbf0c 100644 --- a/tests/test_builders/test_build_warnings.py +++ b/tests/test_builders/test_build_warnings.py @@ -1,3 +1,5 @@ +from __future__ import annotations + import os import re import sys diff --git a/tests/test_builders/test_incremental_reading.py b/tests/test_builders/test_incremental_reading.py index e1ecda7f4af..4ae1b77e1b1 100644 --- a/tests/test_builders/test_incremental_reading.py +++ b/tests/test_builders/test_incremental_reading.py @@ -1,5 +1,7 @@ """Test the Builder class.""" +from __future__ import annotations + import sys import pytest diff --git a/tests/test_config/test_copyright.py b/tests/test_config/test_copyright.py index 72cef60709b..fdabcc08123 100644 --- a/tests/test_config/test_copyright.py +++ b/tests/test_config/test_copyright.py @@ -1,5 +1,7 @@ """Test copyright year adjustment""" +from __future__ import annotations + import time import pytest diff --git a/tests/test_directives/test_directive_code.py b/tests/test_directives/test_directive_code.py index 0b81b65c1d8..5acebd08d60 100644 --- a/tests/test_directives/test_directive_code.py +++ b/tests/test_directives/test_directive_code.py @@ -1,5 +1,7 @@ """Test the code-block directive.""" +from __future__ import annotations + import pytest from docutils import nodes diff --git a/tests/test_directives/test_directive_only.py b/tests/test_directives/test_directive_only.py index 297f304dfdb..7d5f15674c3 100644 --- a/tests/test_directives/test_directive_only.py +++ b/tests/test_directives/test_directive_only.py @@ -1,5 +1,7 @@ """Test the only directive with the test root.""" +from __future__ import annotations + import re import pytest diff --git a/tests/test_directives/test_directive_option.py b/tests/test_directives/test_directive_option.py index 8f07e538545..11369a4346d 100644 --- a/tests/test_directives/test_directive_option.py +++ b/tests/test_directives/test_directive_option.py @@ -1,3 +1,5 @@ +from __future__ import annotations + import pytest diff --git a/tests/test_directives/test_directive_other.py b/tests/test_directives/test_directive_other.py index 45a236bfd5a..3f09bb83789 100644 --- a/tests/test_directives/test_directive_other.py +++ b/tests/test_directives/test_directive_other.py @@ -1,5 +1,7 @@ """Test the other directives.""" +from __future__ import annotations + from pathlib import Path import pytest diff --git a/tests/test_directives/test_directive_patch.py b/tests/test_directives/test_directive_patch.py index a35c3f32e81..a141cc86682 100644 --- a/tests/test_directives/test_directive_patch.py +++ b/tests/test_directives/test_directive_patch.py @@ -1,5 +1,7 @@ """Test the patched directives.""" +from __future__ import annotations + import pytest from docutils import nodes diff --git a/tests/test_directives/test_directives_no_typesetting.py b/tests/test_directives/test_directives_no_typesetting.py index 8692cd387fa..68df4fbefcf 100644 --- a/tests/test_directives/test_directives_no_typesetting.py +++ b/tests/test_directives/test_directives_no_typesetting.py @@ -1,5 +1,7 @@ """Tests the directives""" +from __future__ import annotations + import pytest from docutils import nodes diff --git a/tests/test_domains/test_domain_c.py b/tests/test_domains/test_domain_c.py index d90bcd0b7b5..339938010f9 100644 --- a/tests/test_domains/test_domain_c.py +++ b/tests/test_domains/test_domain_c.py @@ -1,9 +1,11 @@ """Tests the C Domain""" +from __future__ import annotations + import itertools import xml.etree.ElementTree as ET import zlib -from io import StringIO +from typing import TYPE_CHECKING import pytest @@ -29,6 +31,9 @@ from sphinx.util.cfamily import DefinitionError from sphinx.writers.text import STDINDENT +if TYPE_CHECKING: + from io import StringIO + class Config: c_id_attributes = ['id_attr', 'LIGHTGBM_C_EXPORT'] diff --git a/tests/test_domains/test_domain_cpp.py b/tests/test_domains/test_domain_cpp.py index a177991a535..96208bf7409 100644 --- a/tests/test_domains/test_domain_cpp.py +++ b/tests/test_domains/test_domain_cpp.py @@ -1,9 +1,11 @@ """Tests the C++ Domain""" +from __future__ import annotations + import itertools import re import zlib -from io import StringIO +from typing import TYPE_CHECKING import pytest @@ -30,6 +32,9 @@ from sphinx.util.cfamily import DefinitionError, NoOldIdError from sphinx.writers.text import STDINDENT +if TYPE_CHECKING: + from io import StringIO + def parse(name, string): class Config: diff --git a/tests/test_domains/test_domain_js.py b/tests/test_domains/test_domain_js.py index a51dd13be58..ab3ed26c6cb 100644 --- a/tests/test_domains/test_domain_js.py +++ b/tests/test_domains/test_domain_js.py @@ -1,5 +1,7 @@ """Tests the JavaScript Domain""" +from __future__ import annotations + from unittest.mock import Mock import docutils.utils diff --git a/tests/test_domains/test_domain_rst.py b/tests/test_domains/test_domain_rst.py index dc61792dd0a..9f1eec12c28 100644 --- a/tests/test_domains/test_domain_rst.py +++ b/tests/test_domains/test_domain_rst.py @@ -1,5 +1,7 @@ """Tests the reStructuredText domain.""" +from __future__ import annotations + import pytest from sphinx import addnodes diff --git a/tests/test_domains/test_domain_std.py b/tests/test_domains/test_domain_std.py index 4e5d88b63de..4684596bd70 100644 --- a/tests/test_domains/test_domain_std.py +++ b/tests/test_domains/test_domain_std.py @@ -1,5 +1,7 @@ """Tests the std domain""" +from __future__ import annotations + from unittest import mock import pytest diff --git a/tests/test_environment/test_environment.py b/tests/test_environment/test_environment.py index 05427faf56f..c50dc21c484 100644 --- a/tests/test_environment/test_environment.py +++ b/tests/test_environment/test_environment.py @@ -1,5 +1,7 @@ """Test the BuildEnvironment class.""" +from __future__ import annotations + import shutil from pathlib import Path diff --git a/tests/test_environment/test_environment_indexentries.py b/tests/test_environment/test_environment_indexentries.py index 1de861abc49..e54e41e6f64 100644 --- a/tests/test_environment/test_environment_indexentries.py +++ b/tests/test_environment/test_environment_indexentries.py @@ -1,5 +1,7 @@ """Test the sphinx.environment.adapters.indexentries.""" +from __future__ import annotations + import pytest from sphinx.environment.adapters.indexentries import IndexEntries diff --git a/tests/test_environment/test_environment_record_dependencies.py b/tests/test_environment/test_environment_record_dependencies.py index 0a17253c091..f5d80ca2f74 100644 --- a/tests/test_environment/test_environment_record_dependencies.py +++ b/tests/test_environment/test_environment_record_dependencies.py @@ -1,5 +1,7 @@ """Tests for ``record_dependencies``.""" +from __future__ import annotations + import pytest diff --git a/tests/test_environment/test_environment_toctree.py b/tests/test_environment/test_environment_toctree.py index 7f283c42231..0020fb9a161 100644 --- a/tests/test_environment/test_environment_toctree.py +++ b/tests/test_environment/test_environment_toctree.py @@ -1,5 +1,7 @@ """Test the sphinx.environment.adapters.toctree.""" +from __future__ import annotations + import pytest from docutils import nodes from docutils.nodes import bullet_list, list_item, literal, reference, title diff --git a/tests/test_errors.py b/tests/test_errors.py index d551a6ef1e8..e1ef0d389ae 100644 --- a/tests/test_errors.py +++ b/tests/test_errors.py @@ -1,3 +1,5 @@ +from __future__ import annotations + from sphinx.errors import ExtensionError diff --git a/tests/test_events.py b/tests/test_events.py index df175787a17..349b3dde572 100644 --- a/tests/test_events.py +++ b/tests/test_events.py @@ -1,5 +1,7 @@ """Test the EventManager class.""" +from __future__ import annotations + import pytest from sphinx.errors import ExtensionError diff --git a/tests/test_extensions/ext_napoleon_pep526_data_google.py b/tests/test_extensions/ext_napoleon_pep526_data_google.py index d0692e0bab1..f42dfcfc442 100644 --- a/tests/test_extensions/ext_napoleon_pep526_data_google.py +++ b/tests/test_extensions/ext_napoleon_pep526_data_google.py @@ -1,5 +1,7 @@ """Test module for napoleon PEP 526 compatibility with google style""" +from __future__ import annotations + module_level_var: int = 99 """This is an example module level variable""" diff --git a/tests/test_extensions/ext_napoleon_pep526_data_numpy.py b/tests/test_extensions/ext_napoleon_pep526_data_numpy.py index eff7746ebf2..0681a334a1e 100644 --- a/tests/test_extensions/ext_napoleon_pep526_data_numpy.py +++ b/tests/test_extensions/ext_napoleon_pep526_data_numpy.py @@ -1,5 +1,7 @@ """Test module for napoleon PEP 526 compatibility with numpy style""" +from __future__ import annotations + module_level_var: int = 99 """This is an example module level variable""" diff --git a/tests/test_extensions/test_ext_apidoc.py b/tests/test_extensions/test_ext_apidoc.py index 515aff9e930..1e191de9a7b 100644 --- a/tests/test_extensions/test_ext_apidoc.py +++ b/tests/test_extensions/test_ext_apidoc.py @@ -1,13 +1,18 @@ """Test the sphinx.apidoc module.""" +from __future__ import annotations + from collections import namedtuple -from pathlib import Path +from typing import TYPE_CHECKING import pytest import sphinx.ext.apidoc from sphinx.ext.apidoc import main as apidoc_main +if TYPE_CHECKING: + from pathlib import Path + _apidoc = namedtuple('_apidoc', 'coderoot,outdir') # NoQA: PYI024 diff --git a/tests/test_extensions/test_ext_autodoc_autoattribute.py b/tests/test_extensions/test_ext_autodoc_autoattribute.py index 41fcc99011b..51358302a6e 100644 --- a/tests/test_extensions/test_ext_autodoc_autoattribute.py +++ b/tests/test_extensions/test_ext_autodoc_autoattribute.py @@ -4,6 +4,8 @@ source file translated by test_build. """ +from __future__ import annotations + import pytest from tests.test_extensions.autodoc_util import do_autodoc diff --git a/tests/test_extensions/test_ext_autodoc_autodata.py b/tests/test_extensions/test_ext_autodoc_autodata.py index b794666e97e..796e29e1d45 100644 --- a/tests/test_extensions/test_ext_autodoc_autodata.py +++ b/tests/test_extensions/test_ext_autodoc_autodata.py @@ -4,6 +4,8 @@ source file translated by test_build. """ +from __future__ import annotations + import pytest from tests.test_extensions.autodoc_util import do_autodoc diff --git a/tests/test_extensions/test_ext_autodoc_autofunction.py b/tests/test_extensions/test_ext_autodoc_autofunction.py index e70069bfd61..a3cb5d747d4 100644 --- a/tests/test_extensions/test_ext_autodoc_autofunction.py +++ b/tests/test_extensions/test_ext_autodoc_autofunction.py @@ -4,6 +4,8 @@ source file translated by test_build. """ +from __future__ import annotations + from typing import Any import pytest diff --git a/tests/test_extensions/test_ext_autodoc_automodule.py b/tests/test_extensions/test_ext_autodoc_automodule.py index 2f502f12336..c9503a765c3 100644 --- a/tests/test_extensions/test_ext_autodoc_automodule.py +++ b/tests/test_extensions/test_ext_autodoc_automodule.py @@ -4,6 +4,8 @@ source file translated by test_build. """ +from __future__ import annotations + import inspect import sys import typing diff --git a/tests/test_extensions/test_ext_autodoc_autoproperty.py b/tests/test_extensions/test_ext_autodoc_autoproperty.py index 9c9178d3b01..b6d2125e25f 100644 --- a/tests/test_extensions/test_ext_autodoc_autoproperty.py +++ b/tests/test_extensions/test_ext_autodoc_autoproperty.py @@ -4,6 +4,8 @@ source file translated by test_build. """ +from __future__ import annotations + import pytest from tests.test_extensions.autodoc_util import do_autodoc diff --git a/tests/test_extensions/test_ext_autodoc_configs.py b/tests/test_extensions/test_ext_autodoc_configs.py index c3911b8533c..155cbc36347 100644 --- a/tests/test_extensions/test_ext_autodoc_configs.py +++ b/tests/test_extensions/test_ext_autodoc_configs.py @@ -1,10 +1,11 @@ """Test the autodoc extension. This tests mainly for config variables""" +from __future__ import annotations + import platform import sys -from collections.abc import Iterator from contextlib import contextmanager -from pathlib import Path +from typing import TYPE_CHECKING import pytest @@ -12,6 +13,10 @@ from tests.test_extensions.autodoc_util import do_autodoc +if TYPE_CHECKING: + from collections.abc import Iterator + from pathlib import Path + skip_py314_segfault = pytest.mark.skipif( sys.version_info[:2] >= (3, 14), reason='Segmentation fault: https://github.com/python/cpython/issues/125017', diff --git a/tests/test_extensions/test_ext_autodoc_events.py b/tests/test_extensions/test_ext_autodoc_events.py index 8ff8630ad7e..d7139e88019 100644 --- a/tests/test_extensions/test_ext_autodoc_events.py +++ b/tests/test_extensions/test_ext_autodoc_events.py @@ -1,5 +1,7 @@ """Test the autodoc extension. This tests mainly for autodoc events""" +from __future__ import annotations + import pytest from sphinx.ext.autodoc import between, cut_lines diff --git a/tests/test_extensions/test_ext_autodoc_preserve_defaults.py b/tests/test_extensions/test_ext_autodoc_preserve_defaults.py index 1f8b5554c6e..b7b4fcf7027 100644 --- a/tests/test_extensions/test_ext_autodoc_preserve_defaults.py +++ b/tests/test_extensions/test_ext_autodoc_preserve_defaults.py @@ -1,5 +1,7 @@ """Test the autodoc extension.""" +from __future__ import annotations + import pytest from tests.test_extensions.autodoc_util import do_autodoc diff --git a/tests/test_extensions/test_ext_autodoc_private_members.py b/tests/test_extensions/test_ext_autodoc_private_members.py index 2915c74d7af..d5f8df9f03d 100644 --- a/tests/test_extensions/test_ext_autodoc_private_members.py +++ b/tests/test_extensions/test_ext_autodoc_private_members.py @@ -1,5 +1,7 @@ """Test the autodoc extension. This tests mainly for private-members option.""" +from __future__ import annotations + import pytest from tests.test_extensions.autodoc_util import do_autodoc diff --git a/tests/test_extensions/test_ext_autosectionlabel.py b/tests/test_extensions/test_ext_autosectionlabel.py index 2133f64bfaf..7c8dad56036 100644 --- a/tests/test_extensions/test_ext_autosectionlabel.py +++ b/tests/test_extensions/test_ext_autosectionlabel.py @@ -1,5 +1,7 @@ """Test sphinx.ext.autosectionlabel extension.""" +from __future__ import annotations + import re import pytest diff --git a/tests/test_extensions/test_ext_autosummary.py b/tests/test_extensions/test_ext_autosummary.py index 5a96afdd3e3..175abc4e21c 100644 --- a/tests/test_extensions/test_ext_autosummary.py +++ b/tests/test_extensions/test_ext_autosummary.py @@ -1,10 +1,12 @@ """Test the autosummary extension.""" +from __future__ import annotations + import sys from contextlib import chdir from io import StringIO +from typing import TYPE_CHECKING from unittest.mock import Mock, patch -from xml.etree.ElementTree import Element import pytest from docutils import nodes @@ -26,6 +28,9 @@ from sphinx.testing.util import assert_node, etree_parse from sphinx.util.docutils import new_document +if TYPE_CHECKING: + from xml.etree.ElementTree import Element + html_warnfile = StringIO() diff --git a/tests/test_extensions/test_ext_autosummary_imports.py b/tests/test_extensions/test_ext_autosummary_imports.py index 12a0d4624b0..7abee757e3c 100644 --- a/tests/test_extensions/test_ext_autosummary_imports.py +++ b/tests/test_extensions/test_ext_autosummary_imports.py @@ -1,5 +1,7 @@ """Test autosummary for import cycles.""" +from __future__ import annotations + import pytest from docutils import nodes diff --git a/tests/test_extensions/test_ext_coverage.py b/tests/test_extensions/test_ext_coverage.py index 563fd7eb61b..f788314af91 100644 --- a/tests/test_extensions/test_ext_coverage.py +++ b/tests/test_extensions/test_ext_coverage.py @@ -1,5 +1,7 @@ """Test the coverage builder.""" +from __future__ import annotations + import pickle import pytest diff --git a/tests/test_extensions/test_ext_doctest.py b/tests/test_extensions/test_ext_doctest.py index 0f141f276d2..fc86ee47f5d 100644 --- a/tests/test_extensions/test_ext_doctest.py +++ b/tests/test_extensions/test_ext_doctest.py @@ -1,5 +1,7 @@ """Test the doctest extension.""" +from __future__ import annotations + import os from collections import Counter diff --git a/tests/test_extensions/test_ext_duration.py b/tests/test_extensions/test_ext_duration.py index 535786de892..c8984de02e3 100644 --- a/tests/test_extensions/test_ext_duration.py +++ b/tests/test_extensions/test_ext_duration.py @@ -1,5 +1,7 @@ """Test sphinx.ext.duration extension.""" +from __future__ import annotations + import re import pytest diff --git a/tests/test_extensions/test_ext_extlinks.py b/tests/test_extensions/test_ext_extlinks.py index 38614329802..55b80e84e4d 100644 --- a/tests/test_extensions/test_ext_extlinks.py +++ b/tests/test_extensions/test_ext_extlinks.py @@ -1,3 +1,5 @@ +from __future__ import annotations + import pytest diff --git a/tests/test_extensions/test_ext_githubpages.py b/tests/test_extensions/test_ext_githubpages.py index 156b9a3916a..ddf54d96e6f 100644 --- a/tests/test_extensions/test_ext_githubpages.py +++ b/tests/test_extensions/test_ext_githubpages.py @@ -1,5 +1,7 @@ """Test sphinx.ext.githubpages extension.""" +from __future__ import annotations + import pytest diff --git a/tests/test_extensions/test_ext_graphviz.py b/tests/test_extensions/test_ext_graphviz.py index 4be01caf023..b4e0a167316 100644 --- a/tests/test_extensions/test_ext_graphviz.py +++ b/tests/test_extensions/test_ext_graphviz.py @@ -1,5 +1,7 @@ """Test sphinx.ext.graphviz extension.""" +from __future__ import annotations + import re import sys diff --git a/tests/test_extensions/test_ext_ifconfig.py b/tests/test_extensions/test_ext_ifconfig.py index 318c56a5741..7c07405b2f8 100644 --- a/tests/test_extensions/test_ext_ifconfig.py +++ b/tests/test_extensions/test_ext_ifconfig.py @@ -1,5 +1,7 @@ """Test sphinx.ext.ifconfig extension.""" +from __future__ import annotations + import docutils.utils import pytest diff --git a/tests/test_extensions/test_ext_imgconverter.py b/tests/test_extensions/test_ext_imgconverter.py index a37c98c172d..b44fd5782a2 100644 --- a/tests/test_extensions/test_ext_imgconverter.py +++ b/tests/test_extensions/test_ext_imgconverter.py @@ -1,5 +1,7 @@ """Test sphinx.ext.imgconverter extension.""" +from __future__ import annotations + import subprocess import pytest diff --git a/tests/test_extensions/test_ext_imgmockconverter.py b/tests/test_extensions/test_ext_imgmockconverter.py index fc34c24c65e..7d10a522e93 100644 --- a/tests/test_extensions/test_ext_imgmockconverter.py +++ b/tests/test_extensions/test_ext_imgmockconverter.py @@ -1,5 +1,7 @@ """Test image converter with identical basenames""" +from __future__ import annotations + import pytest diff --git a/tests/test_extensions/test_ext_inheritance_diagram.py b/tests/test_extensions/test_ext_inheritance_diagram.py index 2aa4ff99188..c43aa2e28ee 100644 --- a/tests/test_extensions/test_ext_inheritance_diagram.py +++ b/tests/test_extensions/test_ext_inheritance_diagram.py @@ -1,5 +1,7 @@ """Test sphinx.ext.inheritance_diagram extension.""" +from __future__ import annotations + import re import sys import zlib diff --git a/tests/test_extensions/test_ext_math.py b/tests/test_extensions/test_ext_math.py index 5a866520afb..e0445318840 100644 --- a/tests/test_extensions/test_ext_math.py +++ b/tests/test_extensions/test_ext_math.py @@ -1,5 +1,7 @@ """Test math extensions.""" +from __future__ import annotations + import re import shutil import subprocess diff --git a/tests/test_extensions/test_ext_napoleon.py b/tests/test_extensions/test_ext_napoleon.py index 306ab057b0d..1d495add7b0 100644 --- a/tests/test_extensions/test_ext_napoleon.py +++ b/tests/test_extensions/test_ext_napoleon.py @@ -1,5 +1,7 @@ """Tests for :mod:`sphinx.ext.napoleon.__init__` module.""" +from __future__ import annotations + import functools from collections import namedtuple from unittest import mock diff --git a/tests/test_extensions/test_ext_napoleon_docstring.py b/tests/test_extensions/test_ext_napoleon_docstring.py index 1e0cf3ae023..cf78e98f6ab 100644 --- a/tests/test_extensions/test_ext_napoleon_docstring.py +++ b/tests/test_extensions/test_ext_napoleon_docstring.py @@ -1,5 +1,7 @@ """Tests for :mod:`sphinx.ext.napoleon.docstring` module.""" +from __future__ import annotations + import re import zlib from collections import namedtuple diff --git a/tests/test_extensions/test_ext_todo.py b/tests/test_extensions/test_ext_todo.py index ed7f8d60963..626afe1120b 100644 --- a/tests/test_extensions/test_ext_todo.py +++ b/tests/test_extensions/test_ext_todo.py @@ -1,5 +1,7 @@ """Test sphinx.ext.todo extension.""" +from __future__ import annotations + import re import pytest diff --git a/tests/test_extensions/test_extension.py b/tests/test_extensions/test_extension.py index a84ca30b81f..7c44360ae26 100644 --- a/tests/test_extensions/test_extension.py +++ b/tests/test_extensions/test_extension.py @@ -1,5 +1,7 @@ """Test sphinx.extension module.""" +from __future__ import annotations + import pytest from sphinx.errors import VersionRequirementError diff --git a/tests/test_highlighting.py b/tests/test_highlighting.py index 430d569c0e6..c32c1887ebe 100644 --- a/tests/test_highlighting.py +++ b/tests/test_highlighting.py @@ -1,5 +1,7 @@ """Test the Pygments highlighting bridge.""" +from __future__ import annotations + from unittest import mock import pygments diff --git a/tests/test_intl/test_catalogs.py b/tests/test_intl/test_catalogs.py index 5e28df680ec..1a6f3f425df 100644 --- a/tests/test_intl/test_catalogs.py +++ b/tests/test_intl/test_catalogs.py @@ -1,5 +1,7 @@ """Test the base build process.""" +from __future__ import annotations + import shutil from pathlib import Path diff --git a/tests/test_intl/test_intl.py b/tests/test_intl/test_intl.py index 19808f6d538..ab104b55600 100644 --- a/tests/test_intl/test_intl.py +++ b/tests/test_intl/test_intl.py @@ -3,11 +3,13 @@ Runs the text builder in the test root. """ +from __future__ import annotations + import os import re import shutil import time -from io import StringIO +from typing import TYPE_CHECKING import pytest from babel.messages import mofile, pofile @@ -19,6 +21,9 @@ from sphinx.util.console import strip_colors from sphinx.util.nodes import NodeMatcher +if TYPE_CHECKING: + from io import StringIO + _CATALOG_LOCALE = 'xx' sphinx_intl = pytest.mark.sphinx( diff --git a/tests/test_markup/test_markup.py b/tests/test_markup/test_markup.py index 9eb0b83fa89..fa99fa352dd 100644 --- a/tests/test_markup/test_markup.py +++ b/tests/test_markup/test_markup.py @@ -1,5 +1,7 @@ """Test various Sphinx-specific markup extensions.""" +from __future__ import annotations + import re import warnings from types import SimpleNamespace diff --git a/tests/test_markup/test_metadata.py b/tests/test_markup/test_metadata.py index b3fe3e678ee..c84dc43081e 100644 --- a/tests/test_markup/test_metadata.py +++ b/tests/test_markup/test_metadata.py @@ -2,6 +2,7 @@ # adapted from an example of bibliographic metadata at # https://docutils.sourceforge.io/docs/user/rst/demo.txt +from __future__ import annotations import pytest diff --git a/tests/test_markup/test_parser.py b/tests/test_markup/test_parser.py index a047923c720..3f40a21693a 100644 --- a/tests/test_markup/test_parser.py +++ b/tests/test_markup/test_parser.py @@ -1,5 +1,7 @@ """Tests parsers module.""" +from __future__ import annotations + from unittest.mock import Mock, patch import pytest diff --git a/tests/test_markup/test_smartquotes.py b/tests/test_markup/test_smartquotes.py index b57a8a81017..2f34f9f1a21 100644 --- a/tests/test_markup/test_smartquotes.py +++ b/tests/test_markup/test_smartquotes.py @@ -1,5 +1,7 @@ """Test smart quotes.""" +from __future__ import annotations + import pytest from sphinx.testing.util import etree_parse diff --git a/tests/test_project.py b/tests/test_project.py index aa8b04b1f06..0a7688b2cfe 100644 --- a/tests/test_project.py +++ b/tests/test_project.py @@ -1,5 +1,7 @@ """Tests project module.""" +from __future__ import annotations + from pathlib import Path import pytest diff --git a/tests/test_pycode/test_pycode.py b/tests/test_pycode/test_pycode.py index 3a34d6f3a0f..cc08ff62c11 100644 --- a/tests/test_pycode/test_pycode.py +++ b/tests/test_pycode/test_pycode.py @@ -1,5 +1,7 @@ """Test pycode.""" +from __future__ import annotations + import sys from pathlib import Path diff --git a/tests/test_pycode/test_pycode_ast.py b/tests/test_pycode/test_pycode_ast.py index c62600a82dc..bc734e737dc 100644 --- a/tests/test_pycode/test_pycode_ast.py +++ b/tests/test_pycode/test_pycode_ast.py @@ -1,5 +1,7 @@ """Test pycode.ast""" +from __future__ import annotations + import ast import pytest diff --git a/tests/test_pycode/test_pycode_parser.py b/tests/test_pycode/test_pycode_parser.py index 7883a8de3cf..813aa3a413e 100644 --- a/tests/test_pycode/test_pycode_parser.py +++ b/tests/test_pycode/test_pycode_parser.py @@ -1,5 +1,7 @@ """Test pycode.parser.""" +from __future__ import annotations + from sphinx.pycode.parser import Parser from sphinx.util.inspect import signature_from_str diff --git a/tests/test_quickstart.py b/tests/test_quickstart.py index 07a61689575..d0db756db21 100644 --- a/tests/test_quickstart.py +++ b/tests/test_quickstart.py @@ -1,10 +1,10 @@ """Test the sphinx.quickstart module.""" +from __future__ import annotations + import time -from collections.abc import Callable from io import StringIO -from pathlib import Path -from typing import Any +from typing import TYPE_CHECKING, Any import pytest @@ -12,6 +12,10 @@ from sphinx.testing.util import SphinxTestApp from sphinx.util.console import coloron, nocolor +if TYPE_CHECKING: + from collections.abc import Callable + from pathlib import Path + warnfile = StringIO() diff --git a/tests/test_roles.py b/tests/test_roles.py index 127f671a145..11b0a13cf4c 100644 --- a/tests/test_roles.py +++ b/tests/test_roles.py @@ -1,5 +1,7 @@ """Test sphinx.roles""" +from __future__ import annotations + from unittest.mock import Mock import pytest diff --git a/tests/test_theming/test_html_theme.py b/tests/test_theming/test_html_theme.py index bdfca9886d8..7f2d34a93e6 100644 --- a/tests/test_theming/test_html_theme.py +++ b/tests/test_theming/test_html_theme.py @@ -1,3 +1,5 @@ +from __future__ import annotations + import pytest diff --git a/tests/test_theming/test_templating.py b/tests/test_theming/test_templating.py index 6eb2fdffa78..c508716b7c7 100644 --- a/tests/test_theming/test_templating.py +++ b/tests/test_theming/test_templating.py @@ -1,5 +1,7 @@ """Test templating.""" +from __future__ import annotations + import pytest from sphinx.ext.autosummary.generate import setup_documenters diff --git a/tests/test_theming/test_theming.py b/tests/test_theming/test_theming.py index afd15f838b4..bf3025473d0 100644 --- a/tests/test_theming/test_theming.py +++ b/tests/test_theming/test_theming.py @@ -1,5 +1,7 @@ """Test the Theme class.""" +from __future__ import annotations + import shutil from pathlib import Path from xml.etree.ElementTree import ParseError diff --git a/tests/test_transforms/test_transforms_move_module_targets.py b/tests/test_transforms/test_transforms_move_module_targets.py index df72d9c33a5..f64b7d6a500 100644 --- a/tests/test_transforms/test_transforms_move_module_targets.py +++ b/tests/test_transforms/test_transforms_move_module_targets.py @@ -1,3 +1,5 @@ +from __future__ import annotations + import pytest from docutils import nodes diff --git a/tests/test_transforms/test_transforms_post_transforms_code.py b/tests/test_transforms/test_transforms_post_transforms_code.py index 57e9bb054cc..a4243fb85bc 100644 --- a/tests/test_transforms/test_transforms_post_transforms_code.py +++ b/tests/test_transforms/test_transforms_post_transforms_code.py @@ -1,3 +1,5 @@ +from __future__ import annotations + import pytest diff --git a/tests/test_transforms/test_transforms_post_transforms_images.py b/tests/test_transforms/test_transforms_post_transforms_images.py index bb5d076f071..800fb3b986b 100644 --- a/tests/test_transforms/test_transforms_post_transforms_images.py +++ b/tests/test_transforms/test_transforms_post_transforms_images.py @@ -1,3 +1,5 @@ +from __future__ import annotations + from types import SimpleNamespace from docutils import nodes diff --git a/tests/test_transforms/test_transforms_reorder_nodes.py b/tests/test_transforms/test_transforms_reorder_nodes.py index 22fdcf742cf..6afa6ab5b9c 100644 --- a/tests/test_transforms/test_transforms_reorder_nodes.py +++ b/tests/test_transforms/test_transforms_reorder_nodes.py @@ -1,5 +1,7 @@ """Tests the transformations""" +from __future__ import annotations + import pytest from docutils import nodes diff --git a/tests/test_transforms/test_unreferenced_footnotes.py b/tests/test_transforms/test_unreferenced_footnotes.py index b1cf08f783a..ac1b8b15dbd 100644 --- a/tests/test_transforms/test_unreferenced_footnotes.py +++ b/tests/test_transforms/test_unreferenced_footnotes.py @@ -1,10 +1,16 @@ """Test the ``UnreferencedFootnotesDetector`` transform.""" -from pathlib import Path +from __future__ import annotations + +from typing import TYPE_CHECKING -from sphinx.testing.util import SphinxTestApp from sphinx.util.console import strip_colors +if TYPE_CHECKING: + from pathlib import Path + + from sphinx.testing.util import SphinxTestApp + def test_warnings(make_app: type[SphinxTestApp], tmp_path: Path) -> None: """Test that warnings are emitted for unreferenced footnotes.""" diff --git a/tests/test_util/test_util.py b/tests/test_util/test_util.py index b04c30e7aa8..2584206eb1b 100644 --- a/tests/test_util/test_util.py +++ b/tests/test_util/test_util.py @@ -1,5 +1,7 @@ """Tests util functions.""" +from __future__ import annotations + from sphinx.util.osutil import ensuredir diff --git a/tests/test_util/test_util_display.py b/tests/test_util/test_util_display.py index afad5a34d90..7bd70f89cd3 100644 --- a/tests/test_util/test_util_display.py +++ b/tests/test_util/test_util_display.py @@ -1,5 +1,7 @@ """Tests util functions.""" +from __future__ import annotations + import pytest from sphinx.util import logging diff --git a/tests/test_util/test_util_docstrings.py b/tests/test_util/test_util_docstrings.py index 6d416a0754e..9fde0cb14eb 100644 --- a/tests/test_util/test_util_docstrings.py +++ b/tests/test_util/test_util_docstrings.py @@ -1,5 +1,7 @@ """Test sphinx.util.docstrings.""" +from __future__ import annotations + from sphinx.util.docstrings import ( prepare_commentdoc, prepare_docstring, diff --git a/tests/test_util/test_util_fileutil.py b/tests/test_util/test_util_fileutil.py index 7ffa83fc223..ebe252f6264 100644 --- a/tests/test_util/test_util_fileutil.py +++ b/tests/test_util/test_util_fileutil.py @@ -1,5 +1,7 @@ """Tests sphinx.util.fileutil functions.""" +from __future__ import annotations + import re from unittest import mock diff --git a/tests/test_util/test_util_i18n.py b/tests/test_util/test_util_i18n.py index 95d0909f90a..b6e865c331d 100644 --- a/tests/test_util/test_util_i18n.py +++ b/tests/test_util/test_util_i18n.py @@ -1,5 +1,7 @@ """Test i18n util.""" +from __future__ import annotations + import datetime import os import time diff --git a/tests/test_util/test_util_images.py b/tests/test_util/test_util_images.py index afc8587fa30..006d28003ce 100644 --- a/tests/test_util/test_util_images.py +++ b/tests/test_util/test_util_images.py @@ -1,5 +1,7 @@ """Test images util.""" +from __future__ import annotations + import pytest from sphinx.util.images import ( diff --git a/tests/test_util/test_util_importer.py b/tests/test_util/test_util_importer.py index 57fd6e46fb6..af0503c6899 100644 --- a/tests/test_util/test_util_importer.py +++ b/tests/test_util/test_util_importer.py @@ -1,5 +1,7 @@ """Test sphinx.util._importer.""" +from __future__ import annotations + import pytest from sphinx.errors import ExtensionError diff --git a/tests/test_util/test_util_lines.py b/tests/test_util/test_util_lines.py index a4ab41641de..5c9db412d88 100644 --- a/tests/test_util/test_util_lines.py +++ b/tests/test_util/test_util_lines.py @@ -1,3 +1,5 @@ +from __future__ import annotations + import pytest from sphinx.util._lines import parse_line_num_spec diff --git a/tests/test_util/test_util_logging.py b/tests/test_util/test_util_logging.py index 5a9e0be5ada..aa0fc86be59 100644 --- a/tests/test_util/test_util_logging.py +++ b/tests/test_util/test_util_logging.py @@ -1,5 +1,7 @@ """Test logging util.""" +from __future__ import annotations + import codecs import os from pathlib import Path diff --git a/tests/test_util/test_util_matching.py b/tests/test_util/test_util_matching.py index 01fb9fde2e1..9a51123ceff 100644 --- a/tests/test_util/test_util_matching.py +++ b/tests/test_util/test_util_matching.py @@ -1,5 +1,7 @@ """Tests sphinx.util.matching functions.""" +from __future__ import annotations + from sphinx.util.matching import Matcher, compile_matchers, get_matching_files diff --git a/tests/test_util/test_util_rst.py b/tests/test_util/test_util_rst.py index d6e83433af3..4d8d0e09482 100644 --- a/tests/test_util/test_util_rst.py +++ b/tests/test_util/test_util_rst.py @@ -1,5 +1,7 @@ """Tests sphinx.util.rst functions.""" +from __future__ import annotations + from docutils.statemachine import StringList from jinja2 import Environment diff --git a/tests/test_util/test_util_template.py b/tests/test_util/test_util_template.py index 51bb225b2c4..19e8b961ccf 100644 --- a/tests/test_util/test_util_template.py +++ b/tests/test_util/test_util_template.py @@ -1,5 +1,7 @@ """Tests sphinx.util.template functions.""" +from __future__ import annotations + from sphinx.util.template import ReSTRenderer diff --git a/tests/test_util/test_util_typing.py b/tests/test_util/test_util_typing.py index 314639efbe7..aabf84a61be 100644 --- a/tests/test_util/test_util_typing.py +++ b/tests/test_util/test_util_typing.py @@ -1,5 +1,7 @@ """Tests util.typing functions.""" +from __future__ import annotations + import dataclasses import sys import typing as t diff --git a/tests/test_util/test_util_uri.py b/tests/test_util/test_util_uri.py index 43ee1b34e48..fdcb3c2ca8e 100644 --- a/tests/test_util/test_util_uri.py +++ b/tests/test_util/test_util_uri.py @@ -1,3 +1,5 @@ +from __future__ import annotations + from sphinx.util._uri import encode_uri diff --git a/tests/test_versioning.py b/tests/test_versioning.py index 14e7520db49..795868ba84f 100644 --- a/tests/test_versioning.py +++ b/tests/test_versioning.py @@ -1,5 +1,7 @@ """Test the versioning implementation.""" +from __future__ import annotations + import pickle import shutil diff --git a/tests/test_writers/test_api_translator.py b/tests/test_writers/test_api_translator.py index bdbea0dc578..cd34aba058a 100644 --- a/tests/test_writers/test_api_translator.py +++ b/tests/test_writers/test_api_translator.py @@ -1,5 +1,7 @@ """Test the Sphinx API for translator.""" +from __future__ import annotations + import sys import pytest diff --git a/tests/test_writers/test_docutilsconf.py b/tests/test_writers/test_docutilsconf.py index cd74b5bef95..4201c9df831 100644 --- a/tests/test_writers/test_docutilsconf.py +++ b/tests/test_writers/test_docutilsconf.py @@ -1,5 +1,7 @@ """Test docutils.conf support for several writers.""" +from __future__ import annotations + import pytest from docutils import nodes diff --git a/tests/test_writers/test_writer_latex.py b/tests/test_writers/test_writer_latex.py index a0ab3ee5915..55130c44d52 100644 --- a/tests/test_writers/test_writer_latex.py +++ b/tests/test_writers/test_writer_latex.py @@ -1,5 +1,7 @@ """Test the LaTeX writer""" +from __future__ import annotations + import pytest from sphinx.writers.latex import rstdim_to_latexdim diff --git a/utils/babel_runner.py b/utils/babel_runner.py index 8c27f20e3e8..c3a2d030a35 100644 --- a/utils/babel_runner.py +++ b/utils/babel_runner.py @@ -13,6 +13,8 @@ Compile the ".po" catalogue files to ".mo" and ".js" files. """ +from __future__ import annotations + import json import logging import sys diff --git a/utils/bump_docker.py b/utils/bump_docker.py index 570e5ecab49..c36d8a88369 100755 --- a/utils/bump_docker.py +++ b/utils/bump_docker.py @@ -2,6 +2,8 @@ """Usage: bump_docker.py [VERSION]""" +from __future__ import annotations + import re import subprocess import sys diff --git a/utils/convert_attestations.py b/utils/convert_attestations.py index d359c1593b3..b8bd19a4a02 100644 --- a/utils/convert_attestations.py +++ b/utils/convert_attestations.py @@ -3,6 +3,8 @@ See https://github.com/trailofbits/pypi-attestations. """ +from __future__ import annotations + import json import sys from base64 import b64decode diff --git a/utils/generate_js_fixtures.py b/utils/generate_js_fixtures.py index 4e126899026..ecf13a94741 100755 --- a/utils/generate_js_fixtures.py +++ b/utils/generate_js_fixtures.py @@ -1,4 +1,5 @@ #!/usr/bin/env python3 +from __future__ import annotations import shutil import subprocess From b6b01c2755e0ea405275f6695098b9cf305ed5af Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jean-Fran=C3=A7ois=20B=2E?= <2589111+jfbu@users.noreply.github.com> Date: Sat, 23 Nov 2024 15:20:08 +0100 Subject: [PATCH 13/29] LaTeX: fix a `7.4.0` typo in a default for `\sphinxboxsetup` (#13152) --- CHANGES.rst | 3 +++ sphinx/texinputs/sphinx.sty | 7 ++++--- 2 files changed, 7 insertions(+), 3 deletions(-) diff --git a/CHANGES.rst b/CHANGES.rst index ca123d89f43..86e59279491 100644 --- a/CHANGES.rst +++ b/CHANGES.rst @@ -23,6 +23,9 @@ Bugs fixed * #13060: HTML Search: use ``Map`` to store per-file term scores. Patch by James Addison +* LaTeX: fix a ``7.4.0`` typo in a default for ``\sphinxboxsetup`` + (refs: PR #13152). + Patch by Jean-François B. Testing ------- diff --git a/sphinx/texinputs/sphinx.sty b/sphinx/texinputs/sphinx.sty index 0d52676c4fe..8837485c5f7 100644 --- a/sphinx/texinputs/sphinx.sty +++ b/sphinx/texinputs/sphinx.sty @@ -9,7 +9,7 @@ % by the Sphinx LaTeX writer. \NeedsTeXFormat{LaTeX2e}[1995/12/01] -\ProvidesPackage{sphinx}[2024/10/11 v8.1.1 Sphinx LaTeX package (sphinx-doc)] +\ProvidesPackage{sphinx}[2024/11/23 v8.2.0 Sphinx LaTeX package (sphinx-doc)] % provides \ltx@ifundefined % (many packages load ltxcmds: graphicx does for pdftex and lualatex but @@ -1098,12 +1098,13 @@ % Some of these defaults got already set. But we now list them all explicitly % for a complete initial configuration of reset storage. % At 7.4.0, \fboxrule and \fboxsep replaced by 0.4pt and 3pt which are anyhow -% the defaults for these LaTeX dimensions. +% the defaults for these LaTeX dimensions. 8.2.0 corrected border-radius +% default back to 3pt (\fboxsep) not 0.4pt (\fboxrule). \let\spx@boxes@sphinxbox@defaults\@gobble \sphinxboxsetup{% border-width=0.4pt, padding=3pt, - border-radius=0.4pt, + border-radius=3pt, box-shadow=none, % MEMO: as xcolor is loaded, \spx@defineorletcolor has a "\colorlet" branch % which makes this syntax acceptable and avoids duplicating here the values. From d178a5369d74f2a79bca17e545e4414151e5fef1 Mon Sep 17 00:00:00 2001 From: Adam Turner <9087854+aa-turner@users.noreply.github.com> Date: Sun, 24 Nov 2024 22:26:40 +0000 Subject: [PATCH 14/29] Use the right types in {Dummy,Fake}Application --- sphinx/ext/autosummary/__init__.py | 5 +++-- sphinx/ext/autosummary/generate.py | 3 ++- 2 files changed, 5 insertions(+), 3 deletions(-) diff --git a/sphinx/ext/autosummary/__init__.py b/sphinx/ext/autosummary/__init__.py index e77dbe91c79..ddb93a30789 100644 --- a/sphinx/ext/autosummary/__init__.py +++ b/sphinx/ext/autosummary/__init__.py @@ -57,6 +57,7 @@ import re import sys from inspect import Parameter +from pathlib import Path from types import ModuleType from typing import TYPE_CHECKING, Any, ClassVar, cast @@ -154,10 +155,10 @@ class FakeApplication: verbosity = 0 def __init__(self) -> None: - self.doctreedir = None + self.doctreedir = Path() self.events = None self.extensions: dict[str, Extension] = {} - self.srcdir = None + self.srcdir = Path() self.config = Config() self.project = Project('', {}) self.registry = SphinxComponentRegistry() diff --git a/sphinx/ext/autosummary/generate.py b/sphinx/ext/autosummary/generate.py index 1d90438cfec..f2ce63ed406 100644 --- a/sphinx/ext/autosummary/generate.py +++ b/sphinx/ext/autosummary/generate.py @@ -46,6 +46,7 @@ from sphinx.pycode import ModuleAnalyzer from sphinx.registry import SphinxComponentRegistry from sphinx.util import logging, rst +from sphinx.util._pathlib import _StrPath from sphinx.util.inspect import getall, safe_getattr from sphinx.util.osutil import ensuredir from sphinx.util.template import SphinxTemplateLoader @@ -67,7 +68,7 @@ def __init__(self, translator: NullTranslations) -> None: self.config = Config() self.registry = SphinxComponentRegistry() self.messagelog: list[str] = [] - self.srcdir = '/' + self.srcdir = _StrPath('/') self.translator = translator self.verbosity = 0 self._warncount = 0 From 768c6b70dfb2ab8ad049a4f28c1a656c731ecb3f Mon Sep 17 00:00:00 2001 From: Adam Turner <9087854+aa-turner@users.noreply.github.com> Date: Tue, 26 Nov 2024 23:39:01 +0000 Subject: [PATCH 15/29] Create a _StrPath proxy type --- sphinx/util/_pathlib.py | 37 ++++++++++++++++++++++++++++++++++++- 1 file changed, 36 insertions(+), 1 deletion(-) diff --git a/sphinx/util/_pathlib.py b/sphinx/util/_pathlib.py index 31b47ce5a67..1f3209cef48 100644 --- a/sphinx/util/_pathlib.py +++ b/sphinx/util/_pathlib.py @@ -17,7 +17,7 @@ import sys import warnings from pathlib import Path, PosixPath, PurePath, WindowsPath -from typing import Any +from typing import Any, overload from sphinx.deprecation import RemovedInSphinx90Warning @@ -133,3 +133,38 @@ def __getitem__(self, item: int | slice) -> str: def __len__(self) -> int: warnings.warn(_MSG, RemovedInSphinx90Warning, stacklevel=2) return len(self.__str__()) + + +class _StrPathProperty: + def __init__(self) -> None: + self.instance_attr: str = '' + + def __set_name__(self, owner: object, name: str) -> None: + self.instance_attr = f'_{name}' # i.e. '_srcdir' + + @overload + def __get__(self, obj: None, objtype: None) -> _StrPathProperty: ... # NoQA: E704 + + @overload + def __get__(self, obj: object, objtype: type[object]) -> _StrPath: ... # NoQA: E704 + + def __get__( + self, obj: object | None, objtype: type[object] | None = None + ) -> _StrPathProperty | _StrPath: + if obj is None: + return self + if not self.instance_attr: + raise AttributeError + return getattr(obj, self.instance_attr) + + def __set__(self, obj: Any, value: _StrPath | Path) -> None: + try: + setattr(obj, self.instance_attr, _StrPath(value)) + except TypeError as err: + cls_name = type(obj).__qualname__ + name = self.instance_attr.removeprefix('_') + msg = f'{cls_name}.{name} may only be set to path-like objects' + raise TypeError(msg) from err + + def __delete__(self, obj: Any) -> None: + delattr(obj, self.instance_attr) From 3930a1ce10f4dbb24bfad95becbc0515bf87acae Mon Sep 17 00:00:00 2001 From: Adam Turner <9087854+aa-turner@users.noreply.github.com> Date: Wed, 27 Nov 2024 00:24:16 +0000 Subject: [PATCH 16/29] Use the _StrPath proxy for sphinx.application --- sphinx/application.py | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) diff --git a/sphinx/application.py b/sphinx/application.py index 5edc86afb71..0c10575abef 100644 --- a/sphinx/application.py +++ b/sphinx/application.py @@ -26,7 +26,7 @@ from sphinx.project import Project from sphinx.registry import SphinxComponentRegistry from sphinx.util import docutils, logging -from sphinx.util._pathlib import _StrPath +from sphinx.util._pathlib import _StrPath, _StrPathProperty from sphinx.util.build_phase import BuildPhase from sphinx.util.console import bold from sphinx.util.display import progress_message @@ -140,6 +140,11 @@ class Sphinx: warningiserror: Final = False _warncount: int + srcdir = _StrPathProperty() + confdir = _StrPathProperty() + outdir = _StrPathProperty() + doctreedir = _StrPathProperty() + def __init__(self, srcdir: str | os.PathLike[str], confdir: str | os.PathLike[str] | None, outdir: str | os.PathLike[str], doctreedir: str | os.PathLike[str], buildername: str, confoverrides: dict | None = None, From 40dc8b89610796c4b4e1994241e2a3738fd8b13b Mon Sep 17 00:00:00 2001 From: Adam Turner <9087854+aa-turner@users.noreply.github.com> Date: Wed, 27 Nov 2024 00:27:11 +0000 Subject: [PATCH 17/29] Use the _StrPath proxy for sphinx.builders --- sphinx/builders/__init__.py | 15 ++++++++++----- 1 file changed, 10 insertions(+), 5 deletions(-) diff --git a/sphinx/builders/__init__.py b/sphinx/builders/__init__.py index 620ff82abe5..7c9b9cb172c 100644 --- a/sphinx/builders/__init__.py +++ b/sphinx/builders/__init__.py @@ -23,6 +23,7 @@ rst, ) from sphinx.util._importer import import_object +from sphinx.util._pathlib import _StrPathProperty from sphinx.util.build_phase import BuildPhase from sphinx.util.console import bold from sphinx.util.display import progress_message, status_iterator @@ -48,7 +49,6 @@ from sphinx.application import Sphinx from sphinx.config import Config from sphinx.events import EventManager - from sphinx.util._pathlib import _StrPath from sphinx.util.tags import Tags @@ -94,11 +94,16 @@ class Builder: #: The file format produced by the builder allows images to be embedded using data-URIs. supported_data_uri_images: bool = False + srcdir = _StrPathProperty() + confdir = _StrPathProperty() + outdir = _StrPathProperty() + doctreedir = _StrPathProperty() + def __init__(self, app: Sphinx, env: BuildEnvironment) -> None: - self.srcdir: _StrPath = app.srcdir - self.confdir: _StrPath = app.confdir - self.outdir: _StrPath = app.outdir - self.doctreedir: _StrPath = app.doctreedir + self.srcdir = app.srcdir + self.confdir = app.confdir + self.outdir = app.outdir + self.doctreedir = app.doctreedir ensuredir(self.doctreedir) self.app: Sphinx = app From 7fbd5533da08bf439d8c88f0972faec807326234 Mon Sep 17 00:00:00 2001 From: Adam Turner <9087854+aa-turner@users.noreply.github.com> Date: Wed, 27 Nov 2024 00:26:59 +0000 Subject: [PATCH 18/29] Use the _StrPath proxy for sphinx.environment --- sphinx/environment/__init__.py | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/sphinx/environment/__init__.py b/sphinx/environment/__init__.py index b17c1ee453b..e73ecd7a511 100644 --- a/sphinx/environment/__init__.py +++ b/sphinx/environment/__init__.py @@ -22,6 +22,7 @@ from sphinx.transforms import SphinxTransformer from sphinx.util import logging from sphinx.util._files import DownloadFiles, FilenameUniqDict +from sphinx.util._pathlib import _StrPathProperty from sphinx.util._serialise import stable_str from sphinx.util._timestamps import _format_rfc3339_microseconds from sphinx.util.docutils import LoggingReporter @@ -98,10 +99,13 @@ class BuildEnvironment: # --------- ENVIRONMENT INITIALIZATION ------------------------------------- + srcdir = _StrPathProperty() + doctreedir = _StrPathProperty() + def __init__(self, app: Sphinx) -> None: self.app: Sphinx = app - self.doctreedir: _StrPath = app.doctreedir - self.srcdir: _StrPath = app.srcdir + self.doctreedir = app.doctreedir + self.srcdir = app.srcdir self.config: Config = None # type: ignore[assignment] self.config_status: int = CONFIG_UNSET self.config_status_extra: str = '' From d9a28215095b4a8d4669bbde548115e85de10d8d Mon Sep 17 00:00:00 2001 From: Adam Turner <9087854+aa-turner@users.noreply.github.com> Date: Thu, 28 Nov 2024 14:54:49 +0000 Subject: [PATCH 19/29] Bump types-docutils to 0.21.0.20241128 --- pyproject.toml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pyproject.toml b/pyproject.toml index f479619fee4..f22e1824f17 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -87,7 +87,7 @@ lint = [ "sphinx-lint>=0.9", "types-colorama==0.4.15.20240311", "types-defusedxml==0.7.0.20240218", - "types-docutils==0.21.0.20241005", + "types-docutils==0.21.0.20241128", "types-Pillow==10.2.0.20240822", "types-Pygments==2.18.0.20240506", "types-requests==2.32.0.20241016", # align with requests From b458850b329f3112b6e8df7a19ff21fa478fb2d3 Mon Sep 17 00:00:00 2001 From: Adam Turner <9087854+aa-turner@users.noreply.github.com> Date: Fri, 29 Nov 2024 23:09:10 +0000 Subject: [PATCH 20/29] Bump Ruff to 0.8.1 --- pyproject.toml | 2 +- sphinx/builders/latex/transforms.py | 2 +- sphinx/builders/linkcheck.py | 2 +- sphinx/directives/__init__.py | 4 ++-- sphinx/directives/other.py | 2 +- sphinx/directives/patches.py | 6 +++--- sphinx/domains/c/_ast.py | 6 +++--- sphinx/domains/citation.py | 2 +- sphinx/domains/python/__init__.py | 4 ++-- sphinx/domains/std/__init__.py | 11 +++++----- sphinx/environment/collectors/metadata.py | 6 +++--- sphinx/environment/collectors/toctree.py | 2 +- sphinx/ext/autodoc/type_comment.py | 4 ++-- sphinx/ext/autodoc/typehints.py | 9 ++++---- sphinx/ext/autosectionlabel.py | 2 +- sphinx/ext/autosummary/__init__.py | 16 +++++++------- sphinx/ext/inheritance_diagram.py | 5 +++-- sphinx/ext/intersphinx/_resolve.py | 2 +- sphinx/ext/mathjax.py | 4 ++-- sphinx/ext/todo.py | 2 +- sphinx/ext/viewcode.py | 4 ++-- sphinx/transforms/__init__.py | 4 ++-- sphinx/transforms/compact_bullet_list.py | 4 ++-- sphinx/transforms/post_transforms/__init__.py | 7 ++++--- sphinx/util/docfields.py | 18 ++++++++-------- sphinx/util/docutils.py | 2 +- sphinx/util/inspect.py | 2 +- sphinx/util/nodes.py | 2 +- sphinx/util/rst.py | 2 +- sphinx/writers/html.py | 2 +- sphinx/writers/html5.py | 5 +++-- sphinx/writers/latex.py | 19 +++++++++-------- sphinx/writers/manpage.py | 15 ++++++------- sphinx/writers/texinfo.py | 21 ++++++++++--------- sphinx/writers/text.py | 13 ++++++------ 35 files changed, 111 insertions(+), 102 deletions(-) diff --git a/pyproject.toml b/pyproject.toml index f22e1824f17..8186e321362 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -82,7 +82,7 @@ docs = [ ] lint = [ "flake8>=6.0", - "ruff==0.8.0", + "ruff==0.8.1", "mypy==1.13.0", "sphinx-lint>=0.9", "types-colorama==0.4.15.20240311", diff --git a/sphinx/builders/latex/transforms.py b/sphinx/builders/latex/transforms.py index 57864208e17..b6d0439d18a 100644 --- a/sphinx/builders/latex/transforms.py +++ b/sphinx/builders/latex/transforms.py @@ -151,7 +151,7 @@ def renumber_footnotes(self) -> None: break # assign new footnote number - old_label = cast(nodes.label, footnote[0]) + old_label = cast('nodes.label', footnote[0]) old_label.replace_self(nodes.label('', str(num))) if old_label in footnote['names']: footnote['names'].remove(old_label.astext()) diff --git a/sphinx/builders/linkcheck.py b/sphinx/builders/linkcheck.py index 82fc0acc4bc..48c589acc07 100644 --- a/sphinx/builders/linkcheck.py +++ b/sphinx/builders/linkcheck.py @@ -251,7 +251,7 @@ def _add_uri(self, uri: str, node: nodes.Element) -> None: :param uri: URI to add :param node: A node class where the URI was found """ - builder = cast(CheckExternalLinksBuilder, self.app.builder) + builder = cast('CheckExternalLinksBuilder', self.app.builder) hyperlinks = builder.hyperlinks docname = self.env.docname diff --git a/sphinx/directives/__init__.py b/sphinx/directives/__init__.py index d92678ab8f2..de0dc45beec 100644 --- a/sphinx/directives/__init__.py +++ b/sphinx/directives/__init__.py @@ -81,7 +81,7 @@ def get_field_type_map(self) -> dict[str, tuple[Field, bool]]: self._doc_field_type_map[name] = (field, False) if field.is_typed: - typed_field = cast(TypedField, field) + typed_field = cast('TypedField', field) for name in typed_field.typenames: self._doc_field_type_map[name] = (field, True) @@ -346,7 +346,7 @@ def run(self) -> list[Node]: ) messages += [error] - return cast(list[nodes.Node], messages) + return cast('list[nodes.Node]', messages) class DefaultDomain(SphinxDirective): diff --git a/sphinx/directives/other.py b/sphinx/directives/other.py index 32fd3111a65..10fe8ff69f9 100644 --- a/sphinx/directives/other.py +++ b/sphinx/directives/other.py @@ -376,7 +376,7 @@ def run(self) -> list[Node]: # Use these depths to determine where the nested sections should # be placed in the doctree. n_sects_to_raise = current_depth - nested_depth + 1 - parent = cast(nodes.Element, self.state.parent) + parent = cast('nodes.Element', self.state.parent) for _i in range(n_sects_to_raise): if parent.parent: parent = parent.parent diff --git a/sphinx/directives/patches.py b/sphinx/directives/patches.py index ff2989520d8..37144abacdd 100644 --- a/sphinx/directives/patches.py +++ b/sphinx/directives/patches.py @@ -38,7 +38,7 @@ def run(self) -> list[Node]: return result assert len(result) == 1 - figure_node = cast(nodes.figure, result[0]) + figure_node = cast('nodes.figure', result[0]) if name: # set ``name`` to figure_node if given self.options['name'] = name @@ -46,7 +46,7 @@ def run(self) -> list[Node]: # copy lineno from image node if figure_node.line is None and len(figure_node) == 2: - caption = cast(nodes.caption, figure_node[1]) + caption = cast('nodes.caption', figure_node[1]) figure_node.line = caption.line return [figure_node] @@ -163,7 +163,7 @@ def run(self) -> list[Node]: return ret def add_target(self, ret: list[Node]) -> None: - node = cast(nodes.math_block, ret[0]) + node = cast('nodes.math_block', ret[0]) # assign label automatically if math_number_all enabled if node['label'] == '' or (self.config.math_number_all and not node['label']): # NoQA: PLC1901 diff --git a/sphinx/domains/c/_ast.py b/sphinx/domains/c/_ast.py index 09f7388253e..14b89c4773c 100644 --- a/sphinx/domains/c/_ast.py +++ b/sphinx/domains/c/_ast.py @@ -1793,14 +1793,14 @@ def clone(self) -> ASTDeclaration: @property def name(self) -> ASTNestedName: - decl = cast(DeclarationType, self.declaration) + decl = cast('DeclarationType', self.declaration) return decl.name @property def function_params(self) -> list[ASTFunctionParameter] | None: if self.objectType != 'function': return None - decl = cast(ASTType, self.declaration) + decl = cast('ASTType', self.declaration) return decl.function_params def get_id(self, version: int, prefixed: bool = True) -> str: @@ -1851,7 +1851,7 @@ def describe_signature(self, signode: TextElement, mode: str, mainDeclNode += addnodes.desc_sig_keyword('enumerator', 'enumerator') mainDeclNode += addnodes.desc_sig_space() elif self.objectType == 'type': - decl = cast(ASTType, self.declaration) + decl = cast('ASTType', self.declaration) prefix = decl.get_type_declaration_prefix() mainDeclNode += addnodes.desc_sig_keyword(prefix, prefix) mainDeclNode += addnodes.desc_sig_space() diff --git a/sphinx/domains/citation.py b/sphinx/domains/citation.py index 0bac6040c46..2f70131f3da 100644 --- a/sphinx/domains/citation.py +++ b/sphinx/domains/citation.py @@ -117,7 +117,7 @@ def apply(self, **kwargs: Any) -> None: domain.note_citation(node) # mark citation labels as not smartquoted - label = cast(nodes.label, node[0]) + label = cast('nodes.label', node[0]) label['support_smartquotes'] = False diff --git a/sphinx/domains/python/__init__.py b/sphinx/domains/python/__init__.py index 24dfc2a5756..27e4de4ebb8 100644 --- a/sphinx/domains/python/__init__.py +++ b/sphinx/domains/python/__init__.py @@ -542,10 +542,10 @@ def filter_meta_fields(app: Sphinx, domain: str, objtype: str, content: Element) for node in content: if isinstance(node, nodes.field_list): - fields = cast(list[nodes.field], node) + fields = cast('list[nodes.field]', node) # removing list items while iterating the list needs reversed() for field in reversed(fields): - field_name = cast(nodes.field_body, field[0]).astext().strip() + field_name = cast('nodes.field_body', field[0]).astext().strip() if field_name == 'meta' or field_name.startswith('meta '): node.remove(field) diff --git a/sphinx/domains/std/__init__.py b/sphinx/domains/std/__init__.py index 0e214c92a5b..490077f8609 100644 --- a/sphinx/domains/std/__init__.py +++ b/sphinx/domains/std/__init__.py @@ -7,7 +7,6 @@ from typing import TYPE_CHECKING, Any, ClassVar, Final, cast from docutils import nodes -from docutils.nodes import Element, Node, system_message from docutils.parsers.rst import Directive, directives from docutils.statemachine import StringList @@ -25,6 +24,8 @@ if TYPE_CHECKING: from collections.abc import Callable, Iterable, Iterator, Set + from docutils.nodes import Element, Node, system_message + from sphinx.application import Sphinx from sphinx.builders import Builder from sphinx.environment import BuildEnvironment @@ -813,7 +814,7 @@ def process_doc( location=node) self.anonlabels[name] = docname, labelid if node.tagname == 'section': - title = cast(nodes.title, node[0]) + title = cast('nodes.title', node[0]) sectname = clean_astext(title) elif node.tagname == 'rubric': sectname = clean_astext(node) @@ -824,9 +825,9 @@ def process_doc( else: if (isinstance(node, nodes.definition_list | nodes.field_list) and node.children): - node = cast(nodes.Element, node.children[0]) + node = cast('nodes.Element', node.children[0]) if isinstance(node, nodes.field | nodes.definition_list_item): - node = cast(nodes.Element, node.children[0]) + node = cast('nodes.Element', node.children[0]) if isinstance(node, nodes.term | nodes.field_name): sectname = clean_astext(node) else: @@ -1117,7 +1118,7 @@ def is_enumerable_node(self, node: Node) -> bool: def get_numfig_title(self, node: Node) -> str | None: """Get the title of enumerable nodes to refer them using its title""" if self.is_enumerable_node(node): - elem = cast(Element, node) + elem = cast('Element', node) _, title_getter = self.enumerable_nodes.get(elem.__class__, (None, None)) if title_getter: return title_getter(elem) diff --git a/sphinx/environment/collectors/metadata.py b/sphinx/environment/collectors/metadata.py index aadc6af1613..6352adcc2ca 100644 --- a/sphinx/environment/collectors/metadata.py +++ b/sphinx/environment/collectors/metadata.py @@ -43,12 +43,12 @@ def process_doc(self, app: Sphinx, doctree: nodes.document) -> None: for node in doctree[index]: # type: ignore[attr-defined] # nodes are multiply inherited... if isinstance(node, nodes.authors): - authors = cast(list[nodes.author], node) + authors = cast('list[nodes.author]', node) md['authors'] = [author.astext() for author in authors] elif isinstance(node, nodes.field): assert len(node) == 2 - field_name = cast(nodes.field_name, node[0]) - field_body = cast(nodes.field_body, node[1]) + field_name = cast('nodes.field_name', node[0]) + field_body = cast('nodes.field_body', node[1]) md[field_name.astext()] = field_body.astext() elif isinstance(node, nodes.TextElement): # other children must be TextElement diff --git a/sphinx/environment/collectors/toctree.py b/sphinx/environment/collectors/toctree.py index 1de7b2473eb..6067a3856f5 100644 --- a/sphinx/environment/collectors/toctree.py +++ b/sphinx/environment/collectors/toctree.py @@ -232,7 +232,7 @@ def _walk_toc( if 'skip_section_number' in subnode: continue numstack[-1] += 1 - reference = cast(nodes.reference, subnode[0]) + reference = cast('nodes.reference', subnode[0]) if depth > 0: number = numstack.copy() secnums[reference['anchorname']] = tuple(numstack) diff --git a/sphinx/ext/autodoc/type_comment.py b/sphinx/ext/autodoc/type_comment.py index e0a5a63b91d..afdf890489e 100644 --- a/sphinx/ext/autodoc/type_comment.py +++ b/sphinx/ext/autodoc/type_comment.py @@ -94,11 +94,11 @@ def get_type_comment(obj: Any, bound_method: bool = False) -> Signature | None: # this adds if-block before the declaration. module = ast.parse('if True:\n' + source, type_comments=True) subject = cast( - ast.FunctionDef, module.body[0].body[0], # type: ignore[attr-defined] + 'ast.FunctionDef', module.body[0].body[0], # type: ignore[attr-defined] ) else: module = ast.parse(source, type_comments=True) - subject = cast(ast.FunctionDef, module.body[0]) + subject = cast('ast.FunctionDef', module.body[0]) type_comment = getattr(subject, "type_comment", None) if type_comment: diff --git a/sphinx/ext/autodoc/typehints.py b/sphinx/ext/autodoc/typehints.py index a2b05aac705..1a53d483408 100644 --- a/sphinx/ext/autodoc/typehints.py +++ b/sphinx/ext/autodoc/typehints.py @@ -3,7 +3,6 @@ from __future__ import annotations import re -from collections.abc import Iterable from typing import TYPE_CHECKING, Any, cast from docutils import nodes @@ -14,6 +13,8 @@ from sphinx.util.typing import ExtensionMetadata, stringify_annotation if TYPE_CHECKING: + from collections.abc import Iterable + from docutils.nodes import Element from sphinx.application import Sphinx @@ -49,7 +50,7 @@ def merge_typehints(app: Sphinx, domain: str, objtype: str, contentnode: Element return try: - signature = cast(addnodes.desc_signature, contentnode.parent[0]) + signature = cast('addnodes.desc_signature', contentnode.parent[0]) if signature['module']: fullname = f'{signature["module"]}.{signature["fullname"]}' else: @@ -97,7 +98,7 @@ def insert_field_list(node: Element) -> nodes.field_list: def modify_field_list(node: nodes.field_list, annotations: dict[str, str], suppress_rtype: bool = False) -> None: arguments: dict[str, dict[str, bool]] = {} - fields = cast(Iterable[nodes.field], node) + fields = cast('Iterable[nodes.field]', node) for field in fields: field_name = field[0].astext() parts = re.split(' +', field_name) @@ -159,7 +160,7 @@ def augment_descriptions_with_types( annotations: dict[str, str], force_rtype: bool, ) -> None: - fields = cast(Iterable[nodes.field], node) + fields = cast('Iterable[nodes.field]', node) has_description: set[str] = set() has_type: set[str] = set() for field in fields: diff --git a/sphinx/ext/autosectionlabel.py b/sphinx/ext/autosectionlabel.py index 63885e7039f..f3bee10e44f 100644 --- a/sphinx/ext/autosectionlabel.py +++ b/sphinx/ext/autosectionlabel.py @@ -37,7 +37,7 @@ def register_sections_as_label(app: Sphinx, document: Node) -> None: continue labelid = node['ids'][0] docname = app.env.docname - title = cast(nodes.title, node[0]) + title = cast('nodes.title', node[0]) ref_name = getattr(title, 'rawsource', title.astext()) if app.config.autosectionlabel_prefix_document: name = nodes.fully_normalize_name(docname + ':' + ref_name) diff --git a/sphinx/ext/autosummary/__init__.py b/sphinx/ext/autosummary/__init__.py index ddb93a30789..ffae20af0ee 100644 --- a/sphinx/ext/autosummary/__init__.py +++ b/sphinx/ext/autosummary/__init__.py @@ -134,13 +134,13 @@ class autosummary_table(nodes.comment): def autosummary_table_visit_html(self: HTML5Translator, node: autosummary_table) -> None: """Make the first column of the table non-breaking.""" try: - table = cast(nodes.table, node[0]) - tgroup = cast(nodes.tgroup, table[0]) - tbody = cast(nodes.tbody, tgroup[-1]) - rows = cast(list[nodes.row], tbody) + table = cast('nodes.table', node[0]) + tgroup = cast('nodes.tgroup', table[0]) + tbody = cast('nodes.tbody', tgroup[-1]) + rows = cast('list[nodes.row]', tbody) for row in rows: - col1_entry = cast(nodes.entry, row[0]) - par = cast(nodes.paragraph, col1_entry[0]) + col1_entry = cast('nodes.entry', row[0]) + par = cast('nodes.paragraph', col1_entry[0]) for j, subnode in enumerate(list(par)): if isinstance(subnode, nodes.Text): new_text = subnode.astext().replace(" ", "\u00a0") @@ -765,7 +765,7 @@ def run(self) -> tuple[list[Node], list[system_message]]: return objects, errors assert len(objects) == 1 - pending_xref = cast(addnodes.pending_xref, objects[0]) + pending_xref = cast('addnodes.pending_xref', objects[0]) try: # try to import object by name prefixes = get_import_prefixes_from_env(self.env) @@ -778,7 +778,7 @@ def run(self) -> tuple[list[Node], list[system_message]]: ] import_by_name(name, prefixes) except ImportExceptionGroup: - literal = cast(nodes.literal, pending_xref[0]) + literal = cast('nodes.literal', pending_xref[0]) objects[0] = nodes.emphasis(self.rawtext, literal.astext(), classes=literal['classes']) diff --git a/sphinx/ext/inheritance_diagram.py b/sphinx/ext/inheritance_diagram.py index 47dec78aa9f..1bdbc2513ab 100644 --- a/sphinx/ext/inheritance_diagram.py +++ b/sphinx/ext/inheritance_diagram.py @@ -35,7 +35,6 @@ class E(B): pass import inspect import os.path import re -from collections.abc import Iterable, Sequence from importlib import import_module from typing import TYPE_CHECKING, Any, ClassVar, cast @@ -54,6 +53,8 @@ class E(B): pass from sphinx.util.docutils import SphinxDirective if TYPE_CHECKING: + from collections.abc import Iterable, Sequence + from docutils.nodes import Node from sphinx.application import Sphinx @@ -424,7 +425,7 @@ def html_visit_inheritance_diagram(self: HTML5Translator, node: inheritance_diag graphviz_output_format = self.builder.env.config.graphviz_output_format.upper() current_filename = os.path.basename(self.builder.current_docname + self.builder.out_suffix) urls = {} - pending_xrefs = cast(Iterable[addnodes.pending_xref], node) + pending_xrefs = cast('Iterable[addnodes.pending_xref]', node) for child in pending_xrefs: if child.get('refuri') is not None: # Construct the name from the URI if the reference is external via intersphinx diff --git a/sphinx/ext/intersphinx/_resolve.py b/sphinx/ext/intersphinx/_resolve.py index 0dbab63dc69..94101588bfa 100644 --- a/sphinx/ext/intersphinx/_resolve.py +++ b/sphinx/ext/intersphinx/_resolve.py @@ -569,7 +569,7 @@ def run(self, **kwargs: Any) -> None: for node in self.document.findall(pending_xref): if 'intersphinx' not in node: continue - contnode = cast(nodes.TextElement, node[0].deepcopy()) + contnode = cast('nodes.TextElement', node[0].deepcopy()) inv_name = node['inventory'] if inv_name is not None: assert inventory_exists(self.env, inv_name) diff --git a/sphinx/ext/mathjax.py b/sphinx/ext/mathjax.py index ba8cd03e945..e9357fee604 100644 --- a/sphinx/ext/mathjax.py +++ b/sphinx/ext/mathjax.py @@ -13,13 +13,13 @@ from docutils import nodes import sphinx -from sphinx.builders.html import StandaloneHTMLBuilder from sphinx.errors import ExtensionError from sphinx.locale import _ from sphinx.util.math import get_node_equation_number if TYPE_CHECKING: from sphinx.application import Sphinx + from sphinx.builders.html import StandaloneHTMLBuilder from sphinx.util.typing import ExtensionMetadata from sphinx.writers.html5 import HTML5Translator @@ -82,7 +82,7 @@ def install_mathjax(app: Sphinx, pagename: str, templatename: str, context: dict raise ExtensionError(msg) domain = app.env.domains.math_domain - builder = cast(StandaloneHTMLBuilder, app.builder) + builder = cast('StandaloneHTMLBuilder', app.builder) if app.registry.html_assets_policy == 'always' or domain.has_equations(pagename): # Enable mathjax only if equations exists if app.config.mathjax2_config: diff --git a/sphinx/ext/todo.py b/sphinx/ext/todo.py index 0625621b6db..cec0599b4aa 100644 --- a/sphinx/ext/todo.py +++ b/sphinx/ext/todo.py @@ -214,7 +214,7 @@ def latex_visit_todo_node(self: LaTeXTranslator, node: todo_node) -> None: self.body.append('\n\\begin{sphinxtodo}{') self.body.append(self.hypertarget_to(node)) - title_node = cast(nodes.title, node[0]) + title_node = cast('nodes.title', node[0]) title = texescape.escape(title_node.astext(), self.config.latex_engine) self.body.append('%s:}' % title) self.no_latex_floats += 1 diff --git a/sphinx/ext/viewcode.py b/sphinx/ext/viewcode.py index da1713aa6fa..afc4b06f5cc 100644 --- a/sphinx/ext/viewcode.py +++ b/sphinx/ext/viewcode.py @@ -14,7 +14,6 @@ import sphinx from sphinx import addnodes -from sphinx.builders.html import StandaloneHTMLBuilder from sphinx.locale import _, __ from sphinx.pycode import ModuleAnalyzer from sphinx.transforms.post_transforms import SphinxPostTransform @@ -28,6 +27,7 @@ from sphinx.application import Sphinx from sphinx.builders import Builder + from sphinx.builders.html import StandaloneHTMLBuilder from sphinx.environment import BuildEnvironment from sphinx.util._pathlib import _StrPath from sphinx.util.typing import ExtensionMetadata @@ -228,7 +228,7 @@ def should_generate_module_page(app: Sphinx, modname: str) -> bool: # Always (re-)generate module page when module filename is not found. return True - builder = cast(StandaloneHTMLBuilder, app.builder) + builder = cast('StandaloneHTMLBuilder', app.builder) basename = modname.replace('.', '/') + builder.out_suffix page_filename = os.path.join(app.outdir, '_modules/', basename) diff --git a/sphinx/transforms/__init__.py b/sphinx/transforms/__init__.py index ca3a533fab0..a1c9f28c720 100644 --- a/sphinx/transforms/__init__.py +++ b/sphinx/transforms/__init__.py @@ -443,11 +443,11 @@ class GlossarySorter(SphinxTransform): def apply(self, **kwargs: Any) -> None: for glossary in self.document.findall(addnodes.glossary): if glossary['sorted']: - definition_list = cast(nodes.definition_list, glossary[0]) + definition_list = cast('nodes.definition_list', glossary[0]) definition_list[:] = sorted( definition_list, key=lambda item: unicodedata.normalize( - 'NFD', cast(nodes.term, item)[0].astext().lower() + 'NFD', cast('nodes.term', item)[0].astext().lower() ), ) diff --git a/sphinx/transforms/compact_bullet_list.py b/sphinx/transforms/compact_bullet_list.py index f144df87fb2..ffc67007bec 100644 --- a/sphinx/transforms/compact_bullet_list.py +++ b/sphinx/transforms/compact_bullet_list.py @@ -75,8 +75,8 @@ def check_refonly_list(node: Node) -> bool: for node in self.document.findall(nodes.bullet_list): if check_refonly_list(node): for item in node.findall(nodes.list_item): - para = cast(nodes.paragraph, item[0]) - ref = cast(nodes.reference, para[0]) + para = cast('nodes.paragraph', item[0]) + ref = cast('nodes.reference', para[0]) compact_para = addnodes.compact_paragraph() compact_para += ref item.replace(para, compact_para) diff --git a/sphinx/transforms/post_transforms/__init__.py b/sphinx/transforms/post_transforms/__init__.py index cb590b77a10..de880c9c27f 100644 --- a/sphinx/transforms/post_transforms/__init__.py +++ b/sphinx/transforms/post_transforms/__init__.py @@ -7,7 +7,6 @@ from typing import TYPE_CHECKING, Any, cast from docutils import nodes -from docutils.nodes import Element, Node from sphinx import addnodes from sphinx.errors import NoUri @@ -20,6 +19,8 @@ if TYPE_CHECKING: from collections.abc import Sequence + from docutils.nodes import Element, Node + from sphinx.addnodes import pending_xref from sphinx.application import Sphinx from sphinx.domains import Domain @@ -68,9 +69,9 @@ def run(self, **kwargs: Any) -> None: for node in self.document.findall(addnodes.pending_xref): content = self.find_pending_xref_condition(node, ('resolved', '*')) if content: - contnode = cast(Element, content[0].deepcopy()) + contnode = cast('Element', content[0].deepcopy()) else: - contnode = cast(Element, node[0].deepcopy()) + contnode = cast('Element', node[0].deepcopy()) newnode = None diff --git a/sphinx/util/docfields.py b/sphinx/util/docfields.py index 33fda6f5dcd..1bfcd7451e4 100644 --- a/sphinx/util/docfields.py +++ b/sphinx/util/docfields.py @@ -10,7 +10,6 @@ from typing import TYPE_CHECKING, Any, cast from docutils import nodes -from docutils.nodes import Element, Node from sphinx import addnodes from sphinx.locale import __ @@ -18,6 +17,7 @@ from sphinx.util.nodes import get_node_line if TYPE_CHECKING: + from docutils.nodes import Element, Node from docutils.parsers.rst.states import Inliner from sphinx.directives import ObjectDescription @@ -237,7 +237,7 @@ def make_field( listnode += nodes.list_item('', par) if len(items) == 1 and self.can_collapse: - list_item = cast(nodes.list_item, listnode[0]) + list_item = cast('nodes.list_item', listnode[0]) fieldbody = nodes.field_body('', list_item[0]) return nodes.field('', fieldname, fieldbody) @@ -366,10 +366,10 @@ def transform(self, node: nodes.field_list) -> None: types: dict[str, dict] = {} # step 1: traverse all fields and collect field types and content - for field in cast(list[nodes.field], node): + for field in cast('list[nodes.field]', node): assert len(field) == 2 - field_name = cast(nodes.field_name, field[0]) - field_body = cast(nodes.field_body, field[1]) + field_name = cast('nodes.field_name', field[0]) + field_body = cast('nodes.field_body', field[1]) try: # split into field type and argument fieldtype_name, fieldarg = field_name.astext().split(None, 1) @@ -380,7 +380,7 @@ def transform(self, node: nodes.field_list) -> None: # collect the content, trying not to keep unnecessary paragraphs if _is_single_paragraph(field_body): - paragraph = cast(nodes.paragraph, field_body[0]) + paragraph = cast('nodes.paragraph', field_body[0]) content = paragraph.children else: content = field_body.children @@ -403,7 +403,7 @@ def transform(self, node: nodes.field_list) -> None: and len(content) == 1 and isinstance(content[0], nodes.Text) ): - typed_field = cast(TypedField, typedesc) + typed_field = cast('TypedField', typedesc) target = content[0].astext() xrefs = typed_field.make_xrefs( typed_field.typerolename, @@ -413,7 +413,7 @@ def transform(self, node: nodes.field_list) -> None: env=self.directive.state.document.settings.env, ) if _is_single_paragraph(field_body): - paragraph = cast(nodes.paragraph, field_body[0]) + paragraph = cast('nodes.paragraph', field_body[0]) paragraph.clear() paragraph.extend(xrefs) else: @@ -456,7 +456,7 @@ def transform(self, node: nodes.field_list) -> None: if typedesc.is_grouped: if typename in groupindices: group = cast( - tuple[Field, list, Node], entries[groupindices[typename]] + 'tuple[Field, list, Node]', entries[groupindices[typename]] ) else: groupindices[typename] = len(entries) diff --git a/sphinx/util/docutils.py b/sphinx/util/docutils.py index f8fa00f4316..ed8ebdb3544 100644 --- a/sphinx/util/docutils.py +++ b/sphinx/util/docutils.py @@ -355,7 +355,7 @@ def __init__( debug: bool = False, error_handler: str = 'backslashreplace', ) -> None: - stream = cast(IO, WarningStream()) + stream = cast('IO', WarningStream()) super().__init__( source, report_level, halt_level, stream, debug, error_handler=error_handler ) diff --git a/sphinx/util/inspect.py b/sphinx/util/inspect.py index b25d5f06aeb..e2bc8b1e6a7 100644 --- a/sphinx/util/inspect.py +++ b/sphinx/util/inspect.py @@ -848,7 +848,7 @@ def signature_from_str(signature: str) -> Signature: """Create a :class:`~inspect.Signature` object from a string.""" code = 'def func' + signature + ': pass' module = ast.parse(code) - function = typing.cast(ast.FunctionDef, module.body[0]) + function = typing.cast('ast.FunctionDef', module.body[0]) return signature_from_ast(function, code) diff --git a/sphinx/util/nodes.py b/sphinx/util/nodes.py index 7f06ae194fc..19c21986d98 100644 --- a/sphinx/util/nodes.py +++ b/sphinx/util/nodes.py @@ -95,7 +95,7 @@ def findall(self, node: Node) -> Iterator[N]: confounds type checkers' ability to determine the return type of the iterator. """ for found in node.findall(self): - yield cast(N, found) + yield cast('N', found) def get_full_module_name(node: Node) -> str: diff --git a/sphinx/util/rst.py b/sphinx/util/rst.py index 85915ef894b..412edc31b25 100644 --- a/sphinx/util/rst.py +++ b/sphinx/util/rst.py @@ -105,7 +105,7 @@ def append_epilog(content: StringList, epilog: str) -> None: if len(content) > 0: source, lineno = content.info(-1) # lineno will never be None, since len(content) > 0 - lineno = cast(int, lineno) + lineno = cast('int', lineno) else: source = '' lineno = 0 diff --git a/sphinx/writers/html.py b/sphinx/writers/html.py index 4b193345da0..e9640dc2925 100644 --- a/sphinx/writers/html.py +++ b/sphinx/writers/html.py @@ -31,7 +31,7 @@ def __init__(self, builder: StandaloneHTMLBuilder) -> None: def translate(self) -> None: # sadly, this is mostly copied from parent class visitor = self.builder.create_translator(self.document, self.builder) - self.visitor = cast(HTML5Translator, visitor) + self.visitor = cast('HTML5Translator', visitor) self.document.walkabout(visitor) self.output = self.visitor.astext() for attr in ( diff --git a/sphinx/writers/html5.py b/sphinx/writers/html5.py index 608a84f34db..d5c77fee8fe 100644 --- a/sphinx/writers/html5.py +++ b/sphinx/writers/html5.py @@ -5,7 +5,6 @@ import posixpath import re import urllib.parse -from collections.abc import Iterable from typing import TYPE_CHECKING, cast from docutils import nodes @@ -18,6 +17,8 @@ from sphinx.util.images import get_image_size if TYPE_CHECKING: + from collections.abc import Iterable + from docutils.nodes import Element, Node, Text from sphinx.builders import Builder @@ -670,7 +671,7 @@ def depart_literal(self, node: Element) -> None: def visit_productionlist(self, node: Element) -> None: self.body.append(self.starttag(node, 'pre')) - productionlist = cast(Iterable[addnodes.production], node) + productionlist = cast('Iterable[addnodes.production]', node) names = (production['tokenname'] for production in productionlist) maxlen = max(len(name) for name in names) lastname = None diff --git a/sphinx/writers/latex.py b/sphinx/writers/latex.py index 454543b043c..20f0cc0fdbb 100644 --- a/sphinx/writers/latex.py +++ b/sphinx/writers/latex.py @@ -8,7 +8,6 @@ import re from collections import defaultdict -from collections.abc import Iterable from pathlib import Path from typing import TYPE_CHECKING, Any, ClassVar, cast @@ -26,6 +25,8 @@ from sphinx.util.texescape import tex_replace_map if TYPE_CHECKING: + from collections.abc import Iterable + from docutils.nodes import Element, Node, Text from sphinx.builders.latex import LaTeXBuilder @@ -95,7 +96,7 @@ def translate(self) -> None: self.document, self.builder, self.theme ) self.document.walkabout(visitor) - self.output = cast(LaTeXTranslator, visitor).astext() + self.output = cast('LaTeXTranslator', visitor).astext() # Helper classes @@ -1111,7 +1112,7 @@ def depart_rubric(self, node: nodes.rubric) -> None: def visit_footnote(self, node: Element) -> None: self.in_footnote += 1 - label = cast(nodes.label, node[0]) + label = cast('nodes.label', node[0]) if self.in_parsed_literal: self.body.append(r'\begin{footnote}[%s]' % label.astext()) else: @@ -1382,8 +1383,8 @@ def depart_entry(self, node: Element) -> None: def visit_acks(self, node: Element) -> None: # this is a list in the source, but should be rendered as a # comma-separated list here - bullet_list = cast(nodes.bullet_list, node[0]) - list_items = cast(Iterable[nodes.list_item], bullet_list) + bullet_list = cast('nodes.bullet_list', node[0]) + list_items = cast('Iterable[nodes.list_item]', bullet_list) self.body.append(BLANKLINE) self.body.append(', '.join(n.astext() for n in list_items) + '.') self.body.append(BLANKLINE) @@ -2091,8 +2092,8 @@ def depart_title_reference(self, node: Element) -> None: self.body.append('}') def visit_thebibliography(self, node: Element) -> None: - citations = cast(Iterable[nodes.citation], node) - labels = (cast(nodes.label, citation[0]) for citation in citations) + citations = cast('Iterable[nodes.citation]', node) + labels = (cast('nodes.label', citation[0]) for citation in citations) longest_label = max((label.astext() for label in labels), key=len) if len(longest_label) > MAX_CITATION_LABEL_LENGTH: # adjust max width of citation labels not to break the layout @@ -2106,7 +2107,7 @@ def depart_thebibliography(self, node: Element) -> None: self.body.append(r'\end{sphinxthebibliography}' + CR) def visit_citation(self, node: Element) -> None: - label = cast(nodes.label, node[0]) + label = cast('nodes.label', node[0]) self.body.append( rf'\bibitem[{self.encode(label.astext())}]' rf'{{{node["docname"]}:{node["ids"][0]}}}' @@ -2159,7 +2160,7 @@ def depart_footnotemark(self, node: Element) -> None: self.body.append(']') def visit_footnotetext(self, node: Element) -> None: - label = cast(nodes.label, node[0]) + label = cast('nodes.label', node[0]) self.body.append('%' + CR) self.body.append(r'\begin{footnotetext}[%s]' % label.astext()) self.body.append(r'\sphinxAtStartFootnote' + CR) diff --git a/sphinx/writers/manpage.py b/sphinx/writers/manpage.py index 7b7db13961f..0f5a37572e8 100644 --- a/sphinx/writers/manpage.py +++ b/sphinx/writers/manpage.py @@ -2,7 +2,6 @@ from __future__ import annotations -from collections.abc import Iterable from typing import TYPE_CHECKING, Any, cast from docutils import nodes @@ -17,6 +16,8 @@ from sphinx.util.nodes import NodeMatcher if TYPE_CHECKING: + from collections.abc import Iterable + from docutils.nodes import Element from sphinx.builders import Builder @@ -33,7 +34,7 @@ def translate(self) -> None: transform = NestedInlineTransform(self.document) transform.apply() visitor = self.builder.create_translator(self.document, self.builder) - self.visitor = cast(ManualPageTranslator, visitor) + self.visitor = cast('ManualPageTranslator', visitor) self.document.walkabout(visitor) self.output = self.visitor.astext() @@ -277,7 +278,7 @@ def visit_productionlist(self, node: Element) -> None: self.ensure_eol() self.in_productionlist += 1 self.body.append('.sp\n.nf\n') - productionlist = cast(Iterable[addnodes.production], node) + productionlist = cast('Iterable[addnodes.production]', node) names = (production['tokenname'] for production in productionlist) maxlen = max(len(name) for name in names) lastname = None @@ -379,11 +380,11 @@ def depart_glossary(self, node: Element) -> None: pass def visit_acks(self, node: Element) -> None: - bullet_list = cast(nodes.bullet_list, node[0]) - list_items = cast(Iterable[nodes.list_item], bullet_list) + bullet_list = cast('nodes.bullet_list', node[0]) + list_items = cast('Iterable[nodes.list_item]', bullet_list) self.ensure_eol() - bullet_list = cast(nodes.bullet_list, node[0]) - list_items = cast(Iterable[nodes.list_item], bullet_list) + bullet_list = cast('nodes.bullet_list', node[0]) + list_items = cast('Iterable[nodes.list_item]', bullet_list) self.body.append(', '.join(n.astext() for n in list_items) + '.') self.body.append('\n') raise nodes.SkipNode diff --git a/sphinx/writers/texinfo.py b/sphinx/writers/texinfo.py index 396e48bf91a..bae5714425e 100644 --- a/sphinx/writers/texinfo.py +++ b/sphinx/writers/texinfo.py @@ -5,7 +5,6 @@ import os.path import re import textwrap -from collections.abc import Iterable, Iterator from typing import TYPE_CHECKING, Any, ClassVar, cast from docutils import nodes, writers @@ -19,6 +18,8 @@ from sphinx.writers.latex import collected_footnote if TYPE_CHECKING: + from collections.abc import Iterable, Iterator + from docutils.nodes import Element, Node, Text from sphinx.builders.texinfo import TexinfoBuilder @@ -133,7 +134,7 @@ def __init__(self, builder: TexinfoBuilder) -> None: def translate(self) -> None: assert isinstance(self.document, nodes.document) visitor = self.builder.create_translator(self.document, self.builder) - self.visitor = cast(TexinfoTranslator, visitor) + self.visitor = cast('TexinfoTranslator', visitor) self.document.walkabout(visitor) self.visitor.finish() for attr in self.visitor_attributes: @@ -288,7 +289,7 @@ def add_node_name(name: str) -> str: ] # each section is also a node for section in self.document.findall(nodes.section): - title = cast(nodes.TextElement, section.next_node(nodes.Titular)) # type: ignore[type-var] + title = cast('nodes.TextElement', section.next_node(nodes.Titular)) # type: ignore[type-var] name = title.astext() if title else '' section['node_name'] = add_node_name(name) @@ -530,7 +531,7 @@ def footnotes_under(n: Element) -> Iterator[nodes.footnote]: fnotes: dict[str, list[collected_footnote | bool]] = {} for fn in footnotes_under(node): - label = cast(nodes.label, fn[0]) + label = cast('nodes.label', fn[0]) num = label.astext().strip() fnotes[num] = [collected_footnote('', *fn.children), False] return fnotes @@ -609,7 +610,7 @@ def visit_section(self, node: Element) -> None: self.add_anchor(id, node) self.next_section_ids.clear() - self.previous_section = cast(nodes.section, node) + self.previous_section = cast('nodes.section', node) self.section_level += 1 def depart_section(self, node: Element) -> None: @@ -1110,7 +1111,7 @@ def depart_field_body(self, node: Element) -> None: def visit_admonition(self, node: Element, name: str = '') -> None: if not name: - title = cast(nodes.title, node[0]) + title = cast('nodes.title', node[0]) name = self.escape(title.astext()) self.body.append('\n@cartouche\n@quotation %s ' % name) @@ -1173,7 +1174,7 @@ def visit_topic(self, node: Element) -> None: # ignore TOC's since we have to have a "menu" anyway if 'contents' in node.get('classes', []): raise nodes.SkipNode - title = cast(nodes.title, node[0]) + title = cast('nodes.title', node[0]) self.visit_rubric(title) self.body.append('%s\n' % self.escape(title.astext())) self.depart_rubric(title) @@ -1307,7 +1308,7 @@ def unknown_departure(self, node: Node) -> None: def visit_productionlist(self, node: Element) -> None: self.visit_literal_block(None) - productionlist = cast(Iterable[addnodes.production], node) + productionlist = cast('Iterable[addnodes.production]', node) names = (production['tokenname'] for production in productionlist) maxlen = max(len(name) for name in names) @@ -1388,8 +1389,8 @@ def depart_glossary(self, node: Element) -> None: pass def visit_acks(self, node: Element) -> None: - bullet_list = cast(nodes.bullet_list, node[0]) - list_items = cast(Iterable[nodes.list_item], bullet_list) + bullet_list = cast('nodes.bullet_list', node[0]) + list_items = cast('Iterable[nodes.list_item]', bullet_list) self.body.append('\n\n') self.body.append(', '.join(n.astext() for n in list_items) + '.') self.body.append('\n\n') diff --git a/sphinx/writers/text.py b/sphinx/writers/text.py index c16181ec247..43648c5a6b9 100644 --- a/sphinx/writers/text.py +++ b/sphinx/writers/text.py @@ -6,7 +6,6 @@ import os import re import textwrap -from collections.abc import Iterable, Iterator, Sequence from itertools import chain, groupby, pairwise from typing import TYPE_CHECKING, Any, ClassVar, cast @@ -18,6 +17,8 @@ from sphinx.util.docutils import SphinxTranslator if TYPE_CHECKING: + from collections.abc import Iterable, Iterator, Sequence + from docutils.nodes import Element, Text from sphinx.builders.text import TextBuilder @@ -381,7 +382,7 @@ def translate(self) -> None: assert isinstance(self.document, nodes.document) visitor = self.builder.create_translator(self.document, self.builder) self.document.walkabout(visitor) - self.output = cast(TextTranslator, visitor).body + self.output = cast('TextTranslator', visitor).body class TextTranslator(SphinxTranslator): @@ -776,7 +777,7 @@ def depart_caption(self, node: Element) -> None: def visit_productionlist(self, node: Element) -> None: self.new_state() - productionlist = cast(Iterable[addnodes.production], node) + productionlist = cast('Iterable[addnodes.production]', node) names = (production['tokenname'] for production in productionlist) maxlen = max(len(name) for name in names) lastname = None @@ -791,7 +792,7 @@ def visit_productionlist(self, node: Element) -> None: raise nodes.SkipNode def visit_footnote(self, node: Element) -> None: - label = cast(nodes.label, node[0]) + label = cast('nodes.label', node[0]) self._footnote = label.astext().strip() self.new_state(len(self._footnote) + 3) @@ -923,8 +924,8 @@ def depart_table(self, node: Element) -> None: self.end_state(wrap=False) def visit_acks(self, node: Element) -> None: - bullet_list = cast(nodes.bullet_list, node[0]) - list_items = cast(Iterable[nodes.list_item], bullet_list) + bullet_list = cast('nodes.bullet_list', node[0]) + list_items = cast('Iterable[nodes.list_item]', bullet_list) self.new_state(0) self.add_text(', '.join(n.astext() for n in list_items) + '.') self.end_state() From 439bff92c7f0a9b5cfa798e2b674b418f47f825d Mon Sep 17 00:00:00 2001 From: Adam Turner <9087854+aa-turner@users.noreply.github.com> Date: Sat, 30 Nov 2024 19:14:22 +0000 Subject: [PATCH 21/29] Enable the RUF003 lint in Ruff --- .ruff.toml | 2 +- sphinx/util/i18n.py | 16 ++++++++-------- 2 files changed, 9 insertions(+), 9 deletions(-) diff --git a/.ruff.toml b/.ruff.toml index be9ac015a33..22cb9f4bb40 100644 --- a/.ruff.toml +++ b/.ruff.toml @@ -222,7 +222,7 @@ select = [ # ruff-specific rules ('RUF') # "RUF001", # String contains ambiguous {}. Did you mean {}? "RUF002", # Docstring contains ambiguous {}. Did you mean {}? -# "RUF003", # Comment contains ambiguous {}. Did you mean {}? + "RUF003", # Comment contains ambiguous {}. Did you mean {}? "RUF005", # Consider `{expression}` instead of concatenation "RUF006", # Store a reference to the return value of `{expr}.{method}` "RUF007", # Prefer `itertools.pairwise()` over `zip()` when iterating over successive pairs diff --git a/sphinx/util/i18n.py b/sphinx/util/i18n.py index 826696c83e3..48425a03385 100644 --- a/sphinx/util/i18n.py +++ b/sphinx/util/i18n.py @@ -167,11 +167,11 @@ def docname_to_domain(docname: str, compaction: bool | str) -> str: # date_format mappings: ustrftime() to babel.dates.format_datetime() date_format_mappings = { - '%a': 'EEE', # Weekday as locale’s abbreviated name. - '%A': 'EEEE', # Weekday as locale’s full name. - '%b': 'MMM', # Month as locale’s abbreviated name. - '%B': 'MMMM', # Month as locale’s full name. - '%c': 'medium', # Locale’s appropriate date and time representation. + '%a': 'EEE', # Weekday as locale's abbreviated name. + '%A': 'EEEE', # Weekday as locale's full name. + '%b': 'MMM', # Month as locale's abbreviated name. + '%B': 'MMMM', # Month as locale's full name. + '%c': 'medium', # Locale's appropriate date and time representation. '%-d': 'd', # Day of the month as a decimal number. '%d': 'dd', # Day of the month as a zero-padded decimal number. '%-H': 'H', # Hour (24-hour clock) as a decimal number [0,23]. @@ -184,7 +184,7 @@ def docname_to_domain(docname: str, compaction: bool | str) -> str: '%m': 'MM', # Month as a zero-padded decimal number. '%-M': 'm', # Minute as a decimal number [0,59]. '%M': 'mm', # Minute as a zero-padded decimal number [00,59]. - '%p': 'a', # Locale’s equivalent of either AM or PM. + '%p': 'a', # Locale's equivalent of either AM or PM. '%-S': 's', # Second as a decimal number. '%S': 'ss', # Second as a zero-padded decimal number. '%U': 'WW', # Week number of the year (Sunday as the first day of the week) @@ -196,8 +196,8 @@ def docname_to_domain(docname: str, compaction: bool | str) -> str: # Monday are considered to be in week 0. '%W': 'WW', # Week number of the year (Monday as the first day of the week) # as a zero-padded decimal number. - '%x': 'medium', # Locale’s appropriate date representation. - '%X': 'medium', # Locale’s appropriate time representation. + '%x': 'medium', # Locale's appropriate date representation. + '%X': 'medium', # Locale's appropriate time representation. '%y': 'YY', # Year without century as a zero-padded decimal number. '%Y': 'yyyy', # Year with century as a decimal number. '%Z': 'zzz', # Time zone name (no characters if no time zone exists). From 2f1cd36798a42dfe1f1bae7ded43e83174389e3e Mon Sep 17 00:00:00 2001 From: Adam Turner <9087854+aa-turner@users.noreply.github.com> Date: Sat, 30 Nov 2024 19:19:34 +0000 Subject: [PATCH 22/29] Enable the RUF015 lint in Ruff --- .ruff.toml | 2 +- tests/test_domains/test_domain_std.py | 20 ++++++++++++-------- tests/test_extensions/test_ext_coverage.py | 2 +- tests/test_util/test_util_nodes.py | 2 +- 4 files changed, 15 insertions(+), 11 deletions(-) diff --git a/.ruff.toml b/.ruff.toml index 22cb9f4bb40..3523672ca6f 100644 --- a/.ruff.toml +++ b/.ruff.toml @@ -231,7 +231,7 @@ select = [ "RUF010", # Use explicit conversion flag # "RUF012", # Mutable class attributes should be annotated with `typing.ClassVar` "RUF013", # PEP 484 prohibits implicit `Optional` -# "RUF015", # Prefer `next({iterable})` over single element slice + "RUF015", # Prefer `next({iterable})` over single element slice "RUF016", # Slice in indexed access to type `{value_type}` uses type `{index_type}` instead of an integer "RUF017", # Avoid quadratic list summation "RUF018", # Avoid assignment expressions in `assert` statements diff --git a/tests/test_domains/test_domain_std.py b/tests/test_domains/test_domain_std.py index 4684596bd70..993fa197af8 100644 --- a/tests/test_domains/test_domain_std.py +++ b/tests/test_domains/test_domain_std.py @@ -504,19 +504,23 @@ def test_productionlist(app): ul = nodes[2] cases = [] for li in list(ul): - assert len(list(li)) == 1 - p = list(li)[0] + li_list = list(li) + assert len(li_list) == 1 + p = li_list[0] assert p.tag == 'p' text = str(p.text).strip(' :') - assert len(list(p)) == 1 - a = list(p)[0] + p_list = list(p) + assert len(p_list) == 1 + a = p_list[0] assert a.tag == 'a' link = a.get('href') - assert len(list(a)) == 1 - code = list(a)[0] + a_list = list(a) + assert len(a_list) == 1 + code = a_list[0] assert code.tag == 'code' - assert len(list(code)) == 1 - span = list(code)[0] + code_list = list(code) + assert len(code_list) == 1 + span = code_list[0] assert span.tag == 'span' linkText = span.text.strip() cases.append((text, link, linkText)) diff --git a/tests/test_extensions/test_ext_coverage.py b/tests/test_extensions/test_ext_coverage.py index f788314af91..5dc2f95a8bc 100644 --- a/tests/test_extensions/test_ext_coverage.py +++ b/tests/test_extensions/test_ext_coverage.py @@ -39,7 +39,7 @@ def test_build(app): ) assert len(undoc_c) == 1 # the key is the full path to the header file, which isn't testable - assert list(undoc_c.values())[0] == {('function', 'Py_SphinxTest')} + assert next(iter(undoc_c.values())) == {('function', 'Py_SphinxTest')} assert 'autodoc_target' in undoc_py assert 'funcs' in undoc_py['autodoc_target'] diff --git a/tests/test_util/test_util_nodes.py b/tests/test_util/test_util_nodes.py index a330eff366d..9595e4a64df 100644 --- a/tests/test_util/test_util_nodes.py +++ b/tests/test_util/test_util_nodes.py @@ -180,7 +180,7 @@ def test_extract_messages_without_rawsource(): document.append(p) _transform(document) assert_node_count(extract_messages(document), nodes.TextElement, 1) - assert [m for n, m in extract_messages(document)][0], 'text sentence' + assert next(m for n, m in extract_messages(document)), 'text sentence' def test_clean_astext(): From 6375b342dbe2d47552cede6a818fcd258465b312 Mon Sep 17 00:00:00 2001 From: Adam Turner <9087854+aa-turner@users.noreply.github.com> Date: Sat, 30 Nov 2024 19:19:57 +0000 Subject: [PATCH 23/29] Enable the RUF028 lint in Ruff --- .ruff.toml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.ruff.toml b/.ruff.toml index 3523672ca6f..f43f5148179 100644 --- a/.ruff.toml +++ b/.ruff.toml @@ -243,7 +243,7 @@ select = [ "RUF024", # Do not pass mutable objects as values to `dict.fromkeys` "RUF026", # `default_factory` is a positional-only argument to `defaultdict` # "RUF027", # Possible f-string without an `f` prefix -# "RUF028", # This suppression comment is invalid because {} + "RUF028", # This suppression comment is invalid because {} # "RUF029", # Function `{name}` is declared `async`, but doesn't `await` or use `async` features. "RUF030", # `print()` expression in `assert` statement is likely unintentional # "RUF031", # Use parentheses for tuples in subscripts. From d796a8ae21756ce32561df4f342e5886be8ee88e Mon Sep 17 00:00:00 2001 From: Adam Turner <9087854+aa-turner@users.noreply.github.com> Date: Sat, 30 Nov 2024 19:21:35 +0000 Subject: [PATCH 24/29] Enable the RUF029 lint in Ruff --- .ruff.toml | 2 +- tests/test_util/test_util_inspect.py | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/.ruff.toml b/.ruff.toml index f43f5148179..97ba78e7e56 100644 --- a/.ruff.toml +++ b/.ruff.toml @@ -244,7 +244,7 @@ select = [ "RUF026", # `default_factory` is a positional-only argument to `defaultdict` # "RUF027", # Possible f-string without an `f` prefix "RUF028", # This suppression comment is invalid because {} -# "RUF029", # Function `{name}` is declared `async`, but doesn't `await` or use `async` features. + "RUF029", # Function `{name}` is declared `async`, but doesn't `await` or use `async` features. "RUF030", # `print()` expression in `assert` statement is likely unintentional # "RUF031", # Use parentheses for tuples in subscripts. "RUF032", # `Decimal()` called with float literal argument diff --git a/tests/test_util/test_util_inspect.py b/tests/test_util/test_util_inspect.py index 5c868efa6e9..8e6c1e01bcb 100644 --- a/tests/test_util/test_util_inspect.py +++ b/tests/test_util/test_util_inspect.py @@ -62,7 +62,7 @@ async def coroutinefunc(): pass -async def asyncgenerator(): +async def asyncgenerator(): # NoQA: RUF029 yield From 19dfc3f5afa31407df2c009540237bd09731a51b Mon Sep 17 00:00:00 2001 From: Adam Turner <9087854+aa-turner@users.noreply.github.com> Date: Sat, 30 Nov 2024 19:22:30 +0000 Subject: [PATCH 25/29] Enable the RUF031 lint in Ruff --- .ruff.toml | 2 +- sphinx/builders/latex/transforms.py | 4 ++-- sphinx/pycode/parser.py | 4 ++-- sphinx/writers/latex.py | 16 ++++++++-------- tests/test_domains/test_domain_std.py | 8 ++++---- tests/test_extensions/test_ext_doctest.py | 2 +- tests/test_pycode/test_pycode.py | 20 ++++++++++---------- 7 files changed, 28 insertions(+), 28 deletions(-) diff --git a/.ruff.toml b/.ruff.toml index 97ba78e7e56..4bca08fc8a4 100644 --- a/.ruff.toml +++ b/.ruff.toml @@ -246,7 +246,7 @@ select = [ "RUF028", # This suppression comment is invalid because {} "RUF029", # Function `{name}` is declared `async`, but doesn't `await` or use `async` features. "RUF030", # `print()` expression in `assert` statement is likely unintentional -# "RUF031", # Use parentheses for tuples in subscripts. + "RUF031", # Use parentheses for tuples in subscripts. "RUF032", # `Decimal()` called with float literal argument "RUF033", # `__post_init__` method with argument defaults "RUF034", # Useless if-else condition diff --git a/sphinx/builders/latex/transforms.py b/sphinx/builders/latex/transforms.py index b6d0439d18a..df478533770 100644 --- a/sphinx/builders/latex/transforms.py +++ b/sphinx/builders/latex/transforms.py @@ -455,7 +455,7 @@ def visit_footnote_reference(self, node: nodes.footnote_reference) -> None: number = node.astext().strip() docname = node['docname'] if (docname, number) in self.appeared: - footnote = self.appeared[(docname, number)] + footnote = self.appeared[docname, number] footnote['referred'] = True mark = footnotemark('', number, refid=node['refid']) @@ -471,7 +471,7 @@ def visit_footnote_reference(self, node: nodes.footnote_reference) -> None: node.replace_self(footnote) footnote.walkabout(self) - self.appeared[(docname, number)] = footnote + self.appeared[docname, number] = footnote raise nodes.SkipNode def get_footnote_by_reference( diff --git a/sphinx/pycode/parser.py b/sphinx/pycode/parser.py index 26ecb913a50..7c59b158e48 100644 --- a/sphinx/pycode/parser.py +++ b/sphinx/pycode/parser.py @@ -283,13 +283,13 @@ def add_variable_comment(self, name: str, comment: str) -> None: qualname = self.get_qualname_for(name) if qualname: basename = '.'.join(qualname[:-1]) - self.comments[(basename, name)] = comment + self.comments[basename, name] = comment def add_variable_annotation(self, name: str, annotation: ast.AST) -> None: qualname = self.get_qualname_for(name) if qualname: basename = '.'.join(qualname[:-1]) - self.annotations[(basename, name)] = ast_unparse(annotation) + self.annotations[basename, name] = ast_unparse(annotation) def is_final(self, decorators: list[ast.expr]) -> bool: final = [] diff --git a/sphinx/writers/latex.py b/sphinx/writers/latex.py index 20f0cc0fdbb..cb07c0bf615 100644 --- a/sphinx/writers/latex.py +++ b/sphinx/writers/latex.py @@ -215,8 +215,8 @@ def add_cell(self, height: int, width: int) -> None: self.cell_id += 1 for col in range(width): for row in range(height): - assert self.cells[(self.row + row, self.col + col)] == 0 - self.cells[(self.row + row, self.col + col)] = self.cell_id + assert self.cells[self.row + row, self.col + col] == 0 + self.cells[self.row + row, self.col + col] = self.cell_id def cell( self, @@ -242,25 +242,25 @@ class TableCell: """Data of a cell in a table.""" def __init__(self, table: Table, row: int, col: int) -> None: - if table.cells[(row, col)] == 0: + if table.cells[row, col] == 0: raise IndexError self.table = table - self.cell_id = table.cells[(row, col)] + self.cell_id = table.cells[row, col] self.row = row self.col = col # adjust position for multirow/multicol cell - while table.cells[(self.row - 1, self.col)] == self.cell_id: + while table.cells[self.row - 1, self.col] == self.cell_id: self.row -= 1 - while table.cells[(self.row, self.col - 1)] == self.cell_id: + while table.cells[self.row, self.col - 1] == self.cell_id: self.col -= 1 @property def width(self) -> int: """Returns the cell width.""" width = 0 - while self.table.cells[(self.row, self.col + width)] == self.cell_id: + while self.table.cells[self.row, self.col + width] == self.cell_id: width += 1 return width @@ -268,7 +268,7 @@ def width(self) -> int: def height(self) -> int: """Returns the cell height.""" height = 0 - while self.table.cells[(self.row + height, self.col)] == self.cell_id: + while self.table.cells[self.row + height, self.col] == self.cell_id: height += 1 return height diff --git a/tests/test_domains/test_domain_std.py b/tests/test_domains/test_domain_std.py index 993fa197af8..eb01c8e7093 100644 --- a/tests/test_domains/test_domain_std.py +++ b/tests/test_domains/test_domain_std.py @@ -414,7 +414,7 @@ def test_cmdoption(app): entries=[('pair', 'ls command line option; -l', 'cmdoption-ls-l', '', None)], ) assert ('ls', '-l') in domain.progoptions - assert domain.progoptions[('ls', '-l')] == ('index', 'cmdoption-ls-l') + assert domain.progoptions['ls', '-l'] == ('index', 'cmdoption-ls-l') @pytest.mark.sphinx('html', testroot='root') @@ -441,7 +441,7 @@ def test_cmdoption_for_None(app): entries=[('pair', 'command line option; -l', 'cmdoption-l', '', None)], ) assert (None, '-l') in domain.progoptions - assert domain.progoptions[(None, '-l')] == ('index', 'cmdoption-l') + assert domain.progoptions[None, '-l'] == ('index', 'cmdoption-l') @pytest.mark.sphinx('html', testroot='root') @@ -481,8 +481,8 @@ def test_multiple_cmdoptions(app): ) assert ('cmd', '-o') in domain.progoptions assert ('cmd', '--output') in domain.progoptions - assert domain.progoptions[('cmd', '-o')] == ('index', 'cmdoption-cmd-o') - assert domain.progoptions[('cmd', '--output')] == ('index', 'cmdoption-cmd-o') + assert domain.progoptions['cmd', '-o'] == ('index', 'cmdoption-cmd-o') + assert domain.progoptions['cmd', '--output'] == ('index', 'cmdoption-cmd-o') @pytest.mark.sphinx('html', testroot='productionlist') diff --git a/tests/test_extensions/test_ext_doctest.py b/tests/test_extensions/test_ext_doctest.py index fc86ee47f5d..4cfbe69ddde 100644 --- a/tests/test_extensions/test_ext_doctest.py +++ b/tests/test_extensions/test_ext_doctest.py @@ -122,7 +122,7 @@ def test_skipif(app): def record(directive, part, should_skip): - recorded_calls[(directive, part, should_skip)] += 1 + recorded_calls[directive, part, should_skip] += 1 return f'Recorded {directive} {part} {should_skip}' diff --git a/tests/test_pycode/test_pycode.py b/tests/test_pycode/test_pycode.py index cc08ff62c11..2d5ec66d32f 100644 --- a/tests/test_pycode/test_pycode.py +++ b/tests/test_pycode/test_pycode.py @@ -151,15 +151,15 @@ def test_ModuleAnalyzer_find_attr_docs(): ('Foo', 'attr8'), ('Foo', 'attr9'), } - assert docs[('Foo', 'attr1')] == ['comment before attr1', ''] - assert docs[('Foo', 'attr3')] == ['attribute comment for attr3', ''] - assert docs[('Foo', 'attr4')] == ['long attribute comment', ''] - assert docs[('Foo', 'attr4')] == ['long attribute comment', ''] - assert docs[('Foo', 'attr5')] == ['attribute comment for attr5', ''] - assert docs[('Foo', 'attr6')] == ['this comment is ignored', ''] - assert docs[('Foo', 'attr7')] == ['this comment is ignored', ''] - assert docs[('Foo', 'attr8')] == ['attribute comment for attr8', ''] - assert docs[('Foo', 'attr9')] == ['string after attr9', ''] + assert docs['Foo', 'attr1'] == ['comment before attr1', ''] + assert docs['Foo', 'attr3'] == ['attribute comment for attr3', ''] + assert docs['Foo', 'attr4'] == ['long attribute comment', ''] + assert docs['Foo', 'attr4'] == ['long attribute comment', ''] + assert docs['Foo', 'attr5'] == ['attribute comment for attr5', ''] + assert docs['Foo', 'attr6'] == ['this comment is ignored', ''] + assert docs['Foo', 'attr7'] == ['this comment is ignored', ''] + assert docs['Foo', 'attr8'] == ['attribute comment for attr8', ''] + assert docs['Foo', 'attr9'] == ['string after attr9', ''] assert analyzer.tagorder == { 'Foo': 0, 'Foo.__init__': 8, @@ -189,5 +189,5 @@ def test_ModuleAnalyzer_find_attr_docs_for_posonlyargs_method(): analyzer = ModuleAnalyzer.for_string(code, 'module') docs = analyzer.find_attr_docs() assert set(docs) == {('Foo', 'attr')} - assert docs[('Foo', 'attr')] == ['attribute comment', ''] + assert docs['Foo', 'attr'] == ['attribute comment', ''] assert analyzer.tagorder == {'Foo': 0, 'Foo.__init__': 1, 'Foo.attr': 2} From e739368dcf7c81bafcc813224f81f357f1dd5b6f Mon Sep 17 00:00:00 2001 From: Adam Turner <9087854+aa-turner@users.noreply.github.com> Date: Sun, 1 Dec 2024 09:28:58 +0000 Subject: [PATCH 26/29] Update comments in .ruff.toml --- .ruff.toml | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/.ruff.toml b/.ruff.toml index 4bca08fc8a4..f9226aa43c2 100644 --- a/.ruff.toml +++ b/.ruff.toml @@ -245,11 +245,11 @@ select = [ # "RUF027", # Possible f-string without an `f` prefix "RUF028", # This suppression comment is invalid because {} "RUF029", # Function `{name}` is declared `async`, but doesn't `await` or use `async` features. - "RUF030", # `print()` expression in `assert` statement is likely unintentional - "RUF031", # Use parentheses for tuples in subscripts. + "RUF030", # `print()` call in `assert` statement is likely unintentional + "RUF031", # Use parentheses for tuples in subscripts "RUF032", # `Decimal()` called with float literal argument "RUF033", # `__post_init__` method with argument defaults - "RUF034", # Useless if-else condition + "RUF034", # Useless `if`-`else` condition # "RUF100", # Unused `noqa` directive "RUF101", # `{original}` is a redirect to `{target}` "RUF200", # Failed to parse pyproject.toml: {message} From a843532161da56deba9013d013f1878ec2fe0d54 Mon Sep 17 00:00:00 2001 From: Adam Turner <9087854+aa-turner@users.noreply.github.com> Date: Sun, 1 Dec 2024 09:29:39 +0000 Subject: [PATCH 27/29] Enable new RUF rules --- .ruff.toml | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/.ruff.toml b/.ruff.toml index f9226aa43c2..d952eb30b14 100644 --- a/.ruff.toml +++ b/.ruff.toml @@ -250,6 +250,14 @@ select = [ "RUF032", # `Decimal()` called with float literal argument "RUF033", # `__post_init__` method with argument defaults "RUF034", # Useless `if`-`else` condition + "RUF035", # Unsafe use of `{name}` detected +# "RUF036", # `None` not at the end of the type annotation. + "RUF038", # `Literal[True, False, ...]` can be replaced with `Literal[...] | bool` +# "RUF039", # First argument to {call} is not raw string + "RUF040", # Non-string literal used as assert message + "RUF041", # Unnecessary nested `Literal` +# "RUF048", # `__version__` may contain non-integral-like elements + "RUF055", # Plain string pattern passed to `re` function # "RUF100", # Unused `noqa` directive "RUF101", # `{original}` is a redirect to `{target}` "RUF200", # Failed to parse pyproject.toml: {message} From 9f0194de65b321785dee5320a1eba818d662c198 Mon Sep 17 00:00:00 2001 From: Adam Turner <9087854+aa-turner@users.noreply.github.com> Date: Sun, 1 Dec 2024 09:30:55 +0000 Subject: [PATCH 28/29] Enable the RUF036 lint in Ruff --- .ruff.toml | 4 ++-- sphinx/util/logging.py | 6 +++--- 2 files changed, 5 insertions(+), 5 deletions(-) diff --git a/.ruff.toml b/.ruff.toml index d952eb30b14..5efb7d35ac6 100644 --- a/.ruff.toml +++ b/.ruff.toml @@ -251,7 +251,7 @@ select = [ "RUF033", # `__post_init__` method with argument defaults "RUF034", # Useless `if`-`else` condition "RUF035", # Unsafe use of `{name}` detected -# "RUF036", # `None` not at the end of the type annotation. + "RUF036", # `None` not at the end of the type annotation. "RUF038", # `Literal[True, False, ...]` can be replaced with `Literal[...] | bool` # "RUF039", # First argument to {call} is not raw string "RUF040", # Non-string literal used as assert message @@ -367,7 +367,7 @@ select = [ ] # these tests need old ``typing`` generic aliases -"tests/test_util/test_util_typing.py" = ["UP006", "UP007", "UP035"] +"tests/test_util/test_util_typing.py" = ["RUF036", "UP006", "UP007", "UP035"] "tests/test_util/typing_test_data.py" = ["FA100", "I002", "PYI030", "UP006", "UP007", "UP035"] "utils/*" = [ diff --git a/sphinx/util/logging.py b/sphinx/util/logging.py index 20293c8fabe..b40f49259b5 100644 --- a/sphinx/util/logging.py +++ b/sphinx/util/logging.py @@ -161,9 +161,9 @@ def warning( # type: ignore[override] self, msg: object, *args: object, - type: None | str = None, - subtype: None | str = None, - location: None | str | tuple[str | None, int | None] | Node = None, + type: str | None = None, + subtype: str | None = None, + location: str | tuple[str | None, int | None] | Node | None = None, nonl: bool = True, color: str | None = None, once: bool = False, From df3d94ffdad09cc2592caccd179004e31aa63227 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jean-Fran=C3=A7ois=20B=2E?= <2589111+jfbu@users.noreply.github.com> Date: Mon, 2 Dec 2024 11:12:22 +0100 Subject: [PATCH 29/29] Close #13130 (LaTeX docs about index creation) (#13139) --- CHANGES.rst | 3 +++ doc/usage/configuration.rst | 3 +++ 2 files changed, 6 insertions(+) diff --git a/CHANGES.rst b/CHANGES.rst index 86e59279491..e20c1e4571e 100644 --- a/CHANGES.rst +++ b/CHANGES.rst @@ -23,6 +23,9 @@ Bugs fixed * #13060: HTML Search: use ``Map`` to store per-file term scores. Patch by James Addison +* #13130: LaTeX docs: ``pdflatex`` index creation may fail for index entries + in French. See :confval:`latex_use_xindy`. + Patch by Jean-François B. * LaTeX: fix a ``7.4.0`` typo in a default for ``\sphinxboxsetup`` (refs: PR #13152). Patch by Jean-François B. diff --git a/doc/usage/configuration.rst b/doc/usage/configuration.rst index 40fe5931250..b65b9b79f79 100644 --- a/doc/usage/configuration.rst +++ b/doc/usage/configuration.rst @@ -3157,6 +3157,9 @@ These options influence LaTeX output. * The default is :code-py:`False` for :code-py:`'pdflatex'`, but :code-py:`True` is recommended for non-English documents as soon as some indexed terms use non-ASCII characters from the language script. + Attempting to index a term whose first character is non-ASCII + will break the build, if :confval:`latex_use_xindy` is left to its + default :code-py:`False`. Sphinx adds some dedicated support to the :program:`xindy` base distribution for using :code-py:`'pdflatex'` engine with Cyrillic scripts.