@pixelzx/genesis 2026.5.5-2 → 2026.5.5-3

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 (1017) hide show
  1. package/CHANGELOG.md +1 -0
  2. package/dist/.buildstamp +1 -1
  3. package/dist/abort-Yw60gQ-C.js +201 -0
  4. package/dist/abort.runtime-DYQuut_O.js +2 -0
  5. package/dist/abort.runtime.js +1 -1
  6. package/dist/accounts-BSCHGPHp.js +107 -0
  7. package/dist/accounts-Bb0FqxV2.js +104 -0
  8. package/dist/accounts-BjGgx8lz.js +2 -0
  9. package/dist/acp-cli-emMnOaiO.js +2193 -0
  10. package/dist/acp-spawn-BmyUZuJX.js +1093 -0
  11. package/dist/acp-spawn-DPF5zmnN.js +2 -0
  12. package/dist/acp-stateful-target-driver-CJ4JyIHf.js +89 -0
  13. package/dist/action-agents-D1tXt8Kj.js +67 -0
  14. package/dist/action-focus-Bk6AJsNH.js +132 -0
  15. package/dist/action-help-4nFSAuVo.js +7 -0
  16. package/dist/action-info-D-byoefn.js +101 -0
  17. package/dist/action-kill-Bo-NwBVR.js +33 -0
  18. package/dist/action-list-lAa4rlzN.js +21 -0
  19. package/dist/action-log-BasoUC3L.js +30 -0
  20. package/dist/action-send-DwLkiz4x.js +39 -0
  21. package/dist/action-spawn-C70_iR8d.js +47 -0
  22. package/dist/action-unfocus-dUVMx5Fh.js +29 -0
  23. package/dist/actions.runtime-BdSo6rN8.js +18 -0
  24. package/dist/actions.runtime-BvY67XKu.js +5 -0
  25. package/dist/actions.runtime.js +1 -1
  26. package/dist/agent-DDBv3Vgl.js +2 -0
  27. package/dist/agent-command-DC7I_88W.js +874 -0
  28. package/dist/agent-harness-runtime-TA6YH7Ud.js +144 -0
  29. package/dist/agent-runner-utils-CMe-JIDq.js +239 -0
  30. package/dist/agent-runner.runtime-le6AmJRw.js +3455 -0
  31. package/dist/agent-runner.runtime.js +1 -1
  32. package/dist/agent-runtime-CONy4fDD.js +18 -0
  33. package/dist/agents-BlbOM4XW.js +953 -0
  34. package/dist/agents-Byj8CDAP.js +5 -0
  35. package/dist/aliases-C3Ks5x38.js +2 -0
  36. package/dist/aliases-CxBcx9gF.js +96 -0
  37. package/dist/api-CLoj7nYH.js +3 -0
  38. package/dist/api-DZiRkVPk.js +5 -0
  39. package/dist/api-_Y4nIHNo.js +139 -0
  40. package/dist/approval-gateway-resolver-Ba_2kb3B.js +29 -0
  41. package/dist/approval-gateway-runtime-7VMivk0A.js +2 -0
  42. package/dist/approval-handler-runtime-Bh0onoHD.js +439 -0
  43. package/dist/approval-native-runtime-IYFyT8Hf.js +729 -0
  44. package/dist/attempt-execution.runtime-CpL2tsYd.js +509 -0
  45. package/dist/attempt-execution.runtime.js +1 -1
  46. package/dist/attempt.prompt-helpers-Gx9nGKbW.js +221 -0
  47. package/dist/attempt.tool-run-context-BmuljBMs.js +933 -0
  48. package/dist/audit-CsjqOEVE.js +939 -0
  49. package/dist/audit.runtime-BybEaYH4.js +7 -0
  50. package/dist/audit.runtime.js +1 -1
  51. package/dist/auth-BFbLz2gx.js +383 -0
  52. package/dist/auth-DdoNShRm.js +2 -0
  53. package/dist/auth-order-CtO2WQTH.js +2 -0
  54. package/dist/auth-order-zXIiW-Cn.js +96 -0
  55. package/dist/bash-tools-CnDYSbnX.js +2824 -0
  56. package/dist/bash-tools-DiOcKaak.js +3 -0
  57. package/dist/binding-routing-E9Gh_NqO.js +85 -0
  58. package/dist/binding-targets-C5PbBaZ6.js +121 -0
  59. package/dist/bridge-server-DragdaQD.js +113 -0
  60. package/dist/browser-control-auth-D_M31xJ1.js +2 -0
  61. package/dist/browser-node-runtime-Bgu82luB.js +12 -0
  62. package/dist/browser-profiles-C3ffkgOt.js +2 -0
  63. package/dist/browser-runtime-D84IBo4A.js +387 -0
  64. package/dist/browser-setup-tools-VC7rGCyn.js +13 -0
  65. package/dist/build-DeGDX_pJ.js +550 -0
  66. package/dist/build-info.json +3 -3
  67. package/dist/bundled/boot-md/handler.js +2 -2
  68. package/dist/bundled/session-memory/handler.js +1 -1
  69. package/dist/call-CbvF41H8.js +3 -0
  70. package/dist/call-DuDGOVT1.js +331 -0
  71. package/dist/call.runtime-DztWXjMr.js +2 -0
  72. package/dist/call.runtime.js +1 -1
  73. package/dist/capability-cli-DoJLi3eS.js +1401 -0
  74. package/dist/catchup-Dk-5hIwm.js +300 -0
  75. package/dist/channel-BJ9iEC4Y.js +595 -0
  76. package/dist/channel-BdvsbgZE.js +453 -0
  77. package/dist/channel-Bu-7KaIx.js +1802 -0
  78. package/dist/channel-C9vQNMK1.js +350 -0
  79. package/dist/channel-CZoVAZyW.js +491 -0
  80. package/dist/channel-Cr2KTwyo.js +840 -0
  81. package/dist/channel-DCeC6AEQ.js +297 -0
  82. package/dist/channel-DOGp2TfT.js +1320 -0
  83. package/dist/channel-Dfq1lxxI.js +226 -0
  84. package/dist/channel-core-DiltAzdr.js +5 -0
  85. package/dist/channel-hZDaTBpm.js +1100 -0
  86. package/dist/channel-inbound-DvOS-Z1s.js +31 -0
  87. package/dist/channel-plugin-runtime-PFRHQ61T.js +771 -0
  88. package/dist/channel-runtime-BAjxIEfQ.js +425 -0
  89. package/dist/channel-szBwbQZB.js +1174 -0
  90. package/dist/channel.runtime-B1Ry35gt.js +89 -0
  91. package/dist/channel.runtime-BNp6GxaQ.js +34702 -0
  92. package/dist/channel.runtime-BfjvNvbj.js +2364 -0
  93. package/dist/channel.runtime-BlGn4QYV.js +4 -0
  94. package/dist/channel.runtime-D7bv5rJl.js +576 -0
  95. package/dist/channel.runtime-DQ5gSgeW.js +109 -0
  96. package/dist/channel.runtime-IxClQMdT.js +430 -0
  97. package/dist/channel.setup-BsfyGPgV.js +10 -0
  98. package/dist/channels-Dn2VHRmZ.js +733 -0
  99. package/dist/channels-cli-l4xjPvDv.js +268 -0
  100. package/dist/chat-zUeX8mQw.js +2758 -0
  101. package/dist/clawbot-cli-C3BFdHJB.js +9 -0
  102. package/dist/cli/daemon-cli.js +4 -4
  103. package/dist/cli-BTnpFsj1.js +2 -0
  104. package/dist/cli-CAFQ6Cwy.js +72 -0
  105. package/dist/cli-CiqIbwtg.js +219 -0
  106. package/dist/cli-DL1ppK38.js +2 -0
  107. package/dist/cli-runner-DO6lEPJr.js +286 -0
  108. package/dist/cli-runner.runtime-B-Ic4DyC.js +3 -0
  109. package/dist/cli-runner.runtime-C4cAORcg.js +4 -0
  110. package/dist/cli-runner.runtime.js +1 -1
  111. package/dist/cli-startup-metadata.json +2 -2
  112. package/dist/cli.runtime-CYHctHVS.js +1261 -0
  113. package/dist/cli.runtime.js +1 -1
  114. package/dist/client-Bb_miREU.js +138 -0
  115. package/dist/client-CrPccv7a.js +713 -0
  116. package/dist/command-auth-B7P-1IeA.js +76 -0
  117. package/dist/command-config-resolution-BB-SsCob.js +23 -0
  118. package/dist/command-config-resolution-DMKYb0Kh.js +2 -0
  119. package/dist/command-config-resolution.runtime-Coeaa9lz.js +2 -0
  120. package/dist/command-config-resolution.runtime.js +1 -1
  121. package/dist/command-registry-B2JzWHqy.js +9 -0
  122. package/dist/command-registry-CNUpJUp8.js +4 -0
  123. package/dist/command-registry-core-BLap9aka.js +101 -0
  124. package/dist/command-secret-gateway-ChRXthnT.js +528 -0
  125. package/dist/command-status.runtime-D8VWyC3v.js +87 -0
  126. package/dist/command-status.runtime.js +1 -1
  127. package/dist/commands-acp-QLk1cU5f.js +77 -0
  128. package/dist/commands-compact.runtime-BIHG2RSY.js +10 -0
  129. package/dist/commands-compact.runtime.js +1 -1
  130. package/dist/commands-handlers.runtime-DjimbxSJ.js +4597 -0
  131. package/dist/commands-handlers.runtime.js +1 -1
  132. package/dist/commands-status-DiIQSYSt.js +16 -0
  133. package/dist/commands-status.runtime-DBRxICN6.js +3 -0
  134. package/dist/commands-status.runtime.js +1 -1
  135. package/dist/commands-subagents-control.runtime-BmqXiKHu.js +3 -0
  136. package/dist/commands-subagents-control.runtime-CVQSOd8H.js +2 -0
  137. package/dist/commands-subagents-control.runtime.js +1 -1
  138. package/dist/commands-system-prompt-2qwZc3xL.js +158 -0
  139. package/dist/commands-system-prompt-Clnn_Gct.js +2 -0
  140. package/dist/commands.runtime-DAqpiDPr.js +166 -0
  141. package/dist/commands.runtime.js +1 -1
  142. package/dist/compact-DqOkQZYr.js +1118 -0
  143. package/dist/compact.runtime-CC_l29j4.js +12 -0
  144. package/dist/compact.runtime.js +1 -1
  145. package/dist/completion-cli-DZFJPSiF.js +328 -0
  146. package/dist/config-D9XEe0Cd.js +251 -0
  147. package/dist/config-cli-DVDwIjKr.js +1078 -0
  148. package/dist/configure-39oQUHuK.js +1245 -0
  149. package/dist/configure-BGD8tViZ.js +2 -0
  150. package/dist/connect-options-Fpky6zuL.js +699 -0
  151. package/dist/control-auth-BlqH4IVx.js +125 -0
  152. package/dist/control-service-CaqDohiY.js +156 -0
  153. package/dist/conversation-id-BH9GFVB2.js +235 -0
  154. package/dist/conversation-id-UdIegcf4.js +38 -0
  155. package/dist/conversation-runtime-Bao0kIPf.js +31 -0
  156. package/dist/core-DXUh5Xbm.js +275 -0
  157. package/dist/cron-cli-B1mxHnGP.js +713 -0
  158. package/dist/daemon-cli-CaHBNwMy.js +12 -0
  159. package/dist/daemon-install-BBwOxf1q.js +64 -0
  160. package/dist/delegate-7G7R4C-C.js +64 -0
  161. package/dist/detached-task-runtime-BMdSrcuz.js +73 -0
  162. package/dist/devices-cli-DnEQH3R2.js +496 -0
  163. package/dist/diagnostics-C8VwSDtU.js +154 -0
  164. package/dist/direct-dm-BSEkIiOe.js +64 -0
  165. package/dist/dispatch-1gv6It-6.js +1131 -0
  166. package/dist/dispatch-acp-hbhR9aHr.js +981 -0
  167. package/dist/dispatch-acp-manager.runtime-BXQbK1cO.js +3 -0
  168. package/dist/dispatch-acp-manager.runtime.js +1 -1
  169. package/dist/dispatch-acp.runtime-Dhx090J2.js +19 -0
  170. package/dist/dispatch-acp.runtime.js +1 -1
  171. package/dist/doctor-device-pairing-1gf3tk3g.js +307 -0
  172. package/dist/doctor-gateway-daemon-flow-LEnN6pL-.js +250 -0
  173. package/dist/doctor-gateway-health-DwxLkTV4.js +60 -0
  174. package/dist/doctor-gateway-services-BhqUOY_e.js +316 -0
  175. package/dist/doctor-health-CdJkaUYf.js +59 -0
  176. package/dist/doctor-health-contributions-BaTnIzCT.js +486 -0
  177. package/dist/doctor-prompter-BZTb5rOn.js +56 -0
  178. package/dist/doctor-workspace-status-DnBARoPa.js +75 -0
  179. package/dist/dreaming-BAGFdUxh.js +1574 -0
  180. package/dist/dreaming-narrative-XoOfVb0x.js +595 -0
  181. package/dist/embedded-gateway-stub.runtime-xwZ-M7Ei.js +9 -0
  182. package/dist/embedded-gateway-stub.runtime.js +1 -1
  183. package/dist/entry.js +2 -2
  184. package/dist/exec-approvals-cli-CWVWSMk-.js +498 -0
  185. package/dist/extensionAPI.js +1 -1
  186. package/dist/extensions/active-memory/index.js +1 -1
  187. package/dist/extensions/bluebubbles/api.js +3 -3
  188. package/dist/extensions/bluebubbles/channel-plugin-api.js +1 -1
  189. package/dist/extensions/browser/browser-bridge.js +1 -1
  190. package/dist/extensions/browser/browser-config.js +4 -4
  191. package/dist/extensions/browser/browser-control-auth.js +2 -2
  192. package/dist/extensions/browser/browser-doctor.js +2 -2
  193. package/dist/extensions/browser/browser-maintenance.js +2 -2
  194. package/dist/extensions/browser/browser-profiles.js +2 -2
  195. package/dist/extensions/browser/browser-runtime-api.js +10 -10
  196. package/dist/extensions/browser/index.js +1 -1
  197. package/dist/extensions/browser/plugin-registration.js +1 -1
  198. package/dist/extensions/browser/register.runtime.js +3 -3
  199. package/dist/extensions/browser/runtime-api.js +11 -11
  200. package/dist/extensions/browser/test-support.js +1 -1
  201. package/dist/extensions/device-pair/api.js +1 -1
  202. package/dist/extensions/device-pair/index.js +3 -3
  203. package/dist/extensions/device-pair/notify.js +1 -1
  204. package/dist/extensions/device-pair/pair-command-approve.js +1 -1
  205. package/dist/extensions/google-meet/index.js +2 -2
  206. package/dist/extensions/imessage/api.js +3 -3
  207. package/dist/extensions/imessage/channel-plugin-api.js +1 -1
  208. package/dist/extensions/imessage/runtime-api.js +3 -3
  209. package/dist/extensions/irc/api.js +2 -2
  210. package/dist/extensions/irc/channel-plugin-api.js +1 -1
  211. package/dist/extensions/line/api.js +2 -2
  212. package/dist/extensions/line/channel-plugin-api.js +1 -1
  213. package/dist/extensions/line/contract-api.js +1 -1
  214. package/dist/extensions/line/runtime-api.js +4 -4
  215. package/dist/extensions/line/setup-api.js +1 -1
  216. package/dist/extensions/llm-task/index.js +2 -2
  217. package/dist/extensions/lobster/index.js +3 -3
  218. package/dist/extensions/lobster/runtime-api.js +1 -1
  219. package/dist/extensions/mattermost/api.js +1 -1
  220. package/dist/extensions/mattermost/channel-plugin-api.js +1 -1
  221. package/dist/extensions/mattermost/channel-plugin-runtime.js +1 -1
  222. package/dist/extensions/mattermost/policy-api.js +1 -1
  223. package/dist/extensions/mattermost/runtime-api.js +4 -4
  224. package/dist/extensions/mattermost/slash-route-api.js +1 -1
  225. package/dist/extensions/memory-core/api.js +1 -1
  226. package/dist/extensions/memory-core/cli-metadata.js +2 -2
  227. package/dist/extensions/memory-core/index.js +3 -3
  228. package/dist/extensions/memory-lancedb/cli-metadata.js +1 -1
  229. package/dist/extensions/msteams/api.js +1 -1
  230. package/dist/extensions/msteams/channel-plugin-api.js +1 -1
  231. package/dist/extensions/msteams/runtime-api.js +3 -3
  232. package/dist/extensions/msteams/test-api.js +1 -1
  233. package/dist/extensions/nextcloud-talk/api.js +1 -1
  234. package/dist/extensions/nextcloud-talk/channel-plugin-api.js +1 -1
  235. package/dist/extensions/nextcloud-talk/runtime-api.js +2 -2
  236. package/dist/extensions/openshell/index.js +2 -2
  237. package/dist/extensions/signal/api.js +6 -6
  238. package/dist/extensions/signal/channel-plugin-api.js +1 -1
  239. package/dist/extensions/signal/reaction-runtime-api.js +1 -1
  240. package/dist/extensions/signal/runtime-api.js +7 -7
  241. package/dist/extensions/skill-workshop/api.js +1 -1
  242. package/dist/extensions/skill-workshop/index.js +1 -1
  243. package/dist/extensions/synology-chat/api.js +1 -1
  244. package/dist/extensions/synology-chat/channel-plugin-api.js +1 -1
  245. package/dist/extensions/tlon/api.js +2 -2
  246. package/dist/extensions/tlon/channel-plugin-api.js +1 -1
  247. package/dist/extensions/tlon/runtime-api.js +1 -1
  248. package/dist/extensions/tlon/test-api.js +1 -1
  249. package/dist/extensions/twitch/api.js +1 -1
  250. package/dist/extensions/twitch/channel-plugin-api.js +1 -1
  251. package/dist/extensions/twitch/setup-plugin-api.js +1 -1
  252. package/dist/extensions/zalo/api.js +3 -3
  253. package/dist/extensions/zalo/channel-plugin-api.js +1 -1
  254. package/dist/extensions/zalo/runtime-api.js +2 -2
  255. package/dist/extensions/zalo/setup-api.js +2 -2
  256. package/dist/extensions/zalouser/api.js +3 -3
  257. package/dist/extensions/zalouser/channel-plugin-api.js +1 -1
  258. package/dist/extensions/zalouser/runtime-api.js +6 -6
  259. package/dist/extensions/zalouser/setup-plugin-api.js +1 -1
  260. package/dist/extensions/zalouser/test-api.js +1 -1
  261. package/dist/fallbacks-BKA5XrxA.js +31 -0
  262. package/dist/fallbacks-QvezaDAr.js +2 -0
  263. package/dist/fallbacks-shared-DS-qTOLR.js +111 -0
  264. package/dist/gateway-C0YHaLXO.js +115 -0
  265. package/dist/gateway-cli-DJBCD8J5.js +1325 -0
  266. package/dist/gateway-rpc-CmJXd4SU.js +14 -0
  267. package/dist/gateway-rpc.runtime-C9Fs2mSx.js +23 -0
  268. package/dist/gateway-rpc.runtime.js +1 -1
  269. package/dist/gateway-runtime-Bm2_MhFn.js +15 -0
  270. package/dist/gateway-status-DXKMKFtB.js +584 -0
  271. package/dist/genesis-tools-AolFZ-0I.js +8999 -0
  272. package/dist/genesis-tools.runtime-CBWXJqzw.js +2 -0
  273. package/dist/genesis-tools.runtime.js +1 -1
  274. package/dist/get-reply-BpxjHQBp.js +3879 -0
  275. package/dist/get-reply-from-config.runtime-DJKEqLSi.js +2 -0
  276. package/dist/get-reply-from-config.runtime.js +1 -1
  277. package/dist/graph-users-D2Q5UTGQ.js +1337 -0
  278. package/dist/health-BNKpuAPB.js +469 -0
  279. package/dist/health-BkxZfKx8.js +3 -0
  280. package/dist/heartbeat-runner-Bdu2q6k3.js +5 -0
  281. package/dist/heartbeat-runner-C0B-c-fv.js +1292 -0
  282. package/dist/heartbeat-runner.runtime-Cic23eNy.js +4 -0
  283. package/dist/heartbeat-runner.runtime.js +1 -1
  284. package/dist/hooks-cli-BfCr2In5.js +433 -0
  285. package/dist/image-fallbacks-Dgy-LmDo.js +2 -0
  286. package/dist/image-fallbacks-RlfTqOne.js +31 -0
  287. package/dist/inbound-reply-dispatch-D4KJHWLG.js +73 -0
  288. package/dist/index.js +2 -2
  289. package/dist/infra-runtime-BTJv3KK8.js +39 -0
  290. package/dist/init-q4INH9uv.js +59 -0
  291. package/dist/install-BT4BNeMX.js +190 -0
  292. package/dist/install.runtime-BnqPyIyn.js +2 -0
  293. package/dist/launchd-BM_37PLv.js +698 -0
  294. package/dist/library-DGd4UZKC.js +45 -0
  295. package/dist/lifecycle-B_RLw4uY.js +229 -0
  296. package/dist/lifecycle-Bw01evSl.js +571 -0
  297. package/dist/lifecycle-core-CTjCo-k0.js +422 -0
  298. package/dist/lifecycle.runtime-BxpD2Ol7.js +2 -0
  299. package/dist/lifecycle.runtime.js +1 -1
  300. package/dist/list-BXa4zUF-.js +2 -0
  301. package/dist/list-BugTSXmQ.js +131 -0
  302. package/dist/list-CgC_rsta.js +1201 -0
  303. package/dist/list-D2sk1YPs.js +2 -0
  304. package/dist/list.probe-C6zfgLdG.js +419 -0
  305. package/dist/llm-slug-generator-DsWJvqNx.js +79 -0
  306. package/dist/llm-slug-generator.js +1 -1
  307. package/dist/load-config-D6ZKPY7I.js +35 -0
  308. package/dist/local-dispatch.runtime-MWMxyZKj.js +8 -0
  309. package/dist/local-dispatch.runtime.js +1 -1
  310. package/dist/logs-cli-Dn-8jl3z.js +265 -0
  311. package/dist/logs-cli.runtime-DwsccWUJ.js +2 -0
  312. package/dist/logs-cli.runtime.js +1 -1
  313. package/dist/main-session-restart-recovery-6Uh4W8ZJ.js +206 -0
  314. package/dist/managed-image-attachments-DHawPS1-.js +2 -0
  315. package/dist/managed-image-attachments-DjtxwZ9o.js +635 -0
  316. package/dist/manager-D0tkR46e.js +2057 -0
  317. package/dist/manager-g_P1RDky.js +2 -0
  318. package/dist/markdown-to-line-7AxR7QFo.js +790 -0
  319. package/dist/mcp/plugin-tools-serve.js +1 -1
  320. package/dist/mcp-cli-Cw2XQMOA.js +725 -0
  321. package/dist/mcp-http-DFKqya8U.js +529 -0
  322. package/dist/memory-core-host-runtime-cli-DaAsOXiU.js +9 -0
  323. package/dist/message-BDCG7eWj.js +232 -0
  324. package/dist/message-action-runner-6C7kXcC9.js +2 -0
  325. package/dist/message-action-runner-C2P3efhU.js +1407 -0
  326. package/dist/message-actions-BWQWt5gT.js +143 -0
  327. package/dist/message.gateway.runtime-BXChuYAf.js +2 -0
  328. package/dist/message.gateway.runtime.js +1 -1
  329. package/dist/models-auth-status-B8lnM1PS.js +201 -0
  330. package/dist/models-cli-DbUA18AN.js +219 -0
  331. package/dist/monitor-BEJe2ceJ.js +1237 -0
  332. package/dist/monitor-BO9i6U5R.js +1459 -0
  333. package/dist/monitor-BZLrxL_x.js +1661 -0
  334. package/dist/monitor-CDDxHGlD.js +788 -0
  335. package/dist/monitor-D779kLc0.js +671 -0
  336. package/dist/monitor-V5ZNCrQL.js +2 -0
  337. package/dist/monitor-auth-D2rCHNgr.js +207 -0
  338. package/dist/monitor-processing-BLbsNFVv.js +1974 -0
  339. package/dist/monitor.runtime-RrusGXVO.js +2 -0
  340. package/dist/monitor.runtime.js +1 -1
  341. package/dist/monitor.webhook-B0DuBNWa.js +180 -0
  342. package/dist/msteams-hh9nwLlZ.js +35 -0
  343. package/dist/native-hook-relay-BygYZoeR.js +519 -0
  344. package/dist/nextcloud-talk-AadRCqXn.js +17 -0
  345. package/dist/node-cli-FF1zz0JU.js +2506 -0
  346. package/dist/node-service-f8mmgM5L.js +68 -0
  347. package/dist/nodes-cli-tQ5dpecZ.js +1046 -0
  348. package/dist/nodes-utils-BfpGcUca.js +84 -0
  349. package/dist/nodes.helpers-CMXxavTs.js +34 -0
  350. package/dist/notify-DaNP5Zyh.js +315 -0
  351. package/dist/onboard-B9hx80Dm.js +632 -0
  352. package/dist/onboard-helpers-BlPrFJHK.js +204 -0
  353. package/dist/onboard-helpers-kL8mgUkl.js +6 -0
  354. package/dist/onboard-remote-ByHuQd1m.js +193 -0
  355. package/dist/onboard-remote-CRPXcrWy.js +2 -0
  356. package/dist/onboard-skills-ChtyNRCd.js +2 -0
  357. package/dist/onboard-skills-DROlYXya.js +134 -0
  358. package/dist/openai-http-2An5HAJU.js +500 -0
  359. package/dist/openresponses-http-DpbJXZP4.js +1128 -0
  360. package/dist/operator-approvals-client-BoN_VmkB.js +68 -0
  361. package/dist/outbound.runtime-D_aebUn6.js +2 -0
  362. package/dist/outbound.runtime.js +1 -1
  363. package/dist/pair-command-approve-i-1VECJx.js +44 -0
  364. package/dist/persistent-bindings.lifecycle-eZyKQ1D8.js +85 -0
  365. package/dist/persistent-bindings.lifecycle-mxY73TJR.js +2 -0
  366. package/dist/pi-embedded-0fErjTQ7.js +2905 -0
  367. package/dist/pi-embedded-Bc1jC2x0.js +4 -0
  368. package/dist/pi-embedded.runtime-Bn_RZ5z8.js +4 -0
  369. package/dist/pi-embedded.runtime.js +1 -1
  370. package/dist/pi-tool-definition-adapter-thh7LO1t.js +217 -0
  371. package/dist/pi-tools-BtlLDovQ.js +1057 -0
  372. package/dist/pi-tools.before-tool-call-BJ_X8Dq3.js +2 -0
  373. package/dist/pi-tools.before-tool-call-DdPCGWCs.js +433 -0
  374. package/dist/plugin-CdHHk4Xn.js +12195 -0
  375. package/dist/plugin-enabled-DybhbjKQ.js +140 -0
  376. package/dist/plugin-registration-Bgwpsokw.js +23 -0
  377. package/dist/plugin-sdk/.boundary-entry-shims.stamp +1 -1
  378. package/dist/plugin-sdk/acp-binding-runtime.js +1 -1
  379. package/dist/plugin-sdk/acp-runtime.js +2 -2
  380. package/dist/plugin-sdk/agent-harness-runtime.js +5 -5
  381. package/dist/plugin-sdk/agent-harness.js +6 -6
  382. package/dist/plugin-sdk/agent-runtime.js +2 -2
  383. package/dist/plugin-sdk/approval-gateway-runtime.js +2 -2
  384. package/dist/plugin-sdk/approval-handler-runtime.js +3 -3
  385. package/dist/plugin-sdk/approval-runtime.js +1 -1
  386. package/dist/plugin-sdk/browser-node-runtime.js +4 -4
  387. package/dist/plugin-sdk/browser-setup-tools.js +3 -3
  388. package/dist/plugin-sdk/browser-support.js +7 -7
  389. package/dist/plugin-sdk/channel-core.js +2 -2
  390. package/dist/plugin-sdk/channel-inbound.js +2 -2
  391. package/dist/plugin-sdk/command-auth.js +1 -1
  392. package/dist/plugin-sdk/command-status-runtime.js +1 -1
  393. package/dist/plugin-sdk/compat.js +1 -1
  394. package/dist/plugin-sdk/conversation-binding-runtime.js +1 -1
  395. package/dist/plugin-sdk/conversation-runtime.js +3 -3
  396. package/dist/plugin-sdk/core.js +2 -2
  397. package/dist/plugin-sdk/direct-dm.js +1 -1
  398. package/dist/plugin-sdk/gateway-runtime.js +3 -3
  399. package/dist/plugin-sdk/inbound-reply-dispatch.js +1 -1
  400. package/dist/plugin-sdk/index.js +1 -1
  401. package/dist/plugin-sdk/infra-runtime.js +2 -2
  402. package/dist/plugin-sdk/irc.js +2 -2
  403. package/dist/plugin-sdk/matrix.js +1 -1
  404. package/dist/plugin-sdk/memory-core-host-runtime-cli.js +2 -2
  405. package/dist/plugin-sdk/memory-core.js +2 -2
  406. package/dist/plugin-sdk/msteams.js +2 -2
  407. package/dist/plugin-sdk/nextcloud-talk.js +2 -2
  408. package/dist/plugin-sdk/nostr.js +1 -1
  409. package/dist/plugin-sdk/reply-dispatch-runtime.js +1 -1
  410. package/dist/plugin-sdk/reply-runtime.js +4 -4
  411. package/dist/plugin-sdk/runtime-secret-resolution.js +1 -1
  412. package/dist/plugin-sdk/runtime.js +2 -2
  413. package/dist/plugin-sdk/session-visibility.js +1 -1
  414. package/dist/plugin-sdk/testing.js +4 -4
  415. package/dist/plugin-sdk/tlon.js +1 -1
  416. package/dist/plugin-sdk/zalo.js +1 -1
  417. package/dist/plugin-sdk/zalouser.js +1 -1
  418. package/dist/plugin-service-ClhrJfT3.js +2890 -0
  419. package/dist/plugins/runtime/index.js +1 -1
  420. package/dist/policy-CoqHgREn.js +328 -0
  421. package/dist/postinstall-inventory.json +418 -418
  422. package/dist/prepare.runtime-CawO_32K.js +815 -0
  423. package/dist/prepare.runtime.js +1 -1
  424. package/dist/probe-BP3QS65i.js +74 -0
  425. package/dist/probe-BdCXAH_u.js +2 -0
  426. package/dist/probe-Cv5tXOMP.js +2205 -0
  427. package/dist/probe-D-D_1oc4.js +241 -0
  428. package/dist/probe-DO-gll0g.js +2 -0
  429. package/dist/probe-T6c0F73q.js +45 -0
  430. package/dist/probe-dxji7IxZ.js +1443 -0
  431. package/dist/program-GMPfHt5E.js +111 -0
  432. package/dist/prompt-select-styled-BvuJNEJG.js +20 -0
  433. package/dist/protocol-Batc2DmY.js +2234 -0
  434. package/dist/provider-dispatcher-CGYoOdKf.js +2 -0
  435. package/dist/provider-dispatcher-iTEDM88F.js +22 -0
  436. package/dist/qr-cli-6IxenO1C.js +349 -0
  437. package/dist/qr-cli-D63piSog.js +2 -0
  438. package/dist/reaction-runtime-api-DgwRXTz2.js +116 -0
  439. package/dist/reactions-Dnh7t4Zl.js +998 -0
  440. package/dist/register-service-commands-D7w01SKL.js +71 -0
  441. package/dist/register.agent-j1Eno4xr.js +248 -0
  442. package/dist/register.configure-ubwxTkGt.js +15 -0
  443. package/dist/register.maintenance-B7FuM_ZW.js +438 -0
  444. package/dist/register.message-QR3u9rBl.js +329 -0
  445. package/dist/register.onboard-Bdcf_lH0.js +81 -0
  446. package/dist/register.runtime-QDduc4yj.js +81 -0
  447. package/dist/register.runtime.js +1 -1
  448. package/dist/register.setup-DzVV6tdQ.js +150 -0
  449. package/dist/register.status-health-sessions-Brw3VcxG.js +1215 -0
  450. package/dist/register.subclis-B8qDbqPl.js +29 -0
  451. package/dist/register.subclis-DdoN3nZi.js +3 -0
  452. package/dist/register.subclis-core-CWHmnIoe.js +249 -0
  453. package/dist/reply-dispatch-runtime-BVLvCeJ0.js +13 -0
  454. package/dist/reply-runtime-BXkvfLv_.js +11 -0
  455. package/dist/reply.runtime-BTkpxI5R.js +2 -0
  456. package/dist/reply.runtime.js +1 -1
  457. package/dist/restart-health-6cjrRBpF.js +202 -0
  458. package/dist/restart-health-b6Qw43Tj.js +2 -0
  459. package/dist/root-help-DDa1oEtT.js +44 -0
  460. package/dist/routes-CoCMty69.js +2 -0
  461. package/dist/routes-D9uC_Zdy.js +3341 -0
  462. package/dist/rpc-DpLGM3FH.js +61 -0
  463. package/dist/rpc.runtime-B27EA5A0.js +21 -0
  464. package/dist/rpc.runtime.js +1 -1
  465. package/dist/run-delivery.runtime-OmILefLX.js +530 -0
  466. package/dist/run-delivery.runtime.js +1 -1
  467. package/dist/run-embedded.runtime-Dc43u_Z7.js +4 -0
  468. package/dist/run-embedded.runtime.js +1 -1
  469. package/dist/run-execution-cli.runtime-DL5w7Ac8.js +4 -0
  470. package/dist/run-execution-cli.runtime.js +1 -1
  471. package/dist/run-executor.runtime-Bzwt0zP-.js +277 -0
  472. package/dist/run-executor.runtime.js +1 -1
  473. package/dist/run-main-BLD1wTLU.js +516 -0
  474. package/dist/run-subagent-registry.runtime-CddCeg6W.js +2 -0
  475. package/dist/run-subagent-registry.runtime.js +1 -1
  476. package/dist/run-wait-D4Gxuyru.js +135 -0
  477. package/dist/runtime-8BuIPrSY.js +973 -0
  478. package/dist/runtime-Co8r21pw.js +9 -0
  479. package/dist/runtime-api-BG0XWkir.js +4 -0
  480. package/dist/runtime-api-BjBzlgns.js +9 -0
  481. package/dist/runtime-api-DlYbbA-n.js +14 -0
  482. package/dist/runtime-api-faT33SE2.js +9 -0
  483. package/dist/runtime-embedded-pi.runtime-QdJ7K4t0.js +2 -0
  484. package/dist/runtime-embedded-pi.runtime.js +1 -1
  485. package/dist/runtime-internal-C0xc_Zhf.js +2 -0
  486. package/dist/runtime-options-DLv7ygkO.js +275 -0
  487. package/dist/runtime-schema-CQK5R5Pl.js +27780 -0
  488. package/dist/scan-Yz6DoQn5.js +523 -0
  489. package/dist/scan-k38hL_6o.js +2 -0
  490. package/dist/secrets-cli-Bjvr0bOw.js +2101 -0
  491. package/dist/security-cli-C2gkMFcB.js +486 -0
  492. package/dist/selection-DC8EXnRg.js +7736 -0
  493. package/dist/selection-DmSFI5k4.js +2 -0
  494. package/dist/send-CT6EfHrc.js +102 -0
  495. package/dist/send-_ANvjE_C.js +156 -0
  496. package/dist/send.runtime-COIjG4BV.js +2 -0
  497. package/dist/send.runtime.js +1 -1
  498. package/dist/server-COkv9Si4.js +13 -0
  499. package/dist/server-DHLQZJL8.js +77 -0
  500. package/dist/server-context-6jnr-aAw.js +2 -0
  501. package/dist/server-context-DK1QsdLh.js +847 -0
  502. package/dist/server-node-events-q8H0odkq.js +481 -0
  503. package/dist/server-plugin-bootstrap-Bw4a88yA.js +2 -0
  504. package/dist/server-plugin-bootstrap-By_w9ngt.js +11333 -0
  505. package/dist/server-restart-sentinel-CGLYF3mk.js +697 -0
  506. package/dist/server.impl-8j4SImks.js +12735 -0
  507. package/dist/service-CNjb_qXC.js +2 -0
  508. package/dist/service-CyR3mZIU.js +120 -0
  509. package/dist/service-audit-DIL0OiMa.js +260 -0
  510. package/dist/service-audit-pwloDggS.js +2 -0
  511. package/dist/session-kill-http-Dek5w0Ee.js +110 -0
  512. package/dist/session-reset-service-B_9Ps0kr.js +471 -0
  513. package/dist/session-route-Cr6vOd9_.js +93 -0
  514. package/dist/session-status.runtime-C_x1MHH5.js +2 -0
  515. package/dist/session-status.runtime.js +1 -1
  516. package/dist/session-subagent-reactivation.runtime-BKWuNild.js +2 -0
  517. package/dist/session-subagent-reactivation.runtime.js +1 -1
  518. package/dist/session-tab-registry-C08V5jx6.js +491 -0
  519. package/dist/session-visibility-D4j8hN_B.js +147 -0
  520. package/dist/sessions-helpers-DmUVRk16.js +304 -0
  521. package/dist/sessions-history-http-DN3KzEnX.js +383 -0
  522. package/dist/sessions-patch-BPtlF0Lu.js +309 -0
  523. package/dist/sessions-resolve-D6KPfG_4.js +174 -0
  524. package/dist/sessions.runtime-DGL1-G20.js +2 -0
  525. package/dist/sessions.runtime.js +1 -1
  526. package/dist/setup-CkYiQoP0.js +495 -0
  527. package/dist/setup-api-CUGYqGNT.js +29 -0
  528. package/dist/setup-core-BmplfubJ.js +171 -0
  529. package/dist/setup-core-C5ddWpfy.js +176 -0
  530. package/dist/setup-surface-2JmEfowI.js +219 -0
  531. package/dist/setup-surface-BzHQx_GD.js +403 -0
  532. package/dist/setup-surface-DOH6kc6w.js +286 -0
  533. package/dist/setup.finalize-Ci2HBkDq.js +539 -0
  534. package/dist/setup.gateway-config-CyG_Fzaa.js +250 -0
  535. package/dist/shared-BItUH43s.js +121 -0
  536. package/dist/shared-D7diJkX6.js +198 -0
  537. package/dist/shared-DJU_HCJ1.js +76 -0
  538. package/dist/slash-state-C4YD-tLz.js +1911 -0
  539. package/dist/src-8HdytGc7.js +3974 -0
  540. package/dist/startup-context-CfcCyUU0.js +312 -0
  541. package/dist/status-B94yciC9.js +397 -0
  542. package/dist/status-BUkUYtFm.js +190 -0
  543. package/dist/status-BsHC-0MV.js +2 -0
  544. package/dist/status-D4jNATg0.js +209 -0
  545. package/dist/status-U8ut-X2Q.js +2 -0
  546. package/dist/status-all-CIzsXQnV.js +498 -0
  547. package/dist/status-json-alPkfWTd.js +14 -0
  548. package/dist/status-json-command-CAkzWzw7.js +84 -0
  549. package/dist/status-jxNdVQVZ.js +3 -0
  550. package/dist/status-runtime-shared-BSsA48i9.js +257 -0
  551. package/dist/status-subagents.runtime-hIKg6PXV.js +18 -0
  552. package/dist/status-subagents.runtime.js +1 -1
  553. package/dist/status-text-BmLJrLth.js +237 -0
  554. package/dist/status.gateway-connection.runtime-Dr7HJ1wZ.js +2 -0
  555. package/dist/status.gateway-connection.runtime.js +1 -1
  556. package/dist/status.gather-DFY0T97g.js +2 -0
  557. package/dist/status.gather-e6sfNh8I.js +292 -0
  558. package/dist/status.runtime-DBvapXQS.js +2 -0
  559. package/dist/status.runtime.js +1 -1
  560. package/dist/status.scan-DSpkE-Q-.js +65 -0
  561. package/dist/status.scan-overview-Dr72bkbi.js +379 -0
  562. package/dist/status.scan.fast-json-C7k_m46S.js +2 -0
  563. package/dist/status.scan.fast-json-KHh7R3wX.js +132 -0
  564. package/dist/status.summary-C4r9lkCH.js +214 -0
  565. package/dist/status.summary-D0EJwkmX.js +2 -0
  566. package/dist/subagent-announce-Bu5-8O3P.js +351 -0
  567. package/dist/subagent-announce-delivery-4hk9P48s.js +726 -0
  568. package/dist/subagent-announce-output-Csyi1-Kz.js +364 -0
  569. package/dist/subagent-control-DAeNsV_G.js +506 -0
  570. package/dist/subagent-followup.runtime-Bch92LIq.js +68 -0
  571. package/dist/subagent-followup.runtime.js +1 -1
  572. package/dist/subagent-orphan-recovery-mNzyr1b-.js +305 -0
  573. package/dist/subagent-registry-BMjQxBxi.js +3 -0
  574. package/dist/subagent-registry-D4DFKQCL.js +1753 -0
  575. package/dist/subagent-registry.runtime.js +1 -1
  576. package/dist/subagent-spawn-C13uUm47.js +1005 -0
  577. package/dist/system-cli-BU6GhJxG.js +59 -0
  578. package/dist/targets-eYkepVzz.js +67 -0
  579. package/dist/task-executor-DzJLcm4D.js +360 -0
  580. package/dist/task-owner-access-DW9EbhjP.js +74 -0
  581. package/dist/task-registry-DBJkAtFF.js +2366 -0
  582. package/dist/task-registry-delivery-runtime-BdkPeGC1.js +2 -0
  583. package/dist/task-registry-delivery-runtime-Ov88LGnv.js +3 -0
  584. package/dist/task-registry.maintenance-BU6Z4V_9.js +416 -0
  585. package/dist/task-registry.maintenance-sEo6eYdx.js +2 -0
  586. package/dist/telegram/token.js +1 -1
  587. package/dist/testing-B2jHeQft.js +575 -0
  588. package/dist/text-report-HYrgZP8-.js +587 -0
  589. package/dist/tool-resolution-BZxaDOKg.js +90 -0
  590. package/dist/tools-effective-inventory-BCvCprpD.js +152 -0
  591. package/dist/tools-invoke-http-Dt1rsx4a.js +206 -0
  592. package/dist/trash-F4tvjrKS.js +24 -0
  593. package/dist/tui-cli-DlQ1oejy.js +4575 -0
  594. package/dist/update-cli-DHMdwQWA.js +1759 -0
  595. package/dist/upgrade-UR-EI5pO.js +1226 -0
  596. package/dist/video-generation-task-status-qtHrIZLt.js +163 -0
  597. package/dist/wait-for-idle-before-flush-CYB_61O0.js +5986 -0
  598. package/dist/wizard-models-BKqMCaCm.js +161 -0
  599. package/package.json +1 -1
  600. package/dist/abort-BHSttDhY.js +0 -201
  601. package/dist/abort.runtime-VJ9lFsZS.js +0 -2
  602. package/dist/accounts-BNvagEOQ.js +0 -104
  603. package/dist/accounts-CPIuzenz.js +0 -107
  604. package/dist/accounts-r08DQWhZ.js +0 -2
  605. package/dist/acp-cli-i0L-eRs9.js +0 -2193
  606. package/dist/acp-spawn-CHzVLe9q.js +0 -1093
  607. package/dist/acp-spawn-CqvxDuBq.js +0 -2
  608. package/dist/acp-stateful-target-driver-CWRLc_NP.js +0 -89
  609. package/dist/action-agents-SNCscX_-.js +0 -67
  610. package/dist/action-focus-B2s0PF2E.js +0 -132
  611. package/dist/action-help-CXKbyaZ7.js +0 -7
  612. package/dist/action-info-B3IDKxWu.js +0 -101
  613. package/dist/action-kill-idGvCNhT.js +0 -33
  614. package/dist/action-list-Di7gO3qL.js +0 -21
  615. package/dist/action-log-BCsz-gFi.js +0 -30
  616. package/dist/action-send-DLsdZnVc.js +0 -39
  617. package/dist/action-spawn-DhKEOdL0.js +0 -47
  618. package/dist/action-unfocus-Dh7ti5UP.js +0 -29
  619. package/dist/actions.runtime-BPf03SN3.js +0 -18
  620. package/dist/actions.runtime-CJg_lweh.js +0 -5
  621. package/dist/agent-C3PLvvws.js +0 -2
  622. package/dist/agent-command-UroeNrV4.js +0 -874
  623. package/dist/agent-harness-runtime-MXvI9FlJ.js +0 -144
  624. package/dist/agent-runner-utils-CaVgLZrf.js +0 -239
  625. package/dist/agent-runner.runtime-Csqm3m09.js +0 -3455
  626. package/dist/agent-runtime-B9nUYDUz.js +0 -18
  627. package/dist/agents-BP4p-1q2.js +0 -5
  628. package/dist/agents-CDiXfrfc.js +0 -953
  629. package/dist/aliases-BBCtCq2A.js +0 -96
  630. package/dist/aliases-DrFtFq1p.js +0 -2
  631. package/dist/api-BH0oEwR1.js +0 -5
  632. package/dist/api-D-4gHdrl.js +0 -139
  633. package/dist/api-DE6RYTxv.js +0 -3
  634. package/dist/approval-gateway-resolver-URpDQMld.js +0 -29
  635. package/dist/approval-gateway-runtime-B087BMms.js +0 -2
  636. package/dist/approval-handler-runtime-BEl3ua8-.js +0 -439
  637. package/dist/approval-native-runtime-ghjYGufu.js +0 -729
  638. package/dist/attempt-execution.runtime-PCTFYqLh.js +0 -509
  639. package/dist/attempt.prompt-helpers-CkWEozQ2.js +0 -221
  640. package/dist/attempt.tool-run-context-DbqVgXFk.js +0 -933
  641. package/dist/audit-Lb1yNEwg.js +0 -939
  642. package/dist/audit.runtime-DOnixKE_.js +0 -7
  643. package/dist/auth-BrlJ7NM9.js +0 -2
  644. package/dist/auth-CJoP7Yst.js +0 -383
  645. package/dist/auth-order-D0KFgBib.js +0 -96
  646. package/dist/auth-order-_xlUHzCg.js +0 -2
  647. package/dist/bash-tools-BEVEHCl9.js +0 -2824
  648. package/dist/bash-tools-DJ0D8Wr5.js +0 -3
  649. package/dist/binding-routing-IfKqPcfO.js +0 -85
  650. package/dist/binding-targets-2huR0hTX.js +0 -121
  651. package/dist/bridge-server-D8Y8Fzdz.js +0 -113
  652. package/dist/browser-control-auth-Q6UpVLEj.js +0 -2
  653. package/dist/browser-node-runtime-C9KvsOyx.js +0 -12
  654. package/dist/browser-profiles-DFcwiMEf.js +0 -2
  655. package/dist/browser-runtime-sbZ3hKg6.js +0 -387
  656. package/dist/browser-setup-tools-CP4BkZ9z.js +0 -13
  657. package/dist/build-DzHX2LrH.js +0 -550
  658. package/dist/call-ORivR4Db.js +0 -331
  659. package/dist/call-qzfm6lro.js +0 -3
  660. package/dist/call.runtime-BkzJe07B.js +0 -2
  661. package/dist/capability-cli-BDCvm93o.js +0 -1401
  662. package/dist/catchup-CjZa2PrP.js +0 -300
  663. package/dist/channel-B3UAL9Pk.js +0 -840
  664. package/dist/channel-B5L9LJBF.js +0 -453
  665. package/dist/channel-BGXl6A-E.js +0 -297
  666. package/dist/channel-BNUblayu.js +0 -1320
  667. package/dist/channel-Bej-TN7f.js +0 -491
  668. package/dist/channel-BzOKSC4e.js +0 -350
  669. package/dist/channel-CY2xEOzR.js +0 -1802
  670. package/dist/channel-D2kKLWUA.js +0 -226
  671. package/dist/channel-DcjCe46M.js +0 -1100
  672. package/dist/channel-GqA96nmt.js +0 -595
  673. package/dist/channel-YI2aIJFQ.js +0 -1174
  674. package/dist/channel-core-DqPUhmTt.js +0 -5
  675. package/dist/channel-inbound-CzgMd2Xi.js +0 -31
  676. package/dist/channel-plugin-runtime-DI5uUbxx.js +0 -771
  677. package/dist/channel-runtime-BJqKQitD.js +0 -425
  678. package/dist/channel.runtime-BBn9mgbB.js +0 -89
  679. package/dist/channel.runtime-Bhj1b9gQ.js +0 -34702
  680. package/dist/channel.runtime-BzOVx90Z.js +0 -576
  681. package/dist/channel.runtime-CQkLKWlq.js +0 -109
  682. package/dist/channel.runtime-D8qT4A1x.js +0 -4
  683. package/dist/channel.runtime-DjK9Re1w.js +0 -2364
  684. package/dist/channel.runtime-tWwM8hFw.js +0 -430
  685. package/dist/channel.setup-gaiRZ-r8.js +0 -10
  686. package/dist/channels-BXIl3hBo.js +0 -733
  687. package/dist/channels-cli-CrmBqaul.js +0 -268
  688. package/dist/chat-C42usv-F.js +0 -2758
  689. package/dist/clawbot-cli-BGYO_ymZ.js +0 -9
  690. package/dist/cli-BbD25CFK.js +0 -2
  691. package/dist/cli-CYD4Wfcq.js +0 -219
  692. package/dist/cli-D-6ycHAw.js +0 -72
  693. package/dist/cli-DX4CK-bw.js +0 -2
  694. package/dist/cli-runner-C5_PBTxm.js +0 -286
  695. package/dist/cli-runner.runtime-DE7RpQ27.js +0 -3
  696. package/dist/cli-runner.runtime-ueVMICzY.js +0 -4
  697. package/dist/cli.runtime-DSO8Vfrh.js +0 -1261
  698. package/dist/client-BKro32pr.js +0 -138
  699. package/dist/client-Vx7pseEY.js +0 -713
  700. package/dist/command-auth-CvFUTCCJ.js +0 -76
  701. package/dist/command-config-resolution-BnW1XGnW.js +0 -23
  702. package/dist/command-config-resolution-BrlfYyiN.js +0 -2
  703. package/dist/command-config-resolution.runtime-DYMr8J0t.js +0 -2
  704. package/dist/command-registry-Bi9thw1b.js +0 -9
  705. package/dist/command-registry-D__zH4Fg.js +0 -4
  706. package/dist/command-registry-core-DQoNFQZT.js +0 -101
  707. package/dist/command-secret-gateway-B27Zpgo9.js +0 -528
  708. package/dist/command-status.runtime-fERQMzeI.js +0 -87
  709. package/dist/commands-acp-BmORmf3-.js +0 -77
  710. package/dist/commands-compact.runtime-BUaXIw-I.js +0 -10
  711. package/dist/commands-handlers.runtime-BxbkeuTc.js +0 -4597
  712. package/dist/commands-status-4OW5sqmg.js +0 -16
  713. package/dist/commands-status.runtime-Cl7nJp_7.js +0 -3
  714. package/dist/commands-subagents-control.runtime-BR4qBGkk.js +0 -2
  715. package/dist/commands-subagents-control.runtime-C8ufgKPI.js +0 -3
  716. package/dist/commands-system-prompt-BsimvjJt.js +0 -158
  717. package/dist/commands-system-prompt-CKrY0h7P.js +0 -2
  718. package/dist/commands.runtime-BHsDdaEU.js +0 -166
  719. package/dist/compact-BHYLQst3.js +0 -1118
  720. package/dist/compact.runtime-7-tZRdgw.js +0 -12
  721. package/dist/completion-cli-1sAiSIYc.js +0 -328
  722. package/dist/config-BZ3FuFjH.js +0 -251
  723. package/dist/config-cli-B_dNq5hD.js +0 -1078
  724. package/dist/configure-BOvmX2NJ.js +0 -2
  725. package/dist/configure-DR9NYN9e.js +0 -1245
  726. package/dist/connect-options-DlmPTyhG.js +0 -699
  727. package/dist/control-auth-C_zNiV10.js +0 -125
  728. package/dist/control-service-B1TL51jf.js +0 -156
  729. package/dist/conversation-id-B3lLiTfF.js +0 -235
  730. package/dist/conversation-id-DORmTZm_.js +0 -38
  731. package/dist/conversation-runtime-DDxZZZXE.js +0 -31
  732. package/dist/core-DbPzffGG.js +0 -275
  733. package/dist/cron-cli-AoH4jhFt.js +0 -713
  734. package/dist/daemon-cli-BOJdM3wT.js +0 -12
  735. package/dist/daemon-install-D8HPES4u.js +0 -64
  736. package/dist/delegate-Dmdda3kT.js +0 -64
  737. package/dist/detached-task-runtime-BB5az34R.js +0 -73
  738. package/dist/devices-cli-Cy9DF-DP.js +0 -496
  739. package/dist/diagnostics-CTE0TWR1.js +0 -154
  740. package/dist/direct-dm-9jzadx9u.js +0 -64
  741. package/dist/dispatch-DU7cqfv6.js +0 -1131
  742. package/dist/dispatch-acp-Cu_nUtKg.js +0 -981
  743. package/dist/dispatch-acp-manager.runtime-8MOQ5BRn.js +0 -3
  744. package/dist/dispatch-acp.runtime-DatcHNJq.js +0 -19
  745. package/dist/doctor-device-pairing-Qm-r7mwu.js +0 -307
  746. package/dist/doctor-gateway-daemon-flow-BN2uGhFJ.js +0 -250
  747. package/dist/doctor-gateway-health-CD7Vzth9.js +0 -60
  748. package/dist/doctor-gateway-services-CIypZgv_.js +0 -316
  749. package/dist/doctor-health-ByjG3620.js +0 -59
  750. package/dist/doctor-health-contributions-CxNByg_i.js +0 -486
  751. package/dist/doctor-prompter-BYIoKiZ8.js +0 -56
  752. package/dist/doctor-workspace-status-DL2fRPoT.js +0 -75
  753. package/dist/dreaming-CCctNgkQ.js +0 -1574
  754. package/dist/dreaming-narrative-BPtXKc2-.js +0 -595
  755. package/dist/embedded-gateway-stub.runtime-Hi36BeCq.js +0 -9
  756. package/dist/exec-approvals-cli-DyslUWtQ.js +0 -498
  757. package/dist/fallbacks-B3xZC-ms.js +0 -2
  758. package/dist/fallbacks-CWDz-tSa.js +0 -31
  759. package/dist/fallbacks-shared-BlZBjxHL.js +0 -111
  760. package/dist/gateway-NYBkUx5r.js +0 -115
  761. package/dist/gateway-cli-E5gYA0U1.js +0 -1325
  762. package/dist/gateway-rpc-BrG5Lkfm.js +0 -14
  763. package/dist/gateway-rpc.runtime-BrTGchr9.js +0 -23
  764. package/dist/gateway-runtime-DctMRv_p.js +0 -15
  765. package/dist/gateway-status-B6ol1agz.js +0 -584
  766. package/dist/genesis-tools-D_ah3Zek.js +0 -8999
  767. package/dist/genesis-tools.runtime-Cy7aTWEN.js +0 -2
  768. package/dist/get-reply-BvVkxYum.js +0 -3879
  769. package/dist/get-reply-from-config.runtime-DR3yiK1R.js +0 -2
  770. package/dist/graph-users-yBNugoFz.js +0 -1337
  771. package/dist/health-D_wk2s7j.js +0 -3
  772. package/dist/health-MRnjOx-_.js +0 -469
  773. package/dist/heartbeat-runner-NuhhYnxs.js +0 -5
  774. package/dist/heartbeat-runner-Y4NeUV3L.js +0 -1292
  775. package/dist/heartbeat-runner.runtime-eGfKtcpP.js +0 -4
  776. package/dist/hooks-cli-CtfN2vc-.js +0 -433
  777. package/dist/image-fallbacks-CgeDMYZo.js +0 -2
  778. package/dist/image-fallbacks-CzQ85Eo8.js +0 -31
  779. package/dist/inbound-reply-dispatch-jVlLaSqn.js +0 -73
  780. package/dist/infra-runtime-D2Pqjk-r.js +0 -39
  781. package/dist/init-D_MTYqrw.js +0 -59
  782. package/dist/install-X7wgrwQ1.js +0 -190
  783. package/dist/install.runtime-C4wO4Nr9.js +0 -2
  784. package/dist/launchd-lFgfiDrN.js +0 -688
  785. package/dist/library-DIykYIWD.js +0 -45
  786. package/dist/lifecycle-BNuFzffC.js +0 -571
  787. package/dist/lifecycle-IWN9qlwo.js +0 -229
  788. package/dist/lifecycle-core-CFv5qkS8.js +0 -422
  789. package/dist/lifecycle.runtime-9oo2JCI2.js +0 -2
  790. package/dist/list-BZ3mMSpA.js +0 -2
  791. package/dist/list-CzQs_k0U.js +0 -131
  792. package/dist/list-CzTJTFOH.js +0 -1201
  793. package/dist/list-qDXsgKhB.js +0 -2
  794. package/dist/list.probe-EOZR6ueo.js +0 -419
  795. package/dist/llm-slug-generator-DzWIx0nh.js +0 -79
  796. package/dist/load-config-C_8uUnXU.js +0 -35
  797. package/dist/local-dispatch.runtime-DLd10Xb1.js +0 -8
  798. package/dist/logs-cli-mnOTmC9R.js +0 -265
  799. package/dist/logs-cli.runtime-BvP3aise.js +0 -2
  800. package/dist/main-session-restart-recovery-D7Vxmu7e.js +0 -206
  801. package/dist/managed-image-attachments-BB5u0Zq8.js +0 -2
  802. package/dist/managed-image-attachments-BeRGNdL9.js +0 -635
  803. package/dist/manager-BhNWV4EC.js +0 -2
  804. package/dist/manager-CbZ9ncZs.js +0 -2057
  805. package/dist/markdown-to-line-DyB8w7ef.js +0 -790
  806. package/dist/mcp-cli-CWzLBrDF.js +0 -725
  807. package/dist/mcp-http-1jD7LE0a.js +0 -529
  808. package/dist/memory-core-host-runtime-cli-P80xoEfr.js +0 -9
  809. package/dist/message-BAhd_1ud.js +0 -232
  810. package/dist/message-action-runner-CH2jjT21.js +0 -1407
  811. package/dist/message-action-runner-DW7z_bMx.js +0 -2
  812. package/dist/message-actions-BjBGSsJB.js +0 -143
  813. package/dist/message.gateway.runtime-sUc85g7X.js +0 -2
  814. package/dist/models-auth-status-D6lrM56M.js +0 -201
  815. package/dist/models-cli-CmnbAk2M.js +0 -219
  816. package/dist/monitor-AyGe8vyk.js +0 -1237
  817. package/dist/monitor-DQuy2kx-.js +0 -671
  818. package/dist/monitor-DmtJiX5o.js +0 -788
  819. package/dist/monitor-GB8lcnal.js +0 -1459
  820. package/dist/monitor-IuVjATac.js +0 -1661
  821. package/dist/monitor-auth-rF7mr3Cm.js +0 -207
  822. package/dist/monitor-bnOvJyLN.js +0 -2
  823. package/dist/monitor-processing-B1ewr2qF.js +0 -1974
  824. package/dist/monitor.runtime-BrUD5_VN.js +0 -2
  825. package/dist/monitor.webhook-BvJG6Xbh.js +0 -180
  826. package/dist/msteams-DUDRy3xt.js +0 -35
  827. package/dist/native-hook-relay-CTpTLgCA.js +0 -519
  828. package/dist/nextcloud-talk-BH2uWRZ2.js +0 -17
  829. package/dist/node-cli-AWE1ZFfV.js +0 -2506
  830. package/dist/node-service-DCI-s3Mm.js +0 -68
  831. package/dist/nodes-cli-B7IWCrYq.js +0 -1046
  832. package/dist/nodes-utils-Dg2hrC3z.js +0 -84
  833. package/dist/nodes.helpers-CkJO5i0d.js +0 -34
  834. package/dist/notify-bsxfkOWp.js +0 -315
  835. package/dist/onboard-BQaZtwIU.js +0 -632
  836. package/dist/onboard-helpers-Cc2AnoIU.js +0 -6
  837. package/dist/onboard-helpers-DIxfEXNM.js +0 -204
  838. package/dist/onboard-remote-CYhQbOXH.js +0 -2
  839. package/dist/onboard-remote-DrH3yVxE.js +0 -193
  840. package/dist/onboard-skills-DCI3hVXa.js +0 -134
  841. package/dist/onboard-skills-i4KuFzS4.js +0 -2
  842. package/dist/openai-http-D_7Nk3iN.js +0 -500
  843. package/dist/openresponses-http-BITxvD4V.js +0 -1128
  844. package/dist/operator-approvals-client-C0t_-2xC.js +0 -68
  845. package/dist/outbound.runtime-C8_lBQcx.js +0 -2
  846. package/dist/pair-command-approve-D3s1W5go.js +0 -44
  847. package/dist/persistent-bindings.lifecycle-CCR6khHJ.js +0 -85
  848. package/dist/persistent-bindings.lifecycle-Dgoo5bpE.js +0 -2
  849. package/dist/pi-embedded-C2l80M7j.js +0 -4
  850. package/dist/pi-embedded-DMHGJgoQ.js +0 -2905
  851. package/dist/pi-embedded.runtime-fA0EsIue.js +0 -4
  852. package/dist/pi-tool-definition-adapter-B4hSYdYX.js +0 -217
  853. package/dist/pi-tools-YCHBvJG0.js +0 -1057
  854. package/dist/pi-tools.before-tool-call-BsyruRs4.js +0 -433
  855. package/dist/pi-tools.before-tool-call-D1maRTbl.js +0 -2
  856. package/dist/plugin-DmBZwzWQ.js +0 -12195
  857. package/dist/plugin-enabled-rr_Y1Kwh.js +0 -140
  858. package/dist/plugin-registration-J1JWqMZg.js +0 -23
  859. package/dist/plugin-service-CPY8FSW2.js +0 -2890
  860. package/dist/policy-DDUw681i.js +0 -328
  861. package/dist/prepare.runtime-BfnpyhHB.js +0 -815
  862. package/dist/probe-CCdD6_vN.js +0 -1443
  863. package/dist/probe-CakOsrSu.js +0 -241
  864. package/dist/probe-Ctrcd_Q7.js +0 -2205
  865. package/dist/probe-D9IFIte1.js +0 -2
  866. package/dist/probe-DXMI0QlZ.js +0 -74
  867. package/dist/probe-KtX2HjXM.js +0 -45
  868. package/dist/probe-OcOYm1bj.js +0 -2
  869. package/dist/program-DxZA1yAX.js +0 -111
  870. package/dist/prompt-select-styled-CATG4Xp5.js +0 -20
  871. package/dist/protocol-C40iRm9c.js +0 -2234
  872. package/dist/provider-dispatcher-BZp3Stzr.js +0 -2
  873. package/dist/provider-dispatcher-JJxN7BDL.js +0 -22
  874. package/dist/qr-cli-IMKzycXh.js +0 -349
  875. package/dist/qr-cli-W8X3Ha5T.js +0 -2
  876. package/dist/reaction-runtime-api-D7i4oMn9.js +0 -116
  877. package/dist/reactions-D9eRHeXM.js +0 -998
  878. package/dist/register-service-commands-D37K4_BJ.js +0 -71
  879. package/dist/register.agent-p1q6ZWSs.js +0 -248
  880. package/dist/register.configure-tgXCoIHv.js +0 -15
  881. package/dist/register.maintenance-BtBgu8zC.js +0 -438
  882. package/dist/register.message-B5msOUKt.js +0 -329
  883. package/dist/register.onboard-BO11PDUs.js +0 -81
  884. package/dist/register.runtime-D8XvwfoZ.js +0 -81
  885. package/dist/register.setup-Ba19XSJi.js +0 -150
  886. package/dist/register.status-health-sessions-Bne0mmgw.js +0 -1215
  887. package/dist/register.subclis--mdYCZZM.js +0 -3
  888. package/dist/register.subclis-_kuyreQm.js +0 -29
  889. package/dist/register.subclis-core-UkVTqwB1.js +0 -249
  890. package/dist/reply-dispatch-runtime-DcxOLr7n.js +0 -13
  891. package/dist/reply-runtime-B4VQJnqi.js +0 -11
  892. package/dist/reply.runtime-B990ty9C.js +0 -2
  893. package/dist/restart-health-BWUPc4R-.js +0 -202
  894. package/dist/restart-health-CWvsjHVN.js +0 -2
  895. package/dist/root-help-D9aeVXNr.js +0 -44
  896. package/dist/routes-BSmEgP46.js +0 -2
  897. package/dist/routes-Z67chkN5.js +0 -3341
  898. package/dist/rpc-CBQZvJME.js +0 -61
  899. package/dist/rpc.runtime-ClhNVRer.js +0 -21
  900. package/dist/run-delivery.runtime-Bj566BTH.js +0 -530
  901. package/dist/run-embedded.runtime-DuB-mQh2.js +0 -4
  902. package/dist/run-execution-cli.runtime-Oy4FGfj5.js +0 -4
  903. package/dist/run-executor.runtime-B7-lfFh3.js +0 -277
  904. package/dist/run-main-1Fvwazhp.js +0 -516
  905. package/dist/run-subagent-registry.runtime-Bx9Wga3d.js +0 -2
  906. package/dist/run-wait-C-VgbefQ.js +0 -135
  907. package/dist/runtime-BV7JmOaE.js +0 -9
  908. package/dist/runtime-D7npxl4G.js +0 -973
  909. package/dist/runtime-api-BkVIoPU9.js +0 -4
  910. package/dist/runtime-api-CdhgyHMB.js +0 -9
  911. package/dist/runtime-api-CmlBBCCY.js +0 -9
  912. package/dist/runtime-api-DnhXPE_z.js +0 -14
  913. package/dist/runtime-embedded-pi.runtime-DpKqDUrl.js +0 -2
  914. package/dist/runtime-internal-Bu1n9i4P.js +0 -2
  915. package/dist/runtime-options-CN4g8VJE.js +0 -275
  916. package/dist/runtime-schema-Jjsquqvq.js +0 -27780
  917. package/dist/scan-C8jthyct.js +0 -523
  918. package/dist/scan-DEd4q4aN.js +0 -2
  919. package/dist/secrets-cli-iBL8NBAq.js +0 -2101
  920. package/dist/security-cli-CzYBRa4n.js +0 -486
  921. package/dist/selection-BOalx5uy.js +0 -2
  922. package/dist/selection-BZFVKdFz.js +0 -7736
  923. package/dist/send-C4mKkkWW.js +0 -156
  924. package/dist/send-C_36w3lb.js +0 -102
  925. package/dist/send.runtime-vJT5TzDj.js +0 -2
  926. package/dist/server-BTmWWR_9.js +0 -13
  927. package/dist/server-CBQVrNY0.js +0 -77
  928. package/dist/server-context-Dq2VA7I4.js +0 -2
  929. package/dist/server-context-O_WUP8OV.js +0 -847
  930. package/dist/server-node-events-B8PLg0eQ.js +0 -481
  931. package/dist/server-plugin-bootstrap-ChIvqLMw.js +0 -2
  932. package/dist/server-plugin-bootstrap-Rbxd2g6a.js +0 -11333
  933. package/dist/server-restart-sentinel-BCd6AqrQ.js +0 -697
  934. package/dist/server.impl-CEOWGKN5.js +0 -12735
  935. package/dist/service-ChLFBF7Q.js +0 -120
  936. package/dist/service-XI052xiT.js +0 -2
  937. package/dist/service-audit-BFxGImYM.js +0 -254
  938. package/dist/service-audit-cqHyD5gB.js +0 -2
  939. package/dist/session-kill-http-ChCr5CQ3.js +0 -110
  940. package/dist/session-reset-service-DsYIDYdh.js +0 -471
  941. package/dist/session-route-rem428sJ.js +0 -93
  942. package/dist/session-status.runtime-CFmGwoCm.js +0 -2
  943. package/dist/session-subagent-reactivation.runtime-DLkpnURC.js +0 -2
  944. package/dist/session-tab-registry-DWyzIez5.js +0 -491
  945. package/dist/session-visibility-CJocpWP0.js +0 -147
  946. package/dist/sessions-helpers-CMf9gVTZ.js +0 -304
  947. package/dist/sessions-history-http-C5iZkW8x.js +0 -383
  948. package/dist/sessions-patch-2sBcHc9a.js +0 -309
  949. package/dist/sessions-resolve-D1hmolC-.js +0 -174
  950. package/dist/sessions.runtime-DMPmzLE0.js +0 -2
  951. package/dist/setup-D1ekkhVk.js +0 -495
  952. package/dist/setup-api-DilgjV01.js +0 -29
  953. package/dist/setup-core-CFAiauCt.js +0 -171
  954. package/dist/setup-core-RgJfQc-C.js +0 -176
  955. package/dist/setup-surface-C0l9bHY3.js +0 -286
  956. package/dist/setup-surface-CfcFHZln.js +0 -403
  957. package/dist/setup-surface-CyDRXq_1.js +0 -219
  958. package/dist/setup.finalize-JPqhcPh5.js +0 -539
  959. package/dist/setup.gateway-config-Df6Jm4vX.js +0 -250
  960. package/dist/shared-3gP_6DZV.js +0 -76
  961. package/dist/shared-C9fN-WCy.js +0 -198
  962. package/dist/shared-Cz9c46Aj.js +0 -121
  963. package/dist/slash-state-C7nXXf23.js +0 -1911
  964. package/dist/src-LnF0REtW.js +0 -3974
  965. package/dist/startup-context-CxTy7OzN.js +0 -312
  966. package/dist/status-4n9xL7HY.js +0 -2
  967. package/dist/status-B4kD5cCF.js +0 -2
  968. package/dist/status-CARBoVql.js +0 -209
  969. package/dist/status-COK71LP_.js +0 -3
  970. package/dist/status-DxcqzXtf.js +0 -397
  971. package/dist/status-all-r7zyk41X.js +0 -498
  972. package/dist/status-f2DLmjOF.js +0 -190
  973. package/dist/status-json-C_wESP21.js +0 -14
  974. package/dist/status-json-command-C6w9ZAHp.js +0 -84
  975. package/dist/status-runtime-shared-Bq_SUJpC.js +0 -257
  976. package/dist/status-subagents.runtime-03fp3eBE.js +0 -18
  977. package/dist/status-text-CNvz374U.js +0 -237
  978. package/dist/status.gateway-connection.runtime-zjQmormq.js +0 -2
  979. package/dist/status.gather-BhjoJVEs.js +0 -2
  980. package/dist/status.gather-zdywRvBb.js +0 -292
  981. package/dist/status.runtime-DB31ZRnK.js +0 -2
  982. package/dist/status.scan-DjhdTOWP.js +0 -65
  983. package/dist/status.scan-overview-BYxTBsOw.js +0 -379
  984. package/dist/status.scan.fast-json-Dz0JcORG.js +0 -2
  985. package/dist/status.scan.fast-json-FroEIZ9d.js +0 -132
  986. package/dist/status.summary-3UVa_X5B.js +0 -214
  987. package/dist/status.summary-BGzGsyZB.js +0 -2
  988. package/dist/subagent-announce-BvRFZ87M.js +0 -351
  989. package/dist/subagent-announce-delivery-QvuOM46S.js +0 -726
  990. package/dist/subagent-announce-output-fOtzkOiP.js +0 -364
  991. package/dist/subagent-control-DwnzmG6T.js +0 -506
  992. package/dist/subagent-followup.runtime-DxR1eP_x.js +0 -68
  993. package/dist/subagent-orphan-recovery-CiBbDxX3.js +0 -305
  994. package/dist/subagent-registry-D5FrT_tn.js +0 -3
  995. package/dist/subagent-registry-DbPekx8d.js +0 -1753
  996. package/dist/subagent-spawn-W-bTaU9l.js +0 -1005
  997. package/dist/system-cli-B443uVHG.js +0 -59
  998. package/dist/targets-3Lcm8HoL.js +0 -67
  999. package/dist/task-executor-B5e9_Hmt.js +0 -360
  1000. package/dist/task-owner-access-CIEssJ78.js +0 -74
  1001. package/dist/task-registry-M1bHrFvP.js +0 -2366
  1002. package/dist/task-registry-delivery-runtime-CiHBC7sJ.js +0 -3
  1003. package/dist/task-registry-delivery-runtime-M4O7ffju.js +0 -2
  1004. package/dist/task-registry.maintenance-BURvMU0r.js +0 -2
  1005. package/dist/task-registry.maintenance-CgrLoBqi.js +0 -416
  1006. package/dist/testing-CRy_6-wt.js +0 -575
  1007. package/dist/text-report-B3-4Dvfb.js +0 -587
  1008. package/dist/tool-resolution-C52y5Qtc.js +0 -90
  1009. package/dist/tools-effective-inventory-WlIlqM7G.js +0 -152
  1010. package/dist/tools-invoke-http-B04Rc8tG.js +0 -206
  1011. package/dist/trash-BV5Gcx3a.js +0 -24
  1012. package/dist/tui-cli-DMZ07qRK.js +0 -4575
  1013. package/dist/update-cli-DeBm7b4g.js +0 -1759
  1014. package/dist/upgrade-C9ke1SCc.js +0 -1226
  1015. package/dist/video-generation-task-status-I3jvAjUL.js +0 -163
  1016. package/dist/wait-for-idle-before-flush-BZhF2ejZ.js +0 -5986
  1017. package/dist/wizard-models-CMKeAr_M.js +0 -161
