{
  "Selected_candidate": {
    "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))"
      }
    ]
  },
  "Justification": "Candidate E is particularly relevant because it highlights an issue with warning filters relating to test collection, which closely aligns with the current bug involving the creation of temporary directories in pytest. Both reports cover aspects of the pytest testing framework and share implications on how test execution can be affected by user-defined characteristics—whether that be username formats in the current bug or warning filters in E. The potential similarities in root causes and user experience challenges while dealing with pytest make it a valuable resource for debugging the current problem.",
  "instance_id": "pytest-dev__pytest-8365",
  "repo": "pytest-dev/pytest",
  "created_at": "2021-02-22T20:26:35Z",
  "problem_statement": "tmpdir creation fails when the username contains illegal characters for directory names\n`tmpdir`, `tmpdir_factory` and `tmp_path_factory` rely on `getpass.getuser()` for determining the `basetemp` directory. I found that the user name returned by `getpass.getuser()` may return characters that are not allowed for directory names. This may lead to errors while creating the temporary directory.\r\n\r\nThe situation in which I reproduced this issue was while being logged in through an ssh connection into my Windows 10 x64 Enterprise version (1909) using an OpenSSH_for_Windows_7.7p1 server. In this configuration the command `python -c \"import getpass; print(getpass.getuser())\"` returns my domain username e.g. `contoso\\john_doe` instead of `john_doe` as when logged in regularly using a local session.\r\n\r\nWhen trying to create a temp directory in pytest through e.g. `tmpdir_factory.mktemp('foobar')` this fails with the following error message:\r\n```\r\nself = WindowsPath('C:/Users/john_doe/AppData/Local/Temp/pytest-of-contoso/john_doe')\r\nmode = 511, parents = False, exist_ok = True\r\n\r\n    def mkdir(self, mode=0o777, parents=False, exist_ok=False):\r\n        \"\"\"\r\n        Create a new directory at this given path.\r\n        \"\"\"\r\n        if self._closed:\r\n            self._raise_closed()\r\n        try:\r\n>           self._accessor.mkdir(self, mode)\r\nE           FileNotFoundError: [WinError 3] The system cannot find the path specified: 'C:\\\\Users\\\\john_doe\\\\AppData\\\\Local\\\\Temp\\\\pytest-of-contoso\\\\john_doe'\r\n\r\nC:\\Python38\\lib\\pathlib.py:1266: FileNotFoundError\r\n```\r\n\r\nI could also reproduce this without the complicated ssh/windows setup with pytest 6.2.2 using the following commands from a `cmd`:\r\n```bat\r\necho def test_tmpdir(tmpdir):>test_tmp.py\r\necho   pass>>test_tmp.py\r\nset LOGNAME=contoso\\john_doe\r\npy.test test_tmp.py\r\n```\r\n\r\nThanks for having a look at this!\n",
  "patch": "diff --git a/src/_pytest/tmpdir.py b/src/_pytest/tmpdir.py\n--- a/src/_pytest/tmpdir.py\n+++ b/src/_pytest/tmpdir.py\n@@ -115,7 +115,12 @@ def getbasetemp(self) -> Path:\n             # use a sub-directory in the temproot to speed-up\n             # make_numbered_dir() call\n             rootdir = temproot.joinpath(f\"pytest-of-{user}\")\n-            rootdir.mkdir(exist_ok=True)\n+            try:\n+                rootdir.mkdir(exist_ok=True)\n+            except OSError:\n+                # getuser() likely returned illegal characters for the platform, use unknown back off mechanism\n+                rootdir = temproot.joinpath(\"pytest-of-unknown\")\n+                rootdir.mkdir(exist_ok=True)\n             basetemp = make_numbered_dir_with_cleanup(\n                 prefix=\"pytest-\", root=rootdir, keep=3, lock_timeout=LOCK_TIMEOUT\n             )\n"
}