~bzr-pqm/bzr/bzr.dev

« back to all changes in this revision

Viewing changes to bzrlib/branch.py

  • Committer: Canonical.com Patch Queue Manager
  • Date: 2007-11-03 01:53:30 UTC
  • mfrom: (2955.1.1 trunk)
  • Revision ID: pqm@pqm.ubuntu.com-20071103015330-pt1tec7wyxwwcey8
Fix #158972 don't use timeout for HttpServer

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
# Copyright (C) 2005, 2006, 2007, 2008 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., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
16
16
 
17
17
 
18
 
import sys
 
18
from cStringIO import StringIO
19
19
 
20
20
from bzrlib.lazy_import import lazy_import
21
21
lazy_import(globals(), """
22
 
from itertools import chain
 
22
from warnings import warn
 
23
 
 
24
import bzrlib
23
25
from bzrlib import (
24
26
        bzrdir,
25
27
        cache_utf8,
28
30
        errors,
29
31
        lockdir,
30
32
        lockable_files,
31
 
        repository,
 
33
        osutils,
32
34
        revision as _mod_revision,
33
35
        transport,
 
36
        tree,
34
37
        tsort,
35
38
        ui,
36
39
        urlutils,
37
40
        )
38
 
from bzrlib.config import BranchConfig
39
 
from bzrlib.repofmt.pack_repo import RepositoryFormatPackDevelopment1Subtree
 
41
from bzrlib.config import BranchConfig, TreeConfig
 
42
from bzrlib.lockable_files import LockableFiles, TransportLock
40
43
from bzrlib.tag import (
41
44
    BasicTags,
42
45
    DisabledTags,
44
47
""")
45
48
 
46
49
from bzrlib.decorators import needs_read_lock, needs_write_lock
 
50
from bzrlib.errors import (BzrError, BzrCheckError, DivergedBranches,
 
51
                           HistoryMissing, InvalidRevisionId,
 
52
                           InvalidRevisionNumber, LockError, NoSuchFile,
 
53
                           NoSuchRevision, NotVersionedError,
 
54
                           NotBranchError, UninitializableFormat,
 
55
                           UnlistableStore, UnlistableBranch,
 
56
                           )
47
57
from bzrlib.hooks import Hooks
48
 
from bzrlib.symbol_versioning import (
49
 
    deprecated_in,
50
 
    deprecated_method,
51
 
    )
52
 
from bzrlib.trace import mutter, mutter_callsite, note, is_quiet
 
58
from bzrlib.symbol_versioning import (deprecated_function,
 
59
                                      deprecated_method,
 
60
                                      DEPRECATED_PARAMETER,
 
61
                                      deprecated_passed,
 
62
                                      zero_eight, zero_nine, zero_sixteen,
 
63
                                      zero_ninetyone,
 
64
                                      )
 
65
from bzrlib.trace import mutter, mutter_callsite, note
53
66
 
54
67
 
55
68
BZR_BRANCH_FORMAT_4 = "Bazaar-NG branch, format 0.0.4\n"
89
102
        self.tags = self._make_tags()
90
103
        self._revision_history_cache = None
91
104
        self._revision_id_to_revno_cache = None
92
 
        self._last_revision_info_cache = None
93
 
        self._open_hook()
94
 
 
95
 
    def _open_hook(self):
96
 
        """Called by init to allow simpler extension of the base class."""
97
105
 
98
106
    def break_lock(self):
99
107
        """Break a lock if one is present from another instance.
203
211
        :return: A dictionary mapping revision_id => dotted revno.
204
212
        """
205
213
        last_revision = self.last_revision()
206
 
        revision_graph = repository._old_get_graph(self.repository,
207
 
            last_revision)
 
214
        revision_graph = self.repository.get_revision_graph(last_revision)
