@hera-al/browser-server 1.0.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 (69) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +212 -0
  3. package/dist/config.d.ts +44 -0
  4. package/dist/config.js +74 -0
  5. package/dist/core/cdp.d.ts +124 -0
  6. package/dist/core/cdp.helpers.d.ts +14 -0
  7. package/dist/core/cdp.helpers.js +148 -0
  8. package/dist/core/cdp.js +309 -0
  9. package/dist/core/chrome.d.ts +21 -0
  10. package/dist/core/chrome.executables.d.ts +10 -0
  11. package/dist/core/chrome.executables.js +559 -0
  12. package/dist/core/chrome.js +257 -0
  13. package/dist/core/chrome.profile-decoration.d.ts +11 -0
  14. package/dist/core/chrome.profile-decoration.js +148 -0
  15. package/dist/core/constants.d.ts +9 -0
  16. package/dist/core/constants.js +9 -0
  17. package/dist/core/profiles.d.ts +31 -0
  18. package/dist/core/profiles.js +99 -0
  19. package/dist/core/target-id.d.ts +12 -0
  20. package/dist/core/target-id.js +21 -0
  21. package/dist/data-dir.d.ts +2 -0
  22. package/dist/data-dir.js +6 -0
  23. package/dist/logger.d.ts +16 -0
  24. package/dist/logger.js +125 -0
  25. package/dist/playwright/pw-role-snapshot.d.ts +32 -0
  26. package/dist/playwright/pw-role-snapshot.js +337 -0
  27. package/dist/playwright/pw-session.d.ts +119 -0
  28. package/dist/playwright/pw-session.js +530 -0
  29. package/dist/playwright/pw-tools-core.activity.d.ts +22 -0
  30. package/dist/playwright/pw-tools-core.activity.js +47 -0
  31. package/dist/playwright/pw-tools-core.d.ts +9 -0
  32. package/dist/playwright/pw-tools-core.downloads.d.ts +35 -0
  33. package/dist/playwright/pw-tools-core.downloads.js +186 -0
  34. package/dist/playwright/pw-tools-core.interactions.d.ts +104 -0
  35. package/dist/playwright/pw-tools-core.interactions.js +404 -0
  36. package/dist/playwright/pw-tools-core.js +9 -0
  37. package/dist/playwright/pw-tools-core.responses.d.ts +14 -0
  38. package/dist/playwright/pw-tools-core.responses.js +91 -0
  39. package/dist/playwright/pw-tools-core.shared.d.ts +7 -0
  40. package/dist/playwright/pw-tools-core.shared.js +50 -0
  41. package/dist/playwright/pw-tools-core.snapshot.d.ts +65 -0
  42. package/dist/playwright/pw-tools-core.snapshot.js +144 -0
  43. package/dist/playwright/pw-tools-core.state.d.ts +47 -0
  44. package/dist/playwright/pw-tools-core.state.js +154 -0
  45. package/dist/playwright/pw-tools-core.storage.d.ts +48 -0
  46. package/dist/playwright/pw-tools-core.storage.js +76 -0
  47. package/dist/playwright/pw-tools-core.trace.d.ts +13 -0
  48. package/dist/playwright/pw-tools-core.trace.js +26 -0
  49. package/dist/server/browser-context.d.ts +29 -0
  50. package/dist/server/browser-context.js +137 -0
  51. package/dist/server/browser-server.d.ts +7 -0
  52. package/dist/server/browser-server.js +49 -0
  53. package/dist/server/routes/act.d.ts +4 -0
  54. package/dist/server/routes/act.js +176 -0
  55. package/dist/server/routes/basic.d.ts +4 -0
  56. package/dist/server/routes/basic.js +36 -0
  57. package/dist/server/routes/index.d.ts +4 -0
  58. package/dist/server/routes/index.js +16 -0
  59. package/dist/server/routes/snapshot.d.ts +4 -0
  60. package/dist/server/routes/snapshot.js +143 -0
  61. package/dist/server/routes/storage.d.ts +4 -0
  62. package/dist/server/routes/storage.js +117 -0
  63. package/dist/server/routes/tabs.d.ts +4 -0
  64. package/dist/server/routes/tabs.js +51 -0
  65. package/dist/server/standalone.d.ts +9 -0
  66. package/dist/server/standalone.js +42 -0
  67. package/dist/utils.d.ts +18 -0
  68. package/dist/utils.js +58 -0
  69. package/package.json +66 -0
