From 75e8dd8f1aeea749f9279594c68b0c7f14f59387 Mon Sep 17 00:00:00 2001 From: alex501212 Date: Mon, 22 Jul 2024 13:52:34 +0100 Subject: [PATCH 1/3] run_command now can take an array of commands --- .../lib/puppet/functions/run_command.rb | 21 +++++++++++++++++++ 1 file changed, 21 insertions(+) diff --git a/bolt-modules/boltlib/lib/puppet/functions/run_command.rb b/bolt-modules/boltlib/lib/puppet/functions/run_command.rb index 6869eff757..451d1f6fb6 100644 --- a/bolt-modules/boltlib/lib/puppet/functions/run_command.rb +++ b/bolt-modules/boltlib/lib/puppet/functions/run_command.rb @@ -25,6 +25,23 @@ return_type 'ResultSet' end + # Run multiple commands. + # @param commands Commands to run on target. + # @param targets A pattern identifying zero or more targets. See {get_targets} for accepted patterns. + # @param options A hash of additional options. + # @option options [Boolean] _catch_errors Whether to catch raised errors. + # @option options [String] _run_as User to run as using privilege escalation. + # @option options [Hash[String, Any]] _env_vars Map of environment variables to set + # @return A list of results, one entry per target. + # @example Run commands on targets + # run_command(['hostname', 'whoami'], $targets, '_catch_errors' => true) + dispatch :run_commands do + param 'Array', :commands + param 'Boltlib::TargetSpec', :targets + optional_param 'Hash[String[1], Any]', :options + return_type 'ResultSet' + end + # Run a command, logging the provided description. # @param command A command to run on target. # @param targets A pattern identifying zero or more targets. See {get_targets} for accepted patterns. @@ -48,6 +65,10 @@ def run_command(command, targets, options = {}) run_command_with_description(command, targets, nil, options) end + def run_commands(commands, targets, options = {}) + run_command_with_description(commands.join(' && '), targets, nil, options) + end + def run_command_with_description(command, targets, description = nil, options = {}) unless Puppet[:tasks] raise Puppet::ParseErrorWithIssue From 035859f56884af4e3496bdf868664b9ec425701a Mon Sep 17 00:00:00 2001 From: alex501212 Date: Mon, 22 Jul 2024 14:32:38 +0100 Subject: [PATCH 2/3] !feature * **run_command: support an array of commands* ([#3333](https://github.com/puppetlabs/bolt/issues/3333)) Added run_commands method which allows an array of commands to be passed to run_command From bd833746f7cc6f943ca249a6cc406061bc8bc53f Mon Sep 17 00:00:00 2001 From: alex501212 Date: Fri, 26 Jul 2024 11:59:19 +0100 Subject: [PATCH 3/3] allow for array input for commands --- .../lib/puppet/functions/run_command.rb | 71 ++++++++++++++----- 1 file changed, 54 insertions(+), 17 deletions(-) diff --git a/bolt-modules/boltlib/lib/puppet/functions/run_command.rb b/bolt-modules/boltlib/lib/puppet/functions/run_command.rb index 451d1f6fb6..31e7205dc6 100644 --- a/bolt-modules/boltlib/lib/puppet/functions/run_command.rb +++ b/bolt-modules/boltlib/lib/puppet/functions/run_command.rb @@ -2,6 +2,7 @@ require 'bolt/error' require 'json' +require 'shellwords' # Runs a command on the given set of targets and returns the result from each command execution. # This function does nothing if the list of targets is empty. @@ -66,7 +67,7 @@ def run_command(command, targets, options = {}) end def run_commands(commands, targets, options = {}) - run_command_with_description(commands.join(' && '), targets, nil, options) + run_command_with_description(commands, targets, nil, options) end def run_command_with_description(command, targets, description = nil, options = {}) @@ -104,24 +105,60 @@ def run_command_with_description(command, targets, description = nil, options = # Ensure that given targets are all Target instances targets = inventory.get_targets(targets) - if targets.empty? - call_function('debug', "Simulating run_command('#{command}') - no targets given - no action taken") - Bolt::ResultSet.new([]) - else - file_line = Puppet::Pops::PuppetStack.top_of_stack - r = if executor.in_parallel? - executor.run_in_thread do - executor.run_command(targets, command, options, file_line) - end - else - executor.run_command(targets, command, options, file_line) + # if command is an array, run each command in sequence + if command.is_a?(Array) + results = [] + command.each do |cmd| + # Split the command into an array of arguments + split_cmd = Shellwords.split(cmd) + + # Escape each argument + escaped_cmd = split_cmd.map { |arg| Shellwords.escape(arg) }.join(' ') + + if targets.empty? + call_function('debug', "Simulating run_command('#{escaped_cmd}') - no targets given - no action taken") + Bolt::ResultSet.new([]) + else + file_line = Puppet::Pops::PuppetStack.top_of_stack + result = if executor.in_parallel? + executor.run_in_thread do + executor.run_command(targets, escaped_cmd, options, file_line) + end + else + executor.run_command(targets, escaped_cmd, options, file_line) + end + + if !result.ok && !options[:catch_errors] + raise Bolt::RunFailure.new(result, 'run_command', escaped_cmd) end - - if !r.ok && !options[:catch_errors] - raise Bolt::RunFailure.new(r, 'run_command', command) + results.concat(result.results) + end + end + Bolt::ResultSet.new(results) + else + # Split the command into an array of arguments + split_cmd = Shellwords.split(command) + + escaped_cmd = split_cmd.map { |arg| Shellwords.escape(arg) }.join(' ') + + if targets.empty? + call_function('debug', "Simulating run_command('#{escaped_cmd}') - no targets given - no action taken") + Bolt::Result.new([]) + else + file_line = Puppet::Pops::PuppetStack.top_of_stack + result = if executor.in_parallel? + executor.run_in_thread do + executor.run_command(targets, escaped_cmd, options, file_line) + end + else + executor.run_command(targets, escaped_cmd, options, file_line) + end + + if !result.ok && !options[:catch_errors] + raise Bolt::RunFailure.new(result, 'run_command', escaped_cmd) + end + result end - - r end end end