mirror of
https://github.com/mozilla/gecko-dev.git
synced 2025-01-16 06:54:00 +00:00
397 lines
12 KiB
Perl
397 lines
12 KiB
Perl
#!/usr/bin/perl
|
|
#
|
|
# ***** BEGIN LICENSE BLOCK *****
|
|
# Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
|
#
|
|
# The contents of this file are subject to the Mozilla Public License Version
|
|
# 1.1 (the "License"); you may not use this file except in compliance with
|
|
# the License. You may obtain a copy of the License at
|
|
# http://www.mozilla.org/MPL/
|
|
#
|
|
# Software distributed under the License is distributed on an "AS IS" basis,
|
|
# WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
|
# for the specific language governing rights and limitations under the
|
|
# License.
|
|
#
|
|
# The Original Code is Patcher 2, a patch generator for the AUS2 system.
|
|
#
|
|
# The Initial Developer of the Original Code is
|
|
# Mozilla Corporation
|
|
#
|
|
# Portions created by the Initial Developer are Copyright (C) 2006
|
|
# the Initial Developer. All Rights Reserved.
|
|
#
|
|
# Contributor(s):
|
|
# Chase Phillips (chase@mozilla.org)
|
|
# J. Paul Reed (preed@mozilla.com)
|
|
#
|
|
# Alternatively, the contents of this file may be used under the terms of
|
|
# either the GNU General Public License Version 2 or later (the "GPL"), or
|
|
# the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
|
# in which case the provisions of the GPL or the LGPL are applicable instead
|
|
# of those above. If you wish to allow use of your version of this file only
|
|
# under the terms of either the GPL or the LGPL, and not to allow others to
|
|
# use your version of this file under the terms of the MPL, indicate your
|
|
# decision by deleting the provisions above and replace them with the notice
|
|
# and other provisions required by the GPL or the LGPL. If you do not delete
|
|
# the provisions above, a recipient may use your version of this file under
|
|
# the terms of any one of the MPL, the GPL or the LGPL.
|
|
#
|
|
# ***** END LICENSE BLOCK *****
|
|
#
|
|
|
|
package MozAUSLib;
|
|
|
|
use Cwd;
|
|
use File::Path;
|
|
use File::Copy qw(move copy);
|
|
use English;
|
|
|
|
require Exporter;
|
|
|
|
@ISA = qw(Exporter);
|
|
@EXPORT_OK = qw(CreatePartialMarFile CreatePartialMarPatchInfo
|
|
GetAUS2PlatformStrings GetBouncerPlatformStrings
|
|
ValidateToolsDirectory
|
|
MkdirWithPath RunShellCommand
|
|
SubstitutePath
|
|
);
|
|
|
|
use strict;
|
|
|
|
##
|
|
## CONSTANTS
|
|
##
|
|
|
|
use vars qw($MAR_BIN $MBSDIFF_BIN $MAKE_BIN
|
|
$INCREMENTAL_UPDATE_BIN $UNWRAP_FULL_UPDATE_BIN
|
|
$TMPDIR_PREFIX
|
|
$ST_SIZE
|
|
%BOUNCER_PLATFORMS %AUS2_PLATFORMS
|
|
$DEFAULT_PARTIAL_MAR_OUTPUT_FILE
|
|
$EXEC_TIMEOUT );
|
|
|
|
$ST_SIZE = 7;
|
|
|
|
$MAR_BIN = 'dist/host/bin/mar';
|
|
$MBSDIFF_BIN = 'dist/host/bin/mbsdiff';
|
|
|
|
$INCREMENTAL_UPDATE_BIN = 'tools/update-packaging/make_incremental_update.sh';
|
|
$UNWRAP_FULL_UPDATE_BIN = 'tools/update-packaging/unwrap_full_update.pl';
|
|
|
|
$MAKE_BIN = '/usr/bin/make';
|
|
$TMPDIR_PREFIX = '/dev/shm/tmp/MozAUSLib';
|
|
|
|
%BOUNCER_PLATFORMS = ( 'win32' => 'win',
|
|
'linux-i686' => 'linux',
|
|
'mac' => 'osx',
|
|
'unimac' => 'osx',
|
|
);
|
|
|
|
%AUS2_PLATFORMS = ( 'macppc' => 'Darwin_ppc-gcc3',
|
|
'mac' => 'Darwin_Universal-gcc3',
|
|
'linux-i686' => 'Linux_x86-gcc3',
|
|
'win32' => 'WINNT_x86-msvc' );
|
|
|
|
$DEFAULT_PARTIAL_MAR_OUTPUT_FILE = 'partial.mar';
|
|
|
|
$EXEC_TIMEOUT = 600;
|
|
|
|
## This is a wrapper function to get easy true/false return values from a
|
|
## mkpath()-like function. mkpath() *actually* returns the list of directories
|
|
## it created in the pursuit of your request, and keeps its actual success
|
|
## status in $@.
|
|
|
|
sub MkdirWithPath
|
|
{
|
|
my $dirToCreate = shift;
|
|
my $printProgress = shift;
|
|
my $dirMask = shift;
|
|
|
|
die "ASSERT: MkdirWithPath() needs an arg\n" if not defined($dirToCreate);
|
|
|
|
## Defaults based on what mkpath does...
|
|
$printProgress = defined($printProgress) ? $printProgress : 0;
|
|
$dirMask = defined($dirMask) ? $dirMask : 0777;
|
|
|
|
eval { mkpath($dirToCreate, $printProgress, $dirMask) };
|
|
return defined($@);
|
|
}
|
|
|
|
sub ValidateToolsDirectory
|
|
{
|
|
my %args = @_;
|
|
my $toolsDir= $args{'toolsDir'};
|
|
|
|
if ($toolsDir !~ /^\//) {
|
|
die "ASSERT: ValidateToolsDirectory() requires a full path: $toolsDir\n";
|
|
}
|
|
|
|
my $binPrefix = "$toolsDir/mozilla";
|
|
return (-d $binPrefix and
|
|
-x "$binPrefix/$MAR_BIN" and
|
|
-x "$binPrefix/$MBSDIFF_BIN" and
|
|
-x "$binPrefix/$INCREMENTAL_UPDATE_BIN" and
|
|
-x "$binPrefix/$UNWRAP_FULL_UPDATE_BIN");
|
|
}
|
|
|
|
sub GetAUS2PlatformStrings
|
|
{
|
|
my %retHash = %AUS2_PLATFORMS;
|
|
return %retHash;
|
|
}
|
|
|
|
sub GetBouncerPlatformStrings
|
|
{
|
|
my %retHash = %BOUNCER_PLATFORMS;
|
|
return %retHash;
|
|
}
|
|
|
|
sub CreatePartialMarFile
|
|
{
|
|
my %args = @_;
|
|
|
|
my $fromCompleteMar = $args{'from'};
|
|
my $toCompleteMar = $args{'to'};
|
|
my $outputDir = $args{'outputDir'} || getcwd();
|
|
my $outputFile = $args{'outputFile'} || $DEFAULT_PARTIAL_MAR_OUTPUT_FILE;
|
|
my $mozdir = $args{'mozdir'};
|
|
my $forceList = $args{'force'};
|
|
|
|
my $startingWd = getcwd();
|
|
|
|
if (not defined($mozdir)) {
|
|
die "ASSERT: CreatePartialMarFile(): mozdir undefined\n";
|
|
}
|
|
|
|
if (not ValidateToolsDirectory(toolsDir => $mozdir)) {
|
|
print STDERR "Invalid Mozilla working dir: $mozdir\n";
|
|
return -1;
|
|
} else {
|
|
# We actually want the CVS directory itself...
|
|
$mozdir .= '/mozilla';
|
|
}
|
|
|
|
my $mar = "$mozdir/$MAR_BIN";
|
|
my $mbsdiff = "$mozdir/$MBSDIFF_BIN";
|
|
my $makeIncrementalUpdate = "$mozdir/$INCREMENTAL_UPDATE_BIN";
|
|
|
|
# Seed PRNG.
|
|
srand (time() ^ $PID ^ unpack "%L*", `ps axww|gzip`);
|
|
my $tmpDir = "$TMPDIR_PREFIX.CreatePartialMarFile/$PID/" .
|
|
int(rand(1000000));
|
|
|
|
my $fromDir = "$tmpDir/from";
|
|
my $toDir = "$tmpDir/to";
|
|
my $partialDir = "$tmpDir/partial";
|
|
|
|
my @createdDirs = ($tmpDir, $fromDir, $toDir, $partialDir);
|
|
|
|
foreach my $dir (@createdDirs) {
|
|
if (not MkdirWithPath($dir)) {
|
|
print STDERR "MkdirWithPath() failed on $dir: $EVAL_ERROR\n";
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
if ( not -r $fromCompleteMar) {
|
|
print STDERR "CreatePartialMarFile: $fromCompleteMardoesn't exist!";
|
|
return -1;
|
|
}
|
|
|
|
if ( not -r $toCompleteMar ) {
|
|
print STDERR "CreatePartialMarFile: $toCompleteMar doesn't exist!";
|
|
return -1;
|
|
}
|
|
|
|
# XXX - Add check here to verify md5/sha1 checksum of file is as-expected.
|
|
|
|
# Extract the source MAR file.
|
|
my $extractCommand = "MAR=$mar $mozdir/$UNWRAP_FULL_UPDATE_BIN $startingWd/$fromCompleteMar";
|
|
printf("Decompressing $fromCompleteMar with $extractCommand... \n");
|
|
chdir($fromDir) or die "chdir() $fromDir failed: $ERRNO";
|
|
|
|
my $rv = RunShellCommand(command => $extractCommand);
|
|
if ($rv->{'exit_value'} != 0) {
|
|
die "FAILED: $extractCommand: $rv->{'exit_value'}, output: $rv->{'output'}\n";
|
|
}
|
|
|
|
printf("done\n");
|
|
|
|
# Extract the destination MAR file.
|
|
$extractCommand = "MAR=$mar $mozdir/$UNWRAP_FULL_UPDATE_BIN $startingWd/$toCompleteMar";
|
|
printf("Decompressing $toCompleteMar with $extractCommand... \n");
|
|
chdir($toDir) or die "chdir() $toDir failed: $ERRNO";;
|
|
|
|
my $rv = RunShellCommand(command => $extractCommand);
|
|
if ($rv->{'exit_value'} != 0) {
|
|
die "FAILED: $extractCommand: $rv->{'exit_value'}, output: $rv->{'output'}\n";
|
|
}
|
|
|
|
printf("done\n");
|
|
|
|
# Build the partial patch.
|
|
chdir($mozdir);
|
|
|
|
my $forceArgument = (defined($forceList) && scalar(@{$forceList}) > 0) ?
|
|
('-f ' . join(' -f ', @{$forceList})) : '';
|
|
|
|
my $outputMar = "$mozdir/$DEFAULT_PARTIAL_MAR_OUTPUT_FILE";
|
|
my $cmd = "MAR=$mar MBSDIFF=$mbsdiff"
|
|
. " $makeIncrementalUpdate $forceArgument "
|
|
. " $outputMar "
|
|
. " $fromDir $toDir";
|
|
|
|
printf("Building partial update with: $cmd...\n");
|
|
my $rv = RunShellCommand(command => $cmd, output => 1);
|
|
if ($rv->{'exit_value'} != 0) {
|
|
die "FAILED: $cmd: $rv->{'exit_value'}, output: $rv->{'output'}\n";
|
|
}
|
|
|
|
printf("done\n");
|
|
|
|
if (-f $outputMar) {
|
|
printf("Found $outputMar.\n");
|
|
} else {
|
|
print STDERR "Couldn't find partial output mar: $outputMar\n";
|
|
}
|
|
|
|
printf("Moving $outputMar to $outputDir/$outputFile... \n");
|
|
move($outputMar, "$outputDir/$outputFile") or
|
|
die "move($outputMar, $outputDir/$outputFile) failed: $ERRNO";
|
|
printf("done\n");
|
|
|
|
printf("Removing temporary directories...\n");
|
|
foreach my $dir (@createdDirs ) {
|
|
printf("\tRemoving $dir...\n");
|
|
eval { rmtree($dir) };
|
|
if ($EVAL_ERROR) {
|
|
print STDERR "rmtree on $dir failed; $EVAL_ERROR";
|
|
return 1;
|
|
}
|
|
}
|
|
printf("done\n");
|
|
|
|
chdir($startingWd) or die "Couldn't chdir() back to $startingWd: $ERRNO";
|
|
return 1;
|
|
}
|
|
|
|
# Stolen from the code for bug 336463
|
|
|
|
sub RunShellCommand
|
|
{
|
|
my %args = @_;
|
|
my $shellCommand = $args{'command'};
|
|
|
|
# optional
|
|
my $timeout = exists($args{'timeout'}) ? $args{'timeout'} : $EXEC_TIMEOUT;
|
|
my $redirectStderr = exists($args{'redirectStderr'}) ? $args{'redirectStderr'} : 1;
|
|
my $printOutputImmediately = exists($args{'output'}) ? $args{'output'} : 0;
|
|
|
|
my $now = localtime();
|
|
local $_;
|
|
|
|
chomp($shellCommand);
|
|
|
|
my $exit_value = 1;
|
|
my $signal_num;
|
|
my $sig_name;
|
|
my $dumped_core;
|
|
my $timed_out;
|
|
my $output;
|
|
|
|
eval {
|
|
local $SIG{'ALRM'} = sub { die "alarm\n" };
|
|
alarm $timeout;
|
|
|
|
if (! $redirectStderr || $shellCommand =~ "2>&1") {
|
|
open CMD, "$shellCommand |" or die "Could not run command: $!";
|
|
} else {
|
|
open CMD, "$shellCommand 2>&1 |" or die "Could not run command: $!";
|
|
}
|
|
|
|
while (<CMD>) {
|
|
$output .= $_;
|
|
print $_ if ($printOutputImmediately);
|
|
}
|
|
|
|
close CMD or die "Could not close command: $!";
|
|
$exit_value = $? >> 8;
|
|
$signal_num = $? >> 127;
|
|
$dumped_core = $? & 128;
|
|
$timed_out = 0;
|
|
alarm 0;
|
|
};
|
|
|
|
if ($@){
|
|
if ($@ eq "alarm\n") {
|
|
$timed_out = 1;
|
|
} else {
|
|
warn "Error running $shellCommand: $@\n";
|
|
$output = $@;
|
|
}
|
|
}
|
|
|
|
if ($exit_value || $timed_out || $dumped_core || $signal_num) {
|
|
if ($timed_out) {
|
|
# callers expect exit_value to be non-zero if request timed out
|
|
$exit_value = 1;
|
|
}
|
|
}
|
|
|
|
return { timed_out => $timed_out,
|
|
exit_value => $exit_value,
|
|
sig_name => $sig_name,
|
|
output => $output,
|
|
dumped_core => $dumped_core };
|
|
}
|
|
|
|
sub SubstitutePath
|
|
{
|
|
my %args = @_;
|
|
|
|
my $string = $args{'path'};
|
|
my $platform = $args{'platform'};
|
|
my $locale = $args{'locale'};
|
|
my $version = $args{'version'};
|
|
my $app = $args{'app'};
|
|
|
|
my %bouncer_platforms = GetBouncerPlatformStrings();
|
|
my $bouncer_platform = $bouncer_platforms{$platform};
|
|
|
|
$string =~ s/%platform%/$platform/g;
|
|
$string =~ s/%locale%/$locale/g;
|
|
$string =~ s/%bouncer-platform%/$bouncer_platform/g;
|
|
$string =~ s/%version%/$version/g;
|
|
$string =~ s/%app%/$app/g;
|
|
|
|
return $string;
|
|
}
|
|
|
|
sub DownloadFile
|
|
{
|
|
my %args = @_;
|
|
|
|
my $sourceUrl = $args{'url'};
|
|
|
|
die "ASSERT: Invalid Source URL: $sourceUrl\n"
|
|
if ($sourceUrl !~ m|^http://|);
|
|
|
|
my $destArgument = defined($args{'dest'}) ? "-O $args{'dest'}" : '';
|
|
|
|
my $httpUser = defined($args{'user'}) ?
|
|
"--http-user=\"$args{'user'}\"" : '';
|
|
my $httpPassword = defined($args{'password'}) ?
|
|
"--http-password=\"$args{'password'}\"" : '';
|
|
my $httpAuthArgs = "$httpUser $httpPassword";
|
|
|
|
my $wgetCommand = "wget $destArgument $httpAuthArgs --progress=dot:mega " .
|
|
$sourceUrl;
|
|
my $rv = RunShellCommand(command => $wgetCommand);
|
|
if ($rv->{'exit_value'} != 0) {
|
|
die "$wgetCommand FAILED: $rv->{'exit_value'}, output: $rv->{'output'}\n";
|
|
}
|
|
}
|
|
|
|
1;
|