Class: Async::HTTP::Client
- Inherits:
-
Protocol::HTTP::Methods
- Object
- Protocol::HTTP::Methods
- Async::HTTP::Client
- Includes:
- Proxy::Client
- Defined in:
- lib/async/http/client.rb
Instance Attribute Summary collapse
-
#authority ⇒ Object
readonly
Returns the value of attribute authority.
-
#endpoint ⇒ Object
readonly
Returns the value of attribute endpoint.
-
#pool ⇒ Object
readonly
Returns the value of attribute pool.
-
#protocol ⇒ Object
readonly
Returns the value of attribute protocol.
-
#retries ⇒ Object
readonly
Returns the value of attribute retries.
-
#scheme ⇒ Object
readonly
Returns the value of attribute scheme.
Class Method Summary collapse
Instance Method Summary collapse
- #as_json ⇒ Object
- #call(request) ⇒ Object
- #close ⇒ Object
-
#initialize(endpoint, protocol: endpoint.protocol, scheme: endpoint.scheme, authority: endpoint.authority, retries: DEFAULT_RETRIES, connection_limit: DEFAULT_CONNECTION_LIMIT) ⇒ Client
constructor
Provides a robust interface to a server.
- #inspect ⇒ Object
- #secure? ⇒ Boolean
- #to_json ⇒ Object
Methods included from Proxy::Client
#proxied_client, #proxied_endpoint, #proxy
Constructor Details
#initialize(endpoint, protocol: endpoint.protocol, scheme: endpoint.scheme, authority: endpoint.authority, retries: DEFAULT_RETRIES, connection_limit: DEFAULT_CONNECTION_LIMIT) ⇒ Client
Provides a robust interface to a server.
-
If there are no connections, it will create one.
-
If there are already connections, it will reuse it.
-
If a request fails, it will retry it up to N times if it was idempotent.
The client object will never become unusable. It internally manages persistent connections (or non-persistent connections if that’s required).
33 34 35 36 37 38 39 40 41 42 |
# File 'lib/async/http/client.rb', line 33 def initialize(endpoint, protocol: endpoint.protocol, scheme: endpoint.scheme, authority: endpoint., retries: DEFAULT_RETRIES, connection_limit: DEFAULT_CONNECTION_LIMIT) @endpoint = endpoint @protocol = protocol @retries = retries @pool = make_pool(connection_limit) @scheme = scheme @authority = end |
Instance Attribute Details
#authority ⇒ Object (readonly)
Returns the value of attribute authority.
65 66 67 |
# File 'lib/async/http/client.rb', line 65 def @authority end |
#endpoint ⇒ Object (readonly)
Returns the value of attribute endpoint.
58 59 60 |
# File 'lib/async/http/client.rb', line 58 def endpoint @endpoint end |
#pool ⇒ Object (readonly)
Returns the value of attribute pool.
62 63 64 |
# File 'lib/async/http/client.rb', line 62 def pool @pool end |
#protocol ⇒ Object (readonly)
Returns the value of attribute protocol.
59 60 61 |
# File 'lib/async/http/client.rb', line 59 def protocol @protocol end |
#retries ⇒ Object (readonly)
Returns the value of attribute retries.
61 62 63 |
# File 'lib/async/http/client.rb', line 61 def retries @retries end |
#scheme ⇒ Object (readonly)
Returns the value of attribute scheme.
64 65 66 |
# File 'lib/async/http/client.rb', line 64 def scheme @scheme end |
Class Method Details
.open(*arguments, **options, &block) ⇒ Object
71 72 73 74 75 76 77 78 79 80 81 |
# File 'lib/async/http/client.rb', line 71 def self.open(*arguments, **, &block) client = self.new(*arguments, **) return client unless block_given? begin yield client ensure client.close end end |
Instance Method Details
#as_json ⇒ Object
44 45 46 47 48 49 50 51 52 |
# File 'lib/async/http/client.rb', line 44 def as_json(...) { endpoint: @endpoint.to_s, protocol: @protocol, retries: @retries, scheme: @scheme, authority: @authority, } end |
#call(request) ⇒ Object
92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 |
# File 'lib/async/http/client.rb', line 92 def call(request) request.scheme ||= self.scheme request. ||= self. attempt = 0 # We may retry the request if it is possible to do so. https://tools.ietf.org/html/draft-nottingham-httpbis-retry-01 is a good guide for how retrying requests should work. begin attempt += 1 # As we cache pool, it's possible these pool go bad (e.g. closed by remote host). In this case, we need to try again. It's up to the caller to impose a timeout on this. If this is the last attempt, we force a new connection. connection = @pool.acquire response = make_response(request, connection) # This signals that the ensure block below should not try to release the connection, because it's bound into the response which will be returned: connection = nil return response rescue Protocol::RequestFailed # This is a specific case where the entire request wasn't sent before a failure occurred. So, we can even resend non-idempotent requests. if connection @pool.release(connection) connection = nil end if attempt < @retries retry else raise end rescue SocketError, IOError, EOFError, Errno::ECONNRESET, Errno::EPIPE if connection @pool.release(connection) connection = nil end if request.idempotent? and attempt < @retries retry else raise end ensure if connection @pool.release(connection) end end end |
#close ⇒ Object
83 84 85 86 87 88 89 90 |
# File 'lib/async/http/client.rb', line 83 def close while @pool.busy? Console.logger.warn(self) {"Waiting for #{@protocol} pool to drain: #{@pool}"} @pool.wait end @pool.close end |
#inspect ⇒ Object
140 141 142 |
# File 'lib/async/http/client.rb', line 140 def inspect "#<#{self.class} authority=#{@authority.inspect}>" end |
#secure? ⇒ Boolean
67 68 69 |
# File 'lib/async/http/client.rb', line 67 def secure? @endpoint.secure? end |
#to_json ⇒ Object
54 55 56 |
# File 'lib/async/http/client.rb', line 54 def to_json(...) as_json.to_json(...) end |