~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: 2011-04-29 11:07:14 UTC
  • mfrom: (5813.1.1 realname-can-be-empty)
  • Revision ID: pqm@pqm.ubuntu.com-20110429110714-wr9f71ea9600lvb6
(jelmer) Allow realname to be empty in tests. (Jelmer Vernooij)

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
# Copyright (C) 2005-2010 Canonical Ltd
 
1
# Copyright (C) 2005-2011 Canonical Ltd
2
2
#
3
3
# This program is free software; you can redistribute it and/or modify
4
4
# it under the terms of the GNU General Public License as published by
25
25
        bzrdir,
26
26
        cache_utf8,
27
27
        config as _mod_config,
28
 
        controldir,
29
28
        debug,
30
29
        errors,
 
30
        fetch,
 
31
        graph as _mod_graph,
31
32
        lockdir,
32
33
        lockable_files,
33
34
        remote,
41
42
        urlutils,
42
43
        )
43
44
from bzrlib.config import BranchConfig, TransportConfig
44
 
from bzrlib.repofmt.pack_repo import RepositoryFormatKnitPack5RichRoot
45
45
from bzrlib.tag import (
46
46
    BasicTags,
47
47
    DisabledTags,
48
48
    )
49
49
""")
50
50
 
51
 
from bzrlib.decorators import needs_read_lock, needs_write_lock, only_raises
52
 
from bzrlib.hooks import HookPoint, Hooks
 
51
from bzrlib import (
 
52
    controldir,
 
53
    )
 
54
from bzrlib.decorators import (
 
55
    needs_read_lock,
 
56
    needs_write_lock,
 
57
    only_raises,
 
58
    )
 
59
from bzrlib.hooks import Hooks
53
60
from bzrlib.inter import InterObject
54
61
from bzrlib.lock import _RelockDebugMixin, LogicalLockResult
55
62
from bzrlib import registry
71
78
    :ivar base:
72
79
        Base directory/url of the branch; using control_url and
73
80
        control_transport is more standardized.
74
 
 
75
 
    hooks: An instance of BranchHooks.
 
81
    :ivar hooks: An instance of BranchHooks.
 
82
    :ivar _master_branch_cache: cached result of get_master_branch, see
 
83
        _clear_cached_state.
76
84
    """
77
85
    # this is really an instance variable - FIXME move it there
78
86
    # - RBC 20060112
94
102
        self._partial_revision_history_cache = []
95
103
        self._tags_bytes = None
96
104
        self._last_revision_info_cache = None
 
105
        self._master_branch_cache = None
97
106
        self._merge_sorted_revisions_cache = None
98
107
        self._open_hook()
99
108
        hooks = Branch.hooks['open']
105
114
 
106
115
    def _activate_fallback_location(self, url):
107
116
        """Activate the branch/repository from url as a fallback repository."""
 
117
        for existing_fallback_repo in self.repository._fallback_repositories:
 
118
            if existing_fallback_repo.user_url == url:
 
119
                # This fallback is already configured.  This probably only
 
120
                # happens because BzrDir.sprout is a horrible mess.  To avoid
 
121
                # confusing _unstack we don't add this a second time.
 
122
                mutter('duplicate activation of fallback %r on %r', url, self)
 
123
                return
108
124
        repo = self._get_fallback_repository(url)
109
125
        if repo.has_same_location(self.repository):
110
126
            raise errors.UnstackableLocationError(self.user_url, url)
653
669
        raise errors.UnsupportedOperation(self.get_reference_info, self)
654
670
 
655
671
    @needs_write_lock
656
 
    def fetch(self, from_branch, last_revision=None, pb=None):
 
672
    def fetch(self, from_branch, last_revision=None):
657
673
        """Copy revisions from from_branch into this branch.
658
674
 
659
675
        :param from_branch: Where to copy from.
660
676
        :param last_revision: What revision to stop at (None for at the end
661
677
                              of the branch.
662
 
        :param pb: An optional progress bar to use.
663
678
        :return: None
664
679
        """
665
 
        if self.base == from_branch.base:
666
 
            return (0, [])
667
 
        if pb is not None:
668
 
            symbol_versioning.warn(
669
 
                symbol_versioning.deprecated_in((1, 14, 0))
670
 
                % "pb parameter to fetch()")
671
 
        from_branch.lock_read()
672
 
        try:
673
 
            if last_revision is None:
674
 
                last_revision = from_branch.last_revision()
675
 
                last_revision = _mod_revision.ensure_null(last_revision)
676
 
            return self.repository.fetch(from_branch.repository,
677
 
                                         revision_id=last_revision,
678
 
                                         pb=pb)
679
 
        finally:
680
 
            from_branch.unlock()
 
680
        return InterBranch.get(from_branch, self).fetch(last_revision)
681
681
 
682
682
    def get_bound_location(self):
683
683
        """Return the URL of the branch we are bound to.
694
694
 
695
695
    def get_commit_builder(self, parents, config=None, timestamp=None,
696
696
                           timezone=None, committer=None, revprops=None,
697
 
                           revision_id=None):
 
697
                           revision_id=None, lossy=False):
698
698
        """Obtain a CommitBuilder for this branch.
699
699
 
700
700
        :param parents: Revision ids of the parents of the new revision.
704
704
        :param committer: Optional committer to set for commit.
705
705
        :param revprops: Optional dictionary of revision properties.
706
706
        :param revision_id: Optional revision id.
 
707
        :param lossy: Whether to discard data that can not be natively
 
708
            represented, when pushing to a foreign VCS 
707
709
        """
708
710
 
709
711
        if config is None:
710
712
            config = self.get_config()
711
713
 
712
714
        return self.repository.get_commit_builder(self, parents, config,
713
 
            timestamp, timezone, committer, revprops, revision_id)
 
715
            timestamp, timezone, committer, revprops, revision_id,
 
716
            lossy)
714
717
 
715
718
    def get_master_branch(self, possible_transports=None):
716
719
        """Return the branch we are bound to.
794
797
 
795
798
    def _unstack(self):
796
799
        """Change a branch to be unstacked, copying data as needed.
797
 
        
 
800
 
798
801
        Don't call this directly, use set_stacked_on_url(None).
799
802
        """
800
803
        pb = ui.ui_factory.nested_progress_bar()
809
812
            old_repository = self.repository
810
813
            if len(old_repository._fallback_repositories) != 1:
811
814
                raise AssertionError("can't cope with fallback repositories "
812
 
                    "of %r" % (self.repository,))
 
815
                    "of %r (fallbacks: %r)" % (old_repository,
 
816
                        old_repository._fallback_repositories))
813
817
            # Open the new repository object.
