mirror of
https://github.com/penpot/penpot.git
synced 2026-05-22 08:23:42 +00:00
258 lines
9.6 KiB
TypeScript
258 lines
9.6 KiB
TypeScript
import { McpServer } from "@modelcontextprotocol/sdk/server/mcp.js";
|
|
import { AsyncLocalStorage } from "async_hooks";
|
|
import { SSEServerTransport } from "@modelcontextprotocol/sdk/server/sse.js";
|
|
import { StreamableHTTPServerTransport } from "@modelcontextprotocol/sdk/server/streamableHttp.js";
|
|
import { ExecuteCodeTool } from "./tools/ExecuteCodeTool";
|
|
import { PluginBridge } from "./PluginBridge";
|
|
import { ConfigurationLoader } from "./ConfigurationLoader";
|
|
import { createLogger } from "./logger";
|
|
import { Tool } from "./Tool";
|
|
import { HighLevelOverviewTool } from "./tools/HighLevelOverviewTool";
|
|
import { PenpotApiInfoTool } from "./tools/PenpotApiInfoTool";
|
|
import { ExportShapeTool } from "./tools/ExportShapeTool";
|
|
import { ImportImageTool } from "./tools/ImportImageTool";
|
|
import { ReplServer } from "./ReplServer";
|
|
import { ApiDocs } from "./ApiDocs";
|
|
|
|
/**
|
|
* Session context for request-scoped data.
|
|
*/
|
|
export interface SessionContext {
|
|
userToken?: string;
|
|
}
|
|
|
|
export class PenpotMcpServer {
|
|
private readonly logger = createLogger("PenpotMcpServer");
|
|
private readonly server: McpServer;
|
|
private readonly tools: Map<string, Tool<any>>;
|
|
public readonly configLoader: ConfigurationLoader;
|
|
private app: any;
|
|
public readonly pluginBridge: PluginBridge;
|
|
private readonly replServer: ReplServer;
|
|
private apiDocs: ApiDocs;
|
|
|
|
/**
|
|
* Manages session-specific context, particularly user tokens for each request.
|
|
*/
|
|
private readonly sessionContext = new AsyncLocalStorage<SessionContext>();
|
|
|
|
private readonly transports = {
|
|
streamable: {} as Record<string, StreamableHTTPServerTransport>,
|
|
sse: {} as Record<string, { transport: SSEServerTransport; userToken?: string }>,
|
|
};
|
|
|
|
public readonly host: string;
|
|
public readonly port: number;
|
|
public readonly webSocketPort: number;
|
|
public readonly replPort: number;
|
|
|
|
constructor(private isMultiUser: boolean = false) {
|
|
// read port configuration from environment variables
|
|
this.host = process.env.PENPOT_MCP_SERVER_HOST ?? "0.0.0.0";
|
|
this.port = parseInt(process.env.PENPOT_MCP_SERVER_PORT ?? "4401", 10);
|
|
this.webSocketPort = parseInt(process.env.PENPOT_MCP_WEBSOCKET_PORT ?? "4402", 10);
|
|
this.replPort = parseInt(process.env.PENPOT_MCP_REPL_PORT ?? "4403", 10);
|
|
|
|
this.configLoader = new ConfigurationLoader(process.cwd());
|
|
this.apiDocs = new ApiDocs();
|
|
|
|
this.server = new McpServer(
|
|
{
|
|
name: "penpot-mcp-server",
|
|
version: "1.0.0",
|
|
},
|
|
{
|
|
instructions: this.getInitialInstructions(),
|
|
}
|
|
);
|
|
|
|
this.tools = new Map<string, Tool<any>>();
|
|
this.pluginBridge = new PluginBridge(this, this.webSocketPort);
|
|
this.replServer = new ReplServer(this.pluginBridge, this.replPort);
|
|
|
|
this.registerTools();
|
|
}
|
|
|
|
/**
|
|
* Indicates whether the server is running in multi-user mode,
|
|
* where user tokens are required for authentication.
|
|
*/
|
|
public isMultiUserMode(): boolean {
|
|
return this.isMultiUser;
|
|
}
|
|
|
|
/**
|
|
* Indicates whether the server is running in remote mode.
|
|
*
|
|
* In remote mode, the server is not assumed to be accessed only by a local user on the same machine,
|
|
* with corresponding limitations being enforced.
|
|
* Remote mode can be explicitly enabled by setting the environment variable PENPOT_MCP_REMOTE_MODE
|
|
* to "true". Enabling multi-user mode forces remote mode, regardless of the value of the environment
|
|
* variable.
|
|
*/
|
|
public isRemoteMode(): boolean {
|
|
const isRemoteModeRequested: boolean = process.env.PENPOT_MCP_REMOTE_MODE === "true";
|
|
return this.isMultiUserMode() || isRemoteModeRequested;
|
|
}
|
|
|
|
/**
|
|
* Indicates whether file system access is enabled for MCP tools.
|
|
* Access is enabled only in local mode, where the file system is assumed
|
|
* to belong to the user running the server locally.
|
|
*/
|
|
public isFileSystemAccessEnabled(): boolean {
|
|
return !this.isRemoteMode();
|
|
}
|
|
|
|
public getInitialInstructions(): string {
|
|
let instructions = this.configLoader.getInitialInstructions();
|
|
instructions = instructions.replace("$api_types", this.apiDocs.getTypeNames().join(", "));
|
|
return instructions;
|
|
}
|
|
|
|
/**
|
|
* Retrieves the current session context.
|
|
*
|
|
* @returns The session context for the current request, or undefined if not in a request context
|
|
*/
|
|
public getSessionContext(): SessionContext | undefined {
|
|
return this.sessionContext.getStore();
|
|
}
|
|
|
|
private registerTools(): void {
|
|
// Create relevant tool instances (depending on file system access)
|
|
const toolInstances: Tool<any>[] = [
|
|
new ExecuteCodeTool(this),
|
|
new HighLevelOverviewTool(this),
|
|
new PenpotApiInfoTool(this, this.apiDocs),
|
|
new ExportShapeTool(this), // tool adapts to file system access internally
|
|
];
|
|
if (this.isFileSystemAccessEnabled()) {
|
|
toolInstances.push(new ImportImageTool(this));
|
|
}
|
|
|
|
for (const tool of toolInstances) {
|
|
const toolName = tool.getToolName();
|
|
this.tools.set(toolName, tool);
|
|
|
|
// Register each tool with McpServer
|
|
this.logger.info(`Registering tool: ${toolName}`);
|
|
this.server.registerTool(
|
|
toolName,
|
|
{
|
|
description: tool.getToolDescription(),
|
|
inputSchema: tool.getInputSchema(),
|
|
},
|
|
async (args) => {
|
|
return tool.execute(args);
|
|
}
|
|
);
|
|
}
|
|
}
|
|
|
|
private setupHttpEndpoints(): void {
|
|
/**
|
|
* Modern Streamable HTTP connection endpoint
|
|
*/
|
|
this.app.all("/mcp", async (req: any, res: any) => {
|
|
const userToken = req.query.userToken as string | undefined;
|
|
|
|
await this.sessionContext.run({ userToken }, async () => {
|
|
const { randomUUID } = await import("node:crypto");
|
|
|
|
const sessionId = req.headers["mcp-session-id"] as string | undefined;
|
|
let transport: StreamableHTTPServerTransport;
|
|
|
|
if (sessionId && this.transports.streamable[sessionId]) {
|
|
transport = this.transports.streamable[sessionId];
|
|
} else {
|
|
transport = new StreamableHTTPServerTransport({
|
|
sessionIdGenerator: () => randomUUID(),
|
|
onsessioninitialized: (id: string) => {
|
|
this.transports.streamable[id] = transport;
|
|
},
|
|
});
|
|
|
|
transport.onclose = () => {
|
|
if (transport.sessionId) {
|
|
delete this.transports.streamable[transport.sessionId];
|
|
}
|
|
};
|
|
|
|
await this.server.connect(transport);
|
|
}
|
|
|
|
await transport.handleRequest(req, res, req.body);
|
|
});
|
|
});
|
|
|
|
/**
|
|
* Legacy SSE connection endpoint
|
|
*/
|
|
this.app.get("/sse", async (req: any, res: any) => {
|
|
const userToken = req.query.userToken as string | undefined;
|
|
|
|
await this.sessionContext.run({ userToken }, async () => {
|
|
const transport = new SSEServerTransport("/messages", res);
|
|
this.transports.sse[transport.sessionId] = { transport, userToken };
|
|
|
|
res.on("close", () => {
|
|
delete this.transports.sse[transport.sessionId];
|
|
});
|
|
|
|
await this.server.connect(transport);
|
|
});
|
|
});
|
|
|
|
/**
|
|
* SSE message POST endpoint (using previously established session)
|
|
*/
|
|
this.app.post("/messages", async (req: any, res: any) => {
|
|
const sessionId = req.query.sessionId as string;
|
|
const session = this.transports.sse[sessionId];
|
|
|
|
if (session) {
|
|
await this.sessionContext.run({ userToken: session.userToken }, async () => {
|
|
await session.transport.handlePostMessage(req, res, req.body);
|
|
});
|
|
} else {
|
|
res.status(400).send("No transport found for sessionId");
|
|
}
|
|
});
|
|
}
|
|
|
|
async start(): Promise<void> {
|
|
const { default: express } = await import("express");
|
|
this.app = express();
|
|
this.app.use(express.json());
|
|
|
|
this.setupHttpEndpoints();
|
|
|
|
return new Promise((resolve) => {
|
|
this.app.listen(this.port, this.host, async () => {
|
|
this.logger.info(`Multi-user mode: ${this.isMultiUserMode()}`);
|
|
this.logger.info(`Remote mode: ${this.isRemoteMode()}`);
|
|
this.logger.info(`Modern Streamable HTTP endpoint: http://${this.host}:${this.port}/mcp`);
|
|
this.logger.info(`Legacy SSE endpoint: http://${this.host}:${this.port}/sse`);
|
|
this.logger.info(`WebSocket server URL: ws://${this.host}:${this.webSocketPort}`);
|
|
|
|
// start the REPL server
|
|
await this.replServer.start();
|
|
|
|
resolve();
|
|
});
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Stops the MCP server and associated services.
|
|
*
|
|
* Gracefully shuts down the REPL server and other components.
|
|
*/
|
|
public async stop(): Promise<void> {
|
|
this.logger.info("Stopping Penpot MCP Server...");
|
|
await this.replServer.stop();
|
|
this.logger.info("Penpot MCP Server stopped");
|
|
}
|
|
}
|