1
# Copyright (C) 2009, 2010 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
21
from StringIO import StringIO
25
from bzrlib import tests, ui
26
from bzrlib.bzrdir import (
29
from bzrlib.clean_tree import (
33
from bzrlib.osutils import (
36
from bzrlib.tests import (
41
class TestCleanTree(TestCaseInTempDir):
43
def test_symlinks(self):
44
if has_symlinks() is False:
47
BzrDir.create_standalone_workingtree('branch')
48
os.symlink(os.path.realpath('no-die-please'), 'branch/die-please')
49
os.mkdir('no-die-please')
50
self.failUnlessExists('branch/die-please')
51
os.mkdir('no-die-please/child')
53
clean_tree('branch', unknown=True, no_prompt=True)
54
self.failUnlessExists('no-die-please')
55
self.failUnlessExists('no-die-please/child')
57
def test_iter_deletable(self):
58
"""Files are selected for deletion appropriately"""
60
tree = BzrDir.create_standalone_workingtree('branch')
61
transport = tree.bzrdir.root_transport
62
transport.put_bytes('.bzrignore', '*~\n*.pyc\n.bzrignore\n')
63
transport.put_bytes('file.BASE', 'contents')
66
self.assertEqual(len(list(iter_deletables(tree, unknown=True))), 1)
67
transport.put_bytes('file', 'contents')
68
transport.put_bytes('file~', 'contents')
69
transport.put_bytes('file.pyc', 'contents')
70
dels = sorted([r for a,r in iter_deletables(tree, unknown=True)])
71
self.assertEqual(['file', 'file.BASE'], dels)
73
dels = [r for a,r in iter_deletables(tree, detritus=True)]
74
self.assertEqual(sorted(['file~', 'file.BASE']), dels)
76
dels = [r for a,r in iter_deletables(tree, ignored=True)]
77
self.assertEqual(sorted(['file~', 'file.pyc', '.bzrignore']),
80
dels = [r for a,r in iter_deletables(tree, unknown=False)]
81
self.assertEqual([], dels)
85
def test_delete_items_warnings(self):
86
"""Ensure delete_items issues warnings on EACCES. (bug #430785)
88
def _dummy_unlink(path):
89
"""unlink() files other than files named '0foo'.
91
if path.endswith('0foo'):
92
# Simulate 'permission denied' error.
93
# This should show up as a warning for the
96
e.errno = errno.EACCES
99
def _dummy_rmtree(path, ignore_errors=False, onerror=None):
100
"""Call user supplied error handler onerror.
102
self.assertTrue(isinstance(onerror, types.FunctionType))
103
# Indicate failure in removing 'path' if path is subdir0
104
# We later check to ensure that this is indicated
105
# to the user as a warning. We raise OSError to construct
106
# proper excinfo that needs to be passed to onerror
110
e.errno = errno.EACCES
111
excinfo = sys.exc_info()
113
if 'subdir0' not in path:
114
# onerror should show warning only for os.remove
115
# error. For any other failures the error should
116
# be shown to the user.
117
function = os.listdir
118
onerror(function=function,
119
path=path, excinfo=excinfo)
121
self.overrideAttr(os, 'unlink', _dummy_unlink)
122
self.overrideAttr(shutil, 'rmtree', _dummy_rmtree)
123
stdout = tests.StringIOWrapper()
124
stderr = tests.StringIOWrapper()
125
ui.ui_factory = tests.TestUIFactory(stdout=stdout, stderr=stderr)
127
BzrDir.create_standalone_workingtree('.')
128
self.build_tree(['0foo', '1bar', '2baz', 'subdir0/'])
129
clean_tree('.', unknown=True, no_prompt=True)
130
self.assertContainsRe(stderr.getvalue(),
131
'bzr: warning: unable to remove.*0foo')
132
self.assertContainsRe(stderr.getvalue(),
133
'bzr: warning: unable to remove.*subdir0')
135
# Ensure that error other than EACCES during os.remove are
136
# not turned into warnings.
137
self.build_tree(['subdir1/'])
138
self.assertRaises(OSError, clean_tree, '.',
139
unknown=True, no_prompt=True)