--- /dev/null Thu Jan 01 00:00:00 1970 +0000
+++ b/bintools/evalid/EvalidMD5.pm Mon May 10 19:54:49 2010 +0100
@@ -0,0 +1,499 @@
+#
+# Copyright (c) 2003-2009 Nokia Corporation and/or its subsidiary(-ies).
+# All rights reserved.
+# This component and the accompanying materials are made available
+# under the terms of the License "Eclipse Public License v1.0"
+# which accompanies this distribution, and is available
+# at the URL "http://www.eclipse.org/legal/epl-v10.html".
+#
+# Initial Contributors:
+# Nokia Corporation - initial contribution.
+#
+# Contributors:
+#
+# Description:
+# This module implements the MD5 version of the Evalid's compare
+#
+
+package EvalidMD5;
+
+use strict;
+use Carp;
+use File::Find;
+use IO::File;
+use Sys::Hostname;
+
+use FindBin;
+use lib "$FindBin::Bin";
+require Exporter;
+use File::Basename;
+
+use vars qw($VERSION );
+$VERSION = 0.02;
+
+use EvalidCompare;
+
+# Public
+
+# MD5Generate
+#
+# Inputs
+# $iLeftDir - Left side directory or File containing listing of left side
+# $iResultsFile - Name of file to write the list of files and MD5 checksum to
+# $iExclude - Reference to array of regular expression patterns to exclude
+# $iInclude - Reference to array of regular expression patterns to include
+# $iListFile - Filename to read for list of files from [Optional]
+# $iDumpDir - Directory to generate dump of comparison data [Optional]
+#
+# Outputs
+#
+# Description
+# This function generates the MD5 checksum file
+
+sub MD5Generate
+{
+ my ($iLeftDir, $iResultsFile, $iExclude, $iInclude, $iListFile, $iDumpDir) = @_;
+
+ my (@iLeftFiles);
+
+ # This should have been check before getting here but to be on the safe side
+ # check again, as it should never overwrite or append to the file
+ croak "$iResultsFile already exists" if (-e $iResultsFile);
+ croak "$iListFile does exist" if ((defined($iListFile) && (!-e $iListFile)));
+
+ if (-d $iLeftDir){
+ # Make sure all the \ are / for the RegEx
+ $iLeftDir =~ s/\\/\//g;
+ # Make sure it does not have a / on the end
+ $iLeftDir =~ s/\/$//;
+ # If $iFileList is defined then use the filelist to generate @iLeftFiles
+ if (defined($iListFile))
+ {
+ # Generate list of files
+ @iLeftFiles = &FilterDir($iLeftDir,$iExclude,$iInclude, $iListFile);
+ } else {
+ # Generate the the directory listing
+ @iLeftFiles = &FilterDir($iLeftDir,$iExclude,$iInclude);
+ }
+ } else {
+ croak "$iLeftDir is not a directory";
+ }
+
+ # Open the results file for writing
+ my ($fResultsFile) = new IO::File;
+ croak "Cannot for $iResultsFile for writing" unless ($fResultsFile->open("> $iResultsFile"));
+
+ # Write headers to the file
+ print $fResultsFile "Host:".&hostname()."\n";
+ print $fResultsFile "Username:".$ENV{'USERNAME'}."\n";
+ print $fResultsFile "Date-Time:".scalar(localtime)."\n";
+ print $fResultsFile "Version:".$VERSION."\n";
+ print $fResultsFile "Directory:".$iLeftDir."\n";
+ print $fResultsFile "FileList:".$iListFile."\n" if (defined($iListFile));
+ print $fResultsFile "Exclusion(s):";
+ foreach my $iExc (sort @$iExclude)
+ {
+ print $fResultsFile "$iExc ";
+ }
+ print $fResultsFile "\n";
+
+ print $fResultsFile "Inclusion(s):";
+ foreach my $iInc (@$iInclude)
+ {
+ print $fResultsFile "$iInc ";
+ }
+ print $fResultsFile "\n";
+
+ print $fResultsFile "----------------\n";
+
+ # Write out sorted list of files with MD5 Checksums
+ foreach my $iFile (sort @iLeftFiles)
+ {
+ my ($MD5, $type) = &EvalidCompare::GenerateSignature($iLeftDir."/".$iFile, $iDumpDir);
+ print $fResultsFile $iFile." TYPE=".$type." MD5=".$MD5."\n";
+ }
+
+ $fResultsFile->close;
+}
+
+
+# MD5Compare
+#
+# Inputs
+# $iLeftFile - Left side File containing listing and MD5 of left side
+# $iRightFile - Right side File containing listing and MD5 of right side
+# $iVerbose - Verbose Flag
+# $iLog - Logfile name
+#
+# Outputs
+# %iCommon - hash relative filenames with values of file type that are in both directories and Compare results/types
+# %iDiff - hash relative filenames with values of file type and directory side infomation
+#
+# Description
+
+sub MD5Compare
+{
+ my ($iLeftFile) = shift;
+ my ($iRightFile) = shift;
+ my ($iVerbose) = defined($_[0]) ? shift : 0;
+ my ($iLog) = defined($_[0]) ? shift : *STDOUT;
+
+ my (%iCommon, %iDiff);
+
+ my (%iLeftFiles, %iRightFiles);
+ my (%iLeftHeaders, %iRightHeaders);
+
+ # Backup check the files are available to read.
+ croak "$iLeftFile is not a file" unless (-f $iLeftFile);
+ croak "$iRightFile is not a file" unless(-f $iRightFile);
+
+ #Read the files
+ &ReadMD5File($iLeftFile, \%iLeftFiles, \%iLeftHeaders);
+ &ReadMD5File($iRightFile, \%iRightFiles, \%iRightHeaders);
+
+ # Check Critical headers
+ foreach my $iHeader (sort keys %iLeftHeaders)
+ {
+ # Warning is certain headers are not identical
+ if ($iHeader =~ /^Version|Exclusion\(s\)|Inclusion\(s\)/)
+ {
+ print $iLog "WARNING:$iHeader is different\n" if ($iLeftHeaders{$iHeader} ne $iRightHeaders{$iHeader});
+ }
+ }
+
+ # A Hash is used to combine the two directory listing using the filename as the key
+ my %iCombinedFiles;
+
+ # Enter the files from the left side listing and check the right side critical header are the same
+ # Note all filenames are turned to lower case as this is designed to only wotk on Windows
+ foreach my $iFile (sort keys %iLeftFiles)
+ {
+ $iCombinedFiles{$iFile} = "Left";
+ }
+
+ # Enter the files from the right side listing
+ # Note all filenames are turned to lower case as this is designed to only wotk on Windows
+ foreach my $iFile (sort keys %iRightFiles)
+ {
+ # Check to see if any entry for this file exists on the left side
+ if ((defined ($iCombinedFiles{$iFile})) && ( $iCombinedFiles{$iFile} eq "Left"))
+ {
+ # Yes, so add to the Common set
+ # Check if the MD5 checksum matches
+ # The [0] element is the file type
+ if ($iLeftFiles{$iFile}[0] ne $iRightFiles{$iFile}[0])
+ {
+ $iCommon{$iFile} = [$iLeftFiles{$iFile}[0]." to ".$iRightFiles{$iFile}[0], "Type Changed"];
+ } elsif ($iLeftFiles{$iFile}[1] eq $iRightFiles{$iFile}[1]) {
+ $iCommon{$iFile} = [$iLeftFiles{$iFile}[0],"OK"];
+ } else {
+ $iCommon{$iFile} = [$iLeftFiles{$iFile}[0],"Different"];
+ }
+ # The filename key is not needed any more as both sides have been processed, so delete the hash entry
+ delete $iCombinedFiles{$iFile};
+ } elsif (!defined($iCombinedFiles{$iFile})) {
+ # No, the key is not defined, so this filename is only in the right side
+ $iDiff{$iFile} = [$iRightFiles{$iFile}[0],"Right"];
+ # The filename key is not needed any more as both sides have been processed, so delete the hash entry
+ delete $iCombinedFiles{$iFile};
+ }
+ }
+
+ # Add the files left in the hash to the Left side only list
+ foreach my $iFile (sort keys %iCombinedFiles)
+ {
+ $iDiff{$iFile} = [$iLeftFiles{$iFile}[0],"Left"];
+ }
+
+ # Return References to the Arrays
+ return (\%iCommon, \%iLeftHeaders, \%iRightHeaders, \%iDiff);
+}
+
+# FilterDir
+#
+# Inputs
+# $iDir - Directory to process
+# $iExclude - Reference to array of regular expression patterns to exclude
+# $iInclude - Reference to array of regular expression patterns to include
+# $iListFile - Filename to read for list of files from [Optional]
+#
+# Outputs
+# @iFinalFileList - Filtered list relative filenames
+#
+# Description
+# This function produces a filtered list of filenames in the specified directory or from file
+
+sub FilterDir
+{
+ my ($iDir,$iExclude,$iInclude, $iListFile) = @_;
+
+ my (@iFileList, @iFinalFileList, $iFileName);
+
+ if(defined($iListFile))
+ {
+ open LIST, "$iListFile" or croak "Cannot open $iListFile\n";
+ while(<LIST>)
+ {
+ next if /^\s+$/; # skip blank lines
+ my $iFileName = $iDir."/".$_;
+ chomp $iFileName; # Remove new line
+ if(-e $iFileName)
+ {
+ # The listed file exists add to the @iFileList
+ push @iFileList, $iFileName;
+ } else {
+ print "Warning: Cannot find $iFileName\n";
+ }
+ }
+ close LIST;
+ } else {
+ # Produce full filelist listing without directory names
+ find sub { push @iFileList, $File::Find::name if (!-d) ;}, $iDir;
+ }
+
+ foreach my $iFile ( @iFileList)
+ {
+ my $iExcludeFile = 0;
+
+ # Remove the specified directory path from the front of the filename
+ $iFile =~ s#^$iDir/##;
+
+ # Process all Exclude RegEx to see if this file matches
+ foreach my $iExcludeRegEx (@$iExclude)
+ {
+ if ($iFile =~ /$iExcludeRegEx/i)
+ {
+ # Mark this file to be excluded from the final list
+ $iExcludeFile = 1;
+ }
+ }
+
+ # Process all Include RegEx to see if this file matches
+ foreach my $iIncludeRegEx (@$iInclude)
+ {
+ if ($iFile =~ /$iIncludeRegEx/i)
+ {
+ # Mark this file to be Included in the final list
+ $iExcludeFile = 0;
+ }
+ }
+
+ # Added the file to the final list based on the flag
+ push @iFinalFileList, lc($iFile) unless $iExcludeFile;
+ }
+
+ return @iFinalFileList;
+
+}
+
+# MD5ComparePrint
+#
+# Inputs
+# $iCommon - Reference to Hash of common file names and the result the comparision
+# $iLeftHeaders - Reference to Hash contain the left side
+# $iRightHeaders - Reference to Hash contain the right side
+# $iDiff - Reference to Hash of relative filenames with values of file type and left/right directory side
+# $iLog - Logfile name
+#
+# Outputs
+#
+# Description
+# This function prints the output of a Compare
+sub MD5ComparePrint
+{
+ my ($iCommon) = shift;
+ my ($iLeftHeaders) = shift;
+ my ($iRightHeaders) = shift;
+ my ($iDiff) = shift;
+ my ($iLog) = shift;
+
+ my ($iFailed) = 0; # Count of the failed comparisions
+ my ($iPassed) = 0; # Count of the Passed comparisions
+ my ($iLeft) = 0; # Count of the Passed comparisions
+ my ($iRight) = 0; # Count of the Passed comparisions
+
+
+ my ($sec, $min, $hour, $mday, $mon, $year) = localtime(time);
+ printf $iLog "\n----------------\n%02d:%02d %02d/%02d/%04d\n", $hour, $min, $mday, $mon+1, $year+1900;
+ print $iLog "evalid\nLeft Side=$ARGV[0]\nRight Side=$ARGV[1]\n";
+ print $iLog "\nLeft side information\n";
+ foreach my $iHeader (sort keys %$iLeftHeaders)
+ {
+ print $iLog $iHeader.":".$$iLeftHeaders{$iHeader}."\n";
+ }
+ print $iLog "\nRight side information\n";
+ foreach my $iHeader (sort keys %$iRightHeaders)
+ {
+ print $iLog $iHeader.":".$$iRightHeaders{$iHeader}."\n";
+ }
+ print $iLog "\n";
+
+ foreach my $iFile (sort keys %$iCommon)
+ {
+ if ($$iCommon{$iFile}[1] eq "OK")
+ {
+ print $iLog "Passed:$iFile (".$$iCommon{$iFile}[0].")\n";
+ $iPassed++;
+ }
+ }
+ print $iLog "\n";
+
+ foreach my $iFile (sort keys %$iCommon)
+ {
+ if ($$iCommon{$iFile}[1] eq "Type Changed")
+ {
+ print $iLog "Type Changed:$iFile (".$$iCommon{$iFile}[0].")\n";
+ $iFailed++;
+ }
+ }
+ print $iLog "\n";
+
+ foreach my $iFile (sort keys %$iCommon)
+ {
+ if ($$iCommon{$iFile}[1] eq "Different")
+ {
+ print $iLog "Failed:$iFile (".$$iCommon{$iFile}[0].")\n";
+ $iFailed++;
+ }
+ }
+ print $iLog "\n";
+
+ foreach my $iFile (sort keys %$iDiff)
+ {
+ if ($$iDiff{$iFile}[1] eq "Left")
+ {
+ print $iLog "Missing Left:$iFile (".$$iDiff{$iFile}[0].")\n";
+ $iLeft++;
+ }
+ }
+ print $iLog "\n";
+
+ foreach my $iFile (sort keys %$iDiff)
+ {
+ if ($$iDiff{$iFile}[1] eq "Right")
+ {
+ print $iLog "Missing Right:$iFile (".$$iDiff{$iFile}[0].")\n";
+ $iRight++;
+ }
+ }
+
+ print $iLog "\n\nSummary\n";
+ print $iLog "Total files processed=".($iPassed+$iFailed+$iRight+$iLeft)."\n";
+ print $iLog "Files Passed=$iPassed\n";
+ print $iLog "Files Failed=$iFailed\n";
+ print $iLog "Missing Files in Left=".$iRight."\n";
+ print $iLog "Missing Files in Right=".$iLeft."\n";
+
+ return ($iFailed);
+}
+# NameOnly
+#
+# Inputs
+# $filename - may contain path and file extension
+#
+# Outputs
+# $nameOnly - filename without extension and path
+#
+# Description
+# This routine is used to extract the name of the file
+# only from a filename that may include a path and file extension
+#
+sub NameOnly
+{
+ my ($filename) = @_;
+ my $nameOnly = basename($filename);
+ $nameOnly =~ s/(\w+.*)\.\w+$/$1/;
+ return $nameOnly;
+}
+
+# MD5CompareZipDel
+#
+# Inputs
+# $iCommon - Reference to Hash of common file names and the result the comparision
+# $iDiff - Reference to Hash of relative filenames with values of file type and left/right directory side
+# $iLeft - filename of Left side directory
+# $iRight - filename of Right side directory
+#
+# Outputs
+#
+# Description
+# This function prints the output of a Compare results in a format ready for creating the a
+# Zip and batch file to upgrade the left side to the equivalent of the right side.
+sub MD5CompareZipDel
+{
+ my ($iCommon) = shift;
+ my ($iDiff) = shift;
+ my ($iLeft) = shift;
+ my ($iRight) = shift;
+
+ # Build a suitable name for outputfiles based on input filenames
+ $iLeft = NameOnly($iLeft);
+ $iRight = NameOnly($iRight);
+ my ($iBasename) = $iLeft."_to_".$iRight;
+
+ open DELLIST, ">del_$iBasename.bat";
+ open ZIPLIST, ">zip_$iBasename.log";
+ open ZIPBAT, ">zip_$iBasename.bat";
+
+ foreach my $iFile (sort keys %$iCommon)
+ {
+ if (($$iCommon{$iFile}[1] eq "Different") || ($$iCommon{$iFile}[1] eq "Type Changed"))
+ {
+ print ZIPLIST "$iFile\n";
+ }
+ }
+
+ foreach my $iFile (sort keys %$iDiff)
+ {
+ if ($$iDiff{$iFile}[1] eq "Right")
+ {
+ print ZIPLIST "$iFile\n";
+ } else {
+ # DEL needs the / to be \
+ $iFile =~ s/\//\\/g;
+ print DELLIST "del /F $iFile\n";
+ }
+ }
+
+ print ZIPBAT "zip $iBasename.zip -@<zip_$iBasename.log\n";
+
+ close DELLIST;
+ close ZIPLIST;
+ close ZIPBAT;
+
+}
+
+
+# Private
+
+# ReadMD5File
+#
+# Inputs
+# $iResultsFile - Results filename to process
+# $iResults - Reference to Results Hash
+# $iHeaders - Reference to Headers Hash
+#
+# Outputs
+#
+# Description
+# This function reads the conntent of the MD5 File in to a Hash
+sub ReadMD5File
+{
+ my ($iResultsFile, $iResults, $iHeaders) = @_;
+
+ # Open the results file for reading
+ open (INPUT, "$iResultsFile") or croak "Cannot for $iResultsFile for reading";
+ while (<INPUT>)
+ {
+ if (/^(\S+?):(.*)/)
+ {
+ $$iHeaders{$1} = $2;
+ } elsif (/^(\S+?)\sTYPE=(.*?)\sMD5=(.*)/){
+ $$iResults{$1} = [$2,$3];
+ }
+ }
+ close INPUT;
+
+}
+
+1;