使用迷你zip解压缩字符数组

Using minizip to decompress a char array

本文关键字:解压缩 字符 数组 zip      更新时间:2023-10-16

我有一个向量(它只是一个字符数组上的包装器(作为输入。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,所以我知道它适合缓冲区。

对于我的生活,我无法让迷你拉链给前者充气并将其存储到后者中。

这是我到目前为止所拥有的:

#include "minizipzlib.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文件,那么膨胀是错误的功能。zlib,gzip和zip格式都是不同的。您可以使用 zlib 读取 zip,如果您使用正确的函数这样做。如果你没有贡献,也许下载并重建 zlib。

这是我的一些旧代码,它使用 zlib 库适用于 zip 文件。我可能已经移动了一些标题,因为官方的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;
}

如果我正确理解了这个问题,您正在尝试解压缩 6390 字节的压缩数据。压缩数据是一个原始的放气流,没有 zlib 标头或尾部。为此,您需要使用 inflateInit2(&stream, -MAX_WBITS) 而不是 inflateInit(&stream) .-MAX_WBITS要求对原始放气进行减压。

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中。

仅此而已。