Возможно, это поможет кому-то еще искать аналогичные функции тайм-аута, но ему нужно собрать результат из команды оболочки.
Я адаптировал метод @ shurikk для работы с Ruby 2.0 и некоторый код из Выполняет дочерний процесс Fork с тайм-аутом и выходом захвата для сбора выходных данных.
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
Open3.capture*
будет проще. Я предполагаю, что он будет убирать после себя по таймаутам.
– akostadinov
11 August 2015 в 20:43
Обработка процессов, сигналов и таймеров не очень проста. Вот почему вы можете рассмотреть делегирование этой задачи: Используйте команду timeout
для новых версий Linux:
timeout --kill-after 5s 10s my_bash_command -c12 -o text.txt
timeout <duration> <command>
: так, например, для timeout 10s my_bash_command
. Опция --kill-after
зависит от количества времени ожидания после отправки сигнала термина. Используя этот параметр, вам все равно нужно указать начальную продолжительность: timeout --kill-after=5s 10s my_bash_command
.
– yves
17 January 2018 в 11:10
, чтобы правильно остановить порожденное дерево процессов (а не только родительский процесс), нужно подумать о чем-то вроде этого:
def exec_with_timeout(cmd, timeout)
pid = Process.spawn(cmd, {[:err,:out] => :close, :pgroup => true})
begin
Timeout.timeout(timeout) do
Process.waitpid(pid, 0)
$?.exitstatus == 0
end
rescue Timeout::Error
Process.kill(15, -Process.getpgid(pid))
false
end
end
это также позволяет отслеживать статус процесса