core: Remove binding of State type in test dsl

This commit is contained in:
Andras Slemmer 2016-07-04 17:13:55 +01:00
parent 4a89be8785
commit 9b36df607e
4 changed files with 163 additions and 116 deletions

View File

@ -1,18 +1,21 @@
package com.r3corda.core.testing
import com.r3corda.core.contracts.Attachment
import com.r3corda.core.contracts.ContractState
import com.r3corda.core.contracts.*
import com.r3corda.core.crypto.SecureHash
import com.r3corda.core.node.services.IdentityService
import com.r3corda.core.node.services.StorageService
import com.r3corda.core.node.services.testing.MockStorageService
interface LedgerDslInterpreter<State: ContractState, out TransactionInterpreter: TransactionDslInterpreter<State>> {
fun transaction(dsl: TransactionDsl<State, TransactionInterpreter>.() -> Unit): Unit
fun nonVerifiedTransaction(dsl: TransactionDsl<State, TransactionInterpreter>.() -> Unit): Unit
fun tweak(dsl: LedgerDsl<State, TransactionInterpreter, LedgerDslInterpreter<State, TransactionInterpreter>>.() -> Unit)
interface OutputStateLookup {
fun <State: ContractState> retrieveOutputStateAndRef(clazz: Class<State>, label: String): StateAndRef<State>
}
interface LedgerDslInterpreter<out TransactionInterpreter: TransactionDslInterpreter> :
OutputStateLookup {
fun transaction(transactionLabel: String?, dsl: TransactionDsl<TransactionInterpreter>.() -> Unit): WireTransaction
fun nonVerifiedTransaction(transactionLabel: String?, dsl: TransactionDsl<TransactionInterpreter>.() -> Unit): WireTransaction
fun tweak(dsl: LedgerDsl<TransactionInterpreter, LedgerDslInterpreter<TransactionInterpreter>>.() -> Unit)
fun attachment(attachment: Attachment): SecureHash
fun _verifies(identityService: IdentityService, storageService: StorageService)
fun verifies()
}
/**
@ -21,15 +24,18 @@ interface LedgerDslInterpreter<State: ContractState, out TransactionInterpreter:
* covariance of the TransactionInterpreter parameter
*/
class LedgerDsl<
State: ContractState,
out TransactionInterpreter: TransactionDslInterpreter<State>,
out LedgerInterpreter: LedgerDslInterpreter<State, TransactionInterpreter>
out TransactionInterpreter: TransactionDslInterpreter,
out LedgerInterpreter: LedgerDslInterpreter<TransactionInterpreter>
> (val interpreter: LedgerInterpreter)
: LedgerDslInterpreter<State, TransactionDslInterpreter<State>> by interpreter {
: LedgerDslInterpreter<TransactionDslInterpreter> by interpreter {
@JvmOverloads
fun verifies(
identityService: IdentityService = MOCK_IDENTITY_SERVICE,
storageService: StorageService = MockStorageService()
) = _verifies(identityService, storageService)
fun transaction(dsl: TransactionDsl<TransactionDslInterpreter>.() -> Unit) = transaction(null, dsl)
fun nonVerifiedTransaction(dsl: TransactionDsl<TransactionDslInterpreter>.() -> Unit) =
nonVerifiedTransaction(null, dsl)
inline fun <reified State: ContractState> String.outputStateAndRef(): StateAndRef<State> =
retrieveOutputStateAndRef(State::class.java, this)
inline fun <reified State: ContractState> String.output(): TransactionState<State> =
outputStateAndRef<State>().state
fun String.outputRef(): StateRef = outputStateAndRef<ContractState>().ref
}

View File

@ -1,23 +1,34 @@
package com.r3corda.core.testing
import com.r3corda.core.contracts.*
import com.r3corda.core.crypto.DigitalSignature
import com.r3corda.core.crypto.Party
import com.r3corda.core.crypto.SecureHash
import com.r3corda.core.crypto.signWithECDSA
import com.r3corda.core.node.services.IdentityService
import com.r3corda.core.node.services.StorageService
import com.r3corda.core.node.services.testing.MockStorageService
import com.r3corda.core.serialization.serialize
import java.security.KeyPair
import java.security.PublicKey
import java.util.*
inline fun <reified State: ContractState> ledger(
dsl: LedgerDsl<State, TestTransactionDslInterpreter<State>, TestLedgerDslInterpreter<State>>.() -> Unit) =
dsl(LedgerDsl(TestLedgerDslInterpreter.create()))
inline fun ledger(
identityService: IdentityService = MOCK_IDENTITY_SERVICE,
storageService: StorageService = MockStorageService(),
dsl: LedgerDsl<TestTransactionDslInterpreter, TestLedgerDslInterpreter>.() -> Unit
): LedgerDsl<TestTransactionDslInterpreter, TestLedgerDslInterpreter> {
val ledgerDsl = LedgerDsl(TestLedgerDslInterpreter(identityService, storageService))
dsl(ledgerDsl)
return ledgerDsl
}
@Deprecated(
message = "ledger doesn't nest, use tweak",
replaceWith = ReplaceWith("tweak"),
level = DeprecationLevel.ERROR)
fun <State: ContractState> TransactionDslInterpreter<State>.ledger(
dsl: LedgerDsl<State, TestTransactionDslInterpreter<State>, TestLedgerDslInterpreter<State>>.() -> Unit) {
fun TransactionDslInterpreter.ledger(
dsl: LedgerDsl<TestTransactionDslInterpreter, TestLedgerDslInterpreter>.() -> Unit) {
this.toString()
dsl.toString()
}
@ -26,8 +37,8 @@ fun <State: ContractState> TransactionDslInterpreter<State>.ledger(
message = "ledger doesn't nest, use tweak",
replaceWith = ReplaceWith("tweak"),
level = DeprecationLevel.ERROR)
fun <State: ContractState> LedgerDslInterpreter<State, TransactionDslInterpreter<State>>.ledger(
dsl: LedgerDsl<State, TestTransactionDslInterpreter<State>, TestLedgerDslInterpreter<State>>.() -> Unit) {
fun LedgerDslInterpreter<TransactionDslInterpreter>.ledger(
dsl: LedgerDsl<TestTransactionDslInterpreter, TestLedgerDslInterpreter>.() -> Unit) {
this.toString()
dsl.toString()
}
@ -36,17 +47,16 @@ fun <State: ContractState> LedgerDslInterpreter<State, TransactionDslInterpreter
* This interpreter builds a transaction, and [TransactionDsl.verifies] that the resolved transaction is correct. Note
* that transactions corresponding to input states are not verified. Use [LedgerDsl.verifies] for that.
*/
data class TestTransactionDslInterpreter<State: ContractState>(
private val ledgerInterpreter: TestLedgerDslInterpreter<State>,
data class TestTransactionDslInterpreter(
private val ledgerInterpreter: TestLedgerDslInterpreter,
private val inputStateRefs: ArrayList<StateRef> = arrayListOf(),
internal val outputStates: ArrayList<LabeledOutput> = arrayListOf(),
private val attachments: ArrayList<SecureHash> = arrayListOf(),
private val commands: ArrayList<Command> = arrayListOf(),
private val signers: LinkedHashSet<PublicKey> = LinkedHashSet(),
private val transactionType: TransactionType = TransactionType.General()
) : TransactionDslInterpreter<State> {
private fun copy(): TestTransactionDslInterpreter<State> =
) : TransactionDslInterpreter, OutputStateLookup {
private fun copy(): TestTransactionDslInterpreter =
TestTransactionDslInterpreter(
ledgerInterpreter = ledgerInterpreter.copy(),
inputStateRefs = ArrayList(inputStateRefs),
@ -68,18 +78,18 @@ data class TestTransactionDslInterpreter<State: ContractState>(
)
override fun input(stateLabel: String) {
val notary = stateLabel.output.notary.owningKey
signers.add(notary)
inputStateRefs.add(stateLabel.outputRef)
val stateAndRef = retrieveOutputStateAndRef(ContractState::class.java, stateLabel)
signers.add(stateAndRef.state.notary.owningKey)
inputStateRefs.add(stateAndRef.ref)
}
override fun input(stateRef: StateRef) {
val notary = ledgerInterpreter.resolveStateRef(stateRef).notary
val notary = ledgerInterpreter.resolveStateRef<ContractState>(stateRef).notary
signers.add(notary.owningKey)
inputStateRefs.add(stateRef)
}
override fun output(label: String?, notary: Party, contractState: State) {
override fun output(label: String?, notary: Party, contractState: ContractState) {
outputStates.add(LabeledOutput(label, TransactionState(contractState, notary)))
}
@ -92,14 +102,14 @@ data class TestTransactionDslInterpreter<State: ContractState>(
commands.add(Command(commandData, signers))
}
override fun _verifies(identityService: IdentityService) {
val resolvedTransaction = ledgerInterpreter.resolveWireTransaction(toWireTransaction(), identityService)
override fun verifies() {
val resolvedTransaction = ledgerInterpreter.resolveWireTransaction(toWireTransaction())
resolvedTransaction.verify()
}
override fun failsWith(expectedMessage: String?, identityService: IdentityService) {
override fun failsWith(expectedMessage: String?) {
val exceptionThrown = try {
_verifies(identityService)
verifies()
false
} catch (exception: Exception) {
if (expectedMessage != null) {
@ -110,7 +120,7 @@ data class TestTransactionDslInterpreter<State: ContractState>(
)
} else if (!exceptionMessage.toLowerCase().contains(expectedMessage.toLowerCase())) {
throw AssertionError(
"Expected exception containing '$expectedMessage' but raised exception was '$exceptionMessage'"
"Expected exception containing '$expectedMessage' but raised exception was '$exception'"
)
}
}
@ -122,58 +132,60 @@ data class TestTransactionDslInterpreter<State: ContractState>(
}
}
override fun tweak(dsl: TransactionDsl<State, TransactionDslInterpreter<State>>.() -> Unit) =
override fun tweak(dsl: TransactionDsl<TransactionDslInterpreter>.() -> Unit) =
dsl(TransactionDsl(copy()))
override fun retrieveOutputStateAndRef(label: String): StateAndRef<State>? =
ledgerInterpreter.labelToOutputStateAndRefs[label]
override fun <State: ContractState> retrieveOutputStateAndRef(clazz: Class<State>, label: String) = ledgerInterpreter.retrieveOutputStateAndRef(clazz, label)
}
class AttachmentResolutionException(val attachmentId: SecureHash) :
class AttachmentResolutionException(attachmentId: SecureHash) :
Exception("Attachment with id $attachmentId not found")
data class TestLedgerDslInterpreter<State: ContractState> private constructor (
internal val stateClazz: Class<State>,
internal val labelToOutputStateAndRefs: HashMap<String, StateAndRef<State>> = HashMap(),
data class TestLedgerDslInterpreter private constructor (
private val identityService: IdentityService,
private val storageService: StorageService,
internal val labelToOutputStateAndRefs: HashMap<String, StateAndRef<ContractState>> = HashMap(),
private val transactionWithLocations: HashMap<SecureHash, WireTransactionWithLocation> = HashMap(),
private val nonVerifiedTransactionWithLocations: HashMap<SecureHash, WireTransactionWithLocation> = HashMap(),
private val attachments: HashMap<SecureHash, Attachment> = HashMap()
) : LedgerDslInterpreter<State, TestTransactionDslInterpreter<State>> {
private val nonVerifiedTransactionWithLocations: HashMap<SecureHash, WireTransactionWithLocation> = HashMap()
) : LedgerDslInterpreter<TestTransactionDslInterpreter> {
// We specify [labelToOutputStateAndRefs] just so that Kotlin picks the primary constructor instead of cycling
constructor(stateClazz: Class<State>) : this(stateClazz, labelToOutputStateAndRefs = HashMap())
constructor(identityService: IdentityService, storageService: StorageService) : this(
identityService, storageService, labelToOutputStateAndRefs = HashMap()
)
companion object {
/**
* Convenience factory to avoid having to pass in the Class
*/
inline fun <reified State: ContractState> create() = TestLedgerDslInterpreter(State::class.java)
private fun getCallerLocation(offset: Int): String {
val stackTraceElement = Thread.currentThread().stackTrace[3 + offset]
return stackTraceElement.toString()
}
}
private data class WireTransactionWithLocation(val transaction: WireTransaction, val location: String)
private class VerifiesFailed(transactionLocation: String, cause: Throwable) :
internal data class WireTransactionWithLocation(
val label: String?,
val transaction: WireTransaction,
val location: String
)
class VerifiesFailed(transactionLocation: String, cause: Throwable) :
Exception("Transaction defined at ($transactionLocation) didn't verify: $cause", cause)
class TypeMismatch(requested: Class<*>, actual: Class<*>) :
Exception("Actual type $actual is not a subtype of requested type $requested")
internal fun copy(): TestLedgerDslInterpreter<State> =
internal fun copy(): TestLedgerDslInterpreter =
TestLedgerDslInterpreter(
stateClazz = stateClazz,
identityService,
storageService,
labelToOutputStateAndRefs = HashMap(labelToOutputStateAndRefs),
transactionWithLocations = HashMap(transactionWithLocations),
nonVerifiedTransactionWithLocations = HashMap(nonVerifiedTransactionWithLocations),
attachments = HashMap(attachments)
nonVerifiedTransactionWithLocations = HashMap(nonVerifiedTransactionWithLocations)
)
fun resolveWireTransaction(wireTransaction: WireTransaction, identityService: IdentityService): TransactionForVerification {
internal fun resolveWireTransaction(wireTransaction: WireTransaction): TransactionForVerification {
return wireTransaction.run {
val authenticatedCommands = commands.map {
AuthenticatedObject(it.signers, it.signers.mapNotNull { identityService.partyFromKey(it) }, it.value)
}
val resolvedInputStates = inputs.map { resolveStateRef(it) }
val resolvedInputStates = inputs.map { resolveStateRef<ContractState>(it) }
val resolvedAttachments = attachments.map { resolveAttachment(it) }
TransactionForVerification(
inputs = resolvedInputStates,
@ -188,30 +200,30 @@ data class TestLedgerDslInterpreter<State: ContractState> private constructor (
}
}
fun resolveStateRef(stateRef: StateRef): TransactionState<State> {
internal inline fun <reified State: ContractState> resolveStateRef(stateRef: StateRef): TransactionState<State> {
val transactionWithLocation =
transactionWithLocations[stateRef.txhash] ?:
nonVerifiedTransactionWithLocations[stateRef.txhash] ?:
throw TransactionResolutionException(stateRef.txhash)
val output = transactionWithLocation.transaction.outputs[stateRef.index]
return if (stateClazz.isInstance(output.data)) @Suppress("UNCHECKED_CAST") {
return if (State::class.java.isAssignableFrom(output.data.javaClass)) @Suppress("UNCHECKED_CAST") {
output as TransactionState<State>
} else {
throw IllegalArgumentException("Referenced state is of another type than requested")
throw TypeMismatch(requested = State::class.java, actual = output.data.javaClass)
}
}
fun resolveAttachment(attachmentId: SecureHash): Attachment =
attachments[attachmentId] ?: throw AttachmentResolutionException(attachmentId)
internal fun resolveAttachment(attachmentId: SecureHash): Attachment =
storageService.attachments.openAttachment(attachmentId) ?: throw AttachmentResolutionException(attachmentId)
private fun interpretTransactionDsl(dsl: TransactionDsl<State, TestTransactionDslInterpreter<State>>.() -> Unit):
TestTransactionDslInterpreter<State> {
private fun interpretTransactionDsl(dsl: TransactionDsl<TestTransactionDslInterpreter>.() -> Unit):
TestTransactionDslInterpreter {
val transactionInterpreter = TestTransactionDslInterpreter(this)
dsl(TransactionDsl(transactionInterpreter))
return transactionInterpreter
}
private fun toTransactionGroup(identityService: IdentityService, storageService: StorageService): TransactionGroup {
fun toTransactionGroup(): TransactionGroup {
val ledgerTransactions = transactionWithLocations.map {
it.value.transaction.toLedgerTransaction(identityService, storageService.attachments)
}
@ -221,10 +233,23 @@ data class TestLedgerDslInterpreter<State: ContractState> private constructor (
return TransactionGroup(ledgerTransactions.toSet(), nonVerifiedLedgerTransactions.toSet())
}
fun transactionName(transactionHash: SecureHash): String? {
val transactionWithLocation = transactionWithLocations[transactionHash]
return if (transactionWithLocation != null) {
transactionWithLocation.label ?: "TX[${transactionWithLocation.location}]"
} else {
null
}
}
fun outputToLabel(state: ContractState): String? =
labelToOutputStateAndRefs.filter { it.value.state.data == state }.keys.firstOrNull()
private fun recordTransactionWithTransactionMap(
dsl: TransactionDsl<State, TestTransactionDslInterpreter<State>>.() -> Unit,
transactionLabel: String?,
dsl: TransactionDsl<TestTransactionDslInterpreter>.() -> Unit,
transactionMap: HashMap<SecureHash, WireTransactionWithLocation> = HashMap()
) {
): WireTransaction {
val transactionLocation = getCallerLocation(3)
val transactionInterpreter = interpretTransactionDsl(dsl)
// Create the WireTransaction
@ -237,38 +262,67 @@ data class TestLedgerDslInterpreter<State: ContractState> private constructor (
}
transactionMap[wireTransaction.serialized.hash] =
WireTransactionWithLocation(wireTransaction, transactionLocation)
WireTransactionWithLocation(transactionLabel, wireTransaction, transactionLocation)
return wireTransaction
}
override fun transaction(dsl: TransactionDsl<State, TestTransactionDslInterpreter<State>>.() -> Unit) =
recordTransactionWithTransactionMap(dsl, transactionWithLocations)
override fun transaction(transactionLabel: String?, dsl: TransactionDsl<TestTransactionDslInterpreter>.() -> Unit) =
recordTransactionWithTransactionMap(transactionLabel, dsl, transactionWithLocations)
override fun nonVerifiedTransaction(dsl: TransactionDsl<State, TestTransactionDslInterpreter<State>>.() -> Unit) =
recordTransactionWithTransactionMap(dsl, nonVerifiedTransactionWithLocations)
override fun nonVerifiedTransaction(transactionLabel: String?, dsl: TransactionDsl<TestTransactionDslInterpreter>.() -> Unit) =
recordTransactionWithTransactionMap(transactionLabel, dsl, nonVerifiedTransactionWithLocations)
override fun tweak(
dsl: LedgerDsl<State, TestTransactionDslInterpreter<State>,
LedgerDslInterpreter<State, TestTransactionDslInterpreter<State>>>.() -> Unit) =
dsl: LedgerDsl<TestTransactionDslInterpreter,
LedgerDslInterpreter<TestTransactionDslInterpreter>>.() -> Unit) =
dsl(LedgerDsl(copy()))
override fun attachment(attachment: Attachment): SecureHash {
attachments[attachment.id] = attachment
storageService.attachments.importAttachment(attachment.open())
return attachment.id
}
override fun _verifies(identityService: IdentityService, storageService: StorageService) {
val transactionGroup = toTransactionGroup(identityService, storageService)
override fun verifies() {
val transactionGroup = toTransactionGroup()
try {
transactionGroup.verify()
} catch (exception: TransactionVerificationException) {
throw VerifiesFailed(transactionWithLocations[exception.tx.origHash]?.location ?: "<unknown>", exception)
}
}
override fun <State: ContractState> retrieveOutputStateAndRef(clazz: Class<State>, label: String): StateAndRef<State> {
val stateAndRef = labelToOutputStateAndRefs[label]
if (stateAndRef == null) {
throw IllegalArgumentException("State with label '$label' was not found")
} else if (!clazz.isAssignableFrom(stateAndRef.state.data.javaClass)) {
throw TypeMismatch(requested = clazz, actual = stateAndRef.state.data.javaClass)
} else {
@Suppress("UNCHECKED_CAST")
return stateAndRef as StateAndRef<State>
}
}
}
fun signAll(transactionsToSign: List<WireTransaction>, vararg extraKeys: KeyPair): List<SignedTransaction> {
return transactionsToSign.map { wtx ->
val allPubKeys = wtx.signers.toMutableSet()
val bits = wtx.serialize()
require(bits == wtx.serialized)
val signatures = ArrayList<DigitalSignature.WithKey>()
for (key in ALL_TEST_KEYS + extraKeys) {
if (allPubKeys.contains(key.public)) {
signatures += key.signWithECDSA(bits)
allPubKeys -= key.public
}
}
SignedTransaction(bits, signatures)
}
}
fun main(args: Array<String>) {
ledger<ContractState> {
ledger {
nonVerifiedTransaction {
output("hello") { DummyLinearState() }
}

View File

@ -6,7 +6,6 @@ import com.google.common.base.Throwables
import com.google.common.net.HostAndPort
import com.r3corda.core.contracts.*
import com.r3corda.core.crypto.*
import com.r3corda.core.node.services.IdentityService
import com.r3corda.core.node.services.testing.MockIdentityService
import com.r3corda.core.node.services.testing.MockStorageService
import com.r3corda.core.seconds
@ -219,22 +218,21 @@ open class TransactionForTest : AbstractTransactionForTest() {
}
fun input(s: ContractState) = input { s }
protected fun runCommandsAndVerify(time: Instant) {
protected fun runCommandsAndVerify() {
val cmds = commandsToAuthenticatedObjects()
val tx = TransactionForVerification(inStates, outStates.map { it.state }, emptyList(), cmds, SecureHash.Companion.randomSHA256(), signers.toList(), type)
tx.verify()
}
@JvmOverloads
fun accepts(time: Instant = TEST_TX_TIME): LastLineShouldTestForAcceptOrFailure {
runCommandsAndVerify(time)
fun accepts(): LastLineShouldTestForAcceptOrFailure {
runCommandsAndVerify()
return LastLineShouldTestForAcceptOrFailure.Token
}
@JvmOverloads
fun rejects(withMessage: String? = null, time: Instant = TEST_TX_TIME): LastLineShouldTestForAcceptOrFailure {
fun rejects(withMessage: String? = null): LastLineShouldTestForAcceptOrFailure {
val r = try {
runCommandsAndVerify(time)
runCommandsAndVerify()
false
} catch (e: Exception) {
val m = e.message
@ -300,7 +298,7 @@ open class TransactionForTest : AbstractTransactionForTest() {
}
}
class TransactionGroupDSL<T : ContractState>(private val stateType: Class<T>) {
class TransactionGroupDSL<out T : ContractState>(private val stateType: Class<T>) {
open inner class WireTransactionDSL : AbstractTransactionForTest() {
private val inStates = ArrayList<StateRef>()

View File

@ -15,50 +15,39 @@ import java.time.Instant
* TODO: Move the [State] binding to the primitives' level to allow different State types, use reflection to check types
* dynamically, come up with a substitute for primitives relying on early bind
*/
interface TransactionDslInterpreter<State: ContractState> {
interface TransactionDslInterpreter : OutputStateLookup {
fun input(stateLabel: String)
fun input(stateRef: StateRef)
fun output(label: String?, notary: Party, contractState: State)
fun output(label: String?, notary: Party, contractState: ContractState)
fun attachment(attachmentId: SecureHash)
fun _command(signers: List<PublicKey>, commandData: CommandData)
fun _verifies(identityService: IdentityService)
fun failsWith(expectedMessage: String?, identityService: IdentityService)
fun tweak(dsl: TransactionDsl<State, TransactionDslInterpreter<State>>.() -> Unit)
fun retrieveOutputStateAndRef(label: String): StateAndRef<State>?
val String.outputStateAndRef: StateAndRef<State>
get() = retrieveOutputStateAndRef(this) ?: throw IllegalArgumentException("State with label '$this' was not found")
val String.output: TransactionState<State>
get() = outputStateAndRef.state
val String.outputRef: StateRef
get() = outputStateAndRef.ref
fun verifies()
fun failsWith(expectedMessage: String?)
fun tweak(dsl: TransactionDsl<TransactionDslInterpreter>.() -> Unit)
}
class TransactionDsl<
State: ContractState,
out TransactionInterpreter: TransactionDslInterpreter<State>
out TransactionInterpreter: TransactionDslInterpreter
> (val interpreter: TransactionInterpreter)
: TransactionDslInterpreter<State> by interpreter {
: TransactionDslInterpreter by interpreter {
// Convenience functions
fun output(label: String? = null, notary: Party = DUMMY_NOTARY, contractStateClosure: () -> State) =
fun output(label: String? = null, notary: Party = DUMMY_NOTARY, contractStateClosure: () -> ContractState) =
output(label, notary, contractStateClosure())
@JvmOverloads
fun output(label: String? = null, contractState: State) = output(label, DUMMY_NOTARY, contractState)
fun output(label: String? = null, contractState: ContractState) = output(label, DUMMY_NOTARY, contractState)
fun command(vararg signers: PublicKey, commandDataClosure: () -> CommandData) =
_command(listOf(*signers), commandDataClosure())
fun command(signer: PublicKey, commandData: CommandData) = _command(listOf(signer), commandData)
fun verifies(identityService: IdentityService = MOCK_IDENTITY_SERVICE) = _verifies(identityService)
@JvmOverloads
fun timestamp(time: Instant, notary: PublicKey = DUMMY_NOTARY.owningKey) =
timestamp(TimestampCommand(time, 30.seconds), notary)
@JvmOverloads
fun timestamp(data: TimestampCommand, notary: PublicKey = DUMMY_NOTARY.owningKey) = command(notary, data)
fun fails(identityService: IdentityService = MOCK_IDENTITY_SERVICE) = failsWith(null, identityService)
infix fun `fails with`(msg: String) = failsWith(msg, MOCK_IDENTITY_SERVICE)
fun fails() = failsWith(null)
infix fun `fails with`(msg: String) = failsWith(msg)
}