1
0
mirror of https://github.com/mamoe/mirai.git synced 2025-04-03 14:20:10 +08:00

[core] Optimize ExceptionCollector, flatten suppressed exceptions and add them only to the last exception

This commit is contained in:
Him188 2022-08-25 19:35:37 +08:00
parent c1213bb0ca
commit 8dcc93089c
2 changed files with 26 additions and 5 deletions
mirai-core-utils/src
commonMain/kotlin
commonTest/kotlin/net/mamoe/mirai/utils

View File

@ -36,6 +36,7 @@ public open class ExceptionCollector {
@Volatile
private var last: Throwable? = null
private val hashCodes = mutableSetOf<Long>()
private val suppressedList = mutableListOf<Throwable>()
/**
* @return `true` if [e] is new.
@ -52,7 +53,8 @@ public open class ExceptionCollector {
}
protected open fun addSuppressed(receiver: Throwable, e: Throwable) {
receiver.addSuppressed(e)
suppressedList.add(e)
// receiver.addSuppressed(e)
}
public fun collectGet(e: Throwable?): Throwable {
@ -66,7 +68,23 @@ public open class ExceptionCollector {
*/
public fun collectException(e: Throwable?): Boolean = collect(e)
public fun getLast(): Throwable? = last
/**
* Adds [suppressedList] to suppressed exceptions of [last]
*/
@Synchronized
private fun bake() {
last?.let { last ->
for (suppressed in suppressedList.asReversed()) {
last.addSuppressed(suppressed)
}
}
suppressedList.clear()
}
public fun getLast(): Throwable? {
bake()
return last
}
@TerminalOperation // to give it a color for a clearer control flow
public fun collectThrow(exception: Throwable): Nothing {
@ -97,6 +115,7 @@ public open class ExceptionCollector {
public fun dispose() { // help gc
this.last = null
this.hashCodes.clear()
this.suppressedList.clear()
}
public companion object {
@ -130,6 +149,8 @@ public inline fun <R> ExceptionCollector.withExceptionCollector(action: Exceptio
return action()
} catch (e: Throwable) {
collectThrow(e)
} finally {
dispose()
}
}
}

View File

@ -32,7 +32,7 @@ internal class ExceptionCollectorTest {
collector.collect(IllegalStateException())
assertIs<IllegalStateException>(collector.getLast())
assertTrue { collector.getLast()!!.suppressedExceptions.single() is IllegalArgumentException }
assertTrue { collector.getLast()!!.suppressedExceptions[0] is IllegalArgumentException }
assertEquals(2, collector.asSequence().count())
}
@ -45,8 +45,8 @@ internal class ExceptionCollectorTest {
collector.collect(IllegalStateException())
assertIs<IllegalStateException>(collector.getLast())
assertTrue { collector.getLast()!!.suppressedExceptions.single() is IllegalArgumentException }
assertTrue { collector.getLast()!!.suppressedExceptions.single().suppressedExceptions.single() is IndexOutOfBoundsException }
assertTrue { collector.getLast()!!.suppressedExceptions[0] is IllegalArgumentException }
assertTrue { collector.getLast()!!.suppressedExceptions[1] is IndexOutOfBoundsException }
assertEquals(3, collector.asSequence().count())
}