~bzr-pqm/bzr/bzr.dev

« back to all changes in this revision

Viewing changes to bzrlib/revisionspec.py

(jam) Handle bug #382709 by encoding paths as 'mbcs' when spawning
        external diff.

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
# Copyright (C) 2005-2010 Canonical Ltd
 
1
# Copyright (C) 2005, 2006, 2007 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
15
15
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
16
16
 
17
17
 
 
18
import re
 
19
 
18
20
from bzrlib.lazy_import import lazy_import
19
21
lazy_import(globals(), """
20
22
import bisect
21
23
import datetime
 
24
""")
22
25
 
23
26
from bzrlib import (
24
 
    branch as _mod_branch,
 
27
    errors,
25
28
    osutils,
 
29
    registry,
26
30
    revision,
27
31
    symbol_versioning,
28
 
    workingtree,
29
 
    )
30
 
from bzrlib.i18n import gettext
31
 
""")
32
 
 
33
 
from bzrlib import (
34
 
    errors,
35
 
    lazy_regex,
36
 
    registry,
37
32
    trace,
38
33
    )
39
34
 
118
113
        return RevisionInfo(branch, revno, revision_id)
119
114
 
120
115
 
 
116
# classes in this list should have a "prefix" attribute, against which
 
117
# string specs are matched
 
118
_revno_regex = None
 
119
 
 
120
 
121
121
class RevisionSpec(object):
122
122
    """A parsed revision specification."""
123
123
 
124
124
    help_txt = """A parsed revision specification.
125
125
 
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.
 
126
    A revision specification can be an integer, in which case it is
 
127
    assumed to be a revno (though this will translate negative values
 
128
    into positive ones); or it can be a string, in which case it is
 
129
    parsed for something like 'date:' or 'revid:' etc.
130
130
 
131
131
    Revision specs are an UI element, and they have been moved out
132
132
    of the branch class to leave "back-end" classes unaware of such
139
139
 
140
140
    prefix = None
141
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
 
    """
150
142
 
151
143
    @staticmethod
152
144
    def from_string(spec):
168
160
                         spectype.__name__, spec)
169
161
            return spectype(spec, _internal=True)
170
162
        else:
171
 
            # Otherwise treat it as a DWIM, build the RevisionSpec object and
172
 
            # wait for _match_on to be called.
173
 
            return RevisionSpec_dwim(spec, _internal=True)
 
163
            for spectype in SPEC_TYPES:
 
164
                trace.mutter('Returning RevisionSpec %s for %s',
 
165
                             spectype.__name__, spec)
 
166
                if spec.startswith(spectype.prefix):
 
167
                    return spectype(spec, _internal=True)
 
168
            # RevisionSpec_revno is special cased, because it is the only
 
169
            # one that directly handles plain integers
 
170
            # TODO: This should not be special cased rather it should be
 
171
            # a method invocation on spectype.canparse()
 
172
            global _revno_regex
 
173
            if _revno_regex is None:
 
174
                _revno_regex = re.compile(r'^(?:(\d+(\.\d+)*)|-\d+)(:.*)?$')
 
175
            if _revno_regex.match(spec) is not None:
 
176
                return RevisionSpec_revno(spec, _internal=True)
 
177
 
 
178
            raise errors.NoSuchRevisionSpec(spec)
174
179
 
175
180
    def __init__(self, spec, _internal=False):
176
181
        """Create a RevisionSpec referring to the Null revision.
180
185
            called directly. Only from RevisionSpec.from_string()
181
186
        """
182
187
        if not _internal:
 
188
            # XXX: Update this after 0.10 is released
183
189
            symbol_versioning.warn('Creating a RevisionSpec directly has'
184
190
                                   ' been deprecated in version 0.11. Use'
185
191
                                   ' RevisionSpec.from_string()'
209
215
    def in_history(self, branch):
210
216
        if branch:
211
217
            if self.wants_revision_history:
212
 
                # TODO: avoid looking at all of history
213
 
                branch.lock_read()
214
 
                try:
215
 
                    graph = branch.repository.get_graph()
216
 
                    revs = list(graph.iter_lefthand_ancestry(
217
 
                        branch.last_revision(), [revision.NULL_REVISION]))
218
 
                finally:
219
 
                    branch.unlock()
220
 
                revs.reverse()
 
218
                revs = branch.revision_history()
221
219
            else:
222
220
                revs = None
223
221
        else:
293
291
 
294
292
# private API
295
293
 
296
 
class RevisionSpec_dwim(RevisionSpec):
297
 
    """Provides a DWIMish revision specifier lookup.
