Skip to content

Commit dfb8cc7

Browse files
jahrulespre-commit-ci[bot]gaborbernat
authored
Re-raise error if cannot create file for some reason for soft locks
* updated nomenclature to allign with softlock, added test, added changelog * moved fcntl import to the test def and added type hints * another type attempt * flake fixes * flake fixes * changes to resolve issue 147 * fix typo * flake 8 fix * test for OSError in SoftFileLock * flake8 fix * [pre-commit.ci] auto fixes from pre-commit.com hooks for more information, see https://pre-commit.ci * PR Feedback Signed-off-by: Bernát Gábor <[email protected]> * PR Feedback Signed-off-by: Bernát Gábor <[email protected]> --------- Signed-off-by: Bernát Gábor <[email protected]> Co-authored-by: pre-commit-ci[bot] <66853113+pre-commit-ci[bot]@users.noreply.github.com> Co-authored-by: Bernát Gábor <[email protected]>
1 parent 43472a5 commit dfb8cc7

File tree

4 files changed

+28
-22
lines changed

4 files changed

+28
-22
lines changed

docs/changelog.rst

+4
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,9 @@
11
Changelog
22
=========
3+
v3.10.6 (2023-03-25)
4+
--------------------
5+
- Enhance the robustness of the try/catch block in _soft.py. by :user:`jahrules`.
6+
37
v3.10.5 (2023-03-25)
48
--------------------
59
- Add explicit error check as certain UNIX filesystems do not support flock. by :user:`jahrules`.

src/filelock/_soft.py

+8-10
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@
22

33
import os
44
import sys
5-
from errno import EACCES, EEXIST, ENOENT
5+
from errno import EACCES, EEXIST
66

77
from ._api import BaseFileLock
88
from ._util import raise_on_exist_ro_file
@@ -21,17 +21,15 @@ def _acquire(self) -> None:
2121
| os.O_TRUNC # truncate the file to zero byte
2222
)
2323
try:
24-
fd = os.open(self._lock_file, flags, self._mode)
25-
except OSError as exception:
26-
if exception.errno == EEXIST: # expected if cannot lock
27-
pass
28-
elif exception.errno == ENOENT: # No such file or directory - parent directory is missing
24+
file_handler = os.open(self._lock_file, flags, self._mode)
25+
except OSError as exception: # re-raise unless expected exception
26+
if not (
27+
exception.errno == EEXIST # lock already exist
28+
or (exception.errno == EACCES and sys.platform == "win32") # has no access to this lock
29+
): # pragma: win32 no cover
2930
raise
30-
elif exception.errno == EACCES and sys.platform != "win32": # pragma: win32 no cover
31-
# Permission denied - parent dir is R/O
32-
raise # note windows does not allow you to make a folder r/o only files
3331
else:
34-
self._lock_file_fd = fd
32+
self._lock_file_fd = file_handler
3533

3634
def _release(self) -> None:
3735
os.close(self._lock_file_fd) # type: ignore # the lock file is definitely not None

tests/test_filelock.py

+16-11
Original file line numberDiff line numberDiff line change
@@ -185,7 +185,7 @@ def __init__(self, target: Callable[[], None], name: str) -> None:
185185
def run(self) -> None:
186186
try:
187187
super().run()
188-
except Exception: # pragma: no cover
188+
except RuntimeError: # pragma: no cover
189189
self.ex = sys.exc_info() # pragma: no cover
190190

191191
def join(self, timeout: float | None = None) -> None:
@@ -337,8 +337,8 @@ def test_context_release_on_exc(lock_type: type[BaseFileLock], tmp_path: Path) -
337337
with lock as lock_1:
338338
assert lock is lock_1
339339
assert lock.is_locked
340-
raise Exception
341-
except Exception:
340+
raise ValueError
341+
except ValueError:
342342
assert not lock.is_locked
343343

344344

@@ -352,8 +352,8 @@ def test_acquire_release_on_exc(lock_type: type[BaseFileLock], tmp_path: Path) -
352352
with lock.acquire() as lock_1:
353353
assert lock is lock_1
354354
assert lock.is_locked
355-
raise Exception
356-
except Exception:
355+
raise ValueError
356+
except ValueError:
357357
assert not lock.is_locked
358358

359359

@@ -385,9 +385,8 @@ def test_del(lock_type: type[BaseFileLock], tmp_path: Path) -> None:
385385
def test_cleanup_soft_lock(tmp_path: Path) -> None:
386386
# tests if the lock file is removed after use
387387
lock_path = tmp_path / "a"
388-
lock = SoftFileLock(str(lock_path))
389388

390-
with lock:
389+
with SoftFileLock(lock_path):
391390
assert lock_path.exists()
392391
assert not lock_path.exists()
393392

@@ -399,9 +398,9 @@ def test_poll_intervall_deprecated(lock_type: type[BaseFileLock], tmp_path: Path
399398

400399
with pytest.deprecated_call(match="use poll_interval instead of poll_intervall") as checker:
401400
lock.acquire(poll_intervall=0.05) # the deprecation warning will be captured by the checker
402-
frameinfo = getframeinfo(stack()[0][0]) # get frameinfo of current file and lineno (+1 than the above lineno)
401+
frame_info = getframeinfo(stack()[0][0]) # get frame info of current file and lineno (+1 than the above lineno)
403402
for warning in checker:
404-
if warning.filename == frameinfo.filename and warning.lineno + 1 == frameinfo.lineno: # pragma: no cover
403+
if warning.filename == frame_info.filename and warning.lineno + 1 == frame_info.lineno: # pragma: no cover
405404
break
406405
else: # pragma: no cover
407406
pytest.fail("No warnings of stacklevel=2 matching.")
@@ -490,7 +489,7 @@ def test_wrong_platform(tmp_path: Path) -> None:
490489
assert inspect.isabstract(BaseFileLock)
491490

492491
lock_type = UnixFileLock if sys.platform == "win32" else WindowsFileLock
493-
lock = lock_type(str(tmp_path / "lockfile"))
492+
lock = lock_type(tmp_path / "lockfile")
494493

495494
with pytest.raises(NotImplementedError):
496495
lock.acquire()
@@ -502,5 +501,11 @@ def test_wrong_platform(tmp_path: Path) -> None:
502501
def test_flock_not_implemented_unix(tmp_path: Path, mocker: MockerFixture) -> None:
503502
mocker.patch("fcntl.flock", side_effect=OSError(ENOSYS, "mock error"))
504503
with pytest.raises(NotImplementedError):
505-
with FileLock(str(tmp_path / "a.lock")):
504+
with FileLock(tmp_path / "a.lock"):
506505
pass
506+
507+
508+
def test_soft_errors(tmp_path: Path, mocker: MockerFixture) -> None:
509+
mocker.patch("os.open", side_effect=OSError(ENOSYS, "mock error"))
510+
with pytest.raises(OSError, match="mock error"):
511+
SoftFileLock(tmp_path / "a.lock").acquire()

whitelist.txt

-1
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,6 @@ enosys
77
extlinks
88
filelock
99
filemode
10-
frameinfo
1110
fspath
1211
getframeinfo
1312
intersphinx

0 commit comments

Comments
 (0)