13
13
# You should have received a copy of the GNU General Public License
14
14
# along with this program; if not, write to the Free Software
15
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
15
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
17
17
"""Tests for Branch.get_stacked_on_url and set_stacked_on_url."""
19
19
from bzrlib import (
24
25
from bzrlib.revision import NULL_REVISION
25
26
from bzrlib.smart import server
26
27
from bzrlib.tests import TestNotApplicable, KnownFailure, transport_util
27
from bzrlib.tests.branch_implementations import TestCaseWithBranch
28
from bzrlib.tests.per_branch import TestCaseWithBranch
28
29
from bzrlib.transport import get_transport
77
80
self.assertEqual('../target', branch.get_stacked_on_url())
82
def test_set_stacked_on_same_branch_raises(self):
83
# Stacking on the same branch silently raises and doesn't execute the
84
# change. Reported in bug 376243.
85
branch = self.make_branch('branch')
87
self.assertRaises(errors.UnstackableLocationError,
88
branch.set_stacked_on_url, '../branch')
89
except unstackable_format_errors:
90
# if the set failed, so must the get
91
self.assertRaises(unstackable_format_errors, branch.get_stacked_on_url)
93
self.assertRaises(errors.NotStacked, branch.get_stacked_on_url)
95
def test_set_stacked_on_same_branch_after_being_stacked_raises(self):
96
# Stacking on the same branch silently raises and doesn't execute the
98
branch = self.make_branch('branch')
99
target = self.make_branch('target')
101
branch.set_stacked_on_url('../target')
102
except unstackable_format_errors:
103
# if the set failed, so must the get
104
self.assertRaises(unstackable_format_errors, branch.get_stacked_on_url)
106
self.assertRaises(errors.UnstackableLocationError,
107
branch.set_stacked_on_url, '../branch')
108
self.assertEqual('../target', branch.get_stacked_on_url())
79
110
def assertRevisionInRepository(self, repo_path, revid):
80
111
"""Check that a revision is in a repository, disregarding stacking."""
81
112
repo = bzrdir.BzrDir.open(repo_path).open_repository()
119
150
raise TestNotApplicable(e)
120
151
# stacked repository
121
152
self.assertRevisionNotInRepository('newbranch', trunk_revid)
122
new_tree = new_dir.open_workingtree()
123
new_branch_revid = new_tree.commit('something local')
153
tree = new_dir.open_branch().create_checkout('local')
154
new_branch_revid = tree.commit('something local')
124
155
self.assertRevisionNotInRepository('mainline', new_branch_revid)
125
156
self.assertRevisionInRepository('newbranch', new_branch_revid)
127
# XXX: this helper probably belongs on TestCaseWithTransport
128
def make_smart_server(self, path):
129
smart_server = server.SmartTCPServer_for_testing()
130
smart_server.setUp(self.get_server())
131
remote_transport = get_transport(smart_server.get_url()).clone(path)
132
self.addCleanup(smart_server.tearDown)
133
return remote_transport
135
158
def test_sprout_stacked_from_smart_server(self):
136
159
if isinstance(self.branch_format, branch.BzrBranchFormat4):
137
160
raise TestNotApplicable('Branch format 4 is not usable via HPSS.')
162
185
trunk_revid = trunk_tree.commit('revision on mainline')
163
186
# and make branch from it which is stacked
165
new_dir = trunk_tree.bzrdir.sprout('newbranch', stacked=True)
188
new_dir = trunk_tree.bzrdir.sprout(self.get_url('newbranch'),
166
190
except unstackable_format_errors, e:
167
191
raise TestNotApplicable(e)
168
192
# stacked repository
169
193
self.assertRevisionNotInRepository('newbranch', trunk_revid)
194
# TODO: we'd like to commit in the stacked repository; that requires
195
# some care (maybe a BranchBuilder) if it's remote and has no
197
##newbranch_revid = new_dir.open_workingtree().commit('revision in '
170
199
# now when we unstack that should implicitly fetch, to make sure that
171
200
# the branch will still work
172
201
new_branch = new_dir.open_branch()
245
274
self.assertRaises((errors.NotStacked, errors.UnstackableBranchFormat),
246
275
cloned_bzrdir.open_branch().get_stacked_on_url)
277
def make_stacked_on_matching(self, source):
278
if source.repository.supports_rich_root():
279
if source.repository._format.supports_chks:
282
format = "1.9-rich-root"
285
return self.make_branch('stack-on', format)
248
287
def test_sprout_stacking_policy_handling(self):
249
288
"""Obey policy where possible, ignore otherwise."""
250
stack_on = self.make_branch('stack-on')
289
if isinstance(self.branch_format, branch.BzrBranchFormat4):
290
raise TestNotApplicable('Branch format 4 does not autoupgrade.')
291
source = self.make_branch('source')
292
stack_on = self.make_stacked_on_matching(source)
251
293
parent_bzrdir = self.make_bzrdir('.', format='default')
252
294
parent_bzrdir.get_config().set_default_stack_on('stack-on')
253
source = self.make_branch('source')
254
295
target = source.bzrdir.sprout('target').open_branch()
296
# When we sprout we upgrade the branch when there is a default stack_on
297
# set by a config *and* the targeted branch supports stacking.
298
if stack_on._format.supports_stacking():
256
299
self.assertEqual('../stack-on', target.get_stacked_on_url())
257
except errors.UnstackableBranchFormat:
302
errors.UnstackableBranchFormat, target.get_stacked_on_url)
260
304
def test_clone_stacking_policy_handling(self):
261
305
"""Obey policy where possible, ignore otherwise."""
262
stack_on = self.make_branch('stack-on')
306
if isinstance(self.branch_format, branch.BzrBranchFormat4):
307
raise TestNotApplicable('Branch format 4 does not autoupgrade.')
308
source = self.make_branch('source')
309
stack_on = self.make_stacked_on_matching(source)
263
310
parent_bzrdir = self.make_bzrdir('.', format='default')
264
311
parent_bzrdir.get_config().set_default_stack_on('stack-on')
265
source = self.make_branch('source')
266
312
target = source.bzrdir.clone('target').open_branch()
268
self.assertEqual('../stack-on', target.get_stacked_on_url())
269
except errors.UnstackableBranchFormat:
313
# When we clone we upgrade the branch when there is a default stack_on
314
# set by a config *and* the targeted branch supports stacking.
315
if stack_on._format.supports_stacking():
316
self.assertEqual('../stack-on', target.get_stacked_on_url())
319
errors.UnstackableBranchFormat, target.get_stacked_on_url)
321
def test_sprout_to_smart_server_stacking_policy_handling(self):
322
"""Obey policy where possible, ignore otherwise."""
323
if isinstance(self.branch_format, branch.BzrBranchFormat4):
324
raise TestNotApplicable('Branch format 4 is not usable via HPSS.')
325
source = self.make_branch('source')
326
stack_on = self.make_stacked_on_matching(source)
327
parent_bzrdir = self.make_bzrdir('.', format='default')
328
parent_bzrdir.get_config().set_default_stack_on('stack-on')
329
url = self.make_smart_server('target').base
330
target = source.bzrdir.sprout(url).open_branch()
331
# When we sprout we upgrade the branch when there is a default stack_on
332
# set by a config *and* the targeted branch supports stacking.
333
if stack_on._format.supports_stacking():
334
self.assertEqual('../stack-on', target.get_stacked_on_url())
337
errors.UnstackableBranchFormat, target.get_stacked_on_url)
272
339
def prepare_stacked_on_fetch(self):
273
340
stack_on = self.make_branch_and_tree('stack-on')
299
367
# repository boundaries. however, i didn't actually get this test to
300
368
# fail on that code. -- mbp
301
369
# see https://bugs.launchpad.net/bzr/+bug/252821
302
if not self.branch_format.supports_stacking():
370
stack_on = self.make_branch_and_tree('stack-on')
371
if not stack_on.branch._format.supports_stacking():
303
372
raise TestNotApplicable("%r does not support stacking"
304
373
% self.branch_format)
305
stack_on = self.make_branch_and_tree('stack-on')
306
374
text_lines = ['line %d blah blah blah\n' % i for i in range(20)]
307
375
self.build_tree_contents([('stack-on/a', ''.join(text_lines))])
308
376
stack_on.add('a')
309
377
stack_on.commit('base commit')
310
378
stacked_dir = stack_on.bzrdir.sprout('stacked', stacked=True)
311
stacked_tree = stacked_dir.open_workingtree()
379
stacked_branch = stacked_dir.open_branch()
380
local_tree = stack_on.bzrdir.sprout('local').open_workingtree()
312
381
for i in range(20):
313
382
text_lines[0] = 'changed in %d\n' % i
314
self.build_tree_contents([('stacked/a', ''.join(text_lines))])
315
stacked_tree.commit('commit %d' % i)
316
stacked_tree.branch.repository.pack()
317
stacked_tree.branch.check()
383
self.build_tree_contents([('local/a', ''.join(text_lines))])
384
local_tree.commit('commit %d' % i)
385
local_tree.branch.push(stacked_branch)
386
stacked_branch.repository.pack()
387
check.check_dwim(stacked_branch.base, False, True, True)
319
389
def test_pull_delta_when_stacked(self):
320
390
if not self.branch_format.supports_stacking():
409
479
# Ensure that opening the branch doesn't raise.
410
480
branch.Branch.open(transport.base)
482
def test_revision_history_of_stacked(self):
483
# See <https://launchpad.net/bugs/380314>.
484
stack_on = self.make_branch_and_tree('stack-on')
485
stack_on.commit('first commit', rev_id='rev1')
487
stacked_dir = stack_on.bzrdir.sprout(
488
self.get_url('stacked'), stacked=True)
489
except unstackable_format_errors, e:
490
raise TestNotApplicable('Format does not support stacking.')
492
stacked = stacked_dir.open_workingtree()
493
except errors.NoWorkingTree:
494
stacked = stacked_dir.open_branch().create_checkout(
495
'stacked-checkout', lightweight=True)
496
tree = stacked.branch.create_checkout('local')
497
tree.commit('second commit', rev_id='rev2')
498
# Sanity check: stacked's repo should not contain rev1, otherwise this
499
# test isn't testing what it's supposed to.
500
repo = stacked.branch.repository.bzrdir.open_repository()
502
self.addCleanup(repo.unlock)
503
self.assertEqual({}, repo.get_parent_map(['rev1']))
504
# revision_history should work, even though the history is spread over
505
# multiple repositories.
506
self.assertLength(2, stacked.branch.revision_history())
413
509
class TestStackingConnections(
414
510
transport_util.TestCaseWithConnectionHookedTransport):