~bzr-pqm/bzr/bzr.dev

« back to all changes in this revision

Viewing changes to bzrlib/log.py

  • Committer: Patch Queue Manager
  • Date: 2011-10-14 16:54:26 UTC
  • mfrom: (6216.1.1 remove-this-file)
  • Revision ID: pqm@pqm.ubuntu.com-20111014165426-tjix4e6idryf1r2z
(jelmer) Remove an accidentally committed .THIS file. (Jelmer Vernooij)

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
# Copyright (C) 2005, 2006, 2007, 2009 Canonical Ltd
 
1
# Copyright (C) 2005-2011 Canonical Ltd
2
2
#
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
65
65
lazy_import(globals(), """
66
66
 
67
67
from bzrlib import (
68
 
    bzrdir,
69
68
    config,
 
69
    controldir,
70
70
    diff,
71
71
    errors,
72
72
    foreign,
73
73
    repository as _mod_repository,
74
74
    revision as _mod_revision,
75
75
    revisionspec,
76
 
    trace,
77
76
    tsort,
78
77
    )
 
78
from bzrlib.i18n import gettext, ngettext
79
79
""")
80
80
 
81
81
from bzrlib import (
 
82
    lazy_regex,
82
83
    registry,
83
84
    )
84
85
from bzrlib.osutils import (
85
86
    format_date,
 
87
    format_date_with_offset_in_original_timezone,
 
88
    get_diff_header_encoding,
86
89
    get_terminal_encoding,
87
 
    re_compile_checked,
88
90
    terminal_width,
89
91
    )
90
92
 
103
105
    last_ie = None
104
106
    last_path = None
105
107
    revno = 1
106
 
    for revision_id in branch.revision_history():
107
 
        this_inv = branch.repository.get_revision_inventory(revision_id)
108
 
        if file_id in this_inv:
 
108
    graph = branch.repository.get_graph()
 
109
    history = list(graph.iter_lefthand_ancestry(branch.last_revision(),
 
110
        [_mod_revision.NULL_REVISION]))
 
111
    for revision_id in reversed(history):
 
112
        this_inv = branch.repository.get_inventory(revision_id)
 
113
        if this_inv.has_id(file_id):
109
114
            this_ie = this_inv[file_id]
110
115
            this_path = this_inv.id2path(file_id)
111
116
        else:
151
156
             end_revision=None,
152
157
             search=None,
153
158
             limit=None,
154
 
             show_diff=False):
 
159
             show_diff=False,
 
160
             match=None):
155
161
    """Write out human-readable log of commits to this branch.
156
162
 
157
163
    This function is being retained for backwards compatibility but
180
186
        if None or 0.
181
187
 
182
188
    :param show_diff: If True, output a diff after each revision.
 
189
 
 
190
    :param match: Dictionary of search lists to use when matching revision
 
191
      properties.
183
192
    """
184
193
    # Convert old-style parameters to new-style parameters
185
194
    if specific_fileid is not None:
209
218
    Logger(branch, rqst).show(lf)
210
219
 
211
220
 
212
 
# Note: This needs to be kept this in sync with the defaults in
 
221
# Note: This needs to be kept in sync with the defaults in
213
222
# make_log_request_dict() below
214
223
_DEFAULT_REQUEST_PARAMS = {
215
224
    'direction': 'reverse',
216
 
    'levels': 1,
 
225
    'levels': None,
217
226
    'generate_tags': True,
 
227
    'exclude_common_ancestry': False,
218
228
    '_match_using_deltas': True,
219
229
    }
220
230
 
221
231
 
222
232
def make_log_request_dict(direction='reverse', specific_fileids=None,
223
 
    start_revision=None, end_revision=None, limit=None,
224
 
    message_search=None, levels=1, generate_tags=True, delta_type=None,
225
 
    diff_type=None, _match_using_deltas=True):
 
233
                          start_revision=None, end_revision=None, limit=None,
 
234
                          message_search=None, levels=None, generate_tags=True,
 
235
                          delta_type=None,
 
236
                          diff_type=None, _match_using_deltas=True,
 
237
                          exclude_common_ancestry=False, match=None,
 
238
                          signature=False, omit_merges=False,
 
239
                          ):
226
240
    """Convenience function for making a logging request dictionary.
227
241
 
228
242
    Using this function may make code slightly safer by ensuring
248
262
      matching commit messages
249
263
 
250
264
    :param levels: the number of levels of revisions to
251
 
      generate; 1 for just the mainline; 0 for all levels.
 
265
      generate; 1 for just the mainline; 0 for all levels, or None for
 
266
      a sensible default.
252
267
 
253
268
    :param generate_tags: If True, include tags for matched revisions.
254
 
 
 
269
`
255
270
    :param delta_type: Either 'full', 'partial' or None.
256
271
      'full' means generate the complete delta - adds/deletes/modifies/etc;
257
272
      'partial' means filter the delta using specific_fileids;
266
281
      algorithm used for matching specific_fileids. This parameter
267
282
      may be removed in the future so bzrlib client code should NOT
268
283
      use it.
 
284
 
 
285
    :param exclude_common_ancestry: Whether -rX..Y should be interpreted as a
 
286
      range operator or as a graph difference.
 
287
 
 
288
    :param signature: show digital signature information
 
289
 
 
290
    :param match: Dictionary of list of search strings to use when filtering
 
291
      revisions. Keys can be 'message', 'author', 'committer', 'bugs' or
 
292
      the empty string to match any of the preceding properties.
 
293
 
 
294
    :param omit_merges: If True, commits with more than one parent are
 
295
      omitted.
 
296
 
269
297
    """
 
298
    # Take care of old style message_search parameter
 
299
    if message_search:
 
300
        if match:
 
301
            if 'message' in match:
 
302
                match['message'].append(message_search)
 
303
            else:
 
304
                match['message'] = [message_search]
 
305
        else:
 
306
            match={ 'message': [message_search] }
270
307
    return {
271
308
        'direction': direction,
272
309
        'specific_fileids': specific_fileids,
273
310
        'start_revision': start_revision,
274
311
        'end_revision': end_revision,
275
312
        'limit': limit,
276
 
        'message_search': message_search,
277
313
        'levels': levels,
278
314
        'generate_tags': generate_tags,
279
315
        'delta_type': delta_type,
280
316
        'diff_type': diff_type,
 
317
        'exclude_common_ancestry': exclude_common_ancestry,
 
318
        'signature': signature,
 
319
        'match': match,
 
320
        'omit_merges': omit_merges,
281
321
        # Add 'private' attributes for features that may be deprecated
282
322
        '_match_using_deltas': _match_using_deltas,
283
323
    }
285
325
 
286
326
def _apply_log_request_defaults(rqst):
287
327
    """Apply default values to a request dictionary."""
288
 
    result = _DEFAULT_REQUEST_PARAMS
 
328
    result = _DEFAULT_REQUEST_PARAMS.copy()
289
329
    if rqst:
290
330
        result.update(rqst)
291
331
    return result
292
332
 
293
333
 
 
334
def format_signature_validity(rev_id, repo):
 
335
    """get the signature validity
 
336
 
 
337
    :param rev_id: revision id to validate
 
338
    :param repo: repository of revision
 
339
    :return: human readable string to print to log
 
340
    """
 
341
    from bzrlib import gpg
 
342
 
 
343
    gpg_strategy = gpg.GPGStrategy(None)
 
344
    result = repo.verify_revision(rev_id, gpg_strategy)
 
345
    if result[0] == gpg.SIGNATURE_VALID:
 
346
        return "valid signature from {0}".format(result[1])
 
347
    if result[0] == gpg.SIGNATURE_KEY_MISSING:
 
348
        return "unknown key {0}".format(result[1])
 
349
    if result[0] == gpg.SIGNATURE_NOT_VALID:
 
350
        return "invalid signature!"
 
351
    if result[0] == gpg.SIGNATURE_NOT_SIGNED:
 
352
        return "no signature"
 
353
 
 
354
 
294
355
class LogGenerator(object):
295
356
    """A generator of log revisions."""
296
357
 
303
364
 
304
365
 
305
366
class Logger(object):
306
 
    """An object the generates, formats and displays a log."""
 
367
    """An object that generates, formats and displays a log."""
307
368
 
308
369
    def __init__(self, branch, rqst):