814
818
            # Repositories don't offer an interface to remove fallback
815
819
            # repositories today; take the conceptually simpler option and just
863
867
                # XXX: If you unstack a branch while it has a working tree
864
868
                # with a pending merge, the pending-merged revisions will no
865
869
                # longer be present.  You can (probably) revert and remerge.
866
 
                #
867
 
                # XXX: This only fetches up to the tip of the repository; it
868
 
                # doesn't bring across any tags.  That's fairly consistent
869
 
                # with how branch works, but perhaps not ideal.
870
 
                self.repository.fetch(old_repository,
871
 
                    revision_id=self.last_revision(),
872
 
                    find_ghosts=True)
 
870
                try:
 
871
                    tags_to_fetch = set(self.tags.get_reverse_tag_dict())
 
872
                except errors.TagsNotSupported:
 
873
                    tags_to_fetch = set()
 
874
                fetch_spec = _mod_graph.NotInOtherForRevs(self.repository,
 
875
                    old_repository, required_ids=[self.last_revision()],
 
876
                    if_present_ids=tags_to_fetch, find_ghosts=True).execute()
 
877
                self.repository.fetch(old_repository, fetch_spec=fetch_spec)
873
878
            finally:
874
879
                old_repository.unlock()
875
880
        finally:
918
923
        self._revision_history_cache = None
919
924
        self._revision_id_to_revno_cache = None
920
925
        self._last_revision_info_cache = None
 
926
        self._master_branch_cache = None
921
927
        self._merge_sorted_revisions_cache = None
922
928
        self._partial_revision_history_cache = []
923
929
        self._partial_revision_id_to_revno_cache = {}
988
994
        else:
989
995
            return (0, _mod_revision.NULL_REVISION)
990
996
 
991
 
    @deprecated_method(deprecated_in((1, 6, 0)))
992
 
    def missing_revisions(self, other, stop_revision=None):
993
 
        """Return a list of new revisions that would perfectly fit.
994
 
 
995
 
        If self and other have not diverged, return a list of the revisions
996
 
        present in other, but missing from self.
997
 
        """
998
 
        self_history = self.revision_history()
999
 
        self_len = len(self_history)
1000
 
        other_history = other.revision_history()
1001
 
        other_len = len(other_history)
1002
 
        common_index = min(self_len, other_len) -1
1003
 
        if common_index >= 0 and \
1004
 
            self_history[common_index] != other_history[common_index]:
1005
 
            raise errors.DivergedBranches(self, other)
1006
 
 
1007
 
        if stop_revision is None:
1008
 
            stop_revision = other_len
1009
 
        else:
1010
 
            if stop_revision > other_len:
1011
 
                raise errors.NoSuchRevision(self, stop_revision)
1012
 
        return other_history[self_len:stop_revision]
1013
 
 
1014
997
    def update_revisions(self, other, stop_revision=None, overwrite=False,
1015
 
                         graph=None):
 
998
            graph=None):
1016
999
        """Pull in new perfect-fit revisions.
1017
1000
 
1018
1001
        :param other: Another Branch to pull from
1026
1009
        return InterBranch.get(other, self).update_revisions(stop_revision,
1027
1010
            overwrite, graph)
1028
1011
 
 
1012
    @deprecated_method(deprecated_in((2, 4, 0)))
1029
1013
    def import_last_revision_info(self, source_repo, revno, revid):
1030
1014
        """Set the last revision info, importing from another repo if necessary.
1031
1015
 
1032
 
        This is used by the bound branch code to upload a revision to
1033
 
        the master branch first before updating the tip of the local branch.
1034
 
 
1035
1016
        :param source_repo: Source repository to optionally fetch from
1036
1017
        :param revno: Revision number of the new tip
1037
1018
        :param revid: Revision id of the new tip
1040
1021
            self.repository.fetch(source_repo, revision_id=revid)
1041
1022
        self.set_last_revision_info(revno, revid)
1042
1023
 
 
1024
    def import_last_revision_info_and_tags(self, source, revno, revid,
 
1025
                                           lossy=False):
 
1026
        """Set the last revision info, importing from another repo if necessary.
 
1027
 
 
1028
        This is used by the bound branch code to upload a revision to
 
1029
        the master branch first before updating the tip of the local branch.
 
1030
        Revisions referenced by source's tags are also transferred.
 
1031
 
 
1032
        :param source: Source branch to optionally fetch from
 
1033
        :param revno: Revision number of the new tip
 
1034
        :param revid: Revision id of the new tip
 
1035
        :param lossy: Whether to discard metadata that can not be
 
1036
            natively represented
 
1037
        :return: Tuple with the new revision number and revision id
 
1038
            (should only be different from the arguments when lossy=True)
 
1039
        """
 
1040
        if not self.repository.has_same_location(source.repository):
 
1041
            self.fetch(source, revid)
 
1042
        self.set_last_revision_info(revno, revid)
 
1043
        return (revno, revid)
 
1044
 
1043
1045
    def revision_id_to_revno(self, revision_id):
1044
1046
        """Given a revision id, return its revno"""
1045
1047
        if _mod_revision.is_null(revision_id):
1266
1268
        return result
1267
1269
 
1268
1270
    @needs_read_lock
1269
 
    def sprout(self, to_bzrdir, revision_id=None, repository_policy=None):
 
1271
    def sprout(self, to_bzrdir, revision_id=None, repository_policy=None,
 
1272
            repository=None):
1270
1273
        """Create a new line of development from the branch, into to_bzrdir.
1271
1274
 
1272
1275
        to_bzrdir controls the branch format.
1277
1280
        if (repository_policy is not None and
1278
1281
            repository_policy.requires_stacking()):
1279
1282
            to_bzrdir._format.require_stacking(_skip_repo=True)
1280
 
        result = to_bzrdir.create_branch()
 
1283
        result = to_bzrdir.create_branch(repository=repository)
1281
1284
        result.lock_write()
1282
1285
        try:
1283
1286
            if repository_policy is not None:
1371
1374
        """Return the most suitable metadir for a checkout of this branch.
1372
1375
        Weaves are used if this branch's repository uses weaves.
1373
1376
        """
1374
 
        if isinstance(self.bzrdir, bzrdir.BzrDirPreSplitOut):
1375
 
            from bzrlib.repofmt import weaverepo
1376
 
            format = bzrdir.BzrDirMetaFormat1()
1377
 
            format.repository_format = weaverepo.RepositoryFormat7()
1378
 
        else:
1379
 
            format = self.repository.bzrdir.checkout_metadir()
1380
 
            format.set_branch_format(self._format)
 
1377
        format = self.repository.bzrdir.checkout_metadir()
 
1378
        format.set_branch_format(self._format)
1381
1379
        return format
1382
1380
 
1383
1381
    def create_clone_on_transport(self, to_transport, revision_id=None,
1523
1521
        else:
1524
1522
            raise AssertionError("invalid heads: %r" % (heads,))
1525
1523
 
1526
 
 
1527
 
class BranchFormat(object):
 
1524
    def heads_to_fetch(self):
 
1525
        """Return the heads that must and that should be fetched to copy this
 
