diff --git a/package-lock.json b/package-lock.json index 73f1cbba..8338e3c4 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,12 +1,12 @@ { "name": "@modelcontextprotocol/sdk", - "version": "1.7.0", + "version": "1.8.0", "lockfileVersion": 3, "requires": true, "packages": { "": { "name": "@modelcontextprotocol/sdk", - "version": "1.7.0", + "version": "1.8.0", "license": "MIT", "dependencies": { "content-type": "^1.0.5", diff --git a/src/server/mcp.ts b/src/server/mcp.ts index 8f4a909c..d4382155 100644 --- a/src/server/mcp.ts +++ b/src/server/mcp.ts @@ -43,12 +43,22 @@ import { UriTemplate, Variables } from "../shared/uriTemplate.js"; import { RequestHandlerExtra } from "../shared/protocol.js"; import { Transport } from "../shared/transport.js"; +type RenderApi = { + resource: McpServer["resource"]; + tool: McpServer["tool"]; + prompt: McpServer["prompt"]; +}; + +type McpServerOptions = ServerOptions & { + render?: (api: RenderApi, args: T) => void | Promise; +}; + /** * High-level MCP server that provides a simpler API for working with resources, tools, and prompts. * For advanced usage (like sending notifications or setting custom request handlers), use the underlying * Server instance available via the `server` property. */ -export class McpServer { +export class McpServer | undefined = undefined> { /** * The underlying Server instance, useful for advanced operations like sending notifications. */ @@ -61,8 +71,17 @@ export class McpServer { private _registeredTools: { [name: string]: RegisteredTool } = {}; private _registeredPrompts: { [name: string]: RegisteredPrompt } = {}; - constructor(serverInfo: Implementation, options?: ServerOptions) { + private readonly _renderFunction?: ( + api: RenderApi, + args: RenderArgs, + ) => void | Promise; + private readonly _locked: boolean; + private _isConnected: boolean = true; + + constructor(serverInfo: Implementation, options?: McpServerOptions) { this.server = new Server(serverInfo, options); + this._renderFunction = options?.render; + this._locked = Boolean(this._renderFunction); } /** @@ -71,6 +90,7 @@ export class McpServer { * 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. */ async connect(transport: Transport): Promise { + this._isConnected = true return await this.server.connect(transport); } @@ -87,7 +107,7 @@ export class McpServer { if (this._toolHandlersInitialized) { return; } - + this.server.assertCanSetRequestHandler( ListToolsRequestSchema.shape.method.value, ); @@ -96,7 +116,9 @@ export class McpServer { ); this.server.registerCapabilities({ - tools: {}, + tools: { + listChanged: true + }, }); this.server.setRequestHandler( @@ -285,7 +307,9 @@ export class McpServer { ); this.server.registerCapabilities({ - resources: {}, + resources: { + listChanged: true + }, }); this.server.setRequestHandler( @@ -366,7 +390,7 @@ export class McpServer { ); this.setCompletionRequestHandler(); - + this._resourceHandlersInitialized = true; } @@ -385,7 +409,9 @@ export class McpServer { ); this.server.registerCapabilities({ - prompts: {}, + prompts: { + listChanged: true + }, }); this.server.setRequestHandler( @@ -438,17 +464,98 @@ export class McpServer { ); this.setCompletionRequestHandler(); - + this._promptHandlersInitialized = true; } + /** + * Clears existing resources, tools, and prompts, then runs the configured `render` function + * to define a new set based on the provided arguments. + * If the set of registered items changes compared to the previous state (and it's not the first render), + * appropriate `/listChanged` notifications are sent. + * + * @param args Arguments to pass to the configured `render` function. + * @throws Error if no `render` function was provided in the constructor options. + */ + async render(args: RenderArgs): Promise { + if (!this._renderFunction) { + throw new Error( + "Cannot call render(). No render function was provided during McpServer initialization.", + ); + } + + // --- 1. Prepare for new render --- + const newResources: { [uri: string]: RegisteredResource } = {}; + const newResourceTemplates: { [name: string]: RegisteredResourceTemplate } = + {}; + const newTools: { [name: string]: RegisteredTool } = {}; + const newPrompts: { [name: string]: RegisteredPrompt } = {}; + + // --- 2. Create temporary registration API for the render function --- + // These functions capture the definitions into the 'new*' objects above. + // They mirror the public API but don't check for locking or emit events immediately. + const renderApi: RenderApi = { + resource: ( + name: string, + uriOrTemplate: string | ResourceTemplate, + ...rest: unknown[] + ): void => { + addResources(newResources, newResourceTemplates, name, uriOrTemplate, ...rest) + }, + tool: (name: string, ...rest: unknown[]): void => { + addTool(newTools, name, ...rest) + }, + prompt: (name: string, ...rest: unknown[]): void => { + addPrompt(newPrompts, name, ...rest) + }, + }; + + // --- 3. Execute the user's render function --- + this._renderFunction(renderApi, args) + + // --- 4. Compare old state with new state --- + const toolsChanged = haveKeysChanged(this._registeredTools, newTools); + const promptsChanged = haveKeysChanged(this._registeredPrompts, newPrompts); + const resourcesChanged = haveKeysChanged( + { + ...this._registeredResources, + ...mapKeys(this._registeredResourceTemplates, (t) => t.resourceTemplate.uriTemplate.toString()) // Use template URI for comparison consistency if needed, or just name + }, + { + ...newResources, + ...mapKeys(newResourceTemplates, (t) => t.resourceTemplate.uriTemplate.toString()) + } + ) || haveKeysChanged(this._registeredResourceTemplates, newResourceTemplates); // Also check template names directly + + // --- 5. Always update internal state (currently we're not emitting events for changes in parameters or descriptions + // of tools, but we should at least store the new values + this._registeredTools = newTools; + this._registeredPrompts = newPrompts; + this._registeredResources = newResources; + this._registeredResourceTemplates = newResourceTemplates; + + // Ensure handlers are set up + this.setToolRequestHandlers(); + this.setPromptRequestHandlers(); + this.setResourceRequestHandlers(); + + // Emit change events if we have a transport to emit to + if (!this._isConnected) { + if (toolsChanged) this.server.sendToolListChanged() + if (promptsChanged) this.server.sendPromptListChanged() + if (resourcesChanged) this.server.sendResourceListChanged() + } + } + /** * Registers a resource `name` at a fixed URI, which will use the given callback to respond to read requests. + * @throws Error if the server is locked. */ resource(name: string, uri: string, readCallback: ReadResourceCallback): void; /** * Registers a resource `name` at a fixed URI with metadata, which will use the given callback to respond to read requests. + * @throws Error if the server is locked. */ resource( name: string, @@ -459,6 +566,7 @@ export class McpServer { /** * Registers a resource `name` with a template pattern, which will use the given callback to respond to read requests. + * @throws Error if the server is locked. */ resource( name: string, @@ -468,6 +576,7 @@ export class McpServer { /** * Registers a resource `name` with a template pattern and metadata, which will use the given callback to respond to read requests. + * @throws Error if the server is locked. */ resource( name: string, @@ -481,52 +590,33 @@ export class McpServer { uriOrTemplate: string | ResourceTemplate, ...rest: unknown[] ): void { - let metadata: ResourceMetadata | undefined; - if (typeof rest[0] === "object") { - metadata = rest.shift() as ResourceMetadata; + if (this._locked) { + throw new Error( + "Server is locked. Resources can only be registered via the render() method.", + ); } - const readCallback = rest[0] as - | ReadResourceCallback - | ReadResourceTemplateCallback; - - if (typeof uriOrTemplate === "string") { - if (this._registeredResources[uriOrTemplate]) { - throw new Error(`Resource ${uriOrTemplate} is already registered`); - } - - this._registeredResources[uriOrTemplate] = { - name, - metadata, - readCallback: readCallback as ReadResourceCallback, - }; - } else { - if (this._registeredResourceTemplates[name]) { - throw new Error(`Resource template ${name} is already registered`); - } + addResources(this._registeredResources, this._registeredResourceTemplates, name, uriOrTemplate, ...rest) - this._registeredResourceTemplates[name] = { - resourceTemplate: uriOrTemplate, - metadata, - readCallback: readCallback as ReadResourceTemplateCallback, - }; - } - - this.setResourceRequestHandlers(); + this.setResourceRequestHandlers() + this.server.sendResourceListChanged() } /** * Registers a zero-argument tool `name`, which will run the given function when the client calls it. + * @throws Error if the server is locked. */ tool(name: string, cb: ToolCallback): void; /** * Registers a zero-argument tool `name` (with a description) which will run the given function when the client calls it. + * @throws Error if the server is locked. */ tool(name: string, description: string, cb: ToolCallback): void; /** * Registers a tool `name` accepting the given arguments, which must be an object containing named properties associated with Zod schemas. When the client calls it, the function will be run with the parsed and validated arguments. + * @throws Error if the server is locked. */ tool( name: string, @@ -536,6 +626,7 @@ export class McpServer { /** * Registers a tool `name` (with a description) accepting the given arguments, which must be an object containing named properties associated with Zod schemas. When the client calls it, the function will be run with the parsed and validated arguments. + * @throws Error if the server is locked. */ tool( name: string, @@ -545,43 +636,33 @@ export class McpServer { ): void; tool(name: string, ...rest: unknown[]): void { - if (this._registeredTools[name]) { - throw new Error(`Tool ${name} is already registered`); - } - - let description: string | undefined; - if (typeof rest[0] === "string") { - description = rest.shift() as string; - } - - let paramsSchema: ZodRawShape | undefined; - if (rest.length > 1) { - paramsSchema = rest.shift() as ZodRawShape; + if (this._locked) { + throw new Error( + "Server is locked. Tools can only be registered via the render() method.", + ); } - const cb = rest[0] as ToolCallback; - this._registeredTools[name] = { - description, - inputSchema: - paramsSchema === undefined ? undefined : z.object(paramsSchema), - callback: cb, - }; + addTool(this._registeredTools, name, ...rest) this.setToolRequestHandlers(); + this.server.sendToolListChanged(); } /** * Registers a zero-argument prompt `name`, which will run the given function when the client calls it. + * @throws Error if the server is locked. */ prompt(name: string, cb: PromptCallback): void; /** * Registers a zero-argument prompt `name` (with a description) which will run the given function when the client calls it. + * @throws Error if the server is locked. */ prompt(name: string, description: string, cb: PromptCallback): void; /** * Registers a prompt `name` accepting the given arguments, which must be an object containing named properties associated with Zod schemas. When the client calls it, the function will be run with the parsed and validated arguments. + * @throws Error if the server is locked. */ prompt( name: string, @@ -591,6 +672,7 @@ export class McpServer { /** * Registers a prompt `name` (with a description) accepting the given arguments, which must be an object containing named properties associated with Zod schemas. When the client calls it, the function will be run with the parsed and validated arguments. + * @throws Error if the server is locked. */ prompt( name: string, @@ -600,31 +682,129 @@ export class McpServer { ): void; prompt(name: string, ...rest: unknown[]): void { - if (this._registeredPrompts[name]) { - throw new Error(`Prompt ${name} is already registered`); + if (this._locked) { + throw new Error( + "Server is locked. Prompts can only be registered via the render() method.", + ); } - let description: string | undefined; - if (typeof rest[0] === "string") { - description = rest.shift() as string; + addPrompt(this._registeredPrompts, name, ...rest) + + this.setPromptRequestHandlers(); + this.server.sendPromptListChanged() + } +} + +function addResources(resources: { [p: string]: RegisteredResource }, resourceTemplates: { + [p: string]: RegisteredResourceTemplate +}, name: string, uriOrTemplate: string | ResourceTemplate, ...rest: unknown[]) { + let metadata: ResourceMetadata | undefined; + // Check if the first rest arg is metadata (object, not function) + if (rest.length > 1 && typeof rest[0] === "object" && rest[0] !== null && !(rest[0] instanceof Function)) { + metadata = rest.shift() as ResourceMetadata; + } + + const readCallback = rest[0] as + | ReadResourceCallback + | ReadResourceTemplateCallback; + + if (typeof uriOrTemplate === "string") { + if (resources[uriOrTemplate]) { + throw new Error(`Resource ${uriOrTemplate} is already registered`); } - let argsSchema: PromptArgsRawShape | undefined; - if (rest.length > 1) { - argsSchema = rest.shift() as PromptArgsRawShape; + resources[uriOrTemplate] = { + name, + metadata, + readCallback: readCallback as ReadResourceCallback, + }; + } else { + if (resourceTemplates[name]) { + throw new Error(`Resource template ${name} is already registered`); } - const cb = rest[0] as PromptCallback; - this._registeredPrompts[name] = { - description, - argsSchema: argsSchema === undefined ? undefined : z.object(argsSchema), - callback: cb, + resourceTemplates[name] = { + resourceTemplate: uriOrTemplate, + metadata, + readCallback: readCallback as ReadResourceTemplateCallback, }; + } +} - this.setPromptRequestHandlers(); +function addTool(tools: { [p: string]: RegisteredTool }, name: string, ...rest: unknown[]) { + let description: string | undefined; + if (typeof rest[0] === "string") { + description = rest.shift() as string; + } + + let paramsSchema: ZodRawShape | undefined; + if (rest.length > 1) { + paramsSchema = rest.shift() as ZodRawShape; } + + const cb = rest[0] as ToolCallback; + tools[name] = { + description, + inputSchema: + paramsSchema === undefined ? undefined : z.object(paramsSchema), + callback: cb, + }; } +function addPrompt(prompts: { [p: string]: RegisteredPrompt }, name: string, ...rest: unknown[]) { + let description: string | undefined; + if (typeof rest[0] === "string") { + description = rest.shift() as string; + } + + let argsSchema: PromptArgsRawShape | undefined; + if (rest.length > 1) { + argsSchema = rest.shift() as PromptArgsRawShape; + } + + const cb = rest[0] as PromptCallback; + prompts[name] = { + description, + argsSchema: argsSchema === undefined ? undefined : z.object(argsSchema), + callback: cb, + }; +} + +// --- Helper Function for Change Detection --- + +/** Checks if the keys of two objects are different. */ +function haveKeysChanged(oldObj: object, newObj: object): boolean { + const oldKeys = Object.keys(oldObj).sort(); + const newKeys = Object.keys(newObj).sort(); + + if (oldKeys.length !== newKeys.length) { + return true; + } + + for (let i = 0; i < oldKeys.length; i++) { + if (oldKeys[i] !== newKeys[i]) { + return true; + } + } + + return false; +} + +/** Helper to map object keys while preserving values. */ +function mapKeys(obj: Record, keyMapper: (value: V, key: string) => string): Record { + const result: Record = {}; + for(const key in obj) { + if (Object.prototype.hasOwnProperty.call(obj, key)) { + const newKey = keyMapper(obj[key], key); + result[newKey] = obj[key]; + } + } + return result; +} + + +// --- Constants and Type Definitions --- + /** * A callback to complete one variable within a resource template's URI template. */