1
# Copyright (C) 2006, 2007, 2008 Canonical Ltd
1
# Copyright (C) 2006-2010 Canonical Ltd
3
3
# This program is free software; you can redistribute it and/or modify
4
4
# it under the terms of the GNU General Public License as published by
240
242
# incorrect. It's possible some other servers or filesystems will
241
243
# have a similar bug allowing someone to think they got the lock
242
244
# when it's already held.
246
# See <https://bugs.launchpad.net/bzr/+bug/498378> for one case.
248
# Strictly the check is unnecessary and a waste of time for most
249
# people, but probably worth trapping if something is wrong.
243
250
info = self.peek()
244
251
self._trace("after locking, info=%r", info)
245
if info['nonce'] != self.nonce:
253
raise LockFailed(self, "lock was renamed into place, but "
255
if info.get('nonce') != self.nonce:
246
256
self._trace("rename succeeded, "
247
257
"but lock is still held by someone else")
248
258
raise LockContention(self)
325
336
self._trace("... unlock succeeded after %dms",
326
337
(time.time() - start_time) * 1000)
327
338
result = lock.LockResult(self.transport.abspath(self.path),
329
340
for hook in self.hooks['lock_released']:
379
390
raise LockBreakMismatch(self, broken_info, dead_holder_info)
380
391
self.transport.delete(broken_info_path)
381
392
self.transport.rmdir(tmpname)
393
result = lock.LockResult(self.transport.abspath(self.path),
394
current_info.get('nonce'))
395
for hook in self.hooks['lock_broken']:
383
398
def _check_not_locked(self):
384
399
"""If the lock is held by this instance, raise an error."""
411
426
peek() reads the info file of the lock holder, if any.
413
return self._parse_info(self.transport.get(path))
428
return self._parse_info(self.transport.get_bytes(path))
416
431
"""Check if the lock is held by anyone.
418
If it is held, this returns the lock info structure as a rio Stanza,
433
If it is held, this returns the lock info structure as a dict
419
434
which contains some information about the current lock holder.
420
435
Otherwise returns None.
432
447
# XXX: is creating this here inefficient?
433
448
config = bzrlib.config.GlobalConfig()
435
user = config.user_email()
436
except errors.NoEmailInUsername:
437
450
user = config.username()
451
except errors.NoWhoami:
452
user = osutils.getuser_unicode()
438
453
s = rio.Stanza(hostname=get_host_name(),
439
454
pid=str(os.getpid()),
440
455
start_time=str(int(time.time())),
444
459
return s.to_string()
446
def _parse_info(self, info_file):
447
return rio.read_stanza(info_file.readlines()).as_dict()
461
def _parse_info(self, info_bytes):
462
stanza = rio.read_stanza(osutils.split_lines(info_bytes))
464
# see bug 185013; we fairly often end up with the info file being
465
# empty after an interruption; we could log a message here but
466
# there may not be much we can say
469
return stanza.as_dict()
449
471
def attempt_lock(self):
450
472
"""Take the lock; fail if it's already held.
517
539
if deadline_str is None:
518
540
deadline_str = time.strftime('%H:%M:%S',
519
541
time.localtime(deadline))
542
# As local lock urls are correct we display them.
543
# We avoid displaying remote lock urls.
520
544
lock_url = self.transport.abspath(self.path)
521
self._report_function('%s %s\n'
523
'%s\n' # locked ... ago
524
'Will continue to try until %s, unless '
526
'If you\'re sure that it\'s not being '
527
'modified, use bzr break-lock %s',
545
if lock_url.startswith('file://'):
546
lock_url = lock_url.split('.bzr/')[0]
549
user, hostname, pid, time_ago = formatted_info
550
msg = ('%s lock %s ' # lock_url
554
'[process #%s], ' # pid
555
'acquired %s.') # time ago
556
msg_args = [start, lock_url, user, hostname, pid, time_ago]
558
msg += ('\nWill continue to try until %s, unless '
560
msg_args.append(deadline_str)
561
msg += '\nSee "bzr help break-lock" for more.'
562
self._report_function(msg, *msg_args)
535
563
if (max_attempts is not None) and (attempt_count >= max_attempts):
536
564
self._trace("exceeded %d attempts")
537
565
raise LockContention(self)
539
567
self._trace("waiting %ss", poll)
570
# As timeout is always 0 for remote locks
571
# this block is applicable only for local
542
573
self._trace("timeout after waiting %ss", timeout)
543
raise LockContention(self)
574
raise LockContention('(local)', lock_url)
545
576
def leave_in_place(self):
546
577
self._locked_via_token = True
592
623
def _format_lock_info(self, info):
593
624
"""Turn the contents of peek() into something for the user"""
594
lock_url = self.transport.abspath(self.path)
595
delta = time.time() - int(info['start_time'])
625
start_time = info.get('start_time')
626
if start_time is None:
627
time_ago = '(unknown)'
629
time_ago = format_delta(time.time() - int(info['start_time']))
630
user = info.get('user', '<unknown>')
631
hostname = info.get('hostname', '<unknown>')
632
pid = info.get('pid', '<unknown>')
597
'lock %s' % (lock_url,),
598
'held by %(user)s on host %(hostname)s [process #%(pid)s]' % info,
599
'locked %s' % (format_delta(delta),),
602
640
def validate_token(self, token):