onbuzz 4.5.0 → 4.6.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 (1000) hide show
  1. package/bin/cli.js +38 -18
  2. package/package.json +2 -1
  3. package/scripts/e2e-flow-live.js +598 -0
  4. package/src/__tests__/resolvePreferredModel.test.js +15 -1
  5. package/src/core/__test-utils__/flowJudge.js +198 -0
  6. package/src/core/__tests__/flowCheckpointStore.test.js +140 -0
  7. package/src/core/__tests__/flowEndToEnd.test.js +565 -0
  8. package/src/core/__tests__/flowExecutor.test.js +950 -0
  9. package/src/core/__tests__/flowFieldMapping.test.js +189 -0
  10. package/src/core/__tests__/flowJudge.test.js +140 -0
  11. package/src/core/__tests__/flowLint.test.js +119 -0
  12. package/src/core/__tests__/flowLintClientMirror.test.js +98 -0
  13. package/src/core/__tests__/flowMigration.test.js +160 -0
  14. package/src/core/__tests__/flowRetry.test.js +145 -0
  15. package/src/core/__tests__/flowSavePayload.test.js +169 -0
  16. package/src/core/__tests__/flowSchema.test.js +438 -0
  17. package/src/core/__tests__/flowTemplates.test.js +311 -0
  18. package/src/core/__tests__/flowTypes.test.js +147 -0
  19. package/src/core/__tests__/flowVersionStore.test.js +123 -0
  20. package/src/core/__tests__/orchestrator.test.js +38 -0
  21. package/src/core/__tests__/stateManager.test.js +348 -0
  22. package/src/core/agentScheduler.js +37 -6
  23. package/src/core/flowCheckpointStore.js +146 -0
  24. package/src/core/flowExecutor.js +669 -87
  25. package/src/core/flowFieldMapping.js +138 -0
  26. package/src/core/flowLint.js +102 -0
  27. package/src/core/flowMigration.js +88 -0
  28. package/src/core/flowRetry.js +103 -0
  29. package/src/core/flowSchema.js +297 -0
  30. package/src/core/flowTypes.js +81 -0
  31. package/src/core/flowVersionStore.js +122 -0
  32. package/src/core/messageProcessor.js +5 -1
  33. package/src/core/stateManager.js +265 -17
  34. package/src/index.js +17 -0
  35. package/src/interfaces/webServer.js +304 -4
  36. package/src/modules/widget/galleryStore.js +27 -2
  37. package/src/services/__tests__/apiKeyManager.test.js +163 -0
  38. package/src/services/__tests__/flowContextService.test.js +391 -0
  39. package/src/services/apiKeyManager.js +54 -3
  40. package/src/services/flowContextService.js +401 -4
  41. package/src/tools/__tests__/imageFormat.test.js +208 -0
  42. package/src/tools/__tests__/jobDoneTool.test.js +285 -0
  43. package/src/tools/__tests__/platformControlAgentTeamPermissions.test.js +200 -0
  44. package/src/tools/__tests__/platformControlFlow.test.js +412 -0
  45. package/src/tools/imageFormat.js +249 -0
  46. package/src/tools/imageTool.js +107 -18
  47. package/src/tools/jobDoneTool.js +111 -3
  48. package/src/tools/platformControl/permissions.js +111 -0
  49. package/src/tools/platformControlTool.js +222 -0
  50. package/src/tools/videoTool.js +94 -60
  51. package/src/tools/webTool.js +9 -0
  52. package/src/utilities/__tests__/apiKeyPlaceholders.test.js +120 -0
  53. package/src/utilities/__tests__/brand.test.js +115 -0
  54. package/src/utilities/__tests__/flowEdgeIds.test.js +116 -0
  55. package/src/utilities/__tests__/flowPublish.test.js +196 -0
  56. package/src/utilities/__tests__/flowRunFilters.test.js +315 -0
  57. package/src/utilities/__tests__/useIsTouchDevice.detect.test.js +114 -0
  58. package/src/utilities/apiKeyPlaceholders.js +68 -0
  59. package/src/utilities/brand.js +93 -0
  60. package/src/utilities/flowEdgeIds.js +50 -0
  61. package/src/utilities/flowPublish.js +72 -0
  62. package/src/utilities/flowRunFilters.js +149 -0
  63. package/src/utilities/userDataDir.js +5 -0
  64. package/web-ui/build/index.html +2 -2
  65. package/web-ui/build/static/1c-4BRB87En.js +1 -0
  66. package/web-ui/build/static/abap-BXIqiAfL.js +1 -0
  67. package/web-ui/build/static/abnf-0gk1i0eD.js +1 -0
  68. package/web-ui/build/static/abnf-jcv7vN9a.js +1 -0
  69. package/web-ui/build/static/accesslog-bXE2aOZl.js +1 -0
  70. package/web-ui/build/static/actionscript-CF-bqawj.js +1 -0
  71. package/web-ui/build/static/actionscript-CodwLKar.js +1 -0
  72. package/web-ui/build/static/ada-BBEDI84F.js +1 -0
  73. package/web-ui/build/static/ada-Kb6JvaxP.js +1 -0
  74. package/web-ui/build/static/agda-UvnMXUUo.js +1 -0
  75. package/web-ui/build/static/al-D5QZ0ijQ.js +1 -0
  76. package/web-ui/build/static/angelscript-RpvZxbeT.js +1 -0
  77. package/web-ui/build/static/antlr4-BPyUvfii.js +1 -0
  78. package/web-ui/build/static/apache-CtM2pGik.js +1 -0
  79. package/web-ui/build/static/apacheconf-D-i9eISH.js +1 -0
  80. package/web-ui/build/static/apex-Dhn_SJi4.js +1 -0
  81. package/web-ui/build/static/apl-HWtdSaYy.js +1 -0
  82. package/web-ui/build/static/applescript-Br9NrmTP.js +1 -0
  83. package/web-ui/build/static/applescript-DLL0TuGw.js +1 -0
  84. package/web-ui/build/static/aql-MjCgNRUX.js +1 -0
  85. package/web-ui/build/static/arcade--DFvEPVv.js +1 -0
  86. package/web-ui/build/static/arduino-Cqw5W_fL.js +1 -0
  87. package/web-ui/build/static/arduino-DIDBbB7R.js +1 -0
  88. package/web-ui/build/static/arff-GGvbnOhc.js +1 -0
  89. package/web-ui/build/static/armasm-DKLRDLs1.js +1 -0
  90. package/web-ui/build/static/asciidoc-CN5qfm0K.js +1 -0
  91. package/web-ui/build/static/asciidoc-CeQEWYOj.js +1 -0
  92. package/web-ui/build/static/asm6502-B1l2Hc9j.js +1 -0
  93. package/web-ui/build/static/asmatmel-BAojNNWJ.js +1 -0
  94. package/web-ui/build/static/aspectj-DqZFhwdh.js +1 -0
  95. package/web-ui/build/static/aspnet-C0kQPqSr.js +1 -0
  96. package/web-ui/build/static/autohotkey-DY25td-Z.js +1 -0
  97. package/web-ui/build/static/autohotkey-DlqXINgk.js +1 -0
  98. package/web-ui/build/static/autoit-By7J2Y34.js +1 -0
  99. package/web-ui/build/static/autoit-nsu9sBga.js +1 -0
  100. package/web-ui/build/static/avisynth-DmhvVsr-.js +1 -0
  101. package/web-ui/build/static/avrasm-CHfBwwoQ.js +1 -0
  102. package/web-ui/build/static/avro-idl-BKgIWEIF.js +1 -0
  103. package/web-ui/build/static/awk-CpAb4d4E.js +1 -0
  104. package/web-ui/build/static/axapta-CM4aTfCZ.js +1 -0
  105. package/web-ui/build/static/bash-D1KTDlk-.js +1 -0
  106. package/web-ui/build/static/bash-DShkskxb.js +1 -0
  107. package/web-ui/build/static/basic-C4CfKvhg.js +1 -0
  108. package/web-ui/build/static/basic-CkQS4VwL.js +1 -0
  109. package/web-ui/build/static/batch-DZtrClI1.js +1 -0
  110. package/web-ui/build/static/bbcode-CY2FsfaO.js +1 -0
  111. package/web-ui/build/static/bicep-BeSEpm_3.js +1 -0
  112. package/web-ui/build/static/birb-huu22Mwu.js +1 -0
  113. package/web-ui/build/static/bison-CmbxfR57.js +1 -0
  114. package/web-ui/build/static/bnf-c2l_4HXN.js +1 -0
  115. package/web-ui/build/static/bnf-wucEDTqY.js +1 -0
  116. package/web-ui/build/static/brainfuck-CzWBg5U7.js +1 -0
  117. package/web-ui/build/static/brainfuck-I_YQ4sma.js +1 -0
  118. package/web-ui/build/static/brightscript-LOkD4eYW.js +1 -0
  119. package/web-ui/build/static/bro-CLJ7Afqo.js +1 -0
  120. package/web-ui/build/static/bsl-DbUgC2FQ.js +1 -0
  121. package/web-ui/build/static/c-C0S2D2ge.js +1 -0
  122. package/web-ui/build/static/c-KH25O_pL.js +1 -0
  123. package/web-ui/build/static/c-like-DtAFxyl_.js +1 -0
  124. package/web-ui/build/static/cal-uGiVJBNC.js +1 -0
  125. package/web-ui/build/static/capnproto-DSLchjOW.js +1 -0
  126. package/web-ui/build/static/ceylon-Bbxk66YL.js +1 -0
  127. package/web-ui/build/static/cfscript-CjDdgKbl.js +1 -0
  128. package/web-ui/build/static/chaiscript-CnRXaMae.js +1 -0
  129. package/web-ui/build/static/cil-CM9ygF3R.js +1 -0
  130. package/web-ui/build/static/clean-ppAH0SYy.js +1 -0
  131. package/web-ui/build/static/clojure-C5LVccGq.js +1 -0
  132. package/web-ui/build/static/clojure-DtLXiuuC.js +1 -0
  133. package/web-ui/build/static/clojure-repl-fSdrYuup.js +1 -0
  134. package/web-ui/build/static/cmake-AD1fuzSY.js +1 -0
  135. package/web-ui/build/static/cmake-D-o08Yta.js +1 -0
  136. package/web-ui/build/static/cobol-bDh1KHf9.js +1 -0
  137. package/web-ui/build/static/coffeescript-D8InK8kf.js +1 -0
  138. package/web-ui/build/static/coffeescript-MKWUOXb5.js +1 -0
  139. package/web-ui/build/static/concurnas-CR56nj6R.js +1 -0
  140. package/web-ui/build/static/coq-DDLBsBJu.js +1 -0
  141. package/web-ui/build/static/coq-WbhtANeW.js +1 -0
  142. package/web-ui/build/static/cos-Bfmre3op.js +1 -0
  143. package/web-ui/build/static/cpp-Cdy6A7gI.js +1 -0
  144. package/web-ui/build/static/cpp-De2KZejr.js +1 -0
  145. package/web-ui/build/static/crmsh-uf3vbKnN.js +1 -0
  146. package/web-ui/build/static/crystal-BHMyEVPP.js +1 -0
  147. package/web-ui/build/static/crystal-DXAQwftg.js +1 -0
  148. package/web-ui/build/static/csharp-Ci2MRG6L.js +1 -0
  149. package/web-ui/build/static/csharp-D8MV2a7f.js +1 -0
  150. package/web-ui/build/static/cshtml-kVj3v3uY.js +1 -0
  151. package/web-ui/build/static/csp-B83CCdVc.js +1 -0
  152. package/web-ui/build/static/csp-DUO7zlPh.js +1 -0
  153. package/web-ui/build/static/css-BWsx7mtO.js +1 -0
  154. package/web-ui/build/static/css-extras-oHch2Y5C.js +1 -0
  155. package/web-ui/build/static/csv-DIf0iNlc.js +1 -0
  156. package/web-ui/build/static/cypher-B1rOF_f3.js +1 -0
  157. package/web-ui/build/static/d-BUgLS87u.js +1 -0
  158. package/web-ui/build/static/d-ClM8XTCr.js +1 -0
  159. package/web-ui/build/static/dart-D5rVGtrb.js +1 -0
  160. package/web-ui/build/static/dart-Oal__k4m.js +1 -0
  161. package/web-ui/build/static/dataweave-B0uxnNDC.js +1 -0
  162. package/web-ui/build/static/dax-B9z6d1P6.js +1 -0
  163. package/web-ui/build/static/delphi-KMlx8q8A.js +1 -0
  164. package/web-ui/build/static/dhall-DLWFgSR8.js +1 -0
  165. package/web-ui/build/static/diff-DDMGnyhE.js +1 -0
  166. package/web-ui/build/static/diff-yg1oXq3U.js +1 -0
  167. package/web-ui/build/static/django-BDtkIu_c.js +1 -0
  168. package/web-ui/build/static/django-Bkn9HlyQ.js +1 -0
  169. package/web-ui/build/static/dns-Bx1VHfo6.js +1 -0
  170. package/web-ui/build/static/dns-zone-file-Bgn5zn_J.js +1 -0
  171. package/web-ui/build/static/docker-CTdrZJHc.js +1 -0
  172. package/web-ui/build/static/dockerfile-B7ziWbf_.js +1 -0
  173. package/web-ui/build/static/dos-cbBC5Icj.js +1 -0
  174. package/web-ui/build/static/dot-JXSVTLEH.js +1 -0
  175. package/web-ui/build/static/dsconfig-sEua2C2o.js +1 -0
  176. package/web-ui/build/static/dts-Zem_gT_O.js +1 -0
  177. package/web-ui/build/static/dust-B-dZ6-uQ.js +1 -0
  178. package/web-ui/build/static/ebnf-CXfeZoXF.js +1 -0
  179. package/web-ui/build/static/ebnf-raQs4FpT.js +1 -0
  180. package/web-ui/build/static/editorconfig-hp8ARbHf.js +1 -0
  181. package/web-ui/build/static/eiffel-tO3jaG00.js +1 -0
  182. package/web-ui/build/static/ejs-BGmD5Rvl.js +1 -0
  183. package/web-ui/build/static/elixir-CF0JDN9L.js +1 -0
  184. package/web-ui/build/static/elixir-CFSil_Ou.js +1 -0
  185. package/web-ui/build/static/elm-As6XaD-g.js +1 -0
  186. package/web-ui/build/static/elm-pw5GpqTZ.js +1 -0
  187. package/web-ui/build/static/erb-BrpCJIyN.js +1 -0
  188. package/web-ui/build/static/erb-D2_eJU9C.js +1 -0
  189. package/web-ui/build/static/erlang-5rSv6NMZ.js +1 -0
  190. package/web-ui/build/static/erlang-RCAxVMBd.js +1 -0
  191. package/web-ui/build/static/erlang-repl-BN0h_6XR.js +1 -0
  192. package/web-ui/build/static/etlua-BMtMnKtv.js +1 -0
  193. package/web-ui/build/static/excel-CRNS57iM.js +1 -0
  194. package/web-ui/build/static/excel-formula-BNw56R2m.js +1 -0
  195. package/web-ui/build/static/factor-7lsloWTf.js +1 -0
  196. package/web-ui/build/static/false-CQUvGnfd.js +1 -0
  197. package/web-ui/build/static/firestore-security-rules-WC1PabFS.js +1 -0
  198. package/web-ui/build/static/fix-Cs6kLeYV.js +1 -0
  199. package/web-ui/build/static/flix-1fdMgw2I.js +1 -0
  200. package/web-ui/build/static/flow-CRPduKSH.js +1 -0
  201. package/web-ui/build/static/fortran-C-3TUt5R.js +1 -0
  202. package/web-ui/build/static/fortran-CA4bbZCj.js +1 -0
  203. package/web-ui/build/static/fsharp-BScjCFae.js +1 -0
  204. package/web-ui/build/static/fsharp-BtHWuW4o.js +1 -0
  205. package/web-ui/build/static/ftl-Re-S2zvF.js +1 -0
  206. package/web-ui/build/static/gams-Cr12l-cM.js +1 -0
  207. package/web-ui/build/static/gap-H0O9-Zju.js +1 -0
  208. package/web-ui/build/static/gauss-CW8jwURx.js +1 -0
  209. package/web-ui/build/static/gcode-CIxfYtg9.js +1 -0
  210. package/web-ui/build/static/gcode-CVqRWYeB.js +1 -0
  211. package/web-ui/build/static/gdscript-BltZCaKl.js +1 -0
  212. package/web-ui/build/static/gedcom-nMT5ety_.js +1 -0
  213. package/web-ui/build/static/gherkin-B4M7CGOo.js +1 -0
  214. package/web-ui/build/static/gherkin-DumuqAQF.js +1 -0
  215. package/web-ui/build/static/git-BBiBlZ5H.js +1 -0
  216. package/web-ui/build/static/glsl-D8ckiRVo.js +1 -0
  217. package/web-ui/build/static/glsl-Z4VdDyPa.js +1 -0
  218. package/web-ui/build/static/gml-BHuSndHE.js +1 -0
  219. package/web-ui/build/static/gml-DZpfN_A4.js +1 -0
  220. package/web-ui/build/static/gn-BjpAHNh_.js +1 -0
  221. package/web-ui/build/static/go-BeZ9IgZB.js +1 -0
  222. package/web-ui/build/static/go-HEXrcgIc.js +1 -0
  223. package/web-ui/build/static/go-module-BGdPb7sP.js +1 -0
  224. package/web-ui/build/static/golo-Cr70KUbI.js +1 -0
  225. package/web-ui/build/static/gradle-C7MXrDnB.js +1 -0
  226. package/web-ui/build/static/graphql-CJK0wQjf.js +1 -0
  227. package/web-ui/build/static/groovy-Bwe7hkFG.js +1 -0
  228. package/web-ui/build/static/groovy-Dc6WDoKa.js +1 -0
  229. package/web-ui/build/static/haml-BSsAjltt.js +1 -0
  230. package/web-ui/build/static/haml-CZih7X_4.js +1 -0
  231. package/web-ui/build/static/handlebars-CN8hCpsy.js +1 -0
  232. package/web-ui/build/static/handlebars-XBLc7i13.js +1 -0
  233. package/web-ui/build/static/haskell-BH9WhK10.js +1 -0
  234. package/web-ui/build/static/haskell-Djf2J1vd.js +1 -0
  235. package/web-ui/build/static/haxe-BllD-W9e.js +1 -0
  236. package/web-ui/build/static/haxe-DY8Hyn7u.js +1 -0
  237. package/web-ui/build/static/hcl-CfktAP2o.js +1 -0
  238. package/web-ui/build/static/hlsl-B2JSOD9d.js +1 -0
  239. package/web-ui/build/static/hoon-BILyDobB.js +1 -0
  240. package/web-ui/build/static/hpkp-Bb8rXx4p.js +1 -0
  241. package/web-ui/build/static/hsp-BmjeJC21.js +1 -0
  242. package/web-ui/build/static/hsts-DJhW57Fu.js +1 -0
  243. package/web-ui/build/static/htmlbars-CJe_ZLjw.js +1 -0
  244. package/web-ui/build/static/http-D7SlABvQ.js +1 -0
  245. package/web-ui/build/static/http-DEUJqmEm.js +1 -0
  246. package/web-ui/build/static/hy-DVfsgEbr.js +1 -0
  247. package/web-ui/build/static/ichigojam-CPdz2bOt.js +1 -0
  248. package/web-ui/build/static/icon-DYK9RbU9.js +1 -0
  249. package/web-ui/build/static/icu-message-format-HiZB_alB.js +1 -0
  250. package/web-ui/build/static/idris-Clk3tJIE.js +1 -0
  251. package/web-ui/build/static/iecst-D7wkeksq.js +1 -0
  252. package/web-ui/build/static/ignore-GSm-L3jG.js +1 -0
  253. package/web-ui/build/static/index-BJe5td0v.js +13 -0
  254. package/web-ui/build/static/index-BYvljeMN.js +1 -0
  255. package/web-ui/build/static/index-CLelJq9j.js +1156 -0
  256. package/web-ui/build/static/index-f5UnFSsY.css +1 -0
  257. package/web-ui/build/static/inform7-D3gc0hSc.js +1 -0
  258. package/web-ui/build/static/inform7-DLdvZTxX.js +1 -0
  259. package/web-ui/build/static/ini-CVcWx1TA.js +1 -0
  260. package/web-ui/build/static/ini-CxYBdsvZ.js +1 -0
  261. package/web-ui/build/static/io-D760QF0i.js +1 -0
  262. package/web-ui/build/static/irpf90-DfkXdZut.js +1 -0
  263. package/web-ui/build/static/isbl-Bl3W-krY.js +1 -0
  264. package/web-ui/build/static/j-DyJj9FSc.js +1 -0
  265. package/web-ui/build/static/java-CLw1cH97.js +1 -0
  266. package/web-ui/build/static/java-vXaHBzP4.js +1 -0
  267. package/web-ui/build/static/javadoc-p5hIe48g.js +1 -0
  268. package/web-ui/build/static/javadoclike-OXi-sQ0q.js +1 -0
  269. package/web-ui/build/static/javascript-CTSqES8o.js +1 -0
  270. package/web-ui/build/static/javastacktrace-PC9-3moA.js +1 -0
  271. package/web-ui/build/static/jboss-cli-CxOZKDLi.js +1 -0
  272. package/web-ui/build/static/jexl-C4kSXhD1.js +1 -0
  273. package/web-ui/build/static/jolie-CntM8cWx.js +1 -0
  274. package/web-ui/build/static/jq-D_4bTCm0.js +1 -0
  275. package/web-ui/build/static/js-extras-Cx9VAZJd.js +1 -0
  276. package/web-ui/build/static/js-templates-BqyJUDxJ.js +1 -0
  277. package/web-ui/build/static/jsdoc-CNHLgAph.js +1 -0
  278. package/web-ui/build/static/json-BXIFrbVC.js +1 -0
  279. package/web-ui/build/static/json-DRUrJL_F.js +1 -0
  280. package/web-ui/build/static/json5-EsTrOpXr.js +1 -0
  281. package/web-ui/build/static/jsonp-RijEuTJY.js +1 -0
  282. package/web-ui/build/static/jsstacktrace-dM13uE1y.js +1 -0
  283. package/web-ui/build/static/jsx-Cf0j02DZ.js +1 -0
  284. package/web-ui/build/static/julia-BLrtj5Dj.js +1 -0
  285. package/web-ui/build/static/julia-D0PJgQhi.js +1 -0
  286. package/web-ui/build/static/julia-repl-BLnLCve4.js +1 -0
  287. package/web-ui/build/static/keepalived-SvzEYloD.js +1 -0
  288. package/web-ui/build/static/keyman-B5l2mmmE.js +1 -0
  289. package/web-ui/build/static/kotlin-B_6COR1e.js +1 -0
  290. package/web-ui/build/static/kotlin-vd1Xdz6D.js +1 -0
  291. package/web-ui/build/static/kumir-8mIZWgJQ.js +1 -0
  292. package/web-ui/build/static/kusto-CoV2oHqt.js +1 -0
  293. package/web-ui/build/static/lasso-BjjUNa_o.js +1 -0
  294. package/web-ui/build/static/latex-C9KOfuFk.js +1 -0
  295. package/web-ui/build/static/latex-DhLttY-I.js +1 -0
  296. package/web-ui/build/static/latte-CaYNaOoK.js +1 -0
  297. package/web-ui/build/static/ldif-DFxnpH3G.js +1 -0
  298. package/web-ui/build/static/leaf-DrMpC83a.js +1 -0
  299. package/web-ui/build/static/less-DUoYpEFn.js +1 -0
  300. package/web-ui/build/static/less-nh0XCcut.js +1 -0
  301. package/web-ui/build/static/lilypond-BlqjLr3W.js +1 -0
  302. package/web-ui/build/static/liquid-BVGHSG0T.js +1 -0
  303. package/web-ui/build/static/lisp-CEK8BtYQ.js +1 -0
  304. package/web-ui/build/static/lisp-D_jGHxhc.js +1 -0
  305. package/web-ui/build/static/livecodeserver-CFUDw_SK.js +1 -0
  306. package/web-ui/build/static/livescript-C5xzpiLM.js +1 -0
  307. package/web-ui/build/static/livescript-CKoPcK0c.js +1 -0
  308. package/web-ui/build/static/llvm-BBOVIeTl.js +1 -0
  309. package/web-ui/build/static/llvm-ioC5bCUB.js +1 -0
  310. package/web-ui/build/static/log-CPYTqm4T.js +1 -0
  311. package/web-ui/build/static/lolcode-BkT93adT.js +1 -0
  312. package/web-ui/build/static/lsl-M98YPNRW.js +1 -0
  313. package/web-ui/build/static/lua-D7Lum5n9.js +1 -0
  314. package/web-ui/build/static/lua-DHyr0QwE.js +1 -0
  315. package/web-ui/build/static/magma-BwMFjwQu.js +1 -0
  316. package/web-ui/build/static/makefile-BFYZ-SfG.js +1 -0
  317. package/web-ui/build/static/makefile-D6oyR2lD.js +1 -0
  318. package/web-ui/build/static/markdown-6jtg6KZm.js +1 -0
  319. package/web-ui/build/static/markdown-DTLtPgeP.js +1 -0
  320. package/web-ui/build/static/markup-templating-K8uoAIqY.js +1 -0
  321. package/web-ui/build/static/mathematica-cW5hOuB2.js +1 -0
  322. package/web-ui/build/static/matlab-CwSNWEq8.js +1 -0
  323. package/web-ui/build/static/matlab-MQ972BY-.js +1 -0
  324. package/web-ui/build/static/maxima-CMSFLri-.js +1 -0
  325. package/web-ui/build/static/maxscript-BzcoS2KA.js +1 -0
  326. package/web-ui/build/static/mel-Cd20pAs-.js +1 -0
  327. package/web-ui/build/static/mel-KWG5N49g.js +1 -0
  328. package/web-ui/build/static/mercury-eW1ljJh0.js +1 -0
  329. package/web-ui/build/static/mermaid-D3zeCM84.js +1 -0
  330. package/web-ui/build/static/mipsasm-B5g6Ndra.js +1 -0
  331. package/web-ui/build/static/mizar-C_nZ0Ts9.js +1 -0
  332. package/web-ui/build/static/mizar-CuTNEv2t.js +1 -0
  333. package/web-ui/build/static/mojolicious-C1Xq--lT.js +1 -0
  334. package/web-ui/build/static/mongodb-C_3giArw.js +1 -0
  335. package/web-ui/build/static/monkey-DQ1olPd-.js +1 -0
  336. package/web-ui/build/static/monkey-n4Tbrep7.js +1 -0
  337. package/web-ui/build/static/moonscript-CmnQJuIL.js +1 -0
  338. package/web-ui/build/static/moonscript-DXgmWjLy.js +1 -0
  339. package/web-ui/build/static/n1ql-531a3W-1.js +1 -0
  340. package/web-ui/build/static/n1ql-BYH9EvME.js +1 -0
  341. package/web-ui/build/static/n4js-Cy0LcM-V.js +1 -0
  342. package/web-ui/build/static/nand2tetris-hdl-CKOztBDD.js +1 -0
  343. package/web-ui/build/static/naniscript-oLFpxBGE.js +1 -0
  344. package/web-ui/build/static/nasm-DLPnwFkM.js +1 -0
  345. package/web-ui/build/static/neon-Cn03wGUR.js +1 -0
  346. package/web-ui/build/static/nevod-C_VRvbdf.js +1 -0
  347. package/web-ui/build/static/nginx-60SpLv4t.js +1 -0
  348. package/web-ui/build/static/nginx-DZLHid-q.js +1 -0
  349. package/web-ui/build/static/nim-BnrHdqGC.js +1 -0
  350. package/web-ui/build/static/nim-Ds3pmwtE.js +1 -0
  351. package/web-ui/build/static/nix-Dl-dS0w4.js +1 -0
  352. package/web-ui/build/static/nix-jL62FzTz.js +1 -0
  353. package/web-ui/build/static/node-repl-BAuDzNZd.js +1 -0
  354. package/web-ui/build/static/nsis-CEnFjfNr.js +1 -0
  355. package/web-ui/build/static/nsis-CcCz6HIu.js +1 -0
  356. package/web-ui/build/static/objectivec-CZXfVowS.js +1 -0
  357. package/web-ui/build/static/objectivec-CuaFU-Ef.js +1 -0
  358. package/web-ui/build/static/ocaml-BGAzfwAx.js +1 -0
  359. package/web-ui/build/static/ocaml-BoO1lU26.js +1 -0
  360. package/web-ui/build/static/opencl-DAnD5xP-.js +1 -0
  361. package/web-ui/build/static/openqasm-CO7gdVj9.js +1 -0
  362. package/web-ui/build/static/openscad-B2wMEMQR.js +1 -0
  363. package/web-ui/build/static/oxygene-1erv93gF.js +1 -0
  364. package/web-ui/build/static/oz-D3334gi7.js +1 -0
  365. package/web-ui/build/static/parigp-rQZTN8zi.js +1 -0
  366. package/web-ui/build/static/parser-B4ZSZ1ig.js +1 -0
  367. package/web-ui/build/static/parser3-CCCjkYLG.js +1 -0
  368. package/web-ui/build/static/pascal-DeC9Pvom.js +1 -0
  369. package/web-ui/build/static/pascaligo-CT2u6UpO.js +1 -0
  370. package/web-ui/build/static/pcaxis-CvlRg6t4.js +1 -0
  371. package/web-ui/build/static/peoplecode-BYahULz2.js +1 -0
  372. package/web-ui/build/static/perl-CKgvyY59.js +1 -0
  373. package/web-ui/build/static/perl-E7sx7Izv.js +1 -0
  374. package/web-ui/build/static/pf-D9wnFBt7.js +1 -0
  375. package/web-ui/build/static/pgsql-CrlBdRAn.js +1 -0
  376. package/web-ui/build/static/php-D2O-B3y7.js +1 -0
  377. package/web-ui/build/static/php-extras-BKqZuzgt.js +1 -0
  378. package/web-ui/build/static/php-nfXl9WfB.js +1 -0
  379. package/web-ui/build/static/php-template-Cg8ERLNQ.js +1 -0
  380. package/web-ui/build/static/phpdoc-Z1u90OwH.js +1 -0
  381. package/web-ui/build/static/plaintext-DkNVeXWx.js +1 -0
  382. package/web-ui/build/static/plsql-DO5fJNhS.js +1 -0
  383. package/web-ui/build/static/pony-DXIqq5Bk.js +1 -0
  384. package/web-ui/build/static/powerquery-B2ovxJZi.js +1 -0
  385. package/web-ui/build/static/powershell-BVeqqari.js +1 -0
  386. package/web-ui/build/static/powershell-CskV4qWh.js +1 -0
  387. package/web-ui/build/static/processing-D90SJpnq.js +1 -0
  388. package/web-ui/build/static/processing-DP2mf97Z.js +1 -0
  389. package/web-ui/build/static/profile-H6zk8v_f.js +1 -0
  390. package/web-ui/build/static/prolog-AUfxYiJV.js +1 -0
  391. package/web-ui/build/static/prolog-Dz2WdsVN.js +1 -0
  392. package/web-ui/build/static/promql-CloMJmMW.js +1 -0
  393. package/web-ui/build/static/properties-CTl3mnrP.js +1 -0
  394. package/web-ui/build/static/properties-D2RfteTZ.js +1 -0
  395. package/web-ui/build/static/protobuf-Be9oqQCu.js +1 -0
  396. package/web-ui/build/static/protobuf-BgG8zUnY.js +1 -0
  397. package/web-ui/build/static/psl-Bqx8iN9e.js +1 -0
  398. package/web-ui/build/static/pug-CHfFC0Wg.js +1 -0
  399. package/web-ui/build/static/puppet-__1ifzZS.js +1 -0
  400. package/web-ui/build/static/puppet-pbtRCMgd.js +1 -0
  401. package/web-ui/build/static/pure-B9STuf4F.js +1 -0
  402. package/web-ui/build/static/purebasic-7ADyn-Ce.js +1 -0
  403. package/web-ui/build/static/purebasic-Bc06YQcD.js +1 -0
  404. package/web-ui/build/static/purescript-CU-dDebr.js +1 -0
  405. package/web-ui/build/static/python-BEsXrEbi.js +1 -0
  406. package/web-ui/build/static/python-CZFrgH2X.js +1 -0
  407. package/web-ui/build/static/python-repl-CI9GmJce.js +1 -0
  408. package/web-ui/build/static/q-E7xvTGRo.js +1 -0
  409. package/web-ui/build/static/q-hvg9aQoY.js +1 -0
  410. package/web-ui/build/static/qml-DkZWcxEv.js +1 -0
  411. package/web-ui/build/static/qml-DpdWla8G.js +1 -0
  412. package/web-ui/build/static/qore-Dsm7ACR_.js +1 -0
  413. package/web-ui/build/static/qsharp-BBec6MJR.js +1 -0
  414. package/web-ui/build/static/r-CTVkq_Ky.js +1 -0
  415. package/web-ui/build/static/r-DHGNCupF.js +1 -0
  416. package/web-ui/build/static/racket-hoSh4YPx.js +1 -0
  417. package/web-ui/build/static/reason-CQPUIbsm.js +1 -0
  418. package/web-ui/build/static/reasonml-BXtFFSTZ.js +1 -0
  419. package/web-ui/build/static/regex-BNkF_ufb.js +1 -0
  420. package/web-ui/build/static/rego-BXHrjItC.js +1 -0
  421. package/web-ui/build/static/renpy-CBQvK_5t.js +1 -0
  422. package/web-ui/build/static/rest-Bf8hzmF1.js +1 -0
  423. package/web-ui/build/static/rib-D0a4KzlP.js +1 -0
  424. package/web-ui/build/static/rip-DchN4BQN.js +1 -0
  425. package/web-ui/build/static/roboconf-HXnyGNTm.js +1 -0
  426. package/web-ui/build/static/roboconf-zfK6mgmU.js +1 -0
  427. package/web-ui/build/static/robotframework-DnP-vXyd.js +1 -0
  428. package/web-ui/build/static/routeros-CDirYxZY.js +1 -0
  429. package/web-ui/build/static/rsl-BCK_msc-.js +1 -0
  430. package/web-ui/build/static/ruby--mXX6GNK.js +1 -0
  431. package/web-ui/build/static/ruby-BIdQYq0Y.js +1 -0
  432. package/web-ui/build/static/ruleslanguage-BD07s5RU.js +1 -0
  433. package/web-ui/build/static/rust-Bgni9V7U.js +1 -0
  434. package/web-ui/build/static/rust-OywmQ2ox.js +1 -0
  435. package/web-ui/build/static/sas-B6-82htH.js +1 -0
  436. package/web-ui/build/static/sas-k9XEScbR.js +1 -0
  437. package/web-ui/build/static/sass-BOvNBStT.js +1 -0
  438. package/web-ui/build/static/scala-At07fmMd.js +1 -0
  439. package/web-ui/build/static/scala-DUOfqSaU.js +1 -0
  440. package/web-ui/build/static/scheme-B87lC0bK.js +1 -0
  441. package/web-ui/build/static/scheme-BS7OzWRX.js +1 -0
  442. package/web-ui/build/static/scilab-Bmn4jjcu.js +1 -0
  443. package/web-ui/build/static/scss-DTeB95yE.js +1 -0
  444. package/web-ui/build/static/scss-xilnl6Dx.js +1 -0
  445. package/web-ui/build/static/shell-CKetyy4y.js +1 -0
  446. package/web-ui/build/static/shell-session-B6rat2c7.js +1 -0
  447. package/web-ui/build/static/smali-C9O9CXOD.js +1 -0
  448. package/web-ui/build/static/smali-CBkgbinS.js +1 -0
  449. package/web-ui/build/static/smalltalk-CuUjFHiz.js +1 -0
  450. package/web-ui/build/static/smalltalk-D0xhFL9b.js +1 -0
  451. package/web-ui/build/static/smarty-CY0N0d5x.js +1 -0
  452. package/web-ui/build/static/sml-BVAw3fjB.js +1 -0
  453. package/web-ui/build/static/sml-CO9XB1ax.js +1 -0
  454. package/web-ui/build/static/solidity-BNixD4Dx.js +1 -0
  455. package/web-ui/build/static/solution-file-DHyGXf2y.js +1 -0
  456. package/web-ui/build/static/soy-Cxgjntsl.js +1 -0
  457. package/web-ui/build/static/sparql-CDLqu9vy.js +1 -0
  458. package/web-ui/build/static/splunk-spl-Cmkl_Xmw.js +1 -0
  459. package/web-ui/build/static/sqf-CIGKolw3.js +1 -0
  460. package/web-ui/build/static/sqf-Cfp_lC6-.js +1 -0
  461. package/web-ui/build/static/sql-DHOGwYxe.js +1 -0
  462. package/web-ui/build/static/sql-QIw1Q2yi.js +1 -0
  463. package/web-ui/build/static/sql_more-C0b3-InT.js +1 -0
  464. package/web-ui/build/static/squirrel-DASToo1F.js +1 -0
  465. package/web-ui/build/static/stan-DHMkcFZg.js +1 -0
  466. package/web-ui/build/static/stan-No3nsdnQ.js +1 -0
  467. package/web-ui/build/static/stata-Brkzu5Nv.js +1 -0
  468. package/web-ui/build/static/step21-D2l5M5Ml.js +1 -0
  469. package/web-ui/build/static/stylus-Bx6FaZN_.js +1 -0
  470. package/web-ui/build/static/stylus-CVkTaPns.js +1 -0
  471. package/web-ui/build/static/subunit-CDrsAMuL.js +1 -0
  472. package/web-ui/build/static/swift-BSw_pS-l.js +1 -0
  473. package/web-ui/build/static/swift-BWyqwIo8.js +1 -0
  474. package/web-ui/build/static/systemd-BwqpYcJK.js +1 -0
  475. package/web-ui/build/static/t4-cs-CwsJLOxV.js +1 -0
  476. package/web-ui/build/static/t4-templating-BfxY01xf.js +1 -0
  477. package/web-ui/build/static/t4-vb-BhsKKFhp.js +1 -0
  478. package/web-ui/build/static/taggerscript-Bkew95or.js +1 -0
  479. package/web-ui/build/static/tap-DHovcHbS.js +1 -0
  480. package/web-ui/build/static/tap-DPl9uPo4.js +1 -0
  481. package/web-ui/build/static/tcl-BeozHO-f.js +1 -0
  482. package/web-ui/build/static/tcl-Df_8ItQH.js +1 -0
  483. package/web-ui/build/static/textile-B3m_f58b.js +1 -0
  484. package/web-ui/build/static/thrift-D20n-HpW.js +1 -0
  485. package/web-ui/build/static/toml-C-fTAonK.js +1 -0
  486. package/web-ui/build/static/tp-hGbXe5Eo.js +1 -0
  487. package/web-ui/build/static/tremor-BaVOtRFz.js +1 -0
  488. package/web-ui/build/static/tsx-ChPInNl0.js +1 -0
  489. package/web-ui/build/static/tt2-DAIJ8cpz.js +1 -0
  490. package/web-ui/build/static/turtle-B7z_y-PK.js +1 -0
  491. package/web-ui/build/static/twig-TUEkve6W.js +1 -0
  492. package/web-ui/build/static/twig-stKDItz7.js +1 -0
  493. package/web-ui/build/static/typescript-BPdM-QOM.js +1 -0
  494. package/web-ui/build/static/typescript-DjSY4MvL.js +1 -0
  495. package/web-ui/build/static/typoscript-BQHgQtEL.js +1 -0
  496. package/web-ui/build/static/unrealscript-g8yNPQQ5.js +1 -0
  497. package/web-ui/build/static/uorazor-CnuMiQz9.js +1 -0
  498. package/web-ui/build/static/uri-CfioiemU.js +1 -0
  499. package/web-ui/build/static/v-B69Oau14.js +1 -0
  500. package/web-ui/build/static/vala-C0PjlK3l.js +1 -0
  501. package/web-ui/build/static/vala-COgjD3gU.js +1 -0
  502. package/web-ui/build/static/vbnet-CklOnyuR.js +1 -0
  503. package/web-ui/build/static/vbnet-MxHUC5Mz.js +1 -0
  504. package/web-ui/build/static/vbscript-BxQnA97w.js +1 -0
  505. package/web-ui/build/static/vbscript-html-DOtde7dD.js +1 -0
  506. package/web-ui/build/static/velocity-l8B9m66X.js +1 -0
  507. package/web-ui/build/static/verilog-DF6dF-Vp.js +1 -0
  508. package/web-ui/build/static/verilog-Dgb9SDRX.js +1 -0
  509. package/web-ui/build/static/vhdl-0GarU_AK.js +1 -0
  510. package/web-ui/build/static/vhdl-BfPQnOdI.js +1 -0
  511. package/web-ui/build/static/vim-Likg_-II.js +1 -0
  512. package/web-ui/build/static/vim-k53Sj7A3.js +1 -0
  513. package/web-ui/build/static/visual-basic-BNVZCzlV.js +1 -0
  514. package/web-ui/build/static/warpscript-BkLK5iuW.js +1 -0
  515. package/web-ui/build/static/wasm-CQhA1ece.js +1 -0
  516. package/web-ui/build/static/web-idl-DmNsI0kE.js +1 -0
  517. package/web-ui/build/static/wiki-B3z4pmra.js +1 -0
  518. package/web-ui/build/static/wolfram-Dsf21d_g.js +1 -0
  519. package/web-ui/build/static/wren-BZXZlzrL.js +1 -0
  520. package/web-ui/build/static/x86asm-DpWGZhJU.js +1 -0
  521. package/web-ui/build/static/xeora-8xPZN4DH.js +1 -0
  522. package/web-ui/build/static/xl-XkdWYMGl.js +1 -0
  523. package/web-ui/build/static/xml-BBdROEWt.js +1 -0
  524. package/web-ui/build/static/xml-doc-Cjs4OpTe.js +1 -0
  525. package/web-ui/build/static/xojo-nCen1FPx.js +1 -0
  526. package/web-ui/build/static/xquery-CBFkAxo1.js +1 -0
  527. package/web-ui/build/static/xquery-Dwsc1weX.js +1 -0
  528. package/web-ui/build/static/yaml-DzSAh6jl.js +1 -0
  529. package/web-ui/build/static/yaml-Lg8gnEUU.js +1 -0
  530. package/web-ui/build/static/yang-Bd-v2b0k.js +1 -0
  531. package/web-ui/build/static/zephir-wbJOizIa.js +1 -0
  532. package/web-ui/build/static/zig-Cz384XjE.js +1 -0
  533. package/web-ui/build/static/1c-Cxgysx0K.js +0 -1
  534. package/web-ui/build/static/abap-D9Pd8DUG.js +0 -1
  535. package/web-ui/build/static/abnf-CHs5P8yl.js +0 -1
  536. package/web-ui/build/static/abnf-D5OtFEZO.js +0 -1
  537. package/web-ui/build/static/accesslog-CTD5Frmr.js +0 -1
  538. package/web-ui/build/static/actionscript-BN0BKQI7.js +0 -1
  539. package/web-ui/build/static/actionscript-DQNwbYlU.js +0 -1
  540. package/web-ui/build/static/ada-8O7AmGii.js +0 -1
  541. package/web-ui/build/static/ada-p8y2qCLB.js +0 -1
  542. package/web-ui/build/static/agda-BzT4bKSx.js +0 -1
  543. package/web-ui/build/static/al-Caq7o9RG.js +0 -1
  544. package/web-ui/build/static/angelscript-DivMpSHf.js +0 -1
  545. package/web-ui/build/static/antlr4-B1D_tnKH.js +0 -1
  546. package/web-ui/build/static/apache-l88_LooT.js +0 -1
  547. package/web-ui/build/static/apacheconf-kaxXABFE.js +0 -1
  548. package/web-ui/build/static/apex-CSdSRWUv.js +0 -1
  549. package/web-ui/build/static/apl-CHwVg89T.js +0 -1
  550. package/web-ui/build/static/applescript-C64qdYZf.js +0 -1
  551. package/web-ui/build/static/applescript-CqTGrBVp.js +0 -1
  552. package/web-ui/build/static/aql-BwKKr7z7.js +0 -1
  553. package/web-ui/build/static/arcade-C6oqHEwh.js +0 -1
  554. package/web-ui/build/static/arduino-DB4E20Km.js +0 -1
  555. package/web-ui/build/static/arduino-DKpgmHjE.js +0 -1
  556. package/web-ui/build/static/arff-CBbf8E8O.js +0 -1
  557. package/web-ui/build/static/armasm-Bogg7lrz.js +0 -1
  558. package/web-ui/build/static/asciidoc-3ZlDsfKD.js +0 -1
  559. package/web-ui/build/static/asciidoc-Cc88g_au.js +0 -1
  560. package/web-ui/build/static/asm6502-B4rN-vZ1.js +0 -1
  561. package/web-ui/build/static/asmatmel-B8zZLWGk.js +0 -1
  562. package/web-ui/build/static/aspectj-XdNtCtbN.js +0 -1
  563. package/web-ui/build/static/aspnet-DZVKB5n1.js +0 -1
  564. package/web-ui/build/static/autohotkey-CxDWLQqM.js +0 -1
  565. package/web-ui/build/static/autohotkey-DJStPrtS.js +0 -1
  566. package/web-ui/build/static/autoit-BWirJOCg.js +0 -1
  567. package/web-ui/build/static/autoit-CJcfmsnl.js +0 -1
  568. package/web-ui/build/static/avisynth-C7JQe3L7.js +0 -1
  569. package/web-ui/build/static/avrasm-CMiuHQvu.js +0 -1
  570. package/web-ui/build/static/avro-idl-kaRVyxMP.js +0 -1
  571. package/web-ui/build/static/awk-CQ70Cyh1.js +0 -1
  572. package/web-ui/build/static/axapta-DRxAzlG8.js +0 -1
  573. package/web-ui/build/static/bash-B2XqEYpT.js +0 -1
  574. package/web-ui/build/static/bash-D-D1taMA.js +0 -1
  575. package/web-ui/build/static/basic-Da1x7OIZ.js +0 -1
  576. package/web-ui/build/static/basic-Dslk2ZwT.js +0 -1
  577. package/web-ui/build/static/batch-B0wJNPoF.js +0 -1
  578. package/web-ui/build/static/bbcode-BXDdc2HS.js +0 -1
  579. package/web-ui/build/static/bicep-iiH-nx9J.js +0 -1
  580. package/web-ui/build/static/birb-DuTqqibI.js +0 -1
  581. package/web-ui/build/static/bison-D1Hw6owA.js +0 -1
  582. package/web-ui/build/static/bnf-BRhzfr87.js +0 -1
  583. package/web-ui/build/static/bnf-u1Iy1Ku7.js +0 -1
  584. package/web-ui/build/static/brainfuck-C4R5VJ3_.js +0 -1
  585. package/web-ui/build/static/brainfuck-C8niWF0i.js +0 -1
  586. package/web-ui/build/static/brightscript-DavMfkHx.js +0 -1
  587. package/web-ui/build/static/bro-CEMPoYum.js +0 -1
  588. package/web-ui/build/static/bsl-COeqiMqc.js +0 -1
  589. package/web-ui/build/static/c-BaaVQwFt.js +0 -1
  590. package/web-ui/build/static/c-BvALLiMv.js +0 -1
  591. package/web-ui/build/static/c-like-C4yx6s7b.js +0 -1
  592. package/web-ui/build/static/cal-CSxgKgNR.js +0 -1
  593. package/web-ui/build/static/capnproto-C9cMEmIo.js +0 -1
  594. package/web-ui/build/static/ceylon-Dj67Lf2T.js +0 -1
  595. package/web-ui/build/static/cfscript-BhALECgj.js +0 -1
  596. package/web-ui/build/static/chaiscript-Cg5weNRB.js +0 -1
  597. package/web-ui/build/static/cil-BR-JCsPu.js +0 -1
  598. package/web-ui/build/static/clean-xJMmqZ4X.js +0 -1
  599. package/web-ui/build/static/clojure-9C3tRo_T.js +0 -1
  600. package/web-ui/build/static/clojure-C2KNwa0h.js +0 -1
  601. package/web-ui/build/static/clojure-repl-B-pke3nS.js +0 -1
  602. package/web-ui/build/static/cmake-C0BFHpjZ.js +0 -1
  603. package/web-ui/build/static/cmake-DYoa_Kz_.js +0 -1
  604. package/web-ui/build/static/cobol-Bj_qcMsY.js +0 -1
  605. package/web-ui/build/static/coffeescript-C2KEz8d0.js +0 -1
  606. package/web-ui/build/static/coffeescript-CQsP9N_M.js +0 -1
  607. package/web-ui/build/static/concurnas-Ceu24KQb.js +0 -1
  608. package/web-ui/build/static/coq-CP7VW0bo.js +0 -1
  609. package/web-ui/build/static/coq-DQry0I8v.js +0 -1
  610. package/web-ui/build/static/cos-DAjhu7pi.js +0 -1
  611. package/web-ui/build/static/cpp-BwHJ0JtK.js +0 -1
  612. package/web-ui/build/static/cpp-Dro3stzS.js +0 -1
  613. package/web-ui/build/static/crmsh-Bnwknpi9.js +0 -1
  614. package/web-ui/build/static/crystal-B7TVeIo6.js +0 -1
  615. package/web-ui/build/static/crystal-Cy5JlSrW.js +0 -1
  616. package/web-ui/build/static/csharp-CeouYZK3.js +0 -1
  617. package/web-ui/build/static/csharp-Dv2oxLAS.js +0 -1
  618. package/web-ui/build/static/cshtml-C9IRHsFH.js +0 -1
  619. package/web-ui/build/static/csp-C85pVqYT.js +0 -1
  620. package/web-ui/build/static/csp-MJntgwIj.js +0 -1
  621. package/web-ui/build/static/css-DWTfQjN4.js +0 -1
  622. package/web-ui/build/static/css-extras-BmVbsPjr.js +0 -1
  623. package/web-ui/build/static/csv-zUw-JkSo.js +0 -1
  624. package/web-ui/build/static/cypher-DQOOp0bG.js +0 -1
  625. package/web-ui/build/static/d-B7Cku9c8.js +0 -1
  626. package/web-ui/build/static/d-Lnzvs4Wd.js +0 -1
  627. package/web-ui/build/static/dart-CeK_W9Mg.js +0 -1
  628. package/web-ui/build/static/dart-Z5nwVCn_.js +0 -1
  629. package/web-ui/build/static/dataweave-Bv7GTsiJ.js +0 -1
  630. package/web-ui/build/static/dax-CxkNT7ml.js +0 -1
  631. package/web-ui/build/static/delphi-BnBBHmHI.js +0 -1
  632. package/web-ui/build/static/dhall-DzzkunJZ.js +0 -1
  633. package/web-ui/build/static/diff-BKZzYJzc.js +0 -1
  634. package/web-ui/build/static/diff-BfTiUZV3.js +0 -1
  635. package/web-ui/build/static/django-BY4l8873.js +0 -1
  636. package/web-ui/build/static/django-NRVGlUZ_.js +0 -1
  637. package/web-ui/build/static/dns-Dvc8YawT.js +0 -1
  638. package/web-ui/build/static/dns-zone-file-CkaiOrlK.js +0 -1
  639. package/web-ui/build/static/docker-BQXYNHPO.js +0 -1
  640. package/web-ui/build/static/dockerfile-CkHfYmB1.js +0 -1
  641. package/web-ui/build/static/dos-Cj0YcQAg.js +0 -1
  642. package/web-ui/build/static/dot-zgBkKTS-.js +0 -1
  643. package/web-ui/build/static/dsconfig-Yy07ZSap.js +0 -1
  644. package/web-ui/build/static/dts-DyTWnuFT.js +0 -1
  645. package/web-ui/build/static/dust-CKq7f4u7.js +0 -1
  646. package/web-ui/build/static/ebnf-DJRrwYUt.js +0 -1
  647. package/web-ui/build/static/ebnf-RHbRUELR.js +0 -1
  648. package/web-ui/build/static/editorconfig-BPnyfBW2.js +0 -1
  649. package/web-ui/build/static/eiffel-yj-2JB74.js +0 -1
  650. package/web-ui/build/static/ejs-DitQDEAj.js +0 -1
  651. package/web-ui/build/static/elixir-3QGdRPiZ.js +0 -1
  652. package/web-ui/build/static/elixir-DjVihWJd.js +0 -1
  653. package/web-ui/build/static/elm-BY6TIO5U.js +0 -1
  654. package/web-ui/build/static/elm-CRkIBVgc.js +0 -1
  655. package/web-ui/build/static/erb-Djhpl6Qv.js +0 -1
  656. package/web-ui/build/static/erb-T-e_0q4M.js +0 -1
  657. package/web-ui/build/static/erlang-BHne0Sow.js +0 -1
  658. package/web-ui/build/static/erlang-BwFMbXMn.js +0 -1
  659. package/web-ui/build/static/erlang-repl-BkeLNBB1.js +0 -1
  660. package/web-ui/build/static/etlua-C7wAJl9b.js +0 -1
  661. package/web-ui/build/static/excel-DxE1g8nj.js +0 -1
  662. package/web-ui/build/static/excel-formula-CJfyOSIQ.js +0 -1
  663. package/web-ui/build/static/factor-CoKZ2w9G.js +0 -1
  664. package/web-ui/build/static/false-R3UgYwdZ.js +0 -1
  665. package/web-ui/build/static/firestore-security-rules-Na-FK95T.js +0 -1
  666. package/web-ui/build/static/fix-CvUjGqrP.js +0 -1
  667. package/web-ui/build/static/flix-DZcdobtd.js +0 -1
  668. package/web-ui/build/static/flow-Bs4ybDfF.js +0 -1
  669. package/web-ui/build/static/fortran-Bo7HQHVz.js +0 -1
  670. package/web-ui/build/static/fortran-C01QKAjv.js +0 -1
  671. package/web-ui/build/static/fsharp-Bl0db-E4.js +0 -1
  672. package/web-ui/build/static/fsharp-CyZIu__H.js +0 -1
  673. package/web-ui/build/static/ftl-CWXbnuJ8.js +0 -1
  674. package/web-ui/build/static/gams-BqHeBOiK.js +0 -1
  675. package/web-ui/build/static/gap-Dotaw9CX.js +0 -1
  676. package/web-ui/build/static/gauss-BT78eAPL.js +0 -1
  677. package/web-ui/build/static/gcode-MKCsCHbV.js +0 -1
  678. package/web-ui/build/static/gcode-bZIzS7dq.js +0 -1
  679. package/web-ui/build/static/gdscript-BfFdzROd.js +0 -1
  680. package/web-ui/build/static/gedcom-C-0fkvQT.js +0 -1
  681. package/web-ui/build/static/gherkin-Bgefs1Ac.js +0 -1
  682. package/web-ui/build/static/gherkin-BwLhttSG.js +0 -1
  683. package/web-ui/build/static/git-Cf3XNFwK.js +0 -1
  684. package/web-ui/build/static/glsl-BRBvgtKX.js +0 -1
  685. package/web-ui/build/static/glsl-Ci0Ywihb.js +0 -1
  686. package/web-ui/build/static/gml-BGED3ZV5.js +0 -1
  687. package/web-ui/build/static/gml-Bo5ybH2Z.js +0 -1
  688. package/web-ui/build/static/gn-Dvq3uq8q.js +0 -1
  689. package/web-ui/build/static/go-DEXubI7V.js +0 -1
  690. package/web-ui/build/static/go-DQ-rOBtr.js +0 -1
  691. package/web-ui/build/static/go-module-CJ0C_rZQ.js +0 -1
  692. package/web-ui/build/static/golo-BMZyESOG.js +0 -1
  693. package/web-ui/build/static/gradle-DamtVT5m.js +0 -1
  694. package/web-ui/build/static/graphql-BERukF58.js +0 -1
  695. package/web-ui/build/static/groovy-B-WmRKEQ.js +0 -1
  696. package/web-ui/build/static/groovy-BzgGHCpW.js +0 -1
  697. package/web-ui/build/static/haml-B50hpu0J.js +0 -1
  698. package/web-ui/build/static/haml-vZAqNnLr.js +0 -1
  699. package/web-ui/build/static/handlebars-ChvSiZA5.js +0 -1
  700. package/web-ui/build/static/handlebars-z32GTNvC.js +0 -1
  701. package/web-ui/build/static/haskell-BWLVCHRl.js +0 -1
  702. package/web-ui/build/static/haskell-CZ4HT-QV.js +0 -1
  703. package/web-ui/build/static/haxe-Dff2UzKb.js +0 -1
  704. package/web-ui/build/static/haxe-DtUMfT4Y.js +0 -1
  705. package/web-ui/build/static/hcl-CPdCn21C.js +0 -1
  706. package/web-ui/build/static/hlsl-Ddnzpafa.js +0 -1
  707. package/web-ui/build/static/hoon-BmzYN5g6.js +0 -1
  708. package/web-ui/build/static/hpkp-Cf4BtYzp.js +0 -1
  709. package/web-ui/build/static/hsp-CtNdizh5.js +0 -1
  710. package/web-ui/build/static/hsts-ChmUnrWY.js +0 -1
  711. package/web-ui/build/static/htmlbars-FvxUz397.js +0 -1
  712. package/web-ui/build/static/http-C91BqIc4.js +0 -1
  713. package/web-ui/build/static/http-DUkMpz6R.js +0 -1
  714. package/web-ui/build/static/hy-BgEorqc6.js +0 -1
  715. package/web-ui/build/static/ichigojam-CyJ85XqZ.js +0 -1
  716. package/web-ui/build/static/icon-CWVwUSUB.js +0 -1
  717. package/web-ui/build/static/icu-message-format-DQ5Vf-57.js +0 -1
  718. package/web-ui/build/static/idris-25epvnkz.js +0 -1
  719. package/web-ui/build/static/iecst-D_Q4A3Ao.js +0 -1
  720. package/web-ui/build/static/ignore-Dn3TUem-.js +0 -1
  721. package/web-ui/build/static/index-1GAqwN2P.js +0 -1029
  722. package/web-ui/build/static/index-Bbt0N_qA.css +0 -1
  723. package/web-ui/build/static/index-COM0znG6.js +0 -13
  724. package/web-ui/build/static/index-Cq1FjdFd.js +0 -1
  725. package/web-ui/build/static/inform7-CFrNuUCE.js +0 -1
  726. package/web-ui/build/static/inform7-DKEqg7Td.js +0 -1
  727. package/web-ui/build/static/ini-C-WeV3rA.js +0 -1
  728. package/web-ui/build/static/ini-DNeLlQLb.js +0 -1
  729. package/web-ui/build/static/io-CHx2lS-R.js +0 -1
  730. package/web-ui/build/static/irpf90-D4fNrYKT.js +0 -1
  731. package/web-ui/build/static/isbl-BwxgZUmw.js +0 -1
  732. package/web-ui/build/static/j-EkuSi-Ss.js +0 -1
  733. package/web-ui/build/static/java-BtphFsNb.js +0 -1
  734. package/web-ui/build/static/java-D6Ksxtks.js +0 -1
  735. package/web-ui/build/static/javadoc-COfrpXIF.js +0 -1
  736. package/web-ui/build/static/javadoclike-C0DIBgSo.js +0 -1
  737. package/web-ui/build/static/javascript-D9_bdns_.js +0 -1
  738. package/web-ui/build/static/javastacktrace-BgesTIoE.js +0 -1
  739. package/web-ui/build/static/jboss-cli-BCBm8RaA.js +0 -1
  740. package/web-ui/build/static/jexl-Bur7gjJl.js +0 -1
  741. package/web-ui/build/static/jolie-CG2HtqVF.js +0 -1
  742. package/web-ui/build/static/jq-BjcZo4Ta.js +0 -1
  743. package/web-ui/build/static/js-extras-DPWP-xWb.js +0 -1
  744. package/web-ui/build/static/js-templates-BYQFylC_.js +0 -1
  745. package/web-ui/build/static/jsdoc-CiOxmgl5.js +0 -1
  746. package/web-ui/build/static/json-C2-b3Xpo.js +0 -1
  747. package/web-ui/build/static/json-D7ok2MJS.js +0 -1
  748. package/web-ui/build/static/json5-DRNOdkus.js +0 -1
  749. package/web-ui/build/static/jsonp-FFJzAxUk.js +0 -1
  750. package/web-ui/build/static/jsstacktrace-C4ki-LaP.js +0 -1
  751. package/web-ui/build/static/jsx-ZxGQpUNq.js +0 -1
  752. package/web-ui/build/static/julia-7ooB8a4b.js +0 -1
  753. package/web-ui/build/static/julia-DzjCRNEr.js +0 -1
  754. package/web-ui/build/static/julia-repl-pceaZSLt.js +0 -1
  755. package/web-ui/build/static/keepalived-BQgTZddo.js +0 -1
  756. package/web-ui/build/static/keyman-Cqlba6tn.js +0 -1
  757. package/web-ui/build/static/kotlin-BygmvqBG.js +0 -1
  758. package/web-ui/build/static/kotlin-DrN0ycN8.js +0 -1
  759. package/web-ui/build/static/kumir-YhasANvI.js +0 -1
  760. package/web-ui/build/static/kusto-LQa6MD2B.js +0 -1
  761. package/web-ui/build/static/lasso-Ca3RFbcE.js +0 -1
  762. package/web-ui/build/static/latex-BoQj21fG.js +0 -1
  763. package/web-ui/build/static/latex-ZxewREOs.js +0 -1
  764. package/web-ui/build/static/latte-40SpSG4q.js +0 -1
  765. package/web-ui/build/static/ldif-CzTJxBjJ.js +0 -1
  766. package/web-ui/build/static/leaf-DO9LaVNe.js +0 -1
  767. package/web-ui/build/static/less-BoClFdvD.js +0 -1
  768. package/web-ui/build/static/less-C1s1U6-n.js +0 -1
  769. package/web-ui/build/static/lilypond-D5wHKW8A.js +0 -1
  770. package/web-ui/build/static/liquid-BRQzL-S2.js +0 -1
  771. package/web-ui/build/static/lisp-8q-QRYKi.js +0 -1
  772. package/web-ui/build/static/lisp-J8TPaPPb.js +0 -1
  773. package/web-ui/build/static/livecodeserver-Dcr3yGmD.js +0 -1
  774. package/web-ui/build/static/livescript-BUEQmlAe.js +0 -1
  775. package/web-ui/build/static/livescript-DsId63Ot.js +0 -1
  776. package/web-ui/build/static/llvm-B3CCzYxo.js +0 -1
  777. package/web-ui/build/static/llvm-kQGfyqnj.js +0 -1
  778. package/web-ui/build/static/log-BlA5152b.js +0 -1
  779. package/web-ui/build/static/lolcode-8WRAMeb2.js +0 -1
  780. package/web-ui/build/static/lsl-BZUMQNE2.js +0 -1
  781. package/web-ui/build/static/lua-C4hJ-Gql.js +0 -1
  782. package/web-ui/build/static/lua-RxOeZV_z.js +0 -1
  783. package/web-ui/build/static/magma-U9kYv2QD.js +0 -1
  784. package/web-ui/build/static/makefile-1Cu2rVDg.js +0 -1
  785. package/web-ui/build/static/makefile-mmZ8lbad.js +0 -1
  786. package/web-ui/build/static/markdown-CCqDOXhR.js +0 -1
  787. package/web-ui/build/static/markdown-CVOmqzdD.js +0 -1
  788. package/web-ui/build/static/markup-templating-C7eiVR0j.js +0 -1
  789. package/web-ui/build/static/mathematica-BbKHQ6zH.js +0 -1
  790. package/web-ui/build/static/matlab-BbMwJ6T0.js +0 -1
  791. package/web-ui/build/static/matlab-gHn5UhMb.js +0 -1
  792. package/web-ui/build/static/maxima-CuB3n0ri.js +0 -1
  793. package/web-ui/build/static/maxscript-jhOix4gn.js +0 -1
  794. package/web-ui/build/static/mel-Bpjqa7Ec.js +0 -1
  795. package/web-ui/build/static/mel-QxMxBq03.js +0 -1
  796. package/web-ui/build/static/mercury-C8IdDyFB.js +0 -1
  797. package/web-ui/build/static/mermaid-v41WDixf.js +0 -1
  798. package/web-ui/build/static/mipsasm-kYH27KXy.js +0 -1
  799. package/web-ui/build/static/mizar-BCJI9Jxa.js +0 -1
  800. package/web-ui/build/static/mizar-BCxP_6pj.js +0 -1
  801. package/web-ui/build/static/mojolicious-C7DC_CUp.js +0 -1
  802. package/web-ui/build/static/mongodb-DvBX00WY.js +0 -1
  803. package/web-ui/build/static/monkey-B7KjOZtt.js +0 -1
  804. package/web-ui/build/static/monkey-CFKZd6lx.js +0 -1
  805. package/web-ui/build/static/moonscript-44jD_uAZ.js +0 -1
  806. package/web-ui/build/static/moonscript-B3KahKf5.js +0 -1
  807. package/web-ui/build/static/n1ql-CxJOkBbD.js +0 -1
  808. package/web-ui/build/static/n1ql-FJFE6P7E.js +0 -1
  809. package/web-ui/build/static/n4js-D5jJTGdO.js +0 -1
  810. package/web-ui/build/static/nand2tetris-hdl-DlsCaky2.js +0 -1
  811. package/web-ui/build/static/naniscript-C3YsvWgN.js +0 -1
  812. package/web-ui/build/static/nasm-DCGKvOZq.js +0 -1
  813. package/web-ui/build/static/neon-nyFNinx-.js +0 -1
  814. package/web-ui/build/static/nevod-C5PnMzAW.js +0 -1
  815. package/web-ui/build/static/nginx-BUgMakda.js +0 -1
  816. package/web-ui/build/static/nginx-C5JBKsuz.js +0 -1
  817. package/web-ui/build/static/nim-BFu6fQgc.js +0 -1
  818. package/web-ui/build/static/nim-D0uh-nR9.js +0 -1
  819. package/web-ui/build/static/nix-BiOn3l1e.js +0 -1
  820. package/web-ui/build/static/nix-tapJszPl.js +0 -1
  821. package/web-ui/build/static/node-repl-BTp9CFzE.js +0 -1
  822. package/web-ui/build/static/nsis-B7hV5A06.js +0 -1
  823. package/web-ui/build/static/nsis-DLshnCv-.js +0 -1
  824. package/web-ui/build/static/objectivec-BKjt_sKE.js +0 -1
  825. package/web-ui/build/static/objectivec-KUMYgk5w.js +0 -1
  826. package/web-ui/build/static/ocaml-CV2nKA_A.js +0 -1
  827. package/web-ui/build/static/ocaml-ZyZ-dXmE.js +0 -1
  828. package/web-ui/build/static/opencl-igIWfc0n.js +0 -1
  829. package/web-ui/build/static/openqasm-ByCPmCpS.js +0 -1
  830. package/web-ui/build/static/openscad-Db6ypam1.js +0 -1
  831. package/web-ui/build/static/oxygene-CBbaQWRB.js +0 -1
  832. package/web-ui/build/static/oz-lEl_mXVw.js +0 -1
  833. package/web-ui/build/static/parigp-c2FbheBr.js +0 -1
  834. package/web-ui/build/static/parser-Ba6uivHf.js +0 -1
  835. package/web-ui/build/static/parser3-Cm5cgVSQ.js +0 -1
  836. package/web-ui/build/static/pascal-JgY-dd52.js +0 -1
  837. package/web-ui/build/static/pascaligo-BjSdyegq.js +0 -1
  838. package/web-ui/build/static/pcaxis-CDdxDfqJ.js +0 -1
  839. package/web-ui/build/static/peoplecode-BCPhsbK6.js +0 -1
  840. package/web-ui/build/static/perl-ClVPikLH.js +0 -1
  841. package/web-ui/build/static/perl-pRjXJ4uc.js +0 -1
  842. package/web-ui/build/static/pf-D6VrFfuf.js +0 -1
  843. package/web-ui/build/static/pgsql-DAv4ezCU.js +0 -1
  844. package/web-ui/build/static/php-Bz3Bn1G_.js +0 -1
  845. package/web-ui/build/static/php-extras-B8k1hZ7O.js +0 -1
  846. package/web-ui/build/static/php-template-DJJiMC8Q.js +0 -1
  847. package/web-ui/build/static/php-zCXGNRnB.js +0 -1
  848. package/web-ui/build/static/phpdoc-CpnPjGDk.js +0 -1
  849. package/web-ui/build/static/plaintext-BRkRKfKN.js +0 -1
  850. package/web-ui/build/static/plsql-BARhkvBo.js +0 -1
  851. package/web-ui/build/static/pony-guYTDEtj.js +0 -1
  852. package/web-ui/build/static/powerquery-DcM2Yj3O.js +0 -1
  853. package/web-ui/build/static/powershell-DJ8ftCxt.js +0 -1
  854. package/web-ui/build/static/powershell-DSRUQzqt.js +0 -1
  855. package/web-ui/build/static/processing-BCmW5Adu.js +0 -1
  856. package/web-ui/build/static/processing-DDMjYIa-.js +0 -1
  857. package/web-ui/build/static/profile-C9J13WXr.js +0 -1
  858. package/web-ui/build/static/prolog-CUyUfIUU.js +0 -1
  859. package/web-ui/build/static/prolog-wjvrQEy3.js +0 -1
  860. package/web-ui/build/static/promql-BkN3_kL1.js +0 -1
  861. package/web-ui/build/static/properties-B7zGUFLC.js +0 -1
  862. package/web-ui/build/static/properties-DHKWbrVz.js +0 -1
  863. package/web-ui/build/static/protobuf-CVwMi4py.js +0 -1
  864. package/web-ui/build/static/protobuf-CWUzVSVD.js +0 -1
  865. package/web-ui/build/static/psl-DdX6HMd3.js +0 -1
  866. package/web-ui/build/static/pug-D-Sn469u.js +0 -1
  867. package/web-ui/build/static/puppet-B6ZR9-D4.js +0 -1
  868. package/web-ui/build/static/puppet-m66LRiIo.js +0 -1
  869. package/web-ui/build/static/pure-7deZH-sc.js +0 -1
  870. package/web-ui/build/static/purebasic-DQLUcurt.js +0 -1
  871. package/web-ui/build/static/purebasic-DYMw9jTh.js +0 -1
  872. package/web-ui/build/static/purescript-DpysoAya.js +0 -1
  873. package/web-ui/build/static/python-D66_vufV.js +0 -1
  874. package/web-ui/build/static/python-DYxaIENO.js +0 -1
  875. package/web-ui/build/static/python-repl-DPjvQNxf.js +0 -1
  876. package/web-ui/build/static/q-DDHNKITB.js +0 -1
  877. package/web-ui/build/static/q-xV1Q9ILw.js +0 -1
  878. package/web-ui/build/static/qml-DSENM1u-.js +0 -1
  879. package/web-ui/build/static/qml-DfhbPQ-t.js +0 -1
  880. package/web-ui/build/static/qore-LQCffrzv.js +0 -1
  881. package/web-ui/build/static/qsharp-BcSf2ZGH.js +0 -1
  882. package/web-ui/build/static/r-Dc6GR0yQ.js +0 -1
  883. package/web-ui/build/static/r-vludgk3t.js +0 -1
  884. package/web-ui/build/static/racket-DJYIkQwh.js +0 -1
  885. package/web-ui/build/static/reason-D75i_ede.js +0 -1
  886. package/web-ui/build/static/reasonml-DhRXKjo3.js +0 -1
  887. package/web-ui/build/static/regex-D67iENuu.js +0 -1
  888. package/web-ui/build/static/rego-CJ2Eeb4g.js +0 -1
  889. package/web-ui/build/static/renpy-DUDNyFqb.js +0 -1
  890. package/web-ui/build/static/rest-Dn7SYTNt.js +0 -1
  891. package/web-ui/build/static/rib-BVUY5dIb.js +0 -1
  892. package/web-ui/build/static/rip-Bu9K3Hu9.js +0 -1
  893. package/web-ui/build/static/roboconf-CUzY2sLf.js +0 -1
  894. package/web-ui/build/static/roboconf-DBEmoD-Z.js +0 -1
  895. package/web-ui/build/static/robotframework-70EV7ffL.js +0 -1
  896. package/web-ui/build/static/routeros-KgfwmDVO.js +0 -1
  897. package/web-ui/build/static/rsl-D3oCnFGa.js +0 -1
  898. package/web-ui/build/static/ruby-Dqa0HtB4.js +0 -1
  899. package/web-ui/build/static/ruby-PeJYeL87.js +0 -1
  900. package/web-ui/build/static/ruleslanguage-vo3eM_mz.js +0 -1
  901. package/web-ui/build/static/rust-CwU6t4_Y.js +0 -1
  902. package/web-ui/build/static/rust-WXADYh9r.js +0 -1
  903. package/web-ui/build/static/sas-Cg6ADdDH.js +0 -1
  904. package/web-ui/build/static/sas-DWoEaeXv.js +0 -1
  905. package/web-ui/build/static/sass-DbHJHeBV.js +0 -1
  906. package/web-ui/build/static/scala-B6oE2QKr.js +0 -1
  907. package/web-ui/build/static/scala-NFQGPaYa.js +0 -1
  908. package/web-ui/build/static/scheme-C8Jx4oX7.js +0 -1
  909. package/web-ui/build/static/scheme-CUMBJaUf.js +0 -1
  910. package/web-ui/build/static/scilab-D1VqsdSy.js +0 -1
  911. package/web-ui/build/static/scss-CSH87oUD.js +0 -1
  912. package/web-ui/build/static/scss-GLX6NNDB.js +0 -1
  913. package/web-ui/build/static/shell-CbOhN21M.js +0 -1
  914. package/web-ui/build/static/shell-session-BQPdJJZS.js +0 -1
  915. package/web-ui/build/static/smali-BYbEOQX5.js +0 -1
  916. package/web-ui/build/static/smali-DcoYSibw.js +0 -1
  917. package/web-ui/build/static/smalltalk-CQkedD8N.js +0 -1
  918. package/web-ui/build/static/smalltalk-DihI81Cd.js +0 -1
  919. package/web-ui/build/static/smarty-RRvllERL.js +0 -1
  920. package/web-ui/build/static/sml-BJkMhBLU.js +0 -1
  921. package/web-ui/build/static/sml-Bm2IN0wP.js +0 -1
  922. package/web-ui/build/static/solidity-CsCW8kyX.js +0 -1
  923. package/web-ui/build/static/solution-file-B5YSxWQt.js +0 -1
  924. package/web-ui/build/static/soy-Dek2_YCW.js +0 -1
  925. package/web-ui/build/static/sparql-BeTxhWa4.js +0 -1
  926. package/web-ui/build/static/splunk-spl-hq6BW85c.js +0 -1
  927. package/web-ui/build/static/sqf-C-u4en-i.js +0 -1
  928. package/web-ui/build/static/sqf-DP0IhDYt.js +0 -1
  929. package/web-ui/build/static/sql-BzL0uonb.js +0 -1
  930. package/web-ui/build/static/sql-Zy6tV_us.js +0 -1
  931. package/web-ui/build/static/sql_more-BP2G6Vh5.js +0 -1
  932. package/web-ui/build/static/squirrel-C3MnMYx5.js +0 -1
  933. package/web-ui/build/static/stan-7iKo211G.js +0 -1
  934. package/web-ui/build/static/stan-C1lcIZjJ.js +0 -1
  935. package/web-ui/build/static/stata-DJ6GKcaw.js +0 -1
  936. package/web-ui/build/static/step21-Pg-uClDB.js +0 -1
  937. package/web-ui/build/static/stylus-DFAxYuRi.js +0 -1
  938. package/web-ui/build/static/stylus-DT6xc_Nv.js +0 -1
  939. package/web-ui/build/static/subunit-aGvGJXZZ.js +0 -1
  940. package/web-ui/build/static/swift-Sqmdmb4O.js +0 -1
  941. package/web-ui/build/static/swift-cUX-MVl4.js +0 -1
  942. package/web-ui/build/static/systemd-qIaU0Trg.js +0 -1
  943. package/web-ui/build/static/t4-cs-DtO6Bgg1.js +0 -1
  944. package/web-ui/build/static/t4-templating-DSSM2N5M.js +0 -1
  945. package/web-ui/build/static/t4-vb-tKIiE5uC.js +0 -1
  946. package/web-ui/build/static/taggerscript-Dt2Kgr-W.js +0 -1
  947. package/web-ui/build/static/tap-BMEgEMIk.js +0 -1
  948. package/web-ui/build/static/tap-C4TdzsNz.js +0 -1
  949. package/web-ui/build/static/tcl-B6k4Af9f.js +0 -1
  950. package/web-ui/build/static/tcl-Dq4_i5q5.js +0 -1
  951. package/web-ui/build/static/textile-C0uqAPk0.js +0 -1
  952. package/web-ui/build/static/thrift-DybQEjaA.js +0 -1
  953. package/web-ui/build/static/toml-DG1a2FSS.js +0 -1
  954. package/web-ui/build/static/tp-BwdfEEST.js +0 -1
  955. package/web-ui/build/static/tremor-JSTukmt-.js +0 -1
  956. package/web-ui/build/static/tsx-YUfsqzPf.js +0 -1
  957. package/web-ui/build/static/tt2-D9A_ACnF.js +0 -1
  958. package/web-ui/build/static/turtle-BKxwLa_-.js +0 -1
  959. package/web-ui/build/static/twig-BeEDK9Ri.js +0 -1
  960. package/web-ui/build/static/twig-TnOWwD7-.js +0 -1
  961. package/web-ui/build/static/typescript-BjZl8C3q.js +0 -1
  962. package/web-ui/build/static/typescript-qtW5Xh-j.js +0 -1
  963. package/web-ui/build/static/typoscript-Dj4sO7Jx.js +0 -1
  964. package/web-ui/build/static/unrealscript-DInimzNQ.js +0 -1
  965. package/web-ui/build/static/uorazor-B_9tFVKw.js +0 -1
  966. package/web-ui/build/static/uri-CTuMA7_k.js +0 -1
  967. package/web-ui/build/static/v-CvE0Ujqx.js +0 -1
  968. package/web-ui/build/static/vala-CLGf3xEs.js +0 -1
  969. package/web-ui/build/static/vala-Dm8LmEpm.js +0 -1
  970. package/web-ui/build/static/vbnet-Cwm2TjGP.js +0 -1
  971. package/web-ui/build/static/vbnet-LCd9GiMJ.js +0 -1
  972. package/web-ui/build/static/vbscript-DVj94_-y.js +0 -1
  973. package/web-ui/build/static/vbscript-html-CDHtSX-o.js +0 -1
  974. package/web-ui/build/static/velocity-C-OrhdDf.js +0 -1
  975. package/web-ui/build/static/verilog-Bh18N7TM.js +0 -1
  976. package/web-ui/build/static/verilog-C-YNXb8j.js +0 -1
  977. package/web-ui/build/static/vhdl-BFNm41_R.js +0 -1
  978. package/web-ui/build/static/vhdl-CyLgBDKk.js +0 -1
  979. package/web-ui/build/static/vim-BTafUqR0.js +0 -1
  980. package/web-ui/build/static/vim-Xg1pEHJe.js +0 -1
  981. package/web-ui/build/static/visual-basic-CDgUOlvq.js +0 -1
  982. package/web-ui/build/static/warpscript-1ELDmkt5.js +0 -1
  983. package/web-ui/build/static/wasm-C25KdQk5.js +0 -1
  984. package/web-ui/build/static/web-idl-CsdVYgpL.js +0 -1
  985. package/web-ui/build/static/wiki-4RrZuyhV.js +0 -1
  986. package/web-ui/build/static/wolfram-DMiRIwvp.js +0 -1
  987. package/web-ui/build/static/wren-Vk5PBcVB.js +0 -1
  988. package/web-ui/build/static/x86asm-D9TFTgye.js +0 -1
  989. package/web-ui/build/static/xeora-SAn_K3Ts.js +0 -1
  990. package/web-ui/build/static/xl-Da2Fl8D9.js +0 -1
  991. package/web-ui/build/static/xml-CeLm6l4V.js +0 -1
  992. package/web-ui/build/static/xml-doc-DypYH408.js +0 -1
  993. package/web-ui/build/static/xojo-CrfMiv1-.js +0 -1
  994. package/web-ui/build/static/xquery-8tOqVkgP.js +0 -1
  995. package/web-ui/build/static/xquery-CLOnNPtx.js +0 -1
  996. package/web-ui/build/static/yaml-CXiycPmU.js +0 -1
  997. package/web-ui/build/static/yaml-cIKjKW-H.js +0 -1
  998. package/web-ui/build/static/yang-KVshO9Ce.js +0 -1
  999. package/web-ui/build/static/zephir-B88JNqlj.js +0 -1
  1000. package/web-ui/build/static/zig-C6GYIeFs.js +0 -1