@@ -0,0 +1,530 @@
1
+ import { chromium } from "playwright-core";
2
+ import { formatErrorMessage } from "../utils.js";
3
+ import { appendCdpPath, fetchJson, getHeadersWithAuth, withCdpSocket } from "../core/cdp.helpers.js";
4
+ import { normalizeCdpWsUrl } from "../core/cdp.js";
5
+ import { getChromeWebSocketUrl } from "../core/chrome.js";
6
+ const pageStates = new WeakMap();
7
+ const contextStates = new WeakMap();
8
+ const observedContexts = new WeakSet();
9
+ const observedPages = new WeakSet();
10
+ const roleRefsByTarget = new Map();
11
+ const MAX_ROLE_REFS_CACHE = 50;
12
+ const MAX_CONSOLE_MESSAGES = 500;
13
+ const MAX_PAGE_ERRORS = 200;
14
+ const MAX_NETWORK_REQUESTS = 500;
15
+ let cached = null;
16
+ let connecting = null;
17
+ function normalizeCdpUrl(raw) {
18
+ return raw.replace(/\/$/, "");
19
+ }
20
+ function roleRefsKey(cdpUrl, targetId) {
21
+ return `${normalizeCdpUrl(cdpUrl)}::${targetId}`;
22
+ }
23
+ export function rememberRoleRefsForTarget(opts) {
24
+ const targetId = opts.targetId.trim();
25
+ if (!targetId) {
26
+ return;
27
+ }
28
+ roleRefsByTarget.set(roleRefsKey(opts.cdpUrl, targetId), {
29
+ refs: opts.refs,
30
+ ...(opts.frameSelector ? { frameSelector: opts.frameSelector } : {}),
31
+ ...(opts.mode ? { mode: opts.mode } : {}),
32
+ });
33
+ while (roleRefsByTarget.size > MAX_ROLE_REFS_CACHE) {
34
+ const first = roleRefsByTarget.keys().next();
35
+ if (first.done) {
36
+ break;
37
+ }
38
+ roleRefsByTarget.delete(first.value);
39
+ }
40
+ }
41
+ export function storeRoleRefsForTarget(opts) {
42
+ const state = ensurePageState(opts.page);
43
+ state.roleRefs = opts.refs;
44
+ state.roleRefsFrameSelector = opts.frameSelector;
45
+ state.roleRefsMode = opts.mode;
46
+ if (!opts.targetId?.trim()) {
47
+ return;
48
+ }
49
+ rememberRoleRefsForTarget({
50
+ cdpUrl: opts.cdpUrl,
51
+ targetId: opts.targetId,
52
+ refs: opts.refs,
53
+ frameSelector: opts.frameSelector,
54
+ mode: opts.mode,
55
+ });
56
+ }
57
+ export function restoreRoleRefsForTarget(opts) {
58
+ const targetId = opts.targetId?.trim() || "";
59
+ if (!targetId) {
60
+ return;
61
+ }
62
+ const cachedEntry = roleRefsByTarget.get(roleRefsKey(opts.cdpUrl, targetId));
63
+ if (!cachedEntry) {
64
+ return;
65
+ }
66
+ const state = ensurePageState(opts.page);
67
+ if (state.roleRefs) {
68
+ return;
69
+ }
70
+ state.roleRefs = cachedEntry.refs;
71
+ state.roleRefsFrameSelector = cachedEntry.frameSelector;
72
+ state.roleRefsMode = cachedEntry.mode;
73
+ }
74
+ export function ensurePageState(page) {
75
+ const existing = pageStates.get(page);
76
+ if (existing) {
77
+ return existing;
78
+ }
79
+ const state = {
80
+ console: [],
81
+ errors: [],
82
+ requests: [],
83
+ requestIds: new WeakMap(),
84
+ nextRequestId: 0,
85
+ armIdUpload: 0,
86
+ armIdDialog: 0,
87
+ armIdDownload: 0,
88
+ };
89
+ pageStates.set(page, state);
90
+ if (!observedPages.has(page)) {
91
+ observedPages.add(page);
92
+ page.on("console", (msg) => {
93
+ const entry = {
94
+ type: msg.type(),
95
+ text: msg.text(),
96
+ timestamp: new Date().toISOString(),
97
+ location: msg.location(),
98
+ };
99
+ state.console.push(entry);
100
+ if (state.console.length > MAX_CONSOLE_MESSAGES) {
101
+ state.console.shift();
102
+ }
103
+ });
104
+ page.on("pageerror", (err) => {
105
+ state.errors.push({
106
+ message: err?.message ? String(err.message) : String(err),
107
+ name: err?.name ? String(err.name) : undefined,
108
+ stack: err?.stack ? String(err.stack) : undefined,
109
+ timestamp: new Date().toISOString(),
110
+ });
111
+ if (state.errors.length > MAX_PAGE_ERRORS) {
112
+ state.errors.shift();
113
+ }
114
+ });
115
+ page.on("request", (req) => {
116
+ state.nextRequestId += 1;
117
+ const id = `r${state.nextRequestId}`;
118
+ state.requestIds.set(req, id);
119
+ state.requests.push({
120
+ id,
121
+ timestamp: new Date().toISOString(),
122
+ method: req.method(),
123
+ url: req.url(),
124
+ resourceType: req.resourceType(),
125
+ });
126
+ if (state.requests.length > MAX_NETWORK_REQUESTS) {
127
+ state.requests.shift();
128
+ }
129
+ });
130
+ page.on("response", (resp) => {
131
+ const req = resp.request();
132
+ const id = state.requestIds.get(req);
133
+ if (!id) {
134
+ return;
135
+ }
136
+ let rec;
137
+ for (let i = state.requests.length - 1; i >= 0; i -= 1) {
138
+ const candidate = state.requests[i];
139
+ if (candidate && candidate.id === id) {
140
+ rec = candidate;
141
+ break;
142
+ }
143
+ }
144
+ if (!rec) {
145
+ return;
146
+ }
147
+ rec.status = resp.status();
148
+ rec.ok = resp.ok();
149
+ });
150
+ page.on("requestfailed", (req) => {
151
+ const id = state.requestIds.get(req);
152
+ if (!id) {
153
+ return;
154
+ }
155
+ let rec;
156
+ for (let i = state.requests.length - 1; i >= 0; i -= 1) {
157
+ const candidate = state.requests[i];
158
+ if (candidate && candidate.id === id) {
159
+ rec = candidate;
160
+ break;
161
+ }
162
+ }
163
+ if (!rec) {
164
+ return;
165
+ }
166
+ rec.failureText = req.failure()?.errorText;
167
+ rec.ok = false;
168
+ });
169
+ page.on("close", () => {
170
+ pageStates.delete(page);
171
+ observedPages.delete(page);
172
+ });
173
+ }
174
+ return state;
175
+ }
176
+ function observeContext(context) {
177
+ if (observedContexts.has(context)) {
178
+ return;
179
+ }
180
+ observedContexts.add(context);
181
+ ensureContextState(context);
182
+ for (const page of context.pages()) {
183
+ ensurePageState(page);
184
+ }
185
+ context.on("page", (page) => ensurePageState(page));
186
+ }
187
+ export function ensureContextState(context) {
188
+ const existing = contextStates.get(context);
189
+ if (existing) {
190
+ return existing;
191
+ }
192
+ const state = { traceActive: false };
193
+ contextStates.set(context, state);
194
+ return state;
195
+ }
196
+ function observeBrowser(browser) {
197
+ for (const context of browser.contexts()) {
198
+ observeContext(context);
199
+ }
200
+ }
201
+ async function connectBrowser(cdpUrl) {
202
+ const normalized = normalizeCdpUrl(cdpUrl);
203
+ if (cached?.cdpUrl === normalized) {
204
+ return cached;
205
+ }
206
+ if (connecting) {
207
+ return await connecting;
208
+ }
209
+ const connectWithRetry = async () => {
210
+ let lastErr;
211
+ for (let attempt = 0; attempt < 3; attempt += 1) {
212
+ try {
213
+ const timeout = 5000 + attempt * 2000;
214
+ const wsUrl = await getChromeWebSocketUrl(normalized, timeout).catch(() => null);
215
+ const endpoint = wsUrl ?? normalized;
216
+ const headers = getHeadersWithAuth(endpoint);
217
+ const browser = await chromium.connectOverCDP(endpoint, { timeout, headers });
218
+ const onDisconnected = () => {
219
+ if (cached?.browser === browser) {
220
+ cached = null;
221
+ }
222
+ };
223
+ const connected = { browser, cdpUrl: normalized, onDisconnected };
224
+ cached = connected;
225
+ browser.on("disconnected", onDisconnected);
226
+ observeBrowser(browser);
227
+ return connected;
228
+ }
229
+ catch (err) {
230
+ lastErr = err;
231
+ const delay = 250 + attempt * 250;
232
+ await new Promise((r) => setTimeout(r, delay));
233
+ }
234
+ }
235
+ if (lastErr instanceof Error) {
236
+ throw lastErr;
237
+ }
238
+ const message = lastErr ? formatErrorMessage(lastErr) : "CDP connect failed";
239
+ throw new Error(message);
240
+ };
241
+ connecting = connectWithRetry().finally(() => {
242
+ connecting = null;
243
+ });
244
+ return await connecting;
245
+ }
246
+ async function getAllPages(browser) {
247
+ const contexts = browser.contexts();
248
+ const pages = contexts.flatMap((c) => c.pages());
249
+ return pages;
250
+ }
251
+ async function pageTargetId(page) {
252
+ const session = await page.context().newCDPSession(page);
253
+ try {
254
+ const info = (await session.send("Target.getTargetInfo"));
255
+ const targetId = String(info?.targetInfo?.targetId ?? "").trim();
256
+ return targetId || null;
257
+ }
258
+ finally {
259
+ await session.detach().catch(() => { });
260
+ }
261
+ }
262
+ async function findPageByTargetId(browser, targetId, cdpUrl) {
263
+ const pages = await getAllPages(browser);
264
+ for (const page of pages) {
265
+ const tid = await pageTargetId(page).catch(() => null);
266
+ if (tid && tid === targetId) {
267
+ return page;
268
+ }
269
+ }
270
+ if (cdpUrl) {
271
+ try {
272
+ const baseUrl = cdpUrl
273
+ .replace(/\/+$/, "")
274
+ .replace(/^ws:/, "http:")
275
+ .replace(/\/cdp$/, "");
276
+ const listUrl = `${baseUrl}/json/list`;
277
+ const response = await fetch(listUrl, { headers: getHeadersWithAuth(listUrl) });
278
+ if (response.ok) {
279
+ const targets = (await response.json());
280
+ const target = targets.find((t) => t.id === targetId);
281
+ if (target) {
282
+ const urlMatch = pages.filter((p) => p.url() === target.url);
283
+ if (urlMatch.length === 1) {
284
+ return urlMatch[0];
285
+ }
286
+ if (urlMatch.length > 1) {
287
+ const sameUrlTargets = targets.filter((t) => t.url === target.url);
288
+ if (sameUrlTargets.length === urlMatch.length) {
289
+ const idx = sameUrlTargets.findIndex((t) => t.id === targetId);
290
+ if (idx >= 0 && idx < urlMatch.length) {
291
+ return urlMatch[idx];
292
+ }
293
+ }
294
+ }
295
+ }
296
+ }
297
+ }
298
+ catch {
299
+ // Ignore fetch errors
300
+ }
301
+ }
302
+ return null;
303
+ }
304
+ export async function getPageForTargetId(opts) {
305
+ const { browser } = await connectBrowser(opts.cdpUrl);
306
+ const pages = await getAllPages(browser);
307
+ if (!pages.length) {
308
+ throw new Error("No pages available in the connected browser.");
309
+ }
310
+ const first = pages[0];
311
+ if (!opts.targetId) {
312
+ return first;
313
+ }
314
+ const found = await findPageByTargetId(browser, opts.targetId, opts.cdpUrl);
315
+ if (!found) {
316
+ if (pages.length === 1) {
317
+ return first;
318
+ }
319
+ throw new Error("tab not found");
320
+ }
321
+ return found;
322
+ }
323
+ export function refLocator(page, ref) {
324
+ const normalized = ref.startsWith("@")
325
+ ? ref.slice(1)
326
+ : ref.startsWith("ref=")
327
+ ? ref.slice(4)
328
+ : ref;
329
+ if (/^e\d+$/.test(normalized)) {
330
+ const state = pageStates.get(page);
331
+ if (state?.roleRefsMode === "aria") {
332
+ const scope = state.roleRefsFrameSelector
333
+ ? page.frameLocator(state.roleRefsFrameSelector)
334
+ : page;
335
+ return scope.locator(`aria-ref=${normalized}`);
336
+ }
337
+ const info = state?.roleRefs?.[normalized];
338
+ if (!info) {
339
+ throw new Error(`Unknown ref "${normalized}". Run a new snapshot and use a ref from that snapshot.`);
340
+ }
341
+ const scope = state?.roleRefsFrameSelector
342
+ ? page.frameLocator(state.roleRefsFrameSelector)
343
+ : page;
344
+ const locAny = scope;
345
+ const locator = info.name
346
+ ? locAny.getByRole(info.role, { name: info.name, exact: true })
347
+ : locAny.getByRole(info.role);
348
+ return info.nth !== undefined ? locator.nth(info.nth) : locator;
349
+ }
350
+ return page.locator(`aria-ref=${normalized}`);
351
+ }
352
+ export async function closePlaywrightBrowserConnection() {
353
+ const cur = cached;
354
+ cached = null;
355
+ connecting = null;
356
+ if (!cur) {
357
+ return;
358
+ }
359
+ if (cur.onDisconnected && typeof cur.browser.off === "function") {
360
+ cur.browser.off("disconnected", cur.onDisconnected);
361
+ }
362
+ await cur.browser.close().catch(() => { });
363
+ }
364
+ function normalizeCdpHttpBaseForJsonEndpoints(cdpUrl) {
365
+ try {
366
+ const url = new URL(cdpUrl);
367
+ if (url.protocol === "ws:") {
368
+ url.protocol = "http:";
369
+ }
370
+ else if (url.protocol === "wss:") {
371
+ url.protocol = "https:";
372
+ }
373
+ url.pathname = url.pathname.replace(/\/devtools\/browser\/.*$/, "");
374
+ url.pathname = url.pathname.replace(/\/cdp$/, "");
375
+ return url.toString().replace(/\/$/, "");
376
+ }
377
+ catch {
378
+ return cdpUrl
379
+ .replace(/^ws:/, "http:")
380
+ .replace(/^wss:/, "https:")
381
+ .replace(/\/devtools\/browser\/.*$/, "")
382
+ .replace(/\/cdp$/, "")
383
+ .replace(/\/$/, "");
384
+ }
385
+ }
386
+ function cdpSocketNeedsAttach(wsUrl) {
387
+ try {
388
+ const pathname = new URL(wsUrl).pathname;
389
+ return (pathname === "/cdp" || pathname.endsWith("/cdp") || pathname.includes("/devtools/browser/"));
390
+ }
391
+ catch {
392
+ return false;
393
+ }
394
+ }
395
+ async function tryTerminateExecutionViaCdp(opts) {
396
+ const cdpHttpBase = normalizeCdpHttpBaseForJsonEndpoints(opts.cdpUrl);
397
+ const listUrl = appendCdpPath(cdpHttpBase, "/json/list");
398
+ const pages = await fetchJson(listUrl, 2000).catch(() => null);
399
+ if (!pages || pages.length === 0) {
400
+ return;
401
+ }
402
+ const target = pages.find((p) => String(p.id ?? "").trim() === opts.targetId);
403
+ const wsUrlRaw = String(target?.webSocketDebuggerUrl ?? "").trim();
404
+ if (!wsUrlRaw) {
405
+ return;
406
+ }
407
+ const wsUrl = normalizeCdpWsUrl(wsUrlRaw, cdpHttpBase);
408
+ const needsAttach = cdpSocketNeedsAttach(wsUrl);
409
+ const runWithTimeout = async (work, ms) => {
410
+ let timer;
411
+ const timeoutPromise = new Promise((_, reject) => {
412
+ timer = setTimeout(() => reject(new Error("CDP command timed out")), ms);
413
+ });
414
+ try {
415
+ return await Promise.race([work, timeoutPromise]);
416
+ }
417
+ finally {
418
+ if (timer) {
419
+ clearTimeout(timer);
420
+ }
421
+ }
422
+ };
423
+ await withCdpSocket(wsUrl, async (send) => {
424
+ let sessionId;
425
+ try {
426
+ if (needsAttach) {
427
+ const attached = (await runWithTimeout(send("Target.attachToTarget", { targetId: opts.targetId, flatten: true }), 1500));
428
+ if (typeof attached?.sessionId === "string" && attached.sessionId.trim()) {
429
+ sessionId = attached.sessionId;
430
+ }
431
+ }
432
+ await runWithTimeout(send("Runtime.terminateExecution", undefined, sessionId), 1500);
433
+ if (sessionId) {
434
+ void send("Target.detachFromTarget", { sessionId }).catch(() => { });
435
+ }
436
+ }
437
+ catch {
438
+ // Best-effort; ignore
439
+ }
440
+ }, { handshakeTimeoutMs: 2000 }).catch(() => { });
441
+ }
442
+ export async function forceDisconnectPlaywrightForTarget(opts) {
443
+ const normalized = normalizeCdpUrl(opts.cdpUrl);
444
+ if (cached?.cdpUrl !== normalized) {
445
+ return;
446
+ }
447
+ const cur = cached;
448
+ cached = null;
449
+ connecting = null;
450
+ if (cur) {
451
+ if (cur.onDisconnected && typeof cur.browser.off === "function") {
452
+ cur.browser.off("disconnected", cur.onDisconnected);
453
+ }
454
+ const targetId = opts.targetId?.trim() || "";
455
+ if (targetId) {
456
+ await tryTerminateExecutionViaCdp({ cdpUrl: normalized, targetId }).catch(() => { });
457
+ }
458
+ cur.browser.close().catch(() => { });
459
+ }
460
+ }
461
+ export async function listPagesViaPlaywright(opts) {
462
+ const { browser } = await connectBrowser(opts.cdpUrl);
463
+ const pages = await getAllPages(browser);
464
+ const results = [];
465
+ for (const page of pages) {
466
+ const tid = await pageTargetId(page).catch(() => null);
467
+ if (tid) {
468
+ results.push({
469
+ targetId: tid,
470
+ title: await page.title().catch(() => ""),
471
+ url: page.url(),
472
+ type: "page",
473
+ });
474
+ }
475
+ }
476
+ return results;
477
+ }
478
+ export async function createPageViaPlaywright(opts) {
479
+ const { browser } = await connectBrowser(opts.cdpUrl);
480
+ const context = browser.contexts()[0] ?? (await browser.newContext());
481
+ ensureContextState(context);
482
+ const page = await context.newPage();
483
+ ensurePageState(page);
484
+ const targetUrl = opts.url.trim() || "about:blank";
485
+ if (targetUrl !== "about:blank") {
486
+ await page.goto(targetUrl, { timeout: 30_000 }).catch(() => { });
487
+ }
488
+ const tid = await pageTargetId(page).catch(() => null);
489
+ if (!tid) {
490
+ throw new Error("Failed to get targetId for new page");
491
+ }
492
+ return {
493
+ targetId: tid,
494
+ title: await page.title().catch(() => ""),
495
+ url: page.url(),
496
+ type: "page",
497
+ };
498
+ }
499
+ export async function closePageByTargetIdViaPlaywright(opts) {
500
+ const { browser } = await connectBrowser(opts.cdpUrl);
501
+ const page = await findPageByTargetId(browser, opts.targetId, opts.cdpUrl);
502
+ if (!page) {
503
+ throw new Error("tab not found");
504
+ }
505
+ await page.close();
506
+ }
507
+ export async function focusPageByTargetIdViaPlaywright(opts) {
508
+ const { browser } = await connectBrowser(opts.cdpUrl);
509
+ const page = await findPageByTargetId(browser, opts.targetId, opts.cdpUrl);
510
+ if (!page) {
511
+ throw new Error("tab not found");
512
+ }
513
+ try {
514
+ await page.bringToFront();
515
+ }
516
+ catch (err) {
517
+ const session = await page.context().newCDPSession(page);
518
+ try {
519
+ await session.send("Page.bringToFront");
520
+ return;
521
+ }
522
+ catch {
523
+ throw err;
524
+ }
525
+ finally {
526
+ await session.detach().catch(() => { });
527
+ }
528
+ }
529
+ }
530
+ //# sourceMappingURL=pw-session.js.map
@@ -0,0 +1,22 @@
1
+ import type { BrowserConsoleMessage, BrowserNetworkRequest, BrowserPageError } from "./pw-session.js";
2
+ export declare function getPageErrorsViaPlaywright(opts: {
3
+ cdpUrl: string;
4
+ targetId?: string;
5
+ clear?: boolean;
6
+ }): Promise<{
7
+ errors: BrowserPageError[];
8
+ }>;
9
+ export declare function getNetworkRequestsViaPlaywright(opts: {
10
+ cdpUrl: string;
11
+ targetId?: string;
12
+ filter?: string;
13
+ clear?: boolean;
14
+ }): Promise<{
15
+ requests: BrowserNetworkRequest[];
16
+ }>;
17
+ export declare function getConsoleMessagesViaPlaywright(opts: {
18
+ cdpUrl: string;
19
+ targetId?: string;
20
+ level?: string;
21
+ }): Promise<BrowserConsoleMessage[]>;
22
+ //# sourceMappingURL=pw-tools-core.activity.d.ts.map
@@ -0,0 +1,47 @@
1
+ import { ensurePageState, getPageForTargetId } from "./pw-session.js";
2
+ export async function getPageErrorsViaPlaywright(opts) {
3
+ const page = await getPageForTargetId(opts);
4
+ const state = ensurePageState(page);
5
+ const errors = [...state.errors];
6
+ if (opts.clear) {
7
+ state.errors = [];
8
+ }
9
+ return { errors };
10
+ }
11
+ export async function getNetworkRequestsViaPlaywright(opts) {
12
+ const page = await getPageForTargetId(opts);
13
+ const state = ensurePageState(page);
14
+ const raw = [...state.requests];
15
+ const filter = typeof opts.filter === "string" ? opts.filter.trim() : "";
16
+ const requests = filter ? raw.filter((r) => r.url.includes(filter)) : raw;
17
+ if (opts.clear) {
18
+ state.requests = [];
19
+ state.requestIds = new WeakMap();
20
+ }
21
+ return { requests };
22
+ }
23
+ function consolePriority(level) {
24
+ switch (level) {
25
+ case "error":
26
+ return 3;
27
+ case "warning":
28
+ return 2;
29
+ case "info":
30
+ case "log":
31
+ return 1;
32
+ case "debug":
33
+ return 0;
34
+ default:
35
+ return 1;
36
+ }
37
+ }
38
+ export async function getConsoleMessagesViaPlaywright(opts) {
39
+ const page = await getPageForTargetId(opts);
40
+ const state = ensurePageState(page);
41
+ if (!opts.level) {
42
+ return [...state.console];
43
+ }
44
+ const min = consolePriority(opts.level);
45
+ return state.console.filter((msg) => consolePriority(msg.type) >= min);
46
+ }
47
+ //# sourceMappingURL=pw-tools-core.activity.js.map
@@ -0,0 +1,9 @@
1
+ export * from "./pw-tools-core.activity.js";
2
+ export * from "./pw-tools-core.downloads.js";
3
+ export * from "./pw-tools-core.interactions.js";
4
+ export * from "./pw-tools-core.responses.js";
5
+ export * from "./pw-tools-core.snapshot.js";
6
+ export * from "./pw-tools-core.state.js";
7
+ export * from "./pw-tools-core.storage.js";
8
+ export * from "./pw-tools-core.trace.js";
9
+ //# sourceMappingURL=pw-tools-core.d.ts.map
@@ -0,0 +1,35 @@
1
+ export declare function armFileUploadViaPlaywright(opts: {
2
+ cdpUrl: string;
3
+ targetId?: string;
4
+ paths?: string[];
5
+ timeoutMs?: number;
6
+ }): Promise<void>;
7
+ export declare function armDialogViaPlaywright(opts: {
8
+ cdpUrl: string;
9
+ targetId?: string;
10
+ accept: boolean;
11
+ promptText?: string;
12
+ timeoutMs?: number;
13
+ }): Promise<void>;
14
+ export declare function waitForDownloadViaPlaywright(opts: {
15
+ cdpUrl: string;
16
+ targetId?: string;
17
+ path?: string;
18
+ timeoutMs?: number;
19
+ }): Promise<{
20
+ url: string;
21
+ suggestedFilename: string;
22
+ path: string;
23
+ }>;
24
+ export declare function downloadViaPlaywright(opts: {
25
+ cdpUrl: string;
26
+ targetId?: string;
27
+ ref: string;
28
+ path: string;
29
+ timeoutMs?: number;
30
+ }): Promise<{
31
+ url: string;
32
+ suggestedFilename: string;
33
+ path: string;
34
+ }>;
35
+ //# sourceMappingURL=pw-tools-core.downloads.d.ts.map