@@ -0,0 +1,3341 @@
1
+ import { i as formatErrorMessage } from "./errors-Jbvi20TW.js";
2
+ import { a as normalizeLowercaseStringOrEmpty, c as normalizeOptionalString$4 } from "./string-coerce-C1IzJjqi.js";
3
+ import { m as resolveUserPath } from "./utils-DaGfogP-.js";
4
+ import { a as loadConfig, y as writeConfigFile } from "./io-DNd8mbjy.js";
5
+ import { a as getImageMetadata, l as resizeToJpeg, r as buildImageResizeSideGrid, t as IMAGE_REDUCE_QUALITY_STEPS } from "./image-ops-DXau4yMa.js";
6
+ import { a as ensureMediaDir, l as saveMediaBuffer } from "./store-Fge4lGwN.js";
7
+ import { n as deriveDefaultBrowserCdpPortRange } from "./port-defaults-grVhyUfZ.js";
8
+ import "./text-runtime-B4pK_Jpd.js";
9
+ import "./browser-setup-tools-VC7rGCyn.js";
10
+ import "./control-auth-BlqH4IVx.js";
11
+ import { c as DEFAULT_AI_SNAPSHOT_MAX_CHARS, f as DEFAULT_BROWSER_SCREENSHOT_TIMEOUT_MS, i as resolveExistingPathsWithinRoot, n as DEFAULT_TRACE_DIR, o as resolveWritablePathWithinRoot, r as DEFAULT_UPLOAD_DIR, s as DEFAULT_AI_SNAPSHOT_EFFICIENT_MAX_CHARS, t as DEFAULT_DOWNLOAD_DIR } from "./paths-6HUizwZc.js";
12
+ import { n as resolveProfile } from "./config-D9XEe0Cd.js";
13
+ import { d as parseBrowserHttpUrl, n as assertCdpEndpointAllowed } from "./cdp.helpers-D-cKiAJ-.js";
14
+ import { a as BrowserProfileUnavailableError, c as BrowserTabNotFoundError, d as toBrowserErrorResponse, i as BrowserProfileNotFoundError, l as BrowserTargetAmbiguousError, n as BrowserConflictError, r as BrowserError, s as BrowserResourceExhaustedError, u as BrowserValidationError } from "./errors-DyjYUGQK.js";
15
+ import { a as shouldUsePlaywrightForAriaSnapshot, i as resolveDefaultSnapshotFormat, n as getPwAiModule$1, o as shouldUsePlaywrightForScreenshot, r as getBrowserProfileCapabilities, t as resolveTargetIdFromTabs } from "./target-id-0_iIeBLg.js";
16
+ import { _ as assertBrowserNavigationAllowed, a as resolveGenesisUserDataDir, h as resolveBrowserNavigationProxyMode, m as snapshotAria, u as captureScreenshot, x as withBrowserNavigationPolicy, y as assertBrowserNavigationResultAllowed } from "./chrome-CgLTMlX5.js";
17
+ import { n as normalizeString } from "./record-shared-DxabpimX.js";
18
+ import { E as uploadChromeMcpFile, T as takeChromeMcpSnapshot, _ as navigateChromeMcpPage, a as closeChromeMcpTab, c as evaluateChromeMcpScript, f as getChromeMcpPid, l as fillChromeMcpElement, m as hoverChromeMcpElement, n as clickChromeMcpCoords, o as dragChromeMcpElement, r as clickChromeMcpElement, u as fillChromeMcpForm, w as takeChromeMcpScreenshot, x as resizeChromeMcpPage, y as pressChromeMcpKey } from "./chrome-mcp-BjiLGZmi.js";
19
+ import { a as CONTENT_ROLES, d as ACT_MAX_CLICK_DELAY_MS, f as ACT_MAX_WAIT_TIME_MS, g as matchBrowserUrlPattern, l as normalizeBrowserFormField, o as INTERACTIVE_ROLES, p as normalizeActBoundedNonNegativeMs, r as getRoleSnapshotStats, s as STRUCTURAL_ROLES } from "./pw-role-snapshot-D65unjCD.js";
20
+ import { r as resolveBrowserExecutableForPlatform } from "./chrome.executables-CoZsfQLC.js";
21
+ import { t as movePathToTrash } from "./trash-F4tvjrKS.js";
22
+ import fs from "node:fs";
23
+ import path from "node:path";
24
+ import fs$1 from "node:fs/promises";
25
+ import crypto from "node:crypto";
26
+ //#region extensions/browser/src/browser/routes/utils.ts
27
+ function normalizeOptionalString$3(value) {
28
+ return value.trim() || void 0;
29
+ }
30
+ function asyncBrowserRoute(handler) {
31
+ return (req, res) => handler(req, res);
32
+ }
33
+ /**
34
+ * Extract profile name from query string or body and get profile context.
35
+ * Query string takes precedence over body for consistency with GET routes.
36
+ */
37
+ function getProfileContext(req, ctx) {
38
+ let profileName;
39
+ if (typeof req.query.profile === "string") profileName = normalizeOptionalString$3(req.query.profile);
40
+ if (!profileName && req.body && typeof req.body === "object") {
41
+ const body = req.body;
42
+ if (typeof body.profile === "string") profileName = normalizeOptionalString$3(body.profile);
43
+ }
44
+ try {
45
+ return ctx.forProfile(profileName);
46
+ } catch (err) {
47
+ return {
48
+ error: String(err),
49
+ status: 404
50
+ };
51
+ }
52
+ }
53
+ function jsonError(res, status, message) {
54
+ res.status(status).json({ error: message });
55
+ }
56
+ function toStringOrEmpty(value) {
57
+ if (typeof value === "string" || typeof value === "number" || typeof value === "boolean") return normalizeOptionalString$3(String(value)) ?? "";
58
+ return "";
59
+ }
60
+ function toNumber(value) {
61
+ if (typeof value === "number" && Number.isFinite(value)) return value;
62
+ const normalized = typeof value === "string" ? normalizeOptionalString$3(value) : void 0;
63
+ if (normalized) {
64
+ const parsed = Number(normalized);
65
+ return Number.isFinite(parsed) ? parsed : void 0;
66
+ }
67
+ }
68
+ function toBoolean(value) {
69
+ if (typeof value === "boolean") return value;
70
+ if (typeof value !== "string" && typeof value !== "number") return;
71
+ const normalized = String(value).trim().toLowerCase();
72
+ if (normalized === "true" || normalized === "1" || normalized === "yes") return true;
73
+ if (normalized === "false" || normalized === "0" || normalized === "no") return false;
74
+ }
75
+ function toStringArray(value) {
76
+ if (!Array.isArray(value)) return;
77
+ const strings = value.map((v) => toStringOrEmpty(v)).filter(Boolean);
78
+ return strings.length ? strings : void 0;
79
+ }
80
+ //#endregion
81
+ //#region extensions/browser/src/browser/routes/agent.shared.ts
82
+ function normalizeOptionalString$2(value) {
83
+ return typeof value === "string" ? value.trim() || void 0 : void 0;
84
+ }
85
+ const SELECTOR_UNSUPPORTED_MESSAGE = [
86
+ "Error: 'selector' is not supported. Use 'ref' from snapshot instead.",
87
+ "",
88
+ "Example workflow:",
89
+ "1. snapshot action to get page state with refs",
90
+ "2. act with ref: \"e123\" to interact with element",
91
+ "",
92
+ "This is more reliable for modern SPAs."
93
+ ].join("\n");
94
+ function readBody(req) {
95
+ const body = req.body;
96
+ if (!body || typeof body !== "object" || Array.isArray(body)) return {};
97
+ return body;
98
+ }
99
+ function resolveTargetIdFromBody(body) {
100
+ return (normalizeOptionalString$2(body.targetId) ?? "") || void 0;
101
+ }
102
+ function resolveTargetIdFromQuery(query) {
103
+ return (normalizeOptionalString$2(query.targetId) ?? "") || void 0;
104
+ }
105
+ function handleRouteError(ctx, res, err) {
106
+ const mapped = ctx.mapTabError(err);
107
+ if (mapped) return jsonError(res, mapped.status, mapped.message);
108
+ const browserMapped = toBrowserErrorResponse(err);
109
+ if (browserMapped) return jsonError(res, browserMapped.status, browserMapped.message);
110
+ jsonError(res, 500, String(err));
111
+ }
112
+ function resolveProfileContext(req, res, ctx) {
113
+ const profileCtx = getProfileContext(req, ctx);
114
+ if ("error" in profileCtx) {
115
+ jsonError(res, profileCtx.status, profileCtx.error);
116
+ return null;
117
+ }
118
+ return profileCtx;
119
+ }
120
+ async function getPwAiModule() {
121
+ return await getPwAiModule$1({ mode: "soft" });
122
+ }
123
+ async function requirePwAi(res, feature) {
124
+ const mod = await getPwAiModule();
125
+ if (mod) return mod;
126
+ jsonError(res, 501, [
127
+ `Playwright is not available in this gateway build; '${feature}' is unsupported.`,
128
+ "Repair the bundled browser plugin runtime dependencies so playwright-core is installed, then restart the gateway. In Docker, also install Chromium with the bundled playwright-core CLI.",
129
+ "Docs: /tools/browser#playwright-requirement"
130
+ ].join("\n"));
131
+ return null;
132
+ }
133
+ async function withRouteTabContext(params) {
134
+ const profileCtx = resolveProfileContext(params.req, params.res, params.ctx);
135
+ if (!profileCtx) return;
136
+ try {
137
+ const tab = await profileCtx.ensureTabAvailable(params.targetId);
138
+ return await params.run({
139
+ profileCtx,
140
+ tab,
141
+ cdpUrl: profileCtx.profile.cdpUrl
142
+ });
143
+ } catch (err) {
144
+ handleRouteError(params.ctx, params.res, err);
145
+ return;
146
+ }
147
+ }
148
+ async function withPlaywrightRouteContext(params) {
149
+ return await withRouteTabContext({
150
+ req: params.req,
151
+ res: params.res,
152
+ ctx: params.ctx,
153
+ targetId: params.targetId,
154
+ run: async ({ profileCtx, tab, cdpUrl }) => {
155
+ const pw = await requirePwAi(params.res, params.feature);
156
+ if (!pw) return;
157
+ return await params.run({
158
+ profileCtx,
159
+ tab,
160
+ cdpUrl,
161
+ pw
162
+ });
163
+ }
164
+ });
165
+ }
166
+ //#endregion
167
+ //#region extensions/browser/src/browser/routes/existing-session-limits.ts
168
+ const EXISTING_SESSION_LIMITS = {
169
+ act: {
170
+ clickSelector: "existing-session click does not support selector targeting yet; use ref.",
171
+ clickButtonOrModifiers: "existing-session click currently supports left-click only (no button overrides/modifiers).",
172
+ typeSelector: "existing-session type does not support selector targeting yet; use ref.",
173
+ typeSlowly: "existing-session type does not support slowly=true; use fill/press instead.",
174
+ typeTimeout: "existing-session type does not support timeoutMs overrides.",
175
+ pressDelay: "existing-session press does not support delayMs.",
176
+ hoverSelector: "existing-session hover does not support selector targeting yet; use ref.",
177
+ hoverTimeout: "existing-session hover does not support timeoutMs overrides.",
178
+ scrollSelector: "existing-session scrollIntoView does not support selector targeting yet; use ref.",
179
+ scrollTimeout: "existing-session scrollIntoView does not support timeoutMs overrides.",
180
+ dragSelector: "existing-session drag does not support selector targeting yet; use startRef/endRef.",
181
+ dragTimeout: "existing-session drag does not support timeoutMs overrides.",
182
+ selectSelector: "existing-session select does not support selector targeting yet; use ref.",
183
+ selectSingleValue: "existing-session select currently supports a single value only.",
184
+ selectTimeout: "existing-session select does not support timeoutMs overrides.",
185
+ fillTimeout: "existing-session fill does not support timeoutMs overrides.",
186
+ waitNetworkIdle: "existing-session wait does not support loadState=networkidle yet.",
187
+ evaluateTimeout: "existing-session evaluate does not support timeoutMs overrides.",
188
+ batch: "existing-session batch is not supported yet; send actions individually."
189
+ },
190
+ hooks: {
191
+ uploadElement: "existing-session file uploads do not support element selectors; use ref/inputRef.",
192
+ uploadSingleFile: "existing-session file uploads currently support one file at a time.",
193
+ uploadRefRequired: "existing-session file uploads require ref or inputRef.",
194
+ dialogTimeout: "existing-session dialog handling does not support timeoutMs."
195
+ },
196
+ download: {
197
+ waitUnsupported: "download waiting is not supported for existing-session profiles yet.",
198
+ downloadUnsupported: "downloads are not supported for existing-session profiles yet."
199
+ },
200
+ snapshot: {
201
+ pdfUnsupported: "pdf is not supported for existing-session profiles yet; use screenshot/snapshot instead.",
202
+ screenshotElement: "element screenshots are not supported for existing-session profiles; use ref from snapshot.",
203
+ snapshotSelector: "selector/frame snapshots are not supported for existing-session profiles; snapshot the whole page and use refs."
204
+ },
205
+ responseBody: "response body is not supported for existing-session profiles yet."
206
+ };
207
+ //#endregion
208
+ //#region extensions/browser/src/browser/routes/output-paths.ts
209
+ async function ensureOutputRootDir(rootDir) {
210
+ await fs$1.mkdir(rootDir, { recursive: true });
211
+ }
212
+ async function resolveWritableOutputPathOrRespond(params) {
213
+ if (params.ensureRootDir) await ensureOutputRootDir(params.rootDir);
214
+ const pathResult = await resolveWritablePathWithinRoot({
215
+ rootDir: params.rootDir,
216
+ requestedPath: params.requestedPath,
217
+ scopeLabel: params.scopeLabel,
218
+ defaultFileName: params.defaultFileName
219
+ });
220
+ if (!pathResult.ok) {
221
+ params.res.status(400).json({ error: pathResult.error });
222
+ return null;
223
+ }
224
+ return pathResult.path;
225
+ }
226
+ //#endregion
227
+ //#region extensions/browser/src/browser/routes/agent.act.download.ts
228
+ function buildDownloadRequestBase(cdpUrl, targetId, timeoutMs) {
229
+ return {
230
+ cdpUrl,
231
+ targetId,
232
+ timeoutMs: timeoutMs ?? void 0
233
+ };
234
+ }
235
+ function registerBrowserAgentActDownloadRoutes(app, ctx) {
236
+ app.post("/wait/download", asyncBrowserRoute(async (req, res) => {
237
+ const body = readBody(req);
238
+ const targetId = resolveTargetIdFromBody(body);
239
+ const out = toStringOrEmpty(body.path) || "";
240
+ const timeoutMs = toNumber(body.timeoutMs);
241
+ await withRouteTabContext({
242
+ req,
243
+ res,
244
+ ctx,
245
+ targetId,
246
+ run: async ({ profileCtx, cdpUrl, tab }) => {
247
+ if (getBrowserProfileCapabilities(profileCtx.profile).usesChromeMcp) return jsonError(res, 501, EXISTING_SESSION_LIMITS.download.waitUnsupported);
248
+ const pw = await requirePwAi(res, "wait for download");
249
+ if (!pw) return;
250
+ await ensureOutputRootDir(DEFAULT_DOWNLOAD_DIR);
251
+ let downloadPath;
252
+ if (out.trim()) {
253
+ const resolvedDownloadPath = await resolveWritableOutputPathOrRespond({
254
+ res,
255
+ rootDir: DEFAULT_DOWNLOAD_DIR,
256
+ requestedPath: out,
257
+ scopeLabel: "downloads directory"
258
+ });
259
+ if (!resolvedDownloadPath) return;
260
+ downloadPath = resolvedDownloadPath;
261
+ }
262
+ const requestBase = buildDownloadRequestBase(cdpUrl, tab.targetId, timeoutMs);
263
+ const result = await pw.waitForDownloadViaPlaywright({
264
+ ...requestBase,
265
+ path: downloadPath
266
+ });
267
+ res.json({
268
+ ok: true,
269
+ targetId: tab.targetId,
270
+ download: result
271
+ });
272
+ }
273
+ });
274
+ }));
275
+ app.post("/download", asyncBrowserRoute(async (req, res) => {
276
+ const body = readBody(req);
277
+ const targetId = resolveTargetIdFromBody(body);
278
+ const ref = toStringOrEmpty(body.ref);
279
+ const out = toStringOrEmpty(body.path);
280
+ const timeoutMs = toNumber(body.timeoutMs);
281
+ if (!ref) return jsonError(res, 400, "ref is required");
282
+ if (!out) return jsonError(res, 400, "path is required");
283
+ await withRouteTabContext({
284
+ req,
285
+ res,
286
+ ctx,
287
+ targetId,
288
+ run: async ({ profileCtx, cdpUrl, tab }) => {
289
+ if (getBrowserProfileCapabilities(profileCtx.profile).usesChromeMcp) return jsonError(res, 501, EXISTING_SESSION_LIMITS.download.downloadUnsupported);
290
+ const pw = await requirePwAi(res, "download");
291
+ if (!pw) return;
292
+ await ensureOutputRootDir(DEFAULT_DOWNLOAD_DIR);
293
+ const downloadPath = await resolveWritableOutputPathOrRespond({
294
+ res,
295
+ rootDir: DEFAULT_DOWNLOAD_DIR,
296
+ requestedPath: out,
297
+ scopeLabel: "downloads directory"
298
+ });
299
+ if (!downloadPath) return;
300
+ const requestBase = buildDownloadRequestBase(cdpUrl, tab.targetId, timeoutMs);
301
+ const result = await pw.downloadViaPlaywright({
302
+ ...requestBase,
303
+ ref,
304
+ path: downloadPath
305
+ });
306
+ res.json({
307
+ ok: true,
308
+ targetId: tab.targetId,
309
+ download: result
310
+ });
311
+ }
312
+ });
313
+ }));
314
+ }
315
+ //#endregion
316
+ //#region extensions/browser/src/browser/routes/agent.act.errors.ts
317
+ const ACT_ERROR_CODES = {
318
+ kindRequired: "ACT_KIND_REQUIRED",
319
+ invalidRequest: "ACT_INVALID_REQUEST",
320
+ selectorUnsupported: "ACT_SELECTOR_UNSUPPORTED",
321
+ evaluateDisabled: "ACT_EVALUATE_DISABLED",
322
+ unsupportedForExistingSession: "ACT_EXISTING_SESSION_UNSUPPORTED",
323
+ targetIdMismatch: "ACT_TARGET_ID_MISMATCH"
324
+ };
325
+ function jsonActError(res, status, code, message) {
326
+ res.status(status).json({
327
+ error: message,
328
+ code
329
+ });
330
+ }
331
+ function browserEvaluateDisabledMessage(action) {
332
+ return [action === "wait" ? "wait --fn is disabled by config (browser.evaluateEnabled=false)." : "act:evaluate is disabled by config (browser.evaluateEnabled=false).", "Docs: /gateway/configuration#browser-genesis-managed-browser"].join("\n");
333
+ }
334
+ //#endregion
335
+ //#region extensions/browser/src/browser/routes/agent.act.hooks.ts
336
+ function registerBrowserAgentActHookRoutes(app, ctx) {
337
+ app.post("/hooks/file-chooser", asyncBrowserRoute(async (req, res) => {
338
+ const body = readBody(req);
339
+ const targetId = resolveTargetIdFromBody(body);
340
+ const ref = toStringOrEmpty(body.ref) || void 0;
341
+ const inputRef = toStringOrEmpty(body.inputRef) || void 0;
342
+ const element = toStringOrEmpty(body.element) || void 0;
343
+ const paths = toStringArray(body.paths) ?? [];
344
+ const timeoutMs = toNumber(body.timeoutMs);
345
+ if (!paths.length) return jsonError(res, 400, "paths are required");
346
+ await withRouteTabContext({
347
+ req,
348
+ res,
349
+ ctx,
350
+ targetId,
351
+ run: async ({ profileCtx, cdpUrl, tab }) => {
352
+ const uploadPathsResult = await resolveExistingPathsWithinRoot({
353
+ rootDir: DEFAULT_UPLOAD_DIR,
354
+ requestedPaths: paths,
355
+ scopeLabel: `uploads directory (${DEFAULT_UPLOAD_DIR})`
356
+ });
357
+ if (!uploadPathsResult.ok) {
358
+ res.status(400).json({ error: uploadPathsResult.error });
359
+ return;
360
+ }
361
+ const resolvedPaths = uploadPathsResult.paths;
362
+ if (getBrowserProfileCapabilities(profileCtx.profile).usesChromeMcp) {
363
+ if (element) return jsonError(res, 501, EXISTING_SESSION_LIMITS.hooks.uploadElement);
364
+ if (resolvedPaths.length !== 1) return jsonError(res, 501, EXISTING_SESSION_LIMITS.hooks.uploadSingleFile);
365
+ const uid = inputRef || ref;
366
+ if (!uid) return jsonError(res, 501, EXISTING_SESSION_LIMITS.hooks.uploadRefRequired);
367
+ await uploadChromeMcpFile({
368
+ profileName: profileCtx.profile.name,
369
+ userDataDir: profileCtx.profile.userDataDir,
370
+ targetId: tab.targetId,
371
+ uid,
372
+ filePath: resolvedPaths[0] ?? ""
373
+ });
374
+ return res.json({ ok: true });
375
+ }
376
+ const pw = await requirePwAi(res, "file chooser hook");
377
+ if (!pw) return;
378
+ if (inputRef || element) {
379
+ if (ref) return jsonError(res, 400, "ref cannot be combined with inputRef/element");
380
+ await pw.setInputFilesViaPlaywright({
381
+ cdpUrl,
382
+ targetId: tab.targetId,
383
+ inputRef,
384
+ element,
385
+ paths: resolvedPaths
386
+ });
387
+ } else {
388
+ await pw.armFileUploadViaPlaywright({
389
+ cdpUrl,
390
+ targetId: tab.targetId,
391
+ paths: resolvedPaths,
392
+ timeoutMs: timeoutMs ?? void 0
393
+ });
394
+ if (ref) await pw.clickViaPlaywright({
395
+ cdpUrl,
396
+ targetId: tab.targetId,
397
+ ssrfPolicy: ctx.state().resolved.ssrfPolicy,
398
+ ref
399
+ });
400
+ }
401
+ res.json({ ok: true });
402
+ }
403
+ });
404
+ }));
405
+ app.post("/hooks/dialog", asyncBrowserRoute(async (req, res) => {
406
+ const body = readBody(req);
407
+ const targetId = resolveTargetIdFromBody(body);
408
+ const accept = toBoolean(body.accept);
409
+ const promptText = toStringOrEmpty(body.promptText) || void 0;
410
+ const timeoutMs = toNumber(body.timeoutMs);
411
+ if (accept === void 0) return jsonError(res, 400, "accept is required");
412
+ await withRouteTabContext({
413
+ req,
414
+ res,
415
+ ctx,
416
+ targetId,
417
+ run: async ({ profileCtx, cdpUrl, tab }) => {
418
+ if (getBrowserProfileCapabilities(profileCtx.profile).usesChromeMcp) {
419
+ if (timeoutMs) return jsonError(res, 501, EXISTING_SESSION_LIMITS.hooks.dialogTimeout);
420
+ await evaluateChromeMcpScript({
421
+ profileName: profileCtx.profile.name,
422
+ userDataDir: profileCtx.profile.userDataDir,
423
+ targetId: tab.targetId,
424
+ fn: `() => {
425
+ const state = (window.__genesisDialogHook ??= {});
426
+ if (!state.originals) {
427
+ state.originals = {
428
+ alert: window.alert.bind(window),
429
+ confirm: window.confirm.bind(window),
430
+ prompt: window.prompt.bind(window),
431
+ };
432
+ }
433
+ const originals = state.originals;
434
+ const restore = () => {
435
+ window.alert = originals.alert;
436
+ window.confirm = originals.confirm;
437
+ window.prompt = originals.prompt;
438
+ delete window.__genesisDialogHook;
439
+ };
440
+ window.alert = (...args) => {
441
+ try {
442
+ return undefined;
443
+ } finally {
444
+ restore();
445
+ }
446
+ };
447
+ window.confirm = (...args) => {
448
+ try {
449
+ return ${accept ? "true" : "false"};
450
+ } finally {
451
+ restore();
452
+ }
453
+ };
454
+ window.prompt = (...args) => {
455
+ try {
456
+ return ${accept ? JSON.stringify(promptText ?? "") : "null"};
457
+ } finally {
458
+ restore();
459
+ }
460
+ };
461
+ return true;
462
+ }`
463
+ });
464
+ return res.json({ ok: true });
465
+ }
466
+ const pw = await requirePwAi(res, "dialog hook");
467
+ if (!pw) return;
468
+ await pw.armDialogViaPlaywright({
469
+ cdpUrl,
470
+ targetId: tab.targetId,
471
+ accept,
472
+ promptText,
473
+ timeoutMs: timeoutMs ?? void 0
474
+ });
475
+ res.json({ ok: true });
476
+ }
477
+ });
478
+ }));
479
+ }
480
+ //#endregion
481
+ //#region extensions/browser/src/browser/routes/agent.act.shared.ts
482
+ const ACT_KINDS = [
483
+ "batch",
484
+ "click",
485
+ "clickCoords",
486
+ "close",
487
+ "drag",
488
+ "evaluate",
489
+ "fill",
490
+ "hover",
491
+ "scrollIntoView",
492
+ "press",
493
+ "resize",
494
+ "select",
495
+ "type",
496
+ "wait"
497
+ ];
498
+ function isActKind(value) {
499
+ if (typeof value !== "string") return false;
500
+ return ACT_KINDS.includes(value);
501
+ }
502
+ const ALLOWED_CLICK_MODIFIERS = new Set([
503
+ "Alt",
504
+ "Control",
505
+ "ControlOrMeta",
506
+ "Meta",
507
+ "Shift"
508
+ ]);
509
+ function parseClickButton(raw) {
510
+ if (raw === "left" || raw === "right" || raw === "middle") return raw;
511
+ }
512
+ function parseClickModifiers(raw) {
513
+ if (raw.filter((m) => !ALLOWED_CLICK_MODIFIERS.has(m)).length) return { error: "modifiers must be Alt|Control|ControlOrMeta|Meta|Shift" };
514
+ return { modifiers: raw.length ? raw : void 0 };
515
+ }
516
+ //#endregion
517
+ //#region extensions/browser/src/browser/routes/agent.act.normalize.ts
518
+ function normalizeActKind(raw) {
519
+ const kind = toStringOrEmpty(raw);
520
+ if (!isActKind(kind)) throw new Error("kind is required");
521
+ return kind;
522
+ }
523
+ function countBatchActions(actions) {
524
+ let count = 0;
525
+ for (const action of actions) {
526
+ count += 1;
527
+ if (action.kind === "batch") count += countBatchActions(action.actions);
528
+ }
529
+ return count;
530
+ }
531
+ function validateBatchTargetIds(actions, targetId) {
532
+ for (const action of actions) {
533
+ if (action.targetId && action.targetId !== targetId) return "batched action targetId must match request targetId";
534
+ if (action.kind === "batch") {
535
+ const nestedError = validateBatchTargetIds(action.actions, targetId);
536
+ if (nestedError) return nestedError;
537
+ }
538
+ }
539
+ return null;
540
+ }
541
+ function normalizeFields(rawFields) {
542
+ return (Array.isArray(rawFields) ? rawFields : []).map((field) => {
543
+ if (!field || typeof field !== "object") return null;
544
+ return normalizeBrowserFormField(field);
545
+ }).filter((field) => field !== null);
546
+ }
547
+ function normalizeBatchAction(value) {
548
+ if (!value || typeof value !== "object" || Array.isArray(value)) throw new Error("batch actions must be objects");
549
+ return normalizeActRequest(value, { source: "batch" });
550
+ }
551
+ function normalizeActRequest(body, options) {
552
+ const source = options?.source ?? "request";
553
+ const kind = normalizeActKind(body.kind);
554
+ switch (kind) {
555
+ case "click": {
556
+ const ref = toStringOrEmpty(body.ref) || void 0;
557
+ const selector = toStringOrEmpty(body.selector) || void 0;
558
+ if (!ref && !selector) throw new Error("click requires ref or selector");
559
+ const buttonRaw = toStringOrEmpty(body.button);
560
+ const button = buttonRaw ? parseClickButton(buttonRaw) : void 0;
561
+ if (buttonRaw && !button) throw new Error("click button must be left|right|middle");
562
+ const parsedModifiers = parseClickModifiers(toStringArray(body.modifiers) ?? []);
563
+ if (parsedModifiers.error) throw new Error(parsedModifiers.error);
564
+ const doubleClick = toBoolean(body.doubleClick);
565
+ const delayMs = normalizeActBoundedNonNegativeMs(toNumber(body.delayMs), "click delayMs", ACT_MAX_CLICK_DELAY_MS);
566
+ const timeoutMs = toNumber(body.timeoutMs);
567
+ const targetId = toStringOrEmpty(body.targetId) || void 0;
568
+ return {
569
+ kind,
570
+ ...ref ? { ref } : {},
571
+ ...selector ? { selector } : {},
572
+ ...targetId ? { targetId } : {},
573
+ ...doubleClick !== void 0 ? { doubleClick } : {},
574
+ ...button ? { button } : {},
575
+ ...parsedModifiers.modifiers ? { modifiers: parsedModifiers.modifiers } : {},
576
+ ...delayMs !== void 0 ? { delayMs } : {},
577
+ ...timeoutMs !== void 0 ? { timeoutMs } : {}
578
+ };
579
+ }
580
+ case "clickCoords": {
581
+ const x = toNumber(body.x);
582
+ const y = toNumber(body.y);
583
+ if (x === void 0 || y === void 0 || x < 0 || y < 0) throw new Error("clickCoords requires non-negative x and y");
584
+ const buttonRaw = toStringOrEmpty(body.button);
585
+ const button = buttonRaw ? parseClickButton(buttonRaw) : void 0;
586
+ if (buttonRaw && !button) throw new Error("clickCoords button must be left|right|middle");
587
+ const doubleClick = toBoolean(body.doubleClick);
588
+ const delayMs = normalizeActBoundedNonNegativeMs(toNumber(body.delayMs), "clickCoords delayMs", ACT_MAX_CLICK_DELAY_MS);
589
+ const timeoutMs = toNumber(body.timeoutMs);
590
+ const targetId = toStringOrEmpty(body.targetId) || void 0;
591
+ return {
592
+ kind,
593
+ x,
594
+ y,
595
+ ...targetId ? { targetId } : {},
596
+ ...doubleClick !== void 0 ? { doubleClick } : {},
597
+ ...button ? { button } : {},
598
+ ...delayMs !== void 0 ? { delayMs } : {},
599
+ ...timeoutMs !== void 0 ? { timeoutMs } : {}
600
+ };
601
+ }
602
+ case "type": {
603
+ const ref = toStringOrEmpty(body.ref) || void 0;
604
+ const selector = toStringOrEmpty(body.selector) || void 0;
605
+ const text = body.text;
606
+ if (!ref && !selector) throw new Error("type requires ref or selector");
607
+ if (typeof text !== "string") throw new Error("type requires text");
608
+ const targetId = toStringOrEmpty(body.targetId) || void 0;
609
+ const submit = toBoolean(body.submit);
610
+ const slowly = toBoolean(body.slowly);
611
+ const timeoutMs = toNumber(body.timeoutMs);
612
+ return {
613
+ kind,
614
+ ...ref ? { ref } : {},
615
+ ...selector ? { selector } : {},
616
+ text,
617
+ ...targetId ? { targetId } : {},
618
+ ...submit !== void 0 ? { submit } : {},
619
+ ...slowly !== void 0 ? { slowly } : {},
620
+ ...timeoutMs !== void 0 ? { timeoutMs } : {}
621
+ };
622
+ }
623
+ case "press": {
624
+ const key = toStringOrEmpty(body.key);
625
+ if (!key) throw new Error("press requires key");
626
+ const targetId = toStringOrEmpty(body.targetId) || void 0;
627
+ const delayMs = toNumber(body.delayMs);
628
+ return {
629
+ kind,
630
+ key,
631
+ ...targetId ? { targetId } : {},
632
+ ...delayMs !== void 0 ? { delayMs } : {}
633
+ };
634
+ }
635
+ case "hover":
636
+ case "scrollIntoView": {
637
+ const ref = toStringOrEmpty(body.ref) || void 0;
638
+ const selector = toStringOrEmpty(body.selector) || void 0;
639
+ if (!ref && !selector) throw new Error(`${kind} requires ref or selector`);
640
+ const targetId = toStringOrEmpty(body.targetId) || void 0;
641
+ const timeoutMs = toNumber(body.timeoutMs);
642
+ return {
643
+ kind,
644
+ ...ref ? { ref } : {},
645
+ ...selector ? { selector } : {},
646
+ ...targetId ? { targetId } : {},
647
+ ...timeoutMs !== void 0 ? { timeoutMs } : {}
648
+ };
649
+ }
650
+ case "drag": {
651
+ const startRef = toStringOrEmpty(body.startRef) || void 0;
652
+ const startSelector = toStringOrEmpty(body.startSelector) || void 0;
653
+ const endRef = toStringOrEmpty(body.endRef) || void 0;
654
+ const endSelector = toStringOrEmpty(body.endSelector) || void 0;
655
+ if (!startRef && !startSelector) throw new Error("drag requires startRef or startSelector");
656
+ if (!endRef && !endSelector) throw new Error("drag requires endRef or endSelector");
657
+ const targetId = toStringOrEmpty(body.targetId) || void 0;
658
+ const timeoutMs = toNumber(body.timeoutMs);
659
+ return {
660
+ kind,
661
+ ...startRef ? { startRef } : {},
662
+ ...startSelector ? { startSelector } : {},
663
+ ...endRef ? { endRef } : {},
664
+ ...endSelector ? { endSelector } : {},
665
+ ...targetId ? { targetId } : {},
666
+ ...timeoutMs !== void 0 ? { timeoutMs } : {}
667
+ };
668
+ }
669
+ case "select": {
670
+ const ref = toStringOrEmpty(body.ref) || void 0;
671
+ const selector = toStringOrEmpty(body.selector) || void 0;
672
+ const values = toStringArray(body.values);
673
+ if (!ref && !selector || !values?.length) throw new Error("select requires ref/selector and values");
674
+ const targetId = toStringOrEmpty(body.targetId) || void 0;
675
+ const timeoutMs = toNumber(body.timeoutMs);
676
+ return {
677
+ kind,
678
+ ...ref ? { ref } : {},
679
+ ...selector ? { selector } : {},
680
+ values,
681
+ ...targetId ? { targetId } : {},
682
+ ...timeoutMs !== void 0 ? { timeoutMs } : {}
683
+ };
684
+ }
685
+ case "fill": {
686
+ const fields = normalizeFields(body.fields);
687
+ if (!fields.length) throw new Error("fill requires fields");
688
+ const targetId = toStringOrEmpty(body.targetId) || void 0;
689
+ const timeoutMs = toNumber(body.timeoutMs);
690
+ return {
691
+ kind,
692
+ fields,
693
+ ...targetId ? { targetId } : {},
694
+ ...timeoutMs !== void 0 ? { timeoutMs } : {}
695
+ };
696
+ }
697
+ case "resize": {
698
+ const width = toNumber(body.width);
699
+ const height = toNumber(body.height);
700
+ if (width === void 0 || height === void 0 || width <= 0 || height <= 0) throw new Error("resize requires positive width and height");
701
+ const targetId = toStringOrEmpty(body.targetId) || void 0;
702
+ return {
703
+ kind,
704
+ width,
705
+ height,
706
+ ...targetId ? { targetId } : {}
707
+ };
708
+ }
709
+ case "wait": {
710
+ const loadStateRaw = toStringOrEmpty(body.loadState);
711
+ const loadState = loadStateRaw === "load" || loadStateRaw === "domcontentloaded" || loadStateRaw === "networkidle" ? loadStateRaw : void 0;
712
+ const timeMs = normalizeActBoundedNonNegativeMs(toNumber(body.timeMs), "wait timeMs", ACT_MAX_WAIT_TIME_MS);
713
+ const text = toStringOrEmpty(body.text) || void 0;
714
+ const textGone = toStringOrEmpty(body.textGone) || void 0;
715
+ const selector = toStringOrEmpty(body.selector) || void 0;
716
+ const url = toStringOrEmpty(body.url) || void 0;
717
+ const fn = toStringOrEmpty(body.fn) || void 0;
718
+ if (timeMs === void 0 && !text && !textGone && !selector && !url && !loadState && !fn) throw new Error("wait requires at least one of: timeMs, text, textGone, selector, url, loadState, fn");
719
+ const targetId = toStringOrEmpty(body.targetId) || void 0;
720
+ const timeoutMs = toNumber(body.timeoutMs);
721
+ return {
722
+ kind,
723
+ ...timeMs !== void 0 ? { timeMs } : {},
724
+ ...text ? { text } : {},
725
+ ...textGone ? { textGone } : {},
726
+ ...selector ? { selector } : {},
727
+ ...url ? { url } : {},
728
+ ...loadState ? { loadState } : {},
729
+ ...fn ? { fn } : {},
730
+ ...targetId ? { targetId } : {},
731
+ ...timeoutMs !== void 0 ? { timeoutMs } : {}
732
+ };
733
+ }
734
+ case "evaluate": {
735
+ const fn = toStringOrEmpty(body.fn);
736
+ if (!fn) throw new Error("evaluate requires fn");
737
+ const ref = toStringOrEmpty(body.ref) || void 0;
738
+ const targetId = toStringOrEmpty(body.targetId) || void 0;
739
+ const timeoutMs = toNumber(body.timeoutMs);
740
+ return {
741
+ kind,
742
+ fn,
743
+ ...ref ? { ref } : {},
744
+ ...targetId ? { targetId } : {},
745
+ ...timeoutMs !== void 0 ? { timeoutMs } : {}
746
+ };
747
+ }
748
+ case "close": {
749
+ const targetId = toStringOrEmpty(body.targetId) || void 0;
750
+ return {
751
+ kind,
752
+ ...targetId ? { targetId } : {}
753
+ };
754
+ }
755
+ case "batch": {
756
+ const actions = Array.isArray(body.actions) ? body.actions.map(normalizeBatchAction) : [];
757
+ if (!actions.length) throw new Error(source === "batch" ? "batch requires actions" : "actions are required");
758
+ if (countBatchActions(actions) > 100) throw new Error(`batch exceeds maximum of 100 actions`);
759
+ const targetId = toStringOrEmpty(body.targetId) || void 0;
760
+ const stopOnError = toBoolean(body.stopOnError);
761
+ return {
762
+ kind,
763
+ actions,
764
+ ...targetId ? { targetId } : {},
765
+ ...stopOnError !== void 0 ? { stopOnError } : {}
766
+ };
767
+ }
768
+ }
769
+ throw new Error("Unsupported browser act kind");
770
+ }
771
+ //#endregion
772
+ //#region extensions/browser/src/browser/routes/agent.act.ts
773
+ function sleep(ms) {
774
+ return new Promise((resolve) => setTimeout(resolve, ms));
775
+ }
776
+ const EXISTING_SESSION_INTERACTION_NAVIGATION_RECHECK_DELAYS_MS = [
777
+ 0,
778
+ 250,
779
+ 500
780
+ ];
781
+ async function readExistingSessionLocationHref(params) {
782
+ const currentUrl = await evaluateChromeMcpScript({
783
+ profileName: params.profileName,
784
+ userDataDir: params.userDataDir,
785
+ targetId: params.targetId,
786
+ fn: "() => window.location.href"
787
+ });
788
+ if (typeof currentUrl !== "string") throw new Error("Location probe returned a non-string result");
789
+ const normalizedUrl = currentUrl.trim();
790
+ if (!normalizedUrl) throw new Error("Location probe returned an empty URL");
791
+ return normalizedUrl;
792
+ }
793
+ async function assertExistingSessionPostInteractionNavigationAllowed(params) {
794
+ const ssrfPolicyOpts = withBrowserNavigationPolicy(params.ssrfPolicy);
795
+ if (!ssrfPolicyOpts.ssrfPolicy) return;
796
+ const listTabs = params.listTabs;
797
+ const initialTabTargetIds = params.initialTabTargetIds;
798
+ const assertNewTabsAllowed = async () => {
799
+ const tabs = await listTabs();
800
+ for (const tab of tabs) {
801
+ if (initialTabTargetIds.has(tab.targetId)) continue;
802
+ await assertBrowserNavigationResultAllowed({
803
+ url: tab.url,
804
+ ...ssrfPolicyOpts
805
+ });
806
+ }
807
+ };
808
+ let lastObservedUrl;
809
+ let sawStableAllowedUrl = false;
810
+ for (const delayMs of EXISTING_SESSION_INTERACTION_NAVIGATION_RECHECK_DELAYS_MS) {
811
+ if (delayMs > 0) await sleep(delayMs);
812
+ let currentUrl;
813
+ try {
814
+ currentUrl = await readExistingSessionLocationHref(params);
815
+ } catch {
816
+ sawStableAllowedUrl = false;
817
+ continue;
818
+ }
819
+ await assertBrowserNavigationResultAllowed({
820
+ url: currentUrl,
821
+ ...ssrfPolicyOpts
822
+ });
823
+ if (currentUrl === lastObservedUrl) sawStableAllowedUrl = true;
824
+ else sawStableAllowedUrl = false;
825
+ lastObservedUrl = currentUrl;
826
+ }
827
+ if (sawStableAllowedUrl) {
828
+ await assertNewTabsAllowed();
829
+ return;
830
+ }
831
+ if (lastObservedUrl) {
832
+ const lastDelay = EXISTING_SESSION_INTERACTION_NAVIGATION_RECHECK_DELAYS_MS[EXISTING_SESSION_INTERACTION_NAVIGATION_RECHECK_DELAYS_MS.length - 1];
833
+ await sleep(lastDelay);
834
+ try {
835
+ const followUpUrl = await readExistingSessionLocationHref(params);
836
+ await assertBrowserNavigationResultAllowed({
837
+ url: followUpUrl,
838
+ ...ssrfPolicyOpts
839
+ });
840
+ if (followUpUrl === lastObservedUrl) {
841
+ await assertNewTabsAllowed();
842
+ return;
843
+ }
844
+ } catch {}
845
+ }
846
+ throw new Error("Unable to verify stable post-interaction navigation");
847
+ }
848
+ async function runExistingSessionActionWithNavigationGuard(params) {
849
+ let actionError;
850
+ let result;
851
+ try {
852
+ result = await params.execute();
853
+ } catch (error) {
854
+ actionError = error;
855
+ }
856
+ if (params.guard) await assertExistingSessionPostInteractionNavigationAllowed(params.guard);
857
+ if (actionError) throw actionError;
858
+ return result;
859
+ }
860
+ function buildExistingSessionWaitPredicate(params) {
861
+ const checks = [];
862
+ if (params.text) checks.push(`Boolean(document.body?.innerText?.includes(${JSON.stringify(params.text)}))`);
863
+ if (params.textGone) checks.push(`!document.body?.innerText?.includes(${JSON.stringify(params.textGone)})`);
864
+ if (params.selector) checks.push(`Boolean(document.querySelector(${JSON.stringify(params.selector)}))`);
865
+ if (params.loadState === "domcontentloaded") checks.push(`document.readyState === "interactive" || document.readyState === "complete"`);
866
+ else if (params.loadState === "load") checks.push(`document.readyState === "complete"`);
867
+ if (params.fn) checks.push(`Boolean(await (${params.fn})())`);
868
+ if (checks.length === 0) return null;
869
+ return checks.length === 1 ? checks[0] : checks.map((check) => `(${check})`).join(" && ");
870
+ }
871
+ async function waitForExistingSessionCondition(params) {
872
+ if (params.timeMs && params.timeMs > 0) await sleep(params.timeMs);
873
+ const predicate = buildExistingSessionWaitPredicate(params);
874
+ if (!predicate && !params.url) return;
875
+ const timeoutMs = Math.max(250, params.timeoutMs ?? 1e4);
876
+ const deadline = Date.now() + timeoutMs;
877
+ while (Date.now() < deadline) {
878
+ let ready = true;
879
+ if (predicate) ready = Boolean(await evaluateChromeMcpScript({
880
+ profileName: params.profileName,
881
+ userDataDir: params.userDataDir,
882
+ targetId: params.targetId,
883
+ fn: `async () => ${predicate}`
884
+ }));
885
+ if (ready && params.url) {
886
+ const currentUrl = await evaluateChromeMcpScript({
887
+ profileName: params.profileName,
888
+ userDataDir: params.userDataDir,
889
+ targetId: params.targetId,
890
+ fn: "() => window.location.href"
891
+ });
892
+ ready = typeof currentUrl === "string" && matchBrowserUrlPattern(params.url, currentUrl);
893
+ }
894
+ if (ready) return;
895
+ await sleep(250);
896
+ }
897
+ throw new Error("Timed out waiting for condition");
898
+ }
899
+ const SELECTOR_ALLOWED_KINDS = new Set([
900
+ "batch",
901
+ "click",
902
+ "drag",
903
+ "hover",
904
+ "scrollIntoView",
905
+ "select",
906
+ "type",
907
+ "wait"
908
+ ]);
909
+ function getExistingSessionUnsupportedMessage(action) {
910
+ switch (action.kind) {
911
+ case "click":
912
+ if (action.selector) return EXISTING_SESSION_LIMITS.act.clickSelector;
913
+ if (action.button && action.button !== "left" || Array.isArray(action.modifiers) && action.modifiers.length > 0) return EXISTING_SESSION_LIMITS.act.clickButtonOrModifiers;
914
+ return null;
915
+ case "clickCoords": return null;
916
+ case "type":
917
+ if (action.selector) return EXISTING_SESSION_LIMITS.act.typeSelector;
918
+ if (action.slowly) return EXISTING_SESSION_LIMITS.act.typeSlowly;
919
+ return action.timeoutMs ? EXISTING_SESSION_LIMITS.act.typeTimeout : null;
920
+ case "press": return action.delayMs ? EXISTING_SESSION_LIMITS.act.pressDelay : null;
921
+ case "hover":
922
+ if (action.selector) return EXISTING_SESSION_LIMITS.act.hoverSelector;
923
+ return action.timeoutMs ? EXISTING_SESSION_LIMITS.act.hoverTimeout : null;
924
+ case "scrollIntoView":
925
+ if (action.selector) return EXISTING_SESSION_LIMITS.act.scrollSelector;
926
+ return action.timeoutMs ? EXISTING_SESSION_LIMITS.act.scrollTimeout : null;
927
+ case "drag":
928
+ if (action.startSelector || action.endSelector) return EXISTING_SESSION_LIMITS.act.dragSelector;
929
+ return action.timeoutMs ? EXISTING_SESSION_LIMITS.act.dragTimeout : null;
930
+ case "select":
931
+ if (action.selector) return EXISTING_SESSION_LIMITS.act.selectSelector;
932
+ if (action.values.length !== 1) return EXISTING_SESSION_LIMITS.act.selectSingleValue;
933
+ return action.timeoutMs ? EXISTING_SESSION_LIMITS.act.selectTimeout : null;
934
+ case "fill": return action.timeoutMs ? EXISTING_SESSION_LIMITS.act.fillTimeout : null;
935
+ case "wait": return action.loadState === "networkidle" ? EXISTING_SESSION_LIMITS.act.waitNetworkIdle : null;
936
+ case "evaluate": return action.timeoutMs !== void 0 ? EXISTING_SESSION_LIMITS.act.evaluateTimeout : null;
937
+ case "batch": return EXISTING_SESSION_LIMITS.act.batch;
938
+ case "resize":
939
+ case "close": return null;
940
+ }
941
+ throw new Error("Unsupported browser act kind");
942
+ }
943
+ function registerBrowserAgentActRoutes(app, ctx) {
944
+ app.post("/act", asyncBrowserRoute(async (req, res) => {
945
+ const body = readBody(req);
946
+ const kindRaw = toStringOrEmpty(body.kind);
947
+ if (!isActKind(kindRaw)) return jsonActError(res, 400, ACT_ERROR_CODES.kindRequired, "kind is required");
948
+ const kind = kindRaw;
949
+ let action;
950
+ try {
951
+ action = normalizeActRequest(body);
952
+ } catch (err) {
953
+ return jsonActError(res, 400, ACT_ERROR_CODES.invalidRequest, formatErrorMessage(err));
954
+ }
955
+ const targetId = resolveTargetIdFromBody(body);
956
+ if (Object.hasOwn(body, "selector") && !SELECTOR_ALLOWED_KINDS.has(kind)) return jsonActError(res, 400, ACT_ERROR_CODES.selectorUnsupported, SELECTOR_UNSUPPORTED_MESSAGE);
957
+ const earlyFn = action.kind === "wait" || action.kind === "evaluate" ? action.fn : "";
958
+ if ((action.kind === "evaluate" || action.kind === "wait" && earlyFn) && !ctx.state().resolved.evaluateEnabled) return jsonActError(res, 403, ACT_ERROR_CODES.evaluateDisabled, browserEvaluateDisabledMessage(action.kind === "evaluate" ? "evaluate" : "wait"));
959
+ await withRouteTabContext({
960
+ req,
961
+ res,
962
+ ctx,
963
+ targetId,
964
+ run: async ({ profileCtx, cdpUrl, tab }) => {
965
+ const evaluateEnabled = ctx.state().resolved.evaluateEnabled;
966
+ const ssrfPolicy = ctx.state().resolved.ssrfPolicy;
967
+ if (action.targetId && action.targetId !== tab.targetId) return jsonActError(res, 403, ACT_ERROR_CODES.targetIdMismatch, "action targetId must match request targetId");
968
+ const isExistingSession = getBrowserProfileCapabilities(profileCtx.profile).usesChromeMcp;
969
+ const profileName = profileCtx.profile.name;
970
+ if (isExistingSession) {
971
+ const initialTabTargetIds = withBrowserNavigationPolicy(ssrfPolicy).ssrfPolicy ? new Set((await profileCtx.listTabs()).map((currentTab) => currentTab.targetId)) : /* @__PURE__ */ new Set();
972
+ const existingSessionNavigationGuard = {
973
+ profileName,
974
+ userDataDir: profileCtx.profile.userDataDir,
975
+ targetId: tab.targetId,
976
+ ssrfPolicy,
977
+ listTabs: () => profileCtx.listTabs(),
978
+ initialTabTargetIds
979
+ };
980
+ const unsupportedMessage = getExistingSessionUnsupportedMessage(action);
981
+ if (unsupportedMessage) return jsonActError(res, 501, ACT_ERROR_CODES.unsupportedForExistingSession, unsupportedMessage);
982
+ switch (action.kind) {
983
+ case "click":
984
+ await runExistingSessionActionWithNavigationGuard({
985
+ execute: () => clickChromeMcpElement({
986
+ profileName,
987
+ userDataDir: profileCtx.profile.userDataDir,
988
+ targetId: tab.targetId,
989
+ uid: action.ref,
990
+ doubleClick: action.doubleClick ?? false,
991
+ timeoutMs: action.timeoutMs,
992
+ signal: req.signal
993
+ }),
994
+ guard: existingSessionNavigationGuard
995
+ });
996
+ return res.json({
997
+ ok: true,
998
+ targetId: tab.targetId,
999
+ url: tab.url
1000
+ });
1001
+ case "clickCoords":
1002
+ await runExistingSessionActionWithNavigationGuard({
1003
+ execute: () => clickChromeMcpCoords({
1004
+ profileName,
1005
+ userDataDir: profileCtx.profile.userDataDir,
1006
+ targetId: tab.targetId,
1007
+ x: action.x,
1008
+ y: action.y,
1009
+ doubleClick: action.doubleClick ?? false,
1010
+ button: action.button,
1011
+ delayMs: action.delayMs
1012
+ }),
1013
+ guard: existingSessionNavigationGuard
1014
+ });
1015
+ return res.json({
1016
+ ok: true,
1017
+ targetId: tab.targetId,
1018
+ url: tab.url
1019
+ });
1020
+ case "type":
1021
+ await runExistingSessionActionWithNavigationGuard({
1022
+ execute: async () => {
1023
+ await fillChromeMcpElement({
1024
+ profileName,
1025
+ userDataDir: profileCtx.profile.userDataDir,
1026
+ targetId: tab.targetId,
1027
+ uid: action.ref,
1028
+ value: action.text
1029
+ });
1030
+ if (action.submit) await pressChromeMcpKey({
1031
+ profileName,
1032
+ userDataDir: profileCtx.profile.userDataDir,
1033
+ targetId: tab.targetId,
1034
+ key: "Enter"
1035
+ });
1036
+ },
1037
+ guard: existingSessionNavigationGuard
1038
+ });
1039
+ return res.json({
1040
+ ok: true,
1041
+ targetId: tab.targetId
1042
+ });
1043
+ case "press":
1044
+ await runExistingSessionActionWithNavigationGuard({
1045
+ execute: () => pressChromeMcpKey({
1046
+ profileName,
1047
+ userDataDir: profileCtx.profile.userDataDir,
1048
+ targetId: tab.targetId,
1049
+ key: action.key
1050
+ }),
1051
+ guard: existingSessionNavigationGuard
1052
+ });
1053
+ return res.json({
1054
+ ok: true,
1055
+ targetId: tab.targetId
1056
+ });
1057
+ case "hover":
1058
+ await runExistingSessionActionWithNavigationGuard({
1059
+ execute: () => hoverChromeMcpElement({
1060
+ profileName,
1061
+ userDataDir: profileCtx.profile.userDataDir,
1062
+ targetId: tab.targetId,
1063
+ uid: action.ref
1064
+ }),
1065
+ guard: existingSessionNavigationGuard
1066
+ });
1067
+ return res.json({
1068
+ ok: true,
1069
+ targetId: tab.targetId
1070
+ });
1071
+ case "scrollIntoView":
1072
+ await runExistingSessionActionWithNavigationGuard({
1073
+ execute: () => evaluateChromeMcpScript({
1074
+ profileName,
1075
+ userDataDir: profileCtx.profile.userDataDir,
1076
+ targetId: tab.targetId,
1077
+ fn: `(el) => { el.scrollIntoView({ block: "center", inline: "center" }); return true; }`,
1078
+ args: [action.ref]
1079
+ }),
1080
+ guard: existingSessionNavigationGuard
1081
+ });
1082
+ return res.json({
1083
+ ok: true,
1084
+ targetId: tab.targetId
1085
+ });
1086
+ case "drag":
1087
+ await runExistingSessionActionWithNavigationGuard({
1088
+ execute: () => dragChromeMcpElement({
1089
+ profileName,
1090
+ userDataDir: profileCtx.profile.userDataDir,
1091
+ targetId: tab.targetId,
1092
+ fromUid: action.startRef,
1093
+ toUid: action.endRef
1094
+ }),
1095
+ guard: existingSessionNavigationGuard
1096
+ });
1097
+ return res.json({
1098
+ ok: true,
1099
+ targetId: tab.targetId
1100
+ });
1101
+ case "select":
1102
+ await runExistingSessionActionWithNavigationGuard({
1103
+ execute: () => fillChromeMcpElement({
1104
+ profileName,
1105
+ userDataDir: profileCtx.profile.userDataDir,
1106
+ targetId: tab.targetId,
1107
+ uid: action.ref,
1108
+ value: action.values[0] ?? ""
1109
+ }),
1110
+ guard: existingSessionNavigationGuard
1111
+ });
1112
+ return res.json({
1113
+ ok: true,
1114
+ targetId: tab.targetId
1115
+ });
1116
+ case "fill":
1117
+ await runExistingSessionActionWithNavigationGuard({
1118
+ execute: () => fillChromeMcpForm({
1119
+ profileName,
1120
+ userDataDir: profileCtx.profile.userDataDir,
1121
+ targetId: tab.targetId,
1122
+ elements: action.fields.map((field) => ({
1123
+ uid: field.ref,
1124
+ value: String(field.value ?? "")
1125
+ }))
1126
+ }),
1127
+ guard: existingSessionNavigationGuard
1128
+ });
1129
+ return res.json({
1130
+ ok: true,
1131
+ targetId: tab.targetId
1132
+ });
1133
+ case "resize":
1134
+ await resizeChromeMcpPage({
1135
+ profileName,
1136
+ userDataDir: profileCtx.profile.userDataDir,
1137
+ targetId: tab.targetId,
1138
+ width: action.width,
1139
+ height: action.height
1140
+ });
1141
+ return res.json({
1142
+ ok: true,
1143
+ targetId: tab.targetId,
1144
+ url: tab.url
1145
+ });
1146
+ case "wait":
1147
+ await waitForExistingSessionCondition({
1148
+ profileName,
1149
+ userDataDir: profileCtx.profile.userDataDir,
1150
+ targetId: tab.targetId,
1151
+ timeMs: action.timeMs,
1152
+ text: action.text,
1153
+ textGone: action.textGone,
1154
+ selector: action.selector,
1155
+ url: action.url,
1156
+ loadState: action.loadState,
1157
+ fn: action.fn,
1158
+ timeoutMs: action.timeoutMs
1159
+ });
1160
+ return res.json({
1161
+ ok: true,
1162
+ targetId: tab.targetId
1163
+ });
1164
+ case "evaluate": {
1165
+ const result = await runExistingSessionActionWithNavigationGuard({
1166
+ execute: () => evaluateChromeMcpScript({
1167
+ profileName,
1168
+ userDataDir: profileCtx.profile.userDataDir,
1169
+ targetId: tab.targetId,
1170
+ fn: action.fn,
1171
+ args: action.ref ? [action.ref] : void 0
1172
+ }),
1173
+ guard: existingSessionNavigationGuard
1174
+ });
1175
+ return res.json({
1176
+ ok: true,
1177
+ targetId: tab.targetId,
1178
+ url: tab.url,
1179
+ result
1180
+ });
1181
+ }
1182
+ case "close":
1183
+ await closeChromeMcpTab(profileName, tab.targetId, profileCtx.profile.userDataDir);
1184
+ return res.json({
1185
+ ok: true,
1186
+ targetId: tab.targetId
1187
+ });
1188
+ case "batch": return jsonActError(res, 501, ACT_ERROR_CODES.unsupportedForExistingSession, EXISTING_SESSION_LIMITS.act.batch);
1189
+ }
1190
+ }
1191
+ const pw = await requirePwAi(res, `act:${kind}`);
1192
+ if (!pw) return;
1193
+ if (action.kind === "batch") {
1194
+ const targetIdError = validateBatchTargetIds(action.actions, tab.targetId);
1195
+ if (targetIdError) return jsonActError(res, 403, ACT_ERROR_CODES.targetIdMismatch, targetIdError);
1196
+ }
1197
+ const result = await pw.executeActViaPlaywright({
1198
+ cdpUrl,
1199
+ action,
1200
+ targetId: tab.targetId,
1201
+ evaluateEnabled,
1202
+ ssrfPolicy,
1203
+ signal: req.signal
1204
+ });
1205
+ switch (action.kind) {
1206
+ case "batch": return res.json({
1207
+ ok: true,
1208
+ targetId: tab.targetId,
1209
+ results: result.results ?? []
1210
+ });
1211
+ case "evaluate": return res.json({
1212
+ ok: true,
1213
+ targetId: tab.targetId,
1214
+ url: tab.url,
1215
+ result: result.result
1216
+ });
1217
+ case "click":
1218
+ case "clickCoords":
1219
+ case "resize": return res.json({
1220
+ ok: true,
1221
+ targetId: tab.targetId,
1222
+ url: tab.url
1223
+ });
1224
+ default: return res.json({
1225
+ ok: true,
1226
+ targetId: tab.targetId
1227
+ });
1228
+ }
1229
+ }
1230
+ });
1231
+ }));
1232
+ registerBrowserAgentActHookRoutes(app, ctx);
1233
+ registerBrowserAgentActDownloadRoutes(app, ctx);
1234
+ app.post("/response/body", asyncBrowserRoute(async (req, res) => {
1235
+ const body = readBody(req);
1236
+ const targetId = resolveTargetIdFromBody(body);
1237
+ const url = toStringOrEmpty(body.url);
1238
+ const timeoutMs = toNumber(body.timeoutMs);
1239
+ const maxChars = toNumber(body.maxChars);
1240
+ if (!url) return jsonError(res, 400, "url is required");
1241
+ await withRouteTabContext({
1242
+ req,
1243
+ res,
1244
+ ctx,
1245
+ targetId,
1246
+ run: async ({ profileCtx, cdpUrl, tab }) => {
1247
+ if (getBrowserProfileCapabilities(profileCtx.profile).usesChromeMcp) return jsonError(res, 501, EXISTING_SESSION_LIMITS.responseBody);
1248
+ const pw = await requirePwAi(res, "response body");
1249
+ if (!pw) return;
1250
+ const result = await pw.responseBodyViaPlaywright({
1251
+ cdpUrl,
1252
+ targetId: tab.targetId,
1253
+ url,
1254
+ timeoutMs: timeoutMs ?? void 0,
1255
+ maxChars: maxChars ?? void 0
1256
+ });
1257
+ res.json({
1258
+ ok: true,
1259
+ targetId: tab.targetId,
1260
+ response: result
1261
+ });
1262
+ }
1263
+ });
1264
+ }));
1265
+ app.post("/highlight", asyncBrowserRoute(async (req, res) => {
1266
+ const body = readBody(req);
1267
+ const targetId = resolveTargetIdFromBody(body);
1268
+ const ref = toStringOrEmpty(body.ref);
1269
+ if (!ref) return jsonError(res, 400, "ref is required");
1270
+ await withRouteTabContext({
1271
+ req,
1272
+ res,
1273
+ ctx,
1274
+ targetId,
1275
+ run: async ({ profileCtx, cdpUrl, tab }) => {
1276
+ if (getBrowserProfileCapabilities(profileCtx.profile).usesChromeMcp) {
1277
+ await evaluateChromeMcpScript({
1278
+ profileName: profileCtx.profile.name,
1279
+ userDataDir: profileCtx.profile.userDataDir,
1280
+ targetId: tab.targetId,
1281
+ args: [ref],
1282
+ fn: `(el) => {
1283
+ if (!(el instanceof Element)) {
1284
+ return false;
1285
+ }
1286
+ el.scrollIntoView({ block: "center", inline: "center" });
1287
+ const previousOutline = el.style.outline;
1288
+ const previousOffset = el.style.outlineOffset;
1289
+ el.style.outline = "3px solid #FF4500";
1290
+ el.style.outlineOffset = "2px";
1291
+ setTimeout(() => {
1292
+ el.style.outline = previousOutline;
1293
+ el.style.outlineOffset = previousOffset;
1294
+ }, 2000);
1295
+ return true;
1296
+ }`
1297
+ });
1298
+ return res.json({
1299
+ ok: true,
1300
+ targetId: tab.targetId
1301
+ });
1302
+ }
1303
+ const pw = await requirePwAi(res, "highlight");
1304
+ if (!pw) return;
1305
+ await pw.highlightViaPlaywright({
1306
+ cdpUrl,
1307
+ targetId: tab.targetId,
1308
+ ref
1309
+ });
1310
+ res.json({
1311
+ ok: true,
1312
+ targetId: tab.targetId
1313
+ });
1314
+ }
1315
+ });
1316
+ }));
1317
+ }
1318
+ //#endregion
1319
+ //#region extensions/browser/src/browser/routes/agent.debug.ts
1320
+ function registerBrowserAgentDebugRoutes(app, ctx) {
1321
+ app.get("/console", asyncBrowserRoute(async (req, res) => {
1322
+ const targetId = resolveTargetIdFromQuery(req.query);
1323
+ const level = typeof req.query.level === "string" ? req.query.level : "";
1324
+ await withPlaywrightRouteContext({
1325
+ req,
1326
+ res,
1327
+ ctx,
1328
+ targetId,
1329
+ feature: "console messages",
1330
+ run: async ({ cdpUrl, tab, pw }) => {
1331
+ const messages = await pw.getConsoleMessagesViaPlaywright({
1332
+ cdpUrl,
1333
+ targetId: tab.targetId,
1334
+ level: normalizeOptionalString$4(level)
1335
+ });
1336
+ res.json({
1337
+ ok: true,
1338
+ messages,
1339
+ targetId: tab.targetId
1340
+ });
1341
+ }
1342
+ });
1343
+ }));
1344
+ app.get("/errors", asyncBrowserRoute(async (req, res) => {
1345
+ const targetId = resolveTargetIdFromQuery(req.query);
1346
+ const clear = toBoolean(req.query.clear) ?? false;
1347
+ await withPlaywrightRouteContext({
1348
+ req,
1349
+ res,
1350
+ ctx,
1351
+ targetId,
1352
+ feature: "page errors",
1353
+ run: async ({ cdpUrl, tab, pw }) => {
1354
+ const result = await pw.getPageErrorsViaPlaywright({
1355
+ cdpUrl,
1356
+ targetId: tab.targetId,
1357
+ clear
1358
+ });
1359
+ res.json({
1360
+ ok: true,
1361
+ targetId: tab.targetId,
1362
+ ...result
1363
+ });
1364
+ }
1365
+ });
1366
+ }));
1367
+ app.get("/requests", asyncBrowserRoute(async (req, res) => {
1368
+ const targetId = resolveTargetIdFromQuery(req.query);
1369
+ const filter = typeof req.query.filter === "string" ? req.query.filter : "";
1370
+ const clear = toBoolean(req.query.clear) ?? false;
1371
+ await withPlaywrightRouteContext({
1372
+ req,
1373
+ res,
1374
+ ctx,
1375
+ targetId,
1376
+ feature: "network requests",
1377
+ run: async ({ cdpUrl, tab, pw }) => {
1378
+ const result = await pw.getNetworkRequestsViaPlaywright({
1379
+ cdpUrl,
1380
+ targetId: tab.targetId,
1381
+ filter: normalizeOptionalString$4(filter),
1382
+ clear
1383
+ });
1384
+ res.json({
1385
+ ok: true,
1386
+ targetId: tab.targetId,
1387
+ ...result
1388
+ });
1389
+ }
1390
+ });
1391
+ }));
1392
+ app.post("/trace/start", asyncBrowserRoute(async (req, res) => {
1393
+ const body = readBody(req);
1394
+ const targetId = resolveTargetIdFromBody(body);
1395
+ const screenshots = toBoolean(body.screenshots) ?? void 0;
1396
+ const snapshots = toBoolean(body.snapshots) ?? void 0;
1397
+ const sources = toBoolean(body.sources) ?? void 0;
1398
+ await withPlaywrightRouteContext({
1399
+ req,
1400
+ res,
1401
+ ctx,
1402
+ targetId,
1403
+ feature: "trace start",
1404
+ run: async ({ cdpUrl, tab, pw }) => {
1405
+ await pw.traceStartViaPlaywright({
1406
+ cdpUrl,
1407
+ targetId: tab.targetId,
1408
+ screenshots,
1409
+ snapshots,
1410
+ sources
1411
+ });
1412
+ res.json({
1413
+ ok: true,
1414
+ targetId: tab.targetId
1415
+ });
1416
+ }
1417
+ });
1418
+ }));
1419
+ app.post("/trace/stop", asyncBrowserRoute(async (req, res) => {
1420
+ const body = readBody(req);
1421
+ const targetId = resolveTargetIdFromBody(body);
1422
+ const out = toStringOrEmpty(body.path) || "";
1423
+ await withPlaywrightRouteContext({
1424
+ req,
1425
+ res,
1426
+ ctx,
1427
+ targetId,
1428
+ feature: "trace stop",
1429
+ run: async ({ cdpUrl, tab, pw }) => {
1430
+ const tracePath = await resolveWritableOutputPathOrRespond({
1431
+ res,
1432
+ rootDir: DEFAULT_TRACE_DIR,
1433
+ requestedPath: out,
1434
+ scopeLabel: "trace directory",
1435
+ defaultFileName: `browser-trace-${crypto.randomUUID()}.zip`,
1436
+ ensureRootDir: true
1437
+ });
1438
+ if (!tracePath) return;
1439
+ await pw.traceStopViaPlaywright({
1440
+ cdpUrl,
1441
+ targetId: tab.targetId,
1442
+ path: tracePath
1443
+ });
1444
+ res.json({
1445
+ ok: true,
1446
+ targetId: tab.targetId,
1447
+ path: path.resolve(tracePath)
1448
+ });
1449
+ }
1450
+ });
1451
+ }));
1452
+ }
1453
+ //#endregion
1454
+ //#region extensions/browser/src/browser/chrome-mcp.snapshot.ts
1455
+ function normalizeRole(node) {
1456
+ return normalizeLowercaseStringOrEmpty(node.role) || "generic";
1457
+ }
1458
+ function escapeQuoted(value) {
1459
+ return value.replaceAll("\\", "\\\\").replaceAll("\"", "\\\"");
1460
+ }
1461
+ function shouldIncludeNode(params) {
1462
+ if (params.options?.interactive && !INTERACTIVE_ROLES.has(params.role)) return false;
1463
+ if (params.options?.compact && STRUCTURAL_ROLES.has(params.role) && !params.name) return false;
1464
+ return true;
1465
+ }
1466
+ function shouldCreateRef(role, name) {
1467
+ return INTERACTIVE_ROLES.has(role) || CONTENT_ROLES.has(role) && Boolean(name);
1468
+ }
1469
+ function createDuplicateTracker() {
1470
+ return {
1471
+ counts: /* @__PURE__ */ new Map(),
1472
+ keysByRef: /* @__PURE__ */ new Map(),
1473
+ duplicates: /* @__PURE__ */ new Set()
1474
+ };
1475
+ }
1476
+ function registerRef(tracker, ref, role, name) {
1477
+ const key = `${role}:${name ?? ""}`;
1478
+ const count = tracker.counts.get(key) ?? 0;
1479
+ tracker.counts.set(key, count + 1);
1480
+ tracker.keysByRef.set(ref, key);
1481
+ if (count > 0) {
1482
+ tracker.duplicates.add(key);
1483
+ return count;
1484
+ }
1485
+ }
1486
+ function flattenChromeMcpSnapshotToAriaNodes(root, limit = 500) {
1487
+ const boundedLimit = Math.max(1, Math.min(2e3, Math.floor(limit)));
1488
+ const out = [];
1489
+ const visit = (node, depth) => {
1490
+ if (out.length >= boundedLimit) return;
1491
+ const ref = normalizeString(node.id);
1492
+ if (ref) out.push({
1493
+ ref,
1494
+ role: normalizeRole(node),
1495
+ name: normalizeString(node.name) ?? "",
1496
+ value: normalizeString(node.value),
1497
+ description: normalizeString(node.description),
1498
+ depth
1499
+ });
1500
+ for (const child of node.children ?? []) {
1501
+ visit(child, depth + 1);
1502
+ if (out.length >= boundedLimit) return;
1503
+ }
1504
+ };
1505
+ visit(root, 0);
1506
+ return out;
1507
+ }
1508
+ function buildAiSnapshotFromChromeMcpSnapshot(params) {
1509
+ const refs = {};
1510
+ const tracker = createDuplicateTracker();
1511
+ const lines = [];
1512
+ const visit = (node, depth) => {
1513
+ const role = normalizeRole(node);
1514
+ const name = normalizeString(node.name);
1515
+ const value = normalizeString(node.value);
1516
+ const description = normalizeString(node.description);
1517
+ const maxDepth = params.options?.maxDepth;
1518
+ if (maxDepth !== void 0 && depth > maxDepth) return;
1519
+ if (shouldIncludeNode({
1520
+ role,
1521
+ name,
1522
+ options: params.options
1523
+ })) {
1524
+ let line = `${" ".repeat(depth)}- ${role}`;
1525
+ if (name) line += ` "${escapeQuoted(name)}"`;
1526
+ const ref = normalizeString(node.id);
1527
+ if (ref && shouldCreateRef(role, name)) {
1528
+ const nth = registerRef(tracker, ref, role, name);
1529
+ refs[ref] = nth === void 0 ? {
1530
+ role,
1531
+ name
1532
+ } : {
1533
+ role,
1534
+ name,
1535
+ nth
1536
+ };
1537
+ line += ` [ref=${ref}]`;
1538
+ }
1539
+ if (value) line += ` value="${escapeQuoted(value)}"`;
1540
+ if (description) line += ` description="${escapeQuoted(description)}"`;
1541
+ lines.push(line);
1542
+ }
1543
+ for (const child of node.children ?? []) visit(child, depth + 1);
1544
+ };
1545
+ visit(params.root, 0);
1546
+ for (const [ref, data] of Object.entries(refs)) {
1547
+ const key = tracker.keysByRef.get(ref);
1548
+ if (key && !tracker.duplicates.has(key)) delete data.nth;
1549
+ }
1550
+ let snapshot = lines.join("\n");
1551
+ let truncated = false;
1552
+ const maxChars = typeof params.maxChars === "number" && Number.isFinite(params.maxChars) && params.maxChars > 0 ? Math.floor(params.maxChars) : void 0;
1553
+ if (maxChars && snapshot.length > maxChars) {
1554
+ snapshot = `${snapshot.slice(0, maxChars)}\n\n[...TRUNCATED - page too large]`;
1555
+ truncated = true;
1556
+ }
1557
+ const stats = getRoleSnapshotStats(snapshot, refs);
1558
+ return truncated ? {
1559
+ snapshot,
1560
+ truncated,
1561
+ refs,
1562
+ stats
1563
+ } : {
1564
+ snapshot,
1565
+ refs,
1566
+ stats
1567
+ };
1568
+ }
1569
+ //#endregion
1570
+ //#region extensions/browser/src/browser/screenshot.ts
1571
+ const DEFAULT_BROWSER_SCREENSHOT_MAX_SIDE = 2e3;
1572
+ const DEFAULT_BROWSER_SCREENSHOT_MAX_BYTES = 5 * 1024 * 1024;
1573
+ async function normalizeBrowserScreenshot(buffer, opts) {
1574
+ const maxSide = Math.max(1, Math.round(opts?.maxSide ?? 2e3));
1575
+ const maxBytes = Math.max(1, Math.round(opts?.maxBytes ?? 5242880));
1576
+ const meta = await getImageMetadata(buffer);
1577
+ const width = meta?.width ?? 0;
1578
+ const height = meta?.height ?? 0;
1579
+ const maxDim = Math.max(width, height);
1580
+ if (buffer.byteLength <= maxBytes && (maxDim === 0 || width <= maxSide && height <= maxSide)) return { buffer };
1581
+ const sideGrid = buildImageResizeSideGrid(maxSide, maxDim > 0 ? Math.min(maxSide, maxDim) : maxSide);
1582
+ let smallest = null;
1583
+ for (const side of sideGrid) for (const quality of IMAGE_REDUCE_QUALITY_STEPS) {
1584
+ const out = await resizeToJpeg({
1585
+ buffer,
1586
+ maxSide: side,
1587
+ quality,
1588
+ withoutEnlargement: true
1589
+ });
1590
+ if (!smallest || out.byteLength < smallest.size) smallest = {
1591
+ buffer: out,
1592
+ size: out.byteLength
1593
+ };
1594
+ if (out.byteLength <= maxBytes) return {
1595
+ buffer: out,
1596
+ contentType: "image/jpeg"
1597
+ };
1598
+ }
1599
+ const best = smallest?.buffer ?? buffer;
1600
+ throw new Error(`Browser screenshot could not be reduced below ${(maxBytes / (1024 * 1024)).toFixed(0)}MB (got ${(best.byteLength / (1024 * 1024)).toFixed(2)}MB)`);
1601
+ }
1602
+ //#endregion
1603
+ //#region extensions/browser/src/browser/routes/agent.snapshot-target.ts
1604
+ /** Resolve the correct targetId after a navigation that may trigger a renderer swap. */
1605
+ async function resolveTargetIdAfterNavigate(opts) {
1606
+ let currentTargetId = opts.oldTargetId;
1607
+ try {
1608
+ const pickReplacement = (tabs, options) => {
1609
+ if (tabs.some((tab) => tab.targetId === opts.oldTargetId)) return {
1610
+ targetId: opts.oldTargetId,
1611
+ shouldRetry: false
1612
+ };
1613
+ const byUrl = tabs.filter((tab) => tab.url === opts.navigatedUrl);
1614
+ if (byUrl.length === 1) return {
1615
+ targetId: byUrl[0]?.targetId ?? opts.oldTargetId,
1616
+ shouldRetry: false
1617
+ };
1618
+ const uniqueReplacement = byUrl.filter((tab) => tab.targetId !== opts.oldTargetId);
1619
+ if (uniqueReplacement.length === 1) return {
1620
+ targetId: uniqueReplacement[0]?.targetId ?? opts.oldTargetId,
1621
+ shouldRetry: false
1622
+ };
1623
+ if (options?.allowSingleTabFallback && tabs.length === 1) return {
1624
+ targetId: tabs[0]?.targetId ?? opts.oldTargetId,
1625
+ shouldRetry: false
1626
+ };
1627
+ return {
1628
+ targetId: opts.oldTargetId,
1629
+ shouldRetry: true
1630
+ };
1631
+ };
1632
+ const first = pickReplacement(await opts.listTabs());
1633
+ currentTargetId = first.targetId;
1634
+ if (first.shouldRetry) {
1635
+ await new Promise((r) => setTimeout(r, opts.retryDelayMs ?? 800));
1636
+ currentTargetId = pickReplacement(await opts.listTabs(), { allowSingleTabFallback: true }).targetId;
1637
+ }
1638
+ } catch {}
1639
+ return currentTargetId;
1640
+ }
1641
+ //#endregion
1642
+ //#region extensions/browser/src/browser/routes/agent.snapshot.plan.ts
1643
+ function readStringValue$1(value) {
1644
+ return typeof value === "string" ? value : void 0;
1645
+ }
1646
+ function normalizeOptionalString$1(value) {
1647
+ return readStringValue$1(value)?.trim() || void 0;
1648
+ }
1649
+ function resolveSnapshotPlan(params) {
1650
+ const mode = params.query.mode === "efficient" ? "efficient" : void 0;
1651
+ const labels = toBoolean(params.query.labels) ?? void 0;
1652
+ const urls = toBoolean(params.query.urls) ?? void 0;
1653
+ const explicitFormat = params.query.format === "aria" ? "aria" : params.query.format === "ai" ? "ai" : void 0;
1654
+ const format = resolveDefaultSnapshotFormat({
1655
+ profile: params.profile,
1656
+ hasPlaywright: params.hasPlaywright,
1657
+ explicitFormat,
1658
+ mode
1659
+ });
1660
+ const limitRaw = readStringValue$1(params.query.limit);
1661
+ const hasMaxChars = Object.hasOwn(params.query, "maxChars");
1662
+ const maxCharsRaw = readStringValue$1(params.query.maxChars);
1663
+ const limit = Number.isFinite(Number(limitRaw)) ? Number(limitRaw) : void 0;
1664
+ const maxChars = Number.isFinite(Number(maxCharsRaw)) && Number(maxCharsRaw) > 0 ? Math.floor(Number(maxCharsRaw)) : void 0;
1665
+ const resolvedMaxChars = format === "ai" ? hasMaxChars ? maxChars : mode === "efficient" ? DEFAULT_AI_SNAPSHOT_EFFICIENT_MAX_CHARS : DEFAULT_AI_SNAPSHOT_MAX_CHARS : void 0;
1666
+ const interactiveRaw = toBoolean(params.query.interactive);
1667
+ const compactRaw = toBoolean(params.query.compact);
1668
+ const depthRaw = toNumber(params.query.depth);
1669
+ const refsModeRaw = toStringOrEmpty(params.query.refs).trim();
1670
+ const refsMode = refsModeRaw === "aria" ? "aria" : refsModeRaw === "role" ? "role" : void 0;
1671
+ const interactive = interactiveRaw ?? (mode === "efficient" ? true : void 0);
1672
+ const compact = compactRaw ?? (mode === "efficient" ? true : void 0);
1673
+ const depth = depthRaw ?? (mode === "efficient" ? 6 : void 0);
1674
+ const selectorValue = normalizeOptionalString$1(toStringOrEmpty(params.query.selector));
1675
+ const frameSelectorValue = normalizeOptionalString$1(toStringOrEmpty(params.query.frame));
1676
+ return {
1677
+ format,
1678
+ mode,
1679
+ labels,
1680
+ urls,
1681
+ limit,
1682
+ resolvedMaxChars,
1683
+ interactive,
1684
+ compact,
1685
+ depth,
1686
+ refsMode,
1687
+ selectorValue,
1688
+ frameSelectorValue,
1689
+ wantsRoleSnapshot: labels === true || urls === true || mode === "efficient" || interactive === true || compact === true || depth !== void 0 || Boolean(selectorValue) || Boolean(frameSelectorValue)
1690
+ };
1691
+ }
1692
+ //#endregion
1693
+ //#region extensions/browser/src/browser/routes/agent.snapshot.ts
1694
+ const CHROME_MCP_OVERLAY_ATTR = "data-genesis-mcp-overlay";
1695
+ function browserNavigationPolicyForProfile$1(ctx, profileCtx) {
1696
+ return withBrowserNavigationPolicy(ctx.state().resolved.ssrfPolicy, { browserProxyMode: resolveBrowserNavigationProxyMode({
1697
+ resolved: ctx.state().resolved,
1698
+ profile: profileCtx.profile
1699
+ }) });
1700
+ }
1701
+ async function collectChromeMcpSnapshotUrls(params) {
1702
+ const result = await evaluateChromeMcpScript({
1703
+ profileName: params.profileName,
1704
+ userDataDir: params.userDataDir,
1705
+ targetId: params.targetId,
1706
+ fn: `() => {
1707
+ const seen = new Set();
1708
+ const out = [];
1709
+ for (const anchor of Array.from(document.querySelectorAll("a[href]"))) {
1710
+ const href = anchor.href || "";
1711
+ if (!href || seen.has(href)) continue;
1712
+ const text = (anchor.innerText || anchor.textContent || anchor.getAttribute("aria-label") || "")
1713
+ .replace(/\\s+/g, " ")
1714
+ .trim()
1715
+ .slice(0, 120) || href;
1716
+ seen.add(href);
1717
+ out.push({ text, url: href });
1718
+ if (out.length >= 100) break;
1719
+ }
1720
+ return out;
1721
+ }`
1722
+ }).catch(() => []);
1723
+ return Array.isArray(result) ? result.filter((entry) => entry && typeof entry === "object" && typeof entry.text === "string" && typeof entry.url === "string") : [];
1724
+ }
1725
+ function appendSnapshotUrls(snapshot, urls) {
1726
+ if (urls.length === 0) return snapshot;
1727
+ return `${snapshot}\n\nLinks:\n${urls.map((entry, index) => `${index + 1}. ${entry.text} -> ${entry.url}`).join("\n")}`;
1728
+ }
1729
+ async function clearChromeMcpOverlay(params) {
1730
+ await evaluateChromeMcpScript({
1731
+ profileName: params.profileName,
1732
+ userDataDir: params.userDataDir,
1733
+ targetId: params.targetId,
1734
+ fn: `() => {
1735
+ document.querySelectorAll("[${CHROME_MCP_OVERLAY_ATTR}]").forEach((node) => node.remove());
1736
+ return true;
1737
+ }`
1738
+ }).catch(() => {});
1739
+ }
1740
+ async function renderChromeMcpLabels(params) {
1741
+ const refList = JSON.stringify(params.refs);
1742
+ const result = await evaluateChromeMcpScript({
1743
+ profileName: params.profileName,
1744
+ userDataDir: params.userDataDir,
1745
+ targetId: params.targetId,
1746
+ args: params.refs,
1747
+ fn: `(...elements) => {
1748
+ const refs = ${refList};
1749
+ document.querySelectorAll("[${CHROME_MCP_OVERLAY_ATTR}]").forEach((node) => node.remove());
1750
+ const root = document.createElement("div");
1751
+ root.setAttribute("${CHROME_MCP_OVERLAY_ATTR}", "labels");
1752
+ root.style.position = "fixed";
1753
+ root.style.inset = "0";
1754
+ root.style.pointerEvents = "none";
1755
+ root.style.zIndex = "2147483647";
1756
+ let labels = 0;
1757
+ let skipped = 0;
1758
+ elements.forEach((el, index) => {
1759
+ if (!(el instanceof Element)) {
1760
+ skipped += 1;
1761
+ return;
1762
+ }
1763
+ const rect = el.getBoundingClientRect();
1764
+ if (rect.width <= 0 && rect.height <= 0) {
1765
+ skipped += 1;
1766
+ return;
1767
+ }
1768
+ labels += 1;
1769
+ const badge = document.createElement("div");
1770
+ badge.setAttribute("${CHROME_MCP_OVERLAY_ATTR}", "label");
1771
+ badge.textContent = refs[index] || String(labels);
1772
+ badge.style.position = "fixed";
1773
+ badge.style.left = \`\${Math.max(0, rect.left)}px\`;
1774
+ badge.style.top = \`\${Math.max(0, rect.top)}px\`;
1775
+ badge.style.transform = "translateY(-100%)";
1776
+ badge.style.padding = "2px 6px";
1777
+ badge.style.borderRadius = "999px";
1778
+ badge.style.background = "#FF4500";
1779
+ badge.style.color = "#fff";
1780
+ badge.style.font = "600 12px ui-monospace, SFMono-Regular, Menlo, monospace";
1781
+ badge.style.boxShadow = "0 2px 6px rgba(0,0,0,0.35)";
1782
+ badge.style.whiteSpace = "nowrap";
1783
+ root.appendChild(badge);
1784
+ });
1785
+ document.documentElement.appendChild(root);
1786
+ return { labels, skipped };
1787
+ }`
1788
+ });
1789
+ return {
1790
+ labels: result && typeof result === "object" && typeof result.labels === "number" ? result.labels : 0,
1791
+ skipped: result && typeof result === "object" && typeof result.skipped === "number" ? result.skipped : 0
1792
+ };
1793
+ }
1794
+ async function saveNormalizedScreenshotResponse(params) {
1795
+ const normalized = await normalizeBrowserScreenshot(params.buffer, {
1796
+ maxSide: DEFAULT_BROWSER_SCREENSHOT_MAX_SIDE,
1797
+ maxBytes: DEFAULT_BROWSER_SCREENSHOT_MAX_BYTES
1798
+ });
1799
+ await saveBrowserMediaResponse({
1800
+ res: params.res,
1801
+ buffer: normalized.buffer,
1802
+ contentType: normalized.contentType ?? `image/${params.type}`,
1803
+ maxBytes: DEFAULT_BROWSER_SCREENSHOT_MAX_BYTES,
1804
+ targetId: params.targetId,
1805
+ url: params.url,
1806
+ labels: params.labels,
1807
+ labelsCount: params.labelsCount,
1808
+ labelsSkipped: params.labelsSkipped
1809
+ });
1810
+ }
1811
+ async function saveBrowserMediaResponse(params) {
1812
+ await ensureMediaDir();
1813
+ const saved = await saveMediaBuffer(params.buffer, params.contentType, "browser", params.maxBytes);
1814
+ params.res.json({
1815
+ ok: true,
1816
+ path: path.resolve(saved.path),
1817
+ targetId: params.targetId,
1818
+ url: params.url,
1819
+ ...params.labels ? { labels: true } : {},
1820
+ ...typeof params.labelsCount === "number" ? { labelsCount: params.labelsCount } : {},
1821
+ ...typeof params.labelsSkipped === "number" ? { labelsSkipped: params.labelsSkipped } : {}
1822
+ });
1823
+ }
1824
+ function registerBrowserAgentSnapshotRoutes(app, ctx) {
1825
+ app.post("/navigate", asyncBrowserRoute(async (req, res) => {
1826
+ const body = readBody(req);
1827
+ const url = toStringOrEmpty(body.url);
1828
+ const targetId = toStringOrEmpty(body.targetId) || void 0;
1829
+ if (!url) return jsonError(res, 400, "url is required");
1830
+ await withRouteTabContext({
1831
+ req,
1832
+ res,
1833
+ ctx,
1834
+ targetId,
1835
+ run: async ({ profileCtx, tab, cdpUrl }) => {
1836
+ if (getBrowserProfileCapabilities(profileCtx.profile).usesChromeMcp) {
1837
+ const ssrfPolicyOpts = browserNavigationPolicyForProfile$1(ctx, profileCtx);
1838
+ await assertBrowserNavigationAllowed({
1839
+ url,
1840
+ ...ssrfPolicyOpts
1841
+ });
1842
+ const result = await navigateChromeMcpPage({
1843
+ profileName: profileCtx.profile.name,
1844
+ userDataDir: profileCtx.profile.userDataDir,
1845
+ targetId: tab.targetId,
1846
+ url
1847
+ });
1848
+ await assertBrowserNavigationResultAllowed({
1849
+ url: result.url,
1850
+ ...ssrfPolicyOpts
1851
+ });
1852
+ return res.json({
1853
+ ok: true,
1854
+ targetId: tab.targetId,
1855
+ ...result
1856
+ });
1857
+ }
1858
+ const pw = await requirePwAi(res, "navigate");
1859
+ if (!pw) return;
1860
+ const result = await pw.navigateViaPlaywright({
1861
+ cdpUrl,
1862
+ targetId: tab.targetId,
1863
+ url,
1864
+ ...browserNavigationPolicyForProfile$1(ctx, profileCtx)
1865
+ });
1866
+ const currentTargetId = await resolveTargetIdAfterNavigate({
1867
+ oldTargetId: tab.targetId,
1868
+ navigatedUrl: result.url,
1869
+ listTabs: () => profileCtx.listTabs()
1870
+ });
1871
+ res.json({
1872
+ ok: true,
1873
+ targetId: currentTargetId,
1874
+ ...result
1875
+ });
1876
+ }
1877
+ });
1878
+ }));
1879
+ app.post("/pdf", asyncBrowserRoute(async (req, res) => {
1880
+ const targetId = toStringOrEmpty(readBody(req).targetId) || void 0;
1881
+ const profileCtx = resolveProfileContext(req, res, ctx);
1882
+ if (!profileCtx) return;
1883
+ if (getBrowserProfileCapabilities(profileCtx.profile).usesChromeMcp) return jsonError(res, 501, EXISTING_SESSION_LIMITS.snapshot.pdfUnsupported);
1884
+ await withPlaywrightRouteContext({
1885
+ req,
1886
+ res,
1887
+ ctx,
1888
+ targetId,
1889
+ feature: "pdf",
1890
+ run: async ({ cdpUrl, tab, pw }) => {
1891
+ const pdf = await pw.pdfViaPlaywright({
1892
+ cdpUrl,
1893
+ targetId: tab.targetId
1894
+ });
1895
+ await saveBrowserMediaResponse({
1896
+ res,
1897
+ buffer: pdf.buffer,
1898
+ contentType: "application/pdf",
1899
+ maxBytes: pdf.buffer.byteLength,
1900
+ targetId: tab.targetId,
1901
+ url: tab.url
1902
+ });
1903
+ }
1904
+ });
1905
+ }));
1906
+ app.post("/screenshot", asyncBrowserRoute(async (req, res) => {
1907
+ const body = readBody(req);
1908
+ const targetId = toStringOrEmpty(body.targetId) || void 0;
1909
+ const fullPage = toBoolean(body.fullPage) ?? false;
1910
+ const ref = toStringOrEmpty(body.ref) || void 0;
1911
+ const element = toStringOrEmpty(body.element) || void 0;
1912
+ const labels = toBoolean(body.labels) ?? false;
1913
+ const type = body.type === "jpeg" ? "jpeg" : "png";
1914
+ const timeoutMsRaw = toNumber(body.timeoutMs);
1915
+ const timeoutMs = timeoutMsRaw !== void 0 ? Math.max(1, Math.floor(timeoutMsRaw)) : DEFAULT_BROWSER_SCREENSHOT_TIMEOUT_MS;
1916
+ if (fullPage && (ref || element)) return jsonError(res, 400, "fullPage is not supported for element screenshots");
1917
+ await withRouteTabContext({
1918
+ req,
1919
+ res,
1920
+ ctx,
1921
+ targetId,
1922
+ run: async ({ profileCtx, tab, cdpUrl }) => {
1923
+ if (getBrowserProfileCapabilities(profileCtx.profile).usesChromeMcp) {
1924
+ const ssrfPolicyOpts = browserNavigationPolicyForProfile$1(ctx, profileCtx);
1925
+ if (element) return jsonError(res, 400, EXISTING_SESSION_LIMITS.snapshot.screenshotElement);
1926
+ if (ssrfPolicyOpts.ssrfPolicy) await assertBrowserNavigationResultAllowed({
1927
+ url: tab.url,
1928
+ ...ssrfPolicyOpts
1929
+ });
1930
+ if (labels) {
1931
+ const built = buildAiSnapshotFromChromeMcpSnapshot({ root: await takeChromeMcpSnapshot({
1932
+ profileName: profileCtx.profile.name,
1933
+ userDataDir: profileCtx.profile.userDataDir,
1934
+ targetId: tab.targetId
1935
+ }) });
1936
+ const labelResult = await renderChromeMcpLabels({
1937
+ profileName: profileCtx.profile.name,
1938
+ userDataDir: profileCtx.profile.userDataDir,
1939
+ targetId: tab.targetId,
1940
+ refs: Object.keys(built.refs)
1941
+ });
1942
+ try {
1943
+ await saveNormalizedScreenshotResponse({
1944
+ res,
1945
+ buffer: await takeChromeMcpScreenshot({
1946
+ profileName: profileCtx.profile.name,
1947
+ userDataDir: profileCtx.profile.userDataDir,
1948
+ targetId: tab.targetId,
1949
+ fullPage,
1950
+ format: type,
1951
+ timeoutMs
1952
+ }),
1953
+ type,
1954
+ targetId: tab.targetId,
1955
+ url: tab.url,
1956
+ labels: true,
1957
+ labelsCount: labelResult.labels,
1958
+ labelsSkipped: labelResult.skipped
1959
+ });
1960
+ } finally {
1961
+ await clearChromeMcpOverlay({
1962
+ profileName: profileCtx.profile.name,
1963
+ userDataDir: profileCtx.profile.userDataDir,
1964
+ targetId: tab.targetId
1965
+ });
1966
+ }
1967
+ return;
1968
+ }
1969
+ await saveNormalizedScreenshotResponse({
1970
+ res,
1971
+ buffer: await takeChromeMcpScreenshot({
1972
+ profileName: profileCtx.profile.name,
1973
+ userDataDir: profileCtx.profile.userDataDir,
1974
+ targetId: tab.targetId,
1975
+ uid: ref,
1976
+ fullPage,
1977
+ format: type,
1978
+ timeoutMs
1979
+ }),
1980
+ type,
1981
+ targetId: tab.targetId,
1982
+ url: tab.url
1983
+ });
1984
+ return;
1985
+ }
1986
+ let buffer;
1987
+ if (labels || shouldUsePlaywrightForScreenshot({
1988
+ profile: profileCtx.profile,
1989
+ wsUrl: tab.wsUrl,
1990
+ ref,
1991
+ element
1992
+ })) {
1993
+ const pw = await requirePwAi(res, "screenshot");
1994
+ if (!pw) return;
1995
+ if (labels) {
1996
+ const snap = await pw.snapshotRoleViaPlaywright({
1997
+ cdpUrl,
1998
+ targetId: tab.targetId,
1999
+ ssrfPolicy: ctx.state().resolved.ssrfPolicy
2000
+ });
2001
+ const labeled = await pw.screenshotWithLabelsViaPlaywright({
2002
+ cdpUrl,
2003
+ targetId: tab.targetId,
2004
+ refs: snap.refs,
2005
+ type,
2006
+ timeoutMs
2007
+ });
2008
+ await saveNormalizedScreenshotResponse({
2009
+ res,
2010
+ buffer: labeled.buffer,
2011
+ type,
2012
+ targetId: tab.targetId,
2013
+ url: tab.url,
2014
+ labels: true,
2015
+ labelsCount: labeled.labels,
2016
+ labelsSkipped: labeled.skipped
2017
+ });
2018
+ return;
2019
+ }
2020
+ buffer = (await pw.takeScreenshotViaPlaywright({
2021
+ cdpUrl,
2022
+ targetId: tab.targetId,
2023
+ ref,
2024
+ element,
2025
+ fullPage,
2026
+ type,
2027
+ timeoutMs
2028
+ })).buffer;
2029
+ } else buffer = await captureScreenshot({
2030
+ wsUrl: tab.wsUrl ?? "",
2031
+ fullPage,
2032
+ format: type,
2033
+ quality: type === "jpeg" ? 85 : void 0,
2034
+ timeoutMs
2035
+ });
2036
+ await saveNormalizedScreenshotResponse({
2037
+ res,
2038
+ buffer,
2039
+ type,
2040
+ targetId: tab.targetId,
2041
+ url: tab.url
2042
+ });
2043
+ }
2044
+ });
2045
+ }));
2046
+ app.get("/snapshot", asyncBrowserRoute(async (req, res) => {
2047
+ const profileCtx = resolveProfileContext(req, res, ctx);
2048
+ if (!profileCtx) return;
2049
+ const targetId = typeof req.query.targetId === "string" ? req.query.targetId.trim() : "";
2050
+ const hasPlaywright = Boolean(await getPwAiModule());
2051
+ const plan = resolveSnapshotPlan({
2052
+ profile: profileCtx.profile,
2053
+ query: req.query,
2054
+ hasPlaywright
2055
+ });
2056
+ try {
2057
+ const tab = await profileCtx.ensureTabAvailable(targetId || void 0);
2058
+ if ((plan.labels || plan.mode === "efficient") && plan.format === "aria") return jsonError(res, 400, "labels/mode=efficient require format=ai");
2059
+ if (getBrowserProfileCapabilities(profileCtx.profile).usesChromeMcp) {
2060
+ const ssrfPolicyOpts = browserNavigationPolicyForProfile$1(ctx, profileCtx);
2061
+ if (plan.selectorValue || plan.frameSelectorValue) return jsonError(res, 400, EXISTING_SESSION_LIMITS.snapshot.snapshotSelector);
2062
+ if (ssrfPolicyOpts.ssrfPolicy) await assertBrowserNavigationResultAllowed({
2063
+ url: tab.url,
2064
+ ...ssrfPolicyOpts
2065
+ });
2066
+ const snapshot = await takeChromeMcpSnapshot({
2067
+ profileName: profileCtx.profile.name,
2068
+ userDataDir: profileCtx.profile.userDataDir,
2069
+ targetId: tab.targetId
2070
+ });
2071
+ if (plan.format === "aria") return res.json({
2072
+ ok: true,
2073
+ format: "aria",
2074
+ targetId: tab.targetId,
2075
+ url: tab.url,
2076
+ nodes: flattenChromeMcpSnapshotToAriaNodes(snapshot, plan.limit)
2077
+ });
2078
+ const built = buildAiSnapshotFromChromeMcpSnapshot({
2079
+ root: snapshot,
2080
+ options: {
2081
+ interactive: plan.interactive ?? void 0,
2082
+ compact: plan.compact ?? void 0,
2083
+ maxDepth: plan.depth ?? void 0
2084
+ },
2085
+ maxChars: plan.resolvedMaxChars
2086
+ });
2087
+ const builtWithUrls = plan.urls ? {
2088
+ ...built,
2089
+ snapshot: appendSnapshotUrls(built.snapshot, await collectChromeMcpSnapshotUrls({
2090
+ profileName: profileCtx.profile.name,
2091
+ userDataDir: profileCtx.profile.userDataDir,
2092
+ targetId: tab.targetId
2093
+ }))
2094
+ } : built;
2095
+ if (plan.labels) {
2096
+ const refs = Object.keys(builtWithUrls.refs);
2097
+ const labelResult = await renderChromeMcpLabels({
2098
+ profileName: profileCtx.profile.name,
2099
+ userDataDir: profileCtx.profile.userDataDir,
2100
+ targetId: tab.targetId,
2101
+ refs
2102
+ });
2103
+ try {
2104
+ const normalized = await normalizeBrowserScreenshot(await takeChromeMcpScreenshot({
2105
+ profileName: profileCtx.profile.name,
2106
+ userDataDir: profileCtx.profile.userDataDir,
2107
+ targetId: tab.targetId,
2108
+ format: "png"
2109
+ }), {
2110
+ maxSide: DEFAULT_BROWSER_SCREENSHOT_MAX_SIDE,
2111
+ maxBytes: DEFAULT_BROWSER_SCREENSHOT_MAX_BYTES
2112
+ });
2113
+ await ensureMediaDir();
2114
+ const saved = await saveMediaBuffer(normalized.buffer, normalized.contentType ?? "image/png", "browser", DEFAULT_BROWSER_SCREENSHOT_MAX_BYTES);
2115
+ return res.json({
2116
+ ok: true,
2117
+ format: "ai",
2118
+ targetId: tab.targetId,
2119
+ url: tab.url,
2120
+ labels: true,
2121
+ labelsCount: labelResult.labels,
2122
+ labelsSkipped: labelResult.skipped,
2123
+ imagePath: path.resolve(saved.path),
2124
+ imageType: normalized.contentType?.includes("jpeg") ? "jpeg" : "png",
2125
+ ...builtWithUrls
2126
+ });
2127
+ } finally {
2128
+ await clearChromeMcpOverlay({
2129
+ profileName: profileCtx.profile.name,
2130
+ userDataDir: profileCtx.profile.userDataDir,
2131
+ targetId: tab.targetId
2132
+ });
2133
+ }
2134
+ }
2135
+ return res.json({
2136
+ ok: true,
2137
+ format: "ai",
2138
+ targetId: tab.targetId,
2139
+ url: tab.url,
2140
+ ...builtWithUrls
2141
+ });
2142
+ }
2143
+ if (plan.format === "ai") {
2144
+ const pw = await requirePwAi(res, "ai snapshot");
2145
+ if (!pw) return;
2146
+ const roleSnapshotArgs = {
2147
+ cdpUrl: profileCtx.profile.cdpUrl,
2148
+ targetId: tab.targetId,
2149
+ selector: plan.selectorValue,
2150
+ frameSelector: plan.frameSelectorValue,
2151
+ refsMode: plan.refsMode,
2152
+ ssrfPolicy: ctx.state().resolved.ssrfPolicy,
2153
+ urls: plan.urls,
2154
+ options: {
2155
+ interactive: plan.interactive ?? void 0,
2156
+ compact: plan.compact ?? void 0,
2157
+ maxDepth: plan.depth ?? void 0
2158
+ }
2159
+ };
2160
+ const snap = plan.wantsRoleSnapshot ? await pw.snapshotRoleViaPlaywright(roleSnapshotArgs) : await pw.snapshotAiViaPlaywright({
2161
+ cdpUrl: profileCtx.profile.cdpUrl,
2162
+ targetId: tab.targetId,
2163
+ ssrfPolicy: ctx.state().resolved.ssrfPolicy,
2164
+ urls: plan.urls,
2165
+ ...typeof plan.resolvedMaxChars === "number" ? { maxChars: plan.resolvedMaxChars } : {}
2166
+ });
2167
+ if (plan.labels) {
2168
+ const labeled = await pw.screenshotWithLabelsViaPlaywright({
2169
+ cdpUrl: profileCtx.profile.cdpUrl,
2170
+ targetId: tab.targetId,
2171
+ refs: "refs" in snap ? snap.refs : {},
2172
+ type: "png"
2173
+ });
2174
+ const normalized = await normalizeBrowserScreenshot(labeled.buffer, {
2175
+ maxSide: DEFAULT_BROWSER_SCREENSHOT_MAX_SIDE,
2176
+ maxBytes: DEFAULT_BROWSER_SCREENSHOT_MAX_BYTES
2177
+ });
2178
+ await ensureMediaDir();
2179
+ const saved = await saveMediaBuffer(normalized.buffer, normalized.contentType ?? "image/png", "browser", DEFAULT_BROWSER_SCREENSHOT_MAX_BYTES);
2180
+ const imageType = normalized.contentType?.includes("jpeg") ? "jpeg" : "png";
2181
+ return res.json({
2182
+ ok: true,
2183
+ format: plan.format,
2184
+ targetId: tab.targetId,
2185
+ url: tab.url,
2186
+ labels: true,
2187
+ labelsCount: labeled.labels,
2188
+ labelsSkipped: labeled.skipped,
2189
+ imagePath: path.resolve(saved.path),
2190
+ imageType,
2191
+ ...snap
2192
+ });
2193
+ }
2194
+ return res.json({
2195
+ ok: true,
2196
+ format: plan.format,
2197
+ targetId: tab.targetId,
2198
+ url: tab.url,
2199
+ ...snap
2200
+ });
2201
+ }
2202
+ const snap = shouldUsePlaywrightForAriaSnapshot({
2203
+ profile: profileCtx.profile,
2204
+ wsUrl: tab.wsUrl
2205
+ }) ? requirePwAi(res, "aria snapshot").then(async (pw) => {
2206
+ if (!pw) return null;
2207
+ return await pw.snapshotAriaViaPlaywright({
2208
+ cdpUrl: profileCtx.profile.cdpUrl,
2209
+ targetId: tab.targetId,
2210
+ limit: plan.limit,
2211
+ ssrfPolicy: ctx.state().resolved.ssrfPolicy
2212
+ });
2213
+ }) : snapshotAria({
2214
+ wsUrl: tab.wsUrl ?? "",
2215
+ limit: plan.limit
2216
+ });
2217
+ const resolved = await Promise.resolve(snap);
2218
+ if (!resolved) return;
2219
+ return res.json({
2220
+ ok: true,
2221
+ format: plan.format,
2222
+ targetId: tab.targetId,
2223
+ url: tab.url,
2224
+ ...resolved
2225
+ });
2226
+ } catch (err) {
2227
+ handleRouteError(ctx, res, err);
2228
+ }
2229
+ }));
2230
+ }
2231
+ //#endregion
2232
+ //#region extensions/browser/src/browser/routes/agent.storage.ts
2233
+ function readStringValue(value) {
2234
+ return typeof value === "string" ? value : void 0;
2235
+ }
2236
+ function normalizeOptionalString(value) {
2237
+ return readStringValue(value)?.trim() || void 0;
2238
+ }
2239
+ function parseStorageKind(raw) {
2240
+ if (raw === "local" || raw === "session") return raw;
2241
+ return null;
2242
+ }
2243
+ function parseStorageMutationRequest(kindParam, body) {
2244
+ return {
2245
+ kind: parseStorageKind(toStringOrEmpty(kindParam)),
2246
+ targetId: resolveTargetIdFromBody(body)
2247
+ };
2248
+ }
2249
+ function parseRequiredStorageMutationRequest(kindParam, body) {
2250
+ const parsed = parseStorageMutationRequest(kindParam, body);
2251
+ if (!parsed.kind) return null;
2252
+ return {
2253
+ kind: parsed.kind,
2254
+ targetId: parsed.targetId
2255
+ };
2256
+ }
2257
+ function parseStorageMutationOrRespond(res, kindParam, body) {
2258
+ const parsed = parseRequiredStorageMutationRequest(kindParam, body);
2259
+ if (!parsed) {
2260
+ jsonError(res, 400, "kind must be local|session");
2261
+ return null;
2262
+ }
2263
+ return parsed;
2264
+ }
2265
+ function parseStorageMutationFromRequest(req, res) {
2266
+ const body = readBody(req);
2267
+ const parsed = parseStorageMutationOrRespond(res, req.params.kind, body);
2268
+ if (!parsed) return null;
2269
+ return {
2270
+ body,
2271
+ parsed
2272
+ };
2273
+ }
2274
+ function registerBrowserAgentStorageRoutes(app, ctx) {
2275
+ app.get("/cookies", asyncBrowserRoute(async (req, res) => {
2276
+ await withPlaywrightRouteContext({
2277
+ req,
2278
+ res,
2279
+ ctx,
2280
+ targetId: resolveTargetIdFromQuery(req.query),
2281
+ feature: "cookies",
2282
+ run: async ({ cdpUrl, tab, pw }) => {
2283
+ const result = await pw.cookiesGetViaPlaywright({
2284
+ cdpUrl,
2285
+ targetId: tab.targetId
2286
+ });
2287
+ res.json({
2288
+ ok: true,
2289
+ targetId: tab.targetId,
2290
+ ...result
2291
+ });
2292
+ }
2293
+ });
2294
+ }));
2295
+ app.post("/cookies/set", asyncBrowserRoute(async (req, res) => {
2296
+ const body = readBody(req);
2297
+ const targetId = resolveTargetIdFromBody(body);
2298
+ const cookie = body.cookie && typeof body.cookie === "object" && !Array.isArray(body.cookie) ? body.cookie : null;
2299
+ if (!cookie) return jsonError(res, 400, "cookie is required");
2300
+ await withPlaywrightRouteContext({
2301
+ req,
2302
+ res,
2303
+ ctx,
2304
+ targetId,
2305
+ feature: "cookies set",
2306
+ run: async ({ cdpUrl, tab, pw }) => {
2307
+ await pw.cookiesSetViaPlaywright({
2308
+ cdpUrl,
2309
+ targetId: tab.targetId,
2310
+ cookie: {
2311
+ name: toStringOrEmpty(cookie.name),
2312
+ value: toStringOrEmpty(cookie.value),
2313
+ url: toStringOrEmpty(cookie.url) || void 0,
2314
+ domain: toStringOrEmpty(cookie.domain) || void 0,
2315
+ path: toStringOrEmpty(cookie.path) || void 0,
2316
+ expires: toNumber(cookie.expires) ?? void 0,
2317
+ httpOnly: toBoolean(cookie.httpOnly) ?? void 0,
2318
+ secure: toBoolean(cookie.secure) ?? void 0,
2319
+ sameSite: cookie.sameSite === "Lax" || cookie.sameSite === "None" || cookie.sameSite === "Strict" ? cookie.sameSite : void 0
2320
+ }
2321
+ });
2322
+ res.json({
2323
+ ok: true,
2324
+ targetId: tab.targetId
2325
+ });
2326
+ }
2327
+ });
2328
+ }));
2329
+ app.post("/cookies/clear", asyncBrowserRoute(async (req, res) => {
2330
+ await withPlaywrightRouteContext({
2331
+ req,
2332
+ res,
2333
+ ctx,
2334
+ targetId: resolveTargetIdFromBody(readBody(req)),
2335
+ feature: "cookies clear",
2336
+ run: async ({ cdpUrl, tab, pw }) => {
2337
+ await pw.cookiesClearViaPlaywright({
2338
+ cdpUrl,
2339
+ targetId: tab.targetId
2340
+ });
2341
+ res.json({
2342
+ ok: true,
2343
+ targetId: tab.targetId
2344
+ });
2345
+ }
2346
+ });
2347
+ }));
2348
+ app.get("/storage/:kind", asyncBrowserRoute(async (req, res) => {
2349
+ const kind = parseStorageKind(toStringOrEmpty(req.params.kind));
2350
+ if (!kind) return jsonError(res, 400, "kind must be local|session");
2351
+ const targetId = resolveTargetIdFromQuery(req.query);
2352
+ const key = toStringOrEmpty(req.query.key);
2353
+ await withPlaywrightRouteContext({
2354
+ req,
2355
+ res,
2356
+ ctx,
2357
+ targetId,
2358
+ feature: "storage get",
2359
+ run: async ({ cdpUrl, tab, pw }) => {
2360
+ const result = await pw.storageGetViaPlaywright({
2361
+ cdpUrl,
2362
+ targetId: tab.targetId,
2363
+ kind,
2364
+ key: normalizeOptionalString(key)
2365
+ });
2366
+ res.json({
2367
+ ok: true,
2368
+ targetId: tab.targetId,
2369
+ ...result
2370
+ });
2371
+ }
2372
+ });
2373
+ }));
2374
+ app.post("/storage/:kind/set", asyncBrowserRoute(async (req, res) => {
2375
+ const mutation = parseStorageMutationFromRequest(req, res);
2376
+ if (!mutation) return;
2377
+ const key = toStringOrEmpty(mutation.body.key);
2378
+ if (!key) return jsonError(res, 400, "key is required");
2379
+ const value = typeof mutation.body.value === "string" ? mutation.body.value : "";
2380
+ await withPlaywrightRouteContext({
2381
+ req,
2382
+ res,
2383
+ ctx,
2384
+ targetId: mutation.parsed.targetId,
2385
+ feature: "storage set",
2386
+ run: async ({ cdpUrl, tab, pw }) => {
2387
+ await pw.storageSetViaPlaywright({
2388
+ cdpUrl,
2389
+ targetId: tab.targetId,
2390
+ kind: mutation.parsed.kind,
2391
+ key,
2392
+ value
2393
+ });
2394
+ res.json({
2395
+ ok: true,
2396
+ targetId: tab.targetId
2397
+ });
2398
+ }
2399
+ });
2400
+ }));
2401
+ app.post("/storage/:kind/clear", asyncBrowserRoute(async (req, res) => {
2402
+ const mutation = parseStorageMutationFromRequest(req, res);
2403
+ if (!mutation) return;
2404
+ await withPlaywrightRouteContext({
2405
+ req,
2406
+ res,
2407
+ ctx,
2408
+ targetId: mutation.parsed.targetId,
2409
+ feature: "storage clear",
2410
+ run: async ({ cdpUrl, tab, pw }) => {
2411
+ await pw.storageClearViaPlaywright({
2412
+ cdpUrl,
2413
+ targetId: tab.targetId,
2414
+ kind: mutation.parsed.kind
2415
+ });
2416
+ res.json({
2417
+ ok: true,
2418
+ targetId: tab.targetId
2419
+ });
2420
+ }
2421
+ });
2422
+ }));
2423
+ app.post("/set/offline", asyncBrowserRoute(async (req, res) => {
2424
+ const body = readBody(req);
2425
+ const targetId = resolveTargetIdFromBody(body);
2426
+ const offline = toBoolean(body.offline);
2427
+ if (offline === void 0) return jsonError(res, 400, "offline is required");
2428
+ await withPlaywrightRouteContext({
2429
+ req,
2430
+ res,
2431
+ ctx,
2432
+ targetId,
2433
+ feature: "offline",
2434
+ run: async ({ cdpUrl, tab, pw }) => {
2435
+ await pw.setOfflineViaPlaywright({
2436
+ cdpUrl,
2437
+ targetId: tab.targetId,
2438
+ offline
2439
+ });
2440
+ res.json({
2441
+ ok: true,
2442
+ targetId: tab.targetId
2443
+ });
2444
+ }
2445
+ });
2446
+ }));
2447
+ app.post("/set/headers", asyncBrowserRoute(async (req, res) => {
2448
+ const body = readBody(req);
2449
+ const targetId = resolveTargetIdFromBody(body);
2450
+ const headers = body.headers && typeof body.headers === "object" && !Array.isArray(body.headers) ? body.headers : null;
2451
+ if (!headers) return jsonError(res, 400, "headers is required");
2452
+ const parsed = {};
2453
+ for (const [k, v] of Object.entries(headers)) if (typeof v === "string") parsed[k] = v;
2454
+ await withPlaywrightRouteContext({
2455
+ req,
2456
+ res,
2457
+ ctx,
2458
+ targetId,
2459
+ feature: "headers",
2460
+ run: async ({ cdpUrl, tab, pw }) => {
2461
+ await pw.setExtraHTTPHeadersViaPlaywright({
2462
+ cdpUrl,
2463
+ targetId: tab.targetId,
2464
+ headers: parsed
2465
+ });
2466
+ res.json({
2467
+ ok: true,
2468
+ targetId: tab.targetId
2469
+ });
2470
+ }
2471
+ });
2472
+ }));
2473
+ app.post("/set/credentials", asyncBrowserRoute(async (req, res) => {
2474
+ const body = readBody(req);
2475
+ const targetId = resolveTargetIdFromBody(body);
2476
+ const clear = toBoolean(body.clear) ?? false;
2477
+ const username = toStringOrEmpty(body.username) || void 0;
2478
+ const password = readStringValue(body.password);
2479
+ await withPlaywrightRouteContext({
2480
+ req,
2481
+ res,
2482
+ ctx,
2483
+ targetId,
2484
+ feature: "http credentials",
2485
+ run: async ({ cdpUrl, tab, pw }) => {
2486
+ await pw.setHttpCredentialsViaPlaywright({
2487
+ cdpUrl,
2488
+ targetId: tab.targetId,
2489
+ username,
2490
+ password,
2491
+ clear
2492
+ });
2493
+ res.json({
2494
+ ok: true,
2495
+ targetId: tab.targetId
2496
+ });
2497
+ }
2498
+ });
2499
+ }));
2500
+ app.post("/set/geolocation", asyncBrowserRoute(async (req, res) => {
2501
+ const body = readBody(req);
2502
+ const targetId = resolveTargetIdFromBody(body);
2503
+ const clear = toBoolean(body.clear) ?? false;
2504
+ const latitude = toNumber(body.latitude);
2505
+ const longitude = toNumber(body.longitude);
2506
+ const accuracy = toNumber(body.accuracy) ?? void 0;
2507
+ const origin = toStringOrEmpty(body.origin) || void 0;
2508
+ await withPlaywrightRouteContext({
2509
+ req,
2510
+ res,
2511
+ ctx,
2512
+ targetId,
2513
+ feature: "geolocation",
2514
+ run: async ({ cdpUrl, tab, pw }) => {
2515
+ await pw.setGeolocationViaPlaywright({
2516
+ cdpUrl,
2517
+ targetId: tab.targetId,
2518
+ latitude,
2519
+ longitude,
2520
+ accuracy,
2521
+ origin,
2522
+ clear
2523
+ });
2524
+ res.json({
2525
+ ok: true,
2526
+ targetId: tab.targetId
2527
+ });
2528
+ }
2529
+ });
2530
+ }));
2531
+ app.post("/set/media", asyncBrowserRoute(async (req, res) => {
2532
+ const body = readBody(req);
2533
+ const targetId = resolveTargetIdFromBody(body);
2534
+ const schemeRaw = toStringOrEmpty(body.colorScheme);
2535
+ const colorScheme = schemeRaw === "dark" || schemeRaw === "light" || schemeRaw === "no-preference" ? schemeRaw : schemeRaw === "none" ? null : void 0;
2536
+ if (colorScheme === void 0) return jsonError(res, 400, "colorScheme must be dark|light|no-preference|none");
2537
+ await withPlaywrightRouteContext({
2538
+ req,
2539
+ res,
2540
+ ctx,
2541
+ targetId,
2542
+ feature: "media emulation",
2543
+ run: async ({ cdpUrl, tab, pw }) => {
2544
+ await pw.emulateMediaViaPlaywright({
2545
+ cdpUrl,
2546
+ targetId: tab.targetId,
2547
+ colorScheme
2548
+ });
2549
+ res.json({
2550
+ ok: true,
2551
+ targetId: tab.targetId
2552
+ });
2553
+ }
2554
+ });
2555
+ }));
2556
+ app.post("/set/timezone", asyncBrowserRoute(async (req, res) => {
2557
+ const body = readBody(req);
2558
+ const targetId = resolveTargetIdFromBody(body);
2559
+ const timezoneId = toStringOrEmpty(body.timezoneId);
2560
+ if (!timezoneId) return jsonError(res, 400, "timezoneId is required");
2561
+ await withPlaywrightRouteContext({
2562
+ req,
2563
+ res,
2564
+ ctx,
2565
+ targetId,
2566
+ feature: "timezone",
2567
+ run: async ({ cdpUrl, tab, pw }) => {
2568
+ await pw.setTimezoneViaPlaywright({
2569
+ cdpUrl,
2570
+ targetId: tab.targetId,
2571
+ timezoneId
2572
+ });
2573
+ res.json({
2574
+ ok: true,
2575
+ targetId: tab.targetId
2576
+ });
2577
+ }
2578
+ });
2579
+ }));
2580
+ app.post("/set/locale", asyncBrowserRoute(async (req, res) => {
2581
+ const body = readBody(req);
2582
+ const targetId = resolveTargetIdFromBody(body);
2583
+ const locale = toStringOrEmpty(body.locale);
2584
+ if (!locale) return jsonError(res, 400, "locale is required");
2585
+ await withPlaywrightRouteContext({
2586
+ req,
2587
+ res,
2588
+ ctx,
2589
+ targetId,
2590
+ feature: "locale",
2591
+ run: async ({ cdpUrl, tab, pw }) => {
2592
+ await pw.setLocaleViaPlaywright({
2593
+ cdpUrl,
2594
+ targetId: tab.targetId,
2595
+ locale
2596
+ });
2597
+ res.json({
2598
+ ok: true,
2599
+ targetId: tab.targetId
2600
+ });
2601
+ }
2602
+ });
2603
+ }));
2604
+ app.post("/set/device", asyncBrowserRoute(async (req, res) => {
2605
+ const body = readBody(req);
2606
+ const targetId = resolveTargetIdFromBody(body);
2607
+ const name = toStringOrEmpty(body.name);
2608
+ if (!name) return jsonError(res, 400, "name is required");
2609
+ await withPlaywrightRouteContext({
2610
+ req,
2611
+ res,
2612
+ ctx,
2613
+ targetId,
2614
+ feature: "device emulation",
2615
+ run: async ({ cdpUrl, tab, pw }) => {
2616
+ await pw.setDeviceViaPlaywright({
2617
+ cdpUrl,
2618
+ targetId: tab.targetId,
2619
+ name
2620
+ });
2621
+ res.json({
2622
+ ok: true,
2623
+ targetId: tab.targetId
2624
+ });
2625
+ }
2626
+ });
2627
+ }));
2628
+ }
2629
+ //#endregion
2630
+ //#region extensions/browser/src/browser/routes/agent.ts
2631
+ function registerBrowserAgentRoutes(app, ctx) {
2632
+ registerBrowserAgentSnapshotRoutes(app, ctx);
2633
+ registerBrowserAgentActRoutes(app, ctx);
2634
+ registerBrowserAgentDebugRoutes(app, ctx);
2635
+ registerBrowserAgentStorageRoutes(app, ctx);
2636
+ }
2637
+ //#endregion
2638
+ //#region extensions/browser/src/browser/doctor.ts
2639
+ function buildBrowserDoctorReport(params) {
2640
+ const status = params.status;
2641
+ const checks = [];
2642
+ const transport = status.transport === "chrome-mcp" ? "chrome-mcp" : "cdp";
2643
+ checks.push({
2644
+ id: "plugin-enabled",
2645
+ label: "Browser plugin",
2646
+ status: status.enabled ? "pass" : "fail",
2647
+ summary: status.enabled ? "enabled" : "disabled",
2648
+ ...status.enabled ? {} : { fixHint: "Enable the browser plugin and restart the Gateway." }
2649
+ });
2650
+ checks.push({
2651
+ id: "profile",
2652
+ label: "Profile",
2653
+ status: "pass",
2654
+ summary: `${status.profile ?? "genesis"} via ${transport}`
2655
+ });
2656
+ if (transport === "chrome-mcp") checks.push({
2657
+ id: "attach-target",
2658
+ label: "Existing browser attach",
2659
+ status: status.running ? "pass" : "fail",
2660
+ summary: status.running ? "Chrome MCP target is reachable" : "Chrome MCP target is not reachable",
2661
+ ...status.running ? {} : { fixHint: "Keep the matching Chromium browser running, enable remote debugging in chrome://inspect, and accept the attach prompt." }
2662
+ });
2663
+ else {
2664
+ checks.push({
2665
+ id: "managed-executable",
2666
+ label: "Chromium executable",
2667
+ status: status.detectError ? "fail" : status.detectedExecutablePath ? "pass" : "warn",
2668
+ summary: status.detectError ? status.detectError : status.detectedExecutablePath ? `${status.detectedBrowser ?? "chromium"} at ${status.detectedExecutablePath}` : "No Chromium executable detected",
2669
+ ...status.detectedExecutablePath || status.detectError ? {} : { fixHint: "Install Chrome/Chromium/Brave/Edge or set browser.executablePath." }
2670
+ });
2671
+ const platform = params.platform ?? process.platform;
2672
+ const env = params.env ?? process.env;
2673
+ const uid = params.uid ?? process.getuid?.();
2674
+ if (platform === "linux" && !status.headless && !env.DISPLAY && !env.WAYLAND_DISPLAY) checks.push({
2675
+ id: "display",
2676
+ label: "Display",
2677
+ status: "warn",
2678
+ summary: "No DISPLAY or WAYLAND_DISPLAY is set while browser.headless is false",
2679
+ fixHint: "Use a desktop session, Xvfb, or set browser.headless: true."
2680
+ });
2681
+ if (platform === "linux" && uid === 0 && !status.noSandbox) checks.push({
2682
+ id: "linux-sandbox",
2683
+ label: "Linux sandbox",
2684
+ status: "warn",
2685
+ summary: "Gateway is running as root while browser.noSandbox is false",
2686
+ fixHint: "Set browser.noSandbox: true for container/root Chromium runtimes."
2687
+ });
2688
+ checks.push({
2689
+ id: "cdp-http",
2690
+ label: "CDP HTTP",
2691
+ status: status.cdpHttp ? "pass" : status.running ? "fail" : "info",
2692
+ summary: status.cdpHttp ? "CDP HTTP endpoint is reachable" : status.running ? "CDP HTTP endpoint is not reachable" : "Browser is not currently running",
2693
+ ...status.cdpHttp || !status.running ? {} : { fixHint: "Run genesis browser start or inspect browser.cdpUrl/CDP port reachability." }
2694
+ });
2695
+ checks.push({
2696
+ id: "cdp-websocket",
2697
+ label: "CDP WebSocket",
2698
+ status: status.cdpReady ? "pass" : status.running ? "fail" : "info",
2699
+ summary: status.cdpReady ? "CDP WebSocket is reachable" : status.running ? "CDP WebSocket is not reachable" : "Browser is launchable but not running",
2700
+ ...status.cdpReady || !status.running ? {} : { fixHint: "Check Chrome launch logs, stale locks, proxy env, and port conflicts." }
2701
+ });
2702
+ }
2703
+ return {
2704
+ ok: checks.every((check) => check.status !== "fail"),
2705
+ profile: status.profile ?? "genesis",
2706
+ transport,
2707
+ checks,
2708
+ status
2709
+ };
2710
+ }
2711
+ const PROFILE_NAME_REGEX = /^[a-z0-9][a-z0-9-]*$/;
2712
+ function isValidProfileName(name) {
2713
+ if (!name || name.length > 64) return false;
2714
+ return PROFILE_NAME_REGEX.test(name);
2715
+ }
2716
+ function allocateCdpPort(usedPorts, range) {
2717
+ const start = range?.start ?? 18800;
2718
+ const end = range?.end ?? 18899;
2719
+ if (!Number.isFinite(start) || !Number.isFinite(end) || start <= 0 || end <= 0) return null;
2720
+ if (start > end) return null;
2721
+ for (let port = start; port <= end; port++) if (!usedPorts.has(port)) return port;
2722
+ return null;
2723
+ }
2724
+ function getUsedPorts(profiles) {
2725
+ if (!profiles) return /* @__PURE__ */ new Set();
2726
+ const used = /* @__PURE__ */ new Set();
2727
+ for (const profile of Object.values(profiles)) {
2728
+ if (typeof profile.cdpPort === "number") {
2729
+ used.add(profile.cdpPort);
2730
+ continue;
2731
+ }
2732
+ const rawUrl = profile.cdpUrl?.trim();
2733
+ if (!rawUrl) continue;
2734
+ try {
2735
+ const parsed = new URL(rawUrl);
2736
+ const port = parsed.port && Number.parseInt(parsed.port, 10) > 0 ? Number.parseInt(parsed.port, 10) : parsed.protocol === "https:" ? 443 : 80;
2737
+ if (!Number.isNaN(port) && port > 0 && port <= 65535) used.add(port);
2738
+ } catch {}
2739
+ }
2740
+ return used;
2741
+ }
2742
+ const PROFILE_COLORS = [
2743
+ "#FF4500",
2744
+ "#0066CC",
2745
+ "#00AA00",
2746
+ "#9933FF",
2747
+ "#FF6699",
2748
+ "#00CCCC",
2749
+ "#FF9900",
2750
+ "#6666FF",
2751
+ "#CC3366",
2752
+ "#339966"
2753
+ ];
2754
+ function allocateColor(usedColors) {
2755
+ for (const color of PROFILE_COLORS) if (!usedColors.has(color.toUpperCase())) return color;
2756
+ return PROFILE_COLORS[usedColors.size % PROFILE_COLORS.length] ?? PROFILE_COLORS[0];
2757
+ }
2758
+ function getUsedColors(profiles) {
2759
+ if (!profiles) return /* @__PURE__ */ new Set();
2760
+ return new Set(Object.values(profiles).map((p) => p.color.toUpperCase()));
2761
+ }
2762
+ //#endregion
2763
+ //#region extensions/browser/src/browser/profiles-service.ts
2764
+ const HEX_COLOR_RE = /^#[0-9A-Fa-f]{6}$/;
2765
+ const cdpPortRange = (resolved) => {
2766
+ const start = resolved.cdpPortRangeStart;
2767
+ const end = resolved.cdpPortRangeEnd;
2768
+ if (typeof start === "number" && Number.isFinite(start) && Number.isInteger(start) && typeof end === "number" && Number.isFinite(end) && Number.isInteger(end) && start > 0 && end >= start && end <= 65535) return {
2769
+ start,
2770
+ end
2771
+ };
2772
+ return deriveDefaultBrowserCdpPortRange(resolved.controlPort);
2773
+ };
2774
+ function createBrowserProfilesService(ctx) {
2775
+ const listProfiles = async () => {
2776
+ return await ctx.listProfiles();
2777
+ };
2778
+ const createProfile = async (params) => {
2779
+ const name = params.name.trim();
2780
+ const rawCdpUrl = normalizeOptionalString$4(params.cdpUrl);
2781
+ const rawUserDataDir = normalizeOptionalString$4(params.userDataDir);
2782
+ const normalizedUserDataDir = rawUserDataDir ? resolveUserPath(rawUserDataDir) : void 0;
2783
+ const driver = params.driver === "existing-session" ? "existing-session" : void 0;
2784
+ if (!isValidProfileName(name)) throw new BrowserValidationError("invalid profile name: use lowercase letters, numbers, and hyphens only");
2785
+ const state = ctx.state();
2786
+ const resolvedProfiles = state.resolved.profiles;
2787
+ if (name in resolvedProfiles) throw new BrowserConflictError(`profile "${name}" already exists`);
2788
+ const cfg = loadConfig();
2789
+ const rawProfiles = cfg.browser?.profiles ?? {};
2790
+ if (name in rawProfiles) throw new BrowserConflictError(`profile "${name}" already exists`);
2791
+ const usedColors = getUsedColors(resolvedProfiles);
2792
+ const profileColor = params.color && HEX_COLOR_RE.test(params.color) ? params.color : allocateColor(usedColors);
2793
+ let profileConfig;
2794
+ if (normalizedUserDataDir && driver !== "existing-session") throw new BrowserValidationError("driver=existing-session is required when userDataDir is provided");
2795
+ if (normalizedUserDataDir && !fs.existsSync(normalizedUserDataDir)) throw new BrowserValidationError(`browser user data directory not found: ${normalizedUserDataDir}`);
2796
+ if (params.tor && (rawCdpUrl || driver === "existing-session")) throw new BrowserValidationError("Tor routing is only supported for locally launched managed browser profiles");
2797
+ if (rawCdpUrl) {
2798
+ if (driver === "existing-session") throw new BrowserValidationError("driver=existing-session does not accept cdpUrl; it attaches via the Chrome MCP auto-connect flow");
2799
+ let parsed;
2800
+ try {
2801
+ parsed = parseBrowserHttpUrl(rawCdpUrl, "browser.profiles.cdpUrl");
2802
+ await assertCdpEndpointAllowed(parsed.normalized, state.resolved.ssrfPolicy);
2803
+ } catch (err) {
2804
+ throw new BrowserValidationError(formatErrorMessage(err));
2805
+ }
2806
+ profileConfig = {
2807
+ cdpUrl: parsed.normalized,
2808
+ ...driver ? { driver } : {},
2809
+ color: profileColor
2810
+ };
2811
+ } else if (driver === "existing-session") profileConfig = {
2812
+ driver,
2813
+ attachOnly: true,
2814
+ ...normalizedUserDataDir ? { userDataDir: normalizedUserDataDir } : {},
2815
+ color: profileColor
2816
+ };
2817
+ else {
2818
+ const cdpPort = allocateCdpPort(getUsedPorts(resolvedProfiles), cdpPortRange(state.resolved));
2819
+ if (cdpPort === null) throw new BrowserResourceExhaustedError("no available CDP ports in range");
2820
+ profileConfig = {
2821
+ cdpPort,
2822
+ ...driver ? { driver } : {},
2823
+ ...params.tor ? {
2824
+ headless: true,
2825
+ tor: {
2826
+ enabled: true,
2827
+ mode: "managed"
2828
+ }
2829
+ } : {},
2830
+ color: profileColor
2831
+ };
2832
+ }
2833
+ await writeConfigFile({
2834
+ ...cfg,
2835
+ browser: {
2836
+ ...cfg.browser,
2837
+ profiles: {
2838
+ ...rawProfiles,
2839
+ [name]: profileConfig
2840
+ }
2841
+ }
2842
+ });
2843
+ state.resolved.profiles[name] = profileConfig;
2844
+ const resolved = resolveProfile(state.resolved, name);
2845
+ if (!resolved) throw new BrowserProfileNotFoundError(`profile "${name}" not found after creation`);
2846
+ const capabilities = getBrowserProfileCapabilities(resolved);
2847
+ return {
2848
+ ok: true,
2849
+ profile: name,
2850
+ transport: capabilities.usesChromeMcp ? "chrome-mcp" : "cdp",
2851
+ cdpPort: capabilities.usesChromeMcp ? null : resolved.cdpPort,
2852
+ cdpUrl: capabilities.usesChromeMcp ? null : resolved.cdpUrl,
2853
+ userDataDir: resolved.userDataDir ?? null,
2854
+ color: resolved.color,
2855
+ isRemote: !resolved.cdpIsLoopback
2856
+ };
2857
+ };
2858
+ const deleteProfile = async (nameRaw) => {
2859
+ const name = nameRaw.trim();
2860
+ if (!name) throw new BrowserValidationError("profile name is required");
2861
+ if (!isValidProfileName(name)) throw new BrowserValidationError("invalid profile name");
2862
+ const state = ctx.state();
2863
+ const cfg = loadConfig();
2864
+ const profiles = cfg.browser?.profiles ?? {};
2865
+ if (name === (cfg.browser?.defaultProfile ?? state.resolved.defaultProfile)) throw new BrowserValidationError(`cannot delete the default profile "${name}"; change browser.defaultProfile first`);
2866
+ if (!(name in profiles)) throw new BrowserProfileNotFoundError(`profile "${name}" not found`);
2867
+ let deleted = false;
2868
+ const resolved = resolveProfile(state.resolved, name);
2869
+ if (resolved?.cdpIsLoopback && resolved.driver === "genesis") {
2870
+ try {
2871
+ await ctx.forProfile(name).stopRunningBrowser();
2872
+ } catch {}
2873
+ const userDataDir = resolveGenesisUserDataDir(name);
2874
+ const profileDir = path.dirname(userDataDir);
2875
+ if (fs.existsSync(profileDir)) {
2876
+ await movePathToTrash(profileDir);
2877
+ deleted = true;
2878
+ }
2879
+ }
2880
+ const { [name]: _removed, ...remainingProfiles } = profiles;
2881
+ await writeConfigFile({
2882
+ ...cfg,
2883
+ browser: {
2884
+ ...cfg.browser,
2885
+ profiles: remainingProfiles
2886
+ }
2887
+ });
2888
+ delete state.resolved.profiles[name];
2889
+ state.profiles.delete(name);
2890
+ return {
2891
+ ok: true,
2892
+ profile: name,
2893
+ deleted
2894
+ };
2895
+ };
2896
+ return {
2897
+ listProfiles,
2898
+ createProfile,
2899
+ deleteProfile
2900
+ };
2901
+ }
2902
+ //#endregion
2903
+ //#region extensions/browser/src/browser/routes/basic.ts
2904
+ function handleBrowserRouteError(res, err) {
2905
+ const mapped = toBrowserErrorResponse(err);
2906
+ if (mapped) return jsonError(res, mapped.status, mapped.message);
2907
+ jsonError(res, 500, String(err));
2908
+ }
2909
+ function buildTorStatus(profileCtx, running) {
2910
+ const tor = profileCtx.profile.tor;
2911
+ if (!tor?.enabled) return null;
2912
+ return {
2913
+ enabled: true,
2914
+ mode: tor.mode,
2915
+ routeMode: tor.routeMode,
2916
+ socksHost: tor.socksHost,
2917
+ socksPort: tor.socksPort,
2918
+ running
2919
+ };
2920
+ }
2921
+ async function withBasicProfileRoute(params) {
2922
+ const profileCtx = resolveProfileContext(params.req, params.res, params.ctx);
2923
+ if (!profileCtx) return;
2924
+ try {
2925
+ await params.run(profileCtx);
2926
+ } catch (err) {
2927
+ return handleBrowserRouteError(params.res, err);
2928
+ }
2929
+ }
2930
+ async function withProfilesServiceMutation(params) {
2931
+ try {
2932
+ const service = createBrowserProfilesService(params.ctx);
2933
+ const result = await params.run(service);
2934
+ params.res.json(result);
2935
+ } catch (err) {
2936
+ return handleBrowserRouteError(params.res, err);
2937
+ }
2938
+ }
2939
+ async function buildBrowserStatus(req, ctx) {
2940
+ let current;
2941
+ try {
2942
+ current = ctx.state();
2943
+ } catch {
2944
+ throw new BrowserError("browser server not started", 503);
2945
+ }
2946
+ const profileCtx = getProfileContext(req, ctx);
2947
+ if ("error" in profileCtx) throw new BrowserError(profileCtx.error, profileCtx.status);
2948
+ const capabilities = getBrowserProfileCapabilities(profileCtx.profile);
2949
+ const [cdpHttp, cdpReady] = capabilities.usesChromeMcp ? await (async () => {
2950
+ const ready = await profileCtx.isTransportAvailable(600);
2951
+ return [ready, ready];
2952
+ })() : await Promise.all([profileCtx.isHttpReachable(300), profileCtx.isTransportAvailable(600)]);
2953
+ const profileState = current.profiles.get(profileCtx.profile.name);
2954
+ let detectedBrowser = null;
2955
+ let detectedExecutablePath = null;
2956
+ let detectError = null;
2957
+ try {
2958
+ const detected = resolveBrowserExecutableForPlatform(current.resolved, process.platform);
2959
+ if (detected) {
2960
+ detectedBrowser = detected.kind;
2961
+ detectedExecutablePath = detected.path;
2962
+ }
2963
+ } catch (err) {
2964
+ detectError = String(err);
2965
+ }
2966
+ return {
2967
+ enabled: current.resolved.enabled,
2968
+ profile: profileCtx.profile.name,
2969
+ driver: profileCtx.profile.driver,
2970
+ transport: capabilities.usesChromeMcp ? "chrome-mcp" : "cdp",
2971
+ running: cdpReady,
2972
+ cdpReady,
2973
+ cdpHttp,
2974
+ pid: capabilities.usesChromeMcp ? getChromeMcpPid(profileCtx.profile.name) : profileState?.running?.pid ?? null,
2975
+ cdpPort: capabilities.usesChromeMcp ? null : profileCtx.profile.cdpPort,
2976
+ cdpUrl: capabilities.usesChromeMcp ? null : profileCtx.profile.cdpUrl,
2977
+ chosenBrowser: profileState?.running?.exe.kind ?? null,
2978
+ detectedBrowser,
2979
+ detectedExecutablePath,
2980
+ detectError,
2981
+ userDataDir: profileState?.running?.userDataDir ?? profileCtx.profile.userDataDir ?? null,
2982
+ color: profileCtx.profile.color,
2983
+ headless: profileCtx.profile.headless,
2984
+ noSandbox: current.resolved.noSandbox,
2985
+ executablePath: profileCtx.profile.executablePath ?? null,
2986
+ attachOnly: profileCtx.profile.attachOnly,
2987
+ tor: buildTorStatus(profileCtx, profileState?.running?.tor ? true : null)
2988
+ };
2989
+ }
2990
+ function registerBrowserBasicRoutes(app, ctx) {
2991
+ app.get("/profiles", asyncBrowserRoute(async (_req, res) => {
2992
+ try {
2993
+ const profiles = await createBrowserProfilesService(ctx).listProfiles();
2994
+ res.json({ profiles });
2995
+ } catch (err) {
2996
+ jsonError(res, 500, String(err));
2997
+ }
2998
+ }));
2999
+ app.get("/", asyncBrowserRoute(async (req, res) => {
3000
+ try {
3001
+ res.json(await buildBrowserStatus(req, ctx));
3002
+ } catch (err) {
3003
+ const mapped = toBrowserErrorResponse(err);
3004
+ if (mapped) return jsonError(res, mapped.status, mapped.message);
3005
+ jsonError(res, 500, String(err));
3006
+ }
3007
+ }));
3008
+ app.get("/doctor", asyncBrowserRoute(async (req, res) => {
3009
+ try {
3010
+ const status = await buildBrowserStatus(req, ctx);
3011
+ res.json(buildBrowserDoctorReport({ status }));
3012
+ } catch (err) {
3013
+ const mapped = toBrowserErrorResponse(err);
3014
+ if (mapped) return jsonError(res, mapped.status, mapped.message);
3015
+ jsonError(res, 500, String(err));
3016
+ }
3017
+ }));
3018
+ app.post("/start", asyncBrowserRoute(async (req, res) => {
3019
+ await withBasicProfileRoute({
3020
+ req,
3021
+ res,
3022
+ ctx,
3023
+ run: async (profileCtx) => {
3024
+ await profileCtx.ensureBrowserAvailable();
3025
+ res.json({
3026
+ ok: true,
3027
+ profile: profileCtx.profile.name
3028
+ });
3029
+ }
3030
+ });
3031
+ }));
3032
+ app.post("/stop", asyncBrowserRoute(async (req, res) => {
3033
+ await withBasicProfileRoute({
3034
+ req,
3035
+ res,
3036
+ ctx,
3037
+ run: async (profileCtx) => {
3038
+ const result = await profileCtx.stopRunningBrowser();
3039
+ res.json({
3040
+ ok: true,
3041
+ stopped: result.stopped,
3042
+ profile: profileCtx.profile.name
3043
+ });
3044
+ }
3045
+ });
3046
+ }));
3047
+ app.post("/reset-profile", asyncBrowserRoute(async (req, res) => {
3048
+ await withBasicProfileRoute({
3049
+ req,
3050
+ res,
3051
+ ctx,
3052
+ run: async (profileCtx) => {
3053
+ const result = await profileCtx.resetProfile();
3054
+ res.json({
3055
+ ok: true,
3056
+ profile: profileCtx.profile.name,
3057
+ ...result
3058
+ });
3059
+ }
3060
+ });
3061
+ }));
3062
+ app.post("/profiles/create", asyncBrowserRoute(async (req, res) => {
3063
+ const name = toStringOrEmpty(req.body?.name);
3064
+ const color = toStringOrEmpty(req.body?.color);
3065
+ const cdpUrl = toStringOrEmpty(req.body?.cdpUrl);
3066
+ const userDataDir = toStringOrEmpty(req.body?.userDataDir);
3067
+ const driver = toStringOrEmpty(req.body?.driver);
3068
+ const tor = req.body?.tor === true;
3069
+ if (!name) return jsonError(res, 400, "name is required");
3070
+ if (driver && driver !== "genesis" && driver !== "clawd" && driver !== "existing-session") return jsonError(res, 400, `unsupported profile driver "${driver}"; use "genesis", "clawd", or "existing-session"`);
3071
+ await withProfilesServiceMutation({
3072
+ res,
3073
+ ctx,
3074
+ run: async (service) => await service.createProfile({
3075
+ name,
3076
+ color: color || void 0,
3077
+ cdpUrl: cdpUrl || void 0,
3078
+ userDataDir: userDataDir || void 0,
3079
+ tor,
3080
+ driver: driver === "existing-session" ? "existing-session" : driver === "genesis" || driver === "clawd" ? "genesis" : void 0
3081
+ })
3082
+ });
3083
+ }));
3084
+ app.delete("/profiles/:name", asyncBrowserRoute(async (req, res) => {
3085
+ const name = toStringOrEmpty(req.params.name);
3086
+ if (!name) return jsonError(res, 400, "profile name is required");
3087
+ await withProfilesServiceMutation({
3088
+ res,
3089
+ ctx,
3090
+ run: async (service) => await service.deleteProfile(name)
3091
+ });
3092
+ }));
3093
+ }
3094
+ //#endregion
3095
+ //#region extensions/browser/src/browser/routes/tabs.ts
3096
+ function resolveTabsProfileContext(req, res, ctx) {
3097
+ const profileCtx = getProfileContext(req, ctx);
3098
+ if ("error" in profileCtx) {
3099
+ jsonError(res, profileCtx.status, profileCtx.error);
3100
+ return null;
3101
+ }
3102
+ return profileCtx;
3103
+ }
3104
+ function browserNavigationPolicyForProfile(ctx, profileCtx) {
3105
+ return withBrowserNavigationPolicy(ctx.state().resolved.ssrfPolicy, { browserProxyMode: resolveBrowserNavigationProxyMode({
3106
+ resolved: ctx.state().resolved,
3107
+ profile: profileCtx.profile
3108
+ }) });
3109
+ }
3110
+ function handleTabsRouteError(ctx, res, err, opts) {
3111
+ if (opts?.mapTabError) {
3112
+ const mapped = ctx.mapTabError(err);
3113
+ if (mapped) return jsonError(res, mapped.status, mapped.message);
3114
+ }
3115
+ return jsonError(res, 500, String(err));
3116
+ }
3117
+ async function withTabsProfileRoute(params) {
3118
+ const profileCtx = resolveTabsProfileContext(params.req, params.res, params.ctx);
3119
+ if (!profileCtx) return;
3120
+ try {
3121
+ await params.run(profileCtx);
3122
+ } catch (err) {
3123
+ handleTabsRouteError(params.ctx, params.res, err, { mapTabError: params.mapTabError });
3124
+ }
3125
+ }
3126
+ async function ensureBrowserRunning(profileCtx, res) {
3127
+ if (!await profileCtx.isReachable(300)) {
3128
+ jsonError(res, new BrowserProfileUnavailableError("browser not running").status, "browser not running");
3129
+ return false;
3130
+ }
3131
+ return true;
3132
+ }
3133
+ async function redactBlockedTabUrls(params) {
3134
+ const ssrfPolicyOpts = withBrowserNavigationPolicy(params.ssrfPolicy);
3135
+ if (!ssrfPolicyOpts.ssrfPolicy) return params.tabs;
3136
+ const redactedTabs = [];
3137
+ for (const tab of params.tabs) try {
3138
+ await assertBrowserNavigationResultAllowed({
3139
+ url: tab.url,
3140
+ ...ssrfPolicyOpts
3141
+ });
3142
+ redactedTabs.push(tab);
3143
+ } catch {
3144
+ redactedTabs.push({
3145
+ ...tab,
3146
+ url: ""
3147
+ });
3148
+ }
3149
+ return redactedTabs;
3150
+ }
3151
+ function resolveIndexedTab(tabs, index) {
3152
+ return typeof index === "number" ? tabs[index] : tabs.at(0);
3153
+ }
3154
+ function parseRequiredTargetId(res, rawTargetId) {
3155
+ const targetId = toStringOrEmpty(rawTargetId);
3156
+ if (!targetId) {
3157
+ jsonError(res, 400, "targetId is required");
3158
+ return null;
3159
+ }
3160
+ return targetId;
3161
+ }
3162
+ function readOptionalTabLabel(body) {
3163
+ return toStringOrEmpty(body?.label) || void 0;
3164
+ }
3165
+ async function runTabTargetMutation(params) {
3166
+ await withTabsProfileRoute({
3167
+ req: params.req,
3168
+ res: params.res,
3169
+ ctx: params.ctx,
3170
+ mapTabError: true,
3171
+ run: async (profileCtx) => {
3172
+ if (!await ensureBrowserRunning(profileCtx, params.res)) return;
3173
+ await params.mutate(profileCtx, params.targetId);
3174
+ params.res.json({ ok: true });
3175
+ }
3176
+ });
3177
+ }
3178
+ function registerBrowserTabRoutes(app, ctx) {
3179
+ app.get("/tabs", asyncBrowserRoute(async (req, res) => {
3180
+ await withTabsProfileRoute({
3181
+ req,
3182
+ res,
3183
+ ctx,
3184
+ run: async (profileCtx) => {
3185
+ if (!await profileCtx.isReachable(300)) return res.json({
3186
+ running: false,
3187
+ tabs: []
3188
+ });
3189
+ const tabs = await redactBlockedTabUrls({
3190
+ tabs: await profileCtx.listTabs(),
3191
+ ssrfPolicy: ctx.state().resolved.ssrfPolicy
3192
+ });
3193
+ res.json({
3194
+ running: true,
3195
+ tabs
3196
+ });
3197
+ }
3198
+ });
3199
+ }));
3200
+ app.post("/tabs/open", asyncBrowserRoute(async (req, res) => {
3201
+ const url = toStringOrEmpty(req.body?.url);
3202
+ const label = readOptionalTabLabel(req.body);
3203
+ if (!url) return jsonError(res, 400, "url is required");
3204
+ await withTabsProfileRoute({
3205
+ req,
3206
+ res,
3207
+ ctx,
3208
+ mapTabError: true,
3209
+ run: async (profileCtx) => {
3210
+ await assertBrowserNavigationAllowed({
3211
+ url,
3212
+ ...browserNavigationPolicyForProfile(ctx, profileCtx)
3213
+ });
3214
+ await profileCtx.ensureBrowserAvailable();
3215
+ const tab = await profileCtx.openTab(url, { label });
3216
+ res.json(tab);
3217
+ }
3218
+ });
3219
+ }));
3220
+ app.post("/tabs/focus", asyncBrowserRoute(async (req, res) => {
3221
+ const targetId = parseRequiredTargetId(res, req.body?.targetId);
3222
+ if (!targetId) return;
3223
+ await runTabTargetMutation({
3224
+ req,
3225
+ res,
3226
+ ctx,
3227
+ targetId,
3228
+ mutate: async (profileCtx, id) => {
3229
+ const tabs = await profileCtx.listTabs();
3230
+ const resolved = resolveTargetIdFromTabs(id, tabs);
3231
+ if (!resolved.ok) {
3232
+ if (resolved.reason === "ambiguous") throw new BrowserTargetAmbiguousError();
3233
+ throw new BrowserTabNotFoundError({ input: id });
3234
+ }
3235
+ const tab = tabs.find((currentTab) => currentTab.targetId === resolved.targetId);
3236
+ if (!tab) throw new BrowserTabNotFoundError({ input: id });
3237
+ const ssrfPolicyOpts = browserNavigationPolicyForProfile(ctx, profileCtx);
3238
+ if (ssrfPolicyOpts.ssrfPolicy) await assertBrowserNavigationResultAllowed({
3239
+ url: tab.url,
3240
+ ...ssrfPolicyOpts
3241
+ });
3242
+ await profileCtx.focusTab(resolved.targetId);
3243
+ }
3244
+ });
3245
+ }));
3246
+ app.delete("/tabs/:targetId", asyncBrowserRoute(async (req, res) => {
3247
+ const targetId = parseRequiredTargetId(res, req.params.targetId);
3248
+ if (!targetId) return;
3249
+ await runTabTargetMutation({
3250
+ req,
3251
+ res,
3252
+ ctx,
3253
+ targetId,
3254
+ mutate: async (profileCtx, id) => {
3255
+ await profileCtx.closeTab(id);
3256
+ }
3257
+ });
3258
+ }));
3259
+ app.post("/tabs/action", asyncBrowserRoute(async (req, res) => {
3260
+ const action = toStringOrEmpty(req.body?.action);
3261
+ const index = toNumber(req.body?.index);
3262
+ await withTabsProfileRoute({
3263
+ req,
3264
+ res,
3265
+ ctx,
3266
+ mapTabError: true,
3267
+ run: async (profileCtx) => {
3268
+ if (action === "list") {
3269
+ if (!await profileCtx.isReachable(300)) return res.json({
3270
+ ok: true,
3271
+ tabs: []
3272
+ });
3273
+ const tabs = await redactBlockedTabUrls({
3274
+ tabs: await profileCtx.listTabs(),
3275
+ ssrfPolicy: ctx.state().resolved.ssrfPolicy
3276
+ });
3277
+ return res.json({
3278
+ ok: true,
3279
+ tabs
3280
+ });
3281
+ }
3282
+ if (action === "new") {
3283
+ await profileCtx.ensureBrowserAvailable();
3284
+ const tab = await profileCtx.openTab("about:blank", { label: readOptionalTabLabel(req.body) });
3285
+ return res.json({
3286
+ ok: true,
3287
+ tab
3288
+ });
3289
+ }
3290
+ if (action === "label") {
3291
+ if (!await ensureBrowserRunning(profileCtx, res)) return;
3292
+ const targetId = parseRequiredTargetId(res, req.body?.targetId);
3293
+ if (!targetId) return;
3294
+ const label = readOptionalTabLabel(req.body);
3295
+ if (!label) return jsonError(res, 400, "label is required");
3296
+ const tab = await profileCtx.labelTab(targetId, label);
3297
+ return res.json({
3298
+ ok: true,
3299
+ tab
3300
+ });
3301
+ }
3302
+ if (action === "close") {
3303
+ if (!await ensureBrowserRunning(profileCtx, res)) return;
3304
+ const target = resolveIndexedTab(await profileCtx.listTabs(), index);
3305
+ if (!target) throw new BrowserTabNotFoundError();
3306
+ await profileCtx.closeTab(target.targetId);
3307
+ return res.json({
3308
+ ok: true,
3309
+ targetId: target.targetId
3310
+ });
3311
+ }
3312
+ if (action === "select") {
3313
+ if (typeof index !== "number") return jsonError(res, 400, "index is required");
3314
+ if (!await ensureBrowserRunning(profileCtx, res)) return;
3315
+ const target = (await profileCtx.listTabs())[index];
3316
+ if (!target) throw new BrowserTabNotFoundError();
3317
+ const ssrfPolicyOpts = browserNavigationPolicyForProfile(ctx, profileCtx);
3318
+ if (ssrfPolicyOpts.ssrfPolicy) await assertBrowserNavigationResultAllowed({
3319
+ url: target.url,
3320
+ ...ssrfPolicyOpts
3321
+ });
3322
+ await profileCtx.focusTab(target.targetId);
3323
+ return res.json({
3324
+ ok: true,
3325
+ targetId: target.targetId
3326
+ });
3327
+ }
3328
+ return jsonError(res, 400, "unknown tab action");
3329
+ }
3330
+ });
3331
+ }));
3332
+ }
3333
+ //#endregion
3334
+ //#region extensions/browser/src/browser/routes/index.ts
3335
+ function registerBrowserRoutes(app, ctx) {
3336
+ registerBrowserBasicRoutes(app, ctx);
3337
+ registerBrowserTabRoutes(app, ctx);
3338
+ registerBrowserAgentRoutes(app, ctx);
3339
+ }
3340
+ //#endregion
3341
+ export { registerBrowserRoutes as t };