atoo-studio 0.0.1 → 0.0.2

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 (408) hide show
  1. package/LICENSE +21 -0
  2. package/README.github.md +322 -0
  3. package/README.md +112 -0
  4. package/README.npm.md +112 -0
  5. package/bin/atoo-studio.js +90 -0
  6. package/dist/src/agents/claude-code-terminal/adapter.d.ts +42 -0
  7. package/dist/src/agents/claude-code-terminal/adapter.js +166 -0
  8. package/dist/src/agents/claude-code-terminal/index.d.ts +13 -0
  9. package/dist/src/agents/claude-code-terminal/index.js +45 -0
  10. package/dist/src/agents/claude-code-terminal/spawner.d.ts +9 -0
  11. package/dist/src/agents/claude-code-terminal/spawner.js +37 -0
  12. package/dist/src/agents/claude-code-terminal-chatro/adapter.d.ts +51 -0
  13. package/dist/src/agents/claude-code-terminal-chatro/adapter.js +301 -0
  14. package/dist/src/agents/claude-code-terminal-chatro/index.d.ts +13 -0
  15. package/dist/src/agents/claude-code-terminal-chatro/index.js +45 -0
  16. package/dist/src/agents/claude-code-terminal-chatro/jsonl-watcher.d.ts +67 -0
  17. package/dist/src/agents/claude-code-terminal-chatro/jsonl-watcher.js +431 -0
  18. package/dist/src/agents/claude-code-terminal-chatro/spawner.d.ts +9 -0
  19. package/dist/src/agents/claude-code-terminal-chatro/spawner.js +37 -0
  20. package/dist/src/agents/codex-terminal/adapter.d.ts +40 -0
  21. package/dist/src/agents/codex-terminal/adapter.js +160 -0
  22. package/dist/src/agents/codex-terminal/index.d.ts +13 -0
  23. package/dist/src/agents/codex-terminal/index.js +47 -0
  24. package/dist/src/agents/codex-terminal/spawner.d.ts +9 -0
  25. package/dist/src/agents/codex-terminal/spawner.js +56 -0
  26. package/dist/src/agents/codex-terminal-chatro/adapter.d.ts +58 -0
  27. package/dist/src/agents/codex-terminal-chatro/adapter.js +266 -0
  28. package/dist/src/agents/codex-terminal-chatro/index.d.ts +13 -0
  29. package/dist/src/agents/codex-terminal-chatro/index.js +50 -0
  30. package/dist/src/agents/codex-terminal-chatro/jsonl-watcher.d.ts +36 -0
  31. package/dist/src/agents/codex-terminal-chatro/jsonl-watcher.js +205 -0
  32. package/dist/src/agents/codex-terminal-chatro/spawner.d.ts +9 -0
  33. package/dist/src/agents/codex-terminal-chatro/spawner.js +57 -0
  34. package/dist/src/agents/lib/chain-builder.d.ts +21 -0
  35. package/dist/src/agents/lib/chain-builder.js +139 -0
  36. package/dist/src/agents/lib/claude/fs-sessions.d.ts +31 -0
  37. package/dist/src/agents/lib/claude/fs-sessions.js +329 -0
  38. package/dist/src/agents/lib/claude/jsonl-writer.d.ts +32 -0
  39. package/dist/src/agents/lib/claude/jsonl-writer.js +342 -0
  40. package/dist/src/agents/lib/claude/workspace-trust.d.ts +1 -0
  41. package/dist/src/agents/lib/claude/workspace-trust.js +29 -0
  42. package/dist/src/agents/lib/codex/fs-sessions.d.ts +34 -0
  43. package/dist/src/agents/lib/codex/fs-sessions.js +255 -0
  44. package/dist/src/agents/lib/codex/jsonl-mapper.d.ts +11 -0
  45. package/dist/src/agents/lib/codex/jsonl-mapper.js +154 -0
  46. package/dist/src/agents/lib/codex/jsonl-writer.d.ts +8 -0
  47. package/dist/src/agents/lib/codex/jsonl-writer.js +440 -0
  48. package/dist/src/agents/lib/fs-tracking.d.ts +36 -0
  49. package/dist/src/agents/lib/fs-tracking.js +109 -0
  50. package/dist/src/agents/lib/pty-activity-tracker.d.ts +37 -0
  51. package/dist/src/agents/lib/pty-activity-tracker.js +105 -0
  52. package/dist/src/agents/lib/session-id-utils.d.ts +46 -0
  53. package/dist/src/agents/lib/session-id-utils.js +147 -0
  54. package/dist/src/agents/lib/session-precreate.d.ts +17 -0
  55. package/dist/src/agents/lib/session-precreate.js +177 -0
  56. package/dist/src/agents/registry.d.ts +72 -0
  57. package/dist/src/agents/registry.js +337 -0
  58. package/dist/src/agents/types.d.ts +135 -0
  59. package/dist/src/agents/types.js +1 -0
  60. package/dist/src/auth/crypto-key.d.ts +6 -0
  61. package/dist/src/auth/crypto-key.js +45 -0
  62. package/dist/src/auth/middleware.d.ts +18 -0
  63. package/dist/src/auth/middleware.js +54 -0
  64. package/dist/src/auth/password.d.ts +2 -0
  65. package/dist/src/auth/password.js +12 -0
  66. package/dist/src/auth/session.d.ts +10 -0
  67. package/dist/src/auth/session.js +33 -0
  68. package/dist/src/auth/totp.d.ts +12 -0
  69. package/dist/src/auth/totp.js +61 -0
  70. package/dist/src/auth/webauthn.d.ts +6 -0
  71. package/dist/src/auth/webauthn.js +117 -0
  72. package/dist/src/config.d.ts +10 -0
  73. package/dist/src/config.js +16 -0
  74. package/dist/src/database/connection-manager.d.ts +25 -0
  75. package/dist/src/database/connection-manager.js +211 -0
  76. package/dist/src/database/discovery/container.d.ts +6 -0
  77. package/dist/src/database/discovery/container.js +226 -0
  78. package/dist/src/database/discovery/env-parser.d.ts +9 -0
  79. package/dist/src/database/discovery/env-parser.js +525 -0
  80. package/dist/src/database/discovery/local-files.d.ts +6 -0
  81. package/dist/src/database/discovery/local-files.js +58 -0
  82. package/dist/src/database/discovery/port-scan.d.ts +7 -0
  83. package/dist/src/database/discovery/port-scan.js +61 -0
  84. package/dist/src/database/drivers/cassandra.d.ts +12 -0
  85. package/dist/src/database/drivers/cassandra.js +91 -0
  86. package/dist/src/database/drivers/clickhouse.d.ts +11 -0
  87. package/dist/src/database/drivers/clickhouse.js +127 -0
  88. package/dist/src/database/drivers/elasticsearch.d.ts +12 -0
  89. package/dist/src/database/drivers/elasticsearch.js +169 -0
  90. package/dist/src/database/drivers/influxdb.d.ts +14 -0
  91. package/dist/src/database/drivers/influxdb.js +194 -0
  92. package/dist/src/database/drivers/memcached.d.ts +11 -0
  93. package/dist/src/database/drivers/memcached.js +117 -0
  94. package/dist/src/database/drivers/mongodb.d.ts +12 -0
  95. package/dist/src/database/drivers/mongodb.js +128 -0
  96. package/dist/src/database/drivers/mysql.d.ts +11 -0
  97. package/dist/src/database/drivers/mysql.js +112 -0
  98. package/dist/src/database/drivers/neo4j.d.ts +11 -0
  99. package/dist/src/database/drivers/neo4j.js +158 -0
  100. package/dist/src/database/drivers/postgresql.d.ts +11 -0
  101. package/dist/src/database/drivers/postgresql.js +133 -0
  102. package/dist/src/database/drivers/redis.d.ts +11 -0
  103. package/dist/src/database/drivers/redis.js +91 -0
  104. package/dist/src/database/drivers/sqlite.d.ts +10 -0
  105. package/dist/src/database/drivers/sqlite.js +100 -0
  106. package/dist/src/database/query-stream.d.ts +5 -0
  107. package/dist/src/database/query-stream.js +75 -0
  108. package/dist/src/database/types.d.ts +71 -0
  109. package/dist/src/database/types.js +1 -0
  110. package/dist/src/events/index.d.ts +3 -0
  111. package/dist/src/events/index.js +3 -0
  112. package/dist/src/events/types.d.ts +214 -0
  113. package/dist/src/events/types.js +22 -0
  114. package/dist/src/events/wire.d.ts +114 -0
  115. package/dist/src/events/wire.js +296 -0
  116. package/dist/src/fs-monitor-types.d.ts +24 -0
  117. package/dist/src/fs-monitor-types.js +1 -0
  118. package/dist/src/fs-monitor.d.ts +80 -0
  119. package/dist/src/fs-monitor.js +637 -0
  120. package/dist/src/handlers/auth.d.ts +1 -0
  121. package/dist/src/handlers/auth.js +170 -0
  122. package/dist/src/handlers/changes.d.ts +1 -0
  123. package/dist/src/handlers/changes.js +203 -0
  124. package/dist/src/handlers/containers.d.ts +12 -0
  125. package/dist/src/handlers/containers.js +379 -0
  126. package/dist/src/handlers/databases.d.ts +3 -0
  127. package/dist/src/handlers/databases.js +327 -0
  128. package/dist/src/handlers/environments.d.ts +3 -0
  129. package/dist/src/handlers/environments.js +286 -0
  130. package/dist/src/handlers/github.d.ts +1 -0
  131. package/dist/src/handlers/github.js +153 -0
  132. package/dist/src/handlers/projects.d.ts +1 -0
  133. package/dist/src/handlers/projects.js +895 -0
  134. package/dist/src/handlers/ssh.d.ts +1 -0
  135. package/dist/src/handlers/ssh.js +162 -0
  136. package/dist/src/handlers/users.d.ts +1 -0
  137. package/dist/src/handlers/users.js +195 -0
  138. package/dist/src/index.d.ts +1 -0
  139. package/dist/src/index.js +228 -0
  140. package/dist/src/mcp/config.d.ts +32 -0
  141. package/dist/src/mcp/config.js +227 -0
  142. package/dist/src/mcp/server.d.ts +1 -0
  143. package/dist/src/mcp/server.js +574 -0
  144. package/dist/src/serial/cuse-device.d.ts +19 -0
  145. package/dist/src/serial/cuse-device.js +260 -0
  146. package/dist/src/serial/manager.d.ts +63 -0
  147. package/dist/src/serial/manager.js +206 -0
  148. package/dist/src/serial/pty-pair.d.ts +16 -0
  149. package/dist/src/serial/pty-pair.js +68 -0
  150. package/dist/src/services/fs-browser.d.ts +14 -0
  151. package/dist/src/services/fs-browser.js +98 -0
  152. package/dist/src/services/git-ops.d.ts +78 -0
  153. package/dist/src/services/git-ops.js +288 -0
  154. package/dist/src/services/github-ops.d.ts +104 -0
  155. package/dist/src/services/github-ops.js +192 -0
  156. package/dist/src/services/obfuscation.d.ts +2 -0
  157. package/dist/src/services/obfuscation.js +16 -0
  158. package/dist/src/services/preview/headless-backend.d.ts +62 -0
  159. package/dist/src/services/preview/headless-backend.js +698 -0
  160. package/dist/src/services/preview/injected-scripts.d.ts +9 -0
  161. package/dist/src/services/preview/injected-scripts.js +232 -0
  162. package/dist/src/services/preview/preview-backend.d.ts +92 -0
  163. package/dist/src/services/preview/preview-backend.js +15 -0
  164. package/dist/src/services/preview/universal-setter.d.ts +7 -0
  165. package/dist/src/services/preview/universal-setter.js +46 -0
  166. package/dist/src/services/preview-manager.d.ts +50 -0
  167. package/dist/src/services/preview-manager.js +216 -0
  168. package/dist/src/services/project-watcher.d.ts +6 -0
  169. package/dist/src/services/project-watcher.js +307 -0
  170. package/dist/src/services/remote-fs-browser.d.ts +11 -0
  171. package/dist/src/services/remote-fs-browser.js +50 -0
  172. package/dist/src/services/remote-git-ops.d.ts +71 -0
  173. package/dist/src/services/remote-git-ops.js +215 -0
  174. package/dist/src/services/session-search.d.ts +56 -0
  175. package/dist/src/services/session-search.js +303 -0
  176. package/dist/src/services/ssh-manager.d.ts +44 -0
  177. package/dist/src/services/ssh-manager.js +359 -0
  178. package/dist/src/session-writer.d.ts +9 -0
  179. package/dist/src/session-writer.js +66 -0
  180. package/dist/src/spawner.d.ts +56 -0
  181. package/dist/src/spawner.js +135 -0
  182. package/dist/src/state/db.d.ts +214 -0
  183. package/dist/src/state/db.js +897 -0
  184. package/dist/src/state/store.d.ts +37 -0
  185. package/dist/src/state/store.js +108 -0
  186. package/dist/src/state/types.d.ts +13 -0
  187. package/dist/src/state/types.js +1 -0
  188. package/dist/src/web/devtools-proxy.d.ts +7 -0
  189. package/dist/src/web/devtools-proxy.js +176 -0
  190. package/dist/src/web/port-proxy.d.ts +15 -0
  191. package/dist/src/web/port-proxy.js +124 -0
  192. package/dist/src/web/preview-ws.d.ts +5 -0
  193. package/dist/src/web/preview-ws.js +207 -0
  194. package/dist/src/web/server.d.ts +6 -0
  195. package/dist/src/web/server.js +1694 -0
  196. package/dist/src/ws/agent-ws.d.ts +5 -0
  197. package/dist/src/ws/agent-ws.js +93 -0
  198. package/frontend/dist/assets/_basePickBy-B-LibQ4-.js +1 -0
  199. package/frontend/dist/assets/_baseUniq-CprifHap.js +1 -0
  200. package/frontend/dist/assets/_createAssigner-ByDUqGii.js +1 -0
  201. package/frontend/dist/assets/abap-DuT-3z4x.js +1 -0
  202. package/frontend/dist/assets/addon-fit-CxQet2ja.js +1 -0
  203. package/frontend/dist/assets/addon-web-links-D_jRkPIl.js +1 -0
  204. package/frontend/dist/assets/apex-B-em86xX.js +1 -0
  205. package/frontend/dist/assets/api-SUPuHhSY.js +2 -0
  206. package/frontend/dist/assets/arc-Z0_eVteO.js +1 -0
  207. package/frontend/dist/assets/architecture-PBZL5I3N-hvVXGhqd.js +1 -0
  208. package/frontend/dist/assets/architectureDiagram-2XIMDMQ5-DiHPxX4j.js +36 -0
  209. package/frontend/dist/assets/array-CwG8vNfn.js +1 -0
  210. package/frontend/dist/assets/auth-store-R7eW5SVu.js +1 -0
  211. package/frontend/dist/assets/azcli-Bg9wQloi.js +1 -0
  212. package/frontend/dist/assets/bat-BM46z99L.js +1 -0
  213. package/frontend/dist/assets/bicep-DcBsJUfh.js +2 -0
  214. package/frontend/dist/assets/blockDiagram-WCTKOSBZ-C40u_hLo.js +132 -0
  215. package/frontend/dist/assets/c4Diagram-IC4MRINW-Ct7LjWFQ.js +10 -0
  216. package/frontend/dist/assets/cameligo-zw7JTtim.js +1 -0
  217. package/frontend/dist/assets/channel-ClCsE6HN.js +1 -0
  218. package/frontend/dist/assets/chunk-4BX2VUAB-zZ6P90VO.js +1 -0
  219. package/frontend/dist/assets/chunk-55IACEB6-DXllTDQl.js +1 -0
  220. package/frontend/dist/assets/chunk-7E7YKBS2-7zRaOLjj.js +1 -0
  221. package/frontend/dist/assets/chunk-7R4GIKGN-Csst1274.js +80 -0
  222. package/frontend/dist/assets/chunk-C72U2L5F-_JbQPbLN.js +1 -0
  223. package/frontend/dist/assets/chunk-CFjPhJqf.js +1 -0
  224. package/frontend/dist/assets/chunk-EGIJ26TM-B--aFyPw.js +1 -0
  225. package/frontend/dist/assets/chunk-FMBD7UC4-DVR34RNb.js +15 -0
  226. package/frontend/dist/assets/chunk-GEFDOKGD-CnmN6cC8.js +2 -0
  227. package/frontend/dist/assets/chunk-JSJVCQXG-CWxHBzeJ.js +1 -0
  228. package/frontend/dist/assets/chunk-KX2RTZJC-DkRk56s7.js +1 -0
  229. package/frontend/dist/assets/chunk-KYZI473N-DCCsG2dK.js +53 -0
  230. package/frontend/dist/assets/chunk-L3YUKLVL-C-DkZTMr.js +1 -0
  231. package/frontend/dist/assets/chunk-MX3YWQON-OUdzv5sZ.js +1 -0
  232. package/frontend/dist/assets/chunk-NQ4KR5QH-Bpu9FsM7.js +220 -0
  233. package/frontend/dist/assets/chunk-O4XLMI2P-BMLK6_ib.js +7 -0
  234. package/frontend/dist/assets/chunk-OZEHJAEY-CNNiJtG0.js +1 -0
  235. package/frontend/dist/assets/chunk-PQ6SQG4A-evVHD3KM.js +1 -0
  236. package/frontend/dist/assets/chunk-PU5JKC2W-DPFTYuvl.js +70 -0
  237. package/frontend/dist/assets/chunk-QZHKN3VN-JRdddPvu.js +1 -0
  238. package/frontend/dist/assets/chunk-R5LLSJPH-CHQzVVOV.js +1 -0
  239. package/frontend/dist/assets/chunk-WL4C6EOR-BNFU6IIi.js +189 -0
  240. package/frontend/dist/assets/chunk-XIRO2GV7-98T93G85.js +1 -0
  241. package/frontend/dist/assets/chunk-XZSTWKYB-BcW3cyNp.js +94 -0
  242. package/frontend/dist/assets/chunk-YBOYWFTD-BgKO1qAJ.js +1 -0
  243. package/frontend/dist/assets/classDiagram-VBA2DB6C-DikXzgcD.js +1 -0
  244. package/frontend/dist/assets/classDiagram-v2-RAHNMMFH-D7E3tQUK.js +1 -0
  245. package/frontend/dist/assets/clojure-FspFoNNQ.js +1 -0
  246. package/frontend/dist/assets/clone-mOXuZa7C.js +1 -0
  247. package/frontend/dist/assets/codicon-ngg6Pgfi.ttf +0 -0
  248. package/frontend/dist/assets/coffee-13n8Bk2W.js +1 -0
  249. package/frontend/dist/assets/cose-bilkent-S5V4N54A-zUOWQqLe.js +1 -0
  250. package/frontend/dist/assets/cpp-BVm2xGEs.js +1 -0
  251. package/frontend/dist/assets/csharp-D2kAWmUm.js +1 -0
  252. package/frontend/dist/assets/csp-Ezvgpf0e.js +1 -0
  253. package/frontend/dist/assets/css-CYxRwcFy.js +3 -0
  254. package/frontend/dist/assets/css.worker-Cd5h-ZOL.js +89 -0
  255. package/frontend/dist/assets/cssMode-CrXej49V.js +1 -0
  256. package/frontend/dist/assets/cypher-jg3SGErc.js +1 -0
  257. package/frontend/dist/assets/cytoscape.esm-kyyvzxNV.js +321 -0
  258. package/frontend/dist/assets/dagre-DH4bgZO7.js +1 -0
  259. package/frontend/dist/assets/dagre-KLK3FWXG-DNSqDkwT.js +4 -0
  260. package/frontend/dist/assets/dart-179jqhK4.js +1 -0
  261. package/frontend/dist/assets/defaultLocale-Dda4OpKy.js +1 -0
  262. package/frontend/dist/assets/diagram-E7M64L7V-RqPNT5Vs.js +24 -0
  263. package/frontend/dist/assets/diagram-IFDJBPK2-B-5NRyaE.js +43 -0
  264. package/frontend/dist/assets/diagram-P4PSJMXO-BrP69Hk0.js +24 -0
  265. package/frontend/dist/assets/dist-CU_Nb1G5.js +1 -0
  266. package/frontend/dist/assets/dockerfile-CIAtSGxS.js +1 -0
  267. package/frontend/dist/assets/ecl-CGVKfDxD.js +1 -0
  268. package/frontend/dist/assets/editor-Br_kD0ds.css +1 -0
  269. package/frontend/dist/assets/editor.api2-YXkDn0Gm.js +872 -0
  270. package/frontend/dist/assets/editor.main-fBaXZjJ0.js +6 -0
  271. package/frontend/dist/assets/elixir-BZ-6w0y3.js +1 -0
  272. package/frontend/dist/assets/erDiagram-INFDFZHY-BYiB9NYg.js +70 -0
  273. package/frontend/dist/assets/flow9-CVuOjTMv.js +1 -0
  274. package/frontend/dist/assets/flowDiagram-PKNHOUZH-Cwq47rsR.js +162 -0
  275. package/frontend/dist/assets/freemarker2-DM-pztJU.js +3 -0
  276. package/frontend/dist/assets/fsharp-q0pGJYr6.js +1 -0
  277. package/frontend/dist/assets/ganttDiagram-A5KZAMGK-Dnx3szD9.js +292 -0
  278. package/frontend/dist/assets/gitGraph-HDMCJU4V-COlTQ7bA.js +1 -0
  279. package/frontend/dist/assets/gitGraphDiagram-K3NZZRJ6-BaUxboNc.js +65 -0
  280. package/frontend/dist/assets/go-dzSPfdEO.js +1 -0
  281. package/frontend/dist/assets/graphlib-kEFlkt3U.js +1 -0
  282. package/frontend/dist/assets/graphql-CG4OUoEV.js +1 -0
  283. package/frontend/dist/assets/handlebars-BbK53Vec.js +1 -0
  284. package/frontend/dist/assets/hcl-Cy14JPk3.js +1 -0
  285. package/frontend/dist/assets/html-DYtTQNOG.js +1 -0
  286. package/frontend/dist/assets/html.worker-BjVEKLoU.js +502 -0
  287. package/frontend/dist/assets/htmlMode-C6GTouth.js +1 -0
  288. package/frontend/dist/assets/index-DMLxes_u.js +157 -0
  289. package/frontend/dist/assets/index-DmzeqkB1.css +1 -0
  290. package/frontend/dist/assets/info-3K5VOQVL-DBtHyA4C.js +1 -0
  291. package/frontend/dist/assets/infoDiagram-LFFYTUFH-yBXLgMPI.js +2 -0
  292. package/frontend/dist/assets/ini-Pbg8HGVD.js +1 -0
  293. package/frontend/dist/assets/init-D6KNwrax.js +1 -0
  294. package/frontend/dist/assets/ishikawaDiagram-PHBUUO56-Bld4two_.js +70 -0
  295. package/frontend/dist/assets/java-BmVu6Qrl.js +1 -0
  296. package/frontend/dist/assets/javascript-PbfQEdcJ.js +1 -0
  297. package/frontend/dist/assets/journeyDiagram-4ABVD52K-4HyMd4R2.js +139 -0
  298. package/frontend/dist/assets/json.worker-DqU5Wxnl.js +58 -0
  299. package/frontend/dist/assets/jsonMode-CASsGppE.js +7 -0
  300. package/frontend/dist/assets/julia-3cGnieBq.js +1 -0
  301. package/frontend/dist/assets/kanban-definition-K7BYSVSG-DpgsZmpG.js +89 -0
  302. package/frontend/dist/assets/katex-CEw3x5bf.js +261 -0
  303. package/frontend/dist/assets/kotlin-BuWkVcfV.js +1 -0
  304. package/frontend/dist/assets/less-CJ_VPy2C.js +2 -0
  305. package/frontend/dist/assets/lexon-BygAuZPu.js +1 -0
  306. package/frontend/dist/assets/line-CA_wh_TY.js +1 -0
  307. package/frontend/dist/assets/linear-BAcLW45z.js +1 -0
  308. package/frontend/dist/assets/liquid-kz84dle6.js +1 -0
  309. package/frontend/dist/assets/lspLanguageFeatures-C7hAHFn1.js +4 -0
  310. package/frontend/dist/assets/lua-C8Xs3dCx.js +1 -0
  311. package/frontend/dist/assets/m3-DTJeKBk4.js +1 -0
  312. package/frontend/dist/assets/markdown-QCgx8JqZ.js +1 -0
  313. package/frontend/dist/assets/math-D0YcMJAn.js +1 -0
  314. package/frontend/dist/assets/mdx-yRw0ap-E.js +1 -0
  315. package/frontend/dist/assets/mermaid-parser.core-DAeTodBQ.js +4 -0
  316. package/frontend/dist/assets/mindmap-definition-YRQLILUH-CoNlFyVl.js +68 -0
  317. package/frontend/dist/assets/mips-DopWaYgE.js +1 -0
  318. package/frontend/dist/assets/monaco.contribution-DeY0Qei-.js +2 -0
  319. package/frontend/dist/assets/msdax-BDis4ARV.js +1 -0
  320. package/frontend/dist/assets/mysql-BV6MLsOI.js +1 -0
  321. package/frontend/dist/assets/objective-c-B1UuzKs6.js +1 -0
  322. package/frontend/dist/assets/ordinal-jM7S0YHN.js +1 -0
  323. package/frontend/dist/assets/packet-RMMSAZCW-FF6-Tmai.js +1 -0
  324. package/frontend/dist/assets/pascal-BkvESCrc.js +1 -0
  325. package/frontend/dist/assets/pascaligo-lTy0kZYr.js +1 -0
  326. package/frontend/dist/assets/path-DNPd7Py7.js +1 -0
  327. package/frontend/dist/assets/perl-CrtUPXLV.js +1 -0
  328. package/frontend/dist/assets/pgsql-B9IbNWx2.js +1 -0
  329. package/frontend/dist/assets/php-CXvQBY2p.js +1 -0
  330. package/frontend/dist/assets/pie-UPGHQEXC-CFvXY2o-.js +1 -0
  331. package/frontend/dist/assets/pieDiagram-SKSYHLDU-CM_hbCcn.js +30 -0
  332. package/frontend/dist/assets/pla-DxBxuqWu.js +1 -0
  333. package/frontend/dist/assets/postiats-OkEuT5YF.js +1 -0
  334. package/frontend/dist/assets/powerquery-CMx5Tq4K.js +1 -0
  335. package/frontend/dist/assets/powershell-CstRxrEc.js +1 -0
  336. package/frontend/dist/assets/preload-helper-D4M6sveU.js +1 -0
  337. package/frontend/dist/assets/protobuf-Bx0Z-uRj.js +2 -0
  338. package/frontend/dist/assets/pug--W8vanWl.js +1 -0
  339. package/frontend/dist/assets/python-DA0rnlw3.js +1 -0
  340. package/frontend/dist/assets/qsharp-CRtr0YbN.js +1 -0
  341. package/frontend/dist/assets/quadrantDiagram-337W2JSQ-B3n3IUhC.js +7 -0
  342. package/frontend/dist/assets/r-C6E1d6iv.js +1 -0
  343. package/frontend/dist/assets/radar-KQ55EAFF-MPZu7SdX.js +1 -0
  344. package/frontend/dist/assets/razor-yd73uata.js +1 -0
  345. package/frontend/dist/assets/redis-Dx13voP3.js +1 -0
  346. package/frontend/dist/assets/redshift-D66HwlyV.js +1 -0
  347. package/frontend/dist/assets/requirementDiagram-Z7DCOOCP-CorP7L7F.js +73 -0
  348. package/frontend/dist/assets/restructuredtext-DQT2NKJ2.js +1 -0
  349. package/frontend/dist/assets/rough.esm-DxAX5Vpo.js +1 -0
  350. package/frontend/dist/assets/ruby-iFXI8hwH.js +1 -0
  351. package/frontend/dist/assets/rust-CSKiei34.js +1 -0
  352. package/frontend/dist/assets/sankeyDiagram-WA2Y5GQK-RDx6Bd-B.js +10 -0
  353. package/frontend/dist/assets/sb-Bo3ttdP2.js +1 -0
  354. package/frontend/dist/assets/scala-BC1D-Nxp.js +1 -0
  355. package/frontend/dist/assets/scheme-Z4OAo4Lv.js +1 -0
  356. package/frontend/dist/assets/scss-BvrdPs6B.js +3 -0
  357. package/frontend/dist/assets/sequenceDiagram-2WXFIKYE-JMqJSFq6.js +145 -0
  358. package/frontend/dist/assets/shell-Bh_aCyF-.js +1 -0
  359. package/frontend/dist/assets/solidity-CWHj6tSe.js +1 -0
  360. package/frontend/dist/assets/sophia-raoNtKtm.js +1 -0
  361. package/frontend/dist/assets/sparql-XzmoGnue.js +1 -0
  362. package/frontend/dist/assets/sql-BD0i9Gvg.js +1 -0
  363. package/frontend/dist/assets/src-Bn-kKzs7.js +1 -0
  364. package/frontend/dist/assets/st-DtVKyms6.js +1 -0
  365. package/frontend/dist/assets/stateDiagram-RAJIS63D-CgFfENdy.js +1 -0
  366. package/frontend/dist/assets/stateDiagram-v2-FVOUBMTO-C4Hh2P-U.js +1 -0
  367. package/frontend/dist/assets/swift--UZs77wT.js +1 -0
  368. package/frontend/dist/assets/systemverilog-CDnBSWUd.js +1 -0
  369. package/frontend/dist/assets/tcl-DdCEuTHZ.js +1 -0
  370. package/frontend/dist/assets/timeline-definition-YZTLITO2-BnatPBR5.js +61 -0
  371. package/frontend/dist/assets/treemap-KZPCXAKY-qb1Pl9la.js +1 -0
  372. package/frontend/dist/assets/ts.worker-DyPAEIuH.js +67719 -0
  373. package/frontend/dist/assets/tsMode-iuvyEpyO.js +11 -0
  374. package/frontend/dist/assets/twig-SSL-Altf.js +1 -0
  375. package/frontend/dist/assets/typescript-17918Hud.js +1 -0
  376. package/frontend/dist/assets/typespec-BT7S0ETg.js +1 -0
  377. package/frontend/dist/assets/vb-CrIgucua.js +1 -0
  378. package/frontend/dist/assets/vennDiagram-LZ73GAT5-DygS4Zzd.js +34 -0
  379. package/frontend/dist/assets/wgsl-BeKc3oEp.js +298 -0
  380. package/frontend/dist/assets/workers-DTfwKVoM.js +1 -0
  381. package/frontend/dist/assets/xml-CBMr_Wbw.js +1 -0
  382. package/frontend/dist/assets/xterm-BrP-ENHg.css +1 -0
  383. package/frontend/dist/assets/xterm-CBX2m0YM.js +36 -0
  384. package/frontend/dist/assets/xychartDiagram-JWTSCODW-D6wY1Jwd.js +7 -0
  385. package/frontend/dist/assets/yaml-CTjCH7Bv.js +1 -0
  386. package/frontend/dist/fonts/inter-300.ttf +0 -0
  387. package/frontend/dist/fonts/inter-400.ttf +0 -0
  388. package/frontend/dist/fonts/inter-500.ttf +0 -0
  389. package/frontend/dist/fonts/inter-600.ttf +0 -0
  390. package/frontend/dist/fonts/inter-700.ttf +0 -0
  391. package/frontend/dist/index.html +49 -0
  392. package/frontend/dist/logo_192x192.png +0 -0
  393. package/frontend/dist/logo_32x32.png +0 -0
  394. package/frontend/dist/logo_512x512.png +0 -0
  395. package/frontend/dist/logo_64x64.png +0 -0
  396. package/frontend/dist/logobg_192x192.png +0 -0
  397. package/frontend/dist/logobg_512x512.png +0 -0
  398. package/frontend/dist/logobg_64x64.png +0 -0
  399. package/frontend/dist/manifest.json +25 -0
  400. package/frontend/dist/sw.js +22 -0
  401. package/package.json +74 -7
  402. package/preload/Makefile +12 -0
  403. package/preload/atoo-studio-preload.c +647 -0
  404. package/preload/atoo-studio-preload.so +0 -0
  405. package/setup-cuse.sh +260 -0
  406. package/setup.sh +81 -0
  407. package/src/serial/native/binding.gyp +10 -0
  408. package/src/serial/native/pty_pair.c +222 -0
