diff --git a/logic/src/commonMain/kotlin/com/wire/kalium/logic/data/conversation/ConversationGroupRepository.kt b/logic/src/commonMain/kotlin/com/wire/kalium/logic/data/conversation/ConversationGroupRepository.kt index b3e6cc1a030..adbc6fa2021 100644 --- a/logic/src/commonMain/kotlin/com/wire/kalium/logic/data/conversation/ConversationGroupRepository.kt +++ b/logic/src/commonMain/kotlin/com/wire/kalium/logic/data/conversation/ConversationGroupRepository.kt @@ -205,7 +205,8 @@ internal class ConversationGroupRepositoryImpl( eventMapper.conversationMemberJoin( LocalId.generate(), response.event, - true + true, + false ) ) } @@ -244,7 +245,7 @@ internal class ConversationGroupRepositoryImpl( conversationId: ConversationId ) = if (apiResult.value is ConversationMemberAddedResponse.Changed) { memberJoinEventHandler.handle( - eventMapper.conversationMemberJoin(LocalId.generate(), apiResult.value.event, true) + eventMapper.conversationMemberJoin(LocalId.generate(), apiResult.value.event, true, false) ).flatMap { if (failedUsersList.isNotEmpty()) { newGroupConversationSystemMessagesCreator.value.conversationFailedToAddMembers(conversationId, failedUsersList) @@ -319,7 +320,7 @@ internal class ConversationGroupRepositoryImpl( if (response is ConversationMemberAddedResponse.Changed) { val conversationId = response.event.qualifiedConversation.toModel() - memberJoinEventHandler.handle(eventMapper.conversationMemberJoin(LocalId.generate(), response.event, true)) + memberJoinEventHandler.handle(eventMapper.conversationMemberJoin(LocalId.generate(), response.event, true, false)) .flatMap { wrapStorageRequest { conversationDAO.getConversationProtocolInfo(conversationId.toDao()) } .flatMap { protocol -> @@ -367,6 +368,7 @@ internal class ConversationGroupRepositoryImpl( eventMapper.conversationMemberLeave( LocalId.generate(), response.event, + false, false ) ) @@ -406,7 +408,8 @@ internal class ConversationGroupRepositoryImpl( eventMapper.conversationMessageTimerUpdate( LocalId.generate(), it, - true + true, + false ) ) } diff --git a/logic/src/commonMain/kotlin/com/wire/kalium/logic/data/conversation/MLSConversationRepository.kt b/logic/src/commonMain/kotlin/com/wire/kalium/logic/data/conversation/MLSConversationRepository.kt index ccff7691cec..100254cfb85 100644 --- a/logic/src/commonMain/kotlin/com/wire/kalium/logic/data/conversation/MLSConversationRepository.kt +++ b/logic/src/commonMain/kotlin/com/wire/kalium/logic/data/conversation/MLSConversationRepository.kt @@ -326,7 +326,7 @@ internal class MLSConversationDataSource( private suspend fun processCommitBundleEvents(events: List) { events.forEach { eventContentDTO -> - val event = MapperProvider.eventMapper().fromEventContentDTO("", eventContentDTO, true) + val event = MapperProvider.eventMapper().fromEventContentDTO("", eventContentDTO, true, false) if (event is Event.Conversation) { commitBundleEventReceiver.onEvent(event) } diff --git a/logic/src/commonMain/kotlin/com/wire/kalium/logic/data/event/Event.kt b/logic/src/commonMain/kotlin/com/wire/kalium/logic/data/event/Event.kt index c3df023ac29..eeecc51db20 100644 --- a/logic/src/commonMain/kotlin/com/wire/kalium/logic/data/event/Event.kt +++ b/logic/src/commonMain/kotlin/com/wire/kalium/logic/data/event/Event.kt @@ -45,7 +45,7 @@ import com.wire.kalium.util.DateTimeUtil import com.wire.kalium.util.serialization.toJsonElement import kotlinx.serialization.json.JsonNull -sealed class Event(open val id: String, open val transient: Boolean) { +sealed class Event(open val id: String, open val transient: Boolean, open val live: Boolean) { private companion object { const val typeKey = "type" @@ -70,15 +70,17 @@ sealed class Event(open val id: String, open val transient: Boolean) { sealed class Conversation( id: String, override val transient: Boolean, + override val live: Boolean, open val conversationId: ConversationId - ) : Event(id, transient) { + ) : Event(id, transient, live) { data class AccessUpdate( override val id: String, override val conversationId: ConversationId, val data: ConversationResponse, val qualifiedFrom: UserId, - override val transient: Boolean - ) : Conversation(id, transient, conversationId) { + override val transient: Boolean, + override val live: Boolean + ) : Conversation(id, transient, live, conversationId) { override fun toLogMap(): Map = mapOf( typeKey to "Conversation.AccessUpdate", @@ -92,12 +94,13 @@ sealed class Event(open val id: String, open val transient: Boolean) { override val id: String, override val conversationId: ConversationId, override val transient: Boolean, + override val live: Boolean, val senderUserId: UserId, val senderClientId: ClientId, val timestampIso: String, val content: String, val encryptedExternalContent: EncryptedData? - ) : Conversation(id, transient, conversationId) { + ) : Conversation(id, transient, live, conversationId) { override fun toLogMap(): Map = mapOf( typeKey to "Conversation.NewMessage", @@ -113,11 +116,12 @@ sealed class Event(open val id: String, open val transient: Boolean) { override val id: String, override val conversationId: ConversationId, override val transient: Boolean, + override val live: Boolean, val subconversationId: SubconversationId?, val senderUserId: UserId, val timestampIso: String, val content: String - ) : Conversation(id, transient, conversationId) { + ) : Conversation(id, transient, live, conversationId) { override fun toLogMap(): Map = mapOf( typeKey to "Conversation.NewMLSMessage", @@ -132,10 +136,11 @@ sealed class Event(open val id: String, open val transient: Boolean) { override val id: String, override val conversationId: ConversationId, override val transient: Boolean, + override val live: Boolean, val senderUserId: UserId, val timestampIso: String, val conversation: ConversationResponse - ) : Conversation(id, transient, conversationId) { + ) : Conversation(id, transient, live, conversationId) { override fun toLogMap(): Map = mapOf( typeKey to "Conversation.NewConversation", @@ -149,10 +154,11 @@ sealed class Event(open val id: String, open val transient: Boolean) { override val id: String, override val conversationId: ConversationId, override val transient: Boolean, + override val live: Boolean, val addedBy: UserId, val members: List, val timestampIso: String - ) : Conversation(id, transient, conversationId) { + ) : Conversation(id, transient, live, conversationId) { override fun toLogMap(): Map = mapOf( typeKey to "Conversation.MemberJoin", @@ -168,10 +174,11 @@ sealed class Event(open val id: String, open val transient: Boolean) { override val id: String, override val conversationId: ConversationId, override val transient: Boolean, + override val live: Boolean, val removedBy: UserId, val removedList: List, val timestampIso: String - ) : Conversation(id, transient, conversationId) { + ) : Conversation(id, transient, live, conversationId) { override fun toLogMap(): Map = mapOf( typeKey to "Conversation.MemberLeave", @@ -186,15 +193,17 @@ sealed class Event(open val id: String, open val transient: Boolean) { override val id: String, override val conversationId: ConversationId, open val timestampIso: String, - transient: Boolean, - ) : Conversation(id, transient, conversationId) { + override val transient: Boolean, + override val live: Boolean, + ) : Conversation(id, transient, live, conversationId) { class MemberChangedRole( override val id: String, override val conversationId: ConversationId, override val timestampIso: String, override val transient: Boolean, + override val live: Boolean, val member: Member?, - ) : MemberChanged(id, conversationId, timestampIso, transient) { + ) : MemberChanged(id, conversationId, timestampIso, transient, live) { override fun toLogMap(): Map = mapOf( typeKey to "Conversation.MemberChangedRole", @@ -210,9 +219,10 @@ sealed class Event(open val id: String, open val transient: Boolean) { override val conversationId: ConversationId, override val timestampIso: String, override val transient: Boolean, + override val live: Boolean, val mutedConversationStatus: MutedConversationStatus, val mutedConversationChangedTime: String - ) : MemberChanged(id, conversationId, timestampIso, transient) { + ) : MemberChanged(id, conversationId, timestampIso, transient, live) { override fun toLogMap(): Map = mapOf( typeKey to "Conversation.MemberMutedStatusChanged", @@ -229,9 +239,10 @@ sealed class Event(open val id: String, open val transient: Boolean) { override val conversationId: ConversationId, override val timestampIso: String, override val transient: Boolean, + override val live: Boolean, val archivedConversationChangedTime: String, val isArchiving: Boolean - ) : MemberChanged(id, conversationId, timestampIso, transient) { + ) : MemberChanged(id, conversationId, timestampIso, transient, live) { override fun toLogMap(): Map = mapOf( typeKey to "Conversation.MemberArchivedStatusChanged", @@ -246,8 +257,9 @@ sealed class Event(open val id: String, open val transient: Boolean) { data class IgnoredMemberChanged( override val id: String, override val conversationId: ConversationId, - override val transient: Boolean - ) : MemberChanged(id, conversationId, "", transient) { + override val transient: Boolean, + override val live: Boolean, + ) : MemberChanged(id, conversationId, "", transient, live) { override fun toLogMap(): Map = mapOf( typeKey to "Conversation.IgnoredMemberChanged", @@ -261,10 +273,11 @@ sealed class Event(open val id: String, open val transient: Boolean) { override val id: String, override val conversationId: ConversationId, override val transient: Boolean, + override val live: Boolean, val senderUserId: UserId, val message: String, val timestampIso: String = DateTimeUtil.currentIsoDateTimeString() - ) : Conversation(id, transient, conversationId) { + ) : Conversation(id, transient, live, conversationId) { override fun toLogMap(): Map = mapOf( typeKey to "Conversation.MLSWelcome", idKey to id.obfuscateId(), @@ -278,9 +291,10 @@ sealed class Event(open val id: String, open val transient: Boolean) { override val id: String, override val conversationId: ConversationId, override val transient: Boolean, + override val live: Boolean, val senderUserId: UserId, val timestampIso: String, - ) : Conversation(id, transient, conversationId) { + ) : Conversation(id, transient, live, conversationId) { override fun toLogMap(): Map = mapOf( typeKey to "Conversation.DeletedConversation", @@ -295,10 +309,11 @@ sealed class Event(open val id: String, open val transient: Boolean) { override val id: String, override val conversationId: ConversationId, override val transient: Boolean, + override val live: Boolean, val conversationName: String, val senderUserId: UserId, val timestampIso: String, - ) : Conversation(id, transient, conversationId) { + ) : Conversation(id, transient, live, conversationId) { override fun toLogMap(): Map = mapOf( typeKey to "Conversation.RenamedConversation", idKey to id.obfuscateId(), @@ -313,9 +328,10 @@ sealed class Event(open val id: String, open val transient: Boolean) { override val id: String, override val conversationId: ConversationId, override val transient: Boolean, + override val live: Boolean, val receiptMode: ReceiptMode, val senderUserId: UserId - ) : Conversation(id, transient, conversationId) { + ) : Conversation(id, transient, live, conversationId) { override fun toLogMap() = mapOf( typeKey to "Conversation.ConversationReceiptMode", @@ -330,10 +346,11 @@ sealed class Event(open val id: String, open val transient: Boolean) { override val id: String, override val conversationId: ConversationId, override val transient: Boolean, + override val live: Boolean, val messageTimer: Long?, val senderUserId: UserId, val timestampIso: String - ) : Conversation(id, transient, conversationId) { + ) : Conversation(id, transient, live, conversationId) { override fun toLogMap() = mapOf( typeKey to "Conversation.ConversationMessageTimer", @@ -349,11 +366,12 @@ sealed class Event(open val id: String, open val transient: Boolean) { override val id: String, override val conversationId: ConversationId, override val transient: Boolean, + override val live: Boolean, val key: String, val code: String, val uri: String, val isPasswordProtected: Boolean, - ) : Conversation(id, transient, conversationId) { + ) : Conversation(id, transient, live, conversationId) { override fun toLogMap(): Map = mapOf(typeKey to "Conversation.CodeUpdated") } @@ -361,7 +379,8 @@ sealed class Event(open val id: String, open val transient: Boolean) { override val id: String, override val conversationId: ConversationId, override val transient: Boolean, - ) : Conversation(id, transient, conversationId) { + override val live: Boolean, + ) : Conversation(id, transient, live, conversationId) { override fun toLogMap(): Map = mapOf(typeKey to "Conversation.CodeDeleted") } @@ -369,10 +388,11 @@ sealed class Event(open val id: String, open val transient: Boolean) { override val id: String, override val conversationId: ConversationId, override val transient: Boolean, + override val live: Boolean, val senderUserId: UserId, val timestampIso: String, val typingIndicatorMode: TypingIndicatorMode, - ) : Conversation(id, transient, conversationId) { + ) : Conversation(id, transient, live, conversationId) { override fun toLogMap(): Map = mapOf( typeKey to "Conversation.TypingIndicator", conversationIdKey to conversationId.toLogString(), @@ -386,9 +406,10 @@ sealed class Event(open val id: String, open val transient: Boolean) { override val id: String, override val conversationId: ConversationId, override val transient: Boolean, + override val live: Boolean, val protocol: Protocol, val senderUserId: UserId - ) : Conversation(id, transient, conversationId) { + ) : Conversation(id, transient, live, conversationId) { override fun toLogMap() = mapOf( typeKey to "Conversation.ConversationProtocol", idKey to id.obfuscateId(), @@ -403,14 +424,16 @@ sealed class Event(open val id: String, open val transient: Boolean) { id: String, open val teamId: String, transient: Boolean, - ) : Event(id, transient) { + live: Boolean, + ) : Event(id, transient, live) { data class Update( override val id: String, override val transient: Boolean, + override val live: Boolean, override val teamId: String, val icon: String, val name: String, - ) : Team(id, teamId, transient) { + ) : Team(id, teamId, transient, live) { override fun toLogMap(): Map = mapOf( typeKey to "Team.Update", idKey to id.obfuscateId(), @@ -424,8 +447,9 @@ sealed class Event(open val id: String, open val transient: Boolean) { override val id: String, override val teamId: String, override val transient: Boolean, + override val live: Boolean, val memberId: String, - ) : Team(id, teamId, transient) { + ) : Team(id, teamId, transient, live) { override fun toLogMap(): Map = mapOf( typeKey to "Team.MemberJoin", idKey to id.obfuscateId(), @@ -437,10 +461,11 @@ sealed class Event(open val id: String, open val transient: Boolean) { data class MemberLeave( override val id: String, override val transient: Boolean, + override val live: Boolean, override val teamId: String, val memberId: String, val timestampIso: String, - ) : Team(id, teamId, transient) { + ) : Team(id, teamId, transient, live) { override fun toLogMap(): Map = mapOf( typeKey to "Team.MemberLeave", idKey to id.obfuscateId(), @@ -454,9 +479,10 @@ sealed class Event(open val id: String, open val transient: Boolean) { override val id: String, override val teamId: String, override val transient: Boolean, + override val live: Boolean, val memberId: String, val permissionCode: Int?, - ) : Team(id, teamId, transient) { + ) : Team(id, teamId, transient, live) { override fun toLogMap(): Map = mapOf( typeKey to "Team.MemberUpdate", idKey to id.obfuscateId(), @@ -471,12 +497,14 @@ sealed class Event(open val id: String, open val transient: Boolean) { sealed class FeatureConfig( id: String, transient: Boolean, - ) : Event(id, transient) { + live: Boolean, + ) : Event(id, transient, live) { data class FileSharingUpdated( override val id: String, override val transient: Boolean, + override val live: Boolean, val model: ConfigsStatusModel - ) : FeatureConfig(id, transient) { + ) : FeatureConfig(id, transient, live) { override fun toLogMap(): Map = mapOf( typeKey to "FeatureConfig.FileSharingUpdated", idKey to id.obfuscateId(), @@ -487,8 +515,9 @@ sealed class Event(open val id: String, open val transient: Boolean) { data class MLSUpdated( override val id: String, override val transient: Boolean, + override val live: Boolean, val model: MLSModel - ) : FeatureConfig(id, transient) { + ) : FeatureConfig(id, transient, live) { override fun toLogMap(): Map = mapOf( typeKey to "FeatureConfig.MLSUpdated", idKey to id.obfuscateId(), @@ -500,8 +529,9 @@ sealed class Event(open val id: String, open val transient: Boolean) { data class ClassifiedDomainsUpdated( override val id: String, override val transient: Boolean, + override val live: Boolean, val model: ClassifiedDomainsModel, - ) : FeatureConfig(id, transient) { + ) : FeatureConfig(id, transient, live) { override fun toLogMap(): Map = mapOf( typeKey to "FeatureConfig.ClassifiedDomainsUpdated", idKey to id.obfuscateId(), @@ -513,8 +543,9 @@ sealed class Event(open val id: String, open val transient: Boolean) { data class ConferenceCallingUpdated( override val id: String, override val transient: Boolean, + override val live: Boolean, val model: ConferenceCallingModel, - ) : FeatureConfig(id, transient) { + ) : FeatureConfig(id, transient, live) { override fun toLogMap() = mapOf( typeKey to "FeatureConfig.ConferenceCallingUpdated", idKey to id.obfuscateId(), @@ -525,8 +556,9 @@ sealed class Event(open val id: String, open val transient: Boolean) { data class GuestRoomLinkUpdated( override val id: String, override val transient: Boolean, + override val live: Boolean, val model: ConfigsStatusModel, - ) : FeatureConfig(id, transient) { + ) : FeatureConfig(id, transient, live) { override fun toLogMap(): Map = mapOf( typeKey to "FeatureConfig.GuestRoomLinkUpdated", idKey to id.obfuscateId(), @@ -537,8 +569,9 @@ sealed class Event(open val id: String, open val transient: Boolean) { data class SelfDeletingMessagesConfig( override val id: String, override val transient: Boolean, + override val live: Boolean, val model: SelfDeletingMessagesModel, - ) : FeatureConfig(id, transient) { + ) : FeatureConfig(id, transient, live) { override fun toLogMap(): Map = mapOf( typeKey to "FeatureConfig.SelfDeletingMessagesConfig", idKey to id.obfuscateId(), @@ -550,8 +583,9 @@ sealed class Event(open val id: String, open val transient: Boolean) { data class MLSE2EIUpdated( override val id: String, override val transient: Boolean, + override val live: Boolean, val model: E2EIModel - ) : FeatureConfig(id, transient) { + ) : FeatureConfig(id, transient, live) { override fun toLogMap(): Map = mapOf( typeKey to "FeatureConfig.MLSE2EIUpdated", idKey to id.obfuscateId(), @@ -563,7 +597,8 @@ sealed class Event(open val id: String, open val transient: Boolean) { data class UnknownFeatureUpdated( override val id: String, override val transient: Boolean, - ) : FeatureConfig(id, transient) { + override val live: Boolean, + ) : FeatureConfig(id, transient, live) { override fun toLogMap(): Map = mapOf( typeKey to "FeatureConfig.UnknownFeatureUpdated", idKey to id.obfuscateId(), @@ -573,12 +608,14 @@ sealed class Event(open val id: String, open val transient: Boolean) { sealed class User( id: String, - transient: Boolean - ) : Event(id, transient) { + transient: Boolean, + live: Boolean, + ) : Event(id, transient, live) { data class Update( override val id: String, override val transient: Boolean, + override val live: Boolean, val userId: String, val accentId: Int?, val ssoIdDeleted: Boolean?, @@ -588,7 +625,7 @@ sealed class Event(open val id: String, open val transient: Boolean) { val previewAssetId: String?, val completeAssetId: String?, val supportedProtocols: Set? - ) : User(id, transient) { + ) : User(id, transient, live) { override fun toLogMap(): Map = mapOf( typeKey to "User.Update", idKey to id.obfuscateId(), @@ -598,9 +635,10 @@ sealed class Event(open val id: String, open val transient: Boolean) { data class NewConnection( override val transient: Boolean, + override val live: Boolean, override val id: String, val connection: Connection - ) : User(id, transient) { + ) : User(id, transient, live) { override fun toLogMap(): Map = mapOf( typeKey to "User.NewConnection", idKey to id.obfuscateId(), @@ -610,9 +648,10 @@ sealed class Event(open val id: String, open val transient: Boolean) { data class ClientRemove( override val transient: Boolean, + override val live: Boolean, override val id: String, val clientId: ClientId - ) : User(id, transient) { + ) : User(id, transient, live) { override fun toLogMap(): Map = mapOf( typeKey to "User.ClientRemove", idKey to id.obfuscateId(), @@ -622,10 +661,11 @@ sealed class Event(open val id: String, open val transient: Boolean) { data class UserDelete( override val transient: Boolean, + override val live: Boolean, override val id: String, val userId: UserId, val timestampIso: String = DateTimeUtil.currentIsoDateTimeString() // TODO we are not receiving it from API - ) : User(id, transient) { + ) : User(id, transient, live) { override fun toLogMap(): Map = mapOf( typeKey to "User.UserDelete", idKey to id.obfuscateId(), @@ -636,9 +676,10 @@ sealed class Event(open val id: String, open val transient: Boolean) { data class NewClient( override val transient: Boolean, + override val live: Boolean, override val id: String, val client: Client, - ) : User(id, transient) { + ) : User(id, transient, live) { override fun toLogMap(): Map = mapOf( typeKey to "User.NewClient", idKey to id.obfuscateId(), @@ -655,14 +696,16 @@ sealed class Event(open val id: String, open val transient: Boolean) { sealed class UserProperty( id: String, - transient: Boolean - ) : Event(id, transient) { + transient: Boolean, + live: Boolean, + ) : Event(id, transient, live) { data class ReadReceiptModeSet( override val id: String, override val transient: Boolean, + override val live: Boolean, val value: Boolean, - ) : UserProperty(id, transient) { + ) : UserProperty(id, transient, live) { override fun toLogMap(): Map = mapOf( typeKey to "User.UserProperty.ReadReceiptModeSet", idKey to id.obfuscateId(), @@ -674,8 +717,9 @@ sealed class Event(open val id: String, open val transient: Boolean) { data class TypingIndicatorModeSet( override val id: String, override val transient: Boolean, + override val live: Boolean, val value: Boolean, - ) : UserProperty(id, transient) { + ) : UserProperty(id, transient, live) { override fun toLogMap(): Map = mapOf( typeKey to "User.UserProperty.TypingIndicatorModeSet", idKey to id.obfuscateId(), @@ -688,9 +732,10 @@ sealed class Event(open val id: String, open val transient: Boolean) { data class Unknown( override val id: String, override val transient: Boolean, + override val live: Boolean, val unknownType: String, val cause: String? = null - ) : Event(id, transient) { + ) : Event(id, transient, live) { override fun toLogMap(): Map = mapOf( typeKey to "User.UnknownEvent", idKey to id.obfuscateId(), @@ -702,13 +747,15 @@ sealed class Event(open val id: String, open val transient: Boolean) { sealed class Federation( id: String, override val transient: Boolean, - ) : Event(id, transient) { + override val live: Boolean, + ) : Event(id, transient, live) { data class Delete( override val id: String, override val transient: Boolean, + override val live: Boolean, val domain: String, - ) : Federation(id, transient) { + ) : Federation(id, transient, live) { override fun toLogMap(): Map = mapOf( typeKey to "Federation.Delete", idKey to id.obfuscateId(), @@ -720,8 +767,9 @@ sealed class Event(open val id: String, open val transient: Boolean) { data class ConnectionRemoved( override val id: String, override val transient: Boolean, + override val live: Boolean, val domains: List, - ) : Federation(id, transient) { + ) : Federation(id, transient, live) { override fun toLogMap(): Map = mapOf( typeKey to "Federation.ConnectionRemoved", idKey to id.obfuscateId(), diff --git a/logic/src/commonMain/kotlin/com/wire/kalium/logic/data/event/EventMapper.kt b/logic/src/commonMain/kotlin/com/wire/kalium/logic/data/event/EventMapper.kt index 45e2b2fb70b..90f32003c7f 100644 --- a/logic/src/commonMain/kotlin/com/wire/kalium/logic/data/event/EventMapper.kt +++ b/logic/src/commonMain/kotlin/com/wire/kalium/logic/data/event/EventMapper.kt @@ -60,61 +60,60 @@ class EventMapper( private val receiptModeMapper: ReceiptModeMapper = MapperProvider.receiptModeMapper(), private val clientMapper: ClientMapper = MapperProvider.clientMapper() ) { - fun fromDTO(eventResponse: EventResponse): List { + fun fromDTO(eventResponse: EventResponse, live: Boolean = false): List { // TODO(edge-case): Multiple payloads in the same event have the same ID, is this an issue when marking lastProcessedEventId? val id = eventResponse.id return eventResponse.payload?.map { eventContentDTO -> - fromEventContentDTO(id, eventContentDTO, eventResponse.transient) + fromEventContentDTO(id, eventContentDTO, eventResponse.transient, live) } ?: listOf() } @Suppress("ComplexMethod") - fun fromEventContentDTO(id: String, eventContentDTO: EventContentDTO, transient: Boolean): Event = + fun fromEventContentDTO(id: String, eventContentDTO: EventContentDTO, transient: Boolean, live: Boolean): Event = when (eventContentDTO) { - is EventContentDTO.Conversation.NewMessageDTO -> newMessage(id, eventContentDTO, transient) - is EventContentDTO.Conversation.NewConversationDTO -> newConversation(id, eventContentDTO, transient) - is EventContentDTO.Conversation.MemberJoinDTO -> conversationMemberJoin(id, eventContentDTO, transient) - is EventContentDTO.Conversation.MemberLeaveDTO -> conversationMemberLeave(id, eventContentDTO, transient) - is EventContentDTO.Conversation.MemberUpdateDTO -> memberUpdate(id, eventContentDTO, transient) - is EventContentDTO.Conversation.MLSWelcomeDTO -> welcomeMessage(id, eventContentDTO, transient) - is EventContentDTO.Conversation.NewMLSMessageDTO -> newMLSMessage(id, eventContentDTO, transient) - is EventContentDTO.User.NewConnectionDTO -> connectionUpdate(id, eventContentDTO, transient) - is EventContentDTO.User.ClientRemoveDTO -> clientRemove(id, eventContentDTO, transient) - is EventContentDTO.User.UserDeleteDTO -> userDelete(id, eventContentDTO, transient) - is EventContentDTO.FeatureConfig.FeatureConfigUpdatedDTO -> featureConfig(id, eventContentDTO, transient) - is EventContentDTO.User.NewClientDTO -> newClient(id, eventContentDTO, transient) - is EventContentDTO.Unknown -> unknown(id, transient, eventContentDTO) - is EventContentDTO.Conversation.AccessUpdate -> unknown(id, transient, eventContentDTO) - is EventContentDTO.Conversation.DeletedConversationDTO -> conversationDeleted(id, eventContentDTO, transient) - - is EventContentDTO.Conversation.ConversationRenameDTO -> conversationRenamed(id, eventContentDTO, transient) - is EventContentDTO.Team.MemberJoin -> teamMemberJoined(id, eventContentDTO, transient) - is EventContentDTO.Team.MemberLeave -> teamMemberLeft(id, eventContentDTO, transient) - is EventContentDTO.Team.MemberUpdate -> teamMemberUpdate(id, eventContentDTO, transient) - is EventContentDTO.Team.Update -> teamUpdate(id, eventContentDTO, transient) - is EventContentDTO.User.UpdateDTO -> userUpdate(id, eventContentDTO, transient) - is EventContentDTO.UserProperty.PropertiesSetDTO -> updateUserProperties(id, eventContentDTO, transient) - is EventContentDTO.UserProperty.PropertiesDeleteDTO -> deleteUserProperties(id, eventContentDTO, transient) - is EventContentDTO.Conversation.ReceiptModeUpdate -> conversationReceiptModeUpdate(id, eventContentDTO, transient) - - is EventContentDTO.Conversation.MessageTimerUpdate -> conversationMessageTimerUpdate(id, eventContentDTO, transient) - - is EventContentDTO.Conversation.CodeDeleted -> conversationCodeDeleted(id, eventContentDTO, transient) - is EventContentDTO.Conversation.CodeUpdated -> conversationCodeUpdated(id, eventContentDTO, transient) - is EventContentDTO.Federation -> federationTerminated(id, eventContentDTO, transient) - is EventContentDTO.Conversation.ConversationTypingDTO -> conversationTyping(id, eventContentDTO, transient) - is EventContentDTO.Conversation.ProtocolUpdate -> conversationProtocolUpdate(id, eventContentDTO, transient) + is EventContentDTO.Conversation.NewMessageDTO -> newMessage(id, eventContentDTO, transient, live) + is EventContentDTO.Conversation.NewConversationDTO -> newConversation(id, eventContentDTO, transient, live) + is EventContentDTO.Conversation.MemberJoinDTO -> conversationMemberJoin(id, eventContentDTO, transient, live) + is EventContentDTO.Conversation.MemberLeaveDTO -> conversationMemberLeave(id, eventContentDTO, transient, live) + is EventContentDTO.Conversation.MemberUpdateDTO -> memberUpdate(id, eventContentDTO, transient, live) + is EventContentDTO.Conversation.MLSWelcomeDTO -> welcomeMessage(id, eventContentDTO, transient, live) + is EventContentDTO.Conversation.NewMLSMessageDTO -> newMLSMessage(id, eventContentDTO, transient, live) + is EventContentDTO.User.NewConnectionDTO -> connectionUpdate(id, eventContentDTO, transient, live) + is EventContentDTO.User.ClientRemoveDTO -> clientRemove(id, eventContentDTO, transient, live) + is EventContentDTO.User.UserDeleteDTO -> userDelete(id, eventContentDTO, transient, live) + is EventContentDTO.FeatureConfig.FeatureConfigUpdatedDTO -> featureConfig(id, eventContentDTO, transient, live) + is EventContentDTO.User.NewClientDTO -> newClient(id, eventContentDTO, transient, live) + is EventContentDTO.Unknown -> unknown(id, transient, live, eventContentDTO) + is EventContentDTO.Conversation.AccessUpdate -> unknown(id, transient, live, eventContentDTO) + is EventContentDTO.Conversation.DeletedConversationDTO -> conversationDeleted(id, eventContentDTO, transient, live) + is EventContentDTO.Conversation.ConversationRenameDTO -> conversationRenamed(id, eventContentDTO, transient, live) + is EventContentDTO.Team.MemberJoin -> teamMemberJoined(id, eventContentDTO, transient, live) + is EventContentDTO.Team.MemberLeave -> teamMemberLeft(id, eventContentDTO, transient, live) + is EventContentDTO.Team.MemberUpdate -> teamMemberUpdate(id, eventContentDTO, transient, live) + is EventContentDTO.Team.Update -> teamUpdate(id, eventContentDTO, transient, live) + is EventContentDTO.User.UpdateDTO -> userUpdate(id, eventContentDTO, transient, live) + is EventContentDTO.UserProperty.PropertiesSetDTO -> updateUserProperties(id, eventContentDTO, transient, live) + is EventContentDTO.UserProperty.PropertiesDeleteDTO -> deleteUserProperties(id, eventContentDTO, transient, live) + is EventContentDTO.Conversation.ReceiptModeUpdate -> conversationReceiptModeUpdate(id, eventContentDTO, transient, live) + is EventContentDTO.Conversation.MessageTimerUpdate -> conversationMessageTimerUpdate(id, eventContentDTO, transient, live) + is EventContentDTO.Conversation.CodeDeleted -> conversationCodeDeleted(id, eventContentDTO, transient, live) + is EventContentDTO.Conversation.CodeUpdated -> conversationCodeUpdated(id, eventContentDTO, transient, live) + is EventContentDTO.Federation -> federationTerminated(id, eventContentDTO, transient, live) + is EventContentDTO.Conversation.ConversationTypingDTO -> conversationTyping(id, eventContentDTO, transient, live) + is EventContentDTO.Conversation.ProtocolUpdate -> conversationProtocolUpdate(id, eventContentDTO, transient, live) } private fun conversationTyping( id: String, eventContentDTO: EventContentDTO.Conversation.ConversationTypingDTO, - transient: Boolean + transient: Boolean, + live: Boolean ): Event = Event.Conversation.TypingIndicator( id, eventContentDTO.qualifiedConversation.toModel(), transient, + live, eventContentDTO.qualifiedFrom.toModel(), eventContentDTO.time, when (eventContentDTO.status.status) { @@ -123,17 +122,19 @@ class EventMapper( } ) - private fun federationTerminated(id: String, eventContentDTO: EventContentDTO.Federation, transient: Boolean): Event = + private fun federationTerminated(id: String, eventContentDTO: EventContentDTO.Federation, transient: Boolean, live: Boolean): Event = when (eventContentDTO) { is EventContentDTO.Federation.FederationConnectionRemovedDTO -> Event.Federation.ConnectionRemoved( id, transient, + live, eventContentDTO.domains ) is EventContentDTO.Federation.FederationDeleteDTO -> Event.Federation.Delete( id, transient, + live, eventContentDTO.domain ) } @@ -141,17 +142,20 @@ class EventMapper( private fun conversationCodeDeleted( id: String, event: EventContentDTO.Conversation.CodeDeleted, - transient: Boolean + transient: Boolean, + live: Boolean ): Event.Conversation.CodeDeleted = Event.Conversation.CodeDeleted( id = id, transient = transient, + live = live, conversationId = event.qualifiedConversation.toModel() ) private fun conversationCodeUpdated( id: String, event: EventContentDTO.Conversation.CodeUpdated, - transient: Boolean + transient: Boolean, + live: Boolean ): Event.Conversation.CodeUpdated = Event.Conversation.CodeUpdated( id = id, key = event.data.key, @@ -159,18 +163,21 @@ class EventMapper( uri = event.data.uri, isPasswordProtected = event.data.hasPassword, conversationId = event.qualifiedConversation.toModel(), - transient = transient + transient = transient, + live = live, ) @OptIn(InternalSerializationApi::class, ExperimentalSerializationApi::class) fun unknown( id: String, transient: Boolean, + live: Boolean, eventContentDTO: EventContentDTO, cause: String? = null ): Event.Unknown = Event.Unknown( id = id, transient = transient, + live = live, unknownType = when (eventContentDTO) { is EventContentDTO.Unknown -> eventContentDTO.type else -> try { @@ -185,11 +192,13 @@ class EventMapper( private fun conversationProtocolUpdate( id: String, eventContentDTO: EventContentDTO.Conversation.ProtocolUpdate, - transient: Boolean + transient: Boolean, + live: Boolean ): Event = Event.Conversation.ConversationProtocol( id = id, conversationId = eventContentDTO.qualifiedConversation.toModel(), transient = transient, + live = live, protocol = eventContentDTO.data.protocol.toModel(), senderUserId = eventContentDTO.qualifiedFrom.toModel() ) @@ -197,11 +206,13 @@ class EventMapper( fun conversationMessageTimerUpdate( id: String, eventContentDTO: EventContentDTO.Conversation.MessageTimerUpdate, - transient: Boolean + transient: Boolean, + live: Boolean ) = Event.Conversation.ConversationMessageTimer( id = id, conversationId = eventContentDTO.qualifiedConversation.toModel(), transient = transient, + live = live, messageTimer = eventContentDTO.data.messageTimer, senderUserId = eventContentDTO.qualifiedFrom.toModel(), timestampIso = eventContentDTO.time @@ -210,11 +221,13 @@ class EventMapper( private fun conversationReceiptModeUpdate( id: String, eventContentDTO: EventContentDTO.Conversation.ReceiptModeUpdate, - transient: Boolean + transient: Boolean, + live: Boolean ): Event = Event.Conversation.ConversationReceiptMode( id = id, conversationId = eventContentDTO.qualifiedConversation.toModel(), transient = transient, + live = live, receiptMode = receiptModeMapper.fromApiToModel(eventContentDTO.data.receiptMode), senderUserId = eventContentDTO.qualifiedFrom.toModel() ) @@ -222,7 +235,8 @@ class EventMapper( private fun updateUserProperties( id: String, eventContentDTO: EventContentDTO.UserProperty.PropertiesSetDTO, - transient: Boolean + transient: Boolean, + live: Boolean ): Event { val fieldKeyValue = eventContentDTO.value val key = eventContentDTO.key @@ -232,18 +246,21 @@ class EventMapper( WIRE_RECEIPT_MODE.key -> ReadReceiptModeSet( id, transient, + live, fieldKeyValue.value == 1 ) WIRE_TYPING_INDICATOR_MODE.key -> TypingIndicatorModeSet( id, transient, + live, fieldKeyValue.value != 0 ) else -> unknown( id = id, transient = transient, + live = live, eventContentDTO = eventContentDTO, cause = "Unknown key: $key " ) @@ -252,6 +269,7 @@ class EventMapper( else -> unknown( id = id, transient = transient, + live = live, eventContentDTO = eventContentDTO, cause = "Unknown value type for key: ${eventContentDTO.key} " ) @@ -261,14 +279,16 @@ class EventMapper( private fun deleteUserProperties( id: String, eventContentDTO: EventContentDTO.UserProperty.PropertiesDeleteDTO, - transient: Boolean + transient: Boolean, + live: Boolean ): Event { return when (eventContentDTO.key) { - WIRE_RECEIPT_MODE.key -> ReadReceiptModeSet(id, transient, false) - WIRE_TYPING_INDICATOR_MODE.key -> TypingIndicatorModeSet(id, transient, true) + WIRE_RECEIPT_MODE.key -> ReadReceiptModeSet(id, transient, live, false) + WIRE_TYPING_INDICATOR_MODE.key -> TypingIndicatorModeSet(id, transient, live, true) else -> unknown( id = id, transient = transient, + live = live, eventContentDTO = eventContentDTO, cause = "Unknown key: ${eventContentDTO.key} " ) @@ -278,11 +298,13 @@ class EventMapper( private fun welcomeMessage( id: String, eventContentDTO: EventContentDTO.Conversation.MLSWelcomeDTO, - transient: Boolean + transient: Boolean, + live: Boolean ) = Event.Conversation.MLSWelcome( id, eventContentDTO.qualifiedConversation.toModel(), transient, + live, eventContentDTO.qualifiedFrom.toModel(), eventContentDTO.message, ) @@ -290,11 +312,13 @@ class EventMapper( private fun newMessage( id: String, eventContentDTO: EventContentDTO.Conversation.NewMessageDTO, - transient: Boolean + transient: Boolean, + live: Boolean ) = Event.Conversation.NewMessage( id, eventContentDTO.qualifiedConversation.toModel(), transient, + live, eventContentDTO.qualifiedFrom.toModel(), ClientId(eventContentDTO.data.sender), eventContentDTO.time, @@ -307,11 +331,13 @@ class EventMapper( private fun newMLSMessage( id: String, eventContentDTO: EventContentDTO.Conversation.NewMLSMessageDTO, - transient: Boolean + transient: Boolean, + live: Boolean ) = Event.Conversation.NewMLSMessage( id, eventContentDTO.qualifiedConversation.toModel(), transient, + live, eventContentDTO.subconversation?.let { SubconversationId(it) }, eventContentDTO.qualifiedFrom.toModel(), eventContentDTO.time, @@ -321,32 +347,42 @@ class EventMapper( private fun connectionUpdate( id: String, eventConnectionDTO: EventContentDTO.User.NewConnectionDTO, - transient: Boolean + transient: Boolean, + live: Boolean ) = Event.User.NewConnection( transient, + live, id, connectionMapper.fromApiToModel(eventConnectionDTO.connection) ) - private fun userDelete(id: String, eventUserDelete: EventContentDTO.User.UserDeleteDTO, transient: Boolean): Event.User.UserDelete { - return Event.User.UserDelete(transient, id, eventUserDelete.userId.toModel()) + private fun userDelete( + id: String, + eventUserDelete: EventContentDTO.User.UserDeleteDTO, + transient: Boolean, + live: Boolean + ): Event.User.UserDelete { + return Event.User.UserDelete(transient, live, id, eventUserDelete.userId.toModel()) } private fun clientRemove( id: String, eventClientRemove: EventContentDTO.User.ClientRemoveDTO, - transient: Boolean + transient: Boolean, + live: Boolean ): Event.User.ClientRemove { - return Event.User.ClientRemove(transient, id, ClientId(eventClientRemove.client.clientId)) + return Event.User.ClientRemove(transient, live, id, ClientId(eventClientRemove.client.clientId)) } private fun newClient( id: String, eventNewClient: EventContentDTO.User.NewClientDTO, - transient: Boolean + transient: Boolean, + live: Boolean ): Event.User.NewClient { return Event.User.NewClient( transient = transient, + live = live, id = id, client = clientMapper.fromClientDto(eventNewClient.client) ) @@ -355,11 +391,13 @@ class EventMapper( private fun newConversation( id: String, eventContentDTO: EventContentDTO.Conversation.NewConversationDTO, - transient: Boolean + transient: Boolean, + live: Boolean ) = Event.Conversation.NewConversation( id, eventContentDTO.qualifiedConversation.toModel(), transient, + live, eventContentDTO.qualifiedFrom.toModel(), eventContentDTO.time, eventContentDTO.data @@ -368,33 +406,38 @@ class EventMapper( fun conversationMemberJoin( id: String, eventContentDTO: EventContentDTO.Conversation.MemberJoinDTO, - transient: Boolean + transient: Boolean, + live: Boolean ) = Event.Conversation.MemberJoin( id = id, conversationId = eventContentDTO.qualifiedConversation.toModel(), addedBy = eventContentDTO.qualifiedFrom.toModel(), members = eventContentDTO.members.users.map { memberMapper.fromApiModel(it) }, timestampIso = eventContentDTO.time, - transient = transient + transient = transient, + live = live, ) fun conversationMemberLeave( id: String, eventContentDTO: EventContentDTO.Conversation.MemberLeaveDTO, - transient: Boolean + transient: Boolean, + live: Boolean ) = Event.Conversation.MemberLeave( id = id, conversationId = eventContentDTO.qualifiedConversation.toModel(), removedBy = eventContentDTO.qualifiedFrom.toModel(), removedList = eventContentDTO.members.qualifiedUserIds.map { it.toModel() }, timestampIso = eventContentDTO.time, - transient = transient + transient = transient, + live = live, ) private fun memberUpdate( id: String, eventContentDTO: EventContentDTO.Conversation.MemberUpdateDTO, - transient: Boolean + transient: Boolean, + live: Boolean ): Event.Conversation.MemberChanged { return when { eventContentDTO.roleChange.role?.isNotEmpty() == true -> { @@ -403,6 +446,7 @@ class EventMapper( conversationId = eventContentDTO.qualifiedConversation.toModel(), timestampIso = eventContentDTO.time, transient = transient, + live = live, member = Conversation.Member( id = eventContentDTO.roleChange.qualifiedUserId.toModel(), role = roleMapper.fromApi(eventContentDTO.roleChange.role.orEmpty()) @@ -417,6 +461,7 @@ class EventMapper( timestampIso = eventContentDTO.time, mutedConversationChangedTime = eventContentDTO.roleChange.mutedRef.orEmpty(), transient = transient, + live = live, mutedConversationStatus = mapConversationMutedStatus(eventContentDTO.roleChange.mutedStatus) ) } @@ -427,6 +472,7 @@ class EventMapper( conversationId = eventContentDTO.qualifiedConversation.toModel(), timestampIso = eventContentDTO.time, transient = transient, + live = live, archivedConversationChangedTime = eventContentDTO.roleChange.archivedRef.orEmpty(), isArchiving = eventContentDTO.roleChange.isArchiving ?: false ) @@ -436,7 +482,8 @@ class EventMapper( Event.Conversation.MemberChanged.IgnoredMemberChanged( id, eventContentDTO.qualifiedConversation.toModel(), - transient + transient, + live ) } } @@ -453,129 +500,150 @@ class EventMapper( private fun featureConfig( id: String, featureConfigUpdatedDTO: EventContentDTO.FeatureConfig.FeatureConfigUpdatedDTO, - transient: Boolean + transient: Boolean, + live: Boolean ) = when (featureConfigUpdatedDTO.data) { is FeatureConfigData.FileSharing -> Event.FeatureConfig.FileSharingUpdated( id, transient, + live, featureConfigMapper.fromDTO(featureConfigUpdatedDTO.data as FeatureConfigData.FileSharing) ) is FeatureConfigData.SelfDeletingMessages -> Event.FeatureConfig.SelfDeletingMessagesConfig( id, transient, + live, featureConfigMapper.fromDTO(featureConfigUpdatedDTO.data as FeatureConfigData.SelfDeletingMessages) ) is FeatureConfigData.MLS -> Event.FeatureConfig.MLSUpdated( id, transient, + live, featureConfigMapper.fromDTO(featureConfigUpdatedDTO.data as FeatureConfigData.MLS) ) is FeatureConfigData.ClassifiedDomains -> Event.FeatureConfig.ClassifiedDomainsUpdated( id, transient, + live, featureConfigMapper.fromDTO(featureConfigUpdatedDTO.data as FeatureConfigData.ClassifiedDomains) ) is FeatureConfigData.ConferenceCalling -> Event.FeatureConfig.ConferenceCallingUpdated( id, transient, + live, featureConfigMapper.fromDTO(featureConfigUpdatedDTO.data as FeatureConfigData.ConferenceCalling) ) is FeatureConfigData.ConversationGuestLinks -> Event.FeatureConfig.GuestRoomLinkUpdated( id, transient, + live, featureConfigMapper.fromDTO(featureConfigUpdatedDTO.data as FeatureConfigData.ConversationGuestLinks) ) is FeatureConfigData.E2EI -> Event.FeatureConfig.MLSE2EIUpdated( id, transient, + live, featureConfigMapper.fromDTO(featureConfigUpdatedDTO.data as FeatureConfigData.E2EI) ) - else -> Event.FeatureConfig.UnknownFeatureUpdated(id, transient) + else -> Event.FeatureConfig.UnknownFeatureUpdated(id, transient, live) } private fun conversationDeleted( id: String, deletedConversationDTO: EventContentDTO.Conversation.DeletedConversationDTO, - transient: Boolean + transient: Boolean, + live: Boolean ) = Event.Conversation.DeletedConversation( id = id, conversationId = deletedConversationDTO.qualifiedConversation.toModel(), senderUserId = deletedConversationDTO.qualifiedFrom.toModel(), transient = transient, + live = live, timestampIso = deletedConversationDTO.time ) fun conversationRenamed( id: String, event: EventContentDTO.Conversation.ConversationRenameDTO, - transient: Boolean + transient: Boolean, + live: Boolean ) = Event.Conversation.RenamedConversation( id = id, conversationId = event.qualifiedConversation.toModel(), senderUserId = event.qualifiedFrom.toModel(), conversationName = event.updateNameData.conversationName, transient = transient, + live = live, timestampIso = event.time, ) private fun teamMemberJoined( id: String, event: EventContentDTO.Team.MemberJoin, - transient: Boolean + transient: Boolean, + live: Boolean ) = Event.Team.MemberJoin( id = id, teamId = event.teamId, transient = transient, + live = live, memberId = event.teamMember.nonQualifiedUserId ) private fun teamMemberLeft( id: String, event: EventContentDTO.Team.MemberLeave, - transient: Boolean + transient: Boolean, + live: Boolean ) = Event.Team.MemberLeave( id = id, teamId = event.teamId, memberId = event.teamMember.nonQualifiedUserId, transient = transient, + live = live, timestampIso = event.time ) private fun teamMemberUpdate( id: String, event: EventContentDTO.Team.MemberUpdate, - transient: Boolean + transient: Boolean, + live: Boolean ) = Event.Team.MemberUpdate( id = id, teamId = event.teamId, memberId = event.permissionsResponse.nonQualifiedUserId, transient = transient, + live = live, permissionCode = event.permissionsResponse.permissions.own ) private fun teamUpdate( id: String, event: EventContentDTO.Team.Update, - transient: Boolean + transient: Boolean, + live: Boolean ) = Event.Team.Update( id = id, teamId = event.teamId, icon = event.teamUpdate.icon, transient = transient, + live = live, name = event.teamUpdate.name ) private fun userUpdate( id: String, event: EventContentDTO.User.UpdateDTO, - transient: Boolean + transient: Boolean, + live: Boolean ) = Event.User.Update( id = id, userId = event.userData.nonQualifiedUserId, @@ -586,6 +654,7 @@ class EventMapper( email = event.userData.email, previewAssetId = event.userData.assets?.getPreviewAssetOrNull()?.key, transient = transient, + live = live, completeAssetId = event.userData.assets?.getCompleteAssetOrNull()?.key, supportedProtocols = event.userData.supportedProtocols?.toModel() ) diff --git a/logic/src/commonMain/kotlin/com/wire/kalium/logic/data/event/EventRepository.kt b/logic/src/commonMain/kotlin/com/wire/kalium/logic/data/event/EventRepository.kt index 655199e9d80..f94d546a33f 100644 --- a/logic/src/commonMain/kotlin/com/wire/kalium/logic/data/event/EventRepository.kt +++ b/logic/src/commonMain/kotlin/com/wire/kalium/logic/data/event/EventRepository.kt @@ -108,7 +108,7 @@ class EventDataSource( } is WebSocketEvent.BinaryPayloadReceived -> { - eventMapper.fromDTO(webSocketEvent.payload).asFlow().map { WebSocketEvent.BinaryPayloadReceived(it) } + eventMapper.fromDTO(webSocketEvent.payload, true).asFlow().map { WebSocketEvent.BinaryPayloadReceived(it) } } } }.flattenConcat() 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 0cb4b6b4abc..0ab25813028 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 @@ -898,7 +898,8 @@ class UserSessionScope internal constructor( get() = OneOnOneResolverImpl( userRepository, oneOnOneProtocolSelector, - oneOnOneMigrator + oneOnOneMigrator, + incrementalSyncRepository ) private val slowSyncWorker: SlowSyncWorker by lazy { diff --git a/logic/src/commonMain/kotlin/com/wire/kalium/logic/feature/conversation/RenameConversationUseCase.kt b/logic/src/commonMain/kotlin/com/wire/kalium/logic/feature/conversation/RenameConversationUseCase.kt index 4d5f71a3377..4ec5e031b19 100644 --- a/logic/src/commonMain/kotlin/com/wire/kalium/logic/feature/conversation/RenameConversationUseCase.kt +++ b/logic/src/commonMain/kotlin/com/wire/kalium/logic/feature/conversation/RenameConversationUseCase.kt @@ -54,7 +54,7 @@ internal class RenameConversationUseCaseImpl( .onSuccess { response -> if (response is ConversationRenameResponse.Changed) renamedConversationEventHandler.handle( - eventMapper.conversationRenamed(LocalId.generate(), response.event, true) + eventMapper.conversationRenamed(LocalId.generate(), response.event, true, false) ) } .fold({ diff --git a/logic/src/commonMain/kotlin/com/wire/kalium/logic/feature/conversation/guestroomlink/GenerateGuestRoomLinkUseCase.kt b/logic/src/commonMain/kotlin/com/wire/kalium/logic/feature/conversation/guestroomlink/GenerateGuestRoomLinkUseCase.kt index 9e482f86f34..42a6e7a77b4 100644 --- a/logic/src/commonMain/kotlin/com/wire/kalium/logic/feature/conversation/guestroomlink/GenerateGuestRoomLinkUseCase.kt +++ b/logic/src/commonMain/kotlin/com/wire/kalium/logic/feature/conversation/guestroomlink/GenerateGuestRoomLinkUseCase.kt @@ -51,6 +51,7 @@ class GenerateGuestRoomLinkUseCaseImpl internal constructor( id = uuid4().toString(), isPasswordProtected = it.data.hasPassword, transient = false, + live = false, key = it.data.key, uri = it.data.uri ) 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 790ebd7b088..f6a899cb78e 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 @@ -20,6 +20,8 @@ package com.wire.kalium.logic.feature.conversation.mls import com.wire.kalium.logic.CoreFailure import com.wire.kalium.logic.StorageFailure import com.wire.kalium.logic.data.id.ConversationId +import com.wire.kalium.logic.data.sync.IncrementalSyncRepository +import com.wire.kalium.logic.data.sync.IncrementalSyncStatus import com.wire.kalium.logic.data.user.OtherUser import com.wire.kalium.logic.data.user.SupportedProtocol import com.wire.kalium.logic.data.user.UserId @@ -31,10 +33,20 @@ import com.wire.kalium.logic.functional.flatMapLeft import com.wire.kalium.logic.functional.foldToEitherWhileRight import com.wire.kalium.logic.functional.map import com.wire.kalium.logic.kaliumLogger +import com.wire.kalium.util.KaliumDispatcher +import com.wire.kalium.util.KaliumDispatcherImpl +import kotlinx.coroutines.CoroutineScope +import kotlinx.coroutines.ExperimentalCoroutinesApi +import kotlinx.coroutines.Job +import kotlinx.coroutines.delay +import kotlinx.coroutines.flow.first import kotlinx.coroutines.flow.firstOrNull +import kotlinx.coroutines.launch +import kotlin.time.Duration interface OneOnOneResolver { suspend fun resolveAllOneOnOneConversations(): Either + suspend fun scheduleResolveOneOnOneConversationWithUserId(userId: UserId, delay: Duration = Duration.ZERO): Job suspend fun resolveOneOnOneConversationWithUserId(userId: UserId): Either suspend fun resolveOneOnOneConversationWithUser(user: OtherUser): Either } @@ -43,8 +55,14 @@ internal class OneOnOneResolverImpl( private val userRepository: UserRepository, private val oneOnOneProtocolSelector: OneOnOneProtocolSelector, private val oneOnOneMigrator: OneOnOneMigrator, + private val incrementalSyncRepository: IncrementalSyncRepository, + kaliumDispatcher: KaliumDispatcher = KaliumDispatcherImpl ) : OneOnOneResolver { + @OptIn(ExperimentalCoroutinesApi::class) + private val dispatcher = kaliumDispatcher.default.limitedParallelism(1) + private val resolveActiveOneOnOneScope = CoroutineScope(dispatcher) + override suspend fun resolveAllOneOnOneConversations(): Either { val usersWithOneOnOne = userRepository.getUsersWithOneOnOneConversation() kaliumLogger.i("Resolving one-on-one protocol for ${usersWithOneOnOne.size} user(s)") @@ -53,6 +71,14 @@ internal class OneOnOneResolverImpl( } } + override suspend fun scheduleResolveOneOnOneConversationWithUserId(userId: UserId, delay: Duration) = + resolveActiveOneOnOneScope.launch { + kaliumLogger.d("Schedule resolving active one-on-one") + incrementalSyncRepository.incrementalSyncState.first { it is IncrementalSyncStatus.Live } + delay(delay) + resolveOneOnOneConversationWithUserId(userId) + } + override suspend fun resolveOneOnOneConversationWithUserId(userId: UserId): Either = userRepository.getKnownUser(userId).firstOrNull()?.let { resolveOneOnOneConversationWithUser(it) diff --git a/logic/src/commonMain/kotlin/com/wire/kalium/logic/sync/receiver/UserEventReceiver.kt b/logic/src/commonMain/kotlin/com/wire/kalium/logic/sync/receiver/UserEventReceiver.kt index ba891d031da..7d419c834fc 100644 --- a/logic/src/commonMain/kotlin/com/wire/kalium/logic/sync/receiver/UserEventReceiver.kt +++ b/logic/src/commonMain/kotlin/com/wire/kalium/logic/sync/receiver/UserEventReceiver.kt @@ -27,7 +27,6 @@ import com.wire.kalium.logic.data.event.Event import com.wire.kalium.logic.data.event.EventLoggingStatus import com.wire.kalium.logic.data.event.logEventProcessing import com.wire.kalium.logic.data.logout.LogoutReason -import com.wire.kalium.logic.data.user.Connection import com.wire.kalium.logic.data.user.ConnectionState import com.wire.kalium.logic.data.user.UserId import com.wire.kalium.logic.data.user.UserRepository @@ -41,6 +40,8 @@ import com.wire.kalium.logic.functional.map import com.wire.kalium.logic.functional.onFailure import com.wire.kalium.logic.functional.onSuccess import com.wire.kalium.logic.kaliumLogger +import kotlin.time.Duration.Companion.ZERO +import kotlin.time.Duration.Companion.seconds internal interface UserEventReceiver : EventReceiver @@ -53,7 +54,7 @@ internal class UserEventReceiverImpl internal constructor( private val logout: LogoutUseCase, private val oneOnOneResolver: OneOnOneResolver, private val selfUserId: UserId, - private val currentClientIdProvider: CurrentClientIdProvider, + private val currentClientIdProvider: CurrentClientIdProvider ) : UserEventReceiver { override suspend fun onEvent(event: Event.User): Either { @@ -96,7 +97,15 @@ internal class UserEventReceiverImpl internal constructor( private suspend fun handleNewConnection(event: Event.User.NewConnection): Either = connectionRepository.insertConnectionFromEvent(event) .flatMap { - resolveActiveOneOnOneConversationUponConnectionAccepted(event.connection) + if (event.connection.status != ConnectionState.ACCEPTED) { + return@flatMap Either.Right(Unit) + } + + oneOnOneResolver.scheduleResolveOneOnOneConversationWithUserId( + event.connection.qualifiedToId, + delay = if (event.live) 3.seconds else ZERO + ) + Either.Right(Unit) } .onSuccess { kaliumLogger @@ -114,13 +123,6 @@ internal class UserEventReceiverImpl internal constructor( ) } - private suspend fun resolveActiveOneOnOneConversationUponConnectionAccepted(connection: Connection): Either = - if (connection.status == ConnectionState.ACCEPTED) { - oneOnOneResolver.resolveOneOnOneConversationWithUserId(connection.qualifiedToId).map { } - } else { - Either.Right(Unit) - } - private suspend fun handleClientRemove(event: Event.User.ClientRemove): Either = currentClientIdProvider().map { currentClientId -> if (currentClientId == event.clientId) { diff --git a/logic/src/commonTest/kotlin/com/wire/kalium/logic/data/conversation/ConversationRepositoryTest.kt b/logic/src/commonTest/kotlin/com/wire/kalium/logic/data/conversation/ConversationRepositoryTest.kt index f267fa575fa..bc77d380f8e 100644 --- a/logic/src/commonTest/kotlin/com/wire/kalium/logic/data/conversation/ConversationRepositoryTest.kt +++ b/logic/src/commonTest/kotlin/com/wire/kalium/logic/data/conversation/ConversationRepositoryTest.kt @@ -127,6 +127,7 @@ class ConversationRepositoryTest { "id", TestConversation.ID, false, + false, TestUser.SELF.id, "time", CONVERSATION_RESPONSE @@ -157,6 +158,7 @@ class ConversationRepositoryTest { "id", TestConversation.ID, false, + false, TestUser.SELF.id, "time", CONVERSATION_RESPONSE @@ -188,6 +190,7 @@ class ConversationRepositoryTest { "id", TestConversation.ID, false, + false, TestUser.SELF.id, "time", CONVERSATION_RESPONSE @@ -219,6 +222,7 @@ class ConversationRepositoryTest { "id", TestConversation.ID, false, + false, TestUser.SELF.id, "time", CONVERSATION_RESPONSE.copy( diff --git a/logic/src/commonTest/kotlin/com/wire/kalium/logic/data/conversation/MLSConversationRepositoryTest.kt b/logic/src/commonTest/kotlin/com/wire/kalium/logic/data/conversation/MLSConversationRepositoryTest.kt index 5ada13d0251..b9898cc8b0e 100644 --- a/logic/src/commonTest/kotlin/com/wire/kalium/logic/data/conversation/MLSConversationRepositoryTest.kt +++ b/logic/src/commonTest/kotlin/com/wire/kalium/logic/data/conversation/MLSConversationRepositoryTest.kt @@ -1286,6 +1286,7 @@ class MLSConversationRepositoryTest { "eventId", TestConversation.ID, false, + false, TestUser.USER_ID, WELCOME.encodeBase64(), timestampIso = "2022-03-30T15:36:00.000Z" 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 677481ce34c..7b0ddf225ab 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.util.arrangement.IncrementalSyncRepositoryArrangement +import com.wire.kalium.logic.util.arrangement.IncrementalSyncRepositoryArrangementImpl import com.wire.kalium.logic.util.arrangement.UserRepositoryArrangement import com.wire.kalium.logic.util.arrangement.UserRepositoryArrangementImpl import com.wire.kalium.logic.util.arrangement.mls.OneOnOneMigratorArrangement @@ -119,14 +121,16 @@ class OneOnOneResolverTest { private class Arrangement(private val block: Arrangement.() -> Unit) : UserRepositoryArrangement by UserRepositoryArrangementImpl(), OneOnOneProtocolSelectorArrangement by OneOnOneProtocolSelectorArrangementImpl(), - OneOnOneMigratorArrangement by OneOnOneMigratorArrangementImpl() + OneOnOneMigratorArrangement by OneOnOneMigratorArrangementImpl(), + IncrementalSyncRepositoryArrangement by IncrementalSyncRepositoryArrangementImpl() { fun arrange() = run { block() this@Arrangement to OneOnOneResolverImpl( userRepository = userRepository, oneOnOneProtocolSelector = oneOnOneProtocolSelector, - oneOnOneMigrator = oneOnOneMigrator + oneOnOneMigrator = oneOnOneMigrator, + incrementalSyncRepository = incrementalSyncRepository ) } } diff --git a/logic/src/commonTest/kotlin/com/wire/kalium/logic/framework/TestEvent.kt b/logic/src/commonTest/kotlin/com/wire/kalium/logic/framework/TestEvent.kt index 19b9b0a9bac..83555104564 100644 --- a/logic/src/commonTest/kotlin/com/wire/kalium/logic/framework/TestEvent.kt +++ b/logic/src/commonTest/kotlin/com/wire/kalium/logic/framework/TestEvent.kt @@ -37,6 +37,7 @@ object TestEvent { eventId, TestConversation.ID, false, + false, TestUser.USER_ID, members, "2022-03-30T15:36:00.000Z" @@ -46,6 +47,7 @@ object TestEvent { eventId, TestConversation.ID, false, + false, TestUser.USER_ID, listOf(), "2022-03-30T15:36:00.000Z" @@ -56,6 +58,7 @@ object TestEvent { TestConversation.ID, "2022-03-30T15:36:00.000Z", false, + false, member ) @@ -64,6 +67,7 @@ object TestEvent { TestConversation.ID, "2022-03-30T15:36:00.000Z", false, + false, MutedConversationStatus.AllAllowed, "2022-03-30T15:36:00.000Zp" ) @@ -82,20 +86,22 @@ object TestEvent { eventId, TestConversation.ID, false, + false ) - fun clientRemove(eventId: String = "eventId", clientId: ClientId) = Event.User.ClientRemove(false, eventId, clientId) - fun userDelete(eventId: String = "eventId", userId: UserId) = Event.User.UserDelete(false, eventId, userId) + fun clientRemove(eventId: String = "eventId", clientId: ClientId) = Event.User.ClientRemove(false, false, eventId, clientId) + fun userDelete(eventId: String = "eventId", userId: UserId) = Event.User.UserDelete(false, false, eventId, userId) fun updateUser(eventId: String = "eventId", userId: UserId) = Event.User.Update( eventId, - false, userId.toString(), null, false, "newName", null, null, null, null, null + false, false, userId.toString(), null, false, "newName", null, null, null, null, null ) fun newClient(eventId: String = "eventId", clientId: ClientId = ClientId("client")) = Event.User.NewClient( - false, eventId, TestClient.CLIENT + false, false, eventId, TestClient.CLIENT ) fun newConnection(eventId: String = "eventId", status: ConnectionState = ConnectionState.PENDING) = Event.User.NewConnection( + false, false, eventId, Connection( @@ -113,6 +119,7 @@ object TestEvent { eventId, TestConversation.ID, false, + false, TestUser.USER_ID, "2022-03-30T15:36:00.000Z" ) @@ -121,6 +128,7 @@ object TestEvent { eventId, TestConversation.ID, false, + false, "newName", TestUser.USER_ID, "2022-03-30T15:36:00.000Z" @@ -130,6 +138,7 @@ object TestEvent { eventId, TestConversation.ID, false, + false, receiptMode = Conversation.ReceiptMode.ENABLED, senderUserId = TestUser.USER_ID ) @@ -139,13 +148,15 @@ object TestEvent { teamId = "teamId", name = "teamName", transient = false, + live = false, icon = "icon", ) fun teamMemberJoin(eventId: String = "eventId") = Event.Team.MemberJoin( eventId, teamId = "teamId", - false, + transient = false, + live = false, memberId = "memberId" ) @@ -154,7 +165,8 @@ object TestEvent { teamId = "teamId", memberId = "memberId", timestampIso = "2022-03-30T15:36:00.000Z", - transient = false + transient = false, + live = false ) fun teamMemberUpdate(eventId: String = "eventId", permissionCode: Int) = Event.Team.MemberUpdate( @@ -162,13 +174,15 @@ object TestEvent { teamId = "teamId", memberId = "memberId", permissionCode = permissionCode, - transient = false + transient = false, + live = false ) fun timerChanged(eventId: String = "eventId") = Event.Conversation.ConversationMessageTimer( id = eventId, conversationId = TestConversation.ID, transient = false, + live = false, messageTimer = 3000, senderUserId = TestUser.USER_ID, timestampIso = "2022-03-30T15:36:00.000Z" @@ -177,6 +191,7 @@ object TestEvent { fun userPropertyReadReceiptMode(eventId: String = "eventId") = Event.UserProperty.ReadReceiptModeSet( id = eventId, transient = false, + live = false, value = true ) @@ -188,6 +203,7 @@ object TestEvent { "eventId", TestConversation.ID, false, + false, senderUserId, TestClient.CLIENT_ID, "time", @@ -201,6 +217,7 @@ object TestEvent { "eventId", TestConversation.ID, false, + false, null, TestUser.USER_ID, timestamp.toIsoDateTimeString(), @@ -211,6 +228,7 @@ object TestEvent { id = "eventId", conversationId = TestConversation.ID, transient = false, + live = false, timestampIso = "timestamp", conversation = TestConversation.CONVERSATION_RESPONSE, senderUserId = TestUser.SELF.id @@ -220,6 +238,7 @@ object TestEvent { "eventId", TestConversation.ID, false, + false, TestUser.USER_ID, "dummy-message", timestampIso = "2022-03-30T15:36:00.000Z" @@ -230,13 +249,15 @@ object TestEvent { conversationId = TestConversation.ID, data = TestConversation.CONVERSATION_RESPONSE, qualifiedFrom = TestUser.USER_ID, - transient = false + transient = false, + live = false ) fun codeUpdated() = Event.Conversation.CodeUpdated( id = "eventId", conversationId = TestConversation.ID, transient = false, + live = false, code = "code", key = "key", uri = "uri", @@ -247,6 +268,7 @@ object TestEvent { id = "eventId", conversationId = TestConversation.ID, transient = false, + live = false ) fun typingIndicator(typingIndicatorMode: Conversation.TypingIndicatorMode) = Event.Conversation.TypingIndicator( diff --git a/logic/src/commonTest/kotlin/com/wire/kalium/logic/sync/receiver/FeatureConfigEventReceiverTest.kt b/logic/src/commonTest/kotlin/com/wire/kalium/logic/sync/receiver/FeatureConfigEventReceiverTest.kt index 49f7f79b2d2..e44def97543 100644 --- a/logic/src/commonTest/kotlin/com/wire/kalium/logic/sync/receiver/FeatureConfigEventReceiverTest.kt +++ b/logic/src/commonTest/kotlin/com/wire/kalium/logic/sync/receiver/FeatureConfigEventReceiverTest.kt @@ -451,19 +451,19 @@ class FeatureConfigEventReceiverTest { fun newMLSUpdatedEvent( model: MLSModel - ) = Event.FeatureConfig.MLSUpdated("eventId", false, model) + ) = Event.FeatureConfig.MLSUpdated("eventId", false, false, model) fun newFileSharingUpdatedEvent( model: ConfigsStatusModel - ) = Event.FeatureConfig.FileSharingUpdated("eventId", false, model) + ) = Event.FeatureConfig.FileSharingUpdated("eventId", false, false, model) fun newConferenceCallingUpdatedEvent( model: ConferenceCallingModel - ) = Event.FeatureConfig.ConferenceCallingUpdated("eventId", false, model) + ) = Event.FeatureConfig.ConferenceCallingUpdated("eventId", false, false, model) fun newSelfDeletingMessagesUpdatedEvent( model: SelfDeletingMessagesModel - ) = Event.FeatureConfig.SelfDeletingMessagesConfig("eventId", false, model) + ) = Event.FeatureConfig.SelfDeletingMessagesConfig("eventId", false, false, model) fun arrange() = this to featureConfigEventReceiver } diff --git a/logic/src/commonTest/kotlin/com/wire/kalium/logic/sync/receiver/FederationEventReceiverTest.kt b/logic/src/commonTest/kotlin/com/wire/kalium/logic/sync/receiver/FederationEventReceiverTest.kt index 0c2c53e712c..b375e0d890a 100644 --- a/logic/src/commonTest/kotlin/com/wire/kalium/logic/sync/receiver/FederationEventReceiverTest.kt +++ b/logic/src/commonTest/kotlin/com/wire/kalium/logic/sync/receiver/FederationEventReceiverTest.kt @@ -100,6 +100,7 @@ class FederationEventReceiverTest { val event = Event.Federation.Delete( "id", true, + false, defederatedDomain ) @@ -170,6 +171,7 @@ class FederationEventReceiverTest { val event = Event.Federation.ConnectionRemoved( "id", true, + false, listOf(defederatedDomain, defederatedDomainTwo) ) diff --git a/logic/src/commonTest/kotlin/com/wire/kalium/logic/sync/receiver/UserEventReceiverTest.kt b/logic/src/commonTest/kotlin/com/wire/kalium/logic/sync/receiver/UserEventReceiverTest.kt index 2792bfa60f1..e38c3259476 100644 --- a/logic/src/commonTest/kotlin/com/wire/kalium/logic/sync/receiver/UserEventReceiverTest.kt +++ b/logic/src/commonTest/kotlin/com/wire/kalium/logic/sync/receiver/UserEventReceiverTest.kt @@ -31,8 +31,8 @@ import com.wire.kalium.logic.feature.CurrentClientIdProvider import com.wire.kalium.logic.feature.auth.LogoutUseCase import com.wire.kalium.logic.framework.TestConversation import com.wire.kalium.logic.framework.TestEvent -import com.wire.kalium.logic.framework.TestUser import com.wire.kalium.logic.functional.Either +import com.wire.kalium.logic.test_util.TestKaliumDispatcher import com.wire.kalium.logic.util.arrangement.UserRepositoryArrangement import com.wire.kalium.logic.util.arrangement.UserRepositoryArrangementImpl import com.wire.kalium.logic.util.arrangement.mls.OneOnOneResolverArrangement @@ -45,10 +45,13 @@ import io.mockative.given import io.mockative.mock import io.mockative.once import io.mockative.verify -import kotlinx.coroutines.flow.flowOf +import kotlinx.coroutines.ExperimentalCoroutinesApi +import kotlinx.coroutines.test.advanceUntilIdle import kotlinx.coroutines.test.runTest import kotlin.test.Test import kotlin.test.assertIs +import kotlin.time.Duration.Companion.ZERO +import kotlin.time.Duration.Companion.seconds class UserEventReceiverTest { @@ -205,18 +208,36 @@ class UserEventReceiverTest { } @Test - fun givenNewConnectionEventWithStatusAccepted_thenActiveOneOnOneConversationIsResolved() = runTest { + fun givenNewConnectionEventWithStatusAccepted_thenResolveActiveOneOnOneConversationIsScheduled() = runTest { val event = TestEvent.newConnection(status = ConnectionState.ACCEPTED).copy() val (arrangement, eventReceiver) = arrange { withInsertConnectionFromEventSucceeding() - withResolveOneOnOneConversationWithUserIdReturning(Either.Right(TestConversation.ID)) + withScheduleResolveOneOnOneConversationWithUserId() } eventReceiver.onEvent(event) verify(arrangement.oneOnOneResolver) - .suspendFunction(arrangement.oneOnOneResolver::resolveOneOnOneConversationWithUserId) - .with(eq(event.connection.qualifiedToId)) + .suspendFunction(arrangement.oneOnOneResolver::scheduleResolveOneOnOneConversationWithUserId) + .with(eq(event.connection.qualifiedToId), eq(ZERO)) + .wasInvoked(exactly = once) + } + + @OptIn(ExperimentalCoroutinesApi::class) + @Test + fun givenLiveNewConnectionEventWithStatusAccepted_thenResolveActiveOneOnOneConversationIsScheduledWithDelay() = runTest(TestKaliumDispatcher.default) { + val event = TestEvent.newConnection(status = ConnectionState.ACCEPTED).copy(live = true) + val (arrangement, eventReceiver) = arrange { + withInsertConnectionFromEventSucceeding() + withScheduleResolveOneOnOneConversationWithUserId() + } + + eventReceiver.onEvent(event) + advanceUntilIdle() + + verify(arrangement.oneOnOneResolver) + .suspendFunction(arrangement.oneOnOneResolver::scheduleResolveOneOnOneConversationWithUserId) + .with(eq(event.connection.qualifiedToId), eq(3.seconds)) .wasInvoked(exactly = once) } diff --git a/logic/src/commonTest/kotlin/com/wire/kalium/logic/sync/receiver/conversation/CodeDeletedHandlerTest.kt b/logic/src/commonTest/kotlin/com/wire/kalium/logic/sync/receiver/conversation/CodeDeletedHandlerTest.kt index 9ae28b1835e..400e3034a0a 100644 --- a/logic/src/commonTest/kotlin/com/wire/kalium/logic/sync/receiver/conversation/CodeDeletedHandlerTest.kt +++ b/logic/src/commonTest/kotlin/com/wire/kalium/logic/sync/receiver/conversation/CodeDeletedHandlerTest.kt @@ -41,7 +41,8 @@ class CodeDeletedHandlerTest { val event = Event.Conversation.CodeDeleted( conversationId = ConversationId("conversationId", "domain"), id = "event-id", - transient = false + transient = false, + live = false ) handler.handle(event) diff --git a/logic/src/commonTest/kotlin/com/wire/kalium/logic/sync/receiver/conversation/CodeUpdateHandlerTest.kt b/logic/src/commonTest/kotlin/com/wire/kalium/logic/sync/receiver/conversation/CodeUpdateHandlerTest.kt index c40c6e37a7f..9c213574329 100644 --- a/logic/src/commonTest/kotlin/com/wire/kalium/logic/sync/receiver/conversation/CodeUpdateHandlerTest.kt +++ b/logic/src/commonTest/kotlin/com/wire/kalium/logic/sync/receiver/conversation/CodeUpdateHandlerTest.kt @@ -45,7 +45,8 @@ class CodeUpdateHandlerTest { code = "code", key = "key", id = "event-id", - transient = false + transient = false, + live = false ) handler.handle(event) diff --git a/logic/src/commonTest/kotlin/com/wire/kalium/logic/sync/receiver/conversation/MLSWelcomeEventHandlerTest.kt b/logic/src/commonTest/kotlin/com/wire/kalium/logic/sync/receiver/conversation/MLSWelcomeEventHandlerTest.kt index b6a66f5ebfa..27095c69945 100644 --- a/logic/src/commonTest/kotlin/com/wire/kalium/logic/sync/receiver/conversation/MLSWelcomeEventHandlerTest.kt +++ b/logic/src/commonTest/kotlin/com/wire/kalium/logic/sync/receiver/conversation/MLSWelcomeEventHandlerTest.kt @@ -242,6 +242,7 @@ class MLSWelcomeEventHandlerTest { "eventId", CONVERSATION_ID, false, + false, TestUser.USER_ID, WELCOME.encodeBase64(), timestampIso = "2022-03-30T15:36:00.000Z" diff --git a/logic/src/commonTest/kotlin/com/wire/kalium/logic/sync/receiver/conversation/MemberLeaveEventHandlerTest.kt b/logic/src/commonTest/kotlin/com/wire/kalium/logic/sync/receiver/conversation/MemberLeaveEventHandlerTest.kt index d5be45c2828..58682933b92 100644 --- a/logic/src/commonTest/kotlin/com/wire/kalium/logic/sync/receiver/conversation/MemberLeaveEventHandlerTest.kt +++ b/logic/src/commonTest/kotlin/com/wire/kalium/logic/sync/receiver/conversation/MemberLeaveEventHandlerTest.kt @@ -129,6 +129,7 @@ class MemberLeaveEventHandlerTest { id = "id", conversationId = conversationId, transient = false, + live = false, removedBy = userId, removedList = listOf(userId), timestampIso = "timestampIso" diff --git a/logic/src/commonTest/kotlin/com/wire/kalium/logic/sync/receiver/conversation/NewConversationEventHandlerTest.kt b/logic/src/commonTest/kotlin/com/wire/kalium/logic/sync/receiver/conversation/NewConversationEventHandlerTest.kt index 5e4b0ab9cdc..6360cfd7bc7 100644 --- a/logic/src/commonTest/kotlin/com/wire/kalium/logic/sync/receiver/conversation/NewConversationEventHandlerTest.kt +++ b/logic/src/commonTest/kotlin/com/wire/kalium/logic/sync/receiver/conversation/NewConversationEventHandlerTest.kt @@ -62,6 +62,7 @@ class NewConversationEventHandlerTest { id = "eventId", conversationId = TestConversation.ID, transient = false, + live = false, timestampIso = "timestamp", conversation = TestConversation.CONVERSATION_RESPONSE, senderUserId = TestUser.SELF.id @@ -103,7 +104,8 @@ class NewConversationEventHandlerTest { val event = Event.Conversation.NewConversation( id = "eventId", conversationId = TestConversation.ID, - false, + transient =false, + live = false, timestampIso = "timestamp", conversation = TestConversation.CONVERSATION_RESPONSE, senderUserId = TestUser.SELF.id @@ -142,6 +144,7 @@ class NewConversationEventHandlerTest { id = "eventId", conversationId = TestConversation.ID, transient = false, + live = false, timestampIso = "timestamp", conversation = TestConversation.CONVERSATION_RESPONSE.copy( creator = "creatorId@creatorDomain", @@ -202,6 +205,7 @@ class NewConversationEventHandlerTest { id = "eventId", conversationId = TestConversation.ID, transient = false, + live = false, timestampIso = "timestamp", conversation = TestConversation.CONVERSATION_RESPONSE.copy( creator = "creatorId@creatorDomain", diff --git a/logic/src/commonTest/kotlin/com/wire/kalium/logic/util/arrangement/mls/OneOnOneResolverArrangement.kt b/logic/src/commonTest/kotlin/com/wire/kalium/logic/util/arrangement/mls/OneOnOneResolverArrangement.kt index 87e368402c0..21fdbe02e99 100644 --- a/logic/src/commonTest/kotlin/com/wire/kalium/logic/util/arrangement/mls/OneOnOneResolverArrangement.kt +++ b/logic/src/commonTest/kotlin/com/wire/kalium/logic/util/arrangement/mls/OneOnOneResolverArrangement.kt @@ -25,10 +25,13 @@ import io.mockative.Mock import io.mockative.any import io.mockative.given import io.mockative.mock +import kotlinx.coroutines.Job interface OneOnOneResolverArrangement { val oneOnOneResolver: OneOnOneResolver + + fun withScheduleResolveOneOnOneConversationWithUserId() fun withResolveOneOnOneConversationWithUserIdReturning(result: Either) fun withResolveOneOnOneConversationWithUserReturning(result: Either) fun withResolveAllOneOnOneConversationsReturning(result: Either) @@ -39,6 +42,13 @@ class OneOnOneResolverArrangementImpl : OneOnOneResolverArrangement { @Mock override val oneOnOneResolver = mock(OneOnOneResolver::class) + override fun withScheduleResolveOneOnOneConversationWithUserId() { + given(oneOnOneResolver) + .suspendFunction(oneOnOneResolver::scheduleResolveOneOnOneConversationWithUserId) + .whenInvokedWith(any(), any()) + .thenReturn(Job()) + } + override fun withResolveOneOnOneConversationWithUserIdReturning(result: Either) { given(oneOnOneResolver) .suspendFunction(oneOnOneResolver::resolveOneOnOneConversationWithUserId)