208
215
        merge_sorted_revisions = tsort.merge_sort(
209
216
            revision_graph,
210
217
            last_revision,
231
238
        """
232
239
        self.control_files.dont_leave_in_place()
233
240
 
 
241
    def abspath(self, name):
 
242
        """Return absolute filename for something in the branch
 
243
        
 
244
        XXX: Robert Collins 20051017 what is this used for? why is it a branch
 
245
        method and not a tree method.
 
246
        """
 
247
        raise NotImplementedError(self.abspath)
 
248
 
234
249
    def bind(self, other):
235
250
        """Bind the local branch the other branch.
236
251
 
264
279
            if last_revision is None:
265
280
                pb.update('get source history')
266
281
                last_revision = from_branch.last_revision()
267
 
                last_revision = _mod_revision.ensure_null(last_revision)
 
282
                if last_revision is None:
 
283
                    last_revision = _mod_revision.NULL_REVISION
268
284
            return self.repository.fetch(from_branch.repository,
269
285
                                         revision_id=last_revision,
270
286
                                         pb=nested_pb)
319
335
        The delta is relative to its mainline predecessor, or the
320
336
        empty tree for revision 1.
321
337
        """
 
338
        assert isinstance(revno, int)
322
339
        rh = self.revision_history()
323
340
        if not (1 <= revno <= len(rh)):
324
 
            raise errors.InvalidRevisionNumber(revno)
 
341
            raise InvalidRevisionNumber(revno)
325
342
        return self.repository.get_revision_delta(rh[revno-1])
326
343
 
327
 
    def get_stacked_on_url(self):
328
 
        """Get the URL this branch is stacked against.
 
344
    @deprecated_method(zero_sixteen)
 
345
    def get_root_id(self):
 
346
        """Return the id of this branches root
329
347
 
330
 
        :raises NotStacked: If the branch is not stacked.
331
 
        :raises UnstackableBranchFormat: If the branch does not support
332
 
            stacking.
 
348
        Deprecated: branches don't have root ids-- trees do.
 
349
        Use basis_tree().get_root_id() instead.
333
350
        """
334
 
        raise NotImplementedError(self.get_stacked_on_url)
 
351
        raise NotImplementedError(self.get_root_id)
335
352
 
336
353
    def print_file(self, file, revision_id):
337
354
        """Print `file` to stdout."""
340
357
    def set_revision_history(self, rev_history):
341
358
        raise NotImplementedError(self.set_revision_history)
342
359
 
343
 
    def set_stacked_on_url(self, url):
344
 
        """Set the URL this branch is stacked against.
345
 
 
346
 
        :raises UnstackableBranchFormat: If the branch does not support
347
 
            stacking.
348
 
        :raises UnstackableRepositoryFormat: If the repository does not support
349
 
            stacking.
350
 
        """
351
 
        raise NotImplementedError(self.set_stacked_on_url)
352
 
 
353
360
    def _cache_revision_history(self, rev_history):
354
361
        """Set the cached revision history to rev_history.
355
362
 
380
387
        """
381
388
        self._revision_history_cache = None
382
389
        self._revision_id_to_revno_cache = None
383
 
        self._last_revision_info_cache = None
384
390
 
385
391
    def _gen_revision_history(self):
386
392
        """Return sequence of revision hashes on to this branch.
419
425
        That is equivalent to the number of revisions committed to
420
426
        this branch.
421
427
        """
422
 
        return self.last_revision_info()[0]
 
428
        return len(self.revision_history())
423
429
 
424
430
    def unbind(self):
425
431
        """Older format branches cannot bind or unbind."""
430
436
        raise errors.UpgradeRequired(self.base)
431
437
 
432
438
    def last_revision(self):
433
 
        """Return last revision id, or NULL_REVISION."""
434
 
        return self.last_revision_info()[1]
 
439
        """Return last revision id, or None"""
 
440
        ph = self.revision_history()
 
441
        if ph:
 
442
            return ph[-1]
 
443
        else:
 
444
            return _mod_revision.NULL_REVISION
435
445
 
436
 
    @needs_read_lock
437
446
    def last_revision_info(self):
438
447
        """Return information about the last revision.
439
448
 
440
 
        :return: A tuple (revno, revision_id).
 
449
        :return: A tuple (revno, last_revision_id).
441
450
        """
442
 
        if self._last_revision_info_cache is None:
443
 
            self._last_revision_info_cache = self._last_revision_info()
444
 
        return self._last_revision_info_cache
445
 
 
446
 
    def _last_revision_info(self):
447
451
        rh = self.revision_history()
448
452
        revno = len(rh)
449
453
        if revno:
451
455
        else:
452
456
            return (0, _mod_revision.NULL_REVISION)
453
457
 
454
 
    @deprecated_method(deprecated_in((1, 6, 0)))
455
458
    def missing_revisions(self, other, stop_revision=None):
456
459
        """Return a list of new revisions that would perfectly fit.
457
460
        
465
468
        common_index = min(self_len, other_len) -1
466
469
        if common_index >= 0 and \
467
470
            self_history[common_index] != other_history[common_index]:
468
 
            raise errors.DivergedBranches(self, other)
 
471
            raise DivergedBranches(self, other)
469
472
 
470
473
        if stop_revision is None:
471
474
            stop_revision = other_len
472
475
        else:
 
476
            assert isinstance(stop_revision, int)
473
477
            if stop_revision > other_len:
474
478
                raise errors.NoSuchRevision(self, stop_revision)
475
479
        return other_history[self_len:stop_revision]
476
480
 
477
 
    @needs_write_lock
478
 
    def update_revisions(self, other, stop_revision=None, overwrite=False,
479
 
                         graph=None):
 
481
    def update_revisions(self, other, stop_revision=None):
480
482
        """Pull in new perfect-fit revisions.
481
483
 
482
484
        :param other: Another Branch to pull from
483
485
        :param stop_revision: Updated until the given revision
484
 
        :param overwrite: Always set the branch pointer, rather than checking
485
 
            to see if it is a proper descendant.
486
 
        :param graph: A Graph object that can be used to query history
487
 
            information. This can be None.
488
486
        :return: None
489
487
        """
490
 
        other.lock_read()
491
 
        try:
492
 
            other_revno, other_last_revision = other.last_revision_info()
493
 
            stop_revno = None # unknown
494
 
            if stop_revision is None:
495
 
                stop_revision = other_last_revision
496
 
                if _mod_revision.is_null(stop_revision):
497
 
                    # if there are no commits, we're done.
498
 
                    return
499
 
                stop_revno = other_revno
500
 
 
501
 
            # what's the current last revision, before we fetch [and change it
502
 
            # possibly]
503
 
            last_rev = _mod_revision.ensure_null(self.last_revision())
504
 
            # we fetch here so that we don't process data twice in the common
505
 
            # case of having something to pull, and so that the check for 
506
 
            # already merged can operate on the just fetched graph, which will
507
 
            # be cached in memory.
508
 
            self.fetch(other, stop_revision)
509
 
            # Check to see if one is an ancestor of the other
510
 
            if not overwrite:
511
 
                if graph is None:
512
 
                    graph = self.repository.get_graph()
513
 
                if self._check_if_descendant_or_diverged(
514
 
                        stop_revision, last_rev, graph, other):
515
 
                    # stop_revision is a descendant of last_rev, but we aren't
516
 
                    # overwriting, so we're done.
517
 
                    return
518
 
            if stop_revno is None:
519
 
                if graph is None:
520
 
                    graph = self.repository.get_graph()
521
 
                this_revno, this_last_revision = self.last_revision_info()
522
 
                stop_revno = graph.find_distance_to_null(stop_revision,
523
 
                                [(other_last_revision, other_revno),
524
 
                                 (this_last_revision, this_revno)])
525
 
            self.set_last_revision_info(stop_revno, stop_revision)
526
 
        finally:
527
 
            other.unlock()
 
488
        raise NotImplementedError(self.update_revisions)
528
489
 
529
490
    def revision_id_to_revno(self, revision_id):
530
491
        """Given a revision id, return its revno"""
547
508
        return history[revno - 1]
548
509
 
549
510
    def pull(self, source, overwrite=False, stop_revision=None,
550
 
             possible_transports=None, _override_hook_target=None):
 
511
             possible_transports=None):
551
512
        """Mirror source into this branch.
552
513
 
553
514
        This branch is considered to be 'local', having low latency.
567
528
        """Return `Tree` object for last revision."""
568
529
        return self.repository.revision_tree(self.last_revision())
569
530
 
 
531
    def rename_one(self, from_rel, to_rel):
 
532
        """Rename one file.
 
533
 
 
534
        This can change the directory or the filename or both.
 
535
        """
 
536
        raise NotImplementedError(self.rename_one)
 
537
 
 
538
    def move(self, from_paths, to_name):
 
539
        """Rename files.
 
540
 
 
541
        to_name must exist as a versioned directory.
 
542
 
 
543
        If to_name exists and is a directory, the files are moved into
 
544
        it, keeping their old names.  If it is a directory, 
 
545
 
 
546
        Note that to_name is only the last component of the new name;
 
547
        this doesn't change the directory.
 
548
 
 
549
        This returns a list of (from_path, to_path) pairs for each
 
550
        entry that is moved.
 
551
        """
 
552
        raise NotImplementedError(self.move)
 
553
 
570
554
    def get_parent(self):
571
555
        """Return the parent location of the branch.
572
556
 
663
647
        Zero (the NULL revision) is considered invalid
664
648
        """
665
649
        if revno < 1 or revno > self.revno():
666
 
            raise errors.InvalidRevisionNumber(revno)
 
650
            raise InvalidRevisionNumber(revno)
667
651
 
668
652
    @needs_read_lock
669
653
    def clone(self, to_bzrdir, revision_id=None):
672
656
        revision_id: if not None, the revision history in the new branch will
673
657
                     be truncated to end with revision_id.
674
658
        """
675
 
        result = to_bzrdir.create_branch()
 
659
        result = self._format.initialize(to_bzrdir)
676
660
        self.copy_content_into(result, revision_id=revision_id)
677
661
        return  result
678
662
 
679
663
    @needs_read_lock
680
664
    def sprout(self, to_bzrdir, revision_id=None):
681
665
        """Create a new line of development from the branch, into to_bzrdir.
682
 
 
683
 
        to_bzrdir controls the branch format.
684
 
 
 
666
        
685
667
        revision_id: if not None, the revision history in the new branch will
686
668
                     be truncated to end with revision_id.
687
669
        """
688
 
        result = to_bzrdir.create_branch()
 
670
        result = self._format.initialize(to_bzrdir)
689
671
        self.copy_content_into(result, revision_id=revision_id)
690
672
        result.set_parent(self.bzrdir.root_transport.base)
691
673
        return result
705
687
        """
706
688
        if revision_id == _mod_revision.NULL_REVISION:
707
689
            new_history = []
708
 
        else:
709
 
            new_history = self.revision_history()
 
690
        new_history = self.revision_history()
710
691
        if revision_id is not None and new_history != []:
711
692
            try:
712
693
                new_history = new_history[:new_history.index(revision_id) + 1]
745
726
        :return: A BranchCheckResult.
746
727
        """
747
728
        mainline_parent_id = None
748
 
        last_revno, last_revision_id = self.last_revision_info()
749
 
        real_rev_history = list(self.repository.iter_reverse_revision_history(
750
 
                                last_revision_id))
751
 
        real_rev_history.reverse()
752
 
        if len(real_rev_history) != last_revno:
753
 
            raise errors.BzrCheckError('revno does not match len(mainline)'
754
 
                ' %s != %s' % (last_revno, len(real_rev_history)))
755
 
        # TODO: We should probably also check that real_rev_history actually
756
 
        #       matches self.revision_history()
757
 
        for revision_id in real_rev_history:
 
729
        for revision_id in self.revision_history():
758
730
            try:
759
731
                revision = self.repository.get_revision(revision_id)
760
732
            except errors.NoSuchRevision, e:
785
757
        return format
786
758
 
787
759
    def create_checkout(self, to_location, revision_id=None,
788
 
                        lightweight=False, accelerator_tree=None,
789
 
                        hardlink=False):
 
760
                        lightweight=False):
790
761
        """Create a checkout of a branch.
791
762
        
792
763
        :param to_location: The url to produce the checkout at
793
764
        :param revision_id: The revision to check out
794
765
        :param lightweight: If True, produce a lightweight checkout, otherwise,
795
766
        produce a bound branch (heavyweight checkout)
796
 
        :param accelerator_tree: A tree which can be used for retrieving file
797
 
            contents more quickly than the revision tree, i.e. a workingtree.
798
 
            The revision tree will be used for cases where accelerator_tree's
799
 
            content is different.
800
 
        :param hardlink: If true, hard-link files from accelerator_tree,
801
 
            where possible.
802
767
        :return: The tree of the created checkout
803
768
        """
804
769
        t = transport.get_transport(to_location)
806
771
        if lightweight:
807
772
            format = self._get_checkout_format()
808
773
            checkout = format.initialize_on_transport(t)
809
 
            from_branch = BranchReferenceFormat().initialize(checkout, self)
 
774
            BranchReferenceFormat().initialize(checkout, self)
810
775
        else:
811
776
            format = self._get_checkout_format()
812
777
            checkout_branch = bzrdir.BzrDir.create_branch_convenience(
816
781
            # pull up to the specified revision_id to set the initial 
817
782
            # branch tip correctly, and seed it with history.
818
783
            checkout_branch.pull(self, stop_revision=revision_id)
819
 
            from_branch=None
820
 
        tree = checkout.create_workingtree(revision_id,
821
 
                                           from_branch=from_branch,
822
 
                                           accelerator_tree=accelerator_tree,
823
 
                                           hardlink=hardlink)
 
784
        tree = checkout.create_workingtree(revision_id)
824
785
        basis_tree = tree.basis_tree()
825
786
        basis_tree.lock_read()
826
787
        try:
833
794
            basis_tree.unlock()
834
795
        return tree
835
796
 
836
 
    @needs_write_lock
837
 
    def reconcile(self, thorough=True):
838
 
        """Make sure the data stored in this branch is consistent."""
839
 
        from bzrlib.reconcile import BranchReconciler
840
 
        reconciler = BranchReconciler(self, thorough=thorough)
841
 
        reconciler.reconcile()
842
 
        return reconciler
843
 
 
844
797
    def reference_parent(self, file_id, path):
845
798
        """Return the parent branch for a tree-reference file_id
846
799
        :param file_id: The file_id of the tree reference
853
806
    def supports_tags(self):
854
807
        return self._format.supports_tags()
855
808
 
856
 
    def _check_if_descendant_or_diverged(self, revision_a, revision_b, graph,
857
 
                                         other_branch):
858
 
        """Ensure that revision_b is a descendant of revision_a.
859
 
 
860
 
        This is a helper function for update_revisions.
861
 
        
862
 
        :raises: DivergedBranches if revision_b has diverged from revision_a.
863
 
        :returns: True if revision_b is a descendant of revision_a.
864
 
        """
865
 
        relation = self._revision_relations(revision_a, revision_b, graph)
866
 
        if relation == 'b_descends_from_a':
867
 
            return True
868
 
        elif relation == 'diverged':
869
 
            raise errors.DivergedBranches(self, other_branch)
870
 
        elif relation == 'a_descends_from_b':
871
 
            return False
872
 
        else:
873
 
            raise AssertionError("invalid relation: %r" % (relation,))
874
 
 
875
 
    def _revision_relations(self, revision_a, revision_b, graph):
876
 
        """Determine the relationship between two revisions.
877
 
        
878
 
        :returns: One of: 'a_descends_from_b', 'b_descends_from_a', 'diverged'
879
 
        """
880
 
        heads = graph.heads([revision_a, revision_b])
881
 
        if heads == set([revision_b]):
882
 
            return 'b_descends_from_a'
883
 
        elif heads == set([revision_a, revision_b]):
884
 
            # These branches have diverged
885
 
            return 'diverged'
886
 
        elif heads == set([revision_a]):
887
 
            return 'a_descends_from_b'
888
 
        else:
889
 
            raise AssertionError("invalid heads: %r" % (heads,))
890
 
 
891
809
 
892
810
class BranchFormat(object):
893
811
    """An encapsulation of the initialization and open routines for a format.
926
844
            transport = a_bzrdir.get_branch_transport(None)
927
845
            format_string = transport.get("format").read()
928
846
            return klass._formats[format_string]
929
 
        except errors.NoSuchFile:
930
 
            raise errors.NotBranchError(path=transport.base)
 
847
        except NoSuchFile:
 
848
            raise NotBranchError(path=transport.base)
931
849
        except KeyError:
932
 
            raise errors.UnknownFormatError(format=format_string, kind='branch')
 
850
            raise errors.UnknownFormatError(format=format_string)
933
851
 
934
852
    @classmethod
935
853
    def get_default_format(klass):
948
866
        """
949
867
        return None
950
868
 
951
 
    @classmethod
952
 
    def set_reference(self, a_bzrdir, to_branch):
953
 
        """Set the target reference of the branch in a_bzrdir.
954
 
 
955
 
        format probing must have been completed before calling
956
 
        this method - it is assumed that the format of the branch
957
 
        in a_bzrdir is correct.
958
 
 
959
 
        :param a_bzrdir: The bzrdir to set the branch reference for.
960
 
        :param to_branch: branch that the checkout is to reference
961
 
        """
962
 
        raise NotImplementedError(self.set_reference)
963
 
 
964
869
    def get_format_string(self):
965
870
        """Return the ASCII format string that identifies this format."""
966
871
        raise NotImplementedError(self.get_format_string)
993
898
        control_files.create_lock()
994
899
        control_files.lock_write()
995
900
        if set_format:
996
 
            utf8_files += [('format', self.get_format_string())]
 
901
            control_files.put_utf8('format', self.get_format_string())
997
902
        try:
998
 
            for (filename, content) in utf8_files:
999
 
                branch_transport.put_bytes(
1000
 
                    filename, content,
1001
 
                    mode=a_bzrdir._get_file_mode())
 
903
            for file, content in utf8_files:
 
904
                control_files.put_utf8(file, content)
1002
905
        finally:
1003
906
            control_files.unlock()
1004
907
        return self.open(a_bzrdir, _found=True)
1032
935
    def set_default_format(klass, format):
1033
936
        klass._default_format = format
1034
937
 
1035
 
    def supports_stacking(self):
1036
 
        """True if this format records a stacked-on branch."""
1037
 
        return False
1038
 
 
1039
938
    @classmethod
1040
939
    def unregister_format(klass, format):
 
940
        assert klass._formats[format.get_format_string()] is format
1041
941
        del klass._formats[format.get_format_string()]
1042
942
 
1043
943
    def __str__(self):
1047
947
        """True if this format supports tags stored in the branch"""
1048
948
        return False  # by default
1049
949
 
 
950
    # XXX: Probably doesn't really belong here -- mbp 20070212
 
951
    def _initialize_control_files(self, a_bzrdir, utf8_files, lock_filename,
 
952
            lock_class):
 
953
        branch_transport = a_bzrdir.get_branch_transport(self)
 
954
        control_files = lockable_files.LockableFiles(branch_transport,
 
955
            lock_filename, lock_class)
 
956
        control_files.create_lock()
 
957
        control_files.lock_write()
 
958
        try:
 
959
            for filename, content in utf8_files:
 
960
                control_files.put_utf8(filename, content)
 
961
        finally:
 
962
            control_files.unlock()
 
963
 
1050
964
 
1051
965
class BranchHooks(Hooks):
1052
966
    """A dictionary mapping hook name to a list of callables for branch hooks.
1109
1023
        # local is the local branch or None, master is the target branch,
1110
1024
        # and an empty branch recieves new_revno of 0, new_revid of None.
1111
1025
        self['post_uncommit'] = []
1112
 
        # Introduced in 1.6
1113
 
        # Invoked before the tip of a branch changes.
1114
 
        # the api signature is
1115
 
        # (params) where params is a ChangeBranchTipParams with the members
1116
 
        # (branch, old_revno, new_revno, old_revid, new_revid)
1117
 
        self['pre_change_branch_tip'] = []
1118
 
        # Introduced in 1.4
1119
 
        # Invoked after the tip of a branch changes.
1120
 
        # the api signature is
1121
 
        # (params) where params is a ChangeBranchTipParams with the members
1122
 
        # (branch, old_revno, new_revno, old_revid, new_revid)
1123
 
        self['post_change_branch_tip'] = []
1124
1026
 
1125
1027
 
1126
1028
# install the default hooks into the Branch class.
1127
1029
Branch.hooks = BranchHooks()
1128
1030
 
1129
1031
 
1130
 
class ChangeBranchTipParams(object):
1131
 
    """Object holding parameters passed to *_change_branch_tip hooks.
1132
 
 
1133
 
    There are 5 fields that hooks may wish to access:
1134
 
 
1135
 
    :ivar branch: the branch being changed
1136
 
    :ivar old_revno: revision number before the change
1137
 
    :ivar new_revno: revision number after the change
1138
 
    :ivar old_revid: revision id before the change
1139
 
    :ivar new_revid: revision id after the change
1140
 
 
1141
 
    The revid fields are strings. The revno fields are integers.
1142
 
    """
1143
 
 
1144
 
    def __init__(self, branch, old_revno, new_revno, old_revid, new_revid):
1145
 
        """Create a group of ChangeBranchTip parameters.
1146
 
 
1147
 
        :param branch: The branch being changed.
1148
 
        :param old_revno: Revision number before the change.
1149
 
        :param new_revno: Revision number after the change.
1150
 
        :param old_revid: Tip revision id before the change.
1151
 
        :param new_revid: Tip revision id after the change.
1152
 
        """
1153
 
        self.branch = branch
1154
 
        self.old_revno = old_revno
1155
 
        self.new_revno = new_revno
1156
 
        self.old_revid = old_revid
1157
 
        self.new_revid = new_revid
1158
 
 
1159
 
    def __eq__(self, other):
1160
 
        return self.__dict__ == other.__dict__
1161
 
    
1162
 
    def __repr__(self):
1163
 
        return "<%s of %s from (%s, %s) to (%s, %s)>" % (
1164
 
            self.__class__.__name__, self.branch, 
1165
 
            self.old_revno, self.old_revid, self.new_revno, self.new_revid)
1166
 
 
1167
 
 
1168
1032
class BzrBranchFormat4(BranchFormat):
1169
1033
    """Bzr branch format 4.
1170
1034
 
1207
1071
        return "Bazaar-NG branch format 4"
1208
1072
 
1209
1073
 
1210
 
class BranchFormatMetadir(BranchFormat):
1211
 
    """Common logic for meta-dir based branch formats."""
1212
 
 
1213
 
    def _branch_class(self):
1214
 
        """What class to instantiate on open calls."""
1215
 
        raise NotImplementedError(self._branch_class)
 
1074
class BzrBranchFormat5(BranchFormat):
 
1075
    """Bzr branch format 5.
 
1076
 
 
1077
    This format has:
 
1078
     - a revision-history file.
 
1079
     - a format string
 
1080
     - a lock dir guarding the branch itself
 
1081
     - all of this stored in a branch/ subdirectory
 
1082
     - works with shared repositories.
 
1083
 
 
1084
    This format is new in bzr 0.8.
 
1085
    """
 
1086
 
 
1087
    def get_format_string(self):
 
1088
        """See BranchFormat.get_format_string()."""
 
1089
        return "Bazaar-NG branch format 5\n"
 
1090
 
 
1091
    def get_format_description(self):
 
1092
        """See BranchFormat.get_format_description()."""
 
1093
        return "Branch format 5"
 
1094
        
 
1095
    def initialize(self, a_bzrdir):
 
1096
        """Create a branch of this format in a_bzrdir."""
 
1097
        utf8_files = [('revision-history', ''),
 
1098
                      ('branch-name', ''),
 
1099
                      ]
 
1100
        return self._initialize_helper(a_bzrdir, utf8_files)
 
1101
 
 
1102
    def __init__(self):
 
1103
        super(BzrBranchFormat5, self).__init__()
 
1104
        self._matchingbzrdir = bzrdir.BzrDirMetaFormat1()
1216
1105
 
1217
1106
    def open(self, a_bzrdir, _found=False):
1218
 
        """Return the branch object for a_bzrdir.
 
1107
        """Return the branch object for a_bzrdir
1219
1108
 
1220
1109
        _found is a private parameter, do not use it. It is used to indicate
1221
1110
               if format probing has already be done.
1222
1111
        """
1223
1112
        if not _found:
1224
1113
            format = BranchFormat.find_format(a_bzrdir)
1225
 
            if format.__class__ != self.__class__:
1226
 
                raise AssertionError("wrong format %r found for %r" %
1227
 
                    (format, self))
 
1114
            assert format.__class__ == self.__class__
1228
1115
        try:
1229
1116
            transport = a_bzrdir.get_branch_transport(None)
1230
1117
            control_files = lockable_files.LockableFiles(transport, 'lock',
1231
1118
                                                         lockdir.LockDir)
1232
 
            return self._branch_class()(_format=self,
 
1119
            return BzrBranch5(_format=self,
1233
1120
                              _control_files=control_files,
1234
1121
                              a_bzrdir=a_bzrdir,
1235
1122
                              _repository=a_bzrdir.find_repository())
1236
 
        except errors.NoSuchFile:
1237
 
            raise errors.NotBranchError(path=transport.base)
1238
 
 
1239
 
    def __init__(self):
1240
 
        super(BranchFormatMetadir, self).__init__()
1241
 
        self._matchingbzrdir = bzrdir.BzrDirMetaFormat1()
1242
 
 
1243
 
    def supports_tags(self):
1244
 
        return True
1245
 
 
1246
 
 
1247
 
class BzrBranchFormat5(BranchFormatMetadir):
1248
 
    """Bzr branch format 5.
1249
 
 
1250
 
    This format has:
1251
 
     - a revision-history file.
1252
 
     - a format string
1253
 
     - a lock dir guarding the branch itself
1254
 
     - all of this stored in a branch/ subdirectory
1255
 
     - works with shared repositories.
1256
 
 
1257
 
    This format is new in bzr 0.8.
1258
 
    """
1259
 
 
1260
 
    def _branch_class(self):
1261
 
        return BzrBranch5
1262
 
 
1263
 
    def get_format_string(self):
1264
 
        """See BranchFormat.get_format_string()."""
1265
 
        return "Bazaar-NG branch format 5\n"
1266
 
 
1267
 
    def get_format_description(self):
1268
 
        """See BranchFormat.get_format_description()."""
1269
 
        return "Branch format 5"
1270
 
        
1271
 
    def initialize(self, a_bzrdir):
1272
 
        """Create a branch of this format in a_bzrdir."""
1273
 
        utf8_files = [('revision-history', ''),
1274
 
                      ('branch-name', ''),
1275
 
                      ]
1276
 
        return self._initialize_helper(a_bzrdir, utf8_files)
1277
 
 
1278
 
    def supports_tags(self):
1279
 
        return False
1280
 
 
1281
 
 
1282
 
class BzrBranchFormat6(BranchFormatMetadir):
 
1123
        except NoSuchFile:
 
1124
            raise NotBranchError(path=transport.base)
 
1125
 
 
1126
 
 
1127
class BzrBranchFormat6(BzrBranchFormat5):
1283
1128
    """Branch format with last-revision and tags.
1284
1129
 
1285
1130
    Unlike previous formats, this has no explicit revision history. Instead,
1290
1135
    and became the default in 0.91.
1291
1136
    """
1292
1137
 
1293
 
    def _branch_class(self):
1294
 
        return BzrBranch6
1295
 
 
1296
1138
    def get_format_string(self):
1297
1139
        """See BranchFormat.get_format_string()."""
1298
1140
        return "Bazaar Branch Format 6 (bzr 0.15)\n"
1304
1146
    def initialize(self, a_bzrdir):
1305
1147
        """Create a branch of this format in a_bzrdir."""
1306
1148
        utf8_files = [('last-revision', '0 null:\n'),
1307
 
                      ('branch.conf', ''),
1308
 
                      ('tags', ''),
1309
 
                      ]
1310
 
        return self._initialize_helper(a_bzrdir, utf8_files)
1311
 
 
1312
 
 
1313
 
class BzrBranchFormat7(BranchFormatMetadir):
1314
 
    """Branch format with last-revision, tags, and a stacked location pointer.
1315
 
 
1316
 
    The stacked location pointer is passed down to the repository and requires
1317
 
    a repository format with supports_external_lookups = True.
1318
 
 
1319
 
    This format was introduced in bzr 1.6.
1320
 
    """
1321
 
 
1322
 
    def _branch_class(self):
1323
 
        return BzrBranch7
1324
 
 
1325
 
    def get_format_string(self):
1326
 
        """See BranchFormat.get_format_string()."""
1327
 
        return "Bazaar Branch Format 7 (needs bzr 1.6)\n"
1328
 
 
1329
 
    def get_format_description(self):
1330
 
        """See BranchFormat.get_format_description()."""
1331
 
        return "Branch format 7"
1332
 
 
1333
 
    def initialize(self, a_bzrdir):
1334
 
        """Create a branch of this format in a_bzrdir."""
1335
 
        utf8_files = [('last-revision', '0 null:\n'),
1336
 
                      ('branch.conf', ''),
1337
 
                      ('tags', ''),
1338
 
                      ]
1339
 
        return self._initialize_helper(a_bzrdir, utf8_files)
1340
 
 
1341
 
    def __init__(self):
1342
 
        super(BzrBranchFormat7, self).__init__()
1343
 
        self._matchingbzrdir.repository_format = \
1344
 
            RepositoryFormatPackDevelopment1Subtree()
1345
 
 
1346
 
    def supports_stacking(self):
 
1149
                      ('branch-name', ''),
 
1150
                      ('branch.conf', ''),
 
1151
                      ('tags', ''),
 
1152
                      ]
 
1153
        return self._initialize_helper(a_bzrdir, utf8_files)
 
1154
 
 
1155
    def open(self, a_bzrdir, _found=False):
 
1156
        """Return the branch object for a_bzrdir
 
1157
 
 
1158
        _found is a private parameter, do not use it. It is used to indicate
 
1159
               if format probing has already be done.
 
1160
        """
 
1161
        if not _found:
 
1162
            format = BranchFormat.find_format(a_bzrdir)
 
1163
            assert format.__class__ == self.__class__
 
1164
        transport = a_bzrdir.get_branch_transport(None)
 
1165
        control_files = lockable_files.LockableFiles(transport, 'lock',
 
1166
                                                     lockdir.LockDir)
 
1167
        return BzrBranch6(_format=self,
 
1168
                          _control_files=control_files,
 
1169
                          a_bzrdir=a_bzrdir,
 
1170
                          _repository=a_bzrdir.find_repository())
 
1171
 
 
1172
    def supports_tags(self):
1347
1173
        return True
1348
1174
 
1349
1175
 
1365
1191
    def get_format_description(self):
1366
1192
        """See BranchFormat.get_format_description()."""
1367
1193
        return "Checkout reference format 1"
1368
 
 
 
1194
        
1369
1195
    def get_reference(self, a_bzrdir):
1370
1196
        """See BranchFormat.get_reference()."""
1371
1197
        transport = a_bzrdir.get_branch_transport(None)
1372
1198
        return transport.get('location').read()
1373
1199
 
1374
 
    def set_reference(self, a_bzrdir, to_branch):
1375
 
        """See BranchFormat.set_reference()."""
1376
 
        transport = a_bzrdir.get_branch_transport(None)
1377
 
        location = transport.put_bytes('location', to_branch.base)
1378
 
 
1379
1200
    def initialize(self, a_bzrdir, target_branch=None):
1380
1201
        """Create a branch of this format in a_bzrdir."""
1381
1202
        if target_branch is None:
1387
1208
        branch_transport.put_bytes('location',
1388
1209
            target_branch.bzrdir.root_transport.base)
1389
1210
        branch_transport.put_bytes('format', self.get_format_string())
1390
 
        return self.open(
1391
 
            a_bzrdir, _found=True,
1392
 
            possible_transports=[target_branch.bzrdir.root_transport])
 
1211
        return self.open(a_bzrdir, _found=True)
1393
1212
 
1394
1213
    def __init__(self):
1395
1214
        super(BranchReferenceFormat, self).__init__()
1405
1224
            # emit some sort of warning/error to the caller ?!
1406
1225
        return clone
1407
1226
 
1408
 
    def open(self, a_bzrdir, _found=False, location=None,
1409
 
             possible_transports=None):
 
1227
    def open(self, a_bzrdir, _found=False, location=None):
1410
1228
        """Return the branch that the branch reference in a_bzrdir points at.
1411
1229
 
1412
1230
        _found is a private parameter, do not use it. It is used to indicate
1414
1232
        """
1415
1233
        if not _found:
1416
1234
            format = BranchFormat.find_format(a_bzrdir)
1417
 
            if format.__class__ != self.__class__:
1418
 
                raise AssertionError("wrong format %r found for %r" %
1419
 
                    (format, self))
 
1235
            assert format.__class__ == self.__class__
1420
1236
        if location is None:
1421
1237
            location = self.get_reference(a_bzrdir)
1422
 
        real_bzrdir = bzrdir.BzrDir.open(
1423
 
            location, possible_transports=possible_transports)
 
1238
        real_bzrdir = bzrdir.BzrDir.open(location)
1424
1239
        result = real_bzrdir.open_branch()
1425
1240
        # this changes the behaviour of result.clone to create a new reference
1426
1241
        # rather than a copy of the content of the branch.
1438
1253
# and not independently creatable, so are not registered.
1439
1254
__format5 = BzrBranchFormat5()
1440
1255
__format6 = BzrBranchFormat6()
1441
 
__format7 = BzrBranchFormat7()
1442
1256
BranchFormat.register_format(__format5)
1443
1257
BranchFormat.register_format(BranchReferenceFormat())
1444
1258
BranchFormat.register_format(__format6)
1445
 
BranchFormat.register_format(__format7)
1446
1259
BranchFormat.set_default_format(__format6)
1447
1260
_legacy_formats = [BzrBranchFormat4(),
1448
1261
                   ]
1453
1266
    Note that it's "local" in the context of the filesystem; it doesn't
1454
1267
    really matter if it's on an nfs/smb/afs/coda/... share, as long as
1455
1268
    it's writable, and can be accessed via the normal filesystem API.
1456
 
 
1457
 
    :ivar _transport: Transport for file operations on this branch's 
1458
 
        control files, typically pointing to the .bzr/branch directory.
1459
 
    :ivar repository: Repository for this branch.
1460
 
    :ivar base: The url of the base directory for this branch; the one 
1461
 
        containing the .bzr directory.
1462
1269
    """
1463
1270
    
1464
1271
    def __init__(self, _format=None,
1465
1272
                 _control_files=None, a_bzrdir=None, _repository=None):
1466
1273
        """Create new branch object at a particular location."""
 
1274
        Branch.__init__(self)
1467
1275
        if a_bzrdir is None:
1468
1276
            raise ValueError('a_bzrdir must be supplied')
1469
1277
        else:
1470
1278
            self.bzrdir = a_bzrdir
 
1279
        # self._transport used to point to the directory containing the
 
1280
        # control directory, but was not used - now it's just the transport
 
1281
        # for the branch control files.  mbp 20070212
1471
1282
        self._base = self.bzrdir.transport.clone('..').base
1472
 
        # XXX: We should be able to just do
1473
 
        #   self.base = self.bzrdir.root_transport.base
1474
 
        # but this does not quite work yet -- mbp 20080522
1475
1283
        self._format = _format
1476
1284
        if _control_files is None:
1477
1285
            raise ValueError('BzrBranch _control_files is None')
1478
1286
        self.control_files = _control_files
1479
1287
        self._transport = _control_files._transport
1480
1288
        self.repository = _repository
1481
 
        Branch.__init__(self)
1482
1289
 
1483
1290
    def __str__(self):
1484
1291
        return '%s(%r)' % (self.__class__.__name__, self.base)
1491
1298
 
1492
1299
    base = property(_get_base, doc="The URL for the root of this branch.")
1493
1300
 
 
1301
    def abspath(self, name):
 
1302
        """See Branch.abspath."""
 
1303
        return self.control_files._transport.abspath(name)
 
1304
 
 
1305
 
 
1306
    @deprecated_method(zero_sixteen)
 
1307
    @needs_read_lock
 
1308
    def get_root_id(self):
 
1309
        """See Branch.get_root_id."""
 
1310
        tree = self.repository.revision_tree(self.last_revision())
 
1311
        return tree.inventory.root.file_id
 
1312
 
1494
1313
    def is_locked(self):
1495
1314
        return self.control_files.is_locked()
1496
1315
 
1540
1359
 
1541
1360
        This performs the actual writing to disk.
1542
1361
        It is intended to be called by BzrBranch5.set_revision_history."""
1543
 
        self._transport.put_bytes(
1544
 
            'revision-history', '\n'.join(history),
1545
 
            mode=self.bzrdir._get_file_mode())
 
1362
        self.control_files.put_bytes(
 
1363
            'revision-history', '\n'.join(history))
1546
1364
 
1547
1365
    @needs_write_lock
1548
1366
    def set_revision_history(self, rev_history):
1549
1367
        """See Branch.set_revision_history."""
1550
1368
        if 'evil' in debug.debug_flags:
1551
1369
            mutter_callsite(3, "set_revision_history scales with history.")
1552
 
        check_not_reserved_id = _mod_revision.check_not_reserved_id
1553
 
        for rev_id in rev_history:
1554
 
            check_not_reserved_id(rev_id)
1555
 
        if Branch.hooks['post_change_branch_tip']:
1556
 
            # Don't calculate the last_revision_info() if there are no hooks
1557
 
            # that will use it.
1558
 
            old_revno, old_revid = self.last_revision_info()
1559
 
        if len(rev_history) == 0:
1560
 
            revid = _mod_revision.NULL_REVISION
1561
 
        else:
1562
 
            revid = rev_history[-1]
1563
 
        self._run_pre_change_branch_tip_hooks(len(rev_history), revid)
 
1370
        self._clear_cached_state()
1564
1371
        self._write_revision_history(rev_history)
1565
 
        self._clear_cached_state()
1566
1372
        self._cache_revision_history(rev_history)
1567
1373
        for hook in Branch.hooks['set_rh']:
1568
1374
            hook(self, rev_history)
1569
 
        if Branch.hooks['post_change_branch_tip']:
1570
 
            self._run_post_change_branch_tip_hooks(old_revno, old_revid)
1571
1375
 
1572
 
    def _run_pre_change_branch_tip_hooks(self, new_revno, new_revid):
1573
 
        """Run the pre_change_branch_tip hooks."""
1574
 
        hooks = Branch.hooks['pre_change_branch_tip']
1575
 
        if not hooks:
1576
 
            return
1577
 
        old_revno, old_revid = self.last_revision_info()
1578
 
        params = ChangeBranchTipParams(
1579
 
            self, old_revno, new_revno, old_revid, new_revid)
1580
 
        for hook in hooks:
1581
 
            try:
1582
 
                hook(params)
1583
 
            except errors.TipChangeRejected:
1584
 
                raise
1585
 
            except Exception:
1586
 
                exc_info = sys.exc_info()
1587
 
                hook_name = Branch.hooks.get_hook_name(hook)
1588
 
                raise errors.HookFailed(
1589
 
                    'pre_change_branch_tip', hook_name, exc_info)
1590
 
 
1591
 
    def _run_post_change_branch_tip_hooks(self, old_revno, old_revid):
1592
 
        """Run the post_change_branch_tip hooks."""
1593
 
        hooks = Branch.hooks['post_change_branch_tip']
1594
 
        if not hooks:
1595
 
            return
1596
 
        new_revno, new_revid = self.last_revision_info()
1597
 
        params = ChangeBranchTipParams(
1598
 
            self, old_revno, new_revno, old_revid, new_revid)
1599
 
        for hook in hooks:
1600
 
            hook(params)
1601
 
 
1602
1376
    @needs_write_lock
1603
1377
    def set_last_revision_info(self, revno, revision_id):
1604
1378
        """Set the last revision of this branch.
1611
1385
        configured to check constraints on history, in which case this may not
1612
1386
        be permitted.
1613
1387
        """
1614
 
        revision_id = _mod_revision.ensure_null(revision_id)
1615
 
        # this old format stores the full history, but this api doesn't
1616
 
        # provide it, so we must generate, and might as well check it's
1617
 
        # correct
1618
1388
        history = self._lefthand_history(revision_id)
1619
 
        if len(history) != revno:
1620
 
            raise AssertionError('%d != %d' % (len(history), revno))
 
1389
        assert len(history) == revno, '%d != %d' % (len(history), revno)
1621
1390
        self.set_revision_history(history)
1622
1391
 
1623
1392
    def _gen_revision_history(self):
1624
 
        history = self._transport.get_bytes('revision-history').split('\n')
 
1393
        history = self.control_files.get('revision-history').read().split('\n')
1625
1394
        if history[-1:] == ['']:
1626
1395
            # There shouldn't be a trailing newline, but just in case.
1627
1396
            history.pop()
1632
1401
        if 'evil' in debug.debug_flags:
1633
1402
            mutter_callsite(4, "_lefthand_history scales with history.")
1634
1403
        # stop_revision must be a descendant of last_revision
1635
 
        graph = self.repository.get_graph()
1636
 
        if last_rev is not None:
1637
 
            if not graph.is_ancestor(last_rev, revision_id):
1638
 
                # our previous tip is not merged into stop_revision
1639
 
                raise errors.DivergedBranches(self, other_branch)
 
1404
        stop_graph = self.repository.get_revision_graph(revision_id)
 
1405
        if (last_rev is not None and last_rev != _mod_revision.NULL_REVISION
 
1406
            and last_rev not in stop_graph):
 
1407
            # our previous tip is not merged into stop_revision
 
1408
            raise errors.DivergedBranches(self, other_branch)
1640
1409
        # make a new revision history from the graph
1641
 
        parents_map = graph.get_parent_map([revision_id])
1642
 
        if revision_id not in parents_map:
1643
 
            raise errors.NoSuchRevision(self, revision_id)
1644
1410
        current_rev_id = revision_id
1645
1411
        new_history = []
1646
 
        check_not_reserved_id = _mod_revision.check_not_reserved_id
1647
 
        # Do not include ghosts or graph origin in revision_history
1648
 
        while (current_rev_id in parents_map and
1649
 
               len(parents_map[current_rev_id]) > 0):
1650
 
            check_not_reserved_id(current_rev_id)
 
1412
        while current_rev_id not in (None, _mod_revision.NULL_REVISION):
1651
1413
            new_history.append(current_rev_id)
1652
 
            current_rev_id = parents_map[current_rev_id][0]
1653
 
            parents_map = graph.get_parent_map([current_rev_id])
 
1414
            current_rev_id_parents = stop_graph[current_rev_id]
 
1415
            try:
 
1416
                current_rev_id = current_rev_id_parents[0]
 
1417
            except IndexError:
 
1418
                current_rev_id = None
1654
1419
        new_history.reverse()
1655
1420
        return new_history
1656
1421
 
1668
1433
        self.set_revision_history(self._lefthand_history(revision_id,
1669
1434
            last_rev, other_branch))
1670
1435
 
 
1436
    @needs_write_lock
 
1437
    def update_revisions(self, other, stop_revision=None):
 
1438
        """See Branch.update_revisions."""
 
1439
        other.lock_read()
 
1440
        try:
 
1441
            if stop_revision is None:
 
1442
                stop_revision = other.last_revision()
 
1443
                if stop_revision is None:
 
1444
                    # if there are no commits, we're done.
 
1445
                    return
 
1446
            # whats the current last revision, before we fetch [and change it
 
1447
            # possibly]
 
1448
            last_rev = _mod_revision.ensure_null(self.last_revision())
 
1449
            # we fetch here so that we don't process data twice in the common
 
1450
            # case of having something to pull, and so that the check for 
 
1451
            # already merged can operate on the just fetched graph, which will
 
1452
            # be cached in memory.
 
1453
            self.fetch(other, stop_revision)
 
1454
            if self.repository.get_graph().is_ancestor(stop_revision,
 
1455
                                                       last_rev):
 
1456
                return
 
1457
            self.generate_revision_history(stop_revision, last_rev=last_rev,
 
1458
                other_branch=other)
 
1459
        finally:
 
1460
            other.unlock()
 
1461
 
1671
1462
    def basis_tree(self):
1672
1463
        """See Branch.basis_tree."""
1673
1464
        return self.repository.revision_tree(self.last_revision())
1674
1465
 
1675
1466
    @needs_write_lock
1676
1467
    def pull(self, source, overwrite=False, stop_revision=None,
1677
 
             _hook_master=None, run_hooks=True, possible_transports=None,
1678
 
             _override_hook_target=None):
 
1468
             _hook_master=None, run_hooks=True, possible_transports=None):
1679
1469
        """See Branch.pull.
1680
1470
 
1681
1471
        :param _hook_master: Private parameter - set the branch to 
1682
 
            be supplied as the master to pull hooks.
 
1472
            be supplied as the master to push hooks.
1683
1473
        :param run_hooks: Private parameter - if false, this branch
1684
1474
            is being called because it's the master of the primary branch,
1685
1475
            so it should not run its hooks.
1686
 
        :param _override_hook_target: Private parameter - set the branch to be
1687
 
            supplied as the target_branch to pull hooks.
1688
1476
        """
1689
1477
        result = PullResult()
1690
1478
        result.source_branch = source
1691
 
        if _override_hook_target is None:
1692
 
            result.target_branch = self
1693
 
        else:
1694
 
            result.target_branch = _override_hook_target
 
1479
        result.target_branch = self
1695
1480
        source.lock_read()
1696
1481
        try:
1697
 
            # We assume that during 'pull' the local repository is closer than
1698
 
            # the remote one.
1699
 
            graph = self.repository.get_graph(source.repository)
1700
1482
            result.old_revno, result.old_revid = self.last_revision_info()
1701
 
            self.update_revisions(source, stop_revision, overwrite=overwrite,
1702
 
                                  graph=graph)
 
1483
            try:
 
1484
                self.update_revisions(source, stop_revision)
 
1485
            except DivergedBranches:
 
1486
                if not overwrite:
 
1487
                    raise
 
1488
            if overwrite:
 
1489
                if stop_revision is None:
 
1490
                    stop_revision = source.last_revision()
 
1491
                self.generate_revision_history(stop_revision)
1703
1492
            result.tag_conflicts = source.tags.merge_to(self.tags, overwrite)
1704
1493
            result.new_revno, result.new_revid = self.last_revision_info()
1705
1494
            if _hook_master:
1706
1495
                result.master_branch = _hook_master
1707
 
                result.local_branch = result.target_branch
 
1496
                result.local_branch = self
1708
1497
            else:
1709
 
                result.master_branch = result.target_branch
 
1498
                result.master_branch = self
1710
1499
                result.local_branch = None
1711
1500
            if run_hooks:
1712
1501
                for hook in Branch.hooks['post_pull']:
1719
1508
        _locs = ['parent', 'pull', 'x-pull']
1720
1509
        for l in _locs:
1721
1510
            try:
1722
 
                return self._transport.get_bytes(l).strip('\n')
1723
 
            except errors.NoSuchFile:
 
1511
                return self.control_files.get(l).read().strip('\n')
 
1512
            except NoSuchFile:
1724
1513
                pass
1725
1514
        return None
1726
1515
 
1802
1591
        result.source_branch = self
1803
1592
        result.target_branch = target
1804
1593
        result.old_revno, result.old_revid = target.last_revision_info()
1805
 
 
1806
 
        # We assume that during 'push' this repository is closer than
1807
 
        # the target.
1808
 
        graph = self.repository.get_graph(target.repository)
1809
 
        target.update_revisions(self, stop_revision, overwrite=overwrite,
1810
 
                                graph=graph)
 
1594
        try:
 
1595
            target.update_revisions(self, stop_revision)
 
1596
        except DivergedBranches:
 
1597
            if not overwrite:
 
1598
                raise
 
1599
        if overwrite:
 
1600
            target.set_revision_history(self.revision_history())
1811
1601
        result.tag_conflicts = self.tags.merge_to(target.tags, overwrite)
1812
1602
        result.new_revno, result.new_revid = target.last_revision_info()
1813
1603
        return result
1814
1604
 
1815
1605
    def get_parent(self):
1816
1606
        """See Branch.get_parent."""
 
1607
 
 
1608
        assert self.base[-1] == '/'
1817
1609
        parent = self._get_parent_location()
1818
1610
        if parent is None:
1819
1611
            return parent
1826
1618
        except errors.InvalidURLJoin, e:
1827
1619
            raise errors.InaccessibleParent(parent, self.base)
1828
1620
 
1829
 
    def get_stacked_on_url(self):
1830
 
        raise errors.UnstackableBranchFormat(self._format, self.base)
1831
 
 
1832
1621
    def set_push_location(self, location):
1833
1622
        """See Branch.set_push_location."""
1834
1623
        self.get_config().set_user_option(
1841
1630
        # TODO: Maybe delete old location files?
1842
1631
        # URLs should never be unicode, even on the local fs,
1843
1632
        # FIXUP this and get_parent in a future branch format bump:
1844
 
        # read and rewrite the file. RBC 20060125
 
1633
        # read and rewrite the file, and have the new format code read
 
1634
        # using .get not .get_utf8. RBC 20060125
1845
1635
        if url is not None:
1846
1636
            if isinstance(url, unicode):
1847
 
                try:
 
1637
                try: 
1848
1638
                    url = url.encode('ascii')
1849
1639
                except UnicodeEncodeError:
1850
1640
                    raise errors.InvalidURL(url,
1855
1645
 
1856
1646
    def _set_parent_location(self, url):
1857
1647
        if url is None:
1858
 
            self._transport.delete('parent')
 
1648
            self.control_files._transport.delete('parent')
1859
1649
        else:
1860
 
            self._transport.put_bytes('parent', url + '\n',
1861
 
                mode=self.bzrdir._get_file_mode())
1862
 
 
1863
 
    def set_stacked_on_url(self, url):
1864
 
        raise errors.UnstackableBranchFormat(self._format, self.base)
 
1650
            assert isinstance(url, str)
 
1651
            self.control_files.put_bytes('parent', url + '\n')
1865
1652
 
1866
1653
 
1867
1654
class BzrBranch5(BzrBranch):
1870
1657
    It has support for a master_branch which is the data for bound branches.
1871
1658
    """
1872
1659
 
 
1660
    def __init__(self,
 
1661
                 _format,
 
1662
                 _control_files,
 
1663
                 a_bzrdir,
 
1664
                 _repository):
 
1665
        super(BzrBranch5, self).__init__(_format=_format,
 
1666
                                         _control_files=_control_files,
 
1667
                                         a_bzrdir=a_bzrdir,
 
1668
                                         _repository=_repository)
 
1669
        
1873
1670
    @needs_write_lock
1874
1671
    def pull(self, source, overwrite=False, stop_revision=None,
1875
 
             run_hooks=True, possible_transports=None,
1876
 
             _override_hook_target=None):
 
1672
             run_hooks=True, possible_transports=None):
