claude-mux 0.7.0

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 (636) hide show
  1. package/.beads/README.md +81 -0
  2. package/.beads/config.yaml +62 -0
  3. package/.beads/interactions.jsonl +0 -0
  4. package/.beads/issues.jsonl +5 -0
  5. package/.beads/metadata.json +4 -0
  6. package/.eslintrc.json +25 -0
  7. package/.gitattributes +3 -0
  8. package/.prettierrc +7 -0
  9. package/CLAUDE.md +123 -0
  10. package/LICENSE +21 -0
  11. package/PLAN-beads-integration.md +250 -0
  12. package/README.md +366 -0
  13. package/dist/app.d.ts +2 -0
  14. package/dist/app.d.ts.map +1 -0
  15. package/dist/app.js +147 -0
  16. package/dist/app.js.map +1 -0
  17. package/dist/cli.d.ts +3 -0
  18. package/dist/cli.d.ts.map +1 -0
  19. package/dist/cli.js +65 -0
  20. package/dist/cli.js.map +1 -0
  21. package/dist/commands/index.d.ts +5 -0
  22. package/dist/commands/index.d.ts.map +1 -0
  23. package/dist/commands/index.js +5 -0
  24. package/dist/commands/index.js.map +1 -0
  25. package/dist/commands/serve.d.ts +8 -0
  26. package/dist/commands/serve.d.ts.map +1 -0
  27. package/dist/commands/serve.js +59 -0
  28. package/dist/commands/serve.js.map +1 -0
  29. package/dist/commands/setup.d.ts +4 -0
  30. package/dist/commands/setup.d.ts.map +1 -0
  31. package/dist/commands/setup.js +12 -0
  32. package/dist/commands/setup.js.map +1 -0
  33. package/dist/commands/tui.d.ts +9 -0
  34. package/dist/commands/tui.d.ts.map +1 -0
  35. package/dist/commands/tui.js +234 -0
  36. package/dist/commands/tui.js.map +1 -0
  37. package/dist/commands/uninstall.d.ts +4 -0
  38. package/dist/commands/uninstall.d.ts.map +1 -0
  39. package/dist/commands/uninstall.js +12 -0
  40. package/dist/commands/uninstall.js.map +1 -0
  41. package/dist/components/Header.d.ts +7 -0
  42. package/dist/components/Header.d.ts.map +1 -0
  43. package/dist/components/Header.js +14 -0
  44. package/dist/components/Header.js.map +1 -0
  45. package/dist/components/HelpDialog.d.ts +6 -0
  46. package/dist/components/HelpDialog.d.ts.map +1 -0
  47. package/dist/components/HelpDialog.js +14 -0
  48. package/dist/components/HelpDialog.js.map +1 -0
  49. package/dist/components/SessionEntry.d.ts +17 -0
  50. package/dist/components/SessionEntry.d.ts.map +1 -0
  51. package/dist/components/SessionEntry.js +102 -0
  52. package/dist/components/SessionEntry.js.map +1 -0
  53. package/dist/components/SessionList.d.ts +14 -0
  54. package/dist/components/SessionList.d.ts.map +1 -0
  55. package/dist/components/SessionList.js +57 -0
  56. package/dist/components/SessionList.js.map +1 -0
  57. package/dist/components/StatusBar.d.ts +6 -0
  58. package/dist/components/StatusBar.d.ts.map +1 -0
  59. package/dist/components/StatusBar.js +7 -0
  60. package/dist/components/StatusBar.js.map +1 -0
  61. package/dist/components/index.d.ts +6 -0
  62. package/dist/components/index.d.ts.map +1 -0
  63. package/dist/components/index.js +6 -0
  64. package/dist/components/index.js.map +1 -0
  65. package/dist/db/index.d.ts +2 -0
  66. package/dist/db/index.d.ts.map +1 -0
  67. package/dist/db/index.js +3 -0
  68. package/dist/db/index.js.map +1 -0
  69. package/dist/db/schema.d.ts +4 -0
  70. package/dist/db/schema.d.ts.map +1 -0
  71. package/dist/db/schema.js +23 -0
  72. package/dist/db/schema.js.map +1 -0
  73. package/dist/db/sessions-json.d.ts +85 -0
  74. package/dist/db/sessions-json.d.ts.map +1 -0
  75. package/dist/db/sessions-json.js +242 -0
  76. package/dist/db/sessions-json.js.map +1 -0
  77. package/dist/db/sessions.d.ts +38 -0
  78. package/dist/db/sessions.d.ts.map +1 -0
  79. package/dist/db/sessions.js +87 -0
  80. package/dist/db/sessions.js.map +1 -0
  81. package/dist/hooks/claude-mux-hook.d.ts +15 -0
  82. package/dist/hooks/claude-mux-hook.d.ts.map +1 -0
  83. package/dist/hooks/claude-mux-hook.js +396 -0
  84. package/dist/hooks/claude-mux-hook.js.map +1 -0
  85. package/dist/hooks/claude-watch-hook.d.ts +15 -0
  86. package/dist/hooks/claude-watch-hook.d.ts.map +1 -0
  87. package/dist/hooks/claude-watch-hook.js +396 -0
  88. package/dist/hooks/claude-watch-hook.js.map +1 -0
  89. package/dist/server/custom-server.d.ts +13 -0
  90. package/dist/server/custom-server.d.ts.map +1 -0
  91. package/dist/server/custom-server.js +63 -0
  92. package/dist/server/custom-server.js.map +1 -0
  93. package/dist/server/index.d.ts +9 -0
  94. package/dist/server/index.d.ts.map +1 -0
  95. package/dist/server/index.js +1143 -0
  96. package/dist/server/index.js.map +1 -0
  97. package/dist/server/middleware/cors.d.ts +2 -0
  98. package/dist/server/middleware/cors.d.ts.map +1 -0
  99. package/dist/server/middleware/cors.js +11 -0
  100. package/dist/server/middleware/cors.js.map +1 -0
  101. package/dist/server/routes/sessions.d.ts +3 -0
  102. package/dist/server/routes/sessions.d.ts.map +1 -0
  103. package/dist/server/routes/sessions.js +78 -0
  104. package/dist/server/routes/sessions.js.map +1 -0
  105. package/dist/server/routes/stream.d.ts +3 -0
  106. package/dist/server/routes/stream.d.ts.map +1 -0
  107. package/dist/server/routes/stream.js +45 -0
  108. package/dist/server/routes/stream.js.map +1 -0
  109. package/dist/server/types.d.ts +28 -0
  110. package/dist/server/types.d.ts.map +1 -0
  111. package/dist/server/types.js +2 -0
  112. package/dist/server/types.js.map +1 -0
  113. package/dist/server/watcher.d.ts +22 -0
  114. package/dist/server/watcher.d.ts.map +1 -0
  115. package/dist/server/watcher.js +119 -0
  116. package/dist/server/watcher.js.map +1 -0
  117. package/dist/server/websocket.d.ts +51 -0
  118. package/dist/server/websocket.d.ts.map +1 -0
  119. package/dist/server/websocket.js +156 -0
  120. package/dist/server/websocket.js.map +1 -0
  121. package/dist/server/ws-handlers.d.ts +174 -0
  122. package/dist/server/ws-handlers.d.ts.map +1 -0
  123. package/dist/server/ws-handlers.js +695 -0
  124. package/dist/server/ws-handlers.js.map +1 -0
  125. package/dist/setup/hooks.d.ts +44 -0
  126. package/dist/setup/hooks.d.ts.map +1 -0
  127. package/dist/setup/hooks.js +267 -0
  128. package/dist/setup/hooks.js.map +1 -0
  129. package/dist/setup/index.d.ts +3 -0
  130. package/dist/setup/index.d.ts.map +1 -0
  131. package/dist/setup/index.js +3 -0
  132. package/dist/setup/index.js.map +1 -0
  133. package/dist/setup/wizard.d.ts +4 -0
  134. package/dist/setup/wizard.d.ts.map +1 -0
  135. package/dist/setup/wizard.js +72 -0
  136. package/dist/setup/wizard.js.map +1 -0
  137. package/dist/tmux/detect.d.ts +25 -0
  138. package/dist/tmux/detect.d.ts.map +1 -0
  139. package/dist/tmux/detect.js +71 -0
  140. package/dist/tmux/detect.js.map +1 -0
  141. package/dist/tmux/navigate.d.ts +13 -0
  142. package/dist/tmux/navigate.d.ts.map +1 -0
  143. package/dist/tmux/navigate.js +41 -0
  144. package/dist/tmux/navigate.js.map +1 -0
  145. package/dist/tmux/pane.d.ts +57 -0
  146. package/dist/tmux/pane.d.ts.map +1 -0
  147. package/dist/tmux/pane.js +156 -0
  148. package/dist/tmux/pane.js.map +1 -0
  149. package/dist/tmux/resize.d.ts +10 -0
  150. package/dist/tmux/resize.d.ts.map +1 -0
  151. package/dist/tmux/resize.js +25 -0
  152. package/dist/tmux/resize.js.map +1 -0
  153. package/dist/utils/paths.d.ts +7 -0
  154. package/dist/utils/paths.d.ts.map +1 -0
  155. package/dist/utils/paths.js +9 -0
  156. package/dist/utils/paths.js.map +1 -0
  157. package/dist/utils/pid.d.ts +5 -0
  158. package/dist/utils/pid.d.ts.map +1 -0
  159. package/dist/utils/pid.js +14 -0
  160. package/dist/utils/pid.js.map +1 -0
  161. package/dist/utils/version.d.ts +6 -0
  162. package/dist/utils/version.d.ts.map +1 -0
  163. package/dist/utils/version.js +6 -0
  164. package/dist/utils/version.js.map +1 -0
  165. package/dist/web/client/_app/immutable/assets/0.WptSHSUl.css +1 -0
  166. package/dist/web/client/_app/immutable/assets/2.s6Kx4oz1.css +1 -0
  167. package/dist/web/client/_app/immutable/assets/4.DoNWy7tW.css +1 -0
  168. package/dist/web/client/_app/immutable/assets/AllSessionsPanel.CGHY3HLy.css +1 -0
  169. package/dist/web/client/_app/immutable/chunks/-3mUPuLP.js +1 -0
  170. package/dist/web/client/_app/immutable/chunks/B5U4_V3d.js +1 -0
  171. package/dist/web/client/_app/immutable/chunks/BHwiZXRv.js +1 -0
  172. package/dist/web/client/_app/immutable/chunks/C9P-coqM.js +1 -0
  173. package/dist/web/client/_app/immutable/chunks/Cegv0r8x.js +1 -0
  174. package/dist/web/client/_app/immutable/chunks/DU91Ml7U.js +3 -0
  175. package/dist/web/client/_app/immutable/chunks/DmdO6ygw.js +1 -0
  176. package/dist/web/client/_app/immutable/chunks/HKNo9LID.js +5 -0
  177. package/dist/web/client/_app/immutable/chunks/U4ip-C0d.js +2 -0
  178. package/dist/web/client/_app/immutable/chunks/cgUjKIhX.js +2 -0
  179. package/dist/web/client/_app/immutable/entry/app.CGIBnoln.js +2 -0
  180. package/dist/web/client/_app/immutable/entry/start.CJk8zB1j.js +1 -0
  181. package/dist/web/client/_app/immutable/nodes/0.CqlJ9a31.js +1 -0
  182. package/dist/web/client/_app/immutable/nodes/1.BQUZh2-w.js +1 -0
  183. package/dist/web/client/_app/immutable/nodes/2.CCV1YdgF.js +1 -0
  184. package/dist/web/client/_app/immutable/nodes/3.D9tDCdq8.js +1 -0
  185. package/dist/web/client/_app/immutable/nodes/4.BqPyNkFA.js +6 -0
  186. package/dist/web/client/_app/version.json +1 -0
  187. package/dist/web/client/robots.txt +3 -0
  188. package/dist/web/env.js +32 -0
  189. package/dist/web/handler.js +744 -0
  190. package/dist/web/index.js +49 -0
  191. package/dist/web/server/chunks/0-BHWsmCJv.js +23 -0
  192. package/dist/web/server/chunks/0-BHWsmCJv.js.map +1 -0
  193. package/dist/web/server/chunks/1-YRx6A8Tm.js +17 -0
  194. package/dist/web/server/chunks/1-YRx6A8Tm.js.map +1 -0
  195. package/dist/web/server/chunks/2-eC6JuGAo.js +22 -0
  196. package/dist/web/server/chunks/2-eC6JuGAo.js.map +1 -0
  197. package/dist/web/server/chunks/3-Bk-wV20p.js +32 -0
  198. package/dist/web/server/chunks/3-Bk-wV20p.js.map +1 -0
  199. package/dist/web/server/chunks/4-nteBgDrW.js +21 -0
  200. package/dist/web/server/chunks/4-nteBgDrW.js.map +1 -0
  201. package/dist/web/server/chunks/AllSessionsPanel.svelte_svelte_type_style_lang-Bt4B0-oi.js +35 -0
  202. package/dist/web/server/chunks/AllSessionsPanel.svelte_svelte_type_style_lang-Bt4B0-oi.js.map +1 -0
  203. package/dist/web/server/chunks/_layout.svelte-BIF9eZCY.js +453 -0
  204. package/dist/web/server/chunks/_layout.svelte-BIF9eZCY.js.map +1 -0
  205. package/dist/web/server/chunks/_page.svelte-Be6iabRn.js +34 -0
  206. package/dist/web/server/chunks/_page.svelte-Be6iabRn.js.map +1 -0
  207. package/dist/web/server/chunks/_page.svelte-C_fGJVSE.js +576 -0
  208. package/dist/web/server/chunks/_page.svelte-C_fGJVSE.js.map +1 -0
  209. package/dist/web/server/chunks/_page.svelte-CnfJk6cu.js +2722 -0
  210. package/dist/web/server/chunks/_page.svelte-CnfJk6cu.js.map +1 -0
  211. package/dist/web/server/chunks/_server.ts-3WAmKvn2.js +34 -0
  212. package/dist/web/server/chunks/_server.ts-3WAmKvn2.js.map +1 -0
  213. package/dist/web/server/chunks/_server.ts-BAWJCSFb.js +29 -0
  214. package/dist/web/server/chunks/_server.ts-BAWJCSFb.js.map +1 -0
  215. package/dist/web/server/chunks/_server.ts-BPpMOZCm.js +24 -0
  216. package/dist/web/server/chunks/_server.ts-BPpMOZCm.js.map +1 -0
  217. package/dist/web/server/chunks/_server.ts-BUKgRb6U.js +13 -0
  218. package/dist/web/server/chunks/_server.ts-BUKgRb6U.js.map +1 -0
  219. package/dist/web/server/chunks/_server.ts-BVHUS8fm.js +41 -0
  220. package/dist/web/server/chunks/_server.ts-BVHUS8fm.js.map +1 -0
  221. package/dist/web/server/chunks/_server.ts-BrF3od0O.js +45 -0
  222. package/dist/web/server/chunks/_server.ts-BrF3od0O.js.map +1 -0
  223. package/dist/web/server/chunks/_server.ts-C4oPmOJR.js +38 -0
  224. package/dist/web/server/chunks/_server.ts-C4oPmOJR.js.map +1 -0
  225. package/dist/web/server/chunks/_server.ts-CDAUUmG_.js +21 -0
  226. package/dist/web/server/chunks/_server.ts-CDAUUmG_.js.map +1 -0
  227. package/dist/web/server/chunks/_server.ts-CR0uWvpz.js +24 -0
  228. package/dist/web/server/chunks/_server.ts-CR0uWvpz.js.map +1 -0
  229. package/dist/web/server/chunks/_server.ts-CSqdCNGi.js +21 -0
  230. package/dist/web/server/chunks/_server.ts-CSqdCNGi.js.map +1 -0
  231. package/dist/web/server/chunks/_server.ts-DI9fzUo9.js +52 -0
  232. package/dist/web/server/chunks/_server.ts-DI9fzUo9.js.map +1 -0
  233. package/dist/web/server/chunks/_server.ts-DNjJTClI.js +46 -0
  234. package/dist/web/server/chunks/_server.ts-DNjJTClI.js.map +1 -0
  235. package/dist/web/server/chunks/_server.ts-DdbOAkOj.js +22 -0
  236. package/dist/web/server/chunks/_server.ts-DdbOAkOj.js.map +1 -0
  237. package/dist/web/server/chunks/_server.ts-Df2isGQc.js +41 -0
  238. package/dist/web/server/chunks/_server.ts-Df2isGQc.js.map +1 -0
  239. package/dist/web/server/chunks/_server.ts-Vpw25_-3.js +12 -0
  240. package/dist/web/server/chunks/_server.ts-Vpw25_-3.js.map +1 -0
  241. package/dist/web/server/chunks/_server.ts-WldRpSRi.js +26 -0
  242. package/dist/web/server/chunks/_server.ts-WldRpSRi.js.map +1 -0
  243. package/dist/web/server/chunks/alert-dialog-description-DDA6u-nS.js +2890 -0
  244. package/dist/web/server/chunks/alert-dialog-description-DDA6u-nS.js.map +1 -0
  245. package/dist/web/server/chunks/auth-DrUf-v4J.js +51 -0
  246. package/dist/web/server/chunks/auth-DrUf-v4J.js.map +1 -0
  247. package/dist/web/server/chunks/button-D6xS9rHt.js +2335 -0
  248. package/dist/web/server/chunks/button-D6xS9rHt.js.map +1 -0
  249. package/dist/web/server/chunks/chunk-EKzHsMy_.js +42 -0
  250. package/dist/web/server/chunks/chunks-DmdqVYC7.js +58 -0
  251. package/dist/web/server/chunks/chunks-DmdqVYC7.js.map +1 -0
  252. package/dist/web/server/chunks/client-CUrSQijh.js +18 -0
  253. package/dist/web/server/chunks/client-CUrSQijh.js.map +1 -0
  254. package/dist/web/server/chunks/clsx-FzI4_gi0.js +332 -0
  255. package/dist/web/server/chunks/clsx-FzI4_gi0.js.map +1 -0
  256. package/dist/web/server/chunks/error.svelte-D-c-9pTE.js +27 -0
  257. package/dist/web/server/chunks/error.svelte-D-c-9pTE.js.map +1 -0
  258. package/dist/web/server/chunks/events-BDBlYddw.js +89 -0
  259. package/dist/web/server/chunks/events-BDBlYddw.js.map +1 -0
  260. package/dist/web/server/chunks/exports-CCurQ-Tl.js +131 -0
  261. package/dist/web/server/chunks/exports-CCurQ-Tl.js.map +1 -0
  262. package/dist/web/server/chunks/hooks.server-BK1bopsh.js +86 -0
  263. package/dist/web/server/chunks/hooks.server-BK1bopsh.js.map +1 -0
  264. package/dist/web/server/chunks/index2-BQnysSj-.js +2588 -0
  265. package/dist/web/server/chunks/index2-BQnysSj-.js.map +1 -0
  266. package/dist/web/server/chunks/internal-DLENj6YI.js +61 -0
  267. package/dist/web/server/chunks/internal-DLENj6YI.js.map +1 -0
  268. package/dist/web/server/chunks/pane-Dg3pKvsm.js +94 -0
  269. package/dist/web/server/chunks/pane-Dg3pKvsm.js.map +1 -0
  270. package/dist/web/server/chunks/sessions-json-DgfkCLO7.js +107 -0
  271. package/dist/web/server/chunks/sessions-json-DgfkCLO7.js.map +1 -0
  272. package/dist/web/server/chunks/sessions.svelte-Ds82MvkB.js +178 -0
  273. package/dist/web/server/chunks/sessions.svelte-Ds82MvkB.js.map +1 -0
  274. package/dist/web/server/chunks/state.svelte-xeAZvWZ6.js +7 -0
  275. package/dist/web/server/chunks/state.svelte-xeAZvWZ6.js.map +1 -0
  276. package/dist/web/server/chunks/ws-handlers-B4r5eSP2.js +733 -0
  277. package/dist/web/server/chunks/ws-handlers-B4r5eSP2.js.map +1 -0
  278. package/dist/web/server/index.js +4907 -0
  279. package/dist/web/server/index.js.map +1 -0
  280. package/dist/web/server/manifest.js +233 -0
  281. package/dist/web/server/manifest.js.map +1 -0
  282. package/docs/images/desktop-dashboard.png +0 -0
  283. package/docs/images/desktop-session.png +0 -0
  284. package/docs/images/mobile-dashboard.png +0 -0
  285. package/docs/images/mobile-session.png +0 -0
  286. package/docs/release-checklist.md +228 -0
  287. package/docs/removing-hooks.md +135 -0
  288. package/docs/state-transitions.md +109 -0
  289. package/package.json +71 -0
  290. package/src/app.tsx +188 -0
  291. package/src/cli.ts +83 -0
  292. package/src/commands/index.ts +4 -0
  293. package/src/commands/serve.ts +75 -0
  294. package/src/commands/setup.ts +13 -0
  295. package/src/commands/tui.ts +255 -0
  296. package/src/commands/uninstall.ts +13 -0
  297. package/src/components/Header.tsx +32 -0
  298. package/src/components/HelpDialog.tsx +45 -0
  299. package/src/components/SessionEntry.tsx +202 -0
  300. package/src/components/SessionList.tsx +98 -0
  301. package/src/components/StatusBar.tsx +26 -0
  302. package/src/components/index.ts +5 -0
  303. package/src/db/index.ts +20 -0
  304. package/src/db/sessions-json.ts +314 -0
  305. package/src/hooks/claude-mux-hook.ts +498 -0
  306. package/src/server/watcher.ts +128 -0
  307. package/src/server/ws-handlers.ts +922 -0
  308. package/src/setup/hooks.ts +333 -0
  309. package/src/setup/index.ts +2 -0
  310. package/src/setup/wizard.ts +81 -0
  311. package/src/tmux/detect.ts +87 -0
  312. package/src/tmux/navigate.ts +42 -0
  313. package/src/tmux/pane.ts +167 -0
  314. package/src/tmux/resize.ts +28 -0
  315. package/src/utils/paths.ts +11 -0
  316. package/src/utils/pid.ts +12 -0
  317. package/src/utils/version.ts +5 -0
  318. package/tests/components/Header.test.tsx +42 -0
  319. package/tests/components/SessionEntry-extended.test.tsx +165 -0
  320. package/tests/components/SessionEntry.test.tsx +138 -0
  321. package/tests/components/SessionList.test.tsx +110 -0
  322. package/tests/components/StatusBar.test.tsx +31 -0
  323. package/tests/db/index.test.ts +78 -0
  324. package/tests/db/sessions.test.ts +230 -0
  325. package/tests/server/integration.test.ts +319 -0
  326. package/tests/server/sessions.test.ts +114 -0
  327. package/tests/setup/hooks-integration.test.ts +148 -0
  328. package/tests/setup/hooks.test.ts +123 -0
  329. package/tests/tmux/detect.test.ts +54 -0
  330. package/tests/tmux/navigate.test.ts +30 -0
  331. package/tests/utils/pid.test.ts +17 -0
  332. package/tsconfig.cli.json +9 -0
  333. package/tsconfig.json +22 -0
  334. package/vitest.config.ts +29 -0
  335. package/web/.svelte-kit/adapter-bun/.vite/manifest.json +408 -0
  336. package/web/.svelte-kit/adapter-bun/_app/immutable/assets/AllSessionsPanel.BKhqOrbV.css +1 -0
  337. package/web/.svelte-kit/adapter-bun/_app/immutable/assets/_layout.WptSHSUl.css +1 -0
  338. package/web/.svelte-kit/adapter-bun/_app/immutable/assets/_page.DldLgTc-.css +1 -0
  339. package/web/.svelte-kit/adapter-bun/_app/immutable/assets/_page.DoNWy7tW.css +1 -0
  340. package/web/.svelte-kit/adapter-bun/chunks/AllSessionsPanel.svelte_svelte_type_style_lang.js +49 -0
  341. package/web/.svelte-kit/adapter-bun/chunks/alert-dialog-description.js +2670 -0
  342. package/web/.svelte-kit/adapter-bun/chunks/auth.js +59 -0
  343. package/web/.svelte-kit/adapter-bun/chunks/button.js +82 -0
  344. package/web/.svelte-kit/adapter-bun/chunks/client.js +29 -0
  345. package/web/.svelte-kit/adapter-bun/chunks/context.js +133 -0
  346. package/web/.svelte-kit/adapter-bun/chunks/environment.js +34 -0
  347. package/web/.svelte-kit/adapter-bun/chunks/events.js +121 -0
  348. package/web/.svelte-kit/adapter-bun/chunks/exports.js +174 -0
  349. package/web/.svelte-kit/adapter-bun/chunks/false.js +4 -0
  350. package/web/.svelte-kit/adapter-bun/chunks/index.js +59 -0
  351. package/web/.svelte-kit/adapter-bun/chunks/index2.js +2828 -0
  352. package/web/.svelte-kit/adapter-bun/chunks/internal.js +920 -0
  353. package/web/.svelte-kit/adapter-bun/chunks/pane.js +82 -0
  354. package/web/.svelte-kit/adapter-bun/chunks/sessions-json.js +124 -0
  355. package/web/.svelte-kit/adapter-bun/chunks/sessions.svelte.js +229 -0
  356. package/web/.svelte-kit/adapter-bun/chunks/shared.js +542 -0
  357. package/web/.svelte-kit/adapter-bun/chunks/state.svelte.js +16 -0
  358. package/web/.svelte-kit/adapter-bun/chunks/utils.js +43 -0
  359. package/web/.svelte-kit/adapter-bun/chunks/ws-handlers.js +782 -0
  360. package/web/.svelte-kit/adapter-bun/entries/endpoints/api/auth/login/_server.ts.js +22 -0
  361. package/web/.svelte-kit/adapter-bun/entries/endpoints/api/auth/logout/_server.ts.js +9 -0
  362. package/web/.svelte-kit/adapter-bun/entries/endpoints/api/beads/_server.ts.js +22 -0
  363. package/web/.svelte-kit/adapter-bun/entries/endpoints/api/browse/_server.ts.js +50 -0
  364. package/web/.svelte-kit/adapter-bun/entries/endpoints/api/chrome/_server.ts.js +30 -0
  365. package/web/.svelte-kit/adapter-bun/entries/endpoints/api/files/image/_server.ts.js +53 -0
  366. package/web/.svelte-kit/adapter-bun/entries/endpoints/api/health/_server.ts.js +7 -0
  367. package/web/.svelte-kit/adapter-bun/entries/endpoints/api/projects/new-session/_server.ts.js +44 -0
  368. package/web/.svelte-kit/adapter-bun/entries/endpoints/api/sessions/_id_/_server.ts.js +20 -0
  369. package/web/.svelte-kit/adapter-bun/entries/endpoints/api/sessions/_id_/kill/_server.ts.js +36 -0
  370. package/web/.svelte-kit/adapter-bun/entries/endpoints/api/sessions/_id_/restart/_server.ts.js +40 -0
  371. package/web/.svelte-kit/adapter-bun/entries/endpoints/api/sessions/_id_/screenshots/_server.ts.js +28 -0
  372. package/web/.svelte-kit/adapter-bun/entries/endpoints/api/sessions/_id_/send/_server.ts.js +29 -0
  373. package/web/.svelte-kit/adapter-bun/entries/endpoints/api/sessions/_server.ts.js +49 -0
  374. package/web/.svelte-kit/adapter-bun/entries/endpoints/api/sessions/_target_/output/_server.ts.js +14 -0
  375. package/web/.svelte-kit/adapter-bun/entries/endpoints/api/tmux/panes/_server.ts.js +21 -0
  376. package/web/.svelte-kit/adapter-bun/entries/fallbacks/error.svelte.js +27 -0
  377. package/web/.svelte-kit/adapter-bun/entries/hooks.server.js +105 -0
  378. package/web/.svelte-kit/adapter-bun/entries/pages/_layout.svelte.js +499 -0
  379. package/web/.svelte-kit/adapter-bun/entries/pages/_page.svelte.js +3057 -0
  380. package/web/.svelte-kit/adapter-bun/entries/pages/login/_page.server.ts.js +15 -0
  381. package/web/.svelte-kit/adapter-bun/entries/pages/login/_page.svelte.js +37 -0
  382. package/web/.svelte-kit/adapter-bun/entries/pages/session/_target_/_page.svelte.js +653 -0
  383. package/web/.svelte-kit/adapter-bun/index.js +3864 -0
  384. package/web/.svelte-kit/adapter-bun/internal.js +13 -0
  385. package/web/.svelte-kit/adapter-bun/manifest-full.js +167 -0
  386. package/web/.svelte-kit/adapter-bun/manifest.js +171 -0
  387. package/web/.svelte-kit/adapter-bun/nodes/0.js +8 -0
  388. package/web/.svelte-kit/adapter-bun/nodes/1.js +8 -0
  389. package/web/.svelte-kit/adapter-bun/nodes/2.js +8 -0
  390. package/web/.svelte-kit/adapter-bun/nodes/3.js +10 -0
  391. package/web/.svelte-kit/adapter-bun/nodes/4.js +8 -0
  392. package/web/.svelte-kit/adapter-bun/remote-entry.js +541 -0
  393. package/web/.svelte-kit/adapter-node/.vite/manifest.json +223 -0
  394. package/web/.svelte-kit/adapter-node/_app/immutable/assets/_layout.4NiX29PU.css +1 -0
  395. package/web/.svelte-kit/adapter-node/_app/immutable/assets/_page.BEMzYUGV.css +1 -0
  396. package/web/.svelte-kit/adapter-node/_app/immutable/assets/_page.DOJn7TG7.css +1 -0
  397. package/web/.svelte-kit/adapter-node/chunks/context.js +121 -0
  398. package/web/.svelte-kit/adapter-node/chunks/environment.js +34 -0
  399. package/web/.svelte-kit/adapter-node/chunks/exports.js +174 -0
  400. package/web/.svelte-kit/adapter-node/chunks/false.js +4 -0
  401. package/web/.svelte-kit/adapter-node/chunks/index.js +59 -0
  402. package/web/.svelte-kit/adapter-node/chunks/index2.js +2710 -0
  403. package/web/.svelte-kit/adapter-node/chunks/internal.js +1005 -0
  404. package/web/.svelte-kit/adapter-node/chunks/sessions-json.js +109 -0
  405. package/web/.svelte-kit/adapter-node/chunks/sessions.svelte.js +67 -0
  406. package/web/.svelte-kit/adapter-node/chunks/shared.js +542 -0
  407. package/web/.svelte-kit/adapter-node/chunks/state.svelte.js +16 -0
  408. package/web/.svelte-kit/adapter-node/chunks/utils.js +43 -0
  409. package/web/.svelte-kit/adapter-node/entries/endpoints/api/browse/_server.ts.js +50 -0
  410. package/web/.svelte-kit/adapter-node/entries/endpoints/api/health/_server.ts.js +7 -0
  411. package/web/.svelte-kit/adapter-node/entries/endpoints/api/projects/new-session/_server.ts.js +44 -0
  412. package/web/.svelte-kit/adapter-node/entries/endpoints/api/sessions/_id_/_server.ts.js +20 -0
  413. package/web/.svelte-kit/adapter-node/entries/endpoints/api/sessions/_id_/kill/_server.ts.js +30 -0
  414. package/web/.svelte-kit/adapter-node/entries/endpoints/api/sessions/_id_/send/_server.ts.js +22 -0
  415. package/web/.svelte-kit/adapter-node/entries/endpoints/api/sessions/_server.ts.js +126 -0
  416. package/web/.svelte-kit/adapter-node/entries/endpoints/api/sessions/_target_/output/_server.ts.js +14 -0
  417. package/web/.svelte-kit/adapter-node/entries/fallbacks/error.svelte.js +44 -0
  418. package/web/.svelte-kit/adapter-node/entries/pages/_layout.svelte.js +12 -0
  419. package/web/.svelte-kit/adapter-node/entries/pages/_page.svelte.js +87 -0
  420. package/web/.svelte-kit/adapter-node/entries/pages/session/_target_/_page.svelte.js +76 -0
  421. package/web/.svelte-kit/adapter-node/index.js +3864 -0
  422. package/web/.svelte-kit/adapter-node/internal.js +13 -0
  423. package/web/.svelte-kit/adapter-node/manifest-full.js +103 -0
  424. package/web/.svelte-kit/adapter-node/manifest.js +107 -0
  425. package/web/.svelte-kit/adapter-node/nodes/0.js +8 -0
  426. package/web/.svelte-kit/adapter-node/nodes/1.js +8 -0
  427. package/web/.svelte-kit/adapter-node/nodes/2.js +8 -0
  428. package/web/.svelte-kit/adapter-node/nodes/3.js +8 -0
  429. package/web/.svelte-kit/adapter-node/remote-entry.js +541 -0
  430. package/web/.svelte-kit/ambient.d.ts +187 -0
  431. package/web/.svelte-kit/generated/client/app.js +33 -0
  432. package/web/.svelte-kit/generated/client/matchers.js +1 -0
  433. package/web/.svelte-kit/generated/client/nodes/0.js +1 -0
  434. package/web/.svelte-kit/generated/client/nodes/1.js +1 -0
  435. package/web/.svelte-kit/generated/client/nodes/2.js +1 -0
  436. package/web/.svelte-kit/generated/client/nodes/3.js +1 -0
  437. package/web/.svelte-kit/generated/client/nodes/4.js +1 -0
  438. package/web/.svelte-kit/generated/client-optimized/app.js +33 -0
  439. package/web/.svelte-kit/generated/client-optimized/matchers.js +1 -0
  440. package/web/.svelte-kit/generated/client-optimized/nodes/0.js +1 -0
  441. package/web/.svelte-kit/generated/client-optimized/nodes/1.js +1 -0
  442. package/web/.svelte-kit/generated/client-optimized/nodes/2.js +1 -0
  443. package/web/.svelte-kit/generated/client-optimized/nodes/3.js +1 -0
  444. package/web/.svelte-kit/generated/client-optimized/nodes/4.js +1 -0
  445. package/web/.svelte-kit/generated/root.js +3 -0
  446. package/web/.svelte-kit/generated/root.svelte +68 -0
  447. package/web/.svelte-kit/generated/server/internal.js +53 -0
  448. package/web/.svelte-kit/non-ambient.d.ts +73 -0
  449. package/web/.svelte-kit/output/client/.vite/manifest.json +203 -0
  450. package/web/.svelte-kit/output/client/_app/immutable/assets/0.WptSHSUl.css +1 -0
  451. package/web/.svelte-kit/output/client/_app/immutable/assets/2.s6Kx4oz1.css +1 -0
  452. package/web/.svelte-kit/output/client/_app/immutable/assets/4.DoNWy7tW.css +1 -0
  453. package/web/.svelte-kit/output/client/_app/immutable/assets/AllSessionsPanel.CGHY3HLy.css +1 -0
  454. package/web/.svelte-kit/output/client/_app/immutable/chunks/-3mUPuLP.js +1 -0
  455. package/web/.svelte-kit/output/client/_app/immutable/chunks/B5U4_V3d.js +1 -0
  456. package/web/.svelte-kit/output/client/_app/immutable/chunks/BHwiZXRv.js +1 -0
  457. package/web/.svelte-kit/output/client/_app/immutable/chunks/C9P-coqM.js +1 -0
  458. package/web/.svelte-kit/output/client/_app/immutable/chunks/Cegv0r8x.js +1 -0
  459. package/web/.svelte-kit/output/client/_app/immutable/chunks/DU91Ml7U.js +3 -0
  460. package/web/.svelte-kit/output/client/_app/immutable/chunks/DmdO6ygw.js +1 -0
  461. package/web/.svelte-kit/output/client/_app/immutable/chunks/HKNo9LID.js +5 -0
  462. package/web/.svelte-kit/output/client/_app/immutable/chunks/U4ip-C0d.js +2 -0
  463. package/web/.svelte-kit/output/client/_app/immutable/chunks/cgUjKIhX.js +2 -0
  464. package/web/.svelte-kit/output/client/_app/immutable/entry/app.CGIBnoln.js +2 -0
  465. package/web/.svelte-kit/output/client/_app/immutable/entry/start.CJk8zB1j.js +1 -0
  466. package/web/.svelte-kit/output/client/_app/immutable/nodes/0.CqlJ9a31.js +1 -0
  467. package/web/.svelte-kit/output/client/_app/immutable/nodes/1.BQUZh2-w.js +1 -0
  468. package/web/.svelte-kit/output/client/_app/immutable/nodes/2.CCV1YdgF.js +1 -0
  469. package/web/.svelte-kit/output/client/_app/immutable/nodes/3.D9tDCdq8.js +1 -0
  470. package/web/.svelte-kit/output/client/_app/immutable/nodes/4.BqPyNkFA.js +6 -0
  471. package/web/.svelte-kit/output/client/_app/version.json +1 -0
  472. package/web/.svelte-kit/output/client/robots.txt +3 -0
  473. package/web/.svelte-kit/output/server/.vite/manifest.json +408 -0
  474. package/web/.svelte-kit/output/server/_app/immutable/assets/AllSessionsPanel.BKhqOrbV.css +1 -0
  475. package/web/.svelte-kit/output/server/_app/immutable/assets/_layout.WptSHSUl.css +1 -0
  476. package/web/.svelte-kit/output/server/_app/immutable/assets/_page.DldLgTc-.css +1 -0
  477. package/web/.svelte-kit/output/server/_app/immutable/assets/_page.DoNWy7tW.css +1 -0
  478. package/web/.svelte-kit/output/server/chunks/AllSessionsPanel.svelte_svelte_type_style_lang.js +49 -0
  479. package/web/.svelte-kit/output/server/chunks/alert-dialog-description.js +2670 -0
  480. package/web/.svelte-kit/output/server/chunks/auth.js +59 -0
  481. package/web/.svelte-kit/output/server/chunks/button.js +82 -0
  482. package/web/.svelte-kit/output/server/chunks/client.js +29 -0
  483. package/web/.svelte-kit/output/server/chunks/context.js +133 -0
  484. package/web/.svelte-kit/output/server/chunks/environment.js +34 -0
  485. package/web/.svelte-kit/output/server/chunks/events.js +121 -0
  486. package/web/.svelte-kit/output/server/chunks/exports.js +174 -0
  487. package/web/.svelte-kit/output/server/chunks/false.js +4 -0
  488. package/web/.svelte-kit/output/server/chunks/index.js +59 -0
  489. package/web/.svelte-kit/output/server/chunks/index2.js +2828 -0
  490. package/web/.svelte-kit/output/server/chunks/internal.js +920 -0
  491. package/web/.svelte-kit/output/server/chunks/pane.js +82 -0
  492. package/web/.svelte-kit/output/server/chunks/sessions-json.js +124 -0
  493. package/web/.svelte-kit/output/server/chunks/sessions.svelte.js +229 -0
  494. package/web/.svelte-kit/output/server/chunks/shared.js +542 -0
  495. package/web/.svelte-kit/output/server/chunks/state.svelte.js +16 -0
  496. package/web/.svelte-kit/output/server/chunks/utils.js +43 -0
  497. package/web/.svelte-kit/output/server/chunks/ws-handlers.js +782 -0
  498. package/web/.svelte-kit/output/server/entries/endpoints/api/auth/login/_server.ts.js +22 -0
  499. package/web/.svelte-kit/output/server/entries/endpoints/api/auth/logout/_server.ts.js +9 -0
  500. package/web/.svelte-kit/output/server/entries/endpoints/api/beads/_server.ts.js +22 -0
  501. package/web/.svelte-kit/output/server/entries/endpoints/api/browse/_server.ts.js +50 -0
  502. package/web/.svelte-kit/output/server/entries/endpoints/api/chrome/_server.ts.js +30 -0
  503. package/web/.svelte-kit/output/server/entries/endpoints/api/files/image/_server.ts.js +53 -0
  504. package/web/.svelte-kit/output/server/entries/endpoints/api/health/_server.ts.js +7 -0
  505. package/web/.svelte-kit/output/server/entries/endpoints/api/projects/new-session/_server.ts.js +44 -0
  506. package/web/.svelte-kit/output/server/entries/endpoints/api/sessions/_id_/_server.ts.js +20 -0
  507. package/web/.svelte-kit/output/server/entries/endpoints/api/sessions/_id_/kill/_server.ts.js +36 -0
  508. package/web/.svelte-kit/output/server/entries/endpoints/api/sessions/_id_/restart/_server.ts.js +40 -0
  509. package/web/.svelte-kit/output/server/entries/endpoints/api/sessions/_id_/screenshots/_server.ts.js +28 -0
  510. package/web/.svelte-kit/output/server/entries/endpoints/api/sessions/_id_/send/_server.ts.js +29 -0
  511. package/web/.svelte-kit/output/server/entries/endpoints/api/sessions/_server.ts.js +49 -0
  512. package/web/.svelte-kit/output/server/entries/endpoints/api/sessions/_target_/output/_server.ts.js +14 -0
  513. package/web/.svelte-kit/output/server/entries/endpoints/api/tmux/panes/_server.ts.js +21 -0
  514. package/web/.svelte-kit/output/server/entries/fallbacks/error.svelte.js +27 -0
  515. package/web/.svelte-kit/output/server/entries/hooks.server.js +105 -0
  516. package/web/.svelte-kit/output/server/entries/pages/_layout.svelte.js +499 -0
  517. package/web/.svelte-kit/output/server/entries/pages/_page.svelte.js +3057 -0
  518. package/web/.svelte-kit/output/server/entries/pages/login/_page.server.ts.js +15 -0
  519. package/web/.svelte-kit/output/server/entries/pages/login/_page.svelte.js +37 -0
  520. package/web/.svelte-kit/output/server/entries/pages/session/_target_/_page.svelte.js +653 -0
  521. package/web/.svelte-kit/output/server/index.js +3864 -0
  522. package/web/.svelte-kit/output/server/internal.js +13 -0
  523. package/web/.svelte-kit/output/server/manifest-full.js +167 -0
  524. package/web/.svelte-kit/output/server/manifest.js +167 -0
  525. package/web/.svelte-kit/output/server/nodes/0.js +8 -0
  526. package/web/.svelte-kit/output/server/nodes/1.js +8 -0
  527. package/web/.svelte-kit/output/server/nodes/2.js +8 -0
  528. package/web/.svelte-kit/output/server/nodes/3.js +10 -0
  529. package/web/.svelte-kit/output/server/nodes/4.js +8 -0
  530. package/web/.svelte-kit/output/server/remote-entry.js +541 -0
  531. package/web/.svelte-kit/tsconfig.json +58 -0
  532. package/web/.svelte-kit/types/route_meta_data.json +55 -0
  533. package/web/.svelte-kit/types/src/routes/$types.d.ts +24 -0
  534. package/web/.svelte-kit/types/src/routes/api/auth/login/$types.d.ts +10 -0
  535. package/web/.svelte-kit/types/src/routes/api/auth/logout/$types.d.ts +10 -0
  536. package/web/.svelte-kit/types/src/routes/api/beads/$types.d.ts +10 -0
  537. package/web/.svelte-kit/types/src/routes/api/browse/$types.d.ts +10 -0
  538. package/web/.svelte-kit/types/src/routes/api/chrome/$types.d.ts +10 -0
  539. package/web/.svelte-kit/types/src/routes/api/files/image/$types.d.ts +10 -0
  540. package/web/.svelte-kit/types/src/routes/api/health/$types.d.ts +10 -0
  541. package/web/.svelte-kit/types/src/routes/api/projects/new-session/$types.d.ts +10 -0
  542. package/web/.svelte-kit/types/src/routes/api/sessions/$types.d.ts +10 -0
  543. package/web/.svelte-kit/types/src/routes/api/sessions/[id]/$types.d.ts +11 -0
  544. package/web/.svelte-kit/types/src/routes/api/sessions/[id]/kill/$types.d.ts +11 -0
  545. package/web/.svelte-kit/types/src/routes/api/sessions/[id]/restart/$types.d.ts +11 -0
  546. package/web/.svelte-kit/types/src/routes/api/sessions/[id]/screenshots/$types.d.ts +11 -0
  547. package/web/.svelte-kit/types/src/routes/api/sessions/[id]/send/$types.d.ts +11 -0
  548. package/web/.svelte-kit/types/src/routes/api/sessions/[target]/output/$types.d.ts +11 -0
  549. package/web/.svelte-kit/types/src/routes/api/tmux/panes/$types.d.ts +10 -0
  550. package/web/.svelte-kit/types/src/routes/login/$types.d.ts +25 -0
  551. package/web/.svelte-kit/types/src/routes/login/proxy+page.server.ts +19 -0
  552. package/web/.svelte-kit/types/src/routes/session/[target]/$types.d.ts +19 -0
  553. package/web/README.md +42 -0
  554. package/web/components.json +16 -0
  555. package/web/package.json +35 -0
  556. package/web/src/app.css +128 -0
  557. package/web/src/app.d.ts +13 -0
  558. package/web/src/app.html +11 -0
  559. package/web/src/hooks.server.ts +156 -0
  560. package/web/src/lib/assets/favicon.svg +1 -0
  561. package/web/src/lib/components/AllSessionsPanel.svelte +789 -0
  562. package/web/src/lib/components/BeadsPanel.svelte +146 -0
  563. package/web/src/lib/components/IssueItem.svelte +287 -0
  564. package/web/src/lib/components/ScreenshotsPanel.svelte +336 -0
  565. package/web/src/lib/components/SessionsSidebar.svelte +312 -0
  566. package/web/src/lib/components/TerminalRenderer.svelte +189 -0
  567. package/web/src/lib/components/ui/alert-dialog/alert-dialog-action.svelte +18 -0
  568. package/web/src/lib/components/ui/alert-dialog/alert-dialog-cancel.svelte +18 -0
  569. package/web/src/lib/components/ui/alert-dialog/alert-dialog-content.svelte +29 -0
  570. package/web/src/lib/components/ui/alert-dialog/alert-dialog-description.svelte +17 -0
  571. package/web/src/lib/components/ui/alert-dialog/alert-dialog-footer.svelte +20 -0
  572. package/web/src/lib/components/ui/alert-dialog/alert-dialog-header.svelte +20 -0
  573. package/web/src/lib/components/ui/alert-dialog/alert-dialog-overlay.svelte +20 -0
  574. package/web/src/lib/components/ui/alert-dialog/alert-dialog-portal.svelte +7 -0
  575. package/web/src/lib/components/ui/alert-dialog/alert-dialog-title.svelte +17 -0
  576. package/web/src/lib/components/ui/alert-dialog/alert-dialog-trigger.svelte +7 -0
  577. package/web/src/lib/components/ui/alert-dialog/alert-dialog.svelte +7 -0
  578. package/web/src/lib/components/ui/alert-dialog/index.ts +37 -0
  579. package/web/src/lib/components/ui/badge/badge.svelte +50 -0
  580. package/web/src/lib/components/ui/badge/index.ts +2 -0
  581. package/web/src/lib/components/ui/button/button.svelte +86 -0
  582. package/web/src/lib/components/ui/button/index.ts +17 -0
  583. package/web/src/lib/components/ui/checkbox/checkbox.svelte +36 -0
  584. package/web/src/lib/components/ui/checkbox/index.ts +6 -0
  585. package/web/src/lib/components/ui/dialog/dialog-close.svelte +7 -0
  586. package/web/src/lib/components/ui/dialog/dialog-content.svelte +45 -0
  587. package/web/src/lib/components/ui/dialog/dialog-description.svelte +17 -0
  588. package/web/src/lib/components/ui/dialog/dialog-footer.svelte +20 -0
  589. package/web/src/lib/components/ui/dialog/dialog-header.svelte +20 -0
  590. package/web/src/lib/components/ui/dialog/dialog-overlay.svelte +20 -0
  591. package/web/src/lib/components/ui/dialog/dialog-portal.svelte +7 -0
  592. package/web/src/lib/components/ui/dialog/dialog-title.svelte +17 -0
  593. package/web/src/lib/components/ui/dialog/dialog-trigger.svelte +7 -0
  594. package/web/src/lib/components/ui/dialog/dialog.svelte +7 -0
  595. package/web/src/lib/components/ui/dialog/index.ts +34 -0
  596. package/web/src/lib/components/ui/scroll-area/index.ts +10 -0
  597. package/web/src/lib/components/ui/scroll-area/scroll-area-scrollbar.svelte +31 -0
  598. package/web/src/lib/components/ui/scroll-area/scroll-area.svelte +43 -0
  599. package/web/src/lib/components/ui/textarea/index.ts +7 -0
  600. package/web/src/lib/components/ui/textarea/textarea.svelte +23 -0
  601. package/web/src/lib/index.ts +1 -0
  602. package/web/src/lib/server/auth.ts +90 -0
  603. package/web/src/lib/stores/beads.svelte.ts +163 -0
  604. package/web/src/lib/stores/input-injection.svelte.ts +39 -0
  605. package/web/src/lib/stores/preferences.svelte.ts +55 -0
  606. package/web/src/lib/stores/sessions.svelte.ts +108 -0
  607. package/web/src/lib/stores/terminal.svelte.ts +96 -0
  608. package/web/src/lib/stores/websocket-base.svelte.ts +209 -0
  609. package/web/src/lib/types/terminal.ts +31 -0
  610. package/web/src/lib/utils/terminal-parser.ts +239 -0
  611. package/web/src/lib/utils.ts +13 -0
  612. package/web/src/routes/+layout.svelte +450 -0
  613. package/web/src/routes/+page.svelte +19 -0
  614. package/web/src/routes/api/auth/login/+server.ts +34 -0
  615. package/web/src/routes/api/auth/logout/+server.ts +10 -0
  616. package/web/src/routes/api/beads/+server.ts +28 -0
  617. package/web/src/routes/api/browse/+server.ts +59 -0
  618. package/web/src/routes/api/chrome/+server.ts +35 -0
  619. package/web/src/routes/api/files/image/+server.ts +64 -0
  620. package/web/src/routes/api/health/+server.ts +6 -0
  621. package/web/src/routes/api/projects/new-session/+server.ts +50 -0
  622. package/web/src/routes/api/sessions/+server.ts +62 -0
  623. package/web/src/routes/api/sessions/[id]/+server.ts +19 -0
  624. package/web/src/routes/api/sessions/[id]/kill/+server.ts +46 -0
  625. package/web/src/routes/api/sessions/[id]/restart/+server.ts +59 -0
  626. package/web/src/routes/api/sessions/[id]/screenshots/+server.ts +32 -0
  627. package/web/src/routes/api/sessions/[id]/send/+server.ts +31 -0
  628. package/web/src/routes/api/sessions/[target]/output/+server.ts +13 -0
  629. package/web/src/routes/api/tmux/panes/+server.ts +32 -0
  630. package/web/src/routes/login/+page.server.ts +18 -0
  631. package/web/src/routes/login/+page.svelte +69 -0
  632. package/web/src/routes/session/[target]/+page.svelte +450 -0
  633. package/web/static/robots.txt +3 -0
  634. package/web/svelte.config.js +20 -0
  635. package/web/tsconfig.json +20 -0
  636. package/web/vite.config.ts +154 -0
