mirror of
https://github.com/corda/corda.git
synced 2024-12-19 04:57:58 +00:00
ENT-4990: Proper AMQP disconnect handling when remote session was not opened
This commit is contained in:
parent
054563e40c
commit
0af88365a1
@ -305,6 +305,17 @@ internal class ConnectionStateMachine(private val serverMode: Boolean,
|
||||
logDebugWithMDC { "Session final $session" }
|
||||
if (session == this.session) {
|
||||
this.session = null
|
||||
|
||||
// If TRANSPORT_CLOSED event was already processed, the 'transport' in all subsequent events is set to null.
|
||||
// There is, however, a chance of missing TRANSPORT_CLOSED event, e.g. when disconnect occurs before opening remote session.
|
||||
// In such cases we must explicitly cleanup the 'transport' in order to guarantee the delivery of CONNECTION_FINAL event.
|
||||
val transport = event.transport
|
||||
if (transport == this.transport) {
|
||||
logDebugWithMDC { "Missed TRANSPORT_CLOSED: force cleanup ${transport.prettyPrint}" }
|
||||
transport.unbind()
|
||||
transport.free()
|
||||
transport.context = null
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -0,0 +1,71 @@
|
||||
package net.corda.nodeapi.internal.protonwrapper.engine
|
||||
|
||||
import com.nhaarman.mockito_kotlin.any
|
||||
import com.nhaarman.mockito_kotlin.doReturn
|
||||
import com.nhaarman.mockito_kotlin.mock
|
||||
import com.nhaarman.mockito_kotlin.whenever
|
||||
import io.netty.channel.Channel
|
||||
import io.netty.channel.ChannelFuture
|
||||
import io.netty.channel.DefaultEventLoop
|
||||
import io.netty.channel.EventLoop
|
||||
import net.corda.nodeapi.internal.protonwrapper.messages.MessageStatus
|
||||
import net.corda.nodeapi.internal.protonwrapper.messages.impl.SendableMessageImpl
|
||||
import net.corda.testing.core.ALICE_NAME
|
||||
import net.corda.testing.core.BOB_NAME
|
||||
import net.corda.testing.internal.rigorousMock
|
||||
import org.apache.qpid.proton.amqp.transport.Begin
|
||||
import org.apache.qpid.proton.amqp.transport.Open
|
||||
import org.apache.qpid.proton.engine.impl.TransportImpl
|
||||
import org.junit.Test
|
||||
import java.util.concurrent.TimeUnit
|
||||
import kotlin.test.assertEquals
|
||||
|
||||
class EventProcessorTest {
|
||||
@Test(timeout=300_000)
|
||||
fun `reject unacknowledged message on disconnect`() {
|
||||
val executor = DefaultEventLoop()
|
||||
val channel = channel(executor)
|
||||
|
||||
val eventProcessor = EventProcessor(channel, false, ALICE_NAME.toString(), BOB_NAME.toString(), "username", "password")
|
||||
eventProcessor.processEventsAsync()
|
||||
|
||||
val msg = SendableMessageImpl("test".toByteArray(), "topic", BOB_NAME.toString(), mock(), mapOf())
|
||||
eventProcessor.transportWriteMessage(msg)
|
||||
eventProcessor.processEventsAsync()
|
||||
|
||||
// Open remote connection and session
|
||||
(eventProcessor.connection.transport as TransportImpl).also {
|
||||
Open().invoke(it, null, 0)
|
||||
Begin().invoke(it, null, 0)
|
||||
}
|
||||
eventProcessor.processEventsAsync()
|
||||
|
||||
executor.execute { eventProcessor.close() }
|
||||
assertEquals(MessageStatus.Rejected, msg.onComplete.get(5, TimeUnit.SECONDS))
|
||||
}
|
||||
|
||||
@Test(timeout=300_000)
|
||||
fun `reject unacknowledged message on disconnect without remote session being open`() {
|
||||
val executor = DefaultEventLoop()
|
||||
val channel = channel(executor)
|
||||
|
||||
val eventProcessor = EventProcessor(channel, false, ALICE_NAME.toString(), BOB_NAME.toString(), "username", "password")
|
||||
eventProcessor.processEventsAsync()
|
||||
|
||||
val msg = SendableMessageImpl("test".toByteArray(), "topic", BOB_NAME.toString(), mock(), mapOf())
|
||||
eventProcessor.transportWriteMessage(msg)
|
||||
eventProcessor.processEventsAsync()
|
||||
|
||||
executor.execute { eventProcessor.close() }
|
||||
assertEquals(MessageStatus.Rejected, msg.onComplete.get(5, TimeUnit.SECONDS))
|
||||
}
|
||||
|
||||
private fun channel(executor: EventLoop) = rigorousMock<Channel>().also {
|
||||
doReturn(executor).whenever(it).eventLoop()
|
||||
doReturn(mock<ChannelFuture>()).whenever(it).writeAndFlush(any())
|
||||
doReturn(true).whenever(it).isActive
|
||||
doReturn(mock<ChannelFuture>()).whenever(it).close()
|
||||
doReturn(null).whenever(it).localAddress()
|
||||
doReturn(null).whenever(it).remoteAddress()
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue
Block a user