mcpWebSocketTransport.ts
utils/mcpWebSocketTransport.ts
201
Lines
6052
Bytes
1
Exports
6
Imports
10
Keywords
What this is
This page documents one file from the repository and includes its full source so you can read it without leaving the docs site.
Beginner explanation
This file is one piece of the larger system. Its name, directory, imports, and exports show where it fits. Start by reading the exports and related files first.
How it is used
Start from the exports list and related files. Those are the easiest clues for where this file fits into the system.
Expert explanation
Architecturally, this file intersects with mcp. It contains 201 lines, 6 detected imports, and 1 detected exports.
Important relationships
Detected exports
WebSocketTransport
Keywords
websocketmessagevoidprivatecloseeventunknowndatareadystateremoveeventlistener
Detected imports
@modelcontextprotocol/sdk/shared/transport.js@modelcontextprotocol/sdk/types.jsws./diagLogs.js./errors.js./slowOperations.js
Source notes
This page embeds the full file contents. Small or leaf files are still indexed honestly instead of being over-explained.
Full source
import type { Transport } from '@modelcontextprotocol/sdk/shared/transport.js'
import {
type JSONRPCMessage,
JSONRPCMessageSchema,
} from '@modelcontextprotocol/sdk/types.js'
import type WsWebSocket from 'ws'
import { logForDiagnosticsNoPII } from './diagLogs.js'
import { toError } from './errors.js'
import { jsonParse, jsonStringify } from './slowOperations.js'
// WebSocket readyState constants (same for both native and ws)
const WS_CONNECTING = 0
const WS_OPEN = 1
// Minimal interface shared by globalThis.WebSocket and ws.WebSocket
type WebSocketLike = {
readonly readyState: number
close(): void
send(data: string): void
}
export class WebSocketTransport implements Transport {
private started = false
private opened: Promise<void>
private isBun = typeof Bun !== 'undefined'
constructor(private ws: WebSocketLike) {
this.opened = new Promise((resolve, reject) => {
if (this.ws.readyState === WS_OPEN) {
resolve()
} else if (this.isBun) {
const nws = this.ws as unknown as globalThis.WebSocket
const onOpen = () => {
nws.removeEventListener('open', onOpen)
nws.removeEventListener('error', onError)
resolve()
}
const onError = (event: Event) => {
nws.removeEventListener('open', onOpen)
nws.removeEventListener('error', onError)
logForDiagnosticsNoPII('error', 'mcp_websocket_connect_fail')
reject(event)
}
nws.addEventListener('open', onOpen)
nws.addEventListener('error', onError)
} else {
const nws = this.ws as unknown as WsWebSocket
nws.on('open', () => {
resolve()
})
nws.on('error', error => {
logForDiagnosticsNoPII('error', 'mcp_websocket_connect_fail')
reject(error)
})
}
})
// Attach persistent event handlers
if (this.isBun) {
const nws = this.ws as unknown as globalThis.WebSocket
nws.addEventListener('message', this.onBunMessage)
nws.addEventListener('error', this.onBunError)
nws.addEventListener('close', this.onBunClose)
} else {
const nws = this.ws as unknown as WsWebSocket
nws.on('message', this.onNodeMessage)
nws.on('error', this.onNodeError)
nws.on('close', this.onNodeClose)
}
}
onclose?: () => void
onerror?: (error: Error) => void
onmessage?: (message: JSONRPCMessage) => void
// Bun (native WebSocket) event handlers
private onBunMessage = (event: MessageEvent) => {
try {
const data =
typeof event.data === 'string' ? event.data : String(event.data)
const messageObj = jsonParse(data)
const message = JSONRPCMessageSchema.parse(messageObj)
this.onmessage?.(message)
} catch (error) {
this.handleError(error)
}
}
private onBunError = () => {
this.handleError(new Error('WebSocket error'))
}
private onBunClose = () => {
this.handleCloseCleanup()
}
// Node (ws package) event handlers
private onNodeMessage = (data: Buffer) => {
try {
const messageObj = jsonParse(data.toString('utf-8'))
const message = JSONRPCMessageSchema.parse(messageObj)
this.onmessage?.(message)
} catch (error) {
this.handleError(error)
}
}
private onNodeError = (error: unknown) => {
this.handleError(error)
}
private onNodeClose = () => {
this.handleCloseCleanup()
}
// Shared error handler
private handleError(error: unknown): void {
logForDiagnosticsNoPII('error', 'mcp_websocket_message_fail')
this.onerror?.(toError(error))
}
// Shared close handler with listener cleanup
private handleCloseCleanup(): void {
this.onclose?.()
// Clean up listeners after close
if (this.isBun) {
const nws = this.ws as unknown as globalThis.WebSocket
nws.removeEventListener('message', this.onBunMessage)
nws.removeEventListener('error', this.onBunError)
nws.removeEventListener('close', this.onBunClose)
} else {
const nws = this.ws as unknown as WsWebSocket
nws.off('message', this.onNodeMessage)
nws.off('error', this.onNodeError)
nws.off('close', this.onNodeClose)
}
}
/**
* Starts listening for messages on the WebSocket.
*/
async start(): Promise<void> {
if (this.started) {
throw new Error('Start can only be called once per transport.')
}
await this.opened
if (this.ws.readyState !== WS_OPEN) {
logForDiagnosticsNoPII('error', 'mcp_websocket_start_not_opened')
throw new Error('WebSocket is not open. Cannot start transport.')
}
this.started = true
// Unlike stdio, WebSocket connections are typically already established when the transport is created.
// No explicit connection action needed here, just attaching listeners.
}
/**
* Closes the WebSocket connection.
*/
async close(): Promise<void> {
if (
this.ws.readyState === WS_OPEN ||
this.ws.readyState === WS_CONNECTING
) {
this.ws.close()
}
// Ensure listeners are removed even if close was called externally or connection was already closed
this.handleCloseCleanup()
}
/**
* Sends a JSON-RPC message over the WebSocket connection.
*/
async send(message: JSONRPCMessage): Promise<void> {
if (this.ws.readyState !== WS_OPEN) {
logForDiagnosticsNoPII('error', 'mcp_websocket_send_not_opened')
throw new Error('WebSocket is not open. Cannot send message.')
}
const json = jsonStringify(message)
try {
if (this.isBun) {
// Native WebSocket.send() is synchronous (no callback)
this.ws.send(json)
} else {
await new Promise<void>((resolve, reject) => {
;(this.ws as unknown as WsWebSocket).send(json, error => {
if (error) {
reject(error)
} else {
resolve()
}
})
})
}
} catch (error) {
this.handleError(error)
throw error
}
}
}