2015-11-25 13:29:51 +00:00
|
|
|
.. highlight:: kotlin
|
|
|
|
.. raw:: html
|
|
|
|
|
|
|
|
<script type="text/javascript" src="_static/jquery.js"></script>
|
|
|
|
<script type="text/javascript" src="_static/codesets.js"></script>
|
|
|
|
|
2015-12-22 15:15:38 +00:00
|
|
|
Writing a contract
|
|
|
|
==================
|
2015-11-25 13:29:51 +00:00
|
|
|
|
2016-11-17 17:58:23 +00:00
|
|
|
This tutorial will take you through writing a contract, using a simple commercial paper contract as an example.
|
|
|
|
Smart contracts in Corda have three key elements:
|
2015-11-25 13:29:51 +00:00
|
|
|
|
2016-11-17 17:58:23 +00:00
|
|
|
* Executable code (validation logic)
|
|
|
|
* State objects
|
|
|
|
* Commands
|
|
|
|
|
|
|
|
The core of a smart contract is the executable code which validates changes to state objects in transactions. State
|
|
|
|
objects are the data held on the ledger, which represent the current state of an instance of a contract, and are used as
|
|
|
|
inputs and outputs of transactions. Commands are additional data included in transactions to describe what is going on,
|
|
|
|
used to instruct the executable code on how to verify the transaction. For example an ``Issue`` command may indicate
|
|
|
|
that the validation logic should expect to see an output which does not exist as an input, issued by the same entity
|
|
|
|
that signed the command.
|
|
|
|
|
|
|
|
The first thing to think about with a new contract is the lifecycle of contract states, how are they issued, what happens
|
|
|
|
to them after they are issued, and how are they destroyed (if applicable). For the commercial paper contract, states are
|
|
|
|
issued by a legal entity which wishes to create a contract to pay money in the future (the maturity date), in return for
|
|
|
|
a lesser payment now. They are then transferred (moved) to another owner as part of a transaction where the issuer
|
|
|
|
receives funds in payment, and later (after the maturity date) are destroyed (redeemed) by paying the owner the face
|
|
|
|
value of the commercial paper.
|
|
|
|
|
|
|
|
This lifecycle for commercial paper is illustrated in the diagram below:
|
|
|
|
|
2017-06-07 15:14:01 +00:00
|
|
|
.. image:: resources/contract-cp.png
|
2016-08-24 13:38:43 +00:00
|
|
|
|
2015-11-25 13:29:51 +00:00
|
|
|
Starting the commercial paper class
|
|
|
|
-----------------------------------
|
|
|
|
|
2016-11-17 17:58:23 +00:00
|
|
|
A smart contract is a class that implements the ``Contract`` interface. This can be either implemented directly, as done
|
|
|
|
here, or by subclassing an abstract contract such as ``OnLedgerAsset``. The heart of any contract in Corda is the
|
2017-10-01 22:33:15 +00:00
|
|
|
``verify`` function, which determines whether a given transaction is valid. This example shows how to write a
|
|
|
|
``verify`` function from scratch.
|
2016-11-17 17:58:23 +00:00
|
|
|
|
2017-10-01 22:33:15 +00:00
|
|
|
The code in this tutorial is available in both Kotlin and Java. You can quickly switch between them to get a feeling
|
|
|
|
for Kotlin's syntax.
|
2015-11-25 13:29:51 +00:00
|
|
|
|
|
|
|
.. container:: codeset
|
|
|
|
|
|
|
|
.. sourcecode:: kotlin
|
|
|
|
|
|
|
|
class CommercialPaper : Contract {
|
2017-07-19 12:06:43 +00:00
|
|
|
override fun verify(tx: LedgerTransaction) {
|
2015-11-25 13:29:51 +00:00
|
|
|
TODO()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
.. sourcecode:: java
|
|
|
|
|
2016-07-12 10:54:50 +00:00
|
|
|
public class CommercialPaper implements Contract {
|
2015-11-25 13:29:51 +00:00
|
|
|
@Override
|
2017-07-19 12:06:43 +00:00
|
|
|
public void verify(LedgerTransaction tx) {
|
2015-11-25 13:29:51 +00:00
|
|
|
throw new UnsupportedOperationException();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-01 22:33:15 +00:00
|
|
|
Every contract must have at least a ``verify`` method. The verify method returns nothing. This is intentional: the
|
|
|
|
function either completes correctly, or throws an exception, in which case the transaction is rejected.
|
2015-11-25 13:29:51 +00:00
|
|
|
|
|
|
|
So far, so simple. Now we need to define the commercial paper *state*, which represents the fact of ownership of a
|
|
|
|
piece of issued paper.
|
|
|
|
|
|
|
|
States
|
|
|
|
------
|
|
|
|
|
2016-11-17 17:58:23 +00:00
|
|
|
A state is a class that stores data that is checked by the contract. A commercial paper state is structured as below:
|
|
|
|
|
2017-06-07 15:14:01 +00:00
|
|
|
.. image:: resources/contract-cp-state.png
|
2016-11-17 17:58:23 +00:00
|
|
|
|
2015-11-25 13:29:51 +00:00
|
|
|
.. container:: codeset
|
|
|
|
|
2017-10-01 22:33:15 +00:00
|
|
|
.. literalinclude:: ../../docs/source/example-code/src/main/kotlin/net/corda/docs/tutorial/contract/TutorialContract.kt
|
|
|
|
:language: kotlin
|
|
|
|
:start-after: DOCSTART 1
|
|
|
|
:end-before: DOCEND 1
|
2015-11-25 13:29:51 +00:00
|
|
|
|
2017-10-01 22:33:15 +00:00
|
|
|
.. literalinclude:: ../../docs/source/example-code/src/main/java/net/corda/docs/java/tutorial/contract/State.java
|
|
|
|
:language: java
|
|
|
|
:start-after: DOCSTART 1
|
|
|
|
:end-before: DOCEND 1
|
2015-11-25 13:29:51 +00:00
|
|
|
|
|
|
|
|
2016-08-24 13:38:43 +00:00
|
|
|
We define a class that implements the ``ContractState`` interface.
|
|
|
|
|
2015-11-25 13:29:51 +00:00
|
|
|
We have four fields in our state:
|
|
|
|
|
2016-08-24 13:38:43 +00:00
|
|
|
* ``issuance``, a reference to a specific piece of commercial paper issued by some party.
|
|
|
|
* ``owner``, the public key of the current owner. This is the same concept as seen in Bitcoin: the public key has no
|
2015-11-25 17:49:58 +00:00
|
|
|
attached identity and is expected to be one-time-use for privacy reasons. However, unlike in Bitcoin, we model
|
2016-08-24 13:38:43 +00:00
|
|
|
ownership at the level of individual states rather than as a platform-level concept as we envisage many
|
2015-11-25 17:49:58 +00:00
|
|
|
(possibly most) contracts on the platform will not represent "owner/issuer" relationships, but "party/party"
|
|
|
|
relationships such as a derivative contract.
|
2016-08-24 13:38:43 +00:00
|
|
|
* ``faceValue``, an ``Amount<Issued<Currency>>``, which wraps an integer number of pennies and a currency that is
|
|
|
|
specific to some issuer (e.g. a regular bank, a central bank, etc). You can read more about this very common
|
2017-06-05 12:37:23 +00:00
|
|
|
type in :doc:`api-core-types`.
|
2016-08-24 13:38:43 +00:00
|
|
|
* ``maturityDate``, an `Instant <https://docs.oracle.com/javase/8/docs/api/java/time/Instant.html>`_, which is a type
|
2015-11-25 13:29:51 +00:00
|
|
|
from the Java 8 standard time library. It defines a point on the timeline.
|
|
|
|
|
2015-11-25 18:04:35 +00:00
|
|
|
States are immutable, and thus the class is defined as immutable as well. The ``data`` modifier in the Kotlin version
|
2015-11-25 13:29:51 +00:00
|
|
|
causes the compiler to generate the equals/hashCode/toString methods automatically, along with a copy method that can
|
|
|
|
be used to create variants of the original object. Data classes are similar to case classes in Scala, if you are
|
2015-11-30 17:00:25 +00:00
|
|
|
familiar with that language. The ``withoutOwner`` method uses the auto-generated copy method to return a version of
|
2015-11-25 13:29:51 +00:00
|
|
|
the state with the owner public key blanked out: this will prove useful later.
|
|
|
|
|
2016-08-24 13:38:43 +00:00
|
|
|
The Java code compiles to almost identical bytecode as the Kotlin version, but as you can see, is much more verbose.
|
2015-11-25 13:29:51 +00:00
|
|
|
|
|
|
|
Commands
|
|
|
|
--------
|
|
|
|
|
2016-11-17 17:58:23 +00:00
|
|
|
The validation logic for a contract may vary depending on what stage of a state's lifecycle it is automating. So it can
|
|
|
|
be useful to pass additional data into the contract code that isn't represented by the states which exist permanently
|
|
|
|
in the ledger, in order to clarify intent of a transaction.
|
2015-11-25 13:29:51 +00:00
|
|
|
|
2016-08-24 13:38:43 +00:00
|
|
|
For this purpose we have commands. Often they don't need to contain any data at all, they just need to exist. A command
|
2015-11-25 13:29:51 +00:00
|
|
|
is a piece of data associated with some *signatures*. By the time the contract runs the signatures have already been
|
|
|
|
checked, so from the contract code's perspective, a command is simply a data structure with a list of attached
|
2016-08-24 13:38:43 +00:00
|
|
|
public keys. Each key had a signature proving that the corresponding private key was used to sign. Because of this
|
|
|
|
approach contracts never actually interact or work with digital signatures directly.
|
2015-11-25 13:29:51 +00:00
|
|
|
|
2015-11-25 18:04:35 +00:00
|
|
|
Let's define a few commands now:
|
2015-11-25 13:29:51 +00:00
|
|
|
|
|
|
|
.. container:: codeset
|
|
|
|
|
2017-10-01 22:33:15 +00:00
|
|
|
.. literalinclude:: ../../docs/source/example-code/src/main/kotlin/net/corda/docs/tutorial/contract/TutorialContract.kt
|
|
|
|
:language: kotlin
|
|
|
|
:start-after: DOCSTART 2
|
|
|
|
:end-before: DOCEND 2
|
|
|
|
:dedent: 4
|
2015-11-25 13:29:51 +00:00
|
|
|
|
2017-10-01 22:33:15 +00:00
|
|
|
.. literalinclude:: ../../docs/source/example-code/src/main/java/net/corda/docs/java/tutorial/contract/CommercialPaper.java
|
|
|
|
:language: java
|
|
|
|
:start-after: DOCSTART 2
|
|
|
|
:end-before: DOCEND 2
|
|
|
|
:dedent: 4
|
2015-11-25 13:29:51 +00:00
|
|
|
|
2016-08-24 13:38:43 +00:00
|
|
|
We define a simple grouping interface or static class, this gives us a type that all our commands have in common,
|
2016-08-26 11:05:26 +00:00
|
|
|
then we go ahead and create three commands: ``Move``, ``Redeem``, ``Issue``. ``TypeOnlyCommandData`` is a helpful utility
|
2016-08-24 13:38:43 +00:00
|
|
|
for the case when there's no data inside the command; only the existence matters. It defines equals and hashCode
|
|
|
|
such that any instances always compare equal and hash to the same value.
|
2015-11-25 13:29:51 +00:00
|
|
|
|
|
|
|
The verify function
|
|
|
|
-------------------
|
|
|
|
|
|
|
|
The heart of a smart contract is the code that verifies a set of state transitions (a *transaction*). The function is
|
|
|
|
simple: it's given a class representing the transaction, and if the function returns then the transaction is considered
|
|
|
|
acceptable. If it throws an exception, the transaction is rejected.
|
|
|
|
|
|
|
|
Each transaction can have multiple input and output states of different types. The set of contracts to run is decided
|
|
|
|
by taking the code references inside each state. Each contract is run only once. As an example, a contract that includes
|
|
|
|
2 cash states and 1 commercial paper state as input, and has as output 1 cash state and 1 commercial paper state, will
|
|
|
|
run two contracts one time each: Cash and CommercialPaper.
|
|
|
|
|
|
|
|
.. container:: codeset
|
|
|
|
|
2017-10-01 22:33:15 +00:00
|
|
|
.. literalinclude:: ../../docs/source/example-code/src/main/kotlin/net/corda/docs/tutorial/contract/TutorialContract.kt
|
|
|
|
:language: kotlin
|
|
|
|
:start-after: DOCSTART 3
|
|
|
|
:end-before: DOCEND 3
|
|
|
|
:dedent: 4
|
2016-08-24 13:38:43 +00:00
|
|
|
|
2017-10-01 22:33:15 +00:00
|
|
|
.. literalinclude:: ../../docs/source/example-code/src/main/java/net/corda/docs/java/tutorial/contract/CommercialPaper.java
|
|
|
|
:language: java
|
|
|
|
:start-after: DOCSTART 3
|
|
|
|
:end-before: DOCEND 3
|
|
|
|
:dedent: 4
|
2015-11-25 13:29:51 +00:00
|
|
|
|
2016-04-29 16:45:29 +00:00
|
|
|
We start by using the ``groupStates`` method, which takes a type and a function. State grouping is a way of ensuring
|
|
|
|
your contract can handle multiple unrelated states of the same type in the same transaction, which is needed for
|
2016-08-24 13:38:43 +00:00
|
|
|
splitting/merging of assets, atomic swaps and so on. More on this next.
|
|
|
|
|
|
|
|
The second line does what the code suggests: it searches for a command object that inherits from the
|
|
|
|
``CommercialPaper.Commands`` supertype, and either returns it, or throws an exception if there's zero or more than one
|
|
|
|
such command.
|
2015-11-25 13:29:51 +00:00
|
|
|
|
2016-04-29 16:45:29 +00:00
|
|
|
Using state groups
|
|
|
|
------------------
|
|
|
|
|
|
|
|
The simplest way to write a smart contract would be to say that each transaction can have a single input state and a
|
2017-01-06 11:05:37 +00:00
|
|
|
single output state of the kind covered by that contract. This would be easy for the developer, but would prevent many
|
2016-04-29 16:45:29 +00:00
|
|
|
important use cases.
|
2015-11-25 13:29:51 +00:00
|
|
|
|
2016-04-29 16:45:29 +00:00
|
|
|
The next easiest way to write a contract would be to iterate over each input state and expect it to have an output
|
|
|
|
state. Now you can build a single transaction that, for instance, moves two different cash states in different currencies
|
|
|
|
simultaneously. But it gets complicated when you want to issue or exit one state at the same time as moving another.
|
|
|
|
|
|
|
|
Things get harder still once you want to split and merge states. We say states are *fungible* if they are
|
|
|
|
treated identically to each other by the recipient, despite the fact that they aren't quite identical. Dollar bills are
|
|
|
|
fungible because even though one may be worn/a bit dirty and another may be crisp and new, they are still both worth
|
|
|
|
exactly $1. Likewise, ten $1 bills are almost exactly equivalent to one $10 bill. On the other hand, $10 and £10 are not
|
|
|
|
fungible: if you tried to pay for something that cost £20 with $10+£10 notes your trade would not be accepted.
|
2015-11-25 13:29:51 +00:00
|
|
|
|
2015-11-25 18:04:35 +00:00
|
|
|
To make all this easier the contract API provides a notion of groups. A group is a set of input states and output states
|
2016-04-29 16:45:29 +00:00
|
|
|
that should be checked for validity together.
|
2015-11-25 13:29:51 +00:00
|
|
|
|
|
|
|
Consider the following simplified currency trade transaction:
|
|
|
|
|
|
|
|
* **Input**: $12,000 owned by Alice (A)
|
|
|
|
* **Input**: $3,000 owned by Alice (A)
|
|
|
|
* **Input**: £10,000 owned by Bob (B)
|
|
|
|
* **Output**: £10,000 owned by Alice (B)
|
|
|
|
* **Output**: $15,000 owned by Bob (A)
|
|
|
|
|
|
|
|
In this transaction Alice and Bob are trading $15,000 for £10,000. Alice has her money in the form of two different
|
|
|
|
inputs e.g. because she received the dollars in two payments. The input and output amounts do balance correctly, but
|
|
|
|
the cash smart contract must consider the pounds and the dollars separately because they are not fungible: they cannot
|
|
|
|
be merged together. So we have two groups: A and B.
|
|
|
|
|
2017-07-19 12:06:43 +00:00
|
|
|
The ``LedgerTransaction.groupStates`` method handles this logic for us: firstly, it selects only states of the
|
2015-11-25 13:29:51 +00:00
|
|
|
given type (as the transaction may include other types of state, such as states representing bond ownership, or a
|
|
|
|
multi-sig state) and then it takes a function that maps a state to a grouping key. All states that share the same key are
|
|
|
|
grouped together. In the case of the cash example above, the grouping key would be the currency.
|
|
|
|
|
2016-08-24 13:38:43 +00:00
|
|
|
In this kind of contract we don't want CP to be fungible: merging and splitting is (in our example) not allowed.
|
2016-04-29 16:45:29 +00:00
|
|
|
So we just use a copy of the state minus the owner field as the grouping key.
|
|
|
|
|
|
|
|
Here are some code examples:
|
|
|
|
|
|
|
|
.. container:: codeset
|
|
|
|
|
|
|
|
.. sourcecode:: kotlin
|
|
|
|
|
|
|
|
// Type of groups is List<InOutGroup<State, Pair<PartyReference, Currency>>>
|
2017-10-01 22:33:15 +00:00
|
|
|
val groups = tx.groupStates { it: Cash.State -> it.amount.token }
|
|
|
|
for ((inputs, outputs, key) in groups) {
|
|
|
|
// Either inputs or outputs could be empty.
|
|
|
|
val (deposit, currency) = key
|
2016-04-29 16:45:29 +00:00
|
|
|
|
2017-10-01 22:33:15 +00:00
|
|
|
...
|
2016-04-29 16:45:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
.. sourcecode:: java
|
|
|
|
|
|
|
|
List<InOutGroup<State, Pair<PartyReference, Currency>>> groups = tx.groupStates(Cash.State.class, s -> Pair(s.deposit, s.amount.currency))
|
|
|
|
for (InOutGroup<State, Pair<PartyReference, Currency>> group : groups) {
|
|
|
|
List<State> inputs = group.getInputs();
|
|
|
|
List<State> outputs = group.getOutputs();
|
|
|
|
Pair<PartyReference, Currency> key = group.getKey();
|
|
|
|
|
|
|
|
...
|
|
|
|
}
|
|
|
|
|
|
|
|
The ``groupStates`` call uses the provided function to calculate a "grouping key". All states that have the same
|
|
|
|
grouping key are placed in the same group. A grouping key can be anything that implements equals/hashCode, but it's
|
|
|
|
always an aggregate of the fields that shouldn't change between input and output. In the above example we picked the
|
2016-05-23 16:03:31 +00:00
|
|
|
fields we wanted and packed them into a ``Pair``. It returns a list of ``InOutGroup``, which is just a holder for the
|
2016-04-29 16:45:29 +00:00
|
|
|
inputs, outputs and the key that was used to define the group. In the Kotlin version we unpack these using destructuring
|
|
|
|
to get convenient access to the inputs, the outputs, the deposit data and the currency. The Java version is more
|
|
|
|
verbose, but equivalent.
|
|
|
|
|
|
|
|
The rules can then be applied to the inputs and outputs as if it were a single transaction. A group may have zero
|
|
|
|
inputs or zero outputs: this can occur when issuing assets onto the ledger, or removing them.
|
|
|
|
|
|
|
|
In this example, we do it differently and use the state class itself as the aggregator. We just
|
|
|
|
blank out fields that are allowed to change, making the grouping key be "everything that isn't that":
|
|
|
|
|
|
|
|
.. container:: codeset
|
|
|
|
|
|
|
|
.. sourcecode:: kotlin
|
|
|
|
|
2017-06-08 10:53:45 +00:00
|
|
|
val groups = tx.groupStates(State::withoutOwner)
|
2016-04-29 16:45:29 +00:00
|
|
|
|
|
|
|
.. sourcecode:: java
|
|
|
|
|
|
|
|
List<InOutGroup<State, State>> groups = tx.groupStates(State.class, State::withoutOwner);
|
|
|
|
|
|
|
|
For large states with many fields that must remain constant and only one or two that are really mutable, it's often
|
|
|
|
easier to do things this way than to specifically name each field that must stay the same. The ``withoutOwner`` function
|
|
|
|
here simply returns a copy of the object but with the ``owner`` field set to ``NullPublicKey``, which is just a public key
|
|
|
|
of all zeros. It's invalid and useless, but that's OK, because all we're doing is preventing the field from mattering
|
|
|
|
in equals and hashCode.
|
2015-11-25 13:29:51 +00:00
|
|
|
|
|
|
|
|
|
|
|
Checking the requirements
|
|
|
|
-------------------------
|
|
|
|
After extracting the command and the groups, we then iterate over each group and verify it meets the required business
|
|
|
|
logic.
|
|
|
|
|
|
|
|
.. container:: codeset
|
|
|
|
|
2017-10-01 22:33:15 +00:00
|
|
|
.. literalinclude:: ../../docs/source/example-code/src/main/kotlin/net/corda/docs/tutorial/contract/TutorialContract.kt
|
|
|
|
:language: kotlin
|
|
|
|
:start-after: DOCSTART 4
|
|
|
|
:end-before: DOCEND 4
|
|
|
|
:dedent: 8
|
2015-11-25 13:29:51 +00:00
|
|
|
|
2017-10-01 22:33:15 +00:00
|
|
|
.. literalinclude:: ../../docs/source/example-code/src/main/java/net/corda/docs/java/tutorial/contract/CommercialPaper.java
|
|
|
|
:language: java
|
|
|
|
:start-after: DOCSTART 4
|
|
|
|
:end-before: DOCEND 4
|
|
|
|
:dedent: 8
|
2015-11-25 13:29:51 +00:00
|
|
|
|
|
|
|
This loop is the core logic of the contract.
|
|
|
|
|
2018-05-24 15:06:33 +00:00
|
|
|
The first line simply gets the time-window out of the transaction. Setting a time-window in transactions is optional, so a time
|
2015-11-27 17:28:02 +00:00
|
|
|
may be missing here. We check for it being null later.
|
|
|
|
|
2016-08-24 13:38:43 +00:00
|
|
|
.. warning:: In the Kotlin version as long as we write a comparison with the transaction time first the compiler will
|
2015-11-27 17:28:02 +00:00
|
|
|
verify we didn't forget to check if it's missing. Unfortunately due to the need for smooth Java interop, this
|
2015-11-30 17:00:25 +00:00
|
|
|
check won't happen if we write e.g. ``someDate > time``, it has to be ``time < someDate``. So it's good practice to
|
2018-05-24 15:06:33 +00:00
|
|
|
always write the transaction time-window first.
|
2015-11-27 17:28:02 +00:00
|
|
|
|
2017-10-01 22:33:15 +00:00
|
|
|
Next, we take one of three paths, depending on what the type of the command object is.
|
|
|
|
|
|
|
|
**If the command is a ``Move`` command:**
|
|
|
|
|
2015-11-25 13:29:51 +00:00
|
|
|
The first line (first three lines in Java) impose a requirement that there be a single piece of commercial paper in
|
|
|
|
this group. We do not allow multiple units of CP to be split or merged even if they are owned by the same owner. The
|
2015-11-25 18:04:35 +00:00
|
|
|
``single()`` method is a static *extension method* defined by the Kotlin standard library: given a list, it throws an
|
2015-11-25 13:29:51 +00:00
|
|
|
exception if the list size is not 1, otherwise it returns the single item in that list. In Java, this appears as a
|
2016-08-24 13:38:43 +00:00
|
|
|
regular static method of the type familiar from many FooUtils type singleton classes and we have statically imported it
|
|
|
|
here. In Kotlin, it appears as a method that can be called on any JDK list. The syntax is slightly different but
|
|
|
|
behind the scenes, the code compiles to the same bytecodes.
|
2015-11-25 13:29:51 +00:00
|
|
|
|
|
|
|
Next, we check that the transaction was signed by the public key that's marked as the current owner of the commercial
|
|
|
|
paper. Because the platform has already verified all the digital signatures before the contract begins execution,
|
|
|
|
all we have to do is verify that the owner's public key was one of the keys that signed the transaction. The Java code
|
2017-10-01 22:33:15 +00:00
|
|
|
is straightforward: we are simply using the ``Preconditions.checkState`` method from Guava. The Kotlin version looks a
|
|
|
|
little odd: we have a *requireThat* construct that looks like it's built into the language. In fact *requireThat* is an
|
|
|
|
ordinary function provided by the platform's contract API. Kotlin supports the creation of *domain specific languages*
|
|
|
|
through the intersection of several features of the language, and we use it here to support the natural listing of
|
|
|
|
requirements. To see what it compiles down to, look at the Java version. Each ``"string" using (expression)`` statement
|
2017-10-06 09:16:40 +00:00
|
|
|
inside a ``requireThat`` turns into an assertion that the given expression is true, with an ``IllegalArgumentException``
|
2017-10-01 22:33:15 +00:00
|
|
|
being thrown that contains the string if not. It's just another way to write out a regular assertion, but with the
|
|
|
|
English-language requirement being put front and center.
|
|
|
|
|
|
|
|
Next, we simply verify that the output state is actually present: a move is not allowed to delete the CP from the ledger.
|
|
|
|
The grouping logic already ensured that the details are identical and haven't been changed, save for the public key of
|
|
|
|
the owner.
|
|
|
|
|
|
|
|
**If the command is a ``Redeem`` command, then the requirements are more complex:**
|
|
|
|
|
|
|
|
1. We still check there is a CP input state.
|
|
|
|
2. We want to see that the face value of the CP is being moved as a cash claim against some party, that is, the
|
2015-11-25 13:29:51 +00:00
|
|
|
issuer of the CP is really paying back the face value.
|
2017-10-06 09:16:40 +00:00
|
|
|
3. The transaction must be happening after the maturity date.
|
|
|
|
4. The commercial paper must *not* be propagated by this transaction: it must be deleted, by the group having no
|
2015-11-25 13:29:51 +00:00
|
|
|
output state. This prevents the same CP being considered redeemable multiple times.
|
|
|
|
|
2016-08-26 11:05:26 +00:00
|
|
|
To calculate how much cash is moving, we use the ``sumCashBy`` utility function. Again, this is an extension function,
|
2015-11-25 18:04:35 +00:00
|
|
|
so in Kotlin code it appears as if it was a method on the ``List<Cash.State>`` type even though JDK provides no such
|
2016-08-24 13:38:43 +00:00
|
|
|
method. In Java we see its true nature: it is actually a static method named ``CashKt.sumCashBy``. This method simply
|
|
|
|
returns an ``Amount`` object containing the sum of all the cash states in the transaction outputs that are owned by
|
|
|
|
that given public key, or throws an exception if there were no such states *or* if there were different currencies
|
|
|
|
represented in the outputs! So we can see that this contract imposes a limitation on the structure of a redemption
|
|
|
|
transaction: you are not allowed to move currencies in the same transaction that the CP does not involve. This
|
|
|
|
limitation could be addressed with better APIs, if it were to be a real limitation.
|
2015-11-25 13:29:51 +00:00
|
|
|
|
2017-10-01 22:33:15 +00:00
|
|
|
**Finally, we support an ``Issue`` command, to create new instances of commercial paper on the ledger.**
|
|
|
|
|
|
|
|
It likewise enforces various invariants upon the issuance, such as, there must be one output CP state, for instance.
|
2015-11-25 17:49:58 +00:00
|
|
|
|
2016-08-24 13:38:43 +00:00
|
|
|
This contract is simple and does not implement all the business logic a real commercial paper lifecycle
|
2015-11-25 13:29:51 +00:00
|
|
|
management program would. For instance, there is no logic requiring a signature from the issuer for redemption:
|
|
|
|
it is assumed that any transfer of money that takes place at the same time as redemption is good enough. Perhaps
|
|
|
|
that is something that should be tightened. Likewise, there is no logic handling what happens if the issuer has gone
|
|
|
|
bankrupt, if there is a dispute, and so on.
|
|
|
|
|
|
|
|
As the prototype evolves, these requirements will be explored and this tutorial updated to reflect improvements in the
|
2015-11-25 17:49:58 +00:00
|
|
|
contracts API.
|
|
|
|
|
|
|
|
How to test your contract
|
|
|
|
-------------------------
|
|
|
|
|
|
|
|
Of course, it is essential to unit test your new nugget of business logic to ensure that it behaves as you expect.
|
2016-08-24 13:38:43 +00:00
|
|
|
As contract code is just a regular Java function you could write out the logic entirely by hand in the usual
|
|
|
|
manner. But this would be inconvenient, and then you'd get bored of writing tests and that would be bad: you
|
|
|
|
might be tempted to skip a few.
|
2015-11-25 17:49:58 +00:00
|
|
|
|
2016-08-24 13:38:43 +00:00
|
|
|
To make contract testing more convenient Corda provides a language-like API for both Kotlin and Java that lets
|
|
|
|
you easily construct chains of transactions and verify that they either pass validation, or fail with a particular
|
|
|
|
error message.
|
2015-11-25 17:49:58 +00:00
|
|
|
|
2016-08-24 13:38:43 +00:00
|
|
|
Testing contracts with this domain specific language is covered in the separate tutorial, :doc:`tutorial-test-dsl`.
|
2015-11-25 17:49:58 +00:00
|
|
|
|
2016-02-25 12:29:28 +00:00
|
|
|
Adding a generation API to your contract
|
|
|
|
----------------------------------------
|
2015-11-25 17:49:58 +00:00
|
|
|
|
2015-11-27 15:23:32 +00:00
|
|
|
Contract classes **must** provide a verify function, but they may optionally also provide helper functions to simplify
|
2016-02-25 12:29:28 +00:00
|
|
|
their usage. A simple class of functions most contracts provide are *generation functions*, which either create or
|
2015-11-27 15:23:32 +00:00
|
|
|
modify a transaction to perform certain actions (an action is normally mappable 1:1 to a command, but doesn't have to
|
|
|
|
be so).
|
2015-11-25 17:49:58 +00:00
|
|
|
|
2016-02-25 12:29:28 +00:00
|
|
|
Generation may involve complex logic. For example, the cash contract has a ``generateSpend`` method that is given a set of
|
2015-11-27 15:23:32 +00:00
|
|
|
cash states and chooses a way to combine them together to satisfy the amount of money that is being sent. In the
|
|
|
|
immutable-state model that we are using ledger entries (states) can only be created and deleted, but never modified.
|
|
|
|
Therefore to send $1200 when we have only $900 and $500 requires combining both states together, and then creating
|
|
|
|
two new output states of $1200 and $200 back to ourselves. This latter state is called the *change* and is a concept
|
|
|
|
that should be familiar to anyone who has worked with Bitcoin.
|
|
|
|
|
2016-02-25 12:29:28 +00:00
|
|
|
As another example, we can imagine code that implements a netting algorithm may generate complex transactions that must
|
2015-11-27 15:23:32 +00:00
|
|
|
be signed by many people. Whilst such code might be too big for a single utility method (it'd probably be sized more
|
|
|
|
like a module), the basic concept is the same: preparation of a transaction using complex logic.
|
|
|
|
|
|
|
|
For our commercial paper contract however, the things that can be done with it are quite simple. Let's start with
|
|
|
|
a method to wrap up the issuance process:
|
|
|
|
|
|
|
|
.. container:: codeset
|
|
|
|
|
2017-10-01 22:33:15 +00:00
|
|
|
.. literalinclude:: ../../docs/source/example-code/src/main/kotlin/net/corda/docs/tutorial/contract/TutorialContract.kt
|
|
|
|
:language: kotlin
|
|
|
|
:start-after: DOCSTART 5
|
|
|
|
:end-before: DOCEND 5
|
|
|
|
:dedent: 4
|
2015-11-27 15:23:32 +00:00
|
|
|
|
2015-11-30 17:06:59 +00:00
|
|
|
We take a reference that points to the issuing party (i.e. the caller) and which can contain any internal
|
2016-08-24 13:38:43 +00:00
|
|
|
bookkeeping/reference numbers that we may require. The reference field is an ideal place to put (for example) a
|
|
|
|
join key. Then the face value of the paper, and the maturity date. It returns a ``TransactionBuilder``.
|
|
|
|
A ``TransactionBuilder`` is one of the few mutable classes the platform provides. It allows you to add inputs,
|
|
|
|
outputs and commands to it and is designed to be passed around, potentially between multiple contracts.
|
2015-11-27 15:23:32 +00:00
|
|
|
|
2016-02-25 12:29:28 +00:00
|
|
|
.. note:: Generation methods should ideally be written to compose with each other, that is, they should take a
|
2015-12-22 15:28:38 +00:00
|
|
|
``TransactionBuilder`` as an argument instead of returning one, unless you are sure it doesn't make sense to
|
2015-11-27 15:23:32 +00:00
|
|
|
combine this type of transaction with others. In this case, issuing CP at the same time as doing other things
|
2015-11-30 17:00:25 +00:00
|
|
|
would just introduce complexity that isn't likely to be worth it, so we return a fresh object each time: instead,
|
|
|
|
an issuer should issue the CP (starting out owned by themselves), and then sell it in a separate transaction.
|
2015-11-27 15:23:32 +00:00
|
|
|
|
|
|
|
The function we define creates a ``CommercialPaper.State`` object that mostly just uses the arguments we were given,
|
2016-08-24 13:38:43 +00:00
|
|
|
but it fills out the owner field of the state to be the same public key as the issuing party.
|
2015-11-27 15:23:32 +00:00
|
|
|
|
2017-10-01 22:33:15 +00:00
|
|
|
We then combine the ``CommercialPaper.State`` object with a reference to the ``CommercialPaper`` contract, which is
|
|
|
|
defined inside the contract itself
|
|
|
|
|
|
|
|
.. container:: codeset
|
|
|
|
|
|
|
|
.. literalinclude:: ../../docs/source/example-code/src/main/kotlin/net/corda/docs/tutorial/contract/TutorialContract.kt
|
|
|
|
:language: kotlin
|
|
|
|
:start-after: DOCSTART 8
|
|
|
|
:end-before: DOCEND 8
|
|
|
|
:dedent: 4
|
|
|
|
|
|
|
|
.. literalinclude:: ../../docs/source/example-code/src/main/java/net/corda/docs/java/tutorial/contract/CommercialPaper.java
|
|
|
|
:language: java
|
|
|
|
:start-after: DOCSTART 1
|
|
|
|
:end-before: DOCEND 1
|
|
|
|
:dedent: 4
|
|
|
|
|
|
|
|
This value, which is the fully qualified class name of the contract, tells the Corda platform where to find the contract
|
|
|
|
code that should be used to validate a transaction containing an output state of this contract type. Typically the contract
|
|
|
|
code will be included in the transaction as an attachment (see :doc:`tutorial-attachments`).
|
|
|
|
|
2016-08-24 13:38:43 +00:00
|
|
|
The returned partial transaction has a ``Command`` object as a parameter. This is a container for any object
|
2016-08-26 11:05:26 +00:00
|
|
|
that implements the ``CommandData`` interface, along with a list of keys that are expected to sign this transaction. In this case,
|
2015-11-30 17:06:59 +00:00
|
|
|
issuance requires that the issuing party sign, so we put the key of the party there.
|
2015-11-27 15:23:32 +00:00
|
|
|
|
2016-08-24 13:38:43 +00:00
|
|
|
The ``TransactionBuilder`` has a convenience ``withItems`` method that takes a variable argument list. You can pass in
|
2017-10-01 22:33:15 +00:00
|
|
|
any ``StateAndRef`` (input), ``StateAndContract`` (output) or ``Command`` objects and it'll build up the transaction
|
2015-11-27 15:23:32 +00:00
|
|
|
for you.
|
|
|
|
|
2016-08-24 13:38:43 +00:00
|
|
|
There's one final thing to be aware of: we ask the caller to select a *notary* that controls this state and
|
2017-06-05 12:37:23 +00:00
|
|
|
prevents it from being double spent. You can learn more about this topic in the :doc:`key-concepts-notaries` article.
|
2016-08-24 13:38:43 +00:00
|
|
|
|
|
|
|
.. note:: For now, don't worry about how to pick a notary. More infrastructure will come later to automate this
|
|
|
|
decision for you.
|
|
|
|
|
2015-11-27 15:23:32 +00:00
|
|
|
What about moving the paper, i.e. reassigning ownership to someone else?
|
|
|
|
|
|
|
|
.. container:: codeset
|
|
|
|
|
2017-10-01 22:33:15 +00:00
|
|
|
.. literalinclude:: ../../docs/source/example-code/src/main/kotlin/net/corda/docs/tutorial/contract/TutorialContract.kt
|
|
|
|
:language: kotlin
|
|
|
|
:start-after: DOCSTART 6
|
|
|
|
:end-before: DOCEND 6
|
|
|
|
:dedent: 4
|
2015-11-27 15:23:32 +00:00
|
|
|
|
2015-12-22 15:28:38 +00:00
|
|
|
Here, the method takes a pre-existing ``TransactionBuilder`` and adds to it. This is correct because typically
|
2015-11-27 15:23:32 +00:00
|
|
|
you will want to combine a sale of CP atomically with the movement of some other asset, such as cash. So both
|
2016-02-25 12:29:28 +00:00
|
|
|
generate methods should operate on the same transaction. You can see an example of this being done in the unit tests
|
2015-11-27 15:23:32 +00:00
|
|
|
for the commercial paper contract.
|
|
|
|
|
|
|
|
The paper is given to us as a ``StateAndRef<CommercialPaper.State>`` object. This is exactly what it sounds like:
|
2017-10-06 09:29:29 +00:00
|
|
|
a small object that has a (copy of a) state object, and also the ``(txhash, index)`` that indicates the location of this
|
2015-11-27 15:23:32 +00:00
|
|
|
state on the ledger.
|
|
|
|
|
2016-08-24 13:38:43 +00:00
|
|
|
We add the existing paper state as an input, the same paper state with the owner field adjusted as an output,
|
|
|
|
and finally a move command that has the old owner's public key: this is what forces the current owner's signature
|
|
|
|
to be present on the transaction, and is what's checked by the contract.
|
|
|
|
|
2015-11-27 15:23:32 +00:00
|
|
|
Finally, we can do redemption.
|
|
|
|
|
|
|
|
.. container:: codeset
|
|
|
|
|
2017-10-01 22:33:15 +00:00
|
|
|
.. literalinclude:: ../../docs/source/example-code/src/main/kotlin/net/corda/docs/tutorial/contract/TutorialContract.kt
|
|
|
|
:language: kotlin
|
|
|
|
:start-after: DOCSTART 7
|
|
|
|
:end-before: DOCEND 7
|
|
|
|
:dedent: 4
|
2015-11-27 15:23:32 +00:00
|
|
|
|
|
|
|
Here we can see an example of composing contracts together. When an owner wishes to redeem the commercial paper, the
|
2017-06-08 10:53:45 +00:00
|
|
|
issuer (i.e. the caller) must gather cash from its vault and send the face value to the owner of the paper.
|
2015-11-27 15:23:32 +00:00
|
|
|
|
2016-08-24 13:38:43 +00:00
|
|
|
.. note:: This contract has no explicit concept of rollover.
|
2015-11-30 17:00:25 +00:00
|
|
|
|
2017-06-08 10:53:45 +00:00
|
|
|
The *vault* is a concept that may be familiar from Bitcoin and Ethereum. It is simply a set of states (such as cash) that are
|
|
|
|
owned by the caller. Here, we use the vault to update the partial transaction we are handed with a movement of cash
|
|
|
|
from the issuer of the commercial paper to the current owner. If we don't have enough quantity of cash in our vault,
|
2016-08-26 11:05:26 +00:00
|
|
|
an exception is thrown. Then we add the paper itself as an input, but, not an output (as we wish to remove it
|
2016-08-24 13:38:43 +00:00
|
|
|
from the ledger). Finally, we add a Redeem command that should be signed by the owner of the commercial paper.
|
|
|
|
|
2017-08-03 16:17:17 +00:00
|
|
|
.. warning:: The amount we pass to the ``Cash.generateSpend`` function has to be treated first with ``withoutIssuer``.
|
2016-08-24 13:38:43 +00:00
|
|
|
This reflects the fact that the way we handle issuer constraints is still evolving; the commercial paper
|
|
|
|
contract requires payment in the form of a currency issued by a specific party (e.g. the central bank,
|
2017-06-08 10:53:45 +00:00
|
|
|
or the issuers own bank perhaps). But the vault wants to assemble spend transactions using cash states from
|
2016-08-24 13:38:43 +00:00
|
|
|
any issuer, thus we must strip it here. This represents a design mismatch that we will resolve in future
|
|
|
|
versions with a more complete way to express issuer constraints.
|
2015-11-27 15:23:32 +00:00
|
|
|
|
2015-12-22 15:28:38 +00:00
|
|
|
A ``TransactionBuilder`` is not by itself ready to be used anywhere, so first, we must convert it to something that
|
2017-05-12 16:42:13 +00:00
|
|
|
is recognised by the network. The most important next step is for the participating entities to sign it. Typically,
|
2017-08-04 10:50:52 +00:00
|
|
|
an initiating flow will create an initial partially signed ``SignedTransaction`` by calling the ``serviceHub.toSignedTransaction`` method.
|
2017-05-12 16:42:13 +00:00
|
|
|
Then the frozen ``SignedTransaction`` can be passed to other nodes by the flow, these can sign using ``serviceHub.createSignature`` and distribute.
|
|
|
|
The ``CollectSignaturesFlow`` provides a generic implementation of this process that can be used as a ``subFlow`` .
|
2015-11-27 15:23:32 +00:00
|
|
|
|
|
|
|
You can see how transactions flow through the different stages of construction by examining the commercial paper
|
|
|
|
unit tests.
|
2015-11-25 17:49:58 +00:00
|
|
|
|
2016-08-24 13:38:43 +00:00
|
|
|
How multi-party transactions are constructed and transmitted
|
|
|
|
------------------------------------------------------------
|
2015-11-25 17:49:58 +00:00
|
|
|
|
2016-08-24 13:38:43 +00:00
|
|
|
OK, so now we know how to define the rules of the ledger, and we know how to construct transactions that satisfy
|
|
|
|
those rules ... and if all we were doing was maintaining our own data that might be enough. But we aren't: Corda
|
|
|
|
is about keeping many different parties all in sync with each other.
|
2015-11-25 17:49:58 +00:00
|
|
|
|
2016-08-24 13:38:43 +00:00
|
|
|
In a classical blockchain system all data is transmitted to everyone and if you want to do something fancy, like
|
|
|
|
a multi-party transaction, you're on your own. In Corda data is transmitted only to parties that need it and
|
|
|
|
multi-party transactions are a way of life, so we provide lots of support for managing them.
|
2015-11-25 17:49:58 +00:00
|
|
|
|
2016-08-24 13:38:43 +00:00
|
|
|
You can learn how transactions are moved between peers and taken through the build-sign-notarise-broadcast
|
2016-11-22 16:30:17 +00:00
|
|
|
process in a separate tutorial, :doc:`flow-state-machines`.
|
2015-11-25 17:49:58 +00:00
|
|
|
|
2016-08-24 13:38:43 +00:00
|
|
|
Non-asset-oriented smart contracts
|
|
|
|
----------------------------------
|
2015-11-25 17:49:58 +00:00
|
|
|
|
2016-08-24 13:38:43 +00:00
|
|
|
Although this tutorial covers how to implement an owned asset, there is no requirement that states and code contracts
|
|
|
|
*must* be concerned with ownership of an asset. It is better to think of states as representing useful facts about the
|
|
|
|
world, and (code) contracts as imposing logical relations on how facts combine to produce new facts. Alternatively
|
|
|
|
you can imagine that states are like rows in a relational database and contracts are like stored procedures and
|
|
|
|
relational constraints.
|
2015-11-25 17:49:58 +00:00
|
|
|
|
2016-08-24 13:38:43 +00:00
|
|
|
When writing a contract that handles deal-like entities rather than asset-like entities, you may wish to refer
|
|
|
|
to ":doc:`contract-irs`" and the accompanying source code. Whilst all the concepts are the same, deals are
|
|
|
|
typically not splittable or mergeable and thus you don't have to worry much about grouping of states.
|
2015-11-25 17:49:58 +00:00
|
|
|
|
2016-08-24 13:38:43 +00:00
|
|
|
Making things happen at a particular time
|
|
|
|
-----------------------------------------
|
2015-11-25 17:49:58 +00:00
|
|
|
|
2016-08-24 13:38:43 +00:00
|
|
|
It would be nice if you could program your node to automatically redeem your commercial paper as soon as it matures.
|
|
|
|
Corda provides a way for states to advertise scheduled events that should occur in future. Whilst this information
|
|
|
|
is by default ignored, if the corresponding *Cordapp* is installed and active in your node, and if the state is
|
2017-06-08 10:53:45 +00:00
|
|
|
considered relevant by your vault (e.g. because you own it), then the node can automatically begin the process
|
2016-08-24 13:38:43 +00:00
|
|
|
of creating a transaction and taking it through the life cycle. You can learn more about this in the article
|
|
|
|
":doc:`event-scheduling`".
|
2016-07-12 09:24:11 +00:00
|
|
|
|
2016-09-06 16:56:01 +00:00
|
|
|
Encumbrances
|
|
|
|
------------
|
|
|
|
|
|
|
|
All contract states may be *encumbered* by up to one other state, which we call an **encumbrance**.
|
|
|
|
|
2017-10-01 22:33:15 +00:00
|
|
|
The encumbrance state, if present, forces additional controls over the encumbered state, since the encumbrance state
|
|
|
|
contract will also be verified during the execution of the transaction. For example, a contract state could be
|
|
|
|
encumbered with a time-lock contract state; the state is then only processable in a transaction that verifies that the
|
|
|
|
time specified in the encumbrance time-lock has passed.
|
2016-09-06 16:56:01 +00:00
|
|
|
|
2017-10-01 22:33:15 +00:00
|
|
|
The encumbered state refers to its encumbrance by index, and the referred encumbrance state is an output state in a
|
|
|
|
particular position on the same transaction that created the encumbered state. Note that an encumbered state that is
|
|
|
|
being consumed must have its encumbrance consumed in the same transaction, otherwise the transaction is not valid.
|
2016-09-06 16:56:01 +00:00
|
|
|
|
|
|
|
The encumbrance reference is optional in the ``ContractState`` interface:
|
|
|
|
|
|
|
|
.. container:: codeset
|
|
|
|
|
|
|
|
.. sourcecode:: kotlin
|
|
|
|
|
|
|
|
val encumbrance: Int? get() = null
|
|
|
|
|
|
|
|
.. sourcecode:: java
|
|
|
|
|
|
|
|
@Nullable
|
|
|
|
@Override
|
|
|
|
public Integer getEncumbrance() {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
The time-lock contract mentioned above can be implemented very simply:
|
|
|
|
|
|
|
|
.. container:: codeset
|
|
|
|
|
|
|
|
.. sourcecode:: kotlin
|
|
|
|
|
|
|
|
class TestTimeLock : Contract {
|
|
|
|
...
|
2017-07-19 12:06:43 +00:00
|
|
|
override fun verify(tx: LedgerTransaction) {
|
2018-05-24 15:06:33 +00:00
|
|
|
val time = tx.timeWindow?.untilTime ?: throw IllegalStateException(...)
|
2016-09-06 16:56:01 +00:00
|
|
|
...
|
|
|
|
requireThat {
|
|
|
|
"the time specified in the time-lock has passed" by
|
|
|
|
(time >= tx.inputs.filterIsInstance<TestTimeLock.State>().single().validFrom)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
...
|
|
|
|
}
|
|
|
|
|
|
|
|
We can then set up an encumbered state:
|
|
|
|
|
|
|
|
.. container:: codeset
|
|
|
|
|
|
|
|
.. sourcecode:: kotlin
|
|
|
|
|
|
|
|
val encumberedState = Cash.State(amount = 1000.DOLLARS `issued by` defaultIssuer, owner = DUMMY_PUBKEY_1, encumbrance = 1)
|
|
|
|
val fourPmTimelock = TestTimeLock.State(Instant.parse("2015-04-17T16:00:00.00Z"))
|
|
|
|
|
|
|
|
When we construct a transaction that generates the encumbered state, we must place the encumbrance in the corresponding output
|
|
|
|
position of that transaction. And when we subsequently consume that encumbered state, the same encumbrance state must be
|
|
|
|
available somewhere within the input set of states.
|
|
|
|
|
2016-09-08 21:49:24 +00:00
|
|
|
In future, we will consider the concept of a *covenant*. This is where the encumbrance travels alongside each iteration of
|
|
|
|
the encumbered state. For example, a cash state may be encumbered with a *domicile* encumbrance, which checks the domicile of
|
|
|
|
the identity of the owner that the cash state is being moved to, in order to uphold sanction screening regulations, and prevent
|
|
|
|
cash being paid to parties domiciled in e.g. North Korea. In this case, the encumbrance should be permanently attached to
|
2016-09-09 09:31:52 +00:00
|
|
|
the all future cash states stemming from this one.
|
2016-09-08 21:49:24 +00:00
|
|
|
|
|
|
|
We will also consider marking states that are capable of being encumbrances as such. This will prevent states being used
|
|
|
|
as encumbrances inadvertently. For example, the time-lock above would be usable as an encumbrance, but it makes no sense to
|
|
|
|
be able to encumber a cash state with another one.
|