use strict;
use Cwd;
use File::Path;
use Getopt::Long;
use Pod::Usage;
my $use_svn_module = eval { require SVN::Client };
my $CWD = getcwd;
my $force = 0;
my $quiet = 0;
my $print = 0;
my $help = 0;
my $man = 0;
my $nonrecursive = 0;
my $path = $CWD;
GetOptions(
"force" => \$force,
"non-recursive|N" => \$nonrecursive,
"quiet" => \$quiet,
"print" => \$print,
"help|?" => \$help,
"man" => \$man
)
or pod2usage(2);
pod2usage(1) if $help;
pod2usage( -exitstatus => 0, -verbose => 2 ) if $man;
$path = Cwd::abs_path( $ARGV[0] ) if @ARGV;
if ( $use_svn_module ) {
my $ctx = new SVN::Client;
$ctx->status( $path, undef, \&clean, !$nonrecursive, 1, 0, 1 );
}
else {
warn "Warning: Not using SVN Perl modules, this might be slow.\n" unless
$quiet;
my @command = qw(svn status --no-ignore -v);
if ($nonrecursive) {
push @command, '-N';
}
open SVN, "-|", @command, @ARGV
or die "Can't call program \"svn\": $!\n";
while (<SVN>) {
if (/^([\?ID])/) {
my $file = (split)[-1];
if ( $1 eq 'D' ) {
next unless -f $file;
}
else {
next unless -e $file;
}
if ($print) {
print "$file\n";
}
else {
rmtree( $file, !$quiet, !$force );
}
}
}
}
sub clean {
my ( $path, $status ) = @_;
if ( $path =~ s/^\Q$CWD\E\/?//o ) {
return if $path =~ /^\//;
}
if ( $status->text_status == $SVN::Wc::Status::unversioned
or $status->text_status == $SVN::Wc::Status::ignored
or $status->text_status == $SVN::Wc::Status::deleted )
{
stat $path;
if (
-e _
and ( not -d _
or $status->text_status != $SVN::Wc::Status::deleted )
)
{
if ($print) {
print "$path\n";
}
else {
rmtree( $path, !$quiet, !$force );
}
}
}
}
__END__
=head1 NAME
svn-clean - Wipes out unversioned files from Subversion working copy
=head1 SYNOPSIS
svn-clean [options] [directory or file ...]
=head1 DESCRIPTION
B<svn-clean> will scan the given files and directories recursively and find
unversioned files and directories (files and directories that are not present in
the Subversion repository). After the scan is done, these files and directories
will be deleted.
If no file or directory is given, B<svn-clean> defaults to the current directory
(".").
B<svn-clean> uses the SVN Perl modules if they are available. This is much
faster than parsing the output of the B<svn> command-line client.
=head1 OPTIONS
=over 8
=item B<-f>, B<--force>
Files to which you do not have delete access (if running under VMS) or write
access (if running under another OS) will not be deleted unless you use this
option.
=item B<-N>, B<--non-recursive>
Do not search recursively for unversioned files and directories. Unversioned
directories will still be deleted along with all their contents.
=item B<-q>, B<--quiet>
Do not print progress info. In particular, do not print a message each time a
file is examined, giving the name of the file, and indicating whether "rmdir" or
"unlink" is used to remove it, or that it’s skipped.
=item B<-p>, B<--print>
Do not delete anything. Instead, print the name of every file and directory that
would have been deleted.
=item B<-?>, B<-h>, B<--help>
Prints a brief help message and exits.
=item B<--man>
Prints the manual page and exits.
=back
=head1 AUTHOR
Simon Perreault <nomis80@nomis80.org>
=cut