mirror of
https://github.com/corda/corda.git
synced 2025-01-18 02:39:51 +00:00
CORDA-2531 State mappings are now also updated for new reference states. (#4681)
* State mappings are now also updated for new reference states. Added a test to prove the above works. * Addressed Rick's review comments.
This commit is contained in:
parent
7dc7313fb7
commit
365471d9f4
@ -85,9 +85,8 @@ class ReferencedStatesFlowTests {
|
||||
// Wait until node 1 stores the new tx.
|
||||
nodes[1].services.validatedTransactions.updates.filter { it.id == newTx.id }.toFuture().getOrThrow()
|
||||
// Check that nodes[1] has finished recording the transaction (and updating the vault.. hopefully!).
|
||||
val allRefStates = nodes[1].services.vaultService.queryBy<RefState.State>()
|
||||
// nodes[1] should have two states. The newly created output and the reference state created by nodes[0].
|
||||
assertEquals(2, allRefStates.states.size)
|
||||
// nodes[1] should have two states. The newly created output of type "Regular.State" and the reference state created by nodes[0].
|
||||
assertEquals(2, nodes[1].services.vaultService.queryBy<LinearState>().states.size)
|
||||
// Now let's find the specific reference state on nodes[1].
|
||||
val refStateLinearId = newRefState.state.data.linearId
|
||||
val query = QueryCriteria.LinearStateQueryCriteria(linearId = listOf(refStateLinearId))
|
||||
@ -99,14 +98,27 @@ class ReferencedStatesFlowTests {
|
||||
val nodeZeroQuery = QueryCriteria.LinearStateQueryCriteria(linearId = listOf(refStateLinearId))
|
||||
val theReferencedStateOnNodeZero = nodes[0].services.vaultService.queryBy<RefState.State>(nodeZeroQuery)
|
||||
assertEquals(newRefState, theReferencedStateOnNodeZero.states.single())
|
||||
println(theReferencedStateOnNodeZero.statesMetadata.single())
|
||||
// nodes[0] sends the tx that created the reference state to nodes[1].
|
||||
nodes[0].services.startFlow(Initiator(newRefState)).resultFuture.getOrThrow()
|
||||
// Query again.
|
||||
val theReferencedStateAgain = nodes[1].services.vaultService.queryBy<RefState.State>(query)
|
||||
// There should be one result - the reference state.
|
||||
assertEquals(newRefState, theReferencedStateAgain.states.single())
|
||||
println(theReferencedStateAgain.statesMetadata.single())
|
||||
}
|
||||
|
||||
@Test
|
||||
fun `check schema mappings are updated for reference states`() {
|
||||
// 1. Create a state to be used as a reference state. Don't share it.
|
||||
val newRefTx = nodes[0].services.startFlow(CreateRefState()).resultFuture.getOrThrow()
|
||||
val newRefState = newRefTx.tx.outRefsOfType<RefState.State>().single()
|
||||
// 2. Use the "newRefState" a transaction involving another party (nodes[1]) which creates a new state. They should store the new state and the reference state.
|
||||
val newTx = nodes[0].services.startFlow(UseRefState(nodes[1].info.legalIdentities.first(), newRefState.state.data.linearId)).resultFuture.getOrThrow()
|
||||
// Wait until node 1 stores the new tx.
|
||||
nodes[1].services.validatedTransactions.updates.filter { it.id == newTx.id }.toFuture().getOrThrow()
|
||||
// Check that nodes[1] has finished recording the transaction (and updating the vault.. hopefully!).
|
||||
val allRefStates = nodes[1].services.vaultService.queryBy<LinearState>()
|
||||
// nodes[1] should have two states. The newly created output and the reference state created by nodes[0].
|
||||
assertEquals(2, allRefStates.states.size)
|
||||
}
|
||||
|
||||
// A dummy reference state contract.
|
||||
@ -126,6 +138,20 @@ class ReferencedStatesFlowTests {
|
||||
class Update : CommandData
|
||||
}
|
||||
|
||||
class RegularState : Contract {
|
||||
companion object {
|
||||
val CONTRACT_ID: String = RegularState::class.java.name
|
||||
}
|
||||
|
||||
override fun verify(tx: LedgerTransaction) = Unit
|
||||
|
||||
data class State(val owner: Party, override val linearId: UniqueIdentifier = UniqueIdentifier()) : LinearState {
|
||||
override val participants: List<AbstractParty> get() = listOf(owner)
|
||||
}
|
||||
|
||||
class Create : CommandData
|
||||
}
|
||||
|
||||
// A flow to create a reference state.
|
||||
class CreateRefState : FlowLogic<SignedTransaction>() {
|
||||
@Suspendable
|
||||
@ -193,7 +219,7 @@ class ReferencedStatesFlowTests {
|
||||
|
||||
val stx = serviceHub.signInitialTransaction(TransactionBuilder(notary = notary).apply {
|
||||
addReferenceState(referenceState.referenced())
|
||||
addOutputState(RefState.State(participant), RefState.CONTRACT_ID)
|
||||
addOutputState(RegularState.State(participant), RefState.CONTRACT_ID)
|
||||
addCommand(RefState.Create(), listOf(ourIdentity.owningKey))
|
||||
})
|
||||
return if (participant != ourIdentity) {
|
||||
|
@ -81,12 +81,12 @@ class NodeVaultService(
|
||||
* Maintain a list of contract state interfaces to concrete types stored in the vault
|
||||
* for usage in generic queries of type queryBy<LinearState> or queryBy<FungibleState<*>>
|
||||
*/
|
||||
private val contractStateTypeMappings = mutableMapOf<String, MutableSet<String>>()
|
||||
private val contractStateTypeMappings = mutableMapOf<String, MutableSet<String>>().toSynchronised()
|
||||
|
||||
override fun start() {
|
||||
bootstrapContractStateTypes()
|
||||
rawUpdates.subscribe { update ->
|
||||
update.produced.forEach {
|
||||
(update.produced + update.references).forEach {
|
||||
val concreteType = it.state.data.javaClass
|
||||
log.trace { "State update of type: $concreteType" }
|
||||
val seen = contractStateTypeMappings.any { it.value.contains(concreteType.name) }
|
||||
|
Loading…
Reference in New Issue
Block a user