Skip to content

Commit d0061e4

Browse files
added multiuser support and associated tests (#192)
Co-authored-by: pre-commit-ci[bot] <66853113+pre-commit-ci[bot]@users.noreply.github.com>
1 parent df6b13c commit d0061e4

File tree

7 files changed

+88
-10
lines changed

7 files changed

+88
-10
lines changed

docs/changelog.rst

+4
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,9 @@
11
Changelog
22
=========
3+
v3.10.0 (2023-03-15)
4+
-------------------
5+
- Add support for explicit file modes for lockfiles :pr:`192 - by :user:`jahrules`.
6+
37
v3.9.1 (2023-03-14)
48
-------------------
59
- Use ``time.perf_counter`` instead of ``time.monotonic`` for calculating timeouts.

src/filelock/_api.py

+15-3
Original file line numberDiff line numberDiff line change
@@ -39,14 +39,20 @@ def __exit__(
3939
class BaseFileLock(ABC, contextlib.ContextDecorator):
4040
"""Abstract base class for a file lock object."""
4141

42-
def __init__(self, lock_file: str | os.PathLike[Any], timeout: float = -1) -> None:
42+
def __init__(
43+
self,
44+
lock_file: str | os.PathLike[Any],
45+
timeout: float = -1,
46+
mode: int = 0o644,
47+
) -> None:
4348
"""
4449
Create a new lock object.
4550
4651
:param lock_file: path to the file
4752
:param timeout: default timeout when acquiring the lock, in seconds. It will be used as fallback value in
4853
the acquire method, if no timeout value (``None``) is given. If you want to disable the timeout, set it
4954
to a negative value. A timeout of 0 means, that there is exactly one attempt to acquire the file lock.
55+
: param mode: file permissions for the lockfile.
5056
"""
5157
# The path to the lock file.
5258
self._lock_file: str = os.fspath(lock_file)
@@ -58,6 +64,9 @@ def __init__(self, lock_file: str | os.PathLike[Any], timeout: float = -1) -> No
5864
# The default timeout value.
5965
self._timeout: float = timeout
6066

67+
# The mode for the lock files
68+
self._mode: int = mode
69+
6170
# We use this lock primarily for the lock counter.
6271
self._thread_lock: Lock = Lock()
6372

@@ -170,8 +179,11 @@ def acquire(
170179
with self._thread_lock:
171180
if not self.is_locked:
172181
_LOGGER.debug("Attempting to acquire lock %s on %s", lock_id, lock_filename)
173-
self._acquire()
174-
182+
previous_umask = os.umask(0)
183+
try:
184+
self._acquire()
185+
finally:
186+
os.umask(previous_umask) # reset umask to initial value
175187
if self.is_locked:
176188
_LOGGER.debug("Lock %s acquired on %s", lock_id, lock_filename)
177189
break

src/filelock/_soft.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -14,14 +14,14 @@ class SoftFileLock(BaseFileLock):
1414
def _acquire(self) -> None:
1515
raise_on_exist_ro_file(self._lock_file)
1616
# first check for exists and read-only mode as the open will mask this case as EEXIST
17-
mode = (
17+
flags = (
1818
os.O_WRONLY # open for writing only
1919
| os.O_CREAT
2020
| os.O_EXCL # together with above raise EEXIST if the file specified by filename exists
2121
| os.O_TRUNC # truncate the file to zero byte
2222
)
2323
try:
24-
fd = os.open(self._lock_file, mode)
24+
fd = os.open(self._lock_file, flags, self._mode)
2525
except OSError as exception:
2626
if exception.errno == EEXIST: # expected if cannot lock
2727
pass

src/filelock/_unix.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -31,8 +31,8 @@ class UnixFileLock(BaseFileLock):
3131
"""Uses the :func:`fcntl.flock` to hard lock the lock file on unix systems."""
3232

3333
def _acquire(self) -> None:
34-
open_mode = os.O_RDWR | os.O_CREAT | os.O_TRUNC
35-
fd = os.open(self._lock_file, open_mode)
34+
open_flags = os.O_RDWR | os.O_CREAT | os.O_TRUNC
35+
fd = os.open(self._lock_file, open_flags, self._mode)
3636
try:
3737
fcntl.flock(fd, fcntl.LOCK_EX | fcntl.LOCK_NB)
3838
except OSError:

src/filelock/_windows.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -16,13 +16,13 @@ class WindowsFileLock(BaseFileLock):
1616

1717
def _acquire(self) -> None:
1818
raise_on_exist_ro_file(self._lock_file)
19-
mode = (
19+
flags = (
2020
os.O_RDWR # open for read and write
2121
| os.O_CREAT # create file if not exists
2222
| os.O_TRUNC # truncate file if not empty
2323
)
2424
try:
25-
fd = os.open(self._lock_file, mode)
25+
fd = os.open(self._lock_file, flags, self._mode)
2626
except OSError as exception:
2727
if exception.errno == ENOENT: # No such file or directory
2828
raise

tests/test_filelock.py

+62-1
Original file line numberDiff line numberDiff line change
@@ -2,12 +2,13 @@
22

33
import inspect
44
import logging
5+
import os
56
import sys
67
import threading
78
from contextlib import contextmanager
89
from inspect import getframeinfo, stack
910
from pathlib import Path, PurePath
10-
from stat import S_IWGRP, S_IWOTH, S_IWUSR
11+
from stat import S_IWGRP, S_IWOTH, S_IWUSR, filemode
1112
from types import TracebackType
1213
from typing import Callable, Iterator, Tuple, Type, Union
1314

@@ -418,6 +419,66 @@ def decorated_method() -> None:
418419
assert not lock.is_locked
419420

420421

422+
def test_lock_mode(tmp_path: Path) -> None:
423+
lock_path = tmp_path / "a.lock"
424+
lock = FileLock(str(lock_path), mode=0o666)
425+
426+
lock.acquire()
427+
assert lock.is_locked
428+
429+
mode = filemode(os.stat(lock_path).st_mode)
430+
assert mode == "-rw-rw-rw-"
431+
432+
lock.release()
433+
434+
435+
def test_lock_mode_soft(tmp_path: Path) -> None:
436+
lock_path = tmp_path / "a.lock"
437+
lock = SoftFileLock(str(lock_path), mode=0o666)
438+
439+
lock.acquire()
440+
assert lock.is_locked
441+
442+
mode = filemode(os.stat(lock_path).st_mode)
443+
assert mode == "-rw-rw-rw-"
444+
445+
lock.release()
446+
447+
448+
def test_umask(tmp_path: Path) -> None:
449+
lock_path = tmp_path / "a.lock"
450+
lock = FileLock(str(lock_path), mode=0o666)
451+
452+
initial_umask = os.umask(0)
453+
os.umask(initial_umask)
454+
455+
lock.acquire()
456+
assert lock.is_locked
457+
458+
current_umask = os.umask(0)
459+
os.umask(current_umask)
460+
assert initial_umask == current_umask
461+
462+
lock.release()
463+
464+
465+
def test_umask_soft(tmp_path: Path) -> None:
466+
lock_path = tmp_path / "a.lock"
467+
lock = SoftFileLock(str(lock_path), mode=0o666)
468+
469+
initial_umask = os.umask(0)
470+
os.umask(initial_umask)
471+
472+
lock.acquire()
473+
assert lock.is_locked
474+
475+
current_umask = os.umask(0)
476+
os.umask(current_umask)
477+
assert initial_umask == current_umask
478+
479+
lock.release()
480+
481+
421482
def test_wrong_platform(tmp_path: Path) -> None:
422483
assert not inspect.isabstract(UnixFileLock)
423484
assert not inspect.isabstract(WindowsFileLock)

whitelist.txt

+1
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@ caplog
55
eacces
66
extlinks
77
filelock
8+
filemode
89
frameinfo
910
fspath
1011
getframeinfo

0 commit comments

Comments
 (0)