mirror of
https://github.com/corda/corda.git
synced 2025-01-18 10:46:38 +00:00
CORDA-3677 FlowExternalOperation serialising reference to FlowLogic (#6094)
* Stop capturing 'FlowLogic' references in flowAsyncOperation; Creating concrete classes removes the implicit reference to FlowLogic (as this) being included in the anonymous object * Modify test code so that lambdas no longer get implicit references to their enclosing 'FlowLogic'
This commit is contained in:
parent
668748b054
commit
f6952963a8
@ -204,12 +204,14 @@ class FlowExternalAsyncOperationTest : AbstractFlowExternalOperationTest() {
|
|||||||
FlowWithExternalProcess(party) {
|
FlowWithExternalProcess(party) {
|
||||||
|
|
||||||
@Suspendable
|
@Suspendable
|
||||||
override fun testCode(): Any =
|
override fun testCode(): Any {
|
||||||
await(ExternalAsyncOperation(serviceHub) { _, _ ->
|
val e = createException()
|
||||||
|
return await(ExternalAsyncOperation(serviceHub) { _, _ ->
|
||||||
CompletableFuture<Any>().apply {
|
CompletableFuture<Any>().apply {
|
||||||
completeExceptionally(createException())
|
completeExceptionally(e)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
}
|
||||||
|
|
||||||
private fun createException() = when (exceptionType) {
|
private fun createException() = when (exceptionType) {
|
||||||
HospitalizeFlowException::class.java -> HospitalizeFlowException("keep it around")
|
HospitalizeFlowException::class.java -> HospitalizeFlowException("keep it around")
|
||||||
|
@ -260,7 +260,10 @@ class FlowExternalOperationTest : AbstractFlowExternalOperationTest() {
|
|||||||
FlowWithExternalProcess(party) {
|
FlowWithExternalProcess(party) {
|
||||||
|
|
||||||
@Suspendable
|
@Suspendable
|
||||||
override fun testCode(): Any = await(ExternalOperation(serviceHub) { _, _ -> throw createException() })
|
override fun testCode() {
|
||||||
|
val e = createException()
|
||||||
|
await(ExternalOperation(serviceHub) { _, _ -> throw e })
|
||||||
|
}
|
||||||
|
|
||||||
private fun createException() = when (exceptionType) {
|
private fun createException() = when (exceptionType) {
|
||||||
HospitalizeFlowException::class.java -> HospitalizeFlowException("keep it around")
|
HospitalizeFlowException::class.java -> HospitalizeFlowException("keep it around")
|
||||||
|
@ -531,12 +531,7 @@ abstract class FlowLogic<out T> {
|
|||||||
@Suspendable
|
@Suspendable
|
||||||
fun <R : Any> await(operation: FlowExternalAsyncOperation<R>): R {
|
fun <R : Any> await(operation: FlowExternalAsyncOperation<R>): R {
|
||||||
// Wraps the passed in [FlowExternalAsyncOperation] so its [CompletableFuture] can be converted into a [CordaFuture]
|
// Wraps the passed in [FlowExternalAsyncOperation] so its [CompletableFuture] can be converted into a [CordaFuture]
|
||||||
val flowAsyncOperation = object : FlowAsyncOperation<R>, WrappedFlowExternalAsyncOperation<R> {
|
val flowAsyncOperation = WrappedFlowExternalAsyncOperation(operation)
|
||||||
override val operation = operation
|
|
||||||
override fun execute(deduplicationId: String): CordaFuture<R> {
|
|
||||||
return this.operation.execute(deduplicationId).asCordaFuture()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
val request = FlowIORequest.ExecuteAsyncOperation(flowAsyncOperation)
|
val request = FlowIORequest.ExecuteAsyncOperation(flowAsyncOperation)
|
||||||
return stateMachine.suspend(request, false)
|
return stateMachine.suspend(request, false)
|
||||||
}
|
}
|
||||||
@ -550,18 +545,7 @@ abstract class FlowLogic<out T> {
|
|||||||
*/
|
*/
|
||||||
@Suspendable
|
@Suspendable
|
||||||
fun <R : Any> await(operation: FlowExternalOperation<R>): R {
|
fun <R : Any> await(operation: FlowExternalOperation<R>): R {
|
||||||
val flowAsyncOperation = object : FlowAsyncOperation<R>, WrappedFlowExternalOperation<R> {
|
val flowAsyncOperation = WrappedFlowExternalOperation(serviceHub as ServiceHubCoreInternal, operation)
|
||||||
override val serviceHub = this@FlowLogic.serviceHub as ServiceHubCoreInternal
|
|
||||||
override val operation = operation
|
|
||||||
override fun execute(deduplicationId: String): CordaFuture<R> {
|
|
||||||
// Using a [CompletableFuture] allows unhandled exceptions to be thrown inside the background operation
|
|
||||||
// the exceptions will be set on the future by [CompletableFuture.AsyncSupply.run]
|
|
||||||
return CompletableFuture.supplyAsync(
|
|
||||||
Supplier { this.operation.execute(deduplicationId) },
|
|
||||||
serviceHub.externalOperationExecutor
|
|
||||||
).asCordaFuture()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
val request = FlowIORequest.ExecuteAsyncOperation(flowAsyncOperation)
|
val request = FlowIORequest.ExecuteAsyncOperation(flowAsyncOperation)
|
||||||
return stateMachine.suspend(request, false)
|
return stateMachine.suspend(request, false)
|
||||||
}
|
}
|
||||||
@ -571,21 +555,32 @@ abstract class FlowLogic<out T> {
|
|||||||
* [WrappedFlowExternalAsyncOperation] is added to allow jackson to properly reference the data stored within the wrapped
|
* [WrappedFlowExternalAsyncOperation] is added to allow jackson to properly reference the data stored within the wrapped
|
||||||
* [FlowExternalAsyncOperation].
|
* [FlowExternalAsyncOperation].
|
||||||
*/
|
*/
|
||||||
private interface WrappedFlowExternalAsyncOperation<R : Any> {
|
private class WrappedFlowExternalAsyncOperation<R : Any>(val operation: FlowExternalAsyncOperation<R>) : FlowAsyncOperation<R> {
|
||||||
val operation: FlowExternalAsyncOperation<R>
|
override fun execute(deduplicationId: String): CordaFuture<R> {
|
||||||
|
return operation.execute(deduplicationId).asCordaFuture()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* [WrappedFlowExternalOperation] is added to allow jackson to properly reference the data stored within the wrapped
|
* [WrappedFlowExternalOperation] is added to allow jackson to properly reference the data stored within the wrapped
|
||||||
* [FlowExternalOperation].
|
* [FlowExternalOperation].
|
||||||
*
|
*
|
||||||
* The reference to [ServiceHub] is is also needed by Kryo to properly keep a reference to [ServiceHub] so that
|
* The reference to [ServiceHub] is also needed by Kryo to properly keep a reference to [ServiceHub] so that
|
||||||
* [FlowExternalOperation] can be run from the [ServiceHubCoreInternal.externalOperationExecutor] without causing errors when retrying a
|
* [FlowExternalOperation] can be run from the [ServiceHubCoreInternal.externalOperationExecutor] without causing errors when retrying a
|
||||||
* flow. A [NullPointerException] is thrown if [FlowLogic.serviceHub] is accessed from [FlowLogic.await] when retrying a flow.
|
* flow. A [NullPointerException] is thrown if [FlowLogic.serviceHub] is accessed from [FlowLogic.await] when retrying a flow.
|
||||||
*/
|
*/
|
||||||
private interface WrappedFlowExternalOperation<R : Any> {
|
private class WrappedFlowExternalOperation<R : Any>(
|
||||||
val serviceHub: ServiceHub
|
val serviceHub: ServiceHubCoreInternal,
|
||||||
val operation: FlowExternalOperation<R>
|
val operation: FlowExternalOperation<R>
|
||||||
|
) : FlowAsyncOperation<R> {
|
||||||
|
override fun execute(deduplicationId: String): CordaFuture<R> {
|
||||||
|
// Using a [CompletableFuture] allows unhandled exceptions to be thrown inside the background operation
|
||||||
|
// the exceptions will be set on the future by [CompletableFuture.AsyncSupply.run]
|
||||||
|
return CompletableFuture.supplyAsync(
|
||||||
|
Supplier { this.operation.execute(deduplicationId) },
|
||||||
|
serviceHub.externalOperationExecutor
|
||||||
|
).asCordaFuture()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
Loading…
Reference in New Issue
Block a user