16
16
*/
17
17
package com .optimizely .ab ;
18
18
19
+ import org .apache .http .HttpException ;
20
+ import org .apache .http .client .HttpRequestRetryHandler ;
19
21
import org .apache .http .client .ResponseHandler ;
22
+ import org .apache .http .client .methods .CloseableHttpResponse ;
20
23
import org .apache .http .client .methods .HttpGet ;
21
24
import org .apache .http .client .methods .HttpUriRequest ;
22
25
import org .apache .http .client .methods .RequestBuilder ;
23
26
import org .apache .http .conn .HttpHostConnectException ;
24
27
import org .apache .http .impl .client .CloseableHttpClient ;
25
- import org .junit .After ;
26
- import org .junit .Before ;
27
- import org .junit .Test ;
28
+ import org .apache .http .impl .client .DefaultHttpRequestRetryHandler ;
29
+ import org .apache .http .protocol .HttpContext ;
30
+ import org .junit .*;
31
+ import org .mockserver .integration .ClientAndServer ;
32
+ import org .mockserver .model .ConnectionOptions ;
33
+ import org .mockserver .model .HttpError ;
34
+ import org .mockserver .model .HttpRequest ;
35
+ import org .mockserver .model .HttpResponse ;
28
36
29
37
import java .io .IOException ;
38
+ import java .util .concurrent .ExecutionException ;
30
39
import java .util .concurrent .TimeUnit ;
31
40
32
41
import static com .optimizely .ab .OptimizelyHttpClient .builder ;
33
42
import static java .util .concurrent .TimeUnit .*;
34
43
import static org .junit .Assert .*;
35
- import static org .mockito .Mockito .mock ;
36
- import static org .mockito .Mockito .when ;
44
+ import static org .mockito .Mockito .*;
45
+ import static org .mockserver .model .HttpForward .forward ;
46
+ import static org .mockserver .model .HttpRequest .request ;
47
+ import static org .mockserver .model .HttpResponse .*;
48
+ import static org .mockserver .model .HttpResponse .response ;
49
+ import static org .mockserver .verify .VerificationTimes .exactly ;
37
50
38
51
public class OptimizelyHttpClientTest {
39
-
40
52
@ Before
41
53
public void setUp () {
42
54
System .setProperty ("https.proxyHost" , "localhost" );
@@ -51,7 +63,13 @@ public void tearDown() {
51
63
52
64
@ Test
53
65
public void testDefaultConfiguration () {
54
- OptimizelyHttpClient optimizelyHttpClient = builder ().build ();
66
+ OptimizelyHttpClient .Builder builder = builder ();
67
+ assertEquals (builder .validateAfterInactivity , 1000 );
68
+ assertEquals (builder .maxTotalConnections , 200 );
69
+ assertEquals (builder .maxPerRoute , 20 );
70
+ assertNull (builder .customRetryHandler );
71
+
72
+ OptimizelyHttpClient optimizelyHttpClient = builder .build ();
55
73
assertTrue (optimizelyHttpClient .getHttpClient () instanceof CloseableHttpClient );
56
74
}
57
75
@@ -101,4 +119,74 @@ public void testExecute() throws IOException {
101
119
OptimizelyHttpClient optimizelyHttpClient = new OptimizelyHttpClient (mockHttpClient );
102
120
assertTrue (optimizelyHttpClient .execute (httpUriRequest , responseHandler ));
103
121
}
122
+
123
+ @ Test
124
+ public void testRetriesWithCustomRetryHandler () throws IOException {
125
+
126
+ // [NOTE] Request retries are all handled inside HttpClient. Not easy for unit test.
127
+ // - "DefaultHttpRetryHandler" in HttpClient retries only with special types of Exceptions
128
+ // like "NoHttpResponseException", etc.
129
+ // Other exceptions (SocketTimeout, ProtocolException, etc.) all ignored.
130
+ // - Not easy to force the specific exception type in the low-level.
131
+ // - This test just validates custom retry handler injected ok by validating the number of retries.
132
+
133
+ class CustomRetryHandler implements HttpRequestRetryHandler {
134
+ private final int maxRetries ;
135
+
136
+ public CustomRetryHandler (int maxRetries ) {
137
+ this .maxRetries = maxRetries ;
138
+ }
139
+
140
+ @ Override
141
+ public boolean retryRequest (IOException exception , int executionCount , HttpContext context ) {
142
+ // override to retry for any type of exceptions
143
+ return executionCount < maxRetries ;
144
+ }
145
+ }
146
+
147
+ int port = 9999 ;
148
+ ClientAndServer mockServer ;
149
+ int retryCount ;
150
+
151
+ // default httpclient (retries enabled by default, but no retry for timeout connection)
152
+
153
+ mockServer = ClientAndServer .startClientAndServer (port );
154
+ mockServer
155
+ .when (request ().withMethod ("GET" ).withPath ("/" ))
156
+ .error (HttpError .error ());
157
+
158
+ OptimizelyHttpClient clientDefault = OptimizelyHttpClient .builder ()
159
+ .setTimeoutMillis (100 )
160
+ .build ();
161
+
162
+ try {
163
+ clientDefault .execute (new HttpGet ("http://localhost:" + port ));
164
+ fail ();
165
+ } catch (Exception e ) {
166
+ retryCount = mockServer .retrieveRecordedRequests (request ()).length ;
167
+ assertEquals (1 , retryCount );
168
+ }
169
+ mockServer .stop ();
170
+
171
+ // httpclient with custom retry handler (5 times retries for any request)
172
+
173
+ mockServer = ClientAndServer .startClientAndServer (port );
174
+ mockServer
175
+ .when (request ().withMethod ("GET" ).withPath ("/" ))
176
+ .error (HttpError .error ());
177
+
178
+ OptimizelyHttpClient clientWithRetries = OptimizelyHttpClient .builder ()
179
+ .withRetryHandler (new CustomRetryHandler (5 ))
180
+ .setTimeoutMillis (100 )
181
+ .build ();
182
+
183
+ try {
184
+ clientWithRetries .execute (new HttpGet ("http://localhost:" + port ));
185
+ fail ();
186
+ } catch (Exception e ) {
187
+ retryCount = mockServer .retrieveRecordedRequests (request ()).length ;
188
+ assertEquals (5 , retryCount );
189
+ }
190
+ mockServer .stop ();
191
+ }
104
192
}
0 commit comments