Skip to main content
Glama

Playwright MCP Server

Official
by microsoft
fixtures.ts9.11 kB
/** * 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. */ import fs from 'fs'; import path from 'path'; import { chromium } from 'playwright'; import { test as baseTest, expect as baseExpect } from '@playwright/test'; import { StdioClientTransport } from '@modelcontextprotocol/sdk/client/stdio.js'; import { Client } from '@modelcontextprotocol/sdk/client/index.js'; import { ListRootsRequestSchema } from '@modelcontextprotocol/sdk/types.js'; import { TestServer } from './testserver/index'; import type { Config } from '../config'; import type { BrowserContext } from 'playwright'; import type { Transport } from '@modelcontextprotocol/sdk/shared/transport.js'; import type { Stream } from 'stream'; export type TestOptions = { mcpArgs: string[] | undefined; mcpBrowser: string | undefined; mcpMode: 'docker' | undefined; }; type CDPServer = { endpoint: string; start: () => Promise<BrowserContext>; }; export type StartClient = (options?: { clientName?: string, args?: string[], config?: Config, roots?: { name: string, uri: string }[], rootsResponseDelay?: number, extensionToken?: string, }) => Promise<{ client: Client, stderr: () => string }>; type TestFixtures = { client: Client; startClient: StartClient; wsEndpoint: string; cdpServer: CDPServer; server: TestServer; httpsServer: TestServer; mcpHeadless: boolean; }; type WorkerFixtures = { _workerServers: { server: TestServer, httpsServer: TestServer }; }; export const test = baseTest.extend<TestFixtures & TestOptions, WorkerFixtures>({ mcpArgs: [undefined, { option: true }], client: async ({ startClient }, use) => { const { client } = await startClient(); await use(client); }, startClient: async ({ mcpHeadless, mcpBrowser, mcpMode, mcpArgs }, use, testInfo) => { const configDir = path.dirname(test.info().config.configFile!); const clients: Client[] = []; await use(async options => { const args: string[] = mcpArgs ?? []; if (process.env.CI && process.platform === 'linux') args.push('--no-sandbox'); if (mcpHeadless) args.push('--headless'); if (mcpBrowser) args.push(`--browser=${mcpBrowser}`); if (options?.args) args.push(...options.args); if (options?.config) { const configFile = testInfo.outputPath('config.json'); await fs.promises.writeFile(configFile, JSON.stringify(options.config, null, 2)); args.push(`--config=${path.relative(configDir, configFile)}`); } const client = new Client({ name: options?.clientName ?? 'test', version: '1.0.0' }, options?.roots ? { capabilities: { roots: {} } } : undefined); if (options?.roots) { client.setRequestHandler(ListRootsRequestSchema, async request => { if (options.rootsResponseDelay) await new Promise(resolve => setTimeout(resolve, options.rootsResponseDelay)); return { roots: options.roots, }; }); } const { transport, stderr } = await createTransport(args, mcpMode, testInfo.outputPath('ms-playwright'), options?.extensionToken); let stderrBuffer = ''; stderr?.on('data', data => { if (process.env.PWMCP_DEBUG) process.stderr.write(data); stderrBuffer += data.toString(); }); clients.push(client); await client.connect(transport); await client.ping(); return { client, stderr: () => stderrBuffer }; }); await Promise.all(clients.map(client => client.close())); }, wsEndpoint: async ({ }, use) => { const browserServer = await chromium.launchServer(); await use(browserServer.wsEndpoint()); await browserServer.close(); }, cdpServer: async ({ mcpBrowser }, use, testInfo) => { test.skip(!['chrome', 'msedge', 'chromium'].includes(mcpBrowser!), 'CDP is not supported for non-Chromium browsers'); let browserContext: BrowserContext | undefined; const port = 3200 + test.info().parallelIndex; await use({ endpoint: `http://localhost:${port}`, start: async () => { if (browserContext) throw new Error('CDP server already exists'); browserContext = await chromium.launchPersistentContext(testInfo.outputPath('cdp-user-data-dir'), { channel: mcpBrowser, headless: true, args: [ `--remote-debugging-port=${port}`, ], }); return browserContext; } }); await browserContext?.close(); }, mcpHeadless: async ({ headless }, use) => { await use(headless); }, mcpBrowser: ['chrome', { option: true }], mcpMode: [undefined, { option: true }], _workerServers: [async ({ }, use, workerInfo) => { const port = 8907 + workerInfo.workerIndex * 4; const server = await TestServer.create(port); const httpsPort = port + 1; const httpsServer = await TestServer.createHTTPS(httpsPort); await use({ server, httpsServer }); await Promise.all([ server.stop(), httpsServer.stop(), ]); }, { scope: 'worker' }], server: async ({ _workerServers }, use) => { _workerServers.server.reset(); await use(_workerServers.server); }, httpsServer: async ({ _workerServers }, use) => { _workerServers.httpsServer.reset(); await use(_workerServers.httpsServer); }, }); async function createTransport(args: string[], mcpMode: TestOptions['mcpMode'], profilesDir: string, extensionToken?: string): Promise<{ transport: Transport, stderr: Stream | null, }> { if (mcpMode === 'docker') { const dockerArgs = ['run', '--rm', '-i', '--network=host', '-v', `${test.info().project.outputDir}:/app/test-results`]; const transport = new StdioClientTransport({ command: 'docker', args: [...dockerArgs, 'playwright-mcp-dev:latest', ...args], }); return { transport, stderr: transport.stderr, }; } const transport = new StdioClientTransport({ command: 'node', args: [path.join(__dirname, '../cli.js'), ...args], cwd: path.dirname(test.info().config.configFile!), stderr: 'pipe', env: { ...process.env, DEBUG: 'pw:mcp:test', DEBUG_COLORS: '0', DEBUG_HIDE_DATE: '1', PWMCP_PROFILES_DIR_FOR_TEST: profilesDir, ...(extensionToken ? { PLAYWRIGHT_MCP_EXTENSION_TOKEN: extensionToken } : {}), }, }); return { transport, stderr: transport.stderr!, }; } type Response = Awaited<ReturnType<Client['callTool']>>; export const expect = baseExpect.extend({ toHaveResponse(response: Response, object: any) { const parsed = parseResponse(response); const isNot = this.isNot; try { if (isNot) expect(parsed).not.toEqual(expect.objectContaining(object)); else expect(parsed).toEqual(expect.objectContaining(object)); } catch (e) { return { pass: isNot, message: () => e.message, }; } return { pass: !isNot, message: () => ``, }; }, }); export function formatOutput(output: string): string[] { return output.split('\n').map(line => line.replace(/^pw:mcp:test /, '').replace(/user data dir.*/, 'user data dir').trim()).filter(Boolean); } function parseResponse(response: any) { const text = response.content[0].text; const sections = parseSections(text); const result = sections.get('Result'); const code = sections.get('Ran Playwright code'); const tabs = sections.get('Open tabs'); const pageState = sections.get('Page state'); const consoleMessages = sections.get('New console messages'); const modalState = sections.get('Modal state'); const downloads = sections.get('Downloads'); const codeNoFrame = code?.replace(/^```js\n/, '').replace(/\n```$/, ''); const isError = response.isError; const attachments = response.content.slice(1); return { result, code: codeNoFrame, tabs, pageState, consoleMessages, modalState, downloads, isError, attachments, }; } function parseSections(text: string): Map<string, string> { const sections = new Map<string, string>(); const sectionHeaders = text.split(/^### /m).slice(1); // Remove empty first element for (const section of sectionHeaders) { const firstNewlineIndex = section.indexOf('\n'); if (firstNewlineIndex === -1) continue; const sectionName = section.substring(0, firstNewlineIndex); const sectionContent = section.substring(firstNewlineIndex + 1).trim(); sections.set(sectionName, sectionContent); } return sections; }

MCP directory API

We provide all the information about MCP servers via our MCP API.

curl -X GET 'https://glama.ai/api/mcp/v1/servers/microsoft/playwright-mcp'

If you have feedback or need assistance with the MCP directory API, please join our Discord server