@mcp-ts/sdk 1.3.2 → 1.3.4

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 (58) hide show
  1. package/README.md +400 -406
  2. package/dist/adapters/agui-adapter.d.mts +1 -1
  3. package/dist/adapters/agui-adapter.d.ts +1 -1
  4. package/dist/adapters/agui-middleware.d.mts +1 -1
  5. package/dist/adapters/agui-middleware.d.ts +1 -1
  6. package/dist/adapters/ai-adapter.d.mts +1 -1
  7. package/dist/adapters/ai-adapter.d.ts +1 -1
  8. package/dist/adapters/langchain-adapter.d.mts +1 -1
  9. package/dist/adapters/langchain-adapter.d.ts +1 -1
  10. package/dist/adapters/mastra-adapter.d.mts +1 -1
  11. package/dist/adapters/mastra-adapter.d.ts +1 -1
  12. package/dist/client/index.d.mts +8 -64
  13. package/dist/client/index.d.ts +8 -64
  14. package/dist/client/index.js +91 -173
  15. package/dist/client/index.js.map +1 -1
  16. package/dist/client/index.mjs +91 -173
  17. package/dist/client/index.mjs.map +1 -1
  18. package/dist/client/react.d.mts +12 -2
  19. package/dist/client/react.d.ts +12 -2
  20. package/dist/client/react.js +119 -182
  21. package/dist/client/react.js.map +1 -1
  22. package/dist/client/react.mjs +119 -182
  23. package/dist/client/react.mjs.map +1 -1
  24. package/dist/client/vue.d.mts +24 -4
  25. package/dist/client/vue.d.ts +24 -4
  26. package/dist/client/vue.js +121 -182
  27. package/dist/client/vue.js.map +1 -1
  28. package/dist/client/vue.mjs +121 -182
  29. package/dist/client/vue.mjs.map +1 -1
  30. package/dist/index.d.mts +2 -2
  31. package/dist/index.d.ts +2 -2
  32. package/dist/index.js +215 -250
  33. package/dist/index.js.map +1 -1
  34. package/dist/index.mjs +215 -250
  35. package/dist/index.mjs.map +1 -1
  36. package/dist/{multi-session-client-B1DBx5yR.d.mts → multi-session-client-DzjmT7FX.d.mts} +1 -0
  37. package/dist/{multi-session-client-DyFzyJUx.d.ts → multi-session-client-FAFpUzZ4.d.ts} +1 -0
  38. package/dist/server/index.d.mts +16 -21
  39. package/dist/server/index.d.ts +16 -21
  40. package/dist/server/index.js +124 -77
  41. package/dist/server/index.js.map +1 -1
  42. package/dist/server/index.mjs +124 -77
  43. package/dist/server/index.mjs.map +1 -1
  44. package/dist/shared/index.d.mts +2 -2
  45. package/dist/shared/index.d.ts +2 -2
  46. package/dist/shared/index.js.map +1 -1
  47. package/dist/shared/index.mjs.map +1 -1
  48. package/dist/{types-PjM1W07s.d.mts → types-CW6lghof.d.mts} +5 -0
  49. package/dist/{types-PjM1W07s.d.ts → types-CW6lghof.d.ts} +5 -0
  50. package/package.json +1 -1
  51. package/src/client/core/sse-client.ts +354 -493
  52. package/src/client/react/use-mcp.ts +75 -23
  53. package/src/client/vue/use-mcp.ts +111 -48
  54. package/src/server/handlers/nextjs-handler.ts +207 -217
  55. package/src/server/handlers/sse-handler.ts +10 -0
  56. package/src/server/mcp/oauth-client.ts +41 -32
  57. package/src/server/storage/types.ts +12 -5
  58. package/src/shared/types.ts +5 -0
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/client/core/sse-client.ts","../../src/client/core/app-host.ts"],"names":["nanoid","AppBridge","PostMessageTransport"],"mappings":"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","file":"index.js","sourcesContent":["/**\r\n * SSE Client for MCP Connections\r\n *\r\n * Browser-side client that manages real-time communication with the MCP server\r\n * using Server-Sent Events (SSE) for server→client streaming and HTTP POST for\r\n * client→server RPC requests.\r\n *\r\n * Key features:\r\n * - Direct HTTP response for RPC calls (bypasses SSE latency)\r\n * - Resource preloading for instant MCP App UI loading\r\n * - Automatic reconnection with exponential backoff\r\n * - Type-safe RPC methods\r\n */\r\n\r\nimport { nanoid } from 'nanoid';\r\nimport type {\r\n McpConnectionEvent,\r\n McpObservabilityEvent,\r\n McpAppsUIEvent\r\n} from '../../shared/events.js';\r\nimport type {\r\n McpRpcRequest,\r\n McpRpcResponse,\r\n McpRpcMethod,\r\n McpRpcParams,\r\n ConnectParams,\r\n SessionListResult,\r\n ConnectResult,\r\n DisconnectResult,\r\n RestoreSessionResult,\r\n FinishAuthResult,\r\n ListToolsRpcResult,\r\n ListPromptsResult,\r\n ListResourcesResult,\r\n} from '../../shared/types.js';\r\n// ============================================\r\n// Types & Interfaces\r\n// ============================================\r\n\r\nexport interface SSEClientOptions {\r\n /** SSE endpoint URL */\r\n url: string;\r\n\r\n /** User/Client identifier */\r\n identity: string;\r\n\r\n /** Optional auth token for authenticated requests */\r\n authToken?: string;\r\n\r\n /** Callback for MCP connection state changes */\r\n onConnectionEvent?: (event: McpConnectionEvent) => void;\r\n\r\n /** Callback for observability/logging events */\r\n onObservabilityEvent?: (event: McpObservabilityEvent) => void;\r\n\r\n /** Callback for connection status changes */\r\n onStatusChange?: (status: ConnectionStatus) => void;\r\n\r\n /** Callback for MCP App UI events */\r\n onEvent?: (event: McpAppsUIEvent) => void;\r\n\r\n /** Request timeout in milliseconds @default 60000 */\r\n requestTimeout?: number;\r\n\r\n /** Enable debug logging @default false */\r\n debug?: boolean;\r\n}\r\n\r\nexport type ConnectionStatus = 'connecting' | 'connected' | 'disconnected' | 'error';\r\n\r\ninterface PendingRequest {\r\n resolve: (value: unknown) => void;\r\n reject: (error: Error) => void;\r\n timeoutId: ReturnType<typeof setTimeout>;\r\n}\r\n\r\ninterface ToolUiMetadata {\r\n resourceUri?: string;\r\n uri?: string;\r\n visibility?: string[];\r\n}\r\n\r\n// ============================================\r\n// Constants\r\n// ============================================\r\n\r\nconst DEFAULT_REQUEST_TIMEOUT = 60000;\r\nconst MAX_RECONNECT_ATTEMPTS = 5;\r\nconst BASE_RECONNECT_DELAY = 1000;\r\n\r\n// ============================================\r\n// SSEClient Class\r\n// ============================================\r\n\r\n/**\r\n * SSE Client for real-time MCP connection management\r\n */\r\nexport class SSEClient {\r\n private eventSource: EventSource | null = null;\r\n private pendingRequests = new Map<string, PendingRequest>();\r\n private resourceCache = new Map<string, Promise<unknown>>();\r\n\r\n private reconnectAttempts = 0;\r\n private isManuallyDisconnected = false;\r\n private connectionPromise: Promise<void> | null = null;\r\n private connectionResolver: (() => void) | null = null;\r\n\r\n constructor(private readonly options: SSEClientOptions) {}\r\n\r\n // ============================================\r\n // Connection Management\r\n // ============================================\r\n\r\n /**\r\n * Connect to the SSE endpoint\r\n */\r\n connect(): void {\r\n if (this.eventSource) {\r\n return; // Already connected\r\n }\r\n\r\n this.isManuallyDisconnected = false;\r\n this.options.onStatusChange?.('connecting');\r\n this.connectionPromise = new Promise((resolve) => {\r\n this.connectionResolver = resolve;\r\n });\r\n\r\n const url = this.buildUrl();\r\n this.eventSource = new EventSource(url);\r\n this.setupEventListeners();\r\n }\r\n\r\n /**\r\n * Disconnect from the SSE endpoint\r\n */\r\n disconnect(): void {\r\n this.isManuallyDisconnected = true;\r\n\r\n if (this.eventSource) {\r\n this.eventSource.close();\r\n this.eventSource = null;\r\n }\r\n\r\n this.connectionPromise = null;\r\n this.connectionResolver = null;\r\n this.rejectAllPendingRequests(new Error('Connection closed'));\r\n this.options.onStatusChange?.('disconnected');\r\n }\r\n\r\n /**\r\n * Check if connected to the SSE endpoint\r\n */\r\n isConnected(): boolean {\r\n return this.eventSource?.readyState === EventSource.OPEN;\r\n }\r\n\r\n // ============================================\r\n // RPC Methods\r\n // ============================================\r\n\r\n async getSessions(): Promise<SessionListResult> {\r\n return this.sendRequest<SessionListResult>('getSessions');\r\n }\r\n\r\n async connectToServer(params: ConnectParams): Promise<ConnectResult> {\r\n return this.sendRequest<ConnectResult>('connect', params);\r\n }\r\n\r\n async disconnectFromServer(sessionId: string): Promise<DisconnectResult> {\r\n return this.sendRequest<DisconnectResult>('disconnect', { sessionId });\r\n }\r\n\r\n async listTools(sessionId: string): Promise<ListToolsRpcResult> {\r\n return this.sendRequest<ListToolsRpcResult>('listTools', { sessionId });\r\n }\r\n\r\n async callTool(\r\n sessionId: string,\r\n toolName: string,\r\n toolArgs: Record<string, unknown>\r\n ): Promise<unknown> {\r\n const result = await this.sendRequest('callTool', { sessionId, toolName, toolArgs });\r\n this.emitUiEventIfPresent(result, sessionId, toolName);\r\n return result;\r\n }\r\n\r\n async restoreSession(sessionId: string): Promise<RestoreSessionResult> {\r\n return this.sendRequest<RestoreSessionResult>('restoreSession', { sessionId });\r\n }\r\n\r\n async finishAuth(sessionId: string, code: string): Promise<FinishAuthResult> {\r\n return this.sendRequest<FinishAuthResult>('finishAuth', { sessionId, code });\r\n }\r\n\r\n async listPrompts(sessionId: string): Promise<ListPromptsResult> {\r\n return this.sendRequest<ListPromptsResult>('listPrompts', { sessionId });\r\n }\r\n\r\n async getPrompt(sessionId: string, name: string, args?: Record<string, string>): Promise<unknown> {\r\n return this.sendRequest('getPrompt', { sessionId, name, args });\r\n }\r\n\r\n async listResources(sessionId: string): Promise<ListResourcesResult> {\r\n return this.sendRequest<ListResourcesResult>('listResources', { sessionId });\r\n }\r\n\r\n async readResource(sessionId: string, uri: string): Promise<unknown> {\r\n return this.sendRequest('readResource', { sessionId, uri });\r\n }\r\n\r\n // ============================================\r\n // Resource Preloading (for instant UI loading)\r\n // ============================================\r\n\r\n /**\r\n * Preload UI resources for tools that have UI metadata.\r\n * Call this when tools are discovered to enable instant MCP App UI loading.\r\n */\r\n preloadToolUiResources(sessionId: string, tools: Array<{ name: string; _meta?: unknown }>): void {\r\n for (const tool of tools) {\r\n const uri = this.extractUiResourceUri(tool);\r\n if (!uri) continue;\r\n\r\n if (this.resourceCache.has(uri)) {\r\n this.log(`Resource already cached: ${uri}`);\r\n continue;\r\n }\r\n\r\n this.log(`Preloading UI resource for tool \"${tool.name}\": ${uri}`);\r\n const promise = this.sendRequest('readResource', { sessionId, uri })\r\n .catch((err) => {\r\n this.log(`Failed to preload resource ${uri}: ${err.message}`, 'warn');\r\n this.resourceCache.delete(uri);\r\n return null;\r\n });\r\n\r\n this.resourceCache.set(uri, promise);\r\n }\r\n }\r\n\r\n /**\r\n * Get a preloaded resource from cache, or fetch if not cached.\r\n */\r\n getOrFetchResource(sessionId: string, uri: string): Promise<unknown> {\r\n const cached = this.resourceCache.get(uri);\r\n if (cached) {\r\n this.log(`Cache hit for resource: ${uri}`);\r\n return cached;\r\n }\r\n\r\n this.log(`Cache miss, fetching resource: ${uri}`);\r\n const promise = this.sendRequest('readResource', { sessionId, uri });\r\n this.resourceCache.set(uri, promise);\r\n return promise;\r\n }\r\n\r\n /**\r\n * Check if a resource is already cached\r\n */\r\n hasPreloadedResource(uri: string): boolean {\r\n return this.resourceCache.has(uri);\r\n }\r\n\r\n /**\r\n * Clear the resource cache\r\n */\r\n clearResourceCache(): void {\r\n this.resourceCache.clear();\r\n }\r\n\r\n // ============================================\r\n // Private: Request Handling\r\n // ============================================\r\n\r\n /**\r\n * Send an RPC request and return the response directly from HTTP.\r\n * This bypasses SSE latency by returning results in the HTTP response body.\r\n */\r\n private async sendRequest<T = unknown>(method: McpRpcMethod, params?: McpRpcParams): Promise<T> {\r\n if (this.connectionPromise) {\r\n await this.connectionPromise;\r\n }\r\n\r\n const request: McpRpcRequest = {\r\n id: `rpc_${nanoid(10)}`,\r\n method,\r\n params,\r\n };\r\n\r\n const response = await fetch(this.buildUrl(), {\r\n method: 'POST',\r\n headers: this.buildHeaders(),\r\n body: JSON.stringify(request),\r\n });\r\n\r\n if (!response.ok) {\r\n throw new Error(`HTTP ${response.status}: ${response.statusText}`);\r\n }\r\n\r\n const data = await response.json() as McpRpcResponse;\r\n return this.parseRpcResponse<T>(data, request.id);\r\n }\r\n\r\n /**\r\n * Parse RPC response and handle different response formats\r\n */\r\n private parseRpcResponse<T>(data: McpRpcResponse, requestId: string): T | Promise<T> {\r\n // Fast path: Direct response (new behavior)\r\n if ('result' in data) {\r\n return data.result as T;\r\n }\r\n\r\n // Error response\r\n if ('error' in data && data.error) {\r\n throw new Error(data.error.message || 'Unknown RPC error');\r\n }\r\n\r\n // Legacy path: Acknowledgment only (wait for SSE)\r\n // Kept for backwards compatibility with older servers\r\n if ('acknowledged' in data) {\r\n return this.waitForSseResponse<T>(requestId);\r\n }\r\n\r\n throw new Error('Invalid RPC response format');\r\n }\r\n\r\n /**\r\n * Wait for RPC response via SSE (legacy fallback)\r\n */\r\n private waitForSseResponse<T>(requestId: string): Promise<T> {\r\n const timeoutMs = this.options.requestTimeout ?? DEFAULT_REQUEST_TIMEOUT;\r\n\r\n return new Promise<T>((resolve, reject) => {\r\n const timeoutId = setTimeout(() => {\r\n this.pendingRequests.delete(requestId);\r\n reject(new Error(`Request timeout after ${timeoutMs}ms`));\r\n }, timeoutMs);\r\n\r\n this.pendingRequests.set(requestId, {\r\n resolve: resolve as (value: unknown) => void,\r\n reject,\r\n timeoutId,\r\n });\r\n });\r\n }\r\n\r\n /**\r\n * Handle RPC response received via SSE (legacy)\r\n */\r\n private handleRpcResponse(response: McpRpcResponse): void {\r\n const pending = this.pendingRequests.get(response.id);\r\n if (!pending) return;\r\n\r\n clearTimeout(pending.timeoutId);\r\n this.pendingRequests.delete(response.id);\r\n\r\n if (response.error) {\r\n pending.reject(new Error(response.error.message));\r\n } else {\r\n pending.resolve(response.result);\r\n }\r\n }\r\n\r\n // ============================================\r\n // Private: Event Handling\r\n // ============================================\r\n\r\n private setupEventListeners(): void {\r\n if (!this.eventSource) return;\r\n\r\n this.eventSource.addEventListener('open', () => {\r\n this.log('Connected');\r\n this.reconnectAttempts = 0;\r\n this.options.onStatusChange?.('connected');\r\n });\r\n\r\n this.eventSource.addEventListener('connected', () => {\r\n this.log('Server ready');\r\n this.connectionResolver?.();\r\n this.connectionResolver = null;\r\n });\r\n\r\n this.eventSource.addEventListener('connection', (e: MessageEvent) => {\r\n const event = JSON.parse(e.data) as McpConnectionEvent;\r\n this.options.onConnectionEvent?.(event);\r\n });\r\n\r\n this.eventSource.addEventListener('observability', (e: MessageEvent) => {\r\n const event = JSON.parse(e.data) as McpObservabilityEvent;\r\n this.options.onObservabilityEvent?.(event);\r\n });\r\n\r\n this.eventSource.addEventListener('rpc-response', (e: MessageEvent) => {\r\n const response = JSON.parse(e.data) as McpRpcResponse;\r\n this.handleRpcResponse(response);\r\n });\r\n\r\n this.eventSource.addEventListener('error', () => {\r\n this.log('Connection error', 'error');\r\n this.options.onStatusChange?.('error');\r\n this.attemptReconnect();\r\n });\r\n }\r\n\r\n private attemptReconnect(): void {\r\n if (this.isManuallyDisconnected || this.reconnectAttempts >= MAX_RECONNECT_ATTEMPTS) {\r\n return;\r\n }\r\n\r\n this.reconnectAttempts++;\r\n const delay = BASE_RECONNECT_DELAY * this.reconnectAttempts;\r\n this.log(`Reconnecting in ${delay}ms (attempt ${this.reconnectAttempts}/${MAX_RECONNECT_ATTEMPTS})`);\r\n\r\n setTimeout(() => {\r\n this.disconnect();\r\n this.connect();\r\n }, delay);\r\n }\r\n\r\n // ============================================\r\n // Private: Utilities\r\n // ============================================\r\n\r\n private buildUrl(): string {\r\n const url = new URL(this.options.url, globalThis.location?.origin);\r\n url.searchParams.set('identity', this.options.identity);\r\n if (this.options.authToken) {\r\n url.searchParams.set('token', this.options.authToken);\r\n }\r\n return url.toString();\r\n }\r\n\r\n private buildHeaders(): HeadersInit {\r\n const headers: HeadersInit = {\r\n 'Content-Type': 'application/json',\r\n };\r\n if (this.options.authToken) {\r\n headers['Authorization'] = `Bearer ${this.options.authToken}`;\r\n }\r\n return headers;\r\n }\r\n\r\n private rejectAllPendingRequests(error: Error): void {\r\n for (const [, pending] of this.pendingRequests) {\r\n clearTimeout(pending.timeoutId);\r\n pending.reject(error);\r\n }\r\n this.pendingRequests.clear();\r\n }\r\n\r\n private extractUiResourceUri(tool: { name: string; _meta?: unknown }): string | undefined {\r\n const meta = (tool._meta as { ui?: ToolUiMetadata })?.ui;\r\n if (!meta || typeof meta !== 'object') return undefined;\r\n\r\n // Check visibility constraint\r\n if (meta.visibility && !meta.visibility.includes('app')) return undefined;\r\n\r\n // Support both 'resourceUri' and 'uri' field names\r\n return meta.resourceUri ?? meta.uri;\r\n }\r\n\r\n private emitUiEventIfPresent(result: unknown, sessionId: string, toolName: string): void {\r\n const meta = (result as { _meta?: { ui?: ToolUiMetadata } })?._meta;\r\n const resourceUri = meta?.ui?.resourceUri ?? (meta as any)?.['ui/resourceUri'];\r\n\r\n if (resourceUri) {\r\n this.options.onEvent?.({\r\n type: 'mcp-apps-ui',\r\n sessionId,\r\n resourceUri,\r\n toolName,\r\n result,\r\n timestamp: Date.now(),\r\n });\r\n }\r\n }\r\n\r\n private log(message: string, level: 'info' | 'warn' | 'error' = 'info'): void {\r\n if (!this.options.debug && level === 'info') return;\r\n\r\n const prefix = '[SSEClient]';\r\n switch (level) {\r\n case 'warn':\r\n console.warn(prefix, message);\r\n break;\r\n case 'error':\r\n console.error(prefix, message);\r\n break;\r\n default:\r\n console.log(prefix, message);\r\n }\r\n }\r\n}\r\n","/**\r\n * MCP App Host\r\n *\r\n * Bridges the gap between an iframe (MCP App) and the SSEClient (MCP Server).\r\n * Handles secure iframe sandboxing, resource loading, and bi-directional\r\n * communication via the AppBridge protocol.\r\n *\r\n * Key features:\r\n * - Secure iframe sandboxing with minimal permissions\r\n * - Resource preloading for instant MCP App UI loading\r\n * - Cache-aware resource fetching (SSEClient cache → local cache → direct fetch)\r\n * - Support for ui:// and mcp-app:// resource URIs\r\n */\r\n\r\nimport { AppBridge, PostMessageTransport } from '@modelcontextprotocol/ext-apps/app-bridge';\r\nimport type { AppHostClient } from './types';\r\n\r\n// ============================================\r\n// Types & Interfaces\r\n// ============================================\r\n\r\nexport interface AppHostOptions {\r\n /** Enable debug logging @default false */\r\n debug?: boolean;\r\n}\r\n\r\nexport interface AppMessageParams {\r\n role: string;\r\n content: unknown;\r\n}\r\n\r\ninterface ToolCallParams {\r\n name: string;\r\n arguments?: Record<string, unknown>;\r\n}\r\n\r\ninterface ResourceContent {\r\n blob?: string;\r\n text?: string;\r\n}\r\n\r\ninterface ResourceResponse {\r\n contents: ResourceContent[];\r\n}\r\n\r\n// ============================================\r\n// Constants\r\n// ============================================\r\n\r\nconst HOST_INFO = { name: 'mcp-ts-host', version: '1.0.0' };\r\n\r\n/** Sandbox permissions - minimal set required for MCP Apps to function */\r\nconst SANDBOX_PERMISSIONS = [\r\n 'allow-scripts', // Required for app JavaScript execution\r\n 'allow-forms', // Required for form submissions\r\n 'allow-same-origin', // Required for Blob URL correctness\r\n 'allow-modals', // Required for dialogs/alerts\r\n 'allow-popups', // Required for opening links\r\n 'allow-downloads' // Required for file downloads\r\n].join(' ');\r\n\r\n/** Supported MCP App URI schemes */\r\nconst MCP_URI_SCHEMES = ['ui://', 'mcp-app://'] as const;\r\n\r\n// ============================================\r\n// AppHost Class\r\n// ============================================\r\n\r\n/**\r\n * Host for MCP Apps embedded in iframes.\r\n * Manages secure communication between the app and the MCP server.\r\n */\r\nexport class AppHost {\r\n private bridge: AppBridge;\r\n private sessionId?: string;\r\n private resourceCache = new Map<string, Promise<ResourceResponse | null>>();\r\n private debug: boolean;\r\n\r\n /** Callback for app messages (e.g., chat messages from the app) */\r\n public onAppMessage?: (params: AppMessageParams) => void;\r\n\r\n constructor(\r\n private readonly client: AppHostClient,\r\n private readonly iframe: HTMLIFrameElement,\r\n options?: AppHostOptions\r\n ) {\r\n this.debug = options?.debug ?? false;\r\n this.configureSandbox();\r\n this.bridge = this.initializeBridge();\r\n }\r\n\r\n // ============================================\r\n // Public API\r\n // ============================================\r\n\r\n /**\r\n * Start the host. This prepares the bridge handlers but doesn't connect yet.\r\n * The actual connection happens in launch() after HTML is loaded.\r\n * @returns Promise that resolves immediately (bridge connects during launch)\r\n */\r\n async start(): Promise<void> {\r\n // Bridge handlers are already registered in constructor.\r\n // Connection happens in launch() after HTML is loaded.\r\n this.log('Host started, ready to launch');\r\n }\r\n\r\n /**\r\n * Preload UI resources to enable instant app loading.\r\n * Call this when tools are discovered to cache their UI resources.\r\n */\r\n preload(tools: Array<{ _meta?: unknown }>): void {\r\n for (const tool of tools) {\r\n const uri = this.extractUiResourceUri(tool);\r\n if (!uri || this.resourceCache.has(uri)) continue;\r\n\r\n const promise = this.preloadResource(uri);\r\n this.resourceCache.set(uri, promise);\r\n }\r\n }\r\n\r\n /**\r\n * Launch an MCP App from a URL or MCP resource URI.\r\n * Loads the HTML first, then establishes bridge connection.\r\n */\r\n async launch(url: string, sessionId?: string): Promise<void> {\r\n if (sessionId) this.sessionId = sessionId;\r\n\r\n // Set up initialization promise BEFORE connecting\r\n const initializedPromise = this.onAppReady();\r\n\r\n // Load HTML into iframe first\r\n if (this.isMcpUri(url)) {\r\n await this.launchMcpApp(url);\r\n } else {\r\n this.iframe.src = url;\r\n }\r\n\r\n // Wait for iframe to load before connecting bridge\r\n await this.onIframeReady();\r\n\r\n // Connect the bridge (HTML is loaded, contentWindow is ready)\r\n await this.connectBridge();\r\n\r\n // Wait for app to signal it's initialized (with timeout)\r\n this.log('Waiting for app initialization');\r\n await Promise.race([\r\n initializedPromise,\r\n new Promise<void>((resolve) => setTimeout(() => {\r\n this.log('Initialization timeout - continuing anyway', 'warn');\r\n resolve();\r\n }, 3000))\r\n ]);\r\n this.log('App launched and ready');\r\n }\r\n\r\n /**\r\n * Wait for app to signal initialization complete\r\n */\r\n private onAppReady(): Promise<void> {\r\n return new Promise<void>((resolve) => {\r\n const originalHandler = this.bridge.oninitialized;\r\n this.bridge.oninitialized = (...args) => {\r\n this.log('App initialized');\r\n resolve();\r\n this.bridge.oninitialized = originalHandler;\r\n originalHandler?.(...args);\r\n };\r\n });\r\n }\r\n\r\n /**\r\n * Wait for iframe to finish loading\r\n */\r\n private onIframeReady(): Promise<void> {\r\n return new Promise((resolve) => {\r\n if (this.iframe.contentDocument?.readyState === 'complete') {\r\n resolve();\r\n return;\r\n }\r\n this.iframe.addEventListener('load', () => resolve(), { once: true });\r\n });\r\n }\r\n\r\n /**\r\n * Send tool input arguments to the MCP App.\r\n * Call this after launch() when tool input is available.\r\n */\r\n sendToolInput(args: Record<string, unknown>): void {\r\n this.log('Sending tool input to app');\r\n this.bridge.sendToolInput({ arguments: args });\r\n }\r\n\r\n /**\r\n * Send tool result to the MCP App.\r\n * Call this when the tool call completes.\r\n */\r\n sendToolResult(result: unknown): void {\r\n this.log('Sending tool result to app');\r\n this.bridge.sendToolResult(result as any);\r\n }\r\n\r\n /**\r\n * Send tool cancellation to the MCP App.\r\n * Call this when the tool call is cancelled or fails.\r\n */\r\n sendToolCancelled(reason: string): void {\r\n this.log('Sending tool cancellation to app');\r\n this.bridge.sendToolCancelled({ reason });\r\n }\r\n\r\n // ============================================\r\n // Private: Initialization\r\n // ============================================\r\n\r\n private configureSandbox(): void {\r\n if (this.iframe.sandbox.value !== SANDBOX_PERMISSIONS) {\r\n this.iframe.sandbox.value = SANDBOX_PERMISSIONS;\r\n }\r\n }\r\n\r\n private initializeBridge(): AppBridge {\r\n const bridge = new AppBridge(\r\n null,\r\n HOST_INFO,\r\n {\r\n openLinks: {},\r\n serverTools: {},\r\n logging: {},\r\n // Declare support for model context updates\r\n updateModelContext: { text: {} },\r\n },\r\n {\r\n // Initial host context\r\n hostContext: {\r\n theme: 'dark',\r\n platform: 'web',\r\n containerDimensions: { maxHeight: 6000 },\r\n displayMode: 'inline',\r\n availableDisplayModes: ['inline', 'fullscreen'],\r\n },\r\n }\r\n );\r\n\r\n // Register handlers - must be done BEFORE connect()\r\n bridge.oncalltool = (params) => this.handleToolCall(params);\r\n bridge.onopenlink = this.handleOpenLink.bind(this);\r\n bridge.onmessage = this.handleMessage.bind(this);\r\n bridge.onloggingmessage = (params) => this.log(`App log [${params.level}]: ${params.data}`);\r\n bridge.onupdatemodelcontext = async () => ({});\r\n bridge.onsizechange = async ({ width, height }) => {\r\n if (height !== undefined) this.iframe.style.height = `${height}px`;\r\n if (width !== undefined) this.iframe.style.minWidth = `min(${width}px, 100%)`;\r\n return {};\r\n };\r\n bridge.onrequestdisplaymode = async (params) => ({\r\n mode: params.mode === 'fullscreen' ? 'fullscreen' : 'inline'\r\n });\r\n\r\n return bridge;\r\n }\r\n\r\n private async connectBridge(): Promise<void> {\r\n this.log('Connecting bridge to iframe');\r\n\r\n const transport = new PostMessageTransport(\r\n this.iframe.contentWindow!,\r\n this.iframe.contentWindow!\r\n );\r\n\r\n try {\r\n await this.bridge.connect(transport);\r\n this.log('Bridge connected successfully');\r\n } catch (error) {\r\n this.log('Bridge connection failed', 'error');\r\n throw error;\r\n }\r\n }\r\n\r\n // ============================================\r\n // Private: Bridge Event Handlers\r\n // ============================================\r\n\r\n private async handleToolCall(params: ToolCallParams) {\r\n if (!this.client.isConnected()) {\r\n throw new Error('Client disconnected');\r\n }\r\n\r\n const sessionId = await this.getSessionId();\r\n if (!sessionId) {\r\n throw new Error('No active session');\r\n }\r\n\r\n const result = await this.client.callTool(\r\n sessionId,\r\n params.name,\r\n params.arguments ?? {}\r\n );\r\n return result as any;\r\n }\r\n\r\n private async handleOpenLink(params: { url: string }): Promise<Record<string, never>> {\r\n window.open(params.url, '_blank', 'noopener,noreferrer');\r\n return {};\r\n }\r\n\r\n private async handleMessage(params: AppMessageParams): Promise<Record<string, never>> {\r\n this.onAppMessage?.(params);\r\n return {};\r\n }\r\n\r\n // ============================================\r\n // Private: Resource Loading\r\n // ============================================\r\n\r\n private async launchMcpApp(uri: string): Promise<void> {\r\n if (!this.client.isConnected()) {\r\n throw new Error('Client must be connected');\r\n }\r\n\r\n const sessionId = await this.getSessionId();\r\n if (!sessionId) {\r\n throw new Error('No active session');\r\n }\r\n\r\n // Fetch resource using cache hierarchy: SSEClient cache → local cache → direct fetch\r\n const response = await this.fetchResourceWithCache(sessionId, uri);\r\n if (!response?.contents?.length) {\r\n throw new Error(`Empty resource: ${uri}`);\r\n }\r\n\r\n const content = response.contents[0];\r\n const html = this.decodeContent(content);\r\n if (!html) {\r\n throw new Error(`Invalid content in resource: ${uri}`);\r\n }\r\n\r\n // Render via Blob URL for clean isolation\r\n const blob = new Blob([html], { type: 'text/html' });\r\n this.iframe.src = URL.createObjectURL(blob);\r\n }\r\n\r\n private async fetchResourceWithCache(sessionId: string, uri: string): Promise<ResourceResponse> {\r\n // Priority 1: SSEClient's built-in cache (best performance)\r\n if (this.hasClientCache()) {\r\n return (this.client as any).getOrFetchResource(sessionId, uri);\r\n }\r\n\r\n // Priority 2: Local preload cache\r\n const cached = this.resourceCache.get(uri);\r\n if (cached) {\r\n const result = await cached;\r\n if (result) return result;\r\n }\r\n\r\n // Priority 3: Direct fetch\r\n return this.client.readResource(sessionId, uri) as Promise<ResourceResponse>;\r\n }\r\n\r\n private async preloadResource(uri: string): Promise<ResourceResponse | null> {\r\n try {\r\n const sessionId = await this.getSessionId();\r\n if (!sessionId) return null;\r\n return await this.client.readResource(sessionId, uri) as ResourceResponse;\r\n } catch (error) {\r\n this.log(`Preload failed for ${uri}`, 'warn');\r\n return null;\r\n }\r\n }\r\n\r\n // ============================================\r\n // Private: Utilities\r\n // ============================================\r\n\r\n private async getSessionId(): Promise<string | undefined> {\r\n if (this.sessionId) return this.sessionId;\r\n const result = await this.client.getSessions();\r\n return result.sessions?.[0]?.sessionId;\r\n }\r\n\r\n private isMcpUri(url: string): boolean {\r\n return MCP_URI_SCHEMES.some(scheme => url.startsWith(scheme));\r\n }\r\n\r\n private hasClientCache(): boolean {\r\n return 'getOrFetchResource' in this.client &&\r\n typeof (this.client as any).getOrFetchResource === 'function';\r\n }\r\n\r\n private extractUiResourceUri(tool: { _meta?: unknown }): string | undefined {\r\n const meta = tool._meta as { ui?: { resourceUri?: string; uri?: string } } | undefined;\r\n if (!meta?.ui) return undefined;\r\n return meta.ui.resourceUri ?? meta.ui.uri;\r\n }\r\n\r\n private decodeContent(content: ResourceContent): string | undefined {\r\n if (content.blob) {\r\n return atob(content.blob);\r\n }\r\n return content.text;\r\n }\r\n\r\n private log(message: string, level: 'info' | 'warn' | 'error' = 'info'): void {\r\n if (!this.debug && level === 'info') return;\r\n\r\n const prefix = '[AppHost]';\r\n switch (level) {\r\n case 'warn':\r\n console.warn(prefix, message);\r\n break;\r\n case 'error':\r\n console.error(prefix, message);\r\n break;\r\n default:\r\n console.log(prefix, message);\r\n }\r\n }\r\n}\r\n"]}
1
+ {"version":3,"sources":["../../src/client/core/sse-client.ts","../../src/client/core/app-host.ts"],"names":["nanoid","data","AppBridge","PostMessageTransport"],"mappings":"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","file":"index.js","sourcesContent":["/**\n * Stateless RPC-over-stream client for MCP connections.\n *\n * Uses single POST requests with `Accept: text/event-stream` for every RPC call.\n * Progress events and the final rpc-response are delivered in the same response.\n */\n\nimport { nanoid } from 'nanoid';\nimport type {\n McpConnectionEvent,\n McpObservabilityEvent,\n McpAppsUIEvent\n} from '../../shared/events.js';\nimport type {\n McpRpcRequest,\n McpRpcResponse,\n McpRpcMethod,\n McpRpcParams,\n ConnectParams,\n SessionListResult,\n ConnectResult,\n DisconnectResult,\n RestoreSessionResult,\n FinishAuthResult,\n ListToolsRpcResult,\n ListPromptsResult,\n ListResourcesResult,\n} from '../../shared/types.js';\n\nexport interface SSEClientOptions {\n /** MCP endpoint URL */\n url: string;\n\n /** User/Client identifier */\n identity: string;\n\n /** Optional auth token for authenticated requests */\n authToken?: string;\n\n /** Callback for MCP connection state changes */\n onConnectionEvent?: (event: McpConnectionEvent) => void;\n\n /** Callback for observability/logging events */\n onObservabilityEvent?: (event: McpObservabilityEvent) => void;\n\n /** Callback for connection status changes */\n onStatusChange?: (status: ConnectionStatus) => void;\n\n /** Callback for MCP App UI events */\n onEvent?: (event: McpAppsUIEvent) => void;\n\n /** Enable debug logging @default false */\n debug?: boolean;\n}\n\nexport type ConnectionStatus = 'connecting' | 'connected' | 'disconnected' | 'error';\n\ninterface ToolUiMetadata {\n resourceUri?: string;\n uri?: string;\n visibility?: string[];\n}\n\nexport class SSEClient {\n private resourceCache = new Map<string, Promise<unknown>>();\n private connected = false;\n\n constructor(private readonly options: SSEClientOptions) {}\n\n connect(): void {\n if (this.connected) {\n return;\n }\n this.connected = true;\n this.options.onStatusChange?.('connected');\n this.log('RPC mode: post_stream');\n }\n\n disconnect(): void {\n this.connected = false;\n this.options.onStatusChange?.('disconnected');\n }\n\n isConnected(): boolean {\n return this.connected;\n }\n\n async getSessions(): Promise<SessionListResult> {\n return this.sendRequest<SessionListResult>('getSessions');\n }\n\n async connectToServer(params: ConnectParams): Promise<ConnectResult> {\n return this.sendRequest<ConnectResult>('connect', params);\n }\n\n async disconnectFromServer(sessionId: string): Promise<DisconnectResult> {\n return this.sendRequest<DisconnectResult>('disconnect', { sessionId });\n }\n\n async listTools(sessionId: string): Promise<ListToolsRpcResult> {\n return this.sendRequest<ListToolsRpcResult>('listTools', { sessionId });\n }\n\n async callTool(\n sessionId: string,\n toolName: string,\n toolArgs: Record<string, unknown>\n ): Promise<unknown> {\n const result = await this.sendRequest('callTool', { sessionId, toolName, toolArgs });\n this.emitUiEventIfPresent(result, sessionId, toolName);\n return result;\n }\n\n async restoreSession(sessionId: string): Promise<RestoreSessionResult> {\n return this.sendRequest<RestoreSessionResult>('restoreSession', { sessionId });\n }\n\n async finishAuth(sessionId: string, code: string): Promise<FinishAuthResult> {\n return this.sendRequest<FinishAuthResult>('finishAuth', { sessionId, code });\n }\n\n async listPrompts(sessionId: string): Promise<ListPromptsResult> {\n return this.sendRequest<ListPromptsResult>('listPrompts', { sessionId });\n }\n\n async getPrompt(sessionId: string, name: string, args?: Record<string, string>): Promise<unknown> {\n return this.sendRequest('getPrompt', { sessionId, name, args });\n }\n\n async listResources(sessionId: string): Promise<ListResourcesResult> {\n return this.sendRequest<ListResourcesResult>('listResources', { sessionId });\n }\n\n async readResource(sessionId: string, uri: string): Promise<unknown> {\n return this.sendRequest('readResource', { sessionId, uri });\n }\n\n preloadToolUiResources(sessionId: string, tools: Array<{ name: string; _meta?: unknown }>): void {\n for (const tool of tools) {\n const uri = this.extractUiResourceUri(tool);\n if (!uri || this.resourceCache.has(uri)) continue;\n const promise = this.sendRequest('readResource', { sessionId, uri }).catch((err) => {\n this.log(`Failed to preload resource ${uri}: ${err.message}`, 'warn');\n this.resourceCache.delete(uri);\n return null;\n });\n this.resourceCache.set(uri, promise);\n }\n }\n\n getOrFetchResource(sessionId: string, uri: string): Promise<unknown> {\n const cached = this.resourceCache.get(uri);\n if (cached) return cached;\n const promise = this.sendRequest('readResource', { sessionId, uri });\n this.resourceCache.set(uri, promise);\n return promise;\n }\n\n hasPreloadedResource(uri: string): boolean {\n return this.resourceCache.has(uri);\n }\n\n clearResourceCache(): void {\n this.resourceCache.clear();\n }\n\n private async sendRequest<T = unknown>(method: McpRpcMethod, params?: McpRpcParams): Promise<T> {\n if (!this.connected) {\n this.connect();\n }\n\n this.log(`RPC request via post_stream: ${method}`);\n\n const request: McpRpcRequest = {\n id: `rpc_${nanoid(10)}`,\n method,\n params,\n };\n\n const response = await fetch(this.buildUrl(), {\n method: 'POST',\n headers: this.buildHeaders(),\n body: JSON.stringify(request),\n });\n\n if (!response.ok) {\n throw new Error(`HTTP ${response.status}: ${response.statusText}`);\n }\n\n const contentType = (response.headers.get('content-type') || '').toLowerCase();\n if (!contentType.includes('text/event-stream')) {\n const data = await response.json() as McpRpcResponse;\n return this.parseRpcResponse<T>(data);\n }\n\n const data = await this.readRpcResponseFromStream(response);\n return this.parseRpcResponse<T>(data);\n }\n\n private async readRpcResponseFromStream(response: Response): Promise<McpRpcResponse> {\n if (!response.body) {\n throw new Error('Streaming response body is missing');\n }\n\n const reader = response.body.getReader();\n const decoder = new TextDecoder();\n let buffer = '';\n let rpcResponse: McpRpcResponse | null = null;\n\n const dispatchBlock = (block: string) => {\n const lines = block.split('\\n');\n let eventName = 'message';\n const dataLines: string[] = [];\n\n for (const rawLine of lines) {\n const line = rawLine.replace(/\\r$/, '');\n if (!line || line.startsWith(':')) continue;\n if (line.startsWith('event:')) {\n eventName = line.slice('event:'.length).trim();\n continue;\n }\n if (line.startsWith('data:')) {\n dataLines.push(line.slice('data:'.length).trimStart());\n }\n }\n\n if (!dataLines.length) return;\n const payloadText = dataLines.join('\\n');\n let payload: unknown = payloadText;\n try {\n payload = JSON.parse(payloadText);\n } catch {\n // Keep raw text\n }\n\n switch (eventName) {\n case 'connected':\n this.options.onStatusChange?.('connected');\n break;\n case 'connection':\n this.options.onConnectionEvent?.(payload as McpConnectionEvent);\n break;\n case 'observability':\n this.options.onObservabilityEvent?.(payload as McpObservabilityEvent);\n break;\n case 'rpc-response':\n rpcResponse = payload as McpRpcResponse;\n break;\n default:\n break;\n }\n };\n\n while (true) {\n const { value, done } = await reader.read();\n if (done) break;\n buffer += decoder.decode(value, { stream: true });\n\n let separatorMatch = buffer.match(/\\r?\\n\\r?\\n/);\n while (separatorMatch && separatorMatch.index !== undefined) {\n const separatorIndex = separatorMatch.index;\n const separatorLength = separatorMatch[0].length;\n const block = buffer.slice(0, separatorIndex);\n buffer = buffer.slice(separatorIndex + separatorLength);\n dispatchBlock(block);\n separatorMatch = buffer.match(/\\r?\\n\\r?\\n/);\n }\n }\n\n if (buffer.trim()) {\n dispatchBlock(buffer);\n }\n\n if (!rpcResponse) {\n throw new Error('Missing rpc-response event in streamed RPC result');\n }\n\n return rpcResponse;\n }\n\n private parseRpcResponse<T>(data: McpRpcResponse): T {\n if ('result' in data) {\n return data.result as T;\n }\n if ('error' in data && data.error) {\n throw new Error(data.error.message || 'Unknown RPC error');\n }\n // JSON omits `result` when it is `undefined` (response becomes `{ id: ... }`).\n // Treat that shape as a successful void result.\n if (data && typeof data === 'object' && 'id' in data) {\n return undefined as T;\n }\n throw new Error('Invalid RPC response format');\n }\n\n private buildUrl(): string {\n const url = new URL(this.options.url, globalThis.location?.origin);\n url.searchParams.set('identity', this.options.identity);\n if (this.options.authToken) {\n url.searchParams.set('token', this.options.authToken);\n }\n return url.toString();\n }\n\n private buildHeaders(): HeadersInit {\n const headers: HeadersInit = {\n 'Content-Type': 'application/json',\n 'Accept': 'text/event-stream',\n };\n if (this.options.authToken) {\n headers['Authorization'] = `Bearer ${this.options.authToken}`;\n }\n return headers;\n }\n\n private extractUiResourceUri(tool: { name: string; _meta?: unknown }): string | undefined {\n const meta = (tool._meta as { ui?: ToolUiMetadata })?.ui;\n if (!meta || typeof meta !== 'object') return undefined;\n if (meta.visibility && !meta.visibility.includes('app')) return undefined;\n return meta.resourceUri ?? meta.uri;\n }\n\n private emitUiEventIfPresent(result: unknown, sessionId: string, toolName: string): void {\n const meta = (result as { _meta?: { ui?: ToolUiMetadata } })?._meta;\n const resourceUri = meta?.ui?.resourceUri ?? (meta as any)?.['ui/resourceUri'];\n\n if (resourceUri) {\n this.options.onEvent?.({\n type: 'mcp-apps-ui',\n sessionId,\n resourceUri,\n toolName,\n result,\n timestamp: Date.now(),\n });\n }\n }\n\n private log(message: string, level: 'info' | 'warn' | 'error' = 'info'): void {\n if (!this.options.debug && level === 'info') return;\n\n const prefix = '[SSEClient]';\n switch (level) {\n case 'warn':\n console.warn(prefix, message);\n break;\n case 'error':\n console.error(prefix, message);\n break;\n default:\n console.log(prefix, message);\n }\n }\n}\n","/**\r\n * MCP App Host\r\n *\r\n * Bridges the gap between an iframe (MCP App) and the SSEClient (MCP Server).\r\n * Handles secure iframe sandboxing, resource loading, and bi-directional\r\n * communication via the AppBridge protocol.\r\n *\r\n * Key features:\r\n * - Secure iframe sandboxing with minimal permissions\r\n * - Resource preloading for instant MCP App UI loading\r\n * - Cache-aware resource fetching (SSEClient cache → local cache → direct fetch)\r\n * - Support for ui:// and mcp-app:// resource URIs\r\n */\r\n\r\nimport { AppBridge, PostMessageTransport } from '@modelcontextprotocol/ext-apps/app-bridge';\r\nimport type { AppHostClient } from './types';\r\n\r\n// ============================================\r\n// Types & Interfaces\r\n// ============================================\r\n\r\nexport interface AppHostOptions {\r\n /** Enable debug logging @default false */\r\n debug?: boolean;\r\n}\r\n\r\nexport interface AppMessageParams {\r\n role: string;\r\n content: unknown;\r\n}\r\n\r\ninterface ToolCallParams {\r\n name: string;\r\n arguments?: Record<string, unknown>;\r\n}\r\n\r\ninterface ResourceContent {\r\n blob?: string;\r\n text?: string;\r\n}\r\n\r\ninterface ResourceResponse {\r\n contents: ResourceContent[];\r\n}\r\n\r\n// ============================================\r\n// Constants\r\n// ============================================\r\n\r\nconst HOST_INFO = { name: 'mcp-ts-host', version: '1.0.0' };\r\n\r\n/** Sandbox permissions - minimal set required for MCP Apps to function */\r\nconst SANDBOX_PERMISSIONS = [\r\n 'allow-scripts', // Required for app JavaScript execution\r\n 'allow-forms', // Required for form submissions\r\n 'allow-same-origin', // Required for Blob URL correctness\r\n 'allow-modals', // Required for dialogs/alerts\r\n 'allow-popups', // Required for opening links\r\n 'allow-downloads' // Required for file downloads\r\n].join(' ');\r\n\r\n/** Supported MCP App URI schemes */\r\nconst MCP_URI_SCHEMES = ['ui://', 'mcp-app://'] as const;\r\n\r\n// ============================================\r\n// AppHost Class\r\n// ============================================\r\n\r\n/**\r\n * Host for MCP Apps embedded in iframes.\r\n * Manages secure communication between the app and the MCP server.\r\n */\r\nexport class AppHost {\r\n private bridge: AppBridge;\r\n private sessionId?: string;\r\n private resourceCache = new Map<string, Promise<ResourceResponse | null>>();\r\n private debug: boolean;\r\n\r\n /** Callback for app messages (e.g., chat messages from the app) */\r\n public onAppMessage?: (params: AppMessageParams) => void;\r\n\r\n constructor(\r\n private readonly client: AppHostClient,\r\n private readonly iframe: HTMLIFrameElement,\r\n options?: AppHostOptions\r\n ) {\r\n this.debug = options?.debug ?? false;\r\n this.configureSandbox();\r\n this.bridge = this.initializeBridge();\r\n }\r\n\r\n // ============================================\r\n // Public API\r\n // ============================================\r\n\r\n /**\r\n * Start the host. This prepares the bridge handlers but doesn't connect yet.\r\n * The actual connection happens in launch() after HTML is loaded.\r\n * @returns Promise that resolves immediately (bridge connects during launch)\r\n */\r\n async start(): Promise<void> {\r\n // Bridge handlers are already registered in constructor.\r\n // Connection happens in launch() after HTML is loaded.\r\n this.log('Host started, ready to launch');\r\n }\r\n\r\n /**\r\n * Preload UI resources to enable instant app loading.\r\n * Call this when tools are discovered to cache their UI resources.\r\n */\r\n preload(tools: Array<{ _meta?: unknown }>): void {\r\n for (const tool of tools) {\r\n const uri = this.extractUiResourceUri(tool);\r\n if (!uri || this.resourceCache.has(uri)) continue;\r\n\r\n const promise = this.preloadResource(uri);\r\n this.resourceCache.set(uri, promise);\r\n }\r\n }\r\n\r\n /**\r\n * Launch an MCP App from a URL or MCP resource URI.\r\n * Loads the HTML first, then establishes bridge connection.\r\n */\r\n async launch(url: string, sessionId?: string): Promise<void> {\r\n if (sessionId) this.sessionId = sessionId;\r\n\r\n // Set up initialization promise BEFORE connecting\r\n const initializedPromise = this.onAppReady();\r\n\r\n // Load HTML into iframe first\r\n if (this.isMcpUri(url)) {\r\n await this.launchMcpApp(url);\r\n } else {\r\n this.iframe.src = url;\r\n }\r\n\r\n // Wait for iframe to load before connecting bridge\r\n await this.onIframeReady();\r\n\r\n // Connect the bridge (HTML is loaded, contentWindow is ready)\r\n await this.connectBridge();\r\n\r\n // Wait for app to signal it's initialized (with timeout)\r\n this.log('Waiting for app initialization');\r\n await Promise.race([\r\n initializedPromise,\r\n new Promise<void>((resolve) => setTimeout(() => {\r\n this.log('Initialization timeout - continuing anyway', 'warn');\r\n resolve();\r\n }, 3000))\r\n ]);\r\n this.log('App launched and ready');\r\n }\r\n\r\n /**\r\n * Wait for app to signal initialization complete\r\n */\r\n private onAppReady(): Promise<void> {\r\n return new Promise<void>((resolve) => {\r\n const originalHandler = this.bridge.oninitialized;\r\n this.bridge.oninitialized = (...args) => {\r\n this.log('App initialized');\r\n resolve();\r\n this.bridge.oninitialized = originalHandler;\r\n originalHandler?.(...args);\r\n };\r\n });\r\n }\r\n\r\n /**\r\n * Wait for iframe to finish loading\r\n */\r\n private onIframeReady(): Promise<void> {\r\n return new Promise((resolve) => {\r\n if (this.iframe.contentDocument?.readyState === 'complete') {\r\n resolve();\r\n return;\r\n }\r\n this.iframe.addEventListener('load', () => resolve(), { once: true });\r\n });\r\n }\r\n\r\n /**\r\n * Send tool input arguments to the MCP App.\r\n * Call this after launch() when tool input is available.\r\n */\r\n sendToolInput(args: Record<string, unknown>): void {\r\n this.log('Sending tool input to app');\r\n this.bridge.sendToolInput({ arguments: args });\r\n }\r\n\r\n /**\r\n * Send tool result to the MCP App.\r\n * Call this when the tool call completes.\r\n */\r\n sendToolResult(result: unknown): void {\r\n this.log('Sending tool result to app');\r\n this.bridge.sendToolResult(result as any);\r\n }\r\n\r\n /**\r\n * Send tool cancellation to the MCP App.\r\n * Call this when the tool call is cancelled or fails.\r\n */\r\n sendToolCancelled(reason: string): void {\r\n this.log('Sending tool cancellation to app');\r\n this.bridge.sendToolCancelled({ reason });\r\n }\r\n\r\n // ============================================\r\n // Private: Initialization\r\n // ============================================\r\n\r\n private configureSandbox(): void {\r\n if (this.iframe.sandbox.value !== SANDBOX_PERMISSIONS) {\r\n this.iframe.sandbox.value = SANDBOX_PERMISSIONS;\r\n }\r\n }\r\n\r\n private initializeBridge(): AppBridge {\r\n const bridge = new AppBridge(\r\n null,\r\n HOST_INFO,\r\n {\r\n openLinks: {},\r\n serverTools: {},\r\n logging: {},\r\n // Declare support for model context updates\r\n updateModelContext: { text: {} },\r\n },\r\n {\r\n // Initial host context\r\n hostContext: {\r\n theme: 'dark',\r\n platform: 'web',\r\n containerDimensions: { maxHeight: 6000 },\r\n displayMode: 'inline',\r\n availableDisplayModes: ['inline', 'fullscreen'],\r\n },\r\n }\r\n );\r\n\r\n // Register handlers - must be done BEFORE connect()\r\n bridge.oncalltool = (params) => this.handleToolCall(params);\r\n bridge.onopenlink = this.handleOpenLink.bind(this);\r\n bridge.onmessage = this.handleMessage.bind(this);\r\n bridge.onloggingmessage = (params) => this.log(`App log [${params.level}]: ${params.data}`);\r\n bridge.onupdatemodelcontext = async () => ({});\r\n bridge.onsizechange = async ({ width, height }) => {\r\n if (height !== undefined) this.iframe.style.height = `${height}px`;\r\n if (width !== undefined) this.iframe.style.minWidth = `min(${width}px, 100%)`;\r\n return {};\r\n };\r\n bridge.onrequestdisplaymode = async (params) => ({\r\n mode: params.mode === 'fullscreen' ? 'fullscreen' : 'inline'\r\n });\r\n\r\n return bridge;\r\n }\r\n\r\n private async connectBridge(): Promise<void> {\r\n this.log('Connecting bridge to iframe');\r\n\r\n const transport = new PostMessageTransport(\r\n this.iframe.contentWindow!,\r\n this.iframe.contentWindow!\r\n );\r\n\r\n try {\r\n await this.bridge.connect(transport);\r\n this.log('Bridge connected successfully');\r\n } catch (error) {\r\n this.log('Bridge connection failed', 'error');\r\n throw error;\r\n }\r\n }\r\n\r\n // ============================================\r\n // Private: Bridge Event Handlers\r\n // ============================================\r\n\r\n private async handleToolCall(params: ToolCallParams) {\r\n if (!this.client.isConnected()) {\r\n throw new Error('Client disconnected');\r\n }\r\n\r\n const sessionId = await this.getSessionId();\r\n if (!sessionId) {\r\n throw new Error('No active session');\r\n }\r\n\r\n const result = await this.client.callTool(\r\n sessionId,\r\n params.name,\r\n params.arguments ?? {}\r\n );\r\n return result as any;\r\n }\r\n\r\n private async handleOpenLink(params: { url: string }): Promise<Record<string, never>> {\r\n window.open(params.url, '_blank', 'noopener,noreferrer');\r\n return {};\r\n }\r\n\r\n private async handleMessage(params: AppMessageParams): Promise<Record<string, never>> {\r\n this.onAppMessage?.(params);\r\n return {};\r\n }\r\n\r\n // ============================================\r\n // Private: Resource Loading\r\n // ============================================\r\n\r\n private async launchMcpApp(uri: string): Promise<void> {\r\n if (!this.client.isConnected()) {\r\n throw new Error('Client must be connected');\r\n }\r\n\r\n const sessionId = await this.getSessionId();\r\n if (!sessionId) {\r\n throw new Error('No active session');\r\n }\r\n\r\n // Fetch resource using cache hierarchy: SSEClient cache → local cache → direct fetch\r\n const response = await this.fetchResourceWithCache(sessionId, uri);\r\n if (!response?.contents?.length) {\r\n throw new Error(`Empty resource: ${uri}`);\r\n }\r\n\r\n const content = response.contents[0];\r\n const html = this.decodeContent(content);\r\n if (!html) {\r\n throw new Error(`Invalid content in resource: ${uri}`);\r\n }\r\n\r\n // Render via Blob URL for clean isolation\r\n const blob = new Blob([html], { type: 'text/html' });\r\n this.iframe.src = URL.createObjectURL(blob);\r\n }\r\n\r\n private async fetchResourceWithCache(sessionId: string, uri: string): Promise<ResourceResponse> {\r\n // Priority 1: SSEClient's built-in cache (best performance)\r\n if (this.hasClientCache()) {\r\n return (this.client as any).getOrFetchResource(sessionId, uri);\r\n }\r\n\r\n // Priority 2: Local preload cache\r\n const cached = this.resourceCache.get(uri);\r\n if (cached) {\r\n const result = await cached;\r\n if (result) return result;\r\n }\r\n\r\n // Priority 3: Direct fetch\r\n return this.client.readResource(sessionId, uri) as Promise<ResourceResponse>;\r\n }\r\n\r\n private async preloadResource(uri: string): Promise<ResourceResponse | null> {\r\n try {\r\n const sessionId = await this.getSessionId();\r\n if (!sessionId) return null;\r\n return await this.client.readResource(sessionId, uri) as ResourceResponse;\r\n } catch (error) {\r\n this.log(`Preload failed for ${uri}`, 'warn');\r\n return null;\r\n }\r\n }\r\n\r\n // ============================================\r\n // Private: Utilities\r\n // ============================================\r\n\r\n private async getSessionId(): Promise<string | undefined> {\r\n if (this.sessionId) return this.sessionId;\r\n const result = await this.client.getSessions();\r\n return result.sessions?.[0]?.sessionId;\r\n }\r\n\r\n private isMcpUri(url: string): boolean {\r\n return MCP_URI_SCHEMES.some(scheme => url.startsWith(scheme));\r\n }\r\n\r\n private hasClientCache(): boolean {\r\n return 'getOrFetchResource' in this.client &&\r\n typeof (this.client as any).getOrFetchResource === 'function';\r\n }\r\n\r\n private extractUiResourceUri(tool: { _meta?: unknown }): string | undefined {\r\n const meta = tool._meta as { ui?: { resourceUri?: string; uri?: string } } | undefined;\r\n if (!meta?.ui) return undefined;\r\n return meta.ui.resourceUri ?? meta.ui.uri;\r\n }\r\n\r\n private decodeContent(content: ResourceContent): string | undefined {\r\n if (content.blob) {\r\n return atob(content.blob);\r\n }\r\n return content.text;\r\n }\r\n\r\n private log(message: string, level: 'info' | 'warn' | 'error' = 'info'): void {\r\n if (!this.debug && level === 'info') return;\r\n\r\n const prefix = '[AppHost]';\r\n switch (level) {\r\n case 'warn':\r\n console.warn(prefix, message);\r\n break;\r\n case 'error':\r\n console.error(prefix, message);\r\n break;\r\n default:\r\n console.log(prefix, message);\r\n }\r\n }\r\n}\r\n"]}
@@ -4,62 +4,27 @@ import { AppBridge, PostMessageTransport } from '@modelcontextprotocol/ext-apps/
4
4
  var __defProp = Object.defineProperty;
