{
  "Selected_candidate": {
    "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"
      }
    ]
  },
  "Justification": "Candidate D is the most helpful as it deals directly with issues regarding class attributes in the context of Sphinx's autodoc. The underlying mechanism causing attributes to be undocumented is relevant to the CURRENT bug, which seeks to control which inherited members are documented. Understanding and applying the fix from Candidate D can provide critical insights into managing document generation when dealing with complex inheritance hierarchies.",
  "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"
}