PrusaSlicer-NonPlainar/src/libslic3r/Zipper.cpp

221 lines
6.7 KiB
C++
Raw Normal View History

2019-03-14 18:07:16 +00:00
#include <exception>
#include "Zipper.hpp"
#include "miniz_extension.hpp"
2019-03-27 12:22:04 +00:00
#include <boost/log/trivial.hpp>
#include "I18N.hpp"
//! macro used to mark string used at localization,
//! return same string
#define L(s) Slic3r::I18N::translate(s)
2019-03-15 11:30:41 +00:00
#if defined(_MSC_VER) && _MSC_VER <= 1800 || __cplusplus < 201103L
#define SLIC3R_NORETURN
#elif __cplusplus >= 201103L
#define SLIC3R_NORETURN [[noreturn]]
#endif
2019-03-14 18:07:16 +00:00
namespace Slic3r {
class Zipper::Impl {
public:
mz_zip_archive arch;
2019-03-15 11:30:41 +00:00
std::string m_zipname;
2019-03-27 12:22:04 +00:00
static std::string get_errorstr(mz_zip_error mz_err)
{
switch (mz_err)
{
case MZ_ZIP_NO_ERROR:
return "no error";
case MZ_ZIP_UNDEFINED_ERROR:
return L("undefined error");
case MZ_ZIP_TOO_MANY_FILES:
return L("too many files");
case MZ_ZIP_FILE_TOO_LARGE:
return L("file too large");
case MZ_ZIP_UNSUPPORTED_METHOD:
return L("unsupported method");
case MZ_ZIP_UNSUPPORTED_ENCRYPTION:
return L("unsupported encryption");
case MZ_ZIP_UNSUPPORTED_FEATURE:
return L("unsupported feature");
case MZ_ZIP_FAILED_FINDING_CENTRAL_DIR:
return L("failed finding central directory");
case MZ_ZIP_NOT_AN_ARCHIVE:
return L("not a ZIP archive");
case MZ_ZIP_INVALID_HEADER_OR_CORRUPTED:
return L("invalid header or archive is corrupted");
case MZ_ZIP_UNSUPPORTED_MULTIDISK:
return L("unsupported multidisk archive");
case MZ_ZIP_DECOMPRESSION_FAILED:
return L("decompression failed or archive is corrupted");
case MZ_ZIP_COMPRESSION_FAILED:
return L("compression failed");
case MZ_ZIP_UNEXPECTED_DECOMPRESSED_SIZE:
return L("unexpected decompressed size");
case MZ_ZIP_CRC_CHECK_FAILED:
return L("CRC-32 check failed");
case MZ_ZIP_UNSUPPORTED_CDIR_SIZE:
return L("unsupported central directory size");
case MZ_ZIP_ALLOC_FAILED:
return L("allocation failed");
case MZ_ZIP_FILE_OPEN_FAILED:
return L("file open failed");
case MZ_ZIP_FILE_CREATE_FAILED:
return L("file create failed");
case MZ_ZIP_FILE_WRITE_FAILED:
return L("file write failed");
case MZ_ZIP_FILE_READ_FAILED:
return L("file read failed");
case MZ_ZIP_FILE_CLOSE_FAILED:
return L("file close failed");
case MZ_ZIP_FILE_SEEK_FAILED:
return L("file seek failed");
case MZ_ZIP_FILE_STAT_FAILED:
return L("file stat failed");
case MZ_ZIP_INVALID_PARAMETER:
return L("invalid parameter");
case MZ_ZIP_INVALID_FILENAME:
return L("invalid filename");
case MZ_ZIP_BUF_TOO_SMALL:
return L("buffer too small");
case MZ_ZIP_INTERNAL_ERROR:
return L("internal error");
case MZ_ZIP_FILE_NOT_FOUND:
return L("file not found");
case MZ_ZIP_ARCHIVE_TOO_LARGE:
return L("archive is too large");
case MZ_ZIP_VALIDATION_FAILED:
return L("validation failed");
case MZ_ZIP_WRITE_CALLBACK_FAILED:
return L("write calledback failed");
default:
break;
}
return "unknown error";
}
2019-03-27 12:22:04 +00:00
std::string formatted_errorstr() const
{
return L("Error with zip archive") + " " + m_zipname + ": " +
get_errorstr(arch.m_last_error) + "!";
}
SLIC3R_NORETURN void blow_up() const
{
throw std::runtime_error(formatted_errorstr());
}
2019-03-28 15:19:30 +00:00
bool is_alive()
{
return arch.m_zip_mode != MZ_ZIP_MODE_WRITING_HAS_BEEN_FINALIZED;
}
2019-03-14 18:07:16 +00:00
};
Zipper::Zipper(const std::string &zipfname, e_compression compression)
{
m_impl.reset(new Impl());
2019-03-15 11:30:41 +00:00
m_compression = compression;
m_impl->m_zipname = zipfname;
2019-03-14 18:07:16 +00:00
memset(&m_impl->arch, 0, sizeof(m_impl->arch));
if (!open_zip_writer(&m_impl->arch, zipfname)) {
m_impl->blow_up();
}
2019-03-14 18:07:16 +00:00
}
Zipper::~Zipper()
{
2019-03-28 15:19:30 +00:00
if(m_impl->is_alive()) {
// Flush the current entry if not finished yet.
try { finish_entry(); } catch(...) {
BOOST_LOG_TRIVIAL(error) << m_impl->formatted_errorstr();
}
if(!mz_zip_writer_finalize_archive(&m_impl->arch))
BOOST_LOG_TRIVIAL(error) << m_impl->formatted_errorstr();
2019-03-27 12:22:04 +00:00
}
2019-03-28 15:19:30 +00:00
// The file should be closed no matter what...
if(!close_zip_writer(&m_impl->arch))
2019-03-28 15:19:30 +00:00
BOOST_LOG_TRIVIAL(error) << m_impl->formatted_errorstr();
}
Zipper::Zipper(Zipper &&m):
m_impl(std::move(m.m_impl)),
m_data(std::move(m.m_data)),
m_entry(std::move(m.m_entry)),
2019-03-15 11:30:41 +00:00
m_compression(m.m_compression) {}
Zipper &Zipper::operator=(Zipper &&m) {
m_impl = std::move(m.m_impl);
m_data = std::move(m.m_data);
m_entry = std::move(m.m_entry);
m_compression = m.m_compression;
return *this;
2019-03-14 18:07:16 +00:00
}
void Zipper::add_entry(const std::string &name)
{
2019-03-28 15:19:30 +00:00
if(!m_impl->is_alive()) return;
2019-03-14 18:07:16 +00:00
finish_entry(); // finish previous business
m_entry = name;
}
2019-03-18 17:02:50 +00:00
void Zipper::add_entry(const std::string &name, const uint8_t *data, size_t l)
{
2019-03-28 15:19:30 +00:00
if(!m_impl->is_alive()) return;
2019-03-18 17:02:50 +00:00
finish_entry();
mz_uint cmpr = MZ_NO_COMPRESSION;
switch (m_compression) {
case NO_COMPRESSION: cmpr = MZ_NO_COMPRESSION; break;
case FAST_COMPRESSION: cmpr = MZ_BEST_SPEED; break;
case TIGHT_COMPRESSION: cmpr = MZ_BEST_COMPRESSION; break;
}
if(!mz_zip_writer_add_mem(&m_impl->arch, name.c_str(), data, l, cmpr))
m_impl->blow_up();
m_entry.clear();
m_data.clear();
}
2019-03-14 18:07:16 +00:00
void Zipper::finish_entry()
{
2019-03-28 15:19:30 +00:00
if(!m_impl->is_alive()) return;
if(!m_data.empty() && !m_entry.empty()) {
2019-03-14 18:07:16 +00:00
mz_uint compression = MZ_NO_COMPRESSION;
switch (m_compression) {
case NO_COMPRESSION: compression = MZ_NO_COMPRESSION; break;
case FAST_COMPRESSION: compression = MZ_BEST_SPEED; break;
case TIGHT_COMPRESSION: compression = MZ_BEST_COMPRESSION; break;
}
if(!mz_zip_writer_add_mem(&m_impl->arch, m_entry.c_str(),
m_data.c_str(),
m_data.size(),
compression)) m_impl->blow_up();
2019-03-14 18:07:16 +00:00
}
m_data.clear();
m_entry.clear();
}
2019-03-28 15:19:30 +00:00
void Zipper::finalize()
2019-03-27 12:22:04 +00:00
{
finish_entry();
2019-03-28 15:19:30 +00:00
if(m_impl->is_alive()) if(!mz_zip_writer_finalize_archive(&m_impl->arch))
m_impl->blow_up();
2019-03-27 12:22:04 +00:00
}
2019-03-14 18:07:16 +00:00
}