openclaw-multi-auto 1.3.7 → 1.3.8

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 (503) hide show
  1. package/dist/{accounts-L9ByEpnP.js → accounts-C9HcPI9h.js} +2 -2
  2. package/dist/{accounts-BOzyfwW4.js → accounts-C_lW3Ag9.js} +2 -2
  3. package/dist/{accounts-yfBeCZtS.js → accounts-Tgelvk0C.js} +17 -17
  4. package/dist/{active-listener-D1yqT1cw.js → active-listener-BEdprTkn.js} +2 -2
  5. package/dist/{api-key-rotation-DtsNS2Nb.js → api-key-rotation-BJpKWXy0.js} +2 -2
  6. package/dist/{audio-preflight-DpxQCpsA.js → audio-preflight-BMvgEQ5j.js} +32 -32
  7. package/dist/{audio-preflight-5FEeDooz.js → audio-preflight-DDBLZBdb.js} +4 -4
  8. package/dist/{audio-transcription-runner-B-UvoDjZ.js → audio-transcription-runner-DZbSWT9E.js} +1 -1
  9. package/dist/{audio-transcription-runner-28fcRNNi.js → audio-transcription-runner-gLFfz8fr.js} +12 -12
  10. package/dist/{audit-membership-runtime-DWyHWAHM.js → audit-membership-runtime-Dntemq07.js} +4 -4
  11. package/dist/build-info.json +3 -3
  12. package/dist/bundled/boot-md/handler.js +51 -51
  13. package/dist/bundled/bootstrap-extra-files/handler.js +6 -6
  14. package/dist/bundled/command-logger/handler.js +2 -2
  15. package/dist/bundled/session-memory/handler.js +51 -51
  16. package/dist/canvas-host/a2ui/.bundle.hash +1 -1
  17. package/dist/{channel-activity-xHOMiarp.js → channel-activity-BDnjYF7B.js} +3 -3
  18. package/dist/{chrome-D45SyhQL.js → chrome-CMU2WVFh.js} +8 -8
  19. package/dist/{chrome-DwizpzOC.js → chrome-DxxEKrY7.js} +18 -18
  20. package/dist/{commands-registry-V1zZ5pPC.js → commands-registry-D5qXbFJn.js} +4 -4
  21. package/dist/{deliver-B9cys0EZ.js → deliver-BXVcFIHL.js} +1 -1
  22. package/dist/{deliver-D4o6VIur.js → deliver-DbdywYJE.js} +21 -21
  23. package/dist/deliver-runtime-BFs7iAZF.js +36 -0
  24. package/dist/{deliver-runtime-DhaQJ0pI.js → deliver-runtime-DTaIS-1i.js} +3 -3
  25. package/dist/deps-send-discord.runtime-DZUccI6Z.js +26 -0
  26. package/dist/deps-send-imessage.runtime-CF3OpoqY.js +25 -0
  27. package/dist/deps-send-signal.runtime-Cw4-ozeO.js +24 -0
  28. package/dist/deps-send-slack.runtime-BDsDhS1P.js +22 -0
  29. package/dist/deps-send-telegram.runtime-D_4xVasO.js +27 -0
  30. package/dist/{deps-send-whatsapp.runtime-DvTL2tzN.js → deps-send-whatsapp.runtime-CIZqFAqb.js} +7 -7
  31. package/dist/deps-send-whatsapp.runtime-DK8jqd14.js +60 -0
  32. package/dist/{diagnostic-Bn4PZjMZ.js → diagnostic-Co6Kghr-.js} +2 -2
  33. package/dist/{plugin-sdk/errors-CtMWwS2Z.js → errors-xt401nuk.js} +1 -1
  34. package/dist/extensionAPI.js +6 -6
  35. package/dist/{fetch-BlJWzEP6.js → fetch-DuraYswo.js} +5 -5
  36. package/dist/{fetch-guard-ChYBwfiy.js → fetch-guard-DWr0d00H.js} +2 -2
  37. package/dist/{frontmatter-CvaMP376.js → frontmatter-BkTfEZ93.js} +3 -3
  38. package/dist/{fs-safe-0jAo_Whb.js → fs-safe-CTYUrIgQ.js} +4 -4
  39. package/dist/{github-copilot-token-D13V9YBz.js → github-copilot-token-BDioPmd6.js} +7 -7
  40. package/dist/{image-DAOPwVXi.js → image-BCVLo0qw.js} +1 -1
  41. package/dist/{image-Bbn53mzj.js → image-eT7Y-nP5.js} +6 -6
  42. package/dist/{image-ops-CehkHxmW.js → image-ops-BuUnEOE0.js} +2 -2
  43. package/dist/image-runtime-BcAK3n8a.js +29 -0
  44. package/dist/{image-runtime-wlCLVvVv.js → image-runtime-DtCKpMPZ.js} +3 -3
  45. package/dist/{ir-DAP-B-Xw.js → ir-B83looB-.js} +8 -8
  46. package/dist/{legacy-names-TyzbVqa_.js → legacy-names-DOC03BkU.js} +1 -1
  47. package/dist/llm-slug-generator.js +51 -51
  48. package/dist/{logger-DMZQQtxK.js → logger-BfjWMCSD.js} +7 -7
  49. package/dist/{login-DiCctRo1.js → login-CrIwcrVI.js} +5 -5
  50. package/dist/{login-qr-MUbXgjtd.js → login-qr-BpPDZdl_.js} +10 -10
  51. package/dist/{manager-BW_NSIMl.js → manager-1bvuGrNR.js} +13 -13
  52. package/dist/manager-runtime-FO1Sx3W8.js +18 -0
  53. package/dist/{model-selection-idoqPmw0.js → model-selection-Dna0Gz1k.js} +43 -43
  54. package/dist/{outbound-C2kanETZ.js → outbound-ChDjtuD6.js} +6 -6
  55. package/dist/{outbound-attachment-DBrYWX8h.js → outbound-attachment-DqHlD21U.js} +2 -2
  56. package/dist/{path-alias-guards-DqXRZmsL.js → path-alias-guards-BzvdLvTI.js} +1 -1
  57. package/dist/{paths-CCxysrzL.js → paths-Bkr-BCxW.js} +4 -4
  58. package/dist/{paths-C6TxBCvO.js → paths-Cvc9EM8Y.js} +5 -5
  59. package/dist/{pi-embedded-BaGj07T0.js → pi-embedded-BQQa91aA.js} +158 -158
  60. package/dist/{pi-embedded-DYU79yGe.js → pi-embedded-CgQ_W6Xs.js} +24 -24
  61. package/dist/{pi-embedded-helpers-wy0DZvx1.js → pi-embedded-helpers-CLXm10bV.js} +52 -52
  62. package/dist/{pi-embedded-helpers-uTRAmQ4n.js → pi-embedded-helpers-CwuBTKza.js} +3 -3
  63. package/dist/{plugin-sdk/pi-model-discovery-v-XPUOOf.js → pi-model-discovery-Dymwdjt0.js} +2 -2
  64. package/dist/pi-model-discovery-runtime-BeY4EUPp.js +11 -0
  65. package/dist/{pi-tools.before-tool-call.runtime-BuLxSyx9.js → pi-tools.before-tool-call.runtime-Cwab_5W1.js} +9 -9
  66. package/dist/plugin-sdk/{accounts-DyFCXtHv.js → accounts-BslAlVYS.js} +2 -2
  67. package/dist/plugin-sdk/{accounts-BJAXxY46.js → accounts-C3m65--E.js} +2 -2
  68. package/dist/plugin-sdk/{accounts-C1j7HSL0.js → accounts-CNCCkdEF.js} +3 -3
  69. package/dist/plugin-sdk/{active-listener-B_sLJTXM.js → active-listener-CkPnMUkB.js} +2 -2
  70. package/dist/plugin-sdk/{api-key-rotation-8nyyt1kx.js → api-key-rotation-BXnNsojA.js} +2 -2
  71. package/dist/plugin-sdk/{audio-preflight-C_aSAPR1.js → audio-preflight-CtO4fFvp.js} +26 -26
  72. package/dist/plugin-sdk/{audio-transcription-runner-CB53F7_7.js → audio-transcription-runner-DnxvOS1-.js} +11 -11
  73. package/dist/plugin-sdk/{audit-membership-runtime-BXndI4LG.js → audit-membership-runtime-BpfoSk8M.js} +2 -2
  74. package/dist/plugin-sdk/{channel-activity-C5y8AgAV.js → channel-activity-WJYxcJ3S.js} +3 -3
  75. package/dist/plugin-sdk/{channel-web-DBTRO03V.js → channel-web-dO5k3ubM.js} +18 -18
  76. package/dist/plugin-sdk/{chrome-f00sZkDX.js → chrome-CjNTuJML.js} +6 -6
  77. package/dist/plugin-sdk/{commands-registry-BJ_NxG2F.js → commands-registry-CdYjoI0i.js} +4 -4
  78. package/dist/plugin-sdk/{common-Cf27Jwxu.js → common-oYc5vPFl.js} +2 -2
  79. package/dist/plugin-sdk/{config-CHQrpx-Q.js → config-B1z-UxQ3.js} +7 -7
  80. package/dist/plugin-sdk/{deliver-DNEuetST.js → deliver-D5_6T567.js} +10 -10
  81. package/dist/plugin-sdk/deliver-runtime-C5dgvvga.js +32 -0
  82. package/dist/plugin-sdk/deps-send-discord.runtime-Dg4N7PHJ.js +23 -0
  83. package/dist/plugin-sdk/deps-send-imessage.runtime-0OEwzMQm.js +22 -0
  84. package/dist/plugin-sdk/deps-send-signal.runtime-BM1jRt3G.js +21 -0
  85. package/dist/plugin-sdk/deps-send-slack.runtime-1E3BYRdF.js +19 -0
  86. package/dist/plugin-sdk/deps-send-telegram.runtime-DNCxIflA.js +24 -0
  87. package/dist/plugin-sdk/deps-send-whatsapp.runtime-OLwr-9c8.js +57 -0
  88. package/dist/plugin-sdk/{diagnostic-LYUUmjJ5.js → diagnostic-Bxxu0ig-.js} +2 -2
  89. package/dist/plugin-sdk/{errors-DaiAM-yU.js → errors-B3cHyZZA.js} +1 -1
  90. package/dist/plugin-sdk/{fetch-guard-CxYB5Kg6.js → fetch-guard-Dcgod0tg.js} +2 -2
  91. package/dist/plugin-sdk/{fs-safe-DtfhxbrI.js → fs-safe-BaKqI3G4.js} +3 -3
  92. package/dist/plugin-sdk/{image-BwjYjRHx.js → image-B2mQW9Rb.js} +6 -6
  93. package/dist/plugin-sdk/{image-ops-BnZKcbd6.js → image-ops-Cbzr4U9l.js} +2 -2
  94. package/dist/plugin-sdk/image-runtime-BFm45j49.js +25 -0
  95. package/dist/plugin-sdk/index.js +50 -50
  96. package/dist/plugin-sdk/{ir-Z4hX67TJ.js → ir-ZEmrTr4J.js} +7 -7
  97. package/dist/plugin-sdk/{local-roots-KhjQw04O.js → local-roots-CIPRxA-4.js} +4 -4
  98. package/dist/plugin-sdk/{logger-DHIIvMxj.js → logger-CvPFVOgT.js} +2 -2
  99. package/dist/plugin-sdk/{login-C31642Ld.js → login-CCTew9bt.js} +4 -4
  100. package/dist/plugin-sdk/{login-qr--y2SG_Ue.js → login-qr-BI3Vi_wJ.js} +5 -5
  101. package/dist/plugin-sdk/{manager-2UZBMCc7.js → manager-BEoYPn7R.js} +8 -8
  102. package/dist/plugin-sdk/manager-runtime-DxclHQ4U.js +15 -0
  103. package/dist/plugin-sdk/{outbound-Ba0QUI5h.js → outbound-ByOw1K6W.js} +5 -5
  104. package/dist/plugin-sdk/{outbound-attachment-B1Laso-8.js → outbound-attachment-BzVhxRRw.js} +2 -2
  105. package/dist/plugin-sdk/{path-alias-guards-C7Vm5DZ1.js → path-alias-guards-sWayacde.js} +1 -1
  106. package/dist/plugin-sdk/{paths-DopV9PQG.js → paths-Dpg3qxcl.js} +1 -1
  107. package/dist/plugin-sdk/{pi-embedded-helpers-DnA_OCzP.js → pi-embedded-helpers-DIxXkGJf.js} +16 -16
  108. package/dist/plugin-sdk/{pi-model-discovery-DdPqXk8f.js → pi-model-discovery-DM_2uFtj.js} +1 -1
  109. package/dist/plugin-sdk/pi-model-discovery-runtime-BuzvkvNR.js +8 -0
  110. package/dist/plugin-sdk/{pi-tools.before-tool-call.runtime-DxFHiLUE.js → pi-tools.before-tool-call.runtime-w1dqL_ty.js} +4 -4
  111. package/dist/plugin-sdk/{plugins-CbCt4osF.js → plugins-C4USiH29.js} +4 -4
  112. package/dist/plugin-sdk/{proxy-env-C63mMdas.js → proxy-env-ET-rp8eg.js} +1 -1
  113. package/dist/plugin-sdk/{proxy-fetch-ChxOhWF4.js → proxy-fetch-uDXGKG3Z.js} +1 -1
  114. package/dist/plugin-sdk/{pw-ai-DpJk62D4.js → pw-ai-CyOt3RDA.js} +9 -9
  115. package/dist/plugin-sdk/{qmd-manager-Ca-iSfEE.js → qmd-manager-BySdoVR7.js} +7 -7
  116. package/dist/plugin-sdk/{query-expansion-B_Xe41Ab.js → query-expansion-C6uS-7lj.js} +4 -4
  117. package/dist/plugin-sdk/{redact-DjVX-1N3.js → redact-Bvxt1T_Q.js} +1 -1
  118. package/dist/plugin-sdk/{reply-CovBlFea.js → reply-CTCSeQqW.js} +73 -73
  119. package/dist/plugin-sdk/{resolve-outbound-target-BbrHgyUk.js → resolve-outbound-target-Bw8YNANu.js} +2 -2
  120. package/dist/plugin-sdk/{run-with-concurrency-BR1DXa8T.js → run-with-concurrency-C_KCHwvf.js} +1 -1
  121. package/dist/plugin-sdk/runtime-whatsapp-login.runtime-BxgRDkhc.js +10 -0
  122. package/dist/plugin-sdk/runtime-whatsapp-outbound.runtime-elOqrkfg.js +19 -0
  123. package/dist/plugin-sdk/{send-BvAtLLPl.js → send-BZ6nYFZr.js} +5 -5
  124. package/dist/plugin-sdk/{send-BTztm3D2.js → send-C0w6xP2x.js} +6 -6
  125. package/dist/plugin-sdk/{send-CWJUuG0i.js → send-CFf-1V89.js} +8 -8
  126. package/dist/plugin-sdk/{send-EcglC4cG.js → send-CY-Qfwia.js} +7 -7
  127. package/dist/plugin-sdk/{send-BXpXBwM_.js → send-qPyNGSe4.js} +13 -13
  128. package/dist/plugin-sdk/{session-k256LJZT.js → session-COrvpvUQ.js} +3 -3
  129. package/dist/plugin-sdk/{skill-commands-DoRqLzxm.js → skill-commands-DZqhtmiv.js} +4 -4
  130. package/dist/plugin-sdk/{skills-QudILG6e.js → skills-Cw_vXEJb.js} +6 -6
  131. package/dist/plugin-sdk/slash-commands.runtime-D67JLweo.js +13 -0
  132. package/dist/plugin-sdk/slash-dispatch.runtime-DvcpvCJ0.js +52 -0
  133. package/dist/plugin-sdk/slash-skill-commands.runtime-BM1x3azR.js +16 -0
  134. package/dist/plugin-sdk/{store-BbDQw3g6.js → store-CMHj6IIw.js} +2 -2
  135. package/dist/plugin-sdk/subagent-registry-runtime-1lbDyRzz.js +52 -0
  136. package/dist/plugin-sdk/{tables-BhvloMKN.js → tables-CSqrHsKL.js} +1 -1
  137. package/dist/plugin-sdk/{thinking-URzkT-3p.js → thinking-DOnsR_A8.js} +7 -7
  138. package/dist/plugin-sdk/{tokens-DgNRBwIg.js → tokens-BDr0Z9o3.js} +1 -1
  139. package/dist/plugin-sdk/{tool-images-xpqbP6RR.js → tool-images-eEfOVkzf.js} +2 -2
  140. package/dist/plugin-sdk/web-BLyT64pW.js +56 -0
  141. package/dist/plugin-sdk/{whatsapp-actions-RcZ6vp61.js → whatsapp-actions-xcleMoMv.js} +17 -17
  142. package/dist/plugin-sdk/whatsapp.js +50 -50
  143. package/dist/plugin-sdk/zalo.js +2 -2
  144. package/dist/{plugins-CWkRQYDj.js → plugins-4Rj4OjLY.js} +11 -11
  145. package/dist/{proxy-env-Cq5gdrbj.js → proxy-env-DlmzDx8x.js} +1 -1
  146. package/dist/{plugin-sdk/proxy-fetch-Ch95c_Y2.js → proxy-fetch-B2pEfjbR.js} +1 -1
  147. package/dist/{pw-ai-GcYO6HPE.js → pw-ai-CmphSzHx.js} +1 -1
  148. package/dist/{pw-ai-Cl1Lc7RC.js → pw-ai-DNMjFMqH.js} +14 -14
  149. package/dist/{qmd-manager-BsYsO9Ii.js → qmd-manager-BtIKUaO9.js} +10 -10
  150. package/dist/{query-expansion-DtLc3wjL.js → query-expansion-CX-1fS52.js} +6 -6
  151. package/dist/{plugin-sdk/redact-hp9TOulW.js → redact-COik8ET1.js} +1 -1
  152. package/dist/{run-with-concurrency-D_ZpbgEG.js → run-with-concurrency-BgYfgkXT.js} +4 -4
  153. package/dist/runtime-whatsapp-login.runtime-DUb55byQ.js +13 -0
  154. package/dist/runtime-whatsapp-outbound.runtime-Bii_xSfI.js +22 -0
  155. package/dist/{send-Dx2RkUOZ.js → send-6lz6rNVP.js} +6 -6
  156. package/dist/{send-vmONuVgL.js → send-BHTiZcH3.js} +26 -26
  157. package/dist/{send-Bj776ESJ.js → send-L7gRiwyd.js} +7 -7
  158. package/dist/{send-DcxmcFi_.js → send-PE6cwoTe.js} +8 -8
  159. package/dist/{send-BQERFNyo.js → send-dfu6_rgf.js} +5 -5
  160. package/dist/{session-A4QhBRvH.js → session-D8ImowSs.js} +8 -8
  161. package/dist/{skill-commands-CMzBZKG2.js → skill-commands-DNqJ-kwn.js} +9 -9
  162. package/dist/{skills-CE_iqvM5.js → skills-7ODkHQYp.js} +22 -22
  163. package/dist/slash-commands.runtime-CVw6566g.js +16 -0
  164. package/dist/{slash-dispatch.runtime-Dh053pQK.js → slash-dispatch.runtime-131yup2e.js} +6 -6
  165. package/dist/slash-dispatch.runtime-B9Ygtzi4.js +56 -0
  166. package/dist/slash-skill-commands.runtime-DxZ4z5h6.js +20 -0
  167. package/dist/{store--eR1R_UX.js → store-D89wDcz9.js} +2 -2
  168. package/dist/subagent-registry-runtime-DL1Wv7nA.js +56 -0
  169. package/dist/{subagent-registry-runtime-DSi5mnCQ.js → subagent-registry-runtime-DbSf_Je6.js} +6 -6
  170. package/dist/{subsystem-Di1z8l0Z.js → subsystem-B45WV3qB.js} +14 -14
  171. package/dist/{tables-d739Y1xW.js → tables-mE4cJBN2.js} +1 -1
  172. package/dist/{target-errors-CBI2Ga0y.js → target-errors-mnlwhAjP.js} +2 -2
  173. package/dist/{thinking-DXYisHiZ.js → thinking-BeGmb5k6.js} +7 -7
  174. package/dist/{tokens-DxnY9ui_.js → tokens-q32vI39c.js} +1 -1
  175. package/dist/{tool-images-2cBx1W8h.js → tool-images-RZdHiZcG.js} +2 -2
  176. package/dist/{web-CzWRVmFt.js → web-Btj-e8kN.js} +55 -55
  177. package/dist/{web-1hWJDzNA.js → web-MR9d7KyB.js} +6 -6
  178. package/dist/{whatsapp-actions-iEArE_Ez.js → whatsapp-actions-BHbJJyqw.js} +21 -21
  179. package/dist/{workspace-CUVC6GX1.js → workspace-U-DyR64O.js} +20 -20
  180. package/package.json +6 -5
  181. package/scripts/create-instance.sh +18 -15
  182. package/scripts/install-maca.sh +1 -1
  183. package/ui/index.html +16 -0
  184. package/ui/node_modules/.bin/jiti +21 -0
  185. package/ui/node_modules/.bin/lessc +21 -0
  186. package/ui/node_modules/.bin/marked +21 -0
  187. package/ui/node_modules/.bin/playwright +21 -0
  188. package/ui/node_modules/.bin/sass +21 -0
  189. package/ui/node_modules/.bin/tsx +21 -0
  190. package/ui/node_modules/.bin/vite +21 -0
  191. package/ui/node_modules/.bin/vitest +21 -0
  192. package/ui/node_modules/.bin/yaml +21 -0
  193. package/ui/package.json +27 -0
  194. package/ui/public/apple-touch-icon.png +0 -0
  195. package/ui/public/favicon-32.png +0 -0
  196. package/ui/public/favicon.ico +0 -0
  197. package/ui/public/favicon.svg +22 -0
  198. package/ui/src/css.d.ts +1 -0
  199. package/ui/src/i18n/index.ts +3 -0
  200. package/ui/src/i18n/lib/lit-controller.ts +22 -0
  201. package/ui/src/i18n/lib/registry.ts +64 -0
  202. package/ui/src/i18n/lib/translate.ts +123 -0
  203. package/ui/src/i18n/lib/types.ts +9 -0
  204. package/ui/src/i18n/locales/de.ts +129 -0
  205. package/ui/src/i18n/locales/en.ts +337 -0
  206. package/ui/src/i18n/locales/pt-BR.ts +128 -0
  207. package/ui/src/i18n/locales/zh-CN.ts +330 -0
  208. package/ui/src/i18n/locales/zh-TW.ts +125 -0
  209. package/ui/src/i18n/test/translate.test.ts +56 -0
  210. package/ui/src/main.ts +2 -0
  211. package/ui/src/styles/base.css +385 -0
  212. package/ui/src/styles/chat/grouped.css +300 -0
  213. package/ui/src/styles/chat/layout.css +481 -0
  214. package/ui/src/styles/chat/sidebar.css +117 -0
  215. package/ui/src/styles/chat/text.css +146 -0
  216. package/ui/src/styles/chat/tool-cards.css +202 -0
  217. package/ui/src/styles/chat.css +5 -0
  218. package/ui/src/styles/components.css +2612 -0
  219. package/ui/src/styles/config.css +1658 -0
  220. package/ui/src/styles/layout.css +621 -0
  221. package/ui/src/styles/layout.mobile.css +374 -0
  222. package/ui/src/styles.css +5 -0
  223. package/ui/src/ui/__screenshots__/config-form.browser.test.ts/config-form-renderer-flags-unsupported-unions-1.png +0 -0
  224. package/ui/src/ui/__screenshots__/config-form.browser.test.ts/config-form-renderer-renders-inputs-and-patches-values-1.png +0 -0
  225. package/ui/src/ui/__screenshots__/config-form.browser.test.ts/config-form-renderer-renders-union-literals-as-select-options-1.png +0 -0
  226. package/ui/src/ui/__screenshots__/navigation.browser.test.ts/control-UI-routing-auto-scrolls-chat-history-to-the-latest-message-1.png +0 -0
  227. package/ui/src/ui/app-channels.ts +279 -0
  228. package/ui/src/ui/app-chat.ts +266 -0
  229. package/ui/src/ui/app-defaults.ts +50 -0
  230. package/ui/src/ui/app-events.ts +5 -0
  231. package/ui/src/ui/app-gateway.node.test.ts +229 -0
  232. package/ui/src/ui/app-gateway.ts +349 -0
  233. package/ui/src/ui/app-lifecycle.node.test.ts +44 -0
  234. package/ui/src/ui/app-lifecycle.ts +109 -0
  235. package/ui/src/ui/app-polling.ts +69 -0
  236. package/ui/src/ui/app-render-usage-tab.ts +273 -0
  237. package/ui/src/ui/app-render.helpers.node.test.ts +286 -0
  238. package/ui/src/ui/app-render.helpers.ts +574 -0
  239. package/ui/src/ui/app-render.ts +1168 -0
  240. package/ui/src/ui/app-scroll.test.ts +275 -0
  241. package/ui/src/ui/app-scroll.ts +179 -0
  242. package/ui/src/ui/app-settings.test.ts +70 -0
  243. package/ui/src/ui/app-settings.ts +440 -0
  244. package/ui/src/ui/app-tool-stream.node.test.ts +139 -0
  245. package/ui/src/ui/app-tool-stream.ts +455 -0
  246. package/ui/src/ui/app-view-state.ts +321 -0
  247. package/ui/src/ui/app.ts +621 -0
  248. package/ui/src/ui/assistant-identity.ts +23 -0
  249. package/ui/src/ui/chat/constants.ts +12 -0
  250. package/ui/src/ui/chat/copy-as-markdown.ts +97 -0
  251. package/ui/src/ui/chat/grouped-render.ts +287 -0
  252. package/ui/src/ui/chat/message-extract.test.ts +64 -0
  253. package/ui/src/ui/chat/message-extract.ts +122 -0
  254. package/ui/src/ui/chat/message-normalizer.test.ts +179 -0
  255. package/ui/src/ui/chat/message-normalizer.ts +101 -0
  256. package/ui/src/ui/chat/tool-cards.ts +156 -0
  257. package/ui/src/ui/chat/tool-helpers.test.ts +141 -0
  258. package/ui/src/ui/chat/tool-helpers.ts +37 -0
  259. package/ui/src/ui/chat-event-reload.test.ts +47 -0
  260. package/ui/src/ui/chat-event-reload.ts +16 -0
  261. package/ui/src/ui/chat-markdown.browser.test.ts +37 -0
  262. package/ui/src/ui/components/resizable-divider.ts +110 -0
  263. package/ui/src/ui/config-form.browser.test.ts +443 -0
  264. package/ui/src/ui/controllers/agent-files.ts +126 -0
  265. package/ui/src/ui/controllers/agent-identity.ts +59 -0
  266. package/ui/src/ui/controllers/agent-skills.ts +33 -0
  267. package/ui/src/ui/controllers/agents.test.ts +61 -0
  268. package/ui/src/ui/controllers/agents.ts +64 -0
  269. package/ui/src/ui/controllers/assistant-identity.ts +34 -0
  270. package/ui/src/ui/controllers/channels.ts +94 -0
  271. package/ui/src/ui/controllers/channels.types.ts +15 -0
  272. package/ui/src/ui/controllers/chat.test.ts +568 -0
  273. package/ui/src/ui/controllers/chat.ts +318 -0
  274. package/ui/src/ui/controllers/config/form-coerce.ts +160 -0
  275. package/ui/src/ui/controllers/config/form-utils.node.test.ts +455 -0
  276. package/ui/src/ui/controllers/config/form-utils.ts +90 -0
  277. package/ui/src/ui/controllers/config.test.ts +289 -0
  278. package/ui/src/ui/controllers/config.ts +219 -0
  279. package/ui/src/ui/controllers/control-ui-bootstrap.test.ts +82 -0
  280. package/ui/src/ui/controllers/control-ui-bootstrap.ts +49 -0
  281. package/ui/src/ui/controllers/cron-filters.test.ts +81 -0
  282. package/ui/src/ui/controllers/cron.test.ts +1070 -0
  283. package/ui/src/ui/controllers/cron.ts +921 -0
  284. package/ui/src/ui/controllers/debug.ts +60 -0
  285. package/ui/src/ui/controllers/devices.ts +159 -0
  286. package/ui/src/ui/controllers/exec-approval.ts +100 -0
  287. package/ui/src/ui/controllers/exec-approvals.ts +170 -0
  288. package/ui/src/ui/controllers/logs.ts +147 -0
  289. package/ui/src/ui/controllers/nodes.ts +32 -0
  290. package/ui/src/ui/controllers/presence.ts +37 -0
  291. package/ui/src/ui/controllers/sessions.test.ts +104 -0
  292. package/ui/src/ui/controllers/sessions.ts +127 -0
  293. package/ui/src/ui/controllers/skills.ts +157 -0
  294. package/ui/src/ui/controllers/usage.node.test.ts +181 -0
  295. package/ui/src/ui/controllers/usage.ts +315 -0
  296. package/ui/src/ui/data/moonshot-kimi-k2.ts +45 -0
  297. package/ui/src/ui/device-auth.ts +73 -0
  298. package/ui/src/ui/device-identity.ts +112 -0
  299. package/ui/src/ui/external-link.test.ts +18 -0
  300. package/ui/src/ui/external-link.ts +19 -0
  301. package/ui/src/ui/focus-mode.browser.test.ts +39 -0
  302. package/ui/src/ui/format.test.ts +101 -0
  303. package/ui/src/ui/format.ts +60 -0
  304. package/ui/src/ui/gateway.ts +360 -0
  305. package/ui/src/ui/icons.ts +256 -0
  306. package/ui/src/ui/markdown.test.ts +85 -0
  307. package/ui/src/ui/markdown.ts +139 -0
  308. package/ui/src/ui/navigation.browser.test.ts +188 -0
  309. package/ui/src/ui/navigation.test.ts +189 -0
  310. package/ui/src/ui/navigation.ts +165 -0
  311. package/ui/src/ui/open-external-url.test.ts +108 -0
  312. package/ui/src/ui/open-external-url.ts +73 -0
  313. package/ui/src/ui/presenter.ts +85 -0
  314. package/ui/src/ui/storage.node.test.ts +63 -0
  315. package/ui/src/ui/storage.ts +99 -0
  316. package/ui/src/ui/test-helpers/app-mount.ts +27 -0
  317. package/ui/src/ui/text-direction.test.ts +24 -0
  318. package/ui/src/ui/text-direction.ts +30 -0
  319. package/ui/src/ui/theme-transition.ts +109 -0
  320. package/ui/src/ui/theme.ts +16 -0
  321. package/ui/src/ui/tool-display.ts +159 -0
  322. package/ui/src/ui/types/chat-types.ts +44 -0
  323. package/ui/src/ui/types.ts +627 -0
  324. package/ui/src/ui/ui-types.ts +54 -0
  325. package/ui/src/ui/usage-helpers.node.test.ts +43 -0
  326. package/ui/src/ui/usage-helpers.ts +321 -0
  327. package/ui/src/ui/usage-types.ts +22 -0
  328. package/ui/src/ui/uuid.test.ts +41 -0
  329. package/ui/src/ui/uuid.ts +57 -0
  330. package/ui/src/ui/views/agents-panels-status-files.ts +461 -0
  331. package/ui/src/ui/views/agents-panels-tools-skills.browser.test.ts +102 -0
  332. package/ui/src/ui/views/agents-panels-tools-skills.ts +537 -0
  333. package/ui/src/ui/views/agents-utils.test.ts +100 -0
  334. package/ui/src/ui/views/agents-utils.ts +502 -0
  335. package/ui/src/ui/views/agents.ts +499 -0
  336. package/ui/src/ui/views/channel-config-extras.ts +49 -0
  337. package/ui/src/ui/views/channels.config.ts +155 -0
  338. package/ui/src/ui/views/channels.discord.ts +65 -0
  339. package/ui/src/ui/views/channels.googlechat.ts +79 -0
  340. package/ui/src/ui/views/channels.imessage.ts +65 -0
  341. package/ui/src/ui/views/channels.nostr-profile-form.ts +321 -0
  342. package/ui/src/ui/views/channels.nostr.ts +237 -0
  343. package/ui/src/ui/views/channels.shared.ts +38 -0
  344. package/ui/src/ui/views/channels.signal.ts +69 -0
  345. package/ui/src/ui/views/channels.slack.ts +65 -0
  346. package/ui/src/ui/views/channels.telegram.ts +120 -0
  347. package/ui/src/ui/views/channels.ts +325 -0
  348. package/ui/src/ui/views/channels.types.ts +62 -0
  349. package/ui/src/ui/views/channels.whatsapp.ts +118 -0
  350. package/ui/src/ui/views/chat-image-open.browser.test.ts +70 -0
  351. package/ui/src/ui/views/chat.test.ts +227 -0
  352. package/ui/src/ui/views/chat.ts +616 -0
  353. package/ui/src/ui/views/config-form.analyze.ts +267 -0
  354. package/ui/src/ui/views/config-form.node.ts +1073 -0
  355. package/ui/src/ui/views/config-form.render.ts +478 -0
  356. package/ui/src/ui/views/config-form.search.node.test.ts +69 -0
  357. package/ui/src/ui/views/config-form.shared.ts +96 -0
  358. package/ui/src/ui/views/config-form.ts +4 -0
  359. package/ui/src/ui/views/config-search.node.test.ts +50 -0
  360. package/ui/src/ui/views/config-search.ts +92 -0
  361. package/ui/src/ui/views/config.browser.test.ts +233 -0
  362. package/ui/src/ui/views/config.ts +820 -0
  363. package/ui/src/ui/views/cron.test.ts +741 -0
  364. package/ui/src/ui/views/cron.ts +1758 -0
  365. package/ui/src/ui/views/debug.ts +151 -0
  366. package/ui/src/ui/views/exec-approval.ts +89 -0
  367. package/ui/src/ui/views/gateway-url-confirmation.ts +40 -0
  368. package/ui/src/ui/views/instances.ts +89 -0
  369. package/ui/src/ui/views/logs.ts +155 -0
  370. package/ui/src/ui/views/markdown-sidebar.ts +40 -0
  371. package/ui/src/ui/views/nodes-exec-approvals.ts +617 -0
  372. package/ui/src/ui/views/nodes-shared.ts +67 -0
  373. package/ui/src/ui/views/nodes.ts +485 -0
  374. package/ui/src/ui/views/overview-hints.ts +16 -0
  375. package/ui/src/ui/views/overview.node.test.ts +39 -0
  376. package/ui/src/ui/views/overview.ts +361 -0
  377. package/ui/src/ui/views/sessions.test.ts +81 -0
  378. package/ui/src/ui/views/sessions.ts +321 -0
  379. package/ui/src/ui/views/skills-grouping.ts +40 -0
  380. package/ui/src/ui/views/skills-shared.ts +52 -0
  381. package/ui/src/ui/views/skills.ts +192 -0
  382. package/ui/src/ui/views/usage-metrics.ts +578 -0
  383. package/ui/src/ui/views/usage-query.ts +277 -0
  384. package/ui/src/ui/views/usage-render-details.test.ts +136 -0
  385. package/ui/src/ui/views/usage-render-details.ts +1083 -0
  386. package/ui/src/ui/views/usage-render-overview.ts +796 -0
  387. package/ui/src/ui/views/usage-styles/usageStyles-part1.ts +701 -0
  388. package/ui/src/ui/views/usage-styles/usageStyles-part2.ts +702 -0
  389. package/ui/src/ui/views/usage-styles/usageStyles-part3.ts +551 -0
  390. package/ui/src/ui/views/usage.ts +836 -0
  391. package/ui/src/ui/views/usageStyles.ts +5 -0
  392. package/ui/src/ui/views/usageTypes.ts +105 -0
  393. package/ui/vite.config.ts +43 -0
  394. package/ui/vitest.config.ts +15 -0
  395. package/ui/vitest.node.config.ts +10 -0
  396. package/dist/deliver-runtime-P-G3bPjW.js +0 -36
  397. package/dist/deps-send-discord.runtime-DnbhTFX9.js +0 -26
  398. package/dist/deps-send-imessage.runtime-BOiQ6mDx.js +0 -25
  399. package/dist/deps-send-signal.runtime-CTcl388M.js +0 -24
  400. package/dist/deps-send-slack.runtime-CCqBz4Kg.js +0 -22
  401. package/dist/deps-send-telegram.runtime-DGSKTCpH.js +0 -27
  402. package/dist/deps-send-whatsapp.runtime-CJkTHkah.js +0 -60
  403. package/dist/errors-CCLeFWAg.js +0 -54
  404. package/dist/image-runtime-CVv2ra9J.js +0 -29
  405. package/dist/manager-runtime-BN6VevdC.js +0 -18
  406. package/dist/pi-model-discovery-BGgOlX8N.js +0 -134
  407. package/dist/pi-model-discovery-runtime-Bwmi4Ev8.js +0 -11
  408. package/dist/plugin-sdk/accounts-CJWOBzwB.js +0 -35
  409. package/dist/plugin-sdk/accounts-DP1-L-QS.js +0 -288
  410. package/dist/plugin-sdk/accounts-DZhWlEg3.js +0 -46
  411. package/dist/plugin-sdk/active-listener-CftX5jLD.js +0 -50
  412. package/dist/plugin-sdk/api-key-rotation-BRE4X2tf.js +0 -181
  413. package/dist/plugin-sdk/audio-preflight-DGEUDxxR.js +0 -69
  414. package/dist/plugin-sdk/audio-transcription-runner-DkoPNPYt.js +0 -2176
  415. package/dist/plugin-sdk/audit-membership-runtime-DSBHHw7o.js +0 -58
  416. package/dist/plugin-sdk/channel-activity-F3d0yUwy.js +0 -94
  417. package/dist/plugin-sdk/channel-web-QF7EpjeP.js +0 -2256
  418. package/dist/plugin-sdk/chrome-BXoCyCkY.js +0 -2415
  419. package/dist/plugin-sdk/commands-registry-t7cXBTfN.js +0 -1125
  420. package/dist/plugin-sdk/config-BkEnz2Po.js +0 -17913
  421. package/dist/plugin-sdk/deliver-B6AG_l67.js +0 -1694
  422. package/dist/plugin-sdk/deliver-runtime-BFdqklJM.js +0 -32
  423. package/dist/plugin-sdk/deliver-runtime-D585kJZc.js +0 -32
  424. package/dist/plugin-sdk/deps-send-discord.runtime-DuqpYwU0.js +0 -23
  425. package/dist/plugin-sdk/deps-send-discord.runtime-a_OKY2js.js +0 -23
  426. package/dist/plugin-sdk/deps-send-imessage.runtime-Baxy9TD4.js +0 -22
  427. package/dist/plugin-sdk/deps-send-imessage.runtime-CZ2rS8Lb.js +0 -22
  428. package/dist/plugin-sdk/deps-send-signal.runtime-BdqiWhIh.js +0 -21
  429. package/dist/plugin-sdk/deps-send-signal.runtime-BwXoCrFl.js +0 -21
  430. package/dist/plugin-sdk/deps-send-slack.runtime-04s36qiC.js +0 -19
  431. package/dist/plugin-sdk/deps-send-slack.runtime-CLmKjgso.js +0 -19
  432. package/dist/plugin-sdk/deps-send-telegram.runtime-BKfdBKnZ.js +0 -24
  433. package/dist/plugin-sdk/deps-send-telegram.runtime-LE5tkPvr.js +0 -24
  434. package/dist/plugin-sdk/deps-send-whatsapp.runtime-BOTwkbx_.js +0 -57
  435. package/dist/plugin-sdk/deps-send-whatsapp.runtime-Bz57lobC.js +0 -57
  436. package/dist/plugin-sdk/diagnostic-CsP-lEkI.js +0 -319
  437. package/dist/plugin-sdk/fetch-guard-DETCcJzQ.js +0 -156
  438. package/dist/plugin-sdk/fs-safe-B8y811FR.js +0 -352
  439. package/dist/plugin-sdk/image-DjTEkYZE.js +0 -2310
  440. package/dist/plugin-sdk/image-ops-BSiMpAw4.js +0 -584
  441. package/dist/plugin-sdk/image-runtime-6xPp8m5a.js +0 -25
  442. package/dist/plugin-sdk/image-runtime-B8twoubs.js +0 -25
  443. package/dist/plugin-sdk/ir-DQ7_HbvK.js +0 -1296
  444. package/dist/plugin-sdk/local-roots-BUP4YBmR.js +0 -186
  445. package/dist/plugin-sdk/logger-CZY9KIoY.js +0 -1163
  446. package/dist/plugin-sdk/login-BxEKLlCo.js +0 -57
  447. package/dist/plugin-sdk/login-qr-BQIpMPr9.js +0 -320
  448. package/dist/plugin-sdk/manager-I6KbPihW.js +0 -3917
  449. package/dist/plugin-sdk/manager-runtime-CFfYYWIQ.js +0 -15
  450. package/dist/plugin-sdk/manager-runtime-CMeLwose.js +0 -15
  451. package/dist/plugin-sdk/outbound-NS6UHnB6.js +0 -212
  452. package/dist/plugin-sdk/outbound-attachment-Dy6fyf6H.js +0 -19
  453. package/dist/plugin-sdk/path-alias-guards-DBjLbIX_.js +0 -43
  454. package/dist/plugin-sdk/paths-vTM3Lh3X.js +0 -166
  455. package/dist/plugin-sdk/pi-embedded-helpers-1R1gu7eX.js +0 -9627
  456. package/dist/plugin-sdk/pi-model-discovery-runtime-D8CJhtJY.js +0 -8
  457. package/dist/plugin-sdk/pi-model-discovery-runtime-Do9o-dUd.js +0 -8
  458. package/dist/plugin-sdk/pi-tools.before-tool-call.runtime-D4sFsIks.js +0 -354
  459. package/dist/plugin-sdk/plugins-DeBZB9l_.js +0 -864
  460. package/dist/plugin-sdk/pw-ai-DEOmCSSC.js +0 -1938
  461. package/dist/plugin-sdk/qmd-manager-HyYKoEch.js +0 -1448
  462. package/dist/plugin-sdk/query-expansion-CeyKUeDW.js +0 -1011
  463. package/dist/plugin-sdk/reply-DAo_Jt8K.js +0 -97916
  464. package/dist/plugin-sdk/resolve-outbound-target-B42qgQS9.js +0 -40
  465. package/dist/plugin-sdk/run-with-concurrency-Bt_ks0Qa.js +0 -1994
  466. package/dist/plugin-sdk/runtime-whatsapp-login.runtime-B6W989eF.js +0 -10
  467. package/dist/plugin-sdk/runtime-whatsapp-login.runtime-SkO91TZH.js +0 -10
  468. package/dist/plugin-sdk/runtime-whatsapp-outbound.runtime-B0VWK5hm.js +0 -19
  469. package/dist/plugin-sdk/runtime-whatsapp-outbound.runtime-c_GDFy37.js +0 -19
  470. package/dist/plugin-sdk/send-CQpMudwO.js +0 -2587
  471. package/dist/plugin-sdk/send-DQHLzVyO.js +0 -414
  472. package/dist/plugin-sdk/send-DTB24bEF.js +0 -3135
  473. package/dist/plugin-sdk/send-DfHadjZ_.js +0 -503
  474. package/dist/plugin-sdk/send-XXlW2iny.js +0 -540
  475. package/dist/plugin-sdk/session-6TF6MyaC.js +0 -169
  476. package/dist/plugin-sdk/skill-commands-CkGeFUMl.js +0 -342
  477. package/dist/plugin-sdk/skills-CBkHBYPq.js +0 -1428
  478. package/dist/plugin-sdk/slash-commands.runtime-CxliuGaP.js +0 -13
  479. package/dist/plugin-sdk/slash-commands.runtime-DS6vCNSL.js +0 -13
  480. package/dist/plugin-sdk/slash-dispatch.runtime-BXrxb2wd.js +0 -52
  481. package/dist/plugin-sdk/slash-dispatch.runtime-DFaeYlJQ.js +0 -52
  482. package/dist/plugin-sdk/slash-skill-commands.runtime-0M0OLCxq.js +0 -16
  483. package/dist/plugin-sdk/slash-skill-commands.runtime-Bd6qQ2oT.js +0 -16
  484. package/dist/plugin-sdk/ssrf-cFtplYtS.js +0 -202
  485. package/dist/plugin-sdk/store-5nyxY3WU.js +0 -81
  486. package/dist/plugin-sdk/subagent-registry-runtime-1uwQbuXj.js +0 -52
  487. package/dist/plugin-sdk/subagent-registry-runtime-DCtmDwna.js +0 -52
  488. package/dist/plugin-sdk/tables-C47P4GTN.js +0 -55
  489. package/dist/plugin-sdk/target-errors-Blia4S69.js +0 -195
  490. package/dist/plugin-sdk/thinking-Bo2eosVa.js +0 -1206
  491. package/dist/plugin-sdk/tokens-B1PW5Ayy.js +0 -52
  492. package/dist/plugin-sdk/tool-images-Gk_-0y2N.js +0 -274
  493. package/dist/plugin-sdk/web-B74yhL2N.js +0 -56
  494. package/dist/plugin-sdk/web-CVxZbXyH.js +0 -56
  495. package/dist/plugin-sdk/whatsapp-actions-Bw0H9g-n.js +0 -80
  496. package/dist/proxy-fetch-CCjEYbFm.js +0 -38
  497. package/dist/redact-ClbcYG1J.js +0 -319
  498. package/dist/runtime-whatsapp-login.runtime-IeylZEl4.js +0 -13
  499. package/dist/runtime-whatsapp-outbound.runtime-ClBRuLsq.js +0 -22
  500. package/dist/slash-commands.runtime-Cpn2tYW4.js +0 -16
  501. package/dist/slash-dispatch.runtime-DoBAQBU5.js +0 -56
  502. package/dist/slash-skill-commands.runtime-DKMvvdDW.js +0 -20
  503. package/dist/subagent-registry-runtime-ppWS3tVu.js +0 -56
