skybridge 0.0.0-dev.f6bca06 → 0.0.0-dev.f706bf6
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 +24 -16
- package/dist/cli/detect-port.d.ts +18 -0
- package/dist/cli/detect-port.js +61 -0
- package/dist/cli/detect-port.js.map +1 -0
- package/dist/cli/header.js +1 -1
- package/dist/cli/header.js.map +1 -1
- 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.d.ts +1 -0
- package/dist/cli/tunnel.test.js +190 -0
- package/dist/cli/tunnel.test.js.map +1 -0
- package/dist/cli/types.d.ts +5 -0
- package/dist/cli/types.js +2 -0
- package/dist/cli/types.js.map +1 -0
- 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 -6
- package/dist/cli/use-nodemon.js +29 -17
- 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 +14 -0
- package/dist/cli/use-tunnel.js +131 -0
- package/dist/cli/use-tunnel.js.map +1 -0
- package/dist/cli/use-typescript-check.d.ts +1 -0
- package/dist/cli/use-typescript-check.js +41 -6
- package/dist/cli/use-typescript-check.js.map +1 -1
- package/dist/commands/build.js +63 -7
- package/dist/commands/build.js.map +1 -1
- package/dist/commands/dev.d.ts +4 -1
- package/dist/commands/dev.js +57 -8
- package/dist/commands/dev.js.map +1 -1
- package/dist/commands/start.d.ts +3 -1
- package/dist/commands/start.js +30 -9
- 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 +5 -6
- package/dist/server/asset-base-url-transform-plugin.js +9 -10
- package/dist/server/asset-base-url-transform-plugin.js.map +1 -1
- package/dist/server/asset-base-url-transform-plugin.test.js +41 -13
- package/dist/server/asset-base-url-transform-plugin.test.js.map +1 -1
- package/dist/server/content-helpers.d.ts +27 -0
- package/dist/server/content-helpers.js +46 -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 +9 -3
- package/dist/server/express.js +64 -26
- package/dist/server/express.js.map +1 -1
- package/dist/server/express.test.d.ts +1 -0
- package/dist/server/express.test.js +430 -0
- package/dist/server/express.test.js.map +1 -0
- package/dist/server/file-ref.d.ts +8 -0
- package/dist/server/file-ref.js +8 -0
- package/dist/server/file-ref.js.map +1 -0
- package/dist/server/index.d.ts +6 -3
- package/dist/server/index.js +4 -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 +124 -0
- package/dist/server/middleware.js +93 -0
- package/dist/server/middleware.js.map +1 -0
- package/dist/server/middleware.test-d.d.ts +1 -0
- package/dist/server/middleware.test-d.js +75 -0
- package/dist/server/middleware.test-d.js.map +1 -0
- package/dist/server/middleware.test.d.ts +1 -0
- package/dist/server/middleware.test.js +493 -0
- package/dist/server/middleware.test.js.map +1 -0
- package/dist/server/server.d.ts +162 -72
- package/dist/server/server.js +368 -101
- package/dist/server/server.js.map +1 -1
- package/dist/server/templateHelper.d.ts +5 -8
- 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 +16 -244
- 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 +523 -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 +11 -7
- package/dist/web/bridges/apps-sdk/adaptor.js +57 -19
- package/dist/web/bridges/apps-sdk/adaptor.js.map +1 -1
- package/dist/web/bridges/apps-sdk/bridge.d.ts +1 -1
- package/dist/web/bridges/apps-sdk/bridge.js.map +1 -1
- package/dist/web/bridges/apps-sdk/index.d.ts +1 -1
- package/dist/web/bridges/apps-sdk/index.js.map +1 -1
- package/dist/web/bridges/apps-sdk/types.d.ts +32 -14
- package/dist/web/bridges/apps-sdk/types.js.map +1 -1
- package/dist/web/bridges/apps-sdk/use-apps-sdk-context.js.map +1 -1
- 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 +23 -9
- package/dist/web/bridges/mcp-app/adaptor.js +150 -67
- package/dist/web/bridges/mcp-app/adaptor.js.map +1 -1
- package/dist/web/bridges/mcp-app/bridge.d.ts +13 -30
- package/dist/web/bridges/mcp-app/bridge.js +43 -196
- 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 +5 -3
- package/dist/web/bridges/mcp-app/use-mcp-app-context.js +2 -2
- 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 +1 -41
- package/dist/web/bridges/mcp-app/use-mcp-app-context.test.js.map +1 -1
- package/dist/web/bridges/types.d.ts +32 -13
- package/dist/web/bridges/types.js.map +1 -1
- package/dist/web/bridges/use-host-context.js.map +1 -1
- package/dist/web/components/modal-provider.js +3 -5
- package/dist/web/components/modal-provider.js.map +1 -1
- package/dist/web/create-store.js +17 -3
- package/dist/web/create-store.js.map +1 -1
- package/dist/web/create-store.test.js +23 -20
- package/dist/web/create-store.test.js.map +1 -1
- package/dist/web/data-llm.d.ts +1 -1
- package/dist/web/data-llm.js +3 -3
- package/dist/web/data-llm.js.map +1 -1
- package/dist/web/data-llm.test.js +33 -30
- package/dist/web/data-llm.test.js.map +1 -1
- package/dist/web/generate-helpers.d.ts +20 -18
- package/dist/web/generate-helpers.js +20 -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 -2
- package/dist/web/hooks/index.js +3 -1
- package/dist/web/hooks/index.js.map +1 -1
- package/dist/web/hooks/test/utils.js +4 -0
- package/dist/web/hooks/test/utils.js.map +1 -1
- 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 +0 -4
- package/dist/web/hooks/use-call-tool.test.js.map +1 -1
- package/dist/web/hooks/use-display-mode.d.ts +3 -3
- package/dist/web/hooks/use-display-mode.js.map +1 -1
- package/dist/web/hooks/use-display-mode.test-d.d.ts +1 -0
- package/dist/web/hooks/use-display-mode.test-d.js +8 -0
- package/dist/web/hooks/use-display-mode.test-d.js.map +1 -0
- package/dist/web/hooks/use-display-mode.test.js.map +1 -1
- package/dist/web/hooks/use-files.d.ts +2 -1
- package/dist/web/hooks/use-files.js +1 -0
- package/dist/web/hooks/use-files.js.map +1 -1
- package/dist/web/hooks/use-files.test.js +27 -3
- package/dist/web/hooks/use-files.test.js.map +1 -1
- package/dist/web/hooks/use-layout.js.map +1 -1
- package/dist/web/hooks/use-layout.test.js +3 -3
- package/dist/web/hooks/use-layout.test.js.map +1 -1
- package/dist/web/hooks/use-open-external.d.ts +3 -1
- package/dist/web/hooks/use-open-external.js +1 -1
- package/dist/web/hooks/use-open-external.js.map +1 -1
- package/dist/web/hooks/use-open-external.test.js +26 -11
- package/dist/web/hooks/use-open-external.test.js.map +1 -1
- package/dist/web/hooks/use-request-close.d.ts +2 -0
- package/dist/web/hooks/use-request-close.js +8 -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 +1 -1
- package/dist/web/hooks/use-request-modal.js +4 -4
- package/dist/web/hooks/use-request-modal.js.map +1 -1
- package/dist/web/hooks/use-request-modal.test.js +5 -1
- package/dist/web/hooks/use-request-modal.test.js.map +1 -1
- package/dist/web/hooks/use-request-size.d.ts +3 -0
- package/dist/web/hooks/use-request-size.js +8 -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 +2 -1
- package/dist/web/hooks/use-send-follow-up-message.js +2 -2
- package/dist/web/hooks/use-send-follow-up-message.js.map +1 -1
- 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 +5 -11
- package/dist/web/hooks/use-set-open-in-app-url.test.js.map +1 -1
- 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 +1 -1
- package/dist/web/hooks/use-tool-info.test.js.map +1 -1
- package/dist/web/hooks/use-user.js +18 -2
- package/dist/web/hooks/use-user.js.map +1 -1
- package/dist/web/hooks/use-user.test.js +29 -1
- package/dist/web/hooks/use-user.test.js.map +1 -1
- package/dist/web/hooks/use-view-state.d.ts +4 -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 +1 -0
- package/dist/web/{mount-widget.js → mount-view.js} +2 -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 +4 -1
- package/dist/web/plugin/plugin.js +135 -18
- 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.js.map +1 -1
- package/package.json +39 -26
- package/tsconfig.base.json +33 -0
- package/dist/server/templates/development.hbs +0 -67
- package/dist/server/templates/production.hbs +0 -6
- package/dist/server/widgetsDevServer.d.ts +0 -12
- package/dist/server/widgetsDevServer.js +0 -57
- package/dist/server/widgetsDevServer.js.map +0 -1
- package/dist/test/widget.test.js +0 -261
- 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 -61
- 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/{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/server/server.js
CHANGED
|
@@ -2,9 +2,13 @@ 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";
|
|
7
|
-
import {
|
|
8
|
+
import express, {} from "express";
|
|
9
|
+
import { createApp } from "./express.js";
|
|
10
|
+
import { createMiddlewareEntry } from "./metric.js";
|
|
11
|
+
import { buildMiddlewareChain, getHandlerMaps } from "./middleware.js";
|
|
8
12
|
import { templateHelper } from "./templateHelper.js";
|
|
9
13
|
const mergeWithUnion = (target, source) => {
|
|
10
14
|
return mergeWith(target, source, (targetVal, sourceVal) => {
|
|
@@ -13,170 +17,364 @@ const mergeWithUnion = (target, source) => {
|
|
|
13
17
|
}
|
|
14
18
|
});
|
|
15
19
|
};
|
|
16
|
-
export
|
|
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
|
+
*/
|
|
17
46
|
express;
|
|
18
|
-
|
|
47
|
+
customErrorMiddleware = [];
|
|
48
|
+
mcpMiddlewareEntries = [];
|
|
49
|
+
mcpMiddlewareApplied = false;
|
|
50
|
+
claimedViews = new Map();
|
|
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
|
+
}
|
|
19
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.
|
|
20
64
|
if (typeof pathOrHandler === "string") {
|
|
21
|
-
this.
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
65
|
+
this.express.use(pathOrHandler, ...handlers);
|
|
66
|
+
}
|
|
67
|
+
else {
|
|
68
|
+
this.express.use(pathOrHandler, ...handlers);
|
|
69
|
+
}
|
|
70
|
+
return this;
|
|
71
|
+
}
|
|
72
|
+
useOnError(pathOrHandler, ...handlers) {
|
|
73
|
+
if (typeof pathOrHandler === "string") {
|
|
74
|
+
this.customErrorMiddleware.push({ path: pathOrHandler, handlers });
|
|
25
75
|
}
|
|
26
76
|
else {
|
|
27
|
-
this.
|
|
77
|
+
this.customErrorMiddleware.push({
|
|
28
78
|
handlers: [pathOrHandler, ...handlers],
|
|
29
79
|
});
|
|
30
80
|
}
|
|
31
81
|
return this;
|
|
32
82
|
}
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
83
|
+
mcpMiddleware(filterOrHandler,
|
|
84
|
+
// biome-ignore lint/suspicious/noExplicitAny: overloads narrow the handler type at call sites; implementation must accept all variants
|
|
85
|
+
maybeHandler) {
|
|
86
|
+
if (this.mcpMiddlewareApplied) {
|
|
87
|
+
throw new Error("Cannot register MCP middleware after run() or connect() has been called");
|
|
88
|
+
}
|
|
89
|
+
const handler = maybeHandler;
|
|
90
|
+
if (typeof filterOrHandler === "function") {
|
|
91
|
+
this.mcpMiddlewareEntries.push({
|
|
92
|
+
filter: null,
|
|
93
|
+
handler: filterOrHandler,
|
|
94
|
+
});
|
|
95
|
+
}
|
|
96
|
+
else if (handler) {
|
|
97
|
+
this.mcpMiddlewareEntries.push({
|
|
98
|
+
filter: filterOrHandler,
|
|
99
|
+
handler,
|
|
37
100
|
});
|
|
38
|
-
for (const middleware of this.customMiddleware) {
|
|
39
|
-
if (middleware.path) {
|
|
40
|
-
this.express.use(middleware.path, ...middleware.handlers);
|
|
41
|
-
}
|
|
42
|
-
else {
|
|
43
|
-
this.express.use(...middleware.handlers);
|
|
44
|
-
}
|
|
45
|
-
}
|
|
46
101
|
}
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
102
|
+
else {
|
|
103
|
+
throw new Error("mcpMiddleware requires a handler function when a filter is provided");
|
|
104
|
+
}
|
|
105
|
+
return this;
|
|
106
|
+
}
|
|
107
|
+
applyMcpMiddleware() {
|
|
108
|
+
if (this.mcpMiddlewareApplied) {
|
|
109
|
+
return;
|
|
110
|
+
}
|
|
111
|
+
this.mcpMiddlewareApplied = true;
|
|
112
|
+
const monitoringEntry = createMiddlewareEntry();
|
|
113
|
+
const entries = monitoringEntry
|
|
114
|
+
? [monitoringEntry, ...this.mcpMiddlewareEntries]
|
|
115
|
+
: this.mcpMiddlewareEntries;
|
|
116
|
+
if (entries.length === 0) {
|
|
117
|
+
return;
|
|
118
|
+
}
|
|
119
|
+
const { requestHandlers, notificationHandlers } = getHandlerMaps(this.server);
|
|
120
|
+
const instrumentMap = (map, isNotification) => {
|
|
121
|
+
for (const [method, handler] of map) {
|
|
122
|
+
map.set(method, buildMiddlewareChain(method, isNotification, handler, entries));
|
|
123
|
+
}
|
|
124
|
+
const originalSet = map.set.bind(map);
|
|
125
|
+
map.set = (method, handler) => originalSet(method, buildMiddlewareChain(method, isNotification, handler, entries));
|
|
126
|
+
};
|
|
127
|
+
instrumentMap(requestHandlers, false);
|
|
128
|
+
instrumentMap(notificationHandlers, true);
|
|
129
|
+
}
|
|
130
|
+
async connect(transport) {
|
|
131
|
+
this.applyMcpMiddleware();
|
|
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);
|
|
152
|
+
}
|
|
153
|
+
async run() {
|
|
154
|
+
this.applyMcpMiddleware();
|
|
155
|
+
const httpServer = http.createServer();
|
|
156
|
+
await createApp({
|
|
157
|
+
mcpServer: this,
|
|
158
|
+
httpServer,
|
|
159
|
+
errorMiddleware: this.customErrorMiddleware,
|
|
160
|
+
});
|
|
161
|
+
httpServer.on("request", this.express);
|
|
162
|
+
const port = parseInt(process.env.__PORT ?? "3000", 10);
|
|
163
|
+
await new Promise((resolve, reject) => {
|
|
164
|
+
httpServer.on("error", (error) => {
|
|
51
165
|
console.error("Failed to start server:", error);
|
|
52
166
|
reject(error);
|
|
53
167
|
});
|
|
54
|
-
|
|
168
|
+
httpServer.listen(port, () => {
|
|
55
169
|
resolve();
|
|
56
170
|
});
|
|
57
171
|
});
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
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") {
|
|
177
|
+
const cloudflareNode = "cloudflare:node";
|
|
178
|
+
const { httpServerHandler } = await import(cloudflareNode);
|
|
179
|
+
return httpServerHandler({ port });
|
|
180
|
+
}
|
|
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();
|
|
63
190
|
};
|
|
64
|
-
|
|
65
|
-
|
|
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.`);
|
|
199
|
+
}
|
|
200
|
+
this.claimedViews.set(component, toolName);
|
|
201
|
+
}
|
|
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 = {
|
|
66
210
|
hostType: "apps-sdk",
|
|
67
|
-
uri: `ui://
|
|
211
|
+
uri: `ui://views/apps-sdk/${view.component}.html${versionParam}`,
|
|
68
212
|
mimeType: "text/html+skybridge",
|
|
69
|
-
buildContentMeta: ({ resourceDomains, connectDomains, domain }) => {
|
|
70
|
-
const userUi = userMeta?.ui;
|
|
71
|
-
const userCsp = userUi?.csp;
|
|
213
|
+
buildContentMeta: ({ resourceDomains, connectDomains, domain }, overrides) => {
|
|
72
214
|
const defaults = {
|
|
73
215
|
"openai/widgetCSP": {
|
|
74
216
|
resource_domains: resourceDomains,
|
|
75
217
|
connect_domains: connectDomains,
|
|
76
218
|
},
|
|
77
219
|
"openai/widgetDomain": domain,
|
|
78
|
-
"openai/widgetDescription":
|
|
220
|
+
"openai/widgetDescription": view.description,
|
|
79
221
|
};
|
|
80
|
-
const
|
|
222
|
+
const fromView = {
|
|
81
223
|
"openai/widgetCSP": {
|
|
82
|
-
resource_domains:
|
|
83
|
-
connect_domains:
|
|
84
|
-
frame_domains:
|
|
85
|
-
redirect_domains:
|
|
224
|
+
resource_domains: view.csp?.resourceDomains,
|
|
225
|
+
connect_domains: view.csp?.connectDomains,
|
|
226
|
+
frame_domains: view.csp?.frameDomains,
|
|
227
|
+
redirect_domains: view.csp?.redirectDomains,
|
|
86
228
|
},
|
|
87
|
-
"openai/widgetDomain":
|
|
88
|
-
"openai/widgetPrefersBorder":
|
|
229
|
+
"openai/widgetDomain": view.domain,
|
|
230
|
+
"openai/widgetPrefersBorder": view.prefersBorder,
|
|
89
231
|
};
|
|
90
|
-
const
|
|
91
|
-
|
|
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;
|
|
92
239
|
},
|
|
93
240
|
};
|
|
94
|
-
this.
|
|
95
|
-
name,
|
|
96
|
-
|
|
97
|
-
|
|
241
|
+
this.registerViewResource({
|
|
242
|
+
name: toolName,
|
|
243
|
+
viewResource,
|
|
244
|
+
view,
|
|
98
245
|
});
|
|
99
|
-
toolMeta["openai/outputTemplate"] =
|
|
246
|
+
toolMeta["openai/outputTemplate"] = viewResource.uri;
|
|
100
247
|
}
|
|
101
|
-
if (
|
|
102
|
-
const
|
|
248
|
+
if (hosts.includes("mcp-app")) {
|
|
249
|
+
const viewResource = {
|
|
103
250
|
hostType: "mcp-app",
|
|
104
|
-
uri: `ui://
|
|
251
|
+
uri: `ui://views/ext-apps/${view.component}.html${versionParam}`,
|
|
105
252
|
mimeType: "text/html;profile=mcp-app",
|
|
106
|
-
buildContentMeta: ({ resourceDomains, connectDomains, domain }) => {
|
|
253
|
+
buildContentMeta: ({ resourceDomains, connectDomains, domain, baseUriDomains }, overrides) => {
|
|
107
254
|
const defaults = {
|
|
108
255
|
ui: {
|
|
109
256
|
csp: {
|
|
110
257
|
resourceDomains,
|
|
111
258
|
connectDomains,
|
|
259
|
+
baseUriDomains,
|
|
112
260
|
},
|
|
113
261
|
domain,
|
|
114
262
|
},
|
|
115
263
|
};
|
|
116
|
-
|
|
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,
|
|
292
|
+
});
|
|
293
|
+
if (view._meta) {
|
|
294
|
+
return { ...base, ...view._meta };
|
|
295
|
+
}
|
|
296
|
+
return base;
|
|
117
297
|
},
|
|
118
298
|
};
|
|
119
|
-
this.
|
|
120
|
-
name,
|
|
121
|
-
|
|
122
|
-
|
|
299
|
+
this.registerViewResource({
|
|
300
|
+
name: toolName,
|
|
301
|
+
viewResource,
|
|
302
|
+
view,
|
|
123
303
|
});
|
|
124
304
|
// @ts-expect-error - For backwards compatibility with Claude current implementation of the specs
|
|
125
|
-
toolMeta["ui/resourceUri"] =
|
|
126
|
-
toolMeta.ui = { resourceUri:
|
|
305
|
+
toolMeta["ui/resourceUri"] = viewResource.uri;
|
|
306
|
+
toolMeta.ui = { ...toolMeta.ui, resourceUri: viewResource.uri };
|
|
127
307
|
}
|
|
128
|
-
this.registerTool(name, {
|
|
129
|
-
...toolConfig,
|
|
130
|
-
_meta: toolMeta,
|
|
131
|
-
}, toolCallback);
|
|
132
|
-
return this;
|
|
133
|
-
}
|
|
134
|
-
registerTool(name, config, cb) {
|
|
135
|
-
super.registerTool(name, config, cb);
|
|
136
|
-
return this;
|
|
137
308
|
}
|
|
138
|
-
|
|
139
|
-
const { hostType, uri:
|
|
140
|
-
this.registerResource(name,
|
|
309
|
+
registerViewResource({ name, viewResource, view, }) {
|
|
310
|
+
const { hostType, uri: viewUri, mimeType, buildContentMeta } = viewResource;
|
|
311
|
+
this.registerResource(name, viewUri, { description: view.description }, async (uri, extra) => {
|
|
141
312
|
const isProduction = process.env.NODE_ENV === "production";
|
|
142
|
-
const useForwardedHost = process.env.SKYBRIDGE_USE_FORWARDED_HOST === "true";
|
|
143
313
|
const isClaude = extra?.requestInfo?.headers?.["user-agent"] === "Claude-User";
|
|
144
|
-
const
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
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
|
+
}
|
|
150
340
|
const html = isProduction
|
|
151
341
|
? templateHelper.renderProduction({
|
|
152
342
|
hostType,
|
|
153
343
|
serverUrl,
|
|
154
|
-
|
|
155
|
-
styleFile: this.lookupDistFile("style.css"),
|
|
344
|
+
viewFile: this.lookupViewFile(view.component),
|
|
345
|
+
styleFile: this.lookupDistFile("style.css") ?? "",
|
|
156
346
|
})
|
|
157
347
|
: templateHelper.renderDevelopment({
|
|
158
348
|
hostType,
|
|
159
349
|
serverUrl,
|
|
160
|
-
|
|
161
|
-
widgetName: name,
|
|
350
|
+
viewName: view.component,
|
|
162
351
|
});
|
|
163
352
|
const connectDomains = [serverUrl];
|
|
164
353
|
if (!isProduction) {
|
|
165
|
-
const
|
|
166
|
-
|
|
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` };
|
|
167
371
|
}
|
|
168
372
|
const contentMeta = buildContentMeta({
|
|
169
373
|
resourceDomains: [serverUrl],
|
|
170
374
|
connectDomains,
|
|
171
|
-
domain:
|
|
172
|
-
? `${crypto
|
|
173
|
-
.createHash("sha256")
|
|
174
|
-
.update(`https://${hostFromHeaders}/mcp`)
|
|
175
|
-
.digest("hex")
|
|
176
|
-
.slice(0, 32)}.claudemcpcontent.com`
|
|
177
|
-
: serverUrl,
|
|
375
|
+
domain: serverUrl,
|
|
178
376
|
baseUriDomains: [serverUrl],
|
|
179
|
-
});
|
|
377
|
+
}, contentMetaOverrides);
|
|
180
378
|
return {
|
|
181
379
|
contents: [
|
|
182
380
|
{ uri: uri.href, mimeType, text: html, _meta: contentMeta },
|
|
@@ -184,18 +382,87 @@ export class McpServer extends McpServerBase {
|
|
|
184
382
|
};
|
|
185
383
|
});
|
|
186
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
|
+
}
|
|
187
429
|
lookupDistFile(key) {
|
|
188
430
|
const manifest = this.readManifest();
|
|
189
431
|
return manifest[key]?.file;
|
|
190
432
|
}
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
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;
|
|
196
442
|
}
|
|
197
443
|
readManifest() {
|
|
444
|
+
if (this.viteManifest) {
|
|
445
|
+
return this.viteManifest;
|
|
446
|
+
}
|
|
198
447
|
return JSON.parse(readFileSync(path.join(process.cwd(), "dist", "assets", ".vite", "manifest.json"), "utf-8"));
|
|
199
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
|
+
}
|
|
200
467
|
}
|
|
201
468
|
//# sourceMappingURL=server.js.map
|