745
788
really matter if it's on an nfs/smb/afs/coda/... share, as long as
746
789
it's writable, and can be accessed via the normal filesystem API.
748
# We actually expect this class to be somewhat short-lived; part of its
749
# purpose is to try to isolate what bits of the branch logic are tied to
750
# filesystem access, so that in a later step, we can extricate them to
751
# a separarte ("storage") class.
752
_inventory_weave = None
754
# Map some sort of prefix into a namespace
755
# stuff like "revno:10", "revid:", etc.
756
# This should match a prefix with a function which accepts
757
REVISION_NAMESPACES = {}
759
792
def __init__(self, transport=DEPRECATED_PARAMETER, init=DEPRECATED_PARAMETER,
760
793
relax_version_check=DEPRECATED_PARAMETER, _format=None,
761
794
_control_files=None, a_bzrdir=None, _repository=None):
1064
1096
def tree_config(self):
1065
1097
return TreeConfig(self)
1067
def _get_truncated_history(self, revision_id):
1068
history = self.revision_history()
1069
if revision_id is None:
1100
class BzrBranch5(BzrBranch):
1101
"""A format 5 branch. This supports new features over plan branches.
1103
It has support for a master_branch which is the data for bound branches.
1111
super(BzrBranch5, self).__init__(_format=_format,
1112
_control_files=_control_files,
1114
_repository=_repository)
1117
def pull(self, source, overwrite=False, stop_revision=None):
1118
"""Updates branch.pull to be bound branch aware."""
1119
bound_location = self.get_bound_location()
1120
if source.base != bound_location:
1121
# not pulling from master, so we need to update master.
1122
master_branch = self.get_master_branch()
1124
master_branch.pull(source)
1125
source = master_branch
1126
return super(BzrBranch5, self).pull(source, overwrite, stop_revision)
1128
def get_bound_location(self):
1072
idx = history.index(revision_id)
1074
raise InvalidRevisionId(revision_id=revision, branch=self)
1075
return history[:idx+1]
1130
return self.control_files.get_utf8('bound').read()[:-1]
1131
except errors.NoSuchFile:
1077
1134
@needs_read_lock
1078
def _clone_weave(self, to_location, revision=None, basis_branch=None):
1080
from bzrlib.workingtree import WorkingTree
1081
assert isinstance(to_location, basestring)
1082
if basis_branch is not None:
1083
note("basis_branch is not supported for fast weave copy yet.")
1085
history = self._get_truncated_history(revision)
1086
if not bzrlib.osutils.lexists(to_location):
1087
os.mkdir(to_location)
1088
bzrdir_to = self.bzrdir._format.initialize(to_location)
1089
self.repository.clone(bzrdir_to)
1090
branch_to = bzrdir_to.create_branch()
1091
mutter("copy branch from %s to %s", self, branch_to)
1093
# FIXME duplicate code with base .clone().
1094
# .. would template method be useful here? RBC 20051207
1095
branch_to.set_parent(self.base)
1096
branch_to.append_revision(*history)
1097
WorkingTree.create(branch_to, branch_to.base)
1135
def get_master_branch(self):
1136
"""Return the branch we are bound to.
1138
:return: Either a Branch, or None
1140
This could memoise the branch, but if thats done
1141
it must be revalidated on each new lock.
1142
So for now we just dont memoise it.
1143
# RBC 20060304 review this decision.
1145
bound_loc = self.get_bound_location()
1149
return Branch.open(bound_loc)
1150
except (errors.NotBranchError, errors.ConnectionError), e:
1151
raise errors.BoundBranchConnectionFailure(
1155
def set_bound_location(self, location):
1156
"""Set the target where this branch is bound to.
1158
:param location: URL to the target branch
1161
self.control_files.put_utf8('bound', location+'\n')
1164
self.control_files._transport.delete('bound')
1170
def bind(self, other):
1171
"""Bind the local branch the other branch.
1173
:param other: The branch to bind to
1176
# TODO: jam 20051230 Consider checking if the target is bound
1177
# It is debatable whether you should be able to bind to
1178
# a branch which is itself bound.
1179
# Committing is obviously forbidden,
1180
# but binding itself may not be.
1181
# Since we *have* to check at commit time, we don't
1182
# *need* to check here
1185
# we are now equal to or a suffix of other.
1187
# Since we have 'pulled' from the remote location,
1188
# now we should try to pull in the opposite direction
1189
# in case the local tree has more revisions than the
1191
# There may be a different check you could do here
1192
# rather than actually trying to install revisions remotely.
1193
# TODO: capture an exception which indicates the remote branch
1195
# If it is up-to-date, this probably should not be a failure
1197
# lock other for write so the revision-history syncing cannot race
1201
# if this does not error, other now has the same last rev we do
1202
# it can only error if the pull from other was concurrent with
1203
# a commit to other from someone else.
1205
# until we ditch revision-history, we need to sync them up:
1206
self.set_revision_history(other.revision_history())
1207
# now other and self are up to date with each other and have the
1208
# same revision-history.
1212
self.set_bound_location(other.base)
1216
"""If bound, unbind"""
1217
return self.set_bound_location(None)
1221
"""Synchronise this branch with the master branch if any.
1223
:return: None or the last_revision that was pivoted out during the
1226
master = self.get_master_branch()
1227
if master is not None:
1228
old_tip = self.last_revision()
1229
self.pull(master, overwrite=True)
1230
if old_tip in self.repository.get_ancestry(self.last_revision()):
1102
1236
class BranchTestProviderAdapter(object):