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 interface conformance of 'WorkingTree.remove'"""
19
19
from bzrlib.tests.workingtree_implementations import TestCaseWithWorkingTree
20
from bzrlib import errors, osutils
20
from bzrlib import errors, ignores, osutils
22
22
class TestRemove(TestCaseWithWorkingTree):
23
23
"""Tests WorkingTree.remove"""
25
25
files = ['a', 'b/', 'b/c', 'd/']
26
26
rfiles = ['b/c', 'b', 'a', 'd']
28
def get_tree(self, files):
32
29
tree = self.make_branch_and_tree('.')
33
self.build_tree(TestRemove.files)
30
self.build_tree(files)
31
self.failUnlessExists(files)
34
def get_committed_tree(self, files, message="Committing"):
35
tree = self.get_tree(files)
38
self.assertInWorkingTree(files)
41
def assertRemovedAndDeleted(self, files):
42
self.assertNotInWorkingTree(files)
43
self.failIfExists(files)
45
def assertRemovedAndNotDeleted(self, files):
46
self.assertNotInWorkingTree(files)
47
self.failUnlessExists(files)
36
49
def test_remove_keep(self):
37
"""Check that files are unversioned but not deleted."""
50
"""Check that files and directories are unversioned but not deleted."""
51
tree = self.get_tree(TestRemove.files)
39
52
tree.add(TestRemove.files)
40
53
self.assertInWorkingTree(TestRemove.files)
42
55
tree.remove(TestRemove.files)
43
self.assertNotInWorkingTree(TestRemove.files)
44
self.failUnlessExists(TestRemove.files)
56
self.assertRemovedAndNotDeleted(TestRemove.files)
58
def test_remove_keep_subtree(self):
59
"""Check that a directory is unversioned but not deleted."""
60
tree = self.make_branch_and_tree('.')
61
subtree = self.make_branch_and_tree('subtree')
62
tree.add('subtree', 'subtree-id')
64
tree.remove('subtree')
65
self.assertRemovedAndNotDeleted('subtree')
46
67
def test_remove_unchanged_files(self):
47
68
"""Check that unchanged files are removed and deleted."""
49
tree.add(TestRemove.files)
50
tree.commit("files must not have changes")
51
self.assertInWorkingTree(TestRemove.files)
69
tree = self.get_committed_tree(TestRemove.files)
53
70
tree.remove(TestRemove.files, keep_files=False)
55
self.assertNotInWorkingTree(TestRemove.files)
56
self.failIfExists(TestRemove.files)
71
self.assertRemovedAndDeleted(TestRemove.files)
58
74
def test_remove_added_files(self):
59
75
"""Removal of newly added files must fail."""
76
tree = self.get_tree(TestRemove.files)
61
77
tree.add(TestRemove.files)
62
78
self.assertInWorkingTree(TestRemove.files)
63
79
err = self.assertRaises(errors.BzrRemoveChangedFilesError, tree.remove,
66
82
'(?s)added:.*a.*b/.*b/c.*d/')
67
83
self.assertInWorkingTree(TestRemove.files)
68
84
self.failUnlessExists(TestRemove.files)
70
87
def test_remove_changed_file(self):
71
88
"""Removal of a changed files must fail."""
73
tree.add(TestRemove.a)
74
tree.commit("make sure a is versioned")
89
tree = self.get_committed_tree(['a'])
75
90
self.build_tree_contents([('a', "some other new content!")])
76
self.assertInWorkingTree(TestRemove.a)
91
self.assertInWorkingTree('a')
77
92
err = self.assertRaises(errors.BzrRemoveChangedFilesError, tree.remove,
78
TestRemove.a, keep_files=False)
93
'a', keep_files=False)
79
94
self.assertContainsRe(err.changes_as_text, '(?s)modified:.*a')
80
self.assertInWorkingTree(TestRemove.a)
81
self.failUnlessExists(TestRemove.a)
95
self.assertInWorkingTree('a')
96
self.failUnlessExists('a')
83
99
def test_remove_deleted_files(self):
84
100
"""Check that files are removed if they don't exist any more."""
86
tree.add(TestRemove.files)
87
tree.commit("make sure files are versioned")
101
tree = self.get_committed_tree(TestRemove.files)
88
102
for f in TestRemove.rfiles:
89
103
osutils.delete_any(f)
90
104
self.assertInWorkingTree(TestRemove.files)
91
105
self.failIfExists(TestRemove.files)
93
106
tree.remove(TestRemove.files, keep_files=False)
95
self.assertNotInWorkingTree(TestRemove.files)
96
self.failIfExists(TestRemove.files)
107
self.assertRemovedAndDeleted(TestRemove.files)
98
110
def test_remove_renamed_files(self):
99
111
"""Check that files are removed even if they are renamed."""
100
tree = self.getTree()
101
tree.add(TestRemove.files)
102
tree.commit("make sure files are versioned")
112
tree = self.get_committed_tree(TestRemove.files)
104
114
for f in TestRemove.rfiles:
105
115
tree.rename_one(f,f+'x')
132
139
'(?s)modified:.*ax.*bx/cx')
133
140
self.assertInWorkingTree(rfilesx)
134
141
self.failUnlessExists(rfilesx)
136
144
def test_force_remove_changed_files(self):
137
145
"""Check that changed files are removed and deleted when forced."""
138
tree = self.getTree()
146
tree = self.get_tree(TestRemove.files)
139
147
tree.add(TestRemove.files)
140
148
self.assertInWorkingTree(TestRemove.files)
142
150
tree.remove(TestRemove.files, keep_files=False, force=True)
144
self.assertNotInWorkingTree(TestRemove.files)
145
self.failIfExists(TestRemove.files)
151
self.assertRemovedAndDeleted(TestRemove.files)
147
154
def test_remove_unknown_files(self):
148
155
"""Try to delete unknown files."""
149
tree = self.getTree()
156
tree = self.get_tree(TestRemove.files)
150
157
err = self.assertRaises(errors.BzrRemoveChangedFilesError, tree.remove,
151
158
TestRemove.files, keep_files=False)
152
159
self.assertContainsRe(err.changes_as_text,
153
160
'(?s)unknown:.*d/.*b/c.*b/.*a.*')
155
163
def test_remove_nonexisting_files(self):
156
164
"""Try to delete non-existing files."""
157
tree = self.getTree()
165
tree = self.get_tree(TestRemove.files)
158
166
tree.remove([''], keep_files=False)
159
167
tree.remove(['xyz', 'abc/def'], keep_files=False)
161
def test_remove_nonempty_directory(self):
162
"""Unchanged non-empty directories should be deleted."""
163
tree = self.getTree()
164
tree.add(TestRemove.files)
165
tree.commit("make sure b is versioned")
166
self.assertInWorkingTree(TestRemove.files)
167
self.failUnlessExists(TestRemove.files)
168
tree.remove(TestRemove.b, keep_files=False)
169
self.assertNotInWorkingTree(TestRemove.b)
170
self.failIfExists(TestRemove.b)
172
def test_remove_nonempty_directory_with_unknowns(self):
173
"""Unchanged non-empty directories should be deleted."""
174
tree = self.getTree()
175
tree.add(TestRemove.files)
176
tree.commit("make sure b is versioned")
177
self.assertInWorkingTree(TestRemove.files)
178
self.failUnlessExists(TestRemove.files)
179
self.build_tree(['b/my_unknown_file'])
180
err = self.assertRaises(errors.BzrRemoveChangedFilesError, tree.remove,
181
TestRemove.b, keep_files=False)
182
self.assertContainsRe(err.changes_as_text,
183
'(?s)unknown:.*b/my_unknown_file')
184
self.assertInWorkingTree(TestRemove.b)
185
self.failUnlessExists(TestRemove.b)
187
def test_force_remove_nonempty_directory(self):
188
tree = self.getTree()
189
tree.add(TestRemove.files)
190
tree.commit("make sure b is versioned")
191
self.assertInWorkingTree(TestRemove.files)
192
self.failUnlessExists(TestRemove.files)
193
tree.remove(TestRemove.b, keep_files=False, force=True)
194
self.assertNotInWorkingTree(TestRemove.b_c)
195
self.failIfExists(TestRemove.b_c)
170
def test_remove_unchanged_directory(self):
171
"""Unchanged directories should be deleted."""
172
files = ['b/', 'b/c', 'b/sub_directory/', 'b/sub_directory/with_file']
173
tree = self.get_committed_tree(files)
174
tree.remove('b', keep_files=False)
175
self.assertRemovedAndDeleted('b')
178
def test_remove_absent_directory(self):
179
"""Removing a absent directory succeeds without corruption (#150438)."""
180
paths = ['a/', 'a/b']
181
tree = self.get_committed_tree(paths)
182
self.get_transport('.').delete_tree('a')
184
self.assertRemovedAndDeleted('b')
187
def test_remove_unknown_ignored_files(self):
188
"""Unknown ignored files should be deleted."""
189
tree = self.get_committed_tree(['b/'])
190
ignores.add_runtime_ignores(["*ignored*"])
192
self.build_tree(['unknown_ignored_file'])
193
self.assertNotEquals(None, tree.is_ignored('unknown_ignored_file'))
194
tree.remove('unknown_ignored_file', keep_files=False)
195
self.assertRemovedAndDeleted('unknown_ignored_file')
197
self.build_tree(['b/unknown_ignored_file', 'b/unknown_ignored_dir/'])
198
self.assertNotEquals(None, tree.is_ignored('b/unknown_ignored_file'))
199
self.assertNotEquals(None, tree.is_ignored('b/unknown_ignored_dir'))
200
tree.remove('b', keep_files=False)
201
self.assertRemovedAndDeleted('b')
204
def test_remove_changed_ignored_files(self):
205
"""Changed ignored files should not be deleted."""
206
files = ['an_ignored_file']
207
tree = self.get_tree(files)
209
ignores.add_runtime_ignores(["*ignored*"])
210
self.assertInWorkingTree(files)
211
self.assertNotEquals(None, tree.is_ignored(files[0]))
212
err = self.assertRaises(errors.BzrRemoveChangedFilesError, tree.remove,
213
files, keep_files=False)
214
self.assertContainsRe(err.changes_as_text,
215
'(?s)added:.*' + files[0])
216
self.assertInWorkingTree(files)
219
def test_dont_remove_directory_with_unknowns(self):
220
"""Directories with unknowns should not be deleted."""
221
directories = ['a/', 'b/', 'c/', 'c/c/']
222
tree = self.get_committed_tree(directories)
224
self.build_tree(['a/unknown_file'])
225
err = self.assertRaises(errors.BzrRemoveChangedFilesError, tree.remove,
226
'a', keep_files=False)
227
self.assertContainsRe(err.changes_as_text,
228
'(?s)unknown:.*a/unknown_file')
230
self.build_tree(['b/unknown_directory'])
231
err = self.assertRaises(errors.BzrRemoveChangedFilesError, tree.remove,
232
'b', keep_files=False)
233
self.assertContainsRe(err.changes_as_text,
234
'(?s)unknown:.*b/unknown_directory')
236
self.build_tree(['c/c/unknown_file'])
237
err = self.assertRaises(errors.BzrRemoveChangedFilesError, tree.remove,
238
'c/c', keep_files=False)
239
self.assertContainsRe(err.changes_as_text,
240
'(?s)unknown:.*c/c/unknown_file')
242
self.assertInWorkingTree(directories)
243
self.failUnlessExists(directories)
246
def test_force_remove_directory_with_unknowns(self):
247
"""Unchanged non-empty directories should be deleted when forced."""
248
files = ['b/', 'b/c']
249
tree = self.get_committed_tree(files)
251
other_files = ['b/unknown_file', 'b/sub_directory/',
252
'b/sub_directory/with_file', 'b/sub_directory/sub_directory/']
253
self.build_tree(other_files)
255
self.assertInWorkingTree(files)
256
self.failUnlessExists(files)
258
tree.remove('b', keep_files=False, force=True)
260
self.assertRemovedAndDeleted(files)
261
self.assertRemovedAndDeleted(other_files)
197
264
def test_remove_directory_with_changed_file(self):
198
265
"""Refuse to delete directories with changed files."""
199
tree = self.getTree()
200
tree.add(TestRemove.b_c)
201
tree.commit("make sure b and c are versioned")
266
files = ['b/', 'b/c']
267
tree = self.get_committed_tree(files)
202
268
self.build_tree_contents([('b/c', "some other new content!")])
203
self.assertInWorkingTree(TestRemove.b_c)
204
270
err = self.assertRaises(errors.BzrRemoveChangedFilesError, tree.remove,
205
TestRemove.b, keep_files=False)
271
'b', keep_files=False)
206
272
self.assertContainsRe(err.changes_as_text, '(?s)modified:.*b/c')
207
self.assertInWorkingTree(TestRemove.b_c)
208
self.failUnlessExists(TestRemove.b_c)
210
#see if we can force it now..
211
tree.remove(TestRemove.b, keep_files=False, force=True)
212
self.assertNotInWorkingTree(TestRemove.b_c)
213
self.failIfExists(TestRemove.b_c)
215
def test_remove_subtree(self):
216
tree = self.make_branch_and_tree('.')
217
subtree = self.make_branch_and_tree('subtree')
218
tree.add('subtree', 'subtree-id')
219
tree.remove('subtree')
220
self.assertIs(None, tree.path2id('subtree'))
273
self.assertInWorkingTree(files)
274
self.failUnlessExists(files)
276
# see if we can force it now..
277
tree.remove('b', keep_files=False, force=True)
278
self.assertRemovedAndDeleted(files)
281
def test_remove_directory_with_renames(self):
282
"""Delete directory with renames in or out."""
284
files = ['a/', 'a/file', 'a/directory/', 'b/']
285
files_to_move = ['a/file', 'a/directory/']
287
tree = self.get_committed_tree(files)
288
# move stuff from a=>b
289
tree.move(['a/file', 'a/directory'], to_dir='b')
291
moved_files = ['b/file', 'b/directory/']
292
self.assertRemovedAndDeleted(files_to_move)
293
self.assertInWorkingTree(moved_files)
294
self.failUnlessExists(moved_files)
296
# check if it works with renames out
297
tree.remove('a', keep_files=False)
298
self.assertRemovedAndDeleted(['a/'])
300
# check if it works with renames in
301
tree.remove('b', keep_files=False)
302
self.assertRemovedAndDeleted(['b/'])
222
305
def test_non_cwd(self):
223
306
tree = self.make_branch_and_tree('tree')