concurrentTest.channels.ChannelCancelUndeliveredElementStressTest.kt Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of kotlinx-coroutines-core
Show all versions of kotlinx-coroutines-core
Coroutines support libraries for Kotlin
/*
* Copyright 2016-2021 JetBrains s.r.o. Use of this source code is governed by the Apache 2.0 license.
*/
package kotlinx.coroutines.channels
import kotlinx.atomicfu.*
import kotlinx.coroutines.*
import kotlinx.coroutines.selects.*
import kotlin.random.*
import kotlin.test.*
class ChannelCancelUndeliveredElementStressTest : TestBase() {
private val repeatTimes = (if (isNative) 1_000 else 10_000) * stressTestMultiplier
// total counters
private var sendCnt = 0
private var trySendFailedCnt = 0
private var receivedCnt = 0
private var undeliveredCnt = 0
// last operation
private var lastReceived = 0
private var dSendCnt = 0
private var dSendExceptionCnt = 0
private var dTrySendFailedCnt = 0
private var dReceivedCnt = 0
private val dUndeliveredCnt = atomic(0)
@Test
fun testStress() = runMtTest {
repeat(repeatTimes) {
val channel = Channel(1) { dUndeliveredCnt.incrementAndGet() }
val j1 = launch(Dispatchers.Default) {
sendOne(channel) // send first
sendOne(channel) // send second
}
val j2 = launch(Dispatchers.Default) {
receiveOne(channel) // receive one element from the channel
channel.cancel() // cancel the channel
}
joinAll(j1, j2)
// All elements must be either received or undelivered (IN every run)
if (dSendCnt - dTrySendFailedCnt != dReceivedCnt + dUndeliveredCnt.value) {
println(" Send: $dSendCnt")
println("Send exception: $dSendExceptionCnt")
println("trySend failed: $dTrySendFailedCnt")
println(" Received: $dReceivedCnt")
println(" Undelivered: ${dUndeliveredCnt.value}")
error("Failed")
}
trySendFailedCnt += dTrySendFailedCnt
receivedCnt += dReceivedCnt
undeliveredCnt += dUndeliveredCnt.value
// clear for next run
dSendCnt = 0
dSendExceptionCnt = 0
dTrySendFailedCnt = 0
dReceivedCnt = 0
dUndeliveredCnt.value = 0
}
// Stats
println(" Send: $sendCnt")
println("trySend failed: $trySendFailedCnt")
println(" Received: $receivedCnt")
println(" Undelivered: $undeliveredCnt")
assertEquals(sendCnt - trySendFailedCnt, receivedCnt + undeliveredCnt)
}
private suspend fun sendOne(channel: Channel) {
dSendCnt++
val i = ++sendCnt
try {
when (Random.nextInt(2)) {
0 -> channel.send(i)
1 -> if (!channel.trySend(i).isSuccess) {
dTrySendFailedCnt++
}
}
} catch (e: Throwable) {
assertTrue(e is CancellationException) // the only exception possible in this test
dSendExceptionCnt++
throw e
}
}
private suspend fun receiveOne(channel: Channel) {
val received = when (Random.nextInt(3)) {
0 -> channel.receive()
1 -> channel.receiveCatching().getOrElse { error("Cannot be closed yet") }
2 -> select {
channel.onReceive { it }
}
else -> error("Cannot happen")
}
assertTrue(received > lastReceived)
dReceivedCnt++
lastReceived = received
}
}