@eggjs/tegg-controller-plugin 3.53.0 → 3.55.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.
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
+
};
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
const path_to_regexp_1 = __importDefault(require("path-to-regexp"));
|
|
7
|
+
exports.default = () => {
|
|
8
|
+
return async function mcpBodyMiddleware(ctx, next) {
|
|
9
|
+
const arr = [ctx.app.config.mcp.sseInitPath, ctx.app.config.mcp.sseMessagePath, ctx.app.config.mcp.streamPath];
|
|
10
|
+
const res = arr.some(igPath => {
|
|
11
|
+
const match = (0, path_to_regexp_1.default)(igPath, [], {
|
|
12
|
+
end: false,
|
|
13
|
+
});
|
|
14
|
+
return match.test(ctx.path);
|
|
15
|
+
});
|
|
16
|
+
if (res) {
|
|
17
|
+
ctx.disableBodyParser = true;
|
|
18
|
+
}
|
|
19
|
+
await next();
|
|
20
|
+
};
|
|
21
|
+
};
|
|
22
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWNwX2JvZHlfbWlkZGxld2FyZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIm1jcF9ib2R5X21pZGRsZXdhcmUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7QUFDQSxvRUFBMEM7QUFFMUMsa0JBQWUsR0FBRyxFQUFFO0lBQ2xCLE9BQU8sS0FBSyxVQUFVLGlCQUFpQixDQUFDLEdBQWUsRUFBRSxJQUFVO1FBQ2pFLE1BQU0sR0FBRyxHQUFHLENBQUUsR0FBRyxDQUFDLEdBQUcsQ0FBQyxNQUFNLENBQUMsR0FBRyxDQUFDLFdBQVcsRUFBRSxHQUFHLENBQUMsR0FBRyxDQUFDLE1BQU0sQ0FBQyxHQUFHLENBQUMsY0FBYyxFQUFFLEdBQUcsQ0FBQyxHQUFHLENBQUMsTUFBTSxDQUFDLEdBQUcsQ0FBQyxVQUFVLENBQUUsQ0FBQztRQUNqSCxNQUFNLEdBQUcsR0FBRyxHQUFHLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxFQUFFO1lBQzVCLE1BQU0sS0FBSyxHQUFHLElBQUEsd0JBQVksRUFBQyxNQUFNLEVBQUUsRUFBRSxFQUFFO2dCQUNyQyxHQUFHLEVBQUUsS0FBSzthQUNYLENBQUMsQ0FBQztZQUNILE9BQU8sS0FBSyxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDOUIsQ0FBQyxDQUFDLENBQUM7UUFDSCxJQUFJLEdBQUcsRUFBRSxDQUFDO1lBQ1IsR0FBRyxDQUFDLGlCQUFpQixHQUFHLElBQUksQ0FBQztRQUMvQixDQUFDO1FBQ0QsTUFBTSxJQUFJLEVBQUUsQ0FBQztJQUNmLENBQUMsQ0FBQztBQUNKLENBQUMsQ0FBQyJ9
|
package/app.js
CHANGED
|
@@ -11,6 +11,7 @@ const ControllerMetadataManager_1 = require("./lib/ControllerMetadataManager");
|
|
|
11
11
|
const EggControllerPrototypeHook_1 = require("./lib/EggControllerPrototypeHook");
|
|
12
12
|
const RootProtoManager_1 = require("./lib/RootProtoManager");
|
|
13
13
|
const EggControllerLoader_1 = require("./lib/EggControllerLoader");
|
|
14
|
+
const MCPControllerRegister_1 = require("./lib/impl/mcp/MCPControllerRegister");
|
|
14
15
|
// Load Controller process
|
|
15
16
|
// 1. await add load unit is ready, controller may depend other load unit
|
|
16
17
|
// 2. load ${app_base_dir}app/controller file
|
|
@@ -40,6 +41,30 @@ class ControllerAppBootHook {
|
|
|
40
41
|
});
|
|
41
42
|
// init http root proto middleware
|
|
42
43
|
this.prepareMiddleware(this.app.config.coreMiddleware);
|
|
44
|
+
if (this.app.mcpProxy) {
|
|
45
|
+
this.controllerRegisterFactory.registerControllerRegister(tegg_1.ControllerType.MCP, MCPControllerRegister_1.MCPControllerRegister.create);
|
|
46
|
+
// Don't let the mcp's body be consumed
|
|
47
|
+
this.app.config.coreMiddleware.unshift('mcpBodyMiddleware');
|
|
48
|
+
if (this.app.config.security.csrf.ignore) {
|
|
49
|
+
if (Array.isArray(this.app.config.security.csrf.ignore)) {
|
|
50
|
+
this.app.config.security.csrf.ignore = [
|
|
51
|
+
this.app.config.mcp.sseInitPath,
|
|
52
|
+
this.app.config.mcp.sseMessagePath,
|
|
53
|
+
this.app.config.mcp.streamPath,
|
|
54
|
+
...(Array.isArray(this.app.config.security.csrf.ignore)
|
|
55
|
+
? this.app.config.security.csrf.ignore
|
|
56
|
+
: [this.app.config.security.csrf.ignore]),
|
|
57
|
+
];
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
else {
|
|
61
|
+
this.app.config.security.csrf.ignore = [
|
|
62
|
+
this.app.config.mcp.sseInitPath,
|
|
63
|
+
this.app.config.mcp.sseMessagePath,
|
|
64
|
+
this.app.config.mcp.streamPath,
|
|
65
|
+
];
|
|
66
|
+
}
|
|
67
|
+
}
|
|
43
68
|
}
|
|
44
69
|
prepareMiddleware(middlewareNames) {
|
|
45
70
|
if (!middlewareNames.includes('teggCtxLifecycleMiddleware')) {
|
|
@@ -71,4 +96,4 @@ class ControllerAppBootHook {
|
|
|
71
96
|
}
|
|
72
97
|
}
|
|
73
98
|
exports.default = ControllerAppBootHook;
|
|
74
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
99
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXBwLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiYXBwLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7O0FBQ0EsaUVBQW9GO0FBQ3BGLCtFQUE0RTtBQUU1RSxzQ0FBMkU7QUFDM0UsbUZBQWdGO0FBQ2hGLCtFQUE0RTtBQUM1RSwrRUFBNEU7QUFDNUUsc0RBQStGO0FBQy9GLCtFQUE0RTtBQUM1RSxpRkFBOEU7QUFDOUUsNkRBQTBEO0FBQzFELG1FQUFnRTtBQUNoRSxnRkFBNkU7QUFFN0UsMEJBQTBCO0FBQzFCLHlFQUF5RTtBQUN6RSw2Q0FBNkM7QUFDN0Msc0RBQXNEO0FBRXRELE1BQXFCLHFCQUFxQjtJQU94QyxZQUFZLEdBQWdCO1FBQzFCLElBQUksQ0FBQyxHQUFHLEdBQUcsR0FBRyxDQUFDO1FBQ2YsSUFBSSxDQUFDLHlCQUF5QixHQUFHLElBQUkscURBQXlCLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDO1FBQ3pFLElBQUksQ0FBQyxHQUFHLENBQUMsZ0JBQWdCLEdBQUcsSUFBSSxtQ0FBZ0IsRUFBRSxDQUFDO1FBQ25ELElBQUksQ0FBQyxHQUFHLENBQUMseUJBQXlCLEdBQUcsSUFBSSxDQUFDLHlCQUF5QixDQUFDO1FBQ3BFLElBQUksQ0FBQyxHQUFHLENBQUMsNEJBQTRCLEdBQUcsbUNBQTRCLENBQUM7UUFDckUsSUFBSSxDQUFDLFlBQVksR0FBRyxJQUFJLHFEQUF5QixDQUFDLElBQUksQ0FBQyx5QkFBeUIsRUFBRSxJQUFJLENBQUMsR0FBRyxDQUFDLGdCQUFnQixDQUFDLENBQUM7UUFDN0csSUFBSSxDQUFDLHVCQUF1QixHQUFHLElBQUksdURBQTBCLEVBQUUsQ0FBQztJQUNsRSxDQUFDO0lBRUQsY0FBYztRQUNaLElBQUksQ0FBQyxHQUFHLENBQUMscUJBQXFCLENBQUMsaUJBQWlCLENBQUMsSUFBSSxDQUFDLFlBQVksQ0FBQyxDQUFDO1FBQ3BFLElBQUksQ0FBQyxHQUFHLENBQUMseUJBQXlCLENBQUMsaUJBQWlC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
|
|
@@ -0,0 +1,46 @@
|
|
|
1
|
+
import type { Application, Context } from 'egg';
|
|
2
|
+
import http, { ServerResponse } from 'node:http';
|
|
3
|
+
import { ControllerMetadata, MCPControllerMeta, MCPPromptMeta, MCPResourceMeta, MCPToolMeta } from '@eggjs/tegg';
|
|
4
|
+
import { EggPrototype } from '@eggjs/tegg-metadata';
|
|
5
|
+
import { ControllerRegister } from '../../ControllerRegister';
|
|
6
|
+
import { SSEServerTransport } from '@modelcontextprotocol/sdk/server/sse.js';
|
|
7
|
+
import { StreamableHTTPServerTransport } from '@modelcontextprotocol/sdk/server/streamableHttp.js';
|
|
8
|
+
import { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js';
|
|
9
|
+
import { MCPProtocols } from '@eggjs/mcp-proxy';
|
|
10
|
+
export interface MCPControllerHook {
|
|
11
|
+
preSSEInitHandle?: (ctx: Context, transport: SSEServerTransport, register: MCPControllerRegister) => Promise<void>;
|
|
12
|
+
preHandleInitHandle?: (ctx: Context) => Promise<void>;
|
|
13
|
+
preHandle?: (ctx: Context) => Promise<void>;
|
|
14
|
+
onStreamSessionInitialized?: (ctx: Context, transport: StreamableHTTPServerTransport, register: MCPControllerRegister) => Promise<void>;
|
|
15
|
+
preProxy?: (ctx: Context, proxyReq: http.IncomingMessage, proxyResp: http.ServerResponse) => Promise<void>;
|
|
16
|
+
schemaLoader?: (controllerMeta: MCPControllerMeta, meta: MCPPromptMeta | MCPToolMeta) => Promise<Parameters<McpServer['tool']>['2'] | Parameters<McpServer['prompt']>['2']>;
|
|
17
|
+
checkAndRunProxy?: (ctx: Context, type: MCPProtocols, sessionId: string) => Promise<boolean>;
|
|
18
|
+
}
|
|
19
|
+
export declare class MCPControllerRegister implements ControllerRegister {
|
|
20
|
+
static instance?: MCPControllerRegister;
|
|
21
|
+
readonly app: Application;
|
|
22
|
+
private readonly eggContainerFactory;
|
|
23
|
+
private readonly router;
|
|
24
|
+
private controllerProtos;
|
|
25
|
+
private registeredControllerProtos;
|
|
26
|
+
transports: Record<string, SSEServerTransport>;
|
|
27
|
+
sseConnections: Map<string, {
|
|
28
|
+
res: ServerResponse;
|
|
29
|
+
intervalId: NodeJS.Timeout;
|
|
30
|
+
}>;
|
|
31
|
+
private mcpServer;
|
|
32
|
+
private controllerMeta;
|
|
33
|
+
streamTransports: Record<string, StreamableHTTPServerTransport>;
|
|
34
|
+
static hooks: MCPControllerHook[];
|
|
35
|
+
static create(proto: EggPrototype, controllerMeta: ControllerMetadata, app: Application): MCPControllerRegister;
|
|
36
|
+
constructor(_proto: EggPrototype, controllerMeta: MCPControllerMeta, app: Application);
|
|
37
|
+
static addHook(hook: MCPControllerHook): void;
|
|
38
|
+
mcpStreamServerInit(): Promise<void>;
|
|
39
|
+
mcpServerInit(): void;
|
|
40
|
+
sseCtxStorageRun(ctx: Context, transport: SSEServerTransport | StreamableHTTPServerTransport): void;
|
|
41
|
+
mcpServerRegister(): void;
|
|
42
|
+
mcpPromptRegister(controllerProto: EggPrototype, promptMeta: MCPPromptMeta): Promise<void>;
|
|
43
|
+
mcpToolRegister(controllerProto: EggPrototype, toolMeta: MCPToolMeta): Promise<void>;
|
|
44
|
+
mcpResourceRegister(controllerProto: EggPrototype, resourceMeta: MCPResourceMeta): Promise<void>;
|
|
45
|
+
register(): Promise<void>;
|
|
46
|
+
}
|
|
@@ -0,0 +1,417 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
+
};
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
exports.MCPControllerRegister = void 0;
|
|
7
|
+
const node_assert_1 = __importDefault(require("node:assert"));
|
|
8
|
+
const tegg_1 = require("@eggjs/tegg");
|
|
9
|
+
const sse_js_1 = require("@modelcontextprotocol/sdk/server/sse.js");
|
|
10
|
+
const streamableHttp_js_1 = require("@modelcontextprotocol/sdk/server/streamableHttp.js");
|
|
11
|
+
const mcp_js_1 = require("@modelcontextprotocol/sdk/server/mcp.js");
|
|
12
|
+
const inMemoryEventStore_js_1 = require("@modelcontextprotocol/sdk/examples/shared/inMemoryEventStore.js");
|
|
13
|
+
const types_js_1 = require("@modelcontextprotocol/sdk/types.js");
|
|
14
|
+
const mcp_proxy_1 = require("@eggjs/mcp-proxy");
|
|
15
|
+
const raw_body_1 = __importDefault(require("raw-body"));
|
|
16
|
+
const content_type_1 = __importDefault(require("content-type"));
|
|
17
|
+
class MCPControllerRegister {
|
|
18
|
+
static create(proto, controllerMeta, app) {
|
|
19
|
+
(0, node_assert_1.default)(controllerMeta.type === tegg_1.ControllerType.MCP, 'controller meta type is not MCP');
|
|
20
|
+
if (!MCPControllerRegister.instance) {
|
|
21
|
+
MCPControllerRegister.instance = new MCPControllerRegister(proto, controllerMeta, app);
|
|
22
|
+
}
|
|
23
|
+
MCPControllerRegister.instance.controllerProtos.push(proto);
|
|
24
|
+
return MCPControllerRegister.instance;
|
|
25
|
+
}
|
|
26
|
+
constructor(_proto, controllerMeta, app) {
|
|
27
|
+
this.controllerProtos = [];
|
|
28
|
+
this.registeredControllerProtos = [];
|
|
29
|
+
this.transports = {};
|
|
30
|
+
this.sseConnections = new Map();
|
|
31
|
+
this.streamTransports = {};
|
|
32
|
+
this.app = app;
|
|
33
|
+
this.eggContainerFactory = app.eggContainerFactory;
|
|
34
|
+
this.router = app.router;
|
|
35
|
+
this.controllerMeta = controllerMeta;
|
|
36
|
+
}
|
|
37
|
+
static addHook(hook) {
|
|
38
|
+
MCPControllerRegister.hooks.push(hook);
|
|
39
|
+
}
|
|
40
|
+
async mcpStreamServerInit() {
|
|
41
|
+
var _a;
|
|
42
|
+
const allRouterFunc = this.router.all;
|
|
43
|
+
const self = this;
|
|
44
|
+
const initHandler = async (ctx) => {
|
|
45
|
+
var _a, _b, _c, _d, _e, _f, _g;
|
|
46
|
+
ctx.respond = false;
|
|
47
|
+
if (MCPControllerRegister.hooks.length > 0) {
|
|
48
|
+
for (const hook of MCPControllerRegister.hooks) {
|
|
49
|
+
await ((_a = hook.preHandle) === null || _a === void 0 ? void 0 : _a.call(hook, self.app.currentContext));
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
const sessionId = ctx.req.headers['mcp-session-id'];
|
|
53
|
+
if (!sessionId) {
|
|
54
|
+
const ct = content_type_1.default.parse((_b = ctx.req.headers['content-type']) !== null && _b !== void 0 ? _b : '');
|
|
55
|
+
let body;
|
|
56
|
+
try {
|
|
57
|
+
const rawBody = await (0, raw_body_1.default)(ctx.req, {
|
|
58
|
+
limit: '4mb',
|
|
59
|
+
encoding: (_c = ct.parameters.charset) !== null && _c !== void 0 ? _c : 'utf-8',
|
|
60
|
+
});
|
|
61
|
+
body = JSON.parse(rawBody);
|
|
62
|
+
}
|
|
63
|
+
catch (e) {
|
|
64
|
+
ctx.status = 400;
|
|
65
|
+
ctx.body = {
|
|
66
|
+
jsonrpc: '2.0',
|
|
67
|
+
error: {
|
|
68
|
+
code: -32000,
|
|
69
|
+
message: `Bad Request: body should is json, ${e.toString()}`,
|
|
70
|
+
},
|
|
71
|
+
id: null,
|
|
72
|
+
};
|
|
73
|
+
return;
|
|
74
|
+
}
|
|
75
|
+
if ((0, types_js_1.isInitializeRequest)(body)) {
|
|
76
|
+
ctx.respond = false;
|
|
77
|
+
const eventStore = (_e = (_d = this.app.config.mcp) === null || _d === void 0 ? void 0 : _d.eventStore) !== null && _e !== void 0 ? _e : new inMemoryEventStore_js_1.InMemoryEventStore();
|
|
78
|
+
const self = this;
|
|
79
|
+
const transport = new streamableHttp_js_1.StreamableHTTPServerTransport({
|
|
80
|
+
sessionIdGenerator: () => { var _a, _b; return (_b = (_a = this.app.config.mcp) === null || _a === void 0 ? void 0 : _a.sessionIdGenerator) === null || _b === void 0 ? void 0 : _b.call(_a); },
|
|
81
|
+
eventStore,
|
|
82
|
+
onsessioninitialized: async () => {
|
|
83
|
+
var _a;
|
|
84
|
+
if (MCPControllerRegister.hooks.length > 0) {
|
|
85
|
+
for (const hook of MCPControllerRegister.hooks) {
|
|
86
|
+
await ((_a = hook.onStreamSessionInitialized) === null || _a === void 0 ? void 0 : _a.call(hook, self.app.currentContext, transport, self));
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
},
|
|
90
|
+
});
|
|
91
|
+
ctx.set({
|
|
92
|
+
'content-type': 'text/event-stream',
|
|
93
|
+
'transfer-encoding': 'chunked',
|
|
94
|
+
});
|
|
95
|
+
await self.mcpServer.connect(transport);
|
|
96
|
+
await ctx.app.ctxStorage.run(ctx, async () => {
|
|
97
|
+
await transport.handleRequest(ctx.req, ctx.res, body);
|
|
98
|
+
});
|
|
99
|
+
}
|
|
100
|
+
else {
|
|
101
|
+
ctx.status = 400;
|
|
102
|
+
ctx.body = {
|
|
103
|
+
jsonrpc: '2.0',
|
|
104
|
+
error: {
|
|
105
|
+
code: -32000,
|
|
106
|
+
message: 'Bad Request: No valid session ID provided',
|
|
107
|
+
},
|
|
108
|
+
id: null,
|
|
109
|
+
};
|
|
110
|
+
return;
|
|
111
|
+
}
|
|
112
|
+
}
|
|
113
|
+
else if (sessionId) {
|
|
114
|
+
const transport = self.streamTransports[sessionId];
|
|
115
|
+
if (transport) {
|
|
116
|
+
if (MCPControllerRegister.hooks.length > 0) {
|
|
117
|
+
for (const hook of MCPControllerRegister.hooks) {
|
|
118
|
+
await ((_f = hook.preHandle) === null || _f === void 0 ? void 0 : _f.call(hook, self.app.currentContext));
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
ctx.respond = false;
|
|
122
|
+
ctx.set({
|
|
123
|
+
'content-type': 'text/event-stream',
|
|
124
|
+
'transfer-encoding': 'chunked',
|
|
125
|
+
});
|
|
126
|
+
await ctx.app.ctxStorage.run(ctx, async () => {
|
|
127
|
+
await transport.handleRequest(ctx.req, ctx.res);
|
|
128
|
+
});
|
|
129
|
+
return;
|
|
130
|
+
}
|
|
131
|
+
if (MCPControllerRegister.hooks.length > 0) {
|
|
132
|
+
for (const hook of MCPControllerRegister.hooks) {
|
|
133
|
+
const checked = await ((_g = hook.checkAndRunProxy) === null || _g === void 0 ? void 0 : _g.call(hook, self.app.currentContext, mcp_proxy_1.MCPProtocols.STREAM, sessionId));
|
|
134
|
+
if (checked) {
|
|
135
|
+
return;
|
|
136
|
+
}
|
|
137
|
+
}
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
return;
|
|
141
|
+
};
|
|
142
|
+
Reflect.apply(allRouterFunc, this.router, [
|
|
143
|
+
'chairMcpStreamInit',
|
|
144
|
+
(_a = self.app.config.mcp) === null || _a === void 0 ? void 0 : _a.streamPath,
|
|
145
|
+
...[],
|
|
146
|
+
initHandler,
|
|
147
|
+
]);
|
|
148
|
+
}
|
|
149
|
+
mcpServerInit() {
|
|
150
|
+
var _a;
|
|
151
|
+
const routerFunc = this.router.get;
|
|
152
|
+
// const aclMiddleware = aclMiddlewareFactory(this.controllerMeta, this.methodMeta);
|
|
153
|
+
// if (aclMiddleware) {
|
|
154
|
+
// methodMiddlewares.push(aclMiddleware);
|
|
155
|
+
// }
|
|
156
|
+
const self = this;
|
|
157
|
+
const initHandler = async (ctx) => {
|
|
158
|
+
var _a, _b, _c;
|
|
159
|
+
const transport = new sse_js_1.SSEServerTransport(self.app.config.mcp.sseMessagePath, ctx.res);
|
|
160
|
+
const id = transport.sessionId;
|
|
161
|
+
if (MCPControllerRegister.hooks.length > 0) {
|
|
162
|
+
for (const hook of MCPControllerRegister.hooks) {
|
|
163
|
+
await ((_a = hook.preSSEInitHandle) === null || _a === void 0 ? void 0 : _a.call(hook, self.app.currentContext, transport, self));
|
|
164
|
+
}
|
|
165
|
+
}
|
|
166
|
+
// https://github.com/modelcontextprotocol/typescript-sdk/issues/270#issuecomment-2789526821
|
|
167
|
+
const intervalId = setInterval(() => {
|
|
168
|
+
if (self.sseConnections.has(id) && !ctx.res.writableEnded) {
|
|
169
|
+
ctx.res.write(': keepalive\n\n');
|
|
170
|
+
}
|
|
171
|
+
else {
|
|
172
|
+
clearInterval(intervalId);
|
|
173
|
+
self.sseConnections.delete(id);
|
|
174
|
+
}
|
|
175
|
+
}, (_c = (_b = self.app.config.mcp) === null || _b === void 0 ? void 0 : _b.sseHeartTime) !== null && _c !== void 0 ? _c : 25000);
|
|
176
|
+
self.sseConnections.set(id, { res: ctx.res, intervalId });
|
|
177
|
+
self.transports[id] = transport;
|
|
178
|
+
ctx.set({
|
|
179
|
+
'content-type': 'text/event-stream',
|
|
180
|
+
'transfer-encoding': 'chunked',
|
|
181
|
+
});
|
|
182
|
+
ctx.respond = false;
|
|
183
|
+
await self.mcpServer.connect(transport);
|
|
184
|
+
return self.sseCtxStorageRun.bind(self)(ctx, transport);
|
|
185
|
+
};
|
|
186
|
+
Reflect.apply(routerFunc, this.router, [
|
|
187
|
+
'chairMcpInit',
|
|
188
|
+
(_a = self.app.config.mcp) === null || _a === void 0 ? void 0 : _a.sseInitPath,
|
|
189
|
+
...[],
|
|
190
|
+
initHandler,
|
|
191
|
+
]);
|
|
192
|
+
}
|
|
193
|
+
sseCtxStorageRun(ctx, transport) {
|
|
194
|
+
const closeFunc = transport.onclose;
|
|
195
|
+
transport.onclose = (...args) => {
|
|
196
|
+
closeFunc === null || closeFunc === void 0 ? void 0 : closeFunc(...args);
|
|
197
|
+
};
|
|
198
|
+
transport.onerror = error => {
|
|
199
|
+
this.app.logger.error('session %s error %o', transport.sessionId, error);
|
|
200
|
+
};
|
|
201
|
+
const messageFunc = transport.onmessage;
|
|
202
|
+
transport.onmessage = async (...args) => {
|
|
203
|
+
await ctx.app.ctxStorage.run(ctx, async () => {
|
|
204
|
+
await messageFunc(...args);
|
|
205
|
+
});
|
|
206
|
+
};
|
|
207
|
+
}
|
|
208
|
+
mcpServerRegister() {
|
|
209
|
+
var _a;
|
|
210
|
+
const routerFunc = this.router.post;
|
|
211
|
+
const self = this;
|
|
212
|
+
// const aclMiddleware = aclMiddlewareFactory(this.controllerMeta, this.methodMeta);
|
|
213
|
+
// if (aclMiddleware) {
|
|
214
|
+
// methodMiddlewares.push(aclMiddleware);
|
|
215
|
+
// }
|
|
216
|
+
const messageHander = async (ctx) => {
|
|
217
|
+
var _a, _b;
|
|
218
|
+
const sessionId = ctx.query.sessionId;
|
|
219
|
+
if (self.transports[sessionId]) {
|
|
220
|
+
if (MCPControllerRegister.hooks.length > 0) {
|
|
221
|
+
for (const hook of MCPControllerRegister.hooks) {
|
|
222
|
+
await ((_a = hook.preHandleInitHandle) === null || _a === void 0 ? void 0 : _a.call(hook, self.app.currentContext));
|
|
223
|
+
}
|
|
224
|
+
}
|
|
225
|
+
self.app.logger.info('message coming', sessionId);
|
|
226
|
+
try {
|
|
227
|
+
await self.transports[sessionId].handlePostMessage(ctx.req, ctx.res);
|
|
228
|
+
}
|
|
229
|
+
catch (error) {
|
|
230
|
+
self.app.logger.error('Error handling MCP message', error);
|
|
231
|
+
if (!ctx.res.headersSent) {
|
|
232
|
+
ctx.status = 500;
|
|
233
|
+
ctx.body = {
|
|
234
|
+
jsonrpc: '2.0',
|
|
235
|
+
error: {
|
|
236
|
+
code: -32603,
|
|
237
|
+
message: `Internal error: ${error.message}`,
|
|
238
|
+
},
|
|
239
|
+
id: null,
|
|
240
|
+
};
|
|
241
|
+
}
|
|
242
|
+
}
|
|
243
|
+
return;
|
|
244
|
+
}
|
|
245
|
+
if (MCPControllerRegister.hooks.length > 0) {
|
|
246
|
+
for (const hook of MCPControllerRegister.hooks) {
|
|
247
|
+
const checked = await ((_b = hook.checkAndRunProxy) === null || _b === void 0 ? void 0 : _b.call(hook, self.app.currentContext, mcp_proxy_1.MCPProtocols.SSE, sessionId));
|
|
248
|
+
if (checked) {
|
|
249
|
+
return;
|
|
250
|
+
}
|
|
251
|
+
}
|
|
252
|
+
}
|
|
253
|
+
};
|
|
254
|
+
Reflect.apply(routerFunc, this.router, [
|
|
255
|
+
'chairMcpMessage',
|
|
256
|
+
(_a = self.app.config.mcp) === null || _a === void 0 ? void 0 : _a.sseMessagePath,
|
|
257
|
+
...[],
|
|
258
|
+
messageHander,
|
|
259
|
+
]);
|
|
260
|
+
}
|
|
261
|
+
async mcpPromptRegister(controllerProto, promptMeta) {
|
|
262
|
+
var _a, _b, _c, _d;
|
|
263
|
+
const controllerMeta = controllerProto.getMetaData(tegg_1.CONTROLLER_META_DATA);
|
|
264
|
+
const args = [(_a = promptMeta.mcpName) !== null && _a !== void 0 ? _a : promptMeta.name];
|
|
265
|
+
if (promptMeta.description) {
|
|
266
|
+
args.push(promptMeta.description);
|
|
267
|
+
}
|
|
268
|
+
let schema;
|
|
269
|
+
if ((_b = promptMeta.detail) === null || _b === void 0 ? void 0 : _b.argsSchema) {
|
|
270
|
+
schema = (_c = promptMeta.detail) === null || _c === void 0 ? void 0 : _c.argsSchema;
|
|
271
|
+
args.push(schema);
|
|
272
|
+
}
|
|
273
|
+
else if (MCPControllerRegister.hooks.length > 0) {
|
|
274
|
+
for (const hook of MCPControllerRegister.hooks) {
|
|
275
|
+
schema = await ((_d = hook.schemaLoader) === null || _d === void 0 ? void 0 : _d.call(hook, controllerMeta, promptMeta));
|
|
276
|
+
if (schema) {
|
|
277
|
+
args.push(schema);
|
|
278
|
+
break;
|
|
279
|
+
}
|
|
280
|
+
}
|
|
281
|
+
}
|
|
282
|
+
const self = this;
|
|
283
|
+
const handler = async (...args) => {
|
|
284
|
+
const eggObj = await self.eggContainerFactory.getOrCreateEggObject(controllerProto, controllerProto.name);
|
|
285
|
+
const realObj = eggObj.obj;
|
|
286
|
+
const realMethod = realObj[promptMeta.name];
|
|
287
|
+
let newArgs = [];
|
|
288
|
+
if (schema && promptMeta.detail) {
|
|
289
|
+
// 如果有 schema 则证明入参第一个就是 schema
|
|
290
|
+
newArgs[promptMeta.detail.index] = args[0];
|
|
291
|
+
// 如果有 schema 则证明入参第二个就是 extra
|
|
292
|
+
if (promptMeta.extra) {
|
|
293
|
+
newArgs[promptMeta.extra] = args[1];
|
|
294
|
+
}
|
|
295
|
+
}
|
|
296
|
+
else if (promptMeta.extra) {
|
|
297
|
+
// 无 schema, 那么入参第一个就是 extra
|
|
298
|
+
newArgs[promptMeta.extra] = args[0];
|
|
299
|
+
}
|
|
300
|
+
newArgs = [...newArgs, ...args];
|
|
301
|
+
return Reflect.apply(realMethod, realObj, newArgs);
|
|
302
|
+
};
|
|
303
|
+
args.push(handler);
|
|
304
|
+
this.mcpServer.prompt(...args);
|
|
305
|
+
}
|
|
306
|
+
async mcpToolRegister(controllerProto, toolMeta) {
|
|
307
|
+
var _a, _b, _c, _d, _e;
|
|
308
|
+
const controllerMeta = controllerProto.getMetaData(tegg_1.CONTROLLER_META_DATA);
|
|
309
|
+
const args = [(_a = toolMeta.mcpName) !== null && _a !== void 0 ? _a : toolMeta.name];
|
|
310
|
+
if (toolMeta.description) {
|
|
311
|
+
args.push(toolMeta.description);
|
|
312
|
+
}
|
|
313
|
+
let schema;
|
|
314
|
+
if ((_b = toolMeta.detail) === null || _b === void 0 ? void 0 : _b.argsSchema) {
|
|
315
|
+
schema = (_c = toolMeta.detail) === null || _c === void 0 ? void 0 : _c.argsSchema;
|
|
316
|
+
args.push((_d = toolMeta.detail) === null || _d === void 0 ? void 0 : _d.argsSchema);
|
|
317
|
+
}
|
|
318
|
+
else if (MCPControllerRegister.hooks.length > 0) {
|
|
319
|
+
for (const hook of MCPControllerRegister.hooks) {
|
|
320
|
+
schema = await ((_e = hook.schemaLoader) === null || _e === void 0 ? void 0 : _e.call(hook, controllerMeta, toolMeta));
|
|
321
|
+
if (schema) {
|
|
322
|
+
args.push(schema);
|
|
323
|
+
break;
|
|
324
|
+
}
|
|
325
|
+
}
|
|
326
|
+
}
|
|
327
|
+
const self = this;
|
|
328
|
+
const handler = async (...args) => {
|
|
329
|
+
const eggObj = await self.eggContainerFactory.getOrCreateEggObject(controllerProto, controllerProto.name);
|
|
330
|
+
const realObj = eggObj.obj;
|
|
331
|
+
const realMethod = realObj[toolMeta.name];
|
|
332
|
+
let newArgs = [];
|
|
333
|
+
if (schema && toolMeta.detail) {
|
|
334
|
+
// 如果有 schema 则证明入参第一个就是 schema
|
|
335
|
+
newArgs[toolMeta.detail.index] = args[0];
|
|
336
|
+
// 如果有 schema 则证明入参第二个就是 extra
|
|
337
|
+
if (toolMeta.extra) {
|
|
338
|
+
newArgs[toolMeta.extra] = args[1];
|
|
339
|
+
}
|
|
340
|
+
}
|
|
341
|
+
else if (toolMeta.extra) {
|
|
342
|
+
// 无 schema, 那么入参第一个就是 extra
|
|
343
|
+
newArgs[toolMeta.extra] = args[0];
|
|
344
|
+
}
|
|
345
|
+
newArgs = [...newArgs, ...args];
|
|
346
|
+
return Reflect.apply(realMethod, realObj, newArgs);
|
|
347
|
+
};
|
|
348
|
+
args.push(handler);
|
|
349
|
+
this.mcpServer.tool(...args);
|
|
350
|
+
}
|
|
351
|
+
async mcpResourceRegister(controllerProto, resourceMeta) {
|
|
352
|
+
var _a;
|
|
353
|
+
const args = [(_a = resourceMeta.mcpName) !== null && _a !== void 0 ? _a : resourceMeta.name];
|
|
354
|
+
if (resourceMeta.uri) {
|
|
355
|
+
args.push(resourceMeta.uri);
|
|
356
|
+
}
|
|
357
|
+
if (resourceMeta.template) {
|
|
358
|
+
const template = resourceMeta.template;
|
|
359
|
+
args.push(template);
|
|
360
|
+
}
|
|
361
|
+
if (resourceMeta.metadata) {
|
|
362
|
+
args.push(resourceMeta.metadata);
|
|
363
|
+
}
|
|
364
|
+
const self = this;
|
|
365
|
+
const handler = async (...args) => {
|
|
366
|
+
const eggObj = await self.eggContainerFactory.getOrCreateEggObject(controllerProto, controllerProto.name);
|
|
367
|
+
const realObj = eggObj.obj;
|
|
368
|
+
const realMethod = realObj[resourceMeta.name];
|
|
369
|
+
return Reflect.apply(realMethod, realObj, args);
|
|
370
|
+
};
|
|
371
|
+
args.push(handler);
|
|
372
|
+
this.mcpServer.resource(...args);
|
|
373
|
+
}
|
|
374
|
+
async register() {
|
|
375
|
+
var _a, _b;
|
|
376
|
+
const promptMap = new Map();
|
|
377
|
+
const resourceMap = new Map();
|
|
378
|
+
const toolMap = new Map();
|
|
379
|
+
for (const proto of this.controllerProtos) {
|
|
380
|
+
if (this.registeredControllerProtos.includes(proto)) {
|
|
381
|
+
continue;
|
|
382
|
+
}
|
|
383
|
+
const metadata = proto.getMetaData(tegg_1.CONTROLLER_META_DATA);
|
|
384
|
+
for (const prompt of metadata.prompts) {
|
|
385
|
+
promptMap.set(prompt, proto);
|
|
386
|
+
}
|
|
387
|
+
for (const resource of metadata.resources) {
|
|
388
|
+
resourceMap.set(resource, proto);
|
|
389
|
+
}
|
|
390
|
+
for (const tool of metadata.tools) {
|
|
391
|
+
toolMap.set(tool, proto);
|
|
392
|
+
}
|
|
393
|
+
this.registeredControllerProtos.push(proto);
|
|
394
|
+
}
|
|
395
|
+
if (!this.mcpServer) {
|
|
396
|
+
this.mcpServer = new mcp_js_1.McpServer({
|
|
397
|
+
name: (_a = this.controllerMeta.name) !== null && _a !== void 0 ? _a : `chair-mcp-${this.app.name}-server`,
|
|
398
|
+
version: (_b = this.controllerMeta.version) !== null && _b !== void 0 ? _b : '1.0.0',
|
|
399
|
+
}, { capabilities: { logging: {} } });
|
|
400
|
+
this.mcpStreamServerInit();
|
|
401
|
+
this.mcpServerInit();
|
|
402
|
+
this.mcpServerRegister();
|
|
403
|
+
}
|
|
404
|
+
for (const [prompt, controllerProto] of promptMap.entries()) {
|
|
405
|
+
await this.mcpPromptRegister(controllerProto, prompt);
|
|
406
|
+
}
|
|
407
|
+
for (const [tool, controllerProto] of toolMap.entries()) {
|
|
408
|
+
await this.mcpToolRegister(controllerProto, tool);
|
|
409
|
+
}
|
|
410
|
+
for (const [resource, controllerProto] of resourceMap.entries()) {
|
|
411
|
+
await this.mcpResourceRegister(controllerProto, resource);
|
|
412
|
+
}
|
|
413
|
+
}
|
|
414
|
+
}
|
|
415
|
+
exports.MCPControllerRegister = MCPControllerRegister;
|
|
416
|
+
MCPControllerRegister.hooks = [];
|
|
417
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiTUNQQ29udHJvbGxlclJlZ2lzdGVyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiTUNQQ29udHJvbGxlclJlZ2lzdGVyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7OztBQUVBLDhEQUFpQztBQUdqQyxzQ0FRcUI7QUFJckIsb0VBQTZFO0FBQzdFLDBGQUFtRztBQUNuRyxvRUFBb0U7QUFDcEUsMkdBQXFHO0FBQ3JHLGlFQUF5RjtBQUN6RixnREFBZ0Q7QUFFaEQsd0RBQWtDO0FBQ2xDLGdFQUF1QztBQWtCdkMsTUFBYSxxQkFBcUI7SUFjaEMsTUFBTSxDQUFDLE1BQU0sQ0FBQyxLQUFtQixFQUFFLGNBQWtDLEVBQUUsR0FBZ0I7UUFDckYsSUFBQSxxQkFBTSxFQUFDLGNBQWMsQ0FBQyxJQUFJLEtBQUsscUJBQWMsQ0FBQyxHQUFHLEVBQUUsaUNBQWlDLENBQUMsQ0FBQztRQUN0RixJQUFJLENBQUMscUJBQXFCLENBQUMsUUFBUSxFQUFFLENBQUM7WUFDcEMscUJBQXFCLENBQUMsUUFBUSxHQUFHLElBQUkscUJBQXFCLENBQUMsS0FBSyxFQUFFLGNBQW1DLEVBQUUsR0FBRyxDQUFDLENBQUM7UUFDOUcsQ0FBQztRQUNELHFCQUFxQixDQUFDLFFBQVEsQ0FBQyxnQkFBZ0IsQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUM7UUFDNUQsT0FBTyxxQkFBcUIsQ0FBQyxRQUFRLENBQUM7SUFDeEMsQ0FBQztJQUVELFlBQVksTUFBb0IsRUFBRSxjQUFpQyxFQUFFLEdBQWdCO1FBbEI3RSxxQkFBZ0IsR0FBbUIsRUFBRSxDQUFDO1FBQ3RDLCtCQUEwQixHQUFtQixFQUFFLENBQUM7UUFDeEQsZUFBVSxHQUF1QyxFQUFFLENBQUM7UUFDcEQsbUJBQWMsR0FBRyxJQUFJLEdBQUcsRUFBK0QsQ0FBQztRQUd4RixxQkFBZ0IsR0FBa0QsRUFBRSxDQUFDO1FBYW5FLElBQUksQ0FBQyxHQUFHLEdBQUcsR0FBRyxDQUFDO1FBQ2YsSUFBSSxDQUFDLG1C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
|
package/package.json
CHANGED
|
@@ -7,7 +7,7 @@
|
|
|
7
7
|
"tegg"
|
|
8
8
|
]
|
|
9
9
|
},
|
|
10
|
-
"version": "3.
|
|
10
|
+
"version": "3.55.0",
|
|
11
11
|
"description": "controller decorator for egg",
|
|
12
12
|
"keywords": [
|
|
13
13
|
"egg",
|
|
@@ -46,23 +46,27 @@
|
|
|
46
46
|
"node": ">=14.0.0"
|
|
47
47
|
},
|
|
48
48
|
"dependencies": {
|
|
49
|
-
"@eggjs/egg-module-common": "^3.
|
|
49
|
+
"@eggjs/egg-module-common": "^3.55.0",
|
|
50
|
+
"@eggjs/mcp-proxy": "^3.55.0",
|
|
50
51
|
"@eggjs/router": "^2.0.1",
|
|
51
|
-
"@eggjs/tegg": "^3.
|
|
52
|
-
"@eggjs/tegg-common-util": "^3.
|
|
53
|
-
"@eggjs/tegg-loader": "^3.
|
|
54
|
-
"@eggjs/tegg-metadata": "^3.
|
|
55
|
-
"@eggjs/tegg-runtime": "^3.
|
|
52
|
+
"@eggjs/tegg": "^3.55.0",
|
|
53
|
+
"@eggjs/tegg-common-util": "^3.55.0",
|
|
54
|
+
"@eggjs/tegg-loader": "^3.55.0",
|
|
55
|
+
"@eggjs/tegg-metadata": "^3.55.0",
|
|
56
|
+
"@eggjs/tegg-runtime": "^3.55.0",
|
|
57
|
+
"@modelcontextprotocol/sdk": "^1.10.0",
|
|
58
|
+
"content-type": "^1.0.5",
|
|
56
59
|
"egg-errors": "^2.3.0",
|
|
57
60
|
"globby": "^10.0.2",
|
|
58
61
|
"koa-compose": "^3.2.1",
|
|
59
62
|
"path-to-regexp": "^1.8.0",
|
|
63
|
+
"raw-body": "^2.5.2",
|
|
60
64
|
"sdk-base": "^4.2.0"
|
|
61
65
|
},
|
|
62
66
|
"devDependencies": {
|
|
63
|
-
"@eggjs/module-test-util": "^3.
|
|
64
|
-
"@eggjs/tegg-config": "^3.
|
|
65
|
-
"@eggjs/tegg-plugin": "^3.
|
|
67
|
+
"@eggjs/module-test-util": "^3.55.0",
|
|
68
|
+
"@eggjs/tegg-config": "^3.55.0",
|
|
69
|
+
"@eggjs/tegg-plugin": "^3.55.0",
|
|
66
70
|
"@types/mocha": "^10.0.1",
|
|
67
71
|
"@types/node": "^20.2.4",
|
|
68
72
|
"cross-env": "^7.0.3",
|
|
@@ -76,5 +80,5 @@
|
|
|
76
80
|
"publishConfig": {
|
|
77
81
|
"access": "public"
|
|
78
82
|
},
|
|
79
|
-
"gitHead": "
|
|
83
|
+
"gitHead": "355924d6771ebad3e2c586ab679ae4110abe3cef"
|
|
80
84
|
}
|
package/typings/index.d.ts
CHANGED
|
@@ -1,11 +1,12 @@
|
|
|
1
1
|
import 'egg';
|
|
2
2
|
import '@eggjs/tegg-plugin';
|
|
3
|
+
import '@eggjs/mcp-proxy';
|
|
3
4
|
import { RootProtoManager } from '../lib/RootProtoManager';
|
|
4
5
|
import { ControllerRegisterFactory } from '../lib/ControllerRegisterFactory';
|
|
5
6
|
import { ControllerMetaBuilderFactory } from '@eggjs/tegg';
|
|
6
7
|
|
|
7
8
|
declare module 'egg' {
|
|
8
|
-
export interface TEggControllerApp {
|
|
9
|
+
export interface TEggControllerApp extends MCPProxyApp {
|
|
9
10
|
rootProtoManager: RootProtoManager;
|
|
10
11
|
controllerRegisterFactory: ControllerRegisterFactory;
|
|
11
12
|
controllerMetaBuilderFactory: typeof ControllerMetaBuilderFactory;
|