{
  "Selected_candidate": {
    "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.\")"
      }
    ]
  },
  "Justification": "Candidate D's bug report addresses caching behavior in pytest, which is directly relevant to the CURRENT bug concerning incorrect caching of the skipif/xfail evaluation. Both bugs involve how pytest processes caching and condition evaluation during test execution, and the suggested fix for Candidate D modifying how last-failed tests are handled could provide valuable insights into fixing the CURRENT bug. The shared focus on pytest's execution model makes D the most useful report to consult for debugging the CURRENT issue.",
  "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"
}