Tutorial: Trigger a workflow from a rule
An equipment class rule triggers a BPMN workflow whenever a data source publishes a value that meets a specified threshold.
Imagine a scenario when an oven must be preheated every time a new order number is published to an MQTT edge device.
You could automate this workflow with a rule that listens to messages published and evaluates a condition.
If the condition evaluates to true
, the rule triggers a BPMN workflow to preheat the oven.
--- title: Rules trigger workflows from data-source changes --- flowchart LR A(Property changed?) -->|yes| B{"rule evaluates to true?"} B -->|no| C(do nothing) B -->|"yes (optional: pass variables)"| D(Run BPMN workflow)
The broad procedure to create a rule is as follows:
- In the Rhize UI or through GraphQL, create models for the data source and its associated unit of measure, equipment, and equipment class.
- In the Rhize UI, write a BPMN workflow that is triggered when this data source changes and executes some business logic.
- In the equipment class, create a rule that triggers the workflow.
The following sections describe how to do these steps in more detail.
Prerequisites
Before you start, ensure you have the following:
- Access your Rhize customer environment
- The Agent configured to listen for your data-source ID
Set up: configure equipment and workflows
The setup involves modeling the objects associated with the rule.
- Data source
- Data source topic
- Unit of measure
- BPMN
- Equipment class with bound properties
Once you have these, you can create a rule and associate it with an actual equipment item.
Create a data source
- From the main menu, navigate to Master Data > Data Sources.
- Create a new data source. The label (ID) must match the one specified in the configuration file for the
libre-agent
microservice. - From the General tab, add a draft data source version.
- Select
MQTT
as the data-source protocol. - Optionally, enter a connection string, such as
mqtt://<REMOTE_HOST>:1883
, that matches the one specified in the configuration file for thelibre-agent
microservice. - Save the data source version to create it.
data:image/s3,"s3://crabby-images/df0da/df0da82117d47dddde01e02e10ea01f51b136919" alt="A new data source and version created in the UI."
Create a data source topic
- Navigate to the Topics tab.
- Add a new property (that is, a topic).
- Select
STRING
for the property data type (this assumes an order number is a string such asOrder1
). - Select your preferred deduplication key. The default option,
Message Value
, is most appropriate for this scenario. - For label, enter the exact topic name as it appears in the data source. Use a slash to access nested topics. For this example, all new order numbers are published to
Oven/OrderNumber
. - Confirm by clicking the green tick icon.
- Navigate to the General tab and change the version state to active.
data:image/s3,"s3://crabby-images/0099e/0099e91414574fdc324040913fb7800fac9b735b" alt="A new data source topic created in the UI."
Create a unit of measure
- From the Main Menu, navigate to Master Data > Units of Measure.
- Add a new unit of measure.
- Enter
Order Number
for the unit name. - Select
STRING
for the data type.
data:image/s3,"s3://crabby-images/e262d/e262dec0130f15468e546d8bf008f4342cab8822" alt="A new unit of measure created in the UI."
Creating a BPMN workflow
A rule must trigger a BPMN workflow. Before setting up a rule, create its workflow. For this example, this 3-node BPMN is enough:
- Navigate to Workflows > Process List.
- Import the BPMN.
- Save it.
- Set the version as active.
data:image/s3,"s3://crabby-images/4c849/4c849cb87ae7d16d5813ab4ee3c96ac20c91568b" alt="A BPMN created in the UI."
The BPMN has a Libre Jsonata Transform
task that contains an expression "Preheating oven for order number " & $.orderNumber"
.
The rule engine triggers this BPMN with a payload that includes the order number value, as follows:
{
"orderNumber": "Order1"
}
Create an equipment class with bound properties
Equipment class and version
- Navigate to Master Data > Equipment Class.
- Create a new equipment class from the sidebar. The label might be
Pizza Line
, for example. - From the General tab, Create a new Draft version.
data:image/s3,"s3://crabby-images/c15f1/c15f19a0b2b78773bb1a3eee87478d654cc3bcb9" alt="A new equipment class and version created in the UI."
Equipment class property
- From the properties tab, create a new property.
- For type, select
BOUND
. - For name, enter
orderNumber
. - For UoM, select the unit of measure created earlier (
Order Number
). - Confirm by clicking the green tick icon.
data:image/s3,"s3://crabby-images/507d9/507d9a2b7886b199b107b5c9a90f4ce785ea8bb4" alt="A new equipment class property created in the UI."
Create a rule
Add a rule to an existing equipment class
- From the Rules tab of an equipment class version, create a new rule.
- Enter
Run BPMN on Order Number
for the name and confirm. - Select
rules_example_bpmn
for the workflow specification. - Add
orderNumber
as a trigger property. - Add a trigger expression that evaluates to true or false.
true
.
It’s common to compare the new value with the previous.In this case, we can compare the new order number to the previous by adding OrderNumber.current.value != OrderNumber.previous.value
.
Note that the root of the object path must match the ID of the equipment class property we set up earlier and all evaluations are case-sensitive.
The entire information that becomes available to the rule engine looks like this:
{
orderNumber: {
current: {
bindingType: "BOUND",
description: "bound prop",
equipmentClassProperty: {
id: "EQCLASS1.10.orderNumber",
iid: "0x2a78",
label: "orderNumber"
},
equipmentVersion: {
equipment: {
id: "EQ1",
iid: "0x1b",
label: "EQ1"
},
id: "EQ1",
iid: "0x22",
version: "2"
},
id: "EQCLASS1.10.orderNumber",
label: "orderNumber",
messageKey: "ns=3;i=1003.1695170450000000000",
propertyType: "DefaultType",
serverPicoseconds: 0,
serverTimestamp: "2023-09-20T00:40:50.028Z",
sourcePicoseconds: 0,
sourceTimestamp: "2023-09-20T00:40:50Z",
value: "Order2",
valueUnitOfMeasure: {
dataType: "FLOAT",
id: "FLOAT",
iid: "0x28"
}
},
previous: {
bindingType: "BOUND",
description: "bound prop",
equipmentClassProperty: {
id: "EQCLASS1.10.orderNumber",
iid: "0x2a78",
label: "orderNumber"
},
equipmentVersion: {
equipment: {
id: "EQ1",
iid: "0x1b",
label: "EQ1"
},
id: "EQ1",
iid: "0x22",
version: "2"
},
id: "EQCLASS1.10.orderNumber",
label: "orderNumber",
messageKey: "ns=3;i=1003.1695170440000000000",
propertyType: "DefaultType",
serverPicoseconds: 0,
serverTimestamp: "2023-09-20T00:40:40.003Z",
sourcePicoseconds: 0,
sourceTimestamp: "2023-09-20T00:40:40Z",
value: "Order1",
valueUnitOfMeasure: {
dataType: "FLOAT",
id: "FLOAT",
iid: "0x28"
}
}
}
}
`OrderNumber.current.value != OrderNumber.previous.value`
True
<button class=“hextra-code-copy-btn hx-group/copybtn hx-transition-all active:hx-opacity-50 hx-bg-primary-700/5 hx-border hx-border-black/5 hx-text-gray-600 hover:hx-text-gray-900 hx-rounded-md hx-p-1.5 dark:hx-bg-primary-300/10 dark:hx-border-white/10 dark:hx-text-gray-400 dark:hover:hx-text-gray-50” title=“Copy code”
<div class="copy-icon group-[.copied]/copybtn:hx-hidden hx-pointer-events-none hx-h-4 hx-w-4"></div>
<div class="success-icon hx-hidden group-[.copied]/copybtn:hx-block hx-pointer-events-none hx-h-4 hx-w-4"></div>
The expression evaluates to false
, because the current
and previous
values differ.
Optionally, pass information to the BPMN by adding a payload message. The message is an object with multiple keys.
- Enter
orderNumber
for the field name. - Enter
orderNumber.current.value
for the JSON expression. - Confirm by clicking the green tick icon.
- Create.
- From the General tab, change the equipment class version state to active.
data:image/s3,"s3://crabby-images/1ef94/1ef9411d32c07900c02f578e6286b7319b60bfef" alt="Creating an equipment class rule in the UI."
Associate an equipment with a bound property
The final steps to setting up a rule are to:
- Create a new equipment version.
- Link it to a data source.
- Set up bound properties.
Create an equipment and version
- From the Main Menu, navigate to Master Data > Equipment.
- Select a piece of equipment. If none, create one called
Line 1
. - From the General tab, Create.
- Link the version to the equipment class you created earlier (
Pizza Line
). - Save the version to create it.
data:image/s3,"s3://crabby-images/7c6e7/7c6e780c668c1dc8f3d0d9db78e8fb84172bb52e" alt="A new equipment class and version created in the UI."
Link a data source
- From the Data Sources tab, link the equipment version to the data source you created in the previous section.
data:image/s3,"s3://crabby-images/867a9/867a9f3588147929b4d27a833faf334e5c921bde" alt="An equipment linked to a data source in the UI."
Set up the bound property
- From the Properties tab, find a property that you want this equipment to inherit and select the binding icon.
- If you chose the property
orderNumber
, add the topicOven/OrderNumber
you added previously.
data:image/s3,"s3://crabby-images/7d2a5/7d2a5f54fdfbaf08302c54e83e8efaacd558a9fd" alt="An equipment property bound to a data source topic in the UI."
Test the binding and the rule
Send a message to test that the value of the property orderNumber
of the equipment Line 1
is bound to the topic Oven/OrderNumber
.
Test using an MQTT client
For example, using MQTT Explorer:
- Open MQTT Explorer and connect to the broker.
The microservice Libre Agent (libre-agent
) should immediately publish a message to indicate the data source topic Oven/OrderNumber
has been set up successfully.
data:image/s3,"s3://crabby-images/68b67/68b67b0271c763ff22a6ddfde6fd797de8ceae68" alt="The Libre Agent has connected to the data source."
- Publish the string
Order1
to the topicOven/OrderNumber
.
data:image/s3,"s3://crabby-images/cb050/cb0503599c34752ae9d1f5909f3f56ae31d50b0e" alt="A new order number was published to the data source."
If the message has been received,
a new topic, Oven
, appears with its subtopic OrderNumber
.
If there is an equipment property bound to this topic,
a topic called MQTT/<DATA_SOURCE_NAME>/ValueChanged
also appears.
In addition, the published value should show in the column Expression
of the equipment property orderNumber
.
data:image/s3,"s3://crabby-images/a758f/a758ff297a0add6d9e65cfc2c9f1080a615d9b9b" alt="The bound property assumes the last value published to the data source."
Core
will show up containing a subtopic called RuleTriggered
to indicate that the rule has indeed been triggered.data:image/s3,"s3://crabby-images/d4cf3/d4cf322ce633daff1aa3d9a607ae164ca6ded294" alt="The rule engine has published a message to indicate that the equipment class rule has indeed been triggered."
Confirm in execution in Tempo
To confirm the intended BPMN was executed, navigate to Grafana (Tempo) and look for a trace containing the expected BPMN ID.
data:image/s3,"s3://crabby-images/86a78/86a788eaa90f1b79b935fba41b11b3477e081a20" alt="Grafana shows a recent trace with the id of the target BPMN."
Video example
- 🎥 Trigger BPMN. This video provides an example of creating a rule based on values for an OPC UA server in a baking process.