-
Notifications
You must be signed in to change notification settings - Fork 436
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Mutations for McpServer
#247
base: main
Are you sure you want to change the base?
Conversation
4d0e10e
to
7a2352e
Compare
Do we need to manage adding / removing tools to the server itself, or just shape which ones are returned on a request? Ex: a server could register a different handler here: https://github.com/modelcontextprotocol/typescript-sdk/blob/main/src/server/mcp.ts#L105 this.server.setRequestHandler(ListToolsRequestSchema, async (): Promise<ListToolsResult> => {
let userState = await fetchUserStateFromDataStore();
let tools: Tool[] = this._registeredTools.filter((tool) => {
// arbitrary filter by userState
});
return {
tools: tools.map(
([name, tool]): Tool => {
return {
name,
description: tool.description,
inputSchema: tool.inputSchema
? (zodToJsonSchema(tool.inputSchema, {
strictUnions: true,
}) as Tool["inputSchema"])
: EMPTY_OBJECT_JSON_SCHEMA,
};
},
),
};
}); Note that the current version does not support async calls Edit: I think this ~aligns with your approach in #233 ? |
…method and accessors
… the render function is enough
… just the new pieces required in the McpServer api
7a2352e
to
7031a17
Compare
The only issue is making sure you're triggering the But yes, this PR is mainly about keeping the existing style of the const server = new McpServer({
name: "Dynamic Example",
version: "1.0.0"
});
server.tool(
"listMessages",
{ channel: z.string() },
async ({ channel }) => ({
content: [{ type: "text", text: await listMessages(channel) }]
})
);
server.tool(
"upgradeAuth",
{ permission: z.enum(["write', vadmin"])},
upgradeAuth
)
// Connect with the existing set of tools
const transport = new StdioServerTransport();
await server.connect(transport);
// Any mutations after connection result in `listChanged` notifications so the client knows to refresh
async function upgradeAuth({permission}) {
const { ok, err, previous } = await upgradeAuthAndStoreToken(permission)
if (!ok) return {content: [{ type: "text", text: `Error: ${err}` }]}
// If we previously had read-only access, we need to add 'putMessage' now we can use it
if (previous === "read") {
server.tool(
"putMessage",
{ channel: z.string(), message: z.string() },
async ({ channel, message }) => ({
content: [{ type: "text", text: await putMessage(channel, string) }]
})
);
}
// If we've just upgraded to 'write' permissions, we can still call 'upgradeAuth' but can only upgrade to 'admin'
if (permission === 'write') {
server.updateTool(
"upgradeAuth",
{ permission: z.enum(["admin"])}, // change param validation
upgradeAuth
)
} else {
// If we're on admin, we no longer have anywhere to upgrade to
server.removeTool("upgradeAuth")
}
} (I've added the above example to the README as part of this PR) But you're right, like in #233, we could build a much more reactive approach with tools being toggled on/off against the base class, which I think will be more maintainable in the long run. But I like using |
McpServer
is a great higher-level builder pattern for defining servers, but following on from #233 I feel it's a bit too limited to be used as a base for building other abstractions upon.This PR adds:
updateTool()
andremoveTool()
updateResource()
andremoveResource()
updatePrompt()
andremovePrompt()
Motivation and Context
This feels like a flexible but low-enough level addition to make it easier to start to build "dynamic" servers on top of the current class, without imposing any particular design decision.
How Has This Been Tested?
Added tests to
mcp.test.ts
Breaking Changes
Purely additive. The only change is that servers now, by default, report that they support
listChanged
notifications (since they do, now).Types of changes
Checklist
Additional context