patchright-bun 1.58.1 → 1.59.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.
Files changed (183) hide show
  1. package/README.md +225 -225
  2. package/ThirdPartyNotices.txt +3918 -5041
  3. package/cli.js +19 -19
  4. package/index.d.ts +17 -17
  5. package/index.js +17 -17
  6. package/index.mjs +18 -18
  7. package/jsx-runtime.js +42 -42
  8. package/jsx-runtime.mjs +21 -21
  9. package/lib/agents/agentParser.js +89 -89
  10. package/lib/agents/copilot-setup-steps.yml +34 -34
  11. package/lib/agents/generateAgents.js +346 -348
  12. package/lib/agents/playwright-test-coverage.prompt.md +31 -31
  13. package/lib/agents/playwright-test-generate.prompt.md +8 -8
  14. package/lib/agents/playwright-test-generator.agent.md +88 -88
  15. package/lib/agents/playwright-test-heal.prompt.md +6 -6
  16. package/lib/agents/playwright-test-healer.agent.md +55 -55
  17. package/lib/agents/playwright-test-plan.prompt.md +9 -9
  18. package/lib/agents/playwright-test-planner.agent.md +73 -73
  19. package/lib/common/config.js +281 -282
  20. package/lib/common/configLoader.js +344 -344
  21. package/lib/common/esmLoaderHost.js +104 -104
  22. package/lib/common/expectBundle.js +28 -28
  23. package/lib/common/expectBundleImpl.js +407 -407
  24. package/lib/common/fixtures.js +302 -302
  25. package/lib/common/globals.js +58 -58
  26. package/lib/common/ipc.js +60 -60
  27. package/lib/common/poolBuilder.js +85 -85
  28. package/lib/common/process.js +133 -132
  29. package/lib/common/suiteUtils.js +140 -140
  30. package/lib/common/test.js +330 -321
  31. package/lib/common/testLoader.js +101 -101
  32. package/lib/common/testType.js +301 -298
  33. package/lib/common/validators.js +68 -68
  34. package/lib/errorContext.js +121 -0
  35. package/lib/fsWatcher.js +67 -67
  36. package/lib/index.js +764 -726
  37. package/lib/internalsForTest.js +42 -42
  38. package/lib/isomorphic/events.js +77 -77
  39. package/lib/isomorphic/folders.js +30 -30
  40. package/lib/isomorphic/stringInternPool.js +69 -69
  41. package/lib/isomorphic/teleReceiver.js +538 -521
  42. package/lib/isomorphic/teleSuiteUpdater.js +157 -157
  43. package/lib/isomorphic/testServerConnection.js +223 -225
  44. package/lib/isomorphic/testServerInterface.js +16 -16
  45. package/lib/isomorphic/testTree.js +329 -329
  46. package/lib/isomorphic/types.d.js +16 -16
  47. package/lib/loader/loaderMain.js +59 -59
  48. package/lib/matchers/expect.js +311 -311
  49. package/lib/matchers/matcherHint.js +44 -44
  50. package/lib/matchers/matchers.js +385 -383
  51. package/lib/matchers/toBeTruthy.js +75 -75
  52. package/lib/matchers/toEqual.js +100 -100
  53. package/lib/matchers/toHaveURL.js +101 -101
  54. package/lib/matchers/toMatchAriaSnapshot.js +163 -159
  55. package/lib/matchers/toMatchSnapshot.js +349 -342
  56. package/lib/matchers/toMatchText.js +99 -99
  57. package/lib/mcp/test/browserBackend.js +121 -98
  58. package/lib/mcp/test/generatorTools.js +122 -122
  59. package/lib/mcp/test/plannerTools.js +145 -145
  60. package/lib/mcp/test/seed.js +82 -82
  61. package/lib/mcp/test/streams.js +44 -44
  62. package/lib/mcp/test/testBackend.js +99 -99
  63. package/lib/mcp/test/testContext.js +283 -285
  64. package/lib/mcp/test/testTool.js +30 -30
  65. package/lib/mcp/test/testTools.js +108 -108
  66. package/lib/plugins/gitCommitInfoPlugin.js +198 -198
  67. package/lib/plugins/index.js +28 -28
  68. package/lib/plugins/webServerPlugin.js +238 -237
  69. package/lib/program.js +239 -417
  70. package/lib/reportActions.js +80 -0
  71. package/lib/reporters/base.js +633 -634
  72. package/lib/reporters/blob.js +138 -138
  73. package/lib/reporters/dot.js +99 -99
  74. package/lib/reporters/empty.js +32 -32
  75. package/lib/reporters/github.js +127 -128
  76. package/lib/reporters/html.js +666 -633
  77. package/lib/reporters/internalReporter.js +138 -138
  78. package/lib/reporters/json.js +254 -254
  79. package/lib/reporters/junit.js +321 -232
  80. package/lib/reporters/line.js +131 -131
  81. package/lib/reporters/list.js +252 -253
  82. package/lib/reporters/listModeReporter.js +69 -69
  83. package/lib/reporters/markdown.js +144 -144
  84. package/lib/reporters/merge.js +579 -558
  85. package/lib/reporters/multiplexer.js +116 -112
  86. package/lib/reporters/reporterV2.js +102 -102
  87. package/lib/reporters/teleEmitter.js +319 -317
  88. package/lib/reporters/versions/blobV1.js +16 -16
  89. package/lib/runner/dispatcher.js +522 -530
  90. package/lib/runner/failureTracker.js +72 -72
  91. package/lib/runner/lastRun.js +77 -77
  92. package/lib/runner/loadUtils.js +340 -334
  93. package/lib/runner/loaderHost.js +89 -89
  94. package/lib/runner/processHost.js +180 -180
  95. package/lib/runner/projectUtils.js +241 -241
  96. package/lib/runner/rebase.js +189 -189
  97. package/lib/runner/reporters.js +143 -138
  98. package/lib/runner/sigIntWatcher.js +96 -96
  99. package/lib/runner/taskRunner.js +127 -127
  100. package/lib/runner/tasks.js +413 -410
  101. package/lib/runner/testGroups.js +125 -125
  102. package/lib/runner/testRunner.js +398 -398
  103. package/lib/runner/testServer.js +269 -269
  104. package/lib/runner/uiModeReporter.js +30 -30
  105. package/lib/runner/vcs.js +72 -72
  106. package/lib/runner/watchMode.js +396 -396
  107. package/lib/runner/workerHost.js +101 -104
  108. package/lib/testActions.js +220 -0
  109. package/lib/third_party/pirates.js +62 -62
  110. package/lib/third_party/tsconfig-loader.js +103 -103
  111. package/lib/transform/babelBundle.js +43 -46
  112. package/lib/transform/babelBundleImpl.js +461 -461
  113. package/lib/transform/compilationCache.js +272 -274
  114. package/lib/transform/esmLoader.js +105 -103
  115. package/lib/transform/portTransport.js +67 -67
  116. package/lib/transform/transform.js +296 -303
  117. package/lib/util.js +400 -400
  118. package/lib/utilsBundle.js +43 -50
  119. package/lib/utilsBundleImpl.js +100 -103
  120. package/lib/worker/fixtureRunner.js +262 -262
  121. package/lib/worker/testInfo.js +532 -536
  122. package/lib/worker/testTracing.js +351 -345
  123. package/lib/worker/timeoutManager.js +185 -174
  124. package/lib/worker/util.js +31 -31
  125. package/lib/worker/workerMain.js +533 -530
  126. package/package.json +2 -6
  127. package/test.d.ts +18 -18
  128. package/test.js +24 -24
  129. package/test.mjs +34 -34
  130. package/types/test.d.ts +10322 -10251
  131. package/types/testReporter.d.ts +822 -822
  132. package/lib/mcp/browser/browserContextFactory.js +0 -329
  133. package/lib/mcp/browser/browserServerBackend.js +0 -84
  134. package/lib/mcp/browser/config.js +0 -421
  135. package/lib/mcp/browser/context.js +0 -244
  136. package/lib/mcp/browser/response.js +0 -278
  137. package/lib/mcp/browser/sessionLog.js +0 -75
  138. package/lib/mcp/browser/tab.js +0 -343
  139. package/lib/mcp/browser/tools/common.js +0 -65
  140. package/lib/mcp/browser/tools/console.js +0 -46
  141. package/lib/mcp/browser/tools/dialogs.js +0 -60
  142. package/lib/mcp/browser/tools/evaluate.js +0 -61
  143. package/lib/mcp/browser/tools/files.js +0 -58
  144. package/lib/mcp/browser/tools/form.js +0 -63
  145. package/lib/mcp/browser/tools/install.js +0 -72
  146. package/lib/mcp/browser/tools/keyboard.js +0 -107
  147. package/lib/mcp/browser/tools/mouse.js +0 -107
  148. package/lib/mcp/browser/tools/navigate.js +0 -71
  149. package/lib/mcp/browser/tools/network.js +0 -63
  150. package/lib/mcp/browser/tools/open.js +0 -57
  151. package/lib/mcp/browser/tools/pdf.js +0 -49
  152. package/lib/mcp/browser/tools/runCode.js +0 -78
  153. package/lib/mcp/browser/tools/screenshot.js +0 -93
  154. package/lib/mcp/browser/tools/snapshot.js +0 -173
  155. package/lib/mcp/browser/tools/tabs.js +0 -67
  156. package/lib/mcp/browser/tools/tool.js +0 -47
  157. package/lib/mcp/browser/tools/tracing.js +0 -74
  158. package/lib/mcp/browser/tools/utils.js +0 -94
  159. package/lib/mcp/browser/tools/verify.js +0 -143
  160. package/lib/mcp/browser/tools/wait.js +0 -63
  161. package/lib/mcp/browser/tools.js +0 -84
  162. package/lib/mcp/browser/watchdog.js +0 -44
  163. package/lib/mcp/config.d.js +0 -16
  164. package/lib/mcp/extension/cdpRelay.js +0 -351
  165. package/lib/mcp/extension/extensionContextFactory.js +0 -76
  166. package/lib/mcp/extension/protocol.js +0 -28
  167. package/lib/mcp/index.js +0 -61
  168. package/lib/mcp/log.js +0 -35
  169. package/lib/mcp/program.js +0 -111
  170. package/lib/mcp/sdk/exports.js +0 -28
  171. package/lib/mcp/sdk/http.js +0 -152
  172. package/lib/mcp/sdk/inProcessTransport.js +0 -71
  173. package/lib/mcp/sdk/server.js +0 -223
  174. package/lib/mcp/sdk/tool.js +0 -47
  175. package/lib/mcp/terminal/cli.js +0 -296
  176. package/lib/mcp/terminal/command.js +0 -56
  177. package/lib/mcp/terminal/commands.js +0 -333
  178. package/lib/mcp/terminal/daemon.js +0 -129
  179. package/lib/mcp/terminal/help.json +0 -32
  180. package/lib/mcp/terminal/helpGenerator.js +0 -88
  181. package/lib/mcp/terminal/socketConnection.js +0 -80
  182. package/lib/runner/storage.js +0 -91
  183. package/lib/transform/md.js +0 -221
