skybridge 0.0.0-dev.fd6e4e8 → 0.0.0-dev.fd767e9

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 (309) hide show
  1. package/README.md +123 -124
  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 +2 -2
  8. package/dist/cli/detect-port.js +9 -20
  9. package/dist/cli/detect-port.js.map +1 -1
  10. package/dist/cli/header.js.map +1 -1
  11. package/dist/cli/resolve-views-dir.d.ts +1 -0
  12. package/dist/cli/resolve-views-dir.js +17 -0
  13. package/dist/cli/resolve-views-dir.js.map +1 -0
  14. package/dist/cli/run-command.js.map +1 -1
  15. package/dist/cli/telemetry.js.map +1 -1
  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.js.map +1 -1
  32. package/dist/cli/use-execute-steps.js.map +1 -1
  33. package/dist/cli/use-messages.js.map +1 -1
  34. package/dist/cli/use-nodemon.js +11 -2
  35. package/dist/cli/use-nodemon.js.map +1 -1
  36. package/dist/cli/use-open-browser.d.ts +1 -0
  37. package/dist/cli/use-open-browser.js +44 -0
  38. package/dist/cli/use-open-browser.js.map +1 -0
  39. package/dist/cli/use-open-tunnel-browser.d.ts +6 -0
  40. package/dist/cli/use-open-tunnel-browser.js +19 -0
  41. package/dist/cli/use-open-tunnel-browser.js.map +1 -0
  42. package/dist/cli/use-tunnel.d.ts +1 -1
  43. package/dist/cli/use-tunnel.js +102 -68
  44. package/dist/cli/use-tunnel.js.map +1 -1
  45. package/dist/cli/use-typescript-check.d.ts +1 -0
  46. package/dist/cli/use-typescript-check.js +42 -7
  47. package/dist/cli/use-typescript-check.js.map +1 -1
  48. package/dist/commands/build.d.ts +0 -1
  49. package/dist/commands/build.js +51 -8
  50. package/dist/commands/build.js.map +1 -1
  51. package/dist/commands/create.d.ts +9 -0
  52. package/dist/commands/create.js +30 -0
  53. package/dist/commands/create.js.map +1 -0
  54. package/dist/commands/dev.d.ts +1 -0
  55. package/dist/commands/dev.js +51 -2
  56. package/dist/commands/dev.js.map +1 -1
  57. package/dist/commands/start.js +7 -1
  58. package/dist/commands/start.js.map +1 -1
  59. package/dist/commands/telemetry/disable.js.map +1 -1
  60. package/dist/commands/telemetry/enable.js.map +1 -1
  61. package/dist/commands/telemetry/status.js.map +1 -1
  62. package/dist/server/asset-base-url-transform-plugin.d.ts +1 -0
  63. package/dist/server/asset-base-url-transform-plugin.js +17 -2
  64. package/dist/server/asset-base-url-transform-plugin.js.map +1 -1
  65. package/dist/server/asset-base-url-transform-plugin.test.js +80 -1
  66. package/dist/server/asset-base-url-transform-plugin.test.js.map +1 -1
  67. package/dist/server/auth.d.ts +20 -0
  68. package/dist/server/auth.js +28 -0
  69. package/dist/server/auth.js.map +1 -0
  70. package/dist/server/build-manifest.test.d.ts +1 -0
  71. package/dist/server/build-manifest.test.js +27 -0
  72. package/dist/server/build-manifest.test.js.map +1 -0
  73. package/dist/server/content-helpers.d.ts +40 -0
  74. package/dist/server/content-helpers.js +33 -0
  75. package/dist/server/content-helpers.js.map +1 -1
  76. package/dist/server/content-helpers.test.js +1 -1
  77. package/dist/server/content-helpers.test.js.map +1 -1
  78. package/dist/server/express.d.ts +1 -5
  79. package/dist/server/express.js +34 -10
  80. package/dist/server/express.js.map +1 -1
  81. package/dist/server/express.test.js +279 -71
  82. package/dist/server/express.test.js.map +1 -1
  83. package/dist/server/file-ref.d.ts +28 -0
  84. package/dist/server/file-ref.js +27 -0
  85. package/dist/server/file-ref.js.map +1 -0
  86. package/dist/server/index.d.ts +5 -3
  87. package/dist/server/index.js +4 -2
  88. package/dist/server/index.js.map +1 -1
  89. package/dist/server/inferUtilityTypes.d.ts +6 -6
  90. package/dist/server/inferUtilityTypes.js.map +1 -1
  91. package/dist/server/metric.js.map +1 -1
  92. package/dist/server/middleware.d.ts +16 -3
  93. package/dist/server/middleware.js.map +1 -1
  94. package/dist/server/middleware.test-d.js.map +1 -1
  95. package/dist/server/middleware.test.js.map +1 -1
  96. package/dist/server/server.d.ts +248 -41
  97. package/dist/server/server.js +327 -114
  98. package/dist/server/server.js.map +1 -1
  99. package/dist/server/templateHelper.d.ts +5 -7
  100. package/dist/server/templateHelper.js +3 -22
  101. package/dist/server/templateHelper.js.map +1 -1
  102. package/dist/server/templates.generated.d.ts +4 -0
  103. package/dist/server/templates.generated.js +47 -0
  104. package/dist/server/templates.generated.js.map +1 -0
  105. package/dist/server/tunnel-proxy-router.d.ts +7 -0
  106. package/dist/server/tunnel-proxy-router.js +110 -0
  107. package/dist/server/tunnel-proxy-router.js.map +1 -0
  108. package/dist/server/tunnel-proxy-router.test.d.ts +1 -0
  109. package/dist/server/tunnel-proxy-router.test.js +229 -0
  110. package/dist/server/tunnel-proxy-router.test.js.map +1 -0
  111. package/dist/server/viewsDevServer.d.ts +14 -0
  112. package/dist/server/{widgetsDevServer.js → viewsDevServer.js} +6 -6
  113. package/dist/server/viewsDevServer.js.map +1 -0
  114. package/dist/test/utils.d.ts +7 -7
  115. package/dist/test/utils.js +21 -21
  116. package/dist/test/utils.js.map +1 -1
  117. package/dist/test/view.test.d.ts +1 -0
  118. package/dist/test/{widget.test.js → view.test.js} +173 -59
  119. package/dist/test/view.test.js.map +1 -0
  120. package/dist/version.js +1 -3
  121. package/dist/version.js.map +1 -1
  122. package/dist/web/bridges/apps-sdk/adaptor.d.ts +8 -3
  123. package/dist/web/bridges/apps-sdk/adaptor.js +35 -5
  124. package/dist/web/bridges/apps-sdk/adaptor.js.map +1 -1
  125. package/dist/web/bridges/apps-sdk/bridge.d.ts +1 -0
  126. package/dist/web/bridges/apps-sdk/bridge.js +1 -0
  127. package/dist/web/bridges/apps-sdk/bridge.js.map +1 -1
  128. package/dist/web/bridges/apps-sdk/index.js.map +1 -1
  129. package/dist/web/bridges/apps-sdk/types.d.ts +8 -1
  130. package/dist/web/bridges/apps-sdk/types.js.map +1 -1
  131. package/dist/web/bridges/apps-sdk/use-apps-sdk-context.d.ts +11 -0
  132. package/dist/web/bridges/apps-sdk/use-apps-sdk-context.js +11 -0
  133. package/dist/web/bridges/apps-sdk/use-apps-sdk-context.js.map +1 -1
  134. package/dist/web/bridges/get-adaptor.d.ts +7 -0
  135. package/dist/web/bridges/get-adaptor.js +7 -0
  136. package/dist/web/bridges/get-adaptor.js.map +1 -1
  137. package/dist/web/bridges/index.js.map +1 -1
  138. package/dist/web/bridges/mcp-app/adaptor.d.ts +12 -7
  139. package/dist/web/bridges/mcp-app/adaptor.js +45 -30
  140. package/dist/web/bridges/mcp-app/adaptor.js.map +1 -1
  141. package/dist/web/bridges/mcp-app/bridge.d.ts +4 -2
  142. package/dist/web/bridges/mcp-app/bridge.js +23 -1
  143. package/dist/web/bridges/mcp-app/bridge.js.map +1 -1
  144. package/dist/web/bridges/mcp-app/index.js.map +1 -1
  145. package/dist/web/bridges/mcp-app/types.js.map +1 -1
  146. package/dist/web/bridges/mcp-app/use-mcp-app-context.d.ts +12 -0
  147. package/dist/web/bridges/mcp-app/use-mcp-app-context.js +12 -0
  148. package/dist/web/bridges/mcp-app/use-mcp-app-context.js.map +1 -1
  149. package/dist/web/bridges/mcp-app/use-mcp-app-context.test.js.map +1 -1
  150. package/dist/web/bridges/mcp-app/view-tools.test.d.ts +1 -0
  151. package/dist/web/bridges/mcp-app/view-tools.test.js +144 -0
  152. package/dist/web/bridges/mcp-app/view-tools.test.js.map +1 -0
  153. package/dist/web/bridges/types.d.ts +105 -10
  154. package/dist/web/bridges/types.js.map +1 -1
  155. package/dist/web/bridges/use-host-context.d.ts +5 -0
  156. package/dist/web/bridges/use-host-context.js +5 -0
  157. package/dist/web/bridges/use-host-context.js.map +1 -1
  158. package/dist/web/components/modal-provider.js +1 -1
  159. package/dist/web/components/modal-provider.js.map +1 -1
  160. package/dist/web/create-store.d.ts +26 -0
  161. package/dist/web/create-store.js +35 -9
  162. package/dist/web/create-store.js.map +1 -1
  163. package/dist/web/create-store.test.js +14 -16
  164. package/dist/web/create-store.test.js.map +1 -1
  165. package/dist/web/data-llm.d.ts +34 -1
  166. package/dist/web/data-llm.js +31 -3
  167. package/dist/web/data-llm.js.map +1 -1
  168. package/dist/web/data-llm.test.js +22 -22
  169. package/dist/web/data-llm.test.js.map +1 -1
  170. package/dist/web/generate-helpers.d.ts +16 -14
  171. package/dist/web/generate-helpers.js +16 -14
  172. package/dist/web/generate-helpers.js.map +1 -1
  173. package/dist/web/generate-helpers.test-d.js +30 -28
  174. package/dist/web/generate-helpers.test-d.js.map +1 -1
  175. package/dist/web/generate-helpers.test.js.map +1 -1
  176. package/dist/web/helpers/state.d.ts +2 -2
  177. package/dist/web/helpers/state.js +11 -11
  178. package/dist/web/helpers/state.js.map +1 -1
  179. package/dist/web/helpers/state.test.js +9 -9
  180. package/dist/web/helpers/state.test.js.map +1 -1
  181. package/dist/web/hooks/index.d.ts +5 -1
  182. package/dist/web/hooks/index.js +5 -1
  183. package/dist/web/hooks/index.js.map +1 -1
  184. package/dist/web/hooks/test/utils.d.ts +6 -2
  185. package/dist/web/hooks/test/utils.js +13 -2
  186. package/dist/web/hooks/test/utils.js.map +1 -1
  187. package/dist/web/hooks/use-call-tool.d.ts +45 -0
  188. package/dist/web/hooks/use-call-tool.js +28 -0
  189. package/dist/web/hooks/use-call-tool.js.map +1 -1
  190. package/dist/web/hooks/use-call-tool.test-d.js.map +1 -1
  191. package/dist/web/hooks/use-call-tool.test.js +27 -6
  192. package/dist/web/hooks/use-call-tool.test.js.map +1 -1
  193. package/dist/web/hooks/use-display-mode.d.ts +20 -0
  194. package/dist/web/hooks/use-display-mode.js +20 -0
  195. package/dist/web/hooks/use-display-mode.js.map +1 -1
  196. package/dist/web/hooks/use-display-mode.test-d.js.map +1 -1
  197. package/dist/web/hooks/use-display-mode.test.js.map +1 -1
  198. package/dist/web/hooks/use-download.d.ts +5 -0
  199. package/dist/web/hooks/use-download.js +8 -0
  200. package/dist/web/hooks/use-download.js.map +1 -0
  201. package/dist/web/hooks/use-download.test.d.ts +1 -0
  202. package/dist/web/hooks/use-download.test.js +95 -0
  203. package/dist/web/hooks/use-download.test.js.map +1 -0
  204. package/dist/web/hooks/use-files.d.ts +32 -0
  205. package/dist/web/hooks/use-files.js +32 -0
  206. package/dist/web/hooks/use-files.js.map +1 -1
  207. package/dist/web/hooks/use-files.test.js.map +1 -1
  208. package/dist/web/hooks/use-layout.d.ts +2 -0
  209. package/dist/web/hooks/use-layout.js +2 -0
  210. package/dist/web/hooks/use-layout.js.map +1 -1
  211. package/dist/web/hooks/use-layout.test.js.map +1 -1
  212. package/dist/web/hooks/use-open-external.d.ts +17 -0
  213. package/dist/web/hooks/use-open-external.js +16 -0
  214. package/dist/web/hooks/use-open-external.js.map +1 -1
  215. package/dist/web/hooks/use-open-external.test.js.map +1 -1
  216. package/dist/web/hooks/use-register-view-tool.d.ts +38 -0
  217. package/dist/web/hooks/use-register-view-tool.js +50 -0
  218. package/dist/web/hooks/use-register-view-tool.js.map +1 -0
  219. package/dist/web/hooks/use-request-close.d.ts +16 -0
  220. package/dist/web/hooks/use-request-close.js +21 -0
  221. package/dist/web/hooks/use-request-close.js.map +1 -0
  222. package/dist/web/hooks/use-request-close.test.d.ts +1 -0
  223. package/dist/web/hooks/use-request-close.test.js +52 -0
  224. package/dist/web/hooks/use-request-close.test.js.map +1 -0
  225. package/dist/web/hooks/use-request-modal.d.ts +16 -1
  226. package/dist/web/hooks/use-request-modal.js +19 -4
  227. package/dist/web/hooks/use-request-modal.js.map +1 -1
  228. package/dist/web/hooks/use-request-modal.test.js +1 -1
  229. package/dist/web/hooks/use-request-modal.test.js.map +1 -1
  230. package/dist/web/hooks/use-request-size.d.ts +20 -0
  231. package/dist/web/hooks/use-request-size.js +24 -0
  232. package/dist/web/hooks/use-request-size.js.map +1 -0
  233. package/dist/web/hooks/use-request-size.test.d.ts +1 -0
  234. package/dist/web/hooks/use-request-size.test.js +65 -0
  235. package/dist/web/hooks/use-request-size.test.js.map +1 -0
  236. package/dist/web/hooks/use-send-follow-up-message.d.ts +19 -1
  237. package/dist/web/hooks/use-send-follow-up-message.js +19 -2
  238. package/dist/web/hooks/use-send-follow-up-message.js.map +1 -1
  239. package/dist/web/hooks/use-set-open-in-app-url.d.ts +17 -0
  240. package/dist/web/hooks/use-set-open-in-app-url.js +17 -0
  241. package/dist/web/hooks/use-set-open-in-app-url.js.map +1 -1
  242. package/dist/web/hooks/use-set-open-in-app-url.test.js.map +1 -1
  243. package/dist/web/hooks/use-tool-info.d.ts +53 -2
  244. package/dist/web/hooks/use-tool-info.js +30 -7
  245. package/dist/web/hooks/use-tool-info.js.map +1 -1
  246. package/dist/web/hooks/use-tool-info.test-d.js +11 -29
  247. package/dist/web/hooks/use-tool-info.test-d.js.map +1 -1
  248. package/dist/web/hooks/use-tool-info.test.js +5 -5
  249. package/dist/web/hooks/use-tool-info.test.js.map +1 -1
  250. package/dist/web/hooks/use-user.d.ts +2 -0
  251. package/dist/web/hooks/use-user.js +2 -0
  252. package/dist/web/hooks/use-user.js.map +1 -1
  253. package/dist/web/hooks/use-user.test.js.map +1 -1
  254. package/dist/web/hooks/use-view-state.d.ts +25 -0
  255. package/dist/web/hooks/use-view-state.js +32 -0
  256. package/dist/web/hooks/use-view-state.js.map +1 -0
  257. package/dist/web/hooks/use-view-state.test.d.ts +1 -0
  258. package/dist/web/hooks/{use-widget-state.test.js → use-view-state.test.js} +17 -17
  259. package/dist/web/hooks/use-view-state.test.js.map +1 -0
  260. package/dist/web/index.d.ts +1 -3
  261. package/dist/web/index.js +1 -2
  262. package/dist/web/index.js.map +1 -1
  263. package/dist/web/mount-view.d.ts +20 -0
  264. package/dist/web/{mount-widget.js → mount-view.js} +21 -2
  265. package/dist/web/mount-view.js.map +1 -0
  266. package/dist/web/plugin/data-llm.test.js.map +1 -1
  267. package/dist/web/plugin/plugin.d.ts +29 -1
  268. package/dist/web/plugin/plugin.js +113 -55
  269. package/dist/web/plugin/plugin.js.map +1 -1
  270. package/dist/web/plugin/scan-views.d.ts +16 -0
  271. package/dist/web/plugin/scan-views.js +88 -0
  272. package/dist/web/plugin/scan-views.js.map +1 -0
  273. package/dist/web/plugin/scan-views.test.d.ts +1 -0
  274. package/dist/web/plugin/scan-views.test.js +99 -0
  275. package/dist/web/plugin/scan-views.test.js.map +1 -0
  276. package/dist/web/plugin/transform-data-llm.js.map +1 -1
  277. package/dist/web/plugin/transform-data-llm.test.js.map +1 -1
  278. package/dist/web/plugin/{validate-widget.js → validate-view.js} +1 -1
  279. package/dist/web/plugin/validate-view.js.map +1 -0
  280. package/dist/web/plugin/validate-view.test.d.ts +1 -0
  281. package/dist/web/plugin/{validate-widget.test.js → validate-view.test.js} +6 -6
  282. package/dist/web/plugin/validate-view.test.js.map +1 -0
  283. package/dist/web/proxy.js.map +1 -1
  284. package/dist/web/types.d.ts +4 -0
  285. package/dist/web/types.js.map +1 -1
  286. package/package.json +20 -8
  287. package/dist/server/templates/development.hbs +0 -12
  288. package/dist/server/templates/production.hbs +0 -6
  289. package/dist/server/widgetsDevServer.d.ts +0 -14
  290. package/dist/server/widgetsDevServer.js.map +0 -1
  291. package/dist/test/widget.test.js.map +0 -1
  292. package/dist/web/hooks/use-widget-state.d.ts +0 -4
  293. package/dist/web/hooks/use-widget-state.js +0 -32
  294. package/dist/web/hooks/use-widget-state.js.map +0 -1
  295. package/dist/web/hooks/use-widget-state.test.js.map +0 -1
  296. package/dist/web/mount-widget.d.ts +0 -1
  297. package/dist/web/mount-widget.js.map +0 -1
  298. package/dist/web/plugin/scan-widgets.d.ts +0 -8
  299. package/dist/web/plugin/scan-widgets.js +0 -68
  300. package/dist/web/plugin/scan-widgets.js.map +0 -1
  301. package/dist/web/plugin/scan-widgets.test.js +0 -96
  302. package/dist/web/plugin/scan-widgets.test.js.map +0 -1
  303. package/dist/web/plugin/validate-widget.js.map +0 -1
  304. package/dist/web/plugin/validate-widget.test.js.map +0 -1
  305. /package/dist/{test/widget.test.d.ts → cli/build-helpers.test.d.ts} +0 -0
  306. /package/dist/{web/hooks/use-widget-state.test.d.ts → cli/tunnel-control-server.test.d.ts} +0 -0
  307. /package/dist/{web/plugin/scan-widgets.test.d.ts → cli/tunnel-handler.test.d.ts} +0 -0
  308. /package/dist/{web/plugin/validate-widget.test.d.ts → cli/tunnel.test.d.ts} +0 -0
  309. /package/dist/web/plugin/{validate-widget.d.ts → validate-view.d.ts} +0 -0
