Подтвердить что ты не робот

Как установить тайм-аут сокета в Ruby?

Как установить таймаут для блокировки операций в Ruby-сокете?

4b9b3361

Ответ 1

Я нашел решение, которое, как представляется, должно использовать Timeout:: timeout:

require 'timeout'
    ...
begin 
    timeout(5) do
        message, client_address = some_socket.recvfrom(1024)
    end
rescue Timeout::Error
    puts "Timed out!"
end

Ответ 2

Объект тайм-аута - хорошее решение.

Это пример асинхронного ввода-вывода (неблокирующий по своей природе и происходит асинхронно с поток приложения.)

IO.select(read_array
[, write_array
[, error_array
[, timeout]]] ) => array or nil

Может использоваться для получения того же эффекта.

require 'socket'

strmSock1 = TCPSocket::new( "www.dn.se", 80 )
strmSock2 = TCPSocket::new( "www.svd.se", 80 )
# Block until one or more events are received
#result = select( [strmSock1, strmSock2, STDIN], nil, nil )
timeout=5

timeout=100
result = select( [strmSock1, strmSock2], nil, nil,timeout )
puts result.inspect
if result

  for inp in result[0]
    if inp == strmSock1 then
      # data avail on strmSock1
      puts "data avail on strmSock1"
    elsif inp == strmSock2 then
      # data avail on strmSock2
      puts "data avail on strmSock2"
    elsif inp == STDIN
      # data avail on STDIN
      puts "data avail on STDIN"
    end
  end
end

Ответ 3

Я думаю, что неблокирующий подход - это путь. Я попробовал вышеупомянутую статью и все еще мог ее повесить.
эта статья не блокирующая сеть, а подход jonke выше меня на правильном пути. Мой сервер блокировал первоначальное соединение, поэтому мне нужно было немного ниже уровня.
сокет rdoc может предоставить более подробную информацию в connect_nonblock

def self.open(host, port, timeout=10)
 addr = Socket.getaddrinfo(host, nil)
 sock = Socket.new(Socket.const_get(addr[0][0]), Socket::SOCK_STREAM, 0)

 begin
  sock.connect_nonblock(Socket.pack_sockaddr_in(port, addr[0][3]))
 rescue Errno::EINPROGRESS
  resp = IO.select([sock],nil, nil, timeout.to_i)
  if resp.nil?
    raise Errno::ECONNREFUSED
  end
  begin
    sock.connect_nonblock(Socket.pack_sockaddr_in(port, addr[0][3]))
  rescue Errno::EISCONN
  end
 end
 sock
end

чтобы получить хороший тест. запускать простой сервер сокетов, а затем сделать ctrl-z для фона

IO.select ожидает, что данные поступят во входной поток в течение 10 секунд. это может не сработать, если это не так.

Это должна быть хорошая замена для открытого метода TCPSocket.