openclaw-multi-auto 1.0.9 → 1.1.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 (490) hide show
  1. package/dist/{accounts-C_lW3Ag9.js → accounts-BOzyfwW4.js} +2 -2
  2. package/dist/{accounts-BVgYdU9W.js → accounts-BtEMxtPK.js} +1 -1
  3. package/dist/{accounts-BU-CeDai.js → accounts-CUYZBSKh.js} +1 -1
  4. package/dist/{accounts-C9HcPI9h.js → accounts-L9ByEpnP.js} +2 -2
  5. package/dist/{accounts-DJaYqD2E.js → accounts-khXX75l1.js} +7 -7
  6. package/dist/{accounts-Tgelvk0C.js → accounts-yfBeCZtS.js} +17 -17
  7. package/dist/{acp-cli-oWFHnS7i.js → acp-cli-CA2oCCEA.js} +8 -8
  8. package/dist/{active-listener-BEdprTkn.js → active-listener-D1yqT1cw.js} +2 -2
  9. package/dist/{agent-scope-BRElciAf.js → agent-scope-NQ9CtXYN.js} +17 -17
  10. package/dist/{agents-CujsWz9d.js → agents-BWVlofyv.js} +14 -14
  11. package/dist/{agents.config-C42STger.js → agents.config-QVfqc4C-.js} +2 -2
  12. package/dist/{api-key-rotation-BJpKWXy0.js → api-key-rotation-DtsNS2Nb.js} +2 -2
  13. package/dist/{api-key-rotation-BbuLHl0_.js → api-key-rotation-Vix80AUw.js} +1 -1
  14. package/dist/{audio-preflight-Yg1vzPE1.js → audio-preflight-DUtdCXjJ.js} +34 -34
  15. package/dist/{audio-preflight-C58LeYZM.js → audio-preflight-DpxQCpsA.js} +32 -32
  16. package/dist/{audio-transcription-runner-B8jbozH5.js → audio-transcription-runner-28fcRNNi.js} +12 -12
  17. package/dist/{audio-transcription-runner-D1cvrZ6s.js → audio-transcription-runner-DYKvqK54.js} +23 -23
  18. package/dist/{audit-membership-runtime-Dntemq07.js → audit-membership-runtime-DWyHWAHM.js} +4 -4
  19. package/dist/{audit-DtlAv66L.js → audit-oC--RZy0.js} +29 -29
  20. package/dist/{auth-Mj21c_GN.js → auth-YB6m93-M.js} +1 -1
  21. package/dist/{auth-choice-CEycltU4.js → auth-choice-CsSQLEkP.js} +13 -13
  22. package/dist/{auth-choice-CGHVedXa.js → auth-choice-gp-h1aBd.js} +11 -11
  23. package/dist/{auth-choice.apply-helpers-D4YQXYfc.js → auth-choice.apply-helpers-CPpuynsT.js} +1 -1
  24. package/dist/{auth-profiles-CdLTlJLc.js → auth-profiles-15pq9j9V.js} +16 -16
  25. package/dist/{auth-token-7kzDLVhb.js → auth-token-Bwn8N6KA.js} +1 -1
  26. package/dist/{banner-CpzKVd4-.js → banner-hcZ0XNXv.js} +2 -2
  27. package/dist/{bonjour-discovery-BuS9wF_p.js → bonjour-discovery-2btw06uD.js} +1 -1
  28. package/dist/{browser-cli-BVtbvQ3Z.js → browser-cli-C4PdLYHy.js} +12 -12
  29. package/dist/build-info.json +3 -3
  30. package/dist/bundled/boot-md/handler.js +51 -51
  31. package/dist/bundled/bootstrap-extra-files/handler.js +6 -6
  32. package/dist/bundled/command-logger/handler.js +2 -2
  33. package/dist/bundled/session-memory/handler.js +51 -51
  34. package/dist/{call-Dp9SDkqK.js → call-CZgRbVYm.js} +10 -10
  35. package/dist/canvas-host/a2ui/.bundle.hash +1 -1
  36. package/dist/{channel-account-context-ESLVGdUf.js → channel-account-context-BYiOuxnf.js} +5 -5
  37. package/dist/{channel-activity-LfmEkdDN.js → channel-activity-D9K6yeu8.js} +1 -1
  38. package/dist/{channel-activity-BDnjYF7B.js → channel-activity-xHOMiarp.js} +3 -3
  39. package/dist/{channel-options-CsYaFB_4.js → channel-options-DJkf_8wi.js} +3 -3
  40. package/dist/{channel-selection-DFJsYFPo.js → channel-selection-BpIsCVV5.js} +1 -1
  41. package/dist/{channel-web-DzRsJEE4.js → channel-web-wb1My1Yi.js} +17 -17
  42. package/dist/{channels-cli-CpfeOXtJ.js → channels-cli-DlQONtY9.js} +93 -93
  43. package/dist/{channels-status-issues-6V9ktlZD.js → channels-status-issues-Hj-GOSVm.js} +1 -1
  44. package/dist/{chrome-CRBG2YP_.js → chrome-DwizpzOC.js} +26 -26
  45. package/dist/{chrome-CyM61Cn2.js → chrome-L_icBVLq.js} +4 -4
  46. package/dist/{clawbot-cli-COtnakIJ.js → clawbot-cli-CyTZUVIJ.js} +11 -11
  47. package/dist/cli/daemon-cli.js +1 -1
  48. package/dist/{cli-jnT1hYPO.js → cli-CNTrB-ni.js} +73 -73
  49. package/dist/{client-C-wA75vx.js → client-g0366Z73.js} +2 -2
  50. package/dist/{command-registry-g5tDqdAU.js → command-registry-VOmumlmD.js} +11 -11
  51. package/dist/{command-secret-targets-u3eWn-W3.js → command-secret-targets-CIMuIdbL.js} +4 -4
  52. package/dist/{commands-C8pWcFs_.js → commands-DU7At3rY.js} +1 -1
  53. package/dist/{commands-registry-J6nVL3BQ.js → commands-registry-B_WUa9KB.js} +3 -3
  54. package/dist/{commands-registry-D5qXbFJn.js → commands-registry-V1zZ5pPC.js} +4 -4
  55. package/dist/{completion-cli-DQsyli3B.js → completion-cli-pqdmKVTo.js} +13 -13
  56. package/dist/{config-cli-DUycVHoC.js → config-cli-BqznmzlM.js} +7 -7
  57. package/dist/{config-guard--Obk2MGi.js → config-guard-D8jiLjr7.js} +3 -3
  58. package/dist/{config-validation-iOxszdEd.js → config-validation-BngLiGoq.js} +3 -3
  59. package/dist/{configure-Al_pz_5z.js → configure-B--Umf_O.js} +17 -17
  60. package/dist/{control-ui-assets-DpjfNcTC.js → control-ui-assets-C9ewbI8z.js} +1 -1
  61. package/dist/{cron-cli-CQ5EZw_b.js → cron-cli-DCRPpI1m.js} +11 -11
  62. package/dist/{daemon-cli-Bb80Mzv5.js → daemon-cli-DtBzKM_d.js} +15 -15
  63. package/dist/{daemon-install-B09ifLcJ.js → daemon-install-ckPBUAjJ.js} +4 -4
  64. package/dist/{daemon-install-helpers-CYpECiEx.js → daemon-install-helpers-BWf-PrgA.js} +11 -11
  65. package/dist/{deliver-ChbFOwjw.js → deliver-D4o6VIur.js} +21 -21
  66. package/dist/{deliver-CxF9h9PE.js → deliver-b93aBgie.js} +7 -7
  67. package/dist/deliver-runtime-CdWd-JQI.js +61 -0
  68. package/dist/deliver-runtime-P-G3bPjW.js +36 -0
  69. package/dist/deps-send-discord.runtime-BltWu9GW.js +36 -0
  70. package/dist/deps-send-discord.runtime-DnbhTFX9.js +26 -0
  71. package/dist/deps-send-imessage.runtime-BOiQ6mDx.js +25 -0
  72. package/dist/deps-send-imessage.runtime-CdCif3t7.js +35 -0
  73. package/dist/deps-send-signal.runtime-CTcl388M.js +24 -0
  74. package/dist/deps-send-signal.runtime-fHqkSmMW.js +34 -0
  75. package/dist/deps-send-slack.runtime-CCqBz4Kg.js +22 -0
  76. package/dist/deps-send-slack.runtime-CkyZO7ln.js +32 -0
  77. package/dist/{deps-send-telegram.runtime-B6wp3kx8.js → deps-send-telegram.runtime-C--B69CN.js} +16 -16
  78. package/dist/deps-send-telegram.runtime-DGSKTCpH.js +27 -0
  79. package/dist/deps-send-whatsapp.runtime-BkoMLlCM.js +119 -0
  80. package/dist/deps-send-whatsapp.runtime-CJkTHkah.js +60 -0
  81. package/dist/{devices-cli-C0dUbeL4.js → devices-cli-D6V059Dh.js} +8 -8
  82. package/dist/{diagnostic-Co6Kghr-.js → diagnostic-Bn4PZjMZ.js} +2 -2
  83. package/dist/{diagnostic-BfiudAAN.js → diagnostic-BpP9UBCE.js} +1 -1
  84. package/dist/{diagnostics-DCWM_8Ur.js → diagnostics---0c2_jo.js} +5 -5
  85. package/dist/{directory-cli-Dlws13Di.js → directory-cli-RYC34-EU.js} +7 -7
  86. package/dist/{dns-cli-But7QdAw.js → dns-cli-CElNoV5S.js} +5 -5
  87. package/dist/{dock-BbUkruOF.js → dock-nA2DVPCV.js} +4 -4
  88. package/dist/{docs-cli-Ck2IKAtw.js → docs-cli-Bv7ltPvi.js} +4 -4
  89. package/dist/{doctor-completion-Nat1HXE1.js → doctor-completion-CQIiN7rY.js} +2 -2
  90. package/dist/{doctor-config-flow-WCCoUXeB.js → doctor-config-flow-BOAryh5P.js} +15 -15
  91. package/dist/{enable-C3H8BtN4.js → enable-688HYBNS.js} +1 -1
  92. package/dist/entry.js +2 -2
  93. package/dist/{errors-xt401nuk.js → errors-CCLeFWAg.js} +1 -1
  94. package/dist/{exec-approvals-allowlist-ySf2Yo5n.js → exec-approvals-allowlist-C0Eya3rT.js} +1 -1
  95. package/dist/{exec-approvals-cli-DhLLOys6.js → exec-approvals-cli-BYYerKp8.js} +16 -16
  96. package/dist/{exec-safe-bin-runtime-policy-DnXViOlF.js → exec-safe-bin-runtime-policy-BmJmfDRo.js} +2 -2
  97. package/dist/{fetch-D-SiVqBm.js → fetch-BhAHVdwx.js} +3 -3
  98. package/dist/{fetch-DuraYswo.js → fetch-BlJWzEP6.js} +5 -5
  99. package/dist/{fetch-guard-CKYBfJ66.js → fetch-guard-BbBdhWz_.js} +1 -1
  100. package/dist/{fetch-guard-DWr0d00H.js → fetch-guard-ChYBwfiy.js} +2 -2
  101. package/dist/{frontmatter-BkTfEZ93.js → frontmatter-CvaMP376.js} +3 -3
  102. package/dist/{fs-safe-CTYUrIgQ.js → fs-safe-0jAo_Whb.js} +4 -4
  103. package/dist/{fs-safe-jMDpsYew.js → fs-safe-CFLs-j60.js} +24 -24
  104. package/dist/{gateway-cli--atF6LYo.js → gateway-cli-DOYWSRqa.js} +153 -153
  105. package/dist/{gateway-rpc-DYyQQ1z9.js → gateway-rpc-w6t8Eq_1.js} +1 -1
  106. package/dist/{github-copilot-token-BDioPmd6.js → github-copilot-token-D13V9YBz.js} +7 -7
  107. package/dist/{health-BDJ72U4Z.js → health-C1EstYHd.js} +11 -11
  108. package/dist/{hooks-cli-r0UX8B9a.js → hooks-cli-BFMve5sG.js} +81 -81
  109. package/dist/{hooks-status-CxBdpBry.js → hooks-status-DBnEWOEP.js} +1 -1
  110. package/dist/{image-C7RmnYxa.js → image-Bbn53mzj.js} +6 -6
  111. package/dist/{image-CyHTO86Q.js → image-Bw71y73Q.js} +5 -5
  112. package/dist/{image-ops-pjs5W0CZ.js → image-ops-AJL9tN3_.js} +10 -10
  113. package/dist/{image-ops-BuUnEOE0.js → image-ops-CehkHxmW.js} +2 -2
  114. package/dist/image-runtime-6jhrqcle.js +55 -0
  115. package/dist/image-runtime-CVv2ra9J.js +29 -0
  116. package/dist/{inspect-BeU4yMp2.js → inspect-CUxeDA8c.js} +4 -4
  117. package/dist/{install-safe-path-BiL8OJvK.js → install-safe-path-BX58wFBl.js} +25 -25
  118. package/dist/{installs-BwKmG0Uy.js → installs-B-xr0Fzq.js} +9 -9
  119. package/dist/{ipv4-BoGwfnEw.js → ipv4-C4Yt-xid.js} +1 -1
  120. package/dist/{ir-B83looB-.js → ir-DAP-B-Xw.js} +8 -8
  121. package/dist/{ir-Da7Ahsqc.js → ir-otKVkb4a.js} +8 -8
  122. package/dist/{issue-format-CJ89_-9v.js → issue-format-DSksfKiV.js} +1 -1
  123. package/dist/{json-files-CuJjdF_0.js → json-files-rR19q30D.js} +8 -8
  124. package/dist/{legacy-names-DOC03BkU.js → legacy-names-TyzbVqa_.js} +1 -1
  125. package/dist/{lifecycle-core-BzuWBBm8.js → lifecycle-core-DVwen0ks.js} +5 -5
  126. package/dist/llm-slug-generator.js +51 -51
  127. package/dist/{logger-BfjWMCSD.js → logger-DMZQQtxK.js} +7 -7
  128. package/dist/{login-VkQ9MW3d.js → login-BcRaGQb6.js} +3 -3
  129. package/dist/{login-CrIwcrVI.js → login-DiCctRo1.js} +5 -5
  130. package/dist/{login-qr-BVMQ-xQz.js → login-qr-C9D47WEZ.js} +6 -6
  131. package/dist/{login-qr-BpPDZdl_.js → login-qr-MUbXgjtd.js} +10 -10
  132. package/dist/{logs-cli-DWHhSnWs.js → logs-cli-B3TG2lp8.js} +9 -9
  133. package/dist/{manager-1bvuGrNR.js → manager-BW_NSIMl.js} +13 -13
  134. package/dist/{manager-BUBwl6M6.js → manager-iwcRf3Xc.js} +14 -14
  135. package/dist/manager-runtime-BN6VevdC.js +18 -0
  136. package/dist/{manager-runtime-H9iQnw64.js → manager-runtime-D3KqpRT7.js} +9 -9
  137. package/dist/{manifest-registry-DVviqWVY.js → manifest-registry-CJb8odRF.js} +1 -1
  138. package/dist/{memory-cli-DarWCbU6.js → memory-cli-DGsJx7YM.js} +12 -12
  139. package/dist/{model-BLV8y_N-.js → model-B792l3Cn.js} +2 -2
  140. package/dist/{model-catalog-DTcBkUlX.js → model-catalog-DZ6kl2Ko.js} +3 -3
  141. package/dist/{model-picker-D-fTOOzb.js → model-picker-6fHys0wK.js} +4 -4
  142. package/dist/{model-selection-Dna0Gz1k.js → model-selection-idoqPmw0.js} +43 -43
  143. package/dist/{models-CmjutMmh.js → models-DerrMmwF.js} +17 -17
  144. package/dist/{models-cli-BZFQX22p.js → models-cli-BkTZg_4v.js} +78 -78
  145. package/dist/{models-config-DNcDbV_G.js → models-config-Dabs9Kdv.js} +6 -6
  146. package/dist/{net-D5fSREu4.js → net-BAIqYNz0.js} +2 -2
  147. package/dist/{node-cli-Cnq-PJTp.js → node-cli-DoEq4zeG.js} +33 -33
  148. package/dist/{node-command-policy-B5BMBBJP.js → node-command-policy-DRhSc90G.js} +1 -1
  149. package/dist/{node-service-BpYZAvpl.js → node-service-CcgtNIeT.js} +1 -1
  150. package/dist/{nodes-cli-CkrDOWpv.js → nodes-cli--d9RiCnK.js} +16 -16
  151. package/dist/{nodes-screen-DfsQohWd.js → nodes-screen-BTND5VDq.js} +7 -7
  152. package/dist/{npm-pack-install-E-mkrZ55.js → npm-pack-install-CGUDOSKz.js} +18 -18
  153. package/dist/{npm-resolution-BUUmg5ON.js → npm-resolution-CAoULCWM.js} +4 -4
  154. package/dist/{onboard-DINGSS9R.js → onboard-CMZRkPVc.js} +6 -6
  155. package/dist/{onboard-channels-Di4MjWDY.js → onboard-channels-SlJu6XWJ.js} +21 -21
  156. package/dist/{onboard-custom-DITBQuWQ.js → onboard-custom-EzfL89yz.js} +4 -4
  157. package/dist/{onboard-helpers-DO7u2ZOy.js → onboard-helpers-lGpaVp03.js} +10 -10
  158. package/dist/{onboard-hooks-K02_ZWDq.js → onboard-hooks-B7TMiJoY.js} +4 -4
  159. package/dist/{onboard-remote-5-ugiSN2.js → onboard-remote-nvWS7on7.js} +4 -4
  160. package/dist/{onboard-skills-CSzbC2KZ.js → onboard-skills-hz91W03u.js} +4 -4
  161. package/dist/{onboarding-BM4dvUK6.js → onboarding-CTPuMVQI.js} +14 -14
  162. package/dist/{onboarding.finalize-p_RDh7ET.js → onboarding.finalize-DLzl5ZOg.js} +87 -87
  163. package/dist/{onboarding.gateway-config-DldlRfms.js → onboarding.gateway-config-C4MoTr60.js} +18 -18
  164. package/dist/{onboarding.secret-input-BIRIJiCU.js → onboarding.secret-input-BFD0OW4X.js} +1 -1
  165. package/dist/{openai-model-default-CY2Nk4cn.js → openai-model-default-h4LbSR7f.js} +2 -2
  166. package/dist/{openclaw-root-BFfBQ6FD.js → openclaw-root-BU3lu8pM.js} +8 -8
  167. package/dist/{outbound-ChDjtuD6.js → outbound-C2kanETZ.js} +6 -6
  168. package/dist/{outbound-attachment-DfWsfe2N.js → outbound-attachment-CAJBGcna.js} +2 -2
  169. package/dist/{outbound-attachment-DqHlD21U.js → outbound-attachment-DBrYWX8h.js} +2 -2
  170. package/dist/{outbound-DPdJe7e1.js → outbound-kbHYt1JW.js} +3 -3
  171. package/dist/{pairing-cli-D2VZxWg3.js → pairing-cli-CXbmkz7Z.js} +8 -8
  172. package/dist/{pairing-labels-D4rnJ5pJ.js → pairing-labels-CaPLIhlh.js} +1 -1
  173. package/dist/{pairing-store-BXArq4hn.js → pairing-store-Brs9aNn_.js} +3 -3
  174. package/dist/{path-alias-guards-DbNvNQar.js → path-alias-guards-DhIwq92y.js} +3 -3
  175. package/dist/{path-alias-guards-BzvdLvTI.js → path-alias-guards-DqXRZmsL.js} +1 -1
  176. package/dist/{path-safety-ClQO4BB6.js → path-safety-BjIM4N4t.js} +1 -1
  177. package/dist/{paths-Cvc9EM8Y.js → paths-C6TxBCvO.js} +5 -5
  178. package/dist/{paths-Bkr-BCxW.js → paths-CCxysrzL.js} +4 -4
  179. package/dist/{paths-BB_1ZWOj.js → paths-Cv63xST_.js} +9 -9
  180. package/dist/{pi-embedded-BC_GWGuw.js → pi-embedded-BaGj07T0.js} +167 -167
  181. package/dist/{pi-embedded-helpers-oXDyXTD8.js → pi-embedded-helpers-DC2OtKrl.js} +6 -6
  182. package/dist/{pi-embedded-helpers-CbIShbOM.js → pi-embedded-helpers-wy0DZvx1.js} +52 -52
  183. package/dist/{pi-model-discovery-Dymwdjt0.js → pi-model-discovery-BGgOlX8N.js} +7 -7
  184. package/dist/{pi-model-discovery-CECkJMCt.js → pi-model-discovery-DOb5RTev.js} +1 -1
  185. package/dist/pi-model-discovery-runtime-Bwmi4Ev8.js +11 -0
  186. package/dist/{pi-model-discovery-runtime-g7UP-SFR.js → pi-model-discovery-runtime-DShjmiiF.js} +5 -5
  187. package/dist/{pi-tools.before-tool-call.runtime-Df7B-ggW.js → pi-tools.before-tool-call.runtime-B079pVah.js} +5 -5
  188. package/dist/{pi-tools.before-tool-call.runtime-Cwab_5W1.js → pi-tools.before-tool-call.runtime-BuLxSyx9.js} +9 -9
  189. package/dist/{pi-tools.policy-CydUEzFi.js → pi-tools.policy-DZ-X86Va.js} +5 -5
  190. package/dist/{plugin-auto-enable-BE4ZVjAL.js → plugin-auto-enable-BCMqEDjQ.js} +3 -3
  191. package/dist/{plugin-registry-DXW3eyib.js → plugin-registry-BPlCWMur.js} +3 -3
  192. package/dist/plugin-sdk/discord.js +6 -6
  193. package/dist/{plugins-4Rj4OjLY.js → plugins-CWkRQYDj.js} +11 -11
  194. package/dist/{plugins-VAZrrfgw.js → plugins-DCxT-37x.js} +2 -2
  195. package/dist/{plugins-cli-BQJOOVMx.js → plugins-cli-D9ILEfyb.js} +83 -83
  196. package/dist/{ports-DMkRSlnH.js → ports-BlCLhwbc.js} +1 -1
  197. package/dist/{ports-DogAV7pa.js → ports-CER5YPnN.js} +2 -2
  198. package/dist/{program-BA11qFx1.js → program-CV56xn7w.js} +81 -81
  199. package/dist/{prompt-select-styled-B0GS28ia.js → prompt-select-styled-BOa5I_PU.js} +40 -40
  200. package/dist/{provider-auth-helpers-CrGL-jik.js → provider-auth-helpers-c6mNCUXO.js} +5 -5
  201. package/dist/{proxy-env-wKO3g8Yv.js → proxy-env-BMrSVckF.js} +1 -1
  202. package/dist/{proxy-env-DlmzDx8x.js → proxy-env-Cq5gdrbj.js} +1 -1
  203. package/dist/{proxy-fetch-B2pEfjbR.js → proxy-fetch-CCjEYbFm.js} +1 -1
  204. package/dist/{push-apns-BEwBjZ0a.js → push-apns-FOkPD05E.js} +5 -5
  205. package/dist/{pw-ai-D2wlMJtN.js → pw-ai-Cl1Lc7RC.js} +14 -14
  206. package/dist/{pw-ai-CnbxziFP.js → pw-ai-CxBU3aK5.js} +18 -18
  207. package/dist/{qmd-manager-BtIKUaO9.js → qmd-manager-BsYsO9Ii.js} +10 -10
  208. package/dist/{qmd-manager-Dp6PJ8zQ.js → qmd-manager-DEJMqoGd.js} +20 -20
  209. package/dist/{qr-cli-CFz9kS5X.js → qr-cli-BD2jK4fg.js} +2 -2
  210. package/dist/{query-expansion-BrSWVbaE.js → query-expansion-1UTIWjP6.js} +12 -12
  211. package/dist/{query-expansion-CX-1fS52.js → query-expansion-DtLc3wjL.js} +6 -6
  212. package/dist/{redact-COik8ET1.js → redact-ClbcYG1J.js} +1 -1
  213. package/dist/{redact-snapshot-Bs4goggz.js → redact-snapshot-D_qQD4A-.js} +1 -1
  214. package/dist/{register.agent-D641ju8B.js → register.agent-tbPA5YAy.js} +94 -94
  215. package/dist/register.configure-DuwRrXc2.js +165 -0
  216. package/dist/{register.maintenance-BFkk8MEH.js → register.maintenance-DrqDlV5b.js} +95 -95
  217. package/dist/{register.message-D5uE_Hop.js → register.message-CoKXNaU0.js} +74 -74
  218. package/dist/{register.onboard-zHFvSwFr.js → register.onboard-CSWOSL9O.js} +18 -18
  219. package/dist/{register.setup-uTbv3_P1.js → register.setup-DBx5JX6h.js} +21 -21
  220. package/dist/{register.status-health-sessions-CB7t-JQx.js → register.status-health-sessions-D1bPtfep.js} +88 -88
  221. package/dist/{register.subclis-MEiNmuy5.js → register.subclis-QGJNmjss.js} +31 -31
  222. package/dist/{rpc-DVfuVmy9.js → rpc-DkMrTUww.js} +1 -1
  223. package/dist/{run-main-mrvunzuy.js → run-main-CPftxqTe.js} +92 -92
  224. package/dist/{run-with-concurrency-BgYfgkXT.js → run-with-concurrency-D_ZpbgEG.js} +4 -4
  225. package/dist/{runtime-C87FQrrv.js → runtime-MAH2Oph4.js} +3 -3
  226. package/dist/{runtime-config-collectors-BQaC477D.js → runtime-config-collectors-BJMV6Mt1.js} +1 -1
  227. package/dist/{runtime-whatsapp-login.runtime-DjdgScUI.js → runtime-whatsapp-login.runtime-CGYWl9eB.js} +7 -7
  228. package/dist/runtime-whatsapp-login.runtime-IeylZEl4.js +13 -0
  229. package/dist/runtime-whatsapp-outbound.runtime-ClBRuLsq.js +22 -0
  230. package/dist/{runtime-whatsapp-outbound.runtime-B74K7opl.js → runtime-whatsapp-outbound.runtime-hZEfYaRM.js} +15 -15
  231. package/dist/{sandbox-Ct-_lzi1.js → sandbox-0TbzPJaS.js} +18 -18
  232. package/dist/{sandbox-cli-C3RoXFRE.js → sandbox-cli-NLaxffXl.js} +25 -25
  233. package/dist/{secrets-cli-B3hqCxvs.js → secrets-cli-KLGbYet6.js} +11 -11
  234. package/dist/{security-cli-DcHIyXr7.js → security-cli-EIEkcYVb.js} +42 -42
  235. package/dist/{send-dfu6_rgf.js → send-BQERFNyo.js} +5 -5
  236. package/dist/{send-CnRP4P-G.js → send-BW-ZtYG3.js} +5 -5
  237. package/dist/{send-L7gRiwyd.js → send-Bj776ESJ.js} +7 -7
  238. package/dist/{send-2zKwf9NW.js → send-BtZAqquW.js} +11 -11
  239. package/dist/{send-DAQAKa9Z.js → send-CxgWxXZc.js} +6 -6
  240. package/dist/{send-5o2p_xjn.js → send-DAMtu9kK.js} +4 -4
  241. package/dist/{send-PE6cwoTe.js → send-DcxmcFi_.js} +8 -8
  242. package/dist/{send-6lz6rNVP.js → send-Dx2RkUOZ.js} +6 -6
  243. package/dist/{send-u1Bo4CSn.js → send-sC6ka831.js} +8 -8
  244. package/dist/{send-BHTiZcH3.js → send-vmONuVgL.js} +26 -26
  245. package/dist/{server-context-8pDe2iyd.js → server-context-HJVwPQYn.js} +12 -12
  246. package/dist/{server-Ci4xtuR9.js → server-kUElNhlY.js} +20 -20
  247. package/dist/{server-lifecycle-k5daSrde.js → server-lifecycle-BE32unpZ.js} +2 -2
  248. package/dist/{server-middleware-DMiFT9xU.js → server-middleware-AS2VOYkK.js} +1 -1
  249. package/dist/{server-node-events-BmnPjNXE.js → server-node-events-DkbZzI6P.js} +74 -74
  250. package/dist/{service-BhOFtHSw.js → service-CyStNr3d.js} +15 -15
  251. package/dist/{session-D8ImowSs.js → session-A4QhBRvH.js} +8 -8
  252. package/dist/{session-DkOjpX3_.js → session-CaCx4rPH.js} +1 -1
  253. package/dist/{session-utils-DKRmXD2l.js → session-utils-CGqb1oeq.js} +6 -6
  254. package/dist/{sessions-svLGrv0Z.js → sessions-BV6HNW4h.js} +4 -4
  255. package/dist/{sessions-BOWPuhe5.js → sessions-BmVDW-7q.js} +15 -15
  256. package/dist/{shared-CjuadLFV.js → shared-DUQavBtY.js} +3 -3
  257. package/dist/{shared-CbAkLNrg.js → shared-JW74idb0.js} +1 -1
  258. package/dist/{skill-commands-DNqJ-kwn.js → skill-commands-CMzBZKG2.js} +9 -9
  259. package/dist/{skill-commands-TpUsdjev.js → skill-commands-WtIJG0CI.js} +5 -5
  260. package/dist/{skill-scanner-BZvvItef.js → skill-scanner-DIFsGcqE.js} +6 -6
  261. package/dist/{skills-DR-vacol.js → skills-7T9PwwL6.js} +3 -3
  262. package/dist/{skills-7ODkHQYp.js → skills-CE_iqvM5.js} +22 -22
  263. package/dist/{skills-cli-oay0tY8Z.js → skills-cli-DPavvthL.js} +5 -5
  264. package/dist/{skills-install-Vmi7xYfa.js → skills-install-BoLfaoWv.js} +6 -6
  265. package/dist/{skills-status-BmN697ff.js → skills-status-CK5Gnf6i.js} +1 -1
  266. package/dist/{slash-commands.runtime-BfaheruW.js → slash-commands.runtime-CMGx2xHy.js} +11 -11
  267. package/dist/slash-commands.runtime-Cpn2tYW4.js +16 -0
  268. package/dist/slash-dispatch.runtime-BAeJXa56.js +114 -0
  269. package/dist/slash-dispatch.runtime-DoBAQBU5.js +56 -0
  270. package/dist/{slash-skill-commands.runtime-D34BKAN-.js → slash-skill-commands.runtime-ChU2tck2.js} +15 -15
  271. package/dist/slash-skill-commands.runtime-DKMvvdDW.js +20 -0
  272. package/dist/{status-BcQchPaC.js → status-Bp-K1BEf.js} +27 -27
  273. package/dist/{status.update-B20UBTDq.js → status.update-ZYUSggzS.js} +2 -2
  274. package/dist/{store-D89wDcz9.js → store--eR1R_UX.js} +2 -2
  275. package/dist/{store-DDkqo1sO.js → store-DeASfYEV.js} +5 -5
  276. package/dist/{subagent-registry-CPxHbyN5.js → subagent-registry-8qHIVhRq.js} +149 -149
  277. package/dist/subagent-registry-runtime-DVomlbm6.js +114 -0
  278. package/dist/subagent-registry-runtime-ppWS3tVu.js +56 -0
  279. package/dist/{subsystem-B45WV3qB.js → subsystem-Di1z8l0Z.js} +14 -14
  280. package/dist/{system-cli-D2yIJoKU.js → system-cli-C2xNfuQM.js} +9 -9
  281. package/dist/{system-run-command-BmhbnLTE.js → system-run-command-BPWZk7KI.js} +1 -1
  282. package/dist/{systemd-DjWVSbAG.js → systemd-Cf-0XKYu.js} +9 -9
  283. package/dist/{systemd-hints-Do-aQ9jw.js → systemd-hints-CYllYKO0.js} +6 -6
  284. package/dist/{systemd-linger-DpmnYgKU.js → systemd-linger-zllO90bD.js} +1 -1
  285. package/dist/{tables-BxyIF0w4.js → tables-CV7Afb0h.js} +1 -1
  286. package/dist/{tables-mE4cJBN2.js → tables-d739Y1xW.js} +1 -1
  287. package/dist/{tailnet-0_FsdHP-.js → tailnet-DJFUq7_R.js} +1 -1
  288. package/dist/{target-errors-KOHiT_JA.js → target-errors-CBI2Ga0y.js} +2 -2
  289. package/dist/{target-errors-mnlwhAjP.js → target-errors-iVxliVqA.js} +4 -4
  290. package/dist/{thinking-BeGmb5k6.js → thinking-DXYisHiZ.js} +7 -7
  291. package/dist/{tokens-q32vI39c.js → tokens-DxnY9ui_.js} +1 -1
  292. package/dist/{tool-images-RZdHiZcG.js → tool-images-2cBx1W8h.js} +2 -2
  293. package/dist/{tool-images-CNPfeCmU.js → tool-images-Bn6dB14u.js} +1 -1
  294. package/dist/{tui-DL6NZZEa.js → tui-DTVy-YhN.js} +6 -6
  295. package/dist/{tui-cli-BNAYhvpu.js → tui-cli-CexRLJ3a.js} +32 -32
  296. package/dist/{update-DlS-d52F.js → update-DijPxK0g.js} +3 -3
  297. package/dist/{update-cli-D8-DqIs2.js → update-cli-9NslG4yR.js} +104 -104
  298. package/dist/{update-runner-CzTQ7BJT.js → update-runner-B6_UqreW.js} +16 -16
  299. package/dist/{web-igmw_EhT.js → web-CzWRVmFt.js} +55 -55
  300. package/dist/web-DGoa03ue.js +118 -0
  301. package/dist/{webhooks-cli-QdaQhvbT.js → webhooks-cli-B4ZcXTtw.js} +6 -6
  302. package/dist/{whatsapp-actions-Bp8F0cOF.js → whatsapp-actions-Jm4VW1Ve.js} +17 -17
  303. package/dist/{whatsapp-actions-BHbJJyqw.js → whatsapp-actions-iEArE_Ez.js} +21 -21
  304. package/dist/{with-timeout-BMMWHlH3.js → with-timeout-C8-tY12i.js} +3 -3
  305. package/dist/{workspace-v76gFdZu.js → workspace-CIGzK2_w.js} +1 -1
  306. package/dist/{workspace-U-DyR64O.js → workspace-CUVC6GX1.js} +20 -20
  307. package/dist/{workspace-dirs-Cz_Zgtg2.js → workspace-dirs-D4SMysgC.js} +1 -1
  308. package/dist/{wsl-CvQfS6aU.js → wsl-CsGe5QCP.js} +2 -2
  309. package/package.json +7 -9
  310. package/scripts/auth-monitor.sh +89 -0
  311. package/scripts/bench-cli-startup.ts +200 -0
  312. package/scripts/bench-model.ts +146 -0
  313. package/scripts/build-and-run-mac.sh +18 -0
  314. package/scripts/build-docs-list.mjs +14 -0
  315. package/scripts/build_icon.sh +59 -0
  316. package/scripts/bundle-a2ui.sh +95 -0
  317. package/scripts/canvas-a2ui-copy.ts +40 -0
  318. package/scripts/changelog-to-html.sh +91 -0
  319. package/scripts/check-channel-agnostic-boundaries.mjs +343 -0
  320. package/scripts/check-composite-action-input-interpolation.py +81 -0
  321. package/scripts/check-ingress-agent-owner-context.mjs +45 -0
  322. package/scripts/check-no-monolithic-plugin-sdk-entry-imports.ts +103 -0
  323. package/scripts/check-no-pairing-store-group-auth.mjs +180 -0
  324. package/scripts/check-no-random-messaging-tmp.mjs +89 -0
  325. package/scripts/check-no-raw-channel-fetch.mjs +107 -0
  326. package/scripts/check-no-raw-window-open.mjs +86 -0
  327. package/scripts/check-no-register-http-handler.mjs +38 -0
  328. package/scripts/check-pairing-account-scope.mjs +100 -0
  329. package/scripts/check-plugin-sdk-exports.mjs +157 -0
  330. package/scripts/check-ts-max-loc.ts +80 -0
  331. package/scripts/check-webhook-auth-body-order.mjs +55 -0
  332. package/scripts/ci-changed-scope.d.mts +9 -0
  333. package/scripts/ci-changed-scope.mjs +141 -0
  334. package/scripts/claude-auth-status.sh +280 -0
  335. package/scripts/clawlog.sh +321 -0
  336. package/scripts/clawtributors-map.json +40 -0
  337. package/scripts/codesign-mac-app.sh +289 -0
  338. package/scripts/codespell-dictionary.txt +3 -0
  339. package/scripts/codespell-ignore.txt +9 -0
  340. package/scripts/committer +117 -0
  341. package/scripts/copy-export-html-templates.ts +59 -0
  342. package/scripts/copy-hook-metadata.ts +55 -0
  343. package/scripts/copy-plugin-sdk-root-alias.mjs +10 -0
  344. package/scripts/create-dmg.sh +176 -0
  345. package/scripts/create-instance.sh +4 -1
  346. package/scripts/cron_usage_report.ts +273 -0
  347. package/scripts/debug-claude-usage.ts +391 -0
  348. package/scripts/dev/discord-acp-plain-language-smoke.ts +868 -0
  349. package/scripts/dev/gateway-smoke.ts +75 -0
  350. package/scripts/dev/gateway-ws-client.ts +132 -0
  351. package/scripts/dev/ios-node-e2e.ts +283 -0
  352. package/scripts/dev/ios-pull-gateway-log.sh +17 -0
  353. package/scripts/dev/test-device-pair-telegram.ts +62 -0
  354. package/scripts/docker/cleanup-smoke/Dockerfile +19 -0
  355. package/scripts/docker/cleanup-smoke/run.sh +35 -0
  356. package/scripts/docker/install-sh-common/cli-verify.sh +47 -0
  357. package/scripts/docker/install-sh-e2e/Dockerfile +17 -0
  358. package/scripts/docker/install-sh-e2e/run.sh +535 -0
  359. package/scripts/docker/install-sh-nonroot/Dockerfile +33 -0
  360. package/scripts/docker/install-sh-nonroot/run.sh +36 -0
  361. package/scripts/docker/install-sh-smoke/Dockerfile +25 -0
  362. package/scripts/docker/install-sh-smoke/run.sh +63 -0
  363. package/scripts/docs-i18n/doc_mode.go +272 -0
  364. package/scripts/docs-i18n/glossary.go +29 -0
  365. package/scripts/docs-i18n/go.mod +10 -0
  366. package/scripts/docs-i18n/go.sum +10 -0
  367. package/scripts/docs-i18n/html_translate.go +160 -0
  368. package/scripts/docs-i18n/main.go +273 -0
  369. package/scripts/docs-i18n/markdown_segments.go +131 -0
  370. package/scripts/docs-i18n/masking.go +89 -0
  371. package/scripts/docs-i18n/order.go +37 -0
  372. package/scripts/docs-i18n/placeholders.go +30 -0
  373. package/scripts/docs-i18n/process.go +202 -0
  374. package/scripts/docs-i18n/prompt.go +146 -0
  375. package/scripts/docs-i18n/segment.go +11 -0
  376. package/scripts/docs-i18n/tm.go +132 -0
  377. package/scripts/docs-i18n/translator.go +247 -0
  378. package/scripts/docs-i18n/util.go +81 -0
  379. package/scripts/docs-link-audit.mjs +233 -0
  380. package/scripts/docs-list.js +173 -0
  381. package/scripts/docs-spellcheck.sh +44 -0
  382. package/scripts/e2e/Dockerfile +30 -0
  383. package/scripts/e2e/Dockerfile.qr-import +13 -0
  384. package/scripts/e2e/doctor-install-switch-docker.sh +160 -0
  385. package/scripts/e2e/gateway-network-docker.sh +145 -0
  386. package/scripts/e2e/onboard-docker.sh +570 -0
  387. package/scripts/e2e/plugins-docker.sh +224 -0
  388. package/scripts/e2e/qr-import-docker.sh +11 -0
  389. package/scripts/firecrawl-compare.ts +139 -0
  390. package/scripts/generate-host-env-security-policy-swift.mjs +74 -0
  391. package/scripts/generate-secretref-credential-matrix.ts +14 -0
  392. package/scripts/ghsa-patch.mjs +168 -0
  393. package/scripts/install-maca.sh +173 -0
  394. package/scripts/install.ps1 +329 -0
  395. package/scripts/ios-configure-signing.sh +100 -0
  396. package/scripts/ios-team-id.sh +207 -0
  397. package/scripts/label-open-issues.ts +893 -0
  398. package/scripts/lib/callsite-guard.mjs +45 -0
  399. package/scripts/lib/pairing-guard-context.mjs +13 -0
  400. package/scripts/lib/ts-guard-utils.mjs +157 -0
  401. package/scripts/make_appcast.sh +70 -0
  402. package/scripts/mobile-reauth.sh +84 -0
  403. package/scripts/notarize-mac-artifact.sh +65 -0
  404. package/scripts/npm_publish.sh +48 -0
  405. package/scripts/package-mac-app.sh +280 -0
  406. package/scripts/package-mac-dist.sh +93 -0
  407. package/scripts/podman/openclaw.container.in +28 -0
  408. package/scripts/pr +1708 -0
  409. package/scripts/pr-merge +44 -0
  410. package/scripts/pr-prepare +40 -0
  411. package/scripts/pr-review +13 -0
  412. package/scripts/pre-commit/filter-staged-files.mjs +39 -0
  413. package/scripts/pre-commit/run-node-tool.sh +31 -0
  414. package/scripts/protocol-gen-swift.ts +247 -0
  415. package/scripts/protocol-gen.ts +51 -0
  416. package/scripts/readability-basic-compare.ts +66 -0
  417. package/scripts/recover-orphaned-processes.sh +191 -0
  418. package/scripts/release-check.ts +362 -0
  419. package/scripts/repro/tsx-name-repro.ts +3 -0
  420. package/scripts/restart-mac.sh +269 -0
  421. package/scripts/run-node.d.mts +22 -0
  422. package/scripts/run-node.mjs +263 -0
  423. package/scripts/run-openclaw-podman.sh +213 -0
  424. package/scripts/sandbox-browser-entrypoint.sh +127 -0
  425. package/scripts/sandbox-browser-setup.sh +7 -0
  426. package/scripts/sandbox-common-setup.sh +40 -0
  427. package/scripts/sandbox-setup.sh +7 -0
  428. package/scripts/setup-auth-system.sh +119 -0
  429. package/scripts/shell-helpers/README.md +226 -0
  430. package/scripts/shell-helpers/clawdock-helpers.sh +417 -0
  431. package/scripts/sparkle-build.ts +76 -0
  432. package/scripts/sqlite-vec-smoke.mjs +38 -0
  433. package/scripts/sync-dist.sh +171 -0
  434. package/scripts/sync-labels.ts +100 -0
  435. package/scripts/sync-moonshot-docs.ts +125 -0
  436. package/scripts/sync-plugin-versions.ts +108 -0
  437. package/scripts/systemd/openclaw-auth-monitor.service +14 -0
  438. package/scripts/systemd/openclaw-auth-monitor.timer +10 -0
  439. package/scripts/termux-auth-widget.sh +81 -0
  440. package/scripts/termux-quick-auth.sh +30 -0
  441. package/scripts/termux-sync-widget.sh +25 -0
  442. package/scripts/test-cleanup-docker.sh +14 -0
  443. package/scripts/test-force.ts +59 -0
  444. package/scripts/test-hotspots.mjs +83 -0
  445. package/scripts/test-install-sh-docker.sh +72 -0
  446. package/scripts/test-install-sh-e2e-docker.sh +29 -0
  447. package/scripts/test-live-gateway-models-docker.sh +33 -0
  448. package/scripts/test-live-models-docker.sh +34 -0
  449. package/scripts/test-parallel.mjs +468 -0
  450. package/scripts/test-perf-budget.mjs +127 -0
  451. package/scripts/test-shell-completion.ts +223 -0
  452. package/scripts/ui.js +203 -0
  453. package/scripts/update-clawtributors.ts +573 -0
  454. package/scripts/update-clawtributors.types.ts +36 -0
  455. package/scripts/update-package-with-mirror.sh +67 -0
  456. package/scripts/watch-node.d.mts +14 -0
  457. package/scripts/watch-node.mjs +92 -0
  458. package/scripts/write-build-info.ts +47 -0
  459. package/scripts/write-cli-compat.ts +74 -0
  460. package/scripts/write-cli-startup-metadata.ts +93 -0
  461. package/scripts/write-plugin-sdk-entry-dts.ts +60 -0
  462. package/scripts/zai-fallback-repro.ts +168 -0
  463. package/dist/deliver-runtime-B80mXYrL.js +0 -36
  464. package/dist/deliver-runtime-DixuU_uB.js +0 -61
  465. package/dist/deps-send-discord.runtime-CDcLcDSt.js +0 -36
  466. package/dist/deps-send-discord.runtime-DZUccI6Z.js +0 -26
  467. package/dist/deps-send-imessage.runtime-BGPUVMIE.js +0 -35
  468. package/dist/deps-send-imessage.runtime-CF3OpoqY.js +0 -25
  469. package/dist/deps-send-signal.runtime-Cw4-ozeO.js +0 -24
  470. package/dist/deps-send-signal.runtime-DQDD44_O.js +0 -34
  471. package/dist/deps-send-slack.runtime-BDsDhS1P.js +0 -22
  472. package/dist/deps-send-slack.runtime-Cpy77gBG.js +0 -32
  473. package/dist/deps-send-telegram.runtime-D_4xVasO.js +0 -27
  474. package/dist/deps-send-whatsapp.runtime-D2sVAclS.js +0 -119
  475. package/dist/deps-send-whatsapp.runtime-UBgpoWg9.js +0 -60
  476. package/dist/image-runtime-B2qh5seQ.js +0 -55
  477. package/dist/image-runtime-DtMbm7AR.js +0 -29
  478. package/dist/manager-runtime-FO1Sx3W8.js +0 -18
  479. package/dist/pi-model-discovery-runtime-BeY4EUPp.js +0 -11
  480. package/dist/register.configure-BrDOSLIq.js +0 -165
  481. package/dist/runtime-whatsapp-login.runtime-DUb55byQ.js +0 -13
  482. package/dist/runtime-whatsapp-outbound.runtime-Bii_xSfI.js +0 -22
  483. package/dist/slash-commands.runtime-CVw6566g.js +0 -16
  484. package/dist/slash-dispatch.runtime-CMK_EPYG.js +0 -56
  485. package/dist/slash-dispatch.runtime-D0xInkf3.js +0 -114
  486. package/dist/slash-skill-commands.runtime-DxZ4z5h6.js +0 -20
  487. package/dist/subagent-registry-runtime-DdGrQBnQ.js +0 -56
  488. package/dist/subagent-registry-runtime-SkB2tTaE.js +0 -114
  489. package/dist/web-DWiOofzq.js +0 -118
  490. package/install.sh +0 -293
