OSCAR-code/oscar/zip.cpp
sawinglogz 78d3155ecd Add zip creation of OSCAR data folder.
Also fix the zip progress bar when compressing more than 2GB data,
and prevent the zip file from ever trying to include itself.
2020-01-22 17:33:15 -05:00

336 lines
9.8 KiB
C++

/* OSCAR ZIP archive creation
* Provides a Qt-convenient wrapper around miniz, see https://github.com/richgel999/miniz
*
* Copyright (c) 2020 The OSCAR Team
*
* This file is subject to the terms and conditions of the GNU General Public
* License. See the file COPYING in the main directory of the source code
* for more details. */
#include "zip.h"
#include <QDebug>
#include <QDateTime>
#include <QCoreApplication>
#include "SleepLib/progressdialog.h"
static const quint64 PROGRESS_SCALE = 1024; // QProgressBar only holds an int, so report progress in KiB.
// Static functions to abstract the details of miniz from the primary logic.
static void* zip_init();
static bool zip_open(void* ctx, QFile & file);
static bool zip_add(void* ctx, const QString & archive_name, const QByteArray & data, const QDateTime & modified);
static void zip_close(void* ctx);
static void zip_done(void* ctx);
ZipFile::ZipFile()
{
m_ctx = zip_init();
}
ZipFile::~ZipFile()
{
Close();
zip_done(m_ctx);
}
bool ZipFile::Open(const QString & filepath)
{
m_file.setFileName(filepath);
bool ok = m_file.open(QIODevice::WriteOnly);
if (!ok) {
qWarning() << "unable to open" << m_file.fileName();
return false;
}
ok = zip_open(m_ctx, m_file);
return ok;
}
void ZipFile::Close()
{
if (m_file.isOpen()) {
zip_close(m_ctx);
m_file.close();
}
}
bool ZipFile::AddDirectory(const QString & path, ProgressDialog* progress)
{
return AddDirectory(path, "", progress);
}
bool ZipFile::AddDirectory(const QString & path, const QString & prefix, ProgressDialog* progress)
{
bool ok;
FileQueue queue;
queue.AddDirectory(path, prefix);
ok = AddFiles(queue, progress);
return ok;
}
bool ZipFile::AddFiles(FileQueue & queue, ProgressDialog* progress)
{
bool ok;
// Exclude the zip file that's being created (if it happens to be in the list).
queue.Remove(QFileInfo(m_file).canonicalFilePath());
qDebug().noquote() << "Adding" << queue.toString();
m_abort = false;
m_progress = 0;
if (progress) {
progress->addAbortButton();
progress->setWindowModality(Qt::ApplicationModal);
progress->open();
connect(this, SIGNAL(setProgressMax(int)), progress, SLOT(setProgressMax(int)));
connect(this, SIGNAL(setProgressValue(int)), progress, SLOT(setProgressValue(int)));
connect(progress, SIGNAL(abortClicked()), this, SLOT(abort()));
}
// Always emit, since the caller may have configured and connected a progress dialog manually.
emit setProgressValue(m_progress/PROGRESS_SCALE);
emit setProgressMax((queue.byteCount() + queue.dirCount())/PROGRESS_SCALE);
QCoreApplication::processEvents();
for (auto & entry : queue.files()) {
ok = AddFile(entry.path, entry.name);
if (!ok || m_abort) {
break;
}
}
if (progress) {
disconnect(progress, SIGNAL(abortClicked()), this, SLOT(abort()));
disconnect(this, SIGNAL(setProgressMax(int)), progress, SLOT(setProgressMax(int)));
disconnect(this, SIGNAL(setProgressValue(int)), progress, SLOT(setProgressValue(int)));
progress->close();
progress->deleteLater();
}
if (!ok) {
qWarning().noquote() << "Unable to create" << m_file.fileName();
Close();
m_file.remove();
} else if (aborted()) {
qDebug().noquote() << "User canceled zip creation.";
Close();
m_file.remove();
} else {
qDebug().noquote() << "Created" << m_file.fileName() << m_file.size() << "bytes";
}
return ok;
}
bool ZipFile::AddFile(const QString & path, const QString & name)
{
if (!m_file.isOpen()) {
qWarning() << m_file.fileName() << "has not been opened for writing";
return false;
}
QFileInfo fi(path);
QByteArray data;
QString archive_name = name;
if (archive_name.isEmpty()) archive_name = fi.fileName();
if (fi.isDir()) {
archive_name += QDir::separator();
m_progress += 1;
} else {
// Open and read file into memory.
QFile f(path);
if (!f.open(QIODevice::ReadOnly)) {
qWarning() << path << "can't open";
return false;
}
data = f.readAll();
m_progress += data.size();
}
//qDebug() << "attempting to add" << archive_name << ":" << data.size() << "bytes";
bool ok = zip_add(m_ctx, archive_name, data, fi.lastModified());
emit setProgressValue(m_progress/PROGRESS_SCALE);
QCoreApplication::processEvents();
return ok;
}
// ==================================================================================================
bool FileQueue::AddDirectory(const QString & path, const QString & prefix)
{
QDir dir(path);
if (!dir.exists() || !dir.isReadable()) {
qWarning() << dir.canonicalPath() << "can't read directory";
return false;
}
QString base = prefix;
if (base.isEmpty()) base = dir.dirName();
// Add directory entry
bool ok = AddFile(dir.canonicalPath(), base);
if (!ok) {
return false;
}
dir.setFilter(QDir::NoDotAndDotDot | QDir::Dirs | QDir::Files | QDir::Hidden);
dir.setSorting(QDir::Name);
QFileInfoList flist = dir.entryInfoList();
for (auto & fi : flist) {
QString canonicalPath = fi.canonicalFilePath();
QString relative_path = base + QDir::separator() + fi.fileName();
if (fi.isSymLink()) {
qWarning() << "skipping symlink" << canonicalPath << fi.symLinkTarget();
} else if (fi.isDir()) {
// Descend and recurse
ok = AddDirectory(canonicalPath, relative_path);
} else {
// Add the file to the zip
ok = AddFile(canonicalPath, relative_path);
}
if (!ok) {
break;
}
}
return ok;
}
bool FileQueue::AddFile(const QString & path, const QString & prefix)
{
QFileInfo fi(path);
QString canonicalPath = fi.canonicalFilePath();
QString archive_name = prefix;
if (archive_name.isEmpty()) archive_name = fi.fileName();
if (fi.isDir()) {
m_dir_count++;
} else if (fi.exists()) {
m_file_count++;
m_byte_count += fi.size();
} else {
qWarning() << "file doesn't exist" << canonicalPath;
return false;
}
Entry entry = { canonicalPath, archive_name };
m_files.append(entry);
QCoreApplication::processEvents();
return true;
}
int FileQueue::Remove(const QString & path)
{
QFileInfo fi(path);
QString canonicalPath = fi.canonicalFilePath();
int removed = 0;
QMutableListIterator<Entry> i(m_files);
while (i.hasNext()) {
Entry & entry = i.next();
if (entry.path == canonicalPath) {
if (fi.isDir()) {
m_dir_count--;
} else {
m_file_count--;
m_byte_count -= fi.size();
}
i.remove();
removed++;
qDebug().noquote() << "skipping file:" << path;
}
}
return removed;
}
const QString FileQueue::toString() const
{
return QString("%1 directories, %2 files, %3 bytes").arg(m_dir_count).arg(m_file_count).arg(m_byte_count);
}
// ==================================================================================================
// Static functions to abstract the details of miniz from the primary logic.
#include "miniz.h"
// Callback for miniz to write compressed data
static size_t zip_write(void *pOpaque, mz_uint64 /*file_ofs*/, const void *pBuf, size_t n)
{
if (pOpaque == nullptr) {
qCritical() << "null pointer passed to ZipFile::Write!";
return 0;
}
QFile* file = (QFile*) pOpaque;
size_t written = file->write((const char*) pBuf, n);
if (written < n) {
qWarning() << "error writing to" << file->fileName();
}
return written;
}
static void* zip_init()
{
mz_zip_archive* pZip = new mz_zip_archive(); // zero-initializes struct
pZip->m_pWrite = zip_write;
return pZip;
}
static void zip_done(void* ctx)
{
Q_ASSERT(ctx);
mz_zip_archive* pZip = (mz_zip_archive*) ctx;
delete pZip;
}
static bool zip_open(void* ctx, QFile & file)
{
Q_ASSERT(ctx);
mz_zip_archive* pZip = (mz_zip_archive*) ctx;
pZip->m_pIO_opaque = &file;
bool ok = mz_zip_writer_init_v2(pZip, 0, MZ_ZIP_FLAG_CASE_SENSITIVE);
if (!ok) {
mz_zip_error mz_err = mz_zip_get_last_error(pZip);
qWarning() << "unable to initialize miniz writer" << MZ_VERSION << mz_zip_get_error_string(mz_err);
}
return ok;
}
static bool zip_add(void* ctx, const QString & archive_name, const QByteArray & data, const QDateTime & modified)
{
Q_ASSERT(ctx);
mz_zip_archive* pZip = (mz_zip_archive*) ctx;
// Add to .zip
time_t last_modified = modified.toTime_t(); // technically deprecated, but miniz expects a time_t
bool ok = mz_zip_writer_add_mem_ex_v2(pZip, archive_name.toLocal8Bit(), data.constData(), data.size(),
nullptr, 0, // no comment
MZ_DEFAULT_COMPRESSION,
0, 0, // not used when compressing data
&last_modified,
nullptr, 0, // no user extra data
nullptr, 0 // no user extra data central
);
if (!ok) {
mz_zip_error mz_err = mz_zip_get_last_error(pZip);
qWarning() << "unable to add" << archive_name << ":" << data.size() << "bytes" << mz_zip_get_error_string(mz_err);
}
return ok;
}
static void zip_close(void* ctx)
{
Q_ASSERT(ctx);
mz_zip_archive* pZip = (mz_zip_archive*) ctx;
mz_zip_writer_finalize_archive(pZip);
mz_zip_writer_end(pZip);
}