{
  "instance_id": "sphinx-doc__sphinx-10325",
  "repo": "sphinx-doc/sphinx",
  "created_at": "2022-04-02T17:05:02Z",
  "problem_statement": "inherited-members should support more than one class\n**Is your feature request related to a problem? Please describe.**\r\nI have two situations:\r\n- A class inherits from multiple other classes. I want to document members from some of the base classes but ignore some of the base classes\r\n- A module contains several class definitions that inherit from different classes that should all be ignored (e.g., classes that inherit from list or set or tuple). I want to ignore members from list, set, and tuple while documenting all other inherited members in classes in the module.\r\n\r\n**Describe the solution you'd like**\r\nThe :inherited-members: option to automodule should accept a list of classes. If any of these classes are encountered as base classes when instantiating autoclass documentation, they should be ignored.\r\n\r\n**Describe alternatives you've considered**\r\nThe alternative is to not use automodule, but instead manually enumerate several autoclass blocks for a module. This only addresses the second bullet in the problem description and not the first. It is also tedious for modules containing many class definitions.\r\n\r\n\n",
  "patch": "diff --git a/sphinx/ext/autodoc/__init__.py b/sphinx/ext/autodoc/__init__.py\n--- a/sphinx/ext/autodoc/__init__.py\n+++ b/sphinx/ext/autodoc/__init__.py\n@@ -109,12 +109,14 @@ def exclude_members_option(arg: Any) -> Union[object, Set[str]]:\n     return {x.strip() for x in arg.split(',') if x.strip()}\n \n \n-def inherited_members_option(arg: Any) -> Union[object, Set[str]]:\n+def inherited_members_option(arg: Any) -> Set[str]:\n     \"\"\"Used to convert the :members: option to auto directives.\"\"\"\n     if arg in (None, True):\n-        return 'object'\n+        return {'object'}\n+    elif arg:\n+        return set(x.strip() for x in arg.split(','))\n     else:\n-        return arg\n+        return set()\n \n \n def member_order_option(arg: Any) -> Optional[str]:\n@@ -680,9 +682,11 @@ def filter_members(self, members: ObjectMembers, want_all: bool\n         ``autodoc-skip-member`` event.\n         \"\"\"\n         def is_filtered_inherited_member(name: str, obj: Any) -> bool:\n+            inherited_members = self.options.inherited_members or set()\n+\n             if inspect.isclass(self.object):\n                 for cls in self.object.__mro__:\n-                    if cls.__name__ == self.options.inherited_members and cls != self.object:\n+                    if cls.__name__ in inherited_members and cls != self.object:\n                         # given member is a member of specified *super class*\n                         return True\n                     elif name in cls.__dict__:\n",
  "similar_bug_items": [
    {
      "pr_number": 10011,
      "pr_title": "Fix #10009: autodoc: Crashes if subject raises an error on getdoc()",
      "pr_body": "### Feature or Bugfix\r\n- Bugfix\r\n\r\n### Purpose\r\n- refs: #10009 ",
      "issue_id": 10009,
      "issue_title": "SystemError when trying to build a doc with Pyside6 resources",
      "issue_body": "### Describe the bug\n\nHi,\r\n\r\nWe have a fairly large project with the following folder structure:\r\n\r\n- docs\r\n- package\r\n- tests\r\n- ui\r\n  - \\_\\_init__.py\r\n  - resources\r\n    - gui.qrc\r\n    - icons\r\n       - icons1.png\r\n       - icon2.png\r\n       - ...\r\n\r\n\r\n`ui/resources` is actually not a valid Python package, but before running the GUI, we use the command `pyside6-rcc ui/resources/gui.qrc -o ui/resources.py`\r\n\r\nThe file `ui\\__init__.py` contains the line `from . import resources`\r\n\r\nWhen we build the doc with sphinx , we get the exception:\r\n\r\n```\r\n  File \"path\\to\\python\\lib\\inspect.py\", line 614, in getdoc\r\n    doc = object.__doc__\r\nSystemError: null argument to internal routine\r\n```\r\n\r\nThe log itself is not more useful either :\r\n\r\n\r\n```\r\n# Sphinx version: 4.3.2\r\n# Python version: 3.9.9 (CPython)\r\n# Docutils version: 0.16 release\r\n# Jinja2 version: 2.11.3\r\n# Last messages:\r\n#   Lecture des sources... [ 43%] package.module1.submodule1\r\n#   Lecture des sources... [ 45%] package.submodules\r\n#   Lecture des sources... [ 47%] index\r\n#   Lecture des sources... [ 50%] modules\r\n#   Lecture des sources... [ 52%] tests\r\n#   Lecture des sources... [ 54%] tests.test_module1\r\n#   Lecture des sources... [ 56%] tests.test_module2\r\n#   Lecture des sources... [ 58%] tests.test_module2.submodule1\r\n#   Lecture des sources... [ 60%] tests.test_module2.submodule2\r\n# Loaded extensions:\r\n#   sphinx.ext.mathjax (4.3.2) from path\\to\\venv\\lib\\site-packages\\sphinx\\ext\\mathjax.py\r\n#   sphinxcontrib.applehelp (1.0.2) from path\\to\\venv\\lib\\site-packages\\sphinxcontrib\\applehelp\\__init__.py\r\n#   sphinxcontrib.devhelp (1.0.2) from path\\to\\venv\\lib\\site-packages\\sphinxcontrib\\devhelp\\__init__.py\r\n#   sphinxcontrib.htmlhelp (2.0.0) from path\\to\\venv\\lib\\site-packages\\sphinxcontrib\\htmlhelp\\__init__.py\r\n#   sphinxcontrib.serializinghtml (1.1.5) from path\\to\\venv\\lib\\site-packages\\sphinxcontrib\\serializinghtml\\__init__.py\r\n#   sphinxcontrib.qthelp (1.0.3) from path\\to\\venv\\lib\\site-packages\\sphinxcontrib\\qthelp\\__init__.py\r\n#   alabaster (0.7.12) from path\\to\\venv\\lib\\site-packages\\alabaster\\__init__.py\r\n#   sphinx_rtd_theme (unknown version) from path\\to\\venv\\lib\\site-packages\\sphinx_rtd_theme\\__init__.py\r\n#   sphinx.ext.autodoc.preserve_defaults (1.0) from path\\to\\venv\\lib\\site-packages\\sphinx\\ext\\autodoc\\preserve_defaults.py\r\n#   sphinx.ext.autodoc.type_comment (4.3.2) from path\\to\\venv\\lib\\site-packages\\sphinx\\ext\\autodoc\\type_comment.py\r\n#   sphinx.ext.autodoc (4.3.2) from path\\to\\venv\\lib\\site-packages\\sphinx\\ext\\autodoc\\__init__.py\r\n#   matplotlib.sphinxext.plot_directive (3.5.0) from path\\to\\venv\\lib\\site-packages\\matplotlib\\sphinxext\\plot_directive.py\r\nTraceback (most recent call last):\r\n  File \"path\\to\\venv\\lib\\site-packages\\sphinx\\cmd\\build.py\", line 280, in build_main\r\n    app.build(args.force_all, filenames)\r\n  File \"path\\to\\venv\\lib\\site-packages\\sphinx\\application.py\", line 344, in build\r\n    self.builder.build_update()\r\n  File \"path\\to\\venv\\lib\\site-packages\\sphinx\\builders\\__init__.py\", line 294, in build_update\r\n    self.build(to_build,\r\n  File \"path\\to\\venv\\lib\\site-packages\\sphinx\\builders\\__init__.py\", line 308, in build\r\n    updated_docnames = set(self.read())\r\n  File \"path\\to\\venv\\lib\\site-packages\\sphinx\\builders\\__init__.py\", line 415, in read\r\n    self._read_serial(docnames)\r\n  File \"path\\to\\venv\\lib\\site-packages\\sphinx\\builders\\__init__.py\", line 436, in _read_serial\r\n    self.read_doc(docname)\r\n  File \"path\\to\\venv\\lib\\site-packages\\sphinx\\builders\\__init__.py\", line 476, in read_doc\r\n    doctree = read_doc(self.app, self.env, self.env.doc2path(docname))\r\n  File \"path\\to\\venv\\lib\\site-packages\\sphinx\\io.py\", line 189, in read_doc\r\n    pub.publish()\r\n  File \"path\\to\\venv\\lib\\site-packages\\docutils\\core.py\", line 217, in publish\r\n    self.document = self.reader.read(self.source, self.parser,\r\n  File \"path\\to\\venv\\lib\\site-packages\\sphinx\\io.py\", line 109, in read\r\n    self.parse()\r\n  File \"path\\to\\venv\\lib\\site-packages\\docutils\\readers\\__init__.py\", line 77, in parse\r\n    self.parser.parse(self.input, document)\r\n  File \"path\\to\\venv\\lib\\site-packages\\sphinx\\parsers.py\", line 101, in parse\r\n    self.statemachine.run(inputlines, document, inliner=self.inliner)\r\n  File \"path\\to\\venv\\lib\\site-packages\\docutils\\parsers\\rst\\states.py\", line 170, in run\r\n    results = StateMachineWS.run(self, input_lines, input_offset,\r\n  File \"path\\to\\venv\\lib\\site-packages\\docutils\\statemachine.py\", line 241, in run\r\n    context, next_state, result = self.check_line(\r\n  File \"path\\to\\venv\\lib\\site-packages\\docutils\\statemachine.py\", line 459, in check_line\r\n    return method(match, context, next_state)\r\n  File \"path\\to\\venv\\lib\\site-packages\\docutils\\parsers\\rst\\states.py\", line 2769, in underline\r\n    self.section(title, source, style, lineno - 1, messages)\r\n  File \"path\\to\\venv\\lib\\site-packages\\docutils\\parsers\\rst\\states.py\", line 327, in section\r\n    self.new_subsection(title, lineno, messages)\r\n  File \"path\\to\\venv\\lib\\site-packages\\docutils\\parsers\\rst\\states.py\", line 393, in new_subsection\r\n    newabsoffset = self.nested_parse(\r\n  File \"path\\to\\venv\\lib\\site-packages\\docutils\\parsers\\rst\\states.py\", line 281, in nested_parse\r\n    state_machine.run(block, input_offset, memo=self.memo,\r\n  File \"path\\to\\venv\\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 \"path\\to\\venv\\lib\\site-packages\\docutils\\statemachine.py\", line 241, in run\r\n    context, next_state, result = self.check_line(\r\n  File \"path\\to\\venv\\lib\\site-packages\\docutils\\statemachine.py\", line 459, in check_line\r\n    return method(match, context, next_state)\r\n  File \"path\\to\\venv\\lib\\site-packages\\docutils\\parsers\\rst\\states.py\", line 2769, in underline\r\n    self.section(title, source, style, lineno - 1, messages)\r\n  File \"path\\to\\venv\\lib\\site-packages\\docutils\\parsers\\rst\\states.py\", line 327, in section\r\n    self.new_subsection(title, lineno, messages)\r\n  File \"path\\to\\venv\\lib\\site-packages\\docutils\\parsers\\rst\\states.py\", line 393, in new_subsection\r\n    newabsoffset = self.nested_parse(\r\n  File \"path\\to\\venv\\lib\\site-packages\\docutils\\parsers\\rst\\states.py\", line 281, in nested_parse\r\n    state_machine.run(block, input_offset, memo=self.memo,\r\n  File \"path\\to\\venv\\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 \"path\\to\\venv\\lib\\site-packages\\docutils\\statemachine.py\", line 241, in run\r\n    context, next_state, result = self.check_line(\r\n  File \"path\\to\\venv\\lib\\site-packages\\docutils\\statemachine.py\", line 459, in check_line\r\n    return method(match, context, next_state)\r\n  File \"path\\to\\venv\\lib\\site-packages\\docutils\\parsers\\rst\\states.py\", line 2342, in explicit_markup\r\n    nodelist, blank_finish = self.explicit_construct(match)\r\n  File \"path\\to\\venv\\lib\\site-packages\\docutils\\parsers\\rst\\states.py\", line 2354, in explicit_construct\r\n    return method(self, expmatch)\r\n  File \"path\\to\\venv\\lib\\site-packages\\docutils\\parsers\\rst\\states.py\", line 2096, in directive\r\n    return self.run_directive(\r\n  File \"path\\to\\venv\\lib\\site-packages\\docutils\\parsers\\rst\\states.py\", line 2146, in run_directive\r\n    result = directive_instance.run()\r\n  File \"path\\to\\venv\\lib\\site-packages\\sphinx\\ext\\autodoc\\directive.py\", line 162, in run\r\n    documenter.generate(more_content=self.content)\r\n  File \"path\\to\\venv\\lib\\site-packages\\sphinx\\ext\\autodoc\\__init__.py\", line 984, in generate\r\n    self.document_members(all_members)\r\n  File \"path\\to\\venv\\lib\\site-packages\\sphinx\\ext\\autodoc\\__init__.py\", line 842, in document_members\r\n    for (mname, member, isattr) in self.filter_members(members, want_all):\r\n  File \"path\\to\\venv\\lib\\site-packages\\sphinx\\ext\\autodoc\\__init__.py\", line 723, in filter_members\r\n    doc = getdoc(member, self.get_attr, self.config.autodoc_inherit_docstrings,\r\n  File \"path\\to\\venv\\lib\\site-packages\\sphinx\\util\\inspect.py\", line 912, in getdoc\r\n    doc = inspect.getdoc(obj)\r\n  File \"path\\to\\python\\lib\\inspect.py\", line 614, in getdoc\r\n    doc = object.__doc__\r\nSystemError: null argument to internal routine\r\n\r\n```\r\n\r\nWhen we build the doc from the sources, we get this error.\r\nWhen we build it after generating `ui\\resources.py`, we get this error\r\nWhen we build it after generating `ui\\resources.py` and deleting the folder `ui\\resources`, we get this error\r\nWhen we build it after deleting the `ui\\resources` and the file `ui\\resources.py`, it works\r\n\r\n\n\n### How to Reproduce\n\nI am trying to create a minimal reproducable example but my attempts failed so far\r\n\r\nThe doc itself was built running the following powershell script\r\n\r\n```\r\ncd .\\docs\r\n\r\n# Remove previously generated auto-doc and plots\r\nRemove-Item .\\source\\* -Include *.rst -Exclude index.rst\r\n.\\make.bat clean\r\n\r\n# Generate source files for documentation from docstrings\r\nsphinx-apidoc -f -o source/ ../\r\n\r\n# build documentation\r\n.\\make.bat html\r\n```\r\n\n\n### Expected behavior\n\nThe main issue I had is not the error itself, as we are going to change how we package the pyside6 resources to the gui to fix it, but the fact that neither the error message nor the log pointed towards the issue\r\n\r\nTo find the culprit, I started to delete folders of my projet until I managed to build the doc and finally find the culprit.\n\n### Your project\n\nStill working on a minimal reproducible example\n\n### Screenshots\n\n_No response_\n\n### OS\n\nWindows 10\n\n### Python version\n\n3.9.9\n\n### Sphinx version\n\n4.3.2\n\n### Sphinx extensions\n\nsphinx_rtd_theme, sphinx.ext.autodoc, matplotlib.sphinxext.plot_directive'\n\n### Extra tools\n\nPyside6==6.2.2.1\n\n### Additional context\n\nThanks a lot for providing a great documentation tool !",
      "issue_closed_at": "2021-12-28T17:07:48Z",
      "base_commit": "8ddf3f09c62e2a4651458759fcc97e751ca63063",
      "changes": [
        {
          "file": "sphinx/ext/autodoc/__init__.py",
          "type": "function",
          "name": "is_filtered_inherited_member",
          "class_name": "Documenter",
          "code": "def is_filtered_inherited_member(name: str, obj: Any) -> bool:\n            if inspect.isclass(self.object):\n                for cls in self.object.__mro__:\n                    if cls.__name__ == self.options.inherited_members and cls != self.object:\n                        # given member is a member of specified *super class*\n                        return True\n                    elif name in cls.__dict__:\n                        return False\n                    elif name in self.get_attr(cls, '__annotations__', {}):\n                        return False\n                    elif isinstance(obj, ObjectMember) and obj.class_ is cls:\n                        return False\n\n            return False"
        }
      ]
    },
    {
      "pr_number": 8147,
      "pr_title": "Fixes #8146: When identifying bases, only use classes from builtins",
      "pr_body": "Subject: When identifying bases, only use class objects from `builtins`\r\n\r\n### Feature or Bugfix\r\n- Feature\r\n- Minor refactoring\r\n\r\n### Purpose\r\nIn inheritance_diagram extension, while iterating over bases,\r\nwe verify if the base class is one of the Python built-in\r\nclass or not.\r\nAs of now, we simply check for its presence in all `builtins`\r\nobjects. Please note, `builtins` not only has built-in classes,\r\nbut also functions (like `open`) and other built-in objects.\r\n\r\nTo avoid any sort of future problem, it seems better to only\r\nuse classes (and of course exception classes).\r\n\r\n### Relates\r\n- Issue #8146 ",
      "issue_id": 8146,
      "issue_title": "[inheritance_diagram] Use class objects only from builtins while evaluating bases",
      "issue_body": "**Is your feature request related to a problem? Please describe.**\r\nIn `sphinx.ext.inheritance_diagram` extension, we iterate over all bases and verify if a class object belongs to one of the Python built-in classes. As of now, we simply check for its presence in all `builtins` objects. This may be simple but we are ignoring the fact that besides classes, `builtins` also contains functions and other built-in Python objects. \r\n\r\nThe above is a direct cause of the following bug:\r\n\r\n**Prerequisite code:**\r\n```\r\nimport builtins\r\nimport inspect\r\nclass classproperty(object):\r\n    def __init__(self, f):\r\n        self.f = f\r\n    def __get__(self, obj, owner):\r\n        return self.f(owner)\r\n\r\nclass NamedAction(object):\r\n    @classproperty\r\n    def name(self):\r\n        raise NotImplementedError()\r\n```\r\n\r\n**As of now:**\r\n```\r\npy_builtins = vars(builtins).values()    # As used in sphinx.ext.inheritance_diagram\r\ncls = NamedAction                        # defined above\r\ncls in py_builtins\r\n---------------------------------------------------------------------------\r\nNotImplementedError                       Traceback (most recent call last)\r\n<ipython-input-6-2e6b4af0d81c> in <module>\r\n----> 1 cls in py_builtins\r\n\r\n/usr/lib/python3.8/importlib/_bootstrap.py in __eq__(self, other)\r\n\r\n<ipython-input-3-a8bb3f065bd0> in __get__(self, obj, owner)\r\n      3         self.f = f\r\n      4     def __get__(self, obj, owner):\r\n----> 5         return self.f(owner)\r\n      6 \r\n\r\n<ipython-input-4-b0467820c849> in name(self)\r\n      2     @classproperty\r\n      3     def name(self):\r\n----> 4         raise NotImplementedError()\r\n      5 \r\n\r\nNotImplementedError: \r\n```\r\n\r\n**After the proposed fix:**\r\n```\r\npy_builtins = [obj for obj in vars(builtins).values()\r\n               if inspect.isclass(obj)]    # proposed fix\r\ncls = NamedAction                          # defined above\r\ncls in py_builtins\r\nFalse\r\n```\r\n\r\n**Describe the solution you'd like**\r\nThe solution here is pretty simple. Instead of relying on the entire `builtins`, let's separate out the classes via `inspect.isclass`\r\n\r\n**Describe alternatives you've considered**\r\nN/A\r\n\r\n**Additional context**\r\nN/A",
      "issue_closed_at": "2020-09-20T09:02:12Z",
      "base_commit": "ae9f0dd29998a9985aea90777e92e03cf1c9d591",
      "changes": [
        {
          "file": "sphinx/ext/inheritance_diagram.py",
          "type": "function",
          "name": "_class_info",
          "class_name": "InheritanceGraph",
          "code": "def _class_info(self, classes: List[Any], show_builtins: bool, private_bases: bool,\n                    parts: int, aliases: Dict[str, str], top_classes: List[Any]\n                    ) -> List[Tuple[str, str, List[str], str]]:\n        \"\"\"Return name and bases for all classes that are ancestors of\n        *classes*.\n\n        *parts* gives the number of dotted name parts to include in the\n        displayed node names, from right to left. If given as a negative, the\n        number of parts to drop from the left. A value of 0 displays the full\n        dotted name. E.g. ``sphinx.ext.inheritance_diagram.InheritanceGraph``\n        with ``parts=2`` or ``parts=-2`` gets displayed as\n        ``inheritance_diagram.InheritanceGraph``, and as\n        ``ext.inheritance_diagram.InheritanceGraph`` with ``parts=3`` or\n        ``parts=-1``.\n\n        *top_classes* gives the name(s) of the top most ancestor class to\n        traverse to. Multiple names can be specified separated by comma.\n        \"\"\"\n        all_classes = {}\n        py_builtins = vars(builtins).values()\n\n        def recurse(cls: Any) -> None:\n            if not show_builtins and cls in py_builtins:\n                return\n            if not private_bases and cls.__name__.startswith('_'):\n                return\n\n            nodename = self.class_name(cls, parts, aliases)\n            fullname = self.class_name(cls, 0, aliases)\n\n            # Use first line of docstring as tooltip, if available\n            tooltip = None\n            try:\n                if cls.__doc__:\n                    doc = cls.__doc__.strip().split(\"\\n\")[0]\n                    if doc:\n                        tooltip = '\"%s\"' % doc.replace('\"', '\\\\\"')\n            except Exception:  # might raise AttributeError for strange classes\n                pass\n\n            baselist = []  # type: List[str]\n            all_classes[cls] = (nodename, fullname, baselist, tooltip)\n\n            if fullname in top_classes:\n                return\n\n            for base in cls.__bases__:\n                if not show_builtins and base in py_builtins:\n                    continue\n                if not private_bases and base.__name__.startswith('_'):\n                    continue\n                baselist.append(self.class_name(base, parts, aliases))\n                if base not in all_classes:\n                    recurse(base)\n\n        for cls in classes:\n            recurse(cls)\n\n        return list(all_classes.values())"
        }
      ]
    },
    {
      "pr_number": 8595,
      "pr_title": "Fix #8594: autodoc: empty __all__ attribute is ignored",
      "pr_body": "### Feature or Bugfix\r\n- Bugfix\r\n\r\n### Purpose\r\n- An empty `__all__` should be represented as \"there is no public items\".\r\nBut autodoc considers all items on the module are public.  This changes\r\nthe behavior to correct one.\r\n- #8594 ",
      "issue_id": 8594,
      "issue_title": "autodoc: empty __all__ attribute is ignored",
      "issue_body": "**Describe the bug**\r\nautodoc: empty `__all__` attribute is ignored\r\n\r\n**To Reproduce**\r\n```\r\n# example.py\r\n__all__ = []\r\n\r\n\r\ndef foo():\r\n    \"docstring\"\r\n\r\n\r\ndef bar():\r\n    \"docstring\"\r\n\r\n\r\ndef baz():\r\n    \"docstring\"\r\n```\r\n```\r\n# index.rst\r\n.. automodule:: example\r\n   :members:\r\n```\r\n\r\nAll foo, bar, and baz are shown.\r\n\r\n**Expected behavior**\r\nNo entries should be shown because `__all__` is empty.\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",
      "issue_closed_at": "2020-12-28T07:47:35Z",
      "base_commit": "b19bce971e82f2497d67fdacdeca8db08ae0ba56",
      "changes": [
        {
          "file": "sphinx/ext/autodoc/__init__.py",
          "type": "function",
          "name": "get_module_members",
          "class_name": "ModuleDocumenter",
          "code": "def get_module_members(self) -> Dict[str, ObjectMember]:\n        \"\"\"Get members of target module.\"\"\"\n        if self.analyzer:\n            attr_docs = self.analyzer.attr_docs\n        else:\n            attr_docs = {}\n\n        members = {}  # type: Dict[str, ObjectMember]\n        for name in dir(self.object):\n            try:\n                value = safe_getattr(self.object, name, None)\n                docstring = attr_docs.get(('', name), [])\n                members[name] = ObjectMember(name, value, docstring=\"\\n\".join(docstring))\n            except AttributeError:\n                continue\n\n        # annotation only member (ex. attr: int)\n        try:\n            for name in inspect.getannotations(self.object):\n                if name not in members:\n                    docstring = attr_docs.get(('', name), [])\n                    members[name] = ObjectMember(name, INSTANCEATTR,\n                                                 docstring=\"\\n\".join(docstring))\n        except AttributeError:\n            pass\n\n        return members"
        }
      ]
    },
    {
      "pr_number": 2299,
      "pr_title": "Fix #2298: automodule fails to document a class attribute.",
      "pr_body": "This pull request adds a test case for bug #2298 (commit de356149cd) and a fix (commit c44608234d).\n\nI didn't create a new test module but extended test_ext_viewcode, because the sphinx dev guide recommends it for performance reasons. The test fails until the second commit get applied.\n\nMy fix is not the only possible fix, but a fairly simple one. It surely needs to be reviewed. Especially I'm not completely sure about its impact on extension classes written in C.\n",
      "issue_id": 2298,
      "issue_title": "automodule fails to document a class attribute",
      "issue_body": "The directive automodule:: fails to document a class attribute under the following conditions:\n\nModule mod1 defines a class C:\n\n``` python\nclass C(object):\n    class_attr = 42\n    \"\"\"This is the class attribute class_attr\"\"\"\n```\n\nModule mod2 imports C from mod1:\n\n``` python\nfrom mod1 import C\n__all__ = ('C',)  # required to trigger the bug\n```\n\nNow document module mod2 using automodule:\n\n```\n.. automodule:: mod2\n   :members:\n```\n\nThe resulting documentation won't contain the doc string for the class attribute class_attr.\n\nThe bug is caused by a call of  class_documenter.generate() with the optional arguments\n`check_module=False` and `real_modname=\"mod2\"`. This causes the analyser to look for the source of class C in mod2 instead of mod1. As a consequence it doesn't find the doc-string of class_attr and fails to document it.\n\nI'll create a pull request with a test case and a fix.\n",
      "issue_closed_at": "2017-11-11T16:53:43Z",
      "base_commit": "bdc230b1ccc189989789f82a55023f369410309d",
      "changes": [
        {
          "file": "sphinx/ext/autodoc.py",
          "type": "function",
          "name": "document_members",
          "class_name": "AttributeDocumenter",
          "code": "def document_members(self, all_members=False):\n        pass"
        }
      ]
    },
    {
      "pr_number": 8571,
      "pr_title": "Fix #8567: autodoc: Instance attributes are incorrectly added to Parent class",
      "pr_body": "### Feature or Bugfix\r\n- Bugfix\r\n\r\n### Purpose\r\n- The instance attributes on subclasses are shown on the document of\r\nparent class unexpectedly because of autodoc modifies `__annotations__`\r\nin place.  This fix creates a copy of `__annotations__` attribute and\r\nattach it to the subclass.\r\n- refs: #8567 ",
      "issue_id": 8567,
      "issue_title": "Instance attributes are incorrectly added to Parent class with Sphinx 3.4.0",
      "issue_body": "**Describe the bug**\r\n\r\nTrying to build the docs of QCoDeS https://github.com/QCoDeS/Qcodes/pull/2549 with Sphinx 3.4.0 we are \r\nseeing a failure where an instance attribute on a class cannot be resolved. This attribute is not defined on that class \r\nbut only on a different subclass. e.g. `visabackend` is defined on `VisaInstrument` but not on the parent class of `VisaInstrument` called `Instrument` but Sphinx tries to get that attribute on a non related subclass of Instrument `ATS`\r\n\r\nThis happens because Sphinx modifies `__attributes__` in place. If a subclass does not define new class attributes it will reuse the `__attributes__` dict of the super class as seen in the following simple example:\r\n\r\n\r\n```python\r\nclass X:\r\n    a: int = 1 \r\n    def __init__(self):\r\n        self.b: int = 2\r\n            \r\nclass Y(X):\r\n    def __init__(self):\r\n        self.d: int = 4\r\n\r\nid(Y.__annotations__)\r\n3051582812872\r\n\r\nid(X.__annotations__)\r\n3051582812872\r\n\r\nid(Y.__annotations__) == id(X.__annotations__)\r\nTrue\r\n\r\n```\r\nAnd `AttributeDocumenter.update_annotations` modifies the annotation dictionary of the subclass in place causing the annotation dictionary of the super class to be modified with it. \r\n\r\nReplacing this with something that does \r\n\r\n```python\r\nannotations = deepcopy(inspect.getannotations(parent))\r\n\r\nfor cls in inspect.getmro(parent):\r\n   .... (existing code to update annotations)\r\n\r\nparent.__annotations__ = annotations\r\n```\r\nSeems to resolve the issue\r\n\r\n**To Reproduce**\r\n\r\n```\r\n\r\n$ sudo apt install pandoc\r\n$ git clone https://github.com/QCoDeS/Qcodes.git\r\n$ cd qcodes\r\n$ pip install -r requirements.txt -r docs_requirements.txt\r\n$ cd docs\r\n$ make htmlfast\r\n```\r\n\r\n**Expected behavior**\r\nOnly the relevant attributes are attempted to be imported and the docs build correctly.\r\n\r\n**Your project**\r\ngithub.com/qcodes/qcodes\r\n\r\n**Screenshots**\r\nIf applicable, add screenshots to help explain your problem.\r\n\r\n**Environment info**\r\n- OS: Linux Ubnutu 18.04 (Github actions )\r\n- Python version: 3.\r\n- Sphinx version: 3.4.0\r\n- Sphinx extensions:  e.g. ['nbsphinx', 'sphinx.ext.autodoc', 'sphinx.ext.autosummary',\r\n              'sphinx.ext.napoleon', 'sphinx-jsonschema', 'sphinx.ext.doctest',\r\n              'sphinx.ext.intersphinx', 'sphinx.ext.todo',\r\n              'sphinx.ext.coverage', 'sphinx.ext.mathjax',\r\n              'sphinx.ext.viewcode', 'sphinx.ext.githubpages',\r\n              'sphinx.ext.todo']\r\n",
      "issue_closed_at": "2020-12-22T14:41:02Z",
      "base_commit": "31cad2ebe7a205154e1374bfa52338111e515719",
      "changes": [
        {
          "file": "sphinx/ext/autodoc/__init__.py",
          "type": "function",
          "name": "can_document_member",
          "class_name": "NewTypeAttributeDocumenter",
          "code": "def can_document_member(cls, member: Any, membername: str, isattr: bool, parent: Any\n                            ) -> bool:\n        return not isinstance(parent, ModuleDocumenter) and inspect.isNewType(member)"
        },
        {
          "file": "sphinx/ext/autodoc/__init__.py",
          "type": "function",
          "name": "isinstanceattribute",
          "class_name": "AttributeDocumenter",
          "code": "def isinstanceattribute(self) -> bool:\n        \"\"\"Check the subject is an instance attribute.\"\"\"\n        # uninitialized instance variable (PEP-526)\n        with mock(self.config.autodoc_mock_imports):\n            try:\n                ret = import_object(self.modname, self.objpath[:-1], 'class',\n                                    attrgetter=self.get_attr,\n                                    warningiserror=self.config.autodoc_warningiserror)\n                self.parent = ret[3]\n                annotations = get_type_hints(self.parent, None,\n                                             self.config.autodoc_type_aliases)\n                if self.objpath[-1] in annotations:\n                    self.object = UNINITIALIZED_ATTR\n                    return True\n            except ImportError:\n                pass\n\n        return False"
        },
        {
          "file": "sphinx/ext/autodoc/__init__.py",
          "type": "function",
          "name": "update_annotations",
          "class_name": "AttributeDocumenter",
          "code": "def update_annotations(self, parent: Any) -> None:\n        \"\"\"Update __annotations__ to support type_comment and so on.\"\"\"\n        try:\n            annotations = inspect.getannotations(parent)\n\n            for cls in inspect.getmro(parent):\n                try:\n                    module = safe_getattr(cls, '__module__')\n                    qualname = safe_getattr(cls, '__qualname__')\n\n                    analyzer = ModuleAnalyzer.for_module(module)\n                    analyzer.analyze()\n                    for (classname, attrname), annotation in analyzer.annotations.items():\n                        if classname == qualname and attrname not in annotations:\n                            annotations[attrname] = annotation  # type: ignore\n                except (AttributeError, PycodeError):\n                    pass\n        except AttributeError:\n            pass"
        }
      ]
    }
  ]
}