309
344
def show(self, revno, rev, delta):
310
345
raise NotImplementedError('not implemented in abstract base')
347
def short_committer(self, rev):
348
return re.sub('<.*@.*>', '', rev.committer).strip(' ')
313
351
class LongLogFormatter(LogFormatter):
314
352
def show(self, revno, rev, delta):
315
from osutils import format_date
353
return self._show_helper(revno=revno, rev=rev, delta=delta)
355
def show_merge(self, rev):
356
return self._show_helper(rev=rev, indent=' ', merged=True, delta=None)
358
def _show_helper(self, rev=None, revno=None, indent='', merged=False, delta=None):
359
"""Show a revision, either merged or not."""
360
from bzrlib.osutils import format_date
317
361
to_file = self.to_file
319
print >>to_file, '-' * 60
320
print >>to_file, 'revno:', revno
362
print >>to_file, indent+'-' * 60
363
if revno is not None:
364
print >>to_file, 'revno:', revno
366
print >>to_file, indent+'merged:', rev.revision_id
368
print >>to_file, indent+'revision-id:', rev.revision_id
321
369
if self.show_ids:
322
print >>to_file, 'revision-id:', rev.revision_id
324
370
for parent_id in rev.parent_ids:
325
print >>to_file, 'parent:', parent_id
327
print >>to_file, 'committer:', rev.committer
371
print >>to_file, indent+'parent:', parent_id
372
print >>to_file, indent+'committer:', rev.committer
374
print >>to_file, indent+'branch nick: %s' % \
375
rev.properties['branch-nick']
329
378
date_str = format_date(rev.timestamp,
330
379
rev.timezone or 0,
331
380
self.show_timezone)
332
print >>to_file, 'timestamp: %s' % date_str
381
print >>to_file, indent+'timestamp: %s' % date_str
334
print >>to_file, 'message:'
383
print >>to_file, indent+'message:'
335
384
if not rev.message:
336
print >>to_file, ' (no message)'
385
print >>to_file, indent+' (no message)'
338
for l in rev.message.split('\n'):
339
print >>to_file, ' ' + l
387
message = rev.message.rstrip('\r\n')
388
for l in message.split('\n'):
389
print >>to_file, indent+' ' + l
341
390
if delta != None:
342
391
delta.show(to_file, self.show_ids)
346
394
class ShortLogFormatter(LogFormatter):
347
395
def show(self, revno, rev, delta):
348
396
from bzrlib.osutils import format_date
350
398
to_file = self.to_file
352
print >>to_file, "%5d %s\t%s" % (revno, rev.committer,
399
date_str = format_date(rev.timestamp, rev.timezone or 0,
401
print >>to_file, "%5d %s\t%s" % (revno, self.short_committer(rev),
353
402
format_date(rev.timestamp, rev.timezone or 0,
403
self.show_timezone, date_fmt="%Y-%m-%d",
355
405
if self.show_ids:
356
406
print >>to_file, ' revision-id:', rev.revision_id
357
407
if not rev.message:
358
408
print >>to_file, ' (no message)'
360
for l in rev.message.split('\n'):
410
message = rev.message.rstrip('\r\n')
411
for l in message.split('\n'):
361
412
print >>to_file, ' ' + l
363
414
# TODO: Why not show the modified files in a shorter form as
364
415
# well? rewrap them single lines of appropriate length
365
416
if delta != None:
366
417
delta.show(to_file, self.show_ids)
420
class LineLogFormatter(LogFormatter):
421
def truncate(self, str, max_len):
422
if len(str) <= max_len:
424
return str[:max_len-3]+'...'
426
def date_string(self, rev):
427
from bzrlib.osutils import format_date
428
return format_date(rev.timestamp, rev.timezone or 0,
429
self.show_timezone, date_fmt="%Y-%m-%d",
432
def message(self, rev):
434
return '(no message)'
438
def show(self, revno, rev, delta):
439
print >> self.to_file, self.log_string(rev, 79)
441
def log_string(self, rev, max_chars):
442
out = [self.truncate(self.short_committer(rev), 20)]
443
out.append(self.date_string(rev))
444
out.append(self.message(rev).replace('\n', ' '))
445
return self.truncate(" ".join(out).rstrip('\n'), max_chars)
447
def line_log(rev, max_chars):
448
lf = LineLogFormatter(None)
449
return lf.log_string(rev, max_chars)
371
451
FORMATTERS = {'long': LongLogFormatter,
372
452
'short': ShortLogFormatter,
453
'line': LineLogFormatter,
376
457
def log_formatter(name, *args, **kwargs):
377
458
"""Construct a formatter from arguments.
379
name -- Name of the formatter to construct; currently 'long' and
380
'short' are supported.
460
name -- Name of the formatter to construct; currently 'long', 'short' and
461
'line' are supported.
382
463
from bzrlib.errors import BzrCommandError
389
470
# deprecated; for compatability
390
471
lf = LongLogFormatter(to_file=to_file, show_timezone=show_timezone)
391
472
lf.show(revno, rev, delta)
474
def show_changed_revisions(branch, old_rh, new_rh, to_file=None, log_format='long'):
475
"""Show the change in revision history comparing the old revision history to the new one.
477
:param branch: The branch where the revisions exist
478
:param old_rh: The old revision history
479
:param new_rh: The new revision history
480
:param to_file: A file to write the results to. If None, stdout will be used
486
to_file = codecs.getwriter(bzrlib.user_encoding)(sys.stdout, errors='replace')
487
lf = log_formatter(log_format,
490
show_timezone='original')
492
# This is the first index which is different between
495
for i in xrange(max(len(new_rh),
499
or new_rh[i] != old_rh[i]):
504
to_file.write('Nothing seems to have changed\n')
506
## TODO: It might be nice to do something like show_log
507
## and show the merged entries. But since this is the
508
## removed revisions, it shouldn't be as important
509
if base_idx < len(old_rh):
510
to_file.write('*'*60)
511
to_file.write('\nRemoved Revisions:\n')
512
for i in range(base_idx, len(old_rh)):
513
rev = branch.get_revision(old_rh[i])
514
lf.show(i+1, rev, None)
515
to_file.write('*'*60)
516
to_file.write('\n\n')
517
if base_idx < len(new_rh):
518
to_file.write('Added Revisions:\n')
524
start_revision=base_idx+1,
525
end_revision=len(new_rh),