Skip to content

Commit c37a5a4

Browse files
millsjustinmattcreasergpanshu
committed
fix(api): When a mutation response code indicates client error only call onFailure (#2255)
Co-authored-by: Matt Creaser <[email protected]> Co-authored-by: gpanshu <[email protected]>
1 parent 1cd1aca commit c37a5a4

File tree

2 files changed

+73
-0
lines changed

2 files changed

+73
-0
lines changed

aws-api/src/main/java/com/amplifyframework/api/aws/AppSyncGraphQLOperation.java

+1
Original file line numberDiff line numberDiff line change
@@ -147,6 +147,7 @@ public void onResponse(@NonNull Call call, @NonNull Response response) {
147147
onFailure.accept(new ApiException
148148
.NonRetryableException("OkHttp client request failed.", "Irrecoverable error")
149149
);
150+
return;
150151
}
151152

152153
try {

aws-api/src/test/java/com/amplifyframework/api/aws/AWSApiPluginTest.java

+72
Original file line numberDiff line numberDiff line change
@@ -36,6 +36,7 @@
3636
import com.amplifyframework.testmodels.commentsblog.BlogOwner;
3737
import com.amplifyframework.testutils.Await;
3838
import com.amplifyframework.testutils.HubAccumulator;
39+
import com.amplifyframework.testutils.Latch;
3940
import com.amplifyframework.testutils.Resources;
4041
import com.amplifyframework.testutils.random.RandomString;
4142
import com.amplifyframework.util.TypeMaker;
@@ -53,7 +54,9 @@
5354
import java.lang.reflect.Type;
5455
import java.util.Arrays;
5556
import java.util.Map;
57+
import java.util.concurrent.CountDownLatch;
5658
import java.util.concurrent.TimeUnit;
59+
import java.util.concurrent.atomic.AtomicReference;
5760

5861
import io.reactivex.rxjava3.core.Observable;
5962
import okhttp3.HttpUrl;
@@ -275,6 +278,75 @@ public void graphQlMutationGetsResponse() throws JSONException, ApiException {
275278
assertEquals(ApiEndpointStatusChangeEvent.ApiEndpointStatus.REACHABLE, eventData.getCurrentStatus());
276279
}
277280

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+
278350
/**
279351
* Given that only one API was configured in {@link #setup()},
280352
* the {@link AWSApiPlugin#getSelectedApiName(EndpointType)} should be able to identify

0 commit comments

Comments
 (0)