27
26
from bzrlib.tests import (
28
27
TestCaseWithTransport,
30
from bzrlib.tests.features import (
31
CaseInsensitiveFilesystemFeature,
33
UnicodeFilenameFeature,
37
32
class TestMove(TestCaseWithTransport):
34
def assertInWorkingTree(self,path):
35
tree = workingtree.WorkingTree.open('.')
36
self.assertIsNot(tree.path2id(path), None,
37
path+' not in working tree.')
39
def assertNotInWorkingTree(self,path):
40
tree = workingtree.WorkingTree.open('.')
41
self.assertIs(tree.path2id(path), None, path+' in working tree.')
39
43
def assertMoved(self,from_path,to_path):
40
44
"""Assert that to_path is existing and versioned but from_path not. """
41
self.assertPathDoesNotExist(from_path)
45
self.failIfExists(from_path)
42
46
self.assertNotInWorkingTree(from_path)
44
self.assertPathExists(to_path)
48
self.failUnlessExists(to_path)
45
49
self.assertInWorkingTree(to_path)
47
51
def test_mv_modes(self):
50
54
files = self.build_tree(['a', 'c', 'subdir/'])
51
55
tree.add(['a', 'c', 'subdir'])
53
self.run_bzr('mv a b')
57
self.run_bzr('mv', 'a', 'b')
54
58
self.assertMoved('a','b')
56
self.run_bzr('mv b subdir')
60
self.run_bzr('mv', 'b', 'subdir')
57
61
self.assertMoved('b','subdir/b')
59
self.run_bzr('mv subdir/b a')
63
self.run_bzr('mv', 'subdir/b', 'a')
60
64
self.assertMoved('subdir/b','a')
62
self.run_bzr('mv a c subdir')
66
self.run_bzr('mv', 'a', 'c', 'subdir')
63
67
self.assertMoved('a','subdir/a')
64
68
self.assertMoved('c','subdir/c')
66
self.run_bzr('mv subdir/a subdir/newa')
70
self.run_bzr('mv', 'subdir/a', 'subdir/newa')
67
71
self.assertMoved('subdir/a','subdir/newa')
69
73
def test_mv_unversioned(self):
70
74
self.build_tree(['unversioned.txt'])
71
75
self.run_bzr_error(
72
76
["^bzr: ERROR: Could not rename unversioned.txt => elsewhere."
73
" .*unversioned.txt is not versioned\.$"],
74
'mv unversioned.txt elsewhere')
77
" .*unversioned.txt is not versioned$"],
78
'mv', 'unversioned.txt', 'elsewhere')
76
80
def test_mv_nonexisting(self):
77
81
self.run_bzr_error(
78
82
["^bzr: ERROR: Could not rename doesnotexist => somewhereelse."
79
" .*doesnotexist is not versioned\.$"],
80
'mv doesnotexist somewhereelse')
83
" .*doesnotexist is not versioned$"],
84
'mv', 'doesnotexist', 'somewhereelse')
82
86
def test_mv_unqualified(self):
83
87
self.run_bzr_error(['^bzr: ERROR: missing file argument$'], 'mv')
85
89
def test_mv_invalid(self):
86
90
tree = self.make_branch_and_tree('.')
87
91
self.build_tree(['test.txt', 'sub1/'])
88
92
tree.add(['test.txt'])
90
94
self.run_bzr_error(
91
["^bzr: ERROR: Could not move to sub1: sub1 is not versioned\.$"],
95
["^bzr: ERROR: Could not move to sub1: sub1 is not versioned$"],
96
'mv', 'test.txt', 'sub1')
94
98
self.run_bzr_error(
95
99
["^bzr: ERROR: Could not move test.txt => .*hello.txt: "
96
"sub1 is not versioned\.$"],
97
'mv test.txt sub1/hello.txt')
100
"sub1 is not versioned$"],
101
'mv', 'test.txt', 'sub1/hello.txt')
99
103
def test_mv_dirs(self):
100
104
tree = self.make_branch_and_tree('.')
101
105
self.build_tree(['hello.txt', 'sub1/'])
102
106
tree.add(['hello.txt', 'sub1'])
104
self.run_bzr('mv sub1 sub2')
108
self.run_bzr('mv', 'sub1', 'sub2')
105
109
self.assertMoved('sub1','sub2')
107
self.run_bzr('mv hello.txt sub2')
111
self.run_bzr('mv', 'hello.txt', 'sub2')
108
112
self.assertMoved('hello.txt','sub2/hello.txt')
110
114
self.build_tree(['sub1/'])
111
115
tree.add(['sub1'])
112
self.run_bzr('mv sub2/hello.txt sub1')
116
self.run_bzr('mv', 'sub2/hello.txt', 'sub1')
113
117
self.assertMoved('sub2/hello.txt','sub1/hello.txt')
115
self.run_bzr('mv sub2 sub1')
119
self.run_bzr('mv', 'sub2', 'sub1')
116
120
self.assertMoved('sub2','sub1/sub2')
118
122
def test_mv_relative(self):
121
125
tree.add(['sub1', 'sub1/sub2', 'sub1/hello.txt'])
123
127
os.chdir('sub1/sub2')
124
self.run_bzr('mv ../hello.txt .')
125
self.assertPathExists('./hello.txt')
128
self.run_bzr('mv', '../hello.txt', '.')
129
self.failUnlessExists('./hello.txt')
128
self.run_bzr('mv sub2/hello.txt .')
132
self.run_bzr('mv', 'sub2/hello.txt', '.')
130
134
self.assertMoved('sub1/sub2/hello.txt','sub1/hello.txt')
132
def test_mv_change_case_file(self):
133
# test for bug #77740 (mv unable change filename case on Windows)
134
tree = self.make_branch_and_tree('.')
135
self.build_tree(['test.txt'])
136
tree.add(['test.txt'])
137
self.run_bzr('mv test.txt Test.txt')
138
# we can't use failUnlessExists on case-insensitive filesystem
139
# so try to check shape of the tree
140
shape = sorted(os.listdir(u'.'))
141
self.assertEqual(['.bzr', 'Test.txt'], shape)
142
self.assertInWorkingTree('Test.txt')
143
self.assertNotInWorkingTree('test.txt')
145
def test_mv_change_case_dir(self):
146
tree = self.make_branch_and_tree('.')
147
self.build_tree(['foo/'])
149
self.run_bzr('mv foo Foo')
150
# we can't use failUnlessExists on case-insensitive filesystem
151
# so try to check shape of the tree
152
shape = sorted(os.listdir(u'.'))
153
self.assertEqual(['.bzr', 'Foo'], shape)
154
self.assertInWorkingTree('Foo')
155
self.assertNotInWorkingTree('foo')
157
def test_mv_change_case_dir_w_files(self):
158
tree = self.make_branch_and_tree('.')
159
self.build_tree(['foo/', 'foo/bar'])
161
self.run_bzr('mv foo Foo')
162
# we can't use failUnlessExists on case-insensitive filesystem
163
# so try to check shape of the tree
164
shape = sorted(os.listdir(u'.'))
165
self.assertEqual(['.bzr', 'Foo'], shape)
166
self.assertInWorkingTree('Foo')
167
self.assertNotInWorkingTree('foo')
169
def test_mv_file_to_wrong_case_dir(self):
170
self.requireFeature(CaseInsensitiveFilesystemFeature)
171
tree = self.make_branch_and_tree('.')
172
self.build_tree(['foo/', 'bar'])
173
tree.add(['foo', 'bar'])
174
out, err = self.run_bzr('mv bar Foo', retcode=3)
175
self.assertEquals('', out)
177
'bzr: ERROR: Could not move to Foo: Foo is not versioned.\n',
180
136
def test_mv_smoke_aliases(self):
181
137
# just test that aliases for mv exist, if their behaviour is changed in
182
138
# the future, then extend the tests.
184
140
tree = self.make_branch_and_tree('.')
187
self.run_bzr('move a b')
188
self.run_bzr('rename b a')
143
self.run_bzr('move', 'a', 'b')
144
self.run_bzr('rename', 'b', 'a')
190
146
def test_mv_through_symlinks(self):
191
self.requireFeature(SymlinkFeature)
147
if not osutils.has_symlinks():
148
raise TestSkipped('Symlinks are not supported on this platform')
192
149
tree = self.make_branch_and_tree('.')
193
150
self.build_tree(['a/', 'a/b'])
194
151
os.symlink('a', 'c')
195
152
os.symlink('.', 'd')
196
153
tree.add(['a', 'a/b', 'c'], ['a-id', 'b-id', 'c-id'])
197
self.run_bzr('mv c/b b')
154
self.run_bzr('mv', 'c/b', 'b')
198
155
tree = workingtree.WorkingTree.open('.')
199
156
self.assertEqual('b-id', tree.path2id('b'))
404
359
self.build_tree(['a1']) #touch a1
405
360
self.build_tree(['a2']) #touch a2
407
self.run_bzr('mv a1 a2 sub --after')
408
self.assertPathExists('a1')
409
self.assertPathExists('a2')
410
self.assertPathExists('sub/a1')
411
self.assertPathExists('sub/a2')
362
self.run_bzr('mv', 'a1', 'a2', 'sub', '--after')
363
self.failUnlessExists('a1')
364
self.failUnlessExists('a2')
365
self.failUnlessExists('sub/a1')
366
self.failUnlessExists('sub/a2')
412
367
self.assertInWorkingTree('sub/a1')
413
368
self.assertInWorkingTree('sub/a2')
415
def test_mv_already_moved_directory(self):
416
"""Use `bzr mv a b` to mark a directory as renamed.
418
https://bugs.launchpad.net/bzr/+bug/107967/
420
self.build_tree(['a/', 'c/'])
421
tree = self.make_branch_and_tree('.')
423
osutils.rename('a', 'b')
424
osutils.rename('c', 'd')
425
# mv a b should work just like it does for already renamed files
426
self.run_bzr('mv a b')
427
self.assertPathDoesNotExist('a')
428
self.assertNotInWorkingTree('a')
429
self.assertPathExists('b')
430
self.assertInWorkingTree('b')
431
# and --after should work, too (technically it's ignored)
432
self.run_bzr('mv --after c d')
433
self.assertPathDoesNotExist('c')
434
self.assertNotInWorkingTree('c')
435
self.assertPathExists('d')
436
self.assertInWorkingTree('d')
438
def make_abcd_tree(self):
439
tree = self.make_branch_and_tree('tree')
440
self.build_tree(['tree/a', 'tree/c'])
442
tree.commit('record old names')
443
osutils.rename('tree/a', 'tree/b')
444
osutils.rename('tree/c', 'tree/d')
447
def test_mv_auto(self):
448
self.make_abcd_tree()
449
out, err = self.run_bzr('mv --auto', working_dir='tree')
450
self.assertEqual(out, '')
451
self.assertEqual(err, 'a => b\nc => d\n')
452
tree = workingtree.WorkingTree.open('tree')
453
self.assertIsNot(None, tree.path2id('b'))
454
self.assertIsNot(None, tree.path2id('d'))
456
def test_mv_auto_one_path(self):
457
self.make_abcd_tree()
458
out, err = self.run_bzr('mv --auto tree')
459
self.assertEqual(out, '')
460
self.assertEqual(err, 'a => b\nc => d\n')
461
tree = workingtree.WorkingTree.open('tree')
462
self.assertIsNot(None, tree.path2id('b'))
463
self.assertIsNot(None, tree.path2id('d'))
465
def test_mv_auto_two_paths(self):
466
self.make_abcd_tree()
467
out, err = self.run_bzr('mv --auto tree tree2', retcode=3)
468
self.assertEqual('bzr: ERROR: Only one path may be specified to'
471
def test_mv_auto_dry_run(self):
472
self.make_abcd_tree()
473
out, err = self.run_bzr('mv --auto --dry-run', working_dir='tree')
474
self.assertEqual(out, '')
475
self.assertEqual(err, 'a => b\nc => d\n')
476
tree = workingtree.WorkingTree.open('tree')
477
self.assertIsNot(None, tree.path2id('a'))
478
self.assertIsNot(None, tree.path2id('c'))
480
def test_mv_no_auto_dry_run(self):
481
self.make_abcd_tree()
482
out, err = self.run_bzr('mv c d --dry-run',
483
working_dir='tree', retcode=3)
484
self.assertEqual('bzr: ERROR: --dry-run requires --auto.\n', err)
486
def test_mv_auto_after(self):
487
self.make_abcd_tree()
488
out, err = self.run_bzr('mv --auto --after', working_dir='tree',
490
self.assertEqual('bzr: ERROR: --after cannot be specified with'
493
def test_mv_quiet(self):
494
tree = self.make_branch_and_tree('.')
495
self.build_tree(['aaa'])
497
out, err = self.run_bzr('mv --quiet aaa bbb')
498
self.assertEqual(out, '')
499
self.assertEqual(err, '')
501
def test_mv_readonly_lightweight_checkout(self):
502
branch = self.make_branch('foo')
503
branch = bzrlib.branch.Branch.open(self.get_readonly_url('foo'))
504
tree = branch.create_checkout('tree', lightweight=True)
505
self.build_tree(['tree/path'])
507
# If this fails, the tree is trying to acquire a branch lock, which it
509
self.run_bzr(['mv', 'tree/path', 'tree/path2'])
511
def test_mv_unversioned_non_ascii(self):
512
"""Clear error on mv of an unversioned non-ascii file, see lp:707954"""
513
self.requireFeature(UnicodeFilenameFeature)
514
tree = self.make_branch_and_tree(".")
515
self.build_tree([u"\xA7"])
516
out, err = self.run_bzr_error(["Could not rename", "not versioned"],
517
["mv", u"\xA7", "b"])