1877
1673
        """Pull from source into self, updating my master if any.
1878
1674
        
1879
1675
        :param run_hooks: Private parameter - if false, this branch
1893
1689
                    run_hooks=False)
1894
1690
            return super(BzrBranch5, self).pull(source, overwrite,
1895
1691
                stop_revision, _hook_master=master_branch,
1896
 
                run_hooks=run_hooks,
1897
 
                _override_hook_target=_override_hook_target)
 
1692
                run_hooks=run_hooks)
1898
1693
        finally:
1899
1694
            if master_branch:
1900
1695
                master_branch.unlock()
1901
1696
 
1902
1697
    def get_bound_location(self):
1903
1698
        try:
1904
 
            return self._transport.get_bytes('bound')[:-1]
 
1699
            return self.control_files.get_utf8('bound').read()[:-1]
1905
1700
        except errors.NoSuchFile:
1906
1701
            return None
1907
1702
 
1933
1728
        :param location: URL to the target branch
1934
1729
        """
1935
1730
        if location:
1936
 
            self._transport.put_bytes('bound', location+'\n',
1937
 
                mode=self.bzrdir._get_file_mode())
 
1731
            self.control_files.put_utf8('bound', location+'\n')
1938
1732
        else:
1939
1733
            try:
1940
 
                self._transport.delete('bound')
