2022-01-14 06:56:15 +01:00
|
|
|
package org.thoughtcrime.securesms.conversation.v2
|
|
|
|
|
2022-09-04 13:03:32 +02:00
|
|
|
import com.goterl.lazysodium.utils.KeyPair
|
2023-08-28 01:51:48 +02:00
|
|
|
import kotlinx.coroutines.flow.emptyFlow
|
2022-01-14 06:56:15 +01:00
|
|
|
import kotlinx.coroutines.flow.first
|
|
|
|
import org.hamcrest.CoreMatchers.endsWith
|
|
|
|
import org.hamcrest.CoreMatchers.equalTo
|
2023-08-28 01:51:48 +02:00
|
|
|
import org.hamcrest.CoreMatchers.notNullValue
|
|
|
|
import org.hamcrest.CoreMatchers.nullValue
|
2022-01-14 06:56:15 +01:00
|
|
|
import org.hamcrest.MatcherAssert.assertThat
|
|
|
|
import org.junit.Before
|
|
|
|
import org.junit.Test
|
|
|
|
import org.mockito.Mockito.anyLong
|
|
|
|
import org.mockito.Mockito.anySet
|
|
|
|
import org.mockito.Mockito.verify
|
|
|
|
import org.mockito.kotlin.any
|
2023-08-28 01:51:48 +02:00
|
|
|
import org.mockito.kotlin.mock
|
|
|
|
import org.mockito.kotlin.whenever
|
2022-01-14 06:56:15 +01:00
|
|
|
import org.session.libsession.utilities.recipients.Recipient
|
|
|
|
import org.thoughtcrime.securesms.BaseViewModelTest
|
2022-09-04 13:03:32 +02:00
|
|
|
import org.thoughtcrime.securesms.database.Storage
|
2022-01-14 06:56:15 +01:00
|
|
|
import org.thoughtcrime.securesms.database.model.MessageRecord
|
|
|
|
import org.thoughtcrime.securesms.repository.ConversationRepository
|
|
|
|
import org.thoughtcrime.securesms.repository.ResultOf
|
|
|
|
|
|
|
|
class ConversationViewModelTest: BaseViewModelTest() {
|
|
|
|
|
2023-08-28 01:51:48 +02:00
|
|
|
private val repository = mock<ConversationRepository>()
|
|
|
|
private val storage = mock<Storage>()
|
2022-01-14 06:56:15 +01:00
|
|
|
|
|
|
|
private val threadId = 123L
|
2023-08-28 01:51:48 +02:00
|
|
|
private val edKeyPair = mock<KeyPair>()
|
2022-01-14 06:56:15 +01:00
|
|
|
private lateinit var recipient: Recipient
|
|
|
|
|
|
|
|
private val viewModel: ConversationViewModel by lazy {
|
2023-08-28 01:51:48 +02:00
|
|
|
ConversationViewModel(threadId, edKeyPair, repository, storage)
|
2022-01-14 06:56:15 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
@Before
|
|
|
|
fun setUp() {
|
2023-08-28 01:51:48 +02:00
|
|
|
recipient = mock()
|
Performance improvements and bug fixes (#869)
* refactor: fail on testSnode instead of recursively using up snode list. add call timeout on http client
* refactor: refactoring batch message receives and pollers
* refactor: reduce thread utils pool count to a 2 thread fixed pool. Do a check against pubkey instead of room names for oxenHostedOpenGroup
* refactor: caching lib with potential loader fixes and no-cache for giphy
* refactor: remove store and instead use ConcurrentHashMap with a backing update coroutine
* refactor: queue trim thread jobs instead of add every message processed
* fix: wrapping auth token and initial sync for open groups in a threadutils queued runnable, getting initial sync times down
* fix: fixing the user contacts cache in ConversationAdapter.kt
* refactor: improve polling and initial sync, move group joins from config messages into a background job fetching image.
* refactor: improving the job queuing for open groups, replacing placeholder avatar generation with a custom glide loader and archiving initial sync of open groups
* feat: add OpenGroupDeleteJob.kt
* feat: add open group delete job to process deletions after batch adding
* feat: add vacuum and fix job queue re-adding jobs forever, only try to set message hash values in DB if they have changed
* refactor: remove redundant inflation for profile image views throughout app
* refactor(wip): reducing layout inflation and starting to refactor the open group deletion issues taking a long time
* refactor(wip): refactoring group deletion to not iterate through and delete messages individually
* refactor(wip): refactoring group deletion to not iterate through and delete messages individually
* fix: group deletion optimisation
* build: bump build number
* build: bump build number and fix batch message receive retry logic
* fix: clear out open group deletes
* fix: update visible ConversationAdapter.kt binding for initial contact fetching and better traces for debugging background jobs
* fix: add in check for / force sync latest encryption key pair from linked devices if we already have that closed group
* Rename .java to .kt
* refactor: change MmsDatabase to kotlin to make list operations easier
* fix: nullable type
* fix: compilation issues and constants in .kt instead of .java
* fix: bug fix expiration timer on closed group recipient
* feat: use the job queue properly across executors
* feat: start on open group dispatcher-specific logic, probably a queue factory based on openGroupId if that is the same across new message and deletion jobs to ensure consistent entry and removal
* refactor: removing redundant code and fixing jobqueue per opengroup
* fix: allow attachments in note to self
* fix: make the minWidth in quote view bind max of text / title and body, wrapped ?
* fix: fixing up layouts and code view layouts
* fix: remove TODO, remove timestamp binding
* feat: fix view logic, avatars and padding, downloading attachments lazily (on bind), fixing potential crash, add WindowDebouncer.kt
* fix: NPE on viewModel recipient from removed thread while tearing down the Recipient observer in ConversationActivityV2.kt
* refactor: replace conversation notification debouncer handler with handlerthread, same as conversation list debouncer
* refactor: UI for groups and poller improvements
* fix: revert some changes in poller
* feat: add header back in for message requests
* refactor: remove Trace calls, add more conditions to the HomeDiffUtil for updating more efficiently
* feat: try update the home adapter if we get a profile picture modified event
* feat: bump build numbers
* fix: try to start with list in homeViewModel if we don't have already, render quotes to be width of attachment slide view instead of fixed
* fix: set channel to be conflated instead of no buffer
* fix: set unreads based off last local user message vs incrementing unreads to be all amount
* feat: add profile update flag, update build number
* fix: link preview thumbnails download on bind
* fix: centercrop placeholder in glide request
* feat: recycle the contact selection list and profile image in unbind
* fix: try to prevent user KP crash at weird times
* fix: remove additional log, improve attachment download success rate, fix share logs dialog issue
2022-06-08 09:12:34 +02:00
|
|
|
whenever(repository.maybeGetRecipientForThreadId(anyLong())).thenReturn(recipient)
|
2023-08-28 01:51:48 +02:00
|
|
|
whenever(repository.recipientUpdateFlow(anyLong())).thenReturn(emptyFlow())
|
2022-01-14 06:56:15 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
fun `should save draft message`() {
|
|
|
|
val draft = "Hi there"
|
|
|
|
|
|
|
|
viewModel.saveDraft(draft)
|
|
|
|
|
|
|
|
verify(repository).saveDraft(threadId, draft)
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
fun `should retrieve draft message`() {
|
|
|
|
val draft = "Hi there"
|
|
|
|
whenever(repository.getDraft(anyLong())).thenReturn(draft)
|
|
|
|
|
|
|
|
val result = viewModel.getDraft()
|
|
|
|
|
|
|
|
verify(repository).getDraft(threadId)
|
|
|
|
assertThat(result, equalTo(draft))
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
fun `should invite contacts`() {
|
|
|
|
val contacts = listOf<Recipient>()
|
|
|
|
|
|
|
|
viewModel.inviteContacts(contacts)
|
|
|
|
|
|
|
|
verify(repository).inviteContacts(threadId, contacts)
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
fun `should unblock contact recipient`() {
|
|
|
|
whenever(recipient.isContactRecipient).thenReturn(true)
|
|
|
|
|
|
|
|
viewModel.unblock()
|
|
|
|
|
2022-09-13 07:06:46 +02:00
|
|
|
verify(repository).setBlocked(recipient, false)
|
2022-01-14 06:56:15 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
fun `should delete locally`() {
|
2023-08-28 01:51:48 +02:00
|
|
|
val message = mock<MessageRecord>()
|
2022-01-14 06:56:15 +01:00
|
|
|
|
|
|
|
viewModel.deleteLocally(message)
|
|
|
|
|
|
|
|
verify(repository).deleteLocally(recipient, message)
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
fun `should emit error message on failure to delete a message for everyone`() = runBlockingTest {
|
2023-08-28 01:51:48 +02:00
|
|
|
val message = mock<MessageRecord>()
|
2022-01-14 06:56:15 +01:00
|
|
|
val error = Throwable()
|
|
|
|
whenever(repository.deleteForEveryone(anyLong(), any(), any()))
|
|
|
|
.thenReturn(ResultOf.Failure(error))
|
|
|
|
|
|
|
|
viewModel.deleteForEveryone(message)
|
|
|
|
|
|
|
|
assertThat(viewModel.uiState.first().uiMessages.first().message, endsWith("$error"))
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
fun `should emit error message on failure to delete messages without unsend request`() =
|
|
|
|
runBlockingTest {
|
2023-08-28 01:51:48 +02:00
|
|
|
val message = mock<MessageRecord>()
|
2022-01-14 06:56:15 +01:00
|
|
|
val error = Throwable()
|
|
|
|
whenever(repository.deleteMessageWithoutUnsendRequest(anyLong(), anySet()))
|
|
|
|
.thenReturn(ResultOf.Failure(error))
|
|
|
|
|
|
|
|
viewModel.deleteMessagesWithoutUnsendRequest(setOf(message))
|
|
|
|
|
|
|
|
assertThat(viewModel.uiState.first().uiMessages.first().message, endsWith("$error"))
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
fun `should emit error message on ban user failure`() = runBlockingTest {
|
|
|
|
val error = Throwable()
|
|
|
|
whenever(repository.banUser(anyLong(), any())).thenReturn(ResultOf.Failure(error))
|
|
|
|
|
|
|
|
viewModel.banUser(recipient)
|
|
|
|
|
|
|
|
assertThat(viewModel.uiState.first().uiMessages.first().message, endsWith("$error"))
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
fun `should emit a message on ban user success`() = runBlockingTest {
|
|
|
|
whenever(repository.banUser(anyLong(), any())).thenReturn(ResultOf.Success(Unit))
|
|
|
|
|
|
|
|
viewModel.banUser(recipient)
|
|
|
|
|
|
|
|
assertThat(
|
|
|
|
viewModel.uiState.first().uiMessages.first().message,
|
|
|
|
equalTo("Successfully banned user")
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
fun `should emit error message on ban user and delete all failure`() = runBlockingTest {
|
|
|
|
val error = Throwable()
|
|
|
|
whenever(repository.banAndDeleteAll(anyLong(), any())).thenReturn(ResultOf.Failure(error))
|
|
|
|
|
|
|
|
viewModel.banAndDeleteAll(recipient)
|
|
|
|
|
|
|
|
assertThat(viewModel.uiState.first().uiMessages.first().message, endsWith("$error"))
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
fun `should emit a message on ban user and delete all success`() = runBlockingTest {
|
|
|
|
whenever(repository.banAndDeleteAll(anyLong(), any())).thenReturn(ResultOf.Success(Unit))
|
|
|
|
|
|
|
|
viewModel.banAndDeleteAll(recipient)
|
|
|
|
|
|
|
|
assertThat(
|
|
|
|
viewModel.uiState.first().uiMessages.first().message,
|
|
|
|
equalTo("Successfully banned user and deleted all their messages")
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2022-03-04 07:46:39 +01:00
|
|
|
@Test
|
|
|
|
fun `should accept message request`() = runBlockingTest {
|
|
|
|
viewModel.acceptMessageRequest()
|
|
|
|
|
|
|
|
verify(repository).acceptMessageRequest(threadId, recipient)
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
fun `should decline message request`() {
|
|
|
|
viewModel.declineMessageRequest()
|
|
|
|
|
2022-09-13 07:06:46 +02:00
|
|
|
verify(repository).declineMessageRequest(threadId)
|
2022-03-04 07:46:39 +01:00
|
|
|
}
|
|
|
|
|
2022-01-14 06:56:15 +01:00
|
|
|
@Test
|
|
|
|
fun `should remove shown message`() = runBlockingTest {
|
|
|
|
// Given that a message is generated
|
|
|
|
whenever(repository.banUser(anyLong(), any())).thenReturn(ResultOf.Success(Unit))
|
|
|
|
viewModel.banUser(recipient)
|
|
|
|
assertThat(viewModel.uiState.value.uiMessages.size, equalTo(1))
|
|
|
|
// When the message is shown
|
|
|
|
viewModel.messageShown(viewModel.uiState.first().uiMessages.first().id)
|
|
|
|
// Then it should be removed
|
|
|
|
assertThat(viewModel.uiState.value.uiMessages.size, equalTo(0))
|
|
|
|
}
|
|
|
|
|
2023-08-28 01:51:48 +02:00
|
|
|
@Test
|
|
|
|
fun `open group recipient should have no blinded recipient`() {
|
|
|
|
whenever(recipient.isOpenGroupRecipient).thenReturn(true)
|
|
|
|
whenever(recipient.isOpenGroupOutboxRecipient).thenReturn(false)
|
|
|
|
whenever(recipient.isOpenGroupInboxRecipient).thenReturn(false)
|
|
|
|
assertThat(viewModel.blindedRecipient, nullValue())
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
fun `local recipient should have input and no blinded recipient`() {
|
|
|
|
whenever(recipient.isLocalNumber).thenReturn(true)
|
|
|
|
assertThat(viewModel.hidesInputBar(), equalTo(false))
|
|
|
|
assertThat(viewModel.blindedRecipient, nullValue())
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
fun `contact recipient should hide input bar if not accepting requests`() {
|
|
|
|
whenever(recipient.isOpenGroupInboxRecipient).thenReturn(true)
|
|
|
|
val blinded = mock<Recipient> {
|
|
|
|
whenever(it.blocksCommunityMessageRequests).thenReturn(true)
|
|
|
|
}
|
|
|
|
whenever(repository.maybeGetBlindedRecipient(recipient)).thenReturn(blinded)
|
|
|
|
assertThat(viewModel.blindedRecipient, notNullValue())
|
|
|
|
assertThat(viewModel.hidesInputBar(), equalTo(true))
|
|
|
|
}
|
|
|
|
|
2022-01-14 06:56:15 +01:00
|
|
|
}
|