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.
Files changed (35) hide show
  1. package/README.md +232 -6
  2. package/build/browser-command-client.d.ts +19 -0
  3. package/build/browser-command-client.js +39 -0
  4. package/build/browser-command-runner.d.ts +34 -0
  5. package/build/browser-command-runner.js +155 -0
  6. package/build/browser-daemon-constants.d.ts +2 -0
  7. package/build/browser-daemon-constants.js +3 -0
  8. package/build/browser-process.d.ts +45 -0
  9. package/build/browser-process.js +134 -0
  10. package/build/browser-registry.d.ts +44 -0
  11. package/build/browser-registry.js +191 -0
  12. package/build/browser.d.ts +240 -0
  13. package/build/browser.js +375 -0
  14. package/build/cli-helpers.d.ts +16 -0
  15. package/build/cli-helpers.js +177 -0
  16. package/build/cli.d.ts +2 -0
  17. package/build/cli.js +81 -0
  18. package/build/drivers/appium-driver.js +21 -21
  19. package/build/drivers/webdriver-driver.d.ts +4 -4
  20. package/build/drivers/webdriver-driver.js +32 -28
  21. package/build/index.d.ts +9 -1
  22. package/build/index.js +10 -3
  23. package/build/system-test-browser-helper.d.ts +6 -12
  24. package/build/system-test-browser-helper.js +12 -13
  25. package/build/system-test.d.ts +3 -189
  26. package/build/system-test.js +6 -220
  27. package/build/use-system-test-expo.d.ts +16 -0
  28. package/build/use-system-test-expo.js +34 -0
  29. package/build/use-system-test-react-native.d.ts +25 -0
  30. package/build/use-system-test-react-native.js +20 -0
  31. package/build/use-system-test-shape-hook.d.ts +35 -0
  32. package/build/use-system-test-shape-hook.js +74 -0
  33. package/build/use-system-test.d.ts +19 -10
  34. package/build/use-system-test.js +26 -72
  35. package/package.json +17 -8
@@ -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":"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","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
@@ -0,0 +1,2 @@
1
+ #!/usr/bin/env node
2
+ export {};