@stablyai/internal-playwright 0.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +202 -0
- package/NOTICE +5 -0
- package/README.md +168 -0
- package/ThirdPartyNotices.txt +6277 -0
- package/cli.js +19 -0
- package/index.d.ts +17 -0
- package/index.js +17 -0
- package/index.mjs +18 -0
- package/jsx-runtime.js +42 -0
- package/jsx-runtime.mjs +21 -0
- package/lib/agents/generateAgents.js +265 -0
- package/lib/agents/generator.md +102 -0
- package/lib/agents/healer.md +78 -0
- package/lib/agents/planner.md +135 -0
- package/lib/cli.js +274 -0
- package/lib/common/config.js +274 -0
- package/lib/common/config.js.map +7 -0
- package/lib/common/configLoader.js +377 -0
- package/lib/common/configLoader.js.map +7 -0
- package/lib/common/esmLoaderHost.js +102 -0
- package/lib/common/esmLoaderHost.js.map +7 -0
- package/lib/common/expectBundle.js +52 -0
- package/lib/common/expectBundle.js.map +7 -0
- package/lib/common/expectBundleImpl.js +389 -0
- package/lib/common/expectBundleImpl.js.map +7 -0
- package/lib/common/fixtures.js +302 -0
- package/lib/common/fixtures.js.map +7 -0
- package/lib/common/globals.js +58 -0
- package/lib/common/globals.js.map +7 -0
- package/lib/common/ipc.js +60 -0
- package/lib/common/ipc.js.map +7 -0
- package/lib/common/poolBuilder.js +85 -0
- package/lib/common/poolBuilder.js.map +7 -0
- package/lib/common/process.js +104 -0
- package/lib/common/process.js.map +7 -0
- package/lib/common/suiteUtils.js +140 -0
- package/lib/common/suiteUtils.js.map +7 -0
- package/lib/common/test.js +321 -0
- package/lib/common/test.js.map +7 -0
- package/lib/common/testLoader.js +100 -0
- package/lib/common/testLoader.js.map +7 -0
- package/lib/common/testType.js +310 -0
- package/lib/common/testType.js.map +7 -0
- package/lib/fsWatcher.js +67 -0
- package/lib/fsWatcher.js.map +7 -0
- package/lib/index.js +696 -0
- package/lib/index.js.map +7 -0
- package/lib/internalsForTest.js +42 -0
- package/lib/internalsForTest.js.map +7 -0
- package/lib/isomorphic/events.js +77 -0
- package/lib/isomorphic/events.js.map +7 -0
- package/lib/isomorphic/folders.js +30 -0
- package/lib/isomorphic/folders.js.map +7 -0
- package/lib/isomorphic/stringInternPool.js +69 -0
- package/lib/isomorphic/stringInternPool.js.map +7 -0
- package/lib/isomorphic/teleReceiver.js +507 -0
- package/lib/isomorphic/teleReceiver.js.map +7 -0
- package/lib/isomorphic/teleSuiteUpdater.js +137 -0
- package/lib/isomorphic/teleSuiteUpdater.js.map +7 -0
- package/lib/isomorphic/testServerConnection.js +211 -0
- package/lib/isomorphic/testServerConnection.js.map +7 -0
- package/lib/isomorphic/testServerInterface.js +16 -0
- package/lib/isomorphic/testServerInterface.js.map +7 -0
- package/lib/isomorphic/testTree.js +334 -0
- package/lib/isomorphic/testTree.js.map +7 -0
- package/lib/isomorphic/types.d.js +16 -0
- package/lib/isomorphic/types.d.js.map +7 -0
- package/lib/loader/loaderMain.js +59 -0
- package/lib/loader/loaderMain.js.map +7 -0
- package/lib/matchers/expect.js +325 -0
- package/lib/matchers/expect.js.map +7 -0
- package/lib/matchers/matcherHint.js +87 -0
- package/lib/matchers/matcherHint.js.map +7 -0
- package/lib/matchers/matchers.js +366 -0
- package/lib/matchers/matchers.js.map +7 -0
- package/lib/matchers/toBeTruthy.js +73 -0
- package/lib/matchers/toBeTruthy.js.map +7 -0
- package/lib/matchers/toEqual.js +99 -0
- package/lib/matchers/toEqual.js.map +7 -0
- package/lib/matchers/toHaveURL.js +102 -0
- package/lib/matchers/toHaveURL.js.map +7 -0
- package/lib/matchers/toMatchAriaSnapshot.js +159 -0
- package/lib/matchers/toMatchAriaSnapshot.js.map +7 -0
- package/lib/matchers/toMatchSnapshot.js +359 -0
- package/lib/matchers/toMatchSnapshot.js.map +7 -0
- package/lib/matchers/toMatchText.js +99 -0
- package/lib/matchers/toMatchText.js.map +7 -0
- package/lib/mcp/browser/actions.d.js +16 -0
- package/lib/mcp/browser/backend.js +93 -0
- package/lib/mcp/browser/backend.js.map +7 -0
- package/lib/mcp/browser/browserContextFactory.js +296 -0
- package/lib/mcp/browser/browserServerBackend.js +76 -0
- package/lib/mcp/browser/codegen.js +66 -0
- package/lib/mcp/browser/config.js +385 -0
- package/lib/mcp/browser/context.js +287 -0
- package/lib/mcp/browser/response.js +228 -0
- package/lib/mcp/browser/sessionLog.js +160 -0
- package/lib/mcp/browser/tab.js +277 -0
- package/lib/mcp/browser/tool.js +30 -0
- package/lib/mcp/browser/tool.js.map +7 -0
- package/lib/mcp/browser/tools/common.js +63 -0
- package/lib/mcp/browser/tools/console.js +44 -0
- package/lib/mcp/browser/tools/dialogs.js +60 -0
- package/lib/mcp/browser/tools/evaluate.js +70 -0
- package/lib/mcp/browser/tools/files.js +58 -0
- package/lib/mcp/browser/tools/form.js +74 -0
- package/lib/mcp/browser/tools/install.js +69 -0
- package/lib/mcp/browser/tools/keyboard.js +85 -0
- package/lib/mcp/browser/tools/mouse.js +107 -0
- package/lib/mcp/browser/tools/navigate.js +62 -0
- package/lib/mcp/browser/tools/network.js +54 -0
- package/lib/mcp/browser/tools/pdf.js +59 -0
- package/lib/mcp/browser/tools/screenshot.js +88 -0
- package/lib/mcp/browser/tools/snapshot.js +182 -0
- package/lib/mcp/browser/tools/tabs.js +67 -0
- package/lib/mcp/browser/tools/tool.js +49 -0
- package/lib/mcp/browser/tools/tracing.js +74 -0
- package/lib/mcp/browser/tools/utils.js +100 -0
- package/lib/mcp/browser/tools/verify.js +154 -0
- package/lib/mcp/browser/tools/wait.js +63 -0
- package/lib/mcp/browser/tools.js +80 -0
- package/lib/mcp/browser/tools.js.map +7 -0
- package/lib/mcp/browser/watchdog.js +44 -0
- package/lib/mcp/config.d.js +16 -0
- package/lib/mcp/extension/cdpRelay.js +351 -0
- package/lib/mcp/extension/extensionContextFactory.js +75 -0
- package/lib/mcp/extension/protocol.js +28 -0
- package/lib/mcp/index.js +61 -0
- package/lib/mcp/log.js +35 -0
- package/lib/mcp/program.js +96 -0
- package/lib/mcp/sdk/bundle.js +81 -0
- package/lib/mcp/sdk/bundle.js.map +7 -0
- package/lib/mcp/sdk/call.js +49 -0
- package/lib/mcp/sdk/call.js.map +7 -0
- package/lib/mcp/sdk/exports.js +32 -0
- package/lib/mcp/sdk/exports.js.map +7 -0
- package/lib/mcp/sdk/http.js +187 -0
- package/lib/mcp/sdk/http.js.map +7 -0
- package/lib/mcp/sdk/inProcessTransport.js +71 -0
- package/lib/mcp/sdk/inProcessTransport.js.map +7 -0
- package/lib/mcp/sdk/mdb.js +206 -0
- package/lib/mcp/sdk/mdb.js.map +7 -0
- package/lib/mcp/sdk/proxyBackend.js +128 -0
- package/lib/mcp/sdk/proxyBackend.js.map +7 -0
- package/lib/mcp/sdk/server.js +189 -0
- package/lib/mcp/sdk/server.js.map +7 -0
- package/lib/mcp/sdk/tool.js +51 -0
- package/lib/mcp/sdk/tool.js.map +7 -0
- package/lib/mcp/test/backend.js +67 -0
- package/lib/mcp/test/backend.js.map +7 -0
- package/lib/mcp/test/browserBackend.js +98 -0
- package/lib/mcp/test/context.js +48 -0
- package/lib/mcp/test/context.js.map +7 -0
- package/lib/mcp/test/generatorTools.js +122 -0
- package/lib/mcp/test/plannerTools.js +46 -0
- package/lib/mcp/test/seed.js +72 -0
- package/lib/mcp/test/streams.js +39 -0
- package/lib/mcp/test/streams.js.map +7 -0
- package/lib/mcp/test/testBackend.js +97 -0
- package/lib/mcp/test/testContext.js +176 -0
- package/lib/mcp/test/testTool.js +30 -0
- package/lib/mcp/test/testTools.js +115 -0
- package/lib/mcp/test/tool.js +30 -0
- package/lib/mcp/test/tool.js.map +7 -0
- package/lib/mcp/test/tools.js +150 -0
- package/lib/mcp/test/tools.js.map +7 -0
- package/lib/mcp/vscode/host.js +187 -0
- package/lib/mcp/vscode/main.js +77 -0
- package/lib/mcpBundleImpl.js +41 -0
- package/lib/mcpBundleImpl.js.map +7 -0
- package/lib/plugins/gitCommitInfoPlugin.js +198 -0
- package/lib/plugins/gitCommitInfoPlugin.js.map +7 -0
- package/lib/plugins/index.js +28 -0
- package/lib/plugins/index.js.map +7 -0
- package/lib/plugins/webServerPlugin.js +209 -0
- package/lib/plugins/webServerPlugin.js.map +7 -0
- package/lib/program.js +412 -0
- package/lib/program.js.map +7 -0
- package/lib/reporters/base.js +609 -0
- package/lib/reporters/base.js.map +7 -0
- package/lib/reporters/blob.js +135 -0
- package/lib/reporters/blob.js.map +7 -0
- package/lib/reporters/dot.js +82 -0
- package/lib/reporters/dot.js.map +7 -0
- package/lib/reporters/empty.js +32 -0
- package/lib/reporters/empty.js.map +7 -0
- package/lib/reporters/github.js +128 -0
- package/lib/reporters/github.js.map +7 -0
- package/lib/reporters/html.js +644 -0
- package/lib/reporters/html.js.map +7 -0
- package/lib/reporters/internalReporter.js +130 -0
- package/lib/reporters/internalReporter.js.map +7 -0
- package/lib/reporters/json.js +254 -0
- package/lib/reporters/json.js.map +7 -0
- package/lib/reporters/junit.js +230 -0
- package/lib/reporters/junit.js.map +7 -0
- package/lib/reporters/line.js +113 -0
- package/lib/reporters/line.js.map +7 -0
- package/lib/reporters/list.js +235 -0
- package/lib/reporters/list.js.map +7 -0
- package/lib/reporters/listModeReporter.js +69 -0
- package/lib/reporters/listModeReporter.js.map +7 -0
- package/lib/reporters/markdown.js +144 -0
- package/lib/reporters/markdown.js.map +7 -0
- package/lib/reporters/merge.js +535 -0
- package/lib/reporters/merge.js.map +7 -0
- package/lib/reporters/multiplexer.js +104 -0
- package/lib/reporters/multiplexer.js.map +7 -0
- package/lib/reporters/reporterV2.js +102 -0
- package/lib/reporters/reporterV2.js.map +7 -0
- package/lib/reporters/teleEmitter.js +297 -0
- package/lib/reporters/teleEmitter.js.map +7 -0
- package/lib/reporters/versions/blobV1.js +16 -0
- package/lib/reporters/versions/blobV1.js.map +7 -0
- package/lib/runner/dispatcher.js +491 -0
- package/lib/runner/dispatcher.js.map +7 -0
- package/lib/runner/failureTracker.js +72 -0
- package/lib/runner/failureTracker.js.map +7 -0
- package/lib/runner/lastRun.js +77 -0
- package/lib/runner/lastRun.js.map +7 -0
- package/lib/runner/loadUtils.js +333 -0
- package/lib/runner/loadUtils.js.map +7 -0
- package/lib/runner/loaderHost.js +89 -0
- package/lib/runner/loaderHost.js.map +7 -0
- package/lib/runner/processHost.js +161 -0
- package/lib/runner/processHost.js.map +7 -0
- package/lib/runner/projectUtils.js +241 -0
- package/lib/runner/projectUtils.js.map +7 -0
- package/lib/runner/rebase.js +189 -0
- package/lib/runner/rebase.js.map +7 -0
- package/lib/runner/reporters.js +137 -0
- package/lib/runner/reporters.js.map +7 -0
- package/lib/runner/runner.js +173 -0
- package/lib/runner/sigIntWatcher.js +96 -0
- package/lib/runner/sigIntWatcher.js.map +7 -0
- package/lib/runner/taskRunner.js +127 -0
- package/lib/runner/taskRunner.js.map +7 -0
- package/lib/runner/tasks.js +410 -0
- package/lib/runner/tasks.js.map +7 -0
- package/lib/runner/testGroups.js +117 -0
- package/lib/runner/testGroups.js.map +7 -0
- package/lib/runner/testRunner.js +390 -0
- package/lib/runner/testRunner.js.map +7 -0
- package/lib/runner/testServer.js +264 -0
- package/lib/runner/testServer.js.map +7 -0
- package/lib/runner/uiMode.js +271 -0
- package/lib/runner/uiModeReporter.js +30 -0
- package/lib/runner/uiModeReporter.js.map +7 -0
- package/lib/runner/vcs.js +72 -0
- package/lib/runner/vcs.js.map +7 -0
- package/lib/runner/watchMode.js +395 -0
- package/lib/runner/watchMode.js.map +7 -0
- package/lib/runner/workerHost.js +95 -0
- package/lib/runner/workerHost.js.map +7 -0
- package/lib/store.js +98 -0
- package/lib/third_party/pirates.js +62 -0
- package/lib/third_party/pirates.js.map +7 -0
- package/lib/third_party/tsconfig-loader.js +103 -0
- package/lib/third_party/tsconfig-loader.js.map +7 -0
- package/lib/transform/babelBundle.js +43 -0
- package/lib/transform/babelBundle.js.map +7 -0
- package/lib/transform/babelBundleImpl.js +461 -0
- package/lib/transform/babelBundleImpl.js.map +7 -0
- package/lib/transform/compilationCache.js +272 -0
- package/lib/transform/compilationCache.js.map +7 -0
- package/lib/transform/esmLoader.js +104 -0
- package/lib/transform/esmLoader.js.map +7 -0
- package/lib/transform/esmUtils.js +32 -0
- package/lib/transform/portTransport.js +67 -0
- package/lib/transform/portTransport.js.map +7 -0
- package/lib/transform/transform.js +293 -0
- package/lib/transform/transform.js.map +7 -0
- package/lib/util.js +403 -0
- package/lib/util.js.map +7 -0
- package/lib/utilsBundle.js +43 -0
- package/lib/utilsBundle.js.map +7 -0
- package/lib/utilsBundleImpl.js +100 -0
- package/lib/utilsBundleImpl.js.map +7 -0
- package/lib/worker/fixtureRunner.js +258 -0
- package/lib/worker/fixtureRunner.js.map +7 -0
- package/lib/worker/stepContext.js +34 -0
- package/lib/worker/testInfo.js +508 -0
- package/lib/worker/testInfo.js.map +7 -0
- package/lib/worker/testTracing.js +344 -0
- package/lib/worker/testTracing.js.map +7 -0
- package/lib/worker/timeoutManager.js +174 -0
- package/lib/worker/timeoutManager.js.map +7 -0
- package/lib/worker/util.js +31 -0
- package/lib/worker/util.js.map +7 -0
- package/lib/worker/workerMain.js +520 -0
- package/lib/worker/workerMain.js.map +7 -0
- package/package.json +74 -0
- package/test.d.ts +18 -0
- package/test.js +24 -0
- package/test.mjs +33 -0
- package/types/test.d.ts +10217 -0
- package/types/testReporter.d.ts +816 -0
@@ -0,0 +1,351 @@
|
|
1
|
+
"use strict";
|
2
|
+
var __create = Object.create;
|
3
|
+
var __defProp = Object.defineProperty;
|
4
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
5
|
+
var __getOwnPropNames = Object.getOwnPropertyNames;
|
6
|
+
var __getProtoOf = Object.getPrototypeOf;
|
7
|
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
8
|
+
var __export = (target, all) => {
|
9
|
+
for (var name in all)
|
10
|
+
__defProp(target, name, { get: all[name], enumerable: true });
|
11
|
+
};
|
12
|
+
var __copyProps = (to, from, except, desc) => {
|
13
|
+
if (from && typeof from === "object" || typeof from === "function") {
|
14
|
+
for (let key of __getOwnPropNames(from))
|
15
|
+
if (!__hasOwnProp.call(to, key) && key !== except)
|
16
|
+
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
17
|
+
}
|
18
|
+
return to;
|
19
|
+
};
|
20
|
+
var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
|
21
|
+
// If the importer is in node compatibility mode or this is not an ESM
|
22
|
+
// file that has been converted to a CommonJS file using a Babel-
|
23
|
+
// compatible transform (i.e. "__esModule" has not been set), then set
|
24
|
+
// "default" to the CommonJS "module.exports" for node compatibility.
|
25
|
+
isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
|
26
|
+
mod
|
27
|
+
));
|
28
|
+
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
29
|
+
var cdpRelay_exports = {};
|
30
|
+
__export(cdpRelay_exports, {
|
31
|
+
CDPRelayServer: () => CDPRelayServer
|
32
|
+
});
|
33
|
+
module.exports = __toCommonJS(cdpRelay_exports);
|
34
|
+
var import_child_process = require("child_process");
|
35
|
+
var import_utilsBundle = require("playwright-core/lib/utilsBundle");
|
36
|
+
var import_registry = require("playwright-core/lib/server/registry/index");
|
37
|
+
var import_utils = require("playwright-core/lib/utils");
|
38
|
+
var import_http2 = require("../sdk/http");
|
39
|
+
var import_log = require("../log");
|
40
|
+
var protocol = __toESM(require("./protocol"));
|
41
|
+
const debugLogger = (0, import_utilsBundle.debug)("pw:mcp:relay");
|
42
|
+
class CDPRelayServer {
|
43
|
+
constructor(server, browserChannel, userDataDir, executablePath) {
|
44
|
+
this._playwrightConnection = null;
|
45
|
+
this._extensionConnection = null;
|
46
|
+
this._nextSessionId = 1;
|
47
|
+
this._wsHost = (0, import_http2.httpAddressToString)(server.address()).replace(/^http/, "ws");
|
48
|
+
this._browserChannel = browserChannel;
|
49
|
+
this._userDataDir = userDataDir;
|
50
|
+
this._executablePath = executablePath;
|
51
|
+
const uuid = crypto.randomUUID();
|
52
|
+
this._cdpPath = `/cdp/${uuid}`;
|
53
|
+
this._extensionPath = `/extension/${uuid}`;
|
54
|
+
this._resetExtensionConnection();
|
55
|
+
this._wss = new import_utilsBundle.wsServer({ server });
|
56
|
+
this._wss.on("connection", this._onConnection.bind(this));
|
57
|
+
}
|
58
|
+
cdpEndpoint() {
|
59
|
+
return `${this._wsHost}${this._cdpPath}`;
|
60
|
+
}
|
61
|
+
extensionEndpoint() {
|
62
|
+
return `${this._wsHost}${this._extensionPath}`;
|
63
|
+
}
|
64
|
+
async ensureExtensionConnectionForMCPContext(clientInfo, abortSignal, toolName) {
|
65
|
+
debugLogger("Ensuring extension connection for MCP context");
|
66
|
+
if (this._extensionConnection)
|
67
|
+
return;
|
68
|
+
this._connectBrowser(clientInfo, toolName);
|
69
|
+
debugLogger("Waiting for incoming extension connection");
|
70
|
+
await Promise.race([
|
71
|
+
this._extensionConnectionPromise,
|
72
|
+
new Promise((_, reject) => setTimeout(() => {
|
73
|
+
reject(new Error(`Extension connection timeout. Make sure the "Playwright MCP Bridge" extension is installed. See https://github.com/microsoft/playwright-mcp/blob/main/extension/README.md for installation instructions.`));
|
74
|
+
}, process.env.PWMCP_TEST_CONNECTION_TIMEOUT ? parseInt(process.env.PWMCP_TEST_CONNECTION_TIMEOUT, 10) : 5e3)),
|
75
|
+
new Promise((_, reject) => abortSignal.addEventListener("abort", reject))
|
76
|
+
]);
|
77
|
+
debugLogger("Extension connection established");
|
78
|
+
}
|
79
|
+
_connectBrowser(clientInfo, toolName) {
|
80
|
+
const mcpRelayEndpoint = `${this._wsHost}${this._extensionPath}`;
|
81
|
+
const url = new URL("chrome-extension://jakfalbnbhgkpmoaakfflhflbfpkailf/connect.html");
|
82
|
+
url.searchParams.set("mcpRelayUrl", mcpRelayEndpoint);
|
83
|
+
const client = {
|
84
|
+
name: clientInfo.name,
|
85
|
+
version: clientInfo.version
|
86
|
+
};
|
87
|
+
url.searchParams.set("client", JSON.stringify(client));
|
88
|
+
url.searchParams.set("protocolVersion", process.env.PWMCP_TEST_PROTOCOL_VERSION ?? protocol.VERSION.toString());
|
89
|
+
if (toolName)
|
90
|
+
url.searchParams.set("newTab", String(toolName === "browser_navigate"));
|
91
|
+
const token = process.env.PLAYWRIGHT_MCP_EXTENSION_TOKEN;
|
92
|
+
if (token)
|
93
|
+
url.searchParams.set("token", token);
|
94
|
+
const href = url.toString();
|
95
|
+
let executablePath = this._executablePath;
|
96
|
+
if (!executablePath) {
|
97
|
+
const executableInfo = import_registry.registry.findExecutable(this._browserChannel);
|
98
|
+
if (!executableInfo)
|
99
|
+
throw new Error(`Unsupported channel: "${this._browserChannel}"`);
|
100
|
+
executablePath = executableInfo.executablePath("javascript");
|
101
|
+
if (!executablePath)
|
102
|
+
throw new Error(`"${this._browserChannel}" executable not found. Make sure it is installed at a standard location.`);
|
103
|
+
}
|
104
|
+
const args = [];
|
105
|
+
if (this._userDataDir)
|
106
|
+
args.push(`--user-data-dir=${this._userDataDir}`);
|
107
|
+
args.push(href);
|
108
|
+
(0, import_child_process.spawn)(executablePath, args, {
|
109
|
+
windowsHide: true,
|
110
|
+
detached: true,
|
111
|
+
shell: false,
|
112
|
+
stdio: "ignore"
|
113
|
+
});
|
114
|
+
}
|
115
|
+
stop() {
|
116
|
+
this.closeConnections("Server stopped");
|
117
|
+
this._wss.close();
|
118
|
+
}
|
119
|
+
closeConnections(reason) {
|
120
|
+
this._closePlaywrightConnection(reason);
|
121
|
+
this._closeExtensionConnection(reason);
|
122
|
+
}
|
123
|
+
_onConnection(ws2, request) {
|
124
|
+
const url = new URL(`http://localhost${request.url}`);
|
125
|
+
debugLogger(`New connection to ${url.pathname}`);
|
126
|
+
if (url.pathname === this._cdpPath) {
|
127
|
+
this._handlePlaywrightConnection(ws2);
|
128
|
+
} else if (url.pathname === this._extensionPath) {
|
129
|
+
this._handleExtensionConnection(ws2);
|
130
|
+
} else {
|
131
|
+
debugLogger(`Invalid path: ${url.pathname}`);
|
132
|
+
ws2.close(4004, "Invalid path");
|
133
|
+
}
|
134
|
+
}
|
135
|
+
_handlePlaywrightConnection(ws2) {
|
136
|
+
if (this._playwrightConnection) {
|
137
|
+
debugLogger("Rejecting second Playwright connection");
|
138
|
+
ws2.close(1e3, "Another CDP client already connected");
|
139
|
+
return;
|
140
|
+
}
|
141
|
+
this._playwrightConnection = ws2;
|
142
|
+
ws2.on("message", async (data) => {
|
143
|
+
try {
|
144
|
+
const message = JSON.parse(data.toString());
|
145
|
+
await this._handlePlaywrightMessage(message);
|
146
|
+
} catch (error) {
|
147
|
+
debugLogger(`Error while handling Playwright message
|
148
|
+
${data.toString()}
|
149
|
+
`, error);
|
150
|
+
}
|
151
|
+
});
|
152
|
+
ws2.on("close", () => {
|
153
|
+
if (this._playwrightConnection !== ws2)
|
154
|
+
return;
|
155
|
+
this._playwrightConnection = null;
|
156
|
+
this._closeExtensionConnection("Playwright client disconnected");
|
157
|
+
debugLogger("Playwright WebSocket closed");
|
158
|
+
});
|
159
|
+
ws2.on("error", (error) => {
|
160
|
+
debugLogger("Playwright WebSocket error:", error);
|
161
|
+
});
|
162
|
+
debugLogger("Playwright MCP connected");
|
163
|
+
}
|
164
|
+
_closeExtensionConnection(reason) {
|
165
|
+
this._extensionConnection?.close(reason);
|
166
|
+
this._extensionConnectionPromise.reject(new Error(reason));
|
167
|
+
this._resetExtensionConnection();
|
168
|
+
}
|
169
|
+
_resetExtensionConnection() {
|
170
|
+
this._connectedTabInfo = void 0;
|
171
|
+
this._extensionConnection = null;
|
172
|
+
this._extensionConnectionPromise = new import_utils.ManualPromise();
|
173
|
+
void this._extensionConnectionPromise.catch(import_log.logUnhandledError);
|
174
|
+
}
|
175
|
+
_closePlaywrightConnection(reason) {
|
176
|
+
if (this._playwrightConnection?.readyState === import_utilsBundle.ws.OPEN)
|
177
|
+
this._playwrightConnection.close(1e3, reason);
|
178
|
+
this._playwrightConnection = null;
|
179
|
+
}
|
180
|
+
_handleExtensionConnection(ws2) {
|
181
|
+
if (this._extensionConnection) {
|
182
|
+
ws2.close(1e3, "Another extension connection already established");
|
183
|
+
return;
|
184
|
+
}
|
185
|
+
this._extensionConnection = new ExtensionConnection(ws2);
|
186
|
+
this._extensionConnection.onclose = (c, reason) => {
|
187
|
+
debugLogger("Extension WebSocket closed:", reason, c === this._extensionConnection);
|
188
|
+
if (this._extensionConnection !== c)
|
189
|
+
return;
|
190
|
+
this._resetExtensionConnection();
|
191
|
+
this._closePlaywrightConnection(`Extension disconnected: ${reason}`);
|
192
|
+
};
|
193
|
+
this._extensionConnection.onmessage = this._handleExtensionMessage.bind(this);
|
194
|
+
this._extensionConnectionPromise.resolve();
|
195
|
+
}
|
196
|
+
_handleExtensionMessage(method, params) {
|
197
|
+
switch (method) {
|
198
|
+
case "forwardCDPEvent":
|
199
|
+
const sessionId = params.sessionId || this._connectedTabInfo?.sessionId;
|
200
|
+
this._sendToPlaywright({
|
201
|
+
sessionId,
|
202
|
+
method: params.method,
|
203
|
+
params: params.params
|
204
|
+
});
|
205
|
+
break;
|
206
|
+
}
|
207
|
+
}
|
208
|
+
async _handlePlaywrightMessage(message) {
|
209
|
+
debugLogger("\u2190 Playwright:", `${message.method} (id=${message.id})`);
|
210
|
+
const { id, sessionId, method, params } = message;
|
211
|
+
try {
|
212
|
+
const result = await this._handleCDPCommand(method, params, sessionId);
|
213
|
+
this._sendToPlaywright({ id, sessionId, result });
|
214
|
+
} catch (e) {
|
215
|
+
debugLogger("Error in the extension:", e);
|
216
|
+
this._sendToPlaywright({
|
217
|
+
id,
|
218
|
+
sessionId,
|
219
|
+
error: { message: e.message }
|
220
|
+
});
|
221
|
+
}
|
222
|
+
}
|
223
|
+
async _handleCDPCommand(method, params, sessionId) {
|
224
|
+
switch (method) {
|
225
|
+
case "Browser.getVersion": {
|
226
|
+
return {
|
227
|
+
protocolVersion: "1.3",
|
228
|
+
product: "Chrome/Extension-Bridge",
|
229
|
+
userAgent: "CDP-Bridge-Server/1.0.0"
|
230
|
+
};
|
231
|
+
}
|
232
|
+
case "Browser.setDownloadBehavior": {
|
233
|
+
return {};
|
234
|
+
}
|
235
|
+
case "Target.setAutoAttach": {
|
236
|
+
if (sessionId)
|
237
|
+
break;
|
238
|
+
const { targetInfo } = await this._extensionConnection.send("attachToTab", {});
|
239
|
+
this._connectedTabInfo = {
|
240
|
+
targetInfo,
|
241
|
+
sessionId: `pw-tab-${this._nextSessionId++}`
|
242
|
+
};
|
243
|
+
debugLogger("Simulating auto-attach");
|
244
|
+
this._sendToPlaywright({
|
245
|
+
method: "Target.attachedToTarget",
|
246
|
+
params: {
|
247
|
+
sessionId: this._connectedTabInfo.sessionId,
|
248
|
+
targetInfo: {
|
249
|
+
...this._connectedTabInfo.targetInfo,
|
250
|
+
attached: true
|
251
|
+
},
|
252
|
+
waitingForDebugger: false
|
253
|
+
}
|
254
|
+
});
|
255
|
+
return {};
|
256
|
+
}
|
257
|
+
case "Target.getTargetInfo": {
|
258
|
+
return this._connectedTabInfo?.targetInfo;
|
259
|
+
}
|
260
|
+
}
|
261
|
+
return await this._forwardToExtension(method, params, sessionId);
|
262
|
+
}
|
263
|
+
async _forwardToExtension(method, params, sessionId) {
|
264
|
+
if (!this._extensionConnection)
|
265
|
+
throw new Error("Extension not connected");
|
266
|
+
if (this._connectedTabInfo?.sessionId === sessionId)
|
267
|
+
sessionId = void 0;
|
268
|
+
return await this._extensionConnection.send("forwardCDPCommand", { sessionId, method, params });
|
269
|
+
}
|
270
|
+
_sendToPlaywright(message) {
|
271
|
+
debugLogger("\u2192 Playwright:", `${message.method ?? `response(id=${message.id})`}`);
|
272
|
+
this._playwrightConnection?.send(JSON.stringify(message));
|
273
|
+
}
|
274
|
+
}
|
275
|
+
class ExtensionConnection {
|
276
|
+
constructor(ws2) {
|
277
|
+
this._callbacks = /* @__PURE__ */ new Map();
|
278
|
+
this._lastId = 0;
|
279
|
+
this._ws = ws2;
|
280
|
+
this._ws.on("message", this._onMessage.bind(this));
|
281
|
+
this._ws.on("close", this._onClose.bind(this));
|
282
|
+
this._ws.on("error", this._onError.bind(this));
|
283
|
+
}
|
284
|
+
async send(method, params) {
|
285
|
+
if (this._ws.readyState !== import_utilsBundle.ws.OPEN)
|
286
|
+
throw new Error(`Unexpected WebSocket state: ${this._ws.readyState}`);
|
287
|
+
const id = ++this._lastId;
|
288
|
+
this._ws.send(JSON.stringify({ id, method, params }));
|
289
|
+
const error = new Error(`Protocol error: ${method}`);
|
290
|
+
return new Promise((resolve, reject) => {
|
291
|
+
this._callbacks.set(id, { resolve, reject, error });
|
292
|
+
});
|
293
|
+
}
|
294
|
+
close(message) {
|
295
|
+
debugLogger("closing extension connection:", message);
|
296
|
+
if (this._ws.readyState === import_utilsBundle.ws.OPEN)
|
297
|
+
this._ws.close(1e3, message);
|
298
|
+
}
|
299
|
+
_onMessage(event) {
|
300
|
+
const eventData = event.toString();
|
301
|
+
let parsedJson;
|
302
|
+
try {
|
303
|
+
parsedJson = JSON.parse(eventData);
|
304
|
+
} catch (e) {
|
305
|
+
debugLogger(`<closing ws> Closing websocket due to malformed JSON. eventData=${eventData} e=${e?.message}`);
|
306
|
+
this._ws.close();
|
307
|
+
return;
|
308
|
+
}
|
309
|
+
try {
|
310
|
+
this._handleParsedMessage(parsedJson);
|
311
|
+
} catch (e) {
|
312
|
+
debugLogger(`<closing ws> Closing websocket due to failed onmessage callback. eventData=${eventData} e=${e?.message}`);
|
313
|
+
this._ws.close();
|
314
|
+
}
|
315
|
+
}
|
316
|
+
_handleParsedMessage(object) {
|
317
|
+
if (object.id && this._callbacks.has(object.id)) {
|
318
|
+
const callback = this._callbacks.get(object.id);
|
319
|
+
this._callbacks.delete(object.id);
|
320
|
+
if (object.error) {
|
321
|
+
const error = callback.error;
|
322
|
+
error.message = object.error;
|
323
|
+
callback.reject(error);
|
324
|
+
} else {
|
325
|
+
callback.resolve(object.result);
|
326
|
+
}
|
327
|
+
} else if (object.id) {
|
328
|
+
debugLogger("\u2190 Extension: unexpected response", object);
|
329
|
+
} else {
|
330
|
+
this.onmessage?.(object.method, object.params);
|
331
|
+
}
|
332
|
+
}
|
333
|
+
_onClose(event) {
|
334
|
+
debugLogger(`<ws closed> code=${event.code} reason=${event.reason}`);
|
335
|
+
this._dispose();
|
336
|
+
this.onclose?.(this, event.reason);
|
337
|
+
}
|
338
|
+
_onError(event) {
|
339
|
+
debugLogger(`<ws error> message=${event.message} type=${event.type} target=${event.target}`);
|
340
|
+
this._dispose();
|
341
|
+
}
|
342
|
+
_dispose() {
|
343
|
+
for (const callback of this._callbacks.values())
|
344
|
+
callback.reject(new Error("WebSocket closed"));
|
345
|
+
this._callbacks.clear();
|
346
|
+
}
|
347
|
+
}
|
348
|
+
// Annotate the CommonJS export names for ESM import in node:
|
349
|
+
0 && (module.exports = {
|
350
|
+
CDPRelayServer
|
351
|
+
});
|
@@ -0,0 +1,75 @@
|
|
1
|
+
"use strict";
|
2
|
+
var __create = Object.create;
|
3
|
+
var __defProp = Object.defineProperty;
|
4
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
5
|
+
var __getOwnPropNames = Object.getOwnPropertyNames;
|
6
|
+
var __getProtoOf = Object.getPrototypeOf;
|
7
|
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
8
|
+
var __export = (target, all) => {
|
9
|
+
for (var name in all)
|
10
|
+
__defProp(target, name, { get: all[name], enumerable: true });
|
11
|
+
};
|
12
|
+
var __copyProps = (to, from, except, desc) => {
|
13
|
+
if (from && typeof from === "object" || typeof from === "function") {
|
14
|
+
for (let key of __getOwnPropNames(from))
|
15
|
+
if (!__hasOwnProp.call(to, key) && key !== except)
|
16
|
+
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
17
|
+
}
|
18
|
+
return to;
|
19
|
+
};
|
20
|
+
var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
|
21
|
+
// If the importer is in node compatibility mode or this is not an ESM
|
22
|
+
// file that has been converted to a CommonJS file using a Babel-
|
23
|
+
// compatible transform (i.e. "__esModule" has not been set), then set
|
24
|
+
// "default" to the CommonJS "module.exports" for node compatibility.
|
25
|
+
isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
|
26
|
+
mod
|
27
|
+
));
|
28
|
+
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
29
|
+
var extensionContextFactory_exports = {};
|
30
|
+
__export(extensionContextFactory_exports, {
|
31
|
+
ExtensionContextFactory: () => ExtensionContextFactory
|
32
|
+
});
|
33
|
+
module.exports = __toCommonJS(extensionContextFactory_exports);
|
34
|
+
var playwright = __toESM(require("playwright-core"));
|
35
|
+
var import_utilsBundle = require("playwright-core/lib/utilsBundle");
|
36
|
+
var import_http = require("../sdk/http");
|
37
|
+
var import_cdpRelay = require("./cdpRelay");
|
38
|
+
const debugLogger = (0, import_utilsBundle.debug)("pw:mcp:relay");
|
39
|
+
class ExtensionContextFactory {
|
40
|
+
constructor(browserChannel, userDataDir, executablePath) {
|
41
|
+
this._browserChannel = browserChannel;
|
42
|
+
this._userDataDir = userDataDir;
|
43
|
+
this._executablePath = executablePath;
|
44
|
+
}
|
45
|
+
async createContext(clientInfo, abortSignal, toolName) {
|
46
|
+
const browser = await this._obtainBrowser(clientInfo, abortSignal, toolName);
|
47
|
+
return {
|
48
|
+
browserContext: browser.contexts()[0],
|
49
|
+
close: async () => {
|
50
|
+
debugLogger("close() called for browser context");
|
51
|
+
await browser.close();
|
52
|
+
}
|
53
|
+
};
|
54
|
+
}
|
55
|
+
async _obtainBrowser(clientInfo, abortSignal, toolName) {
|
56
|
+
const relay = await this._startRelay(abortSignal);
|
57
|
+
await relay.ensureExtensionConnectionForMCPContext(clientInfo, abortSignal, toolName);
|
58
|
+
return await playwright.chromium.connectOverCDP(relay.cdpEndpoint());
|
59
|
+
}
|
60
|
+
async _startRelay(abortSignal) {
|
61
|
+
const httpServer = await (0, import_http.startHttpServer)({});
|
62
|
+
if (abortSignal.aborted) {
|
63
|
+
httpServer.close();
|
64
|
+
throw new Error(abortSignal.reason);
|
65
|
+
}
|
66
|
+
const cdpRelayServer = new import_cdpRelay.CDPRelayServer(httpServer, this._browserChannel, this._userDataDir, this._executablePath);
|
67
|
+
abortSignal.addEventListener("abort", () => cdpRelayServer.stop());
|
68
|
+
debugLogger(`CDP relay server started, extension endpoint: ${cdpRelayServer.extensionEndpoint()}.`);
|
69
|
+
return cdpRelayServer;
|
70
|
+
}
|
71
|
+
}
|
72
|
+
// Annotate the CommonJS export names for ESM import in node:
|
73
|
+
0 && (module.exports = {
|
74
|
+
ExtensionContextFactory
|
75
|
+
});
|
@@ -0,0 +1,28 @@
|
|
1
|
+
"use strict";
|
2
|
+
var __defProp = Object.defineProperty;
|
3
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
4
|
+
var __getOwnPropNames = Object.getOwnPropertyNames;
|
5
|
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
6
|
+
var __export = (target, all) => {
|
7
|
+
for (var name in all)
|
8
|
+
__defProp(target, name, { get: all[name], enumerable: true });
|
9
|
+
};
|
10
|
+
var __copyProps = (to, from, except, desc) => {
|
11
|
+
if (from && typeof from === "object" || typeof from === "function") {
|
12
|
+
for (let key of __getOwnPropNames(from))
|
13
|
+
if (!__hasOwnProp.call(to, key) && key !== except)
|
14
|
+
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
15
|
+
}
|
16
|
+
return to;
|
17
|
+
};
|
18
|
+
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
19
|
+
var protocol_exports = {};
|
20
|
+
__export(protocol_exports, {
|
21
|
+
VERSION: () => VERSION
|
22
|
+
});
|
23
|
+
module.exports = __toCommonJS(protocol_exports);
|
24
|
+
const VERSION = 1;
|
25
|
+
// Annotate the CommonJS export names for ESM import in node:
|
26
|
+
0 && (module.exports = {
|
27
|
+
VERSION
|
28
|
+
});
|
package/lib/mcp/index.js
ADDED
@@ -0,0 +1,61 @@
|
|
1
|
+
"use strict";
|
2
|
+
var __create = Object.create;
|
3
|
+
var __defProp = Object.defineProperty;
|
4
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
5
|
+
var __getOwnPropNames = Object.getOwnPropertyNames;
|
6
|
+
var __getProtoOf = Object.getPrototypeOf;
|
7
|
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
8
|
+
var __export = (target, all) => {
|
9
|
+
for (var name in all)
|
10
|
+
__defProp(target, name, { get: all[name], enumerable: true });
|
11
|
+
};
|
12
|
+
var __copyProps = (to, from, except, desc) => {
|
13
|
+
if (from && typeof from === "object" || typeof from === "function") {
|
14
|
+
for (let key of __getOwnPropNames(from))
|
15
|
+
if (!__hasOwnProp.call(to, key) && key !== except)
|
16
|
+
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
17
|
+
}
|
18
|
+
return to;
|
19
|
+
};
|
20
|
+
var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
|
21
|
+
// If the importer is in node compatibility mode or this is not an ESM
|
22
|
+
// file that has been converted to a CommonJS file using a Babel-
|
23
|
+
// compatible transform (i.e. "__esModule" has not been set), then set
|
24
|
+
// "default" to the CommonJS "module.exports" for node compatibility.
|
25
|
+
isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
|
26
|
+
mod
|
27
|
+
));
|
28
|
+
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
29
|
+
var mcp_exports = {};
|
30
|
+
__export(mcp_exports, {
|
31
|
+
createConnection: () => createConnection
|
32
|
+
});
|
33
|
+
module.exports = __toCommonJS(mcp_exports);
|
34
|
+
var import_browserServerBackend = require("./browser/browserServerBackend");
|
35
|
+
var import_config = require("./browser/config");
|
36
|
+
var import_browserContextFactory = require("./browser/browserContextFactory");
|
37
|
+
var mcpServer = __toESM(require("./sdk/server"));
|
38
|
+
const packageJSON = require("../../package.json");
|
39
|
+
async function createConnection(userConfig = {}, contextGetter) {
|
40
|
+
const config = await (0, import_config.resolveConfig)(userConfig);
|
41
|
+
const factory = contextGetter ? new SimpleBrowserContextFactory(contextGetter) : (0, import_browserContextFactory.contextFactory)(config);
|
42
|
+
return mcpServer.createServer("Playwright", packageJSON.version, new import_browserServerBackend.BrowserServerBackend(config, factory), false);
|
43
|
+
}
|
44
|
+
class SimpleBrowserContextFactory {
|
45
|
+
constructor(contextGetter) {
|
46
|
+
this.name = "custom";
|
47
|
+
this.description = "Connect to a browser using a custom context getter";
|
48
|
+
this._contextGetter = contextGetter;
|
49
|
+
}
|
50
|
+
async createContext() {
|
51
|
+
const browserContext = await this._contextGetter();
|
52
|
+
return {
|
53
|
+
browserContext,
|
54
|
+
close: () => browserContext.close()
|
55
|
+
};
|
56
|
+
}
|
57
|
+
}
|
58
|
+
// Annotate the CommonJS export names for ESM import in node:
|
59
|
+
0 && (module.exports = {
|
60
|
+
createConnection
|
61
|
+
});
|
package/lib/mcp/log.js
ADDED
@@ -0,0 +1,35 @@
|
|
1
|
+
"use strict";
|
2
|
+
var __defProp = Object.defineProperty;
|
3
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
4
|
+
var __getOwnPropNames = Object.getOwnPropertyNames;
|
5
|
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
6
|
+
var __export = (target, all) => {
|
7
|
+
for (var name in all)
|
8
|
+
__defProp(target, name, { get: all[name], enumerable: true });
|
9
|
+
};
|
10
|
+
var __copyProps = (to, from, except, desc) => {
|
11
|
+
if (from && typeof from === "object" || typeof from === "function") {
|
12
|
+
for (let key of __getOwnPropNames(from))
|
13
|
+
if (!__hasOwnProp.call(to, key) && key !== except)
|
14
|
+
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
15
|
+
}
|
16
|
+
return to;
|
17
|
+
};
|
18
|
+
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
19
|
+
var log_exports = {};
|
20
|
+
__export(log_exports, {
|
21
|
+
logUnhandledError: () => logUnhandledError,
|
22
|
+
testDebug: () => testDebug
|
23
|
+
});
|
24
|
+
module.exports = __toCommonJS(log_exports);
|
25
|
+
var import_utilsBundle = require("playwright-core/lib/utilsBundle");
|
26
|
+
const errorDebug = (0, import_utilsBundle.debug)("pw:mcp:error");
|
27
|
+
function logUnhandledError(error) {
|
28
|
+
errorDebug(error);
|
29
|
+
}
|
30
|
+
const testDebug = (0, import_utilsBundle.debug)("pw:mcp:test");
|
31
|
+
// Annotate the CommonJS export names for ESM import in node:
|
32
|
+
0 && (module.exports = {
|
33
|
+
logUnhandledError,
|
34
|
+
testDebug
|
35
|
+
});
|
@@ -0,0 +1,96 @@
|
|
1
|
+
"use strict";
|
2
|
+
var __create = Object.create;
|
3
|
+
var __defProp = Object.defineProperty;
|
4
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
5
|
+
var __getOwnPropNames = Object.getOwnPropertyNames;
|
6
|
+
var __getProtoOf = Object.getPrototypeOf;
|
7
|
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
8
|
+
var __export = (target, all) => {
|
9
|
+
for (var name in all)
|
10
|
+
__defProp(target, name, { get: all[name], enumerable: true });
|
11
|
+
};
|
12
|
+
var __copyProps = (to, from, except, desc) => {
|
13
|
+
if (from && typeof from === "object" || typeof from === "function") {
|
14
|
+
for (let key of __getOwnPropNames(from))
|
15
|
+
if (!__hasOwnProp.call(to, key) && key !== except)
|
16
|
+
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
17
|
+
}
|
18
|
+
return to;
|
19
|
+
};
|
20
|
+
var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
|
21
|
+
// If the importer is in node compatibility mode or this is not an ESM
|
22
|
+
// file that has been converted to a CommonJS file using a Babel-
|
23
|
+
// compatible transform (i.e. "__esModule" has not been set), then set
|
24
|
+
// "default" to the CommonJS "module.exports" for node compatibility.
|
25
|
+
isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
|
26
|
+
mod
|
27
|
+
));
|
28
|
+
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
29
|
+
var program_exports = {};
|
30
|
+
__export(program_exports, {
|
31
|
+
decorateCommand: () => decorateCommand
|
32
|
+
});
|
33
|
+
module.exports = __toCommonJS(program_exports);
|
34
|
+
var import_utilsBundle = require("playwright-core/lib/utilsBundle");
|
35
|
+
var mcpServer = __toESM(require("./sdk/server"));
|
36
|
+
var import_config = require("./browser/config");
|
37
|
+
var import_watchdog = require("./browser/watchdog");
|
38
|
+
var import_browserContextFactory = require("./browser/browserContextFactory");
|
39
|
+
var import_proxyBackend = require("./sdk/proxyBackend");
|
40
|
+
var import_browserServerBackend = require("./browser/browserServerBackend");
|
41
|
+
var import_extensionContextFactory = require("./extension/extensionContextFactory");
|
42
|
+
function decorateCommand(command, version) {
|
43
|
+
command.option("--allowed-hosts <hosts...>", "comma-separated list of hosts this server is allowed to serve from. Defaults to the host the server is bound to. Pass '*' to disable the host check.", import_config.commaSeparatedList).option("--allowed-origins <origins>", "semicolon-separated list of origins to allow the browser to request. Default is to allow all.", import_config.semicolonSeparatedList).option("--blocked-origins <origins>", "semicolon-separated list of origins to block the browser from requesting. Blocklist is evaluated before allowlist. If used without the allowlist, requests not matching the blocklist are still allowed.", import_config.semicolonSeparatedList).option("--block-service-workers", "block service workers").option("--browser <browser>", "browser or chrome channel to use, possible values: chrome, firefox, webkit, msedge.").option("--caps <caps>", "comma-separated list of additional capabilities to enable, possible values: vision, pdf.", import_config.commaSeparatedList).option("--cdp-endpoint <endpoint>", "CDP endpoint to connect to.").option("--cdp-header <headers...>", "CDP headers to send with the connect request, multiple can be specified.", import_config.headerParser).option("--config <path>", "path to the configuration file.").option("--device <device>", 'device to emulate, for example: "iPhone 15"').option("--executable-path <path>", "path to the browser executable.").option("--extension", 'Connect to a running browser instance (Edge/Chrome only). Requires the "Playwright MCP Bridge" browser extension to be installed.').option("--grant-permissions <permissions...>", 'List of permissions to grant to the browser context, for example "geolocation", "clipboard-read", "clipboard-write".', import_config.commaSeparatedList).option("--headless", "run browser in headless mode, headed by default").option("--host <host>", "host to bind server to. Default is localhost. Use 0.0.0.0 to bind to all interfaces.").option("--ignore-https-errors", "ignore https errors").option("--init-script <path...>", "path to JavaScript file to add as an initialization script. The script will be evaluated in every page before any of the page's scripts. Can be specified multiple times.").option("--isolated", "keep the browser profile in memory, do not save it to disk.").option("--image-responses <mode>", 'whether to send image responses to the client. Can be "allow" or "omit", Defaults to "allow".').option("--no-sandbox", "disable the sandbox for all process types that are normally sandboxed.").option("--output-dir <path>", "path to the directory for output files.").option("--port <port>", "port to listen on for SSE transport.").option("--proxy-bypass <bypass>", 'comma-separated domains to bypass proxy, for example ".com,chromium.org,.domain.com"').option("--proxy-server <proxy>", 'specify proxy server, for example "http://myproxy:3128" or "socks5://myproxy:8080"').option("--save-session", "Whether to save the Playwright MCP session into the output directory.").option("--save-trace", "Whether to save the Playwright Trace of the session into the output directory.").option("--save-video <size>", 'Whether to save the video of the session into the output directory. For example "--save-video=800x600"', import_config.resolutionParser.bind(null, "--save-video")).option("--secrets <path>", "path to a file containing secrets in the dotenv format", import_config.dotenvFileLoader).option("--shared-browser-context", "reuse the same browser context between all connected HTTP clients.").option("--storage-state <path>", "path to the storage state file for isolated sessions.").option("--test-id-attribute <attribute>", 'specify the attribute to use for test ids, defaults to "data-testid"').option("--timeout-action <timeout>", "specify action timeout in milliseconds, defaults to 5000ms", import_config.numberParser).option("--timeout-navigation <timeout>", "specify navigation timeout in milliseconds, defaults to 60000ms", import_config.numberParser).option("--user-agent <ua string>", "specify user agent string").option("--user-data-dir <path>", "path to the user data directory. If not specified, a temporary directory will be created.").option("--viewport-size <size>", 'specify browser viewport size in pixels, for example "1280x720"', import_config.resolutionParser.bind(null, "--viewport-size")).addOption(new import_utilsBundle.ProgramOption("--connect-tool", "Allow to switch between different browser connection methods.").hideHelp()).addOption(new import_utilsBundle.ProgramOption("--vision", "Legacy option, use --caps=vision instead").hideHelp()).action(async (options) => {
|
44
|
+
(0, import_watchdog.setupExitWatchdog)();
|
45
|
+
if (options.vision) {
|
46
|
+
console.error("The --vision option is deprecated, use --caps=vision instead");
|
47
|
+
options.caps = "vision";
|
48
|
+
}
|
49
|
+
const config = await (0, import_config.resolveCLIConfig)(options);
|
50
|
+
const browserContextFactory = (0, import_browserContextFactory.contextFactory)(config);
|
51
|
+
const extensionContextFactory = new import_extensionContextFactory.ExtensionContextFactory(config.browser.launchOptions.channel || "chrome", config.browser.userDataDir, config.browser.launchOptions.executablePath);
|
52
|
+
if (options.extension) {
|
53
|
+
const serverBackendFactory = {
|
54
|
+
name: "Playwright w/ extension",
|
55
|
+
nameInConfig: "playwright-extension",
|
56
|
+
version,
|
57
|
+
create: () => new import_browserServerBackend.BrowserServerBackend(config, extensionContextFactory)
|
58
|
+
};
|
59
|
+
await mcpServer.start(serverBackendFactory, config.server);
|
60
|
+
return;
|
61
|
+
}
|
62
|
+
if (options.connectTool) {
|
63
|
+
const providers = [
|
64
|
+
{
|
65
|
+
name: "default",
|
66
|
+
description: "Starts standalone browser",
|
67
|
+
connect: () => mcpServer.wrapInProcess(new import_browserServerBackend.BrowserServerBackend(config, browserContextFactory))
|
68
|
+
},
|
69
|
+
{
|
70
|
+
name: "extension",
|
71
|
+
description: "Connect to a browser using the Playwright MCP extension",
|
72
|
+
connect: () => mcpServer.wrapInProcess(new import_browserServerBackend.BrowserServerBackend(config, extensionContextFactory))
|
73
|
+
}
|
74
|
+
];
|
75
|
+
const factory2 = {
|
76
|
+
name: "Playwright w/ switch",
|
77
|
+
nameInConfig: "playwright-switch",
|
78
|
+
version,
|
79
|
+
create: () => new import_proxyBackend.ProxyBackend(providers)
|
80
|
+
};
|
81
|
+
await mcpServer.start(factory2, config.server);
|
82
|
+
return;
|
83
|
+
}
|
84
|
+
const factory = {
|
85
|
+
name: "Playwright",
|
86
|
+
nameInConfig: "playwright",
|
87
|
+
version,
|
88
|
+
create: () => new import_browserServerBackend.BrowserServerBackend(config, browserContextFactory)
|
89
|
+
};
|
90
|
+
await mcpServer.start(factory, config.server);
|
91
|
+
});
|
92
|
+
}
|
93
|
+
// Annotate the CommonJS export names for ESM import in node:
|
94
|
+
0 && (module.exports = {
|
95
|
+
decorateCommand
|
96
|
+
});
|