309
370
        """Create a Logger.
341
402
        # Tweak the LogRequest based on what the LogFormatter can handle.
342
403
        # (There's no point generating stuff if the formatter can't display it.)
343
404
        rqst = self.rqst
344
 
        rqst['levels'] = lf.get_levels()
 
405
        if rqst['levels'] is None or lf.get_levels() > rqst['levels']:
 
406
            # user didn't specify levels, use whatever the LF can handle:
 
407
            rqst['levels'] = lf.get_levels()
 
408
 
345
409
        if not getattr(lf, 'supports_tags', False):
346
410
            rqst['generate_tags'] = False
347
411
        if not getattr(lf, 'supports_delta', False):
348
412
            rqst['delta_type'] = None
349
413
        if not getattr(lf, 'supports_diff', False):
350
414
            rqst['diff_type'] = None
 
415
        if not getattr(lf, 'supports_signatures', False):
 
416
            rqst['signature'] = False
351
417
 
352
418
        # Find and print the interesting revisions
353
419
        generator = self._generator_factory(self.branch, rqst)
357
423
 
358
424
    def _generator_factory(self, branch, rqst):
359
425
        """Make the LogGenerator object to use.
360
 
        
 
426
 
361
427
        Subclasses may wish to override this.
362
428
        """
363
429
        return _DefaultLogGenerator(branch, rqst)
384
450
        :return: An iterator yielding LogRevision objects.
385
451
        """
386
452
        rqst = self.rqst
 
453
        levels = rqst.get('levels')
 
454
        limit = rqst.get('limit')
 
455
        diff_type = rqst.get('diff_type')
 
456
        show_signature = rqst.get('signature')
 
457
        omit_merges = rqst.get('omit_merges')
387
458
        log_count = 0
388
459
        revision_iterator = self._create_log_revision_iterator()
389
460
        for revs in revision_iterator:
390
461
            for (rev_id, revno, merge_depth), rev, delta in revs:
391
462
                # 0 levels means show everything; merge_depth counts from 0
392
 
                levels = rqst.get('levels')
393
463
                if levels != 0 and merge_depth >= levels:
394
464
                    continue
395
 
                diff = self._format_diff(rev, rev_id)
 
465
                if omit_merges and len(rev.parent_ids) > 1:
 
466
                    continue
 
467
                if diff_type is None:
 
468
                    diff = None
 
469
                else:
 
470
                    diff = self._format_diff(rev, rev_id, diff_type)
 
471
                if show_signature:
 
472
                    signature = format_signature_validity(rev_id,
 
473
                                                self.branch.repository)
 
474
                else:
 
475
                    signature = None
396
476
                yield LogRevision(rev, revno, merge_depth, delta,
397
 
                    self.rev_tag_dict.get(rev_id), diff)
398
 
                limit = rqst.get('limit')
 
477
                    self.rev_tag_dict.get(rev_id), diff, signature)
399
478
                if limit:
400
479
                    log_count += 1
401
480
                    if log_count >= limit:
402
481
                        return
403
482
 
404
 
    def _format_diff(self, rev, rev_id):
405
 
        diff_type = self.rqst.get('diff_type')
406
 
        if diff_type is None:
407
 
            return None
 
483
    def _format_diff(self, rev, rev_id, diff_type):
408
484
        repo = self.branch.repository
409
485
        if len(rev.parent_ids) == 0:
410
486
            ancestor_id = _mod_revision.NULL_REVISION
418
494
        else:
419
495
            specific_files = None
420
496
        s = StringIO()
 
497
        path_encoding = get_diff_header_encoding()
421
498
        diff.show_diff_trees(tree_1, tree_2, s, specific_files, old_label='',
422
 
            new_label='')
 
499
            new_label='', path_encoding=path_encoding)
423
500
        return s.getvalue()
424
501
 
425
502
    def _create_log_revision_iterator(self):
449
526
        generate_merge_revisions = rqst.get('levels') != 1
450
527
        delayed_graph_generation = not rqst.get('specific_fileids') and (
451
528
                rqst.get('limit') or self.start_rev_id or self.end_rev_id)
452
 
        view_revisions = _calc_view_revisions(self.branch, self.start_rev_id,
453
 
            self.end_rev_id, rqst.get('direction'), generate_merge_revisions,
454
 
            delayed_graph_generation=delayed_graph_generation)
 
529
        view_revisions = _calc_view_revisions(
 
530
            self.branch, self.start_rev_id, self.end_rev_id,
 
531
            rqst.get('direction'),
 
532
            generate_merge_revisions=generate_merge_revisions,
 
533
            delayed_graph_generation=delayed_graph_generation,
 
534
            exclude_common_ancestry=rqst.get('exclude_common_ancestry'))
455
535
 
456
536
        # Apply the other filters
457
537
        return make_log_rev_iterator(self.branch, view_revisions,
458
 
            rqst.get('delta_type'), rqst.get('message_search'),
 
538
            rqst.get('delta_type'), rqst.get('match'),
459
539
            file_ids=rqst.get('specific_fileids'),
460
540
            direction=rqst.get('direction'))
461
541
 
464
544
        # Note that we always generate the merge revisions because
465
545
        # filter_revisions_touching_file_id() requires them ...
466
546
        rqst = self.rqst
467
 
        view_revisions = _calc_view_revisions(self.branch, self.start_rev_id,
468
 
            self.end_rev_id, rqst.get('direction'), True)
 
547
        view_revisions = _calc_view_revisions(
 
548
            self.branch, self.start_rev_id, self.end_rev_id,
 
549
            rqst.get('direction'), generate_merge_revisions=True,
 
550
            exclude_common_ancestry=rqst.get('exclude_common_ancestry'))
469
551
        if not isinstance(view_revisions, list):
470
552
            view_revisions = list(view_revisions)
471
553
        view_revisions = _filter_revisions_touching_file_id(self.branch,
472
554
            rqst.get('specific_fileids')[0], view_revisions,
473
555
            include_merges=rqst.get('levels') != 1)
474
556
        return make_log_rev_iterator(self.branch, view_revisions,
475
 
            rqst.get('delta_type'), rqst.get('message_search'))
 
557
            rqst.get('delta_type'), rqst.get('match'))
476
558
 
477
559
 
478
560
def _calc_view_revisions(branch, start_rev_id, end_rev_id, direction,
479
 
    generate_merge_revisions, delayed_graph_generation=False):
 
561
                         generate_merge_revisions,
 
562
                         delayed_graph_generation=False,
 
563
                         exclude_common_ancestry=False,
 
564
                         ):
480
565
    """Calculate the revisions to view.
481
566
 
482
567
    :return: An iterator of (revision_id, dotted_revno, merge_depth) tuples OR
483
568
             a list of the same tuples.
484
569
    """
 
570
    if (exclude_common_ancestry and start_rev_id == end_rev_id):
 
571
        raise errors.BzrCommandError(gettext(
 
572
            '--exclude-common-ancestry requires two different revisions'))
 
573
    if direction not in ('reverse', 'forward'):
 
574
        raise ValueError(gettext('invalid direction %r') % direction)
485
575
    br_revno, br_rev_id = branch.last_revision_info()
486
576
    if br_revno == 0:
487
577
        return []
488
578
 
489
 
    # If a single revision is requested, check we can handle it
490
 
    generate_single_revision = (end_rev_id and start_rev_id == end_rev_id and
491
 
        (not generate_merge_revisions or not _has_merges(branch, end_rev_id)))
492
 
    if generate_single_revision:
493
 
        return _generate_one_revision(branch, end_rev_id, br_rev_id, br_revno)
494
 
 
495
 
    # If we only want to see linear revisions, we can iterate ...
496
 
    if not generate_merge_revisions:
497
 
        return _generate_flat_revisions(branch, start_rev_id, end_rev_id,
498
 
            direction)
 
579
    if (end_rev_id and start_rev_id == end_rev_id
 
580
        and (not generate_merge_revisions
 
581
             or not _has_merges(branch, end_rev_id))):
 
582
        # If a single revision is requested, check we can handle it
 
583
        iter_revs = _generate_one_revision(branch, end_rev_id, br_rev_id,
 
584
                                           br_revno)
 
585
    elif not generate_merge_revisions:
 
586
        # If we only want to see linear revisions, we can iterate ...
 
587
        iter_revs = _generate_flat_revisions(branch, start_rev_id, end_rev_id,
 
588
                                             direction, exclude_common_ancestry)
 
589
        if direction == 'forward':
 
590
            iter_revs = reversed(iter_revs)
499
591
    else:
500
 
        return _generate_all_revisions(branch, start_rev_id, end_rev_id,
501
 
            direction, delayed_graph_generation)
 
592
        iter_revs = _generate_all_revisions(branch, start_rev_id, end_rev_id,
 
593
                                            direction, delayed_graph_generation,
 
594
                                            exclude_common_ancestry)
 
595
        if direction == 'forward':
 
596
            iter_revs = _rebase_merge_depth(reverse_by_depth(list(iter_revs)))
 
