Commit 6198c542 authored by Ronan Abhamon's avatar Ronan Abhamon

feat(FileExtractor): add impl

parent bc2875c4
...@@ -20,15 +20,95 @@ ...@@ -20,15 +20,95 @@
* Author: Ronan Abhamon * Author: Ronan Abhamon
*/ */
#include <mz_strm.h> #include <mz_strm_bzip.h>
#include <mz.h>
#include <QDebug> #include <QDebug>
#include <QDir>
#include "FileExtractor.hpp" #include "FileExtractor.hpp"
// ============================================================================= // =============================================================================
using namespace std;
static const char *minizipErrorToString (int error) {
switch (error) {
case MZ_OK:
return "ok";
case MZ_STREAM_ERROR:
return "stream error";
case MZ_DATA_ERROR:
return "data error";
case MZ_MEM_ERROR:
return "memory error";
case MZ_END_OF_LIST:
return "end of list";
case MZ_END_OF_STREAM:
return "end of stream";
case MZ_PARAM_ERROR:
return "param error";
case MZ_FORMAT_ERROR:
return "format error";
case MZ_INTERNAL_ERROR:
return "internal error";
case MZ_CRC_ERROR:
return "crc error";
case MZ_CRYPT_ERROR:
return "crypt error";
case MZ_EXIST_ERROR:
return "exist error";
case MZ_PASSWORD_ERROR:
return "password error";
}
Q_ASSERT(false);
return "";
}
static int openMinizipStream (void **stream, const char *filePath) {
*stream = nullptr;
if (!mz_stream_bzip_create(stream))
return MZ_MEM_ERROR;
return mz_stream_bzip_open(stream, filePath, MZ_OPEN_MODE_READ);
}
// -----------------------------------------------------------------------------
void FileExtractor::extract () { void FileExtractor::extract () {
// TODO. if (mExtracting) {
qWarning() << "Unable to extract file. Already extracting!";
return;
}
setExtracting(true);
QFileInfo fileInfo(mFile);
setReadBytes(0);
setTotalBytes(fileInfo.size());
// 1. Open archive stream.
// TODO: Test extension.
int error = openMinizipStream(&mStream, mFile.toLatin1().constData());
if (error != MZ_OK) {
emitExtractFailed(error);
return;
}
// 2. Open output file.
// TODO: Deal with existing files.
Q_ASSERT(!mDestinationFile.isOpen());
mDestinationFile.setFileName(
QDir::cleanPath(mExtractFolder) + QDir::separator() + fileInfo.completeBaseName()
);
if (!mDestinationFile.open(QIODevice::WriteOnly)) {
emitOutputError();
return;
}
// 3. Connect!
mTimer = new QTimer(this);
QObject::connect(mTimer, &QTimer::timeout, this, &FileExtractor::handleExtraction);
mTimer->start();
} }
QString FileExtractor::getFile () const { QString FileExtractor::getFile () const {
...@@ -96,17 +176,50 @@ void FileExtractor::setExtracting (bool extracting) { ...@@ -96,17 +176,50 @@ void FileExtractor::setExtracting (bool extracting) {
} }
} }
void FileExtractor::handleReadyData () { void FileExtractor::clean () {
// TODO. mz_stream_bzip_delete(&mStream);
mDestinationFile.close();
mTimer->stop();
mTimer->deleteLater();
setExtracting(false);
}
void FileExtractor::emitExtractFailed (int error) {
qWarning() << QStringLiteral("Unable to open extract file: `%1` (%2).")
.arg(mFile).arg(minizipErrorToString(error));
mDestinationFile.remove();
clean();
emit extractFailed();
} }
void FileExtractor::handleExtractFinished () { void FileExtractor::emitExtractFinished () {
// TODO. clean();
emit extractFinished();
}
void FileExtractor::emitOutputError () {
qWarning() << QStringLiteral("Could not write into `%1` (%2).")
.arg(mDestinationFile.fileName()).arg(mDestinationFile.errorString());
mDestinationFile.remove();
clean();
emit extractFailed();
} }
void FileExtractor::handleExtractProgress (qint64 readBytes, qint64 totalBytes) { void FileExtractor::handleExtraction () {
// TODO. char buffer[4096];
Q_UNUSED(readBytes); int32_t readBytes = mz_stream_bzip_read(mStream, buffer, sizeof buffer);
Q_UNUSED(totalBytes); switch (readBytes) {
case MZ_OK:
break;
case MZ_END_OF_STREAM:
emitExtractFinished();
return;
default:
emitExtractFailed(readBytes);
return;
}
setReadBytes(readBytes);
if (mDestinationFile.write(buffer, sizeof buffer) == -1)
emitOutputError();
} }
...@@ -24,13 +24,15 @@ ...@@ -24,13 +24,15 @@
#define FILE_EXTRACTOR_H_ #define FILE_EXTRACTOR_H_
#include <QFile> #include <QFile>
#include <QTimer>
// ============================================================================= // =============================================================================
// Supports only bzip file.
class FileExtractor : public QObject { class FileExtractor : public QObject {
Q_OBJECT; Q_OBJECT;
Q_PROPERTY(QString file READ getFile WRITE setFile NOTIFY fileChanged); Q_PROPERTY(QString file READ getFile WRITE setFile NOTIFY fileChanged);
Q_PROPERTY(QString extractFolder READ getExtractFolder WRITE setExtractFolder NOTIFY extractFolderChanged); Q_PROPERTY(QString extractFolder READ getExtractFolder WRITE setExtractFolder NOTIFY extractFolderChanged);
Q_PROPERTY(qint64 readBytes READ getReadBytes NOTIFY readBytesChanged); Q_PROPERTY(qint64 readBytes READ getReadBytes NOTIFY readBytesChanged);
Q_PROPERTY(qint64 totalBytes READ getTotalBytes NOTIFY totalBytesChanged); Q_PROPERTY(qint64 totalBytes READ getTotalBytes NOTIFY totalBytesChanged);
...@@ -64,10 +66,13 @@ private: ...@@ -64,10 +66,13 @@ private:
bool getExtracting () const; bool getExtracting () const;
void setExtracting (bool extracting); void setExtracting (bool extracting);
void handleReadyData (); void clean ();
void handleExtractFinished ();
void handleExtractProgress (qint64 readBytes, qint64 totalBytes); void emitExtractFinished ();
void emitExtractFailed (int error);
void emitOutputError ();
void handleExtraction ();
QString mFile; QString mFile;
QString mExtractFolder; QString mExtractFolder;
...@@ -76,6 +81,10 @@ private: ...@@ -76,6 +81,10 @@ private:
qint64 mReadBytes = 0; qint64 mReadBytes = 0;
qint64 mTotalBytes = 0; qint64 mTotalBytes = 0;
bool mExtracting = false; bool mExtracting = false;
void *mStream = nullptr;
QTimer *mTimer = nullptr;
}; };
#endif // FILE_EXTRACTOR_H_ #endif // FILE_EXTRACTOR_H_
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment