2025-03-21 10:58:58 -07:00
|
|
|
/**
|
|
|
|
* Copyright (c) Microsoft Corporation.
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2025-03-25 14:46:39 -07:00
|
|
|
import { Server } from '@modelcontextprotocol/sdk/server/index.js';
|
2025-04-28 16:14:16 -07:00
|
|
|
import { CallToolRequestSchema, ListToolsRequestSchema } from '@modelcontextprotocol/sdk/types.js';
|
2025-04-22 13:24:38 +02:00
|
|
|
import { zodToJsonSchema } from 'zod-to-json-schema';
|
2025-03-21 10:58:58 -07:00
|
|
|
|
2025-04-30 23:06:56 +02:00
|
|
|
import { Context } from './context.js';
|
2025-03-21 10:58:58 -07:00
|
|
|
|
2025-04-30 23:06:56 +02:00
|
|
|
import type { Tool } from './tools/tool.js';
|
|
|
|
import type { Config } from '../config.js';
|
2025-03-21 10:58:58 -07:00
|
|
|
|
2025-04-28 16:14:16 -07:00
|
|
|
type MCPServerOptions = {
|
2025-03-26 15:02:45 -07:00
|
|
|
name: string;
|
|
|
|
version: string;
|
|
|
|
tools: Tool[];
|
|
|
|
};
|
|
|
|
|
2025-04-28 16:14:16 -07:00
|
|
|
export function createServerWithTools(serverOptions: MCPServerOptions, config: Config): Server {
|
|
|
|
const { name, version, tools } = serverOptions;
|
|
|
|
const context = new Context(tools, config);
|
2025-03-25 14:46:39 -07:00
|
|
|
const server = new Server({ name, version }, {
|
|
|
|
capabilities: {
|
|
|
|
tools: {},
|
|
|
|
}
|
|
|
|
});
|
2025-03-21 10:58:58 -07:00
|
|
|
|
2025-03-25 14:46:39 -07:00
|
|
|
server.setRequestHandler(ListToolsRequestSchema, async () => {
|
2025-04-22 13:24:38 +02:00
|
|
|
return {
|
|
|
|
tools: tools.map(tool => ({
|
|
|
|
name: tool.schema.name,
|
|
|
|
description: tool.schema.description,
|
|
|
|
inputSchema: zodToJsonSchema(tool.schema.inputSchema)
|
|
|
|
})),
|
|
|
|
};
|
2025-03-25 14:46:39 -07:00
|
|
|
});
|
2025-03-21 10:58:58 -07:00
|
|
|
|
2025-03-25 14:46:39 -07:00
|
|
|
server.setRequestHandler(CallToolRequestSchema, async request => {
|
2025-04-29 18:48:52 +02:00
|
|
|
const errorResult = (...messages: string[]) => ({
|
|
|
|
content: [{ type: 'text', text: messages.join('\n') }],
|
|
|
|
isError: true,
|
|
|
|
});
|
2025-03-25 14:46:39 -07:00
|
|
|
const tool = tools.find(tool => tool.schema.name === request.params.name);
|
2025-04-29 18:48:52 +02:00
|
|
|
if (!tool)
|
|
|
|
return errorResult(`Tool "${request.params.name}" not found`);
|
|
|
|
|
2025-03-21 10:58:58 -07:00
|
|
|
|
2025-04-16 15:21:45 -07:00
|
|
|
const modalStates = context.modalStates().map(state => state.type);
|
2025-04-29 18:48:52 +02:00
|
|
|
if (tool.clearsModalState && !modalStates.includes(tool.clearsModalState))
|
|
|
|
return errorResult(`The tool "${request.params.name}" can only be used when there is related modal state present.`, ...context.modalStatesMarkdown());
|
|
|
|
if (!tool.clearsModalState && modalStates.length)
|
|
|
|
return errorResult(`Tool "${request.params.name}" does not handle the modal state.`, ...context.modalStatesMarkdown());
|
2025-04-16 15:21:45 -07:00
|
|
|
|
2025-03-25 14:46:39 -07:00
|
|
|
try {
|
2025-04-16 19:36:48 -07:00
|
|
|
return await context.run(tool, request.params.arguments);
|
2025-03-25 14:46:39 -07:00
|
|
|
} catch (error) {
|
2025-04-29 18:48:52 +02:00
|
|
|
return errorResult(String(error));
|
2025-03-25 14:46:39 -07:00
|
|
|
}
|
|
|
|
});
|
2025-03-21 10:58:58 -07:00
|
|
|
|
2025-03-27 16:50:43 -07:00
|
|
|
const oldClose = server.close.bind(server);
|
|
|
|
|
2025-03-25 14:46:39 -07:00
|
|
|
server.close = async () => {
|
2025-03-27 16:50:43 -07:00
|
|
|
await oldClose();
|
2025-03-25 14:46:39 -07:00
|
|
|
await context.close();
|
|
|
|
};
|
2025-03-21 10:58:58 -07:00
|
|
|
|
2025-03-25 14:46:39 -07:00
|
|
|
return server;
|
2025-03-21 10:58:58 -07:00
|
|
|
}
|
2025-03-28 13:24:45 -07:00
|
|
|
|
|
|
|
export class ServerList {
|
|
|
|
private _servers: Server[] = [];
|
2025-04-15 16:39:52 -07:00
|
|
|
private _serverFactory: () => Promise<Server>;
|
2025-03-28 13:24:45 -07:00
|
|
|
|
2025-04-15 16:39:52 -07:00
|
|
|
constructor(serverFactory: () => Promise<Server>) {
|
2025-03-28 13:24:45 -07:00
|
|
|
this._serverFactory = serverFactory;
|
|
|
|
}
|
|
|
|
|
|
|
|
async create() {
|
2025-04-15 16:39:52 -07:00
|
|
|
const server = await this._serverFactory();
|
2025-03-28 13:24:45 -07:00
|
|
|
this._servers.push(server);
|
|
|
|
return server;
|
|
|
|
}
|
|
|
|
|
|
|
|
async close(server: Server) {
|
|
|
|
const index = this._servers.indexOf(server);
|
|
|
|
if (index !== -1)
|
|
|
|
this._servers.splice(index, 1);
|
|
|
|
await server.close();
|
|
|
|
}
|
|
|
|
|
|
|
|
async closeAll() {
|
|
|
|
await Promise.all(this._servers.map(server => server.close()));
|
|
|
|
}
|
|
|
|
}
|