From bc1a65ec6d368b01db138d75074e8891d63de718 Mon Sep 17 00:00:00 2001 From: Yamil Medina Date: Tue, 31 Dec 2024 05:53:47 -0300 Subject: [PATCH] fix: fallback to proteus if conversation already present but MLS is default (WPB-15191) (#3200) * fix: fallback to proteus if conversation already present and no common protocol * fix: test coverage * fix: test coverage --- .../GetOrCreateOneToOneConversationUseCase.kt | 24 +++++++-- .../conversation/mls/OneOnOneMigrator.kt | 52 ++++++++++++++----- .../conversation/mls/OneOnOneResolver.kt | 13 ++++- .../conversation/mls/OneOnOneMigratorTest.kt | 46 +++++++++------- .../conversation/mls/OneOnOneResolverTest.kt | 19 +++++++ .../mls/OneOnOneMigratorArrangement.kt | 8 +++ 6 files changed, 125 insertions(+), 37 deletions(-) diff --git a/logic/src/commonMain/kotlin/com/wire/kalium/logic/feature/conversation/GetOrCreateOneToOneConversationUseCase.kt b/logic/src/commonMain/kotlin/com/wire/kalium/logic/feature/conversation/GetOrCreateOneToOneConversationUseCase.kt index 7597da0ddb3..a601c05c4e8 100644 --- a/logic/src/commonMain/kotlin/com/wire/kalium/logic/feature/conversation/GetOrCreateOneToOneConversationUseCase.kt +++ b/logic/src/commonMain/kotlin/com/wire/kalium/logic/feature/conversation/GetOrCreateOneToOneConversationUseCase.kt @@ -33,10 +33,6 @@ import kotlinx.coroutines.flow.first /** * Operation that creates one-to-one Conversation with specific [UserId] (only if it is absent in local DB) * and returns [Conversation] data. - * - * @param otherUserId [UserId] private conversation with which we are interested in. - * @return Result with [Conversation] in case of success, or [CoreFailure] if something went wrong: - * can't get data from local DB, or can't create a conversation. */ interface GetOrCreateOneToOneConversationUseCase { suspend operator fun invoke(otherUserId: UserId): CreateConversationResult @@ -47,6 +43,14 @@ internal class GetOrCreateOneToOneConversationUseCaseImpl( private val userRepository: UserRepository, private val oneOnOneResolver: OneOnOneResolver ) : GetOrCreateOneToOneConversationUseCase { + + /** + * The use case operation operation params and return type. + * + * @param otherUserId [UserId] private conversation with which we are interested in. + * @return Result with [Conversation] in case of success, or [CoreFailure] if something went wrong: + * can't get data from local DB, or can't create a conversation. + */ override suspend operator fun invoke(otherUserId: UserId): CreateConversationResult { // TODO periodically re-resolve one-on-one return conversationRepository.observeOneToOneConversationWithOtherUser(otherUserId) @@ -66,6 +70,18 @@ internal class GetOrCreateOneToOneConversationUseCaseImpl( }) } + /** + * Resolves one-on-one conversation with the user. + * Resolving conversations is the process of: + * + * - Intersecting the supported protocols of the self user and the other user. + * - Selecting the common protocol, based on the team settings with the highest priority. + * - Get or create a conversation with the other user. + * - If the protocol now is MLS, migrate the existing Proteus conversation to MLS. + * - Mark the conversation as active. + * + * If no common protocol is found, and we have existing Proteus conversations, we do best effort to use them as fallback. + */ private suspend fun resolveOneOnOneConversationWithUser(otherUserId: UserId): Either = userRepository.userById(otherUserId).flatMap { otherUser -> // TODO support lazily establishing mls group for team 1-1 diff --git a/logic/src/commonMain/kotlin/com/wire/kalium/logic/feature/conversation/mls/OneOnOneMigrator.kt b/logic/src/commonMain/kotlin/com/wire/kalium/logic/feature/conversation/mls/OneOnOneMigrator.kt index 5737853c8eb..2022a62263f 100644 --- a/logic/src/commonMain/kotlin/com/wire/kalium/logic/feature/conversation/mls/OneOnOneMigrator.kt +++ b/logic/src/commonMain/kotlin/com/wire/kalium/logic/feature/conversation/mls/OneOnOneMigrator.kt @@ -36,7 +36,19 @@ import com.wire.kalium.logic.functional.map import com.wire.kalium.logic.kaliumLogger interface OneOnOneMigrator { + /** + * Migrates the user's one-on-one Proteus. Without creating a new one since MLS is the default, marking it as active. + */ + suspend fun migrateExistingProteus(user: OtherUser): Either + + /** + * Get one-on-one conversation with the user, if not found, create a new one (Proteus still default) and mark it as active. + */ suspend fun migrateToProteus(user: OtherUser): Either + + /** + * Perform migration of Proteus to MLS keeping history and marking the new conversation as active. + */ suspend fun migrateToMLS(user: OtherUser): Either } @@ -100,19 +112,35 @@ internal class OneOnOneMigratorImpl( } } + override suspend fun migrateExistingProteus(user: OtherUser): Either = + conversationRepository.getOneOnOneConversationsWithOtherUser(user.id, Conversation.Protocol.PROTEUS).flatMap { conversationIds -> + if (conversationIds.isNotEmpty()) { + val conversationId = conversationIds.first() + Either.Right(conversationId) + } else { + Either.Left(StorageFailure.DataNotFound) + } + }.flatMap { conversationId -> + if (user.activeOneOnOneConversationId != conversationId) { + kaliumLogger.d("resolved existing one-on-one to proteus, user = ${user.id.toLogString()}") + userRepository.updateActiveOneOnOneConversation(user.id, conversationId) + } + Either.Right(conversationId) + } + private suspend fun migrateOneOnOneHistory(user: OtherUser, targetConversation: ConversationId): Either { - return conversationRepository.getOneOnOneConversationsWithOtherUser( - otherUserId = user.id, - protocol = Conversation.Protocol.PROTEUS - ).flatMap { proteusOneOnOneConversations -> - // We can theoretically have more than one proteus 1-1 conversation with - // team members since there was no backend safeguards against this - proteusOneOnOneConversations.foldToEitherWhileRight(Unit) { proteusOneOnOneConversation, _ -> - messageRepository.moveMessagesToAnotherConversation( - originalConversation = proteusOneOnOneConversation, - targetConversation = targetConversation - ) - } + return conversationRepository.getOneOnOneConversationsWithOtherUser( + otherUserId = user.id, + protocol = Conversation.Protocol.PROTEUS + ).flatMap { proteusOneOnOneConversations -> + // We can theoretically have more than one proteus 1-1 conversation with + // team members since there was no backend safeguards against this + proteusOneOnOneConversations.foldToEitherWhileRight(Unit) { proteusOneOnOneConversation, _ -> + messageRepository.moveMessagesToAnotherConversation( + originalConversation = proteusOneOnOneConversation, + targetConversation = targetConversation + ) } + } } } diff --git a/logic/src/commonMain/kotlin/com/wire/kalium/logic/feature/conversation/mls/OneOnOneResolver.kt b/logic/src/commonMain/kotlin/com/wire/kalium/logic/feature/conversation/mls/OneOnOneResolver.kt index c9ba938d312..a136f97107c 100644 --- a/logic/src/commonMain/kotlin/com/wire/kalium/logic/feature/conversation/mls/OneOnOneResolver.kt +++ b/logic/src/commonMain/kotlin/com/wire/kalium/logic/feature/conversation/mls/OneOnOneResolver.kt @@ -31,7 +31,9 @@ import com.wire.kalium.logic.feature.protocol.OneOnOneProtocolSelector import com.wire.kalium.logic.functional.Either import com.wire.kalium.logic.functional.flatMap import com.wire.kalium.logic.functional.flatMapLeft +import com.wire.kalium.logic.functional.fold import com.wire.kalium.logic.functional.foldToEitherWhileRight +import com.wire.kalium.logic.functional.left import com.wire.kalium.logic.functional.map import com.wire.kalium.logic.kaliumLogger import com.wire.kalium.util.KaliumDispatcher @@ -154,11 +156,18 @@ internal class OneOnOneResolverImpl( if (invalidateCurrentKnownProtocols) { userRepository.fetchUsersByIds(setOf(user.id)) } - return oneOnOneProtocolSelector.getProtocolForUser(user.id).flatMap { supportedProtocol -> + return oneOnOneProtocolSelector.getProtocolForUser(user.id).fold({ coreFailure -> + if (coreFailure is CoreFailure.NoCommonProtocolFound.OtherNeedToUpdate) { + kaliumLogger.i("Resolving existing proteus 1:1 as not matching protocol found with: ${user.id.toLogString()}") + oneOnOneMigrator.migrateExistingProteus(user) + } else { + coreFailure.left() + } + }, { supportedProtocol -> when (supportedProtocol) { SupportedProtocol.PROTEUS -> oneOnOneMigrator.migrateToProteus(user) SupportedProtocol.MLS -> oneOnOneMigrator.migrateToMLS(user) } - } + }) } } diff --git a/logic/src/commonTest/kotlin/com/wire/kalium/logic/feature/conversation/mls/OneOnOneMigratorTest.kt b/logic/src/commonTest/kotlin/com/wire/kalium/logic/feature/conversation/mls/OneOnOneMigratorTest.kt index d7f199fa13d..a80a0318de9 100644 --- a/logic/src/commonTest/kotlin/com/wire/kalium/logic/feature/conversation/mls/OneOnOneMigratorTest.kt +++ b/logic/src/commonTest/kotlin/com/wire/kalium/logic/feature/conversation/mls/OneOnOneMigratorTest.kt @@ -65,25 +65,6 @@ class OneOnOneMigratorTest { }.wasNotInvoked() } - @Test - fun givenUnassignedOneOnOne_whenMigratingToProteus_thenShouldAssignOneOnOneConversation() = runTest { - val user = TestUser.OTHER.copy( - activeOneOnOneConversationId = null - ) - - val (arrangement, oneOneMigrator) = arrange { - withGetOneOnOneConversationsWithOtherUserReturning(Either.Right(listOf(TestConversation.ID))) - withUpdateOneOnOneConversationReturning(Either.Right(Unit)) - } - - oneOneMigrator.migrateToProteus(user) - .shouldSucceed() - - coVerify { - arrangement.userRepository.updateActiveOneOnOneConversation(eq(user.id), eq(TestConversation.ID)) - }.wasInvoked() - } - @Test fun givenNoExistingTeamOneOnOne_whenMigratingToProteus_thenShouldCreateGroupConversation() = runTest { val user = TestUser.OTHER.copy( @@ -252,6 +233,33 @@ class OneOnOneMigratorTest { }.wasInvoked(exactly = once) } + @Test + fun givenExistingTeamOneOnOne_whenMigratingToProteus_thenShouldNOTCreateGroupConversation() = runTest { + val user = TestUser.OTHER.copy( + activeOneOnOneConversationId = null + ) + + val (arrangement, oneOneMigrator) = arrange { + withGetOneOnOneConversationsWithOtherUserReturning(Either.Right(listOf(TestConversation.ONE_ON_ONE().id))) + withUpdateOneOnOneConversationReturning(Either.Right(Unit)) + } + + oneOneMigrator.migrateExistingProteus(user) + .shouldSucceed() + + coVerify { + arrangement.conversationGroupRepository.createGroupConversation( + name = eq(null), + usersList = eq(listOf(TestUser.OTHER.id)), + options = eq(ConversationOptions()) + ) + }.wasNotInvoked() + + coVerify { + arrangement.userRepository.updateActiveOneOnOneConversation(eq(TestUser.OTHER.id), eq(TestConversation.ONE_ON_ONE().id)) + }.wasInvoked() + } + private class Arrangement(private val block: suspend Arrangement.() -> Unit) : MLSOneOnOneConversationResolverArrangement by MLSOneOnOneConversationResolverArrangementImpl(), MessageRepositoryArrangement by MessageRepositoryArrangementImpl(), diff --git a/logic/src/commonTest/kotlin/com/wire/kalium/logic/feature/conversation/mls/OneOnOneResolverTest.kt b/logic/src/commonTest/kotlin/com/wire/kalium/logic/feature/conversation/mls/OneOnOneResolverTest.kt index 573083d1951..d00102a78d4 100644 --- a/logic/src/commonTest/kotlin/com/wire/kalium/logic/feature/conversation/mls/OneOnOneResolverTest.kt +++ b/logic/src/commonTest/kotlin/com/wire/kalium/logic/feature/conversation/mls/OneOnOneResolverTest.kt @@ -22,6 +22,8 @@ import com.wire.kalium.logic.data.user.SupportedProtocol import com.wire.kalium.logic.framework.TestConversation import com.wire.kalium.logic.framework.TestUser import com.wire.kalium.logic.functional.Either +import com.wire.kalium.logic.functional.left +import com.wire.kalium.logic.functional.right import com.wire.kalium.logic.util.arrangement.IncrementalSyncRepositoryArrangement import com.wire.kalium.logic.util.arrangement.IncrementalSyncRepositoryArrangementImpl import com.wire.kalium.logic.util.arrangement.mls.OneOnOneMigratorArrangement @@ -244,6 +246,23 @@ class OneOnOneResolverTest { }.wasInvoked(exactly = once) } + @Test + fun givenProtocolResolvesToOtherNeedToUpdate_whenResolveOneOnOneConversationWithUser_thenMigrateExistingToProteus() = runTest { + // given + val (arrangement, resolver) = arrange { + withGetProtocolForUser(CoreFailure.NoCommonProtocolFound.OtherNeedToUpdate.left()) + withMigrateExistingToProteusReturns(TestConversation.ID.right()) + } + + // when + resolver.resolveOneOnOneConversationWithUser(OTHER_USER, false).shouldSucceed() + + // then + coVerify { + arrangement.oneOnOneMigrator.migrateExistingProteus(eq(OTHER_USER)) + }.wasInvoked(exactly = once) + } + private class Arrangement(private val block: suspend Arrangement.() -> Unit) : UserRepositoryArrangement by UserRepositoryArrangementImpl(), OneOnOneProtocolSelectorArrangement by OneOnOneProtocolSelectorArrangementImpl(), diff --git a/logic/src/commonTest/kotlin/com/wire/kalium/logic/util/arrangement/mls/OneOnOneMigratorArrangement.kt b/logic/src/commonTest/kotlin/com/wire/kalium/logic/util/arrangement/mls/OneOnOneMigratorArrangement.kt index 36207e4774e..305b88f50fd 100644 --- a/logic/src/commonTest/kotlin/com/wire/kalium/logic/util/arrangement/mls/OneOnOneMigratorArrangement.kt +++ b/logic/src/commonTest/kotlin/com/wire/kalium/logic/util/arrangement/mls/OneOnOneMigratorArrangement.kt @@ -33,6 +33,8 @@ interface OneOnOneMigratorArrangement { suspend fun withMigrateToMLSReturns(result: Either) suspend fun withMigrateToProteusReturns(result: Either) + + suspend fun withMigrateExistingToProteusReturns(result: Either) } class OneOnOneMigratorArrangementImpl : OneOnOneMigratorArrangement { @@ -51,4 +53,10 @@ class OneOnOneMigratorArrangementImpl : OneOnOneMigratorArrangement { oneOnOneMigrator.migrateToProteus(any()) }.returns(result) } + + override suspend fun withMigrateExistingToProteusReturns(result: Either) { + coEvery { + oneOnOneMigrator.migrateExistingProteus(any()) + }.returns(result) + } }