| [497] | 1 | #!/usr/bin/env perl |
|---|
| [446] | 2 | |
|---|
| 3 | use File::Spec::Functions qw(:ALL); |
|---|
| 4 | use File::Copy; |
|---|
| 5 | use File::Path; |
|---|
| 6 | use Data::Dumper; |
|---|
| 7 | use Getopt::Long; |
|---|
| [497] | 8 | use Archive::Tar; |
|---|
| [446] | 9 | use Cwd; |
|---|
| 10 | use strict; |
|---|
| 11 | use warnings; |
|---|
| 12 | |
|---|
| 13 | my $scriptsdev = ""; |
|---|
| [692] | 14 | my $force = 0; |
|---|
| [446] | 15 | |
|---|
| 16 | GetOptions( |
|---|
| 17 | #"redo-delete" => \$redodelete, |
|---|
| 18 | # "redo-add" => \$redoadd, |
|---|
| 19 | # "redo-replace" => \$redoreplace, |
|---|
| 20 | # "redo-diff" => \$redodiff, |
|---|
| 21 | # "redo-all" => sub {$redodelete = $redoadd = $redoreplace = $redodiff = 1;}, |
|---|
| 22 | "dev" => sub {$scriptsdev = "dev";}, |
|---|
| [692] | 23 | "force" => \$force, |
|---|
| [446] | 24 | ); |
|---|
| 25 | |
|---|
| 26 | if (@ARGV < 3) { |
|---|
| 27 | print STDERR "Usage: $0 package oldversion newversion\n"; |
|---|
| 28 | exit(1); |
|---|
| 29 | } |
|---|
| 30 | |
|---|
| 31 | my ($package, $oldversion, $newversion) = @ARGV; |
|---|
| 32 | my ($old, $new, $updatename) = ($package.'-'.$oldversion, $package.'-'.$newversion, $package.'-'.$oldversion.'-to-'.$newversion); |
|---|
| 33 | |
|---|
| 34 | my $proposaldir = $updatename.".proposal"; |
|---|
| 35 | my $outdir = catdir("/mit/scripts/deploy$scriptsdev/updates/", $updatename); |
|---|
| 36 | |
|---|
| 37 | (-d $proposaldir) or die "Can't find $proposaldir, did you run propose-update?\n"; |
|---|
| [692] | 38 | if (-e $outdir) { |
|---|
| 39 | if ($force) { |
|---|
| 40 | print STDERR "Clobbering $outdir.\n"; |
|---|
| 41 | `rm -rf '$outdir'`; |
|---|
| 42 | } else { |
|---|
| 43 | die "Output directory $outdir exists; try --force.\n"; |
|---|
| 44 | } |
|---|
| 45 | } |
|---|
| [497] | 46 | ((! -e $outdir) && mkdir($outdir)) or die "mkdir($outdir) failed: $!"; |
|---|
| [446] | 47 | |
|---|
| 48 | my $olddir = catdir($proposaldir,$old); |
|---|
| 49 | my $newdir = catdir($proposaldir,$new); |
|---|
| 50 | (-d $olddir && -d $newdir) or die "Packages not unpacked?\n"; |
|---|
| 51 | |
|---|
| 52 | sub readFileList($) { |
|---|
| 53 | my ($list) = @_; |
|---|
| 54 | open(LIST, $list) or die "Can't read $list: $!\n"; |
|---|
| 55 | my @files = map { chomp; s|$newdir\/?||g; [split(' ', $_, 2)] } <LIST>; |
|---|
| 56 | close(LIST); |
|---|
| 57 | return @files; |
|---|
| 58 | } |
|---|
| 59 | |
|---|
| 60 | my @todelete = readFileList(catfile($proposaldir, "files.delete")); |
|---|
| 61 | my @toadd = readFileList(catfile($proposaldir, "files.add")); |
|---|
| 62 | my @toreplace = readFileList(catfile($proposaldir, "files.replace")); |
|---|
| 63 | |
|---|
| 64 | open(CHECKMD5, ">", catfile($outdir, "check.md5")) or die "Can't open check.md5: $!"; |
|---|
| [447] | 65 | print CHECKMD5 map { join(" ", @{$_})."\n" } @todelete, @toreplace; |
|---|
| [446] | 66 | close(CHECKMD5); |
|---|
| 67 | |
|---|
| 68 | open(ABSENT, ">", catfile($outdir, "oldfiles.absent")) or die "Can't open oldfiles.absent: $!"; |
|---|
| 69 | print ABSENT map { $_->[1]."\n" } @toadd; |
|---|
| 70 | close(ABSENT); |
|---|
| 71 | |
|---|
| 72 | open(DELETE, ">", catfile($outdir, "files.delete")) or die "Can't open files.delete: $!"; |
|---|
| 73 | print DELETE map { $_->[1]."\n" } @todelete, @toreplace; |
|---|
| 74 | close(DELETE); |
|---|
| 75 | |
|---|
| [497] | 76 | my $outfiletarball = catfile($outdir, "newfiles.tar.gz"); |
|---|
| 77 | my $cwd = getcwd(); |
|---|
| [446] | 78 | |
|---|
| [497] | 79 | chdir($newdir); |
|---|
| 80 | my $tar = Archive::Tar->new; |
|---|
| 81 | $tar->add_files(map { $_->[1] } @toadd, @toreplace); |
|---|
| 82 | $tar->write($outfiletarball, 1); |
|---|
| 83 | chdir($cwd) or die "Couldn't chdir back to $cwd: $!\n"; |
|---|
| 84 | |
|---|
| 85 | if (0) { |
|---|
| 86 | my $outfiledir = catdir($outdir, "newfiles"); |
|---|
| 87 | (-d $outfiledir || mkdir($outfiledir)) or die "Can't mkdir($outfiledir)\n"; |
|---|
| 88 | |
|---|
| 89 | foreach my $file (@toadd, @toreplace) { |
|---|
| 90 | my $filename = $file->[1]; |
|---|
| 91 | my $src = catfile($newdir, $filename); |
|---|
| 92 | my $dest = catfile($outfiledir, $filename); |
|---|
| 93 | my (undef, $dir, undef) = splitpath($dest); |
|---|
| 94 | mkpath($dir); |
|---|
| 95 | copy($src, $dest) or die "Couldn't copy $filename from $src to $dest: $!"; |
|---|
| 96 | } |
|---|
| [446] | 97 | } |
|---|
| 98 | |
|---|
| 99 | copy(catfile($proposaldir, "update.diff"), catfile($outdir, "update.diff")) or die "Couldn't copy update.diff: $!"; |
|---|
| 100 | |
|---|
| [694] | 101 | if (-d catfile($proposaldir, "extra")) { |
|---|
| 102 | system('/bin/cp', '-pr', catfile($proposaldir, "extra"), catfile($outdir, "extra")) |
|---|
| 103 | and die "Couldn't copy extra: $!"; |
|---|
| 104 | } |
|---|
| 105 | |
|---|
| [446] | 106 | open (SCRIPT, ">", catfile($outdir, "update")) or die "Couldn't write update: $!"; |
|---|
| [710] | 107 | printf SCRIPT <<'EOF', catfile("/afs/athena.mit.edu/contrib/scripts/deploy/", $old), catfile("/afs/athena.mit.edu/contrib/scripts/deploy/", $new); |
|---|
| [446] | 108 | #!/bin/bash |
|---|
| 109 | set -e -o noclobber |
|---|
| 110 | |
|---|
| [692] | 111 | die () { echo "$1" >&2; rm .scripts-security-upd-lock; exit 1; } |
|---|
| [446] | 112 | |
|---|
| 113 | patchdir=$(dirname "$0") |
|---|
| 114 | |
|---|
| 115 | >.scripts-security-upd-lock || die "error: Unable to obtain update lock." |
|---|
| 116 | |
|---|
| [695] | 117 | packages=$(tail -1 .scripts-version) |
|---|
| [497] | 118 | |
|---|
| [446] | 119 | echo "[$PWD] begin dry run" |
|---|
| 120 | |
|---|
| [497] | 121 | found="" |
|---|
| 122 | newpackages="" |
|---|
| 123 | for package in $packages; do |
|---|
| 124 | if [ "$package" = "%s" ]; then |
|---|
| 125 | found="yes" |
|---|
| 126 | newpackages="$newpackages%s " |
|---|
| 127 | else |
|---|
| 128 | newpackages="$newpackages$package " |
|---|
| 129 | fi |
|---|
| 130 | done |
|---|
| 131 | |
|---|
| 132 | [ "$found" = "yes" ] || die "error: Correct version not found." |
|---|
| [446] | 133 | xargs -n1 test ! -e <"$patchdir/oldfiles.absent" || die "error: Conflicting files exist." |
|---|
| [693] | 134 | [ -s "$patchdir/check.md5" ] && (md5sum -c "$patchdir/check.md5" || die "error: MD5 check failed.") |
|---|
| [448] | 135 | patch -stN --dry-run --no-backup-if-mismatch -p2 <"$patchdir/update.diff" || die "error: Patch dry run failed." |
|---|
| [446] | 136 | |
|---|
| 137 | echo "[$PWD] dry run OK, applying update" |
|---|
| 138 | |
|---|
| 139 | mv .scripts-version .scripts-version.old |
|---|
| [447] | 140 | patch -stN --no-backup-if-mismatch -p2 <"$patchdir/update.diff" || : |
|---|
| [446] | 141 | xargs rm -f <"$patchdir/files.delete" |
|---|
| [497] | 142 | tar -xzf "$patchdir/newfiles.tar.gz" |
|---|
| [694] | 143 | [ -e "$patchdir/extra/postpatch.sh" ] && . "$patchdir/extra/postpatch.sh" |
|---|
| [585] | 144 | ( |
|---|
| 145 | cat .scripts-version.old |
|---|
| 146 | echo |
|---|
| 147 | date "+%%F %%T %%z" |
|---|
| 148 | echo "$USER@$(hostname)" |
|---|
| 149 | echo "$patchdir" |
|---|
| 150 | echo "$newpackages" |
|---|
| 151 | ) >.scripts-version |
|---|
| [446] | 152 | rm -f .scripts-version.old |
|---|
| 153 | |
|---|
| 154 | rm -f .scripts-security-upd-lock |
|---|
| 155 | |
|---|
| 156 | echo "[$PWD] done" |
|---|
| 157 | |
|---|
| 158 | exit 0 |
|---|
| 159 | EOF |
|---|
| 160 | chmod 0755, catfile($outdir, "update"); |
|---|