1941
 
            except errors.NoSuchFile:
 
1734
                self.control_files._transport.delete('bound')
 
1735
            except NoSuchFile:
1942
1736
                return False
1943
1737
            return True
1944
1738
 
1966
1760
        # last_rev is not in the other_last_rev history, AND
1967
1761
        # other_last_rev is not in our history, and do it without pulling
1968
1762
        # history around
 
1763
        last_rev = _mod_revision.ensure_null(self.last_revision())
 
1764
        if last_rev != _mod_revision.NULL_REVISION:
 
1765
            other.lock_read()
 
1766
            try:
 
1767
                other_last_rev = other.last_revision()
 
1768
                if not _mod_revision.is_null(other_last_rev):
 
1769
                    # neither branch is new, we have to do some work to
 
1770
                    # ascertain diversion.
 
1771
                    remote_graph = other.repository.get_revision_graph(
 
1772
                        other_last_rev)
 
1773
                    local_graph = self.repository.get_revision_graph(last_rev)
 
1774
                    if (last_rev not in remote_graph and
 
1775
                        other_last_rev not in local_graph):
 
1776
                        raise errors.DivergedBranches(self, other)
 
1777
            finally:
 
1778
                other.unlock()
1969
1779
        self.set_bound_location(other.base)
1970
1780
 
