{
  "Selected_candidate": {
    "pr_number": 4076,
    "pr_title": "Improve error message when TestCase functions use a parametrized fixture",
    "pr_body": "Fix #2535\r\n",
    "issue_id": 2535,
    "issue_title": "Better error message needed for parametrized fixtures with unittest",
    "issue_body": "- [x] Include a detailed description of the bug or suggestion\r\n\r\nunittest style tests are allowed to use pytest fixtures through `@pytest.mark.usefixtures()`.\r\nHowever, if the fixture is paramterized, it crashes and burns and gives a bizarre message of:\r\n`E               Failed: The requested fixture has no parameter defined for the current test.`\r\n\r\nThat wouldn't be so terribly bad, if it wasn't for the other 96 lines of error message that come with it.\r\n\r\n- [x] `pip list` of the virtual environment you are using\r\npip (9.0.1)\r\npy (1.4.34)\r\npytest (3.1.2)\r\nsetuptools (36.0.1)\r\nwheel (0.29.0)\r\n\r\n- [x] pytest and operating system versions\r\nplatform linux -- Python 3.6.1, pytest-3.1.2, py-1.4.34, pluggy-0.4.0\r\n\r\npytest 3.1.2, on Ubuntu Linux running on Windows 10.\r\n\r\n- [x] Minimal example if possible\r\n\r\ntest_one() will pass, test_two() will not only Error, it spews lots of rubbish to confuse the user.\r\n\r\n```\r\nimport pytest\r\nimport unittest\r\n\r\n@pytest.fixture()\r\ndef one():\r\n  return 42\r\n\r\n@pytest.mark.usefixtures('one')\r\nclass TestSomething(unittest.TestCase):\r\n  def test_one(self):\r\n    pass\r\n\r\n@pytest.fixture(params=[1,2])\r\ndef two(request):\r\n  return request.param\r\n\r\n@pytest.mark.usefixtures('two')\r\nclass TestSomethingElse(unittest.TestCase):\r\n  def test_two(self):\r\n    pass\r\n```\r\n\r\noutput\r\n```\r\n(unit_venv) okken@RSA22473:~/projects/unit/fail$ pytest test_something.py\r\n================================== test session starts ==================================\r\nplatform linux -- Python 3.6.1, pytest-3.1.2, py-1.4.34, pluggy-0.4.0\r\nrootdir: /home/okken/projects/unit/fail, inifile: pytest.ini\r\ncollected 2 items\r\n\r\ntest_something.py .E\r\n\r\n======================================== ERRORS =========================================\r\n_____________________ ERROR at setup of TestSomethingElse.test_two ______________________\r\n\r\nself = <FixtureRequest for <TestCaseFunction 'test_two'>>\r\nfixturedef = <FixtureDef name='two' scope='function' baseid='test_something.py' >\r\n\r\n    def _getfixturevalue(self, fixturedef):\r\n        # prepare a subrequest object before calling fixture function\r\n        # (latter managed by fixturedef)\r\n        argname = fixturedef.argname\r\n        funcitem = self._pyfuncitem\r\n        scope = fixturedef.scope\r\n        try:\r\n>           param = funcitem.callspec.getparam(argname)\r\nE           AttributeError: 'TestCaseFunction' object has no attribute 'callspec'\r\n\r\n../unit_venv/lib/python3.6/site-packages/_pytest/fixtures.py:472: AttributeError\r\n\r\nDuring handling of the above exception, another exception occurred:\r\n\r\nself = <CallInfo when='setup' exception: The requested fixture has no parameter defined for the current test.\r\n\r\nRequested fixt...mething.py:14\r\n\r\nRequested here:\r\n/home/okken/projects/unit/unit_venv/lib/python3.6/site-packages/_pytest/fixtures.py:382>\r\nfunc = <function call_runtest_hook.<locals>.<lambda> at 0x7f7689467f28>, when = 'setup'\r\n\r\n    def __init__(self, func, when):\r\n        #: context of invocation: one of \"setup\", \"call\",\r\n        #: \"teardown\", \"memocollect\"\r\n        self.when = when\r\n        self.start = time()\r\n        try:\r\n>           self.result = func()\r\n\r\n../unit_venv/lib/python3.6/site-packages/_pytest/runner.py:157:\r\n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _\r\n../unit_venv/lib/python3.6/site-packages/_pytest/runner.py:145: in <lambda>\r\n    return CallInfo(lambda: ihook(item=item, **kwds), when=when)\r\n../unit_venv/lib/python3.6/site-packages/_pytest/vendored_packages/pluggy.py:745: in __call__\r\n    return self._hookexec(self, self._nonwrappers + self._wrappers, kwargs)\r\n../unit_venv/lib/python3.6/site-packages/_pytest/vendored_packages/pluggy.py:339: in _hookexec\r\n    return self._inner_hookexec(hook, methods, kwargs)\r\n../unit_venv/lib/python3.6/site-packages/_pytest/vendored_packages/pluggy.py:334: in <lambda>\r\n    _MultiCall(methods, kwargs, hook.spec_opts).execute()\r\n../unit_venv/lib/python3.6/site-packages/_pytest/vendored_packages/pluggy.py:613: in execute\r\n    return _wrapped_call(hook_impl.function(*args), self.execute)\r\n../unit_venv/lib/python3.6/site-packages/_pytest/vendored_packages/pluggy.py:254: in _wrapped_call\r\n    return call_outcome.get_result()\r\n../unit_venv/lib/python3.6/site-packages/_pytest/vendored_packages/pluggy.py:279: in get_result\r\n    raise ex[1].with_traceback(ex[2])\r\n../unit_venv/lib/python3.6/site-packages/_pytest/vendored_packages/pluggy.py:265: in __init__\r\n    self.result = func()\r\n../unit_venv/lib/python3.6/site-packages/_pytest/vendored_packages/pluggy.py:614: in execute\r\n    res = hook_impl.function(*args)\r\n../unit_venv/lib/python3.6/site-packages/_pytest/runner.py:94: in pytest_runtest_setup\r\n    item.session._setupstate.prepare(item)\r\n../unit_venv/lib/python3.6/site-packages/_pytest/runner.py:449: in prepare\r\n    col.setup()\r\n../unit_venv/lib/python3.6/site-packages/_pytest/unittest.py:79: in setup\r\n    self._request._fillfixtures()\r\n../unit_venv/lib/python3.6/site-packages/_pytest/fixtures.py:382: in _fillfixtures\r\n    item.funcargs[argname] = self.getfixturevalue(argname)\r\n../unit_venv/lib/python3.6/site-packages/_pytest/fixtures.py:424: in getfixturevalue\r\n    return self._get_active_fixturedef(argname).cached_result[0]\r\n../unit_venv/lib/python3.6/site-packages/_pytest/fixtures.py:449: in _get_active_fixturedef\r\n    result = self._getfixturevalue(fixturedef)\r\n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _\r\n\r\nself = <FixtureRequest for <TestCaseFunction 'test_two'>>\r\nfixturedef = <FixtureDef name='two' scope='function' baseid='test_something.py' >\r\n\r\n    def _getfixturevalue(self, fixturedef):\r\n        # prepare a subrequest object before calling fixture function\r\n        # (latter managed by fixturedef)\r\n        argname = fixturedef.argname\r\n        funcitem = self._pyfuncitem\r\n        scope = fixturedef.scope\r\n        try:\r\n            param = funcitem.callspec.getparam(argname)\r\n        except (AttributeError, ValueError):\r\n            param = NOTSET\r\n            param_index = 0\r\n            if fixturedef.params is not None:\r\n                frame = inspect.stack()[3]\r\n                frameinfo = inspect.getframeinfo(frame[0])\r\n                source_path = frameinfo.filename\r\n                source_lineno = frameinfo.lineno\r\n                source_path = py.path.local(source_path)\r\n                if source_path.relto(funcitem.config.rootdir):\r\n                    source_path = source_path.relto(funcitem.config.rootdir)\r\n                msg = (\r\n                    \"The requested fixture has no parameter defined for the \"\r\n                    \"current test.\\n\\nRequested fixture '{0}' defined in:\\n{1}\"\r\n                    \"\\n\\nRequested here:\\n{2}:{3}\".format(\r\n                        fixturedef.argname,\r\n                        getlocation(fixturedef.func, funcitem.config.rootdir),\r\n                        source_path,\r\n                        source_lineno,\r\n                    )\r\n                )\r\n>               fail(msg)\r\nE               Failed: The requested fixture has no parameter defined for the current test.\r\nE\r\nE               Requested fixture 'two' defined in:\r\nE               test_something.py:14\r\nE\r\nE               Requested here:\r\nE               /home/okken/projects/unit/unit_venv/lib/python3.6/site-packages/_pytest/fixtures.py:382\r\n\r\n../unit_venv/lib/python3.6/site-packages/_pytest/fixtures.py:494: Failed\r\n=========================== 1 passed, 1 error in 0.58 seconds ===========================\r\n```\r\n\r\n",
    "issue_closed_at": "2018-10-06T05:43:58Z",
    "base_commit": "e712adc22651d9c9be5adc1a3337353e0dd1b225",
    "changes": [
      {
        "file": "src/_pytest/fixtures.py",
        "type": "function",
        "name": "_compute_fixture_value",
        "class_name": "FixtureRequest",
        "code": "def _compute_fixture_value(self, fixturedef):\n        \"\"\"\n        Creates a SubRequest based on \"self\" and calls the execute method of the given fixturedef object. This will\n        force the FixtureDef object to throw away any previous results and compute a new fixture value, which\n        will be stored into the FixtureDef object itself.\n\n        :param FixtureDef fixturedef:\n        \"\"\"\n        # prepare a subrequest object before calling fixture function\n        # (latter managed by fixturedef)\n        argname = fixturedef.argname\n        funcitem = self._pyfuncitem\n        scope = fixturedef.scope\n        try:\n            param = funcitem.callspec.getparam(argname)\n        except (AttributeError, ValueError):\n            param = NOTSET\n            param_index = 0\n            if fixturedef.params is not None:\n                frame = inspect.stack()[3]\n                frameinfo = inspect.getframeinfo(frame[0])\n                source_path = frameinfo.filename\n                source_lineno = frameinfo.lineno\n                source_path = py.path.local(source_path)\n                if source_path.relto(funcitem.config.rootdir):\n                    source_path = source_path.relto(funcitem.config.rootdir)\n                msg = (\n                    \"The requested fixture has no parameter defined for the \"\n                    \"current test.\\n\\nRequested fixture '{}' defined in:\\n{}\"\n                    \"\\n\\nRequested here:\\n{}:{}\".format(\n                        fixturedef.argname,\n                        getlocation(fixturedef.func, funcitem.config.rootdir),\n                        source_path,\n                        source_lineno,\n                    )\n                )\n                fail(msg)\n        else:\n            # indices might not be set if old-style metafunc.addcall() was used\n            param_index = funcitem.callspec.indices.get(argname, 0)\n            # if a parametrize invocation set a scope it will override\n            # the static scope defined with the fixture function\n            paramscopenum = funcitem.callspec._arg2scopenum.get(argname)\n            if paramscopenum is not None:\n                scope = scopes[paramscopenum]\n\n        subrequest = SubRequest(self, scope, param, param_index, fixturedef)\n\n        # check if a higher-level scoped fixture accesses a lower level one\n        subrequest._check_scope(argname, self.scope, scope)\n\n        # clear sys.exc_info before invoking the fixture (python bug?)\n        # if its not explicitly cleared it will leak into the call\n        exc_clear()\n        try:\n            # call the fixture function\n            fixturedef.execute(request=subrequest)\n        finally:\n            # if fixture function failed it might have registered finalizers\n            self.session._setupstate.addfinalizer(\n                functools.partial(fixturedef.finish, request=subrequest),\n                subrequest.node,\n            )"
      },
      {
        "file": "src/_pytest/fixtures.py",
        "type": "function",
        "name": "_compute_fixture_value",
        "class_name": "FixtureRequest",
        "code": "def _compute_fixture_value(self, fixturedef):\n        \"\"\"\n        Creates a SubRequest based on \"self\" and calls the execute method of the given fixturedef object. This will\n        force the FixtureDef object to throw away any previous results and compute a new fixture value, which\n        will be stored into the FixtureDef object itself.\n\n        :param FixtureDef fixturedef:\n        \"\"\"\n        # prepare a subrequest object before calling fixture function\n        # (latter managed by fixturedef)\n        argname = fixturedef.argname\n        funcitem = self._pyfuncitem\n        scope = fixturedef.scope\n        try:\n            param = funcitem.callspec.getparam(argname)\n        except (AttributeError, ValueError):\n            param = NOTSET\n            param_index = 0\n            if fixturedef.params is not None:\n                frame = inspect.stack()[3]\n                frameinfo = inspect.getframeinfo(frame[0])\n                source_path = frameinfo.filename\n                source_lineno = frameinfo.lineno\n                source_path = py.path.local(source_path)\n                if source_path.relto(funcitem.config.rootdir):\n                    source_path = source_path.relto(funcitem.config.rootdir)\n                msg = (\n                    \"The requested fixture has no parameter defined for the \"\n                    \"current test.\\n\\nRequested fixture '{}' defined in:\\n{}\"\n                    \"\\n\\nRequested here:\\n{}:{}\".format(\n                        fixturedef.argname,\n                        getlocation(fixturedef.func, funcitem.config.rootdir),\n                        source_path,\n                        source_lineno,\n                    )\n                )\n                fail(msg)\n        else:\n            # indices might not be set if old-style metafunc.addcall() was used\n            param_index = funcitem.callspec.indices.get(argname, 0)\n            # if a parametrize invocation set a scope it will override\n            # the static scope defined with the fixture function\n            paramscopenum = funcitem.callspec._arg2scopenum.get(argname)\n            if paramscopenum is not None:\n                scope = scopes[paramscopenum]\n\n        subrequest = SubRequest(self, scope, param, param_index, fixturedef)\n\n        # check if a higher-level scoped fixture accesses a lower level one\n        subrequest._check_scope(argname, self.scope, scope)\n\n        # clear sys.exc_info before invoking the fixture (python bug?)\n        # if its not explicitly cleared it will leak into the call\n        exc_clear()\n        try:\n            # call the fixture function\n            fixturedef.execute(request=subrequest)\n        finally:\n            # if fixture function failed it might have registered finalizers\n            self.session._setupstate.addfinalizer(\n                functools.partial(fixturedef.finish, request=subrequest),\n                subrequest.node,\n            )"
      }
    ]
  },
  "Justification": "Candidate A is the most relevant report as it directly addresses issues of fixture usage in pytest, which is crucial for the CURRENT bug that seeks to improve how fixture scopes are displayed to the user. Both reports deal with the understanding and usability of fixture outputs, making it a strong candidate for assisting with the CURRENT bug.",
  "instance_id": "pytest-dev__pytest-5221",
  "repo": "pytest-dev/pytest",
  "created_at": "2019-05-06T22:36:44Z",
  "problem_statement": "Display fixture scope with `pytest --fixtures`\nIt would be useful to show fixture scopes with `pytest --fixtures`; currently the only way to learn the scope of a fixture is look at the docs (when that is documented) or at the source code.\n",
  "patch": "diff --git a/src/_pytest/python.py b/src/_pytest/python.py\n--- a/src/_pytest/python.py\n+++ b/src/_pytest/python.py\n@@ -1342,17 +1342,19 @@ def _showfixtures_main(config, session):\n                 currentmodule = module\n         if verbose <= 0 and argname[0] == \"_\":\n             continue\n+        tw.write(argname, green=True)\n+        if fixturedef.scope != \"function\":\n+            tw.write(\" [%s scope]\" % fixturedef.scope, cyan=True)\n         if verbose > 0:\n-            funcargspec = \"%s -- %s\" % (argname, bestrel)\n-        else:\n-            funcargspec = argname\n-        tw.line(funcargspec, green=True)\n+            tw.write(\" -- %s\" % bestrel, yellow=True)\n+        tw.write(\"\\n\")\n         loc = getlocation(fixturedef.func, curdir)\n         doc = fixturedef.func.__doc__ or \"\"\n         if doc:\n             write_docstring(tw, doc)\n         else:\n             tw.line(\"    %s: no docstring available\" % (loc,), red=True)\n+        tw.line()\n \n \n def write_docstring(tw, doc, indent=\"    \"):\n"
}