httpx 1.4.1 → 1.4.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -35,6 +35,7 @@ module HTTPX
35
35
  @_timeouts = Array(@resolver_options[:timeouts])
36
36
  @timeouts = Hash.new { |timeouts, host| timeouts[host] = @_timeouts.dup }
37
37
  @connections = []
38
+ @name = nil
38
39
  @queries = {}
39
40
  @read_buffer = "".b
40
41
  @write_buffer = Buffer.new(@resolver_options[:packet_size])
@@ -58,22 +59,6 @@ module HTTPX
58
59
  when :open
59
60
  consume
60
61
  end
61
- nil
62
- rescue Errno::EHOSTUNREACH => e
63
- @ns_index += 1
64
- nameserver = @nameserver
65
- if nameserver && @ns_index < nameserver.size
66
- log do
67
- "resolver #{FAMILY_TYPES[@record_type]}: " \
68
- "failed resolving on nameserver #{@nameserver[@ns_index - 1]} (#{e.message})"
69
- end
70
- transition(:idle)
71
- @timeouts.clear
72
- else
73
- handle_error(e)
74
- end
75
- rescue NativeResolveError => e
76
- handle_error(e)
77
62
  end
78
63
 
79
64
  def interests
@@ -108,9 +93,7 @@ module HTTPX
108
93
  @timeouts.values_at(*hosts).reject(&:empty?).map(&:first).min
109
94
  end
110
95
 
111
- def handle_socket_timeout(interval)
112
- do_retry(interval)
113
- end
96
+ def handle_socket_timeout(interval); end
114
97
 
115
98
  private
116
99
 
@@ -123,32 +106,60 @@ module HTTPX
123
106
  end
124
107
 
125
108
  def consume
126
- dread if calculate_interests == :r
127
- do_retry
128
- dwrite if calculate_interests == :w
109
+ loop do
110
+ dread if calculate_interests == :r
111
+
112
+ break unless calculate_interests == :w
113
+
114
+ # do_retry
115
+ dwrite
116
+
117
+ break unless calculate_interests == :r
118
+ end
119
+ rescue Errno::EHOSTUNREACH => e
120
+ @ns_index += 1
121
+ nameserver = @nameserver
122
+ if nameserver && @ns_index < nameserver.size
123
+ log do
124
+ "resolver #{FAMILY_TYPES[@record_type]}: " \
125
+ "failed resolving on nameserver #{@nameserver[@ns_index - 1]} (#{e.message})"
126
+ end
127
+ transition(:idle)
128
+ @timeouts.clear
129
+ retry
130
+ else
131
+ handle_error(e)
132
+ emit(:close, self)
133
+ end
134
+ rescue NativeResolveError => e
135
+ handle_error(e)
136
+ close_or_resolve
137
+ retry unless closed?
129
138
  end
130
139
 
131
- def do_retry(loop_time = nil)
132
- return if @queries.empty? || !@start_timeout
140
+ def schedule_retry
141
+ h = @name
133
142
 
134
- loop_time ||= Utils.elapsed_time(@start_timeout)
143
+ return unless h
135
144
 
136
- query = @queries.first
145
+ connection = @queries[h]
137
146
 
138
- return unless query
147
+ timeouts = @timeouts[h]
148
+ timeout = timeouts.shift
139
149
 
140
- h, connection = query
141
- host = connection.peer.host
142
- timeout = (@timeouts[host][0] -= loop_time)
150
+ @timer = @current_selector.after(timeout) do
151
+ next unless @connections.include?(connection)
143
152
 
144
- return unless timeout <= 0
153
+ do_retry(h, connection, timeout)
154
+ end
155
+ end
145
156
 
146
- elapsed_after = @_timeouts[@_timeouts.size - @timeouts[host].size]
147
- @timeouts[host].shift
157
+ def do_retry(h, connection, interval)
158
+ timeouts = @timeouts[h]
148
159
 
149
- if !@timeouts[host].empty?
160
+ if !timeouts.empty?
150
161
  log do
151
- "resolver #{FAMILY_TYPES[@record_type]}: timeout after #{elapsed_after}s, retry (with #{@timeouts[host].first}s) #{host}..."
162
+ "resolver #{FAMILY_TYPES[@record_type]}: timeout after #{interval}s, retry (with #{timeouts.first}s) #{h}..."
152
163
  end
