@captain-app/openclaw 2026.2.4

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 (1686) hide show
  1. package/CHANGELOG.md +1528 -0
  2. package/LICENSE +21 -0
  3. package/README-header.png +0 -0
  4. package/README.md +539 -0
  5. package/assets/avatar-placeholder.svg +19 -0
  6. package/assets/chrome-extension/README.md +22 -0
  7. package/assets/chrome-extension/background.js +438 -0
  8. package/assets/chrome-extension/icons/icon128.png +0 -0
  9. package/assets/chrome-extension/icons/icon16.png +0 -0
  10. package/assets/chrome-extension/icons/icon32.png +0 -0
  11. package/assets/chrome-extension/icons/icon48.png +0 -0
  12. package/assets/chrome-extension/manifest.json +25 -0
  13. package/assets/chrome-extension/options.html +196 -0
  14. package/assets/chrome-extension/options.js +59 -0
  15. package/assets/dmg-background-small.png +0 -0
  16. package/assets/dmg-background.png +0 -0
  17. package/dist/accounts-ClnuDahN.js +250 -0
  18. package/dist/accounts-DzBgAM3C.js +251 -0
  19. package/dist/acp-cli-BEDDkzXH.js +926 -0
  20. package/dist/acp-cli-CwV4mdnW.js +923 -0
  21. package/dist/agent-CArjbSeE.js +695 -0
  22. package/dist/agent-CB7x5HLT.js +695 -0
  23. package/dist/agent-scope-BbT4OG2N.js +545 -0
  24. package/dist/agent-scope-C_O6Vpl0.js +545 -0
  25. package/dist/agent-scope-Csu2B6AM.js +606 -0
  26. package/dist/archive-CWrnG1CH.js +85 -0
  27. package/dist/archive-ccN9aDgq.js +85 -0
  28. package/dist/audit-DXPkQ275.js +1686 -0
  29. package/dist/audit-eH7nwgsM.js +1686 -0
  30. package/dist/auth-health-Bj7Gjbv0.js +149 -0
  31. package/dist/auth-health-CjjJhHey.js +149 -0
  32. package/dist/auth-nnRYiqpH.js +192 -0
  33. package/dist/auth-profiles-DVLfuixr.js +2954 -0
  34. package/dist/auth-y1BLPUhX.js +192 -0
  35. package/dist/boolean-Wzu0-e0P.js +30 -0
  36. package/dist/brew-Cqi8b49_.js +46 -0
  37. package/dist/brew-DyBGNK8A.js +46 -0
  38. package/dist/build-info.json +5 -0
  39. package/dist/call-B7EveN4V.js +252 -0
  40. package/dist/call-vuUQIjOj.js +252 -0
  41. package/dist/canvas-host/a2ui/.bundle.hash +1 -0
  42. package/dist/canvas-host/a2ui/a2ui.bundle.js +17780 -0
  43. package/dist/canvas-host/a2ui/index.html +307 -0
  44. package/dist/channel-options-CJfmOkol.js +32 -0
  45. package/dist/channel-options-lFguMTz1.js +62 -0
  46. package/dist/channel-selection-BmND9mWj.js +51 -0
  47. package/dist/channel-selection-cGhL9G0c.js +51 -0
  48. package/dist/channel-summary-B4G513Eb.js +1154 -0
  49. package/dist/channel-summary-MCwBCa5y.js +1154 -0
  50. package/dist/channels-cli-C2Of1mZG.js +1411 -0
  51. package/dist/channels-cli-DFynrP1H.js +1413 -0
  52. package/dist/channels-status-issues-YohTjZ-I.js +18 -0
  53. package/dist/channels-status-issues-ZcR1U-m5.js +18 -0
  54. package/dist/chrome-B3IuUad-.js +1953 -0
  55. package/dist/chrome-BZ9K48w9.js +1973 -0
  56. package/dist/clack-prompter-B9yLhyOm.js +92 -0
  57. package/dist/clack-prompter-BybM9xdL.js +92 -0
  58. package/dist/cli/daemon-cli.js +2 -0
  59. package/dist/cli-BJZdJwug.js +89 -0
  60. package/dist/cli-BQSoKu3d.js +86 -0
  61. package/dist/cli-utils-CO4jEMn0.js +43 -0
  62. package/dist/cli-utils-gtE-0a0D.js +43 -0
  63. package/dist/client-DPqNOpK3.js +1609 -0
  64. package/dist/client-DySXIFCA.js +1609 -0
  65. package/dist/command-format-CFzL448l.js +52 -0
  66. package/dist/command-format-DELazozB.js +52 -0
  67. package/dist/command-format-ayFsmwwz.js +38 -0
  68. package/dist/command-options-CsjxK4cZ.js +33 -0
  69. package/dist/commands-BRe9VTyU.js +229 -0
  70. package/dist/completion-cli-DlkjK0iC.js +390 -0
  71. package/dist/completion-cli-E6Pt41AL.js +387 -0
  72. package/dist/config-BtSTwPcH.js +4882 -0
  73. package/dist/config-DfMIMT-f.js +4881 -0
  74. package/dist/config-_d7_WcRv.js +5623 -0
  75. package/dist/config-guard-wSnm-U8a.js +5628 -0
  76. package/dist/configure-CPHAFKlg.js +895 -0
  77. package/dist/configure-DK1XgXYx.js +894 -0
  78. package/dist/constants-CNTiY-ZN.js +65 -0
  79. package/dist/constants-hpmbslG7.js +65 -0
  80. package/dist/control-service-CqX5g_ko.js +61 -0
  81. package/dist/control-service-o6xe3hEb.js +61 -0
  82. package/dist/control-ui/apple-touch-icon.png +0 -0
  83. package/dist/control-ui/assets/index-RwcW4Xl0.css +1 -0
  84. package/dist/control-ui/assets/index-ryaCcbyp.js +4584 -0
  85. package/dist/control-ui/assets/index-ryaCcbyp.js.map +1 -0
  86. package/dist/control-ui/favicon-32.png +0 -0
  87. package/dist/control-ui/favicon.ico +0 -0
  88. package/dist/control-ui/favicon.svg +22 -0
  89. package/dist/control-ui/index.html +17 -0
  90. package/dist/cron-cli-BoSaDgvH.js +453 -0
  91. package/dist/cron-cli-qVandvsD.js +456 -0
  92. package/dist/daemon-cli-C4gGWa15.js +760 -0
  93. package/dist/daemon-cli-DV_X0Krf.js +761 -0
  94. package/dist/daemon-runtime-D5hbSrdO.js +460 -0
  95. package/dist/daemon-runtime-DXUfrXBC.js +460 -0
  96. package/dist/deliver-BxK5nI2P.js +2587 -0
  97. package/dist/deliver-Dwzg9LUd.js +2557 -0
  98. package/dist/deliver-qpYZp20m.js +2587 -0
  99. package/dist/deps-CUXtMV9d.js +27 -0
  100. package/dist/deps-WBvZpFV_.js +27 -0
  101. package/dist/devices-cli-B2s18Qrh.js +207 -0
  102. package/dist/devices-cli-D1UEtMUJ.js +204 -0
  103. package/dist/directory-cli-BXQbrkfM.js +247 -0
  104. package/dist/directory-cli-CqA7tRbq.js +244 -0
  105. package/dist/dispatcher-Dyv7T-1r.js +160 -0
  106. package/dist/dns-cli-CWxKD22D.js +198 -0
  107. package/dist/dns-cli-CzKr_Fxj.js +201 -0
  108. package/dist/docs-cli-BNyxUbWr.js +159 -0
  109. package/dist/docs-cli-DYpTbo3i.js +161 -0
  110. package/dist/doctor-C8fDYZLq.js +2583 -0
  111. package/dist/doctor-hYcqp7c0.js +2585 -0
  112. package/dist/entry.js +1326 -0
  113. package/dist/env-l7QVNj6j.js +32 -0
  114. package/dist/errors-CMCg46fK.js +1952 -0
  115. package/dist/exec-B8JKbXKW.js +246 -0
  116. package/dist/exec-BMnoMcZW.js +1099 -0
  117. package/dist/exec-HEWTMJ7j.js +246 -0
  118. package/dist/exec-approvals-DtrnHx6M.js +1026 -0
  119. package/dist/exec-approvals-Y42bE8ud.js +1026 -0
  120. package/dist/exec-approvals-cli-CIedYxP3.js +385 -0
  121. package/dist/exec-approvals-cli-xpbxnj4O.js +388 -0
  122. package/dist/extensionAPI.js +66572 -0
  123. package/dist/format-Dzy9uRLE.js +34 -0
  124. package/dist/format-sj0fELBK.js +34 -0
  125. package/dist/gateway-cli-5A-KNLEC.js +16635 -0
  126. package/dist/gateway-cli-B8kS4chb.js +16637 -0
  127. package/dist/gateway-rpc-15n38Ize.js +28 -0
  128. package/dist/gateway-rpc-TMVRgGfj.js +28 -0
  129. package/dist/github-copilot-auth-DVZj4Zgh.js +1104 -0
  130. package/dist/github-copilot-auth-DeGYyLY9.js +1104 -0
  131. package/dist/github-copilot-token-B3SA95yo.js +103 -0
  132. package/dist/github-copilot-token-C8XFYz0i.js +103 -0
  133. package/dist/github-copilot-token-CnxakiSC.js +103 -0
  134. package/dist/gmail-setup-utils-CWPC386a.js +428 -0
  135. package/dist/gmail-setup-utils-eJVB5Ewp.js +428 -0
  136. package/dist/health-format-B0tMTk3C.js +1189 -0
  137. package/dist/health-format-DaURVaUn.js +1188 -0
  138. package/dist/help-format-CEsRHU2f.js +17 -0
  139. package/dist/help-format-GuCWws6r.js +17 -0
  140. package/dist/helpers-BEJ-phFf.js +25 -0
  141. package/dist/helpers-BtbBZVKZ.js +10 -0
  142. package/dist/helpers-C12w9zxf.js +10 -0
  143. package/dist/helpers-CzjGJZmJ.js +25 -0
  144. package/dist/hooks/bundled/boot-md/HOOK.md +19 -0
  145. package/dist/hooks/bundled/command-logger/HOOK.md +122 -0
  146. package/dist/hooks/bundled/session-memory/HOOK.md +109 -0
  147. package/dist/hooks/bundled/soul-evil/HOOK.md +71 -0
  148. package/dist/hooks-cli-BZCMAnW2.js +1058 -0
  149. package/dist/hooks-cli-D0CEFg3P.js +1061 -0
  150. package/dist/hooks-status-Bn7_O8PM.js +443 -0
  151. package/dist/hooks-status-BrWVfIn0.js +443 -0
  152. package/dist/image-BS022pvv.js +1421 -0
  153. package/dist/image-BzJtY34J.js +629 -0
  154. package/dist/image-CBqmIbQQ.js +629 -0
  155. package/dist/index.js +5809 -0
  156. package/dist/installs-BP4K5L33.js +388 -0
  157. package/dist/installs-DOpTt7VZ.js +388 -0
  158. package/dist/is-main-B4o72sqg.js +25 -0
  159. package/dist/is-main-PYGa3tDA.js +25 -0
  160. package/dist/links-B4nk2iDf.js +15 -0
  161. package/dist/links-DwjRqxgR.js +15 -0
  162. package/dist/loader-Cn4EV_pf.js +63690 -0
  163. package/dist/logging-CS3tbYDj.js +15 -0
  164. package/dist/logging-CY-Q5cwf.js +1 -0
  165. package/dist/logging-DhiLkhLw.js +15 -0
  166. package/dist/logging-pqyrk15z.js +1 -0
  167. package/dist/login-qr-CD164Aw1.js +478 -0
  168. package/dist/login-qr-D7Zdgji2.js +478 -0
  169. package/dist/login-qr-YgILJ4VC.js +475 -0
  170. package/dist/logs-cli-BdS0Uv0I.js +227 -0
  171. package/dist/logs-cli-CqSN1GzB.js +230 -0
  172. package/dist/manager-CfGY5zND.js +2870 -0
  173. package/dist/manager-CjuBqFRL.js +2870 -0
  174. package/dist/manager-CoBEAQKm.js +2872 -0
  175. package/dist/manifest-registry-Bwjq9Iev.js +668 -0
  176. package/dist/manifest-registry-D2Yntqcb.js +668 -0
  177. package/dist/message-channel-Cjsiqxok.js +105 -0
  178. package/dist/message-channel-D6v_oPAg.js +105 -0
  179. package/dist/model-selection-Cv5Ox_tY.js +2956 -0
  180. package/dist/model-selection-Dr-5U5-l.js +2708 -0
  181. package/dist/models-cli-B39ckynD.js +2541 -0
  182. package/dist/models-cli-DoiYsBYw.js +2544 -0
  183. package/dist/net-CFCxaipF.js +137 -0
  184. package/dist/net-DKJPqXuW.js +137 -0
  185. package/dist/node-cli-C_FYF-RA.js +1456 -0
  186. package/dist/node-cli-DWPoNsQS.js +1459 -0
  187. package/dist/node-service-DcJREOww.js +67 -0
  188. package/dist/node-service-DuZ9Us9h.js +67 -0
  189. package/dist/nodes-cli-Elo6tlen.js +1210 -0
  190. package/dist/nodes-cli-zqryRUWB.js +1207 -0
  191. package/dist/nodes-screen-C4aCrxie.js +157 -0
  192. package/dist/nodes-screen-D4PSynkR.js +157 -0
  193. package/dist/note-CQhSvgQv.js +73 -0
  194. package/dist/note-_C44YfAQ.js +73 -0
  195. package/dist/onboard-channels-CHBDi-ZA.js +670 -0
  196. package/dist/onboard-channels-DOEKyxaL.js +670 -0
  197. package/dist/onboard-skills-BUTXREDZ.js +3327 -0
  198. package/dist/onboard-skills-CSLYZmZA.js +3327 -0
  199. package/dist/onboarding-CgKb8b39.js +3232 -0
  200. package/dist/openclaw-root-9ILYSmJ9.js +84 -0
  201. package/dist/openclaw-root-Cvotktkd.js +84 -0
  202. package/dist/pairing-cli-B4UGR2at.js +114 -0
  203. package/dist/pairing-cli-BWDDl8cf.js +117 -0
  204. package/dist/pairing-labels-ClZ-fTWT.js +9 -0
  205. package/dist/pairing-labels-Ds7BPOkj.js +9 -0
  206. package/dist/pairing-store-DDLNuzmx.js +391 -0
  207. package/dist/pairing-store-DRn08lZD.js +391 -0
  208. package/dist/parse-87ybtYW1.js +23 -0
  209. package/dist/parse-OCFfznr3.js +23 -0
  210. package/dist/parse-log-line-C9aL5PUL.js +44 -0
  211. package/dist/parse-log-line-DxRaGzQb.js +44 -0
  212. package/dist/parse-timeout-CFqNj7No.js +16 -0
  213. package/dist/parse-timeout-DV8NQQWk.js +16 -0
  214. package/dist/path-env-C7kiJUgG.js +77 -0
  215. package/dist/path-env-DEj4CiFN.js +77 -0
  216. package/dist/paths-B-q1nXdY.js +43 -0
  217. package/dist/paths-B1kfl4h5.js +164 -0
  218. package/dist/paths-B4kigINg.js +40 -0
  219. package/dist/paths-CHGbP1-A.js +43 -0
  220. package/dist/paths-scjhy7N2.js +180 -0
  221. package/dist/pi-embedded-helpers-C19wUpMB.js +8451 -0
  222. package/dist/pi-embedded-helpers-CT5VuLCb.js +1293 -0
  223. package/dist/pi-embedded-helpers-Dl8e5Rf8.js +1293 -0
  224. package/dist/pi-model-discovery-B6CsmK6Y.js +20 -0
  225. package/dist/pi-model-discovery-DsRqYJLy.js +20 -0
  226. package/dist/pi-model-discovery-EhM2JAQo.js +20 -0
  227. package/dist/pi-tools.policy-BvkSDFDN.js +229 -0
  228. package/dist/plugin-auto-enable-Bd_StZzz.js +461 -0
  229. package/dist/plugin-auto-enable-DBhXb_0x.js +461 -0
  230. package/dist/plugin-sdk/agent-scope-DdwUKIOe.js +606 -0
  231. package/dist/plugin-sdk/chrome-G8apFa5p.js +1953 -0
  232. package/dist/plugin-sdk/command-format-qUVxzqYm.js +52 -0
  233. package/dist/plugin-sdk/config-Cm1M7tgH.js +5623 -0
  234. package/dist/plugin-sdk/deliver-Cl8uowiO.js +2557 -0
  235. package/dist/plugin-sdk/exec-Cm9b2r9Q.js +1107 -0
  236. package/dist/plugin-sdk/github-copilot-token-BHNcM4_B.js +103 -0
  237. package/dist/plugin-sdk/image-7PgoS2VD.js +1421 -0
  238. package/dist/plugin-sdk/index.d.ts +8908 -0
  239. package/dist/plugin-sdk/index.js +70888 -0
  240. package/dist/plugin-sdk/login-qr-qTALvWi2.js +475 -0
  241. package/dist/plugin-sdk/manager-Cs3EQZCb.js +2870 -0
  242. package/dist/plugin-sdk/model-selection-BgC1E1a7.js +2708 -0
  243. package/dist/plugin-sdk/paths-BYpoyRv5.js +164 -0
  244. package/dist/plugin-sdk/paths-DNQE-bvr.js +40 -0
  245. package/dist/plugin-sdk/pi-embedded-helpers-5jNqW_dE.js +8755 -0
  246. package/dist/plugin-sdk/pi-model-discovery-BUGEht9A.js +20 -0
  247. package/dist/plugin-sdk/pw-ai-COTtei4a.js +1649 -0
  248. package/dist/plugin-sdk/qmd-manager-ClSwiAJl.js +615 -0
  249. package/dist/plugin-sdk/redact-2AzjOfk2.js +94 -0
  250. package/dist/plugin-sdk/rolldown-runtime-Cbj13DAv.js +20 -0
  251. package/dist/plugin-sdk/sqlite-gCW7MlLs.js +215 -0
  252. package/dist/plugin-sdk/transcript-events-DGF257vD.js +17 -0
  253. package/dist/plugins-C3Bm-HQV.js +494 -0
  254. package/dist/plugins-QJjTXliB.js +495 -0
  255. package/dist/plugins-cli-DTci0JQb.js +443 -0
  256. package/dist/plugins-cli-wJsN1HHK.js +440 -0
  257. package/dist/ports-CiW9dmMq.js +96 -0
  258. package/dist/program-BWpTHh1I.js +188 -0
  259. package/dist/progress-Bcjniu7m.js +133 -0
  260. package/dist/progress-CvaSPjS9.js +133 -0
  261. package/dist/prompt-style-CFsleyxV.js +9 -0
  262. package/dist/prompt-style-DYJdrXyV.js +9 -0
  263. package/dist/prompts-Bt9fwsg2.js +10 -0
  264. package/dist/prompts-CudpZgTI.js +10 -0
  265. package/dist/pw-ai-08F3GD-3.js +1649 -0
  266. package/dist/pw-ai-ZmHxHQnx.js +1651 -0
  267. package/dist/pw-ai-tNPuRNn3.js +1649 -0
  268. package/dist/qmd-manager-2r-4n3sP.js +617 -0
  269. package/dist/qmd-manager-CF52nuBg.js +615 -0
  270. package/dist/qmd-manager-HEm5H2mk.js +616 -0
  271. package/dist/redact-BICFkpn7.js +97 -0
  272. package/dist/redact-BIMJ3ntQ.js +94 -0
  273. package/dist/redact-KzWHRS5J.js +97 -0
  274. package/dist/register.subclis-D2K25c84.js +348 -0
  275. package/dist/register.subclis-Dd8LbOLi.js +342 -0
  276. package/dist/reply-5UNWRwMn.js +63693 -0
  277. package/dist/restart-sentinel-Cr0vUxB8.js +65 -0
  278. package/dist/restart-sentinel-DUemCjgU.js +65 -0
  279. package/dist/rolldown-runtime-Cbj13DAv.js +20 -0
  280. package/dist/routes-C6UpTPas.js +2410 -0
  281. package/dist/routes-ClNyEvlm.js +2410 -0
  282. package/dist/rpc-D0mf7DIw.js +95 -0
  283. package/dist/rpc-DYdOrgd9.js +95 -0
  284. package/dist/run-main-CojI7gWx.js +194 -0
  285. package/dist/runtime-guard-68M_evhb.js +60 -0
  286. package/dist/runtime-guard-DkjmhnBD.js +60 -0
  287. package/dist/sandbox-Ca81z3Tw.js +2924 -0
  288. package/dist/sandbox-cli-D75GApgp.js +459 -0
  289. package/dist/sandbox-cli-E4SJsC1C.js +462 -0
  290. package/dist/sandbox-knontqD9.js +2925 -0
  291. package/dist/security-cli-BLihvXO-.js +503 -0
  292. package/dist/security-cli-IGQCsK4g.js +506 -0
  293. package/dist/server-context-B9GX5GOI.js +740 -0
  294. package/dist/server-context-BFH7HB_M.js +740 -0
  295. package/dist/server-node-events-CTdHBiEA.js +218 -0
  296. package/dist/server-node-events-DAV14qPr.js +215 -0
  297. package/dist/service-BZNBq9hq.js +680 -0
  298. package/dist/service-C-BLXx9U.js +680 -0
  299. package/dist/service-audit-BfJv4NqZ.js +542 -0
  300. package/dist/service-audit-Bw3M2OEI.js +542 -0
  301. package/dist/shared-5SH-45AX.js +74 -0
  302. package/dist/shared-BxRm5uLU.js +74 -0
  303. package/dist/shared-C80Rmxsd.js +150 -0
  304. package/dist/shared-fGK6_D2v.js +150 -0
  305. package/dist/skills-Bhp0l6UK.js +693 -0
  306. package/dist/skills-Tky2kCMO.js +694 -0
  307. package/dist/skills-cli-6rCClAE4.js +287 -0
  308. package/dist/skills-cli-C4nLCrLw.js +290 -0
  309. package/dist/skills-status-CENcKr3I.js +187 -0
  310. package/dist/skills-status-DX1eUYvk.js +187 -0
  311. package/dist/sqlite-CmdZSZRx.js +197 -0
  312. package/dist/sqlite-Dnmf3LS7.js +215 -0
  313. package/dist/sqlite-QDf0yuU0.js +215 -0
  314. package/dist/status-BSfGAp2D.js +27 -0
  315. package/dist/status-Bp_2NMjt.js +27 -0
  316. package/dist/status-C0ANDr0T.js +3140 -0
  317. package/dist/status-CCHBIZnm.js +21 -0
  318. package/dist/status-Vuqbw2Bb.js +21 -0
  319. package/dist/status.update-BZW5r8Su.js +79 -0
  320. package/dist/status.update-BnD93_O8.js +79 -0
  321. package/dist/subsystem-CAq3uyo7.js +834 -0
  322. package/dist/system-cli-Bb9zmCO1.js +83 -0
  323. package/dist/system-cli-TIIQ04ls.js +80 -0
  324. package/dist/systemd-0Qa_nGqe.js +438 -0
  325. package/dist/systemd-Czb0Xsm7.js +438 -0
  326. package/dist/systemd-hints-CWoEOQRb.js +19 -0
  327. package/dist/systemd-hints-Cv3RN_mZ.js +19 -0
  328. package/dist/systemd-linger-CsdvcIoS.js +75 -0
  329. package/dist/systemd-linger-DKUFHcLn.js +75 -0
  330. package/dist/table-DNPESyNj.js +279 -0
  331. package/dist/table-DS4-gmkV.js +279 -0
  332. package/dist/tailnet-Bg_vE5qi.js +42 -0
  333. package/dist/tailnet-CrNWlQRJ.js +42 -0
  334. package/dist/tailscale-CBv58toW.js +252 -0
  335. package/dist/tailscale-DCnMs7_q.js +225 -0
  336. package/dist/tool-display-BEACy9rK.js +795 -0
  337. package/dist/tool-display-NYQnSpdo.js +795 -0
  338. package/dist/transcript-events-CsB1Saa6.js +17 -0
  339. package/dist/transcript-events-DDYvbmRV.js +17 -0
  340. package/dist/transcript-events-JLH5W4He.js +17 -0
  341. package/dist/tui--NY0rnjr.js +2542 -0
  342. package/dist/tui-DqJfGtvM.js +2543 -0
  343. package/dist/tui-cli-BuHNY6wF.js +54 -0
  344. package/dist/tui-cli-LMFV982e.js +57 -0
  345. package/dist/update-CRpHtCgz.js +317 -0
  346. package/dist/update-D3qruxhj.js +317 -0
  347. package/dist/update-cli-CFF-pslM.js +948 -0
  348. package/dist/update-cli-cn9pEMX7.js +951 -0
  349. package/dist/update-runner-CxGU142L.js +1221 -0
  350. package/dist/update-runner-DNobz_ft.js +1221 -0
  351. package/dist/utils-CKSrBNwq.js +192 -0
  352. package/dist/utils-DX85MiPR.js +188 -0
  353. package/dist/webhooks-cli-BGtt2HAR.js +312 -0
  354. package/dist/webhooks-cli-DHLZrEO_.js +309 -0
  355. package/dist/widearea-dns-BpG7ATO8.js +127 -0
  356. package/dist/widearea-dns-D4wkCJly.js +127 -0
  357. package/dist/ws-3zr8WUwL.js +13 -0
  358. package/dist/ws-log-BXcT2xQk.js +267 -0
  359. package/dist/ws-log-DbDIUsgz.js +267 -0
  360. package/dist/ws-lzrgabja.js +13 -0
  361. package/dist/wsl-D2O2qOrl.js +160 -0
  362. package/docs/.i18n/README.md +31 -0
  363. package/docs/.i18n/glossary.zh-CN.json +190 -0
  364. package/docs/.i18n/zh-CN.tm.jsonl +1329 -0
  365. package/docs/CNAME +1 -0
  366. package/docs/_config.yml +53 -0
  367. package/docs/_layouts/default.html +145 -0
  368. package/docs/assets/markdown.css +179 -0
  369. package/docs/assets/openclaw-logo-text-dark.png +0 -0
  370. package/docs/assets/openclaw-logo-text.png +0 -0
  371. package/docs/assets/pixel-lobster.svg +60 -0
  372. package/docs/assets/showcase/agents-ui.jpg +0 -0
  373. package/docs/assets/showcase/bambu-cli.png +0 -0
  374. package/docs/assets/showcase/codexmonitor.png +0 -0
  375. package/docs/assets/showcase/gohome-grafana.png +0 -0
  376. package/docs/assets/showcase/ios-testflight.jpg +0 -0
  377. package/docs/assets/showcase/oura-health.png +0 -0
  378. package/docs/assets/showcase/padel-cli.svg +11 -0
  379. package/docs/assets/showcase/padel-screenshot.jpg +0 -0
  380. package/docs/assets/showcase/papla-tts.jpg +0 -0
  381. package/docs/assets/showcase/pr-review-telegram.jpg +0 -0
  382. package/docs/assets/showcase/roborock-screenshot.jpg +0 -0
  383. package/docs/assets/showcase/roborock-status.svg +13 -0
  384. package/docs/assets/showcase/roof-camera-sky.jpg +0 -0
  385. package/docs/assets/showcase/snag.png +0 -0
  386. package/docs/assets/showcase/tesco-shop.jpg +0 -0
  387. package/docs/assets/showcase/wienerlinien.png +0 -0
  388. package/docs/assets/showcase/wine-cellar-skill.jpg +0 -0
  389. package/docs/assets/showcase/winix-air-purifier.jpg +0 -0
  390. package/docs/assets/showcase/xuezh-pronunciation.jpeg +0 -0
  391. package/docs/assets/terminal.css +473 -0
  392. package/docs/assets/theme.js +55 -0
  393. package/docs/automation/auth-monitoring.md +44 -0
  394. package/docs/automation/cron-jobs.md +468 -0
  395. package/docs/automation/cron-vs-heartbeat.md +282 -0
  396. package/docs/automation/gmail-pubsub.md +256 -0
  397. package/docs/automation/poll.md +69 -0
  398. package/docs/automation/webhook.md +163 -0
  399. package/docs/bedrock.md +176 -0
  400. package/docs/brave-search.md +41 -0
  401. package/docs/broadcast-groups.md +442 -0
  402. package/docs/channels/bluebubbles.md +338 -0
  403. package/docs/channels/discord.md +475 -0
  404. package/docs/channels/feishu.md +507 -0
  405. package/docs/channels/googlechat.md +250 -0
  406. package/docs/channels/grammy.md +31 -0
  407. package/docs/channels/imessage.md +299 -0
  408. package/docs/channels/index.md +46 -0
  409. package/docs/channels/line.md +186 -0
  410. package/docs/channels/location.md +56 -0
  411. package/docs/channels/matrix.md +233 -0
  412. package/docs/channels/mattermost.md +138 -0
  413. package/docs/channels/msteams.md +768 -0
  414. package/docs/channels/nextcloud-talk.md +136 -0
  415. package/docs/channels/nostr.md +233 -0
  416. package/docs/channels/signal.md +202 -0
  417. package/docs/channels/slack.md +548 -0
  418. package/docs/channels/telegram.md +750 -0
  419. package/docs/channels/tlon.md +132 -0
  420. package/docs/channels/troubleshooting.md +29 -0
  421. package/docs/channels/twitch.md +379 -0
  422. package/docs/channels/whatsapp.md +404 -0
  423. package/docs/channels/zalo.md +189 -0
  424. package/docs/channels/zalouser.md +140 -0
  425. package/docs/cli/acp.md +170 -0
  426. package/docs/cli/agent.md +24 -0
  427. package/docs/cli/agents.md +75 -0
  428. package/docs/cli/approvals.md +50 -0
  429. package/docs/cli/browser.md +107 -0
  430. package/docs/cli/channels.md +79 -0
  431. package/docs/cli/config.md +50 -0
  432. package/docs/cli/configure.md +33 -0
  433. package/docs/cli/cron.md +42 -0
  434. package/docs/cli/dashboard.md +16 -0
  435. package/docs/cli/devices.md +67 -0
  436. package/docs/cli/directory.md +63 -0
  437. package/docs/cli/dns.md +23 -0
  438. package/docs/cli/docs.md +15 -0
  439. package/docs/cli/doctor.md +41 -0
  440. package/docs/cli/gateway.md +199 -0
  441. package/docs/cli/health.md +21 -0
  442. package/docs/cli/hooks.md +304 -0
  443. package/docs/cli/index.md +1029 -0
  444. package/docs/cli/logs.md +24 -0
  445. package/docs/cli/memory.md +45 -0
  446. package/docs/cli/message.md +239 -0
  447. package/docs/cli/models.md +79 -0
  448. package/docs/cli/node.md +112 -0
  449. package/docs/cli/nodes.md +73 -0
  450. package/docs/cli/onboard.md +29 -0
  451. package/docs/cli/pairing.md +21 -0
  452. package/docs/cli/plugins.md +62 -0
  453. package/docs/cli/reset.md +17 -0
  454. package/docs/cli/sandbox.md +152 -0
  455. package/docs/cli/security.md +26 -0
  456. package/docs/cli/sessions.md +16 -0
  457. package/docs/cli/setup.md +29 -0
  458. package/docs/cli/skills.md +26 -0
  459. package/docs/cli/status.md +26 -0
  460. package/docs/cli/system.md +60 -0
  461. package/docs/cli/tui.md +23 -0
  462. package/docs/cli/uninstall.md +17 -0
  463. package/docs/cli/update.md +98 -0
  464. package/docs/cli/voicecall.md +34 -0
  465. package/docs/cli/webhooks.md +25 -0
  466. package/docs/concepts/agent-loop.md +146 -0
  467. package/docs/concepts/agent-workspace.md +233 -0
  468. package/docs/concepts/agent.md +123 -0
  469. package/docs/concepts/architecture.md +129 -0
  470. package/docs/concepts/channel-routing.md +114 -0
  471. package/docs/concepts/compaction.md +61 -0
  472. package/docs/concepts/context.md +161 -0
  473. package/docs/concepts/group-messages.md +84 -0
  474. package/docs/concepts/groups.md +373 -0
  475. package/docs/concepts/markdown-formatting.md +130 -0
  476. package/docs/concepts/memory.md +546 -0
  477. package/docs/concepts/messages.md +154 -0
  478. package/docs/concepts/model-failover.md +149 -0
  479. package/docs/concepts/model-providers.md +316 -0
  480. package/docs/concepts/models.md +208 -0
  481. package/docs/concepts/multi-agent.md +376 -0
  482. package/docs/concepts/oauth.md +145 -0
  483. package/docs/concepts/presence.md +102 -0
  484. package/docs/concepts/queue.md +89 -0
  485. package/docs/concepts/retry.md +69 -0
  486. package/docs/concepts/session-pruning.md +122 -0
  487. package/docs/concepts/session-tool.md +193 -0
  488. package/docs/concepts/session.md +188 -0
  489. package/docs/concepts/sessions.md +10 -0
  490. package/docs/concepts/streaming.md +135 -0
  491. package/docs/concepts/system-prompt.md +115 -0
  492. package/docs/concepts/timezone.md +91 -0
  493. package/docs/concepts/typebox.md +289 -0
  494. package/docs/concepts/typing-indicators.md +68 -0
  495. package/docs/concepts/usage-tracking.md +35 -0
  496. package/docs/date-time.md +128 -0
  497. package/docs/debug/node-issue.md +83 -0
  498. package/docs/debugging.md +162 -0
  499. package/docs/diagnostics/flags.md +91 -0
  500. package/docs/docs.json +1587 -0
  501. package/docs/environment.md +81 -0
  502. package/docs/experiments/onboarding-config-protocol.md +40 -0
  503. package/docs/experiments/plans/cron-add-hardening.md +63 -0
  504. package/docs/experiments/plans/group-policy-hardening.md +40 -0
  505. package/docs/experiments/plans/openresponses-gateway.md +123 -0
  506. package/docs/experiments/proposals/model-config.md +36 -0
  507. package/docs/experiments/research/memory.md +228 -0
  508. package/docs/gateway/authentication.md +145 -0
  509. package/docs/gateway/background-process.md +93 -0
  510. package/docs/gateway/bonjour.md +167 -0
  511. package/docs/gateway/bridge-protocol.md +89 -0
  512. package/docs/gateway/cli-backends.md +223 -0
  513. package/docs/gateway/configuration-examples.md +606 -0
  514. package/docs/gateway/configuration.md +3393 -0
  515. package/docs/gateway/discovery.md +116 -0
  516. package/docs/gateway/doctor.md +282 -0
  517. package/docs/gateway/gateway-lock.md +34 -0
  518. package/docs/gateway/health.md +35 -0
  519. package/docs/gateway/heartbeat.md +302 -0
  520. package/docs/gateway/index.md +328 -0
  521. package/docs/gateway/local-models.md +150 -0
  522. package/docs/gateway/logging.md +113 -0
  523. package/docs/gateway/multiple-gateways.md +112 -0
  524. package/docs/gateway/openai-http-api.md +118 -0
  525. package/docs/gateway/openresponses-http-api.md +317 -0
  526. package/docs/gateway/pairing.md +99 -0
  527. package/docs/gateway/protocol.md +221 -0
  528. package/docs/gateway/remote-gateway-readme.md +157 -0
  529. package/docs/gateway/remote.md +127 -0
  530. package/docs/gateway/sandbox-vs-tool-policy-vs-elevated.md +128 -0
  531. package/docs/gateway/sandboxing.md +193 -0
  532. package/docs/gateway/security/formal-verification.md +164 -0
  533. package/docs/gateway/security/index.md +825 -0
  534. package/docs/gateway/tailscale.md +127 -0
  535. package/docs/gateway/tools-invoke-http-api.md +85 -0
  536. package/docs/gateway/troubleshooting.md +767 -0
  537. package/docs/help/faq.md +2830 -0
  538. package/docs/help/index.md +21 -0
  539. package/docs/help/troubleshooting.md +98 -0
  540. package/docs/hooks/soul-evil.md +69 -0
  541. package/docs/hooks.md +913 -0
  542. package/docs/images/feishu-step2-create-app.png +0 -0
  543. package/docs/images/feishu-step3-credentials.png +0 -0
  544. package/docs/images/feishu-step4-permissions.png +0 -0
  545. package/docs/images/feishu-step5-bot-capability.png +0 -0
  546. package/docs/images/feishu-step6-event-subscription.png +0 -0
  547. package/docs/images/groups-flow.svg +52 -0
  548. package/docs/images/mobile-ui-screenshot.png +0 -0
  549. package/docs/index.md +258 -0
  550. package/docs/install/ansible.md +208 -0
  551. package/docs/install/bun.md +59 -0
  552. package/docs/install/development-channels.md +75 -0
  553. package/docs/install/docker.md +567 -0
  554. package/docs/install/index.md +185 -0
  555. package/docs/install/installer.md +123 -0
  556. package/docs/install/migrating.md +192 -0
  557. package/docs/install/nix.md +96 -0
  558. package/docs/install/node.md +78 -0
  559. package/docs/install/uninstall.md +128 -0
  560. package/docs/install/updating.md +228 -0
  561. package/docs/logging.md +350 -0
  562. package/docs/multi-agent-sandbox-tools.md +395 -0
  563. package/docs/network.md +54 -0
  564. package/docs/nodes/audio.md +114 -0
  565. package/docs/nodes/camera.md +156 -0
  566. package/docs/nodes/images.md +72 -0
  567. package/docs/nodes/index.md +341 -0
  568. package/docs/nodes/location-command.md +113 -0
  569. package/docs/nodes/media-understanding.md +379 -0
  570. package/docs/nodes/talk.md +90 -0
  571. package/docs/nodes/voicewake.md +65 -0
  572. package/docs/northflank.mdx +53 -0
  573. package/docs/perplexity.md +80 -0
  574. package/docs/pi-dev.md +70 -0
  575. package/docs/pi.md +612 -0
  576. package/docs/platforms/android.md +148 -0
  577. package/docs/platforms/digitalocean.md +262 -0
  578. package/docs/platforms/exe-dev.md +125 -0
  579. package/docs/platforms/fly.md +486 -0
  580. package/docs/platforms/gcp.md +503 -0
  581. package/docs/platforms/hetzner.md +330 -0
  582. package/docs/platforms/index.md +53 -0
  583. package/docs/platforms/ios.md +107 -0
  584. package/docs/platforms/linux.md +94 -0
  585. package/docs/platforms/mac/bundled-gateway.md +73 -0
  586. package/docs/platforms/mac/canvas.md +125 -0
  587. package/docs/platforms/mac/child-process.md +69 -0
  588. package/docs/platforms/mac/dev-setup.md +102 -0
  589. package/docs/platforms/mac/health.md +34 -0
  590. package/docs/platforms/mac/icon.md +31 -0
  591. package/docs/platforms/mac/logging.md +57 -0
  592. package/docs/platforms/mac/menu-bar.md +81 -0
  593. package/docs/platforms/mac/peekaboo.md +65 -0
  594. package/docs/platforms/mac/permissions.md +44 -0
  595. package/docs/platforms/mac/release.md +85 -0
  596. package/docs/platforms/mac/remote.md +83 -0
  597. package/docs/platforms/mac/signing.md +47 -0
  598. package/docs/platforms/mac/skills.md +33 -0
  599. package/docs/platforms/mac/voice-overlay.md +60 -0
  600. package/docs/platforms/mac/voicewake.md +67 -0
  601. package/docs/platforms/mac/webchat.md +41 -0
  602. package/docs/platforms/mac/xpc.md +61 -0
  603. package/docs/platforms/macos-vm.md +281 -0
  604. package/docs/platforms/macos.md +203 -0
  605. package/docs/platforms/oracle.md +303 -0
  606. package/docs/platforms/raspberry-pi.md +358 -0
  607. package/docs/platforms/windows.md +159 -0
  608. package/docs/plugin.md +664 -0
  609. package/docs/plugins/agent-tools.md +99 -0
  610. package/docs/plugins/manifest.md +71 -0
  611. package/docs/plugins/voice-call.md +284 -0
  612. package/docs/plugins/zalouser.md +81 -0
  613. package/docs/prose.md +134 -0
  614. package/docs/providers/anthropic.md +152 -0
  615. package/docs/providers/claude-max-api-proxy.md +148 -0
  616. package/docs/providers/cloudflare-ai-gateway.md +71 -0
  617. package/docs/providers/deepgram.md +93 -0
  618. package/docs/providers/github-copilot.md +72 -0
  619. package/docs/providers/glm.md +33 -0
  620. package/docs/providers/index.md +63 -0
  621. package/docs/providers/minimax.md +208 -0
  622. package/docs/providers/models.md +51 -0
  623. package/docs/providers/moonshot.md +142 -0
  624. package/docs/providers/ollama.md +223 -0
  625. package/docs/providers/openai.md +62 -0
  626. package/docs/providers/opencode.md +36 -0
  627. package/docs/providers/openrouter.md +37 -0
  628. package/docs/providers/qwen.md +53 -0
  629. package/docs/providers/synthetic.md +99 -0
  630. package/docs/providers/venice.md +267 -0
  631. package/docs/providers/vercel-ai-gateway.md +50 -0
  632. package/docs/providers/xiaomi.md +64 -0
  633. package/docs/providers/zai.md +36 -0
  634. package/docs/railway.mdx +99 -0
  635. package/docs/refactor/clawnet.md +417 -0
  636. package/docs/refactor/exec-host.md +316 -0
  637. package/docs/refactor/outbound-session-mirroring.md +85 -0
  638. package/docs/refactor/plugin-sdk.md +214 -0
  639. package/docs/refactor/strict-config.md +93 -0
  640. package/docs/reference/AGENTS.default.md +124 -0
  641. package/docs/reference/RELEASING.md +120 -0
  642. package/docs/reference/api-usage-costs.md +137 -0
  643. package/docs/reference/device-models.md +47 -0
  644. package/docs/reference/rpc.md +43 -0
  645. package/docs/reference/session-management-compaction.md +285 -0
  646. package/docs/reference/templates/AGENTS.dev.md +83 -0
  647. package/docs/reference/templates/AGENTS.md +218 -0
  648. package/docs/reference/templates/BOOT.md +10 -0
  649. package/docs/reference/templates/BOOTSTRAP.md +61 -0
  650. package/docs/reference/templates/HEARTBEAT.md +11 -0
  651. package/docs/reference/templates/IDENTITY.dev.md +47 -0
  652. package/docs/reference/templates/IDENTITY.md +27 -0
  653. package/docs/reference/templates/SOUL.dev.md +76 -0
  654. package/docs/reference/templates/SOUL.md +42 -0
  655. package/docs/reference/templates/TOOLS.dev.md +24 -0
  656. package/docs/reference/templates/TOOLS.md +46 -0
  657. package/docs/reference/templates/USER.dev.md +18 -0
  658. package/docs/reference/templates/USER.md +22 -0
  659. package/docs/reference/test.md +50 -0
  660. package/docs/reference/transcript-hygiene.md +129 -0
  661. package/docs/render.mdx +165 -0
  662. package/docs/scripts.md +28 -0
  663. package/docs/security/formal-verification.md +164 -0
  664. package/docs/start/getting-started.md +208 -0
  665. package/docs/start/hubs.md +185 -0
  666. package/docs/start/lore.md +219 -0
  667. package/docs/start/onboarding.md +110 -0
  668. package/docs/start/openclaw.md +241 -0
  669. package/docs/start/pairing.md +86 -0
  670. package/docs/start/setup.md +149 -0
  671. package/docs/start/showcase.md +416 -0
  672. package/docs/start/wizard.md +349 -0
  673. package/docs/testing.md +368 -0
  674. package/docs/token-use.md +112 -0
  675. package/docs/tools/agent-send.md +53 -0
  676. package/docs/tools/apply-patch.md +50 -0
  677. package/docs/tools/browser-linux-troubleshooting.md +139 -0
  678. package/docs/tools/browser-login.md +68 -0
  679. package/docs/tools/browser.md +576 -0
  680. package/docs/tools/chrome-extension.md +178 -0
  681. package/docs/tools/clawhub.md +257 -0
  682. package/docs/tools/creating-skills.md +54 -0
  683. package/docs/tools/elevated.md +57 -0
  684. package/docs/tools/exec-approvals.md +246 -0
  685. package/docs/tools/exec.md +179 -0
  686. package/docs/tools/firecrawl.md +61 -0
  687. package/docs/tools/index.md +509 -0
  688. package/docs/tools/llm-task.md +115 -0
  689. package/docs/tools/lobster.md +342 -0
  690. package/docs/tools/reactions.md +22 -0
  691. package/docs/tools/skills-config.md +76 -0
  692. package/docs/tools/skills.md +300 -0
  693. package/docs/tools/slash-commands.md +198 -0
  694. package/docs/tools/subagents.md +151 -0
  695. package/docs/tools/thinking.md +73 -0
  696. package/docs/tools/web.md +261 -0
  697. package/docs/tts.md +396 -0
  698. package/docs/tui.md +159 -0
  699. package/docs/vps.md +43 -0
  700. package/docs/web/control-ui.md +221 -0
  701. package/docs/web/dashboard.md +46 -0
  702. package/docs/web/index.md +116 -0
  703. package/docs/web/webchat.md +49 -0
  704. package/docs/whatsapp-openclaw-ai-zh.jpg +0 -0
  705. package/docs/whatsapp-openclaw.jpg +0 -0
  706. package/docs/zh-CN/AGENTS.md +59 -0
  707. package/docs/zh-CN/automation/auth-monitoring.md +47 -0
  708. package/docs/zh-CN/automation/cron-jobs.md +424 -0
  709. package/docs/zh-CN/automation/cron-vs-heartbeat.md +286 -0
  710. package/docs/zh-CN/automation/gmail-pubsub.md +249 -0
  711. package/docs/zh-CN/automation/poll.md +76 -0
  712. package/docs/zh-CN/automation/webhook.md +163 -0
  713. package/docs/zh-CN/bedrock.md +170 -0
  714. package/docs/zh-CN/brave-search.md +48 -0
  715. package/docs/zh-CN/broadcast-groups.md +449 -0
  716. package/docs/zh-CN/channels/bluebubbles.md +271 -0
  717. package/docs/zh-CN/channels/discord.md +468 -0
  718. package/docs/zh-CN/channels/feishu.md +513 -0
  719. package/docs/zh-CN/channels/googlechat.md +257 -0
  720. package/docs/zh-CN/channels/grammy.md +38 -0
  721. package/docs/zh-CN/channels/imessage.md +302 -0
  722. package/docs/zh-CN/channels/index.md +53 -0
  723. package/docs/zh-CN/channels/line.md +180 -0
  724. package/docs/zh-CN/channels/location.md +63 -0
  725. package/docs/zh-CN/channels/matrix.md +221 -0
  726. package/docs/zh-CN/channels/mattermost.md +144 -0
  727. package/docs/zh-CN/channels/msteams.md +775 -0
  728. package/docs/zh-CN/channels/nextcloud-talk.md +142 -0
  729. package/docs/zh-CN/channels/nostr.md +240 -0
  730. package/docs/zh-CN/channels/signal.md +209 -0
  731. package/docs/zh-CN/channels/slack.md +531 -0
  732. package/docs/zh-CN/channels/telegram.md +751 -0
  733. package/docs/zh-CN/channels/tlon.md +136 -0
  734. package/docs/zh-CN/channels/troubleshooting.md +36 -0
  735. package/docs/zh-CN/channels/twitch.md +385 -0
  736. package/docs/zh-CN/channels/whatsapp.md +411 -0
  737. package/docs/zh-CN/channels/zalo.md +196 -0
  738. package/docs/zh-CN/channels/zalouser.md +147 -0
  739. package/docs/zh-CN/cli/acp.md +173 -0
  740. package/docs/zh-CN/cli/agent.md +30 -0
  741. package/docs/zh-CN/cli/agents.md +82 -0
  742. package/docs/zh-CN/cli/approvals.md +57 -0
  743. package/docs/zh-CN/cli/browser.md +114 -0
  744. package/docs/zh-CN/cli/channels.md +86 -0
  745. package/docs/zh-CN/cli/config.md +57 -0
  746. package/docs/zh-CN/cli/configure.md +38 -0
  747. package/docs/zh-CN/cli/cron.md +43 -0
  748. package/docs/zh-CN/cli/dashboard.md +23 -0
  749. package/docs/zh-CN/cli/devices.md +74 -0
  750. package/docs/zh-CN/cli/directory.md +70 -0
  751. package/docs/zh-CN/cli/dns.md +30 -0
  752. package/docs/zh-CN/cli/docs.md +22 -0
  753. package/docs/zh-CN/cli/doctor.md +48 -0
  754. package/docs/zh-CN/cli/gateway.md +206 -0
  755. package/docs/zh-CN/cli/health.md +28 -0
  756. package/docs/zh-CN/cli/hooks.md +311 -0
  757. package/docs/zh-CN/cli/index.md +1032 -0
  758. package/docs/zh-CN/cli/logs.md +31 -0
  759. package/docs/zh-CN/cli/memory.md +52 -0
  760. package/docs/zh-CN/cli/message.md +246 -0
  761. package/docs/zh-CN/cli/models.md +85 -0
  762. package/docs/zh-CN/cli/node.md +115 -0
  763. package/docs/zh-CN/cli/nodes.md +80 -0
  764. package/docs/zh-CN/cli/onboard.md +36 -0
  765. package/docs/zh-CN/cli/pairing.md +28 -0
  766. package/docs/zh-CN/cli/plugins.md +66 -0
  767. package/docs/zh-CN/cli/reset.md +24 -0
  768. package/docs/zh-CN/cli/sandbox.md +158 -0
  769. package/docs/zh-CN/cli/security.md +33 -0
  770. package/docs/zh-CN/cli/sessions.md +23 -0
  771. package/docs/zh-CN/cli/setup.md +36 -0
  772. package/docs/zh-CN/cli/skills.md +33 -0
  773. package/docs/zh-CN/cli/status.md +33 -0
  774. package/docs/zh-CN/cli/system.md +63 -0
  775. package/docs/zh-CN/cli/tui.md +30 -0
  776. package/docs/zh-CN/cli/uninstall.md +24 -0
  777. package/docs/zh-CN/cli/update.md +101 -0
  778. package/docs/zh-CN/cli/voicecall.md +41 -0
  779. package/docs/zh-CN/cli/webhooks.md +32 -0
  780. package/docs/zh-CN/concepts/agent-loop.md +146 -0
  781. package/docs/zh-CN/concepts/agent-workspace.md +219 -0
  782. package/docs/zh-CN/concepts/agent.md +115 -0
  783. package/docs/zh-CN/concepts/architecture.md +123 -0
  784. package/docs/zh-CN/concepts/channel-routing.md +117 -0
  785. package/docs/zh-CN/concepts/compaction.md +67 -0
  786. package/docs/zh-CN/concepts/context.md +168 -0
  787. package/docs/zh-CN/concepts/group-messages.md +91 -0
  788. package/docs/zh-CN/concepts/groups.md +379 -0
  789. package/docs/zh-CN/concepts/markdown-formatting.md +117 -0
  790. package/docs/zh-CN/concepts/memory.md +412 -0
  791. package/docs/zh-CN/concepts/messages.md +141 -0
  792. package/docs/zh-CN/concepts/model-failover.md +145 -0
  793. package/docs/zh-CN/concepts/model-providers.md +320 -0
  794. package/docs/zh-CN/concepts/models.md +196 -0
  795. package/docs/zh-CN/concepts/multi-agent.md +372 -0
  796. package/docs/zh-CN/concepts/oauth.md +151 -0
  797. package/docs/zh-CN/concepts/presence.md +99 -0
  798. package/docs/zh-CN/concepts/queue.md +94 -0
  799. package/docs/zh-CN/concepts/retry.md +76 -0
  800. package/docs/zh-CN/concepts/session-pruning.md +129 -0
  801. package/docs/zh-CN/concepts/session-tool.md +200 -0
  802. package/docs/zh-CN/concepts/session.md +166 -0
  803. package/docs/zh-CN/concepts/sessions.md +17 -0
  804. package/docs/zh-CN/concepts/streaming.md +133 -0
  805. package/docs/zh-CN/concepts/system-prompt.md +101 -0
  806. package/docs/zh-CN/concepts/timezone.md +96 -0
  807. package/docs/zh-CN/concepts/typebox.md +284 -0
  808. package/docs/zh-CN/concepts/typing-indicators.md +74 -0
  809. package/docs/zh-CN/concepts/usage-tracking.md +42 -0
  810. package/docs/zh-CN/date-time.md +129 -0
  811. package/docs/zh-CN/debug/node-issue.md +90 -0
  812. package/docs/zh-CN/debugging.md +160 -0
  813. package/docs/zh-CN/diagnostics/flags.md +98 -0
  814. package/docs/zh-CN/environment.md +88 -0
  815. package/docs/zh-CN/experiments/onboarding-config-protocol.md +47 -0
  816. package/docs/zh-CN/experiments/plans/cron-add-hardening.md +70 -0
  817. package/docs/zh-CN/experiments/plans/group-policy-hardening.md +45 -0
  818. package/docs/zh-CN/experiments/plans/openresponses-gateway.md +121 -0
  819. package/docs/zh-CN/experiments/proposals/model-config.md +42 -0
  820. package/docs/zh-CN/experiments/research/memory.md +235 -0
  821. package/docs/zh-CN/gateway/authentication.md +142 -0
  822. package/docs/zh-CN/gateway/background-process.md +100 -0
  823. package/docs/zh-CN/gateway/bonjour.md +174 -0
  824. package/docs/zh-CN/gateway/bridge-protocol.md +86 -0
  825. package/docs/zh-CN/gateway/cli-backends.md +213 -0
  826. package/docs/zh-CN/gateway/configuration-examples.md +587 -0
  827. package/docs/zh-CN/gateway/configuration.md +3332 -0
  828. package/docs/zh-CN/gateway/discovery.md +123 -0
  829. package/docs/zh-CN/gateway/doctor.md +238 -0
  830. package/docs/zh-CN/gateway/gateway-lock.md +41 -0
  831. package/docs/zh-CN/gateway/health.md +42 -0
  832. package/docs/zh-CN/gateway/heartbeat.md +274 -0
  833. package/docs/zh-CN/gateway/index.md +335 -0
  834. package/docs/zh-CN/gateway/local-models.md +157 -0
  835. package/docs/zh-CN/gateway/logging.md +114 -0
  836. package/docs/zh-CN/gateway/multiple-gateways.md +119 -0
  837. package/docs/zh-CN/gateway/openai-http-api.md +125 -0
  838. package/docs/zh-CN/gateway/openresponses-http-api.md +317 -0
  839. package/docs/zh-CN/gateway/pairing.md +99 -0
  840. package/docs/zh-CN/gateway/protocol.md +220 -0
  841. package/docs/zh-CN/gateway/remote-gateway-readme.md +164 -0
  842. package/docs/zh-CN/gateway/remote.md +133 -0
  843. package/docs/zh-CN/gateway/sandbox-vs-tool-policy-vs-elevated.md +135 -0
  844. package/docs/zh-CN/gateway/sandboxing.md +188 -0
  845. package/docs/zh-CN/gateway/security/formal-verification.md +169 -0
  846. package/docs/zh-CN/gateway/security/index.md +777 -0
  847. package/docs/zh-CN/gateway/tailscale.md +124 -0
  848. package/docs/zh-CN/gateway/tools-invoke-http-api.md +92 -0
  849. package/docs/zh-CN/gateway/troubleshooting.md +771 -0
  850. package/docs/zh-CN/help/faq.md +2628 -0
  851. package/docs/zh-CN/help/index.md +28 -0
  852. package/docs/zh-CN/help/troubleshooting.md +104 -0
  853. package/docs/zh-CN/hooks/soul-evil.md +72 -0
  854. package/docs/zh-CN/hooks.md +919 -0
  855. package/docs/zh-CN/index.md +264 -0
  856. package/docs/zh-CN/install/ansible.md +215 -0
  857. package/docs/zh-CN/install/bun.md +65 -0
  858. package/docs/zh-CN/install/development-channels.md +81 -0
  859. package/docs/zh-CN/install/docker.md +532 -0
  860. package/docs/zh-CN/install/index.md +193 -0
  861. package/docs/zh-CN/install/installer.md +128 -0
  862. package/docs/zh-CN/install/migrating.md +199 -0
  863. package/docs/zh-CN/install/nix.md +99 -0
  864. package/docs/zh-CN/install/node.md +85 -0
  865. package/docs/zh-CN/install/uninstall.md +135 -0
  866. package/docs/zh-CN/install/updating.md +233 -0
  867. package/docs/zh-CN/logging.md +329 -0
  868. package/docs/zh-CN/multi-agent-sandbox-tools.md +401 -0
  869. package/docs/zh-CN/network.md +59 -0
  870. package/docs/zh-CN/nodes/audio.md +120 -0
  871. package/docs/zh-CN/nodes/camera.md +162 -0
  872. package/docs/zh-CN/nodes/images.md +79 -0
  873. package/docs/zh-CN/nodes/index.md +348 -0
  874. package/docs/zh-CN/nodes/location-command.md +120 -0
  875. package/docs/zh-CN/nodes/media-understanding.md +380 -0
  876. package/docs/zh-CN/nodes/talk.md +97 -0
  877. package/docs/zh-CN/nodes/voicewake.md +72 -0
  878. package/docs/zh-CN/northflank.mdx +60 -0
  879. package/docs/zh-CN/perplexity.md +84 -0
  880. package/docs/zh-CN/pi-dev.md +77 -0
  881. package/docs/zh-CN/pi.md +619 -0
  882. package/docs/zh-CN/platforms/android.md +155 -0
  883. package/docs/zh-CN/platforms/digitalocean.md +269 -0
  884. package/docs/zh-CN/platforms/exe-dev.md +127 -0
  885. package/docs/zh-CN/platforms/fly.md +490 -0
  886. package/docs/zh-CN/platforms/gcp.md +510 -0
  887. package/docs/zh-CN/platforms/hetzner.md +337 -0
  888. package/docs/zh-CN/platforms/index.md +60 -0
  889. package/docs/zh-CN/platforms/ios.md +114 -0
  890. package/docs/zh-CN/platforms/linux.md +101 -0
  891. package/docs/zh-CN/platforms/mac/bundled-gateway.md +75 -0
  892. package/docs/zh-CN/platforms/mac/canvas.md +128 -0
  893. package/docs/zh-CN/platforms/mac/child-process.md +73 -0
  894. package/docs/zh-CN/platforms/mac/dev-setup.md +109 -0
  895. package/docs/zh-CN/platforms/mac/health.md +41 -0
  896. package/docs/zh-CN/platforms/mac/icon.md +38 -0
  897. package/docs/zh-CN/platforms/mac/logging.md +64 -0
  898. package/docs/zh-CN/platforms/mac/menu-bar.md +88 -0
  899. package/docs/zh-CN/platforms/mac/peekaboo.md +62 -0
  900. package/docs/zh-CN/platforms/mac/permissions.md +46 -0
  901. package/docs/zh-CN/platforms/mac/release.md +92 -0
  902. package/docs/zh-CN/platforms/mac/remote.md +90 -0
  903. package/docs/zh-CN/platforms/mac/signing.md +54 -0
  904. package/docs/zh-CN/platforms/mac/skills.md +40 -0
  905. package/docs/zh-CN/platforms/mac/voice-overlay.md +67 -0
  906. package/docs/zh-CN/platforms/mac/voicewake.md +74 -0
  907. package/docs/zh-CN/platforms/mac/webchat.md +43 -0
  908. package/docs/zh-CN/platforms/mac/xpc.md +68 -0
  909. package/docs/zh-CN/platforms/macos-vm.md +288 -0
  910. package/docs/zh-CN/platforms/macos.md +193 -0
  911. package/docs/zh-CN/platforms/oracle.md +310 -0
  912. package/docs/zh-CN/platforms/raspberry-pi.md +365 -0
  913. package/docs/zh-CN/platforms/windows.md +156 -0
  914. package/docs/zh-CN/plugin.md +639 -0
  915. package/docs/zh-CN/plugins/agent-tools.md +99 -0
  916. package/docs/zh-CN/plugins/manifest.md +68 -0
  917. package/docs/zh-CN/plugins/voice-call.md +250 -0
  918. package/docs/zh-CN/plugins/zalouser.md +88 -0
  919. package/docs/zh-CN/prose.md +141 -0
  920. package/docs/zh-CN/providers/anthropic.md +159 -0
  921. package/docs/zh-CN/providers/claude-max-api-proxy.md +155 -0
  922. package/docs/zh-CN/providers/deepgram.md +97 -0
  923. package/docs/zh-CN/providers/github-copilot.md +67 -0
  924. package/docs/zh-CN/providers/glm.md +39 -0
  925. package/docs/zh-CN/providers/index.md +68 -0
  926. package/docs/zh-CN/providers/minimax.md +206 -0
  927. package/docs/zh-CN/providers/models.md +55 -0
  928. package/docs/zh-CN/providers/moonshot.md +145 -0
  929. package/docs/zh-CN/providers/ollama.md +230 -0
  930. package/docs/zh-CN/providers/openai.md +68 -0
  931. package/docs/zh-CN/providers/opencode.md +41 -0
  932. package/docs/zh-CN/providers/openrouter.md +43 -0
  933. package/docs/zh-CN/providers/qwen.md +55 -0
  934. package/docs/zh-CN/providers/synthetic.md +102 -0
  935. package/docs/zh-CN/providers/venice.md +274 -0
  936. package/docs/zh-CN/providers/vercel-ai-gateway.md +57 -0
  937. package/docs/zh-CN/providers/xiaomi.md +68 -0
  938. package/docs/zh-CN/providers/zai.md +41 -0
  939. package/docs/zh-CN/railway.mdx +106 -0
  940. package/docs/zh-CN/refactor/clawnet.md +424 -0
  941. package/docs/zh-CN/refactor/exec-host.md +323 -0
  942. package/docs/zh-CN/refactor/outbound-session-mirroring.md +92 -0
  943. package/docs/zh-CN/refactor/plugin-sdk.md +221 -0
  944. package/docs/zh-CN/refactor/strict-config.md +100 -0
  945. package/docs/zh-CN/reference/AGENTS.default.md +131 -0
  946. package/docs/zh-CN/reference/RELEASING.md +123 -0
  947. package/docs/zh-CN/reference/api-usage-costs.md +136 -0
  948. package/docs/zh-CN/reference/device-models.md +54 -0
  949. package/docs/zh-CN/reference/rpc.md +48 -0
  950. package/docs/zh-CN/reference/session-management-compaction.md +287 -0
  951. package/docs/zh-CN/reference/templates/AGENTS.dev.md +89 -0
  952. package/docs/zh-CN/reference/templates/AGENTS.md +225 -0
  953. package/docs/zh-CN/reference/templates/BOOT.md +17 -0
  954. package/docs/zh-CN/reference/templates/BOOTSTRAP.md +68 -0
  955. package/docs/zh-CN/reference/templates/HEARTBEAT.md +18 -0
  956. package/docs/zh-CN/reference/templates/IDENTITY.dev.md +54 -0
  957. package/docs/zh-CN/reference/templates/IDENTITY.md +35 -0
  958. package/docs/zh-CN/reference/templates/SOUL.dev.md +83 -0
  959. package/docs/zh-CN/reference/templates/SOUL.md +49 -0
  960. package/docs/zh-CN/reference/templates/TOOLS.dev.md +31 -0
  961. package/docs/zh-CN/reference/templates/TOOLS.md +53 -0
  962. package/docs/zh-CN/reference/templates/USER.dev.md +25 -0
  963. package/docs/zh-CN/reference/templates/USER.md +30 -0
  964. package/docs/zh-CN/reference/test.md +57 -0
  965. package/docs/zh-CN/reference/transcript-hygiene.md +109 -0
  966. package/docs/zh-CN/render.mdx +169 -0
  967. package/docs/zh-CN/scripts.md +35 -0
  968. package/docs/zh-CN/security/formal-verification.md +171 -0
  969. package/docs/zh-CN/start/getting-started.md +206 -0
  970. package/docs/zh-CN/start/hubs.md +191 -0
  971. package/docs/zh-CN/start/lore.md +226 -0
  972. package/docs/zh-CN/start/onboarding.md +105 -0
  973. package/docs/zh-CN/start/openclaw.md +248 -0
  974. package/docs/zh-CN/start/pairing.md +89 -0
  975. package/docs/zh-CN/start/setup.md +153 -0
  976. package/docs/zh-CN/start/showcase.md +423 -0
  977. package/docs/zh-CN/start/wizard.md +331 -0
  978. package/docs/zh-CN/testing.md +375 -0
  979. package/docs/zh-CN/token-use.md +119 -0
  980. package/docs/zh-CN/tools/agent-send.md +59 -0
  981. package/docs/zh-CN/tools/apply-patch.md +57 -0
  982. package/docs/zh-CN/tools/browser-linux-troubleshooting.md +144 -0
  983. package/docs/zh-CN/tools/browser-login.md +75 -0
  984. package/docs/zh-CN/tools/browser.md +553 -0
  985. package/docs/zh-CN/tools/chrome-extension.md +183 -0
  986. package/docs/zh-CN/tools/clawhub.md +209 -0
  987. package/docs/zh-CN/tools/creating-skills.md +61 -0
  988. package/docs/zh-CN/tools/elevated.md +64 -0
  989. package/docs/zh-CN/tools/exec-approvals.md +234 -0
  990. package/docs/zh-CN/tools/exec.md +169 -0
  991. package/docs/zh-CN/tools/firecrawl.md +68 -0
  992. package/docs/zh-CN/tools/index.md +515 -0
  993. package/docs/zh-CN/tools/llm-task.md +117 -0
  994. package/docs/zh-CN/tools/lobster.md +349 -0
  995. package/docs/zh-CN/tools/reactions.md +29 -0
  996. package/docs/zh-CN/tools/skills-config.md +78 -0
  997. package/docs/zh-CN/tools/skills.md +279 -0
  998. package/docs/zh-CN/tools/slash-commands.md +205 -0
  999. package/docs/zh-CN/tools/subagents.md +156 -0
  1000. package/docs/zh-CN/tools/thinking.md +80 -0
  1001. package/docs/zh-CN/tools/web.md +257 -0
  1002. package/docs/zh-CN/tts.md +375 -0
  1003. package/docs/zh-CN/tui.md +166 -0
  1004. package/docs/zh-CN/vps.md +47 -0
  1005. package/docs/zh-CN/web/control-ui.md +191 -0
  1006. package/docs/zh-CN/web/dashboard.md +53 -0
  1007. package/docs/zh-CN/web/index.md +118 -0
  1008. package/docs/zh-CN/web/webchat.md +56 -0
  1009. package/extensions/bluebubbles/README.md +45 -0
  1010. package/extensions/bluebubbles/index.ts +19 -0
  1011. package/extensions/bluebubbles/node_modules/.bin/openclaw +21 -0
  1012. package/extensions/bluebubbles/openclaw.plugin.json +9 -0
  1013. package/extensions/bluebubbles/package.json +36 -0
  1014. package/extensions/bluebubbles/src/accounts.ts +88 -0
  1015. package/extensions/bluebubbles/src/actions.test.ts +650 -0
  1016. package/extensions/bluebubbles/src/actions.ts +438 -0
  1017. package/extensions/bluebubbles/src/attachments.test.ts +345 -0
  1018. package/extensions/bluebubbles/src/attachments.ts +300 -0
  1019. package/extensions/bluebubbles/src/channel.ts +414 -0
  1020. package/extensions/bluebubbles/src/chat.test.ts +461 -0
  1021. package/extensions/bluebubbles/src/chat.ts +378 -0
  1022. package/extensions/bluebubbles/src/config-schema.ts +51 -0
  1023. package/extensions/bluebubbles/src/media-send.ts +174 -0
  1024. package/extensions/bluebubbles/src/monitor.test.ts +2342 -0
  1025. package/extensions/bluebubbles/src/monitor.ts +2490 -0
  1026. package/extensions/bluebubbles/src/onboarding.ts +352 -0
  1027. package/extensions/bluebubbles/src/probe.ts +135 -0
  1028. package/extensions/bluebubbles/src/reactions.test.ts +392 -0
  1029. package/extensions/bluebubbles/src/reactions.ts +188 -0
  1030. package/extensions/bluebubbles/src/runtime.ts +14 -0
  1031. package/extensions/bluebubbles/src/send.test.ts +808 -0
  1032. package/extensions/bluebubbles/src/send.ts +467 -0
  1033. package/extensions/bluebubbles/src/targets.test.ts +183 -0
  1034. package/extensions/bluebubbles/src/targets.ts +422 -0
  1035. package/extensions/bluebubbles/src/types.ts +127 -0
  1036. package/extensions/copilot-proxy/README.md +24 -0
  1037. package/extensions/copilot-proxy/index.ts +148 -0
  1038. package/extensions/copilot-proxy/node_modules/.bin/openclaw +21 -0
  1039. package/extensions/copilot-proxy/openclaw.plugin.json +9 -0
  1040. package/extensions/copilot-proxy/package.json +14 -0
  1041. package/extensions/diagnostics-otel/index.ts +15 -0
  1042. package/extensions/diagnostics-otel/node_modules/.bin/openclaw +21 -0
  1043. package/extensions/diagnostics-otel/openclaw.plugin.json +8 -0
  1044. package/extensions/diagnostics-otel/package.json +27 -0
  1045. package/extensions/diagnostics-otel/src/service.test.ts +226 -0
  1046. package/extensions/diagnostics-otel/src/service.ts +635 -0
  1047. package/extensions/discord/index.ts +17 -0
  1048. package/extensions/discord/node_modules/.bin/openclaw +21 -0
  1049. package/extensions/discord/openclaw.plugin.json +9 -0
  1050. package/extensions/discord/package.json +14 -0
  1051. package/extensions/discord/src/channel.ts +422 -0
  1052. package/extensions/discord/src/runtime.ts +14 -0
  1053. package/extensions/feishu/README.md +47 -0
  1054. package/extensions/feishu/index.ts +15 -0
  1055. package/extensions/feishu/node_modules/.bin/openclaw +21 -0
  1056. package/extensions/feishu/openclaw.plugin.json +9 -0
  1057. package/extensions/feishu/package.json +33 -0
  1058. package/extensions/feishu/src/channel.ts +276 -0
  1059. package/extensions/feishu/src/config-schema.ts +46 -0
  1060. package/extensions/feishu/src/onboarding.ts +278 -0
  1061. package/extensions/google-antigravity-auth/README.md +24 -0
  1062. package/extensions/google-antigravity-auth/index.ts +461 -0
  1063. package/extensions/google-antigravity-auth/node_modules/.bin/openclaw +21 -0
  1064. package/extensions/google-antigravity-auth/openclaw.plugin.json +9 -0
  1065. package/extensions/google-antigravity-auth/package.json +14 -0
  1066. package/extensions/google-gemini-cli-auth/README.md +35 -0
  1067. package/extensions/google-gemini-cli-auth/index.ts +88 -0
  1068. package/extensions/google-gemini-cli-auth/node_modules/.bin/openclaw +21 -0
  1069. package/extensions/google-gemini-cli-auth/oauth.test.ts +240 -0
  1070. package/extensions/google-gemini-cli-auth/oauth.ts +662 -0
  1071. package/extensions/google-gemini-cli-auth/openclaw.plugin.json +9 -0
  1072. package/extensions/google-gemini-cli-auth/package.json +14 -0
  1073. package/extensions/googlechat/index.ts +19 -0
  1074. package/extensions/googlechat/node_modules/.bin/openclaw +21 -0
  1075. package/extensions/googlechat/openclaw.plugin.json +9 -0
  1076. package/extensions/googlechat/package.json +39 -0
  1077. package/extensions/googlechat/src/accounts.ts +147 -0
  1078. package/extensions/googlechat/src/actions.ts +181 -0
  1079. package/extensions/googlechat/src/api.test.ts +61 -0
  1080. package/extensions/googlechat/src/api.ts +282 -0
  1081. package/extensions/googlechat/src/auth.ts +123 -0
  1082. package/extensions/googlechat/src/channel.ts +583 -0
  1083. package/extensions/googlechat/src/monitor.test.ts +22 -0
  1084. package/extensions/googlechat/src/monitor.ts +949 -0
  1085. package/extensions/googlechat/src/onboarding.ts +269 -0
  1086. package/extensions/googlechat/src/runtime.ts +14 -0
  1087. package/extensions/googlechat/src/targets.test.ts +32 -0
  1088. package/extensions/googlechat/src/targets.ts +65 -0
  1089. package/extensions/googlechat/src/types.config.ts +3 -0
  1090. package/extensions/googlechat/src/types.ts +73 -0
  1091. package/extensions/imessage/index.ts +17 -0
  1092. package/extensions/imessage/node_modules/.bin/openclaw +21 -0
  1093. package/extensions/imessage/openclaw.plugin.json +9 -0
  1094. package/extensions/imessage/package.json +14 -0
  1095. package/extensions/imessage/src/channel.ts +294 -0
  1096. package/extensions/imessage/src/runtime.ts +14 -0
  1097. package/extensions/line/index.ts +19 -0
  1098. package/extensions/line/node_modules/.bin/openclaw +21 -0
  1099. package/extensions/line/openclaw.plugin.json +9 -0
  1100. package/extensions/line/package.json +29 -0
  1101. package/extensions/line/src/card-command.ts +344 -0
  1102. package/extensions/line/src/channel.logout.test.ts +99 -0
  1103. package/extensions/line/src/channel.sendPayload.test.ts +306 -0
  1104. package/extensions/line/src/channel.ts +780 -0
  1105. package/extensions/line/src/runtime.ts +14 -0
  1106. package/extensions/llm-task/README.md +97 -0
  1107. package/extensions/llm-task/index.ts +6 -0
  1108. package/extensions/llm-task/node_modules/.bin/openclaw +21 -0
  1109. package/extensions/llm-task/openclaw.plugin.json +21 -0
  1110. package/extensions/llm-task/package.json +14 -0
  1111. package/extensions/llm-task/src/llm-task-tool.test.ts +138 -0
  1112. package/extensions/llm-task/src/llm-task-tool.ts +245 -0
  1113. package/extensions/lobster/README.md +75 -0
  1114. package/extensions/lobster/SKILL.md +97 -0
  1115. package/extensions/lobster/index.ts +14 -0
  1116. package/extensions/lobster/node_modules/.bin/openclaw +21 -0
  1117. package/extensions/lobster/openclaw.plugin.json +10 -0
  1118. package/extensions/lobster/package.json +14 -0
  1119. package/extensions/lobster/src/lobster-tool.test.ts +247 -0
  1120. package/extensions/lobster/src/lobster-tool.ts +328 -0
  1121. package/extensions/matrix/CHANGELOG.md +87 -0
  1122. package/extensions/matrix/index.ts +17 -0
  1123. package/extensions/matrix/node_modules/.bin/markdown-it +21 -0
  1124. package/extensions/matrix/node_modules/.bin/openclaw +21 -0
  1125. package/extensions/matrix/openclaw.plugin.json +9 -0
  1126. package/extensions/matrix/package.json +36 -0
  1127. package/extensions/matrix/src/actions.ts +195 -0
  1128. package/extensions/matrix/src/channel.directory.test.ts +64 -0
  1129. package/extensions/matrix/src/channel.ts +439 -0
  1130. package/extensions/matrix/src/config-schema.ts +62 -0
  1131. package/extensions/matrix/src/directory-live.ts +188 -0
  1132. package/extensions/matrix/src/group-mentions.ts +66 -0
  1133. package/extensions/matrix/src/matrix/accounts.test.ts +82 -0
  1134. package/extensions/matrix/src/matrix/accounts.ts +65 -0
  1135. package/extensions/matrix/src/matrix/actions/client.ts +57 -0
  1136. package/extensions/matrix/src/matrix/actions/messages.ts +128 -0
  1137. package/extensions/matrix/src/matrix/actions/pins.ts +76 -0
  1138. package/extensions/matrix/src/matrix/actions/reactions.ts +96 -0
  1139. package/extensions/matrix/src/matrix/actions/room.ts +85 -0
  1140. package/extensions/matrix/src/matrix/actions/summary.ts +75 -0
  1141. package/extensions/matrix/src/matrix/actions/types.ts +84 -0
  1142. package/extensions/matrix/src/matrix/actions.ts +15 -0
  1143. package/extensions/matrix/src/matrix/active-client.ts +11 -0
  1144. package/extensions/matrix/src/matrix/client/config.ts +160 -0
  1145. package/extensions/matrix/src/matrix/client/create-client.ts +123 -0
  1146. package/extensions/matrix/src/matrix/client/logging.ts +36 -0
  1147. package/extensions/matrix/src/matrix/client/runtime.ts +4 -0
  1148. package/extensions/matrix/src/matrix/client/shared.ts +170 -0
  1149. package/extensions/matrix/src/matrix/client/storage.ts +131 -0
  1150. package/extensions/matrix/src/matrix/client/types.ts +34 -0
  1151. package/extensions/matrix/src/matrix/client.test.ts +56 -0
  1152. package/extensions/matrix/src/matrix/client.ts +5 -0
  1153. package/extensions/matrix/src/matrix/credentials.ts +105 -0
  1154. package/extensions/matrix/src/matrix/deps.ts +60 -0
  1155. package/extensions/matrix/src/matrix/format.test.ts +33 -0
  1156. package/extensions/matrix/src/matrix/format.ts +22 -0
  1157. package/extensions/matrix/src/matrix/index.ts +11 -0
  1158. package/extensions/matrix/src/matrix/monitor/allowlist.test.ts +45 -0
  1159. package/extensions/matrix/src/matrix/monitor/allowlist.ts +103 -0
  1160. package/extensions/matrix/src/matrix/monitor/auto-join.ts +71 -0
  1161. package/extensions/matrix/src/matrix/monitor/direct.ts +104 -0
  1162. package/extensions/matrix/src/matrix/monitor/events.ts +101 -0
  1163. package/extensions/matrix/src/matrix/monitor/handler.ts +661 -0
  1164. package/extensions/matrix/src/matrix/monitor/index.ts +338 -0
  1165. package/extensions/matrix/src/matrix/monitor/location.ts +100 -0
  1166. package/extensions/matrix/src/matrix/monitor/media.test.ts +102 -0
  1167. package/extensions/matrix/src/matrix/monitor/media.ts +113 -0
  1168. package/extensions/matrix/src/matrix/monitor/mentions.ts +31 -0
  1169. package/extensions/matrix/src/matrix/monitor/replies.ts +97 -0
  1170. package/extensions/matrix/src/matrix/monitor/room-info.ts +55 -0
  1171. package/extensions/matrix/src/matrix/monitor/rooms.test.ts +39 -0
  1172. package/extensions/matrix/src/matrix/monitor/rooms.ts +47 -0
  1173. package/extensions/matrix/src/matrix/monitor/threads.ts +68 -0
  1174. package/extensions/matrix/src/matrix/monitor/types.ts +39 -0
  1175. package/extensions/matrix/src/matrix/poll-types.test.ts +21 -0
  1176. package/extensions/matrix/src/matrix/poll-types.ts +166 -0
  1177. package/extensions/matrix/src/matrix/probe.ts +70 -0
  1178. package/extensions/matrix/src/matrix/send/client.ts +66 -0
  1179. package/extensions/matrix/src/matrix/send/formatting.ts +89 -0
  1180. package/extensions/matrix/src/matrix/send/media.ts +229 -0
  1181. package/extensions/matrix/src/matrix/send/targets.test.ts +98 -0
  1182. package/extensions/matrix/src/matrix/send/targets.ts +136 -0
  1183. package/extensions/matrix/src/matrix/send/types.ts +109 -0
  1184. package/extensions/matrix/src/matrix/send.test.ts +171 -0
  1185. package/extensions/matrix/src/matrix/send.ts +260 -0
  1186. package/extensions/matrix/src/onboarding.ts +449 -0
  1187. package/extensions/matrix/src/outbound.ts +52 -0
  1188. package/extensions/matrix/src/resolve-targets.test.ts +48 -0
  1189. package/extensions/matrix/src/resolve-targets.ts +135 -0
  1190. package/extensions/matrix/src/runtime.ts +14 -0
  1191. package/extensions/matrix/src/tool-actions.ts +164 -0
  1192. package/extensions/matrix/src/types.ts +95 -0
  1193. package/extensions/mattermost/index.ts +17 -0
  1194. package/extensions/mattermost/node_modules/.bin/openclaw +21 -0
  1195. package/extensions/mattermost/openclaw.plugin.json +9 -0
  1196. package/extensions/mattermost/package.json +28 -0
  1197. package/extensions/mattermost/src/channel.test.ts +48 -0
  1198. package/extensions/mattermost/src/channel.ts +337 -0
  1199. package/extensions/mattermost/src/config-schema.ts +55 -0
  1200. package/extensions/mattermost/src/group-mentions.ts +15 -0
  1201. package/extensions/mattermost/src/mattermost/accounts.ts +128 -0
  1202. package/extensions/mattermost/src/mattermost/client.ts +220 -0
  1203. package/extensions/mattermost/src/mattermost/index.ts +9 -0
  1204. package/extensions/mattermost/src/mattermost/monitor-helpers.ts +166 -0
  1205. package/extensions/mattermost/src/mattermost/monitor.ts +987 -0
  1206. package/extensions/mattermost/src/mattermost/probe.ts +74 -0
  1207. package/extensions/mattermost/src/mattermost/send.ts +231 -0
  1208. package/extensions/mattermost/src/normalize.ts +46 -0
  1209. package/extensions/mattermost/src/onboarding-helpers.ts +44 -0
  1210. package/extensions/mattermost/src/onboarding.ts +186 -0
  1211. package/extensions/mattermost/src/runtime.ts +14 -0
  1212. package/extensions/mattermost/src/types.ts +50 -0
  1213. package/extensions/memory-core/index.ts +38 -0
  1214. package/extensions/memory-core/node_modules/.bin/openclaw +21 -0
  1215. package/extensions/memory-core/openclaw.plugin.json +9 -0
  1216. package/extensions/memory-core/package.json +17 -0
  1217. package/extensions/memory-lancedb/config.ts +139 -0
  1218. package/extensions/memory-lancedb/index.test.ts +295 -0
  1219. package/extensions/memory-lancedb/index.ts +608 -0
  1220. package/extensions/memory-lancedb/node_modules/.bin/openai +21 -0
  1221. package/extensions/memory-lancedb/node_modules/.bin/openclaw +21 -0
  1222. package/extensions/memory-lancedb/openclaw.plugin.json +60 -0
  1223. package/extensions/memory-lancedb/package.json +19 -0
  1224. package/extensions/minimax-portal-auth/README.md +33 -0
  1225. package/extensions/minimax-portal-auth/index.ts +155 -0
  1226. package/extensions/minimax-portal-auth/node_modules/.bin/openclaw +21 -0
  1227. package/extensions/minimax-portal-auth/oauth.ts +247 -0
  1228. package/extensions/minimax-portal-auth/openclaw.plugin.json +9 -0
  1229. package/extensions/minimax-portal-auth/package.json +14 -0
  1230. package/extensions/msteams/CHANGELOG.md +83 -0
  1231. package/extensions/msteams/index.ts +17 -0
  1232. package/extensions/msteams/node_modules/.bin/openclaw +21 -0
  1233. package/extensions/msteams/openclaw.plugin.json +9 -0
  1234. package/extensions/msteams/package.json +39 -0
  1235. package/extensions/msteams/src/attachments/download.ts +283 -0
  1236. package/extensions/msteams/src/attachments/graph.ts +353 -0
  1237. package/extensions/msteams/src/attachments/html.ts +90 -0
  1238. package/extensions/msteams/src/attachments/payload.ts +22 -0
  1239. package/extensions/msteams/src/attachments/shared.ts +291 -0
  1240. package/extensions/msteams/src/attachments/types.ts +37 -0
  1241. package/extensions/msteams/src/attachments.test.ts +459 -0
  1242. package/extensions/msteams/src/attachments.ts +18 -0
  1243. package/extensions/msteams/src/channel.directory.test.ts +48 -0
  1244. package/extensions/msteams/src/channel.ts +459 -0
  1245. package/extensions/msteams/src/conversation-store-fs.test.ts +88 -0
  1246. package/extensions/msteams/src/conversation-store-fs.ts +165 -0
  1247. package/extensions/msteams/src/conversation-store-memory.ts +47 -0
  1248. package/extensions/msteams/src/conversation-store.ts +41 -0
  1249. package/extensions/msteams/src/directory-live.ts +205 -0
  1250. package/extensions/msteams/src/errors.test.ts +45 -0
  1251. package/extensions/msteams/src/errors.ts +190 -0
  1252. package/extensions/msteams/src/file-consent-helpers.test.ts +243 -0
  1253. package/extensions/msteams/src/file-consent-helpers.ts +73 -0
  1254. package/extensions/msteams/src/file-consent.ts +126 -0
  1255. package/extensions/msteams/src/graph-chat.ts +53 -0
  1256. package/extensions/msteams/src/graph-upload.ts +453 -0
  1257. package/extensions/msteams/src/inbound.test.ts +66 -0
  1258. package/extensions/msteams/src/inbound.ts +48 -0
  1259. package/extensions/msteams/src/index.ts +4 -0
  1260. package/extensions/msteams/src/media-helpers.test.ts +189 -0
  1261. package/extensions/msteams/src/media-helpers.ts +86 -0
  1262. package/extensions/msteams/src/messenger.test.ts +248 -0
  1263. package/extensions/msteams/src/messenger.ts +495 -0
  1264. package/extensions/msteams/src/monitor-handler/inbound-media.ts +128 -0
  1265. package/extensions/msteams/src/monitor-handler/message-handler.ts +640 -0
  1266. package/extensions/msteams/src/monitor-handler.ts +162 -0
  1267. package/extensions/msteams/src/monitor-types.ts +5 -0
  1268. package/extensions/msteams/src/monitor.ts +295 -0
  1269. package/extensions/msteams/src/onboarding.ts +431 -0
  1270. package/extensions/msteams/src/outbound.ts +46 -0
  1271. package/extensions/msteams/src/pending-uploads.ts +89 -0
  1272. package/extensions/msteams/src/policy.test.ts +209 -0
  1273. package/extensions/msteams/src/policy.ts +273 -0
  1274. package/extensions/msteams/src/polls-store-memory.ts +32 -0
  1275. package/extensions/msteams/src/polls-store.test.ts +38 -0
  1276. package/extensions/msteams/src/polls.test.ts +72 -0
  1277. package/extensions/msteams/src/polls.ts +315 -0
  1278. package/extensions/msteams/src/probe.test.ts +58 -0
  1279. package/extensions/msteams/src/probe.ts +107 -0
  1280. package/extensions/msteams/src/reply-dispatcher.ts +130 -0
  1281. package/extensions/msteams/src/resolve-allowlist.ts +297 -0
  1282. package/extensions/msteams/src/runtime.ts +14 -0
  1283. package/extensions/msteams/src/sdk-types.ts +19 -0
  1284. package/extensions/msteams/src/sdk.ts +33 -0
  1285. package/extensions/msteams/src/send-context.ts +164 -0
  1286. package/extensions/msteams/src/send.ts +519 -0
  1287. package/extensions/msteams/src/sent-message-cache.test.ts +15 -0
  1288. package/extensions/msteams/src/sent-message-cache.ts +47 -0
  1289. package/extensions/msteams/src/storage.ts +25 -0
  1290. package/extensions/msteams/src/store-fs.ts +83 -0
  1291. package/extensions/msteams/src/token.ts +19 -0
  1292. package/extensions/nextcloud-talk/index.ts +17 -0
  1293. package/extensions/nextcloud-talk/node_modules/.bin/openclaw +21 -0
  1294. package/extensions/nextcloud-talk/openclaw.plugin.json +9 -0
  1295. package/extensions/nextcloud-talk/package.json +33 -0
  1296. package/extensions/nextcloud-talk/src/accounts.ts +174 -0
  1297. package/extensions/nextcloud-talk/src/channel.ts +409 -0
  1298. package/extensions/nextcloud-talk/src/config-schema.ts +78 -0
  1299. package/extensions/nextcloud-talk/src/format.ts +79 -0
  1300. package/extensions/nextcloud-talk/src/inbound.ts +317 -0
  1301. package/extensions/nextcloud-talk/src/monitor.ts +246 -0
  1302. package/extensions/nextcloud-talk/src/normalize.ts +39 -0
  1303. package/extensions/nextcloud-talk/src/onboarding.ts +343 -0
  1304. package/extensions/nextcloud-talk/src/policy.test.ts +33 -0
  1305. package/extensions/nextcloud-talk/src/policy.ts +180 -0
  1306. package/extensions/nextcloud-talk/src/room-info.ts +125 -0
  1307. package/extensions/nextcloud-talk/src/runtime.ts +14 -0
  1308. package/extensions/nextcloud-talk/src/send.ts +210 -0
  1309. package/extensions/nextcloud-talk/src/signature.ts +72 -0
  1310. package/extensions/nextcloud-talk/src/types.ts +179 -0
  1311. package/extensions/nostr/CHANGELOG.md +74 -0
  1312. package/extensions/nostr/README.md +136 -0
  1313. package/extensions/nostr/index.ts +68 -0
  1314. package/extensions/nostr/node_modules/.bin/openclaw +21 -0
  1315. package/extensions/nostr/openclaw.plugin.json +9 -0
  1316. package/extensions/nostr/package.json +34 -0
  1317. package/extensions/nostr/src/channel.test.ts +151 -0
  1318. package/extensions/nostr/src/channel.ts +353 -0
  1319. package/extensions/nostr/src/config-schema.ts +90 -0
  1320. package/extensions/nostr/src/metrics.ts +478 -0
  1321. package/extensions/nostr/src/nostr-bus.fuzz.test.ts +533 -0
  1322. package/extensions/nostr/src/nostr-bus.integration.test.ts +448 -0
  1323. package/extensions/nostr/src/nostr-bus.test.ts +199 -0
  1324. package/extensions/nostr/src/nostr-bus.ts +715 -0
  1325. package/extensions/nostr/src/nostr-profile-http.test.ts +378 -0
  1326. package/extensions/nostr/src/nostr-profile-http.ts +519 -0
  1327. package/extensions/nostr/src/nostr-profile-import.test.ts +119 -0
  1328. package/extensions/nostr/src/nostr-profile-import.ts +262 -0
  1329. package/extensions/nostr/src/nostr-profile.fuzz.test.ts +477 -0
  1330. package/extensions/nostr/src/nostr-profile.test.ts +410 -0
  1331. package/extensions/nostr/src/nostr-profile.ts +277 -0
  1332. package/extensions/nostr/src/nostr-state-store.test.ts +131 -0
  1333. package/extensions/nostr/src/nostr-state-store.ts +226 -0
  1334. package/extensions/nostr/src/runtime.ts +14 -0
  1335. package/extensions/nostr/src/seen-tracker.ts +303 -0
  1336. package/extensions/nostr/src/types.test.ts +157 -0
  1337. package/extensions/nostr/src/types.ts +101 -0
  1338. package/extensions/nostr/test/setup.ts +5 -0
  1339. package/extensions/open-prose/README.md +25 -0
  1340. package/extensions/open-prose/index.ts +5 -0
  1341. package/extensions/open-prose/node_modules/.bin/openclaw +21 -0
  1342. package/extensions/open-prose/openclaw.plugin.json +11 -0
  1343. package/extensions/open-prose/package.json +14 -0
  1344. package/extensions/open-prose/skills/prose/LICENSE +21 -0
  1345. package/extensions/open-prose/skills/prose/SKILL.md +323 -0
  1346. package/extensions/open-prose/skills/prose/alt-borges.md +141 -0
  1347. package/extensions/open-prose/skills/prose/alts/arabian-nights.md +358 -0
  1348. package/extensions/open-prose/skills/prose/alts/borges.md +360 -0
  1349. package/extensions/open-prose/skills/prose/alts/folk.md +322 -0
  1350. package/extensions/open-prose/skills/prose/alts/homer.md +346 -0
  1351. package/extensions/open-prose/skills/prose/alts/kafka.md +373 -0
  1352. package/extensions/open-prose/skills/prose/compiler.md +2971 -0
  1353. package/extensions/open-prose/skills/prose/examples/01-hello-world.prose +4 -0
  1354. package/extensions/open-prose/skills/prose/examples/02-research-and-summarize.prose +6 -0
  1355. package/extensions/open-prose/skills/prose/examples/03-code-review.prose +17 -0
  1356. package/extensions/open-prose/skills/prose/examples/04-write-and-refine.prose +14 -0
  1357. package/extensions/open-prose/skills/prose/examples/05-debug-issue.prose +20 -0
  1358. package/extensions/open-prose/skills/prose/examples/06-explain-codebase.prose +17 -0
  1359. package/extensions/open-prose/skills/prose/examples/07-refactor.prose +20 -0
  1360. package/extensions/open-prose/skills/prose/examples/08-blog-post.prose +20 -0
  1361. package/extensions/open-prose/skills/prose/examples/09-research-with-agents.prose +25 -0
  1362. package/extensions/open-prose/skills/prose/examples/10-code-review-agents.prose +32 -0
  1363. package/extensions/open-prose/skills/prose/examples/11-skills-and-imports.prose +27 -0
  1364. package/extensions/open-prose/skills/prose/examples/12-secure-agent-permissions.prose +43 -0
  1365. package/extensions/open-prose/skills/prose/examples/13-variables-and-context.prose +51 -0
  1366. package/extensions/open-prose/skills/prose/examples/14-composition-blocks.prose +48 -0
  1367. package/extensions/open-prose/skills/prose/examples/15-inline-sequences.prose +23 -0
  1368. package/extensions/open-prose/skills/prose/examples/16-parallel-reviews.prose +19 -0
  1369. package/extensions/open-prose/skills/prose/examples/17-parallel-research.prose +19 -0
  1370. package/extensions/open-prose/skills/prose/examples/18-mixed-parallel-sequential.prose +36 -0
  1371. package/extensions/open-prose/skills/prose/examples/19-advanced-parallel.prose +71 -0
  1372. package/extensions/open-prose/skills/prose/examples/20-fixed-loops.prose +20 -0
  1373. package/extensions/open-prose/skills/prose/examples/21-pipeline-operations.prose +35 -0
  1374. package/extensions/open-prose/skills/prose/examples/22-error-handling.prose +51 -0
  1375. package/extensions/open-prose/skills/prose/examples/23-retry-with-backoff.prose +63 -0
  1376. package/extensions/open-prose/skills/prose/examples/24-choice-blocks.prose +86 -0
  1377. package/extensions/open-prose/skills/prose/examples/25-conditionals.prose +114 -0
  1378. package/extensions/open-prose/skills/prose/examples/26-parameterized-blocks.prose +100 -0
  1379. package/extensions/open-prose/skills/prose/examples/27-string-interpolation.prose +105 -0
  1380. package/extensions/open-prose/skills/prose/examples/28-automated-pr-review.prose +37 -0
  1381. package/extensions/open-prose/skills/prose/examples/28-gas-town.prose +1572 -0
  1382. package/extensions/open-prose/skills/prose/examples/29-captains-chair.prose +218 -0
  1383. package/extensions/open-prose/skills/prose/examples/30-captains-chair-simple.prose +42 -0
  1384. package/extensions/open-prose/skills/prose/examples/31-captains-chair-with-memory.prose +145 -0
  1385. package/extensions/open-prose/skills/prose/examples/33-pr-review-autofix.prose +168 -0
  1386. package/extensions/open-prose/skills/prose/examples/34-content-pipeline.prose +204 -0
  1387. package/extensions/open-prose/skills/prose/examples/35-feature-factory.prose +296 -0
  1388. package/extensions/open-prose/skills/prose/examples/36-bug-hunter.prose +237 -0
  1389. package/extensions/open-prose/skills/prose/examples/37-the-forge.prose +1474 -0
  1390. package/extensions/open-prose/skills/prose/examples/38-skill-scan.prose +455 -0
  1391. package/extensions/open-prose/skills/prose/examples/39-architect-by-simulation.prose +277 -0
  1392. package/extensions/open-prose/skills/prose/examples/40-rlm-self-refine.prose +32 -0
  1393. package/extensions/open-prose/skills/prose/examples/41-rlm-divide-conquer.prose +38 -0
  1394. package/extensions/open-prose/skills/prose/examples/42-rlm-filter-recurse.prose +46 -0
  1395. package/extensions/open-prose/skills/prose/examples/43-rlm-pairwise.prose +50 -0
  1396. package/extensions/open-prose/skills/prose/examples/44-run-endpoint-ux-test.prose +261 -0
  1397. package/extensions/open-prose/skills/prose/examples/45-plugin-release.prose +159 -0
  1398. package/extensions/open-prose/skills/prose/examples/45-run-endpoint-ux-test-with-remediation.prose +637 -0
  1399. package/extensions/open-prose/skills/prose/examples/46-run-endpoint-ux-test-fast.prose +148 -0
  1400. package/extensions/open-prose/skills/prose/examples/46-workflow-crystallizer.prose +225 -0
  1401. package/extensions/open-prose/skills/prose/examples/47-language-self-improvement.prose +356 -0
  1402. package/extensions/open-prose/skills/prose/examples/48-habit-miner.prose +445 -0
  1403. package/extensions/open-prose/skills/prose/examples/49-prose-run-retrospective.prose +210 -0
  1404. package/extensions/open-prose/skills/prose/examples/README.md +391 -0
  1405. package/extensions/open-prose/skills/prose/examples/roadmap/README.md +22 -0
  1406. package/extensions/open-prose/skills/prose/examples/roadmap/iterative-refinement.prose +20 -0
  1407. package/extensions/open-prose/skills/prose/examples/roadmap/parallel-review.prose +18 -0
  1408. package/extensions/open-prose/skills/prose/examples/roadmap/simple-pipeline.prose +17 -0
  1409. package/extensions/open-prose/skills/prose/examples/roadmap/syntax/open-prose-syntax.prose +223 -0
  1410. package/extensions/open-prose/skills/prose/guidance/antipatterns.md +951 -0
  1411. package/extensions/open-prose/skills/prose/guidance/patterns.md +700 -0
  1412. package/extensions/open-prose/skills/prose/guidance/system-prompt.md +180 -0
  1413. package/extensions/open-prose/skills/prose/help.md +144 -0
  1414. package/extensions/open-prose/skills/prose/lib/README.md +108 -0
  1415. package/extensions/open-prose/skills/prose/lib/calibrator.prose +215 -0
  1416. package/extensions/open-prose/skills/prose/lib/cost-analyzer.prose +174 -0
  1417. package/extensions/open-prose/skills/prose/lib/error-forensics.prose +250 -0
  1418. package/extensions/open-prose/skills/prose/lib/inspector.prose +196 -0
  1419. package/extensions/open-prose/skills/prose/lib/profiler.prose +460 -0
  1420. package/extensions/open-prose/skills/prose/lib/program-improver.prose +275 -0
  1421. package/extensions/open-prose/skills/prose/lib/project-memory.prose +118 -0
  1422. package/extensions/open-prose/skills/prose/lib/user-memory.prose +93 -0
  1423. package/extensions/open-prose/skills/prose/lib/vm-improver.prose +243 -0
  1424. package/extensions/open-prose/skills/prose/primitives/session.md +593 -0
  1425. package/extensions/open-prose/skills/prose/prose.md +1237 -0
  1426. package/extensions/open-prose/skills/prose/state/filesystem.md +498 -0
  1427. package/extensions/open-prose/skills/prose/state/in-context.md +384 -0
  1428. package/extensions/open-prose/skills/prose/state/postgres.md +880 -0
  1429. package/extensions/open-prose/skills/prose/state/sqlite.md +574 -0
  1430. package/extensions/qwen-portal-auth/README.md +24 -0
  1431. package/extensions/qwen-portal-auth/index.ts +130 -0
  1432. package/extensions/qwen-portal-auth/oauth.ts +190 -0
  1433. package/extensions/qwen-portal-auth/openclaw.plugin.json +9 -0
  1434. package/extensions/signal/index.ts +17 -0
  1435. package/extensions/signal/node_modules/.bin/openclaw +21 -0
  1436. package/extensions/signal/openclaw.plugin.json +9 -0
  1437. package/extensions/signal/package.json +14 -0
  1438. package/extensions/signal/src/channel.ts +315 -0
  1439. package/extensions/signal/src/runtime.ts +14 -0
  1440. package/extensions/slack/index.ts +17 -0
  1441. package/extensions/slack/node_modules/.bin/openclaw +21 -0
  1442. package/extensions/slack/openclaw.plugin.json +9 -0
  1443. package/extensions/slack/package.json +14 -0
  1444. package/extensions/slack/src/channel.ts +604 -0
  1445. package/extensions/slack/src/runtime.ts +14 -0
  1446. package/extensions/telegram/index.ts +17 -0
  1447. package/extensions/telegram/node_modules/.bin/openclaw +21 -0
  1448. package/extensions/telegram/openclaw.plugin.json +9 -0
  1449. package/extensions/telegram/package.json +14 -0
  1450. package/extensions/telegram/src/channel.ts +482 -0
  1451. package/extensions/telegram/src/runtime.ts +14 -0
  1452. package/extensions/tlon/README.md +5 -0
  1453. package/extensions/tlon/index.ts +17 -0
  1454. package/extensions/tlon/node_modules/.bin/openclaw +21 -0
  1455. package/extensions/tlon/openclaw.plugin.json +9 -0
  1456. package/extensions/tlon/package.json +33 -0
  1457. package/extensions/tlon/src/channel.ts +392 -0
  1458. package/extensions/tlon/src/config-schema.test.ts +31 -0
  1459. package/extensions/tlon/src/config-schema.ts +43 -0
  1460. package/extensions/tlon/src/monitor/discovery.ts +76 -0
  1461. package/extensions/tlon/src/monitor/history.ts +90 -0
  1462. package/extensions/tlon/src/monitor/index.ts +553 -0
  1463. package/extensions/tlon/src/monitor/processed-messages.test.ts +23 -0
  1464. package/extensions/tlon/src/monitor/processed-messages.ts +46 -0
  1465. package/extensions/tlon/src/monitor/utils.ts +105 -0
  1466. package/extensions/tlon/src/onboarding.ts +214 -0
  1467. package/extensions/tlon/src/runtime.ts +14 -0
  1468. package/extensions/tlon/src/targets.ts +89 -0
  1469. package/extensions/tlon/src/types.ts +92 -0
  1470. package/extensions/tlon/src/urbit/auth.ts +18 -0
  1471. package/extensions/tlon/src/urbit/http-api.ts +38 -0
  1472. package/extensions/tlon/src/urbit/send.test.ts +38 -0
  1473. package/extensions/tlon/src/urbit/send.ts +131 -0
  1474. package/extensions/tlon/src/urbit/sse-client.test.ts +40 -0
  1475. package/extensions/tlon/src/urbit/sse-client.ts +395 -0
  1476. package/extensions/twitch/CHANGELOG.md +45 -0
  1477. package/extensions/twitch/README.md +89 -0
  1478. package/extensions/twitch/index.ts +20 -0
  1479. package/extensions/twitch/node_modules/.bin/openclaw +21 -0
  1480. package/extensions/twitch/openclaw.plugin.json +9 -0
  1481. package/extensions/twitch/package.json +20 -0
  1482. package/extensions/twitch/src/access-control.test.ts +489 -0
  1483. package/extensions/twitch/src/access-control.ts +166 -0
  1484. package/extensions/twitch/src/actions.ts +173 -0
  1485. package/extensions/twitch/src/client-manager-registry.ts +115 -0
  1486. package/extensions/twitch/src/config-schema.ts +82 -0
  1487. package/extensions/twitch/src/config.test.ts +87 -0
  1488. package/extensions/twitch/src/config.ts +116 -0
  1489. package/extensions/twitch/src/monitor.ts +261 -0
  1490. package/extensions/twitch/src/onboarding.test.ts +311 -0
  1491. package/extensions/twitch/src/onboarding.ts +417 -0
  1492. package/extensions/twitch/src/outbound.test.ts +373 -0
  1493. package/extensions/twitch/src/outbound.ts +184 -0
  1494. package/extensions/twitch/src/plugin.test.ts +39 -0
  1495. package/extensions/twitch/src/plugin.ts +274 -0
  1496. package/extensions/twitch/src/probe.test.ts +195 -0
  1497. package/extensions/twitch/src/probe.ts +120 -0
  1498. package/extensions/twitch/src/resolver.ts +137 -0
  1499. package/extensions/twitch/src/runtime.ts +14 -0
  1500. package/extensions/twitch/src/send.test.ts +289 -0
  1501. package/extensions/twitch/src/send.ts +136 -0
  1502. package/extensions/twitch/src/status.test.ts +270 -0
  1503. package/extensions/twitch/src/status.ts +178 -0
  1504. package/extensions/twitch/src/token.test.ts +171 -0
  1505. package/extensions/twitch/src/token.ts +91 -0
  1506. package/extensions/twitch/src/twitch-client.test.ts +589 -0
  1507. package/extensions/twitch/src/twitch-client.ts +277 -0
  1508. package/extensions/twitch/src/types.ts +141 -0
  1509. package/extensions/twitch/src/utils/markdown.ts +98 -0
  1510. package/extensions/twitch/src/utils/twitch.ts +78 -0
  1511. package/extensions/twitch/test/setup.ts +7 -0
  1512. package/extensions/voice-call/CHANGELOG.md +109 -0
  1513. package/extensions/voice-call/README.md +139 -0
  1514. package/extensions/voice-call/index.ts +493 -0
  1515. package/extensions/voice-call/node_modules/.bin/openclaw +21 -0
  1516. package/extensions/voice-call/openclaw.plugin.json +559 -0
  1517. package/extensions/voice-call/package.json +19 -0
  1518. package/extensions/voice-call/src/allowlist.ts +19 -0
  1519. package/extensions/voice-call/src/cli.ts +279 -0
  1520. package/extensions/voice-call/src/config.test.ts +234 -0
  1521. package/extensions/voice-call/src/config.ts +523 -0
  1522. package/extensions/voice-call/src/core-bridge.ts +159 -0
  1523. package/extensions/voice-call/src/manager/context.ts +21 -0
  1524. package/extensions/voice-call/src/manager/events.ts +188 -0
  1525. package/extensions/voice-call/src/manager/lookup.ts +35 -0
  1526. package/extensions/voice-call/src/manager/outbound.ts +275 -0
  1527. package/extensions/voice-call/src/manager/state.ts +48 -0
  1528. package/extensions/voice-call/src/manager/store.ts +91 -0
  1529. package/extensions/voice-call/src/manager/timers.ts +89 -0
  1530. package/extensions/voice-call/src/manager/twiml.ts +9 -0
  1531. package/extensions/voice-call/src/manager.test.ts +194 -0
  1532. package/extensions/voice-call/src/manager.ts +887 -0
  1533. package/extensions/voice-call/src/media-stream.test.ts +96 -0
  1534. package/extensions/voice-call/src/media-stream.ts +411 -0
  1535. package/extensions/voice-call/src/providers/base.ts +67 -0
  1536. package/extensions/voice-call/src/providers/index.ts +10 -0
  1537. package/extensions/voice-call/src/providers/mock.ts +165 -0
  1538. package/extensions/voice-call/src/providers/plivo.test.ts +27 -0
  1539. package/extensions/voice-call/src/providers/plivo.ts +515 -0
  1540. package/extensions/voice-call/src/providers/stt-openai-realtime.ts +311 -0
  1541. package/extensions/voice-call/src/providers/telnyx.ts +371 -0
  1542. package/extensions/voice-call/src/providers/tts-openai.ts +259 -0
  1543. package/extensions/voice-call/src/providers/twilio/api.ts +42 -0
  1544. package/extensions/voice-call/src/providers/twilio/webhook.ts +32 -0
  1545. package/extensions/voice-call/src/providers/twilio.test.ts +60 -0
  1546. package/extensions/voice-call/src/providers/twilio.ts +626 -0
  1547. package/extensions/voice-call/src/response-generator.ts +158 -0
  1548. package/extensions/voice-call/src/runtime.ts +212 -0
  1549. package/extensions/voice-call/src/telephony-audio.ts +90 -0
  1550. package/extensions/voice-call/src/telephony-tts.ts +104 -0
  1551. package/extensions/voice-call/src/tunnel.ts +314 -0
  1552. package/extensions/voice-call/src/types.ts +272 -0
  1553. package/extensions/voice-call/src/utils.ts +14 -0
  1554. package/extensions/voice-call/src/voice-mapping.ts +67 -0
  1555. package/extensions/voice-call/src/webhook-security.test.ts +377 -0
  1556. package/extensions/voice-call/src/webhook-security.ts +689 -0
  1557. package/extensions/voice-call/src/webhook.ts +491 -0
  1558. package/extensions/whatsapp/index.ts +17 -0
  1559. package/extensions/whatsapp/node_modules/.bin/openclaw +21 -0
  1560. package/extensions/whatsapp/openclaw.plugin.json +9 -0
  1561. package/extensions/whatsapp/package.json +14 -0
  1562. package/extensions/whatsapp/src/channel.ts +508 -0
  1563. package/extensions/whatsapp/src/runtime.ts +14 -0
  1564. package/extensions/zalo/CHANGELOG.md +89 -0
  1565. package/extensions/zalo/README.md +50 -0
  1566. package/extensions/zalo/index.ts +19 -0
  1567. package/extensions/zalo/node_modules/.bin/openclaw +21 -0
  1568. package/extensions/zalo/openclaw.plugin.json +9 -0
  1569. package/extensions/zalo/package.json +36 -0
  1570. package/extensions/zalo/src/accounts.ts +80 -0
  1571. package/extensions/zalo/src/actions.ts +67 -0
  1572. package/extensions/zalo/src/api.ts +208 -0
  1573. package/extensions/zalo/src/channel.directory.test.ts +43 -0
  1574. package/extensions/zalo/src/channel.ts +414 -0
  1575. package/extensions/zalo/src/config-schema.ts +24 -0
  1576. package/extensions/zalo/src/monitor.ts +753 -0
  1577. package/extensions/zalo/src/monitor.webhook.test.ts +73 -0
  1578. package/extensions/zalo/src/onboarding.ts +401 -0
  1579. package/extensions/zalo/src/probe.ts +46 -0
  1580. package/extensions/zalo/src/proxy.ts +21 -0
  1581. package/extensions/zalo/src/runtime.ts +14 -0
  1582. package/extensions/zalo/src/send.ts +124 -0
  1583. package/extensions/zalo/src/status-issues.ts +53 -0
  1584. package/extensions/zalo/src/token.ts +63 -0
  1585. package/extensions/zalo/src/types.ts +42 -0
  1586. package/extensions/zalouser/CHANGELOG.md +61 -0
  1587. package/extensions/zalouser/README.md +225 -0
  1588. package/extensions/zalouser/index.ts +31 -0
  1589. package/extensions/zalouser/node_modules/.bin/openclaw +21 -0
  1590. package/extensions/zalouser/openclaw.plugin.json +9 -0
  1591. package/extensions/zalouser/package.json +36 -0
  1592. package/extensions/zalouser/src/accounts.ts +135 -0
  1593. package/extensions/zalouser/src/channel.test.ts +18 -0
  1594. package/extensions/zalouser/src/channel.ts +686 -0
  1595. package/extensions/zalouser/src/config-schema.ts +27 -0
  1596. package/extensions/zalouser/src/monitor.ts +590 -0
  1597. package/extensions/zalouser/src/onboarding.ts +504 -0
  1598. package/extensions/zalouser/src/probe.ts +28 -0
  1599. package/extensions/zalouser/src/runtime.ts +14 -0
  1600. package/extensions/zalouser/src/send.ts +160 -0
  1601. package/extensions/zalouser/src/status-issues.test.ts +57 -0
  1602. package/extensions/zalouser/src/status-issues.ts +89 -0
  1603. package/extensions/zalouser/src/tool.ts +164 -0
  1604. package/extensions/zalouser/src/types.ts +108 -0
  1605. package/extensions/zalouser/src/zca.ts +202 -0
  1606. package/openclaw.mjs +14 -0
  1607. package/package.json +245 -0
  1608. package/skills/1password/SKILL.md +70 -0
  1609. package/skills/1password/references/cli-examples.md +29 -0
  1610. package/skills/1password/references/get-started.md +17 -0
  1611. package/skills/apple-notes/SKILL.md +77 -0
  1612. package/skills/apple-reminders/SKILL.md +96 -0
  1613. package/skills/bear-notes/SKILL.md +107 -0
  1614. package/skills/bird/SKILL.md +224 -0
  1615. package/skills/blogwatcher/SKILL.md +69 -0
  1616. package/skills/blucli/SKILL.md +47 -0
  1617. package/skills/bluebubbles/SKILL.md +131 -0
  1618. package/skills/camsnap/SKILL.md +45 -0
  1619. package/skills/canvas/SKILL.md +198 -0
  1620. package/skills/clawhub/SKILL.md +77 -0
  1621. package/skills/coding-agent/SKILL.md +284 -0
  1622. package/skills/discord/SKILL.md +578 -0
  1623. package/skills/eightctl/SKILL.md +50 -0
  1624. package/skills/food-order/SKILL.md +48 -0
  1625. package/skills/gemini/SKILL.md +43 -0
  1626. package/skills/ghostly-projects/SKILL.md +160 -0
  1627. package/skills/gifgrep/SKILL.md +79 -0
  1628. package/skills/github/SKILL.md +77 -0
  1629. package/skills/gog/SKILL.md +116 -0
  1630. package/skills/goplaces/SKILL.md +52 -0
  1631. package/skills/healthcheck/SKILL.md +245 -0
  1632. package/skills/himalaya/SKILL.md +257 -0
  1633. package/skills/himalaya/references/configuration.md +184 -0
  1634. package/skills/himalaya/references/message-composition.md +199 -0
  1635. package/skills/imsg/SKILL.md +74 -0
  1636. package/skills/linear/SKILL.md +111 -0
  1637. package/skills/linear/linear.sh +204 -0
  1638. package/skills/local-places/SERVER_README.md +101 -0
  1639. package/skills/local-places/SKILL.md +102 -0
  1640. package/skills/local-places/pyproject.toml +21 -0
  1641. package/skills/local-places/src/local_places/__init__.py +2 -0
  1642. package/skills/local-places/src/local_places/google_places.py +314 -0
  1643. package/skills/local-places/src/local_places/main.py +65 -0
  1644. package/skills/local-places/src/local_places/schemas.py +107 -0
  1645. package/skills/mcporter/SKILL.md +61 -0
  1646. package/skills/model-usage/SKILL.md +69 -0
  1647. package/skills/model-usage/references/codexbar-cli.md +33 -0
  1648. package/skills/model-usage/scripts/model_usage.py +310 -0
  1649. package/skills/nano-banana-pro/SKILL.md +58 -0
  1650. package/skills/nano-banana-pro/scripts/generate_image.py +184 -0
  1651. package/skills/nano-pdf/SKILL.md +38 -0
  1652. package/skills/notion/SKILL.md +172 -0
  1653. package/skills/obsidian/SKILL.md +81 -0
  1654. package/skills/openai-image-gen/SKILL.md +89 -0
  1655. package/skills/openai-image-gen/scripts/gen.py +240 -0
  1656. package/skills/openai-whisper/SKILL.md +38 -0
  1657. package/skills/openai-whisper-api/SKILL.md +52 -0
  1658. package/skills/openai-whisper-api/scripts/transcribe.sh +85 -0
  1659. package/skills/openhue/SKILL.md +51 -0
  1660. package/skills/oracle/SKILL.md +125 -0
  1661. package/skills/ordercli/SKILL.md +78 -0
  1662. package/skills/peekaboo/SKILL.md +190 -0
  1663. package/skills/sag/SKILL.md +87 -0
  1664. package/skills/session-logs/SKILL.md +115 -0
  1665. package/skills/sherpa-onnx-tts/SKILL.md +103 -0
  1666. package/skills/sherpa-onnx-tts/bin/sherpa-onnx-tts +178 -0
  1667. package/skills/skill-creator/SKILL.md +370 -0
  1668. package/skills/skill-creator/license.txt +202 -0
  1669. package/skills/skill-creator/scripts/init_skill.py +378 -0
  1670. package/skills/skill-creator/scripts/package_skill.py +111 -0
  1671. package/skills/skill-creator/scripts/quick_validate.py +101 -0
  1672. package/skills/slack/SKILL.md +144 -0
  1673. package/skills/songsee/SKILL.md +49 -0
  1674. package/skills/sonoscli/SKILL.md +46 -0
  1675. package/skills/spotify-player/SKILL.md +64 -0
  1676. package/skills/summarize/SKILL.md +87 -0
  1677. package/skills/things-mac/SKILL.md +86 -0
  1678. package/skills/tmux/SKILL.md +135 -0
  1679. package/skills/tmux/scripts/find-sessions.sh +112 -0
  1680. package/skills/tmux/scripts/wait-for-text.sh +83 -0
  1681. package/skills/trello/SKILL.md +95 -0
  1682. package/skills/video-frames/SKILL.md +46 -0
  1683. package/skills/video-frames/scripts/frame.sh +81 -0
  1684. package/skills/voice-call/SKILL.md +45 -0
  1685. package/skills/wacli/SKILL.md +72 -0
  1686. package/skills/weather/SKILL.md +54 -0
@@ -0,0 +1,4882 @@
1
+ import { t as __exportAll } from "./rolldown-runtime-Cbj13DAv.js";
2
+ import { a as resolveCanonicalConfigPath, c as resolveDefaultConfigCandidates, d as resolveIsNixMode, f as resolveLegacyStateDirs, g as resolveStateDir, h as resolveOAuthPath, i as isNixMode, l as resolveGatewayLockDir, m as resolveOAuthDir, n as DEFAULT_GATEWAY_PORT, o as resolveConfigPath, p as resolveNewStateDir, r as STATE_DIR, s as resolveConfigPathCandidate, t as CONFIG_PATH, u as resolveGatewayPort } from "./paths-scjhy7N2.js";
3
+ import { D as DEFAULT_AGENT_ID, N as normalizeAgentId, c as resolveDefaultAgentId, s as resolveAgentWorkspaceDir } from "./agent-scope-BbT4OG2N.js";
4
+ import { S as normalizeChatChannelId, p as CHANNEL_IDS } from "./subsystem-CAq3uyo7.js";
5
+ import { h as resolveUserPath } from "./utils-CKSrBNwq.js";
6
+ import { B as shouldDeferShellEnvFallback, R as loadShellEnvFallback, V as shouldEnableShellEnvFallback, gt as DEFAULT_CONTEXT_TOKENS, s as parseModelRef, z as resolveShellEnvFallbackTimeoutMs } from "./model-selection-Cv5Ox_tY.js";
7
+ import { c as resolveEnableState, l as resolveMemorySlotDecision, s as normalizePluginsConfig, t as loadPluginManifestRegistry } from "./manifest-registry-D2Yntqcb.js";
8
+ import { createRequire } from "node:module";
9
+ import os from "node:os";
10
+ import path from "node:path";
11
+ import fs from "node:fs";
12
+ import JSON5 from "json5";
13
+ import crypto from "node:crypto";
14
+ import AjvPkg from "ajv";
15
+ import { z } from "zod";
16
+
17
+ //#region src/version.ts
18
+ function readVersionFromPackageJson() {
19
+ try {
20
+ return createRequire(import.meta.url)("../package.json").version ?? null;
21
+ } catch {
22
+ return null;
23
+ }
24
+ }
25
+ function readVersionFromBuildInfo() {
26
+ try {
27
+ const require = createRequire(import.meta.url);
28
+ for (const candidate of ["../build-info.json", "./build-info.json"]) try {
29
+ const info = require(candidate);
30
+ if (info.version) return info.version;
31
+ } catch {}
32
+ return null;
33
+ } catch {
34
+ return null;
35
+ }
36
+ }
37
+ const VERSION = typeof __OPENCLAW_VERSION__ === "string" && __OPENCLAW_VERSION__ || process.env.OPENCLAW_BUNDLED_VERSION || readVersionFromPackageJson() || readVersionFromBuildInfo() || "0.0.0";
38
+
39
+ //#endregion
40
+ //#region src/config/agent-dirs.ts
41
+ var DuplicateAgentDirError = class extends Error {
42
+ constructor(duplicates) {
43
+ super(formatDuplicateAgentDirError(duplicates));
44
+ this.name = "DuplicateAgentDirError";
45
+ this.duplicates = duplicates;
46
+ }
47
+ };
48
+ function canonicalizeAgentDir(agentDir) {
49
+ const resolved = path.resolve(agentDir);
50
+ if (process.platform === "darwin" || process.platform === "win32") return resolved.toLowerCase();
51
+ return resolved;
52
+ }
53
+ function collectReferencedAgentIds(cfg) {
54
+ const ids = /* @__PURE__ */ new Set();
55
+ const agents = Array.isArray(cfg.agents?.list) ? cfg.agents?.list : [];
56
+ const defaultAgentId = agents.find((agent) => agent?.default)?.id ?? agents[0]?.id ?? DEFAULT_AGENT_ID;
57
+ ids.add(normalizeAgentId(defaultAgentId));
58
+ for (const entry of agents) if (entry?.id) ids.add(normalizeAgentId(entry.id));
59
+ const bindings = cfg.bindings;
60
+ if (Array.isArray(bindings)) for (const binding of bindings) {
61
+ const id = binding?.agentId;
62
+ if (typeof id === "string" && id.trim()) ids.add(normalizeAgentId(id));
63
+ }
64
+ return [...ids];
65
+ }
66
+ function resolveEffectiveAgentDir(cfg, agentId, deps) {
67
+ const id = normalizeAgentId(agentId);
68
+ const trimmed = (Array.isArray(cfg.agents?.list) ? cfg.agents?.list.find((agent) => normalizeAgentId(agent.id) === id)?.agentDir : void 0)?.trim();
69
+ if (trimmed) return resolveUserPath(trimmed);
70
+ const root = resolveStateDir(deps?.env ?? process.env, deps?.homedir ?? os.homedir);
71
+ return path.join(root, "agents", id, "agent");
72
+ }
73
+ function findDuplicateAgentDirs(cfg, deps) {
74
+ const byDir = /* @__PURE__ */ new Map();
75
+ for (const agentId of collectReferencedAgentIds(cfg)) {
76
+ const agentDir = resolveEffectiveAgentDir(cfg, agentId, deps);
77
+ const key = canonicalizeAgentDir(agentDir);
78
+ const entry = byDir.get(key);
79
+ if (entry) entry.agentIds.push(agentId);
80
+ else byDir.set(key, {
81
+ agentDir,
82
+ agentIds: [agentId]
83
+ });
84
+ }
85
+ return [...byDir.values()].filter((v) => v.agentIds.length > 1);
86
+ }
87
+ function formatDuplicateAgentDirError(dups) {
88
+ return [
89
+ "Duplicate agentDir detected (multi-agent config).",
90
+ "Each agent must have a unique agentDir; sharing it causes auth/session state collisions and token invalidation.",
91
+ "",
92
+ "Conflicts:",
93
+ ...dups.map((d) => `- ${d.agentDir}: ${d.agentIds.map((id) => `"${id}"`).join(", ")}`),
94
+ "",
95
+ "Fix: remove the shared agents.list[].agentDir override (or give each agent its own directory).",
96
+ "If you want to share credentials, copy auth-profiles.json instead of sharing the entire agentDir."
97
+ ].join("\n");
98
+ }
99
+
100
+ //#endregion
101
+ //#region src/config/agent-limits.ts
102
+ const DEFAULT_AGENT_MAX_CONCURRENT = 4;
103
+ const DEFAULT_SUBAGENT_MAX_CONCURRENT = 8;
104
+ function resolveAgentMaxConcurrent(cfg) {
105
+ const raw = cfg?.agents?.defaults?.maxConcurrent;
106
+ if (typeof raw === "number" && Number.isFinite(raw)) return Math.max(1, Math.floor(raw));
107
+ return DEFAULT_AGENT_MAX_CONCURRENT;
108
+ }
109
+ function resolveSubagentMaxConcurrent(cfg) {
110
+ const raw = cfg?.agents?.defaults?.subagents?.maxConcurrent;
111
+ if (typeof raw === "number" && Number.isFinite(raw)) return Math.max(1, Math.floor(raw));
112
+ return DEFAULT_SUBAGENT_MAX_CONCURRENT;
113
+ }
114
+
115
+ //#endregion
116
+ //#region src/config/talk.ts
117
+ function readTalkApiKeyFromProfile(deps = {}) {
118
+ const fsImpl = deps.fs ?? fs;
119
+ const osImpl = deps.os ?? os;
120
+ const pathImpl = deps.path ?? path;
121
+ const home = osImpl.homedir();
122
+ const candidates = [
123
+ ".profile",
124
+ ".zprofile",
125
+ ".zshrc",
126
+ ".bashrc"
127
+ ].map((name) => pathImpl.join(home, name));
128
+ for (const candidate of candidates) {
129
+ if (!fsImpl.existsSync(candidate)) continue;
130
+ try {
131
+ const value = fsImpl.readFileSync(candidate, "utf-8").match(/(?:^|\n)\s*(?:export\s+)?ELEVENLABS_API_KEY\s*=\s*["']?([^\n"']+)["']?/)?.[1]?.trim();
132
+ if (value) return value;
133
+ } catch {}
134
+ }
135
+ return null;
136
+ }
137
+ function resolveTalkApiKey(env = process.env, deps = {}) {
138
+ const envValue = (env.ELEVENLABS_API_KEY ?? "").trim();
139
+ if (envValue) return envValue;
140
+ return readTalkApiKeyFromProfile(deps);
141
+ }
142
+
143
+ //#endregion
144
+ //#region src/config/defaults.ts
145
+ let defaultWarnState = { warned: false };
146
+ const DEFAULT_MODEL_ALIASES = {
147
+ opus: "anthropic/claude-opus-4-5",
148
+ sonnet: "anthropic/claude-sonnet-4-5",
149
+ gpt: "openai/gpt-5.2",
150
+ "gpt-mini": "openai/gpt-5-mini",
151
+ gemini: "google/gemini-3-pro-preview",
152
+ "gemini-flash": "google/gemini-3-flash-preview"
153
+ };
154
+ const DEFAULT_MODEL_COST = {
155
+ input: 0,
156
+ output: 0,
157
+ cacheRead: 0,
158
+ cacheWrite: 0
159
+ };
160
+ const DEFAULT_MODEL_INPUT = ["text"];
161
+ const DEFAULT_MODEL_MAX_TOKENS = 8192;
162
+ function isPositiveNumber(value) {
163
+ return typeof value === "number" && Number.isFinite(value) && value > 0;
164
+ }
165
+ function resolveModelCost(raw) {
166
+ return {
167
+ input: typeof raw?.input === "number" ? raw.input : DEFAULT_MODEL_COST.input,
168
+ output: typeof raw?.output === "number" ? raw.output : DEFAULT_MODEL_COST.output,
169
+ cacheRead: typeof raw?.cacheRead === "number" ? raw.cacheRead : DEFAULT_MODEL_COST.cacheRead,
170
+ cacheWrite: typeof raw?.cacheWrite === "number" ? raw.cacheWrite : DEFAULT_MODEL_COST.cacheWrite
171
+ };
172
+ }
173
+ function resolveAnthropicDefaultAuthMode(cfg) {
174
+ const profiles = cfg.auth?.profiles ?? {};
175
+ const anthropicProfiles = Object.entries(profiles).filter(([, profile]) => profile?.provider === "anthropic");
176
+ const order = cfg.auth?.order?.anthropic ?? [];
177
+ for (const profileId of order) {
178
+ const entry = profiles[profileId];
179
+ if (!entry || entry.provider !== "anthropic") continue;
180
+ if (entry.mode === "api_key") return "api_key";
181
+ if (entry.mode === "oauth" || entry.mode === "token") return "oauth";
182
+ }
183
+ const hasApiKey = anthropicProfiles.some(([, profile]) => profile?.mode === "api_key");
184
+ const hasOauth = anthropicProfiles.some(([, profile]) => profile?.mode === "oauth" || profile?.mode === "token");
185
+ if (hasApiKey && !hasOauth) return "api_key";
186
+ if (hasOauth && !hasApiKey) return "oauth";
187
+ if (process.env.ANTHROPIC_OAUTH_TOKEN?.trim()) return "oauth";
188
+ if (process.env.ANTHROPIC_API_KEY?.trim()) return "api_key";
189
+ return null;
190
+ }
191
+ function resolvePrimaryModelRef(raw) {
192
+ if (!raw || typeof raw !== "string") return null;
193
+ const trimmed = raw.trim();
194
+ if (!trimmed) return null;
195
+ return DEFAULT_MODEL_ALIASES[trimmed.toLowerCase()] ?? trimmed;
196
+ }
197
+ function applyMessageDefaults(cfg) {
198
+ const messages = cfg.messages;
199
+ if (messages?.ackReactionScope !== void 0) return cfg;
200
+ const nextMessages = messages ? { ...messages } : {};
201
+ nextMessages.ackReactionScope = "group-mentions";
202
+ return {
203
+ ...cfg,
204
+ messages: nextMessages
205
+ };
206
+ }
207
+ function applySessionDefaults(cfg, options = {}) {
208
+ const session = cfg.session;
209
+ if (!session || session.mainKey === void 0) return cfg;
210
+ const trimmed = session.mainKey.trim();
211
+ const warn = options.warn ?? console.warn;
212
+ const warnState = options.warnState ?? defaultWarnState;
213
+ const next = {
214
+ ...cfg,
215
+ session: {
216
+ ...session,
217
+ mainKey: "main"
218
+ }
219
+ };
220
+ if (trimmed && trimmed !== "main" && !warnState.warned) {
221
+ warnState.warned = true;
222
+ warn("session.mainKey is ignored; main session is always \"main\".");
223
+ }
224
+ return next;
225
+ }
226
+ function applyTalkApiKey(config) {
227
+ const resolved = resolveTalkApiKey();
228
+ if (!resolved) return config;
229
+ if (config.talk?.apiKey?.trim()) return config;
230
+ return {
231
+ ...config,
232
+ talk: {
233
+ ...config.talk,
234
+ apiKey: resolved
235
+ }
236
+ };
237
+ }
238
+ function applyModelDefaults(cfg) {
239
+ let mutated = false;
240
+ let nextCfg = cfg;
241
+ const providerConfig = nextCfg.models?.providers;
242
+ if (providerConfig) {
243
+ const nextProviders = { ...providerConfig };
244
+ for (const [providerId, provider] of Object.entries(providerConfig)) {
245
+ const models = provider.models;
246
+ if (!Array.isArray(models) || models.length === 0) continue;
247
+ let providerMutated = false;
248
+ const nextModels = models.map((model) => {
249
+ const raw = model;
250
+ let modelMutated = false;
251
+ const reasoning = typeof raw.reasoning === "boolean" ? raw.reasoning : false;
252
+ if (raw.reasoning !== reasoning) modelMutated = true;
253
+ const input = raw.input ?? [...DEFAULT_MODEL_INPUT];
254
+ if (raw.input === void 0) modelMutated = true;
255
+ const cost = resolveModelCost(raw.cost);
256
+ if (!raw.cost || raw.cost.input !== cost.input || raw.cost.output !== cost.output || raw.cost.cacheRead !== cost.cacheRead || raw.cost.cacheWrite !== cost.cacheWrite) modelMutated = true;
257
+ const contextWindow = isPositiveNumber(raw.contextWindow) ? raw.contextWindow : DEFAULT_CONTEXT_TOKENS;
258
+ if (raw.contextWindow !== contextWindow) modelMutated = true;
259
+ const defaultMaxTokens = Math.min(DEFAULT_MODEL_MAX_TOKENS, contextWindow);
260
+ const maxTokens = isPositiveNumber(raw.maxTokens) ? raw.maxTokens : defaultMaxTokens;
261
+ if (raw.maxTokens !== maxTokens) modelMutated = true;
262
+ if (!modelMutated) return model;
263
+ providerMutated = true;
264
+ return {
265
+ ...raw,
266
+ reasoning,
267
+ input,
268
+ cost,
269
+ contextWindow,
270
+ maxTokens
271
+ };
272
+ });
273
+ if (!providerMutated) continue;
274
+ nextProviders[providerId] = {
275
+ ...provider,
276
+ models: nextModels
277
+ };
278
+ mutated = true;
279
+ }
280
+ if (mutated) nextCfg = {
281
+ ...nextCfg,
282
+ models: {
283
+ ...nextCfg.models,
284
+ providers: nextProviders
285
+ }
286
+ };
287
+ }
288
+ const existingAgent = nextCfg.agents?.defaults;
289
+ if (!existingAgent) return mutated ? nextCfg : cfg;
290
+ const existingModels = existingAgent.models ?? {};
291
+ if (Object.keys(existingModels).length === 0) return mutated ? nextCfg : cfg;
292
+ const nextModels = { ...existingModels };
293
+ for (const [alias, target] of Object.entries(DEFAULT_MODEL_ALIASES)) {
294
+ const entry = nextModels[target];
295
+ if (!entry) continue;
296
+ if (entry.alias !== void 0) continue;
297
+ nextModels[target] = {
298
+ ...entry,
299
+ alias
300
+ };
301
+ mutated = true;
302
+ }
303
+ if (!mutated) return cfg;
304
+ return {
305
+ ...nextCfg,
306
+ agents: {
307
+ ...nextCfg.agents,
308
+ defaults: {
309
+ ...existingAgent,
310
+ models: nextModels
311
+ }
312
+ }
313
+ };
314
+ }
315
+ function applyAgentDefaults(cfg) {
316
+ const agents = cfg.agents;
317
+ const defaults = agents?.defaults;
318
+ const hasMax = typeof defaults?.maxConcurrent === "number" && Number.isFinite(defaults.maxConcurrent);
319
+ const hasSubMax = typeof defaults?.subagents?.maxConcurrent === "number" && Number.isFinite(defaults.subagents.maxConcurrent);
320
+ if (hasMax && hasSubMax) return cfg;
321
+ let mutated = false;
322
+ const nextDefaults = defaults ? { ...defaults } : {};
323
+ if (!hasMax) {
324
+ nextDefaults.maxConcurrent = DEFAULT_AGENT_MAX_CONCURRENT;
325
+ mutated = true;
326
+ }
327
+ const nextSubagents = defaults?.subagents ? { ...defaults.subagents } : {};
328
+ if (!hasSubMax) {
329
+ nextSubagents.maxConcurrent = DEFAULT_SUBAGENT_MAX_CONCURRENT;
330
+ mutated = true;
331
+ }
332
+ if (!mutated) return cfg;
333
+ return {
334
+ ...cfg,
335
+ agents: {
336
+ ...agents,
337
+ defaults: {
338
+ ...nextDefaults,
339
+ subagents: nextSubagents
340
+ }
341
+ }
342
+ };
343
+ }
344
+ function applyLoggingDefaults(cfg) {
345
+ const logging = cfg.logging;
346
+ if (!logging) return cfg;
347
+ if (logging.redactSensitive) return cfg;
348
+ return {
349
+ ...cfg,
350
+ logging: {
351
+ ...logging,
352
+ redactSensitive: "tools"
353
+ }
354
+ };
355
+ }
356
+ function applyContextPruningDefaults(cfg) {
357
+ const defaults = cfg.agents?.defaults;
358
+ if (!defaults) return cfg;
359
+ const authMode = resolveAnthropicDefaultAuthMode(cfg);
360
+ if (!authMode) return cfg;
361
+ let mutated = false;
362
+ const nextDefaults = { ...defaults };
363
+ const contextPruning = defaults.contextPruning ?? {};
364
+ const heartbeat = defaults.heartbeat ?? {};
365
+ if (defaults.contextPruning?.mode === void 0) {
366
+ nextDefaults.contextPruning = {
367
+ ...contextPruning,
368
+ mode: "cache-ttl",
369
+ ttl: defaults.contextPruning?.ttl ?? "1h"
370
+ };
371
+ mutated = true;
372
+ }
373
+ if (defaults.heartbeat?.every === void 0) {
374
+ nextDefaults.heartbeat = {
375
+ ...heartbeat,
376
+ every: authMode === "oauth" ? "1h" : "30m"
377
+ };
378
+ mutated = true;
379
+ }
380
+ if (authMode === "api_key") {
381
+ const nextModels = defaults.models ? { ...defaults.models } : {};
382
+ let modelsMutated = false;
383
+ for (const [key, entry] of Object.entries(nextModels)) {
384
+ const parsed = parseModelRef(key, "anthropic");
385
+ if (!parsed || parsed.provider !== "anthropic") continue;
386
+ const current = entry ?? {};
387
+ const params = current.params ?? {};
388
+ if (typeof params.cacheRetention === "string") continue;
389
+ nextModels[key] = {
390
+ ...current,
391
+ params: {
392
+ ...params,
393
+ cacheRetention: "short"
394
+ }
395
+ };
396
+ modelsMutated = true;
397
+ }
398
+ const primary = resolvePrimaryModelRef(defaults.model?.primary ?? void 0);
399
+ if (primary) {
400
+ const parsedPrimary = parseModelRef(primary, "anthropic");
401
+ if (parsedPrimary?.provider === "anthropic") {
402
+ const key = `${parsedPrimary.provider}/${parsedPrimary.model}`;
403
+ const current = nextModels[key] ?? {};
404
+ const params = current.params ?? {};
405
+ if (typeof params.cacheRetention !== "string") {
406
+ nextModels[key] = {
407
+ ...current,
408
+ params: {
409
+ ...params,
410
+ cacheRetention: "short"
411
+ }
412
+ };
413
+ modelsMutated = true;
414
+ }
415
+ }
416
+ }
417
+ if (modelsMutated) {
418
+ nextDefaults.models = nextModels;
419
+ mutated = true;
420
+ }
421
+ }
422
+ if (!mutated) return cfg;
423
+ return {
424
+ ...cfg,
425
+ agents: {
426
+ ...cfg.agents,
427
+ defaults: nextDefaults
428
+ }
429
+ };
430
+ }
431
+ function applyCompactionDefaults(cfg) {
432
+ const defaults = cfg.agents?.defaults;
433
+ if (!defaults) return cfg;
434
+ const compaction = defaults?.compaction;
435
+ if (compaction?.mode) return cfg;
436
+ return {
437
+ ...cfg,
438
+ agents: {
439
+ ...cfg.agents,
440
+ defaults: {
441
+ ...defaults,
442
+ compaction: {
443
+ ...compaction,
444
+ mode: "safeguard"
445
+ }
446
+ }
447
+ }
448
+ };
449
+ }
450
+
451
+ //#endregion
452
+ //#region src/config/env-substitution.ts
453
+ /**
454
+ * Environment variable substitution for config values.
455
+ *
456
+ * Supports `${VAR_NAME}` syntax in string values, substituted at config load time.
457
+ * - Only uppercase env vars are matched: `[A-Z_][A-Z0-9_]*`
458
+ * - Escape with `$${}` to output literal `${}`
459
+ * - Missing env vars throw `MissingEnvVarError` with context
460
+ *
461
+ * @example
462
+ * ```json5
463
+ * {
464
+ * models: {
465
+ * providers: {
466
+ * "vercel-gateway": {
467
+ * apiKey: "${VERCEL_GATEWAY_API_KEY}"
468
+ * }
469
+ * }
470
+ * }
471
+ * }
472
+ * ```
473
+ */
474
+ const ENV_VAR_NAME_PATTERN = /^[A-Z_][A-Z0-9_]*$/;
475
+ var MissingEnvVarError = class extends Error {
476
+ constructor(varName, configPath) {
477
+ super(`Missing env var "${varName}" referenced at config path: ${configPath}`);
478
+ this.varName = varName;
479
+ this.configPath = configPath;
480
+ this.name = "MissingEnvVarError";
481
+ }
482
+ };
483
+ function isPlainObject$4(value) {
484
+ return typeof value === "object" && value !== null && !Array.isArray(value) && Object.prototype.toString.call(value) === "[object Object]";
485
+ }
486
+ function substituteString(value, env, configPath) {
487
+ if (!value.includes("$")) return value;
488
+ const chunks = [];
489
+ for (let i = 0; i < value.length; i += 1) {
490
+ const char = value[i];
491
+ if (char !== "$") {
492
+ chunks.push(char);
493
+ continue;
494
+ }
495
+ const next = value[i + 1];
496
+ const afterNext = value[i + 2];
497
+ if (next === "$" && afterNext === "{") {
498
+ const start = i + 3;
499
+ const end = value.indexOf("}", start);
500
+ if (end !== -1) {
501
+ const name = value.slice(start, end);
502
+ if (ENV_VAR_NAME_PATTERN.test(name)) {
503
+ chunks.push(`\${${name}}`);
504
+ i = end;
505
+ continue;
506
+ }
507
+ }
508
+ }
509
+ if (next === "{") {
510
+ const start = i + 2;
511
+ const end = value.indexOf("}", start);
512
+ if (end !== -1) {
513
+ const name = value.slice(start, end);
514
+ if (ENV_VAR_NAME_PATTERN.test(name)) {
515
+ const envValue = env[name];
516
+ if (envValue === void 0 || envValue === "") throw new MissingEnvVarError(name, configPath);
517
+ chunks.push(envValue);
518
+ i = end;
519
+ continue;
520
+ }
521
+ }
522
+ }
523
+ chunks.push(char);
524
+ }
525
+ return chunks.join("");
526
+ }
527
+ function substituteAny(value, env, path) {
528
+ if (typeof value === "string") return substituteString(value, env, path);
529
+ if (Array.isArray(value)) return value.map((item, index) => substituteAny(item, env, `${path}[${index}]`));
530
+ if (isPlainObject$4(value)) {
531
+ const result = {};
532
+ for (const [key, val] of Object.entries(value)) result[key] = substituteAny(val, env, path ? `${path}.${key}` : key);
533
+ return result;
534
+ }
535
+ return value;
536
+ }
537
+ /**
538
+ * Resolves `${VAR_NAME}` environment variable references in config values.
539
+ *
540
+ * @param obj - The parsed config object (after JSON5 parse and $include resolution)
541
+ * @param env - Environment variables to use for substitution (defaults to process.env)
542
+ * @returns The config object with env vars substituted
543
+ * @throws {MissingEnvVarError} If a referenced env var is not set or empty
544
+ */
545
+ function resolveConfigEnvVars(obj, env = process.env) {
546
+ return substituteAny(obj, env, "");
547
+ }
548
+
549
+ //#endregion
550
+ //#region src/config/env-vars.ts
551
+ function collectConfigEnvVars(cfg) {
552
+ const envConfig = cfg?.env;
553
+ if (!envConfig) return {};
554
+ const entries = {};
555
+ if (envConfig.vars) for (const [key, value] of Object.entries(envConfig.vars)) {
556
+ if (!value) continue;
557
+ entries[key] = value;
558
+ }
559
+ for (const [key, value] of Object.entries(envConfig)) {
560
+ if (key === "shellEnv" || key === "vars") continue;
561
+ if (typeof value !== "string" || !value.trim()) continue;
562
+ entries[key] = value;
563
+ }
564
+ return entries;
565
+ }
566
+
567
+ //#endregion
568
+ //#region src/config/includes.ts
569
+ /**
570
+ * Config includes: $include directive for modular configs
571
+ *
572
+ * @example
573
+ * ```json5
574
+ * {
575
+ * "$include": "./base.json5", // single file
576
+ * "$include": ["./a.json5", "./b.json5"] // merge multiple
577
+ * }
578
+ * ```
579
+ */
580
+ const INCLUDE_KEY = "$include";
581
+ const MAX_INCLUDE_DEPTH = 10;
582
+ var ConfigIncludeError = class extends Error {
583
+ constructor(message, includePath, cause) {
584
+ super(message);
585
+ this.includePath = includePath;
586
+ this.cause = cause;
587
+ this.name = "ConfigIncludeError";
588
+ }
589
+ };
590
+ var CircularIncludeError = class extends ConfigIncludeError {
591
+ constructor(chain) {
592
+ super(`Circular include detected: ${chain.join(" -> ")}`, chain[chain.length - 1]);
593
+ this.chain = chain;
594
+ this.name = "CircularIncludeError";
595
+ }
596
+ };
597
+ function isPlainObject$3(value) {
598
+ return typeof value === "object" && value !== null && !Array.isArray(value) && Object.prototype.toString.call(value) === "[object Object]";
599
+ }
600
+ /** Deep merge: arrays concatenate, objects merge recursively, primitives: source wins */
601
+ function deepMerge(target, source) {
602
+ if (Array.isArray(target) && Array.isArray(source)) return [...target, ...source];
603
+ if (isPlainObject$3(target) && isPlainObject$3(source)) {
604
+ const result = { ...target };
605
+ for (const key of Object.keys(source)) result[key] = key in result ? deepMerge(result[key], source[key]) : source[key];
606
+ return result;
607
+ }
608
+ return source;
609
+ }
610
+ var IncludeProcessor = class IncludeProcessor {
611
+ constructor(basePath, resolver) {
612
+ this.basePath = basePath;
613
+ this.resolver = resolver;
614
+ this.visited = /* @__PURE__ */ new Set();
615
+ this.depth = 0;
616
+ this.visited.add(path.normalize(basePath));
617
+ }
618
+ process(obj) {
619
+ if (Array.isArray(obj)) return obj.map((item) => this.process(item));
620
+ if (!isPlainObject$3(obj)) return obj;
621
+ if (!(INCLUDE_KEY in obj)) return this.processObject(obj);
622
+ return this.processInclude(obj);
623
+ }
624
+ processObject(obj) {
625
+ const result = {};
626
+ for (const [key, value] of Object.entries(obj)) result[key] = this.process(value);
627
+ return result;
628
+ }
629
+ processInclude(obj) {
630
+ const includeValue = obj[INCLUDE_KEY];
631
+ const otherKeys = Object.keys(obj).filter((k) => k !== INCLUDE_KEY);
632
+ const included = this.resolveInclude(includeValue);
633
+ if (otherKeys.length === 0) return included;
634
+ if (!isPlainObject$3(included)) throw new ConfigIncludeError("Sibling keys require included content to be an object", typeof includeValue === "string" ? includeValue : INCLUDE_KEY);
635
+ const rest = {};
636
+ for (const key of otherKeys) rest[key] = this.process(obj[key]);
637
+ return deepMerge(included, rest);
638
+ }
639
+ resolveInclude(value) {
640
+ if (typeof value === "string") return this.loadFile(value);
641
+ if (Array.isArray(value)) return value.reduce((merged, item) => {
642
+ if (typeof item !== "string") throw new ConfigIncludeError(`Invalid $include array item: expected string, got ${typeof item}`, String(item));
643
+ return deepMerge(merged, this.loadFile(item));
644
+ }, {});
645
+ throw new ConfigIncludeError(`Invalid $include value: expected string or array of strings, got ${typeof value}`, String(value));
646
+ }
647
+ loadFile(includePath) {
648
+ const resolvedPath = this.resolvePath(includePath);
649
+ this.checkCircular(resolvedPath);
650
+ this.checkDepth(includePath);
651
+ const raw = this.readFile(includePath, resolvedPath);
652
+ const parsed = this.parseFile(includePath, resolvedPath, raw);
653
+ return this.processNested(resolvedPath, parsed);
654
+ }
655
+ resolvePath(includePath) {
656
+ const resolved = path.isAbsolute(includePath) ? includePath : path.resolve(path.dirname(this.basePath), includePath);
657
+ return path.normalize(resolved);
658
+ }
659
+ checkCircular(resolvedPath) {
660
+ if (this.visited.has(resolvedPath)) throw new CircularIncludeError([...this.visited, resolvedPath]);
661
+ }
662
+ checkDepth(includePath) {
663
+ if (this.depth >= MAX_INCLUDE_DEPTH) throw new ConfigIncludeError(`Maximum include depth (${MAX_INCLUDE_DEPTH}) exceeded at: ${includePath}`, includePath);
664
+ }
665
+ readFile(includePath, resolvedPath) {
666
+ try {
667
+ return this.resolver.readFile(resolvedPath);
668
+ } catch (err) {
669
+ throw new ConfigIncludeError(`Failed to read include file: ${includePath} (resolved: ${resolvedPath})`, includePath, err instanceof Error ? err : void 0);
670
+ }
671
+ }
672
+ parseFile(includePath, resolvedPath, raw) {
673
+ try {
674
+ return this.resolver.parseJson(raw);
675
+ } catch (err) {
676
+ throw new ConfigIncludeError(`Failed to parse include file: ${includePath} (resolved: ${resolvedPath})`, includePath, err instanceof Error ? err : void 0);
677
+ }
678
+ }
679
+ processNested(resolvedPath, parsed) {
680
+ const nested = new IncludeProcessor(resolvedPath, this.resolver);
681
+ nested.visited = new Set([...this.visited, resolvedPath]);
682
+ nested.depth = this.depth + 1;
683
+ return nested.process(parsed);
684
+ }
685
+ };
686
+ const defaultResolver = {
687
+ readFile: (p) => fs.readFileSync(p, "utf-8"),
688
+ parseJson: (raw) => JSON5.parse(raw)
689
+ };
690
+ /**
691
+ * Resolves all $include directives in a parsed config object.
692
+ */
693
+ function resolveConfigIncludes(obj, configPath, resolver = defaultResolver) {
694
+ return new IncludeProcessor(configPath, resolver).process(obj);
695
+ }
696
+
697
+ //#endregion
698
+ //#region src/config/legacy.shared.ts
699
+ const isRecord$1 = (value) => Boolean(value && typeof value === "object" && !Array.isArray(value));
700
+ const getRecord = (value) => isRecord$1(value) ? value : null;
701
+ const ensureRecord = (root, key) => {
702
+ const existing = root[key];
703
+ if (isRecord$1(existing)) return existing;
704
+ const next = {};
705
+ root[key] = next;
706
+ return next;
707
+ };
708
+ const mergeMissing = (target, source) => {
709
+ for (const [key, value] of Object.entries(source)) {
710
+ if (value === void 0) continue;
711
+ const existing = target[key];
712
+ if (existing === void 0) {
713
+ target[key] = value;
714
+ continue;
715
+ }
716
+ if (isRecord$1(existing) && isRecord$1(value)) mergeMissing(existing, value);
717
+ }
718
+ };
719
+ const AUDIO_TRANSCRIPTION_CLI_ALLOWLIST = new Set(["whisper"]);
720
+ const mapLegacyAudioTranscription = (value) => {
721
+ const transcriber = getRecord(value);
722
+ const command = Array.isArray(transcriber?.command) ? transcriber?.command : null;
723
+ if (!command || command.length === 0) return null;
724
+ const rawExecutable = String(command[0] ?? "").trim();
725
+ if (!rawExecutable) return null;
726
+ const executableName = rawExecutable.split(/[\\/]/).pop() ?? rawExecutable;
727
+ if (!AUDIO_TRANSCRIPTION_CLI_ALLOWLIST.has(executableName)) return null;
728
+ const args = command.slice(1).map((part) => String(part));
729
+ const timeoutSeconds = typeof transcriber?.timeoutSeconds === "number" ? transcriber?.timeoutSeconds : void 0;
730
+ const result = {
731
+ command: rawExecutable,
732
+ type: "cli"
733
+ };
734
+ if (args.length > 0) result.args = args;
735
+ if (timeoutSeconds !== void 0) result.timeoutSeconds = timeoutSeconds;
736
+ return result;
737
+ };
738
+ const getAgentsList = (agents) => {
739
+ const list = agents?.list;
740
+ return Array.isArray(list) ? list : [];
741
+ };
742
+ const resolveDefaultAgentIdFromRaw = (raw) => {
743
+ const list = getAgentsList(getRecord(raw.agents));
744
+ const defaultEntry = list.find((entry) => isRecord$1(entry) && entry.default === true && typeof entry.id === "string" && entry.id.trim() !== "");
745
+ if (defaultEntry) return defaultEntry.id.trim();
746
+ const routing = getRecord(raw.routing);
747
+ const routingDefault = typeof routing?.defaultAgentId === "string" ? routing.defaultAgentId.trim() : "";
748
+ if (routingDefault) return routingDefault;
749
+ const firstEntry = list.find((entry) => isRecord$1(entry) && typeof entry.id === "string" && entry.id.trim() !== "");
750
+ if (firstEntry) return firstEntry.id.trim();
751
+ return "main";
752
+ };
753
+ const ensureAgentEntry = (list, id) => {
754
+ const normalized = id.trim();
755
+ const existing = list.find((entry) => isRecord$1(entry) && typeof entry.id === "string" && entry.id.trim() === normalized);
756
+ if (existing) return existing;
757
+ const created = { id: normalized };
758
+ list.push(created);
759
+ return created;
760
+ };
761
+
762
+ //#endregion
763
+ //#region src/config/legacy.migrations.part-1.ts
764
+ const LEGACY_CONFIG_MIGRATIONS_PART_1 = [
765
+ {
766
+ id: "bindings.match.provider->bindings.match.channel",
767
+ describe: "Move bindings[].match.provider to bindings[].match.channel",
768
+ apply: (raw, changes) => {
769
+ const bindings = Array.isArray(raw.bindings) ? raw.bindings : null;
770
+ if (!bindings) return;
771
+ let touched = false;
772
+ for (const entry of bindings) {
773
+ if (!isRecord$1(entry)) continue;
774
+ const match = getRecord(entry.match);
775
+ if (!match) continue;
776
+ if (typeof match.channel === "string" && match.channel.trim()) continue;
777
+ const provider = typeof match.provider === "string" ? match.provider.trim() : "";
778
+ if (!provider) continue;
779
+ match.channel = provider;
780
+ delete match.provider;
781
+ entry.match = match;
782
+ touched = true;
783
+ }
784
+ if (touched) {
785
+ raw.bindings = bindings;
786
+ changes.push("Moved bindings[].match.provider → bindings[].match.channel.");
787
+ }
788
+ }
789
+ },
790
+ {
791
+ id: "bindings.match.accountID->bindings.match.accountId",
792
+ describe: "Move bindings[].match.accountID to bindings[].match.accountId",
793
+ apply: (raw, changes) => {
794
+ const bindings = Array.isArray(raw.bindings) ? raw.bindings : null;
795
+ if (!bindings) return;
796
+ let touched = false;
797
+ for (const entry of bindings) {
798
+ if (!isRecord$1(entry)) continue;
799
+ const match = getRecord(entry.match);
800
+ if (!match) continue;
801
+ if (match.accountId !== void 0) continue;
802
+ const accountID = typeof match.accountID === "string" ? match.accountID.trim() : match.accountID;
803
+ if (!accountID) continue;
804
+ match.accountId = accountID;
805
+ delete match.accountID;
806
+ entry.match = match;
807
+ touched = true;
808
+ }
809
+ if (touched) {
810
+ raw.bindings = bindings;
811
+ changes.push("Moved bindings[].match.accountID → bindings[].match.accountId.");
812
+ }
813
+ }
814
+ },
815
+ {
816
+ id: "session.sendPolicy.rules.match.provider->match.channel",
817
+ describe: "Move session.sendPolicy.rules[].match.provider to match.channel",
818
+ apply: (raw, changes) => {
819
+ const session = getRecord(raw.session);
820
+ if (!session) return;
821
+ const sendPolicy = getRecord(session.sendPolicy);
822
+ if (!sendPolicy) return;
823
+ const rules = Array.isArray(sendPolicy.rules) ? sendPolicy.rules : null;
824
+ if (!rules) return;
825
+ let touched = false;
826
+ for (const rule of rules) {
827
+ if (!isRecord$1(rule)) continue;
828
+ const match = getRecord(rule.match);
829
+ if (!match) continue;
830
+ if (typeof match.channel === "string" && match.channel.trim()) continue;
831
+ const provider = typeof match.provider === "string" ? match.provider.trim() : "";
832
+ if (!provider) continue;
833
+ match.channel = provider;
834
+ delete match.provider;
835
+ rule.match = match;
836
+ touched = true;
837
+ }
838
+ if (touched) {
839
+ sendPolicy.rules = rules;
840
+ session.sendPolicy = sendPolicy;
841
+ raw.session = session;
842
+ changes.push("Moved session.sendPolicy.rules[].match.provider → match.channel.");
843
+ }
844
+ }
845
+ },
846
+ {
847
+ id: "messages.queue.byProvider->byChannel",
848
+ describe: "Move messages.queue.byProvider to messages.queue.byChannel",
849
+ apply: (raw, changes) => {
850
+ const messages = getRecord(raw.messages);
851
+ if (!messages) return;
852
+ const queue = getRecord(messages.queue);
853
+ if (!queue) return;
854
+ if (queue.byProvider === void 0) return;
855
+ if (queue.byChannel === void 0) {
856
+ queue.byChannel = queue.byProvider;
857
+ changes.push("Moved messages.queue.byProvider → messages.queue.byChannel.");
858
+ } else changes.push("Removed messages.queue.byProvider (messages.queue.byChannel already set).");
859
+ delete queue.byProvider;
860
+ messages.queue = queue;
861
+ raw.messages = messages;
862
+ }
863
+ },
864
+ {
865
+ id: "providers->channels",
866
+ describe: "Move provider config sections to channels.*",
867
+ apply: (raw, changes) => {
868
+ const legacyEntries = [
869
+ "whatsapp",
870
+ "telegram",
871
+ "discord",
872
+ "slack",
873
+ "signal",
874
+ "imessage",
875
+ "msteams"
876
+ ].filter((key) => isRecord$1(raw[key]));
877
+ if (legacyEntries.length === 0) return;
878
+ const channels = ensureRecord(raw, "channels");
879
+ for (const key of legacyEntries) {
880
+ const legacy = getRecord(raw[key]);
881
+ if (!legacy) continue;
882
+ const channelEntry = ensureRecord(channels, key);
883
+ const hadEntries = Object.keys(channelEntry).length > 0;
884
+ mergeMissing(channelEntry, legacy);
885
+ channels[key] = channelEntry;
886
+ delete raw[key];
887
+ changes.push(hadEntries ? `Merged ${key} → channels.${key}.` : `Moved ${key} → channels.${key}.`);
888
+ }
889
+ raw.channels = channels;
890
+ }
891
+ },
892
+ {
893
+ id: "routing.allowFrom->channels.whatsapp.allowFrom",
894
+ describe: "Move routing.allowFrom to channels.whatsapp.allowFrom",
895
+ apply: (raw, changes) => {
896
+ const routing = raw.routing;
897
+ if (!routing || typeof routing !== "object") return;
898
+ const allowFrom = routing.allowFrom;
899
+ if (allowFrom === void 0) return;
900
+ const channels = getRecord(raw.channels);
901
+ const whatsapp = channels ? getRecord(channels.whatsapp) : null;
902
+ if (!whatsapp) {
903
+ delete routing.allowFrom;
904
+ if (Object.keys(routing).length === 0) delete raw.routing;
905
+ changes.push("Removed routing.allowFrom (channels.whatsapp not configured).");
906
+ return;
907
+ }
908
+ if (whatsapp.allowFrom === void 0) {
909
+ whatsapp.allowFrom = allowFrom;
910
+ changes.push("Moved routing.allowFrom → channels.whatsapp.allowFrom.");
911
+ } else changes.push("Removed routing.allowFrom (channels.whatsapp.allowFrom already set).");
912
+ delete routing.allowFrom;
913
+ if (Object.keys(routing).length === 0) delete raw.routing;
914
+ channels.whatsapp = whatsapp;
915
+ raw.channels = channels;
916
+ }
917
+ },
918
+ {
919
+ id: "routing.groupChat.requireMention->groups.*.requireMention",
920
+ describe: "Move routing.groupChat.requireMention to channels.whatsapp/telegram/imessage groups",
921
+ apply: (raw, changes) => {
922
+ const routing = raw.routing;
923
+ if (!routing || typeof routing !== "object") return;
924
+ const groupChat = routing.groupChat && typeof routing.groupChat === "object" ? routing.groupChat : null;
925
+ if (!groupChat) return;
926
+ const requireMention = groupChat.requireMention;
927
+ if (requireMention === void 0) return;
928
+ const channels = ensureRecord(raw, "channels");
929
+ const applyTo = (key, options) => {
930
+ if (options?.requireExisting && !isRecord$1(channels[key])) return;
931
+ const section = channels[key] && typeof channels[key] === "object" ? channels[key] : {};
932
+ const groups = section.groups && typeof section.groups === "object" ? section.groups : {};
933
+ const defaultKey = "*";
934
+ const entry = groups[defaultKey] && typeof groups[defaultKey] === "object" ? groups[defaultKey] : {};
935
+ if (entry.requireMention === void 0) {
936
+ entry.requireMention = requireMention;
937
+ groups[defaultKey] = entry;
938
+ section.groups = groups;
939
+ channels[key] = section;
940
+ changes.push(`Moved routing.groupChat.requireMention → channels.${key}.groups."*".requireMention.`);
941
+ } else changes.push(`Removed routing.groupChat.requireMention (channels.${key}.groups."*" already set).`);
942
+ };
943
+ applyTo("whatsapp", { requireExisting: true });
944
+ applyTo("telegram");
945
+ applyTo("imessage");
946
+ delete groupChat.requireMention;
947
+ if (Object.keys(groupChat).length === 0) delete routing.groupChat;
948
+ if (Object.keys(routing).length === 0) delete raw.routing;
949
+ raw.channels = channels;
950
+ }
951
+ },
952
+ {
953
+ id: "gateway.token->gateway.auth.token",
954
+ describe: "Move gateway.token to gateway.auth.token",
955
+ apply: (raw, changes) => {
956
+ const gateway = raw.gateway;
957
+ if (!gateway || typeof gateway !== "object") return;
958
+ const token = gateway.token;
959
+ if (token === void 0) return;
960
+ const gatewayObj = gateway;
961
+ const auth = gatewayObj.auth && typeof gatewayObj.auth === "object" ? gatewayObj.auth : {};
962
+ if (auth.token === void 0) {
963
+ auth.token = token;
964
+ if (!auth.mode) auth.mode = "token";
965
+ changes.push("Moved gateway.token → gateway.auth.token.");
966
+ } else changes.push("Removed gateway.token (gateway.auth.token already set).");
967
+ delete gatewayObj.token;
968
+ if (Object.keys(auth).length > 0) gatewayObj.auth = auth;
969
+ raw.gateway = gatewayObj;
970
+ }
971
+ },
972
+ {
973
+ id: "telegram.requireMention->channels.telegram.groups.*.requireMention",
974
+ describe: "Move telegram.requireMention to channels.telegram.groups.*.requireMention",
975
+ apply: (raw, changes) => {
976
+ const channels = ensureRecord(raw, "channels");
977
+ const telegram = channels.telegram;
978
+ if (!telegram || typeof telegram !== "object") return;
979
+ const requireMention = telegram.requireMention;
980
+ if (requireMention === void 0) return;
981
+ const groups = telegram.groups && typeof telegram.groups === "object" ? telegram.groups : {};
982
+ const defaultKey = "*";
983
+ const entry = groups[defaultKey] && typeof groups[defaultKey] === "object" ? groups[defaultKey] : {};
984
+ if (entry.requireMention === void 0) {
985
+ entry.requireMention = requireMention;
986
+ groups[defaultKey] = entry;
987
+ telegram.groups = groups;
988
+ changes.push("Moved telegram.requireMention → channels.telegram.groups.\"*\".requireMention.");
989
+ } else changes.push("Removed telegram.requireMention (channels.telegram.groups.\"*\" already set).");
990
+ delete telegram.requireMention;
991
+ channels.telegram = telegram;
992
+ raw.channels = channels;
993
+ }
994
+ }
995
+ ];
996
+
997
+ //#endregion
998
+ //#region src/config/legacy.migrations.part-2.ts
999
+ const LEGACY_CONFIG_MIGRATIONS_PART_2 = [
1000
+ {
1001
+ id: "agent.model-config-v2",
1002
+ describe: "Migrate legacy agent.model/allowedModels/modelAliases/modelFallbacks/imageModelFallbacks to agent.models + model lists",
1003
+ apply: (raw, changes) => {
1004
+ const agentRoot = getRecord(raw.agent);
1005
+ const defaults = getRecord(getRecord(raw.agents)?.defaults);
1006
+ const agent = agentRoot ?? defaults;
1007
+ if (!agent) return;
1008
+ const label = agentRoot ? "agent" : "agents.defaults";
1009
+ const legacyModel = typeof agent.model === "string" ? String(agent.model) : void 0;
1010
+ const legacyImageModel = typeof agent.imageModel === "string" ? String(agent.imageModel) : void 0;
1011
+ const legacyAllowed = Array.isArray(agent.allowedModels) ? agent.allowedModels.map(String) : [];
1012
+ const legacyModelFallbacks = Array.isArray(agent.modelFallbacks) ? agent.modelFallbacks.map(String) : [];
1013
+ const legacyImageModelFallbacks = Array.isArray(agent.imageModelFallbacks) ? agent.imageModelFallbacks.map(String) : [];
1014
+ const legacyAliases = agent.modelAliases && typeof agent.modelAliases === "object" ? agent.modelAliases : {};
1015
+ if (!(legacyModel || legacyImageModel || legacyAllowed.length > 0 || legacyModelFallbacks.length > 0 || legacyImageModelFallbacks.length > 0 || Object.keys(legacyAliases).length > 0)) return;
1016
+ const models = agent.models && typeof agent.models === "object" ? agent.models : {};
1017
+ const ensureModel = (rawKey) => {
1018
+ if (typeof rawKey !== "string") return;
1019
+ const key = rawKey.trim();
1020
+ if (!key) return;
1021
+ if (!models[key]) models[key] = {};
1022
+ };
1023
+ ensureModel(legacyModel);
1024
+ ensureModel(legacyImageModel);
1025
+ for (const key of legacyAllowed) ensureModel(key);
1026
+ for (const key of legacyModelFallbacks) ensureModel(key);
1027
+ for (const key of legacyImageModelFallbacks) ensureModel(key);
1028
+ for (const target of Object.values(legacyAliases)) {
1029
+ if (typeof target !== "string") continue;
1030
+ ensureModel(target);
1031
+ }
1032
+ for (const [alias, targetRaw] of Object.entries(legacyAliases)) {
1033
+ if (typeof targetRaw !== "string") continue;
1034
+ const target = targetRaw.trim();
1035
+ if (!target) continue;
1036
+ const entry = models[target] && typeof models[target] === "object" ? models[target] : {};
1037
+ if (!("alias" in entry)) {
1038
+ entry.alias = alias;
1039
+ models[target] = entry;
1040
+ }
1041
+ }
1042
+ const currentModel = agent.model && typeof agent.model === "object" ? agent.model : null;
1043
+ if (currentModel) {
1044
+ if (!currentModel.primary && legacyModel) currentModel.primary = legacyModel;
1045
+ if (legacyModelFallbacks.length > 0 && (!Array.isArray(currentModel.fallbacks) || currentModel.fallbacks.length === 0)) currentModel.fallbacks = legacyModelFallbacks;
1046
+ agent.model = currentModel;
1047
+ } else if (legacyModel || legacyModelFallbacks.length > 0) agent.model = {
1048
+ primary: legacyModel,
1049
+ fallbacks: legacyModelFallbacks.length ? legacyModelFallbacks : []
1050
+ };
1051
+ const currentImageModel = agent.imageModel && typeof agent.imageModel === "object" ? agent.imageModel : null;
1052
+ if (currentImageModel) {
1053
+ if (!currentImageModel.primary && legacyImageModel) currentImageModel.primary = legacyImageModel;
1054
+ if (legacyImageModelFallbacks.length > 0 && (!Array.isArray(currentImageModel.fallbacks) || currentImageModel.fallbacks.length === 0)) currentImageModel.fallbacks = legacyImageModelFallbacks;
1055
+ agent.imageModel = currentImageModel;
1056
+ } else if (legacyImageModel || legacyImageModelFallbacks.length > 0) agent.imageModel = {
1057
+ primary: legacyImageModel,
1058
+ fallbacks: legacyImageModelFallbacks.length ? legacyImageModelFallbacks : []
1059
+ };
1060
+ agent.models = models;
1061
+ if (legacyModel !== void 0) changes.push(`Migrated ${label}.model string → ${label}.model.primary.`);
1062
+ if (legacyModelFallbacks.length > 0) changes.push(`Migrated ${label}.modelFallbacks → ${label}.model.fallbacks.`);
1063
+ if (legacyImageModel !== void 0) changes.push(`Migrated ${label}.imageModel string → ${label}.imageModel.primary.`);
1064
+ if (legacyImageModelFallbacks.length > 0) changes.push(`Migrated ${label}.imageModelFallbacks → ${label}.imageModel.fallbacks.`);
1065
+ if (legacyAllowed.length > 0) changes.push(`Migrated ${label}.allowedModels → ${label}.models.`);
1066
+ if (Object.keys(legacyAliases).length > 0) changes.push(`Migrated ${label}.modelAliases → ${label}.models.*.alias.`);
1067
+ delete agent.allowedModels;
1068
+ delete agent.modelAliases;
1069
+ delete agent.modelFallbacks;
1070
+ delete agent.imageModelFallbacks;
1071
+ }
1072
+ },
1073
+ {
1074
+ id: "routing.agents-v2",
1075
+ describe: "Move routing.agents/defaultAgentId to agents.list",
1076
+ apply: (raw, changes) => {
1077
+ const routing = getRecord(raw.routing);
1078
+ if (!routing) return;
1079
+ const routingAgents = getRecord(routing.agents);
1080
+ const agents = ensureRecord(raw, "agents");
1081
+ const list = getAgentsList(agents);
1082
+ if (routingAgents) {
1083
+ for (const [rawId, entryRaw] of Object.entries(routingAgents)) {
1084
+ const agentId = String(rawId ?? "").trim();
1085
+ const entry = getRecord(entryRaw);
1086
+ if (!agentId || !entry) continue;
1087
+ const target = ensureAgentEntry(list, agentId);
1088
+ const entryCopy = { ...entry };
1089
+ if ("mentionPatterns" in entryCopy) {
1090
+ const mentionPatterns = entryCopy.mentionPatterns;
1091
+ const groupChat = ensureRecord(target, "groupChat");
1092
+ if (groupChat.mentionPatterns === void 0) {
1093
+ groupChat.mentionPatterns = mentionPatterns;
1094
+ changes.push(`Moved routing.agents.${agentId}.mentionPatterns → agents.list (id "${agentId}").groupChat.mentionPatterns.`);
1095
+ } else changes.push(`Removed routing.agents.${agentId}.mentionPatterns (agents.list groupChat mentionPatterns already set).`);
1096
+ delete entryCopy.mentionPatterns;
1097
+ }
1098
+ const legacyGroupChat = getRecord(entryCopy.groupChat);
1099
+ if (legacyGroupChat) {
1100
+ mergeMissing(ensureRecord(target, "groupChat"), legacyGroupChat);
1101
+ delete entryCopy.groupChat;
1102
+ }
1103
+ const legacySandbox = getRecord(entryCopy.sandbox);
1104
+ if (legacySandbox) {
1105
+ const sandboxTools = getRecord(legacySandbox.tools);
1106
+ if (sandboxTools) {
1107
+ mergeMissing(ensureRecord(ensureRecord(ensureRecord(target, "tools"), "sandbox"), "tools"), sandboxTools);
1108
+ delete legacySandbox.tools;
1109
+ changes.push(`Moved routing.agents.${agentId}.sandbox.tools → agents.list (id "${agentId}").tools.sandbox.tools.`);
1110
+ }
1111
+ entryCopy.sandbox = legacySandbox;
1112
+ }
1113
+ mergeMissing(target, entryCopy);
1114
+ }
1115
+ delete routing.agents;
1116
+ changes.push("Moved routing.agents → agents.list.");
1117
+ }
1118
+ const defaultAgentId = typeof routing.defaultAgentId === "string" ? routing.defaultAgentId.trim() : "";
1119
+ if (defaultAgentId) {
1120
+ if (!list.some((entry) => isRecord$1(entry) && entry.default === true)) {
1121
+ const entry = ensureAgentEntry(list, defaultAgentId);
1122
+ entry.default = true;
1123
+ changes.push(`Moved routing.defaultAgentId → agents.list (id "${defaultAgentId}").default.`);
1124
+ } else changes.push("Removed routing.defaultAgentId (agents.list default already set).");
1125
+ delete routing.defaultAgentId;
1126
+ }
1127
+ if (list.length > 0) agents.list = list;
1128
+ if (Object.keys(routing).length === 0) delete raw.routing;
1129
+ }
1130
+ },
1131
+ {
1132
+ id: "routing.config-v2",
1133
+ describe: "Move routing bindings/groupChat/queue/agentToAgent/transcribeAudio",
1134
+ apply: (raw, changes) => {
1135
+ const routing = getRecord(raw.routing);
1136
+ if (!routing) return;
1137
+ if (routing.bindings !== void 0) {
1138
+ if (raw.bindings === void 0) {
1139
+ raw.bindings = routing.bindings;
1140
+ changes.push("Moved routing.bindings → bindings.");
1141
+ } else changes.push("Removed routing.bindings (bindings already set).");
1142
+ delete routing.bindings;
1143
+ }
1144
+ if (routing.agentToAgent !== void 0) {
1145
+ const tools = ensureRecord(raw, "tools");
1146
+ if (tools.agentToAgent === void 0) {
1147
+ tools.agentToAgent = routing.agentToAgent;
1148
+ changes.push("Moved routing.agentToAgent → tools.agentToAgent.");
1149
+ } else changes.push("Removed routing.agentToAgent (tools.agentToAgent already set).");
1150
+ delete routing.agentToAgent;
1151
+ }
1152
+ if (routing.queue !== void 0) {
1153
+ const messages = ensureRecord(raw, "messages");
1154
+ if (messages.queue === void 0) {
1155
+ messages.queue = routing.queue;
1156
+ changes.push("Moved routing.queue → messages.queue.");
1157
+ } else changes.push("Removed routing.queue (messages.queue already set).");
1158
+ delete routing.queue;
1159
+ }
1160
+ const groupChat = getRecord(routing.groupChat);
1161
+ if (groupChat) {
1162
+ const historyLimit = groupChat.historyLimit;
1163
+ if (historyLimit !== void 0) {
1164
+ const messagesGroup = ensureRecord(ensureRecord(raw, "messages"), "groupChat");
1165
+ if (messagesGroup.historyLimit === void 0) {
1166
+ messagesGroup.historyLimit = historyLimit;
1167
+ changes.push("Moved routing.groupChat.historyLimit → messages.groupChat.historyLimit.");
1168
+ } else changes.push("Removed routing.groupChat.historyLimit (messages.groupChat.historyLimit already set).");
1169
+ delete groupChat.historyLimit;
1170
+ }
1171
+ const mentionPatterns = groupChat.mentionPatterns;
1172
+ if (mentionPatterns !== void 0) {
1173
+ const messagesGroup = ensureRecord(ensureRecord(raw, "messages"), "groupChat");
1174
+ if (messagesGroup.mentionPatterns === void 0) {
1175
+ messagesGroup.mentionPatterns = mentionPatterns;
1176
+ changes.push("Moved routing.groupChat.mentionPatterns → messages.groupChat.mentionPatterns.");
1177
+ } else changes.push("Removed routing.groupChat.mentionPatterns (messages.groupChat.mentionPatterns already set).");
1178
+ delete groupChat.mentionPatterns;
1179
+ }
1180
+ if (Object.keys(groupChat).length === 0) delete routing.groupChat;
1181
+ else routing.groupChat = groupChat;
1182
+ }
1183
+ if (routing.transcribeAudio !== void 0) {
1184
+ const mapped = mapLegacyAudioTranscription(routing.transcribeAudio);
1185
+ if (mapped) {
1186
+ const mediaAudio = ensureRecord(ensureRecord(ensureRecord(raw, "tools"), "media"), "audio");
1187
+ if ((Array.isArray(mediaAudio.models) ? mediaAudio.models : []).length === 0) {
1188
+ mediaAudio.enabled = true;
1189
+ mediaAudio.models = [mapped];
1190
+ changes.push("Moved routing.transcribeAudio → tools.media.audio.models.");
1191
+ } else changes.push("Removed routing.transcribeAudio (tools.media.audio.models already set).");
1192
+ } else changes.push("Removed routing.transcribeAudio (unsupported transcription CLI).");
1193
+ delete routing.transcribeAudio;
1194
+ }
1195
+ const audio = getRecord(raw.audio);
1196
+ if (audio?.transcription !== void 0) {
1197
+ const mapped = mapLegacyAudioTranscription(audio.transcription);
1198
+ if (mapped) {
1199
+ const mediaAudio = ensureRecord(ensureRecord(ensureRecord(raw, "tools"), "media"), "audio");
1200
+ if ((Array.isArray(mediaAudio.models) ? mediaAudio.models : []).length === 0) {
1201
+ mediaAudio.enabled = true;
1202
+ mediaAudio.models = [mapped];
1203
+ changes.push("Moved audio.transcription → tools.media.audio.models.");
1204
+ } else changes.push("Removed audio.transcription (tools.media.audio.models already set).");
1205
+ delete audio.transcription;
1206
+ if (Object.keys(audio).length === 0) delete raw.audio;
1207
+ else raw.audio = audio;
1208
+ } else {
1209
+ delete audio.transcription;
1210
+ changes.push("Removed audio.transcription (unsupported transcription CLI).");
1211
+ if (Object.keys(audio).length === 0) delete raw.audio;
1212
+ else raw.audio = audio;
1213
+ }
1214
+ }
1215
+ if (Object.keys(routing).length === 0) delete raw.routing;
1216
+ }
1217
+ }
1218
+ ];
1219
+
1220
+ //#endregion
1221
+ //#region src/config/legacy.migrations.part-3.ts
1222
+ const LEGACY_CONFIG_MIGRATIONS_PART_3 = [
1223
+ {
1224
+ id: "auth.anthropic-claude-cli-mode-oauth",
1225
+ describe: "Switch anthropic:claude-cli auth profile mode to oauth",
1226
+ apply: (raw, changes) => {
1227
+ const profiles = getRecord(getRecord(raw.auth)?.profiles);
1228
+ if (!profiles) return;
1229
+ const claudeCli = getRecord(profiles["anthropic:claude-cli"]);
1230
+ if (!claudeCli) return;
1231
+ if (claudeCli.mode !== "token") return;
1232
+ claudeCli.mode = "oauth";
1233
+ changes.push("Updated auth.profiles[\"anthropic:claude-cli\"].mode → \"oauth\".");
1234
+ }
1235
+ },
1236
+ {
1237
+ id: "tools.bash->tools.exec",
1238
+ describe: "Move tools.bash to tools.exec",
1239
+ apply: (raw, changes) => {
1240
+ const tools = ensureRecord(raw, "tools");
1241
+ const bash = getRecord(tools.bash);
1242
+ if (!bash) return;
1243
+ if (tools.exec === void 0) {
1244
+ tools.exec = bash;
1245
+ changes.push("Moved tools.bash → tools.exec.");
1246
+ } else changes.push("Removed tools.bash (tools.exec already set).");
1247
+ delete tools.bash;
1248
+ }
1249
+ },
1250
+ {
1251
+ id: "messages.tts.enabled->auto",
1252
+ describe: "Move messages.tts.enabled to messages.tts.auto",
1253
+ apply: (raw, changes) => {
1254
+ const tts = getRecord(getRecord(raw.messages)?.tts);
1255
+ if (!tts) return;
1256
+ if (tts.auto !== void 0) {
1257
+ if ("enabled" in tts) {
1258
+ delete tts.enabled;
1259
+ changes.push("Removed messages.tts.enabled (messages.tts.auto already set).");
1260
+ }
1261
+ return;
1262
+ }
1263
+ if (typeof tts.enabled !== "boolean") return;
1264
+ tts.auto = tts.enabled ? "always" : "off";
1265
+ delete tts.enabled;
1266
+ changes.push(`Moved messages.tts.enabled → messages.tts.auto (${String(tts.auto)}).`);
1267
+ }
1268
+ },
1269
+ {
1270
+ id: "agent.defaults-v2",
1271
+ describe: "Move agent config to agents.defaults and tools",
1272
+ apply: (raw, changes) => {
1273
+ const agent = getRecord(raw.agent);
1274
+ if (!agent) return;
1275
+ const agents = ensureRecord(raw, "agents");
1276
+ const defaults = getRecord(agents.defaults) ?? {};
1277
+ const tools = ensureRecord(raw, "tools");
1278
+ const agentTools = getRecord(agent.tools);
1279
+ if (agentTools) {
1280
+ if (tools.allow === void 0 && agentTools.allow !== void 0) {
1281
+ tools.allow = agentTools.allow;
1282
+ changes.push("Moved agent.tools.allow → tools.allow.");
1283
+ }
1284
+ if (tools.deny === void 0 && agentTools.deny !== void 0) {
1285
+ tools.deny = agentTools.deny;
1286
+ changes.push("Moved agent.tools.deny → tools.deny.");
1287
+ }
1288
+ }
1289
+ const elevated = getRecord(agent.elevated);
1290
+ if (elevated) if (tools.elevated === void 0) {
1291
+ tools.elevated = elevated;
1292
+ changes.push("Moved agent.elevated → tools.elevated.");
1293
+ } else changes.push("Removed agent.elevated (tools.elevated already set).");
1294
+ const bash = getRecord(agent.bash);
1295
+ if (bash) if (tools.exec === void 0) {
1296
+ tools.exec = bash;
1297
+ changes.push("Moved agent.bash → tools.exec.");
1298
+ } else changes.push("Removed agent.bash (tools.exec already set).");
1299
+ const sandbox = getRecord(agent.sandbox);
1300
+ if (sandbox) {
1301
+ const sandboxTools = getRecord(sandbox.tools);
1302
+ if (sandboxTools) {
1303
+ mergeMissing(ensureRecord(ensureRecord(tools, "sandbox"), "tools"), sandboxTools);
1304
+ delete sandbox.tools;
1305
+ changes.push("Moved agent.sandbox.tools → tools.sandbox.tools.");
1306
+ }
1307
+ }
1308
+ const subagents = getRecord(agent.subagents);
1309
+ if (subagents) {
1310
+ const subagentTools = getRecord(subagents.tools);
1311
+ if (subagentTools) {
1312
+ mergeMissing(ensureRecord(ensureRecord(tools, "subagents"), "tools"), subagentTools);
1313
+ delete subagents.tools;
1314
+ changes.push("Moved agent.subagents.tools → tools.subagents.tools.");
1315
+ }
1316
+ }
1317
+ const agentCopy = structuredClone(agent);
1318
+ delete agentCopy.tools;
1319
+ delete agentCopy.elevated;
1320
+ delete agentCopy.bash;
1321
+ if (isRecord$1(agentCopy.sandbox)) delete agentCopy.sandbox.tools;
1322
+ if (isRecord$1(agentCopy.subagents)) delete agentCopy.subagents.tools;
1323
+ mergeMissing(defaults, agentCopy);
1324
+ agents.defaults = defaults;
1325
+ raw.agents = agents;
1326
+ delete raw.agent;
1327
+ changes.push("Moved agent → agents.defaults.");
1328
+ }
1329
+ },
1330
+ {
1331
+ id: "identity->agents.list",
1332
+ describe: "Move identity to agents.list[].identity",
1333
+ apply: (raw, changes) => {
1334
+ const identity = getRecord(raw.identity);
1335
+ if (!identity) return;
1336
+ const agents = ensureRecord(raw, "agents");
1337
+ const list = getAgentsList(agents);
1338
+ const defaultId = resolveDefaultAgentIdFromRaw(raw);
1339
+ const entry = ensureAgentEntry(list, defaultId);
1340
+ if (entry.identity === void 0) {
1341
+ entry.identity = identity;
1342
+ changes.push(`Moved identity → agents.list (id "${defaultId}").identity.`);
1343
+ } else changes.push("Removed identity (agents.list identity already set).");
1344
+ agents.list = list;
1345
+ raw.agents = agents;
1346
+ delete raw.identity;
1347
+ }
1348
+ }
1349
+ ];
1350
+
1351
+ //#endregion
1352
+ //#region src/config/legacy.migrations.ts
1353
+ const LEGACY_CONFIG_MIGRATIONS = [
1354
+ ...LEGACY_CONFIG_MIGRATIONS_PART_1,
1355
+ ...LEGACY_CONFIG_MIGRATIONS_PART_2,
1356
+ ...LEGACY_CONFIG_MIGRATIONS_PART_3
1357
+ ];
1358
+
1359
+ //#endregion
1360
+ //#region src/config/legacy.rules.ts
1361
+ const LEGACY_CONFIG_RULES = [
1362
+ {
1363
+ path: ["whatsapp"],
1364
+ message: "whatsapp config moved to channels.whatsapp (auto-migrated on load)."
1365
+ },
1366
+ {
1367
+ path: ["telegram"],
1368
+ message: "telegram config moved to channels.telegram (auto-migrated on load)."
1369
+ },
1370
+ {
1371
+ path: ["discord"],
1372
+ message: "discord config moved to channels.discord (auto-migrated on load)."
1373
+ },
1374
+ {
1375
+ path: ["slack"],
1376
+ message: "slack config moved to channels.slack (auto-migrated on load)."
1377
+ },
1378
+ {
1379
+ path: ["signal"],
1380
+ message: "signal config moved to channels.signal (auto-migrated on load)."
1381
+ },
1382
+ {
1383
+ path: ["imessage"],
1384
+ message: "imessage config moved to channels.imessage (auto-migrated on load)."
1385
+ },
1386
+ {
1387
+ path: ["msteams"],
1388
+ message: "msteams config moved to channels.msteams (auto-migrated on load)."
1389
+ },
1390
+ {
1391
+ path: ["routing", "allowFrom"],
1392
+ message: "routing.allowFrom was removed; use channels.whatsapp.allowFrom instead (auto-migrated on load)."
1393
+ },
1394
+ {
1395
+ path: ["routing", "bindings"],
1396
+ message: "routing.bindings was moved; use top-level bindings instead (auto-migrated on load)."
1397
+ },
1398
+ {
1399
+ path: ["routing", "agents"],
1400
+ message: "routing.agents was moved; use agents.list instead (auto-migrated on load)."
1401
+ },
1402
+ {
1403
+ path: ["routing", "defaultAgentId"],
1404
+ message: "routing.defaultAgentId was moved; use agents.list[].default instead (auto-migrated on load)."
1405
+ },
1406
+ {
1407
+ path: ["routing", "agentToAgent"],
1408
+ message: "routing.agentToAgent was moved; use tools.agentToAgent instead (auto-migrated on load)."
1409
+ },
1410
+ {
1411
+ path: [
1412
+ "routing",
1413
+ "groupChat",
1414
+ "requireMention"
1415
+ ],
1416
+ message: "routing.groupChat.requireMention was removed; use channels.whatsapp/telegram/imessage groups defaults (e.g. channels.whatsapp.groups.\"*\".requireMention) instead (auto-migrated on load)."
1417
+ },
1418
+ {
1419
+ path: [
1420
+ "routing",
1421
+ "groupChat",
1422
+ "mentionPatterns"
1423
+ ],
1424
+ message: "routing.groupChat.mentionPatterns was moved; use agents.list[].groupChat.mentionPatterns or messages.groupChat.mentionPatterns instead (auto-migrated on load)."
1425
+ },
1426
+ {
1427
+ path: ["routing", "queue"],
1428
+ message: "routing.queue was moved; use messages.queue instead (auto-migrated on load)."
1429
+ },
1430
+ {
1431
+ path: ["routing", "transcribeAudio"],
1432
+ message: "routing.transcribeAudio was moved; use tools.media.audio.models instead (auto-migrated on load)."
1433
+ },
1434
+ {
1435
+ path: ["telegram", "requireMention"],
1436
+ message: "telegram.requireMention was removed; use channels.telegram.groups.\"*\".requireMention instead (auto-migrated on load)."
1437
+ },
1438
+ {
1439
+ path: ["identity"],
1440
+ message: "identity was moved; use agents.list[].identity instead (auto-migrated on load)."
1441
+ },
1442
+ {
1443
+ path: ["agent"],
1444
+ message: "agent.* was moved; use agents.defaults (and tools.* for tool/elevated/exec settings) instead (auto-migrated on load)."
1445
+ },
1446
+ {
1447
+ path: ["tools", "bash"],
1448
+ message: "tools.bash was removed; use tools.exec instead (auto-migrated on load)."
1449
+ },
1450
+ {
1451
+ path: ["agent", "model"],
1452
+ message: "agent.model string was replaced by agents.defaults.model.primary/fallbacks and agents.defaults.models (auto-migrated on load).",
1453
+ match: (value) => typeof value === "string"
1454
+ },
1455
+ {
1456
+ path: ["agent", "imageModel"],
1457
+ message: "agent.imageModel string was replaced by agents.defaults.imageModel.primary/fallbacks (auto-migrated on load).",
1458
+ match: (value) => typeof value === "string"
1459
+ },
1460
+ {
1461
+ path: ["agent", "allowedModels"],
1462
+ message: "agent.allowedModels was replaced by agents.defaults.models (auto-migrated on load)."
1463
+ },
1464
+ {
1465
+ path: ["agent", "modelAliases"],
1466
+ message: "agent.modelAliases was replaced by agents.defaults.models.*.alias (auto-migrated on load)."
1467
+ },
1468
+ {
1469
+ path: ["agent", "modelFallbacks"],
1470
+ message: "agent.modelFallbacks was replaced by agents.defaults.model.fallbacks (auto-migrated on load)."
1471
+ },
1472
+ {
1473
+ path: ["agent", "imageModelFallbacks"],
1474
+ message: "agent.imageModelFallbacks was replaced by agents.defaults.imageModel.fallbacks (auto-migrated on load)."
1475
+ },
1476
+ {
1477
+ path: [
1478
+ "messages",
1479
+ "tts",
1480
+ "enabled"
1481
+ ],
1482
+ message: "messages.tts.enabled was replaced by messages.tts.auto (auto-migrated on load)."
1483
+ },
1484
+ {
1485
+ path: ["gateway", "token"],
1486
+ message: "gateway.token is ignored; use gateway.auth.token instead (auto-migrated on load)."
1487
+ }
1488
+ ];
1489
+
1490
+ //#endregion
1491
+ //#region src/config/legacy.ts
1492
+ function findLegacyConfigIssues(raw) {
1493
+ if (!raw || typeof raw !== "object") return [];
1494
+ const root = raw;
1495
+ const issues = [];
1496
+ for (const rule of LEGACY_CONFIG_RULES) {
1497
+ let cursor = root;
1498
+ for (const key of rule.path) {
1499
+ if (!cursor || typeof cursor !== "object") {
1500
+ cursor = void 0;
1501
+ break;
1502
+ }
1503
+ cursor = cursor[key];
1504
+ }
1505
+ if (cursor !== void 0 && (!rule.match || rule.match(cursor, root))) issues.push({
1506
+ path: rule.path.join("."),
1507
+ message: rule.message
1508
+ });
1509
+ }
1510
+ return issues;
1511
+ }
1512
+ function applyLegacyMigrations(raw) {
1513
+ if (!raw || typeof raw !== "object") return {
1514
+ next: null,
1515
+ changes: []
1516
+ };
1517
+ const next = structuredClone(raw);
1518
+ const changes = [];
1519
+ for (const migration of LEGACY_CONFIG_MIGRATIONS) migration.apply(next, changes);
1520
+ if (changes.length === 0) return {
1521
+ next: null,
1522
+ changes: []
1523
+ };
1524
+ return {
1525
+ next,
1526
+ changes
1527
+ };
1528
+ }
1529
+
1530
+ //#endregion
1531
+ //#region src/config/normalize-paths.ts
1532
+ const PATH_VALUE_RE = /^~(?=$|[\\/])/;
1533
+ const PATH_KEY_RE = /(dir|path|paths|file|root|workspace)$/i;
1534
+ const PATH_LIST_KEYS = new Set(["paths", "pathPrepend"]);
1535
+ function isPlainObject$2(value) {
1536
+ return Boolean(value) && typeof value === "object" && !Array.isArray(value);
1537
+ }
1538
+ function normalizeStringValue(key, value) {
1539
+ if (!PATH_VALUE_RE.test(value.trim())) return value;
1540
+ if (!key) return value;
1541
+ if (PATH_KEY_RE.test(key) || PATH_LIST_KEYS.has(key)) return resolveUserPath(value);
1542
+ return value;
1543
+ }
1544
+ function normalizeAny(key, value) {
1545
+ if (typeof value === "string") return normalizeStringValue(key, value);
1546
+ if (Array.isArray(value)) {
1547
+ const normalizeChildren = Boolean(key && PATH_LIST_KEYS.has(key));
1548
+ return value.map((entry) => {
1549
+ if (typeof entry === "string") return normalizeChildren ? normalizeStringValue(key, entry) : entry;
1550
+ if (Array.isArray(entry)) return normalizeAny(void 0, entry);
1551
+ if (isPlainObject$2(entry)) return normalizeAny(void 0, entry);
1552
+ return entry;
1553
+ });
1554
+ }
1555
+ if (!isPlainObject$2(value)) return value;
1556
+ for (const [childKey, childValue] of Object.entries(value)) {
1557
+ const next = normalizeAny(childKey, childValue);
1558
+ if (next !== childValue) value[childKey] = next;
1559
+ }
1560
+ return value;
1561
+ }
1562
+ /**
1563
+ * Normalize "~" paths in path-ish config fields.
1564
+ *
1565
+ * Goal: accept `~/...` consistently across config file + env overrides, while
1566
+ * keeping the surface area small and predictable.
1567
+ */
1568
+ function normalizeConfigPaths(cfg) {
1569
+ if (!cfg || typeof cfg !== "object") return cfg;
1570
+ normalizeAny(void 0, cfg);
1571
+ return cfg;
1572
+ }
1573
+
1574
+ //#endregion
1575
+ //#region src/config/config-paths.ts
1576
+ const BLOCKED_KEYS = new Set([
1577
+ "__proto__",
1578
+ "prototype",
1579
+ "constructor"
1580
+ ]);
1581
+ function parseConfigPath(raw) {
1582
+ const trimmed = raw.trim();
1583
+ if (!trimmed) return {
1584
+ ok: false,
1585
+ error: "Invalid path. Use dot notation (e.g. foo.bar)."
1586
+ };
1587
+ const parts = trimmed.split(".").map((part) => part.trim());
1588
+ if (parts.some((part) => !part)) return {
1589
+ ok: false,
1590
+ error: "Invalid path. Use dot notation (e.g. foo.bar)."
1591
+ };
1592
+ if (parts.some((part) => BLOCKED_KEYS.has(part))) return {
1593
+ ok: false,
1594
+ error: "Invalid path segment."
1595
+ };
1596
+ return {
1597
+ ok: true,
1598
+ path: parts
1599
+ };
1600
+ }
1601
+ function setConfigValueAtPath(root, path, value) {
1602
+ let cursor = root;
1603
+ for (let idx = 0; idx < path.length - 1; idx += 1) {
1604
+ const key = path[idx];
1605
+ const next = cursor[key];
1606
+ if (!isPlainObject$1(next)) cursor[key] = {};
1607
+ cursor = cursor[key];
1608
+ }
1609
+ cursor[path[path.length - 1]] = value;
1610
+ }
1611
+ function unsetConfigValueAtPath(root, path) {
1612
+ const stack = [];
1613
+ let cursor = root;
1614
+ for (let idx = 0; idx < path.length - 1; idx += 1) {
1615
+ const key = path[idx];
1616
+ const next = cursor[key];
1617
+ if (!isPlainObject$1(next)) return false;
1618
+ stack.push({
1619
+ node: cursor,
1620
+ key
1621
+ });
1622
+ cursor = next;
1623
+ }
1624
+ const leafKey = path[path.length - 1];
1625
+ if (!(leafKey in cursor)) return false;
1626
+ delete cursor[leafKey];
1627
+ for (let idx = stack.length - 1; idx >= 0; idx -= 1) {
1628
+ const { node, key } = stack[idx];
1629
+ const child = node[key];
1630
+ if (isPlainObject$1(child) && Object.keys(child).length === 0) delete node[key];
1631
+ else break;
1632
+ }
1633
+ return true;
1634
+ }
1635
+ function getConfigValueAtPath(root, path) {
1636
+ let cursor = root;
1637
+ for (const key of path) {
1638
+ if (!isPlainObject$1(cursor)) return;
1639
+ cursor = cursor[key];
1640
+ }
1641
+ return cursor;
1642
+ }
1643
+ function isPlainObject$1(value) {
1644
+ return typeof value === "object" && value !== null && !Array.isArray(value) && Object.prototype.toString.call(value) === "[object Object]";
1645
+ }
1646
+
1647
+ //#endregion
1648
+ //#region src/config/runtime-overrides.ts
1649
+ let overrides = {};
1650
+ function mergeOverrides(base, override) {
1651
+ if (!isPlainObject(base) || !isPlainObject(override)) return override;
1652
+ const next = { ...base };
1653
+ for (const [key, value] of Object.entries(override)) {
1654
+ if (value === void 0) continue;
1655
+ next[key] = mergeOverrides(base[key], value);
1656
+ }
1657
+ return next;
1658
+ }
1659
+ function isPlainObject(value) {
1660
+ return typeof value === "object" && value !== null && !Array.isArray(value) && Object.prototype.toString.call(value) === "[object Object]";
1661
+ }
1662
+ function getConfigOverrides() {
1663
+ return overrides;
1664
+ }
1665
+ function resetConfigOverrides() {
1666
+ overrides = {};
1667
+ }
1668
+ function setConfigOverride(pathRaw, value) {
1669
+ const parsed = parseConfigPath(pathRaw);
1670
+ if (!parsed.ok || !parsed.path) return {
1671
+ ok: false,
1672
+ error: parsed.error ?? "Invalid path."
1673
+ };
1674
+ setConfigValueAtPath(overrides, parsed.path, value);
1675
+ return { ok: true };
1676
+ }
1677
+ function unsetConfigOverride(pathRaw) {
1678
+ const parsed = parseConfigPath(pathRaw);
1679
+ if (!parsed.ok || !parsed.path) return {
1680
+ ok: false,
1681
+ removed: false,
1682
+ error: parsed.error ?? "Invalid path."
1683
+ };
1684
+ return {
1685
+ ok: true,
1686
+ removed: unsetConfigValueAtPath(overrides, parsed.path)
1687
+ };
1688
+ }
1689
+ function applyConfigOverrides(cfg) {
1690
+ if (!overrides || Object.keys(overrides).length === 0) return cfg;
1691
+ return mergeOverrides(cfg, overrides);
1692
+ }
1693
+
1694
+ //#endregion
1695
+ //#region src/plugins/schema-validator.ts
1696
+ const ajv = new AjvPkg({
1697
+ allErrors: true,
1698
+ strict: false,
1699
+ removeAdditional: false
1700
+ });
1701
+ const schemaCache = /* @__PURE__ */ new Map();
1702
+ function formatAjvErrors(errors) {
1703
+ if (!errors || errors.length === 0) return ["invalid config"];
1704
+ return errors.map((error) => {
1705
+ return `${error.instancePath?.replace(/^\//, "").replace(/\//g, ".") || "<root>"}: ${error.message ?? "invalid"}`;
1706
+ });
1707
+ }
1708
+ function validateJsonSchemaValue(params) {
1709
+ let cached = schemaCache.get(params.cacheKey);
1710
+ if (!cached || cached.schema !== params.schema) {
1711
+ cached = {
1712
+ validate: ajv.compile(params.schema),
1713
+ schema: params.schema
1714
+ };
1715
+ schemaCache.set(params.cacheKey, cached);
1716
+ }
1717
+ if (cached.validate(params.value)) return { ok: true };
1718
+ return {
1719
+ ok: false,
1720
+ errors: formatAjvErrors(cached.validate.errors)
1721
+ };
1722
+ }
1723
+
1724
+ //#endregion
1725
+ //#region src/cli/parse-duration.ts
1726
+ function parseDurationMs(raw, opts) {
1727
+ const trimmed = String(raw ?? "").trim().toLowerCase();
1728
+ if (!trimmed) throw new Error("invalid duration (empty)");
1729
+ const m = /^(\d+(?:\.\d+)?)(ms|s|m|h|d)?$/.exec(trimmed);
1730
+ if (!m) throw new Error(`invalid duration: ${raw}`);
1731
+ const value = Number(m[1]);
1732
+ if (!Number.isFinite(value) || value < 0) throw new Error(`invalid duration: ${raw}`);
1733
+ const unit = m[2] ?? opts?.defaultUnit ?? "ms";
1734
+ const multiplier = unit === "ms" ? 1 : unit === "s" ? 1e3 : unit === "m" ? 6e4 : unit === "h" ? 36e5 : 864e5;
1735
+ const ms = Math.round(value * multiplier);
1736
+ if (!Number.isFinite(ms)) throw new Error(`invalid duration: ${raw}`);
1737
+ return ms;
1738
+ }
1739
+
1740
+ //#endregion
1741
+ //#region src/infra/exec-safety.ts
1742
+ const SHELL_METACHARS = /[;&|`$<>]/;
1743
+ const CONTROL_CHARS = /[\r\n]/;
1744
+ const QUOTE_CHARS = /["']/;
1745
+ const BARE_NAME_PATTERN = /^[A-Za-z0-9._+-]+$/;
1746
+ function isLikelyPath(value) {
1747
+ if (value.startsWith(".") || value.startsWith("~")) return true;
1748
+ if (value.includes("/") || value.includes("\\")) return true;
1749
+ return /^[A-Za-z]:[\\/]/.test(value);
1750
+ }
1751
+ function isSafeExecutableValue(value) {
1752
+ if (!value) return false;
1753
+ const trimmed = value.trim();
1754
+ if (!trimmed) return false;
1755
+ if (trimmed.includes("\0")) return false;
1756
+ if (CONTROL_CHARS.test(trimmed)) return false;
1757
+ if (SHELL_METACHARS.test(trimmed)) return false;
1758
+ if (QUOTE_CHARS.test(trimmed)) return false;
1759
+ if (isLikelyPath(trimmed)) return true;
1760
+ if (trimmed.startsWith("-")) return false;
1761
+ return BARE_NAME_PATTERN.test(trimmed);
1762
+ }
1763
+
1764
+ //#endregion
1765
+ //#region src/config/zod-schema.core.ts
1766
+ const ModelApiSchema = z.union([
1767
+ z.literal("openai-completions"),
1768
+ z.literal("openai-responses"),
1769
+ z.literal("anthropic-messages"),
1770
+ z.literal("google-generative-ai"),
1771
+ z.literal("github-copilot"),
1772
+ z.literal("bedrock-converse-stream")
1773
+ ]);
1774
+ const ModelCompatSchema = z.object({
1775
+ supportsStore: z.boolean().optional(),
1776
+ supportsDeveloperRole: z.boolean().optional(),
1777
+ supportsReasoningEffort: z.boolean().optional(),
1778
+ maxTokensField: z.union([z.literal("max_completion_tokens"), z.literal("max_tokens")]).optional()
1779
+ }).strict().optional();
1780
+ const ModelDefinitionSchema = z.object({
1781
+ id: z.string().min(1),
1782
+ name: z.string().min(1),
1783
+ api: ModelApiSchema.optional(),
1784
+ reasoning: z.boolean().optional(),
1785
+ input: z.array(z.union([z.literal("text"), z.literal("image")])).optional(),
1786
+ cost: z.object({
1787
+ input: z.number().optional(),
1788
+ output: z.number().optional(),
1789
+ cacheRead: z.number().optional(),
1790
+ cacheWrite: z.number().optional()
1791
+ }).strict().optional(),
1792
+ contextWindow: z.number().positive().optional(),
1793
+ maxTokens: z.number().positive().optional(),
1794
+ headers: z.record(z.string(), z.string()).optional(),
1795
+ compat: ModelCompatSchema
1796
+ }).strict();
1797
+ const ModelProviderSchema = z.object({
1798
+ baseUrl: z.string().min(1),
1799
+ apiKey: z.string().optional(),
1800
+ auth: z.union([
1801
+ z.literal("api-key"),
1802
+ z.literal("aws-sdk"),
1803
+ z.literal("oauth"),
1804
+ z.literal("token")
1805
+ ]).optional(),
1806
+ api: ModelApiSchema.optional(),
1807
+ headers: z.record(z.string(), z.string()).optional(),
1808
+ authHeader: z.boolean().optional(),
1809
+ models: z.array(ModelDefinitionSchema)
1810
+ }).strict();
1811
+ const BedrockDiscoverySchema = z.object({
1812
+ enabled: z.boolean().optional(),
1813
+ region: z.string().optional(),
1814
+ providerFilter: z.array(z.string()).optional(),
1815
+ refreshInterval: z.number().int().nonnegative().optional(),
1816
+ defaultContextWindow: z.number().int().positive().optional(),
1817
+ defaultMaxTokens: z.number().int().positive().optional()
1818
+ }).strict().optional();
1819
+ const ModelsConfigSchema = z.object({
1820
+ mode: z.union([z.literal("merge"), z.literal("replace")]).optional(),
1821
+ providers: z.record(z.string(), ModelProviderSchema).optional(),
1822
+ bedrockDiscovery: BedrockDiscoverySchema
1823
+ }).strict().optional();
1824
+ const GroupChatSchema = z.object({
1825
+ mentionPatterns: z.array(z.string()).optional(),
1826
+ historyLimit: z.number().int().positive().optional()
1827
+ }).strict().optional();
1828
+ const DmConfigSchema = z.object({ historyLimit: z.number().int().min(0).optional() }).strict();
1829
+ const IdentitySchema = z.object({
1830
+ name: z.string().optional(),
1831
+ theme: z.string().optional(),
1832
+ emoji: z.string().optional(),
1833
+ avatar: z.string().optional()
1834
+ }).strict().optional();
1835
+ const QueueModeSchema = z.union([
1836
+ z.literal("steer"),
1837
+ z.literal("followup"),
1838
+ z.literal("collect"),
1839
+ z.literal("steer-backlog"),
1840
+ z.literal("steer+backlog"),
1841
+ z.literal("queue"),
1842
+ z.literal("interrupt")
1843
+ ]);
1844
+ const QueueDropSchema = z.union([
1845
+ z.literal("old"),
1846
+ z.literal("new"),
1847
+ z.literal("summarize")
1848
+ ]);
1849
+ const ReplyToModeSchema = z.union([
1850
+ z.literal("off"),
1851
+ z.literal("first"),
1852
+ z.literal("all")
1853
+ ]);
1854
+ const GroupPolicySchema = z.enum([
1855
+ "open",
1856
+ "disabled",
1857
+ "allowlist"
1858
+ ]);
1859
+ const DmPolicySchema = z.enum([
1860
+ "pairing",
1861
+ "allowlist",
1862
+ "open",
1863
+ "disabled"
1864
+ ]);
1865
+ const BlockStreamingCoalesceSchema = z.object({
1866
+ minChars: z.number().int().positive().optional(),
1867
+ maxChars: z.number().int().positive().optional(),
1868
+ idleMs: z.number().int().nonnegative().optional()
1869
+ }).strict();
1870
+ const BlockStreamingChunkSchema = z.object({
1871
+ minChars: z.number().int().positive().optional(),
1872
+ maxChars: z.number().int().positive().optional(),
1873
+ breakPreference: z.union([
1874
+ z.literal("paragraph"),
1875
+ z.literal("newline"),
1876
+ z.literal("sentence")
1877
+ ]).optional()
1878
+ }).strict();
1879
+ const MarkdownTableModeSchema = z.enum([
1880
+ "off",
1881
+ "bullets",
1882
+ "code"
1883
+ ]);
1884
+ const MarkdownConfigSchema = z.object({ tables: MarkdownTableModeSchema.optional() }).strict().optional();
1885
+ const TtsProviderSchema = z.enum([
1886
+ "elevenlabs",
1887
+ "openai",
1888
+ "edge"
1889
+ ]);
1890
+ const TtsModeSchema = z.enum(["final", "all"]);
1891
+ const TtsAutoSchema = z.enum([
1892
+ "off",
1893
+ "always",
1894
+ "inbound",
1895
+ "tagged"
1896
+ ]);
1897
+ const TtsConfigSchema = z.object({
1898
+ auto: TtsAutoSchema.optional(),
1899
+ enabled: z.boolean().optional(),
1900
+ mode: TtsModeSchema.optional(),
1901
+ provider: TtsProviderSchema.optional(),
1902
+ summaryModel: z.string().optional(),
1903
+ modelOverrides: z.object({
1904
+ enabled: z.boolean().optional(),
1905
+ allowText: z.boolean().optional(),
1906
+ allowProvider: z.boolean().optional(),
1907
+ allowVoice: z.boolean().optional(),
1908
+ allowModelId: z.boolean().optional(),
1909
+ allowVoiceSettings: z.boolean().optional(),
1910
+ allowNormalization: z.boolean().optional(),
1911
+ allowSeed: z.boolean().optional()
1912
+ }).strict().optional(),
1913
+ elevenlabs: z.object({
1914
+ apiKey: z.string().optional(),
1915
+ baseUrl: z.string().optional(),
1916
+ voiceId: z.string().optional(),
1917
+ modelId: z.string().optional(),
1918
+ seed: z.number().int().min(0).max(4294967295).optional(),
1919
+ applyTextNormalization: z.enum([
1920
+ "auto",
1921
+ "on",
1922
+ "off"
1923
+ ]).optional(),
1924
+ languageCode: z.string().optional(),
1925
+ voiceSettings: z.object({
1926
+ stability: z.number().min(0).max(1).optional(),
1927
+ similarityBoost: z.number().min(0).max(1).optional(),
1928
+ style: z.number().min(0).max(1).optional(),
1929
+ useSpeakerBoost: z.boolean().optional(),
1930
+ speed: z.number().min(.5).max(2).optional()
1931
+ }).strict().optional()
1932
+ }).strict().optional(),
1933
+ openai: z.object({
1934
+ apiKey: z.string().optional(),
1935
+ model: z.string().optional(),
1936
+ voice: z.string().optional()
1937
+ }).strict().optional(),
1938
+ edge: z.object({
1939
+ enabled: z.boolean().optional(),
1940
+ voice: z.string().optional(),
1941
+ lang: z.string().optional(),
1942
+ outputFormat: z.string().optional(),
1943
+ pitch: z.string().optional(),
1944
+ rate: z.string().optional(),
1945
+ volume: z.string().optional(),
1946
+ saveSubtitles: z.boolean().optional(),
1947
+ proxy: z.string().optional(),
1948
+ timeoutMs: z.number().int().min(1e3).max(12e4).optional()
1949
+ }).strict().optional(),
1950
+ prefsPath: z.string().optional(),
1951
+ maxTextLength: z.number().int().min(1).optional(),
1952
+ timeoutMs: z.number().int().min(1e3).max(12e4).optional()
1953
+ }).strict().optional();
1954
+ const HumanDelaySchema = z.object({
1955
+ mode: z.union([
1956
+ z.literal("off"),
1957
+ z.literal("natural"),
1958
+ z.literal("custom")
1959
+ ]).optional(),
1960
+ minMs: z.number().int().nonnegative().optional(),
1961
+ maxMs: z.number().int().nonnegative().optional()
1962
+ }).strict();
1963
+ const CliBackendSchema = z.object({
1964
+ command: z.string(),
1965
+ args: z.array(z.string()).optional(),
1966
+ output: z.union([
1967
+ z.literal("json"),
1968
+ z.literal("text"),
1969
+ z.literal("jsonl")
1970
+ ]).optional(),
1971
+ resumeOutput: z.union([
1972
+ z.literal("json"),
1973
+ z.literal("text"),
1974
+ z.literal("jsonl")
1975
+ ]).optional(),
1976
+ input: z.union([z.literal("arg"), z.literal("stdin")]).optional(),
1977
+ maxPromptArgChars: z.number().int().positive().optional(),
1978
+ env: z.record(z.string(), z.string()).optional(),
1979
+ clearEnv: z.array(z.string()).optional(),
1980
+ modelArg: z.string().optional(),
1981
+ modelAliases: z.record(z.string(), z.string()).optional(),
1982
+ sessionArg: z.string().optional(),
1983
+ sessionArgs: z.array(z.string()).optional(),
1984
+ resumeArgs: z.array(z.string()).optional(),
1985
+ sessionMode: z.union([
1986
+ z.literal("always"),
1987
+ z.literal("existing"),
1988
+ z.literal("none")
1989
+ ]).optional(),
1990
+ sessionIdFields: z.array(z.string()).optional(),
1991
+ systemPromptArg: z.string().optional(),
1992
+ systemPromptMode: z.union([z.literal("append"), z.literal("replace")]).optional(),
1993
+ systemPromptWhen: z.union([
1994
+ z.literal("first"),
1995
+ z.literal("always"),
1996
+ z.literal("never")
1997
+ ]).optional(),
1998
+ imageArg: z.string().optional(),
1999
+ imageMode: z.union([z.literal("repeat"), z.literal("list")]).optional(),
2000
+ serialize: z.boolean().optional()
2001
+ }).strict();
2002
+ const normalizeAllowFrom = (values) => (values ?? []).map((v) => String(v).trim()).filter(Boolean);
2003
+ const requireOpenAllowFrom = (params) => {
2004
+ if (params.policy !== "open") return;
2005
+ if (normalizeAllowFrom(params.allowFrom).includes("*")) return;
2006
+ params.ctx.addIssue({
2007
+ code: z.ZodIssueCode.custom,
2008
+ path: params.path,
2009
+ message: params.message
2010
+ });
2011
+ };
2012
+ const MSTeamsReplyStyleSchema = z.enum(["thread", "top-level"]);
2013
+ const RetryConfigSchema = z.object({
2014
+ attempts: z.number().int().min(1).optional(),
2015
+ minDelayMs: z.number().int().min(0).optional(),
2016
+ maxDelayMs: z.number().int().min(0).optional(),
2017
+ jitter: z.number().min(0).max(1).optional()
2018
+ }).strict().optional();
2019
+ const QueueModeBySurfaceSchema = z.object({
2020
+ whatsapp: QueueModeSchema.optional(),
2021
+ telegram: QueueModeSchema.optional(),
2022
+ discord: QueueModeSchema.optional(),
2023
+ slack: QueueModeSchema.optional(),
2024
+ mattermost: QueueModeSchema.optional(),
2025
+ signal: QueueModeSchema.optional(),
2026
+ imessage: QueueModeSchema.optional(),
2027
+ msteams: QueueModeSchema.optional(),
2028
+ webchat: QueueModeSchema.optional()
2029
+ }).strict().optional();
2030
+ const DebounceMsBySurfaceSchema = z.record(z.string(), z.number().int().nonnegative()).optional();
2031
+ const QueueSchema = z.object({
2032
+ mode: QueueModeSchema.optional(),
2033
+ byChannel: QueueModeBySurfaceSchema,
2034
+ debounceMs: z.number().int().nonnegative().optional(),
2035
+ debounceMsByChannel: DebounceMsBySurfaceSchema,
2036
+ cap: z.number().int().positive().optional(),
2037
+ drop: QueueDropSchema.optional()
2038
+ }).strict().optional();
2039
+ const InboundDebounceSchema = z.object({
2040
+ debounceMs: z.number().int().nonnegative().optional(),
2041
+ byChannel: DebounceMsBySurfaceSchema
2042
+ }).strict().optional();
2043
+ const TranscribeAudioSchema = z.object({
2044
+ command: z.array(z.string()).superRefine((value, ctx) => {
2045
+ const executable = value[0];
2046
+ if (!isSafeExecutableValue(executable)) ctx.addIssue({
2047
+ code: z.ZodIssueCode.custom,
2048
+ path: [0],
2049
+ message: "expected safe executable name or path"
2050
+ });
2051
+ }),
2052
+ timeoutSeconds: z.number().int().positive().optional()
2053
+ }).strict().optional();
2054
+ const HexColorSchema = z.string().regex(/^#?[0-9a-fA-F]{6}$/, "expected hex color (RRGGBB)");
2055
+ const ExecutableTokenSchema = z.string().refine(isSafeExecutableValue, "expected safe executable name or path");
2056
+ const MediaUnderstandingScopeSchema = z.object({
2057
+ default: z.union([z.literal("allow"), z.literal("deny")]).optional(),
2058
+ rules: z.array(z.object({
2059
+ action: z.union([z.literal("allow"), z.literal("deny")]),
2060
+ match: z.object({
2061
+ channel: z.string().optional(),
2062
+ chatType: z.union([
2063
+ z.literal("direct"),
2064
+ z.literal("group"),
2065
+ z.literal("channel")
2066
+ ]).optional(),
2067
+ keyPrefix: z.string().optional()
2068
+ }).strict().optional()
2069
+ }).strict()).optional()
2070
+ }).strict().optional();
2071
+ const MediaUnderstandingCapabilitiesSchema = z.array(z.union([
2072
+ z.literal("image"),
2073
+ z.literal("audio"),
2074
+ z.literal("video")
2075
+ ])).optional();
2076
+ const MediaUnderstandingAttachmentsSchema = z.object({
2077
+ mode: z.union([z.literal("first"), z.literal("all")]).optional(),
2078
+ maxAttachments: z.number().int().positive().optional(),
2079
+ prefer: z.union([
2080
+ z.literal("first"),
2081
+ z.literal("last"),
2082
+ z.literal("path"),
2083
+ z.literal("url")
2084
+ ]).optional()
2085
+ }).strict().optional();
2086
+ const DeepgramAudioSchema = z.object({
2087
+ detectLanguage: z.boolean().optional(),
2088
+ punctuate: z.boolean().optional(),
2089
+ smartFormat: z.boolean().optional()
2090
+ }).strict().optional();
2091
+ const ProviderOptionValueSchema = z.union([
2092
+ z.string(),
2093
+ z.number(),
2094
+ z.boolean()
2095
+ ]);
2096
+ const ProviderOptionsSchema = z.record(z.string(), z.record(z.string(), ProviderOptionValueSchema)).optional();
2097
+ const MediaUnderstandingModelSchema = z.object({
2098
+ provider: z.string().optional(),
2099
+ model: z.string().optional(),
2100
+ capabilities: MediaUnderstandingCapabilitiesSchema,
2101
+ type: z.union([z.literal("provider"), z.literal("cli")]).optional(),
2102
+ command: z.string().optional(),
2103
+ args: z.array(z.string()).optional(),
2104
+ prompt: z.string().optional(),
2105
+ maxChars: z.number().int().positive().optional(),
2106
+ maxBytes: z.number().int().positive().optional(),
2107
+ timeoutSeconds: z.number().int().positive().optional(),
2108
+ language: z.string().optional(),
2109
+ providerOptions: ProviderOptionsSchema,
2110
+ deepgram: DeepgramAudioSchema,
2111
+ baseUrl: z.string().optional(),
2112
+ headers: z.record(z.string(), z.string()).optional(),
2113
+ profile: z.string().optional(),
2114
+ preferredProfile: z.string().optional()
2115
+ }).strict().optional();
2116
+ const ToolsMediaUnderstandingSchema = z.object({
2117
+ enabled: z.boolean().optional(),
2118
+ scope: MediaUnderstandingScopeSchema,
2119
+ maxBytes: z.number().int().positive().optional(),
2120
+ maxChars: z.number().int().positive().optional(),
2121
+ prompt: z.string().optional(),
2122
+ timeoutSeconds: z.number().int().positive().optional(),
2123
+ language: z.string().optional(),
2124
+ providerOptions: ProviderOptionsSchema,
2125
+ deepgram: DeepgramAudioSchema,
2126
+ baseUrl: z.string().optional(),
2127
+ headers: z.record(z.string(), z.string()).optional(),
2128
+ attachments: MediaUnderstandingAttachmentsSchema,
2129
+ models: z.array(MediaUnderstandingModelSchema).optional()
2130
+ }).strict().optional();
2131
+ const ToolsMediaSchema = z.object({
2132
+ models: z.array(MediaUnderstandingModelSchema).optional(),
2133
+ concurrency: z.number().int().positive().optional(),
2134
+ image: ToolsMediaUnderstandingSchema.optional(),
2135
+ audio: ToolsMediaUnderstandingSchema.optional(),
2136
+ video: ToolsMediaUnderstandingSchema.optional()
2137
+ }).strict().optional();
2138
+ const LinkModelSchema = z.object({
2139
+ type: z.literal("cli").optional(),
2140
+ command: z.string().min(1),
2141
+ args: z.array(z.string()).optional(),
2142
+ timeoutSeconds: z.number().int().positive().optional()
2143
+ }).strict();
2144
+ const ToolsLinksSchema = z.object({
2145
+ enabled: z.boolean().optional(),
2146
+ scope: MediaUnderstandingScopeSchema,
2147
+ maxLinks: z.number().int().positive().optional(),
2148
+ timeoutSeconds: z.number().int().positive().optional(),
2149
+ models: z.array(LinkModelSchema).optional()
2150
+ }).strict().optional();
2151
+ const NativeCommandsSettingSchema = z.union([z.boolean(), z.literal("auto")]);
2152
+ const ProviderCommandsSchema = z.object({
2153
+ native: NativeCommandsSettingSchema.optional(),
2154
+ nativeSkills: NativeCommandsSettingSchema.optional()
2155
+ }).strict().optional();
2156
+
2157
+ //#endregion
2158
+ //#region src/config/zod-schema.agent-runtime.ts
2159
+ const HeartbeatSchema = z.object({
2160
+ every: z.string().optional(),
2161
+ activeHours: z.object({
2162
+ start: z.string().optional(),
2163
+ end: z.string().optional(),
2164
+ timezone: z.string().optional()
2165
+ }).strict().optional(),
2166
+ model: z.string().optional(),
2167
+ session: z.string().optional(),
2168
+ includeReasoning: z.boolean().optional(),
2169
+ target: z.string().optional(),
2170
+ to: z.string().optional(),
2171
+ prompt: z.string().optional(),
2172
+ ackMaxChars: z.number().int().nonnegative().optional()
2173
+ }).strict().superRefine((val, ctx) => {
2174
+ if (!val.every) return;
2175
+ try {
2176
+ parseDurationMs(val.every, { defaultUnit: "m" });
2177
+ } catch {
2178
+ ctx.addIssue({
2179
+ code: z.ZodIssueCode.custom,
2180
+ path: ["every"],
2181
+ message: "invalid duration (use ms, s, m, h)"
2182
+ });
2183
+ }
2184
+ const active = val.activeHours;
2185
+ if (!active) return;
2186
+ const timePattern = /^([01]\d|2[0-3]|24):([0-5]\d)$/;
2187
+ const validateTime = (raw, opts, path) => {
2188
+ if (!raw) return;
2189
+ if (!timePattern.test(raw)) {
2190
+ ctx.addIssue({
2191
+ code: z.ZodIssueCode.custom,
2192
+ path: ["activeHours", path],
2193
+ message: "invalid time (use \"HH:MM\" 24h format)"
2194
+ });
2195
+ return;
2196
+ }
2197
+ const [hourStr, minuteStr] = raw.split(":");
2198
+ const hour = Number(hourStr);
2199
+ if (hour === 24 && Number(minuteStr) !== 0) {
2200
+ ctx.addIssue({
2201
+ code: z.ZodIssueCode.custom,
2202
+ path: ["activeHours", path],
2203
+ message: "invalid time (24:00 is the only allowed 24:xx value)"
2204
+ });
2205
+ return;
2206
+ }
2207
+ if (hour === 24 && !opts.allow24) ctx.addIssue({
2208
+ code: z.ZodIssueCode.custom,
2209
+ path: ["activeHours", path],
2210
+ message: "invalid time (start cannot be 24:00)"
2211
+ });
2212
+ };
2213
+ validateTime(active.start, { allow24: false }, "start");
2214
+ validateTime(active.end, { allow24: true }, "end");
2215
+ }).optional();
2216
+ const SandboxDockerSchema = z.object({
2217
+ image: z.string().optional(),
2218
+ containerPrefix: z.string().optional(),
2219
+ workdir: z.string().optional(),
2220
+ readOnlyRoot: z.boolean().optional(),
2221
+ tmpfs: z.array(z.string()).optional(),
2222
+ network: z.string().optional(),
2223
+ user: z.string().optional(),
2224
+ capDrop: z.array(z.string()).optional(),
2225
+ env: z.record(z.string(), z.string()).optional(),
2226
+ setupCommand: z.string().optional(),
2227
+ pidsLimit: z.number().int().positive().optional(),
2228
+ memory: z.union([z.string(), z.number()]).optional(),
2229
+ memorySwap: z.union([z.string(), z.number()]).optional(),
2230
+ cpus: z.number().positive().optional(),
2231
+ ulimits: z.record(z.string(), z.union([
2232
+ z.string(),
2233
+ z.number(),
2234
+ z.object({
2235
+ soft: z.number().int().nonnegative().optional(),
2236
+ hard: z.number().int().nonnegative().optional()
2237
+ }).strict()
2238
+ ])).optional(),
2239
+ seccompProfile: z.string().optional(),
2240
+ apparmorProfile: z.string().optional(),
2241
+ dns: z.array(z.string()).optional(),
2242
+ extraHosts: z.array(z.string()).optional(),
2243
+ binds: z.array(z.string()).optional()
2244
+ }).strict().optional();
2245
+ const SandboxBrowserSchema = z.object({
2246
+ enabled: z.boolean().optional(),
2247
+ image: z.string().optional(),
2248
+ containerPrefix: z.string().optional(),
2249
+ cdpPort: z.number().int().positive().optional(),
2250
+ vncPort: z.number().int().positive().optional(),
2251
+ noVncPort: z.number().int().positive().optional(),
2252
+ headless: z.boolean().optional(),
2253
+ enableNoVnc: z.boolean().optional(),
2254
+ allowHostControl: z.boolean().optional(),
2255
+ autoStart: z.boolean().optional(),
2256
+ autoStartTimeoutMs: z.number().int().positive().optional()
2257
+ }).strict().optional();
2258
+ const SandboxPruneSchema = z.object({
2259
+ idleHours: z.number().int().nonnegative().optional(),
2260
+ maxAgeDays: z.number().int().nonnegative().optional()
2261
+ }).strict().optional();
2262
+ const ToolPolicyBaseSchema = z.object({
2263
+ allow: z.array(z.string()).optional(),
2264
+ alsoAllow: z.array(z.string()).optional(),
2265
+ deny: z.array(z.string()).optional()
2266
+ }).strict();
2267
+ const ToolPolicySchema = ToolPolicyBaseSchema.superRefine((value, ctx) => {
2268
+ if (value.allow && value.allow.length > 0 && value.alsoAllow && value.alsoAllow.length > 0) ctx.addIssue({
2269
+ code: z.ZodIssueCode.custom,
2270
+ message: "tools policy cannot set both allow and alsoAllow in the same scope (merge alsoAllow into allow, or remove allow and use profile + alsoAllow)"
2271
+ });
2272
+ }).optional();
2273
+ const ToolsWebSearchSchema = z.object({
2274
+ enabled: z.boolean().optional(),
2275
+ provider: z.union([z.literal("brave"), z.literal("perplexity")]).optional(),
2276
+ apiKey: z.string().optional(),
2277
+ maxResults: z.number().int().positive().optional(),
2278
+ timeoutSeconds: z.number().int().positive().optional(),
2279
+ cacheTtlMinutes: z.number().nonnegative().optional(),
2280
+ perplexity: z.object({
2281
+ apiKey: z.string().optional(),
2282
+ baseUrl: z.string().optional(),
2283
+ model: z.string().optional()
2284
+ }).strict().optional()
2285
+ }).strict().optional();
2286
+ const ToolsWebFetchSchema = z.object({
2287
+ enabled: z.boolean().optional(),
2288
+ maxChars: z.number().int().positive().optional(),
2289
+ maxCharsCap: z.number().int().positive().optional(),
2290
+ timeoutSeconds: z.number().int().positive().optional(),
2291
+ cacheTtlMinutes: z.number().nonnegative().optional(),
2292
+ maxRedirects: z.number().int().nonnegative().optional(),
2293
+ userAgent: z.string().optional()
2294
+ }).strict().optional();
2295
+ const ToolsWebSchema = z.object({
2296
+ search: ToolsWebSearchSchema,
2297
+ fetch: ToolsWebFetchSchema
2298
+ }).strict().optional();
2299
+ const ToolProfileSchema = z.union([
2300
+ z.literal("minimal"),
2301
+ z.literal("coding"),
2302
+ z.literal("messaging"),
2303
+ z.literal("full")
2304
+ ]).optional();
2305
+ const ToolPolicyWithProfileSchema = z.object({
2306
+ allow: z.array(z.string()).optional(),
2307
+ alsoAllow: z.array(z.string()).optional(),
2308
+ deny: z.array(z.string()).optional(),
2309
+ profile: ToolProfileSchema
2310
+ }).strict().superRefine((value, ctx) => {
2311
+ if (value.allow && value.allow.length > 0 && value.alsoAllow && value.alsoAllow.length > 0) ctx.addIssue({
2312
+ code: z.ZodIssueCode.custom,
2313
+ message: "tools.byProvider policy cannot set both allow and alsoAllow in the same scope (merge alsoAllow into allow, or remove allow and use profile + alsoAllow)"
2314
+ });
2315
+ });
2316
+ const ElevatedAllowFromSchema = z.record(z.string(), z.array(z.union([z.string(), z.number()]))).optional();
2317
+ const AgentSandboxSchema = z.object({
2318
+ mode: z.union([
2319
+ z.literal("off"),
2320
+ z.literal("non-main"),
2321
+ z.literal("all")
2322
+ ]).optional(),
2323
+ workspaceAccess: z.union([
2324
+ z.literal("none"),
2325
+ z.literal("ro"),
2326
+ z.literal("rw")
2327
+ ]).optional(),
2328
+ sessionToolsVisibility: z.union([z.literal("spawned"), z.literal("all")]).optional(),
2329
+ scope: z.union([
2330
+ z.literal("session"),
2331
+ z.literal("agent"),
2332
+ z.literal("shared")
2333
+ ]).optional(),
2334
+ perSession: z.boolean().optional(),
2335
+ workspaceRoot: z.string().optional(),
2336
+ docker: SandboxDockerSchema,
2337
+ browser: SandboxBrowserSchema,
2338
+ prune: SandboxPruneSchema
2339
+ }).strict().optional();
2340
+ const AgentToolsSchema = z.object({
2341
+ profile: ToolProfileSchema,
2342
+ allow: z.array(z.string()).optional(),
2343
+ alsoAllow: z.array(z.string()).optional(),
2344
+ deny: z.array(z.string()).optional(),
2345
+ byProvider: z.record(z.string(), ToolPolicyWithProfileSchema).optional(),
2346
+ elevated: z.object({
2347
+ enabled: z.boolean().optional(),
2348
+ allowFrom: ElevatedAllowFromSchema
2349
+ }).strict().optional(),
2350
+ exec: z.object({
2351
+ host: z.enum([
2352
+ "sandbox",
2353
+ "gateway",
2354
+ "node"
2355
+ ]).optional(),
2356
+ security: z.enum([
2357
+ "deny",
2358
+ "allowlist",
2359
+ "full"
2360
+ ]).optional(),
2361
+ ask: z.enum([
2362
+ "off",
2363
+ "on-miss",
2364
+ "always"
2365
+ ]).optional(),
2366
+ node: z.string().optional(),
2367
+ pathPrepend: z.array(z.string()).optional(),
2368
+ safeBins: z.array(z.string()).optional(),
2369
+ backgroundMs: z.number().int().positive().optional(),
2370
+ timeoutSec: z.number().int().positive().optional(),
2371
+ approvalRunningNoticeMs: z.number().int().nonnegative().optional(),
2372
+ cleanupMs: z.number().int().positive().optional(),
2373
+ notifyOnExit: z.boolean().optional(),
2374
+ applyPatch: z.object({
2375
+ enabled: z.boolean().optional(),
2376
+ allowModels: z.array(z.string()).optional()
2377
+ }).strict().optional()
2378
+ }).strict().optional(),
2379
+ sandbox: z.object({ tools: ToolPolicySchema }).strict().optional()
2380
+ }).strict().superRefine((value, ctx) => {
2381
+ if (value.allow && value.allow.length > 0 && value.alsoAllow && value.alsoAllow.length > 0) ctx.addIssue({
2382
+ code: z.ZodIssueCode.custom,
2383
+ message: "agent tools cannot set both allow and alsoAllow in the same scope (merge alsoAllow into allow, or remove allow and use profile + alsoAllow)"
2384
+ });
2385
+ }).optional();
2386
+ const MemorySearchSchema = z.object({
2387
+ enabled: z.boolean().optional(),
2388
+ sources: z.array(z.union([z.literal("memory"), z.literal("sessions")])).optional(),
2389
+ extraPaths: z.array(z.string()).optional(),
2390
+ experimental: z.object({ sessionMemory: z.boolean().optional() }).strict().optional(),
2391
+ provider: z.union([
2392
+ z.literal("openai"),
2393
+ z.literal("local"),
2394
+ z.literal("gemini")
2395
+ ]).optional(),
2396
+ remote: z.object({
2397
+ baseUrl: z.string().optional(),
2398
+ apiKey: z.string().optional(),
2399
+ headers: z.record(z.string(), z.string()).optional(),
2400
+ batch: z.object({
2401
+ enabled: z.boolean().optional(),
2402
+ wait: z.boolean().optional(),
2403
+ concurrency: z.number().int().positive().optional(),
2404
+ pollIntervalMs: z.number().int().nonnegative().optional(),
2405
+ timeoutMinutes: z.number().int().positive().optional()
2406
+ }).strict().optional()
2407
+ }).strict().optional(),
2408
+ fallback: z.union([
2409
+ z.literal("openai"),
2410
+ z.literal("gemini"),
2411
+ z.literal("local"),
2412
+ z.literal("none")
2413
+ ]).optional(),
2414
+ model: z.string().optional(),
2415
+ local: z.object({
2416
+ modelPath: z.string().optional(),
2417
+ modelCacheDir: z.string().optional()
2418
+ }).strict().optional(),
2419
+ store: z.object({
2420
+ driver: z.literal("sqlite").optional(),
2421
+ path: z.string().optional(),
2422
+ vector: z.object({
2423
+ enabled: z.boolean().optional(),
2424
+ extensionPath: z.string().optional()
2425
+ }).strict().optional()
2426
+ }).strict().optional(),
2427
+ chunking: z.object({
2428
+ tokens: z.number().int().positive().optional(),
2429
+ overlap: z.number().int().nonnegative().optional()
2430
+ }).strict().optional(),
2431
+ sync: z.object({
2432
+ onSessionStart: z.boolean().optional(),
2433
+ onSearch: z.boolean().optional(),
2434
+ watch: z.boolean().optional(),
2435
+ watchDebounceMs: z.number().int().nonnegative().optional(),
2436
+ intervalMinutes: z.number().int().nonnegative().optional(),
2437
+ sessions: z.object({
2438
+ deltaBytes: z.number().int().nonnegative().optional(),
2439
+ deltaMessages: z.number().int().nonnegative().optional()
2440
+ }).strict().optional()
2441
+ }).strict().optional(),
2442
+ query: z.object({
2443
+ maxResults: z.number().int().positive().optional(),
2444
+ minScore: z.number().min(0).max(1).optional(),
2445
+ hybrid: z.object({
2446
+ enabled: z.boolean().optional(),
2447
+ vectorWeight: z.number().min(0).max(1).optional(),
2448
+ textWeight: z.number().min(0).max(1).optional(),
2449
+ candidateMultiplier: z.number().int().positive().optional()
2450
+ }).strict().optional()
2451
+ }).strict().optional(),
2452
+ cache: z.object({
2453
+ enabled: z.boolean().optional(),
2454
+ maxEntries: z.number().int().positive().optional()
2455
+ }).strict().optional()
2456
+ }).strict().optional();
2457
+ const AgentModelSchema = z.union([z.string(), z.object({
2458
+ primary: z.string().optional(),
2459
+ fallbacks: z.array(z.string()).optional()
2460
+ }).strict()]);
2461
+ const AgentEntrySchema = z.object({
2462
+ id: z.string(),
2463
+ default: z.boolean().optional(),
2464
+ name: z.string().optional(),
2465
+ workspace: z.string().optional(),
2466
+ agentDir: z.string().optional(),
2467
+ model: AgentModelSchema.optional(),
2468
+ skills: z.array(z.string()).optional(),
2469
+ memorySearch: MemorySearchSchema,
2470
+ humanDelay: HumanDelaySchema.optional(),
2471
+ heartbeat: HeartbeatSchema,
2472
+ identity: IdentitySchema,
2473
+ groupChat: GroupChatSchema,
2474
+ subagents: z.object({
2475
+ allowAgents: z.array(z.string()).optional(),
2476
+ model: z.union([z.string(), z.object({
2477
+ primary: z.string().optional(),
2478
+ fallbacks: z.array(z.string()).optional()
2479
+ }).strict()]).optional(),
2480
+ thinking: z.string().optional()
2481
+ }).strict().optional(),
2482
+ sandbox: AgentSandboxSchema,
2483
+ tools: AgentToolsSchema
2484
+ }).strict();
2485
+ const ToolsSchema = z.object({
2486
+ profile: ToolProfileSchema,
2487
+ allow: z.array(z.string()).optional(),
2488
+ alsoAllow: z.array(z.string()).optional(),
2489
+ deny: z.array(z.string()).optional(),
2490
+ byProvider: z.record(z.string(), ToolPolicyWithProfileSchema).optional(),
2491
+ web: ToolsWebSchema,
2492
+ media: ToolsMediaSchema,
2493
+ links: ToolsLinksSchema,
2494
+ message: z.object({
2495
+ allowCrossContextSend: z.boolean().optional(),
2496
+ crossContext: z.object({
2497
+ allowWithinProvider: z.boolean().optional(),
2498
+ allowAcrossProviders: z.boolean().optional(),
2499
+ marker: z.object({
2500
+ enabled: z.boolean().optional(),
2501
+ prefix: z.string().optional(),
2502
+ suffix: z.string().optional()
2503
+ }).strict().optional()
2504
+ }).strict().optional(),
2505
+ broadcast: z.object({ enabled: z.boolean().optional() }).strict().optional()
2506
+ }).strict().optional(),
2507
+ agentToAgent: z.object({
2508
+ enabled: z.boolean().optional(),
2509
+ allow: z.array(z.string()).optional()
2510
+ }).strict().optional(),
2511
+ elevated: z.object({
2512
+ enabled: z.boolean().optional(),
2513
+ allowFrom: ElevatedAllowFromSchema
2514
+ }).strict().optional(),
2515
+ exec: z.object({
2516
+ host: z.enum([
2517
+ "sandbox",
2518
+ "gateway",
2519
+ "node"
2520
+ ]).optional(),
2521
+ security: z.enum([
2522
+ "deny",
2523
+ "allowlist",
2524
+ "full"
2525
+ ]).optional(),
2526
+ ask: z.enum([
2527
+ "off",
2528
+ "on-miss",
2529
+ "always"
2530
+ ]).optional(),
2531
+ node: z.string().optional(),
2532
+ pathPrepend: z.array(z.string()).optional(),
2533
+ safeBins: z.array(z.string()).optional(),
2534
+ backgroundMs: z.number().int().positive().optional(),
2535
+ timeoutSec: z.number().int().positive().optional(),
2536
+ cleanupMs: z.number().int().positive().optional(),
2537
+ notifyOnExit: z.boolean().optional(),
2538
+ applyPatch: z.object({
2539
+ enabled: z.boolean().optional(),
2540
+ allowModels: z.array(z.string()).optional()
2541
+ }).strict().optional()
2542
+ }).strict().optional(),
2543
+ subagents: z.object({ tools: ToolPolicySchema }).strict().optional(),
2544
+ sandbox: z.object({ tools: ToolPolicySchema }).strict().optional()
2545
+ }).strict().superRefine((value, ctx) => {
2546
+ if (value.allow && value.allow.length > 0 && value.alsoAllow && value.alsoAllow.length > 0) ctx.addIssue({
2547
+ code: z.ZodIssueCode.custom,
2548
+ message: "tools cannot set both allow and alsoAllow in the same scope (merge alsoAllow into allow, or remove allow and use profile + alsoAllow)"
2549
+ });
2550
+ }).optional();
2551
+
2552
+ //#endregion
2553
+ //#region src/config/zod-schema.agent-defaults.ts
2554
+ const AgentDefaultsSchema = z.object({
2555
+ model: z.object({
2556
+ primary: z.string().optional(),
2557
+ fallbacks: z.array(z.string()).optional()
2558
+ }).strict().optional(),
2559
+ imageModel: z.object({
2560
+ primary: z.string().optional(),
2561
+ fallbacks: z.array(z.string()).optional()
2562
+ }).strict().optional(),
2563
+ models: z.record(z.string(), z.object({
2564
+ alias: z.string().optional(),
2565
+ params: z.record(z.string(), z.unknown()).optional()
2566
+ }).strict()).optional(),
2567
+ workspace: z.string().optional(),
2568
+ repoRoot: z.string().optional(),
2569
+ skipBootstrap: z.boolean().optional(),
2570
+ bootstrapMaxChars: z.number().int().positive().optional(),
2571
+ userTimezone: z.string().optional(),
2572
+ timeFormat: z.union([
2573
+ z.literal("auto"),
2574
+ z.literal("12"),
2575
+ z.literal("24")
2576
+ ]).optional(),
2577
+ envelopeTimezone: z.string().optional(),
2578
+ envelopeTimestamp: z.union([z.literal("on"), z.literal("off")]).optional(),
2579
+ envelopeElapsed: z.union([z.literal("on"), z.literal("off")]).optional(),
2580
+ contextTokens: z.number().int().positive().optional(),
2581
+ cliBackends: z.record(z.string(), CliBackendSchema).optional(),
2582
+ memorySearch: MemorySearchSchema,
2583
+ contextPruning: z.object({
2584
+ mode: z.union([z.literal("off"), z.literal("cache-ttl")]).optional(),
2585
+ ttl: z.string().optional(),
2586
+ keepLastAssistants: z.number().int().nonnegative().optional(),
2587
+ softTrimRatio: z.number().min(0).max(1).optional(),
2588
+ hardClearRatio: z.number().min(0).max(1).optional(),
2589
+ minPrunableToolChars: z.number().int().nonnegative().optional(),
2590
+ tools: z.object({
2591
+ allow: z.array(z.string()).optional(),
2592
+ deny: z.array(z.string()).optional()
2593
+ }).strict().optional(),
2594
+ softTrim: z.object({
2595
+ maxChars: z.number().int().nonnegative().optional(),
2596
+ headChars: z.number().int().nonnegative().optional(),
2597
+ tailChars: z.number().int().nonnegative().optional()
2598
+ }).strict().optional(),
2599
+ hardClear: z.object({
2600
+ enabled: z.boolean().optional(),
2601
+ placeholder: z.string().optional()
2602
+ }).strict().optional()
2603
+ }).strict().optional(),
2604
+ compaction: z.object({
2605
+ mode: z.union([z.literal("default"), z.literal("safeguard")]).optional(),
2606
+ reserveTokensFloor: z.number().int().nonnegative().optional(),
2607
+ maxHistoryShare: z.number().min(.1).max(.9).optional(),
2608
+ memoryFlush: z.object({
2609
+ enabled: z.boolean().optional(),
2610
+ softThresholdTokens: z.number().int().nonnegative().optional(),
2611
+ prompt: z.string().optional(),
2612
+ systemPrompt: z.string().optional()
2613
+ }).strict().optional()
2614
+ }).strict().optional(),
2615
+ thinkingDefault: z.union([
2616
+ z.literal("off"),
2617
+ z.literal("minimal"),
2618
+ z.literal("low"),
2619
+ z.literal("medium"),
2620
+ z.literal("high"),
2621
+ z.literal("xhigh")
2622
+ ]).optional(),
2623
+ verboseDefault: z.union([
2624
+ z.literal("off"),
2625
+ z.literal("on"),
2626
+ z.literal("full")
2627
+ ]).optional(),
2628
+ elevatedDefault: z.union([
2629
+ z.literal("off"),
2630
+ z.literal("on"),
2631
+ z.literal("ask"),
2632
+ z.literal("full")
2633
+ ]).optional(),
2634
+ blockStreamingDefault: z.union([z.literal("off"), z.literal("on")]).optional(),
2635
+ blockStreamingBreak: z.union([z.literal("text_end"), z.literal("message_end")]).optional(),
2636
+ blockStreamingChunk: BlockStreamingChunkSchema.optional(),
2637
+ blockStreamingCoalesce: BlockStreamingCoalesceSchema.optional(),
2638
+ humanDelay: HumanDelaySchema.optional(),
2639
+ timeoutSeconds: z.number().int().positive().optional(),
2640
+ mediaMaxMb: z.number().positive().optional(),
2641
+ typingIntervalSeconds: z.number().int().positive().optional(),
2642
+ typingMode: z.union([
2643
+ z.literal("never"),
2644
+ z.literal("instant"),
2645
+ z.literal("thinking"),
2646
+ z.literal("message")
2647
+ ]).optional(),
2648
+ heartbeat: HeartbeatSchema,
2649
+ maxConcurrent: z.number().int().positive().optional(),
2650
+ subagents: z.object({
2651
+ maxConcurrent: z.number().int().positive().optional(),
2652
+ archiveAfterMinutes: z.number().int().positive().optional(),
2653
+ model: z.union([z.string(), z.object({
2654
+ primary: z.string().optional(),
2655
+ fallbacks: z.array(z.string()).optional()
2656
+ }).strict()]).optional(),
2657
+ thinking: z.string().optional()
2658
+ }).strict().optional(),
2659
+ sandbox: z.object({
2660
+ mode: z.union([
2661
+ z.literal("off"),
2662
+ z.literal("non-main"),
2663
+ z.literal("all")
2664
+ ]).optional(),
2665
+ workspaceAccess: z.union([
2666
+ z.literal("none"),
2667
+ z.literal("ro"),
2668
+ z.literal("rw")
2669
+ ]).optional(),
2670
+ sessionToolsVisibility: z.union([z.literal("spawned"), z.literal("all")]).optional(),
2671
+ scope: z.union([
2672
+ z.literal("session"),
2673
+ z.literal("agent"),
2674
+ z.literal("shared")
2675
+ ]).optional(),
2676
+ perSession: z.boolean().optional(),
2677
+ workspaceRoot: z.string().optional(),
2678
+ docker: SandboxDockerSchema,
2679
+ browser: SandboxBrowserSchema,
2680
+ prune: SandboxPruneSchema
2681
+ }).strict().optional()
2682
+ }).strict().optional();
2683
+
2684
+ //#endregion
2685
+ //#region src/config/zod-schema.agents.ts
2686
+ const AgentsSchema = z.object({
2687
+ defaults: z.lazy(() => AgentDefaultsSchema).optional(),
2688
+ list: z.array(AgentEntrySchema).optional()
2689
+ }).strict().optional();
2690
+ const BindingsSchema = z.array(z.object({
2691
+ agentId: z.string(),
2692
+ match: z.object({
2693
+ channel: z.string(),
2694
+ accountId: z.string().optional(),
2695
+ peer: z.object({
2696
+ kind: z.union([
2697
+ z.literal("dm"),
2698
+ z.literal("group"),
2699
+ z.literal("channel")
2700
+ ]),
2701
+ id: z.string()
2702
+ }).strict().optional(),
2703
+ guildId: z.string().optional(),
2704
+ teamId: z.string().optional()
2705
+ }).strict()
2706
+ }).strict()).optional();
2707
+ const BroadcastStrategySchema = z.enum(["parallel", "sequential"]);
2708
+ const BroadcastSchema = z.object({ strategy: BroadcastStrategySchema.optional() }).catchall(z.array(z.string())).optional();
2709
+ const AudioSchema = z.object({ transcription: TranscribeAudioSchema }).strict().optional();
2710
+
2711
+ //#endregion
2712
+ //#region src/config/zod-schema.approvals.ts
2713
+ const ExecApprovalForwardTargetSchema = z.object({
2714
+ channel: z.string().min(1),
2715
+ to: z.string().min(1),
2716
+ accountId: z.string().optional(),
2717
+ threadId: z.union([z.string(), z.number()]).optional()
2718
+ }).strict();
2719
+ const ExecApprovalForwardingSchema = z.object({
2720
+ enabled: z.boolean().optional(),
2721
+ mode: z.union([
2722
+ z.literal("session"),
2723
+ z.literal("targets"),
2724
+ z.literal("both")
2725
+ ]).optional(),
2726
+ agentFilter: z.array(z.string()).optional(),
2727
+ sessionFilter: z.array(z.string()).optional(),
2728
+ targets: z.array(ExecApprovalForwardTargetSchema).optional()
2729
+ }).strict().optional();
2730
+ const ApprovalsSchema = z.object({ exec: ExecApprovalForwardingSchema }).strict().optional();
2731
+
2732
+ //#endregion
2733
+ //#region src/config/zod-schema.hooks.ts
2734
+ const HookMappingSchema = z.object({
2735
+ id: z.string().optional(),
2736
+ match: z.object({
2737
+ path: z.string().optional(),
2738
+ source: z.string().optional()
2739
+ }).optional(),
2740
+ action: z.union([z.literal("wake"), z.literal("agent")]).optional(),
2741
+ wakeMode: z.union([z.literal("now"), z.literal("next-heartbeat")]).optional(),
2742
+ name: z.string().optional(),
2743
+ sessionKey: z.string().optional(),
2744
+ messageTemplate: z.string().optional(),
2745
+ textTemplate: z.string().optional(),
2746
+ deliver: z.boolean().optional(),
2747
+ allowUnsafeExternalContent: z.boolean().optional(),
2748
+ channel: z.union([
2749
+ z.literal("last"),
2750
+ z.literal("whatsapp"),
2751
+ z.literal("telegram"),
2752
+ z.literal("discord"),
2753
+ z.literal("slack"),
2754
+ z.literal("signal"),
2755
+ z.literal("imessage"),
2756
+ z.literal("msteams")
2757
+ ]).optional(),
2758
+ to: z.string().optional(),
2759
+ model: z.string().optional(),
2760
+ thinking: z.string().optional(),
2761
+ timeoutSeconds: z.number().int().positive().optional(),
2762
+ transform: z.object({
2763
+ module: z.string(),
2764
+ export: z.string().optional()
2765
+ }).strict().optional()
2766
+ }).strict().optional();
2767
+ const InternalHookHandlerSchema = z.object({
2768
+ event: z.string(),
2769
+ module: z.string(),
2770
+ export: z.string().optional()
2771
+ }).strict();
2772
+ const HookConfigSchema = z.object({
2773
+ enabled: z.boolean().optional(),
2774
+ env: z.record(z.string(), z.string()).optional()
2775
+ }).strict();
2776
+ const HookInstallRecordSchema = z.object({
2777
+ source: z.union([
2778
+ z.literal("npm"),
2779
+ z.literal("archive"),
2780
+ z.literal("path")
2781
+ ]),
2782
+ spec: z.string().optional(),
2783
+ sourcePath: z.string().optional(),
2784
+ installPath: z.string().optional(),
2785
+ version: z.string().optional(),
2786
+ installedAt: z.string().optional(),
2787
+ hooks: z.array(z.string()).optional()
2788
+ }).strict();
2789
+ const InternalHooksSchema = z.object({
2790
+ enabled: z.boolean().optional(),
2791
+ handlers: z.array(InternalHookHandlerSchema).optional(),
2792
+ entries: z.record(z.string(), HookConfigSchema).optional(),
2793
+ load: z.object({ extraDirs: z.array(z.string()).optional() }).strict().optional(),
2794
+ installs: z.record(z.string(), HookInstallRecordSchema).optional()
2795
+ }).strict().optional();
2796
+ const HooksGmailSchema = z.object({
2797
+ account: z.string().optional(),
2798
+ label: z.string().optional(),
2799
+ topic: z.string().optional(),
2800
+ subscription: z.string().optional(),
2801
+ pushToken: z.string().optional(),
2802
+ hookUrl: z.string().optional(),
2803
+ includeBody: z.boolean().optional(),
2804
+ maxBytes: z.number().int().positive().optional(),
2805
+ renewEveryMinutes: z.number().int().positive().optional(),
2806
+ allowUnsafeExternalContent: z.boolean().optional(),
2807
+ serve: z.object({
2808
+ bind: z.string().optional(),
2809
+ port: z.number().int().positive().optional(),
2810
+ path: z.string().optional()
2811
+ }).strict().optional(),
2812
+ tailscale: z.object({
2813
+ mode: z.union([
2814
+ z.literal("off"),
2815
+ z.literal("serve"),
2816
+ z.literal("funnel")
2817
+ ]).optional(),
2818
+ path: z.string().optional(),
2819
+ target: z.string().optional()
2820
+ }).strict().optional(),
2821
+ model: z.string().optional(),
2822
+ thinking: z.union([
2823
+ z.literal("off"),
2824
+ z.literal("minimal"),
2825
+ z.literal("low"),
2826
+ z.literal("medium"),
2827
+ z.literal("high")
2828
+ ]).optional()
2829
+ }).strict().optional();
2830
+
2831
+ //#endregion
2832
+ //#region src/config/zod-schema.channels.ts
2833
+ const ChannelHeartbeatVisibilitySchema = z.object({
2834
+ showOk: z.boolean().optional(),
2835
+ showAlerts: z.boolean().optional(),
2836
+ useIndicator: z.boolean().optional()
2837
+ }).strict().optional();
2838
+
2839
+ //#endregion
2840
+ //#region src/config/telegram-custom-commands.ts
2841
+ const TELEGRAM_COMMAND_NAME_PATTERN = /^[a-z0-9_]{1,32}$/;
2842
+ function normalizeTelegramCommandName(value) {
2843
+ const trimmed = value.trim();
2844
+ if (!trimmed) return "";
2845
+ return (trimmed.startsWith("/") ? trimmed.slice(1) : trimmed).trim().toLowerCase();
2846
+ }
2847
+ function normalizeTelegramCommandDescription(value) {
2848
+ return value.trim();
2849
+ }
2850
+ function resolveTelegramCustomCommands(params) {
2851
+ const entries = Array.isArray(params.commands) ? params.commands : [];
2852
+ const reserved = params.reservedCommands ?? /* @__PURE__ */ new Set();
2853
+ const checkReserved = params.checkReserved !== false;
2854
+ const checkDuplicates = params.checkDuplicates !== false;
2855
+ const seen = /* @__PURE__ */ new Set();
2856
+ const resolved = [];
2857
+ const issues = [];
2858
+ for (let index = 0; index < entries.length; index += 1) {
2859
+ const entry = entries[index];
2860
+ const normalized = normalizeTelegramCommandName(String(entry?.command ?? ""));
2861
+ if (!normalized) {
2862
+ issues.push({
2863
+ index,
2864
+ field: "command",
2865
+ message: "Telegram custom command is missing a command name."
2866
+ });
2867
+ continue;
2868
+ }
2869
+ if (!TELEGRAM_COMMAND_NAME_PATTERN.test(normalized)) {
2870
+ issues.push({
2871
+ index,
2872
+ field: "command",
2873
+ message: `Telegram custom command "/${normalized}" is invalid (use a-z, 0-9, underscore; max 32 chars).`
2874
+ });
2875
+ continue;
2876
+ }
2877
+ if (checkReserved && reserved.has(normalized)) {
2878
+ issues.push({
2879
+ index,
2880
+ field: "command",
2881
+ message: `Telegram custom command "/${normalized}" conflicts with a native command.`
2882
+ });
2883
+ continue;
2884
+ }
2885
+ if (checkDuplicates && seen.has(normalized)) {
2886
+ issues.push({
2887
+ index,
2888
+ field: "command",
2889
+ message: `Telegram custom command "/${normalized}" is duplicated.`
2890
+ });
2891
+ continue;
2892
+ }
2893
+ const description = normalizeTelegramCommandDescription(String(entry?.description ?? ""));
2894
+ if (!description) {
2895
+ issues.push({
2896
+ index,
2897
+ field: "description",
2898
+ message: `Telegram custom command "/${normalized}" is missing a description.`
2899
+ });
2900
+ continue;
2901
+ }
2902
+ if (checkDuplicates) seen.add(normalized);
2903
+ resolved.push({
2904
+ command: normalized,
2905
+ description
2906
+ });
2907
+ }
2908
+ return {
2909
+ commands: resolved,
2910
+ issues
2911
+ };
2912
+ }
2913
+
2914
+ //#endregion
2915
+ //#region src/config/zod-schema.providers-core.ts
2916
+ const ToolPolicyBySenderSchema$1 = z.record(z.string(), ToolPolicySchema).optional();
2917
+ const TelegramInlineButtonsScopeSchema = z.enum([
2918
+ "off",
2919
+ "dm",
2920
+ "group",
2921
+ "all",
2922
+ "allowlist"
2923
+ ]);
2924
+ const TelegramCapabilitiesSchema = z.union([z.array(z.string()), z.object({ inlineButtons: TelegramInlineButtonsScopeSchema.optional() }).strict()]);
2925
+ const TelegramTopicSchema = z.object({
2926
+ requireMention: z.boolean().optional(),
2927
+ skills: z.array(z.string()).optional(),
2928
+ enabled: z.boolean().optional(),
2929
+ allowFrom: z.array(z.union([z.string(), z.number()])).optional(),
2930
+ systemPrompt: z.string().optional()
2931
+ }).strict();
2932
+ const TelegramGroupSchema = z.object({
2933
+ requireMention: z.boolean().optional(),
2934
+ tools: ToolPolicySchema,
2935
+ toolsBySender: ToolPolicyBySenderSchema$1,
2936
+ skills: z.array(z.string()).optional(),
2937
+ enabled: z.boolean().optional(),
2938
+ allowFrom: z.array(z.union([z.string(), z.number()])).optional(),
2939
+ systemPrompt: z.string().optional(),
2940
+ topics: z.record(z.string(), TelegramTopicSchema.optional()).optional()
2941
+ }).strict();
2942
+ const TelegramCustomCommandSchema = z.object({
2943
+ command: z.string().transform(normalizeTelegramCommandName),
2944
+ description: z.string().transform(normalizeTelegramCommandDescription)
2945
+ }).strict();
2946
+ const validateTelegramCustomCommands = (value, ctx) => {
2947
+ if (!value.customCommands || value.customCommands.length === 0) return;
2948
+ const { issues } = resolveTelegramCustomCommands({
2949
+ commands: value.customCommands,
2950
+ checkReserved: false,
2951
+ checkDuplicates: false
2952
+ });
2953
+ for (const issue of issues) ctx.addIssue({
2954
+ code: z.ZodIssueCode.custom,
2955
+ path: [
2956
+ "customCommands",
2957
+ issue.index,
2958
+ issue.field
2959
+ ],
2960
+ message: issue.message
2961
+ });
2962
+ };
2963
+ const TelegramAccountSchemaBase = z.object({
2964
+ name: z.string().optional(),
2965
+ capabilities: TelegramCapabilitiesSchema.optional(),
2966
+ markdown: MarkdownConfigSchema,
2967
+ enabled: z.boolean().optional(),
2968
+ commands: ProviderCommandsSchema,
2969
+ customCommands: z.array(TelegramCustomCommandSchema).optional(),
2970
+ configWrites: z.boolean().optional(),
2971
+ dmPolicy: DmPolicySchema.optional().default("pairing"),
2972
+ botToken: z.string().optional(),
2973
+ tokenFile: z.string().optional(),
2974
+ replyToMode: ReplyToModeSchema.optional(),
2975
+ groups: z.record(z.string(), TelegramGroupSchema.optional()).optional(),
2976
+ allowFrom: z.array(z.union([z.string(), z.number()])).optional(),
2977
+ groupAllowFrom: z.array(z.union([z.string(), z.number()])).optional(),
2978
+ groupPolicy: GroupPolicySchema.optional().default("allowlist"),
2979
+ historyLimit: z.number().int().min(0).optional(),
2980
+ dmHistoryLimit: z.number().int().min(0).optional(),
2981
+ dms: z.record(z.string(), DmConfigSchema.optional()).optional(),
2982
+ textChunkLimit: z.number().int().positive().optional(),
2983
+ chunkMode: z.enum(["length", "newline"]).optional(),
2984
+ blockStreaming: z.boolean().optional(),
2985
+ draftChunk: BlockStreamingChunkSchema.optional(),
2986
+ blockStreamingCoalesce: BlockStreamingCoalesceSchema.optional(),
2987
+ streamMode: z.enum([
2988
+ "off",
2989
+ "partial",
2990
+ "block"
2991
+ ]).optional().default("partial"),
2992
+ mediaMaxMb: z.number().positive().optional(),
2993
+ timeoutSeconds: z.number().int().positive().optional(),
2994
+ retry: RetryConfigSchema,
2995
+ network: z.object({ autoSelectFamily: z.boolean().optional() }).strict().optional(),
2996
+ proxy: z.string().optional(),
2997
+ webhookUrl: z.string().optional(),
2998
+ webhookSecret: z.string().optional(),
2999
+ webhookPath: z.string().optional(),
3000
+ actions: z.object({
3001
+ reactions: z.boolean().optional(),
3002
+ sendMessage: z.boolean().optional(),
3003
+ deleteMessage: z.boolean().optional(),
3004
+ sticker: z.boolean().optional()
3005
+ }).strict().optional(),
3006
+ reactionNotifications: z.enum([
3007
+ "off",
3008
+ "own",
3009
+ "all"
3010
+ ]).optional(),
3011
+ reactionLevel: z.enum([
3012
+ "off",
3013
+ "ack",
3014
+ "minimal",
3015
+ "extensive"
3016
+ ]).optional(),
3017
+ heartbeat: ChannelHeartbeatVisibilitySchema,
3018
+ linkPreview: z.boolean().optional()
3019
+ }).strict();
3020
+ const TelegramAccountSchema = TelegramAccountSchemaBase.superRefine((value, ctx) => {
3021
+ requireOpenAllowFrom({
3022
+ policy: value.dmPolicy,
3023
+ allowFrom: value.allowFrom,
3024
+ ctx,
3025
+ path: ["allowFrom"],
3026
+ message: "channels.telegram.dmPolicy=\"open\" requires channels.telegram.allowFrom to include \"*\""
3027
+ });
3028
+ validateTelegramCustomCommands(value, ctx);
3029
+ });
3030
+ const TelegramConfigSchema = TelegramAccountSchemaBase.extend({ accounts: z.record(z.string(), TelegramAccountSchema.optional()).optional() }).superRefine((value, ctx) => {
3031
+ requireOpenAllowFrom({
3032
+ policy: value.dmPolicy,
3033
+ allowFrom: value.allowFrom,
3034
+ ctx,
3035
+ path: ["allowFrom"],
3036
+ message: "channels.telegram.dmPolicy=\"open\" requires channels.telegram.allowFrom to include \"*\""
3037
+ });
3038
+ validateTelegramCustomCommands(value, ctx);
3039
+ const baseWebhookUrl = typeof value.webhookUrl === "string" ? value.webhookUrl.trim() : "";
3040
+ const baseWebhookSecret = typeof value.webhookSecret === "string" ? value.webhookSecret.trim() : "";
3041
+ if (baseWebhookUrl && !baseWebhookSecret) ctx.addIssue({
3042
+ code: z.ZodIssueCode.custom,
3043
+ message: "channels.telegram.webhookUrl requires channels.telegram.webhookSecret",
3044
+ path: ["webhookSecret"]
3045
+ });
3046
+ if (!value.accounts) return;
3047
+ for (const [accountId, account] of Object.entries(value.accounts)) {
3048
+ if (!account) continue;
3049
+ if (account.enabled === false) continue;
3050
+ if (!(typeof account.webhookUrl === "string" ? account.webhookUrl.trim() : "")) continue;
3051
+ if (!(typeof account.webhookSecret === "string" ? account.webhookSecret.trim() : "") && !baseWebhookSecret) ctx.addIssue({
3052
+ code: z.ZodIssueCode.custom,
3053
+ message: "channels.telegram.accounts.*.webhookUrl requires channels.telegram.webhookSecret or channels.telegram.accounts.*.webhookSecret",
3054
+ path: [
3055
+ "accounts",
3056
+ accountId,
3057
+ "webhookSecret"
3058
+ ]
3059
+ });
3060
+ }
3061
+ });
3062
+ const DiscordDmSchema = z.object({
3063
+ enabled: z.boolean().optional(),
3064
+ policy: DmPolicySchema.optional().default("pairing"),
3065
+ allowFrom: z.array(z.union([z.string(), z.number()])).optional(),
3066
+ groupEnabled: z.boolean().optional(),
3067
+ groupChannels: z.array(z.union([z.string(), z.number()])).optional()
3068
+ }).strict().superRefine((value, ctx) => {
3069
+ requireOpenAllowFrom({
3070
+ policy: value.policy,
3071
+ allowFrom: value.allowFrom,
3072
+ ctx,
3073
+ path: ["allowFrom"],
3074
+ message: "channels.discord.dm.policy=\"open\" requires channels.discord.dm.allowFrom to include \"*\""
3075
+ });
3076
+ });
3077
+ const DiscordGuildChannelSchema = z.object({
3078
+ allow: z.boolean().optional(),
3079
+ requireMention: z.boolean().optional(),
3080
+ tools: ToolPolicySchema,
3081
+ toolsBySender: ToolPolicyBySenderSchema$1,
3082
+ skills: z.array(z.string()).optional(),
3083
+ enabled: z.boolean().optional(),
3084
+ users: z.array(z.union([z.string(), z.number()])).optional(),
3085
+ systemPrompt: z.string().optional(),
3086
+ autoThread: z.boolean().optional()
3087
+ }).strict();
3088
+ const DiscordGuildSchema = z.object({
3089
+ slug: z.string().optional(),
3090
+ requireMention: z.boolean().optional(),
3091
+ tools: ToolPolicySchema,
3092
+ toolsBySender: ToolPolicyBySenderSchema$1,
3093
+ reactionNotifications: z.enum([
3094
+ "off",
3095
+ "own",
3096
+ "all",
3097
+ "allowlist"
3098
+ ]).optional(),
3099
+ users: z.array(z.union([z.string(), z.number()])).optional(),
3100
+ channels: z.record(z.string(), DiscordGuildChannelSchema.optional()).optional()
3101
+ }).strict();
3102
+ const DiscordAccountSchema = z.object({
3103
+ name: z.string().optional(),
3104
+ capabilities: z.array(z.string()).optional(),
3105
+ markdown: MarkdownConfigSchema,
3106
+ enabled: z.boolean().optional(),
3107
+ commands: ProviderCommandsSchema,
3108
+ configWrites: z.boolean().optional(),
3109
+ token: z.string().optional(),
3110
+ allowBots: z.boolean().optional(),
3111
+ groupPolicy: GroupPolicySchema.optional().default("allowlist"),
3112
+ historyLimit: z.number().int().min(0).optional(),
3113
+ dmHistoryLimit: z.number().int().min(0).optional(),
3114
+ dms: z.record(z.string(), DmConfigSchema.optional()).optional(),
3115
+ textChunkLimit: z.number().int().positive().optional(),
3116
+ chunkMode: z.enum(["length", "newline"]).optional(),
3117
+ blockStreaming: z.boolean().optional(),
3118
+ blockStreamingCoalesce: BlockStreamingCoalesceSchema.optional(),
3119
+ maxLinesPerMessage: z.number().int().positive().optional(),
3120
+ mediaMaxMb: z.number().positive().optional(),
3121
+ retry: RetryConfigSchema,
3122
+ actions: z.object({
3123
+ reactions: z.boolean().optional(),
3124
+ stickers: z.boolean().optional(),
3125
+ emojiUploads: z.boolean().optional(),
3126
+ stickerUploads: z.boolean().optional(),
3127
+ polls: z.boolean().optional(),
3128
+ permissions: z.boolean().optional(),
3129
+ messages: z.boolean().optional(),
3130
+ threads: z.boolean().optional(),
3131
+ pins: z.boolean().optional(),
3132
+ search: z.boolean().optional(),
3133
+ memberInfo: z.boolean().optional(),
3134
+ roleInfo: z.boolean().optional(),
3135
+ roles: z.boolean().optional(),
3136
+ channelInfo: z.boolean().optional(),
3137
+ voiceStatus: z.boolean().optional(),
3138
+ events: z.boolean().optional(),
3139
+ moderation: z.boolean().optional(),
3140
+ channels: z.boolean().optional(),
3141
+ presence: z.boolean().optional()
3142
+ }).strict().optional(),
3143
+ replyToMode: ReplyToModeSchema.optional(),
3144
+ dm: DiscordDmSchema.optional(),
3145
+ guilds: z.record(z.string(), DiscordGuildSchema.optional()).optional(),
3146
+ heartbeat: ChannelHeartbeatVisibilitySchema,
3147
+ execApprovals: z.object({
3148
+ enabled: z.boolean().optional(),
3149
+ approvers: z.array(z.union([z.string(), z.number()])).optional(),
3150
+ agentFilter: z.array(z.string()).optional(),
3151
+ sessionFilter: z.array(z.string()).optional()
3152
+ }).strict().optional(),
3153
+ intents: z.object({
3154
+ presence: z.boolean().optional(),
3155
+ guildMembers: z.boolean().optional()
3156
+ }).strict().optional(),
3157
+ pluralkit: z.object({
3158
+ enabled: z.boolean().optional(),
3159
+ token: z.string().optional()
3160
+ }).strict().optional()
3161
+ }).strict();
3162
+ const DiscordConfigSchema = DiscordAccountSchema.extend({ accounts: z.record(z.string(), DiscordAccountSchema.optional()).optional() });
3163
+ const GoogleChatDmSchema = z.object({
3164
+ enabled: z.boolean().optional(),
3165
+ policy: DmPolicySchema.optional().default("pairing"),
3166
+ allowFrom: z.array(z.union([z.string(), z.number()])).optional()
3167
+ }).strict().superRefine((value, ctx) => {
3168
+ requireOpenAllowFrom({
3169
+ policy: value.policy,
3170
+ allowFrom: value.allowFrom,
3171
+ ctx,
3172
+ path: ["allowFrom"],
3173
+ message: "channels.googlechat.dm.policy=\"open\" requires channels.googlechat.dm.allowFrom to include \"*\""
3174
+ });
3175
+ });
3176
+ const GoogleChatGroupSchema = z.object({
3177
+ enabled: z.boolean().optional(),
3178
+ allow: z.boolean().optional(),
3179
+ requireMention: z.boolean().optional(),
3180
+ users: z.array(z.union([z.string(), z.number()])).optional(),
3181
+ systemPrompt: z.string().optional()
3182
+ }).strict();
3183
+ const GoogleChatAccountSchema = z.object({
3184
+ name: z.string().optional(),
3185
+ capabilities: z.array(z.string()).optional(),
3186
+ enabled: z.boolean().optional(),
3187
+ configWrites: z.boolean().optional(),
3188
+ allowBots: z.boolean().optional(),
3189
+ requireMention: z.boolean().optional(),
3190
+ groupPolicy: GroupPolicySchema.optional().default("allowlist"),
3191
+ groupAllowFrom: z.array(z.union([z.string(), z.number()])).optional(),
3192
+ groups: z.record(z.string(), GoogleChatGroupSchema.optional()).optional(),
3193
+ serviceAccount: z.union([z.string(), z.record(z.string(), z.unknown())]).optional(),
3194
+ serviceAccountFile: z.string().optional(),
3195
+ audienceType: z.enum(["app-url", "project-number"]).optional(),
3196
+ audience: z.string().optional(),
3197
+ webhookPath: z.string().optional(),
3198
+ webhookUrl: z.string().optional(),
3199
+ botUser: z.string().optional(),
3200
+ historyLimit: z.number().int().min(0).optional(),
3201
+ dmHistoryLimit: z.number().int().min(0).optional(),
3202
+ dms: z.record(z.string(), DmConfigSchema.optional()).optional(),
3203
+ textChunkLimit: z.number().int().positive().optional(),
3204
+ chunkMode: z.enum(["length", "newline"]).optional(),
3205
+ blockStreaming: z.boolean().optional(),
3206
+ blockStreamingCoalesce: BlockStreamingCoalesceSchema.optional(),
3207
+ mediaMaxMb: z.number().positive().optional(),
3208
+ replyToMode: ReplyToModeSchema.optional(),
3209
+ actions: z.object({ reactions: z.boolean().optional() }).strict().optional(),
3210
+ dm: GoogleChatDmSchema.optional(),
3211
+ typingIndicator: z.enum([
3212
+ "none",
3213
+ "message",
3214
+ "reaction"
3215
+ ]).optional()
3216
+ }).strict();
3217
+ const GoogleChatConfigSchema = GoogleChatAccountSchema.extend({
3218
+ accounts: z.record(z.string(), GoogleChatAccountSchema.optional()).optional(),
3219
+ defaultAccount: z.string().optional()
3220
+ });
3221
+ const SlackDmSchema = z.object({
3222
+ enabled: z.boolean().optional(),
3223
+ policy: DmPolicySchema.optional().default("pairing"),
3224
+ allowFrom: z.array(z.union([z.string(), z.number()])).optional(),
3225
+ groupEnabled: z.boolean().optional(),
3226
+ groupChannels: z.array(z.union([z.string(), z.number()])).optional(),
3227
+ replyToMode: ReplyToModeSchema.optional()
3228
+ }).strict().superRefine((value, ctx) => {
3229
+ requireOpenAllowFrom({
3230
+ policy: value.policy,
3231
+ allowFrom: value.allowFrom,
3232
+ ctx,
3233
+ path: ["allowFrom"],
3234
+ message: "channels.slack.dm.policy=\"open\" requires channels.slack.dm.allowFrom to include \"*\""
3235
+ });
3236
+ });
3237
+ const SlackChannelSchema = z.object({
3238
+ enabled: z.boolean().optional(),
3239
+ allow: z.boolean().optional(),
3240
+ requireMention: z.boolean().optional(),
3241
+ tools: ToolPolicySchema,
3242
+ toolsBySender: ToolPolicyBySenderSchema$1,
3243
+ allowBots: z.boolean().optional(),
3244
+ users: z.array(z.union([z.string(), z.number()])).optional(),
3245
+ skills: z.array(z.string()).optional(),
3246
+ systemPrompt: z.string().optional()
3247
+ }).strict();
3248
+ const SlackThreadSchema = z.object({
3249
+ historyScope: z.enum(["thread", "channel"]).optional(),
3250
+ inheritParent: z.boolean().optional()
3251
+ }).strict();
3252
+ const SlackReplyToModeByChatTypeSchema = z.object({
3253
+ direct: ReplyToModeSchema.optional(),
3254
+ group: ReplyToModeSchema.optional(),
3255
+ channel: ReplyToModeSchema.optional()
3256
+ }).strict();
3257
+ const SlackAccountSchema = z.object({
3258
+ name: z.string().optional(),
3259
+ mode: z.enum(["socket", "http"]).optional(),
3260
+ signingSecret: z.string().optional(),
3261
+ webhookPath: z.string().optional(),
3262
+ capabilities: z.array(z.string()).optional(),
3263
+ markdown: MarkdownConfigSchema,
3264
+ enabled: z.boolean().optional(),
3265
+ commands: ProviderCommandsSchema,
3266
+ configWrites: z.boolean().optional(),
3267
+ botToken: z.string().optional(),
3268
+ appToken: z.string().optional(),
3269
+ userToken: z.string().optional(),
3270
+ userTokenReadOnly: z.boolean().optional().default(true),
3271
+ allowBots: z.boolean().optional(),
3272
+ requireMention: z.boolean().optional(),
3273
+ groupPolicy: GroupPolicySchema.optional().default("allowlist"),
3274
+ historyLimit: z.number().int().min(0).optional(),
3275
+ dmHistoryLimit: z.number().int().min(0).optional(),
3276
+ dms: z.record(z.string(), DmConfigSchema.optional()).optional(),
3277
+ textChunkLimit: z.number().int().positive().optional(),
3278
+ chunkMode: z.enum(["length", "newline"]).optional(),
3279
+ blockStreaming: z.boolean().optional(),
3280
+ blockStreamingCoalesce: BlockStreamingCoalesceSchema.optional(),
3281
+ mediaMaxMb: z.number().positive().optional(),
3282
+ reactionNotifications: z.enum([
3283
+ "off",
3284
+ "own",
3285
+ "all",
3286
+ "allowlist"
3287
+ ]).optional(),
3288
+ reactionAllowlist: z.array(z.union([z.string(), z.number()])).optional(),
3289
+ replyToMode: ReplyToModeSchema.optional(),
3290
+ replyToModeByChatType: SlackReplyToModeByChatTypeSchema.optional(),
3291
+ thread: SlackThreadSchema.optional(),
3292
+ actions: z.object({
3293
+ reactions: z.boolean().optional(),
3294
+ messages: z.boolean().optional(),
3295
+ pins: z.boolean().optional(),
3296
+ search: z.boolean().optional(),
3297
+ permissions: z.boolean().optional(),
3298
+ memberInfo: z.boolean().optional(),
3299
+ channelInfo: z.boolean().optional(),
3300
+ emojiList: z.boolean().optional()
3301
+ }).strict().optional(),
3302
+ slashCommand: z.object({
3303
+ enabled: z.boolean().optional(),
3304
+ name: z.string().optional(),
3305
+ sessionPrefix: z.string().optional(),
3306
+ ephemeral: z.boolean().optional()
3307
+ }).strict().optional(),
3308
+ dm: SlackDmSchema.optional(),
3309
+ channels: z.record(z.string(), SlackChannelSchema.optional()).optional(),
3310
+ heartbeat: ChannelHeartbeatVisibilitySchema
3311
+ }).strict();
3312
+ const SlackConfigSchema = SlackAccountSchema.extend({
3313
+ mode: z.enum(["socket", "http"]).optional().default("socket"),
3314
+ signingSecret: z.string().optional(),
3315
+ webhookPath: z.string().optional().default("/slack/events"),
3316
+ accounts: z.record(z.string(), SlackAccountSchema.optional()).optional()
3317
+ }).superRefine((value, ctx) => {
3318
+ const baseMode = value.mode ?? "socket";
3319
+ if (baseMode === "http" && !value.signingSecret) ctx.addIssue({
3320
+ code: z.ZodIssueCode.custom,
3321
+ message: "channels.slack.mode=\"http\" requires channels.slack.signingSecret",
3322
+ path: ["signingSecret"]
3323
+ });
3324
+ if (!value.accounts) return;
3325
+ for (const [accountId, account] of Object.entries(value.accounts)) {
3326
+ if (!account) continue;
3327
+ if (account.enabled === false) continue;
3328
+ if ((account.mode ?? baseMode) !== "http") continue;
3329
+ if (!(account.signingSecret ?? value.signingSecret)) ctx.addIssue({
3330
+ code: z.ZodIssueCode.custom,
3331
+ message: "channels.slack.accounts.*.mode=\"http\" requires channels.slack.signingSecret or channels.slack.accounts.*.signingSecret",
3332
+ path: [
3333
+ "accounts",
3334
+ accountId,
3335
+ "signingSecret"
3336
+ ]
3337
+ });
3338
+ }
3339
+ });
3340
+ const SignalAccountSchemaBase = z.object({
3341
+ name: z.string().optional(),
3342
+ capabilities: z.array(z.string()).optional(),
3343
+ markdown: MarkdownConfigSchema,
3344
+ enabled: z.boolean().optional(),
3345
+ configWrites: z.boolean().optional(),
3346
+ account: z.string().optional(),
3347
+ httpUrl: z.string().optional(),
3348
+ httpHost: z.string().optional(),
3349
+ httpPort: z.number().int().positive().optional(),
3350
+ cliPath: ExecutableTokenSchema.optional(),
3351
+ autoStart: z.boolean().optional(),
3352
+ startupTimeoutMs: z.number().int().min(1e3).max(12e4).optional(),
3353
+ receiveMode: z.union([z.literal("on-start"), z.literal("manual")]).optional(),
3354
+ ignoreAttachments: z.boolean().optional(),
3355
+ ignoreStories: z.boolean().optional(),
3356
+ sendReadReceipts: z.boolean().optional(),
3357
+ dmPolicy: DmPolicySchema.optional().default("pairing"),
3358
+ allowFrom: z.array(z.union([z.string(), z.number()])).optional(),
3359
+ groupAllowFrom: z.array(z.union([z.string(), z.number()])).optional(),
3360
+ groupPolicy: GroupPolicySchema.optional().default("allowlist"),
3361
+ historyLimit: z.number().int().min(0).optional(),
3362
+ dmHistoryLimit: z.number().int().min(0).optional(),
3363
+ dms: z.record(z.string(), DmConfigSchema.optional()).optional(),
3364
+ textChunkLimit: z.number().int().positive().optional(),
3365
+ chunkMode: z.enum(["length", "newline"]).optional(),
3366
+ blockStreaming: z.boolean().optional(),
3367
+ blockStreamingCoalesce: BlockStreamingCoalesceSchema.optional(),
3368
+ mediaMaxMb: z.number().int().positive().optional(),
3369
+ reactionNotifications: z.enum([
3370
+ "off",
3371
+ "own",
3372
+ "all",
3373
+ "allowlist"
3374
+ ]).optional(),
3375
+ reactionAllowlist: z.array(z.union([z.string(), z.number()])).optional(),
3376
+ actions: z.object({ reactions: z.boolean().optional() }).strict().optional(),
3377
+ reactionLevel: z.enum([
3378
+ "off",
3379
+ "ack",
3380
+ "minimal",
3381
+ "extensive"
3382
+ ]).optional(),
3383
+ heartbeat: ChannelHeartbeatVisibilitySchema
3384
+ }).strict();
3385
+ const SignalAccountSchema = SignalAccountSchemaBase.superRefine((value, ctx) => {
3386
+ requireOpenAllowFrom({
3387
+ policy: value.dmPolicy,
3388
+ allowFrom: value.allowFrom,
3389
+ ctx,
3390
+ path: ["allowFrom"],
3391
+ message: "channels.signal.dmPolicy=\"open\" requires channels.signal.allowFrom to include \"*\""
3392
+ });
3393
+ });
3394
+ const SignalConfigSchema = SignalAccountSchemaBase.extend({ accounts: z.record(z.string(), SignalAccountSchema.optional()).optional() }).superRefine((value, ctx) => {
3395
+ requireOpenAllowFrom({
3396
+ policy: value.dmPolicy,
3397
+ allowFrom: value.allowFrom,
3398
+ ctx,
3399
+ path: ["allowFrom"],
3400
+ message: "channels.signal.dmPolicy=\"open\" requires channels.signal.allowFrom to include \"*\""
3401
+ });
3402
+ });
3403
+ const IMessageAccountSchemaBase = z.object({
3404
+ name: z.string().optional(),
3405
+ capabilities: z.array(z.string()).optional(),
3406
+ markdown: MarkdownConfigSchema,
3407
+ enabled: z.boolean().optional(),
3408
+ configWrites: z.boolean().optional(),
3409
+ cliPath: ExecutableTokenSchema.optional(),
3410
+ dbPath: z.string().optional(),
3411
+ remoteHost: z.string().optional(),
3412
+ service: z.union([
3413
+ z.literal("imessage"),
3414
+ z.literal("sms"),
3415
+ z.literal("auto")
3416
+ ]).optional(),
3417
+ region: z.string().optional(),
3418
+ dmPolicy: DmPolicySchema.optional().default("pairing"),
3419
+ allowFrom: z.array(z.union([z.string(), z.number()])).optional(),
3420
+ groupAllowFrom: z.array(z.union([z.string(), z.number()])).optional(),
3421
+ groupPolicy: GroupPolicySchema.optional().default("allowlist"),
3422
+ historyLimit: z.number().int().min(0).optional(),
3423
+ dmHistoryLimit: z.number().int().min(0).optional(),
3424
+ dms: z.record(z.string(), DmConfigSchema.optional()).optional(),
3425
+ includeAttachments: z.boolean().optional(),
3426
+ mediaMaxMb: z.number().int().positive().optional(),
3427
+ textChunkLimit: z.number().int().positive().optional(),
3428
+ chunkMode: z.enum(["length", "newline"]).optional(),
3429
+ blockStreaming: z.boolean().optional(),
3430
+ blockStreamingCoalesce: BlockStreamingCoalesceSchema.optional(),
3431
+ groups: z.record(z.string(), z.object({
3432
+ requireMention: z.boolean().optional(),
3433
+ tools: ToolPolicySchema,
3434
+ toolsBySender: ToolPolicyBySenderSchema$1
3435
+ }).strict().optional()).optional(),
3436
+ heartbeat: ChannelHeartbeatVisibilitySchema
3437
+ }).strict();
3438
+ const IMessageAccountSchema = IMessageAccountSchemaBase.superRefine((value, ctx) => {
3439
+ requireOpenAllowFrom({
3440
+ policy: value.dmPolicy,
3441
+ allowFrom: value.allowFrom,
3442
+ ctx,
3443
+ path: ["allowFrom"],
3444
+ message: "channels.imessage.dmPolicy=\"open\" requires channels.imessage.allowFrom to include \"*\""
3445
+ });
3446
+ });
3447
+ const IMessageConfigSchema = IMessageAccountSchemaBase.extend({ accounts: z.record(z.string(), IMessageAccountSchema.optional()).optional() }).superRefine((value, ctx) => {
3448
+ requireOpenAllowFrom({
3449
+ policy: value.dmPolicy,
3450
+ allowFrom: value.allowFrom,
3451
+ ctx,
3452
+ path: ["allowFrom"],
3453
+ message: "channels.imessage.dmPolicy=\"open\" requires channels.imessage.allowFrom to include \"*\""
3454
+ });
3455
+ });
3456
+ const BlueBubblesAllowFromEntry = z.union([z.string(), z.number()]);
3457
+ const BlueBubblesActionSchema = z.object({
3458
+ reactions: z.boolean().optional(),
3459
+ edit: z.boolean().optional(),
3460
+ unsend: z.boolean().optional(),
3461
+ reply: z.boolean().optional(),
3462
+ sendWithEffect: z.boolean().optional(),
3463
+ renameGroup: z.boolean().optional(),
3464
+ setGroupIcon: z.boolean().optional(),
3465
+ addParticipant: z.boolean().optional(),
3466
+ removeParticipant: z.boolean().optional(),
3467
+ leaveGroup: z.boolean().optional(),
3468
+ sendAttachment: z.boolean().optional()
3469
+ }).strict().optional();
3470
+ const BlueBubblesGroupConfigSchema = z.object({
3471
+ requireMention: z.boolean().optional(),
3472
+ tools: ToolPolicySchema,
3473
+ toolsBySender: ToolPolicyBySenderSchema$1
3474
+ }).strict();
3475
+ const BlueBubblesAccountSchemaBase = z.object({
3476
+ name: z.string().optional(),
3477
+ capabilities: z.array(z.string()).optional(),
3478
+ markdown: MarkdownConfigSchema,
3479
+ configWrites: z.boolean().optional(),
3480
+ enabled: z.boolean().optional(),
3481
+ serverUrl: z.string().optional(),
3482
+ password: z.string().optional(),
3483
+ webhookPath: z.string().optional(),
3484
+ dmPolicy: DmPolicySchema.optional().default("pairing"),
3485
+ allowFrom: z.array(BlueBubblesAllowFromEntry).optional(),
3486
+ groupAllowFrom: z.array(BlueBubblesAllowFromEntry).optional(),
3487
+ groupPolicy: GroupPolicySchema.optional().default("allowlist"),
3488
+ historyLimit: z.number().int().min(0).optional(),
3489
+ dmHistoryLimit: z.number().int().min(0).optional(),
3490
+ dms: z.record(z.string(), DmConfigSchema.optional()).optional(),
3491
+ textChunkLimit: z.number().int().positive().optional(),
3492
+ chunkMode: z.enum(["length", "newline"]).optional(),
3493
+ mediaMaxMb: z.number().int().positive().optional(),
3494
+ sendReadReceipts: z.boolean().optional(),
3495
+ blockStreaming: z.boolean().optional(),
3496
+ blockStreamingCoalesce: BlockStreamingCoalesceSchema.optional(),
3497
+ groups: z.record(z.string(), BlueBubblesGroupConfigSchema.optional()).optional(),
3498
+ heartbeat: ChannelHeartbeatVisibilitySchema
3499
+ }).strict();
3500
+ const BlueBubblesAccountSchema = BlueBubblesAccountSchemaBase.superRefine((value, ctx) => {
3501
+ requireOpenAllowFrom({
3502
+ policy: value.dmPolicy,
3503
+ allowFrom: value.allowFrom,
3504
+ ctx,
3505
+ path: ["allowFrom"],
3506
+ message: "channels.bluebubbles.accounts.*.dmPolicy=\"open\" requires allowFrom to include \"*\""
3507
+ });
3508
+ });
3509
+ const BlueBubblesConfigSchema = BlueBubblesAccountSchemaBase.extend({
3510
+ accounts: z.record(z.string(), BlueBubblesAccountSchema.optional()).optional(),
3511
+ actions: BlueBubblesActionSchema
3512
+ }).superRefine((value, ctx) => {
3513
+ requireOpenAllowFrom({
3514
+ policy: value.dmPolicy,
3515
+ allowFrom: value.allowFrom,
3516
+ ctx,
3517
+ path: ["allowFrom"],
3518
+ message: "channels.bluebubbles.dmPolicy=\"open\" requires channels.bluebubbles.allowFrom to include \"*\""
3519
+ });
3520
+ });
3521
+ const MSTeamsChannelSchema = z.object({
3522
+ requireMention: z.boolean().optional(),
3523
+ tools: ToolPolicySchema,
3524
+ toolsBySender: ToolPolicyBySenderSchema$1,
3525
+ replyStyle: MSTeamsReplyStyleSchema.optional()
3526
+ }).strict();
3527
+ const MSTeamsTeamSchema = z.object({
3528
+ requireMention: z.boolean().optional(),
3529
+ tools: ToolPolicySchema,
3530
+ toolsBySender: ToolPolicyBySenderSchema$1,
3531
+ replyStyle: MSTeamsReplyStyleSchema.optional(),
3532
+ channels: z.record(z.string(), MSTeamsChannelSchema.optional()).optional()
3533
+ }).strict();
3534
+ const MSTeamsConfigSchema = z.object({
3535
+ enabled: z.boolean().optional(),
3536
+ capabilities: z.array(z.string()).optional(),
3537
+ markdown: MarkdownConfigSchema,
3538
+ configWrites: z.boolean().optional(),
3539
+ appId: z.string().optional(),
3540
+ appPassword: z.string().optional(),
3541
+ tenantId: z.string().optional(),
3542
+ webhook: z.object({
3543
+ port: z.number().int().positive().optional(),
3544
+ path: z.string().optional()
3545
+ }).strict().optional(),
3546
+ dmPolicy: DmPolicySchema.optional().default("pairing"),
3547
+ allowFrom: z.array(z.string()).optional(),
3548
+ groupAllowFrom: z.array(z.string()).optional(),
3549
+ groupPolicy: GroupPolicySchema.optional().default("allowlist"),
3550
+ textChunkLimit: z.number().int().positive().optional(),
3551
+ chunkMode: z.enum(["length", "newline"]).optional(),
3552
+ blockStreamingCoalesce: BlockStreamingCoalesceSchema.optional(),
3553
+ mediaAllowHosts: z.array(z.string()).optional(),
3554
+ mediaAuthAllowHosts: z.array(z.string()).optional(),
3555
+ requireMention: z.boolean().optional(),
3556
+ historyLimit: z.number().int().min(0).optional(),
3557
+ dmHistoryLimit: z.number().int().min(0).optional(),
3558
+ dms: z.record(z.string(), DmConfigSchema.optional()).optional(),
3559
+ replyStyle: MSTeamsReplyStyleSchema.optional(),
3560
+ teams: z.record(z.string(), MSTeamsTeamSchema.optional()).optional(),
3561
+ mediaMaxMb: z.number().positive().optional(),
3562
+ sharePointSiteId: z.string().optional(),
3563
+ heartbeat: ChannelHeartbeatVisibilitySchema
3564
+ }).strict().superRefine((value, ctx) => {
3565
+ requireOpenAllowFrom({
3566
+ policy: value.dmPolicy,
3567
+ allowFrom: value.allowFrom,
3568
+ ctx,
3569
+ path: ["allowFrom"],
3570
+ message: "channels.msteams.dmPolicy=\"open\" requires channels.msteams.allowFrom to include \"*\""
3571
+ });
3572
+ });
3573
+
3574
+ //#endregion
3575
+ //#region src/config/zod-schema.providers-whatsapp.ts
3576
+ const ToolPolicyBySenderSchema = z.record(z.string(), ToolPolicySchema).optional();
3577
+ const WhatsAppAccountSchema = z.object({
3578
+ name: z.string().optional(),
3579
+ capabilities: z.array(z.string()).optional(),
3580
+ markdown: MarkdownConfigSchema,
3581
+ configWrites: z.boolean().optional(),
3582
+ enabled: z.boolean().optional(),
3583
+ sendReadReceipts: z.boolean().optional(),
3584
+ messagePrefix: z.string().optional(),
3585
+ authDir: z.string().optional(),
3586
+ dmPolicy: DmPolicySchema.optional().default("pairing"),
3587
+ selfChatMode: z.boolean().optional(),
3588
+ allowFrom: z.array(z.string()).optional(),
3589
+ groupAllowFrom: z.array(z.string()).optional(),
3590
+ groupPolicy: GroupPolicySchema.optional().default("allowlist"),
3591
+ historyLimit: z.number().int().min(0).optional(),
3592
+ dmHistoryLimit: z.number().int().min(0).optional(),
3593
+ dms: z.record(z.string(), DmConfigSchema.optional()).optional(),
3594
+ textChunkLimit: z.number().int().positive().optional(),
3595
+ chunkMode: z.enum(["length", "newline"]).optional(),
3596
+ mediaMaxMb: z.number().int().positive().optional(),
3597
+ blockStreaming: z.boolean().optional(),
3598
+ blockStreamingCoalesce: BlockStreamingCoalesceSchema.optional(),
3599
+ groups: z.record(z.string(), z.object({
3600
+ requireMention: z.boolean().optional(),
3601
+ tools: ToolPolicySchema,
3602
+ toolsBySender: ToolPolicyBySenderSchema
3603
+ }).strict().optional()).optional(),
3604
+ ackReaction: z.object({
3605
+ emoji: z.string().optional(),
3606
+ direct: z.boolean().optional().default(true),
3607
+ group: z.enum([
3608
+ "always",
3609
+ "mentions",
3610
+ "never"
3611
+ ]).optional().default("mentions")
3612
+ }).strict().optional(),
3613
+ debounceMs: z.number().int().nonnegative().optional().default(0),
3614
+ heartbeat: ChannelHeartbeatVisibilitySchema
3615
+ }).strict().superRefine((value, ctx) => {
3616
+ if (value.dmPolicy !== "open") return;
3617
+ if ((value.allowFrom ?? []).map((v) => String(v).trim()).filter(Boolean).includes("*")) return;
3618
+ ctx.addIssue({
3619
+ code: z.ZodIssueCode.custom,
3620
+ path: ["allowFrom"],
3621
+ message: "channels.whatsapp.accounts.*.dmPolicy=\"open\" requires allowFrom to include \"*\""
3622
+ });
3623
+ });
3624
+ const WhatsAppConfigSchema = z.object({
3625
+ accounts: z.record(z.string(), WhatsAppAccountSchema.optional()).optional(),
3626
+ capabilities: z.array(z.string()).optional(),
3627
+ markdown: MarkdownConfigSchema,
3628
+ configWrites: z.boolean().optional(),
3629
+ sendReadReceipts: z.boolean().optional(),
3630
+ dmPolicy: DmPolicySchema.optional().default("pairing"),
3631
+ messagePrefix: z.string().optional(),
3632
+ selfChatMode: z.boolean().optional(),
3633
+ allowFrom: z.array(z.string()).optional(),
3634
+ groupAllowFrom: z.array(z.string()).optional(),
3635
+ groupPolicy: GroupPolicySchema.optional().default("allowlist"),
3636
+ historyLimit: z.number().int().min(0).optional(),
3637
+ dmHistoryLimit: z.number().int().min(0).optional(),
3638
+ dms: z.record(z.string(), DmConfigSchema.optional()).optional(),
3639
+ textChunkLimit: z.number().int().positive().optional(),
3640
+ chunkMode: z.enum(["length", "newline"]).optional(),
3641
+ mediaMaxMb: z.number().int().positive().optional().default(50),
3642
+ blockStreaming: z.boolean().optional(),
3643
+ blockStreamingCoalesce: BlockStreamingCoalesceSchema.optional(),
3644
+ actions: z.object({
3645
+ reactions: z.boolean().optional(),
3646
+ sendMessage: z.boolean().optional(),
3647
+ polls: z.boolean().optional()
3648
+ }).strict().optional(),
3649
+ groups: z.record(z.string(), z.object({
3650
+ requireMention: z.boolean().optional(),
3651
+ tools: ToolPolicySchema,
3652
+ toolsBySender: ToolPolicyBySenderSchema
3653
+ }).strict().optional()).optional(),
3654
+ ackReaction: z.object({
3655
+ emoji: z.string().optional(),
3656
+ direct: z.boolean().optional().default(true),
3657
+ group: z.enum([
3658
+ "always",
3659
+ "mentions",
3660
+ "never"
3661
+ ]).optional().default("mentions")
3662
+ }).strict().optional(),
3663
+ debounceMs: z.number().int().nonnegative().optional().default(0),
3664
+ heartbeat: ChannelHeartbeatVisibilitySchema
3665
+ }).strict().superRefine((value, ctx) => {
3666
+ if (value.dmPolicy !== "open") return;
3667
+ if ((value.allowFrom ?? []).map((v) => String(v).trim()).filter(Boolean).includes("*")) return;
3668
+ ctx.addIssue({
3669
+ code: z.ZodIssueCode.custom,
3670
+ path: ["allowFrom"],
3671
+ message: "channels.whatsapp.dmPolicy=\"open\" requires channels.whatsapp.allowFrom to include \"*\""
3672
+ });
3673
+ });
3674
+
3675
+ //#endregion
3676
+ //#region src/config/zod-schema.providers.ts
3677
+ const ChannelsSchema = z.object({
3678
+ defaults: z.object({
3679
+ groupPolicy: GroupPolicySchema.optional(),
3680
+ heartbeat: ChannelHeartbeatVisibilitySchema
3681
+ }).strict().optional(),
3682
+ whatsapp: WhatsAppConfigSchema.optional(),
3683
+ telegram: TelegramConfigSchema.optional(),
3684
+ discord: DiscordConfigSchema.optional(),
3685
+ googlechat: GoogleChatConfigSchema.optional(),
3686
+ slack: SlackConfigSchema.optional(),
3687
+ signal: SignalConfigSchema.optional(),
3688
+ imessage: IMessageConfigSchema.optional(),
3689
+ bluebubbles: BlueBubblesConfigSchema.optional(),
3690
+ msteams: MSTeamsConfigSchema.optional()
3691
+ }).passthrough().optional();
3692
+
3693
+ //#endregion
3694
+ //#region src/config/zod-schema.session.ts
3695
+ const SessionResetConfigSchema = z.object({
3696
+ mode: z.union([z.literal("daily"), z.literal("idle")]).optional(),
3697
+ atHour: z.number().int().min(0).max(23).optional(),
3698
+ idleMinutes: z.number().int().positive().optional()
3699
+ }).strict();
3700
+ const SessionSendPolicySchema = z.object({
3701
+ default: z.union([z.literal("allow"), z.literal("deny")]).optional(),
3702
+ rules: z.array(z.object({
3703
+ action: z.union([z.literal("allow"), z.literal("deny")]),
3704
+ match: z.object({
3705
+ channel: z.string().optional(),
3706
+ chatType: z.union([
3707
+ z.literal("direct"),
3708
+ z.literal("group"),
3709
+ z.literal("channel")
3710
+ ]).optional(),
3711
+ keyPrefix: z.string().optional()
3712
+ }).strict().optional()
3713
+ }).strict()).optional()
3714
+ }).strict();
3715
+ const SessionSchema = z.object({
3716
+ scope: z.union([z.literal("per-sender"), z.literal("global")]).optional(),
3717
+ dmScope: z.union([
3718
+ z.literal("main"),
3719
+ z.literal("per-peer"),
3720
+ z.literal("per-channel-peer"),
3721
+ z.literal("per-account-channel-peer")
3722
+ ]).optional(),
3723
+ identityLinks: z.record(z.string(), z.array(z.string())).optional(),
3724
+ resetTriggers: z.array(z.string()).optional(),
3725
+ idleMinutes: z.number().int().positive().optional(),
3726
+ reset: SessionResetConfigSchema.optional(),
3727
+ resetByType: z.object({
3728
+ dm: SessionResetConfigSchema.optional(),
3729
+ group: SessionResetConfigSchema.optional(),
3730
+ thread: SessionResetConfigSchema.optional()
3731
+ }).strict().optional(),
3732
+ resetByChannel: z.record(z.string(), SessionResetConfigSchema).optional(),
3733
+ store: z.string().optional(),
3734
+ typingIntervalSeconds: z.number().int().positive().optional(),
3735
+ typingMode: z.union([
3736
+ z.literal("never"),
3737
+ z.literal("instant"),
3738
+ z.literal("thinking"),
3739
+ z.literal("message")
3740
+ ]).optional(),
3741
+ mainKey: z.string().optional(),
3742
+ sendPolicy: SessionSendPolicySchema.optional(),
3743
+ agentToAgent: z.object({ maxPingPongTurns: z.number().int().min(0).max(5).optional() }).strict().optional()
3744
+ }).strict().optional();
3745
+ const MessagesSchema = z.object({
3746
+ messagePrefix: z.string().optional(),
3747
+ responsePrefix: z.string().optional(),
3748
+ groupChat: GroupChatSchema,
3749
+ queue: QueueSchema,
3750
+ inbound: InboundDebounceSchema,
3751
+ ackReaction: z.string().optional(),
3752
+ ackReactionScope: z.enum([
3753
+ "group-mentions",
3754
+ "group-all",
3755
+ "direct",
3756
+ "all"
3757
+ ]).optional(),
3758
+ removeAckAfterReply: z.boolean().optional(),
3759
+ tts: TtsConfigSchema
3760
+ }).strict().optional();
3761
+ const CommandsSchema = z.object({
3762
+ native: NativeCommandsSettingSchema.optional().default("auto"),
3763
+ nativeSkills: NativeCommandsSettingSchema.optional().default("auto"),
3764
+ text: z.boolean().optional(),
3765
+ bash: z.boolean().optional(),
3766
+ bashForegroundMs: z.number().int().min(0).max(3e4).optional(),
3767
+ config: z.boolean().optional(),
3768
+ debug: z.boolean().optional(),
3769
+ restart: z.boolean().optional(),
3770
+ useAccessGroups: z.boolean().optional()
3771
+ }).strict().optional().default({
3772
+ native: "auto",
3773
+ nativeSkills: "auto"
3774
+ });
3775
+
3776
+ //#endregion
3777
+ //#region src/config/zod-schema.ts
3778
+ const BrowserSnapshotDefaultsSchema = z.object({ mode: z.literal("efficient").optional() }).strict().optional();
3779
+ const NodeHostSchema = z.object({ browserProxy: z.object({
3780
+ enabled: z.boolean().optional(),
3781
+ allowProfiles: z.array(z.string()).optional()
3782
+ }).strict().optional() }).strict().optional();
3783
+ const MemoryQmdPathSchema = z.object({
3784
+ path: z.string(),
3785
+ name: z.string().optional(),
3786
+ pattern: z.string().optional()
3787
+ }).strict();
3788
+ const MemoryQmdSessionSchema = z.object({
3789
+ enabled: z.boolean().optional(),
3790
+ exportDir: z.string().optional(),
3791
+ retentionDays: z.number().int().nonnegative().optional()
3792
+ }).strict();
3793
+ const MemoryQmdUpdateSchema = z.object({
3794
+ interval: z.string().optional(),
3795
+ debounceMs: z.number().int().nonnegative().optional(),
3796
+ onBoot: z.boolean().optional(),
3797
+ embedInterval: z.string().optional()
3798
+ }).strict();
3799
+ const MemoryQmdLimitsSchema = z.object({
3800
+ maxResults: z.number().int().positive().optional(),
3801
+ maxSnippetChars: z.number().int().positive().optional(),
3802
+ maxInjectedChars: z.number().int().positive().optional(),
3803
+ timeoutMs: z.number().int().nonnegative().optional()
3804
+ }).strict();
3805
+ const MemoryQmdSchema = z.object({
3806
+ command: z.string().optional(),
3807
+ includeDefaultMemory: z.boolean().optional(),
3808
+ paths: z.array(MemoryQmdPathSchema).optional(),
3809
+ sessions: MemoryQmdSessionSchema.optional(),
3810
+ update: MemoryQmdUpdateSchema.optional(),
3811
+ limits: MemoryQmdLimitsSchema.optional(),
3812
+ scope: SessionSendPolicySchema.optional()
3813
+ }).strict();
3814
+ const MemorySchema = z.object({
3815
+ backend: z.union([z.literal("builtin"), z.literal("qmd")]).optional(),
3816
+ citations: z.union([
3817
+ z.literal("auto"),
3818
+ z.literal("on"),
3819
+ z.literal("off")
3820
+ ]).optional(),
3821
+ qmd: MemoryQmdSchema.optional()
3822
+ }).strict().optional();
3823
+ const OpenClawSchema = z.object({
3824
+ meta: z.object({
3825
+ lastTouchedVersion: z.string().optional(),
3826
+ lastTouchedAt: z.string().optional()
3827
+ }).strict().optional(),
3828
+ env: z.object({
3829
+ shellEnv: z.object({
3830
+ enabled: z.boolean().optional(),
3831
+ timeoutMs: z.number().int().nonnegative().optional()
3832
+ }).strict().optional(),
3833
+ vars: z.record(z.string(), z.string()).optional()
3834
+ }).catchall(z.string()).optional(),
3835
+ wizard: z.object({
3836
+ lastRunAt: z.string().optional(),
3837
+ lastRunVersion: z.string().optional(),
3838
+ lastRunCommit: z.string().optional(),
3839
+ lastRunCommand: z.string().optional(),
3840
+ lastRunMode: z.union([z.literal("local"), z.literal("remote")]).optional()
3841
+ }).strict().optional(),
3842
+ diagnostics: z.object({
3843
+ enabled: z.boolean().optional(),
3844
+ flags: z.array(z.string()).optional(),
3845
+ otel: z.object({
3846
+ enabled: z.boolean().optional(),
3847
+ endpoint: z.string().optional(),
3848
+ protocol: z.union([z.literal("http/protobuf"), z.literal("grpc")]).optional(),
3849
+ headers: z.record(z.string(), z.string()).optional(),
3850
+ serviceName: z.string().optional(),
3851
+ traces: z.boolean().optional(),
3852
+ metrics: z.boolean().optional(),
3853
+ logs: z.boolean().optional(),
3854
+ sampleRate: z.number().min(0).max(1).optional(),
3855
+ flushIntervalMs: z.number().int().nonnegative().optional()
3856
+ }).strict().optional(),
3857
+ cacheTrace: z.object({
3858
+ enabled: z.boolean().optional(),
3859
+ filePath: z.string().optional(),
3860
+ includeMessages: z.boolean().optional(),
3861
+ includePrompt: z.boolean().optional(),
3862
+ includeSystem: z.boolean().optional()
3863
+ }).strict().optional()
3864
+ }).strict().optional(),
3865
+ logging: z.object({
3866
+ level: z.union([
3867
+ z.literal("silent"),
3868
+ z.literal("fatal"),
3869
+ z.literal("error"),
3870
+ z.literal("warn"),
3871
+ z.literal("info"),
3872
+ z.literal("debug"),
3873
+ z.literal("trace")
3874
+ ]).optional(),
3875
+ file: z.string().optional(),
3876
+ consoleLevel: z.union([
3877
+ z.literal("silent"),
3878
+ z.literal("fatal"),
3879
+ z.literal("error"),
3880
+ z.literal("warn"),
3881
+ z.literal("info"),
3882
+ z.literal("debug"),
3883
+ z.literal("trace")
3884
+ ]).optional(),
3885
+ consoleStyle: z.union([
3886
+ z.literal("pretty"),
3887
+ z.literal("compact"),
3888
+ z.literal("json")
3889
+ ]).optional(),
3890
+ redactSensitive: z.union([z.literal("off"), z.literal("tools")]).optional(),
3891
+ redactPatterns: z.array(z.string()).optional()
3892
+ }).strict().optional(),
3893
+ update: z.object({
3894
+ channel: z.union([
3895
+ z.literal("stable"),
3896
+ z.literal("beta"),
3897
+ z.literal("dev")
3898
+ ]).optional(),
3899
+ checkOnStart: z.boolean().optional()
3900
+ }).strict().optional(),
3901
+ browser: z.object({
3902
+ enabled: z.boolean().optional(),
3903
+ evaluateEnabled: z.boolean().optional(),
3904
+ cdpUrl: z.string().optional(),
3905
+ remoteCdpTimeoutMs: z.number().int().nonnegative().optional(),
3906
+ remoteCdpHandshakeTimeoutMs: z.number().int().nonnegative().optional(),
3907
+ color: z.string().optional(),
3908
+ executablePath: z.string().optional(),
3909
+ headless: z.boolean().optional(),
3910
+ noSandbox: z.boolean().optional(),
3911
+ attachOnly: z.boolean().optional(),
3912
+ defaultProfile: z.string().optional(),
3913
+ snapshotDefaults: BrowserSnapshotDefaultsSchema,
3914
+ profiles: z.record(z.string().regex(/^[a-z0-9-]+$/, "Profile names must be alphanumeric with hyphens only"), z.object({
3915
+ cdpPort: z.number().int().min(1).max(65535).optional(),
3916
+ cdpUrl: z.string().optional(),
3917
+ driver: z.union([z.literal("clawd"), z.literal("extension")]).optional(),
3918
+ color: HexColorSchema
3919
+ }).strict().refine((value) => value.cdpPort || value.cdpUrl, { message: "Profile must set cdpPort or cdpUrl" })).optional()
3920
+ }).strict().optional(),
3921
+ ui: z.object({
3922
+ seamColor: HexColorSchema.optional(),
3923
+ assistant: z.object({
3924
+ name: z.string().max(50).optional(),
3925
+ avatar: z.string().max(200).optional()
3926
+ }).strict().optional()
3927
+ }).strict().optional(),
3928
+ auth: z.object({
3929
+ profiles: z.record(z.string(), z.object({
3930
+ provider: z.string(),
3931
+ mode: z.union([
3932
+ z.literal("api_key"),
3933
+ z.literal("oauth"),
3934
+ z.literal("token")
3935
+ ]),
3936
+ email: z.string().optional()
3937
+ }).strict()).optional(),
3938
+ order: z.record(z.string(), z.array(z.string())).optional(),
3939
+ cooldowns: z.object({
3940
+ billingBackoffHours: z.number().positive().optional(),
3941
+ billingBackoffHoursByProvider: z.record(z.string(), z.number().positive()).optional(),
3942
+ billingMaxHours: z.number().positive().optional(),
3943
+ failureWindowHours: z.number().positive().optional()
3944
+ }).strict().optional()
3945
+ }).strict().optional(),
3946
+ models: ModelsConfigSchema,
3947
+ nodeHost: NodeHostSchema,
3948
+ agents: AgentsSchema,
3949
+ tools: ToolsSchema,
3950
+ bindings: BindingsSchema,
3951
+ broadcast: BroadcastSchema,
3952
+ audio: AudioSchema,
3953
+ media: z.object({ preserveFilenames: z.boolean().optional() }).strict().optional(),
3954
+ messages: MessagesSchema,
3955
+ commands: CommandsSchema,
3956
+ approvals: ApprovalsSchema,
3957
+ session: SessionSchema,
3958
+ cron: z.object({
3959
+ enabled: z.boolean().optional(),
3960
+ store: z.string().optional(),
3961
+ maxConcurrentRuns: z.number().int().positive().optional()
3962
+ }).strict().optional(),
3963
+ hooks: z.object({
3964
+ enabled: z.boolean().optional(),
3965
+ path: z.string().optional(),
3966
+ token: z.string().optional(),
3967
+ maxBodyBytes: z.number().int().positive().optional(),
3968
+ presets: z.array(z.string()).optional(),
3969
+ transformsDir: z.string().optional(),
3970
+ mappings: z.array(HookMappingSchema).optional(),
3971
+ gmail: HooksGmailSchema,
3972
+ internal: InternalHooksSchema
3973
+ }).strict().optional(),
3974
+ web: z.object({
3975
+ enabled: z.boolean().optional(),
3976
+ heartbeatSeconds: z.number().int().positive().optional(),
3977
+ reconnect: z.object({
3978
+ initialMs: z.number().positive().optional(),
3979
+ maxMs: z.number().positive().optional(),
3980
+ factor: z.number().positive().optional(),
3981
+ jitter: z.number().min(0).max(1).optional(),
3982
+ maxAttempts: z.number().int().min(0).optional()
3983
+ }).strict().optional()
3984
+ }).strict().optional(),
3985
+ channels: ChannelsSchema,
3986
+ discovery: z.object({
3987
+ wideArea: z.object({ enabled: z.boolean().optional() }).strict().optional(),
3988
+ mdns: z.object({ mode: z.enum([
3989
+ "off",
3990
+ "minimal",
3991
+ "full"
3992
+ ]).optional() }).strict().optional()
3993
+ }).strict().optional(),
3994
+ canvasHost: z.object({
3995
+ enabled: z.boolean().optional(),
3996
+ root: z.string().optional(),
3997
+ port: z.number().int().positive().optional(),
3998
+ liveReload: z.boolean().optional()
3999
+ }).strict().optional(),
4000
+ talk: z.object({
4001
+ voiceId: z.string().optional(),
4002
+ voiceAliases: z.record(z.string(), z.string()).optional(),
4003
+ modelId: z.string().optional(),
4004
+ outputFormat: z.string().optional(),
4005
+ apiKey: z.string().optional(),
4006
+ interruptOnSpeech: z.boolean().optional()
4007
+ }).strict().optional(),
4008
+ gateway: z.object({
4009
+ port: z.number().int().positive().optional(),
4010
+ mode: z.union([z.literal("local"), z.literal("remote")]).optional(),
4011
+ bind: z.union([
4012
+ z.literal("auto"),
4013
+ z.literal("lan"),
4014
+ z.literal("loopback"),
4015
+ z.literal("custom"),
4016
+ z.literal("tailnet")
4017
+ ]).optional(),
4018
+ controlUi: z.object({
4019
+ enabled: z.boolean().optional(),
4020
+ basePath: z.string().optional(),
4021
+ root: z.string().optional(),
4022
+ allowedOrigins: z.array(z.string()).optional(),
4023
+ allowInsecureAuth: z.boolean().optional(),
4024
+ dangerouslyDisableDeviceAuth: z.boolean().optional()
4025
+ }).strict().optional(),
4026
+ auth: z.object({
4027
+ mode: z.union([z.literal("token"), z.literal("password")]).optional(),
4028
+ token: z.string().optional(),
4029
+ password: z.string().optional(),
4030
+ allowTailscale: z.boolean().optional()
4031
+ }).strict().optional(),
4032
+ trustedProxies: z.array(z.string()).optional(),
4033
+ tailscale: z.object({
4034
+ mode: z.union([
4035
+ z.literal("off"),
4036
+ z.literal("serve"),
4037
+ z.literal("funnel")
4038
+ ]).optional(),
4039
+ resetOnExit: z.boolean().optional()
4040
+ }).strict().optional(),
4041
+ remote: z.object({
4042
+ url: z.string().optional(),
4043
+ transport: z.union([z.literal("ssh"), z.literal("direct")]).optional(),
4044
+ token: z.string().optional(),
4045
+ password: z.string().optional(),
4046
+ tlsFingerprint: z.string().optional(),
4047
+ sshTarget: z.string().optional(),
4048
+ sshIdentity: z.string().optional()
4049
+ }).strict().optional(),
4050
+ reload: z.object({
4051
+ mode: z.union([
4052
+ z.literal("off"),
4053
+ z.literal("restart"),
4054
+ z.literal("hot"),
4055
+ z.literal("hybrid")
4056
+ ]).optional(),
4057
+ debounceMs: z.number().int().min(0).optional()
4058
+ }).strict().optional(),
4059
+ tls: z.object({
4060
+ enabled: z.boolean().optional(),
4061
+ autoGenerate: z.boolean().optional(),
4062
+ certPath: z.string().optional(),
4063
+ keyPath: z.string().optional(),
4064
+ caPath: z.string().optional()
4065
+ }).optional(),
4066
+ http: z.object({ endpoints: z.object({
4067
+ chatCompletions: z.object({ enabled: z.boolean().optional() }).strict().optional(),
4068
+ responses: z.object({
4069
+ enabled: z.boolean().optional(),
4070
+ maxBodyBytes: z.number().int().positive().optional(),
4071
+ files: z.object({
4072
+ allowUrl: z.boolean().optional(),
4073
+ allowedMimes: z.array(z.string()).optional(),
4074
+ maxBytes: z.number().int().positive().optional(),
4075
+ maxChars: z.number().int().positive().optional(),
4076
+ maxRedirects: z.number().int().nonnegative().optional(),
4077
+ timeoutMs: z.number().int().positive().optional(),
4078
+ pdf: z.object({
4079
+ maxPages: z.number().int().positive().optional(),
4080
+ maxPixels: z.number().int().positive().optional(),
4081
+ minTextChars: z.number().int().nonnegative().optional()
4082
+ }).strict().optional()
4083
+ }).strict().optional(),
4084
+ images: z.object({
4085
+ allowUrl: z.boolean().optional(),
4086
+ allowedMimes: z.array(z.string()).optional(),
4087
+ maxBytes: z.number().int().positive().optional(),
4088
+ maxRedirects: z.number().int().nonnegative().optional(),
4089
+ timeoutMs: z.number().int().positive().optional()
4090
+ }).strict().optional()
4091
+ }).strict().optional()
4092
+ }).strict().optional() }).strict().optional(),
4093
+ nodes: z.object({
4094
+ browser: z.object({
4095
+ mode: z.union([
4096
+ z.literal("auto"),
4097
+ z.literal("manual"),
4098
+ z.literal("off")
4099
+ ]).optional(),
4100
+ node: z.string().optional()
4101
+ }).strict().optional(),
4102
+ allowCommands: z.array(z.string()).optional(),
4103
+ denyCommands: z.array(z.string()).optional()
4104
+ }).strict().optional()
4105
+ }).strict().optional(),
4106
+ memory: MemorySchema,
4107
+ skills: z.object({
4108
+ allowBundled: z.array(z.string()).optional(),
4109
+ load: z.object({
4110
+ extraDirs: z.array(z.string()).optional(),
4111
+ watch: z.boolean().optional(),
4112
+ watchDebounceMs: z.number().int().min(0).optional()
4113
+ }).strict().optional(),
4114
+ install: z.object({
4115
+ preferBrew: z.boolean().optional(),
4116
+ nodeManager: z.union([
4117
+ z.literal("npm"),
4118
+ z.literal("pnpm"),
4119
+ z.literal("yarn"),
4120
+ z.literal("bun")
4121
+ ]).optional()
4122
+ }).strict().optional(),
4123
+ entries: z.record(z.string(), z.object({
4124
+ enabled: z.boolean().optional(),
4125
+ apiKey: z.string().optional(),
4126
+ env: z.record(z.string(), z.string()).optional(),
4127
+ config: z.record(z.string(), z.unknown()).optional()
4128
+ }).strict()).optional()
4129
+ }).strict().optional(),
4130
+ plugins: z.object({
4131
+ enabled: z.boolean().optional(),
4132
+ allow: z.array(z.string()).optional(),
4133
+ deny: z.array(z.string()).optional(),
4134
+ load: z.object({ paths: z.array(z.string()).optional() }).strict().optional(),
4135
+ slots: z.object({ memory: z.string().optional() }).strict().optional(),
4136
+ entries: z.record(z.string(), z.object({
4137
+ enabled: z.boolean().optional(),
4138
+ config: z.record(z.string(), z.unknown()).optional()
4139
+ }).strict()).optional(),
4140
+ installs: z.record(z.string(), z.object({
4141
+ source: z.union([
4142
+ z.literal("npm"),
4143
+ z.literal("archive"),
4144
+ z.literal("path")
4145
+ ]),
4146
+ spec: z.string().optional(),
4147
+ sourcePath: z.string().optional(),
4148
+ installPath: z.string().optional(),
4149
+ version: z.string().optional(),
4150
+ installedAt: z.string().optional()
4151
+ }).strict()).optional()
4152
+ }).strict().optional()
4153
+ }).strict().superRefine((cfg, ctx) => {
4154
+ const agents = cfg.agents?.list ?? [];
4155
+ if (agents.length === 0) return;
4156
+ const agentIds = new Set(agents.map((agent) => agent.id));
4157
+ const broadcast = cfg.broadcast;
4158
+ if (!broadcast) return;
4159
+ for (const [peerId, ids] of Object.entries(broadcast)) {
4160
+ if (peerId === "strategy") continue;
4161
+ if (!Array.isArray(ids)) continue;
4162
+ for (let idx = 0; idx < ids.length; idx += 1) {
4163
+ const agentId = ids[idx];
4164
+ if (!agentIds.has(agentId)) ctx.addIssue({
4165
+ code: z.ZodIssueCode.custom,
4166
+ path: [
4167
+ "broadcast",
4168
+ peerId,
4169
+ idx
4170
+ ],
4171
+ message: `Unknown agent id "${agentId}" (not in agents.list).`
4172
+ });
4173
+ }
4174
+ }
4175
+ });
4176
+
4177
+ //#endregion
4178
+ //#region src/config/validation.ts
4179
+ const AVATAR_SCHEME_RE = /^[a-z][a-z0-9+.-]*:/i;
4180
+ const AVATAR_DATA_RE = /^data:/i;
4181
+ const AVATAR_HTTP_RE = /^https?:\/\//i;
4182
+ const WINDOWS_ABS_RE = /^[a-zA-Z]:[\\/]/;
4183
+ function isWorkspaceAvatarPath(value, workspaceDir) {
4184
+ const workspaceRoot = path.resolve(workspaceDir);
4185
+ const resolved = path.resolve(workspaceRoot, value);
4186
+ const relative = path.relative(workspaceRoot, resolved);
4187
+ if (relative === "") return true;
4188
+ if (relative.startsWith("..")) return false;
4189
+ return !path.isAbsolute(relative);
4190
+ }
4191
+ function validateIdentityAvatar(config) {
4192
+ const agents = config.agents?.list;
4193
+ if (!Array.isArray(agents) || agents.length === 0) return [];
4194
+ const issues = [];
4195
+ for (const [index, entry] of agents.entries()) {
4196
+ if (!entry || typeof entry !== "object") continue;
4197
+ const avatarRaw = entry.identity?.avatar;
4198
+ if (typeof avatarRaw !== "string") continue;
4199
+ const avatar = avatarRaw.trim();
4200
+ if (!avatar) continue;
4201
+ if (AVATAR_DATA_RE.test(avatar) || AVATAR_HTTP_RE.test(avatar)) continue;
4202
+ if (avatar.startsWith("~")) {
4203
+ issues.push({
4204
+ path: `agents.list.${index}.identity.avatar`,
4205
+ message: "identity.avatar must be a workspace-relative path, http(s) URL, or data URI."
4206
+ });
4207
+ continue;
4208
+ }
4209
+ if (AVATAR_SCHEME_RE.test(avatar) && !WINDOWS_ABS_RE.test(avatar)) {
4210
+ issues.push({
4211
+ path: `agents.list.${index}.identity.avatar`,
4212
+ message: "identity.avatar must be a workspace-relative path, http(s) URL, or data URI."
4213
+ });
4214
+ continue;
4215
+ }
4216
+ if (!isWorkspaceAvatarPath(avatar, resolveAgentWorkspaceDir(config, entry.id ?? resolveDefaultAgentId(config)))) issues.push({
4217
+ path: `agents.list.${index}.identity.avatar`,
4218
+ message: "identity.avatar must stay within the agent workspace."
4219
+ });
4220
+ }
4221
+ return issues;
4222
+ }
4223
+ function validateConfigObject(raw) {
4224
+ const legacyIssues = findLegacyConfigIssues(raw);
4225
+ if (legacyIssues.length > 0) return {
4226
+ ok: false,
4227
+ issues: legacyIssues.map((iss) => ({
4228
+ path: iss.path,
4229
+ message: iss.message
4230
+ }))
4231
+ };
4232
+ const validated = OpenClawSchema.safeParse(raw);
4233
+ if (!validated.success) return {
4234
+ ok: false,
4235
+ issues: validated.error.issues.map((iss) => ({
4236
+ path: iss.path.join("."),
4237
+ message: iss.message
4238
+ }))
4239
+ };
4240
+ const duplicates = findDuplicateAgentDirs(validated.data);
4241
+ if (duplicates.length > 0) return {
4242
+ ok: false,
4243
+ issues: [{
4244
+ path: "agents.list",
4245
+ message: formatDuplicateAgentDirError(duplicates)
4246
+ }]
4247
+ };
4248
+ const avatarIssues = validateIdentityAvatar(validated.data);
4249
+ if (avatarIssues.length > 0) return {
4250
+ ok: false,
4251
+ issues: avatarIssues
4252
+ };
4253
+ return {
4254
+ ok: true,
4255
+ config: applyModelDefaults(applyAgentDefaults(applySessionDefaults(validated.data)))
4256
+ };
4257
+ }
4258
+ function isRecord(value) {
4259
+ return Boolean(value && typeof value === "object" && !Array.isArray(value));
4260
+ }
4261
+ function validateConfigObjectWithPlugins(raw) {
4262
+ const base = validateConfigObject(raw);
4263
+ if (!base.ok) return {
4264
+ ok: false,
4265
+ issues: base.issues,
4266
+ warnings: []
4267
+ };
4268
+ const config = base.config;
4269
+ const issues = [];
4270
+ const warnings = [];
4271
+ const pluginsConfig = config.plugins;
4272
+ const normalizedPlugins = normalizePluginsConfig(pluginsConfig);
4273
+ const registry = loadPluginManifestRegistry({
4274
+ config,
4275
+ workspaceDir: resolveAgentWorkspaceDir(config, resolveDefaultAgentId(config)) ?? void 0
4276
+ });
4277
+ const knownIds = new Set(registry.plugins.map((record) => record.id));
4278
+ for (const diag of registry.diagnostics) {
4279
+ let path = diag.pluginId ? `plugins.entries.${diag.pluginId}` : "plugins";
4280
+ if (!diag.pluginId && diag.message.includes("plugin path not found")) path = "plugins.load.paths";
4281
+ const message = `${diag.pluginId ? `plugin ${diag.pluginId}` : "plugin"}: ${diag.message}`;
4282
+ if (diag.level === "error") issues.push({
4283
+ path,
4284
+ message
4285
+ });
4286
+ else warnings.push({
4287
+ path,
4288
+ message
4289
+ });
4290
+ }
4291
+ const entries = pluginsConfig?.entries;
4292
+ if (entries && isRecord(entries)) {
4293
+ for (const pluginId of Object.keys(entries)) if (!knownIds.has(pluginId)) issues.push({
4294
+ path: `plugins.entries.${pluginId}`,
4295
+ message: `plugin not found: ${pluginId}`
4296
+ });
4297
+ }
4298
+ const allow = pluginsConfig?.allow ?? [];
4299
+ for (const pluginId of allow) {
4300
+ if (typeof pluginId !== "string" || !pluginId.trim()) continue;
4301
+ if (!knownIds.has(pluginId)) issues.push({
4302
+ path: "plugins.allow",
4303
+ message: `plugin not found: ${pluginId}`
4304
+ });
4305
+ }
4306
+ const deny = pluginsConfig?.deny ?? [];
4307
+ for (const pluginId of deny) {
4308
+ if (typeof pluginId !== "string" || !pluginId.trim()) continue;
4309
+ if (!knownIds.has(pluginId)) issues.push({
4310
+ path: "plugins.deny",
4311
+ message: `plugin not found: ${pluginId}`
4312
+ });
4313
+ }
4314
+ const memorySlot = normalizedPlugins.slots.memory;
4315
+ if (typeof memorySlot === "string" && memorySlot.trim() && !knownIds.has(memorySlot)) issues.push({
4316
+ path: "plugins.slots.memory",
4317
+ message: `plugin not found: ${memorySlot}`
4318
+ });
4319
+ const allowedChannels = new Set(["defaults", ...CHANNEL_IDS]);
4320
+ for (const record of registry.plugins) for (const channelId of record.channels) allowedChannels.add(channelId);
4321
+ if (config.channels && isRecord(config.channels)) for (const key of Object.keys(config.channels)) {
4322
+ const trimmed = key.trim();
4323
+ if (!trimmed) continue;
4324
+ if (!allowedChannels.has(trimmed)) issues.push({
4325
+ path: `channels.${trimmed}`,
4326
+ message: `unknown channel id: ${trimmed}`
4327
+ });
4328
+ }
4329
+ const heartbeatChannelIds = /* @__PURE__ */ new Set();
4330
+ for (const channelId of CHANNEL_IDS) heartbeatChannelIds.add(channelId.toLowerCase());
4331
+ for (const record of registry.plugins) for (const channelId of record.channels) {
4332
+ const trimmed = channelId.trim();
4333
+ if (trimmed) heartbeatChannelIds.add(trimmed.toLowerCase());
4334
+ }
4335
+ const validateHeartbeatTarget = (target, path) => {
4336
+ if (typeof target !== "string") return;
4337
+ const trimmed = target.trim();
4338
+ if (!trimmed) {
4339
+ issues.push({
4340
+ path,
4341
+ message: "heartbeat target must not be empty"
4342
+ });
4343
+ return;
4344
+ }
4345
+ const normalized = trimmed.toLowerCase();
4346
+ if (normalized === "last" || normalized === "none") return;
4347
+ if (normalizeChatChannelId(trimmed)) return;
4348
+ if (heartbeatChannelIds.has(normalized)) return;
4349
+ issues.push({
4350
+ path,
4351
+ message: `unknown heartbeat target: ${target}`
4352
+ });
4353
+ };
4354
+ validateHeartbeatTarget(config.agents?.defaults?.heartbeat?.target, "agents.defaults.heartbeat.target");
4355
+ if (Array.isArray(config.agents?.list)) for (const [index, entry] of config.agents.list.entries()) validateHeartbeatTarget(entry?.heartbeat?.target, `agents.list.${index}.heartbeat.target`);
4356
+ let selectedMemoryPluginId = null;
4357
+ const seenPlugins = /* @__PURE__ */ new Set();
4358
+ for (const record of registry.plugins) {
4359
+ const pluginId = record.id;
4360
+ if (seenPlugins.has(pluginId)) continue;
4361
+ seenPlugins.add(pluginId);
4362
+ const entry = normalizedPlugins.entries[pluginId];
4363
+ const entryHasConfig = Boolean(entry?.config);
4364
+ const enableState = resolveEnableState(pluginId, record.origin, normalizedPlugins);
4365
+ let enabled = enableState.enabled;
4366
+ let reason = enableState.reason;
4367
+ if (enabled) {
4368
+ const memoryDecision = resolveMemorySlotDecision({
4369
+ id: pluginId,
4370
+ kind: record.kind,
4371
+ slot: memorySlot,
4372
+ selectedId: selectedMemoryPluginId
4373
+ });
4374
+ if (!memoryDecision.enabled) {
4375
+ enabled = false;
4376
+ reason = memoryDecision.reason;
4377
+ }
4378
+ if (memoryDecision.selected && record.kind === "memory") selectedMemoryPluginId = pluginId;
4379
+ }
4380
+ if (enabled || entryHasConfig) if (record.configSchema) {
4381
+ const res = validateJsonSchemaValue({
4382
+ schema: record.configSchema,
4383
+ cacheKey: record.schemaCacheKey ?? record.manifestPath ?? pluginId,
4384
+ value: entry?.config ?? {}
4385
+ });
4386
+ if (!res.ok) for (const error of res.errors) issues.push({
4387
+ path: `plugins.entries.${pluginId}.config`,
4388
+ message: `invalid config: ${error}`
4389
+ });
4390
+ } else issues.push({
4391
+ path: `plugins.entries.${pluginId}`,
4392
+ message: `plugin schema missing for ${pluginId}`
4393
+ });
4394
+ if (!enabled && entryHasConfig) warnings.push({
4395
+ path: `plugins.entries.${pluginId}`,
4396
+ message: `plugin disabled (${reason ?? "disabled"}) but config is present`
4397
+ });
4398
+ }
4399
+ if (issues.length > 0) return {
4400
+ ok: false,
4401
+ issues,
4402
+ warnings
4403
+ };
4404
+ return {
4405
+ ok: true,
4406
+ config,
4407
+ warnings
4408
+ };
4409
+ }
4410
+
4411
+ //#endregion
4412
+ //#region src/config/version.ts
4413
+ const VERSION_RE = /^v?(\d+)\.(\d+)\.(\d+)(?:-(\d+))?/;
4414
+ function parseOpenClawVersion(raw) {
4415
+ if (!raw) return null;
4416
+ const match = raw.trim().match(VERSION_RE);
4417
+ if (!match) return null;
4418
+ const [, major, minor, patch, revision] = match;
4419
+ return {
4420
+ major: Number.parseInt(major, 10),
4421
+ minor: Number.parseInt(minor, 10),
4422
+ patch: Number.parseInt(patch, 10),
4423
+ revision: revision ? Number.parseInt(revision, 10) : 0
4424
+ };
4425
+ }
4426
+ function compareOpenClawVersions(a, b) {
4427
+ const parsedA = parseOpenClawVersion(a);
4428
+ const parsedB = parseOpenClawVersion(b);
4429
+ if (!parsedA || !parsedB) return null;
4430
+ if (parsedA.major !== parsedB.major) return parsedA.major < parsedB.major ? -1 : 1;
4431
+ if (parsedA.minor !== parsedB.minor) return parsedA.minor < parsedB.minor ? -1 : 1;
4432
+ if (parsedA.patch !== parsedB.patch) return parsedA.patch < parsedB.patch ? -1 : 1;
4433
+ if (parsedA.revision !== parsedB.revision) return parsedA.revision < parsedB.revision ? -1 : 1;
4434
+ return 0;
4435
+ }
4436
+
4437
+ //#endregion
4438
+ //#region src/config/io.ts
4439
+ const SHELL_ENV_EXPECTED_KEYS = [
4440
+ "OPENAI_API_KEY",
4441
+ "ANTHROPIC_API_KEY",
4442
+ "ANTHROPIC_OAUTH_TOKEN",
4443
+ "GEMINI_API_KEY",
4444
+ "ZAI_API_KEY",
4445
+ "OPENROUTER_API_KEY",
4446
+ "AI_GATEWAY_API_KEY",
4447
+ "MINIMAX_API_KEY",
4448
+ "SYNTHETIC_API_KEY",
4449
+ "ELEVENLABS_API_KEY",
4450
+ "TELEGRAM_BOT_TOKEN",
4451
+ "DISCORD_BOT_TOKEN",
4452
+ "SLACK_BOT_TOKEN",
4453
+ "SLACK_APP_TOKEN",
4454
+ "OPENCLAW_GATEWAY_TOKEN",
4455
+ "OPENCLAW_GATEWAY_PASSWORD"
4456
+ ];
4457
+ const CONFIG_BACKUP_COUNT = 5;
4458
+ const loggedInvalidConfigs = /* @__PURE__ */ new Set();
4459
+ function hashConfigRaw(raw) {
4460
+ return crypto.createHash("sha256").update(raw ?? "").digest("hex");
4461
+ }
4462
+ function resolveConfigSnapshotHash(snapshot) {
4463
+ if (typeof snapshot.hash === "string") {
4464
+ const trimmed = snapshot.hash.trim();
4465
+ if (trimmed) return trimmed;
4466
+ }
4467
+ if (typeof snapshot.raw !== "string") return null;
4468
+ return hashConfigRaw(snapshot.raw);
4469
+ }
4470
+ function coerceConfig(value) {
4471
+ if (!value || typeof value !== "object" || Array.isArray(value)) return {};
4472
+ return value;
4473
+ }
4474
+ async function rotateConfigBackups(configPath, ioFs) {
4475
+ if (CONFIG_BACKUP_COUNT <= 1) return;
4476
+ const backupBase = `${configPath}.bak`;
4477
+ const maxIndex = CONFIG_BACKUP_COUNT - 1;
4478
+ await ioFs.unlink(`${backupBase}.${maxIndex}`).catch(() => {});
4479
+ for (let index = maxIndex - 1; index >= 1; index -= 1) await ioFs.rename(`${backupBase}.${index}`, `${backupBase}.${index + 1}`).catch(() => {});
4480
+ await ioFs.rename(backupBase, `${backupBase}.1`).catch(() => {});
4481
+ }
4482
+ function warnOnConfigMiskeys(raw, logger) {
4483
+ if (!raw || typeof raw !== "object") return;
4484
+ const gateway = raw.gateway;
4485
+ if (!gateway || typeof gateway !== "object") return;
4486
+ if ("token" in gateway) logger.warn("Config uses \"gateway.token\". This key is ignored; use \"gateway.auth.token\" instead.");
4487
+ }
4488
+ function stampConfigVersion(cfg) {
4489
+ const now = (/* @__PURE__ */ new Date()).toISOString();
4490
+ return {
4491
+ ...cfg,
4492
+ meta: {
4493
+ ...cfg.meta,
4494
+ lastTouchedVersion: VERSION,
4495
+ lastTouchedAt: now
4496
+ }
4497
+ };
4498
+ }
4499
+ function warnIfConfigFromFuture(cfg, logger) {
4500
+ const touched = cfg.meta?.lastTouchedVersion;
4501
+ if (!touched) return;
4502
+ const cmp = compareOpenClawVersions(VERSION, touched);
4503
+ if (cmp === null) return;
4504
+ if (cmp < 0) logger.warn(`Config was last written by a newer OpenClaw (${touched}); current version is ${VERSION}.`);
4505
+ }
4506
+ function applyConfigEnv(cfg, env) {
4507
+ const entries = collectConfigEnvVars(cfg);
4508
+ for (const [key, value] of Object.entries(entries)) {
4509
+ if (env[key]?.trim()) continue;
4510
+ env[key] = value;
4511
+ }
4512
+ }
4513
+ function resolveConfigPathForDeps(deps) {
4514
+ if (deps.configPath) return deps.configPath;
4515
+ return resolveConfigPath(deps.env, resolveStateDir(deps.env, deps.homedir));
4516
+ }
4517
+ function normalizeDeps(overrides = {}) {
4518
+ return {
4519
+ fs: overrides.fs ?? fs,
4520
+ json5: overrides.json5 ?? JSON5,
4521
+ env: overrides.env ?? process.env,
4522
+ homedir: overrides.homedir ?? os.homedir,
4523
+ configPath: overrides.configPath ?? "",
4524
+ logger: overrides.logger ?? console
4525
+ };
4526
+ }
4527
+ function parseConfigJson5(raw, json5 = JSON5) {
4528
+ try {
4529
+ return {
4530
+ ok: true,
4531
+ parsed: json5.parse(raw)
4532
+ };
4533
+ } catch (err) {
4534
+ return {
4535
+ ok: false,
4536
+ error: String(err)
4537
+ };
4538
+ }
4539
+ }
4540
+ function createConfigIO(overrides = {}) {
4541
+ const deps = normalizeDeps(overrides);
4542
+ const requestedConfigPath = resolveConfigPathForDeps(deps);
4543
+ const configPath = (deps.configPath ? [requestedConfigPath] : resolveDefaultConfigCandidates(deps.env, deps.homedir)).find((candidate) => deps.fs.existsSync(candidate)) ?? requestedConfigPath;
4544
+ function loadConfig() {
4545
+ try {
4546
+ if (!deps.fs.existsSync(configPath)) {
4547
+ if (shouldEnableShellEnvFallback(deps.env) && !shouldDeferShellEnvFallback(deps.env)) loadShellEnvFallback({
4548
+ enabled: true,
4549
+ env: deps.env,
4550
+ expectedKeys: SHELL_ENV_EXPECTED_KEYS,
4551
+ logger: deps.logger,
4552
+ timeoutMs: resolveShellEnvFallbackTimeoutMs(deps.env)
4553
+ });
4554
+ return {};
4555
+ }
4556
+ const raw = deps.fs.readFileSync(configPath, "utf-8");
4557
+ const resolved = resolveConfigIncludes(deps.json5.parse(raw), configPath, {
4558
+ readFile: (p) => deps.fs.readFileSync(p, "utf-8"),
4559
+ parseJson: (raw) => deps.json5.parse(raw)
4560
+ });
4561
+ if (resolved && typeof resolved === "object" && "env" in resolved) applyConfigEnv(resolved, deps.env);
4562
+ const resolvedConfig = resolveConfigEnvVars(resolved, deps.env);
4563
+ warnOnConfigMiskeys(resolvedConfig, deps.logger);
4564
+ if (typeof resolvedConfig !== "object" || resolvedConfig === null) return {};
4565
+ const preValidationDuplicates = findDuplicateAgentDirs(resolvedConfig, {
4566
+ env: deps.env,
4567
+ homedir: deps.homedir
4568
+ });
4569
+ if (preValidationDuplicates.length > 0) throw new DuplicateAgentDirError(preValidationDuplicates);
4570
+ const validated = validateConfigObjectWithPlugins(resolvedConfig);
4571
+ if (!validated.ok) {
4572
+ const details = validated.issues.map((iss) => `- ${iss.path || "<root>"}: ${iss.message}`).join("\n");
4573
+ if (!loggedInvalidConfigs.has(configPath)) {
4574
+ loggedInvalidConfigs.add(configPath);
4575
+ deps.logger.error(`Invalid config at ${configPath}:\\n${details}`);
4576
+ }
4577
+ const error = /* @__PURE__ */ new Error("Invalid config");
4578
+ error.code = "INVALID_CONFIG";
4579
+ error.details = details;
4580
+ throw error;
4581
+ }
4582
+ if (validated.warnings.length > 0) {
4583
+ const details = validated.warnings.map((iss) => `- ${iss.path || "<root>"}: ${iss.message}`).join("\n");
4584
+ deps.logger.warn(`Config warnings:\\n${details}`);
4585
+ }
4586
+ warnIfConfigFromFuture(validated.config, deps.logger);
4587
+ const cfg = applyModelDefaults(applyCompactionDefaults(applyContextPruningDefaults(applyAgentDefaults(applySessionDefaults(applyLoggingDefaults(applyMessageDefaults(validated.config)))))));
4588
+ normalizeConfigPaths(cfg);
4589
+ const duplicates = findDuplicateAgentDirs(cfg, {
4590
+ env: deps.env,
4591
+ homedir: deps.homedir
4592
+ });
4593
+ if (duplicates.length > 0) throw new DuplicateAgentDirError(duplicates);
4594
+ applyConfigEnv(cfg, deps.env);
4595
+ if ((shouldEnableShellEnvFallback(deps.env) || cfg.env?.shellEnv?.enabled === true) && !shouldDeferShellEnvFallback(deps.env)) loadShellEnvFallback({
4596
+ enabled: true,
4597
+ env: deps.env,
4598
+ expectedKeys: SHELL_ENV_EXPECTED_KEYS,
4599
+ logger: deps.logger,
4600
+ timeoutMs: cfg.env?.shellEnv?.timeoutMs ?? resolveShellEnvFallbackTimeoutMs(deps.env)
4601
+ });
4602
+ return applyConfigOverrides(cfg);
4603
+ } catch (err) {
4604
+ if (err instanceof DuplicateAgentDirError) {
4605
+ deps.logger.error(err.message);
4606
+ throw err;
4607
+ }
4608
+ if (err?.code === "INVALID_CONFIG") return {};
4609
+ deps.logger.error(`Failed to read config at ${configPath}`, err);
4610
+ return {};
4611
+ }
4612
+ }
4613
+ async function readConfigFileSnapshot() {
4614
+ if (!deps.fs.existsSync(configPath)) {
4615
+ const hash = hashConfigRaw(null);
4616
+ return {
4617
+ path: configPath,
4618
+ exists: false,
4619
+ raw: null,
4620
+ parsed: {},
4621
+ valid: true,
4622
+ config: applyTalkApiKey(applyModelDefaults(applyCompactionDefaults(applyContextPruningDefaults(applyAgentDefaults(applySessionDefaults(applyMessageDefaults({}))))))),
4623
+ hash,
4624
+ issues: [],
4625
+ warnings: [],
4626
+ legacyIssues: []
4627
+ };
4628
+ }
4629
+ try {
4630
+ const raw = deps.fs.readFileSync(configPath, "utf-8");
4631
+ const hash = hashConfigRaw(raw);
4632
+ const parsedRes = parseConfigJson5(raw, deps.json5);
4633
+ if (!parsedRes.ok) return {
4634
+ path: configPath,
4635
+ exists: true,
4636
+ raw,
4637
+ parsed: {},
4638
+ valid: false,
4639
+ config: {},
4640
+ hash,
4641
+ issues: [{
4642
+ path: "",
4643
+ message: `JSON5 parse failed: ${parsedRes.error}`
4644
+ }],
4645
+ warnings: [],
4646
+ legacyIssues: []
4647
+ };
4648
+ let resolved;
4649
+ try {
4650
+ resolved = resolveConfigIncludes(parsedRes.parsed, configPath, {
4651
+ readFile: (p) => deps.fs.readFileSync(p, "utf-8"),
4652
+ parseJson: (raw) => deps.json5.parse(raw)
4653
+ });
4654
+ } catch (err) {
4655
+ const message = err instanceof ConfigIncludeError ? err.message : `Include resolution failed: ${String(err)}`;
4656
+ return {
4657
+ path: configPath,
4658
+ exists: true,
4659
+ raw,
4660
+ parsed: parsedRes.parsed,
4661
+ valid: false,
4662
+ config: coerceConfig(parsedRes.parsed),
4663
+ hash,
4664
+ issues: [{
4665
+ path: "",
4666
+ message
4667
+ }],
4668
+ warnings: [],
4669
+ legacyIssues: []
4670
+ };
4671
+ }
4672
+ if (resolved && typeof resolved === "object" && "env" in resolved) applyConfigEnv(resolved, deps.env);
4673
+ let substituted;
4674
+ try {
4675
+ substituted = resolveConfigEnvVars(resolved, deps.env);
4676
+ } catch (err) {
4677
+ const message = err instanceof MissingEnvVarError ? err.message : `Env var substitution failed: ${String(err)}`;
4678
+ return {
4679
+ path: configPath,
4680
+ exists: true,
4681
+ raw,
4682
+ parsed: parsedRes.parsed,
4683
+ valid: false,
4684
+ config: coerceConfig(resolved),
4685
+ hash,
4686
+ issues: [{
4687
+ path: "",
4688
+ message
4689
+ }],
4690
+ warnings: [],
4691
+ legacyIssues: []
4692
+ };
4693
+ }
4694
+ const resolvedConfigRaw = substituted;
4695
+ const legacyIssues = findLegacyConfigIssues(resolvedConfigRaw);
4696
+ const validated = validateConfigObjectWithPlugins(resolvedConfigRaw);
4697
+ if (!validated.ok) return {
4698
+ path: configPath,
4699
+ exists: true,
4700
+ raw,
4701
+ parsed: parsedRes.parsed,
4702
+ valid: false,
4703
+ config: coerceConfig(resolvedConfigRaw),
4704
+ hash,
4705
+ issues: validated.issues,
4706
+ warnings: validated.warnings,
4707
+ legacyIssues
4708
+ };
4709
+ warnIfConfigFromFuture(validated.config, deps.logger);
4710
+ return {
4711
+ path: configPath,
4712
+ exists: true,
4713
+ raw,
4714
+ parsed: parsedRes.parsed,
4715
+ valid: true,
4716
+ config: normalizeConfigPaths(applyTalkApiKey(applyModelDefaults(applyAgentDefaults(applySessionDefaults(applyLoggingDefaults(applyMessageDefaults(validated.config))))))),
4717
+ hash,
4718
+ issues: [],
4719
+ warnings: validated.warnings,
4720
+ legacyIssues
4721
+ };
4722
+ } catch (err) {
4723
+ return {
4724
+ path: configPath,
4725
+ exists: true,
4726
+ raw: null,
4727
+ parsed: {},
4728
+ valid: false,
4729
+ config: {},
4730
+ hash: hashConfigRaw(null),
4731
+ issues: [{
4732
+ path: "",
4733
+ message: `read failed: ${String(err)}`
4734
+ }],
4735
+ warnings: [],
4736
+ legacyIssues: []
4737
+ };
4738
+ }
4739
+ }
4740
+ async function writeConfigFile(cfg) {
4741
+ clearConfigCache();
4742
+ const validated = validateConfigObjectWithPlugins(cfg);
4743
+ if (!validated.ok) {
4744
+ const issue = validated.issues[0];
4745
+ const pathLabel = issue?.path ? issue.path : "<root>";
4746
+ throw new Error(`Config validation failed: ${pathLabel}: ${issue?.message ?? "invalid"}`);
4747
+ }
4748
+ if (validated.warnings.length > 0) {
4749
+ const details = validated.warnings.map((warning) => `- ${warning.path}: ${warning.message}`).join("\n");
4750
+ deps.logger.warn(`Config warnings:\n${details}`);
4751
+ }
4752
+ const dir = path.dirname(configPath);
4753
+ await deps.fs.promises.mkdir(dir, {
4754
+ recursive: true,
4755
+ mode: 448
4756
+ });
4757
+ const json = JSON.stringify(applyModelDefaults(stampConfigVersion(cfg)), null, 2).trimEnd().concat("\n");
4758
+ const tmp = path.join(dir, `${path.basename(configPath)}.${process.pid}.${crypto.randomUUID()}.tmp`);
4759
+ await deps.fs.promises.writeFile(tmp, json, {
4760
+ encoding: "utf-8",
4761
+ mode: 384
4762
+ });
4763
+ if (deps.fs.existsSync(configPath)) {
4764
+ await rotateConfigBackups(configPath, deps.fs.promises);
4765
+ await deps.fs.promises.copyFile(configPath, `${configPath}.bak`).catch(() => {});
4766
+ }
4767
+ try {
4768
+ await deps.fs.promises.rename(tmp, configPath);
4769
+ } catch (err) {
4770
+ const code = err.code;
4771
+ if (code === "EPERM" || code === "EEXIST") {
4772
+ await deps.fs.promises.copyFile(tmp, configPath);
4773
+ await deps.fs.promises.chmod(configPath, 384).catch(() => {});
4774
+ await deps.fs.promises.unlink(tmp).catch(() => {});
4775
+ return;
4776
+ }
4777
+ await deps.fs.promises.unlink(tmp).catch(() => {});
4778
+ throw err;
4779
+ }
4780
+ }
4781
+ return {
4782
+ configPath,
4783
+ loadConfig,
4784
+ readConfigFileSnapshot,
4785
+ writeConfigFile
4786
+ };
4787
+ }
4788
+ const DEFAULT_CONFIG_CACHE_MS = 200;
4789
+ let configCache = null;
4790
+ function resolveConfigCacheMs(env) {
4791
+ const raw = env.OPENCLAW_CONFIG_CACHE_MS?.trim();
4792
+ if (raw === "" || raw === "0") return 0;
4793
+ if (!raw) return DEFAULT_CONFIG_CACHE_MS;
4794
+ const parsed = Number.parseInt(raw, 10);
4795
+ if (!Number.isFinite(parsed)) return DEFAULT_CONFIG_CACHE_MS;
4796
+ return Math.max(0, parsed);
4797
+ }
4798
+ function shouldUseConfigCache(env) {
4799
+ if (env.OPENCLAW_DISABLE_CONFIG_CACHE?.trim()) return false;
4800
+ return resolveConfigCacheMs(env) > 0;
4801
+ }
4802
+ function clearConfigCache() {
4803
+ configCache = null;
4804
+ }
4805
+ function loadConfig() {
4806
+ const io = createConfigIO();
4807
+ const configPath = io.configPath;
4808
+ const now = Date.now();
4809
+ if (shouldUseConfigCache(process.env)) {
4810
+ const cached = configCache;
4811
+ if (cached && cached.configPath === configPath && cached.expiresAt > now) return cached.config;
4812
+ }
4813
+ const config = io.loadConfig();
4814
+ if (shouldUseConfigCache(process.env)) {
4815
+ const cacheMs = resolveConfigCacheMs(process.env);
4816
+ if (cacheMs > 0) configCache = {
4817
+ configPath,
4818
+ expiresAt: now + cacheMs,
4819
+ config
4820
+ };
4821
+ }
4822
+ return config;
4823
+ }
4824
+ async function readConfigFileSnapshot() {
4825
+ return await createConfigIO().readConfigFileSnapshot();
4826
+ }
4827
+ async function writeConfigFile(cfg) {
4828
+ clearConfigCache();
4829
+ await createConfigIO().writeConfigFile(cfg);
4830
+ }
4831
+
4832
+ //#endregion
4833
+ //#region src/config/legacy-migrate.ts
4834
+ function migrateLegacyConfig(raw) {
4835
+ const { next, changes } = applyLegacyMigrations(raw);
4836
+ if (!next) return {
4837
+ config: null,
4838
+ changes: []
4839
+ };
4840
+ const validated = validateConfigObjectWithPlugins(next);
4841
+ if (!validated.ok) {
4842
+ changes.push("Migration applied, but config still invalid; fix remaining issues manually.");
4843
+ return {
4844
+ config: null,
4845
+ changes
4846
+ };
4847
+ }
4848
+ return {
4849
+ config: validated.config,
4850
+ changes
4851
+ };
4852
+ }
4853
+
4854
+ //#endregion
4855
+ //#region src/config/config.ts
4856
+ var config_exports = /* @__PURE__ */ __exportAll({
4857
+ CONFIG_PATH: () => CONFIG_PATH,
4858
+ DEFAULT_GATEWAY_PORT: () => DEFAULT_GATEWAY_PORT,
4859
+ STATE_DIR: () => STATE_DIR,
4860
+ applyConfigOverrides: () => applyConfigOverrides,
4861
+ getConfigOverrides: () => getConfigOverrides,
4862
+ isNixMode: () => isNixMode,
4863
+ loadConfig: () => loadConfig,
4864
+ resetConfigOverrides: () => resetConfigOverrides,
4865
+ resolveCanonicalConfigPath: () => resolveCanonicalConfigPath,
4866
+ resolveConfigPath: () => resolveConfigPath,
4867
+ resolveConfigPathCandidate: () => resolveConfigPathCandidate,
4868
+ resolveDefaultConfigCandidates: () => resolveDefaultConfigCandidates,
4869
+ resolveGatewayLockDir: () => resolveGatewayLockDir,
4870
+ resolveGatewayPort: () => resolveGatewayPort,
4871
+ resolveIsNixMode: () => resolveIsNixMode,
4872
+ resolveLegacyStateDirs: () => resolveLegacyStateDirs,
4873
+ resolveNewStateDir: () => resolveNewStateDir,
4874
+ resolveOAuthDir: () => resolveOAuthDir,
4875
+ resolveOAuthPath: () => resolveOAuthPath,
4876
+ resolveStateDir: () => resolveStateDir,
4877
+ setConfigOverride: () => setConfigOverride,
4878
+ unsetConfigOverride: () => unsetConfigOverride
4879
+ });
4880
+
4881
+ //#endregion
4882
+ export { resolveSubagentMaxConcurrent as A, setConfigValueAtPath as C, MAX_INCLUDE_DEPTH as D, INCLUDE_KEY as E, collectConfigEnvVars as O, parseConfigPath as S, applyLegacyMigrations as T, getConfigOverrides as _, parseConfigJson5 as a, unsetConfigOverride as b, writeConfigFile as c, TELEGRAM_COMMAND_NAME_PATTERN as d, normalizeTelegramCommandName as f, validateJsonSchemaValue as g, parseDurationMs as h, loadConfig as i, VERSION as j, resolveAgentMaxConcurrent as k, validateConfigObjectWithPlugins as l, isSafeExecutableValue as m, migrateLegacyConfig as n, readConfigFileSnapshot as o, resolveTelegramCustomCommands as p, createConfigIO as r, resolveConfigSnapshotHash as s, config_exports as t, OpenClawSchema as u, resetConfigOverrides as v, unsetConfigValueAtPath as w, getConfigValueAtPath as x, setConfigOverride as y };