fix/QA882 voice message duration on android 9 API 28 (#980)

* WIP

* Do not update final outgoing voice message duration as it's possible to miscalculate it on older Android versions

* Revert attempted hardening adjustments to DecodedAudio.kt

* Correct the voice message check & minor tidy

* Minor tidy up

* Addressed PR feedback to simplify

---------

Co-authored-by: alansley <aclansley@gmail.com>
pull/1710/head
AL-Session 1 month ago committed by GitHub
parent d5b80440ad
commit 6f75829bdb
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194

@ -1931,10 +1931,14 @@ class ConversationActivityV2 : ScreenLockActionBarActivity(), InputBarDelegate,
private fun sendAttachments(
attachments: List<Attachment>,
body: String?,
quotedMessage: MessageRecord? = binding.inputBar?.quote,
quotedMessage: MessageRecord? = binding.inputBar.quote,
linkPreview: LinkPreview? = null
): Pair<Address, Long>? {
val recipient = viewModel.recipient ?: return null
if (viewModel.recipient == null) {
Log.w(TAG, "Cannot send attachments to a null recipient")
return null
}
val recipient = viewModel.recipient!!
val sentTimestamp = SnodeAPI.nowWithOffset
viewModel.beforeSendingAttachments()

@ -68,8 +68,7 @@ class VoiceMessageView @JvmOverloads constructor(
return
}
val player = AudioSlidePlayer.createFor(context.applicationContext, audioSlide, this)
this.player = player
this.player = AudioSlidePlayer.createFor(context.applicationContext, audioSlide, this)
// This sets the final duration of the uploaded voice message
(audioSlide.asAttachment() as? DatabaseAttachment)?.let { attachment ->

@ -1,6 +1,5 @@
package org.session.libsession.messaging.jobs
import okhttp3.HttpUrl
import okhttp3.HttpUrl.Companion.toHttpUrlOrNull
import org.session.libsession.database.MessageDataProvider
import org.session.libsession.database.StorageProtocol
@ -150,7 +149,7 @@ class AttachmentDownloadJob(val attachmentID: Long, val databaseMessageID: Long)
// process the duration
try {
InputStreamMediaDataSource(getInputStream(tempFile, attachment)).use { mediaDataSource ->
val durationMs = (DecodedAudio.create(mediaDataSource).totalDuration / 1000.0).toLong()
val durationMs = (DecodedAudio.create(mediaDataSource).totalDurationMicroseconds / 1000.0).toLong()
messageDataProvider.updateAudioAttachmentDuration(
attachment.attachmentId,
durationMs,

@ -57,6 +57,7 @@ class AttachmentUploadJob(val attachmentID: Long, val threadID: String, val mess
val attachment = messageDataProvider.getScaledSignalAttachmentStream(attachmentID)
?: return handleFailure(dispatcherName, Error.NoAttachment)
val openGroup = storage.getOpenGroup(threadID.toLong())
if (openGroup != null) {
val keyAndResult = upload(attachment, openGroup.server, false) {
OpenGroupApi.upload(it, openGroup.room, openGroup.server)
@ -114,20 +115,25 @@ class AttachmentUploadJob(val attachmentID: Long, val threadID: String, val mess
delegate?.handleJobSucceeded(this, dispatcherName)
val messageDataProvider = MessagingModuleConfiguration.shared.messageDataProvider
messageDataProvider.handleSuccessfulAttachmentUpload(attachmentID, attachment, attachmentKey, uploadResult)
if (attachment.contentType.startsWith("audio/")) {
// process the duration
// Outgoing voice messages do not have their final duration set because older Android versions (API 28 and below)
// can have bugs where the media duration is calculated incorrectly. In such cases we leave the correct "interim"
// voice message duration as the final duration as we know that it'll be correct..
if (attachment.contentType.startsWith("audio/") && !attachment.voiceNote) {
// ..but for outgoing audio files we do process the duration to the best of our ability.
try {
val inputStream = messageDataProvider.getAttachmentStream(attachmentID)!!.inputStream!!
InputStreamMediaDataSource(inputStream).use { mediaDataSource ->
val durationMs = (DecodedAudio.create(mediaDataSource).totalDuration / 1000.0).toLong()
val durationMS = (DecodedAudio.create(mediaDataSource).totalDurationMicroseconds / 1000.0).toLong()
messageDataProvider.getDatabaseAttachment(attachmentID)?.attachmentId?.let { attachmentId ->
messageDataProvider.updateAudioAttachmentDuration(attachmentId, durationMs, threadID.toLong())
messageDataProvider.updateAudioAttachmentDuration(attachmentId, durationMS, threadID.toLong())
}
}
} catch (e: Exception) {
Log.e("Loki", "Couldn't process audio attachment", e)
}
}
val storage = MessagingModuleConfiguration.shared.storage
storage.getMessageSendJob(messageSendJobID)?.let {
val destination = it.destination as? Destination.OpenGroup ?: return@let

@ -53,8 +53,8 @@ class DecodedAudio {
val sampleRate: Int
/** In microseconds. */
val totalDuration: Long
/** In microseconds. There are 1 million microseconds in a second. */
val totalDurationMicroseconds: Long
val channels: Int
@ -96,15 +96,15 @@ class DecodedAudio {
channels = mediaFormat.getInteger(MediaFormat.KEY_CHANNEL_COUNT)
sampleRate = mediaFormat.getInteger(MediaFormat.KEY_SAMPLE_RATE)
// On some old APIs (23) this field might be missing.
totalDuration = if (mediaFormat.containsKey(MediaFormat.KEY_DURATION)) {
totalDurationMicroseconds = if (mediaFormat.containsKey(MediaFormat.KEY_DURATION)) {
mediaFormat.getLong(MediaFormat.KEY_DURATION)
} else {
-1L
}
// Expected total number of samples per channel.
val expectedNumSamples = if (totalDuration >= 0) {
((totalDuration / 1000000f) * sampleRate + 0.5f).toInt()
val expectedNumSamples = if (totalDurationMicroseconds >= 0) {
((totalDurationMicroseconds / 1000000f) * sampleRate + 0.5f).toInt()
} else {
Int.MAX_VALUE
}

Loading…
Cancel
Save