{
  "instance_id": "pytest-dev__pytest-7373",
  "repo": "pytest-dev/pytest",
  "created_at": "2020-06-15T17:12:08Z",
  "problem_statement": "Incorrect caching of skipif/xfail string condition evaluation\nVersion: pytest 5.4.3, current master\r\n\r\npytest caches the evaluation of the string in e.g. `@pytest.mark.skipif(\"sys.platform == 'win32'\")`. The caching key is only the string itself (see `cached_eval` in `_pytest/mark/evaluate.py`). However, the evaluation also depends on the item's globals, so the caching can lead to incorrect results. Example:\r\n\r\n```py\r\n# test_module_1.py\r\nimport pytest\r\n\r\nskip = True\r\n\r\n@pytest.mark.skipif(\"skip\")\r\ndef test_should_skip():\r\n    assert False\r\n```\r\n\r\n```py\r\n# test_module_2.py\r\nimport pytest\r\n\r\nskip = False\r\n\r\n@pytest.mark.skipif(\"skip\")\r\ndef test_should_not_skip():\r\n    assert False\r\n```\r\n\r\nRunning `pytest test_module_1.py test_module_2.py`.\r\n\r\nExpected: `test_should_skip` is skipped, `test_should_not_skip` is not skipped.\r\n\r\nActual: both are skipped.\r\n\r\n---\r\n\r\nI think the most appropriate fix is to simply remove the caching, which I don't think is necessary really, and inline `cached_eval` into `MarkEvaluator._istrue`.\n",
  "patch": "diff --git a/src/_pytest/mark/evaluate.py b/src/_pytest/mark/evaluate.py\n--- a/src/_pytest/mark/evaluate.py\n+++ b/src/_pytest/mark/evaluate.py\n@@ -10,25 +10,14 @@\n from ..outcomes import fail\n from ..outcomes import TEST_OUTCOME\n from .structures import Mark\n-from _pytest.config import Config\n from _pytest.nodes import Item\n-from _pytest.store import StoreKey\n \n \n-evalcache_key = StoreKey[Dict[str, Any]]()\n+def compiled_eval(expr: str, d: Dict[str, object]) -> Any:\n+    import _pytest._code\n \n-\n-def cached_eval(config: Config, expr: str, d: Dict[str, object]) -> Any:\n-    default = {}  # type: Dict[str, object]\n-    evalcache = config._store.setdefault(evalcache_key, default)\n-    try:\n-        return evalcache[expr]\n-    except KeyError:\n-        import _pytest._code\n-\n-        exprcode = _pytest._code.compile(expr, mode=\"eval\")\n-        evalcache[expr] = x = eval(exprcode, d)\n-        return x\n+    exprcode = _pytest._code.compile(expr, mode=\"eval\")\n+    return eval(exprcode, d)\n \n \n class MarkEvaluator:\n@@ -98,7 +87,7 @@ def _istrue(self) -> bool:\n                     self.expr = expr\n                     if isinstance(expr, str):\n                         d = self._getglobals()\n-                        result = cached_eval(self.item.config, expr, d)\n+                        result = compiled_eval(expr, d)\n                     else:\n                         if \"reason\" not in mark.kwargs:\n                             # XXX better be checked at collection time\n",
  "similar_bug_items": [
    {
      "pr_number": 3893,
      "pr_title": "travis: run tests on macOS",
      "pr_body": "Closes #3892 \r\nCloses #3888\r\n\r\nThanks for submitting a PR, your contribution is really appreciated!\r\n\r\nHere's a quick checklist that should be present in PRs (you can delete this text from the final description, this is\r\njust a guideline):\r\n\r\n- [ ] Create a new changelog file in the `changelog` folder, with a name like `<ISSUE NUMBER>.<TYPE>.rst`. See [changelog/README.rst](/changelog/README.rst) for details.\r\n- [x] Target the `master` branch for bug fixes, documentation updates and trivial changes.\r\n- [ ] Target the `features` branch for new features and removals/deprecations.\r\n- [ ] Include documentation when adding new features.\r\n- [ ] Include new tests or update existing tests when applicable.\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- [ ] ~Add yourself to `AUTHORS` in alphabetical order;~ trivial change - no code\r\n",
      "issue_id": 3888,
      "issue_title": "doctests regression with v3.7.3 on OS-X",
      "issue_body": "- [x] Include a detailed description of the bug or suggestion\r\n\r\nAfter upgrading to v3.7.3, all doctests started failing with the exception:\r\n\r\n```\r\n<my-pyenv>/lib/python3.7/site-packages/_pytest/doctest.py:206: TypeError: cannot unpack non-iterable NoneType object\r\n```\r\n\r\nEverything is OK with v3.7.2.\r\n\r\nSlightly longer traceback (For the record, looking into the stacktrace, I see some references to \"darwin\"-specific functions):\r\n\r\n```\r\n___________________________________________________________________ [doctest] xxx ___________________________________________________________________\r\n\r\nself = <CallInfo when='call' exception: cannot unpack non-iterable NoneType object>, func = <function call_runtest_hook.<locals>.<lambda> at 0x1112f49d8>, when = 'call'\r\ntreat_keyboard_interrupt_as_exception = False\r\n\r\n    def __init__(self, func, when, treat_keyboard_interrupt_as_exception=False):\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<my-pyenv>/lib/python3.7/site-packages/_pytest/runner.py:201:\r\n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _\r\n\r\n>   lambda: ihook(item=item, **kwds),\r\n    when=when,\r\n    treat_keyboard_interrupt_as_exception=item.config.getvalue(\"usepdb\"),\r\n        )\r\n\r\n<my-pyenv>/lib/python3.7/site-packages/_pytest/runner.py:183:\r\n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _\r\n\r\nself = <_HookCaller 'pytest_runtest_call'>, args = (), kwargs = {'item': <DoctestItem 'somemodule.somefunction'>}, notincall = set()\r\n\r\n    def __call__(self, *args, **kwargs):\r\n        if args:\r\n            raise TypeError(\"hook calling supports only keyword arguments\")\r\n        assert not self.is_historic()\r\n        if self.argnames:\r\n            notincall = set(self.argnames) - set(['__multicall__']) - set(\r\n                kwargs.keys())\r\n            if notincall:\r\n                warnings.warn(\r\n                    \"Argument(s) {} which are declared in the hookspec \"\r\n                    \"can not be found in this hook call\"\r\n                    .format(tuple(notincall)),\r\n                    stacklevel=2,\r\n                )\r\n>       return self._hookexec(self, self._nonwrappers + self._wrappers, kwargs)\r\n\r\n<my-pyenv>/lib/python3.7/site-packages/pluggy/hooks.py:258:\r\n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _\r\n\r\nself = <_pytest.config.PytestPluginManager object at 0x1034317b8>, hook = <_HookCaller 'pytest_runtest_call'>\r\nmethods = [<HookImpl plugin_name='runner', plugin=<module '_pytest.runner' from '<my-pyenv......t 0x103d4f4e0>>, <HookImpl plugin_name='logging-plugin', plugin=<_pytest.logging.LoggingPlugin object at 0x10e8f9860>>]\r\nkwargs = {'item': <DoctestItem 'somemodule.somefunction'>}\r\n\r\n    def _hookexec(self, hook, methods, kwargs):\r\n        # called from all hookcaller instances.\r\n        # enable_tracing will set its own wrapping function at self._inner_hookexec\r\n>       return self._inner_hookexec(hook, methods, kwargs)\r\n\r\n<my-pyenv>/lib/python3.7/site-packages/pluggy/manager.py:67:\r\n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _\r\n\r\nhook = <_HookCaller 'pytest_runtest_call'>\r\nmethods = [<HookImpl plugin_name='runner', plugin=<module '_pytest.runner' from '<my-pyenv......t 0x103d4f4e0>>, <HookImpl plugin_name='logging-plugin', plugin=<_pytest.logging.LoggingPlugin object at 0x10e8f9860>>]\r\nkwargs = {'item': <DoctestItem 'somemodule.somefunction'>}\r\n\r\n    self._inner_hookexec = lambda hook, methods, kwargs: \\\r\n        hook.multicall(\r\n            methods, kwargs,\r\n>           firstresult=hook.spec_opts.get('firstresult'),\r\n        )\r\n\r\n<my-pyenv>/lib/python3.7/site-packages/pluggy/manager.py:61:\r\n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _\r\n\r\nhook_impls = [<HookImpl plugin_name='runner', plugin=<module '_pytest.runner' from '<my-pyenv......t 0x103d4f4e0>>, <HookImpl plugin_name='logging-plugin', plugin=<_pytest.logging.LoggingPlugin object at 0x10e8f9860>>]\r\ncaller_kwargs = {'item': <DoctestItem 'somemodule.somefunction'>}, firstresult = False\r\n\r\n    def _multicall(hook_impls, caller_kwargs, firstresult=False):\r\n        \"\"\"Execute a call into multiple python functions/methods and return the\r\n        result(s).\r\n\r\n        ``caller_kwargs`` comes from _HookCaller.__call__().\r\n        \"\"\"\r\n        __tracebackhide__ = True\r\n        results = []\r\n        excinfo = None\r\n        try:  # run impl and wrapper setup functions in a loop\r\n            teardowns = []\r\n            try:\r\n                for hook_impl in reversed(hook_impls):\r\n                    try:\r\n                        args = [caller_kwargs[argname] for argname in hook_impl.argnames]\r\n                    except KeyError:\r\n                        for argname in hook_impl.argnames:\r\n                            if argname not in caller_kwargs:\r\n                                raise HookCallError(\r\n                                    \"hook call must provide argument %r\" % (argname,))\r\n\r\n                    if hook_impl.hookwrapper:\r\n                        try:\r\n                            gen = hook_impl.function(*args)\r\n                            next(gen)   # first yield\r\n                            teardowns.append(gen)\r\n                        except StopIteration:\r\n                            _raise_wrapfail(gen, \"did not yield\")\r\n                    else:\r\n                        res = hook_impl.function(*args)\r\n                        if res is not None:\r\n                            results.append(res)\r\n                            if firstresult:  # halt further impl calls\r\n                                break\r\n            except BaseException:\r\n                excinfo = sys.exc_info()\r\n        finally:\r\n            if firstresult:  # first result hooks return a single value\r\n                outcome = _Result(results[0] if results else None, excinfo)\r\n            else:\r\n                outcome = _Result(results, excinfo)\r\n\r\n            # run all wrapper post-yield blocks\r\n            for gen in reversed(teardowns):\r\n                try:\r\n                    gen.send(outcome)\r\n                    _raise_wrapfail(gen, \"has second yield\")\r\n                except StopIteration:\r\n                    pass\r\n\r\n>           return outcome.get_result()\r\n\r\n<my-pyenv>/lib/python3.7/site-packages/pluggy/callers.py:201:\r\n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _\r\n\r\nself = <pluggy.callers._Result object at 0x11116efd0>\r\n\r\n    def get_result(self):\r\n        \"\"\"Get the result(s) for this hook call.\r\n\r\n            If the hook was marked as a ``firstresult`` only a single value\r\n            will be returned otherwise a list of results.\r\n            \"\"\"\r\n        __tracebackhide__ = True\r\n        if self._excinfo is None:\r\n            return self._result\r\n        else:\r\n            ex = self._excinfo\r\n            if _py3:\r\n>               raise ex[1].with_traceback(ex[2])\r\n\r\n<my-pyenv>/lib/python3.7/site-packages/pluggy/callers.py:76:\r\n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _\r\n\r\nhook_impls = [<HookImpl plugin_name='runner', plugin=<module '_pytest.runner' from '<my-pyenv......t 0x103d4f4e0>>, <HookImpl plugin_name='logging-plugin', plugin=<_pytest.logging.LoggingPlugin object at 0x10e8f9860>>]\r\ncaller_kwargs = {'item': <DoctestItem 'somemodule.somefunction'>}, firstresult = False\r\n\r\n    def _multicall(hook_impls, caller_kwargs, firstresult=False):\r\n        \"\"\"Execute a call into multiple python functions/methods and return the\r\n        result(s).\r\n\r\n        ``caller_kwargs`` comes from _HookCaller.__call__().\r\n        \"\"\"\r\n        __tracebackhide__ = True\r\n        results = []\r\n        excinfo = None\r\n        try:  # run impl and wrapper setup functions in a loop\r\n            teardowns = []\r\n            try:\r\n                for hook_impl in reversed(hook_impls):\r\n                    try:\r\n                        args = [caller_kwargs[argname] for argname in hook_impl.argnames]\r\n                    except KeyError:\r\n                        for argname in hook_impl.argnames:\r\n                            if argname not in caller_kwargs:\r\n                                raise HookCallError(\r\n                                    \"hook call must provide argument %r\" % (argname,))\r\n\r\n                    if hook_impl.hookwrapper:\r\n                        try:\r\n                            gen = hook_impl.function(*args)\r\n                            next(gen)   # first yield\r\n                            teardowns.append(gen)\r\n                        except StopIteration:\r\n                            _raise_wrapfail(gen, \"did not yield\")\r\n                    else:\r\n>                       res = hook_impl.function(*args)\r\n\r\n<my-pyenv>/lib/python3.7/site-packages/pluggy/callers.py:180:\r\n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _\r\n\r\nitem = <DoctestItem 'somemodule.somefunction'>\r\n\r\n    def pytest_runtest_call(item):\r\n        _update_current_test_var(item, \"call\")\r\n        sys.last_type, sys.last_value, sys.last_traceback = (None, None, None)\r\n        try:\r\n>           item.runtest()\r\n\r\n<my-pyenv>/lib/python3.7/site-packages/_pytest/runner.py:111:\r\n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _\r\n\r\nself = <DoctestItem 'somemodule.somefunction'>\r\n\r\n    def runtest(self):\r\n        _check_all_skipped(self.dtest)\r\n>       self._disable_output_capturing_for_darwin()\r\n\r\n<my-pyenv>/lib/python3.7/site-packages/_pytest/doctest.py:192:\r\n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _\r\n\r\nself = <DoctestItem 'somemodule.somefunction'>\r\n\r\n    def _disable_output_capturing_for_darwin(self):\r\n        \"\"\"\r\n            Disable output capturing. Otherwise, stdout is lost to doctest (#985)\r\n            \"\"\"\r\n        if platform.system() != \"Darwin\":\r\n            return\r\n        capman = self.config.pluginmanager.getplugin(\"capturemanager\")\r\n        if capman:\r\n>           out, err = capman.suspend_global_capture(in_=True)\r\nE           TypeError: cannot unpack non-iterable NoneType object\r\n\r\n<my-pyenv>/lib/python3.7/site-packages/_pytest/doctest.py:206: TypeError\r\n```\r\n\r\n- [x] `pip list` of the virtual environment you are using\r\n\r\n```\r\n...\r\nmypy                     0.620\r\npytest                   3.7.3\r\npytest-cov               2.5.1\r\npytest-django            3.4.2\r\ntox                      3.2.1\r\n...\r\n```\r\n\r\n- [x] pytest and operating system versions\r\n\r\n`pytest v3.7.3` and \r\n\r\n```\r\n$ sw_vers\r\nProductName:\tMac OS X\r\nProductVersion:\t10.13.6\r\nBuildVersion:\t17G65\r\n```\r\n\r\n- [ ] Minimal example if possible\r\n",
      "issue_closed_at": "2018-08-28T20:06:18Z",
      "base_commit": "2a059b1c1bff392252c02413c89913dd672af168",
      "changes": [
        {
          "file": "src/_pytest/doctest.py",
          "type": "function",
          "name": "_disable_output_capturing_for_darwin",
          "class_name": "DoctestItem",
          "code": "def _disable_output_capturing_for_darwin(self):\n        \"\"\"\n        Disable output capturing. Otherwise, stdout is lost to doctest (#985)\n        \"\"\"\n        if platform.system() != \"Darwin\":\n            return\n        capman = self.config.pluginmanager.getplugin(\"capturemanager\")\n        if capman:\n            out, err = capman.suspend_global_capture(in_=True)\n            sys.stdout.write(out)\n            sys.stderr.write(err)"
        }
      ]
    },
    {
      "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"
        }
      ]
    },
    {
      "pr_number": 3295,
      "pr_title": "implemented --last-failed-no-failures",
      "pr_body": "Fixes #3139",
      "issue_id": 3139,
      "issue_title": "pytest cache is empty inspite of that \"--lf\" option execute all the cases",
      "issue_body": "Use Case: \r\n1. I have a huge Web automation suite(and yes with flaky test cases).70k test case execution per week.\r\n2. There are some random failures and I want to rerun them and Voila we have a wonderful feature of --lf\r\n3. Now My all test case got passed in third rerun but note I have 4th rerun also with --lf option.\r\n4. 4th Rerun picks all the test case back again. Driving me crazy.\r\n",
      "issue_closed_at": "2018-03-20T22:17:32Z",
      "base_commit": "d6ddeb395bbf788a708c90f6e3003fb57cdc3b7e",
      "changes": [
        {
          "file": "_pytest/cacheprovider.py",
          "type": "function",
          "name": "__init__",
          "class_name": "NFPlugin",
          "code": "def __init__(self, config):\n        self.config = config\n        self.active = config.option.newfirst\n        self.cached_nodeids = config.cache.get(\"cache/nodeids\", [])"
        },
        {
          "file": "_pytest/cacheprovider.py",
          "type": "function",
          "name": "pytest_collectreport",
          "class_name": "LFPlugin",
          "code": "def pytest_collectreport(self, report):\n        passed = report.outcome in ('passed', 'skipped')\n        if passed:\n            if report.nodeid in self.lastfailed:\n                self.lastfailed.pop(report.nodeid)\n                self.lastfailed.update(\n                    (item.nodeid, True)\n                    for item in report.result)\n        else:\n            self.lastfailed[report.nodeid] = True"
        },
        {
          "file": "_pytest/cacheprovider.py",
          "type": "function",
          "name": "pytest_addoption",
          "class_name": null,
          "code": "def pytest_addoption(parser):\n    group = parser.getgroup(\"general\")\n    group.addoption(\n        '--lf', '--last-failed', action='store_true', dest=\"lf\",\n        help=\"rerun only the tests that failed \"\n             \"at the last run (or all if none failed)\")\n    group.addoption(\n        '--ff', '--failed-first', action='store_true', dest=\"failedfirst\",\n        help=\"run all tests but run the last failures first.  \"\n             \"This may re-order tests and thus lead to \"\n             \"repeated fixture setup/teardown\")\n    group.addoption(\n        '--nf', '--new-first', action='store_true', dest=\"newfirst\",\n        help=\"run tests from new files first, then the rest of the tests \"\n             \"sorted by file mtime\")\n    group.addoption(\n        '--cache-show', action='store_true', dest=\"cacheshow\",\n        help=\"show cache contents, don't perform collection or tests\")\n    group.addoption(\n        '--cache-clear', action='store_true', dest=\"cacheclear\",\n        help=\"remove all cache contents at start of test run.\")\n    parser.addini(\n        \"cache_dir\", default='.pytest_cache',\n        help=\"cache directory path.\")"
        }
      ]
    },
    {
      "pr_number": 4164,
      "pr_title": "don`t parse compat properties as fixtures",
      "pr_body": "Fix #2701 \r\n\r\nits tricky to trigger those warnings under normal circumstances as they happen before pytest hooks into the warnings system\r\n\r\n`python -W always -m pytest testing/test_tmpdir.py --collectonly` could be used to trigger them for example on the cli\r\n",
      "issue_id": 2701,
      "issue_title": "warning filters incorrectly apply for conftests and collection in pytester",
      "issue_body": "#2696 uncovered that pytest invocations with selected warning filters dont pass trough to the nested pytest correctly, in turn making tests even with the filter fail at importing a conftest and running a collection\r\n\r\nCC @nicoddemus ",
      "issue_closed_at": "2018-10-31T15:52:53Z",
      "base_commit": "ff04a1fb0999ad82f26db47e4026c8974bd3f99d",
      "changes": [
        {
          "file": "src/_pytest/fixtures.py",
          "type": "function",
          "name": "pytest_plugin_registered",
          "class_name": "FixtureManager",
          "code": "def pytest_plugin_registered(self, plugin):\n        nodeid = None\n        try:\n            p = py.path.local(plugin.__file__).realpath()\n        except AttributeError:\n            pass\n        else:\n            # construct the base nodeid which is later used to check\n            # what fixtures are visible for particular tests (as denoted\n            # by their test id)\n            if p.basename.startswith(\"conftest.py\"):\n                nodeid = p.dirpath().relto(self.config.rootdir)\n                if p.sep != nodes.SEP:\n                    nodeid = nodeid.replace(p.sep, nodes.SEP)\n        self.parsefactories(plugin, nodeid)"
        },
        {
          "file": "src/_pytest/fixtures.py",
          "type": "function",
          "name": "parsefactories",
          "class_name": "FixtureManager",
          "code": "def parsefactories(self, node_or_obj, nodeid=NOTSET, unittest=False):\n        from _pytest import deprecated\n\n        if nodeid is not NOTSET:\n            holderobj = node_or_obj\n        else:\n            holderobj = node_or_obj.obj\n            nodeid = node_or_obj.nodeid\n        if holderobj in self._holderobjseen:\n            return\n        self._holderobjseen.add(holderobj)\n        autousenames = []\n        for name in dir(holderobj):\n            # The attribute can be an arbitrary descriptor, so the attribute\n            # access below can raise. safe_getatt() ignores such exceptions.\n            obj = safe_getattr(holderobj, name, None)\n            marker = getfixturemarker(obj)\n            # fixture functions have a pytest_funcarg__ prefix (pre-2.3 style)\n            # or are \"@pytest.fixture\" marked\n            if marker is None:\n                if not name.startswith(self._argprefix):\n                    continue\n                if not callable(obj):\n                    continue\n                marker = defaultfuncargprefixmarker\n\n                filename, lineno = getfslineno(obj)\n                warnings.warn_explicit(\n                    deprecated.FUNCARG_PREFIX.format(name=name),\n                    category=None,\n                    filename=str(filename),\n                    lineno=lineno + 1,\n                )\n                name = name[len(self._argprefix) :]\n            elif not isinstance(marker, FixtureFunctionMarker):\n                # magic globals  with __getattr__ might have got us a wrong\n                # fixture attribute\n                continue\n            else:\n                if marker.name:\n                    name = marker.name\n                assert not name.startswith(self._argprefix), FIXTURE_MSG.format(name)\n\n            # during fixture definition we wrap the original fixture function\n            # to issue a warning if called directly, so here we unwrap it in order to not emit the warning\n            # when pytest itself calls the fixture function\n            if six.PY2 and unittest:\n                # hack on Python 2 because of the unbound methods\n                obj = get_real_func(obj)\n            else:\n                obj = get_real_method(obj, holderobj)\n\n            fixture_def = FixtureDef(\n                self,\n                nodeid,\n                name,\n                obj,\n                marker.scope,\n                marker.params,\n                unittest=unittest,\n                ids=marker.ids,\n            )\n\n            faclist = self._arg2fixturedefs.setdefault(name, [])\n            if fixture_def.has_location:\n                faclist.append(fixture_def)\n            else:\n                # fixturedefs with no location are at the front\n                # so this inserts the current fixturedef after the\n                # existing fixturedefs from external plugins but\n                # before the fixturedefs provided in conftests.\n                i = len([f for f in faclist if not f.has_location])\n                faclist.insert(i, fixture_def)\n            if marker.autouse:\n                autousenames.append(name)\n\n        if autousenames:\n            self._nodeid_and_autousenames.append((nodeid or \"\", autousenames))"
        }
      ]
    }
  ]
}