{
  "Selected_candidate": {
    "pr_number": 8274,
    "pr_title": "Fix #8200: autodoc: type aliases break type formatting",
    "pr_body": "### Feature or Bugfix\r\n- Bugfix\r\n\r\n### Purpose\r\n- refs: #8200 \r\n- The annotation option is shared between auto directives unexpectedly.\r\nIt causes supression of type annotations for objects after\r\nGenericAlias definition.\r\n",
    "issue_id": 8200,
    "issue_title": "Certain type aliases break type formatting in automodule",
    "issue_body": "**Describe the bug**\r\nIf `automodule` is applied to a module which defines a documented complex type alias, the type annotations on all attributes later in the module will be ignored.\r\n\r\n**To Reproduce**\r\nApply `automodule` to a module like the following:\r\n```\r\nimport typing\r\n\r\nAlias = typing.Union[int, float]\r\n\"\"\"a type alias\"\"\"\r\n\r\nclass Foo:\r\n    \"\"\"blah\"\"\"\r\n\r\n    bar: object\r\n    \"\"\"an attribute\"\"\"\r\n```\r\nThe generated documentation will not list any type for the attribute `bar`. Removing the definition of `Alias` causes the type of `bar` to be correctly rendered.\r\n\r\n**Expected behavior**\r\nThe presence of a type alias definition should not affect the display of type information for unrelated members.\r\n\r\n**Your project**\r\nHere is a tiny example project: [example3.zip](https://github.com/sphinx-doc/sphinx/files/5212082/example3.zip)\r\n\r\n**Environment info**\r\n- OS: macOS 10.14.6\r\n- Python version: 3.8.0\r\n- Sphinx version: 3.2.1 (also reproduced on the current 3.x branch)\r\n- Sphinx extensions:  `sphinx.ext.autodoc`\r\n\r\n**Additional context**\r\nThe reason for the bug is that `autodoc.GenericAliasDocumenter.add_directive_header` and `autodoc.TypeVarDocumenter.add_directive_header` use the line `self.options.annotation = SUPPRESS` to suppress the printing of the usual type information. Unfortunately, the `self.options` object can be shared across multiple documenters created as part of the same `automodule` directive, so mutating the options for one documenter affects all the others. One way to fix this is to change `autodoc.Documenter.__init__` to make a copy of the `Options` object, using the line `self.options = Options(directive.genopt)`. Making this change causes the bug described above to disappear.\r\n\r\nThanks to @altendky for helping me debug this problem.",
    "issue_closed_at": "2020-10-04T14:44:10Z",
    "base_commit": "38bb3774643d779b708970f941f2b16d1ab81b89",
    "changes": [
      {
        "file": "sphinx/ext/autodoc/__init__.py",
        "type": "function",
        "name": "can_document_member",
        "class_name": "SlotsAttributeDocumenter",
        "code": "def can_document_member(cls, member: Any, membername: str, isattr: bool, parent: Any\n                            ) -> bool:\n        \"\"\"This documents only SLOTSATTR members.\"\"\"\n        return member is SLOTSATTR"
      },
      {
        "file": "sphinx/ext/autodoc/__init__.py",
        "type": "function",
        "name": "can_document_member",
        "class_name": "SlotsAttributeDocumenter",
        "code": "def can_document_member(cls, member: Any, membername: str, isattr: bool, parent: Any\n                            ) -> bool:\n        \"\"\"This documents only SLOTSATTR members.\"\"\"\n        return member is SLOTSATTR"
      }
    ]
  },
  "Justification": "Candidate A is the most relevant bug report as it deals with the autodoc functionality of Sphinx, specifically how it handles type annotations and their documentation. This relates directly to the CURRENT bug, which also concerns the documentation generation for members of a class and how inherited attributes are treated. Both bugs arise from the autodoc behavior, including the issues surrounding showing expected attributes, making the insights from Candidate A's resolution helpful in understanding the underlying issue with `Bar.attr1`. Moreover, Candidate A describes a similar problem of attributes losing their documentation under specific circumstances, indicating a systemic problem in the autodoc implementation that could shed light on solutions applicable to the CURRENT bug.",
  "instance_id": "sphinx-doc__sphinx-8801",
  "repo": "sphinx-doc/sphinx",
  "created_at": "2021-01-31T11:12:59Z",
  "problem_statement": "autodoc: The annotation only member in superclass is treated as \"undocumented\"\n**Describe the bug**\r\nautodoc: The annotation only member in superclass is treated as \"undocumented\".\r\n\r\n**To Reproduce**\r\n\r\n```\r\n# example.py\r\nclass Foo:\r\n    \"\"\"docstring\"\"\"\r\n    attr1: int  #: docstring\r\n\r\n\r\nclass Bar(Foo):\r\n    \"\"\"docstring\"\"\"\r\n    attr2: str  #: docstring\r\n```\r\n```\r\n# index.rst\r\n.. autoclass:: example.Bar\r\n   :members:\r\n   :inherited-members:\r\n```\r\n\r\n`Bar.attr1` is not documented. It will be shown if I give `:undoc-members:` option to the autoclass directive call. It seems the attribute is treated as undocumented.\r\n\r\n**Expected behavior**\r\nIt should be shown.\r\n\r\n**Your project**\r\nNo\r\n\r\n**Screenshots**\r\nNo\r\n\r\n**Environment info**\r\n- OS: Mac\r\n- Python version: 3.9.1\r\n- Sphinx version: HEAD of 3.x\r\n- Sphinx extensions: sphinx.ext.autodoc\r\n- Extra tools: No\r\n\r\n**Additional context**\r\nNo\r\n\n",
  "patch": "diff --git a/sphinx/ext/autodoc/importer.py b/sphinx/ext/autodoc/importer.py\n--- a/sphinx/ext/autodoc/importer.py\n+++ b/sphinx/ext/autodoc/importer.py\n@@ -294,24 +294,35 @@ def get_class_members(subject: Any, objpath: List[str], attrgetter: Callable\n \n     try:\n         for cls in getmro(subject):\n+            try:\n+                modname = safe_getattr(cls, '__module__')\n+                qualname = safe_getattr(cls, '__qualname__')\n+                analyzer = ModuleAnalyzer.for_module(modname)\n+                analyzer.analyze()\n+            except AttributeError:\n+                qualname = None\n+                analyzer = None\n+            except PycodeError:\n+                analyzer = None\n+\n             # annotation only member (ex. attr: int)\n             for name in getannotations(cls):\n                 name = unmangle(cls, name)\n                 if name and name not in members:\n-                    members[name] = ObjectMember(name, INSTANCEATTR, class_=cls)\n+                    if analyzer and (qualname, name) in analyzer.attr_docs:\n+                        docstring = '\\n'.join(analyzer.attr_docs[qualname, name])\n+                    else:\n+                        docstring = None\n+\n+                    members[name] = ObjectMember(name, INSTANCEATTR, class_=cls,\n+                                                 docstring=docstring)\n \n             # append instance attributes (cf. self.attr1) if analyzer knows\n-            try:\n-                modname = safe_getattr(cls, '__module__')\n-                qualname = safe_getattr(cls, '__qualname__')\n-                analyzer = ModuleAnalyzer.for_module(modname)\n-                analyzer.analyze()\n+            if analyzer:\n                 for (ns, name), docstring in analyzer.attr_docs.items():\n                     if ns == qualname and name not in members:\n                         members[name] = ObjectMember(name, INSTANCEATTR, class_=cls,\n                                                      docstring='\\n'.join(docstring))\n-            except (AttributeError, PycodeError):\n-                pass\n     except AttributeError:\n         pass\n \n"
}