@5minds/node-red-contrib-processcube 1.3.0-feature-92414d-m21x3yqt → 1.3.0
Sign up to get free protection for your applications and to get access to all the features.
- package/examples/Wait-for-UserTask-Sample.json +187 -0
- package/externaltask-input.html +14 -0
- package/externaltask-input.js +64 -62
- package/nodered/flows.json +179 -4
- package/package.json +2 -3
- package/processcube-engine-config.html +29 -5
- package/processcube-engine-config.js +8 -5
- package/processes/User-Task-Sample.bpmn +20 -17
- package/processes/Wait-For-Usertask.bpmn +74 -0
- package/processinstance-delete.html +58 -0
- package/processinstance-delete.js +55 -0
- package/wait-for-usertask.js +6 -5
@@ -0,0 +1,187 @@
|
|
1
|
+
[
|
2
|
+
{
|
3
|
+
"id": "d6c450fe98faa45f",
|
4
|
+
"type": "tab",
|
5
|
+
"label": "Wait for usertask",
|
6
|
+
"disabled": false,
|
7
|
+
"info": "",
|
8
|
+
"env": []
|
9
|
+
},
|
10
|
+
{
|
11
|
+
"id": "30b8f9c606a2bb83",
|
12
|
+
"type": "wait-for-usertask",
|
13
|
+
"z": "d6c450fe98faa45f",
|
14
|
+
"name": "",
|
15
|
+
"engine": "42e6796dddd9d4db",
|
16
|
+
"query": "payload",
|
17
|
+
"query_type": "msg",
|
18
|
+
"only_for_new": true,
|
19
|
+
"x": 590,
|
20
|
+
"y": 120,
|
21
|
+
"wires": [
|
22
|
+
[
|
23
|
+
"d9cb8e69e985d6df",
|
24
|
+
"a5c0ea907e64e759"
|
25
|
+
]
|
26
|
+
]
|
27
|
+
},
|
28
|
+
{
|
29
|
+
"id": "20fd37adf0fb6f39",
|
30
|
+
"type": "process-start",
|
31
|
+
"z": "d6c450fe98faa45f",
|
32
|
+
"name": "",
|
33
|
+
"engine": "42e6796dddd9d4db",
|
34
|
+
"processmodel": "Wait-For-Usertask_Process",
|
35
|
+
"startevent": "StartEvent_1",
|
36
|
+
"x": 250,
|
37
|
+
"y": 120,
|
38
|
+
"wires": [
|
39
|
+
[
|
40
|
+
"1e1e9c428172eae2",
|
41
|
+
"c65a4ba5461e7ee0"
|
42
|
+
]
|
43
|
+
]
|
44
|
+
},
|
45
|
+
{
|
46
|
+
"id": "e18451d69a169f5f",
|
47
|
+
"type": "inject",
|
48
|
+
"z": "d6c450fe98faa45f",
|
49
|
+
"name": "",
|
50
|
+
"props": [
|
51
|
+
{
|
52
|
+
"p": "payload"
|
53
|
+
}
|
54
|
+
],
|
55
|
+
"repeat": "",
|
56
|
+
"crontab": "",
|
57
|
+
"once": false,
|
58
|
+
"onceDelay": 0.1,
|
59
|
+
"topic": "",
|
60
|
+
"payload": "{}",
|
61
|
+
"payloadType": "json",
|
62
|
+
"x": 90,
|
63
|
+
"y": 120,
|
64
|
+
"wires": [
|
65
|
+
[
|
66
|
+
"20fd37adf0fb6f39"
|
67
|
+
]
|
68
|
+
]
|
69
|
+
},
|
70
|
+
{
|
71
|
+
"id": "d9cb8e69e985d6df",
|
72
|
+
"type": "delay",
|
73
|
+
"z": "d6c450fe98faa45f",
|
74
|
+
"name": "",
|
75
|
+
"pauseType": "delay",
|
76
|
+
"timeout": "5",
|
77
|
+
"timeoutUnits": "seconds",
|
78
|
+
"rate": "1",
|
79
|
+
"nbRateUnits": "1",
|
80
|
+
"rateUnits": "second",
|
81
|
+
"randomFirst": "1",
|
82
|
+
"randomLast": "5",
|
83
|
+
"randomUnits": "seconds",
|
84
|
+
"drop": false,
|
85
|
+
"allowrate": false,
|
86
|
+
"outputs": 1,
|
87
|
+
"x": 540,
|
88
|
+
"y": 200,
|
89
|
+
"wires": [
|
90
|
+
[
|
91
|
+
"acecfeef70b5c4ad"
|
92
|
+
]
|
93
|
+
]
|
94
|
+
},
|
95
|
+
{
|
96
|
+
"id": "44f4c6c64263d366",
|
97
|
+
"type": "usertask-output",
|
98
|
+
"z": "d6c450fe98faa45f",
|
99
|
+
"name": "",
|
100
|
+
"engine": "42e6796dddd9d4db",
|
101
|
+
"result": "payload",
|
102
|
+
"result_type": "msg",
|
103
|
+
"x": 880,
|
104
|
+
"y": 200,
|
105
|
+
"wires": [
|
106
|
+
[]
|
107
|
+
]
|
108
|
+
},
|
109
|
+
{
|
110
|
+
"id": "acecfeef70b5c4ad",
|
111
|
+
"type": "function",
|
112
|
+
"z": "d6c450fe98faa45f",
|
113
|
+
"name": "pre",
|
114
|
+
"func": "msg.payload = {};\nreturn msg;",
|
115
|
+
"outputs": 1,
|
116
|
+
"timeout": 0,
|
117
|
+
"noerr": 0,
|
118
|
+
"initialize": "",
|
119
|
+
"finalize": "",
|
120
|
+
"libs": [],
|
121
|
+
"x": 710,
|
122
|
+
"y": 200,
|
123
|
+
"wires": [
|
124
|
+
[
|
125
|
+
"44f4c6c64263d366"
|
126
|
+
]
|
127
|
+
]
|
128
|
+
},
|
129
|
+
{
|
130
|
+
"id": "1e1e9c428172eae2",
|
131
|
+
"type": "function",
|
132
|
+
"z": "d6c450fe98faa45f",
|
133
|
+
"name": "query",
|
134
|
+
"func": "msg.payload = {\n \"processInstanceId\": msg.payload.processInstanceId,\n \"flowNodeId\": \"make_it\"\n};\n\nreturn msg;",
|
135
|
+
"outputs": 1,
|
136
|
+
"timeout": 0,
|
137
|
+
"noerr": 0,
|
138
|
+
"initialize": "",
|
139
|
+
"finalize": "",
|
140
|
+
"libs": [],
|
141
|
+
"x": 410,
|
142
|
+
"y": 120,
|
143
|
+
"wires": [
|
144
|
+
[
|
145
|
+
"30b8f9c606a2bb83"
|
146
|
+
]
|
147
|
+
]
|
148
|
+
},
|
149
|
+
{
|
150
|
+
"id": "c65a4ba5461e7ee0",
|
151
|
+
"type": "debug",
|
152
|
+
"z": "d6c450fe98faa45f",
|
153
|
+
"name": "debug 32",
|
154
|
+
"active": true,
|
155
|
+
"tosidebar": true,
|
156
|
+
"console": false,
|
157
|
+
"tostatus": false,
|
158
|
+
"complete": "false",
|
159
|
+
"statusVal": "",
|
160
|
+
"statusType": "auto",
|
161
|
+
"x": 370,
|
162
|
+
"y": 60,
|
163
|
+
"wires": []
|
164
|
+
},
|
165
|
+
{
|
166
|
+
"id": "a5c0ea907e64e759",
|
167
|
+
"type": "debug",
|
168
|
+
"z": "d6c450fe98faa45f",
|
169
|
+
"name": "debug 33",
|
170
|
+
"active": true,
|
171
|
+
"tosidebar": true,
|
172
|
+
"console": false,
|
173
|
+
"tostatus": false,
|
174
|
+
"complete": "false",
|
175
|
+
"statusVal": "",
|
176
|
+
"statusType": "auto",
|
177
|
+
"x": 760,
|
178
|
+
"y": 60,
|
179
|
+
"wires": []
|
180
|
+
},
|
181
|
+
{
|
182
|
+
"id": "42e6796dddd9d4db",
|
183
|
+
"type": "processcube-engine-config",
|
184
|
+
"name": "Engine 8000",
|
185
|
+
"url": "http://engine:8000"
|
186
|
+
}
|
187
|
+
]
|
package/externaltask-input.html
CHANGED
@@ -6,6 +6,7 @@
|
|
6
6
|
name: { value: '' },
|
7
7
|
engine: { value: '', type: 'processcube-engine-config' },
|
8
8
|
topic: { value: '' },
|
9
|
+
workerConfig: { value: '', type: 'json' },
|
9
10
|
},
|
10
11
|
inputs: 0,
|
11
12
|
outputs: 1,
|
@@ -13,6 +14,15 @@
|
|
13
14
|
label: function () {
|
14
15
|
return this.name || 'externaltask-input';
|
15
16
|
},
|
17
|
+
oneditprepare: function () {
|
18
|
+
$('#node-input-workerConfig').typedInput({
|
19
|
+
default: 'json',
|
20
|
+
types: ['json'],
|
21
|
+
});
|
22
|
+
},
|
23
|
+
oneditsave: function () {
|
24
|
+
this.workerConfig = $('#node-input-workerConfig').typedInput('value');
|
25
|
+
},
|
16
26
|
});
|
17
27
|
</script>
|
18
28
|
|
@@ -29,6 +39,10 @@
|
|
29
39
|
<label for="node-input-topic"><i class="fa fa-tag"></i> Topic</label>
|
30
40
|
<input type="text" id="node-input-topic" placeholder="Topic of ExternalTask" />
|
31
41
|
</div>
|
42
|
+
<div class="form-row"></div>
|
43
|
+
<label for="node-input-workerConfig"><i class="fa fa-tag"></i> Worker Config</label>
|
44
|
+
<input type="text" id="node-input-workerConfig" />
|
45
|
+
</div>
|
32
46
|
</script>
|
33
47
|
|
34
48
|
<script type="text/markdown" data-help-name="externaltask-input">
|
package/externaltask-input.js
CHANGED
@@ -16,7 +16,7 @@ module.exports = function (RED) {
|
|
16
16
|
var node = this;
|
17
17
|
var flowContext = node.context().flow;
|
18
18
|
|
19
|
-
const engine =
|
19
|
+
const engine = RED.nodes.getNode(config.engine);
|
20
20
|
|
21
21
|
const client = engine.engineClient;
|
22
22
|
|
@@ -32,83 +32,85 @@ module.exports = function (RED) {
|
|
32
32
|
eventEmitter = flowContext.get('emitter');
|
33
33
|
}
|
34
34
|
|
35
|
-
|
36
|
-
|
37
|
-
|
38
|
-
|
39
|
-
|
40
|
-
|
41
|
-
|
42
|
-
|
43
|
-
};
|
35
|
+
const etwCallback = async (payload, externalTask) => {
|
36
|
+
const saveHandleCallback = (data, callback) => {
|
37
|
+
try {
|
38
|
+
callback(data);
|
39
|
+
} catch (error) {
|
40
|
+
node.error(`Error in callback 'saveHandleCallback': ${error.message}`);
|
41
|
+
}
|
42
|
+
};
|
44
43
|
|
45
|
-
|
46
|
-
|
47
|
-
|
44
|
+
return await new Promise((resolve, reject) => {
|
45
|
+
const handleFinishTask = (msg) => {
|
46
|
+
let result = RED.util.encodeObject(msg.payload);
|
48
47
|
|
49
|
-
|
50
|
-
|
51
|
-
|
48
|
+
node.log(
|
49
|
+
`handle event for *external task flowNodeInstanceId* '${externalTask.flowNodeInstanceId}' and *processInstanceId* ${externalTask.processInstanceId} with result ${result} on msg._msgid ${msg._msgid}.`
|
50
|
+
);
|
52
51
|
|
53
|
-
|
54
|
-
|
55
|
-
|
52
|
+
if (externalTask.flowNodeInstanceId) {
|
53
|
+
delete started_external_tasks[externalTask.flowNodeInstanceId];
|
54
|
+
}
|
56
55
|
|
57
|
-
|
56
|
+
showStatus(node, Object.keys(started_external_tasks).length);
|
58
57
|
|
59
|
-
|
60
|
-
|
61
|
-
|
58
|
+
//resolve(result);
|
59
|
+
saveHandleCallback(result, resolve);
|
60
|
+
};
|
61
|
+
|
62
|
+
const handleErrorTask = (msg) => {
|
63
|
+
node.log(
|
64
|
+
`handle error event for *external task flowNodeInstanceId* '${externalTask.flowNodeInstanceId}' and *processInstanceId* '${externalTask.processInstanceId}' on *msg._msgid* '${msg._msgid}'.`
|
65
|
+
);
|
62
66
|
|
63
|
-
|
64
|
-
|
65
|
-
|
66
|
-
);
|
67
|
+
if (externalTask.flowNodeInstanceId) {
|
68
|
+
delete started_external_tasks[externalTask.flowNodeInstanceId];
|
69
|
+
}
|
67
70
|
|
68
|
-
|
69
|
-
delete started_external_tasks[externalTask.flowNodeInstanceId];
|
70
|
-
}
|
71
|
+
showStatus(node, Object.keys(started_external_tasks).length);
|
71
72
|
|
72
|
-
|
73
|
+
// TODO: with reject, the default error handling is proceed
|
74
|
+
// SEE: https://github.com/5minds/ProcessCube.Engine.Client.ts/blob/develop/src/ExternalTaskWorker.ts#L180
|
75
|
+
// reject(result);
|
76
|
+
//resolve(msg);
|
77
|
+
saveHandleCallback(msg, resolve);
|
78
|
+
};
|
73
79
|
|
74
|
-
|
75
|
-
|
76
|
-
|
77
|
-
|
78
|
-
saveHandleCallback(msg, resolve);
|
79
|
-
};
|
80
|
+
eventEmitter.once(`handle-${externalTask.flowNodeInstanceId}`, (msg, isError = false) => {
|
81
|
+
node.log(
|
82
|
+
`handle event for *external task flowNodeInstanceId* '${externalTask.flowNodeInstanceId}' and *processInstanceId* '${externalTask.processInstanceId}' with *msg._msgid* '${msg._msgid}' and *isError* '${isError}'`
|
83
|
+
);
|
80
84
|
|
81
|
-
|
82
|
-
|
83
|
-
|
84
|
-
);
|
85
|
+
if (isError) {
|
86
|
+
handleErrorTask(msg);
|
87
|
+
} else {
|
88
|
+
handleFinishTask(msg);
|
89
|
+
}
|
90
|
+
});
|
85
91
|
|
86
|
-
|
87
|
-
handleErrorTask(msg);
|
88
|
-
} else {
|
89
|
-
handleFinishTask(msg);
|
90
|
-
}
|
91
|
-
});
|
92
|
+
started_external_tasks[externalTask.flowNodeInstanceId] = externalTask;
|
92
93
|
|
93
|
-
|
94
|
+
showStatus(node, Object.keys(started_external_tasks).length);
|
94
95
|
|
95
|
-
|
96
|
+
let msg = {
|
97
|
+
_msgid: RED.util.generateId(),
|
98
|
+
task: RED.util.encodeObject(externalTask),
|
99
|
+
payload: payload,
|
100
|
+
flowNodeInstanceId: externalTask.flowNodeInstanceId,
|
101
|
+
processInstanceId: externalTask.processInstanceId
|
102
|
+
};
|
96
103
|
|
97
|
-
|
98
|
-
|
99
|
-
|
100
|
-
payload: payload,
|
101
|
-
flowNodeInstanceId: externalTask.flowNodeInstanceId,
|
102
|
-
processInstanceId: externalTask.processInstanceId
|
103
|
-
};
|
104
|
+
node.log(
|
105
|
+
`Received *external task flowNodeInstanceId* '${externalTask.flowNodeInstanceId}' and *processInstanceId* '${externalTask.processInstanceId}' with *msg._msgid* '${msg._msgid}'`
|
106
|
+
);
|
104
107
|
|
105
|
-
|
106
|
-
|
107
|
-
|
108
|
+
node.send(msg);
|
109
|
+
});
|
110
|
+
};
|
108
111
|
|
109
|
-
|
110
|
-
|
111
|
-
})
|
112
|
+
client.externalTasks
|
113
|
+
.subscribeToExternalTaskTopic(config.topic, etwCallback, config.workerConfig)
|
112
114
|
.then(async (externalTaskWorker) => {
|
113
115
|
node.status({ fill: 'blue', shape: 'ring', text: 'subcribed' });
|
114
116
|
|
package/nodered/flows.json
CHANGED
@@ -63,6 +63,14 @@
|
|
63
63
|
"info": "",
|
64
64
|
"env": []
|
65
65
|
},
|
66
|
+
{
|
67
|
+
"id": "d6c450fe98faa45f",
|
68
|
+
"type": "tab",
|
69
|
+
"label": "Wait for usertask",
|
70
|
+
"disabled": false,
|
71
|
+
"info": "",
|
72
|
+
"env": []
|
73
|
+
},
|
66
74
|
{
|
67
75
|
"id": "4c59118134081e05",
|
68
76
|
"type": "group",
|
@@ -1111,8 +1119,6 @@
|
|
1111
1119
|
"engine": "42e6796dddd9d4db",
|
1112
1120
|
"query": "payload",
|
1113
1121
|
"query_type": "msg",
|
1114
|
-
"force_send_array": false,
|
1115
|
-
"multisend": false,
|
1116
1122
|
"x": 180,
|
1117
1123
|
"y": 420,
|
1118
1124
|
"wires": [
|
@@ -1245,8 +1251,6 @@
|
|
1245
1251
|
"engine": "b78932d162d8d468",
|
1246
1252
|
"query": "payload",
|
1247
1253
|
"query_type": "msg",
|
1248
|
-
"force_send_array": false,
|
1249
|
-
"multisend": false,
|
1250
1254
|
"x": 200,
|
1251
1255
|
"y": 440,
|
1252
1256
|
"wires": [
|
@@ -1821,5 +1825,176 @@
|
|
1821
1825
|
"x": 110,
|
1822
1826
|
"y": 260,
|
1823
1827
|
"wires": []
|
1828
|
+
},
|
1829
|
+
{
|
1830
|
+
"id": "30b8f9c606a2bb83",
|
1831
|
+
"type": "wait-for-usertask",
|
1832
|
+
"z": "d6c450fe98faa45f",
|
1833
|
+
"name": "",
|
1834
|
+
"engine": "42e6796dddd9d4db",
|
1835
|
+
"query": "payload",
|
1836
|
+
"query_type": "msg",
|
1837
|
+
"only_for_new": true,
|
1838
|
+
"x": 590,
|
1839
|
+
"y": 120,
|
1840
|
+
"wires": [
|
1841
|
+
[
|
1842
|
+
"d9cb8e69e985d6df",
|
1843
|
+
"a5c0ea907e64e759"
|
1844
|
+
]
|
1845
|
+
]
|
1846
|
+
},
|
1847
|
+
{
|
1848
|
+
"id": "20fd37adf0fb6f39",
|
1849
|
+
"type": "process-start",
|
1850
|
+
"z": "d6c450fe98faa45f",
|
1851
|
+
"name": "",
|
1852
|
+
"engine": "42e6796dddd9d4db",
|
1853
|
+
"processmodel": "Wait-For-Usertask_Process",
|
1854
|
+
"startevent": "StartEvent_1",
|
1855
|
+
"x": 250,
|
1856
|
+
"y": 120,
|
1857
|
+
"wires": [
|
1858
|
+
[
|
1859
|
+
"1e1e9c428172eae2",
|
1860
|
+
"c65a4ba5461e7ee0"
|
1861
|
+
]
|
1862
|
+
]
|
1863
|
+
},
|
1864
|
+
{
|
1865
|
+
"id": "e18451d69a169f5f",
|
1866
|
+
"type": "inject",
|
1867
|
+
"z": "d6c450fe98faa45f",
|
1868
|
+
"name": "",
|
1869
|
+
"props": [
|
1870
|
+
{
|
1871
|
+
"p": "payload"
|
1872
|
+
}
|
1873
|
+
],
|
1874
|
+
"repeat": "",
|
1875
|
+
"crontab": "",
|
1876
|
+
"once": false,
|
1877
|
+
"onceDelay": 0.1,
|
1878
|
+
"topic": "",
|
1879
|
+
"payload": "{}",
|
1880
|
+
"payloadType": "json",
|
1881
|
+
"x": 90,
|
1882
|
+
"y": 120,
|
1883
|
+
"wires": [
|
1884
|
+
[
|
1885
|
+
"20fd37adf0fb6f39"
|
1886
|
+
]
|
1887
|
+
]
|
1888
|
+
},
|
1889
|
+
{
|
1890
|
+
"id": "d9cb8e69e985d6df",
|
1891
|
+
"type": "delay",
|
1892
|
+
"z": "d6c450fe98faa45f",
|
1893
|
+
"name": "",
|
1894
|
+
"pauseType": "delay",
|
1895
|
+
"timeout": "5",
|
1896
|
+
"timeoutUnits": "seconds",
|
1897
|
+
"rate": "1",
|
1898
|
+
"nbRateUnits": "1",
|
1899
|
+
"rateUnits": "second",
|
1900
|
+
"randomFirst": "1",
|
1901
|
+
"randomLast": "5",
|
1902
|
+
"randomUnits": "seconds",
|
1903
|
+
"drop": false,
|
1904
|
+
"allowrate": false,
|
1905
|
+
"outputs": 1,
|
1906
|
+
"x": 540,
|
1907
|
+
"y": 200,
|
1908
|
+
"wires": [
|
1909
|
+
[
|
1910
|
+
"acecfeef70b5c4ad"
|
1911
|
+
]
|
1912
|
+
]
|
1913
|
+
},
|
1914
|
+
{
|
1915
|
+
"id": "44f4c6c64263d366",
|
1916
|
+
"type": "usertask-output",
|
1917
|
+
"z": "d6c450fe98faa45f",
|
1918
|
+
"name": "",
|
1919
|
+
"engine": "42e6796dddd9d4db",
|
1920
|
+
"result": "payload",
|
1921
|
+
"result_type": "msg",
|
1922
|
+
"x": 880,
|
1923
|
+
"y": 200,
|
1924
|
+
"wires": [
|
1925
|
+
[]
|
1926
|
+
]
|
1927
|
+
},
|
1928
|
+
{
|
1929
|
+
"id": "acecfeef70b5c4ad",
|
1930
|
+
"type": "function",
|
1931
|
+
"z": "d6c450fe98faa45f",
|
1932
|
+
"name": "pre",
|
1933
|
+
"func": "msg.payload = {};\nreturn msg;",
|
1934
|
+
"outputs": 1,
|
1935
|
+
"timeout": 0,
|
1936
|
+
"noerr": 0,
|
1937
|
+
"initialize": "",
|
1938
|
+
"finalize": "",
|
1939
|
+
"libs": [],
|
1940
|
+
"x": 710,
|
1941
|
+
"y": 200,
|
1942
|
+
"wires": [
|
1943
|
+
[
|
1944
|
+
"44f4c6c64263d366"
|
1945
|
+
]
|
1946
|
+
]
|
1947
|
+
},
|
1948
|
+
{
|
1949
|
+
"id": "1e1e9c428172eae2",
|
1950
|
+
"type": "function",
|
1951
|
+
"z": "d6c450fe98faa45f",
|
1952
|
+
"name": "query",
|
1953
|
+
"func": "msg.payload = {\n \"processInstanceId\": msg.payload.processInstanceId,\n \"flowNodeId\": \"make_it\"\n};\n\nreturn msg;",
|
1954
|
+
"outputs": 1,
|
1955
|
+
"timeout": 0,
|
1956
|
+
"noerr": 0,
|
1957
|
+
"initialize": "",
|
1958
|
+
"finalize": "",
|
1959
|
+
"libs": [],
|
1960
|
+
"x": 410,
|
1961
|
+
"y": 120,
|
1962
|
+
"wires": [
|
1963
|
+
[
|
1964
|
+
"30b8f9c606a2bb83"
|
1965
|
+
]
|
1966
|
+
]
|
1967
|
+
},
|
1968
|
+
{
|
1969
|
+
"id": "c65a4ba5461e7ee0",
|
1970
|
+
"type": "debug",
|
1971
|
+
"z": "d6c450fe98faa45f",
|
1972
|
+
"name": "debug 32",
|
1973
|
+
"active": true,
|
1974
|
+
"tosidebar": true,
|
1975
|
+
"console": false,
|
1976
|
+
"tostatus": false,
|
1977
|
+
"complete": "false",
|
1978
|
+
"statusVal": "",
|
1979
|
+
"statusType": "auto",
|
1980
|
+
"x": 370,
|
1981
|
+
"y": 60,
|
1982
|
+
"wires": []
|
1983
|
+
},
|
1984
|
+
{
|
1985
|
+
"id": "a5c0ea907e64e759",
|
1986
|
+
"type": "debug",
|
1987
|
+
"z": "d6c450fe98faa45f",
|
1988
|
+
"name": "debug 33",
|
1989
|
+
"active": true,
|
1990
|
+
"tosidebar": true,
|
1991
|
+
"console": false,
|
1992
|
+
"tostatus": false,
|
1993
|
+
"complete": "false",
|
1994
|
+
"statusVal": "",
|
1995
|
+
"statusType": "auto",
|
1996
|
+
"x": 760,
|
1997
|
+
"y": 60,
|
1998
|
+
"wires": []
|
1824
1999
|
}
|
1825
2000
|
]
|
package/package.json
CHANGED
@@ -1,6 +1,6 @@
|
|
1
1
|
{
|
2
2
|
"name": "@5minds/node-red-contrib-processcube",
|
3
|
-
"version": "1.3.0
|
3
|
+
"version": "1.3.0",
|
4
4
|
"license": "MIT",
|
5
5
|
"description": "Node-RED nodes for ProcessCube",
|
6
6
|
"scripts": {
|
@@ -45,12 +45,11 @@
|
|
45
45
|
"processEventListener": "process-event-listener.js",
|
46
46
|
"processcubeEngineConfig": "processcube-engine-config.js",
|
47
47
|
"ProcessinstanceQuery": "processinstance-query.js",
|
48
|
+
"ProcessinstanceDelete": "processinstance-delete.js",
|
48
49
|
"ProcessdefinitionQuery": "processdefinition-query.js",
|
49
50
|
"messageEventTrigger": "message-event-trigger.js",
|
50
51
|
"signalEventTrigger": "signal-event-trigger.js",
|
51
52
|
"userTaskEventListener": "usertask-event-listener.js",
|
52
|
-
"UserTaskNewListener": "usertask-new-listener.js",
|
53
|
-
"UserTaskFinishedListener": "usertask-finished-listener.js",
|
54
53
|
"UserTaskInput": "usertask-input.js",
|
55
54
|
"UserTaskOutput": "usertask-output.js",
|
56
55
|
"WaitForUsertask": "wait-for-usertask.js"
|
@@ -2,16 +2,40 @@
|
|
2
2
|
RED.nodes.registerType('processcube-engine-config', {
|
3
3
|
category: 'config',
|
4
4
|
defaults: {
|
5
|
-
name
|
5
|
+
name: { value: '' },
|
6
6
|
url: { value: 'http://engine:8000', required: true },
|
7
|
+
clientId: { value: '' },
|
8
|
+
clientIdType: { type: 'str' },
|
9
|
+
clientSecret: { value: '' },
|
10
|
+
clientSecretType: { type: 'str' },
|
7
11
|
},
|
8
12
|
label: function () {
|
9
13
|
return this.name || this.url;
|
10
14
|
},
|
11
|
-
|
12
|
-
clientId
|
13
|
-
|
14
|
-
|
15
|
+
oneditprepare: function () {
|
16
|
+
$('#node-config-input-clientId').typedInput({
|
17
|
+
default: 'str',
|
18
|
+
types: ['str', 'global', 'flow', 'env', 'msg', 'cred'],
|
19
|
+
});
|
20
|
+
|
21
|
+
$('#node-config-input-clientSecret').typedInput({
|
22
|
+
default: 'str',
|
23
|
+
types: ['str', 'global', 'flow', 'env', 'msg', 'cred'],
|
24
|
+
});
|
25
|
+
|
26
|
+
$('#node-config-input-clientId').typedInput('value', this.clientId);
|
27
|
+
$('#node-config-input-clientId').typedInput('type', this.clientIdType);
|
28
|
+
|
29
|
+
$('#node-config-input-clientSecret').typedInput('value', this.clientSecret);
|
30
|
+
$('#node-config-input-clientSecret').typedInput('type', this.clientSecretType);
|
31
|
+
},
|
32
|
+
oneditsave: function () {
|
33
|
+
this.clientId = $('#node-config-input-clientId').typedInput('value');
|
34
|
+
this.clientIdType = $('#node-config-input-clientId').typedInput('type');
|
35
|
+
|
36
|
+
this.clientSecret = $('#node-config-input-clientSecret').typedInput('value');
|
37
|
+
this.clientSecretType = $('#node-config-input-clientSecret').typedInput('type');
|
38
|
+
},
|
15
39
|
});
|
16
40
|
</script>
|
17
41
|
|
@@ -11,18 +11,21 @@ module.exports = function (RED) {
|
|
11
11
|
const identityChangedCallbacks = [];
|
12
12
|
this.url = n.url;
|
13
13
|
this.identity = null;
|
14
|
-
|
14
|
+
|
15
|
+
this.credentials.clientId = RED.util.evaluateNodeProperty(n.clientId, n.clientIdType, node);
|
16
|
+
this.credentials.clientSecret = RED.util.evaluateNodeProperty(n.clientSecret, n.clientSecretType, node);
|
17
|
+
|
15
18
|
this.registerOnIdentityChanged = function (callback) {
|
16
19
|
identityChangedCallbacks.push(callback);
|
17
20
|
};
|
18
21
|
|
19
|
-
this.isIdentityReady = function() {
|
22
|
+
this.isIdentityReady = function () {
|
20
23
|
if (this.credentials.clientId && this.credentials.clientSecret) {
|
21
24
|
return this.identity != null;
|
22
25
|
} else {
|
23
26
|
return true;
|
24
27
|
}
|
25
|
-
}
|
28
|
+
};
|
26
29
|
|
27
30
|
this.setIdentity = (identity) => {
|
28
31
|
node.log(`setIdentity: ${JSON.stringify(identity)}`);
|
@@ -50,7 +53,7 @@ module.exports = function (RED) {
|
|
50
53
|
this.credentials.clientId,
|
51
54
|
this.credentials.clientSecret,
|
52
55
|
authorityUrl,
|
53
|
-
node
|
56
|
+
node
|
54
57
|
).catch((reason) => {
|
55
58
|
console.error(reason);
|
56
59
|
node.error(reason);
|
@@ -136,7 +139,7 @@ async function startRefreshingIdentityCycle(clientId, clientSecret, authorityUrl
|
|
136
139
|
if (retries === 0) {
|
137
140
|
console.error(
|
138
141
|
'Could not refresh identity for external task worker processes. Stopping all external task workers.',
|
139
|
-
{ error }
|
142
|
+
{ error }
|
140
143
|
);
|
141
144
|
return;
|
142
145
|
}
|
@@ -16,23 +16,26 @@
|
|
16
16
|
<bpmn:userTask id="user_task" name="User Task">
|
17
17
|
<bpmn:extensionElements>
|
18
18
|
<camunda:formData>
|
19
|
-
<camunda:formField id="checkbox" label="
|
20
|
-
<camunda:formField id="
|
21
|
-
<camunda:formField id="date
|
22
|
-
<camunda:formField id="
|
23
|
-
<camunda:formField id="
|
24
|
-
<camunda:formField id="
|
25
|
-
<camunda:formField id="
|
26
|
-
<camunda:formField id="
|
27
|
-
<camunda:formField id="
|
28
|
-
<camunda:formField id="
|
29
|
-
<camunda:formField id="
|
30
|
-
<camunda:formField id="
|
31
|
-
<camunda:formField id="
|
32
|
-
<camunda:formField id="
|
33
|
-
<camunda:formField id="
|
34
|
-
<camunda:formField id="
|
35
|
-
<camunda:formField id="
|
19
|
+
<camunda:formField id="checkbox" label="Checkbox" type="checkbox" customForm="{"entries":[{"key":"key","value":"value"}],"hint":"hint"}" />
|
20
|
+
<camunda:formField id="colorpicker" label="Colorpicker" type="color" defaultValue="#e32626" customForm="{"hint":"hint"}" />
|
21
|
+
<camunda:formField id="date" label="Date" type="date" defaultValue="2024-10-18" customForm="{"hint":"hint"}" />
|
22
|
+
<camunda:formField id="datetime" label="Datetime" type="datetime-local" defaultValue="2024-10-19T12:41" customForm="{"hint":"hint"}" />
|
23
|
+
<camunda:formField id="email" label="Email" type="email" customForm="{"hint":"hint"}" />
|
24
|
+
<camunda:formField id="header" type="header" defaultValue="Header" />
|
25
|
+
<camunda:formField id="hidden" type="hidden" defaultValue="hidden" />
|
26
|
+
<camunda:formField id="number" label="Number" type="number" customForm="{"step":1.5,"hint":"hint"}" />
|
27
|
+
<camunda:formField id="month" label="Month" type="month" defaultValue="2024-07" customForm="{"hint":"hint"}" />
|
28
|
+
<camunda:formField id="paragraph" type="paragraph" defaultValue="A lot of words..." />
|
29
|
+
<camunda:formField id="password" label="Pasword" type="password" customForm="{"hint":"hint"}" />
|
30
|
+
<camunda:formField id="radio" label="Radio" type="radio" customForm="{"entries":[{"key":"key","value":"value"}],"hint":"hint"}" />
|
31
|
+
<camunda:formField id="range" label="Range" type="range" defaultValue="30" customForm="{"step":0.4,"min":3,"hint":"hint"}" />
|
32
|
+
<camunda:formField id="select" label="Select" type="select" customForm="{"entries":[{"key":"key","value":"value"}],"hint":"hint"}" />
|
33
|
+
<camunda:formField id="tel" label="Tel." type="tel" customForm="{"hint":"hint"}" />
|
34
|
+
<camunda:formField id="text" label="Text" type="string" customForm="{"hint":"hint"}" />
|
35
|
+
<camunda:formField id="textarea" label="Textarea" type="textarea" defaultValue="more words..." customForm="{"rows":14,"hint":"hint"}" />
|
36
|
+
<camunda:formField id="time" label="Time" type="time" defaultValue="11:54" customForm="{"hint":"hint"}" />
|
37
|
+
<camunda:formField id="url" label="URL" type="url" customForm="{"hint":"hint"}" />
|
38
|
+
<camunda:formField id="week" label="Week" type="week" defaultValue="2024-W37" customForm="{"hint":"hint"}" />
|
36
39
|
</camunda:formData>
|
37
40
|
</bpmn:extensionElements>
|
38
41
|
<bpmn:incoming>Flow_142awo6</bpmn:incoming>
|
@@ -0,0 +1,74 @@
|
|
1
|
+
<?xml version="1.0" encoding="UTF-8"?>
|
2
|
+
<bpmn:definitions xmlns:bpmn="http://www.omg.org/spec/BPMN/20100524/MODEL" xmlns:bpmndi="http://www.omg.org/spec/BPMN/20100524/DI" xmlns:dc="http://www.omg.org/spec/DD/20100524/DC" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:di="http://www.omg.org/spec/DD/20100524/DI" id="Wait-For-Usertask_Definition" targetNamespace="http://bpmn.io/schema/bpmn" exporter="5Minds Studio" exporterVersion="1">
|
3
|
+
<bpmn:collaboration id="Collaboration_1cidyxu" name="">
|
4
|
+
<bpmn:participant id="Participant_0px403d" name="Wait-For-Usertask" processRef="Wait-For-Usertask_Process" />
|
5
|
+
</bpmn:collaboration>
|
6
|
+
<bpmn:process id="Wait-For-Usertask_Process" name="Wait-For-Usertask" isExecutable="true">
|
7
|
+
<bpmn:laneSet>
|
8
|
+
<bpmn:lane id="Lane_1xzf0d3" name="Lane">
|
9
|
+
<bpmn:flowNodeRef>StartEvent_1</bpmn:flowNodeRef>
|
10
|
+
<bpmn:flowNodeRef>Event_0zjvsuj</bpmn:flowNodeRef>
|
11
|
+
<bpmn:flowNodeRef>make_it</bpmn:flowNodeRef>
|
12
|
+
<bpmn:flowNodeRef>Event_02k45ju</bpmn:flowNodeRef>
|
13
|
+
</bpmn:lane>
|
14
|
+
</bpmn:laneSet>
|
15
|
+
<bpmn:startEvent id="StartEvent_1" name="Start">
|
16
|
+
<bpmn:outgoing>Flow_085u77p</bpmn:outgoing>
|
17
|
+
</bpmn:startEvent>
|
18
|
+
<bpmn:sequenceFlow id="Flow_085u77p" sourceRef="StartEvent_1" targetRef="Event_02k45ju" />
|
19
|
+
<bpmn:sequenceFlow id="Flow_0rvexxn" sourceRef="Event_02k45ju" targetRef="make_it" />
|
20
|
+
<bpmn:endEvent id="Event_0zjvsuj">
|
21
|
+
<bpmn:incoming>Flow_10jnerm</bpmn:incoming>
|
22
|
+
</bpmn:endEvent>
|
23
|
+
<bpmn:sequenceFlow id="Flow_10jnerm" sourceRef="make_it" targetRef="Event_0zjvsuj" />
|
24
|
+
<bpmn:userTask id="make_it" name="Make it">
|
25
|
+
<bpmn:incoming>Flow_0rvexxn</bpmn:incoming>
|
26
|
+
<bpmn:outgoing>Flow_10jnerm</bpmn:outgoing>
|
27
|
+
</bpmn:userTask>
|
28
|
+
<bpmn:intermediateCatchEvent id="Event_02k45ju">
|
29
|
+
<bpmn:incoming>Flow_085u77p</bpmn:incoming>
|
30
|
+
<bpmn:outgoing>Flow_0rvexxn</bpmn:outgoing>
|
31
|
+
<bpmn:timerEventDefinition id="TimerEventDefinition_157z719">
|
32
|
+
<bpmn:timeDuration xsi:type="bpmn:tFormalExpression">PT5S</bpmn:timeDuration>
|
33
|
+
</bpmn:timerEventDefinition>
|
34
|
+
</bpmn:intermediateCatchEvent>
|
35
|
+
</bpmn:process>
|
36
|
+
<bpmndi:BPMNDiagram id="BPMNDiagram_1">
|
37
|
+
<bpmndi:BPMNPlane id="BPMNPlane_1" bpmnElement="Collaboration_1cidyxu">
|
38
|
+
<bpmndi:BPMNShape id="Participant_0px403d_di" bpmnElement="Participant_0px403d" isHorizontal="true">
|
39
|
+
<dc:Bounds x="5" y="4" width="535" height="346" />
|
40
|
+
</bpmndi:BPMNShape>
|
41
|
+
<bpmndi:BPMNShape id="Lane_1xzf0d3_di" bpmnElement="Lane_1xzf0d3" isHorizontal="true">
|
42
|
+
<dc:Bounds x="35" y="4" width="505" height="346" />
|
43
|
+
</bpmndi:BPMNShape>
|
44
|
+
<bpmndi:BPMNShape id="StartEvent_1_di" bpmnElement="StartEvent_1">
|
45
|
+
<dc:Bounds x="92" y="152" width="36" height="36" />
|
46
|
+
<bpmndi:BPMNLabel>
|
47
|
+
<dc:Bounds x="98" y="195" width="24" height="14" />
|
48
|
+
</bpmndi:BPMNLabel>
|
49
|
+
</bpmndi:BPMNShape>
|
50
|
+
<bpmndi:BPMNShape id="Event_0zjvsuj_di" bpmnElement="Event_0zjvsuj">
|
51
|
+
<dc:Bounds x="442" y="152" width="36" height="36" />
|
52
|
+
</bpmndi:BPMNShape>
|
53
|
+
<bpmndi:BPMNShape id="Activity_0yse9qf_di" bpmnElement="make_it">
|
54
|
+
<dc:Bounds x="280" y="130" width="100" height="80" />
|
55
|
+
<bpmndi:BPMNLabel />
|
56
|
+
</bpmndi:BPMNShape>
|
57
|
+
<bpmndi:BPMNShape id="Event_09boky1_di" bpmnElement="Event_02k45ju">
|
58
|
+
<dc:Bounds x="182" y="152" width="36" height="36" />
|
59
|
+
</bpmndi:BPMNShape>
|
60
|
+
<bpmndi:BPMNEdge id="Flow_085u77p_di" bpmnElement="Flow_085u77p">
|
61
|
+
<di:waypoint x="128" y="170" />
|
62
|
+
<di:waypoint x="182" y="170" />
|
63
|
+
</bpmndi:BPMNEdge>
|
64
|
+
<bpmndi:BPMNEdge id="Flow_0rvexxn_di" bpmnElement="Flow_0rvexxn">
|
65
|
+
<di:waypoint x="218" y="170" />
|
66
|
+
<di:waypoint x="280" y="170" />
|
67
|
+
</bpmndi:BPMNEdge>
|
68
|
+
<bpmndi:BPMNEdge id="Flow_10jnerm_di" bpmnElement="Flow_10jnerm">
|
69
|
+
<di:waypoint x="380" y="170" />
|
70
|
+
<di:waypoint x="442" y="170" />
|
71
|
+
</bpmndi:BPMNEdge>
|
72
|
+
</bpmndi:BPMNPlane>
|
73
|
+
</bpmndi:BPMNDiagram>
|
74
|
+
</bpmn:definitions>
|
@@ -0,0 +1,58 @@
|
|
1
|
+
<script type="text/javascript">
|
2
|
+
RED.nodes.registerType('processinstance-delete', {
|
3
|
+
category: 'ProcessCube DevOps',
|
4
|
+
color: '#02AFD6',
|
5
|
+
defaults: {
|
6
|
+
name: { value: '' },
|
7
|
+
engine: { value: '', type: 'processcube-engine-config' },
|
8
|
+
time: { value: '', type: 'number' },
|
9
|
+
time_type: { value: 'hours' },
|
10
|
+
},
|
11
|
+
inputs: 1,
|
12
|
+
outputs: 1,
|
13
|
+
icon: 'font-awesome/fa-sign-in',
|
14
|
+
label: function () {
|
15
|
+
return this.name || 'processinstance-delete';
|
16
|
+
},
|
17
|
+
});
|
18
|
+
</script>
|
19
|
+
|
20
|
+
<script type="text/html" data-template-name="processinstance-delete">
|
21
|
+
<div class="form-row">
|
22
|
+
<label for="node-input-name"><i class="fa fa-tag"></i> Name</label>
|
23
|
+
<input type="text" id="node-input-name" placeholder="Name" />
|
24
|
+
</div>
|
25
|
+
<div class="form-row">
|
26
|
+
<label for="node-input-engine"><i class="fa fa-tag"></i> Engine-URL</label>
|
27
|
+
<input type="text" id="node-input-engine" placeholder="Engine-URL" />
|
28
|
+
</div>
|
29
|
+
<div class="form-row">
|
30
|
+
<label for="node-input-time"><i class="fa fa-tag"></i> Time</label>
|
31
|
+
<input type="text" id="node-input-time" />
|
32
|
+
</div>
|
33
|
+
<div class="form-row">
|
34
|
+
<label for="node-input-time_type"><i class="fa fa-sliders"></i> Period</label>
|
35
|
+
<select id="node-input-time_type" style="width: 70%;">
|
36
|
+
<option value="hours">Hours</option>
|
37
|
+
<option value="days">Days</option>
|
38
|
+
</select>
|
39
|
+
</div>
|
40
|
+
</script>
|
41
|
+
|
42
|
+
<script type="text/markdown" data-help-name="processinstance-delete">
|
43
|
+
Delete old instances of a process model in the ProcessCube.
|
44
|
+
|
45
|
+
## Inputs
|
46
|
+
|
47
|
+
: payload.time (number): The number of given time periods.
|
48
|
+
: payload.time_type ('hours' | 'days'): The type of time period to use.
|
49
|
+
|
50
|
+
## Outputs
|
51
|
+
|
52
|
+
: payload (string[]): The ids of the processinstances that were deleted.
|
53
|
+
|
54
|
+
### References
|
55
|
+
|
56
|
+
- [The ProcessCube Developer Network](https://processcube.io) - All documentation for the ProcessCube© platform
|
57
|
+
- [Node-RED Integration in ProcessCube©](https://processcube.io/docs/node-red) - Node-RED integration in ProcessCube©
|
58
|
+
</script>
|
@@ -0,0 +1,55 @@
|
|
1
|
+
module.exports = function (RED) {
|
2
|
+
function ProcessInstanceDelete(config) {
|
3
|
+
RED.nodes.createNode(this, config);
|
4
|
+
var node = this;
|
5
|
+
|
6
|
+
node.on('input', async function (msg) {
|
7
|
+
const engine = RED.nodes.getNode(config.engine);
|
8
|
+
const client = engine.engineClient;
|
9
|
+
|
10
|
+
if (!client) {
|
11
|
+
node.error('No engine configured.');
|
12
|
+
return;
|
13
|
+
}
|
14
|
+
let timeMultiplier;
|
15
|
+
if (msg.payload.time_type) {
|
16
|
+
timeMultiplier = msg.payload.time_type == 'hours' ? 1 : 24;
|
17
|
+
} else {
|
18
|
+
timeMultiplier = config.time_type == 'hours' ? 1 : 24;
|
19
|
+
}
|
20
|
+
|
21
|
+
const timeToUse = msg.payload.time ? msg.payload.time : config.time;
|
22
|
+
|
23
|
+
try {
|
24
|
+
const fetchInstancesByState = async (state) => {
|
25
|
+
const result = await client.processInstances.query({ state });
|
26
|
+
return result.processInstances;
|
27
|
+
};
|
28
|
+
|
29
|
+
const finishedInstances = await fetchInstancesByState('finished');
|
30
|
+
const terminatedInstances = await fetchInstancesByState('terminated');
|
31
|
+
const errorInstances = await fetchInstancesByState('error');
|
32
|
+
|
33
|
+
let allInstances = [...finishedInstances, ...terminatedInstances, ...errorInstances];
|
34
|
+
|
35
|
+
const today = new Date();
|
36
|
+
|
37
|
+
const oldTasks = allInstances.filter((instance) => {
|
38
|
+
const finishedDate = new Date(instance.finishedAt);
|
39
|
+
const diffInHours = (today - finishedDate) / (1000 * 60 * 60);
|
40
|
+
return diffInHours > Number(timeToUse) * timeMultiplier;
|
41
|
+
});
|
42
|
+
|
43
|
+
const ids = oldTasks.map((obj) => obj.processInstanceId);
|
44
|
+
msg.payload = ids;
|
45
|
+
|
46
|
+
await client.processInstances.deleteProcessInstances(ids, true, engine.identity);
|
47
|
+
node.send(msg);
|
48
|
+
} catch (error) {
|
49
|
+
node.error(error);
|
50
|
+
}
|
51
|
+
});
|
52
|
+
}
|
53
|
+
|
54
|
+
RED.nodes.registerType('processinstance-delete', ProcessInstanceDelete);
|
55
|
+
};
|
package/wait-for-usertask.js
CHANGED
@@ -3,12 +3,12 @@ module.exports = function (RED) {
|
|
3
3
|
RED.nodes.createNode(this, config);
|
4
4
|
var node = this;
|
5
5
|
|
6
|
-
|
6
|
+
node.engine = RED.nodes.getNode(config.engine);
|
7
7
|
|
8
|
-
const client = engine.engineClient;
|
8
|
+
const client = node.engine.engineClient;
|
9
9
|
|
10
10
|
let subscription = null;
|
11
|
-
let currentIdentity = engine.identity;
|
11
|
+
let currentIdentity = node.engine.identity;
|
12
12
|
let subscribe = null;
|
13
13
|
|
14
14
|
node.on('input', async function (msg) {
|
@@ -30,6 +30,9 @@ module.exports = function (RED) {
|
|
30
30
|
const matchingFlowNodes = await client.userTasks.query(newQuery, { identity: currentIdentity });
|
31
31
|
|
32
32
|
if (matchingFlowNodes.userTasks && matchingFlowNodes.userTasks.length == 1) {
|
33
|
+
// remove subscription
|
34
|
+
client.userTasks.removeSubscription(subscription, currentIdentity);
|
35
|
+
|
33
36
|
const userTask = matchingFlowNodes.userTasks[0];
|
34
37
|
|
35
38
|
msg.payload = { userTask: userTask };
|
@@ -38,8 +41,6 @@ module.exports = function (RED) {
|
|
38
41
|
// nothing todo - wait for next notification
|
39
42
|
}
|
40
43
|
|
41
|
-
// remove subscription
|
42
|
-
client.userTasks.removeSubscription(subscription, currentIdentity);
|
43
44
|
}, { identity: currentIdentity });
|
44
45
|
|
45
46
|
node.log({"Handling old userTasks config.only_for_new": config.only_for_new});
|