@pedropaulovc/playwright 1.59.0-next

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 (181) hide show
  1. package/ThirdPartyNotices.txt +5042 -0
  2. package/cli.js +19 -0
  3. package/index.d.ts +17 -0
  4. package/index.js +17 -0
  5. package/index.mjs +18 -0
  6. package/jsx-runtime.js +42 -0
  7. package/jsx-runtime.mjs +21 -0
  8. package/lib/agents/agentParser.js +89 -0
  9. package/lib/agents/copilot-setup-steps.yml +34 -0
  10. package/lib/agents/generateAgents.js +348 -0
  11. package/lib/agents/playwright-test-coverage.prompt.md +31 -0
  12. package/lib/agents/playwright-test-generate.prompt.md +8 -0
  13. package/lib/agents/playwright-test-generator.agent.md +88 -0
  14. package/lib/agents/playwright-test-heal.prompt.md +6 -0
  15. package/lib/agents/playwright-test-healer.agent.md +55 -0
  16. package/lib/agents/playwright-test-plan.prompt.md +9 -0
  17. package/lib/agents/playwright-test-planner.agent.md +73 -0
  18. package/lib/common/config.js +282 -0
  19. package/lib/common/configLoader.js +344 -0
  20. package/lib/common/esmLoaderHost.js +104 -0
  21. package/lib/common/expectBundle.js +28 -0
  22. package/lib/common/expectBundleImpl.js +407 -0
  23. package/lib/common/fixtures.js +302 -0
  24. package/lib/common/globals.js +58 -0
  25. package/lib/common/ipc.js +60 -0
  26. package/lib/common/poolBuilder.js +85 -0
  27. package/lib/common/process.js +132 -0
  28. package/lib/common/suiteUtils.js +140 -0
  29. package/lib/common/test.js +321 -0
  30. package/lib/common/testLoader.js +101 -0
  31. package/lib/common/testType.js +298 -0
  32. package/lib/common/validators.js +68 -0
  33. package/lib/fsWatcher.js +67 -0
  34. package/lib/index.js +726 -0
  35. package/lib/internalsForTest.js +42 -0
  36. package/lib/isomorphic/events.js +77 -0
  37. package/lib/isomorphic/folders.js +30 -0
  38. package/lib/isomorphic/stringInternPool.js +69 -0
  39. package/lib/isomorphic/teleReceiver.js +521 -0
  40. package/lib/isomorphic/teleSuiteUpdater.js +157 -0
  41. package/lib/isomorphic/testServerConnection.js +225 -0
  42. package/lib/isomorphic/testServerInterface.js +16 -0
  43. package/lib/isomorphic/testTree.js +329 -0
  44. package/lib/isomorphic/types.d.js +16 -0
  45. package/lib/loader/loaderMain.js +59 -0
  46. package/lib/matchers/expect.js +311 -0
  47. package/lib/matchers/matcherHint.js +44 -0
  48. package/lib/matchers/matchers.js +383 -0
  49. package/lib/matchers/toBeTruthy.js +75 -0
  50. package/lib/matchers/toEqual.js +100 -0
  51. package/lib/matchers/toHaveURL.js +101 -0
  52. package/lib/matchers/toMatchAriaSnapshot.js +159 -0
  53. package/lib/matchers/toMatchSnapshot.js +342 -0
  54. package/lib/matchers/toMatchText.js +99 -0
  55. package/lib/mcp/browser/actions.d.js +16 -0
  56. package/lib/mcp/browser/browserContextFactory.js +329 -0
  57. package/lib/mcp/browser/browserServerBackend.js +84 -0
  58. package/lib/mcp/browser/config.js +421 -0
  59. package/lib/mcp/browser/context.js +244 -0
  60. package/lib/mcp/browser/response.js +278 -0
  61. package/lib/mcp/browser/sessionLog.js +75 -0
  62. package/lib/mcp/browser/tab.js +351 -0
  63. package/lib/mcp/browser/tools/common.js +65 -0
  64. package/lib/mcp/browser/tools/console.js +61 -0
  65. package/lib/mcp/browser/tools/dialogs.js +60 -0
  66. package/lib/mcp/browser/tools/evaluate.js +61 -0
  67. package/lib/mcp/browser/tools/files.js +58 -0
  68. package/lib/mcp/browser/tools/form.js +63 -0
  69. package/lib/mcp/browser/tools/install.js +72 -0
  70. package/lib/mcp/browser/tools/keyboard.js +144 -0
  71. package/lib/mcp/browser/tools/mouse.js +159 -0
  72. package/lib/mcp/browser/tools/navigate.js +136 -0
  73. package/lib/mcp/browser/tools/navigateAndWait.js +66 -0
  74. package/lib/mcp/browser/tools/network.js +78 -0
  75. package/lib/mcp/browser/tools/pdf.js +49 -0
  76. package/lib/mcp/browser/tools/runCode.js +78 -0
  77. package/lib/mcp/browser/tools/screenshot.js +93 -0
  78. package/lib/mcp/browser/tools/snapshot.js +205 -0
  79. package/lib/mcp/browser/tools/snapshotViewport.js +63 -0
  80. package/lib/mcp/browser/tools/tabs.js +67 -0
  81. package/lib/mcp/browser/tools/tool.js +47 -0
  82. package/lib/mcp/browser/tools/tracing.js +74 -0
  83. package/lib/mcp/browser/tools/utils.js +94 -0
  84. package/lib/mcp/browser/tools/verify.js +143 -0
  85. package/lib/mcp/browser/tools/wait.js +63 -0
  86. package/lib/mcp/browser/tools.js +82 -0
  87. package/lib/mcp/browser/watchdog.js +44 -0
  88. package/lib/mcp/config.d.js +16 -0
  89. package/lib/mcp/extension/cdpRelay.js +351 -0
  90. package/lib/mcp/extension/extensionContextFactory.js +76 -0
  91. package/lib/mcp/extension/protocol.js +28 -0
  92. package/lib/mcp/index.js +61 -0
  93. package/lib/mcp/log.js +35 -0
  94. package/lib/mcp/program.js +110 -0
  95. package/lib/mcp/sdk/exports.js +28 -0
  96. package/lib/mcp/sdk/http.js +152 -0
  97. package/lib/mcp/sdk/inProcessTransport.js +71 -0
  98. package/lib/mcp/sdk/server.js +223 -0
  99. package/lib/mcp/sdk/tool.js +47 -0
  100. package/lib/mcp/terminal/cli.js +296 -0
  101. package/lib/mcp/terminal/command.js +56 -0
  102. package/lib/mcp/terminal/commands.js +519 -0
  103. package/lib/mcp/terminal/daemon.js +135 -0
  104. package/lib/mcp/terminal/help.json +47 -0
  105. package/lib/mcp/terminal/helpGenerator.js +115 -0
  106. package/lib/mcp/terminal/socketConnection.js +80 -0
  107. package/lib/mcp/test/browserBackend.js +98 -0
  108. package/lib/mcp/test/generatorTools.js +122 -0
  109. package/lib/mcp/test/plannerTools.js +145 -0
  110. package/lib/mcp/test/seed.js +82 -0
  111. package/lib/mcp/test/streams.js +44 -0
  112. package/lib/mcp/test/testBackend.js +99 -0
  113. package/lib/mcp/test/testContext.js +285 -0
  114. package/lib/mcp/test/testTool.js +30 -0
  115. package/lib/mcp/test/testTools.js +108 -0
  116. package/lib/plugins/gitCommitInfoPlugin.js +198 -0
  117. package/lib/plugins/index.js +28 -0
  118. package/lib/plugins/webServerPlugin.js +237 -0
  119. package/lib/program.js +417 -0
  120. package/lib/reporters/base.js +634 -0
  121. package/lib/reporters/blob.js +138 -0
  122. package/lib/reporters/dot.js +99 -0
  123. package/lib/reporters/empty.js +32 -0
  124. package/lib/reporters/github.js +128 -0
  125. package/lib/reporters/html.js +633 -0
  126. package/lib/reporters/internalReporter.js +138 -0
  127. package/lib/reporters/json.js +254 -0
  128. package/lib/reporters/junit.js +232 -0
  129. package/lib/reporters/line.js +131 -0
  130. package/lib/reporters/list.js +253 -0
  131. package/lib/reporters/listModeReporter.js +69 -0
  132. package/lib/reporters/markdown.js +144 -0
  133. package/lib/reporters/merge.js +558 -0
  134. package/lib/reporters/multiplexer.js +112 -0
  135. package/lib/reporters/reporterV2.js +102 -0
  136. package/lib/reporters/teleEmitter.js +317 -0
  137. package/lib/reporters/versions/blobV1.js +16 -0
  138. package/lib/runner/dispatcher.js +530 -0
  139. package/lib/runner/failureTracker.js +72 -0
  140. package/lib/runner/lastRun.js +77 -0
  141. package/lib/runner/loadUtils.js +334 -0
  142. package/lib/runner/loaderHost.js +89 -0
  143. package/lib/runner/processHost.js +180 -0
  144. package/lib/runner/projectUtils.js +241 -0
  145. package/lib/runner/rebase.js +189 -0
  146. package/lib/runner/reporters.js +138 -0
  147. package/lib/runner/sigIntWatcher.js +96 -0
  148. package/lib/runner/storage.js +91 -0
  149. package/lib/runner/taskRunner.js +127 -0
  150. package/lib/runner/tasks.js +410 -0
  151. package/lib/runner/testGroups.js +125 -0
  152. package/lib/runner/testRunner.js +398 -0
  153. package/lib/runner/testServer.js +269 -0
  154. package/lib/runner/uiModeReporter.js +30 -0
  155. package/lib/runner/vcs.js +72 -0
  156. package/lib/runner/watchMode.js +396 -0
  157. package/lib/runner/workerHost.js +104 -0
  158. package/lib/third_party/pirates.js +62 -0
  159. package/lib/third_party/tsconfig-loader.js +103 -0
  160. package/lib/transform/babelBundle.js +46 -0
  161. package/lib/transform/babelBundleImpl.js +461 -0
  162. package/lib/transform/compilationCache.js +274 -0
  163. package/lib/transform/esmLoader.js +103 -0
  164. package/lib/transform/md.js +221 -0
  165. package/lib/transform/portTransport.js +67 -0
  166. package/lib/transform/transform.js +303 -0
  167. package/lib/util.js +400 -0
  168. package/lib/utilsBundle.js +50 -0
  169. package/lib/utilsBundleImpl.js +103 -0
  170. package/lib/worker/fixtureRunner.js +262 -0
  171. package/lib/worker/testInfo.js +536 -0
  172. package/lib/worker/testTracing.js +345 -0
  173. package/lib/worker/timeoutManager.js +174 -0
  174. package/lib/worker/util.js +31 -0
  175. package/lib/worker/workerMain.js +530 -0
  176. package/package.json +72 -0
  177. package/test.d.ts +18 -0
  178. package/test.js +24 -0
  179. package/test.mjs +34 -0
  180. package/types/test.d.ts +10251 -0
  181. package/types/testReporter.d.ts +822 -0