597
    return iter_revs
502
598
 
503
599
 
504
600
def _generate_one_revision(branch, rev_id, br_rev_id, br_revno):
506
602
        # It's the tip
507
603
        return [(br_rev_id, br_revno, 0)]
508
604
    else:
509
 
        revno = branch.revision_id_to_dotted_revno(rev_id)
510
 
        revno_str = '.'.join(str(n) for n in revno)
 
605
        revno_str = _compute_revno_str(branch, rev_id)
511
606
        return [(rev_id, revno_str, 0)]
512
607
 
513
608
 
514
 
def _generate_flat_revisions(branch, start_rev_id, end_rev_id, direction):
515
 
    result = _linear_view_revisions(branch, start_rev_id, end_rev_id)
 
609
def _generate_flat_revisions(branch, start_rev_id, end_rev_id, direction,
 
610
                             exclude_common_ancestry=False):
 
611
    result = _linear_view_revisions(
 
612
        branch, start_rev_id, end_rev_id,
 
613
        exclude_common_ancestry=exclude_common_ancestry)
516
614
    # If a start limit was given and it's not obviously an
517
615
    # ancestor of the end limit, check it before outputting anything
518
616
    if direction == 'forward' or (start_rev_id
520
618
        try:
521
619
            result = list(result)
522
620
        except _StartNotLinearAncestor:
523
 
            raise errors.BzrCommandError('Start revision not found in'
524
 
                ' left-hand history of end revision.')
525
 
    if direction == 'forward':
526
 
        result = reversed(result)
 
621
            raise errors.BzrCommandError(gettext('Start revision not found in'
 
622
                ' left-hand history of end revision.'))
527
623
    return result
528
624
 
529
625
 
530
626
def _generate_all_revisions(branch, start_rev_id, end_rev_id, direction,
531
 
    delayed_graph_generation):
 
627
                            delayed_graph_generation,
 
628
                            exclude_common_ancestry=False):
532
629
    # On large trees, generating the merge graph can take 30-60 seconds
533
630
    # so we delay doing it until a merge is detected, incrementally
534
631
    # returning initial (non-merge) revisions while we can.
 
632
 
 
633
    # The above is only true for old formats (<= 0.92), for newer formats, a
 
634
    # couple of seconds only should be needed to load the whole graph and the
 
635
    # other graph operations needed are even faster than that -- vila 100201
535
636
    initial_revisions = []
536
637
    if delayed_graph_generation:
537
638
        try:
538
 
            for rev_id, revno, depth in \
539
 
                _linear_view_revisions(branch, start_rev_id, end_rev_id):
 
639
            for rev_id, revno, depth in  _linear_view_revisions(
 
640
                branch, start_rev_id, end_rev_id, exclude_common_ancestry):
540
641
                if _has_merges(branch, rev_id):
 
642
                    # The end_rev_id can be nested down somewhere. We need an
 
643
                    # explicit ancestry check. There is an ambiguity here as we
 
644
                    # may not raise _StartNotLinearAncestor for a revision that
 
645
                    # is an ancestor but not a *linear* one. But since we have
 
646
                    # loaded the graph to do the check (or calculate a dotted
 
647
                    # revno), we may as well accept to show the log...  We need
 
648
                    # the check only if start_rev_id is not None as all
 
649
                    # revisions have _mod_revision.NULL_REVISION as an ancestor
 
650
                    # -- vila 20100319
 
651
                    graph = branch.repository.get_graph()
 
652
                    if (start_rev_id is not None
 
653
                        and not graph.is_ancestor(start_rev_id, end_rev_id)):
 
654
                        raise _StartNotLinearAncestor()
 
655
                    # Since we collected the revisions so far, we need to
 
656
                    # adjust end_rev_id.
541
657
                    end_rev_id = rev_id
542
658
                    break
543
659
                else:
544
660
                    initial_revisions.append((rev_id, revno, depth))
545
661
            else:
546
662
                # No merged revisions found
547
 
                if direction == 'reverse':
548
 
                    return initial_revisions
549
 
                elif direction == 'forward':
550
 
                    return reversed(initial_revisions)
551
 
                else:
552
 
                    raise ValueError('invalid direction %r' % direction)
 
663
                return initial_revisions
553
664
        except _StartNotLinearAncestor:
554
665
            # A merge was never detected so the lower revision limit can't
555
666
            # be nested down somewhere
556
 
            raise errors.BzrCommandError('Start revision not found in'
557
 
                ' history of end revision.')
 
667
            raise errors.BzrCommandError(gettext('Start revision not found in'
 
668
                ' history of end revision.'))
 
669
 
 
670
    # We exit the loop above because we encounter a revision with merges, from
 
671
    # this revision, we need to switch to _graph_view_revisions.
558
672
 
559
673
    # A log including nested merges is required. If the direction is reverse,
560
674
    # we rebase the initial merge depths so that the development line is
563
677
    # indented at the end seems slightly nicer in that case.
564
678
    view_revisions = chain(iter(initial_revisions),
565
679
        _graph_view_revisions(branch, start_rev_id, end_rev_id,
566
 
        rebase_initial_depths=direction == 'reverse'))
567
 
    if direction == 'reverse':
568
 
        return view_revisions
569
 
    elif direction == 'forward':
570
 
        # Forward means oldest first, adjusting for depth.
571
 
        view_revisions = reverse_by_depth(list(view_revisions))
572
 
        return _rebase_merge_depth(view_revisions)
573
 
    else:
574
 
        raise ValueError('invalid direction %r' % direction)
 
680
                              rebase_initial_depths=(direction == 'reverse'),
 
681
                              exclude_common_ancestry=exclude_common_ancestry))
 
682
    return view_revisions
575
683
 
576
684
 
577
685
def _has_merges(branch, rev_id):
580
688
    return len(parents) > 1
581
689
 
582
690
 
 
691
def _compute_revno_str(branch, rev_id):
 
692
    """Compute the revno string from a rev_id.
 
693
 
 
694
    :return: The revno string, or None if the revision is not in the supplied
 
695
        branch.
 
696
    """
 
697
    try:
 
698
        revno = branch.revision_id_to_dotted_revno(rev_id)
 
699
    except errors.NoSuchRevision:
 
700
        # The revision must be outside of this branch
 
701
        return None
 
702
    else:
 
703
        return '.'.join(str(n) for n in revno)
 
704
 
 
705
 
583
706
def _is_obvious_ancestor(branch, start_rev_id, end_rev_id):
584
707
    """Is start_rev_id an obvious ancestor of end_rev_id?"""
585
708
    if start_rev_id and end_rev_id:
586
 
        start_dotted = branch.revision_id_to_dotted_revno(start_rev_id)
587
 
        end_dotted = branch.revision_id_to_dotted_revno(end_rev_id)
 
709
        try:
 
710
            start_dotted = branch.revision_id_to_dotted_revno(start_rev_id)
 
711
            end_dotted = branch.revision_id_to_dotted_revno(end_rev_id)
 
712
        except errors.NoSuchRevision:
 
713
            # one or both is not in the branch; not obvious
 
714
            return False
588
715
        if len(start_dotted) == 1 and len(end_dotted) == 1:
589
716
            # both on mainline
590
717
            return start_dotted[0] <= end_dotted[0]
595
722
        else:
596
723
            # not obvious
597
724
            return False
 
725
    # if either start or end is not specified then we use either the first or
 
726
    # the last revision and *they* are obvious ancestors.
598
727
    return True
599
728
 
600
729
 
601
 
def _linear_view_revisions(branch, start_rev_id, end_rev_id):
 
730
def _linear_view_revisions(branch, start_rev_id, end_rev_id,
 
731
                           exclude_common_ancestry=False):
602
732
    """Calculate a sequence of revisions to view, newest to oldest.
603
733
 
604
734
    :param start_rev_id: the lower revision-id
605
735
    :param end_rev_id: the upper revision-id
 
736
    :param exclude_common_ancestry: Whether the start_rev_id should be part of
 
737
        the iterated revisions.
606
738
    :return: An iterator of (revision_id, dotted_revno, merge_depth) tuples.
607
739
    :raises _StartNotLinearAncestor: if a start_rev_id is specified but
608
 
      is not found walking the left-hand history
 
740
        is not found walking the left-hand history
609
741
    """
610
742
    br_revno, br_rev_id = branch.last_revision_info()
611
743
    repo = branch.repository
 
744
    graph = repo.get_graph()
612
745
    if start_rev_id is None and end_rev_id is None:
613
746
        cur_revno = br_revno
614
 
        for revision_id in repo.iter_reverse_revision_history(br_rev_id):
 
