/* === This file is part of Calamares - === * * SPDX-FileCopyrightText: 2018 Adriaan de Groot * SPDX-License-Identifier: GPL-3.0-or-later * * * Calamares is Free Software: see the License-Identifier above. * * */ #include "CalamaresUtilsSystem.h" #include "Entropy.h" #include "Logger.h" #include "RAII.h" #include "String.h" #include "Traits.h" #include "UMask.h" #include "Variant.h" #include "Yaml.h" #include "GlobalStorage.h" #include "JobQueue.h" #include #include #include #include #include class LibCalamaresTests : public QObject { Q_OBJECT public: LibCalamaresTests(); ~LibCalamaresTests() override; private Q_SLOTS: void initTestCase(); void testDebugLevels(); void testLoadSaveYaml(); // Just settings.conf void testLoadSaveYamlExtended(); // Do a find() in the src dir void testCommands(); /** @brief Test that all the UMask objects work correctly. */ void testUmask(); /** @brief Tests the entropy functions. */ void testEntropy(); void testPrintableEntropy(); void testOddSizedPrintable(); /** @brief Tests the RAII bits. */ void testBoolSetter(); /** @brief Tests the Traits bits. */ void testTraits(); void testVariantStringListCode(); void testVariantStringListYAMLDashed(); void testVariantStringListYAMLBracketed(); /** @brief Test smart string truncation. */ void testStringTruncation(); private: void recursiveCompareMap( const QVariantMap& a, const QVariantMap& b, int depth ); }; LibCalamaresTests::LibCalamaresTests() {} LibCalamaresTests::~LibCalamaresTests() {} void LibCalamaresTests::initTestCase() { } void LibCalamaresTests::testDebugLevels() { Logger::setupLogLevel( Logger::LOG_DISABLE ); QCOMPARE( Logger::logLevel(), static_cast< unsigned int >( Logger::LOG_DISABLE ) ); for ( unsigned int level = 0; level <= Logger::LOGVERBOSE; ++level ) { Logger::setupLogLevel( level ); QCOMPARE( Logger::logLevel(), level ); QVERIFY( Logger::logLevelEnabled( level ) ); for ( unsigned int xlevel = 0; xlevel <= Logger::LOGVERBOSE; ++xlevel ) { QCOMPARE( Logger::logLevelEnabled( xlevel ), xlevel <= level ); } } } void LibCalamaresTests::testLoadSaveYaml() { Logger::setupLogLevel( Logger::LOGDEBUG ); QFile f( "settings.conf" ); // Find the nearest settings.conf to read for ( unsigned int up = 0; !f.exists() && ( up < 4 ); ++up ) { f.setFileName( QString( "../" ) + f.fileName() ); } cDebug() << QDir().absolutePath() << f.fileName() << f.exists(); QVERIFY( f.exists() ); auto map = CalamaresUtils::loadYaml( f.fileName() ); QVERIFY( map.contains( "sequence" ) ); QCOMPARE( map[ "sequence" ].type(), QVariant::List ); // The source-repo example `settings.conf` has a show and an exec phase auto sequence = map[ "sequence" ].toList(); cDebug() << "Loaded example `settings.conf` sequence:"; for ( const auto& v : sequence ) { cDebug() << Logger::SubEntry << v; QCOMPARE( v.type(), QVariant::Map ); QVERIFY( v.toMap().contains( "show" ) || v.toMap().contains( "exec" ) ); } CalamaresUtils::saveYaml( "out.yaml", map ); auto other_map = CalamaresUtils::loadYaml( "out.yaml" ); CalamaresUtils::saveYaml( "out2.yaml", other_map ); QCOMPARE( map, other_map ); QFile::remove( "out.yaml" ); QFile::remove( "out2.yaml" ); } static QStringList findConf( const QDir& d ) { QStringList mine; if ( d.exists() ) { QString path = d.absolutePath(); path.append( d.separator() ); for ( const auto& confname : d.entryList( { "*.conf" } ) ) mine.append( path + confname ); for ( const auto& subdirname : d.entryList( QDir::AllDirs | QDir::NoDotAndDotDot ) ) { QDir subdir( d ); subdir.cd( subdirname ); mine.append( findConf( subdir ) ); } } return mine; } void LibCalamaresTests::recursiveCompareMap( const QVariantMap& a, const QVariantMap& b, int depth ) { cDebug() << "Comparing depth" << depth << a.count() << b.count(); QCOMPARE( a.keys(), b.keys() ); for ( const auto& k : a.keys() ) { cDebug() << Logger::SubEntry << k; const auto& av = a[ k ]; const auto& bv = b[ k ]; if ( av.typeName() != bv.typeName() ) { cDebug() << Logger::SubEntry << "a type" << av.typeName() << av; cDebug() << Logger::SubEntry << "b type" << bv.typeName() << bv; } QCOMPARE( av.typeName(), bv.typeName() ); if ( av.canConvert< QVariantMap >() ) { recursiveCompareMap( av.toMap(), bv.toMap(), depth + 1 ); } else { QCOMPARE( av, bv ); } } } void LibCalamaresTests::testLoadSaveYamlExtended() { Logger::setupLogLevel( Logger::LOGDEBUG ); bool loaded_ok; for ( const auto& confname : findConf( QDir( "../src" ) ) ) { loaded_ok = true; cDebug() << "Testing" << confname; auto map = CalamaresUtils::loadYaml( confname, &loaded_ok ); QVERIFY( loaded_ok ); QVERIFY( CalamaresUtils::saveYaml( "out.yaml", map ) ); auto othermap = CalamaresUtils::loadYaml( "out.yaml", &loaded_ok ); QVERIFY( loaded_ok ); QCOMPARE( map.keys(), othermap.keys() ); recursiveCompareMap( map, othermap, 0 ); QCOMPARE( map, othermap ); } QFile::remove( "out.yaml" ); } void LibCalamaresTests::testCommands() { using CalamaresUtils::System; auto r = System::runCommand( System::RunLocation::RunInHost, { "/bin/ls", "/tmp" } ); QVERIFY( r.getExitCode() == 0 ); QTemporaryFile tf( "/tmp/calamares-test-XXXXXX" ); QVERIFY( tf.open() ); QVERIFY( !tf.fileName().isEmpty() ); QFileInfo tfn( tf.fileName() ); QVERIFY( !r.getOutput().contains( tfn.fileName() ) ); // Run ls again, now that the file exists r = System::runCommand( System::RunLocation::RunInHost, { "/bin/ls", "/tmp" } ); QVERIFY( r.getOutput().contains( tfn.fileName() ) ); // .. and without a working directory set, assume builddir != /tmp r = System::runCommand( System::RunLocation::RunInHost, { "/bin/ls" } ); QVERIFY( !r.getOutput().contains( tfn.fileName() ) ); r = System::runCommand( System::RunLocation::RunInHost, { "/bin/ls" }, "/tmp" ); QVERIFY( r.getOutput().contains( tfn.fileName() ) ); } void LibCalamaresTests::testUmask() { struct stat mystat; QTemporaryFile ft; QVERIFY( ft.open() ); // m gets the previous value of the mask (depends on the environment the // test is run in, might be 002, might be 077), .. mode_t m = CalamaresUtils::setUMask( 022 ); QCOMPARE( CalamaresUtils::setUMask( m ), mode_t( 022 ) ); // But now most recently set was 022 for ( mode_t i = 0; i <= 0777 /* octal! */; ++i ) { QByteArray name = ( ft.fileName() + QChar( '.' ) + QString::number( i, 8 ) ).toLatin1(); CalamaresUtils::UMask um( i ); int fd = creat( name, 0777 ); QVERIFY( fd >= 0 ); close( fd ); QFileInfo fi( name ); QVERIFY( fi.exists() ); QCOMPARE( stat( name, &mystat ), 0 ); QCOMPARE( mystat.st_mode & 0777, 0777 & ~i ); QCOMPARE( unlink( name ), 0 ); } QCOMPARE( CalamaresUtils::setUMask( 022 ), m ); QCOMPARE( CalamaresUtils::setUMask( m ), mode_t( 022 ) ); } void LibCalamaresTests::testEntropy() { QByteArray data; auto r0 = CalamaresUtils::getEntropy( 0, data ); QCOMPARE( CalamaresUtils::EntropySource::None, r0 ); QCOMPARE( data.size(), 0 ); auto r1 = CalamaresUtils::getEntropy( 16, data ); QVERIFY( r1 != CalamaresUtils::EntropySource::None ); QCOMPARE( data.size(), 16 ); // This can randomly fail (but not often) QVERIFY( data.at( data.size() - 1 ) != char( 0xcb ) ); auto r2 = CalamaresUtils::getEntropy( 8, data ); QVERIFY( r2 != CalamaresUtils::EntropySource::None ); QCOMPARE( data.size(), 8 ); QCOMPARE( r1, r2 ); // This can randomly fail (but not often) QVERIFY( data.at( data.size() - 1 ) != char( 0xcb ) ); } void LibCalamaresTests::testPrintableEntropy() { QString s; auto r0 = CalamaresUtils::getPrintableEntropy( 0, s ); QCOMPARE( CalamaresUtils::EntropySource::None, r0 ); QCOMPARE( s.length(), 0 ); auto r1 = CalamaresUtils::getPrintableEntropy( 16, s ); QVERIFY( r1 != CalamaresUtils::EntropySource::None ); QCOMPARE( s.length(), 16 ); for ( QChar c : s ) { QVERIFY( c.isPrint() ); QCOMPARE( c.row(), uchar( 0 ) ); QVERIFY( c.cell() > 32 ); // ASCII SPACE QVERIFY( c.cell() < 127 ); } } void LibCalamaresTests::testOddSizedPrintable() { QString s; for ( int l = 0; l <= 37; ++l ) { auto r = CalamaresUtils::getPrintableEntropy( l, s ); if ( l == 0 ) { QCOMPARE( r, CalamaresUtils::EntropySource::None ); } else { QVERIFY( r != CalamaresUtils::EntropySource::None ); } QCOMPARE( s.length(), l ); for ( QChar c : s ) { QVERIFY( c.isPrint() ); QCOMPARE( c.row(), uchar( 0 ) ); QVERIFY( c.cell() > 32 ); // ASCII SPACE QVERIFY( c.cell() < 127 ); } } } void LibCalamaresTests::testBoolSetter() { bool b = false; QVERIFY( !b ); { QVERIFY( !b ); cBoolSetter< true > x( b ); QVERIFY( b ); } QVERIFY( !b ); QVERIFY( !b ); { QVERIFY( !b ); cBoolSetter< false > x( b ); QVERIFY( !b ); // Still! } QVERIFY( b ); } /* Demonstration of Traits support for has-a-method or not. * * We have two classes, c1 and c2; one has a method do_the_thing() and the * other does not. A third class, Thinginator, has a method thingify(), * which should call do_the_thing() of its argument if it exists. */ struct c1 { int do_the_thing() { return 2; } }; struct c2 { }; DECLARE_HAS_METHOD( do_the_thing ) struct Thinginator { public: /// When class T has function do_the_thing() template < class T > int thingify( T& t, const std::true_type& ) { return t.do_the_thing(); } template < class T > int thingify( T&, const std::false_type& ) { return -1; } template < class T > int thingify( T& t ) { return thingify( t, has_do_the_thing< T > {} ); } }; void LibCalamaresTests::testTraits() { has_do_the_thing< c1 > x {}; has_do_the_thing< c2 > y {}; QVERIFY( x ); QVERIFY( !y ); c1 c1 {}; c2 c2 {}; QCOMPARE( c1.do_the_thing(), 2 ); Thinginator t; QCOMPARE( t.thingify( c1 ), 2 ); // Calls c1::do_the_thing() QCOMPARE( t.thingify( c2 ), -1 ); } void LibCalamaresTests::testVariantStringListCode() { using namespace CalamaresUtils; const QString key( "strings" ); { // Things that are not stringlists QVariantMap m; QCOMPARE( getStringList( m, key ), QStringList {} ); m.insert( key, 17 ); QCOMPARE( getStringList( m, key ), QStringList {} ); m.insert( key, QString( "more strings" ) ); QCOMPARE( getStringList( m, key ), QStringList { "more strings" } ); // A single string **can** be considered a stringlist! m.insert( key, QVariant {} ); QCOMPARE( getStringList( m, key ), QStringList {} ); } { // Things that are stringlists QVariantMap m; m.insert( key, QStringList { "aap", "noot" } ); QVERIFY( getStringList( m, key ).contains( "aap" ) ); QVERIFY( !getStringList( m, key ).contains( "mies" ) ); } } void LibCalamaresTests::testVariantStringListYAMLDashed() { using namespace CalamaresUtils; const QString key( "strings" ); // Looks like a stringlist to me QTemporaryFile f; QVERIFY( f.open() ); f.write( R"(--- strings: - aap - noot - mies )" ); f.close(); bool ok = false; QVariantMap m = loadYaml( f.fileName(), &ok ); QVERIFY( ok ); QCOMPARE( m.count(), 1 ); QVERIFY( m.contains( key ) ); QVERIFY( getStringList( m, key ).contains( "aap" ) ); QVERIFY( getStringList( m, key ).contains( "mies" ) ); QVERIFY( !getStringList( m, key ).contains( "lam" ) ); } void LibCalamaresTests::testVariantStringListYAMLBracketed() { using namespace CalamaresUtils; const QString key( "strings" ); // Looks like a stringlist to me QTemporaryFile f; QVERIFY( f.open() ); f.write( R"(--- strings: [ aap, noot, mies ] )" ); f.close(); bool ok = false; QVariantMap m = loadYaml( f.fileName(), &ok ); QVERIFY( ok ); QCOMPARE( m.count(), 1 ); QVERIFY( m.contains( key ) ); QVERIFY( getStringList( m, key ).contains( "aap" ) ); QVERIFY( getStringList( m, key ).contains( "mies" ) ); QVERIFY( !getStringList( m, key ).contains( "lam" ) ); } void LibCalamaresTests::testStringTruncation() { Logger::setupLogLevel( Logger::LOGDEBUG ); using namespace CalamaresUtils; const QString longString( R"(--- --- src/libcalamares/utils/String.h +++ src/libcalamares/utils/String.h @@ -62,15 +62,22 @@ DLLEXPORT QString removeDiacritics( const QString& string ); */ DLLEXPORT QString obscure( const QString& string ); +/** @brief Parameter for counting lines at beginning and end of string + * + * This is used by truncateMultiLine() to indicate how many lines from + * the beginning and how many from the end should be kept. + */ struct LinesStartEnd { - int atStart; - int atEnd; + int atStart = 0; + int atEnd = 0; )" ); const int sufficientLength = 812; // There's 18 lines in all QCOMPARE( longString.count( '\n' ), 18 ); QVERIFY( longString.length() < sufficientLength ); // If we ask for more, we get everything back QCOMPARE( longString, truncateMultiLine( longString, LinesStartEnd { 20, 0 }, CharCount { sufficientLength } ) ); QCOMPARE( longString, truncateMultiLine( longString, LinesStartEnd { 0, 20 }, CharCount { sufficientLength } ) ); // If we ask for no lines, only characters, we get that { auto s = truncateMultiLine( longString, LinesStartEnd { 0, 0 }, CharCount { 4 } ); QCOMPARE( s.length(), 4 ); QCOMPARE( s, QString( "---\n" ) ); } { auto s = truncateMultiLine( longString, LinesStartEnd { 0, 0 }, CharCount { sufficientLength } ); QCOMPARE( s, longString ); } // Lines at the start { auto s = truncateMultiLine( longString, LinesStartEnd { 4, 0 }, CharCount { sufficientLength } ); QVERIFY( s.length() > 1 ); QVERIFY( longString.startsWith( s ) ); cDebug() << "Result-line" << Logger::Quote << s; QCOMPARE( s.count( '\n' ), 4 ); } // Lines at the end { auto s = truncateMultiLine( longString, LinesStartEnd { 0, 4 }, CharCount { sufficientLength } ); QVERIFY( s.length() > 1 ); QVERIFY( longString.endsWith( s ) ); cDebug() << "Result-line" << Logger::Quote << s; QCOMPARE( s.count( '\n' ), 4 ); } // Lines at both ends { auto s = truncateMultiLine( longString, LinesStartEnd { 2, 2 }, CharCount { sufficientLength } ); QVERIFY( s.length() > 1 ); cDebug() << "Result-line" << Logger::Quote << s; QCOMPARE( s.count( '\n' ), 4 ); auto firsttwo = truncateMultiLine( s, LinesStartEnd { 2, 0 }, CharCount { sufficientLength } ); auto lasttwo = truncateMultiLine( s, LinesStartEnd { 0, 2 }, CharCount { sufficientLength } ); QCOMPARE( firsttwo + lasttwo, s ); QVERIFY( longString.startsWith( firsttwo ) ); QVERIFY( longString.endsWith( lasttwo ) ); } } QTEST_GUILESS_MAIN( LibCalamaresTests ) #include "utils/moc-warnings.h" #include "Tests.moc"