Change request timeout handling to use a longer deadline (#26055)
This commit is contained in:
		
							parent
							
								
									6657695ec6
								
							
						
					
					
						commit
						ff41e5426a
					
				@ -4,14 +4,22 @@ require 'ipaddr'
 | 
				
			|||||||
require 'socket'
 | 
					require 'socket'
 | 
				
			||||||
require 'resolv'
 | 
					require 'resolv'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
# Monkey-patch the HTTP.rb timeout class to avoid using a timeout block
 | 
					# Use our own timeout class to avoid using HTTP.rb's timeout block
 | 
				
			||||||
# around the Socket#open method, since we use our own timeout blocks inside
 | 
					# around the Socket#open method, since we use our own timeout blocks inside
 | 
				
			||||||
# that method
 | 
					# that method
 | 
				
			||||||
#
 | 
					#
 | 
				
			||||||
# Also changes how the read timeout behaves so that it is cumulative (closer
 | 
					# Also changes how the read timeout behaves so that it is cumulative (closer
 | 
				
			||||||
# to HTTP::Timeout::Global, but still having distinct timeouts for other
 | 
					# to HTTP::Timeout::Global, but still having distinct timeouts for other
 | 
				
			||||||
# operation types)
 | 
					# operation types)
 | 
				
			||||||
class HTTP::Timeout::PerOperation
 | 
					class PerOperationWithDeadline < HTTP::Timeout::PerOperation
 | 
				
			||||||
 | 
					  READ_DEADLINE = 30
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  def initialize(*args)
 | 
				
			||||||
 | 
					    super
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @read_deadline = options.fetch(:read_deadline, READ_DEADLINE)
 | 
				
			||||||
 | 
					  end
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  def connect(socket_class, host, port, nodelay = false)
 | 
					  def connect(socket_class, host, port, nodelay = false)
 | 
				
			||||||
    @socket = socket_class.open(host, port)
 | 
					    @socket = socket_class.open(host, port)
 | 
				
			||||||
    @socket.setsockopt(Socket::IPPROTO_TCP, Socket::TCP_NODELAY, 1) if nodelay
 | 
					    @socket.setsockopt(Socket::IPPROTO_TCP, Socket::TCP_NODELAY, 1) if nodelay
 | 
				
			||||||
@ -24,7 +32,7 @@ class HTTP::Timeout::PerOperation
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
  # Read data from the socket
 | 
					  # Read data from the socket
 | 
				
			||||||
  def readpartial(size, buffer = nil)
 | 
					  def readpartial(size, buffer = nil)
 | 
				
			||||||
    @deadline ||= Process.clock_gettime(Process::CLOCK_MONOTONIC) + @read_timeout
 | 
					    @deadline ||= Process.clock_gettime(Process::CLOCK_MONOTONIC) + @read_deadline
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    timeout = false
 | 
					    timeout = false
 | 
				
			||||||
    loop do
 | 
					    loop do
 | 
				
			||||||
@ -33,7 +41,8 @@ class HTTP::Timeout::PerOperation
 | 
				
			|||||||
      return :eof if result.nil?
 | 
					      return :eof if result.nil?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      remaining_time = @deadline - Process.clock_gettime(Process::CLOCK_MONOTONIC)
 | 
					      remaining_time = @deadline - Process.clock_gettime(Process::CLOCK_MONOTONIC)
 | 
				
			||||||
      raise HTTP::TimeoutError, "Read timed out after #{@read_timeout} seconds" if timeout || remaining_time <= 0
 | 
					      raise HTTP::TimeoutError, "Read timed out after #{@read_timeout} seconds" if timeout
 | 
				
			||||||
 | 
					      raise HTTP::TimeoutError, "Read timed out after a total of #{@read_deadline} seconds" if remaining_time <= 0
 | 
				
			||||||
      return result if result != :wait_readable
 | 
					      return result if result != :wait_readable
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      # marking the socket for timeout. Why is this not being raised immediately?
 | 
					      # marking the socket for timeout. Why is this not being raised immediately?
 | 
				
			||||||
@ -46,7 +55,7 @@ class HTTP::Timeout::PerOperation
 | 
				
			|||||||
      # timeout. Else, the first timeout was a proper timeout.
 | 
					      # timeout. Else, the first timeout was a proper timeout.
 | 
				
			||||||
      # This hack has to be done because io/wait#wait_readable doesn't provide a value for when
 | 
					      # This hack has to be done because io/wait#wait_readable doesn't provide a value for when
 | 
				
			||||||
      # the socket is closed by the server, and HTTP::Parser doesn't provide the limit for the chunks.
 | 
					      # the socket is closed by the server, and HTTP::Parser doesn't provide the limit for the chunks.
 | 
				
			||||||
      timeout = true unless @socket.to_io.wait_readable(remaining_time)
 | 
					      timeout = true unless @socket.to_io.wait_readable([remaining_time, @read_timeout].min)
 | 
				
			||||||
    end
 | 
					    end
 | 
				
			||||||
  end
 | 
					  end
 | 
				
			||||||
end
 | 
					end
 | 
				
			||||||
@ -57,7 +66,7 @@ class Request
 | 
				
			|||||||
  # We enforce a 5s timeout on DNS resolving, 5s timeout on socket opening
 | 
					  # We enforce a 5s timeout on DNS resolving, 5s timeout on socket opening
 | 
				
			||||||
  # and 5s timeout on the TLS handshake, meaning the worst case should take
 | 
					  # and 5s timeout on the TLS handshake, meaning the worst case should take
 | 
				
			||||||
  # about 15s in total
 | 
					  # about 15s in total
 | 
				
			||||||
  TIMEOUT = { connect: 5, read: 10, write: 10 }.freeze
 | 
					  TIMEOUT = { connect_timeout: 5, read_timeout: 10, write_timeout: 10, read_deadline: 30 }.freeze
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  include RoutingHelper
 | 
					  include RoutingHelper
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -69,6 +78,7 @@ class Request
 | 
				
			|||||||
    @http_client = options.delete(:http_client)
 | 
					    @http_client = options.delete(:http_client)
 | 
				
			||||||
    @allow_local = options.delete(:allow_local)
 | 
					    @allow_local = options.delete(:allow_local)
 | 
				
			||||||
    @options     = options.merge(socket_class: use_proxy? || @allow_local ? ProxySocket : Socket)
 | 
					    @options     = options.merge(socket_class: use_proxy? || @allow_local ? ProxySocket : Socket)
 | 
				
			||||||
 | 
					    @options     = @options.merge(timeout_class: PerOperationWithDeadline, timeout_options: TIMEOUT)
 | 
				
			||||||
    @options     = @options.merge(proxy_url) if use_proxy?
 | 
					    @options     = @options.merge(proxy_url) if use_proxy?
 | 
				
			||||||
    @headers     = {}
 | 
					    @headers     = {}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -129,7 +139,7 @@ class Request
 | 
				
			|||||||
    end
 | 
					    end
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def http_client
 | 
					    def http_client
 | 
				
			||||||
      HTTP.use(:auto_inflate).timeout(TIMEOUT.dup).follow(max_hops: 3)
 | 
					      HTTP.use(:auto_inflate).follow(max_hops: 3)
 | 
				
			||||||
    end
 | 
					    end
 | 
				
			||||||
  end
 | 
					  end
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
				
			|||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user