@nyaruka/temba-components 0.129.7 → 0.129.9

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 (269) hide show
  1. package/.devcontainer/Dockerfile +11 -4
  2. package/.devcontainer/devcontainer.json +3 -2
  3. package/.github/workflows/build.yml +4 -14
  4. package/CHANGELOG.md +29 -0
  5. package/demo/components/flow/example.html +1 -1
  6. package/demo/components/message-editor/example.html +125 -0
  7. package/demo/components/textinput/completion.html +1 -0
  8. package/demo/data/flows/food-order.json +12 -21
  9. package/demo/data/flows/sample-flow.json +210 -104
  10. package/dist/temba-components.js +715 -364
  11. package/dist/temba-components.js.map +1 -1
  12. package/out-tsc/src/display/Thumbnail.js +2 -1
  13. package/out-tsc/src/display/Thumbnail.js.map +1 -1
  14. package/out-tsc/src/events.js.map +1 -1
  15. package/out-tsc/src/excellent/helpers.js +2 -2
  16. package/out-tsc/src/excellent/helpers.js.map +1 -1
  17. package/out-tsc/src/flow/CanvasNode.js +25 -7
  18. package/out-tsc/src/flow/CanvasNode.js.map +1 -1
  19. package/out-tsc/src/flow/Editor.js +11 -1
  20. package/out-tsc/src/flow/Editor.js.map +1 -1
  21. package/out-tsc/src/flow/NodeEditor.js +342 -276
  22. package/out-tsc/src/flow/NodeEditor.js.map +1 -1
  23. package/out-tsc/src/flow/actions/add_input_labels.js +40 -0
  24. package/out-tsc/src/flow/actions/add_input_labels.js.map +1 -1
  25. package/out-tsc/src/flow/actions/call_llm.js +56 -3
  26. package/out-tsc/src/flow/actions/call_llm.js.map +1 -1
  27. package/out-tsc/src/flow/actions/call_webhook.js +26 -17
  28. package/out-tsc/src/flow/actions/call_webhook.js.map +1 -1
  29. package/out-tsc/src/flow/actions/open_ticket.js +65 -3
  30. package/out-tsc/src/flow/actions/open_ticket.js.map +1 -1
  31. package/out-tsc/src/flow/actions/send_msg.js +147 -6
  32. package/out-tsc/src/flow/actions/send_msg.js.map +1 -1
  33. package/out-tsc/src/flow/actions/set_run_result.js +75 -0
  34. package/out-tsc/src/flow/actions/set_run_result.js.map +1 -1
  35. package/out-tsc/src/flow/config.js +4 -0
  36. package/out-tsc/src/flow/config.js.map +1 -1
  37. package/out-tsc/src/flow/nodes/split_by_llm_categorize.js +227 -0
  38. package/out-tsc/src/flow/nodes/split_by_llm_categorize.js.map +1 -0
  39. package/out-tsc/src/flow/nodes/split_by_ticket.js +18 -0
  40. package/out-tsc/src/flow/nodes/split_by_ticket.js.map +1 -0
  41. package/out-tsc/src/flow/nodes/wait_for_response.js +27 -1
  42. package/out-tsc/src/flow/nodes/wait_for_response.js.map +1 -1
  43. package/out-tsc/src/flow/types.js +0 -65
  44. package/out-tsc/src/flow/types.js.map +1 -1
  45. package/out-tsc/src/form/ArrayEditor.js +87 -57
  46. package/out-tsc/src/form/ArrayEditor.js.map +1 -1
  47. package/out-tsc/src/form/BaseListEditor.js +19 -4
  48. package/out-tsc/src/form/BaseListEditor.js.map +1 -1
  49. package/out-tsc/src/form/FieldRenderer.js +305 -0
  50. package/out-tsc/src/form/FieldRenderer.js.map +1 -0
  51. package/out-tsc/src/form/FormField.js +4 -4
  52. package/out-tsc/src/form/FormField.js.map +1 -1
  53. package/out-tsc/src/form/KeyValueEditor.js +1 -1
  54. package/out-tsc/src/form/KeyValueEditor.js.map +1 -1
  55. package/out-tsc/src/form/MediaPicker.js +13 -1
  56. package/out-tsc/src/form/MediaPicker.js.map +1 -1
  57. package/out-tsc/src/form/MessageEditor.js +422 -0
  58. package/out-tsc/src/form/MessageEditor.js.map +1 -0
  59. package/out-tsc/src/form/TextInput.js +13 -6
  60. package/out-tsc/src/form/TextInput.js.map +1 -1
  61. package/out-tsc/src/form/select/Select.js +52 -24
  62. package/out-tsc/src/form/select/Select.js.map +1 -1
  63. package/out-tsc/src/live/ContactChat.js +66 -15
  64. package/out-tsc/src/live/ContactChat.js.map +1 -1
  65. package/out-tsc/src/markdown.js +13 -11
  66. package/out-tsc/src/markdown.js.map +1 -1
  67. package/out-tsc/temba-modules.js +2 -0
  68. package/out-tsc/temba-modules.js.map +1 -1
  69. package/out-tsc/test/ActionHelper.js +2 -0
  70. package/out-tsc/test/ActionHelper.js.map +1 -1
  71. package/out-tsc/test/NodeHelper.js +148 -0
  72. package/out-tsc/test/NodeHelper.js.map +1 -0
  73. package/out-tsc/test/actions/call_llm.test.js +103 -0
  74. package/out-tsc/test/actions/call_llm.test.js.map +1 -0
  75. package/out-tsc/test/nodes/split_by_llm_categorize.test.js +532 -0
  76. package/out-tsc/test/nodes/split_by_llm_categorize.test.js.map +1 -0
  77. package/out-tsc/test/nodes/split_by_random.test.js +150 -0
  78. package/out-tsc/test/nodes/split_by_random.test.js.map +1 -0
  79. package/out-tsc/test/nodes/wait_for_digits.test.js +150 -0
  80. package/out-tsc/test/nodes/wait_for_digits.test.js.map +1 -0
  81. package/out-tsc/test/nodes/wait_for_response.test.js +171 -0
  82. package/out-tsc/test/nodes/wait_for_response.test.js.map +1 -0
  83. package/out-tsc/test/temba-add-input-labels.test.js +70 -0
  84. package/out-tsc/test/temba-add-input-labels.test.js.map +1 -0
  85. package/out-tsc/test/temba-field-config.test.js +4 -2
  86. package/out-tsc/test/temba-field-config.test.js.map +1 -1
  87. package/out-tsc/test/temba-field-renderer.test.js +296 -0
  88. package/out-tsc/test/temba-field-renderer.test.js.map +1 -0
  89. package/out-tsc/test/temba-markdown.test.js +1 -1
  90. package/out-tsc/test/temba-markdown.test.js.map +1 -1
  91. package/out-tsc/test/temba-message-editor.test.js +194 -0
  92. package/out-tsc/test/temba-message-editor.test.js.map +1 -0
  93. package/out-tsc/test/temba-node-editor.test.js +471 -0
  94. package/out-tsc/test/temba-node-editor.test.js.map +1 -1
  95. package/out-tsc/test/temba-select.test.js +7 -4
  96. package/out-tsc/test/temba-select.test.js.map +1 -1
  97. package/out-tsc/test/temba-textinput.test.js +16 -0
  98. package/out-tsc/test/temba-textinput.test.js.map +1 -1
  99. package/out-tsc/test/temba-webchat.test.js +5 -1
  100. package/out-tsc/test/temba-webchat.test.js.map +1 -1
  101. package/out-tsc/test/utils.test.js +2 -8
  102. package/out-tsc/test/utils.test.js.map +1 -1
  103. package/package.json +7 -4
  104. package/screenshots/truth/actions/add_contact_groups/editor/descriptive-group-names.png +0 -0
  105. package/screenshots/truth/actions/add_contact_groups/editor/long-group-names.png +0 -0
  106. package/screenshots/truth/actions/add_contact_groups/editor/many-groups.png +0 -0
  107. package/screenshots/truth/actions/add_contact_groups/editor/multiple-groups.png +0 -0
  108. package/screenshots/truth/actions/add_contact_groups/editor/single-group.png +0 -0
  109. package/screenshots/truth/actions/call_llm/editor/information-extraction.png +0 -0
  110. package/screenshots/truth/actions/call_llm/editor/sentiment-analysis.png +0 -0
  111. package/screenshots/truth/actions/call_llm/editor/summarization.png +0 -0
  112. package/screenshots/truth/actions/call_llm/editor/translation-task.png +0 -0
  113. package/screenshots/truth/actions/call_llm/render/information-extraction.png +0 -0
  114. package/screenshots/truth/actions/call_llm/render/sentiment-analysis.png +0 -0
  115. package/screenshots/truth/actions/call_llm/render/summarization.png +0 -0
  116. package/screenshots/truth/actions/call_llm/render/translation-task.png +0 -0
  117. package/screenshots/truth/actions/remove_contact_groups/editor/cleanup-groups.png +0 -0
  118. package/screenshots/truth/actions/remove_contact_groups/editor/long-descriptive-group-names.png +0 -0
  119. package/screenshots/truth/actions/remove_contact_groups/editor/many-groups.png +0 -0
  120. package/screenshots/truth/actions/remove_contact_groups/editor/multiple-groups.png +0 -0
  121. package/screenshots/truth/actions/remove_contact_groups/editor/remove-from-all-groups.png +0 -0
  122. package/screenshots/truth/actions/remove_contact_groups/editor/single-group.png +0 -0
  123. package/screenshots/truth/actions/send_email/editor/complex-business-email.png +0 -0
  124. package/screenshots/truth/actions/send_email/editor/empty-body.png +0 -0
  125. package/screenshots/truth/actions/send_email/editor/empty-subject.png +0 -0
  126. package/screenshots/truth/actions/send_email/editor/long-subject.png +0 -0
  127. package/screenshots/truth/actions/send_email/editor/multiline-body.png +0 -0
  128. package/screenshots/truth/actions/send_email/editor/multiple-recipients.png +0 -0
  129. package/screenshots/truth/actions/send_email/editor/simple-email.png +0 -0
  130. package/screenshots/truth/actions/send_email/editor/with-expressions.png +0 -0
  131. package/screenshots/truth/actions/send_msg/editor/long-quick-replies.png +0 -0
  132. package/screenshots/truth/actions/send_msg/editor/multiline-text-with-replies.png +0 -0
  133. package/screenshots/truth/actions/send_msg/editor/simple-text.png +0 -0
  134. package/screenshots/truth/actions/send_msg/editor/text-with-linebreaks.png +0 -0
  135. package/screenshots/truth/actions/send_msg/editor/text-with-many-quick-replies.png +0 -0
  136. package/screenshots/truth/actions/send_msg/editor/text-with-quick-replies.png +0 -0
  137. package/screenshots/truth/actions/send_msg/editor/text-without-quick-replies.png +0 -0
  138. package/screenshots/truth/editor/router.png +0 -0
  139. package/screenshots/truth/editor/send_msg.png +0 -0
  140. package/screenshots/truth/editor/set_contact_language.png +0 -0
  141. package/screenshots/truth/editor/set_contact_name.png +0 -0
  142. package/screenshots/truth/editor/set_run_result.png +0 -0
  143. package/screenshots/truth/editor/wait.png +0 -0
  144. package/screenshots/truth/field-renderer/checkbox-checked.png +0 -0
  145. package/screenshots/truth/field-renderer/checkbox-unchecked.png +0 -0
  146. package/screenshots/truth/field-renderer/checkbox-with-errors.png +0 -0
  147. package/screenshots/truth/field-renderer/context-comparison.png +0 -0
  148. package/screenshots/truth/field-renderer/key-value-with-label.png +0 -0
  149. package/screenshots/truth/field-renderer/message-editor-with-label.png +0 -0
  150. package/screenshots/truth/field-renderer/select-multi.png +0 -0
  151. package/screenshots/truth/field-renderer/select-no-label.png +0 -0
  152. package/screenshots/truth/field-renderer/select-with-label.png +0 -0
  153. package/screenshots/truth/field-renderer/text-evaluated.png +0 -0
  154. package/screenshots/truth/field-renderer/text-no-label.png +0 -0
  155. package/screenshots/truth/field-renderer/text-with-errors.png +0 -0
  156. package/screenshots/truth/field-renderer/text-with-label.png +0 -0
  157. package/screenshots/truth/field-renderer/textarea-evaluated.png +0 -0
  158. package/screenshots/truth/field-renderer/textarea-with-label.png +0 -0
  159. package/screenshots/truth/formfield/markdown-errors.png +0 -0
  160. package/screenshots/truth/formfield/no-errors.png +0 -0
  161. package/screenshots/truth/formfield/plain-text-errors.png +0 -0
  162. package/screenshots/truth/message-editor/autogrow-initial-content.png +0 -0
  163. package/screenshots/truth/message-editor/default.png +0 -0
  164. package/screenshots/truth/message-editor/drag-highlight.png +0 -0
  165. package/screenshots/truth/message-editor/filtered-attachments.png +0 -0
  166. package/screenshots/truth/message-editor/with-completion.png +0 -0
  167. package/screenshots/truth/message-editor/with-properties.png +0 -0
  168. package/screenshots/truth/nodes/split_by_llm_categorize/editor/basic-categorization.png +0 -0
  169. package/screenshots/truth/nodes/split_by_llm_categorize/editor/custom-input-and-result-name.png +0 -0
  170. package/screenshots/truth/nodes/split_by_llm_categorize/editor/feedback-categorization.png +0 -0
  171. package/screenshots/truth/nodes/split_by_llm_categorize/editor/many-categories.png +0 -0
  172. package/screenshots/truth/nodes/split_by_llm_categorize/editor/minimal-categories.png +0 -0
  173. package/screenshots/truth/nodes/split_by_llm_categorize/render/basic-categorization.png +0 -0
  174. package/screenshots/truth/nodes/split_by_llm_categorize/render/custom-input-and-result-name.png +0 -0
  175. package/screenshots/truth/nodes/split_by_llm_categorize/render/feedback-categorization.png +0 -0
  176. package/screenshots/truth/nodes/split_by_llm_categorize/render/many-categories.png +0 -0
  177. package/screenshots/truth/nodes/split_by_llm_categorize/render/minimal-categories.png +0 -0
  178. package/screenshots/truth/nodes/split_by_random/editor/ab-test-multiple-variants.png +0 -0
  179. package/screenshots/truth/nodes/split_by_random/editor/sampling-split.png +0 -0
  180. package/screenshots/truth/nodes/split_by_random/editor/three-way-split.png +0 -0
  181. package/screenshots/truth/nodes/split_by_random/editor/two-bucket-split.png +0 -0
  182. package/screenshots/truth/nodes/split_by_random/render/ab-test-multiple-variants.png +0 -0
  183. package/screenshots/truth/nodes/split_by_random/render/sampling-split.png +0 -0
  184. package/screenshots/truth/nodes/split_by_random/render/three-way-split.png +0 -0
  185. package/screenshots/truth/nodes/split_by_random/render/two-bucket-split.png +0 -0
  186. package/screenshots/truth/nodes/wait_for_digits/editor/basic-digits-wait.png +0 -0
  187. package/screenshots/truth/nodes/wait_for_digits/editor/phone-number-collection.png +0 -0
  188. package/screenshots/truth/nodes/wait_for_digits/editor/single-digit-with-timeout.png +0 -0
  189. package/screenshots/truth/nodes/wait_for_digits/editor/verification-code.png +0 -0
  190. package/screenshots/truth/nodes/wait_for_digits/render/basic-digits-wait.png +0 -0
  191. package/screenshots/truth/nodes/wait_for_digits/render/phone-number-collection.png +0 -0
  192. package/screenshots/truth/nodes/wait_for_digits/render/single-digit-with-timeout.png +0 -0
  193. package/screenshots/truth/nodes/wait_for_digits/render/verification-code.png +0 -0
  194. package/screenshots/truth/nodes/wait_for_response/editor/basic-wait.png +0 -0
  195. package/screenshots/truth/nodes/wait_for_response/editor/custom-result-name.png +0 -0
  196. package/screenshots/truth/nodes/wait_for_response/editor/no-timeout.png +0 -0
  197. package/screenshots/truth/nodes/wait_for_response/editor/short-timeout.png +0 -0
  198. package/screenshots/truth/nodes/wait_for_response/render/basic-wait.png +0 -0
  199. package/screenshots/truth/nodes/wait_for_response/render/custom-result-name.png +0 -0
  200. package/screenshots/truth/nodes/wait_for_response/render/no-timeout.png +0 -0
  201. package/screenshots/truth/nodes/wait_for_response/render/short-timeout.png +0 -0
  202. package/screenshots/truth/omnibox/selected.png +0 -0
  203. package/screenshots/truth/select/functions.png +0 -0
  204. package/screenshots/truth/select/multi-with-endpoint.png +0 -0
  205. package/screenshots/truth/select/search-enabled.png +0 -0
  206. package/screenshots/truth/textinput/autogrow-initial.png +0 -0
  207. package/screenshots/truth/textinput/input-form.png +0 -0
  208. package/src/display/Thumbnail.ts +2 -1
  209. package/src/events.ts +13 -1
  210. package/src/excellent/helpers.ts +2 -2
  211. package/src/flow/CanvasNode.ts +22 -1
  212. package/src/flow/Editor.ts +12 -1
  213. package/src/flow/NodeEditor.ts +412 -354
  214. package/src/flow/actions/add_input_labels.ts +45 -0
  215. package/src/flow/actions/call_llm.ts +57 -3
  216. package/src/flow/actions/call_webhook.ts +28 -18
  217. package/src/flow/actions/open_ticket.ts +74 -3
  218. package/src/flow/actions/send_msg.ts +170 -6
  219. package/src/flow/actions/set_run_result.ts +83 -0
  220. package/src/flow/config.ts +4 -0
  221. package/src/flow/nodes/split_by_llm_categorize.ts +277 -0
  222. package/src/flow/nodes/split_by_ticket.ts +19 -0
  223. package/src/flow/nodes/wait_for_response.ts +28 -1
  224. package/src/flow/types.ts +46 -128
  225. package/src/form/ArrayEditor.ts +96 -66
  226. package/src/form/BaseListEditor.ts +22 -6
  227. package/src/form/FieldRenderer.ts +465 -0
  228. package/src/form/FormField.ts +4 -4
  229. package/src/form/KeyValueEditor.ts +1 -1
  230. package/src/form/MediaPicker.ts +13 -1
  231. package/src/form/MessageEditor.ts +449 -0
  232. package/src/form/TextInput.ts +16 -8
  233. package/src/form/select/Select.ts +55 -24
  234. package/src/live/ContactChat.ts +69 -19
  235. package/src/markdown.ts +19 -11
  236. package/src/store/flow-definition.d.ts +5 -2
  237. package/static/api/labels.json +31 -0
  238. package/static/api/topics.json +24 -9
  239. package/static/api/users.json +35 -16
  240. package/static/css/temba-components.css +5 -3
  241. package/static/mr/docs/en-us/editor.json +2588 -0
  242. package/stress-test.js +143 -0
  243. package/temba-modules.ts +2 -0
  244. package/test/ActionHelper.ts +2 -0
  245. package/test/NodeHelper.ts +184 -0
  246. package/test/actions/call_llm.test.ts +137 -0
  247. package/test/nodes/README.md +78 -0
  248. package/test/nodes/split_by_llm_categorize.test.ts +698 -0
  249. package/test/nodes/split_by_random.test.ts +177 -0
  250. package/test/nodes/wait_for_digits.test.ts +176 -0
  251. package/test/nodes/wait_for_response.test.ts +206 -0
  252. package/test/temba-add-input-labels.test.ts +87 -0
  253. package/test/temba-field-config.test.ts +4 -2
  254. package/test/temba-field-renderer.test.ts +482 -0
  255. package/test/temba-markdown.test.ts +1 -1
  256. package/test/temba-message-editor.test.ts +300 -0
  257. package/test/temba-node-editor.test.ts +590 -0
  258. package/test/temba-select.test.ts +7 -7
  259. package/test/temba-textinput.test.ts +26 -0
  260. package/test/temba-webchat.test.ts +6 -1
  261. package/test/utils.test.ts +2 -13
  262. package/test-assets/contacts/history.json +19 -0
  263. package/test-assets/select/llms.json +18 -0
  264. package/test-assets/style.css +2 -0
  265. package/web-dev-mock.mjs +523 -0
  266. package/web-dev-server.config.mjs +74 -6
  267. package/web-test-runner.config.mjs +9 -4
  268. package/test/temba-flow-editor.test.ts.backup +0 -563
  269. package/test/temba-utils-index.test.ts.backup +0 -1737