747
        for revision_id in graph.iter_lefthand_ancestry(br_rev_id,
 
748
            (_mod_revision.NULL_REVISION,)):
615
749
            yield revision_id, str(cur_revno), 0
616
750
            cur_revno -= 1
617
751
    else:
618
752
        if end_rev_id is None:
619
753
            end_rev_id = br_rev_id
620
754
        found_start = start_rev_id is None
621
 
        for revision_id in repo.iter_reverse_revision_history(end_rev_id):
622
 
            revno = branch.revision_id_to_dotted_revno(revision_id)
623
 
            revno_str = '.'.join(str(n) for n in revno)
 
755
        for revision_id in graph.iter_lefthand_ancestry(end_rev_id,
 
756
                (_mod_revision.NULL_REVISION,)):
 
757
            revno_str = _compute_revno_str(branch, revision_id)
624
758
            if not found_start and revision_id == start_rev_id:
625
 
                yield revision_id, revno_str, 0
 
759
                if not exclude_common_ancestry:
 
760
                    yield revision_id, revno_str, 0
626
761
                found_start = True
627
762
                break
628
763
            else:
633
768
 
634
769
 
635
770
def _graph_view_revisions(branch, start_rev_id, end_rev_id,
636
 
    rebase_initial_depths=True):
 
771
                          rebase_initial_depths=True,
 
772
                          exclude_common_ancestry=False):
637
773
    """Calculate revisions to view including merges, newest to oldest.
638
774
 
639
775
    :param branch: the branch
643
779
      revision is found?
644
780
    :return: An iterator of (revision_id, dotted_revno, merge_depth) tuples.
645
781
    """
 
782
    if exclude_common_ancestry:
 
783
        stop_rule = 'with-merges-without-common-ancestry'
 
784
    else:
 
785
        stop_rule = 'with-merges'
646
786
    view_revisions = branch.iter_merge_sorted_revisions(
647
787
        start_revision_id=end_rev_id, stop_revision_id=start_rev_id,
648
 
        stop_rule="with-merges")
 
788
        stop_rule=stop_rule)
649
789
    if not rebase_initial_depths:
650
790
        for (rev_id, merge_depth, revno, end_of_merge
651
791
             ) in view_revisions:
662
802
                depth_adjustment = merge_depth
663
803
            if depth_adjustment:
664
804
                if merge_depth < depth_adjustment:
 
805
                    # From now on we reduce the depth adjustement, this can be
 
806
                    # surprising for users. The alternative requires two passes
 
807
                    # which breaks the fast display of the first revision
 
808
                    # though.
665
809
                    depth_adjustment = merge_depth
666
810
                merge_depth -= depth_adjustment
667
811
            yield rev_id, '.'.join(map(str, revno)), merge_depth
668
812
 
669
813
 
670
 
def calculate_view_revisions(branch, start_revision, end_revision, direction,
671
 
        specific_fileid, generate_merge_revisions):
672
 
    """Calculate the revisions to view.
673
 
 
674
 
    :return: An iterator of (revision_id, dotted_revno, merge_depth) tuples OR
675
 
             a list of the same tuples.
676
 
    """
677
 
    # This method is no longer called by the main code path.
678
 
    # It is retained for API compatibility and may be deprecated
679
 
    # soon. IGC 20090116
680
 
    start_rev_id, end_rev_id = _get_revision_limits(branch, start_revision,
681
 
        end_revision)
682
 
    view_revisions = list(_calc_view_revisions(branch, start_rev_id, end_rev_id,
683
 
        direction, generate_merge_revisions or specific_fileid))
684
 
    if specific_fileid:
685
 
        view_revisions = _filter_revisions_touching_file_id(branch,
686
 
            specific_fileid, view_revisions,
687
 
            include_merges=generate_merge_revisions)
688
 
    return _rebase_merge_depth(view_revisions)
689
 
 
690
 
 
691
814
def _rebase_merge_depth(view_revisions):
692
815
    """Adjust depths upwards so the top level is 0."""
693
816
    # If either the first or last revision have a merge_depth of 0, we're done
737
860
    return log_rev_iterator
738
861
 
739
862
 
740
 
def _make_search_filter(branch, generate_delta, search, log_rev_iterator):
 
863
def _make_search_filter(branch, generate_delta, match, log_rev_iterator):
741
864
    """Create a filtered iterator of log_rev_iterator matching on a regex.
742
865
 
743
866
    :param branch: The branch being logged.
744
867
    :param generate_delta: Whether to generate a delta for each revision.
745
 
    :param search: A user text search string.
 
868
    :param match: A dictionary with properties as keys and lists of strings
 
869
        as values. To match, a revision may match any of the supplied strings
 
870
        within a single property but must match at least one string for each
 
871
        property.
746
872
    :param log_rev_iterator: An input iterator containing all revisions that
747
873
        could be displayed, in lists.
748
874
    :return: An iterator over lists of ((rev_id, revno, merge_depth), rev,
749
875
        delta).
750
876
    """
751
 
    if search is None:
 
877
    if match is None:
752
878
        return log_rev_iterator
753
 
    searchRE = re_compile_checked(search, re.IGNORECASE,
754
 
            'log message filter')
755
 
    return _filter_message_re(searchRE, log_rev_iterator)
756
 
 
757
 
 
758
 
def _filter_message_re(searchRE, log_rev_iterator):
 
879
    searchRE = [(k, [re.compile(x, re.IGNORECASE) for x in v])
 
880
                for (k,v) in match.iteritems()]
 
881
    return _filter_re(searchRE, log_rev_iterator)
 
882
 
 
883
 
 
884
def _filter_re(searchRE, log_rev_iterator):
759
885
    for revs in log_rev_iterator:
760
 
        new_revs = []
761
 
        for (rev_id, revno, merge_depth), rev, delta in revs:
762
 
            if searchRE.search(rev.message):
763
 
                new_revs.append(((rev_id, revno, merge_depth), rev, delta))
764
 
        yield new_revs
765
 
 
 
886
        new_revs = [rev for rev in revs if _match_filter(searchRE, rev[1])]
 
887
        if new_revs:
 
888
            yield new_revs
 
889
 
 
890
def _match_filter(searchRE, rev):
 
891
    strings = {
 
892
               'message': (rev.message,),
 
893
               'committer': (rev.committer,),
 
894
               'author': (rev.get_apparent_authors()),
 
895
               'bugs': list(rev.iter_bugs())
 
896
               }
 
897
    strings[''] = [item for inner_list in strings.itervalues()
 
898
                   for item in inner_list]
 
899
    for (k,v) in searchRE:
 
900
        if k in strings and not _match_any_filter(strings[k], v):
 
901
            return False
 
902
    return True
 
903
 
 
904
def _match_any_filter(strings, res):
 
905
    return any([filter(None, map(re.search, strings)) for re in res])
766
906
 
767
907
def _make_delta_filter(branch, generate_delta, search, log_rev_iterator,
768
908
    fileids=None, direction='reverse'):
841
981
 
842
982
def _update_fileids(delta, fileids, stop_on):
843
983
    """Update the set of file-ids to search based on file lifecycle events.
844
 
    
 
984
 
845
985
    :param fileids: a set of fileids to update
846
986
    :param stop_on: either 'add' or 'remove' - take file-ids out of the
847
987
      fileids set once their add or remove entry is detected respectively
888
1028
    :return: An iterator over lists of ((rev_id, revno, merge_depth), rev,
889
1029
        delta).
890
1030
    """
891
 
    repository = branch.repository
892
1031
    num = 9
893
1032
    for batch in log_rev_iterator:
894
1033
        batch = iter(batch)
943
1082
    if branch_revno != 0:
944
1083
        if (start_rev_id == _mod_revision.NULL_REVISION
945
1084
            or end_rev_id == _mod_revision.NULL_REVISION):
946
 
            raise errors.BzrCommandError('Logging revision 0 is invalid.')
 
1085
            raise errors.BzrCommandError(gettext('Logging revision 0 is invalid.'))
947
1086
        if start_revno > end_revno:
948
 
            raise errors.BzrCommandError("Start revision must be older than "
949
 
                                         "the end revision.")
 
1087
            raise errors.BzrCommandError(gettext("Start revision must be "
 
1088
                                         "older than the end revision."))
950
1089
    return (start_rev_id, end_rev_id)
951
1090
 
952
1091
 
1001
1140
 
1002
1141
    if ((start_rev_id == _mod_revision.NULL_REVISION)
1003
1142
        or (end_rev_id == _mod_revision.NULL_REVISION)):
1004
 
        raise errors.BzrCommandError('Logging revision 0 is invalid.')
 
