腳本功能
腳本使用Perl編寫,計算文件的MD5值
腳本用途
服務器在安裝完操作系統後,計算PATH變量裡面所有的二進制文件的MD5、計算單個文件MD5值、計算某個目錄(包括子目錄)下所有的文件的MD5值記錄到文件,等日後再生成一份,然後2份文件進行對比,即可知道這些文件是否又被改動過.
腳本用法(5種用法)
- 腳本名 -p [ > 輸出的文件名] ← 該功能是計算PATH變量裡面所有文件的MD5值
- 腳本名 文件1 文件2 文件3 ... [ > 輸出的文件名] ← 該功能是計算指定文件的MD5值
- 腳本名 目錄1 目錄2 目錄3 ... [ > 輸出的文件名] ← 該功能是計算指定目錄下所有文件的MD5值
- 腳本名 -c MD5文件1 MD5文件2 [ > 輸出的文件名] ← 該功能是比較同一個目錄下所有文件生成的2次MD5文件中有哪些MD5值不一致
- 腳本名 目錄 -f MD5文件 [ > 輸出的文件名] ← 該功能是跟上面的功能類似,只是這個不生成文件直接比對
注意事項
不能對/proc目錄進行計算
不能對/(根)進行計算,因為/裡面包含/proc
不能對掛載在Linux上Windows的共享目錄進行計算
腳本內容
- #!/usr/bin/perl
- use Digest::MD5;
- use File::Find;
-
- # 2012-11-24 22:41 Leo [email protected]
- # Blog:http://linux5588.blog.51cto.com
-
- #用法提示
- $usage = "Usage: scriptname -p | Directory1 ... | File1 ... | -c MD5File1 MD5File2 | Directory -f MD5File [ > OUTFILE]";
- $usagecompare = "Usage: scriptname -c MD5File1 MD5File2 [ > OUTFILE]";
- $usagepath = "Usage: scriptname -p [ > OUTFILE]";
- #判斷命令行參數是否為空,為空 則直接計算PATH路徑裡面所有的二進制文件的MD5值,不為空
- #如果第一個參數是目錄,那麼調用getDirectoryAllFileMD5 計算目錄裡面所有文件的MD5值
- #如果第一個參數是文件,那麼調用getSingleFileMD5 計算命令行參數裡面所有文件的MD5值
- if (@ARGV) {
- my $arg = $ARGV[0];
- if (-d $arg ) {
- if ($ARGV[1] eq '-f' && -T $ARGV[2]){
- &compareWithLastMD5File($arg);
- } else {
- &getDirectoryAllFileMD5(@ARGV);
- }
- } elsif ( -f $arg ) {
- &getSingleFileMD5(@ARGV);
- } elsif ($arg eq '-c' && @ARGV == 3 ) {
- die "$usagecompare\n" unless (-T $ARGV[1] && -T $ARGV[2]);
- &compareWithTwoMD5Files;
- } elsif ($arg eq '-p') {
- die "$usagepath\n" unless (@ARGV == 1);
- &getPathBinFileMD5;
- } else {
- die "$usage\n";
- }
- } else {
- die "$usage\n";
- }
- #得到目錄下所有文件(包括子目錄)的MD5值
- sub getDirectoryAllFileMD5 {
- find(\&wantedPrint,@_);
- }
- #得到PATH變量裡面所有的二進制文件的MD5值
- sub getPathBinFileMD5 {
- my @path = split /:/,$ENV{PATH};
- find(\&wantedPrint,@path);
- }
- #得到單個文件的MD5值
- sub getSingleFileMD5 {
- foreach (@_) {
- if (-R $_) {
- print "$_ ",&getMD5($_),"\n";
- } else {
- print "Can't read $_\n";
- next;
- }
- }
- }
- #先生成該目錄下所有文件的MD5值,然後跟上一次該目錄的生成的MD5文件對比
- #本次生成的MD5跟上一次生成的MD5文件比對,不一致的輸出出來,同時將3個時間輸出出來
- #如果是新添加的文件則輸出出來其3個時間值,atime mtime ctime
- sub compareWithLastMD5File {
- find(\&wantedHash,@_);
- my $md5file = $ARGV[2];
- open MD5FILE,"<","$md5file"or die "Can't read $md5file : $!\n";
- my $lastMD5Filerecords = (@lastMD5FilerecordsArray = <MD5FILE>);
- my %lastMD5Hash = map { split } @lastMD5FilerecordsArray;
- close MD5FILE;
- foreach (keys %thisMD5Hash) {
- $thisMD5Filerecords++;
- }
- if ($thisMD5Filerecords >= $lastMD5Filerecords) {
- %hash1 = %thisMD5Hash;
- %hash2 = %lastMD5Hash;
- } else {
- %hash1 = %lastMD5Hash;
- %hash2 = %thisMD5Hash;
- }
- foreach (keys %hash1) {
- if (exists $hash2{$_}) {
- if ( $hash1{$_} ne $hash2{$_} ) {
- ($atime,$mtime,$ctime) = &getFileAMCTime($_);
- print "Different:$_ $hash1{$_} Atime:$atime Mtime:$mtime Ctime:$ctime\n";
- }
- } else {
- if (-e $_) {
- ($atime,$mtime,$ctime) = &getFileAMCTime($_);
- print "Added:$_ $hash1{$_} Atime:$atime Mtime:$mtime Ctime:$ctime\n";
- } else {
- print "Deleted:$_ $hash1{$_}\n";
- }
- }
- }
- }
- #比較2個生成的MD5文件(對同一個目錄生成的),找出不同的或者不存在的
- sub compareWithTwoMD5Files {
- my ($md5file1,$md5file2) = ($ARGV[1],$ARGV[2]);
- open MD5FILE1,"<","$md5file1"or die "Can't read $md5file1 : $!\n";
- open MD5FILE2,"<","$md5file2"or die "Can't read $md5file2 : $!\n";
- my $file1record = (@file1record = <MD5FILE1>);
- my $file2record = (@file2record = <MD5FILE2>);
- close MD5FILE1;
- close MD5FILE2;
- my %hashmap1 = map { split } @file1record;
- my %hashmap2 = map { split } @file2record;
- if ($file1record >= $file2record) {
- %hash1 = %hashmap1;
- %hash2 = %hashmap2;
- } else {
- %hash1 = %hashmap2;
- %hash2 = %hashmap1;
- }
- foreach (keys %hash1) {
- if (exists $hash2{$_}) {
- if ( $hash1{$_} ne $hash2{$_} ) {
- ($atime,$mtime,$ctime) = &getFileAMCTime($_);
- print "Different:$_ $hash1{$_} Atime:$atime Mtime:$mtime Ctime:$ctime\n";
- }
- } else {
- if (-e $_) {
- ($atime,$mtime,$ctime) = &getFileAMCTime($_);
- print "Added:$_ $hash1{$_} Atime:$atime Mtime:$mtime Ctime:$ctime\n";
- } else {
- print "Deleted:$_ $hash1{$_}\n";
- }
- }
- }
- }
- #遍歷條件,找到之後輸出
- sub wantedPrint {
- if (-f $_ && -R $_) {
- print "$File::Find::name ",&getMD5($_),"\n";
- }
- }
- #遍歷條件,找到之後形成一個HASH
- sub wantedHash {
- if (-f $_ && -r $_) {
- $thisMD5Hash{$File::Find::name} = &getMD5($_);
- }
- }
- #計算MD5值
- sub getMD5 {
- my $file = shift @_;
- open(FH, $file) or die "Can't open '$file': $!\n";
- binmode(FH);
- my $filemd5 = Digest::MD5->new->addfile(FH)->hexdigest;
- close FH;
- return $filemd5;
- }
- #獲取文件的atime,mtime,ctime
- sub getFileAMCTime {
- $filename = shift @_;
- my ($atime,$mtime ,$ctime) = (stat ($filename))[8,9,10];
- $atime = &getTime($atime);
- $mtime = &getTime($mtime);
- $ctime = &getTime($ctime);
- #將日期時間格式轉換為比較友好的格式
- sub getTime {
- my $time = shift @_;
- my($sec,$min,$hour,$day,$mon,$year) = (localtime $time)[0..5];
- $time = sprintf "%4d-%02d-%02d %2d:%02d:%02d",$year + 1900,$mon + 1,$day,$hour,$min,$sec;
- return $time;
- }
- return $atime,$mtime,$ctime;
- }