3222
3225
class GenericInterBranch(InterBranch):
3223
"""InterBranch implementation that uses public Branch functions.
3226
"""InterBranch implementation that uses public Branch functions."""
3229
def is_compatible(klass, source, target):
3230
# GenericBranch uses the public API, so always compatible
3227
3234
def _get_branch_formats_to_test():
3228
3235
return BranchFormat._default_format, BranchFormat._default_format
3238
def unwrap_format(klass, format):
3239
if isinstance(format, remote.RemoteBranchFormat):
3240
format._ensure_real()
3241
return format._custom_format
3245
def copy_content_into(self, revision_id=None):
3246
"""Copy the content of source into target
3248
revision_id: if not None, the revision history in the new branch will
3249
be truncated to end with revision_id.
3251
self.source.update_references(self.target)
3252
self.source._synchronize_history(self.target, revision_id)
3254
parent = self.source.get_parent()
3255
except errors.InaccessibleParent, e:
3256
mutter('parent was not accessible to copy: %s', e)
3259
self.target.set_parent(parent)
3260
if self.source._push_should_merge_tags():
3261
self.source.tags.merge_to(self.target.tags)
3230
3264
def update_revisions(self, stop_revision=None, overwrite=False,
3232
3266
"""See InterBranch.update_revisions()."""
3233
self.source.lock_read()
3235
other_revno, other_last_revision = self.source.last_revision_info()
3236
stop_revno = None # unknown
3237
if stop_revision is None:
3238
stop_revision = other_last_revision
3239
if _mod_revision.is_null(stop_revision):
3240
# if there are no commits, we're done.
3242
stop_revno = other_revno
3267
other_revno, other_last_revision = self.source.last_revision_info()
3268
stop_revno = None # unknown
3269
if stop_revision is None:
3270
stop_revision = other_last_revision
3271
if _mod_revision.is_null(stop_revision):
3272
# if there are no commits, we're done.
3274
stop_revno = other_revno
3244
# what's the current last revision, before we fetch [and change it
3246
last_rev = _mod_revision.ensure_null(self.target.last_revision())
3247
# we fetch here so that we don't process data twice in the common
3248
# case of having something to pull, and so that the check for
3249
# already merged can operate on the just fetched graph, which will
3250
# be cached in memory.
3251
self.target.fetch(self.source, stop_revision)
3252
# Check to see if one is an ancestor of the other
3255
graph = self.target.repository.get_graph()
3256
if self.target._check_if_descendant_or_diverged(
3257
stop_revision, last_rev, graph, self.source):
3258
# stop_revision is a descendant of last_rev, but we aren't
3259
# overwriting, so we're done.
3261
if stop_revno is None:
3263
graph = self.target.repository.get_graph()
3264
this_revno, this_last_revision = \
3265
self.target.last_revision_info()
3266
stop_revno = graph.find_distance_to_null(stop_revision,
3267
[(other_last_revision, other_revno),
3268
(this_last_revision, this_revno)])
3269
self.target.set_last_revision_info(stop_revno, stop_revision)
3271
self.source.unlock()
3276
# what's the current last revision, before we fetch [and change it
3278
last_rev = _mod_revision.ensure_null(self.target.last_revision())
3279
# we fetch here so that we don't process data twice in the common
3280
# case of having something to pull, and so that the check for
3281
# already merged can operate on the just fetched graph, which will
3282
# be cached in memory.
3283
self.target.fetch(self.source, stop_revision)
3284
# Check to see if one is an ancestor of the other
3287
graph = self.target.repository.get_graph()
3288
if self.target._check_if_descendant_or_diverged(
3289
stop_revision, last_rev, graph, self.source):
3290
# stop_revision is a descendant of last_rev, but we aren't
3291
# overwriting, so we're done.
3293
if stop_revno is None:
3295
graph = self.target.repository.get_graph()
3296
this_revno, this_last_revision = \
3297
self.target.last_revision_info()
3298
stop_revno = graph.find_distance_to_null(stop_revision,
3299
[(other_last_revision, other_revno),
3300
(this_last_revision, this_revno)])
3301
self.target.set_last_revision_info(stop_revno, stop_revision)
3273
3303
def pull(self, overwrite=False, stop_revision=None,
3274
possible_transports=None, _hook_master=None, run_hooks=True,
3304
possible_transports=None, run_hooks=True,
3275
3305
_override_hook_target=None, local=False):
3306
"""Pull from source into self, updating my master if any.
3278
:param _hook_master: Private parameter - set the branch to
3279
be supplied as the master to pull hooks.
3280
3308
:param run_hooks: Private parameter - if false, this branch
3281
3309
is being called because it's the master of the primary branch,
3282
3310
so it should not run its hooks.
3283
:param _override_hook_target: Private parameter - set the branch to be
3284
supplied as the target_branch to pull hooks.
3285
:param local: Only update the local branch, and not the bound branch.
3287
# This type of branch can't be bound.
3312
bound_location = self.target.get_bound_location()
3313
if local and not bound_location:
3289
3314
raise errors.LocalRequiresBoundBranch()
3290
result = PullResult()
3291
result.source_branch = self.source
3292
if _override_hook_target is None:
3293
result.target_branch = self.target
3295
result.target_branch = _override_hook_target
3296
self.source.lock_read()
3315
master_branch = None
3316
if not local and bound_location and self.source.user_url != bound_location:
3317
# not pulling from master, so we need to update master.
3318
master_branch = self.target.get_master_branch(possible_transports)
3319
master_branch.lock_write()
3298
# We assume that during 'pull' the target repository is closer than
3300
self.source.update_references(self.target)
3301
graph = self.target.repository.get_graph(self.source.repository)
3302
# TODO: Branch formats should have a flag that indicates
3303
# that revno's are expensive, and pull() should honor that flag.
3305
result.old_revno, result.old_revid = \
3306
self.target.last_revision_info()
3307
self.target.update_revisions(self.source, stop_revision,
3308
overwrite=overwrite, graph=graph)
3309
# TODO: The old revid should be specified when merging tags,
3310
# so a tags implementation that versions tags can only
3311
# pull in the most recent changes. -- JRV20090506
3312
result.tag_conflicts = self.source.tags.merge_to(self.target.tags,
3314
result.new_revno, result.new_revid = self.target.last_revision_info()
3316
result.master_branch = _hook_master
3317
result.local_branch = result.target_branch
3319
result.master_branch = result.target_branch
3320
result.local_branch = None
3322
for hook in Branch.hooks['post_pull']:
3322
# pull from source into master.
3323
master_branch.pull(self.source, overwrite, stop_revision,
3325
return self._pull(overwrite,
3326
stop_revision, _hook_master=master_branch,
3327
run_hooks=run_hooks,
3328
_override_hook_target=_override_hook_target)
3325
self.source.unlock()
3331
master_branch.unlock()
3328
3333
def push(self, overwrite=False, stop_revision=None,
3329
3334
_override_hook_source_branch=None):
3395
def is_compatible(self, source, target):
3396
# GenericBranch uses the public API, so always compatible
3400
class InterToBranch5(GenericInterBranch):
3403
def _get_branch_formats_to_test():
3404
return BranchFormat._default_format, BzrBranchFormat5()
3406
def pull(self, overwrite=False, stop_revision=None,
3407
possible_transports=None, run_hooks=True,
3399
def _pull(self, overwrite=False, stop_revision=None,
3400
possible_transports=None, _hook_master=None, run_hooks=True,
3408
3401
_override_hook_target=None, local=False):
3409
"""Pull from source into self, updating my master if any.
3404
This function is the core worker, used by GenericInterBranch.pull to
3405
avoid duplication when pulling source->master and source->local.
3407
:param _hook_master: Private parameter - set the branch to
3408
be supplied as the master to pull hooks.
3411
3409
:param run_hooks: Private parameter - if false, this branch
3412
3410
is being called because it's the master of the primary branch,
3413
3411
so it should not run its hooks.
3412
:param _override_hook_target: Private parameter - set the branch to be
3413
supplied as the target_branch to pull hooks.
3414
:param local: Only update the local branch, and not the bound branch.
3415
bound_location = self.target.get_bound_location()
3416
if local and not bound_location:
3416
# This type of branch can't be bound.
3417
3418
raise errors.LocalRequiresBoundBranch()
3418
master_branch = None
3419
if not local and bound_location and self.source.user_url != bound_location:
3420
# not pulling from master, so we need to update master.
3421
master_branch = self.target.get_master_branch(possible_transports)
3422
master_branch.lock_write()
3419
result = PullResult()
3420
result.source_branch = self.source
3421
if _override_hook_target is None:
3422
result.target_branch = self.target
3424
result.target_branch = _override_hook_target
3425
self.source.lock_read()
3425
# pull from source into master.
3426
master_branch.pull(self.source, overwrite, stop_revision,
3428
return super(InterToBranch5, self).pull(overwrite,
3429
stop_revision, _hook_master=master_branch,
3430
run_hooks=run_hooks,
3431
_override_hook_target=_override_hook_target)
3427
# We assume that during 'pull' the target repository is closer than
3429
self.source.update_references(self.target)
3430
graph = self.target.repository.get_graph(self.source.repository)
3431
# TODO: Branch formats should have a flag that indicates
3432
# that revno's are expensive, and pull() should honor that flag.
3434
result.old_revno, result.old_revid = \
3435
self.target.last_revision_info()
3436
self.target.update_revisions(self.source, stop_revision,
3437
overwrite=overwrite, graph=graph)
3438
# TODO: The old revid should be specified when merging tags,
3439
# so a tags implementation that versions tags can only
3440
# pull in the most recent changes. -- JRV20090506
3441
result.tag_conflicts = self.source.tags.merge_to(self.target.tags,
3443
result.new_revno, result.new_revid = self.target.last_revision_info()
3445
result.master_branch = _hook_master
3446
result.local_branch = result.target_branch
3448
result.master_branch = result.target_branch
3449
result.local_branch = None
3451
for hook in Branch.hooks['post_pull']:
3434
master_branch.unlock()
3454
self.source.unlock()
3437
3458
InterBranch.register_optimiser(GenericInterBranch)
3438
InterBranch.register_optimiser(InterToBranch5)