{
    "Selected_candidate": {
        "pr_number": 1861,
        "pr_title": "Improve error message when passing non-string ids to pytest.mark.parametrize",
        "pr_body": "Fix #1857\n",
        "issue_id": 1857,
        "issue_title": "Parameterized ids can't be integers (was allowed in 2.9.2)",
        "issue_body": "Parameterized id's used to be able to be integers. Now that crashes verbosely.\n- [ x ] Include a detailed description of the bug \n\nUsed to be able to have numerical ids in paramterizations.\n3.0.0 crashes verbosely if you do that.\n- [ x ] `pip list` of the virtual environment you are using\n  (venv) $ pip list\n  numpy (1.11.1)\n  pip (8.1.2)\n  py (1.4.31)\n  pymongo (3.3.0)\n  pytest (3.0.0)\n  setuptools (20.10.1)\n  unnecessary-math (0.0.1)\n- [ x ] pytest and operating system versions\n  pytest 3.0.0 (tested against 2.9.2 and works there)\n  os: mac something\n- [ x ] Minimal example if possible\n\n```\nimport pytest\n\ntestdata = [( 1, 2), ( 2, 4)]\n\ndef times_2(x):\n    return x * 2\n\n@pytest.mark.parametrize(\"x,expected\", testdata, ids=('a','b'))\ndef test_ids_strings(x,expected):\n    '''works in both 3.0.0 and 2.9.2'''\n    assert times_2(x) == expected\n\n\n@pytest.mark.parametrize(\"x,expected\", testdata, ids=(1,2))\ndef test_ids_numbers(x,expected):\n    '''works in 2.9.2, crashes verbosely in 3.0.0'''\n    assert times_2(x) == expected\n\n```\n\n2.9.2:\n\n```\n(venv_2.9.2) $ py.test -v test_ids2.py \n================================ test session starts =================================\nplatform darwin -- Python 3.5.2, pytest-2.9.2, py-1.4.31, pluggy-0.3.1 -- /Users/okken/projects/book/bopytest/Book/code/pytest/um_project/tests/venv_2.9.2/bin/python3.5\ncachedir: ../.cache\nrootdir: /Users/okken/projects/book/bopytest/Book/code/pytest/um_project, inifile: \ncollected 4 items \n\ntest_ids2.py::test_ids_strings[a] PASSED\ntest_ids2.py::test_ids_strings[b] PASSED\ntest_ids2.py::test_ids_numbers[1] PASSED\ntest_ids2.py::test_ids_numbers[2] PASSED\n\n============================== 4 passed in 0.02 seconds ==============================\n```\n\n3.0.0:\n\n```\n(venv) $ pytest -v test_ids2.py \n============================= test session starts ==============================\nplatform darwin -- Python 3.5.2, pytest-3.0.0, py-1.4.31, pluggy-0.3.1 -- /Users/okken/projects/book/bopytest/Book/venv/bin/python3.5\ncachedir: ../.cache\nrootdir: /Users/okken/projects/book/bopytest/Book/code/pytest/um_project, inifile: \ncollected 0 items / 1 errors \n\n==================================== ERRORS ====================================\n_____________________ ERROR collecting tests/test_ids2.py ______________________\n../../../../venv/lib/python3.5/site-packages/_pytest/runner.py:163: in __init__\n    self.result = func()\n../../../../venv/lib/python3.5/site-packages/_pytest/main.py:460: in _memocollect\n    return self._memoizedcall('_collected', lambda: list(self.collect()))\n../../../../venv/lib/python3.5/site-packages/_pytest/main.py:331: in _memoizedcall\n    res = function()\n../../../../venv/lib/python3.5/site-packages/_pytest/main.py:460: in <lambda>\n    return self._memoizedcall('_collected', lambda: list(self.collect()))\n../../../../venv/lib/python3.5/site-packages/_pytest/python.py:404: in collect\n    return super(Module, self).collect()\n../../../../venv/lib/python3.5/site-packages/_pytest/python.py:318: in collect\n    res = self.makeitem(name, obj)\n../../../../venv/lib/python3.5/site-packages/_pytest/python.py:330: in makeitem\n    collector=self, name=name, obj=obj)\n../../../../venv/lib/python3.5/site-packages/_pytest/vendored_packages/pluggy.py:724: in __call__\n    return self._hookexec(self, self._nonwrappers + self._wrappers, kwargs)\n../../../../venv/lib/python3.5/site-packages/_pytest/vendored_packages/pluggy.py:338: in _hookexec\n    return self._inner_hookexec(hook, methods, kwargs)\n../../../../venv/lib/python3.5/site-packages/_pytest/vendored_packages/pluggy.py:333: in <lambda>\n    _MultiCall(methods, kwargs, hook.spec_opts).execute()\n../../../../venv/lib/python3.5/site-packages/_pytest/vendored_packages/pluggy.py:595: in execute\n    return _wrapped_call(hook_impl.function(*args), self.execute)\n../../../../venv/lib/python3.5/site-packages/_pytest/vendored_packages/pluggy.py:249: in _wrapped_call\n    wrap_controller.send(call_outcome)\n../../../../venv/lib/python3.5/site-packages/_pytest/python.py:191: in pytest_pycollect_makeitem\n    res = list(collector._genfunctions(name, obj))\n../../../../venv/lib/python3.5/site-packages/_pytest/python.py:350: in _genfunctions\n    self.ihook.pytest_generate_tests(metafunc=metafunc)\n../../../../venv/lib/python3.5/site-packages/_pytest/vendored_packages/pluggy.py:724: in __call__\n    return self._hookexec(self, self._nonwrappers + self._wrappers, kwargs)\n../../../../venv/lib/python3.5/site-packages/_pytest/vendored_packages/pluggy.py:338: in _hookexec\n    return self._inner_hookexec(hook, methods, kwargs)\n../../../../venv/lib/python3.5/site-packages/_pytest/vendored_packages/pluggy.py:333: in <lambda>\n    _MultiCall(methods, kwargs, hook.spec_opts).execute()\n../../../../venv/lib/python3.5/site-packages/_pytest/vendored_packages/pluggy.py:596: in execute\n    res = hook_impl.function(*args)\n../../../../venv/lib/python3.5/site-packages/_pytest/python.py:104: in pytest_generate_tests\n    metafunc.parametrize(*marker.args, **marker.kwargs)\n../../../../venv/lib/python3.5/site-packages/_pytest/python.py:846: in parametrize\n    ids = idmaker(argnames, argvalues, idfn, ids, self.config)\n../../../../venv/lib/python3.5/site-packages/_pytest/python.py:933: in idmaker\n    for valindex, valset in enumerate(argvalues)]\n../../../../venv/lib/python3.5/site-packages/_pytest/python.py:933: in <listcomp>\n    for valindex, valset in enumerate(argvalues)]\n../../../../venv/lib/python3.5/site-packages/_pytest/python.py:929: in _idvalset\n    return _escape_strings(ids[idx])\n../../../../venv/lib/python3.5/site-packages/_pytest/compat.py:144: in _escape_strings\n    return val.encode('unicode_escape').decode('ascii')\nE   AttributeError: 'int' object has no attribute 'encode'\n!!!!!!!!!!!!!!!!!!! Interrupted: 1 errors during collection !!!!!!!!!!!!!!!!!!!!\n=========================== 1 error in 0.64 seconds ============================\n\n```\n",
        "issue_closed_at": "2016-08-24T04:43:57Z",
        "base_commit": "ea0febad2873da4df524f78fb688a9b182c55587",
        "changes": [
            {
                "file": "_pytest/python.py",
                "type": "function",
                "name": "parametrize",
                "class_name": "Metafunc",
                "code": "def parametrize(self, argnames, argvalues, indirect=False, ids=None,\n        scope=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: list of string ids, or a callable.\n            If strings, each is corresponding to the argvalues so that they are\n            part of the test id. If None is given as id of specific test, the\n            automatically generated id for that argument will be used.\n            If callable, it should take one argument (a single argvalue) and return\n            a string or return None. If None, the automatically generated id for that\n            argument will be used.\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 scopes\n        from _pytest.mark import extract_argvalue\n\n        unwrapped_argvalues = []\n        newkeywords = []\n        for maybe_marked_args in argvalues:\n            argval, newmarks = extract_argvalue(maybe_marked_args)\n            unwrapped_argvalues.append(argval)\n            newkeywords.append(newmarks)\n        argvalues = unwrapped_argvalues\n\n        if not isinstance(argnames, (tuple, list)):\n            argnames = [x.strip() for x in argnames.split(\",\") if x.strip()]\n            if len(argnames) == 1:\n                argvalues = [(val,) for val in argvalues]\n        if not argvalues:\n            argvalues = [(NOTSET,) * len(argnames)]\n            # we passed a empty list to parameterize, skip that test\n            #\n            fs, lineno = getfslineno(self.function)\n            newmark = pytest.mark.skip(\n                reason=\"got empty parameter set %r, function %s at %s:%d\" % (\n                    argnames, self.function.__name__, fs, lineno))\n            newkeywords = [{newmark.markname: newmark}]\n\n        if scope is None:\n            scope = _find_parametrized_scope(argnames, self._arg2fixturedefs, indirect)\n\n        scopenum = scopes.index(scope)\n        valtypes = {}\n        for arg in argnames:\n            if arg not in self.fixturenames:\n                if isinstance(indirect, (tuple, list)):\n                    name = 'fixture' if arg in indirect else 'argument'\n                else:\n                    name = 'fixture' if indirect else 'argument'\n                raise ValueError(\n                    \"%r uses no %s %r\" % (\n                            self.function, name, arg))\n\n        if indirect is True:\n            valtypes = dict.fromkeys(argnames, \"params\")\n        elif indirect is False:\n            valtypes = dict.fromkeys(argnames, \"funcargs\")\n        elif isinstance(indirect, (tuple, list)):\n            valtypes = dict.fromkeys(argnames, \"funcargs\")\n            for arg in indirect:\n                if arg not in argnames:\n                    raise ValueError(\"indirect given to %r: fixture %r doesn't exist\" % (\n                                     self.function, arg))\n                valtypes[arg] = \"params\"\n        idfn = None\n        if callable(ids):\n            idfn = ids\n            ids = None\n        if ids and len(ids) != len(argvalues):\n            raise ValueError('%d tests specified with %d ids' %(\n                             len(argvalues), len(ids)))\n        ids = idmaker(argnames, argvalues, idfn, ids, self.config)\n        newcalls = []\n        for callspec in self._calls or [CallSpec2(self)]:\n            elements = zip(ids, argvalues, newkeywords, count())\n            for a_id, valset, keywords, param_index in elements:\n                assert len(valset) == len(argnames)\n                newcallspec = callspec.copy(self)\n                newcallspec.setmulti(valtypes, argnames, valset, a_id,\n                                     keywords, scopenum, param_index)\n                newcalls.append(newcallspec)\n        self._calls = newcalls"
            },
            {
                "file": "_pytest/python.py",
                "type": "function",
                "name": "parametrize",
                "class_name": "Metafunc",
                "code": "def parametrize(self, argnames, argvalues, indirect=False, ids=None,\n        scope=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: list of string ids, or a callable.\n            If strings, each is corresponding to the argvalues so that they are\n            part of the test id. If None is given as id of specific test, the\n            automatically generated id for that argument will be used.\n            If callable, it should take one argument (a single argvalue) and return\n            a string or return None. If None, the automatically generated id for that\n            argument will be used.\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 scopes\n        from _pytest.mark import extract_argvalue\n\n        unwrapped_argvalues = []\n        newkeywords = []\n        for maybe_marked_args in argvalues:\n            argval, newmarks = extract_argvalue(maybe_marked_args)\n            unwrapped_argvalues.append(argval)\n            newkeywords.append(newmarks)\n        argvalues = unwrapped_argvalues\n\n        if not isinstance(argnames, (tuple, list)):\n            argnames = [x.strip() for x in argnames.split(\",\") if x.strip()]\n            if len(argnames) == 1:\n                argvalues = [(val,) for val in argvalues]\n        if not argvalues:\n            argvalues = [(NOTSET,) * len(argnames)]\n            # we passed a empty list to parameterize, skip that test\n            #\n            fs, lineno = getfslineno(self.function)\n            newmark = pytest.mark.skip(\n                reason=\"got empty parameter set %r, function %s at %s:%d\" % (\n                    argnames, self.function.__name__, fs, lineno))\n            newkeywords = [{newmark.markname: newmark}]\n\n        if scope is None:\n            scope = _find_parametrized_scope(argnames, self._arg2fixturedefs, indirect)\n\n        scopenum = scopes.index(scope)\n        valtypes = {}\n        for arg in argnames:\n            if arg not in self.fixturenames:\n                if isinstance(indirect, (tuple, list)):\n                    name = 'fixture' if arg in indirect else 'argument'\n                else:\n                    name = 'fixture' if indirect else 'argument'\n                raise ValueError(\n                    \"%r uses no %s %r\" % (\n                            self.function, name, arg))\n\n        if indirect is True:\n            valtypes = dict.fromkeys(argnames, \"params\")\n        elif indirect is False:\n            valtypes = dict.fromkeys(argnames, \"funcargs\")\n        elif isinstance(indirect, (tuple, list)):\n            valtypes = dict.fromkeys(argnames, \"funcargs\")\n            for arg in indirect:\n                if arg not in argnames:\n                    raise ValueError(\"indirect given to %r: fixture %r doesn't exist\" % (\n                                     self.function, arg))\n                valtypes[arg] = \"params\"\n        idfn = None\n        if callable(ids):\n            idfn = ids\n            ids = None\n        if ids and len(ids) != len(argvalues):\n            raise ValueError('%d tests specified with %d ids' %(\n                             len(argvalues), len(ids)))\n        ids = idmaker(argnames, argvalues, idfn, ids, self.config)\n        newcalls = []\n        for callspec in self._calls or [CallSpec2(self)]:\n            elements = zip(ids, argvalues, newkeywords, count())\n            for a_id, valset, keywords, param_index in elements:\n                assert len(valset) == len(argnames)\n                newcallspec = callspec.copy(self)\n                newcallspec.setmulti(valtypes, argnames, valset, a_id,\n                                     keywords, scopenum, param_index)\n                newcalls.append(newcallspec)\n        self._calls = newcalls"
            }
        ]
    },
    "Justification": "Candidate B directly relates to pytest's handling of parameters, which is a structural component of how tests are defined. It involves an error behavior in the pytest workflow which may provide insights into how dynamic xfail markers interact with fixed parameters. The fact that it affects the test collection process can also reflect on how pytest is evolving between versions, contributing to better understanding the dynamic handling issues presented in the CURRENT bug report."
}