298
 
 
299
 
    Note that this does not go in the revspec_registry because by definition
300
 
    there is no prefix to identify it.  It's solely called from
301
 
    RevisionSpec.from_string() because the DWIMification happen when _match_on
302
 
    is called so the string describing the revision is kept here until needed.
303
 
    """
304
 
 
305
 
    help_txt = None
306
 
    # We don't need to build the revision history ourself, that's delegated to
307
 
    # each revspec we try.
308
 
    wants_revision_history = False
309
 
 
310
 
    _revno_regex = lazy_regex.lazy_compile(r'^(?:(\d+(\.\d+)*)|-\d+)(:.*)?$')
311
 
 
312
 
    # The revspecs to try
313
 
    _possible_revspecs = []
314
 
 
315
 
    def _try_spectype(self, rstype, branch):
316
 
        rs = rstype(self.spec, _internal=True)
317
 
        # Hit in_history to find out if it exists, or we need to try the
318
 
        # next type.
319
 
        return rs.in_history(branch)
320
 
 
321
 
    def _match_on(self, branch, revs):
322
 
        """Run the lookup and see what we can get."""
323
 
 
324
 
        # First, see if it's a revno
325
 
        if self._revno_regex.match(self.spec) is not None:
326
 
            try:
327
 
                return self._try_spectype(RevisionSpec_revno, branch)
328
 
            except RevisionSpec_revno.dwim_catchable_exceptions:
329
 
                pass
330
 
 
331
 
        # Next see what has been registered
332
 
        for objgetter in self._possible_revspecs:
333
 
            rs_class = objgetter.get_obj()
334
 
            try:
335
 
                return self._try_spectype(rs_class, branch)
336
 
            except rs_class.dwim_catchable_exceptions:
337
 
                pass
338
 
 
339
 
        # Try the old (deprecated) dwim list:
340
 
        for rs_class in dwim_revspecs:
341
 
            try:
342
 
                return self._try_spectype(rs_class, branch)
343
 
            except rs_class.dwim_catchable_exceptions:
344
 
                pass
345
 
 
346
 
        # Well, I dunno what it is. Note that we don't try to keep track of the
347
 
        # first of last exception raised during the DWIM tries as none seems
348
 
        # really relevant.
349
 
        raise errors.InvalidRevisionSpec(self.spec, branch)
350
 
 
351
 
    @classmethod
352
 
    def append_possible_revspec(cls, revspec):
353
 
        """Append a possible DWIM revspec.
354
 
 
355
 
        :param revspec: Revision spec to try.
356
 
        """
357
 
        cls._possible_revspecs.append(registry._ObjectGetter(revspec))
358
 
 
359
 
    @classmethod
360
 
    def append_possible_lazy_revspec(cls, module_name, member_name):
361
 
        """Append a possible lazily loaded DWIM revspec.
362
 
 
363
 
        :param module_name: Name of the module with the revspec
364
 
        :param member_name: Name of the revspec within the module
365
 
        """
366
 
        cls._possible_revspecs.append(
367
 
            registry._LazyObjectGetter(module_name, member_name))
368
 
 
369
 
 
370
294
class RevisionSpec_revno(RevisionSpec):
371
295
    """Selects a revision using a number."""
372
296
 
373
297
    help_txt = """Selects a revision using a number.
374
298
 
375
299
    Use an integer to specify a revision in the history of the branch.
376
 
    Optionally a branch can be specified.  A negative number will count
377
 
    from the end of the branch (-1 is the last revision, -2 the previous
378
 
    one). If the negative number is larger than the branch's history, the
379
 
    first revision is returned.
 
300
    Optionally a branch can be specified. The 'revno:' prefix is optional.
 
301
    A negative number will count from the end of the branch (-1 is the
 
302
    last revision, -2 the previous one). If the negative number is larger
 
303
    than the branch's history, the first revision is returned.
380
304
    Examples::
381
305
 
382
306
      revno:1                   -> return the first revision of this branch
477
401
 
478
402
 
479
403
 
480
 
class RevisionIDSpec(RevisionSpec):
481
 
 
482
 
    def _match_on(self, branch, revs):
