~bzr-pqm/bzr/bzr.dev

« back to all changes in this revision

Viewing changes to bzrlib/revisionspec.py

  • Committer: Andrew Bennetts
  • Date: 2010-10-08 08:15:14 UTC
  • mto: This revision was merged to the branch mainline in revision 5498.
  • Revision ID: andrew.bennetts@canonical.com-20101008081514-dviqzrdfwyzsqbz2
Split NEWS into per-release doc/en/release-notes/bzr-*.txt

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
# Copyright (C) 2005, 2006, 2007 Canonical Ltd
 
1
# Copyright (C) 2005-2010 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
12
12
#
13
13
# You should have received a copy of the GNU General Public License
14
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
16
 
 
17
 
 
 
15
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
 
16
 
 
17
 
 
18
import re
 
19
 
 
20
from bzrlib.lazy_import import lazy_import
 
21
lazy_import(globals(), """
18
22
import bisect
19
23
import datetime
20
 
import re
 
24
""")
21
25
 
22
26
from bzrlib import (
 
27
    branch as _mod_branch,
23
28
    errors,
24
29
    osutils,
 
30
    registry,
25
31
    revision,
26
32
    symbol_versioning,
27
33
    trace,
28
 
    tsort,
 
34
    workingtree,
29
35
    )
30
36
 
31
37
 
100
106
 
101
107
        Use this if you don't know or care what the revno is.
102
108
        """
 
109
        if revision_id == revision.NULL_REVISION:
 
110
            return RevisionInfo(branch, 0, revision_id)
103
111
        try:
104
112
            revno = revs.index(revision_id) + 1
105
113
        except ValueError:
107
115
        return RevisionInfo(branch, revno, revision_id)
108
116
 
109
117
 
110
 
# classes in this list should have a "prefix" attribute, against which
111
 
# string specs are matched
112
 
SPEC_TYPES = []
113
118
_revno_regex = None
114
119
 
115
120
 
118
123
 
119
124
    help_txt = """A parsed revision specification.
120
125
 
121
 
    A revision specification can be an integer, in which case it is
122
 
    assumed to be a revno (though this will translate negative values
123
 
    into positive ones); or it can be a string, in which case it is
124
 
    parsed for something like 'date:' or 'revid:' etc.
 
126
    A revision specification is a string, which may be unambiguous about
 
127
    what it represents by giving a prefix like 'date:' or 'revid:' etc,
 
128
    or it may have no prefix, in which case it's tried against several
 
129
    specifier types in sequence to determine what the user meant.
125
130
 
126
131
    Revision specs are an UI element, and they have been moved out
127
132
    of the branch class to leave "back-end" classes unaware of such
133
138
    """
134
139
 
135
140
    prefix = None
136
 
 
137
 
    def __new__(cls, spec, _internal=False):
138
 
        if _internal:
139
 
            return object.__new__(cls, spec, _internal=_internal)
140
 
 
141
 
        symbol_versioning.warn('Creating a RevisionSpec directly has'
142
 
                               ' been deprecated in version 0.11. Use'
143
 
                               ' RevisionSpec.from_string()'
144
 
                               ' instead.',
145
 
                               DeprecationWarning, stacklevel=2)
146
 
        return RevisionSpec.from_string(spec)
 
141
    wants_revision_history = True
 
142
    dwim_catchable_exceptions = (errors.InvalidRevisionSpec,)
 
143
    """Exceptions that RevisionSpec_dwim._match_on will catch.
 
144
 
 
145
    If the revspec is part of ``dwim_revspecs``, it may be tried with an
 
146
    invalid revspec and raises some exception. The exceptions mentioned here
 
147
    will not be reported to the user but simply ignored without stopping the
 
148
    dwim processing.
 
