use Cwd qw/cwd/;
use File::Spec::Functions qw/catfile path/;
-use App::Rgit::Utils qw/:codes/;
+use App::Rgit::Utils qw/:codes :levels/;
use App::Rgit;
our $VERSION;
BEGIN {
- $VERSION = '0.04';
+ $VERSION = '0.06';
}
my %opts;
BEGIN {
@ARGV = grep {
defined $cmd ? $_
- : ( /^-([IKV]+)$/ ? do { $opts{$_} = 1 for split //, $1; () }
- : do { $cmd = $_ unless /^-/; $_ } )
+ : ( /^-([DIKV]+)$/ ? do { $opts{$_} = 1 for split //, $1; () }
+ : do { $cmd = $_ unless /^-/; $_ } )
} @ARGV;
- warn "rgit $VERSION\n" if $opts{V};
$cmd = ' ' unless defined $cmd;
}
+my $shell;
+
BEGIN {
if (-t && $opts{I}) {
if (eval "require Term::ReadKey; 1") {
Term::ReadKey->import;
- *policy = sub {
- my ($cmd, $conf, $repo, $status, $signal) = @_;
- return NEXT unless $status;
- print STDERR 'The command failed for ' . $repo->work . "\n";
- print STDERR "[a]bort, [i]gnore, [I]gnore all, [r]etry ?";
- ReadMode(4);
- my $key = ReadKey(0);
- ReadMode(1);
- print STDERR "\n";
- my %codes = (
- 'a' => LAST,
- 'i' => NEXT,
- 'I' => NEXT | SAVE,
- 'r' => REDO,
- );
- $key = 'a' unless defined $key;
- my $code = $codes{$key};
- $code = $codes{a} unless defined $code;
- return $code;
- };
+ *policy = \&policy_interactive;
+ for (grep defined, $ENV{SHELL}, '/bin/sh') {
+ if (-x $_) {
+ $shell = $_;
+ last;
+ }
+ }
} else {
warn "You have to install Term::ReadKey to use the interactive mode.\n";
}
}
- *policy = sub { NEXT } if $opts{K} and not defined *policy{CODE};
- *policy = sub {
- my ($cmd, $conf, $repo, $status, $signal) = @_;
- return NEXT unless $status;
- return LAST;
- } unless defined *policy{CODE};
+ *policy = $opts{K} ? \&policy_keep
+ : \&policy_default
+ unless defined *policy{CODE};
}
setpgrp 0, 0 if $Config{d_setpgrp};
my $root = $ENV{GIT_DIR};
$root = cwd unless defined $root;
-exit App::Rgit->new(
+my $ar = App::Rgit->new(
git => $git,
root => $root,
cmd => $cmd,
args => \@ARGV,
policy => \&policy,
-)->run;
+ debug => $opts{D} ? INFO : WARN,
+);
+
+print STDOUT "rgit $VERSION\n" if $opts{V};
+
+exit $ar->run;
+
+sub policy_default {
+ my ($cmd, $conf, $repo, $status, $signal) = @_;
+ return NEXT unless $status;
+ return LAST;
+}
+
+sub policy_keep { NEXT }
+
+sub policy_interactive {
+ my ($cmd, $conf, $repo, $status, $signal) = @_;
+ return NEXT unless $status;
+ my %codes = (
+ 'a' => [ LAST, 'aborting' ],
+ 'i' => [ NEXT, 'ignoring' ],
+ 'I' => [ NEXT | SAVE, 'ignoring all' ],
+ 'r' => [ REDO, 'retrying' ],
+ );
+ my $int = { GetControlChars() }->{INTERRUPT};
+ while (1) {
+ $conf->warn("[a]bort, [i]gnore, [I]gnore all, [r]etry, open [s]hell ?");
+ ReadMode(4);
+ my $key = ReadKey(0);
+ ReadMode(1);
+ print STDERR "\n";
+ next unless defined $key;
+ if ($key eq $int) {
+ $conf->warn("Interrupted, aborting\n");
+ return LAST;
+ } elsif ($key eq 's') {
+ if (defined $shell) {
+ $conf->info('Opening shell in ', $repo->work, "\n");
+ my $cwd = cwd;
+ $repo->chdir;
+ system { $shell } $shell;
+ chdir $cwd;
+ } else {
+ $conf->err("Couldn't find any shell\n");
+ }
+ } elsif (exists $codes{$key}) {
+ my $code = $codes{$key};
+ $conf->info('Okay, ', $code->[1], "\n");
+ return $code->[0];
+ }
+ }
+}
__END__
=head1 VERSION
-Version 0.04
+Version 0.06
=head1 SYNOPSIS
- rgit [-K|-V|-I] [GIT_OPTIONS] COMMAND [COMMAND_ARGS]
+ rgit [-K|-I|-D|-V] [GIT_OPTIONS] COMMAND [COMMAND_ARGS]
=head1 DESCRIPTION
-This utility recursively searches in the current directory (or in the directory given by the C<GIT_DIR> environment variable if it's set) for all git repositories, sort this list by the repository path, C<chdir> into each of them, and executes the specified git command.
-Moreover, those formats are substuted in the arguments before running the command :
+This utility recursively searches in a root directory (which may be the current working directory or - if it has been set - the directory given by the C<GIT_DIR> environment variable) for all git repositories, sort this list by the repository path, C<chdir> into each of them, and executes the specified git command.
+Moreover, those formats are substituted in the arguments before running the command :
=over 4
=item *
-C<^n> with the current repository name.
+C<%n> with the current repository name.
=item *
-C<^g> with the relative path to the current repository.
+C<%g> with the relative path (based from the root directory) to the current repository.
=item *
-C<^G> with the absolute path to the current repository.
+C<%G> with the absolute path to the current repository.
=item *
-C<^w> with the relative path to the current repository's working directory.
+C<%w> with the relative path (based from the root directory) to the current repository's working directory.
=item *
-C<^W> with the absolute path to the current repository's working directory.
+C<%W> with the absolute path to the current repository's working directory.
=item *
-C<^b> with a "bareified" relative path, i.e. C<^g> if this is a bare repository, and C<^w.git> otherwise.
+C<%b> with a "bareified" relative path, i.e. C<%g> if this is a bare repository, and C<%w.git> otherwise.
=item *
-C<^B> with an absolute version of the "bareified" path.
+C<%B> with an absolute version of the "bareified" path.
=item *
-C<^R> with the absolute path to the current root directory.
+C<%R> with the absolute path to the root directory.
=item *
-C<^^> with a bare C<^>.
+C<%%> with a bare C<%>.
=back
-There are actually a few commands that are only executed once in the current directory : C<daemon>, C<gui>, C<help>, C<init> and C<version>.
+There are actually a few commands that are only executed once in the root directory : C<daemon>, C<gui>, C<help>, C<init> and C<version>.
For any of those, no format substitution is done.
You can specify which C<git> executable to use with the C<GIT_EXEC_PATH> environment variable.
=item *
+C<-D>
+
+Outputs diagnostics.
+
+=item *
+
C<-V>
Outputs the version.
Tag all the repositories with their name :
- rgit tag ^n
+ rgit tag %n
Add a remote to all repositories in "/foo/bar" to their bare counterpart in C<qux> on F<host> :
- GIT_DIR="/foo/bar" rgit remote add host git://host/qux/^b
+ GIT_DIR="/foo/bar" rgit remote add host git://host/qux/%b
=head1 DEPENDENCIES
-The core modules L<Carp>, C<Config>, L<Cwd>, L<Exporter>, L<File::Find>, L<File::Spec::Functions> and L<POSIX>.
+The core modules L<Carp>, L<Config>, L<Cwd>, L<Exporter>, L<File::Find>, L<File::Spec::Functions> and L<POSIX>.
L<Object::Tiny>.
=head1 COPYRIGHT & LICENSE
-Copyright 2008 Vincent Pit, all rights reserved.
+Copyright 2008-2009 Vincent Pit, all rights reserved.
This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself.