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-27 18:23:30 +01:00
|
|
|
import http from 'http';
|
2025-03-26 15:02:45 -07:00
|
|
|
import fs from 'fs';
|
|
|
|
import os from 'os';
|
|
|
|
import path from 'path';
|
|
|
|
|
2025-03-21 10:58:58 -07:00
|
|
|
import { program } from 'commander';
|
2025-03-25 14:46:39 -07:00
|
|
|
import { StdioServerTransport } from '@modelcontextprotocol/sdk/server/stdio.js';
|
2025-03-27 18:23:30 +01:00
|
|
|
import { SSEServerTransport } from '@modelcontextprotocol/sdk/server/sse.js';
|
|
|
|
|
2025-03-21 10:58:58 -07:00
|
|
|
|
2025-03-25 14:46:39 -07:00
|
|
|
import { createServer } from './index';
|
2025-03-28 13:24:45 -07:00
|
|
|
import { ServerList } from './server';
|
2025-03-21 10:58:58 -07:00
|
|
|
|
2025-03-25 14:46:39 -07:00
|
|
|
import type { LaunchOptions } from 'playwright';
|
2025-03-27 18:23:30 +01:00
|
|
|
import assert from 'assert';
|
2025-03-21 10:58:58 -07:00
|
|
|
|
|
|
|
const packageJSON = require('../package.json');
|
|
|
|
|
|
|
|
program
|
|
|
|
.version('Version ' + packageJSON.version)
|
|
|
|
.name(packageJSON.name)
|
|
|
|
.option('--headless', 'Run browser in headless mode, headed by default')
|
2025-03-26 15:02:45 -07:00
|
|
|
.option('--user-data-dir <path>', 'Path to the user data directory')
|
2025-03-21 10:58:58 -07:00
|
|
|
.option('--vision', 'Run server that uses screenshots (Aria snapshots are used by default)')
|
2025-03-27 18:23:30 +01:00
|
|
|
.option('--port <port>', 'Port to listen on for SSE transport.')
|
2025-03-30 09:05:58 -07:00
|
|
|
.option('--cdp-endpoint <endpoint>', 'CDP endpoint to connect to.')
|
2025-03-21 10:58:58 -07:00
|
|
|
.action(async options => {
|
|
|
|
const launchOptions: LaunchOptions = {
|
|
|
|
headless: !!options.headless,
|
2025-03-26 15:02:45 -07:00
|
|
|
channel: 'chrome',
|
2025-03-21 10:58:58 -07:00
|
|
|
};
|
2025-03-28 13:24:45 -07:00
|
|
|
const userDataDir = options.userDataDir ?? await createUserDataDir();
|
|
|
|
const serverList = new ServerList(() => createServer({
|
|
|
|
userDataDir,
|
2025-03-26 15:02:45 -07:00
|
|
|
launchOptions,
|
2025-03-27 15:26:37 +01:00
|
|
|
vision: !!options.vision,
|
2025-03-30 09:05:58 -07:00
|
|
|
cdpEndpoint: options.cdpEndpoint,
|
2025-03-28 13:24:45 -07:00
|
|
|
}));
|
|
|
|
setupExitWatchdog(serverList);
|
2025-03-25 14:46:39 -07:00
|
|
|
|
2025-03-27 18:23:30 +01:00
|
|
|
if (options.port) {
|
2025-03-28 13:24:45 -07:00
|
|
|
startSSEServer(+options.port, serverList);
|
2025-03-27 18:23:30 +01:00
|
|
|
} else {
|
2025-03-28 13:24:45 -07:00
|
|
|
const server = await serverList.create();
|
|
|
|
await server.connect(new StdioServerTransport());
|
2025-03-27 18:23:30 +01:00
|
|
|
}
|
2025-03-21 10:58:58 -07:00
|
|
|
});
|
|
|
|
|
2025-03-28 13:24:45 -07:00
|
|
|
function setupExitWatchdog(serverList: ServerList) {
|
2025-03-21 10:58:58 -07:00
|
|
|
process.stdin.on('close', async () => {
|
|
|
|
setTimeout(() => process.exit(0), 15000);
|
2025-03-28 13:24:45 -07:00
|
|
|
await serverList.closeAll();
|
2025-03-21 10:58:58 -07:00
|
|
|
process.exit(0);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
program.parse(process.argv);
|
2025-03-26 15:02:45 -07:00
|
|
|
|
2025-03-28 13:24:45 -07:00
|
|
|
async function createUserDataDir() {
|
2025-03-26 15:02:45 -07:00
|
|
|
let cacheDirectory: string;
|
|
|
|
if (process.platform === 'linux')
|
|
|
|
cacheDirectory = process.env.XDG_CACHE_HOME || path.join(os.homedir(), '.cache');
|
|
|
|
else if (process.platform === 'darwin')
|
|
|
|
cacheDirectory = path.join(os.homedir(), 'Library', 'Caches');
|
|
|
|
else if (process.platform === 'win32')
|
|
|
|
cacheDirectory = process.env.LOCALAPPDATA || path.join(os.homedir(), 'AppData', 'Local');
|
|
|
|
else
|
|
|
|
throw new Error('Unsupported platform: ' + process.platform);
|
2025-03-26 16:03:46 -07:00
|
|
|
const result = path.join(cacheDirectory, 'ms-playwright', 'mcp-chrome-profile');
|
2025-03-26 15:02:45 -07:00
|
|
|
await fs.promises.mkdir(result, { recursive: true });
|
|
|
|
return result;
|
|
|
|
}
|
2025-03-28 13:24:45 -07:00
|
|
|
|
|
|
|
async function startSSEServer(port: number, serverList: ServerList) {
|
|
|
|
const sessions = new Map<string, SSEServerTransport>();
|
|
|
|
const httpServer = http.createServer(async (req, res) => {
|
|
|
|
if (req.method === 'POST') {
|
2025-03-28 13:41:08 -07:00
|
|
|
const searchParams = new URL(`http://localhost${req.url}`).searchParams;
|
|
|
|
const sessionId = searchParams.get('sessionId');
|
2025-03-28 13:24:45 -07:00
|
|
|
if (!sessionId) {
|
|
|
|
res.statusCode = 400;
|
|
|
|
res.end('Missing sessionId');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const transport = sessions.get(sessionId);
|
|
|
|
if (!transport) {
|
|
|
|
res.statusCode = 404;
|
|
|
|
res.end('Session not found');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
await transport.handlePostMessage(req, res);
|
|
|
|
return;
|
|
|
|
} else if (req.method === 'GET') {
|
|
|
|
const transport = new SSEServerTransport('/sse', res);
|
|
|
|
sessions.set(transport.sessionId, transport);
|
|
|
|
const server = await serverList.create();
|
|
|
|
res.on('close', () => {
|
|
|
|
sessions.delete(transport.sessionId);
|
|
|
|
serverList.close(server).catch(e => console.error(e));
|
|
|
|
});
|
|
|
|
await server.connect(transport);
|
|
|
|
return;
|
|
|
|
} else {
|
|
|
|
res.statusCode = 405;
|
|
|
|
res.end('Method not allowed');
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
httpServer.listen(port, () => {
|
|
|
|
const address = httpServer.address();
|
|
|
|
assert(address, 'Could not bind server socket');
|
|
|
|
let url: string;
|
|
|
|
if (typeof address === 'string') {
|
|
|
|
url = address;
|
|
|
|
} else {
|
|
|
|
const resolvedPort = address.port;
|
|
|
|
let resolvedHost = address.family === 'IPv4' ? address.address : `[${address.address}]`;
|
|
|
|
if (resolvedHost === '0.0.0.0' || resolvedHost === '[::]')
|
|
|
|
resolvedHost = 'localhost';
|
|
|
|
url = `http://${resolvedHost}:${resolvedPort}`;
|
|
|
|
}
|
|
|
|
console.log(`Listening on ${url}`);
|
|
|
|
console.log('Put this in your client config:');
|
|
|
|
console.log(JSON.stringify({
|
|
|
|
'mcpServers': {
|
|
|
|
'playwright': {
|
|
|
|
'url': `${url}/sse`
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}, undefined, 2));
|
|
|
|
});
|
|
|
|
}
|