{
  "Selected_candidate": {
    "pr_number": 5470,
    "pr_title": "Fixes #5426: [Napoleon] Better handling of inline attributes",
    "pr_body": "Fixes #5426: [Napoleon] Better handling of inline attributes\r\n\r\n### Feature or Bugfix\r\n- Bugfix\r\n\r\n### Purpose\r\n- Better handling of inline attributes by Napoleon to support both Napoleon style attribute docstrings and the specifically formatted docstring for class data members.\r\n",
    "issue_id": 5426,
    "issue_title": "TypeError in docutils/writers/_html_base.py for autodoc'd class data member",
    "issue_body": "This bug does not occur with previous versions of Sphinx.\r\n\r\nA class's data member must have a specifically formatted docstring.\r\n\r\nThe first part of the docstring (the main text) must match the regular expression /:\\s*$/ (ends with a colon). After the colon, any amount of whitespace may be present. The text thereafter is an enumerated list (both - and * work) of any number of items. The first list item matches the expression /:/ (contains a colon).\r\n\r\n# Environment\r\n\r\n- OS: OSX 10.13.6\r\n- Python version: 3.7 (installed via homebrew). Also happens in 3.6.4.\r\n- Sphinx version: 1.8.0\r\n\r\n# Test Case\r\n\r\nRun autodoc on the following and generate HTML:\r\n\r\n```python\r\nclass SomeClass:\r\n  some_data = 0\r\n  \"\"\"data member description:\r\n\r\n  - a: b\r\n  \"\"\"\r\n```\r\n\r\n# Actual Result\r\n\r\nSphinx crashes with the following output:\r\n\r\n```writing output...                                                                                                                                          \r\nException occurred:\r\n  File \".../site-packages/docutils/writers/_html_base.py\", line 399, in set_class_on_child\r\n    child['classes'].append(class_)\r\nTypeError: string indices must be integers\r\n```\r\n\r\n# Error logs / results\r\n\r\n```\r\n# Sphinx version: 1.8.0\r\n# Python version: 3.7.0 (CPython)\r\n# Docutils version: 0.14 \r\n# Jinja2 version: 2.10\r\n# Last messages:\r\n#   writing output... [ 59%] someclass\r\n#   \r\n# Loaded extensions:\r\n#   sphinx.ext.mathjax (1.8.0) from /.venv/lib/python3.7/site-packages/sphinx/ext/mathjax.py\r\n#   alabaster (0.7.11) from /.venv/lib/python3.7/site-packages/alabaster/__init__.py\r\n#   sphinx.ext.autodoc (1.8.0) from /.venv/lib/python3.7/site-packages/sphinx/ext/autodoc/__init__.py\r\n#   sphinx.ext.napoleon (1.8.0) from /.venv/lib/python3.7/site-packages/sphinx/ext/napoleon/__init__.py\r\nTraceback (most recent call last):\r\n  File \"/.venv/lib/python3.7/site-packages/sphinx/cmd/build.py\", line 304, in build_main\r\n    app.build(args.force_all, filenames)\r\n  File \"/.venv/lib/python3.7/site-packages/sphinx/application.py\", line 341, in build\r\n    self.builder.build_update()\r\n  File \"/.venv/lib/python3.7/site-packages/sphinx/builders/__init__.py\", line 347, in build_update\r\n    len(to_build))\r\n  File \"/.venv/lib/python3.7/site-packages/sphinx/builders/__init__.py\", line 412, in build\r\n    self.write(docnames, list(updated_docnames), method)\r\n  File \"/.venv/lib/python3.7/site-packages/sphinx/builders/__init__.py\", line 593, in write\r\n    self._write_serial(sorted(docnames))\r\n  File \"/.venv/lib/python3.7/site-packages/sphinx/builders/__init__.py\", line 604, in _write_serial\r\n    self.write_doc(docname, doctree)\r\n  File \"/.venv/lib/python3.7/site-packages/sphinx/builders/html.py\", line 730, in write_doc\r\n    self.docwriter.write(doctree, destination)\r\n  File \"/.venv/lib/python3.7/site-packages/docutils/writers/__init__.py\", line 80, in write\r\n    self.translate()\r\n  File \"/.venv/lib/python3.7/site-packages/sphinx/writers/html.py\", line 58, in translate\r\n    self.document.walkabout(visitor)\r\n  File \"/.venv/lib/python3.7/site-packages/docutils/nodes.py\", line 174, in walkabout\r\n    if child.walkabout(visitor):\r\n  File \"/.venv/lib/python3.7/site-packages/docutils/nodes.py\", line 174, in walkabout\r\n    if child.walkabout(visitor):\r\n  File \"/.venv/lib/python3.7/site-packages/docutils/nodes.py\", line 174, in walkabout\r\n    if child.walkabout(visitor):\r\n  [Previous line repeated 4 more times]\r\n  File \"/.venv/lib/python3.7/site-packages/docutils/nodes.py\", line 166, in walkabout\r\n    visitor.dispatch_visit(self)\r\n  File \"/.venv/lib/python3.7/site-packages/docutils/nodes.py\", line 1882, in dispatch_visit\r\n    return method(node)\r\n  File \"/.venv/lib/python3.7/site-packages/docutils/writers/html4css1/__init__.py\", line 402, in visit_field_body\r\n    self.set_class_on_child(node, 'first', 0)\r\n  File \"/.venv/lib/python3.7/site-packages/docutils/writers/_html_base.py\", line 399, in set_class_on_child\r\n    child['classes'].append(class_)\r\nTypeError: string indices must be integers\r\n```\r\n\r\n",
    "issue_closed_at": "2018-09-23T13:26:17Z",
    "base_commit": "37d58ab9d5b27403226c7e6ddc1cc48b905db297",
    "changes": [
      {
        "file": "sphinx/ext/napoleon/__init__.py",
        "type": "line",
        "name": "line 9",
        "code": "    :license: BSD, see LICENSE for details.\n\"\"\"\n\nimport sphinx\nfrom sphinx.application import Sphinx\nfrom sphinx.ext.napoleon.docstring import GoogleDocstring, NumpyDocstring\n"
      },
      {
        "file": "sphinx/ext/napoleon/__init__.py",
        "type": "function",
        "name": "setup",
        "class_name": null,
        "code": "def setup(app):\n    # type: (Sphinx) -> Dict[unicode, Any]\n    \"\"\"Sphinx extension setup function.\n\n    When the extension is loaded, Sphinx imports this module and executes\n    the ``setup()`` function, which in turn notifies Sphinx of everything\n    the extension offers.\n\n    Parameters\n    ----------\n    app : sphinx.application.Sphinx\n        Application object representing the Sphinx process\n\n    See Also\n    --------\n    `The Sphinx documentation on Extensions\n    <http://sphinx-doc.org/extensions.html>`_\n\n    `The Extension Tutorial <http://sphinx-doc.org/extdev/tutorial.html>`_\n\n    `The Extension API <http://sphinx-doc.org/extdev/appapi.html>`_\n\n    \"\"\"\n    if not isinstance(app, Sphinx):\n        return  # probably called by tests\n\n    _patch_python_domain()\n\n    app.setup_extension('sphinx.ext.autodoc')\n    app.connect('autodoc-process-docstring', _process_docstring)\n    app.connect('autodoc-skip-member', _skip_member)\n\n    for name, (default, rebuild) in Config._config_values.items():\n        app.add_config_value(name, default, rebuild)\n    return {'version': sphinx.__display_version__, 'parallel_read_safe': True}"
      },
      {
        "file": "sphinx/ext/napoleon/__init__.py",
        "type": "function",
        "name": "setup",
        "class_name": null,
        "code": "def setup(app):\n    # type: (Sphinx) -> Dict[unicode, Any]\n    \"\"\"Sphinx extension setup function.\n\n    When the extension is loaded, Sphinx imports this module and executes\n    the ``setup()`` function, which in turn notifies Sphinx of everything\n    the extension offers.\n\n    Parameters\n    ----------\n    app : sphinx.application.Sphinx\n        Application object representing the Sphinx process\n\n    See Also\n    --------\n    `The Sphinx documentation on Extensions\n    <http://sphinx-doc.org/extensions.html>`_\n\n    `The Extension Tutorial <http://sphinx-doc.org/extdev/tutorial.html>`_\n\n    `The Extension API <http://sphinx-doc.org/extdev/appapi.html>`_\n\n    \"\"\"\n    if not isinstance(app, Sphinx):\n        return  # probably called by tests\n\n    _patch_python_domain()\n\n    app.setup_extension('sphinx.ext.autodoc')\n    app.connect('autodoc-process-docstring', _process_docstring)\n    app.connect('autodoc-skip-member', _skip_member)\n\n    for name, (default, rebuild) in Config._config_values.items():\n        app.add_config_value(name, default, rebuild)\n    return {'version': sphinx.__display_version__, 'parallel_read_safe': True}"
      },
      {
        "file": "sphinx/ext/napoleon/docstring.py",
        "type": "function",
        "name": "_consume_inline_attribute",
        "class_name": "GoogleDocstring",
        "code": "def _consume_inline_attribute(self):\n        # type: () -> Tuple[unicode, List[unicode]]\n        line = next(self._line_iter)\n        _type, colon, _desc = self._partition_field_on_colon(line)\n        if not colon:\n            _type, _desc = _desc, _type\n        _descs = [_desc] + self._dedent(self._consume_to_end())\n        _descs = self.__class__(_descs, self._config).lines()\n        return _type, _descs"
      }
    ]
  },
  "Justification": "Candidate D is the most relevant bug report because it addresses a specific TypeError that occurs when autodoc is used to document class data members, similar to the issue at hand with attribute visibility issues in the CURRENT bug report. Both bugs involve the processing of Python class members in Sphinx and are related to how docstrings and attributes are formatted and rendered in HTML. The fact that Candidate D specifically mentions the Napoleon extension hints it involves a close tie to the current issue with the attribute `hello_`, which also faces formatting issues within the same context (autodoc and Sphinx). This overlap makes it highly likely that solving the issues in Candidate D may provide valuable clues for fixing the CURRENT bug.",
  "instance_id": "sphinx-doc__sphinx-7738",
  "repo": "sphinx-doc/sphinx",
  "created_at": "2020-05-27T16:48:09Z",
  "problem_statement": "overescaped trailing underscore on attribute with napoleon\n**Describe the bug**\r\nAttribute name `hello_` shows up as `hello\\_` in the html (visible backslash) with napoleon.\r\n\r\n**To Reproduce**\r\nSteps to reproduce the behavior:\r\n\r\nempty `__init__.py`\r\n`a.py` contains\r\n```python\r\nclass A:\r\n    \"\"\"\r\n    Attributes\r\n    ----------\r\n    hello_: int\r\n        hi\r\n    \"\"\"\r\n    pass\r\n```\r\nrun `sphinx-quickstart`\r\nadd `'sphinx.ext.autodoc', 'sphinx.ext.napoleon'` to extensions in conf.py.\r\nadd `.. autoclass:: a.A` to index.rst\r\nPYTHONPATH=. make clean html\r\nopen _build/html/index.html in web browser and see the ugly backslash.\r\n\r\n**Expected behavior**\r\nNo backslash, a similar output to what I get for\r\n```rst\r\n    .. attribute:: hello_\r\n        :type: int\r\n\r\n        hi\r\n```\r\n(the type shows up differently as well, but that's not the point here)\r\nOlder versions like 2.4.3 look ok to me.\r\n\r\n**Environment info**\r\n- OS: Linux debian testing\r\n- Python version: 3.8.3\r\n- Sphinx version: 3.0.4\r\n- Sphinx extensions:  sphinx.ext.autodoc, sphinx.ext.napoleon\r\n- Extra tools:\n",
  "patch": "diff --git a/sphinx/ext/napoleon/docstring.py b/sphinx/ext/napoleon/docstring.py\n--- a/sphinx/ext/napoleon/docstring.py\n+++ b/sphinx/ext/napoleon/docstring.py\n@@ -318,7 +318,7 @@ def _dedent(self, lines: List[str], full: bool = False) -> List[str]:\n             return [line[min_indent:] for line in lines]\n \n     def _escape_args_and_kwargs(self, name: str) -> str:\n-        if name.endswith('_'):\n+        if name.endswith('_') and getattr(self._config, 'strip_signature_backslash', False):\n             name = name[:-1] + r'\\_'\n \n         if name[:2] == '**':\n"
}