{
  "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",
  "similar_bug_items": [
    {
      "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            )"
        }
      ]
    },
    {
      "pr_number": 926,
      "pr_title": "Don't skip fixtures that are substrings of params",
      "pr_body": "Fix #736\n",
      "issue_id": 736,
      "issue_title": "Parametrized argument named similarly to fixture (contains it) breaks that fixture's parametrization",
      "issue_body": "Originally reported by: **Jacobo Giralt (BitBucket: [jgiralt](http://bitbucket.org/jgiralt), GitHub: [jgiralt](http://github.com/jgiralt))**\n\n---\n\n```\n#!python\n\nimport pytest\n\n@pytest.fixture(params=[1,2,3])\ndef foo(request):\n    return request.param\n\n@pytest.mark.parametrize('foobar', [4,5,6])\ndef test_issue(foo, foobar):\n    assert True\n```\n\n```\n#!python\n\nrequest = <SubRequest 'foo' for <Function 'test_issue[4]'>>\n\n    @pytest.fixture(params=[1,2,3])\n    def foo(request):\n>       return request.param\nAttributeError: SubRequest instance has no attribute 'param'\n```\n\nThe issue seems reproducible as long as the parametrized test argument contains the fixture name as a substring, e.g. foobar, barfoo, foox, xfoo, disappears as soon as I modify the fixture name to anything else.\n\nThis same code works in pytest-2.6.4 and fails for 2.7.\n\n---\n- Bitbucket: https://bitbucket.org/pytest-dev/pytest/issue/736\n",
      "issue_closed_at": "2015-08-11T00:12:52Z",
      "base_commit": "681e502c12f5baa3f2da9c9de4a848813a872945",
      "changes": [
        {
          "file": "_pytest/python.py",
          "type": "function",
          "name": "pytest_generate_tests",
          "class_name": "FixtureManager",
          "code": "def pytest_generate_tests(self, metafunc):\n        for argname in metafunc.fixturenames:\n            faclist = metafunc._arg2fixturedefs.get(argname)\n            if faclist:\n                fixturedef = faclist[-1]\n                if fixturedef.params is not None:\n                    func_params = getattr(getattr(metafunc.function, 'parametrize', None), 'args', [[None]])\n                    # skip directly parametrized arguments\n                    if argname not in func_params and argname not in func_params[0]:\n                        metafunc.parametrize(argname, fixturedef.params,\n                                             indirect=True, scope=fixturedef.scope,\n                                             ids=fixturedef.ids)\n            else:\n                continue"
        }
      ]
    },
    {
      "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"
        }
      ]
    },
    {
      "pr_number": 3634,
      "pr_title": "Merge from master",
      "pr_body": "",
      "issue_id": 3605,
      "issue_title": "get_marker doesn't work for parametrized tests using custom markers",
      "issue_body": "When we are using parametrize regarding to [doc](https://docs.pytest.org/en/latest/parametrize.html) and in pytest hook pytest_collection_modifyitems we would like to get marker and it's value we are failing with error:\r\n`ValueError: MarkInfo expects Mark instances, got MarkDecorator(mark=Mark(name='custom_mark', args=('custom mark1',), kwargs={})) (<class '_pytest.mark.structures.MarkDecorator'>)`\r\n\r\n\r\n- \r\n\r\nHow to reproduce:\r\nI attached full [example](https://github.com/pytest-dev/pytest/files/2119572/pytest_marker_issue.tar.gz),, you just need to run_test.sh script.\r\n\r\n- requirement file is part of [attached examle](https://github.com/pytest-dev/pytest/files/2119572/pytest_marker_issue.tar.gz), contains just latest pytest@features branch\r\n- latest - features branch\r\n- Minimal example if possible, attached [pytest_marker_issue.tar.gz](https://github.com/pytest-dev/pytest/files/2119572/pytest_marker_issue.tar.gz)\r\n\r\ntest_file.py:\r\n---------------\r\n```python\r\nimport pytest\r\n\r\nfirst_custom_mark = pytest.mark.custom_marker\r\ncustom_mark = pytest.mark.custom_mark\r\n\r\n@custom_mark(\"custom mark non parametrized\")\r\ndef test_custom_mark_non_parametrized():\r\n    print \"Hey from test\"\r\n\r\n@pytest.mark.parametrize(\r\n    \"obj_type\",\r\n    [\r\n        first_custom_mark(\"first custom mark\")(\"template\"),\r\n        pytest.param( # Think this should be recommended way?\r\n            \"disk\",\r\n            marks=custom_mark('custom mark1')\r\n        ),\r\n        custom_mark(\"custom mark2\")(\"vm\"),  # Tried also this\r\n    ]\r\n)\r\ndef test_custom_mark_parametrized(obj_type):\r\n    print \"obj_type is: %s\" % obj_type\r\n```\r\n\r\nconftest.py\r\n--------------\r\n```python\r\ndef pytest_collection_modifyitems(session, config, items):\r\n    for item in items:\r\n        # here we are failing with:\r\n        # INTERNALERROR> ValueError: MarkInfo expects Mark instances, got\r\n        # MarkDecorator(mark=Mark(name='custom_mark', args=('custom mark1',), kwargs={})) (<class '_pytest.mark.structures.MarkDecorator'>)\r\n        custom_mark = item.get_marker('custom_mark')\r\n        print(\"Custom mark %s\" % custom_mark)\r\n```",
      "issue_closed_at": "2018-06-27T01:14:31Z",
      "base_commit": "e9371a58a0d1562f9d187778e67a7fae3708261f",
      "changes": [
        {
          "file": "src/_pytest/mark/structures.py",
          "type": "function",
          "name": "get_unpacked_marks",
          "class_name": null,
          "code": "def get_unpacked_marks(obj):\n    \"\"\"\n    obtain the unpacked marks that are stored on an object\n    \"\"\"\n    mark_list = getattr(obj, \"pytestmark\", [])\n\n    if not isinstance(mark_list, list):\n        mark_list = [mark_list]\n    return [getattr(mark, \"mark\", mark) for mark in mark_list]"
        },
        {
          "file": "src/_pytest/python.py",
          "type": "line",
          "name": "line 39",
          "code": "    get_default_arg_names,\n)\nfrom _pytest.outcomes import fail\nfrom _pytest.mark.structures import transfer_markers, get_unpacked_marks\n\n\n# relative paths that we use to filter traceback entries from appearing to the user;"
        },
        {
          "file": "src/_pytest/python.py",
          "type": "function",
          "name": "setmulti2",
          "class_name": "CallSpec2",
          "code": "def setmulti2(self, valtypes, argnames, valset, id, marks, scopenum, param_index):\n        for arg, val in zip(argnames, valset):\n            self._checkargnotcontained(arg)\n            valtype_for_arg = valtypes[arg]\n            getattr(self, valtype_for_arg)[arg] = val\n            self.indices[arg] = param_index\n            self._arg2scopenum[arg] = scopenum\n        self._idlist.append(id)\n        self.marks.extend(marks)"
        },
        {
          "file": "src/_pytest/python.py",
          "type": "function",
          "name": "__init__",
          "class_name": "Function",
          "code": "def __init__(\n        self,\n        name,\n        parent,\n        args=None,\n        config=None,\n        callspec=None,\n        callobj=NOTSET,\n        keywords=None,\n        session=None,\n        fixtureinfo=None,\n        originalname=None,\n    ):\n        super(Function, self).__init__(name, parent, config=config, session=session)\n        self._args = args\n        if callobj is not NOTSET:\n            self.obj = callobj\n\n        self.keywords.update(self.obj.__dict__)\n        self.own_markers.extend(get_unpacked_marks(self.obj))\n        if callspec:\n            self.callspec = callspec\n            # this is total hostile and a mess\n            # keywords are broken by design by now\n            # this will be redeemed later\n            for mark in callspec.marks:\n                # feel free to cry, this was broken for years before\n                # and keywords cant fix it per design\n                self.keywords[mark.name] = mark\n            self.own_markers.extend(callspec.marks)\n        if keywords:\n            self.keywords.update(keywords)\n\n        if fixtureinfo is None:\n            fixtureinfo = self.session._fixturemanager.getfixtureinfo(\n                self, self.obj, self.cls, funcargs=not self._isyieldedfunction()\n            )\n        self._fixtureinfo = fixtureinfo\n        self.fixturenames = fixtureinfo.names_closure\n        self._initrequest()\n\n        #: original function name, without any decorations (for example\n        #: parametrization adds a ``\"[...]\"`` suffix to function names).\n        #:\n        #: .. versionadded:: 3.0\n        self.originalname = originalname"
        }
      ]
    },
    {
      "pr_number": 2487,
      "pr_title": "Fix internal error when trying to detect the start of a recursive traceback",
      "pr_body": "Fix #2486\r\n",
      "issue_id": 2486,
      "issue_title": "_truncate_recursive_traceback()'s recursionindex can be None in 3.1.2",
      "issue_body": "```\r\ntest runtests: commands[0] | coverage run --parallel --source tested_library -m pytest --doctest-glob=*.md --junit-xml=report.xml\r\n============================= test session starts ==============================\r\nplatform linux2 -- Python 2.7.5, pytest-3.1.2, py-1.4.34, pluggy-0.4.0 -- workspace/.tox/test/bin/python2\r\ncachedir: .cache\r\nrootdir: workspace, inifile: tox.ini\r\nplugins: mock-1.6.0, pylama-7.3.3\r\ncollecting ... INTERNALERROR> Traceback (most recent call last):\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/main.py\", line 105, in wrap_session\r\nINTERNALERROR>     session.exitstatus = doit(config, session) or 0\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/main.py\", line 140, in _main\r\nINTERNALERROR>     config.hook.pytest_collection(session=session)\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/vendored_packages/pluggy.py\", line 745, in __call__\r\nINTERNALERROR>     return self._hookexec(self, self._nonwrappers + self._wrappers, kwargs)\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/vendored_packages/pluggy.py\", line 339, in _hookexec\r\nINTERNALERROR>     return self._inner_hookexec(hook, methods, kwargs)\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/vendored_packages/pluggy.py\", line 334, in <lambda>\r\nINTERNALERROR>     _MultiCall(methods, kwargs, hook.spec_opts).execute()\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/vendored_packages/pluggy.py\", line 614, in execute\r\nINTERNALERROR>     res = hook_impl.function(*args)\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/main.py\", line 150, in pytest_collection\r\nINTERNALERROR>     return session.perform_collect()\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/main.py\", line 604, in perform_collect\r\nINTERNALERROR>     items = self._perform_collect(args, genitems)\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/main.py\", line 641, in _perform_collect\r\nINTERNALERROR>     self.items.extend(self.genitems(node))\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/main.py\", line 776, in genitems\r\nINTERNALERROR>     rep = collect_one_node(node)\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/runner.py\", line 457, in collect_one_node\r\nINTERNALERROR>     rep = ihook.pytest_make_collect_report(collector=collector)\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/vendored_packages/pluggy.py\", line 745, in __call__\r\nINTERNALERROR>     return self._hookexec(self, self._nonwrappers + self._wrappers, kwargs)\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/vendored_packages/pluggy.py\", line 339, in _hookexec\r\nINTERNALERROR>     return self._inner_hookexec(hook, methods, kwargs)\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/vendored_packages/pluggy.py\", line 334, in <lambda>\r\nINTERNALERROR>     _MultiCall(methods, kwargs, hook.spec_opts).execute()\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/vendored_packages/pluggy.py\", line 613, in execute\r\nINTERNALERROR>     return _wrapped_call(hook_impl.function(*args), self.execute)\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/vendored_packages/pluggy.py\", line 250, in _wrapped_call\r\nINTERNALERROR>     wrap_controller.send(call_outcome)\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/capture.py\", line 118, in pytest_make_collect_report\r\nINTERNALERROR>     rep = outcome.get_result()\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/vendored_packages/pluggy.py\", line 280, in get_result\r\nINTERNALERROR>     _reraise(*ex)  # noqa\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/vendored_packages/pluggy.py\", line 265, in __init__\r\nINTERNALERROR>     self.result = func()\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/vendored_packages/pluggy.py\", line 614, in execute\r\nINTERNALERROR>     res = hook_impl.function(*args)\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/runner.py\", line 342, in pytest_make_collect_report\r\nINTERNALERROR>     errorinfo = collector.repr_failure(call.excinfo)\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/main.py\", line 480, in repr_failure\r\nINTERNALERROR>     return self._repr_failure_py(excinfo, style=\"short\")\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/main.py\", line 457, in _repr_failure_py\r\nINTERNALERROR>     style=style, tbfilter=tbfilter)\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/_code/code.py\", line 429, in getrepr\r\nINTERNALERROR>     return fmt.repr_excinfo(self)\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/_code/code.py\", line 650, in repr_excinfo\r\nINTERNALERROR>     reprtraceback = self.repr_traceback(excinfo)\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/_code/code.py\", line 607, in repr_traceback\r\nINTERNALERROR>     traceback, extraline = self._truncate_recursive_traceback(traceback)\r\nINTERNALERROR>   File \"workspace/test/lib/python2.7/site-packages/_pytest/_code/code.py\", line 644, in _truncate_recursive_traceback\r\nINTERNALERROR>     traceback = traceback[:recursionindex + 1]\r\nINTERNALERROR> TypeError: unsupported operand type(s) for +: 'NoneType' and 'int'\r\n```\r\n",
      "issue_closed_at": "2017-06-22T11:40:33Z",
      "base_commit": "b2d7c26d80aedb8533430d2e70c3d472614c7c0c",
      "changes": [
        {
          "file": "_pytest/_code/code.py",
          "type": "function",
          "name": "_truncate_recursive_traceback",
          "class_name": "FormattedExcinfo",
          "code": "def _truncate_recursive_traceback(self, traceback):\n        \"\"\"\n        Truncate the given recursive traceback trying to find the starting point\n        of the recursion.\n\n        The detection is done by going through each traceback entry and finding the\n        point in which the locals of the frame are equal to the locals of a previous frame (see ``recursionindex()``.\n\n        Handle the situation where the recursion process might raise an exception (for example\n        comparing numpy arrays using equality raises a TypeError), in which case we do our best to\n        warn the user of the error and show a limited traceback.\n        \"\"\"\n        try:\n            recursionindex = traceback.recursionindex()\n        except Exception as e:\n            max_frames = 10\n            extraline = (\n                '!!! Recursion error detected, but an error occurred locating the origin of recursion.\\n'\n                '  The following exception happened when comparing locals in the stack frame:\\n'\n                '    {exc_type}: {exc_msg}\\n'\n                '  Displaying first and last {max_frames} stack frames out of {total}.'\n            ).format(exc_type=type(e).__name__, exc_msg=safe_str(e), max_frames=max_frames, total=len(traceback))\n            traceback = traceback[:max_frames] + traceback[-max_frames:]\n        else:\n            extraline = \"!!! Recursion detected (same locals & position)\"\n            traceback = traceback[:recursionindex + 1]\n            \n        return traceback, extraline"
        }
      ]
    }
  ]
}