153
164
  # must downgrade to tcp AND retry on same host as last
154
165
  downgrade_socket
@@ -157,22 +168,28 @@ module HTTPX
157
168
  # try on the next nameserver
158
169
  @ns_index += 1
159
170
  log do
160
- "resolver #{FAMILY_TYPES[@record_type]}: failed resolving #{host} on nameserver #{@nameserver[@ns_index - 1]} (timeout error)"
171
+ "resolver #{FAMILY_TYPES[@record_type]}: failed resolving #{h} on nameserver #{@nameserver[@ns_index - 1]} (timeout error)"
161
172
  end
162
173
  transition(:idle)
163
174
  @timeouts.clear
164
175
  resolve(connection, h)
165
176
  else
166
177
 
167
- @timeouts.delete(host)
178
+ @timeouts.delete(h)
168
179
  reset_hostname(h, reset_candidates: false)
169
180
 
170
- return unless @queries.empty?
181
+ unless @queries.empty?
182
+ resolve(connection)
183
+ return
184
+ end
171
185
 
172
186
  @connections.delete(connection)
187
+
188
+ host = connection.peer.host
189
+
173
190
  # This loop_time passed to the exception is bogus. Ideally we would pass the total
174
191
  # resolve timeout, including from the previous retries.
175
- ex = ResolveTimeoutError.new(loop_time, "Timed out while resolving #{connection.peer.host}")
192
+ ex = ResolveTimeoutError.new(interval, "Timed out while resolving #{host}")
176
193
  ex.set_backtrace(ex ? ex.backtrace : caller)
177
194
  emit_resolve_error(connection, host, ex)
178
195
 
@@ -225,7 +242,7 @@ module HTTPX
225
242
  parse(@read_buffer)
226
243
  end
227
244
 
228
- return if @state == :closed
245
+ return if @state == :closed || !@write_buffer.empty?
229
246
  end
230
247
  end
231
248
 
@@ -243,11 +260,15 @@ module HTTPX
243
260
 
244
261
  return unless siz.positive?
245
262
 
263
+ schedule_retry if @write_buffer.empty?
264
+
246
265
  return if @state == :closed
247
266
  end
248
267
  end
249
268
 
250
269
  def parse(buffer)
270
+ @timer.cancel
271
+
251
272
  code, result = Resolver.decode_dns_answer(buffer)
252
273
 
253
274
  case code
@@ -258,8 +279,10 @@ module HTTPX
258
279
  hostname, connection = @queries.first
259
280
  reset_hostname(hostname, reset_candidates: false)
260
281
 
261
- if @queries.value?(connection)
262
- resolve
282
+ other_candidate, _ = @queries.find { |_, conn| conn == connection }
283
+
284
+ if other_candidate
285
+ resolve(connection, other_candidate)
263
286
  else
264
287
  @connections.delete(connection)
265
288
  ex = NativeResolveError.new(connection, connection.peer.host, "name or service not known")
@@ -321,8 +344,10 @@ module HTTPX
321
344
  connection = @queries.delete(name)
322
345
  end
323
346
 
324
- if address.key?("alias") # CNAME
325
- hostname_alias = address["alias"]
347
+ alias_addresses, addresses = addresses.partition { |addr| addr.key?("alias") }
348
+
349
+ if addresses.empty? && !alias_addresses.empty? # CNAME
350
+ hostname_alias = alias_addresses.first["alias"]
326
351
  # clean up intermediate queries
327
352
  @timeouts.delete(name) unless connection.peer.host == name
328
353
 
@@ -350,7 +375,11 @@ module HTTPX
350
375
  close_or_resolve
351
376
  end
352
377
 
353
- def resolve(connection = @connections.first, hostname = nil)
378
+ def resolve(connection = nil, hostname = nil)
379
+ @connections.shift until @connections.empty? || @connections.first.state != :closed
380
+
381
+ connection ||= @connections.find { |c| !@queries.value?(c) }
382
+
354
383
  raise Error, "no URI to resolve" unless connection
355
384
 
356
385
  return unless @write_buffer.empty?
@@ -370,6 +399,9 @@ module HTTPX
370
399
  else
371
400
  @queries[hostname] = connection
372
401
  end
