Merge pull request #1525 from tintou/calamares

[partition] Add tests for Layout-constrained partionning
This commit is contained in:
Adriaan de Groot 2020-10-06 21:46:21 +02:00 committed by GitHub
commit 3e94570849
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
6 changed files with 208 additions and 12 deletions

View File

@ -11,8 +11,6 @@
#include "PartUtils.h" #include "PartUtils.h"
#include "PartitionCoreModule.h"
#include "core/DeviceModel.h" #include "core/DeviceModel.h"
#include "core/KPMHelpers.h" #include "core/KPMHelpers.h"
#include "core/PartitionInfo.h" #include "core/PartitionInfo.h"
@ -198,13 +196,12 @@ canBeResized( Partition* candidate )
bool bool
canBeResized( PartitionCoreModule* core, const QString& partitionPath ) canBeResized( DeviceModel* dm, const QString& partitionPath )
{ {
cDebug() << "Checking if" << partitionPath << "can be resized."; cDebug() << "Checking if" << partitionPath << "can be resized.";
QString partitionWithOs = partitionPath; QString partitionWithOs = partitionPath;
if ( partitionWithOs.startsWith( "/dev/" ) ) if ( partitionWithOs.startsWith( "/dev/" ) )
{ {
DeviceModel* dm = core->deviceModel();
for ( int i = 0; i < dm->rowCount(); ++i ) for ( int i = 0; i < dm->rowCount(); ++i )
{ {
Device* dev = dm->deviceForIndex( dm->index( i ) ); Device* dev = dm->deviceForIndex( dm->index( i ) );
@ -358,7 +355,7 @@ findPartitionPathForMountPoint( const FstabEntryList& fstab, const QString& moun
OsproberEntryList OsproberEntryList
runOsprober( PartitionCoreModule* core ) runOsprober( DeviceModel* dm )
{ {
QString osproberOutput; QString osproberOutput;
QProcess osprober; QProcess osprober;
@ -406,7 +403,7 @@ runOsprober( PartitionCoreModule* core )
QString homePath = findPartitionPathForMountPoint( fstabEntries, "/home" ); QString homePath = findPartitionPathForMountPoint( fstabEntries, "/home" );
osproberEntries.append( osproberEntries.append(
{ prettyName, path, QString(), canBeResized( core, path ), lineColumns, fstabEntries, homePath } ); { prettyName, path, QString(), canBeResized( dm, path ), lineColumns, fstabEntries, homePath } );
osproberCleanLines.append( line ); osproberCleanLines.append( line );
} }
} }

View File

@ -22,7 +22,7 @@
// Qt // Qt
#include <QString> #include <QString>
class PartitionCoreModule; class DeviceModel;
class Partition; class Partition;
namespace PartUtils namespace PartUtils
@ -56,19 +56,19 @@ bool canBeResized( Partition* candidate );
/** /**
* @brief canBeReplaced checks whether the given Partition satisfies the criteria * @brief canBeReplaced checks whether the given Partition satisfies the criteria
* for resizing (shrinking) it to make room for a new OS. * for resizing (shrinking) it to make room for a new OS.
* @param core the PartitionCoreModule instance. * @param dm the DeviceModel instance.
* @param partitionPath the device path of the candidate partition to resize. * @param partitionPath the device path of the candidate partition to resize.
* @return true if the criteria are met, otherwise false. * @return true if the criteria are met, otherwise false.
*/ */
bool canBeResized( PartitionCoreModule* core, const QString& partitionPath ); bool canBeResized( DeviceModel* dm, const QString& partitionPath );
/** /**
* @brief runOsprober executes os-prober, parses the output and writes relevant * @brief runOsprober executes os-prober, parses the output and writes relevant
* data to GlobalStorage. * data to GlobalStorage.
* @param core the PartitionCoreModule instance. * @param dm the DeviceModel instance.
* @return a list of os-prober entries, parsed. * @return a list of os-prober entries, parsed.
*/ */
OsproberEntryList runOsprober( PartitionCoreModule* core ); OsproberEntryList runOsprober( DeviceModel* dm );
/** /**
* @brief Is this system EFI-enabled? Decides based on /sys/firmware/efi * @brief Is this system EFI-enabled? Decides based on /sys/firmware/efi

View File

@ -180,7 +180,7 @@ PartitionCoreModule::doInit()
// The following PartUtils::runOsprober call in turn calls PartUtils::canBeResized, // The following PartUtils::runOsprober call in turn calls PartUtils::canBeResized,
// which relies on a working DeviceModel. // which relies on a working DeviceModel.
m_osproberLines = PartUtils::runOsprober( this ); m_osproberLines = PartUtils::runOsprober( this->deviceModel() );
// We perform a best effort of filling out filesystem UUIDs in m_osproberLines // We perform a best effort of filling out filesystem UUIDs in m_osproberLines
// because we will need them later on in PartitionModel if partition paths // because we will need them later on in PartitionModel if partition paths

View File

@ -40,3 +40,21 @@ calamares_add_test(
DEFINITIONS ${_partition_defs} DEFINITIONS ${_partition_defs}
) )
calamares_add_test(
createlayoutstests
SOURCES
${PartitionModule_SOURCE_DIR}/core/KPMHelpers.cpp
${PartitionModule_SOURCE_DIR}/core/PartitionInfo.cpp
${PartitionModule_SOURCE_DIR}/core/PartitionLayout.cpp
${PartitionModule_SOURCE_DIR}/core/PartUtils.cpp
${PartitionModule_SOURCE_DIR}/core/DeviceModel.cpp
CreateLayoutsTests.cpp
LIBRARIES
kpmcore
calamares
calamaresui
Qt5::Gui
DEFINITIONS ${_partition_defs}
)

View File

@ -0,0 +1,145 @@
/* === This file is part of Calamares - <https://calamares.io> ===
*
* SPDX-FileCopyrightText: 2020 Corentin Noël <corentin.noel@collabora.com>
* SPDX-License-Identifier: GPL-3.0-or-later
*
* Calamares is Free Software: see the License-Identifier above.
*
*/
#include "CreateLayoutsTests.h"
#include "core/PartitionLayout.h"
#include "utils/Logger.h"
#include "partition/KPMManager.h"
#include "JobQueue.h"
#include <core/lvmdevice.h>
#include <core/partition.h>
#include <fs/filesystemfactory.h>
#include <memory>
#include <QtTest/QtTest>
class PartitionTable;
class SmartStatus;
QTEST_GUILESS_MAIN( CreateLayoutsTests )
CalamaresUtils::Partition::KPMManager* kpmcore = nullptr;
using CalamaresUtils::operator""_MiB;
using CalamaresUtils::operator""_GiB;
#define LOGICAL_SIZE 512
CreateLayoutsTests::CreateLayoutsTests()
{
Logger::setupLogLevel( Logger::LOGDEBUG );
}
void
CreateLayoutsTests::init()
{
std::unique_ptr< Calamares::JobQueue > jobqueue_p( new Calamares::JobQueue( nullptr ) );
kpmcore = new CalamaresUtils::Partition::KPMManager();
}
void
CreateLayoutsTests::cleanup()
{
delete kpmcore;
}
void
CreateLayoutsTests::testFixedSizePartition()
{
PartitionLayout layout = PartitionLayout();
TestDevice dev( QString( "test" ), LOGICAL_SIZE, 5_GiB/LOGICAL_SIZE );
PartitionRole role( PartitionRole::Role::Any );
QList< Partition* > partitions;
if (!layout.addEntry( QString( "/" ), QString( "5MiB" ) ))
{
QFAIL( qPrintable( "Unable to create / partition" ) );
}
partitions = layout.execute( static_cast<Device *>(&dev), 0, dev.totalLogical(), nullptr, nullptr, role );
QCOMPARE( partitions.count(), 1 );
QCOMPARE( partitions[0]->length(), 5_MiB/LOGICAL_SIZE );
}
void
CreateLayoutsTests::testPercentSizePartition()
{
PartitionLayout layout = PartitionLayout();
TestDevice dev( QString( "test" ), LOGICAL_SIZE, 5_GiB/LOGICAL_SIZE );
PartitionRole role( PartitionRole::Role::Any );
QList< Partition* > partitions;
if (!layout.addEntry( QString( "/" ), QString( "50%" ) ))
{
QFAIL( qPrintable( "Unable to create / partition" ) );
}
partitions = layout.execute( static_cast<Device *>(&dev), 0, dev.totalLogical(), nullptr, nullptr, role );
QCOMPARE( partitions.count(), 1 );
QCOMPARE( partitions[0]->length(), (5_GiB/2)/LOGICAL_SIZE );
}
void
CreateLayoutsTests::testMixedSizePartition()
{
PartitionLayout layout = PartitionLayout();
TestDevice dev( QString( "test" ), LOGICAL_SIZE, 5_GiB/LOGICAL_SIZE );
PartitionRole role( PartitionRole::Role::Any );
QList< Partition* > partitions;
if (!layout.addEntry( QString( "/" ), QString( "5MiB" ) ))
{
QFAIL( qPrintable( "Unable to create / partition" ) );
}
if (!layout.addEntry( QString( "/home" ), QString( "50%" ) ))
{
QFAIL( qPrintable( "Unable to create /home partition" ) );
}
if (!layout.addEntry( QString( "/bkup" ), QString( "50%" ) ))
{
QFAIL( qPrintable( "Unable to create /bkup partition" ) );
}
partitions = layout.execute( static_cast<Device *>(&dev), 0, dev.totalLogical(), nullptr, nullptr, role );
QCOMPARE( partitions.count(), 3 );
QCOMPARE( partitions[0]->length(), 5_MiB/LOGICAL_SIZE );
QCOMPARE( partitions[1]->length(), ((5_GiB - 5_MiB)/2)/LOGICAL_SIZE );
QCOMPARE( partitions[2]->length(), ((5_GiB - 5_MiB)/2)/LOGICAL_SIZE );
}
// TODO: Get a clean way to instanciate a test Device from KPMCore
class DevicePrivate
{
public:
QString m_Name;
QString m_DeviceNode;
qint64 m_LogicalSectorSize;
qint64 m_TotalLogical;
PartitionTable* m_PartitionTable;
QString m_IconName;
std::shared_ptr<SmartStatus> m_SmartStatus;
Device::Type m_Type;
};
TestDevice::TestDevice(const QString& name, const qint64 logicalSectorSize, const qint64 totalLogicalSectors)
: Device (std::make_shared<DevicePrivate>(), name, QString( "node" ), logicalSectorSize, totalLogicalSectors, QString(), Device::Type::Unknown_Device)
{
}

View File

@ -0,0 +1,36 @@
/* === This file is part of Calamares - <https://calamares.io> ===
*
* SPDX-FileCopyrightText: 2020 Corentin Noël <corentin.noel@collabora.com>
* SPDX-License-Identifier: GPL-3.0-or-later
*
* Calamares is Free Software: see the License-Identifier above.
*
*/
#ifndef CLEARMOUNTSJOBTESTS_H
#define CLEARMOUNTSJOBTESTS_H
#include <QObject>
#include <core/device.h>
class CreateLayoutsTests : public QObject
{
Q_OBJECT
public:
CreateLayoutsTests();
private Q_SLOTS:
void testFixedSizePartition();
void testPercentSizePartition();
void testMixedSizePartition();
void init();
void cleanup();
};
class TestDevice : public Device
{
public:
TestDevice(const QString& name, const qint64 logicalSectorSize, const qint64 totalLogicalSectors);
};
#endif