Class: HTTPX::Pool
- Inherits:
-
Object
- Object
- HTTPX::Pool
- Defined in:
- lib/httpx/pool.rb
Constant Summary collapse
- POOL_TIMEOUT =
5
Instance Method Summary collapse
- #checkin_connection(connection) ⇒ Object
- #checkin_resolver(resolver) ⇒ Object
-
#checkout_connection(uri, options) ⇒ Object
opens a connection to the IP reachable through
uri. - #checkout_mergeable_connection(connection) ⇒ Object
- #checkout_resolver(options) ⇒ Object
-
#initialize(options) ⇒ Pool
constructor
Sets up the connection pool with the given
options, which can be the following:. -
#inspect ⇒ Object
:nocov:.
-
#pop_connection ⇒ Object
connections returned by this function are not expected to return to the connection pool.
- #reset_resolvers ⇒ Object
Constructor Details
#initialize(options) ⇒ Pool
Sets up the connection pool with the given options, which can be the following:
- :max_connections
-
the maximum number of connections held in the pool.
- :max_connections_per_origin
-
the maximum number of connections held in the pool pointing to a given origin.
- :pool_timeout
-
the number of seconds to wait for a connection to a given origin (before raising HTTPX::PoolTimeoutError)
21 22 23 24 25 26 27 28 29 30 31 32 33 |
# File 'lib/httpx/pool.rb', line 21 def initialize() @max_connections = .fetch(:max_connections, Float::INFINITY) @max_connections_per_origin = .fetch(:max_connections_per_origin, Float::INFINITY) @pool_timeout = .fetch(:pool_timeout, POOL_TIMEOUT) @resolvers = Hash.new { |hs, resolver_type| hs[resolver_type] = [] } @resolver_mtx = Thread::Mutex.new @connections = [] @connection_mtx = Thread::Mutex.new @connections_counter = 0 @max_connections_cond = ConditionVariable.new @origin_counters = Hash.new(0) @origin_conds = Hash.new { |hs, orig| hs[orig] = ConditionVariable.new } end |
Instance Method Details
#checkin_connection(connection) ⇒ Object
110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 |
# File 'lib/httpx/pool.rb', line 110 def checkin_connection(connection) return if connection..io @connection_mtx.synchronize do if connection.coalesced? || connection.state == :idle # when connections coalesce drop_connection(connection) return end @connections << connection @max_connections_cond.signal @origin_conds[connection.origin.to_s].signal # Observed situations where a session handling multiple requests in a loop # across multiple threads checks the same connection in and out, while another # thread which is waiting on the same connection never gets the chance to pick # it up, because ruby's thread scheduler never switched on to it in the process. Thread.pass end end |
#checkin_resolver(resolver) ⇒ Object
163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 |
# File 'lib/httpx/pool.rb', line 163 def checkin_resolver(resolver) resolver_class = resolver.class resolver = resolver.multi # a multi requires all sub-resolvers being closed in order to be # correctly checked back in. return unless resolver.closed? @resolver_mtx.synchronize do resolvers = @resolvers[resolver_class] resolvers << resolver unless resolvers.include?(resolver) end end |
#checkout_connection(uri, options) ⇒ Object
opens a connection to the IP reachable through uri. Many hostnames are reachable through the same IP, so we try to maximize pipelining by opening as few connections as possible.
46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 |
# File 'lib/httpx/pool.rb', line 46 def checkout_connection(uri, ) return checkout_new_connection(uri, ) if .io @connection_mtx.synchronize do acquire_connection(uri, ) || begin if @connections_counter == @max_connections # this takes precedence over per-origin expires_at = Utils.now + @pool_timeout loop do @max_connections_cond.wait(@connection_mtx, @pool_timeout) if (conn = acquire_connection(uri, )) return conn end # if one can afford to create a new connection, do it break unless @connections_counter == @max_connections # if no matching usable connection was found, the pool will make room and drop a closed connection. if (conn = @connections.find { |c| c.state == :closed }) drop_connection(conn) break end # happens when a condition was signalled, but another thread snatched the available connection before # context was passed back here. next if Utils.now < expires_at raise PoolTimeoutError.new(@pool_timeout, "Timed out after #{@pool_timeout} seconds while waiting for a connection") end end if @origin_counters[uri.origin] == @max_connections_per_origin expires_at = Utils.now + @pool_timeout loop do @origin_conds[uri.origin].wait(@connection_mtx, @pool_timeout) if (conn = acquire_connection(uri, )) return conn end # happens when a condition was signalled, but another thread snatched the available connection before # context was passed back here. next if Utils.now < expires_at raise(PoolTimeoutError.new(@pool_timeout, "Timed out after #{@pool_timeout} seconds while waiting for a connection to #{uri.origin}")) end end @connections_counter += 1 @origin_counters[uri.origin] += 1 checkout_new_connection(uri, ) end end end |
#checkout_mergeable_connection(connection) ⇒ Object
134 135 136 137 138 139 140 141 142 143 |
# File 'lib/httpx/pool.rb', line 134 def checkout_mergeable_connection(connection) return if connection..io @connection_mtx.synchronize do idx = @connections.find_index do |ch| ch != connection && ch.mergeable?(connection) end @connections.delete_at(idx) if idx end end |
#checkout_resolver(options) ⇒ Object
149 150 151 152 153 154 155 156 157 158 159 160 161 |
# File 'lib/httpx/pool.rb', line 149 def checkout_resolver() resolver_type = .resolver_class resolver_type = Resolver.resolver_for(resolver_type, ) @resolver_mtx.synchronize do resolvers = @resolvers[resolver_type] idx = resolvers.find_index do |res| res. == end resolvers.delete_at(idx) if idx end || checkout_new_resolver(resolver_type, ) end |
#inspect ⇒ Object
:nocov:
180 181 182 183 184 185 186 |
# File 'lib/httpx/pool.rb', line 180 def inspect "#<#{self.class}:#{object_id} " \ "@max_connections=#{@max_connections} " \ "@max_connections_per_origin=#{@max_connections_per_origin} " \ "@pool_timeout=#{@pool_timeout} " \ "@connections=#{@connections.size}>" end |
#pop_connection ⇒ Object
connections returned by this function are not expected to return to the connection pool.
36 37 38 39 40 |
# File 'lib/httpx/pool.rb', line 36 def pop_connection @connection_mtx.synchronize do drop_connection end end |
#reset_resolvers ⇒ Object
145 146 147 |
# File 'lib/httpx/pool.rb', line 145 def reset_resolvers @resolver_mtx.synchronize { @resolvers.clear } end |