14
14
use std:: collections:: HashMap ;
15
15
use std:: convert:: TryFrom ;
16
16
use std:: str:: FromStr ;
17
+ use std:: thread;
17
18
18
19
#[ allow( unused_imports) ]
19
20
use log:: { debug, error, info, trace} ;
20
21
21
- use minreq:: { Proxy , Request } ;
22
+ use minreq:: { Proxy , Request , Response } ;
22
23
23
24
use bitcoin:: consensus:: { deserialize, serialize, Decodable } ;
24
25
use bitcoin:: hashes:: { sha256, Hash } ;
@@ -27,7 +28,10 @@ use bitcoin::{
27
28
block:: Header as BlockHeader , Block , BlockHash , MerkleBlock , Script , Transaction , Txid ,
28
29
} ;
29
30
30
- use crate :: { BlockStatus , BlockSummary , Builder , Error , MerkleProof , OutputStatus , Tx , TxStatus } ;
31
+ use crate :: {
32
+ BlockStatus , BlockSummary , Builder , Error , MerkleProof , OutputStatus , Tx , TxStatus ,
33
+ BASE_BACKOFF_MILLIS , RETRYABLE_ERROR_CODES ,
34
+ } ;
31
35
32
36
#[ derive( Debug , Clone ) ]
33
37
pub struct BlockingClient {
@@ -39,6 +43,8 @@ pub struct BlockingClient {
39
43
pub timeout : Option < u64 > ,
40
44
/// HTTP headers to set on every request made to Esplora server
41
45
pub headers : HashMap < String , String > ,
46
+ /// Number of times to retry a request
47
+ pub max_retries : usize ,
42
48
}
43
49
44
50
impl BlockingClient {
@@ -49,6 +55,7 @@ impl BlockingClient {
49
55
proxy : builder. proxy ,
50
56
timeout : builder. timeout ,
51
57
headers : builder. headers ,
58
+ max_retries : builder. max_retries ,
52
59
}
53
60
}
54
61
@@ -80,20 +87,20 @@ impl BlockingClient {
80
87
}
81
88
82
89
fn get_opt_response < T : Decodable > ( & self , path : & str ) -> Result < Option < T > , Error > {
83
- match self . get_request ( path) ? . send ( ) {
90
+ match self . get_with_retry ( path) {
84
91
Ok ( resp) if is_status_not_found ( resp. status_code ) => Ok ( None ) ,
85
92
Ok ( resp) if !is_status_ok ( resp. status_code ) => {
86
93
let status = u16:: try_from ( resp. status_code ) . map_err ( Error :: StatusCode ) ?;
87
94
let message = resp. as_str ( ) . unwrap_or_default ( ) . to_string ( ) ;
88
95
Err ( Error :: HttpResponse { status, message } )
89
96
}
90
97
Ok ( resp) => Ok ( Some ( deserialize :: < T > ( resp. as_bytes ( ) ) ?) ) ,
91
- Err ( e) => Err ( Error :: Minreq ( e ) ) ,
98
+ Err ( e) => Err ( e ) ,
92
99
}
93
100
}
94
101
95
102
fn get_opt_response_txid ( & self , path : & str ) -> Result < Option < Txid > , Error > {
96
- match self . get_request ( path) ? . send ( ) {
103
+ match self . get_with_retry ( path) {
97
104
Ok ( resp) if is_status_not_found ( resp. status_code ) => Ok ( None ) ,
98
105
Ok ( resp) if !is_status_ok ( resp. status_code ) => {
99
106
let status = u16:: try_from ( resp. status_code ) . map_err ( Error :: StatusCode ) ?;
@@ -103,12 +110,12 @@ impl BlockingClient {
103
110
Ok ( resp) => Ok ( Some (
104
111
Txid :: from_str ( resp. as_str ( ) . map_err ( Error :: Minreq ) ?) . map_err ( Error :: HexToArray ) ?,
105
112
) ) ,
106
- Err ( e) => Err ( Error :: Minreq ( e ) ) ,
113
+ Err ( e) => Err ( e ) ,
107
114
}
108
115
}
109
116
110
117
fn get_opt_response_hex < T : Decodable > ( & self , path : & str ) -> Result < Option < T > , Error > {
111
- match self . get_request ( path) ? . send ( ) {
118
+ match self . get_with_retry ( path) {
112
119
Ok ( resp) if is_status_not_found ( resp. status_code ) => Ok ( None ) ,
113
120
Ok ( resp) if !is_status_ok ( resp. status_code ) => {
114
121
let status = u16:: try_from ( resp. status_code ) . map_err ( Error :: StatusCode ) ?;
@@ -122,12 +129,12 @@ impl BlockingClient {
122
129
. map_err ( Error :: BitcoinEncoding )
123
130
. map ( |r| Some ( r) )
124
131
}
125
- Err ( e) => Err ( Error :: Minreq ( e ) ) ,
132
+ Err ( e) => Err ( e ) ,
126
133
}
127
134
}
128
135
129
136
fn get_response_hex < T : Decodable > ( & self , path : & str ) -> Result < T , Error > {
130
- match self . get_request ( path) ? . send ( ) {
137
+ match self . get_with_retry ( path) {
131
138
Ok ( resp) if !is_status_ok ( resp. status_code ) => {
132
139
let status = u16:: try_from ( resp. status_code ) . map_err ( Error :: StatusCode ) ?;
133
140
let message = resp. as_str ( ) . unwrap_or_default ( ) . to_string ( ) ;
@@ -138,51 +145,51 @@ impl BlockingClient {
138
145
let hex_vec = Vec :: from_hex ( hex_str) . unwrap ( ) ;
139
146
deserialize :: < T > ( & hex_vec) . map_err ( Error :: BitcoinEncoding )
140
147
}
141
- Err ( e) => Err ( Error :: Minreq ( e ) ) ,
148
+ Err ( e) => Err ( e ) ,
142
149
}
143
150
}
144
151
145
152
fn get_response_json < ' a , T : serde:: de:: DeserializeOwned > (
146
153
& ' a self ,
147
154
path : & ' a str ,
148
155
) -> Result < T , Error > {
149
- let response = self . get_request ( path) ? . send ( ) ;
156
+ let response = self . get_with_retry ( path) ;
150
157
match response {
151
158
Ok ( resp) if !is_status_ok ( resp. status_code ) => {
152
159
let status = u16:: try_from ( resp. status_code ) . map_err ( Error :: StatusCode ) ?;
153
160
let message = resp. as_str ( ) . unwrap_or_default ( ) . to_string ( ) ;
154
161
Err ( Error :: HttpResponse { status, message } )
155
162
}
156
163
Ok ( resp) => Ok ( resp. json :: < T > ( ) . map_err ( Error :: Minreq ) ?) ,
157
- Err ( e) => Err ( Error :: Minreq ( e ) ) ,
164
+ Err ( e) => Err ( e ) ,
158
165
}
159
166
}
160
167
161
168
fn get_opt_response_json < T : serde:: de:: DeserializeOwned > (
162
169
& self ,
163
170
path : & str ,
164
171
) -> Result < Option < T > , Error > {
165
- match self . get_request ( path) ? . send ( ) {
172
+ match self . get_with_retry ( path) {
166
173
Ok ( resp) if is_status_not_found ( resp. status_code ) => Ok ( None ) ,
167
174
Ok ( resp) if !is_status_ok ( resp. status_code ) => {
168
175
let status = u16:: try_from ( resp. status_code ) . map_err ( Error :: StatusCode ) ?;
169
176
let message = resp. as_str ( ) . unwrap_or_default ( ) . to_string ( ) ;
170
177
Err ( Error :: HttpResponse { status, message } )
171
178
}
172
179
Ok ( resp) => Ok ( Some ( resp. json :: < T > ( ) ?) ) ,
173
- Err ( e) => Err ( Error :: Minreq ( e ) ) ,
180
+ Err ( e) => Err ( e ) ,
174
181
}
175
182
}
176
183
177
184
fn get_response_str ( & self , path : & str ) -> Result < String , Error > {
178
- match self . get_request ( path) ? . send ( ) {
185
+ match self . get_with_retry ( path) {
179
186
Ok ( resp) if !is_status_ok ( resp. status_code ) => {
180
187
let status = u16:: try_from ( resp. status_code ) . map_err ( Error :: StatusCode ) ?;
181
188
let message = resp. as_str ( ) . unwrap_or_default ( ) . to_string ( ) ;
182
189
Err ( Error :: HttpResponse { status, message } )
183
190
}
184
191
Ok ( resp) => Ok ( resp. as_str ( ) ?. to_string ( ) ) ,
185
- Err ( e) => Err ( Error :: Minreq ( e ) ) ,
192
+ Err ( e) => Err ( e ) ,
186
193
}
187
194
}
188
195
@@ -339,6 +346,24 @@ impl BlockingClient {
339
346
} ;
340
347
self . get_response_json ( & path)
341
348
}
349
+
350
+ /// Sends a GET request to the given `url`, retrying failed attempts
351
+ /// for retryable error codes until max retries hit.
352
+ pub fn get_with_retry ( & self , url : & str ) -> Result < Response , Error > {
353
+ let mut delay = BASE_BACKOFF_MILLIS ;
354
+ let mut attempts = 0 ;
355
+
356
+ loop {
357
+ match self . get_request ( url) ?. send ( ) ? {
358
+ resp if attempts < self . max_retries && is_status_retryable ( resp. status_code ) => {
359
+ thread:: sleep ( delay) ;
360
+ attempts += 1 ;
361
+ delay *= 2 ;
362
+ }
363
+ resp => return Ok ( resp) ,
364
+ }
365
+ }
366
+ }
342
367
}
343
368
344
369
fn is_status_ok ( status : i32 ) -> bool {
@@ -348,3 +373,8 @@ fn is_status_ok(status: i32) -> bool {
348
373
fn is_status_not_found ( status : i32 ) -> bool {
349
374
status == 404
350
375
}
376
+
377
+ fn is_status_retryable ( status : i32 ) -> bool {
378
+ let status = status as u16 ;
379
+ RETRYABLE_ERROR_CODES . contains ( & status)
380
+ }
0 commit comments