mirror of
https://github.com/mamoe/mirai.git
synced 2025-01-05 23:50:08 +08:00
Revise exception handling in NetworkHandler, involving:
- HeartbeatProcessor - HeartbeatFailedException: IOException is not recoverable, since this is not even thrown
This commit is contained in:
parent
2f40d3f432
commit
1c7e3bc5a1
@ -1,14 +1,16 @@
|
||||
/*
|
||||
* Copyright 2019-2021 Mamoe Technologies and contributors.
|
||||
* Copyright 2019-2022 Mamoe Technologies and contributors.
|
||||
*
|
||||
* 此源代码的使用受 GNU AFFERO GENERAL PUBLIC LICENSE version 3 许可证的约束, 可以在以下链接找到该许可证.
|
||||
* Use of this source code is governed by the GNU AGPLv3 license that can be found through the following link.
|
||||
*
|
||||
* https://github.com/mamoe/mirai/blob/master/LICENSE
|
||||
* https://github.com/mamoe/mirai/blob/dev/LICENSE
|
||||
*/
|
||||
|
||||
package net.mamoe.mirai.internal.network.components
|
||||
|
||||
import kotlinx.coroutines.CancellationException
|
||||
import kotlinx.coroutines.TimeoutCancellationException
|
||||
import net.mamoe.mirai.internal.network.component.ComponentKey
|
||||
import net.mamoe.mirai.internal.network.handler.NetworkHandler
|
||||
import net.mamoe.mirai.internal.network.protocol.packet.login.Heartbeat
|
||||
@ -16,21 +18,31 @@ import net.mamoe.mirai.internal.network.protocol.packet.login.StatSvc
|
||||
import net.mamoe.mirai.internal.network.protocol.packet.sendAndExpect
|
||||
|
||||
internal interface HeartbeatProcessor {
|
||||
|
||||
@Throws(Exception::class)
|
||||
/**
|
||||
* @throws TimeoutCancellationException if timed out waiting for response.
|
||||
* @throws CancellationException if [networkHandler] closed.
|
||||
* @throws Exception any other exceptions considered critical internal error and will stop SSO (i.e. stop Bot).
|
||||
*/
|
||||
suspend fun doAliveHeartbeatNow(networkHandler: NetworkHandler)
|
||||
|
||||
@Throws(Exception::class)
|
||||
/**
|
||||
* @throws TimeoutCancellationException if timed out waiting for response.
|
||||
* @throws CancellationException if [networkHandler] closed.
|
||||
* @throws Exception any other exceptions considered critical internal error and will stop SSO (i.e. stop Bot).
|
||||
*/
|
||||
suspend fun doStatHeartbeatNow(networkHandler: NetworkHandler)
|
||||
|
||||
@Throws(Exception::class)
|
||||
/**
|
||||
* @throws TimeoutCancellationException if timed out waiting for response.
|
||||
* @throws CancellationException if [networkHandler] closed.
|
||||
* @throws Exception any other exceptions considered critical internal error and will stop SSO (i.e. stop Bot).
|
||||
*/
|
||||
suspend fun doRegisterNow(networkHandler: NetworkHandler): StatSvc.Register.Response
|
||||
|
||||
companion object : ComponentKey<HeartbeatProcessor>
|
||||
}
|
||||
|
||||
internal class HeartbeatProcessorImpl : HeartbeatProcessor {
|
||||
@Throws(Exception::class)
|
||||
override suspend fun doStatHeartbeatNow(networkHandler: NetworkHandler) {
|
||||
StatSvc.SimpleGet(networkHandler.context.bot.client).sendAndExpect(
|
||||
networkHandler,
|
||||
@ -39,7 +51,6 @@ internal class HeartbeatProcessorImpl : HeartbeatProcessor {
|
||||
)
|
||||
}
|
||||
|
||||
@Throws(Exception::class)
|
||||
override suspend fun doAliveHeartbeatNow(networkHandler: NetworkHandler) {
|
||||
Heartbeat.Alive(networkHandler.context.bot.client).sendAndExpect(
|
||||
networkHandler,
|
||||
@ -48,7 +59,6 @@ internal class HeartbeatProcessorImpl : HeartbeatProcessor {
|
||||
)
|
||||
}
|
||||
|
||||
@Throws(Exception::class)
|
||||
override suspend fun doRegisterNow(networkHandler: NetworkHandler): StatSvc.Register.Response {
|
||||
return networkHandler.context[SsoProcessor].sendRegister(networkHandler)
|
||||
}
|
||||
|
@ -13,11 +13,15 @@ import kotlinx.coroutines.*
|
||||
import net.mamoe.mirai.internal.network.component.ComponentKey
|
||||
import net.mamoe.mirai.internal.network.component.ComponentStorage
|
||||
import net.mamoe.mirai.internal.network.handler.NetworkHandlerSupport
|
||||
import net.mamoe.mirai.internal.network.handler.selector.NetworkException
|
||||
import net.mamoe.mirai.internal.network.handler.selector.PacketTimeoutException
|
||||
import net.mamoe.mirai.utils.BotConfiguration.HeartbeatStrategy.*
|
||||
import net.mamoe.mirai.utils.MiraiLogger
|
||||
import net.mamoe.mirai.utils.info
|
||||
|
||||
/**
|
||||
* Accepts any kinds of exceptions. A [NetworkException] can control whether this error is recoverable, while any other ones are regarded as unexpected failure.
|
||||
*/
|
||||
internal typealias HeartbeatFailureHandler = (name: String, e: Throwable) -> Unit
|
||||
|
||||
/**
|
||||
@ -127,13 +131,13 @@ internal class TimeBasedHeartbeatSchedulerImpl(
|
||||
name,
|
||||
PacketTimeoutException(
|
||||
"$coroutineName: Timeout receiving action response",
|
||||
cause
|
||||
cause // cause is TimeoutCancellationException from `action`
|
||||
) // This is a NetworkException that is recoverable
|
||||
)
|
||||
return@async
|
||||
}
|
||||
} catch (e: Throwable) {
|
||||
// catch other errors
|
||||
// catch other errors in `action`, should not happen
|
||||
onHeartFailure(
|
||||
name,
|
||||
IllegalStateException("$coroutineName: Internal error: caught unexpected exception", e)
|
||||
|
@ -9,7 +9,9 @@
|
||||
|
||||
package net.mamoe.mirai.internal.network.handler
|
||||
|
||||
import kotlinx.coroutines.CancellationException
|
||||
import kotlinx.coroutines.CoroutineScope
|
||||
import kotlinx.coroutines.TimeoutCancellationException
|
||||
import kotlinx.coroutines.channels.ReceiveChannel
|
||||
import kotlinx.coroutines.flow.collect
|
||||
import kotlinx.coroutines.flow.consumeAsFlow
|
||||
@ -131,6 +133,10 @@ internal interface NetworkHandler : CoroutineScope {
|
||||
* Coroutine suspension may happen if connection is not yet available however,
|
||||
* [IllegalStateException] is thrown if [NetworkHandler] is already in [State.CLOSED] since closure is final.
|
||||
*
|
||||
* @throws TimeoutCancellationException if timeout has been reached.
|
||||
* @throws CancellationException if the [NetworkHandler] is closed, with the last cause for closure.
|
||||
* @throws IllegalArgumentException if [timeout] or [attempts] are invalid.
|
||||
*
|
||||
* @param attempts ranges `1..INFINITY`
|
||||
*/
|
||||
suspend fun sendAndExpect(packet: OutgoingPacket, timeout: Long = 5000, attempts: Int = 2): Packet?
|
||||
@ -141,6 +147,9 @@ internal interface NetworkHandler : CoroutineScope {
|
||||
* Response is still being processed but not passed as a return value of this function, so it does not suspends this function (due to awaiting for the response).
|
||||
* However, coroutine is still suspended if connection is not yet available,
|
||||
* and [IllegalStateException] is thrown if [NetworkHandler] is already in [State.CLOSED] since closure is final.
|
||||
* legalStateException] is thrown if [NetworkHandler] is already in [State.CLOSED] since closure is final.
|
||||
*
|
||||
* @throws CancellationException if the [NetworkHandler] is closed, with the last cause for closure.
|
||||
*/
|
||||
suspend fun sendWithoutExpect(packet: OutgoingPacket)
|
||||
|
||||
|
@ -43,6 +43,11 @@ internal abstract class NetworkHandlerSupport(
|
||||
.plus(CoroutineExceptionHandler.fromMiraiLogger(logger))
|
||||
|
||||
protected abstract fun initialState(): BaseStateImpl
|
||||
|
||||
/**
|
||||
* It's not guaranteed whether this function sends the packet in-place or launches a coroutine for it.
|
||||
* Caller should not rely on this property.
|
||||
*/
|
||||
protected abstract suspend fun sendPacketImpl(packet: OutgoingPacket)
|
||||
|
||||
protected fun collectUnknownPacket(raw: RawIncomingPacket) {
|
||||
|
@ -1,22 +1,20 @@
|
||||
/*
|
||||
* Copyright 2019-2021 Mamoe Technologies and contributors.
|
||||
* Copyright 2019-2022 Mamoe Technologies and contributors.
|
||||
*
|
||||
* 此源代码的使用受 GNU AFFERO GENERAL PUBLIC LICENSE version 3 许可证的约束, 可以在以下链接找到该许可证.
|
||||
* Use of this source code is governed by the GNU AGPLv3 license that can be found through the following link.
|
||||
*
|
||||
* https://github.com/mamoe/mirai/blob/master/LICENSE
|
||||
* https://github.com/mamoe/mirai/blob/dev/LICENSE
|
||||
*/
|
||||
|
||||
package net.mamoe.mirai.internal.network.impl.netty
|
||||
|
||||
import net.mamoe.mirai.internal.network.handler.selector.NetworkException
|
||||
import net.mamoe.mirai.utils.unwrapCancellationException
|
||||
import java.io.IOException
|
||||
|
||||
internal class HeartbeatFailedException(
|
||||
private val name: String, // kind of HB
|
||||
override val cause: Throwable,
|
||||
recoverable: Boolean = cause.unwrapCancellationException() is IOException || cause is NetworkException && cause.recoverable,
|
||||
recoverable: Boolean = cause is NetworkException && cause.recoverable,
|
||||
) : NetworkException(recoverable) {
|
||||
override val message: String = "Exception in $name job"
|
||||
override fun toString(): String = "HeartbeatFailedException: $name, recoverable=$recoverable, cause=$cause"
|
||||
|
@ -49,7 +49,7 @@ internal class IncomingPacket private constructor(
|
||||
val commandName: String,
|
||||
val sequenceId: Int,
|
||||
|
||||
val result: Either<Throwable, Packet?>
|
||||
val result: Either<Throwable, Packet?> // exception will be the same as caught from PacketFactory.decode. So they can be ISE, NPE, etc.
|
||||
) {
|
||||
companion object {
|
||||
operator fun invoke(commandName: String, sequenceId: Int, data: Packet?) =
|
||||
|
@ -1,10 +1,10 @@
|
||||
/*
|
||||
* Copyright 2019-2021 Mamoe Technologies and contributors.
|
||||
* Copyright 2019-2022 Mamoe Technologies and contributors.
|
||||
*
|
||||
* 此源代码的使用受 GNU AFFERO GENERAL PUBLIC LICENSE version 3 许可证的约束, 可以在以下链接找到该许可证.
|
||||
* Use of this source code is governed by the GNU AGPLv3 license that can be found through the following link.
|
||||
*
|
||||
* https://github.com/mamoe/mirai/blob/master/LICENSE
|
||||
* https://github.com/mamoe/mirai/blob/dev/LICENSE
|
||||
*/
|
||||
|
||||
package net.mamoe.mirai.internal.network.framework
|
||||
@ -117,6 +117,7 @@ internal sealed class AbstractRealNetworkHandlerTest<H : NetworkHandler> : Abstr
|
||||
override suspend fun init() {
|
||||
nhEvents.add(NHEvent.Init)
|
||||
networkLogger.debug { "BotInitProcessor.init" }
|
||||
bot.components[SsoProcessor].firstLoginResult.value = FirstLoginResult.PASSED
|
||||
}
|
||||
})
|
||||
set(ServerList, ServerListImpl())
|
||||
@ -161,6 +162,7 @@ internal sealed class AbstractRealNetworkHandlerTest<H : NetworkHandler> : Abstr
|
||||
}
|
||||
|
||||
val eventDispatcher get() = bot.components[EventDispatcher]
|
||||
val firstLoginResult: FirstLoginResult? get() = bot.components[SsoProcessor].firstLoginResult.value
|
||||
}
|
||||
|
||||
internal fun AbstractRealNetworkHandlerTest<*>.setSsoProcessor(action: suspend SsoProcessor.(handler: NetworkHandler) -> Unit) {
|
||||
|
@ -18,14 +18,26 @@ import net.mamoe.mirai.internal.network.framework.AbstractNettyNHTestWithSelecto
|
||||
import net.mamoe.mirai.internal.network.impl.netty.HeartbeatFailedException
|
||||
import net.mamoe.mirai.internal.network.impl.netty.NettyChannelException
|
||||
import net.mamoe.mirai.internal.test.runBlockingUnit
|
||||
import org.junit.jupiter.api.AfterEach
|
||||
import org.junit.jupiter.api.BeforeEach
|
||||
import org.junit.jupiter.api.Test
|
||||
import java.io.IOException
|
||||
import org.junit.jupiter.api.TestInfo
|
||||
import kotlin.test.assertFails
|
||||
|
||||
/**
|
||||
* Test whether the selector can recover the connection after first successful login.
|
||||
*/
|
||||
internal class SelectorRecoveryTest : AbstractNettyNHTestWithSelector() {
|
||||
@BeforeEach
|
||||
fun beforeTest(info: TestInfo) {
|
||||
println("=".repeat(30) + "BEGIN: ${info.displayName}" + "=".repeat(30))
|
||||
}
|
||||
|
||||
@AfterEach
|
||||
fun afterTest(info: TestInfo) {
|
||||
println("=".repeat(31) + "END: ${info.displayName}" + "=".repeat(31))
|
||||
}
|
||||
|
||||
@Test
|
||||
fun `stop on manual close`() = runBlockingUnit {
|
||||
network.resumeConnection()
|
||||
@ -33,19 +45,6 @@ internal class SelectorRecoveryTest : AbstractNettyNHTestWithSelector() {
|
||||
assertFails { network.resumeConnection() }
|
||||
}
|
||||
|
||||
/**
|
||||
* Emulates system hibernation and network failure.
|
||||
* @see HeartbeatFailedException
|
||||
*/
|
||||
@Test
|
||||
fun `can recover on heartbeat failure with IOException`() = runBlockingUnit {
|
||||
// We allow IOException to cause a reconnect.
|
||||
testRecoverWhenHeartbeatFailWith { IOException("test IO ex") }
|
||||
|
||||
// BotOfflineMonitor immediately launches a recovery which is UNDISPATCHED, so connection is immediately recovered.
|
||||
assertState(NetworkHandler.State.CONNECTING, NetworkHandler.State.LOADING, NetworkHandler.State.OK)
|
||||
}
|
||||
|
||||
/**
|
||||
* Emulates system hibernation and network failure.
|
||||
* @see HeartbeatFailedException
|
||||
|
@ -13,6 +13,7 @@ import kotlinx.coroutines.delay
|
||||
import kotlinx.coroutines.isActive
|
||||
import net.mamoe.mirai.internal.network.components.BotOfflineEventMonitor
|
||||
import net.mamoe.mirai.internal.network.components.BotOfflineEventMonitorImpl
|
||||
import net.mamoe.mirai.internal.network.components.FirstLoginResult
|
||||
import net.mamoe.mirai.internal.network.framework.AbstractNettyNHTest
|
||||
import net.mamoe.mirai.internal.network.framework.TestNettyNH
|
||||
import net.mamoe.mirai.internal.network.framework.setSsoProcessor
|
||||
@ -27,6 +28,7 @@ import net.mamoe.mirai.utils.cast
|
||||
import org.junit.jupiter.api.AfterEach
|
||||
import org.junit.jupiter.api.Test
|
||||
import java.io.IOException
|
||||
import kotlin.test.assertEquals
|
||||
import kotlin.test.assertFailsWith
|
||||
import kotlin.test.assertFalse
|
||||
|
||||
@ -93,8 +95,9 @@ internal class NettyBotNormalLoginTest : AbstractNettyNHTest() {
|
||||
// }
|
||||
|
||||
@Test
|
||||
fun `test resume after MsfOffline received`() = runBlockingUnit {
|
||||
fun `test resume after MsfOffline received after first login`() = runBlockingUnit {
|
||||
bot.login()
|
||||
assertEquals(FirstLoginResult.PASSED, firstLoginResult)
|
||||
bot.network.close(StatSvc.ReqMSFOffline.MsfOfflineToken(0, 0, 0))
|
||||
|
||||
eventDispatcher.joinBroadcast()
|
||||
|
@ -145,6 +145,7 @@ internal class NettyHandlerEventTest : AbstractNettyNHTest() {
|
||||
fun `BotOffline from OK TO CLOSED by bot close`() = runBlockingUnit {
|
||||
bot.login()
|
||||
assertState(OK)
|
||||
assertEquals(FirstLoginResult.PASSED, firstLoginResult)
|
||||
eventDispatcher.joinBroadcast() // `login` launches a job which broadcasts the event
|
||||
assertEventBroadcasts<Event>(1) {
|
||||
assertTrue { bot.isActive }
|
||||
|
Loading…
Reference in New Issue
Block a user