{
  "instance_id": "sphinx-doc__sphinx-8474",
  "repo": "sphinx-doc/sphinx",
  "created_at": "2020-11-22T16:24:25Z",
  "problem_statement": "v3.3 upgrade started generating \"WARNING: no number is assigned for table\" warnings\nWe've updated to Sphinx 3.3 in our documentation, and suddenly the following warning started popping up in our builds when we build either `singlehtml` or `latex`.:\r\n\r\n`WARNING: no number is assigned for table:`\r\n\r\nI looked through the changelog but it didn't seem like there was anything related to `numref` that was changed, but perhaps I missed something? Could anyone point me to a change in the numref logic so I can figure out where these warnings are coming from?\n",
  "patch": "diff --git a/sphinx/domains/std.py b/sphinx/domains/std.py\n--- a/sphinx/domains/std.py\n+++ b/sphinx/domains/std.py\n@@ -852,8 +852,9 @@ def _resolve_numref_xref(self, env: \"BuildEnvironment\", fromdocname: str,\n             if fignumber is None:\n                 return contnode\n         except ValueError:\n-            logger.warning(__(\"no number is assigned for %s: %s\"), figtype, labelid,\n-                           location=node)\n+            logger.warning(__(\"Failed to create a cross reference. Any number is not \"\n+                              \"assigned: %s\"),\n+                           labelid, location=node)\n             return contnode\n \n         try:\n",
  "similar_bug_items": [
    {
      "pr_number": 8076,
      "pr_title": "Fix #6914: figure numbers are unexpectedly assigned to uncaptioned items",
      "pr_body": "### Feature or Bugfix\r\n- Bugfix\r\n\r\n### Purpose\r\n- refs: #6914 \r\n- The figure numbers should be assigned to items only having captions or\r\ntitles.  This uses `get_numfig_title()` to ensures it on assign numbers.\r\n",
      "issue_id": 6914,
      "issue_title": "References to figures without captions: errors in both HTML and LaTeX",
      "issue_body": "\r\n**Describe the bug**\r\nUsing figures without captions causes errors in both HTML (though these are properly reported when source is processed) and in LaTeX (they are not reported until LaTeX says there were undefined references).\r\n\r\nThis was the test document, compiled with sphinx 2.2.2 from pypi; `numfig=True` was added to conf.py, the project was otherwise generated with sphinx-build with no other changes. It is attached here: [sphinx-captions.zip](https://github.com/sphinx-doc/sphinx/files/3947135/sphinx-captions.zip)\r\n\r\n```\r\nWelcome to foo's documentation!\r\n===============================\r\n\r\nReferences:\r\n\r\n* figure without caption\r\n\r\n   * plain reference :ref:`fig-sample-nocaption` (error: HTML, LaTeX)\r\n   * named reference :ref:`figure without caption <fig-sample-nocaption>` (error: LaTeX)\r\n   * numbered reference :numref:`fig-sample-nocaption` (error: LaTeX)\r\n\r\n* figure with caption\r\n\r\n   * plain reference :ref:`fig-sample-caption`\r\n   * named reference :ref:`figure without caption <fig-sample-caption>`\r\n   * numbered reference :numref:`fig-sample-caption`\r\n\r\n.. _fig-sample-nocaption:\r\n.. figure:: sample.png\r\n\r\n\r\n.. _fig-sample-caption:\r\n.. figure:: sample.png\r\n   \r\n   This is some caption.\r\n```\r\n\r\nand these are the results:\r\n\r\n1. misleading warning: **index.rst:8: WARNING: undefined label: fig-sample-nocaption (if the link has no caption the label must precede a section header)**\r\n2. this is HTML output (the error highlighted corresponds to the warning mentioned above):\r\n![html output](https://user-images.githubusercontent.com/1029876/70568432-2b150c00-1b98-11ea-98ac-67e7fbc23927.png)\r\n3. this is LaTeX (pdflatex) output:\r\n```\r\nLaTeX Warning: Hyper reference `index:fig-sample-nocaption' on page 1 undefined\r\n on input line 99.\r\nLaTeX Warning: Hyper reference `index:fig-sample-nocaption' on page 1 undefined\r\n on input line 102.\r\n```\r\n![latex output](https://user-images.githubusercontent.com/1029876/70568602-7fb88700-1b98-11ea-85bd-b7b6fec93e41.png)\r\n\r\n**Expected behavior**\r\nI expect\r\n1. sphinx to produce valid LaTeX input without undefined references;\r\n2. uncaptioned figures to be referencable in LaTeX (this could be an optional setting perhaps causing uncaptioned figured to produce only \"Figure 4.1.\" caption);\r\n3. warning about figure not being captioned to be more meaningful -- I understand that non-numbered figure cannot be referenced via :ref:`label` (as the label will not resolve to any text) but the warning is not pointing to how to fix the issue.\r\n\r\n**Environment info**\r\n- OS: Ubuntu 18.04 LTS\r\n- Python version: 3.6.8\r\n- Sphinx version: 2.2.2\r\n- Sphinx extensions: none\r\n- Extra tools: pdflatex TeXLive\r\n",
      "issue_closed_at": "2020-11-03T02:29:07Z",
      "base_commit": "b4add96a53f30d0e901fa683a8d9e698240e8fe0",
      "changes": [
        {
          "file": "sphinx/environment/collectors/toctree.py",
          "type": "function",
          "name": "assign_figure_numbers",
          "class_name": "TocTreeCollector",
          "code": "def assign_figure_numbers(self, env: BuildEnvironment) -> List[str]:\n        \"\"\"Assign a figure number to each figure under a numbered toctree.\"\"\"\n\n        rewrite_needed = []\n\n        assigned = set()  # type: Set[str]\n        old_fignumbers = env.toc_fignumbers\n        env.toc_fignumbers = {}\n        fignum_counter = {}  # type: Dict[str, Dict[Tuple[int, ...], int]]\n\n        def get_figtype(node: Node) -> str:\n            for domain in env.domains.values():\n                figtype = domain.get_enumerable_node_type(node)\n                if figtype:\n                    return figtype\n\n            return None\n\n        def get_section_number(docname: str, section: nodes.section) -> Tuple[int, ...]:\n            anchorname = '#' + section['ids'][0]\n            secnumbers = env.toc_secnumbers.get(docname, {})\n            if anchorname in secnumbers:\n                secnum = secnumbers.get(anchorname)\n            else:\n                secnum = secnumbers.get('')\n\n            return secnum or tuple()\n\n        def get_next_fignumber(figtype: str, secnum: Tuple[int, ...]) -> Tuple[int, ...]:\n            counter = fignum_counter.setdefault(figtype, {})\n\n            secnum = secnum[:env.config.numfig_secnum_depth]\n            counter[secnum] = counter.get(secnum, 0) + 1\n            return secnum + (counter[secnum],)\n\n        def register_fignumber(docname: str, secnum: Tuple[int, ...],\n                               figtype: str, fignode: Element) -> None:\n            env.toc_fignumbers.setdefault(docname, {})\n            fignumbers = env.toc_fignumbers[docname].setdefault(figtype, {})\n            figure_id = fignode['ids'][0]\n\n            fignumbers[figure_id] = get_next_fignumber(figtype, secnum)\n\n        def _walk_doctree(docname: str, doctree: Element, secnum: Tuple[int, ...]) -> None:\n            for subnode in doctree.children:\n                if isinstance(subnode, nodes.section):\n                    next_secnum = get_section_number(docname, subnode)\n                    if next_secnum:\n                        _walk_doctree(docname, subnode, next_secnum)\n                    else:\n                        _walk_doctree(docname, subnode, secnum)\n                elif isinstance(subnode, addnodes.toctree):\n                    for title, subdocname in subnode['entries']:\n                        if url_re.match(subdocname) or subdocname == 'self':\n                            # don't mess with those\n                            continue\n\n                        _walk_doc(subdocname, secnum)\n                elif isinstance(subnode, nodes.Element):\n                    figtype = get_figtype(subnode)\n                    if figtype and subnode['ids']:\n                        register_fignumber(docname, secnum, figtype, subnode)\n\n                    _walk_doctree(docname, subnode, secnum)\n\n        def _walk_doc(docname: str, secnum: Tuple[int, ...]) -> None:\n            if docname not in assigned:\n                assigned.add(docname)\n                doctree = env.get_doctree(docname)\n                _walk_doctree(docname, doctree, secnum)\n\n        if env.config.numfig:\n            _walk_doc(env.config.master_doc, tuple())\n            for docname, fignums in env.toc_fignumbers.items():\n                if fignums != old_fignumbers.get(docname):\n                    rewrite_needed.append(docname)\n\n        return rewrite_needed"
        }
      ]
    },
    {
      "pr_number": 6532,
      "pr_title": "Fix #6531: Failed to load last environment object when extension added",
      "pr_body": "### Feature or Bugfix\r\n- Bugfix\r\n\r\n### Purpose\r\n- refs: https://github.com/sphinx-doc/sphinx/pull/6419/files#r295377826\r\n",
      "issue_id": 6531,
      "issue_title": "Failed to load last environment object if extension added",
      "issue_body": "**Describe the bug**\r\nFailed to load last environment object if extension added.\r\n\r\n**To Reproduce**\r\nSteps to reproduce the behavior:\r\n\r\n1. Create a new project\r\n2. Build HTML once\r\n3. Add a some extension to conf.py\r\n4. Build HTML again\r\n\r\n```\r\nRunning Sphinx v2.2.0+\r\nloading pickled environment... failed\r\nfailed: No such config value: autosummary_generate\r\nbuilding [mo]: targets for 0 po files that are out of date\r\nbuilding [html]: targets for 4 source files that are out of date\r\n...\r\n```\r\n\r\n**Expected behavior**\r\nLoading succeeded.\r\n\r\n**Your project**\r\nNone\r\n\r\n**Screenshots**\r\nNone\r\n\r\n**Environment info**\r\n- OS: Mac\r\n- Python version: 3.7.3\r\n- Sphinx version: 2.2.0 (dev)\r\n- Sphinx extensions: sphinx.ext.autosummary\r\n",
      "issue_closed_at": "2019-06-30T15:27:06Z",
      "base_commit": "05949f83474ced6f365ac528e4744a428c6da1ca",
      "changes": [
        {
          "file": "sphinx/environment/__init__.py",
          "type": "function",
          "name": "_update_config",
          "class_name": "BuildEnvironment",
          "code": "def _update_config(self, config):\n        # type: (Config) -> None\n        \"\"\"Update configurations by new one.\"\"\"\n        self.config_status = CONFIG_OK\n        if self.config is None:\n            self.config_status = CONFIG_NEW\n        else:\n            # check if a config value was changed that affects how\n            # doctrees are read\n            for item in config.filter('env'):\n                if self.config[item.name] != item.value:\n                    self.config_status = CONFIG_CHANGED\n                    break\n\n            # this value is not covered by the above loop because it is handled\n            # specially by the config class\n            if self.config.extensions != config.extensions:\n                self.config_status = CONFIG_EXTENSIONS_CHANGED\n\n        self.config = config"
        },
        {
          "file": "sphinx/environment/__init__.py",
          "type": "function",
          "name": "_update_config",
          "class_name": "BuildEnvironment",
          "code": "def _update_config(self, config):\n        # type: (Config) -> None\n        \"\"\"Update configurations by new one.\"\"\"\n        self.config_status = CONFIG_OK\n        if self.config is None:\n            self.config_status = CONFIG_NEW\n        else:\n            # check if a config value was changed that affects how\n            # doctrees are read\n            for item in config.filter('env'):\n                if self.config[item.name] != item.value:\n                    self.config_status = CONFIG_CHANGED\n                    break\n\n            # this value is not covered by the above loop because it is handled\n            # specially by the config class\n            if self.config.extensions != config.extensions:\n                self.config_status = CONFIG_EXTENSIONS_CHANGED\n\n        self.config = config"
        }
      ]
    },
    {
      "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"
        }
      ]
    },
    {
      "pr_number": 7501,
      "pr_title": "Fix #7418: std domain: term role role could not match case-insensitively",
      "pr_body": "### Feature or Bugfix\r\n- Bugfix\r\n\r\n### Purpose\r\n- refs: #7418 ",
      "issue_id": 7418,
      "issue_title": "glossary duplicate term with a different case",
      "issue_body": "**Describe the bug**\r\n```\r\nWarning, treated as error:\r\ndoc/glossary.rst:243:duplicate term description of mysql, other instance in glossary\r\n```\r\n\r\n**To Reproduce**\r\nSteps to reproduce the behavior:\r\n[.travis.yml#L168](https://github.com/phpmyadmin/phpmyadmin/blob/f7cc383674b7099190771b1db510c62bfbbf89a7/.travis.yml#L168)\r\n```\r\n$ git clone --depth 1 https://github.com/phpmyadmin/phpmyadmin.git\r\n$ cd doc\r\n$ pip install 'Sphinx'\r\n$ make html\r\n```\r\n\r\n**Expected behavior**\r\nMySQL != mysql term right ?\r\n\r\n**Your project**\r\nhttps://github.com/phpmyadmin/phpmyadmin/blame/master/doc/glossary.rst#L234\r\n\r\n\r\n**Environment info**\r\n- OS: Unix\r\n- Python version: 3.6\r\n- Sphinx version: 3.0.0\r\n\r\n**Additional context**\r\nDid occur some hours ago, maybe you just released the version\r\n\r\n- https://travis-ci.org/github/williamdes/phpmyadmintest/jobs/671352365#L328\r\n\r\n",
      "issue_closed_at": "2020-04-09T15:12:11Z",
      "base_commit": "50d2d289e150cb429de15770bdd48a723de8c45d",
      "changes": [
        {
          "file": "sphinx/domains/std.py",
          "type": "function",
          "name": "resolve_xref",
          "class_name": "StandardDomain",
          "code": "def resolve_xref(self, env: \"BuildEnvironment\", fromdocname: str, builder: \"Builder\",\n                     typ: str, target: str, node: pending_xref, contnode: Element) -> Element:\n        if typ == 'ref':\n            resolver = self._resolve_ref_xref\n        elif typ == 'numref':\n            resolver = self._resolve_numref_xref\n        elif typ == 'keyword':\n            resolver = self._resolve_keyword_xref\n        elif typ == 'doc':\n            resolver = self._resolve_doc_xref\n        elif typ == 'option':\n            resolver = self._resolve_option_xref\n        elif typ == 'citation':\n            warnings.warn('pending_xref(domain=std, type=citation) is deprecated: %r' % node,\n                          RemovedInSphinx40Warning)\n            domain = env.get_domain('citation')\n            return domain.resolve_xref(env, fromdocname, builder, typ, target, node, contnode)\n        else:\n            resolver = self._resolve_obj_xref\n\n        return resolver(env, fromdocname, builder, typ, target, node, contnode)"
        },
        {
          "file": "sphinx/domains/std.py",
          "type": "function",
          "name": "_resolve_option_xref",
          "class_name": "StandardDomain",
          "code": "def _resolve_option_xref(self, env: \"BuildEnvironment\", fromdocname: str,\n                             builder: \"Builder\", typ: str, target: str,\n                             node: pending_xref, contnode: Element) -> Element:\n        progname = node.get('std:program')\n        target = target.strip()\n        docname, labelid = self.progoptions.get((progname, target), ('', ''))\n        if not docname:\n            commands = []\n            while ws_re.search(target):\n                subcommand, target = ws_re.split(target, 1)\n                commands.append(subcommand)\n                progname = \"-\".join(commands)\n\n                docname, labelid = self.progoptions.get((progname, target), ('', ''))\n                if docname:\n                    break\n            else:\n                return None\n\n        return make_refnode(builder, fromdocname, docname,\n                            labelid, contnode)"
        }
      ]
    },
    {
      "pr_number": 6226,
      "pr_title": "C++, fix parsing of full xrefs.",
      "pr_body": "If a full xref has a short xref as prefix, e.g., ``T f()``, parsing would fail.\r\n\r\n### Relates\r\n- Fixes #6208\r\n\r\n",
      "issue_id": 6208,
      "issue_title": "Cross-referencing a function that returns a pointer with :cpp:func: causes error \"Invalid definition: Expected end of definition\"",
      "issue_body": "**Describe the bug**\r\nWhile using the :cpp:func: directive to cross-reference a C++ function that returns a pointer, you will receivean error that the * character in the reference is \"unparseable\". This makes it impossible to cross-reference a function with a pointer return value\r\n\r\n**To Reproduce**\r\nSteps to reproduce the behavior:\r\n\r\nSource (C++): \r\n```\r\nclass Foo\r\n{\r\n    public:\r\n        Foo* Bar();\r\n}   \r\n```\r\nSphinx (reStructuredText):\r\n```\r\n:cpp:func:`Foo* Foo::Bar()`\r\n```\r\n\r\nResult:\r\n```\r\nWARNING: Unpareseable C++ cross-reference: 'Foo*'\r\nInvalid definition: Expected end of definition. [error at 4]\r\n   Foo*\r\n   ---^\r\n```\r\n\r\n**Expected behavior**\r\nThe expected behavior is to output a properly cross-referenced link to the Foo* Foo::Bar() function.\r\n\r\n**Your project**\r\nN/A\r\n\r\n**Screenshots**\r\nN/A\r\n\r\n**Environment info**\r\n- OS: [e.g. Unix/Linux/Mac/Win/other with version]\r\n- Python version: 3.7.0\r\n- Sphinx version: 2.0.0+\r\n- Sphinx extensions:  breathe\r\n- Extra tools: N/A",
      "issue_closed_at": "2019-04-04T16:57:06Z",
      "base_commit": "8925358eca5fb640543d8c531427f5e50063c782",
      "changes": [
        {
          "file": "sphinx/domains/cpp.py",
          "type": "function",
          "name": "parse_xref_object",
          "class_name": "DefinitionParser",
          "code": "def parse_xref_object(self):\n        # type: () -> Tuple[Any, bool]\n        pos = self.pos\n        try:\n            templatePrefix = self._parse_template_declaration_prefix(objectType=\"xref\")\n            name = self._parse_nested_name()\n            # if there are '()' left, just skip them\n            self.skip_ws()\n            self.skip_string('()')\n            templatePrefix = self._check_template_consistency(name, templatePrefix,\n                                                              fullSpecShorthand=True)\n            res1 = ASTNamespace(name, templatePrefix)\n            res1.objectType = 'xref'  # type: ignore\n            return res1, True\n        except DefinitionError as e1:\n            try:\n                self.pos = pos\n                res2 = self.parse_declaration('function', 'function')\n                # if there are '()' left, just skip them\n                self.skip_ws()\n                self.skip_string('()')\n                return res2, False\n            except DefinitionError as e2:\n                errs = []\n                errs.append((e1, \"If shorthand ref\"))\n                errs.append((e2, \"If full function ref\"))\n                msg = \"Error in cross-reference.\"\n                raise self._make_multi_error(errs, msg)"
        },
        {
          "file": "sphinx/domains/cpp.py",
          "type": "function",
          "name": "parse_xref_object",
          "class_name": "DefinitionParser",
          "code": "def parse_xref_object(self):\n        # type: () -> Tuple[Any, bool]\n        pos = self.pos\n        try:\n            templatePrefix = self._parse_template_declaration_prefix(objectType=\"xref\")\n            name = self._parse_nested_name()\n            # if there are '()' left, just skip them\n            self.skip_ws()\n            self.skip_string('()')\n            templatePrefix = self._check_template_consistency(name, templatePrefix,\n                                                              fullSpecShorthand=True)\n            res1 = ASTNamespace(name, templatePrefix)\n            res1.objectType = 'xref'  # type: ignore\n            return res1, True\n        except DefinitionError as e1:\n            try:\n                self.pos = pos\n                res2 = self.parse_declaration('function', 'function')\n                # if there are '()' left, just skip them\n                self.skip_ws()\n                self.skip_string('()')\n                return res2, False\n            except DefinitionError as e2:\n                errs = []\n                errs.append((e1, \"If shorthand ref\"))\n                errs.append((e2, \"If full function ref\"))\n                msg = \"Error in cross-reference.\"\n                raise self._make_multi_error(errs, msg)"
        },
        {
          "file": "sphinx/domains/cpp.py",
          "type": "function",
          "name": "warn",
          "class_name": "Warner",
          "code": "def warn(self, msg):\n                if emitWarnings:\n                    logger.warning(msg, location=node)"
        },
        {
          "file": "sphinx/domains/cpp.py",
          "type": "function",
          "name": "findWarning",
          "class_name": null,
          "code": "def findWarning(e):  # as arg to stop flake8 from complaining\n                if typ != 'any' and typ != 'func':\n                    return target, e\n                # hax on top of the paren hax to try to get correct errors\n                parser2 = DefinitionParser(target[:-2], warner, env.config)\n                try:\n                    parser2.parse_xref_object()\n                    parser2.assert_end()\n                except DefinitionError as e2:\n                    return target[:-2], e2\n                # strange, that we don't get the error now, use the original\n                return target, e"
        }
      ]
    }
  ]
}