使用 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);

我使用调试器单步执行该方法并监视 retinflate 返回值 -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;
}

Zip-Utils

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。