[ide-mode] Stream notifications when the active file changes (#4147)
Co-authored-by: Scott Densmore <scottdensmore@mac.com>
This commit is contained in:
parent
b09bc66560
commit
7effdad3e2
|
@ -8,28 +8,81 @@ import * as vscode from 'vscode';
|
||||||
import { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js';
|
import { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js';
|
||||||
import { StreamableHTTPServerTransport } from '@modelcontextprotocol/sdk/server/streamableHttp.js';
|
import { StreamableHTTPServerTransport } from '@modelcontextprotocol/sdk/server/streamableHttp.js';
|
||||||
import express, { Request, Response } from 'express';
|
import express, { Request, Response } from 'express';
|
||||||
|
import { randomUUID } from 'node:crypto';
|
||||||
|
import {
|
||||||
|
isInitializeRequest,
|
||||||
|
type JSONRPCNotification,
|
||||||
|
} from '@modelcontextprotocol/sdk/types.js';
|
||||||
|
|
||||||
export async function startIDEServer(_context: vscode.ExtensionContext) {
|
export async function startIDEServer(context: vscode.ExtensionContext) {
|
||||||
const app = express();
|
const app = express();
|
||||||
app.use(express.json());
|
app.use(express.json());
|
||||||
|
|
||||||
const mcpServer = createMcpServer();
|
const transports: { [sessionId: string]: StreamableHTTPServerTransport } = {};
|
||||||
const transport = new StreamableHTTPServerTransport({
|
|
||||||
sessionIdGenerator: undefined,
|
|
||||||
enableJsonResponse: true,
|
|
||||||
});
|
|
||||||
|
|
||||||
mcpServer.connect(transport);
|
const disposable = vscode.window.onDidChangeActiveTextEditor((editor) => {
|
||||||
|
const filePath = editor ? editor.document.uri.fsPath : null;
|
||||||
|
const notification: JSONRPCNotification = {
|
||||||
|
jsonrpc: '2.0',
|
||||||
|
method: 'ide/activeFileChanged',
|
||||||
|
params: { filePath },
|
||||||
|
};
|
||||||
|
for (const transport of Object.values(transports)) {
|
||||||
|
transport.send(notification);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
context.subscriptions.push(disposable);
|
||||||
|
|
||||||
app.post('/mcp', async (req: Request, res: Response) => {
|
app.post('/mcp', async (req: Request, res: Response) => {
|
||||||
console.log('Received MCP request:', req.body);
|
const sessionId = req.headers['mcp-session-id'] as string | undefined;
|
||||||
|
let transport: StreamableHTTPServerTransport;
|
||||||
|
|
||||||
|
if (sessionId && transports[sessionId]) {
|
||||||
|
transport = transports[sessionId];
|
||||||
|
} else if (!sessionId && isInitializeRequest(req.body)) {
|
||||||
|
transport = new StreamableHTTPServerTransport({
|
||||||
|
sessionIdGenerator: () => randomUUID(),
|
||||||
|
onsessioninitialized: (newSessionId) => {
|
||||||
|
transports[newSessionId] = transport;
|
||||||
|
const editor = vscode.window.activeTextEditor;
|
||||||
|
const filePath = editor ? editor.document.uri.fsPath : null;
|
||||||
|
const notification: JSONRPCNotification = {
|
||||||
|
jsonrpc: '2.0',
|
||||||
|
method: 'ide/activeFileChanged',
|
||||||
|
params: { filePath },
|
||||||
|
};
|
||||||
|
transport.send(notification);
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
transport.onclose = () => {
|
||||||
|
if (transport.sessionId) {
|
||||||
|
delete transports[transport.sessionId];
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const server = createMcpServer();
|
||||||
|
server.connect(transport);
|
||||||
|
} else {
|
||||||
|
res.status(400).json({
|
||||||
|
jsonrpc: '2.0',
|
||||||
|
error: {
|
||||||
|
code: -32000,
|
||||||
|
message:
|
||||||
|
'Bad Request: No valid session ID provided for non-initialize request.',
|
||||||
|
},
|
||||||
|
id: null,
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
await transport.handleRequest(req, res, req.body);
|
await transport.handleRequest(req, res, req.body);
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('Error handling MCP request:', error);
|
console.error('Error handling MCP request:', error);
|
||||||
if (!res.headersSent) {
|
if (!res.headersSent) {
|
||||||
res.status(500).json({
|
res.status(500).json({
|
||||||
jsonrpc: '2.0',
|
jsonrpc: '2.0' as const,
|
||||||
error: {
|
error: {
|
||||||
code: -32603,
|
code: -32603,
|
||||||
message: 'Internal server error',
|
message: 'Internal server error',
|
||||||
|
@ -40,15 +93,29 @@ export async function startIDEServer(_context: vscode.ExtensionContext) {
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
// Handle GET requests for SSE streams
|
const handleSessionRequest = async (req: Request, res: Response) => {
|
||||||
app.get('/mcp', async (req: Request, res: Response) => {
|
const sessionId = req.headers['mcp-session-id'] as string | undefined;
|
||||||
res.status(405).set('Allow', 'POST').send('Method Not Allowed');
|
if (!sessionId || !transports[sessionId]) {
|
||||||
});
|
res.status(400).send('Invalid or missing session ID');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const transport = transports[sessionId];
|
||||||
|
try {
|
||||||
|
await transport.handleRequest(req, res);
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Error handling MCP GET request:', error);
|
||||||
|
if (!res.headersSent) {
|
||||||
|
res.status(400).send('Bad Request');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
app.get('/mcp', handleSessionRequest);
|
||||||
|
|
||||||
// Start the server
|
|
||||||
// TODO(#3918): Generate dynamically and write to env variable
|
// TODO(#3918): Generate dynamically and write to env variable
|
||||||
const PORT = 3000;
|
const PORT = 3000;
|
||||||
app.listen(PORT, (error) => {
|
app.listen(PORT, (error?: Error) => {
|
||||||
if (error) {
|
if (error) {
|
||||||
console.error('Failed to start server:', error);
|
console.error('Failed to start server:', error);
|
||||||
vscode.window.showErrorMessage(
|
vscode.window.showErrorMessage(
|
||||||
|
@ -60,10 +127,13 @@ export async function startIDEServer(_context: vscode.ExtensionContext) {
|
||||||
}
|
}
|
||||||
|
|
||||||
const createMcpServer = () => {
|
const createMcpServer = () => {
|
||||||
const server = new McpServer({
|
const server = new McpServer(
|
||||||
name: 'vscode-ide-server',
|
{
|
||||||
version: '1.0.0',
|
name: 'gemini-cli-companion-mcp-server',
|
||||||
});
|
version: '1.0.0',
|
||||||
|
},
|
||||||
|
{ capabilities: { logging: {} } },
|
||||||
|
);
|
||||||
server.registerTool(
|
server.registerTool(
|
||||||
'getActiveFile',
|
'getActiveFile',
|
||||||
{
|
{
|
||||||
|
|
Loading…
Reference in New Issue