[libcalamares] Add a word-expander

This is a variant on KMacroExpander, which allows for
reporting of errors after expansion.
This commit is contained in:
Adriaan de Groot 2022-04-12 15:28:07 +02:00
parent 576f244d2d
commit f923dedc3f
4 changed files with 209 additions and 1 deletions

View File

@ -69,6 +69,7 @@ set(libSources
utils/Retranslator.cpp utils/Retranslator.cpp
utils/Runner.cpp utils/Runner.cpp
utils/String.cpp utils/String.cpp
utils/StringExpander.cpp
utils/UMask.cpp utils/UMask.cpp
utils/Variant.cpp utils/Variant.cpp
utils/Yaml.cpp utils/Yaml.cpp
@ -146,7 +147,7 @@ calamares_automoc( calamares )
target_link_libraries( target_link_libraries(
calamares calamares
LINK_PRIVATE ${OPTIONAL_PRIVATE_LIBRARIES} LINK_PRIVATE ${OPTIONAL_PRIVATE_LIBRARIES}
LINK_PUBLIC yamlcpp::yamlcpp Qt5::Core ${OPTIONAL_PUBLIC_LIBRARIES} LINK_PUBLIC yamlcpp::yamlcpp Qt5::Core KF5::CoreAddons ${OPTIONAL_PUBLIC_LIBRARIES}
) )
add_library(Calamares::calamares ALIAS calamares) add_library(Calamares::calamares ALIAS calamares)

View File

@ -0,0 +1,82 @@
/* === This file is part of Calamares - <https://calamares.io> ===
*
* SPDX-FileCopyrightText: 2022 Adriaan de Groot <groot@kde.org>
* SPDX-License-Identifier: GPL-3.0-or-later
*
* Calamares is Free Software: see the License-Identifier above.
*
*
*/
#include "StringExpander.h"
#include "Logger.h"
namespace Calamares
{
namespace String
{
struct DictionaryExpander::Private
{
QHash< QString, QString > dictionary;
QStringList missing;
};
DictionaryExpander::DictionaryExpander()
: KWordMacroExpander( '$' )
, d( std::make_unique< Private >() )
{
}
DictionaryExpander::~DictionaryExpander() {}
void
DictionaryExpander::insert( const QString& key, const QString& value )
{
d->dictionary.insert( key, value );
}
void
DictionaryExpander::clearErrors()
{
d->missing.clear();
}
bool
DictionaryExpander::hasErrors() const
{
return !d->missing.isEmpty();
}
QStringList
DictionaryExpander::errorNames() const
{
return d->missing;
}
QString
DictionaryExpander::expand( QString s )
{
clearErrors();
expandMacros( s );
return s;
}
bool
DictionaryExpander::expandMacro( const QString& str, QStringList& ret )
{
if ( d->dictionary.contains( str ) )
{
ret << d->dictionary[ str ];
return true;
}
else
{
d->missing << str;
return false;
}
}
} // namespace String
} // namespace Calamares

View File

@ -0,0 +1,66 @@
/* === This file is part of Calamares - <https://calamares.io> ===
*
* SPDX-FileCopyrightText: 2022 Adriaan de Groot <groot@kde.org>
* SPDX-License-Identifier: GPL-3.0-or-later
*
* Calamares is Free Software: see the License-Identifier above.
*
*
*/
#ifndef UTILS_STRINGEXPANDER_H
#define UTILS_STRINGEXPANDER_H
#include "DllMacro.h"
#include <KMacroExpander>
#include <QString>
#include <QStringList>
namespace Calamares
{
namespace String
{
/** @brief Expand variables in a string against a dictionary.
*
* This class provides a convenience API for building up a dictionary
* and using it to expand strings. Use the `expand()` method to
* do standard word-based expansion with `$` as macro-symbol.
*
* Unlike straight-up `KMacroExpander::expandMacros()`, this
* provides an API to find out which variables were missing
* from the dictionary during expansion. Use `hasErrors()` and
* `errorNames()` to find out which variables those were.
*
* Call `clearErrors()` to reset the stored errors. Calling
* `expand()` implicitly clears the errors before starting
* a new expansion, as well.
*/
class DictionaryExpander : public KWordMacroExpander
{
public:
DictionaryExpander();
virtual ~DictionaryExpander() override;
void insert( const QString& key, const QString& value );
void clearErrors();
bool hasErrors() const;
QStringList errorNames() const;
QString expand( QString s );
protected:
virtual bool expandMacro( const QString& str, QStringList& ret ) override;
private:
struct Private;
std::unique_ptr< Private > d;
};
} // namespace String
} // namespace Calamares
#endif

View File

@ -15,6 +15,7 @@
#include "RAII.h" #include "RAII.h"
#include "Runner.h" #include "Runner.h"
#include "String.h" #include "String.h"
#include "StringExpander.h"
#include "Traits.h" #include "Traits.h"
#include "UMask.h" #include "UMask.h"
#include "Variant.h" #include "Variant.h"
@ -75,6 +76,10 @@ private Q_SLOTS:
void testStringRemoveTrailing_data(); void testStringRemoveTrailing_data();
void testStringRemoveTrailing(); void testStringRemoveTrailing();
/** @section Test String expansion. */
void testStringMacroExpander_data();
void testStringMacroExpander(); // The KF5::CoreAddons bits
/** @section Test Runner directory-manipulation. */ /** @section Test Runner directory-manipulation. */
void testRunnerDirs(); void testRunnerDirs();
void testCalculateWorkingDirectory(); void testCalculateWorkingDirectory();
@ -817,6 +822,60 @@ LibCalamaresTests::testStringRemoveTrailing()
QCOMPARE( string, result ); QCOMPARE( string, result );
} }
void
LibCalamaresTests::testStringMacroExpander_data()
{
QTest::addColumn< QString >( "source" );
QTest::addColumn< QString >( "result" );
QTest::addColumn< QStringList >( "errors" );
QTest::newRow( "empty " ) << QString() << QString() << QStringList {};
QTest::newRow( "constant" ) << QStringLiteral( "bunnies!" ) << QStringLiteral( "bunnies!" ) << QStringList {};
QTest::newRow( "escaped " ) << QStringLiteral( "$$bun" ) << QStringLiteral( "$bun" )
<< QStringList {}; // Double $$ is an escaped $
QTest::newRow( "whole " ) << QStringLiteral( "${ROOT}" ) << QStringLiteral( "wortel" ) << QStringList {};
QTest::newRow( "unbraced" ) << QStringLiteral( "$ROOT" ) << QStringLiteral( "wortel" )
<< QStringList {}; // Does not need {}
QTest::newRow( "bad-var1" ) << QStringLiteral( "${ROOF}" ) << QStringLiteral( "${ROOF}" )
<< QStringList { QStringLiteral( "ROOF" ) }; // Not replaced
QTest::newRow( "twice " ) << QStringLiteral( "${ROOT}x${ROOT}" ) << QStringLiteral( "wortelxwortel" )
<< QStringList {};
QTest::newRow( "bad-var2" ) << QStringLiteral( "${ROOT}x${ROPE}" ) << QStringLiteral( "wortelx${ROPE}" )
<< QStringList { QStringLiteral( "ROPE" ) }; // Not replaced
// This is a borked string with a "nested" variable. The variable-name-
// scanner goes from ${ to the next } and tries to match that.
QTest::newRow( "confuse1" ) << QStringLiteral( "${RO${ROOT}" ) << QStringLiteral( "${ROwortel" )
<< QStringList { "RO${ROOT" };
// This one doesn't have a { for the first name to match with
QTest::newRow( "confuse2" ) << QStringLiteral( "$RO${ROOT}" ) << QStringLiteral( "$ROwortel" )
<< QStringList { "RO" };
// Here we see it just doesn't nest
QTest::newRow( "confuse3" ) << QStringLiteral( "${RO${ROOT}}" ) << QStringLiteral( "${ROwortel}" )
<< QStringList { "RO${ROOT" };
}
void
LibCalamaresTests::testStringMacroExpander()
{
QHash< QString, QString > dict;
dict.insert( QStringLiteral( "ROOT" ), QStringLiteral( "wortel" ) );
Calamares::String::DictionaryExpander d;
d.insert( QStringLiteral( "ROOT" ), QStringLiteral( "wortel" ) );
QFETCH( QString, source );
QFETCH( QString, result );
QFETCH( QStringList, errors );
QString km_expanded = KMacroExpander::expandMacros( source, dict, '$' );
QCOMPARE( km_expanded, result );
QString de_expanded = d.expand( source );
QCOMPARE( de_expanded, result );
QCOMPARE( d.errorNames(), errors );
QCOMPARE( d.hasErrors(), !errors.isEmpty() );
}
static QString static QString
dirname( const QTemporaryDir& d ) dirname( const QTemporaryDir& d )
{ {