|
36 | 36 | import com.amplifyframework.testmodels.commentsblog.BlogOwner;
|
37 | 37 | import com.amplifyframework.testutils.Await;
|
38 | 38 | import com.amplifyframework.testutils.HubAccumulator;
|
| 39 | +import com.amplifyframework.testutils.Latch; |
39 | 40 | import com.amplifyframework.testutils.Resources;
|
40 | 41 | import com.amplifyframework.testutils.random.RandomString;
|
41 | 42 | import com.amplifyframework.util.TypeMaker;
|
|
53 | 54 | import java.lang.reflect.Type;
|
54 | 55 | import java.util.Arrays;
|
55 | 56 | import java.util.Map;
|
| 57 | +import java.util.concurrent.CountDownLatch; |
56 | 58 | import java.util.concurrent.TimeUnit;
|
| 59 | +import java.util.concurrent.atomic.AtomicReference; |
57 | 60 |
|
58 | 61 | import io.reactivex.rxjava3.core.Observable;
|
59 | 62 | import okhttp3.HttpUrl;
|
@@ -275,6 +278,75 @@ public void graphQlMutationGetsResponse() throws JSONException, ApiException {
|
275 | 278 | assertEquals(ApiEndpointStatusChangeEvent.ApiEndpointStatus.REACHABLE, eventData.getCurrentStatus());
|
276 | 279 | }
|
277 | 280 |
|
| 281 | + /** |
| 282 | + * When the server returns a client error code in response to calling |
| 283 | + * {@link AWSApiPlugin#mutate(GraphQLRequest, Consumer, Consumer)}. |
| 284 | + * only the onFailure callback should be called. |
| 285 | + * |
| 286 | + * @throws InterruptedException if the thread performing the mutation is interrupted. |
| 287 | + */ |
| 288 | + @Test |
| 289 | + public void graphQlMutationWithClientErrorResponseCodeShouldNotCallOnResponse() throws InterruptedException { |
| 290 | + final int CLIENT_ERROR_CODE = 400; |
| 291 | + webServer.enqueue(new MockResponse().setResponseCode(CLIENT_ERROR_CODE)); |
| 292 | + |
| 293 | + final CountDownLatch latch = new CountDownLatch(2); |
| 294 | + final AtomicReference<Throwable> unexpectedErrorContainer = new AtomicReference<>(); |
| 295 | + final AtomicReference<GraphQLResponse<BlogOwner>> responseContainer = new AtomicReference<>(); |
| 296 | + final AtomicReference<ApiException> failureContainer = new AtomicReference<>(); |
| 297 | + |
| 298 | + final Consumer<GraphQLResponse<BlogOwner>> onResponse = (response) -> { |
| 299 | + responseContainer.set(response); |
| 300 | + latch.countDown(); |
| 301 | + }; |
| 302 | + final Consumer<ApiException> onFailure = (failure) -> { |
| 303 | + failureContainer.set(failure); |
| 304 | + latch.countDown(); |
| 305 | + }; |
| 306 | + |
| 307 | + final Thread thread = new Thread(() -> { |
| 308 | + try { |
| 309 | + // Try to perform a mutation. |
| 310 | + final BlogOwner tony = BlogOwner.builder() |
| 311 | + .name(RandomString.string()) |
| 312 | + .build(); |
| 313 | + plugin.mutate(ModelMutation.create(tony), onResponse, onFailure); |
| 314 | + } catch (Throwable unexpectedFailure) { |
| 315 | + unexpectedErrorContainer.set(unexpectedFailure); |
| 316 | + do { |
| 317 | + latch.countDown(); |
| 318 | + } while (latch.getCount() > 0); |
| 319 | + } |
| 320 | + }); |
| 321 | + thread.setDaemon(true); |
| 322 | + thread.start(); |
| 323 | + |
| 324 | + try { |
| 325 | + Latch.await(latch); |
| 326 | + thread.join(); |
| 327 | + } catch (RuntimeException runtimeException) { |
| 328 | + // We expect a RuntimeException here. |
| 329 | + // That means awaiting the latch timed out and both callbacks were not called. |
| 330 | + } |
| 331 | + |
| 332 | + assertNull( |
| 333 | + "An unexpected error occurred while performing the mutation.", |
| 334 | + unexpectedErrorContainer.get() |
| 335 | + ); |
| 336 | + assertTrue( |
| 337 | + "Latch count == 0; both onFailure and onResponse were called.", |
| 338 | + latch.getCount() > 0 |
| 339 | + ); |
| 340 | + assertTrue( |
| 341 | + "Expected client error response code to result in NonRetryableException.", |
| 342 | + failureContainer.get() instanceof ApiException.NonRetryableException |
| 343 | + ); |
| 344 | + assertNull( |
| 345 | + "There was a non-null response but the response code indicates client error.", |
| 346 | + responseContainer.get() |
| 347 | + ); |
| 348 | + } |
| 349 | + |
278 | 350 | /**
|
279 | 351 | * Given that only one API was configured in {@link #setup()},
|
280 | 352 | * the {@link AWSApiPlugin#getSelectedApiName(EndpointType)} should be able to identify
|
|
0 commit comments