mcp-proxy 2.14.3 → 3.0.1
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 +9 -31
- package/dist/bin/mcp-proxy.js +8 -18
- package/dist/bin/mcp-proxy.js.map +1 -1
- package/dist/chunk-EX66KMYB.js +453 -0
- package/dist/chunk-EX66KMYB.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
|
@@ -0,0 +1,444 @@
|
|
|
1
|
+
import { Server } from "@modelcontextprotocol/sdk/server/index.js";
|
|
2
|
+
import { SSEServerTransport } from "@modelcontextprotocol/sdk/server/sse.js";
|
|
3
|
+
import {
|
|
4
|
+
EventStore,
|
|
5
|
+
StreamableHTTPServerTransport,
|
|
6
|
+
} from "@modelcontextprotocol/sdk/server/streamableHttp.js";
|
|
7
|
+
import { isInitializeRequest } from "@modelcontextprotocol/sdk/types.js";
|
|
8
|
+
import http from "http";
|
|
9
|
+
import { randomUUID } from "node:crypto";
|
|
10
|
+
|
|
11
|
+
import { InMemoryEventStore } from "./InMemoryEventStore.js";
|
|
12
|
+
|
|
13
|
+
export type SSEServer = {
|
|
14
|
+
close: () => Promise<void>;
|
|
15
|
+
};
|
|
16
|
+
|
|
17
|
+
type ServerLike = {
|
|
18
|
+
close: Server["close"];
|
|
19
|
+
connect: Server["connect"];
|
|
20
|
+
};
|
|
21
|
+
|
|
22
|
+
const getBody = (request: http.IncomingMessage) => {
|
|
23
|
+
return new Promise((resolve) => {
|
|
24
|
+
const bodyParts: Buffer[] = [];
|
|
25
|
+
let body: string;
|
|
26
|
+
request
|
|
27
|
+
.on("data", (chunk) => {
|
|
28
|
+
bodyParts.push(chunk);
|
|
29
|
+
})
|
|
30
|
+
.on("end", () => {
|
|
31
|
+
body = Buffer.concat(bodyParts).toString();
|
|
32
|
+
resolve(JSON.parse(body));
|
|
33
|
+
});
|
|
34
|
+
});
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
const handleStreamRequest = async <T extends ServerLike>({
|
|
38
|
+
activeTransports,
|
|
39
|
+
createServer,
|
|
40
|
+
endpoint,
|
|
41
|
+
eventStore,
|
|
42
|
+
onClose,
|
|
43
|
+
onConnect,
|
|
44
|
+
req,
|
|
45
|
+
res,
|
|
46
|
+
}: {
|
|
47
|
+
activeTransports: Record<
|
|
48
|
+
string,
|
|
49
|
+
{ server: T; transport: StreamableHTTPServerTransport }
|
|
50
|
+
>;
|
|
51
|
+
createServer: (request: http.IncomingMessage) => Promise<T>;
|
|
52
|
+
endpoint: string;
|
|
53
|
+
eventStore?: EventStore;
|
|
54
|
+
onClose?: (server: T) => void;
|
|
55
|
+
onConnect?: (server: T) => void;
|
|
56
|
+
req: http.IncomingMessage;
|
|
57
|
+
res: http.ServerResponse;
|
|
58
|
+
}) => {
|
|
59
|
+
if (
|
|
60
|
+
req.method === "POST" &&
|
|
61
|
+
new URL(req.url!, "http://localhost").pathname === endpoint
|
|
62
|
+
) {
|
|
63
|
+
try {
|
|
64
|
+
const sessionId = Array.isArray(req.headers["mcp-session-id"])
|
|
65
|
+
? req.headers["mcp-session-id"][0]
|
|
66
|
+
: req.headers["mcp-session-id"];
|
|
67
|
+
let transport: StreamableHTTPServerTransport;
|
|
68
|
+
let server: T;
|
|
69
|
+
|
|
70
|
+
const body = await getBody(req);
|
|
71
|
+
|
|
72
|
+
if (sessionId && activeTransports[sessionId]) {
|
|
73
|
+
transport = activeTransports[sessionId].transport;
|
|
74
|
+
server = activeTransports[sessionId].server;
|
|
75
|
+
} else if (!sessionId && isInitializeRequest(body)) {
|
|
76
|
+
// Create a new transport for the session
|
|
77
|
+
transport = new StreamableHTTPServerTransport({
|
|
78
|
+
eventStore: eventStore || new InMemoryEventStore(),
|
|
79
|
+
onsessioninitialized: (_sessionId) => {
|
|
80
|
+
// add only when the id Sesison id is generated
|
|
81
|
+
activeTransports[_sessionId] = {
|
|
82
|
+
server,
|
|
83
|
+
transport,
|
|
84
|
+
};
|
|
85
|
+
},
|
|
86
|
+
sessionIdGenerator: randomUUID,
|
|
87
|
+
});
|
|
88
|
+
|
|
89
|
+
// Handle the server close event
|
|
90
|
+
transport.onclose = async () => {
|
|
91
|
+
const sid = transport.sessionId;
|
|
92
|
+
if (sid && activeTransports[sid]) {
|
|
93
|
+
onClose?.(server);
|
|
94
|
+
try {
|
|
95
|
+
await server.close();
|
|
96
|
+
} catch (error) {
|
|
97
|
+
console.error("Error closing server:", error);
|
|
98
|
+
}
|
|
99
|
+
delete activeTransports[sid];
|
|
100
|
+
}
|
|
101
|
+
};
|
|
102
|
+
|
|
103
|
+
// Create the server
|
|
104
|
+
try {
|
|
105
|
+
server = await createServer(req);
|
|
106
|
+
} catch (error) {
|
|
107
|
+
if (error instanceof Response) {
|
|
108
|
+
res.writeHead(error.status).end(error.statusText);
|
|
109
|
+
return true;
|
|
110
|
+
}
|
|
111
|
+
res.writeHead(500).end("Error creating server");
|
|
112
|
+
return true;
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
server.connect(transport);
|
|
116
|
+
onConnect?.(server);
|
|
117
|
+
|
|
118
|
+
await transport.handleRequest(req, res, body);
|
|
119
|
+
return true;
|
|
120
|
+
} else {
|
|
121
|
+
// Error if the server is not created but the request is not an initialize request
|
|
122
|
+
res.setHeader("Content-Type", "application/json");
|
|
123
|
+
|
|
124
|
+
res.writeHead(400).end(
|
|
125
|
+
JSON.stringify({
|
|
126
|
+
error: {
|
|
127
|
+
code: -32000,
|
|
128
|
+
message: "Bad Request: No valid session ID provided",
|
|
129
|
+
},
|
|
130
|
+
id: null,
|
|
131
|
+
jsonrpc: "2.0",
|
|
132
|
+
}),
|
|
133
|
+
);
|
|
134
|
+
|
|
135
|
+
return true;
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
// Handle ther request if the server is already created
|
|
139
|
+
await transport.handleRequest(req, res, body);
|
|
140
|
+
} catch (error) {
|
|
141
|
+
console.error("Error handling request:", error);
|
|
142
|
+
res.setHeader("Content-Type", "application/json");
|
|
143
|
+
res.writeHead(500).end(
|
|
144
|
+
JSON.stringify({
|
|
145
|
+
error: { code: -32603, message: "Internal Server Error" },
|
|
146
|
+
id: null,
|
|
147
|
+
jsonrpc: "2.0",
|
|
148
|
+
}),
|
|
149
|
+
);
|
|
150
|
+
}
|
|
151
|
+
return true;
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
if (
|
|
155
|
+
req.method === "GET" &&
|
|
156
|
+
new URL(req.url!, "http://localhost").pathname === endpoint
|
|
157
|
+
) {
|
|
158
|
+
const sessionId = req.headers["mcp-session-id"] as string | undefined;
|
|
159
|
+
const activeTransport:
|
|
160
|
+
| {
|
|
161
|
+
server: T;
|
|
162
|
+
transport: StreamableHTTPServerTransport;
|
|
163
|
+
}
|
|
164
|
+
| undefined = sessionId ? activeTransports[sessionId] : undefined;
|
|
165
|
+
|
|
166
|
+
if (!sessionId) {
|
|
167
|
+
res.writeHead(400).end("No sessionId");
|
|
168
|
+
return true;
|
|
169
|
+
}
|
|
170
|
+
|
|
171
|
+
if (!activeTransport) {
|
|
172
|
+
res.writeHead(400).end("No active transport");
|
|
173
|
+
return true;
|
|
174
|
+
}
|
|
175
|
+
|
|
176
|
+
const lastEventId = req.headers["last-event-id"] as string | undefined;
|
|
177
|
+
if (lastEventId) {
|
|
178
|
+
console.log(`Client reconnecting with Last-Event-ID: ${lastEventId}`);
|
|
179
|
+
} else {
|
|
180
|
+
console.log(`Establishing new SSE stream for session ${sessionId}`);
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
await activeTransport.transport.handleRequest(req, res);
|
|
184
|
+
return true;
|
|
185
|
+
}
|
|
186
|
+
|
|
187
|
+
if (
|
|
188
|
+
req.method === "DELETE" &&
|
|
189
|
+
new URL(req.url!, "http://localhost").pathname === endpoint
|
|
190
|
+
) {
|
|
191
|
+
console.log("received delete request");
|
|
192
|
+
const sessionId = req.headers["mcp-session-id"] as string | undefined;
|
|
193
|
+
if (!sessionId) {
|
|
194
|
+
res.writeHead(400).end("Invalid or missing sessionId");
|
|
195
|
+
return true;
|
|
196
|
+
}
|
|
197
|
+
|
|
198
|
+
console.log("received delete request for session", sessionId);
|
|
199
|
+
|
|
200
|
+
const { server, transport } = activeTransports[sessionId];
|
|
201
|
+
if (!transport) {
|
|
202
|
+
res.writeHead(400).end("No active transport");
|
|
203
|
+
return true;
|
|
204
|
+
}
|
|
205
|
+
|
|
206
|
+
try {
|
|
207
|
+
await transport.handleRequest(req, res);
|
|
208
|
+
onClose?.(server);
|
|
209
|
+
} catch (error) {
|
|
210
|
+
console.error("Error handling delete request:", error);
|
|
211
|
+
res.writeHead(500).end("Error handling delete request");
|
|
212
|
+
}
|
|
213
|
+
|
|
214
|
+
return true;
|
|
215
|
+
}
|
|
216
|
+
|
|
217
|
+
return false;
|
|
218
|
+
};
|
|
219
|
+
|
|
220
|
+
const handleSSERequest = async <T extends ServerLike>({
|
|
221
|
+
activeTransports,
|
|
222
|
+
createServer,
|
|
223
|
+
endpoint,
|
|
224
|
+
onClose,
|
|
225
|
+
onConnect,
|
|
226
|
+
req,
|
|
227
|
+
res,
|
|
228
|
+
}: {
|
|
229
|
+
activeTransports: Record<string, SSEServerTransport>;
|
|
230
|
+
createServer: (request: http.IncomingMessage) => Promise<T>;
|
|
231
|
+
endpoint: string;
|
|
232
|
+
onClose?: (server: T) => void;
|
|
233
|
+
onConnect?: (server: T) => void;
|
|
234
|
+
req: http.IncomingMessage;
|
|
235
|
+
res: http.ServerResponse;
|
|
236
|
+
}) => {
|
|
237
|
+
if (
|
|
238
|
+
req.method === "GET" &&
|
|
239
|
+
new URL(req.url!, "http://localhost").pathname === endpoint
|
|
240
|
+
) {
|
|
241
|
+
const transport = new SSEServerTransport("/messages", res);
|
|
242
|
+
|
|
243
|
+
let server: T;
|
|
244
|
+
|
|
245
|
+
try {
|
|
246
|
+
server = await createServer(req);
|
|
247
|
+
} catch (error) {
|
|
248
|
+
if (error instanceof Response) {
|
|
249
|
+
res.writeHead(error.status).end(error.statusText);
|
|
250
|
+
|
|
251
|
+
return;
|
|
252
|
+
}
|
|
253
|
+
|
|
254
|
+
res.writeHead(500).end("Error creating server");
|
|
255
|
+
|
|
256
|
+
return true;
|
|
257
|
+
}
|
|
258
|
+
|
|
259
|
+
activeTransports[transport.sessionId] = transport;
|
|
260
|
+
|
|
261
|
+
let closed = false;
|
|
262
|
+
|
|
263
|
+
res.on("close", async () => {
|
|
264
|
+
closed = true;
|
|
265
|
+
|
|
266
|
+
try {
|
|
267
|
+
await server.close();
|
|
268
|
+
} catch (error) {
|
|
269
|
+
console.error("Error closing server:", error);
|
|
270
|
+
}
|
|
271
|
+
|
|
272
|
+
delete activeTransports[transport.sessionId];
|
|
273
|
+
|
|
274
|
+
onClose?.(server);
|
|
275
|
+
});
|
|
276
|
+
|
|
277
|
+
try {
|
|
278
|
+
await server.connect(transport);
|
|
279
|
+
|
|
280
|
+
await transport.send({
|
|
281
|
+
jsonrpc: "2.0",
|
|
282
|
+
method: "sse/connection",
|
|
283
|
+
params: { message: "SSE Connection established" },
|
|
284
|
+
});
|
|
285
|
+
|
|
286
|
+
onConnect?.(server);
|
|
287
|
+
} catch (error) {
|
|
288
|
+
if (!closed) {
|
|
289
|
+
console.error("Error connecting to server:", error);
|
|
290
|
+
|
|
291
|
+
res.writeHead(500).end("Error connecting to server");
|
|
292
|
+
}
|
|
293
|
+
}
|
|
294
|
+
|
|
295
|
+
return true;
|
|
296
|
+
}
|
|
297
|
+
|
|
298
|
+
if (req.method === "POST" && req.url?.startsWith("/messages")) {
|
|
299
|
+
const sessionId = new URL(req.url, "https://example.com").searchParams.get(
|
|
300
|
+
"sessionId",
|
|
301
|
+
);
|
|
302
|
+
|
|
303
|
+
if (!sessionId) {
|
|
304
|
+
res.writeHead(400).end("No sessionId");
|
|
305
|
+
|
|
306
|
+
return true;
|
|
307
|
+
}
|
|
308
|
+
|
|
309
|
+
const activeTransport: SSEServerTransport | undefined =
|
|
310
|
+
activeTransports[sessionId];
|
|
311
|
+
|
|
312
|
+
if (!activeTransport) {
|
|
313
|
+
res.writeHead(400).end("No active transport");
|
|
314
|
+
|
|
315
|
+
return true;
|
|
316
|
+
}
|
|
317
|
+
|
|
318
|
+
await activeTransport.handlePostMessage(req, res);
|
|
319
|
+
|
|
320
|
+
return true;
|
|
321
|
+
}
|
|
322
|
+
|
|
323
|
+
return false;
|
|
324
|
+
};
|
|
325
|
+
|
|
326
|
+
export const startHTTPServer = async <T extends ServerLike>({
|
|
327
|
+
createServer,
|
|
328
|
+
eventStore,
|
|
329
|
+
onClose,
|
|
330
|
+
onConnect,
|
|
331
|
+
onUnhandledRequest,
|
|
332
|
+
port,
|
|
333
|
+
}: {
|
|
334
|
+
createServer: (request: http.IncomingMessage) => Promise<T>;
|
|
335
|
+
eventStore?: EventStore;
|
|
336
|
+
onClose?: (server: T) => void;
|
|
337
|
+
onConnect?: (server: T) => void;
|
|
338
|
+
onUnhandledRequest?: (
|
|
339
|
+
req: http.IncomingMessage,
|
|
340
|
+
res: http.ServerResponse,
|
|
341
|
+
) => Promise<void>;
|
|
342
|
+
port: number;
|
|
343
|
+
}): Promise<SSEServer> => {
|
|
344
|
+
const activeSSETransports: Record<string, SSEServerTransport> = {};
|
|
345
|
+
|
|
346
|
+
const activeStreamTransports: Record<
|
|
347
|
+
string,
|
|
348
|
+
{
|
|
349
|
+
server: T;
|
|
350
|
+
transport: StreamableHTTPServerTransport;
|
|
351
|
+
}
|
|
352
|
+
> = {};
|
|
353
|
+
|
|
354
|
+
/**
|
|
355
|
+
* @author https://dev.classmethod.jp/articles/mcp-sse/
|
|
356
|
+
*/
|
|
357
|
+
const httpServer = http.createServer(async (req, res) => {
|
|
358
|
+
if (req.headers.origin) {
|
|
359
|
+
try {
|
|
360
|
+
const origin = new URL(req.headers.origin);
|
|
361
|
+
|
|
362
|
+
res.setHeader("Access-Control-Allow-Origin", origin.origin);
|
|
363
|
+
res.setHeader("Access-Control-Allow-Credentials", "true");
|
|
364
|
+
res.setHeader("Access-Control-Allow-Methods", "GET, POST, OPTIONS");
|
|
365
|
+
res.setHeader("Access-Control-Allow-Headers", "*");
|
|
366
|
+
} catch (error) {
|
|
367
|
+
console.error("Error parsing origin:", error);
|
|
368
|
+
}
|
|
369
|
+
}
|
|
370
|
+
|
|
371
|
+
if (req.method === "OPTIONS") {
|
|
372
|
+
res.writeHead(204);
|
|
373
|
+
res.end();
|
|
374
|
+
return;
|
|
375
|
+
}
|
|
376
|
+
|
|
377
|
+
if (req.method === "GET" && req.url === `/ping`) {
|
|
378
|
+
res.writeHead(200).end("pong");
|
|
379
|
+
return;
|
|
380
|
+
}
|
|
381
|
+
|
|
382
|
+
if (await handleSSERequest({
|
|
383
|
+
activeTransports: activeSSETransports,
|
|
384
|
+
createServer,
|
|
385
|
+
endpoint: "/sse",
|
|
386
|
+
onClose,
|
|
387
|
+
onConnect,
|
|
388
|
+
req,
|
|
389
|
+
res,
|
|
390
|
+
})) {
|
|
391
|
+
return;
|
|
392
|
+
}
|
|
393
|
+
|
|
394
|
+
if (await handleStreamRequest({
|
|
395
|
+
activeTransports: activeStreamTransports,
|
|
396
|
+
createServer,
|
|
397
|
+
endpoint: "/stream",
|
|
398
|
+
eventStore,
|
|
399
|
+
onClose,
|
|
400
|
+
onConnect,
|
|
401
|
+
req,
|
|
402
|
+
res,
|
|
403
|
+
})) {
|
|
404
|
+
return;
|
|
405
|
+
}
|
|
406
|
+
|
|
407
|
+
if (onUnhandledRequest) {
|
|
408
|
+
await onUnhandledRequest(req, res);
|
|
409
|
+
} else {
|
|
410
|
+
res.writeHead(404).end();
|
|
411
|
+
}
|
|
412
|
+
});
|
|
413
|
+
|
|
414
|
+
await new Promise((resolve) => {
|
|
415
|
+
httpServer.listen(port, "::", () => {
|
|
416
|
+
resolve(undefined);
|
|
417
|
+
});
|
|
418
|
+
});
|
|
419
|
+
|
|
420
|
+
return {
|
|
421
|
+
close: async () => {
|
|
422
|
+
for (const transport of Object.values(activeSSETransports)) {
|
|
423
|
+
await transport.close();
|
|
424
|
+
}
|
|
425
|
+
|
|
426
|
+
for (const transport of Object.values(activeStreamTransports)) {
|
|
427
|
+
await transport.transport.close();
|
|
428
|
+
}
|
|
429
|
+
|
|
430
|
+
return new Promise((resolve, reject) => {
|
|
431
|
+
httpServer.close((error) => {
|
|
432
|
+
if (error) {
|
|
433
|
+
reject(error);
|
|
434
|
+
|
|
435
|
+
return;
|
|
436
|
+
}
|
|
437
|
+
|
|
438
|
+
resolve();
|
|
439
|
+
});
|
|
440
|
+
});
|
|
441
|
+
},
|
|
442
|
+
};
|
|
443
|
+
};
|
|
444
|
+
|