1
# Copyright (C) 2005, 2006, 2007 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
18
"""Locking using OS file locks or file existence.
20
Note: This method of locking is generally deprecated in favour of LockDir, but
21
is used to lock local WorkingTrees, and by some old formats. It's accessed
22
through Transport.lock_read(), etc.
24
This module causes two methods, lock() and unlock() to be defined in
25
any way that works on the current platform.
27
It is not specified whether these locks are reentrant (i.e. can be
28
taken repeatedly by a single process) or whether they exclude
29
different threads in a single process. That reentrancy is provided by
32
This defines two classes: ReadLock and WriteLock, which can be
33
implemented in different ways on different platforms. Both have an
41
from bzrlib import errors
42
from bzrlib.errors import LockError, LockContention
43
from bzrlib.osutils import realpath
44
from bzrlib.trace import mutter
47
class _base_Lock(object):
52
def _open(self, filename, filemode):
54
self.f = open(filename, filemode)
57
if e.errno in (errno.EACCES, errno.EPERM):
58
raise errors.ReadOnlyLockError(e)
59
if e.errno != errno.ENOENT:
62
# maybe this is an old branch (before may 2005)
63
mutter("trying to create missing branch lock %r", filename)
65
self.f = open(filename, 'wb+')
69
"""Clear the self.f attribute cleanly."""
76
from warnings import warn
77
warn("lock on %r not released" % self.f)
81
raise NotImplementedError()
84
have_ctypes = have_pywin32 = have_fcntl = False
91
import win32con, win32file, pywintypes, winerror, msvcrt
106
LOCK_SH = fcntl.LOCK_SH
107
LOCK_NB = fcntl.LOCK_NB
108
lock_EX = fcntl.LOCK_EX
111
class _fcntl_FileLock(_base_Lock):
114
fcntl.lockf(self.f, fcntl.LOCK_UN)
118
class _fcntl_WriteLock(_fcntl_FileLock):
122
def __init__(self, filename):
123
# standard IO errors get exposed directly.
124
super(_fcntl_WriteLock, self).__init__()
125
self._open(filename, 'rb+')
126
self.filename = realpath(filename)
127
if self.filename in self.open_locks:
129
raise LockContention(self.filename)
130
# reserve a slot for this lock - even if the lockf call fails,
131
# at thisi point unlock() will be called, because self.f is set.
132
# TODO: make this fully threadsafe, if we decide we care.
133
self.open_locks[self.filename] = self.filename
135
# LOCK_NB will cause IOError to be raised if we can't grab a
137
fcntl.lockf(self.f, fcntl.LOCK_EX | fcntl.LOCK_NB)
139
if e.errno in (errno.EAGAIN, errno.EACCES):
140
# We couldn't grab the lock
142
# we should be more precise about whats a locking
143
# error and whats a random-other error
147
del self.open_locks[self.filename]
151
class _fcntl_ReadLock(_fcntl_FileLock):
155
def __init__(self, filename):
156
super(_fcntl_ReadLock, self).__init__()
157
self._open(filename, 'rb')
159
# LOCK_NB will cause IOError to be raised if we can't grab a
161
fcntl.lockf(self.f, fcntl.LOCK_SH | fcntl.LOCK_NB)
163
# we should be more precise about whats a locking
164
# error and whats a random-other error
171
_lock_classes.append(('fcntl', _fcntl_WriteLock, _fcntl_ReadLock))
174
LOCK_SH = 0 # the default
175
LOCK_EX = win32con.LOCKFILE_EXCLUSIVE_LOCK
176
LOCK_NB = win32con.LOCKFILE_FAIL_IMMEDIATELY
179
class _w32c_FileLock(_base_Lock):
181
def _lock(self, filename, openmode, lockmode):
182
self._open(filename, openmode)
184
self.hfile = msvcrt.get_osfhandle(self.f.fileno())
185
overlapped = pywintypes.OVERLAPPED()
187
win32file.LockFileEx(self.hfile, lockmode, 0, 0x7fff0000,
189
except pywintypes.error, e:
191
if e.args[0] in (winerror.ERROR_LOCK_VIOLATION,):
192
raise errors.LockContention(filename)
193
## import pdb; pdb.set_trace()
200
overlapped = pywintypes.OVERLAPPED()
202
win32file.UnlockFileEx(self.hfile, 0, 0x7fff0000, overlapped)
208
class _w32c_ReadLock(_w32c_FileLock):
209
def __init__(self, filename):
210
super(_w32c_ReadLock, self).__init__()
211
self._lock(filename, 'rb', LOCK_SH + LOCK_NB)
213
class _w32c_WriteLock(_w32c_FileLock):
214
def __init__(self, filename):
215
super(_w32c_WriteLock, self).__init__()
216
self._lock(filename, 'rb+', LOCK_EX + LOCK_NB)
218
_lock_classes.append(('pywin32', _w32c_WriteLock, _w32c_ReadLock))
221
# These constants were copied from the win32con.py module.
222
LOCKFILE_FAIL_IMMEDIATELY = 1
223
LOCKFILE_EXCLUSIVE_LOCK = 2
224
# Constant taken from winerror.py module
225
ERROR_LOCK_VIOLATION = 33
228
LOCK_EX = LOCKFILE_EXCLUSIVE_LOCK
229
LOCK_NB = LOCKFILE_FAIL_IMMEDIATELY
230
_LockFileEx = ctypes.windll.kernel32.LockFileEx
231
_UnlockFileEx = ctypes.windll.kernel32.UnlockFileEx
232
_GetLastError = ctypes.windll.kernel32.GetLastError
234
### Define the OVERLAPPED structure.
235
# http://msdn2.microsoft.com/en-us/library/ms684342.aspx
236
# typedef struct _OVERLAPPED {
237
# ULONG_PTR Internal;
238
# ULONG_PTR InternalHigh;
249
class _inner_struct(ctypes.Structure):
250
_fields_ = [('Offset', ctypes.c_uint), # DWORD
251
('OffsetHigh', ctypes.c_uint), # DWORD
254
class _inner_union(ctypes.Union):
255
_fields_ = [('anon_struct', _inner_struct), # struct
256
('Pointer', ctypes.c_void_p), # PVOID
259
class OVERLAPPED(ctypes.Structure):
260
_fields_ = [('Internal', ctypes.c_void_p), # ULONG_PTR
261
('InternalHigh', ctypes.c_void_p), # ULONG_PTR
262
('_inner_union', _inner_union),
263
('hEvent', ctypes.c_void_p), # HANDLE
266
class _ctypes_FileLock(_base_Lock):
268
def _lock(self, filename, openmode, lockmode):
269
self._open(filename, openmode)
271
self.hfile = msvcrt.get_osfhandle(self.f.fileno())
272
overlapped = OVERLAPPED()
273
p_overlapped = ctypes.pointer(overlapped)
274
result = _LockFileEx(self.hfile, # HANDLE hFile
275
lockmode, # DWORD dwFlags
276
0, # DWORD dwReserved
277
0x7fffffff, # DWORD nNumberOfBytesToLockLow
278
0x00000000, # DWORD nNumberOfBytesToLockHigh
279
p_overlapped, # lpOverlapped
283
last_err = _GetLastError()
284
if last_err in (ERROR_LOCK_VIOLATION,):
285
raise errors.LockContention(filename)
286
raise errors.LockError('Unknown locking error: %s'
290
overlapped = OVERLAPPED()
291
p_overlapped = ctypes.pointer(overlapped)
292
result = _UnlockFileEx(self.hfile, # HANDLE hFile
293
0, # DWORD dwReserved
294
0x7fffffff, # DWORD nNumberOfBytesToLockLow
295
0x00000000, # DWORD nNumberOfBytesToLockHigh
296
p_overlapped, # lpOverlapped
301
last_err = _GetLastError()
302
raise errors.LockError('Unknown unlocking error: %s'
306
class _ctypes_ReadLock(_ctypes_FileLock):
307
def __init__(self, filename):
308
super(_ctypes_ReadLock, self).__init__()
309
self._lock(filename, 'rb', LOCK_SH + LOCK_NB)
311
class _ctypes_WriteLock(_ctypes_FileLock):
312
def __init__(self, filename):
313
super(_ctypes_WriteLock, self).__init__()
314
self._lock(filename, 'rb+', LOCK_EX + LOCK_NB)
316
_lock_classes.append(('ctypes', _ctypes_WriteLock, _ctypes_ReadLock))
319
if len(_lock_classes) == 0:
320
raise NotImplementedError("We only have support for"
321
" fcntl, pywin32 or ctypes locking."
322
" If your platform (windows) does not"
323
" support fcntl locks, you must have"
324
" either pywin32 or ctypes installed.")
326
# We default to using the first available lock class.
327
_lock_type, WriteLock, ReadLock = _lock_classes[0]
330
class LockTreeTestProviderAdapter(object):
331
"""A tool to generate a suite testing multiple lock formats at once.
333
This is done by copying the test once for each lock and injecting the
334
read_lock and write_lock classes.
335
They are also given a new test id.
338
def __init__(self, lock_classes):
339
self._lock_classes = lock_classes
341
def _clone_test(self, test, write_lock, read_lock, variation):
342
"""Clone test for adaption."""
343
new_test = deepcopy(test)
344
new_test.write_lock = write_lock
345
new_test.read_lock = read_lock
346
def make_new_test_id():
347
new_id = "%s(%s)" % (test.id(), variation)
348
return lambda: new_id
349
new_test.id = make_new_test_id()
352
def adapt(self, test):
353
from bzrlib.tests import TestSuite
355
for name, write_lock, read_lock in self._lock_classes:
356
new_test = self._clone_test(test, write_lock, read_lock, name)
357
result.addTest(new_test)