{
  "Selected_candidate": {
    "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"
      }
    ]
  },
  "Justification": "Candidate A is the most relevant since both it and the CURRENT bug involve Sphinx's autodoc functionality and the parsing of type hints, specifically with use of `typing`. The nature of the bug reported in Candidate A suggests a failure when dealing with complex type hints, which resonates with the CURRENT bug regarding overloaded callables and the expected behavior when type hints are used. Additionally, both bugs are linked to how Sphinx processes function signatures, making Candidate A particularly helpful in identifying potential areas of investigation in the CURRENT bug.",
  "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"
}