skybridge 0.0.0-dev.e93cc2e → 0.0.0-dev.e9b8e2a

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 (271) hide show
  1. package/README.md +15 -11
  2. package/dist/cli/detect-port.js.map +1 -1
  3. package/dist/cli/header.js +1 -1
  4. package/dist/cli/header.js.map +1 -1
  5. package/dist/cli/run-command.js.map +1 -1
  6. package/dist/cli/telemetry.js.map +1 -1
  7. package/dist/cli/tunnel-control-server.d.ts +9 -0
  8. package/dist/cli/tunnel-control-server.js +31 -0
  9. package/dist/cli/tunnel-control-server.js.map +1 -0
  10. package/dist/cli/tunnel-control-server.test.js +39 -0
  11. package/dist/cli/tunnel-control-server.test.js.map +1 -0
  12. package/dist/cli/tunnel-handler.d.ts +3 -0
  13. package/dist/cli/tunnel-handler.js +48 -0
  14. package/dist/cli/tunnel-handler.js.map +1 -0
  15. package/dist/cli/tunnel-handler.test.js +105 -0
  16. package/dist/cli/tunnel-handler.test.js.map +1 -0
  17. package/dist/cli/tunnel.d.ts +57 -0
  18. package/dist/cli/tunnel.js +154 -0
  19. package/dist/cli/tunnel.js.map +1 -0
  20. package/dist/cli/tunnel.test.js +190 -0
  21. package/dist/cli/tunnel.test.js.map +1 -0
  22. package/dist/cli/types.d.ts +5 -0
  23. package/dist/cli/types.js +2 -0
  24. package/dist/cli/types.js.map +1 -0
  25. package/dist/cli/use-execute-steps.js.map +1 -1
  26. package/dist/cli/use-messages.d.ts +3 -0
  27. package/dist/cli/use-messages.js +11 -0
  28. package/dist/cli/use-messages.js.map +1 -0
  29. package/dist/cli/use-nodemon.d.ts +2 -7
  30. package/dist/cli/use-nodemon.js +18 -21
  31. package/dist/cli/use-nodemon.js.map +1 -1
  32. package/dist/cli/use-open-browser.d.ts +1 -0
  33. package/dist/cli/use-open-browser.js +44 -0
  34. package/dist/cli/use-open-browser.js.map +1 -0
  35. package/dist/cli/use-tunnel.d.ts +14 -0
  36. package/dist/cli/use-tunnel.js +131 -0
  37. package/dist/cli/use-tunnel.js.map +1 -0
  38. package/dist/cli/use-typescript-check.d.ts +1 -0
  39. package/dist/cli/use-typescript-check.js +41 -6
  40. package/dist/cli/use-typescript-check.js.map +1 -1
  41. package/dist/commands/build.js +60 -13
  42. package/dist/commands/build.js.map +1 -1
  43. package/dist/commands/dev.d.ts +3 -1
  44. package/dist/commands/dev.js +46 -8
  45. package/dist/commands/dev.js.map +1 -1
  46. package/dist/commands/start.js +7 -10
  47. package/dist/commands/start.js.map +1 -1
  48. package/dist/commands/telemetry/disable.js.map +1 -1
  49. package/dist/commands/telemetry/enable.js.map +1 -1
  50. package/dist/commands/telemetry/status.js.map +1 -1
  51. package/dist/server/asset-base-url-transform-plugin.d.ts +5 -6
  52. package/dist/server/asset-base-url-transform-plugin.js +9 -10
  53. package/dist/server/asset-base-url-transform-plugin.js.map +1 -1
  54. package/dist/server/asset-base-url-transform-plugin.test.js +41 -13
  55. package/dist/server/asset-base-url-transform-plugin.test.js.map +1 -1
  56. package/dist/server/content-helpers.d.ts +27 -0
  57. package/dist/server/content-helpers.js +46 -0
  58. package/dist/server/content-helpers.js.map +1 -0
  59. package/dist/server/content-helpers.test.d.ts +1 -0
  60. package/dist/server/content-helpers.test.js +70 -0
  61. package/dist/server/content-helpers.test.js.map +1 -0
  62. package/dist/server/express.d.ts +7 -5
  63. package/dist/server/express.js +53 -26
  64. package/dist/server/express.js.map +1 -1
  65. package/dist/server/express.test.js +381 -25
  66. package/dist/server/express.test.js.map +1 -1
  67. package/dist/server/file-ref.d.ts +8 -0
  68. package/dist/server/file-ref.js +8 -0
  69. package/dist/server/file-ref.js.map +1 -0
  70. package/dist/server/index.d.ts +6 -3
  71. package/dist/server/index.js +4 -1
  72. package/dist/server/index.js.map +1 -1
  73. package/dist/server/inferUtilityTypes.d.ts +6 -6
  74. package/dist/server/inferUtilityTypes.js.map +1 -1
  75. package/dist/server/metric.d.ts +14 -0
  76. package/dist/server/metric.js +62 -0
  77. package/dist/server/metric.js.map +1 -0
  78. package/dist/server/middleware.d.ts +32 -4
  79. package/dist/server/middleware.js.map +1 -1
  80. package/dist/server/middleware.test-d.js +41 -18
  81. package/dist/server/middleware.test-d.js.map +1 -1
  82. package/dist/server/middleware.test.js +115 -5
  83. package/dist/server/middleware.test.js.map +1 -1
  84. package/dist/server/server.d.ts +133 -82
  85. package/dist/server/server.js +305 -112
  86. package/dist/server/server.js.map +1 -1
  87. package/dist/server/templateHelper.d.ts +5 -6
  88. package/dist/server/templateHelper.js +1 -2
  89. package/dist/server/templateHelper.js.map +1 -1
  90. package/dist/server/templates.generated.d.ts +4 -0
  91. package/dist/server/templates.generated.js +47 -0
  92. package/dist/server/templates.generated.js.map +1 -0
  93. package/dist/server/tunnel-proxy-router.d.ts +7 -0
  94. package/dist/server/tunnel-proxy-router.js +110 -0
  95. package/dist/server/tunnel-proxy-router.js.map +1 -0
  96. package/dist/server/tunnel-proxy-router.test.d.ts +1 -0
  97. package/dist/server/tunnel-proxy-router.test.js +229 -0
  98. package/dist/server/tunnel-proxy-router.test.js.map +1 -0
  99. package/dist/server/viewsDevServer.d.ts +14 -0
  100. package/dist/server/viewsDevServer.js +45 -0
  101. package/dist/server/viewsDevServer.js.map +1 -0
  102. package/dist/test/utils.d.ts +13 -21
  103. package/dist/test/utils.js +42 -37
  104. package/dist/test/utils.js.map +1 -1
  105. package/dist/test/view.test.d.ts +1 -0
  106. package/dist/test/view.test.js +523 -0
  107. package/dist/test/view.test.js.map +1 -0
  108. package/dist/version.d.ts +1 -0
  109. package/dist/version.js +3 -0
  110. package/dist/version.js.map +1 -0
  111. package/dist/web/bridges/apps-sdk/adaptor.d.ts +8 -4
  112. package/dist/web/bridges/apps-sdk/adaptor.js +43 -16
  113. package/dist/web/bridges/apps-sdk/adaptor.js.map +1 -1
  114. package/dist/web/bridges/apps-sdk/bridge.js.map +1 -1
  115. package/dist/web/bridges/apps-sdk/index.js.map +1 -1
  116. package/dist/web/bridges/apps-sdk/types.d.ts +18 -6
  117. package/dist/web/bridges/apps-sdk/types.js.map +1 -1
  118. package/dist/web/bridges/apps-sdk/use-apps-sdk-context.js.map +1 -1
  119. package/dist/web/bridges/get-adaptor.js.map +1 -1
  120. package/dist/web/bridges/index.js.map +1 -1
  121. package/dist/web/bridges/mcp-app/adaptor.d.ts +22 -8
  122. package/dist/web/bridges/mcp-app/adaptor.js +143 -62
  123. package/dist/web/bridges/mcp-app/adaptor.js.map +1 -1
  124. package/dist/web/bridges/mcp-app/bridge.d.ts +13 -30
  125. package/dist/web/bridges/mcp-app/bridge.js +43 -201
  126. package/dist/web/bridges/mcp-app/bridge.js.map +1 -1
  127. package/dist/web/bridges/mcp-app/index.js.map +1 -1
  128. package/dist/web/bridges/mcp-app/types.js.map +1 -1
  129. package/dist/web/bridges/mcp-app/use-mcp-app-context.d.ts +5 -3
  130. package/dist/web/bridges/mcp-app/use-mcp-app-context.js +2 -2
  131. package/dist/web/bridges/mcp-app/use-mcp-app-context.js.map +1 -1
  132. package/dist/web/bridges/mcp-app/use-mcp-app-context.test.js +1 -41
  133. package/dist/web/bridges/mcp-app/use-mcp-app-context.test.js.map +1 -1
  134. package/dist/web/bridges/types.d.ts +25 -10
  135. package/dist/web/bridges/types.js.map +1 -1
  136. package/dist/web/bridges/use-host-context.js.map +1 -1
  137. package/dist/web/components/modal-provider.js +3 -5
  138. package/dist/web/components/modal-provider.js.map +1 -1
  139. package/dist/web/create-store.js +17 -3
  140. package/dist/web/create-store.js.map +1 -1
  141. package/dist/web/create-store.test.js +17 -17
  142. package/dist/web/create-store.test.js.map +1 -1
  143. package/dist/web/data-llm.d.ts +1 -1
  144. package/dist/web/data-llm.js +3 -3
  145. package/dist/web/data-llm.js.map +1 -1
  146. package/dist/web/data-llm.test.js +23 -22
  147. package/dist/web/data-llm.test.js.map +1 -1
  148. package/dist/web/generate-helpers.d.ts +20 -18
  149. package/dist/web/generate-helpers.js +20 -18
  150. package/dist/web/generate-helpers.js.map +1 -1
  151. package/dist/web/generate-helpers.test-d.js +26 -26
  152. package/dist/web/generate-helpers.test-d.js.map +1 -1
  153. package/dist/web/generate-helpers.test.js.map +1 -1
  154. package/dist/web/helpers/state.d.ts +2 -2
  155. package/dist/web/helpers/state.js +11 -11
  156. package/dist/web/helpers/state.js.map +1 -1
  157. package/dist/web/helpers/state.test.js +9 -9
  158. package/dist/web/helpers/state.test.js.map +1 -1
  159. package/dist/web/hooks/index.d.ts +3 -1
  160. package/dist/web/hooks/index.js +3 -1
  161. package/dist/web/hooks/index.js.map +1 -1
  162. package/dist/web/hooks/test/utils.js +4 -0
  163. package/dist/web/hooks/test/utils.js.map +1 -1
  164. package/dist/web/hooks/use-call-tool.js.map +1 -1
  165. package/dist/web/hooks/use-call-tool.test-d.js.map +1 -1
  166. package/dist/web/hooks/use-call-tool.test.js +0 -4
  167. package/dist/web/hooks/use-call-tool.test.js.map +1 -1
  168. package/dist/web/hooks/use-display-mode.js.map +1 -1
  169. package/dist/web/hooks/use-display-mode.test-d.js.map +1 -1
  170. package/dist/web/hooks/use-display-mode.test.js.map +1 -1
  171. package/dist/web/hooks/use-files.d.ts +2 -1
  172. package/dist/web/hooks/use-files.js +1 -0
  173. package/dist/web/hooks/use-files.js.map +1 -1
  174. package/dist/web/hooks/use-files.test.js +22 -2
  175. package/dist/web/hooks/use-files.test.js.map +1 -1
  176. package/dist/web/hooks/use-layout.js.map +1 -1
  177. package/dist/web/hooks/use-layout.test.js +3 -3
  178. package/dist/web/hooks/use-layout.test.js.map +1 -1
  179. package/dist/web/hooks/use-open-external.js.map +1 -1
  180. package/dist/web/hooks/use-open-external.test.js +15 -10
  181. package/dist/web/hooks/use-open-external.test.js.map +1 -1
  182. package/dist/web/hooks/use-request-close.d.ts +2 -0
  183. package/dist/web/hooks/use-request-close.js +8 -0
  184. package/dist/web/hooks/use-request-close.js.map +1 -0
  185. package/dist/web/hooks/use-request-close.test.d.ts +1 -0
  186. package/dist/web/hooks/use-request-close.test.js +52 -0
  187. package/dist/web/hooks/use-request-close.test.js.map +1 -0
  188. package/dist/web/hooks/use-request-modal.d.ts +1 -1
  189. package/dist/web/hooks/use-request-modal.js +4 -4
  190. package/dist/web/hooks/use-request-modal.js.map +1 -1
  191. package/dist/web/hooks/use-request-modal.test.js +5 -1
  192. package/dist/web/hooks/use-request-modal.test.js.map +1 -1
  193. package/dist/web/hooks/use-request-size.d.ts +3 -0
  194. package/dist/web/hooks/use-request-size.js +8 -0
  195. package/dist/web/hooks/use-request-size.js.map +1 -0
  196. package/dist/web/hooks/use-request-size.test.d.ts +1 -0
  197. package/dist/web/hooks/use-request-size.test.js +65 -0
  198. package/dist/web/hooks/use-request-size.test.js.map +1 -0
  199. package/dist/web/hooks/use-send-follow-up-message.d.ts +2 -1
  200. package/dist/web/hooks/use-send-follow-up-message.js +2 -2
  201. package/dist/web/hooks/use-send-follow-up-message.js.map +1 -1
  202. package/dist/web/hooks/use-set-open-in-app-url.js.map +1 -1
  203. package/dist/web/hooks/use-set-open-in-app-url.test.js.map +1 -1
  204. package/dist/web/hooks/use-tool-info.js.map +1 -1
  205. package/dist/web/hooks/use-tool-info.test-d.js.map +1 -1
  206. package/dist/web/hooks/use-tool-info.test.js +1 -1
  207. package/dist/web/hooks/use-tool-info.test.js.map +1 -1
  208. package/dist/web/hooks/use-user.js +18 -2
  209. package/dist/web/hooks/use-user.js.map +1 -1
  210. package/dist/web/hooks/use-user.test.js +29 -1
  211. package/dist/web/hooks/use-user.test.js.map +1 -1
  212. package/dist/web/hooks/use-view-state.d.ts +4 -0
  213. package/dist/web/hooks/use-view-state.js +32 -0
  214. package/dist/web/hooks/use-view-state.js.map +1 -0
  215. package/dist/web/hooks/use-view-state.test.d.ts +1 -0
  216. package/dist/web/hooks/use-view-state.test.js +177 -0
  217. package/dist/web/hooks/use-view-state.test.js.map +1 -0
  218. package/dist/web/index.d.ts +1 -2
  219. package/dist/web/index.js +1 -2
  220. package/dist/web/index.js.map +1 -1
  221. package/dist/web/mount-view.d.ts +1 -0
  222. package/dist/web/{mount-widget.js → mount-view.js} +2 -2
  223. package/dist/web/mount-view.js.map +1 -0
  224. package/dist/web/plugin/data-llm.test.js.map +1 -1
  225. package/dist/web/plugin/plugin.d.ts +4 -1
  226. package/dist/web/plugin/plugin.js +134 -25
  227. package/dist/web/plugin/plugin.js.map +1 -1
  228. package/dist/web/plugin/scan-views.d.ts +16 -0
  229. package/dist/web/plugin/scan-views.js +88 -0
  230. package/dist/web/plugin/scan-views.js.map +1 -0
  231. package/dist/web/plugin/scan-views.test.d.ts +1 -0
  232. package/dist/web/plugin/scan-views.test.js +99 -0
  233. package/dist/web/plugin/scan-views.test.js.map +1 -0
  234. package/dist/web/plugin/transform-data-llm.js +1 -1
  235. package/dist/web/plugin/transform-data-llm.js.map +1 -1
  236. package/dist/web/plugin/transform-data-llm.test.js.map +1 -1
  237. package/dist/web/plugin/validate-view.d.ts +1 -0
  238. package/dist/web/plugin/validate-view.js +9 -0
  239. package/dist/web/plugin/validate-view.js.map +1 -0
  240. package/dist/web/plugin/validate-view.test.d.ts +1 -0
  241. package/dist/web/plugin/validate-view.test.js +24 -0
  242. package/dist/web/plugin/validate-view.test.js.map +1 -0
  243. package/dist/web/proxy.js.map +1 -1
  244. package/dist/web/types.js.map +1 -1
  245. package/package.json +33 -23
  246. package/tsconfig.base.json +5 -0
  247. package/dist/server/const.d.ts +0 -1
  248. package/dist/server/const.js +0 -2
  249. package/dist/server/const.js.map +0 -1
  250. package/dist/server/templates/development.js +0 -27
  251. package/dist/server/templates/production.js +0 -23
  252. package/dist/server/widgetsDevServer.d.ts +0 -12
  253. package/dist/server/widgetsDevServer.js +0 -63
  254. package/dist/server/widgetsDevServer.js.map +0 -1
  255. package/dist/test/widget.test.js +0 -261
  256. package/dist/test/widget.test.js.map +0 -1
  257. package/dist/web/hooks/use-widget-state.d.ts +0 -4
  258. package/dist/web/hooks/use-widget-state.js +0 -32
  259. package/dist/web/hooks/use-widget-state.js.map +0 -1
  260. package/dist/web/hooks/use-widget-state.test.js +0 -64
  261. package/dist/web/hooks/use-widget-state.test.js.map +0 -1
  262. package/dist/web/mount-widget.d.ts +0 -1
  263. package/dist/web/mount-widget.js.map +0 -1
  264. package/dist/web/plugin/validate-widget.d.ts +0 -5
  265. package/dist/web/plugin/validate-widget.js +0 -27
  266. package/dist/web/plugin/validate-widget.js.map +0 -1
  267. package/dist/web/plugin/validate-widget.test.js +0 -42
  268. package/dist/web/plugin/validate-widget.test.js.map +0 -1
  269. /package/dist/{test/widget.test.d.ts → cli/tunnel-control-server.test.d.ts} +0 -0
  270. /package/dist/{web/hooks/use-widget-state.test.d.ts → cli/tunnel-handler.test.d.ts} +0 -0
  271. /package/dist/{web/plugin/validate-widget.test.d.ts → cli/tunnel.test.d.ts} +0 -0
