Mirabelle ships with a complete test framework for streams.
First, let’s define and compile some streams:
(streams
(stream
{:name :foo}
(where [:= :service "foo"]
(info)
(tap :foo)))
(stream
{:name :bar}
(where [:> :metric 100]
(info)
(tap :bar))))
The first stream named :foo
will only keep all events with :service
“foo” and log them. These events will also be registered into a tap
named :foo
.
A tap is a Mirabelle action which only has an effect in test mode. Events will be saved into tap, and you will be able to check what was registered on the taps.
Our second stream is also simple: it keeps all events with :metric
greater to 100, log them, and push them into a tap named :bar
.
Let’s write a test file for these streams. The tests file should be referenced in the :test
section of the configuration.
{:test1 {:input [{:metric 10
:service "foo"}
{:metric 101
:service "bar"}]
:taps {:foo [{:metric 10
:service "foo"}]
:bar [{:metric 101
:service "bar"}]}}}
We defined here a test named :test1
. You could have add more tests to the map (or in another files), all tests are run in isolation.
This :test1
key contains a map with two keys:
:input
: the events which will be injected into the streams:taps
: the expected contents of the tap.Here, we see that the tap :foo
should contain the first event (with :service
“foo”), and the tap :bar
the second (with :metric
greater than 100).
You can low run your tests:
export MIRABELLE_CONFIGURATION=/path/to/mirabelle/config.edn
java -jar mirabelle.jar test
All tests successful
If a test fails (let’s say we modify the :foo
tap result, you should get:
1 errors
Error in test :test1
Invalid result for tap :foo
expected: [{:metric 10, :service "not-good"}]
actual: [{:metric 10, :service "foo"}]
Tests can also take a :target
configuration, for example:
{:test1 {:input [{:metric 10
:service "foo"}
{:metric 101
:service "bar"}]
:target :foo
:taps {:foo [{:metric 10
:service "foo"}]}}}
In this example, events in :input
will only be injected into the :foo
stream.
In test mode, some streams behave differently than in the regular mode:
output!
actions) are discarded.publish!
action is also discarded in test mode.async-queue
actions are replaced by a simple stream which forward all events to children. It allows you to test streams using async queues without dealing with their side effects.io
will be discarded. The io
action will in test mode discard all of its children (and on regular mode, just forward events to children).