Fix flaky tests for voice recording feature (#6330)
This commit is contained in:
parent
539d134b77
commit
65bc4acbab
1
changelog.d/6329.misc
Normal file
1
changelog.d/6329.misc
Normal file
@ -0,0 +1 @@
|
|||||||
|
Fix flaky test in voice recording feature.
|
@ -0,0 +1,69 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2022 New Vector Ltd
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package im.vector.app.core.utils
|
||||||
|
|
||||||
|
import kotlinx.coroutines.delay
|
||||||
|
import org.amshove.kluent.fail
|
||||||
|
import kotlin.time.Duration
|
||||||
|
import kotlin.time.Duration.Companion.milliseconds
|
||||||
|
import kotlin.time.Duration.Companion.seconds
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tries a [condition] several times until it returns true or a [timeout] is reached waiting for some [retryDelay] time between retries.
|
||||||
|
* On timeout it fails with an [errorMessage].
|
||||||
|
*/
|
||||||
|
suspend fun waitUntilCondition(
|
||||||
|
errorMessage: String,
|
||||||
|
timeout: Duration = 1.seconds,
|
||||||
|
retryDelay: Duration = 50.milliseconds,
|
||||||
|
condition: () -> Boolean,
|
||||||
|
) {
|
||||||
|
val start = System.currentTimeMillis()
|
||||||
|
do {
|
||||||
|
if (condition()) return
|
||||||
|
delay(retryDelay.inWholeMilliseconds)
|
||||||
|
} while (System.currentTimeMillis() - start < timeout.inWholeMilliseconds)
|
||||||
|
fail(errorMessage)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tries a [block] several times until it runs with no errors or a [timeout] is reached waiting for some [retryDelay] time between retries.
|
||||||
|
* On timeout it fails with a custom [errorMessage] or a caught [AssertionError].
|
||||||
|
*/
|
||||||
|
suspend fun waitUntil(
|
||||||
|
errorMessage: String? = null,
|
||||||
|
timeout: Duration = 1.seconds,
|
||||||
|
retryDelay: Duration = 50.milliseconds,
|
||||||
|
block: () -> Unit,
|
||||||
|
) {
|
||||||
|
var error: AssertionError?
|
||||||
|
val start = System.currentTimeMillis()
|
||||||
|
do {
|
||||||
|
try {
|
||||||
|
block()
|
||||||
|
return
|
||||||
|
} catch (e: AssertionError) {
|
||||||
|
error = e
|
||||||
|
}
|
||||||
|
delay(retryDelay.inWholeMilliseconds)
|
||||||
|
} while (System.currentTimeMillis() - start < timeout.inWholeMilliseconds)
|
||||||
|
if (errorMessage != null) {
|
||||||
|
fail(errorMessage)
|
||||||
|
} else {
|
||||||
|
throw error!!
|
||||||
|
}
|
||||||
|
}
|
@ -21,6 +21,7 @@ import androidx.test.platform.app.InstrumentationRegistry
|
|||||||
import androidx.test.rule.GrantPermissionRule
|
import androidx.test.rule.GrantPermissionRule
|
||||||
import io.mockk.spyk
|
import io.mockk.spyk
|
||||||
import kotlinx.coroutines.Dispatchers
|
import kotlinx.coroutines.Dispatchers
|
||||||
|
import kotlinx.coroutines.runBlocking
|
||||||
import org.amshove.kluent.shouldBeNull
|
import org.amshove.kluent.shouldBeNull
|
||||||
import org.amshove.kluent.shouldExist
|
import org.amshove.kluent.shouldExist
|
||||||
import org.amshove.kluent.shouldNotBeNull
|
import org.amshove.kluent.shouldNotBeNull
|
||||||
@ -42,8 +43,7 @@ class VoiceRecorderLTests {
|
|||||||
getVoiceMessageFile().shouldBeNull()
|
getVoiceMessageFile().shouldBeNull()
|
||||||
|
|
||||||
startRecord("some_room_id")
|
startRecord("some_room_id")
|
||||||
|
runBlocking { waitUntilRecordingFileExists() }
|
||||||
getVoiceMessageFile().shouldNotBeNullAndExist()
|
|
||||||
|
|
||||||
stopRecord()
|
stopRecord()
|
||||||
}
|
}
|
||||||
@ -53,6 +53,7 @@ class VoiceRecorderLTests {
|
|||||||
getVoiceMessageFile().shouldBeNull()
|
getVoiceMessageFile().shouldBeNull()
|
||||||
|
|
||||||
startRecord("some_room_id")
|
startRecord("some_room_id")
|
||||||
|
runBlocking { waitUntilRecordingFileExists() }
|
||||||
stopRecord()
|
stopRecord()
|
||||||
|
|
||||||
getVoiceMessageFile().shouldNotBeNullAndExist()
|
getVoiceMessageFile().shouldNotBeNullAndExist()
|
||||||
@ -61,8 +62,7 @@ class VoiceRecorderLTests {
|
|||||||
@Test
|
@Test
|
||||||
fun cancelRecordRemovesFile() = with(recorder) {
|
fun cancelRecordRemovesFile() = with(recorder) {
|
||||||
startRecord("some_room_id")
|
startRecord("some_room_id")
|
||||||
val file = recorder.getVoiceMessageFile()
|
val file = runBlocking { waitUntilRecordingFileExists() }
|
||||||
file.shouldNotBeNullAndExist()
|
|
||||||
|
|
||||||
cancelRecord()
|
cancelRecord()
|
||||||
|
|
||||||
|
@ -23,7 +23,6 @@ import androidx.test.platform.app.InstrumentationRegistry
|
|||||||
import androidx.test.rule.GrantPermissionRule
|
import androidx.test.rule.GrantPermissionRule
|
||||||
import io.mockk.spyk
|
import io.mockk.spyk
|
||||||
import io.mockk.verify
|
import io.mockk.verify
|
||||||
import kotlinx.coroutines.delay
|
|
||||||
import kotlinx.coroutines.runBlocking
|
import kotlinx.coroutines.runBlocking
|
||||||
import org.amshove.kluent.shouldBeNull
|
import org.amshove.kluent.shouldBeNull
|
||||||
import org.amshove.kluent.shouldExist
|
import org.amshove.kluent.shouldExist
|
||||||
@ -43,50 +42,36 @@ class VoiceRecorderQTests {
|
|||||||
private val recorder = spyk(VoiceRecorderQ(context))
|
private val recorder = spyk(VoiceRecorderQ(context))
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
fun startRecordCreatesOggFile() = runBlocking {
|
fun startRecordCreatesOggFile() = with(recorder) {
|
||||||
with(recorder) {
|
getVoiceMessageFile().shouldBeNull()
|
||||||
getVoiceMessageFile().shouldBeNull()
|
|
||||||
|
|
||||||
startRecord("some_room_id")
|
startRecord("some_room_id")
|
||||||
waitForRecording()
|
runBlocking { waitUntilRecordingFileExists() }
|
||||||
|
|
||||||
getVoiceMessageFile().shouldNotBeNullAndExist()
|
stopRecord()
|
||||||
|
|
||||||
stopRecord()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
fun stopRecordKeepsFile() = runBlocking {
|
fun stopRecordKeepsFile() = with(recorder) {
|
||||||
with(recorder) {
|
getVoiceMessageFile().shouldBeNull()
|
||||||
getVoiceMessageFile().shouldBeNull()
|
|
||||||
|
|
||||||
startRecord("some_room_id")
|
startRecord("some_room_id")
|
||||||
waitForRecording()
|
runBlocking { waitUntilRecordingFileExists() }
|
||||||
stopRecord()
|
stopRecord()
|
||||||
|
|
||||||
getVoiceMessageFile().shouldNotBeNullAndExist()
|
getVoiceMessageFile().shouldNotBeNullAndExist()
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
fun cancelRecordRemovesFileAfterStopping() = runBlocking {
|
fun cancelRecordRemovesFileAfterStopping() = with(recorder) {
|
||||||
with(recorder) {
|
startRecord("some_room_id")
|
||||||
startRecord("some_room_id")
|
val file = runBlocking { waitUntilRecordingFileExists() }
|
||||||
val file = recorder.getVoiceMessageFile()
|
cancelRecord()
|
||||||
file.shouldNotBeNullAndExist()
|
|
||||||
|
|
||||||
waitForRecording()
|
verify { stopRecord() }
|
||||||
cancelRecord()
|
getVoiceMessageFile().shouldBeNull()
|
||||||
|
file!!.shouldNotExist()
|
||||||
verify { stopRecord() }
|
|
||||||
getVoiceMessageFile().shouldBeNull()
|
|
||||||
file!!.shouldNotExist()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Give MediaRecorder some time to actually start recording
|
|
||||||
private suspend fun waitForRecording() = delay(10)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun File?.shouldNotBeNullAndExist() {
|
private fun File?.shouldNotBeNullAndExist() {
|
||||||
|
@ -0,0 +1,36 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2022 New Vector Ltd
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package im.vector.app.features.voice
|
||||||
|
|
||||||
|
import im.vector.app.core.utils.waitUntil
|
||||||
|
import org.amshove.kluent.shouldExist
|
||||||
|
import org.amshove.kluent.shouldNotBeNull
|
||||||
|
import java.io.File
|
||||||
|
import kotlin.time.Duration
|
||||||
|
import kotlin.time.Duration.Companion.milliseconds
|
||||||
|
import kotlin.time.Duration.Companion.seconds
|
||||||
|
|
||||||
|
// Give voice recorders some time to start recording and create the audio file
|
||||||
|
suspend fun VoiceRecorder.waitUntilRecordingFileExists(timeout: Duration = 1.seconds, delay: Duration = 10.milliseconds): File? {
|
||||||
|
waitUntil(timeout = timeout, retryDelay = delay) {
|
||||||
|
getVoiceMessageFile().run {
|
||||||
|
shouldNotBeNull()
|
||||||
|
shouldExist()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return getVoiceMessageFile()
|
||||||
|
}
|
@ -19,6 +19,7 @@ package im.vector.app.features.voice
|
|||||||
import android.content.Context
|
import android.content.Context
|
||||||
import android.media.MediaRecorder
|
import android.media.MediaRecorder
|
||||||
import android.os.Build
|
import android.os.Build
|
||||||
|
import org.matrix.android.sdk.api.extensions.tryOrNull
|
||||||
import org.matrix.android.sdk.api.session.content.ContentAttachmentData
|
import org.matrix.android.sdk.api.session.content.ContentAttachmentData
|
||||||
import org.matrix.android.sdk.api.util.md5
|
import org.matrix.android.sdk.api.util.md5
|
||||||
import java.io.File
|
import java.io.File
|
||||||
@ -80,7 +81,7 @@ abstract class AbstractVoiceRecorder(
|
|||||||
override fun stopRecord() {
|
override fun stopRecord() {
|
||||||
// Can throw when the record is less than 1 second.
|
// Can throw when the record is less than 1 second.
|
||||||
mediaRecorder?.let {
|
mediaRecorder?.let {
|
||||||
it.stop()
|
tryOrNull { it.stop() }
|
||||||
it.reset()
|
it.reset()
|
||||||
it.release()
|
it.release()
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user