1526
        branch into another repo.
 
1527
 
 
1528
        :returns: a 2-tuple of (must_fetch, if_present_fetch).  must_fetch is a
 
1529
            set of heads that must be fetched.  if_present_fetch is a set of
 
1530
            heads that must be fetched if present, but no error is necessary if
 
1531
            they are not present.
 
1532
        """
 
1533
        # For bzr native formats must_fetch is just the tip, and if_present_fetch
 
1534
        # are the tags.
 
1535
        must_fetch = set([self.last_revision()])
 
1536
        try:
 
1537
            if_present_fetch = set(self.tags.get_reverse_tag_dict())
 
1538
        except errors.TagsNotSupported:
 
1539
            if_present_fetch = set()
 
1540
        must_fetch.discard(_mod_revision.NULL_REVISION)
 
1541
        if_present_fetch.discard(_mod_revision.NULL_REVISION)
 
1542
        return must_fetch, if_present_fetch
 
1543
 
 
1544
 
 
1545
class BranchFormat(controldir.ControlComponentFormat):
1528
1546
    """An encapsulation of the initialization and open routines for a format.
1529
1547
 
1530
1548
    Formats provide three things:
1542
1560
    object will be created every time regardless.
1543
1561
    """
1544
1562
 
1545
 
    _default_format = None
1546
 
    """The default format used for new branches."""
1547
 
 
1548
 
    _formats = {}
1549
 
    """The known formats."""
1550
 
 
1551
1563
    can_set_append_revisions_only = True
1552
1564
 
1553
1565
    def __eq__(self, other):
1562
1574
        try:
1563
1575
            transport = a_bzrdir.get_branch_transport(None, name=name)
1564
1576
            format_string = transport.get_bytes("format")
1565
 
            format = klass._formats[format_string]
1566
 
            if isinstance(format, MetaDirBranchFormatFactory):
1567
 
                return format()
1568
 
            return format
 
1577
            return format_registry.get(format_string)
1569
1578
        except errors.NoSuchFile:
1570
1579
            raise errors.NotBranchError(path=transport.base, bzrdir=a_bzrdir)
1571
1580
        except KeyError:
1572
1581
            raise errors.UnknownFormatError(format=format_string, kind='branch')
1573
1582
 
1574
1583
    @classmethod
 
1584
    @deprecated_method(deprecated_in((2, 4, 0)))
1575
1585
    def get_default_format(klass):
1576
1586
        """Return the current default format."""
1577
 
        return klass._default_format
 
1587
        return format_registry.get_default()
1578
1588
 
1579
1589
    @classmethod
 
1590
    @deprecated_method(deprecated_in((2, 4, 0)))
1580
1591
    def get_formats(klass):
1581
1592
        """Get all the known formats.
1582
1593
 
1583
1594
        Warning: This triggers a load of all lazy registered formats: do not
1584
1595
        use except when that is desireed.
1585
1596
        """
1586
 
        result = []
1587
 
        for fmt in klass._formats.values():
1588
 
            if isinstance(fmt, MetaDirBranchFormatFactory):
1589
 
                fmt = fmt()
1590
 
            result.append(fmt)
1591
 
        return result
 
1597
        return format_registry._get_all()
1592
1598
 
1593
1599
    def get_reference(self, a_bzrdir, name=None):
1594
1600
        """Get the target reference of the branch in a_bzrdir.
1633
1639
        for hook in hooks:
1634
1640
            hook(params)
1635
1641
 
1636
 
    def _initialize_helper(self, a_bzrdir, utf8_files, name=None,
1637
 
                           lock_type='metadir', set_format=True):
1638
 
        """Initialize a branch in a bzrdir, with specified files
1639
 
 
1640
 
        :param a_bzrdir: The bzrdir to initialize the branch in
1641
 
        :param utf8_files: The files to create as a list of
1642
 
            (filename, content) tuples
1643
 
        :param name: Name of colocated branch to create, if any
1644
 
        :param set_format: If True, set the format with
1645
 
            self.get_format_string.  (BzrBranch4 has its format set
1646
 
            elsewhere)
1647
 
        :return: a branch in this format
1648
 
        """
1649
 
        mutter('creating branch %r in %s', self, a_bzrdir.user_url)
1650
 
        branch_transport = a_bzrdir.get_branch_transport(self, name=name)
1651
 
        lock_map = {
1652
 
            'metadir': ('lock', lockdir.LockDir),
1653
 
            'branch4': ('branch-lock', lockable_files.TransportLock),
1654
 
        }
1655
 
        lock_name, lock_class = lock_map[lock_type]
1656
 
        control_files = lockable_files.LockableFiles(branch_transport,
1657
 
            lock_name, lock_class)
1658
 
        control_files.create_lock()
1659
 
        try:
1660
 
            control_files.lock_write()
1661
 
        except errors.LockContention:
1662
 
            if lock_type != 'branch4':
1663
 
                raise
1664
 
            lock_taken = False
1665
 
        else:
1666
 
            lock_taken = True
1667
 
        if set_format:
1668
 
            utf8_files += [('format', self.get_format_string())]
1669
 
        try:
1670
 
            for (filename, content) in utf8_files:
1671
 
                branch_transport.put_bytes(
1672
 
                    filename, content,
1673
 
                    mode=a_bzrdir._get_file_mode())
1674
 
        finally:
1675
 
            if lock_taken:
1676
 
                control_files.unlock()
1677
 
        branch = self.open(a_bzrdir, name, _found=True)
1678
 
        self._run_post_branch_init_hooks(a_bzrdir, name, branch)
1679
 
        return branch
1680
 
 
1681
 
    def initialize(self, a_bzrdir, name=None):
 
1642
    def initialize(self, a_bzrdir, name=None, repository=None):
1682
1643
        """Create a branch of this format in a_bzrdir.
1683
1644
        
1684
1645
        :param name: Name of the colocated branch to create.
1718
1679
        """
1719
1680
        raise NotImplementedError(self.network_name)
1720
1681
 
1721
 
    def open(self, a_bzrdir, name=None, _found=False, ignore_fallbacks=False):
 
1682
    def open(self, a_bzrdir, name=None, _found=False, ignore_fallbacks=False,
 
1683
            found_repository=None):