483
 
        revision_id = self.as_revision_id(branch)
484
 
        return RevisionInfo.from_revision_id(branch, revision_id, revs)
485
 
 
486
 
 
487
 
class RevisionSpec_revid(RevisionIDSpec):
 
404
class RevisionSpec_revid(RevisionSpec):
488
405
    """Selects a revision using the revision id."""
489
406
 
490
407
    help_txt = """Selects a revision using the revision id.
499
416
 
500
417
    prefix = 'revid:'
501
418
 
502
 
    def _as_revision_id(self, context_branch):
 
419
    def _match_on(self, branch, revs):
503
420
        # self.spec comes straight from parsing the command line arguments,
504
421
        # so we expect it to be a Unicode string. Switch it to the internal
505
422
        # representation.
 
423
        revision_id = osutils.safe_revision_id(self.spec, warn=False)
 
424
        return RevisionInfo.from_revision_id(branch, revision_id, revs)
 
425
 
 
426
    def _as_revision_id(self, context_branch):
506
427
        return osutils.safe_revision_id(self.spec, warn=False)
507
428
 
508
429
 
641
562
    """
642
563
 
643
564
    prefix = 'tag:'
644
 
    dwim_catchable_exceptions = (errors.NoSuchTag, errors.TagsNotSupported)
645
565
 
646
566
    def _match_on(self, branch, revs):
647
567
        # Can raise tags not supported, NoSuchTag, etc
694
614
                                   August 14th, 2006 at 5:10pm.
695
615
    """
696
616
    prefix = 'date:'
697
 
    _date_regex = lazy_regex.lazy_compile(
 
617
    _date_re = re.compile(
698
618
            r'(?P<date>(?P<year>\d\d\d\d)-(?P<month>\d\d)-(?P<day>\d\d))?'
699
619
            r'(,|T)?\s*'
700
620
            r'(?P<time>(?P<hour>\d\d):(?P<minute>\d\d)(:(?P<second>\d\d))?)?'
718
638
        elif self.spec.lower() == 'tomorrow':
719
639
            dt = today + datetime.timedelta(days=1)
720
640
        else:
721
 
            m = self._date_regex.match(self.spec)
 
641
            m = self._date_re.match(self.spec)
722
642
            if not m or (not m.group('date') and not m.group('time')):
723
643
                raise errors.InvalidRevisionSpec(self.user_spec,
724
644
                                                 branch, 'invalid date')
841
761
      branch:/path/to/branch
842
762
    """
843
763
    prefix = 'branch:'
844
 
    dwim_catchable_exceptions = (errors.NotBranchError,)
845
764
 
846
765
    def _match_on(self, branch, revs):
847
766
        from bzrlib.branch import Branch
849
768
        revision_b = other_branch.last_revision()
850
769
        if revision_b in (None, revision.NULL_REVISION):
851
770
            raise errors.NoCommits(other_branch)
852
 
        if branch is None:
853
 
            branch = other_branch
854
 
        else:
855
 
            try:
856
 
                # pull in the remote revisions so we can diff
857
 
                branch.fetch(other_branch, revision_b)
858
 
            except errors.ReadOnlyError:
859
 
                branch = other_branch
 
771
        # pull in the remote revisions so we can diff
 
772
        branch.fetch(other_branch, revision_b)
860
773
        try:
861
774
            revno = branch.revision_id_to_revno(revision_b)
862
775
        except errors.NoSuchRevision:
882
795
            raise errors.NoCommits(other_branch)
883
796
        return other_branch.repository.revision_tree(last_revision)
884
797
 
885
 
    def needs_branch(self):
886
 
        return False
887
 
 
888
 
    def get_branch(self):
889
 
        return self.spec
890
 
 
891
798
 
892
799
 
893
800
class RevisionSpec_submit(RevisionSpec_ancestor):
919
826
            location_type = 'parent branch'
920
827
        if submit_location is None:
921
828
            raise errors.NoSubmitBranch(branch)
922
 
        trace.note(gettext('Using {0} {1}').format(location_type,
923
 
                                                        submit_location))
 
829
        trace.note('Using %s %s', location_type, submit_location)
924
830
        return submit_location
925
831
 
926
832
    def _match_on(self, branch, revs):
933
839
            self._get_submit_location(context_branch))
934
840
 
935
841
 
936
 
class RevisionSpec_annotate(RevisionIDSpec):
937
 
 
938
 
    prefix = 'annotate:'