1143
        raise errors.BzrCommandError(gettext('Logging revision 0 is invalid.'))
1005
1144
    if start_revno > end_revno:
1006
 
        raise errors.BzrCommandError("Start revision must be older than "
1007
 
                                     "the end revision.")
 
1145
        raise errors.BzrCommandError(gettext("Start revision must be older "
 
1146
                                     "than the end revision."))
1008
1147
 
1009
1148
    if end_revno < start_revno:
1010
1149
        return None, None, None, None
1011
1150
    cur_revno = branch_revno
1012
1151
    rev_nos = {}
1013
1152
    mainline_revs = []
1014
 
    for revision_id in branch.repository.iter_reverse_revision_history(
1015
 
                        branch_last_revision):
 
1153
    graph = branch.repository.get_graph()
 
1154
    for revision_id in graph.iter_lefthand_ancestry(
 
1155
            branch_last_revision, (_mod_revision.NULL_REVISION,)):
1016
1156
        if cur_revno < start_revno:
1017
1157
            # We have gone far enough, but we always add 1 more revision
1018
1158
            rev_nos[revision_id] = cur_revno
1032
1172
    return mainline_revs, rev_nos, start_rev_id, end_rev_id
1033
1173
 
1034
1174
 
1035
 
def _filter_revision_range(view_revisions, start_rev_id, end_rev_id):
1036
 
    """Filter view_revisions based on revision ranges.
1037
 
 
1038
 
    :param view_revisions: A list of (revision_id, dotted_revno, merge_depth)
1039
 
            tuples to be filtered.
1040
 
 
1041
 
    :param start_rev_id: If not NONE specifies the first revision to be logged.
1042
 
            If NONE then all revisions up to the end_rev_id are logged.
1043
 
 
1044
 
    :param end_rev_id: If not NONE specifies the last revision to be logged.
1045
 
            If NONE then all revisions up to the end of the log are logged.
1046
 
 
1047
 
    :return: The filtered view_revisions.
1048
 
    """
1049
 
    # This method is no longer called by the main code path.
1050
 
    # It may be removed soon. IGC 20090127
1051
 
    if start_rev_id or end_rev_id:
1052
 
        revision_ids = [r for r, n, d in view_revisions]
1053
 
        if start_rev_id:
1054
 
            start_index = revision_ids.index(start_rev_id)
1055
 
        else:
1056
 
            start_index = 0
1057
 
        if start_rev_id == end_rev_id:
1058
 
            end_index = start_index
1059
 
        else:
1060
 
            if end_rev_id:
1061
 
                end_index = revision_ids.index(end_rev_id)
1062
 
            else:
1063
 
                end_index = len(view_revisions) - 1
1064
 
        # To include the revisions merged into the last revision,
1065
 
        # extend end_rev_id down to, but not including, the next rev
1066
 
        # with the same or lesser merge_depth
1067
 
        end_merge_depth = view_revisions[end_index][2]
1068
 
        try:
1069
 
            for index in xrange(end_index+1, len(view_revisions)+1):
1070
 
                if view_revisions[index][2] <= end_merge_depth:
1071
 
                    end_index = index - 1
1072
 
                    break
1073
 
        except IndexError:
1074
 
            # if the search falls off the end then log to the end as well
1075
 
            end_index = len(view_revisions) - 1
1076
 
        view_revisions = view_revisions[start_index:end_index+1]
1077
 
    return view_revisions
1078
 
 
1079
 
 
1080
1175
def _filter_revisions_touching_file_id(branch, file_id, view_revisions,
1081
1176
    include_merges=True):
1082
1177
    r"""Return the list of revision ids which touch a given file id.
1085
1180
    This includes the revisions which directly change the file id,
1086
1181
    and the revisions which merge these changes. So if the
1087
1182
    revision graph is::
 
1183
 
1088
1184
        A-.
1089
1185
        |\ \
1090
1186
        B C E
1117
1213
    """
1118
1214
    # Lookup all possible text keys to determine which ones actually modified
1119
1215
    # the file.
 
1216
    graph = branch.repository.get_file_graph()
 
1217
    get_parent_map = graph.get_parent_map
1120
1218
    text_keys = [(file_id, rev_id) for rev_id, revno, depth in view_revisions]
1121
1219
    next_keys = None
1122
1220
    # Looking up keys in batches of 1000 can cut the time in half, as well as
1126
1224
    #       indexing layer. We might consider passing in hints as to the known
1127
1225
    #       access pattern (sparse/clustered, high success rate/low success
1128
1226
    #       rate). This particular access is clustered with a low success rate.
1129
 
    get_parent_map = branch.repository.texts.get_parent_map
1130
1227
    modified_text_revisions = set()
1131
1228
    chunk_size = 1000
1132
1229
    for start in xrange(0, len(text_keys), chunk_size):
1159
1256
    return result
1160
1257
 
1161
1258
 
1162
 
def get_view_revisions(mainline_revs, rev_nos, branch, direction,
1163
 
                       include_merges=True):
1164
 
    """Produce an iterator of revisions to show
1165
 
    :return: an iterator of (revision_id, revno, merge_depth)
1166
 
    (if there is no revno for a revision, None is supplied)
1167
 
    """
1168
 
    # This method is no longer called by the main code path.
1169
 
    # It is retained for API compatibility and may be deprecated
1170
 
    # soon. IGC 20090127
1171
 
    if not include_merges:
1172
 
        revision_ids = mainline_revs[1:]
1173
 
        if direction == 'reverse':
1174
 
            revision_ids.reverse()
1175
 
        for revision_id in revision_ids:
1176
 
            yield revision_id, str(rev_nos[revision_id]), 0
1177
 
        return
1178
 
    graph = branch.repository.get_graph()
1179
 
    # This asks for all mainline revisions, which means we only have to spider
1180
 
    # sideways, rather than depth history. That said, its still size-of-history
1181
 
    # and should be addressed.
1182
 
    # mainline_revisions always includes an extra revision at the beginning, so
1183
 
    # don't request it.
1184
 
    parent_map = dict(((key, value) for key, value in
1185
 
        graph.iter_ancestry(mainline_revs[1:]) if value is not None))
1186
 
    # filter out ghosts; merge_sort errors on ghosts.
1187
 
    rev_graph = _mod_repository._strip_NULL_ghosts(parent_map)
1188
 
    merge_sorted_revisions = tsort.merge_sort(
1189
 
        rev_graph,
1190
 
        mainline_revs[-1],
1191
 
        mainline_revs,
1192
 
        generate_revno=True)
1193
 
 
1194
 
    if direction == 'forward':
1195
 
        # forward means oldest first.
1196
 
        merge_sorted_revisions = reverse_by_depth(merge_sorted_revisions)
1197
 
    elif direction != 'reverse':
1198
 
        raise ValueError('invalid direction %r' % direction)
1199
 
 
1200
 
    for (sequence, rev_id, merge_depth, revno, end_of_merge
1201
 
         ) in merge_sorted_revisions:
1202
 
        yield rev_id, '.'.join(map(str, revno)), merge_depth
1203
 
 
1204
 
 
1205
1259
def reverse_by_depth(merge_sorted_revisions, _depth=0):
1206
1260
    """Reverse revisions by depth.
1207
1261
 
1242
1296
    """
1243
1297
 
1244
1298
    def __init__(self, rev=None, revno=None, merge_depth=0, delta=None,
1245
 
                 tags=None, diff=None):
 
1299
                 tags=None, diff=None, signature=None):
1246
1300
        self.rev = rev
1247
 
        self.revno = str(revno)
 
1301
        if revno is None:
 
1302
            self.revno = None
 
1303
        else:
 
1304
            self.revno = str(revno)
1248
1305
        self.merge_depth = merge_depth
1249
1306
        self.delta = delta
1250
1307
        self.tags = tags
1251
1308
        self.diff = diff
 
1309
        self.signature = signature
1252
1310
 
1253
1311
 
1254
1312
class LogFormatter(object):
1263
1321
    to indicate which LogRevision attributes it supports:
1264
1322
 
1265
1323
    - supports_delta must be True if this log formatter supports delta.
1266
 
        Otherwise the delta attribute may not be populated.  The 'delta_format'
1267
 
        attribute describes whether the 'short_status' format (1) or the long
1268
 
        one (2) should be used.
 
1324
      Otherwise the delta attribute may not be populated.  The 'delta_format'
 
1325
      attribute describes whether the 'short_status' format (1) or the long
 
1326
      one (2) should be used.
1269
1327
 
1270
1328
    - supports_merge_revisions must be True if this log formatter supports
1271
 
        merge revisions.  If not, then only mainline revisions will be passed
