1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119
|
#! /usr/bin/perl
#
# Copyright (C) Heinz-Josef Claes (2009-2022)
# hjclaes@web.de
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
#
use Fcntl;
use Digest::MD5 qw(md5_hex);
use POSIX;
use POSIX ":sys_wait_h";
# $0 comprProg fileToRead fileToSave md5RetFile
# ret:
# 0 ok
# 1 cannot open fileToRead
# 2 cannot exec comprProg
# 3 cannot write md5sum
my ($comprProg, $fileToRead, $fileToSave, $md5RetFile, $tmpdir,
@comprProgFlags) = @ARGV;
unless (sysopen(IN, $fileToRead, O_RDONLY))
{
print STDERR "cannot open <$fileToRead> (", __FILE__, " ", __LINE__, ")\n";
exit 1;
}
my $prefix = "$tmpdir/stbu-md5Exec-";
my $suffix;
do
{
$suffix = sprintf '%08x%08x', rand 0xffffffff, rand 0xffffffff;
}
while (-e $prefix . $suffix);
my $stderr = $prefix . $suffix;
local *PARENT;
my $fd = *PARENT;
my $child;
pipe $child, $fd or die "pipe failed: $!";
my $pid = fork();
die "fork() failed: $!" unless defined $pid;
if ($pid == 0) # in the parent
{
close $Child;
}
elsif ($pid < 0)
{
die "failed to fork in $0\n";
}
else # in the child
{
close PARENT;
open(STDERR, ">", $stderr)
or die "cannot open STDERR";
open(STDIN, "<&=" . fileno($child))
or die "cannot open STDIN";
open(STDOUT, ">", $fileToSave)
or die "cannot open STDOUT";
exec($comprProg, @comprProgFlags);
die "couldn't exec $comprProg @comprProgFlags";
}
chmod 0700, $fileToSave;
if (-s $stderr) # file exists and is > 0
{
open(ERR, "<", $stderr)
or print STDERR "cannot open <$stderr> in $0";
my (@err) = <ERR>;
close(ERR);
print STDERR "@err";
}
unlink $stderr;
my $md5 = Digest::MD5->new();
my ($buffer, $n, $size);
$size = 0;
while ($n = sysread(IN, $buffer, 4096))
{
print STDERR "cannot write to <$fileToSave>\n"
unless syswrite $fd, $buffer;
$md5->add($buffer);
$size += $n;
}
close($fd);
close(IN);
waitpid $pid, 0;
unless (open(MD5, '>', $md5RetFile))
{
print STDERR "cannot write to <$md5RetFile>\n";
exit 3;
}
print MD5 $md5->hexdigest(), "\n$size\n";
close(MD5);
exit 0;
|