{
    "Selected_candidate": {
        "pr_number": 6914,
        "pr_title": "Revert \"[parametrize] enforce explicit argnames declaration (#6330)\"",
        "pr_body": "This reverts commit 9e262038c84a99d1353551e8cbb32f46362b58b4.\r\n\r\nFix #6909\r\n",
        "issue_id": 6909,
        "issue_title": "Parameter \"loop_factory\" should be declared explicitly via indirect or in function itself",
        "issue_body": "As of pytest 5.4.0 there is an issue with async tests, at the very least when using aiohttp. Can confirm that downgrading pytest fixed the issue.\r\n\r\nHere's the same details I submitted over there when I thought aiohttp was the culprit:\r\n\r\nParameter \"loop_factory\" should be declared explicitly via indirect or in function itself\r\n\r\nThe commonality it turns out were async functions, and it appears to be caused somewhere within aiohttp. I'm not that familiar with asyncio or aiohttp, so I'm happy to debug more with guidance, but I'm hoping someone might have more thoughts here.\r\n\r\nRelated packages:\r\n\r\n```\r\naiohttp==3.6.2\r\npytest==5.4.0\r\npytest-aiohttp==0.3.0\r\npytest-cov==2.8.1\r\npytest-forked==1.1.3\r\npytest-mock==2.0.0\r\npytest-responses==0.4.0\r\npytest-xdist==1.31.0\r\n```\r\n\r\nSome detail around the callstack:\r\n\r\n```\r\n➜  ~/D/zeus (feat/increase-lru-cache) ✗ py.test -x --pdb\r\n=========================================================================================== test session starts ============================================================================================\r\nplatform darwin -- Python 3.8.1, pytest-5.4.0, py-1.8.1, pluggy-0.13.1\r\nrootdir: /Users/dcramer/Development/zeus, inifile: setup.cfg\r\nplugins: mock-2.0.0, celery-4.1.1, xdist-1.31.0, aiohttp-0.3.0, forked-1.1.3, responses-0.4.0, cov-2.8.1\r\ncollecting 186 items\r\n>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>> traceback >>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>\r\nIn function \"test_health_check\":\r\nParameter \"loop_factory\" should be declared explicitly via indirect or in function itself\r\n>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>> entering PDB >>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>\r\n\r\n>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>> PDB post_mortem (IO-capturing turned off) >>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>\r\n> /Users/dcramer/Development/zeus/.venv/lib/python3.8/site-packages/_pytest/python.py(1144)_validate_explicit_parameters()\r\n-> fail(msg, pytrace=False)\r\n(Pdb) l\r\n1139 \t                func_name = self.function.__name__\r\n1140 \t                msg = (\r\n1141 \t                    'In function \"{func_name}\":\\n'\r\n1142 \t                    'Parameter \"{arg}\" should be declared explicitly via indirect or in function itself'\r\n1143 \t                ).format(func_name=func_name, arg=arg)\r\n1144 ->\t                fail(msg, pytrace=False)\r\n1145\r\n1146\r\n1147 \tdef _find_parametrized_scope(argnames, arg2fixturedefs, indirect):\r\n1148 \t    \"\"\"Find the most appropriate scope for a parametrized call based on its arguments.\r\n1149\r\n(Pdb) u\r\n> /Users/dcramer/Development/zeus/.venv/lib/python3.8/site-packages/_pytest/python.py(939)parametrize()\r\n-> self._validate_explicit_parameters(argnames, indirect)\r\n(Pdb) l\r\n934\r\n935  \t        self._validate_if_using_arg_names(argnames, indirect)\r\n936\r\n937  \t        arg_values_types = self._resolve_arg_value_types(argnames, indirect)\r\n938\r\n939  ->\t        self._validate_explicit_parameters(argnames, indirect)\r\n940\r\n941  \t        # Use any already (possibly) generated ids with parametrize Marks.\r\n942  \t        if _param_mark and _param_mark._param_ids_from:\r\n943  \t            generated_ids = _param_mark._param_ids_from._param_ids_generated\r\n944  \t            if generated_ids is not None:\r\n(Pdb) u\r\n> /Users/dcramer/Development/zeus/.venv/lib/python3.8/site-packages/aiohttp/pytest_plugin.py(203)pytest_generate_tests()\r\n-> metafunc.parametrize(\"loop_factory\",\r\n(Pdb) l\r\n198  \t                    \"Unknown loop '%s', available loops: %s\" % (\r\n199  \t                        name, list(factories.keys())))\r\n200  \t            else:\r\n201  \t                continue\r\n202  \t        factories[name] = avail_factories[name]\r\n203  ->\t    metafunc.parametrize(\"loop_factory\",\r\n204  \t                         list(factories.values()),\r\n205  \t                         ids=list(factories.keys()))\r\n206\r\n207\r\n208  \t@pytest.fixture\r\n```\r\n\r\nhttps://github.com/aio-libs/aiohttp/issues/4626",
        "issue_closed_at": "2020-03-13T14:07:44Z",
        "base_commit": "59c1bfada7de3b3d239f5d7cfff04ce460c9aecd",
        "changes": [
            {
                "file": "src/_pytest/fixtures.py",
                "type": "line",
                "name": "line 1",
                "code": "import functools\nimport inspect\nimport sys\nimport warnings\nfrom collections import defaultdict"
            },
            {
                "file": "src/_pytest/fixtures.py",
                "type": "function",
                "name": "getfixtureinfo",
                "class_name": "FixtureManager",
                "code": "def getfixtureinfo(self, node, func, cls, funcargs=True):\n        if funcargs and not getattr(node, \"nofuncargs\", False):\n            argnames = getfuncargnames(func, name=node.name, cls=cls)\n        else:\n            argnames = ()\n\n        usefixtures = get_use_fixtures_for_node(node)\n        initialnames = usefixtures + argnames\n        fm = node.session._fixturemanager\n        initialnames, names_closure, arg2fixturedefs = fm.getfixtureclosure(\n            initialnames, node, ignore_args=self._get_direct_parametrize_args(node)\n        )\n        return FuncFixtureInfo(argnames, initialnames, names_closure, arg2fixturedefs)"
            },
            {
                "file": "src/_pytest/fixtures.py",
                "type": "function",
                "name": "_matchfactories",
                "class_name": "FixtureManager",
                "code": "def _matchfactories(self, fixturedefs, nodeid):\n        from _pytest import nodes\n\n        for fixturedef in fixturedefs:\n            if nodes.ischildnode(fixturedef.baseid, nodeid):\n                yield fixturedef"
            },
            {
                "file": "src/_pytest/python.py",
                "type": "function",
                "name": "parametrize",
                "class_name": "Metafunc",
                "code": "def parametrize(\n        self,\n        argnames: Union[str, List[str], Tuple[str, ...]],\n        argvalues: Iterable[Union[ParameterSet, typing.Sequence[object], object]],\n        indirect: Union[bool, typing.Sequence[str]] = False,\n        ids: Optional[\n            Union[\n                Iterable[Union[None, str, float, int, bool]],\n                Callable[[object], Optional[object]],\n            ]\n        ] = None,\n        scope: \"Optional[str]\" = None,\n        *,\n        _param_mark: Optional[Mark] = None\n    ) -> None:\n        \"\"\" Add new invocations to the underlying test function using the list\n        of argvalues for the given argnames.  Parametrization is performed\n        during the collection phase.  If you need to setup expensive resources\n        see about setting indirect to do it rather at test setup time.\n\n        :arg argnames: a comma-separated string denoting one or more argument\n                       names, or a list/tuple of argument strings.\n\n        :arg argvalues: The list of argvalues determines how often a\n            test is invoked with different argument values.  If only one\n            argname was specified argvalues is a list of values.  If N\n            argnames were specified, argvalues must be a list of N-tuples,\n            where each tuple-element specifies a value for its respective\n            argname.\n\n        :arg indirect: The list of argnames or boolean. A list of arguments'\n            names (subset of argnames). If True the list contains all names from\n            the argnames. Each argvalue corresponding to an argname in this list will\n            be passed as request.param to its respective argname fixture\n            function so that it can perform more expensive setups during the\n            setup phase of a test rather than at collection time.\n\n        :arg ids: sequence of (or generator for) ids for ``argvalues``,\n              or a callable to return part of the id for each argvalue.\n\n            With sequences (and generators like ``itertools.count()``) the\n            returned ids should be of type ``string``, ``int``, ``float``,\n            ``bool``, or ``None``.\n            They are mapped to the corresponding index in ``argvalues``.\n            ``None`` means to use the auto-generated id.\n\n            If it is a callable it will be called for each entry in\n            ``argvalues``, and the return value is used as part of the\n            auto-generated id for the whole set (where parts are joined with\n            dashes (\"-\")).\n            This is useful to provide more specific ids for certain items, e.g.\n            dates.  Returning ``None`` will use an auto-generated id.\n\n            If no ids are provided they will be generated automatically from\n            the argvalues.\n\n        :arg scope: if specified it denotes the scope of the parameters.\n            The scope is used for grouping tests by parameter instances.\n            It will also override any fixture-function defined scope, allowing\n            to set a dynamic scope using test context or configuration.\n        \"\"\"\n        from _pytest.fixtures import scope2index\n\n        argnames, parameters = ParameterSet._for_parametrize(\n            argnames,\n            argvalues,\n            self.function,\n            self.config,\n            function_definition=self.definition,\n        )\n        del argvalues\n\n        if \"request\" in argnames:\n            fail(\n                \"'request' is a reserved name and cannot be used in @pytest.mark.parametrize\",\n                pytrace=False,\n            )\n\n        if scope is None:\n            scope = _find_parametrized_scope(argnames, self._arg2fixturedefs, indirect)\n\n        self._validate_if_using_arg_names(argnames, indirect)\n\n        arg_values_types = self._resolve_arg_value_types(argnames, indirect)\n\n        self._validate_explicit_parameters(argnames, indirect)\n\n        # Use any already (possibly) generated ids with parametrize Marks.\n        if _param_mark and _param_mark._param_ids_from:\n            generated_ids = _param_mark._param_ids_from._param_ids_generated\n            if generated_ids is not None:\n                ids = generated_ids\n\n        ids = self._resolve_arg_ids(argnames, ids, parameters, item=self.definition)\n\n        # Store used (possibly generated) ids with parametrize Marks.\n        if _param_mark and _param_mark._param_ids_from and generated_ids is None:\n            object.__setattr__(_param_mark._param_ids_from, \"_param_ids_generated\", ids)\n\n        scopenum = scope2index(\n            scope, descr=\"parametrize() call in {}\".format(self.function.__name__)\n        )\n\n        # create the new calls: if we are parametrize() multiple times (by applying the decorator\n        # more than once) then we accumulate those calls generating the cartesian product\n        # of all calls\n        newcalls = []\n        for callspec in self._calls or [CallSpec2(self)]:\n            for param_index, (param_id, param_set) in enumerate(zip(ids, parameters)):\n                newcallspec = callspec.copy()\n                newcallspec.setmulti2(\n                    arg_values_types,\n                    argnames,\n                    param_set.values,\n                    param_id,\n                    param_set.marks,\n                    scopenum,\n                    param_index,\n                )\n                newcalls.append(newcallspec)\n        self._calls = newcalls"
            },
            {
                "file": "src/_pytest/python.py",
                "type": "function",
                "name": "_validate_if_using_arg_names",
                "class_name": "Metafunc",
                "code": "def _validate_if_using_arg_names(\n        self,\n        argnames: typing.Sequence[str],\n        indirect: Union[bool, typing.Sequence[str]],\n    ) -> None:\n        \"\"\"\n        Check if all argnames are being used, by default values, or directly/indirectly.\n\n        :param List[str] argnames: list of argument names passed to ``parametrize()``.\n        :param indirect: same ``indirect`` parameter of ``parametrize()``.\n        :raise ValueError: if validation fails.\n        \"\"\"\n        default_arg_names = set(get_default_arg_names(self.function))\n        func_name = self.function.__name__\n        for arg in argnames:\n            if arg not in self.fixturenames:\n                if arg in default_arg_names:\n                    fail(\n                        \"In {}: function already takes an argument '{}' with a default value\".format(\n                            func_name, arg\n                        ),\n                        pytrace=False,\n                    )\n                else:\n                    if isinstance(indirect, Sequence):\n                        name = \"fixture\" if arg in indirect else \"argument\"\n                    else:\n                        name = \"fixture\" if indirect else \"argument\"\n                    fail(\n                        \"In {}: function uses no {} '{}'\".format(func_name, name, arg),\n                        pytrace=False,\n                    )"
            }
        ]
    },
    "Justification": "Candidate C deals directly with parameterization and async test handling within pytest, which relates to how pytest interprets and manages test functions. While it isn't a perfect match, the underlying complexities of pytest's test collection mechanisms and asynchronous handling provide insights that could help debug how pytest is interpreting numerically starting expressions erroneously. Furthermore, the patch discusses explicit parameter declaration, which may offer conceptual avenues for understanding the current error."
}