Cordform task
Corda’s cordformation
Gradle plugin provides the Cordform
and Dockerform
tasks. They both allow you to run tasks that automatically generate and configure a local set of nodes for testing and demonstration purposes. This page contains information about the operation of the Cordform task. Visit the Dockerform page for Dockerform configuration options.
- A
Cordform
task creates nodes in thebuild/nodes
directory. The exampleCordform
task used in this document creates three nodes:Notary
,PartyA
, andPartyB
, however you are free to spin up more nodes, specify what nodes you need on the network, change node names, and update node configurations. Cordform
tasks require you to deploy each Corda node and database separately.
Tasks using Cordform
The following example, as defined in the Kotlin CorDapp Template, shows a Cordform
task called deployNodes
that creates three nodes:
A
Notary
node, which:- Provides a validating Notary service.
- Runs the
corda-finance
CorDapp.
PartyA
andPartyB
nodes, each of which:- Does not provide any services.
- Runs the
corda-finance
CorDapp. - Has an RPC (Remote Procedure Call) user (
user1
), which enables you to log in the node via RPC.
All three nodes also include any CorDapps defined in the project’s source directories, even if these CorDapps are not listed in each node’s cordapps
setting. As a result, if you run the deployNodes
task from the template CorDapp, for example, it will automatically build and add the template CorDapp to each node.
Cordform
allows you to specify any number of nodes and define their configurations and names as needed.Run this example task to create the three nodes in the build/nodes
directory. Make sure to use Corda gradle plug-in version 5.0.10 or above:
task deployNodes(type: net.corda.plugins.Cordform, dependsOn: ['jar']) {
directory "./build/nodes"
node {
name "O=Notary,L=London,C=GB"
// The Notary will offer a validating Notary service.
notary = [validating : true]
p2pPort 10002
rpcSettings {
port 10003
adminPort 10023
}
h2Port 10004
// Starts an internal SSH server providing a management shell on the node.
sshdPort 2223
// Includes the corda-finance CorDapp on our node.
cordapps = ["$corda_release_distribution:corda-finance:$corda_release_version"]
extraConfig = [
// Setting the JMX reporter type.
jmxReporterType: 'JOLOKIA',
// Setting the H2 address.
h2Settings: [ address: 'localhost:10030' ]
]
}
node {
name "O=PartyA,L=London,C=GB"
p2pPort 10005
rpcSettings {
port 10006
adminPort 10026
}
h2Port 10008
cordapps = ["$corda_release_distribution:corda-finance:$corda_release_version"]
// Grants user1 all RPC permissions.
rpcUsers = [[ user: "user1", "password": "test", "permissions": ["ALL"]]]
}
node {
name "O=PartyB,L=New York,C=US"
p2pPort 10009
rpcSettings {
port 10010
adminPort 10030
}
h2Port 10012
cordapps = ["$corda_release_distribution:corda-finance:$corda_release_version"]
// Grants user1 the ability to start the MyFlow flow.
rpcUsers = [[ user: "user1", "password": "test", "permissions": ["StartFlow.net.corda.flows.MyFlow"]]]
}
}
You can turn deploying the local project’s CorDapp to each node off by adding the following code to your node configuration:
projectCordapp {
deploy = false
}
The Cordform
and Dockerform
also support a nodeDefaults
block, which can
contain configuration common to all nodes, for example:
nodeDefaults {
cordapp project(':contracts')
cordapp project(':workflows')
runSchemaMigration = true
rpcUsers = [[user: "user1", "password": "test", "permissions": ["ALL"]]
}
You can override these defaults for each node:
node {
name = "O=Notary,L=London,C=GB"
notary = [ validating: true ]
rpcUsers = []
}
Minimum configuration
There are four mandatory fields that must be set for the Cordform plugin:
name
<string> is used to specify the legal identity name of the Corda node. For more information, see myLegalName. For example:
name "O=PartyA,L=London,C=GB"
p2pAddress
<string> is used to specify the address/port the node uses for inbound communication from other nodes. For more information, see p2pAddress. Required ifp2pPort
is not specified. For example:
p2pAddress "example.com:10002"
p2pPort
<integer> is used to specify the port the node uses for inbound communication from other nodes. The assumed IP address islocalhost
. For more information, see p2pAddress. For example:
p2pPort 10006 // "localhost:10006"
rpcSettings
<config> is used to specify RPC settings for the node. For more information, see rpcSettings. For example:
rpcSettings {
port 10006
adminPort 10026
}
Optional configuration
In addition to the four mandatory configuration fields, there are a number of optional configuration options to meet the needs of more specific plugin deployments:
notary
<config> is used to specify the node as a Notary node. Required> for Notary nodes. For more information, see Notary.devMode
<boolean> is used to enable development mode when you set its value totrue
. For more information, see devMode. For example:
devMode true
rpcUsers
<list> is used to set the RPC users for the node. For more information, see rpcUsers. You can use arbitrary values in this configuration block - “incorrect” settings will not cause a DSL error. An example follows below:
rpcUsers = [[ user: "user1", "password": "test", "permissions": ["StartFlow.net.corda.flows.MyFlow"]]]
configFile
<string> is used to generate an extended node configuration. For example:
configFile = "samples/trader-demo/src/main/resources/node-b.conf"
sshdPort
<integer> is used to specify the SSH port for the Docker container. This will be mapped to the same port on the host. IfsshdPort
is specified, then that port must be available on the host and not in use by some other service. IfsshdPort
is not specified, then a default value will be used for the SSH port on the container. Use thedocker port <container_name>
command to check which port has been allocated on the host for your container. For more information, see sshd. For example:
sshd {
port = 2222
}
You can extend the deployNodes
task with more node {}
blocks to generate as many nodes as necessary for your application.
To extend node configuration beyond the properties defined in the deployNodes
task, use the configFile
property with the file path (relative or absolute) set to an additional configuration file. This file should follow the standard Node configuration format of node.conf
. The properties set there will be appended to the generated node configuration.
deployNodes
task, both properties will be present in generated node configuration.You can also add the path to the additional configuration file while running the gradle task via the -PconfigFile
command-line option. However, this will result in the same configuration file being applied to all nodes.
The PartyB
node in the next example has additional configuration options added from a file called none-b.conf
:
task deployNodes(type: net.corda.plugins.Cordform, dependsOn: ['jar']) {
[...]
node {
name "O=PartyB,L=New York,C=US"
[...]
// Grants user1 the ability to start the MyFlow flow.
rpcUsers = [[ user: "user1", "password": "test", "permissions": ["StartFlow.net.corda.flows.MyFlow"]]]
configFile = "samples/trader-demo/src/main/resources/node-b.conf"
}
}
The drivers
parameter in the node
entry lists paths of the files to be copied to the drivers
sub-directory of the node.
To add any drivers as dependencies of the cordaDriver
configuration, use the following code (option recommended over using the drivers
parameter):
dependencies {
cordaDriver "net.corda:corda-shell:$corda_release_version"
cordaDriver files('lib/my_specific_jar.jar')
}
The Cordform
and Dockerform
tasks copy the resolved contents of Gradle’s
cordaDriver
configuration into each node’s drivers
directory before
running the bootstrapper.
Package namespace ownership
To configure package namespace ownership, use the optional networkParameterOverrides
and packageOwnership
blocks, in a similar way to how the configuration file is used by the Network Bootstrapper tool. For example:
task deployNodes(type: net.corda.plugins.Cordform, dependsOn: ['jar']) {
[...]
networkParameterOverrides {
packageOwnership {
"com.mypackagename" {
keystore = "_teststore"
keystorePassword = "MyStorePassword"
keystoreAlias = "MyKeyAlias"
}
}
}
[...]
}
Sign CorDapp JAR files
CorDapp JAR files created by the gradle cordapp
plug-in are signed by a Corda development certificate by default.
You can use the Cordform signing
entry to override and customise the signing of CorDapp JAR files.
Signing a CorDapp enables its contract classes to use signature constraints instead of other types of constraints, such as Contract Constraints.
The signing task may use an external keystore, or create a new one.
You can use the following parameters in the signing
entry:
enabled
is the control flag to enable the signing process. It is set tofalse
by default. Set totrue
to enable signing.all
, if set totrue
(default), all CorDapps inside thecordapp
sub-directory will be signed. If set tofalse
, only the generated Cordapp will be signed.options
covers any relevant parameters of SignJar ANT task and GenKey ANT task. By default, the JAR file is signed by a Corda development key. You can specify the external keystore can be specified. The minimal list of required options is shown below. For other options, see SignJar task.keystore
is the path to the keystore file. The default setting iscordadevcakeys.jks
. The keystore is shipped with the plug-in.alias
is the alias to sign under. The default value iscordaintermediateca
.storepass
is the keystore password. The default value iscordacadevpass
.keypass
is the private key password, if it is different from the keystore password. The default value iscordacadevkeypass
.storetype
is the keystore type. The default value isJKS
.dname
is the distinguished name for the entity. Only use this option whengenerateKeystore
is set totrue
(see below).keyalg
is the method to use when generating a name-value pair. The default value isRSA
because Corda does not supportDSA
. Only use this option whengenerateKeystore
is set totrue
(see below).
generateKeystore
is the flag to generate a keystore. The default value isfalse
. If set totrue
, an “ad hoc” keystore is created and its key is used instead of the default Corda development key or any external key. The sameoptions
to specify an external keystore are used to define the newly created keystore. In addition,dname
andkeyalg
are required.- Other options are described in GenKey task. If the existing keystore is already present, the task will reuse it. However if the file is inside the
build
directory, then it will be deleted when the gradleclean
task is run.
- Other options are described in GenKey task. If the existing keystore is already present, the task will reuse it. However if the file is inside the
The example below shows the minimal set of options
required to create a dummy keystore:
task deployNodes(type: net.corda.plugins.Cordform, dependsOn: ['jar']) {
signing {
enabled true
generateKeystore true
all false
options {
keystore "./build/nodes/jarSignKeystore.p12"
alias "cordapp-signer"
storepass "secret1!"
storetype "PKCS12"
dname "OU=Dummy Cordapp Distributor, O=Corda, L=London, C=GB"
keyalg "RSA"
}
}
//...
Contract classes from signed CorDapp JAR files are checked by signature constraints by default.
You can force them to be checked by zone constraints by adding contract class names to the includeWhitelist
entry - the list will generate an include_whitelist.txt
file used internally by the Network Bootstrapper tool.
Before you add includeWhitelist
to the deployNodes
task, see Contract Constraints to understand the implications of using different constraint types.
The snippet below configures contracts classes from the Finance CorDapp to be verified using zone constraints instead of signature constraints:
task deployNodes(type: net.corda.plugins.Cordform, dependsOn: ['jar']) {
includeWhitelist = [ "net.corda.finance.contracts.asset.Cash", "net.corda.finance.contracts.asset.CommercialPaper" ]
//...
Optional migration step
If you are migrating your database schema from an older Corda version to Corda 4.12, you must add the following parameter to the node section in the build.gradle
and set it to true
:
runSchemaMigration = true
This step runs the full schema migration process as the last step of the Cordform task, and leave the nodes ready to run.
Run the Cordform task
To create the nodes defined in the deployNodes
task example above, run the following command in a command prompt or a terminal window, from the root of the project where the deployNodes
task is defined:
- Linux/macOS:
./gradlew deployNodes
- Windows:
gradlew.bat deployNodes
This command creates the nodes in the build/nodes
directory. A node directory is generated for each node defined in the deployNodes
task, plus a runnodes
shell script (or a batch file on Windows) to run all the nodes at once for testing and development purposes. If you make any changes to your CorDapp source or deployNodes
task, you will need to re-run the task to see the changes take effect.
Was this page helpful?
Thanks for your feedback!
Chat with us
Chat with us on our #docs channel on slack. You can also join a lot of other slack channels there and have access to 1-on-1 communication with members of the R3 team and the online community.
Propose documentation improvements directly
Help us to improve the docs by contributing directly. It's simple - just fork this repository and raise a PR of your own - R3's Technical Writers will review it and apply the relevant suggestions.
We're sorry this page wasn't helpful. Let us know how we can make it better!
Chat with us
Chat with us on our #docs channel on slack. You can also join a lot of other slack channels there and have access to 1-on-1 communication with members of the R3 team and the online community.
Create an issue
Create a new GitHub issue in this repository - submit technical feedback, draw attention to a potential documentation bug, or share ideas for improvement and general feedback.
Propose documentation improvements directly
Help us to improve the docs by contributing directly. It's simple - just fork this repository and raise a PR of your own - R3's Technical Writers will review it and apply the relevant suggestions.