2017-01-06 11:05:37 +00:00
|
|
|
Integration testing
|
|
|
|
===================
|
2016-11-25 15:26:00 +00:00
|
|
|
|
2016-11-28 13:37:38 +00:00
|
|
|
Integration testing involves bringing up nodes locally and testing
|
|
|
|
invariants about them by starting flows and inspecting their state.
|
2016-11-25 15:26:00 +00:00
|
|
|
|
2016-11-28 13:37:38 +00:00
|
|
|
In this tutorial we will bring up three nodes Alice, Bob and a
|
|
|
|
Notary. Alice will issue Cash to Bob, then Bob will send this Cash
|
|
|
|
back to Alice. We will see how to test some simple deterministic and
|
|
|
|
nondeterministic invariants in the meantime.
|
2016-11-25 15:26:00 +00:00
|
|
|
|
2016-11-28 13:37:38 +00:00
|
|
|
(Note that this example where Alice is self-issuing Cash is purely for
|
|
|
|
demonstration purposes, in reality Cash would be issued by a bank and
|
|
|
|
subsequently passed around.)
|
2016-11-28 11:50:12 +00:00
|
|
|
|
2016-11-28 13:37:38 +00:00
|
|
|
In order to spawn nodes we will use the Driver DSL. This DSL allows
|
|
|
|
one to start up node processes from code. It manages a network map
|
|
|
|
service and safe shutting down of nodes in the background.
|
2016-11-25 15:26:00 +00:00
|
|
|
|
|
|
|
.. literalinclude:: example-code/src/integration-test/kotlin/net/corda/docs/IntegrationTestingTutorial.kt
|
|
|
|
:language: kotlin
|
|
|
|
:start-after: START 1
|
|
|
|
:end-before: END 1
|
|
|
|
|
2016-11-28 13:37:38 +00:00
|
|
|
The above code creates a ``User`` permissioned to start the
|
|
|
|
``CashFlow`` protocol. It then starts up Alice and Bob with this user,
|
|
|
|
allowing us to later connect to the nodes.
|
2016-11-25 15:26:00 +00:00
|
|
|
|
2016-11-28 13:37:38 +00:00
|
|
|
Then the notary is started up. Note that we need to add
|
|
|
|
``ValidatingNotaryService`` as an advertised service in order for this
|
|
|
|
node to serve notary functionality. This is also where flows added in
|
|
|
|
plugins should be specified. Note also that we won't connect to the
|
|
|
|
notary directly, so there's no need to pass in the test ``User``.
|
2016-11-25 15:26:00 +00:00
|
|
|
|
2016-11-28 13:37:38 +00:00
|
|
|
The ``startNode`` function returns a future that completes once the
|
|
|
|
node is fully started. This allows starting of the nodes to be
|
|
|
|
parallel. We wait on these futures as we need the information
|
2017-06-30 16:29:57 +01:00
|
|
|
returned; their respective ``NodeHandles`` s. After getting the handles we
|
|
|
|
wait for both parties to register with the network map to ensure we don't
|
|
|
|
have race conditions with network map registration.
|
2016-11-25 15:26:00 +00:00
|
|
|
|
|
|
|
.. literalinclude:: example-code/src/integration-test/kotlin/net/corda/docs/IntegrationTestingTutorial.kt
|
|
|
|
:language: kotlin
|
|
|
|
:start-after: START 2
|
|
|
|
:end-before: END 2
|
|
|
|
|
2016-11-28 13:37:38 +00:00
|
|
|
Next we connect to Alice and Bob respectively from the test process
|
|
|
|
using the test user we created. Then we establish RPC links that allow
|
|
|
|
us to start flows and query state.
|
2016-11-25 15:26:00 +00:00
|
|
|
|
|
|
|
.. literalinclude:: example-code/src/integration-test/kotlin/net/corda/docs/IntegrationTestingTutorial.kt
|
|
|
|
:language: kotlin
|
|
|
|
:start-after: START 3
|
|
|
|
:end-before: END 3
|
|
|
|
|
2016-11-28 13:37:38 +00:00
|
|
|
We will be interested in changes to Alice's and Bob's vault, so we
|
|
|
|
query a stream of vault updates from each.
|
2016-11-25 15:26:00 +00:00
|
|
|
|
|
|
|
Now that we're all set up we can finally get some Cash action going!
|
|
|
|
|
|
|
|
.. literalinclude:: example-code/src/integration-test/kotlin/net/corda/docs/IntegrationTestingTutorial.kt
|
|
|
|
:language: kotlin
|
|
|
|
:start-after: START 4
|
|
|
|
:end-before: END 4
|
|
|
|
|
2016-11-28 13:37:38 +00:00
|
|
|
The first loop creates 10 threads, each starting a ``CashFlow`` flow
|
|
|
|
on the Alice node. We specify that we want to issue ``i`` dollars to
|
|
|
|
Bob, using the Notary as the notary responsible for notarising the
|
|
|
|
created states. Note that no notarisation will occur yet as we're not
|
|
|
|
spending any states, only entering new ones to the ledger.
|
2016-11-25 15:26:00 +00:00
|
|
|
|
2016-11-28 13:37:38 +00:00
|
|
|
We started the flows from different threads for the sake of the
|
|
|
|
tutorial, to demonstrate how to test non-determinism, which is what
|
|
|
|
the ``expectEvents`` block does.
|
2016-11-25 15:26:00 +00:00
|
|
|
|
2016-11-28 13:37:38 +00:00
|
|
|
The Expect DSL allows ordering constraints to be checked on a stream
|
|
|
|
of events. The above code specifies that we are expecting 10 updates
|
|
|
|
to be emitted on the ``bobVaultUpdates`` stream in unspecified order
|
|
|
|
(this is what the ``parallel`` construct does). We specify a
|
|
|
|
(otherwise optional) ``match`` predicate to identify specific updates
|
|
|
|
we are interested in, which we then print.
|
2016-11-25 15:26:00 +00:00
|
|
|
|
2016-11-28 13:37:38 +00:00
|
|
|
If we run the code written so far we should see 4 nodes starting up
|
|
|
|
(Alice,Bob,Notary + implicit Network Map service), then 10 logs of Bob
|
|
|
|
receiving 1,2,...10 dollars from Alice in some unspecified order.
|
2016-11-25 15:26:00 +00:00
|
|
|
|
|
|
|
Next we want Bob to send this Cash back to Alice.
|
|
|
|
|
|
|
|
.. literalinclude:: example-code/src/integration-test/kotlin/net/corda/docs/IntegrationTestingTutorial.kt
|
|
|
|
:language: kotlin
|
|
|
|
:start-after: START 5
|
|
|
|
:end-before: END 5
|
|
|
|
|
2016-11-28 13:37:38 +00:00
|
|
|
This time we'll do it sequentially. We make Bob pay 1,2,..10 dollars
|
|
|
|
to Alice in order. We make sure that a the ``CashFlow`` has finished
|
|
|
|
by waiting on ``startFlow`` 's ``returnValue``.
|
2016-11-25 15:26:00 +00:00
|
|
|
|
2016-11-28 13:37:38 +00:00
|
|
|
Then we use the Expect DSL again, this time using ``sequence`` to test
|
|
|
|
for the updates arriving in the order we expect them to.
|
2016-11-25 15:26:00 +00:00
|
|
|
|
2016-11-28 13:37:38 +00:00
|
|
|
Note that ``parallel`` and ``sequence`` may be nested into each other
|
|
|
|
arbitrarily to test more complex scenarios.
|
2016-11-25 15:26:00 +00:00
|
|
|
|
2016-11-28 13:37:38 +00:00
|
|
|
That's it! We saw how to start up several corda nodes locally, how to
|
|
|
|
connect to them, and how to test some simple invariants about
|
|
|
|
``CashFlow``.
|
2016-11-25 15:26:00 +00:00
|
|
|
|
2016-11-28 13:37:38 +00:00
|
|
|
To run the complete test you can open
|
|
|
|
``example-code/src/integration-test/kotlin/net/corda/docs/IntegrationTestingTutorial.kt``
|
|
|
|
from IntelliJ and run the test, or alternatively use gradle:
|
2016-11-25 15:26:00 +00:00
|
|
|
|
|
|
|
|
|
|
|
.. sourcecode:: bash
|
|
|
|
|
|
|
|
# Run example-code integration tests
|
|
|
|
./gradlew docs/source/example-code:integrationTest -i
|