{
  "Selected_candidate": {
    "pr_number": 8387,
    "pr_title": "Fix #8372: autodoc: autoclass directive became slower than Sphinx-3.2",
    "pr_body": "### Feature or Bugfix\r\n- Bugfix\r\n\r\n### Purpose\r\n* The result of ModuleAnalyzer.parse() is not cached\r\n* autodoc tries to search overloaded constructor methods to the root\r\n  class even if a definition found",
    "issue_id": 8372,
    "issue_title": "Sphinx 3.3 is roughly 3.2x slower than sphinx 3.2.1",
    "issue_body": "**Describe the bug**\r\n\r\nWe recently noticed the doc builds for the 🤗 Transformers library were a lot slower with the recent release. We pinned the version to 3.2.1 for now and it solved the issue, but we wanted to let you know about the slowdown.\r\n\r\n**To Reproduce**\r\nSteps to reproduce the behavior:\r\n```\r\n$ git clone https://github.com/huggingface/transformers/\r\n$ cd transformers\r\n$ pip install --upgrade pip\r\n$ pip install .[tf,torch,sentencepiece,docs]\r\n$ cd docs\r\n$ make html SPHINXOPTS=\"-W\"\r\n```\r\nWith sphinx 3.3.0, the build takes 5min21s on a CircleCI Docker Medium (the size doesn't really impact execution time), whereas sphinx 3.2.1 completes the build in 1min40s (on the same machine). Most of the slowdown is coming in the \"[reading sources]\" part of the build.\r\n\r\n**Expected behavior**\r\nI'd expect the change of version to not impact the time of the build in such a way.\r\n\r\n**Your project**\r\nLink to your sphinx project, or attach zipped small project sample.\r\n\r\n**Environment info**\r\n- OS: Linux\r\n- Python version: 3.7.9\r\n- Sphinx version: 3.2.1/3.3.0\r\n- Sphinx extensions:\r\n    - 'sphinx.ext.autodoc',\r\n    - 'sphinx.ext.coverage',\r\n    - 'sphinx.ext.napoleon',\r\n    - 'recommonmark',\r\n    - 'sphinx.ext.viewcode',\r\n    - 'sphinx_markdown_tables',\r\n    - 'sphinx_copybutton'\r\n\r\n\r\n",
    "issue_closed_at": "2020-11-09T15:31:09Z",
    "base_commit": "1193d83166b7d889343d5aa0268d6c2e9349e692",
    "changes": [
      {
        "file": "sphinx/ext/autodoc/__init__.py",
        "type": "function",
        "name": "get_overloaded_signatures",
        "class_name": "ClassDocumenter",
        "code": "def get_overloaded_signatures(self) -> List[Signature]:\n        if self._signature_class and self._signature_method_name:\n            for cls in self._signature_class.__mro__:\n                try:\n                    analyzer = ModuleAnalyzer.for_module(cls.__module__)\n                    analyzer.parse()\n                    qualname = '.'.join([cls.__qualname__, self._signature_method_name])\n                    if qualname in analyzer.overloads:\n                        return analyzer.overloads.get(qualname)\n                except PycodeError:\n                    pass\n\n        return []"
      },
      {
        "file": "sphinx/pycode/__init__.py",
        "type": "function",
        "name": "__init__",
        "class_name": "ModuleAnalyzer",
        "code": "def __init__(self, source: IO, modname: str, srcname: str, decoded: bool = False) -> None:\n        self.modname = modname  # name of the module\n        self.srcname = srcname  # name of the source file\n\n        # cache the source code as well\n        pos = source.tell()\n        if not decoded:\n            warnings.warn('decode option for ModuleAnalyzer is deprecated.',\n                          RemovedInSphinx40Warning, stacklevel=2)\n            self._encoding, _ = tokenize.detect_encoding(source.readline)\n            source.seek(pos)\n            self.code = source.read().decode(self._encoding)\n        else:\n            self._encoding = None\n            self.code = source.read()\n\n        # will be filled by parse()\n        self.annotations = None  # type: Dict[Tuple[str, str], str]\n        self.attr_docs = None    # type: Dict[Tuple[str, str], List[str]]\n        self.finals = None       # type: List[str]\n        self.overloads = None    # type: Dict[str, List[Signature]]\n        self.tagorder = None     # type: Dict[str, int]\n        self.tags = None"
      },
      {
        "file": "sphinx/pycode/__init__.py",
        "type": "function",
        "name": "parse",
        "class_name": "ModuleAnalyzer",
        "code": "def parse(self) -> None:\n        \"\"\"Parse the source code.\"\"\"\n        try:\n            parser = Parser(self.code, self._encoding)\n            parser.parse()\n\n            self.attr_docs = OrderedDict()\n            for (scope, comment) in parser.comments.items():\n                if comment:\n                    self.attr_docs[scope] = comment.splitlines() + ['']\n                else:\n                    self.attr_docs[scope] = ['']\n\n            self.annotations = parser.annotations\n            self.finals = parser.finals\n            self.overloads = parser.overloads\n            self.tags = parser.definitions\n            self.tagorder = parser.deforders\n        except Exception as exc:\n            raise PycodeError('parsing %r failed: %r' % (self.srcname, exc)) from exc"
      }
    ]
  },
  "Justification": "Candidate C is the most helpful as it directly addresses a performance issue within the `sphinx.ext.autodoc` module, which is relevant to the CURRENT bug involving the autodoc functionality. Both reports pertain to the behavior of Sphinx's autodoc extension, and Candidate C specifically discusses optimizations to the `autoclass` directive, which can shed light on related issues with the installation or reading of modules. The shared focus on autodoc indicates that insights from the logic changes proposed in Candidate C can help identify why an empty `__all__` isn't behaving as expected in the CURRENT bug.",
  "instance_id": "sphinx-doc__sphinx-8595",
  "repo": "sphinx-doc/sphinx",
  "created_at": "2020-12-27T03:07:50Z",
  "problem_statement": "autodoc: empty __all__ attribute is ignored\n**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\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@@ -1074,7 +1074,7 @@ def get_module_members(self) -> Dict[str, ObjectMember]:\n     def get_object_members(self, want_all: bool) -> Tuple[bool, ObjectMembers]:\n         members = self.get_module_members()\n         if want_all:\n-            if not self.__all__:\n+            if self.__all__ is None:\n                 # for implicit module members, check __module__ to avoid\n                 # documenting imported objects\n                 return True, list(members.values())\n"
}