使用minizip解压缩char数组

时间:2017-05-22 18:25:25

标签: c++ compression inflate

我有一个向量(它只是一个char数组的包装器),它是输入。 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 "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的包装库,应该& #39;是吗? 如何修改此功能?

3 个答案:

答案 0 :(得分:2)

根据https://users.cs.jmu.edu/buchhofp/forensics/formats/pkzip.html

,它从50 4B 03 04开始,它是一个PKZIP文件

如果这些是zip文件,那么膨胀是错误的功能。 zlib,gzip和zip格式都不同。如果使用正确的函数,可以使用zlib读取zip。如果您没有contrib,可以下载并重建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;
}

答案 1 :(得分:1)

如果我正确理解了这个问题,那么您正在尝试解压缩6390字节的压缩数据。压缩数据是原始的deflate流,没有zlib头或预告片。为此,您需要使用inflateInit2(&stream, -15)而不是inflateInit(&stream)-15请求解压缩原始deflate。

答案 2 :(得分:0)

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

就是这样。