@nocobase/plugin-workflow 0.10.1-alpha.1 → 0.11.1-alpha.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (289) hide show
  1. package/client.d.ts +2 -3
  2. package/client.js +1 -30
  3. package/lib/client/AddButton.js +13 -11
  4. package/lib/client/Branch.js +10 -8
  5. package/lib/client/CanvasContent.js +12 -10
  6. package/lib/client/ExecutionCanvas.js +37 -33
  7. package/lib/client/ExecutionPage.js +4 -9
  8. package/lib/client/WorkflowCanvas.js +18 -15
  9. package/lib/client/WorkflowPage.js +4 -9
  10. package/lib/client/WorkflowProvider.js +1 -40
  11. package/lib/client/components/CollectionBlockInitializer.js +3 -3
  12. package/lib/client/components/CollectionFieldset.d.ts +1 -1
  13. package/lib/client/components/CollectionFieldset.js +15 -16
  14. package/lib/client/components/Duration.js +5 -5
  15. package/lib/client/components/DynamicExpression.d.ts +3 -3
  16. package/lib/client/components/FieldsSelect.d.ts +1 -1
  17. package/lib/client/components/FieldsSelect.js +10 -7
  18. package/lib/client/components/NodeDescription.js +45 -31
  19. package/lib/client/components/RadioWithTooltip.js +13 -20
  20. package/lib/client/components/ValueBlock.js +14 -21
  21. package/lib/client/components/renderEngineReference.js +1 -8
  22. package/lib/client/index.d.ts +12 -4
  23. package/lib/client/index.js +78 -15
  24. package/lib/client/locale/zh-CN.d.ts +5 -1
  25. package/lib/client/locale/zh-CN.js +6 -2
  26. package/lib/client/nodes/aggregate.d.ts +8 -3
  27. package/lib/client/nodes/aggregate.js +5 -4
  28. package/lib/client/nodes/calculation.d.ts +6 -4
  29. package/lib/client/nodes/calculation.js +22 -28
  30. package/lib/client/nodes/condition.d.ts +2 -10
  31. package/lib/client/nodes/condition.js +19 -37
  32. package/lib/client/nodes/create.d.ts +5 -6
  33. package/lib/client/nodes/create.js +1 -3
  34. package/lib/client/nodes/destroy.d.ts +1 -1
  35. package/lib/client/nodes/index.d.ts +2 -3
  36. package/lib/client/nodes/index.js +95 -102
  37. package/lib/client/nodes/loop.d.ts +1 -1
  38. package/lib/client/nodes/loop.js +46 -54
  39. package/lib/client/nodes/manual/FormBlockInitializer.js +6 -5
  40. package/lib/client/nodes/manual/ModeConfig.js +23 -30
  41. package/lib/client/nodes/manual/SchemaConfig.d.ts +4 -5
  42. package/lib/client/nodes/manual/SchemaConfig.js +180 -25
  43. package/lib/client/nodes/manual/WorkflowTodo.js +95 -110
  44. package/lib/client/nodes/manual/WorkflowTodoBlockInitializer.d.ts +2 -5
  45. package/lib/client/nodes/manual/WorkflowTodoBlockInitializer.js +6 -5
  46. package/lib/client/nodes/manual/forms/create.js +8 -1
  47. package/lib/client/nodes/manual/forms/custom.js +22 -22
  48. package/lib/client/nodes/manual/forms/update.js +8 -1
  49. package/lib/client/nodes/manual/index.d.ts +6 -1
  50. package/lib/client/nodes/manual/index.js +5 -4
  51. package/lib/client/nodes/parallel.js +23 -20
  52. package/lib/client/nodes/query.d.ts +3 -5
  53. package/lib/client/nodes/query.js +1 -3
  54. package/lib/client/nodes/request.d.ts +2 -2
  55. package/lib/client/nodes/request.js +7 -7
  56. package/lib/client/nodes/sql.d.ts +26 -0
  57. package/lib/client/{triggers/schedule/DateFieldsSelect.js → nodes/sql.js} +37 -46
  58. package/lib/client/nodes/update.d.ts +2 -2
  59. package/lib/client/nodes/update.js +1 -1
  60. package/lib/client/schemas/collection.d.ts +3 -4
  61. package/lib/client/schemas/collection.js +11 -17
  62. package/lib/client/style.d.ts +18 -13
  63. package/lib/client/style.js +315 -292
  64. package/lib/client/triggers/collection.d.ts +13 -13
  65. package/lib/client/triggers/collection.js +5 -1
  66. package/lib/client/triggers/index.d.ts +3 -4
  67. package/lib/client/triggers/index.js +51 -53
  68. package/lib/client/triggers/schedule/EndsByField.js +11 -11
  69. package/lib/client/triggers/schedule/OnField.js +45 -33
  70. package/lib/client/triggers/schedule/RepeatField.js +4 -4
  71. package/lib/client/triggers/schedule/ScheduleConfig.js +24 -31
  72. package/lib/client/triggers/schedule/index.d.ts +1 -1
  73. package/lib/client/triggers/schedule/index.js +32 -20
  74. package/lib/client/variable.d.ts +31 -13
  75. package/lib/client/variable.js +44 -29
  76. package/lib/server/Plugin.d.ts +3 -6
  77. package/lib/server/Plugin.js +15 -12
  78. package/lib/server/Processor.d.ts +3 -5
  79. package/lib/server/Processor.js +2 -2
  80. package/lib/server/actions/nodes.js +7 -7
  81. package/lib/server/fields/expression-field.d.ts +1 -2
  82. package/lib/server/fields/expression-field.js +1 -8
  83. package/lib/server/functions/index.d.ts +2 -3
  84. package/lib/server/index.d.ts +1 -0
  85. package/lib/server/index.js +12 -0
  86. package/lib/server/instructions/aggregate.d.ts +1 -1
  87. package/lib/server/instructions/aggregate.js +5 -5
  88. package/lib/server/instructions/condition.d.ts +2 -1
  89. package/lib/server/instructions/create.d.ts +2 -2
  90. package/lib/server/instructions/create.js +13 -13
  91. package/lib/server/instructions/delay.d.ts +3 -3
  92. package/lib/server/instructions/delay.js +66 -64
  93. package/lib/server/instructions/destroy.d.ts +1 -1
  94. package/lib/server/instructions/index.d.ts +5 -5
  95. package/lib/server/instructions/index.js +1 -1
  96. package/lib/server/instructions/loop.d.ts +1 -2
  97. package/lib/server/instructions/manual/actions.js +19 -7
  98. package/lib/server/instructions/manual/forms/create.js +7 -1
  99. package/lib/server/instructions/manual/forms/index.d.ts +1 -1
  100. package/lib/server/instructions/manual/forms/update.js +7 -1
  101. package/lib/server/instructions/manual/index.d.ts +1 -1
  102. package/lib/server/instructions/parallel.d.ts +1 -2
  103. package/lib/server/instructions/query.d.ts +1 -1
  104. package/lib/server/instructions/query.js +8 -1
  105. package/lib/server/instructions/request.d.ts +3 -3
  106. package/lib/server/instructions/request.js +5 -2
  107. package/lib/server/instructions/sql.d.ts +12 -0
  108. package/lib/server/instructions/sql.js +34 -0
  109. package/lib/server/instructions/update.d.ts +1 -1
  110. package/lib/server/migrations/20230221071831-calculation-expression.js +1 -1
  111. package/lib/server/migrations/20230221121203-condition-calculation.js +1 -1
  112. package/lib/server/migrations/20230221162902-jsonb-to-json.js +7 -7
  113. package/lib/server/migrations/20230411034722-manual-multi-form.js +1 -8
  114. package/lib/server/migrations/20230710115902-manual-action-values.d.ts +4 -0
  115. package/lib/server/migrations/20230710115902-manual-action-values.js +97 -0
  116. package/lib/server/triggers/collection.d.ts +1 -1
  117. package/lib/server/triggers/collection.js +15 -13
  118. package/lib/server/triggers/index.d.ts +1 -1
  119. package/lib/server/triggers/schedule.d.ts +1 -1
  120. package/lib/server/triggers/schedule.js +18 -18
  121. package/lib/server/{models → types}/Execution.d.ts +2 -3
  122. package/lib/server/{models → types}/FlowNode.d.ts +1 -2
  123. package/lib/server/{models → types}/Job.d.ts +1 -2
  124. package/lib/server/{models → types}/Workflow.d.ts +1 -2
  125. package/lib/server/types/index.d.ts +4 -0
  126. package/lib/server/types/index.js +5 -0
  127. package/lib/server/utils.d.ts +2 -0
  128. package/lib/server/utils.js +21 -0
  129. package/package.json +39 -18
  130. package/server.d.ts +2 -3
  131. package/server.js +1 -30
  132. package/src/client/AddButton.tsx +111 -0
  133. package/src/client/Branch.tsx +37 -0
  134. package/src/client/CanvasContent.tsx +25 -0
  135. package/src/client/ExecutionCanvas.tsx +166 -0
  136. package/src/client/ExecutionLink.tsx +16 -0
  137. package/src/client/ExecutionPage.tsx +45 -0
  138. package/src/client/ExecutionResourceProvider.tsx +21 -0
  139. package/src/client/FlowContext.ts +7 -0
  140. package/src/client/WorkflowCanvas.tsx +221 -0
  141. package/src/client/WorkflowLink.tsx +16 -0
  142. package/src/client/WorkflowPage.tsx +52 -0
  143. package/src/client/WorkflowProvider.tsx +84 -0
  144. package/src/client/components/CollectionBlockInitializer.tsx +71 -0
  145. package/src/client/components/CollectionFieldset.tsx +160 -0
  146. package/src/client/components/Duration.tsx +45 -0
  147. package/src/client/components/DynamicExpression.tsx +53 -0
  148. package/src/client/components/FieldsSelect.tsx +32 -0
  149. package/src/client/components/FilterDynamicComponent.tsx +15 -0
  150. package/src/client/components/NodeDescription.tsx +51 -0
  151. package/src/client/components/NullRender.tsx +3 -0
  152. package/src/client/components/OpenDrawer.tsx +24 -0
  153. package/src/client/components/RadioWithTooltip.tsx +38 -0
  154. package/src/client/components/ValueBlock.tsx +67 -0
  155. package/src/client/components/renderEngineReference.tsx +30 -0
  156. package/src/client/constants.tsx +91 -0
  157. package/src/client/index.tsx +51 -0
  158. package/src/client/interfaces/expression.tsx +25 -0
  159. package/src/client/locale/en-US.ts +136 -0
  160. package/src/client/locale/es-ES.ts +129 -0
  161. package/src/client/locale/index.ts +18 -0
  162. package/src/client/locale/ja-JP.ts +90 -0
  163. package/src/client/locale/pt-BR.ts +136 -0
  164. package/src/client/locale/ru-RU.ts +90 -0
  165. package/src/client/locale/tr-TR.ts +90 -0
  166. package/src/client/locale/zh-CN.ts +248 -0
  167. package/src/client/nodes/aggregate.tsx +327 -0
  168. package/src/client/nodes/calculation.tsx +216 -0
  169. package/src/client/nodes/condition.tsx +463 -0
  170. package/src/client/nodes/create.tsx +85 -0
  171. package/src/client/nodes/delay.tsx +37 -0
  172. package/src/client/nodes/destroy.tsx +34 -0
  173. package/src/client/nodes/index.tsx +485 -0
  174. package/src/client/nodes/loop.tsx +144 -0
  175. package/src/client/nodes/manual/AssigneesSelect.tsx +33 -0
  176. package/src/client/nodes/manual/DetailsBlockProvider.tsx +80 -0
  177. package/src/client/nodes/manual/FormBlockInitializer.tsx +69 -0
  178. package/src/client/nodes/manual/FormBlockProvider.tsx +75 -0
  179. package/src/client/nodes/manual/ModeConfig.tsx +84 -0
  180. package/src/client/nodes/manual/SchemaConfig.tsx +509 -0
  181. package/src/client/nodes/manual/WorkflowTodo.tsx +607 -0
  182. package/src/client/nodes/manual/WorkflowTodoBlockInitializer.tsx +28 -0
  183. package/src/client/nodes/manual/forms/create.tsx +92 -0
  184. package/src/client/nodes/manual/forms/custom.tsx +392 -0
  185. package/src/client/nodes/manual/forms/update.tsx +134 -0
  186. package/src/client/nodes/manual/index.tsx +162 -0
  187. package/src/client/nodes/manual/utils.ts +28 -0
  188. package/src/client/nodes/parallel.tsx +138 -0
  189. package/src/client/nodes/query.tsx +88 -0
  190. package/src/client/nodes/request.tsx +185 -0
  191. package/src/client/nodes/sql.tsx +37 -0
  192. package/src/client/nodes/update.tsx +99 -0
  193. package/src/client/schemas/collection.ts +75 -0
  194. package/src/client/schemas/executions.tsx +169 -0
  195. package/src/client/schemas/workflows.ts +364 -0
  196. package/src/client/style.tsx +347 -0
  197. package/src/client/triggers/collection.tsx +190 -0
  198. package/src/client/triggers/index.tsx +311 -0
  199. package/src/client/triggers/schedule/EndsByField.tsx +40 -0
  200. package/src/client/triggers/schedule/OnField.tsx +64 -0
  201. package/src/client/triggers/schedule/RepeatField.tsx +116 -0
  202. package/src/client/triggers/schedule/ScheduleConfig.tsx +227 -0
  203. package/src/client/triggers/schedule/constants.ts +4 -0
  204. package/src/client/triggers/schedule/index.tsx +78 -0
  205. package/src/client/triggers/schedule/locale/Cron.zh-CN.ts +79 -0
  206. package/src/client/utils.ts +36 -0
  207. package/src/client/variable.tsx +318 -0
  208. package/src/index.ts +1 -0
  209. package/src/server/Plugin.ts +355 -0
  210. package/src/server/Processor.ts +354 -0
  211. package/src/server/__tests__/Plugin.test.ts +398 -0
  212. package/src/server/__tests__/Processor.test.ts +474 -0
  213. package/src/server/__tests__/actions/workflows.test.ts +419 -0
  214. package/src/server/__tests__/collections/categories.ts +27 -0
  215. package/src/server/__tests__/collections/comments.ts +24 -0
  216. package/src/server/__tests__/collections/posts.ts +42 -0
  217. package/src/server/__tests__/collections/replies.ts +9 -0
  218. package/src/server/__tests__/collections/tags.ts +15 -0
  219. package/src/server/__tests__/index.ts +89 -0
  220. package/src/server/__tests__/instructions/aggregate.test.ts +294 -0
  221. package/src/server/__tests__/instructions/calculation.test.ts +265 -0
  222. package/src/server/__tests__/instructions/condition.test.ts +335 -0
  223. package/src/server/__tests__/instructions/create.test.ts +129 -0
  224. package/src/server/__tests__/instructions/delay.test.ts +182 -0
  225. package/src/server/__tests__/instructions/destroy.test.ts +58 -0
  226. package/src/server/__tests__/instructions/loop.test.ts +331 -0
  227. package/src/server/__tests__/instructions/manual.test.ts +1173 -0
  228. package/src/server/__tests__/instructions/parallel.test.ts +445 -0
  229. package/src/server/__tests__/instructions/query.test.ts +359 -0
  230. package/src/server/__tests__/instructions/request.test.ts +247 -0
  231. package/src/server/__tests__/instructions/sql.test.ts +162 -0
  232. package/src/server/__tests__/instructions/update.test.ts +189 -0
  233. package/src/server/__tests__/triggers/collection.test.ts +333 -0
  234. package/src/server/__tests__/triggers/schedule.test.ts +369 -0
  235. package/src/server/actions/index.ts +25 -0
  236. package/src/server/actions/nodes.ts +214 -0
  237. package/src/server/actions/workflows.ts +178 -0
  238. package/src/server/collections/executions.ts +35 -0
  239. package/src/server/collections/flow_nodes.ts +54 -0
  240. package/src/server/collections/jobs.ts +31 -0
  241. package/src/server/collections/workflows.ts +88 -0
  242. package/src/server/constants.ts +26 -0
  243. package/src/server/fields/expression-field.ts +11 -0
  244. package/src/server/fields/index.ts +7 -0
  245. package/src/server/functions/index.ts +16 -0
  246. package/src/server/index.ts +6 -0
  247. package/src/server/instructions/aggregate.ts +42 -0
  248. package/src/server/instructions/calculation.ts +41 -0
  249. package/src/server/instructions/condition.ts +172 -0
  250. package/src/server/instructions/create.ts +39 -0
  251. package/src/server/instructions/delay.ts +105 -0
  252. package/src/server/instructions/destroy.ts +23 -0
  253. package/src/server/instructions/index.ts +64 -0
  254. package/src/server/instructions/loop.ts +99 -0
  255. package/src/server/instructions/manual/actions.ts +91 -0
  256. package/src/server/instructions/manual/collecions/jobs.ts +17 -0
  257. package/src/server/instructions/manual/collecions/users.ts +15 -0
  258. package/src/server/instructions/manual/collecions/users_jobs.ts +50 -0
  259. package/src/server/instructions/manual/forms/create.ts +23 -0
  260. package/src/server/instructions/manual/forms/index.ts +12 -0
  261. package/src/server/instructions/manual/forms/update.ts +23 -0
  262. package/src/server/instructions/manual/index.ts +184 -0
  263. package/src/server/instructions/parallel.ts +121 -0
  264. package/src/server/instructions/query.ts +42 -0
  265. package/src/server/instructions/request.ts +88 -0
  266. package/src/server/instructions/sql.ts +25 -0
  267. package/src/server/instructions/update.ts +24 -0
  268. package/src/server/migrations/20221129153547-calculation-variables.ts +64 -0
  269. package/src/server/migrations/20230221032941-change-request-body-type.ts +76 -0
  270. package/src/server/migrations/20230221071831-calculation-expression.ts +102 -0
  271. package/src/server/migrations/20230221121203-condition-calculation.ts +82 -0
  272. package/src/server/migrations/20230221162902-jsonb-to-json.ts +51 -0
  273. package/src/server/migrations/20230411034722-manual-multi-form.ts +282 -0
  274. package/src/server/migrations/20230612021134-manual-collection-block.ts +138 -0
  275. package/src/server/migrations/20230710115902-manual-action-values.ts +78 -0
  276. package/src/server/triggers/collection.ts +146 -0
  277. package/src/server/triggers/index.ts +22 -0
  278. package/src/server/triggers/schedule.ts +567 -0
  279. package/src/server/types/Execution.ts +26 -0
  280. package/src/server/types/FlowNode.ts +21 -0
  281. package/src/server/types/Job.ts +18 -0
  282. package/src/server/types/Workflow.ts +36 -0
  283. package/src/server/types/index.ts +4 -0
  284. package/src/server/utils.ts +17 -0
  285. package/lib/client/triggers/schedule/DateFieldsSelect.d.ts +0 -2
  286. /package/lib/server/{models → types}/Execution.js +0 -0
  287. /package/lib/server/{models → types}/FlowNode.js +0 -0
  288. /package/lib/server/{models → types}/Job.js +0 -0
  289. /package/lib/server/{models → types}/Workflow.js +0 -0