@@ -217,6 +217,913 @@ describe('FlowExecutor', () => {
217
217
  await expect(fe.executeFlow('missing-flow', {})).rejects.toThrow('Flow not found');
218
218
  });
219
219
 
220
+ // --- schema validation gate (Phase 0) ---
221
+
222
+ test('executeFlow refuses to run a flow with cycles (schema gate)', async () => {
223
+ // The executor's belt-and-suspenders validator should reject this
224
+ // BEFORE topologicalSort silently drops nodes.
225
+ const cyclicFlow = {
226
+ id: 'flow-cycle',
227
+ name: 'Cyclic flow',
228
+ nodes: [
229
+ { id: 'a', type: 'agent', data: { agentId: 'x' } },
230
+ { id: 'b', type: 'agent', data: { agentId: 'y' } },
231
+ ],
232
+ edges: [
233
+ { source: 'a', target: 'b' },
234
+ { source: 'b', target: 'a' },
235
+ ],
236
+ };
237
+ stateManager.getFlow = jest.fn().mockResolvedValue(cyclicFlow);
238
+ stateManager.createFlowRun = jest.fn();
239
+ await expect(fe.executeFlow('flow-cycle', {})).rejects.toThrow(/invalid|cycle/i);
240
+ // Critical: must not have created a run (we refused before persistence)
241
+ expect(stateManager.createFlowRun).not.toHaveBeenCalled();
242
+ });
243
+
244
+ test('executeFlow refuses to run a flow with dangling edges', async () => {
245
+ const brokenFlow = {
246
+ id: 'flow-bad-edges',
247
+ name: 'Broken edges',
248
+ nodes: [{ id: 'in', type: 'input', data: {} }],
249
+ edges: [{ source: 'in', target: 'ghost' }],
250
+ };
251
+ stateManager.getFlow = jest.fn().mockResolvedValue(brokenFlow);
252
+ stateManager.createFlowRun = jest.fn();
253
+ await expect(fe.executeFlow('flow-bad-edges', {})).rejects.toThrow(/invalid|ghost/i);
254
+ expect(stateManager.createFlowRun).not.toHaveBeenCalled();
255
+ });
256
+
257
+ test('executeFlow refuses to run when an agent node has no agent assigned (clear error)', async () => {
258
+ // Schema allows save with empty agentId (drafts). Executor must
259
+ // refuse to run with a friendly message naming the unbound nodes.
260
+ const noAgentFlow = {
261
+ id: 'flow-no-agent',
262
+ name: 'Agent without ID',
263
+ nodes: [
264
+ { id: 'in', type: 'input', data: {} },
265
+ { id: 'ag', type: 'agent', data: { label: 'Writer' } }, // no agentId
266
+ ],
267
+ edges: [{ source: 'in', target: 'ag' }],
268
+ };
269
+ stateManager.getFlow = jest.fn().mockResolvedValue(noAgentFlow);
270
+ stateManager.createFlowRun = jest.fn().mockResolvedValue({ id: 'run-na' });
271
+ stateManager.updateFlowRun = jest.fn().mockResolvedValue();
272
+
273
+ const result = await fe.executeFlow('flow-no-agent', {});
274
+ expect(result.status).toBe('failed');
275
+ expect(result.error).toMatch(/Writer|no agent assigned/i);
276
+ });
277
+
278
+ // --- Phase 1: typed-input assembly in executeAgentNode ---
279
+
280
+ test('executeAgentNode (v2) exposes typed inputs as template variables', async () => {
281
+ // Build a v2 node that declares two inputs and a prompt template
282
+ // referencing both by name. The mocked agent finishes immediately.
283
+ const node = {
284
+ id: 'w', type: 'agent', data: { agentId: 'writer', promptTemplate: 'Write about {{topic}} ({{count}})' },
285
+ inputs: [{ name: 'topic', type: 'text', required: true },
286
+ { name: 'count', type: 'text', required: true }],
287
+ outputs: [{ name: 'draft', type: 'text' }],
288
+ };
289
+ const flow = {
290
+ id: 'f', name: 'Typed flow', schemaVersion: 2,
291
+ nodes: [
292
+ { id: 'in', type: 'input', data: {}, inputs: [], outputs: [{ name: 'topic', type: 'text' }] },
293
+ node,
294
+ ],
295
+ edges: [
296
+ { source: 'in', sourceField: 'topic', target: 'w', targetField: 'topic' },
297
+ { source: 'in', sourceField: 'topic', target: 'w', targetField: 'count' },
298
+ ],
299
+ };
300
+ const context = {
301
+ input: '',
302
+ // Source result advertises typed outputs map (preferred shape).
303
+ nodeOutputs: { in: { type: 'input', outputs: { topic: 'AI safety' } } },
304
+ variables: {},
305
+ sortedNodes: flow.nodes,
306
+ flow,
307
+ };
308
+
309
+ // Stub agent + completion: agentPool returns a usable agent; the
310
+ // executor queues a message and waits for completion. We resolve it
311
+ // immediately via notifyAgentCompletion to avoid the timeout path.
312
+ const agent = { id: 'writer', name: 'Writer', mode: 'chat', conversationHistory: [] };
313
+ agentPool.getAgent = jest.fn().mockResolvedValue(agent);
314
+ agentPool.persistAgentState = jest.fn().mockResolvedValue();
315
+ agentPool.clearConversation = jest.fn().mockResolvedValue();
316
+ stateManager.updateFlowRun = jest.fn().mockResolvedValue();
317
+ stateManager.getFlowRun = jest.fn().mockResolvedValue({ id: 'run-1', nodeStates: {} });
318
+
319
+ let capturedPrompt = null;
320
+ messageProcessor.processMessage = jest.fn().mockImplementation(async (id, prompt) => {
321
+ capturedPrompt = prompt;
322
+ // Resolve completion on next tick so the awaiter is registered.
323
+ // v2 node declares outputs[draft] → must include it in the bag.
324
+ setImmediate(() => fe.notifyAgentCompletion(id, {
325
+ summary: 'done', success: true,
326
+ outputs: { draft: 'an article' },
327
+ }));
328
+ });
329
+
330
+ await fe.executeAgentNode(node, context, 'run-1', null, flow);
331
+ expect(capturedPrompt).toBe('Write about AI safety (AI safety)');
332
+ });
333
+
334
+ test('executeAgentNode (v2) stores structured outputs on the node result', async () => {
335
+ // When the agent emits an outputs bag via job-done, the executor
336
+ // should store it under .outputs on the node's recorded result, so
337
+ // downstream nodes can read it via assembleNodeInputs.
338
+ const node = {
339
+ id: 'w', type: 'agent', data: { agentId: 'writer', promptTemplate: '{{topic}}' },
340
+ inputs: [{ name: 'topic', type: 'text', required: true }],
341
+ outputs: [{ name: 'draft', type: 'text' }, { name: 'wordCount', type: 'number' }],
342
+ };
343
+ const flow = {
344
+ id: 'f', name: 'Structured outputs', schemaVersion: 2,
345
+ nodes: [
346
+ { id: 'in', type: 'input', data: {}, inputs: [], outputs: [{ name: 'topic', type: 'text' }] },
347
+ node,
348
+ ],
349
+ edges: [{ source: 'in', sourceField: 'topic', target: 'w', targetField: 'topic' }],
350
+ };
351
+ const context = {
352
+ input: '',
353
+ nodeOutputs: { in: { type: 'input', outputs: { topic: 'AI safety' } } },
354
+ variables: {}, sortedNodes: flow.nodes, flow,
355
+ };
356
+ agentPool.getAgent = jest.fn().mockResolvedValue({ id: 'writer', mode: 'chat' });
357
+ agentPool.persistAgentState = jest.fn().mockResolvedValue();
358
+ agentPool.clearConversation = jest.fn().mockResolvedValue();
359
+ stateManager.updateFlowRun = jest.fn().mockResolvedValue();
360
+ stateManager.getFlowRun = jest.fn().mockResolvedValue({ id: 'run-1', nodeStates: {} });
361
+
362
+ messageProcessor.processMessage = jest.fn().mockImplementation(async (id) => {
363
+ setImmediate(() => fe.notifyAgentCompletion(id, {
364
+ summary: 'wrote a draft',
365
+ success: true,
366
+ outputs: { draft: 'long text', wordCount: 850 },
367
+ }));
368
+ });
369
+
370
+ const result = await fe.executeAgentNode(node, context, 'run-1', null, flow);
371
+ expect(result.outputs).toEqual({ draft: 'long text', wordCount: 850 });
372
+ expect(result.type).toBe('agent');
373
+ });
374
+
375
+ test('executeAgentNode (v2) throws when agent omits a required declared output', async () => {
376
+ // Contract says { draft, wordCount } but agent only provides draft.
377
+ // The executor refuses to commit a half-baked handoff.
378
+ const node = {
379
+ id: 'w', type: 'agent', data: { agentId: 'writer', promptTemplate: '{{topic}}' },
380
+ inputs: [{ name: 'topic', type: 'text', required: true }],
381
+ outputs: [{ name: 'draft', type: 'text' }, { name: 'wordCount', type: 'number' }],
382
+ };
383
+ const flow = {
384
+ id: 'f', name: 'Missing output', schemaVersion: 2,
385
+ nodes: [
386
+ { id: 'in', type: 'input', data: {}, inputs: [], outputs: [{ name: 'topic', type: 'text' }] },
387
+ node,
388
+ ],
389
+ edges: [{ source: 'in', sourceField: 'topic', target: 'w', targetField: 'topic' }],
390
+ };
391
+ const context = {
392
+ input: '',
393
+ nodeOutputs: { in: { type: 'input', outputs: { topic: 'x' } } },
394
+ variables: {}, sortedNodes: flow.nodes, flow,
395
+ };
396
+ agentPool.getAgent = jest.fn().mockResolvedValue({ id: 'writer', mode: 'chat' });
397
+ agentPool.persistAgentState = jest.fn().mockResolvedValue();
398
+ agentPool.clearConversation = jest.fn().mockResolvedValue();
399
+ stateManager.updateFlowRun = jest.fn().mockResolvedValue();
400
+ stateManager.getFlowRun = jest.fn().mockResolvedValue({ id: 'run-1', nodeStates: {} });
401
+
402
+ messageProcessor.processMessage = jest.fn().mockImplementation(async (id) => {
403
+ setImmediate(() => fe.notifyAgentCompletion(id, {
404
+ summary: 'wrote a draft', success: true,
405
+ outputs: { draft: 'just text' }, // wordCount missing
406
+ }));
407
+ });
408
+
409
+ await expect(fe.executeAgentNode(node, context, 'run-1', null, flow))
410
+ .rejects.toThrow(/wordCount/);
411
+ });
412
+
413
+ test('executeAgentNode (v2) throws when agent provides no outputs at all on a typed node', async () => {
414
+ const node = {
415
+ id: 'w', type: 'agent', data: { agentId: 'writer', promptTemplate: '{{topic}}' },
416
+ inputs: [{ name: 'topic', type: 'text', required: true }],
417
+ outputs: [{ name: 'draft', type: 'text' }],
418
+ };
419
+ const flow = {
420
+ id: 'f', name: 'No outputs', schemaVersion: 2,
421
+ nodes: [
422
+ { id: 'in', type: 'input', data: {}, inputs: [], outputs: [{ name: 'topic', type: 'text' }] },
423
+ node,
424
+ ],
425
+ edges: [{ source: 'in', sourceField: 'topic', target: 'w', targetField: 'topic' }],
426
+ };
427
+ const context = {
428
+ input: '',
429
+ nodeOutputs: { in: { type: 'input', outputs: { topic: 'x' } } },
430
+ variables: {}, sortedNodes: flow.nodes, flow,
431
+ };
432
+ agentPool.getAgent = jest.fn().mockResolvedValue({ id: 'writer', mode: 'chat' });
433
+ agentPool.persistAgentState = jest.fn().mockResolvedValue();
434
+ agentPool.clearConversation = jest.fn().mockResolvedValue();
435
+ stateManager.updateFlowRun = jest.fn().mockResolvedValue();
436
+ stateManager.getFlowRun = jest.fn().mockResolvedValue({ id: 'run-1', nodeStates: {} });
437
+
438
+ messageProcessor.processMessage = jest.fn().mockImplementation(async (id) => {
439
+ // No outputs field at all
440
+ setImmediate(() => fe.notifyAgentCompletion(id, { summary: 'done', success: true }));
441
+ });
442
+
443
+ await expect(fe.executeAgentNode(node, context, 'run-1', null, flow))
444
+ .rejects.toThrow(/draft/);
445
+ });
446
+
447
+ // -- Phase 8: getActiveContract for tool-level validation --
448
+
449
+ describe('getActiveContract', () => {
450
+ test('returns null when no contract registered for agent', () => {
451
+ expect(fe.getActiveContract('agent-x')).toBe(null);
452
+ });
453
+
454
+ test('returns the registered contract for an agent', () => {
455
+ const contract = { outputs: [{ name: 'draft', type: 'text' }] };
456
+ fe.activeContracts.set('agent-x', contract);
457
+ expect(fe.getActiveContract('agent-x')).toBe(contract);
458
+ });
459
+
460
+ test('contract is registered when executeAgentNode begins, cleared when it ends', async () => {
461
+ // The full executeAgentNode cycle is heavy; this is a focused
462
+ // unit test for the registration/cleanup contract. We check
463
+ // that the contract is registered by attemptOnce and that
464
+ // _activeContracts ends empty after a clean run.
465
+ const node = {
466
+ id: 'w', type: 'agent',
467
+ data: { agentId: 'writer', promptTemplate: '{{input}}' },
468
+ inputs: [{ name: 'input', type: 'text', required: true }],
469
+ outputs: [{ name: 'draft', type: 'text' }],
470
+ };
471
+ const flow = {
472
+ id: 'f', name: 'contract-lifecycle', schemaVersion: 2,
473
+ nodes: [
474
+ { id: 'in', type: 'input', data: {}, inputs: [], outputs: [{ name: 'topic', type: 'text' }] },
475
+ node,
476
+ ],
477
+ edges: [{ source: 'in', sourceField: 'topic', target: 'w', targetField: 'input' }],
478
+ };
479
+ const context = {
480
+ input: 'hi', nodeOutputs: { in: { type: 'input', outputs: { topic: 'hi' } } },
481
+ variables: {}, sortedNodes: flow.nodes, flow,
482
+ };
483
+ agentPool.getAgent = jest.fn().mockResolvedValue({ id: 'writer', mode: 'chat' });
484
+ agentPool.persistAgentState = jest.fn().mockResolvedValue();
485
+ agentPool.clearConversation = jest.fn().mockResolvedValue();
486
+ stateManager.updateFlowRun = jest.fn().mockResolvedValue();
487
+ stateManager.getFlowRun = jest.fn().mockResolvedValue({ id: 'r', nodeStates: {} });
488
+
489
+ messageProcessor.processMessage = jest.fn().mockImplementation(async (id) => {
490
+ // While the call is in-flight, the contract should be registered
491
+ expect(fe.getActiveContract('writer')).toBeTruthy();
492
+ expect(fe.getActiveContract('writer').outputs).toHaveLength(1);
493
+ setImmediate(() => fe.notifyAgentCompletion(id, {
494
+ summary: 'done', success: true, outputs: { draft: 'D' },
495
+ }));
496
+ });
497
+
498
+ await fe.executeAgentNode(node, context, 'r', null, flow);
499
+ // After completion, contract is cleared
500
+ expect(fe.getActiveContract('writer')).toBe(null);
501
+ });
502
+ });
503
+
504
+ // -- ensureAgentsLoaded: broadcast safety --
505
+
506
+ test('ensureAgentsLoaded survives a webSocketManager that lacks broadcast()', async () => {
507
+ // Regression: when wired to the WebServer instance (which only has
508
+ // broadcastToSession), the executor used to crash with
509
+ // "this.webSocketManager.broadcast is not a function" mid-load,
510
+ // killing the run.
511
+ const wsm = { broadcastToSession: jest.fn() }; // no .broadcast
512
+ fe.setWebSocketManager(wsm);
513
+ agentPool.getAgent = jest.fn().mockResolvedValue(null); // not loaded
514
+ stateManager.importArchivedAgent = jest.fn().mockResolvedValue({
515
+ id: 'a1', name: 'A', status: 'active', currentModel: 'm', capabilities: [],
516
+ });
517
+
518
+ await expect(fe.ensureAgentsLoaded([
519
+ { id: 'n', type: 'agent', data: { agentId: 'a1' } },
520
+ ])).resolves.not.toThrow();
521
+
522
+ // Falls back to broadcastToSession(null, msg)
523
+ expect(wsm.broadcastToSession).toHaveBeenCalledWith(null, expect.objectContaining({
524
+ type: 'agent-loaded',
525
+ }));
526
+ });
527
+
528
+ test('ensureAgentsLoaded survives a webSocketManager whose broadcast throws', async () => {
529
+ const wsm = { broadcast: jest.fn().mockImplementation(() => { throw new Error('boom'); }) };
530
+ fe.setWebSocketManager(wsm);
531
+ agentPool.getAgent = jest.fn().mockResolvedValue(null);
532
+ stateManager.importArchivedAgent = jest.fn().mockResolvedValue({ id: 'a2', name: 'B' });
533
+
534
+ // Must not throw — broadcast errors are UX-only
535
+ await expect(fe.ensureAgentsLoaded([
536
+ { id: 'n', type: 'agent', data: { agentId: 'a2' } },
537
+ ])).resolves.not.toThrow();
538
+ });
539
+
540
+ // -- Handoff fixes (data forwarding + re-prompt) --
541
+
542
+ describe('buildPreviousAgentData (handoff data forwarding)', () => {
543
+ test('forwards structured outputs bag from upstream agent', () => {
544
+ const flow = { id: 'f', edges: [{ source: 'a', target: 'b' }] };
545
+ const ctx = {
546
+ nodeOutputs: {
547
+ a: { type: 'agent', agentId: 'a-id', agentName: 'A', output: 'summary',
548
+ outputs: { draft: 'long text', wordCount: 850 } },
549
+ },
550
+ };
551
+ const r = fe.buildPreviousAgentData({ id: 'b' }, ctx, flow);
552
+ expect(r.outputs).toEqual({ draft: 'long text', wordCount: 850 });
553
+ expect(r.summary).toBe('summary');
554
+ expect(r.agentName).toBe('A');
555
+ });
556
+
557
+ test('returns null when no upstream agent contributors', () => {
558
+ const flow = { id: 'f', edges: [{ source: 'in', target: 'b' }] };
559
+ const ctx = { nodeOutputs: { in: { type: 'input', output: 'hi' } } };
560
+ expect(fe.buildPreviousAgentData({ id: 'b' }, ctx, flow)).toBe(null);
561
+ });
562
+
563
+ test('omits outputs field when upstream had no structured bag', () => {
564
+ const flow = { id: 'f', edges: [{ source: 'a', target: 'b' }] };
565
+ const ctx = {
566
+ nodeOutputs: {
567
+ a: { type: 'agent', agentId: 'a', output: 'just text', outputs: undefined },
568
+ },
569
+ };
570
+ const r = fe.buildPreviousAgentData({ id: 'b' }, ctx, flow);
571
+ expect(r.outputs).toBeUndefined();
572
+ });
573
+
574
+ test('merges outputs from multiple upstream agents (fan-in)', () => {
575
+ const flow = { id: 'f', edges: [
576
+ { source: 'a', target: 'c' },
577
+ { source: 'b', target: 'c' },
578
+ ]};
579
+ const ctx = {
580
+ nodeOutputs: {
581
+ a: { type: 'agent', agentId: 'a', output: 'A done', outputs: { findings: 'X' } },
582
+ b: { type: 'agent', agentId: 'b', output: 'B done', outputs: { trends: 'Y' } },
583
+ },
584
+ };
585
+ const r = fe.buildPreviousAgentData({ id: 'c' }, ctx, flow);
586
+ expect(r.outputs).toEqual({ findings: 'X', trends: 'Y' });
587
+ expect(Array.isArray(r.contributors)).toBe(true);
588
+ expect(r.contributors).toHaveLength(2);
589
+ });
590
+
591
+ test('later contributor wins on field-name collision', () => {
592
+ const flow = { id: 'f', edges: [
593
+ { source: 'a', target: 'c' },
594
+ { source: 'b', target: 'c' },
595
+ ]};
596
+ const ctx = {
597
+ nodeOutputs: {
598
+ a: { type: 'agent', agentId: 'a', output: '', outputs: { result: 'first' } },
599
+ b: { type: 'agent', agentId: 'b', output: '', outputs: { result: 'second' } },
600
+ },
601
+ };
602
+ const r = fe.buildPreviousAgentData({ id: 'c' }, ctx, flow);
603
+ expect(r.outputs.result).toBe('second');
604
+ });
605
+ });
606
+
607
+ test('executeAgentNode (v1) prefers full assistant response when summary is too brief', async () => {
608
+ // Bug class we're fixing: agent calls jobdone with {summary: "Done."}
609
+ // and skips details. Old behavior used "Done." as the next agent's
610
+ // entire context. Now we fall back to lastAssistantMessage so the
611
+ // next agent has real content to read.
612
+ const node = { id: 'w', type: 'agent', data: { agentId: 'writer', promptTemplate: '{{input}}' } };
613
+ const flow = {
614
+ id: 'f', name: 'fallback test',
615
+ nodes: [{ id: 'in', type: 'input', data: {} }, node],
616
+ edges: [{ source: 'in', target: 'w' }],
617
+ };
618
+ const context = {
619
+ input: 'hi',
620
+ nodeOutputs: { in: { type: 'input', output: 'hi' } },
621
+ variables: {}, sortedNodes: flow.nodes, flow,
622
+ };
623
+ const fullAssistantMessage = 'Here is the FULL output the agent actually produced — paragraphs of useful work that we want passed downstream.';
624
+ agentPool.getAgent = jest.fn().mockResolvedValue({
625
+ id: 'writer', mode: 'chat',
626
+ conversations: { full: { messages: [
627
+ { role: 'user', content: 'hi' },
628
+ { role: 'assistant', content: fullAssistantMessage },
629
+ ]}},
630
+ });
631
+ agentPool.persistAgentState = jest.fn().mockResolvedValue();
632
+ agentPool.clearConversation = jest.fn().mockResolvedValue();
633
+ stateManager.updateFlowRun = jest.fn().mockResolvedValue();
634
+ stateManager.getFlowRun = jest.fn().mockResolvedValue({ id: 'run-1', nodeStates: {} });
635
+
636
+ messageProcessor.processMessage = jest.fn().mockImplementation(async (id) => {
637
+ // Brief summary, no details — the failure mode the user reported
638
+ setImmediate(() => fe.notifyAgentCompletion(id, { summary: 'Done.', success: true }));
639
+ });
640
+
641
+ const result = await fe.executeAgentNode(node, context, 'run-1', null, flow);
642
+ expect(result.output).toBe(fullAssistantMessage);
643
+ });
644
+
645
+ test('executeAgentNode glues summary + details when both present', async () => {
646
+ const node = { id: 'w', type: 'agent', data: { agentId: 'writer', promptTemplate: '{{input}}' } };
647
+ const flow = {
648
+ id: 'f', name: 'glue test',
649
+ nodes: [{ id: 'in', type: 'input', data: {} }, node],
650
+ edges: [{ source: 'in', target: 'w' }],
651
+ };
652
+ const context = {
653
+ input: 'hi',
654
+ nodeOutputs: { in: { type: 'input', output: 'hi' } },
655
+ variables: {}, sortedNodes: flow.nodes, flow,
656
+ };
657
+ agentPool.getAgent = jest.fn().mockResolvedValue({ id: 'writer', mode: 'chat' });
658
+ agentPool.persistAgentState = jest.fn().mockResolvedValue();
659
+ agentPool.clearConversation = jest.fn().mockResolvedValue();
660
+ stateManager.updateFlowRun = jest.fn().mockResolvedValue();
661
+ stateManager.getFlowRun = jest.fn().mockResolvedValue({ id: 'run-1', nodeStates: {} });
662
+
663
+ messageProcessor.processMessage = jest.fn().mockImplementation(async (id) => {
664
+ setImmediate(() => fe.notifyAgentCompletion(id, {
665
+ summary: 'A complete summary that is sufficiently descriptive',
666
+ details: 'Plus extra details about the work',
667
+ success: true,
668
+ }));
669
+ });
670
+ const r = await fe.executeAgentNode(node, context, 'run-1', null, flow);
671
+ expect(r.output).toContain('complete summary');
672
+ expect(r.output).toContain('extra details');
673
+ });
674
+
675
+ test('executeAgentNode (v2) re-prompts in-conversation when outputs missing, then succeeds', async () => {
676
+ // The agent's first job-done is missing wordCount. The executor
677
+ // must send a corrective message in the SAME conversation (no
678
+ // clearConversation between attempts) and accept the fixed reply.
679
+ const node = {
680
+ id: 'w', type: 'agent', data: { agentId: 'writer', promptTemplate: '{{input}}' },
681
+ inputs: [{ name: 'input', type: 'text', required: true }],
682
+ outputs: [
683
+ { name: 'draft', type: 'text' },
684
+ { name: 'wordCount', type: 'number' },
685
+ ],
686
+ };
687
+ const flow = {
688
+ id: 'f', name: 'reprompt test', schemaVersion: 2,
689
+ nodes: [
690
+ { id: 'in', type: 'input', data: {}, inputs: [], outputs: [{ name: 'topic', type: 'text' }] },
691
+ node,
692
+ ],
693
+ edges: [{ source: 'in', sourceField: 'topic', target: 'w', targetField: 'input' }],
694
+ };
695
+ const context = {
696
+ input: 'AI',
697
+ nodeOutputs: { in: { type: 'input', outputs: { topic: 'AI' } } },
698
+ variables: {}, sortedNodes: flow.nodes, flow,
699
+ };
700
+ agentPool.getAgent = jest.fn().mockResolvedValue({ id: 'writer', mode: 'chat' });
701
+ agentPool.persistAgentState = jest.fn().mockResolvedValue();
702
+ agentPool.clearConversation = jest.fn().mockResolvedValue();
703
+ stateManager.updateFlowRun = jest.fn().mockResolvedValue();
704
+ stateManager.getFlowRun = jest.fn().mockResolvedValue({ id: 'run-1', nodeStates: {} });
705
+
706
+ let call = 0;
707
+ messageProcessor.processMessage = jest.fn().mockImplementation(async (id, _prompt, opts) => {
708
+ call++;
709
+ // First call: incomplete outputs (missing wordCount).
710
+ // Second call (re-prompt): complete outputs.
711
+ if (call === 1) {
712
+ setImmediate(() => fe.notifyAgentCompletion(id, {
713
+ summary: 'wrote draft', success: true,
714
+ outputs: { draft: 'long text' }, // wordCount missing
715
+ }));
716
+ } else {
717
+ // Re-prompt should NOT clear conversation between attempts —
718
+ // verify by ensuring clearConversation count stays at 1.
719
+ setImmediate(() => fe.notifyAgentCompletion(id, {
720
+ summary: 'fixed', success: true,
721
+ outputs: { draft: 'long text', wordCount: 850 },
722
+ }));
723
+ // Sanity: opts.isReprompt should be true on the corrective msg
724
+ expect(opts?.isReprompt).toBe(true);
725
+ }
726
+ });
727
+
728
+ const r = await fe.executeAgentNode(node, context, 'run-1', null, flow);
729
+ expect(r.outputs).toEqual({ draft: 'long text', wordCount: 850 });
730
+ // 1 initial + 1 reprompt = 2 messages
731
+ expect(messageProcessor.processMessage).toHaveBeenCalledTimes(2);
732
+ // Conversation cleared once (start of attempt 0), NOT between
733
+ // re-prompts within an attempt.
734
+ expect(agentPool.clearConversation).toHaveBeenCalledTimes(1);
735
+ });
736
+
737
+ test('executeAgentNode (v2) escalates to outer retry after re-prompts exhausted', async () => {
738
+ // If the agent stubbornly emits incomplete outputs even after
739
+ // re-prompts, attemptOnce throws an agent-error which runWithRetry
740
+ // catches → fresh agent invocation (clearConversation).
741
+ const node = {
742
+ id: 'w', type: 'agent',
743
+ data: { agentId: 'writer', promptTemplate: '{{input}}' },
744
+ execution: { timeoutMs: 1000, maxRetries: 1, backoffBaseMs: 1, backoffMultiplier: 1 },
745
+ inputs: [{ name: 'input', type: 'text', required: true }],
746
+ outputs: [{ name: 'draft', type: 'text' }],
747
+ };
748
+ const flow = {
749
+ id: 'f', name: 'reprompt escalation', schemaVersion: 2,
750
+ nodes: [
751
+ { id: 'in', type: 'input', data: {}, inputs: [], outputs: [{ name: 'topic', type: 'text' }] },
752
+ node,
753
+ ],
754
+ edges: [{ source: 'in', sourceField: 'topic', target: 'w', targetField: 'input' }],
755
+ };
756
+ const context = {
757
+ input: 'X',
758
+ nodeOutputs: { in: { type: 'input', outputs: { topic: 'X' } } },
759
+ variables: {}, sortedNodes: flow.nodes, flow,
760
+ };
761
+ agentPool.getAgent = jest.fn().mockResolvedValue({ id: 'writer', mode: 'chat' });
762
+ agentPool.persistAgentState = jest.fn().mockResolvedValue();
763
+ agentPool.clearConversation = jest.fn().mockResolvedValue();
764
+ stateManager.updateFlowRun = jest.fn().mockResolvedValue();
765
+ stateManager.getFlowRun = jest.fn().mockResolvedValue({ id: 'run-1', nodeStates: {} });
766
+
767
+ // Always emits empty outputs — never satisfies contract
768
+ messageProcessor.processMessage = jest.fn().mockImplementation(async (id) => {
769
+ setImmediate(() => fe.notifyAgentCompletion(id, {
770
+ summary: 'still incomplete', success: true, outputs: {},
771
+ }));
772
+ });
773
+
774
+ await expect(fe.executeAgentNode(node, context, 'run-1', null, flow)).rejects.toThrow(/draft/);
775
+ // Outer retries = 1 → 2 attempts × (1 initial + 2 reprompts) = 6 calls
776
+ expect(messageProcessor.processMessage).toHaveBeenCalledTimes(6);
777
+ // Conversation cleared once per outer attempt (2 total)
778
+ expect(agentPool.clearConversation).toHaveBeenCalledTimes(2);
779
+ });
780
+
781
+ // -- Phase 6.3: flow version stamp on runs --
782
+
783
+ test('executeFlow stamps flow.version onto the run record', async () => {
784
+ const flow = {
785
+ id: 'fv', name: 'versioned',
786
+ version: 7,
787
+ nodes: [{ id: 'in', type: 'input', data: {} }, { id: 'out', type: 'output', data: {} }],
788
+ edges: [{ source: 'in', target: 'out' }],
789
+ };
790
+ stateManager.getFlow = jest.fn().mockResolvedValue(flow);
791
+ stateManager.createFlowRun = jest.fn().mockResolvedValue({ id: 'run-v' });
792
+ stateManager.getFlowRun = jest.fn().mockResolvedValue({ id: 'run-v', nodeStates: {} });
793
+ const updates = [];
794
+ stateManager.updateFlowRun = jest.fn().mockImplementation(async (_, patch) => { updates.push(patch); });
795
+
796
+ await fe.executeFlow('fv', { userInput: 'hi' });
797
+ // First updateFlowRun call after creation should stamp flowVersion
798
+ const stampUpdate = updates.find(u => u.flowVersion !== undefined);
799
+ expect(stampUpdate).toBeDefined();
800
+ expect(stampUpdate.flowVersion).toBe(7);
801
+ });
802
+
803
+ test('executeFlow stamps null flowVersion when flow has no version field', async () => {
804
+ const flow = {
805
+ id: 'fnv', name: 'unversioned',
806
+ nodes: [{ id: 'in', type: 'input', data: {} }, { id: 'out', type: 'output', data: {} }],
807
+ edges: [{ source: 'in', target: 'out' }],
808
+ };
809
+ stateManager.getFlow = jest.fn().mockResolvedValue(flow);
810
+ stateManager.createFlowRun = jest.fn().mockResolvedValue({ id: 'run-nv' });
811
+ stateManager.getFlowRun = jest.fn().mockResolvedValue({ id: 'run-nv', nodeStates: {} });
812
+ const updates = [];
813
+ stateManager.updateFlowRun = jest.fn().mockImplementation(async (_, patch) => { updates.push(patch); });
814
+
815
+ await fe.executeFlow('fnv', { userInput: 'hi' });
816
+ const stampUpdate = updates.find(u => u.flowVersion !== undefined);
817
+ expect(stampUpdate).toBeDefined();
818
+ expect(stampUpdate.flowVersion).toBeNull();
819
+ });
820
+
821
+ // -- Phase 6.1: per-node persisted errors --
822
+
823
+ test('updateNodeState stores errorInfo when provided', async () => {
824
+ const captured = [];
825
+ stateManager.getFlowRun = jest.fn().mockResolvedValue({ id: 'r1', nodeStates: {} });
826
+ stateManager.updateFlowRun = jest.fn().mockImplementation(async (runId, patch) => {
827
+ captured.push(patch);
828
+ });
829
+ const errorInfo = { kind: 'timeout', message: 'too slow', attempts: [{ attempt: 0, error: { kind: 'timeout', message: 't' } }], lastAt: '2026-01-01T00:00:00Z' };
830
+ await fe.updateNodeState('r1', 'n1', 'failed', { error: 'too slow' }, errorInfo);
831
+ const last = captured.at(-1);
832
+ expect(last.nodeStates.n1.status).toBe('failed');
833
+ expect(last.nodeStates.n1.error).toEqual(errorInfo);
834
+ });
835
+
836
+ test('updateNodeState omits error field when no errorInfo', async () => {
837
+ stateManager.getFlowRun = jest.fn().mockResolvedValue({ id: 'r1', nodeStates: {} });
838
+ let captured;
839
+ stateManager.updateFlowRun = jest.fn().mockImplementation(async (_, patch) => { captured = patch; });
840
+ await fe.updateNodeState('r1', 'n1', 'completed', { type: 'agent', output: 'ok' });
841
+ expect(captured.nodeStates.n1.error).toBeUndefined();
842
+ });
843
+
844
+ test('failed agent node persists kind=timeout + attempt history into nodeStates', async () => {
845
+ // Per-node timeout very small + maxRetries=1 → both attempts hang
846
+ // → executor throws with kind:'timeout' + attempts[]. The catch in
847
+ // executeNode must capture both into nodeStates[id].error.
848
+ const node = {
849
+ id: 'w', type: 'agent',
850
+ data: { agentId: 'writer', promptTemplate: '{{input}}' },
851
+ execution: { timeoutMs: 20, maxRetries: 1, backoffBaseMs: 1, backoffMultiplier: 1 },
852
+ };
853
+ const flow = {
854
+ id: 'f', name: 'fail flow',
855
+ nodes: [{ id: 'in', type: 'input', data: {} }, node],
856
+ edges: [{ source: 'in', target: 'w' }],
857
+ };
858
+ const context = {
859
+ input: 'x',
860
+ nodeOutputs: { in: { type: 'input', output: 'x' } },
861
+ variables: {}, sortedNodes: flow.nodes, flow,
862
+ };
863
+ agentPool.getAgent = jest.fn().mockResolvedValue({ id: 'writer', mode: 'chat' });
864
+ agentPool.persistAgentState = jest.fn().mockResolvedValue();
865
+ agentPool.clearConversation = jest.fn().mockResolvedValue();
866
+ messageProcessor.processMessage = jest.fn().mockResolvedValue(); // never notify
867
+
868
+ const captured = [];
869
+ stateManager.getFlowRun = jest.fn().mockResolvedValue({ id: 'run-fail', nodeStates: {} });
870
+ stateManager.updateFlowRun = jest.fn().mockImplementation(async (_, patch) => { captured.push(patch); });
871
+
872
+ await expect(fe.executeNode(node, context, 'run-fail', null, flow)).rejects.toMatchObject({ kind: 'timeout' });
873
+
874
+ // Find the failed update — node state must include error + attempts
875
+ const failedUpdate = captured.find(p => p.nodeStates?.w?.status === 'failed');
876
+ expect(failedUpdate).toBeDefined();
877
+ expect(failedUpdate.nodeStates.w.error.kind).toBe('timeout');
878
+ expect(failedUpdate.nodeStates.w.error.attempts).toBeDefined();
879
+ expect(failedUpdate.nodeStates.w.error.lastAt).toMatch(/^\d{4}-\d{2}-\d{2}T/);
880
+ });
881
+
882
+ // -- Phase 4: disk checkpoint + resume --
883
+
884
+ test('checkpointStore receives node results when set, none when null', async () => {
885
+ const flow = {
886
+ id: 'f', name: 'cp test',
887
+ nodes: [{ id: 'in', type: 'input', data: {} }, { id: 'out', type: 'output', data: {} }],
888
+ edges: [{ source: 'in', target: 'out' }],
889
+ variables: {},
890
+ };
891
+ stateManager.getFlow = jest.fn().mockResolvedValue(flow);
892
+ stateManager.createFlowRun = jest.fn().mockResolvedValue({ id: 'run-cp' });
893
+ stateManager.updateFlowRun = jest.fn().mockResolvedValue();
894
+ stateManager.getFlowRun = jest.fn().mockResolvedValue({ id: 'run-cp', nodeStates: {} });
895
+
896
+ const saveNodeResult = jest.fn().mockResolvedValue();
897
+ fe.setCheckpointStore({ saveNodeResult });
898
+ await fe.executeFlow('f', { userInput: 'hello' });
899
+ // 2 nodes → 2 saves
900
+ expect(saveNodeResult).toHaveBeenCalledTimes(2);
901
+ expect(saveNodeResult).toHaveBeenCalledWith('run-cp', 'in', expect.any(Object));
902
+ expect(saveNodeResult).toHaveBeenCalledWith('run-cp', 'out', expect.any(Object));
903
+ });
904
+
905
+ test('resumeFlow throws when no checkpoint store is configured', async () => {
906
+ fe.setCheckpointStore(null);
907
+ await expect(fe.resumeFlow('run-x')).rejects.toThrow(/checkpoint store/i);
908
+ });
909
+
910
+ test('resumeFlow short-circuits when run is already completed', async () => {
911
+ fe.setCheckpointStore({
912
+ loadAllNodeResults: jest.fn(),
913
+ saveNodeResult: jest.fn(),
914
+ });
915
+ stateManager.getFlowRun = jest.fn().mockResolvedValue({
916
+ id: 'run-done', flowId: 'f', status: 'completed', output: { final: 'yes' },
917
+ });
918
+ const r = await fe.resumeFlow('run-done');
919
+ expect(r.status).toBe('completed');
920
+ expect(r.output).toEqual({ final: 'yes' });
921
+ });
922
+
923
+ test('resumeFlow skips already-completed nodes and runs only the rest', async () => {
924
+ // 3-node flow: in → mid → out. Pretend "in" already completed.
925
+ const flow = {
926
+ id: 'f-resume', name: 'resume test',
927
+ nodes: [
928
+ { id: 'in', type: 'input', data: {} },
929
+ { id: 'mid', type: 'output', data: {} }, // use output node so we don't need agentPool wiring
930
+ { id: 'out', type: 'output', data: {} },
931
+ ],
932
+ edges: [
933
+ { source: 'in', target: 'mid' },
934
+ { source: 'mid', target: 'out' },
935
+ ],
936
+ variables: {},
937
+ };
938
+ stateManager.getFlow = jest.fn().mockResolvedValue(flow);
939
+ stateManager.getFlowRun = jest.fn().mockResolvedValue({
940
+ id: 'run-r', flowId: 'f-resume', status: 'failed',
941
+ initialInput: { userInput: 'hi' }, nodeStates: {},
942
+ });
943
+ stateManager.updateFlowRun = jest.fn().mockResolvedValue();
944
+
945
+ const persistedOutputs = {
946
+ in: { type: 'input', output: 'hi-from-checkpoint' },
947
+ };
948
+ const saveNodeResult = jest.fn().mockResolvedValue();
949
+ fe.setCheckpointStore({
950
+ loadAllNodeResults: jest.fn().mockResolvedValue(persistedOutputs),
951
+ saveNodeResult,
952
+ });
953
+
954
+ // Spy on executeNode so we can verify only mid+out are executed.
955
+ const calls = [];
956
+ const realExec = fe.executeNode.bind(fe);
957
+ fe.executeNode = jest.fn(async (node, context, runId, sessionId, f) => {
958
+ calls.push(node.id);
959
+ return realExec(node, context, runId, sessionId, f);
960
+ });
961
+
962
+ const r = await fe.resumeFlow('run-r');
963
+ expect(r.status).toBe('completed');
964
+ // "in" was checkpointed → executeNode never called for it
965
+ expect(calls).toEqual(['mid', 'out']);
966
+ // Two new checkpoints saved (mid, out)
967
+ expect(saveNodeResult).toHaveBeenCalledTimes(2);
968
+ });
969
+
970
+ // -- Phase 3: per-node retry + per-node timeout --
971
+
972
+ test('_resolveExecutionConfig honors node > flow > config > defaults precedence', () => {
973
+ config.flows = { execution: { timeoutMs: 1000, maxRetries: 5 } };
974
+ const flow = { execution: { timeoutMs: 2000 } };
975
+ const node = { execution: { timeoutMs: 3000 } };
976
+ const r = fe._resolveExecutionConfig(node, flow);
977
+ expect(r.timeoutMs).toBe(3000); // node wins
978
+ expect(r.maxRetries).toBe(5); // falls through to global
979
+ expect(r.retryOn).toEqual(['timeout', 'agent-error']); // default
980
+ });
981
+
982
+ test('_resolveExecutionConfig honors legacy config.flows.nodeTimeout when no execution config given', () => {
983
+ config.flows = { nodeTimeout: 7777 };
984
+ const r = fe._resolveExecutionConfig({}, {});
985
+ expect(r.timeoutMs).toBe(7777);
986
+ });
987
+
988
+ test('_resolveExecutionConfig defaults: maxRetries=1, timeoutMs=300000', () => {
989
+ // Phase 7 hardening: bumped default maxRetries 0→1 so a single
990
+ // structured-output miss doesn't fail the whole flow when re-prompts
991
+ // didn't recover. Override per-node to 0 to disable.
992
+ delete config.flows;
993
+ const r = fe._resolveExecutionConfig({}, {});
994
+ expect(r.maxRetries).toBe(1);
995
+ expect(r.timeoutMs).toBe(300000);
996
+ });
997
+
998
+ test('executeAgentNode retries on timeout and eventually succeeds', async () => {
999
+ // Tiny per-node timeout + 2 retries. First two processMessage calls
1000
+ // never resolve the awaiter (simulated hang); the third does.
1001
+ const node = {
1002
+ id: 'w', type: 'agent',
1003
+ data: { agentId: 'writer', promptTemplate: '{{input}}' },
1004
+ execution: { timeoutMs: 50, maxRetries: 2, backoffBaseMs: 1, backoffMultiplier: 1 },
1005
+ };
1006
+ const flow = {
1007
+ id: 'f', name: 'Retry on timeout',
1008
+ nodes: [{ id: 'in', type: 'input', data: {} }, node],
1009
+ edges: [{ source: 'in', target: 'w' }],
1010
+ };
1011
+ const context = {
1012
+ input: 'hi',
1013
+ nodeOutputs: { in: { type: 'input', output: 'hi' } },
1014
+ variables: {}, sortedNodes: flow.nodes, flow,
1015
+ };
1016
+ agentPool.getAgent = jest.fn().mockResolvedValue({ id: 'writer', mode: 'chat' });
1017
+ agentPool.persistAgentState = jest.fn().mockResolvedValue();
1018
+ agentPool.clearConversation = jest.fn().mockResolvedValue();
1019
+ stateManager.updateFlowRun = jest.fn().mockResolvedValue();
1020
+ stateManager.getFlowRun = jest.fn().mockResolvedValue({ id: 'run-1', nodeStates: {} });
1021
+
1022
+ let calls = 0;
1023
+ messageProcessor.processMessage = jest.fn().mockImplementation(async (id) => {
1024
+ calls++;
1025
+ if (calls < 3) return; // hang the awaiter → timeout
1026
+ setImmediate(() => fe.notifyAgentCompletion(id, { summary: 'finally', success: true }));
1027
+ });
1028
+
1029
+ const result = await fe.executeAgentNode(node, context, 'run-1', null, flow);
1030
+ expect(result.output).toBe('finally');
1031
+ expect(result.attempts).toHaveLength(3);
1032
+ expect(result.attempts[0].error.kind).toBe('timeout');
1033
+ expect(result.attempts[1].error.kind).toBe('timeout');
1034
+ expect(result.attempts[2].error).toBeUndefined();
1035
+ });
1036
+
1037
+ test('executeAgentNode exhausts retries then throws with kind=timeout', async () => {
1038
+ const node = {
1039
+ id: 'w', type: 'agent',
1040
+ data: { agentId: 'writer', promptTemplate: '{{input}}' },
1041
+ execution: { timeoutMs: 30, maxRetries: 1, backoffBaseMs: 1, backoffMultiplier: 1 },
1042
+ };
1043
+ const flow = {
1044
+ id: 'f', name: 'Persistent timeout',
1045
+ nodes: [{ id: 'in', type: 'input', data: {} }, node],
1046
+ edges: [{ source: 'in', target: 'w' }],
1047
+ };
1048
+ const context = {
1049
+ input: 'hi',
1050
+ nodeOutputs: { in: { type: 'input', output: 'hi' } },
1051
+ variables: {}, sortedNodes: flow.nodes, flow,
1052
+ };
1053
+ agentPool.getAgent = jest.fn().mockResolvedValue({ id: 'writer', mode: 'chat' });
1054
+ agentPool.persistAgentState = jest.fn().mockResolvedValue();
1055
+ agentPool.clearConversation = jest.fn().mockResolvedValue();
1056
+ stateManager.updateFlowRun = jest.fn().mockResolvedValue();
1057
+ stateManager.getFlowRun = jest.fn().mockResolvedValue({ id: 'run-1', nodeStates: {} });
1058
+
1059
+ messageProcessor.processMessage = jest.fn().mockResolvedValue(); // never notify
1060
+
1061
+ await expect(fe.executeAgentNode(node, context, 'run-1', null, flow))
1062
+ .rejects.toMatchObject({ kind: 'timeout' });
1063
+ });
1064
+
1065
+ test('executeAgentNode (v1) ignores outputs validation entirely', async () => {
1066
+ // Legacy node has no inputs[]/outputs[] — the executor should NOT
1067
+ // try to validate a non-existent contract.
1068
+ const node = {
1069
+ id: 'w', type: 'agent',
1070
+ data: { agentId: 'writer', promptTemplate: '{{input}}' },
1071
+ };
1072
+ const flow = {
1073
+ id: 'f', name: 'Legacy',
1074
+ nodes: [{ id: 'in', type: 'input', data: {} }, node],
1075
+ edges: [{ source: 'in', target: 'w' }],
1076
+ };
1077
+ const context = {
1078
+ input: 'hello',
1079
+ nodeOutputs: { in: { type: 'input', output: 'hello' } },
1080
+ variables: {}, sortedNodes: flow.nodes, flow,
1081
+ };
1082
+ agentPool.getAgent = jest.fn().mockResolvedValue({ id: 'writer', mode: 'chat' });
1083
+ agentPool.persistAgentState = jest.fn().mockResolvedValue();
1084
+ agentPool.clearConversation = jest.fn().mockResolvedValue();
1085
+ stateManager.updateFlowRun = jest.fn().mockResolvedValue();
1086
+ stateManager.getFlowRun = jest.fn().mockResolvedValue({ id: 'run-1', nodeStates: {} });
1087
+
1088
+ messageProcessor.processMessage = jest.fn().mockImplementation(async (id) => {
1089
+ setImmediate(() => fe.notifyAgentCompletion(id, { summary: 'done', success: true }));
1090
+ });
1091
+ const result = await fe.executeAgentNode(node, context, 'run-1', null, flow);
1092
+ expect(result.type).toBe('agent');
1093
+ expect(result.output).toBe('done');
1094
+ });
1095
+
1096
+ test('executeAgentNode (v2) throws when a required input has no inbound edge', async () => {
1097
+ const node = {
1098
+ id: 'w', type: 'agent', data: { agentId: 'writer', promptTemplate: '{{topic}}' },
1099
+ inputs: [{ name: 'topic', type: 'text', required: true },
1100
+ { name: 'research', type: 'json', required: true }], // unbound
1101
+ outputs: [{ name: 'draft', type: 'text' }],
1102
+ };
1103
+ const flow = {
1104
+ id: 'f', name: 'Missing required',
1105
+ nodes: [
1106
+ { id: 'in', type: 'input', data: {}, inputs: [], outputs: [{ name: 'topic', type: 'text' }] },
1107
+ node,
1108
+ ],
1109
+ edges: [
1110
+ { source: 'in', sourceField: 'topic', target: 'w', targetField: 'topic' },
1111
+ // no edge feeds w.research
1112
+ ],
1113
+ };
1114
+ const context = {
1115
+ input: '',
1116
+ nodeOutputs: { in: { type: 'input', outputs: { topic: 'x' } } },
1117
+ variables: {},
1118
+ sortedNodes: flow.nodes,
1119
+ flow,
1120
+ };
1121
+ agentPool.getAgent = jest.fn().mockResolvedValue({ id: 'writer', mode: 'chat' });
1122
+ // Should throw before queueing a message.
1123
+ await expect(fe.executeAgentNode(node, context, 'run-1', null, flow)).rejects.toThrow(/research/);
1124
+ expect(messageProcessor.processMessage).not.toHaveBeenCalled();
1125
+ });
1126
+
220
1127
  // --- truncateOutput ---
