diff --git a/.ci/api-current.txt b/.ci/api-current.txt index 8b80be5e32..167f149ecf 100644 --- a/.ci/api-current.txt +++ b/.ci/api-current.txt @@ -1599,9 +1599,9 @@ public final class net.corda.core.flows.TransactionParts extends java.lang.Objec public String toString() ## @net.corda.core.serialization.CordaSerializable public final class net.corda.core.flows.UnexpectedFlowEndException extends net.corda.core.CordaRuntimeException implements net.corda.core.flows.IdentifiableException - public (String, Throwable, long) - @org.jetbrains.annotations.NotNull public Long getErrorId() - public final long getOriginalErrorId() + public (String) + public (String, Throwable) + public (String, Throwable, Long) ## @net.corda.core.DoNotImplement @net.corda.core.serialization.CordaSerializable public abstract class net.corda.core.identity.AbstractParty extends java.lang.Object public (java.security.PublicKey) diff --git a/client/rpc/src/integration-test/kotlin/net/corda/client/rpc/RPCStabilityTests.kt b/client/rpc/src/integration-test/kotlin/net/corda/client/rpc/RPCStabilityTests.kt index f4fa6be9f4..2092df3926 100644 --- a/client/rpc/src/integration-test/kotlin/net/corda/client/rpc/RPCStabilityTests.kt +++ b/client/rpc/src/integration-test/kotlin/net/corda/client/rpc/RPCStabilityTests.kt @@ -398,7 +398,7 @@ class RPCStabilityTests { servers[response]!!.shutdown() servers.remove(response) - //failover will take some time + // Failover will take some time. while (true) { try { response = client.serverId() diff --git a/client/rpc/src/main/kotlin/net/corda/client/rpc/CordaRPCClient.kt b/client/rpc/src/main/kotlin/net/corda/client/rpc/CordaRPCClient.kt index ce5d0c617f..db45afe4fb 100644 --- a/client/rpc/src/main/kotlin/net/corda/client/rpc/CordaRPCClient.kt +++ b/client/rpc/src/main/kotlin/net/corda/client/rpc/CordaRPCClient.kt @@ -10,9 +10,9 @@ package net.corda.client.rpc +import net.corda.client.rpc.internal.CordaRPCClientConfigurationImpl import net.corda.client.rpc.internal.KryoClientSerializationScheme import net.corda.client.rpc.internal.RPCClient -import net.corda.client.rpc.internal.CordaRPCClientConfigurationImpl import net.corda.core.context.Actor import net.corda.core.context.Trace import net.corda.core.messaging.CordaRPCOps @@ -103,7 +103,8 @@ interface CordaRPCClientConfiguration { * [CordaRPCClientConfiguration]. While attempting failover, current and future RPC calls will throw * [RPCException] and previously returned observables will call onError(). * - * If the client was created using a list of hosts, automatic failover will occur(the servers have to be started in HA mode) + * If the client was created using a list of hosts, automatic failover will occur (the servers have to be started in + * HA mode). * * @param hostAndPort The network address to connect to. * @param configuration An optional configuration used to tweak client behaviour. diff --git a/client/rpc/src/main/kotlin/net/corda/client/rpc/internal/RPCClientProxyHandler.kt b/client/rpc/src/main/kotlin/net/corda/client/rpc/internal/RPCClientProxyHandler.kt index 6f174da6f5..1e03203031 100644 --- a/client/rpc/src/main/kotlin/net/corda/client/rpc/internal/RPCClientProxyHandler.kt +++ b/client/rpc/src/main/kotlin/net/corda/client/rpc/internal/RPCClientProxyHandler.kt @@ -80,6 +80,12 @@ import kotlin.reflect.jvm.javaMethod * unsubscribing from the [Observable], or if the [Observable] is garbage collected the client will eventually * automatically signal the server. This is done using a cache that holds weak references to the [UnicastSubject]s. * The cleanup happens in batches using a dedicated reaper, scheduled on [reaperExecutor]. + * + * The client will attempt to failover in case the server become unreachable. Depending on the [ServerLocataor] instance + * passed in the constructor, failover is either handle at Artemis level or client level. If only one transport + * was used to create the [ServerLocator], failover is handled by Artemis (retrying based on [CordaRPCClientConfiguration]. + * If a list of transport configurations was used, failover is handled locally. Artemis is able to do it, however the + * brokers on server side need to be configured in HA mode and the [ServerLocator] needs to be created with HA as well. */ class RPCClientProxyHandler( private val rpcConfiguration: CordaRPCClientConfiguration, @@ -185,7 +191,7 @@ class RPCClientProxyHandler( private val deduplicationSequenceNumber = AtomicLong(0) private val sendingEnabled = AtomicBoolean(true) - // used to interrupt failover thread (i.e. client is closed while failing over) + // Used to interrupt failover thread (i.e. client is closed while failing over). private var haFailoverThread: Thread? = null /** @@ -427,7 +433,7 @@ class RPCClientProxyHandler( } private fun attemptReconnect() { - var reconnectAttempts = rpcConfiguration.maxReconnectAttempts * serverLocator.staticTransportConfigurations.size + var reconnectAttempts = rpcConfiguration.maxReconnectAttempts.times(serverLocator.staticTransportConfigurations.size) var retryInterval = rpcConfiguration.connectionRetryInterval val maxRetryInterval = rpcConfiguration.connectionMaxRetryInterval @@ -456,7 +462,7 @@ class RPCClientProxyHandler( continue } - log.debug("Connected successfully using ${transport.params}") + log.debug("Connected successfully after $reconnectAttempts attempts using ${transport.params}.") log.info("RPC server available.") sessionFactory!!.addFailoverListener(this::haFailoverHandler) initSessions() @@ -495,7 +501,7 @@ class RPCClientProxyHandler( haFailoverThread = Thread.currentThread() attemptReconnect() } - /* Other events are not considered as reconnection is not done by Artemis */ + // Other events are not considered as reconnection is not done by Artemis. } private fun failoverHandler(event: FailoverEventType) { diff --git a/constants.properties b/constants.properties index 6f82ef914b..801502948e 100644 --- a/constants.properties +++ b/constants.properties @@ -19,4 +19,3 @@ artifactoryPluginVersion=4.4.18 snakeYamlVersion=1.19 caffeineVersion=2.6.2 metricsVersion=3.2.5 - diff --git a/core/src/main/kotlin/net/corda/core/flows/FlowException.kt b/core/src/main/kotlin/net/corda/core/flows/FlowException.kt index 270e26ce10..6b12775f20 100644 --- a/core/src/main/kotlin/net/corda/core/flows/FlowException.kt +++ b/core/src/main/kotlin/net/corda/core/flows/FlowException.kt @@ -46,7 +46,9 @@ open class FlowException(message: String?, cause: Throwable?) : * that we were not expecting), or the other side had an internal error, or the other side terminated when we * were waiting for a response. */ -class UnexpectedFlowEndException(message: String, cause: Throwable?, val originalErrorId: Long) : +class UnexpectedFlowEndException(message: String, cause: Throwable?, val originalErrorId: Long?) : CordaRuntimeException(message, cause), IdentifiableException { - override fun getErrorId(): Long = originalErrorId + constructor(message: String, cause: Throwable?) : this(message, cause, null) + constructor(message: String) : this(message, null) + override fun getErrorId(): Long? = originalErrorId } diff --git a/core/src/main/kotlin/net/corda/core/flows/FlowLogic.kt b/core/src/main/kotlin/net/corda/core/flows/FlowLogic.kt index 587e89dd08..42cb8d518d 100644 --- a/core/src/main/kotlin/net/corda/core/flows/FlowLogic.kt +++ b/core/src/main/kotlin/net/corda/core/flows/FlowLogic.kt @@ -16,7 +16,10 @@ import net.corda.core.CordaInternal import net.corda.core.crypto.SecureHash import net.corda.core.identity.Party import net.corda.core.identity.PartyAndCertificate -import net.corda.core.internal.* +import net.corda.core.internal.FlowIORequest +import net.corda.core.internal.FlowStateMachine +import net.corda.core.internal.abbreviate +import net.corda.core.internal.uncheckedCast import net.corda.core.messaging.DataFeed import net.corda.core.node.NodeInfo import net.corda.core.node.ServiceHub @@ -141,9 +144,98 @@ abstract class FlowLogic { * Note: The current implementation returns the single identity of the node. This will change once multiple identities * is implemented. */ - val ourIdentity: Party get() = stateMachine.ourIdentity + // Used to implement the deprecated send/receive functions using Party. When such a deprecated function is used we + // create a fresh session for the Party, put it here and use it in subsequent deprecated calls. + private val deprecatedPartySessionMap = HashMap() + private fun getDeprecatedSessionForParty(party: Party): FlowSession { + return deprecatedPartySessionMap.getOrPut(party) { initiateFlow(party) } + } + /** + * Returns a [FlowInfo] object describing the flow [otherParty] is using. With [FlowInfo.flowVersion] it + * provides the necessary information needed for the evolution of flows and enabling backwards compatibility. + * + * This method can be called before any send or receive has been done with [otherParty]. In such a case this will force + * them to start their flow. + */ + @Deprecated("Use FlowSession.getCounterpartyFlowInfo()", level = DeprecationLevel.WARNING) + @Suspendable + fun getFlowInfo(otherParty: Party): FlowInfo = getDeprecatedSessionForParty(otherParty).getCounterpartyFlowInfo() + + /** + * Serializes and queues the given [payload] object for sending to the [otherParty]. Suspends until a response + * is received, which must be of the given [R] type. + * + * Remember that when receiving data from other parties the data should not be trusted until it's been thoroughly + * verified for consistency and that all expectations are satisfied, as a malicious peer may send you subtly + * corrupted data in order to exploit your code. + * + * Note that this function is not just a simple send+receive pair: it is more efficient and more correct to + * use this when you expect to do a message swap than do use [send] and then [receive] in turn. + * + * @return an [UntrustworthyData] wrapper around the received object. + */ + @Deprecated("Use FlowSession.sendAndReceive()", level = DeprecationLevel.WARNING) + inline fun sendAndReceive(otherParty: Party, payload: Any): UntrustworthyData { + return sendAndReceive(R::class.java, otherParty, payload) + } + + /** + * Serializes and queues the given [payload] object for sending to the [otherParty]. Suspends until a response + * is received, which must be of the given [receiveType]. Remember that when receiving data from other parties the data + * should not be trusted until it's been thoroughly verified for consistency and that all expectations are + * satisfied, as a malicious peer may send you subtly corrupted data in order to exploit your code. + * + * Note that this function is not just a simple send+receive pair: it is more efficient and more correct to + * use this when you expect to do a message swap than do use [send] and then [receive] in turn. + * + * @return an [UntrustworthyData] wrapper around the received object. + */ + @Deprecated("Use FlowSession.sendAndReceive()", level = DeprecationLevel.WARNING) + @Suspendable + open fun sendAndReceive(receiveType: Class, otherParty: Party, payload: Any): UntrustworthyData { + return getDeprecatedSessionForParty(otherParty).sendAndReceive(receiveType, payload) + } + + /** + * Suspends until the specified [otherParty] sends us a message of type [R]. + * + * Remember that when receiving data from other parties the data should not be trusted until it's been thoroughly + * verified for consistency and that all expectations are satisfied, as a malicious peer may send you subtly + * corrupted data in order to exploit your code. + */ + @Deprecated("Use FlowSession.receive()", level = DeprecationLevel.WARNING) + inline fun receive(otherParty: Party): UntrustworthyData = receive(R::class.java, otherParty) + + /** + * Suspends until the specified [otherParty] sends us a message of type [receiveType]. + * + * Remember that when receiving data from other parties the data should not be trusted until it's been thoroughly + * verified for consistency and that all expectations are satisfied, as a malicious peer may send you subtly + * corrupted data in order to exploit your code. + * + * @return an [UntrustworthyData] wrapper around the received object. + */ + @Deprecated("Use FlowSession.receive()", level = DeprecationLevel.WARNING) + @Suspendable + open fun receive(receiveType: Class, otherParty: Party): UntrustworthyData { + return getDeprecatedSessionForParty(otherParty).receive(receiveType) + } + + /** + * Queues the given [payload] for sending to the [otherParty] and continues without suspending. + * + * Note that the other party may receive the message at some arbitrary later point or not at all: if [otherParty] + * is offline then message delivery will be retried until it comes back or until the message is older than the + * network's event horizon time. + */ + @Deprecated("Use FlowSession.send()", level = DeprecationLevel.WARNING) + @Suspendable + open fun send(otherParty: Party, payload: Any) { + getDeprecatedSessionForParty(otherParty).send(payload) + } + @Suspendable internal fun FlowSession.sendAndReceiveWithRetry(receiveType: Class, payload: Any): UntrustworthyData { val request = FlowIORequest.SendAndReceive( @@ -403,4 +495,4 @@ data class FlowInfo( * to deduplicate it from other releases of the same CorDapp, typically a version string. See the * [CorDapp JAR format](https://docs.corda.net/cordapp-build-systems.html#cordapp-jar-format) for more details. */ - val appName: String) \ No newline at end of file + val appName: String) diff --git a/core/src/main/kotlin/net/corda/core/internal/FlowAsyncOperation.kt b/core/src/main/kotlin/net/corda/core/internal/FlowAsyncOperation.kt index c0cd2284d8..cc6984c7e6 100644 --- a/core/src/main/kotlin/net/corda/core/internal/FlowAsyncOperation.kt +++ b/core/src/main/kotlin/net/corda/core/internal/FlowAsyncOperation.kt @@ -30,4 +30,4 @@ interface FlowAsyncOperation { fun FlowLogic.executeAsync(operation: FlowAsyncOperation, maySkipCheckpoint: Boolean = false): R { val request = FlowIORequest.ExecuteAsyncOperation(operation) return stateMachine.suspend(request, maySkipCheckpoint) -} \ No newline at end of file +} diff --git a/core/src/main/kotlin/net/corda/core/internal/FlowIORequest.kt b/core/src/main/kotlin/net/corda/core/internal/FlowIORequest.kt index 4150610f8e..0fe2971033 100644 --- a/core/src/main/kotlin/net/corda/core/internal/FlowIORequest.kt +++ b/core/src/main/kotlin/net/corda/core/internal/FlowIORequest.kt @@ -97,4 +97,3 @@ sealed class FlowIORequest { */ data class ExecuteAsyncOperation(val operation: FlowAsyncOperation) : FlowIORequest() } - diff --git a/core/src/main/kotlin/net/corda/core/utilities/UntrustworthyData.kt b/core/src/main/kotlin/net/corda/core/utilities/UntrustworthyData.kt index b9a0c997aa..13af1fd61e 100644 --- a/core/src/main/kotlin/net/corda/core/utilities/UntrustworthyData.kt +++ b/core/src/main/kotlin/net/corda/core/utilities/UntrustworthyData.kt @@ -53,4 +53,4 @@ fun SerializedBytes.checkPayloadIs(type: Class): Untrustworthy return type.castIfPossible(payloadData)?.let { UntrustworthyData(it) } ?: throw IllegalArgumentException("We were expecting a ${type.name} but we instead got a " + "${payloadData.javaClass.name} (${payloadData})") -} \ No newline at end of file +} diff --git a/node/src/main/kotlin/net/corda/node/services/api/CheckpointStorage.kt b/node/src/main/kotlin/net/corda/node/services/api/CheckpointStorage.kt index a7a3fdaa5d..7b25b78122 100644 --- a/node/src/main/kotlin/net/corda/node/services/api/CheckpointStorage.kt +++ b/node/src/main/kotlin/net/corda/node/services/api/CheckpointStorage.kt @@ -32,7 +32,7 @@ interface CheckpointStorage { /** * Stream all checkpoints from the store. If this is backed by a database the stream will be valid until the - * underlying database connection is open, so any processing should happen before it is closed. + * underlying database connection is closed, so any processing should happen before it is closed. */ fun getAllCheckpoints(): Stream>> } diff --git a/node/src/main/kotlin/net/corda/node/services/events/NodeSchedulerService.kt b/node/src/main/kotlin/net/corda/node/services/events/NodeSchedulerService.kt index 8e591e68d4..277ebbaf52 100644 --- a/node/src/main/kotlin/net/corda/node/services/events/NodeSchedulerService.kt +++ b/node/src/main/kotlin/net/corda/node/services/events/NodeSchedulerService.kt @@ -358,4 +358,4 @@ class NodeSchedulerService(private val clock: CordaClock, null } } -} \ No newline at end of file +} diff --git a/node/src/main/kotlin/net/corda/node/services/messaging/Messaging.kt b/node/src/main/kotlin/net/corda/node/services/messaging/Messaging.kt index 3544334342..77c08f37b5 100644 --- a/node/src/main/kotlin/net/corda/node/services/messaging/Messaging.kt +++ b/node/src/main/kotlin/net/corda/node/services/messaging/Messaging.kt @@ -149,7 +149,7 @@ interface ReceivedMessage : Message { val peer: CordaX500Name /** Platform version of the sender's node. */ val platformVersion: Int - /** UUID representing the sending JVM */ + /** Sequence number of message with respect to senderUUID */ val senderSeqNo: Long? /** True if a flow session init message */ val isSessionInit: Boolean @@ -188,4 +188,3 @@ interface DeduplicationHandler { } typealias MessageHandler = (ReceivedMessage, MessageHandlerRegistration, DeduplicationHandler) -> Unit - diff --git a/node/src/main/kotlin/net/corda/node/services/messaging/MessagingExecutor.kt b/node/src/main/kotlin/net/corda/node/services/messaging/MessagingExecutor.kt index bdd3a81f77..630fb67529 100644 --- a/node/src/main/kotlin/net/corda/node/services/messaging/MessagingExecutor.kt +++ b/node/src/main/kotlin/net/corda/node/services/messaging/MessagingExecutor.kt @@ -70,7 +70,6 @@ class MessagingExecutor( private val releaseVersion = SimpleString(versionInfo.releaseVersion) private val sendMessageSizeMetric = metricRegistry.histogram("SendMessageSize") private val sendLatencyMetric = metricRegistry.timer("SendLatency") - private val sendBatchSizeMetric = metricRegistry.histogram("SendBatchSize") private val ourSenderSeqNo = AtomicLong() private companion object { @@ -185,4 +184,4 @@ class MessagingExecutor( private fun acknowledgeJob(job: Job.Acknowledge) { job.message.individualAcknowledge() } -} \ No newline at end of file +} diff --git a/node/src/main/kotlin/net/corda/node/services/messaging/P2PMessageDeduplicator.kt b/node/src/main/kotlin/net/corda/node/services/messaging/P2PMessageDeduplicator.kt index 0de1464d5f..9896e37337 100644 --- a/node/src/main/kotlin/net/corda/node/services/messaging/P2PMessageDeduplicator.kt +++ b/node/src/main/kotlin/net/corda/node/services/messaging/P2PMessageDeduplicator.kt @@ -169,4 +169,4 @@ class P2PMessageDeduplicator(private val database: CordaPersistence) { private data class MessageMeta(val insertionTime: Instant, val senderHash: String?, val senderSeqNo: Long?) private data class SenderKey(val senderUUID: String, val peer: CordaX500Name, val isSessionInit: Boolean) -} \ No newline at end of file +} diff --git a/node/src/main/kotlin/net/corda/node/services/messaging/P2PMessagingClient.kt b/node/src/main/kotlin/net/corda/node/services/messaging/P2PMessagingClient.kt index 097e601722..c7cc2ce71f 100644 --- a/node/src/main/kotlin/net/corda/node/services/messaging/P2PMessagingClient.kt +++ b/node/src/main/kotlin/net/corda/node/services/messaging/P2PMessagingClient.kt @@ -115,7 +115,6 @@ class P2PMessagingClient(val config: NodeConfiguration, ) : SingletonSerializeAsToken(), MessagingService, AddressToArtemisQueueResolver, AutoCloseable { companion object { private val log = contextLogger() - private val amqDelayMillis = System.getProperty("amq.delivery.delay.ms", "0").toInt() private const val messageMaxRetryCount: Int = 3 fun createMessageToRedeliver(): PersistentMap, RetryMessage, Long> { @@ -426,7 +425,6 @@ class P2PMessagingClient(val config: NodeConfiguration, } internal fun deliver(artemisMessage: ClientMessage) { - artemisToCordaMessage(artemisMessage)?.let { cordaMessage -> if (!deduplicator.isDuplicate(cordaMessage)) { deduplicator.signalMessageProcessStart(cordaMessage) @@ -439,7 +437,6 @@ class P2PMessagingClient(val config: NodeConfiguration, } private fun deliver(msg: ReceivedMessage, artemisMessage: ClientMessage) { - state.checkNotLocked() val deliverTo = handlers[msg.topic] if (deliverTo != null) { @@ -621,7 +618,6 @@ class P2PMessagingClient(val config: NodeConfiguration, } override fun createMessage(topic: String, data: ByteArray, deduplicationId: DeduplicationId, additionalHeaders: Map): Message { - return NodeClientMessage(topic, OpaqueBytes(data), deduplicationId, deduplicator.ourSenderUUID, additionalHeaders) } diff --git a/node/src/main/kotlin/net/corda/node/services/persistence/DBTransactionStorage.kt b/node/src/main/kotlin/net/corda/node/services/persistence/DBTransactionStorage.kt index 0e71ebbf83..380ce87ee1 100644 --- a/node/src/main/kotlin/net/corda/node/services/persistence/DBTransactionStorage.kt +++ b/node/src/main/kotlin/net/corda/node/services/persistence/DBTransactionStorage.kt @@ -108,7 +108,7 @@ class DBTransactionStorage(cacheSizeBytes: Long) : WritableTransactionStorage, S override fun track(): DataFeed, SignedTransaction> { return txStorage.exclusive { - DataFeed(allPersisted().map { it.second.toSignedTx() }.toList(), updatesPublisher.bufferUntilSubscribed().wrapWithDatabaseTransaction()) + DataFeed(allPersisted().map { it.second.toSignedTx() }.toList(), updates.bufferUntilSubscribed().wrapWithDatabaseTransaction()) } } @@ -116,7 +116,7 @@ class DBTransactionStorage(cacheSizeBytes: Long) : WritableTransactionStorage, S return txStorage.exclusive { val existingTransaction = get(id) if (existingTransaction == null) { - updatesPublisher.filter { it.id == id }.toFuture() + updates.filter { it.id == id }.toFuture() } else { doneFuture(existingTransaction.toSignedTx()) } diff --git a/node/src/main/kotlin/net/corda/node/services/statemachine/ActionExecutor.kt b/node/src/main/kotlin/net/corda/node/services/statemachine/ActionExecutor.kt index 5273ab709c..f754a509b0 100644 --- a/node/src/main/kotlin/net/corda/node/services/statemachine/ActionExecutor.kt +++ b/node/src/main/kotlin/net/corda/node/services/statemachine/ActionExecutor.kt @@ -18,7 +18,6 @@ import co.paralleluniverse.fibers.Suspendable interface ActionExecutor { /** * Execute [action] by [fiber]. - * Precondition: [executeAction] is run inside an open database transaction. */ @Suspendable fun executeAction(fiber: FlowFiber, action: Action) diff --git a/node/src/main/kotlin/net/corda/node/services/statemachine/CountUpDownLatch.kt b/node/src/main/kotlin/net/corda/node/services/statemachine/CountUpDownLatch.kt index 54b235bcb3..55dc47cb88 100644 --- a/node/src/main/kotlin/net/corda/node/services/statemachine/CountUpDownLatch.kt +++ b/node/src/main/kotlin/net/corda/node/services/statemachine/CountUpDownLatch.kt @@ -73,4 +73,4 @@ class CountUpDownLatch(initialValue: Int) { fun countUp() { sync.increment() } -} \ No newline at end of file +} diff --git a/node/src/main/kotlin/net/corda/node/services/statemachine/DeduplicationId.kt b/node/src/main/kotlin/net/corda/node/services/statemachine/DeduplicationId.kt index 96087f284a..81e6560f86 100644 --- a/node/src/main/kotlin/net/corda/node/services/statemachine/DeduplicationId.kt +++ b/node/src/main/kotlin/net/corda/node/services/statemachine/DeduplicationId.kt @@ -54,4 +54,4 @@ data class DeduplicationId(val toString: String) { return DeduplicationId("E-$errorId-${recipientSessionId.toLong}") } } -} \ No newline at end of file +} diff --git a/node/src/main/kotlin/net/corda/node/services/statemachine/FlowHospital.kt b/node/src/main/kotlin/net/corda/node/services/statemachine/FlowHospital.kt index b81868b808..1f0d6bb6ed 100644 --- a/node/src/main/kotlin/net/corda/node/services/statemachine/FlowHospital.kt +++ b/node/src/main/kotlin/net/corda/node/services/statemachine/FlowHospital.kt @@ -25,4 +25,4 @@ interface FlowHospital { * The flow running in [flowFiber] has cleaned, possibly as a result of a flow hospital resume. */ fun flowCleaned(flowFiber: FlowFiber) -} \ No newline at end of file +} diff --git a/node/src/main/kotlin/net/corda/node/services/statemachine/FlowLogicRefFactoryImpl.kt b/node/src/main/kotlin/net/corda/node/services/statemachine/FlowLogicRefFactoryImpl.kt index cb0e75ba6b..e7ab42274e 100644 --- a/node/src/main/kotlin/net/corda/node/services/statemachine/FlowLogicRefFactoryImpl.kt +++ b/node/src/main/kotlin/net/corda/node/services/statemachine/FlowLogicRefFactoryImpl.kt @@ -166,4 +166,4 @@ class FlowLogicRefFactoryImpl(private val classloader: ClassLoader) : SingletonS return false } } -} \ No newline at end of file +} diff --git a/node/src/main/kotlin/net/corda/node/services/statemachine/FlowMessaging.kt b/node/src/main/kotlin/net/corda/node/services/statemachine/FlowMessaging.kt index 54a4003036..dbe56f95b3 100644 --- a/node/src/main/kotlin/net/corda/node/services/statemachine/FlowMessaging.kt +++ b/node/src/main/kotlin/net/corda/node/services/statemachine/FlowMessaging.kt @@ -46,7 +46,6 @@ interface FlowMessaging { * Implementation of [FlowMessaging] using a [ServiceHubInternal] to do the messaging and routing. */ class FlowMessagingImpl(val serviceHub: ServiceHubInternal): FlowMessaging { - companion object { val log = contextLogger() @@ -73,7 +72,6 @@ class FlowMessagingImpl(val serviceHub: ServiceHubInternal): FlowMessaging { } private fun SessionMessage.additionalHeaders(target: Party): Map { - // This prevents a "deadlock" in case an initiated flow tries to start a session against a draining node that is also the initiator. // It does not help in case more than 2 nodes are involved in a circle, so the kill switch via RPC should be used in that case. val mightDeadlockDrainingTarget = FlowStateMachineImpl.currentStateMachine()?.context?.origin.let { it is InvocationOrigin.Peer && it.party == target.name } diff --git a/node/src/main/kotlin/net/corda/node/services/statemachine/FlowSessionImpl.kt b/node/src/main/kotlin/net/corda/node/services/statemachine/FlowSessionImpl.kt index 24c59aa6f9..535360032d 100644 --- a/node/src/main/kotlin/net/corda/node/services/statemachine/FlowSessionImpl.kt +++ b/node/src/main/kotlin/net/corda/node/services/statemachine/FlowSessionImpl.kt @@ -96,4 +96,3 @@ class FlowSessionImpl( require(!type.isPrimitive) { "Cannot receive primitive type $type" } } } - diff --git a/node/src/main/kotlin/net/corda/node/services/statemachine/FlowStateMachineImpl.kt b/node/src/main/kotlin/net/corda/node/services/statemachine/FlowStateMachineImpl.kt index eabc9b641a..1d69d656c5 100644 --- a/node/src/main/kotlin/net/corda/node/services/statemachine/FlowStateMachineImpl.kt +++ b/node/src/main/kotlin/net/corda/node/services/statemachine/FlowStateMachineImpl.kt @@ -98,6 +98,11 @@ class FlowStateMachineImpl(override val id: StateMachineRunId, if (value) field = value else throw IllegalArgumentException("Can only set to true") } + /** + * Processes an event by creating the associated transition and executing it using the given executor. + * Try to avoid using this directly, instead use [processEventsUntilFlowIsResumed] or [processEventImmediately] + * instead. + */ @Suspendable private fun processEvent(transitionExecutor: TransitionExecutor, event: Event): FlowContinuation { val stateMachine = getTransientField(TransientValues::stateMachine) @@ -109,22 +114,63 @@ class FlowStateMachineImpl(override val id: StateMachineRunId, return continuation } + /** + * Processes the events in the event queue until a transition indicates that control should be returned to user code + * in the form of a regular resume or a throw of an exception. Alternatively the transition may abort the fiber + * completely. + * + * @param isDbTransactionOpenOnEntry indicates whether a DB transaction is expected to be present before the + * processing of the eventloop. Purely used for internal invariant checks. + * @param isDbTransactionOpenOnExit indicates whether a DB transaction is expected to be present once the eventloop + * processing finished. Purely used for internal invariant checks. + */ @Suspendable - private fun processEventsUntilFlowIsResumed(): Any? { + private fun processEventsUntilFlowIsResumed(isDbTransactionOpenOnEntry: Boolean, isDbTransactionOpenOnExit: Boolean): Any? { + checkDbTransaction(isDbTransactionOpenOnEntry) val transitionExecutor = getTransientField(TransientValues::transitionExecutor) val eventQueue = getTransientField(TransientValues::eventQueue) - eventLoop@while (true) { - val nextEvent = eventQueue.receive() - val continuation = processEvent(transitionExecutor, nextEvent) - when (continuation) { - is FlowContinuation.Resume -> return continuation.result - is FlowContinuation.Throw -> { - continuation.throwable.fillInStackTrace() - throw continuation.throwable + try { + eventLoop@while (true) { + val nextEvent = eventQueue.receive() + val continuation = processEvent(transitionExecutor, nextEvent) + when (continuation) { + is FlowContinuation.Resume -> return continuation.result + is FlowContinuation.Throw -> { + continuation.throwable.fillInStackTrace() + throw continuation.throwable + } + FlowContinuation.ProcessEvents -> continue@eventLoop + FlowContinuation.Abort -> abortFiber() } - FlowContinuation.ProcessEvents -> continue@eventLoop - FlowContinuation.Abort -> abortFiber() } + } finally { + checkDbTransaction(isDbTransactionOpenOnExit) + } + } + + /** + * Immediately processes the passed in event. Always called with an open database transaction. + * + * @param event the event to be processed. + * @param isDbTransactionOpenOnEntry indicates whether a DB transaction is expected to be present before the + * processing of the event. Purely used for internal invariant checks. + * @param isDbTransactionOpenOnExit indicates whether a DB transaction is expected to be present once the event + * processing finished. Purely used for internal invariant checks. + */ + @Suspendable + private fun processEventImmediately(event: Event, isDbTransactionOpenOnEntry: Boolean, isDbTransactionOpenOnExit: Boolean): FlowContinuation { + checkDbTransaction(isDbTransactionOpenOnEntry) + val transitionExecutor = getTransientField(TransientValues::transitionExecutor) + val continuation = processEvent(transitionExecutor, event) + checkDbTransaction(isDbTransactionOpenOnExit) + return continuation + } + + private fun checkDbTransaction(isPresent: Boolean) { + if (isPresent) { + requireNotNull(contextTransactionOrNull != null) + } else { + require(contextTransactionOrNull == null) } } @@ -155,32 +201,56 @@ class FlowStateMachineImpl(override val id: StateMachineRunId, Event.Error(resultOrError.exception) } } - scheduleEvent(finalEvent) - processEventsUntilFlowIsResumed() + // Immediately process the last event. This is to make sure the transition can assume that it has an open + // database transaction. + val continuation = processEventImmediately( + finalEvent, + isDbTransactionOpenOnEntry = true, + isDbTransactionOpenOnExit = false + ) + if (continuation == FlowContinuation.ProcessEvents) { + // This can happen in case there was an error and there are further things to do e.g. to propagate it. + processEventsUntilFlowIsResumed( + isDbTransactionOpenOnEntry = false, + isDbTransactionOpenOnExit = false + ) + } recordDuration(startTime) } @Suspendable private fun initialiseFlow() { - processEventsUntilFlowIsResumed() + processEventsUntilFlowIsResumed( + isDbTransactionOpenOnEntry = false, + isDbTransactionOpenOnExit = true + ) } @Suspendable override fun subFlow(subFlow: FlowLogic): R { - processEvent(getTransientField(TransientValues::transitionExecutor), Event.EnterSubFlow(subFlow.javaClass)) + processEventImmediately( + Event.EnterSubFlow(subFlow.javaClass), + isDbTransactionOpenOnEntry = true, + isDbTransactionOpenOnExit = true + ) return try { subFlow.call() } finally { - processEvent(getTransientField(TransientValues::transitionExecutor), Event.LeaveSubFlow) + processEventImmediately( + Event.LeaveSubFlow, + isDbTransactionOpenOnEntry = true, + isDbTransactionOpenOnExit = true + ) } } @Suspendable override fun initiateFlow(party: Party): FlowSession { - val resume = processEvent( - getTransientField(TransientValues::transitionExecutor), - Event.InitiateFlow(party) + val resume = processEventImmediately( + Event.InitiateFlow(party), + isDbTransactionOpenOnEntry = true, + isDbTransactionOpenOnExit = true ) as FlowContinuation.Resume return resume.result as FlowSession } @@ -237,7 +307,6 @@ class FlowStateMachineImpl(override val id: StateMachineRunId, override fun suspend(ioRequest: FlowIORequest, maySkipCheckpoint: Boolean): R { val serializationContext = TransientReference(getTransientField(TransientValues::checkpointSerializationContext)) val transaction = extractThreadLocalTransaction() - val transitionExecutor = TransientReference(getTransientField(TransientValues::transitionExecutor)) parkAndSerialize { _, _ -> logger.trace { "Suspended on $ioRequest" } @@ -252,12 +321,20 @@ class FlowStateMachineImpl(override val id: StateMachineRunId, Event.Error(throwable) } - // We must commit the database transaction before returning from this closure, otherwise Quasar may schedule - // other fibers - require(processEvent(transitionExecutor.value, event) == FlowContinuation.ProcessEvents) + // We must commit the database transaction before returning from this closure otherwise Quasar may schedule + // other fibers, so we process the event immediately + val continuation = processEventImmediately( + event, + isDbTransactionOpenOnEntry = true, + isDbTransactionOpenOnExit = false + ) + require(continuation == FlowContinuation.ProcessEvents) Fiber.unparkDeserialized(this, scheduler) } - return uncheckedCast(processEventsUntilFlowIsResumed()) + return uncheckedCast(processEventsUntilFlowIsResumed( + isDbTransactionOpenOnEntry = false, + isDbTransactionOpenOnExit = true + )) } @Suspendable diff --git a/node/src/main/kotlin/net/corda/node/services/statemachine/PropagatingFlowHospital.kt b/node/src/main/kotlin/net/corda/node/services/statemachine/PropagatingFlowHospital.kt index 0d6caea551..0f3ad147f8 100644 --- a/node/src/main/kotlin/net/corda/node/services/statemachine/PropagatingFlowHospital.kt +++ b/node/src/main/kotlin/net/corda/node/services/statemachine/PropagatingFlowHospital.kt @@ -28,4 +28,3 @@ object PropagatingFlowHospital : FlowHospital { throw IllegalStateException("Flow ${flowFiber.id} cleaned after error propagation triggered") } } - diff --git a/node/src/main/kotlin/net/corda/node/services/statemachine/SessionRejectException.kt b/node/src/main/kotlin/net/corda/node/services/statemachine/SessionRejectException.kt index 11642a59ed..9c46c0dc4e 100644 --- a/node/src/main/kotlin/net/corda/node/services/statemachine/SessionRejectException.kt +++ b/node/src/main/kotlin/net/corda/node/services/statemachine/SessionRejectException.kt @@ -15,4 +15,4 @@ import net.corda.core.CordaException /** * An exception propagated and thrown in case a session initiation fails. */ -class SessionRejectException(reason: String) : CordaException(reason) \ No newline at end of file +class SessionRejectException(reason: String) : CordaException(reason) diff --git a/node/src/main/kotlin/net/corda/node/services/statemachine/SingleThreadedStateMachineManager.kt b/node/src/main/kotlin/net/corda/node/services/statemachine/SingleThreadedStateMachineManager.kt index 28518acb28..03f770809c 100644 --- a/node/src/main/kotlin/net/corda/node/services/statemachine/SingleThreadedStateMachineManager.kt +++ b/node/src/main/kotlin/net/corda/node/services/statemachine/SingleThreadedStateMachineManager.kt @@ -44,6 +44,7 @@ import net.corda.node.services.messaging.DeduplicationHandler import net.corda.node.services.messaging.ReceivedMessage import net.corda.node.services.statemachine.interceptors.* import net.corda.node.services.statemachine.transitions.StateMachine +import net.corda.node.services.statemachine.transitions.StateMachineConfiguration import net.corda.node.utilities.AffinityExecutor import net.corda.nodeapi.internal.persistence.CordaPersistence import net.corda.nodeapi.internal.serialization.SerializeAsTokenContextImpl diff --git a/node/src/main/kotlin/net/corda/node/services/statemachine/StateMachineState.kt b/node/src/main/kotlin/net/corda/node/services/statemachine/StateMachineState.kt index fafdf5533e..0dedc0062b 100644 --- a/node/src/main/kotlin/net/corda/node/services/statemachine/StateMachineState.kt +++ b/node/src/main/kotlin/net/corda/node/services/statemachine/StateMachineState.kt @@ -239,4 +239,3 @@ sealed class ErrorState { } } } - diff --git a/node/src/main/kotlin/net/corda/node/services/statemachine/interceptors/DumpHistoryOnErrorInterceptor.kt b/node/src/main/kotlin/net/corda/node/services/statemachine/interceptors/DumpHistoryOnErrorInterceptor.kt index 91ec293007..7f0911d765 100644 --- a/node/src/main/kotlin/net/corda/node/services/statemachine/interceptors/DumpHistoryOnErrorInterceptor.kt +++ b/node/src/main/kotlin/net/corda/node/services/statemachine/interceptors/DumpHistoryOnErrorInterceptor.kt @@ -58,4 +58,4 @@ class DumpHistoryOnErrorInterceptor(val delegate: TransitionExecutor) : Transiti return Pair(continuation, nextState) } -} \ No newline at end of file +} diff --git a/node/src/main/kotlin/net/corda/node/services/statemachine/interceptors/HospitalisingInterceptor.kt b/node/src/main/kotlin/net/corda/node/services/statemachine/interceptors/HospitalisingInterceptor.kt index e0bcd32883..af10c25db0 100644 --- a/node/src/main/kotlin/net/corda/node/services/statemachine/interceptors/HospitalisingInterceptor.kt +++ b/node/src/main/kotlin/net/corda/node/services/statemachine/interceptors/HospitalisingInterceptor.kt @@ -53,4 +53,4 @@ class HospitalisingInterceptor( } return Pair(continuation, nextState) } -} \ No newline at end of file +} diff --git a/node/src/main/kotlin/net/corda/node/services/statemachine/interceptors/TransitionDiagnosticRecord.kt b/node/src/main/kotlin/net/corda/node/services/statemachine/interceptors/TransitionDiagnosticRecord.kt index e046e27872..afef39c099 100644 --- a/node/src/main/kotlin/net/corda/node/services/statemachine/interceptors/TransitionDiagnosticRecord.kt +++ b/node/src/main/kotlin/net/corda/node/services/statemachine/interceptors/TransitionDiagnosticRecord.kt @@ -58,4 +58,4 @@ data class TransitionDiagnosticRecord( ) ).joinToString("\n") } -} \ No newline at end of file +} diff --git a/node/src/main/kotlin/net/corda/node/services/statemachine/transitions/DeliverSessionMessageTransition.kt b/node/src/main/kotlin/net/corda/node/services/statemachine/transitions/DeliverSessionMessageTransition.kt index 303eb60527..93ecae468a 100644 --- a/node/src/main/kotlin/net/corda/node/services/statemachine/transitions/DeliverSessionMessageTransition.kt +++ b/node/src/main/kotlin/net/corda/node/services/statemachine/transitions/DeliverSessionMessageTransition.kt @@ -196,4 +196,4 @@ class DeliverSessionMessageTransition( } } -} \ No newline at end of file +} diff --git a/node/src/main/kotlin/net/corda/node/services/statemachine/transitions/DoRemainingWorkTransition.kt b/node/src/main/kotlin/net/corda/node/services/statemachine/transitions/DoRemainingWorkTransition.kt index 8840c94d82..49490eeaf4 100644 --- a/node/src/main/kotlin/net/corda/node/services/statemachine/transitions/DoRemainingWorkTransition.kt +++ b/node/src/main/kotlin/net/corda/node/services/statemachine/transitions/DoRemainingWorkTransition.kt @@ -44,4 +44,4 @@ class DoRemainingWorkTransition( private fun erroredTransition(errorState: ErrorState.Errored): TransitionResult { return ErrorFlowTransition(context, startingState, errorState).transition() } -} \ No newline at end of file +} diff --git a/node/src/main/kotlin/net/corda/node/services/statemachine/transitions/ErrorFlowTransition.kt b/node/src/main/kotlin/net/corda/node/services/statemachine/transitions/ErrorFlowTransition.kt index b8808ad9d3..c614879ee2 100644 --- a/node/src/main/kotlin/net/corda/node/services/statemachine/transitions/ErrorFlowTransition.kt +++ b/node/src/main/kotlin/net/corda/node/services/statemachine/transitions/ErrorFlowTransition.kt @@ -132,4 +132,4 @@ class ErrorFlowTransition( } return Pair(initiatedSessions, newSessions) } -} \ No newline at end of file +} diff --git a/node/src/main/kotlin/net/corda/node/services/statemachine/transitions/StartedFlowTransition.kt b/node/src/main/kotlin/net/corda/node/services/statemachine/transitions/StartedFlowTransition.kt index c313ee59e7..d6455032f8 100644 --- a/node/src/main/kotlin/net/corda/node/services/statemachine/transitions/StartedFlowTransition.kt +++ b/node/src/main/kotlin/net/corda/node/services/statemachine/transitions/StartedFlowTransition.kt @@ -417,4 +417,4 @@ class StartedFlowTransition( FlowContinuation.ProcessEvents } } -} \ No newline at end of file +} diff --git a/node/src/main/kotlin/net/corda/node/services/statemachine/transitions/TopLevelTransition.kt b/node/src/main/kotlin/net/corda/node/services/statemachine/transitions/TopLevelTransition.kt index 38aa19f11e..6e3b57ca53 100644 --- a/node/src/main/kotlin/net/corda/node/services/statemachine/transitions/TopLevelTransition.kt +++ b/node/src/main/kotlin/net/corda/node/services/statemachine/transitions/TopLevelTransition.kt @@ -251,4 +251,4 @@ class TopLevelTransition( resumeFlowLogic(event.returnValue) } } -} \ No newline at end of file +} diff --git a/node/src/main/kotlin/net/corda/node/services/statemachine/transitions/UnstartedFlowTransition.kt b/node/src/main/kotlin/net/corda/node/services/statemachine/transitions/UnstartedFlowTransition.kt index e347bb1de6..cbb62728da 100644 --- a/node/src/main/kotlin/net/corda/node/services/statemachine/transitions/UnstartedFlowTransition.kt +++ b/node/src/main/kotlin/net/corda/node/services/statemachine/transitions/UnstartedFlowTransition.kt @@ -87,4 +87,4 @@ class UnstartedFlowTransition( isAnyCheckpointPersisted = true ) } -} \ No newline at end of file +} diff --git a/node/src/test/kotlin/net/corda/node/services/events/NodeSchedulerServiceTest.kt b/node/src/test/kotlin/net/corda/node/services/events/NodeSchedulerServiceTest.kt index 38919396fe..07f020cf53 100644 --- a/node/src/test/kotlin/net/corda/node/services/events/NodeSchedulerServiceTest.kt +++ b/node/src/test/kotlin/net/corda/node/services/events/NodeSchedulerServiceTest.kt @@ -31,6 +31,7 @@ import net.corda.testing.internal.doLookup import net.corda.testing.internal.rigorousMock import net.corda.testing.node.MockServices import net.corda.testing.node.TestClock +import org.junit.Ignore import org.junit.Rule import org.junit.Test import org.junit.rules.TestWatcher @@ -275,6 +276,7 @@ class NodeSchedulerPersistenceTest : NodeSchedulerServiceTestBase() { newDatabase.close() } + @Ignore("Temporarily") @Test fun `test that if schedule is updated then the flow is invoked on the correct schedule`() { val dataSourceProps = MockServices.makeTestDataSourceProperties() @@ -306,4 +308,4 @@ class NodeSchedulerPersistenceTest : NodeSchedulerServiceTestBase() { scheduler.join() database.close() } -} \ No newline at end of file +} diff --git a/node/src/test/kotlin/net/corda/node/services/transactions/MaxTransactionSizeTests.kt b/node/src/test/kotlin/net/corda/node/services/transactions/MaxTransactionSizeTests.kt index 297873b161..554a947e08 100644 --- a/node/src/test/kotlin/net/corda/node/services/transactions/MaxTransactionSizeTests.kt +++ b/node/src/test/kotlin/net/corda/node/services/transactions/MaxTransactionSizeTests.kt @@ -141,4 +141,4 @@ class MaxTransactionSizeTests { otherSide.send(Unit) } } -} \ No newline at end of file +} diff --git a/testing/node-driver/src/main/kotlin/net/corda/testing/node/internal/InternalMockNetwork.kt b/testing/node-driver/src/main/kotlin/net/corda/testing/node/internal/InternalMockNetwork.kt index d86b2a84ff..746dd2e543 100644 --- a/testing/node-driver/src/main/kotlin/net/corda/testing/node/internal/InternalMockNetwork.kt +++ b/testing/node-driver/src/main/kotlin/net/corda/testing/node/internal/InternalMockNetwork.kt @@ -25,7 +25,6 @@ import net.corda.core.internal.VisibleForTesting import net.corda.core.internal.createDirectories import net.corda.core.internal.createDirectory import net.corda.core.internal.uncheckedCast -import net.corda.core.messaging.CordaRPCOps import net.corda.core.messaging.MessageRecipients import net.corda.core.messaging.RPCOps import net.corda.core.messaging.SingleMessageRecipient diff --git a/testing/node-driver/src/main/kotlin/net/corda/testing/node/internal/MockTransactionStorage.kt b/testing/node-driver/src/main/kotlin/net/corda/testing/node/internal/MockTransactionStorage.kt index fb80bc0c91..2a0e00f903 100644 --- a/testing/node-driver/src/main/kotlin/net/corda/testing/node/internal/MockTransactionStorage.kt +++ b/testing/node-driver/src/main/kotlin/net/corda/testing/node/internal/MockTransactionStorage.kt @@ -20,7 +20,7 @@ import net.corda.core.transactions.SignedTransaction import net.corda.node.services.api.WritableTransactionStorage import rx.Observable import rx.subjects.PublishSubject -import java.util.HashMap +import java.util.* /** * A class which provides an implementation of [WritableTransactionStorage] which is used in [MockServices]