1722
1684
        """Return the branch object for a_bzrdir
1723
1685
 
1724
1686
        :param a_bzrdir: A BzrDir that contains a branch.
1731
1693
        raise NotImplementedError(self.open)
1732
1694
 
1733
1695
    @classmethod
 
1696
    @deprecated_method(deprecated_in((2, 4, 0)))
1734
1697
    def register_format(klass, format):
1735
1698
        """Register a metadir format.
1736
 
        
 
1699
 
1737
1700
        See MetaDirBranchFormatFactory for the ability to register a format
1738
1701
        without loading the code the format needs until it is actually used.
1739
1702
        """
1740
 
        klass._formats[format.get_format_string()] = format
1741
 
        # Metadir formats have a network name of their format string, and get
1742
 
        # registered as factories.
1743
 
        if isinstance(format, MetaDirBranchFormatFactory):
1744
 
            network_format_registry.register(format.get_format_string(), format)
1745
 
        else:
1746
 
            network_format_registry.register(format.get_format_string(),
1747
 
                format.__class__)
 
1703
        format_registry.register(format)
1748
1704
 
1749
1705
    @classmethod
 
1706
    @deprecated_method(deprecated_in((2, 4, 0)))
1750
1707
    def set_default_format(klass, format):
1751
 
        klass._default_format = format
 
1708
        format_registry.set_default(format)
1752
1709
 
1753
1710
    def supports_set_append_revisions_only(self):
1754
1711
        """True if this format supports set_append_revisions_only."""
1758
1715
        """True if this format records a stacked-on branch."""
1759
1716
        return False
1760
1717
 
 
1718
    def supports_leaving_lock(self):
 
1719
        """True if this format supports leaving locks in place."""
 
1720
        return False # by default
 
1721
 
1761
1722
    @classmethod
 
1723
    @deprecated_method(deprecated_in((2, 4, 0)))
1762
1724
    def unregister_format(klass, format):
1763
 
        del klass._formats[format.get_format_string()]
 
1725
        format_registry.remove(format)
1764
1726
 
1765
1727
    def __str__(self):
1766
1728
        return self.get_format_description().rstrip()
1811
1773
        These are all empty initially, because by default nothing should get
1812
1774
        notified.
1813
1775
        """
1814
 
        Hooks.__init__(self)
1815
 
        self.create_hook(HookPoint('set_rh',
 
1776
        Hooks.__init__(self, "bzrlib.branch", "Branch.hooks")
 
1777
        self.add_hook('set_rh',
1816
1778
            "Invoked whenever the revision history has been set via "
1817
1779
            "set_revision_history. The api signature is (branch, "
1818
1780
            "revision_history), and the branch will be write-locked. "
1819
1781
            "The set_rh hook can be expensive for bzr to trigger, a better "
1820
 
            "hook to use is Branch.post_change_branch_tip.", (0, 15), None))
1821
 
        self.create_hook(HookPoint('open',
 
1782
            "hook to use is Branch.post_change_branch_tip.", (0, 15))
 
1783
        self.add_hook('open',
1822
1784
            "Called with the Branch object that has been opened after a "
1823
 
            "branch is opened.", (1, 8), None))
1824
 
        self.create_hook(HookPoint('post_push',
 
1785
            "branch is opened.", (1, 8))
 
1786
        self.add_hook('post_push',
1825
1787
            "Called after a push operation completes. post_push is called "
1826
1788
            "with a bzrlib.branch.BranchPushResult object and only runs in the "
1827
 
            "bzr client.", (0, 15), None))
1828
 
        self.create_hook(HookPoint('post_pull',
 
1789
            "bzr client.", (0, 15))
 
1790
        self.add_hook('post_pull',
1829
1791
            "Called after a pull operation completes. post_pull is called "
1830
1792
            "with a bzrlib.branch.PullResult object and only runs in the "
1831
 
            "bzr client.", (0, 15), None))
1832
 
        self.create_hook(HookPoint('pre_commit',
 
1793
            "bzr client.", (0, 15))
 
1794
        self.add_hook('pre_commit',
1833
1795
            "Called after a commit is calculated but before it is "
1834
1796
            "completed. pre_commit is called with (local, master, old_revno, "
1835
1797
            "old_revid, future_revno, future_revid, tree_delta, future_tree"
1838
1800
            "basis revision. hooks MUST NOT modify this delta. "
1839
1801
            " future_tree is an in-memory tree obtained from "
1840
1802
            "CommitBuilder.revision_tree() and hooks MUST NOT modify this "
1841
 
            "tree.", (0,91), None))
1842
 
        self.create_hook(HookPoint('post_commit',
 
1803
            "tree.", (0,91))
 
1804
        self.add_hook('post_commit',
1843
1805
            "Called in the bzr client after a commit has completed. "
1844
1806
            "post_commit is called with (local, master, old_revno, old_revid, "
1845
1807
            "new_revno, new_revid). old_revid is NULL_REVISION for the first "
1846
 
            "commit to a branch.", (0, 15), None))
1847
 
        self.create_hook(HookPoint('post_uncommit',
 
1808
            "commit to a branch.", (0, 15))
 
1809
        self.add_hook('post_uncommit',
1848
1810
            "Called in the bzr client after an uncommit completes. "
1849
1811
            "post_uncommit is called with (local, master, old_revno, "
1850
1812
            "old_revid, new_revno, new_revid) where local is the local branch "
1851
1813
            "or None, master is the target branch, and an empty branch "
1852
 
            "receives new_revno of 0, new_revid of None.", (0, 15), None))
1853
 
        self.create_hook(HookPoint('pre_change_branch_tip',
 
1814
            "receives new_revno of 0, new_revid of None.", (0, 15))
 
1815
        self.add_hook('pre_change_branch_tip',
1854
1816
            "Called in bzr client and server before a change to the tip of a "
1855
1817
            "branch is made. pre_change_branch_tip is called with a "
1856
1818
            "bzrlib.branch.ChangeBranchTipParams. Note that push, pull, "
1857
 
            "commit, uncommit will all trigger this hook.", (1, 6), None))
1858
 
        self.create_hook(HookPoint('post_change_branch_tip',
 
1819
            "commit, uncommit will all trigger this hook.", (1, 6))
 
1820
        self.add_hook('post_change_branch_tip',
1859
1821
            "Called in bzr client and server after a change to the tip of a "
1860
1822
            "branch is made. post_change_branch_tip is called with a "
1861
1823
            "bzrlib.branch.ChangeBranchTipParams. Note that push, pull, "
1862
 
            "commit, uncommit will all trigger this hook.", (1, 4), None))
1863
 
        self.create_hook(HookPoint('transform_fallback_location',
 
1824
            "commit, uncommit will all trigger this hook.", (1, 4))
 
1825
        self.add_hook('transform_fallback_location',
1864
1826
            "Called when a stacked branch is activating its fallback "
1865
1827
            "locations. transform_fallback_location is called with (branch, "
1866
1828
            "url), and should return a new url. Returning the same url "
1871
1833
            "fallback locations have not been activated. When there are "
1872
1834
            "multiple hooks installed for transform_fallback_location, "
1873
1835
            "all are called with the url returned from the previous hook."
1874
 
            "The order is however undefined.", (1, 9), None))
1875
 
        self.create_hook(HookPoint('automatic_tag_name',
 
1836
            "The order is however undefined.", (1, 9))
 
1837
        self.add_hook('automatic_tag_name',
1876
1838
            "Called to determine an automatic tag name for a revision. "
1877
1839
            "automatic_tag_name is called with (branch, revision_id) and "
1878
1840
            "should return a tag name or None if no tag name could be "
1879
1841
            "determined. The first non-None tag name returned will be used.",
1880
 
            (2, 2), None))
1881
 
        self.create_hook(HookPoint('post_branch_init',
 
1842
            (2, 2))
 
1843
        self.add_hook('post_branch_init',
1882
1844
            "Called after new branch initialization completes. "
1883
1845
            "post_branch_init is called with a "
1884
1846
            "bzrlib.branch.BranchInitHookParams. "
1885
1847
            "Note that init, branch and checkout (both heavyweight and "
1886
 
            "lightweight) will all trigger this hook.", (2, 2), None))
1887
 
        self.create_hook(HookPoint('post_switch',
 
1848
            "lightweight) will all trigger this hook.", (2, 2))
 
1849
        self.add_hook('post_switch',
1888
1850
            "Called after a checkout switches branch. "
1889
1851
            "post_switch is called with a "
1890
 
            "bzrlib.branch.SwitchHookParams.", (2, 2), None))
 
1852
            "bzrlib.branch.SwitchHookParams.", (2, 2))
