]> git.vpit.fr Git - perl/modules/rgit.git/blobdiff - bin/rgit
@ wasn't such a good idea for the escape character. Let's use % instead
[perl/modules/rgit.git] / bin / rgit
index 101d63470156611295b448ee7562e4272b966304..2b86e4df02c11734ff081ef0b0a3b9faf75b10d7 100755 (executable)
--- a/bin/rgit
+++ b/bin/rgit
@@ -4,16 +4,53 @@ use strict;
 use warnings;
 
 use Carp qw/croak/;
+use Config qw/%Config/;
 use Cwd qw/cwd/;
 use File::Spec::Functions qw/catfile path/;
-use List::Util qw/first/;
 
+use App::Rgit::Utils qw/:codes :levels/;
 use App::Rgit;
 
-our $VERSION = '0.02';
+our $VERSION;
+BEGIN {
+ $VERSION = '0.05';
+}
+
+my %opts;
+my $cmd;
+
+BEGIN {
+ @ARGV = grep {
+  defined $cmd ? $_
+               : ( /^-([DIKV]+)$/ ? do { $opts{$_} = 1 for split //, $1; () }
+                                  : do { $cmd = $_ unless /^-/; $_ } )
+ } @ARGV;
+ $cmd = ' ' unless defined $cmd;
+}
+
+my $shell;
+
+BEGIN {
+ if (-t && $opts{I}) {
+  if (eval "require Term::ReadKey; 1") {
+   Term::ReadKey->import;
+   *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 = $opts{K} ? \&policy_keep
+                    : \&policy_default
+           unless defined *policy{CODE};
+}
 
-my $cmd = first { !/^-/ } @ARGV;
-$cmd = ' ' unless defined $cmd;
+setpgrp 0, 0 if $Config{d_setpgrp};
 
 my $git = $ENV{GIT_EXEC_PATH};
 unless (defined $git) {
@@ -30,12 +67,64 @@ croak "Couldn't find any valid git executable" unless defined $git;
 my $root = $ENV{GIT_DIR};
 $root = cwd unless defined $root;
 
-exit App::Rgit->new(
- git  => $git,
- root => $root,
- cmd  => $cmd,
- args => \@ARGV
-)->run;
+my $ar = App::Rgit->new(
+ git    => $git,
+ root   => $root,
+ cmd    => $cmd,
+ args   => \@ARGV,
+ policy => \&policy,
+ 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__
 
@@ -45,62 +134,99 @@ rgit - Recursively execute a command on all the git repositories in a directory
 
 =head1 VERSION
 
-Version 0.02
+Version 0.05
 
 =head1 SYNOPSIS
 
-    rgit [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<version>, C<help>, C<daemon> and C<init>.
+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.
 
+=head1 COMMAND LINE SWITCHES
+
+C<rgit> takes its options as the capital switches that comes before the git command.
+It's possible to bundle them together.
+They are removed from the argument list before calling C<git>.
+
+=over 4
+
+=item *
+
+C<-K>
+
+Keep processing on error.
+The default policy is to stop whenever an error occured.
+
+=item *
+
+C<-I>
+
+Enables interactive mode when the standard input is a tty.
+Requires L<Term::ReadKey> to be installed.
+This lets you choose interactively what to do when one of the commands returns a non-zero status.
+
+=item *
+
+C<-D>
+
+Outputs diagnostics.
+
+=item *
+
+C<-V>
+
+Outputs the version.
+
+=back
+
 =head1 EXAMPLES
 
 Execute C<git gc> on all the repositories below the current directory :
@@ -109,15 +235,15 @@ Execute C<git gc> on all the repositories below the current directory :
 
 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>, L<Cwd>, L<Exporter>, L<File::Find>, L<File::Spec::Functions> and L<List::Util>.
+The core modules L<Carp>, L<Config>, L<Cwd>, L<Exporter>, L<File::Find>, L<File::Spec::Functions> and L<POSIX>.
 
 L<Object::Tiny>.