skybridge 0.0.0-dev.f76ccdc → 0.0.0-dev.f792261

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 (419) hide show
  1. package/LICENSE +21 -674
  2. package/README.md +97 -142
  3. package/bin/run.js +5 -0
  4. package/dist/cli/detect-port.d.ts +18 -0
  5. package/dist/cli/detect-port.js +61 -0
  6. package/dist/cli/detect-port.js.map +1 -0
  7. package/dist/cli/header.d.ts +4 -0
  8. package/dist/cli/header.js +6 -0
  9. package/dist/cli/header.js.map +1 -0
  10. package/dist/cli/run-command.d.ts +2 -0
  11. package/dist/cli/run-command.js +43 -0
  12. package/dist/cli/run-command.js.map +1 -0
  13. package/dist/cli/telemetry.d.ts +7 -0
  14. package/dist/cli/telemetry.js +123 -0
  15. package/dist/cli/telemetry.js.map +1 -0
  16. package/dist/cli/tunnel-control-server.d.ts +9 -0
  17. package/dist/cli/tunnel-control-server.js +31 -0
  18. package/dist/cli/tunnel-control-server.js.map +1 -0
  19. package/dist/cli/tunnel-control-server.test.js +39 -0
  20. package/dist/cli/tunnel-control-server.test.js.map +1 -0
  21. package/dist/cli/tunnel-handler.d.ts +3 -0
  22. package/dist/cli/tunnel-handler.js +48 -0
  23. package/dist/cli/tunnel-handler.js.map +1 -0
  24. package/dist/cli/tunnel-handler.test.js +105 -0
  25. package/dist/cli/tunnel-handler.test.js.map +1 -0
  26. package/dist/cli/tunnel.d.ts +57 -0
  27. package/dist/cli/tunnel.js +154 -0
  28. package/dist/cli/tunnel.js.map +1 -0
  29. package/dist/cli/tunnel.test.js +190 -0
  30. package/dist/cli/tunnel.test.js.map +1 -0
  31. package/dist/cli/types.d.ts +5 -0
  32. package/dist/cli/types.js +2 -0
  33. package/dist/cli/types.js.map +1 -0
  34. package/dist/cli/use-execute-steps.d.ts +11 -0
  35. package/dist/cli/use-execute-steps.js +36 -0
  36. package/dist/cli/use-execute-steps.js.map +1 -0
  37. package/dist/cli/use-messages.d.ts +3 -0
  38. package/dist/cli/use-messages.js +11 -0
  39. package/dist/cli/use-messages.js.map +1 -0
  40. package/dist/cli/use-nodemon.d.ts +2 -0
  41. package/dist/cli/use-nodemon.js +73 -0
  42. package/dist/cli/use-nodemon.js.map +1 -0
  43. package/dist/cli/use-open-browser.d.ts +1 -0
  44. package/dist/cli/use-open-browser.js +44 -0
  45. package/dist/cli/use-open-browser.js.map +1 -0
  46. package/dist/cli/use-tunnel.d.ts +14 -0
  47. package/dist/cli/use-tunnel.js +131 -0
  48. package/dist/cli/use-tunnel.js.map +1 -0
  49. package/dist/cli/use-typescript-check.d.ts +9 -0
  50. package/dist/cli/use-typescript-check.js +94 -0
  51. package/dist/cli/use-typescript-check.js.map +1 -0
  52. package/dist/commands/build.d.ts +9 -0
  53. package/dist/commands/build.js +102 -0
  54. package/dist/commands/build.js.map +1 -0
  55. package/dist/commands/dev.d.ts +12 -0
  56. package/dist/commands/dev.js +80 -0
  57. package/dist/commands/dev.js.map +1 -0
  58. package/dist/commands/start.d.ts +9 -0
  59. package/dist/commands/start.js +49 -0
  60. package/dist/commands/start.js.map +1 -0
  61. package/dist/commands/telemetry/disable.d.ts +5 -0
  62. package/dist/commands/telemetry/disable.js +14 -0
  63. package/dist/commands/telemetry/disable.js.map +1 -0
  64. package/dist/commands/telemetry/enable.d.ts +5 -0
  65. package/dist/commands/telemetry/enable.js +14 -0
  66. package/dist/commands/telemetry/enable.js.map +1 -0
  67. package/dist/commands/telemetry/status.d.ts +5 -0
  68. package/dist/commands/telemetry/status.js +14 -0
  69. package/dist/commands/telemetry/status.js.map +1 -0
  70. package/dist/server/asset-base-url-transform-plugin.d.ts +10 -0
  71. package/dist/server/asset-base-url-transform-plugin.js +33 -0
  72. package/dist/server/asset-base-url-transform-plugin.js.map +1 -0
  73. package/dist/server/asset-base-url-transform-plugin.test.js +84 -0
  74. package/dist/server/asset-base-url-transform-plugin.test.js.map +1 -0
  75. package/dist/server/content-helpers.d.ts +27 -0
  76. package/dist/server/content-helpers.js +46 -0
  77. package/dist/server/content-helpers.js.map +1 -0
  78. package/dist/server/content-helpers.test.js +70 -0
  79. package/dist/server/content-helpers.test.js.map +1 -0
  80. package/dist/server/express.d.ts +11 -0
  81. package/dist/server/express.js +101 -0
  82. package/dist/server/express.js.map +1 -0
  83. package/dist/server/express.test.js +430 -0
  84. package/dist/server/express.test.js.map +1 -0
  85. package/dist/server/index.d.ts +6 -0
  86. package/dist/server/index.js +4 -0
  87. package/dist/server/index.js.map +1 -0
  88. package/dist/server/inferUtilityTypes.d.ts +64 -0
  89. package/dist/server/inferUtilityTypes.js +2 -0
  90. package/dist/server/inferUtilityTypes.js.map +1 -0
  91. package/dist/server/metric.d.ts +14 -0
  92. package/dist/server/metric.js +62 -0
  93. package/dist/server/metric.js.map +1 -0
  94. package/dist/server/middleware.d.ts +124 -0
  95. package/dist/server/middleware.js +93 -0
  96. package/dist/server/middleware.js.map +1 -0
  97. package/dist/server/middleware.test-d.js +75 -0
  98. package/dist/server/middleware.test-d.js.map +1 -0
  99. package/dist/server/middleware.test.js +493 -0
  100. package/dist/server/middleware.test.js.map +1 -0
  101. package/dist/server/server.d.ts +196 -0
  102. package/dist/server/server.js +468 -0
  103. package/dist/server/server.js.map +1 -0
  104. package/dist/{src/server → server}/templateHelper.d.ts +5 -4
  105. package/dist/server/templateHelper.js +11 -0
  106. package/dist/server/templateHelper.js.map +1 -0
  107. package/dist/server/templates.generated.d.ts +4 -0
  108. package/dist/server/templates.generated.js +47 -0
  109. package/dist/server/templates.generated.js.map +1 -0
  110. package/dist/server/tunnel-proxy-router.d.ts +7 -0
  111. package/dist/server/tunnel-proxy-router.js +110 -0
  112. package/dist/server/tunnel-proxy-router.js.map +1 -0
  113. package/dist/server/tunnel-proxy-router.test.js +229 -0
  114. package/dist/server/tunnel-proxy-router.test.js.map +1 -0
  115. package/dist/server/viewsDevServer.d.ts +14 -0
  116. package/dist/server/viewsDevServer.js +45 -0
  117. package/dist/server/viewsDevServer.js.map +1 -0
  118. package/dist/test/utils.d.ts +127 -0
  119. package/dist/test/utils.js +247 -0
  120. package/dist/test/utils.js.map +1 -0
  121. package/dist/test/view.test.js +523 -0
  122. package/dist/test/view.test.js.map +1 -0
  123. package/dist/version.d.ts +1 -0
  124. package/dist/version.js +3 -0
  125. package/dist/version.js.map +1 -0
  126. package/dist/web/bridges/apps-sdk/adaptor.d.ts +24 -0
  127. package/dist/web/bridges/apps-sdk/adaptor.js +96 -0
  128. package/dist/web/bridges/apps-sdk/adaptor.js.map +1 -0
  129. package/dist/web/bridges/apps-sdk/bridge.d.ts +10 -0
  130. package/dist/web/bridges/apps-sdk/bridge.js +46 -0
  131. package/dist/web/bridges/apps-sdk/bridge.js.map +1 -0
  132. package/dist/web/bridges/apps-sdk/index.d.ts +5 -0
  133. package/dist/web/bridges/apps-sdk/index.js +5 -0
  134. package/dist/web/bridges/apps-sdk/index.js.map +1 -0
  135. package/dist/web/bridges/apps-sdk/types.d.ts +131 -0
  136. package/dist/web/bridges/apps-sdk/types.js.map +1 -0
  137. package/dist/web/bridges/apps-sdk/use-apps-sdk-context.d.ts +2 -0
  138. package/dist/web/bridges/apps-sdk/use-apps-sdk-context.js +7 -0
  139. package/dist/web/bridges/apps-sdk/use-apps-sdk-context.js.map +1 -0
  140. package/dist/web/bridges/get-adaptor.d.ts +2 -0
  141. package/dist/web/bridges/get-adaptor.js +8 -0
  142. package/dist/web/bridges/get-adaptor.js.map +1 -0
  143. package/dist/web/bridges/index.d.ts +5 -0
  144. package/dist/web/bridges/index.js +6 -0
  145. package/dist/web/bridges/index.js.map +1 -0
  146. package/dist/web/bridges/mcp-app/adaptor.d.ts +48 -0
  147. package/dist/web/bridges/mcp-app/adaptor.js +263 -0
  148. package/dist/web/bridges/mcp-app/adaptor.js.map +1 -0
  149. package/dist/web/bridges/mcp-app/bridge.d.ts +26 -0
  150. package/dist/web/bridges/mcp-app/bridge.js +102 -0
  151. package/dist/web/bridges/mcp-app/bridge.js.map +1 -0
  152. package/dist/web/bridges/mcp-app/index.d.ts +4 -0
  153. package/dist/web/bridges/mcp-app/index.js +4 -0
  154. package/dist/web/bridges/mcp-app/index.js.map +1 -0
  155. package/dist/web/bridges/mcp-app/types.d.ts +8 -0
  156. package/dist/web/bridges/mcp-app/types.js +2 -0
  157. package/dist/web/bridges/mcp-app/types.js.map +1 -0
  158. package/dist/web/bridges/mcp-app/use-mcp-app-context.d.ts +7 -0
  159. package/dist/web/bridges/mcp-app/use-mcp-app-context.js +7 -0
  160. package/dist/web/bridges/mcp-app/use-mcp-app-context.js.map +1 -0
  161. package/dist/web/bridges/mcp-app/use-mcp-app-context.test.js +26 -0
  162. package/dist/web/bridges/mcp-app/use-mcp-app-context.test.js.map +1 -0
  163. package/dist/web/bridges/types.d.ts +111 -0
  164. package/dist/web/bridges/types.js +2 -0
  165. package/dist/web/bridges/types.js.map +1 -0
  166. package/dist/web/bridges/use-host-context.d.ts +2 -0
  167. package/dist/web/bridges/use-host-context.js +8 -0
  168. package/dist/web/bridges/use-host-context.js.map +1 -0
  169. package/dist/web/components/modal-provider.d.ts +4 -0
  170. package/dist/web/components/modal-provider.js +45 -0
  171. package/dist/web/components/modal-provider.js.map +1 -0
  172. package/dist/web/create-store.d.ts +3 -0
  173. package/dist/web/create-store.js +38 -0
  174. package/dist/web/create-store.js.map +1 -0
  175. package/dist/web/create-store.test.d.ts +1 -0
  176. package/dist/web/create-store.test.js +129 -0
  177. package/dist/web/create-store.test.js.map +1 -0
  178. package/dist/web/data-llm.d.ts +14 -0
  179. package/dist/web/data-llm.js +72 -0
  180. package/dist/web/data-llm.js.map +1 -0
  181. package/dist/web/data-llm.test.d.ts +1 -0
  182. package/dist/web/data-llm.test.js +142 -0
  183. package/dist/web/data-llm.test.js.map +1 -0
  184. package/dist/web/generate-helpers.d.ts +118 -0
  185. package/dist/web/generate-helpers.js +113 -0
  186. package/dist/web/generate-helpers.js.map +1 -0
  187. package/dist/web/generate-helpers.test-d.d.ts +1 -0
  188. package/dist/web/generate-helpers.test-d.js +209 -0
  189. package/dist/web/generate-helpers.test-d.js.map +1 -0
  190. package/dist/web/generate-helpers.test.d.ts +1 -0
  191. package/dist/web/generate-helpers.test.js +17 -0
  192. package/dist/web/generate-helpers.test.js.map +1 -0
  193. package/dist/web/helpers/state.d.ts +7 -0
  194. package/dist/web/helpers/state.js +45 -0
  195. package/dist/web/helpers/state.js.map +1 -0
  196. package/dist/web/helpers/state.test.d.ts +1 -0
  197. package/dist/web/helpers/state.test.js +53 -0
  198. package/dist/web/helpers/state.test.js.map +1 -0
  199. package/dist/web/hooks/index.d.ts +11 -0
  200. package/dist/web/hooks/index.js +12 -0
  201. package/dist/web/hooks/index.js.map +1 -0
  202. package/dist/web/hooks/test/utils.d.ts +16 -0
  203. package/dist/web/hooks/test/utils.js +64 -0
  204. package/dist/web/hooks/test/utils.js.map +1 -0
  205. package/dist/web/hooks/use-call-tool.d.ts +101 -0
  206. package/dist/web/hooks/use-call-tool.js +68 -0
  207. package/dist/web/hooks/use-call-tool.js.map +1 -0
  208. package/dist/web/hooks/use-call-tool.test-d.d.ts +1 -0
  209. package/dist/web/hooks/use-call-tool.test-d.js +104 -0
  210. package/dist/web/hooks/use-call-tool.test-d.js.map +1 -0
  211. package/dist/web/hooks/use-call-tool.test.d.ts +1 -0
  212. package/dist/web/hooks/use-call-tool.test.js +186 -0
  213. package/dist/web/hooks/use-call-tool.test.js.map +1 -0
  214. package/dist/web/hooks/use-display-mode.d.ts +4 -0
  215. package/dist/web/hooks/use-display-mode.js +9 -0
  216. package/dist/web/hooks/use-display-mode.js.map +1 -0
  217. package/dist/web/hooks/use-display-mode.test-d.d.ts +1 -0
  218. package/dist/web/hooks/use-display-mode.test-d.js +8 -0
  219. package/dist/web/hooks/use-display-mode.test-d.js.map +1 -0
  220. package/dist/web/hooks/use-display-mode.test.d.ts +1 -0
  221. package/dist/{src/web → web}/hooks/use-display-mode.test.js +3 -2
  222. package/dist/web/hooks/use-display-mode.test.js.map +1 -0
  223. package/dist/web/hooks/use-files.d.ts +7 -0
  224. package/dist/web/hooks/use-files.js +10 -0
  225. package/dist/web/hooks/use-files.js.map +1 -0
  226. package/dist/web/hooks/use-files.test.d.ts +1 -0
  227. package/dist/web/hooks/use-files.test.js +54 -0
  228. package/dist/web/hooks/use-files.test.js.map +1 -0
  229. package/dist/web/hooks/use-layout.d.ts +22 -0
  230. package/dist/web/hooks/use-layout.js +23 -0
  231. package/dist/web/hooks/use-layout.js.map +1 -0
  232. package/dist/web/hooks/use-layout.test.d.ts +1 -0
  233. package/dist/web/hooks/use-layout.test.js +96 -0
  234. package/dist/web/hooks/use-layout.test.js.map +1 -0
  235. package/dist/web/hooks/use-open-external.d.ts +3 -0
  236. package/dist/web/hooks/use-open-external.js +8 -0
  237. package/dist/web/hooks/use-open-external.js.map +1 -0
  238. package/dist/web/hooks/use-open-external.test.d.ts +1 -0
  239. package/dist/web/hooks/use-open-external.test.js +65 -0
  240. package/dist/web/hooks/use-open-external.test.js.map +1 -0
  241. package/dist/web/hooks/use-request-modal.d.ts +9 -0
  242. package/dist/web/hooks/use-request-modal.js +16 -0
  243. package/dist/web/hooks/use-request-modal.js.map +1 -0
  244. package/dist/web/hooks/use-request-modal.test.d.ts +1 -0
  245. package/dist/web/hooks/use-request-modal.test.js +61 -0
  246. package/dist/web/hooks/use-request-modal.test.js.map +1 -0
  247. package/dist/web/hooks/use-send-follow-up-message.d.ts +2 -0
  248. package/dist/web/hooks/use-send-follow-up-message.js +8 -0
  249. package/dist/web/hooks/use-send-follow-up-message.js.map +1 -0
  250. package/dist/web/hooks/use-set-open-in-app-url.d.ts +1 -0
  251. package/dist/web/hooks/use-set-open-in-app-url.js +8 -0
  252. package/dist/web/hooks/use-set-open-in-app-url.js.map +1 -0
  253. package/dist/web/hooks/use-set-open-in-app-url.test.d.ts +1 -0
  254. package/dist/web/hooks/use-set-open-in-app-url.test.js +43 -0
  255. package/dist/web/hooks/use-set-open-in-app-url.test.js.map +1 -0
  256. package/dist/web/hooks/use-tool-info.d.ts +36 -0
  257. package/dist/web/hooks/use-tool-info.js +26 -0
  258. package/dist/web/hooks/use-tool-info.js.map +1 -0
  259. package/dist/web/hooks/use-tool-info.test-d.d.ts +1 -0
  260. package/dist/web/hooks/use-tool-info.test-d.js +109 -0
  261. package/dist/web/hooks/use-tool-info.test-d.js.map +1 -0
  262. package/dist/web/hooks/use-tool-info.test.d.ts +1 -0
  263. package/dist/web/hooks/use-tool-info.test.js +130 -0
  264. package/dist/web/hooks/use-tool-info.test.js.map +1 -0
  265. package/dist/web/hooks/use-user.d.ts +18 -0
  266. package/dist/web/hooks/use-user.js +35 -0
  267. package/dist/web/hooks/use-user.js.map +1 -0
  268. package/dist/web/hooks/use-user.test.d.ts +1 -0
  269. package/dist/web/hooks/use-user.test.js +122 -0
  270. package/dist/web/hooks/use-user.test.js.map +1 -0
  271. package/dist/web/hooks/use-view-state.d.ts +4 -0
  272. package/dist/web/hooks/use-view-state.js +32 -0
  273. package/dist/web/hooks/use-view-state.js.map +1 -0
  274. package/dist/web/hooks/use-view-state.test.d.ts +1 -0
  275. package/dist/web/hooks/use-view-state.test.js +177 -0
  276. package/dist/web/hooks/use-view-state.test.js.map +1 -0
  277. package/dist/web/index.d.ts +7 -0
  278. package/dist/web/index.js +8 -0
  279. package/dist/web/index.js.map +1 -0
  280. package/dist/web/mount-view.d.ts +1 -0
  281. package/dist/web/mount-view.js +27 -0
  282. package/dist/web/mount-view.js.map +1 -0
  283. package/dist/web/plugin/data-llm.test.d.ts +1 -0
  284. package/dist/web/plugin/data-llm.test.js +81 -0
  285. package/dist/web/plugin/data-llm.test.js.map +1 -0
  286. package/dist/web/plugin/plugin.d.ts +5 -0
  287. package/dist/web/plugin/plugin.js +156 -0
  288. package/dist/web/plugin/plugin.js.map +1 -0
  289. package/dist/web/plugin/scan-views.d.ts +16 -0
  290. package/dist/web/plugin/scan-views.js +88 -0
  291. package/dist/web/plugin/scan-views.js.map +1 -0
  292. package/dist/web/plugin/scan-views.test.d.ts +1 -0
  293. package/dist/web/plugin/scan-views.test.js +99 -0
  294. package/dist/web/plugin/scan-views.test.js.map +1 -0
  295. package/dist/web/plugin/transform-data-llm.d.ts +12 -0
  296. package/dist/web/plugin/transform-data-llm.js +96 -0
  297. package/dist/web/plugin/transform-data-llm.js.map +1 -0
  298. package/dist/web/plugin/transform-data-llm.test.d.ts +1 -0
  299. package/dist/web/plugin/transform-data-llm.test.js +81 -0
  300. package/dist/web/plugin/transform-data-llm.test.js.map +1 -0
  301. package/dist/web/plugin/validate-view.d.ts +1 -0
  302. package/dist/web/plugin/validate-view.js +9 -0
  303. package/dist/web/plugin/validate-view.js.map +1 -0
  304. package/dist/web/plugin/validate-view.test.d.ts +1 -0
  305. package/dist/web/plugin/validate-view.test.js +24 -0
  306. package/dist/web/plugin/validate-view.test.js.map +1 -0
  307. package/dist/web/proxy.d.ts +1 -0
  308. package/dist/web/proxy.js +52 -0
  309. package/dist/web/proxy.js.map +1 -0
  310. package/dist/web/types.d.ts +16 -0
  311. package/dist/web/types.js +2 -0
  312. package/dist/web/types.js.map +1 -0
  313. package/package.json +82 -35
  314. package/tsconfig.base.json +33 -0
  315. package/dist/src/server/index.d.ts +0 -2
  316. package/dist/src/server/index.js +0 -3
  317. package/dist/src/server/index.js.map +0 -1
  318. package/dist/src/server/server.d.ts +0 -13
  319. package/dist/src/server/server.js +0 -54
  320. package/dist/src/server/server.js.map +0 -1
  321. package/dist/src/server/templateHelper.js +0 -29
  322. package/dist/src/server/templateHelper.js.map +0 -1
  323. package/dist/src/server/templates/development.hbs +0 -12
  324. package/dist/src/server/templates/production.hbs +0 -6
  325. package/dist/src/server/widgetsDevServer.d.ts +0 -12
  326. package/dist/src/server/widgetsDevServer.js +0 -39
  327. package/dist/src/server/widgetsDevServer.js.map +0 -1
  328. package/dist/src/test/setup.js +0 -9
  329. package/dist/src/test/setup.js.map +0 -1
  330. package/dist/src/test/utils.d.ts +0 -28
  331. package/dist/src/test/utils.js +0 -43
  332. package/dist/src/test/utils.js.map +0 -1
  333. package/dist/src/test/widget.test.js +0 -90
  334. package/dist/src/test/widget.test.js.map +0 -1
  335. package/dist/src/web/hooks/index.d.ts +0 -13
  336. package/dist/src/web/hooks/index.js +0 -14
  337. package/dist/src/web/hooks/index.js.map +0 -1
  338. package/dist/src/web/hooks/use-call-tool.d.ts +0 -54
  339. package/dist/src/web/hooks/use-call-tool.js +0 -44
  340. package/dist/src/web/hooks/use-call-tool.js.map +0 -1
  341. package/dist/src/web/hooks/use-call-tool.test.js +0 -66
  342. package/dist/src/web/hooks/use-call-tool.test.js.map +0 -1
  343. package/dist/src/web/hooks/use-display-mode.d.ts +0 -4
  344. package/dist/src/web/hooks/use-display-mode.js +0 -7
  345. package/dist/src/web/hooks/use-display-mode.js.map +0 -1
  346. package/dist/src/web/hooks/use-display-mode.test.js.map +0 -1
  347. package/dist/src/web/hooks/use-locale.d.ts +0 -1
  348. package/dist/src/web/hooks/use-locale.js +0 -5
  349. package/dist/src/web/hooks/use-locale.js.map +0 -1
  350. package/dist/src/web/hooks/use-locale.test.js +0 -21
  351. package/dist/src/web/hooks/use-locale.test.js.map +0 -1
  352. package/dist/src/web/hooks/use-open-external.d.ts +0 -1
  353. package/dist/src/web/hooks/use-open-external.js +0 -6
  354. package/dist/src/web/hooks/use-open-external.js.map +0 -1
  355. package/dist/src/web/hooks/use-open-external.test.js +0 -24
  356. package/dist/src/web/hooks/use-open-external.test.js.map +0 -1
  357. package/dist/src/web/hooks/use-openai-global.d.ts +0 -2
  358. package/dist/src/web/hooks/use-openai-global.js +0 -21
  359. package/dist/src/web/hooks/use-openai-global.js.map +0 -1
  360. package/dist/src/web/hooks/use-request-modal.d.ts +0 -5
  361. package/dist/src/web/hooks/use-request-modal.js +0 -9
  362. package/dist/src/web/hooks/use-request-modal.js.map +0 -1
  363. package/dist/src/web/hooks/use-request-modal.test.js +0 -24
  364. package/dist/src/web/hooks/use-request-modal.test.js.map +0 -1
  365. package/dist/src/web/hooks/use-send-follow-up-message.d.ts +0 -1
  366. package/dist/src/web/hooks/use-send-follow-up-message.js +0 -11
  367. package/dist/src/web/hooks/use-send-follow-up-message.js.map +0 -1
  368. package/dist/src/web/hooks/use-theme.d.ts +0 -1
  369. package/dist/src/web/hooks/use-theme.js +0 -5
  370. package/dist/src/web/hooks/use-theme.js.map +0 -1
  371. package/dist/src/web/hooks/use-theme.test.js +0 -26
  372. package/dist/src/web/hooks/use-theme.test.js.map +0 -1
  373. package/dist/src/web/hooks/use-tool-info.d.ts +0 -5
  374. package/dist/src/web/hooks/use-tool-info.js +0 -9
  375. package/dist/src/web/hooks/use-tool-info.js.map +0 -1
  376. package/dist/src/web/hooks/use-tool-info.test.js +0 -38
  377. package/dist/src/web/hooks/use-tool-info.test.js.map +0 -1
  378. package/dist/src/web/hooks/use-tool-output.d.ts +0 -4
  379. package/dist/src/web/hooks/use-tool-output.js +0 -9
  380. package/dist/src/web/hooks/use-tool-output.js.map +0 -1
  381. package/dist/src/web/hooks/use-tool-response-metadata.d.ts +0 -4
  382. package/dist/src/web/hooks/use-tool-response-metadata.js +0 -8
  383. package/dist/src/web/hooks/use-tool-response-metadata.js.map +0 -1
  384. package/dist/src/web/hooks/use-user-agent.d.ts +0 -1
  385. package/dist/src/web/hooks/use-user-agent.js +0 -5
  386. package/dist/src/web/hooks/use-user-agent.js.map +0 -1
  387. package/dist/src/web/hooks/use-user-agent.test.js +0 -31
  388. package/dist/src/web/hooks/use-user-agent.test.js.map +0 -1
  389. package/dist/src/web/hooks/use-widget-state.d.ts +0 -4
  390. package/dist/src/web/hooks/use-widget-state.js +0 -30
  391. package/dist/src/web/hooks/use-widget-state.js.map +0 -1
  392. package/dist/src/web/hooks/use-widget-state.test.js +0 -61
  393. package/dist/src/web/hooks/use-widget-state.test.js.map +0 -1
  394. package/dist/src/web/index.d.ts +0 -4
  395. package/dist/src/web/index.js +0 -5
  396. package/dist/src/web/index.js.map +0 -1
  397. package/dist/src/web/mount-widget.d.ts +0 -1
  398. package/dist/src/web/mount-widget.js +0 -14
  399. package/dist/src/web/mount-widget.js.map +0 -1
  400. package/dist/src/web/plugin.d.ts +0 -2
  401. package/dist/src/web/plugin.js +0 -28
  402. package/dist/src/web/plugin.js.map +0 -1
  403. package/dist/src/web/types.d.ts +0 -109
  404. package/dist/src/web/types.js.map +0 -1
  405. package/dist/vitest.config.d.ts +0 -2
  406. package/dist/vitest.config.js +0 -9
  407. package/dist/vitest.config.js.map +0 -1
  408. /package/dist/{src/test/setup.d.ts → cli/tunnel-control-server.test.d.ts} +0 -0
  409. /package/dist/{src/test/widget.test.d.ts → cli/tunnel-handler.test.d.ts} +0 -0
  410. /package/dist/{src/web/hooks/use-call-tool.test.d.ts → cli/tunnel.test.d.ts} +0 -0
  411. /package/dist/{src/web/hooks/use-display-mode.test.d.ts → server/asset-base-url-transform-plugin.test.d.ts} +0 -0
  412. /package/dist/{src/web/hooks/use-locale.test.d.ts → server/content-helpers.test.d.ts} +0 -0
  413. /package/dist/{src/web/hooks/use-open-external.test.d.ts → server/express.test.d.ts} +0 -0
  414. /package/dist/{src/web/hooks/use-request-modal.test.d.ts → server/middleware.test-d.d.ts} +0 -0
  415. /package/dist/{src/web/hooks/use-theme.test.d.ts → server/middleware.test.d.ts} +0 -0
  416. /package/dist/{src/web/hooks/use-tool-info.test.d.ts → server/tunnel-proxy-router.test.d.ts} +0 -0
  417. /package/dist/{src/web/hooks/use-user-agent.test.d.ts → test/view.test.d.ts} +0 -0
  418. /package/dist/{src/web → web/bridges/apps-sdk}/types.js +0 -0
  419. /package/dist/{src/web/hooks/use-widget-state.test.d.ts → web/bridges/mcp-app/use-mcp-app-context.test.d.ts} +0 -0
