Merge branch 'context-wildcard'

FIXES #906
This commit is contained in:
Adriaan de Groot 2018-02-19 07:04:43 -05:00
commit 94014a52d4
6 changed files with 212 additions and 19 deletions

View File

@ -7,3 +7,22 @@ calamares_add_plugin( contextualprocess
calamares calamares
SHARED_LIB SHARED_LIB
) )
if( ECM_FOUND )
find_package( Qt5 COMPONENTS Test REQUIRED )
include( ECMAddTests )
ecm_add_test(
Tests.cpp
ContextualProcessJob.cpp # Builds it a second time
TEST_NAME
contextualprocesstest
LINK_LIBRARIES
${CALAMARES_LIBRARIES}
calamaresui
${YAMLCPP_LIBRARY}
Qt5::Core
Qt5::Test
)
set_target_properties( contextualprocesstest PROPERTIES AUTOMOC TRUE )
endif()

View File

@ -30,31 +30,72 @@
#include "utils/CommandList.h" #include "utils/CommandList.h"
#include "utils/Logger.h" #include "utils/Logger.h"
struct ValueCheck : public QPair<QString, CalamaresUtils::CommandList*>
{
ValueCheck( const QString& value, CalamaresUtils::CommandList* commands )
: QPair<QString, CalamaresUtils::CommandList*>(value, commands)
{
}
~ValueCheck()
{
// We don't own the commandlist, the binding holding this valuecheck
// does, so don't delete. This is closely tied to (temporaries created
// by) pass-by-value in QList::append().
}
QString value() const { return first; }
CalamaresUtils::CommandList* commands() const { return second; }
} ;
struct ContextualProcessBinding struct ContextualProcessBinding
{ {
ContextualProcessBinding( const QString& _n, const QString& _v, CalamaresUtils::CommandList* _c ) ContextualProcessBinding( const QString& varname )
: variable( _n ) : variable( varname )
, value( _v )
, commands( _c )
{ {
} }
~ContextualProcessBinding(); ~ContextualProcessBinding();
int count() const /**
* @brief add commands to be executed when @p value is matched.
*
* Ownership of the CommandList passes to this binding.
*/
void append( const QString& value, CalamaresUtils::CommandList* commands )
{ {
return commands ? commands->count() : 0; checks.append( ValueCheck( value, commands ) );
if ( value == '*' )
wildcard = commands;
}
Calamares::JobResult run( const QString& value ) const
{
for ( const auto& c : checks )
{
if ( value == c.value() )
return c.commands()->run();
}
if ( wildcard )
return wildcard->run();
return Calamares::JobResult::ok();
} }
QString variable; QString variable;
QString value; QList<ValueCheck> checks;
CalamaresUtils::CommandList* commands; CalamaresUtils::CommandList* wildcard{ nullptr };
} ; } ;
ContextualProcessBinding::~ContextualProcessBinding() ContextualProcessBinding::~ContextualProcessBinding()
{ {
delete commands; wildcard = nullptr;
for ( const auto& c : checks )
{
delete c.commands();
}
} }
ContextualProcessJob::ContextualProcessJob( QObject* parent ) ContextualProcessJob::ContextualProcessJob( QObject* parent )
@ -83,12 +124,14 @@ ContextualProcessJob::exec()
for ( const ContextualProcessBinding* binding : m_commands ) for ( const ContextualProcessBinding* binding : m_commands )
{ {
if ( gs->contains( binding->variable ) && ( gs->value( binding->variable ).toString() == binding->value ) ) if ( gs->contains( binding->variable ) )
{ {
Calamares::JobResult r = binding->commands->run(); Calamares::JobResult r = binding->run( gs->value( binding->variable ).toString() );
if ( !r ) if ( !r )
return r; return r;
} }
else
cWarning() << "ContextualProcess checks for unknown variable" << binding->variable;
} }
return Calamares::JobResult::ok(); return Calamares::JobResult::ok();
} }
@ -114,6 +157,8 @@ ContextualProcessJob::setConfigurationMap( const QVariantMap& configurationMap )
continue; continue;
} }
auto binding = new ContextualProcessBinding( variableName );
m_commands.append( binding );
QVariantMap values = iter.value().toMap(); QVariantMap values = iter.value().toMap();
for ( QVariantMap::const_iterator valueiter = values.cbegin(); valueiter != values.cend(); ++valueiter ) for ( QVariantMap::const_iterator valueiter = values.cbegin(); valueiter != values.cend(); ++valueiter )
{ {
@ -126,15 +171,24 @@ ContextualProcessJob::setConfigurationMap( const QVariantMap& configurationMap )
CalamaresUtils::CommandList* commands = new CalamaresUtils::CommandList( valueiter.value(), !dontChroot, timeout ); CalamaresUtils::CommandList* commands = new CalamaresUtils::CommandList( valueiter.value(), !dontChroot, timeout );
if ( commands->count() > 0 ) binding->append( valueString, commands );
{
m_commands.append( new ContextualProcessBinding( variableName, valueString, commands ) );
cDebug() << variableName << '=' << valueString << "will execute" << commands->count() << "commands";
}
else
delete commands;
} }
} }
} }
int
ContextualProcessJob::count()
{
return m_commands.count();
}
int
ContextualProcessJob::count(const QString& variableName)
{
for ( const ContextualProcessBinding* binding : m_commands )
if ( binding->variable == variableName )
return binding->checks.count();
return -1;
}
CALAMARES_PLUGIN_FACTORY_DEFINITION( ContextualProcessJobFactory, registerPlugin<ContextualProcessJob>(); ) CALAMARES_PLUGIN_FACTORY_DEFINITION( ContextualProcessJobFactory, registerPlugin<ContextualProcessJob>(); )