1891
1853
 
1892
1854
 
1893
1855
 
2006
1968
            self.revision_id)
2007
1969
 
2008
1970
 
2009
 
class BzrBranchFormat4(BranchFormat):
2010
 
    """Bzr branch format 4.
2011
 
 
2012
 
    This format has:
2013
 
     - a revision-history file.
2014
 
     - a branch-lock lock file [ to be shared with the bzrdir ]
2015
 
    """
2016
 
 
2017
 
    def get_format_description(self):
2018
 
        """See BranchFormat.get_format_description()."""
2019
 
        return "Branch format 4"
2020
 
 
2021
 
    def initialize(self, a_bzrdir, name=None):
2022
 
        """Create a branch of this format in a_bzrdir."""
2023
 
        utf8_files = [('revision-history', ''),
2024
 
                      ('branch-name', ''),
2025
 
                      ]
2026
 
        return self._initialize_helper(a_bzrdir, utf8_files, name=name,
2027
 
                                       lock_type='branch4', set_format=False)
2028
 
 
2029
 
    def __init__(self):
2030
 
        super(BzrBranchFormat4, self).__init__()
2031
 
        self._matchingbzrdir = bzrdir.BzrDirFormat6()
2032
 
 
2033
 
    def network_name(self):
2034
 
        """The network name for this format is the control dirs disk label."""
2035
 
        return self._matchingbzrdir.get_format_string()
2036
 
 
2037
 
    def open(self, a_bzrdir, name=None, _found=False, ignore_fallbacks=False):
2038
 
        """See BranchFormat.open()."""
2039
 
        if not _found:
2040
 
            # we are being called directly and must probe.
2041
 
            raise NotImplementedError
2042
 
        return BzrBranch(_format=self,
2043
 
                         _control_files=a_bzrdir._control_files,
2044
 
                         a_bzrdir=a_bzrdir,
2045
 
                         name=name,
2046
 
                         _repository=a_bzrdir.open_repository())
2047
 
 
2048
 
    def __str__(self):
2049
 
        return "Bazaar-NG branch format 4"
2050
 
 
2051
 
 
2052
1971
class BranchFormatMetadir(BranchFormat):
2053
1972
    """Common logic for meta-dir based branch formats."""
2054
1973
 
2056
1975
        """What class to instantiate on open calls."""
2057
1976
        raise NotImplementedError(self._branch_class)
2058
1977
 
 
1978
    def _initialize_helper(self, a_bzrdir, utf8_files, name=None,
 
1979
                           repository=None):
 
1980
        """Initialize a branch in a bzrdir, with specified files
 
1981
 
 
1982
        :param a_bzrdir: The bzrdir to initialize the branch in
 
1983
        :param utf8_files: The files to create as a list of
 
1984
            (filename, content) tuples
 
1985
        :param name: Name of colocated branch to create, if any
 
1986
        :return: a branch in this format
 
1987
        """
 
1988
        mutter('creating branch %r in %s', self, a_bzrdir.user_url)
 
1989
        branch_transport = a_bzrdir.get_branch_transport(self, name=name)
 
1990
        control_files = lockable_files.LockableFiles(branch_transport,
 
1991
            'lock', lockdir.LockDir)
 
1992
        control_files.create_lock()
 
1993
        control_files.lock_write()
 
1994
        try:
 
1995
            utf8_files += [('format', self.get_format_string())]
 
1996
            for (filename, content) in utf8_files:
 
1997
                branch_transport.put_bytes(
 
1998
                    filename, content,
 
1999
                    mode=a_bzrdir._get_file_mode())
 
2000
        finally:
 
2001
            control_files.unlock()
 
2002
        branch = self.open(a_bzrdir, name, _found=True,
 
2003
                found_repository=repository)
 
2004
        self._run_post_branch_init_hooks(a_bzrdir, name, branch)
 
2005
        return branch
 
2006
 
2059
2007
    def network_name(self):
2060
2008
        """A simple byte string uniquely identifying this format for RPC calls.
2061
2009
 
2063
2011
        """
2064
2012
        return self.get_format_string()
2065
2013
 
2066
 
    def open(self, a_bzrdir, name=None, _found=False, ignore_fallbacks=False):
 
2014
    def open(self, a_bzrdir, name=None, _found=False, ignore_fallbacks=False,
 
2015
            found_repository=None):
2067
2016
        """See BranchFormat.open()."""
2068
2017
        if not _found:
2069
2018
            format = BranchFormat.find_format(a_bzrdir, name=name)
2074
2023
        try:
2075
2024
            control_files = lockable_files.LockableFiles(transport, 'lock',
2076
2025
                                                         lockdir.LockDir)
 
