{
  "Selected_candidate": {
    "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)"
      }
    ]
  },
  "Justification": "Candidate A is the most relevant because it directly deals with issues arising from the autodoc functionality of Sphinx, specifically related to handling *args and **kwargs, which is part of the CURRENT bug report. Both bugs involve the autodoc_typehints functionality and the nuances of parameter decoration in documentation, making the insights gained from Candidate A particularly valuable in addressing the duplicated parameters issue in the CURRENT bug. The structural similarity in how parameters are processed and the output transformations being affected are critical for debugging the CURRENT issue effectively.",
  "instance_id": "sphinx-doc__sphinx-10451",
  "repo": "sphinx-doc/sphinx",
  "created_at": "2022-05-15T11:49:39Z",
  "problem_statement": "Fix duplicated *args and **kwargs with autodoc_typehints\nFix duplicated *args and **kwargs with autodoc_typehints\r\n\r\n### Bugfix\r\n- Bugfix\r\n\r\n### Detail\r\nConsider this\r\n```python\r\nclass _ClassWithDocumentedInitAndStarArgs:\r\n    \"\"\"Class docstring.\"\"\"\r\n\r\n    def __init__(self, x: int, *args: int, **kwargs: int) -> None:\r\n        \"\"\"Init docstring.\r\n\r\n        :param x: Some integer\r\n        :param *args: Some integer\r\n        :param **kwargs: Some integer\r\n        \"\"\"\r\n```\r\nwhen using the autodoc extension and the setting `autodoc_typehints = \"description\"`.\r\n\r\nWIth sphinx 4.2.0, the current output is\r\n```\r\nClass docstring.\r\n\r\n   Parameters:\r\n      * **x** (*int*) --\r\n\r\n      * **args** (*int*) --\r\n\r\n      * **kwargs** (*int*) --\r\n\r\n   Return type:\r\n      None\r\n\r\n   __init__(x, *args, **kwargs)\r\n\r\n      Init docstring.\r\n\r\n      Parameters:\r\n         * **x** (*int*) -- Some integer\r\n\r\n         * ***args** --\r\n\r\n           Some integer\r\n\r\n         * ****kwargs** --\r\n\r\n           Some integer\r\n\r\n         * **args** (*int*) --\r\n\r\n         * **kwargs** (*int*) --\r\n\r\n      Return type:\r\n         None\r\n```\r\nwhere the *args and **kwargs are duplicated and incomplete.\r\n\r\nThe expected output is\r\n```\r\n  Class docstring.\r\n\r\n   Parameters:\r\n      * **x** (*int*) --\r\n\r\n      * ***args** (*int*) --\r\n\r\n      * ****kwargs** (*int*) --\r\n\r\n   Return type:\r\n      None\r\n\r\n   __init__(x, *args, **kwargs)\r\n\r\n      Init docstring.\r\n\r\n      Parameters:\r\n         * **x** (*int*) -- Some integer\r\n\r\n         * ***args** (*int*) --\r\n\r\n           Some integer\r\n\r\n         * ****kwargs** (*int*) --\r\n\r\n           Some integer\r\n\r\n      Return type:\r\n         None\r\n\r\n```\n",
  "patch": "diff --git a/sphinx/ext/autodoc/typehints.py b/sphinx/ext/autodoc/typehints.py\n--- a/sphinx/ext/autodoc/typehints.py\n+++ b/sphinx/ext/autodoc/typehints.py\n@@ -115,7 +115,15 @@ def modify_field_list(node: nodes.field_list, annotations: Dict[str, str],\n         if name == 'return':\n             continue\n \n-        arg = arguments.get(name, {})\n+        if '*' + name in arguments:\n+            name = '*' + name\n+            arguments.get(name)\n+        elif '**' + name in arguments:\n+            name = '**' + name\n+            arguments.get(name)\n+        else:\n+            arg = arguments.get(name, {})\n+\n         if not arg.get('type'):\n             field = nodes.field()\n             field += nodes.field_name('', 'type ' + name)\n@@ -167,13 +175,19 @@ def augment_descriptions_with_types(\n             has_type.add('return')\n \n     # Add 'type' for parameters with a description but no declared type.\n-    for name in annotations:\n+    for name, annotation in annotations.items():\n         if name in ('return', 'returns'):\n             continue\n+\n+        if '*' + name in has_description:\n+            name = '*' + name\n+        elif '**' + name in has_description:\n+            name = '**' + name\n+\n         if name in has_description and name not in has_type:\n             field = nodes.field()\n             field += nodes.field_name('', 'type ' + name)\n-            field += nodes.field_body('', nodes.paragraph('', annotations[name]))\n+            field += nodes.field_body('', nodes.paragraph('', annotation))\n             node += field\n \n     # Add 'rtype' if 'return' is present and 'rtype' isn't.\n"
}