system-testing 1.0.78 → 1.0.81
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +232 -6
- package/build/browser-command-client.d.ts +19 -0
- package/build/browser-command-client.js +39 -0
- package/build/browser-command-runner.d.ts +34 -0
- package/build/browser-command-runner.js +155 -0
- package/build/browser-daemon-constants.d.ts +2 -0
- package/build/browser-daemon-constants.js +3 -0
- package/build/browser-process.d.ts +45 -0
- package/build/browser-process.js +134 -0
- package/build/browser-registry.d.ts +44 -0
- package/build/browser-registry.js +191 -0
- package/build/browser.d.ts +240 -0
- package/build/browser.js +375 -0
- package/build/cli-helpers.d.ts +16 -0
- package/build/cli-helpers.js +177 -0
- package/build/cli.d.ts +2 -0
- package/build/cli.js +81 -0
- package/build/drivers/appium-driver.js +21 -21
- package/build/drivers/webdriver-driver.d.ts +4 -4
- package/build/drivers/webdriver-driver.js +32 -28
- package/build/index.d.ts +9 -1
- package/build/index.js +10 -3
- package/build/system-test-browser-helper.d.ts +6 -12
- package/build/system-test-browser-helper.js +12 -13
- package/build/system-test.d.ts +3 -189
- package/build/system-test.js +6 -220
- package/build/use-system-test-expo.d.ts +16 -0
- package/build/use-system-test-expo.js +34 -0
- package/build/use-system-test-react-native.d.ts +25 -0
- package/build/use-system-test-react-native.js +20 -0
- package/build/use-system-test-shape-hook.d.ts +35 -0
- package/build/use-system-test-shape-hook.js +74 -0
- package/build/use-system-test.d.ts +19 -10
- package/build/use-system-test.js +26 -72
- package/package.json +17 -8
package/build/browser.js
ADDED
|
@@ -0,0 +1,375 @@
|
|
|
1
|
+
// @ts-check
|
|
2
|
+
import fs from "node:fs/promises";
|
|
3
|
+
import moment from "moment";
|
|
4
|
+
import { prettify } from "htmlfy";
|
|
5
|
+
import timeout from "awaitery/build/timeout.js";
|
|
6
|
+
import SeleniumDriver from "./drivers/selenium-driver.js";
|
|
7
|
+
import AppiumDriver from "./drivers/appium-driver.js";
|
|
8
|
+
/**
|
|
9
|
+
* @typedef {object} BrowserArgs
|
|
10
|
+
* @property {boolean} [debug] Enable debug logging.
|
|
11
|
+
* @property {BrowserDriverConfig} [driver] Driver configuration.
|
|
12
|
+
* @property {import("./system-test-communicator.js").default} [communicator] Optional command communicator for helper-driven navigation.
|
|
13
|
+
* @property {string} [screenshotsPath] Directory used for saved screenshots and browser artifacts.
|
|
14
|
+
*/
|
|
15
|
+
/**
|
|
16
|
+
* @typedef {object} BrowserDriverConfig
|
|
17
|
+
* @property {"selenium"|"appium"} [type] Driver implementation to use.
|
|
18
|
+
* @property {Record<string, any>} [options] Driver-specific options.
|
|
19
|
+
*/
|
|
20
|
+
/**
|
|
21
|
+
* @typedef {object} BrowserNavigationArgs
|
|
22
|
+
* @property {number} [timeout] Override the timeout for this navigation command.
|
|
23
|
+
*/
|
|
24
|
+
/** Generic browser session wrapper around the configured driver. */
|
|
25
|
+
export default class Browser {
|
|
26
|
+
/** @param {BrowserArgs} [args] */
|
|
27
|
+
constructor({ debug = false, driver, communicator, screenshotsPath = `${process.cwd()}/tmp/screenshots`, ...restArgs } = {}) {
|
|
28
|
+
/** @type {import("selenium-webdriver").WebDriver | undefined} */
|
|
29
|
+
this.driver = undefined;
|
|
30
|
+
/** @type {import("./drivers/webdriver-driver.js").default | undefined} */
|
|
31
|
+
this.driverAdapter = undefined;
|
|
32
|
+
this._debug = false;
|
|
33
|
+
/** @type {BrowserDriverConfig | undefined} */
|
|
34
|
+
this._driverConfig = undefined;
|
|
35
|
+
/** @type {Error | undefined} */
|
|
36
|
+
this._httpServerError = undefined;
|
|
37
|
+
/**
|
|
38
|
+
* @param {Error} error
|
|
39
|
+
* @returns {void}
|
|
40
|
+
*/
|
|
41
|
+
this.onHttpServerError = (error) => {
|
|
42
|
+
const errorMessage = error instanceof Error ? error.message : String(error);
|
|
43
|
+
this._httpServerError = error instanceof Error ? error : new Error(errorMessage);
|
|
44
|
+
console.error(`HTTP server error: ${errorMessage}`);
|
|
45
|
+
};
|
|
46
|
+
const restArgsKeys = Object.keys(restArgs);
|
|
47
|
+
if (restArgsKeys.length > 0) {
|
|
48
|
+
throw new Error(`Unknown browser arguments: ${restArgsKeys.join(", ")}`);
|
|
49
|
+
}
|
|
50
|
+
this._debug = debug;
|
|
51
|
+
this._driverConfig = driver;
|
|
52
|
+
this._screenshotsPath = screenshotsPath;
|
|
53
|
+
this.communicator = communicator;
|
|
54
|
+
this.driverAdapter = this.createDriver(driver);
|
|
55
|
+
}
|
|
56
|
+
/**
|
|
57
|
+
* @param {BrowserDriverConfig} [driverConfig]
|
|
58
|
+
* @returns {import("./drivers/webdriver-driver.js").default}
|
|
59
|
+
*/
|
|
60
|
+
createDriver(driverConfig = {}) {
|
|
61
|
+
const { type = "selenium", options, ...restArgs } = driverConfig;
|
|
62
|
+
const restArgsKeys = Object.keys(restArgs);
|
|
63
|
+
if (restArgsKeys.length > 0) {
|
|
64
|
+
throw new Error(`Unknown driver args: ${restArgsKeys.join(", ")}`);
|
|
65
|
+
}
|
|
66
|
+
if (type === "selenium") {
|
|
67
|
+
return new SeleniumDriver({ browser: this, options });
|
|
68
|
+
}
|
|
69
|
+
if (type === "appium") {
|
|
70
|
+
return new AppiumDriver({ browser: this, options });
|
|
71
|
+
}
|
|
72
|
+
throw new Error(`Unsupported driver type: ${type}`);
|
|
73
|
+
}
|
|
74
|
+
/**
|
|
75
|
+
* @param {import("./system-test-communicator.js").default | undefined} communicator
|
|
76
|
+
* @returns {void}
|
|
77
|
+
*/
|
|
78
|
+
setCommunicator(communicator) {
|
|
79
|
+
this.communicator = communicator;
|
|
80
|
+
}
|
|
81
|
+
/** @returns {boolean} */
|
|
82
|
+
communicatorExists() {
|
|
83
|
+
return Boolean(this.communicator);
|
|
84
|
+
}
|
|
85
|
+
/**
|
|
86
|
+
* @param {string} baseSelector
|
|
87
|
+
* @returns {void}
|
|
88
|
+
*/
|
|
89
|
+
setBaseSelector(baseSelector) { this._baseSelector = baseSelector; }
|
|
90
|
+
/** @returns {string | undefined} */
|
|
91
|
+
getBaseSelector() { return this._baseSelector; }
|
|
92
|
+
/**
|
|
93
|
+
* @param {string} selector
|
|
94
|
+
* @returns {string}
|
|
95
|
+
*/
|
|
96
|
+
getSelector(selector) {
|
|
97
|
+
return this.getBaseSelector() ? `${this.getBaseSelector()} ${selector}` : selector;
|
|
98
|
+
}
|
|
99
|
+
/**
|
|
100
|
+
* @param {...any} args
|
|
101
|
+
* @returns {void}
|
|
102
|
+
*/
|
|
103
|
+
debugError(...args) {
|
|
104
|
+
console.error("[Browser error]", ...args);
|
|
105
|
+
}
|
|
106
|
+
/**
|
|
107
|
+
* @param {...any} args
|
|
108
|
+
* @returns {void}
|
|
109
|
+
*/
|
|
110
|
+
debugLog(...args) {
|
|
111
|
+
if (this._debug) {
|
|
112
|
+
console.log("[Browser debug]", ...args);
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
/** @returns {void} */
|
|
116
|
+
throwIfHttpServerError() {
|
|
117
|
+
if (this._httpServerError) {
|
|
118
|
+
throw new Error(`HTTP server error: ${this._httpServerError.message}`);
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
/** @returns {import("selenium-webdriver").WebDriver} */
|
|
122
|
+
getDriver() {
|
|
123
|
+
return this.getDriverAdapter().getWebDriver();
|
|
124
|
+
}
|
|
125
|
+
/** @returns {import("./drivers/webdriver-driver.js").default} */
|
|
126
|
+
getDriverAdapter() {
|
|
127
|
+
if (!this.driverAdapter) {
|
|
128
|
+
throw new Error("Driver hasn't been initialized yet");
|
|
129
|
+
}
|
|
130
|
+
return this.driverAdapter;
|
|
131
|
+
}
|
|
132
|
+
/** @returns {number} */
|
|
133
|
+
getTimeouts() { return this.getDriverAdapter().getTimeouts(); }
|
|
134
|
+
/** @returns {Promise<void>} */
|
|
135
|
+
async restoreTimeouts() {
|
|
136
|
+
await this.getDriverAdapter().restoreTimeouts();
|
|
137
|
+
}
|
|
138
|
+
/**
|
|
139
|
+
* @param {number} newTimeout
|
|
140
|
+
* @returns {Promise<void>}
|
|
141
|
+
*/
|
|
142
|
+
async driverSetTimeouts(newTimeout) {
|
|
143
|
+
await this.getDriverAdapter().driverSetTimeouts(newTimeout);
|
|
144
|
+
}
|
|
145
|
+
/**
|
|
146
|
+
* @param {number} newTimeout
|
|
147
|
+
* @returns {Promise<void>}
|
|
148
|
+
*/
|
|
149
|
+
async setTimeouts(newTimeout) {
|
|
150
|
+
await this.getDriverAdapter().setTimeouts(newTimeout);
|
|
151
|
+
}
|
|
152
|
+
/** @returns {Promise<string[]>} */
|
|
153
|
+
async getBrowserLogs() {
|
|
154
|
+
return await this.getDriverAdapter().getBrowserLogs();
|
|
155
|
+
}
|
|
156
|
+
/** @returns {Promise<string>} */
|
|
157
|
+
async getCurrentUrl() {
|
|
158
|
+
return await this.getDriverAdapter().getCurrentUrl();
|
|
159
|
+
}
|
|
160
|
+
/**
|
|
161
|
+
* @param {string} selector
|
|
162
|
+
* @param {import("./system-test.js").FindArgs} [args]
|
|
163
|
+
* @returns {Promise<import("selenium-webdriver").WebElement[]>}
|
|
164
|
+
*/
|
|
165
|
+
async all(selector, args = {}) {
|
|
166
|
+
return await this.getDriverAdapter().all(selector, args);
|
|
167
|
+
}
|
|
168
|
+
/**
|
|
169
|
+
* @param {string} selector
|
|
170
|
+
* @param {import("./system-test.js").FindArgs} [args]
|
|
171
|
+
* @returns {Promise<import("selenium-webdriver").WebElement>}
|
|
172
|
+
*/
|
|
173
|
+
async find(selector, args = {}) {
|
|
174
|
+
return await this.getDriverAdapter().find(selector, args);
|
|
175
|
+
}
|
|
176
|
+
/**
|
|
177
|
+
* @param {string} testID
|
|
178
|
+
* @param {import("./system-test.js").FindArgs} [args]
|
|
179
|
+
* @returns {Promise<import("selenium-webdriver").WebElement>}
|
|
180
|
+
*/
|
|
181
|
+
async findByTestID(testID, args) {
|
|
182
|
+
return await this.getDriverAdapter().findByTestID(testID, args);
|
|
183
|
+
}
|
|
184
|
+
/**
|
|
185
|
+
* @param {string} selector
|
|
186
|
+
* @param {import("./system-test.js").FindArgs} [args]
|
|
187
|
+
* @returns {Promise<import("selenium-webdriver").WebElement>}
|
|
188
|
+
*/
|
|
189
|
+
async findNoWait(selector, args = {}) {
|
|
190
|
+
return await this.getDriverAdapter().findNoWait(selector, args);
|
|
191
|
+
}
|
|
192
|
+
/**
|
|
193
|
+
* @param {string | import("selenium-webdriver").WebElement} elementOrIdentifier
|
|
194
|
+
* @param {import("./system-test.js").FindArgs} [args]
|
|
195
|
+
* @returns {Promise<void>}
|
|
196
|
+
*/
|
|
197
|
+
async click(elementOrIdentifier, args) {
|
|
198
|
+
await this.getDriverAdapter().click(elementOrIdentifier, args);
|
|
199
|
+
}
|
|
200
|
+
/**
|
|
201
|
+
* @param {import("selenium-webdriver").WebElement|string|{selector: string} & import("./system-test.js").FindArgs} elementOrIdentifier
|
|
202
|
+
* @param {string} methodName
|
|
203
|
+
* @param {...any} args
|
|
204
|
+
* @returns {Promise<any>}
|
|
205
|
+
*/
|
|
206
|
+
async interact(elementOrIdentifier, methodName, ...args) {
|
|
207
|
+
return await this.getDriverAdapter().interact(elementOrIdentifier, methodName, ...args);
|
|
208
|
+
}
|
|
209
|
+
/**
|
|
210
|
+
* @param {string} selector
|
|
211
|
+
* @param {import("./system-test.js").WaitForNoSelectorArgs} [args]
|
|
212
|
+
* @returns {Promise<void>}
|
|
213
|
+
*/
|
|
214
|
+
async waitForNoSelector(selector, args = {}) {
|
|
215
|
+
await this.getDriverAdapter().waitForNoSelector(selector, args);
|
|
216
|
+
}
|
|
217
|
+
/**
|
|
218
|
+
* @param {string} selector
|
|
219
|
+
* @param {import("./system-test.js").FindArgs} [args]
|
|
220
|
+
* @returns {Promise<void>}
|
|
221
|
+
*/
|
|
222
|
+
async expectNoElement(selector, args = {}) {
|
|
223
|
+
let found = false;
|
|
224
|
+
try {
|
|
225
|
+
await this.findNoWait(selector, args);
|
|
226
|
+
found = true;
|
|
227
|
+
}
|
|
228
|
+
catch (error) {
|
|
229
|
+
if (error instanceof Error && error.message.startsWith("Element couldn't be found after ")) {
|
|
230
|
+
// Ignore
|
|
231
|
+
}
|
|
232
|
+
else {
|
|
233
|
+
throw error;
|
|
234
|
+
}
|
|
235
|
+
}
|
|
236
|
+
if (found) {
|
|
237
|
+
throw new Error(`Expected not to find: ${selector}`);
|
|
238
|
+
}
|
|
239
|
+
}
|
|
240
|
+
/** @returns {Promise<string>} */
|
|
241
|
+
async getHTML() {
|
|
242
|
+
return await this.getDriverAdapter().getHTML();
|
|
243
|
+
}
|
|
244
|
+
/**
|
|
245
|
+
* @param {number | undefined} timeoutOverride
|
|
246
|
+
* @returns {number}
|
|
247
|
+
*/
|
|
248
|
+
getCommandTimeout(timeoutOverride) {
|
|
249
|
+
if (timeoutOverride !== undefined) {
|
|
250
|
+
return timeoutOverride;
|
|
251
|
+
}
|
|
252
|
+
return this.getTimeouts();
|
|
253
|
+
}
|
|
254
|
+
/**
|
|
255
|
+
* @param {string} path
|
|
256
|
+
* @returns {Promise<void>}
|
|
257
|
+
*/
|
|
258
|
+
async driverVisit(path) {
|
|
259
|
+
await this.getDriverAdapter().driverVisit(path);
|
|
260
|
+
}
|
|
261
|
+
/**
|
|
262
|
+
* @param {string} type
|
|
263
|
+
* @param {string} path
|
|
264
|
+
* @param {BrowserNavigationArgs} [args]
|
|
265
|
+
* @returns {Promise<void>}
|
|
266
|
+
*/
|
|
267
|
+
async sendBrowserCommand(type, path, args = {}) {
|
|
268
|
+
if (!this.communicator) {
|
|
269
|
+
throw new Error("Communicator hasn't been initialized yet");
|
|
270
|
+
}
|
|
271
|
+
await timeout({ timeout: this.getCommandTimeout(args.timeout), errorMessage: `timeout while sending browser command ${type}: ${path}` }, async () => await this.communicator.sendCommand({ type, path }));
|
|
272
|
+
}
|
|
273
|
+
/**
|
|
274
|
+
* Visits a path using the injected browser helper when available, otherwise navigates directly with the driver.
|
|
275
|
+
* @param {string} path
|
|
276
|
+
* @param {BrowserNavigationArgs} [args]
|
|
277
|
+
* @returns {Promise<void>}
|
|
278
|
+
*/
|
|
279
|
+
async visit(path, args = {}) {
|
|
280
|
+
if (this.communicatorExists()) {
|
|
281
|
+
await this.sendBrowserCommand("visit", path, args);
|
|
282
|
+
}
|
|
283
|
+
else {
|
|
284
|
+
await timeout({ timeout: this.getCommandTimeout(args.timeout), errorMessage: `timeout while visiting path: ${path}` }, async () => await this.driverVisit(path));
|
|
285
|
+
}
|
|
286
|
+
}
|
|
287
|
+
/**
|
|
288
|
+
* Dismisses to a path via the injected browser helper when available, otherwise navigates directly with the driver.
|
|
289
|
+
* @param {string} path
|
|
290
|
+
* @param {BrowserNavigationArgs} [args]
|
|
291
|
+
* @returns {Promise<void>}
|
|
292
|
+
*/
|
|
293
|
+
async dismissTo(path, args = {}) {
|
|
294
|
+
if (this.communicatorExists()) {
|
|
295
|
+
await this.sendBrowserCommand("dismissTo", path, args);
|
|
296
|
+
}
|
|
297
|
+
else {
|
|
298
|
+
await timeout({ timeout: this.getCommandTimeout(args.timeout), errorMessage: `timeout while dismissing to path: ${path}` }, async () => await this.driverVisit(path));
|
|
299
|
+
}
|
|
300
|
+
}
|
|
301
|
+
/**
|
|
302
|
+
* Formats browser logs for console output and truncates overly long output.
|
|
303
|
+
* @param {string[]} logs
|
|
304
|
+
* @param {number} [maxLines]
|
|
305
|
+
* @returns {string[]}
|
|
306
|
+
*/
|
|
307
|
+
formatBrowserLogsForConsole(logs, maxLines = 200) {
|
|
308
|
+
if (!Array.isArray(logs) || logs.length === 0) {
|
|
309
|
+
return ["(no browser logs)"];
|
|
310
|
+
}
|
|
311
|
+
if (logs.length <= maxLines) {
|
|
312
|
+
return logs;
|
|
313
|
+
}
|
|
314
|
+
const keptLogs = logs.slice(logs.length - maxLines);
|
|
315
|
+
const hiddenCount = logs.length - maxLines;
|
|
316
|
+
return [`(showing last ${maxLines} of ${logs.length} browser logs, ${hiddenCount} omitted)`, ...keptLogs];
|
|
317
|
+
}
|
|
318
|
+
/**
|
|
319
|
+
* @param {string[]} logs
|
|
320
|
+
* @returns {void}
|
|
321
|
+
*/
|
|
322
|
+
printBrowserLogsForFailure(logs) {
|
|
323
|
+
console.log("Browser logs:");
|
|
324
|
+
for (const line of this.formatBrowserLogsForConsole(logs)) {
|
|
325
|
+
console.log(line);
|
|
326
|
+
}
|
|
327
|
+
}
|
|
328
|
+
/**
|
|
329
|
+
* Takes a screenshot, writes HTML/browser logs to disk, and returns the collected artifacts.
|
|
330
|
+
* @returns {Promise<{currentUrl: string, html: string, htmlPath: string, logs: string[], logsPath: string, screenshotPath: string}>}
|
|
331
|
+
*/
|
|
332
|
+
async takeScreenshot() {
|
|
333
|
+
this.debugLog("Getting path for screenshots");
|
|
334
|
+
const path = this._screenshotsPath;
|
|
335
|
+
this.debugLog(`Creating dir with recursive: ${path}`);
|
|
336
|
+
await fs.mkdir(path, { recursive: true });
|
|
337
|
+
this.debugLog("Getting screenshot image content");
|
|
338
|
+
const imageContent = await timeout({ timeout: 5000, errorMessage: "timeout while taking screenshot" }, async () => await this.getDriverAdapter().takeScreenshot());
|
|
339
|
+
this.debugLog("Generating date variables");
|
|
340
|
+
const now = new Date();
|
|
341
|
+
const timestamp = moment(now).format("YYYY-MM-DD-HH-MM-SS");
|
|
342
|
+
const screenshotPath = `${path}/${timestamp}.png`;
|
|
343
|
+
const htmlPath = `${path}/${timestamp}.html`;
|
|
344
|
+
const logsPath = `${path}/${timestamp}.logs.txt`;
|
|
345
|
+
this.debugLog("Getting browser logs");
|
|
346
|
+
const logs = await timeout({ timeout: 5000, errorMessage: "timeout while reading browser logs" }, async () => await this.getBrowserLogs());
|
|
347
|
+
const html = await timeout({ timeout: 5000, errorMessage: "timeout while reading page HTML" }, async () => await this.getHTML());
|
|
348
|
+
const htmlPretty = prettify(html);
|
|
349
|
+
this.printBrowserLogsForFailure(logs);
|
|
350
|
+
this.debugLog("Writing files");
|
|
351
|
+
await fs.writeFile(htmlPath, htmlPretty);
|
|
352
|
+
await fs.writeFile(logsPath, logs.join("\n"));
|
|
353
|
+
await fs.writeFile(screenshotPath, imageContent, "base64");
|
|
354
|
+
const currentUrl = await this.getCurrentUrl();
|
|
355
|
+
console.log("Current URL:", currentUrl);
|
|
356
|
+
console.log("Logs:", logsPath);
|
|
357
|
+
console.log("Screenshot:", screenshotPath);
|
|
358
|
+
console.log("HTML:", htmlPath);
|
|
359
|
+
return {
|
|
360
|
+
currentUrl,
|
|
361
|
+
html,
|
|
362
|
+
htmlPath,
|
|
363
|
+
logs,
|
|
364
|
+
logsPath,
|
|
365
|
+
screenshotPath
|
|
366
|
+
};
|
|
367
|
+
}
|
|
368
|
+
/** @returns {Promise<void>} */
|
|
369
|
+
async stopDriver() {
|
|
370
|
+
if (this.driverAdapter) {
|
|
371
|
+
await this.driverAdapter.stop();
|
|
372
|
+
}
|
|
373
|
+
}
|
|
374
|
+
}
|
|
375
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"browser.js","sourceRoot":"","sources":["../src/browser.js"],"names":[],"mappings":"AAAA,YAAY;AAEZ,OAAO,EAAE,MAAM,kBAAkB,CAAA;AACjC,OAAO,MAAM,MAAM,QAAQ,CAAA;AAC3B,OAAO,EAAC,QAAQ,EAAC,MAAM,QAAQ,CAAA;AAC/B,OAAO,OAAO,MAAM,2BAA2B,CAAA;AAC/C,OAAO,cAAc,MAAM,8BAA8B,CAAA;AACzD,OAAO,YAAY,MAAM,4BAA4B,CAAA;AAErD;;;;;;GAMG;AACH;;;;GAIG;AACH;;;GAGG;AAEH,oEAAoE;AACpE,MAAM,CAAC,OAAO,OAAO,OAAO;IAa1B,kCAAkC;IAClC,YAAY,EAAC,KAAK,GAAG,KAAK,EAAE,MAAM,EAAE,YAAY,EAAE,eAAe,GAAG,GAAG,OAAO,CAAC,GAAG,EAAE,kBAAkB,EAAE,GAAG,QAAQ,EAAC,GAAG,EAAE;QAbzH,iEAAiE;QACjE,WAAM,GAAG,SAAS,CAAA;QAElB,0EAA0E;QAC1E,kBAAa,GAAG,SAAS,CAAA;QAEzB,WAAM,GAAG,KAAK,CAAA;QACd,8CAA8C;QAC9C,kBAAa,GAAG,SAAS,CAAA;QACzB,gCAAgC;QAChC,qBAAgB,GAAG,SAAS,CAAA;QA+F5B;;;WAGG;QACH,sBAAiB,GAAG,CAAC,KAAK,EAAE,EAAE;YAC5B,MAAM,YAAY,GAAG,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAA;YAE3E,IAAI,CAAC,gBAAgB,GAAG,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,KAAK,CAAC,YAAY,CAAC,CAAA;YAChF,OAAO,CAAC,KAAK,CAAC,sBAAsB,YAAY,EAAE,CAAC,CAAA;QACrD,CAAC,CAAA;QApGC,MAAM,YAAY,GAAG,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAA;QAE1C,IAAI,YAAY,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAC5B,MAAM,IAAI,KAAK,CAAC,8BAA8B,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,CAAA;QAC1E,CAAC;QAED,IAAI,CAAC,MAAM,GAAG,KAAK,CAAA;QACnB,IAAI,CAAC,aAAa,GAAG,MAAM,CAAA;QAC3B,IAAI,CAAC,gBAAgB,GAAG,eAAe,CAAA;QACvC,IAAI,CAAC,YAAY,GAAG,YAAY,CAAA;QAChC,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,CAAA;IAChD,CAAC;IAED;;;OAGG;IACH,YAAY,CAAC,YAAY,GAAG,EAAE;QAC5B,MAAM,EAAC,IAAI,GAAG,UAAU,EAAE,OAAO,EAAE,GAAG,QAAQ,EAAC,GAAG,YAAY,CAAA;QAC9D,MAAM,YAAY,GAAG,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAA;QAE1C,IAAI,YAAY,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAC5B,MAAM,IAAI,KAAK,CAAC,wBAAwB,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,CAAA;QACpE,CAAC;QAED,IAAI,IAAI,KAAK,UAAU,EAAE,CAAC;YACxB,OAAO,IAAI,cAAc,CAAC,EAAC,OAAO,EAAE,IAAI,EAAE,OAAO,EAAC,CAAC,CAAA;QACrD,CAAC;QAED,IAAI,IAAI,KAAK,QAAQ,EAAE,CAAC;YACtB,OAAO,IAAI,YAAY,CAAC,EAAC,OAAO,EAAE,IAAI,EAAE,OAAO,EAAC,CAAC,CAAA;QACnD,CAAC;QAED,MAAM,IAAI,KAAK,CAAC,4BAA4B,IAAI,EAAE,CAAC,CAAA;IACrD,CAAC;IAED;;;OAGG;IACH,eAAe,CAAC,YAAY;QAC1B,IAAI,CAAC,YAAY,GAAG,YAAY,CAAA;IAClC,CAAC;IAED,yBAAyB;IACzB,kBAAkB;QAChB,OAAO,OAAO,CAAC,IAAI,CAAC,YAAY,CAAC,CAAA;IACnC,CAAC;IAED;;;OAGG;IACH,eAAe,CAAC,YAAY,IAAI,IAAI,CAAC,aAAa,GAAG,YAAY,CAAA,CAAC,CAAC;IAEnE,oCAAoC;IACpC,eAAe,KAAK,OAAO,IAAI,CAAC,aAAa,CAAA,CAAC,CAAC;IAE/C;;;OAGG;IACH,WAAW,CAAC,QAAQ;QAClB,OAAO,IAAI,CAAC,eAAe,EAAE,CAAC,CAAC,CAAC,GAAG,IAAI,CAAC,eAAe,EAAE,IAAI,QAAQ,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAA;IACpF,CAAC;IAED;;;OAGG;IACH,UAAU,CAAC,GAAG,IAAI;QAChB,OAAO,CAAC,KAAK,CAAC,iBAAiB,EAAE,GAAG,IAAI,CAAC,CAAA;IAC3C,CAAC;IAED;;;OAGG;IACH,QAAQ,CAAC,GAAG,IAAI;QACd,IAAI,IAAI,CAAC,MAAM,EAAE,CAAC;YAChB,OAAO,CAAC,GAAG,CAAC,iBAAiB,EAAE,GAAG,IAAI,CAAC,CAAA;QACzC,CAAC;IACH,CAAC;IAED,sBAAsB;IACtB,sBAAsB;QACpB,IAAI,IAAI,CAAC,gBAAgB,EAAE,CAAC;YAC1B,MAAM,IAAI,KAAK,CAAC,sBAAsB,IAAI,CAAC,gBAAgB,CAAC,OAAO,EAAE,CAAC,CAAA;QACxE,CAAC;IACH,CAAC;IAaD,wDAAwD;IACxD,SAAS;QACP,OAAO,IAAI,CAAC,gBAAgB,EAAE,CAAC,YAAY,EAAE,CAAA;IAC/C,CAAC;IAED,iEAAiE;IACjE,gBAAgB;QACd,IAAI,CAAC,IAAI,CAAC,aAAa,EAAE,CAAC;YACxB,MAAM,IAAI,KAAK,CAAC,oCAAoC,CAAC,CAAA;QACvD,CAAC;QAED,OAAO,IAAI,CAAC,aAAa,CAAA;IAC3B,CAAC;IAED,wBAAwB;IACxB,WAAW,KAAK,OAAO,IAAI,CAAC,gBAAgB,EAAE,CAAC,WAAW,EAAE,CAAA,CAAC,CAAC;IAE9D,+BAA+B;IAC/B,KAAK,CAAC,eAAe;QACnB,MAAM,IAAI,CAAC,gBAAgB,EAAE,CAAC,eAAe,EAAE,CAAA;IACjD,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,iBAAiB,CAAC,UAAU;QAChC,MAAM,IAAI,CAAC,gBAAgB,EAAE,CAAC,iBAAiB,CAAC,UAAU,CAAC,CAAA;IAC7D,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,WAAW,CAAC,UAAU;QAC1B,MAAM,IAAI,CAAC,gBAAgB,EAAE,CAAC,WAAW,CAAC,UAAU,CAAC,CAAA;IACvD,CAAC;IAED,mCAAmC;IACnC,KAAK,CAAC,cAAc;QAClB,OAAO,MAAM,IAAI,CAAC,gBAAgB,EAAE,CAAC,cAAc,EAAE,CAAA;IACvD,CAAC;IAED,iCAAiC;IACjC,KAAK,CAAC,aAAa;QACjB,OAAO,MAAM,IAAI,CAAC,gBAAgB,EAAE,CAAC,aAAa,EAAE,CAAA;IACtD,CAAC;IAED;;;;OAIG;IACH,KAAK,CAAC,GAAG,CAAC,QAAQ,EAAE,IAAI,GAAG,EAAE;QAC3B,OAAO,MAAM,IAAI,CAAC,gBAAgB,EAAE,CAAC,GAAG,CAAC,QAAQ,EAAE,IAAI,CAAC,CAAA;IAC1D,CAAC;IAED;;;;OAIG;IACH,KAAK,CAAC,IAAI,CAAC,QAAQ,EAAE,IAAI,GAAG,EAAE;QAC5B,OAAO,MAAM,IAAI,CAAC,gBAAgB,EAAE,CAAC,IAAI,CAAC,QAAQ,EAAE,IAAI,CAAC,CAAA;IAC3D,CAAC;IAED;;;;OAIG;IACH,KAAK,CAAC,YAAY,CAAC,MAAM,EAAE,IAAI;QAC7B,OAAO,MAAM,IAAI,CAAC,gBAAgB,EAAE,CAAC,YAAY,CAAC,MAAM,EAAE,IAAI,CAAC,CAAA;IACjE,CAAC;IAED;;;;OAIG;IACH,KAAK,CAAC,UAAU,CAAC,QAAQ,EAAE,IAAI,GAAG,EAAE;QAClC,OAAO,MAAM,IAAI,CAAC,gBAAgB,EAAE,CAAC,UAAU,CAAC,QAAQ,EAAE,IAAI,CAAC,CAAA;IACjE,CAAC;IAED;;;;OAIG;IACH,KAAK,CAAC,KAAK,CAAC,mBAAmB,EAAE,IAAI;QACnC,MAAM,IAAI,CAAC,gBAAgB,EAAE,CAAC,KAAK,CAAC,mBAAmB,EAAE,IAAI,CAAC,CAAA;IAChE,CAAC;IAED;;;;;OAKG;IACH,KAAK,CAAC,QAAQ,CAAC,mBAAmB,EAAE,UAAU,EAAE,GAAG,IAAI;QACrD,OAAO,MAAM,IAAI,CAAC,gBAAgB,EAAE,CAAC,QAAQ,CAAC,mBAAmB,EAAE,UAAU,EAAE,GAAG,IAAI,CAAC,CAAA;IACzF,CAAC;IAED;;;;OAIG;IACH,KAAK,CAAC,iBAAiB,CAAC,QAAQ,EAAE,IAAI,GAAG,EAAE;QACzC,MAAM,IAAI,CAAC,gBAAgB,EAAE,CAAC,iBAAiB,CAAC,QAAQ,EAAE,IAAI,CAAC,CAAA;IACjE,CAAC;IAED;;;;OAIG;IACH,KAAK,CAAC,eAAe,CAAC,QAAQ,EAAE,IAAI,GAAG,EAAE;QACvC,IAAI,KAAK,GAAG,KAAK,CAAA;QAEjB,IAAI,CAAC;YACH,MAAM,IAAI,CAAC,UAAU,CAAC,QAAQ,EAAE,IAAI,CAAC,CAAA;YACrC,KAAK,GAAG,IAAI,CAAA;QACd,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,IAAI,KAAK,YAAY,KAAK,IAAI,KAAK,CAAC,OAAO,CAAC,UAAU,CAAC,kCAAkC,CAAC,EAAE,CAAC;gBAC3F,SAAS;YACX,CAAC;iBAAM,CAAC;gBACN,MAAM,KAAK,CAAA;YACb,CAAC;QACH,CAAC;QAED,IAAI,KAAK,EAAE,CAAC;YACV,MAAM,IAAI,KAAK,CAAC,yBAAyB,QAAQ,EAAE,CAAC,CAAA;QACtD,CAAC;IACH,CAAC;IAED,iCAAiC;IACjC,KAAK,CAAC,OAAO;QACX,OAAO,MAAM,IAAI,CAAC,gBAAgB,EAAE,CAAC,OAAO,EAAE,CAAA;IAChD,CAAC;IAED;;;OAGG;IACH,iBAAiB,CAAC,eAAe;QAC/B,IAAI,eAAe,KAAK,SAAS,EAAE,CAAC;YAClC,OAAO,eAAe,CAAA;QACxB,CAAC;QAED,OAAO,IAAI,CAAC,WAAW,EAAE,CAAA;IAC3B,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,WAAW,CAAC,IAAI;QACpB,MAAM,IAAI,CAAC,gBAAgB,EAAE,CAAC,WAAW,CAAC,IAAI,CAAC,CAAA;IACjD,CAAC;IAED;;;;;OAKG;IACH,KAAK,CAAC,kBAAkB,CAAC,IAAI,EAAE,IAAI,EAAE,IAAI,GAAG,EAAE;QAC5C,IAAI,CAAC,IAAI,CAAC,YAAY,EAAE,CAAC;YACvB,MAAM,IAAI,KAAK,CAAC,0CAA0C,CAAC,CAAA;QAC7D,CAAC;QAED,MAAM,OAAO,CACX,EAAC,OAAO,EAAE,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,OAAO,CAAC,EAAE,YAAY,EAAE,yCAAyC,IAAI,KAAK,IAAI,EAAE,EAAC,EACvH,KAAK,IAAI,EAAE,CAAC,MAAM,IAAI,CAAC,YAAY,CAAC,WAAW,CAAC,EAAC,IAAI,EAAE,IAAI,EAAC,CAAC,CAC9D,CAAA;IACH,CAAC;IAED;;;;;OAKG;IACH,KAAK,CAAC,KAAK,CAAC,IAAI,EAAE,IAAI,GAAG,EAAE;QACzB,IAAI,IAAI,CAAC,kBAAkB,EAAE,EAAE,CAAC;YAC9B,MAAM,IAAI,CAAC,kBAAkB,CAAC,OAAO,EAAE,IAAI,EAAE,IAAI,CAAC,CAAA;QACpD,CAAC;aAAM,CAAC;YACN,MAAM,OAAO,CACX,EAAC,OAAO,EAAE,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,OAAO,CAAC,EAAE,YAAY,EAAE,gCAAgC,IAAI,EAAE,EAAC,EACrG,KAAK,IAAI,EAAE,CAAC,MAAM,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,CACzC,CAAA;QACH,CAAC;IACH,CAAC;IAED;;;;;OAKG;IACH,KAAK,CAAC,SAAS,CAAC,IAAI,EAAE,IAAI,GAAG,EAAE;QAC7B,IAAI,IAAI,CAAC,kBAAkB,EAAE,EAAE,CAAC;YAC9B,MAAM,IAAI,CAAC,kBAAkB,CAAC,WAAW,EAAE,IAAI,EAAE,IAAI,CAAC,CAAA;QACxD,CAAC;aAAM,CAAC;YACN,MAAM,OAAO,CACX,EAAC,OAAO,EAAE,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,OAAO,CAAC,EAAE,YAAY,EAAE,qCAAqC,IAAI,EAAE,EAAC,EAC1G,KAAK,IAAI,EAAE,CAAC,MAAM,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,CACzC,CAAA;QACH,CAAC;IACH,CAAC;IAED;;;;;OAKG;IACH,2BAA2B,CAAC,IAAI,EAAE,QAAQ,GAAG,GAAG;QAC9C,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,IAAI,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YAC9C,OAAO,CAAC,mBAAmB,CAAC,CAAA;QAC9B,CAAC;QAED,IAAI,IAAI,CAAC,MAAM,IAAI,QAAQ,EAAE,CAAC;YAC5B,OAAO,IAAI,CAAA;QACb,CAAC;QAED,MAAM,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,MAAM,GAAG,QAAQ,CAAC,CAAA;QACnD,MAAM,WAAW,GAAG,IAAI,CAAC,MAAM,GAAG,QAAQ,CAAA;QAE1C,OAAO,CAAC,iBAAiB,QAAQ,OAAO,IAAI,CAAC,MAAM,kBAAkB,WAAW,WAAW,EAAE,GAAG,QAAQ,CAAC,CAAA;IAC3G,CAAC;IAED;;;OAGG;IACH,0BAA0B,CAAC,IAAI;QAC7B,OAAO,CAAC,GAAG,CAAC,eAAe,CAAC,CAAA;QAE5B,KAAK,MAAM,IAAI,IAAI,IAAI,CAAC,2BAA2B,CAAC,IAAI,CAAC,EAAE,CAAC;YAC1D,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,CAAA;QACnB,CAAC;IACH,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,cAAc;QAClB,IAAI,CAAC,QAAQ,CAAC,8BAA8B,CAAC,CAAA;QAC7C,MAAM,IAAI,GAAG,IAAI,CAAC,gBAAgB,CAAA;QAElC,IAAI,CAAC,QAAQ,CAAC,gCAAgC,IAAI,EAAE,CAAC,CAAA;QACrD,MAAM,EAAE,CAAC,KAAK,CAAC,IAAI,EAAE,EAAC,SAAS,EAAE,IAAI,EAAC,CAAC,CAAA;QAEvC,IAAI,CAAC,QAAQ,CAAC,kCAAkC,CAAC,CAAA;QACjD,MAAM,YAAY,GAAG,MAAM,OAAO,CAAC,EAAC,OAAO,EAAE,IAAI,EAAE,YAAY,EAAE,iCAAiC,EAAC,EAAE,KAAK,IAAI,EAAE,CAAC,MAAM,IAAI,CAAC,gBAAgB,EAAE,CAAC,cAAc,EAAE,CAAC,CAAA;QAEhK,IAAI,CAAC,QAAQ,CAAC,2BAA2B,CAAC,CAAA;QAC1C,MAAM,GAAG,GAAG,IAAI,IAAI,EAAE,CAAA;QACtB,MAAM,SAAS,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC,MAAM,CAAC,qBAAqB,CAAC,CAAA;QAC3D,MAAM,cAAc,GAAG,GAAG,IAAI,IAAI,SAAS,MAAM,CAAA;QACjD,MAAM,QAAQ,GAAG,GAAG,IAAI,IAAI,SAAS,OAAO,CAAA;QAC5C,MAAM,QAAQ,GAAG,GAAG,IAAI,IAAI,SAAS,WAAW,CAAA;QAEhD,IAAI,CAAC,QAAQ,CAAC,sBAAsB,CAAC,CAAA;QACrC,MAAM,IAAI,GAAG,MAAM,OAAO,CAAC,EAAC,OAAO,EAAE,IAAI,EAAE,YAAY,EAAE,oCAAoC,EAAC,EAAE,KAAK,IAAI,EAAE,CAAC,MAAM,IAAI,CAAC,cAAc,EAAE,CAAC,CAAA;QACxI,MAAM,IAAI,GAAG,MAAM,OAAO,CAAC,EAAC,OAAO,EAAE,IAAI,EAAE,YAAY,EAAE,iCAAiC,EAAC,EAAE,KAAK,IAAI,EAAE,CAAC,MAAM,IAAI,CAAC,OAAO,EAAE,CAAC,CAAA;QAC9H,MAAM,UAAU,GAAG,QAAQ,CAAC,IAAI,CAAC,CAAA;QACjC,IAAI,CAAC,0BAA0B,CAAC,IAAI,CAAC,CAAA;QAErC,IAAI,CAAC,QAAQ,CAAC,eAAe,CAAC,CAAA;QAC9B,MAAM,EAAE,CAAC,SAAS,CAAC,QAAQ,EAAE,UAAU,CAAC,CAAA;QACxC,MAAM,EAAE,CAAC,SAAS,CAAC,QAAQ,EAAE,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAA;QAC7C,MAAM,EAAE,CAAC,SAAS,CAAC,cAAc,EAAE,YAAY,EAAE,QAAQ,CAAC,CAAA;QAE1D,MAAM,UAAU,GAAG,MAAM,IAAI,CAAC,aAAa,EAAE,CAAA;QAE7C,OAAO,CAAC,GAAG,CAAC,cAAc,EAAE,UAAU,CAAC,CAAA;QACvC,OAAO,CAAC,GAAG,CAAC,OAAO,EAAE,QAAQ,CAAC,CAAA;QAC9B,OAAO,CAAC,GAAG,CAAC,aAAa,EAAE,cAAc,CAAC,CAAA;QAC1C,OAAO,CAAC,GAAG,CAAC,OAAO,EAAE,QAAQ,CAAC,CAAA;QAE9B,OAAO;YACL,UAAU;YACV,IAAI;YACJ,QAAQ;YACR,IAAI;YACJ,QAAQ;YACR,cAAc;SACf,CAAA;IACH,CAAC;IAED,+BAA+B;IAC/B,KAAK,CAAC,UAAU;QACd,IAAI,IAAI,CAAC,aAAa,EAAE,CAAC;YACvB,MAAM,IAAI,CAAC,aAAa,CAAC,IAAI,EAAE,CAAA;QACjC,CAAC;IACH,CAAC;CACF","sourcesContent":["// @ts-check\n\nimport fs from \"node:fs/promises\"\nimport moment from \"moment\"\nimport {prettify} from \"htmlfy\"\nimport timeout from \"awaitery/build/timeout.js\"\nimport SeleniumDriver from \"./drivers/selenium-driver.js\"\nimport AppiumDriver from \"./drivers/appium-driver.js\"\n\n/**\n * @typedef {object} BrowserArgs\n * @property {boolean} [debug] Enable debug logging.\n * @property {BrowserDriverConfig} [driver] Driver configuration.\n * @property {import(\"./system-test-communicator.js\").default} [communicator] Optional command communicator for helper-driven navigation.\n * @property {string} [screenshotsPath] Directory used for saved screenshots and browser artifacts.\n */\n/**\n * @typedef {object} BrowserDriverConfig\n * @property {\"selenium\"|\"appium\"} [type] Driver implementation to use.\n * @property {Record<string, any>} [options] Driver-specific options.\n */\n/**\n * @typedef {object} BrowserNavigationArgs\n * @property {number} [timeout] Override the timeout for this navigation command.\n */\n\n/** Generic browser session wrapper around the configured driver. */\nexport default class Browser {\n  /** @type {import(\"selenium-webdriver\").WebDriver | undefined} */\n  driver = undefined\n\n  /** @type {import(\"./drivers/webdriver-driver.js\").default | undefined} */\n  driverAdapter = undefined\n\n  _debug = false\n  /** @type {BrowserDriverConfig | undefined} */\n  _driverConfig = undefined\n  /** @type {Error | undefined} */\n  _httpServerError = undefined\n\n  /** @param {BrowserArgs} [args] */\n  constructor({debug = false, driver, communicator, screenshotsPath = `${process.cwd()}/tmp/screenshots`, ...restArgs} = {}) {\n    const restArgsKeys = Object.keys(restArgs)\n\n    if (restArgsKeys.length > 0) {\n      throw new Error(`Unknown browser arguments: ${restArgsKeys.join(\", \")}`)\n    }\n\n    this._debug = debug\n    this._driverConfig = driver\n    this._screenshotsPath = screenshotsPath\n    this.communicator = communicator\n    this.driverAdapter = this.createDriver(driver)\n  }\n\n  /**\n   * @param {BrowserDriverConfig} [driverConfig]\n   * @returns {import(\"./drivers/webdriver-driver.js\").default}\n   */\n  createDriver(driverConfig = {}) {\n    const {type = \"selenium\", options, ...restArgs} = driverConfig\n    const restArgsKeys = Object.keys(restArgs)\n\n    if (restArgsKeys.length > 0) {\n      throw new Error(`Unknown driver args: ${restArgsKeys.join(\", \")}`)\n    }\n\n    if (type === \"selenium\") {\n      return new SeleniumDriver({browser: this, options})\n    }\n\n    if (type === \"appium\") {\n      return new AppiumDriver({browser: this, options})\n    }\n\n    throw new Error(`Unsupported driver type: ${type}`)\n  }\n\n  /**\n   * @param {import(\"./system-test-communicator.js\").default | undefined} communicator\n   * @returns {void}\n   */\n  setCommunicator(communicator) {\n    this.communicator = communicator\n  }\n\n  /** @returns {boolean} */\n  communicatorExists() {\n    return Boolean(this.communicator)\n  }\n\n  /**\n   * @param {string} baseSelector\n   * @returns {void}\n   */\n  setBaseSelector(baseSelector) { this._baseSelector = baseSelector }\n\n  /** @returns {string | undefined} */\n  getBaseSelector() { return this._baseSelector }\n\n  /**\n   * @param {string} selector\n   * @returns {string}\n   */\n  getSelector(selector) {\n    return this.getBaseSelector() ? `${this.getBaseSelector()} ${selector}` : selector\n  }\n\n  /**\n   * @param {...any} args\n   * @returns {void}\n   */\n  debugError(...args) {\n    console.error(\"[Browser error]\", ...args)\n  }\n\n  /**\n   * @param {...any} args\n   * @returns {void}\n   */\n  debugLog(...args) {\n    if (this._debug) {\n      console.log(\"[Browser debug]\", ...args)\n    }\n  }\n\n  /** @returns {void} */\n  throwIfHttpServerError() {\n    if (this._httpServerError) {\n      throw new Error(`HTTP server error: ${this._httpServerError.message}`)\n    }\n  }\n\n  /**\n   * @param {Error} error\n   * @returns {void}\n   */\n  onHttpServerError = (error) => {\n    const errorMessage = error instanceof Error ? error.message : String(error)\n\n    this._httpServerError = error instanceof Error ? error : new Error(errorMessage)\n    console.error(`HTTP server error: ${errorMessage}`)\n  }\n\n  /** @returns {import(\"selenium-webdriver\").WebDriver} */\n  getDriver() {\n    return this.getDriverAdapter().getWebDriver()\n  }\n\n  /** @returns {import(\"./drivers/webdriver-driver.js\").default} */\n  getDriverAdapter() {\n    if (!this.driverAdapter) {\n      throw new Error(\"Driver hasn't been initialized yet\")\n    }\n\n    return this.driverAdapter\n  }\n\n  /** @returns {number} */\n  getTimeouts() { return this.getDriverAdapter().getTimeouts() }\n\n  /** @returns {Promise<void>} */\n  async restoreTimeouts() {\n    await this.getDriverAdapter().restoreTimeouts()\n  }\n\n  /**\n   * @param {number} newTimeout\n   * @returns {Promise<void>}\n   */\n  async driverSetTimeouts(newTimeout) {\n    await this.getDriverAdapter().driverSetTimeouts(newTimeout)\n  }\n\n  /**\n   * @param {number} newTimeout\n   * @returns {Promise<void>}\n   */\n  async setTimeouts(newTimeout) {\n    await this.getDriverAdapter().setTimeouts(newTimeout)\n  }\n\n  /** @returns {Promise<string[]>} */\n  async getBrowserLogs() {\n    return await this.getDriverAdapter().getBrowserLogs()\n  }\n\n  /** @returns {Promise<string>} */\n  async getCurrentUrl() {\n    return await this.getDriverAdapter().getCurrentUrl()\n  }\n\n  /**\n   * @param {string} selector\n   * @param {import(\"./system-test.js\").FindArgs} [args]\n   * @returns {Promise<import(\"selenium-webdriver\").WebElement[]>}\n   */\n  async all(selector, args = {}) {\n    return await this.getDriverAdapter().all(selector, args)\n  }\n\n  /**\n   * @param {string} selector\n   * @param {import(\"./system-test.js\").FindArgs} [args]\n   * @returns {Promise<import(\"selenium-webdriver\").WebElement>}\n   */\n  async find(selector, args = {}) {\n    return await this.getDriverAdapter().find(selector, args)\n  }\n\n  /**\n   * @param {string} testID\n   * @param {import(\"./system-test.js\").FindArgs} [args]\n   * @returns {Promise<import(\"selenium-webdriver\").WebElement>}\n   */\n  async findByTestID(testID, args) {\n    return await this.getDriverAdapter().findByTestID(testID, args)\n  }\n\n  /**\n   * @param {string} selector\n   * @param {import(\"./system-test.js\").FindArgs} [args]\n   * @returns {Promise<import(\"selenium-webdriver\").WebElement>}\n   */\n  async findNoWait(selector, args = {}) {\n    return await this.getDriverAdapter().findNoWait(selector, args)\n  }\n\n  /**\n   * @param {string | import(\"selenium-webdriver\").WebElement} elementOrIdentifier\n   * @param {import(\"./system-test.js\").FindArgs} [args]\n   * @returns {Promise<void>}\n   */\n  async click(elementOrIdentifier, args) {\n    await this.getDriverAdapter().click(elementOrIdentifier, args)\n  }\n\n  /**\n   * @param {import(\"selenium-webdriver\").WebElement|string|{selector: string} & import(\"./system-test.js\").FindArgs} elementOrIdentifier\n   * @param {string} methodName\n   * @param {...any} args\n   * @returns {Promise<any>}\n   */\n  async interact(elementOrIdentifier, methodName, ...args) {\n    return await this.getDriverAdapter().interact(elementOrIdentifier, methodName, ...args)\n  }\n\n  /**\n   * @param {string} selector\n   * @param {import(\"./system-test.js\").WaitForNoSelectorArgs} [args]\n   * @returns {Promise<void>}\n   */\n  async waitForNoSelector(selector, args = {}) {\n    await this.getDriverAdapter().waitForNoSelector(selector, args)\n  }\n\n  /**\n   * @param {string} selector\n   * @param {import(\"./system-test.js\").FindArgs} [args]\n   * @returns {Promise<void>}\n   */\n  async expectNoElement(selector, args = {}) {\n    let found = false\n\n    try {\n      await this.findNoWait(selector, args)\n      found = true\n    } catch (error) {\n      if (error instanceof Error && error.message.startsWith(\"Element couldn't be found after \")) {\n        // Ignore\n      } else {\n        throw error\n      }\n    }\n\n    if (found) {\n      throw new Error(`Expected not to find: ${selector}`)\n    }\n  }\n\n  /** @returns {Promise<string>} */\n  async getHTML() {\n    return await this.getDriverAdapter().getHTML()\n  }\n\n  /**\n   * @param {number | undefined} timeoutOverride\n   * @returns {number}\n   */\n  getCommandTimeout(timeoutOverride) {\n    if (timeoutOverride !== undefined) {\n      return timeoutOverride\n    }\n\n    return this.getTimeouts()\n  }\n\n  /**\n   * @param {string} path\n   * @returns {Promise<void>}\n   */\n  async driverVisit(path) {\n    await this.getDriverAdapter().driverVisit(path)\n  }\n\n  /**\n   * @param {string} type\n   * @param {string} path\n   * @param {BrowserNavigationArgs} [args]\n   * @returns {Promise<void>}\n   */\n  async sendBrowserCommand(type, path, args = {}) {\n    if (!this.communicator) {\n      throw new Error(\"Communicator hasn't been initialized yet\")\n    }\n\n    await timeout(\n      {timeout: this.getCommandTimeout(args.timeout), errorMessage: `timeout while sending browser command ${type}: ${path}`},\n      async () => await this.communicator.sendCommand({type, path})\n    )\n  }\n\n  /**\n   * Visits a path using the injected browser helper when available, otherwise navigates directly with the driver.\n   * @param {string} path\n   * @param {BrowserNavigationArgs} [args]\n   * @returns {Promise<void>}\n   */\n  async visit(path, args = {}) {\n    if (this.communicatorExists()) {\n      await this.sendBrowserCommand(\"visit\", path, args)\n    } else {\n      await timeout(\n        {timeout: this.getCommandTimeout(args.timeout), errorMessage: `timeout while visiting path: ${path}`},\n        async () => await this.driverVisit(path)\n      )\n    }\n  }\n\n  /**\n   * Dismisses to a path via the injected browser helper when available, otherwise navigates directly with the driver.\n   * @param {string} path\n   * @param {BrowserNavigationArgs} [args]\n   * @returns {Promise<void>}\n   */\n  async dismissTo(path, args = {}) {\n    if (this.communicatorExists()) {\n      await this.sendBrowserCommand(\"dismissTo\", path, args)\n    } else {\n      await timeout(\n        {timeout: this.getCommandTimeout(args.timeout), errorMessage: `timeout while dismissing to path: ${path}`},\n        async () => await this.driverVisit(path)\n      )\n    }\n  }\n\n  /**\n   * Formats browser logs for console output and truncates overly long output.\n   * @param {string[]} logs\n   * @param {number} [maxLines]\n   * @returns {string[]}\n   */\n  formatBrowserLogsForConsole(logs, maxLines = 200) {\n    if (!Array.isArray(logs) || logs.length === 0) {\n      return [\"(no browser logs)\"]\n    }\n\n    if (logs.length <= maxLines) {\n      return logs\n    }\n\n    const keptLogs = logs.slice(logs.length - maxLines)\n    const hiddenCount = logs.length - maxLines\n\n    return [`(showing last ${maxLines} of ${logs.length} browser logs, ${hiddenCount} omitted)`, ...keptLogs]\n  }\n\n  /**\n   * @param {string[]} logs\n   * @returns {void}\n   */\n  printBrowserLogsForFailure(logs) {\n    console.log(\"Browser logs:\")\n\n    for (const line of this.formatBrowserLogsForConsole(logs)) {\n      console.log(line)\n    }\n  }\n\n  /**\n   * Takes a screenshot, writes HTML/browser logs to disk, and returns the collected artifacts.\n   * @returns {Promise<{currentUrl: string, html: string, htmlPath: string, logs: string[], logsPath: string, screenshotPath: string}>}\n   */\n  async takeScreenshot() {\n    this.debugLog(\"Getting path for screenshots\")\n    const path = this._screenshotsPath\n\n    this.debugLog(`Creating dir with recursive: ${path}`)\n    await fs.mkdir(path, {recursive: true})\n\n    this.debugLog(\"Getting screenshot image content\")\n    const imageContent = await timeout({timeout: 5000, errorMessage: \"timeout while taking screenshot\"}, async () => await this.getDriverAdapter().takeScreenshot())\n\n    this.debugLog(\"Generating date variables\")\n    const now = new Date()\n    const timestamp = moment(now).format(\"YYYY-MM-DD-HH-MM-SS\")\n    const screenshotPath = `${path}/${timestamp}.png`\n    const htmlPath = `${path}/${timestamp}.html`\n    const logsPath = `${path}/${timestamp}.logs.txt`\n\n    this.debugLog(\"Getting browser logs\")\n    const logs = await timeout({timeout: 5000, errorMessage: \"timeout while reading browser logs\"}, async () => await this.getBrowserLogs())\n    const html = await timeout({timeout: 5000, errorMessage: \"timeout while reading page HTML\"}, async () => await this.getHTML())\n    const htmlPretty = prettify(html)\n    this.printBrowserLogsForFailure(logs)\n\n    this.debugLog(\"Writing files\")\n    await fs.writeFile(htmlPath, htmlPretty)\n    await fs.writeFile(logsPath, logs.join(\"\\n\"))\n    await fs.writeFile(screenshotPath, imageContent, \"base64\")\n\n    const currentUrl = await this.getCurrentUrl()\n\n    console.log(\"Current URL:\", currentUrl)\n    console.log(\"Logs:\", logsPath)\n    console.log(\"Screenshot:\", screenshotPath)\n    console.log(\"HTML:\", htmlPath)\n\n    return {\n      currentUrl,\n      html,\n      htmlPath,\n      logs,\n      logsPath,\n      screenshotPath\n    }\n  }\n\n  /** @returns {Promise<void>} */\n  async stopDriver() {\n    if (this.driverAdapter) {\n      await this.driverAdapter.stop()\n    }\n  }\n}\n"]}
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @param {string[]} argv
|
|
3
|
+
* @returns {{_: string[], flags: Record<string, any>}}
|
|
4
|
+
*/
|
|
5
|
+
export function parseArgv(argv: string[]): {
|
|
6
|
+
_: string[];
|
|
7
|
+
flags: Record<string, any>;
|
|
8
|
+
};
|
|
9
|
+
/**
|
|
10
|
+
* @param {Record<string, any>} flags
|
|
11
|
+
* @returns {{command: string, args: Record<string, any>}}
|
|
12
|
+
*/
|
|
13
|
+
export function resolveBrowserCommand(flags: Record<string, any>): {
|
|
14
|
+
command: string;
|
|
15
|
+
args: Record<string, any>;
|
|
16
|
+
};
|
|
@@ -0,0 +1,177 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @param {string[]} argv
|
|
3
|
+
* @returns {{_: string[], flags: Record<string, any>}}
|
|
4
|
+
*/
|
|
5
|
+
export function parseArgv(argv) {
|
|
6
|
+
const result = { _: [], flags: {} };
|
|
7
|
+
const setFlag = (key, value) => {
|
|
8
|
+
if (!(key in result.flags)) {
|
|
9
|
+
result.flags[key] = value;
|
|
10
|
+
return;
|
|
11
|
+
}
|
|
12
|
+
if (!Array.isArray(result.flags[key])) {
|
|
13
|
+
result.flags[key] = [result.flags[key]];
|
|
14
|
+
}
|
|
15
|
+
result.flags[key].push(value);
|
|
16
|
+
};
|
|
17
|
+
for (let index = 0; index < argv.length; index++) {
|
|
18
|
+
const value = argv[index];
|
|
19
|
+
if (!value.startsWith("--")) {
|
|
20
|
+
result._.push(value);
|
|
21
|
+
continue;
|
|
22
|
+
}
|
|
23
|
+
const flag = value.slice(2);
|
|
24
|
+
if (flag.includes("=")) {
|
|
25
|
+
const [key, ...rest] = flag.split("=");
|
|
26
|
+
setFlag(key, rest.join("="));
|
|
27
|
+
continue;
|
|
28
|
+
}
|
|
29
|
+
const nextValue = argv[index + 1];
|
|
30
|
+
if (nextValue && !nextValue.startsWith("--")) {
|
|
31
|
+
setFlag(flag, nextValue);
|
|
32
|
+
index += 1;
|
|
33
|
+
}
|
|
34
|
+
else {
|
|
35
|
+
setFlag(flag, true);
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
return result;
|
|
39
|
+
}
|
|
40
|
+
/**
|
|
41
|
+
* Parses a CLI timeout flag into milliseconds.
|
|
42
|
+
* Bare numeric values are treated as seconds for CLI ergonomics.
|
|
43
|
+
* @param {any} timeoutFlag
|
|
44
|
+
* @returns {number | undefined}
|
|
45
|
+
*/
|
|
46
|
+
function resolveCliTimeout(timeoutFlag) {
|
|
47
|
+
if (timeoutFlag === undefined) {
|
|
48
|
+
return undefined;
|
|
49
|
+
}
|
|
50
|
+
if (typeof timeoutFlag === "number") {
|
|
51
|
+
return timeoutFlag * 1000;
|
|
52
|
+
}
|
|
53
|
+
const timeoutString = String(timeoutFlag).trim();
|
|
54
|
+
if (/^\d+(\.\d+)?ms$/i.test(timeoutString)) {
|
|
55
|
+
return Number(timeoutString.slice(0, -2));
|
|
56
|
+
}
|
|
57
|
+
if (/^\d+(\.\d+)?s$/i.test(timeoutString)) {
|
|
58
|
+
return Number(timeoutString.slice(0, -1)) * 1000;
|
|
59
|
+
}
|
|
60
|
+
if (/^\d+(\.\d+)?$/.test(timeoutString)) {
|
|
61
|
+
return Number(timeoutString) * 1000;
|
|
62
|
+
}
|
|
63
|
+
throw new Error(`Invalid timeout flag: ${timeoutFlag}`);
|
|
64
|
+
}
|
|
65
|
+
/**
|
|
66
|
+
* @param {Record<string, any>} flags
|
|
67
|
+
* @returns {{command: string, args: Record<string, any>}}
|
|
68
|
+
*/
|
|
69
|
+
export function resolveBrowserCommand(flags) {
|
|
70
|
+
const timeout = resolveCliTimeout(flags.timeout);
|
|
71
|
+
if (flags.visit) {
|
|
72
|
+
const args = { url: flags.visit };
|
|
73
|
+
if (timeout !== undefined) {
|
|
74
|
+
args.timeout = timeout;
|
|
75
|
+
}
|
|
76
|
+
return { args, command: "visit" };
|
|
77
|
+
}
|
|
78
|
+
if (flags["dismiss-to"]) {
|
|
79
|
+
const args = { path: flags["dismiss-to"] };
|
|
80
|
+
if (timeout !== undefined) {
|
|
81
|
+
args.timeout = timeout;
|
|
82
|
+
}
|
|
83
|
+
return { args, command: "dismissTo" };
|
|
84
|
+
}
|
|
85
|
+
if (flags["find-by-test-id"]) {
|
|
86
|
+
return {
|
|
87
|
+
args: {
|
|
88
|
+
testID: flags["find-by-test-id"],
|
|
89
|
+
timeout,
|
|
90
|
+
useBaseSelector: flags["use-base-selector"],
|
|
91
|
+
visible: flags.visible
|
|
92
|
+
},
|
|
93
|
+
command: "findByTestID"
|
|
94
|
+
};
|
|
95
|
+
}
|
|
96
|
+
if (flags.find) {
|
|
97
|
+
return {
|
|
98
|
+
args: {
|
|
99
|
+
selector: flags.find,
|
|
100
|
+
timeout,
|
|
101
|
+
useBaseSelector: flags["use-base-selector"],
|
|
102
|
+
visible: flags.visible
|
|
103
|
+
},
|
|
104
|
+
command: "find"
|
|
105
|
+
};
|
|
106
|
+
}
|
|
107
|
+
if (flags.click) {
|
|
108
|
+
return {
|
|
109
|
+
args: {
|
|
110
|
+
selector: flags.click,
|
|
111
|
+
timeout,
|
|
112
|
+
useBaseSelector: flags["use-base-selector"],
|
|
113
|
+
visible: flags.visible
|
|
114
|
+
},
|
|
115
|
+
command: "click"
|
|
116
|
+
};
|
|
117
|
+
}
|
|
118
|
+
if (flags["wait-for-no-selector"]) {
|
|
119
|
+
return {
|
|
120
|
+
args: {
|
|
121
|
+
selector: flags["wait-for-no-selector"],
|
|
122
|
+
timeout,
|
|
123
|
+
useBaseSelector: flags["use-base-selector"]
|
|
124
|
+
},
|
|
125
|
+
command: "waitForNoSelector"
|
|
126
|
+
};
|
|
127
|
+
}
|
|
128
|
+
if (flags["expect-no-element"]) {
|
|
129
|
+
return {
|
|
130
|
+
args: {
|
|
131
|
+
selector: flags["expect-no-element"],
|
|
132
|
+
timeout,
|
|
133
|
+
useBaseSelector: flags["use-base-selector"]
|
|
134
|
+
},
|
|
135
|
+
command: "expectNoElement"
|
|
136
|
+
};
|
|
137
|
+
}
|
|
138
|
+
if (flags["set-base-selector"]) {
|
|
139
|
+
return { args: { selector: flags["set-base-selector"] }, command: "setBaseSelector" };
|
|
140
|
+
}
|
|
141
|
+
if (flags["get-html"]) {
|
|
142
|
+
return { args: {}, command: "getHTML" };
|
|
143
|
+
}
|
|
144
|
+
if (flags["get-browser-logs"]) {
|
|
145
|
+
return { args: {}, command: "getBrowserLogs" };
|
|
146
|
+
}
|
|
147
|
+
if (flags["get-current-url"]) {
|
|
148
|
+
return { args: {}, command: "getCurrentUrl" };
|
|
149
|
+
}
|
|
150
|
+
if (flags["take-screenshot"]) {
|
|
151
|
+
return { args: {}, command: "takeScreenshot" };
|
|
152
|
+
}
|
|
153
|
+
if (flags.command) {
|
|
154
|
+
const args = {};
|
|
155
|
+
if (flags.url)
|
|
156
|
+
args.url = flags.url;
|
|
157
|
+
if (flags.path)
|
|
158
|
+
args.path = flags.path;
|
|
159
|
+
if (flags.selector)
|
|
160
|
+
args.selector = flags.selector;
|
|
161
|
+
if (flags["test-id"])
|
|
162
|
+
args.testID = flags["test-id"];
|
|
163
|
+
if (flags.method)
|
|
164
|
+
args.methodName = flags.method;
|
|
165
|
+
if (flags.arg)
|
|
166
|
+
args.args = Array.isArray(flags.arg) ? flags.arg : [flags.arg];
|
|
167
|
+
if (timeout !== undefined)
|
|
168
|
+
args.timeout = timeout;
|
|
169
|
+
if (flags.visible !== undefined)
|
|
170
|
+
args.visible = flags.visible;
|
|
171
|
+
if (flags["use-base-selector"] !== undefined)
|
|
172
|
+
args.useBaseSelector = flags["use-base-selector"];
|
|
173
|
+
return { args, command: flags.command };
|
|
174
|
+
}
|
|
175
|
+
throw new Error("No browser command was given");
|
|
176
|
+
}
|
|
177
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cli-helpers.js","sourceRoot":"","sources":["../src/cli-helpers.js"],"names":[],"mappings":"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","sourcesContent":["/**\n * @param {string[]} argv\n * @returns {{_: string[], flags: Record<string, any>}}\n */\nexport function parseArgv(argv) {\n  const result = {_: [], flags: {}}\n  const setFlag = (key, value) => {\n    if (!(key in result.flags)) {\n      result.flags[key] = value\n      return\n    }\n\n    if (!Array.isArray(result.flags[key])) {\n      result.flags[key] = [result.flags[key]]\n    }\n\n    result.flags[key].push(value)\n  }\n\n  for (let index = 0; index < argv.length; index++) {\n    const value = argv[index]\n\n    if (!value.startsWith(\"--\")) {\n      result._.push(value)\n      continue\n    }\n\n    const flag = value.slice(2)\n\n    if (flag.includes(\"=\")) {\n      const [key, ...rest] = flag.split(\"=\")\n\n      setFlag(key, rest.join(\"=\"))\n      continue\n    }\n\n    const nextValue = argv[index + 1]\n\n    if (nextValue && !nextValue.startsWith(\"--\")) {\n      setFlag(flag, nextValue)\n      index += 1\n    } else {\n      setFlag(flag, true)\n    }\n  }\n\n  return result\n}\n\n/**\n * Parses a CLI timeout flag into milliseconds.\n * Bare numeric values are treated as seconds for CLI ergonomics.\n * @param {any} timeoutFlag\n * @returns {number | undefined}\n */\nfunction resolveCliTimeout(timeoutFlag) {\n  if (timeoutFlag === undefined) {\n    return undefined\n  }\n\n  if (typeof timeoutFlag === \"number\") {\n    return timeoutFlag * 1000\n  }\n\n  const timeoutString = String(timeoutFlag).trim()\n\n  if (/^\\d+(\\.\\d+)?ms$/i.test(timeoutString)) {\n    return Number(timeoutString.slice(0, -2))\n  }\n\n  if (/^\\d+(\\.\\d+)?s$/i.test(timeoutString)) {\n    return Number(timeoutString.slice(0, -1)) * 1000\n  }\n\n  if (/^\\d+(\\.\\d+)?$/.test(timeoutString)) {\n    return Number(timeoutString) * 1000\n  }\n\n  throw new Error(`Invalid timeout flag: ${timeoutFlag}`)\n}\n\n/**\n * @param {Record<string, any>} flags\n * @returns {{command: string, args: Record<string, any>}}\n */\nexport function resolveBrowserCommand(flags) {\n  const timeout = resolveCliTimeout(flags.timeout)\n\n  if (flags.visit) {\n    const args = {url: flags.visit}\n\n    if (timeout !== undefined) {\n      args.timeout = timeout\n    }\n\n    return {args, command: \"visit\"}\n  }\n\n  if (flags[\"dismiss-to\"]) {\n    const args = {path: flags[\"dismiss-to\"]}\n\n    if (timeout !== undefined) {\n      args.timeout = timeout\n    }\n\n    return {args, command: \"dismissTo\"}\n  }\n\n  if (flags[\"find-by-test-id\"]) {\n    return {\n      args: {\n        testID: flags[\"find-by-test-id\"],\n        timeout,\n        useBaseSelector: flags[\"use-base-selector\"],\n        visible: flags.visible\n      },\n      command: \"findByTestID\"\n    }\n  }\n\n  if (flags.find) {\n    return {\n      args: {\n        selector: flags.find,\n        timeout,\n        useBaseSelector: flags[\"use-base-selector\"],\n        visible: flags.visible\n      },\n      command: \"find\"\n    }\n  }\n\n  if (flags.click) {\n    return {\n      args: {\n        selector: flags.click,\n        timeout,\n        useBaseSelector: flags[\"use-base-selector\"],\n        visible: flags.visible\n      },\n      command: \"click\"\n    }\n  }\n\n  if (flags[\"wait-for-no-selector\"]) {\n    return {\n      args: {\n        selector: flags[\"wait-for-no-selector\"],\n        timeout,\n        useBaseSelector: flags[\"use-base-selector\"]\n      },\n      command: \"waitForNoSelector\"\n    }\n  }\n\n  if (flags[\"expect-no-element\"]) {\n    return {\n      args: {\n        selector: flags[\"expect-no-element\"],\n        timeout,\n        useBaseSelector: flags[\"use-base-selector\"]\n      },\n      command: \"expectNoElement\"\n    }\n  }\n\n  if (flags[\"set-base-selector\"]) {\n    return {args: {selector: flags[\"set-base-selector\"]}, command: \"setBaseSelector\"}\n  }\n\n  if (flags[\"get-html\"]) {\n    return {args: {}, command: \"getHTML\"}\n  }\n\n  if (flags[\"get-browser-logs\"]) {\n    return {args: {}, command: \"getBrowserLogs\"}\n  }\n\n  if (flags[\"get-current-url\"]) {\n    return {args: {}, command: \"getCurrentUrl\"}\n  }\n\n  if (flags[\"take-screenshot\"]) {\n    return {args: {}, command: \"takeScreenshot\"}\n  }\n\n  if (flags.command) {\n    const args = {}\n\n    if (flags.url) args.url = flags.url\n    if (flags.path) args.path = flags.path\n    if (flags.selector) args.selector = flags.selector\n    if (flags[\"test-id\"]) args.testID = flags[\"test-id\"]\n    if (flags.method) args.methodName = flags.method\n    if (flags.arg) args.args = Array.isArray(flags.arg) ? flags.arg : [flags.arg]\n    if (timeout !== undefined) args.timeout = timeout\n    if (flags.visible !== undefined) args.visible = flags.visible\n    if (flags[\"use-base-selector\"] !== undefined) args.useBaseSelector = flags[\"use-base-selector\"]\n\n    return {args, command: flags.command}\n  }\n\n  throw new Error(\"No browser command was given\")\n}\n"]}
|
package/build/cli.d.ts
ADDED