2026
            if found_repository is None:
 
2027
                found_repository = a_bzrdir.find_repository()
2077
2028
            return self._branch_class()(_format=self,
2078
2029
                              _control_files=control_files,
2079
2030
                              name=name,
2080
2031
                              a_bzrdir=a_bzrdir,
2081
 
                              _repository=a_bzrdir.find_repository(),
 
2032
                              _repository=found_repository,
2082
2033
                              ignore_fallbacks=ignore_fallbacks)
2083
2034
        except errors.NoSuchFile:
2084
2035
            raise errors.NotBranchError(path=transport.base, bzrdir=a_bzrdir)
2091
2042
    def supports_tags(self):
2092
2043
        return True
2093
2044
 
 
2045
    def supports_leaving_lock(self):
 
2046
        return True
 
2047
 
2094
2048
 
2095
2049
class BzrBranchFormat5(BranchFormatMetadir):
2096
2050
    """Bzr branch format 5.
2116
2070
        """See BranchFormat.get_format_description()."""
2117
2071
        return "Branch format 5"
2118
2072
 
2119
 
    def initialize(self, a_bzrdir, name=None):
 
2073
    def initialize(self, a_bzrdir, name=None, repository=None):
2120
2074
        """Create a branch of this format in a_bzrdir."""
2121
2075
        utf8_files = [('revision-history', ''),
2122
2076
                      ('branch-name', ''),
2123
2077
                      ]
2124
 
        return self._initialize_helper(a_bzrdir, utf8_files, name)
 
2078
        return self._initialize_helper(a_bzrdir, utf8_files, name, repository)
2125
2079
 
2126
2080
    def supports_tags(self):
2127
2081
        return False
2149
2103
        """See BranchFormat.get_format_description()."""
2150
2104
        return "Branch format 6"
2151
2105
 
2152
 
    def initialize(self, a_bzrdir, name=None):
 
2106
    def initialize(self, a_bzrdir, name=None, repository=None):
2153
2107
        """Create a branch of this format in a_bzrdir."""
2154
2108
        utf8_files = [('last-revision', '0 null:\n'),
2155
2109
                      ('branch.conf', ''),
2156
2110
                      ('tags', ''),
2157
2111
                      ]
2158
 
        return self._initialize_helper(a_bzrdir, utf8_files, name)
 
2112
        return self._initialize_helper(a_bzrdir, utf8_files, name, repository)
2159
2113
 
2160
2114
    def make_tags(self, branch):
2161
2115
        """See bzrlib.branch.BranchFormat.make_tags()."""
2179
2133
        """See BranchFormat.get_format_description()."""
2180
2134
        return "Branch format 8"
2181
2135
 
2182
 
    def initialize(self, a_bzrdir, name=None):
 
2136
    def initialize(self, a_bzrdir, name=None, repository=None):
2183
2137
        """Create a branch of this format in a_bzrdir."""
2184
2138
        utf8_files = [('last-revision', '0 null:\n'),
2185
2139
                      ('branch.conf', ''),
2186
2140
                      ('tags', ''),
2187
2141
                      ('references', '')
2188
2142
                      ]
2189
 
        return self._initialize_helper(a_bzrdir, utf8_files, name)
2190
 
 
2191
 
    def __init__(self):
2192
 
        super(BzrBranchFormat8, self).__init__()
2193
 
        self._matchingbzrdir.repository_format = \
2194
 
            RepositoryFormatKnitPack5RichRoot()
 
2143
        return self._initialize_helper(a_bzrdir, utf8_files, name, repository)
2195
2144
 
2196
2145
    def make_tags(self, branch):
2197
2146
        """See bzrlib.branch.BranchFormat.make_tags()."""
2206
2155
    supports_reference_locations = True
2207
2156
 
2208
2157
 
2209
 
class BzrBranchFormat7(BzrBranchFormat8):
 
2158
class BzrBranchFormat7(BranchFormatMetadir):
2210
2159
    """Branch format with last-revision, tags, and a stacked location pointer.
2211
2160
 
2212
2161
    The stacked location pointer is passed down to the repository and requires
2215
2164
    This format was introduced in bzr 1.6.
2216
2165
    """
2217
2166
 
2218
 
    def initialize(self, a_bzrdir, name=None):
 
2167
    def initialize(self, a_bzrdir, name=None, repository=None):
2219
2168
        """Create a branch of this format in a_bzrdir."""
2220
2169
        utf8_files = [('last-revision', '0 null:\n'),
2221
2170
                      ('branch.conf', ''),
2222
2171
                      ('tags', ''),
2223
2172
                      ]
2224
 
        return self._initialize_helper(a_bzrdir, utf8_files, name)
 
2173
        return self._initialize_helper(a_bzrdir, utf8_files, name, repository)
2225
2174
 
2226
2175
    def _branch_class(self):
2227
2176
        return BzrBranch7
2237
2186
    def supports_set_append_revisions_only(self):
2238
2187
        return True
2239
2188
 
 
2189
    def supports_stacking(self):
 
2190
        return True
 
2191
 
 
2192
    def make_tags(self, branch):
 
2193
        """See bzrlib.branch.BranchFormat.make_tags()."""
 
2194
        return BasicTags(branch)
 
2195
 
2240
2196
    supports_reference_locations = False
2241
2197
 
2242
2198
 
2269
2225
        transport = a_bzrdir.get_branch_transport(None, name=name)
2270
2226
        location = transport.put_bytes('location', to_branch.base)
2271
2227
 
2272
 
    def initialize(self, a_bzrdir, name=None, target_branch=None):
 
2228
    def initialize(self, a_bzrdir, name=None, target_branch=None,
 
2229
            repository=None):
2273
2230
        """Create a branch of this format in a_bzrdir."""
2274
2231
        if target_branch is None:
2275
2232
            # this format does not implement branch itself, thus the implicit
2303
2260
        return clone
2304
2261
 
2305
2262
    def open(self, a_bzrdir, name=None, _found=False, location=None,
2306
 
             possible_transports=None, ignore_fallbacks=False):
 
2263
             possible_transports=None, ignore_fallbacks=False,
 
2264
             found_repository=None):
2307
2265
        """Return the branch that the branch reference in a_bzrdir points at.
2308
2266
 
2309
2267
        :param a_bzrdir: A BzrDir that contains a branch.
2340
2298
        return result
2341
2299
 
2342
2300
 
 
2301
class BranchFormatRegistry(controldir.ControlComponentFormatRegistry):
 
2302
    """Branch format registry."""
 
2303
 
 
2304
    def __init__(self, other_registry=None):
 
2305
        super(BranchFormatRegistry, self).__init__(other_registry)
 