1272
 
        to the formatter.
 
1329
      merge revisions.  If not, then only mainline revisions will be passed
 
1330
      to the formatter.
1273
1331
 
1274
1332
    - preferred_levels is the number of levels this formatter defaults to.
1275
 
        The default value is zero meaning display all levels.
1276
 
        This value is only relevant if supports_merge_revisions is True.
 
1333
      The default value is zero meaning display all levels.
 
1334
      This value is only relevant if supports_merge_revisions is True.
1277
1335
 
1278
1336
    - supports_tags must be True if this log formatter supports tags.
1279
 
        Otherwise the tags attribute may not be populated.
 
1337
      Otherwise the tags attribute may not be populated.
1280
1338
 
1281
1339
    - supports_diff must be True if this log formatter supports diffs.
1282
 
        Otherwise the diff attribute may not be populated.
 
1340
      Otherwise the diff attribute may not be populated.
 
1341
 
 
1342
    - supports_signatures must be True if this log formatter supports GPG
 
1343
      signatures.
1283
1344
 
1284
1345
    Plugins can register functions to show custom revision properties using
1285
1346
    the properties_handler_registry. The registered function
1286
 
    must respect the following interface description:
 
1347
    must respect the following interface description::
 
1348
 
1287
1349
        def my_show_properties(properties_dict):
1288
1350
            # code that returns a dict {'name':'value'} of the properties
1289
1351
            # to be shown
1291
1353
    preferred_levels = 0
1292
1354
 
1293
1355
    def __init__(self, to_file, show_ids=False, show_timezone='original',
1294
 
                 delta_format=None, levels=None, show_advice=False):
 
1356
                 delta_format=None, levels=None, show_advice=False,
 
1357
                 to_exact_file=None, author_list_handler=None):
1295
1358
        """Create a LogFormatter.
1296
1359
 
1297
1360
        :param to_file: the file to output to
 
1361
        :param to_exact_file: if set, gives an output stream to which
 
1362
             non-Unicode diffs are written.
1298
1363
        :param show_ids: if True, revision-ids are to be displayed
1299
1364
        :param show_timezone: the timezone to use
1300
1365
        :param delta_format: the level of delta information to display
1303
1368
          let the log formatter decide.
1304
1369
        :param show_advice: whether to show advice at the end of the
1305
1370
          log or not
 
1371
        :param author_list_handler: callable generating a list of
 
1372
          authors to display for a given revision
1306
1373
        """
1307
1374
        self.to_file = to_file
1308
1375
        # 'exact' stream used to show diff, it should print content 'as is'
1309
1376
        # and should not try to decode/encode it to unicode to avoid bug #328007
1310
 
        self.to_exact_file = getattr(to_file, 'stream', to_file)
 
1377
        if to_exact_file is not None:
 
1378
            self.to_exact_file = to_exact_file
 
1379
        else:
 
1380
            # XXX: somewhat hacky; this assumes it's a codec writer; it's better
 
1381
            # for code that expects to get diffs to pass in the exact file
 
1382
            # stream
 
1383
            self.to_exact_file = getattr(to_file, 'stream', to_file)
1311
1384
        self.show_ids = show_ids
1312
1385
        self.show_timezone = show_timezone
1313
1386
        if delta_format is None:
1317
1390
        self.levels = levels
1318
1391
        self._show_advice = show_advice
1319
1392
        self._merge_count = 0
 
1393
        self._author_list_handler = author_list_handler
1320
1394
 
1321
1395
    def get_levels(self):
1322
1396
        """Get the number of levels to display or 0 for all."""
1341
1415
            if advice_sep:
1342
1416
                self.to_file.write(advice_sep)
1343
1417
            self.to_file.write(
1344
 
                "Use --include-merges or -n0 to see merged revisions.\n")
 
1418
                "Use --include-merged or -n0 to see merged revisions.\n")
1345
1419
 
1346
1420
    def get_advice_separator(self):
1347
1421
        """Get the text separating the log from the closing advice."""
1354
1428
        return address
1355
1429
 
1356
1430
    def short_author(self, rev):
1357
 
        name, address = config.parse_username(rev.get_apparent_authors()[0])
1358
 
        if name:
1359
 
            return name
1360
 
        return address
 
1431
        return self.authors(rev, 'first', short=True, sep=', ')
 
1432
 
 
1433
    def authors(self, rev, who, short=False, sep=None):
 
1434
        """Generate list of authors, taking --authors option into account.
 
1435
 
 
1436
        The caller has to specify the name of a author list handler,
 
1437
        as provided by the author list registry, using the ``who``
 
1438
        argument.  That name only sets a default, though: when the
 
1439
        user selected a different author list generation using the
 
1440
        ``--authors`` command line switch, as represented by the
 
1441
        ``author_list_handler`` constructor argument, that value takes
 
1442
        precedence.
 
1443
 
 
1444
        :param rev: The revision for which to generate the list of authors.
 
1445
        :param who: Name of the default handler.
 
1446
        :param short: Whether to shorten names to either name or address.
 
1447
        :param sep: What separator to use for automatic concatenation.
 
1448
        """
 
1449
        if self._author_list_handler is not None:
 
1450
            # The user did specify --authors, which overrides the default
 
1451
            author_list_handler = self._author_list_handler
 
1452
        else:
 
1453
            # The user didn't specify --authors, so we use the caller's default
 
1454
            author_list_handler = author_list_registry.get(who)
 
1455
        names = author_list_handler(rev)
 
1456
        if short:
 
1457
            for i in range(len(names)):
 
1458
                name, address = config.parse_username(names[i])
 
1459
                if name:
 
1460
                    names[i] = name
 
1461
                else:
 
1462
                    names[i] = address
 
1463
        if sep is not None:
 
1464
            names = sep.join(names)
 
1465
        return names
1361
1466
 
1362
1467
    def merge_marker(self, revision):
1363
1468
        """Get the merge marker to include in the output or '' if none."""
1367
1472
        else:
1368
1473
            return ''
1369
1474
 
1370
 
    def show_foreign_info(self, rev, indent):
 
1475
    def show_properties(self, revision, indent):
 
1476
        """Displays the custom properties returned by each registered handler.
 
1477
 
 
1478
        If a registered handler raises an error it is propagated.
 
1479
        """
 
1480
        for line in self.custom_properties(revision):
 
1481
            self.to_file.write("%s%s\n" % (indent, line))
 
1482
 
 
1483
    def custom_properties(self, revision):
 
1484
        """Format the custom properties returned by each registered handler.
 
1485
 
 
1486
        If a registered handler raises an error it is propagated.
 
1487
 
 
1488
        :return: a list of formatted lines (excluding trailing newlines)
 
1489
        """
 
1490
        lines = self._foreign_info_properties(revision)
 
1491
        for key, handler in properties_handler_registry.iteritems():
 
1492
            lines.extend(self._format_properties(handler(revision)))
 
1493
        return lines
 
1494
 
 
1495
    def _foreign_info_properties(self, rev):
1371
1496
        """Custom log displayer for foreign revision identifiers.
1372
1497
 
1373
1498
        :param rev: Revision object.
1374
1499
        """
1375
1500
        # Revision comes directly from a foreign repository
1376
1501
        if isinstance(rev, foreign.ForeignRevision):
1377
 
            self._write_properties(indent, rev.mapping.vcs.show_foreign_revid(
1378
 
                rev.foreign_revid))
1379
 
            return
 
1502
            return self._format_properties(
 
1503
                rev.mapping.vcs.show_foreign_revid(rev.foreign_revid))
1380
1504
 
1381
1505
        # Imported foreign revision revision ids always contain :
1382
1506
        if not ":" in rev.revision_id:
1383
 
            return
 
1507
            return []
1384
1508
 
1385
1509
        # Revision was once imported from a foreign repository
1386
1510
        try:
1387
1511
            foreign_revid, mapping = \
1388
1512
                foreign.foreign_vcs_registry.parse_revision_id(rev.revision_id)
1389
1513
        except errors.InvalidRevisionId:
1390
 
            return
 
1514
            return []
1391
1515
 