@@ -0,0 +1,698 @@
1
+ import puppeteer from 'puppeteer';
2
+ import { spawn, execSync } from 'child_process';
3
+ import os from 'os';
4
+ import path from 'path';
5
+ import fs from 'fs';
6
+ import net from 'net';
7
+ import { CDP_PORT_START, CDP_PORT_END } from '../../config.js';
8
+ import { broadcastJson, broadcastBinary, } from './preview-backend.js';
9
+ import { getInjectedScript } from './injected-scripts.js';
10
+ import { buildUniversalSetterExpression } from './universal-setter.js';
11
+ const CHROME_PATH = '/home/furti/.cache/puppeteer/chrome/linux-146.0.7680.31/chrome-linux64/chrome';
12
+ let FFMPEG_PATH;
13
+ try {
14
+ FFMPEG_PATH = execSync('which ffmpeg', { encoding: 'utf8', stdio: ['ignore', 'pipe', 'ignore'] }).trim();
15
+ }
16
+ catch {
17
+ FFMPEG_PATH = '/usr/bin/ffmpeg';
18
+ }
19
+ const DOWNLOAD_DIR = path.join(os.tmpdir(), 'atoo-studio-downloads');
20
+ export class HeadlessBackend {
21
+ mode = 'headless';
22
+ usedPorts = new Set();
23
+ cleanupOrphans() {
24
+ for (let port = CDP_PORT_START; port <= CDP_PORT_END; port++) {
25
+ try {
26
+ const result = execSync(`lsof -ti tcp:${port} -s tcp:listen 2>/dev/null || true`, { encoding: 'utf-8' }).trim();
27
+ if (result) {
28
+ for (const pid of result.split('\n').filter(Boolean)) {
29
+ console.log(`[preview/headless] Killing orphaned process on CDP port ${port} (PID ${pid})`);
30
+ try {
31
+ process.kill(parseInt(pid, 10), 'SIGKILL');
32
+ }
33
+ catch { }
34
+ }
35
+ }
36
+ }
37
+ catch { }
38
+ }
39
+ }
40
+ async allocateCdpPort() {
41
+ for (let port = CDP_PORT_START; port <= CDP_PORT_END; port++) {
42
+ if (!this.usedPorts.has(port) && await this.isPortFree(port)) {
43
+ this.usedPorts.add(port);
44
+ return port;
45
+ }
46
+ }
47
+ throw new Error('No available CDP ports');
48
+ }
49
+ isPortFree(port) {
50
+ return new Promise((resolve) => {
51
+ const server = net.createServer();
52
+ server.once('error', () => resolve(false));
53
+ server.once('listening', () => { server.close(() => resolve(true)); });
54
+ server.listen(port, '127.0.0.1');
55
+ });
56
+ }
57
+ releaseCdpPort(port) {
58
+ this.usedPorts.delete(port);
59
+ }
60
+ // --- Screencast ---
61
+ async startScreencast(instance) {
62
+ if (instance.screencastActive)
63
+ return;
64
+ instance.cdpSession.removeAllListeners('Page.screencastFrame');
65
+ const usePng = instance.quality >= 100;
66
+ instance.cdpSession.on('Page.screencastFrame', (frame) => {
67
+ instance.cdpSession.send('Page.screencastFrameAck', { sessionId: frame.sessionId }).catch(() => { });
68
+ instance.lastActivity = Date.now();
69
+ instance.lastFrameTime = Date.now();
70
+ instance.lastStreamRestart = 0; // reset backoff on successful frame
71
+ const imgBuf = Buffer.from(frame.data, 'base64');
72
+ const msg = Buffer.allocUnsafe(1 + imgBuf.length);
73
+ msg[0] = usePng ? 0x02 : 0x01;
74
+ imgBuf.copy(msg, 1);
75
+ broadcastBinary(instance, msg);
76
+ if (instance.recording) {
77
+ instance.recordedFrames.push({ data: imgBuf, timestamp: Date.now() });
78
+ }
79
+ });
80
+ await instance.cdpSession.send('Page.startScreencast', {
81
+ format: usePng ? 'png' : 'jpeg',
82
+ quality: usePng ? undefined : instance.quality,
83
+ maxWidth: instance.viewport.width,
84
+ maxHeight: instance.viewport.height,
85
+ everyNthFrame: 1,
86
+ });
87
+ instance.screencastActive = true;
88
+ instance.lastFrameTime = Date.now();
89
+ }
90
+ async stopScreencast(instance) {
91
+ if (!instance.screencastActive)
92
+ return;
93
+ try {
94
+ await instance.cdpSession.send('Page.stopScreencast');
95
+ }
96
+ catch { }
97
+ instance.cdpSession.removeAllListeners('Page.screencastFrame');
98
+ instance.screencastActive = false;
99
+ }
100
+ // --- CDP interception ---
101
+ async setupInterception(instance) {
102
+ const { cdpSession, browserCdpSession } = instance;
103
+ // 1. JavaScript dialogs
104
+ cdpSession.on('Page.javascriptDialogOpening', (params) => {
105
+ const dialogId = `${Date.now()}-${Math.random().toString(36).slice(2)}`;
106
+ let responded = false;
107
+ instance.pendingDialogs.set(dialogId, {
108
+ resolve: (accept, promptText) => {
109
+ if (responded)
110
+ return;
111
+ responded = true;
112
+ instance.pendingDialogs.delete(dialogId);
113
+ cdpSession.send('Page.handleJavaScriptDialog', { accept, promptText }).catch(() => { });
114
+ },
115
+ timeout: setTimeout(() => {
116
+ if (!responded) {
117
+ responded = true;
118
+ instance.pendingDialogs.delete(dialogId);
119
+ cdpSession.send('Page.handleJavaScriptDialog', { accept: true }).catch(() => { });
120
+ broadcastJson(instance, { type: 'dialog_closed', dialogId, timedOut: true });
121
+ }
122
+ }, 30000),
123
+ });
124
+ broadcastJson(instance, {
125
+ type: 'dialog_opened',
126
+ dialogId,
127
+ dialogType: params.type,
128
+ message: params.message,
129
+ defaultPrompt: params.defaultPrompt,
130
+ url: params.url,
131
+ });
132
+ });
133
+ // 2. File chooser interception
134
+ try {
135
+ await cdpSession.send('Page.setInterceptFileChooserDialog', { enabled: true });
136
+ }
137
+ catch (err) {
138
+ console.warn(`[preview/headless] File chooser interception not supported: ${err.message}`);
139
+ }
140
+ cdpSession.on('Page.fileChooserOpened', (params) => {
141
+ broadcastJson(instance, {
142
+ type: 'file_chooser_opened',
143
+ mode: params.mode,
144
+ frameId: params.frameId,
145
+ backendNodeId: params.backendNodeId,
146
+ });
147
+ });
148
+ // 3. Download interception
149
+ const dlDir = path.join(DOWNLOAD_DIR, instance.id.replace('/', '_'));
150
+ instance.downloadDir = dlDir;
151
+ fs.mkdirSync(dlDir, { recursive: true });
152
+ try {
153
+ await browserCdpSession.send('Browser.setDownloadBehavior', {
154
+ behavior: 'allowAndName',
155
+ downloadPath: dlDir,
156
+ eventsEnabled: true,
157
+ });
158
+ }
159
+ catch (err) {
160
+ console.warn(`[preview/headless] Download interception not supported: ${err.message}`);
161
+ }
162
+ browserCdpSession.on('Browser.downloadWillBegin', (params) => {
163
+ broadcastJson(instance, {
164
+ type: 'download_started',
165
+ guid: params.guid,
166
+ suggestedFilename: params.suggestedFilename,
167
+ url: params.url,
168
+ });
169
+ });
170
+ browserCdpSession.on('Browser.downloadProgress', (params) => {
171
+ if (params.state === 'completed') {
172
+ broadcastJson(instance, { type: 'download_complete', guid: params.guid });
173
+ }
174
+ });
175
+ // 4. New tab/popup interception
176
+ browserCdpSession.on('Target.targetCreated', (params) => {
177
+ const { targetInfo } = params;
178
+ if (targetInfo.type === 'page' && targetInfo.openerId) {
179
+ broadcastJson(instance, {
180
+ type: 'new_tab',
181
+ url: targetInfo.url,
182
+ targetId: targetInfo.targetId,
183
+ });
184
+ }
185
+ });
186
+ try {
187
+ await browserCdpSession.send('Target.setDiscoverTargets', { discover: true });
188
+ }
189
+ catch (err) {
190
+ console.warn(`[preview/headless] Target discovery not supported: ${err.message}`);
191
+ }
192
+ }
193
+ // --- Shadow Overlay: Script injection & bindings ---
194
+ async setupInjectedScripts(instance) {
195
+ const { page, cdpSession } = instance;
196
+ // Use Puppeteer's exposeFunction for reliable binding→server communication
197
+ // (raw Runtime.addBinding + Runtime.bindingCalled has CDP session routing issues)
198
+ const expose = async (name, handler) => {
199
+ try {
200
+ await page.exposeFunction(name, handler);
201
+ }
202
+ catch (err) {
203
+ console.warn(`[preview/headless] Failed to expose ${name}: ${err.message}`);
204
+ }
205
+ };
206
+ await expose('__atoo_selectOpened', (payload) => {
207
+ try {
208
+ const data = JSON.parse(payload);
209
+ broadcastJson(instance, { type: 'select_opened', ...data });
210
+ }
211
+ catch { }
212
+ });
213
+ await expose('__atoo_pickerOpened', (payload) => {
214
+ try {
215
+ const data = JSON.parse(payload);
216
+ // Rename data.type → inputType to avoid overwriting the message 'type' field
217
+ const { type: inputType, ...rest } = data;
218
+ broadcastJson(instance, { type: 'picker_opened', inputType, ...rest });
219
+ }
220
+ catch { }
221
+ });
222
+ await expose('__atoo_tooltipShow', (payload) => {
223
+ try {
224
+ const data = JSON.parse(payload);
225
+ broadcastJson(instance, { type: 'tooltip_show', ...data });
226
+ }
227
+ catch { }
228
+ });
229
+ await expose('__atoo_tooltipHide', (_payload) => {
230
+ broadcastJson(instance, { type: 'tooltip_hide' });
231
+ });
232
+ await expose('__atoo_contextMenu', (payload) => {
233
+ try {
234
+ const data = JSON.parse(payload);
235
+ broadcastJson(instance, { type: 'context_menu', ...data });
236
+ }
237
+ catch { }
238
+ });
239
+ await expose('__atoo_clipboard', (payload) => {
240
+ try {
241
+ const data = JSON.parse(payload);
242
+ broadcastJson(instance, { type: 'clipboard', text: data.text });
243
+ }
244
+ catch { }
245
+ });
246
+ // Inject the capture script — persists across navigations
247
+ try {
248
+ await cdpSession.send('Page.addScriptToEvaluateOnNewDocument', {
249
+ source: getInjectedScript(),
250
+ });
251
+ }
252
+ catch (err) {
253
+ console.warn(`[preview/headless] Failed to inject capture script: ${err.message}`);
254
+ }
255
+ }
256
+ // --- Auth interception via Fetch domain ---
257
+ async setupAuthInterception(instance) {
258
+ const { cdpSession } = instance;
259
+ try {
260
+ // Only intercept auth challenges — use a pattern that won't match any URL
261
+ // to avoid pausing normal requests (empty patterns[] = intercept everything)
262
+ await cdpSession.send('Fetch.enable', {
263
+ handleAuthRequests: true,
264
+ patterns: [{ urlPattern: '__atoo_never_match__' }],
265
+ });
266
+ }
267
+ catch (err) {
268
+ console.warn(`[preview/headless] Fetch/auth interception not supported: ${err.message}`);
269
+ return;
270
+ }
271
+ cdpSession.on('Fetch.authRequired', (params) => {
272
+ const requestId = params.requestId;
273
+ let responded = false;
274
+ const timeout = setTimeout(() => {
275
+ if (!responded) {
276
+ responded = true;
277
+ instance.pendingAuthRequests.delete(requestId);
278
+ cdpSession.send('Fetch.continueWithAuth', {
279
+ requestId,
280
+ authChallengeResponse: { response: 'CancelAuth' },
281
+ }).catch(() => { });
282
+ broadcastJson(instance, { type: 'auth_cancelled', requestId, timedOut: true });
283
+ }
284
+ }, 60000);
285
+ instance.pendingAuthRequests.set(requestId, { requestId, timeout });
286
+ broadcastJson(instance, {
287
+ type: 'auth_required',
288
+ requestId,
289
+ url: params.request?.url || '',
290
+ realm: params.authChallenge?.realm || '',
291
+ scheme: params.authChallenge?.scheme || '',
292
+ });
293
+ });
294
+ }
295
+ // --- Public API ---
296
+ async create(key, opts) {
297
+ const cdpPort = await this.allocateCdpPort();
298
+ const width = opts.width || 1920;
299
+ const height = opts.height || 1080;
300
+ const dpr = opts.dpr || 1;
301
+ const isMobile = opts.isMobile || false;
302
+ const hasTouch = opts.hasTouch || false;
303
+ const quality = opts.quality || 80;
304
+ const protocol = opts.protocol || 'http';
305
+ try {
306
+ const chromeArgs = [
307
+ `--remote-debugging-port=${cdpPort}`,
308
+ '--no-sandbox',
309
+ '--disable-gpu',
310
+ '--disable-dev-shm-usage',
311
+ '--ignore-certificate-errors',
312
+ '--disable-background-timer-throttling',
313
+ '--disable-backgrounding-occluded-windows',
314
+ '--disable-renderer-backgrounding',
315
+ // Permission auto-grant
316
+ '--use-fake-device-for-media-stream',
317
+ '--auto-accept-camera-and-microphone-capture',
318
+ ];
319
+ if (opts.headerHost) {
320
+ const defaultPort = protocol === 'https' ? 443 : 80;
321
+ chromeArgs.push(`--host-resolver-rules=MAP ${opts.headerHost}:${defaultPort} 127.0.0.1:${opts.targetPort}, MAP ${opts.headerHost} 127.0.0.1`);
322
+ }
323
+ const browser = await puppeteer.launch({
324
+ executablePath: CHROME_PATH,
325
+ headless: true,
326
+ args: chromeArgs,
327
+ });
328
+ const page = await browser.newPage();
329
+ await page.setViewport({ width, height, deviceScaleFactor: dpr, isMobile, hasTouch });
330
+ const cdpSession = await page.createCDPSession();
331
+ const browserCdpSession = await browser.target().createCDPSession();
332
+ const instance = {
333
+ id: key,
334
+ browser,
335
+ page,
336
+ cdpSession,
337
+ browserCdpSession,
338
+ cdpPort,
339
+ targetPort: opts.targetPort,
340
+ headerHost: opts.headerHost,
341
+ protocol,
342
+ viewport: { width, height, deviceScaleFactor: dpr, isMobile, hasTouch },
343
+ quality,
344
+ screencastActive: false,
345
+ wsClients: new Set(),
346
+ lastActivity: Date.now(),
347
+ recording: false,
348
+ recordedFrames: [],
349
+ lastFrameTime: Date.now(),
350
+ lastStreamRestart: 0,
351
+ pendingDialogs: new Map(),
352
+ pendingAuthRequests: new Map(),
353
+ downloadDir: '',
354
+ };
355
+ await cdpSession.send('Page.enable');
356
+ // URL change tracking
357
+ cdpSession.on('Page.frameNavigated', (params) => {
358
+ if (params.frame?.parentId)
359
+ return;
360
+ const url = params.frame?.url;
361
+ if (url) {
362
+ broadcastJson(instance, { type: 'url_changed', url });
363
+ }
364
+ });
365
+ await this.setupInterception(instance);
366
+ await this.setupInjectedScripts(instance);
367
+ await this.setupAuthInterception(instance);
368
+ // Grant permissions at browser level
369
+ try {
370
+ await browserCdpSession.send('Browser.grantPermissions', {
371
+ permissions: ['geolocation', 'notifications', 'midi', 'audioCapture', 'videoCapture'],
372
+ });
373
+ }
374
+ catch (err) {
375
+ console.warn(`[preview/headless] Permission grant failed: ${err.message}`);
376
+ }
377
+ // Navigate with retries
378
+ const targetUrl = opts.headerHost
379
+ ? `${protocol}://${opts.headerHost}`
380
+ : `${protocol}://localhost:${opts.targetPort}`;
381
+ for (let attempt = 1; attempt <= 5; attempt++) {
382
+ try {
383
+ await page.goto(targetUrl, { waitUntil: 'domcontentloaded', timeout: 10000 });
384
+ break;
385
+ }
386
+ catch (err) {
387
+ console.warn(`[preview/headless] Navigation to ${targetUrl} attempt ${attempt}/5: ${err.message}`);
388
+ if (attempt < 5)
389
+ await new Promise(r => setTimeout(r, 1000));
390
+ }
391
+ }
392
+ // Also evaluate the injected script immediately in the current page
393
+ // (addScriptToEvaluateOnNewDocument only runs on future navigations)
394
+ try {
395
+ await cdpSession.send('Runtime.evaluate', {
396
+ expression: getInjectedScript(),
397
+ });
398
+ }
399
+ catch (err) {
400
+ console.warn(`[preview/headless] Failed to evaluate capture script: ${err.message}`);
401
+ }
402
+ await this.startScreencast(instance);
403
+ console.log(`[preview/headless] Created instance ${key} → ${targetUrl} (CDP port ${cdpPort})`);
404
+ return instance;
405
+ }
406
+ catch (err) {
407
+ this.releaseCdpPort(cdpPort);
408
+ throw err;
409
+ }
410
+ }
411
+ async destroy(instance) {
412
+ const inst = instance;
413
+ for (const [, dialog] of inst.pendingDialogs) {
414
+ clearTimeout(dialog.timeout);
415
+ }
416
+ inst.pendingDialogs.clear();
417
+ for (const [, auth] of inst.pendingAuthRequests) {
418
+ clearTimeout(auth.timeout);
419
+ }
420
+ inst.pendingAuthRequests.clear();
421
+ await this.stopScreencast(inst);
422
+ try {
423
+ await inst.browser.close();
424
+ }
425
+ catch { }
426
+ this.releaseCdpPort(inst.cdpPort);
427
+ try {
428
+ fs.rmSync(inst.downloadDir, { recursive: true, force: true });
429
+ }
430
+ catch { }
431
+ for (const ws of inst.wsClients) {
432
+ ws.close(1000, 'Preview instance destroyed');
433
+ }
434
+ console.log(`[preview/headless] Destroyed instance ${inst.id}`);
435
+ }
436
+ viewportTimers = new Map();
437
+ async setViewport(instance, viewport) {
438
+ const inst = instance;
439
+ const newVp = {
440
+ width: viewport.width,
441
+ height: viewport.height,
442
+ deviceScaleFactor: viewport.dpr ?? inst.viewport.deviceScaleFactor,
443
+ isMobile: viewport.isMobile ?? inst.viewport.isMobile,
444
+ hasTouch: viewport.hasTouch ?? inst.viewport.hasTouch,
445
+ };
446
+ const cur = inst.viewport;
447
+ if (cur.width === newVp.width && cur.height === newVp.height &&
448
+ cur.deviceScaleFactor === newVp.deviceScaleFactor &&
449
+ cur.isMobile === newVp.isMobile && cur.hasTouch === newVp.hasTouch) {
450
+ return;
451
+ }
452
+ // Debounce rapid viewport changes — only apply after 200ms of no changes
453
+ const existing = this.viewportTimers.get(inst.id);
454
+ if (existing)
455
+ clearTimeout(existing);
456
+ inst.viewport = newVp; // Update immediately so the next comparison works
457
+ this.viewportTimers.set(inst.id, setTimeout(async () => {
458
+ this.viewportTimers.delete(inst.id);
459
+ try {
460
+ await this.stopScreencast(inst);
461
+ await inst.page.setViewport({
462
+ width: newVp.width,
463
+ height: newVp.height,
464
+ deviceScaleFactor: newVp.deviceScaleFactor,
465
+ isMobile: newVp.isMobile,
466
+ hasTouch: newVp.hasTouch,
467
+ });
468
+ await this.startScreencast(inst);
469
+ }
470
+ catch (err) {
471
+ console.warn(`[preview/headless] Viewport update error: ${err.message}`);
472
+ }
473
+ }, 200));
474
+ inst.lastActivity = Date.now();
475
+ }
476
+ async setQuality(instance, quality) {
477
+ const inst = instance;
478
+ inst.quality = quality;
479
+ await this.stopScreencast(inst);
480
+ await this.startScreencast(inst);
481
+ }
482
+ async navigate(instance, url) {
483
+ const inst = instance;
484
+ try {
485
+ await inst.page.goto(url, { waitUntil: 'domcontentloaded', timeout: 10000 });
486
+ }
487
+ catch (err) {
488
+ console.warn(`[preview/headless] Navigate warning: ${err.message}`);
489
+ }
490
+ inst.lastActivity = Date.now();
491
+ }
492
+ async reload(instance) {
493
+ const inst = instance;
494
+ try {
495
+ await inst.page.reload({ waitUntil: 'domcontentloaded', timeout: 10000 });
496
+ }
497
+ catch (err) {
498
+ console.warn(`[preview/headless] Reload warning: ${err.message}`);
499
+ }
500
+ inst.lastActivity = Date.now();
501
+ }
502
+ async dispatchMouseEvent(instance, params) {
503
+ const inst = instance;
504
+ await inst.cdpSession.send('Input.dispatchMouseEvent', params);
505
+ inst.lastActivity = Date.now();
506
+ }
507
+ async dispatchKeyEvent(instance, params) {
508
+ const inst = instance;
509
+ await inst.cdpSession.send('Input.dispatchKeyEvent', params);
510
+ inst.lastActivity = Date.now();
511
+ }
512
+ async insertText(instance, text) {
513
+ const inst = instance;
514
+ await inst.cdpSession.send('Input.insertText', { text });
515
+ inst.lastActivity = Date.now();
516
+ }
517
+ async dispatchScrollEvent(instance, params) {
518
+ const inst = instance;
519
+ await inst.cdpSession.send('Input.dispatchMouseEvent', {
520
+ type: 'mouseWheel',
521
+ x: params.x,
522
+ y: params.y,
523
+ deltaX: params.deltaX,
524
+ deltaY: params.deltaY,
525
+ });
526
+ inst.lastActivity = Date.now();
527
+ }
528
+ async screenshot(instance, fullPage) {
529
+ const inst = instance;
530
+ const data = await inst.page.screenshot({
531
+ fullPage: !!fullPage,
532
+ type: 'png',
533
+ encoding: 'base64',
534
+ });
535
+ return data;
536
+ }
537
+ handleDialogResponse(instance, dialogId, accept, promptText) {
538
+ const pending = instance.pendingDialogs.get(dialogId);
539
+ if (pending) {
540
+ clearTimeout(pending.timeout);
541
+ pending.resolve(accept, promptText);
542
+ broadcastJson(instance, { type: 'dialog_closed', dialogId, accepted: accept });
543
+ }
544
+ }
545
+ async handleFileChooserResponse(instance, backendNodeId, files) {
546
+ const inst = instance;
547
+ try {
548
+ await inst.cdpSession.send('DOM.setFileInputFiles', { files, backendNodeId });
549
+ }
550
+ catch (err) {
551
+ console.warn(`[preview/headless] File chooser response error: ${err.message}`);
552
+ }
553
+ }
554
+ getDownloadPath(instance, guid) {
555
+ const filePath = path.join(instance.downloadDir, guid);
556
+ if (fs.existsSync(filePath))
557
+ return filePath;
558
+ return null;
559
+ }
560
+ startRecording(instance) {
561
+ instance.recording = true;
562
+ instance.recordedFrames = [];
563
+ console.log(`[preview/headless] Recording started for ${instance.id}`);
564
+ }
565
+ async stopRecording(instance) {
566
+ instance.recording = false;
567
+ const frames = instance.recordedFrames;
568
+ instance.recordedFrames = [];
569
+ if (frames.length === 0)
570
+ throw new Error('No frames recorded');
571
+ console.log(`[preview/headless] Recording stopped for ${instance.id}, encoding ${frames.length} frames...`);
572
+ return new Promise((resolve, reject) => {
573
+ const ffmpeg = spawn(FFMPEG_PATH, [
574
+ '-f', 'image2pipe',
575
+ '-framerate', '15',
576
+ '-i', 'pipe:0',
577
+ '-c:v', 'libvpx-vp9',
578
+ '-b:v', '1M',
579
+ '-f', 'webm',
580
+ 'pipe:1',
581
+ ]);
582
+ const chunks = [];
583
+ ffmpeg.stdout.on('data', (chunk) => chunks.push(chunk));
584
+ ffmpeg.stderr.on('data', () => { });
585
+ ffmpeg.on('close', (code) => {
586
+ if (code !== 0) {
587
+ reject(new Error(`ffmpeg exited with code ${code}`));
588
+ return;
589
+ }
590
+ const webm = Buffer.concat(chunks);
591
+ resolve(webm.toString('base64'));
592
+ });
593
+ ffmpeg.on('error', reject);
594
+ for (const frame of frames) {
595
+ ffmpeg.stdin.write(frame.data);
596
+ }
597
+ ffmpeg.stdin.end();
598
+ });
599
+ }
600
+ async restartStream(instance) {
601
+ const inst = instance;
602
+ await this.stopScreencast(inst);
603
+ await this.startScreencast(inst);
604
+ }
605
+ // --- Shadow Overlay response handlers ---
606
+ async handleSelectResponse(instance, selectorPath, value) {
607
+ const inst = instance;
608
+ try {
609
+ await inst.cdpSession.send('Runtime.evaluate', {
610
+ expression: buildUniversalSetterExpression(selectorPath, value),
611
+ });
612
+ }
613
+ catch (err) {
614
+ console.warn(`[preview/headless] Select response error: ${err.message}`);
615
+ }
616
+ inst.lastActivity = Date.now();
617
+ }
618
+ async handlePickerResponse(instance, selectorPath, value, inputType) {
619
+ const inst = instance;
620
+ try {
621
+ await inst.cdpSession.send('Runtime.evaluate', {
622
+ expression: buildUniversalSetterExpression(selectorPath, value),
623
+ });
624
+ }
625
+ catch (err) {
626
+ console.warn(`[preview/headless] Picker response error: ${err.message}`);
627
+ }
628
+ inst.lastActivity = Date.now();
629
+ }
630
+ handleAuthResponse(instance, requestId, username, password) {
631
+ const inst = instance;
632
+ const pending = inst.pendingAuthRequests.get(requestId);
633
+ if (pending) {
634
+ clearTimeout(pending.timeout);
635
+ inst.pendingAuthRequests.delete(requestId);
636
+ }
637
+ inst.cdpSession.send('Fetch.continueWithAuth', {
638
+ requestId,
639
+ authChallengeResponse: {
640
+ response: 'ProvideCredentials',
641
+ username,
642
+ password,
643
+ },
644
+ }).catch((err) => {
645
+ console.warn(`[preview/headless] Auth response error: ${err.message}`);
646
+ });
647
+ inst.lastActivity = Date.now();
648
+ }
649
+ handleAuthCancel(instance, requestId) {
650
+ const inst = instance;
651
+ const pending = inst.pendingAuthRequests.get(requestId);
652
+ if (pending) {
653
+ clearTimeout(pending.timeout);
654
+ inst.pendingAuthRequests.delete(requestId);
655
+ }
656
+ inst.cdpSession.send('Fetch.continueWithAuth', {
657
+ requestId,
658
+ authChallengeResponse: { response: 'CancelAuth' },
659
+ }).catch((err) => {
660
+ console.warn(`[preview/headless] Auth cancel error: ${err.message}`);
661
+ });
662
+ inst.lastActivity = Date.now();
663
+ }
664
+ async handleContextMenuAction(instance, action, params) {
665
+ const inst = instance;
666
+ try {
667
+ switch (action) {
668
+ case 'back':
669
+ await inst.cdpSession.send('Page.navigateToHistoryEntry', {
670
+ entryId: (await inst.cdpSession.send('Page.getNavigationHistory')).currentIndex - 1,
671
+ });
672
+ break;
673
+ case 'forward':
674
+ await inst.cdpSession.send('Page.navigateToHistoryEntry', {
675
+ entryId: (await inst.cdpSession.send('Page.getNavigationHistory')).currentIndex + 1,
676
+ });
677
+ break;
678
+ case 'reload':
679
+ await inst.page.reload({ waitUntil: 'domcontentloaded', timeout: 10000 });
680
+ break;
681
+ case 'copy':
682
+ // selectedText is already sent to the frontend in context_menu message
683
+ // Frontend handles clipboard.writeText() locally
684
+ break;
685
+ case 'copy_link':
686
+ // linkHref is already in the context_menu message, frontend copies it
687
+ break;
688
+ }
689
+ }
690
+ catch (err) {
691
+ console.warn(`[preview/headless] Context menu action '${action}' error: ${err.message}`);
692
+ }
693
+ inst.lastActivity = Date.now();
694
+ }
695
+ async shutdown() {
696
+ // Nothing to do — instances are destroyed individually by the manager
697
+ }
698
+ }