process_executer 1.3.0 → 3.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/CHANGELOG.md +31 -0
- data/README.md +171 -61
- data/lib/process_executer/destination_base.rb +83 -0
- data/lib/process_executer/destinations/child_redirection.rb +23 -0
- data/lib/process_executer/destinations/close.rb +23 -0
- data/lib/process_executer/destinations/file_descriptor.rb +36 -0
- data/lib/process_executer/destinations/file_path.rb +56 -0
- data/lib/process_executer/destinations/file_path_mode.rb +60 -0
- data/lib/process_executer/destinations/file_path_mode_perms.rb +61 -0
- data/lib/process_executer/destinations/io.rb +33 -0
- data/lib/process_executer/destinations/monitored_pipe.rb +39 -0
- data/lib/process_executer/destinations/stderr.rb +31 -0
- data/lib/process_executer/destinations/stdout.rb +31 -0
- data/lib/process_executer/destinations/tee.rb +60 -0
- data/lib/process_executer/destinations/writer.rb +33 -0
- data/lib/process_executer/destinations.rb +70 -0
- data/lib/process_executer/errors.rb +134 -0
- data/lib/process_executer/monitored_pipe.rb +40 -57
- data/lib/process_executer/options/base.rb +240 -0
- data/lib/process_executer/options/option_definition.rb +56 -0
- data/lib/process_executer/options/run_options.rb +48 -0
- data/lib/process_executer/options/spawn_and_wait_options.rb +39 -0
- data/lib/process_executer/options/spawn_options.rb +143 -0
- data/lib/process_executer/options.rb +7 -163
- data/lib/process_executer/result.rb +150 -0
- data/lib/process_executer/runner.rb +155 -0
- data/lib/process_executer/version.rb +1 -1
- data/lib/process_executer.rb +254 -93
- metadata +27 -14
- data/.tool-versions +0 -1
- data/lib/process_executer/command/errors.rb +0 -170
- data/lib/process_executer/command/result.rb +0 -77
- data/lib/process_executer/command/runner.rb +0 -167
- data/lib/process_executer/command.rb +0 -12
- data/lib/process_executer/status.rb +0 -70
@@ -0,0 +1,150 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require 'delegate'
|
4
|
+
|
5
|
+
module ProcessExecuter
|
6
|
+
# A decorator for Process::Status that adds the following attributes:
|
7
|
+
#
|
8
|
+
# * `command`: the command that was used to spawn the process
|
9
|
+
# * `options`: the options that were used to spawn the process
|
10
|
+
# * `elapsed_time`: the secs the command ran
|
11
|
+
# * `stdout`: the captured stdout output
|
12
|
+
# * `stderr`: the captured stderr output
|
13
|
+
# * `timed_out?`: true if the process timed out
|
14
|
+
#
|
15
|
+
# @api public
|
16
|
+
#
|
17
|
+
class Result < SimpleDelegator
|
18
|
+
# Create a new Result object
|
19
|
+
#
|
20
|
+
# @param status [Process::Status] the status to delegate to
|
21
|
+
# @param command [Array] the command that was used to spawn the process
|
22
|
+
# @param options [ProcessExecuter::Options] the options that were used to spawn the process
|
23
|
+
# @param timed_out [Boolean] true if the process timed out
|
24
|
+
# @param elapsed_time [Numeric] the secs the command ran
|
25
|
+
#
|
26
|
+
# @example
|
27
|
+
# command = ['sleep 1']
|
28
|
+
# options = ProcessExecuter::Options.new(timeout_after: 0.5)
|
29
|
+
# start_time = Process.clock_gettime(Process::CLOCK_MONOTONIC)
|
30
|
+
# timed_out = false
|
31
|
+
# status = nil
|
32
|
+
# pid = Process.spawn(*command, **options.spawn_options)
|
33
|
+
# Timeout.timeout(options.timeout_after) do
|
34
|
+
# _pid, status = Process.wait2(pid)
|
35
|
+
# rescue Timeout::Error
|
36
|
+
# Process.kill('KILL', pid)
|
37
|
+
# timed_out = true
|
38
|
+
# _pid, status = Process.wait2(pid)
|
39
|
+
# end
|
40
|
+
# elapsed_time = Process.clock_gettime(Process::CLOCK_MONOTONIC) - start_time
|
41
|
+
#
|
42
|
+
# ProcessExecuter::Result.new(status, command:, options:, timed_out:, elapsed_time:)
|
43
|
+
#
|
44
|
+
# @api public
|
45
|
+
#
|
46
|
+
def initialize(status, command:, options:, timed_out:, elapsed_time:)
|
47
|
+
super(status)
|
48
|
+
@command = command
|
49
|
+
@options = options
|
50
|
+
@timed_out = timed_out
|
51
|
+
@elapsed_time = elapsed_time
|
52
|
+
end
|
53
|
+
|
54
|
+
# The command that was used to spawn the process
|
55
|
+
# @see Process.spawn
|
56
|
+
# @example
|
57
|
+
# result.command #=> [{ 'GIT_DIR' => '/path/to/repo' }, 'git', 'status']
|
58
|
+
# @return [Array]
|
59
|
+
attr_reader :command
|
60
|
+
|
61
|
+
# The options that were used to spawn the process
|
62
|
+
# @see Process.spawn
|
63
|
+
# @example
|
64
|
+
# result.options #=> { chdir: '/path/to/repo', timeout_after: 0.5 }
|
65
|
+
# @return [Hash]
|
66
|
+
# @api public
|
67
|
+
attr_reader :options
|
68
|
+
|
69
|
+
# The secs the command ran
|
70
|
+
# @example
|
71
|
+
# result.elapsed_time #=> 10
|
72
|
+
# @return [Numeric, nil]
|
73
|
+
# @api public
|
74
|
+
attr_reader :elapsed_time
|
75
|
+
|
76
|
+
# @!attribute [r] timed_out?
|
77
|
+
# True if the process timed out and was sent the SIGKILL signal
|
78
|
+
# @example
|
79
|
+
# result = ProcessExecuter.spawn('sleep 10', timeout_after: 0.01)
|
80
|
+
# result.timed_out? # => true
|
81
|
+
# @return [Boolean]
|
82
|
+
#
|
83
|
+
def timed_out?
|
84
|
+
@timed_out
|
85
|
+
end
|
86
|
+
|
87
|
+
# Overrides the default success? method to return nil if the process timed out
|
88
|
+
#
|
89
|
+
# This is because when a timeout occurs, Windows will still return true.
|
90
|
+
#
|
91
|
+
# @example
|
92
|
+
# result = ProcessExecuter.spawn('sleep 10', timeout_after: 0.01)
|
93
|
+
# result.success? # => nil
|
94
|
+
# @return [true, nil]
|
95
|
+
#
|
96
|
+
def success?
|
97
|
+
return nil if timed_out? # rubocop:disable Style/ReturnNilInPredicateMethodDefinition
|
98
|
+
|
99
|
+
super
|
100
|
+
end
|
101
|
+
|
102
|
+
# Return a string representation of the result
|
103
|
+
# @example
|
104
|
+
# result.to_s #=> "pid 70144 SIGKILL (signal 9) timed out after 10s"
|
105
|
+
# @return [String]
|
106
|
+
def to_s
|
107
|
+
"#{super}#{timed_out? ? " timed out after #{options.timeout_after}s" : ''}"
|
108
|
+
end
|
109
|
+
|
110
|
+
# Return the captured stdout output
|
111
|
+
#
|
112
|
+
# This output is only returned if the `:out` option value is a
|
113
|
+
# `ProcessExecuter::MonitoredPipe`.
|
114
|
+
#
|
115
|
+
# @example
|
116
|
+
# # Note that `ProcessExecuter.run` will wrap the given out: object in a
|
117
|
+
# # ProcessExecuter::MonitoredPipe
|
118
|
+
# result = ProcessExecuter.run('echo hello': out: StringIO.new)
|
119
|
+
# result.stdout #=> "hello\n"
|
120
|
+
#
|
121
|
+
# @return [String, nil]
|
122
|
+
#
|
123
|
+
def stdout
|
124
|
+
pipe = options.stdout_redirection_value
|
125
|
+
return nil unless pipe.is_a?(ProcessExecuter::MonitoredPipe)
|
126
|
+
|
127
|
+
pipe.destination.string
|
128
|
+
end
|
129
|
+
|
130
|
+
# Return the captured stderr output
|
131
|
+
#
|
132
|
+
# This output is only returned if the `:err` option value is a
|
133
|
+
# `ProcessExecuter::MonitoredPipe`.
|
134
|
+
#
|
135
|
+
# @example
|
136
|
+
# # Note that `ProcessExecuter.run` will wrap the given err: object in a
|
137
|
+
# # ProcessExecuter::MonitoredPipe
|
138
|
+
# result = ProcessExecuter.run('echo ERROR 1>&2', err: StringIO.new)
|
139
|
+
# resuilt.stderr #=> "ERROR\n"
|
140
|
+
#
|
141
|
+
# @return [String, nil]
|
142
|
+
#
|
143
|
+
def stderr
|
144
|
+
pipe = options.stderr_redirection_value
|
145
|
+
return nil unless pipe.is_a?(ProcessExecuter::MonitoredPipe)
|
146
|
+
|
147
|
+
pipe.destination.string
|
148
|
+
end
|
149
|
+
end
|
150
|
+
end
|
@@ -0,0 +1,155 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require_relative 'errors'
|
4
|
+
|
5
|
+
module ProcessExecuter
|
6
|
+
# The `Runner` class executes subprocess commands and captures their status and output.
|
7
|
+
#
|
8
|
+
# It does the following:
|
9
|
+
# - Run commands (`call`) with options for capturing output, handling timeouts, and merging stdout/stderr.
|
10
|
+
# - Process command results, including logging and error handling.
|
11
|
+
# - Raise detailed exceptions for common command failures, such as timeouts or subprocess errors.
|
12
|
+
#
|
13
|
+
# This class is used internally by {ProcessExecuter.run}.
|
14
|
+
#
|
15
|
+
# @api public
|
16
|
+
#
|
17
|
+
class Runner
|
18
|
+
# Run a command and return the status including stdout and stderr output
|
19
|
+
#
|
20
|
+
# @example
|
21
|
+
# runner = ProcessExecuter::Runner.new()
|
22
|
+
# result = runner.call('echo hello')
|
23
|
+
# result = ProcessExecuter.run('echo hello')
|
24
|
+
# result.success? # => true
|
25
|
+
# result.exitstatus # => 0
|
26
|
+
# result.stdout # => "hello\n"
|
27
|
+
# result.stderr # => ""
|
28
|
+
#
|
29
|
+
# @param command [Array<String>] The command to run
|
30
|
+
# @param options [ProcessExecuter::Options::RunOptions] Options for running the command
|
31
|
+
#
|
32
|
+
# @return [ProcessExecuter::Result] The result of the completed subprocess
|
33
|
+
#
|
34
|
+
def call(command, options)
|
35
|
+
spawn(command, options).tap { |result| process_result(result) }
|
36
|
+
end
|
37
|
+
|
38
|
+
private
|
39
|
+
|
40
|
+
# Wrap the output buffers in pipes and then execute the command
|
41
|
+
#
|
42
|
+
# @param command [Array<String>] The command to execute
|
43
|
+
# @param options [ProcessExecuter::Options::RunOptions] Options for running the command
|
44
|
+
#
|
45
|
+
# @raise [ProcessExecuter::ProcessIOError] If an exception was raised while collecting subprocess output
|
46
|
+
# @raise [ProcessExecuter::TimeoutError] If the command times out
|
47
|
+
#
|
48
|
+
# @return [ProcessExecuter::Result] The result of the completed subprocess
|
49
|
+
#
|
50
|
+
# @api private
|
51
|
+
#
|
52
|
+
def spawn(command, options)
|
53
|
+
opened_pipes = wrap_stdout_stderr(options)
|
54
|
+
ProcessExecuter.spawn_and_wait_with_options(command, options)
|
55
|
+
ensure
|
56
|
+
opened_pipes.each { |key, value| close_pipe(command, key, value) }
|
57
|
+
end
|
58
|
+
|
59
|
+
# Wrap the stdout and stderr redirection options with a MonitoredPipe
|
60
|
+
# @param options [ProcessExecuter::Options::RunOptions] Options for running the command
|
61
|
+
# @return [Hash<Object, ProcessExecuter::MonitoredPipe>] The opened pipes (the Object is the option key)
|
62
|
+
# @api private
|
63
|
+
def wrap_stdout_stderr(options)
|
64
|
+
options.each_with_object({}) do |key_value, opened_pipes|
|
65
|
+
key, value = key_value
|
66
|
+
|
67
|
+
next unless should_wrap?(options, key, value)
|
68
|
+
|
69
|
+
wrapped_destination = ProcessExecuter::MonitoredPipe.new(value)
|
70
|
+
opened_pipes[key] = wrapped_destination
|
71
|
+
options.merge!(key => wrapped_destination)
|
72
|
+
end
|
73
|
+
end
|
74
|
+
|
75
|
+
# Should the redirection option be wrapped by a MonitoredPipe
|
76
|
+
# @param key [Object] The option key
|
77
|
+
# @param value [Object] The option value
|
78
|
+
# @return [Boolean] Whether the option should be wrapped
|
79
|
+
# @api private
|
80
|
+
def should_wrap?(options, key, value)
|
81
|
+
(options.stdout_redirection?(key) || options.stderr_redirection?(key)) &&
|
82
|
+
ProcessExecuter::Destinations.compatible_with_monitored_pipe?(value)
|
83
|
+
end
|
84
|
+
|
85
|
+
# Close the pipe and raise an error if the pipe raised an exception
|
86
|
+
# @return [void]
|
87
|
+
# @raise [ProcessExecuter::ProcessIOError] If an exception was raised while
|
88
|
+
# collecting subprocess output
|
89
|
+
# @api private
|
90
|
+
def close_pipe(command, option_key, pipe)
|
91
|
+
pipe.close
|
92
|
+
raise_pipe_error(command, option_key, pipe) if pipe.exception
|
93
|
+
end
|
94
|
+
|
95
|
+
# Process the result of the command and return a ProcessExecuter::Result
|
96
|
+
#
|
97
|
+
# Log the command and result, and raise an error if the command failed.
|
98
|
+
#
|
99
|
+
# @param result [ProcessExecuter::Result] The result of the command
|
100
|
+
#
|
101
|
+
# @return [Void]
|
102
|
+
#
|
103
|
+
# @raise [ProcessExecuter::FailedError] If the command failed
|
104
|
+
# @raise [ProcessExecuter::SignaledError] If the command was signaled
|
105
|
+
# @raise [ProcessExecuter::TimeoutError] If the command times out
|
106
|
+
# @raise [ProcessExecuter::ProcessIOError] If an exception was raised while collecting subprocess output
|
107
|
+
#
|
108
|
+
# @api private
|
109
|
+
#
|
110
|
+
def process_result(result)
|
111
|
+
log_result(result)
|
112
|
+
|
113
|
+
raise_errors(result) if result.options.raise_errors
|
114
|
+
end
|
115
|
+
|
116
|
+
# Raise an error if the command failed
|
117
|
+
# @return [void]
|
118
|
+
# @raise [ProcessExecuter::FailedError] If the command failed
|
119
|
+
# @raise [ProcessExecuter::SignaledError] If the command was signaled
|
120
|
+
# @raise [ProcessExecuter::TimeoutError] If the command times out
|
121
|
+
# @api private
|
122
|
+
def raise_errors(result)
|
123
|
+
raise TimeoutError, result if result.timed_out?
|
124
|
+
raise SignaledError, result if result.signaled?
|
125
|
+
raise FailedError, result unless result.success?
|
126
|
+
end
|
127
|
+
|
128
|
+
# Log the result of running the command
|
129
|
+
# @param result [ProcessExecuter::Result] the result of the command including
|
130
|
+
# the command, status, stdout, and stderr
|
131
|
+
# @return [void]
|
132
|
+
# @api private
|
133
|
+
def log_result(result)
|
134
|
+
result.options.logger.info { "#{result.command} exited with status #{result}" }
|
135
|
+
result.options.logger.debug { "stdout:\n#{result.stdout.inspect}\nstderr:\n#{result.stderr.inspect}" }
|
136
|
+
end
|
137
|
+
|
138
|
+
# Raise an error when there was exception while collecting the subprocess output
|
139
|
+
#
|
140
|
+
# @param command [Array<String>] The command that was executed
|
141
|
+
# @param option_key [Symbol] The name of the pipe that raised the exception
|
142
|
+
# @param pipe [ProcessExecuter::MonitoredPipe] The pipe that raised the exception
|
143
|
+
#
|
144
|
+
# @raise [ProcessExecuter::ProcessIOError]
|
145
|
+
#
|
146
|
+
# @return [void] This method always raises an error
|
147
|
+
#
|
148
|
+
# @api private
|
149
|
+
#
|
150
|
+
def raise_pipe_error(command, option_key, pipe)
|
151
|
+
error = ProcessExecuter::ProcessIOError.new("Pipe Exception for #{command}: #{option_key.inspect}")
|
152
|
+
raise(error, cause: pipe.exception)
|
153
|
+
end
|
154
|
+
end
|
155
|
+
end
|