149
    """
147
150
 
148
151
    @staticmethod
149
152
    def from_string(spec):
158
161
 
159
162
        if spec is None:
160
163
            return RevisionSpec(None, _internal=True)
161
 
 
162
 
        assert isinstance(spec, basestring), \
163
 
            "You should only supply strings not %s" % (type(spec),)
164
 
 
165
 
        for spectype in SPEC_TYPES:
166
 
            if spec.startswith(spectype.prefix):
167
 
                trace.mutter('Returning RevisionSpec %s for %s',
168
 
                             spectype.__name__, spec)
169
 
                return spectype(spec, _internal=True)
 
164
        match = revspec_registry.get_prefix(spec)
 
165
        if match is not None:
 
166
            spectype, specsuffix = match
 
167
            trace.mutter('Returning RevisionSpec %s for %s',
 
168
                         spectype.__name__, spec)
 
169
            return spectype(spec, _internal=True)
170
170
        else:
171
 
            # RevisionSpec_revno is special cased, because it is the only
172
 
            # one that directly handles plain integers
173
 
            # TODO: This should not be special cased rather it should be
174
 
            # a method invocation on spectype.canparse()
175
 
            global _revno_regex
176
 
            if _revno_regex is None:
177
 
                _revno_regex = re.compile(r'^(?:(\d+(\.\d+)*)|-\d+)(:.*)?$')
178
 
            if _revno_regex.match(spec) is not None:
179
 
                return RevisionSpec_revno(spec, _internal=True)
180
 
 
181
 
            raise errors.NoSuchRevisionSpec(spec)
 
171
            for spectype in SPEC_TYPES:
 
172
                if spec.startswith(spectype.prefix):
 
173
                    trace.mutter('Returning RevisionSpec %s for %s',
 
174
                                 spectype.__name__, spec)
 
175
                    return spectype(spec, _internal=True)
 
176
            # Otherwise treat it as a DWIM, build the RevisionSpec object and
 
177
            # wait for _match_on to be called.
 
178
            return RevisionSpec_dwim(spec, _internal=True)
182
179
 
183
180
    def __init__(self, spec, _internal=False):
184
181
        """Create a RevisionSpec referring to the Null revision.
188
185
            called directly. Only from RevisionSpec.from_string()
189
186
        """
190
187
        if not _internal:
191
 
            # XXX: Update this after 0.10 is released
192
188
            symbol_versioning.warn('Creating a RevisionSpec directly has'
193
189
                                   ' been deprecated in version 0.11. Use'
194
190
                                   ' RevisionSpec.from_string()'
201
197
 
202
198
    def _match_on(self, branch, revs):
203
199
        trace.mutter('Returning RevisionSpec._match_on: None')
204
 
        return RevisionInfo(branch, 0, None)
 
200
        return RevisionInfo(branch, None, None)
205
201
 
206
202
    def _match_on_and_check(self, branch, revs):
207
203
        info = self._match_on(branch, revs)
208
204
        if info:
209
205
            return info
210
 
        elif info == (0, None):
211
 
            # special case - the empty tree
 
206
        elif info == (None, None):
 
207
            # special case - nothing supplied
212
208
            return info
213
209
        elif self.prefix:
214
210
            raise errors.InvalidRevisionSpec(self.user_spec, branch)
217
213
 
218
214
    def in_history(self, branch):
219
215
        if branch:
220
 
            revs = branch.revision_history()
 
216
            if self.wants_revision_history:
 
217
                revs = branch.revision_history()
 
218
            else:
 
219
                revs = None
221
220
        else:
222
221
            # this should never trigger.
223
222
            # TODO: make it a deprecated code path. RBC 20060928
233
232
    # will do what you expect.
234
233
    in_store = in_history
235
234
    in_branch = in_store
236
 
        
 
235
 
 
236
    def as_revision_id(self, context_branch):
 
237
        """Return just the revision_id for this revisions spec.
 
238
 
 
239
        Some revision specs require a context_branch to be able to determine
 
240
        their value. Not all specs will make use of it.
 
241
        """
 
242
        return self._as_revision_id(context_branch)
 
243
 
 
244
    def _as_revision_id(self, context_branch):
 
245
        """Implementation of as_revision_id()
 
246
 
 
247
        Classes should override this function to provide appropriate
 
248
        functionality. The default is to just call '.in_history().rev_id'
 
249
        """
 
250
        return self.in_history(context_branch).rev_id
 
251
 
 
252
    def as_tree(self, context_branch):
 
253
        """Return the tree object for this revisions spec.
 
254
 
 
255
        Some revision specs require a context_branch to be able to determine
 
256
        the revision id and access the repository. Not all specs will make
 
257
        use of it.
 
258
        """
 
259
        return self._as_tree(context_branch)
 
260
 
 
261
    def _as_tree(self, context_branch):
 
262
        """Implementation of as_tree().
 
263
 
 
264
        Classes should override this function to provide appropriate
 
265
        functionality. The default is to just call '.as_revision_id()'
 
266
        and get the revision tree from context_branch's repository.
 
267
        """
 
268
        revision_id = self.as_revision_id(context_branch)
 
269
        return context_branch.repository.revision_tree(revision_id)
 
270
 
237
271
    def __repr__(self):
238
272
        # this is mostly for helping with testing
239
273
        return '<%s %s>' % (self.__class__.__name__,
240
274
                              self.user_spec)
241
 
    
 
275
 
242
276
    def needs_branch(self):
243
277
        """Whether this revision spec needs a branch.
244
278
 
248
282
 
249
283
    def get_branch(self):