402
+
403
+ @name = hostname
404
+
373
405
  log { "resolver #{FAMILY_TYPES[@record_type]}: query for #{hostname}" }
374
406
  begin
375
407
  @write_buffer << encode_dns_query(hostname)
@@ -458,6 +490,7 @@ module HTTPX
458
490
  # these errors may happen during TCP handshake
459
491
  # treat them as resolve errors.
460
492
  handle_error(e)
493
+ emit(:close, self)
461
494
  end
462
495
 
463
496
  def handle_error(error)
@@ -472,13 +505,15 @@ module HTTPX
472
505
  @connections.delete(connection)
473
506
  emit_resolve_error(connection, host, error)
474
507
  end
508
+
509
+ while (connection = @connections.shift)
510
+ emit_resolve_error(connection, host, error)
511
+ end
475
512
  end
476
- close_or_resolve
477
513
  end
478
514
 
479
515
  def reset_hostname(hostname, connection: @queries.delete(hostname), reset_candidates: true)
480
516
  @timeouts.delete(hostname)
481
- @timeouts.delete(hostname)
482
517
 
483
518
  return unless connection && reset_candidates
484
519
 
@@ -490,7 +525,10 @@ module HTTPX
490
525
  end
491
526
 
492
527
  def close_or_resolve
493
- if @connections.empty?
528
+ # drop already closed connections
529
+ @connections.shift until @connections.empty? || @connections.first.state != :closed
530
+
531
+ if (@connections - @queries.values).empty?
494
532
  emit(:close, self)
495
533
  else
496
534
  resolve
@@ -74,14 +74,15 @@ module HTTPX
74
74
 
75
75
  log do
76
76
  "resolver #{FAMILY_TYPES[RECORD_TYPES[family]]}: " \
77
- "answer #{FAMILY_TYPES[RECORD_TYPES[family]]} #{connection.peer.host}: #{addresses.inspect}"
77
+ "answer #{connection.peer.host}: #{addresses.inspect} (early resolve: #{early_resolve})"
78
78
  end
79
79
 
80
- if @current_selector && # if triggered by early resolve, session may not be here yet
81
- !connection.io &&
82
- connection.options.ip_families.size > 1 &&
83
- family == Socket::AF_INET &&
84
- addresses.first.to_s != connection.peer.host.to_s
80
+ if !early_resolve && # do not apply resolution delay for non-dns name resolution
81
+ @current_selector && # just in case...
82
+ family == Socket::AF_INET && # resolution delay only applies to IPv4
83
+ !connection.io && # connection already has addresses and initiated/ended handshake
84
+ connection.options.ip_families.size > 1 && # no need to delay if not supporting dual stack IP
85
+ addresses.first.to_s != connection.peer.host.to_s # connection URL host is already the IP (early resolve included perhaps?)
85
86
  log { "resolver #{FAMILY_TYPES[RECORD_TYPES[family]]}: applying resolution delay..." }
86
87
 
87
88
  @current_selector.after(0.05) do
@@ -52,9 +52,6 @@ module HTTPX
52
52
  # copies the response body to a different location.
53
53
  def_delegator :@body, :copy_to
54
54
 
55
- # closes the body.
56
- def_delegator :@body, :close
57
-
58
55
  # the corresponding request uri.
59
56
  def_delegator :@request, :uri
60
57
 
@@ -74,6 +71,12 @@ module HTTPX
74
71
  @content_type = nil
75
72
  end
76
73
 
74
+ # closes the respective +@request+ and +@body+.
75
+ def close
76
+ @request.close
77
+ @body.close
78
+ end
79
+
77
80
  # merges headers defined in +h+ into the response headers.
78
81
  def merge_headers(h)
79
82
  @headers = @headers.merge(h)
@@ -264,7 +267,7 @@ module HTTPX
264
267
 
265
268
  # closes the error resources.
266
269
  def close
267
- @response.close if @response && @response.respond_to?(:close)
270
+ @response.close if @response
268
271
  end
269
272
 
270
273
  # always true for error responses.
@@ -279,6 +282,8 @@ module HTTPX
279
282
 
280
283
  # buffers lost chunks to error response
281
284
  def <<(data)
285
+ return unless @response
286
+
282
287
  @response << data
283
288
  end
284
289
  end
@@ -19,6 +19,7 @@ module HTTPX
19
19
  def initialize
20
20
  @timers = Timers.new
