purge.py
127 lines
| 4.5 KiB
| text/x-python
|
PythonLexer
/ hgext / purge.py
Emanuele Aina
|
r4311 | # Copyright (C) 2006 - Marco Barisione <marco@barisione.org> | ||
# | ||||
Matt Mackall
|
r26421 | # This is a small extension for Mercurial (https://mercurial-scm.org/) | ||
Emanuele Aina
|
r4311 | # that removes files not known to mercurial | ||
# | ||||
Martin Geisler
|
r9270 | # This program was inspired by the "cvspurge" script contained in CVS | ||
# utilities (http://www.red-bean.com/cvsutils/). | ||||
Emanuele Aina
|
r4311 | # | ||
# For help on the usage of "hg purge" use: | ||||
# hg help purge | ||||
# | ||||
# This program is free software; you can redistribute it and/or modify | ||||
# it under the terms of the GNU General Public License as published by | ||||
# the Free Software Foundation; either version 2 of the License, or | ||||
# (at your option) any later version. | ||||
# | ||||
# This program is distributed in the hope that it will be useful, | ||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||||
# GNU General Public License for more details. | ||||
# | ||||
# You should have received a copy of the GNU General Public License | ||||
Martin Geisler
|
r15782 | # along with this program; if not, see <http://www.gnu.org/licenses/>. | ||
Emanuele Aina
|
r4311 | |||
Dirkjan Ochtman
|
r8934 | '''command to delete untracked files from the working directory''' | ||
timeless
|
r28382 | from __future__ import absolute_import | ||
Dirkjan Ochtman
|
r8873 | |||
timeless
|
r28382 | import os | ||
Yuya Nishihara
|
r29205 | from mercurial.i18n import _ | ||
timeless
|
r28382 | from mercurial import ( | ||
cmdutil, | ||||
commands, | ||||
error, | ||||
scmutil, | ||||
util, | ||||
) | ||||
Emanuele Aina
|
r4311 | |||
Adrian Buehlmann
|
r14310 | cmdtable = {} | ||
command = cmdutil.command(cmdtable) | ||||
Augie Fackler
|
r25186 | # Note for extension authors: ONLY specify testedwith = 'internal' for | ||
# extensions which SHIP WITH MERCURIAL. Non-mainline extensions should | ||||
# be specifying the version(s) of Mercurial they are tested with, or | ||||
# leave the attribute unspecified. | ||||
Augie Fackler
|
r16743 | testedwith = 'internal' | ||
Adrian Buehlmann
|
r14310 | |||
@command('purge|clean', | ||||
[('a', 'abort-on-err', None, _('abort if an error occurs')), | ||||
('', 'all', None, _('purge ignored files too')), | ||||
Ben Kehoe
|
r21853 | ('', 'dirs', None, _('purge empty directories')), | ||
('', 'files', None, _('purge files')), | ||||
Adrian Buehlmann
|
r14310 | ('p', 'print', None, _('print filenames instead of deleting them')), | ||
('0', 'print0', None, _('end filenames with NUL, for use with xargs' | ||||
' (implies -p/--print)')), | ||||
] + commands.walkopts, | ||||
_('hg purge [OPTION]... [DIR]...')) | ||||
Matt Mackall
|
r6573 | def purge(ui, repo, *dirs, **opts): | ||
Benjamin Pollack
|
r7605 | '''removes files not tracked by Mercurial | ||
Matt Mackall
|
r6573 | |||
Martin Geisler
|
r9270 | Delete files not known to Mercurial. This is useful to test local | ||
and uncommitted changes in an otherwise-clean source tree. | ||||
Matt Mackall
|
r6573 | |||
Ben Kehoe
|
r21853 | This means that purge will delete the following by default: | ||
Martin Geisler
|
r9215 | |||
Martin Geisler
|
r10973 | - Unknown files: files marked with "?" by :hg:`status` | ||
Martin Geisler
|
r9270 | - Empty directories: in fact Mercurial ignores directories unless | ||
they contain files under source control management | ||||
Martin Geisler
|
r9215 | |||
Matt Mackall
|
r6573 | But it will leave untouched: | ||
Martin Geisler
|
r9215 | |||
- Modified and unmodified tracked files | ||||
- Ignored files (unless --all is specified) | ||||
Martin Geisler
|
r10973 | - New files added to the repository (with :hg:`add`) | ||
Matt Mackall
|
r6573 | |||
Ben Kehoe
|
r21853 | The --files and --dirs options can be used to direct purge to delete | ||
only files, only directories, or both. If neither option is given, | ||||
both will be deleted. | ||||
Matt Mackall
|
r6573 | If directories are given on the command line, only files in these | ||
directories are considered. | ||||
Martin Geisler
|
r9270 | Be careful with purge, as you could irreversibly delete some files | ||
you forgot to add to the repository. If you only want to print the | ||||
list of files that this program would delete, use the --print | ||||
option. | ||||
Matt Mackall
|
r6573 | ''' | ||
Gregory Szorc
|
r29222 | act = not opts.get('print') | ||
Matt Mackall
|
r6757 | eol = '\n' | ||
Gregory Szorc
|
r29222 | if opts.get('print0'): | ||
Matt Mackall
|
r6757 | eol = '\0' | ||
act = False # --print0 implies --print | ||||
Gregory Szorc
|
r29222 | removefiles = opts.get('files') | ||
removedirs = opts.get('dirs') | ||||
Ben Kehoe
|
r21853 | if not removefiles and not removedirs: | ||
removefiles = True | ||||
removedirs = True | ||||
Emanuele Aina
|
r4311 | |||
def remove(remove_func, name): | ||||
if act: | ||||
try: | ||||
Martin Geisler
|
r7570 | remove_func(repo.wjoin(name)) | ||
Benoit Boissinot
|
r7280 | except OSError: | ||
Matt Mackall
|
r6757 | m = _('%s cannot be removed') % name | ||
Gregory Szorc
|
r29222 | if opts.get('abort_on_err'): | ||
Pierre-Yves David
|
r26587 | raise error.Abort(m) | ||
Matt Mackall
|
r6757 | ui.warn(_('warning: %s\n') % m) | ||
Emanuele Aina
|
r4311 | else: | ||
ui.write('%s%s' % (name, eol)) | ||||
Matt Mackall
|
r14671 | match = scmutil.match(repo[None], dirs, opts) | ||
Siddharth Agarwal
|
r22265 | if removedirs: | ||
directories = [] | ||||
match.explicitdir = match.traversedir = directories.append | ||||
Gregory Szorc
|
r29222 | status = repo.status(match=match, ignored=opts.get('all'), unknown=True) | ||
Emanuele Aina
|
r4311 | |||
Ben Kehoe
|
r21853 | if removefiles: | ||
Martin von Zweigbergk
|
r22920 | for f in sorted(status.unknown + status.ignored): | ||
Ben Kehoe
|
r21853 | if act: | ||
ui.note(_('removing file %s\n') % f) | ||||
Christian Ebert
|
r21983 | remove(util.unlink, f) | ||
Emanuele Aina
|
r4311 | |||
Ben Kehoe
|
r21853 | if removedirs: | ||
for f in sorted(directories, reverse=True): | ||||
if match(f) and not os.listdir(repo.wjoin(f)): | ||||
if act: | ||||
ui.note(_('removing directory %s\n') % f) | ||||
remove(os.rmdir, f) | ||||