{
  "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",
  "similar_bug_items": [
    {
      "pr_number": 8289,
      "pr_title": "Allow to suppress \"duplicated toc entry\" warnings from epub builder",
      "pr_body": "Subject:  Allow to suppress \"duplicated toc entry\" warnings from epub builder #8289 \r\n\r\n\r\n### Feature or Bugfix\r\n<!-- please choose -->\r\n- Feature\r\n\r\n### Purpose\r\nI'd like to suppress this warning, while I find time to fix it\r\n\r\n### Detail\r\n\r\n### Relates\r\n- https://github.com/sphinx-doc/sphinx/commit/6d900c34f12db0d21c581c58a5dd38ab49c8ea35\r\n\r\n",
      "issue_id": 8289,
      "issue_title": "Allow to suppress \"duplicated toc entry\" warnings from epub builder",
      "issue_body": "Subject:  Allow to suppress \"duplicated toc entry\" warnings from epub builder #8289 \r\n\r\n\r\n### Feature or Bugfix\r\n<!-- please choose -->\r\n- Feature\r\n\r\n### Purpose\r\nI'd like to suppress this warning, while I find time to fix it\r\n\r\n### Detail\r\n\r\n### Relates\r\n- https://github.com/sphinx-doc/sphinx/commit/6d900c34f12db0d21c581c58a5dd38ab49c8ea35\r\n\r\n",
      "issue_closed_at": "2020-10-24T11:29:07Z",
      "base_commit": "a8abb9995f71b9bc02b6f83592751c779ae0f75a",
      "changes": [
        {
          "file": "sphinx/builders/_epub_base.py",
          "type": "function",
          "name": "check_refnodes",
          "class_name": "EpubBuilder",
          "code": "def check_refnodes(self, nodes: List[Dict[str, Any]]) -> None:\n        appeared = set()  # type: Set[str]\n        for node in nodes:\n            if node['refuri'] in appeared:\n                logger.warning(__('duplicated ToC entry found: %s'), node['refuri'])\n            else:\n                appeared.add(node['refuri'])"
        }
      ]
    },
    {
      "pr_number": 6532,
      "pr_title": "Fix #6531: Failed to load last environment object when extension added",
      "pr_body": "### Feature or Bugfix\r\n- Bugfix\r\n\r\n### Purpose\r\n- refs: https://github.com/sphinx-doc/sphinx/pull/6419/files#r295377826\r\n",
      "issue_id": 6531,
      "issue_title": "Failed to load last environment object if extension added",
      "issue_body": "**Describe the bug**\r\nFailed to load last environment object if extension added.\r\n\r\n**To Reproduce**\r\nSteps to reproduce the behavior:\r\n\r\n1. Create a new project\r\n2. Build HTML once\r\n3. Add a some extension to conf.py\r\n4. Build HTML again\r\n\r\n```\r\nRunning Sphinx v2.2.0+\r\nloading pickled environment... failed\r\nfailed: No such config value: autosummary_generate\r\nbuilding [mo]: targets for 0 po files that are out of date\r\nbuilding [html]: targets for 4 source files that are out of date\r\n...\r\n```\r\n\r\n**Expected behavior**\r\nLoading succeeded.\r\n\r\n**Your project**\r\nNone\r\n\r\n**Screenshots**\r\nNone\r\n\r\n**Environment info**\r\n- OS: Mac\r\n- Python version: 3.7.3\r\n- Sphinx version: 2.2.0 (dev)\r\n- Sphinx extensions: sphinx.ext.autosummary\r\n",
      "issue_closed_at": "2019-06-30T15:27:06Z",
      "base_commit": "05949f83474ced6f365ac528e4744a428c6da1ca",
      "changes": [
        {
          "file": "sphinx/environment/__init__.py",
          "type": "function",
          "name": "_update_config",
          "class_name": "BuildEnvironment",
          "code": "def _update_config(self, config):\n        # type: (Config) -> None\n        \"\"\"Update configurations by new one.\"\"\"\n        self.config_status = CONFIG_OK\n        if self.config is None:\n            self.config_status = CONFIG_NEW\n        else:\n            # check if a config value was changed that affects how\n            # doctrees are read\n            for item in config.filter('env'):\n                if self.config[item.name] != item.value:\n                    self.config_status = CONFIG_CHANGED\n                    break\n\n            # this value is not covered by the above loop because it is handled\n            # specially by the config class\n            if self.config.extensions != config.extensions:\n                self.config_status = CONFIG_EXTENSIONS_CHANGED\n\n        self.config = config"
        },
        {
          "file": "sphinx/environment/__init__.py",
          "type": "function",
          "name": "_update_config",
          "class_name": "BuildEnvironment",
          "code": "def _update_config(self, config):\n        # type: (Config) -> None\n        \"\"\"Update configurations by new one.\"\"\"\n        self.config_status = CONFIG_OK\n        if self.config is None:\n            self.config_status = CONFIG_NEW\n        else:\n            # check if a config value was changed that affects how\n            # doctrees are read\n            for item in config.filter('env'):\n                if self.config[item.name] != item.value:\n                    self.config_status = CONFIG_CHANGED\n                    break\n\n            # this value is not covered by the above loop because it is handled\n            # specially by the config class\n            if self.config.extensions != config.extensions:\n                self.config_status = CONFIG_EXTENSIONS_CHANGED\n\n        self.config = config"
        }
      ]
    },
    {
      "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 \ud83e\udd17 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"
        }
      ]
    },
    {
      "pr_number": 7515,
      "pr_title": "Close #2044: autodoc: Suppress default value for instance attributes",
      "pr_body": "### Feature or Bugfix\r\n- Feature\r\n- Bugfix\r\n\r\n### Purpose\r\n- refs: #2044 \r\n- This also fixes uninitialized global variables.",
      "issue_id": 2044,
      "issue_title": "None by by default for instance attributes?",
      "issue_body": "When I'm included docstrings for instance attributes in the `__init__` of a class then using autoclass, I always end up with\n\n```\nattr = None\n    Documentation of attr\n```\n\ndue to this line\n\nhttps://github.com/sphinx-doc/sphinx/blob/add4c2467d9f98cadadff51593ca1727b4a81ca1/sphinx/ext/autodoc.py#L1465\n\nIs there a use case for documenting the `= None`. Can it just always be suppressed? There's no way right now to pass a `annotation` option to autoclass AFAICT.\n",
      "issue_closed_at": "2020-04-23T12:57:37Z",
      "base_commit": "12cb90c3fa5451e2c99100d8ce1d4d0a8bb7f527",
      "changes": [
        {
          "file": "sphinx/ext/autodoc/__init__.py",
          "type": "function",
          "name": "identity",
          "class_name": null,
          "code": "def identity(x: Any) -> Any:\n    return x"
        },
        {
          "file": "sphinx/ext/autodoc/__init__.py",
          "type": "function",
          "name": "add_directive_header",
          "class_name": "PropertyDocumenter",
          "code": "def add_directive_header(self, sig: str) -> None:\n        super().add_directive_header(sig)\n        sourcename = self.get_sourcename()\n        if inspect.isabstractmethod(self.object):\n            self.add_line('   :abstractmethod:', sourcename)\n        self.add_line('   :property:', sourcename)"
        },
        {
          "file": "sphinx/ext/autodoc/__init__.py",
          "type": "function",
          "name": "import_object",
          "class_name": "SlotsAttributeDocumenter",
          "code": "def import_object(self) -> Any:\n        \"\"\"Never import anything.\"\"\"\n        # disguise as an attribute\n        self.objtype = 'attribute'\n        self._datadescriptor = True\n\n        with mock(self.env.config.autodoc_mock_imports):\n            try:\n                ret = import_object(self.modname, self.objpath[:-1], 'class',\n                                    attrgetter=self.get_attr,\n                                    warningiserror=self.env.config.autodoc_warningiserror)\n                self.module, _, _, self.parent = ret\n                return True\n            except ImportError as exc:\n                logger.warning(exc.args[0], type='autodoc', subtype='import_object')\n                self.env.note_reread()\n                return False"
        },
        {
          "file": "sphinx/ext/autodoc/__init__.py",
          "type": "function",
          "name": "add_directive_header",
          "class_name": "PropertyDocumenter",
          "code": "def add_directive_header(self, sig: str) -> None:\n        super().add_directive_header(sig)\n        sourcename = self.get_sourcename()\n        if inspect.isabstractmethod(self.object):\n            self.add_line('   :abstractmethod:', sourcename)\n        self.add_line('   :property:', sourcename)"
        },
        {
          "file": "sphinx/ext/autodoc/__init__.py",
          "type": "function",
          "name": "import_object",
          "class_name": "SlotsAttributeDocumenter",
          "code": "def import_object(self) -> Any:\n        \"\"\"Never import anything.\"\"\"\n        # disguise as an attribute\n        self.objtype = 'attribute'\n        self._datadescriptor = True\n\n        with mock(self.env.config.autodoc_mock_imports):\n            try:\n                ret = import_object(self.modname, self.objpath[:-1], 'class',\n                                    attrgetter=self.get_attr,\n                                    warningiserror=self.env.config.autodoc_warningiserror)\n                self.module, _, _, self.parent = ret\n                return True\n            except ImportError as exc:\n                logger.warning(exc.args[0], type='autodoc', subtype='import_object')\n                self.env.note_reread()\n                return False"
        }
      ]
    },
    {
      "pr_number": 7374,
      "pr_title": "Fix #7301: capital characters are not allowed for node_id",
      "pr_body": "### Feature or Bugfix\r\n- Bugfix\r\n\r\n### Purpose\r\n- refs: #7301 ",
      "issue_id": 7301,
      "issue_title": "Breaking change to Python domain IDs",
      "issue_body": "**Describe the bug**\r\n\r\nPreviously, anchors for Python functions were using underscores, #7236 changed this to dashes.\r\n\r\n**To Reproduce**\r\n\r\nDocument some Python function whose name contains underscores:\r\n\r\n```rst\r\n.. py:function:: example_python_function(foo)\r\n\r\n    Some function.\r\n```\r\n\r\n**Expected behavior**\r\n\r\nThis used to create a fragment identifier `#example_python_function` , but since #7236 this creates `#example-python-function`.\r\n\r\n**Your project**\r\n\r\nThis breaks links to python functions when used with `nbsphinx`: https://nbsphinx.readthedocs.io/en/0.5.1/markdown-cells.html#Links-to-Domain-Objects\r\n\r\nApart from that all links (containing underscores) from external sites to Python API docs created by Sphinx (which I guess are a lot) will break!",
      "issue_closed_at": "2020-03-29T15:19:49Z",
      "base_commit": "70c61e44c34b4dadf1a7552be7c5feabd74b98bc",
      "changes": [
        {
          "file": "sphinx/builders/_epub_base.py",
          "type": "function",
          "name": "fix_ids",
          "class_name": "EpubBuilder",
          "code": "def fix_ids(self, tree: nodes.document) -> None:\n        \"\"\"Replace colons with hyphens in href and id attributes.\n\n        Some readers crash because they interpret the part as a\n        transport protocol specification.\n        \"\"\"\n        for reference in tree.traverse(nodes.reference):\n            if 'refuri' in reference:\n                m = self.refuri_re.match(reference['refuri'])\n                if m:\n                    reference['refuri'] = self.fix_fragment(m.group(1), m.group(2))\n            if 'refid' in reference:\n                reference['refid'] = self.fix_fragment('', reference['refid'])\n\n        for target in tree.traverse(nodes.target):\n            for i, node_id in enumerate(target['ids']):\n                if ':' in node_id:\n                    target['ids'][i] = self.fix_fragment('', node_id)\n\n            next_node = target.next_node(ascend=True)  # type: Node\n            if isinstance(next_node, nodes.Element):\n                for i, node_id in enumerate(next_node['ids']):\n                    if ':' in node_id:\n                        next_node['ids'][i] = self.fix_fragment('', node_id)\n\n        for desc_signature in tree.traverse(addnodes.desc_signature):\n            ids = desc_signature.attributes['ids']\n            newids = []\n            for id in ids:\n                newids.append(self.fix_fragment('', id))\n            desc_signature.attributes['ids'] = newids"
        },
        {
          "file": "sphinx/builders/_epub_base.py",
          "type": "function",
          "name": "fix_ids",
          "class_name": "EpubBuilder",
          "code": "def fix_ids(self, tree: nodes.document) -> None:\n        \"\"\"Replace colons with hyphens in href and id attributes.\n\n        Some readers crash because they interpret the part as a\n        transport protocol specification.\n        \"\"\"\n        for reference in tree.traverse(nodes.reference):\n            if 'refuri' in reference:\n                m = self.refuri_re.match(reference['refuri'])\n                if m:\n                    reference['refuri'] = self.fix_fragment(m.group(1), m.group(2))\n            if 'refid' in reference:\n                reference['refid'] = self.fix_fragment('', reference['refid'])\n\n        for target in tree.traverse(nodes.target):\n            for i, node_id in enumerate(target['ids']):\n                if ':' in node_id:\n                    target['ids'][i] = self.fix_fragment('', node_id)\n\n            next_node = target.next_node(ascend=True)  # type: Node\n            if isinstance(next_node, nodes.Element):\n                for i, node_id in enumerate(next_node['ids']):\n                    if ':' in node_id:\n                        next_node['ids'][i] = self.fix_fragment('', node_id)\n\n        for desc_signature in tree.traverse(addnodes.desc_signature):\n            ids = desc_signature.attributes['ids']\n            newids = []\n            for id in ids:\n                newids.append(self.fix_fragment('', id))\n            desc_signature.attributes['ids'] = newids"
        },
        {
          "file": "sphinx/util/nodes.py",
          "type": "function",
          "name": "_make_id",
          "class_name": null,
          "code": "def _make_id(string: str) -> str:\n    \"\"\"Convert `string` into an identifier and return it.\n\n    This function is a modified version of ``docutils.nodes.make_id()`` of\n    docutils-0.16.\n\n    Changes:\n\n    * Allow to use dots (\".\") and underscores (\"_\") for an identifier\n      without a leading character.\n\n    # Author: David Goodger <goodger@python.org>\n    # Maintainer: docutils-develop@lists.sourceforge.net\n    # Copyright: This module has been placed in the public domain.\n    \"\"\"\n    id = string.lower()\n    id = id.translate(_non_id_translate_digraphs)\n    id = id.translate(_non_id_translate)\n    # get rid of non-ascii characters.\n    # 'ascii' lowercase to prevent problems with turkish locale.\n    id = unicodedata.normalize('NFKD', id).encode('ascii', 'ignore').decode('ascii')\n    # shrink runs of whitespace and replace by hyphen\n    id = _non_id_chars.sub('-', ' '.join(id.split()))\n    id = _non_id_at_ends.sub('', id)\n    return str(id)"
        },
        {
          "file": "sphinx/util/nodes.py",
          "type": "function",
          "name": "_make_id",
          "class_name": null,
          "code": "def _make_id(string: str) -> str:\n    \"\"\"Convert `string` into an identifier and return it.\n\n    This function is a modified version of ``docutils.nodes.make_id()`` of\n    docutils-0.16.\n\n    Changes:\n\n    * Allow to use dots (\".\") and underscores (\"_\") for an identifier\n      without a leading character.\n\n    # Author: David Goodger <goodger@python.org>\n    # Maintainer: docutils-develop@lists.sourceforge.net\n    # Copyright: This module has been placed in the public domain.\n    \"\"\"\n    id = string.lower()\n    id = id.translate(_non_id_translate_digraphs)\n    id = id.translate(_non_id_translate)\n    # get rid of non-ascii characters.\n    # 'ascii' lowercase to prevent problems with turkish locale.\n    id = unicodedata.normalize('NFKD', id).encode('ascii', 'ignore').decode('ascii')\n    # shrink runs of whitespace and replace by hyphen\n    id = _non_id_chars.sub('-', ' '.join(id.split()))\n    id = _non_id_at_ends.sub('', id)\n    return str(id)"
        },
        {
          "file": "sphinx/util/nodes.py",
          "type": "function",
          "name": "_make_id",
          "class_name": null,
          "code": "def _make_id(string: str) -> str:\n    \"\"\"Convert `string` into an identifier and return it.\n\n    This function is a modified version of ``docutils.nodes.make_id()`` of\n    docutils-0.16.\n\n    Changes:\n\n    * Allow to use dots (\".\") and underscores (\"_\") for an identifier\n      without a leading character.\n\n    # Author: David Goodger <goodger@python.org>\n    # Maintainer: docutils-develop@lists.sourceforge.net\n    # Copyright: This module has been placed in the public domain.\n    \"\"\"\n    id = string.lower()\n    id = id.translate(_non_id_translate_digraphs)\n    id = id.translate(_non_id_translate)\n    # get rid of non-ascii characters.\n    # 'ascii' lowercase to prevent problems with turkish locale.\n    id = unicodedata.normalize('NFKD', id).encode('ascii', 'ignore').decode('ascii')\n    # shrink runs of whitespace and replace by hyphen\n    id = _non_id_chars.sub('-', ' '.join(id.split()))\n    id = _non_id_at_ends.sub('', id)\n    return str(id)"
        }
      ]
    }
  ]
}