250
284
        """When the revision specifier contains a branch location, return it.
251
 
        
 
285
 
252
286
        Otherwise, return None.
253
287
        """
254
288
        return None
256
290
 
257
291
# private API
258
292
 
 
293
class RevisionSpec_dwim(RevisionSpec):
 
294
    """Provides a DWIMish revision specifier lookup.
 
295
 
 
296
    Note that this does not go in the revspec_registry because by definition
 
297
    there is no prefix to identify it.  It's solely called from
 
298
    RevisionSpec.from_string() because the DWIMification happen when _match_on
 
299
    is called so the string describing the revision is kept here until needed.
 
300
    """
 
301
 
 
302
    help_txt = None
 
303
    # We don't need to build the revision history ourself, that's delegated to
 
304
    # each revspec we try.
 
305
    wants_revision_history = False
 
306
 
 
307
    def _try_spectype(self, rstype, branch):
 
308
        rs = rstype(self.spec, _internal=True)
 
309
        # Hit in_history to find out if it exists, or we need to try the
 
310
        # next type.
 
311
        return rs.in_history(branch)
 
312
 
 
313
    def _match_on(self, branch, revs):
 
314
        """Run the lookup and see what we can get."""
 
315
 
 
316
        # First, see if it's a revno
 
317
        global _revno_regex
 
318
        if _revno_regex is None:
 
319
            _revno_regex = re.compile(r'^(?:(\d+(\.\d+)*)|-\d+)(:.*)?$')
 
320
        if _revno_regex.match(self.spec) is not None:
 
321
            try:
 
322
                return self._try_spectype(RevisionSpec_revno, branch)
 
323
            except RevisionSpec_revno.dwim_catchable_exceptions:
 
324
                pass
 
325
 
 
326
        # Next see what has been registered
 
327
        for rs_class in dwim_revspecs:
 
328
            try:
 
329
                return self._try_spectype(rs_class, branch)
 
330
            except rs_class.dwim_catchable_exceptions:
 
331
                pass
 
332
 
 
333
        # Well, I dunno what it is. Note that we don't try to keep track of the
 
334
        # first of last exception raised during the DWIM tries as none seems
 
335
        # really relevant.
 
336
        raise errors.InvalidRevisionSpec(self.spec, branch)
 
337
 
 
338
 
259
339
class RevisionSpec_revno(RevisionSpec):
260
340
    """Selects a revision using a number."""
261
341
 
262
342
    help_txt = """Selects a revision using a number.
263
343
 
264
344
    Use an integer to specify a revision in the history of the branch.
265
 
    Optionally a branch can be specified. The 'revno:' prefix is optional.
266
 
    A negative number will count from the end of the branch (-1 is the
267
 
    last revision, -2 the previous one). If the negative number is larger
268
 
    than the branch's history, the first revision is returned.
 
345
    Optionally a branch can be specified.  A negative number will count
 
346
    from the end of the branch (-1 is the last revision, -2 the previous
 
347
    one). If the negative number is larger than the branch's history, the
 
348
    first revision is returned.
269
349
    Examples::
270
350
 
271
 
      revno:1                   -> return the first revision
 
351
      revno:1                   -> return the first revision of this branch
272
352
      revno:3:/path/to/branch   -> return the 3rd revision of
273
353
                                   the branch '/path/to/branch'
274
354
      revno:-1                  -> The last revision in a branch.
278
358
                                   your history is very long.
279
359
    """
280
360
    prefix = 'revno:'
 
361
    wants_revision_history = False
281
362
 
282
363
    def _match_on(self, branch, revs):
283
364
        """Lookup a revision by revision number"""
 
365
        branch, revno, revision_id = self._lookup(branch, revs)
 
366
        return RevisionInfo(branch, revno, revision_id)
 
367
 
 
368
    def _lookup(self, branch, revs_or_none):
284
369
        loc = self.spec.find(':')
285
370
        if loc == -1:
286
371
            revno_spec = self.spec
300
385
                dotted = False
301
386
            except ValueError:
302
387
                # dotted decimal. This arguably should not be here
303
 
                # but the from_string method is a little primitive 
 
388
                # but the from_string method is a little primitive
304
389
                # right now - RBC 20060928
305
390
                try:
306
391
                    match_revno = tuple((int(number) for number in revno_spec.split('.')))
315
400
            # the branch object.
316
401
            from bzrlib.branch import Branch
317
402
            branch = Branch.open(branch_spec)
318
 
            # Need to use a new revision history
319
 
            # because we are using a specific branch
320
 
            revs = branch.revision_history()
 
403
            revs_or_none = None
321
404
 
322
405
        if dotted:
323
 
            branch.lock_read()
324
406
            try:
325
 
                revision_id_to_revno = branch.get_revision_id_to_revno_map()
