minizip-ng/minizip.c

297 lines
8.1 KiB
C
Raw Normal View History

/* minizip.c
Version 1.3.0, September 16th, 2017
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
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 <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"
#include "mz_compat.h"
#include "mzstrm.h"
#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
{
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
}
int minizip_addfile(void *zf, const char *path, const char *filenameinzip, int level, const char *password)
{
zip_fileinfo zi = { 0 };
void *stream_entry = NULL;
int size_read = 0;
int err = ZIP_OK;
char buf[UINT16_MAX];
// Get information about the file on disk so we can store it in zip
mz_os_get_file_date(path, &zi.dos_date);
/* 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,
password, 0, 1);
mz_stream_os_create(&stream_entry);
2017-09-29 21:02:09 -07:00
if (err != ZIP_OK)
{
printf("error in opening %s in zip file (%d)\n", filenameinzip, err);
}
else
{
2017-10-03 21:56:07 -07:00
if (mz_stream_os_open(stream_entry, path, MZ_STREAM_MODE_READ) != MZ_OK)
{
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))
{
printf("error in reading %s\n", filenameinzip);
err = ZIP_ERRNO;
}
if (size_read > 0)
{
err = zipWriteInFileInZip(zf, buf, size_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 == ZIP_OK) && (size_read > 0));
}
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
mz_stream_os_delete(&stream_entry);
if (err < 0)
{
err = ZIP_ERRNO;
}
else
{
err = zipCloseFileInZip(zf);
if (err != ZIP_OK)
printf("error in closing %s in the zip file (%d)\n", filenameinzip, err);
}
return err;
}
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 *zf = NULL;
void *stream = NULL;
2012-10-25 00:29:35 -07:00
char *zipfilename = NULL;
const char *password = NULL;
2012-10-25 00:21:50 -07:00
int zipfilenamearg = 0;
int errclose = 0;
int err = 0;
int i = 0;
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();
//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'))
opt_overwrite = MZ_APPEND_STATUS_CREATEAFTER;
2012-10-25 00:21:50 -07:00
if ((c == 'a') || (c == 'A'))
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')
{
2012-10-25 00:21:50 -07:00
opt_overwrite = 2;
}
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
{
const char *filename = argv[i];
const char *filenameinzip;
2012-10-25 00:21:50 -07:00
/* 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;
/* 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. */
2012-10-25 00:21:50 -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;
for (tmpptr = filenameinzip; *tmpptr; tmpptr++)
2012-10-25 00:21:50 -07:00
{
if (*tmpptr == '\\' || *tmpptr == '/')
lastslash = tmpptr;
}
if (lastslash != NULL)
filenameinzip = lastslash + 1; /* base filename follows last slash. */
2012-01-21 14:53:44 -07:00
}
2012-10-25 00:21:50 -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)
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
return err;
2012-01-21 14:53:44 -07:00
}
#endif