From 5d0aac52fa20c5eade4825d062e8662446e4d973 Mon Sep 17 00:00:00 2001 From: toluo-stripe Date: Thu, 9 Jan 2025 13:19:36 -0500 Subject: [PATCH 1/6] Link Cvc & Expiry recollection VM Add confirmation block Update unit tests Lint Update WalletViewModelTest.kt Update FakeLinkConfirmationHandler.kt --- .../ui/core/elements/CardDetailsElement.kt | 9 + .../android/link/LinkActivityViewModel.kt | 3 + .../android/link/ui/wallet/WalletUiState.kt | 13 +- .../android/link/ui/wallet/WalletViewModel.kt | 134 +++++++- .../android/link/LinkActivityViewModelTest.kt | 2 + .../FakeLinkConfirmationHandler.kt | 18 +- .../link/ui/wallet/WalletScreenTest.kt | 6 +- .../link/ui/wallet/WalletUiStateTest.kt | 91 ++++++ .../link/ui/wallet/WalletViewModelTest.kt | 305 +++++++++++++++++- 9 files changed, 576 insertions(+), 5 deletions(-) diff --git a/payments-ui-core/src/main/java/com/stripe/android/ui/core/elements/CardDetailsElement.kt b/payments-ui-core/src/main/java/com/stripe/android/ui/core/elements/CardDetailsElement.kt index 566328b983f..e1c1ee1c3c4 100644 --- a/payments-ui-core/src/main/java/com/stripe/android/ui/core/elements/CardDetailsElement.kt +++ b/payments-ui-core/src/main/java/com/stripe/android/ui/core/elements/CardDetailsElement.kt @@ -1,5 +1,6 @@ package com.stripe.android.ui.core.elements +import androidx.annotation.RestrictTo import com.stripe.android.CardBrandFilter import com.stripe.android.DefaultCardBrandFilter import com.stripe.android.cards.CardAccountRangeRepository @@ -108,6 +109,14 @@ internal class CardDetailsElement( } } +@RestrictTo(RestrictTo.Scope.LIBRARY_GROUP) +fun createExpiryDateFormFieldValues(entry: FormFieldEntry): Map { + return mapOf( + IdentifierSpec.CardExpMonth to getExpiryMonthFormFieldEntry(entry), + IdentifierSpec.CardExpYear to getExpiryYearFormFieldEntry(entry) + ) +} + private fun getExpiryMonthFormFieldEntry(entry: FormFieldEntry): FormFieldEntry { var month = -1 entry.value?.let { date -> diff --git a/paymentsheet/src/main/java/com/stripe/android/link/LinkActivityViewModel.kt b/paymentsheet/src/main/java/com/stripe/android/link/LinkActivityViewModel.kt index 2c20038e7f4..04f11b8f12a 100644 --- a/paymentsheet/src/main/java/com/stripe/android/link/LinkActivityViewModel.kt +++ b/paymentsheet/src/main/java/com/stripe/android/link/LinkActivityViewModel.kt @@ -19,6 +19,7 @@ import com.stripe.android.link.injection.NativeLinkComponent import com.stripe.android.link.model.AccountStatus import com.stripe.android.link.model.LinkAccount import com.stripe.android.link.ui.LinkAppBarState +import com.stripe.android.paymentelement.confirmation.ConfirmationHandler import com.stripe.android.paymentsheet.R import kotlinx.coroutines.flow.MutableStateFlow import kotlinx.coroutines.flow.StateFlow @@ -28,8 +29,10 @@ import javax.inject.Inject internal class LinkActivityViewModel @Inject constructor( val activityRetainedComponent: NativeLinkComponent, + confirmationHandlerFactory: ConfirmationHandler.Factory, private val linkAccountManager: LinkAccountManager, ) : ViewModel(), DefaultLifecycleObserver { + val confirmationHandler = confirmationHandlerFactory.create(viewModelScope) private val _linkState = MutableStateFlow( value = LinkAppBarState( navigationIcon = R.drawable.stripe_link_close, diff --git a/paymentsheet/src/main/java/com/stripe/android/link/ui/wallet/WalletUiState.kt b/paymentsheet/src/main/java/com/stripe/android/link/ui/wallet/WalletUiState.kt index b81809155d6..098016a239d 100644 --- a/paymentsheet/src/main/java/com/stripe/android/link/ui/wallet/WalletUiState.kt +++ b/paymentsheet/src/main/java/com/stripe/android/link/ui/wallet/WalletUiState.kt @@ -5,6 +5,7 @@ import com.stripe.android.core.strings.ResolvableString import com.stripe.android.link.ui.PrimaryButtonState import com.stripe.android.model.ConsumerPaymentDetails import com.stripe.android.model.ConsumerPaymentDetails.Card +import com.stripe.android.uicore.forms.FormFieldEntry @Immutable internal data class WalletUiState( @@ -13,8 +14,14 @@ internal data class WalletUiState( val isProcessing: Boolean, val primaryButtonLabel: ResolvableString, val hasCompleted: Boolean, + val errorMessage: ResolvableString? = null, + val expiryDateInput: FormFieldEntry = FormFieldEntry(null), + val cvcInput: FormFieldEntry = FormFieldEntry(null), + val alertMessage: ResolvableString? = null, ) { + val selectedCard: Card? = selectedItem as? Card + val showBankAccountTerms = selectedItem is ConsumerPaymentDetails.BankAccount val primaryButtonState: PrimaryButtonState @@ -23,7 +30,11 @@ internal data class WalletUiState( val isExpired = card?.isExpired ?: false val requiresCvcRecollection = card?.cvcCheck?.requiresRecollection ?: false - val disableButton = isExpired || requiresCvcRecollection + val isMissingExpiryDateInput = (expiryDateInput.isComplete && cvcInput.isComplete).not() + val isMissingCvcInput = cvcInput.isComplete.not() + + val disableButton = (isExpired && isMissingExpiryDateInput) || + (requiresCvcRecollection && isMissingCvcInput) return if (hasCompleted) { PrimaryButtonState.Completed diff --git a/paymentsheet/src/main/java/com/stripe/android/link/ui/wallet/WalletViewModel.kt b/paymentsheet/src/main/java/com/stripe/android/link/ui/wallet/WalletViewModel.kt index 92bfee9ea41..9d03698a7b3 100644 --- a/paymentsheet/src/main/java/com/stripe/android/link/ui/wallet/WalletViewModel.kt +++ b/paymentsheet/src/main/java/com/stripe/android/link/ui/wallet/WalletViewModel.kt @@ -5,31 +5,46 @@ import androidx.lifecycle.ViewModelProvider import androidx.lifecycle.viewModelScope import androidx.lifecycle.viewmodel.initializer import androidx.lifecycle.viewmodel.viewModelFactory +import com.stripe.android.common.exception.stripeErrorMessage import com.stripe.android.core.Logger import com.stripe.android.core.strings.resolvableString import com.stripe.android.link.LinkActivityResult import com.stripe.android.link.LinkConfiguration import com.stripe.android.link.LinkScreen import com.stripe.android.link.account.LinkAccountManager +import com.stripe.android.link.confirmation.LinkConfirmationHandler import com.stripe.android.link.injection.NativeLinkComponent import com.stripe.android.link.model.LinkAccount import com.stripe.android.link.model.supportedPaymentMethodTypes +import com.stripe.android.model.CardBrand import com.stripe.android.model.ConsumerPaymentDetails +import com.stripe.android.model.ConsumerPaymentDetailsUpdateParams import com.stripe.android.model.PaymentIntent +import com.stripe.android.model.PaymentMethod +import com.stripe.android.model.PaymentMethodCreateParams import com.stripe.android.model.SetupIntent import com.stripe.android.model.StripeIntent import com.stripe.android.ui.core.Amount +import com.stripe.android.ui.core.FieldValuesToParamsMapConverter import com.stripe.android.ui.core.R +import com.stripe.android.ui.core.elements.CvcController +import com.stripe.android.ui.core.elements.createExpiryDateFormFieldValues +import com.stripe.android.uicore.elements.DateConfig +import com.stripe.android.uicore.elements.SimpleTextFieldController +import com.stripe.android.uicore.utils.mapAsStateFlow import kotlinx.coroutines.flow.MutableStateFlow import kotlinx.coroutines.flow.StateFlow +import kotlinx.coroutines.flow.collectLatest import kotlinx.coroutines.flow.update import kotlinx.coroutines.launch import javax.inject.Inject +import com.stripe.android.link.confirmation.Result as LinkConfirmationResult internal class WalletViewModel @Inject constructor( private val configuration: LinkConfiguration, private val linkAccount: LinkAccount, private val linkAccountManager: LinkAccountManager, + private val linkConfirmationHandler: LinkConfirmationHandler, private val logger: Logger, private val navigate: (route: LinkScreen) -> Unit, private val navigateAndClearStack: (route: LinkScreen) -> Unit, @@ -49,8 +64,33 @@ internal class WalletViewModel @Inject constructor( val uiState: StateFlow = _uiState + val expiryDateController = SimpleTextFieldController( + textFieldConfig = DateConfig() + ) + val cvcController = CvcController( + cardBrandFlow = uiState.mapAsStateFlow { + (it.selectedItem as? ConsumerPaymentDetails.Card)?.brand ?: CardBrand.Unknown + } + ) + init { loadPaymentDetails() + + viewModelScope.launch { + expiryDateController.formFieldValue.collectLatest { input -> + _uiState.update { + it.copy(expiryDateInput = input) + } + } + } + + viewModelScope.launch { + cvcController.formFieldValue.collectLatest { input -> + _uiState.update { + it.copy(cvcInput = input) + } + } + } } private fun loadPaymentDetails() { @@ -85,12 +125,92 @@ internal class WalletViewModel @Inject constructor( fun onItemSelected(item: ConsumerPaymentDetails.PaymentDetails) { if (item == uiState.value.selectedItem) return + expiryDateController.onRawValueChange("") + cvcController.onRawValueChange("") + _uiState.update { it.copy(selectedItem = item) } } - fun onPrimaryButtonClicked() = Unit + fun onPrimaryButtonClicked() { + val paymentDetail = _uiState.value.selectedItem ?: return + _uiState.update { + it.copy(isProcessing = true) + } + + viewModelScope.launch { + performPaymentConfirmation(paymentDetail) + } + } + + private suspend fun performPaymentConfirmation( + selectedPaymentDetails: ConsumerPaymentDetails.PaymentDetails, + ) { + val card = selectedPaymentDetails as? ConsumerPaymentDetails.Card + val isExpired = card != null && card.isExpired + + if (isExpired) { + performPaymentDetailsUpdate(selectedPaymentDetails).fold( + onSuccess = { result -> + val updatedPaymentDetails = result.paymentDetails.single { + it.id == selectedPaymentDetails.id + } + performPaymentConfirmation(updatedPaymentDetails) + }, + onFailure = { error -> + _uiState.update { + it.copy( + alertMessage = error.stripeErrorMessage(), + isProcessing = false + ) + } + } + ) + } else { + // Confirm payment with LinkConfirmationHandler + performPaymentConfirmationWithCvc( + selectedPaymentDetails = selectedPaymentDetails, + cvc = cvcController.formFieldValue.value.takeIf { it.isComplete }?.value + ) + } + } + + private suspend fun performPaymentConfirmationWithCvc( + selectedPaymentDetails: ConsumerPaymentDetails.PaymentDetails, + cvc: String? + ) { + val result = linkConfirmationHandler.confirm( + paymentDetails = selectedPaymentDetails, + linkAccount = linkAccount, + cvc = cvc + ) + when (result) { + LinkConfirmationResult.Canceled -> Unit + is LinkConfirmationResult.Failed -> { + _uiState.update { + it.copy(errorMessage = result.message) + } + } + LinkConfirmationResult.Succeeded -> { + dismissWithResult(LinkActivityResult.Completed) + } + } + } + + private suspend fun performPaymentDetailsUpdate( + selectedPaymentDetails: ConsumerPaymentDetails.PaymentDetails + ): Result { + val paymentMethodCreateParams = uiState.value.toPaymentMethodCreateParams() + + val updateParams = ConsumerPaymentDetailsUpdateParams( + id = selectedPaymentDetails.id, + isDefault = selectedPaymentDetails.isDefault, + cardPaymentMethodCreateParamsMap = paymentMethodCreateParams.toParamMap() + ) + + return linkAccountManager.updatePaymentDetails(updateParams) + } fun onPayAnotherWayClicked() { dismissWithResult(LinkActivityResult.Canceled(LinkActivityResult.Canceled.Reason.PayAnotherWay)) @@ -130,6 +250,9 @@ internal class WalletViewModel @Inject constructor( WalletViewModel( configuration = parentComponent.configuration, linkAccountManager = parentComponent.linkAccountManager, + linkConfirmationHandler = parentComponent.linkConfirmationHandlerFactory.create( + confirmationHandler = parentComponent.viewModel.confirmationHandler + ), logger = parentComponent.logger, linkAccount = linkAccount, navigate = navigate, @@ -141,3 +264,12 @@ internal class WalletViewModel @Inject constructor( } } } + +private fun WalletUiState.toPaymentMethodCreateParams(): PaymentMethodCreateParams { + val expiryDateValues = createExpiryDateFormFieldValues(expiryDateInput) + return FieldValuesToParamsMapConverter.transformToPaymentMethodCreateParams( + fieldValuePairs = expiryDateValues, + code = PaymentMethod.Type.Card.code, + requiresMandate = false + ) +} diff --git a/paymentsheet/src/test/java/com/stripe/android/link/LinkActivityViewModelTest.kt b/paymentsheet/src/test/java/com/stripe/android/link/LinkActivityViewModelTest.kt index 22ec47f3415..fe9ad7f0c13 100644 --- a/paymentsheet/src/test/java/com/stripe/android/link/LinkActivityViewModelTest.kt +++ b/paymentsheet/src/test/java/com/stripe/android/link/LinkActivityViewModelTest.kt @@ -20,6 +20,7 @@ import com.google.common.truth.Truth.assertThat import com.stripe.android.link.account.FakeLinkAccountManager import com.stripe.android.link.account.LinkAccountManager import com.stripe.android.link.model.AccountStatus +import com.stripe.android.paymentelement.confirmation.FakeConfirmationHandler import com.stripe.android.testing.CoroutineTestRule import kotlinx.coroutines.test.UnconfinedTestDispatcher import kotlinx.coroutines.test.advanceUntilIdle @@ -317,6 +318,7 @@ internal class LinkActivityViewModelTest { return LinkActivityViewModel( linkAccountManager = linkAccountManager, activityRetainedComponent = mock(), + confirmationHandlerFactory = { FakeConfirmationHandler() } ).apply { this.navController = navController this.dismissWithResult = dismissWithResult diff --git a/paymentsheet/src/test/java/com/stripe/android/link/confirmation/FakeLinkConfirmationHandler.kt b/paymentsheet/src/test/java/com/stripe/android/link/confirmation/FakeLinkConfirmationHandler.kt index ea50e090bee..0cfb2f4dbbd 100644 --- a/paymentsheet/src/test/java/com/stripe/android/link/confirmation/FakeLinkConfirmationHandler.kt +++ b/paymentsheet/src/test/java/com/stripe/android/link/confirmation/FakeLinkConfirmationHandler.kt @@ -5,10 +5,26 @@ import com.stripe.android.model.ConsumerPaymentDetails internal class FakeLinkConfirmationHandler : LinkConfirmationHandler { var confirmResult: Result = Result.Succeeded + val calls = arrayListOf() override suspend fun confirm( paymentDetails: ConsumerPaymentDetails.PaymentDetails, linkAccount: LinkAccount, cvc: String? - ) = confirmResult + ): Result { + calls.add( + element = Call( + paymentDetails = paymentDetails, + linkAccount = linkAccount, + cvc = cvc + ) + ) + return confirmResult + } + + data class Call( + val paymentDetails: ConsumerPaymentDetails.PaymentDetails, + val linkAccount: LinkAccount, + val cvc: String? + ) } diff --git a/paymentsheet/src/test/java/com/stripe/android/link/ui/wallet/WalletScreenTest.kt b/paymentsheet/src/test/java/com/stripe/android/link/ui/wallet/WalletScreenTest.kt index b48396aac48..257d7ed9f19 100644 --- a/paymentsheet/src/test/java/com/stripe/android/link/ui/wallet/WalletScreenTest.kt +++ b/paymentsheet/src/test/java/com/stripe/android/link/ui/wallet/WalletScreenTest.kt @@ -26,6 +26,8 @@ import com.stripe.android.link.TestFactory import com.stripe.android.link.account.FakeLinkAccountManager import com.stripe.android.link.account.LinkAccountManager import com.stripe.android.link.ui.BottomSheetContent +import com.stripe.android.link.confirmation.FakeLinkConfirmationHandler +import com.stripe.android.link.confirmation.LinkConfirmationHandler import com.stripe.android.link.ui.PrimaryButtonTag import com.stripe.android.model.ConsumerPaymentDetails import com.stripe.android.testing.CoroutineTestRule @@ -349,12 +351,14 @@ internal class WalletScreenTest { } private fun createViewModel( - linkAccountManager: LinkAccountManager = FakeLinkAccountManager() + linkAccountManager: LinkAccountManager = FakeLinkAccountManager(), + linkConfirmationHandler: LinkConfirmationHandler = FakeLinkConfirmationHandler() ): WalletViewModel { return WalletViewModel( configuration = TestFactory.LINK_CONFIGURATION, linkAccount = TestFactory.LINK_ACCOUNT, linkAccountManager = linkAccountManager, + linkConfirmationHandler = linkConfirmationHandler, logger = FakeLogger(), navigate = {}, navigateAndClearStack = {}, diff --git a/paymentsheet/src/test/java/com/stripe/android/link/ui/wallet/WalletUiStateTest.kt b/paymentsheet/src/test/java/com/stripe/android/link/ui/wallet/WalletUiStateTest.kt index 575a1965748..9462055522c 100644 --- a/paymentsheet/src/test/java/com/stripe/android/link/ui/wallet/WalletUiStateTest.kt +++ b/paymentsheet/src/test/java/com/stripe/android/link/ui/wallet/WalletUiStateTest.kt @@ -5,6 +5,7 @@ import com.stripe.android.link.TestFactory import com.stripe.android.link.TestFactory.LINK_WALLET_PRIMARY_BUTTON_LABEL import com.stripe.android.link.ui.PrimaryButtonState import com.stripe.android.model.CvcCheck +import com.stripe.android.uicore.forms.FormFieldEntry import org.junit.Test class WalletUiStateTest { @@ -106,4 +107,94 @@ class WalletUiStateTest { assertThat(state.showBankAccountTerms).isFalse() } + + @Test + fun testDisabledButtonStateForExpiredCardWithIncompleteExpiryDate() { + val state = WalletUiState( + paymentDetailsList = TestFactory.CONSUMER_PAYMENT_DETAILS.paymentDetails, + selectedItem = TestFactory.CONSUMER_PAYMENT_DETAILS_CARD.copy(expiryYear = 1900), + hasCompleted = false, + isProcessing = false, + primaryButtonLabel = LINK_WALLET_PRIMARY_BUTTON_LABEL, + expiryDateInput = FormFieldEntry("", isComplete = false) + ) + + assertThat(state.primaryButtonState).isEqualTo(PrimaryButtonState.Disabled) + } + + @Test + fun testEnabledButtonStateForExpiredCardWithCompleteExpiryDateAndIncompleteCvc() { + val state = WalletUiState( + paymentDetailsList = TestFactory.CONSUMER_PAYMENT_DETAILS.paymentDetails, + selectedItem = TestFactory.CONSUMER_PAYMENT_DETAILS_CARD.copy(expiryYear = 1900), + hasCompleted = false, + isProcessing = false, + primaryButtonLabel = LINK_WALLET_PRIMARY_BUTTON_LABEL, + expiryDateInput = FormFieldEntry("12/25", isComplete = true), + ) + + assertThat(state.primaryButtonState).isEqualTo(PrimaryButtonState.Disabled) + } + + @Test + fun testEnabledButtonStateForExpiredCardWithCompleteExpiryDate() { + val state = WalletUiState( + paymentDetailsList = TestFactory.CONSUMER_PAYMENT_DETAILS.paymentDetails, + selectedItem = TestFactory.CONSUMER_PAYMENT_DETAILS_CARD.copy(expiryYear = 1900), + hasCompleted = false, + isProcessing = false, + primaryButtonLabel = LINK_WALLET_PRIMARY_BUTTON_LABEL, + expiryDateInput = FormFieldEntry("12/25", isComplete = true), + cvcInput = FormFieldEntry("123", isComplete = true) + ) + + assertThat(state.primaryButtonState).isEqualTo(PrimaryButtonState.Enabled) + } + + @Test + fun testDisabledButtonStateForCardRequiringCvcWithIncompleteCvc() { + val state = WalletUiState( + paymentDetailsList = TestFactory.CONSUMER_PAYMENT_DETAILS.paymentDetails, + selectedItem = TestFactory.CONSUMER_PAYMENT_DETAILS_CARD.copy( + cvcCheck = CvcCheck.Unchecked + ), + hasCompleted = false, + isProcessing = false, + primaryButtonLabel = LINK_WALLET_PRIMARY_BUTTON_LABEL, + cvcInput = FormFieldEntry("12", isComplete = false) + ) + + assertThat(state.primaryButtonState).isEqualTo(PrimaryButtonState.Disabled) + } + + @Test + fun testEnabledButtonStateForCardRequiringCvcWithCompleteCvc() { + val state = WalletUiState( + paymentDetailsList = TestFactory.CONSUMER_PAYMENT_DETAILS.paymentDetails, + selectedItem = TestFactory.CONSUMER_PAYMENT_DETAILS_CARD.copy( + cvcCheck = CvcCheck.Unchecked + ), + hasCompleted = false, + isProcessing = false, + primaryButtonLabel = LINK_WALLET_PRIMARY_BUTTON_LABEL, + cvcInput = FormFieldEntry("123", isComplete = true) + ) + + assertThat(state.primaryButtonState).isEqualTo(PrimaryButtonState.Enabled) + } + + @Test + fun testEnabledButtonStateForValidCardWithBothInputsComplete() { + val state = WalletUiState( + paymentDetailsList = TestFactory.CONSUMER_PAYMENT_DETAILS.paymentDetails, + selectedItem = TestFactory.CONSUMER_PAYMENT_DETAILS_CARD.copy(expiryYear = 2099), + hasCompleted = false, + isProcessing = false, + primaryButtonLabel = LINK_WALLET_PRIMARY_BUTTON_LABEL, + expiryDateInput = FormFieldEntry("12/25", isComplete = true), + cvcInput = FormFieldEntry("123", isComplete = true) + ) + + assertThat(state.primaryButtonState).isEqualTo(PrimaryButtonState.Enabled) + } } diff --git a/paymentsheet/src/test/java/com/stripe/android/link/ui/wallet/WalletViewModelTest.kt b/paymentsheet/src/test/java/com/stripe/android/link/ui/wallet/WalletViewModelTest.kt index ba41d73b8fd..9239d7d22bc 100644 --- a/paymentsheet/src/test/java/com/stripe/android/link/ui/wallet/WalletViewModelTest.kt +++ b/paymentsheet/src/test/java/com/stripe/android/link/ui/wallet/WalletViewModelTest.kt @@ -1,21 +1,35 @@ package com.stripe.android.link.ui.wallet import com.google.common.truth.Truth.assertThat +import com.stripe.android.common.exception.stripeErrorMessage import com.stripe.android.core.Logger +import com.stripe.android.core.strings.resolvableString import com.stripe.android.link.LinkActivityResult import com.stripe.android.link.LinkScreen import com.stripe.android.link.TestFactory import com.stripe.android.link.account.FakeLinkAccountManager import com.stripe.android.link.account.LinkAccountManager +import com.stripe.android.link.confirmation.FakeLinkConfirmationHandler +import com.stripe.android.link.confirmation.LinkConfirmationHandler import com.stripe.android.model.ConsumerPaymentDetails +import com.stripe.android.model.ConsumerPaymentDetailsUpdateParams import com.stripe.android.testing.CoroutineTestRule import com.stripe.android.testing.FakeLogger +import com.stripe.android.uicore.forms.FormFieldEntry import kotlinx.coroutines.test.UnconfinedTestDispatcher +import kotlinx.coroutines.test.advanceUntilIdle import kotlinx.coroutines.test.runTest import org.junit.Rule import org.junit.Test import org.junit.runner.RunWith import org.robolectric.RobolectricTestRunner +import kotlin.Result +import kotlin.RuntimeException +import kotlin.String +import kotlin.Throwable +import kotlin.Unit +import kotlin.to +import com.stripe.android.link.confirmation.Result as LinkConfirmationResult @RunWith(RobolectricTestRunner::class) class WalletViewModelTest { @@ -47,7 +61,9 @@ class WalletViewModelTest { selectedItem = TestFactory.CONSUMER_PAYMENT_DETAILS.paymentDetails.firstOrNull(), isProcessing = false, hasCompleted = false, - primaryButtonLabel = TestFactory.LINK_WALLET_PRIMARY_BUTTON_LABEL + primaryButtonLabel = TestFactory.LINK_WALLET_PRIMARY_BUTTON_LABEL, + expiryDateInput = FormFieldEntry(""), + cvcInput = FormFieldEntry("") ) ) } @@ -121,9 +137,295 @@ class WalletViewModelTest { assertThat(navScreen).isEqualTo(LinkScreen.PaymentMethod) } + fun `expiryDateController updates uiState when input changes`() = runTest(dispatcher) { + val viewModel = createViewModel() + + viewModel.expiryDateController.onRawValueChange("12") + advanceUntilIdle() + assertThat(viewModel.uiState.value.expiryDateInput).isEqualTo(FormFieldEntry("12", isComplete = false)) + + viewModel.expiryDateController.onRawValueChange("12/25") + assertThat(viewModel.uiState.value.expiryDateInput).isEqualTo(FormFieldEntry("1225", isComplete = true)) + } + + @Test + fun `cvcController updates uiState when input changes`() = runTest(dispatcher) { + val viewModel = createViewModel() + + viewModel.cvcController.onRawValueChange("12") + advanceUntilIdle() + assertThat(viewModel.uiState.value.cvcInput).isEqualTo(FormFieldEntry("12", isComplete = false)) + + viewModel.cvcController.onRawValueChange("123") + advanceUntilIdle() + assertThat(viewModel.uiState.value.cvcInput).isEqualTo(FormFieldEntry("123", isComplete = true)) + } + + @Test + fun `expiryDateController and cvcController reset when new item is selected`() = runTest(dispatcher) { + val viewModel = createViewModel() + + viewModel.expiryDateController.onRawValueChange("12/25") + viewModel.cvcController.onRawValueChange("123") + + assertThat(viewModel.uiState.value.expiryDateInput).isEqualTo(FormFieldEntry("1225", isComplete = true)) + assertThat(viewModel.uiState.value.cvcInput).isEqualTo(FormFieldEntry("123", isComplete = true)) + + val newCard = TestFactory.CONSUMER_PAYMENT_DETAILS_CARD.copy(id = "new_card_id") + viewModel.onItemSelected(newCard) + + assertThat(viewModel.uiState.value.expiryDateInput).isEqualTo(FormFieldEntry("", isComplete = false)) + assertThat(viewModel.uiState.value.cvcInput).isEqualTo(FormFieldEntry("", isComplete = false)) + } + + @Test + fun `performPaymentConfirmation updates expired card successfully`() = runTest(dispatcher) { + val expiredCard = TestFactory.CONSUMER_PAYMENT_DETAILS_CARD.copy(expiryYear = 1999) + val updatedCard = expiredCard.copy(expiryYear = 2099) + val linkAccountManager = object : FakeLinkAccountManager() { + var updateParamsUsed: ConsumerPaymentDetailsUpdateParams? = null + override suspend fun updatePaymentDetails( + updateParams: ConsumerPaymentDetailsUpdateParams + ): Result { + updateParamsUsed = updateParams + return Result.success( + ConsumerPaymentDetails(paymentDetails = listOf(updatedCard)) + ) + } + } + linkAccountManager.listPaymentDetailsResult = Result.success( + ConsumerPaymentDetails(paymentDetails = listOf(expiredCard)) + ) + val linkConfirmationHandler = FakeLinkConfirmationHandler() + + val viewModel = createViewModel( + linkAccountManager = linkAccountManager, + linkConfirmationHandler = linkConfirmationHandler + ) + viewModel.onItemSelected(expiredCard) + viewModel.expiryDateController.onRawValueChange("1299") + viewModel.cvcController.onRawValueChange("123") + + viewModel.onPrimaryButtonClicked() + + assertThat(viewModel.uiState.value.isProcessing).isTrue() + assertThat(viewModel.uiState.value.alertMessage).isNull() + + assertThat(linkAccountManager.updateParamsUsed?.id).isEqualTo(expiredCard.id) + val card = linkAccountManager.updateParamsUsed?.cardPaymentMethodCreateParamsMap + ?.get("card") as? Map + assertThat(card).isEqualTo( + mapOf( + "exp_month" to updatedCard.expiryMonth.toString(), + "exp_year" to updatedCard.expiryYear.toString() + ) + ) + + assertThat(linkConfirmationHandler.calls).containsExactly( + FakeLinkConfirmationHandler.Call( + paymentDetails = updatedCard, + linkAccount = TestFactory.LINK_ACCOUNT, + cvc = "123" + ) + ) + } + + @Test + fun `performPaymentConfirmation handles update failure`() = runTest(dispatcher) { + val error = RuntimeException("Update failed") + val expiredCard = TestFactory.CONSUMER_PAYMENT_DETAILS_CARD.copy(expiryYear = 1999) + val linkAccountManager = FakeLinkAccountManager() + linkAccountManager.listPaymentDetailsResult = Result.success( + ConsumerPaymentDetails(paymentDetails = listOf(expiredCard)) + ) + linkAccountManager.updatePaymentDetailsResult = Result.failure(error) + + val linkConfirmationHandler = FakeLinkConfirmationHandler() + + val viewModel = createViewModel( + linkAccountManager = linkAccountManager, + linkConfirmationHandler = linkConfirmationHandler + ) + viewModel.onItemSelected(expiredCard) + viewModel.expiryDateController.onRawValueChange("1225") + viewModel.cvcController.onRawValueChange("123") + + viewModel.onPrimaryButtonClicked() + + assertThat(viewModel.uiState.value.isProcessing).isFalse() + assertThat(viewModel.uiState.value.alertMessage).isEqualTo(error.stripeErrorMessage()) + + assertThat(linkConfirmationHandler.calls).isEmpty() + } + + @Test + fun `performPaymentConfirmation skips update for non-expired card`() = runTest(dispatcher) { + val validCard = TestFactory.CONSUMER_PAYMENT_DETAILS_CARD.copy(expiryYear = 2099) + val linkAccountManager = object : FakeLinkAccountManager() { + var updatePaymentDetailsCalls = 0 + override suspend fun updatePaymentDetails( + updateParams: ConsumerPaymentDetailsUpdateParams + ): Result { + updatePaymentDetailsCalls += 1 + return super.updatePaymentDetails(updateParams) + } + + override suspend fun listPaymentDetails(paymentMethodTypes: Set): Result { + return Result.success( + value = ConsumerPaymentDetails(paymentDetails = listOf(validCard)) + ) + } + } + + val linkConfirmationHandler = FakeLinkConfirmationHandler() + + val viewModel = createViewModel( + linkAccountManager = linkAccountManager, + linkConfirmationHandler = linkConfirmationHandler + ) + viewModel.onItemSelected(validCard) + + viewModel.onPrimaryButtonClicked() + + assertThat(linkAccountManager.updatePaymentDetailsCalls).isEqualTo(0) + + assertThat(linkConfirmationHandler.calls).containsExactly( + FakeLinkConfirmationHandler.Call( + paymentDetails = validCard, + linkAccount = TestFactory.LINK_ACCOUNT, + cvc = null + ) + ) + } + + @Test + fun `performPaymentConfirmation dismisses with Completed result on success`() = runTest(dispatcher) { + val validCard = TestFactory.CONSUMER_PAYMENT_DETAILS_CARD.copy(expiryYear = 2099) + val linkAccountManager = object : FakeLinkAccountManager() { + var updatePaymentDetailsCalls = 0 + override suspend fun updatePaymentDetails( + updateParams: ConsumerPaymentDetailsUpdateParams + ): Result { + updatePaymentDetailsCalls += 1 + return super.updatePaymentDetails(updateParams) + } + + override suspend fun listPaymentDetails(paymentMethodTypes: Set): Result { + return Result.success( + value = ConsumerPaymentDetails(paymentDetails = listOf(validCard)) + ) + } + } + + val linkConfirmationHandler = FakeLinkConfirmationHandler() + + var result: LinkActivityResult? = null + val viewModel = createViewModel( + linkAccountManager = linkAccountManager, + linkConfirmationHandler = linkConfirmationHandler, + dismissWithResult = { + result = it + } + ) + viewModel.onItemSelected(validCard) + + viewModel.onPrimaryButtonClicked() + + assertThat(linkAccountManager.updatePaymentDetailsCalls).isEqualTo(0) + + assertThat(linkConfirmationHandler.calls).containsExactly( + FakeLinkConfirmationHandler.Call( + paymentDetails = validCard, + cvc = null, + linkAccount = TestFactory.LINK_ACCOUNT + ) + ) + + assertThat(result).isEqualTo(LinkActivityResult.Completed) + } + + @Test + fun `performPaymentConfirmation displays error on failure result`() = runTest(dispatcher) { + val confirmationResult = LinkConfirmationResult.Failed("oops".resolvableString) + val validCard = TestFactory.CONSUMER_PAYMENT_DETAILS_CARD.copy(expiryYear = 2099) + val linkAccountManager = object : FakeLinkAccountManager() { + var updatePaymentDetailsCalls = 0 + override suspend fun updatePaymentDetails( + updateParams: ConsumerPaymentDetailsUpdateParams + ): Result { + updatePaymentDetailsCalls += 1 + return super.updatePaymentDetails(updateParams) + } + + override suspend fun listPaymentDetails(paymentMethodTypes: Set): Result { + return Result.success( + value = ConsumerPaymentDetails(paymentDetails = listOf(validCard)) + ) + } + } + + val linkConfirmationHandler = FakeLinkConfirmationHandler() + linkConfirmationHandler.confirmResult = confirmationResult + + var result: LinkActivityResult? = null + val viewModel = createViewModel( + linkAccountManager = linkAccountManager, + linkConfirmationHandler = linkConfirmationHandler, + dismissWithResult = { + result = it + } + ) + viewModel.onItemSelected(validCard) + + viewModel.onPrimaryButtonClicked() + + assertThat(viewModel.uiState.value.errorMessage).isEqualTo(confirmationResult.message) + assertThat(result).isNull() + } + + @Test + fun `performPaymentConfirmation does nothing on canceled result`() = runTest(dispatcher) { + val confirmationResult = LinkConfirmationResult.Canceled + val validCard = TestFactory.CONSUMER_PAYMENT_DETAILS_CARD.copy(expiryYear = 2099) + val linkAccountManager = object : FakeLinkAccountManager() { + var updatePaymentDetailsCalls = 0 + override suspend fun updatePaymentDetails( + updateParams: ConsumerPaymentDetailsUpdateParams + ): Result { + updatePaymentDetailsCalls += 1 + return super.updatePaymentDetails(updateParams) + } + + override suspend fun listPaymentDetails(paymentMethodTypes: Set): Result { + return Result.success( + value = ConsumerPaymentDetails(paymentDetails = listOf(validCard)) + ) + } + } + + val linkConfirmationHandler = FakeLinkConfirmationHandler() + linkConfirmationHandler.confirmResult = confirmationResult + + var result: LinkActivityResult? = null + val viewModel = createViewModel( + linkAccountManager = linkAccountManager, + linkConfirmationHandler = linkConfirmationHandler, + dismissWithResult = { + result = it + } + ) + viewModel.onItemSelected(validCard) + + viewModel.onPrimaryButtonClicked() + + assertThat(viewModel.uiState.value.errorMessage).isNull() + assertThat(result).isNull() + } + private fun createViewModel( linkAccountManager: LinkAccountManager = FakeLinkAccountManager(), logger: Logger = FakeLogger(), + linkConfirmationHandler: LinkConfirmationHandler = FakeLinkConfirmationHandler(), navigate: (route: LinkScreen) -> Unit = {}, navigateAndClearStack: (route: LinkScreen) -> Unit = {}, dismissWithResult: (LinkActivityResult) -> Unit = {} @@ -132,6 +434,7 @@ class WalletViewModelTest { configuration = TestFactory.LINK_CONFIGURATION, linkAccount = TestFactory.LINK_ACCOUNT, linkAccountManager = linkAccountManager, + linkConfirmationHandler = linkConfirmationHandler, logger = logger, navigate = navigate, navigateAndClearStack = navigateAndClearStack, From 713809e75f7a0755bb1ed92c2ca32fd17c3dab77 Mon Sep 17 00:00:00 2001 From: toluo-stripe Date: Mon, 13 Jan 2025 15:27:55 -0500 Subject: [PATCH 2/6] Update tests to check isProcessing --- .../com/stripe/android/link/ui/wallet/WalletViewModel.kt | 5 ++++- .../com/stripe/android/link/ui/wallet/WalletViewModelTest.kt | 1 + 2 files changed, 5 insertions(+), 1 deletion(-) diff --git a/paymentsheet/src/main/java/com/stripe/android/link/ui/wallet/WalletViewModel.kt b/paymentsheet/src/main/java/com/stripe/android/link/ui/wallet/WalletViewModel.kt index 9d03698a7b3..d3ea2b90b07 100644 --- a/paymentsheet/src/main/java/com/stripe/android/link/ui/wallet/WalletViewModel.kt +++ b/paymentsheet/src/main/java/com/stripe/android/link/ui/wallet/WalletViewModel.kt @@ -189,7 +189,10 @@ internal class WalletViewModel @Inject constructor( LinkConfirmationResult.Canceled -> Unit is LinkConfirmationResult.Failed -> { _uiState.update { - it.copy(errorMessage = result.message) + it.copy( + errorMessage = result.message, + isProcessing = false + ) } } LinkConfirmationResult.Succeeded -> { diff --git a/paymentsheet/src/test/java/com/stripe/android/link/ui/wallet/WalletViewModelTest.kt b/paymentsheet/src/test/java/com/stripe/android/link/ui/wallet/WalletViewModelTest.kt index 9239d7d22bc..a5643252408 100644 --- a/paymentsheet/src/test/java/com/stripe/android/link/ui/wallet/WalletViewModelTest.kt +++ b/paymentsheet/src/test/java/com/stripe/android/link/ui/wallet/WalletViewModelTest.kt @@ -380,6 +380,7 @@ class WalletViewModelTest { viewModel.onPrimaryButtonClicked() assertThat(viewModel.uiState.value.errorMessage).isEqualTo(confirmationResult.message) + assertThat(viewModel.uiState.value.isProcessing).isFalse() assertThat(result).isNull() } From b90d319ad192fdc2d6f719e6fd1f71bd1f396380 Mon Sep 17 00:00:00 2001 From: toluo-stripe Date: Mon, 13 Jan 2025 16:17:23 -0500 Subject: [PATCH 3/6] Fix unchecked cast warning --- .../com/stripe/android/link/ui/wallet/WalletViewModelTest.kt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/paymentsheet/src/test/java/com/stripe/android/link/ui/wallet/WalletViewModelTest.kt b/paymentsheet/src/test/java/com/stripe/android/link/ui/wallet/WalletViewModelTest.kt index a5643252408..3fd69f25d2a 100644 --- a/paymentsheet/src/test/java/com/stripe/android/link/ui/wallet/WalletViewModelTest.kt +++ b/paymentsheet/src/test/java/com/stripe/android/link/ui/wallet/WalletViewModelTest.kt @@ -213,7 +213,7 @@ class WalletViewModelTest { assertThat(linkAccountManager.updateParamsUsed?.id).isEqualTo(expiredCard.id) val card = linkAccountManager.updateParamsUsed?.cardPaymentMethodCreateParamsMap - ?.get("card") as? Map + ?.get("card") as? Map<*, *> assertThat(card).isEqualTo( mapOf( "exp_month" to updatedCard.expiryMonth.toString(), From ec809a374edd43df2fd058e3009fb8fa0b66705c Mon Sep 17 00:00:00 2001 From: toluo-stripe Date: Tue, 14 Jan 2025 01:42:50 -0500 Subject: [PATCH 4/6] Clean up tests --- .../link/ui/wallet/WalletScreenTest.kt | 2 +- .../link/ui/wallet/WalletUiStateTest.kt | 110 +++++--------- .../link/ui/wallet/WalletViewModelTest.kt | 140 ++++++------------ 3 files changed, 87 insertions(+), 165 deletions(-) diff --git a/paymentsheet/src/test/java/com/stripe/android/link/ui/wallet/WalletScreenTest.kt b/paymentsheet/src/test/java/com/stripe/android/link/ui/wallet/WalletScreenTest.kt index 257d7ed9f19..eff7f976530 100644 --- a/paymentsheet/src/test/java/com/stripe/android/link/ui/wallet/WalletScreenTest.kt +++ b/paymentsheet/src/test/java/com/stripe/android/link/ui/wallet/WalletScreenTest.kt @@ -25,9 +25,9 @@ import com.stripe.android.core.strings.resolvableString import com.stripe.android.link.TestFactory import com.stripe.android.link.account.FakeLinkAccountManager import com.stripe.android.link.account.LinkAccountManager -import com.stripe.android.link.ui.BottomSheetContent import com.stripe.android.link.confirmation.FakeLinkConfirmationHandler import com.stripe.android.link.confirmation.LinkConfirmationHandler +import com.stripe.android.link.ui.BottomSheetContent import com.stripe.android.link.ui.PrimaryButtonTag import com.stripe.android.model.ConsumerPaymentDetails import com.stripe.android.testing.CoroutineTestRule diff --git a/paymentsheet/src/test/java/com/stripe/android/link/ui/wallet/WalletUiStateTest.kt b/paymentsheet/src/test/java/com/stripe/android/link/ui/wallet/WalletUiStateTest.kt index 9462055522c..56b8ca0c134 100644 --- a/paymentsheet/src/test/java/com/stripe/android/link/ui/wallet/WalletUiStateTest.kt +++ b/paymentsheet/src/test/java/com/stripe/android/link/ui/wallet/WalletUiStateTest.kt @@ -1,9 +1,11 @@ package com.stripe.android.link.ui.wallet import com.google.common.truth.Truth.assertThat +import com.stripe.android.core.strings.ResolvableString import com.stripe.android.link.TestFactory import com.stripe.android.link.TestFactory.LINK_WALLET_PRIMARY_BUTTON_LABEL import com.stripe.android.link.ui.PrimaryButtonState +import com.stripe.android.model.ConsumerPaymentDetails import com.stripe.android.model.CvcCheck import com.stripe.android.uicore.forms.FormFieldEntry import org.junit.Test @@ -12,12 +14,9 @@ class WalletUiStateTest { @Test fun testCompletedButtonState() { - val state = WalletUiState( - paymentDetailsList = TestFactory.CONSUMER_PAYMENT_DETAILS.paymentDetails, + val state = walletUiState( selectedItem = TestFactory.CONSUMER_PAYMENT_DETAILS.paymentDetails.firstOrNull(), hasCompleted = true, - isProcessing = false, - primaryButtonLabel = LINK_WALLET_PRIMARY_BUTTON_LABEL ) assertThat(state.primaryButtonState).isEqualTo(PrimaryButtonState.Completed) @@ -25,12 +24,9 @@ class WalletUiStateTest { @Test fun testProcessingButtonState() { - val state = WalletUiState( - paymentDetailsList = TestFactory.CONSUMER_PAYMENT_DETAILS.paymentDetails, + val state = walletUiState( selectedItem = TestFactory.CONSUMER_PAYMENT_DETAILS.paymentDetails.firstOrNull(), - hasCompleted = false, - isProcessing = true, - primaryButtonLabel = LINK_WALLET_PRIMARY_BUTTON_LABEL + isProcessing = true ) assertThat(state.primaryButtonState).isEqualTo(PrimaryButtonState.Processing) @@ -38,15 +34,11 @@ class WalletUiStateTest { @Test fun testDisabledButtonStateForExpiredCard() { - val state = WalletUiState( - paymentDetailsList = TestFactory.CONSUMER_PAYMENT_DETAILS.paymentDetails, + val state = walletUiState( selectedItem = TestFactory.CONSUMER_PAYMENT_DETAILS_CARD .copy( expiryYear = 1900 ), - hasCompleted = false, - isProcessing = false, - primaryButtonLabel = LINK_WALLET_PRIMARY_BUTTON_LABEL ) assertThat(state.primaryButtonState).isEqualTo(PrimaryButtonState.Disabled) @@ -54,15 +46,11 @@ class WalletUiStateTest { @Test fun testDisabledButtonStateForCvcRecollection() { - val state = WalletUiState( - paymentDetailsList = TestFactory.CONSUMER_PAYMENT_DETAILS.paymentDetails, + val state = walletUiState( selectedItem = TestFactory.CONSUMER_PAYMENT_DETAILS_CARD .copy( cvcCheck = CvcCheck.Unchecked - ), - hasCompleted = false, - isProcessing = false, - primaryButtonLabel = LINK_WALLET_PRIMARY_BUTTON_LABEL + ) ) assertThat(state.primaryButtonState).isEqualTo(PrimaryButtonState.Disabled) @@ -70,13 +58,8 @@ class WalletUiStateTest { @Test fun testEnabledButtonState() { - val state = WalletUiState( - paymentDetailsList = TestFactory.CONSUMER_PAYMENT_DETAILS.paymentDetails, - selectedItem = TestFactory.CONSUMER_PAYMENT_DETAILS_CARD - .copy(expiryYear = 2099), - hasCompleted = false, - isProcessing = false, - primaryButtonLabel = LINK_WALLET_PRIMARY_BUTTON_LABEL + val state = walletUiState( + selectedItem = TestFactory.CONSUMER_PAYMENT_DETAILS_CARD.copy(expiryYear = 2099), ) assertThat(state.primaryButtonState).isEqualTo(PrimaryButtonState.Enabled) @@ -84,12 +67,8 @@ class WalletUiStateTest { @Test fun testShowBankAccountTermsForSelectedBankPaymentMethod() { - val state = WalletUiState( - paymentDetailsList = TestFactory.CONSUMER_PAYMENT_DETAILS.paymentDetails, - selectedItem = TestFactory.CONSUMER_PAYMENT_DETAILS_BANK_ACCOUNT, - hasCompleted = false, - isProcessing = false, - primaryButtonLabel = LINK_WALLET_PRIMARY_BUTTON_LABEL + val state = walletUiState( + selectedItem = TestFactory.CONSUMER_PAYMENT_DETAILS_BANK_ACCOUNT ) assertThat(state.showBankAccountTerms).isTrue() @@ -97,25 +76,15 @@ class WalletUiStateTest { @Test fun testNoBankAccountTermsForSelectedNonBankPaymentMethod() { - val state = WalletUiState( - paymentDetailsList = TestFactory.CONSUMER_PAYMENT_DETAILS.paymentDetails, - selectedItem = TestFactory.CONSUMER_PAYMENT_DETAILS_CARD, - hasCompleted = false, - isProcessing = false, - primaryButtonLabel = LINK_WALLET_PRIMARY_BUTTON_LABEL - ) + val state = walletUiState() assertThat(state.showBankAccountTerms).isFalse() } @Test fun testDisabledButtonStateForExpiredCardWithIncompleteExpiryDate() { - val state = WalletUiState( - paymentDetailsList = TestFactory.CONSUMER_PAYMENT_DETAILS.paymentDetails, + val state = walletUiState( selectedItem = TestFactory.CONSUMER_PAYMENT_DETAILS_CARD.copy(expiryYear = 1900), - hasCompleted = false, - isProcessing = false, - primaryButtonLabel = LINK_WALLET_PRIMARY_BUTTON_LABEL, expiryDateInput = FormFieldEntry("", isComplete = false) ) @@ -124,12 +93,8 @@ class WalletUiStateTest { @Test fun testEnabledButtonStateForExpiredCardWithCompleteExpiryDateAndIncompleteCvc() { - val state = WalletUiState( - paymentDetailsList = TestFactory.CONSUMER_PAYMENT_DETAILS.paymentDetails, + val state = walletUiState( selectedItem = TestFactory.CONSUMER_PAYMENT_DETAILS_CARD.copy(expiryYear = 1900), - hasCompleted = false, - isProcessing = false, - primaryButtonLabel = LINK_WALLET_PRIMARY_BUTTON_LABEL, expiryDateInput = FormFieldEntry("12/25", isComplete = true), ) @@ -138,12 +103,8 @@ class WalletUiStateTest { @Test fun testEnabledButtonStateForExpiredCardWithCompleteExpiryDate() { - val state = WalletUiState( - paymentDetailsList = TestFactory.CONSUMER_PAYMENT_DETAILS.paymentDetails, + val state = walletUiState( selectedItem = TestFactory.CONSUMER_PAYMENT_DETAILS_CARD.copy(expiryYear = 1900), - hasCompleted = false, - isProcessing = false, - primaryButtonLabel = LINK_WALLET_PRIMARY_BUTTON_LABEL, expiryDateInput = FormFieldEntry("12/25", isComplete = true), cvcInput = FormFieldEntry("123", isComplete = true) ) @@ -153,14 +114,10 @@ class WalletUiStateTest { @Test fun testDisabledButtonStateForCardRequiringCvcWithIncompleteCvc() { - val state = WalletUiState( - paymentDetailsList = TestFactory.CONSUMER_PAYMENT_DETAILS.paymentDetails, + val state = walletUiState( selectedItem = TestFactory.CONSUMER_PAYMENT_DETAILS_CARD.copy( cvcCheck = CvcCheck.Unchecked ), - hasCompleted = false, - isProcessing = false, - primaryButtonLabel = LINK_WALLET_PRIMARY_BUTTON_LABEL, cvcInput = FormFieldEntry("12", isComplete = false) ) @@ -169,14 +126,10 @@ class WalletUiStateTest { @Test fun testEnabledButtonStateForCardRequiringCvcWithCompleteCvc() { - val state = WalletUiState( - paymentDetailsList = TestFactory.CONSUMER_PAYMENT_DETAILS.paymentDetails, + val state = walletUiState( selectedItem = TestFactory.CONSUMER_PAYMENT_DETAILS_CARD.copy( cvcCheck = CvcCheck.Unchecked ), - hasCompleted = false, - isProcessing = false, - primaryButtonLabel = LINK_WALLET_PRIMARY_BUTTON_LABEL, cvcInput = FormFieldEntry("123", isComplete = true) ) @@ -185,16 +138,33 @@ class WalletUiStateTest { @Test fun testEnabledButtonStateForValidCardWithBothInputsComplete() { - val state = WalletUiState( - paymentDetailsList = TestFactory.CONSUMER_PAYMENT_DETAILS.paymentDetails, + val state = walletUiState( selectedItem = TestFactory.CONSUMER_PAYMENT_DETAILS_CARD.copy(expiryYear = 2099), - hasCompleted = false, - isProcessing = false, - primaryButtonLabel = LINK_WALLET_PRIMARY_BUTTON_LABEL, expiryDateInput = FormFieldEntry("12/25", isComplete = true), cvcInput = FormFieldEntry("123", isComplete = true) ) assertThat(state.primaryButtonState).isEqualTo(PrimaryButtonState.Enabled) } + + private fun walletUiState( + paymentDetailsList: List = + TestFactory.CONSUMER_PAYMENT_DETAILS.paymentDetails, + selectedItem: ConsumerPaymentDetails.PaymentDetails? = TestFactory.CONSUMER_PAYMENT_DETAILS_CARD, + hasCompleted: Boolean = false, + isProcessing: Boolean = false, + primaryButtonLabel: ResolvableString = LINK_WALLET_PRIMARY_BUTTON_LABEL, + expiryDateInput: FormFieldEntry = FormFieldEntry(null), + cvcInput: FormFieldEntry = FormFieldEntry(null) + ): WalletUiState { + return WalletUiState( + paymentDetailsList = paymentDetailsList, + selectedItem = selectedItem, + hasCompleted = hasCompleted, + isProcessing = isProcessing, + primaryButtonLabel = primaryButtonLabel, + expiryDateInput = expiryDateInput, + cvcInput = cvcInput + ) + } } diff --git a/paymentsheet/src/test/java/com/stripe/android/link/ui/wallet/WalletViewModelTest.kt b/paymentsheet/src/test/java/com/stripe/android/link/ui/wallet/WalletViewModelTest.kt index 3fd69f25d2a..91679ccc549 100644 --- a/paymentsheet/src/test/java/com/stripe/android/link/ui/wallet/WalletViewModelTest.kt +++ b/paymentsheet/src/test/java/com/stripe/android/link/ui/wallet/WalletViewModelTest.kt @@ -8,7 +8,6 @@ import com.stripe.android.link.LinkActivityResult import com.stripe.android.link.LinkScreen import com.stripe.android.link.TestFactory import com.stripe.android.link.account.FakeLinkAccountManager -import com.stripe.android.link.account.LinkAccountManager import com.stripe.android.link.confirmation.FakeLinkConfirmationHandler import com.stripe.android.link.confirmation.LinkConfirmationHandler import com.stripe.android.model.ConsumerPaymentDetails @@ -24,11 +23,6 @@ import org.junit.Test import org.junit.runner.RunWith import org.robolectric.RobolectricTestRunner import kotlin.Result -import kotlin.RuntimeException -import kotlin.String -import kotlin.Throwable -import kotlin.Unit -import kotlin.to import com.stripe.android.link.confirmation.Result as LinkConfirmationResult @RunWith(RobolectricTestRunner::class) @@ -40,20 +34,14 @@ class WalletViewModelTest { @Test fun `viewmodel should load payment methods on init`() = runTest(dispatcher) { - val linkAccountManager = object : FakeLinkAccountManager() { - var paymentMethodTypes: Set? = null - override suspend fun listPaymentDetails(paymentMethodTypes: Set): Result { - this.paymentMethodTypes = paymentMethodTypes - return super.listPaymentDetails(paymentMethodTypes) - } - } + val linkAccountManager = WalletLinkAccountManager() val viewModel = createViewModel( linkAccountManager = linkAccountManager ) - assertThat(linkAccountManager.paymentMethodTypes) - .containsExactlyElementsIn(TestFactory.LINK_CONFIGURATION.stripeIntent.paymentMethodTypes) + assertThat(linkAccountManager.listPaymentDetailsCalls) + .containsExactly(TestFactory.LINK_CONFIGURATION.stripeIntent.paymentMethodTypes.toSet()) assertThat(viewModel.uiState.value).isEqualTo( WalletUiState( @@ -71,7 +59,7 @@ class WalletViewModelTest { @Test fun `viewmodel should dismiss with failure on load payment method failure`() = runTest(dispatcher) { val error = Throwable("oops") - val linkAccountManager = FakeLinkAccountManager() + val linkAccountManager = WalletLinkAccountManager() linkAccountManager.listPaymentDetailsResult = Result.failure(error) var linkActivityResult: LinkActivityResult? = null @@ -93,7 +81,7 @@ class WalletViewModelTest { @Test fun `viewmodel should open payment method screen when none is available`() = runTest(dispatcher) { - val linkAccountManager = FakeLinkAccountManager() + val linkAccountManager = WalletLinkAccountManager() linkAccountManager.listPaymentDetailsResult = Result.success(ConsumerPaymentDetails(emptyList())) var navScreen: LinkScreen? = null @@ -182,17 +170,10 @@ class WalletViewModelTest { fun `performPaymentConfirmation updates expired card successfully`() = runTest(dispatcher) { val expiredCard = TestFactory.CONSUMER_PAYMENT_DETAILS_CARD.copy(expiryYear = 1999) val updatedCard = expiredCard.copy(expiryYear = 2099) - val linkAccountManager = object : FakeLinkAccountManager() { - var updateParamsUsed: ConsumerPaymentDetailsUpdateParams? = null - override suspend fun updatePaymentDetails( - updateParams: ConsumerPaymentDetailsUpdateParams - ): Result { - updateParamsUsed = updateParams - return Result.success( - ConsumerPaymentDetails(paymentDetails = listOf(updatedCard)) - ) - } - } + val linkAccountManager = WalletLinkAccountManager() + linkAccountManager.updatePaymentDetailsResult = Result.success( + ConsumerPaymentDetails(paymentDetails = listOf(updatedCard)) + ) linkAccountManager.listPaymentDetailsResult = Result.success( ConsumerPaymentDetails(paymentDetails = listOf(expiredCard)) ) @@ -211,9 +192,11 @@ class WalletViewModelTest { assertThat(viewModel.uiState.value.isProcessing).isTrue() assertThat(viewModel.uiState.value.alertMessage).isNull() - assertThat(linkAccountManager.updateParamsUsed?.id).isEqualTo(expiredCard.id) - val card = linkAccountManager.updateParamsUsed?.cardPaymentMethodCreateParamsMap + val updateParamsUsed = linkAccountManager.updatePaymentDetailsCalls.firstOrNull() + val card = updateParamsUsed?.cardPaymentMethodCreateParamsMap ?.get("card") as? Map<*, *> + + assertThat(updateParamsUsed?.id).isEqualTo(expiredCard.id) assertThat(card).isEqualTo( mapOf( "exp_month" to updatedCard.expiryMonth.toString(), @@ -234,7 +217,7 @@ class WalletViewModelTest { fun `performPaymentConfirmation handles update failure`() = runTest(dispatcher) { val error = RuntimeException("Update failed") val expiredCard = TestFactory.CONSUMER_PAYMENT_DETAILS_CARD.copy(expiryYear = 1999) - val linkAccountManager = FakeLinkAccountManager() + val linkAccountManager = WalletLinkAccountManager() linkAccountManager.listPaymentDetailsResult = Result.success( ConsumerPaymentDetails(paymentDetails = listOf(expiredCard)) ) @@ -261,21 +244,7 @@ class WalletViewModelTest { @Test fun `performPaymentConfirmation skips update for non-expired card`() = runTest(dispatcher) { val validCard = TestFactory.CONSUMER_PAYMENT_DETAILS_CARD.copy(expiryYear = 2099) - val linkAccountManager = object : FakeLinkAccountManager() { - var updatePaymentDetailsCalls = 0 - override suspend fun updatePaymentDetails( - updateParams: ConsumerPaymentDetailsUpdateParams - ): Result { - updatePaymentDetailsCalls += 1 - return super.updatePaymentDetails(updateParams) - } - - override suspend fun listPaymentDetails(paymentMethodTypes: Set): Result { - return Result.success( - value = ConsumerPaymentDetails(paymentDetails = listOf(validCard)) - ) - } - } + val linkAccountManager = WalletLinkAccountManager() val linkConfirmationHandler = FakeLinkConfirmationHandler() @@ -287,7 +256,7 @@ class WalletViewModelTest { viewModel.onPrimaryButtonClicked() - assertThat(linkAccountManager.updatePaymentDetailsCalls).isEqualTo(0) + assertThat(linkAccountManager.updatePaymentDetailsCalls).isEmpty() assertThat(linkConfirmationHandler.calls).containsExactly( FakeLinkConfirmationHandler.Call( @@ -301,21 +270,10 @@ class WalletViewModelTest { @Test fun `performPaymentConfirmation dismisses with Completed result on success`() = runTest(dispatcher) { val validCard = TestFactory.CONSUMER_PAYMENT_DETAILS_CARD.copy(expiryYear = 2099) - val linkAccountManager = object : FakeLinkAccountManager() { - var updatePaymentDetailsCalls = 0 - override suspend fun updatePaymentDetails( - updateParams: ConsumerPaymentDetailsUpdateParams - ): Result { - updatePaymentDetailsCalls += 1 - return super.updatePaymentDetails(updateParams) - } - - override suspend fun listPaymentDetails(paymentMethodTypes: Set): Result { - return Result.success( - value = ConsumerPaymentDetails(paymentDetails = listOf(validCard)) - ) - } - } + val linkAccountManager = WalletLinkAccountManager() + linkAccountManager.listPaymentDetailsResult = Result.success( + value = ConsumerPaymentDetails(paymentDetails = listOf(validCard)) + ) val linkConfirmationHandler = FakeLinkConfirmationHandler() @@ -331,7 +289,7 @@ class WalletViewModelTest { viewModel.onPrimaryButtonClicked() - assertThat(linkAccountManager.updatePaymentDetailsCalls).isEqualTo(0) + assertThat(linkAccountManager.updatePaymentDetailsCalls).isEmpty() assertThat(linkConfirmationHandler.calls).containsExactly( FakeLinkConfirmationHandler.Call( @@ -348,21 +306,10 @@ class WalletViewModelTest { fun `performPaymentConfirmation displays error on failure result`() = runTest(dispatcher) { val confirmationResult = LinkConfirmationResult.Failed("oops".resolvableString) val validCard = TestFactory.CONSUMER_PAYMENT_DETAILS_CARD.copy(expiryYear = 2099) - val linkAccountManager = object : FakeLinkAccountManager() { - var updatePaymentDetailsCalls = 0 - override suspend fun updatePaymentDetails( - updateParams: ConsumerPaymentDetailsUpdateParams - ): Result { - updatePaymentDetailsCalls += 1 - return super.updatePaymentDetails(updateParams) - } - - override suspend fun listPaymentDetails(paymentMethodTypes: Set): Result { - return Result.success( - value = ConsumerPaymentDetails(paymentDetails = listOf(validCard)) - ) - } - } + val linkAccountManager = WalletLinkAccountManager() + linkAccountManager.listPaymentDetailsResult = Result.success( + value = ConsumerPaymentDetails(paymentDetails = listOf(validCard)) + ) val linkConfirmationHandler = FakeLinkConfirmationHandler() linkConfirmationHandler.confirmResult = confirmationResult @@ -388,21 +335,10 @@ class WalletViewModelTest { fun `performPaymentConfirmation does nothing on canceled result`() = runTest(dispatcher) { val confirmationResult = LinkConfirmationResult.Canceled val validCard = TestFactory.CONSUMER_PAYMENT_DETAILS_CARD.copy(expiryYear = 2099) - val linkAccountManager = object : FakeLinkAccountManager() { - var updatePaymentDetailsCalls = 0 - override suspend fun updatePaymentDetails( - updateParams: ConsumerPaymentDetailsUpdateParams - ): Result { - updatePaymentDetailsCalls += 1 - return super.updatePaymentDetails(updateParams) - } - - override suspend fun listPaymentDetails(paymentMethodTypes: Set): Result { - return Result.success( - value = ConsumerPaymentDetails(paymentDetails = listOf(validCard)) - ) - } - } + val linkAccountManager = WalletLinkAccountManager() + linkAccountManager.listPaymentDetailsResult = Result.success( + value = ConsumerPaymentDetails(paymentDetails = listOf(validCard)) + ) val linkConfirmationHandler = FakeLinkConfirmationHandler() linkConfirmationHandler.confirmResult = confirmationResult @@ -424,7 +360,7 @@ class WalletViewModelTest { } private fun createViewModel( - linkAccountManager: LinkAccountManager = FakeLinkAccountManager(), + linkAccountManager: WalletLinkAccountManager = WalletLinkAccountManager(), logger: Logger = FakeLogger(), linkConfirmationHandler: LinkConfirmationHandler = FakeLinkConfirmationHandler(), navigate: (route: LinkScreen) -> Unit = {}, @@ -443,3 +379,19 @@ class WalletViewModelTest { ) } } + +private class WalletLinkAccountManager : FakeLinkAccountManager() { + val listPaymentDetailsCalls = arrayListOf>() + val updatePaymentDetailsCalls = arrayListOf() + override suspend fun listPaymentDetails(paymentMethodTypes: Set): Result { + listPaymentDetailsCalls.add(paymentMethodTypes) + return super.listPaymentDetails(paymentMethodTypes) + } + + override suspend fun updatePaymentDetails( + updateParams: ConsumerPaymentDetailsUpdateParams + ): Result { + updatePaymentDetailsCalls.add(updateParams) + return super.updatePaymentDetails(updateParams) + } +} From 7a3ff16a002ae50175add87597dc42c7b55a3d10 Mon Sep 17 00:00:00 2001 From: toluo-stripe Date: Tue, 14 Jan 2025 01:46:42 -0500 Subject: [PATCH 5/6] Move createExpiryDateFormFieldValues to utility class --- .../ui/core/elements/CardDetailsElement.kt | 38 +--------------- .../ui/core/elements/CardDetailsUtil.kt | 45 +++++++++++++++++++ .../android/link/ui/wallet/WalletViewModel.kt | 2 +- 3 files changed, 48 insertions(+), 37 deletions(-) create mode 100644 payments-ui-core/src/main/java/com/stripe/android/ui/core/elements/CardDetailsUtil.kt diff --git a/payments-ui-core/src/main/java/com/stripe/android/ui/core/elements/CardDetailsElement.kt b/payments-ui-core/src/main/java/com/stripe/android/ui/core/elements/CardDetailsElement.kt index e1c1ee1c3c4..4f91c2cb1f2 100644 --- a/payments-ui-core/src/main/java/com/stripe/android/ui/core/elements/CardDetailsElement.kt +++ b/payments-ui-core/src/main/java/com/stripe/android/ui/core/elements/CardDetailsElement.kt @@ -7,6 +7,8 @@ import com.stripe.android.cards.CardAccountRangeRepository import com.stripe.android.core.strings.ResolvableString import com.stripe.android.model.CardBrand import com.stripe.android.ui.core.cbc.CardBrandChoiceEligibility +import com.stripe.android.ui.core.elements.CardDetailsUtil.getExpiryMonthFormFieldEntry +import com.stripe.android.ui.core.elements.CardDetailsUtil.getExpiryYearFormFieldEntry import com.stripe.android.uicore.elements.IdentifierSpec import com.stripe.android.uicore.elements.SectionFieldErrorController import com.stripe.android.uicore.elements.SectionMultiFieldElement @@ -108,39 +110,3 @@ internal class CardDetailsElement( return combineAsStateFlow(flows) { it.toList() } } } - -@RestrictTo(RestrictTo.Scope.LIBRARY_GROUP) -fun createExpiryDateFormFieldValues(entry: FormFieldEntry): Map { - return mapOf( - IdentifierSpec.CardExpMonth to getExpiryMonthFormFieldEntry(entry), - IdentifierSpec.CardExpYear to getExpiryYearFormFieldEntry(entry) - ) -} - -private fun getExpiryMonthFormFieldEntry(entry: FormFieldEntry): FormFieldEntry { - var month = -1 - entry.value?.let { date -> - val newString = convertTo4DigitDate(date) - if (newString.length == 4) { - month = requireNotNull(newString.take(2).toIntOrNull()) - } - } - - return entry.copy( - value = month.toString().padStart(length = 2, padChar = '0') - ) -} - -private fun getExpiryYearFormFieldEntry(entry: FormFieldEntry): FormFieldEntry { - var year = -1 - entry.value?.let { date -> - val newString = convertTo4DigitDate(date) - if (newString.length == 4) { - year = requireNotNull(newString.takeLast(2).toIntOrNull()) + 2000 - } - } - - return entry.copy( - value = year.toString() - ) -} diff --git a/payments-ui-core/src/main/java/com/stripe/android/ui/core/elements/CardDetailsUtil.kt b/payments-ui-core/src/main/java/com/stripe/android/ui/core/elements/CardDetailsUtil.kt new file mode 100644 index 00000000000..70fdbbe73c7 --- /dev/null +++ b/payments-ui-core/src/main/java/com/stripe/android/ui/core/elements/CardDetailsUtil.kt @@ -0,0 +1,45 @@ +package com.stripe.android.ui.core.elements + +import androidx.annotation.RestrictTo +import com.stripe.android.uicore.elements.IdentifierSpec +import com.stripe.android.uicore.elements.convertTo4DigitDate +import com.stripe.android.uicore.forms.FormFieldEntry + +@RestrictTo(RestrictTo.Scope.LIBRARY_GROUP) +object CardDetailsUtil { + @RestrictTo(RestrictTo.Scope.LIBRARY_GROUP) + fun createExpiryDateFormFieldValues(entry: FormFieldEntry): Map { + return mapOf( + IdentifierSpec.CardExpMonth to getExpiryMonthFormFieldEntry(entry), + IdentifierSpec.CardExpYear to getExpiryYearFormFieldEntry(entry) + ) + } + + internal fun getExpiryMonthFormFieldEntry(entry: FormFieldEntry): FormFieldEntry { + var month = -1 + entry.value?.let { date -> + val newString = convertTo4DigitDate(date) + if (newString.length == 4) { + month = requireNotNull(newString.take(2).toIntOrNull()) + } + } + + return entry.copy( + value = month.toString().padStart(length = 2, padChar = '0') + ) + } + + internal fun getExpiryYearFormFieldEntry(entry: FormFieldEntry): FormFieldEntry { + var year = -1 + entry.value?.let { date -> + val newString = convertTo4DigitDate(date) + if (newString.length == 4) { + year = requireNotNull(newString.takeLast(2).toIntOrNull()) + 2000 + } + } + + return entry.copy( + value = year.toString() + ) + } +} \ No newline at end of file diff --git a/paymentsheet/src/main/java/com/stripe/android/link/ui/wallet/WalletViewModel.kt b/paymentsheet/src/main/java/com/stripe/android/link/ui/wallet/WalletViewModel.kt index d3ea2b90b07..d6a1cca7d86 100644 --- a/paymentsheet/src/main/java/com/stripe/android/link/ui/wallet/WalletViewModel.kt +++ b/paymentsheet/src/main/java/com/stripe/android/link/ui/wallet/WalletViewModel.kt @@ -27,8 +27,8 @@ import com.stripe.android.model.StripeIntent import com.stripe.android.ui.core.Amount import com.stripe.android.ui.core.FieldValuesToParamsMapConverter import com.stripe.android.ui.core.R +import com.stripe.android.ui.core.elements.CardDetailsUtil.createExpiryDateFormFieldValues import com.stripe.android.ui.core.elements.CvcController -import com.stripe.android.ui.core.elements.createExpiryDateFormFieldValues import com.stripe.android.uicore.elements.DateConfig import com.stripe.android.uicore.elements.SimpleTextFieldController import com.stripe.android.uicore.utils.mapAsStateFlow From 3f730c74a6f08dcf5272d502e9bdd757947b1302 Mon Sep 17 00:00:00 2001 From: toluo-stripe Date: Tue, 14 Jan 2025 01:48:14 -0500 Subject: [PATCH 6/6] Fix lint --- .../com/stripe/android/ui/core/elements/CardDetailsElement.kt | 2 -- .../com/stripe/android/ui/core/elements/CardDetailsUtil.kt | 4 +++- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/payments-ui-core/src/main/java/com/stripe/android/ui/core/elements/CardDetailsElement.kt b/payments-ui-core/src/main/java/com/stripe/android/ui/core/elements/CardDetailsElement.kt index 4f91c2cb1f2..c992019e11e 100644 --- a/payments-ui-core/src/main/java/com/stripe/android/ui/core/elements/CardDetailsElement.kt +++ b/payments-ui-core/src/main/java/com/stripe/android/ui/core/elements/CardDetailsElement.kt @@ -1,6 +1,5 @@ package com.stripe.android.ui.core.elements -import androidx.annotation.RestrictTo import com.stripe.android.CardBrandFilter import com.stripe.android.DefaultCardBrandFilter import com.stripe.android.cards.CardAccountRangeRepository @@ -12,7 +11,6 @@ import com.stripe.android.ui.core.elements.CardDetailsUtil.getExpiryYearFormFiel import com.stripe.android.uicore.elements.IdentifierSpec import com.stripe.android.uicore.elements.SectionFieldErrorController import com.stripe.android.uicore.elements.SectionMultiFieldElement -import com.stripe.android.uicore.elements.convertTo4DigitDate import com.stripe.android.uicore.forms.FormFieldEntry import com.stripe.android.uicore.utils.combineAsStateFlow import com.stripe.android.uicore.utils.mapAsStateFlow diff --git a/payments-ui-core/src/main/java/com/stripe/android/ui/core/elements/CardDetailsUtil.kt b/payments-ui-core/src/main/java/com/stripe/android/ui/core/elements/CardDetailsUtil.kt index 70fdbbe73c7..96038b0d2ae 100644 --- a/payments-ui-core/src/main/java/com/stripe/android/ui/core/elements/CardDetailsUtil.kt +++ b/payments-ui-core/src/main/java/com/stripe/android/ui/core/elements/CardDetailsUtil.kt @@ -15,6 +15,7 @@ object CardDetailsUtil { ) } + @SuppressWarnings("MagicNumber") internal fun getExpiryMonthFormFieldEntry(entry: FormFieldEntry): FormFieldEntry { var month = -1 entry.value?.let { date -> @@ -29,6 +30,7 @@ object CardDetailsUtil { ) } + @SuppressWarnings("MagicNumber") internal fun getExpiryYearFormFieldEntry(entry: FormFieldEntry): FormFieldEntry { var year = -1 entry.value?.let { date -> @@ -42,4 +44,4 @@ object CardDetailsUtil { value = year.toString() ) } -} \ No newline at end of file +}