使用 minizip 解压缩一个字符数组
Using minizip to decompress a char array
我有一个向量(它只是一个字符数组的包装器)作为输入。 pkzip 是使用 c# sharpZipLib 创建的。
我通过签出的十六进制编辑器 zip 模板将数据存储到一个文件中 运行。输入很好,没有格式错误。除了压缩数据之外的所有内容:
50 4B 03 04 14 00 00 00 08 00 51 B2 8B 4A B3 B6
6C B0 F6 18 00 00 40 07 01 00 07 00 00 00 2D 33
31 2F 31 32 38
<compressed data (6390 bytes)>
50 4B 01 02 14 00 14 00 00 00 08 00 51 B2 8B 4A
B3 B6 6C B0 F6 18 00 00 40 07 01 00 07 00 00 00
00 00 00 00 00 00 00 00 00 00 00 00 00 00 2D 33
31 2F 31 32 38 50 4B 05 06 00 00 00 00 01 00 01
00 35 00 00 00 1B 19 00 00 00 00
我有另一个向量作为输出。膨胀的数据大约有 67-68k,所以我知道它适合缓冲区。
对于我来说,我无法让 minizip 将前者膨胀并将其存储到后者中。
这是我目前拥有的:
#include "minizip\zlib.h"
#define ZLIB_WINAPI
std::vector<unsigned char> data;
/*...*/
std::vector<unsigned char> outBuffer(1024 * 1024);
z_stream stream;
stream.zalloc = Z_NULL;
stream.zfree = Z_NULL;
stream.opaque = Z_NULL;
stream.data_type = Z_BINARY;
stream.avail_in = data.size();
stream.avail_out = outBuffer.size();
stream.next_in = &data[0];
stream.next_out = &outBuffer[0];
int ret = inflateInit(&stream);
ret = inflate(&stream, 1);
ret = inflateEnd(&stream);
我使用调试器单步执行该方法并监视 ret
。 inflate
返回值 -3 和消息 "incorrect header check"
。
这是一个 pkzip
,它是 zlib
的包装器,但 minizip
应该是 zlib
的包装器库,应该支持 pkzip
,不应该吗?
我必须如何修改它才能工作?
因为它以 50 4B 03 04 开头,所以它是一个 PKZIP 文件,根据 https://users.cs.jmu.edu/buchhofp/forensics/formats/pkzip.html
如果这些是 zip 文件,则 inflate 函数是错误的。 zlib、gzip 和 zip 格式都是不同的。如果使用正确的函数,则可以使用 zlib 读取 zip。如果您没有贡献,也许下载并重建 zlib。
这是我使用 zlib 库处理 zip 文件的一些旧代码。我可能移动了一些 headers,因为官方 zlib 将它们放在 zlib/contrib/minizip.
下
参数是文件名,所以你必须修改它,或者将你的数组写入文件。
// #include <zlib/unzip.h>
#include <zlib/contrib/minizip/unzip.h>
/// return list of filenames in zip archive
std::list<std::string> GetZipFilenames(const char *szZipArchive){
std::list<std::string> results;
unzFile zip = unzOpen(szZipArchive);
if (zip){
unz_global_info info;
int rv = unzGetGlobalInfo(zip, &info);
if (UNZ_OK == unzGoToFirstFile(zip)){
do {
char szFilename[BUFSIZ];
if (UNZ_OK == unzGetCurrentFileInfo(zip, NULL, szFilename, sizeof(szFilename), NULL, 0, NULL, 0))
results.push_back(std::string(szFilename));
} while (UNZ_OK == unzGoToNextFile(zip));
}
}
return results;
}
/// extract the contents of szFilename inside szZipArchive
bool ExtractZipFileContents(const char *szZipArchive, const char *szFilename, std::string &contents){
bool result = false;
unzFile zip = unzOpen(szZipArchive);
if (zip){
if (UNZ_OK == unzLocateFile(zip, szFilename, 0)){
if (UNZ_OK == unzOpenCurrentFile(zip)){
char buffer[BUFSIZ];
size_t bytes;
while (0 < (bytes = unzReadCurrentFile(zip, buffer, sizeof(buffer)))){
contents += std::string(buffer, bytes);
}
unzCloseCurrentFile(zip);
result = (bytes == 0);
}
}
unzClose(zip);
}
return result;
}
std::vector<unsigned char> inputBuffer;
std::vector<unsigned char> outBuffer(1024 * 1024);
HZIP hz = OpenZip(&inputBuffer[0], inputBuffer.capacity(), 0);
ZIPENTRY ze;
GetZipItem(hz, 0, &ze);
UnzipItem(hz, 0, &outBuffer[0], 1024 * 1024);
outBuffer.resize(ze.unc_size);
如果 inputBuffer
包含 pkzip
文件,此代码段将解压缩它并将内容存储在 outBuffer
。
就这些了。
如果我没理解错的话,你是想解压6390字节的压缩数据。该压缩数据是原始压缩流,没有 zlib 头或尾。为此,您需要使用 inflateInit2(&stream, -MAX_WBITS)
而不是 inflateInit(&stream)
。 -MAX_WBITS
请求解压缩原始 deflate。
我有一个向量(它只是一个字符数组的包装器)作为输入。 pkzip 是使用 c# sharpZipLib 创建的。
我通过签出的十六进制编辑器 zip 模板将数据存储到一个文件中 运行。输入很好,没有格式错误。除了压缩数据之外的所有内容:
50 4B 03 04 14 00 00 00 08 00 51 B2 8B 4A B3 B6
6C B0 F6 18 00 00 40 07 01 00 07 00 00 00 2D 33
31 2F 31 32 38
<compressed data (6390 bytes)>
50 4B 01 02 14 00 14 00 00 00 08 00 51 B2 8B 4A
B3 B6 6C B0 F6 18 00 00 40 07 01 00 07 00 00 00
00 00 00 00 00 00 00 00 00 00 00 00 00 00 2D 33
31 2F 31 32 38 50 4B 05 06 00 00 00 00 01 00 01
00 35 00 00 00 1B 19 00 00 00 00
我有另一个向量作为输出。膨胀的数据大约有 67-68k,所以我知道它适合缓冲区。
对于我来说,我无法让 minizip 将前者膨胀并将其存储到后者中。
这是我目前拥有的:
#include "minizip\zlib.h"
#define ZLIB_WINAPI
std::vector<unsigned char> data;
/*...*/
std::vector<unsigned char> outBuffer(1024 * 1024);
z_stream stream;
stream.zalloc = Z_NULL;
stream.zfree = Z_NULL;
stream.opaque = Z_NULL;
stream.data_type = Z_BINARY;
stream.avail_in = data.size();
stream.avail_out = outBuffer.size();
stream.next_in = &data[0];
stream.next_out = &outBuffer[0];
int ret = inflateInit(&stream);
ret = inflate(&stream, 1);
ret = inflateEnd(&stream);
我使用调试器单步执行该方法并监视 ret
。 inflate
返回值 -3 和消息 "incorrect header check"
。
这是一个 pkzip
,它是 zlib
的包装器,但 minizip
应该是 zlib
的包装器库,应该支持 pkzip
,不应该吗?
我必须如何修改它才能工作?
因为它以 50 4B 03 04 开头,所以它是一个 PKZIP 文件,根据 https://users.cs.jmu.edu/buchhofp/forensics/formats/pkzip.html
如果这些是 zip 文件,则 inflate 函数是错误的。 zlib、gzip 和 zip 格式都是不同的。如果使用正确的函数,则可以使用 zlib 读取 zip。如果您没有贡献,也许下载并重建 zlib。
这是我使用 zlib 库处理 zip 文件的一些旧代码。我可能移动了一些 headers,因为官方 zlib 将它们放在 zlib/contrib/minizip.
下参数是文件名,所以你必须修改它,或者将你的数组写入文件。
// #include <zlib/unzip.h>
#include <zlib/contrib/minizip/unzip.h>
/// return list of filenames in zip archive
std::list<std::string> GetZipFilenames(const char *szZipArchive){
std::list<std::string> results;
unzFile zip = unzOpen(szZipArchive);
if (zip){
unz_global_info info;
int rv = unzGetGlobalInfo(zip, &info);
if (UNZ_OK == unzGoToFirstFile(zip)){
do {
char szFilename[BUFSIZ];
if (UNZ_OK == unzGetCurrentFileInfo(zip, NULL, szFilename, sizeof(szFilename), NULL, 0, NULL, 0))
results.push_back(std::string(szFilename));
} while (UNZ_OK == unzGoToNextFile(zip));
}
}
return results;
}
/// extract the contents of szFilename inside szZipArchive
bool ExtractZipFileContents(const char *szZipArchive, const char *szFilename, std::string &contents){
bool result = false;
unzFile zip = unzOpen(szZipArchive);
if (zip){
if (UNZ_OK == unzLocateFile(zip, szFilename, 0)){
if (UNZ_OK == unzOpenCurrentFile(zip)){
char buffer[BUFSIZ];
size_t bytes;
while (0 < (bytes = unzReadCurrentFile(zip, buffer, sizeof(buffer)))){
contents += std::string(buffer, bytes);
}
unzCloseCurrentFile(zip);
result = (bytes == 0);
}
}
unzClose(zip);
}
return result;
}
std::vector<unsigned char> inputBuffer;
std::vector<unsigned char> outBuffer(1024 * 1024);
HZIP hz = OpenZip(&inputBuffer[0], inputBuffer.capacity(), 0);
ZIPENTRY ze;
GetZipItem(hz, 0, &ze);
UnzipItem(hz, 0, &outBuffer[0], 1024 * 1024);
outBuffer.resize(ze.unc_size);
如果 inputBuffer
包含 pkzip
文件,此代码段将解压缩它并将内容存储在 outBuffer
。
就这些了。
如果我没理解错的话,你是想解压6390字节的压缩数据。该压缩数据是原始压缩流,没有 zlib 头或尾。为此,您需要使用 inflateInit2(&stream, -MAX_WBITS)
而不是 inflateInit(&stream)
。 -MAX_WBITS
请求解压缩原始 deflate。