/************************************************************************** ** ** Copyright (C) 2020 The Qt Company Ltd. ** Contact: https://www.qt.io/licensing/ ** ** This file is part of the Qt Installer Framework. ** ** $QT_BEGIN_LICENSE:GPL-EXCEPT$ ** 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 https://www.qt.io/terms-conditions. For further ** information use the contact form at https://www.qt.io/contact-us. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 3 as published by the Free Software ** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT ** included in the packaging of this file. Please review the following ** information to ensure the GNU General Public License requirements will ** be met: https://www.gnu.org/licenses/gpl-3.0.html. ** ** $QT_END_LICENSE$ ** **************************************************************************/ #ifndef SDKAPP_H #define SDKAPP_H #include "commandlineparser.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include template class SDKApp : public T { public: SDKApp(int& argc, char** argv) : T(argc, argv), m_runCheck(QDir::tempPath() + QLatin1Char('/') + qApp->applicationName() + QLatin1String("1234865.lock")) { m_parser.parse(QCoreApplication::arguments()); } virtual ~SDKApp() { foreach (const QByteArray &ba, m_resourceMappings) QResource::unregisterResource((const uchar*) ba.data(), QLatin1String(":/metadata")); } bool notify(QObject *receiver, QEvent *event) { try { return T::notify(receiver, event); } catch (std::exception &e) { qFatal("Exception thrown: %s", e.what()); } catch (...) { qFatal("Unknown exception caught."); } return false; } bool init(QString &errorMessage) { QString appname = qApp->applicationName(); if (m_runCheck.isRunning(RunOnceChecker::ConditionFlag::Lockfile)) { // It is possible to install an application and thus the maintenance tool into a // directory that requires elevated permission to create a lock file. Since this // cannot be done without requesting credentials from the user, we silently ignore // the fact that we could not create the lock file and check the running processes. if (m_runCheck.isRunning(RunOnceChecker::ConditionFlag::ProcessList)) { errorMessage = QObject::tr("Another %1 instance is already running. Wait " "until it finishes, close it, or restart your system.").arg(qAppName()); return false; } } QFile binary(binaryFile()); #ifdef Q_OS_WIN // On some admin user installations it is possible that the installer.dat // file is left without reading permissions for non-administrator users, // we should check this and prompt the user to run the executable as admin if needed. if (!binary.open(QIODevice::ReadOnly)) { QFileInfo binaryInfo(binary.fileName()); errorMessage = QObject::tr("Please make sure that the current user has reading access " "to file \"%1\" or try running %2 as an administrator.").arg(binaryInfo.fileName(), qAppName()); return false; } binary.close(); #endif QString fileName = datFile(binaryFile()); quint64 cookie = QInstaller::BinaryContent::MagicCookieDat; if (fileName.isEmpty()) { fileName = binaryFile(); cookie = QInstaller::BinaryContent::MagicCookie; } binary.setFileName(fileName); QInstaller::openForRead(&binary); qint64 magicMarker; QInstaller::ResourceCollectionManager manager; QList oldOperations; QInstaller::BinaryContent::readBinaryContent(&binary, &oldOperations, &manager, &magicMarker, cookie); // Usually resources simply get mapped into memory and therefore the file does not need to be // kept open during application runtime. Though in case of offline installers we need to access // the appended binary content (packages etc.), so we close only in maintenance mode. if (magicMarker != QInstaller::BinaryContent::MagicInstallerMarker) binary.close(); bool isCommandLineInterface = false; foreach (const QString &option, CommandLineOptions::scCommandLineInterfaceOptions) { if (m_parser.positionalArguments().contains(option)) { isCommandLineInterface = true; break; } } if (m_parser.isSet(CommandLineOptions::scDeprecatedCheckUpdates)) isCommandLineInterface = true; QString loggingRules; if (m_parser.isSet(CommandLineOptions::scLoggingRulesLong)) { loggingRules = QLatin1String("ifw.* = false\n"); loggingRules += m_parser.value(CommandLineOptions::scLoggingRulesLong) .split(QLatin1Char(','), QString::SkipEmptyParts) .join(QLatin1Char('\n')); // take rules from command line } else if (isCommandLineInterface) { loggingRules = QLatin1String("ifw.* = false\n" "ifw.installer.* = true\n" "ifw.server = true\n" "ifw.progress.indicator = true\n" "ifw.package.* = true\n"); } else { // enable all except detailed package information and developer specific logging loggingRules = QLatin1String("ifw.* = true\n" "ifw.developer.build = false\n" "ifw.package.* = true\n"); } if (QInstaller::verboseLevel() == QInstaller::VerbosityLevel::Detailed) { loggingRules += QLatin1String("\nifw.developer.build = true\n" "ifw.package.* = true\n"); } QLoggingCategory::setFilterRules(loggingRules); qCDebug(QInstaller::lcInstallerInstallLog).noquote() << "Arguments:" << QCoreApplication::arguments().join(QLatin1String(", ")); dumpResourceTree(); SDKApp::registerMetaResources(manager.collectionByName("QResources")); QInstaller::BinaryFormatEngineHandler::instance()->registerResources(manager.collections()); const QHash userArgs = userArguments(); if (m_parser.isSet(CommandLineOptions::scStartClientLong)) { const QStringList arguments = m_parser.value(CommandLineOptions::scStartClientLong) .split(QLatin1Char(','), QString::SkipEmptyParts); m_core = new QInstaller::PackageManagerCore( magicMarker, oldOperations, arguments.value(0, QLatin1String(QInstaller::Protocol::DefaultSocket)), arguments.value(1, QLatin1String(QInstaller::Protocol::DefaultAuthorizationKey)), QInstaller::Protocol::Mode::Debug, userArgs, isCommandLineInterface); } else { m_core = new QInstaller::PackageManagerCore(magicMarker, oldOperations, QUuid::createUuid().toString(), QUuid::createUuid().toString(), QInstaller::Protocol::Mode::Production, userArgs, isCommandLineInterface); } // From Qt5.8 onwards system proxy is used by default. If Qt is built with // QT_USE_SYSTEM_PROXIES false then system proxies are not used by default. if (m_parser.isSet(CommandLineOptions::scNoProxyLong)) { m_core->settings().setProxyType(QInstaller::Settings::NoProxy); KDUpdater::FileDownloaderFactory::instance().setProxyFactory(m_core->proxyFactory()); } else if (QNetworkProxyFactory::usesSystemConfiguration()) { m_core->settings().setProxyType(QInstaller::Settings::SystemProxy); KDUpdater::FileDownloaderFactory::instance().setProxyFactory(m_core->proxyFactory()); } if (m_parser.isSet(CommandLineOptions::scShowVirtualComponentsLong)) QInstaller::PackageManagerCore::setVirtualComponentsVisible(true); // IFW 3.x.x style --updater option support provided for backward compatibility if (m_parser.isSet(CommandLineOptions::scStartUpdaterLong) || m_parser.isSet(CommandLineOptions::scDeprecatedUpdater)) { if (m_core->isInstaller()) { errorMessage = QObject::tr("Cannot start installer binary as updater."); return false; } m_core->setUserSetBinaryMarker(QInstaller::BinaryContent::MagicUpdaterMarker); } if (m_parser.isSet(CommandLineOptions::scStartPackageManagerLong)) { if (m_core->isInstaller()) { errorMessage = QObject::tr("Cannot start installer binary as package manager."); return false; } m_core->setUserSetBinaryMarker(QInstaller::BinaryContent::MagicPackageManagerMarker); } if (m_parser.isSet(CommandLineOptions::scStartUninstallerLong)) { if (m_core->isInstaller()) { errorMessage = QObject::tr("Cannot start installer binary as uninstaller."); return false; } m_core->setUserSetBinaryMarker(QInstaller::BinaryContent::MagicUninstallerMarker); } if (m_parser.isSet(CommandLineOptions::scAddRepositoryLong)) { const QStringList repoList = repositories(m_parser.value(CommandLineOptions::scAddRepositoryLong)); if (repoList.isEmpty()) { errorMessage = QObject::tr("Empty repository list for option 'addRepository'."); return false; } m_core->addUserRepositories(repoList); } if (m_parser.isSet(CommandLineOptions::scAddTmpRepositoryLong)) { const QStringList repoList = repositories(m_parser.value(CommandLineOptions::scAddTmpRepositoryLong)); if (repoList.isEmpty()) { errorMessage = QObject::tr("Empty repository list for option 'addTempRepository'."); return false; } m_core->setTemporaryRepositories(repoList, false); } if (m_parser.isSet(CommandLineOptions::scSetTmpRepositoryLong)) { const QStringList repoList = repositories(m_parser.value(CommandLineOptions::scSetTmpRepositoryLong)); if (repoList.isEmpty()) { errorMessage = QObject::tr("Empty repository list for option 'setTempRepository'."); return false; } m_core->setTemporaryRepositories(repoList, true); } if (m_parser.isSet(CommandLineOptions::scInstallCompressedRepositoryLong)) { const QStringList repoList = repositories(m_parser.value(CommandLineOptions::scInstallCompressedRepositoryLong)); if (repoList.isEmpty()) { errorMessage = QObject::tr("Empty repository list for option 'installCompressedRepository'."); return false; } foreach (QString repository, repoList) { if (!QFileInfo::exists(repository)) { errorMessage = QObject::tr("The file %1 does not exist.").arg(repository); return false; } } m_core->setTemporaryRepositories(repoList, false, true); } // Disable checking for free space on target if (m_parser.isSet(CommandLineOptions::scNoSizeCheckingLong)) m_core->setCheckAvailableSpace(false); QInstaller::PackageManagerCore::setNoForceInstallation(m_parser .isSet(CommandLineOptions::scNoForceInstallationLong)); QInstaller::PackageManagerCore::setNoDefaultInstallation(m_parser .isSet(CommandLineOptions::scNoDefaultInstallationLong)); QInstaller::PackageManagerCore::setCreateLocalRepositoryFromBinary(m_parser .isSet(CommandLineOptions::scCreateLocalRepositoryLong) || m_core->settings().createLocalRepository()); if (m_parser.isSet(CommandLineOptions::scAcceptLicensesLong)) m_core->setAutoAcceptLicenses(); if (m_parser.isSet(CommandLineOptions::scConfirmCommandLong)) m_core->setAutoConfirmCommand(); // Ignore message acceptance options when running the installer with GUI if (m_core->isCommandLineInstance()) { if (m_parser.isSet(CommandLineOptions::scAcceptMessageQueryLong)) m_core->autoAcceptMessageBoxes(); if (m_parser.isSet(CommandLineOptions::scRejectMessageQueryLong)) m_core->autoRejectMessageBoxes(); if (m_parser.isSet(CommandLineOptions::scMessageDefaultAnswerLong)) m_core->acceptMessageBoxDefaultButton(); if (m_parser.isSet(CommandLineOptions::scMessageAutomaticAnswerLong)) { const QString positionalArguments = m_parser.value(CommandLineOptions::scMessageAutomaticAnswerLong); const QStringList items = positionalArguments.split(QLatin1Char(','), QString::SkipEmptyParts); if (items.count() > 0) { errorMessage = setMessageBoxAutomaticAnswers(items); if (!errorMessage.isEmpty()) return false; } else { errorMessage = QObject::tr("Arguments missing for option %1") .arg(CommandLineOptions::scMessageAutomaticAnswerLong); return false; } } if (m_parser.isSet(CommandLineOptions::scFileDialogAutomaticAnswer)) { const QString positionalArguments = m_parser.value(CommandLineOptions::scFileDialogAutomaticAnswer); const QStringList items = positionalArguments.split(QLatin1Char(','), QString::SkipEmptyParts); foreach (const QString &item, items) { if (item.contains(QLatin1Char('='))) { const QString name = item.section(QLatin1Char('='), 0, 0); QString value = item.section(QLatin1Char('='), 1, 1); m_core->setFileDialogAutomaticAnswer(name, value); } } } } try { ProductKeyCheck::instance()->init(m_core); } catch (const QInstaller::Error &e) { errorMessage = e.message(); return false; } ProductKeyCheck::instance()->addPackagesFromXml(QLatin1String(":/metadata/Updates.xml")); return true; } /*! Returns the installer binary or installer.dat. In case of an installer this will be the installer binary itself, which contains the binary layout and the binary content. In case of an maintenance tool, it will return a .dat file that has just a binary layout as the binary layout cannot be appended to the actual maintenance tool binary itself because of signing. On macOS: This function will return always the .dat file .dat file is located inside the resource folder in the application bundle in macOS. */ QString binaryFile() const { QString binaryFile = QCoreApplication::applicationFilePath(); // The installer binary on macOS and Windows does not contain the binary // content, it's put into the resources folder as separate file. // Adjust the actual binary path. No error checking here since we // will fail later while reading the binary content. QDir resourcePath(QFileInfo(binaryFile).dir()); #ifdef Q_OS_MACOS resourcePath.cdUp(); resourcePath.cd(QLatin1String("Resources")); #endif QString datFilePath = resourcePath.filePath(QLatin1String("installer.dat")); if (QFileInfo::exists(datFilePath)) return datFilePath; return binaryFile; } /*! Returns the corresponding .dat file for a given installer / maintenance tool binary or an empty string if it fails to find one. */ QString datFile(const QString &binaryFile) const { QFile file(binaryFile); QInstaller::openForRead(&file); const quint64 cookiePos = QInstaller::BinaryContent::findMagicCookie(&file, QInstaller::BinaryContent::MagicCookie); if (!file.seek(cookiePos - sizeof(qint64))) // seek to read the marker return QString(); // ignore error, we will fail later const qint64 magicMarker = QInstaller::retrieveInt64(&file); if (magicMarker == QInstaller::BinaryContent::MagicUninstallerMarker) { QFileInfo fi(binaryFile); QString bundlePath; if (QInstaller::isInBundle(fi.absoluteFilePath(), &bundlePath)) fi.setFile(bundlePath); #ifdef Q_OS_MACOS return fi.absoluteDir().filePath(fi.baseName() + QLatin1String(".dat")); #else return fi.absoluteDir().filePath(qApp->applicationName() + QLatin1String(".dat")); #endif } return QString(); } void registerMetaResources(const QInstaller::ResourceCollection &collection) { foreach (const QSharedPointer &resource, collection.resources()) { const bool isOpen = resource->isOpen(); if ((!isOpen) && (!resource->open())) continue; if (!resource->seek(0)) continue; const QByteArray ba = resource->readAll(); if (ba.isEmpty()) continue; if (QResource::registerResource((const uchar*) ba.data(), QLatin1String(":/metadata"))) m_resourceMappings.append(ba); 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. } } QStringList repositories(const QString &list) const { const QStringList items = list.split(QLatin1Char(','), QString::SkipEmptyParts); foreach (const QString &item, items) qCDebug(QInstaller::lcInstallerInstallLog) << "Adding custom repository:" << item; return items; } QHash userArguments() { QHash params; const QStringList positionalArguments = m_parser.positionalArguments(); foreach (const QString &argument, positionalArguments) { if (argument.contains(QLatin1Char('='))) { const QString name = argument.section(QLatin1Char('='), 0, 0); const QString value = argument.section(QLatin1Char('='), 1, 1); params.insert(name, value); } } return params; } QString setMessageBoxAutomaticAnswers(const QStringList &items) { foreach (const QString &item, items) { if (item.contains(QLatin1Char('='))) { const QMetaObject metaObject = QMessageBox::staticMetaObject; int enumIndex = metaObject.indexOfEnumerator("StandardButton"); if (enumIndex != -1) { QMetaEnum en = metaObject.enumerator(enumIndex); const QString name = item.section(QLatin1Char('='), 0, 0); QString value = item.section(QLatin1Char('='), 1, 1); value.prepend(QLatin1String("QMessageBox::")); bool ok = false; int buttonValue = en.keyToValue(value.toLocal8Bit().data(), &ok); if (ok) m_core->setMessageBoxAutomaticAnswer(name, buttonValue); else return QObject::tr("Invalid button value %1 ").arg(value); } } else { return QObject::tr("Incorrect arguments for %1") .arg(CommandLineOptions::scMessageAutomaticAnswerLong); } } return QString(); } void dumpResourceTree() const { qCDebug(QInstaller::lcDeveloperBuild) << "Resource tree:"; QDirIterator it(QLatin1String(":/"), QDir::NoDotAndDotDot | QDir::AllEntries | QDir::Hidden, QDirIterator::Subdirectories); while (it.hasNext()) { if (it.next().startsWith(QLatin1String(":/qt-project.org"))) continue; qCDebug(QInstaller::lcDeveloperBuild) << " " << it.filePath().toUtf8().constData(); } } private: QList m_resourceMappings; public: QInstaller::PackageManagerCore *m_core; CommandLineParser m_parser; RunOnceChecker m_runCheck; }; #endif // SDKAPP_H