summaryrefslogtreecommitdiffstats
path: root/src/libs/installer/installercalculator.cpp
diff options
context:
space:
mode:
authorkh1 <karsten.heimrich@digia.com>2014-10-01 22:37:09 +0200
committerKarsten Heimrich <karsten.heimrich@digia.com>2014-10-06 16:08:23 +0200
commitf288f29e92102f7779101a59515f86bce64a3713 (patch)
treef44ad581f551f272fb03610c401bddc73b9f10d3 /src/libs/installer/installercalculator.cpp
parent8941d7affa7dddf6780e79172bcb3c0d67fca128 (diff)
Move InstallerCalculator into its own header
Make it public, so that it can be used outside (e.g. in tests) Change-Id: I2a307dcd6f2eb017f9ac26de06d77deb0d668fbe Reviewed-by: Jarek Kobus <jaroslaw.kobus@digia.com>
Diffstat (limited to 'src/libs/installer/installercalculator.cpp')
-rw-r--r--src/libs/installer/installercalculator.cpp219
1 files changed, 219 insertions, 0 deletions
diff --git a/src/libs/installer/installercalculator.cpp b/src/libs/installer/installercalculator.cpp
new file mode 100644
index 000000000..2fe3c9915
--- /dev/null
+++ b/src/libs/installer/installercalculator.cpp
@@ -0,0 +1,219 @@
+/**************************************************************************
+**
+** Copyright (C) 2014 Digia Plc and/or its subsidiary(-ies).
+** Contact: http://www.qt-project.org/legal
+**
+** 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 Digia. For licensing terms and
+** conditions see http://qt.digia.com/licensing. For further information
+** use the contact form at http://qt.digia.com/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 as published by the Free Software
+** Foundation and appearing in the file LICENSE.LGPL included in the
+** packaging of this file. Please review the following information to
+** ensure the GNU Lesser General Public License version 2.1 requirements
+** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
+**
+** In addition, as a special exception, Digia gives you certain additional
+** rights. These rights are described in the Digia Qt LGPL Exception
+** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
+**
+** GNU General Public License Usage
+** Alternatively, this file may be used under the terms of the GNU
+** General Public License version 3.0 as published by the Free Software
+** Foundation and appearing in the file LICENSE.GPL included in the
+** packaging of this file. Please review the following information to
+** ensure the GNU General Public License version 3.0 requirements will be
+** met: http://www.gnu.org/copyleft/gpl.html.
+**
+**
+** $QT_END_LICENSE$
+**
+**************************************************************************/
+
+#include "installercalculator.h"
+
+#include "component.h"
+#include "packagemanagercore.h"
+
+#include <QDebug>
+
+namespace QInstaller {
+
+InstallerCalculator::InstallerCalculator(const QList<Component *> &allComponents)
+ : m_allComponents(allComponents)
+{
+}
+
+void InstallerCalculator::insertInstallReason(Component *component,
+ InstallReasonType installReason, const QString &referencedComponentName)
+{
+ // keep the first reason
+ if (m_toInstallComponentIdReasonHash.contains(component->name()))
+ return;
+ m_toInstallComponentIdReasonHash.insert(component->name(),
+ qMakePair(installReason, referencedComponentName));
+}
+
+InstallerCalculator::InstallReasonType InstallerCalculator::installReasonType(Component *c) const
+{
+ return m_toInstallComponentIdReasonHash.value(c->name(),
+ qMakePair(InstallerCalculator::Selected, QString())).first;
+}
+
+QString InstallerCalculator::installReasonReferencedComponent(Component *component) const
+{
+ return m_toInstallComponentIdReasonHash.value(component->name(),
+ qMakePair(InstallerCalculator::Selected, QString())).second;
+}
+
+QString InstallerCalculator::installReason(Component *component) const
+{
+ InstallerCalculator::InstallReasonType reason = installReasonType(component);
+ switch (reason) {
+ case Automatic:
+ return QCoreApplication::translate("InstallerCalculator",
+ "Components added as automatic dependencies:");
+ case Dependent:
+ return QCoreApplication::translate("InstallerCalculator", "Components added as "
+ "dependency for '%1':").arg(installReasonReferencedComponent(component));
+ case Resolved:
+ return QCoreApplication::translate("InstallerCalculator",
+ "Components that have resolved dependencies:");
+ case Selected:
+ return QCoreApplication::translate("InstallerCalculator",
+ "Selected components without dependencies:");
+ }
+ return QString();
+}
+
+QList<Component*> InstallerCalculator::orderedComponentsToInstall() const
+{
+ return m_orderedComponentsToInstall;
+}
+
+QString InstallerCalculator::componentsToInstallError() const
+{
+ return m_componentsToInstallError;
+}
+
+void InstallerCalculator::realAppendToInstallComponents(Component *component)
+{
+ if (!component->isInstalled() || component->updateRequested()) {
+ // remove the checkState method if we don't use selected in scripts
+ component->setCheckState(Qt::Checked);
+
+ m_orderedComponentsToInstall.append(component);
+ m_toInstallComponentIds.insert(component->name());
+ }
+}
+
+QString InstallerCalculator::recursionError(Component *component)
+{
+ return QCoreApplication::translate("InstallerCalculator", "Recursion detected, component '%1' "
+ "already added with reason: '%2'").arg(component->name(), installReason(component));
+}
+
+bool InstallerCalculator::appendComponentsToInstall(const QList<Component *> &components)
+{
+ if (components.isEmpty())
+ return true;
+
+ QList<Component*> notAppendedComponents; // for example components with unresolved dependencies
+ foreach (Component *component, components){
+ if (m_toInstallComponentIds.contains(component->name())) {
+ const QString errorMessage = recursionError(component);
+ qWarning() << errorMessage;
+ m_componentsToInstallError.append(errorMessage);
+ Q_ASSERT_X(!m_toInstallComponentIds.contains(component->name()), Q_FUNC_INFO,
+ qPrintable(errorMessage));
+ return false;
+ }
+
+ if (component->dependencies().isEmpty())
+ realAppendToInstallComponents(component);
+ else
+ notAppendedComponents.append(component);
+ }
+
+ foreach (Component *component, notAppendedComponents) {
+ if (!appendComponentToInstall(component))
+ return false;
+ }
+
+ QList<Component *> foundAutoDependOnList;
+ // All regular dependencies are resolved. Now we are looking for auto depend on components.
+ foreach (Component *component, m_allComponents) {
+ // If a components is already installed or is scheduled for installation, no need to check
+ // for auto depend installation.
+ if ((!component->isInstalled() || component->updateRequested())
+ && !m_toInstallComponentIds.contains(component->name())) {
+ // If we figure out a component requests auto installation, keep it to resolve
+ // their dependencies as well.
+ if (component->isAutoDependOn(m_toInstallComponentIds)) {
+ foundAutoDependOnList.append(component);
+ insertInstallReason(component, InstallerCalculator::Automatic);
+ }
+ }
+ }
+
+ if (!foundAutoDependOnList.isEmpty())
+ return appendComponentsToInstall(foundAutoDependOnList);
+ return true;
+}
+
+bool InstallerCalculator::appendComponentToInstall(Component *component)
+{
+ QSet<QString> allDependencies = component->dependencies().toSet();
+
+ foreach (const QString &dependencyComponentName, allDependencies) {
+ // PackageManagerCore::componentByName returns 0 if dependencyComponentName contains a
+ // version which is not available
+ Component *dependencyComponent =
+ PackageManagerCore::componentByName(dependencyComponentName, m_allComponents);
+ if (!dependencyComponent) {
+ const QString errorMessage = QCoreApplication::translate("InstallerCalculator",
+ "Cannot find missing dependency '%1' for '%2'.").arg(dependencyComponentName,
+ component->name());
+ qWarning() << errorMessage;
+ m_componentsToInstallError.append(errorMessage);
+ return false;
+ }
+
+ if ((!dependencyComponent->isInstalled() || dependencyComponent->updateRequested())
+ && !m_toInstallComponentIds.contains(dependencyComponent->name())) {
+ if (m_visitedComponents.value(component).contains(dependencyComponent)) {
+ const QString errorMessage = recursionError(component);
+ qWarning() << errorMessage;
+ m_componentsToInstallError = errorMessage;
+ Q_ASSERT_X(!m_visitedComponents.value(component).contains(dependencyComponent),
+ Q_FUNC_INFO, qPrintable(errorMessage));
+ return false;
+ }
+ m_visitedComponents[component].insert(dependencyComponent);
+
+ // add needed dependency components to the next run
+ insertInstallReason(dependencyComponent, InstallerCalculator::Dependent,
+ component->name());
+
+ if (!appendComponentToInstall(dependencyComponent))
+ return false;
+ }
+ }
+
+ if (!m_toInstallComponentIds.contains(component->name())) {
+ realAppendToInstallComponents(component);
+ insertInstallReason(component, InstallerCalculator::Resolved);
+ }
+ return true;
+}
+
+} // namespace QInstaller