From c15e908a158ddcdbe84c09d4cce7ac43f928f052 Mon Sep 17 00:00:00 2001 From: Adam Brown Date: Tue, 1 Mar 2022 16:17:34 +0000 Subject: [PATCH 01/20] converting onboarding action to sealed interface --- .../features/onboarding/OnboardingAction.kt | 52 +++++++++---------- 1 file changed, 26 insertions(+), 26 deletions(-) diff --git a/vector/src/main/java/im/vector/app/features/onboarding/OnboardingAction.kt b/vector/src/main/java/im/vector/app/features/onboarding/OnboardingAction.kt index 4f16231747c..4a5445a2628 100644 --- a/vector/src/main/java/im/vector/app/features/onboarding/OnboardingAction.kt +++ b/vector/src/main/java/im/vector/app/features/onboarding/OnboardingAction.kt @@ -25,26 +25,26 @@ import org.matrix.android.sdk.api.auth.data.Credentials import org.matrix.android.sdk.api.auth.registration.RegisterThreePid import org.matrix.android.sdk.internal.network.ssl.Fingerprint -sealed class OnboardingAction : VectorViewModelAction { - data class OnGetStarted(val resetLoginConfig: Boolean, val onboardingFlow: OnboardingFlow) : OnboardingAction() - data class OnIAlreadyHaveAnAccount(val resetLoginConfig: Boolean, val onboardingFlow: OnboardingFlow) : OnboardingAction() - - data class UpdateServerType(val serverType: ServerType) : OnboardingAction() - data class UpdateHomeServer(val homeServerUrl: String) : OnboardingAction() - data class UpdateUseCase(val useCase: FtueUseCase) : OnboardingAction() - object ResetUseCase : OnboardingAction() - data class UpdateSignMode(val signMode: SignMode) : OnboardingAction() - data class LoginWithToken(val loginToken: String) : OnboardingAction() - data class WebLoginSuccess(val credentials: Credentials) : OnboardingAction() - data class InitWith(val loginConfig: LoginConfig?) : OnboardingAction() - data class ResetPassword(val email: String, val newPassword: String) : OnboardingAction() - object ResetPasswordMailConfirmed : OnboardingAction() +sealed interface OnboardingAction : VectorViewModelAction { + data class OnGetStarted(val resetLoginConfig: Boolean, val onboardingFlow: OnboardingFlow) : OnboardingAction + data class OnIAlreadyHaveAnAccount(val resetLoginConfig: Boolean, val onboardingFlow: OnboardingFlow) : OnboardingAction + + data class UpdateServerType(val serverType: ServerType) : OnboardingAction + data class UpdateHomeServer(val homeServerUrl: String) : OnboardingAction + data class UpdateUseCase(val useCase: FtueUseCase) : OnboardingAction + object ResetUseCase : OnboardingAction + data class UpdateSignMode(val signMode: SignMode) : OnboardingAction + data class LoginWithToken(val loginToken: String) : OnboardingAction + data class WebLoginSuccess(val credentials: Credentials) : OnboardingAction + data class InitWith(val loginConfig: LoginConfig?) : OnboardingAction + data class ResetPassword(val email: String, val newPassword: String) : OnboardingAction + object ResetPasswordMailConfirmed : OnboardingAction // Login or Register, depending on the signMode - data class LoginOrRegister(val username: String, val password: String, val initialDeviceName: String) : OnboardingAction() + data class LoginOrRegister(val username: String, val password: String, val initialDeviceName: String) : OnboardingAction // Register actions - open class RegisterAction : OnboardingAction() + open class RegisterAction : OnboardingAction data class AddThreePid(val threePid: RegisterThreePid) : RegisterAction() object SendAgainThreePid : RegisterAction() @@ -60,7 +60,7 @@ sealed class OnboardingAction : VectorViewModelAction { object RegisterDummy : RegisterAction() // Reset actions - open class ResetAction : OnboardingAction() + open class ResetAction : OnboardingAction object ResetHomeServerType : ResetAction() object ResetHomeServerUrl : ResetAction() @@ -69,16 +69,16 @@ sealed class OnboardingAction : VectorViewModelAction { object ResetResetPassword : ResetAction() // Homeserver history - object ClearHomeServerHistory : OnboardingAction() + object ClearHomeServerHistory : OnboardingAction - data class PostViewEvent(val viewEvent: OnboardingViewEvents) : OnboardingAction() + data class PostViewEvent(val viewEvent: OnboardingViewEvents) : OnboardingAction - data class UserAcceptCertificate(val fingerprint: Fingerprint) : OnboardingAction() + data class UserAcceptCertificate(val fingerprint: Fingerprint) : OnboardingAction - object PersonalizeProfile : OnboardingAction() - data class UpdateDisplayName(val displayName: String) : OnboardingAction() - object UpdateDisplayNameSkipped : OnboardingAction() - data class ProfilePictureSelected(val uri: Uri) : OnboardingAction() - object SaveSelectedProfilePicture : OnboardingAction() - object UpdateProfilePictureSkipped : OnboardingAction() + object PersonalizeProfile : OnboardingAction + data class UpdateDisplayName(val displayName: String) : OnboardingAction + object UpdateDisplayNameSkipped : OnboardingAction + data class ProfilePictureSelected(val uri: Uri) : OnboardingAction + object SaveSelectedProfilePicture : OnboardingAction + object UpdateProfilePictureSkipped : OnboardingAction } From 4225f62120540a34c632a27fe138ae1dba4951ed Mon Sep 17 00:00:00 2001 From: Adam Brown Date: Wed, 2 Mar 2022 12:18:50 +0000 Subject: [PATCH 02/20] adding test helper for asserting states whilst combining previous updates --- vector/src/test/java/im/vector/app/test/Extensions.kt | 11 +++++++++++ 1 file changed, 11 insertions(+) diff --git a/vector/src/test/java/im/vector/app/test/Extensions.kt b/vector/src/test/java/im/vector/app/test/Extensions.kt index 3ff041dc11b..136ca58d68b 100644 --- a/vector/src/test/java/im/vector/app/test/Extensions.kt +++ b/vector/src/test/java/im/vector/app/test/Extensions.kt @@ -55,6 +55,17 @@ class ViewModelTest( return this } + fun assertStatesWithPrevious(initial: S, vararg expected: S.() -> S): ViewModelTest { + val reducedExpectedStates = expected.fold(mutableListOf(initial)) { acc, curr -> + val next = curr.invoke(acc.last()) + acc.add(next) + acc + } + + states.assertValues(reducedExpectedStates) + return this + } + fun assertStates(expected: List): ViewModelTest { states.assertValues(expected) return this From 3fa415007c21e2bbad4c350d1f16429c6aae2f4b Mon Sep 17 00:00:00 2001 From: Adam Brown Date: Wed, 2 Mar 2022 16:49:23 +0000 Subject: [PATCH 03/20] extracting registration steps to separate handler to make testing the flow simpler --- .../features/onboarding/OnboardingAction.kt | 18 +- .../onboarding/OnboardingViewModel.kt | 172 ++++-------------- .../onboarding/RegistrationActionHandler.kt | 61 +++++++ .../ftueauth/FtueAuthCaptchaFragment.kt | 3 +- .../FtueAuthGenericTextInputFormFragment.kt | 11 +- .../ftueauth/FtueAuthWaitForEmailFragment.kt | 5 +- .../ftueauth/terms/FtueAuthTermsFragment.kt | 3 +- .../onboarding/OnboardingViewModelTest.kt | 110 ++++++++--- .../RegistrationActionHandlerTest.kt | 73 ++++++++ .../test/fakes/FakeAuthenticationService.kt | 5 + .../test/fakes/FakeRegisterActionHandler.kt | 43 +++++ .../app/test/fakes/FakeRegistrationWizard.kt | 10 +- 12 files changed, 329 insertions(+), 185 deletions(-) create mode 100644 vector/src/main/java/im/vector/app/features/onboarding/RegistrationActionHandler.kt create mode 100644 vector/src/test/java/im/vector/app/features/onboarding/RegistrationActionHandlerTest.kt create mode 100644 vector/src/test/java/im/vector/app/test/fakes/FakeRegisterActionHandler.kt diff --git a/vector/src/main/java/im/vector/app/features/onboarding/OnboardingAction.kt b/vector/src/main/java/im/vector/app/features/onboarding/OnboardingAction.kt index 4a5445a2628..8b17b318c15 100644 --- a/vector/src/main/java/im/vector/app/features/onboarding/OnboardingAction.kt +++ b/vector/src/main/java/im/vector/app/features/onboarding/OnboardingAction.kt @@ -22,7 +22,6 @@ import im.vector.app.features.login.LoginConfig import im.vector.app.features.login.ServerType import im.vector.app.features.login.SignMode import org.matrix.android.sdk.api.auth.data.Credentials -import org.matrix.android.sdk.api.auth.registration.RegisterThreePid import org.matrix.android.sdk.internal.network.ssl.Fingerprint sealed interface OnboardingAction : VectorViewModelAction { @@ -42,22 +41,9 @@ sealed interface OnboardingAction : VectorViewModelAction { // Login or Register, depending on the signMode data class LoginOrRegister(val username: String, val password: String, val initialDeviceName: String) : OnboardingAction + object StopEmailValidationCheck : OnboardingAction - // Register actions - open class RegisterAction : OnboardingAction - - data class AddThreePid(val threePid: RegisterThreePid) : RegisterAction() - object SendAgainThreePid : RegisterAction() - - // TODO Confirm Email (from link in the email, open in the phone, intercepted by the app) - data class ValidateThreePid(val code: String) : RegisterAction() - - data class CheckIfEmailHasBeenValidated(val delayMillis: Long) : RegisterAction() - object StopEmailValidationCheck : RegisterAction() - - data class CaptchaDone(val captchaResponse: String) : RegisterAction() - object AcceptTerms : RegisterAction() - object RegisterDummy : RegisterAction() + data class PostRegisterAction(val registerAction: RegisterAction) : OnboardingAction // Reset actions open class ResetAction : OnboardingAction diff --git a/vector/src/main/java/im/vector/app/features/onboarding/OnboardingViewModel.kt b/vector/src/main/java/im/vector/app/features/onboarding/OnboardingViewModel.kt index 36020fbe61f..303a4d5950e 100644 --- a/vector/src/main/java/im/vector/app/features/onboarding/OnboardingViewModel.kt +++ b/vector/src/main/java/im/vector/app/features/onboarding/OnboardingViewModel.kt @@ -83,6 +83,7 @@ class OnboardingViewModel @AssistedInject constructor( private val vectorFeatures: VectorFeatures, private val analyticsTracker: AnalyticsTracker, private val uriFilenameResolver: UriFilenameResolver, + private val registrationActionHandler: RegistrationActionHandler, private val vectorOverrides: VectorOverrides ) : VectorViewModel(initialState) { @@ -116,16 +117,16 @@ class OnboardingViewModel @AssistedInject constructor( private val matrixOrgUrl = stringProvider.getString(R.string.matrix_org_server_url).ensureTrailingSlash() + private val registrationWizard: RegistrationWizard + get() = authenticationService.getRegistrationWizard() + val currentThreePid: String? - get() = registrationWizard?.currentThreePid + get() = registrationWizard.currentThreePid // True when login and password has been sent with success to the homeserver val isRegistrationStarted: Boolean get() = authenticationService.isRegistrationStarted - private val registrationWizard: RegistrationWizard? - get() = authenticationService.getRegistrationWizard() - private val loginWizard: LoginWizard? get() = authenticationService.getLoginWizard() @@ -153,7 +154,7 @@ class OnboardingViewModel @AssistedInject constructor( is OnboardingAction.WebLoginSuccess -> handleWebLoginSuccess(action) is OnboardingAction.ResetPassword -> handleResetPassword(action) is OnboardingAction.ResetPasswordMailConfirmed -> handleResetPasswordMailConfirmed() - is OnboardingAction.RegisterAction -> handleRegisterAction(action) + is OnboardingAction.PostRegisterAction -> handleRegisterAction(action.registerAction) is OnboardingAction.ResetAction -> handleResetAction(action) is OnboardingAction.UserAcceptCertificate -> handleUserAcceptCertificate(action) OnboardingAction.ClearHomeServerHistory -> handleClearHomeServerHistory() @@ -164,6 +165,7 @@ class OnboardingViewModel @AssistedInject constructor( is OnboardingAction.ProfilePictureSelected -> handleProfilePictureSelected(action) OnboardingAction.SaveSelectedProfilePicture -> updateProfilePicture() is OnboardingAction.PostViewEvent -> _viewEvents.post(action.viewEvent) + OnboardingAction.StopEmailValidationCheck -> currentJob = null }.exhaustive } @@ -266,131 +268,36 @@ class OnboardingViewModel @AssistedInject constructor( } } - private fun handleRegisterAction(action: OnboardingAction.RegisterAction) { - when (action) { - is OnboardingAction.CaptchaDone -> handleCaptchaDone(action) - is OnboardingAction.AcceptTerms -> handleAcceptTerms() - is OnboardingAction.RegisterDummy -> handleRegisterDummy() - is OnboardingAction.AddThreePid -> handleAddThreePid(action) - is OnboardingAction.SendAgainThreePid -> handleSendAgainThreePid() - is OnboardingAction.ValidateThreePid -> handleValidateThreePid(action) - is OnboardingAction.CheckIfEmailHasBeenValidated -> handleCheckIfEmailHasBeenValidated(action) - is OnboardingAction.StopEmailValidationCheck -> handleStopEmailValidationCheck() - } - } - - private fun handleCheckIfEmailHasBeenValidated(action: OnboardingAction.CheckIfEmailHasBeenValidated) { - // We do not want the common progress bar to be displayed, so we do not change asyncRegistration value in the state - currentJob = executeRegistrationStep(withLoading = false) { - it.checkIfEmailHasBeenValidated(action.delayMillis) - } - } - - private fun handleStopEmailValidationCheck() { - currentJob = null - } - - private fun handleValidateThreePid(action: OnboardingAction.ValidateThreePid) { - currentJob = executeRegistrationStep { - it.handleValidateThreePid(action.code) - } - } - - private fun executeRegistrationStep(withLoading: Boolean = true, - block: suspend (RegistrationWizard) -> RegistrationResult): Job { - if (withLoading) { - setState { copy(asyncRegistration = Loading()) } - } - return viewModelScope.launch { - try { - registrationWizard?.let { block(it) } - /* - // Simulate registration disabled - throw Failure.ServerError(MatrixError( - code = MatrixError.FORBIDDEN, - message = "Registration is disabled" - ), 403)) - */ - } catch (failure: Throwable) { - if (failure !is CancellationException) { - _viewEvents.post(OnboardingViewEvents.Failure(failure)) - } - null - } - ?.let { data -> - when (data) { - is RegistrationResult.Success -> onSessionCreated(data.session, isAccountCreated = true) - is RegistrationResult.FlowResponse -> onFlowResponse(data.flowResult) - } - } - - setState { - copy( - asyncRegistration = Uninitialized - ) - } - } - } - - private fun handleAddThreePid(action: OnboardingAction.AddThreePid) { - setState { copy(asyncRegistration = Loading()) } - currentJob = viewModelScope.launch { - try { - registrationWizard?.addThreePid(action.threePid) - } catch (failure: Throwable) { - _viewEvents.post(OnboardingViewEvents.Failure(failure)) - } - setState { - copy( - asyncRegistration = Uninitialized - ) - } - } - } - - private fun handleSendAgainThreePid() { - setState { copy(asyncRegistration = Loading()) } + private fun handleRegisterAction(action: RegisterAction) { currentJob = viewModelScope.launch { - try { - registrationWizard?.sendAgainThreePid() - } catch (failure: Throwable) { - _viewEvents.post(OnboardingViewEvents.Failure(failure)) - } - setState { - copy( - asyncRegistration = Uninitialized - ) - } - } - } - - private fun handleAcceptTerms() { - currentJob = executeRegistrationStep { - it.acceptTerms() - } - } - - private fun handleRegisterDummy() { - currentJob = executeRegistrationStep { - it.dummy() + if (action.hasLoadingState()) { + setState { copy(asyncRegistration = Loading()) } + } + kotlin.runCatching { registrationActionHandler.handleRegisterAction(registrationWizard, action) } + .fold( + onSuccess = { + when (it) { + is RegistrationResult.Success -> onSessionCreated(it.session, isAccountCreated = true) + is RegistrationResult.FlowResponse -> onFlowResponse(it.flowResult) + } + }, + onFailure = { + if (it !is CancellationException) { + _viewEvents.post(OnboardingViewEvents.Failure(it)) + } + } + ) + setState { copy(asyncRegistration = Uninitialized) } } } private fun handleRegisterWith(action: OnboardingAction.LoginOrRegister) { reAuthHelper.data = action.password - currentJob = executeRegistrationStep { - it.createAccount( - action.username, - action.password, - action.initialDeviceName - ) - } - } - - private fun handleCaptchaDone(action: OnboardingAction.CaptchaDone) { - currentJob = executeRegistrationStep { - it.performReCaptcha(action.captchaResponse) - } + handleRegisterAction(RegisterAction.CreateAccount( + action.username, + action.password, + action.initialDeviceName + )) } private fun handleResetAction(action: OnboardingAction.ResetAction) { @@ -461,7 +368,7 @@ class OnboardingViewModel @AssistedInject constructor( } when (action.signMode) { - SignMode.SignUp -> startRegistrationFlow() + SignMode.SignUp -> handleRegisterAction(RegisterAction.RegisterDummy) SignMode.SignIn -> startAuthenticationFlow() SignMode.SignInWithMatrixId -> _viewEvents.post(OnboardingViewEvents.OnSignModeSelected(SignMode.SignInWithMatrixId)) SignMode.Unknown -> Unit @@ -499,7 +406,7 @@ class OnboardingViewModel @AssistedInject constructor( // If there is a pending email validation continue on this step try { - if (registrationWizard?.isRegistrationStarted == true) { + if (registrationWizard.isRegistrationStarted) { currentThreePid?.let { handle(OnboardingAction.PostViewEvent(OnboardingViewEvents.OnSendEmailSuccess(it))) } @@ -730,12 +637,6 @@ class OnboardingViewModel @AssistedInject constructor( } } - private fun startRegistrationFlow() { - currentJob = executeRegistrationStep { - it.getRegistrationFlow() - } - } - private fun startAuthenticationFlow() { // Ensure Wizard is ready loginWizard @@ -745,8 +646,7 @@ class OnboardingViewModel @AssistedInject constructor( private fun onFlowResponse(flowResult: FlowResult) { // If dummy stage is mandatory, and password is already sent, do the dummy stage now - if (isRegistrationStarted && - flowResult.missingStages.any { it is Stage.Dummy && it.mandatory }) { + if (isRegistrationStarted && flowResult.missingStages.any { it is Stage.Dummy && it.mandatory }) { handleRegisterDummy() } else { // Notify the user @@ -754,6 +654,10 @@ class OnboardingViewModel @AssistedInject constructor( } } + private fun handleRegisterDummy() { + handleRegisterAction(RegisterAction.RegisterDummy) + } + private suspend fun onSessionCreated(session: Session, isAccountCreated: Boolean) { val state = awaitState() state.useCase?.let { useCase -> diff --git a/vector/src/main/java/im/vector/app/features/onboarding/RegistrationActionHandler.kt b/vector/src/main/java/im/vector/app/features/onboarding/RegistrationActionHandler.kt new file mode 100644 index 00000000000..2938681ce76 --- /dev/null +++ b/vector/src/main/java/im/vector/app/features/onboarding/RegistrationActionHandler.kt @@ -0,0 +1,61 @@ +/* + * Copyright (c) 2022 New Vector Ltd + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package im.vector.app.features.onboarding + +import org.matrix.android.sdk.api.auth.registration.RegisterThreePid +import org.matrix.android.sdk.api.auth.registration.RegistrationResult +import org.matrix.android.sdk.api.auth.registration.RegistrationWizard +import javax.inject.Inject + +class RegistrationActionHandler @Inject constructor() { + + suspend fun handleRegisterAction(registrationWizard: RegistrationWizard, action: RegisterAction): RegistrationResult { + return when (action) { + RegisterAction.StartRegistration -> registrationWizard.getRegistrationFlow() + is RegisterAction.CaptchaDone -> registrationWizard.performReCaptcha(action.captchaResponse) + is RegisterAction.AcceptTerms -> registrationWizard.acceptTerms() + is RegisterAction.RegisterDummy -> registrationWizard.dummy() + is RegisterAction.AddThreePid -> registrationWizard.addThreePid(action.threePid) + is RegisterAction.SendAgainThreePid -> registrationWizard.sendAgainThreePid() + is RegisterAction.ValidateThreePid -> registrationWizard.handleValidateThreePid(action.code) + is RegisterAction.CheckIfEmailHasBeenValidated -> registrationWizard.checkIfEmailHasBeenValidated(action.delayMillis) + is RegisterAction.CreateAccount -> registrationWizard.createAccount(action.username, action.password, action.initialDeviceName) + } + } +} + +sealed interface RegisterAction { + object StartRegistration : RegisterAction + data class CreateAccount(val username: String, val password: String, val initialDeviceName: String) : RegisterAction + + data class AddThreePid(val threePid: RegisterThreePid) : RegisterAction + object SendAgainThreePid : RegisterAction + + // TODO Confirm Email (from link in the email, open in the phone, intercepted by the app) + data class ValidateThreePid(val code: String) : RegisterAction + + data class CheckIfEmailHasBeenValidated(val delayMillis: Long) : RegisterAction + + data class CaptchaDone(val captchaResponse: String) : RegisterAction + object AcceptTerms : RegisterAction + object RegisterDummy : RegisterAction +} + +fun RegisterAction.hasLoadingState() = when (this) { + is RegisterAction.CheckIfEmailHasBeenValidated -> false + else -> true +} diff --git a/vector/src/main/java/im/vector/app/features/onboarding/ftueauth/FtueAuthCaptchaFragment.kt b/vector/src/main/java/im/vector/app/features/onboarding/ftueauth/FtueAuthCaptchaFragment.kt index e2e390ae2de..47733321386 100644 --- a/vector/src/main/java/im/vector/app/features/onboarding/ftueauth/FtueAuthCaptchaFragment.kt +++ b/vector/src/main/java/im/vector/app/features/onboarding/ftueauth/FtueAuthCaptchaFragment.kt @@ -39,6 +39,7 @@ import im.vector.app.databinding.FragmentLoginCaptchaBinding import im.vector.app.features.login.JavascriptResponse import im.vector.app.features.onboarding.OnboardingAction import im.vector.app.features.onboarding.OnboardingViewState +import im.vector.app.features.onboarding.RegisterAction import kotlinx.parcelize.Parcelize import org.matrix.android.sdk.internal.di.MoshiProvider import timber.log.Timber @@ -181,7 +182,7 @@ class FtueAuthCaptchaFragment @Inject constructor( val response = javascriptResponse?.response if (javascriptResponse?.action == "verifyCallback" && response != null) { - viewModel.handle(OnboardingAction.CaptchaDone(response)) + viewModel.handle(OnboardingAction.PostRegisterAction(RegisterAction.CaptchaDone(response))) } } return true diff --git a/vector/src/main/java/im/vector/app/features/onboarding/ftueauth/FtueAuthGenericTextInputFormFragment.kt b/vector/src/main/java/im/vector/app/features/onboarding/ftueauth/FtueAuthGenericTextInputFormFragment.kt index bd5054f6463..2800530152f 100644 --- a/vector/src/main/java/im/vector/app/features/onboarding/ftueauth/FtueAuthGenericTextInputFormFragment.kt +++ b/vector/src/main/java/im/vector/app/features/onboarding/ftueauth/FtueAuthGenericTextInputFormFragment.kt @@ -37,6 +37,7 @@ import im.vector.app.databinding.FragmentLoginGenericTextInputFormBinding import im.vector.app.features.login.TextInputFormFragmentMode import im.vector.app.features.onboarding.OnboardingAction import im.vector.app.features.onboarding.OnboardingViewEvents +import im.vector.app.features.onboarding.RegisterAction import kotlinx.coroutines.flow.launchIn import kotlinx.coroutines.flow.onEach import kotlinx.parcelize.Parcelize @@ -138,7 +139,7 @@ class FtueAuthGenericTextInputFormFragment @Inject constructor() : AbstractFtueA private fun onOtherButtonClicked() { when (params.mode) { TextInputFormFragmentMode.ConfirmMsisdn -> { - viewModel.handle(OnboardingAction.SendAgainThreePid) + viewModel.handle(OnboardingAction.PostRegisterAction(RegisterAction.SendAgainThreePid)) } else -> { // Should not happen, button is not displayed @@ -152,19 +153,19 @@ class FtueAuthGenericTextInputFormFragment @Inject constructor() : AbstractFtueA if (text.isEmpty()) { // Perform dummy action - viewModel.handle(OnboardingAction.RegisterDummy) + viewModel.handle(OnboardingAction.PostRegisterAction(RegisterAction.RegisterDummy)) } else { when (params.mode) { TextInputFormFragmentMode.SetEmail -> { - viewModel.handle(OnboardingAction.AddThreePid(RegisterThreePid.Email(text))) + viewModel.handle(OnboardingAction.PostRegisterAction(RegisterAction.AddThreePid(RegisterThreePid.Email(text)))) } TextInputFormFragmentMode.SetMsisdn -> { getCountryCodeOrShowError(text)?.let { countryCode -> - viewModel.handle(OnboardingAction.AddThreePid(RegisterThreePid.Msisdn(text, countryCode))) + viewModel.handle(OnboardingAction.PostRegisterAction(RegisterAction.AddThreePid(RegisterThreePid.Msisdn(text, countryCode)))) } } TextInputFormFragmentMode.ConfirmMsisdn -> { - viewModel.handle(OnboardingAction.ValidateThreePid(text)) + viewModel.handle(OnboardingAction.PostRegisterAction(RegisterAction.ValidateThreePid(text))) } } } diff --git a/vector/src/main/java/im/vector/app/features/onboarding/ftueauth/FtueAuthWaitForEmailFragment.kt b/vector/src/main/java/im/vector/app/features/onboarding/ftueauth/FtueAuthWaitForEmailFragment.kt index 94758c7fad4..ec72f52b9ed 100644 --- a/vector/src/main/java/im/vector/app/features/onboarding/ftueauth/FtueAuthWaitForEmailFragment.kt +++ b/vector/src/main/java/im/vector/app/features/onboarding/ftueauth/FtueAuthWaitForEmailFragment.kt @@ -25,6 +25,7 @@ import com.airbnb.mvrx.args import im.vector.app.R import im.vector.app.databinding.FragmentLoginWaitForEmailBinding import im.vector.app.features.onboarding.OnboardingAction +import im.vector.app.features.onboarding.RegisterAction import kotlinx.parcelize.Parcelize import org.matrix.android.sdk.api.failure.is401 import javax.inject.Inject @@ -54,7 +55,7 @@ class FtueAuthWaitForEmailFragment @Inject constructor() : AbstractFtueAuthFragm override fun onResume() { super.onResume() - viewModel.handle(OnboardingAction.CheckIfEmailHasBeenValidated(0)) + viewModel.handle(OnboardingAction.PostRegisterAction(RegisterAction.CheckIfEmailHasBeenValidated(0))) } override fun onPause() { @@ -70,7 +71,7 @@ class FtueAuthWaitForEmailFragment @Inject constructor() : AbstractFtueAuthFragm override fun onError(throwable: Throwable) { if (throwable.is401()) { // Try again, with a delay - viewModel.handle(OnboardingAction.CheckIfEmailHasBeenValidated(10_000)) + viewModel.handle(OnboardingAction.PostRegisterAction(RegisterAction.CheckIfEmailHasBeenValidated(10_000))) } else { super.onError(throwable) } diff --git a/vector/src/main/java/im/vector/app/features/onboarding/ftueauth/terms/FtueAuthTermsFragment.kt b/vector/src/main/java/im/vector/app/features/onboarding/ftueauth/terms/FtueAuthTermsFragment.kt index 5ce9a5350df..03598d3a47f 100755 --- a/vector/src/main/java/im/vector/app/features/onboarding/ftueauth/terms/FtueAuthTermsFragment.kt +++ b/vector/src/main/java/im/vector/app/features/onboarding/ftueauth/terms/FtueAuthTermsFragment.kt @@ -32,6 +32,7 @@ import im.vector.app.features.login.terms.LoginTermsViewState import im.vector.app.features.login.terms.PolicyController import im.vector.app.features.onboarding.OnboardingAction import im.vector.app.features.onboarding.OnboardingViewState +import im.vector.app.features.onboarding.RegisterAction import im.vector.app.features.onboarding.ftueauth.AbstractFtueAuthFragment import kotlinx.parcelize.Parcelize import org.matrix.android.sdk.internal.auth.registration.LocalizedFlowDataLoginTerms @@ -111,7 +112,7 @@ class FtueAuthTermsFragment @Inject constructor( } private fun submit() { - viewModel.handle(OnboardingAction.AcceptTerms) + viewModel.handle(OnboardingAction.PostRegisterAction(RegisterAction.AcceptTerms)) } override fun updateWithState(state: OnboardingViewState) { diff --git a/vector/src/test/java/im/vector/app/features/onboarding/OnboardingViewModelTest.kt b/vector/src/test/java/im/vector/app/features/onboarding/OnboardingViewModelTest.kt index 085e1a8049e..44d44a94b21 100644 --- a/vector/src/test/java/im/vector/app/features/onboarding/OnboardingViewModelTest.kt +++ b/vector/src/test/java/im/vector/app/features/onboarding/OnboardingViewModelTest.kt @@ -29,6 +29,7 @@ import im.vector.app.test.fakes.FakeAuthenticationService import im.vector.app.test.fakes.FakeContext import im.vector.app.test.fakes.FakeHomeServerConnectionConfigFactory import im.vector.app.test.fakes.FakeHomeServerHistoryService +import im.vector.app.test.fakes.FakeRegisterActionHandler import im.vector.app.test.fakes.FakeRegistrationWizard import im.vector.app.test.fakes.FakeSession import im.vector.app.test.fakes.FakeStringProvider @@ -41,6 +42,9 @@ import kotlinx.coroutines.test.runBlockingTest import org.junit.Before import org.junit.Rule import org.junit.Test +import org.matrix.android.sdk.api.auth.registration.FlowResult +import org.matrix.android.sdk.api.auth.registration.RegistrationResult +import org.matrix.android.sdk.api.auth.registration.Stage import org.matrix.android.sdk.api.session.homeserver.HomeServerCapabilities private const val A_DISPLAY_NAME = "a display name" @@ -50,6 +54,7 @@ private val AN_UNSUPPORTED_PERSONALISATION_STATE = PersonalizationState( supportsChangingDisplayName = false, supportsChangingProfilePicture = false ) +private val A_LOADABLE_REGISTER_ACTION = RegisterAction.StartRegistration class OnboardingViewModelTest { @@ -63,6 +68,7 @@ class OnboardingViewModelTest { private val fakeUriFilenameResolver = FakeUriFilenameResolver() private val fakeActiveSessionHolder = FakeActiveSessionHolder(fakeSession) private val fakeAuthenticationService = FakeAuthenticationService() + private val fakeRegisterActionHandler = FakeRegisterActionHandler() lateinit var viewModel: OnboardingViewModel @@ -108,28 +114,67 @@ class OnboardingViewModelTest { .finish() } + @Test + fun `given register action requires more steps when handling action then posts next steps`() = runBlockingTest { + val test = viewModel.test(this) + val flowResult = FlowResult(missingStages = listOf(Stage.Email(true)), completedStages = listOf(Stage.Email(true))) + givenRegistrationResultFor(A_LOADABLE_REGISTER_ACTION, RegistrationResult.FlowResponse(flowResult)) + + viewModel.handle(OnboardingAction.PostRegisterAction(A_LOADABLE_REGISTER_ACTION)) + + test + .assertStatesWithPrevious( + initialState, + { copy(asyncRegistration = Loading()) }, + { copy(asyncRegistration = Uninitialized) } + ) + .assertEvents(OnboardingViewEvents.RegistrationFlowResult(flowResult, isRegistrationStarted = true)) + .finish() + } + + @Test + fun `given registration has started and has dummy step to do when handling action then ignores other steps and executes dummy`() = runBlockingTest { + val test = viewModel.test(this) + + val homeServerCapabilities = HomeServerCapabilities(canChangeDisplayName = true, canChangeAvatar = true) + fakeSession.fakeHomeServerCapabilitiesService.givenCapabilities(homeServerCapabilities) + val flowResult = FlowResult(missingStages = listOf(Stage.Dummy(mandatory = true), Stage.Email(true)), completedStages = emptyList()) + givenRegistrationResultsFor(listOf( + A_LOADABLE_REGISTER_ACTION to RegistrationResult.FlowResponse(flowResult), + RegisterAction.RegisterDummy to RegistrationResult.Success(fakeSession) + )) + givenSuccessfullyCreatesAccount() + + viewModel.handle(OnboardingAction.PostRegisterAction(A_LOADABLE_REGISTER_ACTION)) + + test + .assertStatesWithPrevious( + initialState, + { copy(asyncRegistration = Loading()) }, + { copy(asyncLoginAction = Success(Unit), personalizationState = homeServerCapabilities.toPersonalisationState()) }, + { copy(asyncRegistration = Uninitialized) }, + + ) + .assertEvents(OnboardingViewEvents.OnAccountCreated) + .finish() + } + @Test fun `given homeserver does not support personalisation when registering account then updates state and emits account created event`() = runBlockingTest { - fakeSession.fakeHomeServerCapabilitiesService.givenCapabilities(HomeServerCapabilities(canChangeDisplayName = false, canChangeAvatar = false)) + val homeServerCapabilities = HomeServerCapabilities(canChangeDisplayName = false, canChangeAvatar = false) + fakeSession.fakeHomeServerCapabilitiesService.givenCapabilities(homeServerCapabilities) + givenRegistrationResultFor(A_LOADABLE_REGISTER_ACTION, RegistrationResult.Success(fakeSession)) givenSuccessfullyCreatesAccount() val test = viewModel.test(this) - viewModel.handle(OnboardingAction.RegisterDummy) + viewModel.handle(OnboardingAction.PostRegisterAction(A_LOADABLE_REGISTER_ACTION)) test - .assertStates( + .assertStatesWithPrevious( initialState, - initialState.copy(asyncRegistration = Loading()), - initialState.copy( - asyncLoginAction = Success(Unit), - asyncRegistration = Loading(), - personalizationState = AN_UNSUPPORTED_PERSONALISATION_STATE - ), - initialState.copy( - asyncLoginAction = Success(Unit), - asyncRegistration = Uninitialized, - personalizationState = AN_UNSUPPORTED_PERSONALISATION_STATE - ) + { copy(asyncRegistration = Loading()) }, + { copy(asyncLoginAction = Success(Unit), personalizationState = homeServerCapabilities.toPersonalisationState()) }, + { copy(asyncLoginAction = Success(Unit), asyncRegistration = Uninitialized) } ) .assertEvents(OnboardingViewEvents.OnAccountCreated) .finish() @@ -173,10 +218,10 @@ class OnboardingViewModelTest { viewModel.handle(OnboardingAction.UpdateDisplayName(A_DISPLAY_NAME)) test - .assertStates( + .assertStatesWithPrevious( initialState, - initialState.copy(asyncDisplayName = Loading()), - initialState.copy(asyncDisplayName = Fail(AN_ERROR)), + { copy(asyncDisplayName = Loading()) }, + { copy(asyncDisplayName = Fail(AN_ERROR)) }, ) .assertEvents(OnboardingViewEvents.Failure(AN_ERROR)) .finish() @@ -264,6 +309,7 @@ class OnboardingViewModelTest { FakeVectorFeatures(), FakeAnalyticsTracker(), fakeUriFilenameResolver.instance, + fakeRegisterActionHandler.instance, FakeVectorOverrides() ) } @@ -286,14 +332,6 @@ class OnboardingViewModelTest { state.copy(asyncProfilePicture = Fail(cause)) ) - private fun givenSuccessfullyCreatesAccount() { - fakeActiveSessionHolder.expectSetsActiveSession(fakeSession) - val registrationWizard = FakeRegistrationWizard().also { it.givenSuccessfulDummy(fakeSession) } - fakeAuthenticationService.givenRegistrationWizard(registrationWizard) - fakeAuthenticationService.expectReset() - fakeSession.expectStartsSyncing() - } - private fun expectedSuccessfulDisplayNameUpdateStates(personalisedInitialState: OnboardingViewState): List { return listOf( personalisedInitialState, @@ -304,4 +342,26 @@ class OnboardingViewModelTest { ) ) } + + private fun givenSuccessfullyCreatesAccount() { + fakeActiveSessionHolder.expectSetsActiveSession(fakeSession) + fakeAuthenticationService.expectReset() + fakeSession.expectStartsSyncing() + } + + private fun givenRegistrationResultFor(action: RegisterAction, result: RegistrationResult) { + givenRegistrationResultsFor(listOf(action to result)) + } + + private fun givenRegistrationResultsFor(results: List>) { + fakeAuthenticationService.givenRegistrationStarted(true) + val registrationWizard = FakeRegistrationWizard() + fakeAuthenticationService.givenRegistrationWizard(registrationWizard) + fakeRegisterActionHandler.givenResultsFor(registrationWizard, results) + } } + +private fun HomeServerCapabilities.toPersonalisationState() = PersonalizationState( + supportsChangingDisplayName = canChangeDisplayName, + supportsChangingProfilePicture = canChangeAvatar +) diff --git a/vector/src/test/java/im/vector/app/features/onboarding/RegistrationActionHandlerTest.kt b/vector/src/test/java/im/vector/app/features/onboarding/RegistrationActionHandlerTest.kt new file mode 100644 index 00000000000..87efa0bddc3 --- /dev/null +++ b/vector/src/test/java/im/vector/app/features/onboarding/RegistrationActionHandlerTest.kt @@ -0,0 +1,73 @@ +/* + * Copyright (c) 2022 New Vector Ltd + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package im.vector.app.features.onboarding + +import im.vector.app.test.fakes.FakeRegistrationWizard +import im.vector.app.test.fakes.FakeSession +import kotlinx.coroutines.test.runBlockingTest +import org.amshove.kluent.shouldBeEqualTo +import org.junit.Test +import org.matrix.android.sdk.api.auth.registration.RegisterThreePid +import org.matrix.android.sdk.api.auth.registration.RegistrationResult +import org.matrix.android.sdk.api.auth.registration.RegistrationWizard + +private val A_SESSION = FakeSession() +private val AN_EXPECTED_RESULT = RegistrationResult.Success(A_SESSION) +private const val A_USERNAME = "a username" +private const val A_PASSWORD = "a password" +private const val AN_INITIAL_DEVICE_NAME = "a device name" +private const val A_CAPTCHA_RESPONSE = "a captcha response" +private const val A_PID_CODE = "a pid code" +private const val EMAIL_VALIDATED_DELAY = 10000L +private val A_PID_TO_REGISTER = RegisterThreePid.Email("an email") + +class RegistrationActionHandlerTest { + + private val fakeRegistrationWizard = FakeRegistrationWizard() + private val registrationActionHandler = RegistrationActionHandler() + + @Test + fun `when handling register action then delegates to wizard`() = runBlockingTest { + val cases = listOf( + case(RegisterAction.StartRegistration) { getRegistrationFlow() }, + case(RegisterAction.CaptchaDone(A_CAPTCHA_RESPONSE)) { performReCaptcha(A_CAPTCHA_RESPONSE) }, + case(RegisterAction.AcceptTerms) { acceptTerms() }, + case(RegisterAction.RegisterDummy) { dummy() }, + case(RegisterAction.AddThreePid(A_PID_TO_REGISTER)) { addThreePid(A_PID_TO_REGISTER) }, + case(RegisterAction.SendAgainThreePid) { sendAgainThreePid() }, + case(RegisterAction.ValidateThreePid(A_PID_CODE)) { handleValidateThreePid(A_PID_CODE) }, + case(RegisterAction.CheckIfEmailHasBeenValidated(EMAIL_VALIDATED_DELAY)) { checkIfEmailHasBeenValidated(EMAIL_VALIDATED_DELAY) }, + case(RegisterAction.CreateAccount(A_USERNAME, A_PASSWORD, AN_INITIAL_DEVICE_NAME)) { + createAccount(A_USERNAME, A_PASSWORD, AN_INITIAL_DEVICE_NAME) + } + ) + + cases.forEach { testSuccessfulActionDelegation(it) } + } + + private suspend fun testSuccessfulActionDelegation(case: Case) { + fakeRegistrationWizard.givenSuccessFor(result = A_SESSION, case.expect) + + val result = registrationActionHandler.handleRegisterAction(fakeRegistrationWizard, case.action) + + result shouldBeEqualTo AN_EXPECTED_RESULT + } +} + +private fun case(action: RegisterAction, expect: suspend RegistrationWizard.() -> RegistrationResult) = Case(action, expect) + +private class Case(val action: RegisterAction, val expect: suspend RegistrationWizard.() -> RegistrationResult) diff --git a/vector/src/test/java/im/vector/app/test/fakes/FakeAuthenticationService.kt b/vector/src/test/java/im/vector/app/test/fakes/FakeAuthenticationService.kt index e1a605c7df9..10daf5de1eb 100644 --- a/vector/src/test/java/im/vector/app/test/fakes/FakeAuthenticationService.kt +++ b/vector/src/test/java/im/vector/app/test/fakes/FakeAuthenticationService.kt @@ -23,10 +23,15 @@ import org.matrix.android.sdk.api.auth.AuthenticationService import org.matrix.android.sdk.api.auth.registration.RegistrationWizard class FakeAuthenticationService : AuthenticationService by mockk() { + fun givenRegistrationWizard(registrationWizard: RegistrationWizard) { every { getRegistrationWizard() } returns registrationWizard } + fun givenRegistrationStarted(started: Boolean) { + every { isRegistrationStarted } returns started + } + fun expectReset() { coJustRun { reset() } } diff --git a/vector/src/test/java/im/vector/app/test/fakes/FakeRegisterActionHandler.kt b/vector/src/test/java/im/vector/app/test/fakes/FakeRegisterActionHandler.kt new file mode 100644 index 00000000000..6c01779a023 --- /dev/null +++ b/vector/src/test/java/im/vector/app/test/fakes/FakeRegisterActionHandler.kt @@ -0,0 +1,43 @@ +/* + * Copyright (c) 2022 New Vector Ltd + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package im.vector.app.test.fakes + +import im.vector.app.features.onboarding.RegisterAction +import im.vector.app.features.onboarding.RegistrationActionHandler +import io.mockk.coEvery +import io.mockk.mockk +import org.matrix.android.sdk.api.auth.registration.RegistrationResult +import org.matrix.android.sdk.api.auth.registration.RegistrationWizard + +class FakeRegisterActionHandler { + + val instance = mockk() + + fun givenResultFor(wizard: RegistrationWizard, action: RegisterAction, result: RegistrationResult) { + coEvery { instance.handleRegisterAction(wizard, action) } answers { + it.invocation.args.first() + result + } + } + + fun givenResultsFor(wizard: RegistrationWizard, result: List>) { + coEvery { instance.handleRegisterAction(wizard, any()) } answers { + val actionArg = it.invocation.args[1] as RegisterAction + result.first { it.first == actionArg }.second + } + } +} diff --git a/vector/src/test/java/im/vector/app/test/fakes/FakeRegistrationWizard.kt b/vector/src/test/java/im/vector/app/test/fakes/FakeRegistrationWizard.kt index 6ae394eea1d..2fc830e94a1 100644 --- a/vector/src/test/java/im/vector/app/test/fakes/FakeRegistrationWizard.kt +++ b/vector/src/test/java/im/vector/app/test/fakes/FakeRegistrationWizard.kt @@ -25,6 +25,14 @@ import org.matrix.android.sdk.api.session.Session class FakeRegistrationWizard : RegistrationWizard by mockk() { fun givenSuccessfulDummy(session: Session) { - coEvery { dummy() } returns RegistrationResult.Success(session) + givenSuccessFor(session) { dummy() } + } + + fun givenSuccessFor(result: Session, expect: suspend RegistrationWizard.() -> RegistrationResult) { + coEvery { expect(this@FakeRegistrationWizard) } returns RegistrationResult.Success(result) + } + + fun givenSuccessfulAcceptTerms(session: Session) { + coEvery { acceptTerms() } returns RegistrationResult.Success(session) } } From 434ee67982b9d2b00bc125081716b7dc60852928 Mon Sep 17 00:00:00 2001 From: Adam Brown Date: Wed, 2 Mar 2022 17:27:54 +0000 Subject: [PATCH 04/20] ensure the pid add/resend methods do not trigger the next registration steps - keeps the previous behaviour --- .../app/features/onboarding/OnboardingViewModel.kt | 11 ++++++++--- .../features/onboarding/RegistrationActionHandler.kt | 6 ++++++ 2 files changed, 14 insertions(+), 3 deletions(-) diff --git a/vector/src/main/java/im/vector/app/features/onboarding/OnboardingViewModel.kt b/vector/src/main/java/im/vector/app/features/onboarding/OnboardingViewModel.kt index 303a4d5950e..6c3c7b02c11 100644 --- a/vector/src/main/java/im/vector/app/features/onboarding/OnboardingViewModel.kt +++ b/vector/src/main/java/im/vector/app/features/onboarding/OnboardingViewModel.kt @@ -276,9 +276,14 @@ class OnboardingViewModel @AssistedInject constructor( kotlin.runCatching { registrationActionHandler.handleRegisterAction(registrationWizard, action) } .fold( onSuccess = { - when (it) { - is RegistrationResult.Success -> onSessionCreated(it.session, isAccountCreated = true) - is RegistrationResult.FlowResponse -> onFlowResponse(it.flowResult) + when { + action.ignoresResult() -> { + // do nothing + } + else -> when (it) { + is RegistrationResult.Success -> onSessionCreated(it.session, isAccountCreated = true) + is RegistrationResult.FlowResponse -> onFlowResponse(it.flowResult) + } } }, onFailure = { diff --git a/vector/src/main/java/im/vector/app/features/onboarding/RegistrationActionHandler.kt b/vector/src/main/java/im/vector/app/features/onboarding/RegistrationActionHandler.kt index 2938681ce76..b4998d2ba03 100644 --- a/vector/src/main/java/im/vector/app/features/onboarding/RegistrationActionHandler.kt +++ b/vector/src/main/java/im/vector/app/features/onboarding/RegistrationActionHandler.kt @@ -55,6 +55,12 @@ sealed interface RegisterAction { object RegisterDummy : RegisterAction } +fun RegisterAction.ignoresResult() = when (this) { + is RegisterAction.AddThreePid -> true + is RegisterAction.SendAgainThreePid -> true + else -> false +} + fun RegisterAction.hasLoadingState() = when (this) { is RegisterAction.CheckIfEmailHasBeenValidated -> false else -> true From b2a1aa17bdd4daf7250185e8132d102dd51d924c Mon Sep 17 00:00:00 2001 From: Adam Brown Date: Wed, 2 Mar 2022 17:28:08 +0000 Subject: [PATCH 05/20] adding commas to separate the test name sections --- .../onboarding/OnboardingViewModelTest.kt | 32 ++++++++----------- 1 file changed, 14 insertions(+), 18 deletions(-) diff --git a/vector/src/test/java/im/vector/app/features/onboarding/OnboardingViewModelTest.kt b/vector/src/test/java/im/vector/app/features/onboarding/OnboardingViewModelTest.kt index 44d44a94b21..a36c3cbb7a4 100644 --- a/vector/src/test/java/im/vector/app/features/onboarding/OnboardingViewModelTest.kt +++ b/vector/src/test/java/im/vector/app/features/onboarding/OnboardingViewModelTest.kt @@ -50,10 +50,6 @@ import org.matrix.android.sdk.api.session.homeserver.HomeServerCapabilities private const val A_DISPLAY_NAME = "a display name" private const val A_PICTURE_FILENAME = "a-picture.png" private val AN_ERROR = RuntimeException("an error!") -private val AN_UNSUPPORTED_PERSONALISATION_STATE = PersonalizationState( - supportsChangingDisplayName = false, - supportsChangingProfilePicture = false -) private val A_LOADABLE_REGISTER_ACTION = RegisterAction.StartRegistration class OnboardingViewModelTest { @@ -78,7 +74,7 @@ class OnboardingViewModelTest { } @Test - fun `when handling PostViewEvent then emits contents as view event`() = runBlockingTest { + fun `when handling PostViewEvent, then emits contents as view event`() = runBlockingTest { val test = viewModel.test(this) viewModel.handle(OnboardingAction.PostViewEvent(OnboardingViewEvents.OnTakeMeHome)) @@ -89,7 +85,7 @@ class OnboardingViewModelTest { } @Test - fun `given supports changing display name when handling PersonalizeProfile then emits contents choose display name`() = runBlockingTest { + fun `given supports changing display name, when handling PersonalizeProfile, then emits contents choose display name`() = runBlockingTest { val initialState = initialState.copy(personalizationState = PersonalizationState(supportsChangingDisplayName = true, supportsChangingProfilePicture = false)) viewModel = createViewModel(initialState) val test = viewModel.test(this) @@ -102,7 +98,7 @@ class OnboardingViewModelTest { } @Test - fun `given only supports changing profile picture when handling PersonalizeProfile then emits contents choose profile picture`() = runBlockingTest { + fun `given only supports changing profile picture, when handling PersonalizeProfile, then emits contents choose profile picture`() = runBlockingTest { val initialState = initialState.copy(personalizationState = PersonalizationState(supportsChangingDisplayName = false, supportsChangingProfilePicture = true)) viewModel = createViewModel(initialState) val test = viewModel.test(this) @@ -115,7 +111,7 @@ class OnboardingViewModelTest { } @Test - fun `given register action requires more steps when handling action then posts next steps`() = runBlockingTest { + fun `given register action requires more steps, when handling action, then posts next steps`() = runBlockingTest { val test = viewModel.test(this) val flowResult = FlowResult(missingStages = listOf(Stage.Email(true)), completedStages = listOf(Stage.Email(true))) givenRegistrationResultFor(A_LOADABLE_REGISTER_ACTION, RegistrationResult.FlowResponse(flowResult)) @@ -133,7 +129,7 @@ class OnboardingViewModelTest { } @Test - fun `given registration has started and has dummy step to do when handling action then ignores other steps and executes dummy`() = runBlockingTest { + fun `given registration has started and has dummy step to do, when handling action, then ignores other steps and executes dummy`() = runBlockingTest { val test = viewModel.test(this) val homeServerCapabilities = HomeServerCapabilities(canChangeDisplayName = true, canChangeAvatar = true) @@ -160,7 +156,7 @@ class OnboardingViewModelTest { } @Test - fun `given homeserver does not support personalisation when registering account then updates state and emits account created event`() = runBlockingTest { + fun `given homeserver does not support personalisation, when registering account, then updates state and emits account created event`() = runBlockingTest { val homeServerCapabilities = HomeServerCapabilities(canChangeDisplayName = false, canChangeAvatar = false) fakeSession.fakeHomeServerCapabilitiesService.givenCapabilities(homeServerCapabilities) givenRegistrationResultFor(A_LOADABLE_REGISTER_ACTION, RegistrationResult.Success(fakeSession)) @@ -181,7 +177,7 @@ class OnboardingViewModelTest { } @Test - fun `given changing profile picture is supported when updating display name then updates upstream user display name and moves to choose profile picture`() = runBlockingTest { + fun `given changing profile picture is supported, when updating display name, then updates upstream user display name and moves to choose profile picture`() = runBlockingTest { val personalisedInitialState = initialState.copy(personalizationState = PersonalizationState(supportsChangingProfilePicture = true)) viewModel = createViewModel(personalisedInitialState) val test = viewModel.test(this) @@ -196,7 +192,7 @@ class OnboardingViewModelTest { } @Test - fun `given changing profile picture is not supported when updating display name then updates upstream user display name and completes personalization`() = runBlockingTest { + fun `given changing profile picture is not supported, when updating display name, then updates upstream user display name and completes personalization`() = runBlockingTest { val personalisedInitialState = initialState.copy(personalizationState = PersonalizationState(supportsChangingProfilePicture = false)) viewModel = createViewModel(personalisedInitialState) val test = viewModel.test(this) @@ -211,7 +207,7 @@ class OnboardingViewModelTest { } @Test - fun `given upstream failure when handling display name update then emits failure event`() = runBlockingTest { + fun `given upstream failure, when handling display name update, then emits failure event`() = runBlockingTest { val test = viewModel.test(this) fakeSession.fakeProfileService.givenSetDisplayNameErrors(AN_ERROR) @@ -228,7 +224,7 @@ class OnboardingViewModelTest { } @Test - fun `when handling profile picture selected then updates selected picture state`() = runBlockingTest { + fun `when handling profile picture selected, then updates selected picture state`() = runBlockingTest { val test = viewModel.test(this) viewModel.handle(OnboardingAction.ProfilePictureSelected(fakeUri.instance)) @@ -243,7 +239,7 @@ class OnboardingViewModelTest { } @Test - fun `given a selected picture when handling save selected profile picture then updates upstream avatar and completes personalization`() = runBlockingTest { + fun `given a selected picture, when handling save selected profile picture, then updates upstream avatar and completes personalization`() = runBlockingTest { val initialStateWithPicture = givenPictureSelected(fakeUri.instance, A_PICTURE_FILENAME) viewModel = createViewModel(initialStateWithPicture) val test = viewModel.test(this) @@ -258,7 +254,7 @@ class OnboardingViewModelTest { } @Test - fun `given upstream update avatar fails when saving selected profile picture then emits failure event`() = runBlockingTest { + fun `given upstream update avatar fails, when saving selected profile picture, then emits failure event`() = runBlockingTest { fakeSession.fakeProfileService.givenUpdateAvatarErrors(AN_ERROR) val initialStateWithPicture = givenPictureSelected(fakeUri.instance, A_PICTURE_FILENAME) viewModel = createViewModel(initialStateWithPicture) @@ -273,7 +269,7 @@ class OnboardingViewModelTest { } @Test - fun `given no selected picture when saving selected profile picture then emits failure event`() = runBlockingTest { + fun `given no selected picture, when saving selected profile picture, then emits failure event`() = runBlockingTest { val test = viewModel.test(this) viewModel.handle(OnboardingAction.SaveSelectedProfilePicture) @@ -285,7 +281,7 @@ class OnboardingViewModelTest { } @Test - fun `when handling profile picture skipped then completes personalization`() = runBlockingTest { + fun `when handling profile skipped, then completes personalization`() = runBlockingTest { val test = viewModel.test(this) viewModel.handle(OnboardingAction.UpdateProfilePictureSkipped) From 75cbb727a443e78f972072b485c5b96d7d83bb9b Mon Sep 17 00:00:00 2001 From: Adam Brown Date: Wed, 2 Mar 2022 17:45:29 +0000 Subject: [PATCH 06/20] cleaning up test names and bodies to be clearer --- .../onboarding/OnboardingViewModelTest.kt | 45 ++++++++++--------- .../fixtures/HomeserverCapabilityFixture.kt | 40 +++++++++++++++++ 2 files changed, 63 insertions(+), 22 deletions(-) create mode 100644 vector/src/test/java/im/vector/app/test/fixtures/HomeserverCapabilityFixture.kt diff --git a/vector/src/test/java/im/vector/app/features/onboarding/OnboardingViewModelTest.kt b/vector/src/test/java/im/vector/app/features/onboarding/OnboardingViewModelTest.kt index a36c3cbb7a4..b485a3fa44c 100644 --- a/vector/src/test/java/im/vector/app/features/onboarding/OnboardingViewModelTest.kt +++ b/vector/src/test/java/im/vector/app/features/onboarding/OnboardingViewModelTest.kt @@ -37,6 +37,7 @@ import im.vector.app.test.fakes.FakeUri import im.vector.app.test.fakes.FakeUriFilenameResolver import im.vector.app.test.fakes.FakeVectorFeatures import im.vector.app.test.fakes.FakeVectorOverrides +import im.vector.app.test.fixtures.aHomeServerCapabilities import im.vector.app.test.test import kotlinx.coroutines.test.runBlockingTest import org.junit.Before @@ -51,6 +52,7 @@ private const val A_DISPLAY_NAME = "a display name" private const val A_PICTURE_FILENAME = "a-picture.png" private val AN_ERROR = RuntimeException("an error!") private val A_LOADABLE_REGISTER_ACTION = RegisterAction.StartRegistration +private val A_HOMESERVER_CAPABILITIES = aHomeServerCapabilities(canChangeDisplayName = true, canChangeAvatar = true) class OnboardingViewModelTest { @@ -129,38 +131,27 @@ class OnboardingViewModelTest { } @Test - fun `given registration has started and has dummy step to do, when handling action, then ignores other steps and executes dummy`() = runBlockingTest { + fun `when registering account, then updates state and emits account created event`() = runBlockingTest { + givenRegistrationResultFor(A_LOADABLE_REGISTER_ACTION, RegistrationResult.Success(fakeSession)) + givenSuccessfullyCreatesAccount(A_HOMESERVER_CAPABILITIES) val test = viewModel.test(this) - val homeServerCapabilities = HomeServerCapabilities(canChangeDisplayName = true, canChangeAvatar = true) - fakeSession.fakeHomeServerCapabilitiesService.givenCapabilities(homeServerCapabilities) - val flowResult = FlowResult(missingStages = listOf(Stage.Dummy(mandatory = true), Stage.Email(true)), completedStages = emptyList()) - givenRegistrationResultsFor(listOf( - A_LOADABLE_REGISTER_ACTION to RegistrationResult.FlowResponse(flowResult), - RegisterAction.RegisterDummy to RegistrationResult.Success(fakeSession) - )) - givenSuccessfullyCreatesAccount() - viewModel.handle(OnboardingAction.PostRegisterAction(A_LOADABLE_REGISTER_ACTION)) test .assertStatesWithPrevious( initialState, { copy(asyncRegistration = Loading()) }, - { copy(asyncLoginAction = Success(Unit), personalizationState = homeServerCapabilities.toPersonalisationState()) }, - { copy(asyncRegistration = Uninitialized) }, - - ) + { copy(asyncLoginAction = Success(Unit), personalizationState = A_HOMESERVER_CAPABILITIES.toPersonalisationState()) }, + { copy(asyncLoginAction = Success(Unit), asyncRegistration = Uninitialized) } + ) .assertEvents(OnboardingViewEvents.OnAccountCreated) .finish() } @Test - fun `given homeserver does not support personalisation, when registering account, then updates state and emits account created event`() = runBlockingTest { - val homeServerCapabilities = HomeServerCapabilities(canChangeDisplayName = false, canChangeAvatar = false) - fakeSession.fakeHomeServerCapabilitiesService.givenCapabilities(homeServerCapabilities) - givenRegistrationResultFor(A_LOADABLE_REGISTER_ACTION, RegistrationResult.Success(fakeSession)) - givenSuccessfullyCreatesAccount() + fun `given registration has started and has dummy step to do, when handling action, then ignores other steps and executes dummy`() = runBlockingTest { + givenSuccessfulRegistrationForStartAndDummySteps(missingStages = listOf(Stage.Dummy(mandatory = true))) val test = viewModel.test(this) viewModel.handle(OnboardingAction.PostRegisterAction(A_LOADABLE_REGISTER_ACTION)) @@ -169,8 +160,8 @@ class OnboardingViewModelTest { .assertStatesWithPrevious( initialState, { copy(asyncRegistration = Loading()) }, - { copy(asyncLoginAction = Success(Unit), personalizationState = homeServerCapabilities.toPersonalisationState()) }, - { copy(asyncLoginAction = Success(Unit), asyncRegistration = Uninitialized) } + { copy(asyncLoginAction = Success(Unit), personalizationState = A_HOMESERVER_CAPABILITIES.toPersonalisationState()) }, + { copy(asyncRegistration = Uninitialized) } ) .assertEvents(OnboardingViewEvents.OnAccountCreated) .finish() @@ -339,7 +330,17 @@ class OnboardingViewModelTest { ) } - private fun givenSuccessfullyCreatesAccount() { + private fun givenSuccessfulRegistrationForStartAndDummySteps(missingStages: List) { + val flowResult = FlowResult(missingStages = missingStages, completedStages = emptyList()) + givenRegistrationResultsFor(listOf( + A_LOADABLE_REGISTER_ACTION to RegistrationResult.FlowResponse(flowResult), + RegisterAction.RegisterDummy to RegistrationResult.Success(fakeSession) + )) + givenSuccessfullyCreatesAccount(A_HOMESERVER_CAPABILITIES) + } + + private fun givenSuccessfullyCreatesAccount(homeServerCapabilities: HomeServerCapabilities) { + fakeSession.fakeHomeServerCapabilitiesService.givenCapabilities(homeServerCapabilities) fakeActiveSessionHolder.expectSetsActiveSession(fakeSession) fakeAuthenticationService.expectReset() fakeSession.expectStartsSyncing() diff --git a/vector/src/test/java/im/vector/app/test/fixtures/HomeserverCapabilityFixture.kt b/vector/src/test/java/im/vector/app/test/fixtures/HomeserverCapabilityFixture.kt new file mode 100644 index 00000000000..a4d9869a894 --- /dev/null +++ b/vector/src/test/java/im/vector/app/test/fixtures/HomeserverCapabilityFixture.kt @@ -0,0 +1,40 @@ +/* + * Copyright (c) 2022 New Vector Ltd + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package im.vector.app.test.fixtures + +import org.matrix.android.sdk.api.session.homeserver.HomeServerCapabilities +import org.matrix.android.sdk.api.session.homeserver.RoomVersionCapabilities + +fun aHomeServerCapabilities( + canChangePassword: Boolean = true, + canChangeDisplayName: Boolean = true, + canChangeAvatar: Boolean = true, + canChange3pid: Boolean = true, + maxUploadFileSize: Long = 100L, + lastVersionIdentityServerSupported: Boolean = false, + defaultIdentityServerUrl: String? = null, + roomVersions: RoomVersionCapabilities? = null +) = HomeServerCapabilities( + canChangePassword, + canChangeDisplayName, + canChangeAvatar, + canChange3pid, + maxUploadFileSize, + lastVersionIdentityServerSupported, + defaultIdentityServerUrl, + roomVersions +) From 804513c808de54f318f808cc2d8c07e28d130d6b Mon Sep 17 00:00:00 2001 From: Adam Brown Date: Wed, 2 Mar 2022 17:53:12 +0000 Subject: [PATCH 07/20] adding case for result ignoring register actions --- .../onboarding/OnboardingViewModelTest.kt | 20 +++++++++++++++++++ 1 file changed, 20 insertions(+) diff --git a/vector/src/test/java/im/vector/app/features/onboarding/OnboardingViewModelTest.kt b/vector/src/test/java/im/vector/app/features/onboarding/OnboardingViewModelTest.kt index b485a3fa44c..3c8279bafaf 100644 --- a/vector/src/test/java/im/vector/app/features/onboarding/OnboardingViewModelTest.kt +++ b/vector/src/test/java/im/vector/app/features/onboarding/OnboardingViewModelTest.kt @@ -44,6 +44,7 @@ import org.junit.Before import org.junit.Rule import org.junit.Test import org.matrix.android.sdk.api.auth.registration.FlowResult +import org.matrix.android.sdk.api.auth.registration.RegisterThreePid import org.matrix.android.sdk.api.auth.registration.RegistrationResult import org.matrix.android.sdk.api.auth.registration.Stage import org.matrix.android.sdk.api.session.homeserver.HomeServerCapabilities @@ -52,7 +53,9 @@ private const val A_DISPLAY_NAME = "a display name" private const val A_PICTURE_FILENAME = "a-picture.png" private val AN_ERROR = RuntimeException("an error!") private val A_LOADABLE_REGISTER_ACTION = RegisterAction.StartRegistration +private val A_RESULT_IGNORED_REGISTER_ACTION = RegisterAction.AddThreePid(RegisterThreePid.Email("an email")) private val A_HOMESERVER_CAPABILITIES = aHomeServerCapabilities(canChangeDisplayName = true, canChangeAvatar = true) +private val AN_IGNORED_FLOW_RESULT = FlowResult(missingStages = emptyList(), completedStages = emptyList()) class OnboardingViewModelTest { @@ -130,6 +133,23 @@ class OnboardingViewModelTest { .finish() } + @Test + fun `given register action ignores result, when handling action, then does nothing on success`() = runBlockingTest { + val test = viewModel.test(this) + givenRegistrationResultFor(A_RESULT_IGNORED_REGISTER_ACTION, RegistrationResult.FlowResponse(AN_IGNORED_FLOW_RESULT)) + + viewModel.handle(OnboardingAction.PostRegisterAction(A_RESULT_IGNORED_REGISTER_ACTION)) + + test + .assertStatesWithPrevious( + initialState, + { copy(asyncRegistration = Loading()) }, + { copy(asyncRegistration = Uninitialized) } + ) + .assertNoEvents() + .finish() + } + @Test fun `when registering account, then updates state and emits account created event`() = runBlockingTest { givenRegistrationResultFor(A_LOADABLE_REGISTER_ACTION, RegistrationResult.Success(fakeSession)) From 390ae4344df08f559ce1a02b3276b4c9fdf99edb Mon Sep 17 00:00:00 2001 From: Adam Brown Date: Wed, 2 Mar 2022 18:07:13 +0000 Subject: [PATCH 08/20] allowing test withPrevious to be supplied a list --- .../features/onboarding/OnboardingViewModelTest.kt | 14 +++++--------- .../src/test/java/im/vector/app/test/Extensions.kt | 4 ++++ 2 files changed, 9 insertions(+), 9 deletions(-) diff --git a/vector/src/test/java/im/vector/app/features/onboarding/OnboardingViewModelTest.kt b/vector/src/test/java/im/vector/app/features/onboarding/OnboardingViewModelTest.kt index 3c8279bafaf..b6e92655c0c 100644 --- a/vector/src/test/java/im/vector/app/features/onboarding/OnboardingViewModelTest.kt +++ b/vector/src/test/java/im/vector/app/features/onboarding/OnboardingViewModelTest.kt @@ -196,7 +196,7 @@ class OnboardingViewModelTest { viewModel.handle(OnboardingAction.UpdateDisplayName(A_DISPLAY_NAME)) test - .assertStates(expectedSuccessfulDisplayNameUpdateStates(personalisedInitialState)) + .assertStatesWithPrevious(personalisedInitialState, expectedSuccessfulDisplayNameUpdateStates()) .assertEvents(OnboardingViewEvents.OnChooseProfilePicture) .finish() fakeSession.fakeProfileService.verifyUpdatedName(fakeSession.myUserId, A_DISPLAY_NAME) @@ -211,7 +211,7 @@ class OnboardingViewModelTest { viewModel.handle(OnboardingAction.UpdateDisplayName(A_DISPLAY_NAME)) test - .assertStates(expectedSuccessfulDisplayNameUpdateStates(personalisedInitialState)) + .assertStatesWithPrevious(personalisedInitialState, expectedSuccessfulDisplayNameUpdateStates()) .assertEvents(OnboardingViewEvents.OnPersonalizationComplete) .finish() fakeSession.fakeProfileService.verifyUpdatedName(fakeSession.myUserId, A_DISPLAY_NAME) @@ -339,14 +339,10 @@ class OnboardingViewModelTest { state.copy(asyncProfilePicture = Fail(cause)) ) - private fun expectedSuccessfulDisplayNameUpdateStates(personalisedInitialState: OnboardingViewState): List { + private fun expectedSuccessfulDisplayNameUpdateStates(): List OnboardingViewState> { return listOf( - personalisedInitialState, - personalisedInitialState.copy(asyncDisplayName = Loading()), - personalisedInitialState.copy( - asyncDisplayName = Success(Unit), - personalizationState = personalisedInitialState.personalizationState.copy(displayName = A_DISPLAY_NAME) - ) + { copy(asyncDisplayName = Loading()) }, + { copy(asyncDisplayName = Success(Unit), personalizationState = personalizationState.copy(displayName = A_DISPLAY_NAME)) } ) } diff --git a/vector/src/test/java/im/vector/app/test/Extensions.kt b/vector/src/test/java/im/vector/app/test/Extensions.kt index 136ca58d68b..5fe07b967a3 100644 --- a/vector/src/test/java/im/vector/app/test/Extensions.kt +++ b/vector/src/test/java/im/vector/app/test/Extensions.kt @@ -56,6 +56,10 @@ class ViewModelTest( } fun assertStatesWithPrevious(initial: S, vararg expected: S.() -> S): ViewModelTest { + return assertStatesWithPrevious(initial, expected.toList()) + } + + fun assertStatesWithPrevious(initial: S, expected: List S>): ViewModelTest { val reducedExpectedStates = expected.fold(mutableListOf(initial)) { acc, curr -> val next = curr.invoke(acc.last()) acc.add(next) From 694016fc164139739ad45977fc4f4b83876e289d Mon Sep 17 00:00:00 2001 From: Adam Brown Date: Wed, 2 Mar 2022 18:14:45 +0000 Subject: [PATCH 09/20] adding test case for the non loading registration steps --- .../onboarding/OnboardingViewModelTest.kt | 20 ++++++++++++++++--- 1 file changed, 17 insertions(+), 3 deletions(-) diff --git a/vector/src/test/java/im/vector/app/features/onboarding/OnboardingViewModelTest.kt b/vector/src/test/java/im/vector/app/features/onboarding/OnboardingViewModelTest.kt index b6e92655c0c..ec9851ebc8a 100644 --- a/vector/src/test/java/im/vector/app/features/onboarding/OnboardingViewModelTest.kt +++ b/vector/src/test/java/im/vector/app/features/onboarding/OnboardingViewModelTest.kt @@ -53,9 +53,11 @@ private const val A_DISPLAY_NAME = "a display name" private const val A_PICTURE_FILENAME = "a-picture.png" private val AN_ERROR = RuntimeException("an error!") private val A_LOADABLE_REGISTER_ACTION = RegisterAction.StartRegistration +private val A_NON_LOADABLE_REGISTER_ACTION = RegisterAction.CheckIfEmailHasBeenValidated(delayMillis = -1L) private val A_RESULT_IGNORED_REGISTER_ACTION = RegisterAction.AddThreePid(RegisterThreePid.Email("an email")) private val A_HOMESERVER_CAPABILITIES = aHomeServerCapabilities(canChangeDisplayName = true, canChangeAvatar = true) private val AN_IGNORED_FLOW_RESULT = FlowResult(missingStages = emptyList(), completedStages = emptyList()) +private val ANY_CONTINUING_REGISTRATION_RESULT = RegistrationResult.FlowResponse(AN_IGNORED_FLOW_RESULT) class OnboardingViewModelTest { @@ -118,8 +120,7 @@ class OnboardingViewModelTest { @Test fun `given register action requires more steps, when handling action, then posts next steps`() = runBlockingTest { val test = viewModel.test(this) - val flowResult = FlowResult(missingStages = listOf(Stage.Email(true)), completedStages = listOf(Stage.Email(true))) - givenRegistrationResultFor(A_LOADABLE_REGISTER_ACTION, RegistrationResult.FlowResponse(flowResult)) + givenRegistrationResultFor(A_LOADABLE_REGISTER_ACTION, ANY_CONTINUING_REGISTRATION_RESULT) viewModel.handle(OnboardingAction.PostRegisterAction(A_LOADABLE_REGISTER_ACTION)) @@ -129,7 +130,20 @@ class OnboardingViewModelTest { { copy(asyncRegistration = Loading()) }, { copy(asyncRegistration = Uninitialized) } ) - .assertEvents(OnboardingViewEvents.RegistrationFlowResult(flowResult, isRegistrationStarted = true)) + .assertEvents(OnboardingViewEvents.RegistrationFlowResult(ANY_CONTINUING_REGISTRATION_RESULT.flowResult, isRegistrationStarted = true)) + .finish() + } + + @Test + fun `given register action is non loadable, when handling action, then posts next steps without loading`() = runBlockingTest { + val test = viewModel.test(this) + givenRegistrationResultFor(A_NON_LOADABLE_REGISTER_ACTION, ANY_CONTINUING_REGISTRATION_RESULT) + + viewModel.handle(OnboardingAction.PostRegisterAction(A_NON_LOADABLE_REGISTER_ACTION)) + + test + .assertState(initialState) + .assertEvents(OnboardingViewEvents.RegistrationFlowResult(ANY_CONTINUING_REGISTRATION_RESULT.flowResult, isRegistrationStarted = true)) .finish() } From fc5c0579bbc5b22dac1198f52f903b7723d06d85 Mon Sep 17 00:00:00 2001 From: Adam Brown Date: Wed, 2 Mar 2022 18:24:22 +0000 Subject: [PATCH 10/20] adding changelog entry --- changelog.d/5408.misc | 1 + 1 file changed, 1 insertion(+) create mode 100644 changelog.d/5408.misc diff --git a/changelog.d/5408.misc b/changelog.d/5408.misc new file mode 100644 index 00000000000..3807ee1da83 --- /dev/null +++ b/changelog.d/5408.misc @@ -0,0 +1 @@ +Improved onboarding registration unit test coverage \ No newline at end of file From fe206fe130f4729f61785fa4000e035464a5e2ef Mon Sep 17 00:00:00 2001 From: Adam Brown Date: Fri, 4 Mar 2022 14:00:59 +0000 Subject: [PATCH 11/20] fixing wrong action for starting the sign up --- .../onboarding/OnboardingViewModel.kt | 2 +- .../onboarding/OnboardingViewModelTest.kt | 19 +++++++++++++++++++ 2 files changed, 20 insertions(+), 1 deletion(-) diff --git a/vector/src/main/java/im/vector/app/features/onboarding/OnboardingViewModel.kt b/vector/src/main/java/im/vector/app/features/onboarding/OnboardingViewModel.kt index 6c3c7b02c11..cfa65e0874a 100644 --- a/vector/src/main/java/im/vector/app/features/onboarding/OnboardingViewModel.kt +++ b/vector/src/main/java/im/vector/app/features/onboarding/OnboardingViewModel.kt @@ -373,7 +373,7 @@ class OnboardingViewModel @AssistedInject constructor( } when (action.signMode) { - SignMode.SignUp -> handleRegisterAction(RegisterAction.RegisterDummy) + SignMode.SignUp -> handleRegisterAction(RegisterAction.StartRegistration) SignMode.SignIn -> startAuthenticationFlow() SignMode.SignInWithMatrixId -> _viewEvents.post(OnboardingViewEvents.OnSignModeSelected(SignMode.SignInWithMatrixId)) SignMode.Unknown -> Unit diff --git a/vector/src/test/java/im/vector/app/features/onboarding/OnboardingViewModelTest.kt b/vector/src/test/java/im/vector/app/features/onboarding/OnboardingViewModelTest.kt index ec9851ebc8a..5d419c09c99 100644 --- a/vector/src/test/java/im/vector/app/features/onboarding/OnboardingViewModelTest.kt +++ b/vector/src/test/java/im/vector/app/features/onboarding/OnboardingViewModelTest.kt @@ -23,6 +23,7 @@ import com.airbnb.mvrx.Success import com.airbnb.mvrx.Uninitialized import com.airbnb.mvrx.test.MvRxTestRule import im.vector.app.features.login.ReAuthHelper +import im.vector.app.features.login.SignMode import im.vector.app.test.fakes.FakeActiveSessionHolder import im.vector.app.test.fakes.FakeAnalyticsTracker import im.vector.app.test.fakes.FakeAuthenticationService @@ -117,6 +118,24 @@ class OnboardingViewModelTest { .finish() } + @Test + fun `when handling SignUp then sets sign mode to sign up and starts registration`() = runBlockingTest { + givenRegistrationResultFor(RegisterAction.StartRegistration, ANY_CONTINUING_REGISTRATION_RESULT) + val test = viewModel.test(this) + + viewModel.handle(OnboardingAction.UpdateSignMode(SignMode.SignUp)) + + test + .assertStatesWithPrevious( + initialState, + { copy(signMode = SignMode.SignUp) }, + { copy(asyncRegistration = Loading()) }, + { copy(asyncRegistration = Uninitialized) } + ) + .assertEvents(OnboardingViewEvents.RegistrationFlowResult(ANY_CONTINUING_REGISTRATION_RESULT.flowResult, isRegistrationStarted = true)) + .finish() + } + @Test fun `given register action requires more steps, when handling action, then posts next steps`() = runBlockingTest { val test = viewModel.test(this) From 3d20d46eb3f0d440db2faf3fc0d53e985bd67308 Mon Sep 17 00:00:00 2001 From: Adam Brown Date: Mon, 7 Mar 2022 17:40:07 +0000 Subject: [PATCH 12/20] enabling the personalize step for the unit tests preemptively for the feature to be enabled --- .../test/java/im/vector/app/test/fakes/FakeVectorFeatures.kt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/vector/src/test/java/im/vector/app/test/fakes/FakeVectorFeatures.kt b/vector/src/test/java/im/vector/app/test/fakes/FakeVectorFeatures.kt index 265941a5316..b6e06bcddaf 100644 --- a/vector/src/test/java/im/vector/app/test/fakes/FakeVectorFeatures.kt +++ b/vector/src/test/java/im/vector/app/test/fakes/FakeVectorFeatures.kt @@ -23,5 +23,5 @@ class FakeVectorFeatures : VectorFeatures { override fun isOnboardingAlreadyHaveAccountSplashEnabled() = true override fun isOnboardingSplashCarouselEnabled() = true override fun isOnboardingUseCaseEnabled() = true - override fun isOnboardingPersonalizeEnabled() = false + override fun isOnboardingPersonalizeEnabled() = true } From d77061b2295dab51b7aaabf41468c7c4dd4e0cfc Mon Sep 17 00:00:00 2001 From: Adam Brown Date: Thu, 17 Mar 2022 16:38:42 +0000 Subject: [PATCH 13/20] removing fully qualified import --- .../im/vector/app/features/onboarding/OnboardingViewModel.kt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/vector/src/main/java/im/vector/app/features/onboarding/OnboardingViewModel.kt b/vector/src/main/java/im/vector/app/features/onboarding/OnboardingViewModel.kt index cfa65e0874a..3f2b5adcbe6 100644 --- a/vector/src/main/java/im/vector/app/features/onboarding/OnboardingViewModel.kt +++ b/vector/src/main/java/im/vector/app/features/onboarding/OnboardingViewModel.kt @@ -273,7 +273,7 @@ class OnboardingViewModel @AssistedInject constructor( if (action.hasLoadingState()) { setState { copy(asyncRegistration = Loading()) } } - kotlin.runCatching { registrationActionHandler.handleRegisterAction(registrationWizard, action) } + runCatching { registrationActionHandler.handleRegisterAction(registrationWizard, action) } .fold( onSuccess = { when { From 5df2ae9ae2e65e92c624ac57f862b65ef3809b4f Mon Sep 17 00:00:00 2001 From: Adam Brown Date: Thu, 17 Mar 2022 16:50:20 +0000 Subject: [PATCH 14/20] updating with previous state helper and including javadoc to help explain its usage --- .../onboarding/OnboardingViewModelTest.kt | 16 ++++++++-------- .../test/java/im/vector/app/test/Extensions.kt | 10 +++++++--- 2 files changed, 15 insertions(+), 11 deletions(-) diff --git a/vector/src/test/java/im/vector/app/features/onboarding/OnboardingViewModelTest.kt b/vector/src/test/java/im/vector/app/features/onboarding/OnboardingViewModelTest.kt index 5d419c09c99..f6c322af403 100644 --- a/vector/src/test/java/im/vector/app/features/onboarding/OnboardingViewModelTest.kt +++ b/vector/src/test/java/im/vector/app/features/onboarding/OnboardingViewModelTest.kt @@ -126,7 +126,7 @@ class OnboardingViewModelTest { viewModel.handle(OnboardingAction.UpdateSignMode(SignMode.SignUp)) test - .assertStatesWithPrevious( + .assertStatesChanges( initialState, { copy(signMode = SignMode.SignUp) }, { copy(asyncRegistration = Loading()) }, @@ -144,7 +144,7 @@ class OnboardingViewModelTest { viewModel.handle(OnboardingAction.PostRegisterAction(A_LOADABLE_REGISTER_ACTION)) test - .assertStatesWithPrevious( + .assertStatesChanges( initialState, { copy(asyncRegistration = Loading()) }, { copy(asyncRegistration = Uninitialized) } @@ -174,7 +174,7 @@ class OnboardingViewModelTest { viewModel.handle(OnboardingAction.PostRegisterAction(A_RESULT_IGNORED_REGISTER_ACTION)) test - .assertStatesWithPrevious( + .assertStatesChanges( initialState, { copy(asyncRegistration = Loading()) }, { copy(asyncRegistration = Uninitialized) } @@ -192,7 +192,7 @@ class OnboardingViewModelTest { viewModel.handle(OnboardingAction.PostRegisterAction(A_LOADABLE_REGISTER_ACTION)) test - .assertStatesWithPrevious( + .assertStatesChanges( initialState, { copy(asyncRegistration = Loading()) }, { copy(asyncLoginAction = Success(Unit), personalizationState = A_HOMESERVER_CAPABILITIES.toPersonalisationState()) }, @@ -210,7 +210,7 @@ class OnboardingViewModelTest { viewModel.handle(OnboardingAction.PostRegisterAction(A_LOADABLE_REGISTER_ACTION)) test - .assertStatesWithPrevious( + .assertStatesChanges( initialState, { copy(asyncRegistration = Loading()) }, { copy(asyncLoginAction = Success(Unit), personalizationState = A_HOMESERVER_CAPABILITIES.toPersonalisationState()) }, @@ -229,7 +229,7 @@ class OnboardingViewModelTest { viewModel.handle(OnboardingAction.UpdateDisplayName(A_DISPLAY_NAME)) test - .assertStatesWithPrevious(personalisedInitialState, expectedSuccessfulDisplayNameUpdateStates()) + .assertStatesChanges(personalisedInitialState, expectedSuccessfulDisplayNameUpdateStates()) .assertEvents(OnboardingViewEvents.OnChooseProfilePicture) .finish() fakeSession.fakeProfileService.verifyUpdatedName(fakeSession.myUserId, A_DISPLAY_NAME) @@ -244,7 +244,7 @@ class OnboardingViewModelTest { viewModel.handle(OnboardingAction.UpdateDisplayName(A_DISPLAY_NAME)) test - .assertStatesWithPrevious(personalisedInitialState, expectedSuccessfulDisplayNameUpdateStates()) + .assertStatesChanges(personalisedInitialState, expectedSuccessfulDisplayNameUpdateStates()) .assertEvents(OnboardingViewEvents.OnPersonalizationComplete) .finish() fakeSession.fakeProfileService.verifyUpdatedName(fakeSession.myUserId, A_DISPLAY_NAME) @@ -258,7 +258,7 @@ class OnboardingViewModelTest { viewModel.handle(OnboardingAction.UpdateDisplayName(A_DISPLAY_NAME)) test - .assertStatesWithPrevious( + .assertStatesChanges( initialState, { copy(asyncDisplayName = Loading()) }, { copy(asyncDisplayName = Fail(AN_ERROR)) }, diff --git a/vector/src/test/java/im/vector/app/test/Extensions.kt b/vector/src/test/java/im/vector/app/test/Extensions.kt index 5fe07b967a3..67eff7ca11d 100644 --- a/vector/src/test/java/im/vector/app/test/Extensions.kt +++ b/vector/src/test/java/im/vector/app/test/Extensions.kt @@ -55,11 +55,15 @@ class ViewModelTest( return this } - fun assertStatesWithPrevious(initial: S, vararg expected: S.() -> S): ViewModelTest { - return assertStatesWithPrevious(initial, expected.toList()) + fun assertStatesChanges(initial: S, vararg expected: S.() -> S): ViewModelTest { + return assertStatesChanges(initial, expected.toList()) } - fun assertStatesWithPrevious(initial: S, expected: List S>): ViewModelTest { + /** + * Asserts the expected states are in the same order as the actual state emissions + * Each expected lambda is given the previous expected state, starting with the initial + */ + fun assertStatesChanges(initial: S, expected: List S>): ViewModelTest { val reducedExpectedStates = expected.fold(mutableListOf(initial)) { acc, curr -> val next = curr.invoke(acc.last()) acc.add(next) From d514751ffd5f8e5ea6f35c1837235d29d995b428 Mon Sep 17 00:00:00 2001 From: Adam Brown Date: Thu, 17 Mar 2022 16:52:37 +0000 Subject: [PATCH 15/20] avoiding shadowed lambda parameters --- .../im/vector/app/test/fakes/FakeRegisterActionHandler.kt | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/vector/src/test/java/im/vector/app/test/fakes/FakeRegisterActionHandler.kt b/vector/src/test/java/im/vector/app/test/fakes/FakeRegisterActionHandler.kt index 6c01779a023..3122e29c3b4 100644 --- a/vector/src/test/java/im/vector/app/test/fakes/FakeRegisterActionHandler.kt +++ b/vector/src/test/java/im/vector/app/test/fakes/FakeRegisterActionHandler.kt @@ -28,15 +28,15 @@ class FakeRegisterActionHandler { val instance = mockk() fun givenResultFor(wizard: RegistrationWizard, action: RegisterAction, result: RegistrationResult) { - coEvery { instance.handleRegisterAction(wizard, action) } answers { - it.invocation.args.first() + coEvery { instance.handleRegisterAction(wizard, action) } answers { call -> + call.invocation.args.first() result } } fun givenResultsFor(wizard: RegistrationWizard, result: List>) { - coEvery { instance.handleRegisterAction(wizard, any()) } answers { - val actionArg = it.invocation.args[1] as RegisterAction + coEvery { instance.handleRegisterAction(wizard, any()) } answers { call -> + val actionArg = call.invocation.args[1] as RegisterAction result.first { it.first == actionArg }.second } } From ba76aac9659b6c2af9b5a91bc6d0bea3b6b982ae Mon Sep 17 00:00:00 2001 From: Adam Brown Date: Thu, 17 Mar 2022 16:54:51 +0000 Subject: [PATCH 16/20] removing unused fake helper methods --- .../im/vector/app/test/fakes/FakeRegisterActionHandler.kt | 7 ------- .../im/vector/app/test/fakes/FakeRegistrationWizard.kt | 8 -------- 2 files changed, 15 deletions(-) diff --git a/vector/src/test/java/im/vector/app/test/fakes/FakeRegisterActionHandler.kt b/vector/src/test/java/im/vector/app/test/fakes/FakeRegisterActionHandler.kt index 3122e29c3b4..8d595d91e9e 100644 --- a/vector/src/test/java/im/vector/app/test/fakes/FakeRegisterActionHandler.kt +++ b/vector/src/test/java/im/vector/app/test/fakes/FakeRegisterActionHandler.kt @@ -27,13 +27,6 @@ class FakeRegisterActionHandler { val instance = mockk() - fun givenResultFor(wizard: RegistrationWizard, action: RegisterAction, result: RegistrationResult) { - coEvery { instance.handleRegisterAction(wizard, action) } answers { call -> - call.invocation.args.first() - result - } - } - fun givenResultsFor(wizard: RegistrationWizard, result: List>) { coEvery { instance.handleRegisterAction(wizard, any()) } answers { call -> val actionArg = call.invocation.args[1] as RegisterAction diff --git a/vector/src/test/java/im/vector/app/test/fakes/FakeRegistrationWizard.kt b/vector/src/test/java/im/vector/app/test/fakes/FakeRegistrationWizard.kt index 2fc830e94a1..723ed4ec27d 100644 --- a/vector/src/test/java/im/vector/app/test/fakes/FakeRegistrationWizard.kt +++ b/vector/src/test/java/im/vector/app/test/fakes/FakeRegistrationWizard.kt @@ -24,15 +24,7 @@ import org.matrix.android.sdk.api.session.Session class FakeRegistrationWizard : RegistrationWizard by mockk() { - fun givenSuccessfulDummy(session: Session) { - givenSuccessFor(session) { dummy() } - } - fun givenSuccessFor(result: Session, expect: suspend RegistrationWizard.() -> RegistrationResult) { coEvery { expect(this@FakeRegistrationWizard) } returns RegistrationResult.Success(result) } - - fun givenSuccessfulAcceptTerms(session: Session) { - coEvery { acceptTerms() } returns RegistrationResult.Success(session) - } } From 192d1c4f2dd849495102da149ecad9e0a5bcb61f Mon Sep 17 00:00:00 2001 From: Adam Brown Date: Thu, 17 Mar 2022 17:01:16 +0000 Subject: [PATCH 17/20] converting open class to sealed interface for extra type safety --- .../app/features/onboarding/OnboardingAction.kt | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/vector/src/main/java/im/vector/app/features/onboarding/OnboardingAction.kt b/vector/src/main/java/im/vector/app/features/onboarding/OnboardingAction.kt index 8b17b318c15..7fa75d1544e 100644 --- a/vector/src/main/java/im/vector/app/features/onboarding/OnboardingAction.kt +++ b/vector/src/main/java/im/vector/app/features/onboarding/OnboardingAction.kt @@ -46,13 +46,13 @@ sealed interface OnboardingAction : VectorViewModelAction { data class PostRegisterAction(val registerAction: RegisterAction) : OnboardingAction // Reset actions - open class ResetAction : OnboardingAction + sealed interface ResetAction : OnboardingAction - object ResetHomeServerType : ResetAction() - object ResetHomeServerUrl : ResetAction() - object ResetSignMode : ResetAction() - object ResetLogin : ResetAction() - object ResetResetPassword : ResetAction() + object ResetHomeServerType : ResetAction + object ResetHomeServerUrl : ResetAction + object ResetSignMode : ResetAction + object ResetLogin : ResetAction + object ResetResetPassword : ResetAction // Homeserver history object ClearHomeServerHistory : OnboardingAction From abf62aff475e6003fc3796cd74338a43b23f33e5 Mon Sep 17 00:00:00 2001 From: Adam Brown Date: Thu, 17 Mar 2022 17:51:01 +0000 Subject: [PATCH 18/20] extracting named function out for cancelling the email validation job, giving more context to the currentjob=null --- .../vector/app/features/onboarding/OnboardingViewModel.kt | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/vector/src/main/java/im/vector/app/features/onboarding/OnboardingViewModel.kt b/vector/src/main/java/im/vector/app/features/onboarding/OnboardingViewModel.kt index 3f2b5adcbe6..6659058b4e6 100644 --- a/vector/src/main/java/im/vector/app/features/onboarding/OnboardingViewModel.kt +++ b/vector/src/main/java/im/vector/app/features/onboarding/OnboardingViewModel.kt @@ -165,7 +165,7 @@ class OnboardingViewModel @AssistedInject constructor( is OnboardingAction.ProfilePictureSelected -> handleProfilePictureSelected(action) OnboardingAction.SaveSelectedProfilePicture -> updateProfilePicture() is OnboardingAction.PostViewEvent -> _viewEvents.post(action.viewEvent) - OnboardingAction.StopEmailValidationCheck -> currentJob = null + OnboardingAction.StopEmailValidationCheck -> cancelWaitForEmailValidation() }.exhaustive } @@ -915,6 +915,10 @@ class OnboardingViewModel @AssistedInject constructor( private fun completePersonalization() { _viewEvents.post(OnboardingViewEvents.OnPersonalizationComplete) } + + private fun cancelWaitForEmailValidation() { + currentJob = null + } } private fun LoginMode.supportsSignModeScreen(): Boolean { From 7f943d37fdd1d5413c465680089737bbd04c0f13 Mon Sep 17 00:00:00 2001 From: Adam Brown Date: Fri, 18 Mar 2022 13:15:14 +0000 Subject: [PATCH 19/20] explicitly declaring the fake registrationb wizard as not relaxed and creating new test instances for each case --- .../features/onboarding/RegistrationActionHandlerTest.kt | 7 ++++--- .../im/vector/app/test/fakes/FakeRegistrationWizard.kt | 2 +- 2 files changed, 5 insertions(+), 4 deletions(-) diff --git a/vector/src/test/java/im/vector/app/features/onboarding/RegistrationActionHandlerTest.kt b/vector/src/test/java/im/vector/app/features/onboarding/RegistrationActionHandlerTest.kt index 87efa0bddc3..8e61c2946f3 100644 --- a/vector/src/test/java/im/vector/app/features/onboarding/RegistrationActionHandlerTest.kt +++ b/vector/src/test/java/im/vector/app/features/onboarding/RegistrationActionHandlerTest.kt @@ -37,9 +37,6 @@ private val A_PID_TO_REGISTER = RegisterThreePid.Email("an email") class RegistrationActionHandlerTest { - private val fakeRegistrationWizard = FakeRegistrationWizard() - private val registrationActionHandler = RegistrationActionHandler() - @Test fun `when handling register action then delegates to wizard`() = runBlockingTest { val cases = listOf( @@ -60,6 +57,8 @@ class RegistrationActionHandlerTest { } private suspend fun testSuccessfulActionDelegation(case: Case) { + val registrationActionHandler = RegistrationActionHandler() + val fakeRegistrationWizard = FakeRegistrationWizard() fakeRegistrationWizard.givenSuccessFor(result = A_SESSION, case.expect) val result = registrationActionHandler.handleRegisterAction(fakeRegistrationWizard, case.action) @@ -71,3 +70,5 @@ class RegistrationActionHandlerTest { private fun case(action: RegisterAction, expect: suspend RegistrationWizard.() -> RegistrationResult) = Case(action, expect) private class Case(val action: RegisterAction, val expect: suspend RegistrationWizard.() -> RegistrationResult) + + diff --git a/vector/src/test/java/im/vector/app/test/fakes/FakeRegistrationWizard.kt b/vector/src/test/java/im/vector/app/test/fakes/FakeRegistrationWizard.kt index 723ed4ec27d..4e6e511abb9 100644 --- a/vector/src/test/java/im/vector/app/test/fakes/FakeRegistrationWizard.kt +++ b/vector/src/test/java/im/vector/app/test/fakes/FakeRegistrationWizard.kt @@ -22,7 +22,7 @@ import org.matrix.android.sdk.api.auth.registration.RegistrationResult import org.matrix.android.sdk.api.auth.registration.RegistrationWizard import org.matrix.android.sdk.api.session.Session -class FakeRegistrationWizard : RegistrationWizard by mockk() { +class FakeRegistrationWizard : RegistrationWizard by mockk(relaxed = false) { fun givenSuccessFor(result: Session, expect: suspend RegistrationWizard.() -> RegistrationResult) { coEvery { expect(this@FakeRegistrationWizard) } returns RegistrationResult.Success(result) From ce2c309d72a357430ebc3f8685a3bb556622fd86 Mon Sep 17 00:00:00 2001 From: Adam Brown Date: Fri, 18 Mar 2022 14:00:56 +0000 Subject: [PATCH 20/20] including verification to ensure no other methods are being called --- .../app/features/onboarding/RegistrationActionHandlerTest.kt | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/vector/src/test/java/im/vector/app/features/onboarding/RegistrationActionHandlerTest.kt b/vector/src/test/java/im/vector/app/features/onboarding/RegistrationActionHandlerTest.kt index 8e61c2946f3..2ca9aaef071 100644 --- a/vector/src/test/java/im/vector/app/features/onboarding/RegistrationActionHandlerTest.kt +++ b/vector/src/test/java/im/vector/app/features/onboarding/RegistrationActionHandlerTest.kt @@ -18,6 +18,7 @@ package im.vector.app.features.onboarding import im.vector.app.test.fakes.FakeRegistrationWizard import im.vector.app.test.fakes.FakeSession +import io.mockk.coVerifyAll import kotlinx.coroutines.test.runBlockingTest import org.amshove.kluent.shouldBeEqualTo import org.junit.Test @@ -63,6 +64,7 @@ class RegistrationActionHandlerTest { val result = registrationActionHandler.handleRegisterAction(fakeRegistrationWizard, case.action) + coVerifyAll { case.expect(fakeRegistrationWizard) } result shouldBeEqualTo AN_EXPECTED_RESULT } } @@ -70,5 +72,3 @@ class RegistrationActionHandlerTest { private fun case(action: RegisterAction, expect: suspend RegistrationWizard.() -> RegistrationResult) = Case(action, expect) private class Case(val action: RegisterAction, val expect: suspend RegistrationWizard.() -> RegistrationResult) - -