326
 
                revisions = [revision_id for revision_id, revno
327
 
                             in revision_id_to_revno.iteritems()
328
 
                             if revno == match_revno]
329
 
            finally:
330
 
                branch.unlock()
331
 
            if len(revisions) != 1:
332
 
                return RevisionInfo(branch, None, None)
 
407
                revision_id = branch.dotted_revno_to_revision_id(match_revno,
 
408
                    _cache_reverse=True)
 
409
            except errors.NoSuchRevision:
 
410
                raise errors.InvalidRevisionSpec(self.user_spec, branch)
333
411
            else:
334
412
                # there is no traditional 'revno' for dotted-decimal revnos.
335
413
                # so for  API compatability we return None.
336
 
                return RevisionInfo(branch, None, revisions[0])
 
414
                return branch, None, revision_id
337
415
        else:
 
416
            last_revno, last_revision_id = branch.last_revision_info()
338
417
            if revno < 0:
339
418
                # if get_rev_id supported negative revnos, there would not be a
340
419
                # need for this special case.
341
 
                if (-revno) >= len(revs):
 
420
                if (-revno) >= last_revno:
342
421
                    revno = 1
343
422
                else:
344
 
                    revno = len(revs) + revno + 1
 
423
                    revno = last_revno + revno + 1
345
424
            try:
346
 
                revision_id = branch.get_rev_id(revno, revs)
 
425
                revision_id = branch.get_rev_id(revno, revs_or_none)
347
426
            except errors.NoSuchRevision:
348
427
                raise errors.InvalidRevisionSpec(self.user_spec, branch)
349
 
        return RevisionInfo(branch, revno, revision_id)
350
 
        
 
428
        return branch, revno, revision_id
 
429
 
 
430
    def _as_revision_id(self, context_branch):
 
431
        # We would have the revno here, but we don't really care
 
432
        branch, revno, revision_id = self._lookup(context_branch, None)
 
433
        return revision_id
 
434
 
351
435
    def needs_branch(self):
352
436
        return self.spec.find(':') == -1
353
437
 
357
441
        else:
358
442
            return self.spec[self.spec.find(':')+1:]
359
443
 
360
 
# Old compatibility 
 
444
# Old compatibility
361
445
RevisionSpec_int = RevisionSpec_revno
362
446
 
363
 
SPEC_TYPES.append(RevisionSpec_revno)
364
 
 
365
 
 
366
 
class RevisionSpec_revid(RevisionSpec):
 
447
 
 
448
 
 
449
class RevisionIDSpec(RevisionSpec):
 
450
 
 
451
    def _match_on(self, branch, revs):
 
452
        revision_id = self.as_revision_id(branch)
 
453
        return RevisionInfo.from_revision_id(branch, revision_id, revs)
 
454
 
 
455
 
 
456
class RevisionSpec_revid(RevisionIDSpec):
367
457
    """Selects a revision using the revision id."""
368
458
 
369
459
    help_txt = """Selects a revision using the revision id.
370
460
 
371
461
    Supply a specific revision id, that can be used to specify any
372
 
    revision id in the ancestry of the branch. 
 
462
    revision id in the ancestry of the branch.
373
463
    Including merges, and pending merges.
374
464
    Examples::
375
465
 
376
466
      revid:aaaa@bbbb-123456789 -> Select revision 'aaaa@bbbb-123456789'
377
 
    """    
 
467
    """
 
468
 
378
469
    prefix = 'revid:'
379
470
 
380
 
    def _match_on(self, branch, revs):
 
471
    def _as_revision_id(self, context_branch):
381
472
        # self.spec comes straight from parsing the command line arguments,
382
473
        # so we expect it to be a Unicode string. Switch it to the internal
383
474
        # representation.
384
 
        revision_id = osutils.safe_revision_id(self.spec, warn=False)
385
 
        return RevisionInfo.from_revision_id(branch, revision_id, revs)
 
475
        return osutils.safe_revision_id(self.spec, warn=False)
386
476
 
387
 
SPEC_TYPES.append(RevisionSpec_revid)
388
477
 
389
478
 
390
479
class RevisionSpec_last(RevisionSpec):
398
487
 
399
488
      last:1        -> return the last revision
400
489
      last:3        -> return the revision 2 before the end.
401
 
    """    
 
