15
15
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
21
27
from bzrlib.osutils import isdir
22
28
from bzrlib.trace import note
23
29
from bzrlib.workingtree import WorkingTree
30
from bzrlib.i18n import gettext
26
32
def is_detritus(subp):
27
33
"""Return True if the supplied path is detritus, False otherwise"""
52
58
deletables = list(iter_deletables(tree, unknown=unknown,
53
59
ignored=ignored, detritus=detritus))
60
deletables = _filter_out_nested_bzrdirs(deletables)
54
61
if len(deletables) == 0:
55
note('Nothing to delete.')
62
note(gettext('Nothing to delete.'))
58
65
for path, subp in deletables:
60
val = raw_input('Are you sure you wish to delete these [y/N]?')
61
if val.lower() not in ('y', 'yes'):
66
ui.ui_factory.note(subp)
67
prompt = gettext('Are you sure you wish to delete these')
68
if not ui.ui_factory.get_boolean(prompt):
69
ui.ui_factory.note(gettext('Canceled'))
64
71
delete_items(deletables, dry_run=dry_run)
76
def _filter_out_nested_bzrdirs(deletables):
78
for path, subp in deletables:
79
# bzr won't recurse into unknowns/ignored directories by default
80
# so we don't pay a penalty for checking subdirs of path for nested
82
# That said we won't detect the branch in the subdir of non-branch
83
# directory and therefore delete it. (worth to FIXME?)
86
controldir.ControlDir.open(path)
87
except errors.NotBranchError:
88
result.append((path,subp))
90
# TODO may be we need to notify user about skipped directories?
93
result.append((path,subp))
69
97
def delete_items(deletables, dry_run=False):
70
98
"""Delete files in the deletables iterable"""
99
def onerror(function, path, excinfo):
100
"""Show warning for errors seen by rmtree.
102
# Handle only permission error while removing files.
103
# Other errors are re-raised.
104
if function is not os.remove or excinfo[1].errno != errno.EACCES:
106
ui.ui_factory.show_warning(gettext('unable to remove %s') % path)
71
107
has_deleted = False
72
108
for path, subp in deletables:
73
109
if not has_deleted:
74
note("deleting paths:")
110
note(gettext("deleting paths:"))
75
111
has_deleted = True
114
shutil.rmtree(path, onerror=onerror)
120
# We handle only permission error here
121
if e.errno != errno.EACCES:
123
ui.ui_factory.show_warning(gettext(
124
'unable to remove "{0}": {1}.').format(
82
128
if not has_deleted:
83
note("No files deleted.")
129
note(gettext("No files deleted."))