From e9cad1d240af2bb3198bf1a32e8a4b5ecce43d1b Mon Sep 17 00:00:00 2001 From: Abhash Kumar Singh Date: Mon, 22 Jan 2024 14:10:05 -0800 Subject: [PATCH] fix(datastore): fix syncQuery for partial success-error responses and update retry mechanism for error codes (#3479) * fix(DataStore): retry on session expired requests * fix(datastore): Adding secureConnectionFailed as a retryable error (#3475) * fix(datastore): decode optimistically with paginated list response type (#3474) * fix(datastore): decode optimistically with paginated list response type * add unit test cases * resolve comments * Update PaginatedListTests.swift * fix(DataStore): retry on session expired requests (#3477) * fix(DataStore): retry on session expired requests * add test --------- Co-authored-by: Michael Law <1365977+lawmicha@users.noreply.github.com> Co-authored-by: Harsh <6162866+harsh62@users.noreply.github.com> Co-authored-by: Di Wu --- .../API/Response/GraphQLError.swift | 2 + .../AWSPluginsCore/Sync/PaginatedList.swift | 31 ++++++++++++ .../Sync/PaginatedListTests.swift | 50 +++++++++++++++++++ .../InitialSync/InitialSyncOperation.swift | 10 +++- .../SyncMutationToCloudOperation.swift | 14 +++++- .../Sync/RequestRetryablePolicy.swift | 4 +- .../Sync/RequestRetryablePolicyTests.swift | 22 ++++++++ 7 files changed, 128 insertions(+), 5 deletions(-) diff --git a/Amplify/Categories/API/Response/GraphQLError.swift b/Amplify/Categories/API/Response/GraphQLError.swift index 2afe5e8c63..7d1d21f104 100644 --- a/Amplify/Categories/API/Response/GraphQLError.swift +++ b/Amplify/Categories/API/Response/GraphQLError.swift @@ -44,3 +44,5 @@ extension GraphQLError { public let column: Int } } + +extension GraphQLError: Error { } diff --git a/AmplifyPlugins/Core/AWSPluginsCore/Sync/PaginatedList.swift b/AmplifyPlugins/Core/AWSPluginsCore/Sync/PaginatedList.swift index b77a11cbb9..1458988802 100644 --- a/AmplifyPlugins/Core/AWSPluginsCore/Sync/PaginatedList.swift +++ b/AmplifyPlugins/Core/AWSPluginsCore/Sync/PaginatedList.swift @@ -12,4 +12,35 @@ public struct PaginatedList: Decodable { public let items: [MutationSync] public let nextToken: String? public let startedAt: Int64? + + enum CodingKeys: CodingKey { + case items + case nextToken + case startedAt + } + + public init(items: [MutationSync], nextToken: String?, startedAt: Int64?) { + self.items = items + self.nextToken = nextToken + self.startedAt = startedAt + } + + public init(from decoder: Decoder) throws { + let values = try decoder.container(keyedBy: CodingKeys.self) + let optimisticDecodedResults = try values.decode([OptimisticDecoded>].self, forKey: .items) + items = optimisticDecodedResults.compactMap { try? $0.result.get() } + nextToken = try values.decode(String?.self, forKey: .nextToken) + startedAt = try values.decode(Int64?.self, forKey: .startedAt) + } +} + + +fileprivate struct OptimisticDecoded: Decodable { + let result: Result + + init(from decoder: Decoder) throws { + result = Result(catching: { + try T(from: decoder) + }) + } } diff --git a/AmplifyPlugins/Core/AWSPluginsCoreTests/Sync/PaginatedListTests.swift b/AmplifyPlugins/Core/AWSPluginsCoreTests/Sync/PaginatedListTests.swift index 58fe5ee517..3027a3b68f 100644 --- a/AmplifyPlugins/Core/AWSPluginsCoreTests/Sync/PaginatedListTests.swift +++ b/AmplifyPlugins/Core/AWSPluginsCoreTests/Sync/PaginatedListTests.swift @@ -69,6 +69,7 @@ class PaginatedListTests: XCTestCase { XCTAssertNotNil(paginatedList.startedAt) XCTAssertNotNil(paginatedList.nextToken) XCTAssertNotNil(paginatedList.items) + XCTAssertEqual(paginatedList.items.count, 2) XCTAssert(!paginatedList.items.isEmpty) XCTAssert(paginatedList.items[0].model.title == "title") XCTAssert(paginatedList.items[0].syncMetadata.version == 10) @@ -94,4 +95,53 @@ class PaginatedListTests: XCTestCase { } } + /// - Given: a `Post` Sync query with items, nextToken, and with sync data (startedAt, _version, etc) + /// - When: + /// - some of the JSON items are not able to be decoded to Post + /// - the JSON is decoded into `PaginatedList` + /// - Then: + /// - the result should contain only valid items of type MutationSync, startedAt, nextToken. + func testDecodePaginatedListOptimistically() { + let syncQueryJSON = """ + { + "items": [ + null, + { + "id": "post-id", + "createdAt": "2019-11-27T23:35:39Z", + "_version": 10, + "_lastChangedAt": 1574897753341, + "_deleted": null + }, + { + "id": "post-id", + "title": "title", + "content": "post content", + "createdAt": "2019-11-27T23:35:39Z", + "_version": 11, + "_lastChangedAt": 1574897753341, + "_deleted": null + } + ], + "startedAt": 1575322600038, + "nextToken": "token" + } + """ + do { + let decoder = JSONDecoder(dateDecodingStrategy: ModelDateFormatting.decodingStrategy) + let data = Data(syncQueryJSON.utf8) + let paginatedList = try decoder.decode(PaginatedList.self, from: data) + XCTAssertNotNil(paginatedList) + XCTAssertNotNil(paginatedList.startedAt) + XCTAssertNotNil(paginatedList.nextToken) + XCTAssertNotNil(paginatedList.items) + XCTAssertEqual(paginatedList.items.count, 1) + XCTAssert(paginatedList.items[0].model.title == "title") + XCTAssert(paginatedList.items[0].syncMetadata.version == 11) + XCTAssert(paginatedList.items[0].syncMetadata.lastChangedAt == 1_574_897_753_341) + } catch { + XCTFail(error.localizedDescription) + } + } + } diff --git a/AmplifyPlugins/DataStore/Sources/AWSDataStorePlugin/Sync/InitialSync/InitialSyncOperation.swift b/AmplifyPlugins/DataStore/Sources/AWSDataStorePlugin/Sync/InitialSync/InitialSyncOperation.swift index 6785669c6f..e01f235b88 100644 --- a/AmplifyPlugins/DataStore/Sources/AWSDataStorePlugin/Sync/InitialSync/InitialSyncOperation.swift +++ b/AmplifyPlugins/DataStore/Sources/AWSDataStorePlugin/Sync/InitialSync/InitialSyncOperation.swift @@ -216,11 +216,17 @@ final class InitialSyncOperation: AsynchronousOperation { let syncQueryResult: SyncQueryResult switch graphQLResult { + case .success(let queryResult): + syncQueryResult = queryResult + + case .failure(.partial(let queryResult, let errors)): + syncQueryResult = queryResult + errors.map { DataStoreError.api(APIError(errorDescription: $0.message, error: $0)) } + .forEach { dataStoreConfiguration.errorHandler($0) } + case .failure(let graphQLResponseError): finish(result: .failure(DataStoreError.api(graphQLResponseError))) return - case .success(let queryResult): - syncQueryResult = queryResult } let items = syncQueryResult.items diff --git a/AmplifyPlugins/DataStore/Sources/AWSDataStorePlugin/Sync/MutationSync/OutgoingMutationQueue/SyncMutationToCloudOperation.swift b/AmplifyPlugins/DataStore/Sources/AWSDataStorePlugin/Sync/MutationSync/OutgoingMutationQueue/SyncMutationToCloudOperation.swift index 4577782c41..e9f6cecb28 100644 --- a/AmplifyPlugins/DataStore/Sources/AWSDataStorePlugin/Sync/MutationSync/OutgoingMutationQueue/SyncMutationToCloudOperation.swift +++ b/AmplifyPlugins/DataStore/Sources/AWSDataStorePlugin/Sync/MutationSync/OutgoingMutationQueue/SyncMutationToCloudOperation.swift @@ -294,8 +294,18 @@ class SyncMutationToCloudOperation: AsynchronousOperation { advice = shouldRetryWithDifferentAuthType() // should retry with a different authType if request failed locally with an AuthError case .operationError(_, _, let error) where (error as? AuthError) != nil: - advice = shouldRetryWithDifferentAuthType() - + + // Not all AuthError's are unauthorized errors. If `AuthError.sessionExpired` then + // the request never made it to the server. We should keep trying until the user is signed in. + // Otherwise we may be making the wrong determination to remove this mutation event. + if case .sessionExpired = error as? AuthError { + // Use `userAuthenticationRequired` to ensure advice to retry is true. + advice = requestRetryablePolicy.retryRequestAdvice(urlError: URLError(.userAuthenticationRequired), + httpURLResponse: nil, + attemptNumber: currentAttemptNumber) + } else { + advice = shouldRetryWithDifferentAuthType() + } case .httpStatusError(_, let httpURLResponse): advice = requestRetryablePolicy.retryRequestAdvice(urlError: nil, httpURLResponse: httpURLResponse, diff --git a/AmplifyPlugins/DataStore/Sources/AWSDataStorePlugin/Sync/RequestRetryablePolicy.swift b/AmplifyPlugins/DataStore/Sources/AWSDataStorePlugin/Sync/RequestRetryablePolicy.swift index 39b67528ae..220834d187 100644 --- a/AmplifyPlugins/DataStore/Sources/AWSDataStorePlugin/Sync/RequestRetryablePolicy.swift +++ b/AmplifyPlugins/DataStore/Sources/AWSDataStorePlugin/Sync/RequestRetryablePolicy.swift @@ -40,7 +40,9 @@ class RequestRetryablePolicy: RequestRetryable { .timedOut, .dataNotAllowed, .cannotParseResponse, - .networkConnectionLost: + .networkConnectionLost, + .secureConnectionFailed, + .userAuthenticationRequired: let waitMillis = retryDelayInMillseconds(for: attemptNumber) return RequestRetryAdvice(shouldRetry: true, retryInterval: .milliseconds(waitMillis)) default: diff --git a/AmplifyPlugins/DataStore/Tests/AWSDataStorePluginTests/Sync/RequestRetryablePolicyTests.swift b/AmplifyPlugins/DataStore/Tests/AWSDataStorePluginTests/Sync/RequestRetryablePolicyTests.swift index 498fdaddc6..bf4e51baa7 100644 --- a/AmplifyPlugins/DataStore/Tests/AWSDataStorePluginTests/Sync/RequestRetryablePolicyTests.swift +++ b/AmplifyPlugins/DataStore/Tests/AWSDataStorePluginTests/Sync/RequestRetryablePolicyTests.swift @@ -199,6 +199,17 @@ class RequestRetryablePolicyTests: XCTestCase { assertMilliseconds(retryAdvice.retryInterval, greaterThan: 200, lessThan: 300) } + func testUserAuthenticationRequiredError() { + let retryableErrorCode = URLError.init(.userAuthenticationRequired) + + let retryAdvice = retryPolicy.retryRequestAdvice(urlError: retryableErrorCode, + httpURLResponse: nil, + attemptNumber: 1) + + XCTAssert(retryAdvice.shouldRetry) + assertMilliseconds(retryAdvice.retryInterval, greaterThan: 200, lessThan: 300) + } + func testHTTPTooManyRedirectsError() { let nonRetryableErrorCode = URLError.init(.httpTooManyRedirects) @@ -210,6 +221,17 @@ class RequestRetryablePolicyTests: XCTestCase { XCTAssertEqual(retryAdvice.retryInterval, defaultTimeout) } + func testSecureConnectionFailedError() { + let retryableErrorCode = URLError.init(.secureConnectionFailed) + + let retryAdvice = retryPolicy.retryRequestAdvice(urlError: retryableErrorCode, + httpURLResponse: nil, + attemptNumber: 1) + + XCTAssert(retryAdvice.shouldRetry) + assertMilliseconds(retryAdvice.retryInterval, greaterThan: 200, lessThan: 300) + } + func testMaxValueRetryDelay() { let retryableErrorCode = URLError.init(.timedOut)