{
    "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."
}