@@ -0,0 +1,278 @@
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 response_exports = {};
30
+ __export(response_exports, {
31
+ Response: () => Response,
32
+ parseResponse: () => parseResponse,
33
+ renderTabMarkdown: () => renderTabMarkdown,
34
+ renderTabsMarkdown: () => renderTabsMarkdown,
35
+ requestDebug: () => requestDebug
36
+ });
37
+ module.exports = __toCommonJS(response_exports);
38
+ var import_fs = __toESM(require("fs"));
39
+ var import_path = __toESM(require("path"));
40
+ var import_utilsBundle = require("playwright-core/lib/utilsBundle");
41
+ var import_tab = require("./tab");
42
+ var import_utils = require("./tools/utils");
43
+ const requestDebug = (0, import_utilsBundle.debug)("pw:mcp:request");
44
+ class Response {
45
+ constructor(ordinal, context, toolName, toolArgs) {
46
+ this._results = [];
47
+ this._errors = [];
48
+ this._code = [];
49
+ this._images = [];
50
+ this._includeSnapshot = "none";
51
+ this._ordinal = ordinal;
52
+ this._context = context;
53
+ this.toolName = toolName;
54
+ this.toolArgs = toolArgs;
55
+ }
56
+ static {
57
+ this._ordinal = 0;
58
+ }
59
+ static create(context, toolName, toolArgs) {
60
+ return new Response(++Response._ordinal, context, toolName, toolArgs);
61
+ }
62
+ addTextResult(result) {
63
+ this._results.push({ text: result });
64
+ }
65
+ async addResult(result) {
66
+ if (result.data && !result.suggestedFilename)
67
+ result.suggestedFilename = (0, import_utils.dateAsFileName)(result.ext ?? "bin");
68
+ if (this._context.config.outputMode === "file") {
69
+ if (!result.suggestedFilename)
70
+ result.suggestedFilename = (0, import_utils.dateAsFileName)(result.ext ?? (result.text ? "txt" : "bin"));
71
+ }
72
+ const entry = { text: result.text, data: result.data, title: result.title };
73
+ if (result.suggestedFilename)
74
+ entry.filename = await this._context.outputFile(result.suggestedFilename, { origin: "llm", title: result.title ?? "Saved result" });
75
+ this._results.push(entry);
76
+ return { fileName: entry.filename };
77
+ }
78
+ addError(error) {
79
+ this._errors.push(error);
80
+ }
81
+ addCode(code) {
82
+ this._code.push(code);
83
+ }
84
+ addImage(image) {
85
+ this._images.push(image);
86
+ }
87
+ setIncludeSnapshot() {
88
+ this._includeSnapshot = this._context.config.snapshot.mode;
89
+ }
90
+ setIncludeFullSnapshot(includeSnapshotFileName) {
91
+ this._includeSnapshot = "full";
92
+ this._includeSnapshotFileName = includeSnapshotFileName;
93
+ }
94
+ async build() {
95
+ const rootPath = this._context.firstRootPath();
96
+ const sections = [];
97
+ const addSection = (title) => {
98
+ const section = { title, content: [] };
99
+ sections.push(section);
100
+ return section.content;
101
+ };
102
+ if (this._errors.length) {
103
+ const text = addSection("Error");
104
+ text.push("### Error");
105
+ text.push(this._errors.join("\n"));
106
+ }
107
+ if (this._results.length) {
108
+ const text = addSection("Result");
109
+ for (const result of this._results) {
110
+ if (result.filename) {
111
+ text.push(`- [${result.title}](${rootPath ? import_path.default.relative(rootPath, result.filename) : result.filename})`);
112
+ if (result.data)
113
+ await import_fs.default.promises.writeFile(result.filename, result.data);
114
+ else if (result.text)
115
+ await import_fs.default.promises.writeFile(result.filename, this._redactText(result.text));
116
+ } else if (result.text) {
117
+ text.push(result.text);
118
+ }
119
+ }
120
+ }
121
+ if (this._context.config.codegen !== "none" && this._code.length) {
122
+ const text = addSection("Ran Playwright code");
123
+ text.push(...this._code);
124
+ }
125
+ const tabSnapshot = this._context.currentTab() ? await this._context.currentTabOrDie().captureSnapshot() : void 0;
126
+ const tabHeaders = await Promise.all(this._context.tabs().map((tab) => tab.headerSnapshot()));
127
+ if (tabHeaders.some((header) => header.changed)) {
128
+ if (tabHeaders.length !== 1) {
129
+ const text2 = addSection("Open tabs");
130
+ text2.push(...renderTabsMarkdown(tabHeaders));
131
+ }
132
+ const text = addSection("Page");
133
+ text.push(...renderTabMarkdown(tabHeaders[0]));
134
+ }
135
+ if (tabSnapshot?.modalStates.length) {
136
+ const text = addSection("Modal state");
137
+ text.push(...(0, import_tab.renderModalStates)(tabSnapshot.modalStates));
138
+ }
139
+ if (tabSnapshot && this._includeSnapshot === "full") {
140
+ let fileName;
141
+ if (this._includeSnapshotFileName)
142
+ fileName = await this._context.outputFile(this._includeSnapshotFileName, { origin: "llm", title: "Saved snapshot" });
143
+ else if (this._context.config.outputMode === "file")
144
+ fileName = await this._context.outputFile(`snapshot-${this._ordinal}.yml`, { origin: "code", title: "Saved snapshot" });
145
+ if (fileName) {
146
+ await import_fs.default.promises.writeFile(fileName, tabSnapshot.ariaSnapshot);
147
+ const text = addSection("Snapshot");
148
+ text.push(`- File: ${rootPath ? import_path.default.relative(rootPath, fileName) : fileName}`);
149
+ } else {
150
+ const text = addSection("Snapshot");
151
+ text.push("```yaml");
152
+ text.push(tabSnapshot.ariaSnapshot);
153
+ text.push("```");
154
+ }
155
+ }
156
+ if (tabSnapshot && this._includeSnapshot === "incremental") {
157
+ const text = addSection("Snapshot");
158
+ text.push("```yaml");
159
+ if (tabSnapshot.ariaSnapshotDiff !== void 0)
160
+ text.push(tabSnapshot.ariaSnapshotDiff);
161
+ else
162
+ text.push(tabSnapshot.ariaSnapshot);
163
+ text.push("```");
164
+ }
165
+ if (tabSnapshot?.events.filter((event) => event.type !== "request").length) {
166
+ const text = addSection("Events");
167
+ for (const event of tabSnapshot.events) {
168
+ if (event.type === "console") {
169
+ if ((0, import_tab.shouldIncludeMessage)(this._context.config.console.level, event.message.type))
170
+ text.push(`- ${trimMiddle(event.message.toString(), 100)}`);
171
+ } else if (event.type === "download-start") {
172
+ text.push(`- Downloading file ${event.download.download.suggestedFilename()} ...`);
173
+ } else if (event.type === "download-finish") {
174
+ text.push(`- Downloaded file ${event.download.download.suggestedFilename()} to "${rootPath ? import_path.default.relative(rootPath, event.download.outputFile) : event.download.outputFile}"`);
175
+ }
176
+ }
177
+ }
178
+ const allText = sections.flatMap((section) => {
179
+ const content2 = [];
180
+ content2.push(`### ${section.title}`);
181
+ content2.push(...section.content);
182
+ content2.push("");
183
+ return content2;
184
+ }).join("\n");
185
+ const content = [
186
+ {
187
+ type: "text",
188
+ text: this._redactText(allText)
189
+ }
190
+ ];
191
+ if (this._context.config.imageResponses !== "omit") {
192
+ for (const image of this._images)
193
+ content.push({ type: "image", data: image.data.toString("base64"), mimeType: image.contentType });
194
+ }
195
+ return {
196
+ content,
197
+ ...this._errors.length > 0 ? { isError: true } : {}
198
+ };
199
+ }
200
+ _redactText(text) {
201
+ for (const [secretName, secretValue] of Object.entries(this._context.config.secrets ?? {}))
202
+ text = text.replaceAll(secretValue, `<secret>${secretName}</secret>`);
203
+ return text;
204
+ }
205
+ }
206
+ function renderTabMarkdown(tab) {
207
+ const lines = [`- Page URL: ${tab.url}`];
208
+ if (tab.title)
209
+ lines.push(`- Page Title: ${tab.title}`);
210
+ return lines;
211
+ }
212
+ function renderTabsMarkdown(tabs) {
213
+ if (!tabs.length)
214
+ return ['No open tabs. Use the "browser_navigate" tool to navigate to a page first.'];
215
+ const lines = [];
216
+ for (let i = 0; i < tabs.length; i++) {
217
+ const tab = tabs[i];
218
+ const current = tab.current ? " (current)" : "";
219
+ lines.push(`- ${i}:${current} [${tab.title}](${tab.url})`);
220
+ }
221
+ return lines;
222
+ }
223
+ function trimMiddle(text, maxLength) {
224
+ if (text.length <= maxLength)
225
+ return text;
226
+ return text.slice(0, Math.floor(maxLength / 2)) + "..." + text.slice(-3 - Math.floor(maxLength / 2));
227
+ }
228
+ function parseSections(text) {
229
+ const sections = /* @__PURE__ */ new Map();
230
+ const sectionHeaders = text.split(/^### /m).slice(1);
231
+ for (const section of sectionHeaders) {
232
+ const firstNewlineIndex = section.indexOf("\n");
233
+ if (firstNewlineIndex === -1)
234
+ continue;
235
+ const sectionName = section.substring(0, firstNewlineIndex);
236
+ const sectionContent = section.substring(firstNewlineIndex + 1).trim();
237
+ sections.set(sectionName, sectionContent);
238
+ }
239
+ return sections;
240
+ }
241
+ function parseResponse(response) {
242
+ if (response.content?.[0].type !== "text")
243
+ return void 0;
244
+ const text = response.content[0].text;
245
+ const sections = parseSections(text);
246
+ const error = sections.get("Error");
247
+ const result = sections.get("Result");
248
+ const code = sections.get("Ran Playwright code");
249
+ const tabs = sections.get("Open tabs");
250
+ const page = sections.get("Page");
251
+ const snapshot = sections.get("Snapshot");
252
+ const events = sections.get("Events");
253
+ const modalState = sections.get("Modal state");
254
+ const codeNoFrame = code?.replace(/^```js\n/, "").replace(/\n```$/, "");
255
+ const isError = response.isError;
256
+ const attachments = response.content.length > 1 ? response.content.slice(1) : void 0;
257
+ return {
258
+ result,
259
+ error,
260
+ code: codeNoFrame,
261
+ tabs,
262
+ page,
263
+ snapshot,
264
+ events,
265
+ modalState,
266
+ isError,
267
+ attachments,
268
+ text
269
+ };
270
+ }
271
+ // Annotate the CommonJS export names for ESM import in node:
272
+ 0 && (module.exports = {
273
+ Response,
274
+ parseResponse,
275
+ renderTabMarkdown,
276
+ renderTabsMarkdown,
277
+ requestDebug
278
+ });
@@ -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 sessionLog_exports = {};
30
+ __export(sessionLog_exports, {
31
+ SessionLog: () => SessionLog
32
+ });
33
+ module.exports = __toCommonJS(sessionLog_exports);
34
+ var import_fs = __toESM(require("fs"));
35
+ var import_path = __toESM(require("path"));
36
+ var import_config = require("./config");
37
+ var import_response = require("./response");
38
+ class SessionLog {
39
+ constructor(sessionFolder) {
40
+ this._sessionFileQueue = Promise.resolve();
41
+ this._folder = sessionFolder;
42
+ this._file = import_path.default.join(this._folder, "session.md");
43
+ }
44
+ static async create(config, clientInfo) {
45
+ const sessionFolder = await (0, import_config.outputFile)(config, clientInfo, `session-${Date.now()}`, { origin: "code", title: "Saving session" });
46
+ await import_fs.default.promises.mkdir(sessionFolder, { recursive: true });
47
+ console.error(`Session: ${sessionFolder}`);
48
+ return new SessionLog(sessionFolder);
49
+ }
50
+ logResponse(toolName, toolArgs, responseObject) {
51
+ const parsed = (0, import_response.parseResponse)(responseObject);
52
+ if (parsed)
53
+ delete parsed.text;
54
+ const lines = [""];
55
+ lines.push(
56
+ `### Tool call: ${toolName}`,
57
+ `- Args`,
58
+ "```json",
59
+ JSON.stringify(toolArgs, null, 2),
60
+ "```"
61
+ );
62
+ if (parsed) {
63
+ lines.push(`- Result`);
64
+ lines.push("```json");
65
+ lines.push(JSON.stringify(parsed, null, 2));
66
+ lines.push("```");
67
+ }
68
+ lines.push("");
69
+ this._sessionFileQueue = this._sessionFileQueue.then(() => import_fs.default.promises.appendFile(this._file, lines.join("\n")));
70
+ }
71
+ }
72
+ // Annotate the CommonJS export names for ESM import in node:
73
+ 0 && (module.exports = {
74
+ SessionLog
75
+ });
@@ -0,0 +1,351 @@
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 tab_exports = {};
20
+ __export(tab_exports, {
21
+ Tab: () => Tab,
22
+ TabEvents: () => TabEvents,
23
+ renderModalStates: () => renderModalStates,
24
+ shouldIncludeMessage: () => shouldIncludeMessage
25
+ });
26
+ module.exports = __toCommonJS(tab_exports);
27
+ var import_events = require("events");
28
+ var import_utils = require("playwright-core/lib/utils");
29
+ var import_utils2 = require("./tools/utils");
30
+ var import_log = require("../log");
31
+ var import_dialogs = require("./tools/dialogs");
32
+ var import_files = require("./tools/files");
33
+ var import_transform = require("../../transform/transform");
34
+ const TabEvents = {
35
+ modalState: "modalState"
36
+ };
37
+ class Tab extends import_events.EventEmitter {
38
+ constructor(context, page, onPageClose) {
39
+ super();
40
+ this._lastHeader = { title: "about:blank", url: "about:blank", current: false };
41
+ this._consoleMessages = [];
42
+ this._downloads = [];
43
+ this._requests = /* @__PURE__ */ new Set();
44
+ this._modalStates = [];
45
+ this._needsFullSnapshot = false;
46
+ this._eventEntries = [];
47
+ this._recentEventEntries = [];
48
+ this.context = context;
49
+ this.page = page;
50
+ this._onPageClose = onPageClose;
51
+ page.on("console", (event) => this._handleConsoleMessage(messageToConsoleMessage(event)));
52
+ page.on("pageerror", (error) => this._handleConsoleMessage(pageErrorToConsoleMessage(error)));
53
+ page.on("request", (request) => this._handleRequest(request));
54
+ page.on("close", () => this._onClose());
55
+ page.on("filechooser", (chooser) => {
56
+ this.setModalState({
57
+ type: "fileChooser",
58
+ description: "File chooser",
59
+ fileChooser: chooser,
60
+ clearedBy: import_files.uploadFile.schema.name
61
+ });
62
+ });
63
+ page.on("dialog", (dialog) => this._dialogShown(dialog));
64
+ page.on("download", (download) => {
65
+ void this._downloadStarted(download);
66
+ });
67
+ page.setDefaultNavigationTimeout(this.context.config.timeouts.navigation);
68
+ page.setDefaultTimeout(this.context.config.timeouts.action);
69
+ page[tabSymbol] = this;
70
+ this._initializedPromise = this._initialize();
71
+ }
72
+ static forPage(page) {
73
+ return page[tabSymbol];
74
+ }
75
+ static async collectConsoleMessages(page) {
76
+ const result = [];
77
+ const messages = await page.consoleMessages().catch(() => []);
78
+ for (const message of messages)
79
+ result.push(messageToConsoleMessage(message));
80
+ const errors = await page.pageErrors().catch(() => []);
81
+ for (const error of errors)
82
+ result.push(pageErrorToConsoleMessage(error));
83
+ return result;
84
+ }
85
+ async _initialize() {
86
+ for (const message of await Tab.collectConsoleMessages(this.page))
87
+ this._handleConsoleMessage(message);
88
+ const requests = await this.page.requests().catch(() => []);
89
+ for (const request of requests)
90
+ this._requests.add(request);
91
+ for (const initPage of this.context.config.browser.initPage || []) {
92
+ try {
93
+ const { default: func } = await (0, import_transform.requireOrImport)(initPage);
94
+ await func({ page: this.page });
95
+ } catch (e) {
96
+ (0, import_log.logUnhandledError)(e);
97
+ }
98
+ }
99
+ }
100
+ modalStates() {
101
+ return this._modalStates;
102
+ }
103
+ setModalState(modalState) {
104
+ this._modalStates.push(modalState);
105
+ this.emit(TabEvents.modalState, modalState);
106
+ }
107
+ clearModalState(modalState) {
108
+ this._modalStates = this._modalStates.filter((state) => state !== modalState);
109
+ }
110
+ _dialogShown(dialog) {
111
+ this.setModalState({
112
+ type: "dialog",
113
+ description: `"${dialog.type()}" dialog with message "${dialog.message()}"`,
114
+ dialog,
115
+ clearedBy: import_dialogs.handleDialog.schema.name
116
+ });
117
+ }
118
+ async _downloadStarted(download) {
119
+ const entry = {
120
+ download,
121
+ finished: false,
122
+ outputFile: await this.context.outputFile(download.suggestedFilename(), { origin: "web", title: "Saving download" })
123
+ };
124
+ this._downloads.push(entry);
125
+ this._addLogEntry({ type: "download-start", wallTime: Date.now(), download: entry });
126
+ await download.saveAs(entry.outputFile);
127
+ entry.finished = true;
128
+ this._addLogEntry({ type: "download-finish", wallTime: Date.now(), download: entry });
129
+ }
130
+ _clearCollectedArtifacts() {
131
+ this._consoleMessages.length = 0;
132
+ this._downloads.length = 0;
133
+ this._requests.clear();
134
+ this._eventEntries.length = 0;
135
+ this._recentEventEntries.length = 0;
136
+ }
137
+ _handleRequest(request) {
138
+ this._requests.add(request);
139
+ this._addLogEntry({ type: "request", wallTime: Date.now(), request });
140
+ }
141
+ _handleConsoleMessage(message) {
142
+ this._consoleMessages.push(message);
143
+ this._addLogEntry({ type: "console", wallTime: Date.now(), message });
144
+ }
145
+ _addLogEntry(entry) {
146
+ this._eventEntries.push(entry);
147
+ this._recentEventEntries.push(entry);
148
+ }
149
+ _onClose() {
150
+ this._clearCollectedArtifacts();
151
+ this._onPageClose(this);
152
+ }
153
+ async headerSnapshot() {
154
+ let title;
155
+ await this._raceAgainstModalStates(async () => {
156
+ title = await (0, import_utils2.callOnPageNoTrace)(this.page, (page) => page.title());
157
+ });
158
+ if (this._lastHeader.title !== title || this._lastHeader.url !== this.page.url() || this._lastHeader.current !== this.isCurrentTab()) {
159
+ this._lastHeader = { title: title ?? "", url: this.page.url(), current: this.isCurrentTab() };
160
+ return { ...this._lastHeader, changed: true };
161
+ }
162
+ return { ...this._lastHeader, changed: false };
163
+ }
164
+ isCurrentTab() {
165
+ return this === this.context.currentTab();
166
+ }
167
+ async waitForLoadState(state, options) {
168
+ await this._initializedPromise;
169
+ await (0, import_utils2.callOnPageNoTrace)(this.page, (page) => page.waitForLoadState(state, options).catch(import_log.logUnhandledError));
170
+ }
171
+ async navigate(url) {
172
+ await this._initializedPromise;
173
+ this._clearCollectedArtifacts();
174
+ const { promise: downloadEvent, abort: abortDownloadEvent } = (0, import_utils2.eventWaiter)(this.page, "download", 3e3);
175
+ try {
176
+ await this.page.goto(url, { waitUntil: "domcontentloaded" });
177
+ abortDownloadEvent();
178
+ } catch (_e) {
179
+ const e = _e;
180
+ const mightBeDownload = e.message.includes("net::ERR_ABORTED") || e.message.includes("Download is starting");
181
+ if (!mightBeDownload)
182
+ throw e;
183
+ const download = await downloadEvent;
184
+ if (!download)
185
+ throw e;
186
+ await new Promise((resolve) => setTimeout(resolve, 500));
187
+ return;
188
+ }
189
+ await this.waitForLoadState("load", { timeout: 5e3 });
190
+ }
191
+ async consoleMessages(level) {
192
+ await this._initializedPromise;
193
+ return this._consoleMessages.filter((message) => shouldIncludeMessage(level, message.type));
194
+ }
195
+ async clearConsoleMessages() {
196
+ await this._initializedPromise;
197
+ this._consoleMessages.length = 0;
198
+ }
199
+ async requests() {
200
+ await this._initializedPromise;
201
+ return this._requests;
202
+ }
203
+ async clearRequests() {
204
+ await this._initializedPromise;
205
+ this._requests.clear();
206
+ }
207
+ async captureSnapshot() {
208
+ await this._initializedPromise;
209
+ let tabSnapshot;
210
+ const modalStates = await this._raceAgainstModalStates(async () => {
211
+ const snapshot = await this.page._snapshotForAI({ track: "response" });
212
+ tabSnapshot = {
213
+ ariaSnapshot: snapshot.full,
214
+ ariaSnapshotDiff: this._needsFullSnapshot ? void 0 : snapshot.incremental,
215
+ modalStates: [],
216
+ events: []
217
+ };
218
+ });
219
+ if (tabSnapshot) {
220
+ tabSnapshot.events = this._recentEventEntries;
221
+ this._recentEventEntries = [];
222
+ }
223
+ this._needsFullSnapshot = !tabSnapshot;
224
+ return tabSnapshot ?? {
225
+ ariaSnapshot: "",
226
+ ariaSnapshotDiff: "",
227
+ modalStates,
228
+ events: []
229
+ };
230
+ }
231
+ _javaScriptBlocked() {
232
+ return this._modalStates.some((state) => state.type === "dialog");
233
+ }
234
+ async _raceAgainstModalStates(action) {
235
+ if (this.modalStates().length)
236
+ return this.modalStates();
237
+ const promise = new import_utils.ManualPromise();
238
+ const listener = (modalState) => promise.resolve([modalState]);
239
+ this.once(TabEvents.modalState, listener);
240
+ return await Promise.race([
241
+ action().then(() => {
242
+ this.off(TabEvents.modalState, listener);
243
+ return [];
244
+ }),
245
+ promise
246
+ ]);
247
+ }
248
+ async waitForCompletion(callback) {
249
+ await this._initializedPromise;
250
+ await this._raceAgainstModalStates(() => (0, import_utils2.waitForCompletion)(this, callback));
251
+ }
252
+ async refLocator(params) {
253
+ await this._initializedPromise;
254
+ return (await this.refLocators([params]))[0];
255
+ }
256
+ async refLocators(params) {
257
+ await this._initializedPromise;
258
+ return Promise.all(params.map(async (param) => {
259
+ try {
260
+ let locator = this.page.locator(`aria-ref=${param.ref}`);
261
+ if (param.element)
262
+ locator = locator.describe(param.element);
263
+ const { resolvedSelector } = await locator._resolveSelector();
264
+ return { locator, resolved: (0, import_utils.asLocator)("javascript", resolvedSelector) };
265
+ } catch (e) {
266
+ throw new Error(`Ref ${param.ref} not found in the current page snapshot. Try capturing new snapshot.`);
267
+ }
268
+ }));
269
+ }
270
+ async waitForTimeout(time) {
271
+ if (this._javaScriptBlocked()) {
272
+ await new Promise((f) => setTimeout(f, time));
273
+ return;
274
+ }
275
+ await (0, import_utils2.callOnPageNoTrace)(this.page, (page) => {
276
+ return page.evaluate(() => new Promise((f) => setTimeout(f, 1e3))).catch(() => {
277
+ });
278
+ });
279
+ }
280
+ }
281
+ function messageToConsoleMessage(message) {
282
+ return {
283
+ type: message.type(),
284
+ text: message.text(),
285
+ toString: () => `[${message.type().toUpperCase()}] ${message.text()} @ ${message.location().url}:${message.location().lineNumber}`
286
+ };
287
+ }
288
+ function pageErrorToConsoleMessage(errorOrValue) {
289
+ if (errorOrValue instanceof Error) {
290
+ return {
291
+ type: "error",
292
+ text: errorOrValue.message,
293
+ toString: () => errorOrValue.stack || errorOrValue.message
294
+ };
295
+ }
296
+ return {
297
+ type: "error",
298
+ text: String(errorOrValue),
299
+ toString: () => String(errorOrValue)
300
+ };
301
+ }
302
+ function renderModalStates(modalStates) {
303
+ const result = [];
304
+ if (modalStates.length === 0)
305
+ result.push("- There is no modal state present");
306
+ for (const state of modalStates)
307
+ result.push(`- [${state.description}]: can be handled by the "${state.clearedBy}" tool`);
308
+ return result;
309
+ }
310
+ const consoleMessageLevels = ["error", "warning", "info", "debug"];
311
+ function shouldIncludeMessage(thresholdLevel, type) {
312
+ const messageLevel = consoleLevelForMessageType(type);
313
+ return consoleMessageLevels.indexOf(messageLevel) <= consoleMessageLevels.indexOf(thresholdLevel);
314
+ }
315
+ function consoleLevelForMessageType(type) {
316
+ switch (type) {
317
+ case "assert":
318
+ case "error":
319
+ return "error";
320
+ case "warning":
321
+ return "warning";
322
+ case "count":
323
+ case "dir":
324
+ case "dirxml":
325
+ case "info":
326
+ case "log":
327
+ case "table":
328
+ case "time":
329
+ case "timeEnd":
330
+ return "info";
331
+ case "clear":
332
+ case "debug":
333
+ case "endGroup":
334
+ case "profile":
335
+ case "profileEnd":
336
+ case "startGroup":
337
+ case "startGroupCollapsed":
338
+ case "trace":
339
+ return "debug";
340
+ default:
341
+ return "info";
342
+ }
343
+ }
344
+ const tabSymbol = Symbol("tabSymbol");
345
+ // Annotate the CommonJS export names for ESM import in node:
346
+ 0 && (module.exports = {
347
+ Tab,
348
+ TabEvents,
349
+ renderModalStates,
350
+ shouldIncludeMessage
351
+ });