1971
1781
    @needs_write_lock
1991
1801
        return None
1992
1802
 
1993
1803
 
1994
 
class BzrBranch7(BzrBranch5):
1995
 
    """A branch with support for a fallback repository."""
1996
 
 
1997
 
    def _get_fallback_repository(self, url):
1998
 
        """Get the repository we fallback to at url."""
1999
 
        url = urlutils.join(self.base, url)
2000
 
        return bzrdir.BzrDir.open(url).open_branch().repository
2001
 
 
2002
 
    def _activate_fallback_location(self, url):
2003
 
        """Activate the branch/repository from url as a fallback repository."""
2004
 
        self.repository.add_fallback_repository(
2005
 
            self._get_fallback_repository(url))
2006
 
 
2007
 
    def _open_hook(self):
 
1804
class BzrBranchExperimental(BzrBranch5):
 
1805
    """Bzr experimental branch format
 
1806
 
 
1807
    This format has:
 
1808
     - a revision-history file.
 
1809
     - a format string
 
1810
     - a lock dir guarding the branch itself
 
1811
     - all of this stored in a branch/ subdirectory
 
1812
     - works with shared repositories.
 
1813
     - a tag dictionary in the branch
 
1814
 
 
1815
    This format is new in bzr 0.15, but shouldn't be used for real data, 
 
1816
    only for testing.
 
1817
 
 
1818
    This class acts as it's own BranchFormat.
 
1819
    """
 
