From b68fbed9b2e8f94683ad17fcf14cf3fe0990e74e Mon Sep 17 00:00:00 2001 From: Boris Safonov Date: Thu, 7 Dec 2023 16:45:47 +0200 Subject: [PATCH 1/7] feat: Use enroll MLS UseCase --- .../com/wire/android/ui/WireActivity.kt | 20 ++- .../com/wire/android/ui/home/E2EIDialogs.kt | 126 ++++++++++++++---- .../wire/android/ui/home/FeatureFlagState.kt | 7 + .../sync/FeatureFlagNotificationViewModel.kt | 42 +++++- .../settings/devices/DeviceDetailsScreen.kt | 52 +++++--- .../devices/DeviceDetailsViewModel.kt | 47 +++++-- .../EndToEndIdentityCertificateItem.kt | 74 ++++++---- .../devices/model/DeviceDetailsState.kt | 5 +- 8 files changed, 291 insertions(+), 82 deletions(-) diff --git a/app/src/main/kotlin/com/wire/android/ui/WireActivity.kt b/app/src/main/kotlin/com/wire/android/ui/WireActivity.kt index 39139815318..1b406d7153e 100644 --- a/app/src/main/kotlin/com/wire/android/ui/WireActivity.kt +++ b/app/src/main/kotlin/com/wire/android/ui/WireActivity.kt @@ -45,6 +45,7 @@ import androidx.compose.runtime.setValue import androidx.compose.runtime.staticCompositionLocalOf import androidx.compose.ui.ExperimentalComposeUiApi import androidx.compose.ui.Modifier +import androidx.compose.ui.platform.LocalContext import androidx.compose.ui.platform.LocalSoftwareKeyboardController import androidx.core.splashscreen.SplashScreen.Companion.installSplashScreen import androidx.core.view.WindowCompat @@ -70,6 +71,7 @@ import com.wire.android.ui.common.snackbar.LocalSnackbarHostState import com.wire.android.ui.common.topappbar.CommonTopAppBar import com.wire.android.ui.common.topappbar.CommonTopAppBarViewModel import com.wire.android.ui.destinations.ConversationScreenDestination +import com.wire.android.ui.destinations.E2eiCertificateDetailsScreenDestination import com.wire.android.ui.destinations.HomeScreenDestination import com.wire.android.ui.destinations.ImportMediaScreenDestination import com.wire.android.ui.destinations.IncomingCallScreenDestination @@ -81,6 +83,7 @@ import com.wire.android.ui.destinations.SelfDevicesScreenDestination import com.wire.android.ui.destinations.SelfUserProfileScreenDestination import com.wire.android.ui.destinations.WelcomeScreenDestination import com.wire.android.ui.home.E2EIRequiredDialog +import com.wire.android.ui.home.E2EIResultDialog import com.wire.android.ui.home.E2EISnoozeDialog import com.wire.android.ui.home.appLock.LockCodeTimeManager import com.wire.android.ui.home.sync.FeatureFlagNotificationViewModel @@ -269,6 +272,7 @@ class WireActivity : AppCompatActivity() { LaunchedEffect(userId) { featureFlagNotificationViewModel.loadInitialSync() } + val context = LocalContext.current with(featureFlagNotificationViewModel.featureFlagState) { if (shouldShowTeamAppLockDialog) { TeamAppLockFeatureFlagDialog( @@ -330,8 +334,9 @@ class WireActivity : AppCompatActivity() { e2EIRequired?.let { E2EIRequiredDialog( - result = e2EIRequired, - getCertificate = featureFlagNotificationViewModel::getE2EICertificate, + e2EIRequired = e2EIRequired, + isE2EILoading = isE2EILoading, + getCertificate = { featureFlagNotificationViewModel.getE2EICertificate(it, context) }, snoozeDialog = featureFlagNotificationViewModel::snoozeE2EIdRequiredDialog ) } @@ -343,6 +348,17 @@ class WireActivity : AppCompatActivity() { ) } + e2EIResult?.let { + E2EIResultDialog( + result = e2EIResult, + updateCertificate = { featureFlagNotificationViewModel.getE2EICertificate(it, context) }, + snoozeDialog = featureFlagNotificationViewModel::snoozeE2EIdRequiredDialog, + openCertificateDetails = { navigate(NavigationCommand(E2eiCertificateDetailsScreenDestination(it))) }, + dismissSuccessDialog = featureFlagNotificationViewModel::dismissSuccessE2EIdDialog, + isE2EILoading = isE2EILoading + ) + } + UpdateAppDialog(viewModel.globalAppState.updateAppDialog, ::updateTheApp) JoinConversationDialog( viewModel.globalAppState.conversationJoinedDialog, diff --git a/app/src/main/kotlin/com/wire/android/ui/home/E2EIDialogs.kt b/app/src/main/kotlin/com/wire/android/ui/home/E2EIDialogs.kt index fb247a4008c..2a28cd90e14 100644 --- a/app/src/main/kotlin/com/wire/android/ui/home/E2EIDialogs.kt +++ b/app/src/main/kotlin/com/wire/android/ui/home/E2EIDialogs.kt @@ -50,36 +50,78 @@ import kotlin.time.Duration.Companion.seconds @Composable fun E2EIRequiredDialog( - result: FeatureFlagState.E2EIRequired, - getCertificate: () -> Unit, + e2EIRequired: FeatureFlagState.E2EIRequired, + isE2EILoading: Boolean, + getCertificate: (FeatureFlagState.E2EIRequired) -> Unit, snoozeDialog: (FeatureFlagState.E2EIRequired.WithGracePeriod) -> Unit, ) { - when (result) { - FeatureFlagState.E2EIRequired.NoGracePeriod.Create -> E2EIRequiredNoSnoozeDialog(getCertificate = getCertificate) - FeatureFlagState.E2EIRequired.NoGracePeriod.Renew -> E2EIRenewNoSnoozeDialog(updateCertificate = getCertificate) + when (e2EIRequired) { + FeatureFlagState.E2EIRequired.NoGracePeriod.Create -> E2EIRequiredNoSnoozeDialog( + isLoading = isE2EILoading, + getCertificate = { getCertificate(e2EIRequired) } + ) + + FeatureFlagState.E2EIRequired.NoGracePeriod.Renew -> E2EIRenewNoSnoozeDialog( + isLoading = isE2EILoading, + updateCertificate = { getCertificate(e2EIRequired) } + ) + is FeatureFlagState.E2EIRequired.WithGracePeriod.Create -> E2EIRequiredWithSnoozeDialog( - getCertificate = getCertificate, - snoozeDialog = { snoozeDialog(result) } + isLoading = isE2EILoading, + getCertificate = { getCertificate(e2EIRequired) }, + snoozeDialog = { snoozeDialog(e2EIRequired) } ) is FeatureFlagState.E2EIRequired.WithGracePeriod.Renew -> E2EIRenewWithSnoozeDialog( - updateCertificate = getCertificate, - snoozeDialog = { snoozeDialog(result) } + isLoading = isE2EILoading, + updateCertificate = { getCertificate(e2EIRequired) }, + snoozeDialog = { snoozeDialog(e2EIRequired) } ) } } +@Composable +fun E2EIResultDialog( + result: FeatureFlagState.E2EIResult, + isE2EILoading: Boolean, + updateCertificate: (FeatureFlagState.E2EIRequired) -> Unit, + snoozeDialog: (FeatureFlagState.E2EIRequired.WithGracePeriod) -> Unit, + openCertificateDetails: (String) -> Unit, + dismissSuccessDialog: () -> Unit +) { + when (result) { + is FeatureFlagState.E2EIResult.Failure -> E2EIRenewErrorDialog( + e2EIRequired = result.e2EIRequired, + isE2EILoading = isE2EILoading, + updateCertificate = { updateCertificate(result.e2EIRequired) }, + snoozeDialog = snoozeDialog + ) + + is FeatureFlagState.E2EIResult.Success -> E2EISuccessDialog( + openCertificateDetails = { openCertificateDetails(result.certificate) }, + dismissDialog = dismissSuccessDialog + ) + } + +} + @Composable fun E2EIRenewErrorDialog( - result: FeatureFlagState.E2EIRequired, + e2EIRequired: FeatureFlagState.E2EIRequired, + isE2EILoading: Boolean, updateCertificate: () -> Unit, snoozeDialog: (FeatureFlagState.E2EIRequired.WithGracePeriod) -> Unit, ) { - when (result) { - is FeatureFlagState.E2EIRequired.NoGracePeriod -> E2EIErrorNoSnoozeDialog(updateCertificate = updateCertificate) + when (e2EIRequired) { + is FeatureFlagState.E2EIRequired.NoGracePeriod -> E2EIErrorNoSnoozeDialog( + isE2EILoading = isE2EILoading, + updateCertificate = updateCertificate + ) + is FeatureFlagState.E2EIRequired.WithGracePeriod -> E2EIErrorWithSnoozeDialog( updateCertificate = updateCertificate, - snoozeDialog = { snoozeDialog(result) } + isE2EILoading = isE2EILoading, + snoozeDialog = { snoozeDialog(e2EIRequired) } ) } } @@ -151,19 +193,47 @@ fun E2EISuccessDialog( ) } +@Composable +fun E2EIErrorWithDismissDialog( + isE2EILoading: Boolean, + updateCertificate: () -> Unit, + onDismiss: () -> Unit +) { + WireDialog( + title = stringResource(id = R.string.end_to_end_identity_renew_error_dialog_title), + text = stringResource(id = R.string.end_to_end_identity_renew_error_dialog_text), + onDismiss = onDismiss, + optionButton1Properties = WireDialogButtonProperties( + onClick = updateCertificate, + text = stringResource(id = R.string.label_retry), + type = WireDialogButtonType.Primary, + loading = isE2EILoading + ), + optionButton2Properties = WireDialogButtonProperties( + onClick = onDismiss, + text = stringResource(id = R.string.label_cancel), + type = WireDialogButtonType.Secondary, + ), + buttonsHorizontalAlignment = false, + properties = DialogProperties(usePlatformDefaultWidth = false) + ) +} + @Composable private fun E2EIErrorWithSnoozeDialog( + isE2EILoading: Boolean, updateCertificate: () -> Unit, snoozeDialog: () -> Unit ) { WireDialog( title = stringResource(id = R.string.end_to_end_identity_renew_error_dialog_title), text = stringResource(id = R.string.end_to_end_identity_renew_error_dialog_text), - onDismiss = updateCertificate, + onDismiss = snoozeDialog, optionButton1Properties = WireDialogButtonProperties( onClick = updateCertificate, text = stringResource(id = R.string.label_retry), type = WireDialogButtonType.Primary, + loading = isE2EILoading ), optionButton2Properties = WireDialogButtonProperties( onClick = snoozeDialog, @@ -177,6 +247,7 @@ private fun E2EIErrorWithSnoozeDialog( @Composable private fun E2EIErrorNoSnoozeDialog( + isE2EILoading: Boolean, updateCertificate: () -> Unit ) { WireDialog( @@ -187,6 +258,7 @@ private fun E2EIErrorNoSnoozeDialog( onClick = updateCertificate, text = stringResource(id = R.string.label_retry), type = WireDialogButtonType.Primary, + loading = isE2EILoading ), properties = DialogProperties( usePlatformDefaultWidth = false, @@ -198,6 +270,7 @@ private fun E2EIErrorNoSnoozeDialog( @Composable private fun E2EIRequiredWithSnoozeDialog( + isLoading: Boolean, getCertificate: () -> Unit, snoozeDialog: () -> Unit ) { @@ -209,6 +282,7 @@ private fun E2EIRequiredWithSnoozeDialog( onClick = getCertificate, text = stringResource(id = R.string.end_to_end_identity_required_dialog_positive_button), type = WireDialogButtonType.Primary, + loading = isLoading ), optionButton2Properties = WireDialogButtonProperties( onClick = snoozeDialog, @@ -221,7 +295,7 @@ private fun E2EIRequiredWithSnoozeDialog( } @Composable -private fun E2EIRequiredNoSnoozeDialog(getCertificate: () -> Unit) { +private fun E2EIRequiredNoSnoozeDialog(isLoading: Boolean, getCertificate: () -> Unit) { WireDialog( title = stringResource(id = R.string.end_to_end_identity_required_dialog_title), text = stringResource(id = R.string.end_to_end_identity_required_dialog_text_no_snooze), @@ -230,6 +304,7 @@ private fun E2EIRequiredNoSnoozeDialog(getCertificate: () -> Unit) { onClick = getCertificate, text = stringResource(id = R.string.end_to_end_identity_required_dialog_positive_button), type = WireDialogButtonType.Primary, + loading = isLoading ), buttonsHorizontalAlignment = false, properties = DialogProperties( @@ -242,6 +317,7 @@ private fun E2EIRequiredNoSnoozeDialog(getCertificate: () -> Unit) { @Composable private fun E2EIRenewWithSnoozeDialog( + isLoading: Boolean, updateCertificate: () -> Unit, snoozeDialog: () -> Unit ) { @@ -253,6 +329,7 @@ private fun E2EIRenewWithSnoozeDialog( onClick = updateCertificate, text = stringResource(id = R.string.end_to_end_identity_renew_dialog_positive_button), type = WireDialogButtonType.Primary, + loading = isLoading ), optionButton2Properties = WireDialogButtonProperties( onClick = snoozeDialog, @@ -265,7 +342,7 @@ private fun E2EIRenewWithSnoozeDialog( } @Composable -private fun E2EIRenewNoSnoozeDialog(updateCertificate: () -> Unit) { +private fun E2EIRenewNoSnoozeDialog(isLoading: Boolean, updateCertificate: () -> Unit) { WireDialog( title = stringResource(id = R.string.end_to_end_identity_renew_dialog_title), text = stringResource(id = R.string.end_to_end_identity_renew_dialog_text_no_snooze), @@ -274,6 +351,7 @@ private fun E2EIRenewNoSnoozeDialog(updateCertificate: () -> Unit) { onClick = updateCertificate, text = stringResource(id = R.string.end_to_end_identity_renew_dialog_positive_button), type = WireDialogButtonType.Primary, + loading = isLoading ), buttonsHorizontalAlignment = false, properties = DialogProperties( @@ -288,7 +366,7 @@ private fun E2EIRenewNoSnoozeDialog(updateCertificate: () -> Unit) { @Composable fun previewE2EIdRequiredWithSnoozeDialog() { WireTheme { - E2EIRequiredWithSnoozeDialog({}) {} + E2EIRequiredWithSnoozeDialog(false, {}) {} } } @@ -296,7 +374,7 @@ fun previewE2EIdRequiredWithSnoozeDialog() { @Composable fun previewE2EIdRequiredNoSnoozeDialog() { WireTheme { - E2EIRequiredNoSnoozeDialog {} + E2EIRequiredNoSnoozeDialog(false) {} } } @@ -304,7 +382,7 @@ fun previewE2EIdRequiredNoSnoozeDialog() { @Composable fun previewE2EIdRenewRequiredWithSnoozeDialog() { WireTheme { - E2EIRenewWithSnoozeDialog({}) {} + E2EIRenewWithSnoozeDialog(false, {}) {} } } @@ -312,7 +390,7 @@ fun previewE2EIdRenewRequiredWithSnoozeDialog() { @Composable fun previewE2EIdRenewRequiredNoSnoozeDialog() { WireTheme { - E2EIRenewNoSnoozeDialog {} + E2EIRenewNoSnoozeDialog(false) {} } } @@ -328,7 +406,7 @@ fun previewE2EIdSnoozeDialog() { @Composable fun previewE2EIRenewErrorDialogNoGracePeriod() { WireTheme { - E2EIRenewErrorDialog(FeatureFlagState.E2EIRequired.NoGracePeriod.Renew, { }) {} + E2EIRenewErrorDialog(FeatureFlagState.E2EIRequired.NoGracePeriod.Renew, false, { }) {} } } @@ -336,7 +414,7 @@ fun previewE2EIRenewErrorDialogNoGracePeriod() { @Composable fun previewE2EIRenewErrorDialogWithGracePeriod() { WireTheme { - E2EIRenewErrorDialog(FeatureFlagState.E2EIRequired.WithGracePeriod.Renew(2.days), { }) {} + E2EIRenewErrorDialog(FeatureFlagState.E2EIRequired.WithGracePeriod.Renew(2.days), false, { }) {} } } @@ -352,7 +430,7 @@ fun previewE2EISuccessDialog() { @Composable fun previewE2EIRenewErrorNoSnoozeDialog() { WireTheme { - E2EIErrorNoSnoozeDialog { } + E2EIErrorNoSnoozeDialog(false) { } } } @@ -360,6 +438,6 @@ fun previewE2EIRenewErrorNoSnoozeDialog() { @Composable fun previewE2EIRenewErrorWithSnoozeDialog() { WireTheme { - E2EIErrorWithSnoozeDialog(updateCertificate = {}) { } + E2EIErrorWithSnoozeDialog(isE2EILoading = false, updateCertificate = {}) { } } } diff --git a/app/src/main/kotlin/com/wire/android/ui/home/FeatureFlagState.kt b/app/src/main/kotlin/com/wire/android/ui/home/FeatureFlagState.kt index 9f877f60175..f35bc10d238 100644 --- a/app/src/main/kotlin/com/wire/android/ui/home/FeatureFlagState.kt +++ b/app/src/main/kotlin/com/wire/android/ui/home/FeatureFlagState.kt @@ -36,6 +36,8 @@ data class FeatureFlagState( val areSelfDeletedMessagesEnabled: Boolean = true, val e2EIRequired: E2EIRequired? = null, val e2EISnoozeInfo: E2EISnooze? = null, + val e2EIResult: E2EIResult? = null, + val isE2EILoading: Boolean = false, val showCallEndedBecauseOfConversationDegraded: Boolean = false ) { enum class SharingRestrictedState { @@ -56,4 +58,9 @@ data class FeatureFlagState( data object Renew : NoGracePeriod() } } + + sealed class E2EIResult { + data class Failure(val e2EIRequired: E2EIRequired) : E2EIResult() + data class Success(val certificate: String): E2EIResult() + } } diff --git a/app/src/main/kotlin/com/wire/android/ui/home/sync/FeatureFlagNotificationViewModel.kt b/app/src/main/kotlin/com/wire/android/ui/home/sync/FeatureFlagNotificationViewModel.kt index 77a3f4a6786..8eca260b397 100644 --- a/app/src/main/kotlin/com/wire/android/ui/home/sync/FeatureFlagNotificationViewModel.kt +++ b/app/src/main/kotlin/com/wire/android/ui/home/sync/FeatureFlagNotificationViewModel.kt @@ -20,6 +20,7 @@ package com.wire.android.ui.home.sync +import android.content.Context import androidx.compose.runtime.getValue import androidx.compose.runtime.mutableStateOf import androidx.compose.runtime.setValue @@ -30,17 +31,21 @@ import com.wire.android.datastore.GlobalDataStore import com.wire.android.di.KaliumCoreLogic import com.wire.android.feature.AppLockSource import com.wire.android.feature.DisableAppLockUseCase +import com.wire.android.feature.e2ei.GetE2EICertificateUseCase import com.wire.android.ui.home.FeatureFlagState import com.wire.android.ui.home.conversations.selfdeletion.SelfDeletionMapper.toSelfDeletionDuration import com.wire.android.ui.home.messagecomposer.SelfDeletionDuration +import com.wire.android.util.dispatchers.DispatcherProvider import com.wire.kalium.logic.CoreLogic import com.wire.kalium.logic.configuration.FileSharingStatus import com.wire.kalium.logic.data.message.TeamSelfDeleteTimer import com.wire.kalium.logic.data.sync.SyncState import com.wire.kalium.logic.data.user.UserId +import com.wire.kalium.logic.feature.e2ei.usecase.E2EIEnrollmentResult import com.wire.kalium.logic.feature.session.CurrentSessionResult import com.wire.kalium.logic.feature.session.CurrentSessionUseCase import com.wire.kalium.logic.feature.user.E2EIRequiredResult +import com.wire.kalium.logic.functional.fold import dagger.hilt.android.lifecycle.HiltViewModel import kotlinx.coroutines.flow.collectLatest import kotlinx.coroutines.flow.distinctUntilChanged @@ -54,7 +59,8 @@ class FeatureFlagNotificationViewModel @Inject constructor( @KaliumCoreLogic private val coreLogic: CoreLogic, private val currentSessionUseCase: CurrentSessionUseCase, private val globalDataStore: GlobalDataStore, - private val disableAppLockUseCase: DisableAppLockUseCase + private val disableAppLockUseCase: DisableAppLockUseCase, + private val dispatcherProvider: DispatcherProvider ) : ViewModel() { var featureFlagState by mutableStateOf(FeatureFlagState()) @@ -254,9 +260,32 @@ class FeatureFlagNotificationViewModel @Inject constructor( fun isUserAppLockSet() = globalDataStore.isAppLockPasscodeSet() - fun getE2EICertificate() { - // TODO do the magic - featureFlagState = featureFlagState.copy(e2EIRequired = null) + fun getE2EICertificate(e2eiRequired: FeatureFlagState.E2EIRequired, context: Context) { + featureFlagState = featureFlagState.copy(isE2EILoading = true) + currentUserId?.let { userId -> + GetE2EICertificateUseCase(coreLogic.getSessionScope(userId).enrollE2EI, dispatcherProvider).invoke(context) { result -> + result.fold({ + featureFlagState = featureFlagState.copy( + isE2EILoading = false, + e2EIRequired = null, + e2EIResult = FeatureFlagState.E2EIResult.Failure(e2eiRequired) + ) + }, { + if (it is E2EIEnrollmentResult.Finalized) + featureFlagState = featureFlagState.copy( + isE2EILoading = false, + e2EIRequired = null, + e2EIResult = FeatureFlagState.E2EIResult.Success(it.certificate) + ) + else if (it is E2EIEnrollmentResult.Failed) + featureFlagState = featureFlagState.copy( + isE2EILoading = false, + e2EIRequired = null, + e2EIResult = FeatureFlagState.E2EIResult.Failure(e2eiRequired) + ) + }) + } + } } fun snoozeE2EIdRequiredDialog(result: FeatureFlagState.E2EIRequired.WithGracePeriod) { @@ -278,4 +307,9 @@ class FeatureFlagNotificationViewModel @Inject constructor( fun dismissCallEndedBecauseOfConversationDegraded() { featureFlagState = featureFlagState.copy(showCallEndedBecauseOfConversationDegraded = false) } + + + fun dismissSuccessE2EIdDialog() { + featureFlagState = featureFlagState.copy(e2EIResult = null) + } } diff --git a/app/src/main/kotlin/com/wire/android/ui/settings/devices/DeviceDetailsScreen.kt b/app/src/main/kotlin/com/wire/android/ui/settings/devices/DeviceDetailsScreen.kt index ba9d764ac82..54d69ab6691 100644 --- a/app/src/main/kotlin/com/wire/android/ui/settings/devices/DeviceDetailsScreen.kt +++ b/app/src/main/kotlin/com/wire/android/ui/settings/devices/DeviceDetailsScreen.kt @@ -1,5 +1,6 @@ package com.wire.android.ui.settings.devices +import android.content.Context import androidx.annotation.StringRes import androidx.compose.foundation.background import androidx.compose.foundation.layout.Box @@ -56,6 +57,8 @@ import com.wire.android.ui.common.scaffold.WireScaffold import com.wire.android.ui.common.topappbar.WireCenterAlignedTopAppBar import com.wire.android.ui.common.topappbar.WireTopAppBarTitle import com.wire.android.ui.destinations.E2eiCertificateDetailsScreenDestination +import com.wire.android.ui.home.E2EIErrorWithDismissDialog +import com.wire.android.ui.home.E2EISuccessDialog import com.wire.android.ui.home.conversationslist.common.FolderHeader import com.wire.android.ui.settings.devices.model.DeviceDetailsState import com.wire.android.ui.theme.wireColorScheme @@ -94,7 +97,9 @@ fun DeviceDetailsScreen( navigator.navigate( NavigationCommand(E2eiCertificateDetailsScreenDestination(it)) ) - } + }, + onEnrollE2EIErrorDismiss = viewModel::hideEnrollE2EICertificateError, + onEnrollE2EISuccessDismiss = viewModel::hideEnrollE2EICertificateSuccess ) } } @@ -109,8 +114,10 @@ fun DeviceDetailsContent( onRemoveConfirm: () -> Unit = {}, onDialogDismiss: () -> Unit = {}, onErrorDialogDismiss: () -> Unit = {}, - enrollE2eiCertificate: () -> Unit = {}, - onUpdateClientVerification: (Boolean) -> Unit = {} + enrollE2eiCertificate: (Context) -> Unit = {}, + onUpdateClientVerification: (Boolean) -> Unit = {}, + onEnrollE2EIErrorDismiss: () -> Unit = {}, + onEnrollE2EISuccessDismiss: () -> Unit = {} ) { val screenState = rememberConversationScreenState() WireScaffold( @@ -148,6 +155,7 @@ fun DeviceDetailsContent( } } ) { internalPadding -> + val context = LocalContext.current LazyColumn( modifier = Modifier .fillMaxSize() @@ -161,18 +169,17 @@ fun DeviceDetailsContent( Divider(color = MaterialTheme.wireColorScheme.background) } } - if (BuildConfig.DEBUG) { - item { - EndToEndIdentityCertificateItem( - isE2eiCertificateActivated = state.isE2eiCertificateActivated, - certificate = state.e2eiCertificate, - isSelfClient = state.isSelfClient, - enrollE2eiCertificate = enrollE2eiCertificate, - updateE2eiCertificate = {}, - showCertificate = onNavigateToE2eiCertificateDetailsScreen - ) - Divider(color = colorsScheme().background) - } + item { + EndToEndIdentityCertificateItem( + isE2eiCertificateActivated = state.isE2eiCertificateActivated, + certificate = state.e2eiCertificate, + isSelfClient = state.isSelfClient, + isLoadingCertificate = state.isLoadingCertificate, + enrollE2eiCertificate = { enrollE2eiCertificate(context) }, + updateE2eiCertificate = {}, + showCertificate = onNavigateToE2eiCertificateDetailsScreen + ) + Divider(color = colorsScheme().background) } item { FolderHeader( @@ -246,6 +253,21 @@ fun DeviceDetailsContent( ) } } + + if (state.isE2EICertificateEnrollError) { + E2EIErrorWithDismissDialog( + isE2EILoading = state.isLoadingCertificate, + updateCertificate = { enrollE2eiCertificate(context) }, + onDismiss = onEnrollE2EIErrorDismiss + ) + } + + if (state.isE2EICertificateEnrollSuccess) { + E2EISuccessDialog( + openCertificateDetails = { onNavigateToE2eiCertificateDetailsScreen(state.e2eiCertificate.certificateDetail) }, + dismissDialog = onEnrollE2EISuccessDismiss + ) + } } } diff --git a/app/src/main/kotlin/com/wire/android/ui/settings/devices/DeviceDetailsViewModel.kt b/app/src/main/kotlin/com/wire/android/ui/settings/devices/DeviceDetailsViewModel.kt index b5a81636c13..fa75f81dd24 100644 --- a/app/src/main/kotlin/com/wire/android/ui/settings/devices/DeviceDetailsViewModel.kt +++ b/app/src/main/kotlin/com/wire/android/ui/settings/devices/DeviceDetailsViewModel.kt @@ -1,14 +1,15 @@ package com.wire.android.ui.settings.devices +import android.content.Context import androidx.compose.runtime.getValue import androidx.compose.runtime.mutableStateOf import androidx.compose.runtime.setValue import androidx.compose.ui.text.input.TextFieldValue import androidx.lifecycle.SavedStateHandle import androidx.lifecycle.viewModelScope -import com.wire.android.BuildConfig import com.wire.android.appLogger import com.wire.android.di.CurrentAccount +import com.wire.android.feature.e2ei.GetE2EICertificateUseCase import com.wire.android.navigation.SavedStateViewModel import com.wire.android.ui.authentication.devices.model.Device import com.wire.android.ui.authentication.devices.remove.RemoveDeviceDialogState @@ -25,11 +26,13 @@ import com.wire.kalium.logic.feature.client.GetClientDetailsResult import com.wire.kalium.logic.feature.client.ObserveClientDetailsUseCase import com.wire.kalium.logic.feature.client.Result import com.wire.kalium.logic.feature.client.UpdateClientVerificationStatusUseCase +import com.wire.kalium.logic.feature.e2ei.usecase.E2EIEnrollmentResult import com.wire.kalium.logic.feature.e2ei.usecase.GetE2EICertificateUseCaseResult import com.wire.kalium.logic.feature.e2ei.usecase.GetE2eiCertificateUseCase import com.wire.kalium.logic.feature.user.GetUserInfoResult import com.wire.kalium.logic.feature.user.IsPasswordRequiredUseCase import com.wire.kalium.logic.feature.user.ObserveUserInfoUseCase +import com.wire.kalium.logic.functional.fold import dagger.hilt.android.lifecycle.HiltViewModel import kotlinx.coroutines.launch import javax.inject.Inject @@ -46,7 +49,8 @@ class DeviceDetailsViewModel @Inject constructor( private val fingerprintUseCase: ClientFingerprintUseCase, private val updateClientVerificationStatus: UpdateClientVerificationStatusUseCase, private val observeUserInfo: ObserveUserInfoUseCase, - private val e2eiCertificate: GetE2eiCertificateUseCase + private val e2eiCertificate: GetE2eiCertificateUseCase, + private val enrolE2EICertificateUseCase: GetE2EICertificateUseCase ) : SavedStateViewModel(savedStateHandle) { private val deviceDetailsNavArgs: DeviceDetailsNavArgs = savedStateHandle.navArgs() @@ -60,9 +64,7 @@ class DeviceDetailsViewModel @Inject constructor( observeDeviceDetails() getClientFingerPrint() observeUserName() - if (BuildConfig.DEBUG) { - getE2eiCertificate() - } + getE2eiCertificate() } private val isSelfClient: Boolean @@ -91,16 +93,35 @@ class DeviceDetailsViewModel @Inject constructor( state = if (certificate is GetE2EICertificateUseCaseResult.Success) { state.copy( isE2eiCertificateActivated = true, - e2eiCertificate = certificate.certificate + e2eiCertificate = certificate.certificate, + isLoadingCertificate = false ) } else { - state.copy(isE2eiCertificateActivated = false) + state.copy(isE2eiCertificateActivated = false, isLoadingCertificate = false) } } } - fun enrollE2eiCertificate() { - // TODO invoke correspondent use case + fun enrollE2eiCertificate(context: Context) { + state = state.copy(isLoadingCertificate = true) + enrolE2EICertificateUseCase(context) { result -> + result.fold({ + state = state.copy( + isLoadingCertificate = false, + isE2EICertificateEnrollError = true + ) + }, { + if (it is E2EIEnrollmentResult.Finalized) { + getE2eiCertificate() + state = state.copy(isE2EICertificateEnrollSuccess = true) + } else if (it is E2EIEnrollmentResult.Failed) { + state = state.copy( + isLoadingCertificate = false, + isE2EICertificateEnrollError = true + ) + } + }) + } } private fun getClientFingerPrint() { @@ -223,4 +244,12 @@ class DeviceDetailsViewModel @Inject constructor( fun clearDeleteClientError() { state = state.copy(error = RemoveDeviceError.None) } + + fun hideEnrollE2EICertificateError() { + state = state.copy(isE2EICertificateEnrollError = false) + } + + fun hideEnrollE2EICertificateSuccess() { + state = state.copy(isE2EICertificateEnrollSuccess = false) + } } diff --git a/app/src/main/kotlin/com/wire/android/ui/settings/devices/EndToEndIdentityCertificateItem.kt b/app/src/main/kotlin/com/wire/android/ui/settings/devices/EndToEndIdentityCertificateItem.kt index cf9870c8df1..6d92df3afd8 100644 --- a/app/src/main/kotlin/com/wire/android/ui/settings/devices/EndToEndIdentityCertificateItem.kt +++ b/app/src/main/kotlin/com/wire/android/ui/settings/devices/EndToEndIdentityCertificateItem.kt @@ -46,6 +46,7 @@ fun EndToEndIdentityCertificateItem( isE2eiCertificateActivated: Boolean, certificate: E2eiCertificate, isSelfClient: Boolean, + isLoadingCertificate: Boolean, enrollE2eiCertificate: () -> Unit, updateE2eiCertificate: () -> Unit, showCertificate: (String) -> Unit @@ -83,13 +84,6 @@ fun EndToEndIdentityCertificateItem( icon = R.drawable.ic_certificate_revoked_mls ) SerialNumberBlock(certificate.serialNumber) - ShowE2eiCertificateButton( - enabled = true, - isLoading = false, - onShowCertificateClicked = { - showCertificate(certificate.certificateDetail) - } - ) } CertificateStatus.EXPIRED -> { @@ -99,18 +93,13 @@ fun EndToEndIdentityCertificateItem( icon = R.drawable.ic_certificate_not_activated_mls ) SerialNumberBlock(certificate.serialNumber) - UpdateE2eiCertificateButton( - enabled = true, - isLoading = false, - updateE2eiCertificate - ) - ShowE2eiCertificateButton( - enabled = true, - isLoading = false, - onShowCertificateClicked = { - showCertificate(certificate.certificateDetail) - } - ) + if (isSelfClient) { + UpdateE2eiCertificateButton( + enabled = true, + isLoading = isLoadingCertificate, + onUpdateCertificateClicked = enrollE2eiCertificate + ) + } } CertificateStatus.VALID -> { @@ -120,15 +109,22 @@ fun EndToEndIdentityCertificateItem( icon = R.drawable.ic_certificate_valid_mls ) SerialNumberBlock(certificate.serialNumber) - ShowE2eiCertificateButton( - enabled = true, - isLoading = false, - onShowCertificateClicked = { - showCertificate(certificate.certificateDetail) - } - ) + if (isSelfClient) { + UpdateE2eiCertificateButton( + enabled = true, + isLoading = isLoadingCertificate, + onUpdateCertificateClicked = enrollE2eiCertificate + ) + } } } + ShowE2eiCertificateButton( + enabled = true, + isLoading = false, + onShowCertificateClicked = { + showCertificate(certificate.certificateDetail) + } + ) } else { E2EIStatusRow( label = stringResource(id = R.string.e2ei_certificat_status_not_activated), @@ -136,7 +132,11 @@ fun EndToEndIdentityCertificateItem( icon = R.drawable.ic_certificate_not_activated_mls ) if (isSelfClient) { - GetE2eiCertificateButton(enabled = true, isLoading = false) { } + GetE2eiCertificateButton( + enabled = true, + isLoading = isLoadingCertificate, + onGetCertificateClicked = enrollE2eiCertificate + ) } } } @@ -200,6 +200,26 @@ fun PreviewEndToEndIdentityCertificateItem() { serialNumber = "e5:d5:e6:75:7e:04:86:07:14:3c:a0:ed:9a:8d:e4:fd", certificateDetail = "" ), + isLoadingCertificate = false, + enrollE2eiCertificate = {}, + updateE2eiCertificate = {}, + showCertificate = {} + ) +} + +@PreviewMultipleThemes +@Composable +fun PreviewEndToEndIdentityCertificateSelfItem() { + EndToEndIdentityCertificateItem( + isE2eiCertificateActivated = true, + isSelfClient = true, + certificate = E2eiCertificate( + issuer = "Wire", + status = CertificateStatus.VALID, + serialNumber = "e5:d5:e6:75:7e:04:86:07:14:3c:a0:ed:9a:8d:e4:fd", + certificateDetail = "" + ), + isLoadingCertificate = false, enrollE2eiCertificate = {}, updateE2eiCertificate = {}, showCertificate = {} diff --git a/app/src/main/kotlin/com/wire/android/ui/settings/devices/model/DeviceDetailsState.kt b/app/src/main/kotlin/com/wire/android/ui/settings/devices/model/DeviceDetailsState.kt index 24b01807567..c5625a74109 100644 --- a/app/src/main/kotlin/com/wire/android/ui/settings/devices/model/DeviceDetailsState.kt +++ b/app/src/main/kotlin/com/wire/android/ui/settings/devices/model/DeviceDetailsState.kt @@ -14,5 +14,8 @@ data class DeviceDetailsState( val isSelfClient: Boolean = false, val userName: String? = null, val isE2eiCertificateActivated: Boolean = false, - val e2eiCertificate: E2eiCertificate = E2eiCertificate() + val e2eiCertificate: E2eiCertificate = E2eiCertificate(), + val isLoadingCertificate: Boolean = false, + val isE2EICertificateEnrollSuccess: Boolean = false, + val isE2EICertificateEnrollError: Boolean = false ) From 977ad3147dd32f61ee5ca1cea08ab6816d78f9d8 Mon Sep 17 00:00:00 2001 From: Boris Safonov Date: Mon, 11 Dec 2023 15:19:47 +0200 Subject: [PATCH 2/7] feat: Use enroll MLS UseCase: ready for review --- .../feature/e2ei/GetE2EICertificateUseCase.kt | 5 +++-- .../ui/settings/devices/DeviceDetailsScreen.kt | 2 +- .../ui/settings/devices/DeviceDetailsViewModel.kt | 7 +++++++ .../devices/EndToEndIdentityCertificateItem.kt | 12 ++++++------ kalium | 2 +- 5 files changed, 18 insertions(+), 10 deletions(-) diff --git a/app/src/main/kotlin/com/wire/android/feature/e2ei/GetE2EICertificateUseCase.kt b/app/src/main/kotlin/com/wire/android/feature/e2ei/GetE2EICertificateUseCase.kt index 4354058cfc0..aaf07c43ef9 100644 --- a/app/src/main/kotlin/com/wire/android/feature/e2ei/GetE2EICertificateUseCase.kt +++ b/app/src/main/kotlin/com/wire/android/feature/e2ei/GetE2EICertificateUseCase.kt @@ -27,6 +27,7 @@ import com.wire.kalium.logic.feature.e2ei.usecase.EnrollE2EIUseCase import com.wire.kalium.logic.functional.Either import com.wire.kalium.logic.functional.fold import com.wire.kalium.logic.functional.map +import com.wire.kalium.logic.functional.mapLeft import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.SupervisorJob import kotlinx.coroutines.launch @@ -62,10 +63,10 @@ class GetE2EICertificateUseCase @Inject constructor( scope.launch { when (oAuthResult) { is OAuthUseCase.OAuthResult.Success -> { - enrollE2EI.finalizeEnrollment( + enrollmentResultHandler(enrollE2EI.finalizeEnrollment( oAuthResult.idToken, initialEnrollmentResult - ).map { enrollmentResultHandler(Either.Right(it)) } + )) } is OAuthUseCase.OAuthResult.Failed -> { diff --git a/app/src/main/kotlin/com/wire/android/ui/settings/devices/DeviceDetailsScreen.kt b/app/src/main/kotlin/com/wire/android/ui/settings/devices/DeviceDetailsScreen.kt index 54d69ab6691..9ebe8bab75a 100644 --- a/app/src/main/kotlin/com/wire/android/ui/settings/devices/DeviceDetailsScreen.kt +++ b/app/src/main/kotlin/com/wire/android/ui/settings/devices/DeviceDetailsScreen.kt @@ -173,7 +173,7 @@ fun DeviceDetailsContent( EndToEndIdentityCertificateItem( isE2eiCertificateActivated = state.isE2eiCertificateActivated, certificate = state.e2eiCertificate, - isSelfClient = state.isSelfClient, + isCurrentDevice = state.isCurrentDevice, isLoadingCertificate = state.isLoadingCertificate, enrollE2eiCertificate = { enrollE2eiCertificate(context) }, updateE2eiCertificate = {}, diff --git a/app/src/main/kotlin/com/wire/android/ui/settings/devices/DeviceDetailsViewModel.kt b/app/src/main/kotlin/com/wire/android/ui/settings/devices/DeviceDetailsViewModel.kt index fa75f81dd24..8f3a4e0af68 100644 --- a/app/src/main/kotlin/com/wire/android/ui/settings/devices/DeviceDetailsViewModel.kt +++ b/app/src/main/kotlin/com/wire/android/ui/settings/devices/DeviceDetailsViewModel.kt @@ -91,12 +91,14 @@ class DeviceDetailsViewModel @Inject constructor( viewModelScope.launch { val certificate = e2eiCertificate(deviceId) state = if (certificate is GetE2EICertificateUseCaseResult.Success) { + println("cyka getting cert success") state.copy( isE2eiCertificateActivated = true, e2eiCertificate = certificate.certificate, isLoadingCertificate = false ) } else { + println("cyka getting cert success") state.copy(isE2eiCertificateActivated = false, isLoadingCertificate = false) } } @@ -106,19 +108,24 @@ class DeviceDetailsViewModel @Inject constructor( state = state.copy(isLoadingCertificate = true) enrolE2EICertificateUseCase(context) { result -> result.fold({ + println("cyka error0 $it") state = state.copy( isLoadingCertificate = false, isE2EICertificateEnrollError = true ) }, { if (it is E2EIEnrollmentResult.Finalized) { + println("cyka Finalized, getting cert") getE2eiCertificate() state = state.copy(isE2EICertificateEnrollSuccess = true) } else if (it is E2EIEnrollmentResult.Failed) { + println("cyka error1 $it") state = state.copy( isLoadingCertificate = false, isE2EICertificateEnrollError = true ) + } else { + println("cyka other: $it") } }) } diff --git a/app/src/main/kotlin/com/wire/android/ui/settings/devices/EndToEndIdentityCertificateItem.kt b/app/src/main/kotlin/com/wire/android/ui/settings/devices/EndToEndIdentityCertificateItem.kt index 6d92df3afd8..cf562fb3508 100644 --- a/app/src/main/kotlin/com/wire/android/ui/settings/devices/EndToEndIdentityCertificateItem.kt +++ b/app/src/main/kotlin/com/wire/android/ui/settings/devices/EndToEndIdentityCertificateItem.kt @@ -45,7 +45,7 @@ import com.wire.kalium.logic.feature.e2ei.E2eiCertificate fun EndToEndIdentityCertificateItem( isE2eiCertificateActivated: Boolean, certificate: E2eiCertificate, - isSelfClient: Boolean, + isCurrentDevice: Boolean, isLoadingCertificate: Boolean, enrollE2eiCertificate: () -> Unit, updateE2eiCertificate: () -> Unit, @@ -93,7 +93,7 @@ fun EndToEndIdentityCertificateItem( icon = R.drawable.ic_certificate_not_activated_mls ) SerialNumberBlock(certificate.serialNumber) - if (isSelfClient) { + if (isCurrentDevice) { UpdateE2eiCertificateButton( enabled = true, isLoading = isLoadingCertificate, @@ -109,7 +109,7 @@ fun EndToEndIdentityCertificateItem( icon = R.drawable.ic_certificate_valid_mls ) SerialNumberBlock(certificate.serialNumber) - if (isSelfClient) { + if (isCurrentDevice) { UpdateE2eiCertificateButton( enabled = true, isLoading = isLoadingCertificate, @@ -131,7 +131,7 @@ fun EndToEndIdentityCertificateItem( labelColor = colorsScheme().error, icon = R.drawable.ic_certificate_not_activated_mls ) - if (isSelfClient) { + if (isCurrentDevice) { GetE2eiCertificateButton( enabled = true, isLoading = isLoadingCertificate, @@ -193,7 +193,7 @@ private fun E2EIStatusRow( fun PreviewEndToEndIdentityCertificateItem() { EndToEndIdentityCertificateItem( isE2eiCertificateActivated = true, - isSelfClient = false, + isCurrentDevice = false, certificate = E2eiCertificate( issuer = "Wire", status = CertificateStatus.VALID, @@ -212,7 +212,7 @@ fun PreviewEndToEndIdentityCertificateItem() { fun PreviewEndToEndIdentityCertificateSelfItem() { EndToEndIdentityCertificateItem( isE2eiCertificateActivated = true, - isSelfClient = true, + isCurrentDevice = true, certificate = E2eiCertificate( issuer = "Wire", status = CertificateStatus.VALID, diff --git a/kalium b/kalium index c2e46cdf3c5..454088c4918 160000 --- a/kalium +++ b/kalium @@ -1 +1 @@ -Subproject commit c2e46cdf3c55beb60b35596f961a729b7dede6c3 +Subproject commit 454088c4918db1d0c4f66cb2d937a0eabbe2c67a From 41eabf5b3c4b1adeaccf7ad49e1462a39fbfc38c Mon Sep 17 00:00:00 2001 From: Boris Safonov Date: Mon, 11 Dec 2023 17:42:58 +0200 Subject: [PATCH 3/7] Code style fixes --- .../wire/android/feature/e2ei/GetE2EICertificateUseCase.kt | 2 -- app/src/main/kotlin/com/wire/android/ui/home/E2EIDialogs.kt | 1 - .../kotlin/com/wire/android/ui/home/FeatureFlagState.kt | 2 +- .../ui/home/sync/FeatureFlagNotificationViewModel.kt | 6 +++--- 4 files changed, 4 insertions(+), 7 deletions(-) diff --git a/app/src/main/kotlin/com/wire/android/feature/e2ei/GetE2EICertificateUseCase.kt b/app/src/main/kotlin/com/wire/android/feature/e2ei/GetE2EICertificateUseCase.kt index aaf07c43ef9..0f57f04997a 100644 --- a/app/src/main/kotlin/com/wire/android/feature/e2ei/GetE2EICertificateUseCase.kt +++ b/app/src/main/kotlin/com/wire/android/feature/e2ei/GetE2EICertificateUseCase.kt @@ -26,8 +26,6 @@ import com.wire.kalium.logic.feature.e2ei.usecase.E2EIEnrollmentResult import com.wire.kalium.logic.feature.e2ei.usecase.EnrollE2EIUseCase import com.wire.kalium.logic.functional.Either import com.wire.kalium.logic.functional.fold -import com.wire.kalium.logic.functional.map -import com.wire.kalium.logic.functional.mapLeft import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.SupervisorJob import kotlinx.coroutines.launch diff --git a/app/src/main/kotlin/com/wire/android/ui/home/E2EIDialogs.kt b/app/src/main/kotlin/com/wire/android/ui/home/E2EIDialogs.kt index 2a28cd90e14..4317dfb3121 100644 --- a/app/src/main/kotlin/com/wire/android/ui/home/E2EIDialogs.kt +++ b/app/src/main/kotlin/com/wire/android/ui/home/E2EIDialogs.kt @@ -102,7 +102,6 @@ fun E2EIResultDialog( dismissDialog = dismissSuccessDialog ) } - } @Composable diff --git a/app/src/main/kotlin/com/wire/android/ui/home/FeatureFlagState.kt b/app/src/main/kotlin/com/wire/android/ui/home/FeatureFlagState.kt index f35bc10d238..4a9bd612df1 100644 --- a/app/src/main/kotlin/com/wire/android/ui/home/FeatureFlagState.kt +++ b/app/src/main/kotlin/com/wire/android/ui/home/FeatureFlagState.kt @@ -61,6 +61,6 @@ data class FeatureFlagState( sealed class E2EIResult { data class Failure(val e2EIRequired: E2EIRequired) : E2EIResult() - data class Success(val certificate: String): E2EIResult() + data class Success(val certificate: String) : E2EIResult() } } diff --git a/app/src/main/kotlin/com/wire/android/ui/home/sync/FeatureFlagNotificationViewModel.kt b/app/src/main/kotlin/com/wire/android/ui/home/sync/FeatureFlagNotificationViewModel.kt index db9629cb94e..453abab8376 100644 --- a/app/src/main/kotlin/com/wire/android/ui/home/sync/FeatureFlagNotificationViewModel.kt +++ b/app/src/main/kotlin/com/wire/android/ui/home/sync/FeatureFlagNotificationViewModel.kt @@ -277,18 +277,19 @@ class FeatureFlagNotificationViewModel @Inject constructor( e2EIResult = FeatureFlagState.E2EIResult.Failure(e2eiRequired) ) }, { - if (it is E2EIEnrollmentResult.Finalized) + if (it is E2EIEnrollmentResult.Finalized) { featureFlagState = featureFlagState.copy( isE2EILoading = false, e2EIRequired = null, e2EIResult = FeatureFlagState.E2EIResult.Success(it.certificate) ) - else if (it is E2EIEnrollmentResult.Failed) + } else if (it is E2EIEnrollmentResult.Failed) { featureFlagState = featureFlagState.copy( isE2EILoading = false, e2EIRequired = null, e2EIResult = FeatureFlagState.E2EIResult.Failure(e2eiRequired) ) + } }) } } @@ -314,7 +315,6 @@ class FeatureFlagNotificationViewModel @Inject constructor( featureFlagState = featureFlagState.copy(showCallEndedBecauseOfConversationDegraded = false) } - fun dismissSuccessE2EIdDialog() { featureFlagState = featureFlagState.copy(e2EIResult = null) } From 527bcdff7cb7600c4f89d2cc3f4abe9f9d79ead6 Mon Sep 17 00:00:00 2001 From: Boris Safonov Date: Tue, 12 Dec 2023 11:41:47 +0200 Subject: [PATCH 4/7] Updated kalium --- .../devices/DeviceDetailsViewModel.kt | 7 ----- .../FeatureFlagNotificationViewModelTest.kt | 4 ++- .../devices/DeviceDetailsViewModelTest.kt | 31 ++++++++++++++++++- kalium | 2 +- 4 files changed, 34 insertions(+), 10 deletions(-) diff --git a/app/src/main/kotlin/com/wire/android/ui/settings/devices/DeviceDetailsViewModel.kt b/app/src/main/kotlin/com/wire/android/ui/settings/devices/DeviceDetailsViewModel.kt index ffe199ccf01..be334dff08a 100644 --- a/app/src/main/kotlin/com/wire/android/ui/settings/devices/DeviceDetailsViewModel.kt +++ b/app/src/main/kotlin/com/wire/android/ui/settings/devices/DeviceDetailsViewModel.kt @@ -92,14 +92,12 @@ class DeviceDetailsViewModel @Inject constructor( viewModelScope.launch { val certificate = e2eiCertificate(deviceId) state = if (certificate is GetE2EICertificateUseCaseResult.Success) { - println("cyka getting cert success") state.copy( isE2eiCertificateActivated = true, e2eiCertificate = certificate.certificate, isLoadingCertificate = false ) } else { - println("cyka getting cert success") state.copy(isE2eiCertificateActivated = false, isLoadingCertificate = false) } } @@ -109,24 +107,19 @@ class DeviceDetailsViewModel @Inject constructor( state = state.copy(isLoadingCertificate = true) enrolE2EICertificateUseCase(context) { result -> result.fold({ - println("cyka error0 $it") state = state.copy( isLoadingCertificate = false, isE2EICertificateEnrollError = true ) }, { if (it is E2EIEnrollmentResult.Finalized) { - println("cyka Finalized, getting cert") getE2eiCertificate() state = state.copy(isE2EICertificateEnrollSuccess = true) } else if (it is E2EIEnrollmentResult.Failed) { - println("cyka error1 $it") state = state.copy( isLoadingCertificate = false, isE2EICertificateEnrollError = true ) - } else { - println("cyka other: $it") } }) } diff --git a/app/src/test/kotlin/com/wire/android/ui/home/sync/FeatureFlagNotificationViewModelTest.kt b/app/src/test/kotlin/com/wire/android/ui/home/sync/FeatureFlagNotificationViewModelTest.kt index 8ac17b40cdf..189e86f80dc 100644 --- a/app/src/test/kotlin/com/wire/android/ui/home/sync/FeatureFlagNotificationViewModelTest.kt +++ b/app/src/test/kotlin/com/wire/android/ui/home/sync/FeatureFlagNotificationViewModelTest.kt @@ -1,6 +1,7 @@ package com.wire.android.ui.home.sync import com.wire.android.config.CoroutineTestExtension +import com.wire.android.config.TestDispatcherProvider import com.wire.android.datastore.GlobalDataStore import com.wire.android.feature.AppLockSource import com.wire.android.feature.DisableAppLockUseCase @@ -295,7 +296,8 @@ class FeatureFlagNotificationViewModelTest { coreLogic = coreLogic, currentSessionUseCase = currentSession, globalDataStore = globalDataStore, - disableAppLockUseCase = disableAppLockUseCase + disableAppLockUseCase = disableAppLockUseCase, + dispatcherProvider = TestDispatcherProvider() ) init { diff --git a/app/src/test/kotlin/com/wire/android/ui/settings/devices/DeviceDetailsViewModelTest.kt b/app/src/test/kotlin/com/wire/android/ui/settings/devices/DeviceDetailsViewModelTest.kt index 57ac9e7a771..33ec8f95969 100644 --- a/app/src/test/kotlin/com/wire/android/ui/settings/devices/DeviceDetailsViewModelTest.kt +++ b/app/src/test/kotlin/com/wire/android/ui/settings/devices/DeviceDetailsViewModelTest.kt @@ -1,8 +1,10 @@ package com.wire.android.ui.settings.devices +import android.content.Context import androidx.lifecycle.SavedStateHandle import com.wire.android.config.CoroutineTestExtension import com.wire.android.config.NavigationTestExtension +import com.wire.android.feature.e2ei.GetE2EICertificateUseCase import com.wire.android.framework.TestClient import com.wire.android.framework.TestUser import com.wire.android.ui.authentication.devices.remove.RemoveDeviceDialogState @@ -204,6 +206,7 @@ class DeviceDetailsViewModelTest { // then assertFalse(viewModel.state.canBeRemoved) } + @Test fun `given self temporary client, when fetching state, then canBeRemoved is true`() = runTest { // given @@ -214,6 +217,7 @@ class DeviceDetailsViewModelTest { // then assertFalse(viewModel.state.canBeRemoved) } + @Test fun `given self permanent client, when fetching state, then canBeRemoved is true`() = runTest { // given @@ -224,6 +228,7 @@ class DeviceDetailsViewModelTest { // then assertTrue(viewModel.state.canBeRemoved) } + @Test fun `given self permanent current client, when fetching state, then canBeRemoved is false`() = runTest { // given @@ -234,6 +239,7 @@ class DeviceDetailsViewModelTest { // then assertFalse(viewModel.state.canBeRemoved) } + @Test fun `given other user permanent client, when fetching state, then canBeRemoved is false`() = runTest { // given @@ -245,8 +251,27 @@ class DeviceDetailsViewModelTest { assertFalse(viewModel.state.canBeRemoved) } + @Test + fun `given get certificate clicked, then should call GetE2EICertificate`() = + runTest { + val (arrangement, viewModel) = Arrangement() + .withRequiredMockSetup() + .withClientDetailsResult(GetClientDetailsResult.Success(TestClient.CLIENT, true)) + .arrange() + + viewModel.enrollE2eiCertificate(arrangement.context) + + coVerify { + arrangement.enrolE2EICertificateUseCase(any(), any()) + } + assertTrue(viewModel.state.isLoadingCertificate) + } + private class Arrangement { + @MockK + lateinit var context: Context + @MockK lateinit var savedStateHandle: SavedStateHandle @@ -271,6 +296,9 @@ class DeviceDetailsViewModelTest { @MockK lateinit var getE2eiCertificate: GetE2eiCertificateUseCase + @MockK + lateinit var enrolE2EICertificateUseCase: GetE2EICertificateUseCase + @MockK(relaxed = true) lateinit var onSuccess: () -> Unit @@ -286,7 +314,8 @@ class DeviceDetailsViewModelTest { updateClientVerificationStatus = updateClientVerificationStatus, currentUserId = currentUserId, observeUserInfo = observeUserInfo, - e2eiCertificate = getE2eiCertificate + e2eiCertificate = getE2eiCertificate, + enrolE2EICertificateUseCase = enrolE2EICertificateUseCase ) } diff --git a/kalium b/kalium index 454088c4918..bec2aa30cd7 160000 --- a/kalium +++ b/kalium @@ -1 +1 @@ -Subproject commit 454088c4918db1d0c4f66cb2d937a0eabbe2c67a +Subproject commit bec2aa30cd7361829c82971ce40a9953c16f62c0 From cd7976fda4986b65234d7c34da4ed623b09ae27a Mon Sep 17 00:00:00 2001 From: Boris Safonov Date: Tue, 12 Dec 2023 12:28:14 +0200 Subject: [PATCH 5/7] Fix build --- .../usecase/GetConversationMessagesFromSearchUseCase.kt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/app/src/main/kotlin/com/wire/android/ui/home/conversations/usecase/GetConversationMessagesFromSearchUseCase.kt b/app/src/main/kotlin/com/wire/android/ui/home/conversations/usecase/GetConversationMessagesFromSearchUseCase.kt index 3dd771258c1..c54ec06faee 100644 --- a/app/src/main/kotlin/com/wire/android/ui/home/conversations/usecase/GetConversationMessagesFromSearchUseCase.kt +++ b/app/src/main/kotlin/com/wire/android/ui/home/conversations/usecase/GetConversationMessagesFromSearchUseCase.kt @@ -64,7 +64,7 @@ class GetConversationMessagesFromSearchUseCase @Inject constructor( searchQuery = searchTerm, conversationId = conversationId, pagingConfig = pagingConfig, - startingOffset = max(0, lastReadIndex - PREFETCH_DISTANCE) + startingOffset = max(0, lastReadIndex - PREFETCH_DISTANCE).toLong() ).map { pagingData -> pagingData.flatMap { messageItem -> observeMemberDetailsByIds(messageMapper.memberIdList(listOf(messageItem))) From 8e0ca20ae018a9cee060a5920e5c562ac78fcc7e Mon Sep 17 00:00:00 2001 From: Boris Safonov Date: Tue, 12 Dec 2023 12:38:40 +0200 Subject: [PATCH 6/7] Fix build, again --- .../conversations/usecase/GetMessagesForConversationUseCase.kt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/app/src/main/kotlin/com/wire/android/ui/home/conversations/usecase/GetMessagesForConversationUseCase.kt b/app/src/main/kotlin/com/wire/android/ui/home/conversations/usecase/GetMessagesForConversationUseCase.kt index 9d76db008de..0061493000a 100644 --- a/app/src/main/kotlin/com/wire/android/ui/home/conversations/usecase/GetMessagesForConversationUseCase.kt +++ b/app/src/main/kotlin/com/wire/android/ui/home/conversations/usecase/GetMessagesForConversationUseCase.kt @@ -55,7 +55,7 @@ class GetMessagesForConversationUseCase @Inject constructor( return getMessages( conversationId, pagingConfig = pagingConfig, - startingOffset = max(0, lastReadIndex - PREFETCH_DISTANCE) + startingOffset = max(0, lastReadIndex - PREFETCH_DISTANCE).toLong() ).map { pagingData -> pagingData.flatMap { messageItem -> observeMemberDetailsByIds(messageMapper.memberIdList(listOf(messageItem))) From 4474aae98f2830a9bad9b839acf723e38653ec8a Mon Sep 17 00:00:00 2001 From: Boris Safonov Date: Wed, 13 Dec 2023 10:48:39 +0200 Subject: [PATCH 7/7] Updated kalium --- kalium | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/kalium b/kalium index bec2aa30cd7..1e78df60282 160000 --- a/kalium +++ b/kalium @@ -1 +1 @@ -Subproject commit bec2aa30cd7361829c82971ce40a9953c16f62c0 +Subproject commit 1e78df60282a78e6cc2b4061f3d14d702c3bd9d5