From a0bc84c53d0fa248701d82af27f6b56372d1dc9c Mon Sep 17 00:00:00 2001 From: AndroidBob Date: Tue, 13 Feb 2024 12:32:29 +0100 Subject: [PATCH] fix: serverConfig and notification crashes right after user becomes invalid [WPB-6552] [WPB-6233] (#2692) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Co-authored-by: Michał Saleniuk <30429749+saleniuk@users.noreply.github.com> Co-authored-by: Michał Saleniuk --- .../wire/android/GlobalObserversManager.kt | 24 ++++++--- .../feature/ObserveAppLockConfigUseCase.kt | 12 ++--- .../notification/WireNotificationManager.kt | 21 ++++++-- .../ui/calling/ProximitySensorManager.kt | 11 ++-- .../android/GlobalObserversManagerTest.kt | 17 ++++++ .../ObserveAppLockConfigUseCaseTest.kt | 26 +++++++++- .../WireNotificationManagerTest.kt | 52 +++++++++++++++++++ 7 files changed, 143 insertions(+), 20 deletions(-) diff --git a/app/src/main/kotlin/com/wire/android/GlobalObserversManager.kt b/app/src/main/kotlin/com/wire/android/GlobalObserversManager.kt index 944b629225a..5a5973f40bb 100644 --- a/app/src/main/kotlin/com/wire/android/GlobalObserversManager.kt +++ b/app/src/main/kotlin/com/wire/android/GlobalObserversManager.kt @@ -91,14 +91,26 @@ class GlobalObserversManager @Inject constructor( } coreLogic.getGlobalScope().observeValidAccounts() + .combine(persistentStatusesFlow) { list, persistentStatuses -> + val persistentStatusesMap = persistentStatuses.associate { it.userId to it.isPersistentWebSocketEnabled } + /* + Intersect both lists as they can be slightly out of sync because both lists can be updated at slightly different times. + When user is logged out, at this time one of them can still contain this invalid user - make sure that it's ignored. + When user is logged in, at this time one of them can still not contain this new user - ignore for now, + the user will be handled correctly in the next iteration when the second list becomes updated as well. + */ + list.map { (selfUser, _) -> selfUser } + .filter { persistentStatusesMap.containsKey(it.id) } + .map { it to persistentStatusesMap.getValue(it.id) } + } .distinctUntilChanged() - .combine(persistentStatusesFlow, ::Pair) - .collect { (list, persistentStatuses) -> - notificationChannelsManager.createUserNotificationChannels(list.map { it.first }) + .collectLatest { + // create notification channels for all valid users + notificationChannelsManager.createUserNotificationChannels(it.map { it.first }) - list.map { it.first.id } - // do not observe notifications for users with PersistentWebSocketEnabled, it will be done in PersistentWebSocketService - .filter { userId -> persistentStatuses.none { it.userId == userId && it.isPersistentWebSocketEnabled } } + // do not observe notifications for users with PersistentWebSocketEnabled, it will be done in PersistentWebSocketService + it.filter { (_, isPersistentWebSocketEnabled) -> !isPersistentWebSocketEnabled } + .map { (selfUser, _) -> selfUser.id } .run { notificationManager.observeNotificationsAndCallsWhileRunning(this, scope) } diff --git a/app/src/main/kotlin/com/wire/android/feature/ObserveAppLockConfigUseCase.kt b/app/src/main/kotlin/com/wire/android/feature/ObserveAppLockConfigUseCase.kt index d946074d9b1..19896940ebb 100644 --- a/app/src/main/kotlin/com/wire/android/feature/ObserveAppLockConfigUseCase.kt +++ b/app/src/main/kotlin/com/wire/android/feature/ObserveAppLockConfigUseCase.kt @@ -37,12 +37,8 @@ class ObserveAppLockConfigUseCase @Inject constructor( ) { operator fun invoke(): Flow = channelFlow { coreLogic.getGlobalScope().session.currentSessionFlow().collectLatest { sessionResult -> - when (sessionResult) { - is CurrentSessionResult.Failure -> { - send(AppLockConfig.Disabled(DEFAULT_APP_LOCK_TIMEOUT)) - } - - is CurrentSessionResult.Success -> { + when { + sessionResult is CurrentSessionResult.Success && sessionResult.accountInfo.isValid() -> { val userId = sessionResult.accountInfo.userId val appLockTeamFeatureConfigFlow = coreLogic.getSessionScope(userId).appLockTeamFeatureConfigObserver @@ -67,6 +63,10 @@ class ObserveAppLockConfigUseCase @Inject constructor( send(it) } } + + else -> { + send(AppLockConfig.Disabled(DEFAULT_APP_LOCK_TIMEOUT)) + } } } } diff --git a/app/src/main/kotlin/com/wire/android/notification/WireNotificationManager.kt b/app/src/main/kotlin/com/wire/android/notification/WireNotificationManager.kt index aaabb885f29..56588ef4118 100644 --- a/app/src/main/kotlin/com/wire/android/notification/WireNotificationManager.kt +++ b/app/src/main/kotlin/com/wire/android/notification/WireNotificationManager.kt @@ -18,6 +18,7 @@ package com.wire.android.notification +import androidx.annotation.VisibleForTesting import com.wire.android.R import com.wire.android.appLogger import com.wire.android.di.KaliumCoreLogic @@ -36,6 +37,7 @@ import com.wire.kalium.logic.data.notification.LocalNotificationMessage import com.wire.kalium.logic.data.user.UserId import com.wire.kalium.logic.feature.message.MarkMessagesAsNotifiedUseCase import com.wire.kalium.logic.feature.session.CurrentSessionResult +import com.wire.kalium.logic.feature.session.DoesValidSessionExistResult import com.wire.kalium.logic.feature.session.GetAllSessionsResult import com.wire.kalium.logic.feature.user.E2EIRequiredResult import kotlinx.coroutines.CoroutineScope @@ -244,9 +246,8 @@ class WireNotificationManager @Inject constructor( return } - // start observing notifications only for new users - userIds - .filter { observingJobs.userJobs[it]?.isAllActive() != true } + // start observing notifications only for new users with valid session and without active jobs + newUsersWithValidSessionAndWithoutActiveJobs(userIds) { observingJobs.userJobs[it]?.isAllActive() == true } .forEach { userId -> val jobs = UserObservingJobs( currentScreenJob = scope.launch(dispatcherProvider.default()) { @@ -271,6 +272,20 @@ class WireNotificationManager @Inject constructor( } } + @VisibleForTesting + internal suspend fun newUsersWithValidSessionAndWithoutActiveJobs( + userIds: List, + hasActiveJobs: (UserId) -> Boolean + ): List = userIds + .filter { !hasActiveJobs(it) } + .filter { + // double check if the valid session for the given user still exists + when (val result = coreLogic.getGlobalScope().doesValidSessionExist(it)) { + is DoesValidSessionExistResult.Success -> result.doesValidSessionExist + else -> false + } + } + private fun stopObservingForUser(userId: UserId, observingJobs: ObservingJobs) { messagesNotificationManager.hideAllNotificationsForUser(userId) observingJobs.userJobs[userId]?.cancelAll() diff --git a/app/src/main/kotlin/com/wire/android/ui/calling/ProximitySensorManager.kt b/app/src/main/kotlin/com/wire/android/ui/calling/ProximitySensorManager.kt index d72a972fb9b..b61eaccabf7 100644 --- a/app/src/main/kotlin/com/wire/android/ui/calling/ProximitySensorManager.kt +++ b/app/src/main/kotlin/com/wire/android/ui/calling/ProximitySensorManager.kt @@ -71,8 +71,9 @@ class ProximitySensorManager @Inject constructor( override fun onSensorChanged(event: SensorEvent) { appCoroutineScope.launch { coreLogic.get().globalScope { - when (val currentSession = currentSession.get().invoke()) { - is CurrentSessionResult.Success -> { + val currentSession = currentSession.get().invoke() + when { + currentSession is CurrentSessionResult.Success && currentSession.accountInfo.isValid() -> { val userId = currentSession.accountInfo.userId val isCallRunning = coreLogic.get().getSessionScope(userId).calls.isCallRunning() val distance = event.values.first() @@ -92,8 +93,10 @@ class ProximitySensorManager @Inject constructor( } } - else -> { - // NO SESSION - Nothing to do + else -> { // NO SESSION - just release in case it's still held + if (wakeLock.isHeld) { + wakeLock.release() + } } } } diff --git a/app/src/test/kotlin/com/wire/android/GlobalObserversManagerTest.kt b/app/src/test/kotlin/com/wire/android/GlobalObserversManagerTest.kt index ecb71639f69..e9579c3534b 100644 --- a/app/src/test/kotlin/com/wire/android/GlobalObserversManagerTest.kt +++ b/app/src/test/kotlin/com/wire/android/GlobalObserversManagerTest.kt @@ -146,6 +146,23 @@ class GlobalObserversManagerTest { coVerify(exactly = 0) { arrangement.messageScope.deleteEphemeralMessageEndDate() } } + @Test + fun `given validAccounts and persistentStatuses are out of sync, when setting up notifications, then ignore invalid users`() { + val validAccountsList = listOf(TestUser.SELF_USER) + val persistentStatusesList = listOf( + PersistentWebSocketStatus(TestUser.SELF_USER.id, false), + PersistentWebSocketStatus(TestUser.USER_ID.copy(value = "something else"), true) + ) + val (arrangement, manager) = Arrangement() + .withValidAccounts(validAccountsList.map { it to null }) + .withPersistentWebSocketConnectionStatuses(persistentStatusesList) + .arrange() + manager.observe() + coVerify(exactly = 1) { + arrangement.notificationChannelsManager.createUserNotificationChannels(listOf(TestUser.SELF_USER)) + } + } + private class Arrangement { @MockK diff --git a/app/src/test/kotlin/com/wire/android/feature/ObserveAppLockConfigUseCaseTest.kt b/app/src/test/kotlin/com/wire/android/feature/ObserveAppLockConfigUseCaseTest.kt index e0caa52dfdf..c566f67c9f7 100644 --- a/app/src/test/kotlin/com/wire/android/feature/ObserveAppLockConfigUseCaseTest.kt +++ b/app/src/test/kotlin/com/wire/android/feature/ObserveAppLockConfigUseCaseTest.kt @@ -22,6 +22,7 @@ import com.wire.android.datastore.GlobalDataStore import com.wire.kalium.logic.CoreLogic import com.wire.kalium.logic.configuration.AppLockTeamConfig import com.wire.kalium.logic.data.auth.AccountInfo +import com.wire.kalium.logic.data.logout.LogoutReason import com.wire.kalium.logic.data.user.UserId import com.wire.kalium.logic.feature.UserSessionScope import com.wire.kalium.logic.feature.applock.AppLockTeamFeatureConfigObserver @@ -54,6 +55,22 @@ class ObserveAppLockConfigUseCaseTest { } } + @Test + fun givenInvalidSession_whenObservingAppLock_thenSendDisabledStatus() = runTest { + val (_, useCase) = Arrangement() + .withInvalidSession() + .arrange() + + val result = useCase.invoke() + + result.test { + val appLockStatus = awaitItem() + + assertEquals(AppLockConfig.Disabled(timeout), appLockStatus) + awaitComplete() + } + } + @Test fun givenValidSessionAndAppLockedByTeam_whenObservingAppLock_thenSendEnforcedByTeamStatus() = runTest { @@ -142,6 +159,11 @@ class ObserveAppLockConfigUseCaseTest { flowOf(CurrentSessionResult.Failure.SessionNotFound) } + fun withInvalidSession() = apply { + coEvery { coreLogic.getGlobalScope().session.currentSessionFlow() } returns + flowOf(CurrentSessionResult.Success(accountInfoInvalid)) + } + fun withValidSession() = apply { coEvery { coreLogic.getGlobalScope().session.currentSessionFlow() } returns flowOf(CurrentSessionResult.Success(accountInfo)) @@ -177,7 +199,9 @@ class ObserveAppLockConfigUseCaseTest { } companion object { - private val accountInfo = AccountInfo.Valid(UserId("userId", "domain")) + private val userId = UserId("userId", "domain") + private val accountInfo = AccountInfo.Valid(userId) + private val accountInfoInvalid = AccountInfo.Invalid(userId, LogoutReason.DELETED_ACCOUNT) private val timeout = 60.seconds } } diff --git a/app/src/test/kotlin/com/wire/android/notification/WireNotificationManagerTest.kt b/app/src/test/kotlin/com/wire/android/notification/WireNotificationManagerTest.kt index 8c837d07f5f..5d8c0f2d868 100644 --- a/app/src/test/kotlin/com/wire/android/notification/WireNotificationManagerTest.kt +++ b/app/src/test/kotlin/com/wire/android/notification/WireNotificationManagerTest.kt @@ -55,6 +55,7 @@ import com.wire.kalium.logic.feature.message.MessageScope import com.wire.kalium.logic.feature.message.Result import com.wire.kalium.logic.feature.session.CurrentSessionFlowUseCase import com.wire.kalium.logic.feature.session.CurrentSessionResult +import com.wire.kalium.logic.feature.session.DoesValidSessionExistResult import com.wire.kalium.logic.feature.session.GetAllSessionsResult import com.wire.kalium.logic.feature.session.GetSessionsUseCase import com.wire.kalium.logic.feature.user.E2EIRequiredResult @@ -83,6 +84,7 @@ import kotlinx.coroutines.test.advanceUntilIdle import kotlinx.coroutines.test.runCurrent import kotlinx.coroutines.test.runTest import kotlinx.datetime.Instant +import org.amshove.kluent.internal.assertEquals import org.junit.jupiter.api.Test import kotlin.time.Duration.Companion.minutes @@ -696,6 +698,51 @@ class WireNotificationManagerTest { } } + @Test + fun givenSessionExistsForTheUserAndNoActiveJobs_whenGettingUsersToObserve_thenReturnThatUser() = + runTest(dispatcherProvider.main()) { + // given + val userId = provideUserId() + val (_, manager) = Arrangement() + .withDoesValidSessionExistResult(userId, DoesValidSessionExistResult.Success(true)) + .arrange() + val hasActiveJobs: (UserId) -> Boolean = { false } + // when + val result = manager.newUsersWithValidSessionAndWithoutActiveJobs(listOf(userId), hasActiveJobs) + // then + assertEquals(listOf(userId), result) + } + + @Test + fun givenSessionExistsForTheUserButWithActiveJobs_whenGettingUsersToObserve_thenDoNotReturnThatUser() = + runTest(dispatcherProvider.main()) { + // given + val userId = provideUserId() + val (_, manager) = Arrangement() + .withDoesValidSessionExistResult(userId, DoesValidSessionExistResult.Success(true)) + .arrange() + val hasActiveJobs: (UserId) -> Boolean = { true } + // when + val result = manager.newUsersWithValidSessionAndWithoutActiveJobs(listOf(userId), hasActiveJobs) + // then + assertEquals(listOf(), result) + } + + @Test + fun givenSessionDoesNotExistForTheUserAndNoActiveJobs_whenGettingUsersToObserve_thenDoNotReturnThatUser() = + runTest(dispatcherProvider.main()) { + // given + val userId = provideUserId() + val (_, manager) = Arrangement() + .withDoesValidSessionExistResult(userId, DoesValidSessionExistResult.Success(false)) + .arrange() + val hasActiveJobs: (UserId) -> Boolean = { false } + // when + val result = manager.newUsersWithValidSessionAndWithoutActiveJobs(listOf(userId), hasActiveJobs) + // then + assertEquals(listOf(), result) + } + private inner class Arrangement { @MockK lateinit var coreLogic: CoreLogic @@ -813,6 +860,7 @@ class WireNotificationManagerTest { every { servicesManager.startOngoingCallService() } returns Unit every { servicesManager.stopOngoingCallService() } returns Unit every { pingRinger.ping(any(), any()) } returns Unit + coEvery { globalKaliumScope.doesValidSessionExist.invoke(any()) } returns DoesValidSessionExistResult.Success(true) } private fun mockSpecificUserSession( @@ -890,6 +938,10 @@ class WireNotificationManagerTest { coEvery { observeE2EIRequired.invoke() } returns flowOf(result) } + fun withDoesValidSessionExistResult(userId: UserId, result: DoesValidSessionExistResult) = apply { + coEvery { globalKaliumScope.doesValidSessionExist.invoke(userId) } returns result + } + fun arrange() = this to wireNotificationManager }