Skip to content

Commit 43472a5

Browse files
authored
Fix trying to infinitely acquire a lock if flock is not implemented
1 parent 745da61 commit 43472a5

File tree

7 files changed

+41
-16
lines changed

7 files changed

+41
-16
lines changed

.pre-commit-config.yaml

+4
Original file line numberDiff line numberDiff line change
@@ -44,6 +44,10 @@ repos:
4444
hooks:
4545
- id: tox-ini-fmt
4646
args: ["-p", "fix"]
47+
- repo: https://github.com/tox-dev/pyproject-fmt
48+
rev: "0.9.2"
49+
hooks:
50+
- id: pyproject-fmt
4751
- repo: https://github.com/PyCQA/flake8
4852
rev: 6.0.0
4953
hooks:

docs/changelog.rst

+4
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,9 @@
11
Changelog
22
=========
3+
v3.10.5 (2023-03-25)
4+
--------------------
5+
- Add explicit error check as certain UNIX filesystems do not support flock. by :user:`jahrules`.
6+
37
v3.10.4 (2023-03-24)
48
--------------------
59
- Update os.open to preserve mode= for certain edge cases. by :user:`jahrules`.

pyproject.toml

+17-15
Original file line numberDiff line numberDiff line change
@@ -42,8 +42,10 @@ optional-dependencies.docs = [
4242
optional-dependencies.testing = [
4343
"covdefaults>=2.3",
4444
"coverage>=7.2.2",
45+
"diff-cover>=7.5",
4546
"pytest>=7.2.2",
4647
"pytest-cov>=4",
48+
"pytest-mock>=3.10",
4749
"pytest-timeout>=2.1",
4850
]
4951
urls.Documentation = "https://py-filelock.readthedocs.io"
@@ -56,6 +58,21 @@ build.hooks.vcs.version-file = "src/filelock/version.py"
5658
build.targets.sdist.include = ["/src", "/tests"]
5759
version.source = "vcs"
5860

61+
[tool.black]
62+
line-length = 120
63+
64+
[tool.isort]
65+
profile = "black"
66+
known_first_party = ["filelock"]
67+
add_imports = ["from __future__ import annotations"]
68+
69+
[tool.flake8]
70+
max-complexity = 22
71+
max-line-length = 120
72+
unused-arguments-ignore-abstract-functions = true
73+
noqa-require-code = true
74+
dictionaries = ["en_US", "python", "technical", "django"]
75+
5976
[tool.coverage]
6077
html.show_contexts = true
6178
html.skip_covered = false
@@ -65,14 +82,6 @@ report.fail_under = 76
6582
run.parallel = true
6683
run.plugins = ["covdefaults"]
6784

68-
[tool.black]
69-
line-length = 120
70-
71-
[tool.isort]
72-
profile = "black"
73-
known_first_party = ["filelock"]
74-
add_imports = ["from __future__ import annotations"]
75-
7685
[tool.mypy]
7786
python_version = "3.11"
7887
show_error_codes = true
@@ -81,10 +90,3 @@ overrides = [{ module = ["appdirs.*", "jnius.*"], ignore_missing_imports = true
8190

8291
[tool.pep8]
8392
max-line-length = "120"
84-
85-
[tool.flake8]
86-
max-complexity = 22
87-
max-line-length = 120
88-
unused-arguments-ignore-abstract-functions = true
89-
noqa-require-code = true
90-
dictionaries = ["en_US", "python", "technical", "django"]

src/filelock/_unix.py

+4-1
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@
22

33
import os
44
import sys
5+
from errno import ENOSYS
56
from typing import cast
67

78
from ._api import BaseFileLock
@@ -39,8 +40,10 @@ def _acquire(self) -> None:
3940
pass # This locked is not owned by this UID
4041
try:
4142
fcntl.flock(fd, fcntl.LOCK_EX | fcntl.LOCK_NB)
42-
except OSError:
43+
except OSError as exception:
4344
os.close(fd)
45+
if exception.errno == ENOSYS: # NotImplemented error
46+
raise NotImplementedError("FileSystem does not appear to support flock; user SoftFileLock instead")
4447
else:
4548
self._lock_file_fd = fd
4649

tests/test_filelock.py

+10
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@
66
import sys
77
import threading
88
from contextlib import contextmanager
9+
from errno import ENOSYS
910
from inspect import getframeinfo, stack
1011
from pathlib import Path, PurePath
1112
from stat import S_IWGRP, S_IWOTH, S_IWUSR, filemode
@@ -14,6 +15,7 @@
1415

1516
import pytest
1617
from _pytest.logging import LogCaptureFixture
18+
from pytest_mock import MockerFixture
1719

1820
from filelock import (
1921
BaseFileLock,
@@ -494,3 +496,11 @@ def test_wrong_platform(tmp_path: Path) -> None:
494496
lock.acquire()
495497
with pytest.raises(NotImplementedError):
496498
lock._release()
499+
500+
501+
@pytest.mark.skipif(sys.platform == "win32", reason="flock not run on windows")
502+
def test_flock_not_implemented_unix(tmp_path: Path, mocker: MockerFixture) -> None:
503+
mocker.patch("fcntl.flock", side_effect=OSError(ENOSYS, "mock error"))
504+
with pytest.raises(NotImplementedError):
505+
with FileLock(str(tmp_path / "a.lock")):
506+
pass

tox.ini

+1
Original file line numberDiff line numberDiff line change
@@ -29,6 +29,7 @@ commands =
2929
--cov-config=pyproject.toml --no-cov-on-fail --cov-report term-missing:skip-covered --cov-context=test \
3030
--cov-report html:{envtmpdir}{/}htmlcov --cov-report xml:{toxworkdir}{/}coverage.{envname}.xml \
3131
tests}
32+
diff-cover --compare-branch {env:DIFF_AGAINST:origin/main} {toxworkdir}{/}coverage.{envname}.xml
3233
package = wheel
3334
wheel_build_env = .pkg
3435

whitelist.txt

+1
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@ autodoc
33
autosectionlabel
44
caplog
55
eacces
6+
enosys
67
extlinks
78
filelock
89
filemode

0 commit comments

Comments
 (0)