Merge pull request #291 from corda/aslemmer-loadtest-fixes

Aslemmer loadtest fixes
This commit is contained in:
Andras Slemmer 2017-02-27 14:15:10 +00:00 committed by GitHub
commit 85243a6b76
7 changed files with 62 additions and 47 deletions

View File

@ -176,14 +176,15 @@ fun <A : Any> Generator.Companion.replicatePoisson(meanSize: Double, generator:
fun <A : Any> Generator.Companion.pickOne(list: List<A>) = Generator.intRange(0, list.size - 1).map { list[it] }
fun <A : Any> Generator.Companion.pickN(number: Int, list: List<A>) = Generator<List<A>> {
val mask = BitSet(list.size)
for (i in 0..Math.min(list.size, number) - 1) {
mask[i] = 1
val size = Math.min(list.size, number)
for (i in 0..size - 1) {
mask[i] = true
}
for (i in 0..mask.size() - 1) {
val byte = mask[i]
val swapIndex = i + it.nextInt(mask.size() - i)
for (i in 0..size - 1) {
val bit = mask[i]
val swapIndex = i + it.nextInt(size - i)
mask[i] = mask[swapIndex]
mask[swapIndex] = byte
mask[swapIndex] = bit
}
val resultList = ArrayList<A>()
list.forEachIndexed { index, a ->

View File

@ -72,7 +72,9 @@ class ConnectionManager(private val username: String, private val jSch: JSch) {
remoteMessagingPort: Int,
localTunnelAddress: HostAndPort,
certificatesBaseDirectory: Path,
remoteCertificatesDirectory: Path
remoteCertificatesDirectory: Path,
rpcUsername: String,
rpcPassword: String
): NodeConnection {
val session = jSch.getSession(username, nodeHost, 22)
// We don't check the host fingerprints because they may change often
@ -97,7 +99,7 @@ class ConnectionManager(private val username: String, private val jSch: JSch) {
channel.disconnect()
log.info("Certificates copied!")
val connection = NodeConnection(nodeHost, session, localTunnelAddress, certificatesDirectory)
val connection = NodeConnection(nodeHost, session, localTunnelAddress, certificatesDirectory, rpcUsername, rpcPassword)
connection.startClient()
return connection
}
@ -121,6 +123,8 @@ fun <A> connectToNodes(
remoteMessagingPort: Int,
tunnelPortAllocation: PortAllocation,
certificatesBaseDirectory: Path,
rpcUsername: String,
rpcPassword: String,
withConnections: (List<NodeConnection>) -> A
): A {
val manager = ConnectionManager(username, setupJSchWithSshAgent())
@ -130,7 +134,9 @@ fun <A> connectToNodes(
remoteMessagingPort = remoteMessagingPort,
localTunnelAddress = tunnelPortAllocation.nextHostAndPort(),
certificatesBaseDirectory = certificatesBaseDirectory,
remoteCertificatesDirectory = nodeHostAndCertificatesPath.second
remoteCertificatesDirectory = nodeHostAndCertificatesPath.second,
rpcUsername = rpcUsername,
rpcPassword = rpcPassword
)
}.toList()
@ -151,7 +157,9 @@ class NodeConnection(
val hostName: String,
private val jSchSession: Session,
private val localTunnelAddress: HostAndPort,
private val certificatesDirectory: Path
private val certificatesDirectory: Path,
private val rpcUsername: String,
private val rpcPassword: String
) : Closeable {
private val sslConfig = object : SSLConfiguration {
@ -187,7 +195,7 @@ class NodeConnection(
fun <A> doWhileClientStopped(action: () -> A): A {
val client = client
val proxy = _proxy
check(client == null || proxy == null) { "doWhileClientStopped called with no running client" }
require(client != null && proxy != null) { "doWhileClientStopped called with no running client" }
log.info("Stopping RPC proxy to $hostName, tunnel at $localTunnelAddress")
client!!.close()
try {
@ -196,7 +204,7 @@ class NodeConnection(
log.info("Starting new RPC proxy to $hostName, tunnel at $localTunnelAddress")
val newClient = CordaRPCClient(localTunnelAddress, sslConfig)
// TODO expose these somehow?
newClient.start("user1", "test")
newClient.start(rpcUsername, rpcPassword)
val newProxy = newClient.proxy()
this.client = newClient
this._proxy = newProxy
@ -206,7 +214,7 @@ class NodeConnection(
fun startClient() {
log.info("Creating RPC proxy to $hostName, tunnel at $localTunnelAddress")
val client = CordaRPCClient(localTunnelAddress, sslConfig)
client.start("user1", "test")
client.start(rpcUsername, rpcPassword)
val proxy = client.proxy()
log.info("Proxy created")
this.client = client

View File

@ -163,7 +163,9 @@ fun runLoadTests(configuration: LoadTestConfiguration, tests: List<Pair<LoadTest
configuration.nodeHosts.map { it to configuration.remoteNodeDirectory / "certificates" },
configuration.remoteMessagingPort,
PortAllocation.Incremental(configuration.localTunnelStartingPort),
configuration.localCertificatesBaseDirectory
configuration.localCertificatesBaseDirectory,
configuration.rpcUsername,
configuration.rpcPassword
) { connections ->
log.info("Connected to all nodes!")
val hostNodeHandleMap = ConcurrentHashMap<String, NodeHandle>()
@ -175,6 +177,8 @@ fun runLoadTests(configuration: LoadTestConfiguration, tests: List<Pair<LoadTest
val pubkeysString = otherNodeInfos.map {
" ${it.legalIdentity.name}: ${it.legalIdentity.owningKey.toBase58String()}"
}.joinToString("\n")
log.info("${connection.hostName} waiting for network map")
connection.proxy.waitUntilRegisteredWithNetworkMap().get()
log.info("${connection.hostName} sees\n$pubkeysString")
val nodeHandle = NodeHandle(configuration, connection, nodeInfo)
nodeHandle.waitUntilUp()

View File

@ -1,12 +1,16 @@
package net.corda.loadtest
import com.typesafe.config.Config
import java.nio.file.Path
import net.corda.node.services.config.*
/**
* @param sshUser The UNIX username to use for SSH auth.
* @param localCertificatesBaseDirectory The base directory to put node certificates in.
* @param localTunnelStartingPort The local starting port to allocate tunneling ports from.
* @param nodeHosts The nodes' resolvable addresses.
* @param rpcUsername The RPC user's name to establish the RPC connection as.
* @param rpcPassword The RPC user's password.
* @param remoteNodeDirectory The remote node directory.
* @param remoteMessagingPort The remote Artemis messaging port.
* @param remoteSystemdServiceName The name of the node's systemd service
@ -16,12 +20,16 @@ import java.nio.file.Path
* for disruptions.
*/
data class LoadTestConfiguration(
val sshUser: String,
val localCertificatesBaseDirectory: Path,
val localTunnelStartingPort: Int,
val nodeHosts: List<String>,
val remoteNodeDirectory: Path,
val remoteMessagingPort: Int,
val remoteSystemdServiceName: String,
val seed: Long?
)
val config: Config
) {
val sshUser: String by config
val localCertificatesBaseDirectory: Path by config
val localTunnelStartingPort: Int by config
val nodeHosts: List<String> = config.getStringList("nodeHosts")
val rpcUsername: String by config
val rpcPassword: String by config
val remoteNodeDirectory: Path by config
val remoteMessagingPort: Int by config
val remoteSystemdServiceName: String by config
val seed: Long? by config
}

View File

@ -44,24 +44,16 @@ import java.nio.file.Paths
*/
fun main(args: Array<String>) {
if (args.isEmpty()) {
throw IllegalArgumentException("Usage: <binary> PATH_TO_CONFIG")
}
val defaultConfig = ConfigFactory.parseResources("loadtest-reference.conf", ConfigParseOptions.defaults().setAllowMissing(false))
val customConfig = ConfigFactory.parseFile(File(args[0]), ConfigParseOptions.defaults().setAllowMissing(false))
val resolvedConfig = customConfig.withFallback(defaultConfig).resolve()
val loadTestConfiguration = LoadTestConfiguration(
sshUser = if (resolvedConfig.hasPath("sshUser")) resolvedConfig.getString("sshUser") else System.getProperty("user.name"),
localCertificatesBaseDirectory = Paths.get(resolvedConfig.getString("localCertificatesBaseDirectory")),
localTunnelStartingPort = resolvedConfig.getInt("localTunnelStartingPort"),
nodeHosts = resolvedConfig.getStringList("nodeHosts"),
remoteNodeDirectory = Paths.get("/opt/r3cev"),
remoteMessagingPort = 31337,
remoteSystemdServiceName = "r3cev-node",
seed = if (resolvedConfig.hasPath("seed")) resolvedConfig.getLong("seed") else null
val defaultSshUserConfig = ConfigFactory.parseMap(
if (defaultConfig.hasPath("sshUser")) emptyMap() else mapOf("sshUser" to System.getProperty("user.name"))
)
val customConfig = ConfigFactory.parseFile(File(args[0]), ConfigParseOptions.defaults().setAllowMissing(false))
val resolvedConfig = customConfig.withFallback(defaultConfig).withFallback(defaultSshUserConfig).resolve()
val loadTestConfiguration = LoadTestConfiguration(resolvedConfig)
if (loadTestConfiguration.nodeHosts.isEmpty()) {
throw IllegalArgumentException("Please specify at least one node host")
@ -99,7 +91,7 @@ fun main(args: Array<String>) {
crossCashTest to LoadTest.RunParameters(
parallelism = 4,
generateCount = 2000,
clearDatabaseBeforeRun = true,
clearDatabaseBeforeRun = false,
gatherFrequency = 10,
disruptionPatterns = listOf(
listOf(),

View File

@ -3,16 +3,15 @@ package net.corda.loadtest.tests
import net.corda.client.mock.Generator
import net.corda.client.mock.pickN
import net.corda.contracts.asset.Cash
import net.corda.core.*
import net.corda.core.contracts.Issued
import net.corda.core.contracts.PartyAndReference
import net.corda.core.contracts.USD
import net.corda.core.crypto.AbstractParty
import net.corda.core.crypto.AnonymousParty
import net.corda.core.flows.FlowException
import net.corda.core.getOrThrow
import net.corda.core.messaging.startFlow
import net.corda.core.serialization.OpaqueBytes
import net.corda.core.toFuture
import net.corda.flows.CashException
import net.corda.flows.CashFlowCommand
import net.corda.loadtest.LoadTest
@ -208,11 +207,12 @@ val crossCashTest = LoadTest<CrossCashCommand, CrossCashState>(
},
execute = { command ->
try {
val result = command.command.startFlow(command.node.connection.proxy).returnValue.getOrThrow()
log.info("Success: $result")
} catch (e: FlowException) {
log.error("Failure", e)
val result = command.command.startFlow(command.node.connection.proxy).returnValue
result.failure {
log.error("Failure[$command]", it)
}
result.success {
log.info("Success[$command]: $result")
}
},

View File

@ -2,6 +2,8 @@
# sshUser = "someusername", by default it uses the System property "user.name"
localCertificatesBaseDirectory = "build/load-test/certificates"
localTunnelStartingPort = 10000
remoteNodeDirectory = "/opt/r3cev"
remoteMessagingPort = 31337
remoteSystemdServiceName = "r3cev-node"
remoteNodeDirectory = "/opt/corda"
remoteMessagingPort = 10002
remoteSystemdServiceName = "corda"
rpcUsername = "corda"
rpcPassword = "rgb"