221
1128
 
222
1129
  test('truncateOutput truncates long strings', () => {
@@ -237,6 +1144,49 @@ describe('FlowExecutor', () => {
237
1144
  expect(result.preview).toBeDefined();
238
1145
  });
239
1146
 
1147
+ // Regression lock: agent completion results with structured `outputs`
1148
+ // must preserve `outputs` verbatim. Truncating the whole result blob
1149
+ // (the pre-fix behavior) breaks downstream edge field-mapping —
1150
+ // `writer.bullets → critic.bullets` would deliver `null` because the
1151
+ // captured run dump only had {truncated: true, preview: "..."}
1152
+ // instead of the real list. See e2e dump from 2026-04-29: every
1153
+ // long-output node showed null handoffs even when the agent's
1154
+ // jobdone payload was structurally perfect.
1155
+ test('truncateOutput preserves structured outputs on agent completion shape', () => {
1156
+ const longDraft = 'climate change '.repeat(200); // ~3 KB
1157
+ const longPreview = 'preview text '.repeat(200); // ~2.5 KB
1158
+ const completion = {
1159
+ type: 'agent',
1160
+ agentId: 'agent-1',
1161
+ output: longPreview,
1162
+ summary: 'Wrote a paragraph',
1163
+ outputs: { draft: longDraft, wordCount: 145 },
1164
+ };
1165
+ const result = fe.truncateOutput(completion);
1166
+
1167
+ // outputs object preserved exactly — downstream edges depend on this
1168
+ expect(result.outputs).toBeDefined();
1169
+ expect(result.outputs.draft).toBe(longDraft);
1170
+ expect(result.outputs.wordCount).toBe(145);
1171
+
1172
+ // Long prose fields get the legacy string truncation
1173
+ expect(result.output.length).toBe(1000 + '... (truncated)'.length);
1174
+ expect(result.output.endsWith('... (truncated)')).toBe(true);
1175
+
1176
+ // Top-level metadata (not a string, not 'outputs') stays
1177
+ expect(result.type).toBe('agent');
1178
+ expect(result.agentId).toBe('agent-1');
1179
+ });
1180
+
1181
+ test('truncateOutput still falls back to {truncated, preview} for non-completion blobs', () => {
1182
+ // Random object without a structured `outputs` key behaves like
1183
+ // before — get the legacy preview-marker shape.
1184
+ const obj = { someBigField: 'x'.repeat(2000) };
1185
+ const result = fe.truncateOutput(obj);
1186
+ expect(result.truncated).toBe(true);
1187
+ expect(result.preview).toBeDefined();
1188
+ });
1189
+
240
1190
  // --- applyTemplate ---
241
1191
 
242
1192
  test('applyTemplate substitutes variables', () => {