From 0c97348fb0574038cca9c53aad3e0e1eb5f1f8f8 Mon Sep 17 00:00:00 2001 From: yamilmedina Date: Thu, 7 Dec 2023 16:08:15 +0100 Subject: [PATCH 1/3] fix: commit bundles special handling --- .../wire/kalium/network/utils/NetworkUtils.kt | 51 +++++++++--- .../wire/kalium/api/v5/MLSMessageApiV5Test.kt | 77 +++++++++++++++++++ 2 files changed, 118 insertions(+), 10 deletions(-) diff --git a/network/src/commonMain/kotlin/com/wire/kalium/network/utils/NetworkUtils.kt b/network/src/commonMain/kotlin/com/wire/kalium/network/utils/NetworkUtils.kt index a91ed94a815..3bdffdea062 100644 --- a/network/src/commonMain/kotlin/com/wire/kalium/network/utils/NetworkUtils.kt +++ b/network/src/commonMain/kotlin/com/wire/kalium/network/utils/NetworkUtils.kt @@ -32,6 +32,7 @@ import io.ktor.http.HttpStatusCode import io.ktor.http.URLProtocol import io.ktor.http.Url import io.ktor.http.isSuccess +import io.ktor.serialization.JsonConvertException internal fun HttpRequestBuilder.setWSSUrl(baseUrl: Url, vararg path: String) { url { @@ -235,25 +236,22 @@ private fun toStatusCodeBasedKaliumException( return kException } + /** * Wrap and handles federation aware endpoints that can send errors responses - * And raise proper federated exceptions + * And raise specific federated context exceptions, + * + * i.e. FederationError, FederationUnreachableException, FederationConflictException * - * @delegatedHandler the fallback handler when the response is not a federation error + * @param response the response to wrap + * @param delegatedHandler the fallback handler when the response cannot be handled as a federation error */ suspend fun wrapFederationResponse( response: HttpResponse, delegatedHandler: suspend (HttpResponse) -> NetworkResponse ) = when (response.status.value) { - HttpStatusCode.Conflict.value -> { - val errorResponse = try { - response.body() - } catch (_: NoTransformationFoundException) { - FederationConflictResponse(emptyList()) - } - NetworkResponse.Error(KaliumException.FederationConflictException(errorResponse)) - } + HttpStatusCode.Conflict.value -> resolveStatusCodeBasedFirstOrFederated(response) HttpStatusCode.UnprocessableEntity.value -> { val errorResponse = try { @@ -277,3 +275,36 @@ suspend fun wrapFederationResponse( delegatedHandler.invoke(response) } } + +/** + * Due to the "shared" status code limitations nature of some endpoints. + * We need to first delegate to status code based exceptions and if parse fails go for federated error, + * + * i.e.: '/commit-bundles' 409 for "mls-stale-message" and 409 for "federation-conflict" + */ +private suspend fun resolveStatusCodeBasedFirstOrFederated(response: HttpResponse): NetworkResponse.Error { + val kaliumException = try { + val errorResponse = response.body() + toStatusCodeBasedKaliumException( + response.status, + response, + errorResponse + ) + } catch (exception: Exception) { + when { + exception is JsonConvertException -> { + try { + KaliumException.FederationConflictException(response.body()) + } catch (_: NoTransformationFoundException) { + KaliumException.FederationConflictException(FederationConflictResponse(emptyList())) + } + } + + else -> { + KaliumException.GenericError(exception) + } + } + + } + return NetworkResponse.Error(kaliumException) +} diff --git a/network/src/commonTest/kotlin/com/wire/kalium/api/v5/MLSMessageApiV5Test.kt b/network/src/commonTest/kotlin/com/wire/kalium/api/v5/MLSMessageApiV5Test.kt index d08d1a3cb2f..5b9959b2c66 100644 --- a/network/src/commonTest/kotlin/com/wire/kalium/api/v5/MLSMessageApiV5Test.kt +++ b/network/src/commonTest/kotlin/com/wire/kalium/api/v5/MLSMessageApiV5Test.kt @@ -19,17 +19,24 @@ package com.wire.kalium.api.v5 import com.wire.kalium.api.ApiTest +import com.wire.kalium.api.json.model.ErrorResponseJson +import com.wire.kalium.model.EventContentDTOJson import com.wire.kalium.model.SendMLSMessageResponseJson import com.wire.kalium.network.api.base.authenticated.message.MLSMessageApi +import com.wire.kalium.network.api.base.model.ErrorResponse +import com.wire.kalium.network.api.base.model.FederationConflictResponse import com.wire.kalium.network.api.v0.authenticated.MLSMessageApiV0 import com.wire.kalium.network.api.v5.authenticated.MLSMessageApiV5 +import com.wire.kalium.network.exceptions.KaliumException import com.wire.kalium.network.serialization.Mls +import com.wire.kalium.network.utils.UnreachableRemoteBackends import com.wire.kalium.network.utils.isSuccessful import io.ktor.http.ContentType import io.ktor.http.HttpStatusCode import kotlinx.coroutines.ExperimentalCoroutinesApi import kotlinx.coroutines.test.runTest import kotlin.test.Test +import kotlin.test.assertEquals import kotlin.test.assertFalse import kotlin.test.assertTrue @@ -80,6 +87,76 @@ internal class MLSMessageApiV5Test : ApiTest() { assertFalse(response.isSuccessful()) } + @Test + fun givenCommitBundle_whenSendingBundleFailsUnreachable_theRequestShouldFailWithUnreachableError() = runTest { + val networkClient = mockAuthenticatedNetworkClient( + EventContentDTOJson.jsonProviderMemberJoinFailureUnreachable, + statusCode = HttpStatusCode.UnreachableRemoteBackends, + assertion = + { + assertPost() + assertContentType(ContentType.Message.Mls) + assertPathEqual(PATH_COMMIT_BUNDLES) + } + ) + val mlsMessageApi: MLSMessageApi = MLSMessageApiV5(networkClient) + val response = mlsMessageApi.sendCommitBundle(COMMIT_BUNDLE) + + assertFalse(response.isSuccessful()) + assertEquals(KaliumException.FederationUnreachableException::class, response.kException::class) + } + + @Test + fun givenCommitBundle_whenSendingBundleFailsConflict_theRequestShouldFailWithConflictDomainsError() = runTest { + val nonFederatingBackends = listOf("bella.wire.link", "foma.wire.link") + val networkClient = mockAuthenticatedNetworkClient( + ErrorResponseJson.validFederationConflictingBackends( + FederationConflictResponse(nonFederatingBackends) + ).rawJson, + statusCode = HttpStatusCode.Conflict, + assertion = + { + assertPost() + assertContentType(ContentType.Message.Mls) + assertPathEqual(PATH_COMMIT_BUNDLES) + } + ) + val mlsMessageApi: MLSMessageApi = MLSMessageApiV5(networkClient) + val response = mlsMessageApi.sendCommitBundle(COMMIT_BUNDLE) + + assertFalse(response.isSuccessful()) + assertEquals(KaliumException.FederationConflictException::class, response.kException::class) + assertEquals( + expected = nonFederatingBackends, + actual = (response.kException as KaliumException.FederationConflictException).errorResponse.nonFederatingBackends + ) + } + + @Test + fun givenCommitBundle_whenSendingBundleFailsConflictNonFederated_theRequestShouldFailWithRegularInvalidRequestError() = runTest { + val networkClient = mockAuthenticatedNetworkClient( + ErrorResponseJson.valid( + ErrorResponse( + code = HttpStatusCode.Conflict.value, + message = "some other error", + label = "mls-stale-message" + ) + ).rawJson, + statusCode = HttpStatusCode.Conflict, + assertion = + { + assertPost() + assertContentType(ContentType.Message.Mls) + assertPathEqual(PATH_COMMIT_BUNDLES) + } + ) + val mlsMessageApi: MLSMessageApi = MLSMessageApiV5(networkClient) + val response = mlsMessageApi.sendCommitBundle(COMMIT_BUNDLE) + + assertFalse(response.isSuccessful()) + assertEquals(KaliumException.InvalidRequestError::class, response.kException::class) + } + private companion object { const val PATH_MESSAGE = "/mls/messages" const val PATH_COMMIT_BUNDLES = "mls/commit-bundles" From ee14b6e516702a0ddf95e5f4da0fc7576ff75376 Mon Sep 17 00:00:00 2001 From: yamilmedina Date: Thu, 7 Dec 2023 16:18:03 +0100 Subject: [PATCH 2/3] fix: detekt --- .../kotlin/com/wire/kalium/network/utils/NetworkUtils.kt | 1 - 1 file changed, 1 deletion(-) diff --git a/network/src/commonMain/kotlin/com/wire/kalium/network/utils/NetworkUtils.kt b/network/src/commonMain/kotlin/com/wire/kalium/network/utils/NetworkUtils.kt index 3bdffdea062..031c3b68e14 100644 --- a/network/src/commonMain/kotlin/com/wire/kalium/network/utils/NetworkUtils.kt +++ b/network/src/commonMain/kotlin/com/wire/kalium/network/utils/NetworkUtils.kt @@ -236,7 +236,6 @@ private fun toStatusCodeBasedKaliumException( return kException } - /** * Wrap and handles federation aware endpoints that can send errors responses * And raise specific federated context exceptions, From d353ff1d392a1f3a19fc331f41589e8d2ee131bc Mon Sep 17 00:00:00 2001 From: yamilmedina Date: Thu, 7 Dec 2023 16:19:40 +0100 Subject: [PATCH 3/3] fix: detekt --- .../wire/kalium/network/utils/NetworkUtils.kt | 19 +++++-------------- 1 file changed, 5 insertions(+), 14 deletions(-) diff --git a/network/src/commonMain/kotlin/com/wire/kalium/network/utils/NetworkUtils.kt b/network/src/commonMain/kotlin/com/wire/kalium/network/utils/NetworkUtils.kt index 031c3b68e14..d1617aee9e1 100644 --- a/network/src/commonMain/kotlin/com/wire/kalium/network/utils/NetworkUtils.kt +++ b/network/src/commonMain/kotlin/com/wire/kalium/network/utils/NetworkUtils.kt @@ -289,21 +289,12 @@ private suspend fun resolveStatusCodeBasedFirstOrFederated(response: HttpRespons response, errorResponse ) - } catch (exception: Exception) { - when { - exception is JsonConvertException -> { - try { - KaliumException.FederationConflictException(response.body()) - } catch (_: NoTransformationFoundException) { - KaliumException.FederationConflictException(FederationConflictResponse(emptyList())) - } - } - - else -> { - KaliumException.GenericError(exception) - } + } catch (exception: JsonConvertException) { + try { + KaliumException.FederationConflictException(response.body()) + } catch (_: NoTransformationFoundException) { + KaliumException.FederationConflictException(FederationConflictResponse(emptyList())) } - } return NetworkResponse.Error(kaliumException) }