ENT-10290 Create Enterprise Aliases for all new Recovery Flows (#7440)

This commit is contained in:
Jose Coll 2023-08-15 15:32:54 +01:00 committed by GitHub
parent 0130914c89
commit d2029b3e0c
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 182 additions and 0 deletions

View File

@ -0,0 +1,97 @@
package net.corda.core.flows
import co.paralleluniverse.fibers.Suspendable
import net.corda.core.CordaInternal
import net.corda.core.crypto.SecureHash
import net.corda.core.flows.FinalityFlow.Companion.tracker
import net.corda.core.identity.CordaX500Name
import net.corda.core.serialization.CordaSerializable
import net.corda.core.utilities.ProgressTracker
import java.time.Instant
/**
* TWO_PHASE_FINALITY Recovery Flow
* This flow is exposed via the Core API for use by any CorDapp but its implementation is available in Enterprise only.
*/
@StartableByRPC
@InitiatingFlow
class FinalityRecoveryFlow(
private val txIds: Collection<SecureHash> = emptySet(),
private val flowIds: Collection<StateMachineRunId> = emptySet(),
private val matchingCriteria: FlowRecoveryQuery? = null,
private val forceRecover: Boolean = false,
private val recoverAll: Boolean = false,
override val progressTracker: ProgressTracker = ProgressTracker()) : FlowLogic<Map<FlowTransactionInfo, Boolean>>() {
@CordaInternal
data class ExtraConstructorArgs(val txIds: Collection<SecureHash>,
val flowIds: Collection<StateMachineRunId>,
val matchingCriteria: FlowRecoveryQuery?,
val forceRecover: Boolean,
val recoverAll: Boolean)
@CordaInternal
fun getExtraConstructorArgs() = ExtraConstructorArgs(txIds, flowIds, matchingCriteria, forceRecover, recoverAll)
constructor(txId: SecureHash, forceRecover: Boolean = false) : this(setOf(txId), forceRecover)
constructor(txIds: Collection<SecureHash>, forceRecover: Boolean = false, recoverAll: Boolean = false) : this(txIds, emptySet(), null, forceRecover, recoverAll, tracker())
constructor(flowId: StateMachineRunId, forceRecover: Boolean = false) : this(emptySet(), setOf(flowId), null, forceRecover)
constructor(flowIds: Collection<StateMachineRunId>, forceRecover: Boolean = false) : this(emptySet(), flowIds, null, forceRecover, false, tracker())
constructor(recoverAll: Boolean, forceRecover: Boolean = false) : this(emptySet(), emptySet(), null, forceRecover, recoverAll, tracker())
constructor(matchingCriteria: FlowRecoveryQuery, forceRecover: Boolean = false) : this(emptySet(), emptySet(), matchingCriteria, forceRecover, false, tracker())
@Suspendable
@Throws(FlowRecoveryException::class)
override fun call(): Map<FlowTransactionInfo, Boolean> {
throw NotImplementedError("Enterprise only feature")
}
}
@CordaSerializable
class FlowRecoveryException(message: String, cause: Throwable? = null) : FlowException(message, cause) {
constructor(txnId: SecureHash, message: String, cause: Throwable? = null) : this("Flow recovery failed for transaction $txnId: $message", cause)
}
@CordaSerializable
data class FlowRecoveryQuery(
val timeframe: FlowTimeWindow? = null,
val initiatedBy: CordaX500Name? = null,
val counterParties: List<CordaX500Name>? = null) {
init {
require(timeframe != null || initiatedBy != null || counterParties != null) {
"Must specify at least one recovery criteria"
}
}
}
@CordaSerializable
data class FlowTimeWindow(val fromTime: Instant? = null, val untilTime: Instant? = null) {
init {
if (fromTime == null && untilTime == null)
throw IllegalArgumentException("Must specify one or both of fromTime or/and untilTime")
fromTime?.let { startTime ->
untilTime?.let { endTime ->
if (endTime < startTime) {
throw IllegalArgumentException(FlowTimeWindow::fromTime.name + " must be before or equal to " + FlowTimeWindow::untilTime.name)
}
}
}
}
companion object {
@JvmStatic
fun between(fromTime: Instant, untilTime: Instant): FlowTimeWindow {
return FlowTimeWindow(fromTime, untilTime)
}
@JvmStatic
fun fromOnly(fromTime: Instant): FlowTimeWindow {
return FlowTimeWindow(fromTime = fromTime)
}
@JvmStatic
fun untilOnly(untilTime: Instant): FlowTimeWindow {
return FlowTimeWindow(untilTime = untilTime)
}
}
}

View File

@ -0,0 +1,85 @@
package net.corda.core.flows
import co.paralleluniverse.fibers.Suspendable
import net.corda.core.CordaInternal
import net.corda.core.crypto.SecureHash
import net.corda.core.identity.CordaX500Name
import net.corda.core.identity.Party
import net.corda.core.serialization.CordaSerializable
import net.corda.core.utilities.ProgressTracker
/**
* Ledger Recovery Flow (available in Enterprise only).
*/
@StartableByRPC
@InitiatingFlow
class LedgerRecoveryFlow(
private val recoveryPeers: Collection<Party>,
private val timeWindow: RecoveryTimeWindow,
private val useAllNetworkNodes: Boolean = false,
private val transactionRole: TransactionRole = TransactionRole.ALL,
private val dryRun: Boolean = false,
private val optimisticInitiatorRecovery: Boolean = false,
override val progressTracker: ProgressTracker = ProgressTracker()) : FlowLogic<Map<SecureHash, RecoveryResult>>() {
@CordaInternal
data class ExtraConstructorArgs(val recoveryPeers: Collection<Party>,
val timeWindow: RecoveryTimeWindow,
val useAllNetworkNodes: Boolean,
val transactionRole: TransactionRole,
val dryRun: Boolean,
val optimisticInitiatorRecovery: Boolean)
@CordaInternal
fun getExtraConstructorArgs() = ExtraConstructorArgs(recoveryPeers, timeWindow, useAllNetworkNodes, transactionRole, dryRun, optimisticInitiatorRecovery)
// unused constructors added to facilitate Node Shell command invocation
constructor(recoveryPeer: Party, timeWindow: RecoveryTimeWindow) : this(setOf(recoveryPeer), timeWindow, false, TransactionRole.ALL, false, false)
constructor(recoveryPeer: Party, timeWindow: RecoveryTimeWindow, dryRun: Boolean) : this(setOf(recoveryPeer), timeWindow, false, TransactionRole.ALL, dryRun, false)
constructor(timeWindow: RecoveryTimeWindow, dryRun: Boolean) : this(emptySet(), timeWindow, false, TransactionRole.ALL, dryRun, false)
constructor(timeWindow: RecoveryTimeWindow, dryRun: Boolean, optimisticInitiatorRecovery: Boolean) : this(emptySet(), timeWindow, false, TransactionRole.ALL, dryRun, optimisticInitiatorRecovery)
constructor(recoveryPeers: Collection<Party>, timeWindow: RecoveryTimeWindow, dryRun: Boolean) : this(recoveryPeers, timeWindow, false, TransactionRole.ALL, dryRun, false)
constructor(recoveryPeers: Collection<Party>, timeWindow: RecoveryTimeWindow, dryRun: Boolean, optimisticInitiatorRecovery: Boolean) : this(recoveryPeers, timeWindow, false, TransactionRole.ALL, dryRun, optimisticInitiatorRecovery)
@Suspendable
@Throws(LedgerRecoveryException::class)
override fun call(): Map<SecureHash, RecoveryResult> {
throw NotImplementedError("Enterprise only feature")
}
}
@InitiatedBy(LedgerRecoveryFlow::class)
class ReceiveLedgerRecoveryFlow constructor(private val otherSideSession: FlowSession) : FlowLogic<Unit>() {
@Suspendable
override fun call() {
throw NotImplementedError("Enterprise only feature")
}
}
@CordaSerializable
class LedgerRecoveryException(message: String) : FlowException("Ledger recovery failed: $message")
/**
* This specifies which type of transactions to recover based on the transaction role of the recovering node
*/
@CordaSerializable
enum class TransactionRole {
ALL,
INITIATOR, // only recover transactions that I initiated
PEER, // only recover transactions where I am a participant on a transaction
OBSERVER, // only recover transactions where I am an observer (but not participant) to a transaction
PEER_AND_OBSERVER // recovery transactions where I am either participant or observer
}
@CordaSerializable
data class RecoveryResult(
val transactionId: SecureHash,
val recoveryPeer: CordaX500Name,
val transactionRole: TransactionRole, // what role did I play in this transaction
val synchronised: Boolean, // whether the transaction was successfully synchronised (will always be false when dryRun option specified)
val synchronisedInitiated: Boolean = false, // only attempted if [optimisticInitiatorRecovery] option set to true and [TransactionRecoveryType.INITIATOR]
val failureCause: String? = null // reason why a transaction failed to synchronise
)