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.
Files changed (36) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +31 -0
  3. data/README.md +171 -61
  4. data/lib/process_executer/destination_base.rb +83 -0
  5. data/lib/process_executer/destinations/child_redirection.rb +23 -0
  6. data/lib/process_executer/destinations/close.rb +23 -0
  7. data/lib/process_executer/destinations/file_descriptor.rb +36 -0
  8. data/lib/process_executer/destinations/file_path.rb +56 -0
  9. data/lib/process_executer/destinations/file_path_mode.rb +60 -0
  10. data/lib/process_executer/destinations/file_path_mode_perms.rb +61 -0
  11. data/lib/process_executer/destinations/io.rb +33 -0
  12. data/lib/process_executer/destinations/monitored_pipe.rb +39 -0
  13. data/lib/process_executer/destinations/stderr.rb +31 -0
  14. data/lib/process_executer/destinations/stdout.rb +31 -0
  15. data/lib/process_executer/destinations/tee.rb +60 -0
  16. data/lib/process_executer/destinations/writer.rb +33 -0
  17. data/lib/process_executer/destinations.rb +70 -0
  18. data/lib/process_executer/errors.rb +134 -0
  19. data/lib/process_executer/monitored_pipe.rb +40 -57
  20. data/lib/process_executer/options/base.rb +240 -0
  21. data/lib/process_executer/options/option_definition.rb +56 -0
  22. data/lib/process_executer/options/run_options.rb +48 -0
  23. data/lib/process_executer/options/spawn_and_wait_options.rb +39 -0
  24. data/lib/process_executer/options/spawn_options.rb +143 -0
  25. data/lib/process_executer/options.rb +7 -163
  26. data/lib/process_executer/result.rb +150 -0
  27. data/lib/process_executer/runner.rb +155 -0
  28. data/lib/process_executer/version.rb +1 -1
  29. data/lib/process_executer.rb +254 -93
  30. metadata +27 -14
  31. data/.tool-versions +0 -1
  32. data/lib/process_executer/command/errors.rb +0 -170
  33. data/lib/process_executer/command/result.rb +0 -77
  34. data/lib/process_executer/command/runner.rb +0 -167
  35. data/lib/process_executer/command.rb +0 -12
  36. 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
@@ -2,5 +2,5 @@
2
2
 
3
3
  module ProcessExecuter
4
4
  # The current Gem version
5
- VERSION = '1.3.0'
5
+ VERSION = '3.0.0'
6
6
  end