2306
        self._default_format = None
 
2307
 
 
2308
    def set_default(self, format):
 
2309
        self._default_format = format
 
2310
 
 
2311
    def get_default(self):
 
2312
        return self._default_format
 
2313
 
 
2314
 
2343
2315
network_format_registry = registry.FormatRegistry()
2344
2316
"""Registry of formats indexed by their network name.
2345
2317
 
2348
2320
BranchFormat.network_name() for more detail.
2349
2321
"""
2350
2322
 
 
2323
format_registry = BranchFormatRegistry(network_format_registry)
 
2324
 
2351
2325
 
2352
2326
# formats which have no format string are not discoverable
2353
2327
# and not independently creatable, so are not registered.
2355
2329
__format6 = BzrBranchFormat6()
2356
2330
__format7 = BzrBranchFormat7()
2357
2331
__format8 = BzrBranchFormat8()
2358
 
BranchFormat.register_format(__format5)
2359
 
BranchFormat.register_format(BranchReferenceFormat())
2360
 
BranchFormat.register_format(__format6)
2361
 
BranchFormat.register_format(__format7)
2362
 
BranchFormat.register_format(__format8)
2363
 
BranchFormat.set_default_format(__format7)
2364
 
_legacy_formats = [BzrBranchFormat4(),
2365
 
    ]
2366
 
network_format_registry.register(
2367
 
    _legacy_formats[0].network_name(), _legacy_formats[0].__class__)
 
2332
format_registry.register(__format5)
 
2333
format_registry.register(BranchReferenceFormat())
 
2334
format_registry.register(__format6)
 
2335
format_registry.register(__format7)
 
2336
format_registry.register(__format8)
 
2337
format_registry.set_default(__format7)
2368
2338
 
2369
2339
 
2370
2340
class BranchWriteLockResult(LogicalLockResult):
2527
2497
            'revision-history', '\n'.join(history),
2528
2498
            mode=self.bzrdir._get_file_mode())
2529
2499
 
2530
 
    @needs_write_lock
 
2500
    @deprecated_method(deprecated_in((2, 4, 0)))
2531
2501
    def set_revision_history(self, rev_history):
2532
2502
        """See Branch.set_revision_history."""
 
2503
        self._set_revision_history(rev_history)
 
2504
 
 
2505
    @needs_write_lock
 
2506
    def _set_revision_history(self, rev_history):
2533
2507
        if 'evil' in debug.debug_flags:
2534
2508
            mutter_callsite(3, "set_revision_history scales with history.")
2535
2509
        check_not_reserved_id = _mod_revision.check_not_reserved_id
2579
2553
            except ValueError:
2580
2554
                rev = self.repository.get_revision(revision_id)
2581
2555
                new_history = rev.get_history(self.repository)[1:]
2582
 
        destination.set_revision_history(new_history)
 
2556
        destination._set_revision_history(new_history)
2583
2557
 
2584
2558
    @needs_write_lock
2585
2559
    def set_last_revision_info(self, revno, revision_id):
2593
2567
        configured to check constraints on history, in which case this may not
2594
2568
        be permitted.
2595
2569
        """
2596
 
        revision_id = _mod_revision.ensure_null(revision_id)
 
2570
        if not revision_id or not isinstance(revision_id, basestring):
 
2571
            raise errors.InvalidRevisionId(revision_id=revision_id, branch=self)
2597
2572
        # this old format stores the full history, but this api doesn't
2598
2573
        # provide it, so we must generate, and might as well check it's
2599
2574
        # correct
2600
2575
        history = self._lefthand_history(revision_id)
2601
2576
        if len(history) != revno:
2602
2577
            raise AssertionError('%d != %d' % (len(history), revno))
2603
 
        self.set_revision_history(history)
 
2578
        self._set_revision_history(history)
2604
2579
 
2605
2580
    def _gen_revision_history(self):
2606
2581
        history = self._transport.get_bytes('revision-history').split('\n')
2620
2595
        :param other_branch: The other branch that DivergedBranches should
2621
2596
            raise with respect to.
2622
2597
        """
