mirror of
https://github.com/corda/corda.git
synced 2025-06-21 08:40:03 +00:00
Bogdan - small rework of RPC client sending (#2812)
* rework message handling while connection is down: do not send anything, throw on all obs and futures when failure detected * RPC calls will throw if used during failover; adapted tests
This commit is contained in:
@ -7,16 +7,13 @@ import net.corda.core.crypto.random63BitValue
|
||||
import net.corda.core.internal.concurrent.fork
|
||||
import net.corda.core.internal.concurrent.transpose
|
||||
import net.corda.core.messaging.RPCOps
|
||||
import net.corda.core.node.NodeInfo
|
||||
import net.corda.core.serialization.SerializationDefaults
|
||||
import net.corda.core.serialization.serialize
|
||||
import net.corda.core.utilities.*
|
||||
import net.corda.node.services.messaging.RPCServerConfiguration
|
||||
import net.corda.nodeapi.RPCApi
|
||||
import net.corda.testing.core.ALICE_NAME
|
||||
import net.corda.testing.core.SerializationEnvironmentRule
|
||||
import net.corda.testing.internal.testThreadFactory
|
||||
import net.corda.testing.node.User
|
||||
import net.corda.testing.node.internal.*
|
||||
import org.apache.activemq.artemis.api.core.SimpleString
|
||||
import org.junit.After
|
||||
@ -28,15 +25,12 @@ import rx.Observable
|
||||
import rx.subjects.PublishSubject
|
||||
import rx.subjects.UnicastSubject
|
||||
import java.time.Duration
|
||||
import java.time.Instant
|
||||
import java.util.concurrent.ConcurrentLinkedQueue
|
||||
import java.util.concurrent.Executors
|
||||
import java.util.concurrent.ScheduledExecutorService
|
||||
import java.util.concurrent.TimeUnit
|
||||
import java.util.concurrent.atomic.AtomicInteger
|
||||
import java.util.concurrent.atomic.AtomicLong
|
||||
import kotlin.concurrent.thread
|
||||
import kotlin.test.assertNotNull
|
||||
|
||||
class RPCStabilityTests {
|
||||
@Rule
|
||||
@ -253,42 +247,13 @@ class RPCStabilityTests {
|
||||
assertEquals("pong", client.ping())
|
||||
serverFollower.shutdown()
|
||||
startRpcServer<ReconnectOps>(ops = ops, customPort = serverPort).getOrThrow()
|
||||
Thread.sleep(1000) //wait for the server to come back up
|
||||
val pingFuture = pool.fork(client::ping)
|
||||
assertEquals("pong", pingFuture.getOrThrow(10.seconds))
|
||||
clientFollower.shutdown() // Driver would do this after the new server, causing hang.
|
||||
}
|
||||
}
|
||||
|
||||
@Test
|
||||
fun `client reconnects to server and resends buffered messages`() {
|
||||
rpcDriver(startNodesInProcess = false) {
|
||||
var nodeInfo: NodeInfo? = null
|
||||
var nodeTime: Instant? = null
|
||||
val alice = startNode(providedName = ALICE_NAME,
|
||||
rpcUsers = listOf(User("alice", "alice", setOf("ALL")))).getOrThrow()
|
||||
CordaRPCClient(alice.rpcAddress).use("alice", "alice") { connection ->
|
||||
val proxy = connection.proxy
|
||||
alice.stop()
|
||||
val nodeInfoThread = thread {
|
||||
nodeInfo = proxy.nodeInfo()
|
||||
}
|
||||
|
||||
val currentTimeThread = thread {
|
||||
nodeTime = proxy.currentNodeTime()
|
||||
}
|
||||
|
||||
Thread.sleep(5000)
|
||||
startNode(providedName = ALICE_NAME,
|
||||
rpcUsers = listOf(User("alice", "alice", setOf("ALL"))),
|
||||
customOverrides = mapOf("rpcSettings" to mapOf("address" to "localhost:${alice.rpcAddress.port}")))
|
||||
currentTimeThread.join()
|
||||
nodeInfoThread.join()
|
||||
assertNotNull(nodeInfo)
|
||||
assertNotNull(nodeTime)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@Test
|
||||
fun `connection failover fails, rpc calls throw`() {
|
||||
rpcDriver {
|
||||
@ -356,57 +321,6 @@ class RPCStabilityTests {
|
||||
}
|
||||
}
|
||||
|
||||
interface ThreadOps : RPCOps {
|
||||
fun sendMessage(id: Int, msgNo: Int): String
|
||||
}
|
||||
|
||||
@Test
|
||||
fun `multiple threads with 1000 messages for each thread`() {
|
||||
val messageNo = 1000
|
||||
val threadNo = 8
|
||||
val ops = object : ThreadOps {
|
||||
override val protocolVersion = 0
|
||||
override fun sendMessage(id: Int, msgNo: Int): String {
|
||||
return "($id-$msgNo)"
|
||||
}
|
||||
}
|
||||
|
||||
rpcDriver(startNodesInProcess = false) {
|
||||
val serverFollower = shutdownManager.follower()
|
||||
val serverPort = startRpcServer<ThreadOps>(rpcUser = User("alice", "alice", setOf("ALL")),
|
||||
ops = ops).getOrThrow().broker.hostAndPort!!
|
||||
|
||||
serverFollower.unfollow()
|
||||
val proxy = RPCClient<ThreadOps>(serverPort).start(ThreadOps::class.java, "alice", "alice").proxy
|
||||
val expectedMap = mutableMapOf<Int, StringBuilder>()
|
||||
val resultsMap = mutableMapOf<Int, StringBuilder>()
|
||||
|
||||
(1 until threadNo).forEach { nr ->
|
||||
(1 until messageNo).forEach { msgNo ->
|
||||
expectedMap[nr] = expectedMap.getOrDefault(nr, StringBuilder()).append("($nr-$msgNo)")
|
||||
}
|
||||
}
|
||||
|
||||
val threads = mutableMapOf<Int, Thread>()
|
||||
(1 until threadNo).forEach { nr ->
|
||||
val thread = thread {
|
||||
(1 until messageNo).forEach { msgNo ->
|
||||
resultsMap[nr] = resultsMap.getOrDefault(nr, StringBuilder()).append(proxy.sendMessage(nr, msgNo))
|
||||
}
|
||||
}
|
||||
threads[nr] = thread
|
||||
}
|
||||
// give the threads a chance to start sending some messages
|
||||
Thread.sleep(50)
|
||||
serverFollower.shutdown()
|
||||
startRpcServer<ThreadOps>(rpcUser = User("alice", "alice", setOf("ALL")),
|
||||
ops = ops, customPort = serverPort).getOrThrow()
|
||||
threads.values.forEach { it.join() }
|
||||
(1 until threadNo).forEach { assertEquals(expectedMap[it].toString(), resultsMap[it].toString()) }
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
interface TrackSubscriberOps : RPCOps {
|
||||
fun subscribe(): Observable<Unit>
|
||||
}
|
||||
|
Reference in New Issue
Block a user