skybridge 0.0.0-dev.55ae9f7 → 0.0.0-dev.570aaaa

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.
@@ -0,0 +1,24 @@
1
+ import type { McpUiHostContext, McpUiInitializeRequest } from "@modelcontextprotocol/ext-apps";
2
+ type McpAppInitializationOptions = Pick<McpUiInitializeRequest["params"], "appInfo">;
3
+ export declare class McpAppBridge {
4
+ context: McpUiHostContext | null;
5
+ private listeners;
6
+ private pendingRequests;
7
+ private nextId;
8
+ private initialized;
9
+ private appInitializationOptions;
10
+ private requestTimeout;
11
+ constructor(options: McpAppInitializationOptions, requestTimeout?: number);
12
+ subscribe: (key: keyof McpUiHostContext) => (onChange: () => void) => () => boolean | undefined;
13
+ cleanup(): void;
14
+ private request;
15
+ private emit;
16
+ private setContext;
17
+ private init;
18
+ private handleMessage;
19
+ private connect;
20
+ private notify;
21
+ }
22
+ export declare function getMcpHost(options?: Partial<McpAppInitializationOptions>, requestTimeout?: number): McpAppBridge;
23
+ export declare function useMcpAppBridge<K extends keyof McpUiHostContext>(key: K, options?: Partial<McpAppInitializationOptions>, requestTimeout?: number): McpUiHostContext[K] | undefined;
24
+ export {};
@@ -0,0 +1,119 @@
1
+ import { useSyncExternalStore } from "react";
2
+ const LATEST_PROTOCOL_VERSION = "2025-11-21";
3
+ export class McpAppBridge {
4
+ context = null;
5
+ listeners = new Map();
6
+ pendingRequests = new Map();
7
+ nextId = 1;
8
+ initialized;
9
+ appInitializationOptions;
10
+ requestTimeout;
11
+ constructor(options, requestTimeout = 10_000) {
12
+ this.requestTimeout = requestTimeout;
13
+ this.initialized = false;
14
+ this.appInitializationOptions = {
15
+ appInfo: options.appInfo,
16
+ appCapabilities: {},
17
+ protocolVersion: LATEST_PROTOCOL_VERSION,
18
+ };
19
+ }
20
+ subscribe = (key) => (onChange) => {
21
+ this.listeners.set(key, new Set([...(this.listeners.get(key) || []), onChange]));
22
+ this.init();
23
+ return () => this.listeners.get(key)?.delete(onChange);
24
+ };
25
+ cleanup() {
26
+ window.removeEventListener("message", this.handleMessage);
27
+ this.pendingRequests.clear();
28
+ this.listeners.clear();
29
+ }
30
+ request({ method, params, }) {
31
+ const id = this.nextId++;
32
+ const { promise, resolve, reject } = Promise.withResolvers();
33
+ this.pendingRequests.set(id, {
34
+ resolve: resolve,
35
+ reject,
36
+ });
37
+ window.parent.postMessage({ jsonrpc: "2.0", id, method, params }, "*");
38
+ setTimeout(() => {
39
+ reject(new Error("Request timed out"));
40
+ this.pendingRequests.delete(id);
41
+ }, this.requestTimeout);
42
+ return promise;
43
+ }
44
+ emit(key) {
45
+ this.listeners.get(key)?.forEach((listener) => {
46
+ listener();
47
+ });
48
+ }
49
+ setContext(context) {
50
+ if (context == null)
51
+ return;
52
+ this.context = context;
53
+ for (const key of Object.keys(context)) {
54
+ this.emit(key);
55
+ }
56
+ }
57
+ init() {
58
+ if (this.initialized)
59
+ return;
60
+ this.initialized = true;
61
+ if (typeof window === "undefined" || window.parent === window) {
62
+ return;
63
+ }
64
+ window.addEventListener("message", this.handleMessage);
65
+ this.connect();
66
+ }
67
+ handleMessage = (event) => {
68
+ const data = event.data;
69
+ if (data?.jsonrpc !== "2.0")
70
+ return;
71
+ const request = this.pendingRequests.get(data.id);
72
+ if (request) {
73
+ this.pendingRequests.delete(data.id);
74
+ if (data.error) {
75
+ request.reject(new Error(data.error.message));
76
+ return;
77
+ }
78
+ request.resolve(data.result);
79
+ return;
80
+ }
81
+ if (data.method === "ui/notifications/host-context-changed") {
82
+ this.setContext({ ...this.context, ...data.params });
83
+ }
84
+ };
85
+ async connect() {
86
+ try {
87
+ const result = await this.request({
88
+ method: "ui/initialize",
89
+ params: this.appInitializationOptions,
90
+ });
91
+ this.setContext(result.hostContext);
92
+ this.notify({ method: "ui/notifications/initialized" });
93
+ }
94
+ catch (err) {
95
+ console.error(err);
96
+ }
97
+ }
98
+ notify(notification) {
99
+ window.parent.postMessage({ jsonrpc: "2.0", ...notification }, "*");
100
+ }
101
+ }
102
+ let instance = null;
103
+ const defaultOptions = {
104
+ appInfo: { name: "skybridge-app", version: "0.0.1" },
105
+ };
106
+ export function getMcpHost(options, requestTimeout) {
107
+ if (instance && (options || requestTimeout)) {
108
+ console.warn("getMcpHost: options and requestTimeout ignored, instance already exists");
109
+ }
110
+ if (!instance) {
111
+ instance = new McpAppBridge({ ...defaultOptions, ...options }, requestTimeout);
112
+ }
113
+ return instance;
114
+ }
115
+ export function useMcpAppBridge(key, options, requestTimeout) {
116
+ const host = getMcpHost(options, requestTimeout);
117
+ return useSyncExternalStore(host.subscribe(key), () => host.context?.[key]);
118
+ }
119
+ //# sourceMappingURL=use-mcp-app-bridge.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"use-mcp-app-bridge.js","sourceRoot":"","sources":["../../../../src/web/hooks/use-mcp-app-bridge.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,62 @@
1
+ import { act, renderHook, waitFor } from "@testing-library/react";
2
+ import { afterEach, beforeEach, describe, expect, it, vi } from "vitest";
3
+ describe("useMcpAppBridge", () => {
4
+ let mockPostMessage;
5
+ const importModule = async () => {
6
+ return import("./use-mcp-app-bridge.js");
7
+ };
8
+ beforeEach(() => {
9
+ vi.resetModules();
10
+ mockPostMessage = vi.fn();
11
+ Object.defineProperty(window, "parent", {
12
+ value: { postMessage: mockPostMessage },
13
+ writable: true,
14
+ configurable: true,
15
+ });
16
+ });
17
+ afterEach(() => {
18
+ vi.clearAllMocks();
19
+ });
20
+ it("should return the theme value from host context", async () => {
21
+ const { useMcpAppBridge } = await importModule();
22
+ const { result } = renderHook(() => useMcpAppBridge("theme"));
23
+ const initCall = mockPostMessage.mock.calls.find((call) => call[0].method === "ui/initialize");
24
+ if (initCall) {
25
+ act(() => {
26
+ window.dispatchEvent(new MessageEvent("message", {
27
+ data: {
28
+ jsonrpc: "2.0",
29
+ id: initCall[0].id,
30
+ result: {
31
+ protocolVersion: "2025-06-18",
32
+ hostInfo: { name: "test-host", version: "1.0.0" },
33
+ hostCapabilities: {},
34
+ hostContext: { theme: "light" },
35
+ },
36
+ },
37
+ }));
38
+ });
39
+ }
40
+ await waitFor(() => {
41
+ expect(result.current).toBe("light");
42
+ });
43
+ });
44
+ it("should reject the request after timeout", async () => {
45
+ vi.useFakeTimers();
46
+ const consoleErrorSpy = vi
47
+ .spyOn(console, "error")
48
+ .mockImplementation(() => { });
49
+ const { useMcpAppBridge, getMcpHost } = await importModule();
50
+ const bridge = getMcpHost({ appInfo: { name: "test", version: "1.0.0" } }, 100);
51
+ renderHook(() => useMcpAppBridge("theme"));
52
+ expect(mockPostMessage).toHaveBeenCalledWith(expect.objectContaining({ method: "ui/initialize" }), "*");
53
+ await act(async () => {
54
+ await vi.advanceTimersByTimeAsync(100);
55
+ });
56
+ expect(consoleErrorSpy).toHaveBeenCalledWith(new Error("Request timed out"));
57
+ consoleErrorSpy.mockRestore();
58
+ vi.useRealTimers();
59
+ bridge.cleanup();
60
+ });
61
+ });
62
+ //# sourceMappingURL=use-mcp-app-bridge.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"use-mcp-app-bridge.test.js","sourceRoot":"","sources":["../../../../src/web/hooks/use-mcp-app-bridge.test.ts"],"names":[],"mappings":"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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "skybridge",
3
- "version": "0.0.0-dev.55ae9f7",
3
+ "version": "0.0.0-dev.570aaaa",
4
4
  "description": "Skybridge is a framework for building ChatGPT apps",
5
5
  "type": "module",
6
6
  "files": [
@@ -52,6 +52,7 @@
52
52
  },
53
53
  "devDependencies": {
54
54
  "@biomejs/biome": "2.3.8",
55
+ "@modelcontextprotocol/ext-apps": "^0.2.2",
55
56
  "@modelcontextprotocol/sdk": "^1.24.3",
56
57
  "@testing-library/dom": "^10.4.1",
57
58
  "@testing-library/react": "^16.3.0",