require 'childprocess' require 'tempfile' require 'shellwords' require 'aruba/errors' require 'aruba/processes/basic_process' require 'aruba/platform' module Aruba module Processes class SpawnProcess < BasicProcess # Use as default launcher def self.match?(mode) true end # Create process # # @params [String] cmd # Command string # # @params [Integer] exit_timeout # The timeout until we expect the command to be finished # # @params [Integer] io_wait # The timeout until we expect the io to be finished # # @params [String] working_directory # The directory where the command will be executed def initialize(cmd, exit_timeout, io_wait, working_directory, environment = ENV.to_hash.dup, main_class = nil) super @exit_timeout = exit_timeout @io_wait = io_wait @process = nil @output_cache = nil @error_cache = nil end # Run the command # # @yield [SpawnProcess] # Run code for process which was started # # rubocop:disable Metrics/MethodLength # rubocop:disable Metrics/CyclomaticComplexity def run! # gather fully qualified path cmd = which(command) # rubocop:disable Metrics/LineLength fail LaunchError, %(Command "#{command}" not found in PATH-variable "#{environment['PATH']}".) if cmd.nil? # rubocop:enable Metrics/LineLength cmd = Aruba.platform.command_string.new(cmd) @process = ChildProcess.build(*[cmd.to_a, arguments].flatten) @out = Tempfile.new("aruba-out") @err = Tempfile.new("aruba-err") @exit_status = nil @duplex = true before_run @process.io.stdout = @out @process.io.stderr = @err @process.duplex = @duplex @process.cwd = @working_directory @process.environment.update(environment) begin @process.start rescue ChildProcess::LaunchError => e raise LaunchError, "It tried to start #{cmd}. " + e.message end after_run yield self if block_given? end # rubocop:enable Metrics/MethodLength # rubocop:enable Metrics/CyclomaticComplexity def stdin @process.io.stdin end def stdout wait_for_io do @process.io.stdout.flush read(@out) end || @output_cache end def stderr wait_for_io do @process.io.stderr.flush read(@err) end || @error_cache end def read_stdout warn('The use of "#read_stdout" is deprecated. Use "#stdout" instead.') stdout end def write(input) @process.io.stdin.write(input) @process.io.stdin.flush end def close_io(name) if RUBY_VERSION < '1.9' @process.io.send(name.to_sym).close else @process.io.public_send(name.to_sym).close end end def stop(reader) @stopped = true return @exit_status unless @process begin @process.poll_for_exit(@exit_timeout) unless @process.exited? rescue ChildProcess::TimeoutError @timed_out = true @process.stop end @exit_status = @process.exit_code @process = nil close_and_cache_out close_and_cache_err if reader reader.announce :stdout, stdout reader.announce :stderr, stderr end @exit_status end def terminate return unless @process @process.stop stop nil end private def wait_for_io(&block) return unless @process sleep @io_wait yield end def read(io) io.rewind io.read end def close_and_cache_out @output_cache = read @out @out.close @out = nil end def close_and_cache_err @error_cache = read @err @err.close @err = nil end end end end