In the first article in this series, Designing an event-driven business process at scale: A health management example, Part 1, we began by defining the business use case and data model for a concrete example from the health management industry. We then began implementing the example in jBPM (an open source business automation suite) by creating our trigger process.
Now, in the second article in this series, we will focus on creating the Task subprocess and its many components. In our case, these are:
- The Expired? gate
- The Suppressed? gate
- The human task
- The Reminder subprocess
- The "What type of close?" gate
- The Hard Close embedded subprocess
- The Escalation subprocess
The Task subprocess
Now you can create the Task subprocess with the properties shown in Figure 1.
data:image/s3,"s3://crabby-images/83497/83497099c030587de0c9933eca7e3f610c78bf0d" alt="PHM Processes A490 1 jBPM Diagram properties section, Process subsection"
Figure 1: Create the Task subprocess.">
The Task subprocess needs the variables shown in Figure 2 to be defined.
data:image/s3,"s3://crabby-images/4fd71/4fd71af15089b048d3fed96cae8c357f39a9c617" alt="PHM Processes 0001 jBPM Process Data section, defining process variables"
Figure 2: Define the variables for your Task subprocess.">
Note: The type com.jbpm.document.Document
of the variable sSupplementalData
is available out of the box.
Now, draw the diagram for the Task subprocess, as shown in Figure 3.
data:image/s3,"s3://crabby-images/70b6e/70b6ee52b066c9e2c2672baa2194e12d37d575f9" alt="2020-03-01_09-09-44 Create your Task subprocess diagram"
Figure 3: Create your Task subprocess diagram.">
Once the process variables are initialized in a script task, a user task must be completed. A reminder is set for the completion of the user task and an escalation is defined. Both the reminder and the escalation are implemented with timers that lead to subprocesses that you need to implement. See below for a note concerning timers from a scalability perspective.
Note: We implemented the task suppression requirement using a timer as well.
The hard close requirement is realized as an embedded subprocess that is simply catching a signal. Following the requirements, the escalation occurs due to a timer on this subprocess.
The Expired? gate
The Expired? exclusive gate should have two branches configured using the expirationDate
attribute of the Task class, as shown in Figure 4.
data:image/s3,"s3://crabby-images/2f3ef/2f3efcbaf7655a79fca66b26cf3fb45e1d8c0701" alt="PHM Processes - Fig 211 jBPM diagram showing the expired gate in place"
Figure 4: Checking if the task is expired.">
To configure the Yes branch, set up the condition expressions shown in Figure 5.
data:image/s3,"s3://crabby-images/a2498/a249843b9803c74feb530a46b1ed7fb60869ad71" alt="jBPM settings for what to do if the task is expired."
Figure 5: Defining the behavior if your task is expired.">
The Suppressed? gate
The Suppressed? exclusive gate's two branches should be configured using the suppressed
boolean attribute of the Task class, as shown in Figure 6.
data:image/s3,"s3://crabby-images/b82b7/b82b72d6ebe32bed571e4b2284f777f74d7910ab" alt="2020-02-23_10-02-09 Defining behavior depending on whether or not your task needs to be suppressed."
Figure 6: Defining behavior depending on whether or not your task needs to be suppressed.">
Figure 7 shows the configuration for the Yes branch.
data:image/s3,"s3://crabby-images/67952/67952a20d21d2e79455f7c9688061721b9dd1715" alt="PHM Processes - Fig 213 jBPM configuration for suppressing a task"
Figure 7: Entering the Yes branch.">
The suppression of the task is accomplished by a timer. The timer causes the process to wait for the period specified in the Task object's suppressionPeriod
attribute, as shown in Figure 8.
data:image/s3,"s3://crabby-images/41bf7/41bf7c8763e5926c1ff1988399d68f0122b28aa6" alt="Defining the suppression timer"
Figure 8: Defining the suppression timer.">
The human task
Now, let's focus on the human task, as shown in Figure 9.
data:image/s3,"s3://crabby-images/f680a/f680afe62e8a83da123da9dbaf85b39283d14256" alt="2020-03-01_09-12-25 Defining the human task."
Figure 9: Defining the human task.">
Configure the human task using the settings shown in Figure 10.
data:image/s3,"s3://crabby-images/bb514/bb5141b22b6a01de4a54ca2f6a658ce151875d5a" alt="PHM Processes Task 1 jBPM showing task actor setup"
Figure 10: Implementing the human (actor) task.">
The task actor is resolved from the task actor assignment process variable.
Next, there are a couple of abilities you will want: To capture text as well as supplemental uploaded documentation and to complete the task with a Not Applicable or Not Available response. These requirements can be satisfied by configuring the task parameters as you see in Figure 11.
data:image/s3,"s3://crabby-images/5f674/5f674be6e916fb216cc9106b880df2c86e6884f3" alt="JBPM Data Outputs and Assignments section filled in for this task."
Figure 11: Defining the task's parameters to capture the types of data you need.">
You also need to satisfy the requirement of sending a periodic reminder to the task actor. There is a built-in notification capability in jBPM that allows sending email to notify groups and individuals to complete a task. For example, you could configure notifications for the Task activity as you see in Figure 12.
data:image/s3,"s3://crabby-images/ecd28/ecd283c2e084c0206116848e3e46ddc8b7962e03" alt="PHM Processes 0003 jBPM Notification configuration for task state type not completed."
In many cases, the built-in notification capability may be adequate. However, there are cases where one would need more control over the notification process. The advantage of using timers for this purpose allows you to design as complex a reminder process as you want. In this example, you will implement timer-based reminders.Configure the timer on the task border, which will trigger the first reminder after the period of time defined in the reminderInitiation
attribute of the Task
object as shown in Figure 13.
data:image/s3,"s3://crabby-images/9feb4/9feb41fe976abc20f20330a812e6f3acb4da8122" alt="PHM Processes - Fig 100 jBPM implementing the first task reminder"
The Reminder subprocess
Now you should configure the subprocess Reminder
, as shown in Figure 14. Notice that both the properties "Independent" and "Abort Parent" must be left unchecked while "Wait for Completion" is checked. If you don't configure it in this way the Reminder subprocess will not stop as it should.
data:image/s3,"s3://crabby-images/bda9a/bda9a7dc67d3f296eddad246e34883f2b5e3b1ee" alt="2020-03-01_07-42-47 Creating the Reminder subprocess."
Set the subprocess parameters to be the Reminder
and Task
objects, as shown in Figure 15.
data:image/s3,"s3://crabby-images/1e8db/1e8dbacf8201f3578ed836f15d1dc47346b63178" alt="Configuring the subprocess"
After completion of the task, a signal should be sent to stop the reminder subprocess, as shown in Figure 16.
data:image/s3,"s3://crabby-images/71bbe/71bbec7bef41a1faf9102eadf8011f2f632ab954" alt="2020-03-01_08-18-38.png Creating the stop signal."
The "What type of close?" gate
An exclusive gate decides if the task close state should be soft or hard. The soft close branch of the gate looks like what you see in Figure 17.
data:image/s3,"s3://crabby-images/e79ed/e79ed7b381d89fed2db996fcaeffb0fc41dac948" alt="PHM Processes - Fig 4 jBPM setting up for the value SOFT"
And the hard-close branch is shown in Figure 18.
data:image/s3,"s3://crabby-images/389b7/389b75761045b65e55dbcbc0a20a5a337dfc9f9a" alt="PHM Processes - Fig 5 jBPM setting up for the value HARD."
The Hard Close embedded subprocess
If the task close is hard, the process must wait for a confirmation signal coming from an external system. You have to configure this in an embedded subprocess because of the escalation requirement.
This subprocess is simple. There is just an intermediate signal catch, as shown in Figure 19.
data:image/s3,"s3://crabby-images/7ddb1/7ddb10af74600a71a1d0c2b45784839578e59371" alt="PHM Processes - hard close signal jBPM catching the hard close signal."
You also need to implement an escalation subprocess if the task is not closed in a given time. Again, jBPM has a built-in capability to reassign a task if it is not completed in a timely fashion. For example, you could configure how to reassign the task as you see in Figure 20.
data:image/s3,"s3://crabby-images/2edc2/2edc2d875589ba0e456f924ddd1040d2c98bf856" alt="PHM Processes 00004 jBPM's Reassignment section set to escalate a task that was not completed in time."
However, you need to base the reassignment after receiving confirmation from an external system that the task was closed. Moreover, the period must be configurable as a variable, because this value depends on the specific task. Because of these reasons, you need to implement the escalation as a timer-triggered subprocess.
A border timer determines when an escalation is needed based on the value of the escalationTimer
attribute of the Task
object, as shown in Figure 21.
data:image/s3,"s3://crabby-images/5fb89/5fb89b12a6793a0fcb8a061bac4b50174654d88a" alt="PHM Processes - escalation timer jBPM setting the task to fire once when it's time."
There is yet another way to implement SLA escalations. The ProcessEventListener
interface has two methods that capture the event of an SLA violation and can be implemented with custom code specifying what to do in such an event, as you can see in this example implementation:
/** * @param event */ public void beforeSLAViolated(SLAViolatedEvent event) { System.out.println( "Process <<"+ event.getProcessInstance().getProcessName()+ ">>-<"+ event.getProcessInstance().getId()+ "> ->SLA <<"+ event.getNodeInstance().getNodeName()+">>-<"+ event.getNodeInstance().getNodeId()+">-<"+ event.getNodeInstance().getId()+ "> SLA is about to be violated." ); } /** * @param event */ public void afterSLAViolated(SLAViolatedEvent event) { System.out.println( "Process <<"+ event.getProcessInstance().getProcessName()+ ">>-<"+ event.getProcessInstance().getId()+ "> ->SLA <<"+ event.getNodeInstance().getNodeName()+">>-<"+ event.getNodeInstance().getNodeId()+">-<"+ event.getNodeInstance().getId()+ "> SLA has been violated." ); }
The Escalation subprocess
Now, configure the Escalation subprocess, as shown in Figure 22.
data:image/s3,"s3://crabby-images/7a688/7a6881c4e1c396b7932647e4386e9dafa95c334b" alt="2020-03-01_07-44-09 Creating the Escalation subprocess."
Then define the parameters of the subprocess. You need to pass the Task
and the TaskActorAssignment
objects, as you can see in Figure 23.
data:image/s3,"s3://crabby-images/2d5fe/2d5fe61dd06de1f1eb0201e88a617751056651ef" alt="PHM Processes call Escalate 2"
Conclusion
So far in this series, we have defined the business use case for our health management event-driven business process, and have created our data model, trigger process, and task subprocess with all of its components. In the next installment, we will complete our example configuration.