Skip to content

Latest commit

 

History

History
375 lines (291 loc) · 16.5 KB

README.md

File metadata and controls

375 lines (291 loc) · 16.5 KB

processhub-bpmn-manipulation

Table of contents

Introduction

Processhub API to manipulate BPMN processes. This API provides custom operators for tailoring and process composition by extending the camunda-bpmn-model functions to offer the required operators. It also offers a BPMNt implementation, which can be viewed as a "process tailoring log".

For every operator, the BPMN models will be represented as an instance of the class BpmnModelInstance from Camunda's API or its extensions (TailorableBpmnModelInstance and BpmntModelInstance).

Installation

To install, simply add the following to your project's pom.xml:

<repositories>
	...
	<repository>
		<id>jitpack.io</id>
		<url>https://jitpack.io</url>
	</repository>
	...
</repositories>

<dependencies>
	...
	<dependency>
		<groupId>com.github.owse</groupId>
		<artifactId>processhub-bpmn-manipulation</artifactId>
		<version>-SNAPSHOT</version>
	</dependency>
	...
</dependencies>

The '-SNAPSHOT' version will fetch the latest commit in the master branch. The version can also be a version tag (currently not available), or a short commit hash (eg. 9a6e0eb1f3).

Composition Operators

The composition operators are functions that take two or more processes as input and generate a single new process. The resulting process represents the composition of the processes provided. There are different strategies to perform a processes composition. Each available operator is described below.

1. Serial Composition

This operator takes a list of two or more processes and perform a serial composition. It "glues" the end of a process to the start of the next one. As it relies on BPMN start and end events to identify where to connect the processes, the operator requires that every process have exactly only one start event and one end event.

A simple example is shown in Figure 01.

alt text Figure 01 -Simple serial composition scenario

The above scenario could be achieved by the execution of the following code:

BpmnModelInstance modelInstance1 = Bpmn.readModelFromFile(new File("simple_diagram1.bpmn"));
BpmnModelInstance modelInstance2 = Bpmn.readModelFromFile(new File("simple_diagram2.bpmn"));
List<BpmnModelInstance> modelsToJoin = new ArrayList<BpmnModelInstance>();

modelsToJoin.add(modelInstance1);
modelsToJoin.add(modelInstance2);

BpmnModelComposer bpmnModelComposer = new BpmnModelComposer();
BpmnModelInstance resultModel1 = bpmnModelComposer.joinModelsInSeries(modelsToJoin);

// The same result can be achieved using the variable arguments signature, as shown below
BpmnModelInstance resultModel2 = bpmnModelComposer.joinModelsInSeries(modelInstance1, modelInstance2);
 

2. Parallel Composition

This operator takes a list of two or more processes and perform a parallel composition. It adds (if necessary) parallel gateways after the start event and before the end event of the first process and connects the other processes ends to these gateways. Possible scenarios are showed in Figures 02, 03 and 04.

alt text Figure 02 - Simple parallel composition scenario

alt text Figure 03 - Parallel composition scenario with gateways already present in the process

alt text Figure 04 - Parallel composition scenario with only one gateway already present in the process

The above scenarios can be achieved by the following code:

BpmnModelInstance modelInstance1 = Bpmn.readModelFromFile(new File("simple_diagram1.bpmn"));
BpmnModelInstance modelInstance2 = Bpmn.readModelFromFile(new File("simple_diagram2.bpmn"));
List<BpmnModelInstance> modelsToJoin = new ArrayList<BpmnModelInstance>();

modelsToJoin.add(modelInstance1);
modelsToJoin.add(modelInstance2);

BpmnModelComposer bpmnModelComposer = new BpmnModelComposer();
BpmnModelInstance resultModel1 = bpmnModelComposer.joinModelsInParallel(modelsToJoin);

// The same result can be achieved using the variable arguments signature, as shown below
BpmnModelInstance resultModel2 = bpmnModelComposer.joinModelsInParallel(modelInstance1, modelInstance2);

Restrictions

Currently, the operations rely on the assumption that the processes to be joined have exactly one start event and one end events. It's also assumed that the processes are valid, following Camunda's criteria. The operators work only if both conditions are met.

Tailoring Operators

Tailoring operators are functions that modify an existing process in order to adapt it to a new context. The operators will be supported in an interface and implementation that extends BpmnModelInstance with tailoring operations. The new interface is called TailorableModelInstance. Its operators are divided in two groups, high-level operations and low-level operations.

High-level Operators

The high-level operators are the main tailoring operators. What distinguishes high- from low-level operators is that the high-level ones guarantee that if the base model is valid, the resulting model after an operation is executed is also valid, while the resulting model of a low-level operation might be invalid. The high-level operators are described below.

1. Rename

Rename a flow element.

TailorableBpmnModelInstance modelInstance = TailorableBpmn.readModelFromFile(new File("simple_diagram.bpmn"));
// Pass element id and new name
modelInstance.rename(flowElement.getId(), "New Name");

2. Delete

Delete a flow node or a set of flow nodes from the model. Currently, the nodes before and after the set to be deleted are connected with a single sequence flow.

