skybridge 0.0.0-dev.f29d75c → 0.0.0-dev.f2d6084

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 (331) hide show
  1. package/README.md +123 -116
  2. package/dist/cli/build-helpers.d.ts +7 -0
  3. package/dist/cli/build-helpers.js +82 -0
  4. package/dist/cli/build-helpers.js.map +1 -0
  5. package/dist/cli/build-helpers.test.js +64 -0
  6. package/dist/cli/build-helpers.test.js.map +1 -0
  7. package/dist/cli/detect-port.d.ts +18 -0
  8. package/dist/cli/detect-port.js +50 -0
  9. package/dist/cli/detect-port.js.map +1 -0
  10. package/dist/cli/header.js +1 -1
  11. package/dist/cli/header.js.map +1 -1
  12. package/dist/cli/resolve-views-dir.d.ts +1 -0
  13. package/dist/cli/resolve-views-dir.js +17 -0
  14. package/dist/cli/resolve-views-dir.js.map +1 -0
  15. package/dist/cli/run-command.js.map +1 -1
  16. package/dist/cli/telemetry.js.map +1 -1
  17. package/dist/cli/tunnel-control-server.d.ts +9 -0
  18. package/dist/cli/tunnel-control-server.js +31 -0
  19. package/dist/cli/tunnel-control-server.js.map +1 -0
  20. package/dist/cli/tunnel-control-server.test.js +39 -0
  21. package/dist/cli/tunnel-control-server.test.js.map +1 -0
  22. package/dist/cli/tunnel-handler.d.ts +3 -0
  23. package/dist/cli/tunnel-handler.js +48 -0
  24. package/dist/cli/tunnel-handler.js.map +1 -0
  25. package/dist/cli/tunnel-handler.test.d.ts +1 -0
  26. package/dist/cli/tunnel-handler.test.js +105 -0
  27. package/dist/cli/tunnel-handler.test.js.map +1 -0
  28. package/dist/cli/tunnel.d.ts +57 -0
  29. package/dist/cli/tunnel.js +154 -0
  30. package/dist/cli/tunnel.js.map +1 -0
  31. package/dist/cli/tunnel.test.d.ts +1 -0
  32. package/dist/cli/tunnel.test.js +190 -0
  33. package/dist/cli/tunnel.test.js.map +1 -0
  34. package/dist/cli/types.d.ts +5 -0
  35. package/dist/cli/types.js +2 -0
  36. package/dist/cli/types.js.map +1 -0
  37. package/dist/cli/use-execute-steps.js.map +1 -1
  38. package/dist/cli/use-messages.d.ts +3 -0
  39. package/dist/cli/use-messages.js +11 -0
  40. package/dist/cli/use-messages.js.map +1 -0
  41. package/dist/cli/use-nodemon.d.ts +2 -6
  42. package/dist/cli/use-nodemon.js +18 -14
  43. package/dist/cli/use-nodemon.js.map +1 -1
  44. package/dist/cli/use-open-browser.d.ts +1 -0
  45. package/dist/cli/use-open-browser.js +44 -0
  46. package/dist/cli/use-open-browser.js.map +1 -0
  47. package/dist/cli/use-open-tunnel-browser.d.ts +6 -0
  48. package/dist/cli/use-open-tunnel-browser.js +19 -0
  49. package/dist/cli/use-open-tunnel-browser.js.map +1 -0
  50. package/dist/cli/use-tunnel.d.ts +14 -0
  51. package/dist/cli/use-tunnel.js +131 -0
  52. package/dist/cli/use-tunnel.js.map +1 -0
  53. package/dist/cli/use-typescript-check.d.ts +1 -0
  54. package/dist/cli/use-typescript-check.js +42 -7
  55. package/dist/cli/use-typescript-check.js.map +1 -1
  56. package/dist/commands/build.d.ts +0 -1
  57. package/dist/commands/build.js +52 -8
  58. package/dist/commands/build.js.map +1 -1
  59. package/dist/commands/create.d.ts +9 -0
  60. package/dist/commands/create.js +30 -0
  61. package/dist/commands/create.js.map +1 -0
  62. package/dist/commands/dev.d.ts +4 -1
  63. package/dist/commands/dev.js +75 -8
  64. package/dist/commands/dev.js.map +1 -1
  65. package/dist/commands/start.d.ts +3 -1
  66. package/dist/commands/start.js +34 -12
  67. package/dist/commands/start.js.map +1 -1
  68. package/dist/commands/telemetry/disable.js.map +1 -1
  69. package/dist/commands/telemetry/enable.js.map +1 -1
  70. package/dist/commands/telemetry/status.js.map +1 -1
  71. package/dist/server/asset-base-url-transform-plugin.d.ts +6 -6
  72. package/dist/server/asset-base-url-transform-plugin.js +25 -11
  73. package/dist/server/asset-base-url-transform-plugin.js.map +1 -1
  74. package/dist/server/asset-base-url-transform-plugin.test.js +92 -14
  75. package/dist/server/asset-base-url-transform-plugin.test.js.map +1 -1
  76. package/dist/server/auth.d.ts +20 -0
  77. package/dist/server/auth.js +28 -0
  78. package/dist/server/auth.js.map +1 -0
  79. package/dist/server/build-manifest.test.d.ts +1 -0
  80. package/dist/server/build-manifest.test.js +27 -0
  81. package/dist/server/build-manifest.test.js.map +1 -0
  82. package/dist/server/content-helpers.d.ts +67 -0
  83. package/dist/server/content-helpers.js +79 -0
  84. package/dist/server/content-helpers.js.map +1 -0
  85. package/dist/server/content-helpers.test.d.ts +1 -0
  86. package/dist/server/content-helpers.test.js +70 -0
  87. package/dist/server/content-helpers.test.js.map +1 -0
  88. package/dist/server/express.d.ts +7 -5
  89. package/dist/server/express.js +52 -23
  90. package/dist/server/express.js.map +1 -1
  91. package/dist/server/express.test.js +411 -25
  92. package/dist/server/express.test.js.map +1 -1
  93. package/dist/server/file-ref.d.ts +28 -0
  94. package/dist/server/file-ref.js +27 -0
  95. package/dist/server/file-ref.js.map +1 -0
  96. package/dist/server/index.d.ts +7 -3
  97. package/dist/server/index.js +5 -2
  98. package/dist/server/index.js.map +1 -1
  99. package/dist/server/inferUtilityTypes.d.ts +6 -6
  100. package/dist/server/inferUtilityTypes.js.map +1 -1
  101. package/dist/server/metric.d.ts +14 -0
  102. package/dist/server/metric.js +62 -0
  103. package/dist/server/metric.js.map +1 -0
  104. package/dist/server/middleware.d.ts +137 -0
  105. package/dist/server/middleware.js +93 -0
  106. package/dist/server/middleware.js.map +1 -0
  107. package/dist/server/middleware.test-d.d.ts +1 -0
  108. package/dist/server/middleware.test-d.js +75 -0
  109. package/dist/server/middleware.test-d.js.map +1 -0
  110. package/dist/server/middleware.test.d.ts +1 -0
  111. package/dist/server/middleware.test.js +493 -0
  112. package/dist/server/middleware.test.js.map +1 -0
  113. package/dist/server/server.d.ts +340 -64
  114. package/dist/server/server.js +507 -102
  115. package/dist/server/server.js.map +1 -1
  116. package/dist/server/templateHelper.d.ts +5 -8
  117. package/dist/server/templateHelper.js +3 -22
  118. package/dist/server/templateHelper.js.map +1 -1
  119. package/dist/server/templates.generated.d.ts +4 -0
  120. package/dist/server/templates.generated.js +47 -0
  121. package/dist/server/templates.generated.js.map +1 -0
  122. package/dist/server/tunnel-proxy-router.d.ts +7 -0
  123. package/dist/server/tunnel-proxy-router.js +110 -0
  124. package/dist/server/tunnel-proxy-router.js.map +1 -0
  125. package/dist/server/tunnel-proxy-router.test.d.ts +1 -0
  126. package/dist/server/tunnel-proxy-router.test.js +229 -0
  127. package/dist/server/tunnel-proxy-router.test.js.map +1 -0
  128. package/dist/server/viewsDevServer.d.ts +14 -0
  129. package/dist/server/viewsDevServer.js +45 -0
  130. package/dist/server/viewsDevServer.js.map +1 -0
  131. package/dist/test/utils.d.ts +13 -21
  132. package/dist/test/utils.js +42 -37
  133. package/dist/test/utils.js.map +1 -1
  134. package/dist/test/view.test.d.ts +1 -0
  135. package/dist/test/view.test.js +568 -0
  136. package/dist/test/view.test.js.map +1 -0
  137. package/dist/version.d.ts +1 -0
  138. package/dist/version.js +3 -0
  139. package/dist/version.js.map +1 -0
  140. package/dist/web/bridges/apps-sdk/adaptor.d.ts +14 -7
  141. package/dist/web/bridges/apps-sdk/adaptor.js +66 -29
  142. package/dist/web/bridges/apps-sdk/adaptor.js.map +1 -1
  143. package/dist/web/bridges/apps-sdk/bridge.d.ts +2 -1
  144. package/dist/web/bridges/apps-sdk/bridge.js +1 -0
  145. package/dist/web/bridges/apps-sdk/bridge.js.map +1 -1
  146. package/dist/web/bridges/apps-sdk/index.d.ts +1 -1
  147. package/dist/web/bridges/apps-sdk/index.js.map +1 -1
  148. package/dist/web/bridges/apps-sdk/types.d.ts +26 -11
  149. package/dist/web/bridges/apps-sdk/types.js.map +1 -1
  150. package/dist/web/bridges/apps-sdk/use-apps-sdk-context.d.ts +11 -0
  151. package/dist/web/bridges/apps-sdk/use-apps-sdk-context.js +11 -0
  152. package/dist/web/bridges/apps-sdk/use-apps-sdk-context.js.map +1 -1
  153. package/dist/web/bridges/get-adaptor.d.ts +7 -0
  154. package/dist/web/bridges/get-adaptor.js +7 -0
  155. package/dist/web/bridges/get-adaptor.js.map +1 -1
  156. package/dist/web/bridges/index.js.map +1 -1
  157. package/dist/web/bridges/mcp-app/adaptor.d.ts +26 -9
  158. package/dist/web/bridges/mcp-app/adaptor.js +159 -66
  159. package/dist/web/bridges/mcp-app/adaptor.js.map +1 -1
  160. package/dist/web/bridges/mcp-app/bridge.d.ts +16 -31
  161. package/dist/web/bridges/mcp-app/bridge.js +65 -201
  162. package/dist/web/bridges/mcp-app/bridge.js.map +1 -1
  163. package/dist/web/bridges/mcp-app/index.js.map +1 -1
  164. package/dist/web/bridges/mcp-app/types.js.map +1 -1
  165. package/dist/web/bridges/mcp-app/use-mcp-app-context.d.ts +17 -3
  166. package/dist/web/bridges/mcp-app/use-mcp-app-context.js +14 -2
  167. package/dist/web/bridges/mcp-app/use-mcp-app-context.js.map +1 -1
  168. package/dist/web/bridges/mcp-app/use-mcp-app-context.test.js +1 -41
  169. package/dist/web/bridges/mcp-app/use-mcp-app-context.test.js.map +1 -1
  170. package/dist/web/bridges/mcp-app/view-tools.test.d.ts +1 -0
  171. package/dist/web/bridges/mcp-app/view-tools.test.js +144 -0
  172. package/dist/web/bridges/mcp-app/view-tools.test.js.map +1 -0
  173. package/dist/web/bridges/types.d.ts +120 -14
  174. package/dist/web/bridges/types.js.map +1 -1
  175. package/dist/web/bridges/use-host-context.d.ts +5 -0
  176. package/dist/web/bridges/use-host-context.js +5 -0
  177. package/dist/web/bridges/use-host-context.js.map +1 -1
  178. package/dist/web/components/modal-provider.js +3 -5
  179. package/dist/web/components/modal-provider.js.map +1 -1
  180. package/dist/web/create-store.d.ts +26 -0
  181. package/dist/web/create-store.js +43 -3
  182. package/dist/web/create-store.js.map +1 -1
  183. package/dist/web/create-store.test.js +21 -18
  184. package/dist/web/create-store.test.js.map +1 -1
  185. package/dist/web/data-llm.d.ts +34 -1
  186. package/dist/web/data-llm.js +31 -3
  187. package/dist/web/data-llm.js.map +1 -1
  188. package/dist/web/data-llm.test.js +24 -23
  189. package/dist/web/data-llm.test.js.map +1 -1
  190. package/dist/web/generate-helpers.d.ts +22 -18
  191. package/dist/web/generate-helpers.js +22 -18
  192. package/dist/web/generate-helpers.js.map +1 -1
  193. package/dist/web/generate-helpers.test-d.js +26 -26
  194. package/dist/web/generate-helpers.test-d.js.map +1 -1
  195. package/dist/web/generate-helpers.test.js.map +1 -1
  196. package/dist/web/helpers/state.d.ts +2 -2
  197. package/dist/web/helpers/state.js +11 -11
  198. package/dist/web/helpers/state.js.map +1 -1
  199. package/dist/web/helpers/state.test.js +9 -9
  200. package/dist/web/helpers/state.test.js.map +1 -1
  201. package/dist/web/hooks/index.d.ts +6 -2
  202. package/dist/web/hooks/index.js +5 -1
  203. package/dist/web/hooks/index.js.map +1 -1
  204. package/dist/web/hooks/test/utils.d.ts +6 -2
  205. package/dist/web/hooks/test/utils.js +17 -2
  206. package/dist/web/hooks/test/utils.js.map +1 -1
  207. package/dist/web/hooks/use-call-tool.d.ts +45 -0
  208. package/dist/web/hooks/use-call-tool.js +28 -0
  209. package/dist/web/hooks/use-call-tool.js.map +1 -1
  210. package/dist/web/hooks/use-call-tool.test-d.js.map +1 -1
  211. package/dist/web/hooks/use-call-tool.test.js +27 -6
  212. package/dist/web/hooks/use-call-tool.test.js.map +1 -1
  213. package/dist/web/hooks/use-display-mode.d.ts +23 -3
  214. package/dist/web/hooks/use-display-mode.js +20 -0
  215. package/dist/web/hooks/use-display-mode.js.map +1 -1
  216. package/dist/web/hooks/use-display-mode.test-d.d.ts +1 -0
  217. package/dist/web/hooks/use-display-mode.test-d.js +8 -0
  218. package/dist/web/hooks/use-display-mode.test-d.js.map +1 -0
  219. package/dist/web/hooks/use-display-mode.test.js.map +1 -1
  220. package/dist/web/hooks/use-download.d.ts +5 -0
  221. package/dist/web/hooks/use-download.js +8 -0
  222. package/dist/web/hooks/use-download.js.map +1 -0
  223. package/dist/web/hooks/use-download.test.d.ts +1 -0
  224. package/dist/web/hooks/use-download.test.js +95 -0
  225. package/dist/web/hooks/use-download.test.js.map +1 -0
  226. package/dist/web/hooks/use-files.d.ts +34 -1
  227. package/dist/web/hooks/use-files.js +33 -0
  228. package/dist/web/hooks/use-files.js.map +1 -1
  229. package/dist/web/hooks/use-files.test.js +22 -2
  230. package/dist/web/hooks/use-files.test.js.map +1 -1
  231. package/dist/web/hooks/use-layout.d.ts +2 -0
  232. package/dist/web/hooks/use-layout.js +2 -0
  233. package/dist/web/hooks/use-layout.js.map +1 -1
  234. package/dist/web/hooks/use-layout.test.js +3 -3
  235. package/dist/web/hooks/use-layout.test.js.map +1 -1
  236. package/dist/web/hooks/use-open-external.d.ts +20 -1
  237. package/dist/web/hooks/use-open-external.js +17 -1
  238. package/dist/web/hooks/use-open-external.js.map +1 -1
  239. package/dist/web/hooks/use-open-external.test.js +26 -11
  240. package/dist/web/hooks/use-open-external.test.js.map +1 -1
  241. package/dist/web/hooks/use-register-view-tool.d.ts +38 -0
  242. package/dist/web/hooks/use-register-view-tool.js +50 -0
  243. package/dist/web/hooks/use-register-view-tool.js.map +1 -0
  244. package/dist/web/hooks/use-request-close.d.ts +16 -0
  245. package/dist/web/hooks/use-request-close.js +21 -0
  246. package/dist/web/hooks/use-request-close.js.map +1 -0
  247. package/dist/web/hooks/use-request-close.test.d.ts +1 -0
  248. package/dist/web/hooks/use-request-close.test.js +52 -0
  249. package/dist/web/hooks/use-request-close.test.js.map +1 -0
  250. package/dist/web/hooks/use-request-modal.d.ts +16 -1
  251. package/dist/web/hooks/use-request-modal.js +19 -4
  252. package/dist/web/hooks/use-request-modal.js.map +1 -1
  253. package/dist/web/hooks/use-request-modal.test.js +5 -1
  254. package/dist/web/hooks/use-request-modal.test.js.map +1 -1
  255. package/dist/web/hooks/use-request-size.d.ts +20 -0
  256. package/dist/web/hooks/use-request-size.js +24 -0
  257. package/dist/web/hooks/use-request-size.js.map +1 -0
  258. package/dist/web/hooks/use-request-size.test.d.ts +1 -0
  259. package/dist/web/hooks/use-request-size.test.js +65 -0
  260. package/dist/web/hooks/use-request-size.test.js.map +1 -0
  261. package/dist/web/hooks/use-send-follow-up-message.d.ts +19 -1
  262. package/dist/web/hooks/use-send-follow-up-message.js +19 -2
  263. package/dist/web/hooks/use-send-follow-up-message.js.map +1 -1
  264. package/dist/web/hooks/use-set-open-in-app-url.d.ts +17 -0
  265. package/dist/web/hooks/use-set-open-in-app-url.js +17 -0
  266. package/dist/web/hooks/use-set-open-in-app-url.js.map +1 -1
  267. package/dist/web/hooks/use-set-open-in-app-url.test.js +5 -11
  268. package/dist/web/hooks/use-set-open-in-app-url.test.js.map +1 -1
  269. package/dist/web/hooks/use-tool-info.d.ts +33 -0
  270. package/dist/web/hooks/use-tool-info.js +26 -0
  271. package/dist/web/hooks/use-tool-info.js.map +1 -1
  272. package/dist/web/hooks/use-tool-info.test-d.js.map +1 -1
  273. package/dist/web/hooks/use-tool-info.test.js +1 -1
  274. package/dist/web/hooks/use-tool-info.test.js.map +1 -1
  275. package/dist/web/hooks/use-user.d.ts +2 -0
  276. package/dist/web/hooks/use-user.js +20 -2
  277. package/dist/web/hooks/use-user.js.map +1 -1
  278. package/dist/web/hooks/use-user.test.js +29 -1
  279. package/dist/web/hooks/use-user.test.js.map +1 -1
  280. package/dist/web/hooks/use-view-state.d.ts +25 -0
  281. package/dist/web/hooks/use-view-state.js +32 -0
  282. package/dist/web/hooks/use-view-state.js.map +1 -0
  283. package/dist/web/hooks/use-view-state.test.d.ts +1 -0
  284. package/dist/web/hooks/use-view-state.test.js +177 -0
  285. package/dist/web/hooks/use-view-state.test.js.map +1 -0
  286. package/dist/web/index.d.ts +1 -2
  287. package/dist/web/index.js +1 -2
  288. package/dist/web/index.js.map +1 -1
  289. package/dist/web/mount-view.d.ts +20 -0
  290. package/dist/web/{mount-widget.js → mount-view.js} +21 -2
  291. package/dist/web/mount-view.js.map +1 -0
  292. package/dist/web/plugin/data-llm.test.js.map +1 -1
  293. package/dist/web/plugin/plugin.d.ts +32 -1
  294. package/dist/web/plugin/plugin.js +161 -18
  295. package/dist/web/plugin/plugin.js.map +1 -1
  296. package/dist/web/plugin/scan-views.d.ts +16 -0
  297. package/dist/web/plugin/scan-views.js +88 -0
  298. package/dist/web/plugin/scan-views.js.map +1 -0
  299. package/dist/web/plugin/scan-views.test.d.ts +1 -0
  300. package/dist/web/plugin/scan-views.test.js +99 -0
  301. package/dist/web/plugin/scan-views.test.js.map +1 -0
  302. package/dist/web/plugin/transform-data-llm.js +1 -1
  303. package/dist/web/plugin/transform-data-llm.js.map +1 -1
  304. package/dist/web/plugin/transform-data-llm.test.js.map +1 -1
  305. package/dist/web/plugin/validate-view.d.ts +1 -0
  306. package/dist/web/plugin/validate-view.js +9 -0
  307. package/dist/web/plugin/validate-view.js.map +1 -0
  308. package/dist/web/plugin/validate-view.test.d.ts +1 -0
  309. package/dist/web/plugin/validate-view.test.js +24 -0
  310. package/dist/web/plugin/validate-view.test.js.map +1 -0
  311. package/dist/web/proxy.js.map +1 -1
  312. package/dist/web/types.d.ts +4 -0
  313. package/dist/web/types.js.map +1 -1
  314. package/package.json +39 -23
  315. package/tsconfig.base.json +5 -0
  316. package/dist/server/templates/development.hbs +0 -67
  317. package/dist/server/templates/production.hbs +0 -6
  318. package/dist/server/widgetsDevServer.d.ts +0 -12
  319. package/dist/server/widgetsDevServer.js +0 -57
  320. package/dist/server/widgetsDevServer.js.map +0 -1
  321. package/dist/test/widget.test.js +0 -261
  322. package/dist/test/widget.test.js.map +0 -1
  323. package/dist/web/hooks/use-widget-state.d.ts +0 -4
  324. package/dist/web/hooks/use-widget-state.js +0 -32
  325. package/dist/web/hooks/use-widget-state.js.map +0 -1
  326. package/dist/web/hooks/use-widget-state.test.js +0 -62
  327. package/dist/web/hooks/use-widget-state.test.js.map +0 -1
  328. package/dist/web/mount-widget.d.ts +0 -1
  329. package/dist/web/mount-widget.js.map +0 -1
  330. /package/dist/{test/widget.test.d.ts → cli/build-helpers.test.d.ts} +0 -0
  331. /package/dist/{web/hooks/use-widget-state.test.d.ts → cli/tunnel-control-server.test.d.ts} +0 -0
