Skip to content
This repository has been archived by the owner on May 14, 2024. It is now read-only.

Commit

Permalink
Merge branch 'fix/2293_sending-improvements' into 'prerelease'
Browse files Browse the repository at this point in the history
Improve send message process reliability

See merge request android/mail/proton-mail-android!678
  • Loading branch information
marinomeneghel committed Aug 23, 2021
2 parents af4f859 + 3b3e34f commit 361c440
Show file tree
Hide file tree
Showing 8 changed files with 56 additions and 43 deletions.
4 changes: 1 addition & 3 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -56,6 +56,7 @@ app/src/beta/google-services.json
.idea/**/usage.statistics.xml
.idea/**/dictionaries/**
.idea/assetWizardSettings.xml
.idea/deploymentTargetDropDown.xml
# Do not exclude project Dictionary
!.idea/**/dictionaries/project.xml
.idea/**/shelf
Expand Down Expand Up @@ -126,6 +127,3 @@ fabric.properties
# Android studio 3.1+ serialized cache file
.idea/caches/build_file_checksums.ser

# # # # # # # # #
# IDEA SETTINGS #
# # # # # # # # #
17 changes: 0 additions & 17 deletions .idea/deploymentTargetDropDown.xml

This file was deleted.

Original file line number Diff line number Diff line change
Expand Up @@ -55,7 +55,7 @@ internal const val KEY_INPUT_UPLOAD_ATTACHMENTS_IS_MESSAGE_SENDING = "keyUploadA
internal const val KEY_OUTPUT_RESULT_UPLOAD_ATTACHMENTS_ERROR = "keyUploadAttachmentResultError"

private const val UPLOAD_ATTACHMENTS_WORK_NAME_PREFIX = "uploadAttachmentUniqueWorkName"
private const val UPLOAD_ATTACHMENTS_MAX_RETRIES = 3
private const val UPLOAD_ATTACHMENTS_MAX_RETRIES = 1

class UploadAttachments @WorkerInject constructor(
@Assisted context: Context,
Expand Down Expand Up @@ -214,7 +214,7 @@ class UploadAttachments @WorkerInject constructor(
KEY_INPUT_UPLOAD_ATTACHMENTS_IS_MESSAGE_SENDING to isMessageSending
)
)
.setBackoffCriteria(BackoffPolicy.EXPONENTIAL, 2 * TEN_SECONDS, TimeUnit.SECONDS)
.setBackoffCriteria(BackoffPolicy.LINEAR, TEN_SECONDS, TimeUnit.SECONDS)
.build()

