From 60b8751c5d484c6075c6658f2e926f07508b5bd6 Mon Sep 17 00:00:00 2001 From: ohassine Date: Thu, 21 Nov 2024 17:09:21 +0100 Subject: [PATCH 1/2] feat: update teamId after migrating from personal to team account --- .../kalium/logic/data/user/CreateUserTeam.kt | 2 +- .../kalium/logic/data/user/UserRepository.kt | 7 +++- .../kalium/logic/feature/UserSessionScope.kt | 6 ++- .../MigrateFromPersonalToTeamUseCase.kt | 14 +++++-- .../MigrateFromPersonalToTeamUseCaseTest.kt | 41 ++++++++++++++----- .../com/wire/kalium/persistence/Users.sq | 3 ++ .../wire/kalium/persistence/dao/UserDAO.kt | 1 + .../kalium/persistence/dao/UserDAOImpl.kt | 4 ++ 8 files changed, 61 insertions(+), 17 deletions(-) diff --git a/data/src/commonMain/kotlin/com/wire/kalium/logic/data/user/CreateUserTeam.kt b/data/src/commonMain/kotlin/com/wire/kalium/logic/data/user/CreateUserTeam.kt index f5e78793d5d..622d28ad298 100644 --- a/data/src/commonMain/kotlin/com/wire/kalium/logic/data/user/CreateUserTeam.kt +++ b/data/src/commonMain/kotlin/com/wire/kalium/logic/data/user/CreateUserTeam.kt @@ -1,3 +1,3 @@ package com.wire.kalium.logic.data.user -data class CreateUserTeam(val teamName: String) +data class CreateUserTeam(val teamId: String, val teamName: String) diff --git a/logic/src/commonMain/kotlin/com/wire/kalium/logic/data/user/UserRepository.kt b/logic/src/commonMain/kotlin/com/wire/kalium/logic/data/user/UserRepository.kt index 35360a5aaa8..e8760f83b79 100644 --- a/logic/src/commonMain/kotlin/com/wire/kalium/logic/data/user/UserRepository.kt +++ b/logic/src/commonMain/kotlin/com/wire/kalium/logic/data/user/UserRepository.kt @@ -166,6 +166,7 @@ interface UserRepository { suspend fun getUsersMinimizedByQualifiedIDs(userIds: List): Either> suspend fun getNameAndHandle(userId: UserId): Either suspend fun migrateUserToTeam(teamName: String): Either + suspend fun updateTeamId(userId: UserId, teamId: TeamId): Either } @Suppress("LongParameterList", "TooManyFunctions") @@ -652,7 +653,7 @@ internal class UserDataSource internal constructor( override suspend fun migrateUserToTeam(teamName: String): Either { return wrapApiRequest { upgradePersonalToTeamApi.migrateToTeam(teamName) }.map { dto -> - CreateUserTeam(dto.teamName) + CreateUserTeam(dto.teamId, dto.teamName) } .onSuccess { kaliumLogger.d("Migrated user to team") @@ -663,6 +664,10 @@ internal class UserDataSource internal constructor( } } + override suspend fun updateTeamId(userId: UserId, teamId: TeamId): Either = wrapStorageRequest { + userDAO.updateTeamId(userId.toDao(), teamId.value) + } + companion object { internal const val SELF_USER_ID_KEY = "selfUserID" diff --git a/logic/src/commonMain/kotlin/com/wire/kalium/logic/feature/UserSessionScope.kt b/logic/src/commonMain/kotlin/com/wire/kalium/logic/feature/UserSessionScope.kt index ae927be25bb..d218ed15118 100644 --- a/logic/src/commonMain/kotlin/com/wire/kalium/logic/feature/UserSessionScope.kt +++ b/logic/src/commonMain/kotlin/com/wire/kalium/logic/feature/UserSessionScope.kt @@ -560,6 +560,10 @@ class UserSessionScope internal constructor( } } + private val invalidateTeamId = { + _teamId = Either.Left(CoreFailure.Unknown(Throwable("NotInitialized"))) + } + private val selfTeamId = SelfTeamIdProvider { teamId() } private val accessTokenRepository: AccessTokenRepository @@ -2095,7 +2099,7 @@ class UserSessionScope internal constructor( ) val migrateFromPersonalToTeam: MigrateFromPersonalToTeamUseCase - get() = MigrateFromPersonalToTeamUseCaseImpl(userRepository) + get() = MigrateFromPersonalToTeamUseCaseImpl(userId, userRepository, invalidateTeamId) internal val getProxyCredentials: GetProxyCredentialsUseCase get() = GetProxyCredentialsUseCaseImpl(sessionManager) diff --git a/logic/src/commonMain/kotlin/com/wire/kalium/logic/feature/user/migration/MigrateFromPersonalToTeamUseCase.kt b/logic/src/commonMain/kotlin/com/wire/kalium/logic/feature/user/migration/MigrateFromPersonalToTeamUseCase.kt index 8d3e841940e..9eb493a1b85 100644 --- a/logic/src/commonMain/kotlin/com/wire/kalium/logic/feature/user/migration/MigrateFromPersonalToTeamUseCase.kt +++ b/logic/src/commonMain/kotlin/com/wire/kalium/logic/feature/user/migration/MigrateFromPersonalToTeamUseCase.kt @@ -18,8 +18,11 @@ package com.wire.kalium.logic.feature.user.migration import com.wire.kalium.logic.CoreFailure +import com.wire.kalium.logic.data.id.TeamId +import com.wire.kalium.logic.data.user.UserId import com.wire.kalium.logic.data.user.UserRepository import com.wire.kalium.logic.functional.fold +import com.wire.kalium.logic.functional.map /** * Use case to migrate user personal account to team account. @@ -30,12 +33,14 @@ interface MigrateFromPersonalToTeamUseCase { } sealed class MigrateFromPersonalToTeamResult { - data class Success(val teamName: String) : MigrateFromPersonalToTeamResult() + data object Success : MigrateFromPersonalToTeamResult() data class Error(val failure: CoreFailure) : MigrateFromPersonalToTeamResult() } internal class MigrateFromPersonalToTeamUseCaseImpl internal constructor( + private val selfUserId: UserId, private val userRepository: UserRepository, + private val invalidateTeamId: () -> Unit ) : MigrateFromPersonalToTeamUseCase { override suspend operator fun invoke( teamName: String, @@ -43,9 +48,10 @@ internal class MigrateFromPersonalToTeamUseCaseImpl internal constructor( return userRepository.migrateUserToTeam(teamName) .fold( { error -> return MigrateFromPersonalToTeamResult.Error(error) }, - { success -> - // TODO Invalidate team id in memory so UserSessionScope.selfTeamId got updated data WPB-12187 - MigrateFromPersonalToTeamResult.Success(teamName = success.teamName) + { user -> + userRepository.updateTeamId(selfUserId, TeamId(user.teamId)) + invalidateTeamId() + MigrateFromPersonalToTeamResult.Success } ) } diff --git a/logic/src/commonTest/kotlin/com/wire/kalium/logic/feature/user/migration/MigrateFromPersonalToTeamUseCaseTest.kt b/logic/src/commonTest/kotlin/com/wire/kalium/logic/feature/user/migration/MigrateFromPersonalToTeamUseCaseTest.kt index 425af5a8d1e..d9efa29641f 100644 --- a/logic/src/commonTest/kotlin/com/wire/kalium/logic/feature/user/migration/MigrateFromPersonalToTeamUseCaseTest.kt +++ b/logic/src/commonTest/kotlin/com/wire/kalium/logic/feature/user/migration/MigrateFromPersonalToTeamUseCaseTest.kt @@ -20,8 +20,10 @@ package com.wire.kalium.logic.feature.user.migration import com.wire.kalium.logic.CoreFailure import com.wire.kalium.logic.NetworkFailure +import com.wire.kalium.logic.StorageFailure import com.wire.kalium.logic.data.user.CreateUserTeam import com.wire.kalium.logic.data.user.UserRepository +import com.wire.kalium.logic.framework.TestUser import com.wire.kalium.logic.functional.Either import com.wire.kalium.network.api.model.ErrorResponse import com.wire.kalium.network.exceptions.KaliumException @@ -29,20 +31,30 @@ import io.ktor.http.HttpStatusCode import io.mockative.Mock import io.mockative.any import io.mockative.coEvery +import io.mockative.coVerify import io.mockative.mock +import io.mockative.once import kotlinx.coroutines.test.runTest import kotlin.test.Test import kotlin.test.assertEquals import kotlin.test.assertIs +import kotlin.test.assertTrue class MigrateFromPersonalToTeamUseCaseTest { @Test fun givenRepositorySucceeds_whenMigratingUserToTeam_thenShouldPropagateSuccess() = runTest { - val (_, useCase) = Arrangement().withSuccessRepository().arrange() + val (arrangement, useCase) = Arrangement() + .withUpdateTeamIdReturning(Either.Right(Unit)) + .withMigrationSuccess() + .arrange() val result = useCase(teamName = "teamName") + coVerify { + arrangement.userRepository.updateTeamId(any(), any()) + }.wasInvoked(exactly = once) + assertTrue(arrangement.isCachedTeamIdInvalidated) assertIs(result) } @@ -50,7 +62,7 @@ class MigrateFromPersonalToTeamUseCaseTest { fun givenRepositoryFailsWithNoNetworkConnection_whenMigratingUserToTeam_thenShouldPropagateFailure() = runTest { val coreFailure = NetworkFailure.NoNetworkConnection(null) - val (_, useCase) = Arrangement().withRepositoryReturning(Either.Left(coreFailure)) + val (_, useCase) = Arrangement().withMigrationReturning(Either.Left(coreFailure)) .arrange() val result = useCase(teamName = "teamName") @@ -79,7 +91,7 @@ class MigrateFromPersonalToTeamUseCaseTest { @Test fun givenRepositoryFailsWithNotFound_whenMigratingUserToTeam_thenShouldPropagateFailure() = runTest { - val (_, useCase) = Arrangement().withNotFoundRepository().arrange() + val (_, useCase) = Arrangement().withMigrationFailure().arrange() val result = useCase(teamName = "teamName") @@ -98,10 +110,12 @@ class MigrateFromPersonalToTeamUseCaseTest { @Mock val userRepository: UserRepository = mock(UserRepository::class) - suspend fun withSuccessRepository() = apply { + var isCachedTeamIdInvalidated = false + + suspend fun withMigrationSuccess() = apply { coEvery { userRepository.migrateUserToTeam(any()) }.returns( Either.Right( - CreateUserTeam("teamName") + CreateUserTeam("teamId", "teamName") ) ) } @@ -122,7 +136,7 @@ class MigrateFromPersonalToTeamUseCaseTest { ) } - suspend fun withNotFoundRepository() = apply { + suspend fun withMigrationFailure() = apply { coEvery { userRepository.migrateUserToTeam(any()) }.returns( Either.Left( NetworkFailure.ServerMiscommunication( @@ -138,14 +152,21 @@ class MigrateFromPersonalToTeamUseCaseTest { ) } - suspend fun withRepositoryReturning(result: Either) = + suspend fun withMigrationReturning(result: Either) = apply { coEvery { userRepository.migrateUserToTeam(any()) }.returns(result) } + suspend fun withUpdateTeamIdReturning(result: Either) = apply { + coEvery { userRepository.updateTeamId(any(), any()) }.returns(result) + } + fun arrange() = this to MigrateFromPersonalToTeamUseCaseImpl( - userRepository = userRepository + selfUserId = TestUser.SELF.id, + userRepository = userRepository, + invalidateTeamId = { + isCachedTeamIdInvalidated = true + } ) } - -} \ No newline at end of file +} diff --git a/persistence/src/commonMain/db_user/com/wire/kalium/persistence/Users.sq b/persistence/src/commonMain/db_user/com/wire/kalium/persistence/Users.sq index 56dd1db0d6f..385d13f2602 100644 --- a/persistence/src/commonMain/db_user/com/wire/kalium/persistence/Users.sq +++ b/persistence/src/commonMain/db_user/com/wire/kalium/persistence/Users.sq @@ -273,3 +273,6 @@ SELECT active_one_on_one_conversation_id FROM User WHERE qualified_id = :userId; selectNamesAndHandle: SELECT name, handle FROM User WHERE qualified_id = :userId; + +updateTeamId: +UPDATE User SET team = ? WHERE qualified_id = ?; diff --git a/persistence/src/commonMain/kotlin/com/wire/kalium/persistence/dao/UserDAO.kt b/persistence/src/commonMain/kotlin/com/wire/kalium/persistence/dao/UserDAO.kt index 5c535cd5ccb..c92b50621e0 100644 --- a/persistence/src/commonMain/kotlin/com/wire/kalium/persistence/dao/UserDAO.kt +++ b/persistence/src/commonMain/kotlin/com/wire/kalium/persistence/dao/UserDAO.kt @@ -316,4 +316,5 @@ interface UserDAO { suspend fun getOneOnOnConversationId(userId: UserIDEntity): QualifiedIDEntity? suspend fun getUsersMinimizedByQualifiedIDs(qualifiedIDs: List): List suspend fun getNameAndHandle(userId: UserIDEntity): NameAndHandleEntity? + suspend fun updateTeamId(userId: UserIDEntity, teamId: String) } diff --git a/persistence/src/commonMain/kotlin/com/wire/kalium/persistence/dao/UserDAOImpl.kt b/persistence/src/commonMain/kotlin/com/wire/kalium/persistence/dao/UserDAOImpl.kt index c0c3789d3a0..f601dcc701c 100644 --- a/persistence/src/commonMain/kotlin/com/wire/kalium/persistence/dao/UserDAOImpl.kt +++ b/persistence/src/commonMain/kotlin/com/wire/kalium/persistence/dao/UserDAOImpl.kt @@ -473,4 +473,8 @@ class UserDAOImpl internal constructor( override suspend fun getNameAndHandle(userId: UserIDEntity): NameAndHandleEntity? = withContext(queriesContext) { userQueries.selectNamesAndHandle(userId, ::NameAndHandleEntity).executeAsOneOrNull() } + + override suspend fun updateTeamId(userId: UserIDEntity, teamId: String) { + userQueries.updateTeamId(teamId, userId) + } } From 47b3abe1c333820a6ec2dfe723fe3d6586634329 Mon Sep 17 00:00:00 2001 From: ohassine Date: Fri, 22 Nov 2024 09:11:55 +0100 Subject: [PATCH 2/2] feat: detekt --- .../feature/user/migration/MigrateFromPersonalToTeamUseCase.kt | 1 - 1 file changed, 1 deletion(-) diff --git a/logic/src/commonMain/kotlin/com/wire/kalium/logic/feature/user/migration/MigrateFromPersonalToTeamUseCase.kt b/logic/src/commonMain/kotlin/com/wire/kalium/logic/feature/user/migration/MigrateFromPersonalToTeamUseCase.kt index 9eb493a1b85..d5c356601e9 100644 --- a/logic/src/commonMain/kotlin/com/wire/kalium/logic/feature/user/migration/MigrateFromPersonalToTeamUseCase.kt +++ b/logic/src/commonMain/kotlin/com/wire/kalium/logic/feature/user/migration/MigrateFromPersonalToTeamUseCase.kt @@ -22,7 +22,6 @@ import com.wire.kalium.logic.data.id.TeamId import com.wire.kalium.logic.data.user.UserId import com.wire.kalium.logic.data.user.UserRepository import com.wire.kalium.logic.functional.fold -import com.wire.kalium.logic.functional.map /** * Use case to migrate user personal account to team account.