{
  "instance_id": "sphinx-doc__sphinx-11445",
  "repo": "sphinx-doc/sphinx",
  "created_at": "2023-05-28T19:15:07Z",
  "problem_statement": "Using rst_prolog removes top level headings containing a domain directive\n### Describe the bug\r\n\r\nIf `rst_prolog` is set, then any documents that contain a domain directive as the first heading (eg `:mod:`) do not render the heading correctly or include the heading in the toctree.\r\n\r\nIn the example below, if the heading of `docs/mypackage.rst` were `mypackage2` instead of `:mod:mypackage2` then the heading displays correctly.\r\nSimilarly, if you do not set `rst_prolog` then the heading will display correctly.\r\n\r\nThis appears to have been broken for some time because I can reproduce it in v4.0.0 of Sphinx\r\n\r\n### How to Reproduce\r\n\r\n```bash\r\n$ sphinx-quickstart --no-sep --project mypackage --author me -v 0.1.0 --release 0.1.0 --language en docs\r\n$ echo -e 'Welcome\\n=======\\n\\n.. toctree::\\n\\n   mypackage\\n' > docs/index.rst\r\n$ echo -e ':mod:`mypackage2`\\n=================\\n\\nContent\\n\\nSubheading\\n----------\\n' > docs/mypackage.rst\r\n$ echo -e 'rst_prolog = \"\"\"\\n.. |psf| replace:: Python Software Foundation\\n\"\"\"\\n' >> docs/conf.py\r\n$ sphinx-build -b html . _build\r\n$ grep 'mypackage2' docs/_build/index.html\r\n```\r\n\r\n`docs/index.rst`:\r\n\r\n```rst\r\nWelcome\r\n=======\r\n\r\n.. toctree::\r\n\r\n   mypackage\r\n```\r\n\r\n`docs/mypackage.rst`:\r\n\r\n```rst\r\n:mod:`mypackage2`\r\n=================\r\n\r\nContent\r\n\r\nSubheading\r\n----------\r\n```\r\n\r\n### Environment Information\r\n\r\n```text\r\nPlatform:              linux; (Linux-6.3.2-arch1-1-x86_64-with-glibc2.37)\r\nPython version:        3.11.3 (main, Apr  5 2023, 15:52:25) [GCC 12.2.1 20230201])\r\nPython implementation: CPython\r\nSphinx version:        7.1.0+/d3c91f951\r\nDocutils version:      0.20.1\r\nJinja2 version:        3.1.2\r\nPygments version:      2.15.1\r\n```\r\n\r\n\r\n### Sphinx extensions\r\n\r\n```python\r\n[]\r\n```\r\n\r\n\r\n### Additional context\r\n\r\n_No response_\n",
  "patch": "diff --git a/sphinx/util/rst.py b/sphinx/util/rst.py\n--- a/sphinx/util/rst.py\n+++ b/sphinx/util/rst.py\n@@ -10,22 +10,17 @@\n \n from docutils.parsers.rst import roles\n from docutils.parsers.rst.languages import en as english\n+from docutils.parsers.rst.states import Body\n from docutils.statemachine import StringList\n from docutils.utils import Reporter\n-from jinja2 import Environment\n+from jinja2 import Environment, pass_environment\n \n from sphinx.locale import __\n from sphinx.util import docutils, logging\n \n-try:\n-    from jinja2.utils import pass_environment\n-except ImportError:\n-    from jinja2 import environmentfilter as pass_environment\n-\n-\n logger = logging.getLogger(__name__)\n \n-docinfo_re = re.compile(':\\\\w+:.*?')\n+FIELD_NAME_RE = re.compile(Body.patterns['field_marker'])\n symbols_re = re.compile(r'([!-\\-/:-@\\[-`{-~])')  # symbols without dot(0x2e)\n SECTIONING_CHARS = ['=', '-', '~']\n \n@@ -80,7 +75,7 @@ def prepend_prolog(content: StringList, prolog: str) -> None:\n     if prolog:\n         pos = 0\n         for line in content:\n-            if docinfo_re.match(line):\n+            if FIELD_NAME_RE.match(line):\n                 pos += 1\n             else:\n                 break\n@@ -91,6 +86,7 @@ def prepend_prolog(content: StringList, prolog: str) -> None:\n             pos += 1\n \n         # insert prolog (after docinfo if exists)\n+        lineno = 0\n         for lineno, line in enumerate(prolog.splitlines()):\n             content.insert(pos + lineno, line, '<rst_prolog>', lineno)\n \n",
  "similar_bug_items": [
    {
      "pr_number": 10539,
      "pr_title": "Fix inherited attribute docstrings",
      "pr_body": "Fixes #10538\r\n\r\nThis respects the value of `autodoc_inherit_docstrings` in the autodoc importer.\r\n\r\ncc: @oscarbenjamin -- please can you test if this fixes the issue?\r\n\r\n### Feature or Bugfix\r\n- Bugfix\r\n\r\nA",
      "issue_id": 10538,
      "issue_title": "Class attribute in subclass inherits attribute docstring in Sphinx 5.0.1",
      "issue_body": "### Describe the bug\n\nAs of Sphinx 5.0.1 the SymPy docs build gives lots of warnings like this:\r\n```\r\ndocstring of sympy.polys.domains.field.Field.is_Field:14: WARNING: more than one target found for cross-reference 'is_Ring': sympy.polys.domains.domain.Domain.is_Ring, sympy.polys.domains.ring.Ring.is_Ring\r\n```\r\nFull output in CI can be seen here:\r\nhttps://github.com/sympy/sympy/runs/6784391829?check_suite_focus=true\r\n\r\nThe problem seems to be that there is a class Domain which has a documented attribute `is_Ring`:\r\nhttps://github.com/sympy/sympy/blob/a96dce8b55675a0529d9bed479e2569126e27149/sympy/polys/domains/domain.py#L245-L262\r\nYou can see that in the docs here:\r\nhttps://docs.sympy.org/dev/modules/polys/domainsref.html#sympy.polys.domains.domain.Domain.is_Ring\r\n\r\nThen there is a class Ring which is a subclass of Domain and changes the value of the class attribute but does not document it:\r\nhttps://github.com/sympy/sympy/blob/a96dce8b55675a0529d9bed479e2569126e27149/sympy/polys/domains/ring.py#L13\r\n\r\nIn previous versions of Sphinx the Ring class would not show the `is_Ring` attribute:\r\nhttps://docs.sympy.org/dev/modules/polys/domainsref.html#sympy.polys.domains.ring.Ring\r\n\r\nNow with Sphinx 5.0.1 though it seems that the Sphinx build decides to document the `Ring.is_Ring` attribute with the docstring inherited from the superclass Domain. Then the see also references to `is_Ring` fail due to multiple targets.\n\n### How to Reproduce\n\n```\r\n$ git clone https://github.com/sympy/sympy.git\r\n$ cd sympy/doc\r\n$ pip install -r requirements.txt\r\n$ make html\r\n$ # open _build/html/modules/polys/domainsref.html\r\n```\r\n\n\n### Expected behavior\n\nThe attribute `is_Ring` in the subclass `Ring` would not be documented and there would be only one target for `is_Ring` pointing to the `Domain.is_Ring` attribute.\n\n### Your project\n\nhttps://github.com/sympy/sympy\n\n### Screenshots\n\n_No response_\n\n### OS\n\nAny\n\n### Python version\n\n3.8-3.10\n\n### Sphinx version\n\n5.0.1\n\n### Sphinx extensions\n\nSphinx                        5.0.1 sphinx-autobuild              2021.3.14 sphinx-basic-ng               0.0.1a11 sphinx-copybutton             0.5.0 sphinx-math-dollar            1.2.1 sphinx-press-theme            0.8.0 sphinx-reredirects            0.0.1 sphinxcontrib-applehelp       1.0.2 sphinxcontrib-devhelp         1.0.2 sphinxcontrib-htmlhelp        2.0.0 sphinxcontrib-jsmath          1.0.1 sphinxcontrib-qthelp          1.0.3 sphinxcontrib-serializinghtml 1.1.5\n\n### Extra tools\n\n_No response_\n\n### Additional context\n\n_No response_",
      "issue_closed_at": "2022-06-13T18:06:09Z",
      "base_commit": "60775ec4c4ea08509eee4b564cbf90f316021aff",
      "changes": [
        {
          "file": "sphinx/ext/autodoc/__init__.py",
          "type": "function",
          "name": "add_directive_header",
          "class_name": "PropertyDocumenter",
          "code": "def add_directive_header(self, sig: str) -> None:\n        super().add_directive_header(sig)\n        sourcename = self.get_sourcename()\n        if inspect.isabstractmethod(self.object):\n            self.add_line('   :abstractmethod:', sourcename)\n        if self.isclassmethod:\n            self.add_line('   :classmethod:', sourcename)\n\n        if safe_getattr(self.object, 'fget', None):  # property\n            func = self.object.fget\n        elif safe_getattr(self.object, 'func', None):  # cached_property\n            func = self.object.func\n        else:\n            func = None\n\n        if func and self.config.autodoc_typehints != 'none':\n            try:\n                signature = inspect.signature(func,\n                                              type_aliases=self.config.autodoc_type_aliases)\n                if signature.return_annotation is not Parameter.empty:\n                    if self.config.autodoc_typehints_format == \"short\":\n                        objrepr = stringify_typehint(signature.return_annotation, \"smart\")\n                    else:\n                        objrepr = stringify_typehint(signature.return_annotation)\n                    self.add_line('   :type: ' + objrepr, sourcename)\n            except TypeError as exc:\n                logger.warning(__(\"Failed to get a function signature for %s: %s\"),\n                               self.fullname, exc)\n                return None\n            except ValueError:\n                return None"
        },
        {
          "file": "sphinx/ext/autodoc/importer.py",
          "type": "function",
          "name": "get_object_members",
          "class_name": null,
          "code": "def get_object_members(subject: Any, objpath: List[str], attrgetter: Callable,\n                       analyzer: ModuleAnalyzer = None) -> Dict[str, Attribute]:\n    \"\"\"Get members and attributes of target object.\"\"\"\n    from sphinx.ext.autodoc import INSTANCEATTR\n\n    # the members directly defined in the class\n    obj_dict = attrgetter(subject, '__dict__', {})\n\n    members: Dict[str, Attribute] = {}\n\n    # enum members\n    if isenumclass(subject):\n        for name, value in subject.__members__.items():\n            if name not in members:\n                members[name] = Attribute(name, True, value)\n\n        superclass = subject.__mro__[1]\n        for name in obj_dict:\n            if name not in superclass.__dict__:\n                value = safe_getattr(subject, name)\n                members[name] = Attribute(name, True, value)\n\n    # members in __slots__\n    try:\n        __slots__ = getslots(subject)\n        if __slots__:\n            from sphinx.ext.autodoc import SLOTSATTR\n\n            for name in __slots__:\n                members[name] = Attribute(name, True, SLOTSATTR)\n    except (TypeError, ValueError):\n        pass\n\n    # other members\n    for name in dir(subject):\n        try:\n            value = attrgetter(subject, name)\n            directly_defined = name in obj_dict\n            name = unmangle(subject, name)\n            if name and name not in members:\n                members[name] = Attribute(name, directly_defined, value)\n        except AttributeError:\n            continue\n\n    # annotation only member (ex. attr: int)\n    for i, cls in enumerate(getmro(subject)):\n        for name in getannotations(cls):\n            name = unmangle(cls, name)\n            if name and name not in members:\n                members[name] = Attribute(name, i == 0, INSTANCEATTR)\n\n    if analyzer:\n        # append instance attributes (cf. self.attr1) if analyzer knows\n        namespace = '.'.join(objpath)\n        for (ns, name) in analyzer.find_attr_docs():\n            if namespace == ns and name not in members:\n                members[name] = Attribute(name, True, INSTANCEATTR)\n\n    return members"
        },
        {
          "file": "sphinx/ext/autodoc/importer.py",
          "type": "function",
          "name": "get_class_members",
          "class_name": null,
          "code": "def get_class_members(subject: Any, objpath: List[str], attrgetter: Callable\n                      ) -> Dict[str, \"ObjectMember\"]:\n    \"\"\"Get members and attributes of target class.\"\"\"\n    from sphinx.ext.autodoc import INSTANCEATTR, ObjectMember\n\n    # the members directly defined in the class\n    obj_dict = attrgetter(subject, '__dict__', {})\n\n    members: Dict[str, ObjectMember] = {}\n\n    # enum members\n    if isenumclass(subject):\n        for name, value in subject.__members__.items():\n            if name not in members:\n                members[name] = ObjectMember(name, value, class_=subject)\n\n        superclass = subject.__mro__[1]\n        for name in obj_dict:\n            if name not in superclass.__dict__:\n                value = safe_getattr(subject, name)\n                members[name] = ObjectMember(name, value, class_=subject)\n\n    # members in __slots__\n    try:\n        __slots__ = getslots(subject)\n        if __slots__:\n            from sphinx.ext.autodoc import SLOTSATTR\n\n            for name, docstring in __slots__.items():\n                members[name] = ObjectMember(name, SLOTSATTR, class_=subject,\n                                             docstring=docstring)\n    except (TypeError, ValueError):\n        pass\n\n    # other members\n    for name in dir(subject):\n        try:\n            value = attrgetter(subject, name)\n            if ismock(value):\n                value = undecorate(value)\n\n            unmangled = unmangle(subject, name)\n            if unmangled and unmangled not in members:\n                if name in obj_dict:\n                    members[unmangled] = ObjectMember(unmangled, value, class_=subject)\n                else:\n                    members[unmangled] = ObjectMember(unmangled, value)\n        except AttributeError:\n            continue\n\n    try:\n        for cls in getmro(subject):\n            try:\n                modname = safe_getattr(cls, '__module__')\n                qualname = safe_getattr(cls, '__qualname__')\n                analyzer = ModuleAnalyzer.for_module(modname)\n                analyzer.analyze()\n            except AttributeError:\n                qualname = None\n                analyzer = None\n            except PycodeError:\n                analyzer = None\n\n            # annotation only member (ex. attr: int)\n            for name in getannotations(cls):\n                name = unmangle(cls, name)\n                if name and name not in members:\n                    if analyzer and (qualname, name) in analyzer.attr_docs:\n                        docstring = '\\n'.join(analyzer.attr_docs[qualname, name])\n                    else:\n                        docstring = None\n\n                    members[name] = ObjectMember(name, INSTANCEATTR, class_=cls,\n                                                 docstring=docstring)\n\n            # append or complete instance attributes (cf. self.attr1) if analyzer knows\n            if analyzer:\n                for (ns, name), docstring in analyzer.attr_docs.items():\n                    if ns == qualname and name not in members:\n                        # otherwise unknown instance attribute\n                        members[name] = ObjectMember(name, INSTANCEATTR, class_=cls,\n                                                     docstring='\\n'.join(docstring))\n                    elif (ns == qualname and docstring and\n                          isinstance(members[name], ObjectMember) and\n                          not members[name].docstring):\n                        # attribute is already known, because dir(subject) enumerates it.\n                        # But it has no docstring yet\n                        members[name].docstring = '\\n'.join(docstring)\n    except AttributeError:\n        pass\n\n    return members"
        }
      ]
    },
    {
      "pr_number": 5174,
      "pr_title": "Fix #5164: incremental build has broken with external source_parser",
      "pr_body": "### Feature or Bugfix\r\n- Bugfix\r\n\r\n### Purpose\r\n- refs: #5164 \r\n",
      "issue_id": 5164,
      "issue_title": "All sources are re-built, even if nothing has changed (using nbsphinx extension)",
      "issue_body": "This problem was introduced in PR #4966. Before that, everything was working fine.\r\n\r\nThis only happens when the `nbsphinx` extension is used (but maybe other extensions are affected, too?).\r\n\r\n#### Procedure to reproduce the problem\r\n\r\nThis can be reproduced by building the docs for the `nbsphinx` module.\r\nInstallation instructions: https://github.com/spatialaudio/nbsphinx/blob/master/CONTRIBUTING.rst\r\n\r\nIn short:\r\n\r\n```\r\ngit clone https://github.com/spatialaudio/nbsphinx.git\r\ncd nbsphinx\r\npython3 -m pip install -e . --user\r\npython3 setup.py build_sphinx\r\n```\r\n\r\nAll source files (including Jupyter notebooks) are built at this point.\r\n\r\nIf I re-build immediately after that (without touching any files) ...\r\n\r\n```\r\npython3 setup.py build_sphinx\r\n```\r\n\r\n... all source files are built again.\r\n\r\n#### Expected results\r\n\r\nThe second time the source files should not be processed again.\r\nInstead, there should be this message:\r\n\r\n```\r\nbuilding [mo]: targets for 0 po files that are out of date\r\nbuilding [html]: targets for 0 source files that are out of date\r\nupdating environment: 0 added, 0 changed, 0 removed\r\nlooking for now-outdated files... none found\r\nno targets are out of date.\r\n```\r\n\r\n### Reproducible project / your project\r\n- https://github.com/spatialaudio/nbsphinx\r\n\r\n### Environment info\r\n- OS: Linux\r\n- Python version: Python 3.6.6rc1+\r\n- Sphinx version: `master` branch, every commit after PR #4966 (commit 3ffde92c54c6e9d594b2a41a365a90dc0dcbac6c)",
      "issue_closed_at": "2018-07-17T14:07:25Z",
      "base_commit": "6c12dba968a71e35e390a057a5d13c53cc26c8ce",
      "changes": [
        {
          "file": "sphinx/registry.py",
          "type": "line",
          "name": "line 39",
          "code": "    from docutils.transforms import Transform  # NOQA\n    from sphinx.application import Sphinx  # NOQA\n    from sphinx.builders import Builder  # NOQA\n    from sphinx.domains import Domain, Index  # NOQA\n    from sphinx.environment import BuildEnvironment  # NOQA\n    from sphinx.ext.autodoc import Documenter  # NOQA"
        },
        {
          "file": "sphinx/registry.py",
          "type": "function",
          "name": "add_source_suffix",
          "class_name": "SphinxComponentRegistry",
          "code": "def add_source_suffix(self, suffix, filetype, override=False):\n        # type: (unicode, unicode, bool) -> None\n        logger.debug('[app] adding source_suffix: %r, %r', suffix, filetype)\n        if suffix in self.source_suffix and not override:\n            raise ExtensionError(__('source_parser for %r is already registered') % suffix)\n        else:\n            self.source_suffix[suffix] = filetype"
        },
        {
          "file": "sphinx/registry.py",
          "type": "function",
          "name": "add_source_parser",
          "class_name": "SphinxComponentRegistry",
          "code": "def add_source_parser(self, *args, **kwargs):\n        # type: (Any, bool) -> None\n        logger.debug('[app] adding search source_parser: %r', args)\n        if len(args) == 1:\n            # new sytle arguments: (source_parser)\n            suffix = None       # type: unicode\n            parser = args[0]    # type: Type[Parser]\n        else:\n            # old style arguments: (suffix, source_parser)\n            warnings.warn('app.add_source_parser() does not support suffix argument. '\n                          'Use app.add_source_suffix() instead.',\n                          RemovedInSphinx30Warning)\n            suffix = args[0]\n            parser = args[1]\n\n        if suffix:\n            self.add_source_suffix(suffix, suffix)\n\n        if len(parser.supported) == 0:\n            warnings.warn('Old source_parser has been detected. Please fill Parser.supported '\n                          'attribute: %s' % parser.__name__,\n                          RemovedInSphinx30Warning)\n\n        # create a map from filetype to parser\n        for filetype in parser.supported:\n            if filetype in self.source_parsers and not kwargs.get('override'):\n                raise ExtensionError(__('source_parser for %r is already registered') %\n                                     filetype)\n            else:\n                self.source_parsers[filetype] = parser\n\n        # also maps suffix to parser\n        #\n        # This rescues old styled parsers which does not have ``supported`` filetypes.\n        if suffix:\n            self.source_parsers[suffix] = parser"
        },
        {
          "file": "sphinx/registry.py",
          "type": "function",
          "name": "get_envversion",
          "class_name": "SphinxComponentRegistry",
          "code": "def get_envversion(self, app):\n        # type: (Sphinx) -> Dict[unicode, unicode]\n        from sphinx.environment import ENV_VERSION\n        envversion = {ext.name: ext.metadata['env_version'] for ext in app.extensions.values()\n                      if ext.metadata.get('env_version')}\n        envversion['sphinx'] = ENV_VERSION\n        return envversion"
        },
        {
          "file": "sphinx/registry.py",
          "type": "function",
          "name": "merge_source_suffix",
          "class_name": null,
          "code": "def merge_source_suffix(app):\n    # type: (Sphinx) -> None\n    \"\"\"Merge source_suffix which specified by user and added by extensions.\"\"\"\n    for suffix, filetype in iteritems(app.registry.source_suffix):\n        if suffix not in app.config.source_suffix:\n            app.config.source_suffix[suffix] = filetype\n        elif app.config.source_suffix[suffix] is None:\n            # filetype is not specified (default filetype).\n            # So it overrides default filetype by extensions setting.\n            app.config.source_suffix[suffix] = filetype\n\n    # copy config.source_suffix to registry\n    app.registry.source_suffix = app.config.source_suffix"
        }
      ]
    },
    {
      "pr_number": 9699,
      "pr_title": "Close #9683: Revert the removal of ``add_stylesheet()`` API",
      "pr_body": "### Feature or Bugfix\r\n- Feature\r\n\r\n### Purpose\r\n- It will be kept until the Sphinx-6.0 release.\r\n- Note: Now it emits a warning instead of DeprecationWarning to let the\r\nusers know the deprecation.\r\n- refs: #9683 ",
      "issue_id": 9683,
      "issue_title": "app.add_stylesheet() is gone and replaced by app.add_css_file",
      "issue_body": "### Describe the bug\n\nThis breaks a LOT of projects. Please revert the removal.\r\n\r\nAs a package maintainer in Debian that is in charge of 500+ packages, this is not fun. There's no reason to deprecate and remove methods just because you think it's nicer. Please behave yourself. Nobody does that on the Linux kernel since its inception. It is unacceptable that breaking the world feels OK-ish to python developers, and shows a gave lack of seriousness.\n\n### How to Reproduce\n\nFor example, try to build pyroute2 with Sphinx 4.\n\n### Expected behavior\n\n_No response_\n\n### Your project\n\nDebian packaging of MANY packages\n\n### Screenshots\n\n_No response_\n\n### OS\n\ndebian\n\n### Python version\n\n3.9\n\n### Sphinx version\n\n4.2.0\n\n### Sphinx extensions\n\n_No response_\n\n### Extra tools\n\n_No response_\n\n### Additional context\n\n_No response_",
      "issue_closed_at": "2021-10-09T06:12:45Z",
      "base_commit": "f050a7775dfc9000f55d023d36d925a8d02ccfa8",
      "changes": [
        {
          "file": "sphinx/application.py",
          "type": "function",
          "name": "add_css_file",
          "class_name": "Sphinx",
          "code": "def add_css_file(self, filename: str, priority: int = 500, **kwargs: Any) -> None:\n        \"\"\"Register a stylesheet to include in the HTML output.\n\n        Add *filename* to the list of CSS files that the default HTML template\n        will include in order of *priority* (ascending).  The filename must be\n        relative to the HTML static path, or a full URI with scheme.  If the\n        priority of the CSS file is the same as others, the CSS files will be\n        included in order of registration.  The keyword arguments are also\n        accepted for attributes of ``<link>`` tag.\n\n        Example::\n\n            app.add_css_file('custom.css')\n            # => <link rel=\"stylesheet\" href=\"_static/custom.css\" type=\"text/css\" />\n\n            app.add_css_file('print.css', media='print')\n            # => <link rel=\"stylesheet\" href=\"_static/print.css\"\n            #          type=\"text/css\" media=\"print\" />\n\n            app.add_css_file('fancy.css', rel='alternate stylesheet', title='fancy')\n            # => <link rel=\"alternate stylesheet\" href=\"_static/fancy.css\"\n            #          type=\"text/css\" title=\"fancy\" />\n\n        .. list-table:: priority range for CSS files\n           :widths: 20,80\n\n           * - Priority\n             - Main purpose in Sphinx\n           * - 200\n             - default priority for built-in CSS files\n           * - 500\n             - default priority for extensions\n           * - 800\n             - default priority for :confval:`html_css_files`\n\n        A CSS file can be added to the specific HTML page when an extension calls\n        this method on :event:`html-page-context` event.\n\n        .. versionadded:: 1.0\n\n        .. versionchanged:: 1.6\n           Optional ``alternate`` and/or ``title`` attributes can be supplied\n           with the arguments *alternate* (a Boolean) and *title* (a string).\n           The default is no title and *alternate* = ``False``. For\n           more information, refer to the `documentation\n           <https://mdn.io/Web/CSS/Alternative_style_sheets>`__.\n\n        .. versionchanged:: 1.8\n           Renamed from ``app.add_stylesheet()``.\n           And it allows keyword arguments as attributes of link tag.\n\n        .. versionchanged:: 3.5\n           Take priority argument.  Allow to add a CSS file to the specific page.\n        \"\"\"\n        logger.debug('[app] adding stylesheet: %r', filename)\n        self.registry.add_css_files(filename, priority=priority, **kwargs)\n        if hasattr(self.builder, 'add_css_file'):\n            self.builder.add_css_file(filename, priority=priority, **kwargs)"
        }
      ]
    },
    {
      "pr_number": 6592,
      "pr_title": "Fix #6589: autodoc: Formatting issues with autodoc_typehints='none'",
      "pr_body": "### Feature or Bugfix\r\n- Bugfix\r\n\r\n### Purpose\r\n- refs: #6589 ",
      "issue_id": 6589,
      "issue_title": "Formatting issues with autodoc_typehints='none'",
      "issue_body": "**Describe the bug**\r\n\r\nWhen using `autodoc_typehints='none'`, I see two issues currently:\r\n\r\n1. When an annotated parameter has a default value, spaces are inserted around `=` where they are not for unannotated parameters. Consider:\r\n\r\n   ```python\r\n   def foo(x=True, y: bool = True, z = True):\r\n       return x and y and z\r\n   ```\r\n\r\n   The result looks like:\r\n\r\n   **foo**(*x=True, y = True, z=True*)\r\n\r\n2. Return types are not removed. Consider:\r\n\r\n   ```python\r\n   def bar(x: int) -> int:\r\n       return x * 2\r\n   ```\r\n\r\n   The result looks like:\r\n\r\n   **bar**(*x*) -> int\r\n\r\n**To Reproduce**\r\nSteps to reproduce the behavior:\r\n\r\nUse the above examples with the `autodoc_typehints='none'` option, or with the attached project (see below).\r\n\r\n**Expected behavior**\r\n\r\nI expect the annotated and unannotated parameters to be rendered similarly and for return type annotations to be removed as well, as if they were not there originally.\r\n\r\n**Your project**\r\n\r\nSee attached [foobar.zip](https://github.com/sphinx-doc/sphinx/files/3406054/foobar.zip)\r\n\r\n**Environment info**\r\n- OS: [Pop!_OS 18.10]\r\n- Python version: 3.6.8\r\n- Sphinx version: 2.1.2\r\n- Sphinx extensions:  [sphinx.ext.autodoc]\r\n\r\n**Additional context**\r\n\r\n- #6361 \r\n- #5868 \r\n- https://github.com/agronholm/sphinx-autodoc-typehints/pull/78\r\n\r\n",
      "issue_closed_at": "2019-08-02T13:37:16Z",
      "base_commit": "4732ec5edf9e53e2fa78cd5e1ff6bee92f1b27b7",
      "changes": [
        {
          "file": "sphinx/util/inspect.py",
          "type": "function",
          "name": "format_args",
          "class_name": "Signature",
          "code": "def format_args(self, show_annotation: bool = True) -> str:\n        args = []\n        last_kind = None\n        for i, param in enumerate(self.parameters.values()):\n            # skip first argument if subject is bound method\n            if self.skip_first_argument and i == 0:\n                continue\n\n            arg = StringIO()\n\n            # insert '*' between POSITIONAL args and KEYWORD_ONLY args::\n            #     func(a, b, *, c, d):\n            if param.kind == param.KEYWORD_ONLY and last_kind in (param.POSITIONAL_OR_KEYWORD,\n                                                                  param.POSITIONAL_ONLY,\n                                                                  None):\n                args.append('*')\n\n            if param.kind in (param.POSITIONAL_ONLY,\n                              param.POSITIONAL_OR_KEYWORD,\n                              param.KEYWORD_ONLY):\n                arg.write(param.name)\n                if show_annotation and param.annotation is not param.empty:\n                    if isinstance(param.annotation, str) and param.name in self.annotations:\n                        arg.write(': ')\n                        arg.write(self.format_annotation(self.annotations[param.name]))\n                    else:\n                        arg.write(': ')\n                        arg.write(self.format_annotation(param.annotation))\n                if param.default is not param.empty:\n                    if param.annotation is param.empty:\n                        arg.write('=')\n                        arg.write(object_description(param.default))\n                    else:\n                        arg.write(' = ')\n                        arg.write(object_description(param.default))\n            elif param.kind == param.VAR_POSITIONAL:\n                arg.write('*')\n                arg.write(param.name)\n            elif param.kind == param.VAR_KEYWORD:\n                arg.write('**')\n                arg.write(param.name)\n\n            args.append(arg.getvalue())\n            last_kind = param.kind\n\n        if self.return_annotation is inspect.Parameter.empty:\n            return '(%s)' % ', '.join(args)\n        else:\n            if 'return' in self.annotations:\n                annotation = self.format_annotation(self.annotations['return'])\n            else:\n                annotation = self.format_annotation(self.return_annotation)\n\n            return '(%s) -> %s' % (', '.join(args), annotation)"
        },
        {
          "file": "sphinx/util/inspect.py",
          "type": "function",
          "name": "format_args",
          "class_name": "Signature",
          "code": "def format_args(self, show_annotation: bool = True) -> str:\n        args = []\n        last_kind = None\n        for i, param in enumerate(self.parameters.values()):\n            # skip first argument if subject is bound method\n            if self.skip_first_argument and i == 0:\n                continue\n\n            arg = StringIO()\n\n            # insert '*' between POSITIONAL args and KEYWORD_ONLY args::\n            #     func(a, b, *, c, d):\n            if param.kind == param.KEYWORD_ONLY and last_kind in (param.POSITIONAL_OR_KEYWORD,\n                                                                  param.POSITIONAL_ONLY,\n                                                                  None):\n                args.append('*')\n\n            if param.kind in (param.POSITIONAL_ONLY,\n                              param.POSITIONAL_OR_KEYWORD,\n                              param.KEYWORD_ONLY):\n                arg.write(param.name)\n                if show_annotation and param.annotation is not param.empty:\n                    if isinstance(param.annotation, str) and param.name in self.annotations:\n                        arg.write(': ')\n                        arg.write(self.format_annotation(self.annotations[param.name]))\n                    else:\n                        arg.write(': ')\n                        arg.write(self.format_annotation(param.annotation))\n                if param.default is not param.empty:\n                    if param.annotation is param.empty:\n                        arg.write('=')\n                        arg.write(object_description(param.default))\n                    else:\n                        arg.write(' = ')\n                        arg.write(object_description(param.default))\n            elif param.kind == param.VAR_POSITIONAL:\n                arg.write('*')\n                arg.write(param.name)\n            elif param.kind == param.VAR_KEYWORD:\n                arg.write('**')\n                arg.write(param.name)\n\n            args.append(arg.getvalue())\n            last_kind = param.kind\n\n        if self.return_annotation is inspect.Parameter.empty:\n            return '(%s)' % ', '.join(args)\n        else:\n            if 'return' in self.annotations:\n                annotation = self.format_annotation(self.annotations['return'])\n            else:\n                annotation = self.format_annotation(self.return_annotation)\n\n            return '(%s) -> %s' % (', '.join(args), annotation)"
        }
      ]
    },
    {
      "pr_number": 7444,
      "pr_title": "C++, fix merging overloaded functions in parallel builds.",
      "pr_body": "### Feature or Bugfix\r\n- Bugfix\r\n\r\n### Detail\r\nFixes #7438.\r\n\r\nTo reproduce, have sufficiently many rst files, where one of them contains an overloaded function where a pending xref will be made. E.g.,\r\n```rst\r\n .. cpp:function:: std::string f(int)\r\n .. cpp:function:: std::string f(double)\r\n```\r\n",
      "issue_id": 7438,
      "issue_title": "Version 3.0 breaks builds on certain applications",
      "issue_body": "**Describe the bug**\r\nI built [Botan](https://github.com/randombit/botan/) on Mageia Cauldron x86_64 and the build failed when using Sphix to build the documentation. Using Doxygen worked fine.\r\n\r\nYou can [see the full build log](https://github.com/randombit/botan/issues/2324) and my bug report over at Botan.\r\n\r\n**To Reproduce**\r\nSteps to reproduce the behavior: Build Botan using `--with-sphinx` and see that it fails.\r\n```\r\n$ git clone https://github.com/randombit/botan\r\n$ cd botan\r\n$ ./configure.py --with-sphinx\r\n$ make\r\n```\r\n\r\n**Expected behavior**\r\nI expect Sphinx to build the docs successfully.\r\n\r\n**Environment info**\r\n- OS: Mageia Cauldron x86_64\r\n- Python version: 3.8.2\r\n- Sphinx version: 3.0.0\r\n- Sphinx extensions:  N/A\r\n- Extra tools: N/A\r\n\r\n**Additional context**\r\nFull build log: https://raw.githubusercontent.com/kekePower/mmbl/6ddfe29086def989b9867c1893548acd716a5939/2020/04/07/20%3A07%3A51/botan2-2.14.0-1.mga8.src.rpm/build.0.20200407175517.log\r\n\r\n",
      "issue_closed_at": "2020-04-09T08:05:43Z",
      "base_commit": "4caa7d7c379025052da8774a648dccf29426d5f0",
      "changes": [
        {
          "file": "sphinx/domains/cpp.py",
          "type": "function",
          "name": "merge_with",
          "class_name": "Symbol",
          "code": "def merge_with(self, other: \"Symbol\", docnames: List[str],\n                   env: \"BuildEnvironment\") -> None:\n        if Symbol.debug_lookup:\n            Symbol.debug_indent += 1\n            Symbol.debug_print(\"merge_with:\")\n        assert other is not None\n        for otherChild in other._children:\n            ourChild = self._find_first_named_symbol(\n                identOrOp=otherChild.identOrOp,\n                templateParams=otherChild.templateParams,\n                templateArgs=otherChild.templateArgs,\n                templateShorthand=False, matchSelf=False,\n                recurseInAnon=False, correctPrimaryTemplateArgs=False)\n            if ourChild is None:\n                # TODO: hmm, should we prune by docnames?\n                self._children.append(otherChild)\n                otherChild.parent = self\n                otherChild._assert_invariants()\n                continue\n            if otherChild.declaration and otherChild.docname in docnames:\n                if not ourChild.declaration:\n                    ourChild._fill_empty(otherChild.declaration, otherChild.docname)\n                elif ourChild.docname != otherChild.docname:\n                    name = str(ourChild.declaration)\n                    msg = __(\"Duplicate declaration, also defined in '%s'.\\n\"\n                             \"Declaration is '%s'.\")\n                    msg = msg % (ourChild.docname, name)\n                    logger.warning(msg, location=otherChild.docname)\n                else:\n                    # Both have declarations, and in the same docname.\n                    # This can apparently happen, it should be safe to\n                    # just ignore it, right?\n                    pass\n            ourChild.merge_with(otherChild, docnames, env)\n        if Symbol.debug_lookup:\n            Symbol.debug_indent -= 1"
        },
        {
          "file": "sphinx/domains/cpp.py",
          "type": "function",
          "name": "merge_with",
          "class_name": "Symbol",
          "code": "def merge_with(self, other: \"Symbol\", docnames: List[str],\n                   env: \"BuildEnvironment\") -> None:\n        if Symbol.debug_lookup:\n            Symbol.debug_indent += 1\n            Symbol.debug_print(\"merge_with:\")\n        assert other is not None\n        for otherChild in other._children:\n            ourChild = self._find_first_named_symbol(\n                identOrOp=otherChild.identOrOp,\n                templateParams=otherChild.templateParams,\n                templateArgs=otherChild.templateArgs,\n                templateShorthand=False, matchSelf=False,\n                recurseInAnon=False, correctPrimaryTemplateArgs=False)\n            if ourChild is None:\n                # TODO: hmm, should we prune by docnames?\n                self._children.append(otherChild)\n                otherChild.parent = self\n                otherChild._assert_invariants()\n                continue\n            if otherChild.declaration and otherChild.docname in docnames:\n                if not ourChild.declaration:\n                    ourChild._fill_empty(otherChild.declaration, otherChild.docname)\n                elif ourChild.docname != otherChild.docname:\n                    name = str(ourChild.declaration)\n                    msg = __(\"Duplicate declaration, also defined in '%s'.\\n\"\n                             \"Declaration is '%s'.\")\n                    msg = msg % (ourChild.docname, name)\n                    logger.warning(msg, location=otherChild.docname)\n                else:\n                    # Both have declarations, and in the same docname.\n                    # This can apparently happen, it should be safe to\n                    # just ignore it, right?\n                    pass\n            ourChild.merge_with(otherChild, docnames, env)\n        if Symbol.debug_lookup:\n            Symbol.debug_indent -= 1"
        },
        {
          "file": "sphinx/domains/cpp.py",
          "type": "function",
          "name": "merge_domaindata",
          "class_name": "CPPDomain",
          "code": "def merge_domaindata(self, docnames: List[str], otherdata: Dict) -> None:\n        if Symbol.debug_show_tree:\n            print(\"merge_domaindata:\")\n            print(\"\\tself:\")\n            print(self.data['root_symbol'].dump(1))\n            print(\"\\tself end\")\n            print(\"\\tother:\")\n            print(otherdata['root_symbol'].dump(1))\n            print(\"\\tother end\")\n            print(\"merge_domaindata end\")\n\n        self.data['root_symbol'].merge_with(otherdata['root_symbol'],\n                                            docnames, self.env)\n        ourNames = self.data['names']\n        for name, docname in otherdata['names'].items():\n            if docname in docnames:\n                if name in ourNames:\n                    msg = __(\"Duplicate declaration, also defined in '%s'.\\n\"\n                             \"Name of declaration is '%s'.\")\n                    msg = msg % (ourNames[name], name)\n                    logger.warning(msg, location=docname)\n                else:\n                    ourNames[name] = docname"
        },
        {
          "file": "sphinx/domains/cpp.py",
          "type": "function",
          "name": "merge_domaindata",
          "class_name": "CPPDomain",
          "code": "def merge_domaindata(self, docnames: List[str], otherdata: Dict) -> None:\n        if Symbol.debug_show_tree:\n            print(\"merge_domaindata:\")\n            print(\"\\tself:\")\n            print(self.data['root_symbol'].dump(1))\n            print(\"\\tself end\")\n            print(\"\\tother:\")\n            print(otherdata['root_symbol'].dump(1))\n            print(\"\\tother end\")\n            print(\"merge_domaindata end\")\n\n        self.data['root_symbol'].merge_with(otherdata['root_symbol'],\n                                            docnames, self.env)\n        ourNames = self.data['names']\n        for name, docname in otherdata['names'].items():\n            if docname in docnames:\n                if name in ourNames:\n                    msg = __(\"Duplicate declaration, also defined in '%s'.\\n\"\n                             \"Name of declaration is '%s'.\")\n                    msg = msg % (ourNames[name], name)\n                    logger.warning(msg, location=docname)\n                else:\n                    ourNames[name] = docname"
        }
      ]
    }
  ]
}