system-testing 1.0.73 → 1.0.74
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 +51 -1
- package/build/drivers/appium-driver.d.ts +97 -0
- package/build/drivers/appium-driver.js +316 -0
- package/build/drivers/selenium-driver.d.ts +36 -0
- package/build/drivers/selenium-driver.js +44 -0
- package/build/drivers/webdriver-driver.d.ts +173 -0
- package/build/drivers/webdriver-driver.js +462 -0
- package/build/system-test-browser-helper.d.ts +4 -0
- package/build/system-test-browser-helper.js +28 -3
- package/build/system-test.d.ts +80 -10
- package/build/system-test.js +148 -299
- package/build/use-system-test.js +6 -1
- package/package.json +10 -3
|
@@ -0,0 +1,462 @@
|
|
|
1
|
+
import { By, error as SeleniumError } from "selenium-webdriver";
|
|
2
|
+
import logging from "selenium-webdriver/lib/logging.js";
|
|
3
|
+
import { wait } from "awaitery";
|
|
4
|
+
import timeout from "awaitery/build/timeout.js";
|
|
5
|
+
/**
|
|
6
|
+
* @typedef {object} FindArgs
|
|
7
|
+
* @property {number} [timeout] Override timeout for lookup.
|
|
8
|
+
* @property {boolean} [visible] Whether to require elements to be visible.
|
|
9
|
+
* @property {boolean} [useBaseSelector] Whether to scope by the base selector.
|
|
10
|
+
*/
|
|
11
|
+
/**
|
|
12
|
+
* @typedef {object} WaitForNoSelectorArgs
|
|
13
|
+
* @property {boolean} [useBaseSelector] Whether to scope by the base selector.
|
|
14
|
+
*/
|
|
15
|
+
class ElementNotFoundError extends Error {
|
|
16
|
+
}
|
|
17
|
+
const { WebDriverError } = SeleniumError;
|
|
18
|
+
/**
|
|
19
|
+
* Base driver using selenium-webdriver sessions.
|
|
20
|
+
*/
|
|
21
|
+
export default class WebDriverDriver {
|
|
22
|
+
/**
|
|
23
|
+
* @param {object} args
|
|
24
|
+
* @param {import("../system-test.js").default} args.systemTest
|
|
25
|
+
* @param {Record<string, any>} [args.options]
|
|
26
|
+
*/
|
|
27
|
+
constructor({ systemTest, options = {} }) {
|
|
28
|
+
this.systemTest = systemTest;
|
|
29
|
+
this.options = options;
|
|
30
|
+
this.baseUrl = undefined;
|
|
31
|
+
this.webDriver = undefined;
|
|
32
|
+
this._driverTimeouts = 5000;
|
|
33
|
+
this._timeouts = 5000;
|
|
34
|
+
}
|
|
35
|
+
/**
|
|
36
|
+
* @param {string} baseUrl
|
|
37
|
+
* @returns {void}
|
|
38
|
+
*/
|
|
39
|
+
setBaseUrl(baseUrl) {
|
|
40
|
+
this.baseUrl = baseUrl;
|
|
41
|
+
}
|
|
42
|
+
/** @returns {string} */
|
|
43
|
+
getBaseUrl() {
|
|
44
|
+
if (!this.baseUrl) {
|
|
45
|
+
throw new Error("Driver base URL has not been set");
|
|
46
|
+
}
|
|
47
|
+
return this.baseUrl;
|
|
48
|
+
}
|
|
49
|
+
/** @returns {number} */
|
|
50
|
+
getTimeouts() { return this._timeouts; }
|
|
51
|
+
/**
|
|
52
|
+
* @returns {import("selenium-webdriver").WebDriver}
|
|
53
|
+
*/
|
|
54
|
+
getWebDriver() {
|
|
55
|
+
if (!this.webDriver)
|
|
56
|
+
throw new Error("Driver hasn't been initialized yet");
|
|
57
|
+
this.systemTest.throwIfHttpServerError();
|
|
58
|
+
return this.webDriver;
|
|
59
|
+
}
|
|
60
|
+
/**
|
|
61
|
+
* @param {import("selenium-webdriver").WebDriver} webDriver
|
|
62
|
+
* @returns {void}
|
|
63
|
+
*/
|
|
64
|
+
setWebDriver(webDriver) {
|
|
65
|
+
this.webDriver = webDriver;
|
|
66
|
+
this.systemTest.driver = webDriver;
|
|
67
|
+
}
|
|
68
|
+
/**
|
|
69
|
+
* @returns {Promise<void>}
|
|
70
|
+
*/
|
|
71
|
+
async start() {
|
|
72
|
+
throw new Error("start() must be implemented by the driver");
|
|
73
|
+
}
|
|
74
|
+
/**
|
|
75
|
+
* @returns {Promise<void>}
|
|
76
|
+
*/
|
|
77
|
+
async stop() {
|
|
78
|
+
if (this.webDriver) {
|
|
79
|
+
await timeout({ timeout: this.getTimeouts(), errorMessage: "timeout while quitting WebDriver" }, async () => await this.webDriver.quit());
|
|
80
|
+
}
|
|
81
|
+
this.webDriver = undefined;
|
|
82
|
+
this.systemTest.driver = undefined;
|
|
83
|
+
}
|
|
84
|
+
/**
|
|
85
|
+
* @param {number} newTimeout
|
|
86
|
+
* @returns {Promise<void>}
|
|
87
|
+
*/
|
|
88
|
+
async driverSetTimeouts(newTimeout) {
|
|
89
|
+
this._driverTimeouts = newTimeout;
|
|
90
|
+
await this.getWebDriver().manage().setTimeouts({ implicit: newTimeout });
|
|
91
|
+
}
|
|
92
|
+
/**
|
|
93
|
+
* @returns {Promise<void>}
|
|
94
|
+
*/
|
|
95
|
+
async restoreTimeouts() {
|
|
96
|
+
if (!this.getTimeouts()) {
|
|
97
|
+
throw new Error("Timeouts haven't previously been set");
|
|
98
|
+
}
|
|
99
|
+
await this.driverSetTimeouts(this.getTimeouts());
|
|
100
|
+
}
|
|
101
|
+
/**
|
|
102
|
+
* @param {number} newTimeout
|
|
103
|
+
* @returns {Promise<void>}
|
|
104
|
+
*/
|
|
105
|
+
async setTimeouts(newTimeout) {
|
|
106
|
+
this._timeouts = newTimeout;
|
|
107
|
+
await this.restoreTimeouts();
|
|
108
|
+
}
|
|
109
|
+
/**
|
|
110
|
+
* @param {string} selector
|
|
111
|
+
* @returns {string}
|
|
112
|
+
*/
|
|
113
|
+
getSelector(selector) {
|
|
114
|
+
return this.systemTest.getSelector(selector);
|
|
115
|
+
}
|
|
116
|
+
/**
|
|
117
|
+
* @param {string} path
|
|
118
|
+
* @returns {Promise<void>}
|
|
119
|
+
*/
|
|
120
|
+
async driverVisit(path) {
|
|
121
|
+
const url = `${this.getBaseUrl()}${path}`;
|
|
122
|
+
await this.getWebDriver().get(url);
|
|
123
|
+
}
|
|
124
|
+
/** @returns {Promise<string>} */
|
|
125
|
+
async getCurrentUrl() {
|
|
126
|
+
try {
|
|
127
|
+
return await this.getWebDriver().getCurrentUrl();
|
|
128
|
+
}
|
|
129
|
+
catch {
|
|
130
|
+
return "";
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
/** @returns {Promise<string>} */
|
|
134
|
+
async getHTML() {
|
|
135
|
+
return await this.getWebDriver().getPageSource();
|
|
136
|
+
}
|
|
137
|
+
/** @returns {Promise<string>} */
|
|
138
|
+
async takeScreenshot() {
|
|
139
|
+
return await this.getWebDriver().takeScreenshot();
|
|
140
|
+
}
|
|
141
|
+
/**
|
|
142
|
+
* Gets browser logs
|
|
143
|
+
* @returns {Promise<string[]>}
|
|
144
|
+
*/
|
|
145
|
+
async getBrowserLogs() {
|
|
146
|
+
let entries;
|
|
147
|
+
try {
|
|
148
|
+
entries = await this.getWebDriver().manage().logs().get(logging.Type.BROWSER);
|
|
149
|
+
}
|
|
150
|
+
catch {
|
|
151
|
+
return [];
|
|
152
|
+
}
|
|
153
|
+
const browserLogs = [];
|
|
154
|
+
for (const entry of entries) {
|
|
155
|
+
const messageMatch = entry.message.match(/^(.+) (\d+):(\d+) (.+)$/);
|
|
156
|
+
let message;
|
|
157
|
+
if (messageMatch) {
|
|
158
|
+
message = messageMatch[4];
|
|
159
|
+
}
|
|
160
|
+
else {
|
|
161
|
+
message = entry.message;
|
|
162
|
+
}
|
|
163
|
+
browserLogs.push(`${entry.level.name}: ${message}`);
|
|
164
|
+
}
|
|
165
|
+
return browserLogs;
|
|
166
|
+
}
|
|
167
|
+
/**
|
|
168
|
+
* Finds all elements by CSS selector
|
|
169
|
+
* @param {string} selector
|
|
170
|
+
* @param {FindArgs} [args]
|
|
171
|
+
* @returns {Promise<import("selenium-webdriver").WebElement[]>}
|
|
172
|
+
*/
|
|
173
|
+
async all(selector, args = {}) {
|
|
174
|
+
const { visible = true, timeout, useBaseSelector = true, ...restArgs } = args;
|
|
175
|
+
const restArgsKeys = Object.keys(restArgs);
|
|
176
|
+
let actualTimeout;
|
|
177
|
+
if (timeout === undefined) {
|
|
178
|
+
actualTimeout = this._driverTimeouts;
|
|
179
|
+
}
|
|
180
|
+
else {
|
|
181
|
+
actualTimeout = timeout;
|
|
182
|
+
}
|
|
183
|
+
if (restArgsKeys.length > 0)
|
|
184
|
+
throw new Error(`Unknown arguments: ${restArgsKeys.join(", ")}`);
|
|
185
|
+
const actualSelector = useBaseSelector ? this.getSelector(selector) : selector;
|
|
186
|
+
const startTime = Date.now();
|
|
187
|
+
const getTimeLeft = () => Math.max(actualTimeout - (Date.now() - startTime), 0);
|
|
188
|
+
const getElements = async () => {
|
|
189
|
+
const foundElements = await this.getWebDriver().findElements(By.css(actualSelector));
|
|
190
|
+
if (visible !== true && visible !== false) {
|
|
191
|
+
return foundElements;
|
|
192
|
+
}
|
|
193
|
+
const filteredElements = [];
|
|
194
|
+
for (const element of foundElements) {
|
|
195
|
+
const isDisplayed = await element.isDisplayed();
|
|
196
|
+
if (visible && !isDisplayed)
|
|
197
|
+
continue;
|
|
198
|
+
if (!visible && isDisplayed)
|
|
199
|
+
continue;
|
|
200
|
+
filteredElements.push(element);
|
|
201
|
+
}
|
|
202
|
+
return filteredElements;
|
|
203
|
+
};
|
|
204
|
+
let elements = [];
|
|
205
|
+
while (true) {
|
|
206
|
+
const timeLeft = actualTimeout == 0 ? 0 : getTimeLeft();
|
|
207
|
+
try {
|
|
208
|
+
if (timeLeft == 0) {
|
|
209
|
+
elements = await getElements();
|
|
210
|
+
}
|
|
211
|
+
else {
|
|
212
|
+
await this.getWebDriver().wait(async () => {
|
|
213
|
+
elements = await getElements();
|
|
214
|
+
return elements.length > 0;
|
|
215
|
+
}, timeLeft);
|
|
216
|
+
}
|
|
217
|
+
break;
|
|
218
|
+
}
|
|
219
|
+
catch (error) {
|
|
220
|
+
if (error instanceof SeleniumError.TimeoutError && getTimeLeft() > 0) {
|
|
221
|
+
continue;
|
|
222
|
+
}
|
|
223
|
+
throw new Error(`Couldn't get elements with selector: ${actualSelector}: ${error instanceof Error ? error.message : error}`);
|
|
224
|
+
}
|
|
225
|
+
}
|
|
226
|
+
return elements;
|
|
227
|
+
}
|
|
228
|
+
/**
|
|
229
|
+
* Finds a single element by CSS selector
|
|
230
|
+
* @param {string} selector
|
|
231
|
+
* @param {FindArgs} [args]
|
|
232
|
+
* @returns {Promise<import("selenium-webdriver").WebElement>}
|
|
233
|
+
*/
|
|
234
|
+
async find(selector, args = {}) {
|
|
235
|
+
const startTime = Date.now();
|
|
236
|
+
let elements = [];
|
|
237
|
+
try {
|
|
238
|
+
elements = await this.all(selector, args);
|
|
239
|
+
}
|
|
240
|
+
catch (error) {
|
|
241
|
+
// Re-throw to recover stack trace
|
|
242
|
+
if (error instanceof Error) {
|
|
243
|
+
if (error.message.startsWith("Wait timed out after")) {
|
|
244
|
+
elements = [];
|
|
245
|
+
}
|
|
246
|
+
throw new Error(`${error.constructor.name} - ${error.message} (selector: ${this.getSelector(selector)})`);
|
|
247
|
+
}
|
|
248
|
+
else {
|
|
249
|
+
throw new Error(`${typeof error} - ${error} (selector: ${this.getSelector(selector)})`);
|
|
250
|
+
}
|
|
251
|
+
}
|
|
252
|
+
if (elements.length > 1) {
|
|
253
|
+
throw new Error(`More than 1 elements (${elements.length}) was found by CSS: ${this.getSelector(selector)}`);
|
|
254
|
+
}
|
|
255
|
+
if (!elements[0]) {
|
|
256
|
+
const elapsedSeconds = (Date.now() - startTime) / 1000;
|
|
257
|
+
throw new ElementNotFoundError(`Element couldn't be found after ${elapsedSeconds.toFixed(2)}s by CSS: ${this.getSelector(selector)}`);
|
|
258
|
+
}
|
|
259
|
+
return elements[0];
|
|
260
|
+
}
|
|
261
|
+
/**
|
|
262
|
+
* Finds a single element by test ID
|
|
263
|
+
* @param {string} testId
|
|
264
|
+
* @param {FindArgs} [args]
|
|
265
|
+
* @returns {Promise<import("selenium-webdriver").WebElement>}
|
|
266
|
+
*/
|
|
267
|
+
async findByTestId(testId, args) {
|
|
268
|
+
return await this.find(`[data-testid='${testId}']`, args);
|
|
269
|
+
}
|
|
270
|
+
/**
|
|
271
|
+
* Finds a single element by test ID
|
|
272
|
+
* @param {string} testID
|
|
273
|
+
* @param {FindArgs} [args]
|
|
274
|
+
* @returns {Promise<import("selenium-webdriver").WebElement>}
|
|
275
|
+
*/
|
|
276
|
+
async findByTestID(testID, args) {
|
|
277
|
+
return await this.findByTestId(testID, args);
|
|
278
|
+
}
|
|
279
|
+
/**
|
|
280
|
+
* @param {string|import("selenium-webdriver").WebElement|{selector: string} & FindArgs} elementOrIdentifier
|
|
281
|
+
* @param {FindArgs} [args]
|
|
282
|
+
* @returns {Promise<import("selenium-webdriver").WebElement>}
|
|
283
|
+
*/
|
|
284
|
+
async _findElement(elementOrIdentifier, args) {
|
|
285
|
+
/** @type {import("selenium-webdriver").WebElement} */
|
|
286
|
+
let element;
|
|
287
|
+
if (typeof elementOrIdentifier == "string") {
|
|
288
|
+
element = await this.find(elementOrIdentifier, args);
|
|
289
|
+
}
|
|
290
|
+
else if (typeof elementOrIdentifier == "object" && elementOrIdentifier !== null && "selector" in elementOrIdentifier) {
|
|
291
|
+
const { selector, ...restArgs } = elementOrIdentifier;
|
|
292
|
+
element = await this.find(selector, restArgs);
|
|
293
|
+
}
|
|
294
|
+
else {
|
|
295
|
+
element = /** @type {import("selenium-webdriver").WebElement} */ (elementOrIdentifier);
|
|
296
|
+
}
|
|
297
|
+
return element;
|
|
298
|
+
}
|
|
299
|
+
/**
|
|
300
|
+
* Finds a single element by CSS selector without waiting
|
|
301
|
+
* @param {string} selector
|
|
302
|
+
* @param {FindArgs} [args]
|
|
303
|
+
* @returns {Promise<import("selenium-webdriver").WebElement>}
|
|
304
|
+
*/
|
|
305
|
+
async findNoWait(selector, args = {}) {
|
|
306
|
+
await this.driverSetTimeouts(0);
|
|
307
|
+
try {
|
|
308
|
+
return await this.find(selector, args);
|
|
309
|
+
}
|
|
310
|
+
finally {
|
|
311
|
+
await this.restoreTimeouts();
|
|
312
|
+
}
|
|
313
|
+
}
|
|
314
|
+
/**
|
|
315
|
+
* Clicks an element, allowing selector args when using a CSS selector.
|
|
316
|
+
* @param {string|import("selenium-webdriver").WebElement} elementOrIdentifier
|
|
317
|
+
* @param {FindArgs} [args]
|
|
318
|
+
* @returns {Promise<void>}
|
|
319
|
+
*/
|
|
320
|
+
async click(elementOrIdentifier, args) {
|
|
321
|
+
let tries = 0;
|
|
322
|
+
while (true) {
|
|
323
|
+
tries++;
|
|
324
|
+
try {
|
|
325
|
+
const element = await this._findElement(elementOrIdentifier, args);
|
|
326
|
+
const actions = this.getWebDriver().actions({ async: true });
|
|
327
|
+
await actions.move({ origin: element }).click().perform();
|
|
328
|
+
break;
|
|
329
|
+
}
|
|
330
|
+
catch (error) {
|
|
331
|
+
if (error instanceof Error) {
|
|
332
|
+
if (error.constructor.name === "ElementNotInteractableError") {
|
|
333
|
+
if (tries >= 3) {
|
|
334
|
+
throw new Error(`Element ${elementOrIdentifier.constructor.name} click failed after ${tries} tries - ${error.constructor.name}: ${error.message}`);
|
|
335
|
+
}
|
|
336
|
+
else {
|
|
337
|
+
await wait(50);
|
|
338
|
+
}
|
|
339
|
+
}
|
|
340
|
+
else {
|
|
341
|
+
// Re-throw with un-corrupted stack trace
|
|
342
|
+
throw new Error(`Element ${elementOrIdentifier.constructor.name} click failed - ${error.constructor.name}: ${error.message}`);
|
|
343
|
+
}
|
|
344
|
+
}
|
|
345
|
+
else {
|
|
346
|
+
throw new Error(`Element ${elementOrIdentifier.constructor.name} click failed - ${typeof error}: ${error}`);
|
|
347
|
+
}
|
|
348
|
+
}
|
|
349
|
+
}
|
|
350
|
+
}
|
|
351
|
+
/**
|
|
352
|
+
* Interacts with an element by calling a method on it with the given arguments.
|
|
353
|
+
* Retrying on ElementNotInteractableError, ElementClickInterceptedError, or StaleElementReferenceError.
|
|
354
|
+
* @param {import("selenium-webdriver").WebElement|string|{selector: string} & FindArgs} elementOrIdentifier The element or a CSS selector to find the element.
|
|
355
|
+
* @param {string} methodName The method name to call on the element.
|
|
356
|
+
* @param {...any} args Arguments to pass to the method.
|
|
357
|
+
* @returns {Promise<any>}
|
|
358
|
+
*/
|
|
359
|
+
async interact(elementOrIdentifier, methodName, ...args) {
|
|
360
|
+
let tries = 0;
|
|
361
|
+
while (true) {
|
|
362
|
+
tries++;
|
|
363
|
+
const element = await this._findElement(elementOrIdentifier);
|
|
364
|
+
if (!element[methodName]) {
|
|
365
|
+
throw new Error(`${element.constructor.name} hasn't an attribute named: ${methodName}`);
|
|
366
|
+
}
|
|
367
|
+
else if (typeof element[methodName] != "function") {
|
|
368
|
+
throw new Error(`${element.constructor.name}#${methodName} is not a function`);
|
|
369
|
+
}
|
|
370
|
+
try {
|
|
371
|
+
// Dont call with candidate, because that will bind the function wrong.
|
|
372
|
+
return await element[methodName](...args);
|
|
373
|
+
}
|
|
374
|
+
catch (error) {
|
|
375
|
+
if (error instanceof Error) {
|
|
376
|
+
if (error.constructor.name === "ElementNotInteractableError" ||
|
|
377
|
+
error.constructor.name === "ElementClickInterceptedError" ||
|
|
378
|
+
error.constructor.name === "StaleElementReferenceError") {
|
|
379
|
+
// Retry finding the element and interacting with it
|
|
380
|
+
if (tries >= 3) {
|
|
381
|
+
let elementDescription;
|
|
382
|
+
if (typeof elementOrIdentifier == "string") {
|
|
383
|
+
elementDescription = `CSS selector ${elementOrIdentifier}`;
|
|
384
|
+
}
|
|
385
|
+
else {
|
|
386
|
+
elementDescription = `${element.constructor.name}`;
|
|
387
|
+
}
|
|
388
|
+
throw new Error(`${elementDescription} ${methodName} failed after ${tries} tries - ${error.constructor.name}: ${error.message}`);
|
|
389
|
+
}
|
|
390
|
+
else {
|
|
391
|
+
await wait(50);
|
|
392
|
+
}
|
|
393
|
+
}
|
|
394
|
+
else {
|
|
395
|
+
// Re-throw with un-corrupted stack trace
|
|
396
|
+
throw new Error(`${element.constructor.name} ${methodName} failed - ${error.constructor.name}: ${error.message}`);
|
|
397
|
+
}
|
|
398
|
+
}
|
|
399
|
+
else {
|
|
400
|
+
throw new Error(`${element.constructor.name} ${methodName} failed - ${typeof error}: ${error}`);
|
|
401
|
+
}
|
|
402
|
+
}
|
|
403
|
+
}
|
|
404
|
+
}
|
|
405
|
+
/**
|
|
406
|
+
* @param {string} selector
|
|
407
|
+
* @param {WaitForNoSelectorArgs} [args]
|
|
408
|
+
* @returns {Promise<void>}
|
|
409
|
+
*/
|
|
410
|
+
async waitForNoSelector(selector, args = {}) {
|
|
411
|
+
const { useBaseSelector, ...restArgs } = args;
|
|
412
|
+
if (Object.keys(restArgs).length > 0) {
|
|
413
|
+
throw new Error(`Unexpected args: ${Object.keys(restArgs).join(", ")}`);
|
|
414
|
+
}
|
|
415
|
+
const actualSelector = useBaseSelector ? this.getSelector(selector) : selector;
|
|
416
|
+
await this.driverSetTimeouts(0);
|
|
417
|
+
try {
|
|
418
|
+
await this._withRethrownErrors(async () => {
|
|
419
|
+
await this.getWebDriver().wait(async () => {
|
|
420
|
+
const elements = await this.getWebDriver().findElements(By.css(actualSelector));
|
|
421
|
+
// Not found at all
|
|
422
|
+
if (elements.length === 0) {
|
|
423
|
+
return true;
|
|
424
|
+
}
|
|
425
|
+
// Found but not visible
|
|
426
|
+
try {
|
|
427
|
+
const isDisplayed = await elements[0].isDisplayed();
|
|
428
|
+
return !isDisplayed;
|
|
429
|
+
}
|
|
430
|
+
catch (error) {
|
|
431
|
+
if (error instanceof Error &&
|
|
432
|
+
(error.constructor.name === "StaleElementReferenceError" || error.message.includes("stale element reference"))) {
|
|
433
|
+
return false;
|
|
434
|
+
}
|
|
435
|
+
throw error;
|
|
436
|
+
}
|
|
437
|
+
}, this.getTimeouts());
|
|
438
|
+
});
|
|
439
|
+
}
|
|
440
|
+
finally {
|
|
441
|
+
await this.restoreTimeouts();
|
|
442
|
+
}
|
|
443
|
+
}
|
|
444
|
+
/**
|
|
445
|
+
* @param {() => Promise<any>} callback
|
|
446
|
+
* @returns {Promise<any>}
|
|
447
|
+
*/
|
|
448
|
+
async _withRethrownErrors(callback) {
|
|
449
|
+
try {
|
|
450
|
+
return await callback();
|
|
451
|
+
}
|
|
452
|
+
catch (error) {
|
|
453
|
+
if (error instanceof WebDriverError) {
|
|
454
|
+
throw new Error(`Selenium ${error.constructor.name}: ${error.message}`);
|
|
455
|
+
}
|
|
456
|
+
else {
|
|
457
|
+
throw error;
|
|
458
|
+
}
|
|
459
|
+
}
|
|
460
|
+
}
|
|
461
|
+
}
|
|
462
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"webdriver-driver.js","sourceRoot":"","sources":["../../src/drivers/webdriver-driver.js"],"names":[],"mappings":"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","sourcesContent":["import {By, error as SeleniumError} from \"selenium-webdriver\"\nimport logging from \"selenium-webdriver/lib/logging.js\"\nimport {wait} from \"awaitery\"\nimport timeout from \"awaitery/build/timeout.js\"\n\n/**\n * @typedef {object} FindArgs\n * @property {number} [timeout] Override timeout for lookup.\n * @property {boolean} [visible] Whether to require elements to be visible.\n * @property {boolean} [useBaseSelector] Whether to scope by the base selector.\n */\n/**\n * @typedef {object} WaitForNoSelectorArgs\n * @property {boolean} [useBaseSelector] Whether to scope by the base selector.\n */\n\nclass ElementNotFoundError extends Error { }\nconst {WebDriverError} = SeleniumError\n\n/**\n * Base driver using selenium-webdriver sessions.\n */\nexport default class WebDriverDriver {\n  /**\n   * @param {object} args\n   * @param {import(\"../system-test.js\").default} args.systemTest\n   * @param {Record<string, any>} [args.options]\n   */\n  constructor({systemTest, options = {}}) {\n    this.systemTest = systemTest\n    this.options = options\n    this.baseUrl = undefined\n    this.webDriver = undefined\n    this._driverTimeouts = 5000\n    this._timeouts = 5000\n  }\n\n  /**\n   * @param {string} baseUrl\n   * @returns {void}\n   */\n  setBaseUrl(baseUrl) {\n    this.baseUrl = baseUrl\n  }\n\n  /** @returns {string} */\n  getBaseUrl() {\n    if (!this.baseUrl) {\n      throw new Error(\"Driver base URL has not been set\")\n    }\n\n    return this.baseUrl\n  }\n\n  /** @returns {number} */\n  getTimeouts() { return this._timeouts }\n\n  /**\n   * @returns {import(\"selenium-webdriver\").WebDriver}\n   */\n  getWebDriver() {\n    if (!this.webDriver) throw new Error(\"Driver hasn't been initialized yet\")\n    this.systemTest.throwIfHttpServerError()\n\n    return this.webDriver\n  }\n\n  /**\n   * @param {import(\"selenium-webdriver\").WebDriver} webDriver\n   * @returns {void}\n   */\n  setWebDriver(webDriver) {\n    this.webDriver = webDriver\n    this.systemTest.driver = webDriver\n  }\n\n  /**\n   * @returns {Promise<void>}\n   */\n  async start() {\n    throw new Error(\"start() must be implemented by the driver\")\n  }\n\n  /**\n   * @returns {Promise<void>}\n   */\n  async stop() {\n    if (this.webDriver) {\n      await timeout({timeout: this.getTimeouts(), errorMessage: \"timeout while quitting WebDriver\"}, async () => await this.webDriver.quit())\n    }\n\n    this.webDriver = undefined\n    this.systemTest.driver = undefined\n  }\n\n  /**\n   * @param {number} newTimeout\n   * @returns {Promise<void>}\n   */\n  async driverSetTimeouts(newTimeout) {\n    this._driverTimeouts = newTimeout\n    await this.getWebDriver().manage().setTimeouts({implicit: newTimeout})\n  }\n\n  /**\n   * @returns {Promise<void>}\n   */\n  async restoreTimeouts() {\n    if (!this.getTimeouts()) {\n      throw new Error(\"Timeouts haven't previously been set\")\n    }\n\n    await this.driverSetTimeouts(this.getTimeouts())\n  }\n\n  /**\n   * @param {number} newTimeout\n   * @returns {Promise<void>}\n   */\n  async setTimeouts(newTimeout) {\n    this._timeouts = newTimeout\n    await this.restoreTimeouts()\n  }\n\n  /**\n   * @param {string} selector\n   * @returns {string}\n   */\n  getSelector(selector) {\n    return this.systemTest.getSelector(selector)\n  }\n\n  /**\n   * @param {string} path\n   * @returns {Promise<void>}\n   */\n  async driverVisit(path) {\n    const url = `${this.getBaseUrl()}${path}`\n\n    await this.getWebDriver().get(url)\n  }\n\n  /** @returns {Promise<string>} */\n  async getCurrentUrl() {\n    try {\n      return await this.getWebDriver().getCurrentUrl()\n    } catch {\n      return \"\"\n    }\n  }\n\n  /** @returns {Promise<string>} */\n  async getHTML() {\n    return await this.getWebDriver().getPageSource()\n  }\n\n  /** @returns {Promise<string>} */\n  async takeScreenshot() {\n    return await this.getWebDriver().takeScreenshot()\n  }\n\n  /**\n   * Gets browser logs\n   * @returns {Promise<string[]>}\n   */\n  async getBrowserLogs() {\n    let entries\n\n    try {\n      entries = await this.getWebDriver().manage().logs().get(logging.Type.BROWSER)\n    } catch {\n      return []\n    }\n    const browserLogs = []\n\n    for (const entry of entries) {\n      const messageMatch = entry.message.match(/^(.+) (\\d+):(\\d+) (.+)$/)\n      let message\n\n      if (messageMatch) {\n        message = messageMatch[4]\n      } else {\n        message = entry.message\n      }\n\n      browserLogs.push(`${entry.level.name}: ${message}`)\n    }\n\n    return browserLogs\n  }\n\n  /**\n   * Finds all elements by CSS selector\n   * @param {string} selector\n   * @param {FindArgs} [args]\n   * @returns {Promise<import(\"selenium-webdriver\").WebElement[]>}\n   */\n  async all(selector, args = {}) {\n    const {visible = true, timeout, useBaseSelector = true, ...restArgs} = args\n    const restArgsKeys = Object.keys(restArgs)\n    let actualTimeout\n\n    if (timeout === undefined) {\n      actualTimeout = this._driverTimeouts\n    } else {\n      actualTimeout = timeout\n    }\n\n    if (restArgsKeys.length > 0) throw new Error(`Unknown arguments: ${restArgsKeys.join(\", \")}`)\n\n    const actualSelector = useBaseSelector ? this.getSelector(selector) : selector\n    const startTime = Date.now()\n    const getTimeLeft = () => Math.max(actualTimeout - (Date.now() - startTime), 0)\n    const getElements = async () => {\n      const foundElements = await this.getWebDriver().findElements(By.css(actualSelector))\n\n      if (visible !== true && visible !== false) {\n        return foundElements\n      }\n\n      const filteredElements = []\n\n      for (const element of foundElements) {\n        const isDisplayed = await element.isDisplayed()\n\n        if (visible && !isDisplayed) continue\n        if (!visible && isDisplayed) continue\n\n        filteredElements.push(element)\n      }\n\n      return filteredElements\n    }\n    let elements = []\n\n    while (true) {\n      const timeLeft = actualTimeout == 0 ? 0 : getTimeLeft()\n\n      try {\n        if (timeLeft == 0) {\n          elements = await getElements()\n        } else {\n          await this.getWebDriver().wait(async () => {\n            elements = await getElements()\n\n            return elements.length > 0\n          }, timeLeft)\n        }\n\n        break\n      } catch (error) {\n        if (error instanceof SeleniumError.TimeoutError && getTimeLeft() > 0) {\n          continue\n        }\n\n        throw new Error(`Couldn't get elements with selector: ${actualSelector}: ${error instanceof Error ? error.message : error}`)\n      }\n    }\n    return elements\n  }\n\n  /**\n   * Finds a single element by CSS selector\n   * @param {string} selector\n   * @param {FindArgs} [args]\n   * @returns {Promise<import(\"selenium-webdriver\").WebElement>}\n   */\n  async find(selector, args = {}) {\n    const startTime = Date.now()\n    let elements = []\n\n    try {\n      elements = await this.all(selector, args)\n    } catch (error) {\n      // Re-throw to recover stack trace\n      if (error instanceof Error) {\n        if (error.message.startsWith(\"Wait timed out after\")) {\n          elements = []\n        }\n\n        throw new Error(`${error.constructor.name} - ${error.message} (selector: ${this.getSelector(selector)})`)\n      } else {\n        throw new Error(`${typeof error} - ${error} (selector: ${this.getSelector(selector)})`)\n      }\n    }\n\n    if (elements.length > 1) {\n      throw new Error(`More than 1 elements (${elements.length}) was found by CSS: ${this.getSelector(selector)}`)\n    }\n\n    if (!elements[0]) {\n      const elapsedSeconds = (Date.now() - startTime) / 1000\n      throw new ElementNotFoundError(`Element couldn't be found after ${elapsedSeconds.toFixed(2)}s by CSS: ${this.getSelector(selector)}`)\n    }\n\n    return elements[0]\n  }\n\n  /**\n   * Finds a single element by test ID\n   * @param {string} testId\n   * @param {FindArgs} [args]\n   * @returns {Promise<import(\"selenium-webdriver\").WebElement>}\n   */\n  async findByTestId(testId, args) {\n    return await this.find(`[data-testid='${testId}']`, args)\n  }\n\n  /**\n   * Finds a single element by test ID\n   * @param {string} testID\n   * @param {FindArgs} [args]\n   * @returns {Promise<import(\"selenium-webdriver\").WebElement>}\n   */\n  async findByTestID(testID, args) {\n    return await this.findByTestId(testID, args)\n  }\n\n  /**\n   * @param {string|import(\"selenium-webdriver\").WebElement|{selector: string} & FindArgs} elementOrIdentifier\n   * @param {FindArgs} [args]\n   * @returns {Promise<import(\"selenium-webdriver\").WebElement>}\n   */\n  async _findElement(elementOrIdentifier, args) {\n    /** @type {import(\"selenium-webdriver\").WebElement} */\n    let element\n\n    if (typeof elementOrIdentifier == \"string\") {\n      element = await this.find(elementOrIdentifier, args)\n    } else if (typeof elementOrIdentifier == \"object\" && elementOrIdentifier !== null && \"selector\" in elementOrIdentifier) {\n      const {selector, ...restArgs} = elementOrIdentifier\n\n      element = await this.find(selector, restArgs)\n    } else {\n      element = /** @type {import(\"selenium-webdriver\").WebElement} */ (elementOrIdentifier)\n    }\n\n    return element\n  }\n\n  /**\n   * Finds a single element by CSS selector without waiting\n   * @param {string} selector\n   * @param {FindArgs} [args]\n   * @returns {Promise<import(\"selenium-webdriver\").WebElement>}\n   */\n  async findNoWait(selector, args = {}) {\n    await this.driverSetTimeouts(0)\n\n    try {\n      return await this.find(selector, args)\n    } finally {\n      await this.restoreTimeouts()\n    }\n  }\n\n  /**\n   * Clicks an element, allowing selector args when using a CSS selector.\n   * @param {string|import(\"selenium-webdriver\").WebElement} elementOrIdentifier\n   * @param {FindArgs} [args]\n   * @returns {Promise<void>}\n   */\n  async click(elementOrIdentifier, args) {\n    let tries = 0\n\n    while (true) {\n      tries++\n\n      try {\n        const element = await this._findElement(elementOrIdentifier, args)\n        const actions = this.getWebDriver().actions({async: true})\n\n        await actions.move({origin: element}).click().perform()\n        break\n      } catch (error) {\n        if (error instanceof Error) {\n          if (error.constructor.name === \"ElementNotInteractableError\") {\n            if (tries >= 3) {\n              throw new Error(`Element ${elementOrIdentifier.constructor.name} click failed after ${tries} tries - ${error.constructor.name}: ${error.message}`)\n            } else {\n              await wait(50)\n            }\n          } else {\n            // Re-throw with un-corrupted stack trace\n            throw new Error(`Element ${elementOrIdentifier.constructor.name} click failed - ${error.constructor.name}: ${error.message}`)\n          }\n        } else {\n          throw new Error(`Element ${elementOrIdentifier.constructor.name} click failed - ${typeof error}: ${error}`)\n        }\n      }\n    }\n  }\n\n  /**\n   * Interacts with an element by calling a method on it with the given arguments.\n   * Retrying on ElementNotInteractableError, ElementClickInterceptedError, or StaleElementReferenceError.\n   * @param {import(\"selenium-webdriver\").WebElement|string|{selector: string} & FindArgs} elementOrIdentifier The element or a CSS selector to find the element.\n   * @param {string} methodName The method name to call on the element.\n   * @param {...any} args Arguments to pass to the method.\n   * @returns {Promise<any>}\n   */\n  async interact(elementOrIdentifier, methodName, ...args) {\n    let tries = 0\n\n    while (true) {\n      tries++\n\n      const element = await this._findElement(elementOrIdentifier)\n\n      if (!element[methodName]) {\n        throw new Error(`${element.constructor.name} hasn't an attribute named: ${methodName}`)\n      } else if (typeof element[methodName] != \"function\") {\n        throw new Error(`${element.constructor.name}#${methodName} is not a function`)\n      }\n\n      try {\n        // Dont call with candidate, because that will bind the function wrong.\n        return await element[methodName](...args)\n      } catch (error) {\n        if (error instanceof Error) {\n          if (\n            error.constructor.name === \"ElementNotInteractableError\" ||\n            error.constructor.name === \"ElementClickInterceptedError\" ||\n            error.constructor.name === \"StaleElementReferenceError\"\n          ) {\n            // Retry finding the element and interacting with it\n            if (tries >= 3) {\n              let elementDescription\n\n              if (typeof elementOrIdentifier == \"string\") {\n                elementDescription = `CSS selector ${elementOrIdentifier}`\n              } else {\n                elementDescription = `${element.constructor.name}`\n              }\n\n              throw new Error(`${elementDescription} ${methodName} failed after ${tries} tries - ${error.constructor.name}: ${error.message}`)\n            } else {\n              await wait(50)\n            }\n          } else {\n            // Re-throw with un-corrupted stack trace\n            throw new Error(`${element.constructor.name} ${methodName} failed - ${error.constructor.name}: ${error.message}`)\n          }\n        } else {\n          throw new Error(`${element.constructor.name} ${methodName} failed - ${typeof error}: ${error}`)\n        }\n      }\n    }\n  }\n\n  /**\n   * @param {string} selector\n   * @param {WaitForNoSelectorArgs} [args]\n   * @returns {Promise<void>}\n   */\n  async waitForNoSelector(selector, args = {}) {\n    const {useBaseSelector, ...restArgs} = args\n\n    if (Object.keys(restArgs).length > 0) {\n      throw new Error(`Unexpected args: ${Object.keys(restArgs).join(\", \")}`)\n    }\n\n    const actualSelector = useBaseSelector ? this.getSelector(selector) : selector\n\n    await this.driverSetTimeouts(0)\n\n    try {\n      await this._withRethrownErrors(async () => {\n        await this.getWebDriver().wait(\n          async () => {\n            const elements = await this.getWebDriver().findElements(By.css(actualSelector))\n\n            // Not found at all\n            if (elements.length === 0) {\n              return true\n            }\n\n            // Found but not visible\n            try {\n              const isDisplayed = await elements[0].isDisplayed()\n\n              return !isDisplayed\n            } catch (error) {\n              if (\n                error instanceof Error &&\n                (error.constructor.name === \"StaleElementReferenceError\" || error.message.includes(\"stale element reference\"))\n              ) {\n                return false\n              }\n\n              throw error\n            }\n          },\n          this.getTimeouts()\n        )\n      })\n    } finally {\n      await this.restoreTimeouts()\n    }\n  }\n\n  /**\n   * @param {() => Promise<any>} callback\n   * @returns {Promise<any>}\n   */\n  async _withRethrownErrors(callback) {\n    try {\n      return await callback()\n    } catch (error) {\n      if (error instanceof WebDriverError) {\n        throw new Error(`Selenium ${error.constructor.name}: ${error.message}`)\n      } else {\n        throw error\n      }\n    }\n  }\n}\n"]}
|