@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.
Files changed (261) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +141 -0
  3. package/dist/cjs/client/index.d.ts +588 -0
  4. package/dist/cjs/client/index.d.ts.map +1 -0
  5. package/dist/cjs/client/index.js +629 -0
  6. package/dist/cjs/client/index.js.map +1 -0
  7. package/dist/cjs/client/stdio.d.ts +77 -0
  8. package/dist/cjs/client/stdio.d.ts.map +1 -0
  9. package/dist/cjs/client/stdio.js +199 -0
  10. package/dist/cjs/client/stdio.js.map +1 -0
  11. package/dist/cjs/experimental/index.d.ts +13 -0
  12. package/dist/cjs/experimental/index.d.ts.map +1 -0
  13. package/dist/cjs/experimental/index.js +29 -0
  14. package/dist/cjs/experimental/index.js.map +1 -0
  15. package/dist/cjs/experimental/tasks/client.d.ts +121 -0
  16. package/dist/cjs/experimental/tasks/client.d.ts.map +1 -0
  17. package/dist/cjs/experimental/tasks/client.js +188 -0
  18. package/dist/cjs/experimental/tasks/client.js.map +1 -0
  19. package/dist/cjs/experimental/tasks/helpers.d.ts +47 -0
  20. package/dist/cjs/experimental/tasks/helpers.d.ts.map +1 -0
  21. package/dist/cjs/experimental/tasks/helpers.js +68 -0
  22. package/dist/cjs/experimental/tasks/helpers.js.map +1 -0
  23. package/dist/cjs/experimental/tasks/index.d.ts +16 -0
  24. package/dist/cjs/experimental/tasks/index.d.ts.map +1 -0
  25. package/dist/cjs/experimental/tasks/index.js +39 -0
  26. package/dist/cjs/experimental/tasks/index.js.map +1 -0
  27. package/dist/cjs/experimental/tasks/interfaces.d.ts +232 -0
  28. package/dist/cjs/experimental/tasks/interfaces.d.ts.map +1 -0
  29. package/dist/cjs/experimental/tasks/interfaces.js +19 -0
  30. package/dist/cjs/experimental/tasks/interfaces.js.map +1 -0
  31. package/dist/cjs/experimental/tasks/mcp-server.d.ts +77 -0
  32. package/dist/cjs/experimental/tasks/mcp-server.d.ts.map +1 -0
  33. package/dist/cjs/experimental/tasks/mcp-server.js +36 -0
  34. package/dist/cjs/experimental/tasks/mcp-server.js.map +1 -0
  35. package/dist/cjs/experimental/tasks/server.d.ts +170 -0
  36. package/dist/cjs/experimental/tasks/server.d.ts.map +1 -0
  37. package/dist/cjs/experimental/tasks/server.js +250 -0
  38. package/dist/cjs/experimental/tasks/server.js.map +1 -0
  39. package/dist/cjs/experimental/tasks/stores/in-memory.d.ts +94 -0
  40. package/dist/cjs/experimental/tasks/stores/in-memory.d.ts.map +1 -0
  41. package/dist/cjs/experimental/tasks/stores/in-memory.js +251 -0
  42. package/dist/cjs/experimental/tasks/stores/in-memory.js.map +1 -0
  43. package/dist/cjs/experimental/tasks/types.d.ts +10 -0
  44. package/dist/cjs/experimental/tasks/types.d.ts.map +1 -0
  45. package/dist/cjs/experimental/tasks/types.js +28 -0
  46. package/dist/cjs/experimental/tasks/types.js.map +1 -0
  47. package/dist/cjs/inMemory.d.ts +31 -0
  48. package/dist/cjs/inMemory.d.ts.map +1 -0
  49. package/dist/cjs/inMemory.js +51 -0
  50. package/dist/cjs/inMemory.js.map +1 -0
  51. package/dist/cjs/package.json +1 -0
  52. package/dist/cjs/server/completable.d.ts +38 -0
  53. package/dist/cjs/server/completable.d.ts.map +1 -0
  54. package/dist/cjs/server/completable.js +48 -0
  55. package/dist/cjs/server/completable.js.map +1 -0
  56. package/dist/cjs/server/index.d.ts +196 -0
  57. package/dist/cjs/server/index.d.ts.map +1 -0
  58. package/dist/cjs/server/index.js +444 -0
  59. package/dist/cjs/server/index.js.map +1 -0
  60. package/dist/cjs/server/mcp.d.ts +364 -0
  61. package/dist/cjs/server/mcp.d.ts.map +1 -0
  62. package/dist/cjs/server/mcp.js +918 -0
  63. package/dist/cjs/server/mcp.js.map +1 -0
  64. package/dist/cjs/server/stdio.d.ts +28 -0
  65. package/dist/cjs/server/stdio.d.ts.map +1 -0
  66. package/dist/cjs/server/stdio.js +82 -0
  67. package/dist/cjs/server/stdio.js.map +1 -0
  68. package/dist/cjs/server/zod-compat.d.ts +84 -0
  69. package/dist/cjs/server/zod-compat.d.ts.map +1 -0
  70. package/dist/cjs/server/zod-compat.js +244 -0
  71. package/dist/cjs/server/zod-compat.js.map +1 -0
  72. package/dist/cjs/server/zod-json-schema-compat.d.ts +12 -0
  73. package/dist/cjs/server/zod-json-schema-compat.d.ts.map +1 -0
  74. package/dist/cjs/server/zod-json-schema-compat.js +79 -0
  75. package/dist/cjs/server/zod-json-schema-compat.js.map +1 -0
  76. package/dist/cjs/shared/auth-info.d.ts +32 -0
  77. package/dist/cjs/shared/auth-info.d.ts.map +1 -0
  78. package/dist/cjs/shared/auth-info.js +3 -0
  79. package/dist/cjs/shared/auth-info.js.map +1 -0
  80. package/dist/cjs/shared/metadataUtils.d.ts +16 -0
  81. package/dist/cjs/shared/metadataUtils.d.ts.map +1 -0
  82. package/dist/cjs/shared/metadataUtils.js +25 -0
  83. package/dist/cjs/shared/metadataUtils.js.map +1 -0
  84. package/dist/cjs/shared/protocol.d.ts +443 -0
  85. package/dist/cjs/shared/protocol.d.ts.map +1 -0
  86. package/dist/cjs/shared/protocol.js +1104 -0
  87. package/dist/cjs/shared/protocol.js.map +1 -0
  88. package/dist/cjs/shared/responseMessage.d.ts +45 -0
  89. package/dist/cjs/shared/responseMessage.d.ts.map +1 -0
  90. package/dist/cjs/shared/responseMessage.js +23 -0
  91. package/dist/cjs/shared/responseMessage.js.map +1 -0
  92. package/dist/cjs/shared/stdio.d.ts +13 -0
  93. package/dist/cjs/shared/stdio.d.ts.map +1 -0
  94. package/dist/cjs/shared/stdio.js +37 -0
  95. package/dist/cjs/shared/stdio.js.map +1 -0
  96. package/dist/cjs/shared/toolNameValidation.d.ts +31 -0
  97. package/dist/cjs/shared/toolNameValidation.d.ts.map +1 -0
  98. package/dist/cjs/shared/toolNameValidation.js +97 -0
  99. package/dist/cjs/shared/toolNameValidation.js.map +1 -0
  100. package/dist/cjs/shared/transport.d.ts +89 -0
  101. package/dist/cjs/shared/transport.d.ts.map +1 -0
  102. package/dist/cjs/shared/transport.js +43 -0
  103. package/dist/cjs/shared/transport.js.map +1 -0
  104. package/dist/cjs/shared/uriTemplate.d.ts +25 -0
  105. package/dist/cjs/shared/uriTemplate.d.ts.map +1 -0
  106. package/dist/cjs/shared/uriTemplate.js +243 -0
  107. package/dist/cjs/shared/uriTemplate.js.map +1 -0
  108. package/dist/cjs/spec.types.d.ts +2299 -0
  109. package/dist/cjs/spec.types.d.ts.map +1 -0
  110. package/dist/cjs/spec.types.js +27 -0
  111. package/dist/cjs/spec.types.js.map +1 -0
  112. package/dist/cjs/types.d.ts +8137 -0
  113. package/dist/cjs/types.d.ts.map +1 -0
  114. package/dist/cjs/types.js +2092 -0
  115. package/dist/cjs/types.js.map +1 -0
  116. package/dist/cjs/validation/ajv-provider.d.ts +53 -0
  117. package/dist/cjs/validation/ajv-provider.d.ts.map +1 -0
  118. package/dist/cjs/validation/ajv-provider.js +94 -0
  119. package/dist/cjs/validation/ajv-provider.js.map +1 -0
  120. package/dist/cjs/validation/cfworker-provider.d.ts +51 -0
  121. package/dist/cjs/validation/cfworker-provider.d.ts.map +1 -0
  122. package/dist/cjs/validation/cfworker-provider.js +69 -0
  123. package/dist/cjs/validation/cfworker-provider.js.map +1 -0
  124. package/dist/cjs/validation/index.d.ts +29 -0
  125. package/dist/cjs/validation/index.d.ts.map +1 -0
  126. package/dist/cjs/validation/index.js +30 -0
  127. package/dist/cjs/validation/index.js.map +1 -0
  128. package/dist/cjs/validation/types.d.ts +65 -0
  129. package/dist/cjs/validation/types.d.ts.map +1 -0
  130. package/dist/cjs/validation/types.js +3 -0
  131. package/dist/cjs/validation/types.js.map +1 -0
  132. package/dist/esm/client/index.d.ts +588 -0
  133. package/dist/esm/client/index.d.ts.map +1 -0
  134. package/dist/esm/client/index.js +624 -0
  135. package/dist/esm/client/index.js.map +1 -0
  136. package/dist/esm/client/stdio.d.ts +77 -0
  137. package/dist/esm/client/stdio.d.ts.map +1 -0
  138. package/dist/esm/client/stdio.js +191 -0
  139. package/dist/esm/client/stdio.js.map +1 -0
  140. package/dist/esm/experimental/index.d.ts +13 -0
  141. package/dist/esm/experimental/index.d.ts.map +1 -0
  142. package/dist/esm/experimental/index.js +13 -0
  143. package/dist/esm/experimental/index.js.map +1 -0
  144. package/dist/esm/experimental/tasks/client.d.ts +121 -0
  145. package/dist/esm/experimental/tasks/client.d.ts.map +1 -0
  146. package/dist/esm/experimental/tasks/client.js +184 -0
  147. package/dist/esm/experimental/tasks/client.js.map +1 -0
  148. package/dist/esm/experimental/tasks/helpers.d.ts +47 -0
  149. package/dist/esm/experimental/tasks/helpers.d.ts.map +1 -0
  150. package/dist/esm/experimental/tasks/helpers.js +64 -0
  151. package/dist/esm/experimental/tasks/helpers.js.map +1 -0
  152. package/dist/esm/experimental/tasks/index.d.ts +16 -0
  153. package/dist/esm/experimental/tasks/index.d.ts.map +1 -0
  154. package/dist/esm/experimental/tasks/index.js +20 -0
  155. package/dist/esm/experimental/tasks/index.js.map +1 -0
  156. package/dist/esm/experimental/tasks/interfaces.d.ts +232 -0
  157. package/dist/esm/experimental/tasks/interfaces.d.ts.map +1 -0
  158. package/dist/esm/experimental/tasks/interfaces.js +16 -0
  159. package/dist/esm/experimental/tasks/interfaces.js.map +1 -0
  160. package/dist/esm/experimental/tasks/mcp-server.d.ts +77 -0
  161. package/dist/esm/experimental/tasks/mcp-server.d.ts.map +1 -0
  162. package/dist/esm/experimental/tasks/mcp-server.js +32 -0
  163. package/dist/esm/experimental/tasks/mcp-server.js.map +1 -0
  164. package/dist/esm/experimental/tasks/server.d.ts +170 -0
  165. package/dist/esm/experimental/tasks/server.d.ts.map +1 -0
  166. package/dist/esm/experimental/tasks/server.js +246 -0
  167. package/dist/esm/experimental/tasks/server.js.map +1 -0
  168. package/dist/esm/experimental/tasks/stores/in-memory.d.ts +94 -0
  169. package/dist/esm/experimental/tasks/stores/in-memory.d.ts.map +1 -0
  170. package/dist/esm/experimental/tasks/stores/in-memory.js +246 -0
  171. package/dist/esm/experimental/tasks/stores/in-memory.js.map +1 -0
  172. package/dist/esm/experimental/tasks/types.d.ts +10 -0
  173. package/dist/esm/experimental/tasks/types.d.ts.map +1 -0
  174. package/dist/esm/experimental/tasks/types.js +10 -0
  175. package/dist/esm/experimental/tasks/types.js.map +1 -0
  176. package/dist/esm/inMemory.d.ts +31 -0
  177. package/dist/esm/inMemory.d.ts.map +1 -0
  178. package/dist/esm/inMemory.js +47 -0
  179. package/dist/esm/inMemory.js.map +1 -0
  180. package/dist/esm/package.json +1 -0
  181. package/dist/esm/server/completable.d.ts +38 -0
  182. package/dist/esm/server/completable.d.ts.map +1 -0
  183. package/dist/esm/server/completable.js +41 -0
  184. package/dist/esm/server/completable.js.map +1 -0
  185. package/dist/esm/server/index.d.ts +196 -0
  186. package/dist/esm/server/index.d.ts.map +1 -0
  187. package/dist/esm/server/index.js +440 -0
  188. package/dist/esm/server/index.js.map +1 -0
  189. package/dist/esm/server/mcp.d.ts +364 -0
  190. package/dist/esm/server/mcp.d.ts.map +1 -0
  191. package/dist/esm/server/mcp.js +913 -0
  192. package/dist/esm/server/mcp.js.map +1 -0
  193. package/dist/esm/server/stdio.d.ts +28 -0
  194. package/dist/esm/server/stdio.d.ts.map +1 -0
  195. package/dist/esm/server/stdio.js +75 -0
  196. package/dist/esm/server/stdio.js.map +1 -0
  197. package/dist/esm/server/zod-compat.d.ts +84 -0
  198. package/dist/esm/server/zod-compat.d.ts.map +1 -0
  199. package/dist/esm/server/zod-compat.js +209 -0
  200. package/dist/esm/server/zod-compat.js.map +1 -0
  201. package/dist/esm/server/zod-json-schema-compat.d.ts +12 -0
  202. package/dist/esm/server/zod-json-schema-compat.d.ts.map +1 -0
  203. package/dist/esm/server/zod-json-schema-compat.js +51 -0
  204. package/dist/esm/server/zod-json-schema-compat.js.map +1 -0
  205. package/dist/esm/shared/auth-info.d.ts +32 -0
  206. package/dist/esm/shared/auth-info.d.ts.map +1 -0
  207. package/dist/esm/shared/auth-info.js +2 -0
  208. package/dist/esm/shared/auth-info.js.map +1 -0
  209. package/dist/esm/shared/metadataUtils.d.ts +16 -0
  210. package/dist/esm/shared/metadataUtils.d.ts.map +1 -0
  211. package/dist/esm/shared/metadataUtils.js +22 -0
  212. package/dist/esm/shared/metadataUtils.js.map +1 -0
  213. package/dist/esm/shared/protocol.d.ts +443 -0
  214. package/dist/esm/shared/protocol.d.ts.map +1 -0
  215. package/dist/esm/shared/protocol.js +1099 -0
  216. package/dist/esm/shared/protocol.js.map +1 -0
  217. package/dist/esm/shared/responseMessage.d.ts +45 -0
  218. package/dist/esm/shared/responseMessage.d.ts.map +1 -0
  219. package/dist/esm/shared/responseMessage.js +19 -0
  220. package/dist/esm/shared/responseMessage.js.map +1 -0
  221. package/dist/esm/shared/stdio.d.ts +13 -0
  222. package/dist/esm/shared/stdio.d.ts.map +1 -0
  223. package/dist/esm/shared/stdio.js +31 -0
  224. package/dist/esm/shared/stdio.js.map +1 -0
  225. package/dist/esm/shared/toolNameValidation.d.ts +31 -0
  226. package/dist/esm/shared/toolNameValidation.d.ts.map +1 -0
  227. package/dist/esm/shared/toolNameValidation.js +92 -0
  228. package/dist/esm/shared/toolNameValidation.js.map +1 -0
  229. package/dist/esm/shared/transport.d.ts +89 -0
  230. package/dist/esm/shared/transport.d.ts.map +1 -0
  231. package/dist/esm/shared/transport.js +39 -0
  232. package/dist/esm/shared/transport.js.map +1 -0
  233. package/dist/esm/shared/uriTemplate.d.ts +25 -0
  234. package/dist/esm/shared/uriTemplate.d.ts.map +1 -0
  235. package/dist/esm/shared/uriTemplate.js +239 -0
  236. package/dist/esm/shared/uriTemplate.js.map +1 -0
  237. package/dist/esm/spec.types.d.ts +2299 -0
  238. package/dist/esm/spec.types.d.ts.map +1 -0
  239. package/dist/esm/spec.types.js +24 -0
  240. package/dist/esm/spec.types.js.map +1 -0
  241. package/dist/esm/types.d.ts +8137 -0
  242. package/dist/esm/types.d.ts.map +1 -0
  243. package/dist/esm/types.js +2052 -0
  244. package/dist/esm/types.js.map +1 -0
  245. package/dist/esm/validation/ajv-provider.d.ts +53 -0
  246. package/dist/esm/validation/ajv-provider.d.ts.map +1 -0
  247. package/dist/esm/validation/ajv-provider.js +87 -0
  248. package/dist/esm/validation/ajv-provider.js.map +1 -0
  249. package/dist/esm/validation/cfworker-provider.d.ts +51 -0
  250. package/dist/esm/validation/cfworker-provider.d.ts.map +1 -0
  251. package/dist/esm/validation/cfworker-provider.js +65 -0
  252. package/dist/esm/validation/cfworker-provider.js.map +1 -0
  253. package/dist/esm/validation/index.d.ts +29 -0
  254. package/dist/esm/validation/index.d.ts.map +1 -0
  255. package/dist/esm/validation/index.js +29 -0
  256. package/dist/esm/validation/index.js.map +1 -0
  257. package/dist/esm/validation/types.d.ts +65 -0
  258. package/dist/esm/validation/types.d.ts.map +1 -0
  259. package/dist/esm/validation/types.js +2 -0
  260. package/dist/esm/validation/types.js.map +1 -0
  261. 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