2623
 
        self.set_revision_history(self._lefthand_history(revision_id,
 
2598
        self._set_revision_history(self._lefthand_history(revision_id,
2624
2599
            last_rev, other_branch))
2625
2600
 
2626
2601
    def basis_tree(self):
2646
2621
        result.target_branch = target
2647
2622
        result.old_revno, result.old_revid = target.last_revision_info()
2648
2623
        self.update_references(target)
2649
 
        if result.old_revid != self.last_revision():
 
2624
        if result.old_revid != stop_revision:
2650
2625
            # We assume that during 'push' this repository is closer than
2651
2626
            # the target.
2652
2627
            graph = self.repository.get_graph(target.repository)
2653
2628
            target.update_revisions(self, stop_revision,
2654
2629
                overwrite=overwrite, graph=graph)
2655
2630
        if self._push_should_merge_tags():
2656
 
            result.tag_conflicts = self.tags.merge_to(target.tags,
2657
 
                overwrite)
 
2631
            result.tag_conflicts = self.tags.merge_to(target.tags, overwrite)
2658
2632
        result.new_revno, result.new_revid = target.last_revision_info()
2659
2633
        return result
2660
2634
 
2692
2666
        """Return the branch we are bound to.
2693
2667
 
2694
2668
        :return: Either a Branch, or None
2695
 
 
2696
 
        This could memoise the branch, but if thats done
2697
 
        it must be revalidated on each new lock.
2698
 
        So for now we just don't memoise it.
2699
 
        # RBC 20060304 review this decision.
2700
2669
        """
 
2670
        if self._master_branch_cache is None:
 
2671
            self._master_branch_cache = self._get_master_branch(
 
2672
                possible_transports)
 
2673
        return self._master_branch_cache
 
2674
 
 
2675
    def _get_master_branch(self, possible_transports):
2701
2676
        bound_loc = self.get_bound_location()
2702
2677
        if not bound_loc:
2703
2678
            return None
2714
2689
 
2715
2690
        :param location: URL to the target branch
2716
2691
        """
 
2692
        self._master_branch_cache = None
2717
2693
        if location:
2718
2694
            self._transport.put_bytes('bound', location+'\n',
2719
2695
                mode=self.bzrdir._get_file_mode())
2828
2804
 
2829
2805
    @needs_write_lock
2830
2806
    def set_last_revision_info(self, revno, revision_id):
2831
 
        revision_id = _mod_revision.ensure_null(revision_id)
 
2807
        if not revision_id or not isinstance(revision_id, basestring):
 
2808
            raise errors.InvalidRevisionId(revision_id=revision_id, branch=self)
2832
2809
        old_revno, old_revid = self.last_revision_info()
2833
2810
        if self._get_append_revisions_only():
2834
2811
            self._check_history_violation(revision_id)
2971
2948
 
2972
2949
    def set_bound_location(self, location):
2973
2950
        """See Branch.set_push_location."""
 
2951
        self._master_branch_cache = None
2974
2952
        result = None
2975
2953
        config = self.get_config()
2976
2954
        if location is None:
3053
3031
        try:
3054
3032
            index = self._partial_revision_history_cache.index(revision_id)
3055
3033
        except ValueError:
3056
 
            self._extend_partial_history(stop_revision=revision_id)
 
3034
            try:
 
3035
                self._extend_partial_history(stop_revision=revision_id)
 
3036
            except errors.RevisionNotPresent, e:
 
3037
                raise errors.GhostRevisionsHaveNoRevno(revision_id, e.revision_id)
3057
3038
            index = len(self._partial_revision_history_cache) - 1
3058
3039
            if self._partial_revision_history_cache[index] != revision_id:
3059
3040
                raise errors.NoSuchRevision(self, revision_id)
3307
3288
 
3308
3289
    @needs_write_lock
3309
3290
    def update_revisions(self, stop_revision=None, overwrite=False,
3310
 
                         graph=None):
 
3291
            graph=None):
3311
3292
        """Pull in new perfect-fit revisions.
3312
3293
 
3313
3294
        :param stop_revision: Updated until the given revision
3337
3318
        """
3338
3319
        raise NotImplementedError(self.copy_content_into)
3339
3320
 
 
3321
    @needs_write_lock
 
3322
    def fetch(self, stop_revision=None):
 
3323
        """Fetch revisions.
 
3324
 
 
3325
        :param stop_revision: Last revision to fetch
 
3326
        """
 
3327
        raise NotImplementedError(self.fetch)
 
3328
 
3340
3329
 
3341
3330
class GenericInterBranch(InterBranch):
3342
3331
    """InterBranch implementation that uses public Branch functions."""
3348
3337
 
3349
3338
    @classmethod
3350
3339
    def _get_branch_formats_to_test(klass):
3351
 
        return [(BranchFormat._default_format, BranchFormat._default_format)]
 
3340
        return [(format_registry.get_default(), format_registry.get_default())]
3352
3341
 
3353
3342
    @classmethod
3354
3343
    def unwrap_format(klass, format):
3377
3366
            self.source.tags.merge_to(self.target.tags)
3378
3367
 
3379
3368
    @needs_write_lock
 
3369
    def fetch(self, stop_revision=None):
 
3370
        if self.target.base == self.source.base:
 
3371
            return (0, [])
 
3372
        self.source.lock_read()
 
3373
        try:
 
3374
            fetch_spec_factory = fetch.FetchSpecFactory()
 
3375
            fetch_spec_factory.source_branch = self.source
 
3376
            fetch_spec_factory.source_branch_stop_revision_id = stop_revision
 
3377
            fetch_spec_factory.source_repo = self.source.repository
 
3378
            fetch_spec_factory.target_repo = self.target.repository
 
3379
            fetch_spec_factory.target_repo_kind = fetch.TargetRepoKinds.PREEXISTING
 
3380
            fetch_spec = fetch_spec_factory.make_fetch_spec()
 
3381
            return self.target.repository.fetch(self.source.repository,
 
3382
                fetch_spec=fetch_spec)
 
3383
        finally:
 
3384
            self.source.unlock()
 
3385
 
 
3386
    @needs_write_lock
3380
3387
    def update_revisions(self, stop_revision=None, overwrite=False,
3381
 
        graph=None):
 
3388
            graph=None):
3382
3389
        """See InterBranch.update_revisions()."""
3383
3390
        other_revno, other_last_revision = self.source.last_revision_info()
3384
3391
        stop_revno = None # unknown
3396
3403
        # case of having something to pull, and so that the check for
3397
3404
        # already merged can operate on the just fetched graph, which will
3398
3405
        # be cached in memory.
3399
 
        self.target.fetch(self.source, stop_revision)
 
3406
        self.fetch(stop_revision=stop_revision)
3400
3407
        # Check to see if one is an ancestor of the other
3401
3408
        if not overwrite:
3402
3409
            if graph is None:
3430
3437
        if local and not bound_location:
3431
3438
            raise errors.LocalRequiresBoundBranch()
3432
3439
        master_branch = None
3433
 
        if not local and bound_location and self.source.user_url != bound_location:
 
3440
        source_is_master = (self.source.user_url == bound_location)
 
3441
        if not local and bound_location and not source_is_master:
3434
3442
            # not pulling from master, so we need to update master.
3435
3443
            master_branch = self.target.get_master_branch(possible_transports)
3436
3444
            master_branch.lock_write()
3442
3450
            return self._pull(overwrite,
3443
3451
                stop_revision, _hook_master=master_branch,
3444
3452
                run_hooks=run_hooks,
3445
 
                _override_hook_target=_override_hook_target)
 
3453
                _override_hook_target=_override_hook_target,
 
3454
                merge_tags_to_master=not source_is_master)
3446
3455
        finally:
3447
3456
            if master_branch:
3448
3457
                master_branch.unlock()
3515
3524
 
3516
3525
    def _pull(self, overwrite=False, stop_revision=None,
3517
3526
             possible_transports=None, _hook_master=None, run_hooks=True,
3518
 
             _override_hook_target=None, local=False):
 
3527
             _override_hook_target=None, local=False,
 
3528
             merge_tags_to_master=True):
3519
3529
        """See Branch.pull.
3520
3530
 
3521
3531
        This function is the core worker, used by GenericInterBranch.pull to
3526
3536
        :param run_hooks: Private parameter - if false, this branch
3527
3537
            is being called because it's the master of the primary branch,
3528
3538
            so it should not run its hooks.
 
3539
            is being called because it's the master of the primary branch,
 
3540
            so it should not run its hooks.
3529
3541
        :param _override_hook_target: Private parameter - set the branch to be
3530
3542
            supplied as the target_branch to pull hooks.
3531
3543
        :param local: Only update the local branch, and not the bound branch.
3556
3568
            # so a tags implementation that versions tags can only 
3557
3569
            # pull in the most recent changes. -- JRV20090506
3558
3570
            result.tag_conflicts = self.source.tags.merge_to(self.target.tags,
3559
 
                overwrite)
 
3571
                overwrite, ignore_master=not merge_tags_to_master)
3560
3572
            result.new_revno, result.new_revid = self.target.last_revision_info()
3561
3573
            if _hook_master:
3562
3574
                result.master_branch = _hook_master