system-testing 1.0.103 → 1.0.105
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 +8 -0
- package/build/browser-command-runner.js +53 -1
- package/build/browser.d.ts +109 -20
- package/build/browser.js +207 -1
- package/build/cli-helpers.js +22 -1
- package/build/cli.js +3 -1
- package/build/drivers/webdriver-driver.d.ts +49 -0
- package/build/drivers/webdriver-driver.js +141 -3
- package/build/system-test.d.ts +5 -2
- package/build/system-test.js +28 -4
- package/package.json +1 -1
package/README.md
CHANGED
|
@@ -449,6 +449,14 @@ await systemTest.reinitialize()
|
|
|
449
449
|
|
|
450
450
|
This tears down the browser, servers, and sockets, then starts them again so subsequent steps run against a fresh app instance.
|
|
451
451
|
|
|
452
|
+
`SystemTest.run(...)` does this automatically after a failed callback or teardown by default. Disable it only for tests that intentionally inspect the broken session after failure:
|
|
453
|
+
|
|
454
|
+
```js
|
|
455
|
+
await SystemTest.run({reinitializeAfterFailure: false}, async (systemTest) => {
|
|
456
|
+
await systemTest.findByTestID("brokenState")
|
|
457
|
+
})
|
|
458
|
+
```
|
|
459
|
+
|
|
452
460
|
## Dummy Expo app
|
|
453
461
|
|
|
454
462
|
A ready-to-run Expo Router dummy app that uses `system-testing` lives in `spec/dummy`.
|
|
@@ -1,3 +1,20 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Parse a cookie boolean attribute coming in over the JSON command transport.
|
|
3
|
+
* The transport carries booleans either as native `true`/`false` or as the
|
|
4
|
+
* string forms `"true"`/`"false"`. Anything else is rejected so a typo like
|
|
5
|
+
* `--cookie-secure=TRUE` or `--cookie-secure=1` fails loudly instead of
|
|
6
|
+
* silently producing an insecure cookie.
|
|
7
|
+
* @param {string} fieldName
|
|
8
|
+
* @param {unknown} rawValue
|
|
9
|
+
* @returns {boolean}
|
|
10
|
+
*/
|
|
11
|
+
function parseCookieBoolean(fieldName, rawValue) {
|
|
12
|
+
if (rawValue === true || rawValue === "true")
|
|
13
|
+
return true;
|
|
14
|
+
if (rawValue === false || rawValue === "false")
|
|
15
|
+
return false;
|
|
16
|
+
throw new Error(`addCookie ${fieldName} must be true or false, got ${JSON.stringify(rawValue)}`);
|
|
17
|
+
}
|
|
1
18
|
/** Runs browser commands across CLI and WebSocket transports. */
|
|
2
19
|
export default class BrowserCommandRunner {
|
|
3
20
|
/**
|
|
@@ -144,6 +161,41 @@ export default class BrowserCommandRunner {
|
|
|
144
161
|
await this.browser.expectNoElement(commandArgs.selector, this.normalizeFindArgs(commandArgs));
|
|
145
162
|
return { ok: true };
|
|
146
163
|
}
|
|
164
|
+
if (command === "executeScript") {
|
|
165
|
+
const script = commandArgs.script;
|
|
166
|
+
if (typeof script !== "string" || script.length === 0) {
|
|
167
|
+
throw new Error("executeScript requires script");
|
|
168
|
+
}
|
|
169
|
+
const scriptArgs = Array.isArray(commandArgs.args) ? commandArgs.args : [];
|
|
170
|
+
const result = await this.browser.executeScript(script, ...scriptArgs);
|
|
171
|
+
return { result };
|
|
172
|
+
}
|
|
173
|
+
if (command === "addCookie") {
|
|
174
|
+
const name = commandArgs.name;
|
|
175
|
+
if (typeof name !== "string" || name.length === 0) {
|
|
176
|
+
throw new Error("addCookie requires name");
|
|
177
|
+
}
|
|
178
|
+
const value = commandArgs.value;
|
|
179
|
+
if (typeof value !== "string") {
|
|
180
|
+
throw new Error("addCookie requires string value");
|
|
181
|
+
}
|
|
182
|
+
/** @type {{name: string, value: string, domain?: string, path?: string, secure?: boolean, httpOnly?: boolean, expiry?: number, sameSite?: "Strict" | "Lax" | "None"}} */
|
|
183
|
+
const cookie = { name, value };
|
|
184
|
+
if (typeof commandArgs.domain === "string" && commandArgs.domain.length > 0)
|
|
185
|
+
cookie.domain = commandArgs.domain;
|
|
186
|
+
if (typeof commandArgs.path === "string" && commandArgs.path.length > 0)
|
|
187
|
+
cookie.path = commandArgs.path;
|
|
188
|
+
if (commandArgs.secure !== undefined)
|
|
189
|
+
cookie.secure = parseCookieBoolean("secure", commandArgs.secure);
|
|
190
|
+
if (commandArgs.httpOnly !== undefined)
|
|
191
|
+
cookie.httpOnly = parseCookieBoolean("httpOnly", commandArgs.httpOnly);
|
|
192
|
+
if (commandArgs.expiry !== undefined)
|
|
193
|
+
cookie.expiry = Number(commandArgs.expiry);
|
|
194
|
+
if (typeof commandArgs.sameSite === "string")
|
|
195
|
+
cookie.sameSite = /** @type {"Strict" | "Lax" | "None"} */ (commandArgs.sameSite);
|
|
196
|
+
await this.browser.addCookie(cookie);
|
|
197
|
+
return { ok: true };
|
|
198
|
+
}
|
|
147
199
|
if (command === "interact") {
|
|
148
200
|
const selector = commandArgs.selector;
|
|
149
201
|
const methodName = commandArgs.methodName ?? commandArgs.method;
|
|
@@ -169,4 +221,4 @@ export default class BrowserCommandRunner {
|
|
|
169
221
|
throw new Error(`Unknown browser command: ${command}`);
|
|
170
222
|
}
|
|
171
223
|
}
|
|
172
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"browser-command-runner.js","sourceRoot":"","sources":["../src/browser-command-runner.js"],"names":[],"mappings":"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","sourcesContent":["/** Runs browser commands across CLI and WebSocket transports. */\nexport default class BrowserCommandRunner {\n  /**\n   * @param {object} args\n   * @param {import(\"./browser.js\").default} args.browser\n   */\n  constructor({browser}) {\n    this.browser = browser\n  }\n\n  /**\n   * @param {Record<string, any>} commandArgs\n   * @returns {{timeout?: number}}\n   */\n  normalizeTimeoutArgs(commandArgs) {\n    const normalizedArgs = {}\n\n    if (\"timeout\" in commandArgs && commandArgs.timeout !== undefined) {\n      normalizedArgs.timeout = Number(commandArgs.timeout)\n\n      if (Number.isNaN(normalizedArgs.timeout)) {\n        throw new Error(`Invalid timeout: ${commandArgs.timeout}`)\n      }\n    }\n\n    return normalizedArgs\n  }\n\n  /**\n   * @param {Record<string, any>} commandArgs\n   * @returns {import(\"./system-test.js\").FindArgs}\n   */\n  normalizeFindArgs(commandArgs) {\n    const findArgs = /** @type {import(\"./system-test.js\").FindArgs} */ (this.normalizeTimeoutArgs(commandArgs))\n\n    if (\"visible\" in commandArgs && commandArgs.visible !== undefined) {\n      if (commandArgs.visible === null || commandArgs.visible === \"null\") {\n        findArgs.visible = null\n      } else if (typeof commandArgs.visible === \"boolean\") {\n        findArgs.visible = commandArgs.visible\n      } else {\n        findArgs.visible = commandArgs.visible === \"true\"\n      }\n    }\n\n    if (\"useBaseSelector\" in commandArgs && commandArgs.useBaseSelector !== undefined) {\n      if (typeof commandArgs.useBaseSelector === \"boolean\") {\n        findArgs.useBaseSelector = commandArgs.useBaseSelector\n      } else {\n        findArgs.useBaseSelector = commandArgs.useBaseSelector === \"true\"\n      }\n    }\n\n    if (\"scrollTo\" in commandArgs && commandArgs.scrollTo !== undefined) {\n      if (typeof commandArgs.scrollTo === \"boolean\") {\n        findArgs.scrollTo = commandArgs.scrollTo\n      } else {\n        findArgs.scrollTo = commandArgs.scrollTo === \"true\"\n      }\n    }\n\n    return findArgs\n  }\n\n  /**\n   * @param {import(\"selenium-webdriver\").WebElement} element\n   * @returns {Promise<Record<string, any>>}\n   */\n  async serializeElement(element) {\n    const text = await element.getText()\n    const tagName = await element.getTagName()\n    const displayed = await element.isDisplayed()\n\n    return {displayed, tagName, text}\n  }\n\n  /**\n   * @param {string} command\n   * @param {Record<string, any>} commandArgs\n   * @returns {Promise<any>}\n   */\n  async run(command, commandArgs = {}) {\n    if (command === \"visit\") {\n      const path = commandArgs.path ?? commandArgs.url\n\n      if (!path) {\n        throw new Error(\"visit requires path or url\")\n      }\n\n      await this.browser.visit(path, this.normalizeTimeoutArgs(commandArgs))\n      return {ok: true}\n    }\n\n    if (command === \"dismissTo\") {\n      const path = commandArgs.path ?? commandArgs.url\n\n      if (!path) {\n        throw new Error(\"dismissTo requires path or url\")\n      }\n\n      await this.browser.dismissTo(path, this.normalizeTimeoutArgs(commandArgs))\n      return {ok: true}\n    }\n\n    if (command === \"setBaseSelector\") {\n      if (!commandArgs.selector) {\n        throw new Error(\"setBaseSelector requires selector\")\n      }\n\n      this.browser.setBaseSelector(commandArgs.selector)\n      return {ok: true}\n    }\n\n    if (command === \"getCurrentUrl\") {\n      return {currentUrl: await this.browser.getCurrentUrl()}\n    }\n\n    if (command === \"getHTML\") {\n      return {html: await this.browser.getHTML()}\n    }\n\n    if (command === \"getBrowserLogs\") {\n      return {logs: await this.browser.getBrowserLogs()}\n    }\n\n    if (command === \"takeScreenshot\") {\n      return await this.browser.takeScreenshot()\n    }\n\n    if (command === \"find\") {\n      if (!commandArgs.selector) {\n        throw new Error(\"find requires selector\")\n      }\n\n      const element = await this.browser.find(commandArgs.selector, this.normalizeFindArgs(commandArgs))\n\n      return {element: await this.serializeElement(element)}\n    }\n\n    if (command === \"findByTestID\") {\n      const testID = commandArgs.testID ?? commandArgs.testId\n\n      if (!testID) {\n        throw new Error(\"findByTestID requires testID\")\n      }\n\n      const element = await this.browser.findByTestID(testID, this.normalizeFindArgs(commandArgs))\n\n      return {element: await this.serializeElement(element)}\n    }\n\n    if (command === \"click\") {\n      const selector = commandArgs.selector\n\n      if (!selector) {\n        throw new Error(\"click requires selector\")\n      }\n\n      await this.browser.click(selector, this.normalizeFindArgs(commandArgs))\n      return {ok: true}\n    }\n\n    if (command === \"waitForNoSelector\") {\n      if (!commandArgs.selector) {\n        throw new Error(\"waitForNoSelector requires selector\")\n      }\n\n      await this.browser.waitForNoSelector(commandArgs.selector, this.normalizeFindArgs(commandArgs))\n      return {ok: true}\n    }\n\n    if (command === \"expectNoElement\") {\n      if (!commandArgs.selector) {\n        throw new Error(\"expectNoElement requires selector\")\n      }\n\n      await this.browser.expectNoElement(commandArgs.selector, this.normalizeFindArgs(commandArgs))\n      return {ok: true}\n    }\n\n    if (command === \"interact\") {\n      const selector = commandArgs.selector\n      const methodName = commandArgs.methodName ?? commandArgs.method\n      const methodArgs = Array.isArray(commandArgs.args) ? commandArgs.args : []\n      const interactArgs = /** @type {{selector: string} & import(\"./system-test.js\").InteractArgs} */ ({selector, ...this.normalizeFindArgs(commandArgs)})\n\n      if (!selector) {\n        throw new Error(\"interact requires selector\")\n      }\n\n      if (!methodName) {\n        throw new Error(\"interact requires methodName\")\n      }\n\n      if (\"withFallback\" in commandArgs && commandArgs.withFallback !== undefined) {\n        if (typeof commandArgs.withFallback === \"boolean\") {\n          interactArgs.withFallback = commandArgs.withFallback\n        } else {\n          interactArgs.withFallback = commandArgs.withFallback === \"true\"\n        }\n      }\n\n      const result = await this.browser.interact(interactArgs, methodName, ...methodArgs)\n\n      return {result}\n    }\n\n    throw new Error(`Unknown browser command: ${command}`)\n  }\n}\n"]}
|
|
224
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"browser-command-runner.js","sourceRoot":"","sources":["../src/browser-command-runner.js"],"names":[],"mappings":"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","sourcesContent":["/**\n * Parse a cookie boolean attribute coming in over the JSON command transport.\n * The transport carries booleans either as native `true`/`false` or as the\n * string forms `\"true\"`/`\"false\"`. Anything else is rejected so a typo like\n * `--cookie-secure=TRUE` or `--cookie-secure=1` fails loudly instead of\n * silently producing an insecure cookie.\n * @param {string} fieldName\n * @param {unknown} rawValue\n * @returns {boolean}\n */\nfunction parseCookieBoolean(fieldName, rawValue) {\n  if (rawValue === true || rawValue === \"true\") return true\n  if (rawValue === false || rawValue === \"false\") return false\n\n  throw new Error(`addCookie ${fieldName} must be true or false, got ${JSON.stringify(rawValue)}`)\n}\n\n/** Runs browser commands across CLI and WebSocket transports. */\nexport default class BrowserCommandRunner {\n  /**\n   * @param {object} args\n   * @param {import(\"./browser.js\").default} args.browser\n   */\n  constructor({browser}) {\n    this.browser = browser\n  }\n\n  /**\n   * @param {Record<string, any>} commandArgs\n   * @returns {{timeout?: number}}\n   */\n  normalizeTimeoutArgs(commandArgs) {\n    const normalizedArgs = {}\n\n    if (\"timeout\" in commandArgs && commandArgs.timeout !== undefined) {\n      normalizedArgs.timeout = Number(commandArgs.timeout)\n\n      if (Number.isNaN(normalizedArgs.timeout)) {\n        throw new Error(`Invalid timeout: ${commandArgs.timeout}`)\n      }\n    }\n\n    return normalizedArgs\n  }\n\n  /**\n   * @param {Record<string, any>} commandArgs\n   * @returns {import(\"./system-test.js\").FindArgs}\n   */\n  normalizeFindArgs(commandArgs) {\n    const findArgs = /** @type {import(\"./system-test.js\").FindArgs} */ (this.normalizeTimeoutArgs(commandArgs))\n\n    if (\"visible\" in commandArgs && commandArgs.visible !== undefined) {\n      if (commandArgs.visible === null || commandArgs.visible === \"null\") {\n        findArgs.visible = null\n      } else if (typeof commandArgs.visible === \"boolean\") {\n        findArgs.visible = commandArgs.visible\n      } else {\n        findArgs.visible = commandArgs.visible === \"true\"\n      }\n    }\n\n    if (\"useBaseSelector\" in commandArgs && commandArgs.useBaseSelector !== undefined) {\n      if (typeof commandArgs.useBaseSelector === \"boolean\") {\n        findArgs.useBaseSelector = commandArgs.useBaseSelector\n      } else {\n        findArgs.useBaseSelector = commandArgs.useBaseSelector === \"true\"\n      }\n    }\n\n    if (\"scrollTo\" in commandArgs && commandArgs.scrollTo !== undefined) {\n      if (typeof commandArgs.scrollTo === \"boolean\") {\n        findArgs.scrollTo = commandArgs.scrollTo\n      } else {\n        findArgs.scrollTo = commandArgs.scrollTo === \"true\"\n      }\n    }\n\n    return findArgs\n  }\n\n  /**\n   * @param {import(\"selenium-webdriver\").WebElement} element\n   * @returns {Promise<Record<string, any>>}\n   */\n  async serializeElement(element) {\n    const text = await element.getText()\n    const tagName = await element.getTagName()\n    const displayed = await element.isDisplayed()\n\n    return {displayed, tagName, text}\n  }\n\n  /**\n   * @param {string} command\n   * @param {Record<string, any>} commandArgs\n   * @returns {Promise<any>}\n   */\n  async run(command, commandArgs = {}) {\n    if (command === \"visit\") {\n      const path = commandArgs.path ?? commandArgs.url\n\n      if (!path) {\n        throw new Error(\"visit requires path or url\")\n      }\n\n      await this.browser.visit(path, this.normalizeTimeoutArgs(commandArgs))\n      return {ok: true}\n    }\n\n    if (command === \"dismissTo\") {\n      const path = commandArgs.path ?? commandArgs.url\n\n      if (!path) {\n        throw new Error(\"dismissTo requires path or url\")\n      }\n\n      await this.browser.dismissTo(path, this.normalizeTimeoutArgs(commandArgs))\n      return {ok: true}\n    }\n\n    if (command === \"setBaseSelector\") {\n      if (!commandArgs.selector) {\n        throw new Error(\"setBaseSelector requires selector\")\n      }\n\n      this.browser.setBaseSelector(commandArgs.selector)\n      return {ok: true}\n    }\n\n    if (command === \"getCurrentUrl\") {\n      return {currentUrl: await this.browser.getCurrentUrl()}\n    }\n\n    if (command === \"getHTML\") {\n      return {html: await this.browser.getHTML()}\n    }\n\n    if (command === \"getBrowserLogs\") {\n      return {logs: await this.browser.getBrowserLogs()}\n    }\n\n    if (command === \"takeScreenshot\") {\n      return await this.browser.takeScreenshot()\n    }\n\n    if (command === \"find\") {\n      if (!commandArgs.selector) {\n        throw new Error(\"find requires selector\")\n      }\n\n      const element = await this.browser.find(commandArgs.selector, this.normalizeFindArgs(commandArgs))\n\n      return {element: await this.serializeElement(element)}\n    }\n\n    if (command === \"findByTestID\") {\n      const testID = commandArgs.testID ?? commandArgs.testId\n\n      if (!testID) {\n        throw new Error(\"findByTestID requires testID\")\n      }\n\n      const element = await this.browser.findByTestID(testID, this.normalizeFindArgs(commandArgs))\n\n      return {element: await this.serializeElement(element)}\n    }\n\n    if (command === \"click\") {\n      const selector = commandArgs.selector\n\n      if (!selector) {\n        throw new Error(\"click requires selector\")\n      }\n\n      await this.browser.click(selector, this.normalizeFindArgs(commandArgs))\n      return {ok: true}\n    }\n\n    if (command === \"waitForNoSelector\") {\n      if (!commandArgs.selector) {\n        throw new Error(\"waitForNoSelector requires selector\")\n      }\n\n      await this.browser.waitForNoSelector(commandArgs.selector, this.normalizeFindArgs(commandArgs))\n      return {ok: true}\n    }\n\n    if (command === \"expectNoElement\") {\n      if (!commandArgs.selector) {\n        throw new Error(\"expectNoElement requires selector\")\n      }\n\n      await this.browser.expectNoElement(commandArgs.selector, this.normalizeFindArgs(commandArgs))\n      return {ok: true}\n    }\n\n    if (command === \"executeScript\") {\n      const script = commandArgs.script\n\n      if (typeof script !== \"string\" || script.length === 0) {\n        throw new Error(\"executeScript requires script\")\n      }\n\n      const scriptArgs = Array.isArray(commandArgs.args) ? commandArgs.args : []\n      const result = await this.browser.executeScript(script, ...scriptArgs)\n\n      return {result}\n    }\n\n    if (command === \"addCookie\") {\n      const name = commandArgs.name\n\n      if (typeof name !== \"string\" || name.length === 0) {\n        throw new Error(\"addCookie requires name\")\n      }\n\n      const value = commandArgs.value\n\n      if (typeof value !== \"string\") {\n        throw new Error(\"addCookie requires string value\")\n      }\n\n      /** @type {{name: string, value: string, domain?: string, path?: string, secure?: boolean, httpOnly?: boolean, expiry?: number, sameSite?: \"Strict\" | \"Lax\" | \"None\"}} */\n      const cookie = {name, value}\n\n      if (typeof commandArgs.domain === \"string\" && commandArgs.domain.length > 0) cookie.domain = commandArgs.domain\n      if (typeof commandArgs.path === \"string\" && commandArgs.path.length > 0) cookie.path = commandArgs.path\n      if (commandArgs.secure !== undefined) cookie.secure = parseCookieBoolean(\"secure\", commandArgs.secure)\n      if (commandArgs.httpOnly !== undefined) cookie.httpOnly = parseCookieBoolean(\"httpOnly\", commandArgs.httpOnly)\n      if (commandArgs.expiry !== undefined) cookie.expiry = Number(commandArgs.expiry)\n      if (typeof commandArgs.sameSite === \"string\") cookie.sameSite = /** @type {\"Strict\" | \"Lax\" | \"None\"} */ (commandArgs.sameSite)\n\n      await this.browser.addCookie(cookie)\n\n      return {ok: true}\n    }\n\n    if (command === \"interact\") {\n      const selector = commandArgs.selector\n      const methodName = commandArgs.methodName ?? commandArgs.method\n      const methodArgs = Array.isArray(commandArgs.args) ? commandArgs.args : []\n      const interactArgs = /** @type {{selector: string} & import(\"./system-test.js\").InteractArgs} */ ({selector, ...this.normalizeFindArgs(commandArgs)})\n\n      if (!selector) {\n        throw new Error(\"interact requires selector\")\n      }\n\n      if (!methodName) {\n        throw new Error(\"interact requires methodName\")\n      }\n\n      if (\"withFallback\" in commandArgs && commandArgs.withFallback !== undefined) {\n        if (typeof commandArgs.withFallback === \"boolean\") {\n          interactArgs.withFallback = commandArgs.withFallback\n        } else {\n          interactArgs.withFallback = commandArgs.withFallback === \"true\"\n        }\n      }\n\n      const result = await this.browser.interact(interactArgs, methodName, ...methodArgs)\n\n      return {result}\n    }\n\n    throw new Error(`Unknown browser command: ${command}`)\n  }\n}\n"]}
|
package/build/browser.d.ts
CHANGED
|
@@ -1,23 +1,3 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @typedef {object} BrowserArgs
|
|
3
|
-
* @property {boolean} [debug] Enable debug logging.
|
|
4
|
-
* @property {BrowserDriverConfig} [driver] Driver configuration.
|
|
5
|
-
* @property {import("./system-test-communicator.js").default} [communicator] Optional command communicator for helper-driven navigation.
|
|
6
|
-
* @property {string} [screenshotsPath] Directory used for saved screenshots and browser artifacts.
|
|
7
|
-
*/
|
|
8
|
-
/**
|
|
9
|
-
* @typedef {object} BrowserDriverConfig
|
|
10
|
-
* @property {"selenium"|"appium"} [type] Driver implementation to use.
|
|
11
|
-
* @property {Record<string, any>} [options] Driver-specific options.
|
|
12
|
-
*/
|
|
13
|
-
/**
|
|
14
|
-
* @typedef {object} BrowserNavigationArgs
|
|
15
|
-
* @property {number} [timeout] Override the timeout for this navigation command.
|
|
16
|
-
*/
|
|
17
|
-
/**
|
|
18
|
-
* @typedef {object} BrowserPathWaitArgs
|
|
19
|
-
* @property {number} [timeout] Override the timeout for this path wait.
|
|
20
|
-
*/
|
|
21
1
|
/** Generic browser session wrapper around the configured driver. */
|
|
22
2
|
export default class Browser {
|
|
23
3
|
/** @param {BrowserArgs} [args] */
|
|
@@ -104,6 +84,27 @@ export default class Browser {
|
|
|
104
84
|
* @returns {Promise<void>}
|
|
105
85
|
*/
|
|
106
86
|
waitForPath(expectedPath: string, args?: BrowserPathWaitArgs): Promise<void>;
|
|
87
|
+
/**
|
|
88
|
+
* Waits until the current URL exactly matches the expected URL.
|
|
89
|
+
* @param {string} expectedUrl Exact URL expected.
|
|
90
|
+
* @param {BrowserCurrentUrlWaitArgs} [args] Optional timeout.
|
|
91
|
+
* @returns {Promise<void>}
|
|
92
|
+
*/
|
|
93
|
+
waitForCurrentUrl(expectedUrl: string, args?: BrowserCurrentUrlWaitArgs): Promise<void>;
|
|
94
|
+
/**
|
|
95
|
+
* Waits until the current URL contains a fragment.
|
|
96
|
+
* @param {string} expectedFragment Fragment that should appear.
|
|
97
|
+
* @param {BrowserCurrentUrlWaitArgs} [args] Optional timeout.
|
|
98
|
+
* @returns {Promise<void>}
|
|
99
|
+
*/
|
|
100
|
+
waitForUrlContains(expectedFragment: string, args?: BrowserCurrentUrlWaitArgs): Promise<void>;
|
|
101
|
+
/**
|
|
102
|
+
* Waits until the current URL does not contain a fragment.
|
|
103
|
+
* @param {string} unexpectedFragment Fragment that should disappear.
|
|
104
|
+
* @param {BrowserCurrentUrlWaitArgs} [args] Optional timeout.
|
|
105
|
+
* @returns {Promise<void>}
|
|
106
|
+
*/
|
|
107
|
+
waitForUrlExcludes(unexpectedFragment: string, args?: BrowserCurrentUrlWaitArgs): Promise<void>;
|
|
107
108
|
/**
|
|
108
109
|
* @param {string} selector
|
|
109
110
|
* @param {import("./system-test.js").FindArgs} [args]
|
|
@@ -152,6 +153,40 @@ export default class Browser {
|
|
|
152
153
|
clearAndSendKeys(elementOrIdentifier: import("selenium-webdriver").WebElement | string | ({
|
|
153
154
|
selector: string;
|
|
154
155
|
} & import("./system-test.js").InteractArgs), nextValue: string): Promise<void>;
|
|
156
|
+
/**
|
|
157
|
+
* Replaces an input-like element's value by test id.
|
|
158
|
+
* @param {string} testID Field `data-testid` to target.
|
|
159
|
+
* @param {string} nextValue Text to leave in the field.
|
|
160
|
+
* @param {BrowserTestIDInputArgs} [args] Optional lookup timeout.
|
|
161
|
+
* @returns {Promise<void>}
|
|
162
|
+
*/
|
|
163
|
+
replaceTestIDInputValue(testID: string, nextValue: string, args?: BrowserTestIDInputArgs): Promise<void>;
|
|
164
|
+
/**
|
|
165
|
+
* Waits until a test id contains expected visible text.
|
|
166
|
+
* @param {string} testID Element `data-testid` to inspect.
|
|
167
|
+
* @param {string} expectedText Fragment that must appear in the element text.
|
|
168
|
+
* @param {BrowserTextWaitArgs} [args] Optional timeout.
|
|
169
|
+
* @returns {Promise<void>}
|
|
170
|
+
*/
|
|
171
|
+
waitForTestIDText(testID: string, expectedText: string, args?: BrowserTextWaitArgs): Promise<void>;
|
|
172
|
+
/**
|
|
173
|
+
* Waits until a test id no longer contains excluded visible text.
|
|
174
|
+
* @param {string} testID Element `data-testid` to inspect.
|
|
175
|
+
* @param {string} excludedText Fragment that should disappear from the element text.
|
|
176
|
+
* @param {BrowserTextWaitArgs} [args] Optional timeout.
|
|
177
|
+
* @returns {Promise<void>}
|
|
178
|
+
*/
|
|
179
|
+
waitForTestIDTextExcludes(testID: string, excludedText: string, args?: BrowserTextWaitArgs): Promise<void>;
|
|
180
|
+
/**
|
|
181
|
+
* Asserts a rendered element has a CSS color from the expected palette.
|
|
182
|
+
* @param {string} testID Element `data-testid` to inspect.
|
|
183
|
+
* @param {string} propertyName CSS property to read.
|
|
184
|
+
* @param {string} expectedRgb Expected RGB fragment.
|
|
185
|
+
* @param {string} lightRgb Disallowed RGB fragment.
|
|
186
|
+
* @param {string} description Human-readable element description.
|
|
187
|
+
* @returns {Promise<void>}
|
|
188
|
+
*/
|
|
189
|
+
expectTestIDCssColor(testID: string, propertyName: string, expectedRgb: string, lightRgb: string, description: string): Promise<void>;
|
|
155
190
|
/**
|
|
156
191
|
* Scrolls an element into view.
|
|
157
192
|
* @param {import("selenium-webdriver").WebElement|string|{selector: string} & import("./system-test.js").FindArgs} elementOrIdentifier
|
|
@@ -194,6 +229,42 @@ export default class Browser {
|
|
|
194
229
|
driverVisit(path: string): Promise<void>;
|
|
195
230
|
/** @returns {Promise<void>} */
|
|
196
231
|
deleteAllCookies(): Promise<void>;
|
|
232
|
+
/**
|
|
233
|
+
* Add a cookie to the active driver session for the current document
|
|
234
|
+
* origin. Useful when an out-of-band login (curl, fetch, etc.) returned
|
|
235
|
+
* a `Set-Cookie` value and the test needs the browser to start
|
|
236
|
+
* authenticated without driving the sign-in UI.
|
|
237
|
+
*
|
|
238
|
+
* The driver must already be on a page whose origin/domain matches the
|
|
239
|
+
* cookie domain, otherwise Selenium will reject the call.
|
|
240
|
+
* @param {{name: string, value: string, domain?: string, path?: string, secure?: boolean, httpOnly?: boolean, expiry?: number, sameSite?: "Strict" | "Lax" | "None"}} cookie
|
|
241
|
+
* @returns {Promise<void>}
|
|
242
|
+
*/
|
|
243
|
+
addCookie(cookie: {
|
|
244
|
+
name: string;
|
|
245
|
+
value: string;
|
|
246
|
+
domain?: string;
|
|
247
|
+
path?: string;
|
|
248
|
+
secure?: boolean;
|
|
249
|
+
httpOnly?: boolean;
|
|
250
|
+
expiry?: number;
|
|
251
|
+
sameSite?: "Strict" | "Lax" | "None";
|
|
252
|
+
}): Promise<void>;
|
|
253
|
+
/**
|
|
254
|
+
* Run an arbitrary script in the active browser session and return the
|
|
255
|
+
* resolved value. `script` is the function body executed in the browser
|
|
256
|
+
* (`new Function("...")`-style); `args` are forwarded as `arguments[i]`.
|
|
257
|
+
* Asynchronous scripts must `return` a Promise, which Selenium awaits.
|
|
258
|
+
*
|
|
259
|
+
* Useful for verification flows that need to call into application code
|
|
260
|
+
* (e.g. `fetch("/development/sign-in", {...})`) without going through the
|
|
261
|
+
* UI, or to read browser state the existing finder/interact commands
|
|
262
|
+
* don't expose.
|
|
263
|
+
* @param {string} script
|
|
264
|
+
* @param {...any} args
|
|
265
|
+
* @returns {Promise<any>}
|
|
266
|
+
*/
|
|
267
|
+
executeScript(script: string, ...args: any[]): Promise<any>;
|
|
197
268
|
/**
|
|
198
269
|
* @param {string} type
|
|
199
270
|
* @param {string} path
|
|
@@ -282,3 +353,21 @@ export type BrowserPathWaitArgs = {
|
|
|
282
353
|
*/
|
|
283
354
|
timeout?: number | undefined;
|
|
284
355
|
};
|
|
356
|
+
export type BrowserTextWaitArgs = {
|
|
357
|
+
/**
|
|
358
|
+
* Override the timeout for this text wait.
|
|
359
|
+
*/
|
|
360
|
+
timeout?: number | undefined;
|
|
361
|
+
};
|
|
362
|
+
export type BrowserCurrentUrlWaitArgs = {
|
|
363
|
+
/**
|
|
364
|
+
* Override the timeout for this URL wait.
|
|
365
|
+
*/
|
|
366
|
+
timeout?: number | undefined;
|
|
367
|
+
};
|
|
368
|
+
export type BrowserTestIDInputArgs = {
|
|
369
|
+
/**
|
|
370
|
+
* Override timeout for the input lookup.
|
|
371
|
+
*/
|
|
372
|
+
timeout?: number | undefined;
|
|
373
|
+
};
|