2014-01-12 14:04:54 -07:00
|
|
|
/* minizip.c
|
2017-10-03 23:19:37 -07:00
|
|
|
Version 1.3.0, September 16th, 2017
|
2014-01-12 14:04:54 -07:00
|
|
|
sample 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
|
2014-01-12 14:04:54 -07:00
|
|
|
Copyright (C) 1998-2010 Gilles Vollant
|
|
|
|
http://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 <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <time.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
|
2012-01-21 15:12:36 -07:00
|
|
|
#ifdef _WIN32
|
2014-01-07 19:13:14 -07:00
|
|
|
# include <direct.h>
|
|
|
|
# include <io.h>
|
2012-01-21 15:12:36 -07:00
|
|
|
#else
|
2014-01-07 19:13:14 -07:00
|
|
|
# include <unistd.h>
|
|
|
|
# include <utime.h>
|
|
|
|
# include <sys/types.h>
|
|
|
|
# include <sys/stat.h>
|
2012-01-21 14:53:44 -07:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "zip.h"
|
|
|
|
|
2017-10-02 22:11:03 -07:00
|
|
|
#include "mz_compat.h"
|
2017-10-01 22:42:35 -07:00
|
|
|
#include "mzstrm.h"
|
2017-10-02 22:11:03 -07:00
|
|
|
#include "test.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
|
|
|
{
|
2017-10-03 23:19:37 -07:00
|
|
|
printf("MiniZip 1.3.0, demo of zLib + MiniZip64 package\n");
|
2017-07-25 12:37:14 +08:00
|
|
|
printf("more info on MiniZip at https://github.com/nmoinvaz/minizip\n\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" \
|
|
|
|
" -o Overwrite existing file.zip\n" \
|
|
|
|
" -a Append to existing file.zip\n" \
|
|
|
|
" -0 Store only\n" \
|
|
|
|
" -1 Compress faster\n" \
|
|
|
|
" -9 Compress better\n\n" \
|
|
|
|
" -j exclude path. store only the file name.\n\n");
|
2012-01-21 14:53:44 -07:00
|
|
|
}
|
|
|
|
|
2017-10-02 22:11:03 -07:00
|
|
|
int minizip_addfile(void *zf, const char *path, const char *filenameinzip, int level, const char *password)
|
2017-03-18 17:07:05 -07:00
|
|
|
{
|
|
|
|
zip_fileinfo zi = { 0 };
|
2017-10-02 22:11:03 -07:00
|
|
|
void *stream_entry = NULL;
|
2017-03-18 17:07:05 -07:00
|
|
|
int size_read = 0;
|
|
|
|
int err = ZIP_OK;
|
2017-03-31 10:47:14 -07:00
|
|
|
char buf[UINT16_MAX];
|
2017-03-18 17:07:05 -07:00
|
|
|
|
|
|
|
|
2017-10-03 23:19:37 -07:00
|
|
|
// Get information about the file on disk so we can store it in zip
|
|
|
|
mz_os_get_file_date(path, &zi.dos_date);
|
2017-03-18 17:07:05 -07:00
|
|
|
|
|
|
|
/* Add to zip file */
|
|
|
|
err = zipOpenNewFileInZip3_64(zf, filenameinzip, &zi,
|
|
|
|
NULL, 0, NULL, 0, NULL /* comment*/,
|
|
|
|
(level != 0) ? Z_DEFLATED : 0, level, 0,
|
|
|
|
-MAX_WBITS, DEF_MEM_LEVEL, Z_DEFAULT_STRATEGY,
|
2017-10-02 22:11:03 -07:00
|
|
|
password, 0, 1);
|
2017-03-18 17:07:05 -07:00
|
|
|
|
2017-10-01 22:42:35 -07:00
|
|
|
mz_stream_os_create(&stream_entry);
|
2017-09-29 21:02:09 -07:00
|
|
|
|
2017-03-18 17:07:05 -07:00
|
|
|
if (err != ZIP_OK)
|
|
|
|
{
|
2017-10-03 23:19:37 -07:00
|
|
|
printf("error in opening %s in zip file (%d)\n", filenameinzip, err);
|
2017-03-18 17:07:05 -07:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2017-10-03 21:56:07 -07:00
|
|
|
if (mz_stream_os_open(stream_entry, path, MZ_STREAM_MODE_READ) != MZ_OK)
|
2017-03-18 17:07:05 -07:00
|
|
|
{
|
|
|
|
err = ZIP_ERRNO;
|
|
|
|
printf("error in opening %s for reading\n", path);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (err == ZIP_OK)
|
|
|
|
{
|
|
|
|
/* Read contents of file and write it to zip */
|
|
|
|
do
|
|
|
|
{
|
2017-09-29 21:02:09 -07:00
|
|
|
|
2017-10-01 21:43:24 -07:00
|
|
|
size_read = mz_stream_os_read(stream_entry, buf, sizeof(buf));
|
|
|
|
if ((size_read < (int)sizeof(buf)) && mz_stream_os_error(stream_entry))
|
2017-03-18 17:07:05 -07:00
|
|
|
{
|
|
|
|
printf("error in reading %s\n", filenameinzip);
|
|
|
|
err = ZIP_ERRNO;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (size_read > 0)
|
|
|
|
{
|
|
|
|
err = zipWriteInFileInZip(zf, buf, size_read);
|
|
|
|
if (err < 0)
|
2017-10-03 23:19:37 -07:00
|
|
|
printf("error in writing %s in the zip file (%d)\n", filenameinzip, err);
|
2017-03-18 17:07:05 -07:00
|
|
|
}
|
2017-09-29 21:02:09 -07:00
|
|
|
}
|
|
|
|
while ((err == ZIP_OK) && (size_read > 0));
|
2017-03-18 17:07:05 -07:00
|
|
|
}
|
|
|
|
|
2017-10-01 21:43:24 -07:00
|
|
|
if (mz_stream_os_is_open(stream_entry))
|
|
|
|
mz_stream_os_close(stream_entry);
|
2017-09-29 21:02:09 -07:00
|
|
|
|
2017-10-01 22:42:35 -07:00
|
|
|
mz_stream_os_delete(&stream_entry);
|
2017-03-18 17:07:05 -07:00
|
|
|
|
|
|
|
if (err < 0)
|
|
|
|
{
|
|
|
|
err = ZIP_ERRNO;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
err = zipCloseFileInZip(zf);
|
|
|
|
if (err != ZIP_OK)
|
2017-10-03 23:19:37 -07:00
|
|
|
printf("error in closing %s in the zip file (%d)\n", filenameinzip, err);
|
2017-03-18 17:07:05 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
2017-10-01 20:33:01 -07:00
|
|
|
|
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
|
|
|
{
|
2017-10-02 22:11:03 -07:00
|
|
|
void *zf = NULL;
|
|
|
|
void *stream = NULL;
|
2012-10-25 00:29:35 -07:00
|
|
|
char *zipfilename = NULL;
|
2016-10-09 20:09:50 -07:00
|
|
|
const char *password = NULL;
|
2012-10-25 00:21:50 -07:00
|
|
|
int zipfilenamearg = 0;
|
|
|
|
int errclose = 0;
|
|
|
|
int err = 0;
|
|
|
|
int i = 0;
|
2017-10-02 22:11:03 -07:00
|
|
|
int opt_overwrite = MZ_APPEND_STATUS_CREATE;
|
2012-10-25 00:21:50 -07:00
|
|
|
int opt_compress_level = Z_DEFAULT_COMPRESSION;
|
|
|
|
int opt_exclude_path = 0;
|
2017-10-02 00:44:51 -07:00
|
|
|
//test_crypt();
|
|
|
|
//test_aes();
|
2017-10-02 22:11:03 -07:00
|
|
|
//test_zlib();
|
|
|
|
//test_bzip();
|
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
|
|
|
|
2012-10-25 00:29:35 -07:00
|
|
|
/* 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
|
|
|
{
|
2012-10-25 00:21:50 -07:00
|
|
|
char c = *(p++);;
|
|
|
|
if ((c == 'o') || (c == 'O'))
|
2017-10-02 22:11:03 -07:00
|
|
|
opt_overwrite = MZ_APPEND_STATUS_CREATEAFTER;
|
2012-10-25 00:21:50 -07:00
|
|
|
if ((c == 'a') || (c == 'A'))
|
2017-10-02 22:11:03 -07:00
|
|
|
opt_overwrite = MZ_APPEND_STATUS_ADDINZIP;
|
2012-10-25 00:21:50 -07:00
|
|
|
if ((c >= '0') && (c <= '9'))
|
|
|
|
opt_compress_level = (c - '0');
|
|
|
|
if ((c == 'j') || (c == 'J'))
|
|
|
|
opt_exclude_path = 1;
|
|
|
|
|
|
|
|
if (((c == 'p') || (c == 'P')) && (i+1 < argc))
|
2012-01-21 14:53:44 -07:00
|
|
|
{
|
2012-10-25 00:21:50 -07:00
|
|
|
password=argv[i+1];
|
|
|
|
i++;
|
2012-01-21 14:53:44 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-10-25 00:21:50 -07:00
|
|
|
else
|
|
|
|
{
|
|
|
|
if (zipfilenamearg == 0)
|
|
|
|
zipfilenamearg = i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (zipfilenamearg == 0)
|
|
|
|
{
|
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
|
|
|
}
|
2012-10-25 00:29:35 -07:00
|
|
|
zipfilename = argv[zipfilenamearg];
|
2012-01-21 14:53:44 -07:00
|
|
|
|
2012-10-25 00:21:50 -07:00
|
|
|
if (opt_overwrite == 2)
|
2012-01-21 14:53:44 -07:00
|
|
|
{
|
2012-10-25 00:21:50 -07:00
|
|
|
/* If the file don't exist, we not append file */
|
2017-10-02 00:44:51 -07:00
|
|
|
if (mz_os_file_exists(zipfilename) == 0)
|
2012-10-25 00:21:50 -07:00
|
|
|
opt_overwrite = 1;
|
2012-01-21 14:53:44 -07:00
|
|
|
}
|
2012-10-25 00:21:50 -07:00
|
|
|
else if (opt_overwrite == 0)
|
2012-01-21 14:53:44 -07:00
|
|
|
{
|
2012-10-25 00:21:50 -07:00
|
|
|
/* If ask the user what to do because append and overwrite args not set */
|
2017-10-02 00:44:51 -07:00
|
|
|
if (mz_os_file_exists(zipfilename) != 0)
|
2012-10-25 00:21:50 -07:00
|
|
|
{
|
|
|
|
char rep = 0;
|
|
|
|
do
|
|
|
|
{
|
|
|
|
char answer[128];
|
2012-10-25 00:29:35 -07:00
|
|
|
printf("The file %s exists. Overwrite ? [y]es, [n]o, [a]ppend : ", zipfilename);
|
|
|
|
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')
|
2017-03-18 17:07:05 -07:00
|
|
|
{
|
2012-10-25 00:21:50 -07:00
|
|
|
opt_overwrite = 2;
|
2017-03-18 17:07:05 -07:00
|
|
|
}
|
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
|
|
|
|
|
|
|
zf = zipOpen2(zipfilename, opt_overwrite, NULL, stream);
|
2012-10-25 00:21:50 -07:00
|
|
|
|
|
|
|
if (zf == NULL)
|
2012-01-21 14:53:44 -07:00
|
|
|
{
|
2012-10-25 00:29:35 -07:00
|
|
|
printf("error opening %s\n", zipfilename);
|
2012-10-25 00:21:50 -07:00
|
|
|
err = ZIP_ERRNO;
|
|
|
|
}
|
|
|
|
else
|
2012-10-25 00:29:35 -07:00
|
|
|
printf("creating %s\n", zipfilename);
|
2012-10-25 00:21:50 -07:00
|
|
|
|
|
|
|
/* Go through command line args looking for files to add to zip */
|
2012-10-25 00:40:28 -07:00
|
|
|
for (i = zipfilenamearg + 1; (i < argc) && (err == ZIP_OK); i++)
|
2012-10-25 00:21:50 -07:00
|
|
|
{
|
2017-03-18 17:07:05 -07:00
|
|
|
const char *filename = argv[i];
|
|
|
|
const char *filenameinzip;
|
2012-10-25 00:21:50 -07:00
|
|
|
|
2012-10-25 00:36:46 -07:00
|
|
|
/* Skip command line options */
|
2013-12-18 14:30:33 +01:00
|
|
|
if ((((*(argv[i])) == '-') || ((*(argv[i])) == '/')) && (strlen(argv[i]) == 2) &&
|
2015-06-17 13:11:04 -07:00
|
|
|
((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;
|
|
|
|
|
2013-12-18 14:30:33 +01:00
|
|
|
/* Construct the filename that our file will be stored in the zip as.
|
2017-03-18 17:07:05 -07:00
|
|
|
The path name saved, should not include a leading slash.
|
|
|
|
If it did, windows/xp and dynazip couldn't read the zip file. */
|
2012-10-25 00:21:50 -07:00
|
|
|
|
2017-03-18 17:07:05 -07:00
|
|
|
filenameinzip = filename;
|
|
|
|
while (filenameinzip[0] == '\\' || filenameinzip[0] == '/')
|
|
|
|
filenameinzip++;
|
2012-10-25 00:21:50 -07:00
|
|
|
|
|
|
|
/* Should the file be stored with any path info at all? */
|
|
|
|
if (opt_exclude_path)
|
2012-01-21 14:53:44 -07:00
|
|
|
{
|
2012-10-25 00:21:50 -07:00
|
|
|
const char *tmpptr = NULL;
|
|
|
|
const char *lastslash = 0;
|
|
|
|
|
2017-03-18 17:07:05 -07:00
|
|
|
for (tmpptr = filenameinzip; *tmpptr; tmpptr++)
|
2012-10-25 00:21:50 -07:00
|
|
|
{
|
|
|
|
if (*tmpptr == '\\' || *tmpptr == '/')
|
|
|
|
lastslash = tmpptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (lastslash != NULL)
|
2017-03-18 17:07:05 -07:00
|
|
|
filenameinzip = lastslash + 1; /* base filename follows last slash. */
|
2012-01-21 14:53:44 -07:00
|
|
|
}
|
2012-10-25 00:21:50 -07:00
|
|
|
|
2017-03-18 17:07:05 -07:00
|
|
|
err = minizip_addfile(zf, filename, filenameinzip, opt_compress_level, password);
|
2012-01-21 14:53:44 -07:00
|
|
|
}
|
|
|
|
|
2012-10-25 00:21:50 -07:00
|
|
|
errclose = zipClose(zf, NULL);
|
|
|
|
if (errclose != ZIP_OK)
|
2012-10-25 00:36:46 -07:00
|
|
|
printf("error in closing %s (%d)\n", zipfilename, errclose);
|
2012-10-25 00:21:50 -07:00
|
|
|
|
2017-10-02 00:44:51 -07:00
|
|
|
mz_stream_os_delete(&stream);
|
2017-09-29 21:02:09 -07:00
|
|
|
|
2012-10-25 00:36:46 -07:00
|
|
|
return err;
|
2012-01-21 14:53:44 -07:00
|
|
|
}
|
2017-04-18 10:59:42 -07:00
|
|
|
#endif
|