{
  "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",
  "similar_bug_items": [
    {
      "pr_number": 7501,
      "pr_title": "Fix #7418: std domain: term role role could not match case-insensitively",
      "pr_body": "### Feature or Bugfix\r\n- Bugfix\r\n\r\n### Purpose\r\n- refs: #7418 ",
      "issue_id": 7418,
      "issue_title": "glossary duplicate term with a different case",
      "issue_body": "**Describe the bug**\r\n```\r\nWarning, treated as error:\r\ndoc/glossary.rst:243:duplicate term description of mysql, other instance in glossary\r\n```\r\n\r\n**To Reproduce**\r\nSteps to reproduce the behavior:\r\n[.travis.yml#L168](https://github.com/phpmyadmin/phpmyadmin/blob/f7cc383674b7099190771b1db510c62bfbbf89a7/.travis.yml#L168)\r\n```\r\n$ git clone --depth 1 https://github.com/phpmyadmin/phpmyadmin.git\r\n$ cd doc\r\n$ pip install 'Sphinx'\r\n$ make html\r\n```\r\n\r\n**Expected behavior**\r\nMySQL != mysql term right ?\r\n\r\n**Your project**\r\nhttps://github.com/phpmyadmin/phpmyadmin/blame/master/doc/glossary.rst#L234\r\n\r\n\r\n**Environment info**\r\n- OS: Unix\r\n- Python version: 3.6\r\n- Sphinx version: 3.0.0\r\n\r\n**Additional context**\r\nDid occur some hours ago, maybe you just released the version\r\n\r\n- https://travis-ci.org/github/williamdes/phpmyadmintest/jobs/671352365#L328\r\n\r\n",
      "issue_closed_at": "2020-04-09T15:12:11Z",
      "base_commit": "50d2d289e150cb429de15770bdd48a723de8c45d",
      "changes": [
        {
          "file": "sphinx/domains/std.py",
          "type": "function",
          "name": "resolve_xref",
          "class_name": "StandardDomain",
          "code": "def resolve_xref(self, env: \"BuildEnvironment\", fromdocname: str, builder: \"Builder\",\n                     typ: str, target: str, node: pending_xref, contnode: Element) -> Element:\n        if typ == 'ref':\n            resolver = self._resolve_ref_xref\n        elif typ == 'numref':\n            resolver = self._resolve_numref_xref\n        elif typ == 'keyword':\n            resolver = self._resolve_keyword_xref\n        elif typ == 'doc':\n            resolver = self._resolve_doc_xref\n        elif typ == 'option':\n            resolver = self._resolve_option_xref\n        elif typ == 'citation':\n            warnings.warn('pending_xref(domain=std, type=citation) is deprecated: %r' % node,\n                          RemovedInSphinx40Warning)\n            domain = env.get_domain('citation')\n            return domain.resolve_xref(env, fromdocname, builder, typ, target, node, contnode)\n        else:\n            resolver = self._resolve_obj_xref\n\n        return resolver(env, fromdocname, builder, typ, target, node, contnode)"
        },
        {
          "file": "sphinx/domains/std.py",
          "type": "function",
          "name": "_resolve_option_xref",
          "class_name": "StandardDomain",
          "code": "def _resolve_option_xref(self, env: \"BuildEnvironment\", fromdocname: str,\n                             builder: \"Builder\", typ: str, target: str,\n                             node: pending_xref, contnode: Element) -> Element:\n        progname = node.get('std:program')\n        target = target.strip()\n        docname, labelid = self.progoptions.get((progname, target), ('', ''))\n        if not docname:\n            commands = []\n            while ws_re.search(target):\n                subcommand, target = ws_re.split(target, 1)\n                commands.append(subcommand)\n                progname = \"-\".join(commands)\n\n                docname, labelid = self.progoptions.get((progname, target), ('', ''))\n                if docname:\n                    break\n            else:\n                return None\n\n        return make_refnode(builder, fromdocname, docname,\n                            labelid, contnode)"
        }
      ]
    },
    {
      "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": 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": 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": 6592,
      "pr_title": "Fix #6589: autodoc: Formatting issues with autodoc_typehints='none'",
      "pr_body": "### Feature or Bugfix\r\n- Bugfix\r\n\r\n### Purpose\r\n- refs: #6589 ",
      "issue_id": 6589,
      "issue_title": "Formatting issues with autodoc_typehints='none'",
      "issue_body": "**Describe the bug**\r\n\r\nWhen using `autodoc_typehints='none'`, I see two issues currently:\r\n\r\n1. When an annotated parameter has a default value, spaces are inserted around `=` where they are not for unannotated parameters. Consider:\r\n\r\n   ```python\r\n   def foo(x=True, y: bool = True, z = True):\r\n       return x and y and z\r\n   ```\r\n\r\n   The result looks like:\r\n\r\n   **foo**(*x=True, y = True, z=True*)\r\n\r\n2. Return types are not removed. Consider:\r\n\r\n   ```python\r\n   def bar(x: int) -> int:\r\n       return x * 2\r\n   ```\r\n\r\n   The result looks like:\r\n\r\n   **bar**(*x*) -> int\r\n\r\n**To Reproduce**\r\nSteps to reproduce the behavior:\r\n\r\nUse the above examples with the `autodoc_typehints='none'` option, or with the attached project (see below).\r\n\r\n**Expected behavior**\r\n\r\nI expect the annotated and unannotated parameters to be rendered similarly and for return type annotations to be removed as well, as if they were not there originally.\r\n\r\n**Your project**\r\n\r\nSee attached [foobar.zip](https://github.com/sphinx-doc/sphinx/files/3406054/foobar.zip)\r\n\r\n**Environment info**\r\n- OS: [Pop!_OS 18.10]\r\n- Python version: 3.6.8\r\n- Sphinx version: 2.1.2\r\n- Sphinx extensions:  [sphinx.ext.autodoc]\r\n\r\n**Additional context**\r\n\r\n- #6361 \r\n- #5868 \r\n- https://github.com/agronholm/sphinx-autodoc-typehints/pull/78\r\n\r\n",
      "issue_closed_at": "2019-08-02T13:37:16Z",
      "base_commit": "4732ec5edf9e53e2fa78cd5e1ff6bee92f1b27b7",
      "changes": [
        {
          "file": "sphinx/util/inspect.py",
          "type": "function",
          "name": "format_args",
          "class_name": "Signature",
          "code": "def format_args(self, show_annotation: bool = True) -> str:\n        args = []\n        last_kind = None\n        for i, param in enumerate(self.parameters.values()):\n            # skip first argument if subject is bound method\n            if self.skip_first_argument and i == 0:\n                continue\n\n            arg = StringIO()\n\n            # insert '*' between POSITIONAL args and KEYWORD_ONLY args::\n            #     func(a, b, *, c, d):\n            if param.kind == param.KEYWORD_ONLY and last_kind in (param.POSITIONAL_OR_KEYWORD,\n                                                                  param.POSITIONAL_ONLY,\n                                                                  None):\n                args.append('*')\n\n            if param.kind in (param.POSITIONAL_ONLY,\n                              param.POSITIONAL_OR_KEYWORD,\n                              param.KEYWORD_ONLY):\n                arg.write(param.name)\n                if show_annotation and param.annotation is not param.empty:\n                    if isinstance(param.annotation, str) and param.name in self.annotations:\n                        arg.write(': ')\n                        arg.write(self.format_annotation(self.annotations[param.name]))\n                    else:\n                        arg.write(': ')\n                        arg.write(self.format_annotation(param.annotation))\n                if param.default is not param.empty:\n                    if param.annotation is param.empty:\n                        arg.write('=')\n                        arg.write(object_description(param.default))\n                    else:\n                        arg.write(' = ')\n                        arg.write(object_description(param.default))\n            elif param.kind == param.VAR_POSITIONAL:\n                arg.write('*')\n                arg.write(param.name)\n            elif param.kind == param.VAR_KEYWORD:\n                arg.write('**')\n                arg.write(param.name)\n\n            args.append(arg.getvalue())\n            last_kind = param.kind\n\n        if self.return_annotation is inspect.Parameter.empty:\n            return '(%s)' % ', '.join(args)\n        else:\n            if 'return' in self.annotations:\n                annotation = self.format_annotation(self.annotations['return'])\n            else:\n                annotation = self.format_annotation(self.return_annotation)\n\n            return '(%s) -> %s' % (', '.join(args), annotation)"
        },
        {
          "file": "sphinx/util/inspect.py",
          "type": "function",
          "name": "format_args",
          "class_name": "Signature",
          "code": "def format_args(self, show_annotation: bool = True) -> str:\n        args = []\n        last_kind = None\n        for i, param in enumerate(self.parameters.values()):\n            # skip first argument if subject is bound method\n            if self.skip_first_argument and i == 0:\n                continue\n\n            arg = StringIO()\n\n            # insert '*' between POSITIONAL args and KEYWORD_ONLY args::\n            #     func(a, b, *, c, d):\n            if param.kind == param.KEYWORD_ONLY and last_kind in (param.POSITIONAL_OR_KEYWORD,\n                                                                  param.POSITIONAL_ONLY,\n                                                                  None):\n                args.append('*')\n\n            if param.kind in (param.POSITIONAL_ONLY,\n                              param.POSITIONAL_OR_KEYWORD,\n                              param.KEYWORD_ONLY):\n                arg.write(param.name)\n                if show_annotation and param.annotation is not param.empty:\n                    if isinstance(param.annotation, str) and param.name in self.annotations:\n                        arg.write(': ')\n                        arg.write(self.format_annotation(self.annotations[param.name]))\n                    else:\n                        arg.write(': ')\n                        arg.write(self.format_annotation(param.annotation))\n                if param.default is not param.empty:\n                    if param.annotation is param.empty:\n                        arg.write('=')\n                        arg.write(object_description(param.default))\n                    else:\n                        arg.write(' = ')\n                        arg.write(object_description(param.default))\n            elif param.kind == param.VAR_POSITIONAL:\n                arg.write('*')\n                arg.write(param.name)\n            elif param.kind == param.VAR_KEYWORD:\n                arg.write('**')\n                arg.write(param.name)\n\n            args.append(arg.getvalue())\n            last_kind = param.kind\n\n        if self.return_annotation is inspect.Parameter.empty:\n            return '(%s)' % ', '.join(args)\n        else:\n            if 'return' in self.annotations:\n                annotation = self.format_annotation(self.annotations['return'])\n            else:\n                annotation = self.format_annotation(self.return_annotation)\n\n            return '(%s) -> %s' % (', '.join(args), annotation)"
        }
      ]
    }
  ]
}