@@ -2,8 +2,10 @@ import crypto from "node:crypto";
2
2
  import { readFileSync } from "node:fs";
3
3
  import http from "node:http";
4
4
  import path from "node:path";
5
+ import { Server as SdkServer, } from "@modelcontextprotocol/sdk/server/index.js";
5
6
  import { McpServer as McpServerBase } from "@modelcontextprotocol/sdk/server/mcp.js";
6
7
  import { mergeWith, union } from "es-toolkit";
8
+ import express, {} from "express";
7
9
  import { createApp } from "./express.js";
8
10
  import { createMiddlewareEntry } from "./metric.js";
9
11
  import { buildMiddlewareChain, getHandlerMaps } from "./middleware.js";
@@ -15,10 +17,16 @@ const mergeWithUnion = (target, source) => {
15
17
  }
16
18
  });
17
19
  };
18
- // ---------------------------------------------------------------------------
19
- // Content normalization
20
- // ---------------------------------------------------------------------------
20
+ /**
21
+ * Coerce a tool handler's return value into an MCP `content` array. Strings
22
+ * become a single `TextContent`; a single block is wrapped in an array;
23
+ * `undefined` produces `[]`. Mostly used internally — exported so consumers
24
+ * who build content lazily can apply the same normalization.
25
+ */
21
26
  export function normalizeContent(content) {
27
+ if (content === undefined) {
28
+ return [];
29
+ }
22
30
  if (typeof content === "string") {
23
31
  return [{ type: "text", text: content }];
24
32
  }
@@ -28,24 +36,97 @@ export function normalizeContent(content) {
28
36
  return [content];
29
37
  }
30
38
  const McpServerBaseOmitted = McpServerBase;
39
+ /**
40
+ * The Skybridge server. Extends the MCP SDK's `McpServer` with a typed tool
41
+ * registry, view resources, an embedded Express app, and protocol-level
42
+ * middleware. Construct it with the same `Implementation` info you would pass
43
+ * to the SDK, chain {@link McpServer.registerTool} calls to declare tools,
44
+ * then call {@link McpServer.run} to start the HTTP server.
45
+ *
46
+ * The `TTools` generic accumulates each registered tool's input/output/meta
47
+ * shape, so `typeof server` carries enough information for view-side helpers
48
+ * like {@link generateHelpers} to produce fully-typed hooks.
49
+ *
50
+ * @typeParam TTools - Accumulated tool registry. Filled in by `registerTool`
51
+ * chaining; you almost never set this manually.
52
+ *
53
+ * @example
54
+ * ```ts
55
+ * const server = new McpServer({ name: "my-app", version: "1.0.0" }, {})
56
+ * .registerTool({
57
+ * name: "search",
58
+ * inputSchema: { query: z.string() },
59
+ * view: { component: "search" },
60
+ * }, async ({ query }) => ({ content: `Results for ${query}` }));
61
+ *
62
+ * await server.run();
63
+ * export type AppType = typeof server;
64
+ * ```
65
+ *
66
+ * @see https://docs.skybridge.tech/api-reference/mcp-server
67
+ */
68
+ // Side channel populated by `dist/__entry.js` before user code is imported.
69
+ // Set at module scope rather than passed through the constructor because the
70
+ // wrapper has the manifest before the user's `new McpServer(...)` runs, and
71
+ // threading it through every call site (including user templates) is exactly
72
+ // the boilerplate this design is trying to hide.
73
+ let pendingBuildManifest = null;
74
+ /**
75
+ * Prime the build-time Vite manifest before user code constructs its
76
+ * `McpServer`. Called from the generated `dist/__entry.js`; not part of the
77
+ * user-facing API.
78
+ *
79
+ * @internal
80
+ */
81
+ export function __setBuildManifest(manifest) {
82
+ pendingBuildManifest = manifest;
83
+ }
31
84
  export class McpServer extends McpServerBaseOmitted {
85
+ /**
86
+ * The underlying Express app. Use this to extend the HTTP server with
87
+ * custom routes, middleware, or settings — e.g.
88
+ * `server.express.get("/health", ...)`.
89
+ *
90
+ * `express.json()` is pre-applied. Register your handlers before `run()`;
91
+ * after `run()`, dev-mode middleware, the `/mcp` route, and the default
92
+ * error handler are appended in that order.
93
+ *
94
+ * Note: Alpic Cloud only routes traffic to `/mcp` — custom routes work
95
+ * locally and on self-hosted deployments.
96
+ */
32
97
  express;
33
- customMiddleware = [];
34
98
  customErrorMiddleware = [];
35
99
  mcpMiddlewareEntries = [];
36
100
  mcpMiddlewareApplied = false;
37
- claimedWidgets = new Map();
101
+ claimedViews = new Map();
102
+ viewMetaBuilders = new Map();
103
+ viteManifest = null;
104
+ serverInfo;
105
+ serverOptions;
106
+ constructor(serverInfo, options) {
107
+ super(serverInfo, options);
108
+ this.serverInfo = serverInfo;
109
+ this.serverOptions = options;
110
+ this.express = express();
111
+ this.express.use(express.json());
112
+ // Pick up the manifest if `dist/__entry.js` primed it before importing
113
+ // user code. Consume-once: clear after the first construction so a
114
+ // subsequent test that doesn't prime can't inherit stale state.
115
+ // Explicit `setViteManifest` calls still win because they happen after
116
+ // construction.
117
+ if (pendingBuildManifest) {
118
+ this.setViteManifest(pendingBuildManifest);
119
+ pendingBuildManifest = null;
120
+ }
121
+ }
38
122
  use(pathOrHandler, ...handlers) {
123
+ // Branching is load-bearing: Express's `app.use` overloads can't be
124
+ // resolved against a `string | RequestHandler` union, so we narrow.
39
125
  if (typeof pathOrHandler === "string") {
40
- this.customMiddleware.push({
41
- path: pathOrHandler,
42
- handlers,
43
- });
126
+ this.express.use(pathOrHandler, ...handlers);
44
127
  }
45
128
  else {
46
- this.customMiddleware.push({
47
- handlers: [pathOrHandler, ...handlers],
48
- });
129
+ this.express.use(pathOrHandler, ...handlers);
49
130
  }
50
131
  return this;
51
132
  }
@@ -89,10 +170,32 @@ export class McpServer extends McpServerBaseOmitted {
89
170
  return;
90
171
  }
91
172
  this.mcpMiddlewareApplied = true;
173
+ // Surface view-resource _meta on `resources/list` (per ext-apps spec:
174
+ // hosts/checkers read CSP & domain at list time before fetching content).
175
+ const viewListMetaEntry = {
176
+ filter: "resources/list",
177
+ handler: async (_req, extra, next) => {
178
+ const result = (await next());
179
+ for (const resource of result.resources) {
180
+ const builder = this.viewMetaBuilders.get(resource.uri);
181
+ if (!builder) {
182
+ continue;
183
+ }
184
+ const meta = builder(extra);
185
+ resource._meta = {
186
+ ...(resource._meta ?? {}),
187
+ ...meta,
188
+ };
189
+ }
190
+ return result;
191
+ },
192
+ };
92
193
  const monitoringEntry = createMiddlewareEntry();
93
- const entries = monitoringEntry
94
- ? [monitoringEntry, ...this.mcpMiddlewareEntries]
95
- : this.mcpMiddlewareEntries;
194
+ const entries = [
195
+ ...(monitoringEntry ? [monitoringEntry] : []),
196
+ viewListMetaEntry,
197
+ ...this.mcpMiddlewareEntries,
198
+ ];
96
199
  if (entries.length === 0) {
97
200
  return;
98
201
  }
@@ -107,53 +210,171 @@ export class McpServer extends McpServerBaseOmitted {
107
210
  instrumentMap(requestHandlers, false);
108
211
  instrumentMap(notificationHandlers, true);
109
212
  }
213
+ /**
214
+ * Connect to an MCP transport (override of the SDK's `connect`). Use this
215
+ * when you're embedding Skybridge in a host that already manages its own
216
+ * transport (e.g. stdio for desktop apps); for HTTP, prefer {@link McpServer.run}
217
+ * which sets the transport up for you. Locks in any middleware registered
218
+ * via {@link McpServer.mcpMiddleware} — further calls to that method will
219
+ * throw afterwards.
220
+ */
110
221
  async connect(transport) {
111
222
  this.applyMcpMiddleware();
112
223
  return McpServerBase.prototype.connect.call(this, transport);
113
224
  }
225
+ /**
226
+ * Per-request stateless connect. The SDK's `Protocol` only allows one
227
+ * transport per instance, so we can't reuse this `McpServer` across
228
+ * concurrent requests. The SDK's idiomatic fix is a `() => McpServer`
229
+ * factory, but that would break Skybridge's singleton API — so instead
230
+ * we build a fresh underlying `Server` per request and share the main
231
+ * server's handler maps by reference. The cast is unavoidable: there's
232
+ * no public API to inject handler maps. `getHandlerMaps` validates the
233
+ * read side and fails fast on SDK field renames.
234
+ */
235
+ async connectStatelessTransport(transport) {
236
+ this.applyMcpMiddleware();
237
+ const { requestHandlers, notificationHandlers } = getHandlerMaps(this.server);
238
+ const fresh = new SdkServer(this.serverInfo, this.serverOptions);
239
+ const target = fresh;
240
+ target._requestHandlers = requestHandlers;
241
+ target._notificationHandlers = notificationHandlers;
242
+ await fresh.connect(transport);
243
+ }
244
+ /**
245
+ * Start the HTTP server. Listens on `process.env.__PORT` (default `3000`),
246
+ * mounts the `/mcp` route, applies any custom Express middleware registered
247
+ * via {@link McpServer.use} / {@link McpServer.useOnError}, and locks in
248
+ * any MCP middleware registered via {@link McpServer.mcpMiddleware}.
249
+ *
250
+ * On Cloudflare Workers / workerd, returns an object exposing `fetch` so
251
+ * the runtime can bridge incoming requests to the Node HTTP server. On
252
+ * Vercel (`VERCEL === "1"`), returns the Express app directly so the
253
+ * serverless function entry can call it as a `(req, res)` handler. On
254
+ * Node, returns `undefined` once listening.
255
+ */
114
256
  async run() {
115
257
  this.applyMcpMiddleware();
116
- const httpServer = http.createServer();
117
- if (!this.express) {
118
- this.express = await createApp({
258
+ if (process.env.VERCEL === "1") {
259
+ // createApp only reads httpServer inside its dev-only branch
260
+ // (viewsDevServer); under VERCEL=1 + NODE_ENV=production it's a
261
+ // bare object passed to satisfy the required parameter.
262
+ const httpServer = http.createServer();
263
+ await createApp({
119
264
  mcpServer: this,
120
265
  httpServer,
121
- customMiddleware: this.customMiddleware,
122
266
  errorMiddleware: this.customErrorMiddleware,
123
267
  });
268
+ return this.express;
124
269
  }
270
+ const httpServer = http.createServer();
271
+ await createApp({
272
+ mcpServer: this,
273
+ httpServer,
274
+ errorMiddleware: this.customErrorMiddleware,
275
+ });
125
276
  httpServer.on("request", this.express);
126
- return new Promise((resolve, reject) => {
277
+ const port = parseInt(process.env.__PORT ?? "3000", 10);
278
+ await new Promise((resolve, reject) => {
127
279
  httpServer.on("error", (error) => {
128
280
  console.error("Failed to start server:", error);
129
281
  reject(error);
130
282
  });
131
- const port = parseInt(process.env.__PORT ?? "3000", 10);
132
283
  httpServer.listen(port, () => {
133
284
  resolve();
134
285
  });
135
286
  });
287
+ // On workerd, bridge the Node http server to a Workers fetch handler.
288
+ // The specifier is held in a variable to sidestep tsc's module resolution
289
+ // (`cloudflare:node` only exists under wrangler/workerd).
290
+ if (typeof navigator !== "undefined" &&
291
+ navigator.userAgent === "Cloudflare-Workers") {
292
+ const cloudflareNode = "cloudflare:node";
293
+ const { httpServerHandler } = await import(cloudflareNode);
294
+ return httpServerHandler({ port });
295
+ }
296
+ const shutdown = () => {
297
+ // Drop both handlers so a second signal falls through to Node's default
298
+ // (force-quit on a second Ctrl+C while drain is hanging).
299
+ process.off("SIGTERM", shutdown);
300
+ process.off("SIGINT", shutdown);
301
+ httpServer.close(() => process.exit(0));
302
+ // Force exit if connections don't drain in time so the port is still
303
+ // released promptly (e.g. for nodemon restarts).
304
+ setTimeout(() => process.exit(0), 3000).unref();
305
+ };
306
+ process.on("SIGTERM", shutdown);
307
+ process.on("SIGINT", shutdown);
308
+ return undefined;
136
309
  }
137
- // registerTool is defined via interface merging + prototype assignment below
138
- // to avoid TypeScript override signature incompatibility with the base class.
139
- // -------------------------------------------------------------------------
140
- // View (widget) resource registration
141
- // -------------------------------------------------------------------------
142
- /** @internal */
143
- enforceOneToolPerWidget(component, toolName) {
144
- const existingTool = this.claimedWidgets.get(component);
310
+ enforceOneToolPerView(component, toolName) {
311
+ const existingTool = this.claimedViews.get(component);
145
312
  if (existingTool) {
146
- throw new Error(`skybridge: widget "${component}" is already used by tool "${existingTool}". Tool "${toolName}" cannot also reference it — each widget backs exactly one tool.`);
313
+ throw new Error(`skybridge: view "${component}" is already used by tool "${existingTool}". Tool "${toolName}" cannot also reference it — each view backs exactly one tool.`);
314
+ }
315
+ this.claimedViews.set(component, toolName);
316
+ }
317
+ resolveViewRequestContext(extra) {
318
+ const isProduction = process.env.NODE_ENV === "production";
319
+ const headers = extra?.requestInfo?.headers || {};
320
+ const header = (key) => {
321
+ const val = headers[key];
322
+ return Array.isArray(val) ? val[0] : val;
323
+ };
324
+ const isClaude = header("user-agent") === "Claude-User";
325
+ let serverUrl;
326
+ const forwardedHost = header("x-forwarded-host");
327
+ const origin = header("origin");
328
+ const host = header("host");
329
+ if (forwardedHost) {
330
+ const proto = header("x-forwarded-proto") || "https";
331
+ serverUrl = `${proto}://${forwardedHost}`;
332
+ }
333
+ else if (origin) {
334
+ serverUrl = origin;
335
+ }
336
+ else if (host) {
337
+ const proto = ["127.0.0.1:", "localhost:"].some((p) => host.startsWith(p))
338
+ ? "http"
339
+ : "https";
340
+ serverUrl = `${proto}://${host}`;
341
+ }
342
+ else {
343
+ const devPort = process.env.__PORT || "3000";
344
+ serverUrl = `http://localhost:${devPort}`;
345
+ }
346
+ const connectDomains = [serverUrl];
347
+ if (!isProduction) {
348
+ const wsUrl = new URL(serverUrl);
349
+ wsUrl.protocol = wsUrl.protocol === "https:" ? "wss:" : "ws:";
350
+ connectDomains.push(wsUrl.origin);
351
+ }
352
+ let contentMetaOverrides = {};
353
+ if (isClaude) {
354
+ const pathname = extra?.requestInfo?.url?.pathname ?? "";
355
+ const rawUrl = header("x-alpic-forwarded-url") ?? `${serverUrl}${pathname}`;
356
+ // Strip a lone trailing slash so the hash matches the connector URL
357
+ // as registered with Claude (which has no trailing slash on bare origins).
358
+ const url = rawUrl.endsWith("/") ? rawUrl.slice(0, -1) : rawUrl;
359
+ const hash = crypto
360
+ .createHash("sha256")
361
+ .update(url)
362
+ .digest("hex")
363
+ .slice(0, 32);
364
+ contentMetaOverrides = { domain: `${hash}.claudemcpcontent.com` };
147
365
  }
148
- this.claimedWidgets.set(component, toolName);
366
+ return { serverUrl, connectDomains, contentMetaOverrides };
149
367
  }
150
- /** @internal */
151
368
  registerViewResources(toolName, view, toolMeta) {
152
369
  const hosts = view.hosts ?? ["apps-sdk", "mcp-app"];
370
+ // Append a content-derived version param so hosts (e.g. ChatGPT) bust
371
+ // their cache when the bundle changes, but keep the URI stable across
372
+ // `tools/list` calls when the bundle hasn't changed.
373
+ const versionParam = this.computeViewVersionParam(view.component);
153
374
  if (hosts.includes("apps-sdk")) {
154
- const widgetConfig = {
375
+ const viewResource = {
155
376
  hostType: "apps-sdk",
156
- uri: `ui://widgets/apps-sdk/${view.component}.html`,
377
+ uri: `ui://views/apps-sdk/${view.component}.html${versionParam}`,
157
378
  mimeType: "text/html+skybridge",
158
379
  buildContentMeta: ({ resourceDomains, connectDomains, domain }, overrides) => {
159
380
  const defaults = {
@@ -183,17 +404,17 @@ export class McpServer extends McpServerBaseOmitted {
183
404
  return base;
184
405
  },
185
406
  };
186
- this.registerWidgetResource({
407
+ this.registerViewResource({
187
408
  name: toolName,
188
- widgetConfig,
409
+ viewResource,
189
410
  view,
190
411
  });
191
- toolMeta["openai/outputTemplate"] = widgetConfig.uri;
412
+ toolMeta["openai/outputTemplate"] = viewResource.uri;
192
413
  }
193
414
  if (hosts.includes("mcp-app")) {
194
- const widgetConfig = {
415
+ const viewResource = {
195
416
  hostType: "mcp-app",
196
- uri: `ui://widgets/ext-apps/${view.component}.html`,
417
+ uri: `ui://views/ext-apps/${view.component}.html${versionParam}`,
197
418
  mimeType: "text/html;profile=mcp-app",
198
419
  buildContentMeta: ({ resourceDomains, connectDomains, domain, baseUriDomains }, overrides) => {
199
420
  const defaults = {
@@ -241,96 +462,50 @@ export class McpServer extends McpServerBaseOmitted {
241
462
  return base;
242
463
  },
243
464
  };
244
- this.registerWidgetResource({
465
+ this.registerViewResource({
245
466
  name: toolName,
246
- widgetConfig,
467
+ viewResource,
247
468
  view,
248
469
  });
249
470
  // @ts-expect-error - For backwards compatibility with Claude current implementation of the specs
250
- toolMeta["ui/resourceUri"] = widgetConfig.uri;
251
- toolMeta.ui = { resourceUri: widgetConfig.uri };
471
+ toolMeta["ui/resourceUri"] = viewResource.uri;
472
+ toolMeta.ui = { ...toolMeta.ui, resourceUri: viewResource.uri };
252
473
  }
253
474
  }
254
- registerWidgetResource({ name, widgetConfig, view, }) {
255
- const { hostType, uri: widgetUri, mimeType, buildContentMeta, } = widgetConfig;
256
- this.registerResource(name, widgetUri, { description: view.description }, async (uri, extra) => {
475
+ registerViewResource({ name, viewResource, view, }) {
476
+ const { hostType, uri: viewUri, mimeType, buildContentMeta } = viewResource;
477
+ const buildMeta = (extra) => {
478
+ const { serverUrl, connectDomains, contentMetaOverrides } = this.resolveViewRequestContext(extra);
479
+ return buildContentMeta({
480
+ resourceDomains: [serverUrl],
481
+ connectDomains,
482
+ domain: serverUrl,
483
+ baseUriDomains: [serverUrl],
484
+ }, contentMetaOverrides);
485
+ };
486
+ this.viewMetaBuilders.set(viewUri, buildMeta);
487
+ this.registerResource(name, viewUri, { description: view.description }, async (uri, extra) => {
257
488
  const isProduction = process.env.NODE_ENV === "production";
258
- const isClaude = extra?.requestInfo?.headers?.["user-agent"] === "Claude-User";
259
- const headers = extra?.requestInfo?.headers || {};
260
- const header = (key) => {
261
- const val = headers[key];
262
- return Array.isArray(val) ? val[0] : val;
263
- };
264
- let serverUrl;
265
- const forwardedHost = header("x-forwarded-host");
266
- const origin = header("origin");
267
- const host = header("host");
268
- if (forwardedHost) {
269
- const proto = header("x-forwarded-proto") || "https";
270
- serverUrl = `${proto}://${forwardedHost}`;
271
- }
272
- else if (origin) {
273
- serverUrl = origin;
274
- }
275
- else if (host) {
276
- const proto = ["127.0.0.1:", "localhost:"].some((p) => host.startsWith(p))
277
- ? "http"
278
- : "https";
279
- serverUrl = `${proto}://${host}`;
280
- }
281
- else {
282
- const devPort = process.env.__PORT || "3000";
283
- serverUrl = `http://localhost:${devPort}`;
284
- }
489
+ const { serverUrl } = this.resolveViewRequestContext(extra);
285
490
  const html = isProduction
286
491
  ? templateHelper.renderProduction({
287
492
  hostType,
288
493
  serverUrl,
289
- widgetFile: this.lookupWidgetFile(view.component),
494
+ viewFile: this.lookupViewFile(view.component),
290
495
  styleFile: this.lookupDistFile("style.css") ?? "",
291
496
  })
292
497
  : templateHelper.renderDevelopment({
293
498
  hostType,
294
499
  serverUrl,
295
- widgetName: view.component,
500
+ viewName: view.component,
296
501
  });
297
- const connectDomains = [serverUrl];
298
- if (!isProduction) {
299
- const wsUrl = new URL(serverUrl);
300
- wsUrl.protocol = wsUrl.protocol === "https:" ? "wss:" : "ws:";
301
- connectDomains.push(wsUrl.origin);
302
- }
303
- let contentMetaOverrides = {};
304
- if (isClaude) {
305
- const pathname = extra?.requestInfo?.url?.pathname ?? "";
306
- const rawUrl = header("x-alpic-forwarded-url") ?? `${serverUrl}${pathname}`;
307
- // Strip a lone trailing slash so the hash matches the connector URL
308
- // as registered with Claude (which has no trailing slash on bare origins).
309
- const url = rawUrl.endsWith("/") ? rawUrl.slice(0, -1) : rawUrl;
310
- const hash = crypto
311
- .createHash("sha256")
312
- .update(url)
313
- .digest("hex")
314
- .slice(0, 32);
315
- contentMetaOverrides = { domain: `${hash}.claudemcpcontent.com` };
316
- }
317
- const contentMeta = buildContentMeta({
318
- resourceDomains: [serverUrl],
319
- connectDomains,
320
- domain: serverUrl,
321
- baseUriDomains: [serverUrl],
322
- }, contentMetaOverrides);
323
502
  return {
324
503
  contents: [
325
- { uri: uri.href, mimeType, text: html, _meta: contentMeta },
504
+ { uri: uri.href, mimeType, text: html, _meta: buildMeta(extra) },
326
505
  ],
327
506
  };
328
507
  });
329
508
  }
330
- // -------------------------------------------------------------------------
331
- // Handler wrapping
332
- // -------------------------------------------------------------------------
333
- /** @internal */
334
509
  wrapHandler(cb, { attachViewUUID }) {
335
510
  return async (args, extra) => {
336
511
  const result = await cb(args, extra);
@@ -346,23 +521,53 @@ export class McpServer extends McpServerBaseOmitted {
346
521
  };
347
522
  };
348
523
  }
349
- // -------------------------------------------------------------------------
350
- // Manifest helpers
351
- // -------------------------------------------------------------------------
352
- lookupWidgetFile(widgetName) {
524
+ computeViewVersionParam(viewName) {
525
+ if (process.env.NODE_ENV !== "production") {
526
+ return "";
527
+ }
528
+ try {
529
+ const viewFile = this.lookupViewFile(viewName);
530
+ const styleFile = this.lookupDistFile("style.css") ?? "";
531
+ const hash = crypto
532
+ .createHash("sha256")
533
+ .update(viewFile)
534
+ .update("\0")
535
+ .update(styleFile)
536
+ .digest("hex")
537
+ .slice(0, 8);
538
+ return `?v=${hash}`;
539
+ }
540
+ catch {
541
+ return "";
542
+ }
543
+ }
544
+ lookupViewFile(viewName) {
353
545
  const manifest = this.readManifest();
354
546
  for (const entry of Object.values(manifest)) {
355
- if (entry?.isEntry && entry.name === widgetName && entry.file) {
547
+ if (entry?.isEntry && entry.name === viewName && entry.file) {
356
548
  return entry.file;
357
549
  }
358
550
  }
359
- throw new Error(`Widget "${widgetName}" not found in Vite manifest. Did the build complete successfully? Look for an entry with name "${widgetName}" in dist/assets/.vite/manifest.json.`);
551
+ throw new Error(`View "${viewName}" not found in Vite manifest. Did the build complete successfully? Look for an entry with name "${viewName}" in dist/assets/.vite/manifest.json.`);
360
552
  }
361
553
  lookupDistFile(key) {
362
554
  const manifest = this.readManifest();
363
555
  return manifest[key]?.file;
364
556
  }
557
+ /**
558
+ * Inject the Vite manifest as a value rather than letting `readManifest()`
559
+ * load it from disk. Required for runtimes without a usable filesystem
560
+ * (Cloudflare Workers, etc.) — the user's `skybridge build` emits the
561
+ * manifest as a JS module which the entry imports and passes here.
562
+ */
563
+ setViteManifest(manifest) {
564
+ this.viteManifest = manifest;
565
+ return this;
566
+ }
365
567
  readManifest() {
568
+ if (this.viteManifest) {
569
+ return this.viteManifest;
570
+ }
366
571
  return JSON.parse(readFileSync(path.join(process.cwd(), "dist", "assets", ".vite", "manifest.json"), "utf-8"));
367
572
  }
368
573
  registerTool(...args) {
@@ -373,10 +578,18 @@ export class McpServer extends McpServerBaseOmitted {
373
578
  }
374
579
  const config = args[0];
375
580
  const cb = args[1];
376
- const { name, view, _meta: userToolMeta, ...toolFields } = config;
581
+ const { name, view, securitySchemes, _meta: userToolMeta, ...toolFields } = config;
377
582
  const toolMeta = { ...userToolMeta };
583
+ if (securitySchemes) {
584
+ // SEP-1488 puts `securitySchemes` at the top level of the tool
585
+ // descriptor, but the SDK's `registerTool` drops unknown top-level
586
+ // fields, so the canonical spot isn't reachable without intercepting
587
+ // `tools/list`. Use the `_meta` back-compat mirror documented in the
588
+ // Apps SDK reference until SEP-1488 lands in the spec.
589
+ toolMeta.securitySchemes = securitySchemes;
590
+ }
378
591
  if (view) {
379
- this.enforceOneToolPerWidget(view.component, name);
592
+ this.enforceOneToolPerView(view.component, name);
380
593
  this.registerViewResources(name, view, toolMeta);
381
594
  }
382
595
  const wrappedCb = this.wrapHandler(cb, { attachViewUUID: Boolean(view) });