1820
 
 
1821
    _matchingbzrdir = bzrdir.BzrDirMetaFormat1()
 
1822
 
 
1823
    @classmethod
 
1824
    def get_format_string(cls):
 
1825
        """See BranchFormat.get_format_string()."""
 
1826
        return "Bazaar-NG branch format experimental\n"
 
1827
 
 
1828
    @classmethod
 
1829
    def get_format_description(cls):
 
1830
        """See BranchFormat.get_format_description()."""
 
1831
        return "Experimental branch format"
 
1832
 
 
1833
    @classmethod
 
1834
    def get_reference(cls, a_bzrdir):
 
1835
        """Get the target reference of the branch in a_bzrdir.
 
1836
 
 
1837
        format probing must have been completed before calling
 
1838
        this method - it is assumed that the format of the branch
 
1839
        in a_bzrdir is correct.
 
1840
 
 
1841
        :param a_bzrdir: The bzrdir to get the branch data from.
 
1842
        :return: None if the branch is not a reference branch.
 
1843
        """
 
1844
        return None
 
1845
 
 
1846
    @classmethod
 
1847
    def _initialize_control_files(cls, a_bzrdir, utf8_files, lock_filename,
 
1848
            lock_class):
 
1849
        branch_transport = a_bzrdir.get_branch_transport(cls)
 
