skybridge 0.0.0-dev.ed8e93c → 0.0.0-dev.ed994f4

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 (427) hide show
  1. package/LICENSE +21 -674
  2. package/README.md +140 -15
  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/{src/server → server}/inferUtilityTypes.d.ts +15 -5
  89. package/dist/server/inferUtilityTypes.js.map +1 -0
  90. package/dist/server/metric.d.ts +14 -0
  91. package/dist/server/metric.js +62 -0
  92. package/dist/server/metric.js.map +1 -0
  93. package/dist/server/middleware.d.ts +124 -0
  94. package/dist/server/middleware.js +93 -0
  95. package/dist/server/middleware.js.map +1 -0
  96. package/dist/server/middleware.test-d.js +75 -0
  97. package/dist/server/middleware.test-d.js.map +1 -0
  98. package/dist/server/middleware.test.js +493 -0
  99. package/dist/server/middleware.test.js.map +1 -0
  100. package/dist/server/server.d.ts +196 -0
  101. package/dist/server/server.js +468 -0
  102. package/dist/server/server.js.map +1 -0
  103. package/dist/{src/server → server}/templateHelper.d.ts +5 -4
  104. package/dist/server/templateHelper.js +11 -0
  105. package/dist/server/templateHelper.js.map +1 -0
  106. package/dist/server/templates.generated.d.ts +4 -0
  107. package/dist/server/templates.generated.js +47 -0
  108. package/dist/server/templates.generated.js.map +1 -0
  109. package/dist/server/tunnel-proxy-router.d.ts +7 -0
  110. package/dist/server/tunnel-proxy-router.js +110 -0
  111. package/dist/server/tunnel-proxy-router.js.map +1 -0
  112. package/dist/server/tunnel-proxy-router.test.js +229 -0
  113. package/dist/server/tunnel-proxy-router.test.js.map +1 -0
  114. package/dist/server/viewsDevServer.d.ts +14 -0
  115. package/dist/server/viewsDevServer.js +45 -0
  116. package/dist/server/viewsDevServer.js.map +1 -0
  117. package/dist/test/utils.d.ts +127 -0
  118. package/dist/test/utils.js +247 -0
  119. package/dist/test/utils.js.map +1 -0
  120. package/dist/test/view.test.js +523 -0
  121. package/dist/test/view.test.js.map +1 -0
  122. package/dist/version.d.ts +1 -0
  123. package/dist/version.js +3 -0
  124. package/dist/version.js.map +1 -0
  125. package/dist/web/bridges/apps-sdk/adaptor.d.ts +24 -0
  126. package/dist/web/bridges/apps-sdk/adaptor.js +96 -0
  127. package/dist/web/bridges/apps-sdk/adaptor.js.map +1 -0
  128. package/dist/web/bridges/apps-sdk/bridge.d.ts +10 -0
  129. package/dist/web/bridges/apps-sdk/bridge.js +46 -0
  130. package/dist/web/bridges/apps-sdk/bridge.js.map +1 -0
  131. package/dist/web/bridges/apps-sdk/index.d.ts +5 -0
  132. package/dist/web/bridges/apps-sdk/index.js +5 -0
  133. package/dist/web/bridges/apps-sdk/index.js.map +1 -0
  134. package/dist/web/bridges/apps-sdk/types.d.ts +131 -0
  135. package/dist/{src/web → web/bridges/apps-sdk}/types.js +0 -1
  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.js +38 -0
  173. package/dist/web/create-store.js.map +1 -0
  174. package/dist/web/create-store.test.js +129 -0
  175. package/dist/web/create-store.test.js.map +1 -0
  176. package/dist/{src/web → web}/data-llm.d.ts +1 -1
  177. package/dist/{src/web → web}/data-llm.js +13 -9
  178. package/dist/web/data-llm.js.map +1 -0
  179. package/dist/web/data-llm.test.js +142 -0
  180. package/dist/web/data-llm.test.js.map +1 -0
  181. package/dist/{src/web → web}/generate-helpers.d.ts +29 -24
  182. package/dist/{src/web → web}/generate-helpers.js +24 -20
  183. package/dist/web/generate-helpers.js.map +1 -0
  184. package/dist/{src/web → web}/generate-helpers.test-d.js +79 -23
  185. package/dist/web/generate-helpers.test-d.js.map +1 -0
  186. package/dist/web/generate-helpers.test.js.map +1 -0
  187. package/dist/{src/web → web}/helpers/state.d.ts +2 -2
  188. package/dist/web/helpers/state.js +45 -0
  189. package/dist/web/helpers/state.js.map +1 -0
  190. package/dist/{src/web → web}/helpers/state.test.js +10 -10
  191. package/dist/web/helpers/state.test.js.map +1 -0
  192. package/dist/{src/web → web}/hooks/index.d.ts +5 -6
  193. package/dist/{src/web → web}/hooks/index.js +4 -5
  194. package/dist/web/hooks/index.js.map +1 -0
  195. package/dist/web/hooks/test/utils.d.ts +16 -0
  196. package/dist/web/hooks/test/utils.js +64 -0
  197. package/dist/web/hooks/test/utils.js.map +1 -0
  198. package/dist/{src/web → web}/hooks/use-call-tool.d.ts +3 -2
  199. package/dist/{src/web → web}/hooks/use-call-tool.js +12 -4
  200. package/dist/web/hooks/use-call-tool.js.map +1 -0
  201. package/dist/web/hooks/use-call-tool.test-d.js.map +1 -0
  202. package/dist/{src/web → web}/hooks/use-call-tool.test.js +34 -19
  203. package/dist/web/hooks/use-call-tool.test.js.map +1 -0
  204. package/dist/web/hooks/use-display-mode.d.ts +4 -0
  205. package/dist/web/hooks/use-display-mode.js +9 -0
  206. package/dist/web/hooks/use-display-mode.js.map +1 -0
  207. package/dist/web/hooks/use-display-mode.test-d.js +8 -0
  208. package/dist/web/hooks/use-display-mode.test-d.js.map +1 -0
  209. package/dist/{src/web → web}/hooks/use-display-mode.test.js +1 -0
  210. package/dist/web/hooks/use-display-mode.test.js.map +1 -0
  211. package/dist/web/hooks/use-files.d.ts +7 -0
  212. package/dist/web/hooks/use-files.js +10 -0
  213. package/dist/web/hooks/use-files.js.map +1 -0
  214. package/dist/web/hooks/use-files.test.d.ts +1 -0
  215. package/dist/web/hooks/use-files.test.js +54 -0
  216. package/dist/web/hooks/use-files.test.js.map +1 -0
  217. package/dist/web/hooks/use-layout.d.ts +22 -0
  218. package/dist/web/hooks/use-layout.js +23 -0
  219. package/dist/web/hooks/use-layout.js.map +1 -0
  220. package/dist/web/hooks/use-layout.test.d.ts +1 -0
  221. package/dist/web/hooks/use-layout.test.js +96 -0
  222. package/dist/web/hooks/use-layout.test.js.map +1 -0
  223. package/dist/web/hooks/use-open-external.d.ts +3 -0
  224. package/dist/web/hooks/use-open-external.js +8 -0
  225. package/dist/web/hooks/use-open-external.js.map +1 -0
  226. package/dist/web/hooks/use-open-external.test.d.ts +1 -0
  227. package/dist/web/hooks/use-open-external.test.js +65 -0
  228. package/dist/web/hooks/use-open-external.test.js.map +1 -0
  229. package/dist/web/hooks/use-request-modal.d.ts +9 -0
  230. package/dist/web/hooks/use-request-modal.js +16 -0
  231. package/dist/web/hooks/use-request-modal.js.map +1 -0
  232. package/dist/web/hooks/use-request-modal.test.d.ts +1 -0
  233. package/dist/web/hooks/use-request-modal.test.js +61 -0
  234. package/dist/web/hooks/use-request-modal.test.js.map +1 -0
  235. package/dist/web/hooks/use-send-follow-up-message.d.ts +2 -0
  236. package/dist/web/hooks/use-send-follow-up-message.js +8 -0
  237. package/dist/web/hooks/use-send-follow-up-message.js.map +1 -0
  238. package/dist/web/hooks/use-set-open-in-app-url.d.ts +1 -0
  239. package/dist/web/hooks/use-set-open-in-app-url.js +8 -0
  240. package/dist/web/hooks/use-set-open-in-app-url.js.map +1 -0
  241. package/dist/web/hooks/use-set-open-in-app-url.test.d.ts +1 -0
  242. package/dist/web/hooks/use-set-open-in-app-url.test.js +43 -0
  243. package/dist/web/hooks/use-set-open-in-app-url.test.js.map +1 -0
  244. package/dist/{src/web → web}/hooks/use-tool-info.d.ts +13 -2
  245. package/dist/web/hooks/use-tool-info.js +26 -0
  246. package/dist/web/hooks/use-tool-info.js.map +1 -0
  247. package/dist/web/hooks/use-tool-info.test-d.d.ts +1 -0
  248. package/dist/{src/web → web}/hooks/use-tool-info.test-d.js +40 -5
  249. package/dist/web/hooks/use-tool-info.test-d.js.map +1 -0
  250. package/dist/web/hooks/use-tool-info.test.d.ts +1 -0
  251. package/dist/web/hooks/use-tool-info.test.js +130 -0
  252. package/dist/web/hooks/use-tool-info.test.js.map +1 -0
  253. package/dist/web/hooks/use-user.d.ts +18 -0
  254. package/dist/web/hooks/use-user.js +35 -0
  255. package/dist/web/hooks/use-user.js.map +1 -0
  256. package/dist/web/hooks/use-user.test.d.ts +1 -0
  257. package/dist/web/hooks/use-user.test.js +122 -0
  258. package/dist/web/hooks/use-user.test.js.map +1 -0
  259. package/dist/web/hooks/use-view-state.d.ts +4 -0
  260. package/dist/web/hooks/use-view-state.js +32 -0
  261. package/dist/web/hooks/use-view-state.js.map +1 -0
  262. package/dist/web/hooks/use-view-state.test.d.ts +1 -0
  263. package/dist/web/hooks/use-view-state.test.js +177 -0
  264. package/dist/web/hooks/use-view-state.test.js.map +1 -0
  265. package/dist/{src/web → web}/index.d.ts +2 -2
  266. package/dist/{src/web → web}/index.js +2 -2
  267. package/dist/web/index.js.map +1 -0
  268. package/dist/web/mount-view.d.ts +1 -0
  269. package/dist/{src/web/mount-widget.js → web/mount-view.js} +11 -3
  270. package/dist/web/mount-view.js.map +1 -0
  271. package/dist/web/plugin/data-llm.test.d.ts +1 -0
  272. package/dist/web/plugin/data-llm.test.js.map +1 -0
  273. package/dist/web/plugin/plugin.d.ts +5 -0
  274. package/dist/web/plugin/plugin.js +156 -0
  275. package/dist/web/plugin/plugin.js.map +1 -0
  276. package/dist/web/plugin/scan-views.d.ts +16 -0
  277. package/dist/web/plugin/scan-views.js +88 -0
  278. package/dist/web/plugin/scan-views.js.map +1 -0
  279. package/dist/web/plugin/scan-views.test.d.ts +1 -0
  280. package/dist/web/plugin/scan-views.test.js +99 -0
  281. package/dist/web/plugin/scan-views.test.js.map +1 -0
  282. package/dist/{src/web → web}/plugin/transform-data-llm.js +7 -4
  283. package/dist/web/plugin/transform-data-llm.js.map +1 -0
  284. package/dist/web/plugin/transform-data-llm.test.d.ts +1 -0
  285. package/dist/web/plugin/transform-data-llm.test.js.map +1 -0
  286. package/dist/web/plugin/validate-view.d.ts +1 -0
  287. package/dist/web/plugin/validate-view.js +9 -0
  288. package/dist/web/plugin/validate-view.js.map +1 -0
  289. package/dist/web/plugin/validate-view.test.d.ts +1 -0
  290. package/dist/web/plugin/validate-view.test.js +24 -0
  291. package/dist/web/plugin/validate-view.test.js.map +1 -0
  292. package/dist/{src/web → web}/proxy.js +5 -1
  293. package/dist/web/proxy.js.map +1 -0
  294. package/dist/web/types.d.ts +16 -0
  295. package/dist/web/types.js +2 -0
  296. package/dist/web/types.js.map +1 -0
  297. package/package.json +79 -41
  298. package/tsconfig.base.json +33 -0
  299. package/dist/src/server/index.d.ts +0 -4
  300. package/dist/src/server/index.js +0 -3
  301. package/dist/src/server/index.js.map +0 -1
  302. package/dist/src/server/inferUtilityTypes.js.map +0 -1
  303. package/dist/src/server/server.d.ts +0 -48
  304. package/dist/src/server/server.js +0 -62
  305. package/dist/src/server/server.js.map +0 -1
  306. package/dist/src/server/templateHelper.js +0 -29
  307. package/dist/src/server/templateHelper.js.map +0 -1
  308. package/dist/src/server/templates/development.hbs +0 -12
  309. package/dist/src/server/templates/production.hbs +0 -6
  310. package/dist/src/server/widgetsDevServer.d.ts +0 -12
  311. package/dist/src/server/widgetsDevServer.js +0 -39
  312. package/dist/src/server/widgetsDevServer.js.map +0 -1
  313. package/dist/src/test/utils.d.ts +0 -89
  314. package/dist/src/test/utils.js +0 -164
  315. package/dist/src/test/utils.js.map +0 -1
  316. package/dist/src/test/widget.test.js +0 -89
  317. package/dist/src/test/widget.test.js.map +0 -1
  318. package/dist/src/web/create-store.js +0 -25
  319. package/dist/src/web/create-store.js.map +0 -1
  320. package/dist/src/web/create-store.test.js +0 -70
  321. package/dist/src/web/create-store.test.js.map +0 -1
  322. package/dist/src/web/data-llm.js.map +0 -1
  323. package/dist/src/web/data-llm.test.js +0 -76
  324. package/dist/src/web/data-llm.test.js.map +0 -1
  325. package/dist/src/web/generate-helpers.js.map +0 -1
  326. package/dist/src/web/generate-helpers.test-d.js.map +0 -1
  327. package/dist/src/web/generate-helpers.test.js.map +0 -1
  328. package/dist/src/web/helpers/state.js +0 -40
  329. package/dist/src/web/helpers/state.js.map +0 -1
  330. package/dist/src/web/helpers/state.test.js.map +0 -1
  331. package/dist/src/web/hooks/index.js.map +0 -1
  332. package/dist/src/web/hooks/use-call-tool.js.map +0 -1
  333. package/dist/src/web/hooks/use-call-tool.test-d.js.map +0 -1
  334. package/dist/src/web/hooks/use-call-tool.test.js.map +0 -1
  335. package/dist/src/web/hooks/use-display-mode.d.ts +0 -4
  336. package/dist/src/web/hooks/use-display-mode.js +0 -7
  337. package/dist/src/web/hooks/use-display-mode.js.map +0 -1
  338. package/dist/src/web/hooks/use-display-mode.test.js.map +0 -1
  339. package/dist/src/web/hooks/use-files.d.ts +0 -10
  340. package/dist/src/web/hooks/use-files.js +0 -7
  341. package/dist/src/web/hooks/use-files.js.map +0 -1
  342. package/dist/src/web/hooks/use-files.test.js +0 -29
  343. package/dist/src/web/hooks/use-files.test.js.map +0 -1
  344. package/dist/src/web/hooks/use-locale.d.ts +0 -1
  345. package/dist/src/web/hooks/use-locale.js +0 -5
  346. package/dist/src/web/hooks/use-locale.js.map +0 -1
  347. package/dist/src/web/hooks/use-locale.test.js +0 -21
  348. package/dist/src/web/hooks/use-locale.test.js.map +0 -1
  349. package/dist/src/web/hooks/use-open-external.d.ts +0 -1
  350. package/dist/src/web/hooks/use-open-external.js +0 -6
  351. package/dist/src/web/hooks/use-open-external.js.map +0 -1
  352. package/dist/src/web/hooks/use-open-external.test.js +0 -24
  353. package/dist/src/web/hooks/use-open-external.test.js.map +0 -1
  354. package/dist/src/web/hooks/use-openai-global.d.ts +0 -2
  355. package/dist/src/web/hooks/use-openai-global.js +0 -23
  356. package/dist/src/web/hooks/use-openai-global.js.map +0 -1
  357. package/dist/src/web/hooks/use-request-modal.d.ts +0 -6
  358. package/dist/src/web/hooks/use-request-modal.js +0 -9
  359. package/dist/src/web/hooks/use-request-modal.js.map +0 -1
  360. package/dist/src/web/hooks/use-request-modal.test.js +0 -24
  361. package/dist/src/web/hooks/use-request-modal.test.js.map +0 -1
  362. package/dist/src/web/hooks/use-send-follow-up-message.d.ts +0 -1
  363. package/dist/src/web/hooks/use-send-follow-up-message.js +0 -11
  364. package/dist/src/web/hooks/use-send-follow-up-message.js.map +0 -1
  365. package/dist/src/web/hooks/use-theme.d.ts +0 -1
  366. package/dist/src/web/hooks/use-theme.js +0 -5
  367. package/dist/src/web/hooks/use-theme.js.map +0 -1
  368. package/dist/src/web/hooks/use-theme.test.js +0 -26
  369. package/dist/src/web/hooks/use-theme.test.js.map +0 -1
  370. package/dist/src/web/hooks/use-tool-info.js +0 -22
  371. package/dist/src/web/hooks/use-tool-info.js.map +0 -1
  372. package/dist/src/web/hooks/use-tool-info.test-d.js.map +0 -1
  373. package/dist/src/web/hooks/use-tool-info.test.js +0 -59
  374. package/dist/src/web/hooks/use-tool-info.test.js.map +0 -1
  375. package/dist/src/web/hooks/use-user-agent.d.ts +0 -1
  376. package/dist/src/web/hooks/use-user-agent.js +0 -5
  377. package/dist/src/web/hooks/use-user-agent.js.map +0 -1
  378. package/dist/src/web/hooks/use-user-agent.test.js +0 -31
  379. package/dist/src/web/hooks/use-user-agent.test.js.map +0 -1
  380. package/dist/src/web/hooks/use-widget-state.d.ts +0 -4
  381. package/dist/src/web/hooks/use-widget-state.js +0 -32
  382. package/dist/src/web/hooks/use-widget-state.js.map +0 -1
  383. package/dist/src/web/hooks/use-widget-state.test.js +0 -60
  384. package/dist/src/web/hooks/use-widget-state.test.js.map +0 -1
  385. package/dist/src/web/index.js.map +0 -1
  386. package/dist/src/web/mount-widget.d.ts +0 -1
  387. package/dist/src/web/mount-widget.js.map +0 -1
  388. package/dist/src/web/plugin/data-llm.test.js.map +0 -1
  389. package/dist/src/web/plugin/plugin.d.ts +0 -2
  390. package/dist/src/web/plugin/plugin.js +0 -33
  391. package/dist/src/web/plugin/plugin.js.map +0 -1
  392. package/dist/src/web/plugin/transform-data-llm.js.map +0 -1
  393. package/dist/src/web/plugin/transform-data-llm.test.js.map +0 -1
  394. package/dist/src/web/proxy.js.map +0 -1
  395. package/dist/src/web/types.d.ts +0 -133
  396. package/dist/src/web/types.js.map +0 -1
  397. package/dist/vitest.config.d.ts +0 -2
  398. package/dist/vitest.config.js +0 -8
  399. package/dist/vitest.config.js.map +0 -1
  400. /package/dist/{src/test/widget.test.d.ts → cli/tunnel-control-server.test.d.ts} +0 -0
  401. /package/dist/{src/web/create-store.test.d.ts → cli/tunnel-handler.test.d.ts} +0 -0
  402. /package/dist/{src/web/data-llm.test.d.ts → cli/tunnel.test.d.ts} +0 -0
  403. /package/dist/{src/web/generate-helpers.test-d.d.ts → server/asset-base-url-transform-plugin.test.d.ts} +0 -0
  404. /package/dist/{src/web/generate-helpers.test.d.ts → server/content-helpers.test.d.ts} +0 -0
  405. /package/dist/{src/web/helpers/state.test.d.ts → server/express.test.d.ts} +0 -0
  406. /package/dist/{src/server → server}/inferUtilityTypes.js +0 -0
  407. /package/dist/{src/web/hooks/use-call-tool.test-d.d.ts → server/middleware.test-d.d.ts} +0 -0
  408. /package/dist/{src/web/hooks/use-call-tool.test.d.ts → server/middleware.test.d.ts} +0 -0
  409. /package/dist/{src/web/hooks/use-display-mode.test.d.ts → server/tunnel-proxy-router.test.d.ts} +0 -0
  410. /package/dist/{src/web/hooks/use-files.test.d.ts → test/view.test.d.ts} +0 -0
  411. /package/dist/{src/web/hooks/use-locale.test.d.ts → web/bridges/mcp-app/use-mcp-app-context.test.d.ts} +0 -0
  412. /package/dist/{src/web → web}/create-store.d.ts +0 -0
  413. /package/dist/{src/web/hooks/use-open-external.test.d.ts → web/create-store.test.d.ts} +0 -0
  414. /package/dist/{src/web/plugin → web}/data-llm.test.d.ts +0 -0
  415. /package/dist/{src/web/hooks/use-request-modal.test.d.ts → web/generate-helpers.test-d.d.ts} +0 -0
  416. /package/dist/{src/web/hooks/use-theme.test.d.ts → web/generate-helpers.test.d.ts} +0 -0
  417. /package/dist/{src/web → web}/generate-helpers.test.js +0 -0
  418. /package/dist/{src/web/hooks/use-tool-info.test-d.d.ts → web/helpers/state.test.d.ts} +0 -0
  419. /package/dist/{src/web/hooks/use-tool-info.test.d.ts → web/hooks/use-call-tool.test-d.d.ts} +0 -0
  420. /package/dist/{src/web → web}/hooks/use-call-tool.test-d.js +0 -0
  421. /package/dist/{src/web/hooks/use-user-agent.test.d.ts → web/hooks/use-call-tool.test.d.ts} +0 -0
  422. /package/dist/{src/web/hooks/use-widget-state.test.d.ts → web/hooks/use-display-mode.test-d.d.ts} +0 -0
  423. /package/dist/{src/web/plugin/transform-data-llm.test.d.ts → web/hooks/use-display-mode.test.d.ts} +0 -0
  424. /package/dist/{src/web → web}/plugin/data-llm.test.js +0 -0
  425. /package/dist/{src/web → web}/plugin/transform-data-llm.d.ts +0 -0
  426. /package/dist/{src/web → web}/plugin/transform-data-llm.test.js +0 -0
  427. /package/dist/{src/web → web}/proxy.d.ts +0 -0