@@ -1,285 +1,283 @@
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 testContext_exports = {};
30
- __export(testContext_exports, {
31
- GeneratorJournal: () => GeneratorJournal,
32
- TestContext: () => TestContext,
33
- createScreen: () => createScreen
34
- });
35
- module.exports = __toCommonJS(testContext_exports);
36
- var import_fs = __toESM(require("fs"));
37
- var import_os = __toESM(require("os"));
38
- var import_path = __toESM(require("path"));
39
- var import_utils = require("patchright-core/lib/utils");
40
- var import_base = require("../../reporters/base");
41
- var import_list = __toESM(require("../../reporters/list"));
42
- var import_streams = require("./streams");
43
- var import_util = require("../../util");
44
- var import_testRunner = require("../../runner/testRunner");
45
- var import_seed = require("./seed");
46
- var import_exports = require("../sdk/exports");
47
- var import_configLoader = require("../../common/configLoader");
48
- var import_response = require("../browser/response");
49
- var import_log = require("../log");
50
- class GeneratorJournal {
51
- constructor(rootPath, plan, seed) {
52
- this._rootPath = rootPath;
53
- this._plan = plan;
54
- this._seed = seed;
55
- this._steps = [];
56
- }
57
- logStep(title, code) {
58
- if (title)
59
- this._steps.push({ title, code });
60
- }
61
- journal() {
62
- const result = [];
63
- result.push(`# Plan`);
64
- result.push(this._plan);
65
- result.push(`# Seed file: ${(0, import_utils.toPosixPath)(import_path.default.relative(this._rootPath, this._seed.file))}`);
66
- result.push("```ts");
67
- result.push(this._seed.content);
68
- result.push("```");
69
- result.push(`# Steps`);
70
- result.push(this._steps.map((step) => `### ${step.title}
71
- \`\`\`ts
72
- ${step.code}
73
- \`\`\``).join("\n\n"));
74
- result.push(bestPracticesMarkdown);
75
- return result.join("\n\n");
76
- }
77
- }
78
- class TestContext {
79
- constructor(clientInfo, configPath, options) {
80
- this._clientInfo = clientInfo;
81
- const rootPath = (0, import_exports.firstRootPath)(clientInfo);
82
- this._configLocation = (0, import_configLoader.resolveConfigLocation)(configPath || rootPath);
83
- this.rootPath = rootPath || this._configLocation.configDir;
84
- if (options?.headless !== void 0)
85
- this.computedHeaded = !options.headless;
86
- else
87
- this.computedHeaded = !process.env.CI && !(import_os.default.platform() === "linux" && !process.env.DISPLAY);
88
- }
89
- existingTestRunner() {
90
- return this._testRunnerAndScreen?.testRunner;
91
- }
92
- async _cleanupTestRunner() {
93
- if (!this._testRunnerAndScreen)
94
- return;
95
- await this._testRunnerAndScreen.testRunner.stopTests();
96
- this._testRunnerAndScreen.claimStdio();
97
- try {
98
- await this._testRunnerAndScreen.testRunner.runGlobalTeardown();
99
- } finally {
100
- this._testRunnerAndScreen.releaseStdio();
101
- this._testRunnerAndScreen = void 0;
102
- }
103
- }
104
- async createTestRunner() {
105
- await this._cleanupTestRunner();
106
- const testRunner = new import_testRunner.TestRunner(this._configLocation, {});
107
- await testRunner.initialize({});
108
- const testPaused = new import_utils.ManualPromise();
109
- const testRunnerAndScreen = {
110
- ...createScreen(),
111
- testRunner,
112
- waitForTestPaused: () => testPaused
113
- };
114
- this._testRunnerAndScreen = testRunnerAndScreen;
115
- testRunner.on(import_testRunner.TestRunnerEvent.TestPaused, (params) => {
116
- testRunnerAndScreen.sendMessageToPausedTest = params.sendMessage;
117
- testPaused.resolve();
118
- });
119
- return testRunnerAndScreen;
120
- }
121
- async getOrCreateSeedFile(seedFile, projectName) {
122
- const configDir = this._configLocation.configDir;
123
- const { testRunner } = await this.createTestRunner();
124
- const config = await testRunner.loadConfig();
125
- const project = (0, import_seed.seedProject)(config, projectName);
126
- if (!seedFile) {
127
- seedFile = await (0, import_seed.ensureSeedFile)(project);
128
- } else {
129
- const candidateFiles = [];
130
- const testDir = project.project.testDir;
131
- candidateFiles.push(import_path.default.resolve(testDir, seedFile));
132
- candidateFiles.push(import_path.default.resolve(configDir, seedFile));
133
- candidateFiles.push(import_path.default.resolve(this.rootPath, seedFile));
134
- let resolvedSeedFile;
135
- for (const candidateFile of candidateFiles) {
136
- if (await (0, import_util.fileExistsAsync)(candidateFile)) {
137
- resolvedSeedFile = candidateFile;
138
- break;
139
- }
140
- }
141
- if (!resolvedSeedFile)
142
- throw new Error("seed test not found.");
143
- seedFile = resolvedSeedFile;
144
- }
145
- const seedFileContent = await import_fs.default.promises.readFile(seedFile, "utf8");
146
- return {
147
- file: seedFile,
148
- content: seedFileContent,
149
- projectName: project.project.name
150
- };
151
- }
152
- async runSeedTest(seedFile, projectName) {
153
- const result = await this.runTestsWithGlobalSetupAndPossiblePause({
154
- headed: this.computedHeaded,
155
- locations: ["/" + (0, import_utils.escapeRegExp)(seedFile) + "/"],
156
- projects: [projectName],
157
- timeout: 0,
158
- workers: 1,
159
- pauseAtEnd: true,
160
- disableConfigReporters: true,
161
- failOnLoadErrors: true
162
- });
163
- if (result.status === "passed")
164
- result.output += "\nError: seed test not found.";
165
- else if (result.status !== "paused")
166
- result.output += "\nError while running the seed test.";
167
- return result;
168
- }
169
- async runTestsWithGlobalSetupAndPossiblePause(params) {
170
- const configDir = this._configLocation.configDir;
171
- const testRunnerAndScreen = await this.createTestRunner();
172
- const { testRunner, screen, claimStdio, releaseStdio } = testRunnerAndScreen;
173
- claimStdio();
174
- try {
175
- const setupReporter = new MCPListReporter({ configDir, screen, includeTestId: true });
176
- const { status: status2 } = await testRunner.runGlobalSetup([setupReporter]);
177
- if (status2 !== "passed")
178
- return { output: testRunnerAndScreen.output.join("\n"), status: status2 };
179
- } finally {
180
- releaseStdio();
181
- }
182
- let status = "passed";
183
- const cleanup = async () => {
184
- claimStdio();
185
- try {
186
- const result = await testRunner.runGlobalTeardown();
187
- if (status === "passed")
188
- status = result.status;
189
- } finally {
190
- releaseStdio();
191
- }
192
- };
193
- try {
194
- const reporter = new MCPListReporter({ configDir, screen, includeTestId: true });
195
- status = await Promise.race([
196
- testRunner.runTests(reporter, params).then((result) => result.status),
197
- testRunnerAndScreen.waitForTestPaused().then(() => "paused")
198
- ]);
199
- if (status === "paused") {
200
- const response = await testRunnerAndScreen.sendMessageToPausedTest({ request: { initialize: { clientInfo: this._clientInfo } } });
201
- if (response.error)
202
- throw new Error(response.error.message);
203
- testRunnerAndScreen.output.push(response.response.initialize.pausedMessage);
204
- return { output: testRunnerAndScreen.output.join("\n"), status };
205
- }
206
- } catch (e) {
207
- status = "failed";
208
- testRunnerAndScreen.output.push(String(e));
209
- await cleanup();
210
- return { output: testRunnerAndScreen.output.join("\n"), status };
211
- }
212
- await cleanup();
213
- return { output: testRunnerAndScreen.output.join("\n"), status };
214
- }
215
- async close() {
216
- await this._cleanupTestRunner().catch(import_log.logUnhandledError);
217
- }
218
- async sendMessageToPausedTest(request) {
219
- const sendMessage = this._testRunnerAndScreen?.sendMessageToPausedTest;
220
- if (!sendMessage)
221
- throw new Error("Must setup test before interacting with the page");
222
- const result = await sendMessage({ request });
223
- if (result.error)
224
- throw new Error(result.error.message);
225
- if (typeof request?.callTool?.arguments?.["intent"] === "string") {
226
- const response = (0, import_response.parseResponse)(result.response.callTool);
227
- if (response && !response.isError && response.code)
228
- this.generatorJournal?.logStep(request.callTool.arguments["intent"], response.code);
229
- }
230
- return result.response;
231
- }
232
- }
233
- function createScreen() {
234
- const output = [];
235
- const stdout = new import_streams.StringWriteStream(output, "stdout");
236
- const stderr = new import_streams.StringWriteStream(output, "stderr");
237
- const screen = {
238
- ...import_base.terminalScreen,
239
- isTTY: false,
240
- colors: import_utils.noColors,
241
- stdout,
242
- stderr
243
- };
244
- const originalStdoutWrite = process.stdout.write;
245
- const originalStderrWrite = process.stderr.write;
246
- const claimStdio = () => {
247
- process.stdout.write = (chunk) => {
248
- stdout.write(chunk);
249
- return true;
250
- };
251
- process.stderr.write = (chunk) => {
252
- stderr.write(chunk);
253
- return true;
254
- };
255
- };
256
- const releaseStdio = () => {
257
- process.stdout.write = originalStdoutWrite;
258
- process.stderr.write = originalStderrWrite;
259
- };
260
- return { screen, claimStdio, releaseStdio, output };
261
- }
262
- const bestPracticesMarkdown = `
263
- # Best practices
264
- - Do not improvise, do not add directives that were not asked for
265
- - Use clear, descriptive assertions to validate the expected behavior
266
- - Use reliable locators from this log
267
- - Use local variables for locators that are used multiple times
268
- - Use Playwright waiting assertions and best practices from this log
269
- - NEVER! use page.waitForLoadState()
270
- - NEVER! use page.waitForNavigation()
271
- - NEVER! use page.waitForTimeout()
272
- - NEVER! use page.evaluate()
273
- `;
274
- class MCPListReporter extends import_list.default {
275
- async onTestPaused() {
276
- await new Promise(() => {
277
- });
278
- }
279
- }
280
- // Annotate the CommonJS export names for ESM import in node:
281
- 0 && (module.exports = {
282
- GeneratorJournal,
283
- TestContext,
284
- createScreen
285
- });
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 testContext_exports = {};
30
+ __export(testContext_exports, {
31
+ GeneratorJournal: () => GeneratorJournal,
32
+ TestContext: () => TestContext,
33
+ createScreen: () => createScreen
34
+ });
35
+ module.exports = __toCommonJS(testContext_exports);
36
+ var import_fs = __toESM(require("fs"));
37
+ var import_os = __toESM(require("os"));
38
+ var import_path = __toESM(require("path"));
39
+ var import_utils = require("patchright-bun-core/lib/utils");
40
+ var import_exports = require("patchright-bun-core/lib/tools/exports");
41
+ var import_utilsBundle = require("patchright-bun-core/lib/utilsBundle");
42
+ var import_base = require("../../reporters/base");
43
+ var import_list = __toESM(require("../../reporters/list"));
44
+ var import_streams = require("./streams");
45
+ var import_util = require("../../util");
46
+ var import_testRunner = require("../../runner/testRunner");
47
+ var import_seed = require("./seed");
48
+ var import_configLoader = require("../../common/configLoader");
49
+ class GeneratorJournal {
50
+ constructor(rootPath, plan, seed) {
51
+ this._rootPath = rootPath;
52
+ this._plan = plan;
53
+ this._seed = seed;
54
+ this._steps = [];
55
+ }
56
+ logStep(title, code) {
57
+ if (title)
58
+ this._steps.push({ title, code });
59
+ }
60
+ journal() {
61
+ const result = [];
62
+ result.push(`# Plan`);
63
+ result.push(this._plan);
64
+ result.push(`# Seed file: ${(0, import_utils.toPosixPath)(import_path.default.relative(this._rootPath, this._seed.file))}`);
65
+ result.push("```ts");
66
+ result.push(this._seed.content);
67
+ result.push("```");
68
+ result.push(`# Steps`);
69
+ result.push(this._steps.map((step) => `### ${step.title}
70
+ \`\`\`ts
71
+ ${step.code}
72
+ \`\`\``).join("\n\n"));
73
+ result.push(bestPracticesMarkdown);
74
+ return result.join("\n\n");
75
+ }
76
+ }
77
+ class TestContext {
78
+ constructor(clientInfo, configPath, options) {
79
+ this._clientInfo = clientInfo;
80
+ this._configLocation = (0, import_configLoader.resolveConfigLocation)(configPath || clientInfo.cwd);
81
+ this.rootPath = clientInfo.cwd || this._configLocation.configDir;
82
+ if (options?.headless !== void 0)
83
+ this.computedHeaded = !options.headless;
84
+ else
85
+ this.computedHeaded = !process.env.CI && !(import_os.default.platform() === "linux" && !process.env.DISPLAY);
86
+ }
87
+ existingTestRunner() {
88
+ return this._testRunnerAndScreen?.testRunner;
89
+ }
90
+ async _cleanupTestRunner() {
91
+ if (!this._testRunnerAndScreen)
92
+ return;
93
+ await this._testRunnerAndScreen.testRunner.stopTests();
94
+ this._testRunnerAndScreen.claimStdio();
95
+ try {
96
+ await this._testRunnerAndScreen.testRunner.runGlobalTeardown();
97
+ } finally {
98
+ this._testRunnerAndScreen.releaseStdio();
99
+ this._testRunnerAndScreen = void 0;
100
+ }
101
+ }
102
+ async createTestRunner() {
103
+ await this._cleanupTestRunner();
104
+ const testRunner = new import_testRunner.TestRunner(this._configLocation, {});
105
+ await testRunner.initialize({});
106
+ const testPaused = new import_utils.ManualPromise();
107
+ const testRunnerAndScreen = {
108
+ ...createScreen(),
109
+ testRunner,
110
+ waitForTestPaused: () => testPaused
111
+ };
112
+ this._testRunnerAndScreen = testRunnerAndScreen;
113
+ testRunner.on(import_testRunner.TestRunnerEvent.TestPaused, (params) => {
114
+ testRunnerAndScreen.sendMessageToPausedTest = params.sendMessage;
115
+ testPaused.resolve();
116
+ });
117
+ return testRunnerAndScreen;
118
+ }
119
+ async getOrCreateSeedFile(seedFile, projectName) {
120
+ const configDir = this._configLocation.configDir;
121
+ const { testRunner } = await this.createTestRunner();
122
+ const config = await testRunner.loadConfig();
123
+ const project = (0, import_seed.seedProject)(config, projectName);
124
+ if (!seedFile) {
125
+ seedFile = await (0, import_seed.ensureSeedFile)(project);
126
+ } else {
127
+ const candidateFiles = [];
128
+ const testDir = project.project.testDir;
129
+ candidateFiles.push(import_path.default.resolve(testDir, seedFile));
130
+ candidateFiles.push(import_path.default.resolve(configDir, seedFile));
131
+ candidateFiles.push(import_path.default.resolve(this.rootPath, seedFile));
132
+ let resolvedSeedFile;
133
+ for (const candidateFile of candidateFiles) {
134
+ if (await (0, import_util.fileExistsAsync)(candidateFile)) {
135
+ resolvedSeedFile = candidateFile;
136
+ break;
137
+ }
138
+ }
139
+ if (!resolvedSeedFile)
140
+ throw new Error("seed test not found.");
141
+ seedFile = resolvedSeedFile;
142
+ }
143
+ const seedFileContent = await import_fs.default.promises.readFile(seedFile, "utf8");
144
+ return {
145
+ file: seedFile,
146
+ content: seedFileContent,
147
+ projectName: project.project.name
148
+ };
149
+ }
150
+ async runSeedTest(seedFile, projectName) {
151
+ const result = await this.runTestsWithGlobalSetupAndPossiblePause({
152
+ headed: this.computedHeaded,
153
+ locations: ["/" + (0, import_utils.escapeRegExp)(seedFile) + "/"],
154
+ projects: [projectName],
155
+ timeout: 0,
156
+ workers: 1,
157
+ pauseAtEnd: true,
158
+ disableConfigReporters: true,
159
+ failOnLoadErrors: true
160
+ });
161
+ if (result.status === "passed")
162
+ result.output += "\nError: seed test not found.";
163
+ else if (result.status !== "paused")
164
+ result.output += "\nError while running the seed test.";
165
+ return result;
166
+ }
167
+ async runTestsWithGlobalSetupAndPossiblePause(params) {
168
+ const configDir = this._configLocation.configDir;
169
+ const testRunnerAndScreen = await this.createTestRunner();
170
+ const { testRunner, screen, claimStdio, releaseStdio } = testRunnerAndScreen;
171
+ claimStdio();
172
+ try {
173
+ const setupReporter = new MCPListReporter({ configDir, screen, includeTestId: true });
174
+ const { status: status2 } = await testRunner.runGlobalSetup([setupReporter]);
175
+ if (status2 !== "passed")
176
+ return { output: testRunnerAndScreen.output.join("\n"), status: status2 };
177
+ } finally {
178
+ releaseStdio();
179
+ }
180
+ let status = "passed";
181
+ const cleanup = async () => {
182
+ claimStdio();
183
+ try {
184
+ const result = await testRunner.runGlobalTeardown();
185
+ if (status === "passed")
186
+ status = result.status;
187
+ } finally {
188
+ releaseStdio();
189
+ }
190
+ };
191
+ try {
192
+ const reporter = new MCPListReporter({ configDir, screen, includeTestId: true });
193
+ status = await Promise.race([
194
+ testRunner.runTests(reporter, params).then((result) => result.status),
195
+ testRunnerAndScreen.waitForTestPaused().then(() => "paused")
196
+ ]);
197
+ if (status === "paused") {
198
+ const response = await testRunnerAndScreen.sendMessageToPausedTest({ request: { initialize: { clientInfo: this._clientInfo } } });
199
+ if (response.error)
200
+ throw new Error(response.error.message);
201
+ testRunnerAndScreen.output.push(response.response.initialize.pausedMessage);
202
+ return { output: testRunnerAndScreen.output.join("\n"), status };
203
+ }
204
+ } catch (e) {
205
+ status = "failed";
206
+ testRunnerAndScreen.output.push(String(e));
207
+ await cleanup();
208
+ return { output: testRunnerAndScreen.output.join("\n"), status };
209
+ }
210
+ await cleanup();
211
+ return { output: testRunnerAndScreen.output.join("\n"), status };
212
+ }
213
+ async close() {
214
+ await this._cleanupTestRunner().catch((e) => (0, import_utilsBundle.debug)("pw:mcp:error")(e));
215
+ }
216
+ async sendMessageToPausedTest(request) {
217
+ const sendMessage = this._testRunnerAndScreen?.sendMessageToPausedTest;
218
+ if (!sendMessage)
219
+ throw new Error("Must setup test before interacting with the page");
220
+ const result = await sendMessage({ request });
221
+ if (result.error)
222
+ throw new Error(result.error.message);
223
+ if (typeof request?.callTool?.arguments?.["intent"] === "string") {
224
+ const response = (0, import_exports.parseResponse)(result.response.callTool);
225
+ if (response && !response.isError && response.code)
226
+ this.generatorJournal?.logStep(request.callTool.arguments["intent"], response.code);
227
+ }
228
+ return result.response;
229
+ }
230
+ }
231
+ function createScreen() {
232
+ const output = [];
233
+ const stdout = new import_streams.StringWriteStream(output, "stdout");
234
+ const stderr = new import_streams.StringWriteStream(output, "stderr");
235
+ const screen = {
236
+ ...import_base.terminalScreen,
237
+ isTTY: false,
238
+ colors: import_utils.noColors,
239
+ stdout,
240
+ stderr
241
+ };
242
+ const originalStdoutWrite = process.stdout.write;
243
+ const originalStderrWrite = process.stderr.write;
244
+ const claimStdio = () => {
245
+ process.stdout.write = (chunk) => {
246
+ stdout.write(chunk);
247
+ return true;
248
+ };
249
+ process.stderr.write = (chunk) => {
250
+ stderr.write(chunk);
251
+ return true;
252
+ };
253
+ };
254
+ const releaseStdio = () => {
255
+ process.stdout.write = originalStdoutWrite;
256
+ process.stderr.write = originalStderrWrite;
257
+ };
258
+ return { screen, claimStdio, releaseStdio, output };
259
+ }
260
+ const bestPracticesMarkdown = `
261
+ # Best practices
262
+ - Do not improvise, do not add directives that were not asked for
263
+ - Use clear, descriptive assertions to validate the expected behavior
264
+ - Use reliable locators from this log
265
+ - Use local variables for locators that are used multiple times
266
+ - Use Playwright waiting assertions and best practices from this log
267
+ - NEVER! use page.waitForLoadState()
268
+ - NEVER! use page.waitForNavigation()
269
+ - NEVER! use page.waitForTimeout()
270
+ - NEVER! use page.evaluate()
271
+ `;
272
+ class MCPListReporter extends import_list.default {
273
+ async onTestPaused() {
274
+ await new Promise(() => {
275
+ });
276
+ }
277
+ }
278
+ // Annotate the CommonJS export names for ESM import in node:
279
+ 0 && (module.exports = {
280
+ GeneratorJournal,
281
+ TestContext,
282
+ createScreen
283
+ });
@@ -1,30 +1,30 @@
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 testTool_exports = {};
20
- __export(testTool_exports, {
21
- defineTestTool: () => defineTestTool
22
- });
23
- module.exports = __toCommonJS(testTool_exports);
24
- function defineTestTool(tool) {
25
- return tool;
26
- }
27
- // Annotate the CommonJS export names for ESM import in node:
28
- 0 && (module.exports = {
29
- defineTestTool
30
- });
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 testTool_exports = {};
20
+ __export(testTool_exports, {
21
+ defineTestTool: () => defineTestTool
22
+ });
23
+ module.exports = __toCommonJS(testTool_exports);
24
+ function defineTestTool(tool) {
25
+ return tool;
26
+ }
27
+ // Annotate the CommonJS export names for ESM import in node:
28
+ 0 && (module.exports = {
29
+ defineTestTool
30
+ });