1850
        control_files = lockable_files.LockableFiles(branch_transport,
 
1851
            lock_filename, lock_class)
 
1852
        control_files.create_lock()
 
1853
        control_files.lock_write()
2008
1854
        try:
2009
 
            url = self.get_stacked_on_url()
2010
 
        except (errors.UnstackableRepositoryFormat, errors.NotStacked,
2011
 
            errors.UnstackableBranchFormat):
2012
 
            pass
2013
 
        else:
2014
 
            self._activate_fallback_location(url)
2015
 
 
2016
 
    def _check_stackable_repo(self):
2017
 
        if not self.repository._format.supports_external_lookups:
2018
 
            raise errors.UnstackableRepositoryFormat(self.repository._format,
2019
 
                self.repository.base)
2020
 
 
2021
 
    def __init__(self, *args, **kwargs):
2022
 
        super(BzrBranch7, self).__init__(*args, **kwargs)
2023
 
        self._last_revision_info_cache = None
2024
 
        self._partial_revision_history_cache = []
2025
 
 
2026
 
    def _clear_cached_state(self):
2027
 
        super(BzrBranch7, self)._clear_cached_state()
2028
 
        self._last_revision_info_cache = None
2029
 
        self._partial_revision_history_cache = []
2030
 
 
2031
 
    def _last_revision_info(self):
2032
 
        revision_string = self._transport.get_bytes('last-revision')
 
1855
            for filename, content in utf8_files:
 
1856
                control_files.put_utf8(filename, content)
 
1857
        finally:
 
1858
            control_files.unlock()
 
1859
        
 
1860
    @classmethod
 
1861
    def initialize(cls, a_bzrdir):
 
1862
        """Create a branch of this format in a_bzrdir."""
 
1863
        utf8_files = [('format', cls.get_format_string()),
 
1864
                      ('revision-history', ''),
 
1865
                      ('branch-name', ''),
 
1866
                      ('tags', ''),
 
1867
                      ]
 
1868
        cls._initialize_control_files(a_bzrdir, utf8_files,
 
1869
            'lock', lockdir.LockDir)
 
1870
        return cls.open(a_bzrdir, _found=True)
 
1871
 
 
1872
    @classmethod
 
1873
    def open(cls, a_bzrdir, _found=False):
 
1874
        """Return the branch object for a_bzrdir
 
1875
 
 
1876
        _found is a private parameter, do not use it. It is used to indicate
 
1877
               if format probing has already be done.
 
1878
        """
 
1879
        if not _found:
 
1880
            format = BranchFormat.find_format(a_bzrdir)
 
1881
            assert format.__class__ == cls
 
1882
        transport = a_bzrdir.get_branch_transport(None)
 
1883
        control_files = lockable_files.LockableFiles(transport, 'lock',
 
1884
                                                     lockdir.LockDir)
 
1885
        return cls(_format=cls,
 
1886
            _control_files=control_files,
 
1887
            a_bzrdir=a_bzrdir,
 
1888
            _repository=a_bzrdir.find_repository())
 
1889
 
 
1890
    @classmethod
 
1891
    def is_supported(cls):
 
1892
        return True
 
1893
 
 
1894
    def _make_tags(self):
 
1895
        return BasicTags(self)
 
1896
 
 
1897
    @classmethod
 
1898
    def supports_tags(cls):
 
1899
        return True
 
1900
 
 
1901
 
 
1902
BranchFormat.register_format(BzrBranchExperimental)
 
1903
 
 
1904
 
 
1905
class BzrBranch6(BzrBranch5):
 
1906
 
 
1907
    @needs_read_lock
 
1908
    def last_revision_info(self):
 
1909
        revision_string = self.control_files.get('last-revision').read()
2033
1910
        revno, revision_id = revision_string.rstrip('\n').split(' ', 1)
2034
1911
        revision_id = cache_utf8.get_cached_utf8(revision_id)
2035
1912
        revno = int(revno)
2036
1913
        return revno, revision_id
2037
1914
 
 
1915
    def last_revision(self):
 
1916
        """Return last revision id, or None"""
 
1917
        revision_id = self.last_revision_info()[1]
 
1918
        return revision_id
 
1919
 
2038
1920
    def _write_last_revision_info(self, revno, revision_id):
2039
1921
        """Simply write out the revision id, with no checks.
2040
1922
 
2044
1926
        Intended to be called by set_last_revision_info and
2045
1927
        _write_revision_history.
2046
1928
        """
2047
 
        revision_id = _mod_revision.ensure_null(revision_id)
 
1929
        if revision_id is None:
 
1930
            revision_id = 'null:'
2048
1931
        out_string = '%d %s\n' % (revno, revision_id)
2049
 
        self._transport.put_bytes('last-revision', out_string,
2050
 
            mode=self.bzrdir._get_file_mode())
 
1932
        self.control_files.put_bytes('last-revision', out_string)
2051
1933
 
2052
1934
    @needs_write_lock
2053
1935
    def set_last_revision_info(self, revno, revision_id):
2054
 
        revision_id = _mod_revision.ensure_null(revision_id)
2055
 
        old_revno, old_revid = self.last_revision_info()
2056
1936
        if self._get_append_revisions_only():
2057
1937
            self._check_history_violation(revision_id)
2058
 
        self._run_pre_change_branch_tip_hooks(revno, revision_id)
2059
1938
        self._write_last_revision_info(revno, revision_id)
2060
1939
        self._clear_cached_state()
2061
 
        self._last_revision_info_cache = revno, revision_id
2062
 
        self._run_post_change_branch_tip_hooks(old_revno, old_revid)
2063
1940
 
2064
1941
    def _check_history_violation(self, revision_id):
2065
1942
        last_revision = _mod_revision.ensure_null(self.last_revision())
2071
1948
    def _gen_revision_history(self):
2072
1949
        """Generate the revision history from last revision
2073
1950
        """
2074
 
        last_revno, last_revision = self.last_revision_info()
2075
 
        self._extend_partial_history(stop_index=last_revno-1)
2076
 
        return list(reversed(self._partial_revision_history_cache))
2077
 
 
2078
 
    def _extend_partial_history(self, stop_index=None, stop_revision=None):
