Merge branch 'master' of https://github.com/calamares/calamares into development

This commit is contained in:
Philip Müller 2020-02-19 10:50:50 +01:00
commit 90e7e82078
88 changed files with 1105 additions and 620 deletions

View File

@ -31,6 +31,14 @@ This release contains contributions from (alphabetically by first name):
"fancy" release notes as a QML application, rather than a webview "fancy" release notes as a QML application, rather than a webview
or text widget. Note that this does not replace the slideshow-during- or text widget. Note that this does not replace the slideshow-during-
installation module. installation module.
- The *users* module now has knobs for setting the hostname and writing
the `/etc/hosts` file. The new configuration options are documented
in `users.conf`. #1140
- Multiple *netinstall* modules can exist side-by-side, and they each
control the package installation for their part of the package list.
Previously, a netinstall module would overwrite all of the package
configuration done by other netinstall modules.
#1303
# 3.2.18 (2020-01-28) # # 3.2.18 (2020-01-28) #

View File

@ -549,10 +549,11 @@ install(
"${PROJECT_BINARY_DIR}/CalamaresConfig.cmake" "${PROJECT_BINARY_DIR}/CalamaresConfig.cmake"
"${PROJECT_BINARY_DIR}/CalamaresConfigVersion.cmake" "${PROJECT_BINARY_DIR}/CalamaresConfigVersion.cmake"
"${PROJECT_BINARY_DIR}/CalamaresUse.cmake" "${PROJECT_BINARY_DIR}/CalamaresUse.cmake"
"CMakeModules/CalamaresAddPlugin.cmake"
"CMakeModules/CalamaresAddModuleSubdirectory.cmake"
"CMakeModules/CalamaresAddLibrary.cmake"
"CMakeModules/CalamaresAddBrandingSubdirectory.cmake" "CMakeModules/CalamaresAddBrandingSubdirectory.cmake"
"CMakeModules/CalamaresAddLibrary.cmake"
"CMakeModules/CalamaresAddModuleSubdirectory.cmake"
"CMakeModules/CalamaresAddPlugin.cmake"
"CMakeModules/CalamaresAddTest.cmake"
"CMakeModules/CalamaresAddTranslations.cmake" "CMakeModules/CalamaresAddTranslations.cmake"
"CMakeModules/CalamaresAutomoc.cmake" "CMakeModules/CalamaresAutomoc.cmake"
"CMakeModules/CMakeColors.cmake" "CMakeModules/CMakeColors.cmake"

View File

@ -0,0 +1,58 @@
# === This file is part of Calamares - <https://github.com/calamares> ===
#
# 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/>.
#
# SPDX-License-Identifier: GPL-3.0+
# License-Filename: LICENSE
#
###
#
# Support functions for building Calamares tests.
# This extends KDE's ECM tests with some custom patterns.
#
# calamares_add_test(
# <NAME>
# [GUI]
# SOURCES <FILE..>
# )
include( CMakeParseArguments )
include( CalamaresAutomoc )
function( calamares_add_test )
# parse arguments (name needs to be saved before passing ARGN into the macro)
set( NAME ${ARGV0} )
set( options GUI )
set( multiValueArgs SOURCES LIBRARIES )
cmake_parse_arguments( TEST "${options}" "" "${multiValueArgs}" ${ARGN} )
set( TEST_NAME ${NAME} )
if( ECM_FOUND AND BUILD_TESTING )
ecm_add_test(
${TEST_SOURCES}
TEST_NAME
${TEST_NAME}
LINK_LIBRARIES
calamares
${TEST_LIBRARIES}
Qt5::Core
Qt5::Test
)
calamares_automoc( ${TEST_NAME} )
target_compile_definitions( ${TEST_NAME} PRIVATE -DBUILD_AS_TEST )
if( TEST_GUI )
target_link_libraries( ${TEST_NAME} calamaresui Qt5::Gui )
endif()
endif()
endfunction()

View File

@ -4,6 +4,8 @@
# YAMLCPP_FOUND, if false, do not try to link to yaml-cpp # YAMLCPP_FOUND, if false, do not try to link to yaml-cpp
# YAMLCPP_LIBRARY, where to find yaml-cpp # YAMLCPP_LIBRARY, where to find yaml-cpp
# YAMLCPP_INCLUDE_DIR, where to find yaml.h # YAMLCPP_INCLUDE_DIR, where to find yaml.h
# There is also one IMPORTED library target,
# yamlcpp
# #
# By default, the dynamic libraries of yaml-cpp will be found. To find the static ones instead, # By default, the dynamic libraries of yaml-cpp will be found. To find the static ones instead,
# you must set the YAMLCPP_STATIC_LIBRARY variable to TRUE before calling find_package(YamlCpp ...). # you must set the YAMLCPP_STATIC_LIBRARY variable to TRUE before calling find_package(YamlCpp ...).
@ -48,3 +50,12 @@ find_library(YAMLCPP_LIBRARY
include(FindPackageHandleStandardArgs) include(FindPackageHandleStandardArgs)
FIND_PACKAGE_HANDLE_STANDARD_ARGS(YAMLCPP DEFAULT_MSG YAMLCPP_INCLUDE_DIR YAMLCPP_LIBRARY) FIND_PACKAGE_HANDLE_STANDARD_ARGS(YAMLCPP DEFAULT_MSG YAMLCPP_INCLUDE_DIR YAMLCPP_LIBRARY)
mark_as_advanced(YAMLCPP_INCLUDE_DIR YAMLCPP_LIBRARY) mark_as_advanced(YAMLCPP_INCLUDE_DIR YAMLCPP_LIBRARY)
# Add an imported target
if( YAMLCPP_LIBRARY )
add_library( yamlcpp UNKNOWN IMPORTED )
set_property( TARGET yamlcpp PROPERTY IMPORTED_LOCATION ${YAMLCPP_LIBRARY} )
if ( YAMLCPP_INCLUDE_DIR )
set_property( TARGET yamlcpp PROPERTY INTERFACE_INCLUDE_DIRECTORIES ${YAMLCPP_INCLUDE_DIR} )
endif()
endif()

View File

@ -2,6 +2,7 @@ include( CalamaresAddPlugin )
include( CalamaresAddModuleSubdirectory ) include( CalamaresAddModuleSubdirectory )
include( CalamaresAddLibrary ) include( CalamaresAddLibrary )
include( CalamaresAddBrandingSubdirectory ) include( CalamaresAddBrandingSubdirectory )
include( CalamaresAddTest )
# library # library
add_subdirectory( libcalamares ) add_subdirectory( libcalamares )

View File

@ -1,5 +1,3 @@
add_definitions( -DUIDLLEXPORT_PRO )
set( calamaresSources set( calamaresSources
main.cpp main.cpp
CalamaresApplication.cpp CalamaresApplication.cpp

View File

@ -74,7 +74,6 @@ mark_thirdparty_code( ${kdsagSources} )
include_directories( include_directories(
${CMAKE_CURRENT_BINARY_DIR} ${CMAKE_CURRENT_BINARY_DIR}
${CMAKE_CURRENT_SOURCE_DIR} ${CMAKE_CURRENT_SOURCE_DIR}
${YAMLCPP_INCLUDE_DIR}
) )
### OPTIONAL Python support ### OPTIONAL Python support
@ -126,7 +125,7 @@ target_link_libraries( calamares
LINK_PRIVATE LINK_PRIVATE
${OPTIONAL_PRIVATE_LIBRARIES} ${OPTIONAL_PRIVATE_LIBRARIES}
LINK_PUBLIC LINK_PUBLIC
${YAMLCPP_LIBRARY} yamlcpp
Qt5::Core Qt5::Core
KF5::CoreAddons KF5::CoreAddons
${OPTIONAL_PUBLIC_LIBRARIES} ${OPTIONAL_PUBLIC_LIBRARIES}
@ -159,85 +158,51 @@ install( FILES ${utilsHeaders} DESTINATION include/libcalam
### TESTING ### TESTING
# #
# #
if ( ECM_FOUND AND BUILD_TESTING ) calamares_add_test(
ecm_add_test(
utils/Tests.cpp
TEST_NAME
libcalamarestest libcalamarestest
LINK_LIBRARIES SOURCES
calamares utils/Tests.cpp
Qt5::Core )
Qt5::Test
)
calamares_automoc( libcalamarestest )
ecm_add_test( calamares_add_test(
utils/TestPaths.cpp
TEST_NAME
libcalamarestestpaths libcalamarestestpaths
LINK_LIBRARIES SOURCES
calamares utils/TestPaths.cpp
Qt5::Core )
Qt5::Test
)
calamares_automoc( libcalamarestestpaths )
calamares_add_test(
ecm_add_test( geoiptest
SOURCES
geoip/GeoIPTests.cpp geoip/GeoIPTests.cpp
${geoip_src} ${geoip_src}
TEST_NAME )
geoiptest
LINK_LIBRARIES
calamares
Qt5::Test
${YAMLCPP_LIBRARY}
)
calamares_automoc( geoiptest )
ecm_add_test( calamares_add_test(
partition/Tests.cpp
TEST_NAME
libcalamarespartitiontest libcalamarespartitiontest
LINK_LIBRARIES SOURCES
calamares partition/Tests.cpp
Qt5::Test )
)
calamares_automoc( libcalamarespartitiontest )
ecm_add_test( calamares_add_test(
locale/Tests.cpp
TEST_NAME
libcalamareslocaletest libcalamareslocaletest
LINK_LIBRARIES SOURCES
calamares locale/Tests.cpp
Qt5::Test )
)
calamares_automoc( libcalamareslocaletest )
ecm_add_test( calamares_add_test(
network/Tests.cpp
TEST_NAME
libcalamaresnetworktest libcalamaresnetworktest
LINK_LIBRARIES SOURCES
calamares network/Tests.cpp
Qt5::Test )
)
calamares_automoc( libcalamaresnetworktest )
ecm_add_test( calamares_add_test(
modulesystem/Tests.cpp
TEST_NAME
libcalamaresmodulesystemtest libcalamaresmodulesystemtest
LINK_LIBRARIES SOURCES
calamares modulesystem/Tests.cpp
Qt5::Test )
)
calamares_automoc( libcalamaresmodulesystemtest )
endif()
if( BUILD_TESTING ) if( BUILD_TESTING )
add_executable( test_geoip geoip/test_geoip.cpp ${geoip_src} ) add_executable( test_geoip geoip/test_geoip.cpp ${geoip_src} )
target_link_libraries( test_geoip calamares Qt5::Network ${YAMLCPP_LIBRARY} ) target_link_libraries( test_geoip calamares Qt5::Network yamlcpp )
calamares_automoc( test_geoip ) calamares_automoc( test_geoip )
endif() endif()

View File

@ -1,6 +1,7 @@
/* === This file is part of Calamares - <https://github.com/calamares> === /* === This file is part of Calamares - <https://github.com/calamares> ===
* *
* Copyright 2014, Teo Mrnjavac <teo@kde.org> * Copyright 2014, Teo Mrnjavac <teo@kde.org>
* Copyright 2020, Adriaan de Groot <groot@kde.org>
* *
* Calamares is free software: you can redistribute it and/or modify * Calamares is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by * it under the terms of the GNU General Public License as published by
@ -21,6 +22,10 @@
#include <QtCore/qglobal.h> #include <QtCore/qglobal.h>
/*
* Mark symbols exported from Calamares non-GUI library with DLLEXPORT.
* These are the public API of libcalamares.
*/
#ifndef DLLEXPORT #ifndef DLLEXPORT
#if defined( DLLEXPORT_PRO ) #if defined( DLLEXPORT_PRO )
#define DLLEXPORT Q_DECL_EXPORT #define DLLEXPORT Q_DECL_EXPORT
@ -29,4 +34,42 @@
#endif #endif
#endif #endif
/*
* Mark symbols exported from Calamares GUI library with DLLEXPORT.
* These are the public API of libcalamaresui.
*/
#ifndef UIDLLEXPORT
#if defined( UIDLLEXPORT_PRO )
#define UIDLLEXPORT Q_DECL_EXPORT
#else
#define UIDLLEXPORT Q_DECL_IMPORT
#endif
#endif
/*
* Mark symbols exported from Calamares C++ plugins with PLUGINDLLEXPORT.
* These are the public API of the libraries (generally, the plugin
* entry point)
*/
#ifndef PLUGINDLLEXPORT
#if defined( PLUGINDLLEXPORT_PRO )
#define PLUGINDLLEXPORT Q_DECL_EXPORT
#else
#define PLUGINDLLEXPORT Q_DECL_IMPORT
#endif
#endif
/*
* For functions that should be static in production but also need to
* be tested, use STATICTEST as linkage specifier. When built as part
* of a test, the function will be given normal linkage.
*/
#ifndef STATICTEST
#if defined( BUILD_AS_TEST )
#define STATICTEST
#else
#define STATICTEST static
#endif
#endif
#endif #endif

View File

@ -1,32 +0,0 @@
/* === This file is part of Calamares - <https://github.com/calamares> ===
*
* Copyright 2014, Teo Mrnjavac <teo@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 PLUGINDLLMACRO_H
#define PLUGINDLLMACRO_H
#include <QtCore/qglobal.h>
#ifndef PLUGINDLLEXPORT
#if defined( PLUGINDLLEXPORT_PRO )
#define PLUGINDLLEXPORT Q_DECL_EXPORT
#else
#define PLUGINDLLEXPORT Q_DECL_IMPORT
#endif
#endif
#endif

View File

@ -47,6 +47,6 @@ NetworkTests::testPing()
using namespace CalamaresUtils::Network; using namespace CalamaresUtils::Network;
Logger::setupLogLevel( Logger::LOGVERBOSE ); Logger::setupLogLevel( Logger::LOGVERBOSE );
auto& nam = Manager::instance(); auto& nam = Manager::instance();
auto r = nam.synchronousPing( QUrl( "https://www.kde.org" ), RequestOptions( RequestOptions::FollowRedirect ) ); auto canPing_www_kde_org = nam.synchronousPing( QUrl( "https://www.kde.org" ), RequestOptions( RequestOptions::FollowRedirect ) );
QVERIFY( r ); QVERIFY( canPing_www_kde_org );
} }

View File

@ -293,19 +293,19 @@ System::targetPath( const QString& path ) const
} }
} }
QString CreationResult
System::createTargetFile( const QString& path, const QByteArray& contents ) const System::createTargetFile( const QString& path, const QByteArray& contents ) const
{ {
QString completePath = targetPath( path ); QString completePath = targetPath( path );
if ( completePath.isEmpty() ) if ( completePath.isEmpty() )
{ {
return QString(); return CreationResult( CreationResult::Code::Invalid );
} }
QFile f( completePath ); QFile f( completePath );
if ( f.exists() ) if ( f.exists() )
{ {
return QString(); return CreationResult( CreationResult::Code::AlreadyExists );
} }
QIODevice::OpenMode m = QIODevice::OpenMode m =
@ -317,18 +317,18 @@ System::createTargetFile( const QString& path, const QByteArray& contents ) cons
if ( !f.open( m ) ) if ( !f.open( m ) )
{ {
return QString(); return CreationResult( CreationResult::Code::Failed );
} }
if ( f.write( contents ) != contents.size() ) if ( f.write( contents ) != contents.size() )
{ {
f.close(); f.close();
f.remove(); f.remove();
return QString(); return CreationResult( CreationResult::Code::Failed );
} }
f.close(); f.close();
return QFileInfo( f ).canonicalFilePath(); return CreationResult( QFileInfo( f ).canonicalFilePath() );
} }
void void

View File

@ -84,6 +84,41 @@ public:
} }
}; };
/** @brief The result of a create*() action, for status
*
* A CreationResult has a status field, can be converted to bool
* (true only on success) and can report the full pathname of
* the thing created if it was successful.
*/
class CreationResult : public QPair< int, QString >
{
public:
enum class Code : int
{
// These are "not failed", but only OK is a success
OK = 0,
AlreadyExists = 1,
// These are "failed"
Invalid = -1,
Failed = -2
};
CreationResult( Code r )
: QPair< int, QString >( static_cast< int >( r ), QString() )
{
}
explicit CreationResult( const QString& path )
: QPair< int, QString >( 0, path )
{
}
Code code() const { return static_cast< Code >( first ); }
QString path() const { return second; }
bool failed() const { return first < 0; }
operator bool() const { return first == 0; }
};
/** /**
* @brief The System class is a singleton with utility functions that perform * @brief The System class is a singleton with utility functions that perform
* system-specific operations. * system-specific operations.
@ -244,7 +279,7 @@ public:
* root of the host system, or empty on failure. (Here, it is * root of the host system, or empty on failure. (Here, it is
* possible to be canonical because the file exists). * possible to be canonical because the file exists).
*/ */
DLLEXPORT QString createTargetFile( const QString& path, const QByteArray& contents ) const; DLLEXPORT CreationResult createTargetFile( const QString& path, const QByteArray& contents ) const;
/** @brief Remove a file from the target system. /** @brief Remove a file from the target system.
* *

View File

@ -46,6 +46,7 @@ private Q_SLOTS:
void init(); void init();
void cleanupTestCase(); void cleanupTestCase();
void testCreationResult();
void testTargetPath(); void testTargetPath();
void testCreateTarget(); void testCreateTarget();
void testCreateTargetBasedirs(); void testCreateTargetBasedirs();
@ -95,6 +96,42 @@ TestPaths::init()
m_gs->insert( "rootMountPoint", "/tmp" ); m_gs->insert( "rootMountPoint", "/tmp" );
} }
void TestPaths::testCreationResult()
{
using Code = CalamaresUtils::CreationResult::Code;
for( auto c : { Code::OK, Code::AlreadyExists, Code::Failed, Code::Invalid } )
{
auto r = CalamaresUtils::CreationResult( c );
QVERIFY( r.path().isEmpty() );
QCOMPARE( r.path(), QString() );
// Get a warning from Clang if we're not covering everything
switch( r.code() )
{
case Code::OK:
QVERIFY( !r.failed() );
QVERIFY( r );
break;
case Code::AlreadyExists:
QVERIFY( !r.failed() );
QVERIFY( !r );
break;
case Code::Failed:
case Code::Invalid:
QVERIFY( r.failed() );
QVERIFY( !r );
break;
}
}
QString path( "/etc/os-release" );
auto r = CalamaresUtils::CreationResult( path );
QVERIFY( !r.failed() );
QVERIFY( r );
QCOMPARE( r.code(), Code::OK );
QCOMPARE( r.path(), path );
}
void void
TestPaths::testTargetPath() TestPaths::testTargetPath()
@ -117,7 +154,10 @@ TestPaths::testTargetPath()
void void
TestPaths::testCreateTarget() TestPaths::testCreateTarget()
{ {
QCOMPARE( m_system->createTargetFile( testFile, "Hello" ), QString( absFile ) ); // Success auto r = m_system->createTargetFile( testFile, "Hello" );
QVERIFY( !r.failed() );
QVERIFY( r );
QCOMPARE( r.path(), QString( absFile ) ); // Success
QFileInfo fi( absFile ); QFileInfo fi( absFile );
QVERIFY( fi.exists() ); QVERIFY( fi.exists() );

View File

@ -22,7 +22,7 @@
#ifndef BRANDING_H #ifndef BRANDING_H
#define BRANDING_H #define BRANDING_H
#include "UiDllMacro.h" #include "DllMacro.h"
#include "utils/NamedSuffix.h" #include "utils/NamedSuffix.h"

View File

@ -1,32 +0,0 @@
/* === This file is part of Calamares - <https://github.com/calamares> ===
*
* Copyright 2014, Teo Mrnjavac <teo@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 UIDLLMACRO_H
#define UIDLLMACRO_H
#include <QtCore/qglobal.h>
#ifndef UIDLLEXPORT
#if defined( UIDLLEXPORT_PRO )
#define UIDLLEXPORT Q_DECL_EXPORT
#else
#define UIDLLEXPORT Q_DECL_IMPORT
#endif
#endif
#endif

View File

@ -20,7 +20,7 @@
#ifndef VIEWMANAGER_H #ifndef VIEWMANAGER_H
#define VIEWMANAGER_H #define VIEWMANAGER_H
#include "UiDllMacro.h" #include "DllMacro.h"
#include "viewpages/ViewStep.h" #include "viewpages/ViewStep.h"
#include <QList> #include <QList>

View File

@ -22,7 +22,7 @@
#define CALAMARES_CPPJOBMODULE_H #define CALAMARES_CPPJOBMODULE_H
#include "Module.h" #include "Module.h"
#include "UiDllMacro.h" #include "DllMacro.h"
class QPluginLoader; class QPluginLoader;

View File

@ -22,7 +22,7 @@
#include "Job.h" #include "Job.h"
#include "Requirement.h" #include "Requirement.h"
#include "UiDllMacro.h" #include "DllMacro.h"
#include "modulesystem/Descriptor.h" #include "modulesystem/Descriptor.h"
#include "modulesystem/InstanceKey.h" #include "modulesystem/InstanceKey.h"

View File

@ -22,7 +22,7 @@
#include "Module.h" #include "Module.h"
#include "UiDllMacro.h" #include "DllMacro.h"
#include <chrono> #include <chrono>

View File

@ -21,7 +21,7 @@
#include "Module.h" #include "Module.h"
#include "UiDllMacro.h" #include "DllMacro.h"
namespace Calamares namespace Calamares
{ {

View File

@ -20,7 +20,7 @@
#define CALAMARES_PYTHONQTVIEWMODULE_H #define CALAMARES_PYTHONQTVIEWMODULE_H
#include "Module.h" #include "Module.h"
#include "UiDllMacro.h" #include "DllMacro.h"
namespace Calamares namespace Calamares
{ {

View File

@ -21,7 +21,7 @@
#define CALAMARES_VIEWMODULE_H #define CALAMARES_VIEWMODULE_H
#include "Module.h" #include "Module.h"
#include "UiDllMacro.h" #include "DllMacro.h"
class QPluginLoader; class QPluginLoader;

View File

@ -20,7 +20,7 @@
#ifndef CALAMARESUTILSGUI_H #ifndef CALAMARESUTILSGUI_H
#define CALAMARESUTILSGUI_H #define CALAMARESUTILSGUI_H
#include "UiDllMacro.h" #include "DllMacro.h"
#include <QObject> #include <QObject>
#include <QPixmap> #include <QPixmap>

View File

@ -28,7 +28,7 @@
#include <QPixmap> #include <QPixmap>
#include "UiDllMacro.h" #include "DllMacro.h"
#include "utils/CalamaresUtilsGui.h" #include "utils/CalamaresUtilsGui.h"
class UIDLLEXPORT ImageRegistry class UIDLLEXPORT ImageRegistry

View File

@ -34,7 +34,7 @@ namespace CalamaresUtils
* *
* If there is a return value from the QML method, it is logged (but not otherwise used). * If there is a return value from the QML method, it is logged (but not otherwise used).
*/ */
DLLEXPORT void UIDLLEXPORT void
callQMLFunction( QQuickItem* qmlObject, const char* method ); callQMLFunction( QQuickItem* qmlObject, const char* method );
} // namespace CalamaresUtils } // namespace CalamaresUtils

View File

@ -21,7 +21,7 @@
#define VIEWSTEP_H #define VIEWSTEP_H
#include "Job.h" #include "Job.h"
#include "UiDllMacro.h" #include "DllMacro.h"
#include "modulesystem/InstanceKey.h" #include "modulesystem/InstanceKey.h"
#include "modulesystem/Requirement.h" #include "modulesystem/Requirement.h"

View File

@ -12,7 +12,7 @@ include_directories(
if( BUILD_TESTING ) if( BUILD_TESTING )
add_executable( test_conf test_conf.cpp ) add_executable( test_conf test_conf.cpp )
target_link_libraries( test_conf ${YAMLCPP_LIBRARY} Qt5::Core ) target_link_libraries( test_conf yamlcpp Qt5::Core )
target_include_directories( test_conf PUBLIC ${YAMLCPP_INCLUDE_DIR} ) target_include_directories( test_conf PUBLIC ${YAMLCPP_INCLUDE_DIR} )
endif() endif()

View File

@ -8,18 +8,11 @@ calamares_add_plugin( contextualprocess
SHARED_LIB SHARED_LIB
) )
if( ECM_FOUND AND BUILD_TESTING ) calamares_add_test(
ecm_add_test( contextualprocesstest
SOURCES
Tests.cpp Tests.cpp
ContextualProcessJob.cpp # Builds it a second time ContextualProcessJob.cpp # Builds it a second time
TEST_NAME LIBRARIES
contextualprocesstest yamlcpp
LINK_LIBRARIES )
${CALAMARES_LIBRARIES}
calamaresui
${YAMLCPP_LIBRARY}
Qt5::Core
Qt5::Test
)
calamares_automoc( contextualprocesstest )
endif()

View File

@ -23,7 +23,7 @@
#include <QVariantMap> #include <QVariantMap>
#include "CppJob.h" #include "CppJob.h"
#include "PluginDllMacro.h" #include "DllMacro.h"
#include "utils/PluginFactory.h" #include "utils/PluginFactory.h"

View File

@ -27,7 +27,7 @@
#include <utils/PluginFactory.h> #include <utils/PluginFactory.h>
#include <PluginDllMacro.h> #include <DllMacro.h>
class PLUGINDLLEXPORT DracutLuksCfgJob : public Calamares::CppJob class PLUGINDLLEXPORT DracutLuksCfgJob : public Calamares::CppJob
{ {

View File

@ -27,7 +27,7 @@
#include <utils/PluginFactory.h> #include <utils/PluginFactory.h>
#include <PluginDllMacro.h> #include <DllMacro.h>
class PLUGINDLLEXPORT DummyCppJob : public Calamares::CppJob class PLUGINDLLEXPORT DummyCppJob : public Calamares::CppJob
{ {

View File

@ -25,7 +25,7 @@
#include "utils/PluginFactory.h" #include "utils/PluginFactory.h"
#include "viewpages/ViewStep.h" #include "viewpages/ViewStep.h"
#include "PluginDllMacro.h" #include "DllMacro.h"
class FinishedPage; class FinishedPage;

View File

@ -30,21 +30,15 @@ if ( KPMcore_FOUND AND Qt5DBus_FOUND AND KF5CoreAddons_FOUND AND KF5Config_FOUND
SHARED_LIB SHARED_LIB
) )
if( ECM_FOUND AND BUILD_TESTING ) calamares_add_test(
ecm_add_test(
Tests.cpp
TEST_NAME
fsresizertest fsresizertest
LINK_LIBRARIES SOURCES
${CALAMARES_LIBRARIES} Tests.cpp
calamares LIBRARIES
calamares_job_fsresizer # From above calamares_job_fsresizer # From above
${YAMLCPP_LIBRARY} yamlcpp
Qt5::Core
Qt5::Test
) )
set_target_properties( fsresizertest PROPERTIES AUTOMOC TRUE ) if( TARGET fsresizertest )
target_include_directories( fsresizertest PRIVATE /usr/local/include )
target_compile_definitions( fsresizertest PRIVATE ${_partition_defs} ) target_compile_definitions( fsresizertest PRIVATE ${_partition_defs} )
endif() endif()
else() else()

View File

@ -27,7 +27,7 @@
#include "partition/PartitionSize.h" #include "partition/PartitionSize.h"
#include "utils/PluginFactory.h" #include "utils/PluginFactory.h"
#include <PluginDllMacro.h> #include <DllMacro.h>
class CoreBackend; // From KPMCore class CoreBackend; // From KPMCore
class Device; // From KPMCore class Device; // From KPMCore

View File

@ -32,18 +32,11 @@ if ( KF5CoreAddons_FOUND AND KF5CoreAddons_VERSION VERSION_GREATER_EQUAL 5.58 )
target_compile_definitions( calamares_job_hostinfo PRIVATE WITH_KOSRelease ) target_compile_definitions( calamares_job_hostinfo PRIVATE WITH_KOSRelease )
endif() endif()
if( ECM_FOUND AND BUILD_TESTING ) calamares_add_test(
ecm_add_test(
Tests.cpp
HostInfoJob.cpp # Builds it a second time
TEST_NAME
hostinfotest hostinfotest
LINK_LIBRARIES SOURCES
${CALAMARES_LIBRARIES} Tests.cpp
calamaresui HostInfoJob.cpp
${YAMLCPP_LIBRARY} LIBRARIES
Qt5::Core yamlcpp
Qt5::Test )
)
calamares_automoc( hostinfotest )
endif()

View File

@ -20,7 +20,7 @@
#define HOSTINFOJOB_H #define HOSTINFOJOB_H
#include "CppJob.h" #include "CppJob.h"
#include "PluginDllMacro.h" #include "DllMacro.h"
#include "utils/PluginFactory.h" #include "utils/PluginFactory.h"
#include <QObject> #include <QObject>

View File

@ -8,18 +8,11 @@ calamares_add_plugin( initcpio
SHARED_LIB SHARED_LIB
) )
if( ECM_FOUND AND BUILD_TESTING ) calamares_add_test(
ecm_add_test(
Tests.cpp
TEST_NAME
initcpiotest initcpiotest
LINK_LIBRARIES SOURCES
${CALAMARES_LIBRARIES} Tests.cpp
calamares LIBRARIES
calamares_job_initcpio # From above calamares_job_initcpio # From above
${YAMLCPP_LIBRARY} yamlcpp
Qt5::Core )
Qt5::Test
)
calamares_automoc( initcpiotest )
endif()

View File

@ -20,7 +20,7 @@
#define INITCPIOJOB_H #define INITCPIOJOB_H
#include "CppJob.h" #include "CppJob.h"
#include "PluginDllMacro.h" #include "DllMacro.h"
#include "utils/PluginFactory.h" #include "utils/PluginFactory.h"
#include <QObject> #include <QObject>

View File

@ -8,18 +8,11 @@ calamares_add_plugin( initramfs
SHARED_LIB SHARED_LIB
) )
if( ECM_FOUND AND BUILD_TESTING ) calamares_add_test(
ecm_add_test(
Tests.cpp
TEST_NAME
initramfstest initramfstest
LINK_LIBRARIES SOURCES
${CALAMARES_LIBRARIES} Tests.cpp
calamares LIBRARIES
calamares_job_initramfs # From above calamares_job_initramfs # From above
${YAMLCPP_LIBRARY} yamlcpp
Qt5::Core )
Qt5::Test
)
calamares_automoc( initramfstest )
endif()

View File

@ -54,7 +54,7 @@ InitramfsJob::exec()
// First make sure we generate a safe initramfs with suitable permissions. // First make sure we generate a safe initramfs with suitable permissions.
static const char confFile[] = "/etc/initramfs-tools/conf.d/calamares-safe-initramfs.conf"; static const char confFile[] = "/etc/initramfs-tools/conf.d/calamares-safe-initramfs.conf";
static const char contents[] = "UMASK=0077\n"; static const char contents[] = "UMASK=0077\n";
if ( CalamaresUtils::System::instance()->createTargetFile( confFile, QByteArray( contents ) ).isEmpty() ) if ( CalamaresUtils::System::instance()->createTargetFile( confFile, QByteArray( contents ) ).failed() )
{ {
cWarning() << Logger::SubEntry << "Could not configure safe UMASK for initramfs."; cWarning() << Logger::SubEntry << "Could not configure safe UMASK for initramfs.";
// But continue anyway. // But continue anyway.

View File

@ -20,7 +20,7 @@
#define INITRAMFSJOB_H #define INITRAMFSJOB_H
#include "CppJob.h" #include "CppJob.h"
#include "PluginDllMacro.h" #include "DllMacro.h"
#include "utils/PluginFactory.h" #include "utils/PluginFactory.h"
#include <QObject> #include <QObject>

View File

@ -59,7 +59,10 @@ void InitramfsTests::testCreateHostFile()
{ {
CalamaresUtils::System s( false ); // don't chroot CalamaresUtils::System s( false ); // don't chroot
QString path = s.createTargetFile( confFile, QByteArray( contents ) ); auto r = s.createTargetFile( confFile, QByteArray( contents ) );
QVERIFY( !r.failed() );
QVERIFY( r );
QString path = r.path();
QVERIFY( !path.isEmpty() ); QVERIFY( !path.isEmpty() );
QCOMPARE( path, confFile ); // don't chroot, so path create relative to / QCOMPARE( path, confFile ); // don't chroot, so path create relative to /
QVERIFY( QFile::exists( confFile ) ); QVERIFY( QFile::exists( confFile ) );
@ -76,7 +79,10 @@ void InitramfsTests::testCreateTargetFile()
static const char short_confFile[] = "/calamares-safe-umask"; static const char short_confFile[] = "/calamares-safe-umask";
CalamaresUtils::System s( true ); CalamaresUtils::System s( true );
QString path = s.createTargetFile( short_confFile, QByteArray( contents ) ); auto r = s.createTargetFile( short_confFile, QByteArray( contents ) );
QVERIFY( r.failed() );
QVERIFY( !r );
QString path = r.path();
QVERIFY( path.isEmpty() ); // because no rootmountpoint is set QVERIFY( path.isEmpty() ); // because no rootmountpoint is set
Calamares::JobQueue j; Calamares::JobQueue j;

View File

@ -25,7 +25,7 @@
#include <utils/PluginFactory.h> #include <utils/PluginFactory.h>
#include <viewpages/ViewStep.h> #include <viewpages/ViewStep.h>
#include <PluginDllMacro.h> #include <DllMacro.h>
class InteractiveTerminalPage; class InteractiveTerminalPage;

View File

@ -25,7 +25,7 @@
#include <utils/PluginFactory.h> #include <utils/PluginFactory.h>
#include <viewpages/ViewStep.h> #include <viewpages/ViewStep.h>
#include <PluginDllMacro.h> #include <DllMacro.h>
class KeyboardPage; class KeyboardPage;

View File

@ -20,7 +20,7 @@
#ifndef LICENSEPAGEPLUGIN_H #ifndef LICENSEPAGEPLUGIN_H
#define LICENSEPAGEPLUGIN_H #define LICENSEPAGEPLUGIN_H
#include <PluginDllMacro.h> #include <DllMacro.h>
#include <utils/PluginFactory.h> #include <utils/PluginFactory.h>
#include <viewpages/ViewStep.h> #include <viewpages/ViewStep.h>

View File

@ -27,19 +27,13 @@ calamares_add_plugin( locale
calamaresui calamaresui
Qt5::Network Qt5::Network
${geoip_libs} ${geoip_libs}
${YAMLCPP_LIBRARY} yamlcpp
SHARED_LIB SHARED_LIB
) )
if( ECM_FOUND AND BUILD_TESTING ) calamares_add_test(
ecm_add_test( localetest
SOURCES
Tests.cpp Tests.cpp
LocaleConfiguration.cpp LocaleConfiguration.cpp
TEST_NAME )
localetest
LINK_LIBRARIES
calamares
Qt5::Test
)
calamares_automoc( localetest )
endif()

View File

@ -25,7 +25,7 @@
#include "utils/PluginFactory.h" #include "utils/PluginFactory.h"
#include "viewpages/ViewStep.h" #include "viewpages/ViewStep.h"
#include "PluginDllMacro.h" #include "DllMacro.h"
#include <QFutureWatcher> #include <QFutureWatcher>
#include <QObject> #include <QObject>

View File

@ -205,7 +205,7 @@ LuksBootKeyFileJob::exec()
if ( !setupLuks( d ) ) if ( !setupLuks( d ) )
return Calamares::JobResult::error( return Calamares::JobResult::error(
tr( "Encrypted rootfs setup error" ), tr( "Encrypted rootfs setup error" ),
tr( "Could configure LUKS key file on partition %1." ).arg( d.device ) ); tr( "Could not configure LUKS key file on partition %1." ).arg( d.device ) );
} }
return Calamares::JobResult::ok(); return Calamares::JobResult::ok();

View File

@ -20,7 +20,7 @@
#define LUKSBOOTKEYFILEJOB_H #define LUKSBOOTKEYFILEJOB_H
#include "CppJob.h" #include "CppJob.h"
#include "PluginDllMacro.h" #include "DllMacro.h"
#include "utils/PluginFactory.h" #include "utils/PluginFactory.h"
#include <QObject> #include <QObject>

View File

@ -9,17 +9,10 @@ calamares_add_plugin( machineid
SHARED_LIB SHARED_LIB
) )
if ( ECM_FOUND AND BUILD_TESTING ) calamares_add_test(
ecm_add_test( machineidtest
SOURCES
Tests.cpp Tests.cpp
MachineIdJob.cpp MachineIdJob.cpp
Workers.cpp Workers.cpp
TEST_NAME )
machineidtest
LINK_LIBRARIES
calamares
Qt5::Core
Qt5::Test
)
calamares_automoc( machineidtest )
endif()

View File

@ -26,7 +26,7 @@
#include <utils/PluginFactory.h> #include <utils/PluginFactory.h>
#include <PluginDllMacro.h> #include <DllMacro.h>
class PLUGINDLLEXPORT MachineIdJob : public Calamares::CppJob class PLUGINDLLEXPORT MachineIdJob : public Calamares::CppJob
{ {

View File

@ -28,6 +28,9 @@
#include <QFile> #include <QFile>
#include <QtTest/QtTest> #include <QtTest/QtTest>
// Internals of Workers.cpp
extern int getUrandomPoolSize();
class MachineIdTests : public QObject class MachineIdTests : public QObject
{ {
Q_OBJECT Q_OBJECT
@ -93,10 +96,10 @@ MachineIdTests::testPoolSize()
{ {
#ifdef Q_OS_FREEBSD #ifdef Q_OS_FREEBSD
// It hardly makes sense, but also the /proc entry is missing // It hardly makes sense, but also the /proc entry is missing
QCOMPARE( MachineId::getUrandomPoolSize(), 512 ); QCOMPARE( getUrandomPoolSize(), 512 );
#else #else
// Based on a sample size of 1, Netrunner // Based on a sample size of 1, Netrunner
QCOMPARE( MachineId::getUrandomPoolSize(), 4096 ); QCOMPARE( getUrandomPoolSize(), 4096 );
#endif #endif
} }
@ -122,8 +125,13 @@ MachineIdTests::testJob()
gs->insert( "rootMountPoint", "/tmp" ); gs->insert( "rootMountPoint", "/tmp" );
// Prepare part of the target filesystem // Prepare part of the target filesystem
{
QVERIFY( system->createTargetDirs("/etc") ); QVERIFY( system->createTargetDirs("/etc") );
QVERIFY( !(system->createTargetFile( "/etc/machine-id", "Hello" ).isEmpty() ) ); auto r = system->createTargetFile( "/etc/machine-id", "Hello" );
QVERIFY( !r.failed() );
QVERIFY( r );
QVERIFY( !r.path().isEmpty() );
}
MachineIdJob job( nullptr ); MachineIdJob job( nullptr );
QVERIFY( !job.prettyName().isEmpty() ); QVERIFY( !job.prettyName().isEmpty() );

View File

@ -27,6 +27,34 @@
#include <QFile> #include <QFile>
/// @brief Returns a recommended size for the entropy pool (in bytes)
STATICTEST int
getUrandomPoolSize()
{
QFile f( "/proc/sys/kernel/random/poolsize" );
constexpr const int minimumPoolSize = 512;
int poolSize = minimumPoolSize;
if ( f.exists() && f.open( QIODevice::ReadOnly | QIODevice::Text ) )
{
QByteArray v = f.read( 16 );
if ( v.length() > 2 )
{
if ( v.endsWith( '\n' ) )
{
v.chop( 1 );
}
bool ok = false;
poolSize = v.toInt( &ok );
if ( !ok )
{
poolSize = minimumPoolSize;
}
}
}
return ( poolSize >= minimumPoolSize ) ? poolSize : minimumPoolSize;
}
namespace MachineId namespace MachineId
{ {
@ -59,33 +87,6 @@ copyFile( const QString& rootMountPoint, const QString& fileName )
return Calamares::JobResult::ok(); return Calamares::JobResult::ok();
} }
int
getUrandomPoolSize()
{
QFile f( "/proc/sys/kernel/random/poolsize" );
constexpr const int minimumPoolSize = 512;
int poolSize = minimumPoolSize;
if ( f.exists() && f.open( QIODevice::ReadOnly | QIODevice::Text ) )
{
QByteArray v = f.read( 16 );
if ( v.length() > 2 )
{
if ( v.endsWith( '\n' ) )
{
v.chop( 1 );
}
bool ok = false;
poolSize = v.toInt( &ok );
if ( !ok )
{
poolSize = minimumPoolSize;
}
}
}
return ( poolSize >= minimumPoolSize ) ? poolSize : minimumPoolSize;
}
Calamares::JobResult Calamares::JobResult
createNewEntropy( int poolSize, const QString& rootMountPoint, const QString& fileName ) createNewEntropy( int poolSize, const QString& rootMountPoint, const QString& fileName )
{ {

View File

@ -48,9 +48,6 @@ enum class EntropyGeneration
CopyFromHost CopyFromHost
}; };
/// @brief Returns a recommended size for the entropy pool (in bytes)
int getUrandomPoolSize();
/// @brief Creates a new entropy file @p fileName in the target system at @p rootMountPoint /// @brief Creates a new entropy file @p fileName in the target system at @p rootMountPoint
Calamares::JobResult createNewEntropy( int poolSize, const QString& rootMountPoint, const QString& fileName ); Calamares::JobResult createNewEntropy( int poolSize, const QString& rootMountPoint, const QString& fileName );

View File

@ -1,5 +1,3 @@
include_directories( ${PROJECT_BINARY_DIR}/src/libcalamaresui )
calamares_add_plugin( netinstall calamares_add_plugin( netinstall
TYPE viewmodule TYPE viewmodule
EXPORT_MACRO PLUGINDLLEXPORT_PRO EXPORT_MACRO PLUGINDLLEXPORT_PRO
@ -15,6 +13,6 @@ calamares_add_plugin( netinstall
LINK_PRIVATE_LIBRARIES LINK_PRIVATE_LIBRARIES
calamaresui calamaresui
Qt5::Network Qt5::Network
${YAMLCPP_LIBRARY} yamlcpp
SHARED_LIB SHARED_LIB
) )

View File

@ -20,15 +20,15 @@
#include "NetInstallViewStep.h" #include "NetInstallViewStep.h"
#include "JobQueue.h"
#include "GlobalStorage.h" #include "GlobalStorage.h"
#include "JobQueue.h"
#include "utils/Logger.h" #include "utils/Logger.h"
#include "utils/Variant.h" #include "utils/Variant.h"
#include "NetInstallPage.h" #include "NetInstallPage.h"
CALAMARES_PLUGIN_FACTORY_DEFINITION( NetInstallViewStepFactory, registerPlugin<NetInstallViewStep>(); ) CALAMARES_PLUGIN_FACTORY_DEFINITION( NetInstallViewStepFactory, registerPlugin< NetInstallViewStep >(); )
NetInstallViewStep::NetInstallViewStep( QObject* parent ) NetInstallViewStep::NetInstallViewStep( QObject* parent )
: Calamares::ViewStep( parent ) : Calamares::ViewStep( parent )
@ -36,15 +36,16 @@ NetInstallViewStep::NetInstallViewStep( QObject* parent )
, m_nextEnabled( false ) , m_nextEnabled( false )
{ {
emit nextStatusChanged( true ); emit nextStatusChanged( true );
connect( m_widget, &NetInstallPage::checkReady, connect( m_widget, &NetInstallPage::checkReady, this, &NetInstallViewStep::nextIsReady );
this, &NetInstallViewStep::nextIsReady );
} }
NetInstallViewStep::~NetInstallViewStep() NetInstallViewStep::~NetInstallViewStep()
{ {
if ( m_widget && m_widget->parent() == nullptr ) if ( m_widget && m_widget->parent() == nullptr )
{
m_widget->deleteLater(); m_widget->deleteLater();
}
} }
@ -110,58 +111,70 @@ NetInstallViewStep::onActivate()
m_widget->onActivate(); m_widget->onActivate();
} }
void void
NetInstallViewStep::onLeave() NetInstallViewStep::onLeave()
{ {
cDebug() << "Leaving netinstall, adding packages to be installed"
<< "to global storage";
PackageModel::PackageItemDataList packages = m_widget->selectedPackages(); PackageModel::PackageItemDataList packages = m_widget->selectedPackages();
cDebug() << "Netinstall: Processing" << packages.length() << "packages.";
static const char PACKAGEOP[] = "packageOperations";
// Check if there's already a PACAKGEOP entry in GS, and if so we'll
// extend that one (overwriting the value in GS at the end of this method)
Calamares::GlobalStorage* gs = Calamares::JobQueue::instance()->globalStorage();
QVariantList packageOperations = gs->contains( PACKAGEOP ) ? gs->value( PACKAGEOP ).toList() : QVariantList();
cDebug() << Logger::SubEntry << "Existing package operations length" << packageOperations.length();
// Clear out existing operations for this module, going backwards:
// Sometimes we remove an item, and we don't want the index to
// fall off the end of the list.
for ( int index = packageOperations.length() - 1; 0 <= index ; index-- )
{
const QVariantMap op = packageOperations.at(index).toMap();
if ( op.contains( "source" ) && op.value( "source" ).toString() == moduleInstanceKey().toString() )
{
cDebug() << Logger::SubEntry << "Removing existing operations for" << moduleInstanceKey();
packageOperations.removeAt( index );
}
}
// This netinstall module may add two sub-steps to the packageOperations,
// one for installing and one for try-installing.
QVariantList installPackages; QVariantList installPackages;
QVariantList tryInstallPackages; QVariantList tryInstallPackages;
QVariantList packageOperations;
cDebug() << "Processing" << packages.length() << "packages from netinstall."; for ( const auto& package : packages )
for ( auto package : packages )
{ {
QVariant details( package.packageName );
// If it's a package with a pre- or post-script, replace
// with the more complicated datastructure.
if ( !package.preScript.isEmpty() || !package.postScript.isEmpty() )
{
QMap<QString, QVariant> sdetails;
sdetails.insert( "pre-script", package.preScript );
sdetails.insert( "package", package.packageName );
sdetails.insert( "post-script", package.postScript );
details = sdetails;
}
if ( package.isCritical ) if ( package.isCritical )
installPackages.append( details ); {
installPackages.append( package.toOperation() );
}
else else
tryInstallPackages.append( details ); {
tryInstallPackages.append( package.toOperation() );
}
} }
if ( !installPackages.empty() ) if ( !installPackages.empty() )
{ {
QMap<QString, QVariant> op; QVariantMap op;
op.insert( "install", QVariant( installPackages ) ); op.insert( "install", QVariant( installPackages ) );
op.insert( "source", moduleInstanceKey().toString() );
packageOperations.append( op ); packageOperations.append( op );
cDebug() << Logger::SubEntry << installPackages.length() << "critical packages."; cDebug() << Logger::SubEntry << installPackages.length() << "critical packages.";
} }
if ( !tryInstallPackages.empty() ) if ( !tryInstallPackages.empty() )
{ {
QMap<QString, QVariant> op; QVariantMap op;
op.insert( "try_install", QVariant( tryInstallPackages ) ); op.insert( "try_install", QVariant( tryInstallPackages ) );
op.insert( "source", moduleInstanceKey().toString() );
packageOperations.append( op ); packageOperations.append( op );
cDebug() << Logger::SubEntry << tryInstallPackages.length() << "non-critical packages."; cDebug() << Logger::SubEntry << tryInstallPackages.length() << "non-critical packages.";
} }
if ( !packageOperations.isEmpty() ) if ( !packageOperations.isEmpty() )
{ {
Calamares::GlobalStorage* gs = Calamares::JobQueue::instance()->globalStorage(); gs->insert( PACKAGEOP, packageOperations );
gs->insert( "packageOperations", QVariant( packageOperations ) );
} }
} }

View File

@ -23,7 +23,7 @@
#include <utils/PluginFactory.h> #include <utils/PluginFactory.h>
#include <viewpages/ViewStep.h> #include <viewpages/ViewStep.h>
#include <PluginDllMacro.h> #include <DllMacro.h>
#include <QVariant> #include <QVariant>

View File

@ -21,9 +21,9 @@
#include "utils/Yaml.h" #include "utils/Yaml.h"
PackageModel::PackageModel( const YAML::Node& data, QObject* parent ) : PackageModel::PackageModel( const YAML::Node& data, QObject* parent )
QAbstractItemModel( parent ), : QAbstractItemModel( parent )
m_columnHeadings() , m_columnHeadings()
{ {
m_rootItem = new PackageTreeItem(); m_rootItem = new PackageTreeItem();
setupModelData( data, m_rootItem ); setupModelData( data, m_rootItem );
@ -38,33 +38,47 @@ QModelIndex
PackageModel::index( int row, int column, const QModelIndex& parent ) const PackageModel::index( int row, int column, const QModelIndex& parent ) const
{ {
if ( !hasIndex( row, column, parent ) ) if ( !hasIndex( row, column, parent ) )
{
return QModelIndex(); return QModelIndex();
}
PackageTreeItem* parentItem; PackageTreeItem* parentItem;
if ( !parent.isValid() ) if ( !parent.isValid() )
{
parentItem = m_rootItem; parentItem = m_rootItem;
}
else else
parentItem = static_cast<PackageTreeItem*>( parent.internalPointer() ); {
parentItem = static_cast< PackageTreeItem* >( parent.internalPointer() );
}
PackageTreeItem* childItem = parentItem->child( row ); PackageTreeItem* childItem = parentItem->child( row );
if ( childItem ) if ( childItem )
{
return createIndex( row, column, childItem ); return createIndex( row, column, childItem );
}
else else
{
return QModelIndex(); return QModelIndex();
}
} }
QModelIndex QModelIndex
PackageModel::parent( const QModelIndex& index ) const PackageModel::parent( const QModelIndex& index ) const
{ {
if ( !index.isValid() ) if ( !index.isValid() )
{
return QModelIndex(); return QModelIndex();
}
PackageTreeItem* child = static_cast<PackageTreeItem*>( index.internalPointer() ); PackageTreeItem* child = static_cast< PackageTreeItem* >( index.internalPointer() );
PackageTreeItem* parent = child->parentItem(); PackageTreeItem* parent = child->parentItem();
if ( parent == m_rootItem ) if ( parent == m_rootItem )
{
return QModelIndex(); return QModelIndex();
}
return createIndex( parent->row(), 0, parent ); return createIndex( parent->row(), 0, parent );
} }
@ -72,13 +86,19 @@ int
PackageModel::rowCount( const QModelIndex& parent ) const PackageModel::rowCount( const QModelIndex& parent ) const
{ {
if ( parent.column() > 0 ) if ( parent.column() > 0 )
{
return 0; return 0;
}
PackageTreeItem* parentItem; PackageTreeItem* parentItem;
if ( !parent.isValid() ) if ( !parent.isValid() )
{
parentItem = m_rootItem; parentItem = m_rootItem;
}
else else
parentItem = static_cast<PackageTreeItem*>( parent.internalPointer() ); {
parentItem = static_cast< PackageTreeItem* >( parent.internalPointer() );
}
return parentItem->childCount(); return parentItem->childCount();
} }
@ -87,7 +107,9 @@ int
PackageModel::columnCount( const QModelIndex& parent ) const PackageModel::columnCount( const QModelIndex& parent ) const
{ {
if ( parent.isValid() ) if ( parent.isValid() )
return static_cast<PackageTreeItem*>( parent.internalPointer() )->columnCount(); {
return static_cast< PackageTreeItem* >( parent.internalPointer() )->columnCount();
}
return m_rootItem->columnCount(); return m_rootItem->columnCount();
} }
@ -95,17 +117,25 @@ QVariant
PackageModel::data( const QModelIndex& index, int role ) const PackageModel::data( const QModelIndex& index, int role ) const
{ {
if ( !index.isValid() ) if ( !index.isValid() )
{
return QVariant(); return QVariant();
}
PackageTreeItem* item = static_cast<PackageTreeItem*>( index.internalPointer() ); PackageTreeItem* item = static_cast< PackageTreeItem* >( index.internalPointer() );
if ( index.column() == 0 && role == Qt::CheckStateRole ) if ( index.column() == 0 && role == Qt::CheckStateRole )
{
return item->isSelected(); return item->isSelected();
}
if ( item->isHidden() && role == Qt::DisplayRole ) // Hidden group if ( item->isHidden() && role == Qt::DisplayRole ) // Hidden group
{
return QVariant(); return QVariant();
}
if ( role == Qt::DisplayRole ) if ( role == Qt::DisplayRole )
{
return item->data( index.column() ); return item->data( index.column() );
}
return QVariant(); return QVariant();
} }
@ -114,27 +144,31 @@ PackageModel::setData( const QModelIndex& index, const QVariant& value, int role
{ {
if ( role == Qt::CheckStateRole && index.isValid() ) if ( role == Qt::CheckStateRole && index.isValid() )
{ {
PackageTreeItem* item = static_cast<PackageTreeItem*>( index.internalPointer() ); PackageTreeItem* item = static_cast< PackageTreeItem* >( index.internalPointer() );
item->setSelected( static_cast<Qt::CheckState>( value.toInt() ) ); item->setSelected( static_cast< Qt::CheckState >( value.toInt() ) );
emit dataChanged( this->index( 0, 0 ), index.sibling( index.column(), index.row() + 1 ), emit dataChanged( this->index( 0, 0 ),
QVector<int>( Qt::CheckStateRole ) ); index.sibling( index.column(), index.row() + 1 ),
QVector< int >( Qt::CheckStateRole ) );
} }
return true; return true;
} }
bool bool
PackageModel::setHeaderData( int section, Qt::Orientation orientation, PackageModel::setHeaderData( int section, Qt::Orientation orientation, const QVariant& value, int role )
const QVariant& value, int role )
{ {
Q_UNUSED( role ) Q_UNUSED( role )
if ( orientation == Qt::Horizontal ) if ( orientation == Qt::Horizontal )
{ {
if ( m_columnHeadings.value( section ) != QVariant() ) if ( m_columnHeadings.value( section ) != QVariant() )
{
m_columnHeadings.replace( section, value ); m_columnHeadings.replace( section, value );
}
else else
{
m_columnHeadings.insert( section, value ); m_columnHeadings.insert( section, value );
}
emit headerDataChanged( orientation, section, section ); emit headerDataChanged( orientation, section, section );
} }
return true; return true;
@ -144,9 +178,13 @@ Qt::ItemFlags
PackageModel::flags( const QModelIndex& index ) const PackageModel::flags( const QModelIndex& index ) const
{ {
if ( !index.isValid() ) if ( !index.isValid() )
{
return Qt::ItemFlags(); return Qt::ItemFlags();
}
if ( index.column() == 0 ) if ( index.column() == 0 )
{
return Qt::ItemIsUserCheckable | QAbstractItemModel::flags( index ); return Qt::ItemIsUserCheckable | QAbstractItemModel::flags( index );
}
return QAbstractItemModel::flags( index ); return QAbstractItemModel::flags( index );
} }
@ -154,18 +192,22 @@ QVariant
PackageModel::headerData( int section, Qt::Orientation orientation, int role ) const PackageModel::headerData( int section, Qt::Orientation orientation, int role ) const
{ {
if ( orientation == Qt::Horizontal && role == Qt::DisplayRole ) if ( orientation == Qt::Horizontal && role == Qt::DisplayRole )
{
return m_columnHeadings.value( section ); return m_columnHeadings.value( section );
}
return QVariant(); return QVariant();
} }
QList<PackageTreeItem::ItemData> QList< PackageTreeItem::ItemData >
PackageModel::getPackages() const PackageModel::getPackages() const
{ {
QList<PackageTreeItem*> items = getItemPackages( m_rootItem ); QList< PackageTreeItem* > items = getItemPackages( m_rootItem );
for ( auto package : m_hiddenItems ) for ( auto package : m_hiddenItems )
if ( package->hiddenSelected() ) if ( package->hiddenSelected() )
{
items.append( getItemPackages( package ) ); items.append( getItemPackages( package ) );
QList<PackageTreeItem::ItemData> packages; }
QList< PackageTreeItem::ItemData > packages;
for ( auto item : items ) for ( auto item : items )
{ {
PackageTreeItem::ItemData itemData; PackageTreeItem::ItemData itemData;
@ -178,22 +220,27 @@ PackageModel::getPackages() const
return packages; return packages;
} }
QList<PackageTreeItem*> QList< PackageTreeItem* >
PackageModel::getItemPackages( PackageTreeItem* item ) const PackageModel::getItemPackages( PackageTreeItem* item ) const
{ {
QList<PackageTreeItem*> selectedPackages; QList< PackageTreeItem* > selectedPackages;
for ( int i = 0; i < item->childCount(); i++ ) for ( int i = 0; i < item->childCount(); i++ )
{ {
if ( item->child( i )->isSelected() == Qt::Unchecked ) if ( item->child( i )->isSelected() == Qt::Unchecked )
{
continue; continue;
}
if ( !item->child( i )->childCount() ) // package if ( !item->child( i )->childCount() ) // package
{
selectedPackages.append( item->child( i ) ); selectedPackages.append( item->child( i ) );
}
else else
{
selectedPackages.append( getItemPackages( item->child( i ) ) ); selectedPackages.append( getItemPackages( item->child( i ) ) );
} }
}
return selectedPackages; return selectedPackages;
} }
void void
@ -203,49 +250,57 @@ PackageModel::setupModelData( const YAML::Node& data, PackageTreeItem* parent )
{ {
const YAML::Node itemDefinition = *it; const YAML::Node itemDefinition = *it;
QString name( QString name( tr( CalamaresUtils::yamlToVariant( itemDefinition[ "name" ] ).toByteArray() ) );
tr( CalamaresUtils::yamlToVariant( itemDefinition["name"] ).toByteArray() ) ); QString description( tr( CalamaresUtils::yamlToVariant( itemDefinition[ "description" ] ).toByteArray() ) );
QString description(
tr( CalamaresUtils::yamlToVariant( itemDefinition["description"] ).toByteArray() ) );
PackageTreeItem::ItemData itemData; PackageTreeItem::ItemData itemData;
itemData.name = name; itemData.name = name;
itemData.description = description; itemData.description = description;
if ( itemDefinition["pre-install"] ) if ( itemDefinition[ "pre-install" ] )
itemData.preScript = {
CalamaresUtils::yamlToVariant( itemDefinition["pre-install"] ).toString(); itemData.preScript = CalamaresUtils::yamlToVariant( itemDefinition[ "pre-install" ] ).toString();
if ( itemDefinition["post-install"] ) }
itemData.postScript = if ( itemDefinition[ "post-install" ] )
CalamaresUtils::yamlToVariant( itemDefinition["post-install"] ).toString(); {
itemData.postScript = CalamaresUtils::yamlToVariant( itemDefinition[ "post-install" ] ).toString();
}
PackageTreeItem* item = new PackageTreeItem( itemData, parent ); PackageTreeItem* item = new PackageTreeItem( itemData, parent );
if ( itemDefinition["selected"] ) if ( itemDefinition[ "selected" ] )
item->setSelected( item->setSelected( CalamaresUtils::yamlToVariant( itemDefinition[ "selected" ] ).toBool() ? Qt::Checked
CalamaresUtils::yamlToVariant( itemDefinition["selected"] ).toBool() ? : Qt::Unchecked );
Qt::Checked : Qt::Unchecked );
else else
{
item->setSelected( parent->isSelected() ); // Inherit from it's parent item->setSelected( parent->isSelected() ); // Inherit from it's parent
}
if ( itemDefinition["hidden"] ) if ( itemDefinition[ "hidden" ] )
item->setHidden( {
CalamaresUtils::yamlToVariant( itemDefinition["hidden"] ).toBool() ); item->setHidden( CalamaresUtils::yamlToVariant( itemDefinition[ "hidden" ] ).toBool() );
}
if ( itemDefinition["critical"] ) if ( itemDefinition[ "critical" ] )
item->setCritical( {
CalamaresUtils::yamlToVariant( itemDefinition["critical"] ).toBool() ); item->setCritical( CalamaresUtils::yamlToVariant( itemDefinition[ "critical" ] ).toBool() );
}
if ( itemDefinition["packages"] ) if ( itemDefinition[ "packages" ] )
for ( YAML::const_iterator packageIt = itemDefinition["packages"].begin(); for ( YAML::const_iterator packageIt = itemDefinition[ "packages" ].begin();
packageIt != itemDefinition["packages"].end(); ++packageIt ) packageIt != itemDefinition[ "packages" ].end();
++packageIt )
item->appendChild( item->appendChild(
new PackageTreeItem( CalamaresUtils::yamlToVariant( *packageIt ).toString(), item ) ); new PackageTreeItem( CalamaresUtils::yamlToVariant( *packageIt ).toString(), item ) );
if ( itemDefinition["subgroups"] ) if ( itemDefinition[ "subgroups" ] )
setupModelData( itemDefinition["subgroups"], item ); {
setupModelData( itemDefinition[ "subgroups" ], item );
}
if ( item->isHidden() ) if ( item->isHidden() )
{
m_hiddenItems.append( item ); m_hiddenItems.append( item );
}
else else
{ {
item->setCheckable( true ); item->setCheckable( true );

View File

@ -29,7 +29,7 @@
namespace YAML namespace YAML
{ {
class Node; class Node;
} }
class PackageModel : public QAbstractItemModel class PackageModel : public QAbstractItemModel
@ -43,26 +43,23 @@ public:
~PackageModel() override; ~PackageModel() override;
QVariant data( const QModelIndex& index, int role ) const override; QVariant data( const QModelIndex& index, int role ) const override;
bool setData( const QModelIndex& index, const QVariant& value, bool setData( const QModelIndex& index, const QVariant& value, int role = Qt::EditRole ) override;
int role = Qt::EditRole ) override; bool
bool setHeaderData( int section, Qt::Orientation orientation, setHeaderData( int section, Qt::Orientation orientation, const QVariant& value, int role = Qt::EditRole ) override;
const QVariant& value, int role = Qt::EditRole ) override;
Qt::ItemFlags flags( const QModelIndex& index ) const override; Qt::ItemFlags flags( const QModelIndex& index ) const override;
QVariant headerData( int section, Qt::Orientation orientation, QVariant headerData( int section, Qt::Orientation orientation, int role = Qt::DisplayRole ) const override;
int role = Qt::DisplayRole ) const override; QModelIndex index( int row, int column, const QModelIndex& parent = QModelIndex() ) const override;
QModelIndex index( int row, int column,
const QModelIndex& parent = QModelIndex() ) const override;
QModelIndex parent( const QModelIndex& index ) const override; QModelIndex parent( const QModelIndex& index ) const override;
int rowCount( const QModelIndex& parent = QModelIndex() ) const override; int rowCount( const QModelIndex& parent = QModelIndex() ) const override;
int columnCount( const QModelIndex& parent = QModelIndex() ) const override; int columnCount( const QModelIndex& parent = QModelIndex() ) const override;
PackageItemDataList getPackages() const; PackageItemDataList getPackages() const;
QList<PackageTreeItem*> getItemPackages( PackageTreeItem* item ) const; QList< PackageTreeItem* > getItemPackages( PackageTreeItem* item ) const;
private: private:
void setupModelData( const YAML::Node& data, PackageTreeItem* parent ); void setupModelData( const YAML::Node& data, PackageTreeItem* parent );
PackageTreeItem* m_rootItem; PackageTreeItem* m_rootItem;
QList<PackageTreeItem*> m_hiddenItems; QList< PackageTreeItem* > m_hiddenItems;
QVariantList m_columnHeadings; QVariantList m_columnHeadings;
}; };

View File

@ -21,28 +21,52 @@
#include "utils/Logger.h" #include "utils/Logger.h"
QVariant
PackageTreeItem::ItemData::toOperation() const
{
// If it's a package with a pre- or post-script, replace
// with the more complicated datastructure.
if ( !preScript.isEmpty() || !postScript.isEmpty() )
{
QMap< QString, QVariant > sdetails;
sdetails.insert( "pre-script", preScript );
sdetails.insert( "package", packageName );
sdetails.insert( "post-script", postScript );
return sdetails;
}
else
{
return packageName;
}
}
PackageTreeItem::PackageTreeItem( const ItemData& data, PackageTreeItem* parent ) PackageTreeItem::PackageTreeItem( const ItemData& data, PackageTreeItem* parent )
: m_parentItem( parent ) : m_parentItem( parent )
, m_data( data ) , m_data( data )
{ } {
}
PackageTreeItem::PackageTreeItem( const QString packageName, PackageTreeItem* parent ) : PackageTreeItem::PackageTreeItem( const QString packageName, PackageTreeItem* parent )
m_parentItem( parent ) : m_parentItem( parent )
{ {
m_data.packageName = packageName; m_data.packageName = packageName;
if ( parent != nullptr ) if ( parent != nullptr )
{
m_data.selected = parent->isSelected(); m_data.selected = parent->isSelected();
}
else else
{
m_data.selected = Qt::Unchecked; m_data.selected = Qt::Unchecked;
}
} }
PackageTreeItem::PackageTreeItem( PackageTreeItem* parent ) : PackageTreeItem::PackageTreeItem( PackageTreeItem* parent )
m_parentItem( parent ) : m_parentItem( parent )
{ {
} }
PackageTreeItem::PackageTreeItem::PackageTreeItem() : PackageTreeItem::PackageTreeItem::PackageTreeItem()
PackageTreeItem( QString(), nullptr ) : PackageTreeItem( QString(), nullptr )
{ {
m_data.selected = Qt::Checked; m_data.selected = Qt::Checked;
m_data.name = QLatin1String( "<root>" ); m_data.name = QLatin1String( "<root>" );
@ -75,7 +99,9 @@ int
PackageTreeItem::row() const PackageTreeItem::row() const
{ {
if ( m_parentItem ) if ( m_parentItem )
return m_parentItem->m_childItems.indexOf( const_cast<PackageTreeItem*>( this ) ); {
return m_parentItem->m_childItems.indexOf( const_cast< PackageTreeItem* >( this ) );
}
return 0; return 0;
} }
@ -91,7 +117,9 @@ PackageTreeItem::data( int column ) const
if ( packageName() != nullptr ) // package if ( packageName() != nullptr ) // package
{ {
if ( !column ) if ( !column )
{
return QVariant( packageName() ); return QVariant( packageName() );
}
return QVariant(); return QVariant();
} }
switch ( column ) // group switch ( column ) // group
@ -164,14 +192,18 @@ bool
PackageTreeItem::hiddenSelected() const PackageTreeItem::hiddenSelected() const
{ {
Q_ASSERT( m_data.isHidden ); Q_ASSERT( m_data.isHidden );
if (! m_data.selected ) if ( !m_data.selected )
{
return false; return false;
}
const PackageTreeItem* currentItem = parentItem(); const PackageTreeItem* currentItem = parentItem();
while ( currentItem != nullptr ) while ( currentItem != nullptr )
{ {
if ( !currentItem->isHidden() ) if ( !currentItem->isHidden() )
{
return currentItem->isSelected() != Qt::Unchecked; return currentItem->isSelected() != Qt::Unchecked;
}
currentItem = currentItem->parentItem(); currentItem = currentItem->parentItem();
} }
@ -203,7 +235,9 @@ PackageTreeItem::setSelected( Qt::CheckState isSelected )
{ {
if ( parentItem() == nullptr ) if ( parentItem() == nullptr )
// This is the root, it is always checked so don't change state // This is the root, it is always checked so don't change state
{
return; return;
}
m_data.selected = isSelected; m_data.selected = isSelected;
setChildrenSelected( isSelected ); setChildrenSelected( isSelected );
@ -217,7 +251,9 @@ PackageTreeItem::setSelected( Qt::CheckState isSelected )
} }
if ( currentItem == nullptr ) if ( currentItem == nullptr )
// Reached the root .. don't bother // Reached the root .. don't bother
{
return; return;
}
// Figure out checked-state based on the children // Figure out checked-state based on the children
int childrenSelected = 0; int childrenSelected = 0;
@ -225,16 +261,26 @@ PackageTreeItem::setSelected( Qt::CheckState isSelected )
for ( int i = 0; i < currentItem->childCount(); i++ ) for ( int i = 0; i < currentItem->childCount(); i++ )
{ {
if ( currentItem->child( i )->isSelected() == Qt::Checked ) if ( currentItem->child( i )->isSelected() == Qt::Checked )
{
childrenSelected++; childrenSelected++;
}
if ( currentItem->child( i )->isSelected() == Qt::PartiallyChecked ) if ( currentItem->child( i )->isSelected() == Qt::PartiallyChecked )
{
childrenPartiallySelected++; childrenPartiallySelected++;
} }
if ( !childrenSelected && !childrenPartiallySelected) }
if ( !childrenSelected && !childrenPartiallySelected )
{
currentItem->setSelected( Qt::Unchecked ); currentItem->setSelected( Qt::Unchecked );
}
else if ( childrenSelected == currentItem->childCount() ) else if ( childrenSelected == currentItem->childCount() )
{
currentItem->setSelected( Qt::Checked ); currentItem->setSelected( Qt::Checked );
}
else else
{
currentItem->setSelected( Qt::PartiallyChecked ); currentItem->setSelected( Qt::PartiallyChecked );
}
} }
void void

View File

@ -21,8 +21,8 @@
#define PACKAGETREEITEM_H #define PACKAGETREEITEM_H
#include <QList> #include <QList>
#include <QVariant>
#include <QStandardItem> #include <QStandardItem>
#include <QVariant>
class PackageTreeItem : public QStandardItem class PackageTreeItem : public QStandardItem
{ {
@ -37,6 +37,13 @@ public:
bool isCritical = false; bool isCritical = false;
bool isHidden = false; bool isHidden = false;
Qt::CheckState selected = Qt::Unchecked; Qt::CheckState selected = Qt::Unchecked;
/** @brief Turns this item into a variant for PackageOperations use
*
* For "plain" items, this is just the package name; items with
* scripts return a map. See the package module for how it's interpreted.
*/
QVariant toOperation() const;
}; };
explicit PackageTreeItem( const ItemData& data, PackageTreeItem* parent = nullptr ); explicit PackageTreeItem( const ItemData& data, PackageTreeItem* parent = nullptr );
explicit PackageTreeItem( const QString packageName, PackageTreeItem* parent = nullptr ); explicit PackageTreeItem( const QString packageName, PackageTreeItem* parent = nullptr );
@ -78,9 +85,10 @@ public:
void setSelected( Qt::CheckState isSelected ); void setSelected( Qt::CheckState isSelected );
void setChildrenSelected( Qt::CheckState isSelected ); void setChildrenSelected( Qt::CheckState isSelected );
int type() const override; int type() const override;
private: private:
PackageTreeItem* m_parentItem; PackageTreeItem* m_parentItem;
QList<PackageTreeItem*> m_childItems; QList< PackageTreeItem* > m_childItems;
ItemData m_data; ItemData m_data;
const int m_columns = 2; // Name, description const int m_columns = 2; // Name, description
}; };

View File

@ -20,7 +20,7 @@
#ifndef NOTESQMLVIEWSTEP_H #ifndef NOTESQMLVIEWSTEP_H
#define NOTESQMLVIEWSTEP_H #define NOTESQMLVIEWSTEP_H
#include "PluginDllMacro.h" #include "DllMacro.h"
#include "locale/TranslatableConfiguration.h" #include "locale/TranslatableConfiguration.h"
#include "utils/CalamaresUtilsSystem.h" #include "utils/CalamaresUtilsSystem.h"
#include "utils/Variant.h" #include "utils/Variant.h"

View File

@ -22,7 +22,7 @@
#include <utils/PluginFactory.h> #include <utils/PluginFactory.h>
#include <viewpages/ViewStep.h> #include <viewpages/ViewStep.h>
#include <PluginDllMacro.h> #include <DllMacro.h>
#include <QVariantMap> #include <QVariantMap>

View File

@ -53,18 +53,12 @@ calamares_add_plugin( packagechooser
SHARED_LIB SHARED_LIB
) )
if( ECM_FOUND AND BUILD_TESTING ) calamares_add_test(
ecm_add_test(
Tests.cpp
TEST_NAME
packagechoosertest packagechoosertest
LINK_LIBRARIES GUI
${CALAMARES_LIBRARIES} SOURCES
Tests.cpp
LIBRARIES
calamares_viewmodule_packagechooser calamares_viewmodule_packagechooser
Qt5::Core
Qt5::Test
Qt5::Gui
${_extra_libraries} ${_extra_libraries}
) )
calamares_automoc( packagechoosertest)
endif()

View File

@ -19,7 +19,7 @@
#ifndef PACKAGECHOOSERVIEWSTEP_H #ifndef PACKAGECHOOSERVIEWSTEP_H
#define PACKAGECHOOSERVIEWSTEP_H #define PACKAGECHOOSERVIEWSTEP_H
#include "PluginDllMacro.h" #include "DllMacro.h"
#include "locale/TranslatableConfiguration.h" #include "locale/TranslatableConfiguration.h"
#include "utils/PluginFactory.h" #include "utils/PluginFactory.h"
#include "viewpages/ViewStep.h" #include "viewpages/ViewStep.h"

View File

@ -478,7 +478,10 @@ def run_operations(pkgman, entry):
else: else:
for package in package_list: for package in package_list:
pkgman.install_package(package, from_local=True) pkgman.install_package(package, from_local=True)
elif key == "source":
libcalamares.utils.debug("Package-list from {!s}".format(entry[key]))
else:
libcalamares.utils.warning("Unknown package-operation key {!s}".format(key))
completed_packages += len(package_list) completed_packages += len(package_list)
libcalamares.job.setprogress(completed_packages * 1.0 / total_packages) libcalamares.job.setprogress(completed_packages * 1.0 / total_packages)
libcalamares.utils.debug(pretty_name()) libcalamares.utils.debug(pretty_name())

View File

@ -55,22 +55,25 @@ update_system: false
# that is in this configuration file). # that is in this configuration file).
# #
# Allowed package operations are: # Allowed package operations are:
# - install, try_install: will call the package manager to # - *install*, *try_install*: will call the package manager to
# install one or more packages. The install target will # install one or more packages. The install target will
# abort the whole installation if package-installation # abort the whole installation if package-installation
# fails, while try_install carries on. Packages may be # fails, while try_install carries on. Packages may be
# listed as (localized) names, or as (localized) package-data. # listed as (localized) names, or as (localized) package-data.
# See below for the description of the format. # See below for the description of the format.
# - localInstall: this is used to call the package manager # - *localInstall*: this is used to call the package manager
# to install a package from a path-to-a-package. This is # to install a package from a path-to-a-package. This is
# useful if you have a static package archive on the install media. # useful if you have a static package archive on the install media.
# The *pacman* package manager is the only one to specially support # The *pacman* package manager is the only one to specially support
# this operation (all others treat this the same as *install*). # this operation (all others treat this the same as *install*).
# - remove, try_remove: will call the package manager to # - *remove*, *try_remove*: will call the package manager to
# remove one or more packages. The remove target will # remove one or more packages. The remove target will
# abort the whole installation if package-removal fails, # abort the whole installation if package-removal fails,
# while try_remove carries on. Packages may be listed as # while try_remove carries on. Packages may be listed as
# (localized) names. # (localized) names.
# One additional key is recognized, to help netinstall out:
# - *source*: ignored, does get logged
# Any other key is ignored, and logged as a warning.
# #
# There are two formats for naming packages: as a name or as package-data, # There are two formats for naming packages: as a name or as package-data,
# which is an object notation providing package-name, as well as pre- and # which is an object notation providing package-name, as well as pre- and

View File

@ -648,7 +648,7 @@ PartitionViewStep::checkRequirements()
{ {
QLatin1String( "partitions" ), QLatin1String( "partitions" ),
[]{ return tr( "has at least one disk device available." ); }, []{ return tr( "has at least one disk device available." ); },
[]{ return tr( "There are no partitons to install on." ); }, []{ return tr( "There are no partitions to install on." ); },
m_core->deviceModel()->rowCount() > 0, // satisfied m_core->deviceModel()->rowCount() > 0, // satisfied
#ifdef DEBUG_PARTITION_UNSAFE #ifdef DEBUG_PARTITION_UNSAFE
false // optional false // optional

View File

@ -24,7 +24,7 @@
#include <utils/PluginFactory.h> #include <utils/PluginFactory.h>
#include <viewpages/ViewStep.h> #include <viewpages/ViewStep.h>
#include <PluginDllMacro.h> #include <DllMacro.h>
#include "core/PartitionActions.h" #include "core/PartitionActions.h"

View File

@ -21,16 +21,12 @@ include_directories(
${CMAKE_CURRENT_BINARY_DIR} ${CMAKE_CURRENT_BINARY_DIR}
) )
if( ECM_FOUND AND BUILD_TESTING ) calamares_add_test(
ecm_add_test( ${partitionjobtests_SRCS} partitionjobtests
TEST_NAME partitionjobtests SOURCES ${partitionjobtests_SRCS}
LINK_LIBRARIES LIBRARIES
${CALAMARES_LIBRARIES}
kpmcore kpmcore
Qt5::Core )
Qt5::Test if( TARGET partitionjobtests )
)
set_target_properties( partitionjobtests PROPERTIES AUTOMOC TRUE )
target_compile_definitions( partitionjobtests PRIVATE ${_partition_defs} ) target_compile_definitions( partitionjobtests PRIVATE ${_partition_defs} )
endif() endif()

View File

@ -21,7 +21,7 @@
#include <utils/PluginFactory.h> #include <utils/PluginFactory.h>
#include <viewpages/ViewStep.h> #include <viewpages/ViewStep.h>
#include <PluginDllMacro.h> #include <DllMacro.h>
#include <QObject> #include <QObject>
#include <QUrl> #include <QUrl>

View File

@ -24,7 +24,7 @@
#include <QVariantMap> #include <QVariantMap>
#include "CppJob.h" #include "CppJob.h"
#include "PluginDllMacro.h" #include "DllMacro.h"
#include "utils/PluginFactory.h" #include "utils/PluginFactory.h"

View File

@ -8,17 +8,10 @@ calamares_add_plugin( shellprocess
SHARED_LIB SHARED_LIB
) )
if( ECM_FOUND AND BUILD_TESTING ) calamares_add_test(
ecm_add_test(
Tests.cpp
TEST_NAME
shellprocesstest shellprocesstest
LINK_LIBRARIES SOURCES
${CALAMARES_LIBRARIES} Tests.cpp
calamaresui LIBRARIES
${YAMLCPP_LIBRARY} yamlcpp
Qt5::Core )
Qt5::Test
)
calamares_automoc( shellprocesstest )
endif()

View File

@ -20,7 +20,7 @@
#define SHELLPROCESSJOB_H #define SHELLPROCESSJOB_H
#include "CppJob.h" #include "CppJob.h"
#include "PluginDllMacro.h" #include "DllMacro.h"
#include "utils/CommandList.h" #include "utils/CommandList.h"
#include "utils/PluginFactory.h" #include "utils/PluginFactory.h"

View File

@ -24,7 +24,7 @@
#include <utils/PluginFactory.h> #include <utils/PluginFactory.h>
#include <viewpages/ViewStep.h> #include <viewpages/ViewStep.h>
#include <PluginDllMacro.h> #include <DllMacro.h>
class SummaryPage; class SummaryPage;

View File

@ -21,7 +21,7 @@
#include "TrackingType.h" #include "TrackingType.h"
#include <PluginDllMacro.h> #include <DllMacro.h>
#include <utils/PluginFactory.h> #include <utils/PluginFactory.h>
#include <viewpages/ViewStep.h> #include <viewpages/ViewStep.h>

View File

@ -1,4 +1,4 @@
find_package( Qt5 COMPONENTS Core REQUIRED ) find_package( Qt5 ${QT_VERSION} CONFIG REQUIRED Core DBus Network )
find_package( Crypt REQUIRED ) find_package( Crypt REQUIRED )
# Add optional libraries here # Add optional libraries here
@ -36,20 +36,24 @@ calamares_add_plugin( users
calamaresui calamaresui
${CRYPT_LIBRARIES} ${CRYPT_LIBRARIES}
${USER_EXTRA_LIB} ${USER_EXTRA_LIB}
Qt5::DBus
SHARED_LIB SHARED_LIB
) )
if( ECM_FOUND AND BUILD_TESTING ) calamares_add_test(
ecm_add_test( userspasswordtest
SOURCES
PasswordTests.cpp PasswordTests.cpp
SetPasswordJob.cpp SetPasswordJob.cpp
TEST_NAME LIBRARIES
passwordtest
LINK_LIBRARIES
${CALAMARES_LIBRARIES}
Qt5::Core
Qt5::Test
${CRYPT_LIBRARIES} ${CRYPT_LIBRARIES}
) )
calamares_automoc( passwordtest )
endif() calamares_add_test(
userstest
SOURCES
Tests.cpp
SetHostNameJob.cpp
LIBRARIES
Qt5::DBus
)

View File

@ -2,7 +2,7 @@
* *
* Copyright 2014, Rohan Garg <rohan@kde.org> * Copyright 2014, Rohan Garg <rohan@kde.org>
* Copyright 2015, Teo Mrnjavac <teo@kde.org> * Copyright 2015, Teo Mrnjavac <teo@kde.org>
* Copyright 2018, Adriaan de Groot <groot@kde.org> * Copyright 2018, 2020, Adriaan de Groot <groot@kde.org>
* *
* Calamares is free software: you can redistribute it and/or modify * Calamares is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by * it under the terms of the GNU General Public License as published by
@ -22,14 +22,19 @@
#include "GlobalStorage.h" #include "GlobalStorage.h"
#include "JobQueue.h" #include "JobQueue.h"
#include "utils/CalamaresUtilsSystem.h"
#include "utils/Logger.h" #include "utils/Logger.h"
#include <QDir> #include <QDir>
#include <QFile> #include <QFile>
#include <QtDBus/QDBusConnection>
#include <QtDBus/QDBusInterface>
#include <QtDBus/QDBusReply>
SetHostNameJob::SetHostNameJob( const QString& hostname ) SetHostNameJob::SetHostNameJob( const QString& hostname, Actions a )
: Calamares::Job() : Calamares::Job()
, m_hostname( hostname ) , m_hostname( hostname )
, m_actions( a )
{ {
} }
@ -53,6 +58,68 @@ SetHostNameJob::prettyStatusMessage() const
return tr( "Setting hostname %1." ).arg( m_hostname ); return tr( "Setting hostname %1." ).arg( m_hostname );
} }
STATICTEST bool
setFileHostname( const QString& hostname )
{
return !( CalamaresUtils::System::instance()
->createTargetFile( QStringLiteral( "/etc/hostname" ), ( hostname + '\n' ).toUtf8() )
.failed() );
}
STATICTEST bool
writeFileEtcHosts( const QString& hostname )
{
// The actual hostname gets substituted in at %1
static const char etc_hosts[] = R"(# Host addresses
127.0.0.1 localhost
127.0.1.1 %1
::1 localhost ip6-localhost ip6-loopback
ff02::1 ip6-allnodes
ff02::2 ip6-allrouters
)";
return !( CalamaresUtils::System::instance()
->createTargetFile( QStringLiteral( "/etc/hosts" ), QString( etc_hosts ).arg( hostname ).toUtf8() )
.failed() );
}
STATICTEST bool
setSystemdHostname( const QString& hostname )
{
QDBusInterface hostnamed( "org.freedesktop.hostname1",
"/org/freedesktop/hostname1",
"org.freedesktop.hostname1",
QDBusConnection::systemBus() );
if ( !hostnamed.isValid() )
{
cWarning() << "Interface" << hostnamed.interface() << "is not valid.";
return false;
}
bool success = true;
// Static, writes /etc/hostname
{
QDBusReply< void > r = hostnamed.call( "SetStaticHostname", hostname, false );
if ( !r.isValid() )
{
cWarning() << "Could not set hostname through org.freedesktop.hostname1.SetStaticHostname." << r.error();
success = false;
}
}
// Dynamic, updates kernel
{
QDBusReply< void > r = hostnamed.call( "SetHostname", hostname, false );
if ( !r.isValid() )
{
cWarning() << "Could not set hostname through org.freedesktop.hostname1.SetHostname." << r.error();
success = false;
}
}
return success;
}
Calamares::JobResult Calamares::JobResult
SetHostNameJob::exec() SetHostNameJob::exec()
{ {
@ -71,43 +138,29 @@ SetHostNameJob::exec()
return Calamares::JobResult::error( tr( "Internal Error" ) ); return Calamares::JobResult::error( tr( "Internal Error" ) );
} }
QFile hostfile( destDir + "/etc/hostname" ); if ( m_actions & Action::EtcHostname )
if ( !hostfile.open( QFile::WriteOnly ) ) {
if ( !setFileHostname( m_hostname ) )
{ {
cError() << "Can't write to hostname file"; cError() << "Can't write to hostname file";
return Calamares::JobResult::error( tr( "Cannot write hostname to target system" ) ); return Calamares::JobResult::error( tr( "Cannot write hostname to target system" ) );
} }
}
QTextStream hostfileout( &hostfile ); if ( m_actions & Action::WriteEtcHosts )
hostfileout << m_hostname << "\n"; {
hostfile.close(); if ( !writeFileEtcHosts( m_hostname ) )
QFile hostsfile( destDir + "/etc/hosts" );
if ( !hostsfile.open( QFile::WriteOnly ) )
{ {
cError() << "Can't write to hosts file"; cError() << "Can't write to hosts file";
return Calamares::JobResult::error( tr( "Cannot write hostname to target system" ) ); return Calamares::JobResult::error( tr( "Cannot write hostname to target system" ) );
} }
}
// We also need to write the appropriate entries for /etc/hosts if ( m_actions & Action::SystemdHostname )
QTextStream hostsfileout( &hostsfile ); {
// ipv4 support // Does its own logging
hostsfileout << "127.0.0.1" setSystemdHostname( m_hostname );
<< "\t" }
<< "localhost"
<< "\n";
hostsfileout << "127.0.1.1"
<< "\t" << m_hostname << "\n";
// ipv6 support
hostsfileout << "::1"
<< "\t"
<< "localhost ip6-localhost ip6-loopback"
<< "\n";
hostsfileout << "ff02::1 ip6-allnodes"
<< "\n"
<< "ff02::2 ip6-allrouters"
<< "\n";
hostsfile.close();
return Calamares::JobResult::ok(); return Calamares::JobResult::ok();
} }

View File

@ -2,6 +2,7 @@
* *
* Copyright 2014, Rohan Garg <rohan@kde.org> * Copyright 2014, Rohan Garg <rohan@kde.org>
* Copyright 2015, Teo Mrnjavac <teo@kde.org> * Copyright 2015, Teo Mrnjavac <teo@kde.org>
* Copyright 2020, Adriaan de Groot <groot@kde.org>
* *
* Calamares is free software: you can redistribute it and/or modify * Calamares is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by * it under the terms of the GNU General Public License as published by
@ -26,7 +27,17 @@ class SetHostNameJob : public Calamares::Job
{ {
Q_OBJECT Q_OBJECT
public: public:
SetHostNameJob( const QString& hostname ); enum Action
{
None = 0x0,
EtcHostname = 0x1, // Write to /etc/hostname directly
SystemdHostname = 0x2, // Set via hostnamed(1)
WriteEtcHosts = 0x4 // Write /etc/hosts (127.0.1.1 is this host)
};
Q_DECLARE_FLAGS( Actions, Action )
SetHostNameJob( const QString& hostname, Actions a );
QString prettyName() const override; QString prettyName() const override;
QString prettyDescription() const override; QString prettyDescription() const override;
QString prettyStatusMessage() const override; QString prettyStatusMessage() const override;
@ -34,7 +45,9 @@ public:
private: private:
const QString m_hostname; const QString m_hostname;
const Actions m_actions;
}; };
Q_DECLARE_OPERATORS_FOR_FLAGS( SetHostNameJob::Actions )
#endif // SETHOSTNAMEJOB_CPP_H #endif // SETHOSTNAMEJOB_CPP_H

145
src/modules/users/Tests.cpp Normal file
View File

@ -0,0 +1,145 @@
/* === This file is part of Calamares - <https://github.com/calamares> ===
*
* Copyright 2020, 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 "SetHostNameJob.h"
// Implementation details
extern bool setFileHostname( const QString& );
extern bool writeFileEtcHosts( const QString& );
extern bool setSystemdHostname( const QString& );
#include "GlobalStorage.h"
#include "JobQueue.h"
#include "utils/CalamaresUtilsSystem.h"
#include "utils/Logger.h"
#include "utils/Yaml.h"
#include <QTemporaryDir>
#include <QtTest/QtTest>
class UsersTests : public QObject
{
Q_OBJECT
public:
UsersTests();
virtual ~UsersTests() {}
private Q_SLOTS:
void initTestCase();
void testEtcHostname();
void testEtcHosts();
void testHostnamed();
void cleanup();
private:
QTemporaryDir m_dir;
};
UsersTests::UsersTests()
: m_dir( QStringLiteral( "/tmp/calamares-usertest" ) )
{
}
void
UsersTests::initTestCase()
{
Logger::setupLogLevel( Logger::LOGDEBUG );
cDebug() << "Users test started.";
cDebug() << "Test dir" << m_dir.path();
// Ensure we have a system object, expect it to be a "bogus" one
CalamaresUtils::System* system = CalamaresUtils::System::instance();
QVERIFY( system );
QVERIFY( system->doChroot() );
// Ensure we have a system-wide GlobalStorage with /tmp as root
if ( !Calamares::JobQueue::instance() )
{
cDebug() << "Creating new JobQueue";
(void)new Calamares::JobQueue();
}
Calamares::GlobalStorage* gs
= Calamares::JobQueue::instance() ? Calamares::JobQueue::instance()->globalStorage() : nullptr;
QVERIFY( gs );
gs->insert( "rootMountPoint", m_dir.path() );
}
void
UsersTests::testEtcHostname()
{
cDebug() << "Test dir" << m_dir.path();
QVERIFY( QFile::exists( m_dir.path() ) );
QVERIFY( !QFile::exists( m_dir.filePath( "etc" ) ) );
// Doesn't create intermediate directories
QVERIFY( !setFileHostname( QStringLiteral( "tubophone.calamares.io" ) ) );
QVERIFY( CalamaresUtils::System::instance()->createTargetDirs( "/etc" ) );
QVERIFY( QFile::exists( m_dir.filePath( "etc" ) ) );
// Does write the file
QVERIFY( setFileHostname( QStringLiteral( "tubophone.calamares.io" ) ) );
QVERIFY( QFile::exists( m_dir.filePath( "etc/hostname" ) ) );
// 22 for the test string, above, and 1 for the newline
QCOMPARE( QFileInfo( m_dir.filePath( "etc/hostname" ) ).size(), 22 + 1 );
}
void
UsersTests::testEtcHosts()
{
// Assume previous tests did their work
QVERIFY( QFile::exists( m_dir.path() ) );
QVERIFY( QFile::exists( m_dir.filePath( "etc" ) ) );
QVERIFY( writeFileEtcHosts( QStringLiteral( "tubophone.calamares.io" ) ) );
QVERIFY( QFile::exists( m_dir.filePath( "etc/hosts" ) ) );
// The skeleton contains %1 which has the hostname substituted in, so we lose two,
// and the rest of the blabla is 150 (according to Python)
QCOMPARE( QFileInfo( m_dir.filePath( "etc/hosts" ) ).size(), 150 + 22 - 2 );
}
void
UsersTests::testHostnamed()
{
// Since the service might not be running (e.g. non-systemd systems,
// FreeBSD, docker, ..) we're not going to fail a test here.
// There's also the permissions problem to think of.
QEXPECT_FAIL( "", "Hostname changes are access-controlled", Continue );
QVERIFY( setSystemdHostname( "tubophone.calamares.io" ) );
}
void
UsersTests::cleanup()
{
if ( QTest::currentTestFailed() )
{
m_dir.setAutoRemove( false );
}
}
QTEST_GUILESS_MAIN( UsersTests )
#include "utils/moc-warnings.h"
#include "Tests.moc"

View File

@ -3,6 +3,7 @@
* Copyright 2014-2017, Teo Mrnjavac <teo@kde.org> * Copyright 2014-2017, Teo Mrnjavac <teo@kde.org>
* Copyright 2017-2018, Adriaan de Groot <groot@kde.org> * Copyright 2017-2018, Adriaan de Groot <groot@kde.org>
* Copyright 2019, Collabora Ltd <arnaud.ferraris@collabora.com> * Copyright 2019, Collabora Ltd <arnaud.ferraris@collabora.com>
* Copyright 2020, Gabriel Craciunescu <crazy@frugalware.org>
* *
* Portions from the Manjaro Installation Framework * Portions from the Manjaro Installation Framework
* by Roland Singer <roland@manjaro.org> * by Roland Singer <roland@manjaro.org>
@ -40,6 +41,7 @@
#include "utils/String.h" #include "utils/String.h"
#include <QBoxLayout> #include <QBoxLayout>
#include <QFile>
#include <QLabel> #include <QLabel>
#include <QLineEdit> #include <QLineEdit>
#include <QRegExp> #include <QRegExp>
@ -97,14 +99,12 @@ UsersPage::UsersPage( QWidget* parent )
connect( ui->textBoxUserVerifiedPassword, &QLineEdit::textChanged, this, &UsersPage::onPasswordTextChanged ); connect( ui->textBoxUserVerifiedPassword, &QLineEdit::textChanged, this, &UsersPage::onPasswordTextChanged );
connect( ui->textBoxRootPassword, &QLineEdit::textChanged, this, &UsersPage::onRootPasswordTextChanged ); connect( ui->textBoxRootPassword, &QLineEdit::textChanged, this, &UsersPage::onRootPasswordTextChanged );
connect( ui->textBoxVerifiedRootPassword, &QLineEdit::textChanged, this, &UsersPage::onRootPasswordTextChanged ); connect( ui->textBoxVerifiedRootPassword, &QLineEdit::textChanged, this, &UsersPage::onRootPasswordTextChanged );
connect( ui->checkBoxValidatePassword, &QCheckBox::stateChanged, this, [this]( int ) connect( ui->checkBoxValidatePassword, &QCheckBox::stateChanged, this, [this]( int ) {
{
onPasswordTextChanged( ui->textBoxUserPassword->text() ); onPasswordTextChanged( ui->textBoxUserPassword->text() );
onRootPasswordTextChanged( ui->textBoxRootPassword->text() ); onRootPasswordTextChanged( ui->textBoxRootPassword->text() );
checkReady( isReady() ); checkReady( isReady() );
} ); } );
connect( ui->checkBoxReusePassword, &QCheckBox::stateChanged, this, [this]( int checked ) connect( ui->checkBoxReusePassword, &QCheckBox::stateChanged, this, [this]( int checked ) {
{
ui->labelChooseRootPassword->setVisible( !checked ); ui->labelChooseRootPassword->setVisible( !checked );
ui->labelRootPassword->setVisible( !checked ); ui->labelRootPassword->setVisible( !checked );
ui->labelRootPasswordError->setVisible( !checked ); ui->labelRootPasswordError->setVisible( !checked );
@ -166,6 +166,37 @@ UsersPage::isReady()
return readyFields && m_readyRootPassword; return readyFields && m_readyRootPassword;
} }
QString
UsersPage::getHostname() const
{
return ui->textBoxHostname->text();
}
QString
UsersPage::getRootPassword() const
{
if ( m_writeRootPassword )
{
if ( ui->checkBoxReusePassword->isChecked() )
{
return ui->textBoxUserPassword->text();
}
else
{
return ui->textBoxRootPassword->text();
}
}
else
{
return QString();
}
}
QPair< QString, QString >
UsersPage::getUserPassword() const
{
return QPair< QString, QString >( ui->textBoxUsername->text(), ui->textBoxUserPassword->text() );
}
QList< Calamares::job_ptr > QList< Calamares::job_ptr >
UsersPage::createJobs( const QStringList& defaultGroupsList ) UsersPage::createJobs( const QStringList& defaultGroupsList )
@ -186,32 +217,10 @@ UsersPage::createJobs( const QStringList& defaultGroupsList )
defaultGroupsList ); defaultGroupsList );
list.append( Calamares::job_ptr( j ) ); list.append( Calamares::job_ptr( j ) );
j = new SetPasswordJob( ui->textBoxUsername->text(), ui->textBoxUserPassword->text() );
list.append( Calamares::job_ptr( j ) );
if ( m_writeRootPassword ) if ( m_writeRootPassword )
{ {
gs->insert( "reuseRootPassword", ui->checkBoxReusePassword->isChecked() ); gs->insert( "reuseRootPassword", ui->checkBoxReusePassword->isChecked() );
if ( ui->checkBoxReusePassword->isChecked() )
{
j = new SetPasswordJob( "root", ui->textBoxUserPassword->text() );
} }
else
{
j = new SetPasswordJob( "root", ui->textBoxRootPassword->text() );
}
list.append( Calamares::job_ptr( j ) );
}
else
{
j = new SetPasswordJob( "root",
"" ); //explicitly disable root password
list.append( Calamares::job_ptr( j ) );
}
j = new SetHostNameJob( ui->textBoxHostname->text() );
list.append( Calamares::job_ptr( j ) );
gs->insert( "hostname", ui->textBoxHostname->text() ); gs->insert( "hostname", ui->textBoxHostname->text() );
if ( ui->checkBoxAutoLogin->isChecked() ) if ( ui->checkBoxAutoLogin->isChecked() )
{ {
@ -269,6 +278,38 @@ UsersPage::onFullNameTextEdited( const QString& textRef )
checkReady( isReady() ); checkReady( isReady() );
} }
/** @brief Guess the machine's name
*
* If there is DMI data, use that; otherwise, just call the machine "-pc".
* Reads the DMI data just once.
*/
static QString
guessProductName()
{
static bool tried = false;
static QString dmiProduct;
if ( !tried )
{
// yes validateHostnameText() but these files can be a mess
QRegExp dmirx( "[^a-zA-Z0-9]", Qt::CaseInsensitive );
QFile dmiFile( QStringLiteral( "/sys/devices/virtual/dmi/id/product_name" ) );
if ( dmiFile.exists() && dmiFile.open( QIODevice::ReadOnly ) )
{
dmiProduct = QString::fromLocal8Bit( dmiFile.readAll().simplified().data() )
.toLower()
.replace( dmirx, " " )
.remove( ' ' );
}
if ( dmiProduct.isEmpty() )
{
dmiProduct = QStringLiteral( "-pc" );
}
tried = true;
}
return dmiProduct;
}
void void
UsersPage::fillSuggestions() UsersPage::fillSuggestions()
@ -303,7 +344,9 @@ UsersPage::fillSuggestions()
{ {
if ( !cleanParts.isEmpty() && !cleanParts.first().isEmpty() ) if ( !cleanParts.isEmpty() && !cleanParts.first().isEmpty() )
{ {
QString hostnameSuggestion = QString( "%1-pc" ).arg( cleanParts.first() ); QString hostnameSuggestion;
QString productName = guessProductName();
hostnameSuggestion = QString( "%1-%2" ).arg( cleanParts.first() ).arg( productName );
if ( HOSTNAME_RX.indexIn( hostnameSuggestion ) != -1 ) if ( HOSTNAME_RX.indexIn( hostnameSuggestion ) != -1 )
{ {
ui->textBoxHostname->setText( hostnameSuggestion ); ui->textBoxHostname->setText( hostnameSuggestion );
@ -524,8 +567,8 @@ UsersPage::addPasswordCheck( const QString& key, const QVariant& value )
{ {
if ( value.toBool() ) if ( value.toBool() )
{ {
m_passwordChecks.push_back( PasswordCheck( m_passwordChecks.push_back(
[]() { return QCoreApplication::translate( "PWQ", "Password is empty" ); }, PasswordCheck( []() { return QCoreApplication::translate( "PWQ", "Password is empty" ); },
[]( const QString& s ) { return !s.isEmpty(); }, []( const QString& s ) { return !s.isEmpty(); },
PasswordCheck::Weight( 1 ) ) ); PasswordCheck::Weight( 1 ) ) );
} }

View File

@ -63,6 +63,13 @@ public:
*/ */
void addPasswordCheck( const QString& key, const QVariant& value ); void addPasswordCheck( const QString& key, const QVariant& value );
///@brief Hostname as entered / auto-filled
QString getHostname() const;
///@brief Root password, depends on settings, may be empty
QString getRootPassword() const;
///@brief User name and password
QPair< QString, QString > getUserPassword() const;
protected slots: protected slots:
void onFullNameTextEdited( const QString& ); void onFullNameTextEdited( const QString& );
void fillSuggestions(); void fillSuggestions();

View File

@ -20,10 +20,12 @@
#include "UsersViewStep.h" #include "UsersViewStep.h"
#include "SetHostNameJob.h"
#include "SetPasswordJob.h"
#include "UsersPage.h" #include "UsersPage.h"
// #include "utils/CalamaresUtils.h"
#include "utils/Logger.h" #include "utils/Logger.h"
#include "utils/NamedEnum.h"
#include "utils/Variant.h" #include "utils/Variant.h"
#include "GlobalStorage.h" #include "GlobalStorage.h"
@ -31,9 +33,28 @@
CALAMARES_PLUGIN_FACTORY_DEFINITION( UsersViewStepFactory, registerPlugin< UsersViewStep >(); ) CALAMARES_PLUGIN_FACTORY_DEFINITION( UsersViewStepFactory, registerPlugin< UsersViewStep >(); )
static const NamedEnumTable< SetHostNameJob::Action >&
hostnameActions()
{
using Action = SetHostNameJob::Action;
// *INDENT-OFF*
// clang-format off
static const NamedEnumTable< Action > names {
{ QStringLiteral( "none" ), Action::None },
{ QStringLiteral( "etcfile" ), Action::EtcHostname },
{ QStringLiteral( "hostnamed" ), Action::SystemdHostname }
};
// clang-format on
// *INDENT-ON*
return names;
}
UsersViewStep::UsersViewStep( QObject* parent ) UsersViewStep::UsersViewStep( QObject* parent )
: Calamares::ViewStep( parent ) : Calamares::ViewStep( parent )
, m_widget( new UsersPage() ) , m_widget( new UsersPage() )
, m_actions( SetHostNameJob::Action::None )
{ {
emit nextStatusChanged( true ); emit nextStatusChanged( true );
connect( m_widget, &UsersPage::checkReady, this, &UsersViewStep::nextStatusChanged ); connect( m_widget, &UsersPage::checkReady, this, &UsersViewStep::nextStatusChanged );
@ -109,14 +130,27 @@ void
UsersViewStep::onLeave() UsersViewStep::onLeave()
{ {
m_jobs.clear(); m_jobs.clear();
m_jobs.append( m_widget->createJobs( m_defaultGroups ) ); m_jobs.append( m_widget->createJobs( m_defaultGroups ) );
Calamares::Job* j;
auto userPW = m_widget->getUserPassword();
j = new SetPasswordJob( userPW.first, userPW.second );
m_jobs.append( Calamares::job_ptr( j ) );
j = new SetPasswordJob( "root", m_widget->getRootPassword() );
m_jobs.append( Calamares::job_ptr( j ) );
j = new SetHostNameJob( m_widget->getHostname(), m_actions );
m_jobs.append( Calamares::job_ptr( j ) );
} }
void void
UsersViewStep::setConfigurationMap( const QVariantMap& configurationMap ) UsersViewStep::setConfigurationMap( const QVariantMap& configurationMap )
{ {
using CalamaresUtils::getBool;
if ( configurationMap.contains( "defaultGroups" ) if ( configurationMap.contains( "defaultGroups" )
&& configurationMap.value( "defaultGroups" ).type() == QVariant::List ) && configurationMap.value( "defaultGroups" ).type() == QVariant::List )
{ {
@ -142,25 +176,12 @@ UsersViewStep::setConfigurationMap( const QVariantMap& configurationMap )
configurationMap.value( "sudoersGroup" ).toString() ); configurationMap.value( "sudoersGroup" ).toString() );
} }
if ( configurationMap.contains( "setRootPassword" ) bool setRootPassword = getBool( configurationMap, "setRootPassword", true );
&& configurationMap.value( "setRootPassword" ).type() == QVariant::Bool ) Calamares::JobQueue::instance()->globalStorage()->insert( "setRootPassword", setRootPassword );
{
Calamares::JobQueue::instance()->globalStorage()->insert(
"setRootPassword", configurationMap.value( "setRootPassword" ).toBool() );
m_widget->setWriteRootPassword( configurationMap.value( "setRootPassword" ).toBool() );
}
if ( configurationMap.contains( "doAutologin" ) m_widget->setWriteRootPassword( setRootPassword );
&& configurationMap.value( "doAutologin" ).type() == QVariant::Bool ) m_widget->setAutologinDefault( getBool( configurationMap, "doAutologin", false ) );
{ m_widget->setReusePasswordDefault( getBool( configurationMap, "doReusePassword", false ) );
m_widget->setAutologinDefault( configurationMap.value( "doAutologin" ).toBool() );
}
if ( configurationMap.contains( "doReusePassword" )
&& configurationMap.value( "doReusePassword" ).type() == QVariant::Bool )
{
m_widget->setReusePasswordDefault( configurationMap.value( "doReusePassword" ).toBool() );
}
if ( configurationMap.contains( "passwordRequirements" ) if ( configurationMap.contains( "passwordRequirements" )
&& configurationMap.value( "passwordRequirements" ).type() == QVariant::Map ) && configurationMap.value( "passwordRequirements" ).type() == QVariant::Map )
@ -173,8 +194,8 @@ UsersViewStep::setConfigurationMap( const QVariantMap& configurationMap )
} }
} }
m_widget->setPasswordCheckboxVisible( CalamaresUtils::getBool( configurationMap, "allowWeakPasswords", false ) ); m_widget->setPasswordCheckboxVisible( getBool( configurationMap, "allowWeakPasswords", false ) );
m_widget->setValidatePasswordDefault( !CalamaresUtils::getBool( configurationMap, "allowWeakPasswordsDefault", false) ); m_widget->setValidatePasswordDefault( !getBool( configurationMap, "allowWeakPasswordsDefault", false ) );
QString shell( QLatin1String( "/bin/bash" ) ); // as if it's not set at all QString shell( QLatin1String( "/bin/bash" ) ); // as if it's not set at all
if ( configurationMap.contains( "userShell" ) ) if ( configurationMap.contains( "userShell" ) )
@ -184,4 +205,21 @@ UsersViewStep::setConfigurationMap( const QVariantMap& configurationMap )
// Now it might be explicitly set to empty, which is ok // Now it might be explicitly set to empty, which is ok
Calamares::JobQueue::instance()->globalStorage()->insert( "userShell", shell ); Calamares::JobQueue::instance()->globalStorage()->insert( "userShell", shell );
using Action = SetHostNameJob::Action;
QString hostnameActionString = CalamaresUtils::getString( configurationMap, "setHostname" );
if ( hostnameActionString.isEmpty() )
{
hostnameActionString = QStringLiteral( "EtcFile" );
}
bool ok = false;
auto hostnameAction = hostnameActions().find( hostnameActionString, ok );
if ( !ok )
{
hostnameAction = Action::EtcHostname;
}
Action hostsfileAction = getBool( configurationMap, "writeHostsFile", true ) ? Action::WriteEtcHosts : Action::None;
m_actions = hostsfileAction | hostnameAction;
} }

View File

@ -20,13 +20,13 @@
#ifndef USERSPAGEPLUGIN_H #ifndef USERSPAGEPLUGIN_H
#define USERSPAGEPLUGIN_H #define USERSPAGEPLUGIN_H
#include "SetHostNameJob.h"
#include "DllMacro.h"
#include "utils/PluginFactory.h"
#include "viewpages/ViewStep.h"
#include <QObject> #include <QObject>
#include <utils/PluginFactory.h>
#include <viewpages/ViewStep.h>
#include <PluginDllMacro.h>
#include <QVariant> #include <QVariant>
class UsersPage; class UsersPage;
@ -61,6 +61,7 @@ private:
QList< Calamares::job_ptr > m_jobs; QList< Calamares::job_ptr > m_jobs;
QStringList m_defaultGroups; QStringList m_defaultGroups;
SetHostNameJob::Actions m_actions;
}; };
CALAMARES_PLUGIN_FACTORY_DECLARATION( UsersViewStepFactory ) CALAMARES_PLUGIN_FACTORY_DECLARATION( UsersViewStepFactory )

View File

@ -122,3 +122,17 @@ allowWeakPasswordsDefault: false
# - set, non-empty, use that path as shell. No validation is done # - set, non-empty, use that path as shell. No validation is done
# that the shell actually exists or is executable. # that the shell actually exists or is executable.
# userShell: /bin/bash # userShell: /bin/bash
# Hostname setting
#
# The user can enter a hostname; this is configured into the system
# in some way; pick one of:
# - *None*, to not set the hostname at all
# - *EtcFile*, to write to `/etc/hostname` directly
# - *Hostnamed*, to use systemd hostnamed(1) over DBus
# The default is *EtcFile*.
setHostname: EtcFile
# Should /etc/hosts be written with a hostname for this machine
# (also adds localhost and some ipv6 standard entries).
writeHostsFile: true

View File

@ -26,7 +26,7 @@
#include <utils/PluginFactory.h> #include <utils/PluginFactory.h>
#include <viewpages/ViewStep.h> #include <viewpages/ViewStep.h>
#include <PluginDllMacro.h> #include <DllMacro.h>
#include <QVariantMap> #include <QVariantMap>

View File

@ -25,7 +25,7 @@
#include <utils/PluginFactory.h> #include <utils/PluginFactory.h>
#include <viewpages/ViewStep.h> #include <viewpages/ViewStep.h>
#include <PluginDllMacro.h> #include <DllMacro.h>
#include <QVariantMap> #include <QVariantMap>

View File

@ -25,7 +25,7 @@
#include "utils/PluginFactory.h" #include "utils/PluginFactory.h"
#include "viewpages/ViewStep.h" #include "viewpages/ViewStep.h"
#include <PluginDllMacro.h> #include <DllMacro.h>
#include <QObject> #include <QObject>
#include <QVariantMap> #include <QVariantMap>