21
21
  @selectables = []
22
+ @is_timer_interval = false
22
23
  end
23
24
 
24
25
  def each(&blk)
@@ -43,7 +44,11 @@ module HTTPX
43
44
  rescue StandardError => e
44
45
  emit_error(e)
45
46
  rescue Exception # rubocop:disable Lint/RescueException
46
- each_connection(&:force_reset)
47
+ each_connection do |conn|
48
+ conn.force_reset
49
+ conn.disconnect
50
+ end
51
+
47
52
  raise
48
53
  end
49
54
 
@@ -125,24 +130,22 @@ module HTTPX
125
130
  # first, we group IOs based on interest type. On call to #interests however,
126
131
  # things might already happen, and new IOs might be registered, so we might
127
132
  # have to start all over again. We do this until we group all selectables
128
- begin
129
- @selectables.delete_if do |io|
130
- interests = io.interests
133
+ @selectables.delete_if do |io|
134
+ interests = io.interests
131
135
 
132
- (r ||= []) << io if READABLE.include?(interests)
133
- (w ||= []) << io if WRITABLE.include?(interests)
136
+ (r ||= []) << io if READABLE.include?(interests)
137
+ (w ||= []) << io if WRITABLE.include?(interests)
134
138
 
135
- io.state == :closed
136
- end
139
+ io.state == :closed
140
+ end
137
141
 
138
- # TODO: what to do if there are no selectables?
142
+ # TODO: what to do if there are no selectables?
139
143
 
140
- readers, writers = IO.select(r, w, nil, interval)
144
+ readers, writers = IO.select(r, w, nil, interval)
141
145
 
142
- if readers.nil? && writers.nil? && interval
143
- [*r, *w].each { |io| io.handle_socket_timeout(interval) }
144
- return
145
- end
146
+ if readers.nil? && writers.nil? && interval
147
+ [*r, *w].each { |io| io.handle_socket_timeout(interval) }
148
+ return
146
149
  end
147
150
 
148
151
  if writers
@@ -174,7 +177,7 @@ module HTTPX
174
177
  end
175
178
 
176
179
  unless result || interval.nil?
177
- io.handle_socket_timeout(interval)
180
+ io.handle_socket_timeout(interval) unless @is_timer_interval
178
181
  return
179
182
  end
180
183
  # raise TimeoutError.new(interval, "timed out while waiting on select")
@@ -186,10 +189,21 @@ module HTTPX
186
189
  end
187
190
 
188
191
  def next_timeout
189
- [
190
- @timers.wait_interval,
191
- @selectables.filter_map(&:timeout).min,
192
- ].compact.min
192
+ @is_timer_interval = false
193
+
194
+ timer_interval = @timers.wait_interval
195
+
196
+ connection_interval = @selectables.filter_map(&:timeout).min
197
+
198
+ return connection_interval unless timer_interval
199
+
200
+ if connection_interval.nil? || timer_interval <= connection_interval
201
+ @is_timer_interval = true
202
+
203
+ return timer_interval
204
+ end
205
+
206
+ connection_interval
193
207
  end
194
208
 
195
209
  def emit_error(e)
data/lib/httpx/session.rb CHANGED
@@ -240,11 +240,9 @@ module HTTPX
240
240
  end
241
241
  return unless error && error.is_a?(Exception)
242
242
 
243
- if error.is_a?(Error)
244
- request.emit(:response, ErrorResponse.new(request, error))
245
- else
246
- raise error if selector.empty?
247
- end
243
+ raise error unless error.is_a?(Error)
244
+
245
+ request.emit(:response, ErrorResponse.new(request, error))
248
246
  end
249
247
 
250
248
  # returns a set of HTTPX::Request objects built from the given +args+ and +options+.
data/lib/httpx/timers.rb CHANGED
@@ -26,7 +26,7 @@ module HTTPX
26
26
 
27
27
  @next_interval_at = nil
28
28
 
29
- interval
29
+ Timer.new(interval, callback)
30
30
  end
31
31
 
32
32
  def wait_interval
@@ -48,6 +48,17 @@ module HTTPX
48
48
  @next_interval_at = nil if @intervals.empty?
49
49
  end
50
50
 
