1
# Copyright (C) 2005-2010 Canonical Ltd
3
# This program is free software; you can redistribute it and/or modify
4
# it under the terms of the GNU General Public License as published by
5
# the Free Software Foundation; either version 2 of the License, or
6
# (at your option) any later version.
8
# This program is distributed in the hope that it will be useful,
9
# but WITHOUT ANY WARRANTY; without even the implied warranty of
10
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11
# GNU General Public License for more details.
13
# You should have received a copy of the GNU General Public License
14
# along with this program; if not, write to the Free Software
15
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
17
"""Locking using OS file locks or file existence.
19
Note: This method of locking is generally deprecated in favour of LockDir, but
20
is used to lock local WorkingTrees, and by some old formats. It's accessed
21
through Transport.lock_read(), etc.
23
This module causes two methods, lock() and unlock() to be defined in
24
any way that works on the current platform.
26
It is not specified whether these locks are reentrant (i.e. can be
27
taken repeatedly by a single process) or whether they exclude
28
different threads in a single process. That reentrancy is provided by
31
This defines two classes: ReadLock and WriteLock, which can be
32
implemented in different ways on different platforms. Both have an
36
from __future__ import absolute_import
50
from .hooks import Hooks
51
from .i18n import gettext
54
class LockHooks(Hooks):
57
Hooks.__init__(self, "breezy.lock", "Lock.hooks")
60
"Called with a breezy.lock.LockResult when a physical lock is "
64
"Called with a breezy.lock.LockResult when a physical lock is "
68
"Called with a breezy.lock.LockResult when a physical lock is "
73
"""Base class for locks.
75
:cvar hooks: Hook dictionary for operations on locks.
81
class LockResult(object):
82
"""Result of an operation on a lock; passed to a hook"""
84
def __init__(self, lock_url, details=None):
85
"""Create a lock result for lock with optional details about the lock."""
86
self.lock_url = lock_url
87
self.details = details
89
def __eq__(self, other):
90
return self.lock_url == other.lock_url and self.details == other.details
93
return '%s(%s, %s)' % (self.__class__.__name__,
94
self.lock_url, self.details)
97
class LogicalLockResult(object):
98
"""The result of a lock_read/lock_write/lock_tree_write call on lockables.
100
:ivar unlock: A callable which will unlock the lock.
103
def __init__(self, unlock, token=None):
108
return "LogicalLockResult(%s)" % (self.unlock)
113
def __exit__(self, exc_type, exc_val, exc_tb):
114
# If there was an error raised, prefer the original one
117
except BaseException:
123
def cant_unlock_not_held(locked_object):
124
"""An attempt to unlock failed because the object was not locked.
126
This provides a policy point from which we can generate either a warning or
129
# This is typically masking some other error and called from a finally
130
# block, so it's useful to have the option not to generate a new error
131
# here. You can use -Werror to make it fatal. It should possibly also
133
if 'unlock' in debug.debug_flags:
134
warnings.warn("%r is already unlocked" % (locked_object,),
137
raise errors.LockNotHeld(locked_object)
147
have_ctypes_win32 = False
148
if sys.platform == 'win32':
160
have_ctypes_win32 = True
165
class _OSLock(object):
171
def _open(self, filename, filemode):
172
self.filename = osutils.realpath(filename)
174
self.f = open(self.filename, filemode)
177
if e.errno in (errno.EACCES, errno.EPERM):
178
raise errors.LockFailed(self.filename, str(e))
179
if e.errno != errno.ENOENT:
182
# maybe this is an old branch (before may 2005)
183
trace.mutter("trying to create missing lock %r", self.filename)
185
self.f = open(self.filename, 'wb+')
189
"""Clear the self.f attribute cleanly."""
195
raise NotImplementedError()
203
class _fcntl_FileLock(_OSLock):
206
fcntl.lockf(self.f, fcntl.LOCK_UN)
209
class _fcntl_WriteLock(_fcntl_FileLock):
213
def __init__(self, filename):
214
super(_fcntl_WriteLock, self).__init__()
215
# Check we can grab a lock before we actually open the file.
216
self.filename = osutils.realpath(filename)
217
if self.filename in _fcntl_WriteLock._open_locks:
219
raise errors.LockContention(self.filename)
220
if self.filename in _fcntl_ReadLock._open_locks:
221
if 'strict_locks' in debug.debug_flags:
223
raise errors.LockContention(self.filename)
225
trace.mutter('Write lock taken w/ an open read lock on: %s'
228
self._open(self.filename, 'rb+')
229
# reserve a slot for this lock - even if the lockf call fails,
230
# at this point unlock() will be called, because self.f is set.
231
# TODO: make this fully threadsafe, if we decide we care.
232
_fcntl_WriteLock._open_locks.add(self.filename)
234
# LOCK_NB will cause IOError to be raised if we can't grab a
236
fcntl.lockf(self.f, fcntl.LOCK_EX | fcntl.LOCK_NB)
238
if e.errno in (errno.EAGAIN, errno.EACCES):
239
# We couldn't grab the lock
241
# we should be more precise about whats a locking
242
# error and whats a random-other error
243
raise errors.LockContention(self.filename, e)
246
_fcntl_WriteLock._open_locks.remove(self.filename)
249
class _fcntl_ReadLock(_fcntl_FileLock):
253
def __init__(self, filename):
254
super(_fcntl_ReadLock, self).__init__()
255
self.filename = osutils.realpath(filename)
256
if self.filename in _fcntl_WriteLock._open_locks:
257
if 'strict_locks' in debug.debug_flags:
258
# We raise before calling _open so we don't need to
260
raise errors.LockContention(self.filename)
262
trace.mutter('Read lock taken w/ an open write lock on: %s'
264
_fcntl_ReadLock._open_locks.setdefault(self.filename, 0)
265
_fcntl_ReadLock._open_locks[self.filename] += 1
266
self._open(filename, 'rb')
268
# LOCK_NB will cause IOError to be raised if we can't grab a
270
fcntl.lockf(self.f, fcntl.LOCK_SH | fcntl.LOCK_NB)
272
# we should be more precise about whats a locking
273
# error and whats a random-other error
274
raise errors.LockContention(self.filename, e)
277
count = _fcntl_ReadLock._open_locks[self.filename]
279
del _fcntl_ReadLock._open_locks[self.filename]
281
_fcntl_ReadLock._open_locks[self.filename] = count - 1
284
def temporary_write_lock(self):
285
"""Try to grab a write lock on the file.
287
On platforms that support it, this will upgrade to a write lock
288
without unlocking the file.
289
Otherwise, this will release the read lock, and try to acquire a
292
:return: A token which can be used to switch back to a read lock.
294
if self.filename in _fcntl_WriteLock._open_locks:
295
raise AssertionError('file already locked: %r'
298
wlock = _fcntl_TemporaryWriteLock(self)
299
except errors.LockError:
300
# We didn't unlock, so we can just return 'self'
304
class _fcntl_TemporaryWriteLock(_OSLock):
305
"""A token used when grabbing a temporary_write_lock.
307
Call restore_read_lock() when you are done with the write lock.
310
def __init__(self, read_lock):
311
super(_fcntl_TemporaryWriteLock, self).__init__()
312
self._read_lock = read_lock
313
self.filename = read_lock.filename
315
count = _fcntl_ReadLock._open_locks[self.filename]
317
# Something else also has a read-lock, so we cannot grab a
319
raise errors.LockContention(self.filename)
321
if self.filename in _fcntl_WriteLock._open_locks:
322
raise AssertionError('file already locked: %r'
325
# See if we can open the file for writing. Another process might
326
# have a read lock. We don't use self._open() because we don't want
327
# to create the file if it exists. That would have already been
328
# done by _fcntl_ReadLock
330
new_f = open(self.filename, 'rb+')
332
if e.errno in (errno.EACCES, errno.EPERM):
333
raise errors.LockFailed(self.filename, str(e))
336
# LOCK_NB will cause IOError to be raised if we can't grab a
338
fcntl.lockf(new_f, fcntl.LOCK_EX | fcntl.LOCK_NB)
340
# TODO: Raise a more specific error based on the type of error
341
raise errors.LockContention(self.filename, e)
342
_fcntl_WriteLock._open_locks.add(self.filename)
346
def restore_read_lock(self):
347
"""Restore the original ReadLock."""
348
# For fcntl, since we never released the read lock, just release
349
# the write lock, and return the original lock.
350
fcntl.lockf(self.f, fcntl.LOCK_UN)
352
_fcntl_WriteLock._open_locks.remove(self.filename)
353
# Avoid reference cycles
354
read_lock = self._read_lock
355
self._read_lock = None
358
_lock_classes.append(('fcntl', _fcntl_WriteLock, _fcntl_ReadLock))
361
if have_pywin32 and sys.platform == 'win32':
362
win32file_CreateFile = win32file.CreateFileW
364
class _w32c_FileLock(_OSLock):
366
def _open(self, filename, access, share, cflags, pymode):
367
self.filename = osutils.realpath(filename)
369
self._handle = win32file_CreateFile(
370
filename, access, share, None, win32file.OPEN_ALWAYS,
371
win32file.FILE_ATTRIBUTE_NORMAL, None)
372
except pywintypes.error as e:
373
if e.args[0] == winerror.ERROR_ACCESS_DENIED:
374
raise errors.LockFailed(filename, e)
375
if e.args[0] == winerror.ERROR_SHARING_VIOLATION:
376
raise errors.LockContention(filename, e)
378
fd = win32file._open_osfhandle(self._handle, cflags)
379
self.f = os.fdopen(fd, pymode)
386
class _w32c_ReadLock(_w32c_FileLock):
387
def __init__(self, filename):
388
super(_w32c_ReadLock, self).__init__()
389
self._open(filename, win32file.GENERIC_READ,
390
win32file.FILE_SHARE_READ, os.O_RDONLY, "rb")
392
def temporary_write_lock(self):
393
"""Try to grab a write lock on the file.
395
On platforms that support it, this will upgrade to a write lock
396
without unlocking the file.
397
Otherwise, this will release the read lock, and try to acquire a
400
:return: A token which can be used to switch back to a read lock.
402
# I can't find a way to upgrade a read lock to a write lock without
403
# unlocking first. So here, we do just that.
406
wlock = _w32c_WriteLock(self.filename)
407
except errors.LockError:
408
return False, _w32c_ReadLock(self.filename)
411
class _w32c_WriteLock(_w32c_FileLock):
412
def __init__(self, filename):
413
super(_w32c_WriteLock, self).__init__()
415
win32file.GENERIC_READ | win32file.GENERIC_WRITE, 0,
418
def restore_read_lock(self):
419
"""Restore the original ReadLock."""
420
# For win32 we had to completely let go of the original lock, so we
421
# just unlock and create a new read lock.
423
return _w32c_ReadLock(self.filename)
425
_lock_classes.append(('pywin32', _w32c_WriteLock, _w32c_ReadLock))
428
if have_ctypes_win32:
429
from ctypes.wintypes import DWORD, LPWSTR
430
LPSECURITY_ATTRIBUTES = ctypes.c_void_p # used as NULL no need to declare
431
HANDLE = ctypes.c_int # rather than unsigned as in ctypes.wintypes
432
_function_name = "CreateFileW"
434
# CreateFile <http://msdn.microsoft.com/en-us/library/aa363858.aspx>
435
_CreateFile = ctypes.WINFUNCTYPE(
436
HANDLE, # return value
438
DWORD, # dwDesiredAccess
440
LPSECURITY_ATTRIBUTES, # lpSecurityAttributes
441
DWORD, # dwCreationDisposition
442
DWORD, # dwFlagsAndAttributes
443
HANDLE # hTemplateFile
444
)((_function_name, ctypes.windll.kernel32))
446
INVALID_HANDLE_VALUE = -1
448
GENERIC_READ = 0x80000000
449
GENERIC_WRITE = 0x40000000
452
FILE_ATTRIBUTE_NORMAL = 128
454
ERROR_ACCESS_DENIED = 5
455
ERROR_SHARING_VIOLATION = 32
457
class _ctypes_FileLock(_OSLock):
459
def _open(self, filename, access, share, cflags, pymode):
460
self.filename = osutils.realpath(filename)
461
handle = _CreateFile(filename, access, share, None, OPEN_ALWAYS,
462
FILE_ATTRIBUTE_NORMAL, 0)
463
if handle in (INVALID_HANDLE_VALUE, 0):
464
e = ctypes.WinError()
465
if e.args[0] == ERROR_ACCESS_DENIED:
466
raise errors.LockFailed(filename, e)
467
if e.args[0] == ERROR_SHARING_VIOLATION:
468
raise errors.LockContention(filename, e)
470
fd = msvcrt.open_osfhandle(handle, cflags)
471
self.f = os.fdopen(fd, pymode)
477
class _ctypes_ReadLock(_ctypes_FileLock):
478
def __init__(self, filename):
479
super(_ctypes_ReadLock, self).__init__()
480
self._open(filename, GENERIC_READ, FILE_SHARE_READ, os.O_RDONLY,
483
def temporary_write_lock(self):
484
"""Try to grab a write lock on the file.
486
On platforms that support it, this will upgrade to a write lock
487
without unlocking the file.
488
Otherwise, this will release the read lock, and try to acquire a
491
:return: A token which can be used to switch back to a read lock.
493
# I can't find a way to upgrade a read lock to a write lock without
494
# unlocking first. So here, we do just that.
497
wlock = _ctypes_WriteLock(self.filename)
498
except errors.LockError:
499
return False, _ctypes_ReadLock(self.filename)
502
class _ctypes_WriteLock(_ctypes_FileLock):
503
def __init__(self, filename):
504
super(_ctypes_WriteLock, self).__init__()
505
self._open(filename, GENERIC_READ | GENERIC_WRITE, 0, os.O_RDWR,
508
def restore_read_lock(self):
509
"""Restore the original ReadLock."""
510
# For win32 we had to completely let go of the original lock, so we
511
# just unlock and create a new read lock.
513
return _ctypes_ReadLock(self.filename)
515
_lock_classes.append(('ctypes', _ctypes_WriteLock, _ctypes_ReadLock))
518
if len(_lock_classes) == 0:
519
raise NotImplementedError(
520
"We must have one of fcntl, pywin32, or ctypes available"
521
" to support OS locking."
525
# We default to using the first available lock class.
526
_lock_type, WriteLock, ReadLock = _lock_classes[0]
529
class _RelockDebugMixin(object):
530
"""Mixin support for -Drelock flag.
532
Add this as a base class then call self._note_lock with 'r' or 'w' when
533
acquiring a read- or write-lock. If this object was previously locked (and
534
locked the same way), and -Drelock is set, then this will trace.note a
540
def _note_lock(self, lock_type):
541
if 'relock' in debug.debug_flags and self._prev_lock == lock_type:
546
trace.note(gettext('{0!r} was {1} locked again'), self, type_name)
547
self._prev_lock = lock_type
550
@contextlib.contextmanager
551
def write_locked(lockable):
552
lockable.lock_write()