package/stress-test.js ADDED
@@ -0,0 +1,143 @@
1
+ #!/usr/bin/env node
2
+
3
+ import { execSync } from 'child_process';
4
+ import { performance } from 'perf_hooks';
5
+
6
+ // Parse command line arguments
7
+ const args = process.argv.slice(2);
8
+ let testFile = '';
9
+ let maxRuns = 10;
10
+
11
+ // Parse arguments
12
+ for (let i = 0; i < args.length; i++) {
13
+ const arg = args[i];
14
+
15
+ if (arg.startsWith('--runs=')) {
16
+ maxRuns = parseInt(arg.split('=')[1]);
17
+ if (isNaN(maxRuns) || maxRuns <= 0) {
18
+ console.error('❌ Invalid runs value. Must be a positive integer.');
19
+ process.exit(1);
20
+ }
21
+ } else if (arg.startsWith('test/') && arg.endsWith('.test.ts')) {
22
+ testFile = arg;
23
+ } else if (!arg.startsWith('--')) {
24
+ testFile = arg;
25
+ }
26
+ }
27
+
28
+ // Validate test file
29
+ if (!testFile) {
30
+ console.error('❌ Usage: yarn stress-test <test-file> [--runs=N]');
31
+ console.error(
32
+ ' Example: yarn stress-test test/temba-webchat.test.ts --runs=100'
33
+ );
34
+ process.exit(1);
35
+ }
36
+
37
+ if (!testFile.startsWith('test/') || !testFile.endsWith('.test.ts')) {
38
+ console.error(
39
+ '❌ Test file must be in the test/ directory and end with .test.ts'
40
+ );
41
+ process.exit(1);
42
+ }
43
+
44
+ console.log(`🧪 Stress testing: ${testFile}`);
45
+ console.log(`🔄 Running up to ${maxRuns} times (or until failure)`);
46
+ console.log('');
47
+
48
+ let run = 1;
49
+ let totalTime = 0;
50
+ const runTimes = [];
51
+ let failures = 0;
52
+
53
+ const startTime = performance.now();
54
+
55
+ try {
56
+ while (run <= maxRuns) {
57
+ const runStartTime = performance.now();
58
+
59
+ process.stdout.write(`Run ${run.toString().padStart(3)}/${maxRuns}: `);
60
+
61
+ try {
62
+ // Run the test with minimal output
63
+ const result = execSync(`yarn test ${testFile}`, {
64
+ encoding: 'utf-8',
65
+ stdio: ['pipe', 'pipe', 'pipe']
66
+ });
67
+
68
+ const runEndTime = performance.now();
69
+ const runTime = runEndTime - runStartTime;
70
+ runTimes.push(runTime);
71
+ totalTime += runTime;
72
+
73
+ // Check if the test actually passed by looking for success indicators
74
+ if (result.includes('all tests passed') || result.includes('0 failed')) {
75
+ console.log(`✅ PASS (${(runTime / 1000).toFixed(2)}s)`);
76
+ } else {
77
+ console.log(`❌ FAIL (unexpected output)`);
78
+ console.log('Output:', result);
79
+ failures++;
80
+ break;
81
+ }
82
+ } catch (error) {
83
+ const runEndTime = performance.now();
84
+ const runTime = runEndTime - runStartTime;
85
+
86
+ console.log(`❌ FAIL (${(runTime / 1000).toFixed(2)}s)`);
87
+ console.log('');
88
+ console.log('💥 Test failed on run', run);
89
+ console.log('');
90
+ console.log('Error output:');
91
+ console.log(error.stdout || error.message);
92
+ if (error.stderr) {
93
+ console.log('');
94
+ console.log('Error details:');
95
+ console.log(error.stderr);
96
+ }
97
+ failures++;
98
+ break;
99
+ }
100
+
101
+ run++;
102
+ }
103
+ } catch (error) {
104
+ console.log('');
105
+ console.log('💥 Unexpected error:', error.message);
106
+ process.exit(1);
107
+ }
108
+
109
+ const endTime = performance.now();
110
+ const totalTestTime = endTime - startTime;
111
+
112
+ console.log('');
113
+ console.log('📊 Results Summary');
114
+ console.log('==================');
115
+ console.log(`Test file: ${testFile}`);
116
+ console.log(`Completed runs: ${run - 1}/${maxRuns}`);
117
+ console.log(`Failures: ${failures}`);
118
+ console.log(
119
+ `Success rate: ${(((run - 1 - failures) / (run - 1)) * 100).toFixed(1)}%`
120
+ );
121
+ console.log('');
122
+
123
+ if (runTimes.length > 0) {
124
+ const avgTime = runTimes.reduce((a, b) => a + b, 0) / runTimes.length;
125
+ const minTime = Math.min(...runTimes);
126
+ const maxTime = Math.max(...runTimes);
127
+
128
+ console.log('⏱️ Timing Statistics');
129
+ console.log('=====================');
130
+ console.log(`Total time: ${(totalTestTime / 1000).toFixed(2)}s`);
131
+ console.log(`Average run time: ${(avgTime / 1000).toFixed(2)}s`);
132
+ console.log(`Fastest run: ${(minTime / 1000).toFixed(2)}s`);
133
+ console.log(`Slowest run: ${(maxTime / 1000).toFixed(2)}s`);
134
+ console.log('');
135
+ }
136
+
137
+ if (failures === 0) {
138
+ console.log(`🎉 All ${run - 1} runs passed successfully!`);
139
+ process.exit(0);
140
+ } else {
141
+ console.log(`💥 Test failed after ${run - 1} runs`);
142
+ process.exit(1);
143
+ }
package/temba-modules.ts CHANGED
@@ -70,6 +70,7 @@ import { RangePicker } from './src/form/RangePicker';
70
70
  import { NodeEditor } from './src/flow/NodeEditor';