Restrictions:
  • Start or end events cannot be deleted
  • A gateway cannot be deleted singly
  • Process fragments containing incomplete flow branches cannot be deleted
TailorableBpmnModelInstance modelInstance = TailorableBpmn.readModelFromFile(new File("simple_diagram.bpmn"));
// Delete a single node
modelInstance.delete(flowNode);
// or delete a range of elements from startingNode to endingNode
modelInstance.delete(startingNode, endingNode);
// It's also possible to use the nodes ids' as parameters:
String flowNodeId = flowNode.getId();
modelInstance.delete(flowNodeId);

3. Replace

Replace a flow node or a set of flow nodes from the model. The targets can be replaced by a flow node or a process.

Restrictions:
  • Start or end events can only be placed by other start and end events, respectively
  • Gateways cannot be replaced
  • Process fragments containing incomplete flow branches cannot be replaced
  • A replacing fragment must be a BpmnModelInstance, a Process or a SubProcess
TailorableBpmnModelInstance modelInstance = TailorableBpmn.readModelFromFile(new File("simple_diagram.bpmn"));

// Replace a node with another node
modelInstance.replace(targetNode, replacingNode);

// Replace a node with a fragment
modelInstance.replace(targetNode, replacingFragment);

// Replace a fragment with a node
modelInstance.replace(startingNode, endingNode, replacingNode);

// Replace a fragment with another fragment
modelInstance.replace(startingNode, endingNode, replacingFragment);

4. Move

Move a flow node or a set of flow nodes to another part of the model, marked by the nodes newPositionAfterOf and newPositionBeforeOf.

Restrictions:
  • If only the parameter newPositionAfterOf is provided, it cannot link to a diverging gateway or end event
  • If only the parameter newPositionBeforeOf is provided, it cannot link to a converging gateway or start event
  • If both parameters newPositionAfterOf and newPositionBeforeOf are set, they must be consecutive nodes
  • Start or end events cannot be moved
  • Gateways cannot be moved singly
  • Process fragments containing incomplete flow branches cannot be moved
TailorableBpmnModelInstance modelInstance = TailorableBpmn.readModelFromFile(new File("simple_diagram.bpmn"));
// Move a node
modelInstance.move(targetNode, newPositionAfterOf, newPositionBeforeOf);
// Or
modelInstance.move(targetNode, newPositionAfterOf, null);
// Or
modelInstance.move(targetNode, null, newPositionBeforeOf);

// Move a set of nodes
modelInstance.move(targetStartingNode, targetEndingNode, newPositionAfterOf, newPositionBeforeOf);
// Or
modelInstance.move(targetStartingNode, targetEndingNode, newPositionAfterOf, null);
// Or
modelInstance.move(targetStartingNode, targetEndingNode, null, newPositionBeforeOf);

5. Parallelize

Parallelize the execution of elements from a process fragment that has been defined as a sequential flow

Restrictions:
  • The target fragment cannot contain start or end events
  • The target fragment cannot include gateways
TailorableBpmnModelInstance modelInstance = TailorableBpmn.readModelFromFile(new File("simple_diagram.bpmn"));
// Parallelize nodes between targetStartingNode and targetEndingNode (both included)
modelInstance.move(targetStartingNode, targetEndingNode);

6. Split

Split a single task into a subprocess that details its procedure. Can be seen as task replaced by a subprocess

Restrictions:
  • Only single tasks can be splitted
  • This operation can be defined only by subprocesses
TailorableBpmnModelInstance modelInstance = TailorableBpmn.readModelFromFile(new File("simple_diagram.bpmn"));
// Split the target task in a subprocess (newSubProcessModel is a process inside another BpmnModelInstance)
modelInstance.split(targetNode, newSubProcessModel);

7. Insert

Insert a flow node or a set of flow nodes between afterOf and beforeOf nodes. If afterOf and beforeOf are consecutive nodes, the new nodes are inserted in series. Else, the new nodes are inserted in parallel to the nodes between afterOf and beforeOf

Restrictions:
  • If only the parameter afterOf is provided, it cannot link to a diverging gateway or end event
  • If only the parameter beforeOf is provided, it cannot link to a converging gateway or start event
  • Gateways cannot be inserted singly;
  • The fragment between the parameters afterOf and beforeOf cannot contain incomplete flow branches
  • Process fragments containing incomplete flow branches cannot be inserted.
TailorableBpmnModelInstance modelInstance = TailorableBpmn.readModelFromFile(new File("simple_diagram.bpmn"));
// Insert a node
modelInstance.insert(afterOf, beforeOf, flowNodeToInsert);
// Or
modelInstance.insert(afterOf, null, flowNodeToInsert);
// Or
modelInstance.insert(null, beforeOf, flowNodeToInsert);

// Insert a set of nodes (fragmentToInsert can be a BpmnModelInstance or a Process)
modelInstance.insert(afterOf, beforeOf, fragmentToInsert);
// Or
modelInstance.insert(afterOf, null, fragmentToInsert);
// Or
modelInstance.insert(null, beforeOf, fragmentToInsert);

8. Conditional Insert

