我在Amazon S3上有许多LZO压缩的日志文件,我想从PHP中读取这些文件。 AWS SDK提供了一个很好的StreamWrapper来有效地读取这些文件,但由于文件是压缩的,我需要在处理之前解压缩内容。
我安装了PHP-LZO extension,允许我做lzo_decompress($data)
,但由于我正在处理流而不是完整的文件内容,我假设我需要使用字符串一LZO一次压缩块。换句话说,我想做类似的事情:
$s3 = S3Client::factory( $myAwsCredentials );
$s3->registerStreamWrapper();
$stream = fopen("s3://my_bucket/my_logfile", 'r');
$compressed_data = '';
while (!feof($stream)) {
$compressed_data .= fread($stream, 1024);
// TODO: determine if we have a full LZO block yet
if (contains_full_lzo_block($compressed_data)) {
// TODO: extract the LZO block
$lzo_block = get_lzo_block($compressed_data);
$input = lzo_decompress( $lzo_block );
// ...... and do stuff to the decompressed input
}
}
fclose($stream);
这两个TODO
是我不确定该怎么做的地方:
由于压缩是由亚马逊(s3distCp)完成的,我无法控制块大小,因此我可能需要检查传入流以确定块的大小 - 这是正确的假设吗?
(理想情况下,我会直接在流上使用自定义StreamFilter,但我以前找不到任何人这样做过)
答案 0 :(得分:1)
通过PHP执行命令可以通过多种方式完成,例如:
$command = 'gunzip -c /path/src /path/dest';
$escapedCommand = escapeshellcmd($command);
system($escapedCommand);
或
shell_exec('gunzip -c /path/src /path/dest');
将完成这项工作。 现在这是执行什么命令的问题,在Linux下有一个很好的命令行工具lzop,它提取或压缩lzop文件。
您可以通过以下方式使用它:
lzop -dN sources.lzo
所以你最终的代码可能就像这样简单:
shell_exec('lzop -dN s3://my_bucket/my_logfile');