@contextvm/mcp-sdk 1.27.1-contextvm.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/LICENSE +21 -0
- package/README.md +141 -0
- package/dist/cjs/client/index.d.ts +588 -0
- package/dist/cjs/client/index.d.ts.map +1 -0
- package/dist/cjs/client/index.js +629 -0
- package/dist/cjs/client/index.js.map +1 -0
- package/dist/cjs/client/stdio.d.ts +77 -0
- package/dist/cjs/client/stdio.d.ts.map +1 -0
- package/dist/cjs/client/stdio.js +199 -0
- package/dist/cjs/client/stdio.js.map +1 -0
- package/dist/cjs/experimental/index.d.ts +13 -0
- package/dist/cjs/experimental/index.d.ts.map +1 -0
- package/dist/cjs/experimental/index.js +29 -0
- package/dist/cjs/experimental/index.js.map +1 -0
- package/dist/cjs/experimental/tasks/client.d.ts +121 -0
- package/dist/cjs/experimental/tasks/client.d.ts.map +1 -0
- package/dist/cjs/experimental/tasks/client.js +188 -0
- package/dist/cjs/experimental/tasks/client.js.map +1 -0
- package/dist/cjs/experimental/tasks/helpers.d.ts +47 -0
- package/dist/cjs/experimental/tasks/helpers.d.ts.map +1 -0
- package/dist/cjs/experimental/tasks/helpers.js +68 -0
- package/dist/cjs/experimental/tasks/helpers.js.map +1 -0
- package/dist/cjs/experimental/tasks/index.d.ts +16 -0
- package/dist/cjs/experimental/tasks/index.d.ts.map +1 -0
- package/dist/cjs/experimental/tasks/index.js +39 -0
- package/dist/cjs/experimental/tasks/index.js.map +1 -0
- package/dist/cjs/experimental/tasks/interfaces.d.ts +232 -0
- package/dist/cjs/experimental/tasks/interfaces.d.ts.map +1 -0
- package/dist/cjs/experimental/tasks/interfaces.js +19 -0
- package/dist/cjs/experimental/tasks/interfaces.js.map +1 -0
- package/dist/cjs/experimental/tasks/mcp-server.d.ts +77 -0
- package/dist/cjs/experimental/tasks/mcp-server.d.ts.map +1 -0
- package/dist/cjs/experimental/tasks/mcp-server.js +36 -0
- package/dist/cjs/experimental/tasks/mcp-server.js.map +1 -0
- package/dist/cjs/experimental/tasks/server.d.ts +170 -0
- package/dist/cjs/experimental/tasks/server.d.ts.map +1 -0
- package/dist/cjs/experimental/tasks/server.js +250 -0
- package/dist/cjs/experimental/tasks/server.js.map +1 -0
- package/dist/cjs/experimental/tasks/stores/in-memory.d.ts +94 -0
- package/dist/cjs/experimental/tasks/stores/in-memory.d.ts.map +1 -0
- package/dist/cjs/experimental/tasks/stores/in-memory.js +251 -0
- package/dist/cjs/experimental/tasks/stores/in-memory.js.map +1 -0
- package/dist/cjs/experimental/tasks/types.d.ts +10 -0
- package/dist/cjs/experimental/tasks/types.d.ts.map +1 -0
- package/dist/cjs/experimental/tasks/types.js +28 -0
- package/dist/cjs/experimental/tasks/types.js.map +1 -0
- package/dist/cjs/inMemory.d.ts +31 -0
- package/dist/cjs/inMemory.d.ts.map +1 -0
- package/dist/cjs/inMemory.js +51 -0
- package/dist/cjs/inMemory.js.map +1 -0
- package/dist/cjs/package.json +1 -0
- package/dist/cjs/server/completable.d.ts +38 -0
- package/dist/cjs/server/completable.d.ts.map +1 -0
- package/dist/cjs/server/completable.js +48 -0
- package/dist/cjs/server/completable.js.map +1 -0
- package/dist/cjs/server/index.d.ts +196 -0
- package/dist/cjs/server/index.d.ts.map +1 -0
- package/dist/cjs/server/index.js +444 -0
- package/dist/cjs/server/index.js.map +1 -0
- package/dist/cjs/server/mcp.d.ts +364 -0
- package/dist/cjs/server/mcp.d.ts.map +1 -0
- package/dist/cjs/server/mcp.js +918 -0
- package/dist/cjs/server/mcp.js.map +1 -0
- package/dist/cjs/server/stdio.d.ts +28 -0
- package/dist/cjs/server/stdio.d.ts.map +1 -0
- package/dist/cjs/server/stdio.js +82 -0
- package/dist/cjs/server/stdio.js.map +1 -0
- package/dist/cjs/server/zod-compat.d.ts +84 -0
- package/dist/cjs/server/zod-compat.d.ts.map +1 -0
- package/dist/cjs/server/zod-compat.js +244 -0
- package/dist/cjs/server/zod-compat.js.map +1 -0
- package/dist/cjs/server/zod-json-schema-compat.d.ts +12 -0
- package/dist/cjs/server/zod-json-schema-compat.d.ts.map +1 -0
- package/dist/cjs/server/zod-json-schema-compat.js +79 -0
- package/dist/cjs/server/zod-json-schema-compat.js.map +1 -0
- package/dist/cjs/shared/auth-info.d.ts +32 -0
- package/dist/cjs/shared/auth-info.d.ts.map +1 -0
- package/dist/cjs/shared/auth-info.js +3 -0
- package/dist/cjs/shared/auth-info.js.map +1 -0
- package/dist/cjs/shared/metadataUtils.d.ts +16 -0
- package/dist/cjs/shared/metadataUtils.d.ts.map +1 -0
- package/dist/cjs/shared/metadataUtils.js +25 -0
- package/dist/cjs/shared/metadataUtils.js.map +1 -0
- package/dist/cjs/shared/protocol.d.ts +443 -0
- package/dist/cjs/shared/protocol.d.ts.map +1 -0
- package/dist/cjs/shared/protocol.js +1104 -0
- package/dist/cjs/shared/protocol.js.map +1 -0
- package/dist/cjs/shared/responseMessage.d.ts +45 -0
- package/dist/cjs/shared/responseMessage.d.ts.map +1 -0
- package/dist/cjs/shared/responseMessage.js +23 -0
- package/dist/cjs/shared/responseMessage.js.map +1 -0
- package/dist/cjs/shared/stdio.d.ts +13 -0
- package/dist/cjs/shared/stdio.d.ts.map +1 -0
- package/dist/cjs/shared/stdio.js +37 -0
- package/dist/cjs/shared/stdio.js.map +1 -0
- package/dist/cjs/shared/toolNameValidation.d.ts +31 -0
- package/dist/cjs/shared/toolNameValidation.d.ts.map +1 -0
- package/dist/cjs/shared/toolNameValidation.js +97 -0
- package/dist/cjs/shared/toolNameValidation.js.map +1 -0
- package/dist/cjs/shared/transport.d.ts +89 -0
- package/dist/cjs/shared/transport.d.ts.map +1 -0
- package/dist/cjs/shared/transport.js +43 -0
- package/dist/cjs/shared/transport.js.map +1 -0
- package/dist/cjs/shared/uriTemplate.d.ts +25 -0
- package/dist/cjs/shared/uriTemplate.d.ts.map +1 -0
- package/dist/cjs/shared/uriTemplate.js +243 -0
- package/dist/cjs/shared/uriTemplate.js.map +1 -0
- package/dist/cjs/spec.types.d.ts +2299 -0
- package/dist/cjs/spec.types.d.ts.map +1 -0
- package/dist/cjs/spec.types.js +27 -0
- package/dist/cjs/spec.types.js.map +1 -0
- package/dist/cjs/types.d.ts +8137 -0
- package/dist/cjs/types.d.ts.map +1 -0
- package/dist/cjs/types.js +2092 -0
- package/dist/cjs/types.js.map +1 -0
- package/dist/cjs/validation/ajv-provider.d.ts +53 -0
- package/dist/cjs/validation/ajv-provider.d.ts.map +1 -0
- package/dist/cjs/validation/ajv-provider.js +94 -0
- package/dist/cjs/validation/ajv-provider.js.map +1 -0
- package/dist/cjs/validation/cfworker-provider.d.ts +51 -0
- package/dist/cjs/validation/cfworker-provider.d.ts.map +1 -0
- package/dist/cjs/validation/cfworker-provider.js +69 -0
- package/dist/cjs/validation/cfworker-provider.js.map +1 -0
- package/dist/cjs/validation/index.d.ts +29 -0
- package/dist/cjs/validation/index.d.ts.map +1 -0
- package/dist/cjs/validation/index.js +30 -0
- package/dist/cjs/validation/index.js.map +1 -0
- package/dist/cjs/validation/types.d.ts +65 -0
- package/dist/cjs/validation/types.d.ts.map +1 -0
- package/dist/cjs/validation/types.js +3 -0
- package/dist/cjs/validation/types.js.map +1 -0
- package/dist/esm/client/index.d.ts +588 -0
- package/dist/esm/client/index.d.ts.map +1 -0
- package/dist/esm/client/index.js +624 -0
- package/dist/esm/client/index.js.map +1 -0
- package/dist/esm/client/stdio.d.ts +77 -0
- package/dist/esm/client/stdio.d.ts.map +1 -0
- package/dist/esm/client/stdio.js +191 -0
- package/dist/esm/client/stdio.js.map +1 -0
- package/dist/esm/experimental/index.d.ts +13 -0
- package/dist/esm/experimental/index.d.ts.map +1 -0
- package/dist/esm/experimental/index.js +13 -0
- package/dist/esm/experimental/index.js.map +1 -0
- package/dist/esm/experimental/tasks/client.d.ts +121 -0
- package/dist/esm/experimental/tasks/client.d.ts.map +1 -0
- package/dist/esm/experimental/tasks/client.js +184 -0
- package/dist/esm/experimental/tasks/client.js.map +1 -0
- package/dist/esm/experimental/tasks/helpers.d.ts +47 -0
- package/dist/esm/experimental/tasks/helpers.d.ts.map +1 -0
- package/dist/esm/experimental/tasks/helpers.js +64 -0
- package/dist/esm/experimental/tasks/helpers.js.map +1 -0
- package/dist/esm/experimental/tasks/index.d.ts +16 -0
- package/dist/esm/experimental/tasks/index.d.ts.map +1 -0
- package/dist/esm/experimental/tasks/index.js +20 -0
- package/dist/esm/experimental/tasks/index.js.map +1 -0
- package/dist/esm/experimental/tasks/interfaces.d.ts +232 -0
- package/dist/esm/experimental/tasks/interfaces.d.ts.map +1 -0
- package/dist/esm/experimental/tasks/interfaces.js +16 -0
- package/dist/esm/experimental/tasks/interfaces.js.map +1 -0
- package/dist/esm/experimental/tasks/mcp-server.d.ts +77 -0
- package/dist/esm/experimental/tasks/mcp-server.d.ts.map +1 -0
- package/dist/esm/experimental/tasks/mcp-server.js +32 -0
- package/dist/esm/experimental/tasks/mcp-server.js.map +1 -0
- package/dist/esm/experimental/tasks/server.d.ts +170 -0
- package/dist/esm/experimental/tasks/server.d.ts.map +1 -0
- package/dist/esm/experimental/tasks/server.js +246 -0
- package/dist/esm/experimental/tasks/server.js.map +1 -0
- package/dist/esm/experimental/tasks/stores/in-memory.d.ts +94 -0
- package/dist/esm/experimental/tasks/stores/in-memory.d.ts.map +1 -0
- package/dist/esm/experimental/tasks/stores/in-memory.js +246 -0
- package/dist/esm/experimental/tasks/stores/in-memory.js.map +1 -0
- package/dist/esm/experimental/tasks/types.d.ts +10 -0
- package/dist/esm/experimental/tasks/types.d.ts.map +1 -0
- package/dist/esm/experimental/tasks/types.js +10 -0
- package/dist/esm/experimental/tasks/types.js.map +1 -0
- package/dist/esm/inMemory.d.ts +31 -0
- package/dist/esm/inMemory.d.ts.map +1 -0
- package/dist/esm/inMemory.js +47 -0
- package/dist/esm/inMemory.js.map +1 -0
- package/dist/esm/package.json +1 -0
- package/dist/esm/server/completable.d.ts +38 -0
- package/dist/esm/server/completable.d.ts.map +1 -0
- package/dist/esm/server/completable.js +41 -0
- package/dist/esm/server/completable.js.map +1 -0
- package/dist/esm/server/index.d.ts +196 -0
- package/dist/esm/server/index.d.ts.map +1 -0
- package/dist/esm/server/index.js +440 -0
- package/dist/esm/server/index.js.map +1 -0
- package/dist/esm/server/mcp.d.ts +364 -0
- package/dist/esm/server/mcp.d.ts.map +1 -0
- package/dist/esm/server/mcp.js +913 -0
- package/dist/esm/server/mcp.js.map +1 -0
- package/dist/esm/server/stdio.d.ts +28 -0
- package/dist/esm/server/stdio.d.ts.map +1 -0
- package/dist/esm/server/stdio.js +75 -0
- package/dist/esm/server/stdio.js.map +1 -0
- package/dist/esm/server/zod-compat.d.ts +84 -0
- package/dist/esm/server/zod-compat.d.ts.map +1 -0
- package/dist/esm/server/zod-compat.js +209 -0
- package/dist/esm/server/zod-compat.js.map +1 -0
- package/dist/esm/server/zod-json-schema-compat.d.ts +12 -0
- package/dist/esm/server/zod-json-schema-compat.d.ts.map +1 -0
- package/dist/esm/server/zod-json-schema-compat.js +51 -0
- package/dist/esm/server/zod-json-schema-compat.js.map +1 -0
- package/dist/esm/shared/auth-info.d.ts +32 -0
- package/dist/esm/shared/auth-info.d.ts.map +1 -0
- package/dist/esm/shared/auth-info.js +2 -0
- package/dist/esm/shared/auth-info.js.map +1 -0
- package/dist/esm/shared/metadataUtils.d.ts +16 -0
- package/dist/esm/shared/metadataUtils.d.ts.map +1 -0
- package/dist/esm/shared/metadataUtils.js +22 -0
- package/dist/esm/shared/metadataUtils.js.map +1 -0
- package/dist/esm/shared/protocol.d.ts +443 -0
- package/dist/esm/shared/protocol.d.ts.map +1 -0
- package/dist/esm/shared/protocol.js +1099 -0
- package/dist/esm/shared/protocol.js.map +1 -0
- package/dist/esm/shared/responseMessage.d.ts +45 -0
- package/dist/esm/shared/responseMessage.d.ts.map +1 -0
- package/dist/esm/shared/responseMessage.js +19 -0
- package/dist/esm/shared/responseMessage.js.map +1 -0
- package/dist/esm/shared/stdio.d.ts +13 -0
- package/dist/esm/shared/stdio.d.ts.map +1 -0
- package/dist/esm/shared/stdio.js +31 -0
- package/dist/esm/shared/stdio.js.map +1 -0
- package/dist/esm/shared/toolNameValidation.d.ts +31 -0
- package/dist/esm/shared/toolNameValidation.d.ts.map +1 -0
- package/dist/esm/shared/toolNameValidation.js +92 -0
- package/dist/esm/shared/toolNameValidation.js.map +1 -0
- package/dist/esm/shared/transport.d.ts +89 -0
- package/dist/esm/shared/transport.d.ts.map +1 -0
- package/dist/esm/shared/transport.js +39 -0
- package/dist/esm/shared/transport.js.map +1 -0
- package/dist/esm/shared/uriTemplate.d.ts +25 -0
- package/dist/esm/shared/uriTemplate.d.ts.map +1 -0
- package/dist/esm/shared/uriTemplate.js +239 -0
- package/dist/esm/shared/uriTemplate.js.map +1 -0
- package/dist/esm/spec.types.d.ts +2299 -0
- package/dist/esm/spec.types.d.ts.map +1 -0
- package/dist/esm/spec.types.js +24 -0
- package/dist/esm/spec.types.js.map +1 -0
- package/dist/esm/types.d.ts +8137 -0
- package/dist/esm/types.d.ts.map +1 -0
- package/dist/esm/types.js +2052 -0
- package/dist/esm/types.js.map +1 -0
- package/dist/esm/validation/ajv-provider.d.ts +53 -0
- package/dist/esm/validation/ajv-provider.d.ts.map +1 -0
- package/dist/esm/validation/ajv-provider.js +87 -0
- package/dist/esm/validation/ajv-provider.js.map +1 -0
- package/dist/esm/validation/cfworker-provider.d.ts +51 -0
- package/dist/esm/validation/cfworker-provider.d.ts.map +1 -0
- package/dist/esm/validation/cfworker-provider.js +65 -0
- package/dist/esm/validation/cfworker-provider.js.map +1 -0
- package/dist/esm/validation/index.d.ts +29 -0
- package/dist/esm/validation/index.d.ts.map +1 -0
- package/dist/esm/validation/index.js +29 -0
- package/dist/esm/validation/index.js.map +1 -0
- package/dist/esm/validation/types.d.ts +65 -0
- package/dist/esm/validation/types.d.ts.map +1 -0
- package/dist/esm/validation/types.js +2 -0
- package/dist/esm/validation/types.js.map +1 -0
- package/package.json +124 -0
|
@@ -0,0 +1,913 @@
|
|
|
1
|
+
import { Server } from './index.js';
|
|
2
|
+
import { normalizeObjectSchema, safeParseAsync, getObjectShape, objectFromShape, getParseErrorMessage, getSchemaDescription, isSchemaOptional, getLiteralValue } from './zod-compat.js';
|
|
3
|
+
import { toJsonSchemaCompat } from './zod-json-schema-compat.js';
|
|
4
|
+
import { McpError, ErrorCode, ListResourceTemplatesRequestSchema, ReadResourceRequestSchema, ListToolsRequestSchema, CallToolRequestSchema, ListResourcesRequestSchema, ListPromptsRequestSchema, GetPromptRequestSchema, CompleteRequestSchema, assertCompleteRequestPrompt, assertCompleteRequestResourceTemplate } from '../types.js';
|
|
5
|
+
import { isCompletable, getCompleter } from './completable.js';
|
|
6
|
+
import { UriTemplate } from '../shared/uriTemplate.js';
|
|
7
|
+
import { validateAndWarnToolName } from '../shared/toolNameValidation.js';
|
|
8
|
+
import { ExperimentalMcpServerTasks } from '../experimental/tasks/mcp-server.js';
|
|
9
|
+
import { ZodOptional } from 'zod';
|
|
10
|
+
/**
|
|
11
|
+
* High-level MCP server that provides a simpler API for working with resources, tools, and prompts.
|
|
12
|
+
* For advanced usage (like sending notifications or setting custom request handlers), use the underlying
|
|
13
|
+
* Server instance available via the `server` property.
|
|
14
|
+
*/
|
|
15
|
+
export class McpServer {
|
|
16
|
+
constructor(serverInfo, options) {
|
|
17
|
+
this._registeredResources = {};
|
|
18
|
+
this._registeredResourceTemplates = {};
|
|
19
|
+
this._registeredTools = {};
|
|
20
|
+
this._registeredPrompts = {};
|
|
21
|
+
this._toolHandlersInitialized = false;
|
|
22
|
+
this._completionHandlerInitialized = false;
|
|
23
|
+
this._resourceHandlersInitialized = false;
|
|
24
|
+
this._promptHandlersInitialized = false;
|
|
25
|
+
this.server = new Server(serverInfo, options);
|
|
26
|
+
}
|
|
27
|
+
/**
|
|
28
|
+
* Access experimental features.
|
|
29
|
+
*
|
|
30
|
+
* WARNING: These APIs are experimental and may change without notice.
|
|
31
|
+
*
|
|
32
|
+
* @experimental
|
|
33
|
+
*/
|
|
34
|
+
get experimental() {
|
|
35
|
+
if (!this._experimental) {
|
|
36
|
+
this._experimental = {
|
|
37
|
+
tasks: new ExperimentalMcpServerTasks(this)
|
|
38
|
+
};
|
|
39
|
+
}
|
|
40
|
+
return this._experimental;
|
|
41
|
+
}
|
|
42
|
+
/**
|
|
43
|
+
* Attaches to the given transport, starts it, and starts listening for messages.
|
|
44
|
+
*
|
|
45
|
+
* The `server` object assumes ownership of the Transport, replacing any callbacks that have already been set, and expects that it is the only user of the Transport instance going forward.
|
|
46
|
+
*/
|
|
47
|
+
async connect(transport) {
|
|
48
|
+
return await this.server.connect(transport);
|
|
49
|
+
}
|
|
50
|
+
/**
|
|
51
|
+
* Closes the connection.
|
|
52
|
+
*/
|
|
53
|
+
async close() {
|
|
54
|
+
await this.server.close();
|
|
55
|
+
}
|
|
56
|
+
setToolRequestHandlers() {
|
|
57
|
+
if (this._toolHandlersInitialized) {
|
|
58
|
+
return;
|
|
59
|
+
}
|
|
60
|
+
this.server.assertCanSetRequestHandler(getMethodValue(ListToolsRequestSchema));
|
|
61
|
+
this.server.assertCanSetRequestHandler(getMethodValue(CallToolRequestSchema));
|
|
62
|
+
this.server.registerCapabilities({
|
|
63
|
+
tools: {
|
|
64
|
+
listChanged: true
|
|
65
|
+
}
|
|
66
|
+
});
|
|
67
|
+
this.server.setRequestHandler(ListToolsRequestSchema, () => ({
|
|
68
|
+
tools: Object.entries(this._registeredTools)
|
|
69
|
+
.filter(([, tool]) => tool.enabled)
|
|
70
|
+
.map(([name, tool]) => {
|
|
71
|
+
const toolDefinition = {
|
|
72
|
+
name,
|
|
73
|
+
title: tool.title,
|
|
74
|
+
description: tool.description,
|
|
75
|
+
inputSchema: (() => {
|
|
76
|
+
const obj = normalizeObjectSchema(tool.inputSchema);
|
|
77
|
+
return obj
|
|
78
|
+
? toJsonSchemaCompat(obj, {
|
|
79
|
+
strictUnions: true,
|
|
80
|
+
pipeStrategy: 'input'
|
|
81
|
+
})
|
|
82
|
+
: EMPTY_OBJECT_JSON_SCHEMA;
|
|
83
|
+
})(),
|
|
84
|
+
annotations: tool.annotations,
|
|
85
|
+
execution: tool.execution,
|
|
86
|
+
_meta: tool._meta
|
|
87
|
+
};
|
|
88
|
+
if (tool.outputSchema) {
|
|
89
|
+
const obj = normalizeObjectSchema(tool.outputSchema);
|
|
90
|
+
if (obj) {
|
|
91
|
+
toolDefinition.outputSchema = toJsonSchemaCompat(obj, {
|
|
92
|
+
strictUnions: true,
|
|
93
|
+
pipeStrategy: 'output'
|
|
94
|
+
});
|
|
95
|
+
}
|
|
96
|
+
}
|
|
97
|
+
return toolDefinition;
|
|
98
|
+
})
|
|
99
|
+
}));
|
|
100
|
+
this.server.setRequestHandler(CallToolRequestSchema, async (request, extra) => {
|
|
101
|
+
try {
|
|
102
|
+
const tool = this._registeredTools[request.params.name];
|
|
103
|
+
if (!tool) {
|
|
104
|
+
throw new McpError(ErrorCode.InvalidParams, `Tool ${request.params.name} not found`);
|
|
105
|
+
}
|
|
106
|
+
if (!tool.enabled) {
|
|
107
|
+
throw new McpError(ErrorCode.InvalidParams, `Tool ${request.params.name} disabled`);
|
|
108
|
+
}
|
|
109
|
+
const isTaskRequest = !!request.params.task;
|
|
110
|
+
const taskSupport = tool.execution?.taskSupport;
|
|
111
|
+
const isTaskHandler = 'createTask' in tool.handler;
|
|
112
|
+
// Validate task hint configuration
|
|
113
|
+
if ((taskSupport === 'required' || taskSupport === 'optional') && !isTaskHandler) {
|
|
114
|
+
throw new McpError(ErrorCode.InternalError, `Tool ${request.params.name} has taskSupport '${taskSupport}' but was not registered with registerToolTask`);
|
|
115
|
+
}
|
|
116
|
+
// Handle taskSupport 'required' without task augmentation
|
|
117
|
+
if (taskSupport === 'required' && !isTaskRequest) {
|
|
118
|
+
throw new McpError(ErrorCode.MethodNotFound, `Tool ${request.params.name} requires task augmentation (taskSupport: 'required')`);
|
|
119
|
+
}
|
|
120
|
+
// Handle taskSupport 'optional' without task augmentation - automatic polling
|
|
121
|
+
if (taskSupport === 'optional' && !isTaskRequest && isTaskHandler) {
|
|
122
|
+
return await this.handleAutomaticTaskPolling(tool, request, extra);
|
|
123
|
+
}
|
|
124
|
+
// Normal execution path
|
|
125
|
+
const args = await this.validateToolInput(tool, request.params.arguments, request.params.name);
|
|
126
|
+
const result = await this.executeToolHandler(tool, args, extra);
|
|
127
|
+
// Return CreateTaskResult immediately for task requests
|
|
128
|
+
if (isTaskRequest) {
|
|
129
|
+
return result;
|
|
130
|
+
}
|
|
131
|
+
// Validate output schema for non-task requests
|
|
132
|
+
await this.validateToolOutput(tool, result, request.params.name);
|
|
133
|
+
return result;
|
|
134
|
+
}
|
|
135
|
+
catch (error) {
|
|
136
|
+
if (error instanceof McpError) {
|
|
137
|
+
if (error.code === ErrorCode.UrlElicitationRequired) {
|
|
138
|
+
throw error; // Return the error to the caller without wrapping in CallToolResult
|
|
139
|
+
}
|
|
140
|
+
}
|
|
141
|
+
return this.createToolError(error instanceof Error ? error.message : String(error));
|
|
142
|
+
}
|
|
143
|
+
});
|
|
144
|
+
this._toolHandlersInitialized = true;
|
|
145
|
+
}
|
|
146
|
+
/**
|
|
147
|
+
* Creates a tool error result.
|
|
148
|
+
*
|
|
149
|
+
* @param errorMessage - The error message.
|
|
150
|
+
* @returns The tool error result.
|
|
151
|
+
*/
|
|
152
|
+
createToolError(errorMessage) {
|
|
153
|
+
return {
|
|
154
|
+
content: [
|
|
155
|
+
{
|
|
156
|
+
type: 'text',
|
|
157
|
+
text: errorMessage
|
|
158
|
+
}
|
|
159
|
+
],
|
|
160
|
+
isError: true
|
|
161
|
+
};
|
|
162
|
+
}
|
|
163
|
+
/**
|
|
164
|
+
* Validates tool input arguments against the tool's input schema.
|
|
165
|
+
*/
|
|
166
|
+
async validateToolInput(tool, args, toolName) {
|
|
167
|
+
if (!tool.inputSchema) {
|
|
168
|
+
return undefined;
|
|
169
|
+
}
|
|
170
|
+
// Try to normalize to object schema first (for raw shapes and object schemas)
|
|
171
|
+
// If that fails, use the schema directly (for union/intersection/etc)
|
|
172
|
+
const inputObj = normalizeObjectSchema(tool.inputSchema);
|
|
173
|
+
const schemaToParse = inputObj ?? tool.inputSchema;
|
|
174
|
+
const parseResult = await safeParseAsync(schemaToParse, args);
|
|
175
|
+
if (!parseResult.success) {
|
|
176
|
+
const error = 'error' in parseResult ? parseResult.error : 'Unknown error';
|
|
177
|
+
const errorMessage = getParseErrorMessage(error);
|
|
178
|
+
throw new McpError(ErrorCode.InvalidParams, `Input validation error: Invalid arguments for tool ${toolName}: ${errorMessage}`);
|
|
179
|
+
}
|
|
180
|
+
return parseResult.data;
|
|
181
|
+
}
|
|
182
|
+
/**
|
|
183
|
+
* Validates tool output against the tool's output schema.
|
|
184
|
+
*/
|
|
185
|
+
async validateToolOutput(tool, result, toolName) {
|
|
186
|
+
if (!tool.outputSchema) {
|
|
187
|
+
return;
|
|
188
|
+
}
|
|
189
|
+
// Only validate CallToolResult, not CreateTaskResult
|
|
190
|
+
if (!('content' in result)) {
|
|
191
|
+
return;
|
|
192
|
+
}
|
|
193
|
+
if (result.isError) {
|
|
194
|
+
return;
|
|
195
|
+
}
|
|
196
|
+
if (!result.structuredContent) {
|
|
197
|
+
throw new McpError(ErrorCode.InvalidParams, `Output validation error: Tool ${toolName} has an output schema but no structured content was provided`);
|
|
198
|
+
}
|
|
199
|
+
// if the tool has an output schema, validate structured content
|
|
200
|
+
const outputObj = normalizeObjectSchema(tool.outputSchema);
|
|
201
|
+
const parseResult = await safeParseAsync(outputObj, result.structuredContent);
|
|
202
|
+
if (!parseResult.success) {
|
|
203
|
+
const error = 'error' in parseResult ? parseResult.error : 'Unknown error';
|
|
204
|
+
const errorMessage = getParseErrorMessage(error);
|
|
205
|
+
throw new McpError(ErrorCode.InvalidParams, `Output validation error: Invalid structured content for tool ${toolName}: ${errorMessage}`);
|
|
206
|
+
}
|
|
207
|
+
}
|
|
208
|
+
/**
|
|
209
|
+
* Executes a tool handler (either regular or task-based).
|
|
210
|
+
*/
|
|
211
|
+
async executeToolHandler(tool, args, extra) {
|
|
212
|
+
const handler = tool.handler;
|
|
213
|
+
const isTaskHandler = 'createTask' in handler;
|
|
214
|
+
if (isTaskHandler) {
|
|
215
|
+
if (!extra.taskStore) {
|
|
216
|
+
throw new Error('No task store provided.');
|
|
217
|
+
}
|
|
218
|
+
const taskExtra = { ...extra, taskStore: extra.taskStore };
|
|
219
|
+
if (tool.inputSchema) {
|
|
220
|
+
const typedHandler = handler;
|
|
221
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
222
|
+
return await Promise.resolve(typedHandler.createTask(args, taskExtra));
|
|
223
|
+
}
|
|
224
|
+
else {
|
|
225
|
+
const typedHandler = handler;
|
|
226
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
227
|
+
return await Promise.resolve(typedHandler.createTask(taskExtra));
|
|
228
|
+
}
|
|
229
|
+
}
|
|
230
|
+
if (tool.inputSchema) {
|
|
231
|
+
const typedHandler = handler;
|
|
232
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
233
|
+
return await Promise.resolve(typedHandler(args, extra));
|
|
234
|
+
}
|
|
235
|
+
else {
|
|
236
|
+
const typedHandler = handler;
|
|
237
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
238
|
+
return await Promise.resolve(typedHandler(extra));
|
|
239
|
+
}
|
|
240
|
+
}
|
|
241
|
+
/**
|
|
242
|
+
* Handles automatic task polling for tools with taskSupport 'optional'.
|
|
243
|
+
*/
|
|
244
|
+
async handleAutomaticTaskPolling(tool, request, extra) {
|
|
245
|
+
if (!extra.taskStore) {
|
|
246
|
+
throw new Error('No task store provided for task-capable tool.');
|
|
247
|
+
}
|
|
248
|
+
// Validate input and create task
|
|
249
|
+
const args = await this.validateToolInput(tool, request.params.arguments, request.params.name);
|
|
250
|
+
const handler = tool.handler;
|
|
251
|
+
const taskExtra = { ...extra, taskStore: extra.taskStore };
|
|
252
|
+
const createTaskResult = args // undefined only if tool.inputSchema is undefined
|
|
253
|
+
? await Promise.resolve(handler.createTask(args, taskExtra))
|
|
254
|
+
: // eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
255
|
+
await Promise.resolve(handler.createTask(taskExtra));
|
|
256
|
+
// Poll until completion
|
|
257
|
+
const taskId = createTaskResult.task.taskId;
|
|
258
|
+
let task = createTaskResult.task;
|
|
259
|
+
const pollInterval = task.pollInterval ?? 5000;
|
|
260
|
+
while (task.status !== 'completed' && task.status !== 'failed' && task.status !== 'cancelled') {
|
|
261
|
+
await new Promise(resolve => setTimeout(resolve, pollInterval));
|
|
262
|
+
const updatedTask = await extra.taskStore.getTask(taskId);
|
|
263
|
+
if (!updatedTask) {
|
|
264
|
+
throw new McpError(ErrorCode.InternalError, `Task ${taskId} not found during polling`);
|
|
265
|
+
}
|
|
266
|
+
task = updatedTask;
|
|
267
|
+
}
|
|
268
|
+
// Return the final result
|
|
269
|
+
return (await extra.taskStore.getTaskResult(taskId));
|
|
270
|
+
}
|
|
271
|
+
setCompletionRequestHandler() {
|
|
272
|
+
if (this._completionHandlerInitialized) {
|
|
273
|
+
return;
|
|
274
|
+
}
|
|
275
|
+
this.server.assertCanSetRequestHandler(getMethodValue(CompleteRequestSchema));
|
|
276
|
+
this.server.registerCapabilities({
|
|
277
|
+
completions: {}
|
|
278
|
+
});
|
|
279
|
+
this.server.setRequestHandler(CompleteRequestSchema, async (request) => {
|
|
280
|
+
switch (request.params.ref.type) {
|
|
281
|
+
case 'ref/prompt':
|
|
282
|
+
assertCompleteRequestPrompt(request);
|
|
283
|
+
return this.handlePromptCompletion(request, request.params.ref);
|
|
284
|
+
case 'ref/resource':
|
|
285
|
+
assertCompleteRequestResourceTemplate(request);
|
|
286
|
+
return this.handleResourceCompletion(request, request.params.ref);
|
|
287
|
+
default:
|
|
288
|
+
throw new McpError(ErrorCode.InvalidParams, `Invalid completion reference: ${request.params.ref}`);
|
|
289
|
+
}
|
|
290
|
+
});
|
|
291
|
+
this._completionHandlerInitialized = true;
|
|
292
|
+
}
|
|
293
|
+
async handlePromptCompletion(request, ref) {
|
|
294
|
+
const prompt = this._registeredPrompts[ref.name];
|
|
295
|
+
if (!prompt) {
|
|
296
|
+
throw new McpError(ErrorCode.InvalidParams, `Prompt ${ref.name} not found`);
|
|
297
|
+
}
|
|
298
|
+
if (!prompt.enabled) {
|
|
299
|
+
throw new McpError(ErrorCode.InvalidParams, `Prompt ${ref.name} disabled`);
|
|
300
|
+
}
|
|
301
|
+
if (!prompt.argsSchema) {
|
|
302
|
+
return EMPTY_COMPLETION_RESULT;
|
|
303
|
+
}
|
|
304
|
+
const promptShape = getObjectShape(prompt.argsSchema);
|
|
305
|
+
const field = promptShape?.[request.params.argument.name];
|
|
306
|
+
if (!isCompletable(field)) {
|
|
307
|
+
return EMPTY_COMPLETION_RESULT;
|
|
308
|
+
}
|
|
309
|
+
const completer = getCompleter(field);
|
|
310
|
+
if (!completer) {
|
|
311
|
+
return EMPTY_COMPLETION_RESULT;
|
|
312
|
+
}
|
|
313
|
+
const suggestions = await completer(request.params.argument.value, request.params.context);
|
|
314
|
+
return createCompletionResult(suggestions);
|
|
315
|
+
}
|
|
316
|
+
async handleResourceCompletion(request, ref) {
|
|
317
|
+
const template = Object.values(this._registeredResourceTemplates).find(t => t.resourceTemplate.uriTemplate.toString() === ref.uri);
|
|
318
|
+
if (!template) {
|
|
319
|
+
if (this._registeredResources[ref.uri]) {
|
|
320
|
+
// Attempting to autocomplete a fixed resource URI is not an error in the spec (but probably should be).
|
|
321
|
+
return EMPTY_COMPLETION_RESULT;
|
|
322
|
+
}
|
|
323
|
+
throw new McpError(ErrorCode.InvalidParams, `Resource template ${request.params.ref.uri} not found`);
|
|
324
|
+
}
|
|
325
|
+
const completer = template.resourceTemplate.completeCallback(request.params.argument.name);
|
|
326
|
+
if (!completer) {
|
|
327
|
+
return EMPTY_COMPLETION_RESULT;
|
|
328
|
+
}
|
|
329
|
+
const suggestions = await completer(request.params.argument.value, request.params.context);
|
|
330
|
+
return createCompletionResult(suggestions);
|
|
331
|
+
}
|
|
332
|
+
setResourceRequestHandlers() {
|
|
333
|
+
if (this._resourceHandlersInitialized) {
|
|
334
|
+
return;
|
|
335
|
+
}
|
|
336
|
+
this.server.assertCanSetRequestHandler(getMethodValue(ListResourcesRequestSchema));
|
|
337
|
+
this.server.assertCanSetRequestHandler(getMethodValue(ListResourceTemplatesRequestSchema));
|
|
338
|
+
this.server.assertCanSetRequestHandler(getMethodValue(ReadResourceRequestSchema));
|
|
339
|
+
this.server.registerCapabilities({
|
|
340
|
+
resources: {
|
|
341
|
+
listChanged: true
|
|
342
|
+
}
|
|
343
|
+
});
|
|
344
|
+
this.server.setRequestHandler(ListResourcesRequestSchema, async (request, extra) => {
|
|
345
|
+
const resources = Object.entries(this._registeredResources)
|
|
346
|
+
.filter(([_, resource]) => resource.enabled)
|
|
347
|
+
.map(([uri, resource]) => ({
|
|
348
|
+
uri,
|
|
349
|
+
name: resource.name,
|
|
350
|
+
...resource.metadata
|
|
351
|
+
}));
|
|
352
|
+
const templateResources = [];
|
|
353
|
+
for (const template of Object.values(this._registeredResourceTemplates)) {
|
|
354
|
+
if (!template.resourceTemplate.listCallback) {
|
|
355
|
+
continue;
|
|
356
|
+
}
|
|
357
|
+
const result = await template.resourceTemplate.listCallback(extra);
|
|
358
|
+
for (const resource of result.resources) {
|
|
359
|
+
templateResources.push({
|
|
360
|
+
...template.metadata,
|
|
361
|
+
// the defined resource metadata should override the template metadata if present
|
|
362
|
+
...resource
|
|
363
|
+
});
|
|
364
|
+
}
|
|
365
|
+
}
|
|
366
|
+
return { resources: [...resources, ...templateResources] };
|
|
367
|
+
});
|
|
368
|
+
this.server.setRequestHandler(ListResourceTemplatesRequestSchema, async () => {
|
|
369
|
+
const resourceTemplates = Object.entries(this._registeredResourceTemplates).map(([name, template]) => ({
|
|
370
|
+
name,
|
|
371
|
+
uriTemplate: template.resourceTemplate.uriTemplate.toString(),
|
|
372
|
+
...template.metadata
|
|
373
|
+
}));
|
|
374
|
+
return { resourceTemplates };
|
|
375
|
+
});
|
|
376
|
+
this.server.setRequestHandler(ReadResourceRequestSchema, async (request, extra) => {
|
|
377
|
+
const uri = new URL(request.params.uri);
|
|
378
|
+
// First check for exact resource match
|
|
379
|
+
const resource = this._registeredResources[uri.toString()];
|
|
380
|
+
if (resource) {
|
|
381
|
+
if (!resource.enabled) {
|
|
382
|
+
throw new McpError(ErrorCode.InvalidParams, `Resource ${uri} disabled`);
|
|
383
|
+
}
|
|
384
|
+
return resource.readCallback(uri, extra);
|
|
385
|
+
}
|
|
386
|
+
// Then check templates
|
|
387
|
+
for (const template of Object.values(this._registeredResourceTemplates)) {
|
|
388
|
+
const variables = template.resourceTemplate.uriTemplate.match(uri.toString());
|
|
389
|
+
if (variables) {
|
|
390
|
+
return template.readCallback(uri, variables, extra);
|
|
391
|
+
}
|
|
392
|
+
}
|
|
393
|
+
throw new McpError(ErrorCode.InvalidParams, `Resource ${uri} not found`);
|
|
394
|
+
});
|
|
395
|
+
this._resourceHandlersInitialized = true;
|
|
396
|
+
}
|
|
397
|
+
setPromptRequestHandlers() {
|
|
398
|
+
if (this._promptHandlersInitialized) {
|
|
399
|
+
return;
|
|
400
|
+
}
|
|
401
|
+
this.server.assertCanSetRequestHandler(getMethodValue(ListPromptsRequestSchema));
|
|
402
|
+
this.server.assertCanSetRequestHandler(getMethodValue(GetPromptRequestSchema));
|
|
403
|
+
this.server.registerCapabilities({
|
|
404
|
+
prompts: {
|
|
405
|
+
listChanged: true
|
|
406
|
+
}
|
|
407
|
+
});
|
|
408
|
+
this.server.setRequestHandler(ListPromptsRequestSchema, () => ({
|
|
409
|
+
prompts: Object.entries(this._registeredPrompts)
|
|
410
|
+
.filter(([, prompt]) => prompt.enabled)
|
|
411
|
+
.map(([name, prompt]) => {
|
|
412
|
+
return {
|
|
413
|
+
name,
|
|
414
|
+
title: prompt.title,
|
|
415
|
+
description: prompt.description,
|
|
416
|
+
arguments: prompt.argsSchema ? promptArgumentsFromSchema(prompt.argsSchema) : undefined
|
|
417
|
+
};
|
|
418
|
+
})
|
|
419
|
+
}));
|
|
420
|
+
this.server.setRequestHandler(GetPromptRequestSchema, async (request, extra) => {
|
|
421
|
+
const prompt = this._registeredPrompts[request.params.name];
|
|
422
|
+
if (!prompt) {
|
|
423
|
+
throw new McpError(ErrorCode.InvalidParams, `Prompt ${request.params.name} not found`);
|
|
424
|
+
}
|
|
425
|
+
if (!prompt.enabled) {
|
|
426
|
+
throw new McpError(ErrorCode.InvalidParams, `Prompt ${request.params.name} disabled`);
|
|
427
|
+
}
|
|
428
|
+
if (prompt.argsSchema) {
|
|
429
|
+
const argsObj = normalizeObjectSchema(prompt.argsSchema);
|
|
430
|
+
const parseResult = await safeParseAsync(argsObj, request.params.arguments);
|
|
431
|
+
if (!parseResult.success) {
|
|
432
|
+
const error = 'error' in parseResult ? parseResult.error : 'Unknown error';
|
|
433
|
+
const errorMessage = getParseErrorMessage(error);
|
|
434
|
+
throw new McpError(ErrorCode.InvalidParams, `Invalid arguments for prompt ${request.params.name}: ${errorMessage}`);
|
|
435
|
+
}
|
|
436
|
+
const args = parseResult.data;
|
|
437
|
+
const cb = prompt.callback;
|
|
438
|
+
return await Promise.resolve(cb(args, extra));
|
|
439
|
+
}
|
|
440
|
+
else {
|
|
441
|
+
const cb = prompt.callback;
|
|
442
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
443
|
+
return await Promise.resolve(cb(extra));
|
|
444
|
+
}
|
|
445
|
+
});
|
|
446
|
+
this._promptHandlersInitialized = true;
|
|
447
|
+
}
|
|
448
|
+
resource(name, uriOrTemplate, ...rest) {
|
|
449
|
+
let metadata;
|
|
450
|
+
if (typeof rest[0] === 'object') {
|
|
451
|
+
metadata = rest.shift();
|
|
452
|
+
}
|
|
453
|
+
const readCallback = rest[0];
|
|
454
|
+
if (typeof uriOrTemplate === 'string') {
|
|
455
|
+
if (this._registeredResources[uriOrTemplate]) {
|
|
456
|
+
throw new Error(`Resource ${uriOrTemplate} is already registered`);
|
|
457
|
+
}
|
|
458
|
+
const registeredResource = this._createRegisteredResource(name, undefined, uriOrTemplate, metadata, readCallback);
|
|
459
|
+
this.setResourceRequestHandlers();
|
|
460
|
+
this.sendResourceListChanged();
|
|
461
|
+
return registeredResource;
|
|
462
|
+
}
|
|
463
|
+
else {
|
|
464
|
+
if (this._registeredResourceTemplates[name]) {
|
|
465
|
+
throw new Error(`Resource template ${name} is already registered`);
|
|
466
|
+
}
|
|
467
|
+
const registeredResourceTemplate = this._createRegisteredResourceTemplate(name, undefined, uriOrTemplate, metadata, readCallback);
|
|
468
|
+
this.setResourceRequestHandlers();
|
|
469
|
+
this.sendResourceListChanged();
|
|
470
|
+
return registeredResourceTemplate;
|
|
471
|
+
}
|
|
472
|
+
}
|
|
473
|
+
registerResource(name, uriOrTemplate, config, readCallback) {
|
|
474
|
+
if (typeof uriOrTemplate === 'string') {
|
|
475
|
+
if (this._registeredResources[uriOrTemplate]) {
|
|
476
|
+
throw new Error(`Resource ${uriOrTemplate} is already registered`);
|
|
477
|
+
}
|
|
478
|
+
const registeredResource = this._createRegisteredResource(name, config.title, uriOrTemplate, config, readCallback);
|
|
479
|
+
this.setResourceRequestHandlers();
|
|
480
|
+
this.sendResourceListChanged();
|
|
481
|
+
return registeredResource;
|
|
482
|
+
}
|
|
483
|
+
else {
|
|
484
|
+
if (this._registeredResourceTemplates[name]) {
|
|
485
|
+
throw new Error(`Resource template ${name} is already registered`);
|
|
486
|
+
}
|
|
487
|
+
const registeredResourceTemplate = this._createRegisteredResourceTemplate(name, config.title, uriOrTemplate, config, readCallback);
|
|
488
|
+
this.setResourceRequestHandlers();
|
|
489
|
+
this.sendResourceListChanged();
|
|
490
|
+
return registeredResourceTemplate;
|
|
491
|
+
}
|
|
492
|
+
}
|
|
493
|
+
_createRegisteredResource(name, title, uri, metadata, readCallback) {
|
|
494
|
+
const registeredResource = {
|
|
495
|
+
name,
|
|
496
|
+
title,
|
|
497
|
+
metadata,
|
|
498
|
+
readCallback,
|
|
499
|
+
enabled: true,
|
|
500
|
+
disable: () => registeredResource.update({ enabled: false }),
|
|
501
|
+
enable: () => registeredResource.update({ enabled: true }),
|
|
502
|
+
remove: () => registeredResource.update({ uri: null }),
|
|
503
|
+
update: updates => {
|
|
504
|
+
if (typeof updates.uri !== 'undefined' && updates.uri !== uri) {
|
|
505
|
+
delete this._registeredResources[uri];
|
|
506
|
+
if (updates.uri)
|
|
507
|
+
this._registeredResources[updates.uri] = registeredResource;
|
|
508
|
+
}
|
|
509
|
+
if (typeof updates.name !== 'undefined')
|
|
510
|
+
registeredResource.name = updates.name;
|
|
511
|
+
if (typeof updates.title !== 'undefined')
|
|
512
|
+
registeredResource.title = updates.title;
|
|
513
|
+
if (typeof updates.metadata !== 'undefined')
|
|
514
|
+
registeredResource.metadata = updates.metadata;
|
|
515
|
+
if (typeof updates.callback !== 'undefined')
|
|
516
|
+
registeredResource.readCallback = updates.callback;
|
|
517
|
+
if (typeof updates.enabled !== 'undefined')
|
|
518
|
+
registeredResource.enabled = updates.enabled;
|
|
519
|
+
this.sendResourceListChanged();
|
|
520
|
+
}
|
|
521
|
+
};
|
|
522
|
+
this._registeredResources[uri] = registeredResource;
|
|
523
|
+
return registeredResource;
|
|
524
|
+
}
|
|
525
|
+
_createRegisteredResourceTemplate(name, title, template, metadata, readCallback) {
|
|
526
|
+
const registeredResourceTemplate = {
|
|
527
|
+
resourceTemplate: template,
|
|
528
|
+
title,
|
|
529
|
+
metadata,
|
|
530
|
+
readCallback,
|
|
531
|
+
enabled: true,
|
|
532
|
+
disable: () => registeredResourceTemplate.update({ enabled: false }),
|
|
533
|
+
enable: () => registeredResourceTemplate.update({ enabled: true }),
|
|
534
|
+
remove: () => registeredResourceTemplate.update({ name: null }),
|
|
535
|
+
update: updates => {
|
|
536
|
+
if (typeof updates.name !== 'undefined' && updates.name !== name) {
|
|
537
|
+
delete this._registeredResourceTemplates[name];
|
|
538
|
+
if (updates.name)
|
|
539
|
+
this._registeredResourceTemplates[updates.name] = registeredResourceTemplate;
|
|
540
|
+
}
|
|
541
|
+
if (typeof updates.title !== 'undefined')
|
|
542
|
+
registeredResourceTemplate.title = updates.title;
|
|
543
|
+
if (typeof updates.template !== 'undefined')
|
|
544
|
+
registeredResourceTemplate.resourceTemplate = updates.template;
|
|
545
|
+
if (typeof updates.metadata !== 'undefined')
|
|
546
|
+
registeredResourceTemplate.metadata = updates.metadata;
|
|
547
|
+
if (typeof updates.callback !== 'undefined')
|
|
548
|
+
registeredResourceTemplate.readCallback = updates.callback;
|
|
549
|
+
if (typeof updates.enabled !== 'undefined')
|
|
550
|
+
registeredResourceTemplate.enabled = updates.enabled;
|
|
551
|
+
this.sendResourceListChanged();
|
|
552
|
+
}
|
|
553
|
+
};
|
|
554
|
+
this._registeredResourceTemplates[name] = registeredResourceTemplate;
|
|
555
|
+
// If the resource template has any completion callbacks, enable completions capability
|
|
556
|
+
const variableNames = template.uriTemplate.variableNames;
|
|
557
|
+
const hasCompleter = Array.isArray(variableNames) && variableNames.some(v => !!template.completeCallback(v));
|
|
558
|
+
if (hasCompleter) {
|
|
559
|
+
this.setCompletionRequestHandler();
|
|
560
|
+
}
|
|
561
|
+
return registeredResourceTemplate;
|
|
562
|
+
}
|
|
563
|
+
_createRegisteredPrompt(name, title, description, argsSchema, callback) {
|
|
564
|
+
const registeredPrompt = {
|
|
565
|
+
title,
|
|
566
|
+
description,
|
|
567
|
+
argsSchema: argsSchema === undefined ? undefined : objectFromShape(argsSchema),
|
|
568
|
+
callback,
|
|
569
|
+
enabled: true,
|
|
570
|
+
disable: () => registeredPrompt.update({ enabled: false }),
|
|
571
|
+
enable: () => registeredPrompt.update({ enabled: true }),
|
|
572
|
+
remove: () => registeredPrompt.update({ name: null }),
|
|
573
|
+
update: updates => {
|
|
574
|
+
if (typeof updates.name !== 'undefined' && updates.name !== name) {
|
|
575
|
+
delete this._registeredPrompts[name];
|
|
576
|
+
if (updates.name)
|
|
577
|
+
this._registeredPrompts[updates.name] = registeredPrompt;
|
|
578
|
+
}
|
|
579
|
+
if (typeof updates.title !== 'undefined')
|
|
580
|
+
registeredPrompt.title = updates.title;
|
|
581
|
+
if (typeof updates.description !== 'undefined')
|
|
582
|
+
registeredPrompt.description = updates.description;
|
|
583
|
+
if (typeof updates.argsSchema !== 'undefined')
|
|
584
|
+
registeredPrompt.argsSchema = objectFromShape(updates.argsSchema);
|
|
585
|
+
if (typeof updates.callback !== 'undefined')
|
|
586
|
+
registeredPrompt.callback = updates.callback;
|
|
587
|
+
if (typeof updates.enabled !== 'undefined')
|
|
588
|
+
registeredPrompt.enabled = updates.enabled;
|
|
589
|
+
this.sendPromptListChanged();
|
|
590
|
+
}
|
|
591
|
+
};
|
|
592
|
+
this._registeredPrompts[name] = registeredPrompt;
|
|
593
|
+
// If any argument uses a Completable schema, enable completions capability
|
|
594
|
+
if (argsSchema) {
|
|
595
|
+
const hasCompletable = Object.values(argsSchema).some(field => {
|
|
596
|
+
const inner = field instanceof ZodOptional ? field._def?.innerType : field;
|
|
597
|
+
return isCompletable(inner);
|
|
598
|
+
});
|
|
599
|
+
if (hasCompletable) {
|
|
600
|
+
this.setCompletionRequestHandler();
|
|
601
|
+
}
|
|
602
|
+
}
|
|
603
|
+
return registeredPrompt;
|
|
604
|
+
}
|
|
605
|
+
_createRegisteredTool(name, title, description, inputSchema, outputSchema, annotations, execution, _meta, handler) {
|
|
606
|
+
// Validate tool name according to SEP specification
|
|
607
|
+
validateAndWarnToolName(name);
|
|
608
|
+
const registeredTool = {
|
|
609
|
+
title,
|
|
610
|
+
description,
|
|
611
|
+
inputSchema: getZodSchemaObject(inputSchema),
|
|
612
|
+
outputSchema: getZodSchemaObject(outputSchema),
|
|
613
|
+
annotations,
|
|
614
|
+
execution,
|
|
615
|
+
_meta,
|
|
616
|
+
handler: handler,
|
|
617
|
+
enabled: true,
|
|
618
|
+
disable: () => registeredTool.update({ enabled: false }),
|
|
619
|
+
enable: () => registeredTool.update({ enabled: true }),
|
|
620
|
+
remove: () => registeredTool.update({ name: null }),
|
|
621
|
+
update: updates => {
|
|
622
|
+
if (typeof updates.name !== 'undefined' && updates.name !== name) {
|
|
623
|
+
if (typeof updates.name === 'string') {
|
|
624
|
+
validateAndWarnToolName(updates.name);
|
|
625
|
+
}
|
|
626
|
+
delete this._registeredTools[name];
|
|
627
|
+
if (updates.name)
|
|
628
|
+
this._registeredTools[updates.name] = registeredTool;
|
|
629
|
+
}
|
|
630
|
+
if (typeof updates.title !== 'undefined')
|
|
631
|
+
registeredTool.title = updates.title;
|
|
632
|
+
if (typeof updates.description !== 'undefined')
|
|
633
|
+
registeredTool.description = updates.description;
|
|
634
|
+
if (typeof updates.paramsSchema !== 'undefined')
|
|
635
|
+
registeredTool.inputSchema = objectFromShape(updates.paramsSchema);
|
|
636
|
+
if (typeof updates.outputSchema !== 'undefined')
|
|
637
|
+
registeredTool.outputSchema = objectFromShape(updates.outputSchema);
|
|
638
|
+
if (typeof updates.callback !== 'undefined')
|
|
639
|
+
registeredTool.handler = updates.callback;
|
|
640
|
+
if (typeof updates.annotations !== 'undefined')
|
|
641
|
+
registeredTool.annotations = updates.annotations;
|
|
642
|
+
if (typeof updates._meta !== 'undefined')
|
|
643
|
+
registeredTool._meta = updates._meta;
|
|
644
|
+
if (typeof updates.enabled !== 'undefined')
|
|
645
|
+
registeredTool.enabled = updates.enabled;
|
|
646
|
+
this.sendToolListChanged();
|
|
647
|
+
}
|
|
648
|
+
};
|
|
649
|
+
this._registeredTools[name] = registeredTool;
|
|
650
|
+
this.setToolRequestHandlers();
|
|
651
|
+
this.sendToolListChanged();
|
|
652
|
+
return registeredTool;
|
|
653
|
+
}
|
|
654
|
+
/**
|
|
655
|
+
* tool() implementation. Parses arguments passed to overrides defined above.
|
|
656
|
+
*/
|
|
657
|
+
tool(name, ...rest) {
|
|
658
|
+
if (this._registeredTools[name]) {
|
|
659
|
+
throw new Error(`Tool ${name} is already registered`);
|
|
660
|
+
}
|
|
661
|
+
let description;
|
|
662
|
+
let inputSchema;
|
|
663
|
+
let outputSchema;
|
|
664
|
+
let annotations;
|
|
665
|
+
// Tool properties are passed as separate arguments, with omissions allowed.
|
|
666
|
+
// Support for this style is frozen as of protocol version 2025-03-26. Future additions
|
|
667
|
+
// to tool definition should *NOT* be added.
|
|
668
|
+
if (typeof rest[0] === 'string') {
|
|
669
|
+
description = rest.shift();
|
|
670
|
+
}
|
|
671
|
+
// Handle the different overload combinations
|
|
672
|
+
if (rest.length > 1) {
|
|
673
|
+
// We have at least one more arg before the callback
|
|
674
|
+
const firstArg = rest[0];
|
|
675
|
+
if (isZodRawShapeCompat(firstArg)) {
|
|
676
|
+
// We have a params schema as the first arg
|
|
677
|
+
inputSchema = rest.shift();
|
|
678
|
+
// Check if the next arg is potentially annotations
|
|
679
|
+
if (rest.length > 1 && typeof rest[0] === 'object' && rest[0] !== null && !isZodRawShapeCompat(rest[0])) {
|
|
680
|
+
// Case: tool(name, paramsSchema, annotations, cb)
|
|
681
|
+
// Or: tool(name, description, paramsSchema, annotations, cb)
|
|
682
|
+
annotations = rest.shift();
|
|
683
|
+
}
|
|
684
|
+
}
|
|
685
|
+
else if (typeof firstArg === 'object' && firstArg !== null) {
|
|
686
|
+
// Not a ZodRawShapeCompat, so must be annotations in this position
|
|
687
|
+
// Case: tool(name, annotations, cb)
|
|
688
|
+
// Or: tool(name, description, annotations, cb)
|
|
689
|
+
annotations = rest.shift();
|
|
690
|
+
}
|
|
691
|
+
}
|
|
692
|
+
const callback = rest[0];
|
|
693
|
+
return this._createRegisteredTool(name, undefined, description, inputSchema, outputSchema, annotations, { taskSupport: 'forbidden' }, undefined, callback);
|
|
694
|
+
}
|
|
695
|
+
/**
|
|
696
|
+
* Registers a tool with a config object and callback.
|
|
697
|
+
*/
|
|
698
|
+
registerTool(name, config, cb) {
|
|
699
|
+
if (this._registeredTools[name]) {
|
|
700
|
+
throw new Error(`Tool ${name} is already registered`);
|
|
701
|
+
}
|
|
702
|
+
const { title, description, inputSchema, outputSchema, annotations, _meta } = config;
|
|
703
|
+
return this._createRegisteredTool(name, title, description, inputSchema, outputSchema, annotations, { taskSupport: 'forbidden' }, _meta, cb);
|
|
704
|
+
}
|
|
705
|
+
prompt(name, ...rest) {
|
|
706
|
+
if (this._registeredPrompts[name]) {
|
|
707
|
+
throw new Error(`Prompt ${name} is already registered`);
|
|
708
|
+
}
|
|
709
|
+
let description;
|
|
710
|
+
if (typeof rest[0] === 'string') {
|
|
711
|
+
description = rest.shift();
|
|
712
|
+
}
|
|
713
|
+
let argsSchema;
|
|
714
|
+
if (rest.length > 1) {
|
|
715
|
+
argsSchema = rest.shift();
|
|
716
|
+
}
|
|
717
|
+
const cb = rest[0];
|
|
718
|
+
const registeredPrompt = this._createRegisteredPrompt(name, undefined, description, argsSchema, cb);
|
|
719
|
+
this.setPromptRequestHandlers();
|
|
720
|
+
this.sendPromptListChanged();
|
|
721
|
+
return registeredPrompt;
|
|
722
|
+
}
|
|
723
|
+
/**
|
|
724
|
+
* Registers a prompt with a config object and callback.
|
|
725
|
+
*/
|
|
726
|
+
registerPrompt(name, config, cb) {
|
|
727
|
+
if (this._registeredPrompts[name]) {
|
|
728
|
+
throw new Error(`Prompt ${name} is already registered`);
|
|
729
|
+
}
|
|
730
|
+
const { title, description, argsSchema } = config;
|
|
731
|
+
const registeredPrompt = this._createRegisteredPrompt(name, title, description, argsSchema, cb);
|
|
732
|
+
this.setPromptRequestHandlers();
|
|
733
|
+
this.sendPromptListChanged();
|
|
734
|
+
return registeredPrompt;
|
|
735
|
+
}
|
|
736
|
+
/**
|
|
737
|
+
* Checks if the server is connected to a transport.
|
|
738
|
+
* @returns True if the server is connected
|
|
739
|
+
*/
|
|
740
|
+
isConnected() {
|
|
741
|
+
return this.server.transport !== undefined;
|
|
742
|
+
}
|
|
743
|
+
/**
|
|
744
|
+
* Sends a logging message to the client, if connected.
|
|
745
|
+
* Note: You only need to send the parameters object, not the entire JSON RPC message
|
|
746
|
+
* @see LoggingMessageNotification
|
|
747
|
+
* @param params
|
|
748
|
+
* @param sessionId optional for stateless and backward compatibility
|
|
749
|
+
*/
|
|
750
|
+
async sendLoggingMessage(params, sessionId) {
|
|
751
|
+
return this.server.sendLoggingMessage(params, sessionId);
|
|
752
|
+
}
|
|
753
|
+
/**
|
|
754
|
+
* Sends a resource list changed event to the client, if connected.
|
|
755
|
+
*/
|
|
756
|
+
sendResourceListChanged() {
|
|
757
|
+
if (this.isConnected()) {
|
|
758
|
+
this.server.sendResourceListChanged();
|
|
759
|
+
}
|
|
760
|
+
}
|
|
761
|
+
/**
|
|
762
|
+
* Sends a tool list changed event to the client, if connected.
|
|
763
|
+
*/
|
|
764
|
+
sendToolListChanged() {
|
|
765
|
+
if (this.isConnected()) {
|
|
766
|
+
this.server.sendToolListChanged();
|
|
767
|
+
}
|
|
768
|
+
}
|
|
769
|
+
/**
|
|
770
|
+
* Sends a prompt list changed event to the client, if connected.
|
|
771
|
+
*/
|
|
772
|
+
sendPromptListChanged() {
|
|
773
|
+
if (this.isConnected()) {
|
|
774
|
+
this.server.sendPromptListChanged();
|
|
775
|
+
}
|
|
776
|
+
}
|
|
777
|
+
}
|
|
778
|
+
/**
|
|
779
|
+
* A resource template combines a URI pattern with optional functionality to enumerate
|
|
780
|
+
* all resources matching that pattern.
|
|
781
|
+
*/
|
|
782
|
+
export class ResourceTemplate {
|
|
783
|
+
constructor(uriTemplate, _callbacks) {
|
|
784
|
+
this._callbacks = _callbacks;
|
|
785
|
+
this._uriTemplate = typeof uriTemplate === 'string' ? new UriTemplate(uriTemplate) : uriTemplate;
|
|
786
|
+
}
|
|
787
|
+
/**
|
|
788
|
+
* Gets the URI template pattern.
|
|
789
|
+
*/
|
|
790
|
+
get uriTemplate() {
|
|
791
|
+
return this._uriTemplate;
|
|
792
|
+
}
|
|
793
|
+
/**
|
|
794
|
+
* Gets the list callback, if one was provided.
|
|
795
|
+
*/
|
|
796
|
+
get listCallback() {
|
|
797
|
+
return this._callbacks.list;
|
|
798
|
+
}
|
|
799
|
+
/**
|
|
800
|
+
* Gets the callback for completing a specific URI template variable, if one was provided.
|
|
801
|
+
*/
|
|
802
|
+
completeCallback(variable) {
|
|
803
|
+
return this._callbacks.complete?.[variable];
|
|
804
|
+
}
|
|
805
|
+
}
|
|
806
|
+
const EMPTY_OBJECT_JSON_SCHEMA = {
|
|
807
|
+
type: 'object',
|
|
808
|
+
properties: {}
|
|
809
|
+
};
|
|
810
|
+
/**
|
|
811
|
+
* Checks if a value looks like a Zod schema by checking for parse/safeParse methods.
|
|
812
|
+
*/
|
|
813
|
+
function isZodTypeLike(value) {
|
|
814
|
+
return (value !== null &&
|
|
815
|
+
typeof value === 'object' &&
|
|
816
|
+
'parse' in value &&
|
|
817
|
+
typeof value.parse === 'function' &&
|
|
818
|
+
'safeParse' in value &&
|
|
819
|
+
typeof value.safeParse === 'function');
|
|
820
|
+
}
|
|
821
|
+
/**
|
|
822
|
+
* Checks if an object is a Zod schema instance (v3 or v4).
|
|
823
|
+
*
|
|
824
|
+
* Zod schemas have internal markers:
|
|
825
|
+
* - v3: `_def` property
|
|
826
|
+
* - v4: `_zod` property
|
|
827
|
+
*
|
|
828
|
+
* This includes transformed schemas like z.preprocess(), z.transform(), z.pipe().
|
|
829
|
+
*/
|
|
830
|
+
function isZodSchemaInstance(obj) {
|
|
831
|
+
return '_def' in obj || '_zod' in obj || isZodTypeLike(obj);
|
|
832
|
+
}
|
|
833
|
+
/**
|
|
834
|
+
* Checks if an object is a "raw shape" - a plain object where values are Zod schemas.
|
|
835
|
+
*
|
|
836
|
+
* Raw shapes are used as shorthand: `{ name: z.string() }` instead of `z.object({ name: z.string() })`.
|
|
837
|
+
*
|
|
838
|
+
* IMPORTANT: This must NOT match actual Zod schema instances (like z.preprocess, z.pipe),
|
|
839
|
+
* which have internal properties that could be mistaken for schema values.
|
|
840
|
+
*/
|
|
841
|
+
function isZodRawShapeCompat(obj) {
|
|
842
|
+
if (typeof obj !== 'object' || obj === null) {
|
|
843
|
+
return false;
|
|
844
|
+
}
|
|
845
|
+
// If it's already a Zod schema instance, it's NOT a raw shape
|
|
846
|
+
if (isZodSchemaInstance(obj)) {
|
|
847
|
+
return false;
|
|
848
|
+
}
|
|
849
|
+
// Empty objects are valid raw shapes (tools with no parameters)
|
|
850
|
+
if (Object.keys(obj).length === 0) {
|
|
851
|
+
return true;
|
|
852
|
+
}
|
|
853
|
+
// A raw shape has at least one property that is a Zod schema
|
|
854
|
+
return Object.values(obj).some(isZodTypeLike);
|
|
855
|
+
}
|
|
856
|
+
/**
|
|
857
|
+
* Converts a provided Zod schema to a Zod object if it is a ZodRawShapeCompat,
|
|
858
|
+
* otherwise returns the schema as is.
|
|
859
|
+
*/
|
|
860
|
+
function getZodSchemaObject(schema) {
|
|
861
|
+
if (!schema) {
|
|
862
|
+
return undefined;
|
|
863
|
+
}
|
|
864
|
+
if (isZodRawShapeCompat(schema)) {
|
|
865
|
+
return objectFromShape(schema);
|
|
866
|
+
}
|
|
867
|
+
return schema;
|
|
868
|
+
}
|
|
869
|
+
function promptArgumentsFromSchema(schema) {
|
|
870
|
+
const shape = getObjectShape(schema);
|
|
871
|
+
if (!shape)
|
|
872
|
+
return [];
|
|
873
|
+
return Object.entries(shape).map(([name, field]) => {
|
|
874
|
+
// Get description - works for both v3 and v4
|
|
875
|
+
const description = getSchemaDescription(field);
|
|
876
|
+
// Check if optional - works for both v3 and v4
|
|
877
|
+
const isOptional = isSchemaOptional(field);
|
|
878
|
+
return {
|
|
879
|
+
name,
|
|
880
|
+
description,
|
|
881
|
+
required: !isOptional
|
|
882
|
+
};
|
|
883
|
+
});
|
|
884
|
+
}
|
|
885
|
+
function getMethodValue(schema) {
|
|
886
|
+
const shape = getObjectShape(schema);
|
|
887
|
+
const methodSchema = shape?.method;
|
|
888
|
+
if (!methodSchema) {
|
|
889
|
+
throw new Error('Schema is missing a method literal');
|
|
890
|
+
}
|
|
891
|
+
// Extract literal value - works for both v3 and v4
|
|
892
|
+
const value = getLiteralValue(methodSchema);
|
|
893
|
+
if (typeof value === 'string') {
|
|
894
|
+
return value;
|
|
895
|
+
}
|
|
896
|
+
throw new Error('Schema method literal must be a string');
|
|
897
|
+
}
|
|
898
|
+
function createCompletionResult(suggestions) {
|
|
899
|
+
return {
|
|
900
|
+
completion: {
|
|
901
|
+
values: suggestions.slice(0, 100),
|
|
902
|
+
total: suggestions.length,
|
|
903
|
+
hasMore: suggestions.length > 100
|
|
904
|
+
}
|
|
905
|
+
};
|
|
906
|
+
}
|
|
907
|
+
const EMPTY_COMPLETION_RESULT = {
|
|
908
|
+
completion: {
|
|
909
|
+
values: [],
|
|
910
|
+
hasMore: false
|
|
911
|
+
}
|
|
912
|
+
};
|
|
913
|
+
//# sourceMappingURL=mcp.js.map
|