View File

@ -43,6 +43,11 @@ public:
void setConfigurationMap( const QVariantMap& configurationMap ) override; void setConfigurationMap( const QVariantMap& configurationMap ) override;
/// The number of bindings
int count();
/// The number of value-checks for the named binding (-1 if binding doesn't exist)
int count( const QString& variableName );
private: private:
QList<ContextualProcessBinding*> m_commands; QList<ContextualProcessBinding*> m_commands;
}; };

View File

@ -0,0 +1,71 @@
/* === This file is part of Calamares - <https://github.com/calamares> ===
*
* Copyright 2017, 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 "Tests.h"
#include "ContextualProcessJob.h"
#include "utils/CommandList.h"
#include "utils/YamlUtils.h"
#include <yaml-cpp/yaml.h>
#include <QtTest/QtTest>
#include <QFileInfo>
#include <QStringList>
QTEST_GUILESS_MAIN( ContextualProcessTests )
using CommandList = CalamaresUtils::CommandList;
ContextualProcessTests::ContextualProcessTests()
{
}
ContextualProcessTests::~ContextualProcessTests()
{
}
void
ContextualProcessTests::initTestCase()
{
}
void
ContextualProcessTests::testProcessListSampleConfig()
{
YAML::Node doc;
QStringList dirs { "src/modules/contextualprocess", "." };
for ( const auto& dir : dirs )
{
QString filename = dir + "/contextualprocess.conf";
if ( QFileInfo::exists( filename ) )
{
doc = YAML::LoadFile( filename.toStdString() );
break;
}
}
ContextualProcessJob job;
job.setConfigurationMap( CalamaresUtils::yamlMapToVariant( doc ).toMap() );
QCOMPARE(job.count(), 1); // Only "firmwareType"
QCOMPARE(job.count("firmwareType"), 4);
}

View File

@ -0,0 +1,37 @@
/* === This file is part of Calamares - <https://github.com/calamares> ===
*
* Copyright 2018, 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/>.
*/
#ifndef TESTS_H
#define TESTS_H
#include <QObject>
class ContextualProcessTests : public QObject
{
Q_OBJECT
public:
ContextualProcessTests();
~ContextualProcessTests() override;
private Q_SLOTS:
void initTestCase();
// Check the sample config file is processed correctly
void testProcessListSampleConfig();
};
#endif

View File

@ -15,12 +15,18 @@
# #
# You can check for an empty value with "". # You can check for an empty value with "".
# #
# As a special case, the value-check "*" matches any value, but **only**
# if no other value-check matches. Use it as an *else* form for value-
# checks. Take care to put the asterisk in quotes.
#
# Global configuration variables are not checked in a deterministic # Global configuration variables are not checked in a deterministic
# order, so do not rely on commands from one variable-check to # order, so do not rely on commands from one variable-check to
# always happen before (or after) checks on another # always happen before (or after) checks on another
# variable. Similarly, the value-equality checks are not # variable. Similarly, the value-equality checks are not
# done in a deterministic order, but all of the value-checks # done in a deterministic order, but all of the value-checks
# for a given variable happen together. # for a given variable happen together. As a special case, the
# value-check for "*" (the *else* case) happens after all of the
# other value-checks, and only matches if none of the others do.
# #
# The values after a value sub-keys are the same kinds of values # The values after a value sub-keys are the same kinds of values
# as can be given to the *script* key in the shellprocess module. # as can be given to the *script* key in the shellprocess module.
@ -34,3 +40,4 @@ firmwareType:
timeout: 120 # This is slow timeout: 120 # This is slow
bios: "-pkg remove bios-firmware" bios: "-pkg remove bios-firmware"
"": "/bin/false no-firmware-type-set" "": "/bin/false no-firmware-type-set"
"*": "/bin/false some-other-firmware-value"