synurex 2026.2.19-1

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