From 4432dbe92a4d670b82b430bf6da42c62eb9c9352 Mon Sep 17 00:00:00 2001 From: Jonatan Rhodin Date: Mon, 17 Jun 2024 12:48:09 +0200 Subject: [PATCH] Replace retry with exponential backoff with arrow schedule Also clean up the code related to play purchase verification --- android/app/build.gradle.kts | 1 + .../mullvadvpn/constant/PaymentConstant.kt | 6 ++-- .../mullvadvpn/usecase/PaymentUseCase.kt | 34 ++++++++---------- .../net/mullvad/mullvadvpn/util/FlowUtils.kt | 35 ------------------- .../mullvadvpn/viewmodel/AccountViewModel.kt | 5 +-- .../mullvadvpn/viewmodel/ConnectViewModel.kt | 4 +-- .../viewmodel/OutOfTimeViewModel.kt | 5 +-- .../mullvadvpn/viewmodel/WelcomeViewModel.kt | 5 +-- .../buildSrc/src/main/kotlin/Dependencies.kt | 1 + android/gradle/verification-metadata.xml | 16 +++++++++ .../lib/billing/BillingPaymentRepository.kt | 35 ++++++++----------- android/lib/payment/build.gradle.kts | 1 + .../lib/payment/PaymentRepository.kt | 4 ++- .../lib/payment/model/VerificationError.kt | 7 ++++ .../lib/payment/model/VerificationResult.kt | 14 +------- 15 files changed, 75 insertions(+), 98 deletions(-) create mode 100644 android/lib/payment/src/main/kotlin/net/mullvad/mullvadvpn/lib/payment/model/VerificationError.kt diff --git a/android/app/build.gradle.kts b/android/app/build.gradle.kts index 798667ff3f32..8e523ecb3991 100644 --- a/android/app/build.gradle.kts +++ b/android/app/build.gradle.kts @@ -329,6 +329,7 @@ dependencies { implementation(Dependencies.AndroidX.lifecycleViewmodelKtx) implementation(Dependencies.AndroidX.lifecycleRuntimeCompose) implementation(Dependencies.Arrow.core) + implementation(Dependencies.Arrow.resilience) implementation(Dependencies.Compose.constrainLayout) implementation(Dependencies.Compose.foundation) implementation(Dependencies.Compose.material3) diff --git a/android/app/src/main/kotlin/net/mullvad/mullvadvpn/constant/PaymentConstant.kt b/android/app/src/main/kotlin/net/mullvad/mullvadvpn/constant/PaymentConstant.kt index 1cf1c32b2cf5..ba2413d4fc1c 100644 --- a/android/app/src/main/kotlin/net/mullvad/mullvadvpn/constant/PaymentConstant.kt +++ b/android/app/src/main/kotlin/net/mullvad/mullvadvpn/constant/PaymentConstant.kt @@ -1,5 +1,7 @@ package net.mullvad.mullvadvpn.constant +import kotlin.time.Duration.Companion.seconds + const val VERIFICATION_MAX_ATTEMPTS = 4 -const val VERIFICATION_INITIAL_BACK_OFF_MILLISECONDS = 3000L -const val VERIFICATION_BACK_OFF_FACTOR = 3L +val VERIFICATION_INITIAL_BACK_OFF_DURATION = 3.seconds +const val VERIFICATION_BACK_OFF_FACTOR = 3.toDouble() diff --git a/android/app/src/main/kotlin/net/mullvad/mullvadvpn/usecase/PaymentUseCase.kt b/android/app/src/main/kotlin/net/mullvad/mullvadvpn/usecase/PaymentUseCase.kt index 82efd5d72206..00d716399385 100644 --- a/android/app/src/main/kotlin/net/mullvad/mullvadvpn/usecase/PaymentUseCase.kt +++ b/android/app/src/main/kotlin/net/mullvad/mullvadvpn/usecase/PaymentUseCase.kt @@ -1,20 +1,22 @@ package net.mullvad.mullvadvpn.usecase import android.app.Activity +import arrow.resilience.Schedule +import arrow.resilience.retryEither import kotlinx.coroutines.delay import kotlinx.coroutines.flow.Flow import kotlinx.coroutines.flow.MutableStateFlow import kotlinx.coroutines.flow.asStateFlow import kotlinx.coroutines.flow.transform import net.mullvad.mullvadvpn.constant.VERIFICATION_BACK_OFF_FACTOR -import net.mullvad.mullvadvpn.constant.VERIFICATION_INITIAL_BACK_OFF_MILLISECONDS +import net.mullvad.mullvadvpn.constant.VERIFICATION_INITIAL_BACK_OFF_DURATION import net.mullvad.mullvadvpn.constant.VERIFICATION_MAX_ATTEMPTS import net.mullvad.mullvadvpn.lib.payment.PaymentRepository import net.mullvad.mullvadvpn.lib.payment.model.PaymentAvailability import net.mullvad.mullvadvpn.lib.payment.model.ProductId import net.mullvad.mullvadvpn.lib.payment.model.PurchaseResult +import net.mullvad.mullvadvpn.lib.payment.model.VerificationError import net.mullvad.mullvadvpn.lib.payment.model.VerificationResult -import net.mullvad.mullvadvpn.util.retryWithExponentialBackOff interface PaymentUseCase { val paymentAvailability: Flow @@ -26,7 +28,7 @@ interface PaymentUseCase { suspend fun resetPurchaseResult() - suspend fun verifyPurchases(onSuccessfulVerification: () -> Unit = {}) + suspend fun verifyPurchases(): Boolean } class PlayPaymentUseCase(private val paymentRepository: PaymentRepository) : PaymentUseCase { @@ -56,24 +58,20 @@ class PlayPaymentUseCase(private val paymentRepository: PaymentRepository) : Pay _purchaseResult.emit(null) } - override suspend fun verifyPurchases(onSuccessfulVerification: () -> Unit) { - paymentRepository - .verifyPurchases() - .retryWithExponentialBackOff( - maxAttempts = VERIFICATION_MAX_ATTEMPTS, - initialBackOffDelay = VERIFICATION_INITIAL_BACK_OFF_MILLISECONDS, - backOffDelayFactor = VERIFICATION_BACK_OFF_FACTOR - ) { - it is VerificationResult.Error - } - .collect { + override suspend fun verifyPurchases() = + Schedule.exponential( + VERIFICATION_INITIAL_BACK_OFF_DURATION, + VERIFICATION_BACK_OFF_FACTOR + ) + .and(Schedule.recurs(VERIFICATION_MAX_ATTEMPTS.toLong())) + .retryEither { paymentRepository.verifyPurchases() } + .onRight { if (it == VerificationResult.Success) { // Update the payment availability after a successful verification. queryPaymentAvailability() - onSuccessfulVerification() } } - } + .getOrNull() == VerificationResult.Success private fun PurchaseResult?.shouldDelayLoading() = this is PurchaseResult.FetchingProducts || this is PurchaseResult.VerificationStarted @@ -99,7 +97,5 @@ class EmptyPaymentUseCase : PaymentUseCase { // No op } - override suspend fun verifyPurchases(onSuccessfulVerification: () -> Unit) { - // No op - } + override suspend fun verifyPurchases(): Boolean = false } diff --git a/android/app/src/main/kotlin/net/mullvad/mullvadvpn/util/FlowUtils.kt b/android/app/src/main/kotlin/net/mullvad/mullvadvpn/util/FlowUtils.kt index fbe44a5feabe..13561aa7f858 100644 --- a/android/app/src/main/kotlin/net/mullvad/mullvadvpn/util/FlowUtils.kt +++ b/android/app/src/main/kotlin/net/mullvad/mullvadvpn/util/FlowUtils.kt @@ -3,11 +3,7 @@ package net.mullvad.mullvadvpn.util import kotlinx.coroutines.Deferred -import kotlinx.coroutines.delay import kotlinx.coroutines.flow.Flow -import kotlinx.coroutines.flow.catch -import kotlinx.coroutines.flow.map -import kotlinx.coroutines.flow.retryWhen inline fun combine( flow: Flow, @@ -90,34 +86,3 @@ fun Deferred.getOrDefault(default: T) = } catch (e: IllegalStateException) { default } - -@Suppress("UNCHECKED_CAST") -suspend inline fun Flow.retryWithExponentialBackOff( - maxAttempts: Int, - initialBackOffDelay: Long, - backOffDelayFactor: Long, - crossinline predicate: (T) -> Boolean, -): Flow = - map { - if (predicate(it)) { - throw ExceptionWrapper(it as Any) - } - it - } - .retryWhen { _, attempt -> - if (attempt >= maxAttempts) { - return@retryWhen false - } - val backOffDelay = initialBackOffDelay * backOffDelayFactor.pow(attempt.toInt()) - delay(backOffDelay) - true - } - .catch { - if (it is ExceptionWrapper) { - this.emit(it.item as T) - } else { - throw it - } - } - -class ExceptionWrapper(val item: Any) : Throwable() diff --git a/android/app/src/main/kotlin/net/mullvad/mullvadvpn/viewmodel/AccountViewModel.kt b/android/app/src/main/kotlin/net/mullvad/mullvadvpn/viewmodel/AccountViewModel.kt index a42003d6e200..5a7e3cf3e442 100644 --- a/android/app/src/main/kotlin/net/mullvad/mullvadvpn/viewmodel/AccountViewModel.kt +++ b/android/app/src/main/kotlin/net/mullvad/mullvadvpn/viewmodel/AccountViewModel.kt @@ -90,8 +90,9 @@ class AccountViewModel( private fun verifyPurchases() { viewModelScope.launch { - paymentUseCase.verifyPurchases() - updateAccountExpiry() + if (paymentUseCase.verifyPurchases()) { + updateAccountExpiry() + } } } diff --git a/android/app/src/main/kotlin/net/mullvad/mullvadvpn/viewmodel/ConnectViewModel.kt b/android/app/src/main/kotlin/net/mullvad/mullvadvpn/viewmodel/ConnectViewModel.kt index a27ed9adbd29..0ba20a693c76 100644 --- a/android/app/src/main/kotlin/net/mullvad/mullvadvpn/viewmodel/ConnectViewModel.kt +++ b/android/app/src/main/kotlin/net/mullvad/mullvadvpn/viewmodel/ConnectViewModel.kt @@ -114,8 +114,8 @@ class ConnectViewModel( init { viewModelScope.launch { - paymentUseCase.verifyPurchases { - viewModelScope.launch { accountRepository.getAccountData() } + if (paymentUseCase.verifyPurchases()) { + accountRepository.getAccountData() } } viewModelScope.launch { deviceRepository.updateDevice() } diff --git a/android/app/src/main/kotlin/net/mullvad/mullvadvpn/viewmodel/OutOfTimeViewModel.kt b/android/app/src/main/kotlin/net/mullvad/mullvadvpn/viewmodel/OutOfTimeViewModel.kt index 77841814666f..b42dd1dbfbf6 100644 --- a/android/app/src/main/kotlin/net/mullvad/mullvadvpn/viewmodel/OutOfTimeViewModel.kt +++ b/android/app/src/main/kotlin/net/mullvad/mullvadvpn/viewmodel/OutOfTimeViewModel.kt @@ -76,8 +76,9 @@ class OutOfTimeViewModel( private fun verifyPurchases() { viewModelScope.launch { - paymentUseCase.verifyPurchases() - updateAccountExpiry() + if (paymentUseCase.verifyPurchases()) { + updateAccountExpiry() + } } } diff --git a/android/app/src/main/kotlin/net/mullvad/mullvadvpn/viewmodel/WelcomeViewModel.kt b/android/app/src/main/kotlin/net/mullvad/mullvadvpn/viewmodel/WelcomeViewModel.kt index 4b61468f8ed0..21ad158f29c7 100644 --- a/android/app/src/main/kotlin/net/mullvad/mullvadvpn/viewmodel/WelcomeViewModel.kt +++ b/android/app/src/main/kotlin/net/mullvad/mullvadvpn/viewmodel/WelcomeViewModel.kt @@ -79,8 +79,9 @@ class WelcomeViewModel( private fun verifyPurchases() { viewModelScope.launch { - paymentUseCase.verifyPurchases() - updateAccountExpiry() + if (paymentUseCase.verifyPurchases()) { + updateAccountExpiry() + } } } diff --git a/android/buildSrc/src/main/kotlin/Dependencies.kt b/android/buildSrc/src/main/kotlin/Dependencies.kt index b2417befba98..d13b7c17b3c6 100644 --- a/android/buildSrc/src/main/kotlin/Dependencies.kt +++ b/android/buildSrc/src/main/kotlin/Dependencies.kt @@ -46,6 +46,7 @@ object Dependencies { const val core = "io.arrow-kt:arrow-core:${Versions.Arrow.base}" const val optics = "io.arrow-kt:arrow-optics:${Versions.Arrow.base}" const val opticsKsp = "io.arrow-kt:arrow-optics-ksp-plugin:${Versions.Arrow.base}" + const val resilience = "io.arrow-kt:arrow-resilience:${Versions.Arrow.base}" } object Compose { diff --git a/android/gradle/verification-metadata.xml b/android/gradle/verification-metadata.xml index b3d62044fca8..0f85c1575ee6 100644 --- a/android/gradle/verification-metadata.xml +++ b/android/gradle/verification-metadata.xml @@ -3242,6 +3242,22 @@ + + + + + + + + + + + + + + + + diff --git a/android/lib/billing/src/main/kotlin/net/mullvad/mullvadvpn/lib/billing/BillingPaymentRepository.kt b/android/lib/billing/src/main/kotlin/net/mullvad/mullvadvpn/lib/billing/BillingPaymentRepository.kt index 8b3ad66171f9..b526a10032cb 100644 --- a/android/lib/billing/src/main/kotlin/net/mullvad/mullvadvpn/lib/billing/BillingPaymentRepository.kt +++ b/android/lib/billing/src/main/kotlin/net/mullvad/mullvadvpn/lib/billing/BillingPaymentRepository.kt @@ -1,6 +1,9 @@ package net.mullvad.mullvadvpn.lib.billing import android.app.Activity +import arrow.core.Either +import arrow.core.raise.either +import arrow.core.raise.ensure import com.android.billingclient.api.BillingClient.BillingResponseCode import com.android.billingclient.api.Purchase import kotlinx.coroutines.flow.Flow @@ -22,6 +25,7 @@ import net.mullvad.mullvadvpn.lib.payment.ProductIds import net.mullvad.mullvadvpn.lib.payment.model.PaymentAvailability import net.mullvad.mullvadvpn.lib.payment.model.ProductId import net.mullvad.mullvadvpn.lib.payment.model.PurchaseResult +import net.mullvad.mullvadvpn.lib.payment.model.VerificationError import net.mullvad.mullvadvpn.lib.payment.model.VerificationResult class BillingPaymentRepository( @@ -129,28 +133,19 @@ class BillingPaymentRepository( } } - override fun verifyPurchases(): Flow = flow { - emit(VerificationResult.FetchingUnfinishedPurchases) + override suspend fun verifyPurchases(): Either = either { val purchasesResult = billingRepository.queryPurchases() - when (purchasesResult.responseCode()) { - BillingResponseCode.OK -> { - val purchases = purchasesResult.nonPendingPurchases() - if (purchases.isNotEmpty()) { - emit(VerificationResult.VerificationStarted) - emit( - verifyPurchase(purchases.first()) - .fold( - { VerificationResult.Error.VerificationError(null) }, - { VerificationResult.Success } - ) - ) - } else { - emit(VerificationResult.NothingToVerify) - } - } - else -> - emit(VerificationResult.Error.BillingError(purchasesResult.toBillingException())) + ensure(purchasesResult.responseCode() == BillingResponseCode.OK) { + VerificationError.BillingError(purchasesResult.toBillingException()) + } + val purchases = purchasesResult.nonPendingPurchases() + if (purchases.isEmpty()) { + return@either VerificationResult.NothingToVerify } + verifyPurchase(purchases.first()) + .mapLeft { VerificationError.PlayVerificationError } + .map { VerificationResult.Success } + .bind() } private suspend fun initialisePurchase() = playPurchaseRepository.initializePlayPurchase() diff --git a/android/lib/payment/build.gradle.kts b/android/lib/payment/build.gradle.kts index 23f945b4f9f6..8653be2701e3 100644 --- a/android/lib/payment/build.gradle.kts +++ b/android/lib/payment/build.gradle.kts @@ -41,4 +41,5 @@ android { dependencies { implementation(Dependencies.Kotlin.stdlib) implementation(Dependencies.KotlinX.coroutinesAndroid) + implementation(Dependencies.Arrow.core) } diff --git a/android/lib/payment/src/main/kotlin/net/mullvad/mullvadvpn/lib/payment/PaymentRepository.kt b/android/lib/payment/src/main/kotlin/net/mullvad/mullvadvpn/lib/payment/PaymentRepository.kt index 73fd0c061db0..0f076eab74e1 100644 --- a/android/lib/payment/src/main/kotlin/net/mullvad/mullvadvpn/lib/payment/PaymentRepository.kt +++ b/android/lib/payment/src/main/kotlin/net/mullvad/mullvadvpn/lib/payment/PaymentRepository.kt @@ -1,10 +1,12 @@ package net.mullvad.mullvadvpn.lib.payment import android.app.Activity +import arrow.core.Either import kotlinx.coroutines.flow.Flow import net.mullvad.mullvadvpn.lib.payment.model.PaymentAvailability import net.mullvad.mullvadvpn.lib.payment.model.ProductId import net.mullvad.mullvadvpn.lib.payment.model.PurchaseResult +import net.mullvad.mullvadvpn.lib.payment.model.VerificationError import net.mullvad.mullvadvpn.lib.payment.model.VerificationResult interface PaymentRepository { @@ -14,7 +16,7 @@ interface PaymentRepository { activityProvider: () -> Activity ): Flow - fun verifyPurchases(): Flow + suspend fun verifyPurchases(): Either fun queryPaymentAvailability(): Flow } diff --git a/android/lib/payment/src/main/kotlin/net/mullvad/mullvadvpn/lib/payment/model/VerificationError.kt b/android/lib/payment/src/main/kotlin/net/mullvad/mullvadvpn/lib/payment/model/VerificationError.kt new file mode 100644 index 000000000000..51cf8d1d28dd --- /dev/null +++ b/android/lib/payment/src/main/kotlin/net/mullvad/mullvadvpn/lib/payment/model/VerificationError.kt @@ -0,0 +1,7 @@ +package net.mullvad.mullvadvpn.lib.payment.model + +sealed interface VerificationError { + data class BillingError(val exception: Throwable) : VerificationError + + data object PlayVerificationError : VerificationError +} diff --git a/android/lib/payment/src/main/kotlin/net/mullvad/mullvadvpn/lib/payment/model/VerificationResult.kt b/android/lib/payment/src/main/kotlin/net/mullvad/mullvadvpn/lib/payment/model/VerificationResult.kt index 725ea0af68f2..8cf971194dd9 100644 --- a/android/lib/payment/src/main/kotlin/net/mullvad/mullvadvpn/lib/payment/model/VerificationResult.kt +++ b/android/lib/payment/src/main/kotlin/net/mullvad/mullvadvpn/lib/payment/model/VerificationResult.kt @@ -1,19 +1,7 @@ package net.mullvad.mullvadvpn.lib.payment.model -sealed interface VerificationResult { - data object FetchingUnfinishedPurchases : VerificationResult - - data object VerificationStarted : VerificationResult - - // No verification was needed as there is no purchases to verify +interface VerificationResult { data object NothingToVerify : VerificationResult data object Success : VerificationResult - - // Generic error, add more cases as needed - sealed interface Error : VerificationResult { - data class BillingError(val exception: Throwable?) : Error - - data class VerificationError(val exception: Throwable?) : Error - } }