patchright-bun 1.58.0 → 1.58.2
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/README.md +225 -225
- package/ThirdPartyNotices.txt +5041 -5041
- package/cli.js +19 -19
- package/index.d.ts +17 -17
- package/index.js +17 -17
- package/index.mjs +18 -18
- package/jsx-runtime.js +42 -42
- package/jsx-runtime.mjs +21 -21
- package/lib/agents/agentParser.js +89 -89
- package/lib/agents/copilot-setup-steps.yml +34 -34
- package/lib/agents/generateAgents.js +348 -348
- package/lib/agents/playwright-test-coverage.prompt.md +31 -31
- package/lib/agents/playwright-test-generate.prompt.md +8 -8
- package/lib/agents/playwright-test-generator.agent.md +88 -88
- package/lib/agents/playwright-test-heal.prompt.md +6 -6
- package/lib/agents/playwright-test-healer.agent.md +55 -55
- package/lib/agents/playwright-test-plan.prompt.md +9 -9
- package/lib/agents/playwright-test-planner.agent.md +73 -73
- package/lib/common/config.js +282 -282
- package/lib/common/configLoader.js +344 -344
- package/lib/common/esmLoaderHost.js +104 -104
- package/lib/common/expectBundle.js +28 -28
- package/lib/common/expectBundleImpl.js +407 -407
- package/lib/common/fixtures.js +302 -302
- package/lib/common/globals.js +58 -58
- package/lib/common/ipc.js +60 -60
- package/lib/common/poolBuilder.js +85 -85
- package/lib/common/process.js +132 -132
- package/lib/common/suiteUtils.js +140 -140
- package/lib/common/test.js +321 -321
- package/lib/common/testLoader.js +101 -101
- package/lib/common/testType.js +298 -298
- package/lib/common/validators.js +68 -68
- package/lib/fsWatcher.js +67 -67
- package/lib/index.js +726 -726
- package/lib/internalsForTest.js +42 -42
- package/lib/isomorphic/events.js +77 -77
- package/lib/isomorphic/folders.js +30 -30
- package/lib/isomorphic/stringInternPool.js +69 -69
- package/lib/isomorphic/teleReceiver.js +521 -521
- package/lib/isomorphic/teleSuiteUpdater.js +157 -157
- package/lib/isomorphic/testServerConnection.js +225 -225
- package/lib/isomorphic/testServerInterface.js +16 -16
- package/lib/isomorphic/testTree.js +329 -329
- package/lib/isomorphic/types.d.js +16 -16
- package/lib/loader/loaderMain.js +59 -59
- package/lib/matchers/expect.js +311 -311
- package/lib/matchers/matcherHint.js +44 -44
- package/lib/matchers/matchers.js +383 -383
- package/lib/matchers/toBeTruthy.js +75 -75
- package/lib/matchers/toEqual.js +100 -100
- package/lib/matchers/toHaveURL.js +101 -101
- package/lib/matchers/toMatchAriaSnapshot.js +159 -159
- package/lib/matchers/toMatchSnapshot.js +342 -342
- package/lib/matchers/toMatchText.js +99 -99
- package/lib/mcp/browser/browserContextFactory.js +329 -329
- package/lib/mcp/browser/browserServerBackend.js +84 -84
- package/lib/mcp/browser/config.js +421 -421
- package/lib/mcp/browser/context.js +244 -244
- package/lib/mcp/browser/response.js +278 -278
- package/lib/mcp/browser/sessionLog.js +75 -75
- package/lib/mcp/browser/tab.js +343 -343
- package/lib/mcp/browser/tools/common.js +65 -65
- package/lib/mcp/browser/tools/console.js +46 -46
- package/lib/mcp/browser/tools/dialogs.js +60 -60
- package/lib/mcp/browser/tools/evaluate.js +61 -61
- package/lib/mcp/browser/tools/files.js +58 -58
- package/lib/mcp/browser/tools/form.js +63 -63
- package/lib/mcp/browser/tools/install.js +72 -72
- package/lib/mcp/browser/tools/keyboard.js +107 -107
- package/lib/mcp/browser/tools/mouse.js +107 -107
- package/lib/mcp/browser/tools/navigate.js +71 -71
- package/lib/mcp/browser/tools/network.js +63 -63
- package/lib/mcp/browser/tools/open.js +57 -57
- package/lib/mcp/browser/tools/pdf.js +49 -49
- package/lib/mcp/browser/tools/runCode.js +78 -78
- package/lib/mcp/browser/tools/screenshot.js +93 -93
- package/lib/mcp/browser/tools/snapshot.js +173 -173
- package/lib/mcp/browser/tools/tabs.js +67 -67
- package/lib/mcp/browser/tools/tool.js +47 -47
- package/lib/mcp/browser/tools/tracing.js +74 -74
- package/lib/mcp/browser/tools/utils.js +94 -94
- package/lib/mcp/browser/tools/verify.js +143 -143
- package/lib/mcp/browser/tools/wait.js +63 -63
- package/lib/mcp/browser/tools.js +84 -84
- package/lib/mcp/browser/watchdog.js +44 -44
- package/lib/mcp/config.d.js +16 -16
- package/lib/mcp/extension/cdpRelay.js +351 -351
- package/lib/mcp/extension/extensionContextFactory.js +76 -76
- package/lib/mcp/extension/protocol.js +28 -28
- package/lib/mcp/index.js +61 -61
- package/lib/mcp/log.js +35 -35
- package/lib/mcp/program.js +111 -111
- package/lib/mcp/sdk/exports.js +28 -28
- package/lib/mcp/sdk/http.js +152 -152
- package/lib/mcp/sdk/inProcessTransport.js +71 -71
- package/lib/mcp/sdk/server.js +223 -223
- package/lib/mcp/sdk/tool.js +47 -47
- package/lib/mcp/terminal/cli.js +296 -296
- package/lib/mcp/terminal/command.js +56 -56
- package/lib/mcp/terminal/commands.js +333 -333
- package/lib/mcp/terminal/daemon.js +129 -129
- package/lib/mcp/terminal/help.json +31 -31
- package/lib/mcp/terminal/helpGenerator.js +88 -88
- package/lib/mcp/terminal/socketConnection.js +80 -80
- package/lib/mcp/test/browserBackend.js +98 -98
- package/lib/mcp/test/generatorTools.js +122 -122
- package/lib/mcp/test/plannerTools.js +145 -145
- package/lib/mcp/test/seed.js +82 -82
- package/lib/mcp/test/streams.js +44 -44
- package/lib/mcp/test/testBackend.js +99 -99
- package/lib/mcp/test/testContext.js +285 -285
- package/lib/mcp/test/testTool.js +30 -30
- package/lib/mcp/test/testTools.js +108 -108
- package/lib/plugins/gitCommitInfoPlugin.js +198 -198
- package/lib/plugins/index.js +28 -28
- package/lib/plugins/webServerPlugin.js +237 -237
- package/lib/program.js +417 -417
- package/lib/reporters/base.js +634 -634
- package/lib/reporters/blob.js +138 -138
- package/lib/reporters/dot.js +99 -99
- package/lib/reporters/empty.js +32 -32
- package/lib/reporters/github.js +128 -128
- package/lib/reporters/html.js +633 -633
- package/lib/reporters/internalReporter.js +138 -138
- package/lib/reporters/json.js +254 -254
- package/lib/reporters/junit.js +232 -232
- package/lib/reporters/line.js +131 -131
- package/lib/reporters/list.js +253 -253
- package/lib/reporters/listModeReporter.js +69 -69
- package/lib/reporters/markdown.js +144 -144
- package/lib/reporters/merge.js +558 -558
- package/lib/reporters/multiplexer.js +112 -112
- package/lib/reporters/reporterV2.js +102 -102
- package/lib/reporters/teleEmitter.js +317 -317
- package/lib/reporters/versions/blobV1.js +16 -16
- package/lib/runner/dispatcher.js +530 -530
- package/lib/runner/failureTracker.js +72 -72
- package/lib/runner/lastRun.js +77 -77
- package/lib/runner/loadUtils.js +334 -334
- package/lib/runner/loaderHost.js +89 -89
- package/lib/runner/processHost.js +180 -180
- package/lib/runner/projectUtils.js +241 -241
- package/lib/runner/rebase.js +189 -189
- package/lib/runner/reporters.js +138 -138
- package/lib/runner/sigIntWatcher.js +96 -96
- package/lib/runner/storage.js +91 -91
- package/lib/runner/taskRunner.js +127 -127
- package/lib/runner/tasks.js +410 -410
- package/lib/runner/testGroups.js +125 -125
- package/lib/runner/testRunner.js +398 -398
- package/lib/runner/testServer.js +269 -269
- package/lib/runner/uiModeReporter.js +30 -30
- package/lib/runner/vcs.js +72 -72
- package/lib/runner/watchMode.js +396 -396
- package/lib/runner/workerHost.js +104 -104
- package/lib/third_party/pirates.js +62 -62
- package/lib/third_party/tsconfig-loader.js +103 -103
- package/lib/transform/babelBundle.js +46 -46
- package/lib/transform/babelBundleImpl.js +461 -461
- package/lib/transform/compilationCache.js +274 -274
- package/lib/transform/esmLoader.js +103 -103
- package/lib/transform/md.js +221 -221
- package/lib/transform/portTransport.js +67 -67
- package/lib/transform/transform.js +303 -303
- package/lib/util.js +400 -400
- package/lib/utilsBundle.js +50 -50
- package/lib/utilsBundleImpl.js +103 -103
- package/lib/worker/fixtureRunner.js +262 -262
- package/lib/worker/testInfo.js +536 -536
- package/lib/worker/testTracing.js +345 -345
- package/lib/worker/timeoutManager.js +174 -174
- package/lib/worker/util.js +31 -31
- package/lib/worker/workerMain.js +530 -530
- package/package.json +2 -2
- package/test.d.ts +18 -18
- package/test.js +24 -24
- package/test.mjs +34 -34
- package/types/test.d.ts +10251 -10251
- package/types/testReporter.d.ts +822 -822
package/lib/mcp/terminal/cli.js
CHANGED
|
@@ -1,296 +1,296 @@
|
|
|
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 __copyProps = (to, from, except, desc) => {
|
|
9
|
-
if (from && typeof from === "object" || typeof from === "function") {
|
|
10
|
-
for (let key of __getOwnPropNames(from))
|
|
11
|
-
if (!__hasOwnProp.call(to, key) && key !== except)
|
|
12
|
-
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
|
13
|
-
}
|
|
14
|
-
return to;
|
|
15
|
-
};
|
|
16
|
-
var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
|
|
17
|
-
// If the importer is in node compatibility mode or this is not an ESM
|
|
18
|
-
// file that has been converted to a CommonJS file using a Babel-
|
|
19
|
-
// compatible transform (i.e. "__esModule" has not been set), then set
|
|
20
|
-
// "default" to the CommonJS "module.exports" for node compatibility.
|
|
21
|
-
isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
|
|
22
|
-
mod
|
|
23
|
-
));
|
|
24
|
-
var import_child_process = require("child_process");
|
|
25
|
-
var import_crypto = __toESM(require("crypto"));
|
|
26
|
-
var import_fs = __toESM(require("fs"));
|
|
27
|
-
var import_net = __toESM(require("net"));
|
|
28
|
-
var import_os = __toESM(require("os"));
|
|
29
|
-
var import_path = __toESM(require("path"));
|
|
30
|
-
var import_utilsBundle = require("patchright-core/lib/utilsBundle");
|
|
31
|
-
var import_socketConnection = require("./socketConnection");
|
|
32
|
-
const debugCli = (0, import_utilsBundle.debug)("pw:cli");
|
|
33
|
-
const packageJSON = require("../../../package.json");
|
|
34
|
-
async function runCliCommand(sessionName, args) {
|
|
35
|
-
const session = await connectToDaemon(sessionName);
|
|
36
|
-
const result = await session.runCliCommand(args);
|
|
37
|
-
console.log(result);
|
|
38
|
-
session.dispose();
|
|
39
|
-
}
|
|
40
|
-
async function socketExists(socketPath) {
|
|
41
|
-
try {
|
|
42
|
-
const stat = await import_fs.default.promises.stat(socketPath);
|
|
43
|
-
if (stat?.isSocket())
|
|
44
|
-
return true;
|
|
45
|
-
} catch (e) {
|
|
46
|
-
}
|
|
47
|
-
return false;
|
|
48
|
-
}
|
|
49
|
-
class SocketSession {
|
|
50
|
-
constructor(connection) {
|
|
51
|
-
this._nextMessageId = 1;
|
|
52
|
-
this._callbacks = /* @__PURE__ */ new Map();
|
|
53
|
-
this._connection = connection;
|
|
54
|
-
this._connection.onmessage = (message) => this._onMessage(message);
|
|
55
|
-
this._connection.onclose = () => this.dispose();
|
|
56
|
-
}
|
|
57
|
-
async callTool(name, args) {
|
|
58
|
-
return this._send(name, args);
|
|
59
|
-
}
|
|
60
|
-
async runCliCommand(args) {
|
|
61
|
-
return await this._send("runCliCommand", { args });
|
|
62
|
-
}
|
|
63
|
-
async _send(method, params = {}) {
|
|
64
|
-
const messageId = this._nextMessageId++;
|
|
65
|
-
const message = {
|
|
66
|
-
id: messageId,
|
|
67
|
-
method,
|
|
68
|
-
params
|
|
69
|
-
};
|
|
70
|
-
await this._connection.send(message);
|
|
71
|
-
return new Promise((resolve, reject) => {
|
|
72
|
-
this._callbacks.set(messageId, { resolve, reject });
|
|
73
|
-
});
|
|
74
|
-
}
|
|
75
|
-
dispose() {
|
|
76
|
-
for (const callback of this._callbacks.values())
|
|
77
|
-
callback.reject(new Error("Disposed"));
|
|
78
|
-
this._callbacks.clear();
|
|
79
|
-
this._connection.close();
|
|
80
|
-
}
|
|
81
|
-
_onMessage(object) {
|
|
82
|
-
if (object.id && this._callbacks.has(object.id)) {
|
|
83
|
-
const callback = this._callbacks.get(object.id);
|
|
84
|
-
this._callbacks.delete(object.id);
|
|
85
|
-
if (object.error)
|
|
86
|
-
callback.reject(new Error(object.error));
|
|
87
|
-
else
|
|
88
|
-
callback.resolve(object.result);
|
|
89
|
-
} else if (object.id) {
|
|
90
|
-
throw new Error(`Unexpected message id: ${object.id}`);
|
|
91
|
-
} else {
|
|
92
|
-
throw new Error(`Unexpected message without id: ${JSON.stringify(object)}`);
|
|
93
|
-
}
|
|
94
|
-
}
|
|
95
|
-
}
|
|
96
|
-
function localCacheDir() {
|
|
97
|
-
if (process.platform === "linux")
|
|
98
|
-
return process.env.XDG_CACHE_HOME || import_path.default.join(import_os.default.homedir(), ".cache");
|
|
99
|
-
if (process.platform === "darwin")
|
|
100
|
-
return import_path.default.join(import_os.default.homedir(), "Library", "Caches");
|
|
101
|
-
if (process.platform === "win32")
|
|
102
|
-
return process.env.LOCALAPPDATA || import_path.default.join(import_os.default.homedir(), "AppData", "Local");
|
|
103
|
-
throw new Error("Unsupported platform: " + process.platform);
|
|
104
|
-
}
|
|
105
|
-
function playwrightCacheDir() {
|
|
106
|
-
return import_path.default.join(localCacheDir(), "ms-playwright");
|
|
107
|
-
}
|
|
108
|
-
function calculateSha1(buffer) {
|
|
109
|
-
const hash = import_crypto.default.createHash("sha1");
|
|
110
|
-
hash.update(buffer);
|
|
111
|
-
return hash.digest("hex");
|
|
112
|
-
}
|
|
113
|
-
function socketDirHash() {
|
|
114
|
-
return calculateSha1(__dirname);
|
|
115
|
-
}
|
|
116
|
-
function daemonSocketDir() {
|
|
117
|
-
return import_path.default.resolve(playwrightCacheDir(), "daemon", socketDirHash());
|
|
118
|
-
}
|
|
119
|
-
function daemonSocketPath(sessionName) {
|
|
120
|
-
const socketName = `${sessionName}.sock`;
|
|
121
|
-
if (import_os.default.platform() === "win32")
|
|
122
|
-
return `\\\\.\\pipe\\${socketDirHash()}-${socketName}`;
|
|
123
|
-
return import_path.default.resolve(daemonSocketDir(), socketName);
|
|
124
|
-
}
|
|
125
|
-
async function connectToDaemon(sessionName) {
|
|
126
|
-
const socketPath = daemonSocketPath(sessionName);
|
|
127
|
-
debugCli(`Connecting to daemon at ${socketPath}`);
|
|
128
|
-
if (await socketExists(socketPath)) {
|
|
129
|
-
debugCli(`Socket file exists, attempting to connect...`);
|
|
130
|
-
try {
|
|
131
|
-
return await connectToSocket(socketPath);
|
|
132
|
-
} catch (e) {
|
|
133
|
-
if (import_os.default.platform() !== "win32")
|
|
134
|
-
await import_fs.default.promises.unlink(socketPath).catch(() => {
|
|
135
|
-
});
|
|
136
|
-
}
|
|
137
|
-
}
|
|
138
|
-
const cliPath = import_path.default.join(__dirname, "../../../cli.js");
|
|
139
|
-
debugCli(`Will launch daemon process: ${cliPath}`);
|
|
140
|
-
const userDataDir = import_path.default.resolve(daemonSocketDir(), `${sessionName}-user-data`);
|
|
141
|
-
const child = (0, import_child_process.spawn)(process.execPath, [cliPath, "run-mcp-server", `--daemon=${socketPath}`, `--user-data-dir=${userDataDir}`], {
|
|
142
|
-
detached: true,
|
|
143
|
-
stdio: "ignore",
|
|
144
|
-
cwd: process.cwd()
|
|
145
|
-
// Will be used as root.
|
|
146
|
-
});
|
|
147
|
-
child.unref();
|
|
148
|
-
const maxRetries = 50;
|
|
149
|
-
const retryDelay = 100;
|
|
150
|
-
for (let i = 0; i < maxRetries; i++) {
|
|
151
|
-
await new Promise((resolve) => setTimeout(resolve, 100));
|
|
152
|
-
try {
|
|
153
|
-
return await connectToSocket(socketPath);
|
|
154
|
-
} catch (e) {
|
|
155
|
-
if (e.code !== "ENOENT")
|
|
156
|
-
throw e;
|
|
157
|
-
debugCli(`Retrying to connect to daemon at ${socketPath} (${i + 1}/${maxRetries})`);
|
|
158
|
-
}
|
|
159
|
-
}
|
|
160
|
-
throw new Error(`Failed to connect to daemon at ${socketPath} after ${maxRetries * retryDelay}ms`);
|
|
161
|
-
}
|
|
162
|
-
async function connectToSocket(socketPath) {
|
|
163
|
-
const socket = await new Promise((resolve, reject) => {
|
|
164
|
-
const socket2 = import_net.default.createConnection(socketPath, () => {
|
|
165
|
-
debugCli(`Connected to daemon at ${socketPath}`);
|
|
166
|
-
resolve(socket2);
|
|
167
|
-
});
|
|
168
|
-
socket2.on("error", reject);
|
|
169
|
-
});
|
|
170
|
-
return new SocketSession(new import_socketConnection.SocketConnection(socket));
|
|
171
|
-
}
|
|
172
|
-
function currentSessionPath() {
|
|
173
|
-
return import_path.default.resolve(daemonSocketDir(), "current-session");
|
|
174
|
-
}
|
|
175
|
-
async function getCurrentSession() {
|
|
176
|
-
try {
|
|
177
|
-
const session = await import_fs.default.promises.readFile(currentSessionPath(), "utf-8");
|
|
178
|
-
return session.trim() || "default";
|
|
179
|
-
} catch {
|
|
180
|
-
return "default";
|
|
181
|
-
}
|
|
182
|
-
}
|
|
183
|
-
async function setCurrentSession(sessionName) {
|
|
184
|
-
await import_fs.default.promises.mkdir(daemonSocketDir(), { recursive: true });
|
|
185
|
-
await import_fs.default.promises.writeFile(currentSessionPath(), sessionName);
|
|
186
|
-
}
|
|
187
|
-
async function canConnectToSocket(socketPath) {
|
|
188
|
-
return new Promise((resolve) => {
|
|
189
|
-
const socket = import_net.default.createConnection(socketPath, () => {
|
|
190
|
-
socket.destroy();
|
|
191
|
-
resolve(true);
|
|
192
|
-
});
|
|
193
|
-
socket.on("error", () => {
|
|
194
|
-
resolve(false);
|
|
195
|
-
});
|
|
196
|
-
});
|
|
197
|
-
}
|
|
198
|
-
async function listSessions() {
|
|
199
|
-
const dir = daemonSocketDir();
|
|
200
|
-
try {
|
|
201
|
-
const files = await import_fs.default.promises.readdir(dir);
|
|
202
|
-
const sessions = [];
|
|
203
|
-
for (const file of files) {
|
|
204
|
-
if (file.endsWith("-user-data")) {
|
|
205
|
-
const sessionName = file.slice(0, -"-user-data".length);
|
|
206
|
-
const socketPath = daemonSocketPath(sessionName);
|
|
207
|
-
const live = await canConnectToSocket(socketPath);
|
|
208
|
-
sessions.push({ name: sessionName, live });
|
|
209
|
-
}
|
|
210
|
-
}
|
|
211
|
-
return sessions;
|
|
212
|
-
} catch {
|
|
213
|
-
return [];
|
|
214
|
-
}
|
|
215
|
-
}
|
|
216
|
-
function resolveSessionName(args) {
|
|
217
|
-
if (args.session)
|
|
218
|
-
return args.session;
|
|
219
|
-
if (process.env.PLAYWRIGHT_CLI_SESSION)
|
|
220
|
-
return process.env.PLAYWRIGHT_CLI_SESSION;
|
|
221
|
-
return "default";
|
|
222
|
-
}
|
|
223
|
-
async function handleSessionCommand(args) {
|
|
224
|
-
const subcommand = args._[1];
|
|
225
|
-
if (!subcommand) {
|
|
226
|
-
const current = await getCurrentSession();
|
|
227
|
-
console.log(current);
|
|
228
|
-
return;
|
|
229
|
-
}
|
|
230
|
-
if (subcommand === "list") {
|
|
231
|
-
const sessions = await listSessions();
|
|
232
|
-
const current = await getCurrentSession();
|
|
233
|
-
console.log("Sessions:");
|
|
234
|
-
for (const session of sessions) {
|
|
235
|
-
const marker = session.name === current ? "->" : " ";
|
|
236
|
-
const liveMarker = session.live ? " (live)" : "";
|
|
237
|
-
console.log(`${marker} ${session.name}${liveMarker}`);
|
|
238
|
-
}
|
|
239
|
-
if (sessions.length === 0)
|
|
240
|
-
console.log(" (no sessions)");
|
|
241
|
-
return;
|
|
242
|
-
}
|
|
243
|
-
if (subcommand === "set") {
|
|
244
|
-
const sessionName = args._[2];
|
|
245
|
-
if (!sessionName) {
|
|
246
|
-
console.error("Usage: playwright-cli session set <session-name>");
|
|
247
|
-
process.exit(1);
|
|
248
|
-
}
|
|
249
|
-
await setCurrentSession(sessionName);
|
|
250
|
-
console.log(`Current session set to: ${sessionName}`);
|
|
251
|
-
return;
|
|
252
|
-
}
|
|
253
|
-
console.error(`Unknown session subcommand: ${subcommand}`);
|
|
254
|
-
process.exit(1);
|
|
255
|
-
}
|
|
256
|
-
async function main() {
|
|
257
|
-
const argv = process.argv.slice(2);
|
|
258
|
-
const args = require("minimist")(argv);
|
|
259
|
-
const help = require("./help.json");
|
|
260
|
-
const commandName = args._[0];
|
|
261
|
-
if (args.version || args.v) {
|
|
262
|
-
console.log(packageJSON.version);
|
|
263
|
-
process.exit(0);
|
|
264
|
-
}
|
|
265
|
-
if (commandName === "session") {
|
|
266
|
-
await handleSessionCommand(args);
|
|
267
|
-
return;
|
|
268
|
-
}
|
|
269
|
-
const command = help.commands[commandName];
|
|
270
|
-
if (args.help || args.h) {
|
|
271
|
-
if (command) {
|
|
272
|
-
console.log(command);
|
|
273
|
-
} else {
|
|
274
|
-
console.log("playwright-cli - run playwright mcp commands from terminal\n");
|
|
275
|
-
console.log(help.global);
|
|
276
|
-
}
|
|
277
|
-
process.exit(0);
|
|
278
|
-
}
|
|
279
|
-
if (!command) {
|
|
280
|
-
console.error(`Unknown command: ${commandName}
|
|
281
|
-
`);
|
|
282
|
-
console.log(help.global);
|
|
283
|
-
process.exit(1);
|
|
284
|
-
}
|
|
285
|
-
let sessionName = resolveSessionName(args);
|
|
286
|
-
if (sessionName === "default" && !args.session && !process.env.PLAYWRIGHT_CLI_SESSION)
|
|
287
|
-
sessionName = await getCurrentSession();
|
|
288
|
-
runCliCommand(sessionName, args).catch((e) => {
|
|
289
|
-
console.error(e.message);
|
|
290
|
-
process.exit(1);
|
|
291
|
-
});
|
|
292
|
-
}
|
|
293
|
-
main().catch((e) => {
|
|
294
|
-
console.error(e.message);
|
|
295
|
-
process.exit(1);
|
|
296
|
-
});
|
|
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 __copyProps = (to, from, except, desc) => {
|
|
9
|
+
if (from && typeof from === "object" || typeof from === "function") {
|
|
10
|
+
for (let key of __getOwnPropNames(from))
|
|
11
|
+
if (!__hasOwnProp.call(to, key) && key !== except)
|
|
12
|
+
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
|
13
|
+
}
|
|
14
|
+
return to;
|
|
15
|
+
};
|
|
16
|
+
var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
|
|
17
|
+
// If the importer is in node compatibility mode or this is not an ESM
|
|
18
|
+
// file that has been converted to a CommonJS file using a Babel-
|
|
19
|
+
// compatible transform (i.e. "__esModule" has not been set), then set
|
|
20
|
+
// "default" to the CommonJS "module.exports" for node compatibility.
|
|
21
|
+
isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
|
|
22
|
+
mod
|
|
23
|
+
));
|
|
24
|
+
var import_child_process = require("child_process");
|
|
25
|
+
var import_crypto = __toESM(require("crypto"));
|
|
26
|
+
var import_fs = __toESM(require("fs"));
|
|
27
|
+
var import_net = __toESM(require("net"));
|
|
28
|
+
var import_os = __toESM(require("os"));
|
|
29
|
+
var import_path = __toESM(require("path"));
|
|
30
|
+
var import_utilsBundle = require("patchright-bun-core/lib/utilsBundle");
|
|
31
|
+
var import_socketConnection = require("./socketConnection");
|
|
32
|
+
const debugCli = (0, import_utilsBundle.debug)("pw:cli");
|
|
33
|
+
const packageJSON = require("../../../package.json");
|
|
34
|
+
async function runCliCommand(sessionName, args) {
|
|
35
|
+
const session = await connectToDaemon(sessionName);
|
|
36
|
+
const result = await session.runCliCommand(args);
|
|
37
|
+
console.log(result);
|
|
38
|
+
session.dispose();
|
|
39
|
+
}
|
|
40
|
+
async function socketExists(socketPath) {
|
|
41
|
+
try {
|
|
42
|
+
const stat = await import_fs.default.promises.stat(socketPath);
|
|
43
|
+
if (stat?.isSocket())
|
|
44
|
+
return true;
|
|
45
|
+
} catch (e) {
|
|
46
|
+
}
|
|
47
|
+
return false;
|
|
48
|
+
}
|
|
49
|
+
class SocketSession {
|
|
50
|
+
constructor(connection) {
|
|
51
|
+
this._nextMessageId = 1;
|
|
52
|
+
this._callbacks = /* @__PURE__ */ new Map();
|
|
53
|
+
this._connection = connection;
|
|
54
|
+
this._connection.onmessage = (message) => this._onMessage(message);
|
|
55
|
+
this._connection.onclose = () => this.dispose();
|
|
56
|
+
}
|
|
57
|
+
async callTool(name, args) {
|
|
58
|
+
return this._send(name, args);
|
|
59
|
+
}
|
|
60
|
+
async runCliCommand(args) {
|
|
61
|
+
return await this._send("runCliCommand", { args });
|
|
62
|
+
}
|
|
63
|
+
async _send(method, params = {}) {
|
|
64
|
+
const messageId = this._nextMessageId++;
|
|
65
|
+
const message = {
|
|
66
|
+
id: messageId,
|
|
67
|
+
method,
|
|
68
|
+
params
|
|
69
|
+
};
|
|
70
|
+
await this._connection.send(message);
|
|
71
|
+
return new Promise((resolve, reject) => {
|
|
72
|
+
this._callbacks.set(messageId, { resolve, reject });
|
|
73
|
+
});
|
|
74
|
+
}
|
|
75
|
+
dispose() {
|
|
76
|
+
for (const callback of this._callbacks.values())
|
|
77
|
+
callback.reject(new Error("Disposed"));
|
|
78
|
+
this._callbacks.clear();
|
|
79
|
+
this._connection.close();
|
|
80
|
+
}
|
|
81
|
+
_onMessage(object) {
|
|
82
|
+
if (object.id && this._callbacks.has(object.id)) {
|
|
83
|
+
const callback = this._callbacks.get(object.id);
|
|
84
|
+
this._callbacks.delete(object.id);
|
|
85
|
+
if (object.error)
|
|
86
|
+
callback.reject(new Error(object.error));
|
|
87
|
+
else
|
|
88
|
+
callback.resolve(object.result);
|
|
89
|
+
} else if (object.id) {
|
|
90
|
+
throw new Error(`Unexpected message id: ${object.id}`);
|
|
91
|
+
} else {
|
|
92
|
+
throw new Error(`Unexpected message without id: ${JSON.stringify(object)}`);
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
function localCacheDir() {
|
|
97
|
+
if (process.platform === "linux")
|
|
98
|
+
return process.env.XDG_CACHE_HOME || import_path.default.join(import_os.default.homedir(), ".cache");
|
|
99
|
+
if (process.platform === "darwin")
|
|
100
|
+
return import_path.default.join(import_os.default.homedir(), "Library", "Caches");
|
|
101
|
+
if (process.platform === "win32")
|
|
102
|
+
return process.env.LOCALAPPDATA || import_path.default.join(import_os.default.homedir(), "AppData", "Local");
|
|
103
|
+
throw new Error("Unsupported platform: " + process.platform);
|
|
104
|
+
}
|
|
105
|
+
function playwrightCacheDir() {
|
|
106
|
+
return import_path.default.join(localCacheDir(), "ms-playwright");
|
|
107
|
+
}
|
|
108
|
+
function calculateSha1(buffer) {
|
|
109
|
+
const hash = import_crypto.default.createHash("sha1");
|
|
110
|
+
hash.update(buffer);
|
|
111
|
+
return hash.digest("hex");
|
|
112
|
+
}
|
|
113
|
+
function socketDirHash() {
|
|
114
|
+
return calculateSha1(__dirname);
|
|
115
|
+
}
|
|
116
|
+
function daemonSocketDir() {
|
|
117
|
+
return import_path.default.resolve(playwrightCacheDir(), "daemon", socketDirHash());
|
|
118
|
+
}
|
|
119
|
+
function daemonSocketPath(sessionName) {
|
|
120
|
+
const socketName = `${sessionName}.sock`;
|
|
121
|
+
if (import_os.default.platform() === "win32")
|
|
122
|
+
return `\\\\.\\pipe\\${socketDirHash()}-${socketName}`;
|
|
123
|
+
return import_path.default.resolve(daemonSocketDir(), socketName);
|
|
124
|
+
}
|
|
125
|
+
async function connectToDaemon(sessionName) {
|
|
126
|
+
const socketPath = daemonSocketPath(sessionName);
|
|
127
|
+
debugCli(`Connecting to daemon at ${socketPath}`);
|
|
128
|
+
if (await socketExists(socketPath)) {
|
|
129
|
+
debugCli(`Socket file exists, attempting to connect...`);
|
|
130
|
+
try {
|
|
131
|
+
return await connectToSocket(socketPath);
|
|
132
|
+
} catch (e) {
|
|
133
|
+
if (import_os.default.platform() !== "win32")
|
|
134
|
+
await import_fs.default.promises.unlink(socketPath).catch(() => {
|
|
135
|
+
});
|
|
136
|
+
}
|
|
137
|
+
}
|
|
138
|
+
const cliPath = import_path.default.join(__dirname, "../../../cli.js");
|
|
139
|
+
debugCli(`Will launch daemon process: ${cliPath}`);
|
|
140
|
+
const userDataDir = import_path.default.resolve(daemonSocketDir(), `${sessionName}-user-data`);
|
|
141
|
+
const child = (0, import_child_process.spawn)(process.execPath, [cliPath, "run-mcp-server", `--daemon=${socketPath}`, `--user-data-dir=${userDataDir}`], {
|
|
142
|
+
detached: true,
|
|
143
|
+
stdio: "ignore",
|
|
144
|
+
cwd: process.cwd()
|
|
145
|
+
// Will be used as root.
|
|
146
|
+
});
|
|
147
|
+
child.unref();
|
|
148
|
+
const maxRetries = 50;
|
|
149
|
+
const retryDelay = 100;
|
|
150
|
+
for (let i = 0; i < maxRetries; i++) {
|
|
151
|
+
await new Promise((resolve) => setTimeout(resolve, 100));
|
|
152
|
+
try {
|
|
153
|
+
return await connectToSocket(socketPath);
|
|
154
|
+
} catch (e) {
|
|
155
|
+
if (e.code !== "ENOENT")
|
|
156
|
+
throw e;
|
|
157
|
+
debugCli(`Retrying to connect to daemon at ${socketPath} (${i + 1}/${maxRetries})`);
|
|
158
|
+
}
|
|
159
|
+
}
|
|
160
|
+
throw new Error(`Failed to connect to daemon at ${socketPath} after ${maxRetries * retryDelay}ms`);
|
|
161
|
+
}
|
|
162
|
+
async function connectToSocket(socketPath) {
|
|
163
|
+
const socket = await new Promise((resolve, reject) => {
|
|
164
|
+
const socket2 = import_net.default.createConnection(socketPath, () => {
|
|
165
|
+
debugCli(`Connected to daemon at ${socketPath}`);
|
|
166
|
+
resolve(socket2);
|
|
167
|
+
});
|
|
168
|
+
socket2.on("error", reject);
|
|
169
|
+
});
|
|
170
|
+
return new SocketSession(new import_socketConnection.SocketConnection(socket));
|
|
171
|
+
}
|
|
172
|
+
function currentSessionPath() {
|
|
173
|
+
return import_path.default.resolve(daemonSocketDir(), "current-session");
|
|
174
|
+
}
|
|
175
|
+
async function getCurrentSession() {
|
|
176
|
+
try {
|
|
177
|
+
const session = await import_fs.default.promises.readFile(currentSessionPath(), "utf-8");
|
|
178
|
+
return session.trim() || "default";
|
|
179
|
+
} catch {
|
|
180
|
+
return "default";
|
|
181
|
+
}
|
|
182
|
+
}
|
|
183
|
+
async function setCurrentSession(sessionName) {
|
|
184
|
+
await import_fs.default.promises.mkdir(daemonSocketDir(), { recursive: true });
|
|
185
|
+
await import_fs.default.promises.writeFile(currentSessionPath(), sessionName);
|
|
186
|
+
}
|
|
187
|
+
async function canConnectToSocket(socketPath) {
|
|
188
|
+
return new Promise((resolve) => {
|
|
189
|
+
const socket = import_net.default.createConnection(socketPath, () => {
|
|
190
|
+
socket.destroy();
|
|
191
|
+
resolve(true);
|
|
192
|
+
});
|
|
193
|
+
socket.on("error", () => {
|
|
194
|
+
resolve(false);
|
|
195
|
+
});
|
|
196
|
+
});
|
|
197
|
+
}
|
|
198
|
+
async function listSessions() {
|
|
199
|
+
const dir = daemonSocketDir();
|
|
200
|
+
try {
|
|
201
|
+
const files = await import_fs.default.promises.readdir(dir);
|
|
202
|
+
const sessions = [];
|
|
203
|
+
for (const file of files) {
|
|
204
|
+
if (file.endsWith("-user-data")) {
|
|
205
|
+
const sessionName = file.slice(0, -"-user-data".length);
|
|
206
|
+
const socketPath = daemonSocketPath(sessionName);
|
|
207
|
+
const live = await canConnectToSocket(socketPath);
|
|
208
|
+
sessions.push({ name: sessionName, live });
|
|
209
|
+
}
|
|
210
|
+
}
|
|
211
|
+
return sessions;
|
|
212
|
+
} catch {
|
|
213
|
+
return [];
|
|
214
|
+
}
|
|
215
|
+
}
|
|
216
|
+
function resolveSessionName(args) {
|
|
217
|
+
if (args.session)
|
|
218
|
+
return args.session;
|
|
219
|
+
if (process.env.PLAYWRIGHT_CLI_SESSION)
|
|
220
|
+
return process.env.PLAYWRIGHT_CLI_SESSION;
|
|
221
|
+
return "default";
|
|
222
|
+
}
|
|
223
|
+
async function handleSessionCommand(args) {
|
|
224
|
+
const subcommand = args._[1];
|
|
225
|
+
if (!subcommand) {
|
|
226
|
+
const current = await getCurrentSession();
|
|
227
|
+
console.log(current);
|
|
228
|
+
return;
|
|
229
|
+
}
|
|
230
|
+
if (subcommand === "list") {
|
|
231
|
+
const sessions = await listSessions();
|
|
232
|
+
const current = await getCurrentSession();
|
|
233
|
+
console.log("Sessions:");
|
|
234
|
+
for (const session of sessions) {
|
|
235
|
+
const marker = session.name === current ? "->" : " ";
|
|
236
|
+
const liveMarker = session.live ? " (live)" : "";
|
|
237
|
+
console.log(`${marker} ${session.name}${liveMarker}`);
|
|
238
|
+
}
|
|
239
|
+
if (sessions.length === 0)
|
|
240
|
+
console.log(" (no sessions)");
|
|
241
|
+
return;
|
|
242
|
+
}
|
|
243
|
+
if (subcommand === "set") {
|
|
244
|
+
const sessionName = args._[2];
|
|
245
|
+
if (!sessionName) {
|
|
246
|
+
console.error("Usage: playwright-cli session set <session-name>");
|
|
247
|
+
process.exit(1);
|
|
248
|
+
}
|
|
249
|
+
await setCurrentSession(sessionName);
|
|
250
|
+
console.log(`Current session set to: ${sessionName}`);
|
|
251
|
+
return;
|
|
252
|
+
}
|
|
253
|
+
console.error(`Unknown session subcommand: ${subcommand}`);
|
|
254
|
+
process.exit(1);
|
|
255
|
+
}
|
|
256
|
+
async function main() {
|
|
257
|
+
const argv = process.argv.slice(2);
|
|
258
|
+
const args = require("minimist")(argv);
|
|
259
|
+
const help = require("./help.json");
|
|
260
|
+
const commandName = args._[0];
|
|
261
|
+
if (args.version || args.v) {
|
|
262
|
+
console.log(packageJSON.version);
|
|
263
|
+
process.exit(0);
|
|
264
|
+
}
|
|
265
|
+
if (commandName === "session") {
|
|
266
|
+
await handleSessionCommand(args);
|
|
267
|
+
return;
|
|
268
|
+
}
|
|
269
|
+
const command = help.commands[commandName];
|
|
270
|
+
if (args.help || args.h) {
|
|
271
|
+
if (command) {
|
|
272
|
+
console.log(command);
|
|
273
|
+
} else {
|
|
274
|
+
console.log("playwright-cli - run playwright mcp commands from terminal\n");
|
|
275
|
+
console.log(help.global);
|
|
276
|
+
}
|
|
277
|
+
process.exit(0);
|
|
278
|
+
}
|
|
279
|
+
if (!command) {
|
|
280
|
+
console.error(`Unknown command: ${commandName}
|
|
281
|
+
`);
|
|
282
|
+
console.log(help.global);
|
|
283
|
+
process.exit(1);
|
|
284
|
+
}
|
|
285
|
+
let sessionName = resolveSessionName(args);
|
|
286
|
+
if (sessionName === "default" && !args.session && !process.env.PLAYWRIGHT_CLI_SESSION)
|
|
287
|
+
sessionName = await getCurrentSession();
|
|
288
|
+
runCliCommand(sessionName, args).catch((e) => {
|
|
289
|
+
console.error(e.message);
|
|
290
|
+
process.exit(1);
|
|
291
|
+
});
|
|
292
|
+
}
|
|
293
|
+
main().catch((e) => {
|
|
294
|
+
console.error(e.message);
|
|
295
|
+
process.exit(1);
|
|
296
|
+
});
|