2014-01-12 14:04:54 -07:00
|
|
|
/* minizip.c
|
2018-10-08 23:48:58 -07:00
|
|
|
Version 2.6.0, October 8, 2018
|
2017-10-04 22:50:20 -07:00
|
|
|
part of the MiniZip project
|
2012-01-21 14:53:44 -07:00
|
|
|
|
2018-01-06 08:49:03 -08:00
|
|
|
Copyright (C) 2010-2018 Nathan Moinvaziri
|
2017-09-16 13:25:02 +08:00
|
|
|
https://github.com/nmoinvaz/minizip
|
2014-01-12 14:04:54 -07:00
|
|
|
Copyright (C) 1998-2010 Gilles Vollant
|
2018-10-17 22:39:01 +00:00
|
|
|
https://www.winimage.com/zLibDll/minizip.html
|
2012-01-21 14:53:44 -07:00
|
|
|
|
2014-01-12 14:04:54 -07:00
|
|
|
This program is distributed under the terms of the same license as zlib.
|
|
|
|
See the accompanying LICENSE file for the full text of the license.
|
2012-01-21 14:53:44 -07:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
2018-08-31 16:43:41 -07:00
|
|
|
#include <stdio.h>
|
2012-01-21 14:53:44 -07:00
|
|
|
#include <string.h>
|
2017-10-23 18:58:56 -07:00
|
|
|
#include <inttypes.h>
|
2012-01-21 14:53:44 -07:00
|
|
|
#include <time.h>
|
|
|
|
|
2017-10-16 07:37:11 -07:00
|
|
|
#include "mz.h"
|
2017-10-05 23:32:57 -07:00
|
|
|
#include "mz_os.h"
|
2017-10-04 22:10:11 -07:00
|
|
|
#include "mz_strm.h"
|
2018-01-02 09:22:38 -08:00
|
|
|
#include "mz_strm_buf.h"
|
2017-10-16 07:37:11 -07:00
|
|
|
#include "mz_strm_split.h"
|
2017-10-05 23:32:57 -07:00
|
|
|
#include "mz_zip.h"
|
2018-08-13 23:07:42 -07:00
|
|
|
#include "mz_zip_rw.h"
|
2017-10-04 20:06:25 -07:00
|
|
|
|
|
|
|
/***************************************************************************/
|
2012-01-21 14:53:44 -07:00
|
|
|
|
2018-10-08 22:41:16 -07:00
|
|
|
typedef struct minizip_opt_s {
|
|
|
|
uint8_t include_path;
|
|
|
|
int16_t compress_level;
|
|
|
|
uint8_t compress_method;
|
|
|
|
uint8_t overwrite;
|
|
|
|
uint8_t append;
|
|
|
|
int64_t disk_size;
|
2018-10-20 10:37:10 -07:00
|
|
|
uint8_t zip_cd;
|
2018-10-25 19:35:50 -07:00
|
|
|
uint8_t legacy_encoding;
|
2018-10-25 19:52:48 -07:00
|
|
|
#ifdef HAVE_AES
|
|
|
|
uint8_t aes;
|
|
|
|
#endif
|
|
|
|
#ifndef MZ_NO_ZIP_SIGNING
|
2018-10-23 09:04:04 -07:00
|
|
|
const char *cert_path;
|
|
|
|
const char *cert_pwd;
|
2018-10-25 19:52:48 -07:00
|
|
|
#endif
|
2018-10-08 22:41:16 -07:00
|
|
|
} minizip_opt;
|
|
|
|
|
|
|
|
/***************************************************************************/
|
|
|
|
|
|
|
|
int32_t minizip_banner(void);
|
|
|
|
int32_t minizip_help(void);
|
|
|
|
|
|
|
|
int32_t minizip_list(const char *path);
|
|
|
|
|
|
|
|
int32_t minizip_add_entry_cb(void *handle, void *userdata, mz_zip_file *file_info);
|
2018-10-08 23:31:21 -07:00
|
|
|
int32_t minizip_add_progress_cb(void *handle, void *userdata, mz_zip_file *file_info, int64_t position);
|
2018-10-08 22:41:16 -07:00
|
|
|
int32_t minizip_add_overwrite_cb(void *handle, void *userdata, const char *path);
|
|
|
|
int32_t minizip_add(const char *path, const char *password, minizip_opt *options, int32_t arg_count, const char **args);
|
|
|
|
|
|
|
|
int32_t minizip_extract_entry_cb(void *handle, void *userdata, mz_zip_file *file_info, const char *path);
|
2018-10-08 23:31:21 -07:00
|
|
|
int32_t minizip_extract_progress_cb(void *handle, void *userdata, mz_zip_file *file_info, int64_t position);
|
2018-10-08 22:41:16 -07:00
|
|
|
int32_t minizip_extract_overwrite_cb(void *handle, void *userdata, mz_zip_file *file_info, const char *path);
|
|
|
|
int32_t minizip_extract(const char *path, const char *pattern, const char *destination, const char *password, minizip_opt *options);
|
|
|
|
|
|
|
|
int32_t minizip_erase(const char *src_path, const char *target_path, int32_t arg_count, const char **args);
|
|
|
|
|
|
|
|
/***************************************************************************/
|
|
|
|
|
|
|
|
int32_t minizip_banner(void)
|
2012-01-21 14:53:44 -07:00
|
|
|
{
|
2017-10-16 20:15:35 -07:00
|
|
|
printf("Minizip %s - https://github.com/nmoinvaz/minizip\n", MZ_VERSION);
|
2017-10-05 18:45:23 -07:00
|
|
|
printf("---------------------------------------------------\n");
|
2018-10-08 22:41:16 -07:00
|
|
|
return MZ_OK;
|
2012-10-25 00:21:50 -07:00
|
|
|
}
|
2012-01-21 14:53:44 -07:00
|
|
|
|
2018-10-08 22:41:16 -07:00
|
|
|
int32_t minizip_help(void)
|
2012-10-25 00:21:50 -07:00
|
|
|
{
|
2018-08-27 08:30:57 -07:00
|
|
|
printf("Usage : minizip [-x -d dir|-l|-e] [-o] [-c] [-a] [-j] [-0 to -9] [-b|-m] [-k 512] [-p pwd] [-s] file.zip [files]\n\n" \
|
2017-10-18 16:51:10 -07:00
|
|
|
" -x Extract files\n" \
|
|
|
|
" -l List files\n" \
|
|
|
|
" -d Destination directory\n" \
|
|
|
|
" -o Overwrite existing files\n" \
|
2018-08-27 08:30:57 -07:00
|
|
|
" -c File names use cp437 encoding\n" \
|
2017-10-18 16:51:10 -07:00
|
|
|
" -a Append to existing zip file\n" \
|
2018-05-01 13:45:08 -07:00
|
|
|
" -i Include full path of files\n" \
|
2017-10-18 16:51:10 -07:00
|
|
|
" -0 Store only\n" \
|
|
|
|
" -1 Compress faster\n" \
|
|
|
|
" -9 Compress better\n" \
|
|
|
|
" -k Disk size in KB\n" \
|
2018-10-20 10:37:10 -07:00
|
|
|
" -z Zip central directory" \
|
2017-10-18 16:51:10 -07:00
|
|
|
" -p Encryption password\n");
|
|
|
|
#ifdef HAVE_AES
|
2018-10-23 09:04:04 -07:00
|
|
|
printf(" -s AES encryption\n" \
|
|
|
|
" -h Certificate path\n" \
|
2018-10-25 19:35:50 -07:00
|
|
|
" -w Certificate password\n");
|
2017-10-18 16:51:10 -07:00
|
|
|
#endif
|
2017-10-05 21:02:36 -07:00
|
|
|
#ifdef HAVE_BZIP2
|
|
|
|
printf(" -b BZIP2 compression\n");
|
|
|
|
#endif
|
|
|
|
#ifdef HAVE_LZMA
|
|
|
|
printf(" -m LZMA compression\n");
|
|
|
|
#endif
|
2017-10-18 16:51:10 -07:00
|
|
|
printf("\n");
|
2018-10-08 22:41:16 -07:00
|
|
|
return MZ_OK;
|
2012-01-21 14:53:44 -07:00
|
|
|
}
|
|
|
|
|
2017-10-04 20:06:25 -07:00
|
|
|
/***************************************************************************/
|
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
int32_t minizip_list(const char *path)
|
2017-10-18 16:51:10 -07:00
|
|
|
{
|
|
|
|
mz_zip_file *file_info = NULL;
|
|
|
|
uint32_t ratio = 0;
|
|
|
|
int16_t level = 0;
|
2017-10-20 07:59:39 -07:00
|
|
|
int32_t err = MZ_OK;
|
2018-05-01 13:45:08 -07:00
|
|
|
struct tm tmu_date;
|
2017-10-18 16:51:10 -07:00
|
|
|
const char *string_method = NULL;
|
|
|
|
char crypt = ' ';
|
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
void *reader = NULL;
|
2017-10-18 16:51:10 -07:00
|
|
|
|
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
mz_zip_reader_create(&reader);
|
|
|
|
err = mz_zip_reader_open_file(reader, path);
|
2018-10-20 10:37:10 -07:00
|
|
|
if (err == MZ_OK)
|
|
|
|
err = mz_zip_reader_unzip_cd(reader);
|
2018-08-13 23:07:42 -07:00
|
|
|
if (err != MZ_OK)
|
|
|
|
{
|
|
|
|
printf("Error %d opening zip file %s\n", err, path);
|
|
|
|
mz_zip_reader_delete(&reader);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2018-08-14 09:20:17 -07:00
|
|
|
err = mz_zip_reader_goto_first_entry(reader);
|
2018-09-04 15:06:53 +00:00
|
|
|
|
2017-10-18 16:51:10 -07:00
|
|
|
if (err != MZ_OK && err != MZ_END_OF_LIST)
|
|
|
|
{
|
|
|
|
printf("Error %d going to first entry in zip file\n", err);
|
2018-08-13 23:07:42 -07:00
|
|
|
mz_zip_reader_delete(&reader);
|
2017-10-26 08:07:16 -07:00
|
|
|
return err;
|
2017-10-18 16:51:10 -07:00
|
|
|
}
|
2017-10-26 08:07:16 -07:00
|
|
|
|
2018-05-01 21:54:15 -07:00
|
|
|
printf(" Length Method Size Attribs Ratio Date Time CRC-32 Name\n");
|
|
|
|
printf(" ------ ------ ---- ------- ----- ---- ---- ------ ----\n");
|
2017-10-18 16:51:10 -07:00
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
do
|
2017-10-18 16:51:10 -07:00
|
|
|
{
|
2018-08-14 14:00:16 -07:00
|
|
|
err = mz_zip_reader_entry_get_info(reader, &file_info);
|
2017-10-18 16:51:10 -07:00
|
|
|
|
|
|
|
if (err != MZ_OK)
|
|
|
|
{
|
|
|
|
printf("Error %d getting entry info in zip file\n", err);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2018-05-02 09:54:44 -07:00
|
|
|
ratio = 0;
|
2017-10-18 16:51:10 -07:00
|
|
|
if (file_info->uncompressed_size > 0)
|
|
|
|
ratio = (uint32_t)((file_info->compressed_size * 100) / file_info->uncompressed_size);
|
2018-04-24 10:02:39 +00:00
|
|
|
|
2017-10-18 16:51:10 -07:00
|
|
|
// Display a '*' if the file is encrypted
|
|
|
|
if (file_info->flag & MZ_ZIP_FLAG_ENCRYPTED)
|
|
|
|
crypt = '*';
|
|
|
|
|
|
|
|
switch (file_info->compression_method)
|
|
|
|
{
|
2018-08-08 09:43:14 -07:00
|
|
|
case MZ_COMPRESS_METHOD_STORE:
|
2018-04-24 10:02:39 +00:00
|
|
|
string_method = "Stored";
|
2017-10-18 16:51:10 -07:00
|
|
|
break;
|
|
|
|
case MZ_COMPRESS_METHOD_DEFLATE:
|
|
|
|
level = (int16_t)((file_info->flag & 0x6) / 2);
|
|
|
|
if (level == 0)
|
|
|
|
string_method = "Defl:N";
|
|
|
|
else if (level == 1)
|
|
|
|
string_method = "Defl:X";
|
|
|
|
else if ((level == 2) || (level == 3))
|
|
|
|
string_method = "Defl:F"; // 2: fast , 3: extra fast
|
|
|
|
else
|
|
|
|
string_method = "Defl:?";
|
|
|
|
break;
|
|
|
|
case MZ_COMPRESS_METHOD_BZIP2:
|
|
|
|
string_method = "BZip2";
|
|
|
|
break;
|
|
|
|
case MZ_COMPRESS_METHOD_LZMA:
|
|
|
|
string_method = "LZMA";
|
|
|
|
break;
|
|
|
|
default:
|
2018-09-11 08:18:59 -07:00
|
|
|
string_method = "?";
|
2017-10-18 16:51:10 -07:00
|
|
|
}
|
|
|
|
|
2018-05-02 09:54:44 -07:00
|
|
|
mz_zip_time_t_to_tm(file_info->modified_date, &tmu_date);
|
|
|
|
|
2018-05-01 21:54:15 -07:00
|
|
|
printf(" %7"PRIu64" %6s%c %7"PRIu64" %8"PRIx32" %3"PRIu32"%% %2.2"PRIu32"-%2.2"PRIu32\
|
2018-05-02 13:04:41 -07:00
|
|
|
"-%2.2"PRIu32" %2.2"PRIu32":%2.2"PRIu32" %8.8"PRIx32" %s\n",
|
2018-05-02 19:59:38 +00:00
|
|
|
file_info->uncompressed_size, string_method, crypt,
|
2018-05-01 21:54:15 -07:00
|
|
|
file_info->compressed_size, file_info->external_fa, ratio,
|
2017-10-23 18:58:56 -07:00
|
|
|
(uint32_t)tmu_date.tm_mon + 1, (uint32_t)tmu_date.tm_mday,
|
|
|
|
(uint32_t)tmu_date.tm_year % 100,
|
|
|
|
(uint32_t)tmu_date.tm_hour, (uint32_t)tmu_date.tm_min,
|
2018-05-02 13:04:41 -07:00
|
|
|
file_info->crc, file_info->filename);
|
2017-10-18 16:51:10 -07:00
|
|
|
|
2018-08-14 09:20:17 -07:00
|
|
|
err = mz_zip_reader_goto_next_entry(reader);
|
2017-10-23 18:58:56 -07:00
|
|
|
|
2017-10-26 08:07:16 -07:00
|
|
|
if (err != MZ_OK && err != MZ_END_OF_LIST)
|
|
|
|
{
|
2017-10-23 18:58:56 -07:00
|
|
|
printf("Error %d going to next entry in zip file\n", err);
|
2018-08-13 23:29:11 -07:00
|
|
|
break;
|
2017-10-26 08:07:16 -07:00
|
|
|
}
|
2017-10-18 16:51:10 -07:00
|
|
|
}
|
2018-08-13 23:07:42 -07:00
|
|
|
while (err == MZ_OK);
|
|
|
|
|
|
|
|
mz_zip_reader_delete(&reader);
|
2017-10-18 16:51:10 -07:00
|
|
|
|
|
|
|
if (err == MZ_END_OF_LIST)
|
|
|
|
return MZ_OK;
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
/***************************************************************************/
|
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
int32_t minizip_add_entry_cb(void *handle, void *userdata, mz_zip_file *file_info)
|
2017-10-18 16:51:10 -07:00
|
|
|
{
|
2018-10-08 22:41:16 -07:00
|
|
|
MZ_UNUSED(handle);
|
|
|
|
MZ_UNUSED(userdata);
|
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
printf("Adding %s\n", file_info->filename);
|
|
|
|
return MZ_OK;
|
|
|
|
}
|
2017-10-18 16:51:10 -07:00
|
|
|
|
2018-10-08 23:31:21 -07:00
|
|
|
int32_t minizip_add_progress_cb(void *handle, void *userdata, mz_zip_file *file_info, int64_t position)
|
2018-08-13 23:07:42 -07:00
|
|
|
{
|
|
|
|
double progress = 0;
|
|
|
|
uint8_t raw = 0;
|
2017-10-18 16:51:10 -07:00
|
|
|
|
2018-10-08 22:41:16 -07:00
|
|
|
MZ_UNUSED(userdata);
|
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
mz_zip_writer_get_raw(handle, &raw);
|
2018-05-02 19:59:38 +00:00
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
if (raw && file_info->compressed_size > 0)
|
|
|
|
progress = ((double)position / file_info->compressed_size) * 100;
|
|
|
|
else if (!raw && file_info->uncompressed_size > 0)
|
|
|
|
progress = ((double)position / file_info->uncompressed_size) * 100;
|
2017-10-18 16:51:10 -07:00
|
|
|
|
2018-09-17 07:41:59 -07:00
|
|
|
printf("%s - %"PRIu64" / %"PRIu64" (%.02f%%)\n", file_info->filename, position, file_info->uncompressed_size, progress);
|
2018-08-13 23:07:42 -07:00
|
|
|
return MZ_OK;
|
|
|
|
}
|
2018-10-08 22:41:16 -07:00
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
int32_t minizip_add_overwrite_cb(void *handle, void *userdata, const char *path)
|
|
|
|
{
|
|
|
|
minizip_opt *options = (minizip_opt *)userdata;
|
2017-10-18 16:51:10 -07:00
|
|
|
|
2018-10-08 22:41:16 -07:00
|
|
|
MZ_UNUSED(handle);
|
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
if (options->overwrite == 0)
|
2017-10-18 16:51:10 -07:00
|
|
|
{
|
2018-08-13 23:07:42 -07:00
|
|
|
// If ask the user what to do because append and overwrite args not set
|
2017-10-18 16:51:10 -07:00
|
|
|
char rep = 0;
|
|
|
|
do
|
|
|
|
{
|
|
|
|
char answer[128];
|
2018-08-13 23:07:42 -07:00
|
|
|
printf("The file %s exists. Overwrite ? [y]es, [n]o, [a]ppend : ", path);
|
2017-10-18 16:51:10 -07:00
|
|
|
if (scanf("%1s", answer) != 1)
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
rep = answer[0];
|
2018-08-13 23:07:42 -07:00
|
|
|
|
2017-10-18 16:51:10 -07:00
|
|
|
if ((rep >= 'a') && (rep <= 'z'))
|
|
|
|
rep -= 0x20;
|
|
|
|
}
|
|
|
|
while ((rep != 'Y') && (rep != 'N') && (rep != 'A'));
|
|
|
|
|
|
|
|
if (rep == 'A')
|
2018-08-13 23:07:42 -07:00
|
|
|
{
|
|
|
|
return MZ_EXIST_ERROR;
|
|
|
|
}
|
|
|
|
else if (rep == 'N')
|
|
|
|
{
|
|
|
|
return MZ_INTERNAL_ERROR;
|
|
|
|
}
|
2017-10-18 16:51:10 -07:00
|
|
|
}
|
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
return MZ_OK;
|
|
|
|
}
|
2017-10-18 16:51:10 -07:00
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
int32_t minizip_add(const char *path, const char *password, minizip_opt *options, int32_t arg_count, const char **args)
|
|
|
|
{
|
|
|
|
void *writer = NULL;
|
|
|
|
int32_t err = MZ_OK;
|
|
|
|
int32_t err_close = MZ_OK;
|
|
|
|
int32_t i = 0;
|
2018-10-20 10:37:10 -07:00
|
|
|
int32_t flags = 0;
|
2018-08-13 23:07:42 -07:00
|
|
|
const char *filename_in_zip = NULL;
|
2017-10-18 16:51:10 -07:00
|
|
|
|
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
printf("Archive %s\n", path);
|
|
|
|
|
|
|
|
mz_zip_writer_create(&writer);
|
|
|
|
mz_zip_writer_set_password(writer, password);
|
|
|
|
mz_zip_writer_set_compress_method(writer, options->compress_method);
|
|
|
|
mz_zip_writer_set_compress_level(writer, options->compress_level);
|
|
|
|
mz_zip_writer_set_overwrite_cb(writer, options, minizip_add_overwrite_cb);
|
|
|
|
mz_zip_writer_set_progress_cb(writer, options, minizip_add_progress_cb);
|
2018-10-20 10:37:10 -07:00
|
|
|
if (options->zip_cd)
|
|
|
|
mz_zip_writer_set_flags(writer, MZ_ZIP_FLAG_MASK_LOCAL_INFO);
|
2018-10-23 09:04:04 -07:00
|
|
|
if (options->cert_path != NULL)
|
2018-10-25 19:35:50 -07:00
|
|
|
mz_zip_writer_set_certificate(writer, options->cert_path, options->cert_pwd);
|
2018-08-13 23:07:42 -07:00
|
|
|
|
|
|
|
err = mz_zip_writer_open_file(writer, path, options->disk_size, options->append);
|
2017-10-18 16:51:10 -07:00
|
|
|
|
2018-07-08 18:52:41 -07:00
|
|
|
if (err == MZ_OK)
|
2017-10-18 16:51:10 -07:00
|
|
|
{
|
2018-08-13 23:07:42 -07:00
|
|
|
for (i = 0; i < arg_count; i += 1)
|
2017-10-18 16:51:10 -07:00
|
|
|
{
|
2018-08-13 23:07:42 -07:00
|
|
|
filename_in_zip = args[i];
|
2017-10-18 16:51:10 -07:00
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
err = mz_zip_writer_add_path(writer, filename_in_zip, NULL, options->include_path, 1);
|
|
|
|
if (err != MZ_OK)
|
|
|
|
printf("Error %d adding path to zip %s\n", err, filename_in_zip);
|
|
|
|
}
|
2017-10-18 16:51:10 -07:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-08-13 23:07:42 -07:00
|
|
|
printf("Error %d opening zip for writing\n", err);
|
2017-10-18 16:51:10 -07:00
|
|
|
}
|
2018-09-04 15:06:53 +00:00
|
|
|
|
2018-10-20 10:37:10 -07:00
|
|
|
if (options->zip_cd)
|
|
|
|
{
|
|
|
|
if (password != NULL)
|
|
|
|
flags = MZ_ZIP_FLAG_ENCRYPTED;
|
|
|
|
mz_zip_writer_zip_cd(writer, options->compress_method, flags);
|
|
|
|
}
|
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
err_close = mz_zip_writer_close(writer);
|
2017-10-18 16:51:10 -07:00
|
|
|
if (err_close != MZ_OK)
|
|
|
|
{
|
2018-08-13 23:07:42 -07:00
|
|
|
printf("Error %d closing zip for writing %s\n", err_close, path);
|
2017-10-18 16:51:10 -07:00
|
|
|
err = err_close;
|
|
|
|
}
|
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
mz_zip_writer_delete(&writer);
|
2017-10-18 16:51:10 -07:00
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
/***************************************************************************/
|
2017-10-18 16:51:10 -07:00
|
|
|
|
2018-08-14 14:00:16 -07:00
|
|
|
int32_t minizip_extract_entry_cb(void *handle, void *userdata, mz_zip_file *file_info, const char *path)
|
2018-08-13 23:07:42 -07:00
|
|
|
{
|
2018-10-08 22:41:16 -07:00
|
|
|
MZ_UNUSED(handle);
|
|
|
|
MZ_UNUSED(userdata);
|
|
|
|
MZ_UNUSED(path);
|
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
printf("Extracting %s\n", file_info->filename);
|
|
|
|
return MZ_OK;
|
|
|
|
}
|
2017-10-18 16:51:10 -07:00
|
|
|
|
2018-10-08 23:31:21 -07:00
|
|
|
int32_t minizip_extract_progress_cb(void *handle, void *userdata, mz_zip_file *file_info, int64_t position)
|
2018-08-13 23:07:42 -07:00
|
|
|
{
|
|
|
|
double progress = 0;
|
|
|
|
uint8_t raw = 0;
|
2018-09-04 15:06:53 +00:00
|
|
|
|
2018-10-08 22:41:16 -07:00
|
|
|
MZ_UNUSED(userdata);
|
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
mz_zip_reader_get_raw(handle, &raw);
|
2017-10-18 16:51:10 -07:00
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
if (raw && file_info->compressed_size > 0)
|
|
|
|
progress = ((double)position / file_info->compressed_size) * 100;
|
|
|
|
else if (!raw && file_info->uncompressed_size > 0)
|
|
|
|
progress = ((double)position / file_info->uncompressed_size) * 100;
|
2017-10-18 16:51:10 -07:00
|
|
|
|
2018-09-17 07:41:59 -07:00
|
|
|
printf("%s - %"PRIu64" / %"PRIu64" (%.02f%%)\n", file_info->filename, position, file_info->uncompressed_size, progress);
|
2018-08-13 23:07:42 -07:00
|
|
|
return MZ_OK;
|
2017-10-18 16:51:10 -07:00
|
|
|
}
|
2018-10-08 22:41:16 -07:00
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
int32_t minizip_extract_overwrite_cb(void *handle, void *userdata, mz_zip_file *file_info, const char *path)
|
2017-10-18 16:51:10 -07:00
|
|
|
{
|
2018-08-13 23:07:42 -07:00
|
|
|
minizip_opt *options = (minizip_opt *)userdata;
|
2017-10-18 16:51:10 -07:00
|
|
|
|
2018-10-08 22:41:16 -07:00
|
|
|
MZ_UNUSED(handle);
|
|
|
|
MZ_UNUSED(file_info);
|
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
if (options->overwrite == 0)
|
2017-10-18 16:51:10 -07:00
|
|
|
{
|
2018-08-13 23:07:42 -07:00
|
|
|
char rep = 0;
|
|
|
|
do
|
|
|
|
{
|
|
|
|
char answer[128];
|
|
|
|
printf("The file %s exists. Overwrite ? [y]es, [n]o, [A]ll: ", path);
|
|
|
|
if (scanf("%1s", answer) != 1)
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
rep = answer[0];
|
|
|
|
if ((rep >= 'a') && (rep <= 'z'))
|
|
|
|
rep -= 0x20;
|
|
|
|
} while ((rep != 'Y') && (rep != 'N') && (rep != 'A'));
|
|
|
|
|
|
|
|
if (rep == 'N')
|
|
|
|
return MZ_EXIST_ERROR;
|
|
|
|
if (rep == 'A')
|
|
|
|
options->overwrite = 1;
|
2017-10-18 16:51:10 -07:00
|
|
|
}
|
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
return MZ_OK;
|
2017-10-18 16:51:10 -07:00
|
|
|
}
|
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
int32_t minizip_extract(const char *path, const char *pattern, const char *destination, const char *password, minizip_opt *options)
|
2018-07-31 10:20:21 -07:00
|
|
|
{
|
2018-08-13 23:07:42 -07:00
|
|
|
void *reader = NULL;
|
2018-07-31 10:20:21 -07:00
|
|
|
int32_t err = MZ_OK;
|
2018-08-13 23:07:42 -07:00
|
|
|
int32_t err_close = MZ_OK;
|
2018-07-31 10:20:21 -07:00
|
|
|
|
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
printf("Archive %s\n", path);
|
2018-07-31 10:20:21 -07:00
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
mz_zip_reader_create(&reader);
|
|
|
|
mz_zip_reader_set_pattern(reader, pattern, 1);
|
|
|
|
mz_zip_reader_set_password(reader, password);
|
2018-08-27 08:30:57 -07:00
|
|
|
mz_zip_reader_set_legacy_encoding(reader, options->legacy_encoding);
|
2018-08-13 23:07:42 -07:00
|
|
|
mz_zip_reader_set_entry_cb(reader, options, minizip_extract_entry_cb);
|
|
|
|
mz_zip_reader_set_progress_cb(reader, options, minizip_extract_progress_cb);
|
|
|
|
mz_zip_reader_set_overwrite_cb(reader, options, minizip_extract_overwrite_cb);
|
2018-07-31 10:20:21 -07:00
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
err = mz_zip_reader_open_file(reader, path);
|
2018-07-31 10:20:21 -07:00
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
if (err != MZ_OK)
|
|
|
|
{
|
|
|
|
printf("Error %d opening zip file %s\n", err, path);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-10-20 10:37:10 -07:00
|
|
|
err = mz_zip_reader_unzip_cd(reader);
|
|
|
|
if (err == MZ_OK)
|
|
|
|
err = mz_zip_reader_save_all(reader, destination);
|
2018-08-13 23:07:42 -07:00
|
|
|
if (err == MZ_END_OF_LIST && pattern != NULL)
|
|
|
|
printf("Files matching %s not found in zip file\n", pattern);
|
|
|
|
if (err != MZ_OK)
|
|
|
|
printf("Error %d saving zip entries to disk %s\n", err, path);
|
|
|
|
}
|
2018-10-20 10:37:10 -07:00
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
err_close = mz_zip_reader_close(reader);
|
|
|
|
if (err_close != MZ_OK)
|
|
|
|
{
|
|
|
|
printf("Error %d closing zip for reading\n", err_close);
|
|
|
|
err = err_close;
|
2018-07-31 10:20:21 -07:00
|
|
|
}
|
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
mz_zip_reader_delete(&reader);
|
2018-07-31 10:20:21 -07:00
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
/***************************************************************************/
|
|
|
|
|
|
|
|
int32_t minizip_erase(const char *src_path, const char *target_path, int32_t arg_count, const char **args)
|
2018-07-31 10:20:21 -07:00
|
|
|
{
|
|
|
|
mz_zip_file *file_info = NULL;
|
|
|
|
const char *filename_in_zip = NULL;
|
2018-08-13 23:07:42 -07:00
|
|
|
void *reader = NULL;
|
|
|
|
void *writer = NULL;
|
2018-07-31 10:20:21 -07:00
|
|
|
int32_t err = MZ_OK;
|
|
|
|
int32_t i = 0;
|
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
mz_zip_reader_create(&reader);
|
|
|
|
mz_zip_writer_create(&writer);
|
2018-07-31 10:20:21 -07:00
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
err = mz_zip_reader_open_file(reader, src_path);
|
2018-10-20 10:37:10 -07:00
|
|
|
if (err == MZ_OK)
|
|
|
|
err = mz_zip_reader_unzip_cd(reader);
|
2018-07-31 10:20:21 -07:00
|
|
|
if (err != MZ_OK)
|
2018-08-13 23:07:42 -07:00
|
|
|
{
|
|
|
|
printf("Error %d opening zip for reading %s\n", err, src_path);
|
|
|
|
mz_zip_reader_delete(&reader);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = mz_zip_writer_open_file(writer, target_path, 0, 0);
|
|
|
|
if (err != MZ_OK)
|
|
|
|
{
|
|
|
|
printf("Error %d opening zip for writing %s\n", err, target_path);
|
|
|
|
mz_zip_reader_delete(&reader);
|
|
|
|
mz_zip_writer_delete(&writer);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2018-08-14 09:20:17 -07:00
|
|
|
err = mz_zip_reader_goto_first_entry(reader);
|
2018-08-13 23:07:42 -07:00
|
|
|
|
|
|
|
if (err != MZ_OK && err != MZ_END_OF_LIST)
|
2018-07-31 10:20:21 -07:00
|
|
|
printf("Error %d going to first entry in zip file\n", err);
|
|
|
|
|
|
|
|
while (err == MZ_OK)
|
|
|
|
{
|
2018-08-14 14:00:16 -07:00
|
|
|
err = mz_zip_reader_entry_get_info(reader, &file_info);
|
2018-07-31 10:20:21 -07:00
|
|
|
if (err != MZ_OK)
|
|
|
|
{
|
2018-08-13 23:07:42 -07:00
|
|
|
printf("Error %d getting info from zip\n", err);
|
2018-07-31 10:20:21 -07:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < arg_count; i += 1)
|
|
|
|
{
|
|
|
|
filename_in_zip = args[i];
|
|
|
|
|
2018-07-31 12:48:31 -07:00
|
|
|
if (mz_path_compare_wc(file_info->filename, filename_in_zip, 1) != MZ_OK)
|
2018-07-31 10:20:21 -07:00
|
|
|
{
|
|
|
|
printf("Copying %s\n", file_info->filename);
|
2018-08-13 23:07:42 -07:00
|
|
|
err = mz_zip_writer_copy_from_reader(writer, reader);
|
2018-07-31 10:20:21 -07:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
printf("Skipping %s\n", file_info->filename);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (err != MZ_OK)
|
2018-08-13 23:07:42 -07:00
|
|
|
{
|
|
|
|
printf("Error %d copying entry into new zip\n", err);
|
2018-07-31 10:20:21 -07:00
|
|
|
break;
|
2018-08-13 23:07:42 -07:00
|
|
|
}
|
2018-07-31 10:20:21 -07:00
|
|
|
|
2018-08-14 09:20:17 -07:00
|
|
|
err = mz_zip_reader_goto_next_entry(reader);
|
2018-07-31 10:20:21 -07:00
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
if (err != MZ_OK && err != MZ_END_OF_LIST)
|
2018-07-31 10:20:21 -07:00
|
|
|
printf("Error %d going to next entry in zip file\n", err);
|
|
|
|
}
|
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
mz_zip_reader_close(reader);
|
|
|
|
mz_zip_reader_delete(&reader);
|
|
|
|
|
|
|
|
mz_zip_writer_close(writer);
|
|
|
|
mz_zip_writer_delete(&writer);
|
|
|
|
|
|
|
|
if (err == MZ_END_OF_LIST)
|
|
|
|
return MZ_OK;
|
|
|
|
|
2018-07-31 10:20:21 -07:00
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
/***************************************************************************/
|
|
|
|
|
2018-07-31 14:43:30 -07:00
|
|
|
#if !defined(MZ_NO_MAIN)
|
2018-07-31 11:55:32 -07:00
|
|
|
int main(int argc, const char *argv[])
|
2012-01-21 14:53:44 -07:00
|
|
|
{
|
2018-07-31 10:20:21 -07:00
|
|
|
minizip_opt options;
|
2017-10-04 20:06:25 -07:00
|
|
|
int32_t path_arg = 0;
|
2018-07-31 10:20:21 -07:00
|
|
|
int32_t err = 0;
|
|
|
|
int32_t i = 0;
|
2017-10-18 16:51:10 -07:00
|
|
|
uint8_t do_list = 0;
|
|
|
|
uint8_t do_extract = 0;
|
2018-07-31 10:20:21 -07:00
|
|
|
uint8_t do_erase = 0;
|
|
|
|
char bak_path[256];
|
|
|
|
char tmp_path[256];
|
2018-07-31 11:55:32 -07:00
|
|
|
const char *path = NULL;
|
|
|
|
const char *password = NULL;
|
|
|
|
const char *destination = NULL;
|
|
|
|
const char *filename_to_extract = NULL;
|
2018-07-31 10:20:21 -07:00
|
|
|
|
2017-10-04 20:06:25 -07:00
|
|
|
|
2017-03-18 16:16:06 -07:00
|
|
|
minizip_banner();
|
2012-10-25 00:21:50 -07:00
|
|
|
if (argc == 1)
|
2012-01-21 14:53:44 -07:00
|
|
|
{
|
2017-03-18 16:16:06 -07:00
|
|
|
minizip_help();
|
2012-01-21 14:53:44 -07:00
|
|
|
return 0;
|
|
|
|
}
|
2018-10-25 19:35:50 -07:00
|
|
|
|
2017-10-16 14:50:31 -07:00
|
|
|
memset(&options, 0, sizeof(options));
|
2017-10-05 18:45:23 -07:00
|
|
|
|
2017-10-18 16:51:10 -07:00
|
|
|
options.compress_method = MZ_COMPRESS_METHOD_DEFLATE;
|
|
|
|
options.compress_level = MZ_COMPRESS_LEVEL_DEFAULT;
|
2017-10-05 18:45:23 -07:00
|
|
|
|
2017-10-04 20:06:25 -07:00
|
|
|
// Parse command line options
|
2017-10-23 17:22:36 -07:00
|
|
|
for (i = 1; i < argc; i += 1)
|
2012-01-21 14:53:44 -07:00
|
|
|
{
|
2018-07-08 18:52:41 -07:00
|
|
|
if (argv[i][0] == '-')
|
2012-01-21 14:53:44 -07:00
|
|
|
{
|
2018-07-08 18:52:41 -07:00
|
|
|
char c = argv[i][1];
|
|
|
|
if ((c == 'l') || (c == 'L'))
|
|
|
|
do_list = 1;
|
2018-08-27 08:30:57 -07:00
|
|
|
else if ((c == 'x') || (c == 'X'))
|
2018-07-08 18:52:41 -07:00
|
|
|
do_extract = 1;
|
2018-08-27 08:30:57 -07:00
|
|
|
else if ((c == 'e') || (c == 'E'))
|
2018-07-31 10:20:21 -07:00
|
|
|
do_erase = 1;
|
2018-08-27 08:30:57 -07:00
|
|
|
else if ((c == 'a') || (c == 'A'))
|
2018-08-13 23:07:42 -07:00
|
|
|
options.append = 1;
|
2018-08-27 08:30:57 -07:00
|
|
|
else if ((c == 'o') || (c == 'O'))
|
2018-07-08 18:52:41 -07:00
|
|
|
options.overwrite = 1;
|
2018-08-27 08:30:57 -07:00
|
|
|
else if ((c == 'i') || (c == 'I'))
|
2018-07-08 18:52:41 -07:00
|
|
|
options.include_path = 1;
|
2018-10-20 10:37:10 -07:00
|
|
|
else if ((c == 'z') || (c == 'Z'))
|
|
|
|
options.zip_cd = 1;
|
2018-08-27 08:30:57 -07:00
|
|
|
else if ((c >= '0') && (c <= '9'))
|
2012-01-21 14:53:44 -07:00
|
|
|
{
|
2018-07-08 18:52:41 -07:00
|
|
|
options.compress_level = (c - '0');
|
|
|
|
if (options.compress_level == 0)
|
2018-08-08 09:43:14 -07:00
|
|
|
options.compress_method = MZ_COMPRESS_METHOD_STORE;
|
2018-07-08 18:52:41 -07:00
|
|
|
}
|
2017-10-18 16:51:10 -07:00
|
|
|
|
2017-10-05 21:02:36 -07:00
|
|
|
#ifdef HAVE_BZIP2
|
2018-08-27 08:30:57 -07:00
|
|
|
else if ((c == 'b') || (c == 'B'))
|
2018-07-08 18:52:41 -07:00
|
|
|
options.compress_method = MZ_COMPRESS_METHOD_BZIP2;
|
2017-10-05 21:02:36 -07:00
|
|
|
#endif
|
|
|
|
#ifdef HAVE_LZMA
|
2018-08-27 08:30:57 -07:00
|
|
|
else if ((c == 'm') || (c == 'M'))
|
2018-07-08 18:52:41 -07:00
|
|
|
options.compress_method = MZ_COMPRESS_METHOD_LZMA;
|
2017-10-05 21:02:36 -07:00
|
|
|
#endif
|
|
|
|
#ifdef HAVE_AES
|
2018-08-27 08:30:57 -07:00
|
|
|
else if ((c == 's') || (c == 'S'))
|
2018-07-08 18:52:41 -07:00
|
|
|
options.aes = 1;
|
2018-10-25 19:35:50 -07:00
|
|
|
#endif
|
|
|
|
#ifndef MZ_NO_ZIP_SIGNING
|
2018-10-23 09:04:04 -07:00
|
|
|
else if (((c == 'h') || (c == 'H')) && (i + 1 < argc))
|
|
|
|
{
|
|
|
|
options.cert_path = argv[i + 1];
|
|
|
|
i += 1;
|
|
|
|
}
|
|
|
|
else if (((c == 'w') || (c == 'W')) && (i + 1 < argc))
|
|
|
|
{
|
|
|
|
options.cert_pwd = argv[i + 1];
|
|
|
|
i += 1;
|
|
|
|
}
|
2017-10-05 21:02:36 -07:00
|
|
|
#endif
|
2018-08-27 08:30:57 -07:00
|
|
|
else if ((c == 'c') || (c == 'C'))
|
|
|
|
options.legacy_encoding = 1;
|
|
|
|
else if (((c == 'k') || (c == 'K')) && (i + 1 < argc))
|
2018-07-08 18:52:41 -07:00
|
|
|
{
|
2018-08-31 16:43:41 -07:00
|
|
|
options.disk_size = (int64_t)atoi(argv[i + 1]) * 1024;
|
2018-07-08 18:52:41 -07:00
|
|
|
i += 1;
|
|
|
|
}
|
2018-08-27 08:30:57 -07:00
|
|
|
else if (((c == 'd') || (c == 'D')) && (i + 1 < argc))
|
2018-07-08 18:52:41 -07:00
|
|
|
{
|
|
|
|
destination = argv[i + 1];
|
|
|
|
i += 1;
|
|
|
|
}
|
2018-08-27 08:30:57 -07:00
|
|
|
else if (((c == 'p') || (c == 'P')) && (i + 1 < argc))
|
2018-07-08 18:52:41 -07:00
|
|
|
{
|
|
|
|
password = argv[i + 1];
|
|
|
|
i += 1;
|
2012-01-21 14:53:44 -07:00
|
|
|
}
|
|
|
|
}
|
2018-07-08 18:52:41 -07:00
|
|
|
else if (path_arg == 0)
|
2017-10-04 20:06:25 -07:00
|
|
|
path_arg = i;
|
2012-10-25 00:21:50 -07:00
|
|
|
}
|
|
|
|
|
2017-10-04 20:06:25 -07:00
|
|
|
if (path_arg == 0)
|
2012-10-25 00:21:50 -07:00
|
|
|
{
|
2017-03-18 16:16:06 -07:00
|
|
|
minizip_help();
|
2012-10-25 00:21:50 -07:00
|
|
|
return 0;
|
2012-01-21 14:53:44 -07:00
|
|
|
}
|
2017-10-04 20:06:25 -07:00
|
|
|
|
|
|
|
path = argv[path_arg];
|
2012-01-21 14:53:44 -07:00
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
if (do_list)
|
2012-01-21 14:53:44 -07:00
|
|
|
{
|
2018-08-13 23:07:42 -07:00
|
|
|
err = minizip_list(path);
|
2012-01-21 14:53:44 -07:00
|
|
|
}
|
2018-08-13 23:07:42 -07:00
|
|
|
else if (do_extract)
|
2018-01-02 09:22:38 -08:00
|
|
|
{
|
2018-08-13 23:07:42 -07:00
|
|
|
if (argc > path_arg + 1)
|
|
|
|
filename_to_extract = argv[path_arg + 1];
|
|
|
|
|
|
|
|
err = minizip_extract(path, filename_to_extract, destination, password, &options);
|
2018-01-02 09:22:38 -08:00
|
|
|
}
|
2018-08-13 23:07:42 -07:00
|
|
|
else if (do_erase)
|
2018-01-02 09:22:38 -08:00
|
|
|
{
|
2018-09-06 22:12:49 +00:00
|
|
|
strncpy(tmp_path, path, sizeof(tmp_path) - 1);
|
2018-09-10 08:51:33 -07:00
|
|
|
tmp_path[sizeof(tmp_path) - 1] = 0;
|
2018-08-13 23:07:42 -07:00
|
|
|
strncat(tmp_path, ".tmp", sizeof(tmp_path) - strlen(tmp_path) - 1);
|
2017-10-16 07:37:11 -07:00
|
|
|
|
2018-08-13 23:07:42 -07:00
|
|
|
err = minizip_erase(path, tmp_path, argc - (path_arg + 1), &argv[path_arg + 1]);
|
2012-10-25 00:21:50 -07:00
|
|
|
}
|
2017-10-16 07:37:11 -07:00
|
|
|
else
|
|
|
|
{
|
2018-08-13 23:07:42 -07:00
|
|
|
err = minizip_add(path, password, &options, argc - (path_arg + 1), &argv[path_arg + 1]);
|
2012-01-21 14:53:44 -07:00
|
|
|
}
|
|
|
|
|
2018-07-31 10:20:21 -07:00
|
|
|
if (err == MZ_OK && do_erase)
|
2018-07-14 09:53:56 -07:00
|
|
|
{
|
2018-07-31 10:20:21 -07:00
|
|
|
// Swap zip with temporary zip, backup old zip if possible
|
2018-09-06 22:12:49 +00:00
|
|
|
strncpy(bak_path, path, sizeof(bak_path) - 1);
|
2018-09-10 08:51:33 -07:00
|
|
|
bak_path[sizeof(bak_path) - 1] = 0;
|
2018-07-31 10:34:38 -07:00
|
|
|
strncat(bak_path, ".bak", sizeof(bak_path) - strlen(bak_path) - 1);
|
2018-07-14 09:53:56 -07:00
|
|
|
|
2018-07-31 10:20:21 -07:00
|
|
|
if (mz_os_file_exists(bak_path) == MZ_OK)
|
|
|
|
mz_os_delete(bak_path);
|
2018-07-14 09:53:56 -07:00
|
|
|
|
2018-07-31 10:20:21 -07:00
|
|
|
if (mz_os_rename(path, bak_path) != MZ_OK)
|
|
|
|
printf("Error backing up zip before replacing %s\n", bak_path);
|
2018-07-14 09:53:56 -07:00
|
|
|
|
2018-07-31 10:20:21 -07:00
|
|
|
if (mz_os_rename(tmp_path, path) != MZ_OK)
|
|
|
|
printf("Error replacing zip with temp %s\n", tmp_path);
|
|
|
|
}
|
2018-07-14 09:53:56 -07:00
|
|
|
|
2018-07-31 10:20:21 -07:00
|
|
|
return err;
|
2018-07-31 14:43:30 -07:00
|
|
|
}
|
2018-09-10 08:51:33 -07:00
|
|
|
#endif
|