{
  "instance_id": "sphinx-doc__sphinx-8282",
  "repo": "sphinx-doc/sphinx",
  "created_at": "2020-10-04T09:04:48Z",
  "problem_statement": "autodoc_typehints does not effect to overloaded callables\n**Describe the bug**\r\nautodoc_typehints does not effect to overloaded callables.\r\n\r\n**To Reproduce**\r\n\r\n```\r\n# in conf.py\r\nautodoc_typehints = 'none'\r\n```\r\n```\r\n# in index.rst\r\n.. automodule:: example\r\n   :members:\r\n   :undoc-members:\r\n```\r\n```\r\n# in example.py\r\nfrom typing import overload\r\n\r\n\r\n@overload\r\ndef foo(x: int) -> int:\r\n    ...\r\n\r\n\r\n@overload\r\ndef foo(x: float) -> float:\r\n    ...\r\n\r\n\r\ndef foo(x):\r\n    return x\r\n```\r\n\r\n**Expected behavior**\r\nAll typehints for overloaded callables are obeyed `autodoc_typehints` setting.\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.8.2\r\n- Sphinx version: 3.1.0dev\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@@ -1240,7 +1240,9 @@ def add_directive_header(self, sig: str) -> None:\n \n     def format_signature(self, **kwargs: Any) -> str:\n         sigs = []\n-        if self.analyzer and '.'.join(self.objpath) in self.analyzer.overloads:\n+        if (self.analyzer and\n+                '.'.join(self.objpath) in self.analyzer.overloads and\n+                self.env.config.autodoc_typehints == 'signature'):\n             # Use signatures for overloaded functions instead of the implementation function.\n             overloaded = True\n         else:\n@@ -1474,7 +1476,7 @@ def format_signature(self, **kwargs: Any) -> str:\n         sigs = []\n \n         overloads = self.get_overloaded_signatures()\n-        if overloads:\n+        if overloads and self.env.config.autodoc_typehints == 'signature':\n             # Use signatures for overloaded methods instead of the implementation method.\n             method = safe_getattr(self._signature_class, self._signature_method_name, None)\n             __globals__ = safe_getattr(method, '__globals__', {})\n@@ -1882,7 +1884,9 @@ def document_members(self, all_members: bool = False) -> None:\n \n     def format_signature(self, **kwargs: Any) -> str:\n         sigs = []\n-        if self.analyzer and '.'.join(self.objpath) in self.analyzer.overloads:\n+        if (self.analyzer and\n+                '.'.join(self.objpath) in self.analyzer.overloads and\n+                self.env.config.autodoc_typehints == 'signature'):\n             # Use signatures for overloaded methods instead of the implementation method.\n             overloaded = True\n         else:\n",
  "similar_bug_items": [
    {
      "pr_number": 6404,
      "pr_title": "Fix #6347: autodoc: crashes with a plain Tuple on Python 3.6 and 3.5",
      "pr_body": "### Feature or Bugfix\r\n- Bugfix\r\n\r\n### Purpose\r\n- refs: #6347 ",
      "issue_id": 6347,
      "issue_title": "Fail to parse signature with typing that contains a Union with a plain Tuple on Python 3.6 and 3.5",
      "issue_body": "**Describe the bug**\r\nSphinx fails to parse the signature of a class method that uses type hints and has a parameter with a `Union` type that includes a plain `Tuple` on Python 3.6 and 3.5.\r\nExample signature:\r\n```python\r\n    def mymethod(self, myparam: Union[int, Tuple] = 10) -> List[Dict]:\r\n```\r\n\r\nWhen parsing this code Sphinx fails on Python 3.5 and 3.6 with:\r\n```\r\nWarning, treated as error:\r\nerror while formatting arguments for mymodule.MyClass.mymethod: 'NoneType' object is not iterable\r\n```\r\n\r\nWhile it work as expected on Python 3.7.\r\nJust for reference a plain `Tuple` is a valid type and is equivalent to `Tuple[Any, ...]`.\r\n\r\n**To Reproduce**\r\nTo my understanding any signature with a type hint of `Union[Tuple]` should reproduce the error.\r\n\r\n**Expected behavior**\r\nSphinx should have correctly parsed the signature.\r\n\r\n**Environment info**\r\n- OS: the issue was reproduced on both Linux Debian and Macos\r\n- Python version: 3.5.6, 3.6.7 for the repro, 3.7.3 where it works\r\n- Sphinx version: 2.0.1\r\n- Sphinx extensions:  not applicable\r\n\r\n**Additional context**\r\nI've managed to get a full stacktrace that is pretty long, so pasting here only the final relevant part. This is on 3.5 but the 3.6 one is equivalent:\r\n```python\r\n  File \"/tmp/myproject/.tox/py35-tests/lib/python3.5/site-packages/sphinx/ext/autodoc/__init__.py\", line 1305, in format_args\r\n    args = Signature(self.object, bound_method=True).format_args()\r\n  File \"/tmp/myproject/.tox/py35-tests/lib/python3.5/site-packages/sphinx/util/inspect.py\", line 384, in format_args\r\n    arg.write(self.format_annotation(param.annotation))\r\n  File \"/tmp/myproject/.tox/py35-tests/lib/python3.5/site-packages/sphinx/util/inspect.py\", line 437, in format_annotation\r\n    return self.format_annotation_old(annotation)\r\n  File \"/tmp/myproject/.tox/py35-tests/lib/python3.5/site-packages/sphinx/util/inspect.py\", line 540, in format_annotation_old\r\n    param_str = ', '.join(self.format_annotation(p) for p in params)\r\n  File \"/tmp/myproject/.tox/py35-tests/lib/python3.5/site-packages/sphinx/util/inspect.py\", line 540, in <genexpr>\r\n    param_str = ', '.join(self.format_annotation(p) for p in params)\r\n  File \"/tmp/myproject/.tox/py35-tests/lib/python3.5/site-packages/sphinx/util/inspect.py\", line 437, in format_annotation\r\n    return self.format_annotation_old(annotation)\r\n  File \"/tmp/myproject/.tox/py35-tests/lib/python3.5/site-packages/sphinx/util/inspect.py\", line 499, in format_annotation_old\r\n    param_str = ', '.join(self.format_annotation(p) for p in params)\r\nTypeError: 'NoneType' object is not iterable\r\n```\r\n\r\nAfter some debugging I've noticed that the code enters the first `if` block in https://github.com/sphinx-doc/sphinx/blob/master/sphinx/util/inspect.py#L526 where at line 530-531 does:\r\n```python\r\n            params = annotation.__args__\r\n            param_str = ', '.join(self.format_annotation(p) for p in params)\r\n```\r\n\r\nBut on Python 3.5 and 3.6 a plain Tuple has `__args__` that is `None`:\r\n```python\r\n>>> from typing import Tuple\r\n>>> Tuple.__args__ is None\r\nTrue\r\n```\r\nSo the iteration over `params` fails.\r\n\r\n**Possible Fix**\r\nA quick fix could be to just replace line 530 with:\r\n```python\r\n            params = annotation.__args__ or ()\r\n```\r\n\r\nAlthough I've not sent yet a PR because the comments in that part of the code confuses me as they seem to contradict each other, in particular line 529 `# This is for Python 3.6+, 3.5 case is handled below` vs line 533 `# for py36 or below`.\r\nWhen clearly the first `if` block would not match Python 3.7 given that on 3.7:\r\n```python\r\n>>> hasattr(typing, 'TupleMeta')\r\nFalse\r\n```\r\nSo I would like some advice from people more familiar with that part of the code before sending a patch.",
      "issue_closed_at": "2019-05-29T15:59:23Z",
      "base_commit": "804d5d804a6b9107b9d21a7ab68134f63de1960a",
      "changes": [
        {
          "file": "sphinx/util/inspect.py",
          "type": "function",
          "name": "format_annotation_new",
          "class_name": "Signature",
          "code": "def format_annotation_new(self, annotation):\n        # type: (Any) -> str\n        \"\"\"format_annotation() for py37+\"\"\"\n        module = getattr(annotation, '__module__', None)\n        if module == 'typing':\n            if getattr(annotation, '_name', None):\n                qualname = annotation._name\n            elif getattr(annotation, '__qualname__', None):\n                qualname = annotation.__qualname__\n            elif getattr(annotation, '__forward_arg__', None):\n                qualname = annotation.__forward_arg__\n            else:\n                qualname = self.format_annotation(annotation.__origin__)  # ex. Union\n        elif hasattr(annotation, '__qualname__'):\n            qualname = '%s.%s' % (module, annotation.__qualname__)\n        else:\n            qualname = repr(annotation)\n\n        if getattr(annotation, '__args__', None):\n            if qualname == 'Union':\n                if len(annotation.__args__) == 2 and annotation.__args__[1] is NoneType:  # type: ignore  # NOQA\n                    return 'Optional[%s]' % self.format_annotation(annotation.__args__[0])\n                else:\n                    args = ', '.join(self.format_annotation(a) for a in annotation.__args__)\n                    return '%s[%s]' % (qualname, args)\n            elif qualname == 'Callable':\n                args = ', '.join(self.format_annotation(a) for a in annotation.__args__[:-1])\n                returns = self.format_annotation(annotation.__args__[-1])\n                return '%s[[%s], %s]' % (qualname, args, returns)\n            else:\n                args = ', '.join(self.format_annotation(a) for a in annotation.__args__)\n                return '%s[%s]' % (qualname, args)\n\n        return qualname"
        },
        {
          "file": "sphinx/util/inspect.py",
          "type": "function",
          "name": "format_annotation_old",
          "class_name": "Signature",
          "code": "def format_annotation_old(self, annotation):\n        # type: (Any) -> str\n        \"\"\"format_annotation() for py36 or below\"\"\"\n        module = getattr(annotation, '__module__', None)\n        if module == 'typing':\n            if getattr(annotation, '_name', None):\n                qualname = annotation._name\n            elif getattr(annotation, '__qualname__', None):\n                qualname = annotation.__qualname__\n            elif getattr(annotation, '__forward_arg__', None):\n                qualname = annotation.__forward_arg__\n            elif getattr(annotation, '__origin__', None):\n                qualname = self.format_annotation(annotation.__origin__)  # ex. Union\n            else:\n                qualname = repr(annotation).replace('typing.', '')\n        elif hasattr(annotation, '__qualname__'):\n            qualname = '%s.%s' % (module, annotation.__qualname__)\n        else:\n            qualname = repr(annotation)\n\n        if (hasattr(typing, 'TupleMeta') and\n                isinstance(annotation, typing.TupleMeta) and  # type: ignore\n                not hasattr(annotation, '__tuple_params__')):\n            # This is for Python 3.6+, 3.5 case is handled below\n            params = annotation.__args__\n            param_str = ', '.join(self.format_annotation(p) for p in params)\n            return '%s[%s]' % (qualname, param_str)\n        elif (hasattr(typing, 'GenericMeta') and  # for py36 or below\n              isinstance(annotation, typing.GenericMeta)):\n            # In Python 3.5.2+, all arguments are stored in __args__,\n            # whereas __parameters__ only contains generic parameters.\n            #\n            # Prior to Python 3.5.2, __args__ is not available, and all\n            # arguments are in __parameters__.\n            params = None\n            if hasattr(annotation, '__args__'):\n                if annotation.__args__ is None or len(annotation.__args__) <= 2:  # type: ignore  # NOQA\n                    params = annotation.__args__  # type: ignore\n                else:  # typing.Callable\n                    args = ', '.join(self.format_annotation(arg) for arg\n                                     in annotation.__args__[:-1])  # type: ignore\n                    result = self.format_annotation(annotation.__args__[-1])  # type: ignore\n                    return '%s[[%s], %s]' % (qualname, args, result)\n            elif hasattr(annotation, '__parameters__'):\n                params = annotation.__parameters__  # type: ignore\n            if params is not None:\n                param_str = ', '.join(self.format_annotation(p) for p in params)\n                return '%s[%s]' % (qualname, param_str)\n        elif (hasattr(typing, 'UnionMeta') and  # for py35 or below\n              isinstance(annotation, typing.UnionMeta) and  # type: ignore\n              hasattr(annotation, '__union_params__')):\n            params = annotation.__union_params__\n            if params is not None:\n                if len(params) == 2 and params[1] is NoneType:  # type: ignore\n                    return 'Optional[%s]' % self.format_annotation(params[0])\n                else:\n                    param_str = ', '.join(self.format_annotation(p) for p in params)\n                    return '%s[%s]' % (qualname, param_str)\n        elif (hasattr(typing, 'Union') and  # for py36\n              hasattr(annotation, '__origin__') and\n              annotation.__origin__ is typing.Union):\n            params = annotation.__args__\n            if params is not None:\n                if len(params) == 2 and params[1] is NoneType:  # type: ignore\n                    return 'Optional[%s]' % self.format_annotation(params[0])\n                else:\n                    param_str = ', '.join(self.format_annotation(p) for p in params)\n                    return 'Union[%s]' % param_str\n        elif (hasattr(typing, 'CallableMeta') and  # for py36 or below\n              isinstance(annotation, typing.CallableMeta) and  # type: ignore\n              getattr(annotation, '__args__', None) is not None and\n              hasattr(annotation, '__result__')):\n            # Skipped in the case of plain typing.Callable\n            args = annotation.__args__\n            if args is None:\n                return qualname\n            elif args is Ellipsis:\n                args_str = '...'\n            else:\n                formatted_args = (self.format_annotation(a) for a in args)\n                args_str = '[%s]' % ', '.join(formatted_args)\n            return '%s[%s, %s]' % (qualname,\n                                   args_str,\n                                   self.format_annotation(annotation.__result__))\n        elif (hasattr(typing, 'TupleMeta') and  # for py36 or below\n              isinstance(annotation, typing.TupleMeta) and  # type: ignore\n              hasattr(annotation, '__tuple_params__') and\n              hasattr(annotation, '__tuple_use_ellipsis__')):\n            params = annotation.__tuple_params__\n            if params is not None:\n                param_strings = [self.format_annotation(p) for p in params]\n                if annotation.__tuple_use_ellipsis__:\n                    param_strings.append('...')\n                return '%s[%s]' % (qualname,\n                                   ', '.join(param_strings))\n\n        return qualname"
        }
      ]
    },
    {
      "pr_number": 7823,
      "pr_title": "Fix #7821: autodoc: TypeError is raised for overloaded C-ext function",
      "pr_body": "### Feature or Bugfix\r\n- Bugfix\r\n\r\n### Purpose\r\n- refs: #7821 ",
      "issue_id": 7821,
      "issue_title": "TypeError: can't set attributes of built-in/extension type 'list'",
      "issue_body": "Previous issue: https://github.com/sphinx-doc/sphinx/issues/7791\r\n\r\n**Describe the bug**\r\nRunning sphinx-build after sphinx-apidoc on our codebase results in the error message:\r\n\r\n```\r\n  File \"/home/danielk/.local/lib/python3.7/site-packages/sphinx/ext/autodoc/__init__.py\", line 1240, in annotate_to_first_argument\r\n    func.__signature__ = sig.replace(parameters=params)  # type: ignore\r\nTypeError: can't set attributes of built-in/extension type 'list'\r\n```\r\n\r\n**To Reproduce**\r\nSteps to reproduce the behavior:\r\n\r\nRun `make html` in project folder\r\n\r\n**Expected behavior**\r\nNo error\r\n\r\n**Your project**\r\n\r\nn/a\r\n\r\n**Screenshots**\r\nIf applicable, add screenshots to help explain your problem.\r\n\r\n**Environment info**\r\n- OS: Gentoo Linux, KDE 19.12.3, Plasma 5.70\r\n- Python version: 3.7.7\r\n- Sphinx version: See below\r\n- Sphinx extensions: See below\r\n- Extra tools: n/a\r\n\r\n```\r\nalabaster                          0.7.12            \r\nautodoc                            0.5.0             \r\nBabel                              2.8.0             \r\nbeautifulsoup4                     4.9.1             \r\nBrotli                             1.0.7             \r\nbsddb3                             6.2.7             \r\ncertifi                            2020.4.5.1        \r\ncffi                               1.14.0            \r\nchardet                            3.0.4             \r\ncracklib                           2.9.7             \r\ncryptacular                        1.4.1             /home/danielk/Development/cryptacular/src/cryptacular\r\ncryptography                       2.8               \r\nCython                             0.29.15           \r\ndecorator                          4.4.2             \r\ndistro                             1.0.4             \r\ndocutils                           0.16              \r\nextras                             1.0.0             \r\nfile-magic                         0.4.0             \r\nfixtures                           3.0.0             \r\nFormEncode                         1.3.1             \r\ngdbus-codegen                      2.62.6            \r\ngemato                             14.3              \r\ngentoolkit                         0.4.8             \r\ngpg                                1.13.0            \r\ngpodder                            3.10.5            \r\nhtml5lib                           1.0.1             \r\nidna                               2.8               \r\nimagesize                          1.1.0             \r\nisodate                            0.6.0             \r\nisort                              4.3.15            \r\njava-config                        2.2.0             \r\nJinja2                             2.11.1            \r\nlayman                             2.4.3             \r\nlensfun                            0.3.2             \r\nlibsass                            0.20.0            \r\nlinecache2                         1.0.0             \r\nlxml                               4.5.0             \r\nM2Crypto                           0.31.0            \r\nMako                               1.1.2             \r\nMarkdown                           2.4.1             \r\nMarkups                            3.0.0             \r\nMarkupSafe                         1.1.1             \r\nmercurial                          5.3.2             \r\nmeson                              0.52.1            \r\nmimeparse                          1.6.0             \r\nmock                               4.0.2             \r\nMomoko                             2.2.5.1           \r\nmygpoclient                        1.8               \r\nnetlink                            1.0               \r\nnotify2                            0.3.1             \r\nnumpy                              1.17.4            \r\nolefile                            0.46              \r\npackaging                          20.3              \r\npbkdf2                             1.3               \r\npbr                                5.1.1             \r\npdoc                               0.3.2             \r\nPillow                             7.0.0             \r\npip                                20.0.2            \r\nply                                3.11              \r\npodcastparser                      0.6.4             \r\nportage                            2.3.99            \r\npsycopg2                           2.8.5             \r\npsycopg2-binary                    2.8.5             \r\npwquality                          1.4.2             \r\npycairo                            1.18.2            \r\nPycco                              0.6.0             \r\npycountry                          19.8.18           \r\npycparser                          2.20              \r\npycryptodome                       3.9.4             \r\npyelftools                         0.25              \r\npyenchant                          2.0.0             \r\nPygments                           2.5.2             \r\nPyGObject                          3.34.0            \r\npyOpenSSL                          19.1.0            \r\npyparsing                          2.4.6             \r\npyportmidi                         0.0.7             \r\nPyQt5                              5.14.1            \r\nPyQt5-sip                          4.19.21           \r\nPyQtWebEngine                      5.14.0            \r\npyrsistent                         0.15.6            \r\nPySocks                            1.7.1             \r\npystache                           0.5.4             \r\npython-markdown-math               0.6               \r\npython-mimeparse                   1.6.0             \r\npython-subunit                     1.2.0             \r\npytz                               2019.3            \r\npyxdg                              0.26              \r\nPyYAML                             5.3.1             \r\nrdflib                             4.2.2             \r\nreportlab                          3.5.13            \r\nrequests                           2.23.0            \r\nReText                             7.0.4             \r\nscour                              0.37              \r\nsetuptools                         44.1.0            \r\nsimplejson                         3.17.0            \r\nsip                                4.19.21           \r\nsix                                1.14.0            \r\nsmartypants                        2.0.1             \r\nsnowballstemmer                    2.0.0             \r\nsoupsieve                          2.0.1             \r\nSphinx                             3.1.0             \r\nsphinx-rtd-theme                   0.4.3             \r\nsphinx-rtd-theme-http              1.0.0             \r\nsphinxcontrib-apidoc               0.3.0             \r\nsphinxcontrib-applehelp            1.0.2             \r\nsphinxcontrib-autodoc-doxygen      0.6.0             \r\nsphinxcontrib-autodoc-filterparams 0.0.1             \r\nsphinxcontrib-devhelp              1.0.2             \r\nsphinxcontrib-htmlhelp             1.0.3             \r\nsphinxcontrib-httpdomain           1.7.0             \r\nsphinxcontrib-jsmath               1.0.1             \r\nsphinxcontrib-mockautodoc          0.0.1.dev20130518 \r\nsphinxcontrib-qthelp               1.0.3             \r\nsphinxcontrib-serializinghtml      1.1.3             \r\nssl-fetch                          0.4               \r\ntesttools                          2.3.0             \r\ntornado                            5.1.1             \r\ntraceback2                         1.4.0             \r\ntyped-ast                          1.4.1             \r\nunittest2                          1.1.0             \r\nurllib3                            1.25.8            \r\nwaitress                           1.4.4             \r\nwebencodings                       0.5.1             \r\nWebOb                              1.8.6             \r\nWebTest                            2.0.35            \r\nyoutube-dl                         2020.5.29         \r\nzstandard                          0.13.0 \r\n```\r\n\r\n\r\n\r\n**Additional context**\r\n\r\n```\r\n\r\nRunning Sphinx v3.1.0\r\nbuilding [mo]: targets for 0 po files that are out of date\r\nbuilding [html]: targets for 41 source files that are out of date\r\nupdating environment: [new config] 41 added, 0 changed, 0 removed\r\nreading sources... [ 87%] ... Exception occurred:\r\n  File \"/home/danielk/.local/lib/python3.7/site-packages/sphinx/ext/autodoc/__init__.py\", line 1240, in annotate_to_first_argument\r\n    func.__signature__ = sig.replace(parameters=params)  # type: ignore\r\nTypeError: can't set attributes of built-in/extension type 'list'\r\nThe full traceback has been saved in /tmp/sphinx-err-lf0wsvvp.log\r\n\r\n, if you want to report the issue to the developers.\r\nPlease also report this if it was a user error, so that a better error message can be provided next time.\r\nA bug report can be filed in the tracker at <https://github.com/sphinx-doc/sphinx/issues>. Thanks!\r\nmake[1]: *** [Makefile:20: html] Error 2\r\nmake[1]: Leaving directory '/home/danielk/Development/docs'\r\nmake: *** [Makefile:85: docs] Error 2\r\n```\r\n\r\n[sphinx-err-lf0wsvvp.log](https://github.com/sphinx-doc/sphinx/files/4766780/sphinx-err-lf0wsvvp.log)",
      "issue_closed_at": "2020-06-14T02:48:20Z",
      "base_commit": "46f79c55e1242ae8bb49b14c0a9e226d119460e5",
      "changes": [
        {
          "file": "sphinx/ext/autodoc/__init__.py",
          "type": "function",
          "name": "annotate_to_first_argument",
          "class_name": "MethodDocumenter",
          "code": "def annotate_to_first_argument(self, func: Callable, typ: Type) -> None:\n        \"\"\"Annotate type hint to the first argument of function if needed.\"\"\"\n        try:\n            sig = inspect.signature(func)\n        except TypeError as exc:\n            logger.warning(__(\"Failed to get a method signature for %s: %s\"),\n                           self.fullname, exc)\n            return\n        except ValueError:\n            return\n        if len(sig.parameters) == 1:\n            return\n\n        params = list(sig.parameters.values())\n        if params[1].annotation is Parameter.empty:\n            params[1] = params[1].replace(annotation=typ)\n            func.__signature__ = sig.replace(parameters=params)"
        },
        {
          "file": "sphinx/ext/autodoc/__init__.py",
          "type": "function",
          "name": "annotate_to_first_argument",
          "class_name": "MethodDocumenter",
          "code": "def annotate_to_first_argument(self, func: Callable, typ: Type) -> None:\n        \"\"\"Annotate type hint to the first argument of function if needed.\"\"\"\n        try:\n            sig = inspect.signature(func)\n        except TypeError as exc:\n            logger.warning(__(\"Failed to get a method signature for %s: %s\"),\n                           self.fullname, exc)\n            return\n        except ValueError:\n            return\n        if len(sig.parameters) == 1:\n            return\n\n        params = list(sig.parameters.values())\n        if params[1].annotation is Parameter.empty:\n            params[1] = params[1].replace(annotation=typ)\n            func.__signature__ = sig.replace(parameters=params)"
        }
      ]
    },
    {
      "pr_number": 8112,
      "pr_title": "Close #8100: html: Show a better error message for html_static_files",
      "pr_body": "### Feature or Bugfix\r\n- Feature\r\n\r\n### Purpose\r\n- refs: #8100 \r\n- The HTML Builder crashes if error raised on copying html_static_files.\r\nThis handles the exception and show a better error message to let users\r\nthe reason of errors (ex. failed on extracting Jinja templates).\r\n",
      "issue_id": 8100,
      "issue_title": "Crash trying to extend pydata-sphinx-theme with sphinx-multiversion",
      "issue_body": "**Describe the bug**\r\nTrying to extend `pydata-sphinx-template` with `sphinx-multiversion` resulted in crash.\r\n\r\nError message:\r\n```\r\nException occurred:\r\n  File \"c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\site-packages\\jinja2\\sandbox.py\", line 360, in is_safe_callable\r\n    getattr(obj, \"unsafe_callable\", False) or getattr(obj, \"alters_data\", False)\r\njinja2.exceptions.UndefinedError: 'pathto' is undefined\r\nThe full traceback has been saved in C:\\Users\\riedgar\\AppData\\Local\\Temp\\sphinx-err-tnwtajjw.log, if you want to report the issue to the developers.\r\nPlease also report this if it was a user error, so that a better error message can be provided next time.\r\nA bug report can be filed in the tracker at <https://github.com/sphinx-doc/sphinx/issues>. Thanks!\r\nTraceback (most recent call last):\r\n  File \"C:\\Users\\riedgar\\AppData\\Local\\Continuum\\miniconda3\\envs\\py-36\\Scripts\\sphinx-multiversion-script.py\", line 33, in <module>\r\n    sys.exit(load_entry_point('sphinx-multiversion', 'console_scripts', 'sphinx-multiversion')())\r\n  File \"c:\\users\\riedgar\\source\\repos\\sphinx-multiversion\\sphinx_multiversion\\main.py\", line 302, in main\r\n    subprocess.check_call(cmd, cwd=current_cwd)\r\n  File \"c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\subprocess.py\", line 311, in check_call\r\n    raise CalledProcessError(retcode, cmd)\r\nsubprocess.CalledProcessError: Command '('c:\\\\users\\\\riedgar\\\\appdata\\\\local\\\\continuum\\\\miniconda3\\\\envs\\\\py-36\\\\python.exe', '-m', 'sphinx', '-D', 'smv_metadata_path=C:\\\\Users\\\\riedgar\\\\AppData\\\\Local\\\\Temp\\\\tmp4vfkrz7o\\\\versions.json', '-D', 'smv_current_version=master', '-c', 'C:\\\\Users\\\\riedgar\\\\source\\\\repos\\\\fairlearn\\\\docs', 'C:\\\\Users\\\\riedgar\\\\AppData\\\\Local\\\\Temp\\\\tmp4vfkrz7o\\\\c6c01dc52462a49b95e2aab9f8de4f586de56234\\\\docs', 'C:\\\\Users\\\\riedgar\\\\source\\\\repos\\\\fairlearn\\\\build\\\\html\\\\master')' returned non-zero exit status 2.\r\n```\r\n\r\nThe logfile mentioned has the following contents:\r\n```\r\n# Sphinx version: 3.1.2\r\n# Python version: 3.6.10 (CPython)\r\n# Docutils version: 0.16 release\r\n# Jinja2 version: 2.11.2\r\n# Last messages:\r\n#   copying downloadable files... [ 87%] auto_examples/quickstart/plot_adult_dataset.py\r\n#   \r\n#   copying downloadable files... [100%] auto_examples/quickstart/plot_adult_dataset.ipynb\r\n#   \r\n#   \r\n#   \r\n#   copying static files... ...\r\n#   \r\n#   failed\r\n#   \r\n# Loaded extensions:\r\n#   sphinx.ext.mathjax (3.1.2) from c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\site-packages\\sphinx\\ext\\mathjax.py\r\n#   sphinxcontrib.applehelp (1.0.2) from c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\site-packages\\sphinxcontrib\\applehelp\\__init__.py\r\n#   sphinxcontrib.devhelp (1.0.2) from c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\site-packages\\sphinxcontrib\\devhelp\\__init__.py\r\n#   sphinxcontrib.htmlhelp (1.0.3) from c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\site-packages\\sphinxcontrib\\htmlhelp\\__init__.py\r\n#   sphinxcontrib.serializinghtml (1.1.4) from c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\site-packages\\sphinxcontrib\\serializinghtml\\__init__.py\r\n#   sphinxcontrib.qthelp (1.0.3) from c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\site-packages\\sphinxcontrib\\qthelp\\__init__.py\r\n#   alabaster (0.7.12) from c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\site-packages\\alabaster\\__init__.py\r\n#   sphinx.ext.autodoc.type_comment (3.1.2) from c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\site-packages\\sphinx\\ext\\autodoc\\type_comment.py\r\n#   sphinx.ext.autodoc (3.1.2) from c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\site-packages\\sphinx\\ext\\autodoc\\__init__.py\r\n#   sphinx.ext.doctest (3.1.2) from c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\site-packages\\sphinx\\ext\\doctest.py\r\n#   sphinx.ext.extlinks (3.1.2) from c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\site-packages\\sphinx\\ext\\extlinks.py\r\n#   sphinx.ext.intersphinx (3.1.2) from c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\site-packages\\sphinx\\ext\\intersphinx.py\r\n#   sphinx.ext.linkcode (3.1.2) from c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\site-packages\\sphinx\\ext\\linkcode.py\r\n#   sphinx.ext.napoleon (3.1.2) from c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\site-packages\\sphinx\\ext\\napoleon\\__init__.py\r\n#   sphinx_gallery.gen_gallery (0.7.0) from c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\site-packages\\sphinx_gallery\\gen_gallery.py\r\n#   sphinx_multiversion (0.2) from c:\\users\\riedgar\\source\\repos\\sphinx-multiversion\\sphinx_multiversion\\__init__.py\r\n#   pydata_sphinx_theme (unknown version) from c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\site-packages\\pydata_sphinx_theme\\__init__.py\r\nTraceback (most recent call last):\r\n  File \"c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\site-packages\\sphinx\\cmd\\build.py\", line 280, in build_main\r\n    app.build(args.force_all, filenames)\r\n  File \"c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\site-packages\\sphinx\\application.py\", line 348, in build\r\n    self.builder.build_update()\r\n  File \"c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\site-packages\\sphinx\\builders\\__init__.py\", line 299, in build_update\r\n    len(to_build))\r\n  File \"c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\site-packages\\sphinx\\builders\\__init__.py\", line 364, in build\r\n    self.finish()\r\n  File \"c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\site-packages\\sphinx\\builders\\html\\__init__.py\", line 617, in finish\r\n    self.finish_tasks.add_task(self.copy_static_files)\r\n  File \"c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\site-packages\\sphinx\\util\\parallel.py\", line 49, in add_task\r\n    res = task_func()\r\n  File \"c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\site-packages\\sphinx\\builders\\html\\__init__.py\", line 790, in copy_static_files\r\n    self.copy_html_static_files(context)\r\n  File \"c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\site-packages\\sphinx\\builders\\html\\__init__.py\", line 764, in copy_html_static_files\r\n    excluded, context=context, renderer=self.templates)\r\n  File \"c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\site-packages\\sphinx\\util\\fileutil.py\", line 95, in copy_asset\r\n    context, renderer)\r\n  File \"c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\site-packages\\sphinx\\util\\fileutil.py\", line 53, in copy_asset_file\r\n    fdst.write(renderer.render_string(fsrc.read(), context))\r\n  File \"c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\site-packages\\sphinx\\jinja2glue.py\", line 195, in render_string\r\n    return self.environment.from_string(source).render(context)\r\n  File \"c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\site-packages\\jinja2\\environment.py\", line 1090, in render\r\n    self.environment.handle_exception()\r\n  File \"c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\site-packages\\jinja2\\environment.py\", line 832, in handle_exception\r\n    reraise(*rewrite_traceback_stack(source=source))\r\n  File \"c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\site-packages\\jinja2\\_compat.py\", line 28, in reraise\r\n    raise value.with_traceback(tb)\r\n  File \"<template>\", line 1, in top-level template code\r\n  File \"c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\site-packages\\pydata_sphinx_theme\\docs-sidebar.html\", line 4, in top-level template code\r\n    {%- include \"search-field.html\" %}\r\n  File \"c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\site-packages\\pydata_sphinx_theme\\search-field.html\", line 1, in top-level template code\r\n    <form class=\"bd-search d-flex align-items-center\" action=\"{{ pathto('search') }}\" method=\"get\">\r\n  File \"c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\site-packages\\jinja2\\sandbox.py\", line 460, in call\r\n    if not __self.is_safe_callable(__obj):\r\n  File \"c:\\users\\riedgar\\appdata\\local\\continuum\\miniconda3\\envs\\py-36\\lib\\site-packages\\jinja2\\sandbox.py\", line 360, in is_safe_callable\r\n    getattr(obj, \"unsafe_callable\", False) or getattr(obj, \"alters_data\", False)\r\njinja2.exceptions.UndefinedError: 'pathto' is undefined\r\n```\r\n\r\n\r\n\r\n**To Reproduce**\r\nSteps to reproduce the behaviour:\r\n```\r\ngit clone https://github.com/riedgar-ms/fairlearn.git\r\ncd fairlearn\r\ngit checkout riedgar-ms/versioned-docs-alternate-01\r\npython scripts/install_requirements.py --pinned False\r\nsphinx-multiversion.exe docs build/html\r\n```\r\n\r\n**Expected behavior**\r\nCommand to complete without error, hopefully with `version.html_t` inserted into the sidebar of the emitted documentation. Alternatively, a comprehensible error message. This might actually be a bug in either the theme or sphinx-multiversion, but I'm really not sure based on the above.\r\n\r\n**Your project**\r\nhttps://github.com/riedgar-ms/fairlearn/tree/riedgar-ms/versioned-docs-alternate-01\r\n\r\n**Screenshots**\r\n\r\n\r\n**Environment info**\r\n- OS: Win10\r\n- Python version: 3.6\r\n- Sphinx version: 3.1.2\r\n- Sphinx extensions:  'sphinx.ext.autodoc', 'sphinx.ext.doctest', \r\n'sphinx.ext.extlinks', 'sphinx.ext.intersphinx', 'sphinx.ext.linkcode', 'sphinx.ext.mathjax', 'sphinx.ext.napoleon', 'sphinx_gallery.gen_gallery', 'sphinx_multiversion'\r\n- Extra tools: N/A\r\n\r\n**Additional context**\r\nAdd any other context about the problem here.\r\n\r\n- [e.g. URL or Ticket]\r\n\r\n",
      "issue_closed_at": "2020-09-13T02:06:43Z",
      "base_commit": "4baa7ce99b6a4a79c203140555743eaa50aed28c",
      "changes": [
        {
          "file": "sphinx/builders/html/__init__.py",
          "type": "function",
          "name": "copy_stemmer_js",
          "class_name": "StandaloneHTMLBuilder",
          "code": "def copy_stemmer_js(self) -> None:\n        \"\"\"Copy a JavaScript file for stemmer.\"\"\"\n        if self.indexer is not None:\n            jsfile = self.indexer.get_js_stemmer_rawcode()\n            if jsfile:\n                copyfile(jsfile, path.join(self.outdir, '_static', '_stemmer.js'))"
        },
        {
          "file": "sphinx/util/fileutil.py",
          "type": "line",
          "name": "line 10",
          "code": "\nimport os\nimport posixpath\nfrom typing import Dict\n\nfrom docutils.utils import relative_path\n"
        },
        {
          "file": "sphinx/util/fileutil.py",
          "type": "function",
          "name": "copy_asset_file",
          "class_name": null,
          "code": "def copy_asset_file(source: str, destination: str,\n                    context: Dict = None, renderer: \"BaseRenderer\" = None) -> None:\n    \"\"\"Copy an asset file to destination.\n\n    On copying, it expands the template variables if context argument is given and\n    the asset is a template file.\n\n    :param source: The path to source file\n    :param destination: The path to destination file or directory\n    :param context: The template variables.  If not given, template files are simply copied\n    :param renderer: The template engine.  If not given, SphinxRenderer is used by default\n    \"\"\"\n    if not os.path.exists(source):\n        return\n\n    if os.path.isdir(destination):\n        # Use source filename if destination points a directory\n        destination = os.path.join(destination, os.path.basename(source))\n\n    if source.lower().endswith('_t') and context is not None:\n        if renderer is None:\n            from sphinx.util.template import SphinxRenderer\n            renderer = SphinxRenderer()\n\n        with open(source, encoding='utf-8') as fsrc:\n            if destination.lower().endswith('_t'):\n                destination = destination[:-2]\n            with open(destination, 'w', encoding='utf-8') as fdst:\n                fdst.write(renderer.render_string(fsrc.read(), context))\n    else:\n        copyfile(source, destination)"
        },
        {
          "file": "sphinx/util/fileutil.py",
          "type": "function",
          "name": "copy_asset",
          "class_name": null,
          "code": "def copy_asset(source: str, destination: str, excluded: PathMatcher = lambda path: False,\n               context: Dict = None, renderer: \"BaseRenderer\" = None) -> None:\n    \"\"\"Copy asset files to destination recursively.\n\n    On copying, it expands the template variables if context argument is given and\n    the asset is a template file.\n\n    :param source: The path to source file or directory\n    :param destination: The path to destination directory\n    :param excluded: The matcher to determine the given path should be copied or not\n    :param context: The template variables.  If not given, template files are simply copied\n    :param renderer: The template engine.  If not given, SphinxRenderer is used by default\n    \"\"\"\n    if not os.path.exists(source):\n        return\n\n    if renderer is None:\n        from sphinx.util.template import SphinxRenderer\n        renderer = SphinxRenderer()\n\n    ensuredir(destination)\n    if os.path.isfile(source):\n        copy_asset_file(source, destination, context, renderer)\n        return\n\n    for root, dirs, files in os.walk(source, followlinks=True):\n        reldir = relative_path(source, root)\n        for dir in dirs[:]:\n            if excluded(posixpath.join(reldir, dir)):\n                dirs.remove(dir)\n            else:\n                ensuredir(posixpath.join(destination, reldir, dir))\n\n        for filename in files:\n            if not excluded(posixpath.join(reldir, filename)):\n                copy_asset_file(posixpath.join(root, filename),\n                                posixpath.join(destination, reldir),\n                                context, renderer)"
        },
        {
          "file": "sphinx/util/fileutil.py",
          "type": "function",
          "name": "copy_asset",
          "class_name": null,
          "code": "def copy_asset(source: str, destination: str, excluded: PathMatcher = lambda path: False,\n               context: Dict = None, renderer: \"BaseRenderer\" = None) -> None:\n    \"\"\"Copy asset files to destination recursively.\n\n    On copying, it expands the template variables if context argument is given and\n    the asset is a template file.\n\n    :param source: The path to source file or directory\n    :param destination: The path to destination directory\n    :param excluded: The matcher to determine the given path should be copied or not\n    :param context: The template variables.  If not given, template files are simply copied\n    :param renderer: The template engine.  If not given, SphinxRenderer is used by default\n    \"\"\"\n    if not os.path.exists(source):\n        return\n\n    if renderer is None:\n        from sphinx.util.template import SphinxRenderer\n        renderer = SphinxRenderer()\n\n    ensuredir(destination)\n    if os.path.isfile(source):\n        copy_asset_file(source, destination, context, renderer)\n        return\n\n    for root, dirs, files in os.walk(source, followlinks=True):\n        reldir = relative_path(source, root)\n        for dir in dirs[:]:\n            if excluded(posixpath.join(reldir, dir)):\n                dirs.remove(dir)\n            else:\n                ensuredir(posixpath.join(destination, reldir, dir))\n\n        for filename in files:\n            if not excluded(posixpath.join(reldir, filename)):\n                copy_asset_file(posixpath.join(root, filename),\n                                posixpath.join(destination, reldir),\n                                context, renderer)"
        }
      ]
    },
    {
      "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": 7036,
      "pr_title": "Fix #7023: autodoc: partial functions are listed as module members",
      "pr_body": "### Feature or Bugfix\r\n- Bugfix\r\n\r\n### Purpose\r\n- refs: #7023 ",
      "issue_id": 7023,
      "issue_title": "Problems with autodoc and Python functools.partial and partialmethod",
      "issue_body": "Thanks for making sphinx! I am using sphinx and readthedocs to automatically generate docs whenever I push something to github. I am very impressed with the whole system!\r\n\r\nI have a few minor problems relating to sphinx autodoc and Python's `functools.partial` and `functools.partialmethod`. It is easiest if I show you the examples:\r\n\r\n\r\n### Problem 1:\r\n\r\nSphinx autodoc works fine with a single level of `functools.partial`, but when using it in a nested / recursive fashion, the docs are not shown in sphinx. See for example `load_fundamental` which uses `functools.partial` once, and then see `load_income` which uses `functools.partial` again on `load_fundamental`:\r\n\r\nhttps://github.com/simfin/simfin/blob/master/simfin/load.py#L174-L182\r\n\r\nBut only the docs for `load_fundamental` are generated by sphinx:\r\n\r\nhttps://simfin.readthedocs.io/en/latest/load.html#simfin.load.load_fundamental\r\n\r\n\r\n### Problem 2:\r\n\r\nSphinx autodoc does not seem to work at all for `functools.partialmethod` for class methods. See for example `load_fundamental` and `load_income` in this class:\r\n\r\nhttps://github.com/simfin/simfin/blob/master/simfin/hubs.py#L238-L264\r\n\r\nHere `load_fundamental` is a regular class method which has sphinx docs generated, but `load_income` which is a `partialmethod` does not have any docs generated by sphinx:\r\n\r\nhttps://simfin.readthedocs.io/en/latest/hubs.html#simfin.hubs.StockHub.load_fundamental\r\n\r\n\r\n### Problem 3:\r\n\r\nThe functions created with `functools.partial` in the module `load.py` are imported in another Python module `hubs.py`, but sphinx also shows them as belonging to `hubs.py` when using this rst-file:\r\n\r\nhttps://github.com/SimFin/simfin/blob/master/docs/hubs.rst\r\n\r\nThese are basically the contents of the rst-file:\r\n\r\n    .. automodule:: simfin.hubs\r\n       :members:\r\n\r\nThese two python modules are located here:\r\n\r\nhttps://github.com/SimFin/simfin/blob/master/simfin/load.py\r\nhttps://github.com/SimFin/simfin/blob/master/simfin/hubs.py\r\n\r\n\r\n### Problem 4:\r\n\r\nThis problem is not related to `functools.partial` but it's a very small problem, so rather than opening a separate issue, I include it here.\r\n\r\nThere is a minor inconsistency in how sphinx indents the doc-strings for parameters. Consider this page again:\r\n\r\nhttps://simfin.readthedocs.io/en/latest/load.html\r\n\r\nFor the function `load()` the doc-string for the first parameter `dataset` starts at a new line. This is also the case for the parameters `variant` and `market`. But for the parameter `parse_dates` the doc-string does not start with this newline.\r\n\r\nIf you look again at the Python code, it appears that sphinx inserts a newline at the beginning of the parameter's doc-string, IF there is a blank line somewhere in the doc-string for that parameter.\r\n\r\nhttps://github.com/SimFin/simfin/blob/master/simfin/load.py\r\n\r\n\r\nI hope you can fix these minor issues. Thanks again for making sphinx, it's a very cool system!",
      "issue_closed_at": "2020-01-19T13:47:13Z",
      "base_commit": "334d23f88079b2bceef37af23170815736f23ef7",
      "changes": [
        {
          "file": "sphinx/ext/autodoc/__init__.py",
          "type": "function",
          "name": "check_module",
          "class_name": "Documenter",
          "code": "def check_module(self) -> bool:\n        \"\"\"Check if *self.object* is really defined in the module given by\n        *self.modname*.\n        \"\"\"\n        if self.options.imported_members:\n            return True\n\n        modname = self.get_attr(self.object, '__module__', None)\n        if inspect.ispartial(self.object) and modname == '_functools':  # for pypy\n            return True\n        elif modname and modname != self.modname:\n            return False\n        return True"
        }
      ]
    }
  ]
}