Zeebe Process Test
Zeebe Process Test allows you to unit test your Camunda 8 BPMN processes. It will start a lightweight in-memory Zeebe engine and provide you with a set of assertions you can use to verify your process behaves as expected.
As of 8.5.0, Zeebe Process Test does not support the new REST API and related features (e.g. Zeebe user tasks).
Prerequisites
This library requires the following:
- Java 21+ when running with an embedded engine (
zeebe-process-test-extension
) - Java 8+ and Docker when running using testcontainers (
zeebe-process-test-extension-testcontainer
) - JUnit 5
Dependency
Zeebe Process Test provides you with two dependencies. Which one you need to use is dependent on the Java version you are using.
Testcontainers (JDK 8+)
If you are building your project with a JDK lower than 21, use the testcontainer
dependency. This starts a testcontainer
where a Zeebe engine is running. It is beneficial to use this version instead of the embedded version so your code can be implemented independently of the Java version used by the Zeebe engine. The downside is that testcontainers
provide some overhead, which means tests will be slower. Additionally, Docker must be running to execute the tests.
<dependency>
<groupId>io.camunda</groupId>
<artifactId>zeebe-process-test-extension-testcontainer</artifactId>
<version>X.Y.Z</version>
<scope>test</scope>
</dependency>
Embedded (JDK 21+)
If you are building your project with JDK 21+, you can make use of an embedded Zeebe engine. The advantage of using this instead of the testcontainer
version is that this is the faster solution. This also does not require Docker to be running. The downside to this solution is that the JDK requirement is bound to the Java version of the Zeebe engine.
Whenever this Java version changes, you'll either have to switch to the testcontainer version, or update your own JDK to match the Zeebe engine.
<dependency>
<groupId>io.camunda</groupId>
<artifactId>zeebe-process-test-extension</artifactId>
<version>X.Y.Z</version>
<scope>test</scope>
</dependency>
Camunda Spring SDK integration
You can add testing to your Camunda Spring SDK setup as follows.
Camunda Spring SDK embedded module
To integrate an in-memory test engine and use assertions, add the following Maven dependency:
<dependency>
<groupId>io.camunda</groupId>
<artifactId>spring-boot-starter-camunda-test</artifactId>
<version>X.Y.Z</version>
<scope>test</scope>
</dependency>
Camunda Spring SDK testcontainers module
To integrate a testcontainer engine and use assertions, add the following Maven dependency:
<dependency>
<groupId>io.camunda</groupId>
<artifactId>spring-boot-starter-camunda-test-testcontainer</artifactId>
<version>X.Y.Z</version>
<scope>test</scope>
</dependency>
Usage
Add the @ZeebeSpringTest
annotation to your Spring Boot test case to make the engine and a client available in your test case.
@SpringBootTest
@ZeebeSpringTest
public class TestMyProcess {
private ZeebeTestEngine engine;
private ZeebeClient client;
You can find an example test case in the Camunda Community Hub at Camunda 8 examples.
Do not use zeebeTestEngine.waitForBusyState(...)
to wait for a timer. This will not work as this is also triggered by an incoming job activation.
Annotation
Annotate your test class with the @ZeebeProcessTest
annotation. This annotation will do a couple of things:
- It will manage the lifecycle of the testcontainer/embedded Zeebe engine.
- It will create a client which can be used to interact with the engine.
- It will (optionally) inject three fields in your test class:
ZeebeTestEngine
- This is the engine that will run your process. It will provide some basic functionality to help you write your tests, such as waiting for an idle state and increasing the time.ZeebeClient
- This is the client that allows you to send commands to the engine, such as starting a process instance. The interface of this client is identical to the interface you use to connect to a real Zeebe engine.RecordStream
- This gives you access to all the records processed by the engine. Assertions use the records for verifying expectations. This grants you the freedom to create your own assertions.
// When using the embedded Zeebe engine (Java 21+)
import io.camunda.zeebe.process.test.extension.ZeebeProcessTest;
// When using testcontainers (Java 8+)
import io.camunda.zeebe.process.test.extension.testcontainer.ZeebeProcessTest;
@ZeebeProcessTest
class DeploymentAssertTest {
private ZeebeTestEngine engine;
private ZeebeClient client;
private RecordStream recordStream;
}
Switching between testcontainers and embedded
Switching between testcontainers and embedded requires just two steps:
Switch to the relevant dependency.
- Testcontainers:
zeebe-process-test-extension-testcontainer
- Embedded:
zeebe-process-test-extension
- Testcontainers:
Change the import of
@ZeebeProcessTest
.- Testcontainers:
import io.camunda.zeebe.process.test.extension.testcontainer.ZeebeProcessTest;
- Embedded:
import io.camunda.zeebe.process.test.extension.ZeebeProcessTest;
- Testcontainers:
Assertions
Start an assertion using the following entry points:
Deployment assertions
DeploymentEvent event = client.newDeployResourceCommand()
.addResourceFromClasspath("my-process.bpmn")
.send()
.join();
DeploymentAssert assertions = BpmnAssert.assertThat(event);
Process instance assertions
Started by manually sending an event:
ProcessInstanceEvent event = client.newCreateInstanceCommand()
.bpmnProcessId("<processId>")
.latestVersion()
.send()
.join();
ProcessInstanceAssert assertions = BpmnAssert.assertThat(event);
ProcessInstanceResult event = client.newCreateInstanceCommand()
.bpmnProcessId("<processId>")
.latestVersion()
.withResult()
.send()
.join();
ProcessInstanceAssert assertions = BpmnAssert.assertThat(event);
Started by a timer:
Optional<InspectedProcessInstance> firstProcessInstance = InspectionUtility.findProcessEvents()
.triggeredByTimer(ProcessPackTimerStartEvent.TIMER_ID)
.findFirstProcessInstance();
ProcessInstanceAssert assertions = BpmnAssert.assertThat(firstProcessInstance.get());
Started by a call activity:
Optional<InspectedProcessInstance> firstProcessInstance = InspectionUtility.findProcessInstances()
.withParentProcessInstanceKey(<key>)
.withBpmnProcessId("<called process id>")
.findFirstProcessInstance();
ProcessInstanceAssert assertions = BpmnAssert.assertThat(firstProcessInstance.get());
Job assertions
ActivateJobsResponse response = client.newActivateJobsCommand()
.jobType("<jobType>")
.maxJobsToActivate(1)
.send()
.join();
ActivatedJob activatedJob = response.getJobs().get(0);
JobAssert assertions = BpmnAssert.assertThat(activatedJob);
Message assertions
PublishMessageResponse response = client
.newPublishMessageCommand()
.messageName("<messageName>")
.correlationKey("<correlationKey>")
.send()
.join();
MessageAssert assertions = BpmnAssert.assertThat(response);
Incident assertions
Via a process instance
ProcessInstanceEvent event = client.newCreateInstanceCommand()
.bpmnProcessId("<processId>")
.latestVersion()
.send()
.join();
IncidentAssert assertions = BpmnAssert.assertThat(event)
.extractingLatestIncident();
Via a job:
ActivateJobsResponse response = client.newActivateJobsCommand()
.jobType("<jobType>")
.maxJobsToActivate(1)
.send()
.join();
ActivatedJob activatedJob = response.getJobs().get(0);
IncidentAssert assertions = BpmnAssert.assertThat(activatedJob)
.extractingLatestIncident();
Waiting for idle state
Waiting for idle state is a new feature. When the engine is detected to be idle, it will wait 30ms before checking again. If it is still idle at that stage, it is considered to be in an idle state.
It is unknown if the 30ms delay is sufficient. Using it could result in flaky tests!
Any feedback about the wait for idle state is highly appreciated. Let us know if the delay should be higher or configurable. Leave your feedback on our GitHub page.
engine.waitForIdleState(timeout)
will cause your test to stop executing until the engine has
reached an idle state. If the engine does not reach an idle state within the specified timeout, a
TimeoutException
will be thrown.
We have defined an idle state as a state in which the engine makes no progress and is waiting for new commands or events to trigger. Once the engine has detected it has become idle, it will wait for a delay (30ms) and check if it is still idle. If this is the case, it is considered to be in idle state and continue your test.
Wait for busy state
engine.waitForBusyState(timeout)
will cause your test to stop executing until the engine has
reached a busy state. If the engine does not reach a busy state within the specified timeout, a
TimeoutException
is thrown.
We consider the engine to have reached a busy state when any new record/command is processed since we've started waiting.
Waiting for a busy state is useful in scenarios where you're expecting the engine to start doing something, without explicitly triggering it yourself. An example of this would be a process with a timer event. We can increase the time of the engine, but we cannot trigger the timer explicitly. Because of this, we should wait for a busy state after increasing the engine time.
Examples
For example tests, refer to GitHub.