Skip to content

Commit

Permalink
feat: Handle needToRemoveLocally flag [WPB-14603] (#3158)
Browse files Browse the repository at this point in the history
* feat: Handle needToRemoveLocally flag [WPB-14603]

* Code review

---------

Co-authored-by: Yamil Medina <[email protected]>
  • Loading branch information
m-zagorski and yamilmedina authored Dec 16, 2024
1 parent 5396afa commit 6fb2177
Show file tree
Hide file tree
Showing 6 changed files with 231 additions and 7 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -259,7 +259,8 @@ sealed interface MessageContent {

data class Cleared(
val conversationId: ConversationId,
val time: Instant
val time: Instant,
val needToRemoveLocally: Boolean
) : Signaling

// server message content types
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -574,13 +574,15 @@ class ProtoContentMapperImpl(
Cleared(
conversationId = readableContent.conversationId.value,
qualifiedConversationId = idMapper.toProtoModel(readableContent.conversationId),
clearedTimestamp = readableContent.time.toEpochMilliseconds()
clearedTimestamp = readableContent.time.toEpochMilliseconds(),
needToRemoveLocally = readableContent.needToRemoveLocally
)
)

private fun unpackCleared(protoContent: GenericMessage.Content.Cleared) = MessageContent.Cleared(
conversationId = extractConversationId(protoContent.value.qualifiedConversationId, protoContent.value.conversationId),
time = Instant.fromEpochMilliseconds(protoContent.value.clearedTimestamp)
time = Instant.fromEpochMilliseconds(protoContent.value.clearedTimestamp),
needToRemoveLocally = protoContent.value.needToRemoveLocally ?: false
)

private fun toProtoLegalHoldStatus(legalHoldStatus: Conversation.LegalHoldStatus): LegalHoldStatus =
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -66,7 +66,8 @@ internal class ClearConversationContentUseCaseImpl(
id = uuid4().toString(),
content = MessageContent.Cleared(
conversationId = conversationId,
time = DateTimeUtil.currentInstant()
time = DateTimeUtil.currentInstant(),
needToRemoveLocally = false // TODO Handle in upcoming tasks
),
// sending the message to clear this conversation
conversationId = selfConversationId,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -41,11 +41,18 @@ internal class ClearConversationContentHandlerImpl(
message: Message.Signaling,
messageContent: MessageContent.Cleared
) {
val isMessageComingFromOtherClient = message.senderUserId == selfUserId
val isMessageComingFromAnotherUser = message.senderUserId != selfUserId
val isMessageDestinedForSelfConversation: Boolean = isMessageSentInSelfConversation(message)

if (isMessageComingFromOtherClient && isMessageDestinedForSelfConversation) {
conversationRepository.clearContent(messageContent.conversationId)
if (isMessageComingFromAnotherUser) {
when {
!messageContent.needToRemoveLocally && !isMessageDestinedForSelfConversation -> return
messageContent.needToRemoveLocally && !isMessageDestinedForSelfConversation -> conversationRepository.deleteConversation(
messageContent.conversationId
)

else -> conversationRepository.clearContent(messageContent.conversationId)
}
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,212 @@
/*
* Wire
* Copyright (C) 2024 Wire Swiss GmbH
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see http://www.gnu.org/licenses/.
*/
package com.wire.kalium.logic.sync.receiver.conversation

import com.wire.kalium.logic.data.conversation.ClientId
import com.wire.kalium.logic.data.conversation.ConversationRepository
import com.wire.kalium.logic.data.id.ConversationId
import com.wire.kalium.logic.data.message.IsMessageSentInSelfConversationUseCase
import com.wire.kalium.logic.data.message.Message
import com.wire.kalium.logic.data.message.MessageContent
import com.wire.kalium.logic.data.user.UserId
import com.wire.kalium.logic.framework.TestUser
import com.wire.kalium.logic.functional.Either
import com.wire.kalium.logic.sync.receiver.handler.ClearConversationContentHandler
import com.wire.kalium.logic.sync.receiver.handler.ClearConversationContentHandlerImpl
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 kotlinx.datetime.Instant
import kotlin.test.Test

class ClearConversationContentHandlerTest {

@Test
fun givenMessageFromOtherClient_whenMessageNeedsToBeRemovedLocallyAndUserIsNotPartOfConversation_thenWholeConversationShouldBeDeleted() =
runTest {
// given
val (arrangement, handler) = Arrangement()
.withMessageSentInSelfConversation(false)
.arrange()

// when
handler.handle(
message = MESSAGE,
messageContent = MessageContent.Cleared(
conversationId = CONVERSATION_ID,
time = Instant.DISTANT_PAST,
needToRemoveLocally = true
)
)

// then
coVerify { arrangement.conversationRepository.deleteConversation(any()) }
.wasInvoked(exactly = once)
coVerify { arrangement.conversationRepository.clearContent(any()) }
.wasNotInvoked()
}

@Test
fun givenMessageFromOtherClient_whenMessageNeedsToBeRemovedLocallyAndUserIsPartOfConversation_thenOnlyContentShouldBeCleared() =
runTest {
// given
val (arrangement, handler) = Arrangement()
.withMessageSentInSelfConversation(true)
.arrange()

// when
handler.handle(
message = MESSAGE,
messageContent = MessageContent.Cleared(
conversationId = CONVERSATION_ID,
time = Instant.DISTANT_PAST,
needToRemoveLocally = true
)
)

// then
coVerify { arrangement.conversationRepository.deleteConversation(any()) }
.wasNotInvoked()
coVerify { arrangement.conversationRepository.clearContent(any()) }
.wasInvoked(exactly = once)
}

@Test
fun givenMessageFromOtherClient_whenMessageDoesNotNeedToBeRemovedAndUserIsNotPartOfConversation_thenContentNorConversationShouldBeRemoved() =
runTest {
// given
val (arrangement, handler) = Arrangement()
.withMessageSentInSelfConversation(false)
.arrange()

// when
handler.handle(
message = MESSAGE,
messageContent = MessageContent.Cleared(
conversationId = CONVERSATION_ID,
time = Instant.DISTANT_PAST,
needToRemoveLocally = false
)
)

// then
coVerify { arrangement.conversationRepository.deleteConversation(any()) }
.wasNotInvoked()
coVerify { arrangement.conversationRepository.clearContent(any()) }
.wasNotInvoked()
}

@Test
fun givenMessageFromOtherClient_whenMessageDoesNotNeedToBeRemovedAndUserIsPartOfConversation_thenContentShouldBeRemoved() = runTest {
// given
val (arrangement, handler) = Arrangement()
.withMessageSentInSelfConversation(true)
.arrange()

// when
handler.handle(
message = MESSAGE,
messageContent = MessageContent.Cleared(
conversationId = CONVERSATION_ID,
time = Instant.DISTANT_PAST,
needToRemoveLocally = false
)
)

// then
coVerify { arrangement.conversationRepository.deleteConversation(any()) }
.wasNotInvoked()
coVerify { arrangement.conversationRepository.clearContent(any()) }
.wasInvoked(exactly = once)
}

@Test
fun givenMessageFromTheSameClient_whenHandleIsInvoked_thenContentNorConversationShouldBeRemoved() = runTest {
// given
val (arrangement, handler) = Arrangement()
.withMessageSentInSelfConversation(true)
.arrange()

// when
handler.handle(
message = OWN_MESSAGE,
messageContent = MessageContent.Cleared(
conversationId = CONVERSATION_ID,
time = Instant.DISTANT_PAST,
needToRemoveLocally = false
)
)

// then
coVerify { arrangement.conversationRepository.deleteConversation(any()) }
.wasNotInvoked()
coVerify { arrangement.conversationRepository.clearContent(any()) }
.wasNotInvoked()
}


private class Arrangement {
@Mock
val conversationRepository = mock(ConversationRepository::class)

@Mock
val isMessageSentInSelfConversationUseCase = mock(IsMessageSentInSelfConversationUseCase::class)

suspend fun withMessageSentInSelfConversation(isSentInSelfConv: Boolean) = apply {
coEvery { isMessageSentInSelfConversationUseCase(any()) }.returns(isSentInSelfConv)
}

suspend fun arrange(): Pair<Arrangement, ClearConversationContentHandler> =
this to ClearConversationContentHandlerImpl(
conversationRepository = conversationRepository,
selfUserId = TestUser.USER_ID,
isMessageSentInSelfConversation = isMessageSentInSelfConversationUseCase,
).apply {
coEvery { conversationRepository.deleteConversation(any()) }.returns(Either.Right(Unit))
coEvery { conversationRepository.clearContent(any()) }.returns(Either.Right(Unit))
}
}

companion object {
private val CONVERSATION_ID = ConversationId("conversationId", "domain")
private val OTHER_USER_ID = UserId("otherUserId", "domain")

private val MESSAGE_CONTENT = MessageContent.DataTransfer(
trackingIdentifier = MessageContent.DataTransfer.TrackingIdentifier(
identifier = "abcd-1234-efgh-5678"
)
)
val MESSAGE = Message.Signaling(
id = "messageId",
content = MESSAGE_CONTENT,
conversationId = CONVERSATION_ID,
date = Instant.DISTANT_PAST,
senderUserId = OTHER_USER_ID,
senderClientId = ClientId("deviceId"),
status = Message.Status.Sent,
isSelfMessage = false,
expirationData = null,
)

val OWN_MESSAGE = MESSAGE.copy(senderUserId = TestUser.USER_ID)
}
}
1 change: 1 addition & 0 deletions protobuf-codegen/src/main/proto/messages.proto
Original file line number Diff line number Diff line change
Expand Up @@ -185,6 +185,7 @@ message Cleared {
required int64 cleared_timestamp = 2;
// only optional to maintain backwards compatibility
optional QualifiedConversationId qualified_conversation_id = 3;
optional bool needToRemoveLocally = 4 [default = false];
}

message MessageHide {
Expand Down

0 comments on commit 6fb2177

Please sign in to comment.