@mcp-ts/sdk 1.0.0
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/LICENSE +21 -0
- package/README.md +297 -0
- package/dist/adapters/agui-adapter.d.mts +119 -0
- package/dist/adapters/agui-adapter.d.ts +119 -0
- package/dist/adapters/agui-adapter.js +109 -0
- package/dist/adapters/agui-adapter.js.map +1 -0
- package/dist/adapters/agui-adapter.mjs +107 -0
- package/dist/adapters/agui-adapter.mjs.map +1 -0
- package/dist/adapters/agui-middleware.d.mts +171 -0
- package/dist/adapters/agui-middleware.d.ts +171 -0
- package/dist/adapters/agui-middleware.js +429 -0
- package/dist/adapters/agui-middleware.js.map +1 -0
- package/dist/adapters/agui-middleware.mjs +417 -0
- package/dist/adapters/agui-middleware.mjs.map +1 -0
- package/dist/adapters/ai-adapter.d.mts +38 -0
- package/dist/adapters/ai-adapter.d.ts +38 -0
- package/dist/adapters/ai-adapter.js +82 -0
- package/dist/adapters/ai-adapter.js.map +1 -0
- package/dist/adapters/ai-adapter.mjs +80 -0
- package/dist/adapters/ai-adapter.mjs.map +1 -0
- package/dist/adapters/langchain-adapter.d.mts +46 -0
- package/dist/adapters/langchain-adapter.d.ts +46 -0
- package/dist/adapters/langchain-adapter.js +102 -0
- package/dist/adapters/langchain-adapter.js.map +1 -0
- package/dist/adapters/langchain-adapter.mjs +100 -0
- package/dist/adapters/langchain-adapter.mjs.map +1 -0
- package/dist/adapters/mastra-adapter.d.mts +49 -0
- package/dist/adapters/mastra-adapter.d.ts +49 -0
- package/dist/adapters/mastra-adapter.js +95 -0
- package/dist/adapters/mastra-adapter.js.map +1 -0
- package/dist/adapters/mastra-adapter.mjs +93 -0
- package/dist/adapters/mastra-adapter.mjs.map +1 -0
- package/dist/client/index.d.mts +119 -0
- package/dist/client/index.d.ts +119 -0
- package/dist/client/index.js +225 -0
- package/dist/client/index.js.map +1 -0
- package/dist/client/index.mjs +223 -0
- package/dist/client/index.mjs.map +1 -0
- package/dist/client/react.d.mts +151 -0
- package/dist/client/react.d.ts +151 -0
- package/dist/client/react.js +492 -0
- package/dist/client/react.js.map +1 -0
- package/dist/client/react.mjs +489 -0
- package/dist/client/react.mjs.map +1 -0
- package/dist/client/vue.d.mts +157 -0
- package/dist/client/vue.d.ts +157 -0
- package/dist/client/vue.js +474 -0
- package/dist/client/vue.js.map +1 -0
- package/dist/client/vue.mjs +471 -0
- package/dist/client/vue.mjs.map +1 -0
- package/dist/events-BP6WyRNh.d.mts +110 -0
- package/dist/events-BP6WyRNh.d.ts +110 -0
- package/dist/index.d.mts +10 -0
- package/dist/index.d.ts +10 -0
- package/dist/index.js +2784 -0
- package/dist/index.js.map +1 -0
- package/dist/index.mjs +2723 -0
- package/dist/index.mjs.map +1 -0
- package/dist/multi-session-client-BOFgPypS.d.ts +389 -0
- package/dist/multi-session-client-DMF3ED2O.d.mts +389 -0
- package/dist/server/index.d.mts +269 -0
- package/dist/server/index.d.ts +269 -0
- package/dist/server/index.js +2444 -0
- package/dist/server/index.js.map +1 -0
- package/dist/server/index.mjs +2414 -0
- package/dist/server/index.mjs.map +1 -0
- package/dist/shared/index.d.mts +24 -0
- package/dist/shared/index.d.ts +24 -0
- package/dist/shared/index.js +223 -0
- package/dist/shared/index.js.map +1 -0
- package/dist/shared/index.mjs +190 -0
- package/dist/shared/index.mjs.map +1 -0
- package/dist/types-SbDlA2VX.d.mts +153 -0
- package/dist/types-SbDlA2VX.d.ts +153 -0
- package/dist/utils-0qmYrqoa.d.mts +92 -0
- package/dist/utils-0qmYrqoa.d.ts +92 -0
- package/package.json +165 -0
- package/src/adapters/agui-adapter.ts +210 -0
- package/src/adapters/agui-middleware.ts +512 -0
- package/src/adapters/ai-adapter.ts +115 -0
- package/src/adapters/langchain-adapter.ts +127 -0
- package/src/adapters/mastra-adapter.ts +126 -0
- package/src/client/core/sse-client.ts +340 -0
- package/src/client/index.ts +26 -0
- package/src/client/react/index.ts +10 -0
- package/src/client/react/useMcp.ts +558 -0
- package/src/client/vue/index.ts +10 -0
- package/src/client/vue/useMcp.ts +542 -0
- package/src/index.ts +11 -0
- package/src/server/handlers/nextjs-handler.ts +216 -0
- package/src/server/handlers/sse-handler.ts +699 -0
- package/src/server/index.ts +57 -0
- package/src/server/mcp/multi-session-client.ts +132 -0
- package/src/server/mcp/oauth-client.ts +1168 -0
- package/src/server/mcp/storage-oauth-provider.ts +239 -0
- package/src/server/storage/file-backend.ts +169 -0
- package/src/server/storage/index.ts +115 -0
- package/src/server/storage/memory-backend.ts +132 -0
- package/src/server/storage/redis-backend.ts +210 -0
- package/src/server/storage/redis.ts +160 -0
- package/src/server/storage/types.ts +109 -0
- package/src/shared/constants.ts +29 -0
- package/src/shared/errors.ts +133 -0
- package/src/shared/events.ts +166 -0
- package/src/shared/index.ts +70 -0
- package/src/shared/types.ts +274 -0
- package/src/shared/utils.ts +16 -0
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/server/storage/redis.ts","../src/shared/constants.ts","../src/server/storage/redis-backend.ts","../src/server/storage/memory-backend.ts","../src/server/storage/file-backend.ts","../src/server/storage/index.ts","../src/server/mcp/storage-oauth-provider.ts","../src/shared/utils.ts","../src/shared/events.ts","../src/shared/errors.ts","../src/server/mcp/oauth-client.ts","../src/server/mcp/multi-session-client.ts","../src/server/handlers/sse-handler.ts","../src/server/handlers/nextjs-handler.ts","../src/client/core/sse-client.ts","../src/shared/types.ts"],"names":["redis","firstChar","customAlphabet","rest","fs","getRedis","SDKUnauthorizedError","sendSSE","nanoid"],"mappings":"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","file":"index.mjs","sourcesContent":["/**\r\n * Redis connection management with dependency injection support\r\n * Allows configuration and testing without environment variables\r\n */\r\nimport type { Redis } from 'ioredis';\r\n\r\nexport interface RedisConfig {\r\n /**\r\n * Redis connection URL (defaults to REDIS_URL env var)\r\n */\r\n url?: string;\r\n\r\n /**\r\n * Enable lazy connection (default: true)\r\n */\r\n lazyConnect?: boolean;\r\n\r\n /**\r\n * Maximum retries per request (default: 1)\r\n */\r\n maxRetriesPerRequest?: number;\r\n\r\n /**\r\n * Enable verbose logging (default: false)\r\n */\r\n verbose?: boolean;\r\n /**\r\n * @internal For testing only - bypass ioredis import\r\n */\r\n RedisConstructor?: any;\r\n}\r\n\r\ndeclare global {\r\n // eslint-disable-next-line no-var\r\n var __redis: Redis | undefined;\r\n var __redisConfig: RedisConfig | undefined;\r\n}\r\n\r\nlet redisInstance: Redis | null = null;\r\n\r\n/**\r\n * Initialize Redis with custom configuration\r\n * Call this before any Redis operations if you need custom config\r\n * @param config - Redis configuration options\r\n */\r\nexport async function initRedis(config: RedisConfig): Promise<Redis> {\r\n if (redisInstance) {\r\n // Already initialized, return existing instance\r\n return redisInstance;\r\n }\r\n\r\n const url = config.url ?? process.env.REDIS_URL;\r\n\r\n if (!url) {\r\n throw new Error(\r\n 'Redis URL is required. Set REDIS_URL environment variable or pass url in config.'\r\n );\r\n }\r\n\r\n let Redis: typeof import('ioredis').Redis;\r\n if (config.RedisConstructor) {\r\n Redis = config.RedisConstructor;\r\n } else {\r\n try {\r\n const ioredis = await import('ioredis');\r\n Redis = ioredis.Redis;\r\n } catch (error) {\r\n throw new Error(\r\n 'ioredis is not installed. Install it with:\\n' +\r\n ' npm install ioredis\\n\\n' +\r\n 'Or use a different storage backend:\\n' +\r\n ' MCP_TS_STORAGE_TYPE=memory (for development)\\n' +\r\n ' MCP_TS_STORAGE_TYPE=file (for local persistence)'\r\n );\r\n }\r\n }\r\n\r\n redisInstance = new Redis(url, {\r\n lazyConnect: config.lazyConnect ?? true,\r\n maxRetriesPerRequest: config.maxRetriesPerRequest ?? 1,\r\n });\r\n\r\n if (config.verbose !== false) {\r\n redisInstance.on('ready', () => {\r\n console.log('✅ Redis connected');\r\n });\r\n\r\n redisInstance.on('error', (err) => {\r\n console.error('❌ Redis error:', err.message);\r\n });\r\n\r\n redisInstance.on('reconnecting', () => {\r\n console.log('🔄 Redis reconnecting...');\r\n });\r\n }\r\n\r\n // Store globally for hot reloading scenarios\r\n global.__redis = redisInstance;\r\n global.__redisConfig = config;\r\n\r\n return redisInstance;\r\n}\r\n\r\n/**\r\n * Get the Redis instance\r\n * Automatically initializes with default config if not already initialized\r\n */\r\nexport async function getRedis(): Promise<Redis> {\r\n if (redisInstance) {\r\n return redisInstance;\r\n }\r\n\r\n // Check for existing global instance (hot reload scenario)\r\n if (global.__redis) {\r\n redisInstance = global.__redis;\r\n return redisInstance;\r\n }\r\n\r\n // Initialize with default config\r\n return await initRedis({});\r\n}\r\n\r\n/**\r\n * Set a custom Redis instance (useful for testing with mocks)\r\n * @param instance - Redis instance or mock\r\n */\r\nexport function setRedisInstance(instance: Redis): void {\r\n redisInstance = instance;\r\n global.__redis = instance;\r\n}\r\n\r\n/**\r\n * Close Redis connection and clear instance\r\n */\r\nexport async function closeRedis(): Promise<void> {\r\n if (redisInstance) {\r\n await redisInstance.quit();\r\n redisInstance = null;\r\n global.__redis = undefined;\r\n }\r\n}\r\n\r\n/**\r\n * Default Redis export for backward compatibility\r\n * Will auto-initialize on first access\r\n * Note: This is a lazy proxy that initializes Redis on first method call\r\n */\r\nexport const redis = new Proxy({} as Redis, {\r\n get(_target, prop) {\r\n // Return a function that handles async initialization\r\n return async (...args: any[]) => {\r\n const instance = await getRedis();\r\n const value = (instance as any)[prop];\r\n if (typeof value === 'function') {\r\n return value.apply(instance, args);\r\n }\r\n return value;\r\n };\r\n },\r\n});\r\n","/**\r\n * Centralized constants for MCP Redis library\r\n * Eliminates magic numbers and enables consistent configuration\r\n */\r\n\r\n// Redis TTL and Session Management\r\nexport const SESSION_TTL_SECONDS = 43200; // 12 hours\r\nexport const STATE_EXPIRATION_MS = 10 * 60 * 1000; // 10 minutes for OAuth state\r\n\r\n// Heartbeat and Connection\r\nexport const DEFAULT_HEARTBEAT_INTERVAL_MS = 30000; // 30 seconds\r\n\r\n// Redis Key Prefixes\r\nexport const REDIS_KEY_PREFIX = 'mcp:session:';\r\n\r\n// Token Management\r\nexport const TOKEN_EXPIRY_BUFFER_MS = 5 * 60 * 1000; // 5 minute buffer before expiry\r\n\r\n// Client Information\r\nexport const DEFAULT_CLIENT_NAME = 'MCP Assistant';\r\nexport const DEFAULT_CLIENT_URI = 'https://mcp-assistant.in';\r\nexport const DEFAULT_LOGO_URI = 'https://mcp-assistant.in/logo.png';\r\nexport const DEFAULT_POLICY_URI = 'https://mcp-assistant.in/privacy';\r\nexport const SOFTWARE_ID = '@mcp-ts';\r\nexport const SOFTWARE_VERSION = '1.0.0-beta.5';\r\n\r\n// MCP Client Configuration\r\nexport const MCP_CLIENT_NAME = 'mcp-ts-oauth-client';\r\nexport const MCP_CLIENT_VERSION = '2.0';\r\n","\r\nimport type { Redis } from 'ioredis';\r\nimport { customAlphabet } from 'nanoid';\r\nimport { StorageBackend, SessionData, SetClientOptions } from './types';\r\nimport { SESSION_TTL_SECONDS } from '../../shared/constants.js';\r\n\r\n/** first char: letters only (required by OpenAI) */\r\nconst firstChar = customAlphabet(\r\n 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz',\r\n 1\r\n);\r\n\r\n/** remaining chars: alphanumeric */\r\nconst rest = customAlphabet(\r\n 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789',\r\n 11\r\n);\r\n\r\n/**\r\n * Redis implementation of StorageBackend\r\n */\r\nexport class RedisStorageBackend implements StorageBackend {\r\n private readonly DEFAULT_TTL = SESSION_TTL_SECONDS;\r\n private readonly KEY_PREFIX = 'mcp:session:';\r\n\r\n constructor(private redis: Redis) { }\r\n\r\n /**\r\n * Generates Redis key for a specific session\r\n * @private\r\n */\r\n private getSessionKey(identity: string, sessionId: string): string {\r\n return `${this.KEY_PREFIX}${identity}:${sessionId}`;\r\n }\r\n\r\n /**\r\n * Generates Redis key for tracking all sessions for an identity\r\n * @private\r\n */\r\n private getIdentityKey(identity: string): string {\r\n return `mcp:identity:${identity}:sessions`;\r\n }\r\n\r\n generateSessionId(): string {\r\n return firstChar() + rest();\r\n }\r\n\r\n async createSession(session: SessionData, ttl?: number): Promise<void> {\r\n const { sessionId, identity } = session;\r\n if (!sessionId || !identity) throw new Error('identity and sessionId required');\r\n\r\n const sessionKey = this.getSessionKey(identity, sessionId);\r\n const identityKey = this.getIdentityKey(identity);\r\n const effectiveTtl = ttl ?? this.DEFAULT_TTL;\r\n\r\n /** ioredis syntax: set(key, val, 'EX', ttl, 'NX') */\r\n const result = await this.redis.set(\r\n sessionKey,\r\n JSON.stringify(session),\r\n 'EX',\r\n effectiveTtl,\r\n 'NX'\r\n );\r\n\r\n if (result !== 'OK') {\r\n throw new Error(`Session ${sessionId} already exists`);\r\n }\r\n\r\n await this.redis.sadd(identityKey, sessionId);\r\n }\r\n async updateSession(identity: string, sessionId: string, data: Partial<SessionData>, ttl?: number): Promise<void> {\r\n const sessionKey = this.getSessionKey(identity, sessionId);\r\n const effectiveTtl = ttl ?? this.DEFAULT_TTL;\r\n\r\n /** Lua script for atomic parsing, merging, and saving */\r\n const script = `\r\n local currentStr = redis.call(\"GET\", KEYS[1])\r\n if not currentStr then\r\n return 0\r\n end\r\n\r\n local current = cjson.decode(currentStr)\r\n local updates = cjson.decode(ARGV[1])\r\n\r\n for k,v in pairs(updates) do\r\n current[k] = v\r\n end\r\n\r\n redis.call(\"SET\", KEYS[1], cjson.encode(current), \"EX\", ARGV[2])\r\n return 1\r\n `;\r\n\r\n const result = await this.redis.eval(\r\n script,\r\n 1,\r\n sessionKey,\r\n JSON.stringify(data),\r\n effectiveTtl\r\n );\r\n\r\n if (result === 0) {\r\n throw new Error(`Session ${sessionId} not found for identity ${identity}`);\r\n }\r\n }\r\n\r\n async getSession(identity: string, sessionId: string): Promise<SessionData | null> {\r\n try {\r\n const sessionKey = this.getSessionKey(identity, sessionId);\r\n const sessionDataStr = await this.redis.get(sessionKey);\r\n\r\n if (!sessionDataStr) {\r\n return null;\r\n }\r\n\r\n const sessionData: SessionData = JSON.parse(sessionDataStr);\r\n return sessionData;\r\n } catch (error) {\r\n console.error('[RedisStorage] Failed to get session:', error);\r\n return null;\r\n }\r\n }\r\n\r\n async getIdentityMcpSessions(identity: string): Promise<string[]> {\r\n const identityKey = this.getIdentityKey(identity);\r\n try {\r\n return await this.redis.smembers(identityKey);\r\n } catch (error) {\r\n console.error(`[RedisStorage] Failed to get sessions for ${identity}:`, error);\r\n return [];\r\n }\r\n }\r\n\r\n async getIdentitySessionsData(identity: string): Promise<SessionData[]> {\r\n try {\r\n const sessionIds = await this.redis.smembers(this.getIdentityKey(identity));\r\n if (sessionIds.length === 0) return [];\r\n\r\n const results = await Promise.all(\r\n sessionIds.map(async (sessionId) => {\r\n const data = await this.redis.get(this.getSessionKey(identity, sessionId));\r\n return data ? (JSON.parse(data) as SessionData) : null;\r\n })\r\n );\r\n\r\n return results.filter((session): session is SessionData => session !== null);\r\n } catch (error) {\r\n console.error(`[RedisStorage] Failed to get session data for ${identity}:`, error);\r\n return [];\r\n }\r\n }\r\n\r\n async removeSession(identity: string, sessionId: string): Promise<void> {\r\n try {\r\n const sessionKey = this.getSessionKey(identity, sessionId);\r\n const identityKey = this.getIdentityKey(identity);\r\n\r\n await this.redis.srem(identityKey, sessionId);\r\n await this.redis.del(sessionKey);\r\n } catch (error) {\r\n console.error('[RedisStorage] Failed to remove session:', error);\r\n }\r\n }\r\n\r\n async getAllSessionIds(): Promise<string[]> {\r\n try {\r\n const pattern = `${this.KEY_PREFIX}*`;\r\n const keys = await this.redis.keys(pattern);\r\n return keys.map((key) => key.replace(this.KEY_PREFIX, ''));\r\n } catch (error) {\r\n console.error('[RedisStorage] Failed to get all sessions:', error);\r\n return [];\r\n }\r\n }\r\n\r\n async clearAll(): Promise<void> {\r\n try {\r\n const pattern = `${this.KEY_PREFIX}*`;\r\n const keys = await this.redis.keys(pattern);\r\n if (keys.length > 0) {\r\n await this.redis.del(...keys);\r\n }\r\n } catch (error) {\r\n console.error('[RedisStorage] Failed to clear sessions:', error);\r\n }\r\n }\r\n\r\n async cleanupExpiredSessions(): Promise<void> {\r\n try {\r\n const pattern = `${this.KEY_PREFIX}*`;\r\n const keys = await this.redis.keys(pattern);\r\n\r\n for (const key of keys) {\r\n const ttl = await this.redis.ttl(key);\r\n if (ttl <= 0) {\r\n await this.redis.del(key);\r\n }\r\n }\r\n } catch (error) {\r\n console.error('[RedisStorage] Failed to cleanup expired sessions:', error);\r\n }\r\n }\r\n\r\n async disconnect(): Promise<void> {\r\n try {\r\n await this.redis.quit();\r\n } catch (error) {\r\n console.error('[RedisStorage] Failed to disconnect:', error);\r\n }\r\n }\r\n}\r\n","\r\nimport { customAlphabet } from 'nanoid';\r\nimport { StorageBackend, SessionData, SetClientOptions } from './types';\r\n\r\n// first char: letters only (required by OpenAI)\r\nconst firstChar = customAlphabet(\r\n 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz',\r\n 1\r\n);\r\n\r\n// remaining chars: alphanumeric\r\nconst rest = customAlphabet(\r\n 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789',\r\n 11\r\n);\r\n\r\n/**\r\n * In-memory implementation of StorageBackend\r\n * Useful for local development or testing\r\n */\r\nexport class MemoryStorageBackend implements StorageBackend {\r\n // Map<identity:sessionId, SessionData>\r\n private sessions = new Map<string, SessionData>();\r\n\r\n // Map<identity, Set<sessionId>>\r\n private identitySessions = new Map<string, Set<string>>();\r\n\r\n constructor() { }\r\n\r\n private getSessionKey(identity: string, sessionId: string): string {\r\n return `${identity}:${sessionId}`;\r\n }\r\n\r\n generateSessionId(): string {\r\n return firstChar() + rest();\r\n }\r\n\r\n async createSession(session: SessionData, ttl?: number): Promise<void> {\r\n const { sessionId, identity } = session;\r\n if (!sessionId || !identity) throw new Error('identity and sessionId required');\r\n\r\n const sessionKey = this.getSessionKey(identity, sessionId);\r\n if (this.sessions.has(sessionKey)) {\r\n throw new Error(`Session ${sessionId} already exists`);\r\n }\r\n\r\n this.sessions.set(sessionKey, session);\r\n\r\n // Update index\r\n if (!this.identitySessions.has(identity)) {\r\n this.identitySessions.set(identity, new Set());\r\n }\r\n this.identitySessions.get(identity)!.add(sessionId);\r\n // Note: TTL is ignored in memory backend - sessions don't auto-expire\r\n }\r\n\r\n async updateSession(identity: string, sessionId: string, data: Partial<SessionData>, ttl?: number): Promise<void> {\r\n if (!identity || !sessionId) throw new Error('identity and sessionId required');\r\n\r\n const sessionKey = this.getSessionKey(identity, sessionId);\r\n const current = this.sessions.get(sessionKey);\r\n\r\n if (!current) {\r\n throw new Error(`Session ${sessionId} not found`);\r\n }\r\n\r\n const updated = {\r\n ...current,\r\n ...data\r\n };\r\n\r\n this.sessions.set(sessionKey, updated);\r\n // Note: TTL is ignored in memory backend - sessions don't auto-expire\r\n }\r\n\r\n\r\n async getSession(identity: string, sessionId: string): Promise<SessionData | null> {\r\n const sessionKey = this.getSessionKey(identity, sessionId);\r\n return this.sessions.get(sessionKey) || null;\r\n }\r\n\r\n async getIdentityMcpSessions(identity: string): Promise<string[]> {\r\n const set = this.identitySessions.get(identity);\r\n return set ? Array.from(set) : [];\r\n }\r\n\r\n async getIdentitySessionsData(identity: string): Promise<SessionData[]> {\r\n const set = this.identitySessions.get(identity);\r\n if (!set) return [];\r\n\r\n const results: SessionData[] = [];\r\n for (const sessionId of set) {\r\n const session = this.sessions.get(this.getSessionKey(identity, sessionId));\r\n if (session) {\r\n results.push(session);\r\n }\r\n }\r\n return results;\r\n }\r\n\r\n async removeSession(identity: string, sessionId: string): Promise<void> {\r\n const sessionKey = this.getSessionKey(identity, sessionId);\r\n this.sessions.delete(sessionKey);\r\n\r\n const set = this.identitySessions.get(identity);\r\n if (set) {\r\n set.delete(sessionId);\r\n if (set.size === 0) {\r\n this.identitySessions.delete(identity);\r\n }\r\n }\r\n }\r\n\r\n async getAllSessionIds(): Promise<string[]> {\r\n return Array.from(this.sessions.values()).map(s => s.sessionId);\r\n }\r\n\r\n async clearAll(): Promise<void> {\r\n this.sessions.clear();\r\n this.identitySessions.clear();\r\n }\r\n\r\n async cleanupExpiredSessions(): Promise<void> {\r\n // In-memory doesn't implement TTL automatically, \r\n // but we could check createdAt + TTL here if needed.\r\n // For now, no-op.\r\n }\r\n\r\n async disconnect(): Promise<void> {\r\n // No-op for memory\r\n }\r\n}\r\n","\r\nimport { promises as fs } from 'fs';\r\nimport * as path from 'path';\r\nimport { customAlphabet } from 'nanoid';\r\nimport { StorageBackend, SessionData, SetClientOptions } from './types';\r\n\r\n// first char: letters only (required by OpenAI)\r\nconst firstChar = customAlphabet(\r\n 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz',\r\n 1\r\n);\r\n\r\n// remaining chars: alphanumeric\r\nconst rest = customAlphabet(\r\n 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789',\r\n 11\r\n);\r\n\r\n/**\r\n * File system implementation of StorageBackend\r\n * Persists sessions to a JSON file\r\n */\r\nexport class FileStorageBackend implements StorageBackend {\r\n private filePath: string;\r\n private memoryCache: Map<string, SessionData> | null = null;\r\n private initialized = false;\r\n\r\n /**\r\n * @param options.path Path to the JSON file storage (default: ./sessions.json)\r\n */\r\n constructor(options: { path?: string } = {}) {\r\n this.filePath = options.path || './sessions.json';\r\n }\r\n\r\n /**\r\n * Initialize storage: ensure file exists and load into memory cache\r\n */\r\n async init(): Promise<void> {\r\n if (this.initialized) return;\r\n\r\n try {\r\n // Ensure directory exists\r\n const dir = path.dirname(this.filePath);\r\n await fs.mkdir(dir, { recursive: true });\r\n\r\n // Try to read file\r\n const data = await fs.readFile(this.filePath, 'utf-8');\r\n const json = JSON.parse(data);\r\n\r\n this.memoryCache = new Map();\r\n if (Array.isArray(json)) {\r\n json.forEach((s: SessionData) => {\r\n this.memoryCache!.set(this.getSessionKey(s.identity || 'unknown', s.sessionId), s);\r\n });\r\n }\r\n } catch (error: any) {\r\n if (error.code === 'ENOENT') {\r\n // File does not exist, initialize empty\r\n this.memoryCache = new Map();\r\n await this.flush();\r\n } else {\r\n console.error('[FileStorage] Failed to load sessions:', error);\r\n throw error;\r\n }\r\n }\r\n\r\n this.initialized = true;\r\n }\r\n\r\n private async ensureInitialized() {\r\n if (!this.initialized) await this.init();\r\n }\r\n\r\n private async flush(): Promise<void> {\r\n if (!this.memoryCache) return;\r\n const sessions = Array.from(this.memoryCache.values());\r\n await fs.writeFile(this.filePath, JSON.stringify(sessions, null, 2), 'utf-8');\r\n }\r\n\r\n private getSessionKey(identity: string, sessionId: string): string {\r\n return `${identity}:${sessionId}`;\r\n }\r\n\r\n generateSessionId(): string {\r\n return firstChar() + rest();\r\n }\r\n\r\n async createSession(session: SessionData, ttl?: number): Promise<void> {\r\n await this.ensureInitialized();\r\n const { sessionId, identity } = session;\r\n if (!sessionId || !identity) throw new Error('identity and sessionId required');\r\n\r\n const sessionKey = this.getSessionKey(identity, sessionId);\r\n if (this.memoryCache!.has(sessionKey)) {\r\n throw new Error(`Session ${sessionId} already exists`);\r\n }\r\n\r\n this.memoryCache!.set(sessionKey, session);\r\n await this.flush();\r\n // Note: TTL is ignored in file backend - sessions don't auto-expire\r\n }\r\n\r\n async updateSession(identity: string, sessionId: string, data: Partial<SessionData>, ttl?: number): Promise<void> {\r\n await this.ensureInitialized();\r\n if (!identity || !sessionId) throw new Error('identity and sessionId required');\r\n\r\n const sessionKey = this.getSessionKey(identity, sessionId);\r\n const current = this.memoryCache!.get(sessionKey);\r\n\r\n if (!current) {\r\n throw new Error(`Session ${sessionId} not found`);\r\n }\r\n\r\n const updated = {\r\n ...current,\r\n ...data\r\n };\r\n\r\n this.memoryCache!.set(sessionKey, updated);\r\n await this.flush();\r\n // Note: TTL is ignored in file backend - sessions don't auto-expire\r\n }\r\n\r\n async getSession(identity: string, sessionId: string): Promise<SessionData | null> {\r\n await this.ensureInitialized();\r\n const sessionKey = this.getSessionKey(identity, sessionId);\r\n return this.memoryCache!.get(sessionKey) || null;\r\n }\r\n\r\n async getIdentitySessionsData(identity: string): Promise<SessionData[]> {\r\n await this.ensureInitialized();\r\n return Array.from(this.memoryCache!.values()).filter(s => s.identity === identity);\r\n }\r\n\r\n async getIdentityMcpSessions(identity: string): Promise<string[]> {\r\n await this.ensureInitialized();\r\n return Array.from(this.memoryCache!.values())\r\n .filter(s => s.identity === identity)\r\n .map(s => s.sessionId);\r\n }\r\n\r\n async removeSession(identity: string, sessionId: string): Promise<void> {\r\n await this.ensureInitialized();\r\n const sessionKey = this.getSessionKey(identity, sessionId);\r\n if (this.memoryCache!.delete(sessionKey)) {\r\n await this.flush();\r\n }\r\n }\r\n\r\n async getAllSessionIds(): Promise<string[]> {\r\n await this.ensureInitialized();\r\n return Array.from(this.memoryCache!.values()).map(s => s.sessionId);\r\n }\r\n\r\n async clearAll(): Promise<void> {\r\n await this.ensureInitialized();\r\n this.memoryCache!.clear();\r\n await this.flush();\r\n }\r\n\r\n async cleanupExpiredSessions(): Promise<void> {\r\n // Could implement TTL check here using createdAt\r\n await this.ensureInitialized();\r\n }\r\n\r\n async disconnect(): Promise<void> {\r\n // No explicit disconnect needed for file\r\n }\r\n}\r\n","\r\nimport { RedisStorageBackend } from './redis-backend';\r\nimport { MemoryStorageBackend } from './memory-backend';\r\nimport { FileStorageBackend } from './file-backend';\r\nimport type { StorageBackend } from './types';\r\n\r\n// Re-export types\r\nexport * from './types';\r\nexport { RedisStorageBackend, MemoryStorageBackend, FileStorageBackend };\r\n\r\nlet storageInstance: StorageBackend | null = null;\r\nlet storagePromise: Promise<StorageBackend> | null = null;\r\n\r\nasync function createStorage(): Promise<StorageBackend> {\r\n const type = process.env.MCP_TS_STORAGE_TYPE?.toLowerCase();\r\n\r\n // Explicit selection\r\n if (type === 'redis') {\r\n if (!process.env.REDIS_URL) {\r\n console.warn('[Storage] MCP_TS_STORAGE_TYPE is \"redis\" but REDIS_URL is missing');\r\n }\r\n try {\r\n const { getRedis } = await import('./redis.js');\r\n const redis = await getRedis();\r\n console.log('[Storage] Using Redis storage (Explicit)');\r\n return new RedisStorageBackend(redis);\r\n } catch (error: any) {\r\n console.error('[Storage] Failed to initialize Redis:', error.message);\r\n console.log('[Storage] Falling back to In-Memory storage');\r\n return new MemoryStorageBackend();\r\n }\r\n }\r\n\r\n if (type === 'file') {\r\n const filePath = process.env.MCP_TS_STORAGE_FILE;\r\n if (!filePath) {\r\n console.warn('[Storage] MCP_TS_STORAGE_TYPE is \"file\" but MCP_TS_STORAGE_FILE is missing');\r\n }\r\n console.log(`[Storage] Using File storage (${filePath}) (Explicit)`);\r\n const store = new FileStorageBackend({ path: filePath });\r\n store.init().catch(err => console.error('[Storage] Failed to initialize file storage:', err));\r\n return store;\r\n }\r\n\r\n if (type === 'memory') {\r\n console.log('[Storage] Using In-Memory storage (Explicit)');\r\n return new MemoryStorageBackend();\r\n }\r\n\r\n // Automatic inference (Fallback)\r\n if (process.env.REDIS_URL) {\r\n try {\r\n const { getRedis } = await import('./redis.js');\r\n const redis = await getRedis();\r\n console.log('[Storage] Auto-detected REDIS_URL. Using Redis storage.');\r\n return new RedisStorageBackend(redis);\r\n } catch (error: any) {\r\n console.error('[Storage] Redis auto-detection failed:', error.message);\r\n console.log('[Storage] Falling back to In-Memory storage');\r\n return new MemoryStorageBackend();\r\n }\r\n }\r\n\r\n if (process.env.MCP_TS_STORAGE_FILE) {\r\n console.log(`[Storage] Auto-detected MCP_TS_STORAGE_FILE. Using File storage (${process.env.MCP_TS_STORAGE_FILE}).`);\r\n const store = new FileStorageBackend({ path: process.env.MCP_TS_STORAGE_FILE });\r\n store.init().catch(err => console.error('[Storage] Failed to initialize file storage:', err));\r\n return store;\r\n }\r\n\r\n console.log('[Storage] No storage configured. Using In-Memory storage (Default).');\r\n return new MemoryStorageBackend();\r\n}\r\n\r\nasync function getStorage(): Promise<StorageBackend> {\r\n if (storageInstance) {\r\n return storageInstance;\r\n }\r\n\r\n if (!storagePromise) {\r\n storagePromise = createStorage();\r\n }\r\n\r\n storageInstance = await storagePromise;\r\n return storageInstance;\r\n}\r\n\r\n/**\r\n * Set the storage instance (for testing)\r\n * @internal\r\n * @param instance - StorageBackend instance or null to reset\r\n */\r\nexport function _setStorageInstanceForTesting(instance: StorageBackend | null): void {\r\n storageInstance = instance;\r\n if (!instance) {\r\n storagePromise = null;\r\n }\r\n}\r\n\r\n/**\r\n * Global session store instance\r\n * Uses lazy initialization with a Proxy to handle async setup transparently\r\n */\r\nexport const storage: StorageBackend = new Proxy({} as StorageBackend, {\r\n get(_target, prop) {\r\n return async (...args: any[]) => {\r\n const instance = await getStorage();\r\n const value = (instance as any)[prop];\r\n if (typeof value === 'function') {\r\n return value.apply(instance, args);\r\n }\r\n return value;\r\n };\r\n },\r\n});\r\n","\r\nimport type { OAuthClientProvider } from \"@modelcontextprotocol/sdk/client/auth.js\";\r\nimport type {\r\n OAuthClientInformation,\r\n OAuthClientInformationFull,\r\n OAuthClientMetadata,\r\n OAuthTokens\r\n} from \"@modelcontextprotocol/sdk/shared/auth.js\";\r\nimport { storage, SessionData } from \"../storage/index.js\";\r\nimport { TOKEN_EXPIRY_BUFFER_MS } from '../../shared/constants.js';\r\n\r\n/**\r\n * Extension of OAuthClientProvider interface with additional methods\r\n * Enables server-specific tracking and state management\r\n */\r\nexport interface AgentsOAuthProvider extends OAuthClientProvider {\r\n authUrl: string | undefined;\r\n clientId: string | undefined;\r\n serverId: string | undefined;\r\n checkState(\r\n state: string\r\n ): Promise<{ valid: boolean; serverId?: string; error?: string }>;\r\n consumeState(state: string): Promise<void>;\r\n deleteCodeVerifier(): Promise<void>;\r\n isTokenExpired(): boolean;\r\n setTokenExpiresAt(expiresAt: number): void;\r\n}\r\n\r\n/**\r\n * Storage-backed OAuth provider implementation for MCP\r\n * Stores OAuth tokens, client information, and PKCE verifiers using the configured StorageBackend\r\n */\r\nexport class StorageOAuthClientProvider implements AgentsOAuthProvider {\r\n private _authUrl: string | undefined;\r\n private _clientId: string | undefined;\r\n private onRedirectCallback?: (url: string) => void;\r\n private tokenExpiresAt?: number;\r\n\r\n /**\r\n * Creates a new Storage-backed OAuth provider\r\n * @param identity - User/Client identifier\r\n * @param serverId - Server identifier (for tracking which server this OAuth session belongs to)\r\n * @param sessionId - Session identifier (used as OAuth state)\r\n * @param clientName - OAuth client name\r\n * @param baseRedirectUrl - OAuth callback URL\r\n * @param onRedirect - Optional callback when redirect to authorization is needed\r\n */\r\n constructor(\r\n public identity: string,\r\n public serverId: string,\r\n public sessionId: string,\r\n public clientName: string,\r\n public baseRedirectUrl: string,\r\n onRedirect?: (url: string) => void\r\n ) {\r\n this.onRedirectCallback = onRedirect;\r\n }\r\n\r\n get clientMetadata(): OAuthClientMetadata {\r\n return {\r\n client_name: this.clientName,\r\n client_uri: this.clientUri,\r\n grant_types: [\"authorization_code\", \"refresh_token\"],\r\n redirect_uris: [this.redirectUrl],\r\n response_types: [\"code\"],\r\n token_endpoint_auth_method: \"none\",\r\n ...(this._clientId ? { client_id: this._clientId } : {})\r\n };\r\n }\r\n\r\n get clientUri() {\r\n return new URL(this.redirectUrl).origin;\r\n }\r\n\r\n get redirectUrl() {\r\n return this.baseRedirectUrl;\r\n }\r\n\r\n get clientId() {\r\n return this._clientId;\r\n }\r\n\r\n set clientId(clientId_: string | undefined) {\r\n this._clientId = clientId_;\r\n }\r\n\r\n /**\r\n * Loads OAuth data from storage session\r\n * @private\r\n */\r\n private async getSessionData(): Promise<SessionData> {\r\n const data = await storage.getSession(this.identity, this.sessionId);\r\n if (!data) {\r\n // Return empty/partial object if not found\r\n return {} as SessionData;\r\n }\r\n return data;\r\n }\r\n\r\n /**\r\n * Saves OAuth data to storage\r\n * @param data - Partial OAuth data to save\r\n * @private\r\n * @throws Error if session doesn't exist (session must be created by controller layer)\r\n */\r\n private async saveSessionData(data: Partial<SessionData>): Promise<void> {\r\n await storage.updateSession(this.identity, this.sessionId, data);\r\n }\r\n\r\n /**\r\n * Retrieves stored OAuth client information\r\n */\r\n async clientInformation(): Promise<OAuthClientInformation | undefined> {\r\n const data = await this.getSessionData();\r\n\r\n if (data.clientId && !this._clientId) {\r\n this._clientId = data.clientId;\r\n }\r\n\r\n return data.clientInformation;\r\n }\r\n\r\n /**\r\n * Stores OAuth client information\r\n */\r\n async saveClientInformation(clientInformation: OAuthClientInformationFull): Promise<void> {\r\n await this.saveSessionData({\r\n clientInformation,\r\n clientId: clientInformation.client_id\r\n });\r\n this.clientId = clientInformation.client_id;\r\n }\r\n\r\n /**\r\n * Stores OAuth tokens\r\n */\r\n async saveTokens(tokens: OAuthTokens): Promise<void> {\r\n const data: Partial<SessionData> = { tokens };\r\n\r\n if (tokens.expires_in) {\r\n this.tokenExpiresAt = Date.now() + (tokens.expires_in * 1000) - TOKEN_EXPIRY_BUFFER_MS;\r\n }\r\n\r\n await this.saveSessionData(data);\r\n }\r\n\r\n get authUrl() {\r\n return this._authUrl;\r\n }\r\n\r\n async state(): Promise<string> {\r\n return this.sessionId;\r\n }\r\n\r\n async checkState(state: string): Promise<{ valid: boolean; serverId?: string; error?: string }> {\r\n const data = await storage.getSession(this.identity, this.sessionId);\r\n\r\n if (!data) {\r\n return { valid: false, error: \"Session not found\" };\r\n }\r\n\r\n return { valid: true, serverId: this.serverId };\r\n }\r\n\r\n async consumeState(state: string): Promise<void> {\r\n // No-op\r\n }\r\n\r\n async redirectToAuthorization(authUrl: URL): Promise<void> {\r\n this._authUrl = authUrl.toString();\r\n if (this.onRedirectCallback) {\r\n this.onRedirectCallback(authUrl.toString());\r\n }\r\n }\r\n\r\n async invalidateCredentials(\r\n scope: \"all\" | \"client\" | \"tokens\" | \"verifier\"\r\n ): Promise<void> {\r\n if (scope === \"all\") {\r\n await storage.removeSession(this.identity, this.sessionId);\r\n } else {\r\n const data = await this.getSessionData();\r\n // Create a copy to modify\r\n const updates: Partial<SessionData> = {};\r\n\r\n if (scope === \"client\") {\r\n updates.clientInformation = undefined;\r\n updates.clientId = undefined;\r\n } else if (scope === \"tokens\") {\r\n updates.tokens = undefined;\r\n } else if (scope === \"verifier\") {\r\n updates.codeVerifier = undefined;\r\n }\r\n await this.saveSessionData(updates);\r\n }\r\n }\r\n\r\n async saveCodeVerifier(verifier: string): Promise<void> {\r\n await this.saveSessionData({ codeVerifier: verifier });\r\n }\r\n\r\n async codeVerifier(): Promise<string> {\r\n const data = await this.getSessionData();\r\n\r\n if (data.clientId && !this._clientId) {\r\n this._clientId = data.clientId;\r\n }\r\n\r\n if (!data.codeVerifier) {\r\n throw new Error(\"No code verifier found\");\r\n }\r\n return data.codeVerifier;\r\n }\r\n\r\n async deleteCodeVerifier(): Promise<void> {\r\n await this.saveSessionData({ codeVerifier: undefined });\r\n }\r\n\r\n async tokens(): Promise<OAuthTokens | undefined> {\r\n const data = await this.getSessionData();\r\n\r\n if (data.clientId && !this._clientId) {\r\n this._clientId = data.clientId;\r\n }\r\n\r\n return data.tokens;\r\n }\r\n\r\n isTokenExpired(): boolean {\r\n if (!this.tokenExpiresAt) {\r\n return false;\r\n }\r\n return Date.now() >= this.tokenExpiresAt;\r\n }\r\n\r\n setTokenExpiresAt(expiresAt: number): void {\r\n this.tokenExpiresAt = expiresAt;\r\n }\r\n}\r\n","/**\r\n * Sanitize server name to create a valid server label\r\n * Must start with a letter and contain only letters, digits, '-' and '_'\r\n */\r\nexport function sanitizeServerLabel(name: string): string {\r\n let sanitized = name\r\n .replace(/[^a-zA-Z0-9-_]/g, '_')\r\n .replace(/_{2,}/g, '_')\r\n .toLowerCase();\r\n\r\n if (!/^[a-zA-Z]/.test(sanitized)) {\r\n sanitized = 's_' + sanitized;\r\n }\r\n\r\n return sanitized;\r\n}\r\n","/**\r\n * Simple event emitter pattern for MCP connection events\r\n * Inspired by Cloudflare's agents pattern but adapted for serverless\r\n */\r\n\r\nexport type Disposable = {\r\n dispose(): void;\r\n};\r\n\r\nexport type Event<T> = (listener: (event: T) => void) => Disposable;\r\n\r\n/**\r\n * Event emitter class for type-safe event handling\r\n * Similar to Cloudflare's Emitter but simplified for our use case\r\n */\r\nexport class Emitter<T> {\r\n private listeners: Set<(event: T) => void> = new Set();\r\n\r\n /**\r\n * Subscribe to events\r\n * @param listener - Callback function to handle events\r\n * @returns Disposable to unsubscribe\r\n */\r\n get event(): Event<T> {\r\n return (listener: (event: T) => void) => {\r\n this.listeners.add(listener);\r\n return {\r\n dispose: () => {\r\n this.listeners.delete(listener);\r\n },\r\n };\r\n };\r\n }\r\n\r\n /**\r\n * Fire an event to all listeners\r\n * @param event - Event data to emit\r\n */\r\n fire(event: T): void {\r\n for (const listener of this.listeners) {\r\n try {\r\n listener(event);\r\n } catch (error) {\r\n console.error('[Emitter] Error in event listener:', error);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Clear all listeners\r\n */\r\n dispose(): void {\r\n this.listeners.clear();\r\n }\r\n\r\n /**\r\n * Get number of active listeners\r\n */\r\n get listenerCount(): number {\r\n return this.listeners.size;\r\n }\r\n}\r\n\r\n/**\r\n * Connection state types matching your existing ConnectionStatus\r\n * Extended with more granular states for better observability\r\n */\r\nexport type McpConnectionState =\r\n | 'DISCONNECTED' // Not connected\r\n | 'CONNECTING' // Establishing transport connection to MCP server\r\n | 'AUTHENTICATING' // OAuth flow in progress\r\n | 'AUTHENTICATED' // OAuth complete, pre-connect\r\n | 'DISCOVERING' // Discovering server capabilities (tools, resources, prompts)\r\n | 'CONNECTED' // Transport connection established\r\n | 'READY' // Fully connected and ready to use\r\n | 'VALIDATING' // Validating existing session\r\n | 'RECONNECTING' // Attempting to reconnect\r\n | 'INITIALIZING' // Initializing session or connection\r\n | 'FAILED'; // Connection error at some point\r\n\r\n/**\r\n * MCP Connection Event Types\r\n * Discriminated union for type-safe event handling\r\n */\r\nexport type McpConnectionEvent =\r\n | {\r\n type: 'state_changed';\r\n sessionId: string;\r\n serverId: string;\r\n serverName: string;\r\n state: McpConnectionState;\r\n previousState: McpConnectionState;\r\n timestamp: number;\r\n }\r\n | {\r\n type: 'tools_discovered';\r\n sessionId: string;\r\n serverId: string;\r\n toolCount: number;\r\n tools: any[];\r\n timestamp: number;\r\n }\r\n | {\r\n type: 'auth_required';\r\n sessionId: string;\r\n serverId: string;\r\n authUrl: string;\r\n timestamp: number;\r\n }\r\n | {\r\n type: 'error';\r\n sessionId: string;\r\n serverId: string;\r\n error: string;\r\n errorType: 'connection' | 'auth' | 'validation' | 'unknown';\r\n timestamp: number;\r\n }\r\n | {\r\n type: 'disconnected';\r\n sessionId: string;\r\n serverId: string;\r\n reason?: string;\r\n timestamp: number;\r\n }\r\n | {\r\n type: 'progress';\r\n sessionId: string;\r\n serverId: string;\r\n message: string;\r\n timestamp: number;\r\n };\r\n\r\n/**\r\n * Observability event for debugging and monitoring\r\n */\r\nexport interface McpObservabilityEvent {\r\n type?: string;\r\n level?: 'debug' | 'info' | 'warn' | 'error';\r\n message?: string;\r\n displayMessage?: string;\r\n sessionId?: string;\r\n serverId?: string;\r\n payload?: Record<string, any>;\r\n metadata?: Record<string, any>; // Kept for backward compatibility\r\n timestamp: number;\r\n id?: string;\r\n}\r\n\r\n/**\r\n * DisposableStore for managing multiple disposables\r\n * Useful for cleanup in React hooks\r\n */\r\nexport class DisposableStore {\r\n private disposables: Set<Disposable> = new Set();\r\n\r\n add(disposable: Disposable): void {\r\n this.disposables.add(disposable);\r\n }\r\n\r\n dispose(): void {\r\n for (const disposable of this.disposables) {\r\n disposable.dispose();\r\n }\r\n this.disposables.clear();\r\n }\r\n}\r\n","/**\r\n * Standardized error classes for MCP Redis library\r\n * Provides consistent error handling across the codebase\r\n */\r\n\r\n/**\r\n * Base error class for all MCP-related errors\r\n */\r\nexport class McpError extends Error {\r\n constructor(\r\n public readonly code: string,\r\n message: string,\r\n public readonly cause?: Error\r\n ) {\r\n super(message);\r\n this.name = 'McpError';\r\n // Maintain proper prototype chain for instanceof checks\r\n Object.setPrototypeOf(this, new.target.prototype);\r\n }\r\n\r\n toJSON() {\r\n return {\r\n name: this.name,\r\n code: this.code,\r\n message: this.message,\r\n ...(this.cause ? { cause: this.cause.message } : {}),\r\n };\r\n }\r\n}\r\n\r\n/**\r\n * Thrown when OAuth authorization is required\r\n */\r\nexport class UnauthorizedError extends McpError {\r\n constructor(message: string = 'OAuth authorization required', cause?: Error) {\r\n super('UNAUTHORIZED', message, cause);\r\n this.name = 'UnauthorizedError';\r\n }\r\n}\r\n\r\n/**\r\n * Thrown when connection to MCP server fails\r\n */\r\nexport class ConnectionError extends McpError {\r\n constructor(message: string, cause?: Error) {\r\n super('CONNECTION_ERROR', message, cause);\r\n this.name = 'ConnectionError';\r\n }\r\n}\r\n\r\n/**\r\n * Thrown when session is not found or expired\r\n */\r\nexport class SessionNotFoundError extends McpError {\r\n constructor(sessionId: string, cause?: Error) {\r\n super('SESSION_NOT_FOUND', `Session not found: ${sessionId}`, cause);\r\n this.name = 'SessionNotFoundError';\r\n }\r\n}\r\n\r\n/**\r\n * Thrown when session validation fails\r\n */\r\nexport class SessionValidationError extends McpError {\r\n constructor(message: string, cause?: Error) {\r\n super('SESSION_VALIDATION_ERROR', message, cause);\r\n this.name = 'SessionValidationError';\r\n }\r\n}\r\n\r\n/**\r\n * Thrown when authentication fails\r\n */\r\nexport class AuthenticationError extends McpError {\r\n constructor(message: string, cause?: Error) {\r\n super('AUTH_ERROR', message, cause);\r\n this.name = 'AuthenticationError';\r\n }\r\n}\r\n\r\n/**\r\n * Thrown when OAuth state validation fails\r\n */\r\nexport class InvalidStateError extends McpError {\r\n constructor(message: string = 'Invalid OAuth state', cause?: Error) {\r\n super('INVALID_STATE', message, cause);\r\n this.name = 'InvalidStateError';\r\n }\r\n}\r\n\r\n/**\r\n * Thrown when client is not connected\r\n */\r\nexport class NotConnectedError extends McpError {\r\n constructor(message: string = 'Not connected to server', cause?: Error) {\r\n super('NOT_CONNECTED', message, cause);\r\n this.name = 'NotConnectedError';\r\n }\r\n}\r\n\r\n/**\r\n * Thrown when required configuration is missing\r\n */\r\nexport class ConfigurationError extends McpError {\r\n constructor(message: string, cause?: Error) {\r\n super('CONFIGURATION_ERROR', message, cause);\r\n this.name = 'ConfigurationError';\r\n }\r\n}\r\n\r\n/**\r\n * Thrown when tool execution fails\r\n */\r\nexport class ToolExecutionError extends McpError {\r\n constructor(toolName: string, message: string, cause?: Error) {\r\n super('TOOL_EXECUTION_ERROR', `Tool '${toolName}' failed: ${message}`, cause);\r\n this.name = 'ToolExecutionError';\r\n }\r\n}\r\n\r\n/**\r\n * RPC error codes for SSE communication\r\n */\r\nexport const RpcErrorCodes = {\r\n EXECUTION_ERROR: 'EXECUTION_ERROR',\r\n MISSING_IDENTITY: 'MISSING_IDENTITY',\r\n UNAUTHORIZED: 'UNAUTHORIZED',\r\n NO_CONNECTION: 'NO_CONNECTION',\r\n UNKNOWN_METHOD: 'UNKNOWN_METHOD',\r\n INVALID_PARAMS: 'INVALID_PARAMS',\r\n} as const;\r\n\r\nexport type RpcErrorCode = typeof RpcErrorCodes[keyof typeof RpcErrorCodes];\r\n","import { Client } from '@modelcontextprotocol/sdk/client/index.js';\r\nimport { nanoid } from 'nanoid';\r\nimport { StreamableHTTPClientTransport } from '@modelcontextprotocol/sdk/client/streamableHttp.js';\r\nimport { SSEClientTransport } from '@modelcontextprotocol/sdk/client/sse.js';\r\nimport { Transport } from '@modelcontextprotocol/sdk/shared/transport.js'; /** Import base Transport type */\r\nimport {\r\n UnauthorizedError as SDKUnauthorizedError,\r\n refreshAuthorization,\r\n discoverOAuthProtectedResourceMetadata,\r\n discoverAuthorizationServerMetadata,\r\n auth\r\n} from '@modelcontextprotocol/sdk/client/auth.js';\r\nimport {\r\n ListToolsRequest,\r\n ListToolsResult,\r\n ListToolsResultSchema,\r\n CallToolRequest,\r\n CallToolResult,\r\n CallToolResultSchema,\r\n ListPromptsRequest,\r\n ListPromptsResult,\r\n ListPromptsResultSchema,\r\n GetPromptRequest,\r\n GetPromptResult,\r\n GetPromptResultSchema,\r\n ListResourcesRequest,\r\n ListResourcesResult,\r\n ListResourcesResultSchema,\r\n ReadResourceRequest,\r\n ReadResourceResult,\r\n ReadResourceResultSchema,\r\n} from '@modelcontextprotocol/sdk/types.js';\r\nimport type { OAuthClientMetadata, OAuthTokens, OAuthClientInformationFull } from '@modelcontextprotocol/sdk/shared/auth.js';\r\nimport { StorageOAuthClientProvider, type AgentsOAuthProvider } from './storage-oauth-provider.js';\r\nimport { sanitizeServerLabel } from '../../shared/utils.js';\r\nimport { Emitter, type McpConnectionEvent, type McpObservabilityEvent, type McpConnectionState } from '../../shared/events.js';\r\nimport { UnauthorizedError } from '../../shared/errors.js';\r\nimport { storage } from '../storage/index.js';\r\nimport { SESSION_TTL_SECONDS, STATE_EXPIRATION_MS } from '../../shared/constants.js';\r\n\r\n/**\r\n * Supported MCP transport types\r\n */\r\nexport type TransportType = 'sse' | 'streamable_http';\r\n\r\nexport interface MCPOAuthClientOptions {\r\n serverUrl?: string;\r\n serverName?: string;\r\n callbackUrl?: string;\r\n onRedirect?: (url: string) => void;\r\n identity: string;\r\n serverId?: string; /** Optional - loaded from session if not provided */\r\n sessionId: string; /** Required - primary key for session lookup */\r\n transportType?: TransportType;\r\n tokens?: OAuthTokens;\r\n tokenExpiresAt?: number;\r\n clientInformation?: OAuthClientInformationFull;\r\n clientId?: string;\r\n clientSecret?: string;\r\n onSaveTokens?: (tokens: OAuthTokens) => void;\r\n headers?: Record<string, string>;\r\n /** OAuth Client Metadata (optional - user application info) */\r\n clientName?: string;\r\n clientUri?: string;\r\n logoUri?: string;\r\n policyUri?: string;\r\n}\r\n\r\n/**\r\n * MCP Client with OAuth 2.1 authentication support\r\n * Manages connections to MCP servers with automatic token refresh and session restoration\r\n * Emits connection lifecycle events for observability\r\n */\r\nexport class MCPClient {\r\n private client: Client | null = null;\r\n public oauthProvider: AgentsOAuthProvider | null = null;\r\n private transport: StreamableHTTPClientTransport | SSEClientTransport | null = null;\r\n private identity: string;\r\n private serverId?: string;\r\n private sessionId: string;\r\n private serverName?: string;\r\n private transportType: TransportType | undefined;\r\n private serverUrl: string | undefined;\r\n private callbackUrl: string | undefined;\r\n private onRedirect: ((url: string) => void) | undefined;\r\n private tokens?: OAuthTokens;\r\n private tokenExpiresAt?: number;\r\n private clientInformation?: OAuthClientInformationFull;\r\n private clientId?: string;\r\n private clientSecret?: string;\r\n private onSaveTokens?: (tokens: OAuthTokens) => void;\r\n private headers?: Record<string, string>;\r\n /** OAuth Client Metadata */\r\n private clientName?: string;\r\n private clientUri?: string;\r\n private logoUri?: string;\r\n private policyUri?: string;\r\n\r\n\r\n /** Event emitters for connection lifecycle */\r\n private readonly _onConnectionEvent = new Emitter<McpConnectionEvent>();\r\n public readonly onConnectionEvent = this._onConnectionEvent.event;\r\n\r\n private readonly _onObservabilityEvent = new Emitter<McpObservabilityEvent>();\r\n public readonly onObservabilityEvent = this._onObservabilityEvent.event;\r\n\r\n private currentState: McpConnectionState = 'DISCONNECTED';\r\n\r\n /**\r\n * Creates a new MCP client instance\r\n * Can be initialized with minimal options (identity + sessionId) for session restoration\r\n * @param options - Client configuration options\r\n */\r\n constructor(options: MCPOAuthClientOptions) {\r\n this.serverUrl = options.serverUrl;\r\n this.serverName = options.serverName;\r\n this.callbackUrl = options.callbackUrl;\r\n this.onRedirect = options.onRedirect;\r\n this.identity = options.identity;\r\n this.serverId = options.serverId;\r\n this.sessionId = options.sessionId;\r\n this.transportType = options.transportType;\r\n this.tokens = options.tokens;\r\n this.tokenExpiresAt = options.tokenExpiresAt;\r\n this.clientInformation = options.clientInformation;\r\n this.clientId = options.clientId;\r\n this.clientSecret = options.clientSecret;\r\n this.onSaveTokens = options.onSaveTokens;\r\n this.headers = options.headers;\r\n this.clientName = options.clientName;\r\n this.clientUri = options.clientUri;\r\n this.logoUri = options.logoUri;\r\n this.policyUri = options.policyUri;\r\n }\r\n\r\n /**\r\n * Emit a connection state change event\r\n * @private\r\n */\r\n private emitStateChange(newState: McpConnectionState): void {\r\n const previousState = this.currentState;\r\n this.currentState = newState;\r\n\r\n if (!this.serverId) return;\r\n\r\n this._onConnectionEvent.fire({\r\n type: 'state_changed',\r\n sessionId: this.sessionId,\r\n serverId: this.serverId,\r\n serverName: this.serverName || this.serverId,\r\n state: newState,\r\n previousState,\r\n timestamp: Date.now(),\r\n });\r\n\r\n this._onObservabilityEvent.fire({\r\n type: 'mcp:client:state_change',\r\n level: 'info',\r\n message: `Connection state: ${previousState} → ${newState}`,\r\n displayMessage: `State changed to ${newState}`,\r\n sessionId: this.sessionId,\r\n serverId: this.serverId,\r\n payload: { previousState, newState },\r\n timestamp: Date.now(),\r\n id: nanoid(),\r\n });\r\n }\r\n\r\n /**\r\n * Emit an error event\r\n * @private\r\n */\r\n private emitError(error: string, errorType: 'connection' | 'auth' | 'validation' | 'unknown' = 'unknown'): void {\r\n if (!this.serverId) return;\r\n\r\n this._onConnectionEvent.fire({\r\n type: 'error',\r\n sessionId: this.sessionId,\r\n serverId: this.serverId,\r\n error,\r\n errorType,\r\n timestamp: Date.now(),\r\n });\r\n\r\n this._onObservabilityEvent.fire({\r\n type: 'mcp:client:error',\r\n level: 'error',\r\n message: error,\r\n displayMessage: error,\r\n sessionId: this.sessionId,\r\n serverId: this.serverId,\r\n payload: { errorType, error },\r\n timestamp: Date.now(),\r\n id: nanoid(),\r\n });\r\n }\r\n\r\n /**\r\n * Emit a progress event\r\n * @private\r\n */\r\n private emitProgress(message: string): void {\r\n if (!this.serverId) return;\r\n\r\n this._onConnectionEvent.fire({\r\n type: 'progress',\r\n sessionId: this.sessionId,\r\n serverId: this.serverId,\r\n message,\r\n timestamp: Date.now(),\r\n });\r\n }\r\n\r\n /**\r\n * Get current connection state\r\n */\r\n getConnectionState(): McpConnectionState {\r\n return this.currentState;\r\n }\r\n\r\n /**\r\n * Helper to create a transport instance\r\n * @param type - The transport type to create\r\n * @returns Configured transport instance\r\n * @private\r\n */\r\n private getTransport(type: TransportType): StreamableHTTPClientTransport | SSEClientTransport {\r\n if (!this.serverUrl) {\r\n throw new Error('Server URL is required to create transport');\r\n }\r\n\r\n const baseUrl = new URL(this.serverUrl);\r\n const transportOptions = {\r\n authProvider: this.oauthProvider!,\r\n ...(this.headers && { headers: this.headers }),\r\n /**\r\n * Custom fetch implementation to handle connection timeouts.\r\n * Observation: SDK 1.24.0+ connections may hang indefinitely in some environments.\r\n * This wrapper enforces a timeout and properly uses AbortController to unblock the request.\r\n */\r\n fetch: (url: RequestInfo | URL, init?: RequestInit) => {\r\n const timeout = 30000;\r\n const controller = new AbortController();\r\n const timeoutId = setTimeout(() => controller.abort(), timeout);\r\n const signal = init?.signal ?\r\n // @ts-ignore: AbortSignal.any is available in Node 20+\r\n (AbortSignal.any ? AbortSignal.any([init.signal, controller.signal]) : controller.signal) :\r\n controller.signal;\r\n\r\n return fetch(url, { ...init, signal }).finally(() => clearTimeout(timeoutId));\r\n }\r\n };\r\n\r\n if (type === 'sse') {\r\n return new SSEClientTransport(baseUrl, transportOptions);\r\n } else {\r\n return new StreamableHTTPClientTransport(baseUrl, transportOptions);\r\n }\r\n }\r\n\r\n /**\r\n * Initializes client components (client, transport, OAuth provider)\r\n * Loads missing configuration from Redis session store if needed\r\n * This method is idempotent and safe to call multiple times\r\n * @private\r\n */\r\n private async initialize(): Promise<void> {\r\n if (this.client && this.oauthProvider) {\r\n return;\r\n }\r\n\r\n this.emitStateChange('INITIALIZING');\r\n this.emitProgress('Loading session configuration...');\r\n\r\n if (!this.serverUrl || !this.callbackUrl || !this.serverId) {\r\n const sessionData = await storage.getSession(this.identity, this.sessionId);\r\n if (!sessionData) {\r\n throw new Error(`Session not found: ${this.sessionId}`);\r\n }\r\n\r\n this.serverUrl = this.serverUrl || sessionData.serverUrl;\r\n this.callbackUrl = this.callbackUrl || sessionData.callbackUrl;\r\n /**\r\n * Do NOT load transportType from session if not explicitly provided.\r\n * We want to re-negotiate (try streamable -> sse) on new connections if in \"Auto\" mode.\r\n * this.transportType = this.transportType || sessionData.transportType; \r\n */\r\n this.serverName = this.serverName || sessionData.serverName;\r\n this.serverId = this.serverId || sessionData.serverId || 'unknown';\r\n this.headers = this.headers || sessionData.headers;\r\n }\r\n\r\n if (!this.serverUrl || !this.callbackUrl || !this.serverId) {\r\n throw new Error('Missing required connection metadata');\r\n }\r\n\r\n const clientMetadata: OAuthClientMetadata = {\r\n client_name: this.clientName || 'MCP Assistant',\r\n redirect_uris: [this.callbackUrl],\r\n grant_types: ['authorization_code', 'refresh_token'],\r\n response_types: ['code'],\r\n token_endpoint_auth_method: this.clientSecret ? 'client_secret_basic' : 'none',\r\n client_uri: this.clientUri || 'https://mcp-assistant.in',\r\n logo_uri: this.logoUri || 'https://mcp-assistant.in/logo.png',\r\n policy_uri: this.policyUri || 'https://mcp-assistant.in/privacy',\r\n software_id: '@mcp-ts',\r\n software_version: '1.0.0-beta.4',\r\n ...(this.clientId ? { client_id: this.clientId } : {}),\r\n ...(this.clientSecret ? { client_secret: this.clientSecret } : {}),\r\n };\r\n\r\n if (!this.oauthProvider) {\r\n if (!this.serverId) {\r\n throw new Error('serverId required for OAuth provider initialization');\r\n }\r\n\r\n this.oauthProvider = new StorageOAuthClientProvider(\r\n this.identity,\r\n this.serverId,\r\n this.sessionId,\r\n clientMetadata.client_name ?? 'MCP Assistant',\r\n this.callbackUrl,\r\n (redirectUrl: string) => {\r\n if (this.onRedirect) {\r\n this.onRedirect(redirectUrl);\r\n }\r\n }\r\n );\r\n\r\n if (this.clientId && this.oauthProvider) {\r\n this.oauthProvider.clientId = this.clientId;\r\n }\r\n }\r\n\r\n if (!this.client) {\r\n this.client = new Client(\r\n {\r\n name: 'mcp-ts-oauth-client',\r\n version: '2.0',\r\n },\r\n { capabilities: {} }\r\n );\r\n }\r\n\r\n // Create session in storage if it doesn't exist yet\r\n // This is needed BEFORE OAuth flow starts because the OAuth provider\r\n // will call saveCodeVerifier() which requires the session to exist\r\n const existingSession = await storage.getSession(this.identity, this.sessionId);\r\n if (!existingSession && this.serverId && this.serverUrl && this.callbackUrl) {\r\n console.log(`[MCPClient] Creating initial session ${this.sessionId} for OAuth flow`);\r\n await storage.createSession({\r\n sessionId: this.sessionId,\r\n identity: this.identity,\r\n serverId: this.serverId,\r\n serverName: this.serverName,\r\n serverUrl: this.serverUrl,\r\n callbackUrl: this.callbackUrl,\r\n transportType: this.transportType || 'streamable_http',\r\n createdAt: Date.now(),\r\n }, Math.floor(STATE_EXPIRATION_MS / 1000)); // Short TTL until connection succeeds\r\n }\r\n }\r\n\r\n /**\r\n * Saves current session state to storage\r\n * Creates new session if it doesn't exist, updates if it does\r\n * @param ttl - Time-to-live in seconds (defaults to 12hr for connected sessions)\r\n * @private\r\n */\r\n private async saveSession(ttl: number = SESSION_TTL_SECONDS): Promise<void> {\r\n if (!this.sessionId || !this.serverId || !this.serverUrl || !this.callbackUrl) {\r\n return;\r\n }\r\n\r\n const sessionData = {\r\n sessionId: this.sessionId,\r\n identity: this.identity,\r\n serverId: this.serverId,\r\n serverName: this.serverName,\r\n serverUrl: this.serverUrl,\r\n callbackUrl: this.callbackUrl,\r\n transportType: this.transportType || 'streamable_http' as TransportType,\r\n createdAt: Date.now(),\r\n };\r\n\r\n // Try to update first, create if doesn't exist\r\n const existingSession = await storage.getSession(this.identity, this.sessionId);\r\n if (existingSession) {\r\n await storage.updateSession(this.identity, this.sessionId, sessionData, ttl);\r\n } else {\r\n await storage.createSession(sessionData, ttl);\r\n }\r\n }\r\n\r\n /**\r\n * Try to connect using available transports\r\n * @returns The corrected transport type object if successful\r\n * @private\r\n */\r\n private async tryConnect(): Promise<{ transportType: TransportType }> {\r\n /**\r\n * If exact transport type is known, only try that.\r\n * Otherwise (auto mode), try streamable_http first, then sse.\r\n */\r\n const transportsToTry: TransportType[] = this.transportType\r\n ? [this.transportType]\r\n : ['streamable_http', 'sse'];\r\n\r\n let lastError: unknown;\r\n\r\n for (const currentType of transportsToTry) {\r\n const isLastAttempt = currentType === transportsToTry[transportsToTry.length - 1];\r\n\r\n try {\r\n const transport = this.getTransport(currentType);\r\n\r\n /** Update local state with the transport we are about to try */\r\n this.transport = transport;\r\n\r\n /** Race connection against timeout */\r\n await this.client!.connect(transport);\r\n\r\n /** Success! Return the type that worked */\r\n return { transportType: currentType };\r\n\r\n } catch (error: any) {\r\n lastError = error;\r\n\r\n /** Check for Auth Errors - these should fail immediately, no fallback */\r\n const isAuthError = error instanceof SDKUnauthorizedError ||\r\n (error instanceof Error && error.message.toLowerCase().includes('unauthorized'));\r\n\r\n if (isAuthError) {\r\n throw error;\r\n }\r\n\r\n /** If this was the last transport to try, throw the error */\r\n if (isLastAttempt) {\r\n throw error;\r\n }\r\n\r\n /** Otherwise, log and continue to next transport */\r\n const errorMessage = error instanceof Error ? error.message : String(error);\r\n this.emitProgress(`Connection attempt with ${currentType} failed: ${errorMessage}. Retrying...`);\r\n this._onObservabilityEvent.fire({\r\n level: 'warn',\r\n message: `Transport ${currentType} failed, falling back`,\r\n sessionId: this.sessionId,\r\n serverId: this.serverId,\r\n metadata: {\r\n failedTransport: currentType,\r\n error: errorMessage\r\n },\r\n timestamp: Date.now(),\r\n });\r\n }\r\n }\r\n\r\n throw lastError || new Error('No transports available');\r\n }\r\n\r\n /**\r\n * Connects to the MCP server\r\n * Automatically validates and refreshes OAuth tokens if needed\r\n * Saves session to Redis on first successful connection\r\n * @throws {UnauthorizedError} When OAuth authorization is required\r\n * @throws {Error} When connection fails for other reasons\r\n */\r\n async connect(): Promise<void> {\r\n await this.initialize();\r\n\r\n if (!this.client || !this.oauthProvider) {\r\n const error = 'Client or OAuth provider not initialized';\r\n this.emitError(error, 'connection');\r\n this.emitStateChange('FAILED');\r\n throw new Error(error);\r\n }\r\n\r\n try {\r\n this.emitProgress('Validating OAuth tokens...');\r\n await this.getValidTokens();\r\n\r\n this.emitStateChange('CONNECTING');\r\n\r\n /** Use the tryConnect loop to handle transport fallbacks */\r\n const { transportType } = await this.tryConnect();\r\n\r\n /** Update transport type to the one that actually worked */\r\n this.transportType = transportType;\r\n\r\n this.emitStateChange('CONNECTED');\r\n this.emitProgress('Connected successfully');\r\n\r\n // Only save/update session if transport type changed (connection negotiation)\r\n // This avoids unnecessary writes to storage on every connect\r\n const existingSession = await storage.getSession(this.identity, this.sessionId);\r\n if (!existingSession || existingSession.transportType !== this.transportType) {\r\n console.log(`[MCPClient] Saving session ${this.sessionId} (new or transport changed)`);\r\n await this.saveSession(SESSION_TTL_SECONDS);\r\n }\r\n } catch (error) {\r\n /** Handle Authentication Errors */\r\n if (\r\n error instanceof SDKUnauthorizedError ||\r\n (error instanceof Error && error.message.toLowerCase().includes('unauthorized'))\r\n ) {\r\n this.emitStateChange('AUTHENTICATING');\r\n // Save session with 10min TTL for OAuth pending state\r\n console.log(`[MCPClient] Saving session ${this.sessionId} with 10min TTL (OAuth pending)`);\r\n await this.saveSession(Math.floor(STATE_EXPIRATION_MS / 1000));\r\n\r\n /** Get OAuth authorization URL if available */\r\n let authUrl = '';\r\n if (this.oauthProvider) {\r\n authUrl = this.oauthProvider.authUrl || '';\r\n }\r\n\r\n if (this.serverId) {\r\n this._onConnectionEvent.fire({\r\n type: 'auth_required',\r\n sessionId: this.sessionId,\r\n serverId: this.serverId,\r\n authUrl,\r\n timestamp: Date.now(),\r\n });\r\n\r\n if (authUrl && this.onRedirect) {\r\n this.onRedirect(authUrl);\r\n }\r\n }\r\n\r\n throw new UnauthorizedError('OAuth authorization required');\r\n }\r\n\r\n /** Handle Generic Errors */\r\n const errorMessage = error instanceof Error ? error.message : 'Connection failed';\r\n this.emitError(errorMessage, 'connection');\r\n this.emitStateChange('FAILED');\r\n throw error;\r\n }\r\n }\r\n\r\n /**\r\n * Completes OAuth authorization flow by exchanging authorization code for tokens\r\n * Creates new authenticated client and transport, then establishes connection\r\n * Saves active session to Redis after successful authentication\r\n * @param authCode - Authorization code received from OAuth callback\r\n */\r\n\r\n // TODO: needs to be optimized\r\n async finishAuth(authCode: string): Promise<void> {\r\n this.emitStateChange('AUTHENTICATING');\r\n this.emitProgress('Exchanging authorization code for tokens...');\r\n\r\n await this.initialize();\r\n\r\n if (!this.oauthProvider) {\r\n const error = 'OAuth provider not initialized';\r\n this.emitError(error, 'auth');\r\n this.emitStateChange('FAILED');\r\n throw new Error(error);\r\n }\r\n\r\n /**\r\n * Determine which transports to try for finishing auth\r\n * If transportType is set, use only that. Otherwise try streamable_http then sse.\r\n */\r\n const transportsToTry: TransportType[] = this.transportType\r\n ? [this.transportType]\r\n : ['streamable_http', 'sse'];\r\n\r\n let lastError: unknown;\r\n let tokensExchanged = false;\r\n\r\n for (const currentType of transportsToTry) {\r\n const isLastAttempt = currentType === transportsToTry[transportsToTry.length - 1];\r\n\r\n try {\r\n const transport = this.getTransport(currentType);\r\n\r\n /** Update local state with the transport we are about to try */\r\n this.transport = transport;\r\n\r\n if (!tokensExchanged) {\r\n await transport.finishAuth(authCode);\r\n tokensExchanged = true;\r\n } else {\r\n this.emitProgress(`Tokens already exchanged, skipping auth step for ${currentType}...`);\r\n }\r\n\r\n /** Success! Update transport type */\r\n this.transportType = currentType;\r\n\r\n this.emitStateChange('AUTHENTICATED');\r\n this.emitProgress('Creating authenticated client...');\r\n\r\n this.client = new Client(\r\n {\r\n name: 'mcp-ts-oauth-client',\r\n version: '2.0',\r\n },\r\n { capabilities: {} }\r\n );\r\n\r\n this.emitStateChange('CONNECTING');\r\n\r\n /** We explicitly try to connect with the transport we just auth'd with first */\r\n await this.client.connect(this.transport);\r\n\r\n this.emitStateChange('CONNECTED');\r\n // Update session with 12hr TTL after successful OAuth\r\n console.log(`[MCPClient] Updating session ${this.sessionId} to 12hr TTL (OAuth complete)`);\r\n await this.saveSession(SESSION_TTL_SECONDS);\r\n\r\n return; // Success, exit function\r\n\r\n } catch (error) {\r\n lastError = error;\r\n\r\n const isAuthError = error instanceof SDKUnauthorizedError ||\r\n (error instanceof Error && error.message.toLowerCase().includes('unauthorized'));\r\n\r\n if (isAuthError) {\r\n throw error;\r\n }\r\n\r\n const errorMessage = error instanceof Error ? error.message : String(error);\r\n\r\n // Don't retry if the authorization code was rejected (it's one-time use)\r\n if (!tokensExchanged && errorMessage.toLowerCase().includes('invalid authorization code')) {\r\n const msg = error instanceof Error ? error.message : 'Authentication failed';\r\n this.emitError(msg, 'auth');\r\n this.emitStateChange('FAILED');\r\n throw error;\r\n }\r\n\r\n if (isLastAttempt) {\r\n const msg = error instanceof Error ? error.message : 'Authentication failed';\r\n this.emitError(msg, 'auth');\r\n this.emitStateChange('FAILED');\r\n throw error;\r\n }\r\n\r\n // Log and retry\r\n this.emitProgress(`Auth attempt with ${currentType} failed: ${errorMessage}. Retrying...`);\r\n }\r\n }\r\n\r\n if (lastError) {\r\n const errorMessage = lastError instanceof Error ? lastError.message : 'Authentication failed';\r\n this.emitError(errorMessage, 'auth');\r\n this.emitStateChange('FAILED');\r\n throw lastError;\r\n }\r\n }\r\n\r\n /**\r\n * Lists all available tools from the connected MCP server\r\n * @returns List of tools with their schemas and descriptions\r\n * @throws {Error} When client is not connected\r\n */\r\n async listTools(): Promise<ListToolsResult> {\r\n if (!this.client) {\r\n throw new Error('Not connected to server');\r\n }\r\n\r\n this.emitStateChange('DISCOVERING');\r\n\r\n try {\r\n const request: ListToolsRequest = {\r\n method: 'tools/list',\r\n params: {},\r\n };\r\n\r\n const result = await this.client.request(request, ListToolsResultSchema);\r\n\r\n if (this.serverId) {\r\n this._onConnectionEvent.fire({\r\n type: 'tools_discovered',\r\n sessionId: this.sessionId,\r\n serverId: this.serverId,\r\n toolCount: result.tools.length,\r\n tools: result.tools,\r\n timestamp: Date.now(),\r\n });\r\n }\r\n\r\n this.emitStateChange('READY');\r\n this.emitProgress(`Discovered ${result.tools.length} tools`);\r\n\r\n return result;\r\n } catch (error) {\r\n const errorMessage = error instanceof Error ? error.message : 'Failed to list tools';\r\n this.emitError(errorMessage, 'validation');\r\n this.emitStateChange('FAILED');\r\n throw error;\r\n }\r\n }\r\n\r\n /**\r\n * Executes a tool on the connected MCP server\r\n * @param toolName - Name of the tool to execute\r\n * @param toolArgs - Arguments to pass to the tool\r\n * @returns Tool execution result\r\n * @throws {Error} When client is not connected\r\n */\r\n async callTool(toolName: string, toolArgs: Record<string, unknown>): Promise<CallToolResult> {\r\n if (!this.client) {\r\n throw new Error('Not connected to server');\r\n }\r\n\r\n const request: CallToolRequest = {\r\n method: 'tools/call',\r\n params: {\r\n name: toolName,\r\n arguments: toolArgs,\r\n },\r\n };\r\n\r\n try {\r\n const result = await this.client.request(request, CallToolResultSchema);\r\n\r\n this._onObservabilityEvent.fire({\r\n type: 'mcp:client:tool_call',\r\n level: 'info',\r\n message: `Tool ${toolName} called successfully`,\r\n displayMessage: `Called tool ${toolName}`,\r\n sessionId: this.sessionId,\r\n serverId: this.serverId,\r\n payload: {\r\n toolName,\r\n args: toolArgs,\r\n },\r\n timestamp: Date.now(),\r\n id: nanoid(),\r\n });\r\n\r\n return result;\r\n } catch (error) {\r\n const errorMessage = error instanceof Error ? error.message : `Failed to call tool ${toolName}`;\r\n\r\n this._onObservabilityEvent.fire({\r\n type: 'mcp:client:error',\r\n level: 'error',\r\n message: errorMessage,\r\n displayMessage: `Failed to call tool ${toolName}`,\r\n sessionId: this.sessionId,\r\n serverId: this.serverId,\r\n payload: {\r\n errorType: 'tool_execution',\r\n error: errorMessage,\r\n toolName,\r\n args: toolArgs,\r\n },\r\n timestamp: Date.now(),\r\n id: nanoid(),\r\n });\r\n\r\n throw error;\r\n }\r\n }\r\n\r\n /**\r\n * Lists all available prompts from the connected MCP server\r\n * @returns List of available prompts\r\n * @throws {Error} When client is not connected\r\n */\r\n async listPrompts(): Promise<ListPromptsResult> {\r\n if (!this.client) {\r\n throw new Error('Not connected to server');\r\n }\r\n\r\n this.emitStateChange('DISCOVERING');\r\n\r\n try {\r\n const request: ListPromptsRequest = {\r\n method: 'prompts/list',\r\n params: {},\r\n };\r\n\r\n const result = await this.client.request(request, ListPromptsResultSchema);\r\n\r\n this.emitStateChange('READY');\r\n this.emitProgress(`Discovered ${result.prompts.length} prompts`);\r\n\r\n return result;\r\n } catch (error) {\r\n const errorMessage = error instanceof Error ? error.message : 'Failed to list prompts';\r\n this.emitError(errorMessage, 'validation');\r\n this.emitStateChange('FAILED');\r\n throw error;\r\n }\r\n }\r\n\r\n /**\r\n * Gets a specific prompt with arguments\r\n * @param name - Name of the prompt\r\n * @param args - Arguments for the prompt\r\n * @returns Prompt content\r\n * @throws {Error} When client is not connected\r\n */\r\n async getPrompt(name: string, args?: Record<string, string>): Promise<GetPromptResult> {\r\n if (!this.client) {\r\n throw new Error('Not connected to server');\r\n }\r\n\r\n const request: GetPromptRequest = {\r\n method: 'prompts/get',\r\n params: {\r\n name,\r\n arguments: args,\r\n },\r\n };\r\n\r\n return await this.client.request(request, GetPromptResultSchema);\r\n }\r\n\r\n /**\r\n * Lists all available resources from the connected MCP server\r\n * @returns List of available resources\r\n * @throws {Error} When client is not connected\r\n */\r\n async listResources(): Promise<ListResourcesResult> {\r\n if (!this.client) {\r\n throw new Error('Not connected to server');\r\n }\r\n\r\n this.emitStateChange('DISCOVERING');\r\n\r\n try {\r\n const request: ListResourcesRequest = {\r\n method: 'resources/list',\r\n params: {},\r\n };\r\n\r\n const result = await this.client.request(request, ListResourcesResultSchema);\r\n\r\n this.emitStateChange('READY');\r\n this.emitProgress(`Discovered ${result.resources.length} resources`);\r\n\r\n return result;\r\n } catch (error) {\r\n const errorMessage = error instanceof Error ? error.message : 'Failed to list resources';\r\n this.emitError(errorMessage, 'validation');\r\n this.emitStateChange('FAILED');\r\n throw error;\r\n }\r\n }\r\n\r\n /**\r\n * Reads a specific resource\r\n * @param uri - URI of the resource to read\r\n * @returns Resource content\r\n * @throws {Error} When client is not connected\r\n */\r\n async readResource(uri: string): Promise<ReadResourceResult> {\r\n if (!this.client) {\r\n throw new Error('Not connected to server');\r\n }\r\n\r\n const request: ReadResourceRequest = {\r\n method: 'resources/read',\r\n params: {\r\n uri,\r\n },\r\n };\r\n\r\n return await this.client.request(request, ReadResourceResultSchema);\r\n }\r\n\r\n /**\r\n * Refreshes the OAuth access token using the refresh token\r\n * Discovers OAuth metadata from server and exchanges refresh token for new access token\r\n * @returns True if refresh was successful, false otherwise\r\n */\r\n async refreshToken(): Promise<boolean> {\r\n await this.initialize();\r\n\r\n if (!this.oauthProvider) {\r\n return false;\r\n }\r\n\r\n const tokens = await this.oauthProvider.tokens();\r\n if (!tokens || !tokens.refresh_token) {\r\n return false;\r\n }\r\n\r\n const clientInformation = await this.oauthProvider.clientInformation();\r\n if (!clientInformation) {\r\n return false;\r\n }\r\n\r\n try {\r\n const resourceMetadata = await discoverOAuthProtectedResourceMetadata(this.serverUrl!);\r\n const authServerUrl = resourceMetadata?.authorization_servers?.[0] || this.serverUrl!;\r\n const authMetadata = await discoverAuthorizationServerMetadata(authServerUrl);\r\n\r\n const newTokens = await refreshAuthorization(authServerUrl, {\r\n metadata: authMetadata,\r\n clientInformation,\r\n refreshToken: tokens.refresh_token,\r\n });\r\n\r\n await this.oauthProvider.saveTokens(newTokens);\r\n return true;\r\n } catch (error) {\r\n console.error('[OAuth] Token refresh failed:', error);\r\n return false;\r\n }\r\n }\r\n\r\n /**\r\n * Ensures OAuth tokens are valid, refreshing them if expired\r\n * Called automatically by connect() - rarely needs to be called manually\r\n * @returns True if valid tokens are available, false otherwise\r\n */\r\n async getValidTokens(): Promise<boolean> {\r\n await this.initialize();\r\n\r\n if (!this.oauthProvider) {\r\n return false;\r\n }\r\n\r\n const tokens = await this.oauthProvider.tokens();\r\n if (!tokens) {\r\n return false;\r\n }\r\n\r\n if (this.oauthProvider.isTokenExpired()) {\r\n return await this.refreshToken();\r\n }\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Reconnects to MCP server using existing OAuth provider from Redis\r\n * Used for session restoration in serverless environments\r\n * Creates new client and transport without re-initializing OAuth provider\r\n * @throws {Error} When OAuth provider is not initialized\r\n */\r\n async reconnect(): Promise<void> {\r\n await this.initialize();\r\n\r\n if (!this.oauthProvider) {\r\n throw new Error('OAuth provider not initialized');\r\n }\r\n\r\n this.client = new Client(\r\n {\r\n name: 'mcp-ts-oauth-client',\r\n version: '2.0',\r\n },\r\n { capabilities: {} }\r\n );\r\n\r\n // Use default logic to get transport, defaulting to what's stored or auto\r\n const tt = this.transportType || 'streamable_http';\r\n this.transport = this.getTransport(tt);\r\n\r\n await this.client.connect(this.transport);\r\n }\r\n\r\n /**\r\n * Completely removes the session from Redis including all OAuth data\r\n * Invalidates credentials and disconnects the client\r\n */\r\n async clearSession(): Promise<void> {\r\n try {\r\n await this.initialize();\r\n } catch (error) {\r\n console.warn('[MCPClient] Initialization failed during clearSession:', error);\r\n }\r\n\r\n if (this.oauthProvider) {\r\n await (this.oauthProvider as any).invalidateCredentials('all');\r\n }\r\n\r\n await storage.removeSession(this.identity, this.sessionId);\r\n this.disconnect();\r\n }\r\n\r\n /**\r\n * Checks if the client is currently connected to an MCP server\r\n * @returns True if connected, false otherwise\r\n */\r\n isConnected(): boolean {\r\n return this.client !== null;\r\n }\r\n\r\n /**\r\n * Disconnects from the MCP server and cleans up resources\r\n * Does not remove session from Redis - use clearSession() for that\r\n */\r\n disconnect(reason?: string): void {\r\n if (this.client) {\r\n this.client.close();\r\n }\r\n this.client = null;\r\n this.oauthProvider = null;\r\n this.transport = null;\r\n\r\n // Emit disconnected event\r\n if (this.serverId) {\r\n this._onConnectionEvent.fire({\r\n type: 'disconnected',\r\n sessionId: this.sessionId,\r\n serverId: this.serverId,\r\n reason,\r\n timestamp: Date.now(),\r\n });\r\n\r\n this._onObservabilityEvent.fire({\r\n type: 'mcp:client:disconnect',\r\n level: 'info',\r\n message: `Disconnected from ${this.serverId}`,\r\n sessionId: this.sessionId,\r\n serverId: this.serverId,\r\n payload: {\r\n reason: reason || 'unknown',\r\n },\r\n timestamp: Date.now(),\r\n id: nanoid(),\r\n });\r\n }\r\n\r\n this.emitStateChange('DISCONNECTED');\r\n }\r\n\r\n /**\r\n * Dispose of all event emitters\r\n * Call this when the client is no longer needed\r\n */\r\n dispose(): void {\r\n this._onConnectionEvent.dispose();\r\n this._onObservabilityEvent.dispose();\r\n }\r\n\r\n /**\r\n * Gets the server URL\r\n * @returns Server URL or empty string if not set\r\n */\r\n getServerUrl(): string {\r\n return this.serverUrl || '';\r\n }\r\n\r\n /**\r\n * Gets the OAuth callback URL\r\n * @returns Callback URL or empty string if not set\r\n */\r\n getCallbackUrl(): string {\r\n return this.callbackUrl || '';\r\n }\r\n\r\n /**\r\n * Gets the transport type being used\r\n * @returns Transport type (defaults to 'streamable_http')\r\n */\r\n getTransportType(): TransportType {\r\n return this.transportType || 'streamable_http';\r\n }\r\n\r\n /**\r\n * Gets the human-readable server name\r\n * @returns Server name or undefined\r\n */\r\n getServerName(): string | undefined {\r\n return this.serverName;\r\n }\r\n\r\n /**\r\n * Gets the server ID\r\n * @returns Server ID or undefined\r\n */\r\n getServerId(): string | undefined {\r\n return this.serverId;\r\n }\r\n\r\n /**\r\n * Gets the session ID\r\n * @returns Session ID\r\n */\r\n getSessionId(): string {\r\n return this.sessionId;\r\n }\r\n\r\n /**\r\n * Gets MCP server configuration for all active user sessions\r\n * Loads sessions from Redis, validates OAuth tokens, refreshes if expired\r\n * Returns ready-to-use configuration with valid auth headers\r\n * @param identity - User ID to fetch sessions for\r\n * @returns Object keyed by sanitized server labels containing transport, url, headers, etc.\r\n * @static\r\n */\r\n static async getMcpServerConfig(identity: string): Promise<Record<string, any>> {\r\n const mcpConfig: Record<string, any> = {};\r\n const sessions = await storage.getIdentitySessionsData(identity);\r\n\r\n await Promise.all(\r\n sessions.map(async (sessionData) => {\r\n const { sessionId } = sessionData;\r\n\r\n try {\r\n // Validate session - remove if missing required fields\r\n if (\r\n !sessionData.serverId ||\r\n !sessionData.transportType ||\r\n !sessionData.serverUrl ||\r\n !sessionData.callbackUrl\r\n ) {\r\n await storage.removeSession(identity, sessionId);\r\n return;\r\n }\r\n\r\n // Get OAuth headers if session requires authentication\r\n let headers: Record<string, string> | undefined;\r\n try {\r\n // Inject existing session data to avoid redundant storage reads in initialize()\r\n const client = new MCPClient({\r\n identity,\r\n sessionId,\r\n serverId: sessionData.serverId,\r\n serverUrl: sessionData.serverUrl,\r\n callbackUrl: sessionData.callbackUrl,\r\n serverName: sessionData.serverName,\r\n transportType: sessionData.transportType,\r\n headers: sessionData.headers,\r\n });\r\n\r\n await client.initialize();\r\n\r\n const hasValidTokens = await client.getValidTokens();\r\n if (hasValidTokens && client.oauthProvider) {\r\n const tokens = await client.oauthProvider.tokens();\r\n if (tokens?.access_token) {\r\n headers = { Authorization: `Bearer ${tokens.access_token}` };\r\n }\r\n }\r\n } catch (error) {\r\n console.warn(`[MCP] Failed to get OAuth tokens for ${sessionId}:`, error);\r\n }\r\n\r\n // Build server config\r\n const label = sanitizeServerLabel(\r\n sessionData.serverName || sessionData.serverId || 'server'\r\n );\r\n\r\n mcpConfig[label] = {\r\n transport: sessionData.transportType,\r\n url: sessionData.serverUrl,\r\n ...(sessionData.serverName && {\r\n serverName: sessionData.serverName,\r\n serverLabel: label,\r\n }),\r\n ...(headers && { headers }),\r\n };\r\n } catch (error) {\r\n await storage.removeSession(identity, sessionId);\r\n console.warn(`[MCP] Failed to process session ${sessionId}:`, error);\r\n }\r\n })\r\n );\r\n\r\n return mcpConfig;\r\n }\r\n\r\n}\r\n\r\n","\r\n\r\nimport { MCPClient } from './oauth-client.js';\r\nimport { storage, type SessionData } from '../storage/index.js';\r\n\r\n/**\r\n * Manages multiple MCP connections for a single user identity.\r\n * Allows aggregating tools from all connected servers.\r\n */\r\nexport interface MultiSessionOptions {\r\n /**\r\n * Connection timeout in milliseconds\r\n * @default 15000\r\n */\r\n timeout?: number;\r\n /**\r\n * Maximum number of retry attempts\r\n * @default 2\r\n */\r\n maxRetries?: number;\r\n /**\r\n * Delay between retries in milliseconds\r\n * @default 1000\r\n */\r\n retryDelay?: number;\r\n}\r\n\r\n/**\r\n * Manages multiple MCP connections for a single user identity.\r\n * Allows aggregating tools from all connected servers.\r\n */\r\nexport class MultiSessionClient {\r\n private clients: MCPClient[] = [];\r\n private identity: string;\r\n private options: MultiSessionOptions;\r\n\r\n constructor(identity: string, options: MultiSessionOptions = {}) {\r\n this.identity = identity;\r\n this.options = {\r\n timeout: 15000,\r\n maxRetries: 2,\r\n retryDelay: 1000,\r\n ...options\r\n };\r\n }\r\n\r\n private async getActiveSessions(): Promise<SessionData[]> {\r\n const sessions = await storage.getIdentitySessionsData(this.identity);\r\n console.log(`[MultiSessionClient] All sessions for ${this.identity}:`,\r\n sessions.map(s => ({ sessionId: s.sessionId, serverId: s.serverId }))\r\n );\r\n const valid = sessions.filter(s => s.serverId && s.serverUrl && s.callbackUrl);\r\n console.log(`[MultiSessionClient] Filtered valid sessions:`, valid.length);\r\n return valid;\r\n }\r\n\r\n private async connectInBatches(sessions: SessionData[]): Promise<void> {\r\n const BATCH_SIZE = 5;\r\n for (let i = 0; i < sessions.length; i += BATCH_SIZE) {\r\n const batch = sessions.slice(i, i + BATCH_SIZE);\r\n await Promise.all(batch.map(session => this.connectSession(session)));\r\n }\r\n }\r\n\r\n private async connectSession(session: SessionData): Promise<void> {\r\n const existingClient = this.clients.find(c => c.getSessionId() === session.sessionId);\r\n if (existingClient?.isConnected()) {\r\n return;\r\n }\r\n\r\n const maxRetries = this.options.maxRetries ?? 2;\r\n const retryDelay = this.options.retryDelay ?? 1000;\r\n let lastError: unknown;\r\n\r\n for (let attempt = 0; attempt <= maxRetries; attempt++) {\r\n try {\r\n const client = await this.createAndConnectClient(session);\r\n this.clients.push(client);\r\n return;\r\n } catch (error) {\r\n lastError = error;\r\n if (attempt < maxRetries) {\r\n await new Promise(resolve => setTimeout(resolve, retryDelay));\r\n }\r\n }\r\n }\r\n\r\n console.error(`[MultiSessionClient] Failed to connect to session ${session.sessionId} after ${maxRetries + 1} attempts:`, lastError);\r\n }\r\n\r\n private async createAndConnectClient(session: SessionData): Promise<MCPClient> {\r\n const client = new MCPClient({\r\n identity: this.identity,\r\n sessionId: session.sessionId,\r\n serverId: session.serverId,\r\n serverUrl: session.serverUrl,\r\n callbackUrl: session.callbackUrl,\r\n serverName: session.serverName,\r\n transportType: session.transportType,\r\n headers: session.headers,\r\n });\r\n\r\n const timeoutMs = this.options.timeout ?? 15000;\r\n const timeoutPromise = new Promise<never>((_, reject) => {\r\n setTimeout(() => reject(new Error(`Connection timed out after ${timeoutMs}ms`)), timeoutMs);\r\n });\r\n\r\n await Promise.race([client.connect(), timeoutPromise]);\r\n return client;\r\n }\r\n\r\n async connect(): Promise<void> {\r\n const sessions = await this.getActiveSessions();\r\n await this.connectInBatches(sessions);\r\n }\r\n\r\n /**\r\n * Returns the array of currently connected clients.\r\n */\r\n getClients(): MCPClient[] {\r\n return this.clients;\r\n }\r\n\r\n /**\r\n * Disconnects all clients.\r\n */\r\n disconnect(): void {\r\n this.clients.forEach((client) => client.disconnect());\r\n this.clients = [];\r\n }\r\n}\r\n\r\n","/**\r\n * SSE (Server-Sent Events) Handler for MCP Connections\r\n * Provides real-time connection state updates to clients\r\n * Based on Cloudflare's agents pattern but adapted for HTTP/SSE\r\n */\r\n\r\nimport type { McpConnectionEvent, McpObservabilityEvent } from '../../shared/events.js';\r\nimport type {\r\n McpRpcRequest,\r\n McpRpcResponse,\r\n // RPC Param types\r\n ConnectParams,\r\n DisconnectParams,\r\n SessionParams,\r\n CallToolParams,\r\n GetPromptParams,\r\n ReadResourceParams,\r\n FinishAuthParams,\r\n // RPC Result types\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\nimport { RpcErrorCodes } from '../../shared/errors.js';\r\nimport { MCPClient } from '../mcp/oauth-client.js';\r\nimport { storage } from '../storage/index.js';\r\n\r\nexport interface ClientMetadata {\r\n clientName?: string;\r\n clientUri?: string;\r\n logoUri?: string;\r\n policyUri?: string;\r\n}\r\n\r\nexport interface SSEHandlerOptions {\r\n /**\r\n * User/Client identifier\r\n */\r\n identity: string;\r\n\r\n /**\r\n * Optional callback for authentication/authorization\r\n */\r\n onAuth?: (identity: string) => Promise<boolean>;\r\n\r\n /**\r\n * Heartbeat interval in ms (default: 30000)\r\n */\r\n heartbeatInterval?: number;\r\n\r\n /**\r\n * Static OAuth client metadata defaults (for all connections)\r\n */\r\n clientDefaults?: ClientMetadata;\r\n\r\n /**\r\n * Dynamic OAuth client metadata getter (per-request, useful for multi-tenant)\r\n * Takes precedence over clientDefaults\r\n */\r\n getClientMetadata?: (request?: any) => ClientMetadata | Promise<ClientMetadata>;\r\n}\r\n\r\n/**\r\n * SSE Connection Manager\r\n * Handles a single SSE connection and manages MCP operations\r\n */\r\nexport class SSEConnectionManager {\r\n private identity: string;\r\n private clients: Map<string, MCPClient> = new Map();\r\n private heartbeatTimer?: NodeJS.Timeout;\r\n private isActive: boolean = true;\r\n\r\n constructor(\r\n private options: SSEHandlerOptions,\r\n private sendEvent: (event: McpConnectionEvent | McpObservabilityEvent | McpRpcResponse) => void\r\n ) {\r\n this.identity = options.identity;\r\n this.startHeartbeat();\r\n }\r\n\r\n /**\r\n * Get resolved client metadata (dynamic > static > defaults)\r\n */\r\n private async getResolvedClientMetadata(request?: any): Promise<ClientMetadata> {\r\n // Priority: getClientMetadata() > clientDefaults > empty object\r\n let metadata: ClientMetadata = {};\r\n\r\n // Start with static defaults\r\n if (this.options.clientDefaults) {\r\n metadata = { ...this.options.clientDefaults };\r\n }\r\n\r\n // Override with dynamic metadata if provided\r\n if (this.options.getClientMetadata) {\r\n const dynamicMetadata = await this.options.getClientMetadata(request);\r\n metadata = { ...metadata, ...dynamicMetadata };\r\n }\r\n\r\n return metadata;\r\n }\r\n\r\n /**\r\n * Start heartbeat to keep connection alive\r\n */\r\n private startHeartbeat(): void {\r\n const interval = this.options.heartbeatInterval || 30000;\r\n this.heartbeatTimer = setInterval(() => {\r\n if (this.isActive) {\r\n this.sendEvent({\r\n level: 'debug',\r\n message: 'heartbeat',\r\n timestamp: Date.now(),\r\n } as McpObservabilityEvent);\r\n }\r\n }, interval);\r\n }\r\n\r\n /**\r\n * Handle incoming RPC requests\r\n */\r\n async handleRequest(request: McpRpcRequest): Promise<void> {\r\n try {\r\n let result: SessionListResult | ConnectResult | DisconnectResult | RestoreSessionResult | FinishAuthResult | ListToolsRpcResult | ListPromptsResult | ListResourcesResult | unknown;\r\n\r\n switch (request.method) {\r\n case 'getSessions':\r\n result = await this.getSessions();\r\n break;\r\n\r\n case 'connect':\r\n result = await this.connect(request.params as ConnectParams);\r\n break;\r\n\r\n case 'disconnect':\r\n result = await this.disconnect(request.params as DisconnectParams);\r\n break;\r\n\r\n case 'listTools':\r\n result = await this.listTools(request.params as SessionParams);\r\n break;\r\n\r\n case 'callTool':\r\n result = await this.callTool(request.params as CallToolParams);\r\n break;\r\n\r\n case 'restoreSession':\r\n result = await this.restoreSession(request.params as SessionParams);\r\n break;\r\n\r\n case 'finishAuth':\r\n result = await this.finishAuth(request.params as FinishAuthParams);\r\n break;\r\n\r\n case 'listPrompts':\r\n result = await this.listPrompts(request.params as SessionParams);\r\n break;\r\n\r\n case 'getPrompt':\r\n result = await this.getPrompt(request.params as GetPromptParams);\r\n break;\r\n\r\n case 'listResources':\r\n result = await this.listResources(request.params as SessionParams);\r\n break;\r\n\r\n case 'readResource':\r\n result = await this.readResource(request.params as ReadResourceParams);\r\n break;\r\n\r\n default:\r\n throw new Error(`Unknown method: ${request.method}`);\r\n }\r\n\r\n this.sendEvent({\r\n id: request.id,\r\n result,\r\n });\r\n } catch (error) {\r\n this.sendEvent({\r\n id: request.id,\r\n error: {\r\n code: RpcErrorCodes.EXECUTION_ERROR,\r\n message: error instanceof Error ? error.message : 'Unknown error',\r\n },\r\n });\r\n }\r\n }\r\n\r\n /**\r\n * Get all user sessions\r\n */\r\n private async getSessions(): Promise<SessionListResult> {\r\n const sessions = await storage.getIdentitySessionsData(this.identity);\r\n\r\n this.sendEvent({\r\n level: 'debug',\r\n message: `Retrieved ${sessions.length} sessions for identity ${this.identity}`,\r\n timestamp: Date.now(),\r\n metadata: {\r\n identity: this.identity,\r\n sessionCount: sessions.length,\r\n sessions: sessions.map(s => ({\r\n sessionId: s.sessionId,\r\n serverId: s.serverId,\r\n serverName: s.serverName,\r\n })),\r\n },\r\n });\r\n\r\n return {\r\n sessions: sessions.map((s) => ({\r\n sessionId: s.sessionId,\r\n serverId: s.serverId,\r\n serverName: s.serverName,\r\n serverUrl: s.serverUrl,\r\n transport: s.transportType,\r\n })),\r\n };\r\n }\r\n\r\n /**\r\n * Connect to an MCP server\r\n */\r\n private async connect(params: ConnectParams): Promise<ConnectResult> {\r\n const { serverId, serverName, serverUrl, callbackUrl, transportType } = params;\r\n\r\n // Check for existing connections\r\n const existingSessions = await storage.getIdentitySessionsData(this.identity);\r\n const duplicate = existingSessions.find(s =>\r\n s.serverId === serverId || s.serverUrl === serverUrl\r\n );\r\n\r\n if (duplicate) {\r\n throw new Error(`Connection already exists for server: ${duplicate.serverUrl || duplicate.serverId} (${duplicate.serverName})`);\r\n }\r\n\r\n // Generate session ID\r\n const sessionId = await storage.generateSessionId();\r\n\r\n // Emit connecting state\r\n this.emitConnectionEvent({\r\n type: 'state_changed',\r\n sessionId,\r\n serverId,\r\n serverName,\r\n state: 'CONNECTING',\r\n previousState: 'DISCONNECTED',\r\n timestamp: Date.now(),\r\n });\r\n\r\n try {\r\n // Get resolved client metadata\r\n const clientMetadata = await this.getResolvedClientMetadata();\r\n\r\n // Create MCP client\r\n const client = new MCPClient({\r\n identity: this.identity,\r\n sessionId,\r\n serverId,\r\n serverName,\r\n serverUrl,\r\n callbackUrl,\r\n transportType,\r\n ...clientMetadata, // Spread client metadata (clientName, clientUri, logoUri, policyUri)\r\n onRedirect: (authUrl) => {\r\n // Emit auth required event\r\n this.emitConnectionEvent({\r\n type: 'auth_required',\r\n sessionId,\r\n serverId,\r\n authUrl,\r\n timestamp: Date.now(),\r\n });\r\n },\r\n });\r\n\r\n // Note: Session will be created by MCPClient after successful connection\r\n // This ensures sessions only exist for successful or OAuth-pending connections\r\n\r\n // Store client\r\n this.clients.set(sessionId, client);\r\n\r\n // Subscribe to client events\r\n client.onConnectionEvent((event) => {\r\n this.emitConnectionEvent(event);\r\n });\r\n\r\n client.onObservabilityEvent((event) => {\r\n this.sendEvent(event);\r\n });\r\n\r\n // Attempt connection\r\n await client.connect();\r\n\r\n // Fetch tools\r\n const tools = await client.listTools();\r\n\r\n // Debug: Check session state after connection\r\n const sessionAfterConnect = await storage.getSession(this.identity, sessionId);\r\n console.log(`[SSE Handler] After connect() - Session ${sessionId}:`, {\r\n serverId: sessionAfterConnect?.serverId,\r\n });\r\n\r\n this.emitConnectionEvent({\r\n type: 'tools_discovered',\r\n sessionId,\r\n serverId,\r\n toolCount: tools.tools.length,\r\n tools: tools.tools,\r\n timestamp: Date.now(),\r\n });\r\n\r\n return {\r\n sessionId,\r\n success: true,\r\n };\r\n } catch (error) {\r\n this.emitConnectionEvent({\r\n type: 'error',\r\n sessionId,\r\n serverId,\r\n error: error instanceof Error ? error.message : 'Connection failed',\r\n errorType: 'connection',\r\n timestamp: Date.now(),\r\n });\r\n\r\n // Clean up client\r\n this.clients.delete(sessionId);\r\n\r\n throw error;\r\n }\r\n }\r\n\r\n /**\r\n * Disconnect from an MCP server\r\n */\r\n private async disconnect(params: DisconnectParams): Promise<DisconnectResult> {\r\n const { sessionId } = params;\r\n const client = this.clients.get(sessionId);\r\n\r\n if (client) {\r\n await client.clearSession();\r\n client.disconnect();\r\n this.clients.delete(sessionId);\r\n } else {\r\n // Handle orphaned sessions (e.g., OAuth flow failed before client was stored)\r\n // Directly remove from storage since there's no active client\r\n await storage.removeSession(this.identity, sessionId);\r\n }\r\n\r\n return { success: true };\r\n }\r\n\r\n /**\r\n * Helper to get or restore a client\r\n */\r\n private async getOrCreateClient(sessionId: string): Promise<MCPClient> {\r\n let client = this.clients.get(sessionId);\r\n\r\n if (!client) {\r\n client = new MCPClient({\r\n identity: this.identity,\r\n sessionId,\r\n });\r\n\r\n // Subscribe to events\r\n client.onConnectionEvent((event) => {\r\n this.emitConnectionEvent(event);\r\n });\r\n\r\n client.onObservabilityEvent((event) => {\r\n this.sendEvent(event);\r\n });\r\n\r\n await client.connect();\r\n this.clients.set(sessionId, client);\r\n }\r\n\r\n return client;\r\n }\r\n\r\n /**\r\n * List tools from a session\r\n */\r\n private async listTools(params: SessionParams): Promise<ListToolsRpcResult> {\r\n const { sessionId } = params;\r\n const client = await this.getOrCreateClient(sessionId);\r\n const result = await client.listTools();\r\n return { tools: result.tools };\r\n }\r\n\r\n /**\r\n * Call a tool\r\n */\r\n private async callTool(params: CallToolParams): Promise<unknown> {\r\n const { sessionId, toolName, toolArgs } = params;\r\n const client = await this.getOrCreateClient(sessionId);\r\n return await client.callTool(toolName, toolArgs);\r\n }\r\n\r\n /**\r\n * Refresh/validate a session\r\n */\r\n private async restoreSession(params: SessionParams): Promise<RestoreSessionResult> {\r\n const { sessionId } = params;\r\n\r\n this.sendEvent({\r\n level: 'debug',\r\n message: `Starting session refresh for ${sessionId}`,\r\n timestamp: Date.now(),\r\n metadata: { sessionId, identity: this.identity },\r\n });\r\n\r\n // Emit validating state\r\n const session = await storage.getSession(this.identity, sessionId);\r\n if (!session) {\r\n this.sendEvent({\r\n level: 'error',\r\n message: `Session not found: ${sessionId}`,\r\n timestamp: Date.now(),\r\n metadata: { sessionId, identity: this.identity },\r\n });\r\n throw new Error('Session not found');\r\n }\r\n\r\n this.sendEvent({\r\n level: 'debug',\r\n message: `Session found in Redis`,\r\n timestamp: Date.now(),\r\n metadata: {\r\n sessionId,\r\n serverId: session.serverId,\r\n serverName: session.serverName,\r\n serverUrl: session.serverUrl,\r\n transportType: session.transportType,\r\n },\r\n });\r\n\r\n this.emitConnectionEvent({\r\n type: 'state_changed',\r\n sessionId,\r\n serverId: session.serverId || 'unknown',\r\n serverName: session.serverName || 'Unknown',\r\n state: 'VALIDATING',\r\n previousState: 'DISCONNECTED',\r\n timestamp: Date.now(),\r\n });\r\n\r\n try {\r\n // Get resolved client metadata\r\n const clientMetadata = await this.getResolvedClientMetadata();\r\n\r\n // Try to restore and validate\r\n const client = new MCPClient({\r\n identity: this.identity,\r\n sessionId,\r\n ...clientMetadata, // Include metadata for consistency\r\n });\r\n\r\n // Subscribe to events\r\n client.onConnectionEvent((event) => {\r\n this.emitConnectionEvent(event);\r\n });\r\n\r\n client.onObservabilityEvent((event) => {\r\n this.sendEvent(event);\r\n });\r\n\r\n await client.connect();\r\n this.clients.set(sessionId, client);\r\n\r\n const tools = await client.listTools();\r\n\r\n\r\n\r\n this.emitConnectionEvent({\r\n type: 'tools_discovered',\r\n sessionId,\r\n serverId: session.serverId || 'unknown',\r\n toolCount: tools.tools.length,\r\n tools: tools.tools,\r\n timestamp: Date.now(),\r\n });\r\n\r\n return { success: true, toolCount: tools.tools.length };\r\n } catch (error) {\r\n this.emitConnectionEvent({\r\n type: 'error',\r\n sessionId,\r\n serverId: session.serverId || 'unknown',\r\n error: error instanceof Error ? error.message : 'Validation failed',\r\n errorType: 'validation',\r\n timestamp: Date.now(),\r\n });\r\n\r\n throw error;\r\n }\r\n }\r\n\r\n /**\r\n * Complete OAuth authorization\r\n */\r\n private async finishAuth(params: FinishAuthParams): Promise<FinishAuthResult> {\r\n const { sessionId, code } = params;\r\n\r\n this.sendEvent({\r\n level: 'debug',\r\n message: `Completing OAuth for session ${sessionId}`,\r\n timestamp: Date.now(),\r\n metadata: { sessionId, identity: this.identity },\r\n });\r\n\r\n const session = await storage.getSession(this.identity, sessionId);\r\n if (!session) {\r\n throw new Error('Session not found');\r\n }\r\n\r\n this.emitConnectionEvent({\r\n type: 'state_changed',\r\n sessionId,\r\n serverId: session.serverId || 'unknown',\r\n serverName: session.serverName || 'Unknown',\r\n state: 'AUTHENTICATING',\r\n previousState: 'DISCONNECTED',\r\n timestamp: Date.now(),\r\n });\r\n\r\n try {\r\n const client = new MCPClient({\r\n identity: this.identity,\r\n sessionId,\r\n });\r\n\r\n // Subscribe to events\r\n client.onConnectionEvent((event) => {\r\n this.emitConnectionEvent(event);\r\n });\r\n\r\n await client.finishAuth(code);\r\n this.clients.set(sessionId, client);\r\n\r\n const tools = await client.listTools();\r\n\r\n\r\n\r\n this.emitConnectionEvent({\r\n type: 'tools_discovered',\r\n sessionId,\r\n serverId: session.serverId || 'unknown',\r\n toolCount: tools.tools.length,\r\n tools: tools.tools,\r\n timestamp: Date.now(),\r\n });\r\n\r\n return { success: true, toolCount: tools.tools.length };\r\n } catch (error) {\r\n this.emitConnectionEvent({\r\n type: 'error',\r\n sessionId,\r\n serverId: session.serverId || 'unknown',\r\n error: error instanceof Error ? error.message : 'OAuth completion failed',\r\n errorType: 'auth',\r\n timestamp: Date.now(),\r\n });\r\n\r\n throw error;\r\n }\r\n }\r\n\r\n /**\r\n * List prompts from a session\r\n */\r\n private async listPrompts(params: SessionParams): Promise<ListPromptsResult> {\r\n const { sessionId } = params;\r\n const client = await this.getOrCreateClient(sessionId);\r\n const result = await client.listPrompts();\r\n return { prompts: result.prompts };\r\n }\r\n\r\n /**\r\n * Get a specific prompt\r\n */\r\n private async getPrompt(params: GetPromptParams): Promise<unknown> {\r\n const { sessionId, name, args } = params;\r\n const client = await this.getOrCreateClient(sessionId);\r\n return await client.getPrompt(name, args);\r\n }\r\n\r\n /**\r\n * List resources from a session\r\n */\r\n private async listResources(params: SessionParams): Promise<ListResourcesResult> {\r\n const { sessionId } = params;\r\n const client = await this.getOrCreateClient(sessionId);\r\n const result = await client.listResources();\r\n return { resources: result.resources };\r\n }\r\n\r\n /**\r\n * Read a specific resource\r\n */\r\n private async readResource(params: ReadResourceParams): Promise<unknown> {\r\n const { sessionId, uri } = params;\r\n const client = await this.getOrCreateClient(sessionId);\r\n return await client.readResource(uri);\r\n }\r\n\r\n /**\r\n * Emit connection event\r\n */\r\n private emitConnectionEvent(event: McpConnectionEvent): void {\r\n this.sendEvent(event);\r\n }\r\n\r\n /**\r\n * Cleanup and close all connections\r\n */\r\n dispose(): void {\r\n this.isActive = false;\r\n\r\n if (this.heartbeatTimer) {\r\n clearInterval(this.heartbeatTimer);\r\n }\r\n\r\n for (const client of this.clients.values()) {\r\n client.disconnect();\r\n }\r\n\r\n this.clients.clear();\r\n }\r\n}\r\n\r\n/**\r\n * Create SSE endpoint handler\r\n * Compatible with various Node.js frameworks\r\n */\r\nexport function createSSEHandler(options: SSEHandlerOptions) {\r\n return async (req: any, res: any) => {\r\n // Set SSE headers\r\n res.writeHead(200, {\r\n 'Content-Type': 'text/event-stream',\r\n 'Cache-Control': 'no-cache',\r\n 'Connection': 'keep-alive',\r\n 'Access-Control-Allow-Origin': '*',\r\n });\r\n\r\n // Send initial connection event\r\n sendSSE(res, 'connected', { timestamp: Date.now() });\r\n\r\n // Create connection manager\r\n const manager = new SSEConnectionManager(options, (event) => {\r\n // Determine event type\r\n if ('id' in event) {\r\n // RPC response\r\n sendSSE(res, 'rpc-response', event);\r\n } else if ('type' in event && 'sessionId' in event) {\r\n // Connection event\r\n sendSSE(res, 'connection', event);\r\n } else {\r\n // Observability event\r\n sendSSE(res, 'observability', event);\r\n }\r\n });\r\n\r\n // Handle client disconnect\r\n req.on('close', () => {\r\n manager.dispose();\r\n });\r\n\r\n // Handle incoming messages (if using POST body or other methods)\r\n if (req.method === 'POST') {\r\n let body = '';\r\n req.on('data', (chunk: Buffer) => {\r\n body += chunk.toString();\r\n });\r\n req.on('end', async () => {\r\n try {\r\n const request: McpRpcRequest = JSON.parse(body);\r\n await manager.handleRequest(request);\r\n } catch (error) {\r\n console.error('[SSE] Error handling request:', error);\r\n }\r\n });\r\n }\r\n };\r\n}\r\n\r\n/**\r\n * Send SSE event\r\n */\r\nfunction sendSSE(res: any, event: string, data: any): void {\r\n res.write(`event: ${event}\\n`);\r\n res.write(`data: ${JSON.stringify(data)}\\n\\n`);\r\n}\r\n","/**\r\n * Next.js App Router Handler for MCP SSE\r\n * Provides a clean, zero-boilerplate API for Next.js applications\r\n */\r\n\r\nimport { SSEConnectionManager, type ClientMetadata } from './sse-handler.js';\r\nimport type { McpConnectionEvent, McpObservabilityEvent } from '../../shared/events.js';\r\nimport type { McpRpcResponse } from '../../shared/types.js';\r\n\r\nexport interface NextMcpHandlerOptions {\r\n /**\r\n * Extract identity from request (default: from 'identity' query param)\r\n */\r\n getIdentity?: (request: Request) => string | null;\r\n\r\n /**\r\n * Extract auth token from request (default: from 'token' query param or Authorization header)\r\n */\r\n getAuthToken?: (request: Request) => string | null;\r\n\r\n /**\r\n * Authenticate user and verify access (optional)\r\n * Return true if user is authenticated, false otherwise\r\n */\r\n authenticate?: (identity: string, token: string | null) => Promise<boolean> | boolean;\r\n\r\n /**\r\n * Heartbeat interval in milliseconds (default: 30000)\r\n */\r\n heartbeatInterval?: number;\r\n\r\n /**\r\n * Static OAuth client metadata defaults (for all connections)\r\n * Use this for single-tenant applications with fixed branding\r\n */\r\n clientDefaults?: ClientMetadata;\r\n\r\n /**\r\n * Dynamic OAuth client metadata getter (per-request, useful for multi-tenant)\r\n * Use this when you need different branding based on request (tenant, domain, etc.)\r\n * Takes precedence over clientDefaults\r\n */\r\n getClientMetadata?: (request: Request) => ClientMetadata | Promise<ClientMetadata>;\r\n}\r\n\r\n// Global manager store - shared across requests for the same user\r\nconst managers = new Map<string, SSEConnectionManager>();\r\n\r\n/**\r\n * Creates Next.js App Router handlers (GET and POST) for MCP SSE endpoint\r\n *\r\n * @example\r\n * ```typescript\r\n * // app/api/mcp/route.ts\r\n * import { createNextMcpHandler } from '@mcp-ts/core/server';\r\n *\r\n * export const { GET, POST } = createNextMcpHandler();\r\n * ```\r\n */\r\nexport function createNextMcpHandler(options: NextMcpHandlerOptions = {}) {\r\n const {\r\n getIdentity = (request: Request) => new URL(request.url).searchParams.get('identity'),\r\n getAuthToken = (request: Request) => {\r\n const url = new URL(request.url);\r\n return url.searchParams.get('token') || request.headers.get('authorization');\r\n },\r\n authenticate = () => true,\r\n heartbeatInterval = 30000,\r\n clientDefaults,\r\n getClientMetadata,\r\n } = options;\r\n\r\n /**\r\n * GET handler - Establishes SSE connection\r\n */\r\n async function GET(request: Request): Promise<Response> {\r\n const identity = getIdentity(request);\r\n const authToken = getAuthToken(request);\r\n\r\n if (!identity) {\r\n return new Response('Missing identity', { status: 400 });\r\n }\r\n\r\n // Validate auth\r\n const isAuthorized = await authenticate(identity, authToken);\r\n if (!isAuthorized) {\r\n return new Response('Unauthorized', { status: 401 });\r\n }\r\n\r\n // Create TransformStream for SSE\r\n const stream = new TransformStream();\r\n const writer = stream.writable.getWriter();\r\n const encoder = new TextEncoder();\r\n\r\n // Helper to send SSE events\r\n const sendSSE = (event: string, data: any) => {\r\n const message = `event: ${event}\\ndata: ${JSON.stringify(data)}\\n\\n`;\r\n writer.write(encoder.encode(message)).catch(() => {\r\n // Client disconnected, ignore write errors\r\n });\r\n };\r\n\r\n // Send initial connection event\r\n sendSSE('connected', { timestamp: Date.now() });\r\n\r\n // Clean up previous manager if exists (prevents memory leaks on reconnect)\r\n const previousManager = managers.get(identity);\r\n if (previousManager) {\r\n previousManager.dispose();\r\n }\r\n\r\n // Resolve client metadata (dynamic takes precedence over static)\r\n const resolvedClientMetadata = getClientMetadata\r\n ? await getClientMetadata(request)\r\n : clientDefaults;\r\n\r\n // Create new manager\r\n const manager = new SSEConnectionManager(\r\n {\r\n identity,\r\n heartbeatInterval,\r\n clientDefaults: resolvedClientMetadata, // Pass resolved metadata\r\n },\r\n (event: McpConnectionEvent | McpObservabilityEvent | McpRpcResponse) => {\r\n // Determine event type and send via SSE\r\n if ('id' in event) {\r\n // RPC response\r\n sendSSE('rpc-response', event);\r\n } else if ('type' in event && 'sessionId' in event) {\r\n // Connection event\r\n sendSSE('connection', event);\r\n } else {\r\n // Observability event\r\n sendSSE('observability', event);\r\n }\r\n }\r\n );\r\n\r\n managers.set(identity, manager);\r\n\r\n // Handle client disconnect\r\n const abortController = new AbortController();\r\n request.signal?.addEventListener('abort', () => {\r\n manager.dispose();\r\n managers.delete(identity);\r\n writer.close().catch(() => { });\r\n abortController.abort();\r\n });\r\n\r\n // Return SSE response\r\n return new Response(stream.readable, {\r\n status: 200,\r\n headers: {\r\n 'Content-Type': 'text/event-stream',\r\n 'Cache-Control': 'no-cache, no-transform',\r\n 'Connection': 'keep-alive',\r\n 'X-Accel-Buffering': 'no',\r\n },\r\n });\r\n }\r\n\r\n /**\r\n * POST handler - Handles RPC requests\r\n */\r\n async function POST(request: Request): Promise<Response> {\r\n const identity = getIdentity(request);\r\n const authToken = getAuthToken(request);\r\n\r\n if (!identity) {\r\n return Response.json({ error: { code: 'MISSING_IDENTITY', message: 'Missing identity' } }, { status: 400 });\r\n }\r\n\r\n // Validate auth\r\n const isAuthorized = await authenticate(identity, authToken);\r\n if (!isAuthorized) {\r\n return Response.json({ error: { code: 'UNAUTHORIZED', message: 'Unauthorized' } }, { status: 401 });\r\n }\r\n\r\n try {\r\n const body = await request.json();\r\n\r\n // Get existing manager (created by GET endpoint)\r\n const manager = managers.get(identity);\r\n\r\n if (!manager) {\r\n return Response.json(\r\n {\r\n error: {\r\n code: 'NO_CONNECTION',\r\n message: 'No SSE connection found. Please establish SSE connection first.',\r\n },\r\n },\r\n { status: 400 }\r\n );\r\n }\r\n\r\n // Handle the request - response will be sent via SSE\r\n await manager.handleRequest(body);\r\n\r\n // Return acknowledgment (actual response goes through SSE)\r\n return Response.json({ acknowledged: true });\r\n } catch (error) {\r\n return Response.json(\r\n {\r\n error: {\r\n code: 'EXECUTION_ERROR',\r\n message: error instanceof Error ? error.message : 'Unknown error',\r\n },\r\n },\r\n { status: 500 }\r\n );\r\n }\r\n }\r\n\r\n return { GET, POST };\r\n}\r\n","/**\r\n * SSE Client for MCP Connections\r\n * Browser-side client that connects to SSE endpoint\r\n */\r\n\r\nimport { nanoid } from 'nanoid';\r\nimport type { McpConnectionEvent, McpObservabilityEvent } from '../../shared/events';\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';\r\n\r\nexport interface SSEClientOptions {\r\n /**\r\n * SSE endpoint URL\r\n */\r\n url: string;\r\n\r\n /**\r\n * User/Client identifier\r\n */\r\n identity: string;\r\n\r\n /**\r\n * Optional auth token\r\n */\r\n authToken?: string;\r\n\r\n /**\r\n * Connection event callback\r\n */\r\n onConnectionEvent?: (event: McpConnectionEvent) => void;\r\n\r\n /**\r\n * Observability event callback\r\n */\r\n onObservabilityEvent?: (event: McpObservabilityEvent) => void;\r\n\r\n /**\r\n * Connection status callback\r\n */\r\n onStatusChange?: (status: 'connecting' | 'connected' | 'disconnected' | 'error') => void;\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: Map<\r\n string,\r\n { resolve: (value: unknown) => void; reject: (error: Error) => void }\r\n > = new Map();\r\n private reconnectAttempts: number = 0;\r\n private maxReconnectAttempts: number = 5;\r\n private reconnectDelay: number = 1000;\r\n private isManuallyDisconnected: boolean = false;\r\n private connectionPromise: Promise<void> | null = null;\r\n private connectionResolver: (() => void) | null = null;\r\n\r\n constructor(private options: SSEClientOptions) { }\r\n\r\n /**\r\n * Connect to 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\r\n // Create connection promise\r\n this.connectionPromise = new Promise((resolve) => {\r\n this.connectionResolver = resolve;\r\n });\r\n\r\n // Build URL with query params\r\n // Handle both relative and absolute URLs\r\n const url = new URL(this.options.url, typeof window !== 'undefined' ? window.location.origin : undefined);\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\r\n // Create EventSource\r\n this.eventSource = new EventSource(url.toString());\r\n\r\n // Handle connection open\r\n this.eventSource.addEventListener('open', () => {\r\n console.log('[SSEClient] Connected');\r\n this.reconnectAttempts = 0;\r\n this.options.onStatusChange?.('connected');\r\n });\r\n\r\n // Handle 'connected' event - server confirms manager is ready\r\n this.eventSource.addEventListener('connected', (e: MessageEvent) => {\r\n const data = JSON.parse(e.data);\r\n console.log('[SSEClient] Server ready:', data);\r\n\r\n // Resolve connection promise - now safe to send requests\r\n if (this.connectionResolver) {\r\n this.connectionResolver();\r\n this.connectionResolver = null;\r\n }\r\n });\r\n\r\n // Handle 'connection' events (MCP connection state changes)\r\n this.eventSource.addEventListener('connection', (e: MessageEvent) => {\r\n const event: McpConnectionEvent = JSON.parse(e.data);\r\n this.options.onConnectionEvent?.(event);\r\n });\r\n\r\n // Handle 'observability' events (debugging/logging)\r\n this.eventSource.addEventListener('observability', (e: MessageEvent) => {\r\n const event: McpObservabilityEvent = JSON.parse(e.data);\r\n this.options.onObservabilityEvent?.(event);\r\n });\r\n\r\n // Handle 'rpc-response' events (RPC method responses)\r\n this.eventSource.addEventListener('rpc-response', (e: MessageEvent) => {\r\n const response: McpRpcResponse = JSON.parse(e.data);\r\n this.handleRpcResponse(response);\r\n });\r\n\r\n // Handle errors\r\n this.eventSource.addEventListener('error', () => {\r\n console.error('[SSEClient] Connection error');\r\n this.options.onStatusChange?.('error');\r\n\r\n // Attempt reconnection\r\n if (!this.isManuallyDisconnected && this.reconnectAttempts < this.maxReconnectAttempts) {\r\n this.reconnectAttempts++;\r\n console.log(`[SSEClient] Reconnecting (attempt ${this.reconnectAttempts})...`);\r\n\r\n setTimeout(() => {\r\n this.disconnect();\r\n this.connect();\r\n }, this.reconnectDelay * this.reconnectAttempts);\r\n }\r\n });\r\n }\r\n\r\n /**\r\n * Disconnect from 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 // Reset connection promise\r\n this.connectionPromise = null;\r\n this.connectionResolver = null;\r\n\r\n // Reject all pending requests with a specific error type\r\n for (const [id, { reject }] of this.pendingRequests.entries()) {\r\n const error = new Error('Connection closed');\r\n error.name = 'ConnectionClosedError';\r\n reject(error);\r\n }\r\n this.pendingRequests.clear();\r\n\r\n this.options.onStatusChange?.('disconnected');\r\n }\r\n\r\n /**\r\n * Send RPC request via SSE\r\n * Note: SSE is unidirectional (server->client), so we need to send requests via POST\r\n */\r\n private async sendRequest<T = unknown>(method: McpRpcMethod, params?: McpRpcParams): Promise<T> {\r\n // Wait for connection to be fully established\r\n if (this.connectionPromise) {\r\n await this.connectionPromise;\r\n }\r\n\r\n // Generate unique request ID using nanoid (e.g., \"rpc_V1StGXR8_Z5jdHi\")\r\n const id = `rpc_${nanoid(10)}`;\r\n\r\n const request: McpRpcRequest = {\r\n id,\r\n method,\r\n params,\r\n };\r\n\r\n // Create promise for response\r\n const promise = new Promise<T>((resolve, reject) => {\r\n this.pendingRequests.set(id, { resolve: resolve as (value: unknown) => void, reject });\r\n\r\n // Timeout after 30 seconds\r\n setTimeout(() => {\r\n if (this.pendingRequests.has(id)) {\r\n this.pendingRequests.delete(id);\r\n reject(new Error('Request timeout'));\r\n }\r\n }, 30000);\r\n });\r\n\r\n // Send request via POST to same endpoint\r\n try {\r\n // Handle both relative and absolute URLs\r\n const url = new URL(this.options.url, typeof window !== 'undefined' ? window.location.origin : undefined);\r\n url.searchParams.set('identity', this.options.identity);\r\n\r\n await fetch(url.toString(), {\r\n method: 'POST',\r\n headers: {\r\n 'Content-Type': 'application/json',\r\n ...(this.options.authToken && { Authorization: `Bearer ${this.options.authToken}` }),\r\n },\r\n body: JSON.stringify(request),\r\n });\r\n } catch (error) {\r\n this.pendingRequests.delete(id);\r\n throw error;\r\n }\r\n\r\n return promise;\r\n }\r\n\r\n /**\r\n * Handle RPC response\r\n */\r\n private handleRpcResponse(response: McpRpcResponse): void {\r\n const pending = this.pendingRequests.get(response.id);\r\n\r\n if (pending) {\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 /**\r\n * Get all user sessions\r\n */\r\n async getSessions(): Promise<SessionListResult> {\r\n return this.sendRequest<SessionListResult>('getSessions');\r\n }\r\n\r\n /**\r\n * Connect to an MCP server\r\n */\r\n async connectToServer(params: ConnectParams): Promise<ConnectResult> {\r\n return this.sendRequest<ConnectResult>('connect', params);\r\n }\r\n\r\n /**\r\n * Disconnect from an MCP server\r\n */\r\n async disconnectFromServer(sessionId: string): Promise<DisconnectResult> {\r\n return this.sendRequest<DisconnectResult>('disconnect', { sessionId });\r\n }\r\n\r\n /**\r\n * List tools from a session\r\n */\r\n async listTools(sessionId: string): Promise<ListToolsRpcResult> {\r\n return this.sendRequest<ListToolsRpcResult>('listTools', { sessionId });\r\n }\r\n\r\n /**\r\n * Call a tool\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 return this.sendRequest('callTool', { sessionId, toolName, toolArgs });\r\n }\r\n\r\n /**\r\n * Refresh/validate a session\r\n */\r\n async restoreSession(sessionId: string): Promise<RestoreSessionResult> {\r\n return this.sendRequest<RestoreSessionResult>('restoreSession', { sessionId });\r\n }\r\n\r\n /**\r\n * Complete OAuth authorization\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 /**\r\n * List available prompts\r\n */\r\n async listPrompts(sessionId: string): Promise<ListPromptsResult> {\r\n return this.sendRequest<ListPromptsResult>('listPrompts', { sessionId });\r\n }\r\n\r\n /**\r\n * Get a specific prompt with arguments\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 /**\r\n * List available resources\r\n */\r\n async listResources(sessionId: string): Promise<ListResourcesResult> {\r\n return this.sendRequest<ListResourcesResult>('listResources', { sessionId });\r\n }\r\n\r\n /**\r\n * Read a specific resource\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 * Check if connected\r\n */\r\n isConnected(): boolean {\r\n return this.eventSource !== null && this.eventSource.readyState === EventSource.OPEN;\r\n }\r\n}\r\n","/**\r\n * Type definitions for MCP operations\r\n */\r\n\r\nimport { Tool } from '@modelcontextprotocol/sdk/types.js';\r\n\r\n// Connect API types\r\nexport interface ConnectRequest {\r\n serverUrl: string;\r\n callbackUrl: string;\r\n}\r\n\r\nexport interface ConnectSuccessResponse {\r\n success: true;\r\n sessionId: string;\r\n}\r\n\r\nexport interface ConnectAuthRequiredResponse {\r\n requiresAuth: true;\r\n authUrl: string;\r\n sessionId: string;\r\n}\r\n\r\nexport interface ConnectErrorResponse {\r\n error: string;\r\n}\r\n\r\nexport type ConnectResponse =\r\n | ConnectSuccessResponse\r\n | ConnectAuthRequiredResponse\r\n | ConnectErrorResponse;\r\n\r\n// Callback API types\r\nexport interface CallbackSuccessResponse {\r\n success: true;\r\n message: string;\r\n}\r\n\r\nexport interface CallbackErrorResponse {\r\n error: string;\r\n}\r\n\r\nexport type CallbackResponse = CallbackSuccessResponse | CallbackErrorResponse;\r\n\r\n// Disconnect API types\r\nexport interface DisconnectRequest {\r\n sessionId: string;\r\n}\r\n\r\nexport interface DisconnectSuccessResponse {\r\n success: true;\r\n message: string;\r\n}\r\n\r\nexport interface DisconnectErrorResponse {\r\n error: string;\r\n}\r\n\r\nexport type DisconnectResponse =\r\n | DisconnectSuccessResponse\r\n | DisconnectErrorResponse;\r\n\r\n// List Tools API types\r\nexport interface ListToolsSuccessResponse {\r\n tools: Tool[];\r\n}\r\n\r\nexport interface ListToolsErrorResponse {\r\n error: string;\r\n}\r\n\r\nexport type ListToolsResponse =\r\n | ListToolsSuccessResponse\r\n | ListToolsErrorResponse;\r\n\r\n// Call Tool API types\r\nexport interface CallToolRequest {\r\n sessionId: string;\r\n toolName: string;\r\n toolArgs: Record<string, unknown>;\r\n}\r\n\r\nexport interface CallToolSuccessResponse {\r\n content: Array<{\r\n type: string;\r\n text?: string;\r\n [key: string]: unknown;\r\n }>;\r\n isError: boolean;\r\n}\r\n\r\nexport interface CallToolErrorResponse {\r\n error: string;\r\n}\r\n\r\nexport type CallToolResponse =\r\n | CallToolSuccessResponse\r\n | CallToolErrorResponse;\r\n\r\n// Helper type guards\r\nexport function isConnectSuccess(\r\n response: ConnectResponse\r\n): response is ConnectSuccessResponse {\r\n return 'success' in response && response.success === true;\r\n}\r\n\r\nexport function isConnectAuthRequired(\r\n response: ConnectResponse\r\n): response is ConnectAuthRequiredResponse {\r\n return 'requiresAuth' in response && response.requiresAuth === true;\r\n}\r\n\r\nexport function isConnectError(\r\n response: ConnectResponse\r\n): response is ConnectErrorResponse {\r\n return 'error' in response;\r\n}\r\n\r\nexport function isListToolsSuccess(\r\n response: ListToolsResponse\r\n): response is ListToolsSuccessResponse {\r\n return 'tools' in response;\r\n}\r\n\r\nexport function isCallToolSuccess(\r\n response: CallToolResponse\r\n): response is CallToolSuccessResponse {\r\n return 'content' in response;\r\n}\r\n\r\n// Generic tool info type\r\nexport type ToolInfo = {\r\n name: string;\r\n description?: string;\r\n inputSchema?: unknown;\r\n};\r\n\r\n// Transport type\r\nexport type TransportType = 'sse' | 'streamable_http';\r\n\r\n// SSE/RPC types\r\nexport type McpRpcMethod =\r\n | 'connect'\r\n | 'disconnect'\r\n | 'listTools'\r\n | 'callTool'\r\n | 'getSessions'\r\n | 'restoreSession'\r\n | 'finishAuth'\r\n | 'listPrompts'\r\n | 'getPrompt'\r\n | 'listResources'\r\n | 'readResource';\r\n\r\nexport interface McpRpcRequest {\r\n id: string;\r\n method: McpRpcMethod;\r\n params?: McpRpcParams;\r\n}\r\n\r\nexport interface McpRpcResponse<T = unknown> {\r\n id: string;\r\n result?: T;\r\n error?: {\r\n code: string;\r\n message: string;\r\n };\r\n}\r\n\r\n// RPC Parameter Types\r\nexport interface ConnectParams {\r\n serverId: string;\r\n serverName: string;\r\n serverUrl: string;\r\n callbackUrl: string;\r\n transportType?: TransportType;\r\n}\r\n\r\nexport interface DisconnectParams {\r\n sessionId: string;\r\n}\r\n\r\nexport interface SessionParams {\r\n sessionId: string;\r\n}\r\n\r\nexport interface CallToolParams {\r\n sessionId: string;\r\n toolName: string;\r\n toolArgs: Record<string, unknown>;\r\n}\r\n\r\nexport interface GetPromptParams {\r\n sessionId: string;\r\n name: string;\r\n args?: Record<string, string>;\r\n}\r\n\r\nexport interface ReadResourceParams {\r\n sessionId: string;\r\n uri: string;\r\n}\r\n\r\nexport interface FinishAuthParams {\r\n sessionId: string;\r\n code: string;\r\n}\r\n\r\nexport type McpRpcParams =\r\n | ConnectParams\r\n | DisconnectParams\r\n | SessionParams\r\n | CallToolParams\r\n | GetPromptParams\r\n | ReadResourceParams\r\n | FinishAuthParams\r\n | undefined;\r\n\r\n// RPC Result Types\r\nexport interface SessionInfo {\r\n sessionId: string;\r\n serverId?: string;\r\n serverName?: string;\r\n serverUrl: string;\r\n transport: TransportType;\r\n}\r\n\r\nexport interface SessionListResult {\r\n sessions: SessionInfo[];\r\n}\r\n\r\nexport interface ConnectResult {\r\n sessionId: string;\r\n success: boolean;\r\n}\r\n\r\nexport interface DisconnectResult {\r\n success: boolean;\r\n}\r\n\r\nexport interface RestoreSessionResult {\r\n success: boolean;\r\n toolCount: number;\r\n}\r\n\r\nexport interface FinishAuthResult {\r\n success: boolean;\r\n toolCount: number;\r\n}\r\n\r\nexport interface ListToolsRpcResult {\r\n tools: Tool[];\r\n}\r\n\r\nexport interface ListPromptsResult {\r\n prompts: Array<{\r\n name: string;\r\n description?: string;\r\n arguments?: Array<{\r\n name: string;\r\n description?: string;\r\n required?: boolean;\r\n }>;\r\n }>;\r\n}\r\n\r\nexport interface ListResourcesResult {\r\n resources: Array<{\r\n uri: string;\r\n name: string;\r\n description?: string;\r\n mimeType?: string;\r\n }>;\r\n}\r\n"]}
|