@@ -0,0 +1,868 @@
1
+ #!/usr/bin/env bun
2
+ import { execFile } from "node:child_process";
3
+ // Manual ACP thread smoke for plain-language routing.
4
+ // Keep this script available for regression/debug validation. Do not delete.
5
+ import { randomUUID } from "node:crypto";
6
+ import fs from "node:fs/promises";
7
+ import path from "node:path";
8
+ import { promisify } from "node:util";
9
+
10
+ type ThreadBindingRecord = {
11
+ accountId?: string;
12
+ channelId?: string;
13
+ threadId?: string;
14
+ targetKind?: string;
15
+ targetSessionKey?: string;
16
+ agentId?: string;
17
+ boundBy?: string;
18
+ boundAt?: number;
19
+ };
20
+
21
+ type ThreadBindingsPayload = {
22
+ version?: number;
23
+ bindings?: Record<string, ThreadBindingRecord>;
24
+ };
25
+
26
+ type DiscordMessage = {
27
+ id: string;
28
+ content?: string;
29
+ timestamp?: string;
30
+ author?: {
31
+ id?: string;
32
+ username?: string;
33
+ bot?: boolean;
34
+ };
35
+ };
36
+
37
+ type DiscordUser = {
38
+ id: string;
39
+ username: string;
40
+ bot?: boolean;
41
+ };
42
+
43
+ const execFileAsync = promisify(execFile);
44
+
45
+ type DriverMode = "token" | "webhook" | "openclaw";
46
+
47
+ type Args = {
48
+ channelId: string;
49
+ driverMode: DriverMode;
50
+ driverToken: string;
51
+ driverTokenPrefix: string;
52
+ botToken: string;
53
+ botTokenPrefix: string;
54
+ targetAgent: string;
55
+ timeoutMs: number;
56
+ pollMs: number;
57
+ mentionUserId?: string;
58
+ instruction?: string;
59
+ threadBindingsPath: string;
60
+ openclawBin: string;
61
+ json: boolean;
62
+ };
63
+
64
+ type SuccessResult = {
65
+ ok: true;
66
+ smokeId: string;
67
+ ackToken: string;
68
+ sentMessageId: string;
69
+ binding: {
70
+ threadId: string;
71
+ targetSessionKey: string;
72
+ targetKind: string;
73
+ agentId: string;
74
+ boundAt: number;
75
+ accountId?: string;
76
+ channelId?: string;
77
+ };
78
+ ackMessage: {
79
+ id: string;
80
+ authorId?: string;
81
+ authorUsername?: string;
82
+ timestamp?: string;
83
+ content?: string;
84
+ };
85
+ };
86
+
87
+ type FailureResult = {
88
+ ok: false;
89
+ smokeId: string;
90
+ stage: "validation" | "send-message" | "wait-binding" | "wait-ack" | "discord-api" | "unexpected";
91
+ error: string;
92
+ diagnostics?: {
93
+ parentChannelRecent?: Array<{
94
+ id: string;
95
+ author?: string;
96
+ bot?: boolean;
97
+ content?: string;
98
+ }>;
99
+ bindingCandidates?: Array<{
100
+ threadId: string;
101
+ targetSessionKey: string;
102
+ targetKind?: string;
103
+ agentId?: string;
104
+ boundAt?: number;
105
+ }>;
106
+ };
107
+ };
108
+
109
+ const DISCORD_API_BASE = "https://discord.com/api/v10";
110
+
111
+ function sleep(ms: number): Promise<void> {
112
+ return new Promise((resolve) => setTimeout(resolve, ms));
113
+ }
114
+
115
+ function parseNumber(value: string | undefined, fallback: number): number {
116
+ if (!value) {
117
+ return fallback;
118
+ }
119
+ const parsed = Number.parseInt(value, 10);
120
+ return Number.isFinite(parsed) && parsed > 0 ? parsed : fallback;
121
+ }
122
+
123
+ function resolveStateDir(): string {
124
+ const override = process.env.OPENCLAW_STATE_DIR?.trim() || process.env.CLAWDBOT_STATE_DIR?.trim();
125
+ if (override) {
126
+ return override.startsWith("~")
127
+ ? path.resolve(process.env.HOME || "", override.slice(1))
128
+ : path.resolve(override);
129
+ }
130
+ const home = process.env.OPENCLAW_HOME?.trim() || process.env.HOME || "";
131
+ return path.join(home, ".openclaw");
132
+ }
133
+
134
+ function resolveArg(flag: string): string | undefined {
135
+ const argv = process.argv.slice(2);
136
+ const eq = argv.find((entry) => entry.startsWith(`${flag}=`));
137
+ if (eq) {
138
+ return eq.slice(flag.length + 1);
139
+ }
140
+ const idx = argv.indexOf(flag);
141
+ if (idx >= 0 && idx + 1 < argv.length) {
142
+ return argv[idx + 1];
143
+ }
144
+ return undefined;
145
+ }
146
+
147
+ function hasFlag(flag: string): boolean {
148
+ return process.argv.slice(2).includes(flag);
149
+ }
150
+
151
+ function usage(): string {
152
+ return (
153
+ "Usage: bun scripts/dev/discord-acp-plain-language-smoke.ts " +
154
+ "--channel <discord-channel-id> [--token <driver-token> | --driver webhook --bot-token <bot-token> | --driver openclaw] [options]\n\n" +
155
+ "Manual live smoke only (not CI). Sends a plain-language instruction in Discord and verifies:\n" +
156
+ "1) OpenClaw spawned an ACP thread binding\n" +
157
+ "2) agent replied in that bound thread with the expected ACK token\n\n" +
158
+ "Options:\n" +
159
+ " --channel <id> Parent Discord channel id (required)\n" +
160
+ " --driver <token|webhook|openclaw> Driver transport mode (default: token)\n" +
161
+ " --token <token> Driver Discord token (required for driver=token)\n" +
162
+ " --token-prefix <prefix> Auth prefix for --token (default: Bot)\n" +
163
+ " --bot-token <token> Bot token for webhook driver mode\n" +
164
+ " --bot-token-prefix <prefix> Auth prefix for --bot-token (default: Bot)\n" +
165
+ " --agent <id> Expected ACP agent id (default: codex)\n" +
166
+ " --mention <user-id> Mention this user in the instruction (optional)\n" +
167
+ " --instruction <text> Custom instruction template (optional)\n" +
168
+ " --timeout-ms <n> Total timeout in ms (default: 240000)\n" +
169
+ " --poll-ms <n> Poll interval in ms (default: 1500)\n" +
170
+ " --thread-bindings-path <p> Override thread-bindings json path\n" +
171
+ " --openclaw-bin <path> OpenClaw CLI binary for driver=openclaw (default: openclaw)\n" +
172
+ " --json Emit JSON output\n" +
173
+ "\n" +
174
+ "Environment fallbacks:\n" +
175
+ " OPENCLAW_DISCORD_SMOKE_CHANNEL_ID\n" +
176
+ " OPENCLAW_DISCORD_SMOKE_DRIVER\n" +
177
+ " OPENCLAW_DISCORD_SMOKE_DRIVER_TOKEN\n" +
178
+ " OPENCLAW_DISCORD_SMOKE_DRIVER_TOKEN_PREFIX\n" +
179
+ " OPENCLAW_DISCORD_SMOKE_BOT_TOKEN\n" +
180
+ " OPENCLAW_DISCORD_SMOKE_BOT_TOKEN_PREFIX\n" +
181
+ " OPENCLAW_DISCORD_SMOKE_AGENT\n" +
182
+ " OPENCLAW_DISCORD_SMOKE_MENTION_USER_ID\n" +
183
+ " OPENCLAW_DISCORD_SMOKE_TIMEOUT_MS\n" +
184
+ " OPENCLAW_DISCORD_SMOKE_POLL_MS\n" +
185
+ " OPENCLAW_DISCORD_SMOKE_THREAD_BINDINGS_PATH\n" +
186
+ " OPENCLAW_DISCORD_SMOKE_OPENCLAW_BIN"
187
+ );
188
+ }
189
+
190
+ function parseArgs(): Args {
191
+ const channelId =
192
+ resolveArg("--channel") ||
193
+ process.env.OPENCLAW_DISCORD_SMOKE_CHANNEL_ID ||
194
+ process.env.CLAWDBOT_DISCORD_SMOKE_CHANNEL_ID ||
195
+ "";
196
+ const driverModeRaw =
197
+ resolveArg("--driver") ||
198
+ process.env.OPENCLAW_DISCORD_SMOKE_DRIVER ||
199
+ process.env.CLAWDBOT_DISCORD_SMOKE_DRIVER ||
200
+ "token";
201
+ const normalizedDriverMode = driverModeRaw.trim().toLowerCase();
202
+ const driverMode: DriverMode =
203
+ normalizedDriverMode === "webhook"
204
+ ? "webhook"
205
+ : normalizedDriverMode === "openclaw"
206
+ ? "openclaw"
207
+ : normalizedDriverMode === "token"
208
+ ? "token"
209
+ : "token";
210
+ const driverToken =
211
+ resolveArg("--token") ||
212
+ process.env.OPENCLAW_DISCORD_SMOKE_DRIVER_TOKEN ||
213
+ process.env.CLAWDBOT_DISCORD_SMOKE_DRIVER_TOKEN ||
214
+ "";
215
+ const driverTokenPrefix =
216
+ resolveArg("--token-prefix") || process.env.OPENCLAW_DISCORD_SMOKE_DRIVER_TOKEN_PREFIX || "Bot";
217
+ const botToken =
218
+ resolveArg("--bot-token") ||
219
+ process.env.OPENCLAW_DISCORD_SMOKE_BOT_TOKEN ||
220
+ process.env.CLAWDBOT_DISCORD_SMOKE_BOT_TOKEN ||
221
+ process.env.DISCORD_BOT_TOKEN ||
222
+ "";
223
+ const botTokenPrefix =
224
+ resolveArg("--bot-token-prefix") ||
225
+ process.env.OPENCLAW_DISCORD_SMOKE_BOT_TOKEN_PREFIX ||
226
+ "Bot";
227
+ const targetAgent =
228
+ resolveArg("--agent") ||
229
+ process.env.OPENCLAW_DISCORD_SMOKE_AGENT ||
230
+ process.env.CLAWDBOT_DISCORD_SMOKE_AGENT ||
231
+ "codex";
232
+ const mentionUserId =
233
+ resolveArg("--mention") ||
234
+ process.env.OPENCLAW_DISCORD_SMOKE_MENTION_USER_ID ||
235
+ process.env.CLAWDBOT_DISCORD_SMOKE_MENTION_USER_ID ||
236
+ undefined;
237
+ const instruction =
238
+ resolveArg("--instruction") ||
239
+ process.env.OPENCLAW_DISCORD_SMOKE_INSTRUCTION ||
240
+ process.env.CLAWDBOT_DISCORD_SMOKE_INSTRUCTION ||
241
+ undefined;
242
+ const timeoutMs = parseNumber(
243
+ resolveArg("--timeout-ms") || process.env.OPENCLAW_DISCORD_SMOKE_TIMEOUT_MS,
244
+ 240_000,
245
+ );
246
+ const pollMs = parseNumber(
247
+ resolveArg("--poll-ms") || process.env.OPENCLAW_DISCORD_SMOKE_POLL_MS,
248
+ 1_500,
249
+ );
250
+ const defaultBindingsPath = path.join(resolveStateDir(), "discord", "thread-bindings.json");
251
+ const threadBindingsPath =
252
+ resolveArg("--thread-bindings-path") ||
253
+ process.env.OPENCLAW_DISCORD_SMOKE_THREAD_BINDINGS_PATH ||
254
+ defaultBindingsPath;
255
+ const openclawBin =
256
+ resolveArg("--openclaw-bin") || process.env.OPENCLAW_DISCORD_SMOKE_OPENCLAW_BIN || "openclaw";
257
+ const json = hasFlag("--json");
258
+
259
+ if (!channelId) {
260
+ throw new Error(usage());
261
+ }
262
+ if (driverMode === "token" && !driverToken) {
263
+ throw new Error(usage());
264
+ }
265
+ if (driverMode === "webhook" && !botToken) {
266
+ throw new Error(usage());
267
+ }
268
+
269
+ return {
270
+ channelId,
271
+ driverMode,
272
+ driverToken,
273
+ driverTokenPrefix,
274
+ botToken,
275
+ botTokenPrefix,
276
+ targetAgent,
277
+ timeoutMs,
278
+ pollMs,
279
+ mentionUserId,
280
+ instruction,
281
+ threadBindingsPath,
282
+ openclawBin,
283
+ json,
284
+ };
285
+ }
286
+
287
+ async function openclawCliJson<T>(params: { openclawBin: string; args: string[] }): Promise<T> {
288
+ const result = await execFileAsync(params.openclawBin, params.args, {
289
+ maxBuffer: 8 * 1024 * 1024,
290
+ env: process.env,
291
+ });
292
+ const stdout = (result.stdout || "").trim();
293
+ if (!stdout) {
294
+ throw new Error(`openclaw ${params.args.join(" ")} returned empty stdout`);
295
+ }
296
+ return JSON.parse(stdout) as T;
297
+ }
298
+
299
+ async function readMessagesWithOpenclaw(params: {
300
+ openclawBin: string;
301
+ target: string;
302
+ limit: number;
303
+ }): Promise<DiscordMessage[]> {
304
+ const response = await openclawCliJson<{
305
+ payload?: {
306
+ messages?: DiscordMessage[];
307
+ };
308
+ }>({
309
+ openclawBin: params.openclawBin,
310
+ args: [
311
+ "message",
312
+ "read",
313
+ "--channel",
314
+ "discord",
315
+ "--target",
316
+ params.target,
317
+ "--limit",
318
+ String(params.limit),
319
+ "--json",
320
+ ],
321
+ });
322
+ return Array.isArray(response.payload?.messages) ? response.payload.messages : [];
323
+ }
324
+
325
+ function resolveAuthorizationHeader(params: { token: string; tokenPrefix: string }): string {
326
+ const token = params.token.trim();
327
+ if (!token) {
328
+ throw new Error("Missing Discord driver token.");
329
+ }
330
+ if (token.includes(" ")) {
331
+ return token;
332
+ }
333
+ return `${params.tokenPrefix.trim() || "Bot"} ${token}`;
334
+ }
335
+
336
+ async function discordApi<T>(params: {
337
+ method: "GET" | "POST";
338
+ path: string;
339
+ authHeader: string;
340
+ body?: unknown;
341
+ retries?: number;
342
+ }): Promise<T> {
343
+ return requestDiscordJson<T>({
344
+ method: params.method,
345
+ path: params.path,
346
+ headers: {
347
+ Authorization: params.authHeader,
348
+ "Content-Type": "application/json",
349
+ },
350
+ body: params.body,
351
+ retries: params.retries,
352
+ errorPrefix: "Discord API",
353
+ });
354
+ }
355
+
356
+ async function discordWebhookApi<T>(params: {
357
+ method: "POST" | "DELETE";
358
+ webhookId: string;
359
+ webhookToken: string;
360
+ body?: unknown;
361
+ query?: string;
362
+ retries?: number;
363
+ }): Promise<T> {
364
+ const suffix = params.query ? `?${params.query}` : "";
365
+ const path = `/webhooks/${encodeURIComponent(params.webhookId)}/${encodeURIComponent(params.webhookToken)}${suffix}`;
366
+ return requestDiscordJson<T>({
367
+ method: params.method,
368
+ path,
369
+ headers: {
370
+ "Content-Type": "application/json",
371
+ },
372
+ body: params.body,
373
+ retries: params.retries,
374
+ errorPrefix: "Discord webhook API",
375
+ });
376
+ }
377
+
378
+ async function requestDiscordJson<T>(params: {
379
+ method: string;
380
+ path: string;
381
+ headers: Record<string, string>;
382
+ body?: unknown;
383
+ retries?: number;
384
+ errorPrefix: string;
385
+ }): Promise<T> {
386
+ const retries = params.retries ?? 6;
387
+ for (let attempt = 0; attempt <= retries; attempt += 1) {
388
+ const response = await fetch(`${DISCORD_API_BASE}${params.path}`, {
389
+ method: params.method,
390
+ headers: params.headers,
391
+ body: params.body === undefined ? undefined : JSON.stringify(params.body),
392
+ });
393
+
394
+ if (response.status === 429) {
395
+ const body = (await response.json().catch(() => ({}))) as { retry_after?: number };
396
+ const waitSeconds = typeof body.retry_after === "number" ? body.retry_after : 1;
397
+ await sleep(Math.ceil(waitSeconds * 1000));
398
+ continue;
399
+ }
400
+
401
+ if (!response.ok) {
402
+ const text = await response.text().catch(() => "");
403
+ throw new Error(
404
+ `${params.errorPrefix} ${params.method} ${params.path} failed: ${response.status} ${response.statusText}${text ? ` :: ${text}` : ""}`,
405
+ );
406
+ }
407
+
408
+ if (response.status === 204) {
409
+ return undefined as T;
410
+ }
411
+
412
+ return (await response.json()) as T;
413
+ }
414
+
415
+ throw new Error(`${params.errorPrefix} ${params.method} ${params.path} exceeded retry budget.`);
416
+ }
417
+
418
+ async function readThreadBindings(filePath: string): Promise<ThreadBindingRecord[]> {
419
+ const raw = await fs.readFile(filePath, "utf8");
420
+ const payload = JSON.parse(raw) as ThreadBindingsPayload;
421
+ const entries = Object.values(payload.bindings ?? {});
422
+ return entries.filter((entry) => Boolean(entry?.threadId && entry?.targetSessionKey));
423
+ }
424
+
425
+ function normalizeBoundAt(record: ThreadBindingRecord): number {
426
+ if (typeof record.boundAt === "number" && Number.isFinite(record.boundAt)) {
427
+ return record.boundAt;
428
+ }
429
+ return 0;
430
+ }
431
+
432
+ function resolveCandidateBindings(params: {
433
+ entries: ThreadBindingRecord[];
434
+ minBoundAt: number;
435
+ targetAgent: string;
436
+ }): ThreadBindingRecord[] {
437
+ const normalizedTargetAgent = params.targetAgent.trim().toLowerCase();
438
+ return params.entries
439
+ .filter((entry) => {
440
+ const targetKind = String(entry.targetKind || "")
441
+ .trim()
442
+ .toLowerCase();
443
+ if (targetKind !== "acp") {
444
+ return false;
445
+ }
446
+ if (normalizeBoundAt(entry) < params.minBoundAt) {
447
+ return false;
448
+ }
449
+ const agentId = String(entry.agentId || "")
450
+ .trim()
451
+ .toLowerCase();
452
+ if (normalizedTargetAgent && agentId && agentId !== normalizedTargetAgent) {
453
+ return false;
454
+ }
455
+ return true;
456
+ })
457
+ .toSorted((a, b) => normalizeBoundAt(b) - normalizeBoundAt(a));
458
+ }
459
+
460
+ function buildInstruction(params: {
461
+ smokeId: string;
462
+ ackToken: string;
463
+ targetAgent: string;
464
+ mentionUserId?: string;
465
+ template?: string;
466
+ }): string {
467
+ const mentionPrefix = params.mentionUserId?.trim() ? `<@${params.mentionUserId.trim()}> ` : "";
468
+ if (params.template?.trim()) {
469
+ return mentionPrefix + params.template.trim();
470
+ }
471
+ return (
472
+ mentionPrefix +
473
+ `Manual smoke ${params.smokeId}: Please spawn a ${params.targetAgent} ACP coding agent in a thread for this request, keep it persistent, and in that thread reply with exactly "${params.ackToken}" and nothing else.`
474
+ );
475
+ }
476
+
477
+ function toRecentMessageRow(message: DiscordMessage) {
478
+ return {
479
+ id: message.id,
480
+ author: message.author?.username || message.author?.id || "unknown",
481
+ bot: Boolean(message.author?.bot),
482
+ content: (message.content || "").slice(0, 500),
483
+ };
484
+ }
485
+
486
+ async function loadParentRecentMessages(params: {
487
+ args: Args;
488
+ readAuthHeader: string;
489
+ }): Promise<DiscordMessage[]> {
490
+ if (params.args.driverMode === "openclaw") {
491
+ return await readMessagesWithOpenclaw({
492
+ openclawBin: params.args.openclawBin,
493
+ target: params.args.channelId,
494
+ limit: 20,
495
+ });
496
+ }
497
+ return await discordApi<DiscordMessage[]>({
498
+ method: "GET",
499
+ path: `/channels/${encodeURIComponent(params.args.channelId)}/messages?limit=20`,
500
+ authHeader: params.readAuthHeader,
501
+ });
502
+ }
503
+
504
+ function printOutput(params: { json: boolean; payload: SuccessResult | FailureResult }) {
505
+ if (params.json) {
506
+ // eslint-disable-next-line no-console
507
+ console.log(JSON.stringify(params.payload, null, 2));
508
+ return;
509
+ }
510
+ if (params.payload.ok) {
511
+ const success = params.payload;
512
+ // eslint-disable-next-line no-console
513
+ console.log("PASS");
514
+ // eslint-disable-next-line no-console
515
+ console.log(`smokeId: ${success.smokeId}`);
516
+ // eslint-disable-next-line no-console
517
+ console.log(`sentMessageId: ${success.sentMessageId}`);
518
+ // eslint-disable-next-line no-console
519
+ console.log(`threadId: ${success.binding.threadId}`);
520
+ // eslint-disable-next-line no-console
521
+ console.log(`sessionKey: ${success.binding.targetSessionKey}`);
522
+ // eslint-disable-next-line no-console
523
+ console.log(`ackMessageId: ${success.ackMessage.id}`);
524
+ // eslint-disable-next-line no-console
525
+ console.log(
526
+ `ackAuthor: ${success.ackMessage.authorUsername || success.ackMessage.authorId || "unknown"}`,
527
+ );
528
+ return;
529
+ }
530
+ const failure = params.payload;
531
+ // eslint-disable-next-line no-console
532
+ console.error("FAIL");
533
+ // eslint-disable-next-line no-console
534
+ console.error(`stage: ${failure.stage}`);
535
+ // eslint-disable-next-line no-console
536
+ console.error(`smokeId: ${failure.smokeId}`);
537
+ // eslint-disable-next-line no-console
538
+ console.error(`error: ${failure.error}`);
539
+ if (failure.diagnostics?.bindingCandidates?.length) {
540
+ // eslint-disable-next-line no-console
541
+ console.error("binding candidates:");
542
+ for (const candidate of failure.diagnostics.bindingCandidates) {
543
+ // eslint-disable-next-line no-console
544
+ console.error(
545
+ ` thread=${candidate.threadId} kind=${candidate.targetKind || "?"} agent=${candidate.agentId || "?"} boundAt=${candidate.boundAt || 0} session=${candidate.targetSessionKey}`,
546
+ );
547
+ }
548
+ }
549
+ if (failure.diagnostics?.parentChannelRecent?.length) {
550
+ // eslint-disable-next-line no-console
551
+ console.error("recent parent channel messages:");
552
+ for (const row of failure.diagnostics.parentChannelRecent) {
553
+ // eslint-disable-next-line no-console
554
+ console.error(` ${row.id} ${row.author}${row.bot ? " [bot]" : ""}: ${row.content || ""}`);
555
+ }
556
+ }
557
+ }
558
+
559
+ async function run(): Promise<SuccessResult | FailureResult> {
560
+ let args: Args;
561
+ try {
562
+ args = parseArgs();
563
+ } catch (err) {
564
+ return {
565
+ ok: false,
566
+ stage: "validation",
567
+ smokeId: "n/a",
568
+ error: err instanceof Error ? err.message : String(err),
569
+ };
570
+ }
571
+
572
+ const smokeId = `acp-smoke-${Date.now()}-${randomUUID().slice(0, 8)}`;
573
+ const ackToken = `ACP_SMOKE_ACK_${smokeId}`;
574
+ const instruction = buildInstruction({
575
+ smokeId,
576
+ ackToken,
577
+ targetAgent: args.targetAgent,
578
+ mentionUserId: args.mentionUserId,
579
+ template: args.instruction,
580
+ });
581
+
582
+ let readAuthHeader = "";
583
+ let sentMessageId = "";
584
+ let setupStage: "discord-api" | "send-message" = "discord-api";
585
+ let senderAuthorId: string | undefined;
586
+ let webhookForCleanup:
587
+ | {
588
+ id: string;
589
+ token: string;
590
+ }
591
+ | undefined;
592
+
593
+ try {
594
+ if (args.driverMode === "token") {
595
+ const authHeader = resolveAuthorizationHeader({
596
+ token: args.driverToken,
597
+ tokenPrefix: args.driverTokenPrefix,
598
+ });
599
+ readAuthHeader = authHeader;
600
+
601
+ const driverUser = await discordApi<DiscordUser>({
602
+ method: "GET",
603
+ path: "/users/@me",
604
+ authHeader,
605
+ });
606
+ senderAuthorId = driverUser.id;
607
+
608
+ setupStage = "send-message";
609
+ const sent = await discordApi<DiscordMessage>({
610
+ method: "POST",
611
+ path: `/channels/${encodeURIComponent(args.channelId)}/messages`,
612
+ authHeader,
613
+ body: {
614
+ content: instruction,
615
+ allowed_mentions: args.mentionUserId
616
+ ? { parse: [], users: [args.mentionUserId] }
617
+ : { parse: [] },
618
+ },
619
+ });
620
+ sentMessageId = sent.id;
621
+ } else if (args.driverMode === "webhook") {
622
+ const botAuthHeader = resolveAuthorizationHeader({
623
+ token: args.botToken,
624
+ tokenPrefix: args.botTokenPrefix,
625
+ });
626
+ readAuthHeader = botAuthHeader;
627
+
628
+ await discordApi<DiscordUser>({
629
+ method: "GET",
630
+ path: "/users/@me",
631
+ authHeader: botAuthHeader,
632
+ });
633
+
634
+ setupStage = "send-message";
635
+ const webhook = await discordApi<{ id: string; token?: string | null }>({
636
+ method: "POST",
637
+ path: `/channels/${encodeURIComponent(args.channelId)}/webhooks`,
638
+ authHeader: botAuthHeader,
639
+ body: {
640
+ name: `openclaw-acp-smoke-${smokeId.slice(-8)}`,
641
+ },
642
+ });
643
+ if (!webhook.id || !webhook.token) {
644
+ return {
645
+ ok: false,
646
+ stage: "send-message",
647
+ smokeId,
648
+ error:
649
+ "Discord webhook creation succeeded but no webhook token was returned; cannot post smoke message.",
650
+ };
651
+ }
652
+ webhookForCleanup = { id: webhook.id, token: webhook.token };
653
+
654
+ const sent = await discordWebhookApi<DiscordMessage>({
655
+ method: "POST",
656
+ webhookId: webhook.id,
657
+ webhookToken: webhook.token,
658
+ query: "wait=true",
659
+ body: {
660
+ content: instruction,
661
+ allowed_mentions: args.mentionUserId
662
+ ? { parse: [], users: [args.mentionUserId] }
663
+ : { parse: [] },
664
+ },
665
+ });
666
+ sentMessageId = sent.id;
667
+ senderAuthorId = sent.author?.id;
668
+ } else {
669
+ setupStage = "send-message";
670
+ const sent = await openclawCliJson<{
671
+ payload?: {
672
+ result?: {
673
+ messageId?: string;
674
+ };
675
+ };
676
+ }>({
677
+ openclawBin: args.openclawBin,
678
+ args: [
679
+ "message",
680
+ "send",
681
+ "--channel",
682
+ "discord",
683
+ "--target",
684
+ args.channelId,
685
+ "--message",
686
+ instruction,
687
+ "--json",
688
+ ],
689
+ });
690
+ sentMessageId = String(sent.payload?.result?.messageId || "");
691
+ if (!sentMessageId) {
692
+ throw new Error("openclaw message send did not return payload.result.messageId");
693
+ }
694
+ }
695
+ } catch (err) {
696
+ return {
697
+ ok: false,
698
+ stage: setupStage,
699
+ smokeId,
700
+ error: err instanceof Error ? err.message : String(err),
701
+ };
702
+ }
703
+
704
+ const startedAt = Date.now();
705
+
706
+ const deadline = startedAt + args.timeoutMs;
707
+ let winningBinding: ThreadBindingRecord | undefined;
708
+ let latestCandidates: ThreadBindingRecord[] = [];
709
+
710
+ try {
711
+ while (Date.now() < deadline && !winningBinding) {
712
+ try {
713
+ const entries = await readThreadBindings(args.threadBindingsPath);
714
+ latestCandidates = resolveCandidateBindings({
715
+ entries,
716
+ minBoundAt: startedAt - 3_000,
717
+ targetAgent: args.targetAgent,
718
+ });
719
+ winningBinding = latestCandidates[0];
720
+ } catch {
721
+ // Keep polling; file may not exist yet or may be mid-write.
722
+ }
723
+ if (!winningBinding) {
724
+ await sleep(args.pollMs);
725
+ }
726
+ }
727
+
728
+ if (!winningBinding?.threadId || !winningBinding?.targetSessionKey) {
729
+ let parentRecent: DiscordMessage[] = [];
730
+ try {
731
+ parentRecent = await loadParentRecentMessages({ args, readAuthHeader });
732
+ } catch {
733
+ // Best effort diagnostics only.
734
+ }
735
+ return {
736
+ ok: false,
737
+ stage: "wait-binding",
738
+ smokeId,
739
+ error: `Timed out waiting for new ACP thread binding (path: ${args.threadBindingsPath}).`,
740
+ diagnostics: {
741
+ bindingCandidates: latestCandidates.slice(0, 6).map((entry) => ({
742
+ threadId: entry.threadId || "",
743
+ targetSessionKey: entry.targetSessionKey || "",
744
+ targetKind: entry.targetKind,
745
+ agentId: entry.agentId,
746
+ boundAt: entry.boundAt,
747
+ })),
748
+ parentChannelRecent: parentRecent.map(toRecentMessageRow),
749
+ },
750
+ };
751
+ }
752
+
753
+ const threadId = winningBinding.threadId;
754
+ let ackMessage: DiscordMessage | undefined;
755
+ while (Date.now() < deadline && !ackMessage) {
756
+ try {
757
+ const threadMessages =
758
+ args.driverMode === "openclaw"
759
+ ? await readMessagesWithOpenclaw({
760
+ openclawBin: args.openclawBin,
761
+ target: threadId,
762
+ limit: 50,
763
+ })
764
+ : await discordApi<DiscordMessage[]>({
765
+ method: "GET",
766
+ path: `/channels/${encodeURIComponent(threadId)}/messages?limit=50`,
767
+ authHeader: readAuthHeader,
768
+ });
769
+ ackMessage = threadMessages.find((message) => {
770
+ const content = message.content || "";
771
+ if (!content.includes(ackToken)) {
772
+ return false;
773
+ }
774
+ const authorId = message.author?.id || "";
775
+ return !senderAuthorId || authorId !== senderAuthorId;
776
+ });
777
+ } catch {
778
+ // Keep polling; thread can appear before read permissions settle.
779
+ }
780
+ if (!ackMessage) {
781
+ await sleep(args.pollMs);
782
+ }
783
+ }
784
+
785
+ if (!ackMessage) {
786
+ let parentRecent: DiscordMessage[] = [];
787
+ try {
788
+ parentRecent = await loadParentRecentMessages({ args, readAuthHeader });
789
+ } catch {
790
+ // Best effort diagnostics only.
791
+ }
792
+
793
+ return {
794
+ ok: false,
795
+ stage: "wait-ack",
796
+ smokeId,
797
+ error: `Thread bound (${threadId}) but timed out waiting for ACK token "${ackToken}" from OpenClaw.`,
798
+ diagnostics: {
799
+ bindingCandidates: [
800
+ {
801
+ threadId: winningBinding.threadId || "",
802
+ targetSessionKey: winningBinding.targetSessionKey || "",
803
+ targetKind: winningBinding.targetKind,
804
+ agentId: winningBinding.agentId,
805
+ boundAt: winningBinding.boundAt,
806
+ },
807
+ ],
808
+ parentChannelRecent: parentRecent.map(toRecentMessageRow),
809
+ },
810
+ };
811
+ }
812
+
813
+ return {
814
+ ok: true,
815
+ smokeId,
816
+ ackToken,
817
+ sentMessageId,
818
+ binding: {
819
+ threadId,
820
+ targetSessionKey: winningBinding.targetSessionKey,
821
+ targetKind: String(winningBinding.targetKind || "acp"),
822
+ agentId: String(winningBinding.agentId || args.targetAgent),
823
+ boundAt: normalizeBoundAt(winningBinding),
824
+ accountId: winningBinding.accountId,
825
+ channelId: winningBinding.channelId,
826
+ },
827
+ ackMessage: {
828
+ id: ackMessage.id,
829
+ authorId: ackMessage.author?.id,
830
+ authorUsername: ackMessage.author?.username,
831
+ timestamp: ackMessage.timestamp,
832
+ content: ackMessage.content,
833
+ },
834
+ };
835
+ } finally {
836
+ if (webhookForCleanup) {
837
+ await discordWebhookApi<void>({
838
+ method: "DELETE",
839
+ webhookId: webhookForCleanup.id,
840
+ webhookToken: webhookForCleanup.token,
841
+ }).catch(() => {
842
+ // Best-effort cleanup only.
843
+ });
844
+ }
845
+ }
846
+ }
847
+
848
+ if (hasFlag("--help") || hasFlag("-h")) {
849
+ // eslint-disable-next-line no-console
850
+ console.log(usage());
851
+ process.exit(0);
852
+ }
853
+
854
+ const result = await run().catch(
855
+ (err): FailureResult => ({
856
+ ok: false,
857
+ stage: "unexpected",
858
+ smokeId: "n/a",
859
+ error: err instanceof Error ? err.message : String(err),
860
+ }),
861
+ );
862
+
863
+ printOutput({
864
+ json: hasFlag("--json"),
865
+ payload: result,
866
+ });
867
+
868
+ process.exit(result.ok ? 0 : 1);