@@ -0,0 +1,1143 @@
1
+ import { Hono } from "hono";
2
+ import { html } from "hono/html";
3
+ import { sessionsRoutes } from "./routes/sessions.js";
4
+ import { streamRoute } from "./routes/stream.js";
5
+ import { corsMiddleware } from "./middleware/cors.js";
6
+ import { DEFAULT_SERVER_PORT } from "../utils/paths.js";
7
+ import { deleteSession, upsertSession } from "../db/index.js";
8
+ export function createApp() {
9
+ const app = new Hono();
10
+ app.use("*", corsMiddleware());
11
+ app.get("/health", (c) => c.json({ status: "ok", timestamp: Date.now() }));
12
+ // Simple HTML dashboard (mobile-first)
13
+ app.get("/", (c) => {
14
+ return c.html(html `
15
+ <!DOCTYPE html>
16
+ <html>
17
+ <head>
18
+ <title>claude-watch</title>
19
+ <meta name="viewport" content="width=device-width, initial-scale=1, user-scalable=no">
20
+ <meta name="apple-mobile-web-app-capable" content="yes">
21
+ <script src="https://code.iconify.design/iconify-icon/2.1.0/iconify-icon.min.js"></script>
22
+ <style>
23
+ iconify-icon { font-size: 18px; }
24
+ iconify-icon.spin { animation: spin 1s linear infinite; }
25
+ @keyframes spin { from { transform: rotate(0deg); } to { transform: rotate(360deg); } }
26
+ * { box-sizing: border-box; }
27
+ body {
28
+ font-family: -apple-system, system-ui, sans-serif;
29
+ background: #0a0a0a;
30
+ color: #eee;
31
+ padding: 16px;
32
+ margin: 0;
33
+ min-height: 100vh;
34
+ }
35
+ .header {
36
+ display: flex;
37
+ justify-content: space-between;
38
+ align-items: center;
39
+ margin-bottom: 20px;
40
+ padding-bottom: 16px;
41
+ border-bottom: 1px solid #222;
42
+ }
43
+ h1 { color: #fff; font-size: 24px; margin: 0; }
44
+ .btn {
45
+ background: #333;
46
+ color: #fff;
47
+ border: none;
48
+ padding: 12px 24px;
49
+ font-size: 16px;
50
+ cursor: pointer;
51
+ border-radius: 8px;
52
+ touch-action: manipulation;
53
+ }
54
+ .btn:active { background: #444; }
55
+ .btn.paused { background: #c0392b; }
56
+ .session {
57
+ padding: 16px;
58
+ margin: 12px 0;
59
+ background: #1a1a1a;
60
+ border-radius: 12px;
61
+ border-left: 4px solid #444;
62
+ touch-action: manipulation;
63
+ display: flex;
64
+ align-items: center;
65
+ gap: 12px;
66
+ }
67
+ .session.permission, .session.waiting { border-left-color: #e74c3c; }
68
+ .session.idle { border-left-color: #f39c12; }
69
+ .session.busy { border-left-color: #27ae60; }
70
+ .session-header { display: flex; align-items: center; margin-bottom: 8px; }
71
+ .state {
72
+ width: 14px;
73
+ height: 14px;
74
+ border-radius: 50%;
75
+ margin-right: 12px;
76
+ flex-shrink: 0;
77
+ }
78
+ .target { font-weight: 600; font-size: 18px; overflow: hidden; text-overflow: ellipsis; white-space: nowrap; }
79
+ .action {
80
+ color: #aaa;
81
+ font-size: 14px;
82
+ overflow: hidden;
83
+ text-overflow: ellipsis;
84
+ white-space: nowrap;
85
+ }
86
+ .cwd {
87
+ color: #666;
88
+ font-size: 13px;
89
+ word-break: break-all;
90
+ }
91
+ .empty { color: #555; text-align: center; padding: 40px; font-size: 16px; }
92
+ .count { color: #666; font-size: 14px; margin: 0 0 16px 0; }
93
+ .project-group { margin-bottom: 20px; }
94
+ .project-header {
95
+ display: flex;
96
+ align-items: center;
97
+ gap: 10px;
98
+ padding: 10px 14px;
99
+ border-radius: 8px 8px 0 0;
100
+ font-weight: 700;
101
+ font-size: 16px;
102
+ }
103
+ .project-header .dot {
104
+ width: 12px;
105
+ height: 12px;
106
+ border-radius: 50%;
107
+ flex-shrink: 0;
108
+ }
109
+ .project-header button {
110
+ background: transparent;
111
+ border: none;
112
+ font-size: 18px;
113
+ cursor: pointer;
114
+ color: #888;
115
+ padding: 0;
116
+ display: flex;
117
+ align-items: center;
118
+ justify-content: center;
119
+ }
120
+ .project-header button:hover { color: #fff; }
121
+ .project-header button.delete:hover { color: #e74c3c; }
122
+ .project-sessions .session {
123
+ border-radius: 0;
124
+ margin: 0;
125
+ border-top: 1px solid #333;
126
+ }
127
+ .project-sessions .session:last-child {
128
+ border-radius: 0 0 8px 8px;
129
+ }
130
+ .session-info { cursor: default; flex: 1; min-width: 0; }
131
+ .session-info.clickable { cursor: pointer; }
132
+ .session-info.clickable:hover { background: #252525; border-radius: 8px; }
133
+ .session-info.clickable:active { opacity: 0.7; }
134
+ .actions { display: flex; gap: 6px; flex-shrink: 0; }
135
+ .actions button {
136
+ background: #222;
137
+ color: #aaa;
138
+ border: none;
139
+ padding: 8px 10px;
140
+ font-size: 16px;
141
+ border-radius: 8px;
142
+ cursor: pointer;
143
+ display: flex;
144
+ align-items: center;
145
+ justify-content: center;
146
+ touch-action: manipulation;
147
+ -webkit-tap-highlight-color: rgba(255,255,255,0.2);
148
+ transition: all 0.15s;
149
+ }
150
+ .actions button:hover { color: #fff; background: #333; }
151
+ .actions button:active { background: #444; transform: scale(0.95); }
152
+ .project-header button {
153
+ min-width: 44px;
154
+ min-height: 44px;
155
+ touch-action: manipulation;
156
+ -webkit-tap-highlight-color: rgba(255,255,255,0.2);
157
+ }
158
+ .actions button.danger { background: #7f1d1d; color: #fca5a5; }
159
+ .actions button.danger:hover { background: #991b1b; color: #fff; }
160
+
161
+ /* Folder Browser Modal */
162
+ .folder-modal {
163
+ position: fixed;
164
+ top: 0;
165
+ left: 0;
166
+ right: 0;
167
+ bottom: 0;
168
+ background: rgba(0,0,0,0.85);
169
+ display: flex;
170
+ align-items: center;
171
+ justify-content: center;
172
+ z-index: 100;
173
+ }
174
+ .folder-browser {
175
+ background: #1a1a1a;
176
+ border-radius: 12px;
177
+ width: 94%;
178
+ max-width: 500px;
179
+ max-height: 80vh;
180
+ display: flex;
181
+ flex-direction: column;
182
+ overflow: hidden;
183
+ }
184
+ .fb-header {
185
+ padding: 16px;
186
+ border-bottom: 1px solid #333;
187
+ display: flex;
188
+ justify-content: space-between;
189
+ align-items: center;
190
+ }
191
+ .fb-header h3 { margin: 0; font-size: 18px; }
192
+ .fb-close {
193
+ background: transparent;
194
+ border: none;
195
+ color: #888;
196
+ font-size: 20px;
197
+ cursor: pointer;
198
+ padding: 4px;
199
+ }
200
+ .fb-close:hover { color: #fff; }
201
+ .fb-breadcrumbs {
202
+ padding: 12px 16px;
203
+ background: #111;
204
+ overflow-x: auto;
205
+ white-space: nowrap;
206
+ display: flex;
207
+ align-items: center;
208
+ gap: 4px;
209
+ font-size: 14px;
210
+ -webkit-overflow-scrolling: touch;
211
+ }
212
+ .fb-breadcrumbs span {
213
+ color: #666;
214
+ flex-shrink: 0;
215
+ }
216
+ .fb-crumb {
217
+ color: #3498db;
218
+ cursor: pointer;
219
+ flex-shrink: 0;
220
+ padding: 4px 8px;
221
+ border-radius: 4px;
222
+ }
223
+ .fb-crumb:hover { background: #222; }
224
+ .fb-crumb:last-of-type { color: #fff; cursor: default; }
225
+ .fb-crumb:last-of-type:hover { background: transparent; }
226
+ .fb-options {
227
+ padding: 8px 16px;
228
+ border-bottom: 1px solid #333;
229
+ display: flex;
230
+ align-items: center;
231
+ gap: 8px;
232
+ font-size: 14px;
233
+ color: #888;
234
+ }
235
+ .fb-options input[type="checkbox"] {
236
+ width: 18px;
237
+ height: 18px;
238
+ accent-color: #3498db;
239
+ }
240
+ .fb-list {
241
+ flex: 1;
242
+ overflow-y: auto;
243
+ min-height: 200px;
244
+ max-height: 40vh;
245
+ -webkit-overflow-scrolling: touch;
246
+ }
247
+ .fb-folder {
248
+ padding: 14px 16px;
249
+ display: flex;
250
+ align-items: center;
251
+ gap: 12px;
252
+ cursor: pointer;
253
+ border-bottom: 1px solid #222;
254
+ transition: background 0.1s;
255
+ }
256
+ .fb-folder:hover { background: #252525; }
257
+ .fb-folder:active { background: #333; }
258
+ .fb-folder iconify-icon { color: #f39c12; font-size: 20px; }
259
+ .fb-folder span { flex: 1; overflow: hidden; text-overflow: ellipsis; }
260
+ .fb-folder .fb-arrow { color: #555; font-size: 16px; }
261
+ .fb-empty {
262
+ padding: 40px;
263
+ text-align: center;
264
+ color: #555;
265
+ }
266
+ .fb-loading {
267
+ padding: 40px;
268
+ text-align: center;
269
+ color: #888;
270
+ }
271
+ .fb-error {
272
+ padding: 20px;
273
+ text-align: center;
274
+ color: #e74c3c;
275
+ background: #2a1a1a;
276
+ }
277
+ .fb-footer {
278
+ padding: 16px;
279
+ border-top: 1px solid #333;
280
+ display: flex;
281
+ gap: 10px;
282
+ }
283
+ .fb-footer button {
284
+ flex: 1;
285
+ padding: 14px;
286
+ border: none;
287
+ border-radius: 8px;
288
+ font-size: 16px;
289
+ cursor: pointer;
290
+ touch-action: manipulation;
291
+ }
292
+ .fb-cancel { background: #333; color: #fff; }
293
+ .fb-cancel:hover { background: #444; }
294
+ .fb-select { background: #3498db; color: #fff; }
295
+ .fb-select:hover { background: #2980b9; }
296
+ </style>
297
+ </head>
298
+ <body>
299
+ <div class="header">
300
+ <h1>claude-watch</h1>
301
+ <button class="btn" onclick="newProject()"><iconify-icon icon="ph:folder-plus"></iconify-icon></button>
302
+ <button class="btn" id="toggleBtn" onclick="toggle()"><iconify-icon icon="ph:pause"></iconify-icon></button>
303
+ </div>
304
+ <p class="count" id="count">Loading...</p>
305
+ <div id="sessions"></div>
306
+ <script>
307
+ let paused = false;
308
+ let lastData = '';
309
+
310
+ function stateColor(s) {
311
+ if (s === 'permission' || s === 'waiting') return '#e74c3c';
312
+ if (s === 'idle') return '#f39c12';
313
+ if (s === 'busy') return '#27ae60';
314
+ return '#555';
315
+ }
316
+
317
+ const projectColors = ['#3498db', '#9b59b6', '#1abc9c', '#e67e22', '#e91e63', '#00bcd4', '#8bc34a', '#ff5722'];
318
+
319
+ function getProjectColor(cwd) {
320
+ let hash = 0;
321
+ for (let i = 0; i < cwd.length; i++) {
322
+ hash = ((hash << 5) - hash) + cwd.charCodeAt(i);
323
+ hash = hash & hash;
324
+ }
325
+ return projectColors[Math.abs(hash) % projectColors.length];
326
+ }
327
+
328
+ function getProjectName(cwd) {
329
+ return cwd.split('/').filter(Boolean).pop() || cwd;
330
+ }
331
+
332
+ // Saved projects in localStorage
333
+ function getSavedProjects() {
334
+ try {
335
+ return JSON.parse(localStorage.getItem('claude-watch-projects') || '[]');
336
+ } catch { return []; }
337
+ }
338
+
339
+ function saveProject(cwd) {
340
+ const projects = getSavedProjects();
341
+ if (!projects.includes(cwd)) {
342
+ projects.push(cwd);
343
+ localStorage.setItem('claude-watch-projects', JSON.stringify(projects));
344
+ }
345
+ }
346
+
347
+ function removeProject(cwd) {
348
+ const projects = getSavedProjects().filter(p => p !== cwd);
349
+ localStorage.setItem('claude-watch-projects', JSON.stringify(projects));
350
+ }
351
+
352
+ async function deleteProject(cwd, sessions) {
353
+ // Kill all sessions in this project
354
+ for (const s of sessions) {
355
+ await fetch('/api/sessions/' + encodeURIComponent(s.id) + '/kill', {
356
+ method: 'POST',
357
+ headers: { 'Content-Type': 'application/json' },
358
+ body: JSON.stringify({ pid: s.pid, tmux_target: s.tmux_target })
359
+ });
360
+ }
361
+ // Remove from localStorage
362
+ removeProject(cwd);
363
+ lastData = ''; // Force refresh
364
+ refresh();
365
+ }
366
+
367
+ async function refresh() {
368
+ if (paused) return;
369
+ try {
370
+ const res = await fetch('/api/sessions');
371
+ const data = await res.json();
372
+ const dataStr = JSON.stringify(data.sessions);
373
+ const savedProjects = getSavedProjects();
374
+ const cacheKey = dataStr + JSON.stringify(savedProjects);
375
+ if (cacheKey === lastData) return;
376
+ lastData = cacheKey;
377
+ document.getElementById('count').textContent = data.count + ' session(s)';
378
+ const el = document.getElementById('sessions');
379
+
380
+ // Group sessions by project (cwd)
381
+ const groups = {};
382
+ data.sessions.forEach(s => {
383
+ const key = s.cwd || 'unknown';
384
+ if (!groups[key]) groups[key] = [];
385
+ groups[key].push(s);
386
+ // Auto-save projects with active sessions
387
+ saveProject(key);
388
+ });
389
+
390
+ // Add saved projects that have no active sessions
391
+ savedProjects.forEach(cwd => {
392
+ if (!groups[cwd]) groups[cwd] = [];
393
+ });
394
+
395
+ // Sort projects alphabetically
396
+ const sortedProjects = Object.keys(groups).sort();
397
+
398
+ if (sortedProjects.length === 0) {
399
+ el.innerHTML = '<p class="empty">No projects. Click + to add one.</p>';
400
+ } else {
401
+ el.innerHTML = sortedProjects.map(cwd => {
402
+ const sessions = groups[cwd];
403
+ const color = getProjectColor(cwd);
404
+ const name = getProjectName(cwd);
405
+ const sessionsJson = JSON.stringify(sessions).replace(/'/g, "\\\\'").replace(/"/g, '&quot;');
406
+ return '<div class="project-group">' +
407
+ '<div class="project-header" style="background:' + color + '22;border-left:4px solid ' + color + '">' +
408
+ '<span class="dot" style="background:' + color + '"></span>' +
409
+ '<span>' + name + '</span>' +
410
+ '<span style="font-weight:400;font-size:12px;color:#888;margin-left:auto">' + sessions.length + '</span>' +
411
+ '<button onclick="event.stopPropagation();newSessionInProject(\\'' + cwd.replace(/'/g, "\\\\'") + '\\')"><iconify-icon icon="ph:plus"></iconify-icon></button>' +
412
+ '<button class="delete" onclick="event.stopPropagation();deleteProject(\\'' + cwd.replace(/'/g, "\\\\'") + '\\', JSON.parse(decodeURIComponent(\\'' + encodeURIComponent(JSON.stringify(sessions)) + '\\')))"><iconify-icon icon="ph:trash"></iconify-icon></button>' +
413
+ '</div>' +
414
+ (sessions.length > 0 ? '<div class="project-sessions">' +
415
+ sessions.map(s => {
416
+ const isLocal = !s.tmux_target;
417
+ const targetDisplay = s.pane_title || s.tmux_target || '—';
418
+ const clickHandler = isLocal ? '' : 'onclick="openSession(\\'' + s.tmux_target + '\\')"';
419
+ const clickableClass = isLocal ? '' : ' clickable';
420
+ return '<div class="session ' + s.state + '" style="border-left-color:' + color + '">' +
421
+ '<div class="session-info' + clickableClass + '" ' + clickHandler + '>' +
422
+ '<div class="session-header">' +
423
+ '<span class="state" style="background:' + stateColor(s.state) + '"></span>' +
424
+ '<span class="target">' + targetDisplay + '</span>' +
425
+ '</div>' +
426
+ '<div class="action">' + (s.current_action || s.state) + '</div>' +
427
+ '</div>' +
428
+ '<div class="actions">' +
429
+ (isLocal ? '' : '<button onclick="send(\\'' + s.tmux_target + '\\', \\'Escape\\')"><iconify-icon icon="ph:stop"></iconify-icon></button>') +
430
+ (isLocal ? '' : '<button onclick="copyCmd(\\'' + s.tmux_target + '\\')"><iconify-icon icon="ph:copy"></iconify-icon></button>') +
431
+ '<button onclick="killSession(\\'' + s.id + '\\', ' + s.pid + ', \\'' + (s.tmux_target || '') + '\\', this)" class="danger"><iconify-icon icon="ph:skull"></iconify-icon></button>' +
432
+ '</div>' +
433
+ '</div>';
434
+ }).join('') +
435
+ '</div>' : '') +
436
+ '</div>';
437
+ }).join('');
438
+ }
439
+ } catch (e) { console.error(e); }
440
+ }
441
+
442
+ function toggle() {
443
+ paused = !paused;
444
+ const btn = document.getElementById('toggleBtn');
445
+ btn.innerHTML = paused ? '<iconify-icon icon="ph:play"></iconify-icon>' : '<iconify-icon icon="ph:pause"></iconify-icon>';
446
+ btn.className = paused ? 'btn paused' : 'btn';
447
+ }
448
+
449
+ async function send(target, keys) {
450
+ await fetch('/api/sessions/' + encodeURIComponent(target) + '/send', {
451
+ method: 'POST',
452
+ headers: { 'Content-Type': 'application/json' },
453
+ body: JSON.stringify({ keys })
454
+ });
455
+ refresh();
456
+ }
457
+
458
+ async function copyCmd(target) {
459
+ const cmd = 'tmux attach -t "' + target + '"';
460
+ try {
461
+ await navigator.clipboard.writeText(cmd);
462
+ } catch (e) {}
463
+ }
464
+
465
+ async function killSession(id, pid, tmuxTarget, btn) {
466
+ if (btn) {
467
+ btn.disabled = true;
468
+ btn.innerHTML = '<iconify-icon icon="ph:spinner" class="spin"></iconify-icon>';
469
+ }
470
+ try {
471
+ const res = await fetch('/api/sessions/' + encodeURIComponent(id) + '/kill', {
472
+ method: 'POST',
473
+ headers: { 'Content-Type': 'application/json' },
474
+ body: JSON.stringify({ pid, tmux_target: tmuxTarget })
475
+ });
476
+ if (res.ok) {
477
+ // Remove from UI immediately
478
+ if (btn) btn.closest('.session').remove();
479
+ }
480
+ } catch (e) {
481
+ console.error('Kill failed:', e);
482
+ if (btn) {
483
+ btn.disabled = false;
484
+ btn.innerHTML = '<iconify-icon icon="ph:skull"></iconify-icon>';
485
+ }
486
+ }
487
+ refresh();
488
+ }
489
+
490
+ async function newSessionInProject(cwd) {
491
+ await fetch('/api/projects/new-session', {
492
+ method: 'POST',
493
+ headers: { 'Content-Type': 'application/json' },
494
+ body: JSON.stringify({ cwd })
495
+ });
496
+ // Force refresh even if paused
497
+ const wasPaused = paused;
498
+ paused = false;
499
+ await refresh();
500
+ paused = wasPaused;
501
+ }
502
+
503
+ async function newProject() {
504
+ showFolderBrowser((cwd) => {
505
+ fetch('/api/projects/new-session', {
506
+ method: 'POST',
507
+ headers: { 'Content-Type': 'application/json' },
508
+ body: JSON.stringify({ cwd })
509
+ }).then(refresh);
510
+ });
511
+ }
512
+
513
+ function showFolderBrowser(onSelect) {
514
+ let currentPath = '';
515
+ let showHidden = false;
516
+
517
+ const modal = document.createElement('div');
518
+ modal.id = 'folder-modal';
519
+ modal.className = 'folder-modal';
520
+ modal.innerHTML = '<div class="folder-browser">' +
521
+ '<div class="fb-header">' +
522
+ '<h3>Select Project Folder</h3>' +
523
+ '<button class="fb-close" onclick="closeFolderBrowser()"><iconify-icon icon="ph:x"></iconify-icon></button>' +
524
+ '</div>' +
525
+ '<div class="fb-breadcrumbs" id="fb-breadcrumbs"></div>' +
526
+ '<div class="fb-options">' +
527
+ '<input type="checkbox" id="fb-hidden">' +
528
+ '<label for="fb-hidden">Show hidden folders</label>' +
529
+ '</div>' +
530
+ '<div class="fb-list" id="fb-list"><div class="fb-loading"><iconify-icon icon="ph:spinner" class="spin"></iconify-icon> Loading...</div></div>' +
531
+ '<div class="fb-footer">' +
532
+ '<button class="fb-cancel" onclick="closeFolderBrowser()">Cancel</button>' +
533
+ '<button class="fb-select" id="fb-select">Select This Folder</button>' +
534
+ '</div>' +
535
+ '</div>';
536
+ document.body.appendChild(modal);
537
+
538
+ // Close on escape key
539
+ const escHandler = (e) => { if (e.key === 'Escape') closeFolderBrowser(); };
540
+ document.addEventListener('keydown', escHandler);
541
+
542
+ // Close on backdrop click
543
+ modal.onclick = (e) => { if (e.target === modal) closeFolderBrowser(); };
544
+
545
+ window.closeFolderBrowser = () => {
546
+ document.removeEventListener('keydown', escHandler);
547
+ modal.remove();
548
+ };
549
+
550
+ // Toggle hidden folders
551
+ document.getElementById('fb-hidden').onchange = (e) => {
552
+ showHidden = e.target.checked;
553
+ loadFolder(currentPath);
554
+ };
555
+
556
+ // Select current folder
557
+ document.getElementById('fb-select').onclick = () => {
558
+ if (currentPath) {
559
+ closeFolderBrowser();
560
+ onSelect(currentPath);
561
+ }
562
+ };
563
+
564
+ // Navigate to folder
565
+ window.navigateFolder = (path) => {
566
+ loadFolder(path);
567
+ };
568
+
569
+ async function loadFolder(path) {
570
+ const list = document.getElementById('fb-list');
571
+ const crumbs = document.getElementById('fb-breadcrumbs');
572
+ list.innerHTML = '<div class="fb-loading"><iconify-icon icon="ph:spinner" class="spin"></iconify-icon> Loading...</div>';
573
+
574
+ try {
575
+ const url = '/api/browse' + (path ? '?path=' + encodeURIComponent(path) : '') + (showHidden ? (path ? '&' : '?') + 'showHidden=true' : '');
576
+ const res = await fetch(url);
577
+ const data = await res.json();
578
+
579
+ if (data.error) {
580
+ list.innerHTML = '<div class="fb-error"><iconify-icon icon="ph:warning"></iconify-icon> ' + data.error + '</div>';
581
+ return;
582
+ }
583
+
584
+ currentPath = data.current;
585
+
586
+ // Build breadcrumbs
587
+ const parts = data.current.split('/').filter(Boolean);
588
+ let breadcrumbHtml = '<span class="fb-crumb" onclick="navigateFolder(\\'/\\')"><iconify-icon icon="ph:house"></iconify-icon></span>';
589
+ let buildPath = '';
590
+ for (let i = 0; i < parts.length; i++) {
591
+ buildPath += '/' + parts[i];
592
+ const isLast = i === parts.length - 1;
593
+ breadcrumbHtml += '<span>/</span>';
594
+ if (isLast) {
595
+ breadcrumbHtml += '<span class="fb-crumb">' + parts[i] + '</span>';
596
+ } else {
597
+ breadcrumbHtml += '<span class="fb-crumb" onclick="navigateFolder(\\'' + buildPath.replace(/'/g, "\\\\'") + '\\')">' + parts[i] + '</span>';
598
+ }
599
+ }
600
+ crumbs.innerHTML = breadcrumbHtml;
601
+
602
+ // Build folder list
603
+ if (data.folders.length === 0) {
604
+ list.innerHTML = '<div class="fb-empty"><iconify-icon icon="ph:folder-open"></iconify-icon><br>No subfolders</div>';
605
+ } else {
606
+ list.innerHTML = data.folders.map(f =>
607
+ '<div class="fb-folder" onclick="navigateFolder(\\'' + f.path.replace(/'/g, "\\\\'") + '\\')">' +
608
+ '<iconify-icon icon="ph:folder-fill"></iconify-icon>' +
609
+ '<span>' + f.name + '</span>' +
610
+ '<iconify-icon icon="ph:caret-right" class="fb-arrow"></iconify-icon>' +
611
+ '</div>'
612
+ ).join('');
613
+ }
614
+ } catch (e) {
615
+ list.innerHTML = '<div class="fb-error"><iconify-icon icon="ph:warning"></iconify-icon> Failed to load folder</div>';
616
+ }
617
+ }
618
+
619
+ // Start with home directory
620
+ loadFolder('');
621
+ }
622
+
623
+ function openSession(target) {
624
+ window.location.href = '/session/' + encodeURIComponent(target);
625
+ }
626
+
627
+ refresh();
628
+ setInterval(refresh, 1000);
629
+ </script>
630
+ </body>
631
+ </html>
632
+ `);
633
+ });
634
+ // Send keys to tmux pane
635
+ app.post("/api/sessions/:target/send", async (c) => {
636
+ const target = decodeURIComponent(c.req.param("target"));
637
+ const body = await c.req.json();
638
+ const keys = body.keys || "Escape";
639
+ const text = body.text; // For literal text input
640
+ try {
641
+ const { execSync } = await import("child_process");
642
+ if (text) {
643
+ // Send literal text then Enter
644
+ execSync(`tmux send-keys -t "${target}" -l ${JSON.stringify(text)}`, { stdio: "ignore" });
645
+ execSync(`tmux send-keys -t "${target}" Enter`, { stdio: "ignore" });
646
+ }
647
+ else {
648
+ execSync(`tmux send-keys -t "${target}" ${keys}`, { stdio: "ignore" });
649
+ }
650
+ return c.json({ ok: true });
651
+ }
652
+ catch {
653
+ return c.json({ error: "Failed to send keys" }, 500);
654
+ }
655
+ });
656
+ // Browse directories for folder picker
657
+ app.get("/api/browse", async (c) => {
658
+ const os = await import("os");
659
+ const fs = await import("fs");
660
+ const path = await import("path");
661
+ let targetPath = c.req.query("path") || os.homedir();
662
+ const showHidden = c.req.query("showHidden") === "true";
663
+ // Normalize and resolve path
664
+ if (targetPath.startsWith("~")) {
665
+ targetPath = targetPath.replace("~", os.homedir());
666
+ }
667
+ targetPath = path.resolve(targetPath);
668
+ try {
669
+ const stat = fs.statSync(targetPath);
670
+ if (!stat.isDirectory()) {
671
+ return c.json({ error: "Not a directory" }, 400);
672
+ }
673
+ const entries = fs.readdirSync(targetPath, { withFileTypes: true });
674
+ const folders = entries
675
+ .filter((e) => {
676
+ if (!e.isDirectory())
677
+ return false;
678
+ if (!showHidden && e.name.startsWith("."))
679
+ return false;
680
+ return true;
681
+ })
682
+ .map((e) => ({
683
+ name: e.name,
684
+ path: path.join(targetPath, e.name),
685
+ }))
686
+ .sort((a, b) => a.name.toLowerCase().localeCompare(b.name.toLowerCase()));
687
+ const parent = path.dirname(targetPath);
688
+ const isRoot = targetPath === "/" || targetPath === parent;
689
+ return c.json({
690
+ current: targetPath,
691
+ parent: isRoot ? null : parent,
692
+ isRoot,
693
+ folders,
694
+ });
695
+ }
696
+ catch (err) {
697
+ if (err.code === "ENOENT") {
698
+ return c.json({ error: "Directory not found" }, 404);
699
+ }
700
+ if (err.code === "EACCES") {
701
+ return c.json({ error: "Permission denied" }, 403);
702
+ }
703
+ return c.json({ error: "Failed to read directory" }, 500);
704
+ }
705
+ });
706
+ // Kill session (tmux or local process)
707
+ app.post("/api/sessions/:id/kill", async (c) => {
708
+ const id = decodeURIComponent(c.req.param("id"));
709
+ const body = await c.req.json().catch(() => ({}));
710
+ const { pid, tmux_target } = body;
711
+ try {
712
+ const { execSync } = await import("child_process");
713
+ if (tmux_target) {
714
+ // Kill tmux session
715
+ const session = tmux_target.split(":")[0];
716
+ execSync(`tmux kill-session -t "${session}" 2>/dev/null || true`, { stdio: "ignore" });
717
+ }
718
+ else if (pid && pid > 0) {
719
+ // Kill process directly for non-tmux sessions
720
+ execSync(`kill ${pid} 2>/dev/null || true`, { stdio: "ignore" });
721
+ }
722
+ // Remove session file
723
+ try {
724
+ deleteSession(id);
725
+ }
726
+ catch {
727
+ // Cleanup is best-effort
728
+ }
729
+ return c.json({ ok: true });
730
+ }
731
+ catch {
732
+ // Even if kill fails, clean up session
733
+ try {
734
+ deleteSession(id);
735
+ }
736
+ catch {
737
+ // Ignore
738
+ }
739
+ return c.json({ ok: true });
740
+ }
741
+ });
742
+ // Create new session in project
743
+ app.post("/api/projects/new-session", async (c) => {
744
+ const { cwd } = await c.req.json();
745
+ if (!cwd) {
746
+ return c.json({ error: "cwd required" }, 400);
747
+ }
748
+ // Check if the folder exists
749
+ const fs = await import("fs");
750
+ if (!fs.existsSync(cwd)) {
751
+ return c.json({ error: "Folder does not exist" }, 400);
752
+ }
753
+ const stat = fs.statSync(cwd);
754
+ if (!stat.isDirectory()) {
755
+ return c.json({ error: "Path is not a directory" }, 400);
756
+ }
757
+ try {
758
+ const { execSync } = await import("child_process");
759
+ // Create new tmux window and run claude in the specified directory
760
+ const sessionName = "claude-" + Date.now();
761
+ const tmuxTarget = sessionName + ":1.1";
762
+ execSync(`tmux new-session -d -s "${sessionName}" -c "${cwd}" -- claude --dangerously-skip-permissions`, { stdio: "ignore" });
763
+ // Add session immediately so it shows up in UI
764
+ try {
765
+ const id = crypto.randomUUID();
766
+ console.log("[new-session] Creating session:", { id, cwd, tmuxTarget });
767
+ upsertSession({
768
+ id,
769
+ pid: 0,
770
+ cwd,
771
+ tmux_target: tmuxTarget,
772
+ state: "idle",
773
+ });
774
+ console.log("[new-session] Session created successfully");
775
+ }
776
+ catch (err) {
777
+ console.error("[new-session] Session creation failed:", err);
778
+ }
779
+ return c.json({ ok: true, session: sessionName });
780
+ }
781
+ catch (e) {
782
+ return c.json({ error: "Failed to create session" }, 500);
783
+ }
784
+ });
785
+ // Get tmux pane output
786
+ app.get("/api/sessions/:target/output", async (c) => {
787
+ const target = decodeURIComponent(c.req.param("target"));
788
+ try {
789
+ const { execSync } = await import("child_process");
790
+ const output = execSync(`tmux capture-pane -t "${target}" -p -S -100`, { encoding: "utf-8" });
791
+ return c.json({ output, timestamp: Date.now() });
792
+ }
793
+ catch {
794
+ return c.json({ error: "Failed to capture pane" }, 500);
795
+ }
796
+ });
797
+ // Session detail page (mobile-first chat UI)
798
+ app.get("/session/:target", (c) => {
799
+ const target = decodeURIComponent(c.req.param("target"));
800
+ return c.html(html `
801
+ <!DOCTYPE html>
802
+ <html>
803
+ <head>
804
+ <title>${target}</title>
805
+ <meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1, user-scalable=no">
806
+ <meta name="apple-mobile-web-app-capable" content="yes">
807
+ <script src="https://code.iconify.design/iconify-icon/2.1.0/iconify-icon.min.js"></script>
808
+ <style>
809
+ iconify-icon { font-size: 18px; }
810
+ * { box-sizing: border-box; margin: 0; padding: 0; }
811
+ html, body { height: 100%; overflow: hidden; }
812
+ body {
813
+ font-family: -apple-system, system-ui, sans-serif;
814
+ background: #000;
815
+ color: #fff;
816
+ display: flex;
817
+ flex-direction: column;
818
+ position: fixed;
819
+ width: 100%;
820
+ }
821
+ .header {
822
+ padding: 16px;
823
+ background: #111;
824
+ display: flex;
825
+ align-items: center;
826
+ gap: 16px;
827
+ position: sticky;
828
+ top: 0;
829
+ z-index: 10;
830
+ }
831
+ .back { background: #222; border: none; font-size: 16px; padding: 8px 10px; border-radius: 8px; color: #aaa; display: flex; align-items: center; justify-content: center; cursor: pointer; transition: all 0.15s; text-decoration: none; }
832
+ .back:hover { color: #fff; background: #333; }
833
+ .header-btns { display: flex; gap: 6px; margin-left: auto; align-items: center; }
834
+ .header-btns button { background: #222; border: none; font-size: 16px; padding: 8px 10px; border-radius: 8px; color: #aaa; display: flex; align-items: center; justify-content: center; cursor: pointer; transition: all 0.15s; }
835
+ .header-btns button:hover { color: #fff; background: #333; }
836
+ .header-btns button:active { background: #444; transform: scale(0.95); }
837
+ .header-btns button.stop { background: #b91c1c; color: #fff; }
838
+ .header-btns button.stop:hover { background: #dc2626; }
839
+ .header-btns button.kill { background: #581c87; color: #fff; }
840
+ .header-btns button.kill:hover { background: #7c3aed; }
841
+ .title-row { display: flex; align-items: center; gap: 8px; }
842
+ .state-dot { width: 12px; height: 12px; border-radius: 50%; flex-shrink: 0; background: #555; }
843
+ .title { font-size: 17px; font-weight: 600; }
844
+ .status { font-size: 13px; color: #888; }
845
+ .output {
846
+ flex: 1;
847
+ overflow-y: auto;
848
+ padding: 16px;
849
+ font-size: 15px;
850
+ line-height: 1.6;
851
+ background: #000;
852
+ -webkit-overflow-scrolling: touch;
853
+ }
854
+ .output pre {
855
+ white-space: pre;
856
+ font-family: ui-monospace, monospace;
857
+ font-size: 10px;
858
+ line-height: 1.3;
859
+ color: #ccc;
860
+ background: #111;
861
+ padding: 12px;
862
+ border-radius: 12px;
863
+ overflow: auto;
864
+ -webkit-overflow-scrolling: touch;
865
+ }
866
+ .toolbar {
867
+ display: flex;
868
+ background: #111;
869
+ padding: 4px;
870
+ gap: 2px;
871
+ }
872
+ .toolbar button {
873
+ flex: 1;
874
+ background: transparent;
875
+ border: none;
876
+ padding: 6px 0;
877
+ font-size: 14px;
878
+ color: #aaa;
879
+ cursor: pointer;
880
+ -webkit-tap-highlight-color: transparent;
881
+ touch-action: manipulation;
882
+ display: flex;
883
+ flex-direction: column;
884
+ align-items: center;
885
+ justify-content: center;
886
+ gap: 2px;
887
+ transition: all 0.15s;
888
+ border-radius: 6px;
889
+ }
890
+ .toolbar button iconify-icon { font-size: 20px; }
891
+ .toolbar button span { font-size: 9px; text-transform: uppercase; letter-spacing: 0.5px; }
892
+ .toolbar button:hover { color: #fff; background: #222; }
893
+ .toolbar button:active { background: #333; transform: scale(0.95); }
894
+ .input-row {
895
+ padding: 12px 16px 24px;
896
+ background: #111;
897
+ display: flex;
898
+ gap: 10px;
899
+ }
900
+ .input-row input {
901
+ flex: 1;
902
+ background: #222;
903
+ border: 2px solid transparent;
904
+ color: #fff;
905
+ padding: 14px 16px;
906
+ font-size: 16px;
907
+ border-radius: 24px;
908
+ outline: none;
909
+ transition: border-color 0.15s;
910
+ }
911
+ .input-row input:focus { border-color: #3b82f6; }
912
+ .input-row input::placeholder { color: #555; }
913
+ .input-row button {
914
+ background: #3b82f6;
915
+ color: #fff;
916
+ border: none;
917
+ width: 48px;
918
+ height: 48px;
919
+ font-size: 20px;
920
+ border-radius: 50%;
921
+ cursor: pointer;
922
+ transition: all 0.15s;
923
+ }
924
+ .input-row button:hover { background: #60a5fa; }
925
+ .input-row button:active { background: #2563eb; transform: scale(0.95); }
926
+ </style>
927
+ </head>
928
+ <body>
929
+ <div class="header">
930
+ <a href="/" class="back"><iconify-icon icon="ph:caret-left"></iconify-icon></a>
931
+ <span class="state-dot" id="stateDot"></span>
932
+ <div>
933
+ <div class="title" id="title">${target}</div>
934
+ <div class="status" id="status">Connecting...</div>
935
+ </div>
936
+ <div class="header-btns">
937
+ <button onclick="copyCmd()"><iconify-icon icon="ph:copy"></iconify-icon></button>
938
+ <button class="stop" onclick="send('Escape')"><iconify-icon icon="ph:hand-palm"></iconify-icon></button>
939
+ <button class="kill" onclick="killSession()"><iconify-icon icon="ph:power"></iconify-icon></button>
940
+ </div>
941
+ </div>
942
+ <div class="output" id="output">
943
+ <pre id="terminal">Loading...</pre>
944
+ </div>
945
+ <div class="toolbar">
946
+ <button onclick="send('Up')"><iconify-icon icon="ph:arrow-up"></iconify-icon><span>Up</span></button>
947
+ <button onclick="send('Down')"><iconify-icon icon="ph:arrow-down"></iconify-icon><span>Down</span></button>
948
+ <button onclick="send('Space')"><iconify-icon icon="ph:selection-all"></iconify-icon><span>Space</span></button>
949
+ <button onclick="send('Tab')"><iconify-icon icon="ph:arrow-elbow-down-right"></iconify-icon><span>Tab</span></button>
950
+ <button onclick="send('Enter')"><iconify-icon icon="ph:corner-down-left"></iconify-icon><span>Enter</span></button>
951
+ <button onclick="send('C-l')"><iconify-icon icon="ph:eraser"></iconify-icon><span>Clear</span></button>
952
+ <button onclick="send('C-c')"><iconify-icon icon="ph:command"></iconify-icon><span>Ctrl-C</span></button>
953
+ </div>
954
+ <div class="input-row">
955
+ <input type="text" id="input" placeholder="Send a message..." autocomplete="off" autocorrect="off">
956
+ <button onclick="sendInput()">↑</button>
957
+ </div>
958
+ <script>
959
+ const target = "${target}";
960
+ const terminal = document.getElementById('terminal');
961
+ const status = document.getElementById('status');
962
+ const stateDot = document.getElementById('stateDot');
963
+ const titleEl = document.getElementById('title');
964
+ const input = document.getElementById('input');
965
+ const output = document.getElementById('output');
966
+ let lastOutput = '';
967
+
968
+ function stateColor(s) {
969
+ if (s === 'permission' || s === 'waiting') return '#e74c3c';
970
+ if (s === 'idle') return '#f39c12';
971
+ if (s === 'busy') return '#27ae60';
972
+ return '#555';
973
+ }
974
+
975
+ async function refresh() {
976
+ try {
977
+ const [outputRes, sessionsRes] = await Promise.all([
978
+ fetch('/api/sessions/' + encodeURIComponent(target) + '/output'),
979
+ fetch('/api/sessions')
980
+ ]);
981
+ const data = await outputRes.json();
982
+ const sessionsData = await sessionsRes.json();
983
+
984
+ // Update state dot and title from sessions data
985
+ const session = sessionsData.sessions.find(s => s.tmux_target === target);
986
+ if (session) {
987
+ stateDot.style.background = stateColor(session.state);
988
+ status.textContent = session.current_action || session.state;
989
+ status.style.color = '#888';
990
+ titleEl.textContent = session.pane_title || target;
991
+ }
992
+
993
+ if (data.output) {
994
+ const trimmed = data.output.trim();
995
+ if (trimmed !== lastOutput) {
996
+ lastOutput = trimmed;
997
+ terminal.textContent = trimmed;
998
+ }
999
+ }
1000
+ } catch (e) {
1001
+ status.textContent = 'Disconnected';
1002
+ status.style.color = '#ef4444';
1003
+ stateDot.style.background = '#555';
1004
+ }
1005
+ }
1006
+
1007
+ async function send(keys) {
1008
+ status.textContent = 'Sending...';
1009
+ status.style.color = '#eab308';
1010
+ await fetch('/api/sessions/' + encodeURIComponent(target) + '/send', {
1011
+ method: 'POST',
1012
+ headers: { 'Content-Type': 'application/json' },
1013
+ body: JSON.stringify({ keys })
1014
+ });
1015
+ setTimeout(refresh, 300);
1016
+ }
1017
+
1018
+ async function sendInput() {
1019
+ const text = input.value.trim();
1020
+ input.value = '';
1021
+ if (!text) {
1022
+ await send('Enter');
1023
+ return;
1024
+ }
1025
+ status.textContent = 'Sending...';
1026
+ status.style.color = '#eab308';
1027
+ await fetch('/api/sessions/' + encodeURIComponent(target) + '/send', {
1028
+ method: 'POST',
1029
+ headers: { 'Content-Type': 'application/json' },
1030
+ body: JSON.stringify({ text })
1031
+ });
1032
+ setTimeout(refresh, 300);
1033
+ }
1034
+
1035
+ input.addEventListener('keypress', (e) => {
1036
+ if (e.key === 'Enter') sendInput();
1037
+ });
1038
+
1039
+ async function copyCmd() {
1040
+ const cmd = 'tmux attach -t "' + target + '"';
1041
+ try {
1042
+ await navigator.clipboard.writeText(cmd);
1043
+ status.textContent = 'Copied!';
1044
+ status.style.color = '#22c55e';
1045
+ setTimeout(refresh, 1000);
1046
+ } catch (e) {
1047
+ alert(cmd);
1048
+ }
1049
+ }
1050
+
1051
+ function killSession() {
1052
+ showConfirm('Kill Session', 'Kill this session?', async () => {
1053
+ await fetch('/api/sessions/' + encodeURIComponent(target) + '/kill', { method: 'POST' });
1054
+ window.location.href = '/';
1055
+ });
1056
+ }
1057
+
1058
+ function showConfirm(title, message, onConfirm) {
1059
+ const modal = document.createElement('div');
1060
+ modal.id = 'modal';
1061
+ modal.innerHTML = '<div style="position:fixed;top:0;left:0;right:0;bottom:0;background:rgba(0,0,0,0.8);display:flex;align-items:center;justify-content:center;z-index:100">' +
1062
+ '<div style="background:#222;padding:20px;border-radius:12px;width:90%;max-width:300px">' +
1063
+ '<div style="font-weight:600;margin-bottom:12px">' + title + '</div>' +
1064
+ '<div style="margin-bottom:16px;color:#aaa">' + message + '</div>' +
1065
+ '<div style="display:flex;gap:8px">' +
1066
+ '<button onclick="document.getElementById(\\'modal\\').remove()" style="flex:1;padding:12px;background:#444;border:none;border-radius:6px;color:#fff;font-size:16px">Cancel</button>' +
1067
+ '<button id="confirmOk" style="flex:1;padding:12px;background:#b91c1c;border:none;border-radius:6px;color:#fff;font-size:16px">Kill</button>' +
1068
+ '</div>' +
1069
+ '</div>' +
1070
+ '</div>';
1071
+ document.body.appendChild(modal);
1072
+ document.getElementById('confirmOk').onclick = () => {
1073
+ modal.remove();
1074
+ onConfirm();
1075
+ };
1076
+ }
1077
+
1078
+ function showModal(title, label, defaultVal, onSubmit) {
1079
+ const modal = document.createElement('div');
1080
+ modal.id = 'modal';
1081
+ modal.innerHTML = '<div style="position:fixed;top:0;left:0;right:0;bottom:0;background:rgba(0,0,0,0.8);display:flex;align-items:center;justify-content:center;z-index:100">' +
1082
+ '<div style="background:#222;padding:20px;border-radius:12px;width:90%;max-width:300px">' +
1083
+ '<div style="font-weight:600;margin-bottom:12px">' + title + '</div>' +
1084
+ '<div style="margin-bottom:8px">' + label + '</div>' +
1085
+ '<input type="text" id="modalInput" value="' + (defaultVal || '').replace(/"/g, '&quot;') + '" style="width:100%;padding:12px;border:1px solid #444;border-radius:6px;background:#111;color:#fff;font-size:16px;box-sizing:border-box">' +
1086
+ '<div style="display:flex;gap:8px;margin-top:12px">' +
1087
+ '<button onclick="document.getElementById(\\'modal\\').remove()" style="flex:1;padding:12px;background:#444;border:none;border-radius:6px;color:#fff;font-size:16px">Cancel</button>' +
1088
+ '<button id="modalOk" style="flex:1;padding:12px;background:#3498db;border:none;border-radius:6px;color:#fff;font-size:16px">OK</button>' +
1089
+ '</div>' +
1090
+ '</div>' +
1091
+ '</div>';
1092
+ document.body.appendChild(modal);
1093
+ const input = document.getElementById('modalInput');
1094
+ input.focus();
1095
+ input.select();
1096
+ document.getElementById('modalOk').onclick = () => {
1097
+ const val = input.value.trim();
1098
+ modal.remove();
1099
+ if (val) onSubmit(val);
1100
+ };
1101
+ input.onkeypress = (e) => { if (e.key === 'Enter') document.getElementById('modalOk').click(); };
1102
+ }
1103
+
1104
+ refresh();
1105
+ setInterval(refresh, 500);
1106
+ </script>
1107
+ </body>
1108
+ </html>
1109
+ `);
1110
+ });
1111
+ // Register stream route BEFORE sessions routes to prevent /:id from matching "stream"
1112
+ app.get("/api/sessions/stream", streamRoute);
1113
+ app.route("/api/sessions", sessionsRoutes);
1114
+ return app;
1115
+ }
1116
+ export function startServer(options = {}) {
1117
+ const port = options.port ?? parseInt(process.env.CLAUDE_WATCH_PORT ?? String(DEFAULT_SERVER_PORT));
1118
+ const host = options.host ?? "127.0.0.1";
1119
+ const app = createApp();
1120
+ console.log(`claude-watch server running at http://${host}:${port}`);
1121
+ console.log(` REST API: http://${host}:${port}/api/sessions`);
1122
+ console.log(` SSE stream: http://${host}:${port}/api/sessions/stream`);
1123
+ // Use Bun.serve if available, otherwise fall back to node adapter
1124
+ if (typeof Bun !== "undefined") {
1125
+ return Bun.serve({
1126
+ fetch: app.fetch,
1127
+ port,
1128
+ hostname: host,
1129
+ });
1130
+ }
1131
+ else {
1132
+ // Fallback for Node.js - dynamic import to avoid bundling issues
1133
+ return import("@hono/node-server").then(({ serve }) => {
1134
+ return serve({
1135
+ fetch: app.fetch,
1136
+ port,
1137
+ hostname: host,
1138
+ });
1139
+ });
1140
+ }
1141
+ }
1142
+ export { createApp as default };
1143
+ //# sourceMappingURL=index.js.map