490
    """
402
491
 
403
492
    prefix = 'last:'
404
493
 
405
494
    def _match_on(self, branch, revs):
 
495
        revno, revision_id = self._revno_and_revision_id(branch, revs)
 
496
        return RevisionInfo(branch, revno, revision_id)
 
497
 
 
498
    def _revno_and_revision_id(self, context_branch, revs_or_none):
 
499
        last_revno, last_revision_id = context_branch.last_revision_info()
 
500
 
406
501
        if self.spec == '':
407
 
            if not revs:
408
 
                raise errors.NoCommits(branch)
409
 
            return RevisionInfo(branch, len(revs), revs[-1])
 
502
            if not last_revno:
 
503
                raise errors.NoCommits(context_branch)
 
504
            return last_revno, last_revision_id
410
505
 
411
506
        try:
412
507
            offset = int(self.spec)
413
508
        except ValueError, e:
414
 
            raise errors.InvalidRevisionSpec(self.user_spec, branch, e)
 
509
            raise errors.InvalidRevisionSpec(self.user_spec, context_branch, e)
415
510
 
416
511
        if offset <= 0:
417
 
            raise errors.InvalidRevisionSpec(self.user_spec, branch,
 
512
            raise errors.InvalidRevisionSpec(self.user_spec, context_branch,
418
513
                                             'you must supply a positive value')
419
 
        revno = len(revs) - offset + 1
 
514
 
 
515
        revno = last_revno - offset + 1
420
516
        try:
421
 
            revision_id = branch.get_rev_id(revno, revs)
 
517
            revision_id = context_branch.get_rev_id(revno, revs_or_none)
422
518
        except errors.NoSuchRevision:
423
 
            raise errors.InvalidRevisionSpec(self.user_spec, branch)
424
 
        return RevisionInfo(branch, revno, revision_id)
425
 
 
426
 
SPEC_TYPES.append(RevisionSpec_last)
 
519
            raise errors.InvalidRevisionSpec(self.user_spec, context_branch)
 
520
        return revno, revision_id
 
521
 
 
522
    def _as_revision_id(self, context_branch):
 
523
        # We compute the revno as part of the process, but we don't really care
 
524
        # about it.
 
525
        revno, revision_id = self._revno_and_revision_id(context_branch, None)
 
526
        return revision_id
 
527
 
427
528
 
428
529
 
429
530
class RevisionSpec_before(RevisionSpec):
431
532
 
432
533
    help_txt = """Selects the parent of the revision specified.
433
534
 
434
 
    Supply any revision spec to return the parent of that revision.
 
535
    Supply any revision spec to return the parent of that revision.  This is
 
536
    mostly useful when inspecting revisions that are not in the revision history
 
537
    of a branch.
 
538
 
435
539
    It is an error to request the parent of the null revision (before:0).
436
 
    This is mostly useful when inspecting revisions that are not in the
437
 
    revision history of a branch.
438
540
 
439
541
    Examples::
440
542
 
441
543
      before:1913    -> Return the parent of revno 1913 (revno 1912)
442
544
      before:revid:aaaa@bbbb-1234567890  -> return the parent of revision
443
545
                                            aaaa@bbbb-1234567890
444
 
      bzr diff -r before:revid:aaaa..revid:aaaa
445
 
            -> Find the changes between revision 'aaaa' and its parent.
446
 
               (what changes did 'aaaa' introduce)
 
546
      bzr diff -r before:1913..1913
 
547
            -> Find the changes between revision 1913 and its parent (1912).
 
548
               (What changes did revision 1913 introduce).
 
549
               This is equivalent to:  bzr diff -c 1913
447
550
    """
448
551
 
449
552
    prefix = 'before:'
450
 
    
 
553
 
451
554
    def _match_on(self, branch, revs):
452
555
        r = RevisionSpec.from_string(self.spec)._match_on(branch, revs)
453
556
        if r.revno == 0:
474
577
                                                 branch)
475
578
        return RevisionInfo(branch, revno, revision_id)
476
579
 
477
 
SPEC_TYPES.append(RevisionSpec_before)
 
580
    def _as_revision_id(self, context_branch):
 
581
        base_revspec = RevisionSpec.from_string(self.spec)
 
582
        base_revision_id = base_revspec.as_revision_id(context_branch)
 
583
        if base_revision_id == revision.NULL_REVISION:
 
584
            raise errors.InvalidRevisionSpec(self.user_spec, context_branch,
 
585
                                         'cannot go before the null: revision')
 
586
        context_repo = context_branch.repository
 
587
        context_repo.lock_read()
 
588
        try:
 
589
            parent_map = context_repo.get_parent_map([base_revision_id])
 
590
        finally:
 
591
            context_repo.unlock()
 
592
        if base_revision_id not in parent_map:
 
593
            # Ghost, or unknown revision id
 
594
            raise errors.InvalidRevisionSpec(self.user_spec, context_branch,
 
595
                'cannot find the matching revision')
 
596
        parents = parent_map[base_revision_id]
 
597
        if len(parents) < 1:
 
598
            raise errors.InvalidRevisionSpec(self.user_spec, context_branch,
 
599
                'No parents for revision.')
 
600
        return parents[0]
 
601
 
478
602
 
479
603
 
480
604
class RevisionSpec_tag(RevisionSpec):
486
610
    """