5
5
  var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;
6
6
  var __publicField = (obj, key, value) => __defNormalProp(obj, typeof key !== "symbol" ? key + "" : key, value);
7
- var DEFAULT_REQUEST_TIMEOUT = 6e4;
8
- var MAX_RECONNECT_ATTEMPTS = 5;
9
- var BASE_RECONNECT_DELAY = 1e3;
10
7
  var SSEClient = class {
11
8
  constructor(options) {
12
9
  this.options = options;
13
- __publicField(this, "eventSource", null);
14
- __publicField(this, "pendingRequests", /* @__PURE__ */ new Map());
15
10
  __publicField(this, "resourceCache", /* @__PURE__ */ new Map());
16
- __publicField(this, "reconnectAttempts", 0);
17
- __publicField(this, "isManuallyDisconnected", false);
18
- __publicField(this, "connectionPromise", null);
19
- __publicField(this, "connectionResolver", null);
11
+ __publicField(this, "connected", false);
20
12
  }
21
- // ============================================
22
- // Connection Management
23
- // ============================================
24
- /**
25
- * Connect to the SSE endpoint
26
- */
27
13
  connect() {
28
- if (this.eventSource) {
14
+ if (this.connected) {
29
15
  return;
30
16
  }
31
- this.isManuallyDisconnected = false;
32
- this.options.onStatusChange?.("connecting");
33
- this.connectionPromise = new Promise((resolve) => {
34
- this.connectionResolver = resolve;
35
- });
36
- const url = this.buildUrl();
37
- this.eventSource = new EventSource(url);
38
- this.setupEventListeners();
17
+ this.connected = true;
18
+ this.options.onStatusChange?.("connected");
19
+ this.log("RPC mode: post_stream");
39
20
  }
40
- /**
41
- * Disconnect from the SSE endpoint
42
- */
43
21
  disconnect() {
44
- this.isManuallyDisconnected = true;
45
- if (this.eventSource) {
46
- this.eventSource.close();
47
- this.eventSource = null;
48
- }
49
- this.connectionPromise = null;
50
- this.connectionResolver = null;
51
- this.rejectAllPendingRequests(new Error("Connection closed"));
22
+ this.connected = false;
52
23
  this.options.onStatusChange?.("disconnected");
53
24
  }
54
- /**
55
- * Check if connected to the SSE endpoint
56
- */
57
25
  isConnected() {
58
- return this.eventSource?.readyState === EventSource.OPEN;
26
+ return this.connected;
59
27
  }
60
- // ============================================
61
- // RPC Methods
62
- // ============================================
63
28
  async getSessions() {
64
29
  return this.sendRequest("getSessions");
65
30
  }
@@ -95,22 +60,10 @@ var SSEClient = class {
95
60
  async readResource(sessionId, uri) {
96
61
  return this.sendRequest("readResource", { sessionId, uri });
97
62
  }
98
- // ============================================
99
- // Resource Preloading (for instant UI loading)
100
- // ============================================
101
- /**
102
- * Preload UI resources for tools that have UI metadata.
103
- * Call this when tools are discovered to enable instant MCP App UI loading.
104
- */
105
63
  preloadToolUiResources(sessionId, tools) {
106
64
  for (const tool of tools) {
107
65
  const uri = this.extractUiResourceUri(tool);
108
- if (!uri) continue;
109
- if (this.resourceCache.has(uri)) {
110
- this.log(`Resource already cached: ${uri}`);
111
- continue;
112
- }
113
- this.log(`Preloading UI resource for tool "${tool.name}": ${uri}`);
66
+ if (!uri || this.resourceCache.has(uri)) continue;
114
67
  const promise = this.sendRequest("readResource", { sessionId, uri }).catch((err) => {
115
68
  this.log(`Failed to preload resource ${uri}: ${err.message}`, "warn");
116
69
  this.resourceCache.delete(uri);
@@ -119,43 +72,24 @@ var SSEClient = class {
119
72
  this.resourceCache.set(uri, promise);
120
73
  }
121
74
  }
122
- /**
123
- * Get a preloaded resource from cache, or fetch if not cached.
124
- */
125
75
  getOrFetchResource(sessionId, uri) {
126
76
  const cached = this.resourceCache.get(uri);
127
- if (cached) {
128
- this.log(`Cache hit for resource: ${uri}`);
129
- return cached;
130
- }
131
- this.log(`Cache miss, fetching resource: ${uri}`);
77
+ if (cached) return cached;
132
78
  const promise = this.sendRequest("readResource", { sessionId, uri });
133
79
  this.resourceCache.set(uri, promise);
134
80
  return promise;
135
81
  }
136
- /**
137
- * Check if a resource is already cached
138
- */
139
82
  hasPreloadedResource(uri) {
140
83
  return this.resourceCache.has(uri);
141
84
  }
142
- /**
143
- * Clear the resource cache
144
- */
145
85
  clearResourceCache() {
146
86
  this.resourceCache.clear();
147
87
  }
148
- // ============================================
149
- // Private: Request Handling
150
- // ============================================
151
- /**
152
- * Send an RPC request and return the response directly from HTTP.
153
- * This bypasses SSE latency by returning results in the HTTP response body.
154
- */
155
88
  async sendRequest(method, params) {
156
- if (this.connectionPromise) {
157
- await this.connectionPromise;
89
+ if (!this.connected) {
90
+ this.connect();
158
91
  }
92
+ this.log(`RPC request via post_stream: ${method}`);
159
93
  const request = {
160
94
  id: `rpc_${nanoid(10)}`,
161
95
  method,
@@ -169,103 +103,93 @@ var SSEClient = class {
169
103
  if (!response.ok) {
170
104
  throw new Error(`HTTP ${response.status}: ${response.statusText}`);
171
105
  }
172
- const data = await response.json();
173
- return this.parseRpcResponse(data, request.id);
106
+ const contentType = (response.headers.get("content-type") || "").toLowerCase();
107
+ if (!contentType.includes("text/event-stream")) {
108
+ const data2 = await response.json();
109
+ return this.parseRpcResponse(data2);
110
+ }
111
+ const data = await this.readRpcResponseFromStream(response);
112
+ return this.parseRpcResponse(data);
174
113
  }
175
- /**
176
- * Parse RPC response and handle different response formats
177
- */
178
- parseRpcResponse(data, requestId) {
114
+ async readRpcResponseFromStream(response) {
115
+ if (!response.body) {
116
+ throw new Error("Streaming response body is missing");
117
+ }
118
+ const reader = response.body.getReader();
119
+ const decoder = new TextDecoder();
120
+ let buffer = "";
121
+ let rpcResponse = null;
122
+ const dispatchBlock = (block) => {
123
+ const lines = block.split("\n");
124
+ let eventName = "message";
125
+ const dataLines = [];
126
+ for (const rawLine of lines) {
127
+ const line = rawLine.replace(/\r$/, "");
128
+ if (!line || line.startsWith(":")) continue;
129
+ if (line.startsWith("event:")) {
130
+ eventName = line.slice("event:".length).trim();
131
+ continue;
132
+ }
133
+ if (line.startsWith("data:")) {
134
+ dataLines.push(line.slice("data:".length).trimStart());
135
+ }
136
+ }
137
+ if (!dataLines.length) return;
138
+ const payloadText = dataLines.join("\n");
139
+ let payload = payloadText;
140
+ try {
141
+ payload = JSON.parse(payloadText);
142
+ } catch {
143
+ }
144
+ switch (eventName) {
145
+ case "connected":
146
+ this.options.onStatusChange?.("connected");
147
+ break;
148
+ case "connection":
149
+ this.options.onConnectionEvent?.(payload);
150
+ break;
151
+ case "observability":
152
+ this.options.onObservabilityEvent?.(payload);
153
+ break;
154
+ case "rpc-response":
155
+ rpcResponse = payload;
156
+ break;
157
+ }
158
+ };
159
+ while (true) {
160
+ const { value, done } = await reader.read();
161
+ if (done) break;
162
+ buffer += decoder.decode(value, { stream: true });
163
+ let separatorMatch = buffer.match(/\r?\n\r?\n/);
164
+ while (separatorMatch && separatorMatch.index !== void 0) {
165
+ const separatorIndex = separatorMatch.index;
166
+ const separatorLength = separatorMatch[0].length;
167
+ const block = buffer.slice(0, separatorIndex);
168
+ buffer = buffer.slice(separatorIndex + separatorLength);
169
+ dispatchBlock(block);
170
+ separatorMatch = buffer.match(/\r?\n\r?\n/);
171
+ }
172
+ }
173
+ if (buffer.trim()) {
174
+ dispatchBlock(buffer);
175
+ }
176
+ if (!rpcResponse) {
177
+ throw new Error("Missing rpc-response event in streamed RPC result");
178
+ }
179
+ return rpcResponse;
180
+ }
181
+ parseRpcResponse(data) {
179
182
  if ("result" in data) {
180
183
  return data.result;
181
184
  }
182
185
  if ("error" in data && data.error) {
183
186
  throw new Error(data.error.message || "Unknown RPC error");
184
187
  }
185
- if ("acknowledged" in data) {
186
- return this.waitForSseResponse(requestId);
188
+ if (data && typeof data === "object" && "id" in data) {
189
+ return void 0;
187
190
  }
188
191
  throw new Error("Invalid RPC response format");
189
192
  }
190
- /**
191
- * Wait for RPC response via SSE (legacy fallback)
192
- */
193
- waitForSseResponse(requestId) {
194
- const timeoutMs = this.options.requestTimeout ?? DEFAULT_REQUEST_TIMEOUT;
195
- return new Promise((resolve, reject) => {
196
- const timeoutId = setTimeout(() => {
197
- this.pendingRequests.delete(requestId);
198
- reject(new Error(`Request timeout after ${timeoutMs}ms`));
199
- }, timeoutMs);
200
- this.pendingRequests.set(requestId, {
201
- resolve,
202
- reject,
203
- timeoutId
204
- });
205
- });
206
- }
207
- /**
208
- * Handle RPC response received via SSE (legacy)
209
- */
210
- handleRpcResponse(response) {
211
- const pending = this.pendingRequests.get(response.id);
212
- if (!pending) return;
213
- clearTimeout(pending.timeoutId);
214
- this.pendingRequests.delete(response.id);
215
- if (response.error) {
216
- pending.reject(new Error(response.error.message));
217
- } else {
218
- pending.resolve(response.result);
219
- }
220
- }
221
- // ============================================
222
- // Private: Event Handling
223
- // ============================================
224
- setupEventListeners() {
225
- if (!this.eventSource) return;
226
- this.eventSource.addEventListener("open", () => {
227
- this.log("Connected");
228
- this.reconnectAttempts = 0;
229
- this.options.onStatusChange?.("connected");
230
- });
231
- this.eventSource.addEventListener("connected", () => {
232
- this.log("Server ready");
233
- this.connectionResolver?.();
234
- this.connectionResolver = null;
235
- });
236
- this.eventSource.addEventListener("connection", (e) => {
237
- const event = JSON.parse(e.data);
238
- this.options.onConnectionEvent?.(event);
239
- });
240
- this.eventSource.addEventListener("observability", (e) => {
241
- const event = JSON.parse(e.data);
242
- this.options.onObservabilityEvent?.(event);
243
- });
244
- this.eventSource.addEventListener("rpc-response", (e) => {
245
- const response = JSON.parse(e.data);
246
- this.handleRpcResponse(response);
247
- });
248
- this.eventSource.addEventListener("error", () => {
249
- this.log("Connection error", "error");
250
- this.options.onStatusChange?.("error");
251
- this.attemptReconnect();
252
- });
253
- }
254
- attemptReconnect() {
255
- if (this.isManuallyDisconnected || this.reconnectAttempts >= MAX_RECONNECT_ATTEMPTS) {
256
- return;
257
- }
258
- this.reconnectAttempts++;
259
- const delay = BASE_RECONNECT_DELAY * this.reconnectAttempts;
260
- this.log(`Reconnecting in ${delay}ms (attempt ${this.reconnectAttempts}/${MAX_RECONNECT_ATTEMPTS})`);
261
- setTimeout(() => {
262
- this.disconnect();
263
- this.connect();
264
- }, delay);
265
- }
266
- // ============================================
267
- // Private: Utilities
268
- // ============================================
269
193
  buildUrl() {
270
194
  const url = new URL(this.options.url, globalThis.location?.origin);
271
195
  url.searchParams.set("identity", this.options.identity);
@@ -276,20 +200,14 @@ var SSEClient = class {
276
200
  }
277
201
  buildHeaders() {
278
202
  const headers = {
279
- "Content-Type": "application/json"
203
+ "Content-Type": "application/json",
204
+ "Accept": "text/event-stream"
280
205
  };
281
206
  if (this.options.authToken) {
282
207
  headers["Authorization"] = `Bearer ${this.options.authToken}`;
283
208
  }
284
209
  return headers;
285
210
  }
286
- rejectAllPendingRequests(error) {
287
- for (const [, pending] of this.pendingRequests) {
288
- clearTimeout(pending.timeoutId);
289
- pending.reject(error);
290
- }
291
- this.pendingRequests.clear();
292
- }
293
211
  extractUiResourceUri(tool) {
294
212
  const meta = tool._meta?.ui;
295
213
  if (!meta || typeof meta !== "object") return void 0;