abtars 0.2.1-alpha.8 → 0.2.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 (257) hide show
  1. package/README.md +98 -42
  2. package/bundle/{_registry.generated-ADOYFJJ4.js → _registry.generated-FNJOX7VV.js} +19 -16
  3. package/bundle/{_registry.generated-ADOYFJJ4.js.map → _registry.generated-FNJOX7VV.js.map} +1 -1
  4. package/bundle/abtars-browser.js +5 -6
  5. package/bundle/abtars-browser.js.map +2 -2
  6. package/bundle/abtars-cli.js +339 -271
  7. package/bundle/abtars-cli.js.map +4 -4
  8. package/bundle/abtars-restart.js +4 -4
  9. package/bundle/abtars-rss.js +25 -81
  10. package/bundle/abtars-rss.js.map +2 -2
  11. package/bundle/abtars-task.js +4 -5
  12. package/bundle/abtars-task.js.map +2 -2
  13. package/bundle/abtars-todo.js +133 -0
  14. package/bundle/abtars-todo.js.map +7 -0
  15. package/bundle/abtars.js +239 -155
  16. package/bundle/abtars.js.map +3 -3
  17. package/bundle/action-gate-DYV2XQBP.js +191 -0
  18. package/bundle/action-gate-DYV2XQBP.js.map +7 -0
  19. package/bundle/{agent-api-rate-limit-C25WGSFF.js → agent-api-rate-limit-R2OFAQ3N.js} +4 -4
  20. package/bundle/{agent-registry-SYUFNSVB.js → agent-registry-S2MNHQYQ.js} +8 -8
  21. package/bundle/{bridge-lock-transport-HO545SBK.js → bridge-lock-transport-N6OGDOSE.js} +5 -5
  22. package/bundle/{browse-delivery-VTLEAVYA.js → browse-delivery-DXGMDMXA.js} +8 -7
  23. package/bundle/{browser-REIXOJ6S.js → browser-QMYGSP5W.js} +11 -10
  24. package/bundle/{capability-ILW3D5HS.js → capability-733TLH4W.js} +6 -6
  25. package/bundle/{chunk-ZVDVNSXK.js → chunk-2SWKJX64.js} +7 -7
  26. package/bundle/{chunk-PZE3J7ER.js → chunk-3OXQWII3.js} +2 -2
  27. package/bundle/{chunk-R36WIOYX.js → chunk-3X6VGRL6.js} +33 -6
  28. package/bundle/chunk-3X6VGRL6.js.map +7 -0
  29. package/bundle/{chunk-4KJ76TTE.js → chunk-4XW7YA3K.js} +3 -3
  30. package/bundle/{chunk-Q62SXS73.js → chunk-BBDKU4EH.js} +9 -9
  31. package/bundle/{chunk-EX2SRTUE.js → chunk-CYSGXNBY.js} +2 -2
  32. package/bundle/{chunk-LYEAHE5V.js → chunk-DCY7DGMT.js} +2 -2
  33. package/bundle/chunk-DGS7J4P6.js +13 -0
  34. package/bundle/chunk-DGS7J4P6.js.map +7 -0
  35. package/bundle/{chunk-LDKSCXGL.js → chunk-DHPFI7OF.js} +8 -6
  36. package/bundle/{chunk-LDKSCXGL.js.map → chunk-DHPFI7OF.js.map} +1 -1
  37. package/bundle/{chunk-G6IXMYIO.js → chunk-DO4INSXE.js} +2 -2
  38. package/bundle/{chunk-VA5WKN3Z.js → chunk-EGA6JQVV.js} +4 -4
  39. package/bundle/chunk-EKHNWFEQ.js +85 -0
  40. package/bundle/chunk-EKHNWFEQ.js.map +7 -0
  41. package/bundle/{chunk-URAQLQ2U.js → chunk-F3HMZFIL.js} +4 -4
  42. package/bundle/{chunk-OOKLEC6V.js → chunk-FY3QUO2L.js} +7 -7
  43. package/bundle/{chunk-2BY6I4P5.js → chunk-GUTRAMK3.js} +2 -2
  44. package/bundle/{chunk-GPL57SRN.js → chunk-H7RX7UCR.js} +3 -3
  45. package/bundle/{chunk-BYDUMHXT.js → chunk-HAF2AFBW.js} +2 -2
  46. package/bundle/{chunk-A5OJYQNU.js → chunk-HCYENZAB.js} +49 -22
  47. package/bundle/chunk-HCYENZAB.js.map +7 -0
  48. package/bundle/{chunk-OZ4IZFV4.js → chunk-HJQZP5CK.js} +9 -9
  49. package/bundle/{chunk-OZ4IZFV4.js.map → chunk-HJQZP5CK.js.map} +2 -2
  50. package/bundle/{chunk-HEHD3GG5.js → chunk-ITB2K6LI.js} +6 -13
  51. package/bundle/{chunk-HEHD3GG5.js.map → chunk-ITB2K6LI.js.map} +3 -3
  52. package/bundle/{chunk-KSEIWT4T.js → chunk-JFKOPNKL.js} +10 -10
  53. package/bundle/chunk-JFKOPNKL.js.map +7 -0
  54. package/bundle/{chunk-4BUOO6WI.js → chunk-KL5QRHHK.js} +31 -11
  55. package/bundle/chunk-KL5QRHHK.js.map +7 -0
  56. package/bundle/{chunk-KFENC7BM.js → chunk-L33WNMCP.js} +2 -2
  57. package/bundle/{chunk-JRG4EFMP.js → chunk-LBMETTUP.js} +3 -3
  58. package/bundle/{chunk-TYVI3ZWA.js → chunk-LJAG2URA.js} +10 -7
  59. package/bundle/chunk-LJAG2URA.js.map +7 -0
  60. package/bundle/{chunk-RV54J75Q.js → chunk-MHK4UPM6.js} +13 -12
  61. package/bundle/chunk-MHK4UPM6.js.map +7 -0
  62. package/bundle/chunk-NIYVCGBC.js +330 -0
  63. package/bundle/chunk-NIYVCGBC.js.map +7 -0
  64. package/bundle/{chunk-P2BZSLJJ.js → chunk-OGZXYN6E.js} +392 -125
  65. package/bundle/chunk-OGZXYN6E.js.map +7 -0
  66. package/bundle/{chunk-TOUZC6NR.js → chunk-OKMN6J4Z.js} +3 -3
  67. package/bundle/{chunk-CELR236Q.js → chunk-Q7CH5DA3.js} +2 -2
  68. package/bundle/{chunk-XZSYMCLF.js → chunk-QIAFGDRL.js} +7 -7
  69. package/bundle/chunk-RITEGAW6.js +138 -0
  70. package/bundle/chunk-RITEGAW6.js.map +7 -0
  71. package/bundle/{chunk-UDZIZB5F.js → chunk-RTL7HO3N.js} +3 -3
  72. package/bundle/{chunk-ITG6XGBS.js → chunk-SA44ITVX.js} +10 -10
  73. package/bundle/{chunk-3MO2MDXJ.js → chunk-SA6YEFNG.js} +3 -3
  74. package/bundle/{chunk-GBBTK6H2.js → chunk-SRFEIZQT.js} +4 -4
  75. package/bundle/{chunk-ELRAH7VL.js → chunk-VXUVKC66.js} +3 -3
  76. package/bundle/{chunk-X5FBUA53.js → chunk-WLAVZSVZ.js} +69 -30
  77. package/bundle/chunk-WLAVZSVZ.js.map +7 -0
  78. package/bundle/{chunk-RSWUPUNA.js → chunk-WMWI3SJ7.js} +30 -6
  79. package/bundle/chunk-WMWI3SJ7.js.map +7 -0
  80. package/bundle/{chunk-MJ6PHMOK.js → chunk-WW5F2DCO.js} +11 -2
  81. package/bundle/chunk-WW5F2DCO.js.map +7 -0
  82. package/bundle/chunk-YWZPKBO6.js +22 -0
  83. package/bundle/chunk-YWZPKBO6.js.map +7 -0
  84. package/bundle/{chunk-KWBGYWDO.js → chunk-Z4SWEFIY.js} +12 -14
  85. package/bundle/chunk-Z4SWEFIY.js.map +7 -0
  86. package/bundle/chunk-ZAA7B5BN.js +22 -0
  87. package/bundle/chunk-ZAA7B5BN.js.map +7 -0
  88. package/bundle/{commands-WUGPBPHI.js → commands-AXW7L2MZ.js} +15 -14
  89. package/bundle/{config-DQIGDX4W.js → config-NDEYF4AQ.js} +7 -7
  90. package/bundle/{daemon-NPKYZ3CJ.js → daemon-WOQXCKNL.js} +4 -4
  91. package/bundle/{delegation-tools-6FICZQ5G.js → delegation-tools-Z5OM3TXS.js} +5 -5
  92. package/bundle/{deploy-lib-import-SBKXDD3F.js → deploy-lib-import-6VJTYXEG.js} +2 -2
  93. package/bundle/{deps-HN6CEXA4.js → deps-65V7XXG4.js} +4 -4
  94. package/bundle/{direct-api-transport-TRV45NO6.js → direct-api-transport-LSAUIP5S.js} +27 -15
  95. package/bundle/direct-api-transport-LSAUIP5S.js.map +7 -0
  96. package/bundle/{discord-adapter-WA2MFRK3.js → discord-adapter-LNWTIOUK.js} +27 -24
  97. package/bundle/discord-adapter-LNWTIOUK.js.map +7 -0
  98. package/bundle/{dns-wakeup-RYOCQ6GR.js → dns-wakeup-N46RPU5E.js} +3 -3
  99. package/bundle/{doctor-R54GZPKL.js → doctor-PIPSGI3H.js} +18 -7
  100. package/bundle/{doctor-R54GZPKL.js.map → doctor-PIPSGI3H.js.map} +2 -2
  101. package/bundle/{ensure-invariants-BJIEOSJ2.js → ensure-invariants-3NOBCYWS.js} +4 -4
  102. package/bundle/{env-schema-XCPAJ6IZ.js → env-schema-T43X43BU.js} +4 -4
  103. package/bundle/{hook-system-POI5VRIX.js → hook-system-ZCVOFFRD.js} +4 -4
  104. package/bundle/hotskills-DTROJY6G.js +17 -0
  105. package/bundle/{install-GEXWJYJC.js → install-FZT43PTH.js} +3 -3
  106. package/bundle/{install-manifest-QRWID3KZ.js → install-manifest-KBYD7SAY.js} +3 -3
  107. package/bundle/{irc-adapter-AIEP6OX6.js → irc-adapter-HXO5D4SW.js} +3 -3
  108. package/bundle/{irc-config-6VY67UPQ.js → irc-config-XN5VW2V4.js} +5 -5
  109. package/bundle/{lazy-require-UFYFFX2R.js → lazy-require-R3JYCV5M.js} +4 -4
  110. package/bundle/{media-utils-MOE36VWY.js → media-utils-W7XW3SVV.js} +4 -4
  111. package/bundle/{message-pipeline-2MBT44FO.js → message-pipeline-QX272U5X.js} +17 -14
  112. package/bundle/meta.json +2965 -2294
  113. package/bundle/{notification-U6F5ZBSG.js → notification-ULESRDHB.js} +7 -6
  114. package/bundle/{openrouter-credits-7XXO6QGQ.js → openrouter-credits-PLIKRY5D.js} +4 -4
  115. package/bundle/{paths-ZJYIDND2.js → paths-QQM74XYT.js} +4 -2
  116. package/bundle/{peer-client-T44VI7NB.js → peer-client-D2F5QWRV.js} +8 -8
  117. package/bundle/{peer-config-D5A4454H.js → peer-config-5SUIBJLG.js} +5 -5
  118. package/bundle/{phase-transport-FEZ4SIJJ.js → phase-transport-BSGROTHY.js} +10 -10
  119. package/bundle/restore-Z6MF54HS.js +151 -0
  120. package/bundle/restore-Z6MF54HS.js.map +7 -0
  121. package/bundle/{install-SH4UVUXQ.js → rollback-5RXXLUD6.js} +5 -7
  122. package/bundle/{self-healer-utils-7NFH22VJ.js → self-healer-utils-WPKOVXJD.js} +4 -4
  123. package/bundle/{skill-stats-IPVKMWN3.js → skill-stats-NHNH47QW.js} +5 -5
  124. package/bundle/{sleep-BPWX3FCN.js → sleep-K7EXAFGW.js} +8 -8
  125. package/bundle/{soul-bundle-BRIUDEQ2.js → soul-bundle-QTPWDJB2.js} +7 -7
  126. package/bundle/{soul-loader-GBXJ7EBH.js → soul-loader-LCPTN4PK.js} +8 -8
  127. package/bundle/{sse-parser-anthropic-H42TTLBD.js → sse-parser-anthropic-PYDJM3UC.js} +4 -4
  128. package/bundle/{sse-parser-responses-WG2LY2ML.js → sse-parser-responses-FYT7A5WT.js} +4 -4
  129. package/bundle/{ssrf-guard-E2KBBC5E.js → ssrf-guard-R4P5OCTO.js} +4 -4
  130. package/bundle/{start-CBVKNEAT.js → start-4DNURGIY.js} +1 -1
  131. package/bundle/{stt-CF3CPFDC.js → stt-YN77NND6.js} +5 -5
  132. package/bundle/{subagent-runtime-4MTYUBIZ.js → subagent-runtime-FQAT3564.js} +5 -5
  133. package/bundle/system-event-buffer-OEPPNUGK.js +17 -0
  134. package/bundle/system-event-buffer-OEPPNUGK.js.map +7 -0
  135. package/bundle/{system-message-TALP6GP2.js → system-message-BRU267FW.js} +3 -3
  136. package/bundle/{system-status-GLYXXDE3.js → system-status-7K2QTH3J.js} +58 -51
  137. package/bundle/system-status-7K2QTH3J.js.map +7 -0
  138. package/bundle/{hotskills-6ECHLXTJ.js → task-failure-buffer-DPM5MWZ5.js} +8 -7
  139. package/bundle/{task-store-LC7ZMS72.js → task-store-VCBHAB43.js} +5 -5
  140. package/bundle/{telegram-adapter-BJJYXN7J.js → telegram-adapter-LXLSG4SK.js} +51 -33
  141. package/bundle/telegram-adapter-LXLSG4SK.js.map +7 -0
  142. package/bundle/{tool-registry-T7XLTI2Q.js → tool-registry-5PXNSYOI.js} +12 -9
  143. package/bundle/{tool-sandbox-OZMXJZLQ.js → tool-sandbox-TLAL55QP.js} +5 -5
  144. package/bundle/{transport-config-G5NKQXPJ.js → transport-config-JIKHB7GT.js} +8 -8
  145. package/bundle/update-check-AJMIBQGQ.js +81 -0
  146. package/bundle/update-check-AJMIBQGQ.js.map +7 -0
  147. package/bundle/{user-registry-NUVNEHJU.js → user-registry-PEFDZ5AV.js} +5 -5
  148. package/core/skills/tools/rss/SKILL.md +51 -0
  149. package/install-manifest.json +4 -2
  150. package/package.json +4 -4
  151. package/scripts/build-and-deploy.sh +28 -0
  152. package/scripts/daily-backup.sh +1 -79
  153. package/scripts/doctor.sh +38 -0
  154. package/scripts/emergency-deploy.sh +95 -0
  155. package/scripts/watchdog.sh +51 -5
  156. package/bundle/chunk-4BUOO6WI.js.map +0 -7
  157. package/bundle/chunk-A5OJYQNU.js.map +0 -7
  158. package/bundle/chunk-JX3ZZU3O.js +0 -82
  159. package/bundle/chunk-JX3ZZU3O.js.map +0 -7
  160. package/bundle/chunk-KJOCXWJ5.js +0 -131
  161. package/bundle/chunk-KJOCXWJ5.js.map +0 -7
  162. package/bundle/chunk-KSEIWT4T.js.map +0 -7
  163. package/bundle/chunk-KWBGYWDO.js.map +0 -7
  164. package/bundle/chunk-MJ6PHMOK.js.map +0 -7
  165. package/bundle/chunk-P2BZSLJJ.js.map +0 -7
  166. package/bundle/chunk-R36WIOYX.js.map +0 -7
  167. package/bundle/chunk-RE3F3CFW.js +0 -300
  168. package/bundle/chunk-RE3F3CFW.js.map +0 -7
  169. package/bundle/chunk-RJJWPMUZ.js +0 -411
  170. package/bundle/chunk-RJJWPMUZ.js.map +0 -7
  171. package/bundle/chunk-RSWUPUNA.js.map +0 -7
  172. package/bundle/chunk-RV54J75Q.js.map +0 -7
  173. package/bundle/chunk-RVERPUHT.js +0 -289
  174. package/bundle/chunk-RVERPUHT.js.map +0 -7
  175. package/bundle/chunk-TYVI3ZWA.js.map +0 -7
  176. package/bundle/chunk-X5FBUA53.js.map +0 -7
  177. package/bundle/deploy-lib-import-HCMZCKZD.js +0 -50
  178. package/bundle/direct-api-transport-TRV45NO6.js.map +0 -7
  179. package/bundle/discord-adapter-WA2MFRK3.js.map +0 -7
  180. package/bundle/restore-ROJF22R2.js +0 -47
  181. package/bundle/restore-ROJF22R2.js.map +0 -7
  182. package/bundle/system-status-GLYXXDE3.js.map +0 -7
  183. package/bundle/telegram-adapter-BJJYXN7J.js.map +0 -7
  184. package/bundle/update-check-O5MS6B3L.js +0 -13
  185. package/bundle/user-registry-NUVNEHJU.js.map +0 -7
  186. /package/bundle/{agent-api-rate-limit-C25WGSFF.js.map → agent-api-rate-limit-R2OFAQ3N.js.map} +0 -0
  187. /package/bundle/{agent-registry-SYUFNSVB.js.map → agent-registry-S2MNHQYQ.js.map} +0 -0
  188. /package/bundle/{bridge-lock-transport-HO545SBK.js.map → bridge-lock-transport-N6OGDOSE.js.map} +0 -0
  189. /package/bundle/{browse-delivery-VTLEAVYA.js.map → browse-delivery-DXGMDMXA.js.map} +0 -0
  190. /package/bundle/{browser-REIXOJ6S.js.map → browser-QMYGSP5W.js.map} +0 -0
  191. /package/bundle/{capability-ILW3D5HS.js.map → capability-733TLH4W.js.map} +0 -0
  192. /package/bundle/{chunk-ZVDVNSXK.js.map → chunk-2SWKJX64.js.map} +0 -0
  193. /package/bundle/{chunk-PZE3J7ER.js.map → chunk-3OXQWII3.js.map} +0 -0
  194. /package/bundle/{chunk-4KJ76TTE.js.map → chunk-4XW7YA3K.js.map} +0 -0
  195. /package/bundle/{chunk-Q62SXS73.js.map → chunk-BBDKU4EH.js.map} +0 -0
  196. /package/bundle/{chunk-EX2SRTUE.js.map → chunk-CYSGXNBY.js.map} +0 -0
  197. /package/bundle/{chunk-LYEAHE5V.js.map → chunk-DCY7DGMT.js.map} +0 -0
  198. /package/bundle/{chunk-G6IXMYIO.js.map → chunk-DO4INSXE.js.map} +0 -0
  199. /package/bundle/{chunk-VA5WKN3Z.js.map → chunk-EGA6JQVV.js.map} +0 -0
  200. /package/bundle/{chunk-URAQLQ2U.js.map → chunk-F3HMZFIL.js.map} +0 -0
  201. /package/bundle/{chunk-OOKLEC6V.js.map → chunk-FY3QUO2L.js.map} +0 -0
  202. /package/bundle/{chunk-2BY6I4P5.js.map → chunk-GUTRAMK3.js.map} +0 -0
  203. /package/bundle/{chunk-GPL57SRN.js.map → chunk-H7RX7UCR.js.map} +0 -0
  204. /package/bundle/{chunk-BYDUMHXT.js.map → chunk-HAF2AFBW.js.map} +0 -0
  205. /package/bundle/{chunk-KFENC7BM.js.map → chunk-L33WNMCP.js.map} +0 -0
  206. /package/bundle/{chunk-JRG4EFMP.js.map → chunk-LBMETTUP.js.map} +0 -0
  207. /package/bundle/{chunk-TOUZC6NR.js.map → chunk-OKMN6J4Z.js.map} +0 -0
  208. /package/bundle/{chunk-CELR236Q.js.map → chunk-Q7CH5DA3.js.map} +0 -0
  209. /package/bundle/{chunk-XZSYMCLF.js.map → chunk-QIAFGDRL.js.map} +0 -0
  210. /package/bundle/{chunk-UDZIZB5F.js.map → chunk-RTL7HO3N.js.map} +0 -0
  211. /package/bundle/{chunk-ITG6XGBS.js.map → chunk-SA44ITVX.js.map} +0 -0
  212. /package/bundle/{chunk-3MO2MDXJ.js.map → chunk-SA6YEFNG.js.map} +0 -0
  213. /package/bundle/{chunk-GBBTK6H2.js.map → chunk-SRFEIZQT.js.map} +0 -0
  214. /package/bundle/{chunk-ELRAH7VL.js.map → chunk-VXUVKC66.js.map} +0 -0
  215. /package/bundle/{commands-WUGPBPHI.js.map → commands-AXW7L2MZ.js.map} +0 -0
  216. /package/bundle/{config-DQIGDX4W.js.map → config-NDEYF4AQ.js.map} +0 -0
  217. /package/bundle/{daemon-NPKYZ3CJ.js.map → daemon-WOQXCKNL.js.map} +0 -0
  218. /package/bundle/{delegation-tools-6FICZQ5G.js.map → delegation-tools-Z5OM3TXS.js.map} +0 -0
  219. /package/bundle/{deploy-lib-import-HCMZCKZD.js.map → deploy-lib-import-6VJTYXEG.js.map} +0 -0
  220. /package/bundle/{deps-HN6CEXA4.js.map → deps-65V7XXG4.js.map} +0 -0
  221. /package/bundle/{dns-wakeup-RYOCQ6GR.js.map → dns-wakeup-N46RPU5E.js.map} +0 -0
  222. /package/bundle/{ensure-invariants-BJIEOSJ2.js.map → ensure-invariants-3NOBCYWS.js.map} +0 -0
  223. /package/bundle/{deploy-lib-import-SBKXDD3F.js.map → env-schema-T43X43BU.js.map} +0 -0
  224. /package/bundle/{env-schema-XCPAJ6IZ.js.map → hook-system-ZCVOFFRD.js.map} +0 -0
  225. /package/bundle/{hook-system-POI5VRIX.js.map → hotskills-DTROJY6G.js.map} +0 -0
  226. /package/bundle/{hotskills-6ECHLXTJ.js.map → install-FZT43PTH.js.map} +0 -0
  227. /package/bundle/{install-manifest-QRWID3KZ.js.map → install-manifest-KBYD7SAY.js.map} +0 -0
  228. /package/bundle/{irc-adapter-AIEP6OX6.js.map → irc-adapter-HXO5D4SW.js.map} +0 -0
  229. /package/bundle/{irc-config-6VY67UPQ.js.map → irc-config-XN5VW2V4.js.map} +0 -0
  230. /package/bundle/{install-GEXWJYJC.js.map → lazy-require-R3JYCV5M.js.map} +0 -0
  231. /package/bundle/{media-utils-MOE36VWY.js.map → media-utils-W7XW3SVV.js.map} +0 -0
  232. /package/bundle/{install-SH4UVUXQ.js.map → message-pipeline-QX272U5X.js.map} +0 -0
  233. /package/bundle/{lazy-require-UFYFFX2R.js.map → notification-ULESRDHB.js.map} +0 -0
  234. /package/bundle/{openrouter-credits-7XXO6QGQ.js.map → openrouter-credits-PLIKRY5D.js.map} +0 -0
  235. /package/bundle/{message-pipeline-2MBT44FO.js.map → paths-QQM74XYT.js.map} +0 -0
  236. /package/bundle/{peer-client-T44VI7NB.js.map → peer-client-D2F5QWRV.js.map} +0 -0
  237. /package/bundle/{notification-U6F5ZBSG.js.map → peer-config-5SUIBJLG.js.map} +0 -0
  238. /package/bundle/{paths-ZJYIDND2.js.map → phase-transport-BSGROTHY.js.map} +0 -0
  239. /package/bundle/{peer-config-D5A4454H.js.map → rollback-5RXXLUD6.js.map} +0 -0
  240. /package/bundle/{self-healer-utils-7NFH22VJ.js.map → self-healer-utils-WPKOVXJD.js.map} +0 -0
  241. /package/bundle/{phase-transport-FEZ4SIJJ.js.map → skill-stats-NHNH47QW.js.map} +0 -0
  242. /package/bundle/{skill-stats-IPVKMWN3.js.map → sleep-K7EXAFGW.js.map} +0 -0
  243. /package/bundle/{sleep-BPWX3FCN.js.map → soul-bundle-QTPWDJB2.js.map} +0 -0
  244. /package/bundle/{soul-bundle-BRIUDEQ2.js.map → soul-loader-LCPTN4PK.js.map} +0 -0
  245. /package/bundle/{sse-parser-anthropic-H42TTLBD.js.map → sse-parser-anthropic-PYDJM3UC.js.map} +0 -0
  246. /package/bundle/{sse-parser-responses-WG2LY2ML.js.map → sse-parser-responses-FYT7A5WT.js.map} +0 -0
  247. /package/bundle/{ssrf-guard-E2KBBC5E.js.map → ssrf-guard-R4P5OCTO.js.map} +0 -0
  248. /package/bundle/{start-CBVKNEAT.js.map → start-4DNURGIY.js.map} +0 -0
  249. /package/bundle/{soul-loader-GBXJ7EBH.js.map → stt-YN77NND6.js.map} +0 -0
  250. /package/bundle/{stt-CF3CPFDC.js.map → subagent-runtime-FQAT3564.js.map} +0 -0
  251. /package/bundle/{system-message-TALP6GP2.js.map → system-message-BRU267FW.js.map} +0 -0
  252. /package/bundle/{subagent-runtime-4MTYUBIZ.js.map → task-failure-buffer-DPM5MWZ5.js.map} +0 -0
  253. /package/bundle/{task-store-LC7ZMS72.js.map → task-store-VCBHAB43.js.map} +0 -0
  254. /package/bundle/{tool-registry-T7XLTI2Q.js.map → tool-registry-5PXNSYOI.js.map} +0 -0
  255. /package/bundle/{tool-sandbox-OZMXJZLQ.js.map → tool-sandbox-TLAL55QP.js.map} +0 -0
  256. /package/bundle/{transport-config-G5NKQXPJ.js.map → transport-config-JIKHB7GT.js.map} +0 -0
  257. /package/bundle/{update-check-O5MS6B3L.js.map → user-registry-PEFDZ5AV.js.map} +0 -0