@@ -0,0 +1,190 @@
1
+ import { EventEmitter } from "node:events";
2
+ import { Readable } from "node:stream";
3
+ import { afterEach, beforeEach, describe, expect, it, vi } from "vitest";
4
+ import { parseStdoutLine, TunnelManager } from "./tunnel.js";
5
+ describe("parseStdoutLine", () => {
6
+ it("returns a connected event when the forwarding line is seen", () => {
7
+ const result = parseStdoutLine("Forwarding: https://abc.tunnel.example -> http://localhost:3000");
8
+ expect(result).toEqual({
9
+ kind: "connected",
10
+ url: "https://abc.tunnel.example",
11
+ });
12
+ });
13
+ it("strips a trailing slash from the forwarded URL", () => {
14
+ const result = parseStdoutLine("Forwarding: https://abc.tunnel.example/ -> http://localhost:3000");
15
+ expect(result).toEqual({
16
+ kind: "connected",
17
+ url: "https://abc.tunnel.example",
18
+ });
19
+ });
20
+ it("returns a starting event for any other non-empty line", () => {
21
+ expect(parseStdoutLine("Opening tunnel...")).toEqual({
22
+ kind: "starting",
23
+ message: "Opening tunnel...",
24
+ });
25
+ });
26
+ it("returns null for empty lines", () => {
27
+ expect(parseStdoutLine("")).toBeNull();
28
+ expect(parseStdoutLine(" ")).toBeNull();
29
+ });
30
+ });
31
+ function makeFakeChild() {
32
+ const child = new EventEmitter();
33
+ child.stdout = new Readable({ read() { } });
34
+ child.stderr = new Readable({ read() { } });
35
+ child.kill = vi.fn(() => true);
36
+ return child;
37
+ }
38
+ describe("TunnelManager", () => {
39
+ beforeEach(() => {
40
+ vi.useFakeTimers();
41
+ });
42
+ afterEach(() => {
43
+ vi.useRealTimers();
44
+ });
45
+ it("starts idle", () => {
46
+ const manager = new TunnelManager({
47
+ getPort: () => 3000,
48
+ spawn: () => makeFakeChild(),
49
+ });
50
+ expect(manager.getState()).toEqual({ status: "idle" });
51
+ });
52
+ it("transitions idle -> starting -> connected as stdout reports progress", () => {
53
+ const child = makeFakeChild();
54
+ const manager = new TunnelManager({
55
+ getPort: () => 3000,
56
+ spawn: () => child,
57
+ });
58
+ const states = [];
59
+ manager.on("state", (s) => states.push(s.status));
60
+ manager.start();
61
+ child.stdout.emit("data", Buffer.from("Opening tunnel...\n"));
62
+ child.stdout.emit("data", Buffer.from("Forwarding: https://abc.tunnel.example -> http://localhost:3000\n"));
63
+ expect(states).toEqual(["starting", "starting", "connected"]);
64
+ expect(manager.getState()).toEqual({
65
+ status: "connected",
66
+ url: "https://abc.tunnel.example",
67
+ });
68
+ });
69
+ it("emits activity events for stdout lines after connect", () => {
70
+ const child = makeFakeChild();
71
+ const manager = new TunnelManager({
72
+ getPort: () => 3000,
73
+ spawn: () => child,
74
+ });
75
+ const activity = [];
76
+ manager.on("activity", (a) => activity.push({ text: a.text, level: a.level }));
77
+ manager.start();
78
+ child.stdout.emit("data", Buffer.from("Forwarding: https://abc.tunnel.example -> http://localhost:3000\n"));
79
+ child.stdout.emit("data", Buffer.from("GET /widgets/foo\n"));
80
+ child.stderr.emit("data", Buffer.from("connection wobble\n"));
81
+ expect(activity).toEqual([
82
+ { text: "GET /widgets/foo", level: "log" },
83
+ { text: "connection wobble", level: "error" },
84
+ ]);
85
+ });
86
+ it("calling start twice does not spawn twice", () => {
87
+ const spawn = vi.fn(() => makeFakeChild());
88
+ const manager = new TunnelManager({ getPort: () => 3000, spawn });
89
+ manager.start();
90
+ manager.start();
91
+ expect(spawn).toHaveBeenCalledTimes(1);
92
+ });
93
+ it("transitions to error when the connection times out", () => {
94
+ const child = makeFakeChild();
95
+ const manager = new TunnelManager({
96
+ getPort: () => 3000,
97
+ spawn: () => child,
98
+ });
99
+ manager.start();
100
+ vi.advanceTimersByTime(60_000);
101
+ expect(manager.getState().status).toBe("error");
102
+ expect(child.kill).toHaveBeenCalled();
103
+ });
104
+ it("preserves the timeout error message when the killed child later emits close", () => {
105
+ const child = makeFakeChild();
106
+ const manager = new TunnelManager({
107
+ getPort: () => 3000,
108
+ spawn: () => child,
109
+ });
110
+ manager.start();
111
+ vi.advanceTimersByTime(60_000);
112
+ // The killed child emits a non-zero close after the timeout fired.
113
+ child.emit("close", 1);
114
+ expect(manager.getState()).toEqual({
115
+ status: "error",
116
+ message: "Tunnel connection timed out after one minute",
117
+ });
118
+ });
119
+ it("stop() kills the subprocess and goes idle", () => {
120
+ const child = makeFakeChild();
121
+ const manager = new TunnelManager({
122
+ getPort: () => 3000,
123
+ spawn: () => child,
124
+ });
125
+ manager.start();
126
+ manager.stop();
127
+ expect(child.kill).toHaveBeenCalled();
128
+ expect(manager.getState()).toEqual({ status: "idle" });
129
+ });
130
+ it("subscribers get the current state on subscribe", () => {
131
+ const child = makeFakeChild();
132
+ const manager = new TunnelManager({
133
+ getPort: () => 3000,
134
+ spawn: () => child,
135
+ });
136
+ manager.start();
137
+ child.stdout.emit("data", Buffer.from("Forwarding: https://abc.tunnel.example -> http://localhost:3000\n"));
138
+ let received;
139
+ const unsubscribe = manager.subscribe((s) => {
140
+ received = s;
141
+ });
142
+ expect(received).toEqual({
143
+ status: "connected",
144
+ url: "https://abc.tunnel.example",
145
+ });
146
+ unsubscribe();
147
+ });
148
+ it("ignores deferred close from a child that was replaced via stop()+start()", () => {
149
+ const childA = makeFakeChild();
150
+ const childB = makeFakeChild();
151
+ let spawnCount = 0;
152
+ const manager = new TunnelManager({
153
+ getPort: () => 3000,
154
+ spawn: () => (spawnCount++ === 0 ? childA : childB),
155
+ });
156
+ manager.start();
157
+ manager.stop();
158
+ manager.start();
159
+ childB.stdout.emit("data", Buffer.from("Forwarding: https://abc.tunnel.example -> http://localhost:3000\n"));
160
+ expect(manager.getState()).toEqual({
161
+ status: "connected",
162
+ url: "https://abc.tunnel.example",
163
+ });
164
+ // Stale close from childA arrives only now — must not clobber state.
165
+ childA.emit("close", null);
166
+ expect(manager.getState()).toEqual({
167
+ status: "connected",
168
+ url: "https://abc.tunnel.example",
169
+ });
170
+ });
171
+ it("ignores deferred error from a child that was replaced via stop()+start()", () => {
172
+ const childA = makeFakeChild();
173
+ const childB = makeFakeChild();
174
+ let spawnCount = 0;
175
+ const manager = new TunnelManager({
176
+ getPort: () => 3000,
177
+ spawn: () => (spawnCount++ === 0 ? childA : childB),
178
+ });
179
+ manager.start();
180
+ manager.stop();
181
+ manager.start();
182
+ childB.stdout.emit("data", Buffer.from("Forwarding: https://abc.tunnel.example -> http://localhost:3000\n"));
183
+ childA.emit("error", new Error("late spawn failure"));
184
+ expect(manager.getState()).toEqual({
185
+ status: "connected",
186
+ url: "https://abc.tunnel.example",
187
+ });
188
+ });
189
+ });
190
+ //# sourceMappingURL=tunnel.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tunnel.test.js","sourceRoot":"","sources":["../../src/cli/tunnel.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,aAAa,CAAC;AAC3C,OAAO,EAAE,QAAQ,EAAE,MAAM,aAAa,CAAC;AACvC,OAAO,EAAE,SAAS,EAAE,UAAU,EAAE,QAAQ,EAAE,MAAM,EAAE,EAAE,EAAE,EAAE,EAAE,MAAM,QAAQ,CAAC;AACzE,OAAO,EAAE,eAAe,EAAE,aAAa,EAAoB,MAAM,aAAa,CAAC;AAE/E,QAAQ,CAAC,iBAAiB,EAAE,GAAG,EAAE;IAC/B,EAAE,CAAC,4DAA4D,EAAE,GAAG,EAAE;QACpE,MAAM,MAAM,GAAG,eAAe,CAC5B,iEAAiE,CAClE,CAAC;QACF,MAAM,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC;YACrB,IAAI,EAAE,WAAW;YACjB,GAAG,EAAE,4BAA4B;SAClC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,gDAAgD,EAAE,GAAG,EAAE;QACxD,MAAM,MAAM,GAAG,eAAe,CAC5B,kEAAkE,CACnE,CAAC;QACF,MAAM,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC;YACrB,IAAI,EAAE,WAAW;YACjB,GAAG,EAAE,4BAA4B;SAClC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,uDAAuD,EAAE,GAAG,EAAE;QAC/D,MAAM,CAAC,eAAe,CAAC,mBAAmB,CAAC,CAAC,CAAC,OAAO,CAAC;YACnD,IAAI,EAAE,UAAU;YAChB,OAAO,EAAE,mBAAmB;SAC7B,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,8BAA8B,EAAE,GAAG,EAAE;QACtC,MAAM,CAAC,eAAe,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,EAAE,CAAC;QACvC,MAAM,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC,CAAC,QAAQ,EAAE,CAAC;IAC5C,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC;AAQH,SAAS,aAAa;IACpB,MAAM,KAAK,GAAG,IAAI,YAAY,EAAe,CAAC;IAC9C,KAAK,CAAC,MAAM,GAAG,IAAI,QAAQ,CAAC,EAAE,IAAI,KAAI,CAAC,EAAE,CAAC,CAAC;IAC3C,KAAK,CAAC,MAAM,GAAG,IAAI,QAAQ,CAAC,EAAE,IAAI,KAAI,CAAC,EAAE,CAAC,CAAC;IAC3C,KAAK,CAAC,IAAI,GAAG,EAAE,CAAC,EAAE,CAAgB,GAAG,EAAE,CAAC,IAAI,CAAC,CAAC;IAC9C,OAAO,KAAK,CAAC;AACf,CAAC;AAED,QAAQ,CAAC,eAAe,EAAE,GAAG,EAAE;IAC7B,UAAU,CAAC,GAAG,EAAE;QACd,EAAE,CAAC,aAAa,EAAE,CAAC;IACrB,CAAC,CAAC,CAAC;IACH,SAAS,CAAC,GAAG,EAAE;QACb,EAAE,CAAC,aAAa,EAAE,CAAC;IACrB,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,aAAa,EAAE,GAAG,EAAE;QACrB,MAAM,OAAO,GAAG,IAAI,aAAa,CAAC;YAChC,OAAO,EAAE,GAAG,EAAE,CAAC,IAAI;YACnB,KAAK,EAAE,GAAG,EAAE,CAAC,aAAa,EAAE;SAC7B,CAAC,CAAC;QACH,MAAM,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC,CAAC,OAAO,CAAC,EAAE,MAAM,EAAE,MAAM,EAAE,CAAC,CAAC;IACzD,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,sEAAsE,EAAE,GAAG,EAAE;QAC9E,MAAM,KAAK,GAAG,aAAa,EAAE,CAAC;QAC9B,MAAM,OAAO,GAAG,IAAI,aAAa,CAAC;YAChC,OAAO,EAAE,GAAG,EAAE,CAAC,IAAI;YACnB,KAAK,EAAE,GAAG,EAAE,CAAC,KAAK;SACnB,CAAC,CAAC;QACH,MAAM,MAAM,GAAa,EAAE,CAAC;QAC5B,OAAO,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,CAAc,EAAE,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC;QAE/D,OAAO,CAAC,KAAK,EAAE,CAAC;QAChB,KAAK,CAAC,MAAM,CAAC,IAAI,CAAC,MAAM,EAAE,MAAM,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC;QAC9D,KAAK,CAAC,MAAM,CAAC,IAAI,CACf,MAAM,EACN,MAAM,CAAC,IAAI,CACT,mEAAmE,CACpE,CACF,CAAC;QAEF,MAAM,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC,CAAC,UAAU,EAAE,UAAU,EAAE,WAAW,CAAC,CAAC,CAAC;QAC9D,MAAM,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC,CAAC,OAAO,CAAC;YACjC,MAAM,EAAE,WAAW;YACnB,GAAG,EAAE,4BAA4B;SAClC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,sDAAsD,EAAE,GAAG,EAAE;QAC9D,MAAM,KAAK,GAAG,aAAa,EAAE,CAAC;QAC9B,MAAM,OAAO,GAAG,IAAI,aAAa,CAAC;YAChC,OAAO,EAAE,GAAG,EAAE,CAAC,IAAI;YACnB,KAAK,EAAE,GAAG,EAAE,CAAC,KAAK;SACnB,CAAC,CAAC;QACH,MAAM,QAAQ,GAA2C,EAAE,CAAC;QAC5D,OAAO,CAAC,EAAE,CAAC,UAAU,EAAE,CAAC,CAAkC,EAAE,EAAE,CAC5D,QAAQ,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,CAAC,CAAC,IAAI,EAAE,KAAK,EAAE,CAAC,CAAC,KAAK,EAAE,CAAC,CAChD,CAAC;QAEF,OAAO,CAAC,KAAK,EAAE,CAAC;QAChB,KAAK,CAAC,MAAM,CAAC,IAAI,CACf,MAAM,EACN,MAAM,CAAC,IAAI,CACT,mEAAmE,CACpE,CACF,CAAC;QACF,KAAK,CAAC,MAAM,CAAC,IAAI,CAAC,MAAM,EAAE,MAAM,CAAC,IAAI,CAAC,oBAAoB,CAAC,CAAC,CAAC;QAC7D,KAAK,CAAC,MAAM,CAAC,IAAI,CAAC,MAAM,EAAE,MAAM,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC;QAE9D,MAAM,CAAC,QAAQ,CAAC,CAAC,OAAO,CAAC;YACvB,EAAE,IAAI,EAAE,kBAAkB,EAAE,KAAK,EAAE,KAAK,EAAE;YAC1C,EAAE,IAAI,EAAE,mBAAmB,EAAE,KAAK,EAAE,OAAO,EAAE;SAC9C,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,0CAA0C,EAAE,GAAG,EAAE;QAClD,MAAM,KAAK,GAAG,EAAE,CAAC,EAAE,CAAC,GAAG,EAAE,CAAC,aAAa,EAAE,CAAC,CAAC;QAC3C,MAAM,OAAO,GAAG,IAAI,aAAa,CAAC,EAAE,OAAO,EAAE,GAAG,EAAE,CAAC,IAAI,EAAE,KAAK,EAAE,CAAC,CAAC;QAClE,OAAO,CAAC,KAAK,EAAE,CAAC;QAChB,OAAO,CAAC,KAAK,EAAE,CAAC;QAChB,MAAM,CAAC,KAAK,CAAC,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC;IACzC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,oDAAoD,EAAE,GAAG,EAAE;QAC5D,MAAM,KAAK,GAAG,aAAa,EAAE,CAAC;QAC9B,MAAM,OAAO,GAAG,IAAI,aAAa,CAAC;YAChC,OAAO,EAAE,GAAG,EAAE,CAAC,IAAI;YACnB,KAAK,EAAE,GAAG,EAAE,CAAC,KAAK;SACnB,CAAC,CAAC;QACH,OAAO,CAAC,KAAK,EAAE,CAAC;QAEhB,EAAE,CAAC,mBAAmB,CAAC,MAAM,CAAC,CAAC;QAE/B,MAAM,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAChD,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,gBAAgB,EAAE,CAAC;IACxC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,6EAA6E,EAAE,GAAG,EAAE;QACrF,MAAM,KAAK,GAAG,aAAa,EAAE,CAAC;QAC9B,MAAM,OAAO,GAAG,IAAI,aAAa,CAAC;YAChC,OAAO,EAAE,GAAG,EAAE,CAAC,IAAI;YACnB,KAAK,EAAE,GAAG,EAAE,CAAC,KAAK;SACnB,CAAC,CAAC;QACH,OAAO,CAAC,KAAK,EAAE,CAAC;QAEhB,EAAE,CAAC,mBAAmB,CAAC,MAAM,CAAC,CAAC;QAC/B,mEAAmE;QACnE,KAAK,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC;QAEvB,MAAM,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC,CAAC,OAAO,CAAC;YACjC,MAAM,EAAE,OAAO;YACf,OAAO,EAAE,8CAA8C;SACxD,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,2CAA2C,EAAE,GAAG,EAAE;QACnD,MAAM,KAAK,GAAG,aAAa,EAAE,CAAC;QAC9B,MAAM,OAAO,GAAG,IAAI,aAAa,CAAC;YAChC,OAAO,EAAE,GAAG,EAAE,CAAC,IAAI;YACnB,KAAK,EAAE,GAAG,EAAE,CAAC,KAAK;SACnB,CAAC,CAAC;QACH,OAAO,CAAC,KAAK,EAAE,CAAC;QAChB,OAAO,CAAC,IAAI,EAAE,CAAC;QACf,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,gBAAgB,EAAE,CAAC;QACtC,MAAM,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC,CAAC,OAAO,CAAC,EAAE,MAAM,EAAE,MAAM,EAAE,CAAC,CAAC;IACzD,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,gDAAgD,EAAE,GAAG,EAAE;QACxD,MAAM,KAAK,GAAG,aAAa,EAAE,CAAC;QAC9B,MAAM,OAAO,GAAG,IAAI,aAAa,CAAC;YAChC,OAAO,EAAE,GAAG,EAAE,CAAC,IAAI;YACnB,KAAK,EAAE,GAAG,EAAE,CAAC,KAAK;SACnB,CAAC,CAAC;QACH,OAAO,CAAC,KAAK,EAAE,CAAC;QAChB,KAAK,CAAC,MAAM,CAAC,IAAI,CACf,MAAM,EACN,MAAM,CAAC,IAAI,CACT,mEAAmE,CACpE,CACF,CAAC;QAEF,IAAI,QAAiC,CAAC;QACtC,MAAM,WAAW,GAAG,OAAO,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE;YAC1C,QAAQ,GAAG,CAAC,CAAC;QACf,CAAC,CAAC,CAAC;QACH,MAAM,CAAC,QAAQ,CAAC,CAAC,OAAO,CAAC;YACvB,MAAM,EAAE,WAAW;YACnB,GAAG,EAAE,4BAA4B;SAClC,CAAC,CAAC;QACH,WAAW,EAAE,CAAC;IAChB,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,0EAA0E,EAAE,GAAG,EAAE;QAClF,MAAM,MAAM,GAAG,aAAa,EAAE,CAAC;QAC/B,MAAM,MAAM,GAAG,aAAa,EAAE,CAAC;QAC/B,IAAI,UAAU,GAAG,CAAC,CAAC;QACnB,MAAM,OAAO,GAAG,IAAI,aAAa,CAAC;YAChC,OAAO,EAAE,GAAG,EAAE,CAAC,IAAI;YACnB,KAAK,EAAE,GAAG,EAAE,CAAC,CAAC,UAAU,EAAE,KAAK,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC;SACpD,CAAC,CAAC;QAEH,OAAO,CAAC,KAAK,EAAE,CAAC;QAChB,OAAO,CAAC,IAAI,EAAE,CAAC;QACf,OAAO,CAAC,KAAK,EAAE,CAAC;QAChB,MAAM,CAAC,MAAM,CAAC,IAAI,CAChB,MAAM,EACN,MAAM,CAAC,IAAI,CACT,mEAAmE,CACpE,CACF,CAAC;QACF,MAAM,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC,CAAC,OAAO,CAAC;YACjC,MAAM,EAAE,WAAW;YACnB,GAAG,EAAE,4BAA4B;SAClC,CAAC,CAAC;QAEH,qEAAqE;QACrE,MAAM,CAAC,IAAI,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC;QAC3B,MAAM,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC,CAAC,OAAO,CAAC;YACjC,MAAM,EAAE,WAAW;YACnB,GAAG,EAAE,4BAA4B;SAClC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,0EAA0E,EAAE,GAAG,EAAE;QAClF,MAAM,MAAM,GAAG,aAAa,EAAE,CAAC;QAC/B,MAAM,MAAM,GAAG,aAAa,EAAE,CAAC;QAC/B,IAAI,UAAU,GAAG,CAAC,CAAC;QACnB,MAAM,OAAO,GAAG,IAAI,aAAa,CAAC;YAChC,OAAO,EAAE,GAAG,EAAE,CAAC,IAAI;YACnB,KAAK,EAAE,GAAG,EAAE,CAAC,CAAC,UAAU,EAAE,KAAK,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC;SACpD,CAAC,CAAC;QAEH,OAAO,CAAC,KAAK,EAAE,CAAC;QAChB,OAAO,CAAC,IAAI,EAAE,CAAC;QACf,OAAO,CAAC,KAAK,EAAE,CAAC;QAChB,MAAM,CAAC,MAAM,CAAC,IAAI,CAChB,MAAM,EACN,MAAM,CAAC,IAAI,CACT,mEAAmE,CACpE,CACF,CAAC;QAEF,MAAM,CAAC,IAAI,CAAC,OAAO,EAAE,IAAI,KAAK,CAAC,oBAAoB,CAAC,CAAC,CAAC;QACtD,MAAM,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC,CAAC,OAAO,CAAC;YACjC,MAAM,EAAE,WAAW;YACnB,GAAG,EAAE,4BAA4B;SAClC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC","sourcesContent":["import { EventEmitter } from \"node:events\";\nimport { Readable } from \"node:stream\";\nimport { afterEach, beforeEach, describe, expect, it, vi } from \"vitest\";\nimport { parseStdoutLine, TunnelManager, type TunnelState } from \"./tunnel.js\";\n\ndescribe(\"parseStdoutLine\", () => {\n it(\"returns a connected event when the forwarding line is seen\", () => {\n const result = parseStdoutLine(\n \"Forwarding: https://abc.tunnel.example -> http://localhost:3000\",\n );\n expect(result).toEqual({\n kind: \"connected\",\n url: \"https://abc.tunnel.example\",\n });\n });\n\n it(\"strips a trailing slash from the forwarded URL\", () => {\n const result = parseStdoutLine(\n \"Forwarding: https://abc.tunnel.example/ -> http://localhost:3000\",\n );\n expect(result).toEqual({\n kind: \"connected\",\n url: \"https://abc.tunnel.example\",\n });\n });\n\n it(\"returns a starting event for any other non-empty line\", () => {\n expect(parseStdoutLine(\"Opening tunnel...\")).toEqual({\n kind: \"starting\",\n message: \"Opening tunnel...\",\n });\n });\n\n it(\"returns null for empty lines\", () => {\n expect(parseStdoutLine(\"\")).toBeNull();\n expect(parseStdoutLine(\" \")).toBeNull();\n });\n});\n\ntype FakeChild = EventEmitter & {\n stdout: Readable;\n stderr: Readable;\n kill: ReturnType<typeof vi.fn<() => boolean>>;\n};\n\nfunction makeFakeChild(): FakeChild {\n const child = new EventEmitter() as FakeChild;\n child.stdout = new Readable({ read() {} });\n child.stderr = new Readable({ read() {} });\n child.kill = vi.fn<() => boolean>(() => true);\n return child;\n}\n\ndescribe(\"TunnelManager\", () => {\n beforeEach(() => {\n vi.useFakeTimers();\n });\n afterEach(() => {\n vi.useRealTimers();\n });\n\n it(\"starts idle\", () => {\n const manager = new TunnelManager({\n getPort: () => 3000,\n spawn: () => makeFakeChild(),\n });\n expect(manager.getState()).toEqual({ status: \"idle\" });\n });\n\n it(\"transitions idle -> starting -> connected as stdout reports progress\", () => {\n const child = makeFakeChild();\n const manager = new TunnelManager({\n getPort: () => 3000,\n spawn: () => child,\n });\n const states: string[] = [];\n manager.on(\"state\", (s: TunnelState) => states.push(s.status));\n\n manager.start();\n child.stdout.emit(\"data\", Buffer.from(\"Opening tunnel...\\n\"));\n child.stdout.emit(\n \"data\",\n Buffer.from(\n \"Forwarding: https://abc.tunnel.example -> http://localhost:3000\\n\",\n ),\n );\n\n expect(states).toEqual([\"starting\", \"starting\", \"connected\"]);\n expect(manager.getState()).toEqual({\n status: \"connected\",\n url: \"https://abc.tunnel.example\",\n });\n });\n\n it(\"emits activity events for stdout lines after connect\", () => {\n const child = makeFakeChild();\n const manager = new TunnelManager({\n getPort: () => 3000,\n spawn: () => child,\n });\n const activity: Array<{ text: string; level: string }> = [];\n manager.on(\"activity\", (a: { text: string; level: string }) =>\n activity.push({ text: a.text, level: a.level }),\n );\n\n manager.start();\n child.stdout.emit(\n \"data\",\n Buffer.from(\n \"Forwarding: https://abc.tunnel.example -> http://localhost:3000\\n\",\n ),\n );\n child.stdout.emit(\"data\", Buffer.from(\"GET /widgets/foo\\n\"));\n child.stderr.emit(\"data\", Buffer.from(\"connection wobble\\n\"));\n\n expect(activity).toEqual([\n { text: \"GET /widgets/foo\", level: \"log\" },\n { text: \"connection wobble\", level: \"error\" },\n ]);\n });\n\n it(\"calling start twice does not spawn twice\", () => {\n const spawn = vi.fn(() => makeFakeChild());\n const manager = new TunnelManager({ getPort: () => 3000, spawn });\n manager.start();\n manager.start();\n expect(spawn).toHaveBeenCalledTimes(1);\n });\n\n it(\"transitions to error when the connection times out\", () => {\n const child = makeFakeChild();\n const manager = new TunnelManager({\n getPort: () => 3000,\n spawn: () => child,\n });\n manager.start();\n\n vi.advanceTimersByTime(60_000);\n\n expect(manager.getState().status).toBe(\"error\");\n expect(child.kill).toHaveBeenCalled();\n });\n\n it(\"preserves the timeout error message when the killed child later emits close\", () => {\n const child = makeFakeChild();\n const manager = new TunnelManager({\n getPort: () => 3000,\n spawn: () => child,\n });\n manager.start();\n\n vi.advanceTimersByTime(60_000);\n // The killed child emits a non-zero close after the timeout fired.\n child.emit(\"close\", 1);\n\n expect(manager.getState()).toEqual({\n status: \"error\",\n message: \"Tunnel connection timed out after one minute\",\n });\n });\n\n it(\"stop() kills the subprocess and goes idle\", () => {\n const child = makeFakeChild();\n const manager = new TunnelManager({\n getPort: () => 3000,\n spawn: () => child,\n });\n manager.start();\n manager.stop();\n expect(child.kill).toHaveBeenCalled();\n expect(manager.getState()).toEqual({ status: \"idle\" });\n });\n\n it(\"subscribers get the current state on subscribe\", () => {\n const child = makeFakeChild();\n const manager = new TunnelManager({\n getPort: () => 3000,\n spawn: () => child,\n });\n manager.start();\n child.stdout.emit(\n \"data\",\n Buffer.from(\n \"Forwarding: https://abc.tunnel.example -> http://localhost:3000\\n\",\n ),\n );\n\n let received: TunnelState | undefined;\n const unsubscribe = manager.subscribe((s) => {\n received = s;\n });\n expect(received).toEqual({\n status: \"connected\",\n url: \"https://abc.tunnel.example\",\n });\n unsubscribe();\n });\n\n it(\"ignores deferred close from a child that was replaced via stop()+start()\", () => {\n const childA = makeFakeChild();\n const childB = makeFakeChild();\n let spawnCount = 0;\n const manager = new TunnelManager({\n getPort: () => 3000,\n spawn: () => (spawnCount++ === 0 ? childA : childB),\n });\n\n manager.start();\n manager.stop();\n manager.start();\n childB.stdout.emit(\n \"data\",\n Buffer.from(\n \"Forwarding: https://abc.tunnel.example -> http://localhost:3000\\n\",\n ),\n );\n expect(manager.getState()).toEqual({\n status: \"connected\",\n url: \"https://abc.tunnel.example\",\n });\n\n // Stale close from childA arrives only now — must not clobber state.\n childA.emit(\"close\", null);\n expect(manager.getState()).toEqual({\n status: \"connected\",\n url: \"https://abc.tunnel.example\",\n });\n });\n\n it(\"ignores deferred error from a child that was replaced via stop()+start()\", () => {\n const childA = makeFakeChild();\n const childB = makeFakeChild();\n let spawnCount = 0;\n const manager = new TunnelManager({\n getPort: () => 3000,\n spawn: () => (spawnCount++ === 0 ? childA : childB),\n });\n\n manager.start();\n manager.stop();\n manager.start();\n childB.stdout.emit(\n \"data\",\n Buffer.from(\n \"Forwarding: https://abc.tunnel.example -> http://localhost:3000\\n\",\n ),\n );\n\n childA.emit(\"error\", new Error(\"late spawn failure\"));\n expect(manager.getState()).toEqual({\n status: \"connected\",\n url: \"https://abc.tunnel.example\",\n });\n });\n});\n"]}
@@ -0,0 +1,5 @@
1
+ export type Message = {
2
+ id: string;
3
+ text: string;
4
+ type: "log" | "restart" | "error";
5
+ };
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.js","sourceRoot":"","sources":["../../src/cli/types.ts"],"names":[],"mappings":"","sourcesContent":["export type Message = {\n id: string;\n text: string;\n type: \"log\" | \"restart\" | \"error\";\n};\n"]}
@@ -0,0 +1,11 @@
1
+ export interface CommandStep {
2
+ label: string;
3
+ command?: string;
4
+ run?: () => void | Promise<void>;
5
+ }
6
+ export declare const useExecuteSteps: (steps: CommandStep[]) => {
7
+ currentStep: number;
8
+ status: "error" | "success" | "running";
9
+ error: string | null;
10
+ execute: () => Promise<void>;
11
+ };
@@ -0,0 +1,36 @@
1
+ import { useCallback, useState } from "react";
2
+ import { runCommand } from "./run-command.js";
3
+ export const useExecuteSteps = (steps) => {
4
+ const [currentStep, setCurrentStep] = useState(0);
5
+ const [status, setStatus] = useState("running");
6
+ const [error, setError] = useState(null);
7
+ const execute = useCallback(async () => {
8
+ try {
9
+ for (let i = 0; i < steps.length; i++) {
10
+ const step = steps[i];
11
+ if (step) {
12
+ setCurrentStep(i);
13
+ if (step.run) {
14
+ await step.run();
15
+ }
16
+ if (step.command) {
17
+ await runCommand(step.command);
18
+ }
19
+ }
20
+ }
21
+ setStatus("success");
22
+ setImmediate(() => {
23
+ process.exit(0);
24
+ });
25
+ }
26
+ catch (err) {
27
+ setStatus("error");
28
+ setError(err instanceof Error ? err.message : String(err));
29
+ setImmediate(() => {
30
+ process.exit(1);
31
+ });
32
+ }
33
+ }, [steps]);
34
+ return { currentStep, status, error, execute };
35
+ };
36
+ //# sourceMappingURL=use-execute-steps.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"use-execute-steps.js","sourceRoot":"","sources":["../../src/cli/use-execute-steps.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,QAAQ,EAAE,MAAM,OAAO,CAAC;AAC9C,OAAO,EAAE,UAAU,EAAE,MAAM,kBAAkB,CAAC;AAQ9C,MAAM,CAAC,MAAM,eAAe,GAAG,CAAC,KAAoB,EAAE,EAAE;IACtD,MAAM,CAAC,WAAW,EAAE,cAAc,CAAC,GAAG,QAAQ,CAAS,CAAC,CAAC,CAAC;IAC1D,MAAM,CAAC,MAAM,EAAE,SAAS,CAAC,GAAG,QAAQ,CAClC,SAAS,CACV,CAAC;IACF,MAAM,CAAC,KAAK,EAAE,QAAQ,CAAC,GAAG,QAAQ,CAAgB,IAAI,CAAC,CAAC;IAExD,MAAM,OAAO,GAAG,WAAW,CAAC,KAAK,IAAI,EAAE;QACrC,IAAI,CAAC;YACH,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,KAAK,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;gBACtC,MAAM,IAAI,GAAG,KAAK,CAAC,CAAC,CAAC,CAAC;gBACtB,IAAI,IAAI,EAAE,CAAC;oBACT,cAAc,CAAC,CAAC,CAAC,CAAC;oBAClB,IAAI,IAAI,CAAC,GAAG,EAAE,CAAC;wBACb,MAAM,IAAI,CAAC,GAAG,EAAE,CAAC;oBACnB,CAAC;oBACD,IAAI,IAAI,CAAC,OAAO,EAAE,CAAC;wBACjB,MAAM,UAAU,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;oBACjC,CAAC;gBACH,CAAC;YACH,CAAC;YACD,SAAS,CAAC,SAAS,CAAC,CAAC;YACrB,YAAY,CAAC,GAAG,EAAE;gBAChB,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;YAClB,CAAC,CAAC,CAAC;QACL,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,SAAS,CAAC,OAAO,CAAC,CAAC;YACnB,QAAQ,CAAC,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC;YAC3D,YAAY,CAAC,GAAG,EAAE;gBAChB,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;YAClB,CAAC,CAAC,CAAC;QACL,CAAC;IACH,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC,CAAC;IAEZ,OAAO,EAAE,WAAW,EAAE,MAAM,EAAE,KAAK,EAAE,OAAO,EAAE,CAAC;AACjD,CAAC,CAAC","sourcesContent":["import { useCallback, useState } from \"react\";\nimport { runCommand } from \"./run-command.js\";\n\nexport interface CommandStep {\n label: string;\n command?: string;\n run?: () => void | Promise<void>;\n}\n\nexport const useExecuteSteps = (steps: CommandStep[]) => {\n const [currentStep, setCurrentStep] = useState<number>(0);\n const [status, setStatus] = useState<\"running\" | \"success\" | \"error\">(\n \"running\",\n );\n const [error, setError] = useState<string | null>(null);\n\n const execute = useCallback(async () => {\n try {\n for (let i = 0; i < steps.length; i++) {\n const step = steps[i];\n if (step) {\n setCurrentStep(i);\n if (step.run) {\n await step.run();\n }\n if (step.command) {\n await runCommand(step.command);\n }\n }\n }\n setStatus(\"success\");\n setImmediate(() => {\n process.exit(0);\n });\n } catch (err) {\n setStatus(\"error\");\n setError(err instanceof Error ? err.message : String(err));\n setImmediate(() => {\n process.exit(1);\n });\n }\n }, [steps]);\n\n return { currentStep, status, error, execute };\n};\n"]}
@@ -0,0 +1,3 @@
1
+ import type { Message } from "./types.js";
2
+ export type PushMessage = (text: string, type: Message["type"]) => void;
3
+ export declare function useMessages(): [Array<Message>, PushMessage];
@@ -0,0 +1,11 @@
1
+ import { randomUUID } from "node:crypto";
2
+ import { useCallback, useState } from "react";
3
+ const MAX_MESSAGES = 10;
4
+ export function useMessages() {
5
+ const [messages, setMessages] = useState([]);
6
+ const push = useCallback((text, type) => {
7
+ setMessages((prev) => [...prev, { id: randomUUID(), text, type }].slice(-MAX_MESSAGES));
8
+ }, []);
9
+ return [messages, push];
10
+ }
11
+ //# sourceMappingURL=use-messages.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"use-messages.js","sourceRoot":"","sources":["../../src/cli/use-messages.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,aAAa,CAAC;AACzC,OAAO,EAAE,WAAW,EAAE,QAAQ,EAAE,MAAM,OAAO,CAAC;AAG9C,MAAM,YAAY,GAAG,EAAE,CAAC;AAIxB,MAAM,UAAU,WAAW;IACzB,MAAM,CAAC,QAAQ,EAAE,WAAW,CAAC,GAAG,QAAQ,CAAiB,EAAE,CAAC,CAAC;IAE7D,MAAM,IAAI,GAAG,WAAW,CAAc,CAAC,IAAI,EAAE,IAAI,EAAE,EAAE;QACnD,WAAW,CAAC,CAAC,IAAI,EAAE,EAAE,CACnB,CAAC,GAAG,IAAI,EAAE,EAAE,EAAE,EAAE,UAAU,EAAE,EAAE,IAAI,EAAE,IAAI,EAAE,CAAC,CAAC,KAAK,CAAC,CAAC,YAAY,CAAC,CACjE,CAAC;IACJ,CAAC,EAAE,EAAE,CAAC,CAAC;IAEP,OAAO,CAAC,QAAQ,EAAE,IAAI,CAAC,CAAC;AAC1B,CAAC","sourcesContent":["import { randomUUID } from \"node:crypto\";\nimport { useCallback, useState } from \"react\";\nimport type { Message } from \"./types.js\";\n\nconst MAX_MESSAGES = 10;\n\nexport type PushMessage = (text: string, type: Message[\"type\"]) => void;\n\nexport function useMessages(): [Array<Message>, PushMessage] {\n const [messages, setMessages] = useState<Array<Message>>([]);\n\n const push = useCallback<PushMessage>((text, type) => {\n setMessages((prev) =>\n [...prev, { id: randomUUID(), text, type }].slice(-MAX_MESSAGES),\n );\n }, []);\n\n return [messages, push];\n}\n"]}
@@ -0,0 +1,2 @@
1
+ import type { PushMessage } from "./use-messages.js";
2
+ export declare function useNodemon(env: NodeJS.ProcessEnv, pushMessage: PushMessage): void;
@@ -0,0 +1,73 @@
1
+ import { existsSync } from "node:fs";
2
+ import { resolve } from "node:path";
3
+ import nodemonOriginal from "nodemon";
4
+ import { useEffect } from "react";
5
+ const nodemon = nodemonOriginal;
6
+ const SOURCEMAP_WARNING = /^Sourcemap for ".*" points to missing source files$/;
7
+ export function useNodemon(env, pushMessage) {
8
+ useEffect(() => {
9
+ const configFile = resolve(process.cwd(), "nodemon.json");
10
+ const config = existsSync(configFile)
11
+ ? {
12
+ configFile,
13
+ }
14
+ : {
15
+ watch: ["src"],
16
+ ext: "ts,json",
17
+ exec: "tsx src/server.ts",
18
+ };
19
+ nodemon({ ...config, env, stdout: false });
20
+ const handleStdoutData = (chunk) => {
21
+ const message = chunk.toString().trim();
22
+ if (message) {
23
+ pushMessage(message, "log");
24
+ }
25
+ };
26
+ const handleStderrData = (chunk) => {
27
+ const message = chunk.toString().trim();
28
+ if (!message) {
29
+ return;
30
+ }
31
+ // Node's source-map warnings for third-party deps (superjson, @mcp/sdk, …) — not actionable.
32
+ const filtered = message
33
+ .split("\n")
34
+ .filter((line) => !SOURCEMAP_WARNING.test(line))
35
+ .join("\n");
36
+ if (filtered) {
37
+ pushMessage(filtered, "error");
38
+ }
39
+ };
40
+ const setupStdoutListener = () => {
41
+ if (nodemon.stdout) {
42
+ nodemon.stdout.off("data", handleStdoutData);
43
+ nodemon.stdout.on("data", handleStdoutData);
44
+ }
45
+ };
46
+ const setupStderrListener = () => {
47
+ if (nodemon.stderr) {
48
+ nodemon.stderr.off("data", handleStderrData);
49
+ nodemon.stderr.on("data", handleStderrData);
50
+ }
51
+ };
52
+ nodemon.on("readable", () => {
53
+ setupStdoutListener();
54
+ setupStderrListener();
55
+ });
56
+ nodemon.on("restart", (files) => {
57
+ const restartMessage = `Server restarted due to file changes: ${files.join(", ")}`;
58
+ pushMessage(restartMessage, "restart");
59
+ setupStdoutListener();
60
+ setupStderrListener();
61
+ });
62
+ return () => {
63
+ if (nodemon.stdout) {
64
+ nodemon.stdout.off("data", handleStdoutData);
65
+ }
66
+ if (nodemon.stderr) {
67
+ nodemon.stderr.off("data", handleStderrData);
68
+ }
69
+ nodemon.emit("quit");
70
+ };
71
+ }, [env, pushMessage]);
72
+ }
73
+ //# sourceMappingURL=use-nodemon.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"use-nodemon.js","sourceRoot":"","sources":["../../src/cli/use-nodemon.ts"],"names":[],"mappings":"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","sourcesContent":["import { existsSync } from \"node:fs\";\nimport { resolve } from \"node:path\";\nimport nodemonOriginal from \"nodemon\";\nimport { useEffect } from \"react\";\nimport type { ExtendedNodemon } from \"./nodemon.d.ts\";\nimport type { PushMessage } from \"./use-messages.js\";\n\nconst nodemon = nodemonOriginal as ExtendedNodemon;\n\nconst SOURCEMAP_WARNING = /^Sourcemap for \".*\" points to missing source files$/;\n\nexport function useNodemon(\n env: NodeJS.ProcessEnv,\n pushMessage: PushMessage,\n): void {\n useEffect(() => {\n const configFile = resolve(process.cwd(), \"nodemon.json\");\n\n const config = existsSync(configFile)\n ? {\n configFile,\n }\n : {\n watch: [\"src\"],\n ext: \"ts,json\",\n exec: \"tsx src/server.ts\",\n };\n\n nodemon({ ...config, env, stdout: false });\n\n const handleStdoutData = (chunk: Buffer) => {\n const message = chunk.toString().trim();\n if (message) {\n pushMessage(message, \"log\");\n }\n };\n\n const handleStderrData = (chunk: Buffer) => {\n const message = chunk.toString().trim();\n if (!message) {\n return;\n }\n // Node's source-map warnings for third-party deps (superjson, @mcp/sdk, …) — not actionable.\n const filtered = message\n .split(\"\\n\")\n .filter((line) => !SOURCEMAP_WARNING.test(line))\n .join(\"\\n\");\n if (filtered) {\n pushMessage(filtered, \"error\");\n }\n };\n\n const setupStdoutListener = () => {\n if (nodemon.stdout) {\n nodemon.stdout.off(\"data\", handleStdoutData);\n nodemon.stdout.on(\"data\", handleStdoutData);\n }\n };\n\n const setupStderrListener = () => {\n if (nodemon.stderr) {\n nodemon.stderr.off(\"data\", handleStderrData);\n nodemon.stderr.on(\"data\", handleStderrData);\n }\n };\n\n nodemon.on(\"readable\", () => {\n setupStdoutListener();\n setupStderrListener();\n });\n\n nodemon.on(\"restart\", (files: string[]) => {\n const restartMessage = `Server restarted due to file changes: ${files.join(\", \")}`;\n pushMessage(restartMessage, \"restart\");\n setupStdoutListener();\n setupStderrListener();\n });\n\n return () => {\n if (nodemon.stdout) {\n nodemon.stdout.off(\"data\", handleStdoutData);\n }\n if (nodemon.stderr) {\n nodemon.stderr.off(\"data\", handleStderrData);\n }\n nodemon.emit(\"quit\");\n };\n }, [env, pushMessage]);\n}\n"]}
@@ -0,0 +1 @@
1
+ export declare function useOpenBrowser(port: number, enabled: boolean): void;
@@ -0,0 +1,44 @@
1
+ import net from "node:net";
2
+ import open from "open";
3
+ import { useEffect, useRef } from "react";
4
+ const POLL_INTERVAL_MS = 200;
5
+ const POLL_TIMEOUT_MS = 5_000;
6
+ const isPortListening = (port) => new Promise((resolve) => {
7
+ const socket = net.createConnection({ port, host: "127.0.0.1" });
8
+ socket.once("connect", () => {
9
+ socket.end();
10
+ resolve(true);
11
+ });
12
+ socket.once("error", () => {
13
+ socket.destroy();
14
+ resolve(false);
15
+ });
16
+ });
17
+ export function useOpenBrowser(port, enabled) {
18
+ const opened = useRef(false);
19
+ useEffect(() => {
20
+ if (!enabled || opened.current) {
21
+ return;
22
+ }
23
+ let cancelled = false;
24
+ const deadline = Date.now() + POLL_TIMEOUT_MS;
25
+ const tick = async () => {
26
+ while (!cancelled && Date.now() < deadline) {
27
+ if (await isPortListening(port)) {
28
+ if (cancelled || opened.current) {
29
+ return;
30
+ }
31
+ opened.current = true;
32
+ await open(`http://localhost:${port}/`).catch(() => { });
33
+ return;
34
+ }
35
+ await new Promise((r) => setTimeout(r, POLL_INTERVAL_MS));
36
+ }
37
+ };
38
+ void tick();
39
+ return () => {
40
+ cancelled = true;
41
+ };
42
+ }, [port, enabled]);
43
+ }
44
+ //# sourceMappingURL=use-open-browser.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"use-open-browser.js","sourceRoot":"","sources":["../../src/cli/use-open-browser.ts"],"names":[],"mappings":"AAAA,OAAO,GAAG,MAAM,UAAU,CAAC;AAC3B,OAAO,IAAI,MAAM,MAAM,CAAC;AACxB,OAAO,EAAE,SAAS,EAAE,MAAM,EAAE,MAAM,OAAO,CAAC;AAE1C,MAAM,gBAAgB,GAAG,GAAG,CAAC;AAC7B,MAAM,eAAe,GAAG,KAAK,CAAC;AAE9B,MAAM,eAAe,GAAG,CAAC,IAAY,EAAoB,EAAE,CACzD,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE;IACtB,MAAM,MAAM,GAAG,GAAG,CAAC,gBAAgB,CAAC,EAAE,IAAI,EAAE,IAAI,EAAE,WAAW,EAAE,CAAC,CAAC;IACjE,MAAM,CAAC,IAAI,CAAC,SAAS,EAAE,GAAG,EAAE;QAC1B,MAAM,CAAC,GAAG,EAAE,CAAC;QACb,OAAO,CAAC,IAAI,CAAC,CAAC;IAChB,CAAC,CAAC,CAAC;IACH,MAAM,CAAC,IAAI,CAAC,OAAO,EAAE,GAAG,EAAE;QACxB,MAAM,CAAC,OAAO,EAAE,CAAC;QACjB,OAAO,CAAC,KAAK,CAAC,CAAC;IACjB,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC;AAEL,MAAM,UAAU,cAAc,CAAC,IAAY,EAAE,OAAgB;IAC3D,MAAM,MAAM,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;IAE7B,SAAS,CAAC,GAAG,EAAE;QACb,IAAI,CAAC,OAAO,IAAI,MAAM,CAAC,OAAO,EAAE,CAAC;YAC/B,OAAO;QACT,CAAC;QAED,IAAI,SAAS,GAAG,KAAK,CAAC;QACtB,MAAM,QAAQ,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,eAAe,CAAC;QAE9C,MAAM,IAAI,GAAG,KAAK,IAAI,EAAE;YACtB,OAAO,CAAC,SAAS,IAAI,IAAI,CAAC,GAAG,EAAE,GAAG,QAAQ,EAAE,CAAC;gBAC3C,IAAI,MAAM,eAAe,CAAC,IAAI,CAAC,EAAE,CAAC;oBAChC,IAAI,SAAS,IAAI,MAAM,CAAC,OAAO,EAAE,CAAC;wBAChC,OAAO;oBACT,CAAC;oBACD,MAAM,CAAC,OAAO,GAAG,IAAI,CAAC;oBACtB,MAAM,IAAI,CAAC,oBAAoB,IAAI,GAAG,CAAC,CAAC,KAAK,CAAC,GAAG,EAAE,GAAE,CAAC,CAAC,CAAC;oBACxD,OAAO;gBACT,CAAC;gBACD,MAAM,IAAI,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,UAAU,CAAC,CAAC,EAAE,gBAAgB,CAAC,CAAC,CAAC;YAC5D,CAAC;QACH,CAAC,CAAC;QAEF,KAAK,IAAI,EAAE,CAAC;QAEZ,OAAO,GAAG,EAAE;YACV,SAAS,GAAG,IAAI,CAAC;QACnB,CAAC,CAAC;IACJ,CAAC,EAAE,CAAC,IAAI,EAAE,OAAO,CAAC,CAAC,CAAC;AACtB,CAAC","sourcesContent":["import net from \"node:net\";\nimport open from \"open\";\nimport { useEffect, useRef } from \"react\";\n\nconst POLL_INTERVAL_MS = 200;\nconst POLL_TIMEOUT_MS = 5_000;\n\nconst isPortListening = (port: number): Promise<boolean> =>\n new Promise((resolve) => {\n const socket = net.createConnection({ port, host: \"127.0.0.1\" });\n socket.once(\"connect\", () => {\n socket.end();\n resolve(true);\n });\n socket.once(\"error\", () => {\n socket.destroy();\n resolve(false);\n });\n });\n\nexport function useOpenBrowser(port: number, enabled: boolean): void {\n const opened = useRef(false);\n\n useEffect(() => {\n if (!enabled || opened.current) {\n return;\n }\n\n let cancelled = false;\n const deadline = Date.now() + POLL_TIMEOUT_MS;\n\n const tick = async () => {\n while (!cancelled && Date.now() < deadline) {\n if (await isPortListening(port)) {\n if (cancelled || opened.current) {\n return;\n }\n opened.current = true;\n await open(`http://localhost:${port}/`).catch(() => {});\n return;\n }\n await new Promise((r) => setTimeout(r, POLL_INTERVAL_MS));\n }\n };\n\n void tick();\n\n return () => {\n cancelled = true;\n };\n }, [port, enabled]);\n}\n"]}
@@ -0,0 +1,14 @@
1
+ import type { PushMessage } from "./use-messages.js";
2
+ export type TunnelState = {
3
+ status: "idle";
4
+ } | {
5
+ status: "starting";
6
+ message: string;
7
+ } | {
8
+ status: "connected";
9
+ url: string;
10
+ } | {
11
+ status: "error";
12
+ message: string;
13
+ };
14
+ export declare function useTunnel(port: number | null, pushMessage: PushMessage, verbose: boolean, autoStart: boolean): TunnelState;
@@ -0,0 +1,131 @@
1
+ import { useEffect, useState } from "react";
2
+ const POST_RETRY_DELAY_MS = 250;
3
+ export function useTunnel(port, pushMessage, verbose, autoStart) {
4
+ const [state, setState] = useState(port !== null && autoStart
5
+ ? { status: "starting", message: "Starting tunnel…" }
6
+ : { status: "idle" });
7
+ useEffect(() => {
8
+ if (port === null) {
9
+ return;
10
+ }
11
+ const baseUrl = `http://localhost:${port}`;
12
+ const controller = new AbortController();
13
+ let cancelled = false;
14
+ const pushLog = (text, type) => {
15
+ const time = new Date().toLocaleTimeString("en-US", {
16
+ hour: "numeric",
17
+ minute: "2-digit",
18
+ second: "2-digit",
19
+ hour12: true,
20
+ });
21
+ pushMessage(`${time} [tunnel] ${text}`, type);
22
+ };
23
+ const handleEvent = (event, data) => {
24
+ if (event === "state") {
25
+ const next = JSON.parse(data);
26
+ setState(next);
27
+ return;
28
+ }
29
+ if (event === "activity") {
30
+ if (!verbose) {
31
+ return;
32
+ }
33
+ const activity = JSON.parse(data);
34
+ pushLog(activity.text, activity.level);
35
+ }
36
+ };
37
+ const consumeSse = async () => {
38
+ const res = await fetch(`${baseUrl}/__skybridge/tunnel/events`, {
39
+ signal: controller.signal,
40
+ headers: { Accept: "text/event-stream" },
41
+ });
42
+ if (!res.ok || !res.body) {
43
+ throw new Error(`SSE connection failed (${res.status})`);
44
+ }
45
+ const reader = res.body.getReader();
46
+ const decoder = new TextDecoder();
47
+ let buffer = "";
48
+ while (!cancelled) {
49
+ const { value, done } = await reader.read();
50
+ if (done) {
51
+ return;
52
+ }
53
+ buffer += decoder.decode(value, { stream: true });
54
+ // SSE frames are separated by a blank line ("\n\n").
55
+ let sep = buffer.indexOf("\n\n");
56
+ while (sep !== -1) {
57
+ const frame = buffer.slice(0, sep);
58
+ buffer = buffer.slice(sep + 2);
59
+ let eventName = "message";
60
+ const dataLines = [];
61
+ for (const rawLine of frame.split("\n")) {
62
+ const line = rawLine.replace(/\r$/, "");
63
+ if (line.startsWith("event:")) {
64
+ eventName = line.slice(6).trim();
65
+ }
66
+ else if (line.startsWith("data:")) {
67
+ dataLines.push(line.slice(5).trimStart());
68
+ }
69
+ }
70
+ if (dataLines.length > 0) {
71
+ handleEvent(eventName, dataLines.join("\n"));
72
+ }
73
+ sep = buffer.indexOf("\n\n");
74
+ }
75
+ }
76
+ };
77
+ const postUntilStarted = async () => {
78
+ // Retry indefinitely until POST lands once. Bounded by the effect
79
+ // lifetime via controller.abort() on unmount. Once the manager has
80
+ // been started, this returns and never POSTs again — a user-driven
81
+ // DELETE /tunnel won't be auto-undone.
82
+ while (!cancelled) {
83
+ try {
84
+ const res = await fetch(`${baseUrl}/__skybridge/tunnel`, {
85
+ method: "POST",
86
+ signal: controller.signal,
87
+ });
88
+ if (res.ok) {
89
+ return;
90
+ }
91
+ }
92
+ catch {
93
+ // dev server not up yet (or restarting under nodemon) — wait, retry
94
+ }
95
+ await new Promise((r) => setTimeout(r, POST_RETRY_DELAY_MS));
96
+ }
97
+ };
98
+ // Always observe the tunnel state so external triggers (curl, future
99
+ // devtools UI) update the cli UI. `autoStart` only decides whether we
100
+ // also POST /tunnel on mount. Reconnects indefinitely so we survive
101
+ // dev-server boot delay and nodemon restarts; the cli owns the actual
102
+ // subprocess so a temporarily-unreachable dev server is fine.
103
+ const observe = async () => {
104
+ while (!cancelled) {
105
+ try {
106
+ await consumeSse();
107
+ }
108
+ catch {
109
+ // network error or stream ended abnormally — fall through to retry
110
+ }
111
+ if (cancelled) {
112
+ return;
113
+ }
114
+ await new Promise((r) => setTimeout(r, POST_RETRY_DELAY_MS));
115
+ }
116
+ };
117
+ // observe() always runs — it owns the cli's view of tunnel state. POST is
118
+ // a fire-and-forget side-effect that nudges the manager into starting and
119
+ // retries until it lands at least once.
120
+ if (autoStart) {
121
+ void postUntilStarted();
122
+ }
123
+ void observe();
124
+ return () => {
125
+ cancelled = true;
126
+ controller.abort();
127
+ };
128
+ }, [port, pushMessage, verbose, autoStart]);
129
+ return state;
130
+ }
131
+ //# sourceMappingURL=use-tunnel.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"use-tunnel.js","sourceRoot":"","sources":["../../src/cli/use-tunnel.ts"],"names":[],"mappings":"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","sourcesContent":["import { useEffect, useState } from \"react\";\nimport type { PushMessage } from \"./use-messages.js\";\n\nexport type TunnelState =\n | { status: \"idle\" }\n | { status: \"starting\"; message: string }\n | { status: \"connected\"; url: string }\n | { status: \"error\"; message: string };\n\ntype TunnelActivity = {\n time: string;\n text: string;\n level: \"log\" | \"error\";\n};\n\nconst POST_RETRY_DELAY_MS = 250;\n\nexport function useTunnel(\n port: number | null,\n pushMessage: PushMessage,\n verbose: boolean,\n autoStart: boolean,\n): TunnelState {\n const [state, setState] = useState<TunnelState>(\n port !== null && autoStart\n ? { status: \"starting\", message: \"Starting tunnel…\" }\n : { status: \"idle\" },\n );\n\n useEffect(() => {\n if (port === null) {\n return;\n }\n\n const baseUrl = `http://localhost:${port}`;\n const controller = new AbortController();\n let cancelled = false;\n\n const pushLog = (text: string, type: \"log\" | \"error\") => {\n const time = new Date().toLocaleTimeString(\"en-US\", {\n hour: \"numeric\",\n minute: \"2-digit\",\n second: \"2-digit\",\n hour12: true,\n });\n pushMessage(`${time} [tunnel] ${text}`, type);\n };\n\n const handleEvent = (event: string, data: string) => {\n if (event === \"state\") {\n const next = JSON.parse(data) as TunnelState;\n setState(next);\n return;\n }\n if (event === \"activity\") {\n if (!verbose) {\n return;\n }\n const activity = JSON.parse(data) as TunnelActivity;\n pushLog(activity.text, activity.level);\n }\n };\n\n const consumeSse = async () => {\n const res = await fetch(`${baseUrl}/__skybridge/tunnel/events`, {\n signal: controller.signal,\n headers: { Accept: \"text/event-stream\" },\n });\n if (!res.ok || !res.body) {\n throw new Error(`SSE connection failed (${res.status})`);\n }\n\n const reader = res.body.getReader();\n const decoder = new TextDecoder();\n let buffer = \"\";\n\n while (!cancelled) {\n const { value, done } = await reader.read();\n if (done) {\n return;\n }\n buffer += decoder.decode(value, { stream: true });\n\n // SSE frames are separated by a blank line (\"\\n\\n\").\n let sep = buffer.indexOf(\"\\n\\n\");\n while (sep !== -1) {\n const frame = buffer.slice(0, sep);\n buffer = buffer.slice(sep + 2);\n let eventName = \"message\";\n const dataLines: string[] = [];\n for (const rawLine of frame.split(\"\\n\")) {\n const line = rawLine.replace(/\\r$/, \"\");\n if (line.startsWith(\"event:\")) {\n eventName = line.slice(6).trim();\n } else if (line.startsWith(\"data:\")) {\n dataLines.push(line.slice(5).trimStart());\n }\n }\n if (dataLines.length > 0) {\n handleEvent(eventName, dataLines.join(\"\\n\"));\n }\n sep = buffer.indexOf(\"\\n\\n\");\n }\n }\n };\n\n const postUntilStarted = async () => {\n // Retry indefinitely until POST lands once. Bounded by the effect\n // lifetime via controller.abort() on unmount. Once the manager has\n // been started, this returns and never POSTs again — a user-driven\n // DELETE /tunnel won't be auto-undone.\n while (!cancelled) {\n try {\n const res = await fetch(`${baseUrl}/__skybridge/tunnel`, {\n method: \"POST\",\n signal: controller.signal,\n });\n if (res.ok) {\n return;\n }\n } catch {\n // dev server not up yet (or restarting under nodemon) — wait, retry\n }\n await new Promise((r) => setTimeout(r, POST_RETRY_DELAY_MS));\n }\n };\n\n // Always observe the tunnel state so external triggers (curl, future\n // devtools UI) update the cli UI. `autoStart` only decides whether we\n // also POST /tunnel on mount. Reconnects indefinitely so we survive\n // dev-server boot delay and nodemon restarts; the cli owns the actual\n // subprocess so a temporarily-unreachable dev server is fine.\n const observe = async () => {\n while (!cancelled) {\n try {\n await consumeSse();\n } catch {\n // network error or stream ended abnormally — fall through to retry\n }\n if (cancelled) {\n return;\n }\n await new Promise((r) => setTimeout(r, POST_RETRY_DELAY_MS));\n }\n };\n\n // observe() always runs — it owns the cli's view of tunnel state. POST is\n // a fire-and-forget side-effect that nudges the manager into starting and\n // retries until it lands at least once.\n if (autoStart) {\n void postUntilStarted();\n }\n void observe();\n\n return () => {\n cancelled = true;\n controller.abort();\n };\n }, [port, pushMessage, verbose, autoStart]);\n\n return state;\n}\n"]}