1
# Copyright (C) 2005, 2007, 2008 Canonical Ltd
3
# This program is free software; you can redistribute it and/or modify
4
# it under the terms of the GNU General Public License as published by
5
# the Free Software Foundation; either version 2 of the License, or
6
# (at your option) any later version.
8
# This program is distributed in the hope that it will be useful,
9
# but WITHOUT ANY WARRANTY; without even the implied warranty of
10
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11
# GNU General Public License for more details.
13
# You should have received a copy of the GNU General Public License
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
18
"""Black-box tests for bzr push."""
28
from bzrlib.branch import Branch
29
from bzrlib.bzrdir import BzrDirMetaFormat1
30
from bzrlib.osutils import abspath
31
from bzrlib.repofmt.knitrepo import RepositoryFormatKnit1
32
from bzrlib.tests.blackbox import ExternalBase
33
from bzrlib.tests.http_server import HttpServer
34
from bzrlib.transport.memory import MemoryServer, MemoryTransport
35
from bzrlib.uncommit import uncommit
36
from bzrlib.urlutils import local_path_from_url
37
from bzrlib.workingtree import WorkingTree
40
class TestPush(ExternalBase):
42
def test_push_remember(self):
43
"""Push changes from one branch to another and test push location."""
44
transport = self.get_transport()
45
tree_a = self.make_branch_and_tree('branch_a')
46
branch_a = tree_a.branch
47
self.build_tree(['branch_a/a'])
49
tree_a.commit('commit a')
50
tree_b = branch_a.bzrdir.sprout('branch_b').open_workingtree()
51
branch_b = tree_b.branch
52
tree_c = branch_a.bzrdir.sprout('branch_c').open_workingtree()
53
branch_c = tree_c.branch
54
self.build_tree(['branch_a/b'])
56
tree_a.commit('commit b')
57
self.build_tree(['branch_b/c'])
59
tree_b.commit('commit c')
60
# initial push location must be empty
61
self.assertEqual(None, branch_b.get_push_location())
63
# test push for failure without push location set
65
out = self.run_bzr('push', retcode=3)
66
self.assertEquals(out,
67
('','bzr: ERROR: No push location known or specified.\n'))
69
# test not remembered if cannot actually push
70
self.run_bzr('push ../path/which/doesnt/exist', retcode=3)
71
out = self.run_bzr('push', retcode=3)
73
('', 'bzr: ERROR: No push location known or specified.\n'),
76
# test implicit --remember when no push location set, push fails
77
out = self.run_bzr('push ../branch_b', retcode=3)
78
self.assertEquals(out,
79
('','bzr: ERROR: These branches have diverged. '
80
'Try using "merge" and then "push".\n'))
81
self.assertEquals(abspath(branch_a.get_push_location()),
82
abspath(branch_b.bzrdir.root_transport.base))
84
# test implicit --remember after resolving previous failure
85
uncommit(branch=branch_b, tree=tree_b)
86
transport.delete('branch_b/c')
87
out, err = self.run_bzr('push')
88
path = branch_a.get_push_location()
89
self.assertEquals(out,
90
'Using saved push location: %s\n'
91
'Pushed up to revision 2.\n'
92
% local_path_from_url(path))
94
'All changes applied successfully.\n')
95
self.assertEqual(path,
96
branch_b.bzrdir.root_transport.base)
97
# test explicit --remember
98
self.run_bzr('push ../branch_c --remember')
99
self.assertEquals(branch_a.get_push_location(),
100
branch_c.bzrdir.root_transport.base)
102
def test_push_without_tree(self):
103
# bzr push from a branch that does not have a checkout should work.
104
b = self.make_branch('.')
105
out, err = self.run_bzr('push pushed-location')
106
self.assertEqual('', out)
107
self.assertEqual('Created new branch.\n', err)
108
b2 = Branch.open('pushed-location')
109
self.assertEndsWith(b2.base, 'pushed-location/')
111
def test_push_new_branch_revision_count(self):
112
# bzr push of a branch with revisions to a new location
113
# should print the number of revisions equal to the length of the
115
t = self.make_branch_and_tree('tree')
116
self.build_tree(['tree/file'])
120
out, err = self.run_bzr('push pushed-to')
122
self.assertEqual('', out)
123
self.assertEqual('Created new branch.\n', err)
125
def test_push_only_pushes_history(self):
126
# Knit branches should only push the history for the current revision.
127
format = BzrDirMetaFormat1()
128
format.repository_format = RepositoryFormatKnit1()
129
shared_repo = self.make_repository('repo', format=format, shared=True)
130
shared_repo.set_make_working_trees(True)
132
def make_shared_tree(path):
133
shared_repo.bzrdir.root_transport.mkdir(path)
134
shared_repo.bzrdir.create_branch_convenience('repo/' + path)
135
return WorkingTree.open('repo/' + path)
136
tree_a = make_shared_tree('a')
137
self.build_tree(['repo/a/file'])
139
tree_a.commit('commit a-1', rev_id='a-1')
140
f = open('repo/a/file', 'ab')
141
f.write('more stuff\n')
143
tree_a.commit('commit a-2', rev_id='a-2')
145
tree_b = make_shared_tree('b')
146
self.build_tree(['repo/b/file'])
148
tree_b.commit('commit b-1', rev_id='b-1')
150
self.assertTrue(shared_repo.has_revision('a-1'))
151
self.assertTrue(shared_repo.has_revision('a-2'))
152
self.assertTrue(shared_repo.has_revision('b-1'))
154
# Now that we have a repository with shared files, make sure
155
# that things aren't copied out by a 'push'
157
self.run_bzr('push ../../push-b')
158
pushed_tree = WorkingTree.open('../../push-b')
159
pushed_repo = pushed_tree.branch.repository
160
self.assertFalse(pushed_repo.has_revision('a-1'))
161
self.assertFalse(pushed_repo.has_revision('a-2'))
162
self.assertTrue(pushed_repo.has_revision('b-1'))
164
def test_push_funky_id(self):
165
t = self.make_branch_and_tree('tree')
167
self.build_tree(['filename'])
168
t.add('filename', 'funky-chars<>%&;"\'')
169
t.commit('commit filename')
170
self.run_bzr('push ../new-tree')
172
def test_push_dash_d(self):
173
t = self.make_branch_and_tree('from')
174
t.commit(allow_pointless=True,
175
message='first commit')
176
self.run_bzr('push -d from to-one')
177
self.failUnlessExists('to-one')
178
self.run_bzr('push -d %s %s'
179
% tuple(map(urlutils.local_path_to_url, ['from', 'to-two'])))
180
self.failUnlessExists('to-two')
182
def create_simple_tree(self):
183
tree = self.make_branch_and_tree('tree')
184
self.build_tree(['tree/a'])
185
tree.add(['a'], ['a-id'])
186
tree.commit('one', rev_id='r1')
189
def test_push_create_prefix(self):
190
"""'bzr push --create-prefix' will create leading directories."""
191
tree = self.create_simple_tree()
193
self.run_bzr_error(['Parent directory of ../new/tree does not exist'],
196
self.run_bzr('push ../new/tree --create-prefix',
198
new_tree = WorkingTree.open('new/tree')
199
self.assertEqual(tree.last_revision(), new_tree.last_revision())
200
self.failUnlessExists('new/tree/a')
202
def test_push_use_existing(self):
203
"""'bzr push --use-existing-dir' can push into an existing dir.
205
By default, 'bzr push' will not use an existing, non-versioned dir.
207
tree = self.create_simple_tree()
208
self.build_tree(['target/'])
210
self.run_bzr_error(['Target directory ../target already exists',
211
'Supply --use-existing-dir',
213
'push ../target', working_dir='tree')
215
self.run_bzr('push --use-existing-dir ../target',
218
new_tree = WorkingTree.open('target')
219
self.assertEqual(tree.last_revision(), new_tree.last_revision())
220
# The push should have created target/a
221
self.failUnlessExists('target/a')
223
def test_push_onto_repo(self):
224
"""We should be able to 'bzr push' into an existing bzrdir."""
225
tree = self.create_simple_tree()
226
repo = self.make_repository('repo', shared=True)
228
self.run_bzr('push ../repo',
231
# Pushing onto an existing bzrdir will create a repository and
232
# branch as needed, but will only create a working tree if there was
234
self.assertRaises(errors.NoWorkingTree, WorkingTree.open, 'repo')
235
new_branch = Branch.open('repo')
236
self.assertEqual(tree.last_revision(), new_branch.last_revision())
238
def test_push_onto_just_bzrdir(self):
239
"""We don't handle when the target is just a bzrdir.
241
Because you shouldn't be able to create *just* a bzrdir in the wild.
243
# TODO: jam 20070109 Maybe it would be better to create the repository
245
tree = self.create_simple_tree()
246
a_bzrdir = self.make_bzrdir('dir')
248
self.run_bzr_error(['At ../dir you have a valid .bzr control'],
252
def test_push_with_revisionspec(self):
253
"""We should be able to push a revision older than the tip."""
254
tree_from = self.make_branch_and_tree('from')
255
tree_from.commit("One.", rev_id="from-1")
256
tree_from.commit("Two.", rev_id="from-2")
258
self.run_bzr('push -r1 ../to', working_dir='from')
260
tree_to = WorkingTree.open('to')
261
repo_to = tree_to.branch.repository
262
self.assertTrue(repo_to.has_revision('from-1'))
263
self.assertFalse(repo_to.has_revision('from-2'))
264
self.assertEqual(tree_to.branch.last_revision_info()[1], 'from-1')
267
"bzr: ERROR: bzr push --revision takes one value.\n",
268
'push -r0..2 ../to', working_dir='from')
270
def create_trunk_and_feature_branch(self):
272
trunk_tree = self.make_branch_and_tree('target',
273
format='development')
274
trunk_tree.commit('mainline')
275
# and a branch from it
276
branch_tree = self.make_branch_and_tree('branch',
277
format='development')
278
branch_tree.pull(trunk_tree.branch)
279
branch_tree.branch.set_parent(trunk_tree.branch.base)
280
# with some work on it
281
branch_tree.commit('moar work plz')
282
return trunk_tree, branch_tree
284
def assertPublished(self, branch_revid, stacked_on):
285
"""Assert that the branch 'published' has been published correctly."""
286
published_branch = Branch.open('published')
287
# The published branch refers to the mainline
288
self.assertEqual(stacked_on, published_branch.get_stacked_on_url())
289
# and the branch's work was pushed
290
self.assertTrue(published_branch.repository.has_revision(branch_revid))
292
def test_push_new_branch_stacked_on(self):
293
"""Pushing a new branch with --stacked-on creates a stacked branch."""
294
trunk_tree, branch_tree = self.create_trunk_and_feature_branch()
295
# we publish branch_tree with a reference to the mainline.
296
out, err = self.run_bzr(['push', '--stacked-on', trunk_tree.branch.base,
297
self.get_url('published')], working_dir='branch')
298
self.assertEqual('', out)
299
self.assertEqual('Created new stacked branch referring to %s.\n' %
300
trunk_tree.branch.base, err)
301
self.assertPublished(branch_tree.last_revision(),
302
trunk_tree.branch.base)
304
def test_push_new_branch_stacked_uses_parent_when_no_public_url(self):
305
"""When the parent has no public url the parent is used as-is."""
306
trunk_tree, branch_tree = self.create_trunk_and_feature_branch()
307
# now we do a stacked push, which should determine the public location
309
out, err = self.run_bzr(['push', '--stacked',
310
self.get_url('published')], working_dir='branch')
311
self.assertEqual('', out)
312
self.assertEqual('Created new stacked branch referring to %s.\n' %
313
trunk_tree.branch.base, err)
314
self.assertPublished(branch_tree.last_revision(), trunk_tree.branch.base)
316
def test_push_new_branch_stacked_uses_parent_public(self):
317
"""Pushing a new branch with --stacked creates a stacked branch."""
318
trunk_tree, branch_tree = self.create_trunk_and_feature_branch()
319
# the trunk is published on a web server
320
self.transport_readonly_server = HttpServer
321
trunk_public = self.make_branch('public_trunk', format='development')
322
trunk_public.pull(trunk_tree.branch)
323
trunk_public_url = self.get_readonly_url('public_trunk')
324
trunk_tree.branch.set_public_branch(trunk_public_url)
325
# now we do a stacked push, which should determine the public location
327
out, err = self.run_bzr(['push', '--stacked',
328
self.get_url('published')], working_dir='branch')
329
self.assertEqual('', out)
330
self.assertEqual('Created new stacked branch referring to %s.\n' %
331
trunk_public_url, err)
332
self.assertPublished(branch_tree.last_revision(), trunk_public_url)
334
def test_push_new_branch_stacked_no_parent(self):
335
"""Pushing with --stacked and no parent branch errors."""
336
branch = self.make_branch_and_tree('branch', format='development')
337
# now we do a stacked push, which should fail as the place to refer too
338
# cannot be determined.
339
out, err = self.run_bzr_error(
340
['Could not determine branch to refer to\\.'], ['push', '--stacked',
341
self.get_url('published')], working_dir='branch')
342
self.assertEqual('', out)
343
self.assertFalse(self.get_transport('published').has('.'))
345
def test_push_notifies_default_stacking(self):
346
self.make_branch('stack_on', format='1.6')
347
self.make_bzrdir('.').get_config().set_default_stack_on('stack_on')
348
self.make_branch('from', format='1.6')
349
out, err = self.run_bzr('push -d from to')
350
self.assertContainsRe(err,
351
'Using default stacking branch stack_on at .*')
353
def test_push_doesnt_create_broken_branch(self):
354
"""Pushing a new standalone branch works even when there's a default
355
stacking policy at the destination.
357
The new branch will preserve the repo format (even if it isn't the
358
default for the branch), and will be stacked when the repo format
359
allows (which means that the branch format isn't necessarly preserved).
361
self.make_repository('repo', shared=True, format='1.6')
362
builder = self.make_branch_builder('repo/local', format='pack-0.92')
363
builder.start_series()
364
builder.build_snapshot('rev-1', None, [
365
('add', ('', 'root-id', 'directory', '')),
366
('add', ('filename', 'f-id', 'file', 'content\n'))])
367
builder.build_snapshot('rev-2', ['rev-1'], [])
368
builder.build_snapshot('rev-3', ['rev-2'],
369
[('modify', ('f-id', 'new-content\n'))])
370
builder.finish_series()
371
branch = builder.get_branch()
372
# Push rev-1 to "trunk", so that we can stack on it.
373
self.run_bzr('push -d repo/local trunk -r 1')
374
# Set a default stacking policy so that new branches will automatically
376
self.make_bzrdir('.').get_config().set_default_stack_on('trunk')
377
# Push rev-2 to a new branch "remote". It will be stacked on "trunk".
378
out, err = self.run_bzr('push -d repo/local remote -r 2')
379
self.assertContainsRe(
380
err, 'Using default stacking branch trunk at .*')
381
# Push rev-3 onto "remote". If "remote" not stacked and is missing the
382
# fulltext record for f-id @ rev-1, then this will fail.
383
out, err = self.run_bzr('push -d repo/local remote -r 3')
386
class RedirectingMemoryTransport(MemoryTransport):
388
def mkdir(self, relpath, mode=None):
389
from bzrlib.trace import mutter
390
mutter('cwd: %r, rel: %r, abs: %r' % (self._cwd, relpath, abspath))
391
if self._cwd == '/source/':
392
raise errors.RedirectRequested(self.abspath(relpath),
393
self.abspath('../target'),
395
elif self._cwd == '/infinite-loop/':
396
raise errors.RedirectRequested(self.abspath(relpath),
397
self.abspath('../infinite-loop'),
400
return super(RedirectingMemoryTransport, self).mkdir(
403
def _redirected_to(self, source, target):
404
# We do accept redirections
405
return transport.get_transport(target)
408
class RedirectingMemoryServer(MemoryServer):
411
self._dirs = {'/': None}
414
self._scheme = 'redirecting-memory+%s:///' % id(self)
415
transport.register_transport(self._scheme, self._memory_factory)
417
def _memory_factory(self, url):
418
result = RedirectingMemoryTransport(url)
419
result._dirs = self._dirs
420
result._files = self._files
421
result._locks = self._locks
425
transport.unregister_transport(self._scheme, self._memory_factory)
428
class TestPushRedirect(ExternalBase):
431
ExternalBase.setUp(self)
432
self.memory_server = RedirectingMemoryServer()
433
self.memory_server.setUp()
434
self.addCleanup(self.memory_server.tearDown)
436
# Make the branch and tree that we'll be pushing.
437
t = self.make_branch_and_tree('tree')
438
self.build_tree(['tree/file'])
442
def test_push_redirects_on_mkdir(self):
443
"""If the push requires a mkdir, push respects redirect requests.
445
This is added primarily to handle lp:/ URI support, so that users can
446
push to new branches by specifying lp:/ URIs.
448
destination_url = self.memory_server.get_url() + 'source'
449
self.run_bzr(['push', '-d', 'tree', destination_url])
451
local_revision = Branch.open('tree').last_revision()
452
remote_revision = Branch.open(
453
self.memory_server.get_url() + 'target').last_revision()
454
self.assertEqual(remote_revision, local_revision)
456
def test_push_gracefully_handles_too_many_redirects(self):
457
"""Push fails gracefully if the mkdir generates a large number of
460
destination_url = self.memory_server.get_url() + 'infinite-loop'
461
out, err = self.run_bzr_error(
462
['Too many redirections trying to make %s\\.\n'
463
% re.escape(destination_url)],
464
['push', '-d', 'tree', destination_url], retcode=3)
465
self.assertEqual('', out)