@@ -1,7 +0,0 @@
1
- {
2
- "version": 3,
3
- "sources": ["../src/components/transport/direct-api-transport.ts", "../src/components/retry.ts", "../src/components/transport/conversation-session.ts", "../src/components/transport/sse-parser.ts", "../src/components/transport/transport-utils.ts"],
4
- "sourcesContent": ["import { getEnv } from \"../env-schema.js\";\n/**\n * Direct API Transport \u2014 talks to any OpenAI-compatible endpoint.\n * Implements IKiroTransport with its own agent loop (send \u2192 stream \u2192 tools \u2192 loop).\n */\n\nimport { logInfo, logWarn, logDebug, logTrace } from \"../logger.js\";\nimport { logAndSwallow } from \"../log-and-swallow.js\";\nimport { withRetry, isFatal } from \"../retry.js\";\nimport { ConversationSession, type ToolCall, type ContentPart } from \"./conversation-session.js\";\nimport { parseSSEStream, type SSEToolCallDelta } from \"./sse-parser.js\";\nimport { getToolSchemas, executeToolCall } from \"./tool-registry.js\";\nimport { classifyError } from \"./model-health-registry.js\";\nimport { normalizeToolCalls, parseErrorStatus, parseRetryAfter, parseUsageLimitCooldown } from \"./transport-utils.js\";\nimport { recordUsage } from \"../usage-tracker.js\";\nimport type { FallbackPolicy } from \"./fallback-policy.js\";\nimport type { IKiroTransport } from \"./kiro-transport.js\";\n\nconst TAG = \"direct-api\";\n\n\nexport interface DirectApiConfig {\n endpoint: string;\n apiKey?: string;\n model: string;\n maxContext: number;\n maxOutput: number;\n maxTurns: number;\n apiFormat?: \"chat\" | \"responses\" | \"anthropic\";\n thinking?: { style: \"effort\"; default: string } | { style: \"extended\"; default: number };\n fallbacks?: Array<{ endpoint: string; apiKey?: string; model: string; maxContext?: number }>;\n}\n\nexport { normalizeToolCalls } from \"./transport-utils.js\";\n\nexport class DirectApiTransport implements IKiroTransport {\n private readonly config: DirectApiConfig;\n private readonly sessions = new Map<string, ConversationSession>();\n private readonly abortControllers = new Map<string, AbortController>();\n private systemPrompt = \"\";\n private _contextPercent = -1;\n private _lastAnswer = \"\";\n private _timeoutOverrideMs: number | null = null;\n private _toolCallsSucceeded = 0;\n private _intermediateText = \"\";\n private _promptStartedAt: number | null = null;\n private _lastActivityAt: number | null = null;\n private activeEndpoint: string;\n private activeApiKey?: string;\n private activeModel: string;\n private _lastPromptTokens = 0;\n private _activeSessionKey = \"\";\n private _activeUserId = \"master\";\n\n /** Currently active model (may differ from config if on fallback). */\n get currentModel(): string { return this.activeModel; }\n\n onIntermediateResponse?: (text: string) => void;\n onToolCallStart?: (toolName: string) => void;\n onSegmentBreak?: (text: string) => void;\n /** Sandbox policy for tool access control (#681). Set by pipeline before prompt. */\n sandboxPolicy?: import(\"../tool-sandbox.js\").SandboxPolicy;\n /** Called when fallback model is selected \u2014 send notification before response. */\n onFallback?: (model: string, ctxPercent: number, reason?: string) => void;\n /** Cooperative pause check \u2014 if returns true, agent loop breaks between tool calls. */\n isPaused?: () => boolean;\n /** Returns a pending instruction from parent, if any. Consumed once. */\n getPendingInstruction?: () => string | undefined;\n\n /** Context orchestrator \u2014 when set, messages are built from DB instead of in-memory session. */\n contextOrchestrator?: import(\"abmind\").ContextOrchestrator;\n\n private policy: FallbackPolicy | null;\n private emergencyOverride: { endpoint: string; apiKey?: string; model: string; maxContext: number } | null = null;\n\n /** Activate emergency (hailMary) mode \u2014 next prompts bypass the fallback policy. */\n setEmergencyMode(override: { endpoint: string; apiKey?: string; model: string; maxContext: number } | null): void {\n if (!override && !this.emergencyOverride) return; // no-op if already off\n this.emergencyOverride = override;\n if (override) logWarn(TAG, `\uD83D\uDEA8 EMERGENCY MODE: using ${override.model} (paid) \u2014 bypassing fallback chain`);\n else { this.activeModel = this.config.model; logInfo(TAG, `Emergency mode cleared \u2014 restored ${this.config.model}`); }\n }\n\n /** True if emergency (hailMary) mode is active. */\n get isEmergencyMode(): boolean { return this.emergencyOverride !== null; }\n\n constructor(config: DirectApiConfig, policy?: FallbackPolicy) {\n this.config = config;\n this.activeEndpoint = config.endpoint;\n this.activeApiKey = config.apiKey;\n this.activeModel = config.model;\n this.policy = policy ?? null;\n }\n\n async initialize(): Promise<void> {\n const count = this.policy ? this.policy.candidates.length : (this.config.fallbacks?.length ?? 0) + 1;\n const fb = count > 1 ? ` (+${count - 1} fallback${count > 2 ? \"s\" : \"\"})` : \"\";\n logInfo(TAG, `\uD83D\uDD0C Direct API transport (${this.config.endpoint}, model: ${this.config.model}${fb})`);\n }\n\n setSystemPrompt(prompt: string): void {\n this.systemPrompt = prompt;\n }\n\n async sendPrompt(sessionKey: string, message: string, image?: { mime: string; base64: string }, userId?: string): Promise<string> {\n const session = this.getOrCreateSession(sessionKey);\n this._activeSessionKey = sessionKey;\n this._activeUserId = userId || \"master\";\n\n // If context orchestrator is active, rebuild messages from DB\n if (this.contextOrchestrator) {\n try {\n const ctx = await this.contextOrchestrator.getContext(sessionKey, this.config.maxContext);\n // Replace session messages with DB-backed context + system prompt\n session.messages = [\n { role: \"system\" as const, content: this.systemPrompt },\n ...ctx.messages.map(m => ({ role: m.role as \"user\" | \"assistant\" | \"tool\", content: m.content })),\n ];\n if (ctx.compacted) logDebug(TAG, `Context compacted for ${sessionKey}`);\n } catch (err) {\n logWarn(TAG, `Context engine failed, falling back to in-memory: ${err}`);\n }\n }\n\n session.addUser(message, image);\n\n this._lastAnswer = \"\";\n this._toolCallsSucceeded = 0;\n this._intermediateText = \"\";\n this._promptStartedAt = Date.now();\n this._lastActivityAt = Date.now();\n\n const ac = new AbortController();\n this.abortControllers.set(sessionKey, ac);\n\n try {\n if (this.emergencyOverride) return await this.sendEmergency(session, ac.signal);\n if (!this.policy) throw new Error(\"DirectApiTransport requires a FallbackPolicy\");\n return await this.sendWithPolicy(session, ac.signal);\n } finally {\n // AfterPrompt hook \u2014 observe-only, fire-and-forget\n const durationMs = Date.now() - (this._promptStartedAt ?? Date.now());\n import(\"../hooks/hook-system.js\").then(({ hasHooks, fire }) => {\n if (!hasHooks(\"AfterPrompt\")) return;\n fire(\"AfterPrompt\", {\n event: \"AfterPrompt\", timestamp: new Date().toISOString(),\n sessionKey, platform: \"\", userId: \"\",\n model: this.activeModel, durationMs,\n inputTokens: this._lastPromptTokens || null,\n outputTokens: null, // not tracked per-prompt in DirectApi\n }).catch(err => logAndSwallow(TAG, \"fire AfterPrompt\", err));\n }).catch(err => logAndSwallow(TAG, \"import hook-system\", err));\n this._promptStartedAt = null;\n this.abortControllers.delete(sessionKey);\n }\n }\n\n private async sendEmergency(session: ConversationSession, signal: AbortSignal): Promise<string> {\n const em = this.emergencyOverride!;\n this.activeEndpoint = em.endpoint;\n this.activeApiKey = em.apiKey;\n this.activeModel = em.model;\n this._lastActivityAt = Date.now();\n logWarn(TAG, `\uD83D\uDEA8 Emergency mode: using ${em.model}`);\n const result = await this.agentLoop(session, signal);\n this._lastAnswer = result;\n return result;\n }\n\n private async sendWithPolicy(session: ConversationSession, signal: AbortSignal): Promise<string> {\n const policy = this.policy!;\n const failedAttempts: Array<{ model: string; kind: string; bucket: number }> = [];\n const isPrimary = (m: string): boolean => m === this.config.model;\n\n // Try each candidate via policy\n let candidate = policy.selectModel(this._lastPromptTokens);\n while (candidate) {\n this.activeEndpoint = candidate.endpoint;\n this.activeApiKey = candidate.apiKey;\n this.activeModel = candidate.model;\n this._lastActivityAt = Date.now();\n logDebug(TAG, `Trying model: ${candidate.model}`);\n\n if (!isPrimary(candidate.model) && this.onFallback) {\n const ctxPct = candidate.maxContext > 0 ? Math.round((this._lastPromptTokens / candidate.maxContext) * 100) : -1;\n const lastFail = failedAttempts[failedAttempts.length - 1];\n this.onFallback(candidate.model, ctxPct, lastFail?.kind);\n }\n\n try {\n const result = await this.agentLoop(session, signal);\n this._lastAnswer = result;\n if (!result || !result.trim()) {\n policy.recordError(candidate, \"empty\");\n } else {\n policy.recordSuccess(candidate);\n }\n return result;\n } catch (err) {\n if (signal.aborted) { session.rollbackToLastUser(); throw err; }\n const errMsg = err instanceof Error ? err.message : String(err);\n // Capability miss \u2014 strip image and retry text-only (#670)\n if (errMsg.includes(\"does not support image input\") || errMsg.includes(\"No endpoints found that support image\")) {\n session.rollbackToLastUser();\n // Strip image parts from session, retry same model text-only\n for (const m of session.messages) {\n if (Array.isArray(m.content)) {\n const textParts = (m.content as Array<{ type: string; text?: string }>).filter(p => p.type === \"text\");\n m.content = textParts.map(p => p.text ?? \"\").join(\"\\n\") || \"User sent an image (not supported by this model).\";\n }\n }\n logWarn(TAG, `${candidate.model} doesn't support images \u2014 retrying text-only`);\n if (this.onIntermediateResponse) this.onIntermediateResponse(\"\u26A0\uFE0F Model doesn't support images via this provider. Sending text-only.\\n\");\n try {\n const result = await this.agentLoop(session, signal);\n this._lastAnswer = result;\n policy.recordSuccess(candidate);\n return result;\n } catch (retryErr) {\n const retryStatus = this.parseErrorStatus(retryErr);\n const retryMsg = retryErr instanceof Error ? retryErr.message : String(retryErr);\n policy.recordError(candidate, classifyError(retryStatus, retryMsg));\n session.rollbackToLastUser();\n throw retryErr;\n }\n }\n const status = this.parseErrorStatus(err);\n const kind = classifyError(status, errMsg);\n const retryAfterMs = this.parseRetryAfter(err) ?? parseUsageLimitCooldown(errMsg);\n policy.recordError(candidate, kind, retryAfterMs);\n const bucket = policy.registry.getBucketLevel(candidate.model, candidate.endpoint);\n failedAttempts.push({ model: candidate.model, kind, bucket });\n session.rollbackToLastUser();\n logWarn(TAG, `${candidate.model} failed (${kind}, bucket: ${bucket}%${retryAfterMs ? `, retry-after: ${Math.round(retryAfterMs / 1000)}s` : \"\"}): ${errMsg.slice(0, 100)}`);\n }\n\n candidate = policy.selectModel(this._lastPromptTokens);\n }\n\n // Compaction fallback: pick smallest surviving candidate, truncate to fit\n const surviving = policy.survivingCandidates()\n .filter(c => c.maxContext > 0)\n .sort((a, b) => a.maxContext - b.maxContext);\n const smallest = surviving[0];\n\n if (smallest && this._lastPromptTokens > smallest.maxContext * 0.95) {\n logWarn(TAG, `Compacting session to fit ${smallest.model} (${smallest.maxContext} tokens)`);\n session.truncateToFit(smallest.maxContext);\n this.activeEndpoint = smallest.endpoint;\n this.activeApiKey = smallest.apiKey;\n this.activeModel = smallest.model;\n if (this.onFallback) {\n this.onFallback(`${smallest.model} (compacted)`, Math.round((session.estimateTokens() / smallest.maxContext) * 100));\n }\n try {\n const result = await this.agentLoop(session, signal);\n this._lastAnswer = result;\n if (!result || !result.trim()) policy.recordError(smallest, \"empty\");\n else policy.recordSuccess(smallest);\n return result;\n } catch (err) {\n const status = this.parseErrorStatus(err);\n const errMsg2 = err instanceof Error ? err.message : String(err);\n policy.recordError(smallest, classifyError(status, errMsg2));\n failedAttempts.push({ model: smallest.model, kind: classifyError(status, errMsg2), bucket: policy.registry.getBucketLevel(smallest.model, smallest.endpoint) });\n }\n }\n\n const summary = failedAttempts.map(a => ` - ${a.model}: ${a.kind} (bucket: ${a.bucket}%)`).join(\"\\n\");\n if (policy.lastDecision) {\n logDebug(TAG, `Last decision: ${JSON.stringify(policy.lastDecision)}`);\n }\n throw new Error(`All models exhausted:\\n${summary}`);\n }\n\n private async agentLoop(session: ConversationSession, signal: AbortSignal): Promise<string> {\n let zeroTokenRetries = 0;\n const loopStart = Date.now();\n for (let turn = 0; turn < this.config.maxTurns; turn++) {\n if (signal.aborted) throw new Error(\"Aborted\");\n if (this.isPaused?.()) return \"\u23F8 Session paused. Use `/session resume` to continue.\";\n\n const pendingInstruction = this.getPendingInstruction?.();\n if (pendingInstruction) session.addUser(pendingInstruction);\n\n const { content, toolCalls, usage } = await this.streamCompletion(session, signal);\n\n if (usage) {\n session.updateTokens(usage.prompt_tokens);\n this._contextPercent = session.contextPercent;\n this._lastPromptTokens = usage.prompt_tokens;\n this.contextOrchestrator?.onApiResponse(this._activeSessionKey, usage.prompt_tokens, this.config.maxContext);\n logTrace(TAG, `${this.activeModel} \u2014 ${usage.prompt_tokens}\u2192${usage.completion_tokens ?? 0} tokens, ${Date.now() - (this._lastActivityAt ?? Date.now())}ms`);\n recordUsage(this.activeModel, usage.prompt_tokens, usage.completion_tokens ?? 0);\n }\n\n if (toolCalls.length > 0) {\n session.addAssistant(content, toolCalls);\n logDebug(TAG, `Tool calls: ${toolCalls.map(tc => tc.function.name).join(\", \")}`);\n logTrace(TAG, `Tool args: ${toolCalls.map(tc => {\n // #621: redact abmind_store args based on classification\n if ((tc.function.name === \"abmind_store\" || tc.function.name === \"memory_store\") && /class(?:ification)?[\":\\s]+[23]/.test(tc.function.arguments)) {\n return `${tc.function.name}([REDACTED])`;\n }\n return `${tc.function.name}(${tc.function.arguments})`;\n }).join(\", \")}`);\n\n // Deliver pre-tool text immediately (segment break)\n if (content?.trim()) {\n this.onSegmentBreak?.(content.trim());\n }\n\n for (const tc of toolCalls) {\n if (signal.aborted) throw new Error(\"Aborted\");\n this._lastActivityAt = Date.now();\n this.onToolCallStart?.(tc.function.name ?? \"tool\");\n\n let args: Record<string, string>;\n try { args = JSON.parse(tc.function.arguments); } catch (err) { logAndSwallow(TAG, \"JSON.parse tool args\", err); args = {}; }\n\n const result = await executeToolCall(tc.function.name, args, { userId: this._activeUserId, signal, sandboxPolicy: this.sandboxPolicy });\n session.addToolResult(tc.id, tc.function.name, result);\n\n // #621: scrub secret values from conversation history after store\n if ((tc.function.name === \"abmind_store\" || tc.function.name === \"memory_store\") && parseInt(args.classification ?? args.class ?? \"1\", 10) >= 2) {\n const secretValue = args.content ?? args.value ?? args.translated;\n if (secretValue && secretValue.length > 4) {\n session.scrubFromHistory(secretValue);\n }\n }\n\n try { if (!JSON.parse(result).error) this._toolCallsSucceeded++; } catch (err) { logAndSwallow(TAG, \"JSON.parse tool result\", err); this._toolCallsSucceeded++; }\n }\n continue;\n }\n\n // No tool calls \u2014 final response\n const answer = content ?? \"\";\n // Budget-aware backoff on 0\u21920 token response (#732)\n if (!answer && usage && usage.prompt_tokens === 0) {\n const elapsed = Date.now() - loopStart;\n const remaining = (this._timeoutOverrideMs ?? 60_000) - elapsed;\n const retryDelay = 5000 + (zeroTokenRetries * 3000);\n if (remaining > retryDelay + 5000) {\n zeroTokenRetries++;\n logWarn(TAG, `API returned 0 tokens \u2014 retry #${zeroTokenRetries} after ${retryDelay / 1000}s (${Math.round(remaining / 1000)}s left)`);\n await new Promise(r => setTimeout(r, retryDelay));\n continue;\n }\n }\n session.addAssistant(answer);\n return answer;\n }\n\n logWarn(TAG, `Max turns (${this.config.maxTurns}) reached`);\n const last = session.messages.at(-1)?.content;\n return (typeof last === \"string\" ? last : null) ?? \"(max turns reached)\";\n }\n\n private parseErrorStatus(err: unknown): number { return parseErrorStatus(err); }\n\n /** Extract Retry-After from error (seconds or date). Returns ms or undefined. */\n private parseRetryAfter(err: unknown): number | undefined { return parseRetryAfter(err); }\n\n private async streamCompletion(\n session: ConversationSession,\n signal: AbortSignal,\n ): Promise<{ content: string | null; toolCalls: ToolCall[]; usage: { prompt_tokens: number; completion_tokens: number } | null }> {\n // Compose pipeline signal (user /stop) with per-request timeout\n const timeoutCtrl = new AbortController();\n const timer = setTimeout(() => timeoutCtrl.abort(new Error(\"model API timeout\")), this._timeoutOverrideMs ?? getEnv().modelApiTimeoutMs);\n const composed = AbortSignal.any([signal, timeoutCtrl.signal]);\n\n try {\n // Responses API format (#465, streaming #472)\n if (this.config.apiFormat === \"responses\") {\n const { toResponsesRequest } = await import(\"./responses-adapter.js\");\n const { parseResponsesSSE } = await import(\"./sse-parser-responses.js\");\n const msgs = session.messages.map(m => ({ role: m.role, content: m.content ?? \"\" as string | ContentPart[] }));\n const reqBody = { ...toResponsesRequest(this.activeModel, msgs, getToolSchemas(this.sandboxPolicy), this.config.maxOutput), stream: true };\n const hdrs: Record<string, string> = { \"Content-Type\": \"application/json\" };\n if (this.activeApiKey) hdrs[\"Authorization\"] = `Bearer ${this.activeApiKey}`;\n const res = await fetch(`${this.activeEndpoint}/responses`, {\n method: \"POST\", headers: hdrs, body: JSON.stringify(reqBody), signal: composed,\n });\n if (!res.ok) { const text = await res.text().catch(err => { logAndSwallow(TAG, \"read error body\", err); return \"\"; }); throw new Error(`API error ${res.status}: ${text.slice(0, 500)}`); }\n\n let content = \"\";\n let usage: { prompt_tokens: number; completion_tokens: number } | null = null;\n const toolCallAcc = new Map<string, { id: string; name: string; arguments: string }>();\n for await (const event of parseResponsesSSE(res, composed)) {\n this._lastActivityAt = Date.now();\n if (event.type === \"chunk\") { content += event.content; this._intermediateText += event.content; this.onIntermediateResponse?.(event.content); }\n else if (event.type === \"tool_call_delta\") { this.accumulateToolCall(toolCallAcc, event); }\n else if (event.type === \"done\") { usage = event.usage; }\n }\n clearTimeout(timer);\n const toolCalls = this.finalizeToolCalls(toolCallAcc);\n return { content: content || null, toolCalls, usage };\n }\n\n // Anthropic Messages API format (#467, streaming #472)\n if (this.config.apiFormat === \"anthropic\") {\n const { toAnthropicRequest, buildAnthropicHeaders } = await import(\"./anthropic-adapter.js\");\n const { parseAnthropicSSE } = await import(\"./sse-parser-anthropic.js\");\n const msgs = session.messages.map(m => ({ role: m.role, content: m.content ?? \"\" as string | ContentPart[], tool_call_id: m.tool_call_id }));\n const reqBody = { ...toAnthropicRequest(this.activeModel, msgs, this.config.maxOutput, getToolSchemas(this.sandboxPolicy)), stream: true };\n const hdrs = buildAnthropicHeaders(this.activeApiKey ?? \"\");\n const res = await fetch(`${this.activeEndpoint}/messages`, {\n method: \"POST\", headers: hdrs, body: JSON.stringify(reqBody), signal: composed,\n });\n if (!res.ok) { const text = await res.text().catch(err => { logAndSwallow(TAG, \"read error body\", err); return \"\"; }); throw new Error(`API error ${res.status}: ${text.slice(0, 500)}`); }\n\n let content = \"\";\n let usage: { prompt_tokens: number; completion_tokens: number } | null = null;\n const toolCallAcc = new Map<string, { id: string; name: string; arguments: string }>();\n for await (const event of parseAnthropicSSE(res, composed)) {\n this._lastActivityAt = Date.now();\n if (event.type === \"chunk\") { content += event.content; this._intermediateText += event.content; this.onIntermediateResponse?.(event.content); }\n else if (event.type === \"tool_call_delta\") { this.accumulateToolCall(toolCallAcc, event); }\n else if (event.type === \"done\") { usage = event.usage; }\n }\n clearTimeout(timer);\n const toolCalls = this.finalizeToolCalls(toolCallAcc);\n return { content: content || null, toolCalls, usage };\n }\n\n const body: Record<string, unknown> = {\n model: this.activeModel,\n messages: session.messages,\n tools: getToolSchemas(this.sandboxPolicy),\n max_tokens: this.config.maxOutput,\n stream: true,\n stream_options: { include_usage: true },\n };\n\n // #466: inject thinking/reasoning parameters\n if (this.config.thinking) {\n if (this.config.thinking.style === \"effort\") {\n body.reasoning_effort = this.config.thinking.default;\n } else if (this.config.thinking.style === \"extended\") {\n body.thinking = { type: \"enabled\", budget_tokens: this.config.thinking.default };\n }\n }\n\n const headers: Record<string, string> = { \"Content-Type\": \"application/json\" };\n if (this.activeApiKey) headers[\"Authorization\"] = `Bearer ${this.activeApiKey}`;\n\n const response = await withRetry(\n async () => {\n const res = await fetch(`${this.activeEndpoint}/chat/completions`, {\n method: \"POST\",\n headers,\n body: JSON.stringify(body),\n signal: composed,\n });\n if (!res.ok) {\n const text = await res.text().catch(err => { logAndSwallow(TAG, \"read error body\", err); return \"\"; });\n throw new Error(`API error ${res.status}: ${text.slice(0, 500)}`);\n }\n return res;\n },\n {\n attempts: 3, minDelayMs: 3000,\n isRecoverable: (err) => {\n const msg = err instanceof Error ? err.message : String(err);\n if (/model API timeout/.test(msg)) return false;\n // Don't retry 429/401/402/403 \u2014 let the bucket loop handle model switching\n if (/API error (429|401|402|403)/.test(msg)) return false;\n return !isFatal(err);\n },\n },\n );\n\n let content = \"\";\n const toolCallAccumulator = new Map<string, { id: string; name: string; arguments: string }>();\n let usage: { prompt_tokens: number; completion_tokens: number } | null = null;\n\n for await (const event of parseSSEStream(response, composed)) {\n this._lastActivityAt = Date.now();\n\n switch (event.type) {\n case \"chunk\":\n content += event.content;\n this._intermediateText += event.content;\n this.onIntermediateResponse?.(event.content);\n break;\n\n case \"tool_call_delta\":\n this.accumulateToolCall(toolCallAccumulator, event);\n break;\n\n case \"done\":\n usage = event.usage;\n break;\n }\n }\n\n const toolCalls = this.finalizeToolCalls(toolCallAccumulator);\n\n return { content: content || null, toolCalls, usage };\n } finally {\n clearTimeout(timer);\n }\n }\n\n /** Accumulate streaming tool call deltas by ID (Ollama-safe: tracks by ID not index). */\n private accumulateToolCall(\n acc: Map<string, { id: string; name: string; arguments: string }>,\n delta: SSEToolCallDelta,\n ): void {\n // Use ID if available, fall back to index-based key\n const key = delta.id ?? `idx:${delta.index}`;\n const existing = acc.get(key);\n if (existing) {\n if (delta.arguments) existing.arguments += delta.arguments;\n } else {\n acc.set(key, { id: delta.id ?? `call_${acc.size}`, name: delta.name ?? \"\", arguments: delta.arguments ?? \"\" });\n }\n }\n\n /**\n * Convert accumulated tool call map \u2192 normalized ToolCall array.\n * Handles model fragmentation: some models (nemotron, mistral-free) split a\n * single tool call across multiple SSE entries with different indices/IDs.\n * Pattern: [name=\"execute_bash\" args=\"{}\"], [name=\"\" args=\"\"], [name=\"\" args='{\"command\":\"...\"}']\n * Fix: merge adjacent unnamed entries' args into the preceding named entry.\n */\n private finalizeToolCalls(acc: Map<string, { id: string; name: string; arguments: string }>): ToolCall[] {\n const raw: ToolCall[] = [...acc.values()].map(tc => ({\n id: tc.id, type: \"function\" as const, function: { name: tc.name, arguments: tc.arguments },\n }));\n return normalizeToolCalls(raw);\n }\n\n private getOrCreateSession(sessionKey: string): ConversationSession {\n let session = this.sessions.get(sessionKey);\n if (!session) {\n session = new ConversationSession(this.systemPrompt, this.config.maxContext);\n this.sessions.set(sessionKey, session);\n }\n return session;\n }\n\n async resetSession(sessionKey: string): Promise<void> {\n const session = this.sessions.get(sessionKey);\n if (session) session.reset(this.systemPrompt);\n else this.sessions.delete(sessionKey);\n logInfo(TAG, `Session ${sessionKey} reset`);\n }\n\n async sendInterrupt(): Promise<void> {\n for (const ac of this.abortControllers.values()) ac.abort();\n }\n\n destroy(): void {\n for (const ac of this.abortControllers.values()) ac.abort();\n this.sessions.clear();\n this.abortControllers.clear();\n }\n\n get isReady(): boolean { return true; }\n get contextPercent(): number { return this._contextPercent; }\n get answerOnly(): string { return this._lastAnswer; }\n get toolCallsSucceeded(): number { return this._toolCallsSucceeded; }\n\n /** Hot-swap the active model. Takes effect on next API call. */\n setModel(model: string): void {\n this.activeModel = model;\n (this.config as { model: string }).model = model;\n logInfo(TAG, `Model switched (user): ${model}`);\n }\n\n /** Hot-swap provider+model+policy. Rejects if prompt is in flight. */\n switchProvider(opts: { endpoint: string; apiKey?: string; model: string; maxContext: number; policy: FallbackPolicy }): void {\n if (this._promptStartedAt !== null) {\n throw new Error(\"Cannot switch provider while a prompt is in progress \u2014 try after the response\");\n }\n this.activeEndpoint = opts.endpoint;\n this.activeApiKey = opts.apiKey;\n this.activeModel = opts.model;\n (this.config as { model: string; maxContext: number }).model = opts.model;\n (this.config as { maxContext: number }).maxContext = opts.maxContext;\n this.policy = opts.policy;\n logInfo(TAG, `Provider switched: ${opts.model} @ ${opts.endpoint} (maxCtx=${opts.maxContext})`);\n }\n\n /** Get current active model name. */\n getModel(): string { return this.activeModel; }\n get intermediateDeliveredText(): string { return this._intermediateText; }\n get transportCommands(): string[] { return []; }\n\n // Watchdog support\n get promptStartedAt(): number | null { return this._promptStartedAt; }\n setTimeoutOverride(ms: number | null): void { this._timeoutOverrideMs = ms; }\n\n get lastActivityAt(): number | null { return this._lastActivityAt; }\n\n private readonly _stuckTimeout = getEnv().watchdogSilentSec * 1000;\n\n async healthCheck(): Promise<void> {\n if (!this._promptStartedAt) return;\n const idle = Date.now() - (this._lastActivityAt ?? this._promptStartedAt);\n if (idle > this._stuckTimeout) {\n logWarn(TAG, `[transport-health] Prompt stuck (${Math.round(idle / 1000)}s idle) \u2014 aborting`);\n await this.sendInterrupt();\n }\n }\n}\n", "/**\n * Generic retry utility with escalating backoff, jitter, and fatal error detection.\n */\n\nexport interface RetryPolicy {\n /** Max attempts (including first). Default: 3 */\n attempts?: number;\n /** Initial delay in ms. Default: 300 */\n minDelayMs?: number;\n /** Max delay cap in ms. Default: 30_000 */\n maxDelayMs?: number;\n /** Jitter factor 0-1 (0.1 = \u00B110%). Default: 0.1 */\n jitter?: number;\n /** Return false to stop retrying (fatal error). Default: always retry. */\n isRecoverable?: (err: unknown) => boolean;\n /** Extract delay hint from error (e.g. rate limit header). */\n getDelayHint?: (err: unknown) => number | undefined;\n /** Called before each retry. */\n onAttempt?: (info: { attempt: number; maxAttempts: number; err: unknown; delayMs: number }) => void;\n}\n\nconst DEFAULTS = { attempts: 3, minDelayMs: 300, maxDelayMs: 30_000, jitter: 0.1 };\n\n/** Known-fatal error patterns \u2014 don't retry these. */\nexport const FATAL_PATTERNS = [\n /auth.*fail|invalid.*key|unauthorized/i,\n /model.*not found|not supported/i,\n /account.*suspended|quota.*exceeded/i,\n /bot was blocked/i,\n];\n\n/** Check if an error matches a known-fatal pattern. */\nexport function isFatal(err: unknown): boolean {\n const msg = err instanceof Error ? err.message : String(err);\n return FATAL_PATTERNS.some(p => p.test(msg));\n}\n\n/** Execute `fn` with retry, escalating backoff, and jitter. */\nexport async function withRetry<T>(fn: () => Promise<T>, policy?: RetryPolicy): Promise<T> {\n const attempts = Math.max(1, policy?.attempts ?? DEFAULTS.attempts);\n const minDelay = Math.max(0, policy?.minDelayMs ?? DEFAULTS.minDelayMs);\n const maxDelay = Math.max(minDelay, policy?.maxDelayMs ?? DEFAULTS.maxDelayMs);\n const jitter = Math.max(0, Math.min(1, policy?.jitter ?? DEFAULTS.jitter));\n const isRecoverable = policy?.isRecoverable ?? ((err: unknown) => !isFatal(err));\n\n let lastErr: unknown;\n for (let i = 0; i < attempts; i++) {\n try {\n return await fn();\n } catch (err) {\n lastErr = err;\n if (i === attempts - 1) break;\n if (!isRecoverable(err)) break;\n\n const hint = policy?.getDelayHint?.(err);\n const baseDelay = hint ?? Math.min(maxDelay, minDelay * Math.pow(2, i));\n const offset = jitter > 0 ? (Math.random() * 2 - 1) * jitter * baseDelay : 0;\n const delayMs = Math.max(0, Math.round(baseDelay + offset));\n\n policy?.onAttempt?.({ attempt: i + 1, maxAttempts: attempts, err, delayMs });\n await new Promise(r => setTimeout(r, delayMs));\n }\n }\n throw lastErr;\n}\n", "/**\n * Per-session conversation state for DirectApiTransport.\n * Manages message history and token tracking.\n */\n\nexport type ContentPart =\n | { type: \"text\"; text: string }\n | { type: \"image_url\"; image_url: { url: string } };\n\nexport type ChatMessage = {\n role: \"system\" | \"user\" | \"assistant\" | \"tool\";\n content: string | ContentPart[] | null;\n tool_calls?: ToolCall[];\n tool_call_id?: string;\n name?: string;\n};\n\nexport type ToolCall = {\n id: string;\n type: \"function\";\n function: { name: string; arguments: string };\n};\n\nexport class ConversationSession {\n messages: ChatMessage[] = [];\n totalPromptTokens = 0;\n private readonly maxContext: number;\n\n constructor(systemPrompt: string, maxContext: number) {\n this.maxContext = maxContext;\n this.messages.push({ role: \"system\", content: systemPrompt });\n }\n\n addUser(content: string, image?: { mime: string; base64: string }): void {\n if (image) {\n this.messages.push({ role: \"user\", content: [\n { type: \"image_url\", image_url: { url: `data:${image.mime};base64,${image.base64}` } },\n { type: \"text\", text: content },\n ]});\n } else {\n this.messages.push({ role: \"user\", content });\n }\n }\n\n addAssistant(content: string | null, toolCalls?: ToolCall[]): void {\n const msg: ChatMessage = { role: \"assistant\", content };\n if (toolCalls?.length) msg.tool_calls = toolCalls;\n this.messages.push(msg);\n }\n\n addToolResult(toolCallId: string, name: string, content: string): void {\n this.messages.push({ role: \"tool\", content, tool_call_id: toolCallId, name });\n }\n\n updateTokens(promptTokens: number): void {\n this.totalPromptTokens = promptTokens;\n }\n\n get contextPercent(): number {\n if (this.maxContext <= 0) return 0;\n return Math.round((this.totalPromptTokens / this.maxContext) * 100);\n }\n\n reset(systemPrompt: string): void {\n this.messages = [{ role: \"system\", content: systemPrompt }];\n this.totalPromptTokens = 0;\n }\n\n /** Roll back to last user message \u2014 remove everything after it for clean fallback. */\n rollbackToLastUser(): void {\n for (let i = this.messages.length - 1; i >= 0; i--) {\n if (this.messages[i]!.role === \"user\") {\n this.messages.splice(i + 1);\n return;\n }\n }\n }\n\n /** Rough token estimate: ~4 chars per token. */\n estimateTokens(): number {\n return Math.round(this.messages.reduce((sum, m) => sum + (typeof m.content === \"string\" ? m.content.length : 0), 0) / 4);\n }\n\n /** Drop oldest non-system messages until estimated tokens fit within limit. */\n truncateToFit(maxTokens: number): void {\n while (this.messages.length > 2 && this.estimateTokens() > maxTokens * 0.85) {\n this.messages.splice(1, 1); // remove oldest after system prompt\n }\n this.totalPromptTokens = this.estimateTokens();\n }\n\n /** #621: Replace a literal secret value with [REDACTED] across all messages. */\n scrubFromHistory(value: string): void {\n for (const msg of this.messages) {\n if (typeof msg.content === \"string\" && msg.content.includes(value)) {\n msg.content = msg.content.replaceAll(value, \"[REDACTED]\");\n }\n }\n }\n}\n", "/**\n * SSE stream parser for OpenAI-compatible chat completions.\n * Parses Server-Sent Events into typed events.\n */\n\nimport { logAndSwallow } from \"../log-and-swallow.js\";\n\nconst TAG = \"sse_parser\";\n\nexport type SSEChunkEvent = { type: \"chunk\"; content: string };\nexport type SSEToolCallDelta = { type: \"tool_call_delta\"; index: number; id?: string; name?: string; arguments?: string };\nexport type SSEDoneEvent = { type: \"done\"; usage: { prompt_tokens: number; completion_tokens: number } | null };\nexport type SSEEvent = SSEChunkEvent | SSEToolCallDelta | SSEDoneEvent;\n\nconst STALE_TIMEOUT_MS = 90_000;\n\nexport async function* parseSSEStream(\n response: Response,\n signal: AbortSignal,\n): AsyncGenerator<SSEEvent> {\n const reader = response.body?.getReader();\n if (!reader) throw new Error(\"Response body is not readable\");\n\n const decoder = new TextDecoder();\n let buffer = \"\";\n let lastChunkAt = Date.now();\n let staleTimer: ReturnType<typeof setTimeout> | null = null;\n\n const resetStaleTimer = (): void => {\n lastChunkAt = Date.now();\n if (staleTimer) clearTimeout(staleTimer);\n staleTimer = setTimeout(() => {\n if (Date.now() - lastChunkAt >= STALE_TIMEOUT_MS) {\n reader.cancel(\"stale stream\").catch(err => logAndSwallow(TAG, \"reader.cancel stale\", err));\n }\n }, STALE_TIMEOUT_MS);\n };\n\n try {\n resetStaleTimer();\n while (true) {\n if (signal.aborted) break;\n const { done, value } = await reader.read();\n if (done) break;\n\n resetStaleTimer();\n buffer += decoder.decode(value, { stream: true });\n\n const lines = buffer.split(\"\\n\");\n buffer = lines.pop() ?? \"\";\n\n for (const line of lines) {\n if (!line.startsWith(\"data: \")) continue;\n const data = line.slice(6).trim();\n if (data === \"[DONE]\") return;\n\n let parsed: Record<string, unknown>;\n try { parsed = JSON.parse(data); } catch (err) { logAndSwallow(TAG, \"JSON.parse SSE chunk\", err); continue; }\n\n // Usage in final chunk (stream_options: { include_usage: true })\n if (parsed[\"usage\"]) {\n const u = parsed[\"usage\"] as Record<string, number>;\n yield { type: \"done\", usage: { prompt_tokens: u[\"prompt_tokens\"] ?? 0, completion_tokens: u[\"completion_tokens\"] ?? 0 } };\n }\n\n const choices = parsed[\"choices\"] as Array<Record<string, unknown>> | undefined;\n if (!choices?.length) continue;\n const delta = choices[0]![\"delta\"] as Record<string, unknown> | undefined;\n if (!delta) continue;\n\n // Text content\n if (typeof delta[\"content\"] === \"string\" && delta[\"content\"]) {\n yield { type: \"chunk\", content: delta[\"content\"] as string };\n }\n\n // Tool calls\n const toolCalls = delta[\"tool_calls\"] as Array<Record<string, unknown>> | undefined;\n if (toolCalls) {\n for (const tc of toolCalls) {\n const fn = tc[\"function\"] as Record<string, string> | undefined;\n yield {\n type: \"tool_call_delta\",\n index: (tc[\"index\"] as number) ?? 0,\n id: tc[\"id\"] as string | undefined,\n name: fn?.[\"name\"],\n arguments: fn?.[\"arguments\"],\n };\n }\n }\n }\n }\n } finally {\n if (staleTimer) clearTimeout(staleTimer);\n reader.releaseLock();\n }\n}\n", "/**\n * transport-utils.ts \u2014 Pure utility functions for the transport layer.\n * No I/O, no state, no side effects (except logging).\n */\n\nimport { logWarn } from \"../logger.js\";\nimport type { ToolCall } from \"./conversation-session.js\";\n\nconst TAG = \"direct-api\";\n\n/**\n * Normalize tool calls from models that fragment a single call across multiple entries.\n * Pattern: [name=\"execute_bash\" args=\"{}\"], [name=\"\" args=\"\"], [name=\"\" args='{\"command\":\"...\"}']\n * Fix: merge next unnamed entry's args into preceding named entry. Drop remaining unnamed.\n */\nexport function normalizeToolCalls(raw: ToolCall[]): ToolCall[] {\n if (raw.length <= 1) return raw;\n\n const result: ToolCall[] = [];\n for (let i = 0; i < raw.length; i++) {\n const tc = raw[i]!;\n if (tc.function.name) {\n if (!tc.function.arguments || tc.function.arguments === \"{}\") {\n // Look ahead for the next unnamed entry with real args\n for (let j = i + 1; j < raw.length; j++) {\n const next = raw[j]!;\n if (next.function.name) break; // hit another named entry, stop\n if (next.function.arguments && next.function.arguments !== \"{}\") {\n tc.function.arguments = next.function.arguments;\n i = j; // skip all entries up to and including the merged one\n break;\n }\n }\n }\n result.push(tc);\n }\n }\n\n if (result.length !== raw.length) {\n logWarn(TAG, `Normalized ${raw.length} tool call entries \u2192 ${result.length} (model fragmentation): ${raw.map(tc => `${tc.function.name || \"(unnamed)\"}(${tc.function.arguments.slice(0, 60)})`).join(\", \")}`);\n }\n return result;\n}\n\n/** Extract HTTP status code from error message. Returns 0 if not found. */\nexport function parseErrorStatus(err: unknown): number {\n const msg = err instanceof Error ? err.message : String(err);\n const m = /API error (\\d+)/.exec(msg);\n return m ? parseInt(m[1]!, 10) : 0;\n}\n\n/** Extract Retry-After from error (seconds or date). Returns ms or undefined. */\nexport function parseRetryAfter(err: unknown): number | undefined {\n const msg = err instanceof Error ? err.message : String(err);\n const jsonMatch = /retry[_-]after[\"\\s:]+(\\d+(?:\\.\\d+)?)/i.exec(msg);\n if (jsonMatch) return Math.ceil(parseFloat(jsonMatch[1]!) * 1000);\n const resetMatch = /x-ratelimit-reset[\"\\s:]+(\\d{10,13})/i.exec(msg);\n if (resetMatch) {\n const ts = parseInt(resetMatch[1]!, 10);\n const ms = ts < 1e12 ? ts * 1000 : ts;\n const delta = ms - Date.now();\n return delta > 0 ? delta : undefined;\n }\n return undefined;\n}\n\n/** Detect \"day/week/month limit\" in error message and return cooldown ms. */\nexport function parseUsageLimitCooldown(message: string): number | undefined {\n const lower = message.toLowerCase();\n if (!lower.includes(\"limit\")) return undefined;\n if (lower.includes(\"day\")) {\n const tomorrow = new Date();\n tomorrow.setDate(tomorrow.getDate() + 1);\n tomorrow.setHours(0, 0, 0, 0);\n return tomorrow.getTime() - Date.now();\n }\n if (lower.includes(\"week\")) {\n const now = new Date();\n const daysUntilMonday = (8 - now.getDay()) % 7 || 7;\n const monday = new Date(now);\n monday.setDate(monday.getDate() + daysUntilMonday);\n monday.setHours(0, 0, 0, 0);\n return monday.getTime() - Date.now();\n }\n if (lower.includes(\"month\")) {\n const now = new Date();\n const nextMonth = new Date(now.getFullYear(), now.getMonth() + 1, 1);\n return nextMonth.getTime() - Date.now();\n }\n return undefined;\n}\n"],
5
- "mappings": "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",
6
- "names": ["TAG", "TAG", "content", "usage", "toolCalls"]
7
- }
@@ -1,7 +0,0 @@
1
- {
2
- "version": 3,
3
- "sources": ["../src/platforms/discord/discord-api.ts", "../src/platforms/discord/discord-poller.ts", "../src/platforms/discord/discord-adapter.ts"],
4
- "sourcesContent": ["import { logAndSwallow } from \"../../components/log-and-swallow.js\";\nimport {\n Client,\n GatewayIntentBits,\n Partials,\n type Message,\n type MessageReaction,\n type User,\n type TextChannel,\n} from \"discord.js\";\nimport { logInfo, logError, logDebug, logWarn } from \"../../components/logger.js\";\n\nconst TAG = \"DiscordApi\";\n\n/**\n * Thin wrapper around discord.js Client.\n * Provides typed methods for Gateway connection, message listening, and sending.\n */\nexport class DiscordApi {\n private readonly client: Client;\n private readonly token: string;\n private ready = false;\n\n constructor(botToken: string) {\n this.token = botToken;\n this.client = new Client({\n intents: [\n GatewayIntentBits.Guilds,\n GatewayIntentBits.GuildMessages,\n GatewayIntentBits.GuildMessageReactions,\n GatewayIntentBits.DirectMessages,\n GatewayIntentBits.MessageContent,\n ],\n partials: [Partials.Channel, Partials.Message, Partials.Reaction],\n });\n\n this.client.on(\"clientReady\", () => {\n this.ready = true;\n logInfo(TAG, `Connected as ${this.client.user?.tag ?? \"unknown\"}`);\n });\n\n this.client.on(\"error\", (err) => {\n logError(TAG, \"Client error\", err);\n });\n }\n\n /** Connect to the Discord Gateway. Resolves when the client is ready. */\n async connect(): Promise<void> {\n logInfo(TAG, \"Connecting to Discord Gateway\u2026\");\n return new Promise<void>((resolve, reject) => {\n const onReady = () => {\n cleanup();\n resolve();\n };\n const onError = (err: Error) => {\n cleanup();\n reject(err);\n };\n const cleanup = () => {\n this.client.removeListener(\"clientReady\", onReady);\n this.client.removeListener(\"error\", onError);\n };\n\n this.client.once(\"clientReady\", onReady);\n this.client.once(\"error\", onError);\n\n this.client.login(this.token).catch((err: unknown) => {\n cleanup();\n reject(err instanceof Error ? err : new Error(String(err)));\n });\n });\n }\n\n /** Register a handler for MESSAGE_CREATE events. */\n onMessage(handler: (message: Message) => void | Promise<void>): void {\n this.client.on(\"messageCreate\", (message) => {\n logDebug(TAG, `Message from ${message.author.tag} in #${message.channel.id}`);\n try {\n const result = handler(message);\n if (result instanceof Promise) {\n result.catch((err: unknown) => {\n logError(TAG, \"Message handler error\", err);\n });\n }\n } catch (err) {\n logError(TAG, \"Message handler error\", err);\n }\n });\n }\n\n /** Register a handler for MESSAGE_REACTION_ADD events. */\n onReaction(handler: (reaction: MessageReaction, user: User) => void | Promise<void>): void {\n this.client.on(\"messageReactionAdd\", async (reaction, user) => {\n if (user.bot) return;\n try {\n if (reaction.partial) await reaction.fetch();\n if (user.partial) await user.fetch();\n } catch (err) { logAndSwallow(TAG, \"fetch reaction/user partial\", err); return; }\n try {\n const result = handler(reaction as MessageReaction, user as User);\n if (result instanceof Promise) result.catch((err: unknown) => logError(TAG, \"Reaction handler error\", err));\n } catch (err) { logError(TAG, \"Reaction handler error\", err); }\n });\n }\n\n /** Register a handler for slash command interactions. */\n onInteraction(handler: (interaction: import(\"discord.js\").ChatInputCommandInteraction) => void | Promise<void>): void {\n this.client.on(\"interactionCreate\", async (interaction) => {\n if (interaction.isChatInputCommand()) {\n try {\n const result = handler(interaction);\n if (result instanceof Promise) result.catch((err: unknown) => logError(TAG, \"Interaction handler error\", err));\n } catch (err) { logError(TAG, \"Interaction handler error\", err); }\n }\n if (interaction.isStringSelectMenu()) {\n this.selectMenuHandlers.get(interaction.customId)?.(interaction);\n }\n });\n }\n\n private selectMenuHandlers = new Map<string, (i: import(\"discord.js\").StringSelectMenuInteraction) => void | Promise<void>>();\n\n onSelectMenu(customId: string, handler: (interaction: import(\"discord.js\").StringSelectMenuInteraction) => void | Promise<void>): void {\n this.selectMenuHandlers.set(customId, handler);\n }\n\n /** Register global application commands. Idempotent \u2014 Discord diffs and updates. */\n async registerCommands(commands: Array<{ name: string; description: string }>): Promise<void> {\n if (!this.client.application) {\n logWarn(TAG, \"Cannot register commands \u2014 application not available\");\n return;\n }\n await this.client.application.commands.set(commands.map(c => ({ name: c.name, description: c.description, type: 1 as const })));\n logInfo(TAG, `Registered ${commands.length} slash commands`);\n }\n\n /** Send a text message to a channel. Returns the sent message ID. */\n async sendMessage(channelId: string, text: string): Promise<string> {\n const channel = await this.client.channels.fetch(channelId);\n if (!channel?.isTextBased()) {\n throw new Error(`Channel ${channelId} is not a text channel`);\n }\n const sent = await (channel as TextChannel).send(text);\n logDebug(TAG, `Sent message ${sent.id} to channel ${channelId}`);\n return sent.id;\n }\n\n /** Send typing indicator to a channel. */\n async sendTyping(channelId: string): Promise<void> {\n try {\n const channel = await this.client.channels.fetch(channelId);\n if (channel?.isTextBased()) await (channel as TextChannel).sendTyping();\n } catch (err) { logAndSwallow(\"discord_api\", \"op\", err); }\n }\n\n /** Add or remove a reaction on a message. Empty emoji = remove bot's reactions. */\n async setReaction(channelId: string, messageId: string, emoji: string): Promise<void> {\n try {\n const channel = await this.client.channels.fetch(channelId);\n if (!channel?.isTextBased()) return;\n const message = await (channel as TextChannel).messages.fetch(messageId);\n if (!emoji) {\n // Remove bot's reactions\n const botId = this.client.user?.id;\n if (botId) {\n for (const reaction of message.reactions.cache.values()) {\n await reaction.users.remove(botId).catch(err => logAndSwallow(TAG, \"remove bot reaction\", err));\n }\n }\n } else {\n await message.react(emoji);\n }\n } catch (err) { logAndSwallow(\"discord_api\", \"op\", err); }\n }\n\n /** Edit a previously sent message. */\n async editMessage(channelId: string, messageId: string, text: string): Promise<void> {\n try {\n const channel = await this.client.channels.fetch(channelId);\n if (!channel?.isTextBased()) return;\n const message = await (channel as TextChannel).messages.fetch(messageId);\n await message.edit(text);\n } catch (err) { logAndSwallow(\"discord_api\", \"op\", err); }\n }\n\n /** Gracefully close the Gateway connection. */\n async disconnect(): Promise<void> {\n logInfo(TAG, \"Disconnecting from Discord Gateway\u2026\");\n this.ready = false;\n this.client.destroy();\n }\n\n /** Whether the client is connected and ready. */\n get isReady(): boolean {\n return this.ready;\n }\n\n /** Get the bot's own user ID (for filtering self-messages). */\n get botUserId(): string | null {\n return this.client.user?.id ?? null;\n }\n\n /** Fetch a specific message by channel + message ID. Returns null on failure. (#388) */\n async fetchMessage(channelId: string, messageId: string): Promise<{ authorId: string } | null> {\n try {\n const channel = await this.client.channels.fetch(channelId);\n if (!channel?.isTextBased()) return null;\n const msg = await (channel as TextChannel).messages.fetch(messageId);\n return msg ? { authorId: msg.author.id } : null;\n } catch (err) { logAndSwallow(TAG, \"fetchMessage\", err); return null; }\n }\n}\n", "import type { Message } from \"discord.js\";\nimport type { DiscordInboundMessage } from \"../../types/discord.js\";\nimport type { DiscordApi } from \"./discord-api.js\";\nimport { logInfo, logDebug, logWarn } from \"../../components/logger.js\";\n\nconst TAG = \"DiscordPoller\";\n\n/**\n * Event-driven listener for Discord messages via the Gateway WebSocket.\n * Mirrors TelegramPoller in lifecycle (start/stop) but uses discord.js\n * event handlers instead of long-polling. Reconnection and heartbeat\n * are handled by discord.js internally.\n */\nexport class DiscordPoller {\n private readonly api: DiscordApi;\n private readonly appId: string;\n private readonly onMessage: (message: DiscordInboundMessage) => void | Promise<void>;\n private started = false;\n\n constructor(\n api: DiscordApi,\n appId: string,\n onMessage: (message: DiscordInboundMessage) => void | Promise<void>,\n ) {\n this.api = api;\n this.appId = appId;\n this.onMessage = onMessage;\n }\n\n /** Connect to Gateway and start listening. */\n async start(): Promise<void> {\n if (this.started) return;\n this.started = true;\n\n await this.api.connect();\n\n this.api.onMessage((raw: Message) => {\n // Filter out self-messages (use appId from config, fallback to client user id)\n const selfId = this.api.botUserId ?? this.appId;\n if (raw.author.id === selfId) {\n logDebug(TAG, `Ignoring self-message ${raw.id}`);\n return;\n }\n\n const inbound = toDiscordInboundMessage(raw, this.appId);\n logDebug(TAG, `Dispatching message ${inbound.id} from ${inbound.authorUsername}`);\n\n try {\n const result = this.onMessage(inbound);\n if (result instanceof Promise) {\n result.catch((err: unknown) => {\n logWarn(TAG, `Error in message callback: ${err instanceof Error ? err.message : String(err)}`);\n });\n }\n } catch (err) {\n logWarn(TAG, `Error in message callback: ${err instanceof Error ? err.message : String(err)}`);\n }\n });\n\n logInfo(TAG, \"Started \u2014 listening for Discord messages\");\n }\n\n /** Disconnect from Gateway cleanly. */\n stop(): void {\n if (!this.started) return;\n this.started = false;\n this.api.disconnect();\n logInfo(TAG, \"Stopped\");\n }\n}\n\n/** Convert a raw discord.js Message to a DiscordInboundMessage. */\nfunction toDiscordInboundMessage(raw: Message, botId: string | null): DiscordInboundMessage {\n const parentId = \"parentId\" in raw.channel ? (raw.channel.parentId ?? null) : null;\n const channelName = \"name\" in raw.channel ? (raw.channel.name ?? null) : null;\n // Check both discord.js parsed mentions AND raw content for the bot's mention tag\n const mentionsBotId = botId\n ? (raw.mentions.users.has(botId) || new RegExp(`<@!?${botId}>`).test(raw.content))\n : false;\n\n // #388 \u2014 role-mention detection. Look up the bot's own guild member and check\n // if any of its role IDs appear in raw.mentions.roles. If the bot's member isn't\n // cached yet (cold start, first message in a guild), stay safe by returning false \u2014\n // a missed role-mention is better than an accidental always-respond.\n const botMember = raw.guild?.members.me ?? null;\n const botRoleIds = botMember ? new Set([...botMember.roles.cache.keys()]) : new Set<string>();\n const mentionsBotRole = botMember\n ? [...raw.mentions.roles.keys()].some(rid => botRoleIds.has(rid))\n : false;\n\n logDebug(TAG, `mentionsBotId=${mentionsBotId} mentionsBotRole=${mentionsBotRole} (appId=${botId}, mentions=${[...raw.mentions.users.keys()].join(\",\")}, content=${raw.content.slice(0, 60)})`);\n return {\n id: raw.id,\n channelId: raw.channelId,\n parentChannelId: parentId,\n channelName,\n isDM: !raw.guildId,\n authorId: raw.author.id,\n authorUsername: raw.author.username,\n authorIsBot: raw.author.bot ?? false,\n content: raw.content,\n timestamp: raw.createdTimestamp,\n mentionsBotId,\n mentionsBotRole,\n mentionsEveryone: raw.mentions.everyone,\n hasUserMentions: raw.mentions.users.size > 0,\n replyReferenceMessageId: raw.reference?.messageId ?? null,\n attachments: raw.attachments.size > 0\n ? [...raw.attachments.values()].map(a => ({ url: a.url, filename: a.name ?? \"file\", contentType: a.contentType ?? undefined, size: a.size }))\n : undefined,\n };\n}\n", "/**\n * Discord platform adapter \u2014 wraps DiscordApi, DiscordPoller, DiscordSecurityGate.\n * Handles Discord-specific pre-processing (mention stripping, A2A routing, sender prefix)\n * then delegates to the shared message pipeline.\n */\n\nimport { DiscordApi } from \"./discord-api.js\";\nimport { DiscordPoller } from \"./discord-poller.js\";\nimport { SecurityGate } from \"../../components/security-gate.js\";\nimport { loadUsers } from \"../../components/user-registry.js\";\nimport { BOT_COMMANDS } from \"../../components/command-registry.js\";\nimport { ResponseFormatter } from \"../../components/response-formatter.js\";\nimport { formatReactionSignal } from \"../../components/reactions.js\";\n\nexport const DISCORD_CAPABILITIES: PlatformCapabilities = { voice: false, reactions: true, typing: true, threads: true };\nimport { emojiToScore } from \"../../utils/emoji-score.js\";\nimport { logInfo, logWarn, logDebug } from \"../../components/logger.js\";\nimport { logAndSwallow } from \"../../components/log-and-swallow.js\";\nimport { getEnv } from \"../../components/env-schema.js\";\nimport { handleInboundMessage, type PipelineDeps } from \"../../components/message-pipeline.js\";\nimport type { PlatformAdapter, PlatformCapabilities, InboundMessage, SendOpts } from \"../../types/platform.js\";\nimport type { DiscordInboundMessage } from \"../../types/index.js\";\nimport type { IKiroTransport } from \"../../components/transport/kiro-transport.js\";\nimport type { IMemorySystem } from \"abmind\";\nimport type { ConversationBuffer } from \"../../components/conversation-buffer.js\";\n\nconst TAG = \"discord\";\n\nexport interface DiscordAdapterConfig {\n botToken: string;\n appId: string;\n allowedUserIds: Set<string>;\n}\n\nexport interface DiscordAdapterDeps {\n pipeline: PipelineDeps;\n transport: IKiroTransport;\n memory: IMemorySystem | null;\n conversationBuffer: ConversationBuffer;\n}\n\nexport class DiscordAdapter implements PlatformAdapter {\n readonly name = \"discord\" as const;\n readonly capabilities: PlatformCapabilities = DISCORD_CAPABILITIES;\n\n private readonly api: DiscordApi;\n private readonly securityGate: SecurityGate;\n private readonly formatter = new ResponseFormatter();\n private readonly config: DiscordAdapterConfig;\n private readonly deps: DiscordAdapterDeps;\n private poller: DiscordPoller | null = null;\n\n constructor(config: DiscordAdapterConfig, deps: DiscordAdapterDeps) {\n this.api = new DiscordApi(config.botToken);\n this.securityGate = new SecurityGate(loadUsers());\n this.config = config;\n this.deps = deps;\n }\n\n async start(): Promise<void> {\n this.poller = new DiscordPoller(this.api, this.config.appId, (m) => this.handleMessage(m));\n this.api.onReaction((reaction, user) => this.handleReaction(reaction, user));\n this.api.onInteraction((interaction) => this.handleInteraction(interaction));\n await this.poller.start();\n\n // Register slash commands (idempotent)\n await this.api.registerCommands([...BOT_COMMANDS]).catch(err =>\n logWarn(TAG, `Slash command registration failed: ${err instanceof Error ? err.message : String(err)}`),\n );\n }\n\n /** Handle Discord slash command interactions. */\n private async handleInteraction(interaction: import(\"discord.js\").ChatInputCommandInteraction): Promise<void> {\n // Interactive model picker for /model and /models\n if (interaction.commandName === \"model\" || interaction.commandName === \"models\") {\n await this.handleModelPicker(interaction);\n return;\n }\n\n await interaction.deferReply();\n\n const registry = loadUsers();\n const userEntry = registry.byPlatformId.get(`discord:${interaction.user.id}`);\n if (!this.securityGate.authorizeById(interaction.user.id, interaction.channelId)) {\n await interaction.editReply(\"\u26D4 Unauthorized.\");\n return;\n }\n\n const commandText = `/${interaction.commandName}`;\n const userId = userEntry?.userId ?? \"unknown\";\n const channelId = interaction.channelId;\n\n // Wrap adapter so pipeline responses route through the interaction reply\n let initialReplied = false;\n const interactionAdapter: PlatformAdapter = {\n ...this,\n sendMessage: async (_ch: string, text: string): Promise<string | undefined> => {\n if (!text?.trim()) return undefined;\n const chunks = text.length <= 2000 ? [text] : text.match(/.{1,2000}/gs) ?? [text];\n let lastId: string | undefined;\n for (const chunk of chunks) {\n if (!initialReplied) {\n initialReplied = true;\n const sent = await interaction.editReply(chunk);\n lastId = sent.id;\n } else {\n const sent = await interaction.followUp(chunk);\n lastId = sent.id;\n }\n }\n return lastId;\n },\n editMessage: async (_ch: string, _messageId: number | string, text: string): Promise<void> => {\n // For interaction replies, editReply edits the initial deferred response\n await interaction.editReply(text);\n },\n };\n\n const msg: InboundMessage = {\n text: commandText,\n channelId,\n userId,\n senderId: interaction.user.id,\n senderName: interaction.user.username ?? \"unknown\",\n platform: \"discord\",\n timestamp: Date.now(),\n isGroup: !!interaction.guildId,\n isVoice: false,\n };\n\n await handleInboundMessage(msg, interactionAdapter, this.deps.pipeline);\n }\n\n /** Interactive model picker \u2014 ephemeral select menus. */\n private async handleModelPicker(interaction: import(\"discord.js\").ChatInputCommandInteraction): Promise<void> {\n const { ActionRowBuilder, StringSelectMenuBuilder } = await import(\"discord.js\");\n const { loadTransport } = await import(\"../../components/transport-config.js\");\n const tc = loadTransport();\n if (!tc) { await interaction.reply({ content: \"No transport config.\", ephemeral: true }); return; }\n\n const providers = Object.entries(tc.providers).map(([id, p]) => ({ id, name: (p as any).name ?? id }));\n if (providers.length === 0) {\n await interaction.reply({ content: \"No providers configured.\", ephemeral: true });\n return;\n }\n\n const menu = new StringSelectMenuBuilder()\n .setCustomId(\"model_picker_provider\")\n .setPlaceholder(\"Select provider\")\n .addOptions(providers.slice(0, 25).map(p => ({ label: p.name, value: p.id })));\n\n const row = new ActionRowBuilder<import(\"discord.js\").StringSelectMenuBuilder>().addComponents(menu);\n await interaction.reply({ content: \"\uD83D\uDD27 Select provider:\", components: [row], ephemeral: true });\n\n // Handle provider selection \u2192 show models\n this.api.onSelectMenu(\"model_picker_provider\", async (selectInteraction) => {\n const providerId = selectInteraction.values[0]!;\n const provider = tc.providers[providerId];\n if (!provider) { await selectInteraction.update({ content: \"Provider not found.\", components: [] }); return; }\n\n const { getModelsForProvider } = await import(\"../../components/transport-config.js\");\n const models = getModelsForProvider(providerId).filter((m) => m.entry.status === \"alive\" || !m.entry.status).slice(0, 25);\n if (models.length === 0) { await selectInteraction.update({ content: `No models for ${providerId}.`, components: [] }); return; }\n\n const modelMenu = new StringSelectMenuBuilder()\n .setCustomId(\"model_picker_model\")\n .setPlaceholder(\"Select model\")\n .addOptions(models.map((m) => ({ label: m.id, value: m.id })));\n\n const modelRow = new ActionRowBuilder<import(\"discord.js\").StringSelectMenuBuilder>().addComponents(modelMenu);\n await selectInteraction.update({ content: `Provider: **${providerId}**\\nSelect model:`, components: [modelRow] });\n\n // Handle model selection \u2192 apply\n this.api.onSelectMenu(\"model_picker_model\", async (modelInteraction) => {\n const modelId = modelInteraction.values[0]!;\n // Route through command handler as /model <provider> <model>\n await modelInteraction.update({ content: `\u2705 Switching to **${providerId}/${modelId}**...`, components: [] });\n\n // Trigger the actual switch via the command pipeline\n const msg: InboundMessage = {\n text: `/model ${providerId} ${modelId}`,\n channelId: modelInteraction.channelId,\n userId: interaction.user.id,\n senderId: interaction.user.id,\n senderName: interaction.user.username ?? \"unknown\",\n platform: \"discord\",\n timestamp: Date.now(),\n isGroup: !!interaction.guildId,\n isVoice: false,\n };\n await handleInboundMessage(msg, this, this.deps.pipeline);\n });\n });\n }\n\n stop(): void {\n this.poller?.stop();\n this.poller = null;\n }\n\n authorize(msg: InboundMessage): boolean {\n // Discord security uses string IDs + channel check\n return this.securityGate.authorizeById(msg.senderId, msg.channelId);\n }\n\n async sendMessage(channelId: string, text: string, _opts?: SendOpts): Promise<string | undefined> {\n const id = await this.api.sendMessage(channelId, text);\n return id || undefined;\n }\n\n async sendTyping(channelId: string): Promise<void> {\n await this.api.sendTyping(channelId);\n }\n\n async setReaction(channelId: string, messageId: number | string, emoji: string): Promise<void> {\n await this.api.setReaction(channelId, String(messageId), emoji);\n }\n\n async editMessage(channelId: string, messageId: number | string, text: string): Promise<void> {\n await this.api.editMessage(channelId, String(messageId), text);\n }\n\n chunkResponse(text: string): string[] {\n return this.formatter.chunkForPlatform(text, \"discord\");\n }\n\n injectMessage(msg: InboundMessage): void {\n // Discord doesn't support synthetic injection the same way.\n // For sleep replay, we call handleInboundMessage directly.\n handleInboundMessage(msg, this, this.deps.pipeline).catch((err) => {\n logWarn(TAG, `Failed to replay queued message: ${err instanceof Error ? err.message : String(err)}`);\n });\n }\n\n // --- Internal: Discord message handler ---\n\n private async handleMessage(message: DiscordInboundMessage): Promise<void> {\n logDebug(TAG, `Message from ${message.authorUsername} in ${message.channelId}`);\n\n const effectiveChannelId = message.parentChannelId ?? message.channelId;\n\n if (!this.securityGate.authorizeById(message.authorId, effectiveChannelId)) {\n logDebug(TAG, `Unauthorized user=${message.authorId} channel=${effectiveChannelId}`);\n return;\n }\n\n const rawText = message.content.trim();\n if (!rawText && !message.attachments?.length) return;\n\n // Strip bot's own mention\n let text = rawText.replace(new RegExp(`<@!?${this.config.appId}>`, \"g\"), \"\").replace(/\\s{2,}/g, \" \").trim();\n\n // Download attachments\n let mediaPath: string | undefined;\n if (message.attachments?.length) {\n try {\n const { saveInboundMedia } = await import(\"../../components/media-utils.js\");\n const att = message.attachments[0]!; // handle first attachment\n const res = await fetch(att.url);\n if (res.ok) {\n const buf = Buffer.from(await res.arrayBuffer());\n const extHint = att.filename ? \".\" + (att.filename.split(\".\").pop() ?? \"\") : undefined;\n const saved = await saveInboundMedia(buf, message.channelId, { extHint, claimedMime: att.contentType });\n if (saved) {\n mediaPath = saved.path;\n if (!text) text = `User sent a ${saved.isImage ? \"photo\" : \"file\"}.`;\n }\n }\n } catch (err) {\n logWarn(TAG, `Attachment download failed: ${err instanceof Error ? err.message : String(err)}`);\n }\n }\n\n if (!text) return;\n\n // Mention filter: in non-DM channels, check DISCORD_GROUP_MENTIONS mode.\n const isDM = message.isDM;\n if (!isDM) {\n const mentionMode = getEnv().discordGroupMentions; // \"required\" | \"optional\"\n\n if (mentionMode === \"optional\") {\n // Skip messages that mention someone else (not us, not @everyone)\n if (message.hasUserMentions && !message.mentionsBotId && !message.mentionsBotRole && !message.mentionsEveryone) {\n logDebug(TAG, `Skipped \u2014 mentions another user (optional mode, channel=${effectiveChannelId})`);\n return;\n }\n } else {\n // Required mode: must be explicitly addressed\n let isReplyToBot = false;\n if (message.replyReferenceMessageId && this.config.appId) {\n try {\n const referenced = await this.api.fetchMessage(message.channelId, message.replyReferenceMessageId);\n isReplyToBot = referenced?.authorId === this.config.appId;\n } catch (err) { logAndSwallow(TAG, \"fetchMessage reply ref\", err); }\n }\n\n const addressed = message.mentionsBotId || message.mentionsBotRole || isReplyToBot;\n if (!addressed) {\n logDebug(TAG, `Skipped \u2014 not addressed (channel=${effectiveChannelId})`);\n return;\n }\n }\n }\n\n const channelLabel = message.parentChannelId ? message.channelName ?? \"thread\" : message.channelName ?? \"DM\";\n const senderPrefix = isDM ? \"\" : `[${message.authorUsername}${message.authorIsBot ? \" (bot)\" : \"\"}] in #${channelLabel}: `;\n\n const inbound: InboundMessage = {\n platform: \"discord\",\n channelId: message.channelId,\n userId: loadUsers().byPlatformId.get(\"discord:\" + message.authorId)?.userId ?? \"unknown\",\n senderId: message.authorId,\n senderName: message.authorUsername,\n text: senderPrefix + text,\n timestamp: message.timestamp,\n isGroup: !isDM,\n isVoice: false,\n mediaPath,\n rawPlatformData: message,\n };\n\n // #512: commands bypass sequential await\n if (text.startsWith(\"/\") && !text.startsWith(\"//\")) {\n handleInboundMessage(inbound, this, this.deps.pipeline).catch(err => logAndSwallow(TAG, \"handleInboundMessage command\", err));\n return;\n }\n\n await handleInboundMessage(inbound, this, this.deps.pipeline);\n }\n\n private async handleReaction(\n reaction: import(\"discord.js\").MessageReaction,\n user: import(\"discord.js\").User,\n ): Promise<void> {\n const channelId = reaction.message.channelId;\n const messageId = Number(reaction.message.id);\n const emoji = reaction.emoji.name ?? \"\";\n if (!emoji) return;\n\n const isAuthorized = this.securityGate.authorizeById(user.id, channelId);\n const senderName = user.username || `id:${user.id}`;\n logInfo(TAG, `Reaction ${emoji} from ${senderName} on msg ${reaction.message.id}`);\n\n // Emotion scoring on authorized reactions\n if (isAuthorized && this.deps.memory) {\n const score = emojiToScore(emoji);\n const resolvedUserId = loadUsers().byPlatformId.get(\"discord:\" + user.id)?.userId ?? \"unknown\";\n const updated = this.deps.memory.updateEmotionByPlatformId(resolvedUserId, messageId, score);\n if (updated) logDebug(TAG, `Emotion score ${score} set on msg ${reaction.message.id}`);\n }\n\n if (!isAuthorized) {\n logDebug(TAG, `Unauthorized reaction from ${user.id}, discarding`);\n return;\n }\n\n // Buffer reaction signal for next message context\n const signal = formatReactionSignal(senderName, [emoji]);\n const bufKey = `discord:${channelId}`;\n this.deps.conversationBuffer.push(bufKey, senderName, signal);\n logDebug(TAG, `Buffered reaction signal for channel ${channelId}`);\n }\n}\n"],
5
- "mappings": "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",
6
- "names": ["TAG", "TAG"]
7
- }
@@ -1,47 +0,0 @@
1
- #!/usr/bin/env node
2
- import { createRequire as __bundleCreateRequire } from 'node:module'; import { fileURLToPath as __bundleFileURLToPath } from 'node:url'; import { dirname as __bundleDirname } from 'node:path'; const require = __bundleCreateRequire(import.meta.url); const __chunk_filename = __bundleFileURLToPath(import.meta.url); const __chunk_dirname = __bundleDirname(__chunk_filename);
3
- import {
4
- abtarsHome,
5
- init_paths
6
- } from "./chunk-MJ6PHMOK.js";
7
- import "./chunk-7K2YZTLD.js";
8
-
9
- // src/cli/commands/restore.ts
10
- init_paths();
11
- import { existsSync } from "node:fs";
12
- import { spawnSync } from "node:child_process";
13
- async function restore(archivePath) {
14
- if (!archivePath) {
15
- process.stderr.write("Usage: abtars restore <file.zip|.7z>\n");
16
- return 1;
17
- }
18
- if (!existsSync(archivePath)) {
19
- process.stderr.write(`File not found: ${archivePath}
20
- `);
21
- return 1;
22
- }
23
- const home = abtarsHome();
24
- const is7z = archivePath.endsWith(".7z");
25
- const listCmd = is7z ? spawnSync("7z", ["l", archivePath], { encoding: "utf-8" }) : spawnSync("unzip", ["-l", archivePath], { encoding: "utf-8" });
26
- if (!listCmd.stdout?.includes("config/")) {
27
- process.stderr.write("Error: archive does not contain config/ \u2014 not a valid abtars backup\n");
28
- return 1;
29
- }
30
- let result;
31
- if (is7z) {
32
- result = spawnSync("7z", ["x", `-o${home}`, "-aoa", archivePath], { encoding: "utf-8", stdio: "inherit" });
33
- } else {
34
- result = spawnSync("unzip", ["-o", archivePath, "-d", home], { encoding: "utf-8", stdio: "inherit" });
35
- }
36
- if (result.status !== 0) {
37
- process.stderr.write("Restore failed\n");
38
- return 1;
39
- }
40
- process.stdout.write(`\u2713 Restored to ${home}
41
- `);
42
- return 0;
43
- }
44
- export {
45
- restore
46
- };
47
- //# sourceMappingURL=restore-ROJF22R2.js.map
@@ -1,7 +0,0 @@
1
- {
2
- "version": 3,
3
- "sources": ["../src/cli/commands/restore.ts"],
4
- "sourcesContent": ["/**\n * `abtars restore` \u2014 extract user data from backup archive to ~/.abtars/\n */\n\nimport { existsSync } from \"node:fs\";\nimport { spawnSync } from \"node:child_process\";\nimport { abtarsHome } from \"../../paths.js\";\n\nexport async function restore(archivePath: string): Promise<number> {\n if (!archivePath) {\n process.stderr.write(\"Usage: abtars restore <file.zip|.7z>\\n\");\n return 1;\n }\n if (!existsSync(archivePath)) {\n process.stderr.write(`File not found: ${archivePath}\\n`);\n return 1;\n }\n\n const home = abtarsHome();\n const is7z = archivePath.endsWith(\".7z\");\n\n // Sanity check: verify archive contains config/\n const listCmd = is7z\n ? spawnSync(\"7z\", [\"l\", archivePath], { encoding: \"utf-8\" })\n : spawnSync(\"unzip\", [\"-l\", archivePath], { encoding: \"utf-8\" });\n if (!listCmd.stdout?.includes(\"config/\")) {\n process.stderr.write(\"Error: archive does not contain config/ \u2014 not a valid abtars backup\\n\");\n return 1;\n }\n\n // Extract\n let result;\n if (is7z) {\n result = spawnSync(\"7z\", [\"x\", `-o${home}`, \"-aoa\", archivePath], { encoding: \"utf-8\", stdio: \"inherit\" });\n } else {\n result = spawnSync(\"unzip\", [\"-o\", archivePath, \"-d\", home], { encoding: \"utf-8\", stdio: \"inherit\" });\n }\n\n if (result.status !== 0) {\n process.stderr.write(\"Restore failed\\n\");\n return 1;\n }\n\n process.stdout.write(`\u2713 Restored to ${home}\\n`);\n return 0;\n}\n"],
5
- "mappings": ";;;;;;;;;AAMA;AAFA,SAAS,kBAAkB;AAC3B,SAAS,iBAAiB;AAG1B,eAAsB,QAAQ,aAAsC;AAClE,MAAI,CAAC,aAAa;AAChB,YAAQ,OAAO,MAAM,wCAAwC;AAC7D,WAAO;AAAA,EACT;AACA,MAAI,CAAC,WAAW,WAAW,GAAG;AAC5B,YAAQ,OAAO,MAAM,mBAAmB,WAAW;AAAA,CAAI;AACvD,WAAO;AAAA,EACT;AAEA,QAAM,OAAO,WAAW;AACxB,QAAM,OAAO,YAAY,SAAS,KAAK;AAGvC,QAAM,UAAU,OACZ,UAAU,MAAM,CAAC,KAAK,WAAW,GAAG,EAAE,UAAU,QAAQ,CAAC,IACzD,UAAU,SAAS,CAAC,MAAM,WAAW,GAAG,EAAE,UAAU,QAAQ,CAAC;AACjE,MAAI,CAAC,QAAQ,QAAQ,SAAS,SAAS,GAAG;AACxC,YAAQ,OAAO,MAAM,4EAAuE;AAC5F,WAAO;AAAA,EACT;AAGA,MAAI;AACJ,MAAI,MAAM;AACR,aAAS,UAAU,MAAM,CAAC,KAAK,KAAK,IAAI,IAAI,QAAQ,WAAW,GAAG,EAAE,UAAU,SAAS,OAAO,UAAU,CAAC;AAAA,EAC3G,OAAO;AACL,aAAS,UAAU,SAAS,CAAC,MAAM,aAAa,MAAM,IAAI,GAAG,EAAE,UAAU,SAAS,OAAO,UAAU,CAAC;AAAA,EACtG;AAEA,MAAI,OAAO,WAAW,GAAG;AACvB,YAAQ,OAAO,MAAM,kBAAkB;AACvC,WAAO;AAAA,EACT;AAEA,UAAQ,OAAO,MAAM,sBAAiB,IAAI;AAAA,CAAI;AAC9C,SAAO;AACT;",
6
- "names": []
7
- }
@@ -1,7 +0,0 @@
1
- {
2
- "version": 3,
3
- "sources": ["../src/components/system-status.ts"],
4
- "sourcesContent": ["/**\n * system-status.ts \u2014 Single source of truth for bridge status.\n * One collector, many renderers. Used by /status (text), dashboard (HTML), future API (JSON).\n */\n\nimport { logAndSwallow } from \"./log-and-swallow.js\";\nimport { readFileSync, readlinkSync, existsSync } from \"node:fs\";\nimport { join, basename } from \"node:path\";\nimport { homedir } from \"node:os\";\nimport type { ServiceState } from \"./service-registry.js\";\n\nconst TAG = \"system_status\";\n\nexport interface SubsystemHealth {\n name: string;\n status: \"ok\" | \"failed\" | \"skipped\" | \"stopped\" | \"retrying\";\n detail?: string;\n}\n\nexport interface SystemStatus {\n version: string;\n commit: string;\n model: string;\n transportType: string;\n transportProvider: string;\n transportReady: boolean;\n uptimeMs: number;\n subsystems: SubsystemHealth[];\n tasks: { recurring: number; pending: number; paused: number };\n lastBackup: string | null;\n}\n\nexport interface StatusContext {\n phaseHealth: Map<string, { status: \"ok\" | \"failed\" | \"skipped\"; error?: string }>;\n registry: { getStates(): Record<string, ServiceState> };\n transport: { isReady: boolean } | null;\n startedAt: number;\n bridgeLockPath: string;\n heartbeat: { intervalMs: number } | null;\n}\n\nexport async function getSystemStatus(ctx: StatusContext): Promise<SystemStatus> {\n // Version + commit from release symlink\n let version = \"?\";\n let commit = \"?\";\n try {\n const target = basename(readlinkSync(join(homedir(), \".abtars\", \"current\")));\n const dash = target.lastIndexOf(\"-\");\n if (dash > 0) { version = target.slice(0, dash); commit = target.slice(dash + 1); }\n } catch (err) { logAndSwallow(\"system_status\", \"op\", err); }\n\n // Model + transport\n let model = \"unknown\";\n let transportType = \"unknown\";\n let transportProvider = \"unknown\";\n try {\n const { loadTransport, resolveAgent } = await import(\"./transport-config.js\");\n const tc = loadTransport();\n const prof = tc ? resolveAgent(\"professor\", tc) : null;\n if (prof) {\n model = prof.model;\n transportType = (prof.provider.transport ?? \"acp\").toUpperCase();\n transportProvider = prof.providerName ?? \"unknown\";\n }\n } catch (err) { logAndSwallow(\"system_status\", \"op\", err); }\n\n // Subsystem health: merge phaseHealth + ServiceRegistry live state\n const serviceStates = ctx.registry.getStates();\n const subsystems: SubsystemHealth[] = [];\n\n for (const [name, health] of ctx.phaseHealth) {\n const entry: SubsystemHealth = { name, status: health.status };\n\n // For platform services, override with live ServiceRegistry state\n const svcName = phaseToService(name);\n if (svcName && serviceStates[svcName]) {\n const svc = serviceStates[svcName] as ServiceState;\n if (health.status === \"ok\" && !svc.running) {\n entry.status = svc.retrying ? \"retrying\" : \"stopped\";\n }\n if (svc.retrying) {\n entry.detail = `retry #${svc.retrying.attempt}`;\n }\n }\n\n // Add contextual detail for specific phases\n if (name === \"phaseTransport\" && health.status === \"ok\" && ctx.transport) {\n entry.detail = `${transportType} (${transportProvider}), ${ctx.transport.isReady ? \"ready\" : \"not ready\"}`;\n }\n if (name === \"phaseHeartbeat\" && health.status === \"ok\" && ctx.heartbeat) {\n try {\n const lock = JSON.parse(readFileSync(ctx.bridgeLockPath, \"utf-8\"));\n const ago = Math.round((Date.now() - (lock.lastHeartbeat || 0)) / 60000);\n entry.detail = `${Math.round(ctx.heartbeat.intervalMs / 1000)}s, last tick ${ago}m ago`;\n } catch (err) { logAndSwallow(\"system_status\", \"op\", err); }\n }\n if (name === \"phaseDashboard\" && health.status === \"ok\") {\n entry.detail = `:${process.env[\"WEB_PORT\"] || \"3000\"}`;\n }\n\n subsystems.push(entry);\n }\n\n // Tasks\n let tasks = { recurring: 0, pending: 0, paused: 0 };\n try {\n const { readEntries } = await import(\"./tasks/task-store.js\");\n const entries = readEntries();\n tasks = {\n recurring: entries.filter((e: any) => e.schedule && !e.paused).length,\n pending: entries.filter((e: any) => !e.fired && !e.schedule).length,\n paused: entries.filter((e: any) => e.paused).length,\n };\n } catch (err) { logAndSwallow(\"system_status\", \"op\", err); }\n\n // Last backup\n let lastBackup: string | null = null;\n try {\n const { readdirSync } = await import(\"node:fs\");\n const bd = join(homedir(), \".backup-abtars\");\n const bk = readdirSync(bd).filter((f: string) => f.startsWith(\"abtars-\")).sort();\n if (bk.length > 0) lastBackup = bk[bk.length - 1] ?? null;\n } catch (err) { logAndSwallow(\"system_status\", \"op\", err); }\n\n return {\n version,\n commit,\n model,\n transportType,\n transportProvider,\n transportReady: ctx.transport?.isReady ?? false,\n uptimeMs: Date.now() - ctx.startedAt,\n subsystems,\n tasks,\n lastBackup,\n };\n}\n\n/** Map phase names to ServiceRegistry service names (for live state merge). */\nfunction phaseToService(phaseName: string): string | null {\n switch (phaseName) {\n case \"phasePlatforms\": return null; // platforms registers telegram + discord separately\n case \"phaseDashboard\": return null;\n case \"phaseAgentApi\": return \"agent-api\";\n default: return null;\n }\n}\n\n/** Render SystemStatus as plain text for Telegram/Discord /status command. */\nexport function renderStatusText(status: SystemStatus): string {\n const uptime = formatUptime(status.uptimeMs);\n const lines: string[] = [\n `abTARS v${status.version} (${status.commit})`,\n ];\n\n // abmind version\n try {\n const abmindManifest = join(homedir(), \".abmind\", \"manifest.json\");\n if (existsSync(abmindManifest)) {\n const m = JSON.parse(readFileSync(abmindManifest, \"utf-8\"));\n const ver = m.version?.replace(/-[a-f0-9]{7,}$/, \"\") || null;\n const commit = m.commit?.slice(0, 7) ?? \"\";\n if (ver) {\n lines.push(`abmind v${ver}${commit ? ` (${commit})` : \"\"}`);\n } else {\n lines.push(\"abmind: installed (version unknown)\");\n }\n } else {\n lines.push(\"abmind: not installed\");\n }\n } catch (err) { logAndSwallow(TAG, \"read abmind manifest\", err); }\n\n // Update check (#440)\n try {\n const { checkForUpdate } = require(\"./update-check.js\") as typeof import(\"./update-check.js\");\n const result = checkForUpdate(\"abtars\", status.version);\n if (result?.updateAvailable) {\n lines.push(`\uD83D\uDCE6 Update available: ${result.current} \u2192 ${result.latest}`);\n }\n } catch (err) { logAndSwallow(TAG, \"update check\", err); }\n\n lines.push(\n `\uD83E\uDD16 Model: ${status.model}`,\n `\u23F1\uFE0F Uptime: ${uptime}`,\n \"\",\n \"\uD83C\uDFE5 Subsystems:\",\n );\n\n for (const s of status.subsystems) {\n const icon = s.status === \"ok\" ? \"\u2713\" : s.status === \"skipped\" ? \"\u25CB\" : \"\u2717\";\n const label = s.name.replace(\"phase\", \"\").replace(/([A-Z])/g, \" $1\").trim().toLowerCase();\n const detail = s.detail ? ` \u2014 ${s.detail}` : \"\";\n lines.push(` ${icon} ${label}${detail}`);\n }\n\n lines.push(\"\");\n lines.push(`\u23F0 Tasks: ${status.tasks.recurring} recurring, ${status.tasks.pending} pending${status.tasks.paused ? `, ${status.tasks.paused} paused` : \"\"}`);\n if (status.lastBackup) lines.push(`\uD83D\uDCBE Last backup: ${status.lastBackup}`);\n\n return lines.join(\"\\n\");\n}\n\nfunction formatUptime(ms: number): string {\n const s = Math.floor(ms / 1000);\n const m = Math.floor(s / 60) % 60;\n const h = Math.floor(s / 3600) % 24;\n const d = Math.floor(s / 86400);\n if (d > 0) return `${d}d ${h}h ${m}m`;\n if (h > 0) return `${h}h ${m}m`;\n return `${m}m`;\n}\n"],
5
- "mappings": "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",
6
- "names": []
7
- }
@@ -1,7 +0,0 @@
1
- {
2
- "version": 3,
3
- "sources": ["../src/platforms/telegram/telegram-adapter.ts", "../src/platforms/telegram/telegram-api.ts", "../src/platforms/telegram/telegram-poller.ts", "../src/platforms/telegram/offset-store.ts", "../src/platforms/telegram/telegram-model-picker.ts"],
4
- "sourcesContent": ["import { BOT_COMMANDS } from \"../../components/command-registry.js\";\nimport { abtarsHome } from \"../../paths.js\";\n/**\n * Telegram platform adapter \u2014 wraps TelegramApi, TelegramPoller, SecurityGate.\n * Handles Telegram-specific pre-processing (voice, reactions, groups, mentions)\n * then delegates to the shared message pipeline.\n */\n\nimport { TelegramApi } from \"./telegram-api.js\";\nimport { TelegramPoller } from \"./telegram-poller.js\";\nimport { createFileOffsetStore } from \"./offset-store.js\";\nimport { SecurityGate } from \"../../components/security-gate.js\";\nimport { ResponseFormatter } from \"../../components/response-formatter.js\";\nimport { formatReactionSignal, routeReaction } from \"../../components/reactions.js\";\nimport { cleanResponse } from \"../../components/clean-response.js\";\n\nexport const TELEGRAM_CAPABILITIES: PlatformCapabilities = { voice: true, reactions: true, typing: true, threads: true };\nimport { emojiToScore } from \"../../utils/emoji-score.js\";\nimport { logInfo, logWarn, logError, logDebug } from \"../../components/logger.js\";\nimport { logAndSwallow } from \"../../components/log-and-swallow.js\";\nimport { handleInboundMessage, resetAndPrepare, type PipelineDeps } from \"../../components/message-pipeline.js\";\nimport type { PlatformAdapter, PlatformCapabilities, InboundMessage, SendOpts } from \"../../types/platform.js\";\nimport type { TelegramUpdate } from \"../../types/index.js\";\nimport type { ConversationBuffer } from \"../../components/conversation-buffer.js\";\nimport type { IKiroTransport } from \"../../components/transport/kiro-transport.js\";\nimport type { IMemorySystem } from \"abmind\";\nimport { loadUsers } from \"../../components/user-registry.js\";\nimport { isModelPickerCallback, handleModelPickerCallback } from \"./telegram-model-picker.js\";\n\nconst TAG = \"telegram\";\n\nexport interface TelegramAdapterConfig {\n botToken: string;\n allowedUserIds: Set<number>;\n pollTimeoutS: number;\n}\n\nexport interface TelegramAdapterDeps {\n pipeline: PipelineDeps;\n conversationBuffer: ConversationBuffer;\n transport: IKiroTransport;\n memory: IMemorySystem | null;\n sessionManager: { getActiveSessionId(userId: string, platform: string): string };\n}\n\nexport class TelegramAdapter implements PlatformAdapter {\n readonly name = \"telegram\" as const;\n readonly capabilities: PlatformCapabilities = TELEGRAM_CAPABILITIES;\n\n private readonly api: TelegramApi;\n private readonly securityGate: SecurityGate;\n private readonly formatter = new ResponseFormatter();\n private readonly config: TelegramAdapterConfig;\n private readonly deps: TelegramAdapterDeps;\n private poller: TelegramPoller | null = null;\n private botUsername = \"\";\n _pendingSlot: string | undefined;\n _modelPickerCache: string[] = [];\n\n constructor(config: TelegramAdapterConfig, deps: TelegramAdapterDeps) {\n this.api = new TelegramApi(config.botToken);\n this.securityGate = new SecurityGate(loadUsers());\n this.config = config;\n this.deps = deps;\n }\n\n /** Send a system notification to a chat (fire-and-forget). */\n sendNotification(chatId: string, text: string): void {\n this.api.sendMessage(parseInt(chatId, 10), text).catch(err => logAndSwallow(TAG, \"sendNotification\", err));\n }\n\n /** Reset session after model switch \u2014 saves idle state, clears buffer, marks pendingStart. */\n private async resetSessionForModelSwitch(chatId: number, reason = \"model-switch\"): Promise<void> {\n await this.api.sendMessage(chatId, \"Resetting session\u2026\");\n const p = this.deps.pipeline;\n const sessionKey = `telegram:${chatId}`;\n const bufKey = `telegram:${chatId}`;\n await p.idleSave.save(sessionKey, chatId);\n await resetAndPrepare({\n transport: this.deps.transport, sessionKey, reason,\n sessions: p.sessions, conversationBuffer: this.deps.conversationBuffer, bufKey,\n });\n if (p.memoryConfig.memoryEnabled) {\n const reg = loadUsers();\n const user = reg.byPlatformId.get(String(chatId));\n if (user) p.updateCtxStart(p.memoryConfig.memoryDir, user.userId);\n }\n }\n\n async start(): Promise<void> {\n const botInfo = await this.api.getMe();\n this.botUsername = botInfo.username?.toLowerCase() ?? \"\";\n logInfo(TAG, `\uD83E\uDD16 Bot: @${botInfo.username}`);\n\n await this.api.setMyCommands(BOT_COMMANDS.map(c => ({ command: c.name, description: c.description })))\n .catch((err) => logWarn(TAG, `setMyCommands failed: ${err instanceof Error ? err.message : String(err)}`));\n\n const home = abtarsHome();\n const offsetStore = createFileOffsetStore(`${home}/state/telegram-offset`);\n this.poller = new TelegramPoller(this.api, this.config.pollTimeoutS, (u) => this.handleUpdate(u), offsetStore);\n await this.poller.start();\n }\n\n stop(): void {\n this.poller?.stop();\n this.poller = null;\n }\n\n authorize(msg: InboundMessage): boolean {\n return this.securityGate.authorizeById(msg.senderId);\n }\n\n async sendMessage(channelId: string, text: string, opts?: SendOpts): Promise<number | undefined> {\n const chatId = parseInt(channelId, 10);\n const sendOpts: Record<string, unknown> = {};\n if (opts?.threadId) sendOpts.message_thread_id = parseInt(opts.threadId, 10);\n if (opts?.parseMode) sendOpts.parse_mode = opts.parseMode;\n if (opts?.reply_markup) sendOpts.reply_markup = opts.reply_markup;\n return this.api.sendMessage(chatId, text, sendOpts);\n }\n\n async editMessage(channelId: string, messageId: number, text: string): Promise<void> {\n const chatId = parseInt(channelId, 10);\n await this.api.editMessageText(chatId, messageId, text);\n }\n\n chunkResponse(text: string): string[] {\n return this.formatter.chunkText(text);\n }\n\n async sendTyping(channelId: string, threadId?: string): Promise<void> {\n await this.api.sendChatAction(parseInt(channelId, 10), \"typing\", threadId ? parseInt(threadId, 10) : undefined);\n }\n\n async setReaction(channelId: string, messageId: number, emoji: string): Promise<void> {\n if (messageId <= 0) return;\n try {\n const reaction = emoji ? [{ type: \"emoji\" as const, emoji }] : [];\n await this.api.setMessageReaction(parseInt(channelId, 10), messageId, reaction);\n } catch (err) {\n logDebug(TAG, `React failed (${emoji || \"remove\"}): ${err instanceof Error ? err.message : String(err)}`);\n }\n }\n\n async downloadVoice(fileId: string): Promise<Buffer> {\n const fileInfo = await this.api.getFile(fileId);\n if (!fileInfo.file_path) throw new Error(\"No file_path returned\");\n return this.api.downloadFile(fileInfo.file_path);\n }\n\n async sendVoice(channelId: string, audio: Buffer, opts?: SendOpts): Promise<void> {\n const sendOpts: Record<string, unknown> = {};\n if (opts?.threadId) sendOpts.message_thread_id = parseInt(opts.threadId, 10);\n await this.api.sendVoice(parseInt(channelId, 10), audio, sendOpts);\n }\n\n /** Send a file from disk as a Telegram document. */\n async sendDocument(channelId: string, filePath: string, caption?: string, opts?: SendOpts): Promise<number> {\n const sendOpts: { message_thread_id?: number } = {};\n if (opts?.threadId) sendOpts.message_thread_id = parseInt(opts.threadId, 10);\n return this.api.sendDocument(parseInt(channelId, 10), filePath, caption, sendOpts);\n }\n\n injectMessage(msg: InboundMessage): void {\n if (!this.poller) return;\n this.poller.injectUpdate({\n update_id: 0,\n message: {\n message_id: 0,\n from: { id: parseInt(msg.channelId, 10), is_bot: false, first_name: \"queued\" },\n chat: { id: parseInt(msg.channelId, 10), type: \"private\" },\n date: Math.floor(Date.now() / 1000),\n text: msg.text,\n ...(msg.threadId ? { message_thread_id: parseInt(msg.threadId, 10) } : {}),\n },\n });\n }\n\n // --- Internal: Telegram update handler ---\n\n private async handleUpdate(update: TelegramUpdate): Promise<void> {\n logDebug(TAG, `Update: ${JSON.stringify(update).slice(0, 200)}`);\n\n if (update.callback_query) {\n const data = update.callback_query.data ?? \"\";\n const chatId = update.callback_query.message?.chat?.id;\n this.api.answerCallbackQuery(update.callback_query.id).catch(err => logAndSwallow(TAG, \"answerCallbackQuery\", err));\n if (!chatId) return;\n\n try {\n if (isModelPickerCallback(data)) {\n await handleModelPickerCallback(data, chatId, this.api, this, {\n transport: this.deps.transport as any,\n pipeline: this.deps.pipeline as any,\n resetSessionForModelSwitch: (cid, reason) => this.resetSessionForModelSwitch(cid, reason),\n });\n } else if (data.startsWith(\"model:\")) {\n // Legacy callback \u2014 direct model switch\n const newModel = data.slice(6);\n const transport = this.deps.transport;\n if (\"setModel\" in transport && typeof (transport as { setModel: unknown }).setModel === \"function\") {\n try {\n await (transport as { setModel: (m: string) => Promise<void> | void }).setModel(newModel);\n await this.resetSessionForModelSwitch(chatId);\n if (chatId) await this.api.sendMessage(chatId, `\uD83E\uDD16 Model switched \u2192 ${newModel}`);\n } catch (err) {\n if (chatId) await this.api.sendMessage(chatId, `\u274C Model switch failed: ${err instanceof Error ? err.message : String(err)}`);\n }\n }\n }\n } catch (err) {\n logWarn(TAG, `Callback handler error: ${err instanceof Error ? err.message : String(err)}`);\n this.api.sendMessage(chatId, `\u26A0\uFE0F Action failed \u2014 try again.`).catch(err2 => logAndSwallow(TAG, \"sendMessage action failed\", err2));\n }\n return;\n }\n\n if (update.message_reaction) {\n await this.handleReaction(update);\n return;\n }\n\n const message = update.message;\n if (!message?.from) return;\n\n const hasText = Boolean(message.text);\n const hasVoice = Boolean(message.voice || message.audio);\n const hasPhoto = Boolean(message.photo?.length);\n const hasDocument = Boolean(message.document);\n if (!hasText && !hasVoice && !hasPhoto && !hasDocument) return;\n\n const chatId = message.chat.id;\n const isGroup = message.chat.type === \"group\" || message.chat.type === \"supergroup\";\n const threadId = isGroup ? message.message_thread_id : undefined;\n const messageId = message.message_id;\n const senderName = message.from.first_name || message.from.username || `id:${message.from.id}`;\n const bufKey = threadId != null ? `tg:${chatId}:${threadId}` : `tg:${chatId}`;\n\n let text = message.text ?? \"\";\n let isVoiceNote = false;\n let voiceFileId: string | undefined;\n\n // --- Voice note pre-processing ---\n if (hasVoice && !hasText) {\n if (!this.deps.pipeline.sttConfig) {\n if (isGroup) {\n this.deps.conversationBuffer.push(bufKey, senderName, \"[voice note - STT disabled]\");\n } else if (this.securityGate.authorizeById(String(message.from?.id))) {\n await this.api.sendMessage(chatId, \"\uD83C\uDFA4 Voice notes require STT (set GROQ_API_KEY).\", { message_thread_id: threadId });\n }\n return;\n }\n\n if (!this.securityGate.authorizeById(String(message.from?.id))) {\n if (isGroup) this.deps.conversationBuffer.push(bufKey, senderName, \"[voice note]\");\n return;\n }\n\n // For groups: we need to transcribe first to check for bot mention\n if (isGroup) {\n try {\n await this.setReaction(String(chatId), messageId, \"\uD83D\uDC40\");\n const voiceFile = message.voice || message.audio;\n const audioBuffer = await this.downloadVoice(voiceFile!.file_id);\n const { transcribeAudio } = await import(\"../../components/stt.js\");\n const result = await transcribeAudio(audioBuffer, \"voice.ogg\", this.deps.pipeline.sttConfig!);\n const transcript = result.text;\n\n if (!transcript) {\n await this.setReaction(String(chatId), messageId, \"\");\n this.deps.conversationBuffer.push(bufKey, senderName, \"[voice note - empty]\");\n return;\n }\n\n const mentionRe = new RegExp(`@?${this.botUsername}\\\\b`, \"i\");\n if (!mentionRe.test(transcript) && !transcript.startsWith(\"/\")) {\n await this.setReaction(String(chatId), messageId, \"\");\n this.deps.conversationBuffer.push(bufKey, senderName, `[voice] ${transcript}`);\n logDebug(TAG, `Buffered voice transcript: \"${transcript.slice(0, 60)}\"`);\n return;\n }\n text = transcript.replace(mentionRe, \"\").trim();\n isVoiceNote = true;\n if (!text) { await this.setReaction(String(chatId), messageId, \"\"); return; }\n } catch (err) {\n logError(TAG, \"Voice transcription failed\", err);\n await this.setReaction(String(chatId), messageId, \"\");\n return;\n }\n } else {\n // DM voice: let pipeline handle STT\n isVoiceNote = true;\n voiceFileId = (message.voice || message.audio)!.file_id;\n }\n }\n\n // --- Group text filtering ---\n if (!isVoiceNote && isGroup) {\n const mentionRe = new RegExp(`@${this.botUsername}\\\\b`, \"i\");\n const isMention = mentionRe.test(text);\n const isCommand = text.startsWith(\"/\");\n\n if (!isMention && !isCommand) {\n this.deps.conversationBuffer.push(bufKey, senderName, text);\n logDebug(TAG, `Buffered group msg from ${senderName}: \"${text.slice(0, 60)}\"`);\n return;\n }\n\n if (isMention) {\n text = text.replace(mentionRe, \"\").trim();\n if (!text) return;\n }\n }\n\n // --- Security ---\n if (!isVoiceNote && !this.securityGate.authorizeById(String(message.from?.id))) {\n if (isGroup) this.deps.conversationBuffer.push(bufKey, senderName, text);\n logWarn(TAG, `Unauthorized user ${message.from.id}`);\n return;\n }\n\n // --- Photo/document handling ---\n let mediaPath: string | undefined;\n\n if ((hasPhoto || hasDocument) && this.securityGate.authorizeById(String(message.from?.id))) {\n try {\n const { saveInboundMedia } = await import(\"../../components/media-utils.js\");\n let fileId: string;\n let extHint: string | undefined;\n let claimedMime: string | undefined;\n\n if (hasPhoto) {\n const photo = message.photo![message.photo!.length - 1]!;\n fileId = photo.file_id;\n extHint = \".jpg\";\n } else {\n fileId = message.document!.file_id;\n extHint = message.document!.file_name ? \".\" + (message.document!.file_name.split(\".\").pop() ?? \"\") : undefined;\n claimedMime = message.document!.mime_type;\n }\n\n const buf = await this.downloadVoice(fileId); // reuse file download\n const saved = await saveInboundMedia(buf, chatId, { extHint, claimedMime });\n if (saved) {\n mediaPath = saved.path;\n if (!text) text = message.caption ?? `User sent a ${saved.isImage ? \"photo\" : \"file\"}.`;\n } else {\n if (!text) text = \"\u26A0\uFE0F File too large (max 16MB).\";\n }\n } catch (err) {\n logWarn(TAG, `Media download failed: ${err instanceof Error ? err.message : String(err)}`);\n if (!text) text = \"\u26A0\uFE0F Failed to download media.\";\n }\n }\n\n // --- Reply context ---\n const reply = message.reply_to_message;\n if (reply?.text) {\n const replyFrom = reply.from?.first_name ?? \"someone\";\n text = `[Replying to ${replyFrom}: \"${reply.text.slice(0, 500)}\"]\\n${text}`;\n }\n\n // --- Reply-to-photo: re-download and inject (#668) ---\n if (reply?.photo?.length && !mediaPath && this.securityGate.authorizeById(String(message.from?.id))) {\n try {\n const { saveInboundMedia } = await import(\"../../components/media-utils.js\");\n const photo = reply.photo[reply.photo.length - 1]!;\n const buf = await this.downloadVoice(photo.file_id);\n const saved = await saveInboundMedia(buf, chatId, { extHint: \".jpg\" });\n if (saved) mediaPath = saved.path;\n } catch (err) {\n logWarn(TAG, `Reply-to-photo download failed: ${err instanceof Error ? err.message : String(err)}`);\n }\n }\n\n // --- Dispatch to pipeline ---\n const resolvedUser = loadUsers().byPlatformId.get(\"telegram:\" + message.from.id)?.userId ?? \"unknown\";\n const inbound: InboundMessage = {\n platform: \"telegram\",\n channelId: String(chatId),\n userId: resolvedUser,\n senderId: String(message.from.id),\n senderName,\n text,\n timestamp: message.date * 1000,\n threadId: threadId != null ? String(threadId) : undefined,\n messageId,\n isGroup,\n isVoice: isVoiceNote,\n voiceFileId,\n mediaPath,\n rawPlatformData: message,\n };\n\n // /stop, /ctrlc, /restart now handled in commandMiddleware (platform-agnostic)\n\n // #512: commands bypass the sequential await \u2014 execute immediately even if agent is mid-stream\n if (text.startsWith(\"/\") && !text.startsWith(\"//\")) {\n handleInboundMessage(inbound, this, this.deps.pipeline).catch((err) => {\n logError(TAG, `Command dispatch error: ${err instanceof Error ? err.message : String(err)}`);\n });\n return;\n }\n\n await handleInboundMessage(inbound, this, this.deps.pipeline);\n }\n\n private async handleReaction(update: TelegramUpdate): Promise<void> {\n const reaction = update.message_reaction!;\n const user = reaction.user;\n if (!user) { logDebug(TAG, \"Reaction update missing user field\"); return; }\n if (user.is_bot) return;\n\n const oldEmojis = new Set(reaction.old_reaction.map((r) => r.emoji));\n const added = reaction.new_reaction.filter((r) => !oldEmojis.has(r.emoji));\n if (added.length === 0) return;\n\n const senderName = user.first_name || user.username || `id:${user.id}`;\n const emojis = added.map((r) => r.emoji);\n logInfo(TAG, `Reaction ${emojis.join(\"\")} from ${senderName} on msg ${reaction.message_id}`);\n\n const isAuthorized = this.securityGate.authorizeById(String(user.id));\n const signal = formatReactionSignal(senderName, emojis);\n const chatId = reaction.chat.id;\n const route = routeReaction(isAuthorized, reaction.chat.type);\n\n if (isAuthorized && this.deps.memory) {\n const score = emojiToScore(emojis[0]!);\n const updated = this.deps.memory.updateEmotionByPlatformId(loadUsers().byPlatformId.get(`telegram:${chatId}`)?.userId ?? \"master\", reaction.message_id, score);\n if (updated) logDebug(TAG, `Emotion score ${score} set on platform msg ${reaction.message_id}`);\n }\n\n if (route === \"discard\") {\n logDebug(TAG, `Unauthorized reaction from user ${user.id}, discarding`);\n return;\n }\n\n if (route === \"buffer\") {\n const bufKey = `tg:${chatId}`;\n this.deps.conversationBuffer.push(bufKey, senderName, signal);\n logDebug(TAG, `Buffered reaction signal for group ${chatId}`);\n } else {\n const reactionUser = loadUsers().byPlatformId.get(\"telegram:\" + user.id)?.userId ?? \"unknown\";\n const activeId = this.deps.sessionManager.getActiveSessionId(reactionUser, \"telegram\");\n const { sessions } = this.deps.pipeline;\n const entry = sessions.getOrCreate(activeId);\n if (entry.busy) {\n entry.queue.push({ msg: { userId: reactionUser, channelId: String(chatId), senderName, senderId: String(user.id), text: signal, messageId: reaction.message_id, platform: \"telegram\", timestamp: Date.now(), isGroup: false, isVoice: false }, adapter: this });\n logDebug(TAG, `Queued reaction signal for busy ${activeId} (${entry.queue.length} pending)`);\n } else {\n try {\n const response = await this.deps.transport.sendPrompt(activeId, signal, undefined, reactionUser);\n logDebug(TAG, `Sent reaction signal to transport for chat ${chatId}`);\n if (response) {\n const { text, reactionEmoji, noReply } = cleanResponse(response);\n if (noReply) { /* silent */ }\n else if (reactionEmoji) {\n try { await this.api.setMessageReaction(chatId, reaction.message_id, [{ type: \"emoji\", emoji: reactionEmoji }]); }\n catch { await this.sendMessage(String(chatId), reactionEmoji); }\n } else if (text.trim()) {\n await this.sendMessage(String(chatId), text.trim());\n }\n }\n } catch (err) {\n logError(TAG, `Failed to send reaction signal for chat ${chatId}`, err);\n }\n }\n }\n }\n}\n", "import { getEnv } from \"../../components/env-schema.js\";\nimport { logDebug } from \"../../components/logger.js\";\nimport { logAndSwallow } from \"../../components/log-and-swallow.js\";\nimport type {\n TelegramUpdate,\n TelegramInlineKeyboardMarkup,\n} from \"../../types/index.js\";\nimport { readFileSync } from \"node:fs\";\nimport { basename } from \"node:path\";\n\ntype SendMessageOptions = {\n parse_mode?: \"MarkdownV2\" | \"HTML\";\n reply_markup?: TelegramInlineKeyboardMarkup;\n message_thread_id?: number;\n};\n\nconst TAG = \"telegram-api\";\n\n/** Options for fetchWithRetry. */\ninterface FetchRetryOpts {\n method: string;\n timeoutMs?: number;\n outerSignal?: AbortSignal;\n /** Retry on 4xx responses. Default false \u2014 4xx usually permanent (404 expired, 403 auth). */\n retryable4xx?: boolean;\n}\n\n/**\n * Thin wrapper around the Telegram Bot API using native fetch.\n * All methods throw on HTTP errors.\n */\nexport class TelegramApi {\n private readonly baseUrl: string;\n private readonly token: string;\n\n constructor(botToken: string) {\n this.token = botToken;\n this.baseUrl = `https://api.telegram.org/bot${botToken}`;\n }\n\n /** Get bot info. */\n async getMe(): Promise<{ id: number; username?: string }> {\n const res = await this.call(\"getMe\", {});\n return res as { id: number; username?: string };\n }\n\n async getChat(chatId: number): Promise<{ id: number; first_name?: string; username?: string }> {\n const res = await this.call(\"getChat\", { chat_id: chatId });\n return res as { id: number; first_name?: string; username?: string };\n }\n\n /** Get file path for downloading. */\n async getFile(fileId: string): Promise<{ file_id: string; file_path?: string; file_size?: number }> {\n const res = await this.call(\"getFile\", { file_id: fileId });\n return res as { file_id: string; file_path?: string; file_size?: number };\n }\n\n /** Download a file by its file_path (from getFile). Returns the raw buffer. */\n async downloadFile(filePath: string): Promise<Buffer> {\n const url = `https://api.telegram.org/file/bot${this.token}/${filePath}`;\n // 4xx = expired file / bad path / wrong token \u2192 permanent, don't retry\n const response = await this.fetchWithRetry(\n (signal) => fetch(url, { signal }),\n { method: \"downloadFile\", timeoutMs: getEnv().telegramFileTimeoutMs, retryable4xx: false },\n );\n const arrayBuffer = await response.arrayBuffer();\n return Buffer.from(arrayBuffer);\n }\n\n /** Send a text message. Returns the sent message_id. */\n async sendMessage(\n chatId: number,\n text: string,\n options?: SendMessageOptions,\n ): Promise<number> {\n const body: Record<string, unknown> = { chat_id: chatId, text, ...options };\n const res = await this.call(\"sendMessage\", body);\n return (res as { message_id: number }).message_id;\n }\n\n /** Edit an existing message's text. */\n async editMessageText(\n chatId: number,\n messageId: number,\n text: string,\n options?: { parse_mode?: string },\n ): Promise<void> {\n await this.call(\"editMessageText\", { chat_id: chatId, message_id: messageId, text, ...options });\n }\n\n /** Acknowledge a callback query (inline keyboard press). */\n async answerCallbackQuery(callbackQueryId: string): Promise<void> {\n await this.call(\"answerCallbackQuery\", { callback_query_id: callbackQueryId });\n }\n\n /** Send a chat action (typing indicator, etc.). */\n async sendChatAction(chatId: number, action: string = \"typing\", messageThreadId?: number): Promise<void> {\n const body: Record<string, unknown> = { chat_id: chatId, action };\n if (messageThreadId) body.message_thread_id = messageThreadId;\n await this.call(\"sendChatAction\", body);\n }\n\n /** Set an emoji reaction on a message. Pass empty array to remove. */\n async setMessageReaction(\n chatId: number,\n messageId: number,\n reaction: Array<{ type: \"emoji\"; emoji: string }>,\n ): Promise<void> {\n await this.call(\"setMessageReaction\", {\n chat_id: chatId,\n message_id: messageId,\n reaction,\n });\n }\n\n /** Send a voice note (OGG Opus). Returns the sent message_id. */\n async sendVoice(\n chatId: number,\n audioBuffer: Buffer,\n options?: { message_thread_id?: number },\n ): Promise<number> {\n // Factory rebuilds FormData per attempt \u2014 FormData is single-use in some runtimes.\n const makeForm = (): FormData => {\n const form = new FormData();\n form.append(\"chat_id\", String(chatId));\n const blob = new Blob([audioBuffer], { type: \"audio/webm\" });\n form.append(\"voice\", blob, \"voice.webm\");\n if (options?.message_thread_id) form.append(\"message_thread_id\", String(options.message_thread_id));\n return form;\n };\n const response = await this.fetchWithRetry(\n (signal) => fetch(`${this.baseUrl}/sendVoice`, { method: \"POST\", body: makeForm(), signal }),\n { method: \"sendVoice\" },\n );\n const json = (await response.json()) as { ok: boolean; result: { message_id: number } };\n if (!json.ok) throw new Error(\"Telegram API sendVoice returned ok=false\");\n return json.result.message_id;\n }\n\n /** Send a file from disk as a Telegram document. Returns the sent message_id. */\n async sendDocument(\n chatId: number,\n filePath: string,\n caption?: string,\n options?: { message_thread_id?: number },\n ): Promise<number> {\n const buf = readFileSync(filePath);\n const fileName = basename(filePath);\n // Factory rebuilds FormData per attempt \u2014 FormData is single-use in some runtimes.\n const makeForm = (): FormData => {\n const form = new FormData();\n form.append(\"chat_id\", String(chatId));\n const blob = new Blob([buf], { type: \"text/markdown\" });\n form.append(\"document\", blob, fileName);\n if (caption) form.append(\"caption\", caption.slice(0, 1024));\n if (options?.message_thread_id) form.append(\"message_thread_id\", String(options.message_thread_id));\n return form;\n };\n const response = await this.fetchWithRetry(\n (signal) => fetch(`${this.baseUrl}/sendDocument`, { method: \"POST\", body: makeForm(), signal }),\n { method: \"sendDocument\", timeoutMs: getEnv().telegramFileTimeoutMs },\n );\n const json = (await response.json()) as { ok: boolean; result: { message_id: number } };\n if (!json.ok) throw new Error(\"Telegram API sendDocument returned ok=false\");\n return json.result.message_id;\n }\n\n /** Register bot command menu (shown when user types /). */\n async setMyCommands(commands: Array<{ command: string; description: string }>): Promise<void> {\n await this.call(\"setMyCommands\", { commands });\n }\n\n /** Long-poll for updates. */\n async getUpdates(\n offset: number,\n timeout: number,\n signal?: AbortSignal,\n ): Promise<TelegramUpdate[]> {\n const body = {\n offset,\n timeout,\n allowed_updates: [\n \"message\",\n \"callback_query\",\n \"message_reaction\",\n ],\n };\n return (await this.call(\"getUpdates\", body, signal, (timeout + 10) * 1000)) as TelegramUpdate[];\n }\n\n private static readonly MAX_ATTEMPTS = 3;\n private static readonly BACKOFF = [1000, 3000, 9000];\n\n private static readonly PERMANENT_ERRORS = [\n /chat not found/i, /bot was blocked/i, /user is deactivated/i,\n /chat_id is empty/i, /forbidden/i, /not enough rights/i,\n /CHAT_WRITE_FORBIDDEN/i, /have no rights/i,\n /wrong file_path/i, /file is temporarily unavailable/i,\n ];\n\n private static isPermanent(msg: string): boolean {\n return this.PERMANENT_ERRORS.some(re => re.test(msg));\n }\n\n private static isTransient(msg: string): boolean {\n return /429|5\\d\\d|bad gateway|service unavailable|timeout|timed out|ETIMEDOUT|ECONNRESET|ECONNREFUSED|socket hang up|network/i.test(msg);\n }\n\n /**\n * Retry + timeout wrapper for any Telegram fetch. Returns the Response on 2xx,\n * throws on non-OK or transient-exhaustion. Factory MUST build a fresh request\n * per attempt (required for FormData; single-use in some runtimes).\n */\n private async fetchWithRetry(\n requestFn: (signal: AbortSignal) => Promise<Response>,\n opts: FetchRetryOpts,\n ): Promise<Response> {\n const timeoutMs = opts.timeoutMs ?? getEnv().telegramTimeoutMs;\n for (let attempt = 0; attempt < TelegramApi.MAX_ATTEMPTS; attempt++) {\n const ctrl = new AbortController();\n const timer = setTimeout(() => ctrl.abort(new Error(\"telegram timeout\")), timeoutMs);\n const composed = opts.outerSignal\n ? AbortSignal.any([opts.outerSignal, ctrl.signal])\n : ctrl.signal;\n\n try {\n const response = await requestFn(composed);\n if (!response.ok) {\n const text = await response.text().catch(err2 => { logAndSwallow(TAG, \"read TG API error body\", err2); return \"\"; });\n const err = new Error(`Telegram API ${opts.method} failed (${response.status}): ${text}`);\n // 4xx treated as permanent unless caller opts in\n if (response.status >= 400 && response.status < 500 && !opts.retryable4xx) throw err;\n throw err;\n }\n return response;\n } catch (err) {\n const msg = err instanceof Error ? err.message : String(err);\n if (TelegramApi.isPermanent(msg) || opts.outerSignal?.aborted) throw err;\n // Non-retryable 4xx (when retryable4xx is false) \u2014 already thrown above with status code\n if (!opts.retryable4xx && /failed \\([4]\\d\\d\\)/.test(msg)) throw err;\n if (attempt < TelegramApi.MAX_ATTEMPTS - 1 && TelegramApi.isTransient(msg)) {\n logDebug(TAG, `${opts.method} attempt ${attempt + 1}/${TelegramApi.MAX_ATTEMPTS} \u2014 ${msg.slice(0, 80)}`);\n await new Promise(r => setTimeout(r, TelegramApi.BACKOFF[attempt]!));\n continue;\n }\n throw err;\n } finally {\n clearTimeout(timer);\n }\n }\n throw new Error(`Telegram API ${opts.method}: unreachable`);\n }\n\n private async call(\n method: string,\n body: Record<string, unknown>,\n signal?: AbortSignal,\n timeoutMs?: number,\n ): Promise<unknown> {\n const response = await this.fetchWithRetry(\n (composed) => fetch(`${this.baseUrl}/${method}`, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify(body),\n signal: composed,\n }),\n { method, timeoutMs, outerSignal: signal },\n );\n const json = (await response.json()) as { ok: boolean; result: unknown };\n if (!json.ok) throw new Error(`Telegram API ${method} returned ok=false`);\n return json.result;\n }\n}\n", "import type { TelegramUpdate } from \"../../types/index.js\";\nimport type { TelegramApi } from \"./telegram-api.js\";\nimport type { OffsetStore } from \"./offset-store.js\";\nimport { logError, logWarn } from \"../../components/logger.js\";\n\n/**\n * Long-polls the Telegram Bot API for updates. Never self-terminates \u2014\n * uses exponential backoff with jitter on transient errors.\n *\n * Offset is advanced per-update AFTER the handler settles (success only).\n * Handlers run fire-and-forget to keep /stop responsive; the disk offset\n * only advances once the contiguous prefix of settled handlers is known.\n */\nexport class TelegramPoller {\n private readonly api: TelegramApi;\n private readonly pollTimeout: number;\n private readonly onUpdate: (update: TelegramUpdate) => void | Promise<void>;\n private readonly offsetStore: OffsetStore;\n private offset = 0;\n private running = false;\n private abortController: AbortController | null = null;\n\n /** Timestamp of last successful poll cycle. */\n lastPollAt = Date.now();\n\n constructor(\n api: TelegramApi,\n pollTimeoutS: number,\n onUpdate: (update: TelegramUpdate) => void | Promise<void>,\n offsetStore?: OffsetStore,\n ) {\n this.api = api;\n this.pollTimeout = pollTimeoutS;\n this.onUpdate = onUpdate;\n this.offsetStore = offsetStore ?? { read: async () => 0, write: async () => {} };\n }\n\n /** Start the long-poll loop. Non-blocking \u2014 runs in background. */\n async start(): Promise<void> {\n if (this.running) return;\n this.offset = await this.offsetStore.read();\n this.running = true;\n this.poll().catch((err) => {\n logError(\"poller\", \"Fatal error in poll loop\", err);\n });\n }\n\n /** Stop polling. Cancels any in-flight request. */\n stop(): void {\n this.running = false;\n this.abortController?.abort();\n this.abortController = null;\n }\n\n /** Inject a synthetic update into the handler (used for queued messages after sleep). */\n injectUpdate(update: TelegramUpdate): void {\n void this.onUpdate(update);\n }\n\n private async poll(): Promise<void> {\n let failures = 0;\n\n while (this.running) {\n try {\n this.abortController = new AbortController();\n const updates = await this.api.getUpdates(\n this.offset,\n this.pollTimeout,\n this.abortController.signal,\n );\n\n failures = 0;\n this.lastPollAt = Date.now();\n\n // Process each update: dispatch handler, advance offset only on success.\n // Handlers run in background (fire-and-forget) so /stop stays responsive.\n // We track settlements and advance the disk offset for the contiguous\n // prefix of successfully-handled updates.\n if (updates.length > 0) {\n const sorted = [...updates].sort((a, b) => a.update_id - b.update_id);\n\n for (const update of sorted) {\n try {\n const result = this.onUpdate(update);\n if (result instanceof Promise) {\n result.catch((err: unknown) => logError(\"poller\", \"Error in update handler\", err));\n }\n } catch (err) {\n logError(\"poller\", \"Error in update handler (sync)\", err);\n }\n }\n\n // Advance offset immediately \u2014 don't wait for handlers to settle.\n // Next getUpdates will fetch new messages without blocking on in-flight handlers.\n this.offset = sorted[sorted.length - 1]!.update_id + 1;\n await this.offsetStore.write(this.offset);\n }\n } catch (err) {\n if (!this.running) break;\n\n failures++;\n const baseDelay = Math.min(2 ** failures * 1000, 60_000);\n const jitter = Math.random() * baseDelay;\n const delay = baseDelay + jitter;\n const is409 = String(err).includes(\"409\") || String(err).includes(\"terminated by other getUpdates\");\n const log = is409 ? (failures >= 50 ? logError : logWarn) : (failures < 3 ? logWarn : logError);\n if (is409 && failures < 50) {\n log(\"poller\", `409 conflict (attempt ${failures}) \u2014 another instance running, backing off`);\n } else {\n log(\"poller\", `Error (attempt ${failures}), retrying in ${Math.round(delay)}ms`, err);\n }\n await sleep(delay);\n }\n }\n }\n}\n\nfunction sleep(ms: number): Promise<void> {\n return new Promise((resolve) => setTimeout(resolve, ms));\n}\n", "/**\n * Atomic offset store for Telegram poller.\n * Persists the last-acked update_id to disk so crash recovery resumes\n * from the correct position. Writes atomically (temp + rename).\n */\n\nimport { readFile, writeFile, rename, mkdir } from \"node:fs/promises\";\nimport { dirname } from \"node:path\";\nimport { logAndSwallow } from \"../../components/log-and-swallow.js\";\n\nconst TAG = \"offset_store\";\n\nexport interface OffsetStore {\n read(): Promise<number>;\n write(offset: number): Promise<void>;\n}\n\n/** File-backed offset store with atomic writes. */\nexport function createFileOffsetStore(filePath: string): OffsetStore {\n let pending: Promise<void> = Promise.resolve();\n\n return {\n async read(): Promise<number> {\n try {\n const raw = await readFile(filePath, \"utf-8\");\n const val = parseInt(raw.trim(), 10);\n return Number.isFinite(val) ? val : 0;\n } catch (err) {\n logAndSwallow(TAG, \"read offset file\", err);\n return 0;\n }\n },\n write(offset: number): Promise<void> {\n // Serialize writes via promise chain (async mutex).\n pending = pending.then(async () => {\n const tmp = filePath + \".tmp\";\n await mkdir(dirname(filePath), { recursive: true });\n await writeFile(tmp, String(offset), \"utf-8\");\n await rename(tmp, filePath);\n }).catch(err => logAndSwallow(TAG, \"write offset file\", err));\n return pending;\n },\n };\n}\n\n/** In-memory offset store for tests. */\nexport function createMemoryOffsetStore(initial = 0): OffsetStore & { value: number } {\n const store = {\n value: initial,\n async read(): Promise<number> { return store.value; },\n async write(offset: number): Promise<void> { store.value = offset; },\n };\n return store;\n}\n", "/**\n * telegram-model-picker.ts \u2014 Extracted model picker callback handlers.\n * Handles: mb:, mslot:, mprov:, mpos:, mprov2:, mset:, model: prefixes.\n */\nimport type { TelegramApi } from \"./telegram-api.js\";\nimport { getEnv } from \"../../components/env-schema.js\";\nimport { logAndSwallow } from \"../../components/log-and-swallow.js\";\n\nconst TAG = \"telegram\";\n\nexport interface PickerState {\n _pendingSlot: string | undefined;\n _modelPickerCache: string[];\n}\n\nexport interface PickerDeps {\n transport: { setModel?: (m: string) => Promise<void> | void; switchProvider?: (o: unknown) => void; [k: string]: unknown };\n pipeline: { rebuildTransport?: () => Promise<void>; [k: string]: unknown };\n resetSessionForModelSwitch: (chatId: number, reason?: string) => Promise<void>;\n}\n\nconst MODEL_PREFIXES = [\"mb:\", \"mslot:\", \"mprov:\", \"mpos:\", \"mprov2:\", \"mset:\", \"model:\"];\n\nexport function isModelPickerCallback(data: string): boolean {\n return MODEL_PREFIXES.some(p => data.startsWith(p));\n}\n\nexport async function handleModelPickerCallback(\n data: string, chatId: number, api: TelegramApi, state: PickerState, deps: PickerDeps,\n): Promise<void> {\n\n if (data.startsWith(\"mb:\")) {\n const target = data.slice(3);\n if (!target) { await api.sendMessage(chatId, \"\uD83D\uDC4C Cancelled.\"); return; }\n if (target === \"a\") {\n const AGENT_LABELS = [\n { key: \"professor\", label: \"Professor\" },\n { key: \"dreamy\", label: \"Dreamy (sleep)\" },\n { key: \"browsie\", label: \"Browsie (browse)\" },\n { key: \"coding\", label: \"Cody (coding)\" },\n ];\n const buttons = AGENT_LABELS.map(a => [{ text: a.label, callback_data: `mslot:${a.key}` }]);\n buttons.push([{ text: \"\u2190 Cancel\", callback_data: \"mb:\" }]);\n await api.sendMessage(chatId, \"\uD83E\uDD16 Which agent to change?\", { reply_markup: { inline_keyboard: buttons } });\n return;\n }\n if (target.startsWith(\"p:\")) {\n const agent = target.slice(2);\n const { loadTransport, resolveAgent, getAvailableProviders, getModelsForProvider } = await import(\"../../components/transport-config.js\");\n const tc = loadTransport();\n if (!tc) { await api.sendMessage(chatId, \"\u274C transport.json not loaded\"); return; }\n let providers = getAvailableProviders(tc).filter(p => p.config.transport !== \"tmux\");\n if (providers.length === 0) { await api.sendMessage(chatId, \"\u274C No compatible providers\"); return; }\n const currentProvider = resolveAgent(agent, tc)?.providerName;\n const prefix = agent === \"professor\" ? \"mprov2:professor\" : `mprov:${agent}`;\n const buttons = providers.map(p => {\n const count = getModelsForProvider(p.name).length;\n const label = p.name === currentProvider ? `\u2705 ${p.name} (${count})` : `${p.name} (${count})`;\n return [{ text: label, callback_data: `${prefix}:${p.name}` }];\n });\n buttons.push([{ text: \"\u2190 Back\", callback_data: \"mb:a\" }]);\n await api.sendMessage(chatId, `\uD83D\uDD0C Pick provider:`, { reply_markup: { inline_keyboard: buttons } });\n return;\n }\n if (target.startsWith(\"s:\")) {\n const slot = target.slice(2);\n state._pendingSlot = slot;\n const { loadTransport, resolveAgent, getAvailableProviders, getModelsForProvider } = await import(\"../../components/transport-config.js\");\n const tc = loadTransport();\n if (!tc) { await api.sendMessage(chatId, \"\u274C transport.json not loaded\"); return; }\n let providers = getAvailableProviders(tc).filter(p => p.config.transport !== \"tmux\");\n const currentProvider = resolveAgent(\"professor\", tc)?.providerName;\n const slotLabel = slot === \"professor\" ? \"Main\" : slot.replace(\"professor_fb\", \"Fb\");\n const buttons = providers.map(p => {\n const count = getModelsForProvider(p.name).length;\n const label = p.name === currentProvider ? `\u2705 ${p.name} (${count})` : `${p.name} (${count})`;\n return [{ text: label, callback_data: `mprov2:${slot}:${p.name}` }];\n });\n buttons.push([{ text: \"\u2190 Back\", callback_data: \"mslot:professor\" }]);\n await api.sendMessage(chatId, `\uD83D\uDD0C Provider for ${slotLabel}:`, { reply_markup: { inline_keyboard: buttons } });\n return;\n }\n } else if (data.startsWith(\"mslot:\")) {\n const agent = data.slice(6);\n const { loadTransport, resolveAgent, getAvailableProviders, getModelsForProvider } = await import(\"../../components/transport-config.js\");\n const tc = loadTransport();\n if (!tc) { await api.sendMessage(chatId, \"\u274C transport.json not loaded\"); return; }\n\n if (agent === \"professor\") {\n const profResolved = resolveAgent(\"professor\", tc);\n const fallbacks = tc.agents[\"professor\"]?.fallbacks ?? [];\n const slots: Array<{ label: string; key: string }> = [\n { label: `\u2605 Main: ${profResolved?.model ?? \"?\"}`, key: `mpos:professor::professor` },\n ];\n for (let i = 0; i < fallbacks.length; i++) {\n slots.push({ label: `\u21B3 Fb${i + 1}: ${fallbacks[i]!.model}`, key: `mpos:professor::professor_fb${i + 1}` });\n }\n if (fallbacks.length < 3) {\n slots.push({ label: `\u21B3 Fb${fallbacks.length + 1}: (add)`, key: `mpos:professor::professor_fb${fallbacks.length + 1}` });\n }\n const buttons = slots.map(s => [{ text: s.label, callback_data: s.key }]);\n buttons.push([{ text: \"\u2190 Back\", callback_data: \"mb:a\" }]);\n await api.sendMessage(chatId, `\uD83C\uDFAF Which slot?`, { reply_markup: { inline_keyboard: buttons } });\n return;\n }\n\n let providers = getAvailableProviders(tc).filter(p => p.config.transport !== \"tmux\");\n if (providers.length === 0) { await api.sendMessage(chatId, \"\u274C No compatible providers\"); return; }\n const currentProvider = resolveAgent(agent, tc)?.providerName;\n const buttons = providers.map(p => {\n const count = getModelsForProvider(p.name).length;\n const label = p.name === currentProvider ? `\u2705 ${p.name} (${count})` : `${p.name} (${count})`;\n return [{ text: label, callback_data: `mprov:${agent}:${p.name}` }];\n });\n buttons.push([{ text: \"\u2190 Back\", callback_data: \"mb:a\" }]);\n await api.sendMessage(chatId, `\uD83D\uDD0C Pick provider:`, { reply_markup: { inline_keyboard: buttons } });\n\n } else if (data.startsWith(\"mprov:\")) {\n const [, agent, providerName] = data.split(\":\");\n const { loadTransport, getModelsForProvider, formatRank, formatCost } = await import(\"../../components/transport-config.js\");\n const tc = loadTransport();\n let models = getModelsForProvider(providerName!);\n const providerConfig = tc?.providers[providerName!];\n if (providerConfig?.transport === \"api\") models = models.filter(m => !m.entry.status || m.entry.status === \"alive\");\n if (models.length === 0) { await api.sendMessage(chatId, `\u274C No alive models for ${providerName}`); return; }\n state._pendingSlot = agent;\n state._modelPickerCache = models.map(m => m.id);\n const buttons = models.map((m, i) => [{ text: `${m.id} (${formatRank(m.entry.rank)}, ${formatCost(m.entry.cost)})`, callback_data: `mset:${providerName}:${i}` }]);\n buttons.push([{ text: \"\u2190 Back\", callback_data: `mb:p:${agent}` }]);\n await api.sendMessage(chatId, `\uD83D\uDCCB Models on ${providerName}:`, { reply_markup: { inline_keyboard: buttons } });\n\n } else if (data.startsWith(\"mpos:\")) {\n const [, , , slot] = data.split(\":\");\n const { loadTransport, resolveAgent, getAvailableProviders, getModelsForProvider } = await import(\"../../components/transport-config.js\");\n const tc = loadTransport();\n if (!tc) { await api.sendMessage(chatId, \"\u274C transport.json not loaded\"); return; }\n let providers = getAvailableProviders(tc).filter(p => p.config.transport !== \"tmux\");\n const profResolved = resolveAgent(\"professor\", tc);\n const mainTransport = profResolved?.provider.transport;\n if (slot && slot.startsWith(\"professor_fb\") && mainTransport) {\n if (mainTransport === \"api\") { providers = providers.filter(p => p.config.transport === \"api\"); }\n else { providers = providers.filter(p => p.name === profResolved!.providerName); }\n }\n if (providers.length === 0) { await api.sendMessage(chatId, \"\u274C No compatible providers for this slot\"); return; }\n const currentProvider = profResolved?.providerName;\n const slotLabel = slot === \"professor\" ? \"Main\" : slot!.replace(\"professor_fb\", \"Fb\");\n const buttons = providers.map(p => {\n const count = getModelsForProvider(p.name).length;\n const label = p.name === currentProvider ? `\u2705 ${p.name} (${count})` : `${p.name} (${count})`;\n return [{ text: label, callback_data: `mprov2:${slot}:${p.name}` }];\n });\n buttons.push([{ text: \"\u2190 Back\", callback_data: \"mslot:professor\" }]);\n await api.sendMessage(chatId, `\uD83D\uDD0C Provider for ${slotLabel}:`, { reply_markup: { inline_keyboard: buttons } });\n\n } else if (data.startsWith(\"mprov2:\")) {\n const [, slot, providerName] = data.split(\":\");\n const { getModelsForProvider, formatRank, formatCost, loadTransport } = await import(\"../../components/transport-config.js\");\n const tc = loadTransport();\n const providerConfig = tc?.providers[providerName!];\n let models = getModelsForProvider(providerName!);\n if (providerConfig?.transport === \"api\") models = models.filter(m => !m.entry.status || m.entry.status === \"alive\");\n if (models.length === 0) { await api.sendMessage(chatId, `\u274C No alive models for ${providerName}`); return; }\n state._pendingSlot = slot;\n const slotLabel = slot === \"professor\" ? \"Main\" : slot!.replace(\"professor_fb\", \"Fb\");\n state._modelPickerCache = models.map(m => m.id);\n const buttons = models.map((m, i) => [{ text: `${m.id} (${formatRank(m.entry.rank)}, ${formatCost(m.entry.cost)})`, callback_data: `mset:${providerName}:${i}` }]);\n buttons.push([{ text: \"\u2190 Back\", callback_data: `mb:s:${slot}` }]);\n await api.sendMessage(chatId, `\uD83D\uDCCB Pick model for ${slotLabel}:`, { reply_markup: { inline_keyboard: buttons } });\n\n } else if (data.startsWith(\"mset:\")) {\n const parts = data.split(\":\");\n const providerName = parts[1]!;\n const modelIdx = parseInt(parts[2]!, 10);\n const model = Number.isFinite(modelIdx) && state._modelPickerCache[modelIdx]\n ? state._modelPickerCache[modelIdx]!\n : parts.slice(2).join(\":\");\n const slot = state._pendingSlot ?? \"professor\";\n state._pendingSlot = undefined;\n state._modelPickerCache = [];\n const { loadTransport, writeTransportConfig, resolveAgent, getModelsForProvider, validateProviderReady, formatValidationError } = await import(\"../../components/transport-config.js\");\n const tc = loadTransport();\n if (!tc) { await api.sendMessage(chatId, \"\u274C transport.json not loaded\"); return; }\n\n const validModels = getModelsForProvider(providerName);\n if (!validModels.some(m => m.id === model)) { await api.sendMessage(chatId, `\u274C ${model} is not available on ${providerName}. Pick another.`); return; }\n\n const providerConfig = tc.providers[providerName];\n if (!providerConfig) { await api.sendMessage(chatId, `\u274C Provider ${providerName} not found`); return; }\n const validation = validateProviderReady(providerName, providerConfig, getEnv());\n if (!validation.ok) { await api.sendMessage(chatId, formatValidationError(providerName, validation)); return; }\n\n if (providerConfig?.transport === \"api\") {\n try {\n const endpoint = providerConfig.endpoint ?? \"\";\n const apiKey = getEnv().getApiKey(providerConfig.apiKeyEnv ?? \"API_KEY\");\n const headers: Record<string, string> = {};\n if (apiKey) headers[\"Authorization\"] = `Bearer ${apiKey}`;\n const res = await fetch(`${endpoint}/models`, { headers, signal: AbortSignal.timeout(5000) });\n if (!res.ok) { await api.sendMessage(chatId, `\u26A0\uFE0F ${providerName} unreachable (${res.status}). Try another?`); return; }\n } catch (err) { logAndSwallow(TAG, \"provider reachability check\", err); await api.sendMessage(chatId, `\u26A0\uFE0F ${providerName} unreachable. Try another?`); return; }\n }\n\n const agentKey = slot.startsWith(\"professor_fb\") ? \"professor\" : slot;\n const fbIndex = slot === \"professor_fb1\" ? 0 : slot === \"professor_fb2\" ? 1 : slot === \"professor_fb3\" ? 2 : -1;\n\n if (fbIndex >= 0) {\n if (!tc.agents[\"professor\"]) tc.agents[\"professor\"] = { model: \"\", provider: \"\" };\n if (!tc.agents[\"professor\"]!.fallbacks) tc.agents[\"professor\"]!.fallbacks = [];\n tc.agents[\"professor\"]!.fallbacks[fbIndex] = { model, provider: providerName };\n const { cleanDemotedModels } = await import(\"../../components/transport-config.js\");\n cleanDemotedModels(tc, model);\n writeTransportConfig(tc, `professor fallback ${fbIndex + 1} \u2192 ${model} (${providerName})`);\n await api.sendMessage(chatId, `\u2705 Fallback ${fbIndex + 1} \u2192 ${model} (${providerName})`);\n } else {\n const oldProvider = tc.agents[agentKey]?.provider;\n tc.agents[agentKey] = { ...tc.agents[agentKey]!, model, provider: providerName };\n const { cleanDemotedModels } = await import(\"../../components/transport-config.js\");\n cleanDemotedModels(tc, model);\n writeTransportConfig(tc, `${agentKey} \u2192 ${model} (${providerName})`);\n\n const providerChanged = oldProvider !== providerName;\n const isProfessor = agentKey === \"professor\";\n\n let oldType: string | undefined;\n let newType: string | undefined;\n let newResolved: ReturnType<typeof resolveAgent> | undefined;\n if (isProfessor && providerChanged) {\n const oldResolved = resolveAgent(\"_old\", { ...tc, agents: { ...tc.agents, _old: { model: \"\", provider: oldProvider! } } });\n newResolved = resolveAgent(\"_new\", { ...tc, agents: { ...tc.agents, _new: { model, provider: providerName } } });\n oldType = oldResolved?.provider.transport ?? \"api\";\n newType = newResolved?.provider.transport ?? \"api\";\n if (oldType !== newType) {\n const resetAgents: string[] = [];\n for (const [a, assignment] of Object.entries(tc.agents)) {\n if (a === \"professor\") continue;\n const ap = tc.providers[assignment.provider];\n if (ap && ap.transport !== newType) { tc.agents[a] = { model, provider: providerName }; resetAgents.push(a); }\n }\n if (resetAgents.length > 0) writeTransportConfig(tc, `cascade: ${resetAgents.join(\", \")} \u2192 ${providerName}`);\n }\n }\n\n if (isProfessor && !providerChanged && \"setModel\" in deps.transport) {\n await (deps.transport as unknown as { setModel: (m: string) => Promise<void> }).setModel(model);\n await api.sendMessage(chatId, `\u2705 Switched to ${model}`);\n } else if (isProfessor && providerChanged && oldType === newType && \"switchProvider\" in deps.transport) {\n try {\n const { FallbackPolicy } = await import(\"../../components/transport/fallback-policy.js\");\n const { ModelHealthRegistry } = await import(\"../../components/transport/model-health-registry.js\");\n const apiKey = getEnv().getApiKey(newResolved?.provider.apiKeyEnv ?? \"API_KEY\");\n const candidates = [{ endpoint: newResolved!.provider.endpoint!, apiKey, model, maxContext: newResolved!.contextWindow }];\n for (const fb of (tc.agents[\"professor\"]?.fallbacks ?? [])) {\n const fbRes = resolveAgent(\"_fb\", { ...tc, agents: { ...tc.agents, _fb: { model: fb.model, provider: fb.provider } } });\n if (fbRes) candidates.push({ endpoint: fbRes.provider.endpoint!, apiKey: fbRes.provider.apiKeyEnv ? getEnv().getApiKey(fbRes.provider.apiKeyEnv) : apiKey, model: fb.model, maxContext: fbRes.contextWindow });\n }\n const registry = (deps.transport as unknown as { policy?: { registry: InstanceType<typeof ModelHealthRegistry> } }).policy?.registry ?? new ModelHealthRegistry();\n const policy = new FallbackPolicy(candidates, registry);\n (deps.transport as unknown as { switchProvider: (o: unknown) => void }).switchProvider({ endpoint: newResolved!.provider.endpoint!, apiKey, model, maxContext: newResolved!.contextWindow, policy });\n } catch (err) {\n await api.sendMessage(chatId, `\u26A0\uFE0F Hot swap failed: ${err instanceof Error ? err.message : String(err)}. Use /reset to apply.`);\n return;\n }\n try { await api.sendMessage(chatId, `\u2705 Switched to ${model} (${providerName})`); } catch (err) { logAndSwallow(TAG, \"sendMessage model switch confirm\", err); }\n } else if (isProfessor && providerChanged) {\n const cascadeNote = oldType !== newType ? \" Subagents also reset.\" : \"\";\n try {\n if (deps.pipeline.rebuildTransport) await deps.pipeline.rebuildTransport();\n await deps.resetSessionForModelSwitch(chatId, \"cross-transport-switch\");\n await api.sendMessage(chatId, `\uD83D\uDD04 Switched to ${model} (${providerName}). Transport rebuilt.${cascadeNote}`);\n } catch (err) {\n await api.sendMessage(chatId, `\u26A0\uFE0F Transport rebuild failed: ${err instanceof Error ? err.message : String(err)}. Try /reset manually.`);\n }\n } else {\n await api.sendMessage(chatId, `\u2705 ${agentKey} \u2192 ${model} (${providerName})`);\n }\n }\n } else if (data.startsWith(\"model:\")) {\n const newModel = data.slice(6);\n if (\"setModel\" in deps.transport && typeof deps.transport.setModel === \"function\") {\n try {\n await deps.transport.setModel(newModel);\n await deps.resetSessionForModelSwitch(chatId);\n await api.sendMessage(chatId, `\uD83E\uDD16 Model switched \u2192 ${newModel}`);\n } catch (err) {\n await api.sendMessage(chatId, `\u274C Model switch failed: ${err instanceof Error ? err.message : String(err)}`);\n }\n }\n }\n}\n"],
5
- "mappings": "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",
6
- "names": ["TAG", "TAG", "buttons", "TAG", "chatId"]
7
- }