487
611
 
488
612
    prefix = 'tag:'
 
613
    dwim_catchable_exceptions = (errors.NoSuchTag, errors.TagsNotSupported)
489
614
 
490
615
    def _match_on(self, branch, revs):
491
616
        # Can raise tags not supported, NoSuchTag, etc
493
618
            branch.tags.lookup_tag(self.spec),
494
619
            revs)
495
620
 
496
 
SPEC_TYPES.append(RevisionSpec_tag)
 
621
    def _as_revision_id(self, context_branch):
 
622
        return context_branch.tags.lookup_tag(self.spec)
 
623
 
497
624
 
498
625
 
499
626
class _RevListToTimestamps(object):
527
654
 
528
655
    One way to display all the changes since yesterday would be::
529
656
 
530
 
        bzr log -r date:yesterday..-1
 
657
        bzr log -r date:yesterday..
531
658
 
532
659
    Examples::
533
660
 
534
661
      date:yesterday            -> select the first revision since yesterday
535
662
      date:2006-08-14,17:10:14  -> select the first revision after
536
663
                                   August 14th, 2006 at 5:10pm.
537
 
    """    
 
664
    """
538
665
    prefix = 'date:'
539
666
    _date_re = re.compile(
540
667
            r'(?P<date>(?P<year>\d\d\d\d)-(?P<month>\d\d)-(?P<day>\d\d))?'
596
723
        finally:
597
724
            branch.unlock()
598
725
        if rev == len(revs):
599
 
            return RevisionInfo(branch, None)
 
726
            raise errors.InvalidRevisionSpec(self.user_spec, branch)
600
727
        else:
601
728
            return RevisionInfo(branch, rev + 1)
602
729
 
603
 
SPEC_TYPES.append(RevisionSpec_date)
604
730
 
605
731
 
606
732
class RevisionSpec_ancestor(RevisionSpec):
629
755
        trace.mutter('matching ancestor: on: %s, %s', self.spec, branch)
630
756
        return self._find_revision_info(branch, self.spec)
631
757
 
 
758
    def _as_revision_id(self, context_branch):
 
759
        return self._find_revision_id(context_branch, self.spec)
 
760
 
632
761
    @staticmethod
633
762
    def _find_revision_info(branch, other_location):
 
763
        revision_id = RevisionSpec_ancestor._find_revision_id(branch,
 
764
                                                              other_location)
 
765
        try:
 
766
            revno = branch.revision_id_to_revno(revision_id)
 
767
        except errors.NoSuchRevision:
 
768
            revno = None
 
769
        return RevisionInfo(branch, revno, revision_id)
 
770
 
 
771
    @staticmethod
 
772
    def _find_revision_id(branch, other_location):
634
773
        from bzrlib.branch import Branch
635
774
 
636
 
        other_branch = Branch.open(other_location)
637
 
        revision_a = branch.last_revision()
638
 
        revision_b = other_branch.last_revision()
639
 
        for r, b in ((revision_a, branch), (revision_b, other_branch)):
640
 
            if r in (None, revision.NULL_REVISION):
641
 
                raise errors.NoCommits(b)
642
 
        revision_source = revision.MultipleRevisionSources(
643
 
                branch.repository, other_branch.repository)
644
 
        graph = branch.repository.get_graph(other_branch.repository)
645
 
        revision_a = revision.ensure_null(revision_a)
646
 
        revision_b = revision.ensure_null(revision_b)
647
 
        if revision.NULL_REVISION in (revision_a, revision_b):
648
 
            rev_id = revision.NULL_REVISION
649
 
        else:
650
 
            rev_id = graph.find_unique_lca(revision_a, revision_b)
 
775
        branch.lock_read()
 
776
        try:
 
777
            revision_a = revision.ensure_null(branch.last_revision())
 
778
            if revision_a == revision.NULL_REVISION:
 
779
                raise errors.NoCommits(branch)
 
780
            if other_location == '':
 
781
                other_location = branch.get_parent()
 
782
            other_branch = Branch.open(other_location)
 
783
            other_branch.lock_read()
 
784
            try:
 
785
                revision_b = revision.ensure_null(other_branch.last_revision())
 
786
                if revision_b == revision.NULL_REVISION:
 
787
                    raise errors.NoCommits(other_branch)
 
788
                graph = branch.repository.get_graph(other_branch.repository)
 
789
                rev_id = graph.find_unique_lca(revision_a, revision_b)
 
790
            finally:
 
791
                other_branch.unlock()
651
792
            if rev_id == revision.NULL_REVISION:
652
793
                raise errors.NoCommonAncestor(revision_a, revision_b)
653
 
        try:
654
 
            revno = branch.revision_id_to_revno(rev_id)
655
 
        except errors.NoSuchRevision:
656
 
            revno = None
657
 
        return RevisionInfo(branch, revno, rev_id)
658
 
 
659
 
 
660
 
SPEC_TYPES.append(RevisionSpec_ancestor)
 
794
            return rev_id
 
795
        finally:
 
796
            branch.unlock()
 
797
 
 
798
 
661
799
 
662
800
 
663
801
class RevisionSpec_branch(RevisionSpec):
672
810
      branch:/path/to/branch
673
811
    """
674
812
    prefix = 'branch:'
 
813
    dwim_catchable_exceptions = (errors.NotBranchError,)
675
814
 
676
815
    def _match_on(self, branch, revs):
677
816
        from bzrlib.branch import Branch
679
818
        revision_b = other_branch.last_revision()
680
819
        if revision_b in (None, revision.NULL_REVISION):
681
820
            raise errors.NoCommits(other_branch)
682
 
        # pull in the remote revisions so we can diff
683
 
        branch.fetch(other_branch, revision_b)
 
821
        if branch is None:
 
822
            branch = other_branch
 
823
        else:
 
824
            try:
 
825
                # pull in the remote revisions so we can diff
 
826
                branch.fetch(other_branch, revision_b)
 
827
            except errors.ReadOnlyError:
 
828
                branch = other_branch
684
829
        try:
685
830
            revno = branch.revision_id_to_revno(revision_b)
686
831
        except errors.NoSuchRevision:
687
832
            revno = None
688
833
        return RevisionInfo(branch, revno, revision_b)
689
 
        
690
 
SPEC_TYPES.append(RevisionSpec_branch)
 
834
 
 
835
    def _as_revision_id(self, context_branch):
 
836
        from bzrlib.branch import Branch
 
837
        other_branch = Branch.open(self.spec)
 
838
        last_revision = other_branch.last_revision()
 
839
        last_revision = revision.ensure_null(last_revision)
 
840
        context_branch.fetch(other_branch, last_revision)
 
841
        if last_revision == revision.NULL_REVISION:
 
842
            raise errors.NoCommits(other_branch)
 
843
        return last_revision
 
844
 
 
845
    def _as_tree(self, context_branch):
 
846
        from bzrlib.branch import Branch
 
847
        other_branch = Branch.open(self.spec)
 
848
        last_revision = other_branch.last_revision()
 
849
        last_revision = revision.ensure_null(last_revision)
 
850
        if last_revision == revision.NULL_REVISION:
 
851
            raise errors.NoCommits(other_branch)
 
852
        return other_branch.repository.revision_tree(last_revision)
 
853
 
 
854
    def needs_branch(self):
 
855
        return False
 
856
 
 
857
    def get_branch(self):
 
858
        return self.spec
 
859
 
691
860
 
692
861
 
693
862
class RevisionSpec_submit(RevisionSpec_ancestor):
697
866
 
698
867
    Diffing against this shows all the changes that were made in this branch,
699
868
    and is a good predictor of what merge will do.  The submit branch is
700
 
    used by the bundle and merge directive comands.  If no submit branch
 
869
    used by the bundle and merge directive commands.  If no submit branch
701
870
    is specified, the parent branch is used instead.
702
871
 
703
872
    The common ancestor is the last revision that existed in both
711
880
 
712
881
    prefix = 'submit:'
713
882
 
714
 
    def _match_on(self, branch, revs):
715
 
        trace.mutter('matching ancestor: on: %s, %s', self.spec, branch)
 
883
    def _get_submit_location(self, branch):
716
884
        submit_location = branch.get_submit_branch()
717
885
        location_type = 'submit branch'
718
886
        if submit_location is None:
721
889
        if submit_location is None:
722
890
            raise errors.NoSubmitBranch(branch)
723
891
        trace.note('Using %s %s', location_type, submit_location)
724
 
        return self._find_revision_info(branch, submit_location)
725
 
 
726
 
 
727
 
SPEC_TYPES.append(RevisionSpec_submit)
 
892
        return submit_location
 
893
 
 
894
    def _match_on(self, branch, revs):
 
895
        trace.mutter('matching ancestor: on: %s, %s', self.spec, branch)
 
896
        return self._find_revision_info(branch,
 
897
            self._get_submit_location(branch))
 
898
 
 
899
    def _as_revision_id(self, context_branch):
 
900
        return self._find_revision_id(context_branch,
 
901
            self._get_submit_location(context_branch))
 
902
 
 
903
 
 
904
class RevisionSpec_annotate(RevisionIDSpec):
 
905
 
 
906
    prefix = 'annotate:'
 
907
 
 
908
    help_txt = """Select the revision that last modified the specified line.
 
909
 
 
910
    Select the revision that last modified the specified line.  Line is
 
911
    specified as path:number.  Path is a relative path to the file.  Numbers
 
912
    start at 1, and are relative to the current version, not the last-
 
913
    committed version of the file.
 
914
    """
 
915
 
 
916
    def _raise_invalid(self, numstring, context_branch):
 
917
        raise errors.InvalidRevisionSpec(self.user_spec, context_branch,
 
918
            'No such line: %s' % numstring)
 
919
 
 
920
    def _as_revision_id(self, context_branch):
 
921
        path, numstring = self.spec.rsplit(':', 1)
 
922
        try:
 
923
            index = int(numstring) - 1
 
924
        except ValueError:
 
925
            self._raise_invalid(numstring, context_branch)
 
926
        tree, file_path = workingtree.WorkingTree.open_containing(path)
 
927
        tree.lock_read()
 
928
        try:
 
929
            file_id = tree.path2id(file_path)
 
930
            if file_id is None:
 
931
                raise errors.InvalidRevisionSpec(self.user_spec,
 
932
                    context_branch, "File '%s' is not versioned." %
 
933
                    file_path)
 
934
            revision_ids = [r for (r, l) in tree.annotate_iter(file_id)]
 
935
        finally:
 
936
            tree.unlock()
 
937
        try:
 
938
            revision_id = revision_ids[index]
 
939
        except IndexError:
 
940
            self._raise_invalid(numstring, context_branch)
 
941
        if revision_id == revision.CURRENT_REVISION:
 
942
            raise errors.InvalidRevisionSpec(self.user_spec, context_branch,
 
943
                'Line %s has not been committed.' % numstring)
 
944
        return revision_id
 
945
 
 
946
 
 
947
class RevisionSpec_mainline(RevisionIDSpec):
 
948
 
 
949
    help_txt = """Select mainline revision that merged the specified revision.
 
950
 
 
951
    Select the revision that merged the specified revision into mainline.
 
952
    """
 
953
 
 
954
    prefix = 'mainline:'
 
955
 
 
956
    def _as_revision_id(self, context_branch):
 
957
        revspec = RevisionSpec.from_string(self.spec)
 
958
        if revspec.get_branch() is None:
 
959
            spec_branch = context_branch
 
960
        else:
 
961
            spec_branch = _mod_branch.Branch.open(revspec.get_branch())
 
962
        revision_id = revspec.as_revision_id(spec_branch)
 
963
        graph = context_branch.repository.get_graph()
 
964
        result = graph.find_lefthand_merger(revision_id,
 
965
                                            context_branch.last_revision())
 
966
        if result is None:
 
967
            raise errors.InvalidRevisionSpec(self.user_spec, context_branch)
 
968
        return result
 
969
 
 
970
 
 
971
# The order in which we want to DWIM a revision spec without any prefix.
 
972
# revno is always tried first and isn't listed here, this is used by
 
973
# RevisionSpec_dwim._match_on
 
974
dwim_revspecs = [
 
975
    RevisionSpec_tag, # Let's try for a tag
 
976
    RevisionSpec_revid, # Maybe it's a revid?
 
977
    RevisionSpec_date, # Perhaps a date?
 
978
    RevisionSpec_branch, # OK, last try, maybe it's a branch
 
979
    ]
 
980
 
 
981
 
 
982
revspec_registry = registry.Registry()
 
983
def _register_revspec(revspec):
 
984
    revspec_registry.register(revspec.prefix, revspec)
 
985
 
 
986
_register_revspec(RevisionSpec_revno)
 
987
_register_revspec(RevisionSpec_revid)
 
988
_register_revspec(RevisionSpec_last)
 
989
_register_revspec(RevisionSpec_before)
 
990
_register_revspec(RevisionSpec_tag)
 
991
_register_revspec(RevisionSpec_date)
 
992
_register_revspec(RevisionSpec_ancestor)
 
993
_register_revspec(RevisionSpec_branch)
 
994
_register_revspec(RevisionSpec_submit)
 
995
_register_revspec(RevisionSpec_annotate)
 
996
_register_revspec(RevisionSpec_mainline)
 
997
 
 
998
# classes in this list should have a "prefix" attribute, against which
 
999
# string specs are matched
 
1000
SPEC_TYPES = symbol_versioning.deprecated_list(
 
1001
    symbol_versioning.deprecated_in((1, 12, 0)), "SPEC_TYPES", [])