mcp-proxy 2.14.3 → 3.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/README.md +7 -29
- package/dist/bin/mcp-proxy.js +8 -18
- package/dist/bin/mcp-proxy.js.map +1 -1
- package/dist/chunk-F6SQOZNP.js +453 -0
- package/dist/chunk-F6SQOZNP.js.map +1 -0
- package/dist/index.d.ts +3 -20
- package/dist/index.js +3 -5
- package/dist/index.js.map +1 -1
- package/jsr.json +1 -1
- package/package.json +1 -1
- package/src/bin/mcp-proxy.ts +7 -17
- package/src/index.ts +1 -2
- package/src/{startHTTPStreamServer.test.ts → startHTTPServer.test.ts} +114 -3
- package/src/startHTTPServer.ts +444 -0
- package/dist/chunk-43AXMLZU.js +0 -471
- package/dist/chunk-43AXMLZU.js.map +0 -1
- package/src/startHTTPStreamServer.ts +0 -281
- package/src/startSSEServer.test.ts +0 -127
- package/src/startSSEServer.ts +0 -187
|
@@ -1,281 +0,0 @@
|
|
|
1
|
-
import { Server } from "@modelcontextprotocol/sdk/server/index.js";
|
|
2
|
-
import {
|
|
3
|
-
EventStore,
|
|
4
|
-
StreamableHTTPServerTransport,
|
|
5
|
-
} from "@modelcontextprotocol/sdk/server/streamableHttp.js";
|
|
6
|
-
import { isInitializeRequest } from "@modelcontextprotocol/sdk/types.js";
|
|
7
|
-
import http from "http";
|
|
8
|
-
import { randomUUID } from "node:crypto";
|
|
9
|
-
|
|
10
|
-
import { InMemoryEventStore } from "./InMemoryEventStore.js";
|
|
11
|
-
|
|
12
|
-
export type SSEServer = {
|
|
13
|
-
close: () => Promise<void>;
|
|
14
|
-
};
|
|
15
|
-
|
|
16
|
-
type ServerLike = {
|
|
17
|
-
close: Server["close"];
|
|
18
|
-
connect: Server["connect"];
|
|
19
|
-
};
|
|
20
|
-
|
|
21
|
-
export const startHTTPStreamServer = async <T extends ServerLike>({
|
|
22
|
-
createServer,
|
|
23
|
-
endpoint,
|
|
24
|
-
eventStore,
|
|
25
|
-
onClose,
|
|
26
|
-
onConnect,
|
|
27
|
-
onUnhandledRequest,
|
|
28
|
-
port,
|
|
29
|
-
}: {
|
|
30
|
-
createServer: (request: http.IncomingMessage) => Promise<T>;
|
|
31
|
-
endpoint: string;
|
|
32
|
-
eventStore?: EventStore;
|
|
33
|
-
onClose?: (server: T) => void;
|
|
34
|
-
onConnect?: (server: T) => void;
|
|
35
|
-
onUnhandledRequest?: (
|
|
36
|
-
req: http.IncomingMessage,
|
|
37
|
-
res: http.ServerResponse,
|
|
38
|
-
) => Promise<void>;
|
|
39
|
-
port: number;
|
|
40
|
-
}): Promise<SSEServer> => {
|
|
41
|
-
const activeTransports: Record<
|
|
42
|
-
string,
|
|
43
|
-
{
|
|
44
|
-
server: T;
|
|
45
|
-
transport: StreamableHTTPServerTransport;
|
|
46
|
-
}
|
|
47
|
-
> = {};
|
|
48
|
-
|
|
49
|
-
/**
|
|
50
|
-
* @author https://dev.classmethod.jp/articles/mcp-sse/
|
|
51
|
-
*/
|
|
52
|
-
const httpServer = http.createServer(async (req, res) => {
|
|
53
|
-
if (req.headers.origin) {
|
|
54
|
-
try {
|
|
55
|
-
const origin = new URL(req.headers.origin);
|
|
56
|
-
|
|
57
|
-
res.setHeader("Access-Control-Allow-Origin", origin.origin);
|
|
58
|
-
res.setHeader("Access-Control-Allow-Credentials", "true");
|
|
59
|
-
res.setHeader("Access-Control-Allow-Methods", "GET, POST, OPTIONS");
|
|
60
|
-
res.setHeader("Access-Control-Allow-Headers", "*");
|
|
61
|
-
} catch (error) {
|
|
62
|
-
console.error("Error parsing origin:", error);
|
|
63
|
-
}
|
|
64
|
-
}
|
|
65
|
-
|
|
66
|
-
if (req.method === "OPTIONS") {
|
|
67
|
-
res.writeHead(204);
|
|
68
|
-
res.end();
|
|
69
|
-
return;
|
|
70
|
-
}
|
|
71
|
-
|
|
72
|
-
if (req.method === "GET" && req.url === `/ping`) {
|
|
73
|
-
res.writeHead(200).end("pong");
|
|
74
|
-
return;
|
|
75
|
-
}
|
|
76
|
-
|
|
77
|
-
if (
|
|
78
|
-
req.method === "POST" &&
|
|
79
|
-
new URL(req.url!, "http://localhost").pathname === endpoint
|
|
80
|
-
) {
|
|
81
|
-
try {
|
|
82
|
-
const sessionId = Array.isArray(req.headers["mcp-session-id"])
|
|
83
|
-
? req.headers["mcp-session-id"][0]
|
|
84
|
-
: req.headers["mcp-session-id"];
|
|
85
|
-
let transport: StreamableHTTPServerTransport;
|
|
86
|
-
let server: T;
|
|
87
|
-
|
|
88
|
-
const body = await getBody(req);
|
|
89
|
-
|
|
90
|
-
if (sessionId && activeTransports[sessionId]) {
|
|
91
|
-
transport = activeTransports[sessionId].transport;
|
|
92
|
-
server = activeTransports[sessionId].server;
|
|
93
|
-
} else if (!sessionId && isInitializeRequest(body)) {
|
|
94
|
-
// Create a new transport for the session
|
|
95
|
-
transport = new StreamableHTTPServerTransport({
|
|
96
|
-
eventStore: eventStore || new InMemoryEventStore(),
|
|
97
|
-
onsessioninitialized: (_sessionId) => {
|
|
98
|
-
// add only when the id Sesison id is generated
|
|
99
|
-
activeTransports[_sessionId] = {
|
|
100
|
-
server,
|
|
101
|
-
transport,
|
|
102
|
-
};
|
|
103
|
-
},
|
|
104
|
-
sessionIdGenerator: randomUUID,
|
|
105
|
-
});
|
|
106
|
-
|
|
107
|
-
// Handle the server close event
|
|
108
|
-
transport.onclose = async () => {
|
|
109
|
-
const sid = transport.sessionId;
|
|
110
|
-
if (sid && activeTransports[sid]) {
|
|
111
|
-
onClose?.(server);
|
|
112
|
-
try {
|
|
113
|
-
await server.close();
|
|
114
|
-
} catch (error) {
|
|
115
|
-
console.error("Error closing server:", error);
|
|
116
|
-
}
|
|
117
|
-
delete activeTransports[sid];
|
|
118
|
-
}
|
|
119
|
-
};
|
|
120
|
-
|
|
121
|
-
// Create the server
|
|
122
|
-
try {
|
|
123
|
-
server = await createServer(req);
|
|
124
|
-
} catch (error) {
|
|
125
|
-
if (error instanceof Response) {
|
|
126
|
-
res.writeHead(error.status).end(error.statusText);
|
|
127
|
-
return;
|
|
128
|
-
}
|
|
129
|
-
res.writeHead(500).end("Error creating server");
|
|
130
|
-
return;
|
|
131
|
-
}
|
|
132
|
-
|
|
133
|
-
server.connect(transport);
|
|
134
|
-
onConnect?.(server);
|
|
135
|
-
|
|
136
|
-
await transport.handleRequest(req, res, body);
|
|
137
|
-
return;
|
|
138
|
-
} else {
|
|
139
|
-
// Error if the server is not created but the request is not an initialize request
|
|
140
|
-
res.setHeader("Content-Type", "application/json");
|
|
141
|
-
res.writeHead(400).end(
|
|
142
|
-
JSON.stringify({
|
|
143
|
-
error: {
|
|
144
|
-
code: -32000,
|
|
145
|
-
message: "Bad Request: No valid session ID provided",
|
|
146
|
-
},
|
|
147
|
-
id: null,
|
|
148
|
-
jsonrpc: "2.0",
|
|
149
|
-
}),
|
|
150
|
-
);
|
|
151
|
-
|
|
152
|
-
return;
|
|
153
|
-
}
|
|
154
|
-
|
|
155
|
-
// Handle ther request if the server is already created
|
|
156
|
-
await transport.handleRequest(req, res, body);
|
|
157
|
-
} catch (error) {
|
|
158
|
-
console.error("Error handling request:", error);
|
|
159
|
-
res.setHeader("Content-Type", "application/json");
|
|
160
|
-
res.writeHead(500).end(
|
|
161
|
-
JSON.stringify({
|
|
162
|
-
error: { code: -32603, message: "Internal Server Error" },
|
|
163
|
-
id: null,
|
|
164
|
-
jsonrpc: "2.0",
|
|
165
|
-
}),
|
|
166
|
-
);
|
|
167
|
-
}
|
|
168
|
-
return;
|
|
169
|
-
}
|
|
170
|
-
|
|
171
|
-
if (
|
|
172
|
-
req.method === "GET" &&
|
|
173
|
-
new URL(req.url!, "http://localhost").pathname === endpoint
|
|
174
|
-
) {
|
|
175
|
-
const sessionId = req.headers["mcp-session-id"] as string | undefined;
|
|
176
|
-
const activeTransport:
|
|
177
|
-
| {
|
|
178
|
-
server: T;
|
|
179
|
-
transport: StreamableHTTPServerTransport;
|
|
180
|
-
}
|
|
181
|
-
| undefined = sessionId ? activeTransports[sessionId] : undefined;
|
|
182
|
-
|
|
183
|
-
if (!sessionId) {
|
|
184
|
-
res.writeHead(400).end("No sessionId");
|
|
185
|
-
return;
|
|
186
|
-
}
|
|
187
|
-
|
|
188
|
-
if (!activeTransport) {
|
|
189
|
-
res.writeHead(400).end("No active transport");
|
|
190
|
-
return;
|
|
191
|
-
}
|
|
192
|
-
|
|
193
|
-
const lastEventId = req.headers["last-event-id"] as string | undefined;
|
|
194
|
-
if (lastEventId) {
|
|
195
|
-
console.log(`Client reconnecting with Last-Event-ID: ${lastEventId}`);
|
|
196
|
-
} else {
|
|
197
|
-
console.log(`Establishing new SSE stream for session ${sessionId}`);
|
|
198
|
-
}
|
|
199
|
-
|
|
200
|
-
await activeTransport.transport.handleRequest(req, res);
|
|
201
|
-
return;
|
|
202
|
-
}
|
|
203
|
-
|
|
204
|
-
if (
|
|
205
|
-
req.method === "DELETE" &&
|
|
206
|
-
new URL(req.url!, "http://localhost").pathname === endpoint
|
|
207
|
-
) {
|
|
208
|
-
console.log("received delete request");
|
|
209
|
-
const sessionId = req.headers["mcp-session-id"] as string | undefined;
|
|
210
|
-
if (!sessionId) {
|
|
211
|
-
res.writeHead(400).end("Invalid or missing sessionId");
|
|
212
|
-
return;
|
|
213
|
-
}
|
|
214
|
-
|
|
215
|
-
console.log("received delete request for session", sessionId);
|
|
216
|
-
|
|
217
|
-
const { server, transport } = activeTransports[sessionId];
|
|
218
|
-
if (!transport) {
|
|
219
|
-
res.writeHead(400).end("No active transport");
|
|
220
|
-
return;
|
|
221
|
-
}
|
|
222
|
-
|
|
223
|
-
try {
|
|
224
|
-
await transport.handleRequest(req, res);
|
|
225
|
-
onClose?.(server);
|
|
226
|
-
} catch (error) {
|
|
227
|
-
console.error("Error handling delete request:", error);
|
|
228
|
-
res.writeHead(500).end("Error handling delete request");
|
|
229
|
-
}
|
|
230
|
-
|
|
231
|
-
return;
|
|
232
|
-
}
|
|
233
|
-
|
|
234
|
-
if (onUnhandledRequest) {
|
|
235
|
-
await onUnhandledRequest(req, res);
|
|
236
|
-
} else {
|
|
237
|
-
res.writeHead(404).end();
|
|
238
|
-
}
|
|
239
|
-
});
|
|
240
|
-
|
|
241
|
-
await new Promise((resolve) => {
|
|
242
|
-
httpServer.listen(port, "::", () => {
|
|
243
|
-
resolve(undefined);
|
|
244
|
-
});
|
|
245
|
-
});
|
|
246
|
-
|
|
247
|
-
return {
|
|
248
|
-
close: async () => {
|
|
249
|
-
for (const transport of Object.values(activeTransports)) {
|
|
250
|
-
await transport.transport.close();
|
|
251
|
-
}
|
|
252
|
-
|
|
253
|
-
return new Promise((resolve, reject) => {
|
|
254
|
-
httpServer.close((error) => {
|
|
255
|
-
if (error) {
|
|
256
|
-
reject(error);
|
|
257
|
-
|
|
258
|
-
return;
|
|
259
|
-
}
|
|
260
|
-
|
|
261
|
-
resolve();
|
|
262
|
-
});
|
|
263
|
-
});
|
|
264
|
-
},
|
|
265
|
-
};
|
|
266
|
-
};
|
|
267
|
-
|
|
268
|
-
function getBody(request: http.IncomingMessage) {
|
|
269
|
-
return new Promise((resolve) => {
|
|
270
|
-
const bodyParts: Buffer[] = [];
|
|
271
|
-
let body: string;
|
|
272
|
-
request
|
|
273
|
-
.on("data", (chunk) => {
|
|
274
|
-
bodyParts.push(chunk);
|
|
275
|
-
})
|
|
276
|
-
.on("end", () => {
|
|
277
|
-
body = Buffer.concat(bodyParts).toString();
|
|
278
|
-
resolve(JSON.parse(body));
|
|
279
|
-
});
|
|
280
|
-
});
|
|
281
|
-
}
|
|
@@ -1,127 +0,0 @@
|
|
|
1
|
-
import { Client } from "@modelcontextprotocol/sdk/client/index.js";
|
|
2
|
-
import { SSEClientTransport } from "@modelcontextprotocol/sdk/client/sse.js";
|
|
3
|
-
import { StdioClientTransport } from "@modelcontextprotocol/sdk/client/stdio.js";
|
|
4
|
-
import { Server } from "@modelcontextprotocol/sdk/server/index.js";
|
|
5
|
-
import { EventSource } from "eventsource";
|
|
6
|
-
import { getRandomPort } from "get-port-please";
|
|
7
|
-
import { setTimeout as delay } from "node:timers/promises";
|
|
8
|
-
import { expect, it, vi } from "vitest";
|
|
9
|
-
|
|
10
|
-
import { proxyServer } from "./proxyServer.js";
|
|
11
|
-
import { startSSEServer } from "./startSSEServer.js";
|
|
12
|
-
|
|
13
|
-
if (!("EventSource" in global)) {
|
|
14
|
-
// @ts-expect-error - figure out how to use --experimental-eventsource with vitest
|
|
15
|
-
global.EventSource = EventSource;
|
|
16
|
-
}
|
|
17
|
-
|
|
18
|
-
it("proxies messages between SSE and stdio servers", async () => {
|
|
19
|
-
const stdioTransport = new StdioClientTransport({
|
|
20
|
-
args: ["src/simple-stdio-server.ts"],
|
|
21
|
-
command: "tsx",
|
|
22
|
-
});
|
|
23
|
-
|
|
24
|
-
const stdioClient = new Client(
|
|
25
|
-
{
|
|
26
|
-
name: "mcp-proxy",
|
|
27
|
-
version: "1.0.0",
|
|
28
|
-
},
|
|
29
|
-
{
|
|
30
|
-
capabilities: {},
|
|
31
|
-
},
|
|
32
|
-
);
|
|
33
|
-
|
|
34
|
-
await stdioClient.connect(stdioTransport);
|
|
35
|
-
|
|
36
|
-
const serverVersion = stdioClient.getServerVersion() as {
|
|
37
|
-
name: string;
|
|
38
|
-
version: string;
|
|
39
|
-
};
|
|
40
|
-
|
|
41
|
-
const serverCapabilities = stdioClient.getServerCapabilities() as {
|
|
42
|
-
capabilities: Record<string, unknown>;
|
|
43
|
-
};
|
|
44
|
-
|
|
45
|
-
const port = await getRandomPort();
|
|
46
|
-
|
|
47
|
-
const onConnect = vi.fn();
|
|
48
|
-
const onClose = vi.fn();
|
|
49
|
-
|
|
50
|
-
await startSSEServer({
|
|
51
|
-
createServer: async () => {
|
|
52
|
-
const mcpServer = new Server(serverVersion, {
|
|
53
|
-
capabilities: serverCapabilities,
|
|
54
|
-
});
|
|
55
|
-
|
|
56
|
-
await proxyServer({
|
|
57
|
-
client: stdioClient,
|
|
58
|
-
server: mcpServer,
|
|
59
|
-
serverCapabilities,
|
|
60
|
-
});
|
|
61
|
-
|
|
62
|
-
return mcpServer;
|
|
63
|
-
},
|
|
64
|
-
endpoint: "/sse",
|
|
65
|
-
onClose,
|
|
66
|
-
onConnect,
|
|
67
|
-
port,
|
|
68
|
-
});
|
|
69
|
-
|
|
70
|
-
const sseClient = new Client(
|
|
71
|
-
{
|
|
72
|
-
name: "sse-client",
|
|
73
|
-
version: "1.0.0",
|
|
74
|
-
},
|
|
75
|
-
{
|
|
76
|
-
capabilities: {},
|
|
77
|
-
},
|
|
78
|
-
);
|
|
79
|
-
|
|
80
|
-
const transport = new SSEClientTransport(
|
|
81
|
-
new URL(`http://localhost:${port}/sse`),
|
|
82
|
-
);
|
|
83
|
-
|
|
84
|
-
await sseClient.connect(transport);
|
|
85
|
-
|
|
86
|
-
const result = await sseClient.listResources();
|
|
87
|
-
expect(result).toEqual({
|
|
88
|
-
resources: [
|
|
89
|
-
{
|
|
90
|
-
name: "Example Resource",
|
|
91
|
-
uri: "file:///example.txt",
|
|
92
|
-
},
|
|
93
|
-
],
|
|
94
|
-
});
|
|
95
|
-
|
|
96
|
-
expect(
|
|
97
|
-
await sseClient.readResource({ uri: result.resources[0].uri }, {}),
|
|
98
|
-
).toEqual({
|
|
99
|
-
contents: [
|
|
100
|
-
{
|
|
101
|
-
mimeType: "text/plain",
|
|
102
|
-
text: "This is the content of the example resource.",
|
|
103
|
-
uri: "file:///example.txt",
|
|
104
|
-
},
|
|
105
|
-
],
|
|
106
|
-
});
|
|
107
|
-
expect(await sseClient.subscribeResource({ uri: "xyz" })).toEqual({});
|
|
108
|
-
expect(await sseClient.unsubscribeResource({ uri: "xyz" })).toEqual({});
|
|
109
|
-
expect(await sseClient.listResourceTemplates()).toEqual({
|
|
110
|
-
resourceTemplates: [
|
|
111
|
-
{
|
|
112
|
-
description: "Specify the filename to retrieve",
|
|
113
|
-
name: "Example resource template",
|
|
114
|
-
uriTemplate: `file://{filename}`,
|
|
115
|
-
},
|
|
116
|
-
],
|
|
117
|
-
});
|
|
118
|
-
|
|
119
|
-
expect(onConnect).toHaveBeenCalled();
|
|
120
|
-
expect(onClose).not.toHaveBeenCalled();
|
|
121
|
-
|
|
122
|
-
await sseClient.close();
|
|
123
|
-
|
|
124
|
-
await delay(100);
|
|
125
|
-
|
|
126
|
-
expect(onClose).toHaveBeenCalled();
|
|
127
|
-
});
|
package/src/startSSEServer.ts
DELETED
|
@@ -1,187 +0,0 @@
|
|
|
1
|
-
import { Server } from "@modelcontextprotocol/sdk/server/index.js";
|
|
2
|
-
import { SSEServerTransport } from "@modelcontextprotocol/sdk/server/sse.js";
|
|
3
|
-
import http from "http";
|
|
4
|
-
|
|
5
|
-
export type SSEServer = {
|
|
6
|
-
close: () => Promise<void>;
|
|
7
|
-
};
|
|
8
|
-
|
|
9
|
-
type ServerLike = {
|
|
10
|
-
close: Server["close"];
|
|
11
|
-
connect: Server["connect"];
|
|
12
|
-
};
|
|
13
|
-
|
|
14
|
-
export const startSSEServer = async <T extends ServerLike>({
|
|
15
|
-
createServer,
|
|
16
|
-
endpoint,
|
|
17
|
-
onClose,
|
|
18
|
-
onConnect,
|
|
19
|
-
onUnhandledRequest,
|
|
20
|
-
port,
|
|
21
|
-
}: {
|
|
22
|
-
createServer: (request: http.IncomingMessage) => Promise<T>;
|
|
23
|
-
endpoint: string;
|
|
24
|
-
onClose?: (server: T) => void;
|
|
25
|
-
onConnect?: (server: T) => void;
|
|
26
|
-
onUnhandledRequest?: (
|
|
27
|
-
req: http.IncomingMessage,
|
|
28
|
-
res: http.ServerResponse,
|
|
29
|
-
) => Promise<void>;
|
|
30
|
-
port: number;
|
|
31
|
-
}): Promise<SSEServer> => {
|
|
32
|
-
const activeTransports: Record<string, SSEServerTransport> = {};
|
|
33
|
-
|
|
34
|
-
/**
|
|
35
|
-
* @author https://dev.classmethod.jp/articles/mcp-sse/
|
|
36
|
-
*/
|
|
37
|
-
const httpServer = http.createServer(async (req, res) => {
|
|
38
|
-
if (req.headers.origin) {
|
|
39
|
-
try {
|
|
40
|
-
const origin = new URL(req.headers.origin);
|
|
41
|
-
|
|
42
|
-
res.setHeader("Access-Control-Allow-Origin", origin.origin);
|
|
43
|
-
res.setHeader("Access-Control-Allow-Credentials", "true");
|
|
44
|
-
res.setHeader("Access-Control-Allow-Methods", "GET, POST, OPTIONS");
|
|
45
|
-
res.setHeader("Access-Control-Allow-Headers", "*");
|
|
46
|
-
} catch (error) {
|
|
47
|
-
console.error("Error parsing origin:", error);
|
|
48
|
-
}
|
|
49
|
-
}
|
|
50
|
-
|
|
51
|
-
if (req.method === "OPTIONS") {
|
|
52
|
-
res.writeHead(204);
|
|
53
|
-
res.end();
|
|
54
|
-
return;
|
|
55
|
-
}
|
|
56
|
-
|
|
57
|
-
if (req.method === "GET" && req.url === "/health") {
|
|
58
|
-
res.writeHead(200, { "Content-Type": "text/plain" }).end("OK");
|
|
59
|
-
return;
|
|
60
|
-
}
|
|
61
|
-
|
|
62
|
-
if (req.method === "GET" && req.url === `/ping`) {
|
|
63
|
-
res.writeHead(200).end("pong");
|
|
64
|
-
|
|
65
|
-
return;
|
|
66
|
-
}
|
|
67
|
-
|
|
68
|
-
if (
|
|
69
|
-
req.method === "GET" &&
|
|
70
|
-
new URL(req.url!, "http://localhost").pathname === endpoint
|
|
71
|
-
) {
|
|
72
|
-
const transport = new SSEServerTransport("/messages", res);
|
|
73
|
-
|
|
74
|
-
let server: T;
|
|
75
|
-
|
|
76
|
-
try {
|
|
77
|
-
server = await createServer(req);
|
|
78
|
-
} catch (error) {
|
|
79
|
-
if (error instanceof Response) {
|
|
80
|
-
res.writeHead(error.status).end(error.statusText);
|
|
81
|
-
|
|
82
|
-
return;
|
|
83
|
-
}
|
|
84
|
-
|
|
85
|
-
res.writeHead(500).end("Error creating server");
|
|
86
|
-
|
|
87
|
-
return;
|
|
88
|
-
}
|
|
89
|
-
|
|
90
|
-
activeTransports[transport.sessionId] = transport;
|
|
91
|
-
|
|
92
|
-
let closed = false;
|
|
93
|
-
|
|
94
|
-
res.on("close", async () => {
|
|
95
|
-
closed = true;
|
|
96
|
-
|
|
97
|
-
try {
|
|
98
|
-
await server.close();
|
|
99
|
-
} catch (error) {
|
|
100
|
-
console.error("Error closing server:", error);
|
|
101
|
-
}
|
|
102
|
-
|
|
103
|
-
delete activeTransports[transport.sessionId];
|
|
104
|
-
|
|
105
|
-
onClose?.(server);
|
|
106
|
-
});
|
|
107
|
-
|
|
108
|
-
try {
|
|
109
|
-
await server.connect(transport);
|
|
110
|
-
|
|
111
|
-
await transport.send({
|
|
112
|
-
jsonrpc: "2.0",
|
|
113
|
-
method: "sse/connection",
|
|
114
|
-
params: { message: "SSE Connection established" },
|
|
115
|
-
});
|
|
116
|
-
|
|
117
|
-
onConnect?.(server);
|
|
118
|
-
} catch (error) {
|
|
119
|
-
if (!closed) {
|
|
120
|
-
console.error("Error connecting to server:", error);
|
|
121
|
-
|
|
122
|
-
res.writeHead(500).end("Error connecting to server");
|
|
123
|
-
}
|
|
124
|
-
}
|
|
125
|
-
|
|
126
|
-
return;
|
|
127
|
-
}
|
|
128
|
-
|
|
129
|
-
if (req.method === "POST" && req.url?.startsWith("/messages")) {
|
|
130
|
-
const sessionId = new URL(
|
|
131
|
-
req.url,
|
|
132
|
-
"https://example.com",
|
|
133
|
-
).searchParams.get("sessionId");
|
|
134
|
-
|
|
135
|
-
if (!sessionId) {
|
|
136
|
-
res.writeHead(400).end("No sessionId");
|
|
137
|
-
|
|
138
|
-
return;
|
|
139
|
-
}
|
|
140
|
-
|
|
141
|
-
const activeTransport: SSEServerTransport | undefined =
|
|
142
|
-
activeTransports[sessionId];
|
|
143
|
-
|
|
144
|
-
if (!activeTransport) {
|
|
145
|
-
res.writeHead(400).end("No active transport");
|
|
146
|
-
|
|
147
|
-
return;
|
|
148
|
-
}
|
|
149
|
-
|
|
150
|
-
await activeTransport.handlePostMessage(req, res);
|
|
151
|
-
|
|
152
|
-
return;
|
|
153
|
-
}
|
|
154
|
-
|
|
155
|
-
if (onUnhandledRequest) {
|
|
156
|
-
await onUnhandledRequest(req, res);
|
|
157
|
-
} else {
|
|
158
|
-
res.writeHead(404).end();
|
|
159
|
-
}
|
|
160
|
-
});
|
|
161
|
-
|
|
162
|
-
await new Promise((resolve) => {
|
|
163
|
-
httpServer.listen(port, "::", () => {
|
|
164
|
-
resolve(undefined);
|
|
165
|
-
});
|
|
166
|
-
});
|
|
167
|
-
|
|
168
|
-
return {
|
|
169
|
-
close: async () => {
|
|
170
|
-
for (const transport of Object.values(activeTransports)) {
|
|
171
|
-
await transport.close();
|
|
172
|
-
}
|
|
173
|
-
|
|
174
|
-
return new Promise((resolve, reject) => {
|
|
175
|
-
httpServer.close((error) => {
|
|
176
|
-
if (error) {
|
|
177
|
-
reject(error);
|
|
178
|
-
|
|
179
|
-
return;
|
|
180
|
-
}
|
|
181
|
-
|
|
182
|
-
resolve();
|
|
183
|
-
});
|
|
184
|
-
});
|
|
185
|
-
},
|
|
186
|
-
};
|
|
187
|
-
};
|