minizip-ng/minizip.c

370 lines
10 KiB
C
Raw Normal View History

/* minizip.c
Version 2.0.0, October 4th, 2017
part of the MiniZip project
2012-01-21 14:53:44 -07:00
2017-09-16 13:25:02 +08:00
Copyright (C) 2012-2017 Nathan Moinvaziri
https://github.com/nmoinvaz/minizip
Copyright (C) 2009-2010 Mathias Svensson
Modifications for Zip64 support
http://result42.com
Copyright (C) 2007-2008 Even Rouault
Modifications of Unzip for Zip64
Copyright (C) 1998-2010 Gilles Vollant
http://www.winimage.com/zLibDll/minizip.html
2012-01-21 14:53:44 -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 <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <time.h>
#include <errno.h>
#include "mz_error.h"
#include "mz_os.h"
2017-10-04 22:10:11 -07:00
#include "mz_strm.h"
#include "mz_zip.h"
/***************************************************************************/
2012-01-21 14:53:44 -07:00
2017-03-18 16:16:06 -07:00
void minizip_banner()
2012-01-21 14:53:44 -07:00
{
printf("Minizip 2.0.0 - https://github.com/nmoinvaz/minizip\n");
printf("---------------------------------------------------\n");
2012-10-25 00:21:50 -07:00
}
2012-01-21 14:53:44 -07:00
2017-03-18 16:16:06 -07:00
void minizip_help()
2012-10-25 00:21:50 -07:00
{
printf("Usage : minizip [-o] [-a] [-0 to -9] [-p password] [-j] file.zip [files_to_add]\n\n");
printf(" -o Overwrite existing file.zip\n");
printf(" -a Append to existing file.zip\n");
printf(" -0 Store only\n");
printf(" -1 Compress faster\n");
printf(" -9 Compress better\n");
#ifdef HAVE_BZIP2
printf(" -b BZIP2 compression\n");
#endif
#ifdef HAVE_LZMA
printf(" -m LZMA compression\n");
#endif
#ifdef HAVE_AES
printf(" -s AES encryption\n");
#endif
printf(" -j Exclude path and store only the file name\n\n");
2012-01-21 14:53:44 -07:00
}
/***************************************************************************/
int32_t minizip_add_file(void *handle, const char *path, uint8_t opt_exclude_path, mz_zip_compress *compress_info, mz_zip_crypt *crypt_info)
{
mz_zip_file file_info = { 0 };
int32_t read = 0;
int16_t err = MZ_OK;
int16_t err_close = MZ_OK;
void *stream = NULL;
const char *filenameinzip = NULL;
char buf[UINT16_MAX];
// Construct the filename that our file will be stored in the zip as.
// The path name saved, should not include a leading slash.
// If it did, windows/xp and dynazip couldn't read the zip file.
filenameinzip = path;
while (filenameinzip[0] == '\\' || filenameinzip[0] == '/')
filenameinzip += 1;
// Should the file be stored with any path info at all?
if (opt_exclude_path)
{
const char *match = NULL;
const char *last_slash = NULL;
for (match = filenameinzip; *match; match += 1)
{
if (*match == '\\' || *match == '/')
last_slash = match;
}
if (last_slash != NULL)
filenameinzip = last_slash + 1; // base filename follows last slash
}
// Get information about the file on disk so we can store it in zip
printf("Adding: %s\n", filenameinzip);
file_info.filename = filenameinzip;
if (mz_os_file_get_size(path) >= UINT32_MAX)
file_info.zip64 = 1;
mz_os_get_file_date(path, &file_info.dos_date);
// Add to zip
err = mz_zip_entry_open(handle, &file_info, compress_info, crypt_info);
if (err != MZ_OK)
{
printf("Error in opening %s in zip file (%d)\n", filenameinzip, err);
return err;
}
mz_stream_os_create(&stream);
err = mz_stream_os_open(stream, path, MZ_STREAM_MODE_READ);
if (err == MZ_OK)
{
// Read contents of file and write it to zip
do
{
read = mz_stream_os_read(stream, buf, sizeof(buf));
if (read < 0)
{
err = mz_stream_os_error(stream);
printf("Error %d in reading %s\n", err, filenameinzip);
break;
}
if (read == 0)
break;
err = mz_zip_entry_write(handle, buf, read);
if (err < 0)
printf("Error in writing %s in the zip file (%d)\n", filenameinzip, err);
2017-09-29 21:02:09 -07:00
}
while (err == MZ_OK);
mz_stream_os_close(stream);
}
else
{
printf("Error in opening %s for reading\n", path);
}
2017-09-29 21:02:09 -07:00
mz_stream_os_delete(&stream);
err_close = mz_zip_entry_close(handle);
if (err_close != MZ_OK)
printf("Error in closing %s in the zip file (%d)\n", filenameinzip, err_close);
return err;
}
int32_t minizip_add(void *handle, const char *path, uint8_t opt_exclude_path, mz_zip_compress *compress_info, mz_zip_crypt *crypt_info, uint8_t recursive)
{
DIR *dir = NULL;
2017-10-06 00:02:28 -07:00
struct dirent *entry = NULL;
int16_t err = 0;
2017-10-06 00:02:28 -07:00
int16_t full_path_len = 0;
char full_path[320];
if (mz_os_is_dir(path) != MZ_OK)
return minizip_add_file(handle, path, opt_exclude_path, compress_info, crypt_info);
dir = mz_os_open_dir(path);
if (dir == NULL)
{
printf("Cannot enumerate directory %s\n", path);
return MZ_EXIST_ERROR;
}
while ((entry = mz_os_read_dir(dir)) != NULL)
{
if (strcmp(entry->d_name, ".") == 0 || strcmp(entry->d_name, "..") == 0)
continue;
strncpy(full_path, path, sizeof(full_path));
2017-10-06 00:08:29 -07:00
full_path_len = (int16_t)strlen(full_path);
2017-10-06 00:02:28 -07:00
if (full_path_len > 0 && full_path[full_path_len - 1] != '\\')
strncat(full_path, "\\", sizeof(full_path) - full_path_len - 1);
strncat(full_path, entry->d_name, sizeof(full_path) - full_path_len - 2);
if (!recursive && mz_os_is_dir(full_path))
continue;
err = minizip_add(handle, full_path, opt_exclude_path, compress_info, crypt_info, recursive);
if (err != MZ_OK)
return err;
}
mz_os_close_dir(dir);
return MZ_OK;
}
2017-03-18 16:32:30 -07:00
#ifndef NOMAIN
2012-10-25 00:29:35 -07:00
int main(int argc, char *argv[])
2012-01-21 14:53:44 -07:00
{
void *handle = NULL;
void *stream = NULL;
char *path = NULL;
mz_zip_compress compress_info;
mz_zip_crypt crypt_info;
int32_t path_arg = 0;
uint8_t opt_append = 0;
uint8_t opt_open_existing = 0;
uint8_t opt_exclude_path = 0;
int16_t mode = 0;
int16_t err_close = 0;
int16_t err = 0;
int16_t i = 0;
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;
}
2012-10-25 00:21:50 -07:00
memset(&compress_info, 0, sizeof(compress_info));
memset(&crypt_info, 0, sizeof(crypt_info));
compress_info.method = MZ_METHOD_DEFLATE;
compress_info.level = MZ_COMPRESS_LEVEL_DEFAULT;
// Parse command line options
2012-10-25 00:21:50 -07:00
for (i = 1; i < argc; i++)
2012-01-21 14:53:44 -07:00
{
2012-10-25 00:21:50 -07:00
if ((*argv[i]) == '-')
2012-01-21 14:53:44 -07:00
{
2012-10-25 00:21:50 -07:00
const char *p = argv[i]+1;
2012-01-21 14:53:44 -07:00
2012-10-25 00:21:50 -07:00
while ((*p) != '\0')
2012-01-21 14:53:44 -07:00
{
char c = *(p++);
2012-10-25 00:21:50 -07:00
if ((c == 'o') || (c == 'O'))
opt_append = 1;
2012-10-25 00:21:50 -07:00
if ((c == 'a') || (c == 'A'))
opt_open_existing = 1;
2012-10-25 00:21:50 -07:00
if ((c >= '0') && (c <= '9'))
{
compress_info.level = (c - '0');
if (compress_info.level == 0)
compress_info.method = MZ_METHOD_RAW;
}
2012-10-25 00:21:50 -07:00
if ((c == 'j') || (c == 'J'))
opt_exclude_path = 1;
#ifdef HAVE_BZIP2
if ((c == 'b') || (c == 'B'))
compress_info.method = MZ_METHOD_BZIP2;
#endif
#ifdef HAVE_LZMA
if ((c == 'm') || (c == 'M'))
compress_info.method = MZ_METHOD_LZMA;
#endif
#ifdef HAVE_AES
if ((c == 's') || (c == 'S'))
crypt_info.aes = 1;
#endif
if (((c == 'p') || (c == 'P')) && (i + 1 < argc))
2012-01-21 14:53:44 -07:00
{
crypt_info.password = argv[i + 1];
i += 1;
2012-01-21 14:53:44 -07:00
}
}
continue;
2012-01-21 14:53:44 -07:00
}
if (path_arg == 0)
path_arg = i;
2012-10-25 00:21:50 -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
}
path = argv[path_arg];
2012-01-21 14:53:44 -07:00
if (opt_open_existing)
2012-01-21 14:53:44 -07:00
{
2017-10-06 11:43:21 +08:00
// If the file doesn't exist, we don't append file
if (mz_os_file_exists(path) != MZ_OK)
opt_append = 0;
2012-01-21 14:53:44 -07:00
}
else if (opt_append == 0)
2012-01-21 14:53:44 -07:00
{
// If ask the user what to do because append and overwrite args not set
if (mz_os_file_exists(path) != 0)
2012-10-25 00:21:50 -07:00
{
char rep = 0;
do
{
char answer[128];
printf("The file %s exists. Overwrite ? [y]es, [n]o, [a]ppend : ", path);
2012-10-25 00:29:35 -07:00
if (scanf("%1s", answer) != 1)
exit(EXIT_FAILURE);
2012-10-25 00:21:50 -07:00
rep = answer[0];
2012-01-21 14:53:44 -07:00
2012-10-25 00:29:35 -07:00
if ((rep >= 'a') && (rep <= 'z'))
2012-10-25 00:21:50 -07:00
rep -= 0x20;
}
2012-10-25 00:29:35 -07:00
while ((rep != 'Y') && (rep != 'N') && (rep != 'A'));
2012-01-21 14:53:44 -07:00
2012-10-25 00:21:50 -07:00
if (rep == 'A')
{
opt_open_existing = 1;
}
2012-10-25 00:29:35 -07:00
else if (rep == 'N')
{
2017-03-18 16:16:06 -07:00
minizip_help();
2012-10-25 00:29:35 -07:00
return 0;
}
2012-01-21 14:53:44 -07:00
}
}
2017-10-02 00:44:51 -07:00
mz_stream_os_create(&stream);
2017-09-29 21:02:09 -07:00
mode = MZ_STREAM_MODE_READWRITE;
if (opt_append)
mode |= MZ_STREAM_MODE_APPEND;
else
mode |= MZ_STREAM_MODE_CREATE;
if (mz_stream_open(stream, path, mode) != MZ_OK)
{
mz_stream_os_delete(&stream);
printf("Error opening file %s\n", path);
return 1;
}
handle = mz_zip_open(opt_open_existing, 0, stream);
2012-10-25 00:21:50 -07:00
if (handle == NULL)
2012-01-21 14:53:44 -07:00
{
mz_stream_os_delete(&stream);
printf("Error opening zip %s\n", path);
return 1;
2012-10-25 00:21:50 -07:00
}
printf("Creating %s\n", path);
// Go through command line args looking for files to add to zip
for (i = path_arg + 1; (i < argc) && (err == MZ_OK); i++)
2012-10-25 00:21:50 -07:00
{
const char *filename = argv[i];
// Skip command line options
if ((((*(argv[i])) == '-') || ((*(argv[i])) == '/')) && (strlen(argv[i]) == 2) &&
((argv[i][1] == 'o') || (argv[i][1] == 'O') || (argv[i][1] == 'a') || (argv[i][1] == 'A') ||
(argv[i][1] == 'p') || (argv[i][1] == 'P') || ((argv[i][1] >= '0') && (argv[i][1] <= '9'))))
2012-10-25 00:21:50 -07:00
continue;
err = minizip_add(handle, filename, opt_exclude_path, &compress_info, &crypt_info, 1);
2012-01-21 14:53:44 -07:00
}
err_close = mz_zip_close(handle, NULL, 0);
if (err_close != MZ_OK)
printf("Error in closing %s (%d)\n", path, err_close);
2012-10-25 00:21:50 -07:00
mz_stream_os_close(stream);
2017-10-02 00:44:51 -07:00
mz_stream_os_delete(&stream);
2017-09-29 21:02:09 -07:00
return err;
2012-01-21 14:53:44 -07:00
}
#endif