This commit is contained in:
bipproduction
2025-10-27 10:16:03 +08:00
parent bbd12702aa
commit 4fde9c1087

View File

@@ -3,18 +3,6 @@ import { v4 as uuidv4 } from "uuid";
const API_KEY = process.env.MCP_API_KEY ?? "super-secret-key"; const API_KEY = process.env.MCP_API_KEY ?? "super-secret-key";
const PORT = Number(process.env.PORT ?? 3000); const PORT = Number(process.env.PORT ?? 3000);
const PING_INTERVAL_MS = 25_000;
// =====================
// Store session & clients
// =====================
type Client = {
id: string;
send: (data: string) => void;
close: () => void;
};
const sessions = new Map<string, Set<Client>>();
// ===================== // =====================
// Helper Functions // Helper Functions
@@ -28,48 +16,18 @@ function isAuthorized(headers: Headers) {
return headers.get("x-api-key") === API_KEY; return headers.get("x-api-key") === API_KEY;
} }
function formatSSE(event: string, data: any, id?: string) {
const payload = typeof data === "string" ? data : JSON.stringify(data);
const lines: string[] = [];
if (id) lines.push(`id: ${id}`);
if (event) lines.push(`event: ${event}`);
// Split data into multiple data: lines if needed
payload.split("\n").forEach(line => {
lines.push(`data: ${line}`);
});
lines.push(""); // Empty line to end the message
return lines.join("\n") + "\n";
}
function broadcast(sessionId: string, event: string, data: any) {
const clients = sessions.get(sessionId);
if (!clients) return 0;
const messageId = uuidv4();
const message = formatSSE(event, data, messageId);
for (const client of clients) {
try {
client.send(message);
} catch {
clients.delete(client);
}
}
return clients.size;
}
// ===================== // =====================
// Tools Definition // Tools Definition
// ===================== // =====================
type Tool = { type Tool = {
name: string; name: string;
description: string; description: string;
inputSchema?: { inputSchema: {
type: string; type: string;
properties?: Record<string, any>; properties: Record<string, any>;
required?: string[]; required?: string[];
additionalProperties?: boolean;
$schema?: string;
}; };
run: (input?: any) => Promise<any>; run: (input?: any) => Promise<any>;
}; };
@@ -81,6 +39,8 @@ const tools: Tool[] = [
inputSchema: { inputSchema: {
type: "object", type: "object",
properties: {}, properties: {},
additionalProperties: true,
$schema: "http://json-schema.org/draft-07/schema#",
}, },
run: async () => ({ pong: Date.now() }), run: async () => ({ pong: Date.now() }),
}, },
@@ -90,6 +50,8 @@ const tools: Tool[] = [
inputSchema: { inputSchema: {
type: "object", type: "object",
properties: {}, properties: {},
additionalProperties: true,
$schema: "http://json-schema.org/draft-07/schema#",
}, },
run: async () => ({ uuid: uuidv4() }), run: async () => ({ uuid: uuidv4() }),
}, },
@@ -99,265 +61,299 @@ const tools: Tool[] = [
inputSchema: { inputSchema: {
type: "object", type: "object",
properties: { properties: {
message: { input: {
type: "string", type: "string",
description: "Message to echo back", description: "Message to echo back",
}, },
}, },
required: ["input"],
additionalProperties: true,
$schema: "http://json-schema.org/draft-07/schema#",
}, },
run: async (input) => ({ echo: input }), run: async (input) => ({ echo: input }),
}, },
{
name: "Calculator",
description: "Useful for getting the result of a math expression. The input to this tool should be a valid mathematical expression that could be executed by a simple calculator.",
inputSchema: {
type: "object",
properties: {
input: {
type: "string",
},
},
required: ["input"],
additionalProperties: true,
$schema: "http://json-schema.org/draft-07/schema#",
},
run: async (input) => {
try {
// Simple math evaluation (be careful in production!)
const result = Function(`"use strict"; return (${input.input})`)();
return { result: String(result) };
} catch (error: any) {
throw new Error(`Invalid expression: ${error.message}`);
}
},
},
]; ];
// =====================
// MCP Protocol Types
// =====================
type JSONRPCRequest = {
jsonrpc: "2.0";
id: string | number;
method: string;
params?: any;
};
type JSONRPCResponse = {
jsonrpc: "2.0";
id: string | number;
result?: any;
error?: {
code: number;
message: string;
data?: any;
};
};
type JSONRPCNotification = {
jsonrpc: "2.0";
method: string;
params?: any;
};
// =====================
// MCP Handler
// =====================
function handleMCPRequest(request: JSONRPCRequest): JSONRPCResponse {
const { id, method, params } = request;
switch (method) {
case "initialize":
return {
jsonrpc: "2.0",
id,
result: {
protocolVersion: "2024-11-05",
capabilities: {
tools: {},
},
serverInfo: {
name: "elysia-mcp-server",
version: "1.0.0",
},
},
};
case "tools/list":
return {
jsonrpc: "2.0",
id,
result: {
tools: tools.map(({ name, description, inputSchema }) => ({
name,
description,
inputSchema,
})),
},
};
case "tools/call":
const toolName = params?.name;
const tool = tools.find((t) => t.name === toolName);
if (!tool) {
return {
jsonrpc: "2.0",
id,
error: {
code: -32601,
message: `Tool '${toolName}' not found`,
},
};
}
try {
// Note: This is synchronous for simplicity
// In real implementation, you'd need to handle async properly
let result: any;
tool.run(params?.arguments || {}).then((r) => (result = r));
return {
jsonrpc: "2.0",
id,
result: {
content: [
{
type: "text",
text: JSON.stringify(result || { pending: true }),
},
],
},
};
} catch (error: any) {
return {
jsonrpc: "2.0",
id,
error: {
code: -32603,
message: error.message,
},
};
}
case "ping":
return {
jsonrpc: "2.0",
id,
result: {},
};
default:
return {
jsonrpc: "2.0",
id,
error: {
code: -32601,
message: `Method '${method}' not found`,
},
};
}
}
async function handleMCPRequestAsync(request: JSONRPCRequest): Promise<JSONRPCResponse> {
const { id, method, params } = request;
if (method === "tools/call") {
const toolName = params?.name;
const tool = tools.find((t) => t.name === toolName);
if (!tool) {
return {
jsonrpc: "2.0",
id,
error: {
code: -32601,
message: `Tool '${toolName}' not found`,
},
};
}
try {
const result = await tool.run(params?.arguments || {});
return {
jsonrpc: "2.0",
id,
result: {
content: [
{
type: "text",
text: JSON.stringify(result),
},
],
},
};
} catch (error: any) {
return {
jsonrpc: "2.0",
id,
error: {
code: -32603,
message: error.message,
},
};
}
}
// For other methods, use sync handler
return handleMCPRequest(request);
}
// ===================== // =====================
// Server Initialization // Server Initialization
// ===================== // =====================
export const MCPRoute = new Elysia() export const MCPRoute = new Elysia()
// ===================== // =====================
// SSE Stream // MCP HTTP Streamable Endpoint
// =====================
.get("/mcp/:sessionId", ({ params, set, request }) => {
const { sessionId } = params;
// Check authorization for SSE connection
if (!isAuthorized(request.headers)) {
set.status = 401;
return { error: "Unauthorized" };
}
set.headers["Content-Type"] = "text/event-stream";
set.headers["Cache-Control"] = "no-cache, no-transform";
set.headers["Connection"] = "keep-alive";
set.headers["Access-Control-Allow-Origin"] = "*";
set.headers["X-Accel-Buffering"] = "no";
const stream = new TransformStream();
const writer = stream.writable.getWriter();
const encoder = new TextEncoder();
const client: Client = {
id: uuidv4(),
send: (data) => {
try {
writer.write(encoder.encode(data));
} catch (e) {
console.error("Error writing to stream:", e);
}
},
close: () => {
try {
writer.close();
} catch (e) {
// Stream already closed
}
const set = sessions.get(sessionId);
if (set) {
set.delete(client);
if (set.size === 0) sessions.delete(sessionId);
}
},
};
if (!sessions.has(sessionId)) sessions.set(sessionId, new Set());
sessions.get(sessionId)!.add(client);
// Send initial connection message
client.send(formatSSE("connected", {
sessionId,
clientId: client.id,
timestamp: Date.now()
}));
// Send tools list on connection
client.send(formatSSE("tools", {
tools: tools.map(({ name, description, inputSchema }) => ({
name,
description,
inputSchema
}))
}));
// Setup ping interval
const ping = setInterval(() => {
try {
client.send(formatSSE("ping", { ts: Date.now() }));
} catch (e) {
clearInterval(ping);
}
}, PING_INTERVAL_MS);
const readable = stream.readable;
// Cleanup on connection close
readable.pipeTo(new WritableStream()).catch(() => {
clearInterval(ping);
client.close();
});
return new Response(readable, {
headers: {
"Content-Type": "text/event-stream",
"Cache-Control": "no-cache, no-transform",
"Connection": "keep-alive",
"Access-Control-Allow-Origin": "*",
"X-Accel-Buffering": "no",
},
status: 200,
});
})
// =====================
// MCP Session Status
// =====================
.get("/mcp/:sessionId/status", ({ params, set }) => {
set.headers["Access-Control-Allow-Origin"] = "*";
const clients = sessions.get(params.sessionId);
return {
sessionId: params.sessionId,
connected: clients?.size ?? 0,
clients: Array.from(clients || []).map(c => c.id),
};
})
// =====================
// MCP Broadcast
// ===================== // =====================
.post("/mcp/:sessionId", async ({ params, request, set }) => { .post("/mcp/:sessionId", async ({ params, request, set }) => {
set.headers["Access-Control-Allow-Origin"] = "*"; set.headers["Content-Type"] = "application/json";
if (!isAuthorized(request.headers)) {
set.status = 401;
return { error: "Unauthorized" };
}
const body = await request.json();
const event = body.event ?? "message";
const data = body.data ?? body;
const sentTo = broadcast(params.sessionId, event, data);
return { ok: true, sentTo };
})
// =====================
// Delete /mcp/:sessionId
// =====================
.delete("/mcp/:sessionId", ({ params, request, set }) => {
set.headers["Access-Control-Allow-Origin"] = "*";
if (!isAuthorized(request.headers)) {
set.status = 401;
return { error: "Unauthorized" };
}
const clients = sessions.get(params.sessionId);
if (clients) {
for (const c of clients) c.close();
sessions.delete(params.sessionId);
}
return { ok: true };
})
// =====================
// Tools Introspection (Fixed path)
// =====================
.get("/mcp/:sessionId/tools", ({ params, set, request }) => {
set.headers["Access-Control-Allow-Origin"] = "*"; set.headers["Access-Control-Allow-Origin"] = "*";
// Optional: Check auth if needed // Optional: Check authorization
// if (!isAuthorized(request.headers)) { // if (!isAuthorized(request.headers)) {
// set.status = 401; // set.status = 401;
// return { error: "Unauthorized" }; // return { error: "Unauthorized" };
// } // }
return {
tools: tools.map(({ name, description, inputSchema }) => ({
name,
description,
inputSchema
}))
};
})
// Alternative global tools endpoint
.get("/tools", ({ set }) => {
set.headers["Access-Control-Allow-Origin"] = "*";
return {
tools: tools.map(({ name, description, inputSchema }) => ({
name,
description,
inputSchema
}))
};
})
// =====================
// Run Tool (Fixed path)
// =====================
.post("/mcp/:sessionId/tools/:toolName", async ({ params, request, set }) => {
set.headers["Access-Control-Allow-Origin"] = "*";
if (!isAuthorized(request.headers)) {
set.status = 401;
return { error: "Unauthorized" };
}
const tool = tools.find((t) => t.name === params.toolName);
if (!tool) {
set.status = 404;
return { error: `Tool '${params.toolName}' not found` };
}
try { try {
const body = await request.json().catch(() => ({})); const body = await request.json();
const result = await tool.run(body);
// Broadcast tool execution result to session // Handle single request
broadcast(params.sessionId, "tool_result", { if (!Array.isArray(body)) {
tool: tool.name, const response = await handleMCPRequestAsync(body as JSONRPCRequest);
result, return response;
}
// Handle batch requests
const responses = await Promise.all(
body.map((req) => handleMCPRequestAsync(req as JSONRPCRequest))
);
return responses;
} catch (error: any) {
set.status = 400;
return {
jsonrpc: "2.0",
id: null,
error: {
code: -32700,
message: "Parse error",
data: error.message,
},
};
}
})
// =====================
// Simple tools list endpoint (for debugging)
// =====================
.get("/mcp/:sessionId/tools", ({ set }) => {
set.headers["Access-Control-Allow-Origin"] = "*";
return {
data: tools.map(({ name, description, inputSchema }) => ({
name,
value: name,
description,
inputSchema,
})),
};
})
// =====================
// Session Status
// =====================
.get("/mcp/:sessionId/status", ({ params, set }) => {
set.headers["Access-Control-Allow-Origin"] = "*";
return {
sessionId: params.sessionId,
status: "active",
timestamp: Date.now(), timestamp: Date.now(),
});
return {
ok: true,
tool: tool.name,
result
}; };
} catch (error: any) {
set.status = 500;
return {
error: "Tool execution failed",
message: error.message
};
}
}) })
// Alternative global tool execution // =====================
.post("/tools/:toolName", async ({ params, request, set }) => { // Health Check
// =====================
.get("/health", ({ set }) => {
set.headers["Access-Control-Allow-Origin"] = "*"; set.headers["Access-Control-Allow-Origin"] = "*";
if (!isAuthorized(request.headers)) {
set.status = 401;
return { error: "Unauthorized" };
}
const tool = tools.find((t) => t.name === params.toolName);
if (!tool) {
set.status = 404;
return { error: `Tool '${params.toolName}' not found` };
}
try {
const body = await request.json().catch(() => ({}));
const result = await tool.run(body);
return { return {
ok: true, status: "ok",
tool: tool.name, timestamp: Date.now(),
result tools: tools.length,
}; };
} catch (error: any) {
set.status = 500;
return {
error: "Tool execution failed",
message: error.message
};
}
}) })
// ===================== // =====================
@@ -365,24 +361,16 @@ export const MCPRoute = new Elysia()
// ===================== // =====================
.options("/mcp/:sessionId", ({ set }) => { .options("/mcp/:sessionId", ({ set }) => {
set.headers["Access-Control-Allow-Origin"] = "*"; set.headers["Access-Control-Allow-Origin"] = "*";
set.headers["Access-Control-Allow-Methods"] = "GET,POST,DELETE,OPTIONS"; set.headers["Access-Control-Allow-Methods"] = "GET,POST,OPTIONS";
set.headers["Access-Control-Allow-Headers"] = "Content-Type,X-API-Key,Authorization"; set.headers["Access-Control-Allow-Headers"] = "Content-Type,Authorization,X-API-Key";
set.status = 204; set.status = 204;
return ""; return "";
}) })
.options("/mcp/:sessionId/tools", ({ set }) => { .options("/mcp/:sessionId/tools", ({ set }) => {
set.headers["Access-Control-Allow-Origin"] = "*"; set.headers["Access-Control-Allow-Origin"] = "*";
set.headers["Access-Control-Allow-Methods"] = "GET,POST,OPTIONS"; set.headers["Access-Control-Allow-Methods"] = "GET,OPTIONS";
set.headers["Access-Control-Allow-Headers"] = "Content-Type,X-API-Key,Authorization"; set.headers["Access-Control-Allow-Headers"] = "Content-Type,Authorization,X-API-Key";
set.status = 204;
return "";
})
.options("/mcp/:sessionId/tools/:toolName", ({ set }) => {
set.headers["Access-Control-Allow-Origin"] = "*";
set.headers["Access-Control-Allow-Methods"] = "POST,OPTIONS";
set.headers["Access-Control-Allow-Headers"] = "Content-Type,X-API-Key,Authorization";
set.status = 204; set.status = 204;
return ""; return "";
}); });