2079
 
        """Extend the partial history to include a given index
2080
 
 
2081
 
        If a stop_index is supplied, stop when that index has been reached.
2082
 
        If a stop_revision is supplied, stop when that revision is
2083
 
        encountered.  Otherwise, stop when the beginning of history is
2084
 
        reached.
2085
 
 
2086
 
        :param stop_index: The index which should be present.  When it is
2087
 
            present, history extension will stop.
2088
 
        :param revision_id: The revision id which should be present.  When
2089
 
            it is encountered, history extension will stop.
2090
 
        """
2091
 
        repo = self.repository
2092
 
        if len(self._partial_revision_history_cache) == 0:
2093
 
            iterator = repo.iter_reverse_revision_history(self.last_revision())
2094
 
        else:
2095
 
            start_revision = self._partial_revision_history_cache[-1]
2096
 
            iterator = repo.iter_reverse_revision_history(start_revision)
2097
 
            #skip the last revision in the list
2098
 
            next_revision = iterator.next()
2099
 
        for revision_id in iterator:
2100
 
            self._partial_revision_history_cache.append(revision_id)
2101
 
            if (stop_index is not None and
2102
 
                len(self._partial_revision_history_cache) > stop_index):
2103
 
                break
2104
 
            if revision_id == stop_revision:
2105
 
                break
 
1951
        history = list(self.repository.iter_reverse_revision_history(
 
1952
            self.last_revision()))
 
1953
        history.reverse()
 
1954
        return history
2106
1955
 
2107
1956
    def _write_revision_history(self, history):
2108
1957
        """Factored out of set_revision_history.
2168
2017
        """See Branch.get_old_bound_location"""
2169
2018
        return self._get_bound_location(False)
2170
2019
 
2171
 
    def get_stacked_on_url(self):
2172
 
        self._check_stackable_repo()
2173
 
        stacked_url = self._get_config_location('stacked_on_location')
2174
 
        if stacked_url is None:
2175
 
            raise errors.NotStacked(self)
2176
 
        return stacked_url
2177
 
 
2178
2020
    def set_append_revisions_only(self, enabled):
2179
2021
        if enabled:
2180
2022
            value = 'True'
2183
2025
        self.get_config().set_user_option('append_revisions_only', value,
2184
2026
            warn_masked=True)
2185
2027
 
2186
 
    def set_stacked_on_url(self, url):
2187
 
        self._check_stackable_repo()
2188
 
        if not url:
2189
 
            try:
2190
 
                old_url = self.get_stacked_on_url()
2191
 
            except (errors.NotStacked, errors.UnstackableBranchFormat,
2192
 
                errors.UnstackableRepositoryFormat):
2193
 
                return
2194
 
            url = ''
2195
 
            # repositories don't offer an interface to remove fallback
2196
 
            # repositories today; take the conceptually simpler option and just
2197
 
            # reopen it.
2198
 
            self.repository = self.bzrdir.find_repository()
2199
 
            # for every revision reference the branch has, ensure it is pulled
2200
 
            # in.
2201
 
            source_repository = self._get_fallback_repository(old_url)
2202
 
            for revision_id in chain([self.last_revision()],
2203
 
                self.tags.get_reverse_tag_dict()):
2204
 
                self.repository.fetch(source_repository, revision_id,
2205
 
                    find_ghosts=True)
2206
 
        else:
2207
 
            self._activate_fallback_location(url)
2208
 
        # write this out after the repository is stacked to avoid setting a
2209
 
        # stacked config that doesn't work.
2210
 
        self._set_config_location('stacked_on_location', url)
2211
 
 
2212
2028
    def _get_append_revisions_only(self):
2213
2029
        value = self.get_config().get_user_option('append_revisions_only')
2214
2030
        return value == 'True'
2245
2061
    def _make_tags(self):
2246
2062
        return BasicTags(self)
2247
2063
 
2248
 
    @needs_write_lock
2249
 
    def generate_revision_history(self, revision_id, last_rev=None,
2250
 
                                  other_branch=None):
2251
 
        """See BzrBranch5.generate_revision_history"""
2252
 
        history = self._lefthand_history(revision_id, last_rev, other_branch)
2253
 
        revno = len(history)
2254
 
        self.set_last_revision_info(revno, revision_id)
2255
 
 
2256
 
    @needs_read_lock
2257
 
    def get_rev_id(self, revno, history=None):
2258
 
        """Find the revision id of the specified revno."""
2259
 
        if revno == 0:
2260
 
            return _mod_revision.NULL_REVISION
2261
 
 
2262
 
        last_revno, last_revision_id = self.last_revision_info()
2263
 
        if revno <= 0 or revno > last_revno:
2264
 
            raise errors.NoSuchRevision(self, revno)
2265
 
 
2266
 
        if history is not None:
2267
 
            return history[revno - 1]
2268
 
 
2269
 
        index = last_revno - revno
2270
 
        if len(self._partial_revision_history_cache) <= index:
2271
 
            self._extend_partial_history(stop_index=index)
2272
 
        if len(self._partial_revision_history_cache) > index:
2273
 
            return self._partial_revision_history_cache[index]
2274
 
        else:
2275
 
            raise errors.NoSuchRevision(self, revno)
2276
 
 
2277
 
    @needs_read_lock
2278
 
    def revision_id_to_revno(self, revision_id):
2279
 
        """Given a revision id, return its revno"""
2280
 
        if _mod_revision.is_null(revision_id):
2281
 
            return 0
2282
 
        try:
2283
 
            index = self._partial_revision_history_cache.index(revision_id)
2284
 
        except ValueError:
2285
 
            self._extend_partial_history(stop_revision=revision_id)
2286
 
            index = len(self._partial_revision_history_cache) - 1
2287
 
            if self._partial_revision_history_cache[index] != revision_id:
2288
 
                raise errors.NoSuchRevision(self, revision_id)
2289
 
        return self.revno() - index
2290
 
 
2291
 
 
2292
 
class BzrBranch6(BzrBranch7):
2293
 
    """See BzrBranchFormat6 for the capabilities of this branch.
2294
 
 
2295
 
    This subclass of BzrBranch7 disables the new features BzrBranch7 added,
2296
 
    i.e. stacking.
2297
 
    """
2298
 
 
2299
 
    def get_stacked_on_url(self):
2300
 
        raise errors.UnstackableBranchFormat(self._format, self.base)
2301
 
 
2302
 
    def set_stacked_on_url(self, url):
2303
 
        raise errors.UnstackableBranchFormat(self._format, self.base)
2304
 
 
2305
2064
 
2306
2065
######################################################################
2307
2066
# results of operations
2325
2084
    :ivar old_revid: Tip revision id before pull.
2326
2085
    :ivar new_revid: Tip revision id after pull.
2327
2086
    :ivar source_branch: Source (local) branch object.
2328
 
    :ivar master_branch: Master branch of the target, or the target if no
2329
 
        Master
2330
 
    :ivar local_branch: target branch if there is a Master, else None
 
2087
    :ivar master_branch: Master branch of the target, or None.
2331
2088
    :ivar target_branch: Target/destination branch object.
2332
 
    :ivar tag_conflicts: A list of tag conflicts, see BasicTags.merge_to
2333
2089
    """
2334
2090
 
2335
2091
    def __int__(self):
2337
2093
        return self.new_revno - self.old_revno
2338
2094
 
2339
2095
    def report(self, to_file):
2340
 
        if not is_quiet():
2341
 
            if self.old_revid == self.new_revid:
2342
 
                to_file.write('No revisions to pull.\n')
2343
 
            else:
2344
 
                to_file.write('Now on revision %d.\n' % self.new_revno)
 
2096
        if self.old_revid == self.new_revid:
 
2097
            to_file.write('No revisions to pull.\n')
 
2098
        else:
 
2099
            to_file.write('Now on revision %d.\n' % self.new_revno)
2345
2100
        self._show_tag_conficts(to_file)
2346
2101
 
2347
2102
 
2399
2154
        new_branch = format.open(branch.bzrdir, _found=True)
2400
2155
 
2401
2156
        # Copy source data into target
2402
 
        new_branch._write_last_revision_info(*branch.last_revision_info())
 
2157
        new_branch.set_last_revision_info(*branch.last_revision_info())
2403
2158
        new_branch.set_parent(branch.get_parent())
2404
2159
        new_branch.set_bound_location(branch.get_bound_location())
2405
2160
        new_branch.set_push_location(branch.get_push_location())
2408
2163
        new_branch.tags._set_tag_dict({})
2409
2164
 
2410
2165
        # Copying done; now update target format
2411
 
        new_branch._transport.put_bytes('format',
2412
 
            format.get_format_string(),
2413
 
            mode=new_branch.bzrdir._get_file_mode())
 
2166
        new_branch.control_files.put_utf8('format',
 
2167
            format.get_format_string())
2414
2168
 
2415
2169
        # Clean up old files
2416
 
        new_branch._transport.delete('revision-history')
 
2170
        new_branch.control_files._transport.delete('revision-history')
2417
2171
        try:
2418
2172
            branch.set_parent(None)
2419
 
        except errors.NoSuchFile:
 
2173
        except NoSuchFile:
2420
2174
            pass
2421
2175
        branch.set_bound_location(None)
2422
 
 
2423
 
 
2424
 
class Converter6to7(object):
2425
 
    """Perform an in-place upgrade of format 6 to format 7"""
2426
 
 
2427
 
    def convert(self, branch):
2428
 
        format = BzrBranchFormat7()
2429
 
        branch._set_config_location('stacked_on_location', '')
2430
 
        # update target format
2431
 
        branch._transport.put_bytes('format', format.get_format_string())