1392
 
        self._write_properties(indent, 
 
1516
        return self._format_properties(
1393
1517
            mapping.vcs.show_foreign_revid(foreign_revid))
1394
1518
 
1395
 
    def show_properties(self, revision, indent):
1396
 
        """Displays the custom properties returned by each registered handler.
1397
 
 
1398
 
        If a registered handler raises an error it is propagated.
1399
 
        """
1400
 
        for key, handler in properties_handler_registry.iteritems():
1401
 
            self._write_properties(indent, handler(revision))
1402
 
 
1403
 
    def _write_properties(self, indent, properties):
 
1519
    def _format_properties(self, properties):
 
1520
        lines = []
1404
1521
        for key, value in properties.items():
1405
 
            self.to_file.write(indent + key + ': ' + value + '\n')
 
1522
            lines.append(key + ': ' + value)
 
1523
        return lines
1406
1524
 
1407
1525
    def show_diff(self, to_file, diff, indent):
1408
1526
        for l in diff.rstrip().split('\n'):
1409
1527
            to_file.write(indent + '%s\n' % (l,))
1410
1528
 
1411
1529
 
 
1530
# Separator between revisions in long format
 
1531
_LONG_SEP = '-' * 60
 
1532
 
 
1533
 
1412
1534
class LongLogFormatter(LogFormatter):
1413
1535
 
1414
1536
    supports_merge_revisions = True
1416
1538
    supports_delta = True
1417
1539
    supports_tags = True
1418
1540
    supports_diff = True
 
1541
    supports_signatures = True
 
1542
 
 
1543
    def __init__(self, *args, **kwargs):
 
1544
        super(LongLogFormatter, self).__init__(*args, **kwargs)
 
1545
        if self.show_timezone == 'original':
 
1546
            self.date_string = self._date_string_original_timezone
 
1547
        else:
 
1548
            self.date_string = self._date_string_with_timezone
 
1549
 
 
1550
    def _date_string_with_timezone(self, rev):
 
1551
        return format_date(rev.timestamp, rev.timezone or 0,
 
1552
                           self.show_timezone)
 
1553
 
 
1554
    def _date_string_original_timezone(self, rev):
 
1555
        return format_date_with_offset_in_original_timezone(rev.timestamp,
 
1556
            rev.timezone or 0)
1419
1557
 
1420
1558
    def log_revision(self, revision):
1421
1559
        """Log a revision, either merged or not."""
1422
1560
        indent = '    ' * revision.merge_depth
1423
 
        to_file = self.to_file
1424
 
        to_file.write(indent + '-' * 60 + '\n')
 
1561
        lines = [_LONG_SEP]
1425
1562
        if revision.revno is not None:
1426
 
            to_file.write(indent + 'revno: %s%s\n' % (revision.revno,
 
1563
            lines.append('revno: %s%s' % (revision.revno,
1427
1564
                self.merge_marker(revision)))
1428
1565
        if revision.tags:
1429
 
            to_file.write(indent + 'tags: %s\n' % (', '.join(revision.tags)))
 
1566
            lines.append('tags: %s' % (', '.join(revision.tags)))
 
1567
        if self.show_ids or revision.revno is None:
 
1568
            lines.append('revision-id: %s' % (revision.rev.revision_id,))
1430
1569
        if self.show_ids:
1431
 
            to_file.write(indent + 'revision-id: ' + revision.rev.revision_id)
1432
 
            to_file.write('\n')
1433
1570
            for parent_id in revision.rev.parent_ids:
1434
 
                to_file.write(indent + 'parent: %s\n' % (parent_id,))
1435
 
        self.show_foreign_info(revision.rev, indent)
1436
 
        self.show_properties(revision.rev, indent)
 
1571
                lines.append('parent: %s' % (parent_id,))
 
1572
        lines.extend(self.custom_properties(revision.rev))
1437
1573
 
1438
1574
        committer = revision.rev.committer
1439
 
        authors = revision.rev.get_apparent_authors()
 
1575
        authors = self.authors(revision.rev, 'all')
1440
1576
        if authors != [committer]:
1441
 
            to_file.write(indent + 'author: %s\n' % (", ".join(authors),))
1442
 
        to_file.write(indent + 'committer: %s\n' % (committer,))
 
1577
            lines.append('author: %s' % (", ".join(authors),))
 
1578
        lines.append('committer: %s' % (committer,))
1443
1579
 
1444
1580
        branch_nick = revision.rev.properties.get('branch-nick', None)
1445
1581
        if branch_nick is not None:
1446
 
            to_file.write(indent + 'branch nick: %s\n' % (branch_nick,))
1447
 
 
1448
 
        date_str = format_date(revision.rev.timestamp,
1449
 
                               revision.rev.timezone or 0,
1450
 
                               self.show_timezone)
1451
 
        to_file.write(indent + 'timestamp: %s\n' % (date_str,))
1452
 
 
1453
 
        to_file.write(indent + 'message:\n')
 
1582
            lines.append('branch nick: %s' % (branch_nick,))
 
1583
 
 
1584
        lines.append('timestamp: %s' % (self.date_string(revision.rev),))
 
1585
 
 
1586
        if revision.signature is not None:
 
1587
            lines.append('signature: ' + revision.signature)
 
1588
 
 
1589
        lines.append('message:')
1454
1590
        if not revision.rev.message:
1455
 
            to_file.write(indent + '  (no message)\n')
 
1591
            lines.append('  (no message)')
1456
1592
        else:
1457
1593
            message = revision.rev.message.rstrip('\r\n')
1458
1594
            for l in message.split('\n'):
1459
 
                to_file.write(indent + '  %s\n' % (l,))
 
1595
                lines.append('  %s' % (l,))
 
1596
 
 
1597
        # Dump the output, appending the delta and diff if requested
 
1598
        to_file = self.to_file
 
1599
        to_file.write("%s%s\n" % (indent, ('\n' + indent).join(lines)))
1460
1600
        if revision.delta is not None:
1461
 
            # We don't respect delta_format for compatibility
1462
 
            revision.delta.show(to_file, self.show_ids, indent=indent,
1463
 
                                short_status=False)
 
1601
            # Use the standard status output to display changes
 
1602
            from bzrlib.delta import report_delta
 
1603
            report_delta(to_file, revision.delta, short_status=False,
 
1604
                         show_ids=self.show_ids, indent=indent)
1464
1605
        if revision.diff is not None:
1465
1606
            to_file.write(indent + 'diff:\n')
 
1607
            to_file.flush()
1466
1608
            # Note: we explicitly don't indent the diff (relative to the
1467
1609
            # revision information) so that the output can be fed to patch -p0
1468
1610
            self.show_diff(self.to_exact_file, revision.diff, indent)
 
1611
            self.to_exact_file.flush()
1469
1612
 
1470
1613
    def get_advice_separator(self):
1471
1614
        """Get the text separating the log from the closing advice."""
1495
1638
        indent = '    ' * depth
1496
1639
        revno_width = self.revno_width_by_depth.get(depth)
1497
1640
        if revno_width is None:
1498
 
            if revision.revno.find('.') == -1:
 
1641
            if revision.revno is None or revision.revno.find('.') == -1:
1499
1642
                # mainline revno, e.g. 12345
1500
1643
                revno_width = 5
1501
1644
            else:
1509
1652
        if revision.tags:
1510
1653
            tags = ' {%s}' % (', '.join(revision.tags))
1511
1654
        to_file.write(indent + "%*s %s\t%s%s%s\n" % (revno_width,
1512
 
                revision.revno, self.short_author(revision.rev),
 
1655
                revision.revno or "", self.short_author(revision.rev),
1513
1656
                format_date(revision.rev.timestamp,
1514
1657
                            revision.rev.timezone or 0,
1515
1658
                            self.show_timezone, date_fmt="%Y-%m-%d",
1516
1659
                            show_offset=False),
1517
1660
                tags, self.merge_marker(revision)))
1518
 
        self.show_foreign_info(revision.rev, indent+offset)
1519
1661
        self.show_properties(revision.rev, indent+offset)
1520
 
        if self.show_ids:
 
1662
        if self.show_ids or revision.revno is None:
1521
1663
            to_file.write(indent + offset + 'revision-id:%s\n'
1522
1664
                          % (revision.rev.revision_id,))
1523
1665
        if not revision.rev.message:
1528
1670
                to_file.write(indent + offset + '%s\n' % (l,))
1529
1671
 
1530
1672
        if revision.delta is not None:
1531
 
            revision.delta.show(to_file, self.show_ids, indent=indent + offset,
1532
 
                                short_status=self.delta_format==1)
 
1673
            # Use the standard status output to display changes
 
1674
            from bzrlib.delta import report_delta
 
1675
            report_delta(to_file, revision.delta,
 
1676
                         short_status=self.delta_format==1,
 
1677
                         show_ids=self.show_ids, indent=indent + offset)
1533
1678
        if revision.diff is not None:
1534
1679
            self.show_diff(self.to_exact_file, revision.diff, '      ')
1535
1680
        to_file.write('\n')
1543
1688
 
1544
1689
    def __init__(self, *args, **kwargs):
1545
1690
        super(LineLogFormatter, self).__init__(*args, **kwargs)
1546
 
        self._max_chars = terminal_width() - 1
 
1691
        width = terminal_width()
 
1692
        if width is not None:
 
1693
            # we need one extra space for terminals that wrap on last char
 
1694
            width = width - 1
 
1695
        self._max_chars = width
1547
1696
 
1548
1697
    def truncate(self, str, max_len):
1549
 
        if len(str) <= max_len:
 
1698
        if max_len is None or len(str) <= max_len:
1550
1699
            return str
1551
 
        return str[:max_len-3]+'...'
 
1700
        return str[:max_len-3] + '...'
1552
1701
 
1553
1702
    def date_string(self, rev):
1554
1703
        return format_date(rev.timestamp, rev.timezone or 0,
1569
1718
 
1570
1719
    def log_string(self, revno, rev, max_chars, tags=None, prefix=''):
1571
1720
        """Format log info into one string. Truncate tail of string
1572
 
        :param  revno:      revision number or None.
1573
 
                            Revision numbers counts from 1.
1574
 
        :param  rev:        revision object
1575
 
        :param  max_chars:  maximum length of resulting string
1576
 
        :param  tags:       list of tags or None
1577
 
        :param  prefix:     string to prefix each line
1578
 
        :return:            formatted truncated string
 
1721
 
 
1722
        :param revno:      revision number or None.
 
1723
                           Revision numbers counts from 1.
 
1724
        :param rev:        revision object
 
1725
        :param max_chars:  maximum length of resulting string
 
1726
        :param tags:       list of tags or None
 
1727
        :param prefix:     string to prefix each line
 
1728
        :return:           formatted truncated string
1579
1729
        """
1580
1730
        out = []
1581
1731
        if revno:
1582
1732
            # show revno only when is not None
1583
1733
            out.append("%s:" % revno)
1584
 
        out.append(self.truncate(self.short_author(rev), 20))
 
1734
        if max_chars is not None:
 
1735
            out.append(self.truncate(self.short_author(rev), (max_chars+3)/4))
 
1736
        else:
 
1737
            out.append(self.short_author(rev))
1585
1738
        out.append(self.date_string(rev))
1586
1739
        if len(rev.parent_ids) > 1:
1587
1740
            out.append('[merge]')
1606
1759
                               self.show_timezone,
1607
1760
                               date_fmt='%Y-%m-%d',
1608
1761
                               show_offset=False)
1609
 
        committer_str = revision.rev.committer.replace (' <', '  <')
 
1762
        committer_str = self.authors(revision.rev, 'first', sep=', ')
 
1763
        committer_str = committer_str.replace(' <', '  <')
1610
1764
        to_file.write('%s  %s\n\n' % (date_str,committer_str))
1611
1765
 
1612
1766
        if revision.delta is not None and revision.delta.has_changed():
1645
1799
        return self.get(name)(*args, **kwargs)
1646
1800
 
1647
1801
    def get_default(self, branch):
1648
 
        return self.get(branch.get_config().log_format())
 
1802
        c = branch.get_config_stack()
 
1803
        return self.get(c.get('log_format'))
1649
1804
 
1650
1805
 
1651
1806
log_formatter_registry = LogFormatterRegistry()
1674
1829
    try:
1675
1830
        return log_formatter_registry.make_formatter(name, *args, **kwargs)
1676
1831
    except KeyError:
1677
 
        raise errors.BzrCommandError("unknown log formatter: %r" % name)
1678
 
 
1679
 
 
1680
 
def show_one_log(revno, rev, delta, verbose, to_file, show_timezone):
1681
 
    # deprecated; for compatibility
1682
 
    lf = LongLogFormatter(to_file=to_file, show_timezone=show_timezone)
1683
 
    lf.show(revno, rev, delta)
 
1832
        raise errors.BzrCommandError(gettext("unknown log formatter: %r") % name)
 
1833
 
 
1834
 
 
1835
def author_list_all(rev):
 
1836
    return rev.get_apparent_authors()[:]
 
1837
 
 
1838
 
 
1839
def author_list_first(rev):
 
1840
    lst = rev.get_apparent_authors()
 
1841
    try:
 
1842
        return [lst[0]]
 
1843
    except IndexError:
 
1844
        return []
 
1845
 
 
1846
 
 
1847
def author_list_committer(rev):
 
1848
    return [rev.committer]
 
1849
 
 
1850
 
 
1851
author_list_registry = registry.Registry()
 
1852
 
 
1853
author_list_registry.register('all', author_list_all,
 
1854
                              'All authors')
 
1855
 
 
1856
author_list_registry.register('first', author_list_first,
 
1857
                              'The first author')
 
1858
 
 
1859
author_list_registry.register('committer', author_list_committer,
 
1860
                              'The committer')
1684
1861
 
1685
1862
 
1686
1863
def show_changed_revisions(branch, old_rh, new_rh, to_file=None,
1751
1928
    old_revisions = set()
1752
1929
    new_history = []
1753
1930
    new_revisions = set()
1754
 
    new_iter = repository.iter_reverse_revision_history(new_revision_id)
1755
 
    old_iter = repository.iter_reverse_revision_history(old_revision_id)
 
1931
    graph = repository.get_graph()
 
1932
    new_iter = graph.iter_lefthand_ancestry(new_revision_id)
 
1933
    old_iter = graph.iter_lefthand_ancestry(old_revision_id)
1756
1934
    stop_revision = None
1757
1935
    do_old = True
1758
1936
    do_new = True
1833
2011
        lf.log_revision(lr)
1834
2012
 
1835
2013
 
1836
 
def _get_info_for_log_files(revisionspec_list, file_list):
 
2014
def _get_info_for_log_files(revisionspec_list, file_list, add_cleanup):
1837
2015
    """Find file-ids and kinds given a list of files and a revision range.
1838
2016
 
1839
2017
    We search for files at the end of the range. If not found there,
1843
2021
    :param file_list: the list of paths given on the command line;
1844
2022
      the first of these can be a branch location or a file path,
1845
2023
      the remainder must be file paths
 
2024
    :param add_cleanup: When the branch returned is read locked,
 
2025
      an unlock call will be queued to the cleanup.
1846
2026
    :return: (branch, info_list, start_rev_info, end_rev_info) where
1847
2027
      info_list is a list of (relative_path, file_id, kind) tuples where
1848
2028
      kind is one of values 'directory', 'file', 'symlink', 'tree-reference'.
 
2029
      branch will be read-locked.
1849
2030
    """
1850
 
    from builtins import _get_revision_range, safe_relpath_files
1851
 
    tree, b, path = bzrdir.BzrDir.open_containing_tree_or_branch(file_list[0])
 
2031
    from builtins import _get_revision_range
 
2032
    tree, b, path = controldir.ControlDir.open_containing_tree_or_branch(
 
2033
        file_list[0])
 
2034
    add_cleanup(b.lock_read().unlock)
1852
2035
    # XXX: It's damn messy converting a list of paths to relative paths when
1853
2036
    # those paths might be deleted ones, they might be on a case-insensitive
1854
2037
    # filesystem and/or they might be in silly locations (like another branch).
1858
2041
    # case of running log in a nested directory, assuming paths beyond the
1859
2042
    # first one haven't been deleted ...
1860
2043
    if tree:
1861
 
        relpaths = [path] + safe_relpath_files(tree, file_list[1:])
 
2044
        relpaths = [path] + tree.safe_relpath_files(file_list[1:])
1862
2045
    else:
1863
2046
        relpaths = [path] + file_list[1:]
1864
2047
    info_list = []
1933
2116
 
1934
2117
properties_handler_registry = registry.Registry()
1935
2118
 
 
2119
# Use the properties handlers to print out bug information if available
 
2120
def _bugs_properties_handler(revision):
 
2121
    if revision.properties.has_key('bugs'):
 
2122
        bug_lines = revision.properties['bugs'].split('\n')
 
2123
        bug_rows = [line.split(' ', 1) for line in bug_lines]
 
2124
        fixed_bug_urls = [row[0] for row in bug_rows if
 
2125
                          len(row) > 1 and row[1] == 'fixed']
 
2126
 
 
2127
        if fixed_bug_urls:
 
2128
            return {ngettext('fixes bug', 'fixes bugs', len(fixed_bug_urls)):\
 
2129
                    ' '.join(fixed_bug_urls)}
 
2130
    return {}
 
2131
 
 
2132
properties_handler_registry.register('bugs_properties_handler',
 
2133
                                     _bugs_properties_handler)
 
2134
 
1936
2135
 
1937
2136
# adapters which revision ids to log are filtered. When log is called, the
1938
2137
# log_rev_iterator is adapted through each of these factory methods.