All patches and comments are welcome. Please squash your changes to logical
commits before using git-format-patch and git-send-email to
patches@git.madduck.net.
If you'd read over the Git project's submission guidelines and adhered to them,
I'd be especially grateful.
   5 mr - a Multiple Repository management tool
 
   9 B<mr> [options] checkout
 
  11 B<mr> [options] update
 
  13 B<mr> [options] status
 
  15 B<mr> [options] commit [-m "message"]
 
  17 B<mr> [options] action [params ...]
 
  21 B<mr> is a Multiple Repository management tool. It allows you to register a
 
  22 set of repositories in a .mrconfig file, and then checkout, update, or
 
  23 perform other actions on all of the repositories at once.
 
  25 Any mix of revision control systems can be used with B<mr>, and you can
 
  26 define arbitrary actions for commands like "update", "checkout", or "commit".
 
  28 The predefined commands should be fairly familiar to users of any revision
 
  33 =item checkout (or co)
 
  35 Checks out all the registered repositories that are not already checked
 
  40 Updates each registered repository from its configured remote repository.
 
  42 If a repository isn't checked out yet, it will first check it out.
 
  46 Displays a status report for each registered repository, showing what
 
  47 uncommitted changes are present in the repository.
 
  51 Commits changes to each registered repository. (By default, changes
 
  52 are pushed to the remote repository too, when using distributed systems
 
  55 The optional -m parameter allows specifying a commit message.
 
  59 Actions can be abbreviated to any unambiguous subsctring, so
 
  60 "mr st" is equivilant to "mr status".
 
  68 Specifies the topmost directory that B<mr> should work in. The default is
 
  69 the current working directory. B<mr> will operate on all registered
 
  70 repositories at or under the directory.
 
  74 Use the specified mrconfig file, instead of looking for on in your home
 
  85 B<mr> is configured by .mrconfig files. It starts by reading the .mrconfig
 
  86 file in your home directory. Each repository specified in a .mrconfig file
 
  87 can also have its own .mrconfig file in its root directory that can
 
  88 optionally be used as well. So you could have a ~/.mrconfig that registers a
 
  89 repository ~/src, that itself contains a ~/src/.mrconfig file, that in turn
 
  90 registers several additional repositories.
 
  92 The .mrconfig file uses a variant of the INI file format. Lines starting with
 
  93 "#" are comments. Lines ending with "\" are continued on to the next line.
 
  94 Sections specify where each repository is located, relative to the
 
  95 directory that contains the .mrconfig file.
 
  97 Within a section, each parameter defines a shell command to run to handle a
 
  98 given action. Note that these shell commands are run in a "set -e" shell
 
  99 environment, where any additional parameters you pass are available in
 
 100 "$@". B<mr> cds into the repository directory before running
 
 101 a command, except for the "checkout" command, which is run in the parent
 
 102 of the repository directory, since the repository isn't checked out yet.
 
 104 There are three special parameters. If the "skip" parameter is set and
 
 105 its command returns nonzero, then B<mr> will skip acting on that repository.
 
 106 If the "chain" parameter is set and its command returns nonzero, then B<mr>
 
 107 will try to load a .mrconfig file from the root of the repository. (You
 
 108 should avoid chaining from repositories with untrusted committers.) The
 
 109 "lib" parameter can specify some shell code that will be run before each
 
 110 command, this can be a useful way to define shell functions other commands
 
 113 The "default" section allows setting up default handlers for each action,
 
 114 and is overridden by the contents of other sections. mr contains default
 
 115 handlers for the "update", "status", and "commit" actions, so normally
 
 116 you only need to specify what to do for "checkout".
 
 118 The "alias" section allows adding aliases for commands. Each parameter
 
 119 is an alias, and its value is the command to run.
 
 124   checkout = svn co svn://svn.example.com/src/trunk src
 
 129   checkout = git clone git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux-2.6.git
 
 134         case "$(hostname)" in; \
 
 143 Copyright 2007 Joey Hess <joey@kitenet.net>
 
 145 Licensed under the GNU GPL version 2 or higher.
 
 147 http://kitenet.net/~joey/code/mr/
 
 154 use Cwd qw(getcwd abs_path);
 
 156 my $directory=getcwd();
 
 157 my $config="$ENV{HOME}/.mrconfig";
 
 163 Getopt::Long::Configure("no_permute");
 
 164 my $result=GetOptions(
 
 165         "d=s" => sub { $directory=abs_path($_[1]) },
 
 169 if (! $result || @ARGV < 1) {
 
 170         die("Usage: mr [-d directory] action [params ...]\n");
 
 176 #print Dumper(\%config);
 
 178 my $action=shift @ARGV;
 
 179 if (! exists $knownactions{$action}) {
 
 180         if (exists $alias{$action}) {
 
 181                 $action=$alias{$action};
 
 184                 my @matches = grep { /^\Q$action\E/ }
 
 185                         keys %knownactions, keys %alias;
 
 190                         die "mr: ambiguous action \"$action\" (matches @matches)\n";
 
 195 my (@failed, @successful, @skipped);
 
 197 foreach my $topdir (sort keys %config) {
 
 198         foreach my $subdir (sort keys %{$config{$topdir}}) {
 
 199                 next if $subdir eq 'default';
 
 201                 my $dir=$topdir.$subdir;
 
 203                 if (defined $directory &&
 
 204                     $dir ne $directory &&
 
 205                     $dir !~ /^\Q$directory\E\//) {
 
 206                         print "mr $action: $dir skipped per -d parameter ($directory)\n" if $verbose;
 
 211                 print "\n" unless $first;
 
 214                 action($action, $dir, $topdir, $subdir);
 
 220         my ($action, $dir, $topdir, $subdir) = @_;
 
 222         my $lib= exists $config{$topdir}{$subdir}{lib} ?
 
 223                         $config{$topdir}{$subdir}{lib} : "";
 
 225         if ($action eq 'checkout') {
 
 227                         print "mr $action: $dir already exists, skipping checkout\n" if $verbose;
 
 231                 $dir=~s/^(.*)\/[^\/]+\/?$/$1/;
 
 233         elsif ($action eq 'update') {
 
 235                         return action("checkout", $dir, $topdir, $subdir);
 
 240                 print STDERR "mr $action: failed to chdir to $dir: $!\n";
 
 244         if (exists $config{$topdir}{$subdir}{skip}) {
 
 245                 my $ret=system($lib.$config{$topdir}{$subdir}{skip});
 
 246                 if ($ret >> 8 == 0) {
 
 247                         print "mr $action: $dir skipped per config file\n" if $verbose;
 
 253         if (! exists $config{$topdir}{$subdir}{$action}) {
 
 254                 print STDERR "mr $action: no defined $action command for $topdir$subdir, skipping\n";
 
 258                 print "mr $action: in $dir\n";
 
 259                 my $command="set -e; ".$lib.
 
 260                         "my_action(){ $config{$topdir}{$subdir}{$action} ; }; my_action ".
 
 261                         join(" ", map { s/\//\/\//g; s/"/\"/g; '"'.$_.'"' } @ARGV);
 
 262                 my $ret=system($command);
 
 264                         print STDERR "mr $action: failed to run: $command\n" if $verbose;
 
 265                         push @failed, $topdir.$subdir;
 
 266                         if ($ret >> 8 != 0) {
 
 267                                 print STDERR "mr $action: command failed\n";
 
 270                                 print STDERR "mr $action: command died ($ret)\n";
 
 274                         push @successful, $dir;
 
 284                 return "$count ".($count > 1 ? $plural : $singular);
 
 288 print "\nmr $action: finished (".join("; ",
 
 289         showstat($#successful+1, "successful", "successful"),
 
 290         showstat($#failed+1, "failed", "failed"),
 
 291         showstat($#skipped+1, "skipped", "skipped"),
 
 296 elsif (! @successful && @skipped) {
 
 309         if (ref $f eq 'GLOB') {
 
 314                 # $f might be a symlink
 
 315                 my $absf=abs_path($f);
 
 316                 if ($loaded{$absf}) {
 
 321                 print "mr: loading config $f\n" if $verbose;
 
 322                 open($in, "<", $f) || die "mr: open $f: $!\n";
 
 323                 ($dir)=$f=~/^(.*\/)[^\/]+$/;
 
 324                 if (! defined $dir) {
 
 327                 $dir=abs_path($dir)."/";
 
 329                 # copy in defaults from first parent
 
 331                 while ($parent=~s/^(.*)\/[^\/]+\/?$/$1/) {
 
 332                         if (exists $config{$parent} &&
 
 333                             exists $config{$parent}{default}) {
 
 334                                 $config{$dir}{default}={ %{$config{$parent}{default}} };
 
 343                 next if /^\s*\#/ || /^\s*$/;
 
 344                 if (/^\s*\[([^\]]*)\]\s*$/) {
 
 347                 elsif (/^\s*(\w+)\s*=\s*(.*)/) {
 
 352                         while ($value=~/(.*)\\$/) {
 
 357                         if (! defined $section) {
 
 358                                 die "$f line $.: parameter ($parameter) not in section\n";
 
 360                         if ($section ne 'alias' &&
 
 361                             ! exists $config{$dir}{$section} &&
 
 362                             exists $config{$dir}{default}) {
 
 364                                 $config{$dir}{$section}={ %{$config{$dir}{default}} };
 
 366                         if ($section eq 'alias') {
 
 367                                 $alias{$parameter}=$value;
 
 369                         elsif ($parameter eq 'lib') {
 
 370                                 $config{$dir}{$section}{lib}.=$value." ; ";
 
 373                                 $config{$dir}{$section}{$parameter}=$value;
 
 374                                 $knownactions{$parameter}=1;
 
 375                                 if ($parameter eq 'chain' &&
 
 376                                     length $dir && $section ne "default" &&
 
 377                                     -e $dir.$section."/.mrconfig" &&
 
 378                                     system($value) >> 8 == 0) {
 
 379                                         push @toload, $dir.$section."/.mrconfig";
 
 384                         die "$f line $.: parse error\n";
 
 394 # Finally, some useful actions that mr knows about by default.
 
 395 # These can be overridden in ~/.mrconfig.
 
 407         if [ -d .svn ]; then \
 
 409         elif [ -d .git ]; then \
 
 410                 git pull origin master; \
 
 412                 error "unknown repo type"; \
 
 415         if [ -d .svn ]; then \
 
 417         elif [ -d .git ]; then \
 
 418                 git status || true; \
 
 420                 error "unknown repo type"; \
 
 423         if [ -d .svn ]; then \
 
 425         elif [ -d .git ]; then \
 
 426                 git commit -a "$@" && git push --all; \
 
 428                 error "unknown repo type"; \