mirror of https://github.com/cutefishos/calamares
You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
153 lines
4.5 KiB
C++
153 lines
4.5 KiB
C++
/* === This file is part of Calamares - <http://github.com/calamares> ===
|
|
*
|
|
* Copyright 2019, Adriaan de Groot <groot@kde.org>
|
|
*
|
|
* Calamares is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* Calamares is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with Calamares. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "RequirementsChecker.h"
|
|
|
|
#include "Module.h"
|
|
#include "Requirement.h"
|
|
|
|
#include "utils/Logger.h"
|
|
|
|
#include <algorithm>
|
|
|
|
#include <QtConcurrent/QtConcurrent>
|
|
#include <QFuture>
|
|
#include <QFutureWatcher>
|
|
#include <QTimer>
|
|
|
|
|
|
namespace Calamares
|
|
{
|
|
|
|
static void
|
|
registerMetatypes()
|
|
{
|
|
static bool done = false;
|
|
|
|
if ( !done )
|
|
{
|
|
qRegisterMetaType< RequirementEntry >( "RequirementEntry" );
|
|
// It's sensitive to the names of types in parameters; in particular
|
|
// althrough QList<RequirementEntry> is the same as RequirementsList,
|
|
// because we *name* the type as RequirementsList in the parameters,
|
|
// we need to register that (as well). Here, be safe and register
|
|
// both names.
|
|
qRegisterMetaType< QList< RequirementEntry > >( "QList<RequirementEntry>" );
|
|
qRegisterMetaType< RequirementsList >( "RequirementsList" );
|
|
done = true;
|
|
}
|
|
}
|
|
|
|
static void
|
|
check( Module * const &m, RequirementsChecker *c )
|
|
{
|
|
RequirementsList l = m->checkRequirements();
|
|
if ( l.count() > 0 )
|
|
c->addCheckedRequirements( l );
|
|
c->requirementsProgress( QObject::tr( "Requirements checking for module <i>%1</i> is complete." ).arg( m->name() ) );
|
|
}
|
|
|
|
RequirementsChecker::RequirementsChecker( QVector< Module* > modules, QObject* parent )
|
|
: QObject( parent )
|
|
, m_modules( std::move( modules ) )
|
|
, m_progressTimer( nullptr )
|
|
, m_progressTimeouts( 0 )
|
|
{
|
|
m_watchers.reserve( m_modules.count() );
|
|
m_collectedRequirements.reserve( m_modules.count() );
|
|
|
|
registerMetatypes();
|
|
}
|
|
|
|
RequirementsChecker::~RequirementsChecker()
|
|
{
|
|
}
|
|
|
|
void
|
|
RequirementsChecker::run()
|
|
{
|
|
m_progressTimer = new QTimer( this );
|
|
connect( m_progressTimer, &QTimer::timeout, this, &RequirementsChecker::reportProgress );
|
|
m_progressTimer->start( 1200 ); // msec
|
|
|
|
for (const auto& module : m_modules )
|
|
{
|
|
Watcher *watcher = new Watcher( this );
|
|
watcher->setFuture( QtConcurrent::run( check, module, this ) );
|
|
m_watchers.append( watcher );
|
|
connect( watcher, &Watcher::finished, this, &RequirementsChecker::finished );
|
|
}
|
|
}
|
|
|
|
void
|
|
RequirementsChecker::finished()
|
|
{
|
|
if ( std::all_of( m_watchers.cbegin(), m_watchers.cend(), []( const Watcher *w ) { return w && w->isFinished(); } ) )
|
|
{
|
|
cDebug() << "All requirements have been checked.";
|
|
|
|
if ( m_progressTimer )
|
|
m_progressTimer->stop();
|
|
|
|
bool acceptable = true;
|
|
int count = 0;
|
|
for ( const auto& r : m_collectedRequirements )
|
|
{
|
|
if ( r.mandatory && !r.satisfied )
|
|
{
|
|
cDebug() << " .. requirement" << count << r.name << "is not satisfied.";
|
|
acceptable = false;
|
|
}
|
|
++count;
|
|
}
|
|
|
|
emit requirementsComplete( acceptable );
|
|
QTimer::singleShot(0, this, &RequirementsChecker::done );
|
|
}
|
|
}
|
|
|
|
void
|
|
RequirementsChecker::addCheckedRequirements( RequirementsList l )
|
|
{
|
|
static QMutex addMutex;
|
|
{
|
|
QMutexLocker lock( &addMutex );
|
|
m_collectedRequirements.append( l );
|
|
}
|
|
cDebug() << "Added" << l.count() << "requirement results";
|
|
emit requirementsResult( l );
|
|
}
|
|
|
|
void
|
|
RequirementsChecker::reportProgress()
|
|
{
|
|
m_progressTimeouts++;
|
|
|
|
auto remaining = std::count_if( m_watchers.cbegin(), m_watchers.cend(), []( const Watcher *w ) { return w && !w->isFinished(); } );
|
|
if ( remaining > 0 )
|
|
{
|
|
QString waiting = tr( "Waiting for %n module(s).", "", remaining );
|
|
QString elapsed = tr( "(%n second(s))", "", m_progressTimeouts * m_progressTimer->interval() / 1000 );
|
|
emit requirementsProgress( waiting + QString( " " ) + elapsed );
|
|
}
|
|
else
|
|
emit requirementsProgress( tr( "System-requirements checking is complete." ) );
|
|
}
|
|
|
|
}
|