51
+ class Timer
52
+ def initialize(interval, callback)
53
+ @interval = interval
54
+ @callback = callback
55
+ end
56
+
57
+ def cancel
58
+ @interval.delete(@callback)
59
+ end
60
+ end
61
+
51
62
  class Interval
52
63
  include Comparable
53
64
 
@@ -63,6 +74,10 @@ module HTTPX
63
74
  @on_empty = blk
64
75
  end
65
76
 
77
+ def cancel
78
+ @on_empty.call
79
+ end
80
+
66
81
  def <=>(other)
67
82
  @interval <=> other.interval
68
83
  end
@@ -11,6 +11,7 @@ module HTTPX
11
11
  @buffer = "".b
12
12
 
13
13
  @form = form
14
+ @bytesize = 0
14
15
  @parts = to_parts(form)
15
16
  end
16
17
 
@@ -42,6 +43,7 @@ module HTTPX
42
43
  aux << [key, val]
43
44
  end
44
45
  @form = form
46
+ @bytesize = 0
45
47
  @parts = to_parts(form)
46
48
  @part_index = 0
47
49
  end
@@ -49,7 +51,6 @@ module HTTPX
49
51
  private
50
52
 
51
53
  def to_parts(form)
52
- @bytesize = 0
53
54
  params = form.each_with_object([]) do |(key, val), aux|
54
55
  Transcoder.normalize_keys(key, val, MULTIPART_VALUE_COND) do |k, v|
55
56
  next if v.nil?
data/lib/httpx/version.rb CHANGED
@@ -1,5 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module HTTPX
4
- VERSION = "1.4.1"
4
+ VERSION = "1.4.3"
5
5
  end
data/sig/callbacks.rbs CHANGED
@@ -4,8 +4,8 @@ module HTTPX
4
4
  end
5
5
 
6
6
  module Callbacks
7
- def on: (Symbol) { (*untyped) -> void } -> self
8
- def once: (Symbol) { (*untyped) -> void } -> self
7
+ def on: (Symbol) { (*untyped) -> void } -> ^(*untyped) -> void
8
+ def once: (Symbol) { (*untyped) -> void } -> ^(*untyped) -> void
9
9
  def emit: (Symbol, *untyped) -> void
10
10
 
11
11
  def callbacks_for?: (Symbol) -> bool
@@ -8,6 +8,7 @@ module HTTPX
8
8
  attr_reader streams: Hash[Request, ::HTTP2::Stream]
9
9
  attr_reader pending: Array[Request]
10
10
 
11
+ @connection: HTTP2::Client
11
12
  @options: Options
12
13
  @settings: Hash[Symbol, Integer | bool]
13
14
  @max_concurrent_requests: Integer
@@ -95,5 +96,8 @@ module HTTPX
95
96
 
96
97
  class GoawayError < Error
97
98
  end
99
+
100
+ class PingError < Error
101
+ end
98
102
  end
99
103
  end
data/sig/connection.rbs CHANGED
@@ -43,7 +43,6 @@ module HTTPX
43
43
  @parser: Object & _Parser
44
44
  @connected_at: Float
45
45
  @response_received_at: Float
46
- @intervals: Array[Timers::Interval]
47
46
  @exhausted: bool
48
47
  @cloned: bool
49
48
  @coalesced_connection: instance?
@@ -111,6 +110,8 @@ module HTTPX
111
110
 
112
111
  def handle_connect_error: (StandardError error) -> void
113
112
 
113
+ def disconnect: () -> void
114
+
114
115
  private
115
116
 
116
117
  def initialize: (http_uri uri, Options options) -> void
@@ -119,8 +120,6 @@ module HTTPX
119
120
 
120
121
  def connect: () -> void
121
122
 
122
- def disconnect: () -> void
123
-
124
123
  def exhausted?: () -> boolish
125
124
 
126
125
  def consume: () -> void
@@ -163,7 +162,7 @@ module HTTPX
163
162
 
164
163
  def read_timeout_callback: (Request request, Numeric read_timeout, ?singleton(RequestTimeoutError) error_type) -> void
165
164
 
166
- def set_request_timeout: (Request request, Numeric timeout, Symbol start_event, Symbol | Array[Symbol] finish_events) { () -> void } -> void
165
+ def set_request_timeout: (Symbol label, Request request, Numeric timeout, Symbol start_event, Symbol | Array[Symbol] finish_events) { () -> void } -> void
167
166
 
