|
|
if (!$__IITPRINTLPM ) { use iit::printl; } if (!$__IITUTILPM ) { use iit::util; } if (!$__IITFILEPM ) { use iit::file; } if (!$__IITSENDHTMLMAILPM ) { use iit::sendhtmlmail; }
package main;
use strict 'subs';
use Carp; #debugging library (carp, carp, etc.) use Env; #allows use of $ENVVAR instead of $ENV{ENVVAR} use win32::console;
$PROC = $PROCESSOR_ARCHITECTURE; # prefer constant PROC (see below)
# CONSTANTS
use constant PROC => $PROCESSOR_ARCHITECTURE;
use constant BC_FAILED => 2; use constant BC_NOTHINGDONE => 4; use constant BC_COPYFAILED => 8; use constant BC_BVTFAILED => 16; use constant BC_CABFAILED => 32; use constant BC_CHKSHIPFAILED => 64;
####################################################################################
# SetLocalGlobalsAndBegin()
# creates a separate enclosed variable scope for your script through use of 'local' variables # any variable declared in this function will be visible in all child functions, but invisible # in parent functions # pass a function name (with syntax *main::<fcnname>) as first argument, and any arguments # to pass to that function as additional arguments # return value is return value of the function name passed
# a-jbilas, 05/10/99 - created
####################################################################################
sub SetLocalGlobalsAndBegin { local($sShortBuildName) = $_[0]; #get filenames from the function name $sShortBuildName =~ s/\*main\:\://;
if ($PROJROOT eq "") { die("Project root MUST be defined"); }
#status local($bcStatus) = BC_NOTHINGDONE;
#numbers local($nMajorVersion) = 3; local($nMinorVersion) = 0; local($nBuildStartYear) = 1999; local($nErrorNumber) = 1; local($nMaxErrLines) = 10; local($nScriptStartTime) = time(); local($nLoggingMode) = 2; # 0 (least) - 2 (most) local($nTotalBuilds) = 0; local($nFailedBuilds) = 0;
#paths local($sLibDir) = $PROJROOT."\\lib\\".PROC; local($sBinExeDir) = $PROJROOT."\\bin\\".PROC; local($sBinBatDir) = $PROJROOT."\\bin"; local($sOldPath) = $PATH; local($sOldInclude) = $INCLUDE; local($sOldLib) = $LIB;
#strings local($sBuildName) = "*Unknown Build*"; local($sLanguage) = "ENGLISH"; local($sBuildNumber) = "0000"; local($sLogDir) = $PROJROOT."\\logs"; local($sRootDropDir) = "\\\\b11nlbuilds\\".$PROJ; local($sTestRootDropDir) = "\\\\nlp\\build\\".$PROJ."\\testdrop"; local($sDropDir) = $sRootDropDir."\\".$sLanguage."\\".$sBuildNumber."\\".PROC; local($sLogDropDir) = $sDropDir."\\logs"; local($sRemoteBuildLog) = $sShortBuildName.PROC.$sBuildNumber.".html"; local($sRemoteTOC) = ""; local($sMailfile) = $sLogDir."\\".$sShortBuildName."msg.html"; local($sBuildLog) = $sLogDir."\\".$sShortBuildName."log.html"; local($sVarsLog) = $sLogDir."\\".$sShortBuildName."vars.log"; local($sTyposLog) = $sLogDir."\\".$sShortBuildName."typos.log"; local($sSyncLog) = $sLogDir."\\".$sShortBuildName."sync.log"; local($sUpdateLog) = $sLogDir."\\".$sShortBuildName."update.log"; local($sDHTMLIncFile) = $sBinBatDir."\\htmlinc.htm"; local($sOfficialBuildAccount) = ""; local($sRegKeyBase) = "Software\\Microsoft\\Intelligent Interface Technologies\\".$PROJ;
if (!defined $strBuildMsg) { $strBuildMsg = ""; #one of our few 'absolute' globals }
#bools (flags) local($bGlobalsSet) = 1; local($bBVT) = 0; local($bNoCopy) = 0; local($bOfficialBuild) = 0; local($bShipBuild) = 0; local($bColor) = 1; local($bUpdate) = 0; local($bWin98) = 0; local($bCopyFailed) = 0; local($bBuildFailed) = 0; local($bAddLanguageString) = 0; # <- TODO: is there a better way to do this? local($bNothingDone) = 1; local($bVerbose) = 0; local($bSendMail) = 0; local($bErrorConcat) = 0; local($bDieOnError) = 0;
#lists local(@lArgs) = (); local(@lBuilds) = (); local(@lLanguages) = (); local(@lModifiers) = (); local(@lComponents) = ();
local(@lAllowedArgs) = (); local(@lAllowedComponents) = (); local(@lAllowedLanguages) = (); local(@lAllowedBuilds) = ("DEBUG", "RELEASE"); local(@lAllowedModifiers) = ("ALL", "REBUILD", "RESYNC", "TYPO", "UPDATE", "QUIET", "DEFAULT", "VERBOSE", "TEST", "MAIL");
local(@lAccelList) = (); local(@lAccelParam) = (); local(@lDefaultArgs) = ("SHIP", "REBUILD"); local(@lMailRecipients) = ($USERNAME); local(@lOfficialMailRecipients)= ($USERNAME);
local(@lSyncDirs) = (); local(@lCleanDirs) = (); local(@lStdSyncDirs) = ("RECURSE:".$sLibDir, "RECURSE:".$sBinExeDir, "RECURSE:".$sBinBatDir, "RECURSE:".$PROJROOT."\\inc");
#commands local($cmdIn) = $sBinExeDir."\\in.exe"; local($cmdOut) = $sBinExeDir."\\out.exe"; local($cmdSync) = $sBinExeDir."\\ssync.exe"; local($cmdShowVer) = $sBinExeDir."\\showver.exe"; local($cmdWindiff) = $sBinExeDir."\\windiff.exe"; local($cmdChkShip) = $sBinExeDir."\\chkship.exe"; local($cmdKillOpen) = $sBinExeDir."\\killopen.exe";
if (!-d $sLogDir) { EchoedMkdir($sLogDir); }
# Set OS version my($x, $sOSVer) = `ver`; #first line is blank $bWin98 = ($sOSVer =~ /windows 98/i);
local(*Main) = "*main::".$sShortBuildName; shift(@_);
if (!IsMemberOf("NONEWLOG", @_)) { local($fhBuildLog) = ""; #fwd declaration (so that begin build can use it) if (defined &SetLocalGlobalsAndBeginCustom) { return(SetLocalGlobalsAndBeginCustom(@_)); } else { return(Main(@_)); } } else { if (defined &SetLocalGlobalsAndBeginCustom) { return(SetLocalGlobalsAndBeginCustom(@_)); } else { return(Main(@_)); } } }
#################################################################################### # HASHES ####################################################################################
#descriptions of available options (if you don't define it here, it won't show up in usage) #capitalized letters are used as 'accelerators' (make sure there are no duplicates, the #script doesn't check for that)
#no single quotes or parens allowed (tooltips don't like them) %hOptionDescription = ( # <----------------------------- SCREEN WIDTH -------------------------------------> (accel) "Debug" => " include debug version - default", #D "Release" => " include release version", #R "All" => " include all buildtypes for this build", #A "REbuild" => " delete old build files and rebuild", #RE "TYpo" => " check for typos after build finishes", #TY "Test" => " test build - don't do official build", #T "DEFault" => " (+) include the default parameters with your custom parameters", #DEF "Verbose" => " increased script output", #V "Mail" => " send mail after build completes", #M "NoCopy" => " prevent copying of files", #NC "NoNewLog" => "don't open new log for build - log to currently open log, if exist", #NNL "ReSync" => " resync dirs before building - may not get all dependencies", #RS "Ship" => " build buildtypes for each specific component -shipping- to server", #S "bvt" => " run BVT tests after building", #BVT "bbt" => " BBT optimize build product (available in release build only)", #BBT "Halt" => " halt on error", #H "Quiet" => " suppress pop-up windows [html log open on exit, windiff, etc.]", #Q "AllLang" => " include all languages", #AL "AllComp" => " include all components", #AC # <----------------------------- SCREEN WIDTH -------------------------------------> (accel) );
####################################################################################
# ChangeTextColor()
# changes current html logging text to color passed in argument # if null argument, reverts to previous color
# a-jbilas, 04/20/99 - created
####################################################################################
sub ChangeTextColor { if ($bColor) { local($sColor) = @_; if ($sColor eq "") #reset color { # system("color 0f"); if ($fhBuildLog) { print($fhBuildLog "<\/font>"); } } else { if ($fhBuildLog) { print($fhBuildLog "<font color\=\"$sColor\">"); #remember to reset color first (so that there are no hanging font tags) } # system("color $colorcodes{$sColor}"); } } return(1); }
####################################################################################
# ParseArgs()
# Check all passed args, ensure that they are valid (members of @lAllowedArgs) and returns them # removes leading whitespace,-,/ and is case insensitive # takes expanded language names (english => en) and a buildnumber # if __BUILDNUMBER is member of @lAllowedArgs, will set 4-digit input to $sBuildNumber
# a-jbilas, 04/10/99
####################################################################################
sub ParseArgs { local(@args) = @_; @lPassedArguments = ();
if (@args == ()) { if (@lDefaultArgs == ()) { print(GetUsage()); exit(1); } print(STDOUT "No arguments specified, using build defaults : "); foreach $item (@lDefaultArgs) { print(STDOUT $item." "); } print(STDOUT "\n\n"); return("DEFAULT", @lDefaultArgs); } else { foreach $item (@args) { if ($item ne "") { $item =~ s/^\s*(\/|\-)//; #remove spaces, '/', '-' from beginning (allow -debug, /debug opt.) if ($item eq "?") { print(GetUsage()); exit(1); } # is the argument in AllowedArgs? (test expanded short languages as well) if (!IsMemberOf($item, @lAllowedArgs) && !IsMemberOf($longtoshlang{lc($item)}, @lAllowedArgs)) { # if we allow buildnumbers, is the argument a 4 digit build number? if ((IsMemberOf("__BUILDNUMBER", @lAllowedArgs) || IsMemberOf("__BUILDNUMBER", @args)) && $item =~ /^\d\d\d\d$/) { $sBuildNumber = $item; } # is the argument an accelerator abbreviation? elsif (IsMemberOf($item, @lAccelList)) { my($bAccelFound) = 0; for ($index = 0 ; !$bAccelFound ; ++$index) { if (lc($lAccelList[$index]) eq lc($item)) { if (!IsMemberOf($item, @lPassedArguments)) { @lPassedArguments = (uc($lAccelParam[$index]), @lPassedArguments); } $bAccelFound = 1; } elsif ($index >= @lAccelList) { carp("Error in ParseArgs(): end of accel list reached "); $bAccelFound = 1; #exit the loop } } } elsif (IsMemberOf("__IGNORE", @args)) { if (!IsMemberOf($item, @lPassedArguments)) { @lPassedArguments = (uc($item), @lPassedArguments); } } # must be an invalid argument, print usage list and quit else { print(STDERR "Error: What do you mean by: \'$item\' ?\n"); print(STDOUT GetUsage()."\n\n"); exit(1); } } # make sure the argument isn't inserted twice elsif (!IsMemberOf($item, @lPassedArguments) && !IsMemberOf($longtoshlang{lc($item)}, @lPassedArguments)) { if ($longtoshlang{lc($item)} ne "") { @lPassedArguments = (uc($longtoshlang{lc($item)}), @lPassedArguments); } else { @lPassedArguments = (uc($item), @lPassedArguments); } } } }
# append the default arguments (if DEFAULT was passed) if (IsMemberOf("DEFAULT", @lPassedArguments) && (@lDefaultArgs != ())) { foreach $elem (@lDefaultArgs) { if (!IsMemberOf($elem, @lPassedArguments)) { push(@lPassedArguments, $elem); } } } }
return(@lPassedArguments); }
####################################################################################
# Execute()
# executes first argument in eval block and tees output all to log (if open), failures to $sBuildMsg # if second argument non-null, will exit the script when an error is hit # outputs results to log and screen; returns 1 upon success, 0 upon failure
# a-jbilas, 04/20/99 - created # a-jbilas, 05/24/99 - added win98 support # a-jbilas, 06/15/99 - added bookmark support # a-jbilas, 06/16/99 - added $_Execute string support (will write output to $_Execute if equal to 1)
####################################################################################
sub Execute($;$$$) { my($sCmd, $bDieIfError, $bQuiet, $bIgnoreError) = @_; my($rc) = 1; my($sMsg) = ""; my($bLogExecute) = 0;
if ($_ExecuteQuiet) { $bQuiet = 1; }
if ($_Execute == 1) { $_Execute = ""; $bLogExecute = 1; } if (!$bQuiet) { PrintL(" - Executing \'".($bVerbose ? $sCmd : RemovePath($sCmd))."\'\n", PL_BLUE); } eval { if ($bWin98) { open (CMDIN, $sCmd.' |'); } else { open (CMDIN, $sCmd.' 2>&1 |'); } while (<CMDIN>) { if ($bLogExecute) { $_Execute .= $_; } elsif (!$bQuiet) { PrintL($_); } $sMsg .= $_; } close (CMDIN); };
if (!$bIgnoreError && $CHILD_ERROR) { if (!$bQuiet) { if (IsCritical()) { PrintLTip("Execution of ".RemovePath($sCmd)." FAILED\n", "Return code: ".($CHILD_ERROR/256), PL_BIGERROR | PL_SETERROR); PrintMsgBlock(split(/\n/, $sMsg)); } else { PrintLTip("Execution of ".RemovePath($sCmd)." FAILED\n", "Return code: ".($CHILD_ERROR/256), PL_ERROR | PL_SETERROR); } }
if ($bDieIfError || (IsCritical() && $bDieOnError)) # NOTE: bDieOnError is global, bDieIfError is local { exit($CHILD_ERROR/256); } $rc = 0; } if (!$bIgnoreError && !$rc && IsCritical()) { $bBuildFailed = 1; $bcStatus |= BC_FAILED; } return($rc); }
####################################################################################
# ExecuteAndOutputToFile()
# Executes the command in the first argument (string) and outputs it to a file # named in the second argument (string) # if the third argument is non-null, it will die() upon failure # reports success to screen and log; returns 1 upon success, 0 otherwise
# a-jbilas, 04/20/99 - created # a-jbilas, 06/15/99 - added bookmark support
####################################################################################
sub ExecuteAndOutputToFile($$;$$$) { my($sCmd, $sFile, $bDieIfError, $bQuiet, $bIgnoreError) = @_; my($rc) = 1; my($sMsg) = ""; my($pipe) = ($_ExecuteNoSTDERR ? "" : " 2>&1")." |"; if ($_ExecuteQuiet) { $bQuiet = 1; }
if (!open(FOUT, ">>$sFile")) { PrintL("Cannot open output file for $sCmd \>\> $sFile\n", PL_STDERR | PL_RED); $rc = 0; } else { if (!$bQuiet) { PrintL(" - Executing '".RemovePath($sCmd)." >> ".$sFile."'\n", PL_BLUE); } eval { if ($bWin98) { open (CMDIN, $sCmd.' |'); } else { open (CMDIN, $sCmd.' '.$pipe); } while (<CMDIN>) { print(FOUT $_); } close (CMDIN); };
if (!$bIgnoreError && $CHILD_ERROR) { if (!$bQuiet) { if (IsCritical()) { PrintLTip("Execution of ".RemovePath($sCmd)." FAILED\n", "Return code: ".($CHILD_ERROR/256), PL_BIGERROR | PL_SETERROR); } else { PrintLTip("Execution of ".RemovePath($sCmd)." FAILED\n", "Return code: ".($CHILD_ERROR/256), PL_ERROR | PL_SETERROR); } }
if ($bDieIfError || (IsCritical() && $bDieOnError)) { exit($CHILD_ERROR/256); }
$rc = 0; } close(FOUT); } if (!$bIgnoreError && !$rc && IsCritical()) { $bBuildFailed = 1; $bcStatus |= BC_FAILED; } return($rc); }
####################################################################################
# GetArgs()
# Builds and returns a list of allowed args in build
# a-jbilas, 06/21/99 - created
####################################################################################
sub GetArgs() { local(@m_lArgs) = @lAllowedArgs; @m_lArgs = Union(*m_lArgs, *lAllowedLanguages); #TODO: fix @m_lArgs = Union(*m_lArgs, *lAllowedBuilds); @m_lArgs = Union(*m_lArgs, *lAllowedModifiers); @m_lArgs = Union(*m_lArgs, *lAllowedComponents); return(@m_lArgs); }
####################################################################################
# GetSummary()
# returns a text summary of the build, based upon messages in $strBuildMsg # removes any html/non-interesting info before returning (preserves old $strBuildMsg as well)
# a-jbilas, 05/27/99 - created
####################################################################################
sub GetSummary { local($strTempBuildMsg) = $strBuildMsg; $strTempBuildMsg =~ s/\n//g; $strTempBuildMsg =~ s/<BR>/\n/ig; $strTempBuildMsg =~ s/<dd>/\n/ig; $strTempBuildMsg =~ s/<dl compact>.+?<\/dl>//igs; $strTempBuildMsg =~ s/<! DHTML ACTIVATION SCRIPT >.+?<! END DHTML ACTIVATION SCRIPT >//gs; $strTempBuildMsg =~ s/<[^>]*>//g; $strTempBuildMsg =~ s/\n[^\n]*log file[^\n]*\n//g; return("\n SUMMARY:\n-------------------------------------------------\n".$strTempBuildMsg."\n"); }
####################################################################################
# PrintToLogLarge()
# Prints string argument to STDOUT and, if $fhBuildLog is defined, to the # html log (in strong font)
# -USE FOR SECTION HEADER OUTPUT-
# a-jbilas, 04/20/99 - created # a-jbilas, 08/13/99 - Legacy, prefer PrintL()
####################################################################################
sub PrintToLogLarge { if ($fhBuildLog) { print($fhBuildLog "<font size=\"4\"><strong>"); PrintToLog(@_); print($fhBuildLog "<\/font><\/strong>"); } else { PrintL(@_); } }
####################################################################################
# PrintToLog()
# prints string argument to STDOUT and, if $fhBuildLog is defined, to the html log # searches input string on words such as 'fail' and 'warn', changes text color if found
# -USE FOR NORMAL OUTPUT-
# a-jbilas, 04/20/99 - created # a-jbilas, 08/13/99 - Legacy, prefer PrintL()
####################################################################################
sub PrintToLog { local(@output) = @_; local($sColor) = "";
foreach $elem (@output) { if (/fail/i) { $sColor = "red"; } elsif ((/warn/i) && ($sColor ne "red")) { $sColor = "purple"; } }
if ($sColor ne "") { ChangeTextColor($sColor); }
print(STDOUT @output);
if ($fhBuildLog) { foreach $elem (@output) { $elem =~ s/\n/<br>\n/g; print($fhBuildLog $elem); } }
if ($sColor ne "") { ChangeTextColor(); } }
####################################################################################
# PrintToLogErr()
# Prints string argument to STDERR and, if $fhBuildLog is defined, to the # html log (in red text) # -USE FOR ERROR OUTPUT-
# a-jbilas, 04/20/99 - created # a-jbilas, 08/13/99 - Legacy, prefer PrintL()
####################################################################################
sub PrintToLogErr { local(@lOutput) = @_; ChangeTextColor("red"); print(STDERR @lOutput); if ($fhBuildLog) { foreach $elem (@lOutput) { $elem =~ s/\n/<br>\n/g; print($fhBuildLog $elem); } } ChangeTextColor(); }
####################################################################################
# DumpVars()
# Appends huge list of every var in perl environment to file $sVarsLog # useful only for doing searches on specific variables
# a-jbilas, 04/10/99 - created
####################################################################################
sub DumpVars() { open(VARSLOG, ">>$sVarsLog"); print(VARSLOG "\n\n***********************************************************\nVARS AT "); local($package, $file, $line) = caller(); print(VARSLOG $package.' '.$file.' line: '.$line."\n\n\n"); foreach $i (%main::) { print(VARSLOG $i."=".$$i."\n"); } close(VARSLOG); }
####################################################################################
# SLMOperation
# does a slm operation, ignores the return # (it doesn't seem to mean anything) # and suppresses all the warnings - which are pretty much noise # second argument is for teeing output to file # (useful for checking if anything was changed)
# dougp, 04/10/99 - created
####################################################################################
sub SLMOperation { carp("Usage: SLMOperation(args, [teeToFile]) ") unless(@_ == 1 || @_ == 2);
my ($cmd, $sFileName) = @_; my ($op, $args) = split ' ', $cmd, 2; # echo to user $op .= ' "-f&"'; # this has to be on all commands anyway $cmd = "$op $args"; print $cmd, "\n"; # run eval { if ($sFileName ne "") { if(!open(FOUT, ">>$sFileName")) { PrintToLogErr("SLMOperation(@_) error: cannot open $sFileName for output"); } } if ($bWin98) { open(FPSYS, $cmd. ' |'); } else { open(FPSYS, $cmd. ' 2>&1 |'); } while (<FPSYS>) { if (!/warning:/ && !/^$/ && !/is not ghosted/) { print; if ($sFileName ne "") { print(FOUT); } } } if ($sFileName ne "") { close(FOUT); } close(FPSYS); }; if ($@) { warn("Run Time Error: $@"); } sleep 1; return $? == 0; }
####################################################################################
# CopyWithEchoOnError
# copies file in argument, echoes errors to $strBuildMsg on failure
# dougp, 5/10/99
####################################################################################
sub CopyWithEchoOnError { my ($cmd) = @_; print "copy ".$cmd, "\n"; if ($bWin98) { open (FPIN, 'copy '.$cmd.' |'); } else { open (FPIN, 'copy '.$cmd.' 2>&1 |'); } my $msg=""; while (<FPIN>) { print; $msg .= "<dd>".$_; } close (FPIN); if ($? != 0) { $strBuildMsg .= "<dd>copy ".$cmd." <b>FAILED</b>\n<dl compact><em>\n".$msg."</dl></em>\n"; $bCopyFailed = 1; } }
####################################################################################
# CopyLogs()
# copies logs to $sRootDropDir # use main build function name appended with x86/alpha and build number.html for log file name # will also append www toc for build (if exists) with build log ref and status
# a-jbilas, 05/14/99 - created # a-jbilas, 05/28/99 - will now only append if no log of same build exists and will update status of # existing log # a-jbilas, 07/01/99 - use http addresses instead of unc addresses
####################################################################################
sub CopyLogs() { my($rc) = 1;
EchoedMkdir($sLogDropDir);
if ($bOfficialBuild && !$bNoCopy) { my $sLinkCurBuild = '<td><img border=0 src="'. (($bcStatus & BC_NOTHINGDONE) ? "NothingDone" : (($bcStatus & BC_FAILED) ? "fail" : "succeed")). '.gif"> <a href="'.$sLogDropDir."\\".$sRemoteBuildLog.'">'.PROC.'</a></td>'."\n";
if (!EchoedCopy($sBuildLog, $sLogDropDir."\\".$sRemoteBuildLog)) { $rc = 0; } elsif (-e $sRemoteTOC) { PrintL(" - Updating web log TOC\n", PL_BLUE); my($fhTOC) = OpenFile($sRemoteTOC, "r"); my($sTOC) = "";
if ($fhTOC) { while (!$fhTOC->eof()) { my($sCurLine) = $fhTOC->getline();
if ($sCurLine =~ /Build $sBuildNumber/i) { $sTOC .= $sCurLine; # skip build header $sTOC .= $fhTOC->getline(); # skip <tr> if (PROC ne "x86") { $sTOC .= $fhTOC->getline(); # skip x86 build status link } $sCurLine = $fhTOC->getline(); $sCurLine = $sLinkCurBuild; }
$sTOC .= $sCurLine; } CloseFile($fhTOC); }
unlink($sRemoteTOC); $fhTOC = OpenFile($sRemoteTOC, "w"); if ($fhTOC) { $fhTOC->print($sTOC); CloseFile($fhTOC); } else { PrintL("Could not write to TOC (no write access?)\n", PL_ERROR); $rc = 0; } } }
return($rc); }
####################################################################################
# UpdateLogTOC()
# Update the logging TOC to include current build with status 'yellow' and a link to log location
# a-jbilas, 06/01/99 - created # a-jbilas, 06/02/99 - added to nlglib
####################################################################################
sub UpdateLogTOC($$) { my($remotetoc, $logname) = @_;
# TODO: potential file sync bug if ($bOfficialBuild && !$bNoCopy && (-e $remotetoc) && ($COMPUTERNAME ne "")) { PrintL(" - Updating web logs TOC ...\n\n", PL_NOLOG);
my($fhTOCFile) = OpenFile($remotetoc, "r"); if (!$fhTOCFile) { return(0); }
my($sTOCFile) = "";
my($sBuildHeader) = '<tr><td colspan="2"><center><b>Build '.$sBuildNumber.'</b></center></td></tr>'."\n"; my($sBuildBlank) = '<td></td>'."\n"; my($sBuildCur) = '<td><img border=0 src="waiting.gif"> <a href="' .TranslateToHTTP("\\\\".$COMPUTERNAME."\\".$PROJ."logs\\".RemovePath($logname))."\">" .PROC."</a></td>\n";
my($bUpdateIt) = 1;
while(!$fhTOCFile->eof()) { my($sCurLine) = $fhTOCFile->getline();
if ((($sCurLine =~ /Build \d\d\d\d/i) || ($sCurLine =~ /<\/table>/i)) && $bUpdateIt) { if ($sCurLine =~ /$sBuildNumber/) # we must have either done a previous build or another build beat us here # either way, make certain that the status is 'waiting' { # don't change the build header $sTOCFile .= $sCurLine; # skip the <tr> $sTOCFile .= $fhTOCFile->getline(); # if alpha, skip the first (x86) build link if (IsAlpha()) { $sTOCFile .= $fhTOCFile->getline(); } # rewrite our waiting build line $sCurLine = $fhTOCFile->getline(); $sCurLine = $sBuildCur; # if x86, skip the second (alpha) build link if (Isx86()) { $sTOCFile .= $sCurLine; $sCurLine = $fhTOCFile->getline(); } $bUpdateIt = 0; } else # this is not our build, insert ours before this build (or end of table) { $sTOCFile .= $sBuildHeader."<tr>\n"; if (IsAlpha()) { $sTOCFile .= $sBuildBlank; } $sTOCFile .= $sBuildCur; if (Isx86()) { $sTOCFile .= $sBuildBlank; } $sTOCFile .= "<\/tr>\n\n"; $bUpdateIt = 0; } } $sTOCFile .= $sCurLine; } CloseFile($fhTOCFile);
# output everything to new revised log file
unlink($remotetoc); $fhTOC = OpenFile($remotetoc, "w"); if ($fhTOC) { $fhTOC->print($sTOCFile); CloseFile($fhTOC); } else { PrintL("Could not write to TOC (no write access?)\n", PL_BIGERROR); return(0); } } return(1); }
####################################################################################
# InsertSummaryIntoLog()
# Inserts a summarized version of $strBuildMsg into the build at first '<! $name SUMMARY ENTRY POINT >' found
# a-jbilas, 06/03/99 - created
####################################################################################
sub InsertSummaryIntoLog($) { local($sLogFile) = @_; local($rc) = 1;
unlink($sLogFile.".tmp"); if ((-e $sLogFile) && copy($sLogFile, $sLogFile.".tmp")) { unlink($sLogFile); my($fhLogIn) = OpenFile($sLogFile.".tmp", "read"); my($fhLogOut) = OpenFile($sLogFile, "write"); while (<$fhLogIn>) { if (/<\! $sShortBuildName $nScriptStartTime SUMMARY ENTRY POINT >/) { print($fhLogOut "<font size=5><b>".BuildCodeToHTML($bcStatus)."</font></b>". "  <strong><font size=3>(". FmtDeltaTime(time() - $nScriptStartTime).")</strong></font><BR>\n". "<h3><strong>Summary:</h3></strong><BR>\n".$strBuildMsg."\n<BR>\n<BR>"); } print($fhLogOut $_); } CloseFile($fhLogIn); CloseFile($fhLogOut); unlink($sLogFile.".tmp"); } elsif ($bVerbose) { print(STDERR "InsertSummaryIntoLog() Error: Cannot copy $sLogFile to temp file"); $rc = 0 }
return($rc); }
####################################################################################
# Bookmark()
# if $fhBuildLog is defined, a <a name=> bookmark will be appended to the log and # the string passed to the function will be returned with an href to the bookmark's location # (this function is meant for adding bookmarks to $strBuildMsg)
# a-jbilas, 06/04/99 - created # a-jbilas, 09/20/99 - if second arg non-null, search for existing tag at beginning of str and concatinate href within
####################################################################################
sub Bookmark { my($string) = $_[0];
if ($fhBuildLog && $sShortBuildName && ($sBuildLog || $sRemoteBuildLog) && (defined $nErrorNumber)) { print($fhBuildLog "<a name=".$sShortBuildName.$nErrorNumber."><\/a><BR>\n"); my($log); if ($bOfficialBuild) { $log = TranslateToHTTP(($sLogDropDir ne "" ? $sLogDropDir."\\" : "").$sRemoteBuildLog); } else { $log = TranslateToHTTP($sBuildLog); } $log =~ s/\\/\//g; #replace \ with / for http links if ($_[1]) { $string =~ s/(<a [^>]*>)//; my($hrefstr) = $1; $hrefstr =~ s/<a /<a href="$log#$sShortBuildName$nErrorNumber"/; $string = $hrefstr.$string; } else { if ($string =~ /\n$/) { $string =~ s/\n$//g; $string = "<a href=\"".$log."#".$sShortBuildName.$nErrorNumber."\">".$string."<\/a>\n"; } else { $string = "<a href=\"".$log."#".$sShortBuildName.$nErrorNumber."\">".$string."<\/a>"; } } ++$nErrorNumber; } return($string); }
####################################################################################
# BuildAcceleratorLists()
# Extracts the accelerator (abbreviation) keys and inserts them into @lAccelList # (just the accelerators) and the matching param for the accel into @lAccelParam
# a-jbilas, 06/09/99 - created
####################################################################################
sub BuildAcceleratorLists() { my(@lKeys) = keys(%hOptionDescription); @lAccelParam = (); @lAccelList = (); foreach $key (@lKeys) { my($keyAccel) = ""; for ($index = 0 ; $index < length($key) ; ++$index) { if ((vec($key, $index, 8) > 64) && (vec($key, $index, 8) < 91)) { $keyAccel .= substr($key, $index, 1); } } if ($keyAccel ne "") { push(@lAccelParam, $key); push(@lAccelList, $keyAccel); } }
# special accelerator key for default settings push(@lAccelParam, "DEFAULT"); push(@lAccelList, "+"); }
####################################################################################
# GetLatestBuildDir()
# given a directory (and an optional subdirectory), will return the latest 4 digit # build number named subdirectory of the specified directory (containing the optional # subdirectory in the build (such as 'x86'), if specified) # if no valid dirs exist, will return a null string
# a-jbilas, 06/15/99 - created
####################################################################################
sub GetLatestBuildDir($;$) { my($sBuildDir, $sSubDir) = @_;
PrintL("looking for latest build dir in $sBuildDir, $sSubDir\n", PL_VERBOSE);
my($sLatestBuild) = "0000"; local(@lDirs) = grep(/\d\d\d\d$/, GetSubdirs($sBuildDir));
foreach $dir (@lDirs) { my($sBldNum) = $dir; $sBldNum =~ s/.*(\d\d\d\d)$/$1/; if ((-d $sBuildDir."\\".$sBldNum.($sSubDir ne "" ? "\\".$sSubDir : "")) && ($sBldNum > $sLatestBuild)) { $sLatestBuild = $sBldNum; } }
if (($sLatestBuild eq "0000") && !(-d $sBuildDir."\\".$sLatestBuild.($sSubDir ne "" ? "\\".$sSubDir : ""))) { return(""); } else { return($sBuildDir."\\".$sLatestBuild.($sSubDir ne "" ? "\\".$sSubDir : "")); } }
####################################################################################
# GrabCookie()
# Grabs the cookie -- when passed r (read) or w (write) string as parameter, if cookie # grab fails, will wait 10 minutes before trying another grab. If cookie could not be # grabbed after 30 attempts (5 hours), function returns 0, it otherwise returns 1
# a-jbilas, 07/14/99 - created
####################################################################################
sub GrabCookie { my($rc) = 1; my($nMaxAttempts) = 30; my($bCookieGrabbed) = 0;
if (($_[0] ne "r") && ($_[0] ne "w")) { carp("Usage: GrabCookie(r/w) "); $rc = 0; } else { PrintL("Attempting to obtain a ".$_[0]." lock on cookie\n"); for ($nAttempt = 1 ; (!$bCookieGrabbed && ($nAttempt <= $nMaxAttempts)) ; ++$nAttempt) { if (Execute('cookie -v'.$_[0].'c "Locked for the '.PROC.' build"')) { PrintL("Cookie successfully grabbed\n"); $bCookieGrabbed = 1; } elsif ($nAttempt != 30) { PrintL("Cookie grab failed, waiting 10 minutes for cookie to be freed ", PL_WARNING);
for ($time = 1 ; $time <= 10 ; ++$time) #sleep ten minutes { print("."); sleep(60); } PrintL("\n"); } } }
if (!$bCookieGrabbed) { PrintL("GrabCookie() Error: Cookie could not be obtained\n", PL_BIGERROR); $rc = 0; } return($rc); }
sub FreeCookie() { return(Execute('cookie -f')); }
####################################################################################
# PrintToMsg()
# Outputs 1st string parameter to $strBuildMsg with optional additional string # parameters output as subsets to 1st string (all properly formatted)
# a-jbilas, 07/22/99 - created
####################################################################################
sub PrintToMsg { local(@lOutput) = @_; if ($lOutput[0] =~ /fail/i) { PrintToLogErr($lOutput[0]); } else { PrintToLog($lOutput[0]); }
$lOutput[0] =~ s/(failed|succeeded|succeeds)/<bold>$1<\/bold>/gi; $strBuildMsg .= "<dd>".$lOutput[0]."\n"; shift(@lOutput);
if ($lOutput) { $strBuildMsg .= "<dl compact><em>\n"; foreach $msg (@lOutput) { PrintToLog($msg); $msg =~ s/\n/<BR>\n/g; $strBuildMsg .= "<dd>".$msg; } $strBuildMsg .= "<\/dl><\/em>\n"; } }
sub PrintMsgBlock { if (scalar(@_) == 0) { return(); } my($lineNum) = 0; my($maxReached) = 0; PrintL("<dl compact>", PL_MSGONLY | PL_MSGCONCAT);
foreach $elem (@_) { foreach $line (split(/\n+/, $elem)) { if ((!defined $nMaxErrLines) || (!$maxReached && ($lineNum < $nMaxErrLines))) { if ($line eq "") { PrintL("<BR>\n", PL_MSGONLY); } else { PrintL($line."\n", PL_ITALIC | PL_MSGONLY); } } elsif (!$maxReached) { PrintL("Too many errors to display, click here to view continuation\n", PL_ITALIC | PL_MSGONLY | PL_RED | PL_BOLD | PL_BOOKMARK); $maxReached = 1; } ++$lineNum; } } PrintL("</dl>", PL_MSG | PL_NOSTD | PL_NOLOG | PL_MSGCONCAT); }
sub GetSLMLog { my($strArg) = ""; my($dir) = ""; my($time) = ""; my(%log) = "";
foreach $i (@_) { if ($i eq "today") { my($sec, $min, $hour, $mday, $mon, $year) = localtime(time); $strArg .= " -t ".($mon + 1)."/$mday/$year"; } elsif ($i eq "user") { $strArg .= " -u $ENV{COMPUTERNAME}"; } else { $strArg .= " $i"; } } open(FPIN, 'log "-rfvi&" '.$strArg.' |'); while (<FPIN>) { if (/^time/ || /^log : warning: /) { # skip header and warnings } elsif (/Log for (.*):/) { $dir = $1.$2; #print "Directory is ".$dir."\n"; } elsif (/^(\d\d)-(\d\d)-(\d\d)\@(\d\d):(\d\d):(\d\d)\b(.*)$/) { $time = "$3/$1/$2 $4:$5:$6 ";
my($day, $who, $what, $file, $ver1, $comment) = split ' ', $7, 6; if ($file =~ /.+\\([\w.]+)/) { $file = "$dir\\$1"; } if ($comment =~ /I\d+ +(.*)/) { $comment = $1; } if ($what ne "release") { $log{"$time $who $what $file"} = " - $comment\n"; } } } close(FPIN); my($retVal) = ""; foreach $k (reverse sort keys %log) { $retVal .= $k.$log{$k}; }
return($retVal); }
sub FormatLogAsHTML($) { if ($_[0] eq "") { return('<font size=4><b>No History Available</b></font>'); }
my($result) = "<table border=1><caption><font size=4><b>Recent History</b></font></caption>\n". "<tr><th>when</th><th>who</th><th>what</th><th>file</th><th>comment</th></tr>\n";
foreach $line (split(/\n/, $_[0])) { my($date, $time, $who, $what, $file, $comment) = split(' ', $line, 6); if ($comment =~ /^- (.*)/) { $comment = $1; } $result .= "<tr><td>$date $time</td><td>$who</td><td>$what</td><td>$file</td><td>$comment</td></tr>\n"; } close (FPIN); return($result."</table>\n"); }
sub IsCritical() { if (!defined $__CRITICAL_SECTION) { $__CRITICAL_SECTION = 1; }
if ($__CRITICAL_SECTION > 0) { return(1); } else { return(0); } }
sub UpdateDir { my($sSLMDir, $sSrcDir, $bRecurse, $bCheckForNew, $bForceAdd, $bCheckInAfterUpdate) = @_;
PushD($sSrcDir); foreach $dir (GetSubdirs("", $bRecurse)) { if (!-e $sSLMDir."\\".$dir."\\slm.ini") { if ($bCheckForNew) { my($ret) = ""; if (!$bForceAdd) { PrintL("Add new dir ".$dir."? (y\/n\/a) "); $ret = <STDIN>; } if ($ret =~ /a/) { $bForceAdd = 1; } if ($bForceAdd || ($ret =~ /y/i)) { EchoedMkdir($sSLMDir."\\".$dir); PushD(GetPath($sSLMDir)); Execute("addfile -f -c \"ActivePerl Update Dir\" ".$dir); PopD(); } } else { PrintL("Warning: ".$dir." not found in current perl version\n", PL_WARNING); } } } # foreach $file (grep(!/\.dll$/, GetFiles("", $bRecurse))) foreach $file (grep(!/^slm\.ini$/i, GetFiles("", $bRecurse))) { if (!-e $sSLMDir."\\".$file) { if ($bCheckForNew) { my($ret) = ""; if (!$bForceAdd) { PrintL("Add new file ".$file."? (y\/n\/a) "); $ret = <STDIN>; } if ($ret =~ /a/) { $bForceAdd = 1; } if ($bForceAdd || ($ret =~ /y/i)) { Execute("copy ".$file." ".$sSLMDir."\\".$file); PushD(GetPath($sSLMDir."\\".$file)); Execute("addfile -f -c \"ActivePerl Build 519 File\" ".RemovePath($file)); PopD(); } } else { PrintL("Warning: ".$file." not found in current perl version\n", PL_WARNING); } } else { if (!EchoedCompare($file, $sSLMDir."\\".$file)) { PrintL(" - Updating ".$file."\n", PL_BLUE); PushD(GetPath($sSLMDir."\\".$file)); Execute("out -f ".RemovePath($file)); PopD(); Execute("copy ".$file." ".$sSLMDir."\\".$file); if ($bCheckInAfterUpdate) { PushD(GetPath($sSLMDir."\\".$file)); Execute("in -f -c \"Update\" ".RemovePath($file)); PopD(); } } } }
PopD(); #$sSrcDir }
sub Depends { foreach $var (@_) { if (!defined $$var) { PrintL("build script warning: variable dependency ".$var." not defined\n", PL_BIGWARNING); carp("Location:"); } } }
sub BuildCodeToHTML($) { my($str) = "";
if ($_[0] & BC_FAILED) { $str .= "<font color=\"red\">FAILED<\/font> "; } elsif ($_[0] & BC_NOTHINGDONE) { $str .= "<font color=\"blue\">NOTHING DONE<\/font> "; } else { $str .= "<font color=\"green\">SUCCEEDED<\/font> "; }
local(@lSecondaryFailures) = (); if ($_[0] & BC_COPYFAILED) { push(@lSecondaryFailures, "copy"); } if ($_[0] & BC_BVTFAILED) { push(@lSecondaryFailures, "bvt"); } if ($_[0] & BC_CABFAILED) { push(@lSecondaryFailures, "msi build"); } if ($_[0] & BC_CHKSHIPFAILED) { push(@lSecondaryFailures, "chkship"); }
if (@lSecondaryFailures != ()) { $str .= "<font color=\"orange\">(with ".join(" and ", @lSecondaryFailures)." failures)<\/font>"; } return($str); }
# ARGS: # [str] err # OPT ARGS: # [bool] concat (default=0)
sub SetError($;$) { if ($bErrorConcat) { if ($_[1] || ($_[0] =~ /\n$/)) { $ERROR .= $_[0]; } else { $ERROR .= $_[0]."\n"; } } else { $ERROR = $_[0]; } }
sub ResetError() { $ERROR = ""; }
######################################################################## ######################## SECTION BLOCKS ################################ ########################################################################
sub BEGIN_CRITICAL_SECTION() { if (!defined $__CRITICAL_SECTION) { $__CRITICAL_SECTION = 1; } else { ++$__CRITICAL_SECTION; }
}
sub END_CRITICAL_SECTION() { if (!defined $__CRITICAL_SECTION) { $__CRITICAL_SECTION = 0; } else { --$__CRITICAL_SECTION; } }
sub BEGIN_NON_CRITICAL_SECTION() { END_CRITICAL_SECTION(); }
sub END_NON_CRITICAL_SECTION() { BEGIN_CRITICAL_SECTION(); }
sub BEGIN_DHTML_NODE { if ($bDHTMLActive) { PrintL("<div class=\"parent\">" ."<img src=http://iit/images/node.bmp> ".(($_[0] eq "") ? "(click to expand)" : $_[0]) ."<BR><span class=\"childContainer\"><div>", PL_NOSTD); } }
sub END_DHTML_NODE() { if ($bDHTMLActive) { PrintL("</div></span></div>", PL_NOSTD); } }
sub ParseArgs2 { local(@lUnparsedArgs) = @_; local(%hArgs) = ();
foreach $elem (@_) { # first make sure that no spaces were paired with commas if (($elem =~ /^\,/) || ($elem =~ /\,$/)) { PrintL("ParseArgs() Fatal Error: separate sub-elements with commas only (no spaces)\n\n", PL_BIGERROR); %hArgs->{"__FATAL"} = 1; } elsif ($elem =~ /:/) { my($arg, $subargs) = split(":", $elem, 2); $subargs =~ s/\,/ /g; %hArgs->{uc($arg)} = uc($subargs); } else { %hArgs->{uc($elem)} = 1; } }
return(%hArgs); }
sub CheckArgs { my($hArgs, $hAcceptedArgs) = @_; # local(%hArgs) = %$phArgs; # local(%hAcceptedArgs) = %$phAcceptedArgs;
my($rc) = 1;
%hAcceptedArgs->{"__OFFICIAL"} = 1; %hAcceptedArgs->{"__BUILDNUMBER"} = 1;
if (%hArgs->{"__FATAL"}) { $rc = 0; } elsif (!%hArgs->{"__IGNORE"}) { foreach $arg (keys(%hArgs)) { local(@lAcceptedVals) = StrToL(%hAcceptedArgs->{$arg}); if (@lAcceptedVals == ()) { PrintL("CheckArgs() Error: ".$arg." is not a valid parameter\n\n", PL_BIGERROR); $rc = 0; } foreach $val (StrToL(%hArgs->{$arg})) { if (!IsMemberOf($val, @lAcceptedVals)) { PrintL("CheckArgs() Error: ".$val." is not a valid sub-parameter to ".$arg."\n\n", PL_BIGERROR); $rc = 0; } } } } return($rc); }
$__IITENVPM = 1; 1;
|