{
  "instance_id": "pytest-dev__pytest-11143",
  "repo": "pytest-dev/pytest",
  "created_at": "2023-06-26T06:44:43Z",
  "problem_statement": "Rewrite fails when first expression of file is a number and mistaken as docstring \n<!--\r\nThanks for submitting an issue!\r\n\r\nQuick check-list while reporting bugs:\r\n-->\r\n\r\n- [x] a detailed description of the bug or problem you are having\r\n- [x] output of `pip list` from the virtual environment you are using\r\n- [x] pytest and operating system versions\r\n- [x] minimal example if possible\r\n```\r\nInstalling collected packages: zipp, six, PyYAML, python-dateutil, MarkupSafe, importlib-metadata, watchdog, tomli, soupsieve, pyyaml-env-tag, pycparser, pluggy, packaging, mergedeep, Markdown, jinja2, iniconfig, ghp-import, exceptiongroup, click, websockets, urllib3, tqdm, smmap, pytest, pyee, mkdocs, lxml, importlib-resources, idna, cssselect, charset-normalizer, cffi, certifi, beautifulsoup4, attrs, appdirs, w3lib, typing-extensions, texttable, requests, pyzstd, pytest-metadata, pyquery, pyppmd, pyppeteer, pynacl, pymdown-extensions, pycryptodomex, pybcj, pyasn1, py, psutil, parse, multivolumefile, mkdocs-autorefs, inflate64, gitdb, fake-useragent, cryptography, comtypes, bs4, brotli, bcrypt, allure-python-commons, xlwt, xlrd, rsa, requests-html, pywinauto, python-i18n, python-dotenv, pytest-rerunfailures, pytest-html, pytest-check, PySocks, py7zr, paramiko, mkdocstrings, loguru, GitPython, ftputil, crcmod, chardet, brotlicffi, allure-pytest\r\nSuccessfully installed GitPython-3.1.31 Markdown-3.3.7 MarkupSafe-2.1.3 PySocks-1.7.1 PyYAML-6.0 allure-pytest-2.13.2 allure-python-commons-2.13.2 appdirs-1.4.4 attrs-23.1.0 bcrypt-4.0.1 beautifulsoup4-4.12.2 brotli-1.0.9 brotlicffi-1.0.9.2 bs4-0.0.1 certifi-2023.5.7 cffi-1.15.1 chardet-5.1.0 charset-normalizer-3.1.0 click-8.1.3 comtypes-1.2.0 crcmod-1.7 cryptography-41.0.1 cssselect-1.2.0 exceptiongroup-1.1.1 fake-useragent-1.1.3 ftputil-5.0.4 ghp-import-2.1.0 gitdb-4.0.10 idna-3.4 importlib-metadata-6.7.0 importlib-resources-5.12.0 inflate64-0.3.1 iniconfig-2.0.0 jinja2-3.1.2 loguru-0.7.0 lxml-4.9.2 mergedeep-1.3.4 mkdocs-1.4.3 mkdocs-autorefs-0.4.1 mkdocstrings-0.22.0 multivolumefile-0.2.3 packaging-23.1 paramiko-3.2.0 parse-1.19.1 pluggy-1.2.0 psutil-5.9.5 py-1.11.0 py7zr-0.20.5 pyasn1-0.5.0 pybcj-1.0.1 pycparser-2.21 pycryptodomex-3.18.0 pyee-8.2.2 pymdown-extensions-10.0.1 pynacl-1.5.0 pyppeteer-1.0.2 pyppmd-1.0.0 pyquery-2.0.0 pytest-7.4.0 pytest-check-2.1.5 pytest-html-3.2.0 pytest-metadata-3.0.0 pytest-rerunfailures-11.1.2 python-dateutil-2.8.2 python-dotenv-1.0.0 python-i18n-0.3.9 pywinauto-0.6.6 pyyaml-env-tag-0.1 pyzstd-0.15.9 requests-2.31.0 requests-html-0.10.0 rsa-4.9 six-1.16.0 smmap-5.0.0 soupsieve-2.4.1 texttable-1.6.7 tomli-2.0.1 tqdm-4.65.0 typing-extensions-4.6.3 urllib3-1.26.16 w3lib-2.1.1 watchdog-3.0.0 websockets-10.4 xlrd-2.0.1 xlwt-1.3.0 zipp-3.15.0\r\n```\r\nuse `pytest -k xxx`\uff0c report an error\uff1a`TypeError: argument of type 'int' is not iterable`\r\n\r\nit seems a error in collecting testcase\r\n```\r\n==================================== ERRORS ====================================\r\n_ ERROR collecting testcases/\u57fa\u7ebf/\u4ee3\u7406\u7b56\u7565/SOCKS\u4e8c\u7ea7\u4ee3\u7406\u8fed\u4ee3\u4e8c/\u5728\u7ebf\u7528\u6237/\u5728\u7ebf\u7528\u6237\u66f4\u65b0/\u4e0a\u7ebf\u7528\u6237/test_socks_user_011.py _\r\n/usr/local/lib/python3.8/site-packages/_pytest/runner.py:341: in from_call\r\n    result: Optional[TResult] = func()\r\n/usr/local/lib/python3.8/site-packages/_pytest/runner.py:372: in <lambda>\r\n    call = CallInfo.from_call(lambda: list(collector.collect()), \"collect\")\r\n/usr/local/lib/python3.8/site-packages/_pytest/python.py:531: in collect\r\n    self._inject_setup_module_fixture()\r\n/usr/local/lib/python3.8/site-packages/_pytest/python.py:545: in _inject_setup_module_fixture\r\n    self.obj, (\"setUpModule\", \"setup_module\")\r\n/usr/local/lib/python3.8/site-packages/_pytest/python.py:310: in obj\r\n    self._obj = obj = self._getobj()\r\n/usr/local/lib/python3.8/site-packages/_pytest/python.py:528: in _getobj\r\n    return self._importtestmodule()\r\n/usr/local/lib/python3.8/site-packages/_pytest/python.py:617: in _importtestmodule\r\n    mod = import_path(self.path, mode=importmode, root=self.config.rootpath)\r\n/usr/local/lib/python3.8/site-packages/_pytest/pathlib.py:565: in import_path\r\n    importlib.import_module(module_name)\r\n/usr/local/lib/python3.8/importlib/__init__.py:127: in import_module\r\n    return _bootstrap._gcd_import(name[level:], package, level)\r\n<frozen importlib._bootstrap>:1014: in _gcd_import\r\n    ???\r\n<frozen importlib._bootstrap>:991: in _find_and_load\r\n    ???\r\n<frozen importlib._bootstrap>:975: in _find_and_load_unlocked\r\n    ???\r\n<frozen importlib._bootstrap>:671: in _load_unlocked\r\n    ???\r\n/usr/local/lib/python3.8/site-packages/_pytest/assertion/rewrite.py:169: in exec_module\r\n    source_stat, co = _rewrite_test(fn, self.config)\r\n/usr/local/lib/python3.8/site-packages/_pytest/assertion/rewrite.py:352: in _rewrite_test\r\n    rewrite_asserts(tree, source, strfn, config)\r\n/usr/local/lib/python3.8/site-packages/_pytest/assertion/rewrite.py:413: in rewrite_asserts\r\n    AssertionRewriter(module_path, config, source).run(mod)\r\n/usr/local/lib/python3.8/site-packages/_pytest/assertion/rewrite.py:695: in run\r\n    if self.is_rewrite_disabled(doc):\r\n/usr/local/lib/python3.8/site-packages/_pytest/assertion/rewrite.py:760: in is_rewrite_disabled\r\n    return \"PYTEST_DONT_REWRITE\" in docstring\r\nE   TypeError: argument of type 'int' is not iterable\r\n```\n",
  "patch": "diff --git a/src/_pytest/assertion/rewrite.py b/src/_pytest/assertion/rewrite.py\n--- a/src/_pytest/assertion/rewrite.py\n+++ b/src/_pytest/assertion/rewrite.py\n@@ -676,6 +676,7 @@ def run(self, mod: ast.Module) -> None:\n                 expect_docstring\n                 and isinstance(item, ast.Expr)\n                 and isinstance(item.value, ast.Constant)\n+                and isinstance(item.value.value, str)\n             ):\n                 doc = item.value.value\n                 if self.is_rewrite_disabled(doc):\n",
  "similar_bug_items": [
    {
      "pr_number": 5559,
      "pr_title": "Merge master into features",
      "pr_body": "",
      "issue_id": 5547,
      "issue_title": "pytest stepwise doesn't work with xfail strict failures",
      "issue_body": "```\r\ngraingert@onomastic:~/projects/foo$ cat tests/test_foo.py \r\nimport pytest\r\n\r\n\r\n@pytest.mark.xfail(reason=\"pass\")\r\ndef test_a():\r\n    pass\r\n\r\n\r\n@pytest.mark.xfail(reason=\"pass\")\r\ndef test_b():\r\n    pass\r\ngraingert@onomastic:~/projects/foo$ cat tests/pytest.ini \r\n[pytest]\r\naddopts = --strict\r\nxfail_strict=true\r\ngraingert@onomastic:~/projects/foo$ pytest --sw tests/\r\n================================ test session starts ================================\r\nplatform linux -- Python 3.7.3, pytest-5.0.0, py-1.8.0, pluggy-0.12.0\r\nrootdir: /home/graingert/projects/foo/tests, inifile: pytest.ini\r\ncollected 2 items                                                                   \r\nstepwise: no previously failed tests, not skipping.\r\n\r\ntests/test_foo.py FF                                                          [100%]\r\n\r\n===================================== FAILURES ======================================\r\n______________________________________ test_a _______________________________________\r\n[XPASS(strict)] pass\r\n______________________________________ test_b _______________________________________\r\n[XPASS(strict)] pass\r\n============================= 2 failed in 0.01 seconds ==============================\r\n```",
      "issue_closed_at": "2019-07-04T23:50:34Z",
      "base_commit": "60a358fa2dc82a571c68d1be2d25703b51351538",
      "changes": [
        {
          "file": "doc/en/example/nonpython/conftest.py",
          "type": "line",
          "name": "line 3",
          "code": "\n\ndef pytest_collect_file(parent, path):\n    if path.ext == \".yml\" and path.basename.startswith(\"test\"):\n        return YamlFile(path, parent)\n\n"
        },
        {
          "file": "src/_pytest/_code/code.py",
          "type": "function",
          "name": "match",
          "class_name": "ExceptionInfo",
          "code": "def match(self, regexp):\n        \"\"\"\n        Check whether the regular expression 'regexp' is found in the string\n        representation of the exception using ``re.search``. If it matches\n        then True is returned (so that it is possible to write\n        ``assert excinfo.match()``). If it doesn't match an AssertionError is\n        raised.\n        \"\"\"\n        __tracebackhide__ = True\n        if not re.search(regexp, str(self.value)):\n            assert 0, \"Pattern '{!s}' not found in '{!s}'\".format(regexp, self.value)\n        return True"
        },
        {
          "file": "src/_pytest/stepwise.py",
          "type": "function",
          "name": "pytest_collection_modifyitems",
          "class_name": "StepwisePlugin",
          "code": "def pytest_collection_modifyitems(self, session, config, items):\n        if not self.active:\n            return\n        if not self.lastfailed:\n            self.report_status = \"no previously failed tests, not skipping.\"\n            return\n\n        already_passed = []\n        found = False\n\n        # Make a list of all tests that have been run before the last failing one.\n        for item in items:\n            if item.nodeid == self.lastfailed:\n                found = True\n                break\n            else:\n                already_passed.append(item)\n\n        # If the previously failed test was not found among the test items,\n        # do not skip any tests.\n        if not found:\n            self.report_status = \"previously failed test not found, not skipping.\"\n            already_passed = []\n        else:\n            self.report_status = \"skipping {} already passed items.\".format(\n                len(already_passed)\n            )\n\n        for item in already_passed:\n            items.remove(item)\n\n        config.hook.pytest_deselected(items=already_passed)"
        }
      ]
    },
    {
      "pr_number": 4524,
      "pr_title": "Merge master into features",
      "pr_body": "",
      "issue_id": 4487,
      "issue_title": "Logging done late enough might happen when capture already stopped.",
      "issue_body": "```\r\n22:56:26,997 [pytestsalt.utils:206 ][INFO    ][1024] Stopping process psutil.Process(pid=1228, name='/home/vampas/.d', started='22:55:17') and respective children: [psutil.Process(pid=1245, name='/home/vampas/.d', started='22:55:18'), psut\r\nil.Process(pid=1262, name='/home/vampas/.d', started='22:55:19'), psutil.Process(pid=1263, name='/home/vampas/.d', started='22:55:19')]                                                                                                        \r\nError in atexit._run_exitfuncs:                                                                                                                                                                                                                \r\nTraceback (most recent call last):                                                                                                                                                                                                             \r\n  File \"/home/vampas/.dotfiles/.ext/pyenv/versions/3.5.4/lib/python3.5/logging/__init__.py\", line 861, in handle                                                                                                                               \r\n    self.emit(record)                                                                                                                                                                                                                          \r\n  File \"/home/vampas/.dotfiles/.ext/pyenv/versions/3.5.4/envs/Salt-3.5.4-Develop/lib/python3.5/site-packages/_pytest/logging.py\", line 625, in emit                                                                                            \r\n    logging.StreamHandler.emit(self, record)\r\n  File \"/home/vampas/.dotfiles/.ext/pyenv/versions/3.5.4/lib/python3.5/contextlib.py\", line 66, in __exit__\r\n    next(self.gen)\r\n  File \"/home/vampas/.dotfiles/.ext/pyenv/versions/3.5.4/envs/Salt-3.5.4-Develop/lib/python3.5/site-packages/_pytest/capture.py\", line 167, in global_and_fixture_disabled\r\n    self.resume_global_capture()\r\n  File \"/home/vampas/.dotfiles/.ext/pyenv/versions/3.5.4/envs/Salt-3.5.4-Develop/lib/python3.5/site-packages/_pytest/capture.py\", line 120, in resume_global_capture\r\n    self._global_capturing.resume_capturing()\r\nAttributeError: 'NoneType' object has no attribute 'resume_capturing'\r\n```\r\n- [x] Create a new changelog file in the `changelog` folder, with a name like `<ISSUE NUMBER>.<TYPE>.rst`. See [changelog/README.rst](https://github.com/pytest-dev/pytest/blob/master/changelog/README.rst) for details.\r\n- [x] Target the `master` branch for bug fixes, documentation updates and trivial changes.",
      "issue_closed_at": "2018-12-02T11:10:00Z",
      "base_commit": "76884c73bf1ba0cd9fa5ed6e4ba65931fa38096d",
      "changes": [
        {
          "file": "src/_pytest/capture.py",
          "type": "function",
          "name": "stop_global_capturing",
          "class_name": "CaptureManager",
          "code": "def stop_global_capturing(self):\n        if self._global_capturing is not None:\n            self._global_capturing.pop_outerr_to_orig()\n            self._global_capturing.stop_capturing()\n            self._global_capturing = None"
        },
        {
          "file": "src/_pytest/config/__init__.py",
          "type": "function",
          "name": "_mark_plugins_for_rewrite",
          "class_name": "Config",
          "code": "def _mark_plugins_for_rewrite(self, hook):\n        \"\"\"\n        Given an importhook, mark for rewrite any top-level\n        modules or packages in the distribution package for\n        all pytest plugins.\n        \"\"\"\n        import pkg_resources\n\n        self.pluginmanager.rewrite_hook = hook\n\n        if os.environ.get(\"PYTEST_DISABLE_PLUGIN_AUTOLOAD\"):\n            # We don't autoload from setuptools entry points, no need to continue.\n            return\n\n        # 'RECORD' available for plugins installed normally (pip install)\n        # 'SOURCES.txt' available for plugins installed in dev mode (pip install -e)\n        # for installed plugins 'SOURCES.txt' returns an empty list, and vice-versa\n        # so it shouldn't be an issue\n        metadata_files = \"RECORD\", \"SOURCES.txt\"\n\n        package_files = (\n            entry.split(\",\")[0]\n            for entrypoint in pkg_resources.iter_entry_points(\"pytest11\")\n            for metadata in metadata_files\n            for entry in entrypoint.dist._get_metadata(metadata)\n        )\n\n        for name in _iter_rewritable_modules(package_files):\n            hook.mark_rewrite(name)"
        }
      ]
    },
    {
      "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\u279c  ~/D/zeus (feat/increase-lru-cache) \u2717 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                    )"
        }
      ]
    },
    {
      "pr_number": 2113,
      "pr_title": "Use a simple ``+-`` ASCII string in the string representation of pytest.approx In Python 2",
      "pr_body": "Fix #2111",
      "issue_id": 2111,
      "issue_title": "Printing detailed diff with pytest.approx fails due to \u00b1 character",
      "issue_body": "OS: OS X 10.11.6 (also present on Ubuntu 14.04)\r\nPython version: 2.7.12\r\npytest version: 3.0.4\r\nOutput of `pip list`:\r\n```\r\npip (9.0.1)\r\npy (1.4.31)\r\npytest (3.0.4)\r\nsetuptools (30.0.0)\r\nwheel (0.30.0a0)\r\n```\r\n---\r\nThis code:\r\n```\r\nimport pytest\r\n\r\ndef test_foo():\r\n    assert [3] == [pytest.approx(4)]\r\n```\r\nproduces the below output. It looks like the detailed diff can't render due to the \u00b1 character in the `__repr__` of `pytest.approx`.\r\n```\r\n(venv) \u279c  /tmp py.test pytest_broken.py \r\n============================= test session starts ==============================\r\nplatform darwin -- Python 2.7.12, pytest-3.0.4, py-1.4.31, pluggy-0.4.0\r\nrootdir: /private/tmp, inifile: \r\nplugins: cov-2.2.1, mock-0.11.0\r\ncollected 1 items \r\n\r\npytest_broken.py F\r\n\r\n=================================== FAILURES ===================================\r\n___________________________________ test_foo ___________________________________\r\n\r\n    def test_foo():\r\n>       assert [3] == [pytest.approx(4)]\r\nE       assert [3] == [4 \u00b1 4.0e-06]\r\nE         (pytest_assertion plugin: representation of details failed.  Probably an object has a faulty __repr__.)\r\nE         /usr/local/lib/python2.7/site-packages/_pytest/assertion/util.py:228: UnicodeDecodeError: 'ascii' codec can't decode byte 0xc2 in position 2: ordinal not in range(128)\r\n\r\npytest_broken.py:4: AssertionError\r\n============================ pytest-warning summary ============================\r\nWC1 None pytest_funcarg__cov: declaring fixtures using \"pytest_funcarg__\" prefix is deprecated and scheduled to be removed in pytest 4.0.  Please remove the prefix and use the @pytest.fixture decorator instead.\r\n================= 1 failed, 1 pytest-warnings in 0.03 seconds ==================\r\n\r\n```\r\n",
      "issue_closed_at": "2016-12-05T12:02:07Z",
      "base_commit": "5365f7c9ca7323f2df932a6ea34fcee280e2672d",
      "changes": [
        {
          "file": "_pytest/python.py",
          "type": "function",
          "name": "__repr__",
          "class_name": "ApproxNonIterable",
          "code": "def __repr__(self):\n        if isinstance(self.expected, complex):\n            return str(self.expected)\n\n        # Infinities aren't compared using tolerances, so don't show a\n        # tolerance.\n        if math.isinf(self.expected):\n            return str(self.expected)\n\n        # If a sensible tolerance can't be calculated, self.tolerance will\n        # raise a ValueError.  In this case, display '???'.\n        try:\n            vetted_tolerance = '{:.1e}'.format(self.tolerance)\n        except ValueError:\n            vetted_tolerance = '???'\n\n        plus_minus = u'{0} \\u00b1 {1}'.format(self.expected, vetted_tolerance)\n\n        # In python2, __repr__() must return a string (i.e. not a unicode\n        # object).  In python3, __repr__() must return a unicode object\n        # (although now strings are unicode objects and bytes are what\n        # strings were).\n        if sys.version_info[0] == 2:\n            return plus_minus.encode('utf-8')\n        else:\n            return plus_minus"
        }
      ]
    },
    {
      "pr_number": 5539,
      "pr_title": "Replace importlib_metadata with importlib.metadata on Python 3.8+",
      "pr_body": "Fixes https://github.com/pytest-dev/pytest/issues/5537\r\n\r\n<!--\r\nThanks for submitting a PR, your contribution is really appreciated!\r\n\r\nHere is a quick checklist that should be present in PRs.\r\n(please delete this text from the final description, this is just a guideline)\r\n-->\r\n\r\n- [x] Target the `features` branch\r\n\r\nUnless your change is trivial or a small documentation fix (e.g.,  a typo or reword of a small section) please:\r\n\r\n- [x] Create a new changelog file in the `changelog` folder, with a name like `<ISSUE NUMBER>.<TYPE>.rst`. See [changelog/README.rst](https://github.com/pytest-dev/pytest/blob/master/changelog/README.rst) for details.\r\n- [x] Add yourself to `AUTHORS` in alphabetical order;\r\n\r\n\r\nI'm still running tox locally, there are some 3.8 failures, not yet sure if related.",
      "issue_id": 5523,
      "issue_title": "Command line parsing error with Python 3.8",
      "issue_body": "- [X] a detailed description of the bug or suggestion\r\nCommand line parsing error with Python 3.8 when concatenating short options\r\n- [X] pytest and operating system versions\r\nPytest 5.0.0, Python 3.8.0b1, Windows 10\r\n- [X] minimal example if possible\r\n```\r\n> pytest -vx\r\nERROR: usage: pytest [options] [file_or_dir] [file_or_dir] [...]\r\npytest: error: unrecognized arguments: -vx\r\n  inifile: None\r\n  rootdir: F:\\Documents\\Scripts\\Runscript\r\n```\r\n`pytest -v -x` works\r\n",
      "issue_closed_at": "2019-06-29T15:33:00Z",
      "base_commit": "4f9bf028f503dbf99a5339db8466757e71647918",
      "changes": [
        {
          "file": "setup.py",
          "type": "line",
          "name": "line 11",
          "code": "    'pathlib2>=2.2.0;python_version<\"3.6\"',\n    'colorama;sys_platform==\"win32\"',\n    \"pluggy>=0.12,<1.0\",\n    \"importlib-metadata>=0.12\",\n    \"wcwidth\",\n]\n"
        },
        {
          "file": "src/_pytest/compat.py",
          "type": "line",
          "name": "line 26",
          "code": ")\n\n\ndef _format_args(func):\n    return str(signature(func))\n"
        },
        {
          "file": "src/_pytest/config/__init__.py",
          "type": "line",
          "name": "line 9",
          "code": "import warnings\nfrom functools import lru_cache\n\nimport importlib_metadata\nimport py\nfrom packaging.version import Version\nfrom pluggy import HookimplMarker"
        },
        {
          "file": "src/_pytest/config/__init__.py",
          "type": "line",
          "name": "line 25",
          "code": "from .findpaths import exists\nfrom _pytest._code import ExceptionInfo\nfrom _pytest._code import filter_traceback\nfrom _pytest.outcomes import fail\nfrom _pytest.outcomes import Skipped\nfrom _pytest.warning_types import PytestConfigWarning"
        },
        {
          "file": "src/_pytest/config/argparsing.py",
          "type": "function",
          "name": "parse_args",
          "class_name": "MyOptionParser",
          "code": "def parse_args(self, args=None, namespace=None):\n        \"\"\"allow splitting of positional arguments\"\"\"\n        args, argv = self.parse_known_args(args, namespace)\n        if argv:\n            for arg in argv:\n                if arg and arg[0] == \"-\":\n                    lines = [\"unrecognized arguments: %s\" % (\" \".join(argv))]\n                    for k, v in sorted(self.extra_info.items()):\n                        lines.append(\"  {}: {}\".format(k, v))\n                    self.error(\"\\n\".join(lines))\n            getattr(args, FILE_OR_DIR).extend(argv)\n        return args"
        }
      ]
    }
  ]
}