workManager.enqueueUniqueWork(
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -69,7 +69,10 @@ import ch.protonmail.android.usecase.compose.SaveDraft
import ch.protonmail.android.usecase.compose.SaveDraftResult
import ch.protonmail.android.utils.notifier.UserNotifier
import kotlinx.coroutines.CancellationException
import kotlinx.coroutines.NonCancellable
import kotlinx.coroutines.flow.Flow
import kotlinx.coroutines.withContext
import me.proton.core.util.kotlin.EMPTY_STRING
import me.proton.core.util.kotlin.deserialize
import me.proton.core.util.kotlin.serialize
import timber.log.Timber
Expand All @@ -89,7 +92,7 @@ internal const val KEY_INPUT_SEND_MESSAGE_SECURITY_OPTIONS_SERIALIZED = "keySend
internal const val KEY_OUTPUT_RESULT_SEND_MESSAGE_ERROR_ENUM = "keySendMessageErrorResult"

private const val INPUT_MESSAGE_DB_ID_NOT_FOUND = -1L
private const val SEND_MESSAGE_MAX_RETRIES = 3
private const val SEND_MESSAGE_MAX_RETRIES = 2
private const val NO_CONTACTS_AUTO_SAVE = 0
private const val SEND_MESSAGE_WORK_NAME_PREFIX = "sendMessageUniqueWorkName"
private const val NO_SUBJECT = ""
Expand All @@ -109,8 +112,12 @@ class SendMessageWorker @WorkerInject constructor(

override suspend fun doWork(): Result {
val messageDatabaseId = getInputMessageDatabaseId()
Timber.i("Send Message Worker executing with messageDatabaseId $messageDatabaseId")
val inputMessageId = getInputMessageId()
Timber.i(
"Send Message Worker executing with messageDatabaseId $messageDatabaseId - messageID $inputMessageId"
)
val message = messageDetailsRepository.findMessageByMessageDbId(messageDatabaseId)
?: messageDetailsRepository.findMessageById(inputMessageId)
if (message == null) {
showSendMessageError(NO_SUBJECT)
pendingActionsDao.deletePendingSendByDbId(messageDatabaseId)
Expand All @@ -122,8 +129,7 @@ class SendMessageWorker @WorkerInject constructor(
val previousSenderAddressId = requireNotNull(getInputPreviousSenderAddressId())
val username = requireNotNull(getInputCurrentUsername())

val result = saveDraft(message, previousSenderAddressId)
return when (result) {
return when (val result = saveDraft(message, previousSenderAddressId)) {
is SaveDraftResult.Success -> {
val messageId = result.draftId
Timber.i("Send Message Worker saved draft successfully for messageId $messageId")
Expand All @@ -148,8 +154,10 @@ class SendMessageWorker @WorkerInject constructor(
}

return try {
val response = apiManager.sendMessage(messageId, requestBody, RetrofitTag(username))
handleSendMessageResponse(messageId, response, savedDraftMessage)
withContext(NonCancellable) {
val response = apiManager.sendMessage(messageId, requestBody, RetrofitTag(username))
handleSendMessageResponse(messageId, response, savedDraftMessage)
}
} catch (exception: IOException) {
retryOrFail(ErrorPerformingApiRequest, savedDraftMessage, exception)
} catch (exception: Exception) {
Expand Down Expand Up @@ -323,6 +331,9 @@ class SendMessageWorker @WorkerInject constructor(

private fun getInputParentId() = inputData.getString(KEY_INPUT_SEND_MESSAGE_MSG_PARENT_ID)

private fun getInputMessageId() =
inputData.getString(KEY_INPUT_SEND_MESSAGE_MESSAGE_ID) ?: EMPTY_STRING

private fun getInputMessageDatabaseId() =
inputData.getLong(KEY_INPUT_SEND_MESSAGE_MSG_DB_ID, INPUT_MESSAGE_DB_ID_NOT_FOUND)

Expand Down Expand Up @@ -359,16 +370,15 @@ class SendMessageWorker @WorkerInject constructor(
KEY_INPUT_SEND_MESSAGE_SECURITY_OPTIONS_SERIALIZED to securityOptions.serialize()
)
)
.setBackoffCriteria(BackoffPolicy.EXPONENTIAL, 2 * TEN_SECONDS, TimeUnit.SECONDS)
.setBackoffCriteria(BackoffPolicy.LINEAR, TEN_SECONDS, TimeUnit.SECONDS)
.build()

workManager.enqueueUniqueWork(
"${SEND_MESSAGE_WORK_NAME_PREFIX}-${requireNotNull(message.messageId)}",
"$SEND_MESSAGE_WORK_NAME_PREFIX-${requireNotNull(message.messageId)}",
ExistingWorkPolicy.REPLACE,
sendMessageRequest
)
return workManager.getWorkInfoByIdLiveData(sendMessageRequest.id).asFlow()
}
}

}
Original file line number Diff line number Diff line change
Expand Up @@ -77,7 +77,7 @@ internal const val KEY_OUTPUT_RESULT_SAVE_DRAFT_ERROR_ENUM = "keySaveDraftErrorR
internal const val KEY_OUTPUT_RESULT_SAVE_DRAFT_MESSAGE_ID = "keySaveDraftSuccessResultDbId"

private const val INPUT_MESSAGE_DB_ID_NOT_FOUND = -1L
private const val SAVE_DRAFT_MAX_RETRIES = 3
private const val SAVE_DRAFT_MAX_RETRIES = 1

class CreateDraftWorker @WorkerInject constructor(
@Assisted context: Context,
Expand Down Expand Up @@ -336,7 +336,7 @@ class CreateDraftWorker @WorkerInject constructor(
KEY_INPUT_SAVE_DRAFT_PREV_SENDER_ADDR_ID to previousSenderAddressId
)
)
.setBackoffCriteria(BackoffPolicy.EXPONENTIAL, 2 * TEN_SECONDS, TimeUnit.SECONDS)
.setBackoffCriteria(BackoffPolicy.LINEAR, TEN_SECONDS, TimeUnit.SECONDS)
.build()

val uniqueWorkId = "$SAVE_DRAFT_UNIQUE_WORK_ID_PREFIX-${message.messageId}"
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -139,8 +139,8 @@ class UploadAttachmentsTest : CoroutinesTest {
assertEquals(true, actualIsMessageSending)
assertArrayEquals(attachmentIds.toTypedArray(), actualAttachmentIds)
assertEquals(NetworkType.CONNECTED, constraints.requiredNetworkType)
assertEquals(BackoffPolicy.EXPONENTIAL, workSpec.backoffPolicy)
assertEquals(20000, workSpec.backoffDelayDuration)
assertEquals(BackoffPolicy.LINEAR, workSpec.backoffPolicy)
assertEquals(10000, workSpec.backoffDelayDuration)
verify { workManager.getWorkInfoByIdLiveData(any()) }
}
}
Expand Down Expand Up @@ -261,7 +261,7 @@ class UploadAttachmentsTest : CoroutinesTest {
coEvery { attachmentsRepository.upload(attachment2, crypto) } answers {
AttachmentsRepository.Result.Failure("Failed to upload attachment2")
}
every { parameters.runAttemptCount } returns 2
every { parameters.runAttemptCount } returns 0

val result = uploadAttachments.doWork()

Expand Down Expand Up @@ -325,7 +325,7 @@ class UploadAttachmentsTest : CoroutinesTest {
coEvery { attachmentsRepository.uploadPublicKey(message, crypto) } answers {
AttachmentsRepository.Result.Failure("Failed to upload public key")
}
every { parameters.runAttemptCount } returns 2
every { parameters.runAttemptCount } returns 0

val result = uploadAttachments.doWork()

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -180,8 +180,8 @@ class SendMessageWorkerTest : CoroutinesTest {
assertEquals(previousSenderAddressId, actualPreviousSenderAddress)
assertEquals(securityOptions, actualMessageSecurityOptions?.deserialize(MessageSecurityOptions.serializer()))
assertEquals(NetworkType.CONNECTED, constraints.requiredNetworkType)
assertEquals(BackoffPolicy.EXPONENTIAL, workSpec.backoffPolicy)
assertEquals(20000, workSpec.backoffDelayDuration)
assertEquals(BackoffPolicy.LINEAR, workSpec.backoffPolicy)
assertEquals(10000, workSpec.backoffDelayDuration)
verify { workManager.getWorkInfoByIdLiveData(any()) }
}
}
Expand Down Expand Up @@ -220,12 +220,33 @@ class SendMessageWorkerTest : CoroutinesTest {
coVerify { saveDraft(expectedParameters) }
}

@Test
fun workerTriesFindingTheMessageByMessageIdWhenMessageIsNotFoundByDatabaseId() = runBlockingTest {
val messageDbId = 23712L
val messageId = "8322224-1341"
val message = Message(messageId = messageId)
val createdDraftId = "createdDraftId"
givenFullValidInput(messageDbId, messageId)
coEvery { messageDetailsRepository.findMessageByMessageDbId(messageDbId) } returns null
coEvery { messageDetailsRepository.findMessageById(messageId) } returns message
coEvery { messageDetailsRepository.findMessageById(createdDraftId) } returns null
coEvery { saveDraft.invoke(any()) } returns SaveDraftResult.Success(createdDraftId)

worker.doWork()

verify { userNotifier wasNot Called }
val paramsSlot = slot<SaveDraft.SaveDraftParameters>()
coVerify { saveDraft.invoke(capture(paramsSlot)) }
assertEquals(message, paramsSlot.captured.message)
}

@Test
fun workerNotifiesUserAndFailsWhenMessageIsNotFoundInTheDatabase() = runBlockingTest {
val messageDbId = 2373L
val messageId = "8322223"
givenFullValidInput(messageDbId, messageId)
coEvery { messageDetailsRepository.findMessageByMessageDbId(messageDbId) } returns null
coEvery { messageDetailsRepository.findMessageById(messageId) } returns null
every { context.getString(R.string.message_drafted) } returns "error message 9214"

val result = worker.doWork()
Expand Down Expand Up @@ -685,7 +706,7 @@ class SendMessageWorkerTest : CoroutinesTest {
coEvery { saveDraft(any()) } returns SaveDraftResult.Success(savedDraftMessageId)
every { sendPreferencesFactory.fetch(any()) } returns mapOf()
every { packageFactory.generatePackages(any(), any(), any(), any()) } throws exception
every { parameters.runAttemptCount } returns 2
every { parameters.runAttemptCount } returns 1
mockkStatic(Timber::class)

val result = worker.doWork()
Expand Down Expand Up @@ -1027,3 +1048,4 @@ class SendMessageWorkerTest : CoroutinesTest {
}
}
}

Original file line number Diff line number Diff line change
Expand Up @@ -174,8 +174,8 @@ class CreateDraftWorkerTest : CoroutinesTest {
assertEquals(messageActionType.messageActionTypeValue, actualMessageActionType)
assertEquals(previousSenderAddressId, actualPreviousSenderAddress)
assertEquals(NetworkType.CONNECTED, constraints.requiredNetworkType)
assertEquals(BackoffPolicy.EXPONENTIAL, workSpec.backoffPolicy)
assertEquals(20000, workSpec.backoffDelayDuration)
assertEquals(BackoffPolicy.LINEAR, workSpec.backoffPolicy)
assertEquals(10000, workSpec.backoffDelayDuration)
verify { workManager.getWorkInfoByIdLiveData(any()) }
}
}
Expand Down Expand Up @@ -954,7 +954,7 @@ class CreateDraftWorkerTest : CoroutinesTest {
every { this@mockk.message.subject } returns "Subject001"
}
coEvery { apiManager.createDraft(any()) } throws IOException(errorMessage)
every { parameters.runAttemptCount } returns 2
every { parameters.runAttemptCount } returns 0
val attachment = Attachment("attachment", keyPackets = "OriginalAttachmentPackets", inline = true)
val parentMessage = mockk<Message> {
coEvery { this@mockk.Attachments } returns listOf(attachment)
Expand Down

0 comments on commit 361c440

Please sign in to comment.