{
  "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 B directly deals with parsing issues related to type hints in Sphinx, which is highly relevant to the current issue being faced with `struct.Struct` type annotations. Both reports involve problems that arise during the autodoc processing of types, and the similar error message indicating a failure in handling annotations provides a crucial context for diagnosing and resolving the current bug. Strategies for patching that are applicable between the two further facilitate debugging efforts.",
  "instance_id": "sphinx-doc__sphinx-8627",
  "repo": "sphinx-doc/sphinx",
  "created_at": "2020-12-31T05:21:06Z",
  "problem_statement": "autodoc isn't able to resolve struct.Struct type annotations\n**Describe the bug**\r\nIf `struct.Struct` is declared in any type annotations, I get `class reference target not found: Struct`\r\n\r\n**To Reproduce**\r\nSimple `index.rst`\r\n```\r\nHello World\r\n===========\r\n\r\ncode docs\r\n=========\r\n\r\n.. automodule:: helloworld.helloworld\r\n```\r\n\r\nSimple `helloworld.py`\r\n```\r\nimport struct\r\nimport pathlib\r\n\r\ndef consume_struct(_: struct.Struct) -> None:\r\n    pass\r\n\r\ndef make_struct() -> struct.Struct:\r\n    mystruct = struct.Struct('HH')\r\n    return mystruct\r\n\r\ndef make_path() -> pathlib.Path:\r\n    return pathlib.Path()\r\n```\r\n\r\nCommand line:\r\n```\r\npython3 -m sphinx -b html docs/ doc-out -nvWT\r\n```\r\n\r\n**Expected behavior**\r\nIf you comment out the 2 functions that have `Struct` type annotations, you'll see that `pathlib.Path` resolves fine and shows up in the resulting documentation. I'd expect that `Struct` would also resolve correctly.\r\n\r\n**Your project**\r\nn/a\r\n\r\n**Screenshots**\r\nn/a\r\n\r\n**Environment info**\r\n- OS: Ubuntu 18.04, 20.04\r\n- Python version: 3.8.2\r\n- Sphinx version: 3.2.1\r\n- Sphinx extensions:  'sphinx.ext.autodoc',\r\n              'sphinx.ext.autosectionlabel',\r\n              'sphinx.ext.intersphinx',\r\n              'sphinx.ext.doctest',\r\n              'sphinx.ext.todo'\r\n- Extra tools: \r\n\r\n**Additional context**\r\n\r\n\r\n- [e.g. URL or Ticket]\r\n\r\n\n",
  "patch": "diff --git a/sphinx/util/typing.py b/sphinx/util/typing.py\n--- a/sphinx/util/typing.py\n+++ b/sphinx/util/typing.py\n@@ -10,6 +10,7 @@\n \n import sys\n import typing\n+from struct import Struct\n from typing import Any, Callable, Dict, Generator, List, Optional, Tuple, TypeVar, Union\n \n from docutils import nodes\n@@ -94,6 +95,9 @@ def restify(cls: Optional[\"Type\"]) -> str:\n         return ':obj:`None`'\n     elif cls is Ellipsis:\n         return '...'\n+    elif cls is Struct:\n+        # Before Python 3.9, struct.Struct class has incorrect __module__.\n+        return ':class:`struct.Struct`'\n     elif inspect.isNewType(cls):\n         return ':class:`%s`' % cls.__name__\n     elif cls.__module__ in ('__builtin__', 'builtins'):\n@@ -305,6 +309,9 @@ def stringify(annotation: Any) -> str:\n         return annotation.__qualname__\n     elif annotation is Ellipsis:\n         return '...'\n+    elif annotation is Struct:\n+        # Before Python 3.9, struct.Struct class has incorrect __module__.\n+        return 'struct.Struct'\n \n     if sys.version_info >= (3, 7):  # py37+\n         return _stringify_py37(annotation)\n"
}