blob: 777e4a3b55cbdad21a665663b0aefa5be20efad7 [file] [log] [blame]
package kotlinx.coroutines.experimental.channels
import kotlinx.coroutines.experimental.*
import org.junit.Test
import org.junit.Assert.*
class RendezvousChannelTest : TestBase() {
@Test
fun testSimple() = runBlocking {
val q = RendezvousChannel<Int>()
check(q.isEmpty && q.isFull)
expect(1)
val sender = launch(context) {
expect(4)
q.send(1) // suspend -- the first to come to rendezvous
expect(7)
q.send(2) // does not suspend -- receiver is there
expect(8)
}
expect(2)
val receiver = launch(context) {
expect(5)
check(q.receive() == 1) // does not suspend -- sender was there
expect(6)
check(q.receive() == 2) // suspends
expect(9)
}
expect(3)
sender.join()
receiver.join()
check(q.isEmpty && q.isFull)
finish(10)
}
@Test
fun testStress() = runBlocking {
val n = 100_000
val q = RendezvousChannel<Int>()
val sender = launch(context) {
for (i in 1..n) q.send(i)
expect(2)
}
val receiver = launch(context) {
for (i in 1..n) check(q.receive() == i)
expect(3)
}
expect(1)
sender.join()
receiver.join()
finish(4)
}
@Test
fun testClosedReceiveOrNull() = runBlocking {
val q = RendezvousChannel<Int>()
check(q.isEmpty && q.isFull && !q.isClosedForSend && !q.isClosedForReceive)
expect(1)
launch(context) {
expect(3)
assertEquals(42, q.receiveOrNull())
expect(4)
assertEquals(null, q.receiveOrNull())
expect(6)
}
expect(2)
q.send(42)
expect(5)
q.close()
check(!q.isEmpty && !q.isFull && q.isClosedForSend && q.isClosedForReceive)
yield()
check(!q.isEmpty && !q.isFull && q.isClosedForSend && q.isClosedForReceive)
finish(7)
}
@Test
fun testClosedExceptions() = runBlocking {
val q = RendezvousChannel<Int>()
expect(1)
launch(context) {
expect(4)
try { q.receive() }
catch (e: ClosedReceiveChannelException) {
expect(5)
}
}
expect(2)
q.close()
expect(3)
yield()
expect(6)
try { q.send(42) }
catch (e: ClosedSendChannelException) {
finish(7)
}
}
@Test
fun testOfferAndPool() = runBlocking {
val q = RendezvousChannel<Int>()
assertFalse(q.offer(1))
expect(1)
launch(context) {
expect(3)
assertEquals(null, q.poll())
expect(4)
assertEquals(2, q.receive())
expect(7)
assertEquals(null, q.poll())
yield()
expect(9)
assertEquals(3, q.poll())
expect(10)
}
expect(2)
yield()
expect(5)
assertTrue(q.offer(2))
expect(6)
yield()
expect(8)
q.send(3)
finish(11)
}
@Test
fun testIteratorClosed() = runBlocking {
val q = RendezvousChannel<Int>()
expect(1)
launch(context) {
expect(3)
q.close()
expect(4)
}
expect(2)
for (x in q) {
expectUnreached()
}
finish(5)
}
@Test
fun testIteratorOne() = runBlocking {
val q = RendezvousChannel<Int>()
expect(1)
launch(context) {
expect(3)
q.send(1)
expect(4)
q.close()
expect(5)
}
expect(2)
for (x in q) {
expect(6)
assertEquals(1, x)
}
finish(7)
}
@Test
fun testIteratorOneWithYield() = runBlocking {
val q = RendezvousChannel<Int>()
expect(1)
launch(context) {
expect(3)
q.send(1) // will suspend
expect(6)
q.close()
expect(7)
}
expect(2)
yield() // yield to sender coroutine right before starting for loop
expect(4)
for (x in q) {
expect(5)
assertEquals(1, x)
}
finish(8)
}
@Test
fun testIteratorTwo() = runBlocking {
val q = RendezvousChannel<Int>()
expect(1)
launch(context) {
expect(3)
q.send(1)
expect(4)
q.send(2)
expect(7)
q.close()
expect(8)
}
expect(2)
for (x in q) {
when (x) {
1 -> expect(5)
2 -> expect(6)
else -> expectUnreached()
}
}
finish(9)
}
@Test
fun testIteratorTwoWithYield() = runBlocking {
val q = RendezvousChannel<Int>()
expect(1)
launch(context) {
expect(3)
q.send(1) // will suspend
expect(6)
q.send(2)
expect(7)
q.close()
expect(8)
}
expect(2)
yield() // yield to sender coroutine right before starting for loop
expect(4)
for (x in q) {
when (x) {
1 -> expect(5)
2 -> expect(9)
else -> expectUnreached()
}
}
finish(10)
}
class BadClass {
override fun equals(other: Any?): Boolean = error("equals")
override fun hashCode(): Int = error("hashCode")
override fun toString(): String = error("toString")
}
@Test
fun testDeferBadClass() = runBlocking {
val bad = BadClass()
val c = buildChannel(context) {
expect(1)
send(bad)
}
assertTrue(c.receive() === bad)
finish(2)
}
}