| [363] | 1 | #!/usr/athena/bin/perl | 
|---|
|  | 2 |  | 
|---|
|  | 3 | use File::Spec::Functions; | 
|---|
|  | 4 | use Data::Dumper; | 
|---|
|  | 5 | use Getopt::Long; | 
|---|
|  | 6 | use Cwd; | 
|---|
|  | 7 |  | 
|---|
|  | 8 | my ($redodelete, $redoadd, $redoreplace, $redodiff) = (0,0,0,0); | 
|---|
|  | 9 |  | 
|---|
| [411] | 10 | my $scriptsdev = ""; | 
|---|
|  | 11 |  | 
|---|
| [363] | 12 | GetOptions("redo-delete" => \$redodelete, | 
|---|
| [411] | 13 | "redo-add" => \$redoadd, | 
|---|
|  | 14 | "redo-replace" => \$redoreplace, | 
|---|
|  | 15 | "redo-diff" => \$redodiff, | 
|---|
|  | 16 | "redo-all" => sub {$redodelete = $redoadd = $redoreplace = $redodiff = 1;}, | 
|---|
|  | 17 | "dev" => sub {$scriptsdev = "dev";}, | 
|---|
| [363] | 18 | ); | 
|---|
|  | 19 |  | 
|---|
|  | 20 | if (@ARGV < 3) { | 
|---|
| [747] | 21 | print STDERR "Usage: $0 [--redo-{delete|add|replace|diff|all}] [--dev] package oldversion newversion\n"; | 
|---|
| [363] | 22 | exit(1); | 
|---|
|  | 23 | } | 
|---|
|  | 24 |  | 
|---|
|  | 25 | my ($package, $oldversion, $newversion) = @ARGV; | 
|---|
|  | 26 | my ($old, $new, $updatename) = ($package.'-'.$oldversion, $package.'-'.$newversion, $package.'-'.$oldversion.'-to-'.$newversion); | 
|---|
|  | 27 |  | 
|---|
|  | 28 | my $outdir = $updatename.".proposal"; | 
|---|
|  | 29 |  | 
|---|
|  | 30 | (-d $outdir || mkdir($outdir)) or die "mkdir($outdir) failed: $!"; | 
|---|
|  | 31 |  | 
|---|
|  | 32 | my $olddir = catdir($outdir,$old); | 
|---|
|  | 33 | my $newdir = catdir($outdir,$new); | 
|---|
|  | 34 |  | 
|---|
|  | 35 | unpackPackage($old, $olddir); | 
|---|
| [446] | 36 | unpackPackage($new, $newdir); | 
|---|
| [363] | 37 |  | 
|---|
|  | 38 | sub unpackPackage($$) { | 
|---|
|  | 39 | my ($package, $dir) = @_; | 
|---|
| [446] | 40 | print STDERR "Extracting $package to $dir... "; | 
|---|
| [363] | 41 | if (-d $dir) { | 
|---|
|  | 42 | warn "$dir already exists; assuming unpacking was successful"; | 
|---|
|  | 43 | return; | 
|---|
|  | 44 | } | 
|---|
|  | 45 | mkdir($dir) or die "mkdir($dir) failed: $!"; | 
|---|
|  | 46 | my $cwd = cwd(); | 
|---|
|  | 47 | chdir($dir) or die $!; | 
|---|
| [497] | 48 | `athrun scripts gtar zxf "/mit/scripts/deploy$scriptsdev/$package/$package.tar.gz"`; | 
|---|
|  | 49 | if ($?) { chdir($cwd); system("rmdir", "$dir"); die "Failed to unpack $package.tar.gz: $?"; } | 
|---|
| [363] | 50 | my @files=`athrun scripts gfind . -mindepth 1 -maxdepth 1 | grep -v .admin`; | 
|---|
|  | 51 | if (@files <= 1) { | 
|---|
| [745] | 52 | `athrun scripts gfind . -mindepth 2 -maxdepth 2 | xargs -I{} mv \{} .`; | 
|---|
| [363] | 53 | rmdir($files[0]); | 
|---|
|  | 54 | } | 
|---|
|  | 55 | chdir($cwd) or die "Couldn't return to $cwd"; | 
|---|
|  | 56 | print "done.\n"; | 
|---|
|  | 57 | } | 
|---|
|  | 58 |  | 
|---|
| [745] | 59 | my @oldfiles = sort { $a->[1] cmp $b->[1] } map { chomp; s|$olddir\/?||g; [split(' ', $_, 2)] } `athrun scripts gfind $olddir -type f | xargs -I{} md5sum \{}`; | 
|---|
| [363] | 60 | #print Dumper(\@oldfiles); | 
|---|
| [745] | 61 | my @newfiles = sort { $a->[1] cmp $b->[1] } map { chomp; s|$newdir\/?||g; [split(' ', $_, 2)] } `athrun scripts gfind $newdir -type f | xargs -I{} md5sum \{}`; | 
|---|
| [363] | 62 | #print Dumper(\@newfiles); | 
|---|
|  | 63 |  | 
|---|
|  | 64 | sub compareDirectories($$) { | 
|---|
|  | 65 | my ($alist, $blist) = @_; | 
|---|
|  | 66 | my @a = @$alist; | 
|---|
|  | 67 | my @b = @$blist; | 
|---|
|  | 68 | my @aonly, @bonly, @both; | 
|---|
|  | 69 | $a = $b = 0; | 
|---|
| [497] | 70 | my $debug = 0; | 
|---|
| [363] | 71 | local $Data::Dumper::Indent = 0; | 
|---|
|  | 72 | while ($a <= $#a || $b <= $#a) { | 
|---|
|  | 73 | my $fa = $a[$a]; | 
|---|
|  | 74 | my $fb = $b[$b]; | 
|---|
|  | 75 | print STDERR "Comparing ".Dumper($fa, $fb)."\n" if $debug; | 
|---|
|  | 76 | if ($fa->[1] eq $fb->[1]) { # Same file exists on both | 
|---|
|  | 77 | print STDERR "Same file\n" if $debug; | 
|---|
|  | 78 | if ($fa->[0] ne $fb->[0]) { # File has changed in some way | 
|---|
|  | 79 | print STDERR "Different md5, pushing on \@both\n" if $debug; | 
|---|
|  | 80 | push(@both, [$fa->[1], $fa, $fb]); | 
|---|
|  | 81 | } | 
|---|
|  | 82 | $a++; $b++; # increment both counters | 
|---|
|  | 83 | } else { | 
|---|
|  | 84 | my $a2 = $a; | 
|---|
|  | 85 | while ($a2 <= $#a && $a[$a2]->[1] lt $fb->[1]) { | 
|---|
|  | 86 | $a2++; | 
|---|
|  | 87 | } | 
|---|
|  | 88 | if ($a2 <= $#a && $a[$a2]->[1] eq $fb->[1]) { | 
|---|
|  | 89 | for my $i ($a..$a2-1) { | 
|---|
|  | 90 | push @aonly, $a[$i]; | 
|---|
|  | 91 | } | 
|---|
|  | 92 | $a = $a2; | 
|---|
|  | 93 | } else { | 
|---|
|  | 94 | my $b2 = $b; | 
|---|
|  | 95 | while ($b2 <= $#b && $b[$b2]->[1] lt $fa->[1]) { | 
|---|
|  | 96 | $b2++; | 
|---|
|  | 97 | } | 
|---|
|  | 98 | if ($b2 <= $#b && $b[$b2]->[1] eq $fa->[1]) { | 
|---|
|  | 99 | for my $i ($b..$b2-1) { | 
|---|
|  | 100 | push @bonly, $b[$i]; | 
|---|
|  | 101 | } | 
|---|
|  | 102 | $b = $b2; | 
|---|
|  | 103 | } else { | 
|---|
|  | 104 | push @aonly, $a[$a]; | 
|---|
|  | 105 | push @bonly, $b[$b]; | 
|---|
|  | 106 | $a++; $b++; | 
|---|
|  | 107 | } | 
|---|
|  | 108 | } | 
|---|
|  | 109 | } | 
|---|
|  | 110 | } | 
|---|
|  | 111 | return (\@aonly, \@bonly, \@both); | 
|---|
|  | 112 | } | 
|---|
|  | 113 |  | 
|---|
|  | 114 | my (@todelete, @toadd, @changed); | 
|---|
|  | 115 | my @comp = compareDirectories(\@oldfiles, \@newfiles); | 
|---|
| [446] | 116 | open(DIFF, ">", catfile($outdir, "diff.pl")); | 
|---|
|  | 117 | print DIFF Dumper(@comp); | 
|---|
|  | 118 | close(DIFF); | 
|---|
| [363] | 119 | @todelete = @{$comp[0]}; | 
|---|
|  | 120 | @toadd = @{$comp[1]}; | 
|---|
|  | 121 | @changed = @{$comp[2]}; | 
|---|
|  | 122 |  | 
|---|
|  | 123 | if ($redodelete or ! -e catfile($outdir, "files.delete")) { | 
|---|
|  | 124 | open(TODELETE, ">", catfile($outdir, "files.delete")) or die "Can't open files.delete: $!"; | 
|---|
|  | 125 | foreach my $file (@todelete) { | 
|---|
|  | 126 | printf TODELETE "%s %s\n", $file->[0], $file->[1]; | 
|---|
|  | 127 | } | 
|---|
|  | 128 | close(TODELETE); | 
|---|
| [446] | 129 | printf "Wrote %d filenames to files.delete\n", scalar(@todelete); | 
|---|
|  | 130 | } else { printf "Not overwriting existing files.delete\n"; } | 
|---|
| [363] | 131 |  | 
|---|
|  | 132 | if ($redoadd or ! -e catfile($outdir, "files.add")) { | 
|---|
|  | 133 | open(TOADD, ">", catfile($outdir, "files.add")) or die "Can't open files.add: $!"; | 
|---|
|  | 134 | foreach my $file (@toadd) { | 
|---|
| [446] | 135 | printf TOADD "%s %s\n", $file->[0], $file->[1]; | 
|---|
| [363] | 136 | } | 
|---|
|  | 137 | close(TOADD); | 
|---|
| [446] | 138 | printf "Wrote %d filenames to files.add\n", scalar(@toadd); | 
|---|
|  | 139 | } else { printf "Not overwriting existing files.add\n"; } | 
|---|
| [363] | 140 |  | 
|---|
|  | 141 | my @toreplace; | 
|---|
|  | 142 | my @topatch; | 
|---|
|  | 143 |  | 
|---|
|  | 144 | foreach my $file (@changed) { | 
|---|
|  | 145 | if (-B catdir($newdir, $file->[0])) { | 
|---|
|  | 146 | push (@toreplace, $file); | 
|---|
|  | 147 | } else { | 
|---|
|  | 148 | push (@topatch, $file); | 
|---|
|  | 149 | } | 
|---|
|  | 150 | } | 
|---|
|  | 151 |  | 
|---|
|  | 152 | if ($redoreplace or ! -e catfile($outdir, "files.replace")) { | 
|---|
|  | 153 | open(TOREPLACE, ">", catfile($outdir, "files.replace")) or die "Can't open files.replace: $!"; | 
|---|
|  | 154 | foreach my $file (@toreplace) { | 
|---|
| [446] | 155 | printf TOREPLACE "%s %s\n", $file->[1][0], $file->[0]; | 
|---|
| [363] | 156 | } | 
|---|
|  | 157 | close(TOREPLACE); | 
|---|
| [446] | 158 | printf "Wrote %d filenames to files.replace\n", scalar(@toreplace); | 
|---|
|  | 159 | } else { printf "Not overwriting existing files.replace\n"; } | 
|---|
|  | 160 |  | 
|---|
|  | 161 | if ($redodiff or ! -e catfile($outdir, "update.diff")) { | 
|---|
|  | 162 | open(DIFF, ">", catfile($outdir, "update.diff")) or die "Can't open update.diff: $!"; | 
|---|
|  | 163 | foreach my $file (@topatch) { | 
|---|
|  | 164 | my $filename = $file->[0]; | 
|---|
|  | 165 | my $oldfile = catfile($olddir, $file->[1][1]); | 
|---|
|  | 166 | my $newfile = catfile($newdir, $file->[2][1]); | 
|---|
|  | 167 | my $cmd = "diff -urN $oldfile $newfile"; | 
|---|
|  | 168 | print DIFF "$cmd\n"; | 
|---|
|  | 169 | print DIFF `$cmd`; | 
|---|
|  | 170 | } | 
|---|
|  | 171 | close(DIFF); | 
|---|
|  | 172 | printf "Wrote %d diffs to update.diff\n", scalar(@topatch); | 
|---|
|  | 173 | } else { printf "Not overwriting existing update.patch\n"; } | 
|---|