2023-01-06 11:47:44 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
|
|
|
import os
|
|
|
|
import sys
|
2023-09-17 19:40:34 +00:00
|
|
|
from contextlib import suppress
|
2023-04-12 13:51:13 +00:00
|
|
|
from errno import ENOSYS
|
2023-01-06 11:47:44 +00:00
|
|
|
from typing import cast
|
|
|
|
|
|
|
|
from ._api import BaseFileLock
|
2023-09-17 19:40:34 +00:00
|
|
|
from ._util import ensure_directory_exists
|
2023-01-06 11:47:44 +00:00
|
|
|
|
|
|
|
#: a flag to indicate if the fcntl API is available
|
|
|
|
has_fcntl = False
|
|
|
|
if sys.platform == "win32": # pragma: win32 cover
|
|
|
|
|
|
|
|
class UnixFileLock(BaseFileLock):
|
|
|
|
"""Uses the :func:`fcntl.flock` to hard lock the lock file on unix systems."""
|
|
|
|
|
|
|
|
def _acquire(self) -> None:
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
def _release(self) -> None:
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
else: # pragma: win32 no cover
|
|
|
|
try:
|
|
|
|
import fcntl
|
|
|
|
except ImportError:
|
|
|
|
pass
|
|
|
|
else:
|
|
|
|
has_fcntl = True
|
|
|
|
|
|
|
|
class UnixFileLock(BaseFileLock):
|
|
|
|
"""Uses the :func:`fcntl.flock` to hard lock the lock file on unix systems."""
|
|
|
|
|
|
|
|
def _acquire(self) -> None:
|
2023-09-17 19:40:34 +00:00
|
|
|
ensure_directory_exists(self.lock_file)
|
2023-04-12 13:51:13 +00:00
|
|
|
open_flags = os.O_RDWR | os.O_CREAT | os.O_TRUNC
|
2023-05-05 14:08:05 +00:00
|
|
|
fd = os.open(self.lock_file, open_flags, self._context.mode)
|
2023-09-17 19:40:34 +00:00
|
|
|
with suppress(PermissionError): # This locked is not owned by this UID
|
2023-05-05 14:08:05 +00:00
|
|
|
os.fchmod(fd, self._context.mode)
|
2023-01-06 11:47:44 +00:00
|
|
|
try:
|
|
|
|
fcntl.flock(fd, fcntl.LOCK_EX | fcntl.LOCK_NB)
|
2023-04-12 13:51:13 +00:00
|
|
|
except OSError as exception:
|
2023-01-06 11:47:44 +00:00
|
|
|
os.close(fd)
|
2023-04-12 13:51:13 +00:00
|
|
|
if exception.errno == ENOSYS: # NotImplemented error
|
2023-09-17 19:40:34 +00:00
|
|
|
msg = "FileSystem does not appear to support flock; user SoftFileLock instead"
|
|
|
|
raise NotImplementedError(msg) from exception
|
2023-01-06 11:47:44 +00:00
|
|
|
else:
|
2023-05-05 14:08:05 +00:00
|
|
|
self._context.lock_file_fd = fd
|
2023-01-06 11:47:44 +00:00
|
|
|
|
|
|
|
def _release(self) -> None:
|
|
|
|
# Do not remove the lockfile:
|
|
|
|
# https://github.com/tox-dev/py-filelock/issues/31
|
|
|
|
# https://stackoverflow.com/questions/17708885/flock-removing-locked-file-without-race-condition
|
2023-05-05 14:08:05 +00:00
|
|
|
fd = cast(int, self._context.lock_file_fd)
|
|
|
|
self._context.lock_file_fd = None
|
2023-01-06 11:47:44 +00:00
|
|
|
fcntl.flock(fd, fcntl.LOCK_UN)
|
|
|
|
os.close(fd)
|
|
|
|
|
|
|
|
|
|
|
|
__all__ = [
|
|
|
|
"has_fcntl",
|
|
|
|
"UnixFileLock",
|
|
|
|
]
|