From dc91255ff56b7d5905ef65d537d766a2799c0901 Mon Sep 17 00:00:00 2001 From: Adriaan de Groot Date: Tue, 20 Feb 2024 14:11:23 +0100 Subject: [PATCH 01/13] Changes: document process changes FIXES #2212 --- CHANGES-3.3 | 17 +++++++++++++++++ src/modules/README.md | 5 ++++- 2 files changed, 21 insertions(+), 1 deletion(-) diff --git a/CHANGES-3.3 b/CHANGES-3.3 index 5161fbf1b..19dc4d6de 100644 --- a/CHANGES-3.3 +++ b/CHANGES-3.3 @@ -9,11 +9,28 @@ the history of the 3.2 series (2018-05 - 2022-08). # 3.3.4 (unreleased) +In this release, process jobmodules -- a particular kind of module +recognizable by `type: job` and `interface: process` in the descriptor +file -- undergo a large change to resemble *shellprocess* more. + +Users of process jobmodules are encouraged to double-check the Functionality +of those modules in this release. + This release contains contributions from (alphabetically by first name): + - Adriaan de Groot ## Core ## + - Process jobs (a job type provided by Calamares core) now share more + code with *contextualprocess* and *shellprocess* jobs. The execution + mechanism is the same, and always invokes the shell, whether the command + runs in the host or in the target system. It is no longer necessary to + add `/bin/sh` in the *command* key -- this is always present. ## Modules ## + - *contextualprocess* and *shellprocess* can now set environment variables + as part of the configuration. See *shellprocess* documentation for details. + This is optional, and does not do anything that could not already be done + by putting `export VAR=value ;` in front of the command before. # 3.3.3 (2024-02-24) diff --git a/src/modules/README.md b/src/modules/README.md index 108b3200d..bb2c9cc21 100644 --- a/src/modules/README.md +++ b/src/modules/README.md @@ -508,7 +508,10 @@ while the module type must be *job* or *jobmodule*. The module-descriptor key *command* should have a string as value, which is passed to the shell -- remember to quote it properly. It is generally recommended to use a *shellprocess* job module instead (less configuration, -easier to have multiple instances). +easier to have multiple instances). There is no configuration outside +of the module-descriptor. + +Optional keys are *timeout* and *chroot*. `CMakeLists.txt` is *not* used for process jobmodules. From b795fd82bbf2ae8faa3adbef2ca90bd664bc6eec Mon Sep 17 00:00:00 2001 From: Adriaan de Groot Date: Tue, 20 Feb 2024 14:21:25 +0100 Subject: [PATCH 02/13] [contextualprocess] Improve config docs --- src/modules/contextualprocess/contextualprocess.conf | 9 ++++----- 1 file changed, 4 insertions(+), 5 deletions(-) diff --git a/src/modules/contextualprocess/contextualprocess.conf b/src/modules/contextualprocess/contextualprocess.conf index b86fd922c..ba8a8bf1d 100644 --- a/src/modules/contextualprocess/contextualprocess.conf +++ b/src/modules/contextualprocess/contextualprocess.conf @@ -7,14 +7,13 @@ # When a given global value (string) equals a given value, then # the associated command is executed. # -# The special top-level keys *dontChroot* and *timeout* have -# meaning just like in shellprocess.conf. They are excluded from -# the comparison with global variables. -# # Configuration consists of keys for global variable names (except # *dontChroot* and *timeout*), and the sub-keys are strings to compare # to the variable's value. If the variable has that particular value, the -# corresponding value (script) is executed. +# corresponding value (script) is executed. The top-level keys *dontChroot* +# and *timeout* are not global variable names. They have +# meaning just like in shellprocess.conf, that is they +# determine **where** the command runs and how long it has. # # The variable **may** contain dots, in which case the dot is used # to select into maps inside global storage, e.g. From 15c514326cc57a383346b876107f26b198ef4351 Mon Sep 17 00:00:00 2001 From: Adriaan de Groot Date: Tue, 20 Feb 2024 14:26:27 +0100 Subject: [PATCH 03/13] [shellprocess] Improve config docs --- src/modules/shellprocess/shellprocess.conf | 27 ++++++++++++++++++---- 1 file changed, 22 insertions(+), 5 deletions(-) diff --git a/src/modules/shellprocess/shellprocess.conf b/src/modules/shellprocess/shellprocess.conf index 9ff83221e..881181a0d 100644 --- a/src/modules/shellprocess/shellprocess.conf +++ b/src/modules/shellprocess/shellprocess.conf @@ -14,6 +14,10 @@ # - `USER` is replaced by the username, set on the user page. # # Variables are written as `${var}`, e.g. `${ROOT}`. +# Write `$$` to get a shell-escaped `\$` in the shell command. +# It is not possible to get an un-escaped `$` in the shell command +# (either the command will fail because of undefined variables, or +# you get a shell-escaped `\$`). # # The (global) timeout for the command list can be set with # the *timeout* key. The value is a time in seconds, default @@ -35,20 +39,33 @@ # # The value of *script* may be: # - a single string; this is one command that is executed. -# - a single object (this is not useful). +# - a single object (see below). # - a list of items; these are executed one at a time, by # separate shells (/bin/sh -c is invoked for each command). # Each list item may be: # - a single string; this is one command that is executed. # - a single object, specifying a key *command* and (optionally) # a key *timeout* to set the timeout for this specific -# command differently from the global setting. +# command differently from the global setting. An optional +# key *environment* is a list of strings to put into the +# environment of the command. # -# Using a single object is not useful because the same effect can -# be obtained with a single string and a global timeout, but when -# there are multiple commands to execute, one of them might have +# Using a single object is not generally useful because the same effect +# can be obtained with a single string and a global timeout, except +# when the command needs environment-settings. When there are +# multiple commands to execute, one of them might have # a different timeout than the others. # +# The environment strings should all be "KEY='some value'" strings, +# as if they can be typed into the shell. Quoting the environment +# strings with "" in YAML is recommended. Adding the '' quotes ensures +# that the value will not be interpreted by the shell. Writing +# environment strings is the same as placing `export KEY='some value' ;` +# in front of the *command*. +# +# Calamares variable expansion is **also** done on the environment strings. +# Write `$$` to get a literal `$` in the shell command. +# # To change the description of the job, set the *name* entries in *i18n*. --- # Set to true to run in host, rather than target system From 0d9d2ac59a03b1cacbd9aa65702372c0c8e54dc3 Mon Sep 17 00:00:00 2001 From: Adriaan de Groot Date: Tue, 20 Feb 2024 16:51:23 +0100 Subject: [PATCH 04/13] [libcalamares] Extend CommandLine - rename fields so they are meaningful (this is a leftover from it inheriting std::pair) - add environment list member - add constructor that consumes a QVariantMap --- src/libcalamares/utils/CommandList.cpp | 21 +++++++++++++-- src/libcalamares/utils/CommandList.h | 36 +++++++++++++++++++------- 2 files changed, 45 insertions(+), 12 deletions(-) diff --git a/src/libcalamares/utils/CommandList.cpp b/src/libcalamares/utils/CommandList.cpp index 96d5ea5d1..800dcab78 100644 --- a/src/libcalamares/utils/CommandList.cpp +++ b/src/libcalamares/utils/CommandList.cpp @@ -94,12 +94,29 @@ get_gs_expander( System::RunLocation location ) return expander; } +CommandLine::CommandLine( const QVariantMap& m ) +{ + const QString command = Calamares::getString( m, "command" ); + const qint64 timeout = Calamares::getInteger( m, "timeout", -1 ); + if ( !command.isEmpty() ) + { + m_command = command; + m_timeout = timeout >= 0 ? std::chrono::seconds( timeout ) : CommandLine::TimeoutNotSet(); + m_environment = Calamares::getStringList( m, "environment" ); + } + else + { + cWarning() << "Bad CommandLine element" << m; + // this CommandLine is invalid + } +} + CommandLine CommandLine::expand( KMacroExpanderBase& expander ) const { - QString c = first; + QString c = m_command; expander.expandMacrosShellQuote( c ); - return { c, second }; + return { c, m_environment, m_timeout }; } Calamares::CommandLine diff --git a/src/libcalamares/utils/CommandList.h b/src/libcalamares/utils/CommandList.h index 9969dc165..3db0102d6 100644 --- a/src/libcalamares/utils/CommandList.h +++ b/src/libcalamares/utils/CommandList.h @@ -28,30 +28,43 @@ namespace Calamares * Each command can have an associated timeout in seconds. The timeout * defaults to 10 seconds. Provide some convenience naming and construction. */ -struct CommandLine +class CommandLine { +public: static inline constexpr std::chrono::seconds TimeoutNotSet() { return std::chrono::seconds( -1 ); } /// An invalid command line CommandLine() = default; CommandLine( const QString& s ) - : first( s ) - , second( TimeoutNotSet() ) + : m_command( s ) { } CommandLine( const QString& s, std::chrono::seconds t ) - : first( s ) - , second( t ) + : m_command( s ) + , m_timeout( t ) { } - QString command() const { return first; } + CommandLine( const QString& s, const QStringList& env, std::chrono::seconds t ) + : m_command( s ) + , m_environment( env ) + , m_timeout( t ) + { + } - std::chrono::seconds timeout() const { return second; } + /** @brief Constructs a CommandLine from a map with keys + * + * Relevant keys are *command*, *environment* and *timeout*. + */ + CommandLine( const QVariantMap& m ); - bool isValid() const { return !first.isEmpty(); } + QString command() const { return m_command; } + [[nodiscard]] QStringList environment() const { return m_environment; } + std::chrono::seconds timeout() const { return m_timeout; } + + bool isValid() const { return !m_command.isEmpty(); } /** @brief Returns a copy of this one command, with variables expanded * @@ -60,6 +73,7 @@ struct CommandLine * instance, which handles the ROOT and USER variables. */ DLLEXPORT CommandLine expand( KMacroExpanderBase& expander ) const; + /** @brief As above, with a default macro-expander. * * The default macro-expander assumes RunInHost (e.g. ROOT will @@ -68,8 +82,9 @@ struct CommandLine DLLEXPORT CommandLine expand() const; private: - QString first; - std::chrono::seconds second = std::chrono::seconds( -1 ); + QString m_command; + QStringList m_environment; + std::chrono::seconds m_timeout = TimeoutNotSet(); }; /** @brief Abbreviation, used internally. */ @@ -109,6 +124,7 @@ public: * @see CommandLine::expand() for details. */ CommandList expand( KMacroExpanderBase& expander ) const; + /** @brief As above, with a default macro-expander. * * Each command-line in the list is expanded with that default macro-expander. From 4aa2c4988c1d90c8919d59c754d59db4ebe8bf12 Mon Sep 17 00:00:00 2001 From: Adriaan de Groot Date: Tue, 20 Feb 2024 16:53:55 +0100 Subject: [PATCH 05/13] [libcalamares] Replace factory method with constructor --- src/libcalamares/utils/CommandList.cpp | 18 ++---------------- 1 file changed, 2 insertions(+), 16 deletions(-) diff --git a/src/libcalamares/utils/CommandList.cpp b/src/libcalamares/utils/CommandList.cpp index 800dcab78..d9ceca10b 100644 --- a/src/libcalamares/utils/CommandList.cpp +++ b/src/libcalamares/utils/CommandList.cpp @@ -25,20 +25,6 @@ namespace Calamares { -static CommandLine -get_variant_object( const QVariantMap& m ) -{ - QString command = Calamares::getString( m, "command" ); - qint64 timeout = Calamares::getInteger( m, "timeout", -1 ); - - if ( !command.isEmpty() ) - { - return CommandLine( command, timeout >= 0 ? std::chrono::seconds( timeout ) : CommandLine::TimeoutNotSet() ); - } - cWarning() << "Bad CommandLine element" << m; - return CommandLine(); -} - static CommandList_t get_variant_stringlist( const QVariantList& l ) { @@ -52,7 +38,7 @@ get_variant_stringlist( const QVariantList& l ) } else if ( Calamares::typeOf( v ) == Calamares::MapVariantType ) { - auto command( get_variant_object( v.toMap() ) ); + CommandLine command( v.toMap() ); if ( command.isValid() ) { retl.append( command ); @@ -153,7 +139,7 @@ CommandList::CommandList::CommandList( const QVariant& v, bool doChroot, std::ch } else if ( Calamares::typeOf( v ) == Calamares::MapVariantType ) { - auto c( get_variant_object( v.toMap() ) ); + CommandLine c( v.toMap() ); if ( c.isValid() ) { append( c ); From ae3e60902478405923277be75755fe2f63fed678 Mon Sep 17 00:00:00 2001 From: Adriaan de Groot Date: Tue, 20 Feb 2024 22:21:01 +0100 Subject: [PATCH 06/13] [libcalamares] Get default timeout from CommandList --- src/libcalamares/utils/CommandList.h | 1 + 1 file changed, 1 insertion(+) diff --git a/src/libcalamares/utils/CommandList.h b/src/libcalamares/utils/CommandList.h index 3db0102d6..17a251a7c 100644 --- a/src/libcalamares/utils/CommandList.h +++ b/src/libcalamares/utils/CommandList.h @@ -106,6 +106,7 @@ public: CommandList( const QVariant& v, bool doChroot = true, std::chrono::seconds timeout = std::chrono::seconds( 10 ) ); bool doChroot() const { return m_doChroot; } + std::chrono::seconds defaultTimeout() const { return m_timeout; } Calamares::JobResult run(); From 6770f781e387c747c002ec11981f4011ce8bd59f Mon Sep 17 00:00:00 2001 From: Adriaan de Groot Date: Tue, 20 Feb 2024 22:41:17 +0100 Subject: [PATCH 07/13] [libcalamares] Tests for new CommandLine constructors --- src/libcalamares/utils/Tests.cpp | 75 ++++++++++++++++++++++++++++++++ 1 file changed, 75 insertions(+) diff --git a/src/libcalamares/utils/Tests.cpp b/src/libcalamares/utils/Tests.cpp index ae0be4615..e9414bd86 100644 --- a/src/libcalamares/utils/Tests.cpp +++ b/src/libcalamares/utils/Tests.cpp @@ -52,6 +52,8 @@ private Q_SLOTS: void testCommands(); void testCommandExpansion_data(); void testCommandExpansion(); // See also shellprocess tests + void testCommandConstructors(); + void testCommandConstructorsYAML(); /** @section Test that all the UMask objects work correctly. */ void testUmask(); @@ -300,6 +302,79 @@ LibCalamaresTests::testCommandExpansion() QCOMPARE( e.command(), expected ); } +void +LibCalamaresTests::testCommandConstructors() +{ + const QString command( "do this" ); + Calamares::CommandLine c0( command ); + + QCOMPARE( c0.command(), command ); + QCOMPARE( c0.timeout(), Calamares::CommandLine::TimeoutNotSet() ); + QVERIFY( c0.environment().isEmpty() ); + + const QStringList env { "-la", "/tmp" }; + Calamares::CommandLine c1( command, env, Calamares::CommandLine::TimeoutNotSet() ); + + QCOMPARE( c1.command(), command ); + QCOMPARE( c1.timeout(), Calamares::CommandLine::TimeoutNotSet() ); + QVERIFY( !c1.environment().isEmpty() ); + QCOMPARE( c1.environment().count(), 2 ); + QCOMPARE( c1.environment(), env ); +} + +void +LibCalamaresTests::testCommandConstructorsYAML() +{ + QTemporaryFile f; + QVERIFY( f.open() ); + f.write( R"(--- +commands: + - one-string-command + - command: only-command + - command: with-timeout + timeout: 12 + - command: all-three + timeout: 20 + environment: + - PATH=/USER + - DISPLAY=:0 + )" ); + f.close(); + bool ok = false; + QVariantMap m = Calamares::YAML::load( f.fileName(), &ok ); + + QVERIFY( ok ); + QCOMPARE( m.count(), 1 ); + QCOMPARE( m[ "commands" ].toList().count(), 4 ); + + { + // Take care! The second parameter is a bool, so "3" here means "true" + Calamares::CommandList cmds( m[ "commands" ], 3 ); + QCOMPARE( cmds.defaultTimeout(), std::chrono::seconds( 10 ) ); + // But the 4 commands are there anyway + QCOMPARE( cmds.count(), 4 ); + QCOMPARE( cmds.at( 0 ).command(), QString( "one-string-command" ) ); + QCOMPARE( cmds.at( 0 ).environment(), QStringList() ); + QCOMPARE( cmds.at( 0 ).timeout(), Calamares::CommandLine::TimeoutNotSet() ); + QCOMPARE( cmds.at( 1 ).command(), QString( "only-command" ) ); + QCOMPARE( cmds.at( 2 ).command(), QString( "with-timeout" ) ); + QCOMPARE( cmds.at( 2 ).environment(), QStringList() ); + QCOMPARE( cmds.at( 2 ).timeout(), std::chrono::seconds( 12 ) ); + + QStringList expectedEnvironment = { "PATH=/USER", "DISPLAY=:0" }; + QCOMPARE( cmds.at( 3 ).command(), QString( "all-three" ) ); + QCOMPARE( cmds.at( 3 ).environment(), expectedEnvironment ); + QCOMPARE( cmds.at( 3 ).timeout(), std::chrono::seconds( 20 ) ); + } + + { + Calamares::CommandList cmds( m[ "commands" ], true, std::chrono::seconds( 3 ) ); + QCOMPARE( cmds.defaultTimeout(), std::chrono::seconds( 3 ) ); + QCOMPARE( cmds.at( 0 ).timeout(), Calamares::CommandLine::TimeoutNotSet() ); + QCOMPARE( cmds.at( 2 ).timeout(), std::chrono::seconds( 12 ) ); + } +} + void LibCalamaresTests::testUmask() { From d5555eba3241e6905fdfa3c20849d02d4644d171 Mon Sep 17 00:00:00 2001 From: Adriaan de Groot Date: Sat, 24 Feb 2024 18:38:06 +0100 Subject: [PATCH 08/13] [libcalamares] Add environment-setting to CommandLine This class is shared -- used by shellprocess and contextualprocess -- to collect the command settings for a single shell invocation. --- src/libcalamares/utils/CommandList.cpp | 18 +++++++++++++++++- 1 file changed, 17 insertions(+), 1 deletion(-) diff --git a/src/libcalamares/utils/CommandList.cpp b/src/libcalamares/utils/CommandList.cpp index d9ceca10b..35295fedc 100644 --- a/src/libcalamares/utils/CommandList.cpp +++ b/src/libcalamares/utils/CommandList.cpp @@ -100,8 +100,14 @@ CommandLine::CommandLine( const QVariantMap& m ) CommandLine CommandLine::expand( KMacroExpanderBase& expander ) const { + // Calamares variable expansion in the command QString c = m_command; expander.expandMacrosShellQuote( c ); + + // .. and expand in each environment key=value string. + QStringList e = m_environment; + std::for_each(e.begin(), e.end(), [&expander](QString & s) { expander.expandMacrosShellQuote(s);}); + return { c, m_environment, m_timeout }; } @@ -181,8 +187,18 @@ CommandList::run() processed_cmd.remove( 0, 1 ); // Drop the - } + const QString environmentSetting = []( const QStringList& l ) -> QString + { + if ( l.isEmpty() ) + { + return {}; + } + + return QStringLiteral( "export " ) + l.join( " " ) + QStringLiteral( " ; " ); + }( i->environment() ); + QStringList shell_cmd { "/bin/sh", "-c" }; - shell_cmd << processed_cmd; + shell_cmd << ( environmentSetting + processed_cmd ); std::chrono::seconds timeout = i->timeout() >= std::chrono::seconds::zero() ? i->timeout() : m_timeout; ProcessResult r = System::runCommand( location, shell_cmd, QString(), QString(), timeout ); From 0ced01ddc36e375d2d7ba7f348481a1b1572cdfa Mon Sep 17 00:00:00 2001 From: Adriaan de Groot Date: Sat, 24 Feb 2024 18:52:56 +0100 Subject: [PATCH 09/13] [libcalamares] Test environment-setting for CommandLine --- src/libcalamares/utils/Tests.cpp | 13 +++++++++++++ 1 file changed, 13 insertions(+) diff --git a/src/libcalamares/utils/Tests.cpp b/src/libcalamares/utils/Tests.cpp index e9414bd86..c58d44cce 100644 --- a/src/libcalamares/utils/Tests.cpp +++ b/src/libcalamares/utils/Tests.cpp @@ -54,6 +54,7 @@ private Q_SLOTS: void testCommandExpansion(); // See also shellprocess tests void testCommandConstructors(); void testCommandConstructorsYAML(); + void testCommandRunning(); /** @section Test that all the UMask objects work correctly. */ void testUmask(); @@ -375,6 +376,18 @@ commands: } } +void LibCalamaresTests::testCommandRunning() +{ + const QString echoCommand = QStringLiteral("echo \"$calamares_test_variable\""); + + Calamares::CommandList l(false); // no chroot + Calamares::CommandLine c(echoCommand, {}, std::chrono::seconds(2)); + l.push_back(c); + + const auto r = l.run(); + const auto output = r.readAll(); +} + void LibCalamaresTests::testUmask() { From 4329b824d47db3e8a4647e1400e2de7b13cd20dc Mon Sep 17 00:00:00 2001 From: Adriaan de Groot Date: Sat, 24 Feb 2024 23:18:35 +0100 Subject: [PATCH 10/13] [libcalamares] Test command-expansion with environment --- src/libcalamares/utils/Tests.cpp | 60 ++++++++++++++++++++++++++++---- 1 file changed, 53 insertions(+), 7 deletions(-) diff --git a/src/libcalamares/utils/Tests.cpp b/src/libcalamares/utils/Tests.cpp index c58d44cce..2e1b5655c 100644 --- a/src/libcalamares/utils/Tests.cpp +++ b/src/libcalamares/utils/Tests.cpp @@ -12,6 +12,7 @@ #include "CommandList.h" #include "Entropy.h" #include "Logger.h" +#include "Permissions.h" #include "RAII.h" #include "Runner.h" #include "String.h" @@ -376,16 +377,61 @@ commands: } } -void LibCalamaresTests::testCommandRunning() +void +LibCalamaresTests::testCommandRunning() { - const QString echoCommand = QStringLiteral("echo \"$calamares_test_variable\""); - Calamares::CommandList l(false); // no chroot - Calamares::CommandLine c(echoCommand, {}, std::chrono::seconds(2)); - l.push_back(c); + QTemporaryDir tempRoot( QDir::tempPath() + QStringLiteral( "/test-job-XXXXXX" ) ); + tempRoot.setAutoRemove( false ); - const auto r = l.run(); - const auto output = r.readAll(); + const QString testExecutable = tempRoot.filePath( "example.sh" ); + const QString testFile = tempRoot.filePath( "example.txt" ); + + { + QFile f( testExecutable ); + QVERIFY( f.open( QIODevice::WriteOnly ) ); + f.write( "#! /bin/sh\necho \"$calamares_test_variable\"\n" ); + f.close(); + Calamares::Permissions::apply( testExecutable, 0755 ); + } + + const QString echoCommand = testExecutable + QStringLiteral( " > " ) + testFile; + + // Without an environment, the variable echoed in the example + // executable is empty, and we write a single newline to stdout, + // which is redirected to testFile. + { + Calamares::CommandList l( false ); // no chroot + Calamares::CommandLine c( echoCommand, {}, std::chrono::seconds( 2 ) ); + l.push_back( c ); + + const auto r = l.run(); + QVERIFY( bool( r ) ); + + QCOMPARE( QFileInfo( testFile ).size(), 1 ); // single newline + } + + // With an environment, echoes the value of the variable and a newline + { + const QString world = QStringLiteral( "Hello world" ); + Calamares::CommandList l( false ); // no chroot + Calamares::CommandLine c( + echoCommand, + { QStringLiteral( "calamares_test_variable=" ) + QChar( '"' ) + world + QChar( '"' ) }, + std::chrono::seconds( 2 ) ); + l.push_back( c ); + + const auto r = l.run(); + QVERIFY( bool( r ) ); + + QCOMPARE( QFileInfo( testFile ).size(), world.length() + 1 ); // plus newline + QFile f( testFile ); + QVERIFY( f.open( QIODevice::ReadOnly ) ); + QCOMPARE( f.readAll(), world + QChar( '\n' ) ); + } + + + tempRoot.setAutoRemove( true ); } void From 2697c35fc7509144bf50a9937149cc25eb8edb65 Mon Sep 17 00:00:00 2001 From: Adriaan de Groot Date: Sun, 25 Feb 2024 22:57:35 +0100 Subject: [PATCH 11/13] [libcalamares] Run ProcessJobs with CommandLine This makes the behavior of process jobs and shellcommands the same. --- src/libcalamares/ProcessJob.cpp | 22 ++++------------------ 1 file changed, 4 insertions(+), 18 deletions(-) diff --git a/src/libcalamares/ProcessJob.cpp b/src/libcalamares/ProcessJob.cpp index 70506f5ba..d3d76afb8 100644 --- a/src/libcalamares/ProcessJob.cpp +++ b/src/libcalamares/ProcessJob.cpp @@ -10,8 +10,8 @@ #include "ProcessJob.h" +#include "utils/CommandList.h" #include "utils/Logger.h" -#include "utils/System.h" #include @@ -57,23 +57,9 @@ ProcessJob::prettyStatusMessage() const JobResult ProcessJob::exec() { - using Calamares::System; - - if ( m_runInChroot ) - { - return Calamares::System::instance() - ->targetEnvCommand( { m_command }, m_workingPath, QString(), m_timeoutSec ) - .explainProcess( m_command, m_timeoutSec ); - } - else - { - return System::runCommand( System::RunLocation::RunInHost, - { "/bin/sh", "-c", m_command }, - m_workingPath, - QString(), - m_timeoutSec ) - .explainProcess( m_command, m_timeoutSec ); - } + Calamares::CommandList l( m_runInChroot, m_timeoutSec ); + l.push_back( Calamares::CommandLine { m_command } ); + return l.run(); } } // namespace Calamares From 0f2bceb72f37eff9cdd90037f288cc15aa84fd22 Mon Sep 17 00:00:00 2001 From: Adriaan de Groot Date: Mon, 26 Feb 2024 21:58:40 +0100 Subject: [PATCH 12/13] Docs: process jobmodules now expand variables --- src/modules/README.md | 9 ++++++--- 1 file changed, 6 insertions(+), 3 deletions(-) diff --git a/src/modules/README.md b/src/modules/README.md index bb2c9cc21..86aa4d726 100644 --- a/src/modules/README.md +++ b/src/modules/README.md @@ -497,7 +497,8 @@ LC_ALL and LANG to "C" for the called command. ## Process modules -Use of this kind of module is **not** recommended. +Use of this kind of module is **not** recommended. Use *shellprocess* +instead, which is more configurable. > Type: jobmodule > Interface: process @@ -506,10 +507,12 @@ A process jobmodule runs a (single) command. The interface is *process*, while the module type must be *job* or *jobmodule*. The module-descriptor key *command* should have a string as value, which is -passed to the shell -- remember to quote it properly. It is generally +passed to the shell -- remember to quote it properly in YAML. It is generally recommended to use a *shellprocess* job module instead (less configuration, easier to have multiple instances). There is no configuration outside -of the module-descriptor. +of the module-descriptor. The *command* undergoes Calamares variable- +expansion (e.g. replacing `${ROOT}` by the target of the installation). +See *shellprocess* documentation for details. Optional keys are *timeout* and *chroot*. From 028e9e68f9d9bc89b6e0ef2cb71169f53cae15a1 Mon Sep 17 00:00:00 2001 From: Adriaan de Groot Date: Mon, 26 Feb 2024 22:33:17 +0100 Subject: [PATCH 13/13] [libcalamares] Add ${LANG} to process expansions --- src/libcalamares/utils/CommandList.cpp | 13 ++++++++++++- src/modules/shellprocess/shellprocess.conf | 3 +++ 2 files changed, 15 insertions(+), 1 deletion(-) diff --git a/src/libcalamares/utils/CommandList.cpp b/src/libcalamares/utils/CommandList.cpp index 35295fedc..86efe1d38 100644 --- a/src/libcalamares/utils/CommandList.cpp +++ b/src/libcalamares/utils/CommandList.cpp @@ -14,6 +14,7 @@ #include "JobQueue.h" #include "compat/Variant.h" +#include "locale/Global.h" #include "utils/Logger.h" #include "utils/StringExpander.h" #include "utils/System.h" @@ -77,6 +78,16 @@ get_gs_expander( System::RunLocation location ) expander.insert( QStringLiteral( "USER" ), gs->value( "username" ).toString() ); } + if ( gs ) + { + const auto key = QStringLiteral( "LANG" ); + const QString lang = Calamares::Locale::readGS( *gs, key ); + if ( !lang.isEmpty() ) + { + expander.insert( key, lang ); + } + } + return expander; } @@ -106,7 +117,7 @@ CommandLine::expand( KMacroExpanderBase& expander ) const // .. and expand in each environment key=value string. QStringList e = m_environment; - std::for_each(e.begin(), e.end(), [&expander](QString & s) { expander.expandMacrosShellQuote(s);}); + std::for_each( e.begin(), e.end(), [ &expander ]( QString& s ) { expander.expandMacrosShellQuote( s ); } ); return { c, m_environment, m_timeout }; } diff --git a/src/modules/shellprocess/shellprocess.conf b/src/modules/shellprocess/shellprocess.conf index 881181a0d..87e31c58c 100644 --- a/src/modules/shellprocess/shellprocess.conf +++ b/src/modules/shellprocess/shellprocess.conf @@ -12,6 +12,9 @@ # system from the point of view of the command (when run in the target # system, e.g. when *dontChroot* is false, that will be `/`). # - `USER` is replaced by the username, set on the user page. +# - `LANG` is replaced by the language chosen for the user-interface +# of Calamares, set on the welcome page. This may not reflect the +# chosen system language from the locale page. # # Variables are written as `${var}`, e.g. `${ROOT}`. # Write `$$` to get a shell-escaped `\$` in the shell command.