71
71
  import { KeyValueEditor } from './src/form/KeyValueEditor';
72
72
  import { TembaArrayEditor } from './src/form/ArrayEditor';
73
+ import { MessageEditor } from './src/form/MessageEditor';
73
74
  import './src/form/BaseListEditor'; // Import base class
74
75
 
75
76
  export function addCustomElement(name: string, comp: any) {
@@ -151,3 +152,4 @@ addCustomElement('temba-workspace-select', WorkspaceSelect);
151
152
  addCustomElement('temba-chart', TembaChart);
152
153
  addCustomElement('temba-key-value-editor', KeyValueEditor);
153
154
  addCustomElement('temba-array-editor', TembaArrayEditor);
155
+ addCustomElement('temba-message-editor', MessageEditor);
@@ -8,6 +8,8 @@ import '../temba-modules';
8
8
  /**
9
9
  * Generic action test framework
10
10
  * Tests the complete action lifecycle: render → edit → save → validate
11
+ *
12
+ * For node configuration testing, see NodeHelper.ts
11
13
  */
12
14
  export class ActionTest<T extends Action> {
13
15
  constructor(private actionConfig: any, private actionName: string) {}
@@ -0,0 +1,184 @@
1
+ import { fixture, expect } from '@open-wc/testing';
2
+ import { html } from 'lit';
3
+ import { Node } from '../src/store/flow-definition';
4
+ import { assertScreenshot, getClip } from './utils.test';
5
+ import { Editor } from '../src/flow/Editor';
6
+ import '../temba-modules';
7
+
8
+ /**
9
+ * Generic node test framework
10
+ * Tests the complete node lifecycle: render → edit → save → validate
11
+ *
12
+ * This is the node configuration equivalent of ActionHelper.ts for action configurations.
13
+ * It provides uniform testing for all types of nodes: simple wait nodes, router-based
14
+ * split nodes, and complex form-configured nodes.
15
+ */
16
+ export class NodeTest<T extends Node> {
17
+ constructor(private nodeConfig: any, private nodeName: string) {}
18
+
19
+ /**
20
+ * Renders a node in the flow editor and returns the flow node
21
+ */
22
+ private async renderNode(node: T, nodeUI: any): Promise<HTMLElement> {
23
+ const mockDefinition = {
24
+ nodes: [node],
25
+ _ui: {
26
+ nodes: {
27
+ [node.uuid]: {
28
+ type: nodeUI.type,
29
+ position: { left: 50, top: 50 },
30
+ ...nodeUI
31
+ }
32
+ }
33
+ }
34
+ };
35
+
36
+ const editor = (await fixture(html`
37
+ <temba-flow-editor>
38
+ <div id="canvas"></div>
39
+ </temba-flow-editor>
40
+ `)) as Editor;
41
+
42
+ (editor as any).definition = mockDefinition;
43
+ (editor as any).canvasSize = { width: 400, height: 300 };
44
+ await editor.updateComplete;
45
+
46
+ const flowNode = editor.querySelector('temba-flow-node') as HTMLElement;
47
+ expect(flowNode).to.exist;
48
+
49
+ return flowNode;
50
+ }
51
+
52
+ /**
53
+ * Opens the node editor for a node and returns the editor element
54
+ */
55
+ private async openNodeEditor(node: T, nodeUI: any): Promise<HTMLElement> {
56
+ const nodeEditor = (await fixture(html`
57
+ <temba-node-editor
58
+ .node=${node}
59
+ .nodeUI=${nodeUI}
60
+ .isOpen=${true}
61
+ ></temba-node-editor>
62
+ `)) as HTMLElement;
63
+
64
+ await (nodeEditor as any).updateComplete;
65
+
66
+ // Wait for form data initialization if needed
67
+ await new Promise((resolve) => setTimeout(resolve, 200));
68
+ await (nodeEditor as any).updateComplete;
69
+
70
+ expect(nodeEditor).to.exist;
71
+
72
+ return nodeEditor;
73
+ }
74
+
75
+ /**
76
+ * Takes a screenshot of the dialog container within a node editor
77
+ */
78
+ private async assertDialogScreenshot(
79
+ el: HTMLElement,
80
+ screenshotName: string
81
+ ) {
82
+ const dialog = el.shadowRoot
83
+ ?.querySelector('temba-dialog')
84
+ ?.shadowRoot?.querySelector('.dialog-container') as HTMLElement;
85
+ await assertScreenshot(screenshotName, getClip(dialog));
86
+ }
87
+
88
+ /**
89
+ * Complete test for a node configuration
90
+ * 1. Renders the node in a flow node (with screenshot)
91
+ * 2. Opens the node editor (with screenshot)
92
+ * 3. Simulates save and validates round-trip conversion
93
+ */
94
+ async testNode(node: T, nodeUI: any, testName: string) {
95
+ it(`${testName}`, async () => {
96
+ // Step 1: Render node in flow node
97
+ const flowNode = await this.renderNode(node, nodeUI);
98
+
99
+ // For execute_actions nodes, check for .body, for router nodes check for .router or .categories
100
+ const hasContent =
101
+ flowNode.querySelector('.body') ||
102
+ flowNode.querySelector('.router') ||
103
+ flowNode.querySelector('.categories') ||
104
+ flowNode.querySelector('.action') ||
105
+ flowNode.textContent?.trim();
106
+
107
+ expect(hasContent).to.exist;
108
+ await assertScreenshot(
109
+ `nodes/${this.nodeName}/render/${testName}`,
110
+ getClip(flowNode)
111
+ );
112
+
113
+ // Step 2: Open node editor
114
+ const nodeEditor = await this.openNodeEditor(node, nodeUI);
115
+ await this.assertDialogScreenshot(
116
+ nodeEditor,
117
+ `nodes/${this.nodeName}/editor/${testName}`
118
+ );
119
+
120
+ // Step 3: Test round-trip conversion (simulates save workflow)
121
+ if (this.nodeConfig.toFormData && this.nodeConfig.fromFormData) {
122
+ const formData = this.nodeConfig.toFormData(node);
123
+ const convertedNode = this.nodeConfig.fromFormData(formData, node) as T;
124
+
125
+ // Validate the round trip worked
126
+ expect(convertedNode.uuid).to.equal(node.uuid);
127
+
128
+ // Validate the converted node has expected structure
129
+ expect(convertedNode).to.have.property('actions');
130
+ expect(convertedNode).to.have.property('exits');
131
+
132
+ expect(convertedNode).to.deep.equal(node);
133
+ }
134
+ });
135
+ }
136
+
137
+ /**
138
+ * Run basic property tests
139
+ */
140
+ testBasicProperties() {
141
+ it('has correct basic properties', () => {
142
+ expect(this.nodeConfig.type).to.be.a('string');
143
+
144
+ // Name is optional - only some node configs have it
145
+ if (this.nodeConfig.name) {
146
+ expect(this.nodeConfig.name).to.be.a('string');
147
+ }
148
+
149
+ // Color is optional
150
+ if (this.nodeConfig.color) {
151
+ expect(this.nodeConfig.color).to.be.a('string');
152
+ }
153
+
154
+ // toFormData and fromFormData are optional - only needed for complex data transformations
155
+ if (this.nodeConfig.toFormData) {
156
+ expect(this.nodeConfig.toFormData).to.be.a('function');
157
+ }
158
+ if (this.nodeConfig.fromFormData) {
159
+ expect(this.nodeConfig.fromFormData).to.be.a('function');
160
+ }
161
+
162
+ // Form configuration is optional
163
+ if (this.nodeConfig.form) {
164
+ expect(this.nodeConfig.form).to.be.an('object');
165
+ }
166
+
167
+ // Layout is optional
168
+ if (this.nodeConfig.layout) {
169
+ expect(this.nodeConfig.layout).to.be.an('array');
170
+ }
171
+
172
+ // Router config is optional
173
+ if (this.nodeConfig.router) {
174
+ expect(this.nodeConfig.router).to.be.an('object');
175
+ expect(this.nodeConfig.router.type).to.exist;
176
+ }
177
+
178
+ // Render function is optional
179
+ if (this.nodeConfig.render) {
180
+ expect(this.nodeConfig.render).to.be.a('function');
181
+ }
182
+ });
183
+ }
184
+ }
@@ -0,0 +1,137 @@
1
+ import { expect } from '@open-wc/testing';
2
+ import { call_llm } from '../../src/flow/actions/call_llm';
3
+ import { CallLLM } from '../../src/store/flow-definition';
4
+ import { ActionTest } from '../ActionHelper';
5
+
6
+ /**
7
+ * Test suite for the call_llm action configuration.
8
+ */
9
+ describe('call_llm action config', () => {
10
+ const helper = new ActionTest(call_llm, 'call_llm');
11
+
12
+ describe('basic properties', () => {
13
+ helper.testBasicProperties();
14
+
15
+ it('has correct name', () => {
16
+ expect(call_llm.name).to.equal('Call AI');
17
+ });
18
+
19
+ it('has form configuration', () => {
20
+ expect(call_llm.form).to.exist;
21
+ expect(call_llm.form.llm).to.exist;
22
+ expect(call_llm.form.instructions).to.exist;
23
+ expect(call_llm.form.input).to.exist;
24
+ });
25
+
26
+ it('has layout configuration', () => {
27
+ expect(call_llm.layout).to.exist;
28
+ expect(call_llm.layout).to.deep.equal(['llm', 'input', 'instructions']);
29
+ });
30
+
31
+ it('has data transformation functions', () => {
32
+ expect(call_llm.toFormData).to.be.a('function');
33
+ expect(call_llm.fromFormData).to.be.a('function');
34
+ });
35
+ });
36
+
37
+ describe('data transformations', () => {
38
+ it('converts action to form data correctly', () => {
39
+ const action: CallLLM = {
40
+ uuid: 'test-llm-1',
41
+ type: 'call_llm',
42
+ input: '@input',
43
+ llm: { uuid: 'gpt-4', name: 'GPT 4.1' },
44
+ instructions: 'Translate to French',
45
+ result_name: 'translated_text'
46
+ };
47
+
48
+ const formData = call_llm.toFormData(action);
49
+
50
+ expect(formData.uuid).to.equal('test-llm-1');
51
+ expect(formData.llm).to.deep.equal([{ value: 'gpt-4', name: 'GPT 4.1' }]);
52
+ expect(formData.instructions).to.equal('Translate to French');
53
+ expect(formData.input).to.equal('@input');
54
+ });
55
+
56
+ it('converts form data to action correctly', () => {
57
+ const formData = {
58
+ uuid: 'test-llm-2',
59
+ llm: [{ value: 'gpt-5', name: 'GPT 5' }],
60
+ instructions: 'Summarize the following text',
61
+ input: '@input'
62
+ };
63
+
64
+ const action = call_llm.fromFormData(formData) as CallLLM;
65
+
66
+ expect(action.uuid).to.equal('test-llm-2');
67
+ expect(action.type).to.equal('call_llm');
68
+ expect(action.llm).to.deep.equal({ uuid: 'gpt-5', name: 'GPT 5' });
69
+ expect(action.instructions).to.equal('Summarize the following text');
70
+ expect(action.input).to.equal('@input');
71
+ });
72
+
73
+ it('handles empty form data', () => {
74
+ const formData = {
75
+ uuid: 'test-llm-3',
76
+ llm: [],
77
+ instructions: '',
78
+ input: ''
79
+ };
80
+
81
+ const action = call_llm.fromFormData(formData) as CallLLM;
82
+
83
+ expect(action.llm).to.deep.equal({ uuid: '', name: '' });
84
+ expect(action.instructions).to.equal('');
85
+ expect(action.input).to.equal('@input');
86
+ });
87
+ });
88
+
89
+ describe('action scenarios', () => {
90
+ helper.testAction(
91
+ {
92
+ uuid: 'test-action-1',
93
+ type: 'call_llm',
94
+ llm: { uuid: 'gpt-4', name: 'GPT 4.1' },
95
+ instructions: 'Translate to French',
96
+ input: '@input'
97
+ } as CallLLM,
98
+ 'translation-task'
99
+ );
100
+
101
+ helper.testAction(
102
+ {
103
+ uuid: 'test-action-2',
104
+ type: 'call_llm',
105
+ llm: { uuid: 'gpt-5', name: 'GPT 5' },
106
+ instructions:
107
+ 'Analyze the sentiment of the following message and classify it as positive, negative, or neutral. Provide a brief explanation for your classification.',
108
+ input: '@input'
109
+ } as CallLLM,
110
+ 'sentiment-analysis'
111
+ );
112
+
113
+ helper.testAction(
114
+ {
115
+ uuid: 'test-action-3',
116
+ type: 'call_llm',
117
+ llm: { uuid: 'gpt-4', name: 'GPT 4.1' },
118
+ instructions:
119
+ 'Summarize the key points from the conversation above in bullet format.',
120
+ input: '@input'
121
+ } as CallLLM,
122
+ 'summarization'
123
+ );
124
+
125
+ helper.testAction(
126
+ {
127
+ uuid: 'test-action-4',
128
+ type: 'call_llm',
129
+ llm: { uuid: 'gpt-5', name: 'GPT 5' },
130
+ instructions:
131
+ 'Extract any contact information (phone numbers, email addresses) from the text and format them as a JSON object.',
132
+ input: '@input'
133
+ } as CallLLM,
134
+ 'information-extraction'
135
+ );
136
+ });
137
+ });
@@ -0,0 +1,78 @@
1
+ # Node Configuration Tests
2
+
3
+ This directory contains tests for node configurations using the `NodeHelper` testing framework.
4
+
5
+ ## Overview
6
+
7
+ Similar to how the `ActionHelper` provides a uniform testing strategy for action configurations, the `NodeHelper` class provides a structured approach to testing node configurations. It handles:
8
+
9
+ 1. **Rendering Tests**: Verifies nodes render correctly in the flow editor
10
+ 2. **Editor Tests**: Verifies the node editor opens and displays correctly
11
+ 3. **Round-trip Conversion**: Tests form data conversion for nodes with `toFormData`/`fromFormData`
12
+ 4. **Screenshot Testing**: Captures visual screenshots for regression testing
13
+
14
+ ## Test Structure
15
+
16
+ Each node test file follows this pattern:
17
+
18
+ ```typescript
19
+ import { expect } from '@open-wc/testing';
20
+ import { node_config } from '../../src/flow/nodes/node_config';
21
+ import { Node } from '../../src/store/flow-definition';
22
+ import { NodeTest } from '../NodeHelper';
23
+
24
+ describe('node_config node config', () => {
25
+ const helper = new NodeTest(node_config, 'node_config');
26
+
27
+ describe('basic properties', () => {
28
+ helper.testBasicProperties();
29
+
30
+ // Additional property tests...
31
+ });
32
+
33
+ describe('node scenarios', () => {
34
+ helper.testNode(nodeData, nodeUI, 'test-name');
35
+
36
+ // More scenarios...
37
+ });
38
+
39
+ // Optional: data transformation tests for nodes with form configuration
40
+ describe('data transformation', () => {
41
+ // Round-trip conversion tests...
42
+ });
43
+ });
44
+ ```
45
+
46
+ ## Node Types Covered
47
+
48
+ ### Simple Nodes
49
+
50
+ - `wait_for_digits`: Basic node with no form configuration
51
+ - `wait_for_audio`, `wait_for_image`, etc.: Similar wait nodes
52
+
53
+ ### Form-Configured Nodes
54
+
55
+ - `wait_for_response`: Node with form fields and data transformation
56
+ - `split_by_llm_categorize`: Complex node with form configuration
57
+
58
+ ### Router-Based Nodes
59
+
60
+ - `split_by_random`: Random distribution node
61
+ - Other split nodes with router configurations
62
+
63
+ ## Screenshots
64
+
65
+ Screenshots are automatically generated and stored in:
66
+
67
+ - `screenshots/nodes/{node_name}/render/{test_name}.png` - Flow editor rendering
68
+ - `screenshots/nodes/{node_name}/editor/{test_name}.png` - Node editor dialog
69
+
70
+ ## Running Tests
71
+
72
+ ```bash
73
+ # Run all node tests
74
+ yarn test test/nodes/*.test.ts --no-watch
75
+
76
+ # Run specific node test
77
+ yarn test test/nodes/wait_for_response.test.ts --no-watch
78
+ ```