Sandboxie/SandboxiePlus/MiscHelpers/Archive/ArchiveFS.cpp

363 lines
7.9 KiB
C++

#include "stdafx.h"
#include "ArchiveFS.h"
#include "../Common/Common.h"
//////////////////////////////////////////////////////////////////////////////////////////
// C7zFileEngineIterator
//
class C7zFileEngineIterator : public QAbstractFileEngineIterator
{
public:
#if QT_VERSION < QT_VERSION_CHECK(6, 8, 0)
C7zFileEngineIterator(QDir::Filters filters, const QStringList& filterNames,
const QStringList& allEntries)
: QAbstractFileEngineIterator(filters, filterNames), entries(allEntries), index(0) {}
~C7zFileEngineIterator() {}
QString next() override
{
if (!hasNext())
return QString();
++index;
return currentFilePath();
}
bool hasNext() const override
{
return index < entries.size();
}
#else
C7zFileEngineIterator(const QString &path, QDir::Filters filters, const QStringList& filterNames,
const QStringList& allEntries)
: QAbstractFileEngineIterator(path, filters, filterNames), entries(allEntries), index(0) {}
~C7zFileEngineIterator() {}
bool advance() override
{
if (index >= entries.size())
return false;
++index;
return true;
}
#endif
QString currentFileName() const override
{
if (index <= 0 || index > entries.size())
return QString();
return entries.at(index - 1);
}
private:
mutable QStringList entries;
mutable int index;
};
//////////////////////////////////////////////////////////////////////////////////////////
// C7zFileEngine
//
//QAtomicInt g_7zFileEngineCount;
C7zFileEngine::C7zFileEngine(const QString& filename, CArchive* pArchive, QMutex* pMutex)
: _flags(0), _size(0), _pos(0), _pArchive(pArchive), _pMutex(pMutex)
{
//g_7zFileEngineCount++;
setFileName(filename);
}
C7zFileEngine::~C7zFileEngine()
{
close();
//g_7zFileEngineCount--;
//qDebug() << "g_7zFileEngineCount" << g_7zFileEngineCount;
}
#if QT_VERSION < QT_VERSION_CHECK(6, 0, 0)
bool C7zFileEngine::open(QIODevice::OpenMode openMode)
#else
bool C7zFileEngine::open(QIODevice::OpenMode openMode, std::optional<QFile::Permissions> permissions)
#endif
{
close();
//if (openMode & QIODevice::WriteOnly) {
//
//} else
if (openMode & QIODevice::ReadOnly) {
QMutexLocker Lock(_pMutex);
QMap<int, QIODevice*> Files;
Files.insert(_index, new QBuffer(&_data));
_pArchive->Extract(&Files);
_size = _data.size();
_pos = 0;
} else
//if (openMode & QIODevice::Append) {
//
//} else
{
qWarning("Bad file open mode: %d", (int)openMode);
return false;
}
return true;
}
bool C7zFileEngine::close()
{
if (!isOpened())
return false;
_data.clear();
_size = 0;
_pos = 0;
return true;
}
bool C7zFileEngine::flush()
{
return true;
}
qint64 C7zFileEngine::size() const
{
return _size;
}
qint64 C7zFileEngine::pos() const
{
return _pos;
}
bool C7zFileEngine::seek(qint64 pos)
{
_pos = pos;
return true;
}
bool C7zFileEngine::isSequential() const
{
return false;
}
bool C7zFileEngine::remove()
{
return false;
}
bool C7zFileEngine::mkdir(const QString& dirName, bool createParentDirectories) const
{
return false;
}
bool C7zFileEngine::rmdir(const QString& dirName, bool recurseParentDirectories) const
{
return false;
}
bool C7zFileEngine::caseSensitive() const
{
return false;
}
bool C7zFileEngine::isRelativePath() const
{
return false;
}
QAbstractFileEngine::Iterator *C7zFileEngine::beginEntryList(QDir::Filters filters, const QStringList &filterNames)
{
QMutexLocker Lock(_pMutex);
QStringList allEntries;
for (int i = 0; i < _pArchive->FileCount(); i++) {
QString Path = _pArchive->FileProperty(i, "Path").toString();
if (!_filename.isEmpty()) {
if (!Path.startsWith(QString(_filename).replace("/", "\\")))
continue;
Path = Path.mid(_filename.length() + 1);
}
if (!Path.isEmpty() && !Path.contains("\\"))
allEntries.append(Path);
}
#if QT_VERSION < QT_VERSION_CHECK(6, 8, 0)
return new C7zFileEngineIterator(filters, filterNames, allEntries);
#else
return new C7zFileEngineIterator("", filters, filterNames, allEntries);
#endif
}
QAbstractFileEngine::FileFlags C7zFileEngine::fileFlags(FileFlags type) const
{
return type & _flags;
}
QString C7zFileEngine::fileName(FileName file) const
{
auto filename = Split2(_filename, "/", true);
if (filename.second.isEmpty()) {
filename.second = filename.first;
filename.first.clear();
}
switch (file)
{
case BaseName:
case AbsoluteName:
return filename.second;
case PathName:
case AbsolutePathName:
return filename.first;
//default:
//case DefaultName:
}
return _filename;
}
#if QT_VERSION < QT_VERSION_CHECK(6, 7, 0)
QDateTime C7zFileEngine::fileTime(FileTime time) const
#else
QDateTime C7zFileEngine::fileTime(QFile::FileTime time) const
#endif
{
switch (time)
{
#if QT_VERSION < QT_VERSION_CHECK(6, 7, 0)
case QAbstractFileEngine::ModificationTime:
#else
case QFile::FileModificationTime:
#endif
default:
return _datetime;
break;
};
}
void C7zFileEngine::setFileName(const QString& file)
{
int pos = file.indexOf(":") + 1;
_filename = file.mid(pos);
while (_filename.left(1) == "\\" || _filename.left(1) == "/")
_filename.remove(0, 1);
if (_filename.isEmpty()) { // root
_flags = ExistsFlag | DirectoryType | ReadOwnerPerm | ReadUserPerm | ReadGroupPerm | ReadOtherPerm;
return;
}
QMutexLocker Lock(_pMutex);
_index = _pArchive->FindByPath(_filename);
if (_index != -1) {
_flags = ExistsFlag | ReadOwnerPerm | ReadUserPerm | ReadGroupPerm | ReadOtherPerm;
if (_pArchive->FileProperty(_index, "IsDir").toBool())
_flags |= DirectoryType;
else
_flags |= FileType;
}
}
bool C7zFileEngine::atEnd() const
{
return _pos >= _data.size();
}
qint64 C7zFileEngine::read(char* data, qint64 maxlen)
{
qint64 togo = qMin(maxlen, _size - _pos);
if (togo > 0) {
memcpy(data, _data.constData() + _pos, togo);
_pos += togo;
}
return togo;
}
qint64 C7zFileEngine::readLine(char* data, qint64 maxlen)
{
return -1; // todo
}
qint64 C7zFileEngine::write(const char* data, qint64 len)
{
return -1;
}
bool C7zFileEngine::isOpened() const
{
return _size != 0;
}
QFile::FileError C7zFileEngine::error() const
{
return QFile::UnspecifiedError;
}
QString C7zFileEngine::errorString() const
{
return "";
}
bool C7zFileEngine::supportsExtension(Extension extension) const
{
return extension == QAbstractFileEngine::AtEndExtension;
}
//////////////////////////////////////////////////////////////////////////////////////////
// C7zFileEngineHandler
//
C7zFileEngineHandler::C7zFileEngineHandler(const QString& Scheme, QObject* parent)
: QObject(parent), m_pArchive(NULL)
{
m_Scheme = Scheme + ":";
}
C7zFileEngineHandler::~C7zFileEngineHandler()
{
Close();
}
bool C7zFileEngineHandler::Open(const QString& ArchivePath)
{
Close();
CArchive* pArchive = new CArchive(ArchivePath);
if (pArchive->Open() != ERR_7Z_OK) {
delete pArchive;
return false;
}
m_pArchive = pArchive;
return true;
}
void C7zFileEngineHandler::Close()
{
delete m_pArchive;
m_pArchive = NULL;
}
#if QT_VERSION < QT_VERSION_CHECK(6, 8, 0)
QAbstractFileEngine* C7zFileEngineHandler::create(const QString& filename) const
{
if (m_pArchive && filename.startsWith(m_Scheme))
return new C7zFileEngine(filename, m_pArchive, &m_Mutex);
return NULL;
}
#else
std::unique_ptr<QAbstractFileEngine> C7zFileEngineHandler::create(const QString& filename) const
{
if (m_pArchive && filename.startsWith(m_Scheme))
return std::unique_ptr<QAbstractFileEngine>(new C7zFileEngine(filename, m_pArchive, &m_Mutex));
return std::unique_ptr<QAbstractFileEngine>();
}
#endif