@@ -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"]}
@@ -0,0 +1,9 @@
1
+ type TsError = {
2
+ file: string;
3
+ line: number;
4
+ col: number;
5
+ code: string;
6
+ message: string;
7
+ };
8
+ export declare function useTypeScriptCheck(): Array<TsError>;
9
+ export {};
@@ -0,0 +1,94 @@
1
+ import { spawn } from "node:child_process";
2
+ import { isAbsolute, relative } from "node:path";
3
+ import { useEffect, useRef, useState } from "react";
4
+ // TypeScript nests from general to specific — the deepest line is the root cause.
5
+ function extractBestMessage(message, continuationLines) {
6
+ if (!continuationLines.length) {
7
+ return message;
8
+ }
9
+ let maxIndent = 0;
10
+ for (const line of continuationLines) {
11
+ const indent = line.length - line.trimStart().length;
12
+ if (indent > maxIndent) {
13
+ maxIndent = indent;
14
+ }
15
+ }
16
+ const deepest = continuationLines
17
+ .filter((l) => l.length - l.trimStart().length === maxIndent)
18
+ .map((l) => l.trim())
19
+ .filter(Boolean)[0];
20
+ return deepest ?? message;
21
+ }
22
+ export function useTypeScriptCheck() {
23
+ const tsProcessRef = useRef(null);
24
+ const [tsErrors, setTsErrors] = useState([]);
25
+ useEffect(() => {
26
+ const tsProcess = spawn("npx", ["tsc", "--noEmit", "--watch", "--pretty", "false"], {
27
+ stdio: ["ignore", "pipe", "pipe"],
28
+ shell: true,
29
+ });
30
+ tsProcessRef.current = tsProcess;
31
+ let outputBuffer = "";
32
+ let currentErrors = [];
33
+ let pendingError = null;
34
+ let continuationLines = [];
35
+ const flushPending = () => {
36
+ if (!pendingError) {
37
+ return;
38
+ }
39
+ pendingError.message = extractBestMessage(pendingError.message, continuationLines);
40
+ currentErrors.push(pendingError);
41
+ pendingError = null;
42
+ continuationLines = [];
43
+ };
44
+ const processOutput = (data) => {
45
+ outputBuffer += data.toString();
46
+ const lines = outputBuffer.split("\n");
47
+ outputBuffer = lines.pop() || "";
48
+ for (const line of lines) {
49
+ const trimmed = line.trim();
50
+ const errorMatch = trimmed.match(/^(.+?)\((\d+),(\d+)\):\s+error\s+(TS\d+)?\s*:?\s*(.+)$/);
51
+ if (errorMatch) {
52
+ flushPending();
53
+ const [, file, lineStr, colStr, code, message] = errorMatch;
54
+ if (file && lineStr && colStr && message) {
55
+ let cleanFile = file.trim();
56
+ if (isAbsolute(cleanFile)) {
57
+ cleanFile = relative(process.cwd(), cleanFile);
58
+ }
59
+ pendingError = {
60
+ file: cleanFile,
61
+ line: Number.parseInt(lineStr, 10),
62
+ col: Number.parseInt(colStr, 10),
63
+ code: code ?? "",
64
+ message: message.trim(),
65
+ };
66
+ }
67
+ continue;
68
+ }
69
+ if (pendingError && line.startsWith(" ")) {
70
+ continuationLines.push(line);
71
+ continue;
72
+ }
73
+ if (trimmed.includes("Found") && trimmed.includes("error")) {
74
+ flushPending();
75
+ setTsErrors(trimmed.match(/Found 0 error/) ? [] : [...currentErrors]);
76
+ currentErrors = [];
77
+ }
78
+ }
79
+ };
80
+ if (tsProcess.stdout) {
81
+ tsProcess.stdout.on("data", processOutput);
82
+ }
83
+ if (tsProcess.stderr) {
84
+ tsProcess.stderr.on("data", processOutput);
85
+ }
86
+ return () => {
87
+ if (tsProcessRef.current) {
88
+ tsProcessRef.current.kill();
89
+ }
90
+ };
91
+ }, []);
92
+ return tsErrors;
93
+ }
94
+ //# sourceMappingURL=use-typescript-check.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"use-typescript-check.js","sourceRoot":"","sources":["../../src/cli/use-typescript-check.ts"],"names":[],"mappings":"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","sourcesContent":["import { spawn } from \"node:child_process\";\nimport { isAbsolute, relative } from \"node:path\";\nimport { useEffect, useRef, useState } from \"react\";\n\ntype TsError = {\n file: string;\n line: number;\n col: number;\n code: string;\n message: string;\n};\n\n// TypeScript nests from general to specific — the deepest line is the root cause.\nfunction extractBestMessage(\n message: string,\n continuationLines: Array<string>,\n): string {\n if (!continuationLines.length) {\n return message;\n }\n let maxIndent = 0;\n for (const line of continuationLines) {\n const indent = line.length - line.trimStart().length;\n if (indent > maxIndent) {\n maxIndent = indent;\n }\n }\n const deepest = continuationLines\n .filter((l) => l.length - l.trimStart().length === maxIndent)\n .map((l) => l.trim())\n .filter(Boolean)[0];\n return deepest ?? message;\n}\n\nexport function useTypeScriptCheck(): Array<TsError> {\n const tsProcessRef = useRef<ReturnType<typeof spawn> | null>(null);\n const [tsErrors, setTsErrors] = useState<Array<TsError>>([]);\n\n useEffect(() => {\n const tsProcess = spawn(\n \"npx\",\n [\"tsc\", \"--noEmit\", \"--watch\", \"--pretty\", \"false\"],\n {\n stdio: [\"ignore\", \"pipe\", \"pipe\"],\n shell: true,\n },\n );\n\n tsProcessRef.current = tsProcess;\n\n let outputBuffer = \"\";\n let currentErrors: Array<TsError> = [];\n let pendingError: TsError | null = null;\n let continuationLines: Array<string> = [];\n\n const flushPending = () => {\n if (!pendingError) {\n return;\n }\n pendingError.message = extractBestMessage(\n pendingError.message,\n continuationLines,\n );\n currentErrors.push(pendingError);\n pendingError = null;\n continuationLines = [];\n };\n\n const processOutput = (data: Buffer) => {\n outputBuffer += data.toString();\n const lines = outputBuffer.split(\"\\n\");\n outputBuffer = lines.pop() || \"\";\n\n for (const line of lines) {\n const trimmed = line.trim();\n\n const errorMatch = trimmed.match(\n /^(.+?)\\((\\d+),(\\d+)\\):\\s+error\\s+(TS\\d+)?\\s*:?\\s*(.+)$/,\n );\n if (errorMatch) {\n flushPending();\n const [, file, lineStr, colStr, code, message] = errorMatch;\n if (file && lineStr && colStr && message) {\n let cleanFile = file.trim();\n if (isAbsolute(cleanFile)) {\n cleanFile = relative(process.cwd(), cleanFile);\n }\n pendingError = {\n file: cleanFile,\n line: Number.parseInt(lineStr, 10),\n col: Number.parseInt(colStr, 10),\n code: code ?? \"\",\n message: message.trim(),\n };\n }\n continue;\n }\n\n if (pendingError && line.startsWith(\" \")) {\n continuationLines.push(line);\n continue;\n }\n\n if (trimmed.includes(\"Found\") && trimmed.includes(\"error\")) {\n flushPending();\n setTsErrors(trimmed.match(/Found 0 error/) ? [] : [...currentErrors]);\n currentErrors = [];\n }\n }\n };\n\n if (tsProcess.stdout) {\n tsProcess.stdout.on(\"data\", processOutput);\n }\n if (tsProcess.stderr) {\n tsProcess.stderr.on(\"data\", processOutput);\n }\n\n return () => {\n if (tsProcessRef.current) {\n tsProcessRef.current.kill();\n }\n };\n }, []);\n\n return tsErrors;\n}\n"]}
@@ -0,0 +1,9 @@
1
+ import { Command } from "@oclif/core";
2
+ import { type CommandStep } from "../cli/use-execute-steps.js";
3
+ export declare const commandSteps: CommandStep[];
4
+ export default class Build extends Command {
5
+ static description: string;
6
+ static examples: string[];
7
+ static flags: {};
8
+ run(): Promise<void>;
9
+ }