Возможно, это поможет кому-то еще надеяться на достижение аналогичных функций тайм-аута, но необходимо собрать вывод из команды оболочки.
Я адаптировал метод @ shurikk для работы с Ruby 2.0 и некоторый код от Fork child process with timeout and capture output для сбора результатов.
def exec_with_timeout(cmd, timeout)
begin
# stdout, stderr pipes
rout, wout = IO.pipe
rerr, werr = IO.pipe
stdout, stderr = nil
pid = Process.spawn(cmd, pgroup: true, :out => wout, :err => werr)
Timeout.timeout(timeout) do
Process.waitpid(pid)
# close write ends so we can read from them
wout.close
werr.close
stdout = rout.readlines.join
stderr = rerr.readlines.join
end
rescue Timeout::Error
Process.kill(-9, pid)
Process.detach(pid)
ensure
wout.close unless wout.closed?
werr.close unless werr.closed?
# dispose the read ends of the pipes
rout.close
rerr.close
end
stdout
end
Это неправда. Sub-shell, выполняющий команду, должен завершиться, когда родительский рубиновый процесс завершается. Пожалуйста, дайте более конкретный пример. –
@BenLee: родительский процесс не заканчивается, когда истекает время ожидания. –
@ MladenJablanović, в быстром эксперименте. Я создал файл ruby, который ничего не делал: 'require 'timeout'; Таймаут :: таймаут (100) {'sleep 500'}'. Во время запуска я делаю 'ps aux | grep sleep' и посмотреть процесс сна. Затем я отправляю SIGKILL в рубиновый процесс и снова запускаю 'ps aux | grep sleep' и больше не видят дочерний процесс. –