mirror of
https://github.com/QuasarApp/installer-framework.git
synced 2025-05-16 23:09:34 +00:00
Change-Id: I8dde6629cfd461104364d5cdc255cb54b58283fa Reviewed-by: Niels Weber <niels.weber@theqtcompany.com>
389 lines
14 KiB
C++
389 lines
14 KiB
C++
/**************************************************************************
|
|
**
|
|
** Copyright (C) 2015 The Qt Company Ltd.
|
|
** Contact: http://www.qt.io/licensing/
|
|
**
|
|
** This file is part of the Qt Installer Framework.
|
|
**
|
|
** $QT_BEGIN_LICENSE:LGPL$
|
|
** Commercial License Usage
|
|
** Licensees holding valid commercial Qt licenses may use this file in
|
|
** accordance with the commercial license agreement provided with the
|
|
** Software or, alternatively, in accordance with the terms contained in
|
|
** a written agreement between you and The Qt Company. For licensing terms
|
|
** and conditions see http://qt.io/terms-conditions. For further
|
|
** information use the contact form at http://www.qt.io/contact-us.
|
|
**
|
|
** GNU Lesser General Public License Usage
|
|
** Alternatively, this file may be used under the terms of the GNU Lesser
|
|
** General Public License version 2.1 or version 3 as published by the Free
|
|
** Software Foundation and appearing in the file LICENSE.LGPLv21 and
|
|
** LICENSE.LGPLv3 included in the packaging of this file. Please review the
|
|
** following information to ensure the GNU Lesser General Public License
|
|
** requirements will be met: https://www.gnu.org/licenses/lgpl.html and
|
|
** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
|
**
|
|
** As a special exception, The Qt Company gives you certain additional
|
|
** rights. These rights are described in The Qt Company LGPL Exception
|
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|
**
|
|
**
|
|
** $QT_END_LICENSE$
|
|
**
|
|
**************************************************************************/
|
|
|
|
#include "createlocalrepositoryoperation.h"
|
|
|
|
#include "binarycontent.h"
|
|
#include "binaryformat.h"
|
|
#include "errors.h"
|
|
#include "fileio.h"
|
|
#include "fileutils.h"
|
|
#include "copydirectoryoperation.h"
|
|
#include "lib7z_facade.h"
|
|
#include "packagemanagercore.h"
|
|
#include "productkeycheck.h"
|
|
|
|
#include "kdupdaterupdateoperations.h"
|
|
|
|
#include <QtCore/QDir>
|
|
#include <QtCore/QDirIterator>
|
|
|
|
#include <cerrno>
|
|
|
|
namespace QInstaller {
|
|
|
|
|
|
// -- AutoHelper
|
|
|
|
struct AutoHelper
|
|
{
|
|
public:
|
|
AutoHelper(CreateLocalRepositoryOperation *op)
|
|
: m_op(op)
|
|
{
|
|
}
|
|
|
|
~AutoHelper()
|
|
{
|
|
m_op->emitFullProgress();
|
|
m_op->setValue(QLatin1String("files"), m_files);
|
|
}
|
|
|
|
QStringList m_files;
|
|
CreateLocalRepositoryOperation *m_op;
|
|
};
|
|
|
|
|
|
// statics
|
|
|
|
namespace Static {
|
|
|
|
static void fixPermissions(const QString &repoPath)
|
|
{
|
|
QDirIterator it(repoPath, QDirIterator::Subdirectories);
|
|
while (it.hasNext() && !it.next().isEmpty()) {
|
|
if (!it.fileInfo().isFile())
|
|
continue;
|
|
|
|
if (!QFile::setPermissions(it.filePath(), QFile::ReadOwner | QFile::WriteOwner
|
|
| QFile::ReadUser | QFile::WriteUser | QFile::ReadGroup | QFile::ReadOther)) {
|
|
throw Error(CreateLocalRepositoryOperation::tr("Could not set file permissions %1!")
|
|
.arg(it.filePath()));
|
|
}
|
|
}
|
|
}
|
|
|
|
static void removeDirectory(const QString &path, AutoHelper *const helper)
|
|
{
|
|
QInstaller::removeDirectory(path);
|
|
QStringList files = helper->m_files.filter(path);
|
|
foreach (const QString &file, files)
|
|
helper->m_files.removeAll(file);
|
|
}
|
|
|
|
static void removeFiles(const QString &path, AutoHelper *const helper)
|
|
{
|
|
const QFileInfoList entries = QDir(path).entryInfoList(QDir::AllEntries | QDir::Hidden);
|
|
foreach (const QFileInfo &fi, entries) {
|
|
if (fi.isSymLink() || fi.isFile()) {
|
|
QFile f(fi.filePath());
|
|
if (!f.remove()) {
|
|
throw Error(CreateLocalRepositoryOperation::tr("Could not remove file %1: %2")
|
|
.arg(f.fileName(), f.errorString()));
|
|
}
|
|
helper->m_files.removeAll(f.fileName());
|
|
}
|
|
}
|
|
}
|
|
|
|
static QString createArchive(const QString repoPath, const QString &sourceDir, const QString &version
|
|
, AutoHelper *const helper)
|
|
{
|
|
const QString fileName = QString::fromLatin1("/%1meta.7z").arg(version);
|
|
|
|
QFile archive(repoPath + fileName);
|
|
QInstaller::openForWrite(&archive);
|
|
Lib7z::createArchive(&archive, QStringList() << sourceDir);
|
|
removeFiles(sourceDir, helper); // cleanup the files we compressed
|
|
if (!archive.rename(sourceDir + fileName)) {
|
|
throw Error(CreateLocalRepositoryOperation::tr("Could not move file %1 to %2. Error: %3")
|
|
.arg(archive.fileName(), sourceDir + fileName, archive.errorString()));
|
|
}
|
|
return archive.fileName();
|
|
}
|
|
|
|
} // namespace Statics
|
|
|
|
|
|
// -- CreateLocalRepositoryOperation
|
|
|
|
CreateLocalRepositoryOperation::CreateLocalRepositoryOperation()
|
|
{
|
|
setName(QLatin1String("CreateLocalRepository"));
|
|
}
|
|
|
|
void CreateLocalRepositoryOperation::backup()
|
|
{
|
|
}
|
|
|
|
bool CreateLocalRepositoryOperation::performOperation()
|
|
{
|
|
AutoHelper helper(this);
|
|
emit progressChanged(0.0);
|
|
|
|
const QStringList args = arguments();
|
|
|
|
if (args.count() != 2) {
|
|
setError(InvalidArguments);
|
|
setErrorString(tr("Invalid arguments in %0: %1 arguments given, %2 expected%3.")
|
|
.arg(name()).arg(arguments().count()).arg(tr("exactly 2"), QLatin1String("")));
|
|
return false;
|
|
}
|
|
|
|
try {
|
|
const QString binaryPath = QFileInfo(args.at(0)).absoluteFilePath();
|
|
// Note the "/" at the end, important to make copy directory operation behave well
|
|
const QString repoPath = QFileInfo(args.at(1)).absoluteFilePath() + QLatin1Char('/');
|
|
|
|
// check if this is an offline version, otherwise there will be no binary data
|
|
PackageManagerCore *const core = value(QLatin1String("installer")).value<PackageManagerCore*>();
|
|
if (core && !core->isOfflineOnly()) {
|
|
throw QInstaller::Error(tr("Installer needs to be an offline version: %1.")
|
|
.arg(QDir::toNativeSeparators(binaryPath)));
|
|
}
|
|
|
|
// if we're running as installer and install into an existing target, remove possible previous repos
|
|
if (QFile::exists(repoPath)) {
|
|
Static::fixPermissions(repoPath);
|
|
QInstaller::removeDirectory(repoPath);
|
|
}
|
|
|
|
// create the local repository target dir
|
|
KDUpdater::MkdirOperation mkDirOp;
|
|
mkDirOp.setArguments(QStringList() << repoPath);
|
|
mkDirOp.backup();
|
|
if (!mkDirOp.performOperation()) {
|
|
setError(mkDirOp.error());
|
|
setErrorString(mkDirOp.errorString());
|
|
return false;
|
|
}
|
|
setValue(QLatin1String("createddir"), mkDirOp.value(QLatin1String("createddir")));
|
|
|
|
// copy the whole meta data into local repository
|
|
CopyDirectoryOperation copyDirOp;
|
|
copyDirOp.setArguments(QStringList() << QLatin1String(":/metadata/") << repoPath);
|
|
connect(©DirOp, SIGNAL(outputTextChanged(QString)), this, SIGNAL(outputTextChanged(QString)));
|
|
|
|
const bool success = copyDirOp.performOperation();
|
|
helper.m_files = copyDirOp.value(QLatin1String("files")).toStringList();
|
|
if (!success) {
|
|
setError(copyDirOp.error());
|
|
setErrorString(copyDirOp.errorString());
|
|
return false;
|
|
}
|
|
|
|
emit progressChanged(0.25);
|
|
|
|
// we need to fix the folder and file permissions here, as copying from read only resource file
|
|
// systems sets all permissions to a completely bogus value...
|
|
Static::fixPermissions(repoPath);
|
|
|
|
// open the updates xml file we previously copied
|
|
QFile updatesXml(repoPath + QLatin1String("Updates.xml"));
|
|
if (!updatesXml.exists() || !updatesXml.open(QIODevice::ReadOnly))
|
|
throw QInstaller::Error(tr("Could not open file: %1").arg(updatesXml.fileName()));
|
|
|
|
// read the content of the updates xml
|
|
QString error;
|
|
QDomDocument doc;
|
|
if (!doc.setContent(&updatesXml, &error))
|
|
throw QInstaller::Error(tr("Could not read: %1. Error: %2").arg(updatesXml.fileName(), error));
|
|
|
|
// build for each available package a name - version mapping
|
|
QHash<QString, QString> nameVersionHash;
|
|
const QDomElement root = doc.documentElement();
|
|
const QDomNodeList rootChildNodes = root.childNodes();
|
|
for (int i = 0; i < rootChildNodes.count(); ++i) {
|
|
const QDomElement element = rootChildNodes.at(i).toElement();
|
|
if (element.isNull())
|
|
continue;
|
|
|
|
QString name, version;
|
|
if (element.tagName() == QLatin1String("PackageUpdate")) {
|
|
const QDomNodeList elementChildNodes = element.childNodes();
|
|
for (int j = 0; j < elementChildNodes.count(); ++j) {
|
|
const QDomElement e = elementChildNodes.at(j).toElement();
|
|
if (e.tagName() == QLatin1String("Name"))
|
|
name = e.text();
|
|
else if (e.tagName() == QLatin1String("Version"))
|
|
version = e.text();
|
|
}
|
|
if (ProductKeyCheck::instance()->isValidPackage(name))
|
|
nameVersionHash.insert(name, version);
|
|
}
|
|
}
|
|
|
|
emit progressChanged(0.50);
|
|
|
|
QFile file(binaryPath);
|
|
if (!file.open(QIODevice::ReadOnly)) {
|
|
throw QInstaller::Error(tr("Could not open file: %1. Error: %2").arg(file.fileName(),
|
|
file.errorString()));
|
|
}
|
|
|
|
// start to read the binary layout
|
|
ResourceCollectionManager manager;
|
|
BinaryContent::readBinaryContent(&file, 0, &manager, 0, BinaryContent::MagicCookie);
|
|
|
|
emit progressChanged(0.65);
|
|
|
|
QDir repo(repoPath);
|
|
if (!nameVersionHash.isEmpty()) {
|
|
// extract meta and binary data
|
|
|
|
const QStringList names = nameVersionHash.keys();
|
|
for (int i = 0; i < names.count(); ++i) {
|
|
const QString name = names.at(i);
|
|
if (!repo.mkpath(name)) {
|
|
throw QInstaller::Error(tr("Could not create target dir: %1.")
|
|
.arg(repo.filePath(name)));
|
|
}
|
|
// zip the meta files that come with the offline installer
|
|
helper.m_files.prepend(Static::createArchive(repoPath,
|
|
repo.filePath(name), nameVersionHash.value(name), &helper));
|
|
emit outputTextChanged(helper.m_files.first());
|
|
|
|
// copy the 7z files that are inside the component index into the target
|
|
const ResourceCollection collection = manager.collectionByName(name.toUtf8());
|
|
foreach (const QSharedPointer<Resource> &resource, collection.resources()) {
|
|
const bool isOpen = resource->isOpen();
|
|
if ((!isOpen) && (!resource->open()))
|
|
continue;
|
|
|
|
QFile target(repo.filePath(name) + QDir::separator()
|
|
+ QString::fromUtf8(resource->name()));
|
|
QInstaller::openForWrite(&target);
|
|
resource->copyData(&target);
|
|
helper.m_files.prepend(target.fileName());
|
|
emit outputTextChanged(helper.m_files.first());
|
|
|
|
if (!isOpen) // If we reach that point, either the resource was opened already.
|
|
resource->close(); // or we did open it and have to close it again.
|
|
}
|
|
emit progressChanged(.65f + ((double(i) / double(names.count())) * .25f));
|
|
}
|
|
}
|
|
|
|
emit progressChanged(0.95);
|
|
|
|
try {
|
|
// remove these, if we fail it doesn't hurt
|
|
Static::removeDirectory(QDir::cleanPath(repoPath + QLatin1String("/installer-config")),
|
|
&helper);
|
|
Static::removeDirectory(QDir::cleanPath(repoPath + QLatin1String("/config")), &helper);
|
|
const QStringList files = repo.entryList(QStringList() << QLatin1String("*.qrc"), QDir::Files);
|
|
foreach (const QString &file, files) {
|
|
if (repo.remove(file))
|
|
helper.m_files.removeAll(QDir::cleanPath(repoPath + file));
|
|
}
|
|
} catch (...) {}
|
|
setValue(QLatin1String("local-repo"), repoPath);
|
|
} catch (const Lib7z::SevenZipException &e) {
|
|
setError(UserDefinedError);
|
|
setErrorString(e.message());
|
|
return false;
|
|
} catch (const QInstaller::Error &e) {
|
|
setError(UserDefinedError);
|
|
setErrorString(e.message());
|
|
return false;
|
|
} catch (...) {
|
|
setError(UserDefinedError);
|
|
setErrorString(tr("Unknown exception caught: %1.").arg(QLatin1String(Q_FUNC_INFO)));
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool CreateLocalRepositoryOperation::undoOperation()
|
|
{
|
|
Q_ASSERT(arguments().count() == 2);
|
|
|
|
AutoHelper _(this);
|
|
emit progressChanged(0.0);
|
|
|
|
QDir dir;
|
|
const QStringList files = value(QLatin1String("files")).toStringList();
|
|
foreach (const QString &file, files) {
|
|
emit outputTextChanged(tr("Removing file: %0").arg(file));
|
|
if (!QFile::remove(file)) {
|
|
setError(InvalidArguments);
|
|
setErrorString(tr("Could not remove %0.").arg(file));
|
|
return false;
|
|
}
|
|
dir.rmpath(QFileInfo(file).absolutePath());
|
|
}
|
|
setValue(QLatin1String("files"), QStringList());
|
|
|
|
QDir createdDir = QDir(value(QLatin1String("createddir")).toString());
|
|
if (createdDir == QDir::root() || !createdDir.exists())
|
|
return true;
|
|
|
|
QInstaller::removeSystemGeneratedFiles(createdDir.path());
|
|
|
|
errno = 0;
|
|
const bool result = QDir::root().rmdir(createdDir.path());
|
|
if (!result) {
|
|
#if defined(Q_OS_WIN) && !defined(Q_CC_MINGW)
|
|
char msg[128];
|
|
if (strerror_s(msg, sizeof msg, errno) != 0) {
|
|
setError(UserDefinedError, tr("Cannot remove directory %1: %2").arg(createdDir.path(),
|
|
QString::fromLocal8Bit(msg)));
|
|
}
|
|
#else
|
|
setError(UserDefinedError, tr("Cannot remove directory %1: %2").arg(createdDir.path(),
|
|
QString::fromLocal8Bit(strerror(errno))));
|
|
#endif
|
|
}
|
|
setValue(QLatin1String("files"), QStringList());
|
|
|
|
return result;
|
|
}
|
|
|
|
bool CreateLocalRepositoryOperation::testOperation()
|
|
{
|
|
return true;
|
|
}
|
|
|
|
Operation *CreateLocalRepositoryOperation::clone() const
|
|
{
|
|
return new CreateLocalRepositoryOperation();
|
|
}
|
|
|
|
void CreateLocalRepositoryOperation::emitFullProgress()
|
|
{
|
|
emit progressChanged(1.0);
|
|
}
|
|
|
|
} // namespace QInstaller
|