Insert a flow node or a set of flow nodes between afterOf and beforeOf nodes. The included nodes are only executed with the given condition

Restrictions:
  • Gateways cannot be inserted singly;
  • The fragment between the parameters afterOf and beforeOf cannot contain incomplete flow branches
  • Process fragments containing incomplete flow branches cannot be inserted.
TailorableBpmnModelInstance modelInstance = TailorableBpmn.readModelFromFile(new File("simple_diagram.bpmn"));
// Insert a node
void conditionalInsert(afterOf, beforeOf, flowNodeToInsert, condition, inLoop);

// Insert a set of nodes (fragmentToInsert can be a BpmnModelInstance or a Process)
void conditionalInsert(afterOf, beforeOf, flowNodeToInsert, condition, inLoop);

Low-level Operators

Low-level operators are elementary functions, and aside from 'extend', should be avoided, as they can render the model invalid. Whenever possible, use high-level operators instead.

1. Extend

The 'extend' operation is a special one. Its only function is to instance a BpmntModelInstance with its content copied from the TailorableBpmnModelInstance that invoked it, and initialize the BpmntOperators list. For more information, see the BPMNt section

TailorableBpmnModelInstance modelInstance = TailorableBpmn.readModelFromFile(new File("simple_diagram.bpmn"));

BpmntModelInstance bpmntModelInstance = modelInstance.extend();

2. Suppress

Remove a flow element from the model.

TailorableBpmnModelInstance modelInstance = TailorableBpmn.readModelFromFile(new File("simple_diagram.bpmn"));

modelInstance.suppress(flowElement);
// It's also possible to suppress multiple flow elements at once
Collection<FlowElement> flowElements = someLoadElementsMethod();
modelInstance.suppress(flowElements);

3. Contribute

Insert a flow element in the model.

TailorableBpmnModelInstance modelInstance = TailorableBpmn.readModelFromFile(new File("simple_diagram.bpmn"));

modelInstance.contribute(flowElement);
// It's also possible to add the new element to a specific parent
ModelElementInstance parent = someLoadParentMethod();
modelInstance.contribute(parent, flowElement);

4. Modify

Add or modify a property of a flow element.

TailorableBpmnModelInstance modelInstance = TailorableBpmn.readModelFromFile(new File("simple_diagram.bpmn"));
String property = "Property name";
String value = "Property value";

modelInstance.modify(flowElement, property, value);

BPMNt

The BPMNt is an extension of BPMN, with support to tailoring operations, except for the Extend operation. The main difference of BPMNt from the TailorableBPMN is that it contains an operation log that registers every operation executed. Possible uses are:

  • To verify the exact sequence of operations executed to generate the resulting model;
  • In case that the base process changes, to verify if any breaking changes were made;
  • To replay the operations made to the base process.

Basic usage example:

TailorableBpmnModelInstance modelInstance = TailorableBpmn.readModelFromFile(new File("simple_diagram.bpmn"));

BpmntModelInstance bpmntModelInstance = modelInstance.extend();
bpmntModelInstance.move(targetNode, newPositionAfterOf, newPositionBeforeOf);
bpmntModelInstance.replace(startingNode, endingNode, replacingNode);

BpmnModelInstance bpmntModel = bpmntModelInstance.getBpmntModel();
System.out.println(Bpmn.convertToString(bpmntModel));

The code above outputs a BPMN model with the following structure:

<?xml version="1.0" encoding="UTF-8" standalone="no"?>
<definitions xmlns:camunda="http://activiti.org/bpmn" xmlns:ns0="http://www.processhub.net" targetNamespace="http://www.omg.org/spec/BPMN/20100524/MODEL" xmlns="http://www.omg.org/spec/BPMN/20100524/MODEL">
  <process id="BPMNt_Process_1" name="BPMNt_Process_1">
    <extensionElements>
      <ns0:Extend baseProcessId="Process_1" newProcessId="BPMNt_Process_1" order="1"/>
      <ns0:MoveNode afterOfId="UserTask_0qz0mkl" beforeOfId="EndEvent_1i5whxc" nodeId="ScriptTask_1rseic1" order="2"/>
    </extensionElements>
    <subProcess id="ReplaceFragmentWithNode_3" name="ReplaceFragmentWithNode 3">
      <extensionElements>
        <ns0:ReplaceFragmentWithNode endingNodeId="UserTask_0qz0mkl" order="3" startingNodeId="ScriptTask_1rseic1"/>
      </extensionElements>
      <scriptTask id="ScriptTask_1rseic1" name="Task 1"/>
    </subProcess>
  </process>
</definitions>

If there is already a BPMN with a BPMNt associated, they can be loaded directly with:

BpmntModelInstance modelInstance = Bpmnt.readModelAndBpmntFromStrings(modelXml, bpmntXml);

Another possibility is to replay the BPMNt on the base process. The following code accomplishes this:

// Load the base model and the BPMNt
BpmntModelInstance modelInstance = Bpmnt.readModelAndBpmntFromStrings(baseModelXml, bpmntXml);

// Replay the BPMNt operations
modelInstance.executeOwnBpmnt();