Таймауты рубинов и системные команды
У меня есть тайм-аут ruby, который вызывает команду системы (bash), подобную этой.
Timeout::timeout(10) {
`my_bash_command -c12 -o text.txt`
}
но я думаю, что даже если поток рубина прерывается, фактическая команда продолжает работать в фоновом режиме.. это нормально? Как я могу его убить?
Ответы
Ответ 1
Я думаю, что вы должны kill
его вручную:
require 'timeout'
puts 'starting process'
pid = Process.spawn('sleep 20')
begin
Timeout.timeout(5) do
puts 'waiting for the process to end'
Process.wait(pid)
puts 'process finished in time'
end
rescue Timeout::Error
puts 'process not finished in time, killing it'
Process.kill('TERM', pid)
end
Ответ 2
чтобы правильно остановить порожденное дерево процессов (а не только родительский процесс), нужно подумать над чем-то вроде этого:
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
это также позволяет отслеживать состояние процесса
Ответ 3
Возможно, это поможет кому-то еще искать аналогичные функции тайм-аута, но ему нужно собрать результат из команды оболочки.
Я адаптировал метод @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
Ответ 4
Обработка процессов, сигналов и таймеров не очень проста. Вот почему вы можете рассмотреть делегирование этой задачи: Используйте команду timeout
для новых версий Linux:
timeout --kill-after 5s 10s my_bash_command -c12 -o text.txt