package/README.md CHANGED
@@ -1,145 +1,152 @@
1
- <div align="center">
2
-
3
- <img alt="Skybridge" src="https://raw.githubusercontent.com/alpic-ai/skybridge/main/docs/images/github-banner.png" width="100%">
4
-
5
- <br />
6
-
7
- # Skybridge
8
-
9
- **Build ChatGPT & MCP Apps. The Modern TypeScript Way.**
10
-
11
- The fullstack TypeScript framework for AI-embedded widgets.<br />
12
- **Type-safe. React-powered. Platform-agnostic.**
13
-
14
- <br />
15
-
16
- [![NPM Version](https://img.shields.io/npm/v/skybridge?color=e90060&style=for-the-badge)](https://www.npmjs.com/package/skybridge)
17
- [![NPM Downloads](https://img.shields.io/npm/dm/skybridge?color=e90060&style=for-the-badge)](https://www.npmjs.com/package/skybridge)
18
- [![GitHub License](https://img.shields.io/github/license/alpic-ai/skybridge?color=e90060&style=for-the-badge)](https://github.com/alpic-ai/skybridge/blob/main/LICENSE)
19
-
20
- <br />
21
-
22
- [Documentation](https://docs.skybridge.tech) · [Quick Start](https://docs.skybridge.tech/quickstart/create-new-app) · [Showcase](https://docs.skybridge.tech/showcase)
23
-
24
- </div>
25
-
26
- <br />
27
-
28
- ## ✨ Why Skybridge?
29
-
30
- ChatGPT Apps and MCP Apps let you embed **rich, interactive UIs** directly in AI conversations. But the raw SDKs are low-level—no hooks, no type safety, no dev tools, and no HMR.
31
-
32
- **Skybridge fixes that.**
33
-
34
- | | |
35
- |:--|:--|
36
- | 👨‍💻 **Full Dev Environment** — HMR, debug traces, and local devtools. No more refresh loops. | ✅ **End-to-End Type Safety** — tRPC-style inference from server to widget. Autocomplete everywhere. |
37
- | 🔄 **Widget-to-Model Sync** — Keep the model aware of UI state with `data-llm`. Dual surfaces, one source of truth. | ⚒️ **React Query-style Hooks** — `isPending`, `isError`, callbacks. State management you already know. |
38
- | 🌐 **Platform Agnostic** — Write once, run anywhere. Works with ChatGPT (Apps SDK) and MCP-compatible clients. | 📦 **Showcase Examples** — Production-ready examples to learn from and build upon. |
39
-
40
- <br />
41
-
42
- ## 🚀 Get Started
43
-
44
- **Create a new app:**
45
-
1
+ # Skybridge - the MCP Apps framework
2
+
3
+ <p align="center">
4
+ <a href="https://docs.skybridge.tech">
5
+ <picture>
6
+ <source media="(prefers-color-scheme: dark)" srcset="https://raw.githubusercontent.com/alpic-ai/skybridge/main/docs/images/skybridge-readme-banner-dark.png" />
7
+ <img alt="Skybridge, the full-stack React framework for MCP apps and MCP servers" src="https://raw.githubusercontent.com/alpic-ai/skybridge/main/docs/images/skybridge-readme-banner-light.png" width="100%" />
8
+ </picture>
9
+ </a>
10
+ </p>
11
+
12
+ <p align="center">
13
+ <strong>The full-stack React framework for MCP Apps and MCP Servers.</strong>
14
+ </p>
15
+
16
+ <p align="center">
17
+ <a href="https://docs.skybridge.tech">Documentation</a> ·
18
+ <a href="https://docs.skybridge.tech/quickstart/create-new-app">Quickstart</a> ·
19
+ <a href="https://github.com/alpic-ai/skybridge/tree/main/examples">Examples</a>
20
+ </p>
21
+
22
+ <p align="center">
23
+ <a href="https://www.npmjs.com/package/skybridge"><picture><source media="(prefers-color-scheme: dark)" srcset="https://img.shields.io/npm/v/skybridge?color=77F5EE&amp;labelColor=161B22&amp;style=for-the-badge"><img alt="npm version" src="https://img.shields.io/npm/v/skybridge?color=E3FAF7&amp;labelColor=F6F8FA&amp;style=for-the-badge"></picture></a>
24
+ <a href="https://www.npmjs.com/package/skybridge"><picture><source media="(prefers-color-scheme: dark)" srcset="https://img.shields.io/npm/dm/skybridge?color=D7FFC8&amp;labelColor=161B22&amp;style=for-the-badge"><img alt="npm downloads" src="https://img.shields.io/npm/dm/skybridge?color=E8FBD9&amp;labelColor=F6F8FA&amp;style=for-the-badge"></picture></a>
25
+ <a href="https://discord.com/invite/gNAazGueab"><picture><source media="(prefers-color-scheme: dark)" srcset="https://img.shields.io/badge/Discord-community-77F5EE?style=for-the-badge&amp;logo=discord&amp;logoColor=77F5EE&amp;labelColor=161B22"><img alt="Discord community" src="https://img.shields.io/badge/Discord-community-E3FAF7?style=for-the-badge&amp;logo=discord&amp;logoColor=5865F2&amp;labelColor=F6F8FA"></picture></a>
26
+ <a href="https://github.com/alpic-ai/skybridge/blob/main/LICENSE"><picture><source media="(prefers-color-scheme: dark)" srcset="https://img.shields.io/github/license/alpic-ai/skybridge?color=D7FFC8&amp;labelColor=161B22&amp;style=for-the-badge"><img alt="License: MIT" src="https://img.shields.io/github/license/alpic-ai/skybridge?color=E8FBD9&amp;labelColor=F6F8FA&amp;style=for-the-badge"></picture></a>
27
+ </p>
28
+
29
+ ## About Skybridge
30
+
31
+ Skybridge helps developers build type-safe MCP apps for Claude, ChatGPT and other UI-enabled MCP clients, with a complete set of tooling designed for both humans and agents.
32
+
33
+ Why? MCP apps extend the [Model Context Protocol](https://modelcontextprotocol.io/docs/getting-started/intro) with **rich, interactive UI views** rendered from MCP servers. Conversational apps need seamless interaction between the user, the UI, and the model. This means new UX patterns, developer tooling, and abstractions.
34
+ Plus, the raw SDKs are low-level: no hooks, type safety, HMR, etc.
35
+
36
+ That's why we built *Skybridge*.
37
+
38
+ Features include:
39
+
40
+ - **Delightful dev environment**: Skybridge provides a dev server with a local emulator, hot module reload, and a permanent tunnel to connect your local app to Claude and ChatGPT.
41
+ - **Write once, run everywhere**: the framework abstracts implementation differences between MCP clients, so your app runs seamlessly in Claude, ChatGPT, VSCode, and any other MCP apps compatible client.
42
+ - **Agent-ready**: powerful skills, CLI, and programmatic dev tool APIs, everything your coding agent needs to build MCP apps end-to-end.
43
+ - **Type-safe end-to-end**: tRPC-style inference from MCP server tool definition to React view for type safety from server to frontend.
44
+ - **React-first**: Intuitive React Query-style hooks, with advanced state management.
45
+ - **Example library**: get started quickly with ChatGPT- and Claude-ready app examples for ecommerce, travel, SaaS, and more.
46
+
47
+ They chose to build their MCP apps with Skybridge:
48
+
49
+ <p align="center">
50
+ <a href="https://www.datadoghq.com"><picture><source media="(prefers-color-scheme: dark)" srcset="https://raw.githubusercontent.com/alpic-ai/skybridge/main/docs/images/user-logos/datadog-dark.svg"><img src="https://raw.githubusercontent.com/alpic-ai/skybridge/main/docs/images/user-logos/datadog-light.svg" alt="Datadog" height="24"></picture></a>
51
+ &nbsp;&nbsp;
52
+ <a href="https://bitmovin.com"><picture><source media="(prefers-color-scheme: dark)" srcset="https://raw.githubusercontent.com/alpic-ai/skybridge/main/docs/images/user-logos/bitmovin-dark.svg"><img src="https://raw.githubusercontent.com/alpic-ai/skybridge/main/docs/images/user-logos/bitmovin-light.svg" alt="Bitmovin" height="22"></picture></a>
53
+ &nbsp;&nbsp;
54
+ <a href="https://www.evaneos.com"><picture><source media="(prefers-color-scheme: dark)" srcset="https://raw.githubusercontent.com/alpic-ai/skybridge/main/docs/images/user-logos/evaneos-dark.svg"><img src="https://raw.githubusercontent.com/alpic-ai/skybridge/main/docs/images/user-logos/evaneos-light.svg" alt="Evaneos" height="18"></picture></a>
55
+ &nbsp;&nbsp;
56
+ <a href="https://www.touchstream.media"><picture><source media="(prefers-color-scheme: dark)" srcset="https://raw.githubusercontent.com/alpic-ai/skybridge/main/docs/images/user-logos/touchstream-dark.svg"><img src="https://raw.githubusercontent.com/alpic-ai/skybridge/main/docs/images/user-logos/touchstream-light.svg" alt="Touchstream" height="24"></picture></a>
57
+ &nbsp;&nbsp;
58
+ <a href="https://www.cottages.com"><picture><source media="(prefers-color-scheme: dark)" srcset="https://raw.githubusercontent.com/alpic-ai/skybridge/main/docs/images/user-logos/cottages-dark.svg"><img src="https://raw.githubusercontent.com/alpic-ai/skybridge/main/docs/images/user-logos/cottages-light.svg" alt="Cottages.com" height="24"></picture></a>
59
+ </p>
60
+
61
+ ## Get started
62
+
63
+ **For agents**
64
+
65
+ Install our [skill](https://docs.skybridge.tech/devtools/skills) for building MCP apps and ChatGPT apps:
46
66
  ```bash
47
- npm create skybridge@latest
67
+ npx skills add alpic-ai/skybridge -s skybridge
48
68
  ```
69
+ Once installed, ask your agent "What skills do you have?" to confirm, then try:
70
+
71
+ - _Create a new MCP app_
72
+ - _Migrate my MCP server to the Skybridge framework_
73
+ - _Add a new view to my MCP app_
49
74
 
50
- **Or add to an existing project:**
75
+ **For humans**
51
76
 
77
+ Bootstrap a new project with:
52
78
  ```bash
53
- npm i skybridge
54
- yarn add skybridge
55
- pnpm add skybridge
56
- bun add skybridge
57
- deno add skybridge
79
+ npm create skybridge@latest my-app
58
80
  ```
81
+ For full install instructions, read our [**Quickstart guide**](https://docs.skybridge.tech/quickstart/create-new-app).
59
82
 
60
- <div align="center">
61
-
62
- **👉 [Read the Docs](https://docs.skybridge.tech) 👈**
83
+ ## Documentation
63
84
 
64
- </div>
85
+ The [Skybridge documentation](https://docs.skybridge.tech) covers the full lifecycle of building MCP Apps:
65
86
 
66
- <br />
87
+ - [Fundamentals](https://docs.skybridge.tech/fundamentals): understand MCP Apps, ChatGPT Apps, and how Skybridge bridges both runtimes.
88
+ - [Core concepts](https://docs.skybridge.tech/concepts): learn about server <> model <> UI data flows, LLM context sync, type safety, and instant local iteration with our devtools.
89
+ - [Guides](https://docs.skybridge.tech/guides/fetching-data): build real app behavior with tools, views, state, and model communication.
90
+ - [API Reference](https://docs.skybridge.tech/api-reference): browse our MCP server APIs, React hooks, CLI commands, and runtime compatibility.
67
91
 
68
- ## 📦 Architecture
92
+ ## Deploy
69
93
 
70
- Skybridge is a fullstack framework with unified server and client modules:
94
+ Deploy Skybridge apps instantly on [Alpic](https://alpic.ai) for scalable hosting, MCP-specific analytics, permanent tunneling, app store compliance auditing and submission help. You can also self-host on any Node.js-compatible platform.
71
95
 
72
- - **`skybridge/server`** Define tools and widgets with full type inference. Extends the MCP SDK.
73
- - **`skybridge/web`** — React hooks that consume your server types. Works with Apps SDK (ChatGPT) and MCP Apps.
74
- - **Dev Environment** — Vite plugin with HMR, DevTools emulator, and optimized builds.
96
+ See our [deployment guide](https://docs.skybridge.tech/quickstart/deploy) for the full production path.
75
97
 
76
- ### Server
98
+ ## Community & Contributing
77
99
 
78
- ```ts
79
- import { McpServer } from "skybridge/server";
80
-
81
- server.registerWidget("flights", {}, {
82
- inputSchema: { destination: z.string() },
83
- }, async ({ destination }) => {
84
- const flights = await searchFlights(destination);
85
- return { structuredContent: { flights } };
86
- });
87
- ```
88
-
89
- ### Widget
90
-
91
- ```tsx
92
- import { useToolInfo } from "skybridge/web";
93
-
94
- function FlightsWidget() {
95
- const { output } = useToolInfo();
96
-
97
- return output.structuredContent.flights.map(flight =>
98
- <FlightCard key={flight.id} flight={flight} />
99
- );
100
- }
101
- ```
100
+ We'd love your help improving Skybridge. Here are a few ways to get involved:
102
101
 
103
- <br />
102
+ - **Bugs**: If you run into a bug or unexpected behavior, open a [GitHub Issue](https://github.com/alpic-ai/skybridge/issues) with a clear reproduction.
103
+ - **Questions and ideas**: Need help building with Skybridge or have ideas to improve the framework, docs, examples, or developer experience? [Open an issue](https://github.com/alpic-ai/skybridge/issues) or share them on our [Discord](https://discord.com/invite/gNAazGueab).
104
+ - **Pull requests**: For code or documentation changes, read the [Contributing Guide](https://github.com/alpic-ai/skybridge/blob/main/CONTRIBUTING.md) before opening a PR.
104
105
 
105
- ## 🎯 Features at a Glance
106
+ Skybridge is released under the [MIT License](https://github.com/alpic-ai/skybridge/blob/main/LICENSE).
106
107
 
107
- - **Live Reload** — Vite HMR. See changes instantly without reinstalling.
108
- - **Typed Hooks** — Full autocomplete for tools, inputs, outputs.
109
- - **Widget → Tool Calls** — Trigger server actions from UI.
110
- - **Dual Surface Sync** — Keep model aware of what users see with `data-llm`.
111
- - **React Query-style API** — `isPending`, `isError`, callbacks.
112
- - **Platform Agnostic** — Works with ChatGPT (Apps SDK) and MCP Apps clients (Goose, VSCode, etc.).
113
- - **MCP Compatible** — Extends the official SDK. Works with any MCP client.
108
+ ### Contributors
114
109
 
115
- <br />
110
+ Built and maintained with ❤️ by [Harijoe](https://github.com/harijoe), [Fred Barthelet](https://github.com/fredericbarthelet), and the [Alpic](https://alpic.ai) team.
116
111
 
117
- ## 📖 Showcase
112
+ <a href="https://github.com/alpic-ai/skybridge/graphs/contributors">
113
+ <img src="https://contrib.rocks/image?repo=alpic-ai/skybridge" alt="Skybridge contributors">
114
+ </a>
118
115
 
119
- Explore production-ready examples:
116
+ ## Example templates
120
117
 
121
- | Example | Description | Demo | Code |
122
- |------------------------|----------------------------------------------------------------------------------|-----------------------------------------------------|-------------------------------------------------------------------------------------|
123
- | **Capitals Explorer** | Interactive world map with geolocation and Wikipedia integration | [Try Demo](https://capitals.skybridge.tech/try) | [View Code](https://github.com/alpic-ai/skybridge/tree/main/examples/capitals) |
124
- | **Ecommerce Carousel** | Product carousel with cart, localization, and modals | [Try Demo](https://ecommerce.skybridge.tech/try) | [View Code](https://github.com/alpic-ai/skybridge/tree/main/examples/ecom-carousel) |
125
- | **Everything** | Comprehensive playground showcasing all hooks and features | [Try Demo](https://everything.skybridge.tech/try) | [View Code](https://github.com/alpic-ai/skybridge/tree/main/examples/everything) |
126
- | **Productivity** | Data visualization dashboard demonstrating Skybridge capabilities for MCP Apps | [Try Demo](https://productivity.skybridge.tech/try) | [View Code](https://github.com/alpic-ai/skybridge/tree/main/examples/productivity) |
127
- | **Manifest Starter** | Starter app with Manifest UI agentic components out-of-the-box | [Try Demo](https://manifest-ui.skybridge.tech/try) | [View Code](https://github.com/alpic-ai/skybridge/tree/main/examples/manifest-ui) |
118
+ Explore all our example templates in the [Examples](https://docs.skybridge.tech/examples) section of the documentation.
128
119
 
129
- See all examples in the [Showcase](https://docs.skybridge.tech/showcase) or browse the [examples/](examples/) directory.
120
+ ### Basic
130
121
 
131
- <br />
122
+ | Preview | App | Description | Demo | Code |
123
+ | --- | --- | --- | --- | --- |
124
+ | <img src="docs/images/showcase-example.png" alt="Everything" width="160" /> | Everything | Comprehensive playground app showcasing all Skybridge hooks and features. | [Try Demo](https://everything.skybridge.tech/try) | [View code](https://github.com/alpic-ai/skybridge/tree/main/examples/everything) |
132
125
 
133
- <div align="center">
126
+ ### Use cases
134
127
 
135
- [![GitHub Discussions](https://img.shields.io/badge/Discussions-Ask%20Questions-blue?style=flat-square&logo=github)](https://github.com/alpic-ai/skybridge/discussions)
136
- [![GitHub Issues](https://img.shields.io/badge/Issues-Report%20Bugs-red?style=flat-square&logo=github)](https://github.com/alpic-ai/skybridge/issues)
137
- [![Discord](https://img.shields.io/badge/Discord-Chat-5865F2?style=flat-square&logo=discord&logoColor=white)](https://discord.com/invite/gNAazGueab)
128
+ | Preview | App | Description | Demo | Code |
129
+ | --- | --- | --- | --- | --- |
130
+ | <img src="docs/images/showcase-capitals.png" alt="Capitals Explorer" width="160" /> | Capitals Explorer | Interactive world map with geolocation, country information, and dynamic capital exploration. | [Try Demo](https://capitals.skybridge.tech/try) | [View code](https://github.com/alpic-ai/skybridge/tree/main/examples/capitals) |
131
+ | <img src="docs/images/showcase-flight-booking.png" alt="Flight Booking" width="160" /> | Flight Booking | Flight search carousel with route details, pricing comparison, and external booking. | [Try Demo](https://flight-booking.skybridge.tech/try) | [View code](https://github.com/alpic-ai/skybridge/tree/main/examples/flight-booking) |
132
+ | <img src="docs/images/showcase-ecommerce.png" alt="Ecommerce Carousel" width="160" /> | Ecommerce Carousel | Product carousel with persistent cart, localization, theme switching, and modal dialogs. | [Try Demo](https://ecommerce.skybridge.tech/try) | [View code](https://github.com/alpic-ai/skybridge/tree/main/examples/ecom-carousel) |
133
+ | <img src="docs/images/showcase-investigation-game.png" alt="Investigation Game" width="160" /> | Investigation Game | Multi-screen mystery game with fullscreen mode, dynamic story progression and context asynchronicity demonstration | [Try Demo](https://investigation-game.skybridge.tech/try) | [View code](https://github.com/alpic-ai/skybridge/tree/main/examples/investigation-game) |
134
+ | <img src="docs/images/showcase-productivity.png" alt="Productivity" width="160" /> | Productivity | Interactive analytics dashboard with charts, theme adaptation, localization, fullscreen mode, and bidirectional tool calls. | [Try Demo](https://productivity.skybridge.tech/try) | [View code](https://github.com/alpic-ai/skybridge/tree/main/examples/productivity) |
135
+ | <img src="docs/images/showcase-times-up.png" alt="Time's Up" width="160" /> | Time's Up | Word-guessing party game where the user gives hints and the AI tries to guess. | [Try Demo](https://times-up.skybridge.tech/try) | [View code](https://github.com/alpic-ai/skybridge/tree/main/examples/times-up) |
136
+ | <img src="docs/images/showcase-lumo.png" alt="Lumo Interactive AI Tutor" width="160" /> | Lumo — Interactive AI Tutor | Adaptive tutor with Mermaid diagrams, mind maps, quizzes, and fill-in-the-blank exercises. | [Try Demo](https://lumo-mcp-app-39519fdd.alpic.live/try) | [View code](https://github.com/connorads/lumo-mcp-app) |
138
137
 
139
- See [CONTRIBUTING.md](CONTRIBUTING.md) for setup instructions
138
+ ### Auth
140
139
 
141
- <br />
140
+ | Preview | Provider | Description | Code |
141
+ | --- | --- | --- | --- |
142
+ | <img src="docs/images/showcase-clerk.png" alt="Auth Clerk" width="160" /> | Clerk | Full OAuth authentication with Clerk and personalized coffee shop search. | [View code](https://github.com/alpic-ai/skybridge/tree/main/examples/auth-clerk) |
143
+ | <img src="docs/images/showcase-workos.png" alt="Auth WorkOS AuthKit" width="160" /> | WorkOS AuthKit | Full OAuth authentication with WorkOS AuthKit and personalized coffee shop search. | [View code](https://github.com/alpic-ai/skybridge/tree/main/examples/auth-workos) |
144
+ | <img src="docs/images/showcase-stytch.png" alt="Auth Stytch" width="160" /> | Stytch | Full OAuth authentication with Stytch and personalized coffee shop search. | [View code](https://github.com/alpic-ai/skybridge/tree/main/examples/auth-stytch) |
145
+ | <img src="docs/images/showcase-auth0.png" alt="Auth Auth0" width="160" /> | Auth0 | Full OAuth authentication with Auth0 and personalized coffee shop search. | [View code](https://github.com/alpic-ai/skybridge/tree/main/examples/auth-auth0) |
142
146
 
143
- **[MIT License](LICENSE)** · Made with ❤️ by **[Alpic](https://alpic.ai)**
147
+ ### UI and component libraries
144
148
 
145
- </div>
149
+ | Preview | App | Description | Demo | Code |
150
+ | --- | --- | --- | --- | --- |
151
+ | <img src="docs/images/showcase-manifest-ui.png" alt="Manifest UI" width="160" /> | Manifest UI | Agentic component library example for rich AI-powered experiences. | [Try Demo](https://manifest-ui.skybridge.tech/try) | [View code](https://github.com/alpic-ai/skybridge/tree/main/examples/manifest-ui) |
152
+ | <img src="docs/images/showcase-generative-ui.png" alt="Generative UI" width="160" /> | Generative UI | LLM-generated dynamic UIs with json-render and 36 pre-built shadcn/ui components. | [Try Demo](https://generative-ui.skybridge.tech/try) | [View code](https://github.com/alpic-ai/skybridge/tree/main/examples/generative-ui) |
@@ -0,0 +1,7 @@
1
+ export declare const ENTRY_WRAPPER_CONTENT = "import { __setBuildManifest } from \"skybridge/server\";\nimport manifest from \"./vite-manifest.js\";\n\n__setBuildManifest(manifest);\n\nconst userMod = await import(\"./server.js\");\nexport default userMod.default;\n";
2
+ export declare function emitEntryWrapper(distDir: string): void;
3
+ export declare function emitManifestModule(manifestPath: string, outPath: string): void;
4
+ export declare const VERCEL_FUNCTION_NAME = "mcp";
5
+ export declare const VERCEL_CONFIG: unknown;
6
+ export declare const VERCEL_VC_CONFIG: unknown;
7
+ export declare function emitVercelBuildOutput(root: string): Promise<void>;
@@ -0,0 +1,82 @@
1
+ import { cpSync, mkdirSync, readFileSync, rmSync, writeFileSync, } from "node:fs";
2
+ import path from "node:path";
3
+ // Primes the manifest in skybridge's module scope, then dynamically imports
4
+ // `./server.js` so user code runs *after* the side channel is set. The
5
+ // dynamic import is load-bearing: a static `export { default } from ...` is
6
+ // hoisted with the rest of the static graph and would evaluate `server.js`
7
+ // before `__setBuildManifest` runs.
8
+ export const ENTRY_WRAPPER_CONTENT = `import { __setBuildManifest } from "skybridge/server";
9
+ import manifest from "./vite-manifest.js";
10
+
11
+ __setBuildManifest(manifest);
12
+
13
+ const userMod = await import("./server.js");
14
+ export default userMod.default;
15
+ `;
16
+ export function emitEntryWrapper(distDir) {
17
+ writeFileSync(path.join(distDir, "__entry.js"), ENTRY_WRAPPER_CONTENT);
18
+ }
19
+ export function emitManifestModule(manifestPath, outPath) {
20
+ const manifest = readFileSync(manifestPath, "utf-8");
21
+ writeFileSync(outPath, `export default ${manifest};\n`);
22
+ }
23
+ export const VERCEL_FUNCTION_NAME = "mcp";
24
+ export const VERCEL_CONFIG = {
25
+ version: 3,
26
+ routes: [
27
+ {
28
+ src: "/assets/(.*)",
29
+ headers: { "Access-Control-Allow-Origin": "*" },
30
+ continue: true,
31
+ },
32
+ { handle: "filesystem" },
33
+ { src: "/(.*)", dest: `/${VERCEL_FUNCTION_NAME}` },
34
+ ],
35
+ };
36
+ export const VERCEL_VC_CONFIG = {
37
+ runtime: "nodejs22.x",
38
+ handler: "index.js",
39
+ launcherType: "Nodejs",
40
+ shouldAddHelpers: true,
41
+ };
42
+ // Emit a Build Output API tree under `.vercel/output/`. Bundling the server
43
+ // with esbuild produces a self-contained function bundle, so we don't ship
44
+ // `node_modules` and don't touch tracked paths like `api/` or `public/`.
45
+ //
46
+ // Entry is `dist/__entry.js` — the wrapper that primes the Vite manifest via
47
+ // `__setBuildManifest` before importing user code. Bundling `dist/server.js`
48
+ // directly would skip that priming and 500 on view resource reads when the
49
+ // function falls back to `readFileSync('dist/assets/.vite/manifest.json')`
50
+ // (Vercel functions don't ship `dist/`).
51
+ export async function emitVercelBuildOutput(root) {
52
+ const outputDir = path.join(root, ".vercel", "output");
53
+ const funcDir = path.join(outputDir, "functions", `${VERCEL_FUNCTION_NAME}.func`);
54
+ const staticAssetsDir = path.join(outputDir, "static", "assets");
55
+ rmSync(outputDir, { recursive: true, force: true });
56
+ mkdirSync(funcDir, { recursive: true });
57
+ const { build } = await import("esbuild");
58
+ await build({
59
+ entryPoints: [path.join(root, "dist", "__entry.js")],
60
+ bundle: true,
61
+ platform: "node",
62
+ target: "node22",
63
+ format: "esm",
64
+ outfile: path.join(funcDir, "index.js"),
65
+ // Lets esbuild DCE dev-only branches that pull in vite/devtools.
66
+ define: { "process.env.NODE_ENV": '"production"' },
67
+ // Dev-only deps reachable from re-exports; safe to leave unresolved since
68
+ // the code paths that touch them are eliminated by the NODE_ENV define.
69
+ external: ["vite", "@skybridge/devtools"],
70
+ banner: {
71
+ // ESM bundles miss CJS interop globals that some deps reach for.
72
+ js: "import{createRequire}from'node:module';const require=createRequire(import.meta.url);",
73
+ },
74
+ });
75
+ writeFileSync(path.join(funcDir, ".vc-config.json"), `${JSON.stringify(VERCEL_VC_CONFIG, null, 2)}\n`);
76
+ writeFileSync(path.join(funcDir, "package.json"), `${JSON.stringify({ type: "module" }, null, 2)}\n`);
77
+ cpSync(path.join(root, "dist", "assets"), staticAssetsDir, {
78
+ recursive: true,
79
+ });
80
+ writeFileSync(path.join(outputDir, "config.json"), `${JSON.stringify(VERCEL_CONFIG, null, 2)}\n`);
81
+ }
82
+ //# sourceMappingURL=build-helpers.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"build-helpers.js","sourceRoot":"","sources":["../../src/cli/build-helpers.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n cpSync,\n mkdirSync,\n readFileSync,\n rmSync,\n writeFileSync,\n} from \"node:fs\";\nimport path from \"node:path\";\n\n// Primes the manifest in skybridge's module scope, then dynamically imports\n// `./server.js` so user code runs *after* the side channel is set. The\n// dynamic import is load-bearing: a static `export { default } from ...` is\n// hoisted with the rest of the static graph and would evaluate `server.js`\n// before `__setBuildManifest` runs.\nexport const ENTRY_WRAPPER_CONTENT = `import { __setBuildManifest } from \"skybridge/server\";\nimport manifest from \"./vite-manifest.js\";\n\n__setBuildManifest(manifest);\n\nconst userMod = await import(\"./server.js\");\nexport default userMod.default;\n`;\n\nexport function emitEntryWrapper(distDir: string): void {\n writeFileSync(path.join(distDir, \"__entry.js\"), ENTRY_WRAPPER_CONTENT);\n}\n\nexport function emitManifestModule(\n manifestPath: string,\n outPath: string,\n): void {\n const manifest = readFileSync(manifestPath, \"utf-8\");\n writeFileSync(outPath, `export default ${manifest};\\n`);\n}\n\nexport const VERCEL_FUNCTION_NAME = \"mcp\";\n\nexport const VERCEL_CONFIG: unknown = {\n version: 3,\n routes: [\n {\n src: \"/assets/(.*)\",\n headers: { \"Access-Control-Allow-Origin\": \"*\" },\n continue: true,\n },\n { handle: \"filesystem\" },\n { src: \"/(.*)\", dest: `/${VERCEL_FUNCTION_NAME}` },\n ],\n};\n\nexport const VERCEL_VC_CONFIG: unknown = {\n runtime: \"nodejs22.x\",\n handler: \"index.js\",\n launcherType: \"Nodejs\",\n shouldAddHelpers: true,\n};\n\n// Emit a Build Output API tree under `.vercel/output/`. Bundling the server\n// with esbuild produces a self-contained function bundle, so we don't ship\n// `node_modules` and don't touch tracked paths like `api/` or `public/`.\n//\n// Entry is `dist/__entry.js` — the wrapper that primes the Vite manifest via\n// `__setBuildManifest` before importing user code. Bundling `dist/server.js`\n// directly would skip that priming and 500 on view resource reads when the\n// function falls back to `readFileSync('dist/assets/.vite/manifest.json')`\n// (Vercel functions don't ship `dist/`).\nexport async function emitVercelBuildOutput(root: string): Promise<void> {\n const outputDir = path.join(root, \".vercel\", \"output\");\n const funcDir = path.join(\n outputDir,\n \"functions\",\n `${VERCEL_FUNCTION_NAME}.func`,\n );\n const staticAssetsDir = path.join(outputDir, \"static\", \"assets\");\n\n rmSync(outputDir, { recursive: true, force: true });\n mkdirSync(funcDir, { recursive: true });\n\n const { build } = await import(\"esbuild\");\n await build({\n entryPoints: [path.join(root, \"dist\", \"__entry.js\")],\n bundle: true,\n platform: \"node\",\n target: \"node22\",\n format: \"esm\",\n outfile: path.join(funcDir, \"index.js\"),\n // Lets esbuild DCE dev-only branches that pull in vite/devtools.\n define: { \"process.env.NODE_ENV\": '\"production\"' },\n // Dev-only deps reachable from re-exports; safe to leave unresolved since\n // the code paths that touch them are eliminated by the NODE_ENV define.\n external: [\"vite\", \"@skybridge/devtools\"],\n banner: {\n // ESM bundles miss CJS interop globals that some deps reach for.\n js: \"import{createRequire}from'node:module';const require=createRequire(import.meta.url);\",\n },\n });\n\n writeFileSync(\n path.join(funcDir, \".vc-config.json\"),\n `${JSON.stringify(VERCEL_VC_CONFIG, null, 2)}\\n`,\n );\n writeFileSync(\n path.join(funcDir, \"package.json\"),\n `${JSON.stringify({ type: \"module\" }, null, 2)}\\n`,\n );\n\n cpSync(path.join(root, \"dist\", \"assets\"), staticAssetsDir, {\n recursive: true,\n });\n\n writeFileSync(\n path.join(outputDir, \"config.json\"),\n `${JSON.stringify(VERCEL_CONFIG, null, 2)}\\n`,\n );\n}\n"]}
@@ -0,0 +1,64 @@
1
+ // @vitest-environment node
2
+ // esbuild's invariant check on TextEncoder/Uint8Array trips jsdom's polyfill.
3
+ import { existsSync, mkdirSync, mkdtempSync, readFileSync, writeFileSync, } from "node:fs";
4
+ import { tmpdir } from "node:os";
5
+ import path from "node:path";
6
+ import { describe, expect, it } from "vitest";
7
+ import { ENTRY_WRAPPER_CONTENT, emitEntryWrapper, emitManifestModule, emitVercelBuildOutput, VERCEL_CONFIG, VERCEL_VC_CONFIG, } from "./build-helpers.js";
8
+ function mkTmp(prefix = "skybridge-build-helpers-") {
9
+ return mkdtempSync(path.join(tmpdir(), prefix));
10
+ }
11
+ describe("emitEntryWrapper", () => {
12
+ it("writes dist/__entry.js that primes the manifest before importing user code", () => {
13
+ const dir = mkTmp();
14
+ emitEntryWrapper(dir);
15
+ const out = readFileSync(path.join(dir, "__entry.js"), "utf-8");
16
+ expect(out).toBe(ENTRY_WRAPPER_CONTENT);
17
+ expect(out).toContain('import { __setBuildManifest } from "skybridge/server"');
18
+ expect(out).toContain('import manifest from "./vite-manifest.js"');
19
+ expect(out).toContain("__setBuildManifest(manifest)");
20
+ // Dynamic import is load-bearing: `server.js` must evaluate after the
21
+ // setter runs, so a static re-export wouldn't work.
22
+ expect(out).toContain('await import("./server.js")');
23
+ expect(out).toContain("export default userMod.default");
24
+ });
25
+ });
26
+ describe("emitManifestModule", () => {
27
+ it("inlines the JSON manifest as an ESM default export", () => {
28
+ const dir = mkTmp();
29
+ const inPath = path.join(dir, "manifest.json");
30
+ const outPath = path.join(dir, "vite-manifest.js");
31
+ const manifest = { "src/views/foo.tsx": { file: "assets/foo-abc.js" } };
32
+ writeFileSync(inPath, JSON.stringify(manifest));
33
+ emitManifestModule(inPath, outPath);
34
+ const out = readFileSync(outPath, "utf-8");
35
+ expect(out.startsWith("export default ")).toBe(true);
36
+ const literal = out
37
+ .slice("export default ".length)
38
+ .trim()
39
+ .replace(/;$/, "");
40
+ expect(JSON.parse(literal)).toEqual(manifest);
41
+ });
42
+ });
43
+ describe("emitVercelBuildOutput", () => {
44
+ it("emits a Build Output API tree with bundled function and static assets", async () => {
45
+ const root = mkTmp();
46
+ mkdirSync(path.join(root, "dist", "assets"), { recursive: true });
47
+ writeFileSync(path.join(root, "dist", "server.js"), "export default function handler(_req, res) { res.end('ok'); }\n");
48
+ // Minimal `dist/__entry.js` (the real wrapper imports `skybridge/server`,
49
+ // which isn't resolvable in this test's tmp dir — the function-bundling
50
+ // contract we care about here is just "bundle whatever `__entry.js`
51
+ // imports into a single function file").
52
+ writeFileSync(path.join(root, "dist", "__entry.js"), "const userMod = await import('./server.js');\nexport default userMod.default;\n");
53
+ writeFileSync(path.join(root, "dist", "assets", "view-abc.js"), "/* bundled view */\n");
54
+ await emitVercelBuildOutput(root);
55
+ const outputDir = path.join(root, ".vercel", "output");
56
+ const funcDir = path.join(outputDir, "functions", "mcp.func");
57
+ expect(existsSync(path.join(funcDir, "index.js"))).toBe(true);
58
+ expect(JSON.parse(readFileSync(path.join(funcDir, ".vc-config.json"), "utf-8"))).toEqual(VERCEL_VC_CONFIG);
59
+ expect(JSON.parse(readFileSync(path.join(funcDir, "package.json"), "utf-8"))).toEqual({ type: "module" });
60
+ expect(JSON.parse(readFileSync(path.join(outputDir, "config.json"), "utf-8"))).toEqual(VERCEL_CONFIG);
61
+ expect(readFileSync(path.join(outputDir, "static", "assets", "view-abc.js"), "utf-8")).toContain("bundled view");
62
+ });
63
+ });
64
+ //# sourceMappingURL=build-helpers.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"build-helpers.test.js","sourceRoot":"","sources":["../../src/cli/build-helpers.test.ts"],"names":[],"mappings":"AAAA,2BAA2B;AAC3B,8EAA8E;AAC9E,OAAO,EACL,UAAU,EACV,SAAS,EACT,WAAW,EACX,YAAY,EACZ,aAAa,GACd,MAAM,SAAS,CAAC;AACjB,OAAO,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AACjC,OAAO,IAAI,MAAM,WAAW,CAAC;AAC7B,OAAO,EAAE,QAAQ,EAAE,MAAM,EAAE,EAAE,EAAE,MAAM,QAAQ,CAAC;AAC9C,OAAO,EACL,qBAAqB,EACrB,gBAAgB,EAChB,kBAAkB,EAClB,qBAAqB,EACrB,aAAa,EACb,gBAAgB,GACjB,MAAM,oBAAoB,CAAC;AAE5B,SAAS,KAAK,CAAC,MAAM,GAAG,0BAA0B;IAChD,OAAO,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,EAAE,MAAM,CAAC,CAAC,CAAC;AAClD,CAAC;AAED,QAAQ,CAAC,kBAAkB,EAAE,GAAG,EAAE;IAChC,EAAE,CAAC,4EAA4E,EAAE,GAAG,EAAE;QACpF,MAAM,GAAG,GAAG,KAAK,EAAE,CAAC;QACpB,gBAAgB,CAAC,GAAG,CAAC,CAAC;QACtB,MAAM,GAAG,GAAG,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,EAAE,YAAY,CAAC,EAAE,OAAO,CAAC,CAAC;QAChE,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC;QACxC,MAAM,CAAC,GAAG,CAAC,CAAC,SAAS,CACnB,uDAAuD,CACxD,CAAC;QACF,MAAM,CAAC,GAAG,CAAC,CAAC,SAAS,CAAC,2CAA2C,CAAC,CAAC;QACnE,MAAM,CAAC,GAAG,CAAC,CAAC,SAAS,CAAC,8BAA8B,CAAC,CAAC;QACtD,sEAAsE;QACtE,oDAAoD;QACpD,MAAM,CAAC,GAAG,CAAC,CAAC,SAAS,CAAC,6BAA6B,CAAC,CAAC;QACrD,MAAM,CAAC,GAAG,CAAC,CAAC,SAAS,CAAC,gCAAgC,CAAC,CAAC;IAC1D,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC;AAEH,QAAQ,CAAC,oBAAoB,EAAE,GAAG,EAAE;IAClC,EAAE,CAAC,oDAAoD,EAAE,GAAG,EAAE;QAC5D,MAAM,GAAG,GAAG,KAAK,EAAE,CAAC;QACpB,MAAM,MAAM,GAAG,IAAI,CAAC,IAAI,CAAC,GAAG,EAAE,eAAe,CAAC,CAAC;QAC/C,MAAM,OAAO,GAAG,IAAI,CAAC,IAAI,CAAC,GAAG,EAAE,kBAAkB,CAAC,CAAC;QACnD,MAAM,QAAQ,GAAG,EAAE,mBAAmB,EAAE,EAAE,IAAI,EAAE,mBAAmB,EAAE,EAAE,CAAC;QACxE,aAAa,CAAC,MAAM,EAAE,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC,CAAC;QAChD,kBAAkB,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;QACpC,MAAM,GAAG,GAAG,YAAY,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC;QAC3C,MAAM,CAAC,GAAG,CAAC,UAAU,CAAC,iBAAiB,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACrD,MAAM,OAAO,GAAG,GAAG;aAChB,KAAK,CAAC,iBAAiB,CAAC,MAAM,CAAC;aAC/B,IAAI,EAAE;aACN,OAAO,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC;QACrB,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;IAChD,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC;AAEH,QAAQ,CAAC,uBAAuB,EAAE,GAAG,EAAE;IACrC,EAAE,CAAC,uEAAuE,EAAE,KAAK,IAAI,EAAE;QACrF,MAAM,IAAI,GAAG,KAAK,EAAE,CAAC;QACrB,SAAS,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,MAAM,EAAE,QAAQ,CAAC,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;QAClE,aAAa,CACX,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,MAAM,EAAE,WAAW,CAAC,EACpC,iEAAiE,CAClE,CAAC;QACF,0EAA0E;QAC1E,wEAAwE;QACxE,oEAAoE;QACpE,yCAAyC;QACzC,aAAa,CACX,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,MAAM,EAAE,YAAY,CAAC,EACrC,iFAAiF,CAClF,CAAC;QACF,aAAa,CACX,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,MAAM,EAAE,QAAQ,EAAE,aAAa,CAAC,EAChD,sBAAsB,CACvB,CAAC;QAEF,MAAM,qBAAqB,CAAC,IAAI,CAAC,CAAC;QAElC,MAAM,SAAS,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,SAAS,EAAE,QAAQ,CAAC,CAAC;QACvD,MAAM,OAAO,GAAG,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,WAAW,EAAE,UAAU,CAAC,CAAC;QAE9D,MAAM,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,UAAU,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAC9D,MAAM,CACJ,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,iBAAiB,CAAC,EAAE,OAAO,CAAC,CAAC,CACzE,CAAC,OAAO,CAAC,gBAAgB,CAAC,CAAC;QAC5B,MAAM,CACJ,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,cAAc,CAAC,EAAE,OAAO,CAAC,CAAC,CACtE,CAAC,OAAO,CAAC,EAAE,IAAI,EAAE,QAAQ,EAAE,CAAC,CAAC;QAC9B,MAAM,CACJ,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,aAAa,CAAC,EAAE,OAAO,CAAC,CAAC,CACvE,CAAC,OAAO,CAAC,aAAa,CAAC,CAAC;QACzB,MAAM,CACJ,YAAY,CACV,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,QAAQ,EAAE,QAAQ,EAAE,aAAa,CAAC,EACvD,OAAO,CACR,CACF,CAAC,SAAS,CAAC,cAAc,CAAC,CAAC;IAC9B,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC","sourcesContent":["// @vitest-environment node\n// esbuild's invariant check on TextEncoder/Uint8Array trips jsdom's polyfill.\nimport {\n existsSync,\n mkdirSync,\n mkdtempSync,\n readFileSync,\n writeFileSync,\n} from \"node:fs\";\nimport { tmpdir } from \"node:os\";\nimport path from \"node:path\";\nimport { describe, expect, it } from \"vitest\";\nimport {\n ENTRY_WRAPPER_CONTENT,\n emitEntryWrapper,\n emitManifestModule,\n emitVercelBuildOutput,\n VERCEL_CONFIG,\n VERCEL_VC_CONFIG,\n} from \"./build-helpers.js\";\n\nfunction mkTmp(prefix = \"skybridge-build-helpers-\") {\n return mkdtempSync(path.join(tmpdir(), prefix));\n}\n\ndescribe(\"emitEntryWrapper\", () => {\n it(\"writes dist/__entry.js that primes the manifest before importing user code\", () => {\n const dir = mkTmp();\n emitEntryWrapper(dir);\n const out = readFileSync(path.join(dir, \"__entry.js\"), \"utf-8\");\n expect(out).toBe(ENTRY_WRAPPER_CONTENT);\n expect(out).toContain(\n 'import { __setBuildManifest } from \"skybridge/server\"',\n );\n expect(out).toContain('import manifest from \"./vite-manifest.js\"');\n expect(out).toContain(\"__setBuildManifest(manifest)\");\n // Dynamic import is load-bearing: `server.js` must evaluate after the\n // setter runs, so a static re-export wouldn't work.\n expect(out).toContain('await import(\"./server.js\")');\n expect(out).toContain(\"export default userMod.default\");\n });\n});\n\ndescribe(\"emitManifestModule\", () => {\n it(\"inlines the JSON manifest as an ESM default export\", () => {\n const dir = mkTmp();\n const inPath = path.join(dir, \"manifest.json\");\n const outPath = path.join(dir, \"vite-manifest.js\");\n const manifest = { \"src/views/foo.tsx\": { file: \"assets/foo-abc.js\" } };\n writeFileSync(inPath, JSON.stringify(manifest));\n emitManifestModule(inPath, outPath);\n const out = readFileSync(outPath, \"utf-8\");\n expect(out.startsWith(\"export default \")).toBe(true);\n const literal = out\n .slice(\"export default \".length)\n .trim()\n .replace(/;$/, \"\");\n expect(JSON.parse(literal)).toEqual(manifest);\n });\n});\n\ndescribe(\"emitVercelBuildOutput\", () => {\n it(\"emits a Build Output API tree with bundled function and static assets\", async () => {\n const root = mkTmp();\n mkdirSync(path.join(root, \"dist\", \"assets\"), { recursive: true });\n writeFileSync(\n path.join(root, \"dist\", \"server.js\"),\n \"export default function handler(_req, res) { res.end('ok'); }\\n\",\n );\n // Minimal `dist/__entry.js` (the real wrapper imports `skybridge/server`,\n // which isn't resolvable in this test's tmp dir — the function-bundling\n // contract we care about here is just \"bundle whatever `__entry.js`\n // imports into a single function file\").\n writeFileSync(\n path.join(root, \"dist\", \"__entry.js\"),\n \"const userMod = await import('./server.js');\\nexport default userMod.default;\\n\",\n );\n writeFileSync(\n path.join(root, \"dist\", \"assets\", \"view-abc.js\"),\n \"/* bundled view */\\n\",\n );\n\n await emitVercelBuildOutput(root);\n\n const outputDir = path.join(root, \".vercel\", \"output\");\n const funcDir = path.join(outputDir, \"functions\", \"mcp.func\");\n\n expect(existsSync(path.join(funcDir, \"index.js\"))).toBe(true);\n expect(\n JSON.parse(readFileSync(path.join(funcDir, \".vc-config.json\"), \"utf-8\")),\n ).toEqual(VERCEL_VC_CONFIG);\n expect(\n JSON.parse(readFileSync(path.join(funcDir, \"package.json\"), \"utf-8\")),\n ).toEqual({ type: \"module\" });\n expect(\n JSON.parse(readFileSync(path.join(outputDir, \"config.json\"), \"utf-8\")),\n ).toEqual(VERCEL_CONFIG);\n expect(\n readFileSync(\n path.join(outputDir, \"static\", \"assets\", \"view-abc.js\"),\n \"utf-8\",\n ),\n ).toContain(\"bundled view\");\n });\n});\n"]}
@@ -0,0 +1,18 @@
1
+ export declare function resolvePort(flagPort?: number): Promise<{
2
+ port: number;
3
+ fallback: boolean;
4
+ envWarning?: undefined;
5
+ } | {
6
+ port: number;
7
+ fallback: boolean;
8
+ envWarning: string;
9
+ }>;
10
+ /**
11
+ * Returns the first available port at or after `startPort`, incrementing
12
+ * by one until a free port is found or `MAX_PORT_INCREMENT` is reached.
13
+ *
14
+ * @param host - Bind address for the check. Pass `"localhost"` for
15
+ * services that bind to 127.0.0.1 (e.g. Vite HMR). Omit for
16
+ * services that bind to all interfaces (e.g. the HTTP server).
17
+ */
18
+ export declare function detectAvailablePort(startPort: number, host?: string): Promise<number>;
@@ -0,0 +1,50 @@
1
+ import net from "node:net";
2
+ const DEFAULT_PORT = 3000;
3
+ const MAX_PORT_INCREMENT = 100;
4
+ export async function resolvePort(flagPort) {
5
+ if (flagPort && flagPort > 1) {
6
+ return { port: flagPort, fallback: false };
7
+ }
8
+ const rawEnv = process.env.PORT;
9
+ if (rawEnv) {
10
+ const parsed = Number(rawEnv);
11
+ if (Number.isInteger(parsed) && parsed > 0) {
12
+ return { port: parsed, fallback: false };
13
+ }
14
+ return {
15
+ port: await detectAvailablePort(DEFAULT_PORT),
16
+ fallback: false,
17
+ envWarning: `Invalid PORT="${rawEnv}", ignoring and using default`,
18
+ };
19
+ }
20
+ const port = await detectAvailablePort(DEFAULT_PORT);
21
+ return { port, fallback: port !== DEFAULT_PORT };
22
+ }
23
+ /**
24
+ * Returns the first available port at or after `startPort`, incrementing
25
+ * by one until a free port is found or `MAX_PORT_INCREMENT` is reached.
26
+ *
27
+ * @param host - Bind address for the check. Pass `"localhost"` for
28
+ * services that bind to 127.0.0.1 (e.g. Vite HMR). Omit for
29
+ * services that bind to all interfaces (e.g. the HTTP server).
30
+ */
31
+ export async function detectAvailablePort(startPort, host) {
32
+ for (let port = startPort; port < startPort + MAX_PORT_INCREMENT; port++) {
33
+ if (await isPortAvailable(port, host)) {
34
+ return port;
35
+ }
36
+ console.log(`Port ${port} is in use, trying another one...`);
37
+ }
38
+ throw new Error(`No available port found between ${startPort} and ${startPort + MAX_PORT_INCREMENT - 1}`);
39
+ }
40
+ function isPortAvailable(port, host) {
41
+ return new Promise((resolve) => {
42
+ const server = net.createServer();
43
+ server.once("error", () => resolve(false));
44
+ server.once("listening", () => {
45
+ server.close(() => resolve(true));
46
+ });
47
+ server.listen(port, host);
48
+ });
49
+ }
50
+ //# sourceMappingURL=detect-port.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"detect-port.js","sourceRoot":"","sources":["../../src/cli/detect-port.ts"],"names":[],"mappings":"AAAA,OAAO,GAAG,MAAM,UAAU,CAAC;AAE3B,MAAM,YAAY,GAAG,IAAI,CAAC;AAC1B,MAAM,kBAAkB,GAAG,GAAG,CAAC;AAE/B,MAAM,CAAC,KAAK,UAAU,WAAW,CAAC,QAAiB;IACjD,IAAI,QAAQ,IAAI,QAAQ,GAAG,CAAC,EAAE,CAAC;QAC7B,OAAO,EAAE,IAAI,EAAE,QAAQ,EAAE,QAAQ,EAAE,KAAK,EAAE,CAAC;IAC7C,CAAC;IAED,MAAM,MAAM,GAAG,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC;IAChC,IAAI,MAAM,EAAE,CAAC;QACX,MAAM,MAAM,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC;QAC9B,IAAI,MAAM,CAAC,SAAS,CAAC,MAAM,CAAC,IAAI,MAAM,GAAG,CAAC,EAAE,CAAC;YAC3C,OAAO,EAAE,IAAI,EAAE,MAAM,EAAE,QAAQ,EAAE,KAAK,EAAE,CAAC;QAC3C,CAAC;QACD,OAAO;YACL,IAAI,EAAE,MAAM,mBAAmB,CAAC,YAAY,CAAC;YAC7C,QAAQ,EAAE,KAAK;YACf,UAAU,EAAE,iBAAiB,MAAM,+BAA+B;SACnE,CAAC;IACJ,CAAC;IAED,MAAM,IAAI,GAAG,MAAM,mBAAmB,CAAC,YAAY,CAAC,CAAC;IACrD,OAAO,EAAE,IAAI,EAAE,QAAQ,EAAE,IAAI,KAAK,YAAY,EAAE,CAAC;AACnD,CAAC;AAED;;;;;;;GAOG;AACH,MAAM,CAAC,KAAK,UAAU,mBAAmB,CACvC,SAAiB,EACjB,IAAa;IAEb,KAAK,IAAI,IAAI,GAAG,SAAS,EAAE,IAAI,GAAG,SAAS,GAAG,kBAAkB,EAAE,IAAI,EAAE,EAAE,CAAC;QACzE,IAAI,MAAM,eAAe,CAAC,IAAI,EAAE,IAAI,CAAC,EAAE,CAAC;YACtC,OAAO,IAAI,CAAC;QACd,CAAC;QACD,OAAO,CAAC,GAAG,CAAC,QAAQ,IAAI,mCAAmC,CAAC,CAAC;IAC/D,CAAC;IACD,MAAM,IAAI,KAAK,CACb,mCAAmC,SAAS,QAAQ,SAAS,GAAG,kBAAkB,GAAG,CAAC,EAAE,CACzF,CAAC;AACJ,CAAC;AAED,SAAS,eAAe,CAAC,IAAY,EAAE,IAAa;IAClD,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE;QAC7B,MAAM,MAAM,GAAG,GAAG,CAAC,YAAY,EAAE,CAAC;QAElC,MAAM,CAAC,IAAI,CAAC,OAAO,EAAE,GAAG,EAAE,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC;QAC3C,MAAM,CAAC,IAAI,CAAC,WAAW,EAAE,GAAG,EAAE;YAC5B,MAAM,CAAC,KAAK,CAAC,GAAG,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC;QACpC,CAAC,CAAC,CAAC;QAEH,MAAM,CAAC,MAAM,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC;IAC5B,CAAC,CAAC,CAAC;AACL,CAAC","sourcesContent":["import net from \"node:net\";\n\nconst DEFAULT_PORT = 3000;\nconst MAX_PORT_INCREMENT = 100;\n\nexport async function resolvePort(flagPort?: number) {\n if (flagPort && flagPort > 1) {\n return { port: flagPort, fallback: false };\n }\n\n const rawEnv = process.env.PORT;\n if (rawEnv) {\n const parsed = Number(rawEnv);\n if (Number.isInteger(parsed) && parsed > 0) {\n return { port: parsed, fallback: false };\n }\n return {\n port: await detectAvailablePort(DEFAULT_PORT),\n fallback: false,\n envWarning: `Invalid PORT=\"${rawEnv}\", ignoring and using default`,\n };\n }\n\n const port = await detectAvailablePort(DEFAULT_PORT);\n return { port, fallback: port !== DEFAULT_PORT };\n}\n\n/**\n * Returns the first available port at or after `startPort`, incrementing\n * by one until a free port is found or `MAX_PORT_INCREMENT` is reached.\n *\n * @param host - Bind address for the check. Pass `\"localhost\"` for\n * services that bind to 127.0.0.1 (e.g. Vite HMR). Omit for\n * services that bind to all interfaces (e.g. the HTTP server).\n */\nexport async function detectAvailablePort(\n startPort: number,\n host?: string,\n): Promise<number> {\n for (let port = startPort; port < startPort + MAX_PORT_INCREMENT; port++) {\n if (await isPortAvailable(port, host)) {\n return port;\n }\n console.log(`Port ${port} is in use, trying another one...`);\n }\n throw new Error(\n `No available port found between ${startPort} and ${startPort + MAX_PORT_INCREMENT - 1}`,\n );\n}\n\nfunction isPortAvailable(port: number, host?: string): Promise<boolean> {\n return new Promise((resolve) => {\n const server = net.createServer();\n\n server.once(\"error\", () => resolve(false));\n server.once(\"listening\", () => {\n server.close(() => resolve(true));\n });\n\n server.listen(port, host);\n });\n}\n"]}
@@ -1,6 +1,6 @@
1
1
  import { jsxs as _jsxs } from "react/jsx-runtime";
2
2
  import { Box, Text } from "ink";
3
3
  export const Header = ({ version, children, }) => {
4
- return (_jsxs(Box, { marginBottom: 1, children: [_jsxs(Text, { color: "cyan", bold: true, children: ["\u26F0", " ", "Welcome to Skybridge"] }), _jsxs(Text, { color: "cyan", children: [" v", version] }), children] }));
4
+ return (_jsxs(Box, { marginBottom: 1, children: [_jsxs(Text, { color: "cyan", bold: true, children: ["\u26F0", " ", "Skybridge"] }), _jsxs(Text, { color: "cyan", children: [" v", version] }), children] }));
5
5
  };
6
6
  //# sourceMappingURL=header.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"header.js","sourceRoot":"","sources":["../../src/cli/header.tsx"],"names":[],"mappings":";AAAA,OAAO,EAAE,GAAG,EAAE,IAAI,EAAE,MAAM,KAAK,CAAC;AAEhC,MAAM,CAAC,MAAM,MAAM,GAAG,CAAC,EACrB,OAAO,EACP,QAAQ,GAIT,EAAE,EAAE;IACH,OAAO,CACL,MAAC,GAAG,IAAC,YAAY,EAAE,CAAC,aAClB,MAAC,IAAI,IAAC,KAAK,EAAC,MAAM,EAAC,IAAI,6BACnB,IAAI,4BACD,EACP,MAAC,IAAI,IAAC,KAAK,EAAC,MAAM,mBAAI,OAAO,IAAQ,EACpC,QAAQ,IACL,CACP,CAAC;AACJ,CAAC,CAAC"}
1
+ {"version":3,"file":"header.js","sourceRoot":"","sources":["../../src/cli/header.tsx"],"names":[],"mappings":";AAAA,OAAO,EAAE,GAAG,EAAE,IAAI,EAAE,MAAM,KAAK,CAAC;AAEhC,MAAM,CAAC,MAAM,MAAM,GAAG,CAAC,EACrB,OAAO,EACP,QAAQ,GAIT,EAAE,EAAE;IACH,OAAO,CACL,MAAC,GAAG,IAAC,YAAY,EAAE,CAAC,aAClB,MAAC,IAAI,IAAC,KAAK,EAAC,MAAM,EAAC,IAAI,6BACnB,IAAI,iBACD,EACP,MAAC,IAAI,IAAC,KAAK,EAAC,MAAM,mBAAI,OAAO,IAAQ,EACpC,QAAQ,IACL,CACP,CAAC;AACJ,CAAC,CAAC","sourcesContent":["import { Box, Text } from \"ink\";\n\nexport const Header = ({\n version,\n children,\n}: {\n version: string;\n children?: React.ReactNode;\n}) => {\n return (\n <Box marginBottom={1}>\n <Text color=\"cyan\" bold>\n ⛰{\" \"}Skybridge\n </Text>\n <Text color=\"cyan\"> v{version}</Text>\n {children}\n </Box>\n );\n};\n"]}
@@ -0,0 +1 @@
1
+ export declare function resolveViewsDir(root: string): Promise<string | undefined>;
@@ -0,0 +1,17 @@
1
+ export async function resolveViewsDir(root) {
2
+ const { loadConfigFromFile } = await import("vite");
3
+ const loaded = await loadConfigFromFile({ command: "build", mode: "production" }, undefined, root);
4
+ const isPluginCandidate = (value) => typeof value === "object" && value !== null;
5
+ const plugins = [];
6
+ const walk = (value) => {
7
+ if (Array.isArray(value)) {
8
+ value.forEach(walk);
9
+ }
10
+ else if (isPluginCandidate(value)) {
11
+ plugins.push(value);
12
+ }
13
+ };
14
+ walk(loaded?.config.plugins ?? []);
15
+ return plugins.find((p) => p.name === "skybridge")?.api?.viewsDir;
16
+ }
17
+ //# sourceMappingURL=resolve-views-dir.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"resolve-views-dir.js","sourceRoot":"","sources":["../../src/cli/resolve-views-dir.ts"],"names":[],"mappings":"AAAA,MAAM,CAAC,KAAK,UAAU,eAAe,CACnC,IAAY;IAEZ,MAAM,EAAE,kBAAkB,EAAE,GAAG,MAAM,MAAM,CAAC,MAAM,CAAC,CAAC;IACpD,MAAM,MAAM,GAAG,MAAM,kBAAkB,CACrC,EAAE,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,YAAY,EAAE,EACxC,SAAS,EACT,IAAI,CACL,CAAC;IAEF,MAAM,iBAAiB,GAAG,CACxB,KAAc,EAC2C,EAAE,CAC3D,OAAO,KAAK,KAAK,QAAQ,IAAI,KAAK,KAAK,IAAI,CAAC;IAE9C,MAAM,OAAO,GAA0D,EAAE,CAAC;IAC1E,MAAM,IAAI,GAAG,CAAC,KAAc,EAAE,EAAE;QAC9B,IAAI,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC;YACzB,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;QACtB,CAAC;aAAM,IAAI,iBAAiB,CAAC,KAAK,CAAC,EAAE,CAAC;YACpC,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACtB,CAAC;IACH,CAAC,CAAC;IACF,IAAI,CAAC,MAAM,EAAE,MAAM,CAAC,OAAO,IAAI,EAAE,CAAC,CAAC;IACnC,OAAO,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,WAAW,CAAC,EAAE,GAAG,EAAE,QAAQ,CAAC;AACpE,CAAC","sourcesContent":["export async function resolveViewsDir(\n root: string,\n): Promise<string | undefined> {\n const { loadConfigFromFile } = await import(\"vite\");\n const loaded = await loadConfigFromFile(\n { command: \"build\", mode: \"production\" },\n undefined,\n root,\n );\n\n const isPluginCandidate = (\n value: unknown,\n ): value is { name?: string; api?: { viewsDir?: string } } =>\n typeof value === \"object\" && value !== null;\n\n const plugins: Array<{ name?: string; api?: { viewsDir?: string } }> = [];\n const walk = (value: unknown) => {\n if (Array.isArray(value)) {\n value.forEach(walk);\n } else if (isPluginCandidate(value)) {\n plugins.push(value);\n }\n };\n walk(loaded?.config.plugins ?? []);\n return plugins.find((p) => p.name === \"skybridge\")?.api?.viewsDir;\n}\n"]}