939
 
 
940
 
    help_txt = """Select the revision that last modified the specified line.
941
 
 
942
 
    Select the revision that last modified the specified line.  Line is
943
 
    specified as path:number.  Path is a relative path to the file.  Numbers
944
 
    start at 1, and are relative to the current version, not the last-
945
 
    committed version of the file.
946
 
    """
947
 
 
948
 
    def _raise_invalid(self, numstring, context_branch):
949
 
        raise errors.InvalidRevisionSpec(self.user_spec, context_branch,
950
 
            'No such line: %s' % numstring)
951
 
 
952
 
    def _as_revision_id(self, context_branch):
953
 
        path, numstring = self.spec.rsplit(':', 1)
954
 
        try:
955
 
            index = int(numstring) - 1
956
 
        except ValueError:
957
 
            self._raise_invalid(numstring, context_branch)
958
 
        tree, file_path = workingtree.WorkingTree.open_containing(path)
959
 
        tree.lock_read()
960
 
        try:
961
 
            file_id = tree.path2id(file_path)
962
 
            if file_id is None:
963
 
                raise errors.InvalidRevisionSpec(self.user_spec,
964
 
                    context_branch, "File '%s' is not versioned." %
965
 
                    file_path)
966
 
            revision_ids = [r for (r, l) in tree.annotate_iter(file_id)]
967
 
        finally:
968
 
            tree.unlock()
969
 
        try:
970
 
            revision_id = revision_ids[index]
971
 
        except IndexError:
972
 
            self._raise_invalid(numstring, context_branch)
973
 
        if revision_id == revision.CURRENT_REVISION:
974
 
            raise errors.InvalidRevisionSpec(self.user_spec, context_branch,
975
 
                'Line %s has not been committed.' % numstring)
976
 
        return revision_id
977
 
 
978
 
 
979
 
class RevisionSpec_mainline(RevisionIDSpec):
980
 
 
981
 
    help_txt = """Select mainline revision that merged the specified revision.
982
 
 
983
 
    Select the revision that merged the specified revision into mainline.
984
 
    """
985
 
 
986
 
    prefix = 'mainline:'
987
 
 
988
 
    def _as_revision_id(self, context_branch):
989
 
        revspec = RevisionSpec.from_string(self.spec)
990
 
        if revspec.get_branch() is None:
991
 
            spec_branch = context_branch
992
 
        else:
993
 
            spec_branch = _mod_branch.Branch.open(revspec.get_branch())
994
 
        revision_id = revspec.as_revision_id(spec_branch)
995
 
        graph = context_branch.repository.get_graph()
996
 
        result = graph.find_lefthand_merger(revision_id,
997
 
                                            context_branch.last_revision())
998
 
        if result is None:
999
 
            raise errors.InvalidRevisionSpec(self.user_spec, context_branch)
1000
 
        return result
1001
 
 
1002
 
 
1003
 
# The order in which we want to DWIM a revision spec without any prefix.
1004
 
# revno is always tried first and isn't listed here, this is used by
1005
 
# RevisionSpec_dwim._match_on
1006
 
dwim_revspecs = symbol_versioning.deprecated_list(
1007
 
    symbol_versioning.deprecated_in((2, 4, 0)), "dwim_revspecs", [])
1008
 
 
1009
 
RevisionSpec_dwim.append_possible_revspec(RevisionSpec_tag)
1010
 
RevisionSpec_dwim.append_possible_revspec(RevisionSpec_revid)
1011
 
RevisionSpec_dwim.append_possible_revspec(RevisionSpec_date)
1012
 
RevisionSpec_dwim.append_possible_revspec(RevisionSpec_branch)
1013
 
 
1014
842
revspec_registry = registry.Registry()
1015
843
def _register_revspec(revspec):
1016
844
    revspec_registry.register(revspec.prefix, revspec)
1024
852
_register_revspec(RevisionSpec_ancestor)
1025
853
_register_revspec(RevisionSpec_branch)
1026
854
_register_revspec(RevisionSpec_submit)
1027
 
_register_revspec(RevisionSpec_annotate)
1028
 
_register_revspec(RevisionSpec_mainline)
 
855
 
 
856
SPEC_TYPES = symbol_versioning.deprecated_list(
 
857
    symbol_versioning.deprecated_in((1, 12, 0)), "SPEC_TYPES", [])