@@ -2,10 +2,12 @@ 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 { McpServer as McpServerBase, } from "@modelcontextprotocol/sdk/server/mcp.js";
5
+ import { Server as SdkServer, } from "@modelcontextprotocol/sdk/server/index.js";
6
+ import { McpServer as McpServerBase } from "@modelcontextprotocol/sdk/server/mcp.js";
6
7
  import { mergeWith, union } from "es-toolkit";
7
- import { DEFAULT_HMR_PORT } from "./const.js";
8
- import { createServer } from "./express.js";
8
+ import express, {} from "express";
9
+ import { createApp } from "./express.js";
10
+ import { createMiddlewareEntry } from "./metric.js";
9
11
  import { buildMiddlewareChain, getHandlerMaps } from "./middleware.js";
10
12
  import { templateHelper } from "./templateHelper.js";
11
13
  const mergeWithUnion = (target, source) => {
@@ -15,28 +17,67 @@ const mergeWithUnion = (target, source) => {
15
17
  }
16
18
  });
17
19
  };
18
- export class McpServer extends McpServerBase {
20
+ export function normalizeContent(content) {
21
+ if (content === undefined) {
22
+ return [];
23
+ }
24
+ if (typeof content === "string") {
25
+ return [{ type: "text", text: content }];
26
+ }
27
+ if (Array.isArray(content)) {
28
+ return content;
29
+ }
30
+ return [content];
31
+ }
32
+ const McpServerBaseOmitted = McpServerBase;
33
+ export class McpServer extends McpServerBaseOmitted {
34
+ /**
35
+ * The underlying Express app. Use this to extend the HTTP server with
36
+ * custom routes, middleware, or settings — e.g.
37
+ * `server.express.get("/health", ...)`.
38
+ *
39
+ * `express.json()` is pre-applied. Register your handlers before `run()`;
40
+ * after `run()`, dev-mode middleware, the `/mcp` route, and the default
41
+ * error handler are appended in that order.
42
+ *
43
+ * Note: Alpic Cloud only routes traffic to `/mcp` — custom routes work
44
+ * locally and on self-hosted deployments.
45
+ */
19
46
  express;
20
- customMiddleware = [];
47
+ customErrorMiddleware = [];
21
48
  mcpMiddlewareEntries = [];
22
49
  mcpMiddlewareApplied = false;
50
+ claimedViews = new Map();
23
51
  viteManifest = null;
52
+ serverInfo;
53
+ serverOptions;
54
+ constructor(serverInfo, options) {
55
+ super(serverInfo, options);
56
+ this.serverInfo = serverInfo;
57
+ this.serverOptions = options;
58
+ this.express = express();
59
+ this.express.use(express.json());
60
+ }
24
61
  use(pathOrHandler, ...handlers) {
62
+ // Branching is load-bearing: Express's `app.use` overloads can't be
63
+ // resolved against a `string | RequestHandler` union, so we narrow.
25
64
  if (typeof pathOrHandler === "string") {
26
- this.customMiddleware.push({
27
- path: pathOrHandler,
28
- handlers,
29
- });
65
+ this.express.use(pathOrHandler, ...handlers);
30
66
  }
31
67
  else {
32
- this.customMiddleware.push({
33
- handlers: [pathOrHandler, ...handlers],
34
- });
68
+ this.express.use(pathOrHandler, ...handlers);
35
69
  }
36
70
  return this;
37
71
  }
38
- setViteManifest(manifest) {
39
- this.viteManifest = manifest;
72
+ useOnError(pathOrHandler, ...handlers) {
73
+ if (typeof pathOrHandler === "string") {
74
+ this.customErrorMiddleware.push({ path: pathOrHandler, handlers });
75
+ }
76
+ else {
77
+ this.customErrorMiddleware.push({
78
+ handlers: [pathOrHandler, ...handlers],
79
+ });
80
+ }
40
81
  return this;
41
82
  }
42
83
  mcpMiddleware(filterOrHandler,
@@ -68,12 +109,14 @@ export class McpServer extends McpServerBase {
68
109
  return;
69
110
  }
70
111
  this.mcpMiddlewareApplied = true;
71
- if (this.mcpMiddlewareEntries.length === 0) {
112
+ const monitoringEntry = createMiddlewareEntry();
113
+ const entries = monitoringEntry
114
+ ? [monitoringEntry, ...this.mcpMiddlewareEntries]
115
+ : this.mcpMiddlewareEntries;
116
+ if (entries.length === 0) {
72
117
  return;
73
118
  }
74
119
  const { requestHandlers, notificationHandlers } = getHandlerMaps(this.server);
75
- const entries = this.mcpMiddlewareEntries;
76
- // Wrap existing handlers and proxy future .set() for lazy SDK registration
77
120
  const instrumentMap = (map, isNotification) => {
78
121
  for (const [method, handler] of map) {
79
122
  map.set(method, buildMiddlewareChain(method, isNotification, handler, entries));
@@ -86,162 +129,246 @@ export class McpServer extends McpServerBase {
86
129
  }
87
130
  async connect(transport) {
88
131
  this.applyMcpMiddleware();
89
- return super.connect(transport);
132
+ return McpServerBase.prototype.connect.call(this, transport);
133
+ }
134
+ /**
135
+ * Per-request stateless connect. The SDK's `Protocol` only allows one
136
+ * transport per instance, so we can't reuse this `McpServer` across
137
+ * concurrent requests. The SDK's idiomatic fix is a `() => McpServer`
138
+ * factory, but that would break Skybridge's singleton API — so instead
139
+ * we build a fresh underlying `Server` per request and share the main
140
+ * server's handler maps by reference. The cast is unavoidable: there's
141
+ * no public API to inject handler maps. `getHandlerMaps` validates the
142
+ * read side and fails fast on SDK field renames.
143
+ */
144
+ async connectStatelessTransport(transport) {
145
+ this.applyMcpMiddleware();
146
+ const { requestHandlers, notificationHandlers } = getHandlerMaps(this.server);
147
+ const fresh = new SdkServer(this.serverInfo, this.serverOptions);
148
+ const target = fresh;
149
+ target._requestHandlers = requestHandlers;
150
+ target._notificationHandlers = notificationHandlers;
151
+ await fresh.connect(transport);
90
152
  }
91
153
  async run() {
92
154
  this.applyMcpMiddleware();
93
- if (!this.express) {
94
- this.express = await createServer({
95
- server: this,
96
- customMiddleware: this.customMiddleware,
97
- });
98
- }
99
- const express = this.express;
155
+ const httpServer = http.createServer();
156
+ await createApp({
157
+ mcpServer: this,
158
+ httpServer,
159
+ errorMiddleware: this.customErrorMiddleware,
160
+ });
161
+ httpServer.on("request", this.express);
100
162
  const port = parseInt(process.env.__PORT ?? "3000", 10);
101
163
  await new Promise((resolve, reject) => {
102
- const server = http.createServer(express);
103
- server.on("error", (error) => {
164
+ httpServer.on("error", (error) => {
104
165
  console.error("Failed to start server:", error);
105
166
  reject(error);
106
167
  });
107
- server.listen(port, () => {
168
+ httpServer.listen(port, () => {
108
169
  resolve();
109
170
  });
110
171
  });
111
- try {
172
+ // On workerd, bridge the Node http server to a Workers fetch handler.
173
+ // The specifier is held in a variable to sidestep tsc's module resolution
174
+ // (`cloudflare:node` only exists under wrangler/workerd).
175
+ if (typeof navigator !== "undefined" &&
176
+ navigator.userAgent === "Cloudflare-Workers") {
112
177
  const cloudflareNode = "cloudflare:node";
113
178
  const { httpServerHandler } = await import(cloudflareNode);
114
179
  return httpServerHandler({ port });
115
180
  }
116
- catch {
117
- return undefined;
181
+ const shutdown = () => {
182
+ // Drop both handlers so a second signal falls through to Node's default
183
+ // (force-quit on a second Ctrl+C while drain is hanging).
184
+ process.off("SIGTERM", shutdown);
185
+ process.off("SIGINT", shutdown);
186
+ httpServer.close(() => process.exit(0));
187
+ // Force exit if connections don't drain in time so the port is still
188
+ // released promptly (e.g. for nodemon restarts).
189
+ setTimeout(() => process.exit(0), 3000).unref();
190
+ };
191
+ process.on("SIGTERM", shutdown);
192
+ process.on("SIGINT", shutdown);
193
+ return undefined;
194
+ }
195
+ enforceOneToolPerView(component, toolName) {
196
+ const existingTool = this.claimedViews.get(component);
197
+ if (existingTool) {
198
+ throw new Error(`skybridge: view "${component}" is already used by tool "${existingTool}". Tool "${toolName}" cannot also reference it — each view backs exactly one tool.`);
118
199
  }
200
+ this.claimedViews.set(component, toolName);
119
201
  }
120
- registerWidget(name, resourceConfig, toolConfig, toolCallback) {
121
- const userMeta = resourceConfig._meta;
122
- const toolMeta = {
123
- ...toolConfig._meta,
124
- };
125
- if (!resourceConfig.hosts || resourceConfig.hosts.includes("apps-sdk")) {
126
- const widgetConfig = {
202
+ registerViewResources(toolName, view, toolMeta) {
203
+ const hosts = view.hosts ?? ["apps-sdk", "mcp-app"];
204
+ // Append a content-derived version param so hosts (e.g. ChatGPT) bust
205
+ // their cache when the bundle changes, but keep the URI stable across
206
+ // `tools/list` calls when the bundle hasn't changed.
207
+ const versionParam = this.computeViewVersionParam(view.component);
208
+ if (hosts.includes("apps-sdk")) {
209
+ const viewResource = {
127
210
  hostType: "apps-sdk",
128
- uri: `ui://widgets/apps-sdk/${name}.html`,
211
+ uri: `ui://views/apps-sdk/${view.component}.html${versionParam}`,
129
212
  mimeType: "text/html+skybridge",
130
213
  buildContentMeta: ({ resourceDomains, connectDomains, domain }, overrides) => {
131
- const userUi = userMeta?.ui;
132
- const userCsp = userUi?.csp;
133
214
  const defaults = {
134
215
  "openai/widgetCSP": {
135
216
  resource_domains: resourceDomains,
136
217
  connect_domains: connectDomains,
137
218
  },
138
219
  "openai/widgetDomain": domain,
139
- "openai/widgetDescription": resourceConfig.description,
220
+ "openai/widgetDescription": view.description,
140
221
  };
141
- const fromUi = {
222
+ const fromView = {
142
223
  "openai/widgetCSP": {
143
- resource_domains: userCsp?.resourceDomains,
144
- connect_domains: userCsp?.connectDomains,
145
- frame_domains: userCsp?.frameDomains,
146
- redirect_domains: userCsp?.redirectDomains,
224
+ resource_domains: view.csp?.resourceDomains,
225
+ connect_domains: view.csp?.connectDomains,
226
+ frame_domains: view.csp?.frameDomains,
227
+ redirect_domains: view.csp?.redirectDomains,
147
228
  },
148
- "openai/widgetDomain": userUi?.domain,
149
- "openai/widgetPrefersBorder": userUi?.prefersBorder,
229
+ "openai/widgetDomain": view.domain,
230
+ "openai/widgetPrefersBorder": view.prefersBorder,
150
231
  };
151
- const directOpenaiKeys = Object.fromEntries(Object.entries(userMeta ?? {}).filter(([key]) => key.startsWith("openai/")));
152
- return mergeWithUnion(mergeWithUnion(mergeWithUnion(defaults, fromUi), directOpenaiKeys), { "openai/widgetDomain": overrides.domain });
232
+ const base = mergeWithUnion(mergeWithUnion(defaults, fromView), {
233
+ "openai/widgetDomain": overrides.domain,
234
+ });
235
+ if (view._meta) {
236
+ return { ...base, ...view._meta };
237
+ }
238
+ return base;
153
239
  },
154
240
  };
155
- this.registerWidgetResource({
156
- name,
157
- widgetConfig,
158
- resourceConfig,
241
+ this.registerViewResource({
242
+ name: toolName,
243
+ viewResource,
244
+ view,
159
245
  });
160
- toolMeta["openai/outputTemplate"] = widgetConfig.uri;
246
+ toolMeta["openai/outputTemplate"] = viewResource.uri;
161
247
  }
162
- if (!resourceConfig.hosts || resourceConfig.hosts.includes("mcp-app")) {
163
- const widgetConfig = {
248
+ if (hosts.includes("mcp-app")) {
249
+ const viewResource = {
164
250
  hostType: "mcp-app",
165
- uri: `ui://widgets/ext-apps/${name}.html`,
251
+ uri: `ui://views/ext-apps/${view.component}.html${versionParam}`,
166
252
  mimeType: "text/html;profile=mcp-app",
167
- buildContentMeta: ({ resourceDomains, connectDomains, domain }, overrides) => {
253
+ buildContentMeta: ({ resourceDomains, connectDomains, domain, baseUriDomains }, overrides) => {
168
254
  const defaults = {
169
255
  ui: {
170
256
  csp: {
171
257
  resourceDomains,
172
258
  connectDomains,
259
+ baseUriDomains,
173
260
  },
174
261
  domain,
175
262
  },
176
263
  };
177
- return mergeWithUnion(defaults, {
178
- ui: { ...userMeta?.ui, ...overrides },
264
+ const fromView = {
265
+ ui: {
266
+ ...(view.description && { description: view.description }),
267
+ ...(view.prefersBorder !== undefined && {
268
+ prefersBorder: view.prefersBorder,
269
+ }),
270
+ ...(view.domain && { domain: view.domain }),
271
+ csp: {
272
+ ...(view.csp?.resourceDomains && {
273
+ resourceDomains: view.csp.resourceDomains,
274
+ }),
275
+ ...(view.csp?.connectDomains && {
276
+ connectDomains: view.csp.connectDomains,
277
+ }),
278
+ ...(view.csp?.frameDomains && {
279
+ frameDomains: view.csp.frameDomains,
280
+ }),
281
+ ...(view.csp?.baseUriDomains && {
282
+ baseUriDomains: view.csp.baseUriDomains,
283
+ }),
284
+ ...(view.csp?.redirectDomains && {
285
+ redirectDomains: view.csp.redirectDomains,
286
+ }),
287
+ },
288
+ },
289
+ };
290
+ const base = mergeWithUnion(mergeWithUnion(defaults, fromView), {
291
+ ui: overrides,
179
292
  });
293
+ if (view._meta) {
294
+ return { ...base, ...view._meta };
295
+ }
296
+ return base;
180
297
  },
181
298
  };
182
- this.registerWidgetResource({
183
- name,
184
- widgetConfig,
185
- resourceConfig,
299
+ this.registerViewResource({
300
+ name: toolName,
301
+ viewResource,
302
+ view,
186
303
  });
187
304
  // @ts-expect-error - For backwards compatibility with Claude current implementation of the specs
188
- toolMeta["ui/resourceUri"] = widgetConfig.uri;
189
- toolMeta.ui = { resourceUri: widgetConfig.uri };
305
+ toolMeta["ui/resourceUri"] = viewResource.uri;
306
+ toolMeta.ui = { resourceUri: viewResource.uri };
190
307
  }
191
- this.registerTool(name, {
192
- ...toolConfig,
193
- _meta: toolMeta,
194
- }, toolCallback);
195
- return this;
196
308
  }
197
- registerTool(name, config, cb) {
198
- super.registerTool(name, config, cb);
199
- return this;
200
- }
201
- registerWidgetResource({ name, widgetConfig, resourceConfig, }) {
202
- const { hostType, uri: widgetUri, mimeType, buildContentMeta, } = widgetConfig;
203
- this.registerResource(name, widgetUri, { ...resourceConfig, _meta: resourceConfig._meta }, async (uri, extra) => {
309
+ registerViewResource({ name, viewResource, view, }) {
310
+ const { hostType, uri: viewUri, mimeType, buildContentMeta } = viewResource;
311
+ this.registerResource(name, viewUri, { description: view.description }, async (uri, extra) => {
204
312
  const isProduction = process.env.NODE_ENV === "production";
205
- const useForwardedHost = process.env.SKYBRIDGE_USE_FORWARDED_HOST === "true";
206
313
  const isClaude = extra?.requestInfo?.headers?.["user-agent"] === "Claude-User";
207
- const hostFromHeaders = extra?.requestInfo?.headers?.["x-forwarded-host"] ??
208
- extra?.requestInfo?.headers?.host;
209
- const useExternalHost = isProduction || useForwardedHost || isClaude;
210
- const devPort = process.env.__PORT || "3000";
211
- const serverUrl = useExternalHost
212
- ? `https://${hostFromHeaders}`
213
- : `http://localhost:${devPort}`;
314
+ const headers = extra?.requestInfo?.headers || {};
315
+ const header = (key) => {
316
+ const val = headers[key];
317
+ return Array.isArray(val) ? val[0] : val;
318
+ };
319
+ let serverUrl;
320
+ const forwardedHost = header("x-forwarded-host");
321
+ const origin = header("origin");
322
+ const host = header("host");
323
+ if (forwardedHost) {
324
+ const proto = header("x-forwarded-proto") || "https";
325
+ serverUrl = `${proto}://${forwardedHost}`;
326
+ }
327
+ else if (origin) {
328
+ serverUrl = origin;
329
+ }
330
+ else if (host) {
331
+ const proto = ["127.0.0.1:", "localhost:"].some((p) => host.startsWith(p))
332
+ ? "http"
333
+ : "https";
334
+ serverUrl = `${proto}://${host}`;
335
+ }
336
+ else {
337
+ const devPort = process.env.__PORT || "3000";
338
+ serverUrl = `http://localhost:${devPort}`;
339
+ }
214
340
  const html = isProduction
215
341
  ? templateHelper.renderProduction({
216
342
  hostType,
217
343
  serverUrl,
218
- widgetFile: this.lookupDistFileWithIndexFallback(`src/widgets/${name}`),
219
- styleFile: this.lookupDistFile("style.css"),
344
+ viewFile: this.lookupViewFile(view.component),
345
+ styleFile: this.lookupDistFile("style.css") ?? "",
220
346
  })
221
347
  : templateHelper.renderDevelopment({
222
348
  hostType,
223
349
  serverUrl,
224
- useLocalNetworkAccess: !useExternalHost,
225
- widgetName: name,
350
+ viewName: view.component,
226
351
  });
227
352
  const connectDomains = [serverUrl];
228
353
  if (!isProduction) {
229
- const hmrPort = process.env.__SKYBRIDGE_HMR_PORT ?? DEFAULT_HMR_PORT;
230
- const VITE_HMR_WEBSOCKET_DEFAULT_URL = `ws://localhost:${hmrPort}`;
231
- connectDomains.push(VITE_HMR_WEBSOCKET_DEFAULT_URL);
354
+ const wsUrl = new URL(serverUrl);
355
+ wsUrl.protocol = wsUrl.protocol === "https:" ? "wss:" : "ws:";
356
+ connectDomains.push(wsUrl.origin);
357
+ }
358
+ let contentMetaOverrides = {};
359
+ if (isClaude) {
360
+ const pathname = extra?.requestInfo?.url?.pathname ?? "";
361
+ const rawUrl = header("x-alpic-forwarded-url") ?? `${serverUrl}${pathname}`;
362
+ // Strip a lone trailing slash so the hash matches the connector URL
363
+ // as registered with Claude (which has no trailing slash on bare origins).
364
+ const url = rawUrl.endsWith("/") ? rawUrl.slice(0, -1) : rawUrl;
365
+ const hash = crypto
366
+ .createHash("sha256")
367
+ .update(url)
368
+ .digest("hex")
369
+ .slice(0, 32);
370
+ contentMetaOverrides = { domain: `${hash}.claudemcpcontent.com` };
232
371
  }
233
- const pathname = extra?.requestInfo?.url?.pathname ?? "";
234
- const url = `https://${hostFromHeaders}${pathname}`;
235
- const hash = crypto
236
- .createHash("sha256")
237
- .update(url)
238
- .digest("hex")
239
- .slice(0, 32);
240
- const contentMetaOverrides = isClaude
241
- ? {
242
- domain: `${hash}.claudemcpcontent.com`,
243
- }
244
- : {};
245
372
  const contentMeta = buildContentMeta({
246
373
  resourceDomains: [serverUrl],
247
374
  connectDomains,
@@ -255,15 +382,63 @@ export class McpServer extends McpServerBase {
255
382
  };
256
383
  });
257
384
  }
385
+ wrapHandler(cb, { attachViewUUID }) {
386
+ return async (args, extra) => {
387
+ const result = await cb(args, extra);
388
+ return {
389
+ ...result,
390
+ content: normalizeContent(result.content),
391
+ ...(attachViewUUID && {
392
+ _meta: {
393
+ ...result._meta,
394
+ viewUUID: crypto.randomUUID(),
395
+ },
396
+ }),
397
+ };
398
+ };
399
+ }
400
+ computeViewVersionParam(viewName) {
401
+ if (process.env.NODE_ENV !== "production") {
402
+ return "";
403
+ }
404
+ try {
405
+ const viewFile = this.lookupViewFile(viewName);
406
+ const styleFile = this.lookupDistFile("style.css") ?? "";
407
+ const hash = crypto
408
+ .createHash("sha256")
409
+ .update(viewFile)
410
+ .update("\0")
411
+ .update(styleFile)
412
+ .digest("hex")
413
+ .slice(0, 8);
414
+ return `?v=${hash}`;
415
+ }
416
+ catch {
417
+ return "";
418
+ }
419
+ }
420
+ lookupViewFile(viewName) {
421
+ const manifest = this.readManifest();
422
+ for (const entry of Object.values(manifest)) {
423
+ if (entry?.isEntry && entry.name === viewName && entry.file) {
424
+ return entry.file;
425
+ }
426
+ }
427
+ 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.`);
428
+ }
258
429
  lookupDistFile(key) {
259
430
  const manifest = this.readManifest();
260
431
  return manifest[key]?.file;
261
432
  }
262
- lookupDistFileWithIndexFallback(basePath) {
263
- const manifest = this.readManifest();
264
- const flatFileKey = `${basePath}.tsx`;
265
- const indexFileKey = `${basePath}/index.tsx`;
266
- return manifest[flatFileKey]?.file ?? manifest[indexFileKey]?.file;
433
+ /**
434
+ * Inject the Vite manifest as a value rather than letting `readManifest()`
435
+ * load it from disk. Required for runtimes without a usable filesystem
436
+ * (Cloudflare Workers, etc.) — the user's `skybridge build` emits the
437
+ * manifest as a JS module which the entry imports and passes here.
438
+ */
439
+ setViteManifest(manifest) {
440
+ this.viteManifest = manifest;
441
+ return this;
267
442
  }
268
443
  readManifest() {
269
444
  if (this.viteManifest) {
@@ -271,5 +446,23 @@ export class McpServer extends McpServerBase {
271
446
  }
272
447
  return JSON.parse(readFileSync(path.join(process.cwd(), "dist", "assets", ".vite", "manifest.json"), "utf-8"));
273
448
  }
449
+ registerTool(...args) {
450
+ const baseFn = McpServerBase.prototype.registerTool;
451
+ if (typeof args[0] === "string") {
452
+ baseFn.call(this, args[0], args[1], args[2]);
453
+ return this;
454
+ }
455
+ const config = args[0];
456
+ const cb = args[1];
457
+ const { name, view, _meta: userToolMeta, ...toolFields } = config;
458
+ const toolMeta = { ...userToolMeta };
459
+ if (view) {
460
+ this.enforceOneToolPerView(view.component, name);
461
+ this.registerViewResources(name, view, toolMeta);
462
+ }
463
+ const wrappedCb = this.wrapHandler(cb, { attachViewUUID: Boolean(view) });
464
+ baseFn.call(this, name, { ...toolFields, _meta: toolMeta }, wrappedCb);
465
+ return this;
466
+ }
274
467
  }
275
468
  //# sourceMappingURL=server.js.map