mirror of
https://github.com/corda/corda.git
synced 2025-01-28 07:04:12 +00:00
Include fix for NPEs with RPC exceptions.
This commit is contained in:
commit
0c1a6aad6b
@ -1,6 +1,8 @@
|
|||||||
package net.corda.client
|
package net.corda.client
|
||||||
|
|
||||||
import net.corda.core.contracts.DOLLARS
|
import net.corda.core.contracts.DOLLARS
|
||||||
|
import net.corda.core.contracts.issuedBy
|
||||||
|
import net.corda.core.flows.FlowException
|
||||||
import net.corda.core.getOrThrow
|
import net.corda.core.getOrThrow
|
||||||
import net.corda.core.messaging.startFlow
|
import net.corda.core.messaging.startFlow
|
||||||
import net.corda.core.node.services.ServiceInfo
|
import net.corda.core.node.services.ServiceInfo
|
||||||
@ -8,26 +10,27 @@ import net.corda.core.random63BitValue
|
|||||||
import net.corda.core.serialization.OpaqueBytes
|
import net.corda.core.serialization.OpaqueBytes
|
||||||
import net.corda.flows.CashCommand
|
import net.corda.flows.CashCommand
|
||||||
import net.corda.flows.CashFlow
|
import net.corda.flows.CashFlow
|
||||||
import net.corda.node.driver.DriverBasedTest
|
import net.corda.node.internal.Node
|
||||||
import net.corda.node.driver.NodeHandle
|
|
||||||
import net.corda.node.driver.driver
|
|
||||||
import net.corda.node.services.User
|
import net.corda.node.services.User
|
||||||
|
import net.corda.node.services.config.configureTestSSL
|
||||||
import net.corda.node.services.messaging.CordaRPCClient
|
import net.corda.node.services.messaging.CordaRPCClient
|
||||||
import net.corda.node.services.startFlowPermission
|
import net.corda.node.services.startFlowPermission
|
||||||
import net.corda.node.services.transactions.ValidatingNotaryService
|
import net.corda.node.services.transactions.ValidatingNotaryService
|
||||||
|
import net.corda.testing.node.NodeBasedTest
|
||||||
import org.apache.activemq.artemis.api.core.ActiveMQSecurityException
|
import org.apache.activemq.artemis.api.core.ActiveMQSecurityException
|
||||||
import org.assertj.core.api.Assertions.assertThatExceptionOfType
|
import org.assertj.core.api.Assertions.assertThatExceptionOfType
|
||||||
|
import org.junit.Before
|
||||||
import org.junit.Test
|
import org.junit.Test
|
||||||
|
|
||||||
class CordaRPCClientTest : DriverBasedTest() {
|
class CordaRPCClientTest : NodeBasedTest() {
|
||||||
private val rpcUser = User("user1", "test", permissions = setOf(startFlowPermission<CashFlow>()))
|
private val rpcUser = User("user1", "test", permissions = setOf(startFlowPermission<CashFlow>()))
|
||||||
private lateinit var node: NodeHandle
|
private lateinit var node: Node
|
||||||
private lateinit var client: CordaRPCClient
|
private lateinit var client: CordaRPCClient
|
||||||
|
|
||||||
override fun setup() = driver(isDebug = true) {
|
@Before
|
||||||
node = startNode(rpcUsers = listOf(rpcUser), advertisedServices = setOf(ServiceInfo(ValidatingNotaryService.type))).getOrThrow()
|
fun setUp() {
|
||||||
client = node.rpcClientToNode()
|
node = startNode("Alice", rpcUsers = listOf(rpcUser), advertisedServices = setOf(ServiceInfo(ValidatingNotaryService.type))).getOrThrow()
|
||||||
runTest()
|
client = CordaRPCClient(node.configuration.artemisAddress, configureTestSSL())
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
@ -50,7 +53,7 @@ class CordaRPCClientTest : DriverBasedTest() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
fun `indefinite block bug`() {
|
fun `close-send deadlock and premature shutdown on empty observable`() {
|
||||||
println("Starting client")
|
println("Starting client")
|
||||||
client.start(rpcUser.username, rpcUser.password)
|
client.start(rpcUser.username, rpcUser.password)
|
||||||
println("Creating proxy")
|
println("Creating proxy")
|
||||||
@ -58,11 +61,25 @@ class CordaRPCClientTest : DriverBasedTest() {
|
|||||||
println("Starting flow")
|
println("Starting flow")
|
||||||
val flowHandle = proxy.startFlow(
|
val flowHandle = proxy.startFlow(
|
||||||
::CashFlow,
|
::CashFlow,
|
||||||
CashCommand.IssueCash(20.DOLLARS, OpaqueBytes.of(0), node.nodeInfo.legalIdentity, node.nodeInfo.legalIdentity))
|
CashCommand.IssueCash(20.DOLLARS, OpaqueBytes.of(0), node.info.legalIdentity, node.info.legalIdentity))
|
||||||
println("Started flow, waiting on result")
|
println("Started flow, waiting on result")
|
||||||
flowHandle.progress.subscribe {
|
flowHandle.progress.subscribe {
|
||||||
println("PROGRESS $it")
|
println("PROGRESS $it")
|
||||||
}
|
}
|
||||||
println("Result: ${flowHandle.returnValue.toBlocking().first()}")
|
println("Result: ${flowHandle.returnValue.getOrThrow()}")
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun `FlowException thrown by flow`() {
|
||||||
|
client.start(rpcUser.username, rpcUser.password)
|
||||||
|
val proxy = client.proxy()
|
||||||
|
val handle = proxy.startFlow(::CashFlow, CashCommand.PayCash(
|
||||||
|
amount = 100.DOLLARS.issuedBy(node.info.legalIdentity.ref(1)),
|
||||||
|
recipient = node.info.legalIdentity
|
||||||
|
))
|
||||||
|
// TODO Restrict this to CashException once RPC serialisation has been fixed
|
||||||
|
assertThatExceptionOfType(FlowException::class.java).isThrownBy {
|
||||||
|
handle.returnValue.getOrThrow()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -124,7 +124,7 @@ class NodeMonitorModelTest : DriverBasedTest() {
|
|||||||
issueRef = OpaqueBytes(ByteArray(1, { 1 })),
|
issueRef = OpaqueBytes(ByteArray(1, { 1 })),
|
||||||
recipient = aliceNode.legalIdentity,
|
recipient = aliceNode.legalIdentity,
|
||||||
notary = notaryNode.notaryIdentity
|
notary = notaryNode.notaryIdentity
|
||||||
)).returnValue.toBlocking().first()
|
)).returnValue.getOrThrow()
|
||||||
|
|
||||||
rpc.startFlow(::CashFlow, CashCommand.PayCash(
|
rpc.startFlow(::CashFlow, CashCommand.PayCash(
|
||||||
amount = Amount(100, Issued(PartyAndReference(aliceNode.legalIdentity, OpaqueBytes(ByteArray(1, { 1 }))), USD)),
|
amount = Amount(100, Issued(PartyAndReference(aliceNode.legalIdentity, OpaqueBytes(ByteArray(1, { 1 }))), USD)),
|
||||||
|
@ -6,10 +6,7 @@ package net.corda.core
|
|||||||
import com.google.common.base.Function
|
import com.google.common.base.Function
|
||||||
import com.google.common.base.Throwables
|
import com.google.common.base.Throwables
|
||||||
import com.google.common.io.ByteStreams
|
import com.google.common.io.ByteStreams
|
||||||
import com.google.common.util.concurrent.Futures
|
import com.google.common.util.concurrent.*
|
||||||
import com.google.common.util.concurrent.ListenableFuture
|
|
||||||
import com.google.common.util.concurrent.MoreExecutors
|
|
||||||
import com.google.common.util.concurrent.SettableFuture
|
|
||||||
import kotlinx.support.jdk7.use
|
import kotlinx.support.jdk7.use
|
||||||
import net.corda.core.crypto.newSecureRandom
|
import net.corda.core.crypto.newSecureRandom
|
||||||
import org.slf4j.Logger
|
import org.slf4j.Logger
|
||||||
@ -115,7 +112,7 @@ inline fun <T> SettableFuture<T>.catch(block: () -> T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fun <A> ListenableFuture<A>.toObservable(): Observable<A> {
|
fun <A> ListenableFuture<out A>.toObservable(): Observable<A> {
|
||||||
return Observable.create { subscriber ->
|
return Observable.create { subscriber ->
|
||||||
success {
|
success {
|
||||||
subscriber.onNext(it)
|
subscriber.onNext(it)
|
||||||
@ -384,15 +381,24 @@ fun <T> Observer<T>.tee(vararg teeTo: Observer<T>): Observer<T> {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns a [ListenableFuture] bound to the *first* item emitted by this Observable. The future will complete with a
|
* Returns a [ListenableFuture] bound to the *first* item emitted by this Observable. The future will complete with a
|
||||||
* NoSuchElementException if no items are emitted or any other error thrown by the Observable.
|
* NoSuchElementException if no items are emitted or any other error thrown by the Observable. If it's cancelled then
|
||||||
|
* it will unsubscribe from the observable.
|
||||||
*/
|
*/
|
||||||
fun <T> Observable<T>.toFuture(): ListenableFuture<T> {
|
fun <T> Observable<T>.toFuture(): ListenableFuture<T> = ObservableToFuture(this)
|
||||||
val future = SettableFuture.create<T>()
|
|
||||||
first().subscribe(
|
private class ObservableToFuture<T>(observable: Observable<T>) : AbstractFuture<T>(), Observer<T> {
|
||||||
{ future.set(it) },
|
private val subscription = observable.first().subscribe(this)
|
||||||
{ future.setException(it) }
|
override fun onNext(value: T) {
|
||||||
)
|
set(value)
|
||||||
return future
|
}
|
||||||
|
override fun onError(e: Throwable) {
|
||||||
|
setException(e)
|
||||||
|
}
|
||||||
|
override fun cancel(mayInterruptIfRunning: Boolean): Boolean {
|
||||||
|
subscription.unsubscribe()
|
||||||
|
return super.cancel(mayInterruptIfRunning)
|
||||||
|
}
|
||||||
|
override fun onCompleted() {}
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Return the sum of an Iterable of [BigDecimal]s. */
|
/** Return the sum of an Iterable of [BigDecimal]s. */
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
package net.corda.core.messaging
|
package net.corda.core.messaging
|
||||||
|
|
||||||
|
import com.google.common.util.concurrent.ListenableFuture
|
||||||
import net.corda.core.contracts.ContractState
|
import net.corda.core.contracts.ContractState
|
||||||
import net.corda.core.contracts.StateAndRef
|
import net.corda.core.contracts.StateAndRef
|
||||||
import net.corda.core.crypto.CompositeKey
|
import net.corda.core.crypto.CompositeKey
|
||||||
@ -107,15 +108,16 @@ interface CordaRPCOps : RPCOps {
|
|||||||
fun uploadFile(dataType: String, name: String?, file: InputStream): String
|
fun uploadFile(dataType: String, name: String?, file: InputStream): String
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the node-local current time.
|
* Returns the node's current time.
|
||||||
*/
|
*/
|
||||||
fun currentNodeTime(): Instant
|
fun currentNodeTime(): Instant
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns an Observable emitting a single Unit once the node is registered with the network map.
|
* Returns a [ListenableFuture] which completes when the node has registered wih the network map service. It can also
|
||||||
|
* complete with an exception if it is unable to.
|
||||||
*/
|
*/
|
||||||
@RPCReturnsObservables
|
@RPCReturnsObservables
|
||||||
fun waitUntilRegisteredWithNetworkMap(): Observable<Unit>
|
fun waitUntilRegisteredWithNetworkMap(): ListenableFuture<Unit>
|
||||||
|
|
||||||
// TODO These need rethinking. Instead of these direct calls we should have a way of replicating a subset of
|
// TODO These need rethinking. Instead of these direct calls we should have a way of replicating a subset of
|
||||||
// the node's state locally and query that directly.
|
// the node's state locally and query that directly.
|
||||||
@ -179,13 +181,10 @@ inline fun <T : Any, A, B, C, D, reified R : FlowLogic<T>> CordaRPCOps.startFlow
|
|||||||
*
|
*
|
||||||
* @param id The started state machine's ID.
|
* @param id The started state machine's ID.
|
||||||
* @param progress The stream of progress tracker events.
|
* @param progress The stream of progress tracker events.
|
||||||
* @param returnValue An Observable emitting a single event containing the flow's return value.
|
* @param returnValue A [ListenableFuture] of the flow's return value.
|
||||||
* To block on this value:
|
|
||||||
* val returnValue = rpc.startFlow(::MyFlow).returnValue.toBlocking().first()
|
|
||||||
*/
|
*/
|
||||||
data class FlowHandle<A>(
|
data class FlowHandle<A>(
|
||||||
val id: StateMachineRunId,
|
val id: StateMachineRunId,
|
||||||
val progress: Observable<String>,
|
val progress: Observable<String>,
|
||||||
// TODO This should be ListenableFuture<A>
|
val returnValue: ListenableFuture<A>
|
||||||
val returnValue: Observable<A>
|
|
||||||
)
|
)
|
||||||
|
@ -72,12 +72,7 @@ interface NetworkMapCache {
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
/** Look up the node info for a specific peer key. */
|
/** Look up the node info for a specific peer key. */
|
||||||
fun getNodeByLegalIdentityKey(compositeKey: CompositeKey): NodeInfo? {
|
fun getNodeByLegalIdentityKey(compositeKey: CompositeKey): NodeInfo?
|
||||||
// Although we should never have more than one match, it is theoretically possible. Report an error if it happens.
|
|
||||||
val candidates = partyNodes.filter { it.legalIdentity.owningKey == compositeKey }
|
|
||||||
check(candidates.size <= 1) { "Found more than one match for key $compositeKey" }
|
|
||||||
return candidates.singleOrNull()
|
|
||||||
}
|
|
||||||
/** Look up all nodes advertising the service owned by [compositeKey] */
|
/** Look up all nodes advertising the service owned by [compositeKey] */
|
||||||
fun getNodesByAdvertisedServiceIdentityKey(compositeKey: CompositeKey): List<NodeInfo> {
|
fun getNodesByAdvertisedServiceIdentityKey(compositeKey: CompositeKey): List<NodeInfo> {
|
||||||
return partyNodes.filter { it.advertisedServices.any { it.identity.owningKey == compositeKey } }
|
return partyNodes.filter { it.advertisedServices.any { it.identity.owningKey == compositeKey } }
|
||||||
|
@ -5,6 +5,7 @@ import co.paralleluniverse.io.serialization.kryo.KryoSerializer
|
|||||||
import com.esotericsoftware.kryo.Kryo
|
import com.esotericsoftware.kryo.Kryo
|
||||||
import com.esotericsoftware.kryo.Kryo.DefaultInstantiatorStrategy
|
import com.esotericsoftware.kryo.Kryo.DefaultInstantiatorStrategy
|
||||||
import com.esotericsoftware.kryo.KryoException
|
import com.esotericsoftware.kryo.KryoException
|
||||||
|
import com.esotericsoftware.kryo.Registration
|
||||||
import com.esotericsoftware.kryo.Serializer
|
import com.esotericsoftware.kryo.Serializer
|
||||||
import com.esotericsoftware.kryo.io.Input
|
import com.esotericsoftware.kryo.io.Input
|
||||||
import com.esotericsoftware.kryo.io.Output
|
import com.esotericsoftware.kryo.io.Output
|
||||||
@ -66,7 +67,7 @@ import kotlin.reflect.jvm.javaType
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
// A convenient instance of Kryo pre-configured with some useful things. Used as a default by various functions.
|
// A convenient instance of Kryo pre-configured with some useful things. Used as a default by various functions.
|
||||||
val THREAD_LOCAL_KRYO = ThreadLocal.withInitial { createKryo() }
|
val THREAD_LOCAL_KRYO: ThreadLocal<Kryo> = ThreadLocal.withInitial { createKryo() }
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A type safe wrapper around a byte array that contains a serialised object. You can call [SerializedBytes.deserialize]
|
* A type safe wrapper around a byte array that contains a serialised object. You can call [SerializedBytes.deserialize]
|
||||||
@ -76,7 +77,7 @@ class SerializedBytes<T : Any>(bytes: ByteArray) : OpaqueBytes(bytes) {
|
|||||||
// It's OK to use lazy here because SerializedBytes is configured to use the ImmutableClassSerializer.
|
// It's OK to use lazy here because SerializedBytes is configured to use the ImmutableClassSerializer.
|
||||||
val hash: SecureHash by lazy { bytes.sha256() }
|
val hash: SecureHash by lazy { bytes.sha256() }
|
||||||
|
|
||||||
fun writeToFile(path: Path) = Files.write(path, bytes)
|
fun writeToFile(path: Path): Path = Files.write(path, bytes)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Some extension functions that make deserialisation convenient and provide auto-casting of the result.
|
// Some extension functions that make deserialisation convenient and provide auto-casting of the result.
|
||||||
@ -385,8 +386,7 @@ object KotlinObjectSerializer : Serializer<DeserializeAsKotlinObjectDef>() {
|
|||||||
return type.getField("INSTANCE").get(null) as DeserializeAsKotlinObjectDef
|
return type.getField("INSTANCE").get(null) as DeserializeAsKotlinObjectDef
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun write(kryo: Kryo, output: Output, obj: DeserializeAsKotlinObjectDef) {
|
override fun write(kryo: Kryo, output: Output, obj: DeserializeAsKotlinObjectDef) {}
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fun createKryo(k: Kryo = Kryo()): Kryo {
|
fun createKryo(k: Kryo = Kryo()): Kryo {
|
||||||
@ -402,12 +402,10 @@ fun createKryo(k: Kryo = Kryo()): Kryo {
|
|||||||
// Because we like to stick a Kryo object in a ThreadLocal to speed things up a bit, we can end up trying to
|
// Because we like to stick a Kryo object in a ThreadLocal to speed things up a bit, we can end up trying to
|
||||||
// serialise the Kryo object itself when suspending a fiber. That's dumb, useless AND can cause crashes, so
|
// serialise the Kryo object itself when suspending a fiber. That's dumb, useless AND can cause crashes, so
|
||||||
// we avoid it here.
|
// we avoid it here.
|
||||||
register(Kryo::class.java, object : Serializer<Kryo>() {
|
register(Kryo::class,
|
||||||
override fun read(kryo: Kryo, input: Input, type: Class<Kryo>): Kryo {
|
read = { kryo, input -> createKryo((Fiber.getFiberSerializer() as KryoSerializer).kryo) },
|
||||||
return createKryo((Fiber.getFiberSerializer() as KryoSerializer).kryo)
|
write = { kryo, output, obj -> }
|
||||||
}
|
)
|
||||||
override fun write(kryo: Kryo, output: Output, obj: Kryo) {}
|
|
||||||
})
|
|
||||||
|
|
||||||
register(EdDSAPublicKey::class.java, Ed25519PublicKeySerializer)
|
register(EdDSAPublicKey::class.java, Ed25519PublicKeySerializer)
|
||||||
register(EdDSAPrivateKey::class.java, Ed25519PrivateKeySerializer)
|
register(EdDSAPrivateKey::class.java, Ed25519PrivateKeySerializer)
|
||||||
@ -435,6 +433,8 @@ fun createKryo(k: Kryo = Kryo()): Kryo {
|
|||||||
// This ensures a NonEmptySetSerializer is constructed with an initial value.
|
// This ensures a NonEmptySetSerializer is constructed with an initial value.
|
||||||
register(NonEmptySet::class.java, NonEmptySetSerializer)
|
register(NonEmptySet::class.java, NonEmptySetSerializer)
|
||||||
|
|
||||||
|
register(Array<StackTraceElement>::class, read = { kryo, input -> emptyArray() }, write = { kryo, output, o -> })
|
||||||
|
|
||||||
/** This ensures any kotlin objects that implement [DeserializeAsKotlinObjectDef] are read back in as singletons. */
|
/** This ensures any kotlin objects that implement [DeserializeAsKotlinObjectDef] are read back in as singletons. */
|
||||||
addDefaultSerializer(DeserializeAsKotlinObjectDef::class.java, KotlinObjectSerializer)
|
addDefaultSerializer(DeserializeAsKotlinObjectDef::class.java, KotlinObjectSerializer)
|
||||||
|
|
||||||
@ -451,6 +451,19 @@ fun createKryo(k: Kryo = Kryo()): Kryo {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
inline fun <T : Any> Kryo.register(
|
||||||
|
type: KClass<T>,
|
||||||
|
crossinline read: (Kryo, Input) -> T,
|
||||||
|
crossinline write: (Kryo, Output, T) -> Unit): Registration {
|
||||||
|
return register(
|
||||||
|
type.java,
|
||||||
|
object : Serializer<T>() {
|
||||||
|
override fun read(kryo: Kryo, input: Input, type: Class<T>): T = read(kryo, input)
|
||||||
|
override fun write(kryo: Kryo, output: Output, obj: T) = write(kryo, output, obj)
|
||||||
|
}
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Use this method to mark any types which can have the same instance within it more than once. This will make sure
|
* Use this method to mark any types which can have the same instance within it more than once. This will make sure
|
||||||
* the serialised form is stable across multiple serialise-deserialise cycles. Using this on a type with internal cyclic
|
* the serialised form is stable across multiple serialise-deserialise cycles. Using this on a type with internal cyclic
|
||||||
@ -517,7 +530,7 @@ var Kryo.attachmentStorage: AttachmentStorage?
|
|||||||
//Used in Merkle tree calculation. It doesn't cover all the cases of unstable serialization format.
|
//Used in Merkle tree calculation. It doesn't cover all the cases of unstable serialization format.
|
||||||
fun extendKryoHash(kryo: Kryo): Kryo {
|
fun extendKryoHash(kryo: Kryo): Kryo {
|
||||||
return kryo.apply {
|
return kryo.apply {
|
||||||
setReferences(false)
|
references = false
|
||||||
register(LinkedHashMap::class.java, MapSerializer())
|
register(LinkedHashMap::class.java, MapSerializer())
|
||||||
register(HashMap::class.java, OrderedSerializer)
|
register(HashMap::class.java, OrderedSerializer)
|
||||||
}
|
}
|
||||||
|
@ -4,6 +4,7 @@ import org.assertj.core.api.Assertions.*
|
|||||||
import org.junit.Test
|
import org.junit.Test
|
||||||
import rx.subjects.PublishSubject
|
import rx.subjects.PublishSubject
|
||||||
import java.util.*
|
import java.util.*
|
||||||
|
import java.util.concurrent.CancellationException
|
||||||
|
|
||||||
class UtilsTest {
|
class UtilsTest {
|
||||||
@Test
|
@Test
|
||||||
@ -44,4 +45,16 @@ class UtilsTest {
|
|||||||
future.getOrThrow()
|
future.getOrThrow()
|
||||||
}.isSameAs(exception)
|
}.isSameAs(exception)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun `toFuture - cancel`() {
|
||||||
|
val subject = PublishSubject.create<String>()
|
||||||
|
val future = subject.toFuture()
|
||||||
|
future.cancel(false)
|
||||||
|
assertThat(subject.hasObservers()).isFalse()
|
||||||
|
subject.onNext("Hello")
|
||||||
|
assertThatExceptionOfType(CancellationException::class.java).isThrownBy {
|
||||||
|
future.get()
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
@ -46,6 +46,14 @@ through to the server where the corresponding server-side observables are also u
|
|||||||
a warning printed to the logs and the proxy will be closed for you. But don't rely on this, as garbage
|
a warning printed to the logs and the proxy will be closed for you. But don't rely on this, as garbage
|
||||||
collection is non-deterministic.
|
collection is non-deterministic.
|
||||||
|
|
||||||
|
Futures
|
||||||
|
-------
|
||||||
|
|
||||||
|
A method can also return a ``ListenableFuture`` in its object graph and it will be treated in a similar manner to
|
||||||
|
observables, including needing to mark the RPC with the ``@RPCReturnsObservables`` annotation. Unlike for an observable,
|
||||||
|
once the single value (or an exception) has been received all server-side resources will be released automatically. Calling
|
||||||
|
the ``cancel`` method on the future will unsubscribe it from any future value and release any resources.
|
||||||
|
|
||||||
Versioning
|
Versioning
|
||||||
----------
|
----------
|
||||||
|
|
||||||
|
@ -9,7 +9,6 @@ import net.corda.core.messaging.startFlow
|
|||||||
import net.corda.core.node.services.ServiceInfo
|
import net.corda.core.node.services.ServiceInfo
|
||||||
import net.corda.core.node.services.Vault
|
import net.corda.core.node.services.Vault
|
||||||
import net.corda.core.serialization.OpaqueBytes
|
import net.corda.core.serialization.OpaqueBytes
|
||||||
import net.corda.core.toFuture
|
|
||||||
import net.corda.flows.CashCommand
|
import net.corda.flows.CashCommand
|
||||||
import net.corda.flows.CashFlow
|
import net.corda.flows.CashFlow
|
||||||
import net.corda.node.driver.driver
|
import net.corda.node.driver.driver
|
||||||
@ -87,7 +86,7 @@ class IntegrationTestingTutorial {
|
|||||||
amount = i.DOLLARS.issuedBy(alice.nodeInfo.legalIdentity.ref(issueRef)),
|
amount = i.DOLLARS.issuedBy(alice.nodeInfo.legalIdentity.ref(issueRef)),
|
||||||
recipient = alice.nodeInfo.legalIdentity
|
recipient = alice.nodeInfo.legalIdentity
|
||||||
))
|
))
|
||||||
flowHandle.returnValue.toFuture().getOrThrow()
|
flowHandle.returnValue.getOrThrow()
|
||||||
}
|
}
|
||||||
|
|
||||||
aliceVaultUpdates.expectEvents {
|
aliceVaultUpdates.expectEvents {
|
||||||
|
@ -54,9 +54,11 @@ task buildCordaJAR(type: FatCapsule, dependsOn: ['buildCertSigningRequestUtility
|
|||||||
applicationSource = files(project.tasks.findByName('jar'), '../build/classes/main/CordaCaplet.class', 'config/dev/log4j2.xml')
|
applicationSource = files(project.tasks.findByName('jar'), '../build/classes/main/CordaCaplet.class', 'config/dev/log4j2.xml')
|
||||||
|
|
||||||
capsuleManifest {
|
capsuleManifest {
|
||||||
|
applicationVersion = corda_version
|
||||||
appClassPath = ["jolokia-agent-war-${project.rootProject.ext.jolokia_version}.war"]
|
appClassPath = ["jolokia-agent-war-${project.rootProject.ext.jolokia_version}.war"]
|
||||||
javaAgents = ["quasar-core-${quasar_version}-jdk8.jar"]
|
javaAgents = ["quasar-core-${quasar_version}-jdk8.jar"]
|
||||||
systemProperties['visualvm.display.name'] = 'Corda'
|
systemProperties['visualvm.display.name'] = 'Corda'
|
||||||
|
systemProperties['corda.version'] = corda_version
|
||||||
minJavaVersion = '1.8.0'
|
minJavaVersion = '1.8.0'
|
||||||
// This version is known to work and avoids earlier 8u versions that have bugs.
|
// This version is known to work and avoids earlier 8u versions that have bugs.
|
||||||
minUpdateVersion['1.8'] = '102'
|
minUpdateVersion['1.8'] = '102'
|
||||||
|
@ -11,7 +11,6 @@ import net.corda.core.messaging.StateMachineUpdate
|
|||||||
import net.corda.core.messaging.startFlow
|
import net.corda.core.messaging.startFlow
|
||||||
import net.corda.core.node.NodeInfo
|
import net.corda.core.node.NodeInfo
|
||||||
import net.corda.core.serialization.OpaqueBytes
|
import net.corda.core.serialization.OpaqueBytes
|
||||||
import net.corda.core.toFuture
|
|
||||||
import net.corda.flows.CashCommand
|
import net.corda.flows.CashCommand
|
||||||
import net.corda.flows.CashFlow
|
import net.corda.flows.CashFlow
|
||||||
import net.corda.node.driver.DriverBasedTest
|
import net.corda.node.driver.DriverBasedTest
|
||||||
@ -138,13 +137,13 @@ class DistributedServiceTests : DriverBasedTest() {
|
|||||||
val issueHandle = aliceProxy.startFlow(
|
val issueHandle = aliceProxy.startFlow(
|
||||||
::CashFlow,
|
::CashFlow,
|
||||||
CashCommand.IssueCash(amount, OpaqueBytes.of(0), alice.nodeInfo.legalIdentity, raftNotaryIdentity))
|
CashCommand.IssueCash(amount, OpaqueBytes.of(0), alice.nodeInfo.legalIdentity, raftNotaryIdentity))
|
||||||
issueHandle.returnValue.toFuture().getOrThrow()
|
issueHandle.returnValue.getOrThrow()
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun paySelf(amount: Amount<Currency>) {
|
private fun paySelf(amount: Amount<Currency>) {
|
||||||
val payHandle = aliceProxy.startFlow(
|
val payHandle = aliceProxy.startFlow(
|
||||||
::CashFlow,
|
::CashFlow,
|
||||||
CashCommand.PayCash(amount.issuedBy(alice.nodeInfo.legalIdentity.ref(0)), alice.nodeInfo.legalIdentity))
|
CashCommand.PayCash(amount.issuedBy(alice.nodeInfo.legalIdentity.ref(0)), alice.nodeInfo.legalIdentity))
|
||||||
payHandle.returnValue.toFuture().getOrThrow()
|
payHandle.returnValue.getOrThrow()
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -380,7 +380,7 @@ open class DriverDSL(
|
|||||||
registerProcess(processFuture)
|
registerProcess(processFuture)
|
||||||
return processFuture.flatMap { process ->
|
return processFuture.flatMap { process ->
|
||||||
establishRpc(messagingAddress, configuration).flatMap { rpc ->
|
establishRpc(messagingAddress, configuration).flatMap { rpc ->
|
||||||
rpc.waitUntilRegisteredWithNetworkMap().toFuture().map {
|
rpc.waitUntilRegisteredWithNetworkMap().map {
|
||||||
NodeHandle(rpc.nodeIdentity(), rpc, configuration, process)
|
NodeHandle(rpc.nodeIdentity(), rpc, configuration, process)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -15,7 +15,6 @@ import net.corda.core.node.ServiceHub
|
|||||||
import net.corda.core.node.services.NetworkMapCache
|
import net.corda.core.node.services.NetworkMapCache
|
||||||
import net.corda.core.node.services.StateMachineTransactionMapping
|
import net.corda.core.node.services.StateMachineTransactionMapping
|
||||||
import net.corda.core.node.services.Vault
|
import net.corda.core.node.services.Vault
|
||||||
import net.corda.core.toObservable
|
|
||||||
import net.corda.core.transactions.SignedTransaction
|
import net.corda.core.transactions.SignedTransaction
|
||||||
import net.corda.node.services.messaging.requirePermission
|
import net.corda.node.services.messaging.requirePermission
|
||||||
import net.corda.node.services.startFlowPermission
|
import net.corda.node.services.startFlowPermission
|
||||||
@ -97,7 +96,7 @@ class CordaRPCOpsImpl(
|
|||||||
return FlowHandle(
|
return FlowHandle(
|
||||||
id = stateMachine.id,
|
id = stateMachine.id,
|
||||||
progress = stateMachine.logic.track()?.second ?: Observable.empty(),
|
progress = stateMachine.logic.track()?.second ?: Observable.empty(),
|
||||||
returnValue = stateMachine.resultFuture.toObservable()
|
returnValue = stateMachine.resultFuture
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -111,7 +110,7 @@ class CordaRPCOpsImpl(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun waitUntilRegisteredWithNetworkMap() = services.networkMapCache.mapServiceRegistered.toObservable()
|
override fun waitUntilRegisteredWithNetworkMap() = services.networkMapCache.mapServiceRegistered
|
||||||
override fun partyFromKey(key: CompositeKey) = services.identityService.partyFromKey(key)
|
override fun partyFromKey(key: CompositeKey) = services.identityService.partyFromKey(key)
|
||||||
override fun partyFromName(name: String) = services.identityService.partyFromName(name)
|
override fun partyFromName(name: String) = services.identityService.partyFromName(name)
|
||||||
|
|
||||||
|
@ -31,6 +31,8 @@ import org.jetbrains.exposed.sql.Database
|
|||||||
import java.io.RandomAccessFile
|
import java.io.RandomAccessFile
|
||||||
import java.lang.management.ManagementFactory
|
import java.lang.management.ManagementFactory
|
||||||
import java.nio.channels.FileLock
|
import java.nio.channels.FileLock
|
||||||
|
import java.nio.file.Files
|
||||||
|
import java.nio.file.Paths
|
||||||
import java.time.Clock
|
import java.time.Clock
|
||||||
import javax.management.ObjectName
|
import javax.management.ObjectName
|
||||||
import javax.servlet.*
|
import javax.servlet.*
|
||||||
@ -100,6 +102,27 @@ class Node(override val configuration: FullNodeConfiguration,
|
|||||||
|
|
||||||
private lateinit var userService: RPCUserService
|
private lateinit var userService: RPCUserService
|
||||||
|
|
||||||
|
init {
|
||||||
|
checkVersionUnchanged()
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Abort starting the node if an existing deployment with a different version is detected in the current directory.
|
||||||
|
* The current version is expected to be specified as a system property. If not provided, the check will be ignored.
|
||||||
|
*/
|
||||||
|
private fun checkVersionUnchanged() {
|
||||||
|
val currentVersion = System.getProperty("corda.version") ?: return
|
||||||
|
val versionFile = Paths.get("version")
|
||||||
|
if (Files.exists(versionFile)) {
|
||||||
|
val existingVersion = Files.readAllLines(versionFile)[0]
|
||||||
|
check(existingVersion == currentVersion) {
|
||||||
|
"Version change detected - current: $currentVersion, existing: $existingVersion. Node upgrades are not yet supported."
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
Files.write(versionFile, currentVersion.toByteArray())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
override fun makeMessagingService(): MessagingServiceInternal {
|
override fun makeMessagingService(): MessagingServiceInternal {
|
||||||
userService = RPCUserServiceImpl(configuration)
|
userService = RPCUserServiceImpl(configuration)
|
||||||
|
|
||||||
|
@ -8,9 +8,10 @@ import com.esotericsoftware.kryo.Registration
|
|||||||
import com.esotericsoftware.kryo.Serializer
|
import com.esotericsoftware.kryo.Serializer
|
||||||
import com.esotericsoftware.kryo.io.Input
|
import com.esotericsoftware.kryo.io.Input
|
||||||
import com.esotericsoftware.kryo.io.Output
|
import com.esotericsoftware.kryo.io.Output
|
||||||
import com.esotericsoftware.kryo.serializers.JavaSerializer
|
|
||||||
import com.google.common.net.HostAndPort
|
import com.google.common.net.HostAndPort
|
||||||
|
import com.google.common.util.concurrent.ListenableFuture
|
||||||
import de.javakaffee.kryoserializers.ArraysAsListSerializer
|
import de.javakaffee.kryoserializers.ArraysAsListSerializer
|
||||||
|
import de.javakaffee.kryoserializers.UnmodifiableCollectionsSerializer
|
||||||
import de.javakaffee.kryoserializers.guava.*
|
import de.javakaffee.kryoserializers.guava.*
|
||||||
import net.corda.contracts.asset.Cash
|
import net.corda.contracts.asset.Cash
|
||||||
import net.corda.core.ErrorOr
|
import net.corda.core.ErrorOr
|
||||||
@ -19,6 +20,8 @@ import net.corda.core.crypto.CompositeKey
|
|||||||
import net.corda.core.crypto.DigitalSignature
|
import net.corda.core.crypto.DigitalSignature
|
||||||
import net.corda.core.crypto.Party
|
import net.corda.core.crypto.Party
|
||||||
import net.corda.core.crypto.SecureHash
|
import net.corda.core.crypto.SecureHash
|
||||||
|
import net.corda.core.flows.FlowException
|
||||||
|
import net.corda.core.flows.IllegalFlowLogicException
|
||||||
import net.corda.core.flows.StateMachineRunId
|
import net.corda.core.flows.StateMachineRunId
|
||||||
import net.corda.core.messaging.FlowHandle
|
import net.corda.core.messaging.FlowHandle
|
||||||
import net.corda.core.messaging.StateMachineInfo
|
import net.corda.core.messaging.StateMachineInfo
|
||||||
@ -26,12 +29,15 @@ import net.corda.core.messaging.StateMachineUpdate
|
|||||||
import net.corda.core.node.*
|
import net.corda.core.node.*
|
||||||
import net.corda.core.node.services.*
|
import net.corda.core.node.services.*
|
||||||
import net.corda.core.serialization.*
|
import net.corda.core.serialization.*
|
||||||
|
import net.corda.core.toFuture
|
||||||
|
import net.corda.core.toObservable
|
||||||
import net.corda.core.transactions.SignedTransaction
|
import net.corda.core.transactions.SignedTransaction
|
||||||
import net.corda.core.transactions.WireTransaction
|
import net.corda.core.transactions.WireTransaction
|
||||||
import net.corda.node.internal.AbstractNode
|
import net.corda.node.internal.AbstractNode
|
||||||
import net.corda.node.services.User
|
import net.corda.node.services.User
|
||||||
import net.corda.node.services.messaging.ArtemisMessagingComponent.Companion.NODE_USER
|
import net.corda.node.services.messaging.ArtemisMessagingComponent.Companion.NODE_USER
|
||||||
import net.corda.node.services.messaging.ArtemisMessagingComponent.NetworkMapAddress
|
import net.corda.node.services.messaging.ArtemisMessagingComponent.NetworkMapAddress
|
||||||
|
import net.corda.node.services.statemachine.FlowSessionException
|
||||||
import net.i2p.crypto.eddsa.EdDSAPrivateKey
|
import net.i2p.crypto.eddsa.EdDSAPrivateKey
|
||||||
import net.i2p.crypto.eddsa.EdDSAPublicKey
|
import net.i2p.crypto.eddsa.EdDSAPublicKey
|
||||||
import org.apache.activemq.artemis.api.core.SimpleString
|
import org.apache.activemq.artemis.api.core.SimpleString
|
||||||
@ -138,6 +144,7 @@ private class RPCKryo(observableSerializer: Serializer<Observable<Any>>? = null)
|
|||||||
register(Array<Any>(0,{}).javaClass)
|
register(Array<Any>(0,{}).javaClass)
|
||||||
register(Class::class.java, ClassSerializer)
|
register(Class::class.java, ClassSerializer)
|
||||||
|
|
||||||
|
UnmodifiableCollectionsSerializer.registerSerializers(this)
|
||||||
ImmutableListSerializer.registerSerializers(this)
|
ImmutableListSerializer.registerSerializers(this)
|
||||||
ImmutableSetSerializer.registerSerializers(this)
|
ImmutableSetSerializer.registerSerializers(this)
|
||||||
ImmutableSortedSetSerializer.registerSerializers(this)
|
ImmutableSortedSetSerializer.registerSerializers(this)
|
||||||
@ -207,16 +214,18 @@ private class RPCKryo(observableSerializer: Serializer<Observable<Any>>? = null)
|
|||||||
register(SimpleString::class.java)
|
register(SimpleString::class.java)
|
||||||
register(ServiceEntry::class.java)
|
register(ServiceEntry::class.java)
|
||||||
// Exceptions. We don't bother sending the stack traces as the client will fill in its own anyway.
|
// Exceptions. We don't bother sending the stack traces as the client will fill in its own anyway.
|
||||||
|
register(Array<StackTraceElement>::class, read = { kryo, input -> emptyArray() }, write = { kryo, output, obj -> })
|
||||||
|
register(FlowException::class.java)
|
||||||
|
register(FlowSessionException::class.java)
|
||||||
|
register(IllegalFlowLogicException::class.java)
|
||||||
register(RuntimeException::class.java)
|
register(RuntimeException::class.java)
|
||||||
register(IllegalArgumentException::class.java)
|
register(IllegalArgumentException::class.java)
|
||||||
register(ArrayIndexOutOfBoundsException::class.java)
|
register(ArrayIndexOutOfBoundsException::class.java)
|
||||||
register(IndexOutOfBoundsException::class.java)
|
register(IndexOutOfBoundsException::class.java)
|
||||||
// Kryo couldn't serialize Collections.unmodifiableCollection in Throwable correctly, causing null pointer exception when try to access the deserialize object.
|
register(NoSuchElementException::class.java)
|
||||||
register(NoSuchElementException::class.java, JavaSerializer())
|
|
||||||
register(RPCException::class.java)
|
register(RPCException::class.java)
|
||||||
register(Array<StackTraceElement>::class.java, read = { kryo, input -> emptyArray() }, write = { kryo, output, o -> })
|
|
||||||
register(Collections.unmodifiableList(emptyList<String>()).javaClass)
|
|
||||||
register(PermissionException::class.java)
|
register(PermissionException::class.java)
|
||||||
|
register(Throwable::class.java)
|
||||||
register(FlowHandle::class.java)
|
register(FlowHandle::class.java)
|
||||||
register(KryoException::class.java)
|
register(KryoException::class.java)
|
||||||
register(StringBuffer::class.java)
|
register(StringBuffer::class.java)
|
||||||
@ -229,20 +238,45 @@ private class RPCKryo(observableSerializer: Serializer<Observable<Any>>? = null)
|
|||||||
pluginRegistries.forEach { it.registerRPCKryoTypes(this) }
|
pluginRegistries.forEach { it.registerRPCKryoTypes(this) }
|
||||||
}
|
}
|
||||||
|
|
||||||
// Helper method, attempt to reduce boiler plate code
|
// TODO: workaround to prevent Observable registration conflict when using plugin registered kyro classes
|
||||||
private fun <T> register(type: Class<T>, read: (Kryo, Input) -> T, write: (Kryo, Output, T) -> Unit) {
|
private val observableRegistration: Registration? = observableSerializer?.let { register(Observable::class.java, it, 10000) }
|
||||||
register(type, object : Serializer<T>() {
|
|
||||||
override fun read(kryo: Kryo, input: Input, type: Class<T>): T = read(kryo, input)
|
private val listenableFutureRegistration: Registration? = observableSerializer?.let {
|
||||||
override fun write(kryo: Kryo, output: Output, o: T) = write(kryo, output, o)
|
// Register ListenableFuture by making use of Observable serialisation.
|
||||||
})
|
// TODO Serialisation could be made more efficient as a future can only emit one value (or exception)
|
||||||
|
@Suppress("UNCHECKED_CAST")
|
||||||
|
register(ListenableFuture::class,
|
||||||
|
read = { kryo, input -> it.read(kryo, input, Observable::class.java as Class<Observable<Any>>).toFuture() },
|
||||||
|
write = { kryo, output, obj -> it.write(kryo, output, obj.toObservable()) }
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: workaround to prevent Observable registration conflict when using plugin registered kyro classes
|
// Avoid having to worry about the subtypes of FlowException by converting all of them to just FlowException.
|
||||||
val observableRegistration: Registration? = if (observableSerializer != null) register(Observable::class.java, observableSerializer, 10000) else null
|
// This is a temporary hack until a proper serialisation mechanism is in place.
|
||||||
|
private val flowExceptionRegistration: Registration = register(
|
||||||
|
FlowException::class,
|
||||||
|
read = { kryo, input ->
|
||||||
|
val message = input.readString()
|
||||||
|
val cause = kryo.readObjectOrNull(input, Throwable::class.java)
|
||||||
|
FlowException(message, cause)
|
||||||
|
},
|
||||||
|
write = { kryo, output, obj ->
|
||||||
|
// The subclass may have overridden toString so we use that
|
||||||
|
val message = if (obj.javaClass != FlowException::class.java) obj.toString() else obj.message
|
||||||
|
output.writeString(message)
|
||||||
|
kryo.writeObjectOrNull(output, obj.cause, Throwable::class.java)
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
override fun getRegistration(type: Class<*>): Registration {
|
override fun getRegistration(type: Class<*>): Registration {
|
||||||
if (Observable::class.java.isAssignableFrom(type))
|
if (Observable::class.java.isAssignableFrom(type))
|
||||||
return observableRegistration ?: throw IllegalStateException("This RPC was not annotated with @RPCReturnsObservables")
|
return observableRegistration ?:
|
||||||
|
throw IllegalStateException("This RPC was not annotated with @RPCReturnsObservables")
|
||||||
|
if (ListenableFuture::class.java.isAssignableFrom(type))
|
||||||
|
return listenableFutureRegistration ?:
|
||||||
|
throw IllegalStateException("This RPC was not annotated with @RPCReturnsObservables")
|
||||||
|
if (FlowException::class.java.isAssignableFrom(type))
|
||||||
|
return flowExceptionRegistration
|
||||||
return super.getRegistration(type)
|
return super.getRegistration(type)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -4,6 +4,7 @@ import com.google.common.annotations.VisibleForTesting
|
|||||||
import com.google.common.util.concurrent.ListenableFuture
|
import com.google.common.util.concurrent.ListenableFuture
|
||||||
import com.google.common.util.concurrent.SettableFuture
|
import com.google.common.util.concurrent.SettableFuture
|
||||||
import net.corda.core.bufferUntilSubscribed
|
import net.corda.core.bufferUntilSubscribed
|
||||||
|
import net.corda.core.crypto.CompositeKey
|
||||||
import net.corda.core.crypto.Party
|
import net.corda.core.crypto.Party
|
||||||
import net.corda.core.map
|
import net.corda.core.map
|
||||||
import net.corda.core.messaging.MessagingService
|
import net.corda.core.messaging.MessagingService
|
||||||
@ -70,6 +71,8 @@ open class InMemoryNetworkMapCache : SingletonSerializeAsToken(), NetworkMapCach
|
|||||||
return null
|
return null
|
||||||
}
|
}
|
||||||
|
|
||||||
|
override fun getNodeByLegalIdentityKey(compositeKey: CompositeKey): NodeInfo? = registeredNodes[Party("", compositeKey)]
|
||||||
|
|
||||||
override fun track(): Pair<List<NodeInfo>, Observable<MapChange>> {
|
override fun track(): Pair<List<NodeInfo>, Observable<MapChange>> {
|
||||||
synchronized(_changed) {
|
synchronized(_changed) {
|
||||||
return Pair(partyNodes, _changed.bufferUntilSubscribed().wrapWithDatabaseTransaction())
|
return Pair(partyNodes, _changed.bufferUntilSubscribed().wrapWithDatabaseTransaction())
|
||||||
|
@ -368,9 +368,6 @@ class StateMachineManager(val serviceHub: ServiceHubInternal,
|
|||||||
}
|
}
|
||||||
|
|
||||||
private fun endAllFiberSessions(fiber: FlowStateMachineImpl<*>, errorResponse: Pair<FlowException, Boolean>?) {
|
private fun endAllFiberSessions(fiber: FlowStateMachineImpl<*>, errorResponse: Pair<FlowException, Boolean>?) {
|
||||||
// TODO Blanking the stack trace prevents the receiving flow from filling in its own stack trace
|
|
||||||
// @Suppress("PLATFORM_CLASS_MAPPED_TO_KOTLIN")
|
|
||||||
// (errorResponse?.first as java.lang.Throwable?)?.stackTrace = emptyArray()
|
|
||||||
openSessions.values.removeIf { session ->
|
openSessions.values.removeIf { session ->
|
||||||
if (session.fiber == fiber) {
|
if (session.fiber == fiber) {
|
||||||
session.endSession(errorResponse)
|
session.endSession(errorResponse)
|
||||||
|
@ -1,8 +1,13 @@
|
|||||||
package net.corda.node.messaging
|
package net.corda.node.messaging
|
||||||
|
|
||||||
|
import com.google.common.util.concurrent.Futures
|
||||||
|
import com.google.common.util.concurrent.ListenableFuture
|
||||||
|
import com.google.common.util.concurrent.SettableFuture
|
||||||
|
import net.corda.core.getOrThrow
|
||||||
import net.corda.core.messaging.RPCOps
|
import net.corda.core.messaging.RPCOps
|
||||||
import net.corda.core.messaging.RPCReturnsObservables
|
import net.corda.core.messaging.RPCReturnsObservables
|
||||||
import net.corda.core.serialization.SerializedBytes
|
import net.corda.core.serialization.SerializedBytes
|
||||||
|
import net.corda.core.success
|
||||||
import net.corda.core.utilities.LogHelper
|
import net.corda.core.utilities.LogHelper
|
||||||
import net.corda.node.services.RPCUserService
|
import net.corda.node.services.RPCUserService
|
||||||
import net.corda.node.services.User
|
import net.corda.node.services.User
|
||||||
@ -120,30 +125,31 @@ class ClientRPCInfrastructureTests {
|
|||||||
@RPCReturnsObservables
|
@RPCReturnsObservables
|
||||||
fun makeComplicatedObservable(): Observable<Pair<String, Observable<String>>>
|
fun makeComplicatedObservable(): Observable<Pair<String, Observable<String>>>
|
||||||
|
|
||||||
|
@RPCReturnsObservables
|
||||||
|
fun makeListenableFuture(): ListenableFuture<Int>
|
||||||
|
|
||||||
|
@RPCReturnsObservables
|
||||||
|
fun makeComplicatedListenableFuture(): ListenableFuture<Pair<String, ListenableFuture<String>>>
|
||||||
|
|
||||||
@RPCSinceVersion(2)
|
@RPCSinceVersion(2)
|
||||||
fun addedLater()
|
fun addedLater()
|
||||||
|
|
||||||
fun captureUser(): String
|
fun captureUser(): String
|
||||||
}
|
}
|
||||||
|
|
||||||
lateinit var complicatedObservable: Observable<Pair<String, Observable<String>>>
|
private lateinit var complicatedObservable: Observable<Pair<String, Observable<String>>>
|
||||||
|
private lateinit var complicatedListenableFuturee: ListenableFuture<Pair<String, ListenableFuture<String>>>
|
||||||
|
|
||||||
inner class TestOpsImpl : TestOps {
|
inner class TestOpsImpl : TestOps {
|
||||||
override val protocolVersion = 1
|
override val protocolVersion = 1
|
||||||
|
|
||||||
override fun barf(): Unit = throw IllegalArgumentException("Barf!")
|
override fun barf(): Unit = throw IllegalArgumentException("Barf!")
|
||||||
|
override fun void() {}
|
||||||
override fun void() {
|
|
||||||
}
|
|
||||||
|
|
||||||
override fun someCalculation(str: String, num: Int) = "$str $num"
|
override fun someCalculation(str: String, num: Int) = "$str $num"
|
||||||
|
|
||||||
override fun makeObservable(): Observable<Int> = Observable.just(1, 2, 3, 4)
|
override fun makeObservable(): Observable<Int> = Observable.just(1, 2, 3, 4)
|
||||||
|
override fun makeListenableFuture(): ListenableFuture<Int> = Futures.immediateFuture(1)
|
||||||
override fun makeComplicatedObservable() = complicatedObservable
|
override fun makeComplicatedObservable() = complicatedObservable
|
||||||
|
override fun makeComplicatedListenableFuture(): ListenableFuture<Pair<String, ListenableFuture<String>>> = complicatedListenableFuturee
|
||||||
override fun addedLater(): Unit = throw UnsupportedOperationException("not implemented")
|
override fun addedLater(): Unit = throw UnsupportedOperationException("not implemented")
|
||||||
|
|
||||||
override fun captureUser(): String = CURRENT_RPC_USER.get().username
|
override fun captureUser(): String = CURRENT_RPC_USER.get().username
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -212,6 +218,41 @@ class ClientRPCInfrastructureTests {
|
|||||||
assertEquals(1, clientSession.addressQuery(rpcQueuesQuery).queueNames.size)
|
assertEquals(1, clientSession.addressQuery(rpcQueuesQuery).queueNames.size)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun `simple ListenableFuture`() {
|
||||||
|
val value = proxy.makeListenableFuture().getOrThrow()
|
||||||
|
assertThat(value).isEqualTo(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun `complex ListenableFuture`() {
|
||||||
|
val serverQuote = SettableFuture.create<Pair<String, ListenableFuture<String>>>()
|
||||||
|
complicatedListenableFuturee = serverQuote
|
||||||
|
|
||||||
|
val twainQuote = "Mark Twain" to Futures.immediateFuture("I have never let my schooling interfere with my education.")
|
||||||
|
|
||||||
|
val clientQuotes = LinkedBlockingQueue<String>()
|
||||||
|
val clientFuture = proxy.makeComplicatedListenableFuture()
|
||||||
|
|
||||||
|
clientFuture.success {
|
||||||
|
val name = it.first
|
||||||
|
it.second.success {
|
||||||
|
clientQuotes += "Quote by $name: $it"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
val rpcQueuesQuery = SimpleString("clients.${authenticatedUser.username}.rpc.*")
|
||||||
|
assertEquals(2, clientSession.addressQuery(rpcQueuesQuery).queueNames.size)
|
||||||
|
|
||||||
|
assertThat(clientQuotes).isEmpty()
|
||||||
|
|
||||||
|
serverQuote.set(twainQuote)
|
||||||
|
assertThat(clientQuotes.take()).isEqualTo("Quote by Mark Twain: I have never let my schooling interfere with my education.")
|
||||||
|
|
||||||
|
// TODO This final assert sometimes fails because the relevant queue hasn't been removed yet
|
||||||
|
// assertEquals(1, clientSession.addressQuery(rpcQueuesQuery).queueNames.size)
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
fun versioning() {
|
fun versioning() {
|
||||||
assertFailsWith<UnsupportedOperationException> { proxy.addedLater() }
|
assertFailsWith<UnsupportedOperationException> { proxy.addedLater() }
|
||||||
@ -221,5 +262,4 @@ class ClientRPCInfrastructureTests {
|
|||||||
fun `authenticated user is available to RPC`() {
|
fun `authenticated user is available to RPC`() {
|
||||||
assertThat(proxy.captureUser()).isEqualTo(authenticatedUser.username)
|
assertThat(proxy.captureUser()).isEqualTo(authenticatedUser.username)
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -4,7 +4,6 @@ import net.corda.core.getOrThrow
|
|||||||
import net.corda.core.node.services.ServiceInfo
|
import net.corda.core.node.services.ServiceInfo
|
||||||
import net.corda.node.services.network.NetworkMapService
|
import net.corda.node.services.network.NetworkMapService
|
||||||
import net.corda.node.utilities.databaseTransaction
|
import net.corda.node.utilities.databaseTransaction
|
||||||
import net.corda.testing.expect
|
|
||||||
import net.corda.testing.node.MockNetwork
|
import net.corda.testing.node.MockNetwork
|
||||||
import org.junit.Test
|
import org.junit.Test
|
||||||
import java.math.BigInteger
|
import java.math.BigInteger
|
||||||
@ -34,12 +33,7 @@ class InMemoryNetworkMapCacheTest {
|
|||||||
databaseTransaction(nodeA.database) {
|
databaseTransaction(nodeA.database) {
|
||||||
nodeA.netMapCache.addNode(nodeB.info)
|
nodeA.netMapCache.addNode(nodeB.info)
|
||||||
}
|
}
|
||||||
// Now both nodes match, so it throws an error
|
// The details of node B write over those for node A
|
||||||
expect<IllegalStateException> {
|
assertEquals(nodeA.netMapCache.getNodeByLegalIdentityKey(nodeA.info.legalIdentity.owningKey), nodeB.info)
|
||||||
nodeA.netMapCache.getNodeByLegalIdentityKey(nodeA.info.legalIdentity.owningKey)
|
|
||||||
}
|
|
||||||
expect<IllegalStateException> {
|
|
||||||
nodeA.netMapCache.getNodeByLegalIdentityKey(nodeB.info.legalIdentity.owningKey)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -390,9 +390,8 @@ class StateMachineManagerTests {
|
|||||||
node2 sent sessionConfirm to node1,
|
node2 sent sessionConfirm to node1,
|
||||||
node2 sent sessionEnd(errorFlow.exceptionThrown) to node1
|
node2 sent sessionEnd(errorFlow.exceptionThrown) to node1
|
||||||
)
|
)
|
||||||
// TODO see StateMachineManager.endAllFiberSessions
|
// Make sure the original stack trace isn't sent down the wire
|
||||||
// // Make sure the original stack trace isn't sent down the wire
|
assertThat((sessionTransfers.last().message as SessionEnd).errorResponse!!.stackTrace).isEmpty()
|
||||||
// assertThat((sessionTransfers.last().message as SessionEnd).errorResponse!!.stackTrace).isEmpty()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
|
@ -6,6 +6,7 @@ import net.corda.core.contracts.TransactionType
|
|||||||
import net.corda.core.crypto.Party
|
import net.corda.core.crypto.Party
|
||||||
import net.corda.core.crypto.SecureHash
|
import net.corda.core.crypto.SecureHash
|
||||||
import net.corda.core.div
|
import net.corda.core.div
|
||||||
|
import net.corda.core.getOrThrow
|
||||||
import net.corda.core.messaging.CordaRPCOps
|
import net.corda.core.messaging.CordaRPCOps
|
||||||
import net.corda.core.messaging.startFlow
|
import net.corda.core.messaging.startFlow
|
||||||
import net.corda.core.utilities.Emoji
|
import net.corda.core.utilities.Emoji
|
||||||
@ -83,9 +84,9 @@ fun sender(rpc: CordaRPCOps) {
|
|||||||
// Send the transaction to the other recipient
|
// Send the transaction to the other recipient
|
||||||
val stx = ptx.toSignedTransaction()
|
val stx = ptx.toSignedTransaction()
|
||||||
println("Sending ${stx.id}")
|
println("Sending ${stx.id}")
|
||||||
val protocolHandle = rpc.startFlow(::FinalityFlow, stx, setOf(otherSide))
|
val flowHandle = rpc.startFlow(::FinalityFlow, stx, setOf(otherSide))
|
||||||
protocolHandle.progress.subscribe(::println)
|
flowHandle.progress.subscribe(::println)
|
||||||
protocolHandle.returnValue.toBlocking().first()
|
flowHandle.returnValue.getOrThrow()
|
||||||
}
|
}
|
||||||
|
|
||||||
fun recipient(rpc: CordaRPCOps) {
|
fun recipient(rpc: CordaRPCOps) {
|
||||||
|
@ -5,7 +5,6 @@ import net.corda.core.contracts.DOLLARS
|
|||||||
import net.corda.core.getOrThrow
|
import net.corda.core.getOrThrow
|
||||||
import net.corda.core.messaging.startFlow
|
import net.corda.core.messaging.startFlow
|
||||||
import net.corda.core.node.services.ServiceInfo
|
import net.corda.core.node.services.ServiceInfo
|
||||||
import net.corda.core.transactions.SignedTransaction
|
|
||||||
import net.corda.flows.IssuerFlow.IssuanceRequester
|
import net.corda.flows.IssuerFlow.IssuanceRequester
|
||||||
import net.corda.node.driver.driver
|
import net.corda.node.driver.driver
|
||||||
import net.corda.node.services.User
|
import net.corda.node.services.User
|
||||||
@ -16,7 +15,6 @@ import net.corda.testing.expect
|
|||||||
import net.corda.testing.expectEvents
|
import net.corda.testing.expectEvents
|
||||||
import net.corda.testing.sequence
|
import net.corda.testing.sequence
|
||||||
import org.junit.Test
|
import org.junit.Test
|
||||||
import kotlin.test.assertTrue
|
|
||||||
|
|
||||||
class BankOfCordaRPCClientTest {
|
class BankOfCordaRPCClientTest {
|
||||||
@Test
|
@Test
|
||||||
@ -45,13 +43,12 @@ class BankOfCordaRPCClientTest {
|
|||||||
val vaultUpdatesBigCorp = bigCorpProxy.vaultAndUpdates().second
|
val vaultUpdatesBigCorp = bigCorpProxy.vaultAndUpdates().second
|
||||||
|
|
||||||
// Kick-off actual Issuer Flow
|
// Kick-off actual Issuer Flow
|
||||||
val result = bocProxy.startFlow(
|
bocProxy.startFlow(
|
||||||
::IssuanceRequester,
|
::IssuanceRequester,
|
||||||
1000.DOLLARS,
|
1000.DOLLARS,
|
||||||
nodeBigCorporation.nodeInfo.legalIdentity,
|
nodeBigCorporation.nodeInfo.legalIdentity,
|
||||||
BOC_PARTY_REF,
|
BOC_PARTY_REF,
|
||||||
nodeBankOfCorda.nodeInfo.legalIdentity).returnValue.toBlocking().first()
|
nodeBankOfCorda.nodeInfo.legalIdentity).returnValue.getOrThrow()
|
||||||
assertTrue { result is SignedTransaction }
|
|
||||||
|
|
||||||
// Check Bank of Corda Vault Updates
|
// Check Bank of Corda Vault Updates
|
||||||
vaultUpdatesBoc.expectEvents {
|
vaultUpdatesBoc.expectEvents {
|
||||||
|
@ -2,14 +2,15 @@ package net.corda.bank.api
|
|||||||
|
|
||||||
import com.google.common.net.HostAndPort
|
import com.google.common.net.HostAndPort
|
||||||
import net.corda.bank.api.BankOfCordaWebApi.IssueRequestParams
|
import net.corda.bank.api.BankOfCordaWebApi.IssueRequestParams
|
||||||
import net.corda.flows.IssuerFlow.IssuanceRequester
|
|
||||||
import net.corda.node.services.messaging.CordaRPCClient
|
|
||||||
import net.corda.core.contracts.Amount
|
import net.corda.core.contracts.Amount
|
||||||
import net.corda.core.contracts.currency
|
import net.corda.core.contracts.currency
|
||||||
|
import net.corda.core.getOrThrow
|
||||||
import net.corda.core.messaging.startFlow
|
import net.corda.core.messaging.startFlow
|
||||||
import net.corda.core.serialization.OpaqueBytes
|
import net.corda.core.serialization.OpaqueBytes
|
||||||
import net.corda.core.transactions.SignedTransaction
|
import net.corda.core.transactions.SignedTransaction
|
||||||
|
import net.corda.flows.IssuerFlow.IssuanceRequester
|
||||||
import net.corda.node.services.config.configureTestSSL
|
import net.corda.node.services.config.configureTestSSL
|
||||||
|
import net.corda.node.services.messaging.CordaRPCClient
|
||||||
import net.corda.testing.http.HttpApi
|
import net.corda.testing.http.HttpApi
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -43,6 +44,6 @@ class BankOfCordaClientApi(val hostAndPort: HostAndPort) {
|
|||||||
val amount = Amount(params.amount, currency(params.currency))
|
val amount = Amount(params.amount, currency(params.currency))
|
||||||
val issuerToPartyRef = OpaqueBytes.of(params.issueToPartyRefAsString.toByte())
|
val issuerToPartyRef = OpaqueBytes.of(params.issueToPartyRefAsString.toByte())
|
||||||
|
|
||||||
return proxy.startFlow(::IssuanceRequester, amount, issueToParty, issuerToPartyRef, issuerBankParty).returnValue.toBlocking().first()
|
return proxy.startFlow(::IssuanceRequester, amount, issueToParty, issuerToPartyRef, issuerBankParty).returnValue.getOrThrow()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,13 +1,14 @@
|
|||||||
package net.corda.bank.api
|
package net.corda.bank.api
|
||||||
|
|
||||||
import net.corda.flows.IssuerFlow.IssuanceRequester
|
|
||||||
import net.corda.core.contracts.Amount
|
import net.corda.core.contracts.Amount
|
||||||
import net.corda.core.contracts.currency
|
import net.corda.core.contracts.currency
|
||||||
|
import net.corda.core.flows.FlowException
|
||||||
|
import net.corda.core.getOrThrow
|
||||||
import net.corda.core.messaging.CordaRPCOps
|
import net.corda.core.messaging.CordaRPCOps
|
||||||
import net.corda.core.messaging.startFlow
|
import net.corda.core.messaging.startFlow
|
||||||
import net.corda.core.serialization.OpaqueBytes
|
import net.corda.core.serialization.OpaqueBytes
|
||||||
import net.corda.core.transactions.SignedTransaction
|
|
||||||
import net.corda.core.utilities.loggerFor
|
import net.corda.core.utilities.loggerFor
|
||||||
|
import net.corda.flows.IssuerFlow.IssuanceRequester
|
||||||
import java.time.LocalDateTime
|
import java.time.LocalDateTime
|
||||||
import javax.ws.rs.*
|
import javax.ws.rs.*
|
||||||
import javax.ws.rs.core.MediaType
|
import javax.ws.rs.core.MediaType
|
||||||
@ -46,12 +47,13 @@ class BankOfCordaWebApi(val rpc: CordaRPCOps) {
|
|||||||
|
|
||||||
// invoke client side of Issuer Flow: IssuanceRequester
|
// invoke client side of Issuer Flow: IssuanceRequester
|
||||||
// The line below blocks and waits for the future to resolve.
|
// The line below blocks and waits for the future to resolve.
|
||||||
val result = rpc.startFlow(::IssuanceRequester, amount, issueToParty, issuerToPartyRef, issuerBankParty).returnValue.toBlocking().first()
|
val status = try {
|
||||||
if (result is SignedTransaction) {
|
rpc.startFlow(::IssuanceRequester, amount, issueToParty, issuerToPartyRef, issuerBankParty).returnValue.getOrThrow()
|
||||||
logger.info("Issue request completed successfully: ${params}")
|
logger.info("Issue request completed successfully: $params")
|
||||||
return Response.status(Response.Status.CREATED).build()
|
Response.Status.CREATED
|
||||||
} else {
|
} catch (e: FlowException) {
|
||||||
return Response.status(Response.Status.BAD_REQUEST).build()
|
Response.Status.BAD_REQUEST
|
||||||
}
|
}
|
||||||
|
return Response.status(status).build()
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -1,6 +1,7 @@
|
|||||||
package net.corda.irs.api
|
package net.corda.irs.api
|
||||||
|
|
||||||
import net.corda.core.contracts.filterStatesOfType
|
import net.corda.core.contracts.filterStatesOfType
|
||||||
|
import net.corda.core.getOrThrow
|
||||||
import net.corda.core.messaging.CordaRPCOps
|
import net.corda.core.messaging.CordaRPCOps
|
||||||
import net.corda.core.messaging.startFlow
|
import net.corda.core.messaging.startFlow
|
||||||
import net.corda.core.utilities.loggerFor
|
import net.corda.core.utilities.loggerFor
|
||||||
@ -66,12 +67,12 @@ class InterestRateSwapAPI(val rpc: CordaRPCOps) {
|
|||||||
@Path("deals")
|
@Path("deals")
|
||||||
@Consumes(MediaType.APPLICATION_JSON)
|
@Consumes(MediaType.APPLICATION_JSON)
|
||||||
fun storeDeal(newDeal: InterestRateSwap.State): Response {
|
fun storeDeal(newDeal: InterestRateSwap.State): Response {
|
||||||
try {
|
return try {
|
||||||
rpc.startFlow(AutoOfferFlow::Requester, newDeal).returnValue.toBlocking().first()
|
rpc.startFlow(AutoOfferFlow::Requester, newDeal).returnValue.getOrThrow()
|
||||||
return Response.created(URI.create(generateDealLink(newDeal))).build()
|
Response.created(URI.create(generateDealLink(newDeal))).build()
|
||||||
} catch (ex: Throwable) {
|
} catch (ex: Throwable) {
|
||||||
logger.info("Exception when creating deal: $ex")
|
logger.info("Exception when creating deal: $ex")
|
||||||
return Response.status(Response.Status.INTERNAL_SERVER_ERROR).entity(ex.toString()).build()
|
Response.status(Response.Status.INTERNAL_SERVER_ERROR).entity(ex.toString()).build()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -94,7 +95,7 @@ class InterestRateSwapAPI(val rpc: CordaRPCOps) {
|
|||||||
val priorDemoDate = fetchDemoDate()
|
val priorDemoDate = fetchDemoDate()
|
||||||
// Can only move date forwards
|
// Can only move date forwards
|
||||||
if (newDemoDate.isAfter(priorDemoDate)) {
|
if (newDemoDate.isAfter(priorDemoDate)) {
|
||||||
rpc.startFlow(UpdateBusinessDayFlow::Broadcast, newDemoDate).returnValue.toBlocking().first()
|
rpc.startFlow(UpdateBusinessDayFlow::Broadcast, newDemoDate).returnValue.getOrThrow()
|
||||||
return Response.ok().build()
|
return Response.ok().build()
|
||||||
}
|
}
|
||||||
val msg = "demodate is already $priorDemoDate and can only be updated with a later date"
|
val msg = "demodate is already $priorDemoDate and can only be updated with a later date"
|
||||||
@ -113,7 +114,7 @@ class InterestRateSwapAPI(val rpc: CordaRPCOps) {
|
|||||||
@Path("restart")
|
@Path("restart")
|
||||||
@Consumes(MediaType.APPLICATION_JSON)
|
@Consumes(MediaType.APPLICATION_JSON)
|
||||||
fun exitServer(): Response {
|
fun exitServer(): Response {
|
||||||
rpc.startFlow(ExitServerFlow::Broadcast, 83).returnValue.toBlocking().first()
|
rpc.startFlow(ExitServerFlow::Broadcast, 83).returnValue.getOrThrow()
|
||||||
return Response.ok().build()
|
return Response.ok().build()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,8 +1,10 @@
|
|||||||
package net.corda.notarydemo
|
package net.corda.notarydemo
|
||||||
|
|
||||||
import com.google.common.net.HostAndPort
|
import com.google.common.net.HostAndPort
|
||||||
|
import com.google.common.util.concurrent.Futures
|
||||||
import net.corda.core.crypto.toStringShort
|
import net.corda.core.crypto.toStringShort
|
||||||
import net.corda.core.div
|
import net.corda.core.div
|
||||||
|
import net.corda.core.getOrThrow
|
||||||
import net.corda.core.messaging.CordaRPCOps
|
import net.corda.core.messaging.CordaRPCOps
|
||||||
import net.corda.core.messaging.startFlow
|
import net.corda.core.messaging.startFlow
|
||||||
import net.corda.core.transactions.SignedTransaction
|
import net.corda.core.transactions.SignedTransaction
|
||||||
@ -60,9 +62,9 @@ private class NotaryDemoClientApi(val rpc: CordaRPCOps) {
|
|||||||
*/
|
*/
|
||||||
private fun buildTransactions(count: Int): List<SignedTransaction> {
|
private fun buildTransactions(count: Int): List<SignedTransaction> {
|
||||||
val moveTransactions = (1..count).map {
|
val moveTransactions = (1..count).map {
|
||||||
rpc.startFlow(::DummyIssueAndMove, notary, counterpartyNode.legalIdentity).returnValue.toBlocking().toFuture()
|
rpc.startFlow(::DummyIssueAndMove, notary, counterpartyNode.legalIdentity).returnValue
|
||||||
}
|
}
|
||||||
return moveTransactions.map { it.get() }
|
return Futures.allAsList(moveTransactions).getOrThrow()
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -72,10 +74,8 @@ private class NotaryDemoClientApi(val rpc: CordaRPCOps) {
|
|||||||
* @return a list of encoded signer public keys - one for every transaction
|
* @return a list of encoded signer public keys - one for every transaction
|
||||||
*/
|
*/
|
||||||
private fun notariseTransactions(transactions: List<SignedTransaction>): List<String> {
|
private fun notariseTransactions(transactions: List<SignedTransaction>): List<String> {
|
||||||
val signatureFutures = transactions.map {
|
val signatureFutures = transactions.map { rpc.startFlow(NotaryFlow::Client, it).returnValue }
|
||||||
rpc.startFlow(NotaryFlow::Client, it).returnValue.toBlocking().toFuture()
|
return Futures.allAsList(signatureFutures).getOrThrow().map { it.by.toStringShort() }
|
||||||
}
|
|
||||||
return signatureFutures.map { it.get().by.toStringShort() }
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -42,8 +42,8 @@ class SimmValuationTest : IntegrationTestCategory {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun getPartyWithName(partyApi: HttpApi, countryparty: String): PortfolioApi.ApiParty =
|
private fun getPartyWithName(partyApi: HttpApi, counterparty: String): PortfolioApi.ApiParty =
|
||||||
getAvailablePartiesFor(partyApi).counterparties.single { it.text == countryparty }
|
getAvailablePartiesFor(partyApi).counterparties.single { it.text == counterparty }
|
||||||
|
|
||||||
private fun getAvailablePartiesFor(partyApi: HttpApi): PortfolioApi.AvailableParties {
|
private fun getAvailablePartiesFor(partyApi: HttpApi): PortfolioApi.AvailableParties {
|
||||||
return partyApi.getJson<PortfolioApi.AvailableParties>("whoami")
|
return partyApi.getJson<PortfolioApi.AvailableParties>("whoami")
|
||||||
@ -68,4 +68,4 @@ class SimmValuationTest : IntegrationTestCategory {
|
|||||||
val valuations = partyApi.getJson<PortfolioApiUtils.ValuationsView>("${counterparty.id}/portfolio/valuations")
|
val valuations = partyApi.getJson<PortfolioApiUtils.ValuationsView>("${counterparty.id}/portfolio/valuations")
|
||||||
return (valuations.initialMargin.call["total"] != 0.0)
|
return (valuations.initialMargin.call["total"] != 0.0)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -6,6 +6,7 @@ import net.corda.core.contracts.StateAndRef
|
|||||||
import net.corda.core.contracts.filterStatesOfType
|
import net.corda.core.contracts.filterStatesOfType
|
||||||
import net.corda.core.crypto.CompositeKey
|
import net.corda.core.crypto.CompositeKey
|
||||||
import net.corda.core.crypto.Party
|
import net.corda.core.crypto.Party
|
||||||
|
import net.corda.core.getOrThrow
|
||||||
import net.corda.core.messaging.CordaRPCOps
|
import net.corda.core.messaging.CordaRPCOps
|
||||||
import net.corda.core.messaging.startFlow
|
import net.corda.core.messaging.startFlow
|
||||||
import net.corda.vega.analytics.InitialMarginTriple
|
import net.corda.vega.analytics.InitialMarginTriple
|
||||||
@ -159,7 +160,7 @@ class PortfolioApi(val rpc: CordaRPCOps) {
|
|||||||
return withParty(partyName) {
|
return withParty(partyName) {
|
||||||
val buyer = if (swap.buySell.isBuy) ownParty else it
|
val buyer = if (swap.buySell.isBuy) ownParty else it
|
||||||
val seller = if (swap.buySell.isSell) ownParty else it
|
val seller = if (swap.buySell.isSell) ownParty else it
|
||||||
rpc.startFlow(IRSTradeFlow::Requester, swap.toData(buyer, seller), it).returnValue.toBlocking().first()
|
rpc.startFlow(IRSTradeFlow::Requester, swap.toData(buyer, seller), it).returnValue.getOrThrow()
|
||||||
Response.accepted().entity("{}").build()
|
Response.accepted().entity("{}").build()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -266,12 +267,12 @@ class PortfolioApi(val rpc: CordaRPCOps) {
|
|||||||
fun startPortfolioCalculations(params: ValuationCreationParams = ValuationCreationParams(LocalDate.of(2016, 6, 6)), @PathParam("party") partyName: String): Response {
|
fun startPortfolioCalculations(params: ValuationCreationParams = ValuationCreationParams(LocalDate.of(2016, 6, 6)), @PathParam("party") partyName: String): Response {
|
||||||
return withParty(partyName) { otherParty ->
|
return withParty(partyName) { otherParty ->
|
||||||
val existingSwap = getPortfolioWith(otherParty)
|
val existingSwap = getPortfolioWith(otherParty)
|
||||||
if (existingSwap == null) {
|
val flowHandle = if (existingSwap == null) {
|
||||||
rpc.startFlow(SimmFlow::Requester, otherParty, params.valuationDate).returnValue.toBlocking().first()
|
rpc.startFlow(SimmFlow::Requester, otherParty, params.valuationDate)
|
||||||
} else {
|
} else {
|
||||||
val handle = rpc.startFlow(SimmRevaluation::Initiator, getPortfolioStateAndRefWith(otherParty).ref, params.valuationDate)
|
rpc.startFlow(SimmRevaluation::Initiator, getPortfolioStateAndRefWith(otherParty).ref, params.valuationDate)
|
||||||
handle.returnValue.toBlocking().first()
|
|
||||||
}
|
}
|
||||||
|
flowHandle.returnValue.getOrThrow()
|
||||||
|
|
||||||
withPortfolio(otherParty) { portfolioState ->
|
withPortfolio(otherParty) { portfolioState ->
|
||||||
val portfolio = portfolioState.portfolio.toStateAndRef<IRSState>(rpc).toPortfolio()
|
val portfolio = portfolioState.portfolio.toStateAndRef<IRSState>(rpc).toPortfolio()
|
||||||
|
@ -1,19 +1,17 @@
|
|||||||
package net.corda.traderdemo
|
package net.corda.traderdemo
|
||||||
|
|
||||||
import com.google.common.net.HostAndPort
|
import com.google.common.util.concurrent.Futures
|
||||||
import net.corda.contracts.testing.calculateRandomlySizedAmounts
|
import net.corda.contracts.testing.calculateRandomlySizedAmounts
|
||||||
import net.corda.core.contracts.Amount
|
import net.corda.core.contracts.Amount
|
||||||
import net.corda.core.contracts.DOLLARS
|
import net.corda.core.contracts.DOLLARS
|
||||||
|
import net.corda.core.getOrThrow
|
||||||
import net.corda.core.messaging.CordaRPCOps
|
import net.corda.core.messaging.CordaRPCOps
|
||||||
import net.corda.core.messaging.startFlow
|
import net.corda.core.messaging.startFlow
|
||||||
import net.corda.core.serialization.OpaqueBytes
|
import net.corda.core.serialization.OpaqueBytes
|
||||||
import net.corda.core.transactions.SignedTransaction
|
|
||||||
import net.corda.core.utilities.Emoji
|
import net.corda.core.utilities.Emoji
|
||||||
import net.corda.core.utilities.loggerFor
|
import net.corda.core.utilities.loggerFor
|
||||||
import net.corda.flows.IssuerFlow.IssuanceRequester
|
import net.corda.flows.IssuerFlow.IssuanceRequester
|
||||||
import net.corda.node.services.messaging.CordaRPCClient
|
|
||||||
import net.corda.testing.BOC
|
import net.corda.testing.BOC
|
||||||
import net.corda.testing.http.HttpApi
|
|
||||||
import net.corda.traderdemo.flow.SellerFlow
|
import net.corda.traderdemo.flow.SellerFlow
|
||||||
import java.util.*
|
import java.util.*
|
||||||
import kotlin.test.assertEquals
|
import kotlin.test.assertEquals
|
||||||
@ -26,20 +24,17 @@ class TraderDemoClientApi(val rpc: CordaRPCOps) {
|
|||||||
val logger = loggerFor<TraderDemoClientApi>()
|
val logger = loggerFor<TraderDemoClientApi>()
|
||||||
}
|
}
|
||||||
|
|
||||||
fun runBuyer(amount: Amount<Currency> = 30000.0.DOLLARS, notary: String = "Notary"): Boolean {
|
fun runBuyer(amount: Amount<Currency> = 30000.0.DOLLARS): Boolean {
|
||||||
val bankOfCordaParty = rpc.partyFromName(BOC.name)
|
val bankOfCordaParty = rpc.partyFromName(BOC.name)
|
||||||
?: throw Exception("Unable to locate ${BOC.name} in Network Map Service")
|
?: throw Exception("Unable to locate ${BOC.name} in Network Map Service")
|
||||||
val me = rpc.nodeIdentity()
|
val me = rpc.nodeIdentity()
|
||||||
// TODO: revert back to multiple issue request amounts (3,10) when soft locking implemented
|
// TODO: revert back to multiple issue request amounts (3,10) when soft locking implemented
|
||||||
val amounts = calculateRandomlySizedAmounts(amount, 1, 1, Random())
|
val amounts = calculateRandomlySizedAmounts(amount, 1, 1, Random())
|
||||||
val handles = amounts.map {
|
val resultFutures = amounts.map {
|
||||||
rpc.startFlow(::IssuanceRequester, amount, me.legalIdentity, OpaqueBytes.of(1), bankOfCordaParty)
|
rpc.startFlow(::IssuanceRequester, amount, me.legalIdentity, OpaqueBytes.of(1), bankOfCordaParty).returnValue
|
||||||
}
|
|
||||||
|
|
||||||
handles.forEach {
|
|
||||||
require(it.returnValue.toBlocking().first() is SignedTransaction)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Futures.allAsList(resultFutures).getOrThrow()
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -60,7 +55,7 @@ class TraderDemoClientApi(val rpc: CordaRPCOps) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// The line below blocks and waits for the future to resolve.
|
// The line below blocks and waits for the future to resolve.
|
||||||
val stx = rpc.startFlow(::SellerFlow, otherParty, amount).returnValue.toBlocking().first()
|
val stx = rpc.startFlow(::SellerFlow, otherParty, amount).returnValue.getOrThrow()
|
||||||
logger.info("Sale completed - we have a happy customer!\n\nFinal transaction is:\n\n${Emoji.renderIfSupported(stx.tx)}")
|
logger.info("Sale completed - we have a happy customer!\n\nFinal transaction is:\n\n${Emoji.renderIfSupported(stx.tx)}")
|
||||||
return true
|
return true
|
||||||
} else {
|
} else {
|
||||||
|
@ -21,7 +21,6 @@ import net.corda.core.getOrThrow
|
|||||||
import net.corda.core.messaging.startFlow
|
import net.corda.core.messaging.startFlow
|
||||||
import net.corda.core.node.NodeInfo
|
import net.corda.core.node.NodeInfo
|
||||||
import net.corda.core.serialization.OpaqueBytes
|
import net.corda.core.serialization.OpaqueBytes
|
||||||
import net.corda.core.toFuture
|
|
||||||
import net.corda.core.transactions.SignedTransaction
|
import net.corda.core.transactions.SignedTransaction
|
||||||
import net.corda.explorer.model.CashTransaction
|
import net.corda.explorer.model.CashTransaction
|
||||||
import net.corda.explorer.model.IssuerModel
|
import net.corda.explorer.model.IssuerModel
|
||||||
@ -101,7 +100,7 @@ class NewTransaction : Fragment() {
|
|||||||
rpcProxy.value!!.startFlow(::CashFlow, it)
|
rpcProxy.value!!.startFlow(::CashFlow, it)
|
||||||
}
|
}
|
||||||
val response = try {
|
val response = try {
|
||||||
handle?.returnValue?.toFuture()?.getOrThrow()
|
handle?.returnValue?.getOrThrow()
|
||||||
} catch (e: FlowException) {
|
} catch (e: FlowException) {
|
||||||
e
|
e
|
||||||
}
|
}
|
||||||
|
@ -2,15 +2,20 @@ package net.corda.explorer.views
|
|||||||
|
|
||||||
import org.junit.Assert.assertEquals
|
import org.junit.Assert.assertEquals
|
||||||
import org.junit.Test
|
import org.junit.Test
|
||||||
|
import java.text.DecimalFormatSymbols
|
||||||
|
import java.util.*
|
||||||
|
|
||||||
class GuiUtilitiesKtTest {
|
class GuiUtilitiesKtTest {
|
||||||
@Test
|
@Test
|
||||||
fun `test to string with suffix`() {
|
fun `test to string with suffix`() {
|
||||||
assertEquals("10.5k", 10500.toStringWithSuffix())
|
//Required for this test to be independent of the default Locale.
|
||||||
|
val ds = DecimalFormatSymbols(Locale.getDefault()).decimalSeparator
|
||||||
|
|
||||||
|
assertEquals("10${ds}5k", 10500.toStringWithSuffix())
|
||||||
assertEquals("100", 100.toStringWithSuffix())
|
assertEquals("100", 100.toStringWithSuffix())
|
||||||
assertEquals("5.0M", 5000000.toStringWithSuffix())
|
assertEquals("5${ds}0M", 5000000.toStringWithSuffix())
|
||||||
assertEquals("1.0B", 1000000000.toStringWithSuffix())
|
assertEquals("1${ds}0B", 1000000000.toStringWithSuffix())
|
||||||
assertEquals("1.5T", 1500000000000.toStringWithSuffix())
|
assertEquals("1${ds}5T", 1500000000000.toStringWithSuffix())
|
||||||
assertEquals("1000.0T", 1000000000000000.toStringWithSuffix())
|
assertEquals("1000${ds}0T", 1000000000000000.toStringWithSuffix())
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -7,12 +7,11 @@ import net.corda.core.contracts.Issued
|
|||||||
import net.corda.core.contracts.PartyAndReference
|
import net.corda.core.contracts.PartyAndReference
|
||||||
import net.corda.core.contracts.USD
|
import net.corda.core.contracts.USD
|
||||||
import net.corda.core.crypto.Party
|
import net.corda.core.crypto.Party
|
||||||
|
import net.corda.core.flows.FlowException
|
||||||
import net.corda.core.getOrThrow
|
import net.corda.core.getOrThrow
|
||||||
import net.corda.core.messaging.startFlow
|
import net.corda.core.messaging.startFlow
|
||||||
import net.corda.core.serialization.OpaqueBytes
|
import net.corda.core.serialization.OpaqueBytes
|
||||||
import net.corda.core.toFuture
|
|
||||||
import net.corda.flows.CashCommand
|
import net.corda.flows.CashCommand
|
||||||
import net.corda.flows.CashException
|
|
||||||
import net.corda.flows.CashFlow
|
import net.corda.flows.CashFlow
|
||||||
import net.corda.loadtest.LoadTest
|
import net.corda.loadtest.LoadTest
|
||||||
import net.corda.loadtest.NodeHandle
|
import net.corda.loadtest.NodeHandle
|
||||||
@ -208,9 +207,9 @@ val crossCashTest = LoadTest<CrossCashCommand, CrossCashState>(
|
|||||||
|
|
||||||
execute = { command ->
|
execute = { command ->
|
||||||
try {
|
try {
|
||||||
val result = command.node.connection.proxy.startFlow(::CashFlow, command.command).returnValue.toFuture().getOrThrow()
|
val result = command.node.connection.proxy.startFlow(::CashFlow, command.command).returnValue.getOrThrow()
|
||||||
log.info("Success: $result")
|
log.info("Success: $result")
|
||||||
} catch (e: CashException) {
|
} catch (e: FlowException) {
|
||||||
log.error("Failure", e)
|
log.error("Failure", e)
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
@ -7,11 +7,10 @@ import net.corda.client.mock.replicatePoisson
|
|||||||
import net.corda.contracts.asset.Cash
|
import net.corda.contracts.asset.Cash
|
||||||
import net.corda.core.contracts.USD
|
import net.corda.core.contracts.USD
|
||||||
import net.corda.core.crypto.Party
|
import net.corda.core.crypto.Party
|
||||||
|
import net.corda.core.flows.FlowException
|
||||||
import net.corda.core.getOrThrow
|
import net.corda.core.getOrThrow
|
||||||
import net.corda.core.messaging.startFlow
|
import net.corda.core.messaging.startFlow
|
||||||
import net.corda.core.toFuture
|
|
||||||
import net.corda.flows.CashCommand
|
import net.corda.flows.CashCommand
|
||||||
import net.corda.flows.CashException
|
|
||||||
import net.corda.flows.CashFlow
|
import net.corda.flows.CashFlow
|
||||||
import net.corda.loadtest.LoadTest
|
import net.corda.loadtest.LoadTest
|
||||||
import net.corda.loadtest.NodeHandle
|
import net.corda.loadtest.NodeHandle
|
||||||
@ -63,9 +62,9 @@ val selfIssueTest = LoadTest<SelfIssueCommand, SelfIssueState>(
|
|||||||
|
|
||||||
execute = { command ->
|
execute = { command ->
|
||||||
try {
|
try {
|
||||||
val result = command.node.connection.proxy.startFlow(::CashFlow, command.command).returnValue.toFuture().getOrThrow()
|
val result = command.node.connection.proxy.startFlow(::CashFlow, command.command).returnValue.getOrThrow()
|
||||||
log.info("Success: $result")
|
log.info("Success: $result")
|
||||||
} catch (e: CashException) {
|
} catch (e: FlowException) {
|
||||||
log.error("Failure", e)
|
log.error("Failure", e)
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
Loading…
x
Reference in New Issue
Block a user