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