parent
0821c5c81d
commit
ab331c906e
|
@ -0,0 +1,2 @@
|
||||||
|
Fix a possible race condition when trying to remove lock files used to control access to folders
|
||||||
|
created by ``tmp_path`` and ``tmpdir``.
|
|
@ -1,4 +1,5 @@
|
||||||
import atexit
|
import atexit
|
||||||
|
import contextlib
|
||||||
import fnmatch
|
import fnmatch
|
||||||
import itertools
|
import itertools
|
||||||
import os
|
import os
|
||||||
|
@ -290,10 +291,14 @@ def ensure_deletable(path: Path, consider_lock_dead_if_created_before: float) ->
|
||||||
return False
|
return False
|
||||||
else:
|
else:
|
||||||
if lock_time < consider_lock_dead_if_created_before:
|
if lock_time < consider_lock_dead_if_created_before:
|
||||||
lock.unlink()
|
# wa want to ignore any errors while trying to remove the lock such as:
|
||||||
return True
|
# - PermissionDenied, like the file permissions have changed since the lock creation
|
||||||
else:
|
# - FileNotFoundError, in case another pytest process got here first.
|
||||||
return False
|
# and any other cause of failure.
|
||||||
|
with contextlib.suppress(OSError):
|
||||||
|
lock.unlink()
|
||||||
|
return True
|
||||||
|
return False
|
||||||
|
|
||||||
|
|
||||||
def try_cleanup(path: Path, consider_lock_dead_if_created_before: float) -> None:
|
def try_cleanup(path: Path, consider_lock_dead_if_created_before: float) -> None:
|
||||||
|
|
|
@ -1,9 +1,11 @@
|
||||||
import os.path
|
import os.path
|
||||||
import sys
|
import sys
|
||||||
|
import unittest.mock
|
||||||
|
|
||||||
import py
|
import py
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
|
from _pytest.pathlib import ensure_deletable
|
||||||
from _pytest.pathlib import fnmatch_ex
|
from _pytest.pathlib import fnmatch_ex
|
||||||
from _pytest.pathlib import get_extended_length_path_str
|
from _pytest.pathlib import get_extended_length_path_str
|
||||||
from _pytest.pathlib import get_lock_path
|
from _pytest.pathlib import get_lock_path
|
||||||
|
@ -113,3 +115,22 @@ def test_get_extended_length_path_str():
|
||||||
assert get_extended_length_path_str(r"\\share\foo") == r"\\?\UNC\share\foo"
|
assert get_extended_length_path_str(r"\\share\foo") == r"\\?\UNC\share\foo"
|
||||||
assert get_extended_length_path_str(r"\\?\UNC\share\foo") == r"\\?\UNC\share\foo"
|
assert get_extended_length_path_str(r"\\?\UNC\share\foo") == r"\\?\UNC\share\foo"
|
||||||
assert get_extended_length_path_str(r"\\?\c:\foo") == r"\\?\c:\foo"
|
assert get_extended_length_path_str(r"\\?\c:\foo") == r"\\?\c:\foo"
|
||||||
|
|
||||||
|
|
||||||
|
def test_suppress_error_removing_lock(tmp_path):
|
||||||
|
"""ensure_deletable should not raise an exception if the lock file cannot be removed (#5456)"""
|
||||||
|
path = tmp_path / "dir"
|
||||||
|
path.mkdir()
|
||||||
|
lock = get_lock_path(path)
|
||||||
|
lock.touch()
|
||||||
|
mtime = lock.stat().st_mtime
|
||||||
|
|
||||||
|
with unittest.mock.patch.object(Path, "unlink", side_effect=OSError):
|
||||||
|
assert not ensure_deletable(
|
||||||
|
path, consider_lock_dead_if_created_before=mtime + 30
|
||||||
|
)
|
||||||
|
assert lock.is_file()
|
||||||
|
|
||||||
|
# check now that we can remove the lock file in normal circumstances
|
||||||
|
assert ensure_deletable(path, consider_lock_dead_if_created_before=mtime + 30)
|
||||||
|
assert not lock.is_file()
|
||||||
|
|
Loading…
Reference in New Issue