@@ -0,0 +1,354 @@
1
+ import { Model, Transaction, Transactionable } from '@nocobase/database';
2
+ import { appendArrayColumn } from '@nocobase/evaluators';
3
+ import { Logger } from '@nocobase/logger';
4
+ import { parse } from '@nocobase/utils';
5
+ import Plugin from '.';
6
+ import { EXECUTION_STATUS, JOB_STATUS } from './constants';
7
+ import { Runner } from './instructions';
8
+ import type { ExecutionModel, FlowNodeModel, JobModel } from './types';
9
+
10
+ export interface ProcessorOptions extends Transactionable {
11
+ plugin: Plugin;
12
+ }
13
+
14
+ export default class Processor {
15
+ static StatusMap = {
16
+ [JOB_STATUS.PENDING]: EXECUTION_STATUS.STARTED,
17
+ [JOB_STATUS.RESOLVED]: EXECUTION_STATUS.RESOLVED,
18
+ [JOB_STATUS.FAILED]: EXECUTION_STATUS.FAILED,
19
+ [JOB_STATUS.ERROR]: EXECUTION_STATUS.ERROR,
20
+ [JOB_STATUS.ABORTED]: EXECUTION_STATUS.ABORTED,
21
+ [JOB_STATUS.CANCELED]: EXECUTION_STATUS.CANCELED,
22
+ [JOB_STATUS.REJECTED]: EXECUTION_STATUS.REJECTED,
23
+ };
24
+
25
+ logger: Logger;
26
+
27
+ transaction?: Transaction;
28
+
29
+ nodes: FlowNodeModel[] = [];
30
+ nodesMap = new Map<number, FlowNodeModel>();
31
+ jobsMap = new Map<number, JobModel>();
32
+ jobsMapByNodeId: { [key: number]: any } = {};
33
+
34
+ constructor(public execution: ExecutionModel, public options: ProcessorOptions) {
35
+ this.logger = options.plugin.getLogger(execution.workflowId);
36
+ }
37
+
38
+ // make dual linked nodes list then cache
39
+ private makeNodes(nodes: FlowNodeModel[] = []) {
40
+ this.nodes = nodes;
41
+
42
+ nodes.forEach((node) => {
43
+ this.nodesMap.set(node.id, node);
44
+ });
45
+
46
+ nodes.forEach((node) => {
47
+ if (node.upstreamId) {
48
+ node.upstream = this.nodesMap.get(node.upstreamId) as FlowNodeModel;
49
+ }
50
+
51
+ if (node.downstreamId) {
52
+ node.downstream = this.nodesMap.get(node.downstreamId) as FlowNodeModel;
53
+ }
54
+ });
55
+ }
56
+
57
+ private makeJobs(jobs: Array<JobModel>) {
58
+ jobs.forEach((job) => {
59
+ this.jobsMap.set(job.id, job);
60
+ // TODO: should consider cycle, and from previous job
61
+ this.jobsMapByNodeId[job.nodeId] = job.result;
62
+ });
63
+ }
64
+
65
+ private async getTransaction() {
66
+ if (!this.execution.useTransaction) {
67
+ return;
68
+ }
69
+
70
+ const { options } = this;
71
+
72
+ // @ts-ignore
73
+ return options.transaction && !options.transaction.finished
74
+ ? options.transaction
75
+ : await options.plugin.db.sequelize.transaction();
76
+ }
77
+
78
+ public async prepare() {
79
+ const transaction = await this.getTransaction();
80
+ this.transaction = transaction;
81
+
82
+ const { execution } = this;
83
+ if (!execution.workflow) {
84
+ execution.workflow = await execution.getWorkflow({ transaction });
85
+ }
86
+
87
+ const nodes = await execution.workflow.getNodes({ transaction });
88
+
89
+ this.makeNodes(nodes);
90
+
91
+ const jobs = await execution.getJobs({
92
+ order: [['id', 'ASC']],
93
+ transaction,
94
+ });
95
+
96
+ this.makeJobs(jobs);
97
+ }
98
+
99
+ public async start() {
100
+ const { execution } = this;
101
+ if (execution.status !== EXECUTION_STATUS.STARTED) {
102
+ throw new Error(`execution was ended with status ${execution.status}`);
103
+ }
104
+ await this.prepare();
105
+ if (this.nodes.length) {
106
+ const head = this.nodes.find((item) => !item.upstream);
107
+ await this.run(head, { result: execution.context });
108
+ } else {
109
+ await this.exit(JOB_STATUS.RESOLVED);
110
+ }
111
+ }
112
+
113
+ public async resume(job: JobModel) {
114
+ const { execution } = this;
115
+ if (execution.status !== EXECUTION_STATUS.STARTED) {
116
+ throw new Error(`execution was ended with status ${execution.status}`);
117
+ }
118
+ await this.prepare();
119
+ const node = this.nodesMap.get(job.nodeId);
120
+ await this.recall(node, job);
121
+ }
122
+
123
+ private async commit() {
124
+ // @ts-ignore
125
+ if (this.transaction && (!this.options.transaction || this.options.transaction.finished)) {
126
+ await this.transaction.commit();
127
+ }
128
+ }
129
+
130
+ private async exec(instruction: Runner, node: FlowNodeModel, prevJob) {
131
+ let job;
132
+ try {
133
+ // call instruction to get result and status
134
+ this.logger.info(`execution (${this.execution.id}) run instruction [${node.type}] for node (${node.id})`);
135
+ this.logger.debug(`config of node`, { data: node.config });
136
+ job = await instruction(node, prevJob, this);
137
+ if (!job) {
138
+ return null;
139
+ }
140
+ } catch (err) {
141
+ // for uncaught error, set to error
142
+ this.logger.error(
143
+ `execution (${this.execution.id}) run instruction [${node.type}] for node (${node.id}) failed: `,
144
+ { error: err },
145
+ );
146
+ job = {
147
+ result:
148
+ err instanceof Error
149
+ ? { message: err.message, stack: process.env.NODE_ENV === 'production' ? [] : err.stack }
150
+ : err,
151
+ status: JOB_STATUS.ERROR,
152
+ };
153
+ // if previous job is from resuming
154
+ if (prevJob && prevJob.nodeId === node.id) {
155
+ prevJob.set(job);
156
+ job = prevJob;
157
+ }
158
+ }
159
+
160
+ if (!(job instanceof Model)) {
161
+ job.upstreamId = prevJob instanceof Model ? prevJob.get('id') : null;
162
+ job.nodeId = node.id;
163
+ }
164
+ const savedJob = await this.saveJob(job);
165
+
166
+ this.logger.info(
167
+ `execution (${this.execution.id}) run instruction [${node.type}] for node (${node.id}) finished as status: ${savedJob.status}`,
168
+ );
169
+ this.logger.debug(`result of node`, { data: savedJob.result });
170
+
171
+ if (savedJob.status === JOB_STATUS.RESOLVED && node.downstream) {
172
+ // run next node
173
+ this.logger.debug(`run next node (${node.downstreamId})`);
174
+ return this.run(node.downstream, savedJob);
175
+ }
176
+
177
+ // all nodes in scope have been executed
178
+ return this.end(node, savedJob);
179
+ }
180
+
181
+ public async run(node, input?) {
182
+ const { instructions } = this.options.plugin;
183
+ const instruction = instructions.get(node.type);
184
+ if (typeof instruction.run !== 'function') {
185
+ return Promise.reject(new Error('`run` should be implemented for customized execution of the node'));
186
+ }
187
+
188
+ return this.exec(instruction.run.bind(instruction), node, input);
189
+ }
190
+
191
+ // parent node should take over the control
192
+ public async end(node, job: JobModel) {
193
+ this.logger.debug(`branch ended at node (${node.id})`);
194
+ const parentNode = this.findBranchParentNode(node);
195
+ // no parent, means on main flow
196
+ if (parentNode) {
197
+ this.logger.debug(`not on main, recall to parent entry node (${node.id})})`);
198
+ await this.recall(parentNode, job);
199
+ return job;
200
+ }
201
+
202
+ // really done for all nodes
203
+ // * should mark execution as done with last job status
204
+ return this.exit(job.status);
205
+ }
206
+
207
+ async recall(node, job) {
208
+ const { instructions } = this.options.plugin;
209
+ const instruction = instructions.get(node.type);
210
+ if (typeof instruction.resume !== 'function') {
211
+ return Promise.reject(new Error('`resume` should be implemented'));
212
+ }
213
+
214
+ return this.exec(instruction.resume.bind(instruction), node, job);
215
+ }
216
+
217
+ async exit(s?: number) {
218
+ if (typeof s === 'number') {
219
+ const status = (<typeof Processor>this.constructor).StatusMap[s] ?? Math.sign(s);
220
+ await this.execution.update({ status }, { transaction: this.transaction });
221
+ }
222
+ this.logger.info(`execution (${this.execution.id}) exiting with status ${this.execution.status}`);
223
+ await this.commit();
224
+ return null;
225
+ }
226
+
227
+ // TODO(optimize)
228
+ async saveJob(payload) {
229
+ const { database } = <typeof ExecutionModel>this.execution.constructor;
230
+ const { model } = database.getCollection('jobs');
231
+ let job;
232
+ if (payload instanceof model) {
233
+ job = await payload.save({ transaction: this.transaction });
234
+ } else if (payload.id) {
235
+ job = await model.findByPk(payload.id);
236
+ await job.update(payload, {
237
+ transaction: this.transaction,
238
+ });
239
+ } else {
240
+ job = await model.create(
241
+ {
242
+ ...payload,
243
+ executionId: this.execution.id,
244
+ },
245
+ {
246
+ transaction: this.transaction,
247
+ },
248
+ );
249
+ }
250
+ this.jobsMap.set(job.id, job);
251
+ this.jobsMapByNodeId[job.nodeId] = job.result;
252
+
253
+ return job;
254
+ }
255
+
256
+ getBranches(node: FlowNodeModel): FlowNodeModel[] {
257
+ return this.nodes
258
+ .filter((item) => item.upstream === node && item.branchIndex !== null)
259
+ .sort((a, b) => Number(a.branchIndex) - Number(b.branchIndex));
260
+ }
261
+
262
+ // find the first node in current branch
263
+ findBranchStartNode(node: FlowNodeModel, parent?: FlowNodeModel): FlowNodeModel | null {
264
+ for (let n = node; n; n = n.upstream) {
265
+ if (!parent) {
266
+ if (n.branchIndex !== null) {
267
+ return n;
268
+ }
269
+ } else {
270
+ if (n.upstream === parent) {
271
+ return n;
272
+ }
273
+ }
274
+ }
275
+ return null;
276
+ }
277
+
278
+ // find the node start current branch
279
+ findBranchParentNode(node: FlowNodeModel): FlowNodeModel | null {
280
+ for (let n = node; n; n = n.upstream) {
281
+ if (n.branchIndex !== null) {
282
+ return n.upstream;
283
+ }
284
+ }
285
+ return null;
286
+ }
287
+
288
+ findBranchEndNode(node: FlowNodeModel): FlowNodeModel | null {
289
+ for (let n = node; n; n = n.downstream) {
290
+ if (!n.downstream) {
291
+ return n;
292
+ }
293
+ }
294
+ return null;
295
+ }
296
+
297
+ findBranchParentJob(job: JobModel, node: FlowNodeModel): JobModel | null {
298
+ for (let j: JobModel | undefined = job; j; j = this.jobsMap.get(j.upstreamId)) {
299
+ if (j.nodeId === node.id) {
300
+ return j;
301
+ }
302
+ }
303
+ return null;
304
+ }
305
+
306
+ findBranchLastJob(node: FlowNodeModel): JobModel | null {
307
+ for (let n = this.findBranchEndNode(node); n && n !== node.upstream; n = n.upstream) {
308
+ const jobs = Array.from(this.jobsMap.values())
309
+ .filter((item) => item.nodeId === n.id)
310
+ .sort((a, b) => a.createdAt.getTime() - b.createdAt.getTime());
311
+ if (jobs.length) {
312
+ return jobs[jobs.length - 1];
313
+ }
314
+ }
315
+ return null;
316
+ }
317
+
318
+ public getScope(node?) {
319
+ const systemFns = {};
320
+ const scope = {
321
+ execution: this.execution,
322
+ node,
323
+ };
324
+ for (const [name, fn] of this.options.plugin.functions.getEntities()) {
325
+ systemFns[name] = fn.bind(scope);
326
+ }
327
+
328
+ const $scopes = {};
329
+ if (node) {
330
+ for (let n = this.findBranchParentNode(node); n; n = this.findBranchParentNode(n)) {
331
+ const instruction = this.options.plugin.instructions.get(n.type);
332
+ if (typeof instruction.getScope === 'function') {
333
+ $scopes[n.id] = instruction.getScope(n, this.jobsMapByNodeId[n.id], this);
334
+ }
335
+ }
336
+ }
337
+
338
+ return {
339
+ $context: this.execution.context,
340
+ $jobsMapByNodeId: this.jobsMapByNodeId,
341
+ $system: systemFns,
342
+ $scopes,
343
+ };
344
+ }
345
+
346
+ public getParsedValue(value, node?, additionalScope?: object) {
347
+ const template = parse(value);
348
+ const scope = Object.assign(this.getScope(node), additionalScope);
349
+ template.parameters.forEach(({ key }) => {
350
+ appendArrayColumn(scope, key);
351
+ });
352
+ return template(scope);
353
+ }
354
+ }