@nuxtjs/mcp-toolkit 0.7.0 → 0.8.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/dist/module.d.mts +17 -0
- package/dist/module.json +1 -1
- package/dist/module.mjs +12 -6
- package/dist/runtime/server/mcp/config.d.ts +5 -0
- package/dist/runtime/server/mcp/config.js +6 -1
- package/dist/runtime/server/mcp/providers/cloudflare.js +2 -1
- package/dist/runtime/server/mcp/providers/node.js +76 -6
- package/dist/runtime/server/mcp/providers/types.d.ts +1 -1
- package/dist/runtime/server/mcp/utils.d.ts +4 -1
- package/dist/runtime/server/mcp/utils.js +16 -3
- package/dist/runtime/server/types/hooks.d.ts +4 -0
- package/package.json +16 -12
package/dist/module.d.mts
CHANGED
|
@@ -35,6 +35,23 @@ interface ModuleOptions {
|
|
|
35
35
|
* @default 'mcp'
|
|
36
36
|
*/
|
|
37
37
|
dir?: string;
|
|
38
|
+
/**
|
|
39
|
+
* Enable MCP session management (stateful transport).
|
|
40
|
+
* When enabled, the server assigns session IDs and maintains state across requests,
|
|
41
|
+
* enabling SSE streaming, server-to-client notifications, and resumability.
|
|
42
|
+
*
|
|
43
|
+
* Pass `true` for defaults or an object to configure session behavior.
|
|
44
|
+
* @default false
|
|
45
|
+
* @see https://modelcontextprotocol.io/specification/2025-11-25/basic/transports#session-management
|
|
46
|
+
*/
|
|
47
|
+
sessions?: boolean | {
|
|
48
|
+
enabled?: boolean;
|
|
49
|
+
/**
|
|
50
|
+
* Maximum session duration in milliseconds. Sessions inactive longer than this are cleaned up.
|
|
51
|
+
* @default 1800000 (30 minutes)
|
|
52
|
+
*/
|
|
53
|
+
maxDuration?: number;
|
|
54
|
+
};
|
|
38
55
|
}
|
|
39
56
|
declare const _default: _nuxt_schema.NuxtModule<ModuleOptions, ModuleOptions, false>;
|
|
40
57
|
|
package/dist/module.json
CHANGED
package/dist/module.mjs
CHANGED
|
@@ -72,7 +72,7 @@ function generateDeeplinkUrl(baseUrl, route, ide, serverName) {
|
|
|
72
72
|
}
|
|
73
73
|
|
|
74
74
|
const name = "@nuxtjs/mcp-toolkit";
|
|
75
|
-
const version = "0.
|
|
75
|
+
const version = "0.8.0";
|
|
76
76
|
|
|
77
77
|
const log = logger.withTag("@nuxtjs/mcp-toolkit");
|
|
78
78
|
const { resolve } = createResolver(import.meta.url);
|
|
@@ -86,11 +86,15 @@ const module$1 = defineNuxtModule({
|
|
|
86
86
|
},
|
|
87
87
|
defaults: defaultMcpConfig,
|
|
88
88
|
async setup(options, nuxt) {
|
|
89
|
-
|
|
89
|
+
const nitroOptions = nuxt.options.nitro;
|
|
90
|
+
if (nitroOptions?.static || nuxt.options._generate) {
|
|
90
91
|
log.warn("@nuxtjs/mcp-toolkit is not compatible with `nuxt generate` as it needs a server to run.");
|
|
91
92
|
return;
|
|
92
93
|
}
|
|
93
94
|
const resolver = createResolver(import.meta.url);
|
|
95
|
+
if (typeof options.sessions === "boolean") {
|
|
96
|
+
options.sessions = { enabled: options.sessions };
|
|
97
|
+
}
|
|
94
98
|
const mcpConfig = getMcpConfig(options);
|
|
95
99
|
if (!options.enabled) {
|
|
96
100
|
return;
|
|
@@ -163,10 +167,12 @@ const module$1 = defineNuxtModule({
|
|
|
163
167
|
path: resolver.resolve("runtime/server/types.server.d.ts")
|
|
164
168
|
});
|
|
165
169
|
});
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
+
if (nitroOptions) {
|
|
171
|
+
nitroOptions.typescript ??= {};
|
|
172
|
+
nitroOptions.typescript.tsConfig ??= {};
|
|
173
|
+
nitroOptions.typescript.tsConfig.include ??= [];
|
|
174
|
+
nitroOptions.typescript.tsConfig.include.push(resolver.resolve("runtime/server/types.server.d.ts"));
|
|
175
|
+
}
|
|
170
176
|
let isCloudflare = false;
|
|
171
177
|
if (!nuxt.options.dev) {
|
|
172
178
|
nuxt.hook("nitro:config", (nitroConfig) => {
|
|
@@ -1,3 +1,7 @@
|
|
|
1
|
+
export interface McpSessionsConfig {
|
|
2
|
+
enabled: boolean;
|
|
3
|
+
maxDuration: number;
|
|
4
|
+
}
|
|
1
5
|
export interface McpConfig {
|
|
2
6
|
enabled: boolean;
|
|
3
7
|
route: string;
|
|
@@ -5,6 +9,7 @@ export interface McpConfig {
|
|
|
5
9
|
name: string;
|
|
6
10
|
version: string;
|
|
7
11
|
dir: string;
|
|
12
|
+
sessions: McpSessionsConfig;
|
|
8
13
|
}
|
|
9
14
|
export declare const defaultMcpConfig: McpConfig;
|
|
10
15
|
export declare function getMcpConfig(partial?: Partial<McpConfig>): McpConfig;
|
|
@@ -5,7 +5,12 @@ export const defaultMcpConfig = {
|
|
|
5
5
|
browserRedirect: "/",
|
|
6
6
|
name: "",
|
|
7
7
|
version: "1.0.0",
|
|
8
|
-
dir: "mcp"
|
|
8
|
+
dir: "mcp",
|
|
9
|
+
sessions: {
|
|
10
|
+
enabled: false,
|
|
11
|
+
maxDuration: 30 * 60 * 1e3
|
|
12
|
+
// 30 minutes
|
|
13
|
+
}
|
|
9
14
|
};
|
|
10
15
|
export function getMcpConfig(partial) {
|
|
11
16
|
return defu(partial, defaultMcpConfig);
|
|
@@ -6,7 +6,8 @@ const fallbackCtx = {
|
|
|
6
6
|
passThroughOnException: () => {
|
|
7
7
|
}
|
|
8
8
|
};
|
|
9
|
-
export default createMcpTransportHandler(async (
|
|
9
|
+
export default createMcpTransportHandler(async (createServer, event) => {
|
|
10
|
+
const server = createServer();
|
|
10
11
|
const { createMcpHandler } = await import("agents/mcp");
|
|
11
12
|
const handler = createMcpHandler(server, {
|
|
12
13
|
route: ""
|
|
@@ -1,13 +1,83 @@
|
|
|
1
1
|
import { StreamableHTTPServerTransport } from "@modelcontextprotocol/sdk/server/streamableHttp.js";
|
|
2
|
-
import {
|
|
2
|
+
import { randomUUID } from "uncrypto";
|
|
3
|
+
import { readBody, getHeader, getMethod } from "h3";
|
|
4
|
+
import config from "#nuxt-mcp-toolkit/config.mjs";
|
|
3
5
|
import { createMcpTransportHandler } from "./types.js";
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
6
|
+
const sessions = /* @__PURE__ */ new Map();
|
|
7
|
+
let cleanupInterval = null;
|
|
8
|
+
function ensureCleanup(maxDuration) {
|
|
9
|
+
if (cleanupInterval) return;
|
|
10
|
+
cleanupInterval = setInterval(() => {
|
|
11
|
+
const now = Date.now();
|
|
12
|
+
for (const [id, session] of sessions) {
|
|
13
|
+
if (now - session.lastAccessed > maxDuration) {
|
|
14
|
+
session.transport.close();
|
|
15
|
+
session.server.close();
|
|
16
|
+
sessions.delete(id);
|
|
17
|
+
}
|
|
18
|
+
}
|
|
19
|
+
if (sessions.size === 0 && cleanupInterval) {
|
|
20
|
+
clearInterval(cleanupInterval);
|
|
21
|
+
cleanupInterval = null;
|
|
22
|
+
}
|
|
23
|
+
}, 6e4);
|
|
24
|
+
}
|
|
25
|
+
export default createMcpTransportHandler(async (createServer, event) => {
|
|
26
|
+
const sessionsConfig = config.sessions;
|
|
27
|
+
const sessionsEnabled = sessionsConfig?.enabled ?? false;
|
|
28
|
+
if (!sessionsEnabled) {
|
|
29
|
+
const server2 = createServer();
|
|
30
|
+
const transport2 = new StreamableHTTPServerTransport({ sessionIdGenerator: void 0 });
|
|
31
|
+
event.node.res.on("close", () => {
|
|
32
|
+
transport2.close();
|
|
33
|
+
server2.close();
|
|
34
|
+
});
|
|
35
|
+
await server2.connect(transport2);
|
|
36
|
+
const body2 = await readBody(event);
|
|
37
|
+
await transport2.handleRequest(event.node.req, event.node.res, body2);
|
|
38
|
+
return;
|
|
39
|
+
}
|
|
40
|
+
const maxDuration = sessionsConfig?.maxDuration ?? 30 * 60 * 1e3;
|
|
41
|
+
const method = getMethod(event);
|
|
42
|
+
const sessionId = getHeader(event, "mcp-session-id");
|
|
43
|
+
if (sessionId) {
|
|
44
|
+
const session = sessions.get(sessionId);
|
|
45
|
+
if (!session) {
|
|
46
|
+
event.node.res.writeHead(404, { "Content-Type": "application/json" });
|
|
47
|
+
event.node.res.end(JSON.stringify({
|
|
48
|
+
jsonrpc: "2.0",
|
|
49
|
+
error: { code: -32001, message: "Session not found" },
|
|
50
|
+
id: null
|
|
51
|
+
}));
|
|
52
|
+
return;
|
|
53
|
+
}
|
|
54
|
+
session.lastAccessed = Date.now();
|
|
55
|
+
const body2 = method === "POST" ? await readBody(event) : void 0;
|
|
56
|
+
await session.transport.handleRequest(event.node.req, event.node.res, body2);
|
|
57
|
+
return;
|
|
58
|
+
}
|
|
59
|
+
const server = createServer();
|
|
60
|
+
let sessionStored = false;
|
|
61
|
+
const transport = new StreamableHTTPServerTransport({
|
|
62
|
+
sessionIdGenerator: () => randomUUID(),
|
|
63
|
+
onsessioninitialized: (id) => {
|
|
64
|
+
sessionStored = true;
|
|
65
|
+
sessions.set(id, { server, transport, lastAccessed: Date.now() });
|
|
66
|
+
ensureCleanup(maxDuration);
|
|
67
|
+
}
|
|
9
68
|
});
|
|
69
|
+
transport.onclose = () => {
|
|
70
|
+
const sid = transport.sessionId;
|
|
71
|
+
if (sid && sessions.has(sid)) {
|
|
72
|
+
sessions.delete(sid);
|
|
73
|
+
}
|
|
74
|
+
server.close();
|
|
75
|
+
};
|
|
10
76
|
await server.connect(transport);
|
|
11
77
|
const body = await readBody(event);
|
|
12
78
|
await transport.handleRequest(event.node.req, event.node.res, body);
|
|
79
|
+
if (!sessionStored) {
|
|
80
|
+
transport.close();
|
|
81
|
+
server.close();
|
|
82
|
+
}
|
|
13
83
|
});
|
|
@@ -1,4 +1,4 @@
|
|
|
1
1
|
import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js';
|
|
2
2
|
import type { H3Event } from 'h3';
|
|
3
|
-
export type McpTransportHandler = (
|
|
3
|
+
export type McpTransportHandler = (createServer: () => McpServer, event: H3Event) => Promise<Response | void> | Response | void;
|
|
4
4
|
export declare const createMcpTransportHandler: (handler: McpTransportHandler) => McpTransportHandler;
|
|
@@ -1,6 +1,9 @@
|
|
|
1
1
|
import { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js';
|
|
2
2
|
import type { H3Event } from 'h3';
|
|
3
|
-
import type {
|
|
3
|
+
import type { McpMiddleware } from './definitions/handlers.js';
|
|
4
|
+
import type { McpPromptDefinition } from './definitions/prompts.js';
|
|
5
|
+
import type { McpResourceDefinition } from './definitions/resources.js';
|
|
6
|
+
import type { McpToolDefinition } from './definitions/tools.js';
|
|
4
7
|
export type { McpTransportHandler } from './providers/types.js';
|
|
5
8
|
export { createMcpTransportHandler } from './providers/types.js';
|
|
6
9
|
export interface ResolvedMcpConfig {
|
|
@@ -1,11 +1,24 @@
|
|
|
1
1
|
import { McpServer } from "@modelcontextprotocol/sdk/server/mcp.js";
|
|
2
2
|
import { sendRedirect, getHeader, defineEventHandler } from "h3";
|
|
3
|
-
import {
|
|
3
|
+
import { registerPromptFromDefinition } from "./definitions/prompts.js";
|
|
4
|
+
import { registerResourceFromDefinition } from "./definitions/resources.js";
|
|
5
|
+
import { registerToolFromDefinition } from "./definitions/tools.js";
|
|
4
6
|
import handleMcpRequest from "#nuxt-mcp-toolkit/transport.mjs";
|
|
5
7
|
export { createMcpTransportHandler } from "./providers/types.js";
|
|
6
8
|
function resolveConfig(config, event) {
|
|
7
9
|
return typeof config === "function" ? config(event) : config;
|
|
8
10
|
}
|
|
11
|
+
function registerEmptyDefinitionFallbacks(server, config) {
|
|
12
|
+
if (!config.tools?.length) {
|
|
13
|
+
server.registerTool("__init__", {}, async () => ({ content: [] })).remove();
|
|
14
|
+
}
|
|
15
|
+
if (!config.resources?.length) {
|
|
16
|
+
server.registerResource("__init__", "noop://init", {}, async () => ({ contents: [] })).remove();
|
|
17
|
+
}
|
|
18
|
+
if (!config.prompts?.length) {
|
|
19
|
+
server.registerPrompt("__init__", {}, async () => ({ messages: [] })).remove();
|
|
20
|
+
}
|
|
21
|
+
}
|
|
9
22
|
export function createMcpServer(config) {
|
|
10
23
|
const server = new McpServer({
|
|
11
24
|
name: config.name,
|
|
@@ -20,6 +33,7 @@ export function createMcpServer(config) {
|
|
|
20
33
|
for (const prompt of config.prompts || []) {
|
|
21
34
|
registerPromptFromDefinition(server, prompt);
|
|
22
35
|
}
|
|
36
|
+
registerEmptyDefinitionFallbacks(server, config);
|
|
23
37
|
return server;
|
|
24
38
|
}
|
|
25
39
|
export function createMcpHandler(config) {
|
|
@@ -29,8 +43,7 @@ export function createMcpHandler(config) {
|
|
|
29
43
|
return sendRedirect(event, resolvedConfig.browserRedirect);
|
|
30
44
|
}
|
|
31
45
|
const handler = async () => {
|
|
32
|
-
|
|
33
|
-
return handleMcpRequest(server, event);
|
|
46
|
+
return handleMcpRequest(() => createMcpServer(resolvedConfig), event);
|
|
34
47
|
};
|
|
35
48
|
if (resolvedConfig.middleware) {
|
|
36
49
|
let nextCalled = false;
|
|
@@ -3,6 +3,10 @@ declare module '@nuxt/schema' {
|
|
|
3
3
|
/**
|
|
4
4
|
* Add additional directories to scan for MCP definition files (tools, resources, prompts, handlers).
|
|
5
5
|
* @param paths - Object containing arrays of directory paths for each definition type.
|
|
6
|
+
* @param paths.tools - Array of directory paths to scan for tool definitions.
|
|
7
|
+
* @param paths.resources - Array of directory paths to scan for resource definitions.
|
|
8
|
+
* @param paths.prompts - Array of directory paths to scan for prompt definitions.
|
|
9
|
+
* @param paths.handlers - Array of directory paths to scan for handler definitions.
|
|
6
10
|
* @returns void | Promise<void>
|
|
7
11
|
*/
|
|
8
12
|
'mcp:definitions:paths': (paths: {
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@nuxtjs/mcp-toolkit",
|
|
3
|
-
"version": "0.
|
|
3
|
+
"version": "0.8.0",
|
|
4
4
|
"description": "Create MCP servers directly in your Nuxt application. Define tools, resources, and prompts with a simple and intuitive API.",
|
|
5
5
|
"repository": {
|
|
6
6
|
"type": "git",
|
|
@@ -39,20 +39,24 @@
|
|
|
39
39
|
"dist"
|
|
40
40
|
],
|
|
41
41
|
"dependencies": {
|
|
42
|
-
"@modelcontextprotocol/sdk": "^1.
|
|
43
|
-
"@nuxt/kit": "^4.
|
|
42
|
+
"@modelcontextprotocol/sdk": "^1.27.1",
|
|
43
|
+
"@nuxt/kit": "^4.4.2",
|
|
44
44
|
"defu": "^6.1.4",
|
|
45
45
|
"ms": "^2.1.3",
|
|
46
46
|
"pathe": "^2.0.3",
|
|
47
|
-
"satori": "^0.
|
|
47
|
+
"satori": "^0.25.0",
|
|
48
48
|
"scule": "^1.3.0",
|
|
49
49
|
"tinyglobby": "^0.2.15"
|
|
50
50
|
},
|
|
51
51
|
"peerDependencies": {
|
|
52
|
+
"h3": "^1.15.6",
|
|
52
53
|
"zod": "^4.1.13",
|
|
53
|
-
"agents": ">=0.
|
|
54
|
+
"agents": ">=0.7.6"
|
|
54
55
|
},
|
|
55
56
|
"peerDependenciesMeta": {
|
|
57
|
+
"h3": {
|
|
58
|
+
"optional": false
|
|
59
|
+
},
|
|
56
60
|
"zod": {
|
|
57
61
|
"optional": false
|
|
58
62
|
},
|
|
@@ -61,18 +65,18 @@
|
|
|
61
65
|
}
|
|
62
66
|
},
|
|
63
67
|
"devDependencies": {
|
|
64
|
-
"@nuxt/devtools": "^3.2.
|
|
65
|
-
"@nuxt/eslint-config": "^1.15.
|
|
68
|
+
"@nuxt/devtools": "^3.2.3",
|
|
69
|
+
"@nuxt/eslint-config": "^1.15.2",
|
|
66
70
|
"@nuxt/module-builder": "^1.0.2",
|
|
67
|
-
"@nuxt/schema": "^4.
|
|
71
|
+
"@nuxt/schema": "^4.4.2",
|
|
68
72
|
"@nuxt/test-utils": "^4.0.0",
|
|
69
73
|
"@types/node": "latest",
|
|
70
74
|
"changelogen": "^0.6.2",
|
|
71
|
-
"eslint": "^9.39.
|
|
72
|
-
"nuxt": "^4.
|
|
75
|
+
"eslint": "^9.39.4",
|
|
76
|
+
"nuxt": "^4.4.2",
|
|
73
77
|
"typescript": "~5.9.3",
|
|
74
|
-
"vitest": "^4.0
|
|
75
|
-
"vue-tsc": "^3.2.
|
|
78
|
+
"vitest": "^4.1.0",
|
|
79
|
+
"vue-tsc": "^3.2.5"
|
|
76
80
|
},
|
|
77
81
|
"publishConfig": {
|
|
78
82
|
"access": "public"
|