168
167
  def self.parser_type: (String protocol) -> (singleton(HTTP1) | singleton(HTTP2))
169
168
  end
data/sig/errors.rbs CHANGED
@@ -45,6 +45,9 @@ module HTTPX
45
45
  class WriteTimeoutError < RequestTimeoutError
46
46
  end
47
47
 
48
+ class OperationTimeoutError < TimeoutError
49
+ end
50
+
48
51
  class ResolveError < Error
49
52
  end
50
53
 
@@ -64,7 +67,4 @@ module HTTPX
64
67
 
65
68
  def initialize: (Connection connection, String hostname, ?String message) -> untyped
66
69
  end
67
-
68
- class MisdirectedRequestError < HTTPError
69
- end
70
70
  end
data/sig/loggable.rbs CHANGED
@@ -8,8 +8,8 @@ module HTTPX
8
8
 
9
9
  COLORS: Hash[Symbol, Integer]
10
10
 
11
- def log: (?level: Integer?, ?color: Symbol?) { () -> String } -> void
11
+ def log: (?level: Integer?, ?color: Symbol?, ?debug_level: Integer, ?debug: _IOLogger?) { () -> String } -> void
12
12
 
13
- def log_exception: (Exception error, ?level: Integer, ?color: Symbol) -> void
13
+ def log_exception: (Exception error, ?level: Integer, ?color: Symbol, ?debug_level: Integer, ?debug: _IOLogger?) -> void
14
14
  end
15
15
  end
@@ -0,0 +1,18 @@
1
+ module HTTPX
2
+ module Plugins
3
+ module Query
4
+ def self.subplugins: () -> Hash[Symbol, Module]
5
+
6
+ module InstanceMethods
7
+ def query: (uri | [uri], **untyped) -> response
8
+ | (_Each[uri | [uri, request_params]], **untyped) -> Array[response]
9
+ end
10
+
11
+ module QueryRetries
12
+ module InstanceMethods
13
+ end
14
+ end
15
+ end
16
+ type sessionQuery = Session & Query::InstanceMethods
17
+ end
18
+ end
data/sig/pool.rbs CHANGED
@@ -5,6 +5,8 @@ module HTTPX
5
5
  }
6
6
 
7
7
  class Pool
8
+ POOL_TIMEOUT: Integer
9
+
8
10
  type resolver_manager = Resolver::Multi | Resolver::System
9
11
 
10
12
  @max_connections_per_origin: Integer
data/sig/request.rbs CHANGED
@@ -14,6 +14,7 @@ module HTTPX
14
14
  attr_reader options: Options
15
15
  attr_reader response: response?
16
16
  attr_reader drain_error: StandardError?
17
+ attr_reader active_timeouts: Array[Symbol]
17
18
 
18
19
  attr_accessor peer_address: ipaddr?
19
20
 
@@ -27,6 +28,10 @@ module HTTPX
27
28
 
28
29
  def initialize: (Symbol | String verb, generic_uri uri, Options options, ?request_params params) -> untyped
29
30
 
31
+ def empty?: () -> bool
32
+
33
+ def close: () -> void
34
+
30
35
  def interests: () -> (:r | :w)
31
36
 
32
37
  def merge_headers: (_Each[[String, headers_value]]) -> void
@@ -63,6 +68,8 @@ module HTTPX
63
68
 
64
69
  def request_timeout: () -> Numeric?
65
70
 
71
+ def set_timeout_callback: (Symbol event) { (*untyped) -> void } -> void
72
+
66
73
  private
67
74
 
68
75
  def initialize_body: (Options options) -> Transcoder::_Encoder?
@@ -21,6 +21,7 @@ module HTTPX
21
21
  @write_buffer: Buffer
22
22
  @large_packet: Buffer?
23
23
  @io: UDP | TCP
24
+ @name: String?
24
25
 
25
26
  attr_reader state: Symbol
26
27
 
@@ -42,7 +43,9 @@ module HTTPX
42
43
 
43
44
  def consume: () -> void
44
45
 
45
- def do_retry: (?Numeric? loop_time) -> void
46
+ def schedule_retry: () -> void
47
+
48
+ def do_retry: (String host, Connection connection, Numeric interval) -> void
46
49
 
47
50
  def dread: (Integer) -> void
48
51
  | () -> void