miniz C 无法使用绝对路径压缩文件

miniz C can't zip file with absolute path

我在 Windows 上使用 miniz 在 C 中创建一个 .zip 文件。

我使用文档生成了我的代码并且它有效。我可以用我想要的文件创建一个存档,前提是我给了 zip 函数的相对路径。

我不明白为什么 "file_name" 变量必须是类似“../test/file.txt”而不是 "C:/../test/file.txt".

if (!(status = mz_zip_add_mem_to_archive_file_in_place(archive, file_name, data, strlen(data) + 1, s_pComment,
                                                           (uint16) strlen(s_pComment), MZ_BEST_COMPRESSION)))
        return (merror("add file to archive failed !!"));

在这个函数之前,我打开我的文件,获取里面的数据并用它调用zip_function。

    if (!(src = fopen(file_name, "r")))
        return (merror("can't open this file"));
    char *line = NULL;
    char *data= NULL;
    size_t n = 0;
    getline(&line, &n, src);
    data= strdup(line);
    while (getline(&line, &n, src) != -1){
        data = realloc(save, sizeof(char) * (strlen(data) + strlen(line)) + 1);
        data = strcat(data, line);
    }
    fopen(src);

所以我用存档名称、文件名(带绝对路径)和其中的数据(以 char * 格式)调用 zip 函数。

这是 "full" 代码:函数 init_zip 是我的程序调用的第一个函数。 arg 参数是我要创建的存档名称(它可以是绝对路径并且有效),args 参数是我要添加到存档文件的不同文件的名称(相对路径有效但不是绝对路径)。

typedef unsigned char uint8;
typedef unsigned short uint16;
typedef unsigned int uint;

static const char *s_pComment = "";

static int isDirectory(const char *path) {
    struct stat statbuf;
    if (stat(path, &statbuf) != 0)
        return 0;
    return S_ISDIR(statbuf.st_mode);
}

int get_data(const char *archive, const char *file)
{
    FILE               *src;

    if (!isDirectory(file)) {
        if (!(src = fopen(file, "r")))
            return (merror("can't open this file"));
        char *line = NULL;
        char *save = NULL;
        size_t n = 0;
        getline(&line, &n, src);
        save = strdup(line);
        while (getline(&line, &n, src) != -1) {
            save = realloc(save, sizeof(char) * (strlen(save) + strlen(line)) + 1);
            save = strcat(save, line);
        }
        printf("compressing %s ..\n", file);
        if (m_compress(archive, file, save))
            return (merror("compress function failed"));
        printf(("\tOK.\n"));
        fclose(src);
    }
    else
    {
        DIR                *dir;
        struct dirent      *entry;
        char                *new_file;

        if (!(dir = opendir(file)))
            return (merror("opendir failed: ", "wrong directory path in init_zip.get_data command : ", file, NULL));
        while ((entry = readdir(dir)) != NULL)
        {
            if (strcmp(entry->d_name, ".") && strcmp(entry->d_name, "..")) {
                new_file = add_path(file, entry->d_name);
                get_data(archive, new_file);
            }
        }
        if (new_file)
            free(new_file);
        closedir(dir);
    }
}

int init_zip(const char *arg, const char **args)
{
    printf("\nZIP cmd:\n >");
    remove(arg);
    for (int counter = 0; args[counter]; ++counter)
    {
       get_data(arg, args[counter]);
    }
    printf("All the files are added to %s archive file.\n", arg);
    return (0);
}

int m_compress(const char *archive, const char *file_name, const char *data)
{
    mz_bool status;

    if (data)
        if (!(status = mz_zip_add_mem_to_archive_file_in_place(archive, file_name, data, strlen(data) + 1, s_pComment,
                                                               (uint16) strlen(s_pComment), MZ_BEST_COMPRESSION)))
            return (merror("add file to archive failed !!"));
    else
        if (!(status = mz_zip_add_mem_to_archive_file_in_place(archive, file_name, NULL, 0, "no comment", (uint16)strlen("no comment"), MZ_BEST_COMPRESSION)))
            return (merror("add directory to archive failed !!"));

    return (0);
}

这是get_data()中使用的add_path()函数:

char          *add_path(const char *str1, const char *str2)
{
  char *path;

  path = malloc(sizeof(char) * (strlen(str1) + 1 + strlen(str2) + 1));
  path = strcpy(path, str1);
  path = strcat(path, "/");
  path = strcat(path, str2);
  return (path);
}

有人知道吗?

如果没有任何帮助,那么您应该查找来源。按照 miniz on Github 中的代码,文件 miniz_zip.c 第 4297 行,我看到:

mz_bool mz_zip_add_mem_to_archive_file_in_place(...

它调用函数 mz_zip_writer_validate_archive_name 来检查第二个文件名,前提是它不能以驱动器号开头(第 3069 行),如果是的话 returns FALSE,错误设置为 MZ_ZIP_INVALID_FILENAME

至于为什么第二个文件名可能不是绝对路径,我不知道。如果它对您很重要,您可以从 Github 获取代码并进行改编。