тайм-ауты и системные команды - PullRequest
23 голосов
/ 28 ноября 2011

У меня есть тайм-аут ruby, который вызывает системную (bash) команду, подобную этой.

Timeout::timeout(10) {
  `my_bash_command -c12 -o text.txt`
}

, но я думаю, что даже если поток ruby ​​прерывается, фактическая команда продолжает работать в фоновом режиме.это нормально?Как я могу убить его?

Ответы [ 4 ]

33 голосов
/ 28 ноября 2011

Я думаю, вам нужно 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
11 голосов
/ 10 марта 2013

Чтобы правильно остановить порожденное дерево процессов (а не только родительский процесс), нужно рассмотреть что-то вроде этого:

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

это также позволяет отслеживать состояние процесса

7 голосов
/ 17 июля 2015

Возможно, это поможет кому-то другому, ищущему подобную функциональность тайм-аута, но ему нужно собрать выходные данные из команды оболочки.

Я адаптировал метод @ shurikk для работы с Ruby 2.0 и некоторым кодом из Разветвить дочерний процесс с тайм-аутом и захватить вывод , чтобы собрать вывод.

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
1 голос
/ 21 июня 2017

Обработка процессов, сигналов и таймеров не очень проста.Вот почему вы можете подумать о делегировании этой задачи: используйте команду timeout в новых версиях Linux:

timeout --kill-after 5s 10s my_bash_command -c12 -o text.txt
...