1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
|
/*
* Copyright 2016-2019 JetBrains s.r.o. Use of this source code is governed by the Apache 2.0 license.
*/
package kotlinx.coroutines.exceptions
import kotlinx.coroutines.*
import kotlinx.coroutines.channels.*
import org.junit.Test
import kotlin.test.*
@Suppress("UNREACHABLE_CODE", "UNUSED", "UNUSED_PARAMETER")
class StackTraceRecoveryCustomExceptionsTest : TestBase() {
internal class NonCopyable(val customData: Int) : Throwable() {
// Bait
public constructor(cause: Throwable) : this(42)
}
internal class Copyable(val customData: Int) : Throwable(), CopyableThrowable<Copyable> {
// Bait
public constructor(cause: Throwable) : this(42)
override fun createCopy(): Copyable {
val copy = Copyable(customData)
copy.initCause(this)
return copy
}
}
@Test
fun testStackTraceNotRecovered() = runTest {
try {
withContext(wrapperDispatcher(coroutineContext)) {
throw NonCopyable(239)
}
expectUnreached()
} catch (e: NonCopyable) {
assertEquals(239, e.customData)
assertNull(e.cause)
}
}
@Test
fun testStackTraceRecovered() = runTest {
try {
withContext(wrapperDispatcher(coroutineContext)) {
throw Copyable(239)
}
expectUnreached()
} catch (e: Copyable) {
assertEquals(239, e.customData)
val cause = e.cause
assertTrue(cause is Copyable)
assertEquals(239, cause.customData)
}
}
internal class WithDefault(message: String = "default") : Exception(message)
@Test
fun testStackTraceRecoveredWithCustomMessage() = runTest {
try {
withContext(wrapperDispatcher(coroutineContext)) {
throw WithDefault("custom")
}
expectUnreached()
} catch (e: WithDefault) {
assertEquals("custom", e.message)
val cause = e.cause
assertTrue(cause is WithDefault)
assertEquals("custom", cause.message)
}
}
class WrongMessageException(token: String) : RuntimeException("Token $token")
@Test
fun testWrongMessageException() = runTest {
val result = runCatching {
coroutineScope<Unit> {
throw WrongMessageException("OK")
}
}
val ex = result.exceptionOrNull() ?: error("Expected to fail")
assertTrue(ex is WrongMessageException)
assertEquals("Token OK", ex.message)
}
@Test
fun testWrongMessageExceptionInChannel() = runTest {
val result = produce<Unit>(SupervisorJob() + Dispatchers.Unconfined) {
throw WrongMessageException("OK")
}
val ex = runCatching {
@Suppress("ControlFlowWithEmptyBody")
for (unit in result) {
// Iterator has a special code path
}
}.exceptionOrNull() ?: error("Expected to fail")
assertTrue(ex is WrongMessageException)
assertEquals("Token OK", ex.message)
}
class CopyableWithCustomMessage(
message: String?,
cause: Throwable? = null
) : RuntimeException(message, cause),
CopyableThrowable<CopyableWithCustomMessage> {
override fun createCopy(): CopyableWithCustomMessage {
return CopyableWithCustomMessage("Recovered: [$message]", cause)
}
}
@Test
fun testCustomCopyableMessage() = runTest {
val result = runCatching {
coroutineScope<Unit> {
throw CopyableWithCustomMessage("OK")
}
}
val ex = result.exceptionOrNull() ?: error("Expected to fail")
assertTrue(ex is CopyableWithCustomMessage)
assertEquals("Recovered: [OK]", ex.message)
}
}
|