@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,294 @@
1
+ import { Application } from '@nocobase/server';
2
+ import Database from '@nocobase/database';
3
+ import { getApp, sleep } from '..';
4
+
5
+ describe('workflow > instructions > aggregate', () => {
6
+ let app: Application;
7
+ let db: Database;
8
+ let PostRepo;
9
+ let CommentRepo;
10
+ let TagRepo;
11
+ let WorkflowModel;
12
+ let workflow;
13
+
14
+ beforeEach(async () => {
15
+ app = await getApp();
16
+
17
+ db = app.db;
18
+ WorkflowModel = db.getCollection('workflows').model;
19
+ PostRepo = db.getCollection('posts').repository;
20
+ CommentRepo = db.getCollection('comments').repository;
21
+ TagRepo = db.getCollection('tags').repository;
22
+
23
+ workflow = await WorkflowModel.create({
24
+ title: 'test workflow',
25
+ enabled: true,
26
+ type: 'collection',
27
+ config: {
28
+ mode: 1,
29
+ collection: 'posts',
30
+ },
31
+ });
32
+ });
33
+
34
+ afterEach(() => db.close());
35
+
36
+ describe('based on collection', () => {
37
+ it('count', async () => {
38
+ const n1 = await workflow.createNode({
39
+ type: 'aggregate',
40
+ config: {
41
+ aggregator: 'count',
42
+ collection: 'posts',
43
+ params: {
44
+ field: 'id',
45
+ },
46
+ },
47
+ });
48
+
49
+ const post = await PostRepo.create({ values: { title: 't1' } });
50
+
51
+ await sleep(500);
52
+
53
+ const [execution] = await workflow.getExecutions();
54
+ const [job] = await execution.getJobs();
55
+ expect(job.result).toBe(1);
56
+ });
57
+
58
+ it('sum', async () => {
59
+ const n1 = await workflow.createNode({
60
+ type: 'aggregate',
61
+ config: {
62
+ aggregator: 'sum',
63
+ collection: 'posts',
64
+ params: {
65
+ field: 'read',
66
+ },
67
+ },
68
+ });
69
+
70
+ const p1 = await PostRepo.create({ values: { title: 't1', read: 1 } });
71
+
72
+ await sleep(500);
73
+
74
+ const [e1] = await workflow.getExecutions();
75
+ const [j1] = await e1.getJobs();
76
+ expect(j1.result).toBe(1);
77
+
78
+ const p2 = await PostRepo.create({ values: { title: 't2', read: 2 } });
79
+
80
+ await sleep(500);
81
+
82
+ const [e2] = await workflow.getExecutions({ order: [['id', 'desc']] });
83
+ const [j2] = await e2.getJobs();
84
+ expect(j2.result).toBe(3);
85
+ });
86
+
87
+ it('avg', async () => {
88
+ const n1 = await workflow.createNode({
89
+ type: 'aggregate',
90
+ config: {
91
+ aggregator: 'avg',
92
+ collection: 'posts',
93
+ params: {
94
+ field: 'read',
95
+ },
96
+ },
97
+ });
98
+
99
+ const p1 = await PostRepo.create({ values: { title: 't1', read: 1 } });
100
+
101
+ await sleep(500);
102
+
103
+ const [e1] = await workflow.getExecutions();
104
+ const [j1] = await e1.getJobs();
105
+ expect(j1.result).toBe(1);
106
+
107
+ const p2 = await PostRepo.create({ values: { title: 't2', read: 2 } });
108
+
109
+ await sleep(500);
110
+
111
+ const [e2] = await workflow.getExecutions({ order: [['id', 'desc']] });
112
+ const [j2] = await e2.getJobs();
113
+ expect(j2.result).toBe(1.5);
114
+ });
115
+
116
+ it('min', async () => {
117
+ const n1 = await workflow.createNode({
118
+ type: 'aggregate',
119
+ config: {
120
+ aggregator: 'min',
121
+ collection: 'posts',
122
+ params: {
123
+ field: 'read',
124
+ },
125
+ },
126
+ });
127
+
128
+ const p1 = await PostRepo.create({ values: { title: 't1', read: 1 } });
129
+
130
+ await sleep(500);
131
+
132
+ const [e1] = await workflow.getExecutions();
133
+ const [j1] = await e1.getJobs();
134
+ expect(j1.result).toBe(1);
135
+
136
+ const p2 = await PostRepo.create({ values: { title: 't2', read: 2 } });
137
+
138
+ await sleep(500);
139
+
140
+ const [e2] = await workflow.getExecutions({ order: [['id', 'desc']] });
141
+ const [j2] = await e2.getJobs();
142
+ expect(j2.result).toBe(1);
143
+ });
144
+
145
+ it('max', async () => {
146
+ const n1 = await workflow.createNode({
147
+ type: 'aggregate',
148
+ config: {
149
+ aggregator: 'max',
150
+ collection: 'posts',
151
+ params: {
152
+ field: 'read',
153
+ },
154
+ },
155
+ });
156
+
157
+ const p1 = await PostRepo.create({ values: { title: 't1', read: 1 } });
158
+
159
+ await sleep(500);
160
+
161
+ const [e1] = await workflow.getExecutions();
162
+ const [j1] = await e1.getJobs();
163
+ expect(j1.result).toBe(1);
164
+
165
+ const p2 = await PostRepo.create({ values: { title: 't2', read: 2 } });
166
+
167
+ await sleep(500);
168
+
169
+ const [e2] = await workflow.getExecutions({ order: [['id', 'desc']] });
170
+ const [j2] = await e2.getJobs();
171
+ expect(j2.result).toBe(2);
172
+ });
173
+ });
174
+
175
+ describe('based on data associated collection', () => {
176
+ it('count', async () => {
177
+ const n1 = await workflow.createNode({
178
+ type: 'aggregate',
179
+ config: {
180
+ aggregator: 'count',
181
+ collection: 'comments',
182
+ associated: true,
183
+ association: {
184
+ name: 'comments',
185
+ associatedKey: '{{$context.data.id}}',
186
+ associatedCollection: 'posts',
187
+ },
188
+ params: {
189
+ field: 'id',
190
+ },
191
+ },
192
+ });
193
+ const n2 = await workflow.createNode({
194
+ upstreamId: n1.id,
195
+ type: 'aggregate',
196
+ config: {
197
+ aggregator: 'count',
198
+ collection: 'comments',
199
+ associated: true,
200
+ association: {
201
+ name: 'comments',
202
+ associatedKey: '{{$context.data.id}}',
203
+ associatedCollection: 'posts',
204
+ },
205
+ params: {
206
+ field: 'id',
207
+ filter: {
208
+ $and: [{ status: 1 }],
209
+ },
210
+ },
211
+ },
212
+ });
213
+ await n1.setDownstream(n2);
214
+
215
+ await CommentRepo.create({ values: [{}, {}] });
216
+
217
+ const p1 = await PostRepo.create({ values: { title: 't1', comments: [{}, { status: 1 }] } });
218
+
219
+ await sleep(500);
220
+
221
+ const [e1] = await workflow.getExecutions();
222
+ const [j1, j2] = await e1.getJobs({ order: [['id', 'ASC']] });
223
+ expect(j1.result).toBe(2);
224
+ expect(j2.result).toBe(1);
225
+ });
226
+
227
+ it('sum', async () => {
228
+ const PostModel = db.getCollection('posts').model;
229
+ const p1 = await PostModel.create({ title: 't1', read: 1 });
230
+
231
+ const n1 = await workflow.createNode({
232
+ type: 'create',
233
+ config: {
234
+ collection: 'tags',
235
+ params: {
236
+ values: {
237
+ posts: [p1.id, '{{$context.data.id}}'],
238
+ },
239
+ },
240
+ },
241
+ });
242
+ const n2 = await workflow.createNode({
243
+ upstreamId: n1.id,
244
+ type: 'aggregate',
245
+ config: {
246
+ aggregator: 'sum',
247
+ collection: 'posts',
248
+ associated: true,
249
+ association: {
250
+ name: 'posts',
251
+ associatedKey: `{{$jobsMapByNodeId.${n1.id}.id}}`,
252
+ associatedCollection: 'tags',
253
+ },
254
+ params: {
255
+ field: 'read',
256
+ },
257
+ },
258
+ });
259
+ await n1.setDownstream(n2);
260
+ const n3 = await workflow.createNode({
261
+ upstreamId: n2.id,
262
+ type: 'aggregate',
263
+ config: {
264
+ aggregator: 'sum',
265
+ collection: 'posts',
266
+ associated: true,
267
+ association: {
268
+ name: 'posts',
269
+ associatedKey: `{{$jobsMapByNodeId.${n1.id}.id}}`,
270
+ associatedCollection: 'tags',
271
+ },
272
+ params: {
273
+ field: 'read',
274
+ filter: {
275
+ $and: [{ title: 't1' }],
276
+ },
277
+ },
278
+ },
279
+ });
280
+ await n2.setDownstream(n3);
281
+
282
+ await TagRepo.create({ values: [{}, {}] });
283
+
284
+ const p2 = await PostRepo.create({ values: { title: 't2', read: 2 } });
285
+
286
+ await sleep(500);
287
+
288
+ const [e1] = await workflow.getExecutions();
289
+ const [j1, j2, j3] = await e1.getJobs({ order: [['id', 'ASC']] });
290
+ expect(j2.result).toBe(3);
291
+ expect(j3.result).toBe(1);
292
+ });
293
+ });
294
+ });
@@ -0,0 +1,265 @@
1
+ import { Application } from '@nocobase/server';
2
+ import Database from '@nocobase/database';
3
+ import { getApp, sleep } from '..';
4
+ import { JOB_STATUS } from '../../constants';
5
+
6
+ describe('workflow > instructions > calculation', () => {
7
+ let app: Application;
8
+ let db: Database;
9
+ let PostRepo;
10
+ let CategoryRepo;
11
+ let WorkflowModel;
12
+ let workflow;
13
+
14
+ beforeEach(async () => {
15
+ app = await getApp();
16
+
17
+ db = app.db;
18
+ WorkflowModel = db.getCollection('workflows').model;
19
+ PostRepo = db.getCollection('posts').repository;
20
+ CategoryRepo = db.getCollection('categories').repository;
21
+
22
+ workflow = await WorkflowModel.create({
23
+ title: 'test workflow',
24
+ enabled: true,
25
+ type: 'collection',
26
+ config: {
27
+ mode: 1,
28
+ collection: 'posts',
29
+ },
30
+ });
31
+ });
32
+
33
+ afterEach(() => db.close());
34
+
35
+ describe('math.js', () => {
36
+ it('syntax error', async () => {
37
+ const n1 = await workflow.createNode({
38
+ type: 'calculation',
39
+ config: {
40
+ engine: 'math.js',
41
+ expression: '1 1',
42
+ },
43
+ });
44
+
45
+ const post = await PostRepo.create({ values: { title: 't1' } });
46
+
47
+ await sleep(500);
48
+
49
+ const [execution] = await workflow.getExecutions();
50
+ const [job] = await execution.getJobs();
51
+ expect(job.status).toBe(JOB_STATUS.ERROR);
52
+ expect(job.result.startsWith('SyntaxError: ')).toBe(true);
53
+ });
54
+
55
+ it('constant', async () => {
56
+ const n1 = await workflow.createNode({
57
+ type: 'calculation',
58
+ config: {
59
+ engine: 'math.js',
60
+ expression: ' 1 + 1 ',
61
+ },
62
+ });
63
+
64
+ const post = await PostRepo.create({ values: { title: 't1' } });
65
+
66
+ await sleep(500);
67
+
68
+ const [execution] = await workflow.getExecutions();
69
+ const [job] = await execution.getJobs();
70
+ expect(job.result).toBe(2);
71
+ });
72
+
73
+ it('$context', async () => {
74
+ const n1 = await workflow.createNode({
75
+ type: 'calculation',
76
+ config: {
77
+ engine: 'math.js',
78
+ expression: '{{$context.data.read}} + 1',
79
+ },
80
+ });
81
+
82
+ const post = await PostRepo.create({ values: { title: 't1', read: 1 } });
83
+
84
+ await sleep(500);
85
+
86
+ const [execution] = await workflow.getExecutions();
87
+ const [job] = await execution.getJobs();
88
+ expect(job.result).toBe(2);
89
+ });
90
+
91
+ it('$jobsMapByNodeId', async () => {
92
+ const n1 = await workflow.createNode({
93
+ type: 'echo',
94
+ });
95
+
96
+ const n2 = await workflow.createNode({
97
+ type: 'calculation',
98
+ config: {
99
+ engine: 'math.js',
100
+ expression: `{{$jobsMapByNodeId.${n1.id}.data.read}} + 1`,
101
+ },
102
+ upstreamId: n1.id,
103
+ });
104
+
105
+ await n1.setDownstream(n2);
106
+
107
+ const post = await PostRepo.create({ values: { title: 't1' } });
108
+
109
+ await sleep(500);
110
+
111
+ const [execution] = await workflow.getExecutions();
112
+ const [n1Job, n2Job] = await execution.getJobs({ order: [['id', 'ASC']] });
113
+ expect(n2Job.result).toBe(1);
114
+ });
115
+
116
+ it('$system', async () => {
117
+ const n1 = await workflow.createNode({
118
+ type: 'calculation',
119
+ config: {
120
+ engine: 'math.js',
121
+ expression: '1 + {{$system.no1}}',
122
+ },
123
+ });
124
+
125
+ const post = await PostRepo.create({ values: { title: 't1' } });
126
+
127
+ await sleep(500);
128
+
129
+ const [execution] = await workflow.getExecutions();
130
+ const [job] = await execution.getJobs();
131
+ expect(job.result).toBe(2);
132
+ });
133
+
134
+ it('$system.now()', async () => {
135
+ const n1 = await workflow.createNode({
136
+ type: 'calculation',
137
+ config: {
138
+ engine: 'math.js',
139
+ expression: '{{$system.now}}',
140
+ },
141
+ });
142
+
143
+ const post = await PostRepo.create({ values: { title: 't1' } });
144
+
145
+ await sleep(500);
146
+
147
+ const [execution] = await workflow.getExecutions();
148
+ const [job] = await execution.getJobs();
149
+ expect(job.result).toMatch(/\d{4}-\d{2}-\d{2}T\d{2}:\d{2}:\d{2}(\.\d{1,3})?Z/);
150
+ });
151
+ });
152
+
153
+ describe('formula.js', () => {
154
+ it('string variable with quote should be wrong result', async () => {
155
+ const n1 = await workflow.createNode({
156
+ type: 'calculation',
157
+ config: {
158
+ engine: 'formula.js',
159
+ expression: `CONCATENATE('a', '{{$context.data.title}}')`,
160
+ },
161
+ });
162
+
163
+ const post = await PostRepo.create({ values: { title: 't1' } });
164
+
165
+ await sleep(500);
166
+
167
+ const [execution] = await workflow.getExecutions();
168
+ const [job] = await execution.getJobs();
169
+ expect(job.result).toBe('a $context.data.title ');
170
+ });
171
+
172
+ it('text', async () => {
173
+ const n1 = await workflow.createNode({
174
+ type: 'calculation',
175
+ config: {
176
+ engine: 'formula.js',
177
+ expression: `CONCATENATE('a', {{$context.data.title}})`,
178
+ },
179
+ });
180
+
181
+ const post = await PostRepo.create({ values: { title: 't1' } });
182
+
183
+ await sleep(500);
184
+
185
+ const [execution] = await workflow.getExecutions();
186
+ const [job] = await execution.getJobs();
187
+ expect(job.result).toBe('at1');
188
+ });
189
+ });
190
+
191
+ describe('dynamic expression', () => {
192
+ it('dynamic expression field in current table', async () => {
193
+ const n1 = await workflow.createNode({
194
+ type: 'calculation',
195
+ config: {
196
+ dynamic: '{{$context.data.category}}',
197
+ scope: '{{$context.data}}',
198
+ },
199
+ });
200
+
201
+ const post = await PostRepo.create({
202
+ values: {
203
+ title: 't1',
204
+ category: {
205
+ engine: 'math.js',
206
+ expression: '1 + {{read}}',
207
+ },
208
+ },
209
+ });
210
+
211
+ await sleep(500);
212
+
213
+ const [execution] = await workflow.getExecutions();
214
+ const [job] = await execution.getJobs();
215
+ expect(job.result).toBe(1);
216
+ });
217
+
218
+ it('dynamic expression field in association table', async () => {
219
+ const n1 = await workflow.createNode({
220
+ type: 'query',
221
+ config: {
222
+ collection: 'categories',
223
+ params: {
224
+ filter: {
225
+ $and: [{ id: '{{$context.data.categoryId}}' }],
226
+ },
227
+ },
228
+ },
229
+ });
230
+
231
+ const n2 = await workflow.createNode({
232
+ type: 'calculation',
233
+ config: {
234
+ dynamic: `{{$jobsMapByNodeId.${n1.id}}}`,
235
+ scope: '{{$context.data}}',
236
+ },
237
+ upstreamId: n1.id,
238
+ });
239
+
240
+ await n1.setDownstream(n2);
241
+
242
+ const category = await CategoryRepo.create({
243
+ values: {
244
+ title: 'c1',
245
+ engine: 'math.js',
246
+ expression: '1 + {{read}}',
247
+ },
248
+ });
249
+
250
+ const post = await PostRepo.create({
251
+ values: {
252
+ title: 't1',
253
+ categoryId: category.id,
254
+ },
255
+ });
256
+
257
+ await sleep(500);
258
+
259
+ const [execution] = await workflow.getExecutions();
260
+ const jobs = await execution.getJobs({ order: [['id', 'ASC']] });
261
+ expect(jobs.length).toBe(2);
262
+ expect(jobs[1].result).toBe(1);
263
+ });
264
+ });
265
+ });