#!/usr/bin/perl # # Copyright 2005, Ryan Anderson <ryan@michonline.com> # Josef Weidendorfer <Josef.Weidendorfer@gmx.de> # # This file is licensed under the GPL v2, or a later version # at the discretion of Linus Torvalds. use warnings; use strict; use Getopt::Std; sub usage() { print <<EOT; $0 [-f] [-n] <source> <destination> $0 [-f] [-n] [-k] <source> ... <destination directory> EOT exit(1); } our ($opt_n, $opt_f, $opt_h, $opt_k, $opt_v); getopts("hnfkv") || usage; usage() if $opt_h; @ARGV >= 1 or usage; my $GIT_DIR = `git rev-parse --git-dir`; exit 1 if $?; # rev-parse would have given "not a git dir" message. chomp($GIT_DIR); my (@srcArgs, @dstArgs, @srcs, @dsts); my ($src, $dst, $base, $dstDir); # remove any trailing slash in arguments for (@ARGV) { s/\/*$//; } my $argCount = scalar @ARGV; if (-d $ARGV[$argCount-1]) { $dstDir = $ARGV[$argCount-1]; @srcArgs = @ARGV[0..$argCount-2]; foreach $src (@srcArgs) { $base = $src; $base =~ s/^.*\///; $dst = "$dstDir/". $base; push @dstArgs, $dst; } } else { if ($argCount < 2) { print "Error: need at least two arguments\n"; exit(1); } if ($argCount > 2) { print "Error: moving to directory '" . $ARGV[$argCount-1] . "' not possible; not existing\n"; exit(1); } @srcArgs = ($ARGV[0]); @dstArgs = ($ARGV[1]); $dstDir = ""; } my $subdir_prefix = `git rev-parse --show-prefix`; chomp($subdir_prefix); # run in git base directory, so that git-ls-files lists all revisioned files chdir "$GIT_DIR/.."; # normalize paths, needed to compare against versioned files and update-index # also, this is nicer to end-users by doing ".//a/./b/.//./c" ==> "a/b/c" for (@srcArgs, @dstArgs) { # prepend git prefix as we run from base directory $_ = $subdir_prefix.$_; s|^\./||; s|/\./|/| while (m|/\./|); s|//+|/|g; # Also "a/b/../c" ==> "a/c" 1 while (s,(^|/)[^/]+/\.\./,$1,); } my (@allfiles,@srcfiles,@dstfiles); my $safesrc; my (%overwritten, %srcForDst); $/ = "\0"; open(F, 'git-ls-files -z |') or die "Failed to open pipe from git-ls-files: " . $!; @allfiles = map { chomp; $_; } <F>; close(F); my ($i, $bad); while(scalar @srcArgs > 0) { $src = shift @srcArgs; $dst = shift @dstArgs; $bad = ""; for ($src, $dst) { # Be nicer to end-users by doing ".//a/./b/.//./c" ==> "a/b/c" s|^\./||; s|/\./|/| while (m|/\./|); s|//+|/|g; # Also "a/b/../c" ==> "a/c" 1 while (s,(^|/)[^/]+/\.\./,$1,); } if ($opt_v) { print "Checking rename of '$src' to '$dst'\n"; } unless (-f $src || -l $src || -d $src) { $bad = "bad source '$src'"; } $safesrc = quotemeta($src); @srcfiles = grep /^$safesrc(\/|$)/, @allfiles; $overwritten{$dst} = 0; if (($bad eq "") && -e $dst) { $bad = "destination '$dst' already exists"; if ($opt_f) { # only files can overwrite each other: check both source and destination if (-f $dst && (scalar @srcfiles == 1)) { print "Warning: $bad; will overwrite!\n"; $bad = ""; $overwritten{$dst} = 1; } else { $bad = "Can not overwrite '$src' with '$dst'"; } } } if (($bad eq "") && ($dst =~ /^$safesrc\//)) { $bad = "can not move directory '$src' into itself"; } if ($bad eq "") { if (scalar @srcfiles == 0) { $bad = "'$src' not under version control"; } } if ($bad eq "") { if (defined $srcForDst{$dst}) { $bad = "can not move '$src' to '$dst'; already target of "; $bad .= "'".$srcForDst{$dst}."'"; } else { $srcForDst{$dst} = $src; } } if ($bad ne "") { if ($opt_k) { print "Warning: $bad; skipping\n"; next; } print "Error: $bad\n"; exit(1); } push @srcs, $src; push @dsts, $dst; } # Final pass: rename/move my (@deletedfiles,@addedfiles,@changedfiles); $bad = ""; while(scalar @srcs > 0) { $src = shift @srcs; $dst = shift @dsts; if ($opt_n || $opt_v) { print "Renaming $src to $dst\n"; } if (!$opt_n) { if (!rename($src,$dst)) { $bad = "renaming '$src' failed: $!"; if ($opt_k) { print "Warning: skipped: $bad\n"; $bad = ""; next; } last; } } $safesrc = quotemeta($src); @srcfiles = grep /^$safesrc(\/|$)/, @allfiles; @dstfiles = @srcfiles; s/^$safesrc(\/|$)/$dst$1/ for @dstfiles; push @deletedfiles, @srcfiles; if (scalar @srcfiles == 1) { # $dst can be a directory with 1 file inside if ($overwritten{$dst} ==1) { push @changedfiles, $dstfiles[0]; } else { push @addedfiles, $dstfiles[0]; } } else { push @addedfiles, @dstfiles; } } if ($opt_n) { if (@changedfiles) { print "Changed : ". join(", ", @changedfiles) ."\n"; } if (@addedfiles) { print "Adding : ". join(", ", @addedfiles) ."\n"; } if (@deletedfiles) { print "Deleting : ". join(", ", @deletedfiles) ."\n"; } } else { if (@changedfiles) { open(H, "| git-update-index -z --stdin") or die "git-update-index failed to update changed files with code $!\n"; foreach my $fileName (@changedfiles) { print H "$fileName\0"; } close(H); } if (@addedfiles) { open(H, "| git-update-index --add -z --stdin") or die "git-update-index failed to add new names with code $!\n"; foreach my $fileName (@addedfiles) { print H "$fileName\0"; } close(H); } if (@deletedfiles) { open(H, "| git-update-index --remove -z --stdin") or die "git-update-index failed to remove old names with code $!\n"; foreach my $fileName (@deletedfiles) { print H "$fileName\0"; } close(H); } } if ($bad ne "") { print "Error: $bad\n"; exit(1); }