@@ -0,0 +1,921 @@
1
+ import { t } from "../../i18n/index.ts";
2
+ import { DEFAULT_CRON_FORM } from "../app-defaults.ts";
3
+ import { toNumber } from "../format.ts";
4
+ import type { GatewayBrowserClient } from "../gateway.ts";
5
+ import type {
6
+ CronJob,
7
+ CronDeliveryStatus,
8
+ CronJobsEnabledFilter,
9
+ CronJobsListResult,
10
+ CronJobsSortBy,
11
+ CronRunScope,
12
+ CronRunLogEntry,
13
+ CronRunsResult,
14
+ CronRunsStatusFilter,
15
+ CronRunsStatusValue,
16
+ CronSortDir,
17
+ CronStatus,
18
+ } from "../types.ts";
19
+ import { CRON_CHANNEL_LAST } from "../ui-types.ts";
20
+ import type { CronFormState } from "../ui-types.ts";
21
+
22
+ export type CronFieldKey =
23
+ | "name"
24
+ | "scheduleAt"
25
+ | "everyAmount"
26
+ | "cronExpr"
27
+ | "staggerAmount"
28
+ | "payloadText"
29
+ | "payloadModel"
30
+ | "payloadThinking"
31
+ | "timeoutSeconds"
32
+ | "deliveryTo"
33
+ | "failureAlertAfter"
34
+ | "failureAlertCooldownSeconds";
35
+
36
+ export type CronFieldErrors = Partial<Record<CronFieldKey, string>>;
37
+
38
+ export type CronJobsScheduleKindFilter = "all" | "at" | "every" | "cron";
39
+ export type CronJobsLastStatusFilter = "all" | "ok" | "error" | "skipped";
40
+
41
+ export type CronState = {
42
+ client: GatewayBrowserClient | null;
43
+ connected: boolean;
44
+ cronLoading: boolean;
45
+ cronJobsLoadingMore: boolean;
46
+ cronJobs: CronJob[];
47
+ cronJobsTotal: number;
48
+ cronJobsHasMore: boolean;
49
+ cronJobsNextOffset: number | null;
50
+ cronJobsLimit: number;
51
+ cronJobsQuery: string;
52
+ cronJobsEnabledFilter: CronJobsEnabledFilter;
53
+ cronJobsScheduleKindFilter: CronJobsScheduleKindFilter;
54
+ cronJobsLastStatusFilter: CronJobsLastStatusFilter;
55
+ cronJobsSortBy: CronJobsSortBy;
56
+ cronJobsSortDir: CronSortDir;
57
+ cronStatus: CronStatus | null;
58
+ cronError: string | null;
59
+ cronForm: CronFormState;
60
+ cronFieldErrors: CronFieldErrors;
61
+ cronEditingJobId: string | null;
62
+ cronRunsJobId: string | null;
63
+ cronRunsLoadingMore: boolean;
64
+ cronRuns: CronRunLogEntry[];
65
+ cronRunsTotal: number;
66
+ cronRunsHasMore: boolean;
67
+ cronRunsNextOffset: number | null;
68
+ cronRunsLimit: number;
69
+ cronRunsScope: CronRunScope;
70
+ cronRunsStatuses: CronRunsStatusValue[];
71
+ cronRunsDeliveryStatuses: CronDeliveryStatus[];
72
+ cronRunsStatusFilter: CronRunsStatusFilter;
73
+ cronRunsQuery: string;
74
+ cronRunsSortDir: CronSortDir;
75
+ cronBusy: boolean;
76
+ };
77
+
78
+ export type CronModelSuggestionsState = {
79
+ client: GatewayBrowserClient | null;
80
+ connected: boolean;
81
+ cronModelSuggestions: string[];
82
+ };
83
+
84
+ export function supportsAnnounceDelivery(
85
+ form: Pick<CronFormState, "sessionTarget" | "payloadKind">,
86
+ ) {
87
+ return form.sessionTarget === "isolated" && form.payloadKind === "agentTurn";
88
+ }
89
+
90
+ export function normalizeCronFormState(form: CronFormState): CronFormState {
91
+ if (form.deliveryMode !== "announce") {
92
+ return form;
93
+ }
94
+ if (supportsAnnounceDelivery(form)) {
95
+ return form;
96
+ }
97
+ return {
98
+ ...form,
99
+ deliveryMode: "none",
100
+ };
101
+ }
102
+
103
+ export function validateCronForm(form: CronFormState): CronFieldErrors {
104
+ const errors: CronFieldErrors = {};
105
+ if (!form.name.trim()) {
106
+ errors.name = "cron.errors.nameRequired";
107
+ }
108
+ if (form.scheduleKind === "at") {
109
+ const ms = Date.parse(form.scheduleAt);
110
+ if (!Number.isFinite(ms)) {
111
+ errors.scheduleAt = "cron.errors.scheduleAtInvalid";
112
+ }
113
+ } else if (form.scheduleKind === "every") {
114
+ const amount = toNumber(form.everyAmount, 0);
115
+ if (amount <= 0) {
116
+ errors.everyAmount = "cron.errors.everyAmountInvalid";
117
+ }
118
+ } else {
119
+ if (!form.cronExpr.trim()) {
120
+ errors.cronExpr = "cron.errors.cronExprRequired";
121
+ }
122
+ if (!form.scheduleExact) {
123
+ const staggerAmount = form.staggerAmount.trim();
124
+ if (staggerAmount) {
125
+ const stagger = toNumber(staggerAmount, 0);
126
+ if (stagger <= 0) {
127
+ errors.staggerAmount = "cron.errors.staggerAmountInvalid";
128
+ }
129
+ }
130
+ }
131
+ }
132
+ if (!form.payloadText.trim()) {
133
+ errors.payloadText =
134
+ form.payloadKind === "systemEvent"
135
+ ? "cron.errors.systemTextRequired"
136
+ : "cron.errors.agentMessageRequired";
137
+ }
138
+ if (form.payloadKind === "agentTurn") {
139
+ const timeoutRaw = form.timeoutSeconds.trim();
140
+ if (timeoutRaw) {
141
+ const timeout = toNumber(timeoutRaw, 0);
142
+ if (timeout <= 0) {
143
+ errors.timeoutSeconds = "cron.errors.timeoutInvalid";
144
+ }
145
+ }
146
+ }
147
+ if (form.deliveryMode === "webhook") {
148
+ const target = form.deliveryTo.trim();
149
+ if (!target) {
150
+ errors.deliveryTo = "cron.errors.webhookUrlRequired";
151
+ } else if (!/^https?:\/\//i.test(target)) {
152
+ errors.deliveryTo = "cron.errors.webhookUrlInvalid";
153
+ }
154
+ }
155
+ if (form.failureAlertMode === "custom") {
156
+ const afterRaw = form.failureAlertAfter.trim();
157
+ if (afterRaw) {
158
+ const after = toNumber(afterRaw, 0);
159
+ if (!Number.isFinite(after) || after <= 0) {
160
+ errors.failureAlertAfter = "Failure alert threshold must be greater than 0.";
161
+ }
162
+ }
163
+ const cooldownRaw = form.failureAlertCooldownSeconds.trim();
164
+ if (cooldownRaw) {
165
+ const cooldown = toNumber(cooldownRaw, -1);
166
+ if (!Number.isFinite(cooldown) || cooldown < 0) {
167
+ errors.failureAlertCooldownSeconds = "Cooldown must be 0 or greater.";
168
+ }
169
+ }
170
+ }
171
+ return errors;
172
+ }
173
+
174
+ export function hasCronFormErrors(errors: CronFieldErrors): boolean {
175
+ return Object.keys(errors).length > 0;
176
+ }
177
+
178
+ export async function loadCronStatus(state: CronState) {
179
+ if (!state.client || !state.connected) {
180
+ return;
181
+ }
182
+ try {
183
+ const res = await state.client.request<CronStatus>("cron.status", {});
184
+ state.cronStatus = res;
185
+ } catch (err) {
186
+ state.cronError = String(err);
187
+ }
188
+ }
189
+
190
+ export async function loadCronModelSuggestions(state: CronModelSuggestionsState) {
191
+ if (!state.client || !state.connected) {
192
+ return;
193
+ }
194
+ try {
195
+ const res = await state.client.request("models.list", {});
196
+ const models = (res as { models?: unknown[] } | null)?.models;
197
+ if (!Array.isArray(models)) {
198
+ state.cronModelSuggestions = [];
199
+ return;
200
+ }
201
+ const ids = models
202
+ .map((entry) => {
203
+ if (!entry || typeof entry !== "object") {
204
+ return "";
205
+ }
206
+ const id = (entry as { id?: unknown }).id;
207
+ return typeof id === "string" ? id.trim() : "";
208
+ })
209
+ .filter(Boolean);
210
+ state.cronModelSuggestions = Array.from(new Set(ids)).toSorted((a, b) => a.localeCompare(b));
211
+ } catch {
212
+ state.cronModelSuggestions = [];
213
+ }
214
+ }
215
+
216
+ export async function loadCronJobs(state: CronState) {
217
+ return await loadCronJobsPage(state, { append: false });
218
+ }
219
+
220
+ function normalizeCronPageMeta(params: {
221
+ totalRaw: unknown;
222
+ limitRaw: unknown;
223
+ offsetRaw: unknown;
224
+ nextOffsetRaw: unknown;
225
+ hasMoreRaw: unknown;
226
+ pageCount: number;
227
+ }) {
228
+ const total =
229
+ typeof params.totalRaw === "number" && Number.isFinite(params.totalRaw)
230
+ ? Math.max(0, Math.floor(params.totalRaw))
231
+ : params.pageCount;
232
+ const limit =
233
+ typeof params.limitRaw === "number" && Number.isFinite(params.limitRaw)
234
+ ? Math.max(1, Math.floor(params.limitRaw))
235
+ : Math.max(1, params.pageCount);
236
+ const offset =
237
+ typeof params.offsetRaw === "number" && Number.isFinite(params.offsetRaw)
238
+ ? Math.max(0, Math.floor(params.offsetRaw))
239
+ : 0;
240
+ const hasMore =
241
+ typeof params.hasMoreRaw === "boolean"
242
+ ? params.hasMoreRaw
243
+ : offset + params.pageCount < Math.max(total, offset + params.pageCount);
244
+ const nextOffset =
245
+ typeof params.nextOffsetRaw === "number" && Number.isFinite(params.nextOffsetRaw)
246
+ ? Math.max(0, Math.floor(params.nextOffsetRaw))
247
+ : hasMore
248
+ ? offset + params.pageCount
249
+ : null;
250
+ return { total, limit, offset, hasMore, nextOffset };
251
+ }
252
+
253
+ export async function loadCronJobsPage(state: CronState, opts?: { append?: boolean }) {
254
+ if (!state.client || !state.connected) {
255
+ return;
256
+ }
257
+ if (state.cronLoading || state.cronJobsLoadingMore) {
258
+ return;
259
+ }
260
+ const append = opts?.append === true;
261
+ if (append) {
262
+ if (!state.cronJobsHasMore) {
263
+ return;
264
+ }
265
+ state.cronJobsLoadingMore = true;
266
+ } else {
267
+ state.cronLoading = true;
268
+ }
269
+ state.cronError = null;
270
+ try {
271
+ const offset = append ? Math.max(0, state.cronJobsNextOffset ?? state.cronJobs.length) : 0;
272
+ const res = await state.client.request<CronJobsListResult>("cron.list", {
273
+ includeDisabled: state.cronJobsEnabledFilter === "all",
274
+ limit: state.cronJobsLimit,
275
+ offset,
276
+ query: state.cronJobsQuery.trim() || undefined,
277
+ enabled: state.cronJobsEnabledFilter,
278
+ sortBy: state.cronJobsSortBy,
279
+ sortDir: state.cronJobsSortDir,
280
+ });
281
+ const jobs = Array.isArray(res.jobs) ? res.jobs : [];
282
+ state.cronJobs = append ? [...state.cronJobs, ...jobs] : jobs;
283
+ const meta = normalizeCronPageMeta({
284
+ totalRaw: res.total,
285
+ limitRaw: res.limit,
286
+ offsetRaw: res.offset,
287
+ nextOffsetRaw: res.nextOffset,
288
+ hasMoreRaw: res.hasMore,
289
+ pageCount: jobs.length,
290
+ });
291
+ state.cronJobsTotal = Math.max(meta.total, state.cronJobs.length);
292
+ state.cronJobsHasMore = meta.hasMore;
293
+ state.cronJobsNextOffset = meta.nextOffset;
294
+ if (
295
+ state.cronEditingJobId &&
296
+ !state.cronJobs.some((job) => job.id === state.cronEditingJobId)
297
+ ) {
298
+ clearCronEditState(state);
299
+ }
300
+ } catch (err) {
301
+ state.cronError = String(err);
302
+ } finally {
303
+ if (append) {
304
+ state.cronJobsLoadingMore = false;
305
+ } else {
306
+ state.cronLoading = false;
307
+ }
308
+ }
309
+ }
310
+
311
+ export async function loadMoreCronJobs(state: CronState) {
312
+ await loadCronJobsPage(state, { append: true });
313
+ }
314
+
315
+ export async function reloadCronJobs(state: CronState) {
316
+ await loadCronJobsPage(state, { append: false });
317
+ }
318
+
319
+ export function updateCronJobsFilter(
320
+ state: CronState,
321
+ patch: Partial<
322
+ Pick<
323
+ CronState,
324
+ | "cronJobsQuery"
325
+ | "cronJobsEnabledFilter"
326
+ | "cronJobsScheduleKindFilter"
327
+ | "cronJobsLastStatusFilter"
328
+ | "cronJobsSortBy"
329
+ | "cronJobsSortDir"
330
+ >
331
+ >,
332
+ ) {
333
+ if (typeof patch.cronJobsQuery === "string") {
334
+ state.cronJobsQuery = patch.cronJobsQuery;
335
+ }
336
+ if (patch.cronJobsEnabledFilter) {
337
+ state.cronJobsEnabledFilter = patch.cronJobsEnabledFilter;
338
+ }
339
+ if (patch.cronJobsScheduleKindFilter) {
340
+ state.cronJobsScheduleKindFilter = patch.cronJobsScheduleKindFilter;
341
+ }
342
+ if (patch.cronJobsLastStatusFilter) {
343
+ state.cronJobsLastStatusFilter = patch.cronJobsLastStatusFilter;
344
+ }
345
+ if (patch.cronJobsSortBy) {
346
+ state.cronJobsSortBy = patch.cronJobsSortBy;
347
+ }
348
+ if (patch.cronJobsSortDir) {
349
+ state.cronJobsSortDir = patch.cronJobsSortDir;
350
+ }
351
+ }
352
+
353
+ export function getVisibleCronJobs(
354
+ state: Pick<CronState, "cronJobs" | "cronJobsScheduleKindFilter" | "cronJobsLastStatusFilter">,
355
+ ): CronJob[] {
356
+ return state.cronJobs.filter((job) => {
357
+ if (
358
+ state.cronJobsScheduleKindFilter !== "all" &&
359
+ job.schedule.kind !== state.cronJobsScheduleKindFilter
360
+ ) {
361
+ return false;
362
+ }
363
+ if (
364
+ state.cronJobsLastStatusFilter !== "all" &&
365
+ job.state?.lastStatus !== state.cronJobsLastStatusFilter
366
+ ) {
367
+ return false;
368
+ }
369
+ return true;
370
+ });
371
+ }
372
+
373
+ function clearCronEditState(state: CronState) {
374
+ state.cronEditingJobId = null;
375
+ }
376
+
377
+ function resetCronFormToDefaults(state: CronState) {
378
+ state.cronForm = { ...DEFAULT_CRON_FORM };
379
+ state.cronFieldErrors = validateCronForm(state.cronForm);
380
+ }
381
+
382
+ function formatDateTimeLocal(input: string): string {
383
+ const ms = Date.parse(input);
384
+ if (!Number.isFinite(ms)) {
385
+ return "";
386
+ }
387
+ const date = new Date(ms);
388
+ const year = date.getFullYear();
389
+ const month = String(date.getMonth() + 1).padStart(2, "0");
390
+ const day = String(date.getDate()).padStart(2, "0");
391
+ const hour = String(date.getHours()).padStart(2, "0");
392
+ const minute = String(date.getMinutes()).padStart(2, "0");
393
+ return `${year}-${month}-${day}T${hour}:${minute}`;
394
+ }
395
+
396
+ function parseEverySchedule(everyMs: number): Pick<CronFormState, "everyAmount" | "everyUnit"> {
397
+ if (everyMs % 86_400_000 === 0) {
398
+ return { everyAmount: String(Math.max(1, everyMs / 86_400_000)), everyUnit: "days" };
399
+ }
400
+ if (everyMs % 3_600_000 === 0) {
401
+ return { everyAmount: String(Math.max(1, everyMs / 3_600_000)), everyUnit: "hours" };
402
+ }
403
+ const minutes = Math.max(1, Math.ceil(everyMs / 60_000));
404
+ return { everyAmount: String(minutes), everyUnit: "minutes" };
405
+ }
406
+
407
+ function parseStaggerSchedule(
408
+ staggerMs?: number,
409
+ ): Pick<CronFormState, "scheduleExact" | "staggerAmount" | "staggerUnit"> {
410
+ if (staggerMs === 0) {
411
+ return { scheduleExact: true, staggerAmount: "", staggerUnit: "seconds" };
412
+ }
413
+ if (typeof staggerMs !== "number" || !Number.isFinite(staggerMs) || staggerMs < 0) {
414
+ return { scheduleExact: false, staggerAmount: "", staggerUnit: "seconds" };
415
+ }
416
+ if (staggerMs % 60_000 === 0) {
417
+ return {
418
+ scheduleExact: false,
419
+ staggerAmount: String(Math.max(1, staggerMs / 60_000)),
420
+ staggerUnit: "minutes",
421
+ };
422
+ }
423
+ return {
424
+ scheduleExact: false,
425
+ staggerAmount: String(Math.max(1, Math.ceil(staggerMs / 1_000))),
426
+ staggerUnit: "seconds",
427
+ };
428
+ }
429
+
430
+ function jobToForm(job: CronJob, prev: CronFormState): CronFormState {
431
+ const failureAlert = job.failureAlert;
432
+ const next: CronFormState = {
433
+ ...prev,
434
+ name: job.name,
435
+ description: job.description ?? "",
436
+ agentId: job.agentId ?? "",
437
+ sessionKey: job.sessionKey ?? "",
438
+ clearAgent: false,
439
+ enabled: job.enabled,
440
+ deleteAfterRun: job.deleteAfterRun ?? false,
441
+ scheduleKind: job.schedule.kind,
442
+ scheduleAt: "",
443
+ everyAmount: prev.everyAmount,
444
+ everyUnit: prev.everyUnit,
445
+ cronExpr: prev.cronExpr,
446
+ cronTz: "",
447
+ scheduleExact: false,
448
+ staggerAmount: "",
449
+ staggerUnit: "seconds",
450
+ sessionTarget: job.sessionTarget,
451
+ wakeMode: job.wakeMode,
452
+ payloadKind: job.payload.kind,
453
+ payloadText: job.payload.kind === "systemEvent" ? job.payload.text : job.payload.message,
454
+ payloadModel: job.payload.kind === "agentTurn" ? (job.payload.model ?? "") : "",
455
+ payloadThinking: job.payload.kind === "agentTurn" ? (job.payload.thinking ?? "") : "",
456
+ payloadLightContext:
457
+ job.payload.kind === "agentTurn" ? job.payload.lightContext === true : false,
458
+ deliveryMode: job.delivery?.mode ?? "none",
459
+ deliveryChannel: job.delivery?.channel ?? CRON_CHANNEL_LAST,
460
+ deliveryTo: job.delivery?.to ?? "",
461
+ deliveryAccountId: job.delivery?.accountId ?? "",
462
+ deliveryBestEffort: job.delivery?.bestEffort ?? false,
463
+ failureAlertMode:
464
+ failureAlert === false
465
+ ? "disabled"
466
+ : failureAlert && typeof failureAlert === "object"
467
+ ? "custom"
468
+ : "inherit",
469
+ failureAlertAfter:
470
+ failureAlert && typeof failureAlert === "object" && typeof failureAlert.after === "number"
471
+ ? String(failureAlert.after)
472
+ : DEFAULT_CRON_FORM.failureAlertAfter,
473
+ failureAlertCooldownSeconds:
474
+ failureAlert &&
475
+ typeof failureAlert === "object" &&
476
+ typeof failureAlert.cooldownMs === "number"
477
+ ? String(Math.floor(failureAlert.cooldownMs / 1000))
478
+ : DEFAULT_CRON_FORM.failureAlertCooldownSeconds,
479
+ failureAlertChannel:
480
+ failureAlert && typeof failureAlert === "object"
481
+ ? (failureAlert.channel ?? CRON_CHANNEL_LAST)
482
+ : CRON_CHANNEL_LAST,
483
+ failureAlertTo: failureAlert && typeof failureAlert === "object" ? (failureAlert.to ?? "") : "",
484
+ failureAlertDeliveryMode:
485
+ failureAlert && typeof failureAlert === "object"
486
+ ? (failureAlert.mode ?? "announce")
487
+ : "announce",
488
+ failureAlertAccountId:
489
+ failureAlert && typeof failureAlert === "object" ? (failureAlert.accountId ?? "") : "",
490
+ timeoutSeconds:
491
+ job.payload.kind === "agentTurn" && typeof job.payload.timeoutSeconds === "number"
492
+ ? String(job.payload.timeoutSeconds)
493
+ : "",
494
+ };
495
+
496
+ if (job.schedule.kind === "at") {
497
+ next.scheduleAt = formatDateTimeLocal(job.schedule.at);
498
+ } else if (job.schedule.kind === "every") {
499
+ const parsed = parseEverySchedule(job.schedule.everyMs);
500
+ next.everyAmount = parsed.everyAmount;
501
+ next.everyUnit = parsed.everyUnit;
502
+ } else {
503
+ next.cronExpr = job.schedule.expr;
504
+ next.cronTz = job.schedule.tz ?? "";
505
+ const staggerFields = parseStaggerSchedule(job.schedule.staggerMs);
506
+ next.scheduleExact = staggerFields.scheduleExact;
507
+ next.staggerAmount = staggerFields.staggerAmount;
508
+ next.staggerUnit = staggerFields.staggerUnit;
509
+ }
510
+
511
+ return normalizeCronFormState(next);
512
+ }
513
+
514
+ export function buildCronSchedule(form: CronFormState) {
515
+ if (form.scheduleKind === "at") {
516
+ const ms = Date.parse(form.scheduleAt);
517
+ if (!Number.isFinite(ms)) {
518
+ throw new Error(t("cron.errors.invalidRunTime"));
519
+ }
520
+ return { kind: "at" as const, at: new Date(ms).toISOString() };
521
+ }
522
+ if (form.scheduleKind === "every") {
523
+ const amount = toNumber(form.everyAmount, 0);
524
+ if (amount <= 0) {
525
+ throw new Error(t("cron.errors.invalidIntervalAmount"));
526
+ }
527
+ const unit = form.everyUnit;
528
+ const mult = unit === "minutes" ? 60_000 : unit === "hours" ? 3_600_000 : 86_400_000;
529
+ return { kind: "every" as const, everyMs: amount * mult };
530
+ }
531
+ const expr = form.cronExpr.trim();
532
+ if (!expr) {
533
+ throw new Error(t("cron.errors.cronExprRequiredShort"));
534
+ }
535
+ if (form.scheduleExact) {
536
+ return { kind: "cron" as const, expr, tz: form.cronTz.trim() || undefined, staggerMs: 0 };
537
+ }
538
+ const staggerAmount = form.staggerAmount.trim();
539
+ if (!staggerAmount) {
540
+ return { kind: "cron" as const, expr, tz: form.cronTz.trim() || undefined };
541
+ }
542
+ const staggerValue = toNumber(staggerAmount, 0);
543
+ if (staggerValue <= 0) {
544
+ throw new Error(t("cron.errors.invalidStaggerAmount"));
545
+ }
546
+ const staggerMs = form.staggerUnit === "minutes" ? staggerValue * 60_000 : staggerValue * 1_000;
547
+ return { kind: "cron" as const, expr, tz: form.cronTz.trim() || undefined, staggerMs };
548
+ }
549
+
550
+ export function buildCronPayload(form: CronFormState) {
551
+ if (form.payloadKind === "systemEvent") {
552
+ const text = form.payloadText.trim();
553
+ if (!text) {
554
+ throw new Error(t("cron.errors.systemEventTextRequired"));
555
+ }
556
+ return { kind: "systemEvent" as const, text };
557
+ }
558
+ const message = form.payloadText.trim();
559
+ if (!message) {
560
+ throw new Error(t("cron.errors.agentMessageRequiredShort"));
561
+ }
562
+ const payload: {
563
+ kind: "agentTurn";
564
+ message: string;
565
+ model?: string;
566
+ thinking?: string;
567
+ timeoutSeconds?: number;
568
+ lightContext?: boolean;
569
+ } = { kind: "agentTurn", message };
570
+ const model = form.payloadModel.trim();
571
+ if (model) {
572
+ payload.model = model;
573
+ }
574
+ const thinking = form.payloadThinking.trim();
575
+ if (thinking) {
576
+ payload.thinking = thinking;
577
+ }
578
+ const timeoutSeconds = toNumber(form.timeoutSeconds, 0);
579
+ if (timeoutSeconds > 0) {
580
+ payload.timeoutSeconds = timeoutSeconds;
581
+ }
582
+ if (form.payloadLightContext) {
583
+ payload.lightContext = true;
584
+ }
585
+ return payload;
586
+ }
587
+
588
+ function buildFailureAlert(form: CronFormState) {
589
+ if (form.failureAlertMode === "disabled") {
590
+ return false as const;
591
+ }
592
+ if (form.failureAlertMode !== "custom") {
593
+ return undefined;
594
+ }
595
+ const after = toNumber(form.failureAlertAfter.trim(), 0);
596
+ const cooldownRaw = form.failureAlertCooldownSeconds.trim();
597
+ const cooldownSeconds = cooldownRaw.length > 0 ? toNumber(cooldownRaw, 0) : undefined;
598
+ const cooldownMs =
599
+ cooldownSeconds !== undefined && Number.isFinite(cooldownSeconds) && cooldownSeconds >= 0
600
+ ? Math.floor(cooldownSeconds * 1000)
601
+ : undefined;
602
+ const deliveryMode = form.failureAlertDeliveryMode;
603
+ const accountId = form.failureAlertAccountId.trim();
604
+ const patch: Record<string, unknown> = {
605
+ after: after > 0 ? Math.floor(after) : undefined,
606
+ channel: form.failureAlertChannel.trim() || CRON_CHANNEL_LAST,
607
+ to: form.failureAlertTo.trim() || undefined,
608
+ ...(cooldownMs !== undefined ? { cooldownMs } : {}),
609
+ };
610
+ // Always include mode and accountId so users can switch/clear them
611
+ if (deliveryMode) {
612
+ patch.mode = deliveryMode;
613
+ }
614
+ // Include accountId if explicitly set, or send undefined to allow clearing
615
+ patch.accountId = accountId || undefined;
616
+ return patch;
617
+ }
618
+
619
+ export async function addCronJob(state: CronState) {
620
+ if (!state.client || !state.connected || state.cronBusy) {
621
+ return;
622
+ }
623
+ state.cronBusy = true;
624
+ state.cronError = null;
625
+ try {
626
+ const form = normalizeCronFormState(state.cronForm);
627
+ if (form !== state.cronForm) {
628
+ state.cronForm = form;
629
+ }
630
+ const fieldErrors = validateCronForm(form);
631
+ state.cronFieldErrors = fieldErrors;
632
+ if (hasCronFormErrors(fieldErrors)) {
633
+ return;
634
+ }
635
+
636
+ const schedule = buildCronSchedule(form);
637
+ const payload = buildCronPayload(form);
638
+ const editingJob = state.cronEditingJobId
639
+ ? state.cronJobs.find((job) => job.id === state.cronEditingJobId)
640
+ : undefined;
641
+ if (payload.kind === "agentTurn") {
642
+ const existingLightContext =
643
+ editingJob?.payload.kind === "agentTurn" ? editingJob.payload.lightContext : undefined;
644
+ if (
645
+ !form.payloadLightContext &&
646
+ state.cronEditingJobId &&
647
+ existingLightContext !== undefined
648
+ ) {
649
+ payload.lightContext = false;
650
+ }
651
+ }
652
+ const selectedDeliveryMode = form.deliveryMode;
653
+ const delivery =
654
+ selectedDeliveryMode && selectedDeliveryMode !== "none"
655
+ ? {
656
+ mode: selectedDeliveryMode,
657
+ channel:
658
+ selectedDeliveryMode === "announce"
659
+ ? form.deliveryChannel.trim() || "last"
660
+ : undefined,
661
+ to: form.deliveryTo.trim() || undefined,
662
+ accountId:
663
+ selectedDeliveryMode === "announce" ? form.deliveryAccountId.trim() : undefined,
664
+ bestEffort: form.deliveryBestEffort,
665
+ }
666
+ : selectedDeliveryMode === "none"
667
+ ? ({ mode: "none" } as const)
668
+ : undefined;
669
+ const failureAlert = buildFailureAlert(form);
670
+ const agentId = form.clearAgent ? null : form.agentId.trim();
671
+ const sessionKeyRaw = form.sessionKey.trim();
672
+ const sessionKey = sessionKeyRaw || (editingJob?.sessionKey ? null : undefined);
673
+ const job = {
674
+ name: form.name.trim(),
675
+ description: form.description.trim(),
676
+ agentId: agentId === null ? null : agentId || undefined,
677
+ sessionKey,
678
+ enabled: form.enabled,
679
+ deleteAfterRun: form.deleteAfterRun,
680
+ schedule,
681
+ sessionTarget: form.sessionTarget,
682
+ wakeMode: form.wakeMode,
683
+ payload,
684
+ delivery,
685
+ failureAlert,
686
+ };
687
+ if (!job.name) {
688
+ throw new Error(t("cron.errors.nameRequiredShort"));
689
+ }
690
+ if (state.cronEditingJobId) {
691
+ await state.client.request("cron.update", {
692
+ id: state.cronEditingJobId,
693
+ patch: job,
694
+ });
695
+ clearCronEditState(state);
696
+ } else {
697
+ await state.client.request("cron.add", job);
698
+ resetCronFormToDefaults(state);
699
+ }
700
+ await loadCronJobs(state);
701
+ await loadCronStatus(state);
702
+ } catch (err) {
703
+ state.cronError = String(err);
704
+ } finally {
705
+ state.cronBusy = false;
706
+ }
707
+ }
708
+
709
+ export async function toggleCronJob(state: CronState, job: CronJob, enabled: boolean) {
710
+ if (!state.client || !state.connected || state.cronBusy) {
711
+ return;
712
+ }
713
+ state.cronBusy = true;
714
+ state.cronError = null;
715
+ try {
716
+ await state.client.request("cron.update", { id: job.id, patch: { enabled } });
717
+ await loadCronJobs(state);
718
+ await loadCronStatus(state);
719
+ } catch (err) {
720
+ state.cronError = String(err);
721
+ } finally {
722
+ state.cronBusy = false;
723
+ }
724
+ }
725
+
726
+ export async function runCronJob(state: CronState, job: CronJob, mode: "force" | "due" = "force") {
727
+ if (!state.client || !state.connected || state.cronBusy) {
728
+ return;
729
+ }
730
+ state.cronBusy = true;
731
+ state.cronError = null;
732
+ try {
733
+ await state.client.request("cron.run", { id: job.id, mode });
734
+ if (state.cronRunsScope === "all") {
735
+ await loadCronRuns(state, null);
736
+ } else {
737
+ await loadCronRuns(state, job.id);
738
+ }
739
+ } catch (err) {
740
+ state.cronError = String(err);
741
+ } finally {
742
+ state.cronBusy = false;
743
+ }
744
+ }
745
+
746
+ export async function removeCronJob(state: CronState, job: CronJob) {
747
+ if (!state.client || !state.connected || state.cronBusy) {
748
+ return;
749
+ }
750
+ state.cronBusy = true;
751
+ state.cronError = null;
752
+ try {
753
+ await state.client.request("cron.remove", { id: job.id });
754
+ if (state.cronEditingJobId === job.id) {
755
+ clearCronEditState(state);
756
+ }
757
+ if (state.cronRunsJobId === job.id) {
758
+ state.cronRunsJobId = null;
759
+ state.cronRuns = [];
760
+ state.cronRunsTotal = 0;
761
+ state.cronRunsHasMore = false;
762
+ state.cronRunsNextOffset = null;
763
+ }
764
+ await loadCronJobs(state);
765
+ await loadCronStatus(state);
766
+ } catch (err) {
767
+ state.cronError = String(err);
768
+ } finally {
769
+ state.cronBusy = false;
770
+ }
771
+ }
772
+
773
+ export async function loadCronRuns(
774
+ state: CronState,
775
+ jobId: string | null,
776
+ opts?: { append?: boolean },
777
+ ) {
778
+ if (!state.client || !state.connected) {
779
+ return;
780
+ }
781
+ const scope = state.cronRunsScope;
782
+ const activeJobId = jobId ?? state.cronRunsJobId;
783
+ if (scope === "job" && !activeJobId) {
784
+ state.cronRuns = [];
785
+ state.cronRunsTotal = 0;
786
+ state.cronRunsHasMore = false;
787
+ state.cronRunsNextOffset = null;
788
+ return;
789
+ }
790
+ const append = opts?.append === true;
791
+ if (append && !state.cronRunsHasMore) {
792
+ return;
793
+ }
794
+ try {
795
+ if (append) {
796
+ state.cronRunsLoadingMore = true;
797
+ }
798
+ const offset = append ? Math.max(0, state.cronRunsNextOffset ?? state.cronRuns.length) : 0;
799
+ const res = await state.client.request<CronRunsResult>("cron.runs", {
800
+ scope,
801
+ id: scope === "job" ? (activeJobId ?? undefined) : undefined,
802
+ limit: state.cronRunsLimit,
803
+ offset,
804
+ statuses: state.cronRunsStatuses.length > 0 ? state.cronRunsStatuses : undefined,
805
+ status: state.cronRunsStatusFilter,
806
+ deliveryStatuses:
807
+ state.cronRunsDeliveryStatuses.length > 0 ? state.cronRunsDeliveryStatuses : undefined,
808
+ query: state.cronRunsQuery.trim() || undefined,
809
+ sortDir: state.cronRunsSortDir,
810
+ });
811
+ const entries = Array.isArray(res.entries) ? res.entries : [];
812
+ state.cronRuns =
813
+ append && (scope === "all" || state.cronRunsJobId === activeJobId)
814
+ ? [...state.cronRuns, ...entries]
815
+ : entries;
816
+ if (scope === "job") {
817
+ state.cronRunsJobId = activeJobId ?? null;
818
+ }
819
+ const meta = normalizeCronPageMeta({
820
+ totalRaw: res.total,
821
+ limitRaw: res.limit,
822
+ offsetRaw: res.offset,
823
+ nextOffsetRaw: res.nextOffset,
824
+ hasMoreRaw: res.hasMore,
825
+ pageCount: entries.length,
826
+ });
827
+ state.cronRunsTotal = Math.max(meta.total, state.cronRuns.length);
828
+ state.cronRunsHasMore = meta.hasMore;
829
+ state.cronRunsNextOffset = meta.nextOffset;
830
+ } catch (err) {
831
+ state.cronError = String(err);
832
+ } finally {
833
+ if (append) {
834
+ state.cronRunsLoadingMore = false;
835
+ }
836
+ }
837
+ }
838
+
839
+ export async function loadMoreCronRuns(state: CronState) {
840
+ if (state.cronRunsScope === "job" && !state.cronRunsJobId) {
841
+ return;
842
+ }
843
+ await loadCronRuns(state, state.cronRunsJobId, { append: true });
844
+ }
845
+
846
+ export function updateCronRunsFilter(
847
+ state: CronState,
848
+ patch: Partial<
849
+ Pick<
850
+ CronState,
851
+ | "cronRunsScope"
852
+ | "cronRunsStatuses"
853
+ | "cronRunsDeliveryStatuses"
854
+ | "cronRunsStatusFilter"
855
+ | "cronRunsQuery"
856
+ | "cronRunsSortDir"
857
+ >
858
+ >,
859
+ ) {
860
+ if (patch.cronRunsScope) {
861
+ state.cronRunsScope = patch.cronRunsScope;
862
+ }
863
+ if (Array.isArray(patch.cronRunsStatuses)) {
864
+ state.cronRunsStatuses = patch.cronRunsStatuses;
865
+ state.cronRunsStatusFilter =
866
+ patch.cronRunsStatuses.length === 1 ? patch.cronRunsStatuses[0] : "all";
867
+ }
868
+ if (Array.isArray(patch.cronRunsDeliveryStatuses)) {
869
+ state.cronRunsDeliveryStatuses = patch.cronRunsDeliveryStatuses;
870
+ }
871
+ if (patch.cronRunsStatusFilter) {
872
+ state.cronRunsStatusFilter = patch.cronRunsStatusFilter;
873
+ state.cronRunsStatuses =
874
+ patch.cronRunsStatusFilter === "all" ? [] : [patch.cronRunsStatusFilter];
875
+ }
876
+ if (typeof patch.cronRunsQuery === "string") {
877
+ state.cronRunsQuery = patch.cronRunsQuery;
878
+ }
879
+ if (patch.cronRunsSortDir) {
880
+ state.cronRunsSortDir = patch.cronRunsSortDir;
881
+ }
882
+ }
883
+
884
+ export function startCronEdit(state: CronState, job: CronJob) {
885
+ state.cronEditingJobId = job.id;
886
+ state.cronRunsJobId = job.id;
887
+ state.cronForm = jobToForm(job, state.cronForm);
888
+ state.cronFieldErrors = validateCronForm(state.cronForm);
889
+ }
890
+
891
+ function buildCloneName(name: string, existingNames: Set<string>) {
892
+ const base = name.trim() || "Job";
893
+ const first = `${base} copy`;
894
+ if (!existingNames.has(first.toLowerCase())) {
895
+ return first;
896
+ }
897
+ let index = 2;
898
+ while (index < 1000) {
899
+ const next = `${base} copy ${index}`;
900
+ if (!existingNames.has(next.toLowerCase())) {
901
+ return next;
902
+ }
903
+ index += 1;
904
+ }
905
+ return `${base} copy ${Date.now()}`;
906
+ }
907
+
908
+ export function startCronClone(state: CronState, job: CronJob) {
909
+ clearCronEditState(state);
910
+ state.cronRunsJobId = job.id;
911
+ const existingNames = new Set(state.cronJobs.map((entry) => entry.name.trim().toLowerCase()));
912
+ const cloned = jobToForm(job, state.cronForm);
913
+ cloned.name = buildCloneName(job.name, existingNames);
914
+ state.cronForm = cloned;
915
+ state.cronFieldErrors = validateCronForm(state.cronForm);
916
+ }
917
+
918
+ export function cancelCronEdit(state: CronState) {
919
+ clearCronEditState(state);
920
+ resetCronFormToDefaults(state);
921
+ }