{
  "instance_id": "sphinx-doc__sphinx-8506",
  "repo": "sphinx-doc/sphinx",
  "created_at": "2020-11-28T17:28:05Z",
  "problem_statement": "Sphinx 3.2 complains about option:: syntax that earlier versions accepted\nSphinx 3.2 complains about use of the option:: directive that earlier versions accepted without complaint.\r\n\r\nThe QEMU documentation includes this:\r\n```\r\n.. option:: [enable=]PATTERN\r\n\r\n   Immediately enable events matching *PATTERN*\r\n```\r\n\r\nas part of the documentation of the command line options of one of its programs. Earlier versions of Sphinx were fine with this, but Sphinx 3.2 complains:\r\n\r\n```\r\nWarning, treated as error:\r\n../../docs/qemu-option-trace.rst.inc:4:Malformed option description '[enable=]PATTERN', should look like \"opt\", \"-opt args\", \"--opt args\", \"/opt args\" or \"+opt args\"\r\n```\r\n\r\nSphinx ideally shouldn't change in ways that break the building of documentation that worked in older versions, because this makes it unworkably difficult to have documentation that builds with whatever the Linux distro's sphinx-build is.\r\n\r\nThe error message suggests that Sphinx has a very restrictive idea of what option syntax is; it would be better if it just accepted any string, because not all programs and OSes have option syntax that matches the limited list the error message indicates.\r\n\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@@ -43,7 +43,7 @@\n \n \n # RE for option descriptions\n-option_desc_re = re.compile(r'((?:/|--|-|\\+)?[^\\s=[]+)(=?\\s*.*)')\n+option_desc_re = re.compile(r'((?:/|--|-|\\+)?[^\\s=]+)(=?\\s*.*)')\n # RE for grammar tokens\n token_re = re.compile(r'`(\\w+)`', re.U)\n \n@@ -197,6 +197,11 @@ def handle_signature(self, sig: str, signode: desc_signature) -> str:\n                                location=signode)\n                 continue\n             optname, args = m.groups()\n+            if optname.endswith('[') and args.endswith(']'):\n+                # optional value surrounded by brackets (ex. foo[=bar])\n+                optname = optname[:-1]\n+                args = '[' + args\n+\n             if count:\n                 signode += addnodes.desc_addname(', ', ', ')\n             signode += addnodes.desc_name(optname, optname)\n",
  "similar_bug_items": [
    {
      "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": 5816,
      "pr_title": "C++, fix assertion in partial builds with duplicates",
      "pr_body": "### Feature or Bugfix\r\n- Bugfix\r\n\r\n### Purpose\r\nEverything can happen with partial builds, so the this simply removes an assertion.\r\n\r\n### Relates\r\n- Fixes #5496 \r\n\r\n",
      "issue_id": 5496,
      "issue_title": "asserts out with template specialization for pointer types with breathe",
      "issue_body": "### Problem\r\nBugs out with the following assertion\r\n\r\n```\r\nE:\\projekte\\breathe-doxygen-filter-test\\source\\contents.rst:5: WARNING: Too many template argument lists compared to parameter lists. Argume\r\nnt lists: 1, Parameter lists: 0, Extra empty parameters lists prepended: 1. Declaration:\r\n        TemplateClass<T *>\r\nE:\\projekte\\breathe-doxygen-filter-test\\source\\output\\class\\classTemplateClass.rst:4: WARNING: Duplicate declaration.\r\nE:\\projekte\\breathe-doxygen-filter-test\\source\\output\\class\\classTemplateClass_3_01T_01_5_01_4.rst:4: WARNING: Too many template argument li\r\nsts compared to parameter lists. Argument lists: 1, Parameter lists: 0, Extra empty parameters lists prepended: 1. Declaration:\r\n        TemplateClass<T *>\r\nE:\\projekte\\breathe-doxygen-filter-test\\source\\output\\class\\classTemplateClass_3_01T_01_5_01_4.rst:4: WARNING: Duplicate declaration.\r\n\r\nException occurred:\r\n  File \"c:\\anaconda2\\lib\\site-packages\\sphinx\\domains\\cpp.py\", line 4060, in _add_symbols\r\n    assert len(withDecl) <= 1\r\nAssertionError\r\nThe full traceback has been saved in c:\\users\\thomas\\appdata\\local\\temp\\sphinx-err-jugxnl.log, if you want to report the issue to the develo\r\npers.\r\nPlease also report this if it was a user error, so that a better error message can be provided next time.\r\nA bug report can be filed in the tracker at <https://github.com/sphinx-doc/sphinx/issues>. Thanks!\r\n```\r\n\r\nC++ declarations:\r\n\r\n```cpp\r\n/*!\r\nA generic template class.\r\n*/\r\ntemplate<typename T>\r\nclass TemplateClass\r\n{\r\n};\r\n\r\n/*!\r\nA partial specialization of TemplateClass for pointer types.\r\n*/\r\ntemplate<typename T>\r\nclass TemplateClass<T*>\r\n{\r\n};\r\n```\r\n\r\nbreathe code.\r\n```rst\r\n.. doxygenclass:: TemplateClass\r\n   :project: test\r\n   :no-link:\r\n\r\n.. doxygenclass:: TemplateClass< T * >\r\n   :project: test\r\n   :no-link:\r\n```\r\n\r\n#### Procedure to reproduce the problem\r\n```\r\ngit clone https://github.com/t-b/breathe-doxygen-filter-test\r\ncd breathe-doxygen-filter-test\r\n./run.sh\r\n```\r\n\r\n#### Error logs / results\r\n```\r\n# Sphinx version: 1.8.1\r\n# Python version: 2.7.12 (CPython)\r\n# Docutils version: 0.14 \r\n# Jinja2 version: 2.10\r\n# Last messages:\r\n#   Running Sphinx v1.8.1\r\n#   building [mo]: targets for 0 po files that are out of date\r\n#   building [html]: targets for 6 source files that are out of date\r\n#   updating environment:\r\n#   6 added, 0 changed, 0 removed\r\n#   reading sources... [ 16%] contents\r\n#   reading sources... [ 33%] output/class/classTemplateClass\r\n#   reading sources... [ 50%] output/class/classTemplateClass_3_01T_01_5_01_4\r\n#   reading sources... [ 66%] output/classlist\r\n#   reading sources... [ 83%] output/file/test_8cpp\r\n# Loaded extensions:\r\n#   sphinx.ext.mathjax (1.8.1) from c:\\anaconda2\\lib\\site-packages\\sphinx\\ext\\mathjax.pyc\r\n#   alabaster (0.7.10) from c:\\anaconda2\\lib\\site-packages\\alabaster\\__init__.pyc\r\n#   breathe (4.10.0) from c:\\anaconda2\\lib\\site-packages\\breathe\\__init__.pyc\r\nTraceback (most recent call last):\r\n  File \"c:\\anaconda2\\lib\\site-packages\\sphinx\\cmd\\build.py\", line 304, in build_main\r\n    app.build(args.force_all, filenames)\r\n  File \"c:\\anaconda2\\lib\\site-packages\\sphinx\\application.py\", line 341, in build\r\n    self.builder.build_update()\r\n  File \"c:\\anaconda2\\lib\\site-packages\\sphinx\\builders\\__init__.py\", line 347, in build_update\r\n    len(to_build))\r\n  File \"c:\\anaconda2\\lib\\site-packages\\sphinx\\builders\\__init__.py\", line 360, in build\r\n    updated_docnames = set(self.read())\r\n  File \"c:\\anaconda2\\lib\\site-packages\\sphinx\\builders\\__init__.py\", line 468, in read\r\n    self._read_serial(docnames)\r\n  File \"c:\\anaconda2\\lib\\site-packages\\sphinx\\builders\\__init__.py\", line 490, in _read_serial\r\n    self.read_doc(docname)\r\n  File \"c:\\anaconda2\\lib\\site-packages\\sphinx\\builders\\__init__.py\", line 534, in read_doc\r\n    doctree = read_doc(self.app, self.env, self.env.doc2path(docname))\r\n  File \"c:\\anaconda2\\lib\\site-packages\\sphinx\\io.py\", line 318, in read_doc\r\n    pub.publish()\r\n  File \"c:\\anaconda2\\lib\\site-packages\\docutils\\core.py\", line 217, in publish\r\n    self.settings)\r\n  File \"c:\\anaconda2\\lib\\site-packages\\docutils\\readers\\__init__.py\", line 72, in read\r\n    self.parse()\r\n  File \"c:\\anaconda2\\lib\\site-packages\\docutils\\readers\\__init__.py\", line 78, in parse\r\n    self.parser.parse(self.input, document)\r\n  File \"c:\\anaconda2\\lib\\site-packages\\sphinx\\parsers.py\", line 88, in parse\r\n    self.statemachine.run(inputstring, document, inliner=self.inliner)\r\n  File \"c:\\anaconda2\\lib\\site-packages\\docutils\\parsers\\rst\\states.py\", line 171, in run\r\n    input_source=document['source'])\r\n  File \"c:\\anaconda2\\lib\\site-packages\\docutils\\statemachine.py\", line 239, in run\r\n    context, state, transitions)\r\n  File \"c:\\anaconda2\\lib\\site-packages\\docutils\\statemachine.py\", line 460, in check_line\r\n    return method(match, context, next_state)\r\n  File \"c:\\anaconda2\\lib\\site-packages\\docutils\\parsers\\rst\\states.py\", line 2753, in underline\r\n    self.section(title, source, style, lineno - 1, messages)\r\n  File \"c:\\anaconda2\\lib\\site-packages\\docutils\\parsers\\rst\\states.py\", line 327, in section\r\n    self.new_subsection(title, lineno, messages)\r\n  File \"c:\\anaconda2\\lib\\site-packages\\docutils\\parsers\\rst\\states.py\", line 395, in new_subsection\r\n    node=section_node, match_titles=True)\r\n  File \"c:\\anaconda2\\lib\\site-packages\\docutils\\parsers\\rst\\states.py\", line 282, in nested_parse\r\n    node=node, match_titles=match_titles)\r\n  File \"c:\\anaconda2\\lib\\site-packages\\docutils\\parsers\\rst\\states.py\", line 196, in run\r\n    results = StateMachineWS.run(self, input_lines, input_offset)\r\n  File \"c:\\anaconda2\\lib\\site-packages\\docutils\\statemachine.py\", line 239, in run\r\n    context, state, transitions)\r\n  File \"c:\\anaconda2\\lib\\site-packages\\docutils\\statemachine.py\", line 460, in check_line\r\n    return method(match, context, next_state)\r\n  File \"c:\\anaconda2\\lib\\site-packages\\docutils\\parsers\\rst\\states.py\", line 2326, in explicit_markup\r\n    nodelist, blank_finish = self.explicit_construct(match)\r\n  File \"c:\\anaconda2\\lib\\site-packages\\docutils\\parsers\\rst\\states.py\", line 2338, in explicit_construct\r\n    return method(self, expmatch)\r\n  File \"c:\\anaconda2\\lib\\site-packages\\docutils\\parsers\\rst\\states.py\", line 2081, in directive\r\n    directive_class, match, type_name, option_presets)\r\n  File \"c:\\anaconda2\\lib\\site-packages\\docutils\\parsers\\rst\\states.py\", line 2130, in run_directive\r\n    result = directive_instance.run()\r\n  File \"c:\\anaconda2\\lib\\site-packages\\breathe\\directive\\file.py\", line 91, in run\r\n    return self.handle_contents(file_, project_info)\r\n  File \"c:\\anaconda2\\lib\\site-packages\\breathe\\directive\\file.py\", line 61, in handle_contents\r\n    node_list.extend(object_renderer.render(node_stack[0], context))\r\n  File \"c:\\anaconda2\\lib\\site-packages\\breathe\\renderer\\sphinxrenderer.py\", line 1304, in render\r\n    result = method(self, node)\r\n  File \"c:\\anaconda2\\lib\\site-packages\\breathe\\renderer\\sphinxrenderer.py\", line 630, in visit_compounddef\r\n    nodelist.extend(self.render_iterable(node.innerclass))\r\n  File \"c:\\anaconda2\\lib\\site-packages\\breathe\\renderer\\sphinxrenderer.py\", line 1314, in render_iterable\r\n    output.extend(self.render(entry))\r\n  File \"c:\\anaconda2\\lib\\site-packages\\breathe\\renderer\\sphinxrenderer.py\", line 1304, in render\r\n    result = method(self, node)\r\n  File \"c:\\anaconda2\\lib\\site-packages\\breathe\\renderer\\sphinxrenderer.py\", line 907, in visit_ref\r\n    return self.visit_compound(node, False, get_node_info=get_node_info)\r\n  File \"c:\\anaconda2\\lib\\site-packages\\breathe\\renderer\\sphinxrenderer.py\", line 508, in visit_compound\r\n    name, kind)\r\n  File \"c:\\anaconda2\\lib\\site-packages\\breathe\\renderer\\sphinxrenderer.py\", line 486, in render_signature\r\n    nodes = self.run_domain_directive(kind, self.context.directive_args[1], augment=augment)\r\n  File \"c:\\anaconda2\\lib\\site-packages\\breathe\\renderer\\sphinxrenderer.py\", line 351, in run_domain_directive\r\n    nodes = domain_directive.run()\r\n  File \"c:\\anaconda2\\lib\\site-packages\\sphinx\\domains\\cpp.py\", line 6312, in run\r\n    return ObjectDescription.run(self)\r\n  File \"c:\\anaconda2\\lib\\site-packages\\sphinx\\directives\\__init__.py\", line 161, in run\r\n    name = self.handle_signature(sig, signode)\r\n  File \"c:\\anaconda2\\lib\\site-packages\\sphinx\\domains\\cpp.py\", line 6332, in handle_signature\r\n    symbol = parentSymbol.add_declaration(ast, docname=self.env.docname)\r\n  File \"c:\\anaconda2\\lib\\site-packages\\sphinx\\domains\\cpp.py\", line 4151, in add_declaration\r\n    return self._add_symbols(nestedName, templateDecls, declaration, docname)\r\n  File \"c:\\anaconda2\\lib\\site-packages\\sphinx\\domains\\cpp.py\", line 4060, in _add_symbols\r\n    assert len(withDecl) <= 1\r\nAssertionError\r\n```\r\n\r\n### Environment info\r\n- OS: Win\r\n- Python version: 2.7.12\r\n- Sphinx version: 1.8.1",
      "issue_closed_at": "2018-12-16T19:38:25Z",
      "base_commit": "73a8b7e6581d094cc88c36cbeccab56031351d27",
      "changes": [
        {
          "file": "sphinx/domains/cpp.py",
          "type": "function",
          "name": "__init__",
          "class_name": "CPPExprRole",
          "code": "def __init__(self, asCode):\n        if asCode:\n            # render the expression as inline code\n            self.class_type = 'cpp-expr'\n            self.node_type = nodes.literal\n        else:\n            # render the expression as inline text\n            self.class_type = 'cpp-texpr'\n            self.node_type = nodes.inline"
        },
        {
          "file": "sphinx/domains/cpp.py",
          "type": "function",
          "name": "onMissingQualifiedSymbol",
          "class_name": "Symbol",
          "code": "def onMissingQualifiedSymbol(parentSymbol, identOrOp, templateParams, templateArgs):\n            # type: (Symbol, Union[ASTIdentifier, ASTOperator], Any, ASTTemplateArgs) -> Symbol\n            return None"
        },
        {
          "file": "sphinx/domains/cpp.py",
          "type": "function",
          "name": "to_string",
          "class_name": "Symbol",
          "code": "def to_string(self, indent):\n        # type: (int) -> unicode\n        res = ['\\t' * indent]  # type: List[unicode]\n        if not self.parent:\n            res.append('::')\n        else:\n            if self.templateParams:\n                res.append(text_type(self.templateParams))\n                res.append('\\n')\n                res.append('\\t' * indent)\n            if self.identOrOp:\n                res.append(text_type(self.identOrOp))\n            else:\n                res.append(text_type(self.declaration))\n            if self.templateArgs:\n                res.append(text_type(self.templateArgs))\n            if self.declaration:\n                res.append(\": \")\n                res.append(text_type(self.declaration))\n        if self.docname:\n            res.append('\\t(')\n            res.append(self.docname)\n            res.append(')')\n        res.append('\\n')\n        return ''.join(res)"
        },
        {
          "file": "sphinx/domains/cpp.py",
          "type": "class",
          "name": "CPPDomain",
          "code": "class CPPDomain(Domain):\n    \"\"\"C++ language domain.\"\"\"\n    name = 'cpp'\n    label = 'C++'\n    object_types = {\n        'class':      ObjType(_('class'),      'class',             'type', 'identifier'),\n        'union':      ObjType(_('union'),      'union',             'type', 'identifier'),\n        'function':   ObjType(_('function'),   'function',  'func', 'type', 'identifier'),\n        'member':     ObjType(_('member'),     'member',    'var'),\n        'type':       ObjType(_('type'),                            'type', 'identifier'),\n        'concept':    ObjType(_('concept'),    'concept',                   'identifier'),\n        'enum':       ObjType(_('enum'),       'enum',              'type', 'identifier'),\n        'enumerator': ObjType(_('enumerator'), 'enumerator')\n    }\n\n    directives = {\n        'class': CPPClassObject,\n        'union': CPPUnionObject,\n        'function': CPPFunctionObject,\n        'member': CPPMemberObject,\n        'var': CPPMemberObject,\n        'type': CPPTypeObject,\n        'concept': CPPConceptObject,\n        'enum': CPPEnumObject,\n        'enum-struct': CPPEnumObject,\n        'enum-class': CPPEnumObject,\n        'enumerator': CPPEnumeratorObject,\n        'namespace': CPPNamespaceObject,\n        'namespace-push': CPPNamespacePushObject,\n        'namespace-pop': CPPNamespacePopObject\n    }\n    roles = {\n        'any': CPPXRefRole(),\n        'class': CPPXRefRole(),\n        'union': CPPXRefRole(),\n        'func': CPPXRefRole(fix_parens=True),\n        'member': CPPXRefRole(),\n        'var': CPPXRefRole(),\n        'type': CPPXRefRole(),\n        'concept': CPPXRefRole(),\n        'enum': CPPXRefRole(),\n        'enumerator': CPPXRefRole(),\n        'expr': CPPExprRole(asCode=True),\n        'texpr': CPPExprRole(asCode=False)\n    }\n    initial_data = {\n        'root_symbol': Symbol(None, None, None, None, None, None),\n        'names': {}  # full name for indexing -> docname\n    }\n\n    def clear_doc(self, docname):\n        # type: (unicode) -> None\n        rootSymbol = self.data['root_symbol']\n        rootSymbol.clear_doc(docname)\n        for name, nDocname in list(self.data['names'].items()):\n            if nDocname == docname:\n                del self.data['names'][name]\n\n    def process_doc(self, env, docname, document):\n        # type: (BuildEnvironment, unicode, nodes.Node) -> None\n        # just for debugging\n        # print(\"process_doc:\", docname)\n        # print(self.data['root_symbol'].dump(0))\n        pass\n\n    def process_field_xref(self, pnode):\n        # type: (nodes.Node) -> None\n        pnode.attributes.update(self.env.ref_context)\n\n    def merge_domaindata(self, docnames, otherdata):\n        # type: (List[unicode], Dict) -> None\n        # print(\"merge_domaindata:\")\n        # print(\"self\")\n        # print(self.data['root_symbol'].dump(0))\n        # print(\"other:\")\n        # print(otherdata['root_symbol'].dump(0))\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\n\n    def _resolve_xref_inner(self, env, fromdocname, builder, typ,\n                            target, node, contnode, emitWarnings=True):\n        # type: (BuildEnvironment, unicode, Builder, unicode, unicode, nodes.Node, nodes.Node, bool) -> nodes.Node  # NOQA\n\n        class Warner(object):\n            def warn(self, msg):\n                if emitWarnings:\n                    logger.warning(msg, location=node)\n        warner = Warner()\n        # add parens again for those that could be functions\n        if typ == 'any' or typ == 'func':\n            target += '()'\n        parser = DefinitionParser(target, warner, env.config)\n        try:\n            ast, isShorthand = parser.parse_xref_object()\n            parser.assert_end()\n        except DefinitionError as e:\n            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\n            t, ex = findWarning(e)\n            warner.warn('Unparseable C++ cross-reference: %r\\n%s'\n                        % (t, text_type(ex.description)))\n            return None, None\n        parentKey = node.get(\"cpp:parent_key\", None)\n        rootSymbol = self.data['root_symbol']\n        if parentKey:\n            parentSymbol = rootSymbol.direct_lookup(parentKey)\n            if not parentSymbol:\n                print(\"Target: \", target)\n                print(\"ParentKey: \", parentKey)\n                print(rootSymbol.dump(1))\n            assert parentSymbol  # should be there\n        else:\n            parentSymbol = rootSymbol\n\n        if isShorthand:\n            ns = ast  # type: ASTNamespace\n            name = ns.nestedName\n            if ns.templatePrefix:\n                templateDecls = ns.templatePrefix.templates\n            else:\n                templateDecls = []\n            s = parentSymbol.find_name(name, templateDecls, typ,\n                                       templateShorthand=True,\n                                       matchSelf=True, recurseInAnon=True)\n        else:\n            decl = ast  # type: ASTDeclaration\n            name = decl.name\n            s = parentSymbol.find_declaration(decl, typ,\n                                              templateShorthand=True,\n                                              matchSelf=True, recurseInAnon=True)\n        if s is None or s.declaration is None:\n            txtName = text_type(name)\n            if txtName.startswith('std::') or txtName == 'std':\n                raise NoUri()\n            return None, None\n\n        if typ.startswith('cpp:'):\n            typ = typ[4:]\n        if typ == 'func':\n            typ = 'function'\n        declTyp = s.declaration.objectType\n\n        def checkType():\n            if typ == 'any' or typ == 'identifier':\n                return True\n            if declTyp == 'templateParam':\n                return True\n            objtypes = self.objtypes_for_role(typ)\n            if objtypes:\n                return declTyp in objtypes\n            print(\"Type is %s, declType is %s\" % (typ, declTyp))\n            assert False\n        if not checkType():\n            warner.warn(\"cpp:%s targets a %s (%s).\"\n                        % (typ, s.declaration.objectType,\n                           s.get_full_nested_name()))\n\n        declaration = s.declaration\n        if isShorthand:\n            fullNestedName = s.get_full_nested_name()\n            displayName = fullNestedName.get_display_string().lstrip(':')\n        else:\n            displayName = decl.get_display_string()\n        docname = s.docname\n        assert docname\n\n        # the non-identifier refs are cross-references, which should be processed:\n        # - fix parenthesis due to operator() and add_function_parentheses\n        if typ != \"identifier\":\n            title = contnode.pop(0).astext()\n            # If it's operator(), we need to add '()' if explicit function parens\n            # are requested. Then the Sphinx machinery will add another pair.\n            # Also, if it's an 'any' ref that resolves to a function, we need to add\n            # parens as well.\n            # However, if it's a non-shorthand function ref, for a function that\n            # takes no arguments, then we may need to add parens again as well.\n            addParen = 0\n            if not node.get('refexplicit', False) and declaration.objectType == 'function':\n                if isShorthand:\n                    # this is just the normal haxing for 'any' roles\n                    if env.config.add_function_parentheses and typ == 'any':\n                        addParen += 1\n                    # and now this stuff for operator()\n                    if (env.config.add_function_parentheses and typ == 'function' and\n                            title.endswith('operator()')):\n                        addParen += 1\n                    if ((typ == 'any' or typ == 'function') and\n                            title.endswith('operator') and\n                            displayName.endswith('operator()')):\n                        addParen += 1\n                else:\n                    # our job here is to essentially nullify add_function_parentheses\n                    if env.config.add_function_parentheses:\n                        if typ == 'any' and displayName.endswith('()'):\n                            addParen += 1\n                        elif typ == 'function':\n                            if title.endswith('()') and not displayName.endswith('()'):\n                                title = title[:-2]\n                    else:\n                        if displayName.endswith('()'):\n                            addParen += 1\n            if addParen > 0:\n                title += '()' * addParen\n            # and reconstruct the title again\n            contnode += nodes.Text(title)\n        return make_refnode(builder, fromdocname, docname,\n                            declaration.get_newest_id(), contnode, displayName\n                            ), declaration.objectType\n\n    def resolve_xref(self, env, fromdocname, builder,\n                     typ, target, node, contnode):\n        # type: (BuildEnvironment, unicode, Builder, unicode, unicode, nodes.Node, nodes.Node) -> nodes.Node  # NOQA\n        return self._resolve_xref_inner(env, fromdocname, builder, typ,\n                                        target, node, contnode)[0]\n\n    def resolve_any_xref(self, env, fromdocname, builder, target,\n                         node, contnode):\n        # type: (BuildEnvironment, unicode, Builder, unicode, nodes.Node, nodes.Node) -> List[Tuple[unicode, nodes.Node]]  # NOQA\n        node, objtype = self._resolve_xref_inner(env, fromdocname, builder,\n                                                 'any', target, node, contnode,\n                                                 emitWarnings=False)\n        if node:\n            if objtype == 'templateParam':\n                return [('cpp:templateParam', node)]\n            else:\n                return [('cpp:' + self.role_for_objtype(objtype), node)]\n        return []\n\n    def get_objects(self):\n        # type: () -> Iterator[Tuple[unicode, unicode, unicode, unicode, unicode, int]]\n        rootSymbol = self.data['root_symbol']\n        for symbol in rootSymbol.get_all_symbols():\n            if symbol.declaration is None:\n                continue\n            assert symbol.docname\n            fullNestedName = symbol.get_full_nested_name()\n            name = text_type(fullNestedName).lstrip(':')\n            dispname = fullNestedName.get_display_string().lstrip(':')\n            objectType = symbol.declaration.objectType\n            docname = symbol.docname\n            newestId = symbol.declaration.get_newest_id()\n            yield (name, dispname, objectType, docname, newestId, 1)\n\n    def get_full_qualified_name(self, node):\n        # type: (nodes.Node) -> unicode\n        target = node.get('reftarget', None)\n        if target is None:\n            return None\n        parentKey = node.get(\"cpp:parent_key\", None)\n        if parentKey is None or len(parentKey) <= 0:\n            return None\n\n        rootSymbol = self.data['root_symbol']\n        parentSymbol = rootSymbol.direct_lookup(parentKey)\n        parentName = parentSymbol.get_full_nested_name()\n        return '::'.join([text_type(parentName), target])"
        },
        {
          "file": "sphinx/domains/cpp.py",
          "type": "function",
          "name": "process_field_xref",
          "class_name": "CPPDomain",
          "code": "def process_field_xref(self, pnode):\n        # type: (nodes.Node) -> None\n        pnode.attributes.update(self.env.ref_context)"
        }
      ]
    },
    {
      "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": 7062,
      "pr_title": "Close #7032: html_scaled_image_link is disabled for individual image",
      "pr_body": "### Feature or Bugfix\r\n- Feature\r\n\r\n### Purpose\r\n- refs: #7032 ",
      "issue_id": 7032,
      "issue_title": "html_scaled_image_link behavior is too restrictive for extension developers",
      "issue_body": "I'm  developing sphinx-extensions and my code is using ``nodes.image`` from docutils to add images to the documentation which shall be used as buttons, status or other indicators.\r\n\r\nBut Sphinx is surrounding my images with ``nodes.reference``, so that they become clickable links, which open the image in the browser.\r\n\r\nThis is not the desired behavior, as I bind the images to javascrip (html only) to use them as buttons for specific actions.\r\n\r\n**Problem**\r\nhttps://github.com/sphinx-doc/sphinx/blob/cad1f86dd895836e36e6a57407a5d82c990dfaa8/sphinx/builders/html.py#L808\r\n\r\nThe handling of ``html_scaled_image_link`` does only look on the global config, but not on the image node itself, if maybe some attributes are set to avoid adding of references.\r\n\r\nSo I can only switch off the behavior globally, which for sure is not an option.\r\n\r\n**Solution / Proposal**\r\nI propose the usage of a ``target`` attribute of the image node.\r\nIf this is set, the target-value is used as refuri.\r\nIf it is set to None but part of node.attributes-dict, no reference should be added.\r\nIf it is not set (which is currently always the case), normal handling as of now.\r\n\r\nAs far as I have understood the docutils code, the handlers of images-nodes do not care about a ``target`` attribute.\r\n\r\n**Workaround**\r\nLuckily I found on, but it is very ugly.\r\n\r\nFirst off all, you need to add the image-node to a reference-node.\r\nIn this case, the ``html_scaled_image_link`` handling is aborted, as it stops if parent of an image-node is already a reference-node.\r\n\r\nBut now you have a not wanted reference in your html-code, which forces the browser to jump/scroll the page when clicked on it.\r\nSo I must use some javascript to avoid this :(\r\n\r\nHere the code for the important parts:\r\n```\r\n# Wrap image-node inside a reference node\r\n...\r\nref_node = nodes.reference('', '', refuri='#', classes=['no_link'])  # Set a specific class for later js handling\r\nref_node.append(image_node)\r\nreturn ref_node\r\n```\r\n\r\nA loaded js-file needs to have this in it:\r\n```\r\n// JQuery is used here\r\n$(document).ready(function() {\r\n    $('a.no_link').click(function (e) {\r\n        e.preventDefault();\r\n    });\r\n});\r\n```\r\n\r\n**So, what do you think?**.\r\nI feel not really happy with the proposed solution.\r\nMaybe someone else has a better idea, how to \"deactivate\" this behavior for sphinx extension developers.\r\n",
      "issue_closed_at": "2020-02-22T08:38:06Z",
      "base_commit": "f6882d74663438b281a1260684a16e97696ed6d9",
      "changes": [
        {
          "file": "sphinx/builders/html/__init__.py",
          "type": "function",
          "name": "post_process_images",
          "class_name": "StandaloneHTMLBuilder",
          "code": "def post_process_images(self, doctree: Node) -> None:\n        \"\"\"Pick the best candidate for an image and link down-scaled images to\n        their high res version.\n        \"\"\"\n        Builder.post_process_images(self, doctree)\n\n        if self.config.html_scaled_image_link and self.html_scaled_image_link:\n            for node in doctree.traverse(nodes.image):\n                scale_keys = ('scale', 'width', 'height')\n                if not any((key in node) for key in scale_keys) or \\\n                   isinstance(node.parent, nodes.reference):\n                    # docutils does unfortunately not preserve the\n                    # ``target`` attribute on images, so we need to check\n                    # the parent node here.\n                    continue\n                uri = node['uri']\n                reference = nodes.reference('', '', internal=True)\n                if uri in self.images:\n                    reference['refuri'] = posixpath.join(self.imgpath,\n                                                         self.images[uri])\n                else:\n                    reference['refuri'] = uri\n                node.replace_self(reference)\n                reference.append(node)"
        }
      ]
    },
    {
      "pr_number": 2950,
      "pr_title": "2695 errhandling on setup command",
      "pr_body": "refs: #2695 \n",
      "issue_id": 2695,
      "issue_title": "Why does BuildDoc.run() not catch exceptions?",
      "issue_body": "`sphinx-build` catches exceptions and displays them in a non-Python-programmer-friendly way, so I was surprised that the same error is not intercepted when using `python setup.py build_sphinx`.\n\nI thought that both do basically the same thing, but it turns out that there is not a lot of code re-use between the two.\n\nIt would be possible to add (more) exception handling to [setup_command.py](https://github.com/sphinx-doc/sphinx/blob/c7c07db86b30b811bd8cbd31e6b2a26224c675e4/sphinx/setup_command.py), but wouldn't it be better to use the same exception handling for both use cases?\n",
      "issue_closed_at": "2016-09-14T17:28:22Z",
      "base_commit": "8fa2e337a6f970149ae238a00ca927ae4741c310",
      "changes": [
        {
          "file": "sphinx/cmdline.py",
          "type": "function",
          "name": "format_help",
          "class_name": "MyFormatter",
          "code": "def format_help(self, formatter):\n        result = []\n        if self.description:\n            result.append(self.format_description(formatter))\n        if self.option_list:\n            result.append(self.format_option_help(formatter))\n        return \"\\n\".join(result)"
        },
        {
          "file": "sphinx/cmdline.py",
          "type": "function",
          "name": "main",
          "class_name": null,
          "code": "def main(argv):\n    if not color_terminal():\n        nocolor()\n\n    parser = optparse.OptionParser(USAGE, epilog=EPILOG, formatter=MyFormatter())\n    parser.add_option('--version', action='store_true', dest='version',\n                      help='show version information and exit')\n\n    group = parser.add_option_group('General options')\n    group.add_option('-b', metavar='BUILDER', dest='builder', default='html',\n                     help='builder to use; default is html')\n    group.add_option('-a', action='store_true', dest='force_all',\n                     help='write all files; default is to only write new and '\n                     'changed files')\n    group.add_option('-E', action='store_true', dest='freshenv',\n                     help='don\\'t use a saved environment, always read '\n                     'all files')\n    group.add_option('-d', metavar='PATH', default=None, dest='doctreedir',\n                     help='path for the cached environment and doctree files '\n                     '(default: outdir/.doctrees)')\n    group.add_option('-j', metavar='N', default=1, type='int', dest='jobs',\n                     help='build in parallel with N processes where possible')\n    # this option never gets through to this point (it is intercepted earlier)\n    # group.add_option('-M', metavar='BUILDER', dest='make_mode',\n    #                 help='\"make\" mode -- as used by Makefile, like '\n    #                 '\"sphinx-build -M html\"')\n\n    group = parser.add_option_group('Build configuration options')\n    group.add_option('-c', metavar='PATH', dest='confdir',\n                     help='path where configuration file (conf.py) is located '\n                     '(default: same as sourcedir)')\n    group.add_option('-C', action='store_true', dest='noconfig',\n                     help='use no config file at all, only -D options')\n    group.add_option('-D', metavar='setting=value', action='append',\n                     dest='define', default=[],\n                     help='override a setting in configuration file')\n    group.add_option('-A', metavar='name=value', action='append',\n                     dest='htmldefine', default=[],\n                     help='pass a value into HTML templates')\n    group.add_option('-t', metavar='TAG', action='append',\n                     dest='tags', default=[],\n                     help='define tag: include \"only\" blocks with TAG')\n    group.add_option('-n', action='store_true', dest='nitpicky',\n                     help='nit-picky mode, warn about all missing references')\n\n    group = parser.add_option_group('Console output options')\n    group.add_option('-v', action='count', dest='verbosity', default=0,\n                     help='increase verbosity (can be repeated)')\n    group.add_option('-q', action='store_true', dest='quiet',\n                     help='no output on stdout, just warnings on stderr')\n    group.add_option('-Q', action='store_true', dest='really_quiet',\n                     help='no output at all, not even warnings')\n    group.add_option('-N', action='store_true', dest='nocolor',\n                     help='do not emit colored output')\n    group.add_option('-w', metavar='FILE', dest='warnfile',\n                     help='write warnings (and errors) to given file')\n    group.add_option('-W', action='store_true', dest='warningiserror',\n                     help='turn warnings into errors')\n    group.add_option('-T', action='store_true', dest='traceback',\n                     help='show full traceback on exception')\n    group.add_option('-P', action='store_true', dest='pdb',\n                     help='run Pdb on exception')\n\n    # parse options\n    try:\n        opts, args = parser.parse_args(list(argv[1:]))\n    except SystemExit as err:\n        return err.code\n\n    # handle basic options\n    if opts.version:\n        print('Sphinx (sphinx-build) %s' % __display_version__)\n        return 0\n\n    # get paths (first and second positional argument)\n    try:\n        srcdir = abspath(args[0])\n        confdir = abspath(opts.confdir or srcdir)\n        if opts.noconfig:\n            confdir = None\n        if not path.isdir(srcdir):\n            print('Error: Cannot find source directory `%s\\'.' % srcdir,\n                  file=sys.stderr)\n            return 1\n        if not opts.noconfig and not path.isfile(path.join(confdir, 'conf.py')):\n            print('Error: Config directory doesn\\'t contain a conf.py file.',\n                  file=sys.stderr)\n            return 1\n        outdir = abspath(args[1])\n        if srcdir == outdir:\n            print('Error: source directory and destination directory are same.',\n                  file=sys.stderr)\n            return 1\n    except IndexError:\n        parser.print_help()\n        return 1\n    except UnicodeError:\n        print(\n            'Error: Multibyte filename not supported on this filesystem '\n            'encoding (%r).' % fs_encoding, file=sys.stderr)\n        return 1\n\n    # handle remaining filename arguments\n    filenames = args[2:]\n    err = 0\n    for filename in filenames:\n        if not path.isfile(filename):\n            print('Error: Cannot find file %r.' % filename, file=sys.stderr)\n            err = 1\n    if err:\n        return 1\n\n    # likely encoding used for command-line arguments\n    try:\n        locale = __import__('locale')  # due to submodule of the same name\n        likely_encoding = locale.getpreferredencoding()\n    except Exception:\n        likely_encoding = None\n\n    if opts.force_all and filenames:\n        print('Error: Cannot combine -a option and filenames.', file=sys.stderr)\n        return 1\n\n    if opts.nocolor:\n        nocolor()\n\n    doctreedir = abspath(opts.doctreedir or path.join(outdir, '.doctrees'))\n\n    status = sys.stdout\n    warning = sys.stderr\n    error = sys.stderr\n\n    if opts.quiet:\n        status = None\n    if opts.really_quiet:\n        status = warning = None\n    if warning and opts.warnfile:\n        try:\n            warnfp = open(opts.warnfile, 'w')\n        except Exception as exc:\n            print('Error: Cannot open warning file %r: %s' %\n                  (opts.warnfile, exc), file=sys.stderr)\n            sys.exit(1)\n        warning = Tee(warning, warnfp)\n        error = warning\n\n    confoverrides = {}\n    for val in opts.define:\n        try:\n            key, val = val.split('=')\n        except ValueError:\n            print('Error: -D option argument must be in the form name=value.',\n                  file=sys.stderr)\n            return 1\n        if likely_encoding and isinstance(val, binary_type):\n            try:\n                val = val.decode(likely_encoding)\n            except UnicodeError:\n                pass\n        confoverrides[key] = val\n\n    for val in opts.htmldefine:\n        try:\n            key, val = val.split('=')\n        except ValueError:\n            print('Error: -A option argument must be in the form name=value.',\n                  file=sys.stderr)\n            return 1\n        try:\n            val = int(val)\n        except ValueError:\n            if likely_encoding and isinstance(val, binary_type):\n                try:\n                    val = val.decode(likely_encoding)\n                except UnicodeError:\n                    pass\n        confoverrides['html_context.%s' % key] = val\n\n    if opts.nitpicky:\n        confoverrides['nitpicky'] = True\n\n    app = None\n    try:\n        app = Sphinx(srcdir, confdir, outdir, doctreedir, opts.builder,\n                     confoverrides, status, warning, opts.freshenv,\n                     opts.warningiserror, opts.tags, opts.verbosity, opts.jobs)\n        app.build(opts.force_all, filenames)\n        return app.statuscode\n    except (Exception, KeyboardInterrupt) as err:\n        if opts.pdb:\n            import pdb\n            print(red('Exception occurred while building, starting debugger:'),\n                  file=error)\n            traceback.print_exc()\n            pdb.post_mortem(sys.exc_info()[2])\n        else:\n            print(file=error)\n            if opts.verbosity or opts.traceback:\n                traceback.print_exc(None, error)\n                print(file=error)\n            if isinstance(err, KeyboardInterrupt):\n                print('interrupted!', file=error)\n            elif isinstance(err, SystemMessage):\n                print(red('reST markup error:'), file=error)\n                print(terminal_safe(err.args[0]), file=error)\n            elif isinstance(err, SphinxError):\n                print(red('%s:' % err.category), file=error)\n                print(terminal_safe(text_type(err)), file=error)\n            elif isinstance(err, UnicodeError):\n                print(red('Encoding error:'), file=error)\n                print(terminal_safe(text_type(err)), file=error)\n                tbpath = save_traceback(app)\n                print(red('The full traceback has been saved in %s, if you want '\n                          'to report the issue to the developers.' % tbpath),\n                      file=error)\n            elif isinstance(err, RuntimeError) and 'recursion depth' in str(err):\n                print(red('Recursion error:'), file=error)\n                print(terminal_safe(text_type(err)), file=error)\n                print(file=error)\n                print('This can happen with very large or deeply nested source '\n                      'files.  You can carefully increase the default Python '\n                      'recursion limit of 1000 in conf.py with e.g.:', file=error)\n                print('    import sys; sys.setrecursionlimit(1500)', file=error)\n            else:\n                print(red('Exception occurred:'), file=error)\n                print(format_exception_cut_frames().rstrip(), file=error)\n                tbpath = save_traceback(app)\n                print(red('The full traceback has been saved in %s, if you '\n                          'want to report the issue to the developers.' % tbpath),\n                      file=error)\n                print('Please also report this if it was a user error, so '\n                      'that a better error message can be provided next time.',\n                      file=error)\n                print('A bug report can be filed in the tracker at '\n                      '<https://github.com/sphinx-doc/sphinx/issues>. Thanks!',\n                      file=error)\n            return 1"
        },
        {
          "file": "sphinx/setup_command.py",
          "type": "line",
          "name": "line 15",
          "code": "\nimport sys\nimport os\nimport traceback\n\nfrom distutils.cmd import Command\nfrom distutils.errors import DistutilsOptionError, DistutilsExecError\n\nfrom six import StringIO, string_types\n\nfrom sphinx.application import Sphinx\nfrom sphinx.util.console import darkred, nocolor, color_terminal\nfrom sphinx.util.osutil import abspath\n\n"
        },
        {
          "file": "sphinx/setup_command.py",
          "type": "function",
          "name": "initialize_options",
          "class_name": "BuildDoc",
          "code": "def initialize_options(self):\n        self.fresh_env = self.all_files = False\n        self.pdb = False\n        self.source_dir = self.build_dir = None\n        self.builder = 'html'\n        self.warning_is_error = False\n        self.project = ''\n        self.version = ''\n        self.release = ''\n        self.today = ''\n        self.config_dir = None\n        self.link_index = False\n        self.copyright = ''"
        },
        {
          "file": "sphinx/setup_command.py",
          "type": "function",
          "name": "run",
          "class_name": "BuildDoc",
          "code": "def run(self):\n        if not color_terminal():\n            nocolor()\n        if not self.verbose:\n            status_stream = StringIO()\n        else:\n            status_stream = sys.stdout\n        confoverrides = {}\n        if self.project:\n            confoverrides['project'] = self.project\n        if self.version:\n            confoverrides['version'] = self.version\n        if self.release:\n            confoverrides['release'] = self.release\n        if self.today:\n            confoverrides['today'] = self.today\n        if self.copyright:\n            confoverrides['copyright'] = self.copyright\n        app = Sphinx(self.source_dir, self.config_dir,\n                     self.builder_target_dir, self.doctree_dir,\n                     self.builder, confoverrides, status_stream,\n                     freshenv=self.fresh_env,\n                     warningiserror=self.warning_is_error)\n\n        try:\n            app.build(force_all=self.all_files)\n            if app.statuscode:\n                raise DistutilsExecError(\n                    'caused by %s builder.' % app.builder.name)\n        except Exception as err:\n            if self.pdb:\n                import pdb\n                print(darkred('Exception occurred while building, starting debugger:'),\n                      file=sys.stderr)\n                traceback.print_exc()\n                pdb.post_mortem(sys.exc_info()[2])\n            else:\n                from docutils.utils import SystemMessage\n                if isinstance(err, SystemMessage):\n                    print(darkred('reST markup error:'), file=sys.stderr)\n                    print(err.args[0].encode('ascii', 'backslashreplace'),\n                          file=sys.stderr)\n                else:\n                    raise\n\n        if self.link_index:\n            src = app.config.master_doc + app.builder.out_suffix\n            dst = app.builder.get_outfilename('index')\n            os.symlink(src, dst)"
        }
      ]
    }
  ]
}