system-testing 1.0.77 → 1.0.79
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 +220 -7
- package/build/browser-command-client.d.ts +19 -0
- package/build/browser-command-client.js +39 -0
- package/build/browser-command-runner.d.ts +27 -0
- package/build/browser-command-runner.js +144 -0
- package/build/browser-process.d.ts +45 -0
- package/build/browser-process.js +127 -0
- package/build/browser-registry.d.ts +34 -0
- package/build/browser-registry.js +110 -0
- package/build/browser.d.ts +222 -0
- package/build/browser.js +358 -0
- package/build/cli-helpers.d.ts +16 -0
- package/build/cli-helpers.js +141 -0
- package/build/cli.d.ts +2 -0
- package/build/cli.js +72 -0
- package/build/drivers/appium-driver.d.ts +3 -3
- package/build/drivers/appium-driver.js +2 -2
- package/build/drivers/webdriver-driver.d.ts +6 -6
- package/build/drivers/webdriver-driver.js +11 -10
- package/build/index.d.ts +9 -1
- package/build/index.js +10 -3
- package/build/system-test-browser-helper.d.ts +6 -12
- package/build/system-test-browser-helper.js +12 -13
- package/build/system-test.d.ts +6 -192
- package/build/system-test.js +7 -221
- package/build/use-system-test-expo.d.ts +16 -0
- package/build/use-system-test-expo.js +34 -0
- package/build/use-system-test-react-native.d.ts +25 -0
- package/build/use-system-test-react-native.js +20 -0
- package/build/use-system-test-shape-hook.d.ts +35 -0
- package/build/use-system-test-shape-hook.js +74 -0
- package/build/use-system-test.d.ts +19 -10
- package/build/use-system-test.js +26 -72
- package/package.json +10 -4
|
@@ -0,0 +1,127 @@
|
|
|
1
|
+
import Browser from "./browser.js";
|
|
2
|
+
import BrowserCommandRunner from "./browser-command-runner.js";
|
|
3
|
+
import BrowserRegistry from "./browser-registry.js";
|
|
4
|
+
import { WebSocketServer } from "ws";
|
|
5
|
+
/** Long-running browser daemon exposing browser commands over WebSocket. */
|
|
6
|
+
export default class BrowserProcess {
|
|
7
|
+
/**
|
|
8
|
+
* @param {object} args
|
|
9
|
+
* @param {string} args.name
|
|
10
|
+
* @param {Browser} [args.browser]
|
|
11
|
+
* @param {Record<string, any>} [args.browserArgs]
|
|
12
|
+
* @param {string} [args.baseUrl]
|
|
13
|
+
* @param {boolean} [args.debug]
|
|
14
|
+
* @param {number} [args.port]
|
|
15
|
+
*/
|
|
16
|
+
constructor({ name, browser, browserArgs = {}, baseUrl, debug = false, port = 0 }) {
|
|
17
|
+
/**
|
|
18
|
+
* @param {import("ws").WebSocket} ws
|
|
19
|
+
* @returns {void}
|
|
20
|
+
*/
|
|
21
|
+
this.onConnection = (ws) => {
|
|
22
|
+
ws.on("message", async (rawData) => {
|
|
23
|
+
const requestId = `${Date.now()}-${this.requestCount++}`;
|
|
24
|
+
try {
|
|
25
|
+
const payload = JSON.parse(rawData.toString());
|
|
26
|
+
const result = await this.handlePayload(payload);
|
|
27
|
+
ws.send(JSON.stringify({ ok: true, requestId, result, type: "browser-command-result" }));
|
|
28
|
+
}
|
|
29
|
+
catch (error) {
|
|
30
|
+
ws.send(JSON.stringify({
|
|
31
|
+
error: error instanceof Error ? error.message : String(error),
|
|
32
|
+
ok: false,
|
|
33
|
+
requestId,
|
|
34
|
+
type: "browser-command-result"
|
|
35
|
+
}));
|
|
36
|
+
}
|
|
37
|
+
});
|
|
38
|
+
};
|
|
39
|
+
this.name = name;
|
|
40
|
+
this.browser = browser ?? new Browser({ debug, ...browserArgs });
|
|
41
|
+
this.baseUrl = baseUrl;
|
|
42
|
+
this.debug = debug;
|
|
43
|
+
this.requestRunner = new BrowserCommandRunner({ browser: this.browser });
|
|
44
|
+
this.requestCount = 0;
|
|
45
|
+
this.port = port;
|
|
46
|
+
}
|
|
47
|
+
/** @returns {Promise<void>} */
|
|
48
|
+
async start() {
|
|
49
|
+
if (!this.name) {
|
|
50
|
+
throw new Error("Browser process requires a name");
|
|
51
|
+
}
|
|
52
|
+
if (this.baseUrl) {
|
|
53
|
+
this.browser.getDriverAdapter().setBaseUrl(this.baseUrl);
|
|
54
|
+
}
|
|
55
|
+
await this.browser.getDriverAdapter().start();
|
|
56
|
+
await this.browser.setTimeouts(10000);
|
|
57
|
+
this.wss = new WebSocketServer({ port: this.port });
|
|
58
|
+
await new Promise((resolve) => {
|
|
59
|
+
this.wss.once("listening", resolve);
|
|
60
|
+
});
|
|
61
|
+
const address = this.wss.address();
|
|
62
|
+
if (!address || typeof address === "string") {
|
|
63
|
+
throw new Error("Could not resolve browser process port");
|
|
64
|
+
}
|
|
65
|
+
this.port = address.port;
|
|
66
|
+
this.wss.on("connection", this.onConnection);
|
|
67
|
+
await BrowserRegistry.register({
|
|
68
|
+
baseUrl: this.baseUrl,
|
|
69
|
+
name: this.name,
|
|
70
|
+
pid: process.pid,
|
|
71
|
+
port: this.port,
|
|
72
|
+
startedAt: new Date().toISOString()
|
|
73
|
+
});
|
|
74
|
+
const stop = async () => {
|
|
75
|
+
await this.stop();
|
|
76
|
+
process.exit(0);
|
|
77
|
+
};
|
|
78
|
+
process.once("SIGINT", stop);
|
|
79
|
+
process.once("SIGTERM", stop);
|
|
80
|
+
}
|
|
81
|
+
/** @returns {Promise<void>} */
|
|
82
|
+
async stop() {
|
|
83
|
+
if (this.stopped) {
|
|
84
|
+
return;
|
|
85
|
+
}
|
|
86
|
+
this.stopped = true;
|
|
87
|
+
await BrowserRegistry.unregister(this.name);
|
|
88
|
+
if (this.wss) {
|
|
89
|
+
await new Promise((resolve, reject) => {
|
|
90
|
+
this.wss.close((error) => {
|
|
91
|
+
if (error) {
|
|
92
|
+
reject(error);
|
|
93
|
+
}
|
|
94
|
+
else {
|
|
95
|
+
resolve(undefined);
|
|
96
|
+
}
|
|
97
|
+
});
|
|
98
|
+
});
|
|
99
|
+
}
|
|
100
|
+
await this.browser.stopDriver();
|
|
101
|
+
}
|
|
102
|
+
/**
|
|
103
|
+
* @param {Record<string, any>} payload
|
|
104
|
+
* @returns {Promise<any>}
|
|
105
|
+
*/
|
|
106
|
+
async handlePayload(payload) {
|
|
107
|
+
if (payload.type !== "browser-command") {
|
|
108
|
+
throw new Error(`Unknown payload type: ${payload.type}`);
|
|
109
|
+
}
|
|
110
|
+
const command = payload.command;
|
|
111
|
+
const commandArgs = payload.args ? { ...payload.args } : {};
|
|
112
|
+
if (payload.url && !commandArgs.url) {
|
|
113
|
+
commandArgs.url = payload.url;
|
|
114
|
+
}
|
|
115
|
+
if (payload.path && !commandArgs.path) {
|
|
116
|
+
commandArgs.path = payload.path;
|
|
117
|
+
}
|
|
118
|
+
if (payload.selector && !commandArgs.selector) {
|
|
119
|
+
commandArgs.selector = payload.selector;
|
|
120
|
+
}
|
|
121
|
+
if (payload.testID && !commandArgs.testID) {
|
|
122
|
+
commandArgs.testID = payload.testID;
|
|
123
|
+
}
|
|
124
|
+
return await this.requestRunner.run(command, commandArgs);
|
|
125
|
+
}
|
|
126
|
+
}
|
|
127
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"browser-process.js","sourceRoot":"","sources":["../src/browser-process.js"],"names":[],"mappings":"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","sourcesContent":["import Browser from \"./browser.js\"\nimport BrowserCommandRunner from \"./browser-command-runner.js\"\nimport BrowserRegistry from \"./browser-registry.js\"\nimport {WebSocketServer} from \"ws\"\n\n/** Long-running browser daemon exposing browser commands over WebSocket. */\nexport default class BrowserProcess {\n  /**\n   * @param {object} args\n   * @param {string} args.name\n   * @param {Browser} [args.browser]\n   * @param {Record<string, any>} [args.browserArgs]\n   * @param {string} [args.baseUrl]\n   * @param {boolean} [args.debug]\n   * @param {number} [args.port]\n   */\n  constructor({name, browser, browserArgs = {}, baseUrl, debug = false, port = 0}) {\n    this.name = name\n    this.browser = browser ?? new Browser({debug, ...browserArgs})\n    this.baseUrl = baseUrl\n    this.debug = debug\n    this.requestRunner = new BrowserCommandRunner({browser: this.browser})\n    this.requestCount = 0\n    this.port = port\n  }\n\n  /** @returns {Promise<void>} */\n  async start() {\n    if (!this.name) {\n      throw new Error(\"Browser process requires a name\")\n    }\n\n    if (this.baseUrl) {\n      this.browser.getDriverAdapter().setBaseUrl(this.baseUrl)\n    }\n\n    await this.browser.getDriverAdapter().start()\n    await this.browser.setTimeouts(10000)\n\n    this.wss = new WebSocketServer({port: this.port})\n    await new Promise((resolve) => {\n      this.wss.once(\"listening\", resolve)\n    })\n\n    const address = this.wss.address()\n\n    if (!address || typeof address === \"string\") {\n      throw new Error(\"Could not resolve browser process port\")\n    }\n\n    this.port = address.port\n    this.wss.on(\"connection\", this.onConnection)\n\n    await BrowserRegistry.register({\n      baseUrl: this.baseUrl,\n      name: this.name,\n      pid: process.pid,\n      port: this.port,\n      startedAt: new Date().toISOString()\n    })\n\n    const stop = async () => {\n      await this.stop()\n      process.exit(0)\n    }\n\n    process.once(\"SIGINT\", stop)\n    process.once(\"SIGTERM\", stop)\n  }\n\n  /** @returns {Promise<void>} */\n  async stop() {\n    if (this.stopped) {\n      return\n    }\n\n    this.stopped = true\n    await BrowserRegistry.unregister(this.name)\n\n    if (this.wss) {\n      await new Promise((resolve, reject) => {\n        this.wss.close((error) => {\n          if (error) {\n            reject(error)\n          } else {\n            resolve(undefined)\n          }\n        })\n      })\n    }\n\n    await this.browser.stopDriver()\n  }\n\n  /**\n   * @param {import(\"ws\").WebSocket} ws\n   * @returns {void}\n   */\n  onConnection = (ws) => {\n    ws.on(\"message\", async (rawData) => {\n      const requestId = `${Date.now()}-${this.requestCount++}`\n\n      try {\n        const payload = JSON.parse(rawData.toString())\n        const result = await this.handlePayload(payload)\n\n        ws.send(JSON.stringify({ok: true, requestId, result, type: \"browser-command-result\"}))\n      } catch (error) {\n        ws.send(JSON.stringify({\n          error: error instanceof Error ? error.message : String(error),\n          ok: false,\n          requestId,\n          type: \"browser-command-result\"\n        }))\n      }\n    })\n  }\n\n  /**\n   * @param {Record<string, any>} payload\n   * @returns {Promise<any>}\n   */\n  async handlePayload(payload) {\n    if (payload.type !== \"browser-command\") {\n      throw new Error(`Unknown payload type: ${payload.type}`)\n    }\n\n    const command = payload.command\n    const commandArgs = payload.args ? {...payload.args} : {}\n\n    if (payload.url && !commandArgs.url) {\n      commandArgs.url = payload.url\n    }\n\n    if (payload.path && !commandArgs.path) {\n      commandArgs.path = payload.path\n    }\n\n    if (payload.selector && !commandArgs.selector) {\n      commandArgs.selector = payload.selector\n    }\n\n    if (payload.testID && !commandArgs.testID) {\n      commandArgs.testID = payload.testID\n    }\n\n    return await this.requestRunner.run(command, commandArgs)\n  }\n}\n"]}
|
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
/** Browser process registry. */
|
|
2
|
+
export default class BrowserRegistry {
|
|
3
|
+
/** @returns {string} */
|
|
4
|
+
static getRegistryPath(): string;
|
|
5
|
+
/** @returns {Promise<Array<Record<string, any>>>} */
|
|
6
|
+
static list(): Promise<Array<Record<string, any>>>;
|
|
7
|
+
/**
|
|
8
|
+
* @param {Record<string, any>} entry
|
|
9
|
+
* @returns {Promise<void>}
|
|
10
|
+
*/
|
|
11
|
+
static register(entry: Record<string, any>): Promise<void>;
|
|
12
|
+
/**
|
|
13
|
+
* @param {string} name
|
|
14
|
+
* @returns {Promise<void>}
|
|
15
|
+
*/
|
|
16
|
+
static unregister(name: string): Promise<void>;
|
|
17
|
+
/**
|
|
18
|
+
* @param {string} [name]
|
|
19
|
+
* @returns {Promise<Record<string, any>>}
|
|
20
|
+
*/
|
|
21
|
+
static resolve(name?: string): Promise<Record<string, any>>;
|
|
22
|
+
/**
|
|
23
|
+
* @param {number} pid
|
|
24
|
+
* @returns {boolean}
|
|
25
|
+
*/
|
|
26
|
+
static isProcessAlive(pid: number): boolean;
|
|
27
|
+
/** @returns {Promise<Array<Record<string, any>>>} */
|
|
28
|
+
static _readRegistry(): Promise<Array<Record<string, any>>>;
|
|
29
|
+
/**
|
|
30
|
+
* @param {Array<Record<string, any>>} entries
|
|
31
|
+
* @returns {Promise<void>}
|
|
32
|
+
*/
|
|
33
|
+
static _writeRegistry(entries: Array<Record<string, any>>): Promise<void>;
|
|
34
|
+
}
|
|
@@ -0,0 +1,110 @@
|
|
|
1
|
+
import fs from "node:fs/promises";
|
|
2
|
+
import os from "node:os";
|
|
3
|
+
import path from "node:path";
|
|
4
|
+
const registryPath = path.join(os.tmpdir(), "system-testing-browser-registry.json");
|
|
5
|
+
/** Browser process registry. */
|
|
6
|
+
export default class BrowserRegistry {
|
|
7
|
+
/** @returns {string} */
|
|
8
|
+
static getRegistryPath() {
|
|
9
|
+
return registryPath;
|
|
10
|
+
}
|
|
11
|
+
/** @returns {Promise<Array<Record<string, any>>>} */
|
|
12
|
+
static async list() {
|
|
13
|
+
const entries = await this._readRegistry();
|
|
14
|
+
const aliveEntries = [];
|
|
15
|
+
let dirty = false;
|
|
16
|
+
for (const entry of entries) {
|
|
17
|
+
if (this.isProcessAlive(entry.pid)) {
|
|
18
|
+
aliveEntries.push(entry);
|
|
19
|
+
}
|
|
20
|
+
else {
|
|
21
|
+
dirty = true;
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
if (dirty) {
|
|
25
|
+
await this._writeRegistry(aliveEntries);
|
|
26
|
+
}
|
|
27
|
+
return aliveEntries;
|
|
28
|
+
}
|
|
29
|
+
/**
|
|
30
|
+
* @param {Record<string, any>} entry
|
|
31
|
+
* @returns {Promise<void>}
|
|
32
|
+
*/
|
|
33
|
+
static async register(entry) {
|
|
34
|
+
const entries = await this.list();
|
|
35
|
+
const filteredEntries = entries.filter((existingEntry) => existingEntry.name !== entry.name);
|
|
36
|
+
filteredEntries.push(entry);
|
|
37
|
+
await this._writeRegistry(filteredEntries);
|
|
38
|
+
}
|
|
39
|
+
/**
|
|
40
|
+
* @param {string} name
|
|
41
|
+
* @returns {Promise<void>}
|
|
42
|
+
*/
|
|
43
|
+
static async unregister(name) {
|
|
44
|
+
const entries = await this.list();
|
|
45
|
+
const filteredEntries = entries.filter((entry) => entry.name !== name);
|
|
46
|
+
await this._writeRegistry(filteredEntries);
|
|
47
|
+
}
|
|
48
|
+
/**
|
|
49
|
+
* @param {string} [name]
|
|
50
|
+
* @returns {Promise<Record<string, any>>}
|
|
51
|
+
*/
|
|
52
|
+
static async resolve(name) {
|
|
53
|
+
const entries = await this.list();
|
|
54
|
+
if (name) {
|
|
55
|
+
const entry = entries.find((candidate) => candidate.name === name);
|
|
56
|
+
if (!entry) {
|
|
57
|
+
throw new Error(`No running browser process found with name: ${name}`);
|
|
58
|
+
}
|
|
59
|
+
return entry;
|
|
60
|
+
}
|
|
61
|
+
if (entries.length === 1) {
|
|
62
|
+
return entries[0];
|
|
63
|
+
}
|
|
64
|
+
if (entries.length === 0) {
|
|
65
|
+
throw new Error("No running browser processes found");
|
|
66
|
+
}
|
|
67
|
+
throw new Error(`Multiple browser processes are running (${entries.length}); pass --name`);
|
|
68
|
+
}
|
|
69
|
+
/**
|
|
70
|
+
* @param {number} pid
|
|
71
|
+
* @returns {boolean}
|
|
72
|
+
*/
|
|
73
|
+
static isProcessAlive(pid) {
|
|
74
|
+
if (!pid || typeof pid !== "number") {
|
|
75
|
+
return false;
|
|
76
|
+
}
|
|
77
|
+
try {
|
|
78
|
+
process.kill(pid, 0);
|
|
79
|
+
return true;
|
|
80
|
+
}
|
|
81
|
+
catch {
|
|
82
|
+
return false;
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
/** @returns {Promise<Array<Record<string, any>>>} */
|
|
86
|
+
static async _readRegistry() {
|
|
87
|
+
try {
|
|
88
|
+
const fileContent = await fs.readFile(this.getRegistryPath(), "utf8");
|
|
89
|
+
const parsed = JSON.parse(fileContent);
|
|
90
|
+
if (!Array.isArray(parsed)) {
|
|
91
|
+
return [];
|
|
92
|
+
}
|
|
93
|
+
return parsed;
|
|
94
|
+
}
|
|
95
|
+
catch (error) {
|
|
96
|
+
if (error instanceof Error && "code" in error && error.code === "ENOENT") {
|
|
97
|
+
return [];
|
|
98
|
+
}
|
|
99
|
+
throw error;
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
/**
|
|
103
|
+
* @param {Array<Record<string, any>>} entries
|
|
104
|
+
* @returns {Promise<void>}
|
|
105
|
+
*/
|
|
106
|
+
static async _writeRegistry(entries) {
|
|
107
|
+
await fs.writeFile(this.getRegistryPath(), JSON.stringify(entries, null, 2));
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"browser-registry.js","sourceRoot":"","sources":["../src/browser-registry.js"],"names":[],"mappings":"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","sourcesContent":["import fs from \"node:fs/promises\"\nimport os from \"node:os\"\nimport path from \"node:path\"\n\nconst registryPath = path.join(os.tmpdir(), \"system-testing-browser-registry.json\")\n\n/** Browser process registry. */\nexport default class BrowserRegistry {\n  /** @returns {string} */\n  static getRegistryPath() {\n    return registryPath\n  }\n\n  /** @returns {Promise<Array<Record<string, any>>>} */\n  static async list() {\n    const entries = await this._readRegistry()\n    const aliveEntries = []\n    let dirty = false\n\n    for (const entry of entries) {\n      if (this.isProcessAlive(entry.pid)) {\n        aliveEntries.push(entry)\n      } else {\n        dirty = true\n      }\n    }\n\n    if (dirty) {\n      await this._writeRegistry(aliveEntries)\n    }\n\n    return aliveEntries\n  }\n\n  /**\n   * @param {Record<string, any>} entry\n   * @returns {Promise<void>}\n   */\n  static async register(entry) {\n    const entries = await this.list()\n    const filteredEntries = entries.filter((existingEntry) => existingEntry.name !== entry.name)\n\n    filteredEntries.push(entry)\n    await this._writeRegistry(filteredEntries)\n  }\n\n  /**\n   * @param {string} name\n   * @returns {Promise<void>}\n   */\n  static async unregister(name) {\n    const entries = await this.list()\n    const filteredEntries = entries.filter((entry) => entry.name !== name)\n\n    await this._writeRegistry(filteredEntries)\n  }\n\n  /**\n   * @param {string} [name]\n   * @returns {Promise<Record<string, any>>}\n   */\n  static async resolve(name) {\n    const entries = await this.list()\n\n    if (name) {\n      const entry = entries.find((candidate) => candidate.name === name)\n\n      if (!entry) {\n        throw new Error(`No running browser process found with name: ${name}`)\n      }\n\n      return entry\n    }\n\n    if (entries.length === 1) {\n      return entries[0]\n    }\n\n    if (entries.length === 0) {\n      throw new Error(\"No running browser processes found\")\n    }\n\n    throw new Error(`Multiple browser processes are running (${entries.length}); pass --name`)\n  }\n\n  /**\n   * @param {number} pid\n   * @returns {boolean}\n   */\n  static isProcessAlive(pid) {\n    if (!pid || typeof pid !== \"number\") {\n      return false\n    }\n\n    try {\n      process.kill(pid, 0)\n      return true\n    } catch {\n      return false\n    }\n  }\n\n  /** @returns {Promise<Array<Record<string, any>>>} */\n  static async _readRegistry() {\n    try {\n      const fileContent = await fs.readFile(this.getRegistryPath(), \"utf8\")\n      const parsed = JSON.parse(fileContent)\n\n      if (!Array.isArray(parsed)) {\n        return []\n      }\n\n      return parsed\n    } catch (error) {\n      if (error instanceof Error && \"code\" in error && error.code === \"ENOENT\") {\n        return []\n      }\n\n      throw error\n    }\n  }\n\n  /**\n   * @param {Array<Record<string, any>>} entries\n   * @returns {Promise<void>}\n   */\n  static async _writeRegistry(entries) {\n    await fs.writeFile(this.getRegistryPath(), JSON.stringify(entries, null, 2))\n  }\n}\n"]}
|
|
@@ -0,0 +1,222 @@
|
|
|
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
|
+
/** Generic browser session wrapper around the configured driver. */
|
|
14
|
+
export default class Browser {
|
|
15
|
+
/** @param {BrowserArgs} [args] */
|
|
16
|
+
constructor({ debug, driver, communicator, screenshotsPath, ...restArgs }?: BrowserArgs);
|
|
17
|
+
/** @type {import("selenium-webdriver").WebDriver | undefined} */
|
|
18
|
+
driver: import("selenium-webdriver").WebDriver | undefined;
|
|
19
|
+
/** @type {import("./drivers/webdriver-driver.js").default | undefined} */
|
|
20
|
+
driverAdapter: import("./drivers/webdriver-driver.js").default | undefined;
|
|
21
|
+
_debug: boolean;
|
|
22
|
+
/** @type {BrowserDriverConfig | undefined} */
|
|
23
|
+
_driverConfig: BrowserDriverConfig | undefined;
|
|
24
|
+
/** @type {Error | undefined} */
|
|
25
|
+
_httpServerError: Error | undefined;
|
|
26
|
+
_screenshotsPath: string;
|
|
27
|
+
communicator: import("./system-test-communicator.js").default;
|
|
28
|
+
/**
|
|
29
|
+
* @param {BrowserDriverConfig} [driverConfig]
|
|
30
|
+
* @returns {import("./drivers/webdriver-driver.js").default}
|
|
31
|
+
*/
|
|
32
|
+
createDriver(driverConfig?: BrowserDriverConfig): import("./drivers/webdriver-driver.js").default;
|
|
33
|
+
/**
|
|
34
|
+
* @param {import("./system-test-communicator.js").default | undefined} communicator
|
|
35
|
+
* @returns {void}
|
|
36
|
+
*/
|
|
37
|
+
setCommunicator(communicator: import("./system-test-communicator.js").default | undefined): void;
|
|
38
|
+
/** @returns {boolean} */
|
|
39
|
+
communicatorExists(): boolean;
|
|
40
|
+
/**
|
|
41
|
+
* @param {string} baseSelector
|
|
42
|
+
* @returns {void}
|
|
43
|
+
*/
|
|
44
|
+
setBaseSelector(baseSelector: string): void;
|
|
45
|
+
_baseSelector: string;
|
|
46
|
+
/** @returns {string | undefined} */
|
|
47
|
+
getBaseSelector(): string | undefined;
|
|
48
|
+
/**
|
|
49
|
+
* @param {string} selector
|
|
50
|
+
* @returns {string}
|
|
51
|
+
*/
|
|
52
|
+
getSelector(selector: string): string;
|
|
53
|
+
/**
|
|
54
|
+
* @param {...any} args
|
|
55
|
+
* @returns {void}
|
|
56
|
+
*/
|
|
57
|
+
debugError(...args: any[]): void;
|
|
58
|
+
/**
|
|
59
|
+
* @param {...any} args
|
|
60
|
+
* @returns {void}
|
|
61
|
+
*/
|
|
62
|
+
debugLog(...args: any[]): void;
|
|
63
|
+
/** @returns {void} */
|
|
64
|
+
throwIfHttpServerError(): void;
|
|
65
|
+
/**
|
|
66
|
+
* @param {Error} error
|
|
67
|
+
* @returns {void}
|
|
68
|
+
*/
|
|
69
|
+
onHttpServerError: (error: Error) => void;
|
|
70
|
+
/** @returns {import("selenium-webdriver").WebDriver} */
|
|
71
|
+
getDriver(): import("selenium-webdriver").WebDriver;
|
|
72
|
+
/** @returns {import("./drivers/webdriver-driver.js").default} */
|
|
73
|
+
getDriverAdapter(): import("./drivers/webdriver-driver.js").default;
|
|
74
|
+
/** @returns {number} */
|
|
75
|
+
getTimeouts(): number;
|
|
76
|
+
/** @returns {Promise<void>} */
|
|
77
|
+
restoreTimeouts(): Promise<void>;
|
|
78
|
+
/**
|
|
79
|
+
* @param {number} newTimeout
|
|
80
|
+
* @returns {Promise<void>}
|
|
81
|
+
*/
|
|
82
|
+
driverSetTimeouts(newTimeout: number): Promise<void>;
|
|
83
|
+
/**
|
|
84
|
+
* @param {number} newTimeout
|
|
85
|
+
* @returns {Promise<void>}
|
|
86
|
+
*/
|
|
87
|
+
setTimeouts(newTimeout: number): Promise<void>;
|
|
88
|
+
/** @returns {Promise<string[]>} */
|
|
89
|
+
getBrowserLogs(): Promise<string[]>;
|
|
90
|
+
/** @returns {Promise<string>} */
|
|
91
|
+
getCurrentUrl(): Promise<string>;
|
|
92
|
+
/**
|
|
93
|
+
* @param {string} selector
|
|
94
|
+
* @param {import("./system-test.js").FindArgs} [args]
|
|
95
|
+
* @returns {Promise<import("selenium-webdriver").WebElement[]>}
|
|
96
|
+
*/
|
|
97
|
+
all(selector: string, args?: import("./system-test.js").FindArgs): Promise<import("selenium-webdriver").WebElement[]>;
|
|
98
|
+
/**
|
|
99
|
+
* @param {string} selector
|
|
100
|
+
* @param {import("./system-test.js").FindArgs} [args]
|
|
101
|
+
* @returns {Promise<import("selenium-webdriver").WebElement>}
|
|
102
|
+
*/
|
|
103
|
+
find(selector: string, args?: import("./system-test.js").FindArgs): Promise<import("selenium-webdriver").WebElement>;
|
|
104
|
+
/**
|
|
105
|
+
* @param {string} testID
|
|
106
|
+
* @param {import("./system-test.js").FindArgs} [args]
|
|
107
|
+
* @returns {Promise<import("selenium-webdriver").WebElement>}
|
|
108
|
+
*/
|
|
109
|
+
findByTestID(testID: string, args?: import("./system-test.js").FindArgs): Promise<import("selenium-webdriver").WebElement>;
|
|
110
|
+
/**
|
|
111
|
+
* @param {string} selector
|
|
112
|
+
* @param {import("./system-test.js").FindArgs} [args]
|
|
113
|
+
* @returns {Promise<import("selenium-webdriver").WebElement>}
|
|
114
|
+
*/
|
|
115
|
+
findNoWait(selector: string, args?: import("./system-test.js").FindArgs): Promise<import("selenium-webdriver").WebElement>;
|
|
116
|
+
/**
|
|
117
|
+
* @param {string | import("selenium-webdriver").WebElement} elementOrIdentifier
|
|
118
|
+
* @param {import("./system-test.js").FindArgs} [args]
|
|
119
|
+
* @returns {Promise<void>}
|
|
120
|
+
*/
|
|
121
|
+
click(elementOrIdentifier: string | import("selenium-webdriver").WebElement, args?: import("./system-test.js").FindArgs): Promise<void>;
|
|
122
|
+
/**
|
|
123
|
+
* @param {import("selenium-webdriver").WebElement|string|{selector: string} & import("./system-test.js").FindArgs} elementOrIdentifier
|
|
124
|
+
* @param {string} methodName
|
|
125
|
+
* @param {...any} args
|
|
126
|
+
* @returns {Promise<any>}
|
|
127
|
+
*/
|
|
128
|
+
interact(elementOrIdentifier: import("selenium-webdriver").WebElement | string | ({
|
|
129
|
+
selector: string;
|
|
130
|
+
} & import("./system-test.js").FindArgs), methodName: string, ...args: any[]): Promise<any>;
|
|
131
|
+
/**
|
|
132
|
+
* @param {string} selector
|
|
133
|
+
* @param {import("./system-test.js").WaitForNoSelectorArgs} [args]
|
|
134
|
+
* @returns {Promise<void>}
|
|
135
|
+
*/
|
|
136
|
+
waitForNoSelector(selector: string, args?: import("./system-test.js").WaitForNoSelectorArgs): Promise<void>;
|
|
137
|
+
/**
|
|
138
|
+
* @param {string} selector
|
|
139
|
+
* @param {import("./system-test.js").FindArgs} [args]
|
|
140
|
+
* @returns {Promise<void>}
|
|
141
|
+
*/
|
|
142
|
+
expectNoElement(selector: string, args?: import("./system-test.js").FindArgs): Promise<void>;
|
|
143
|
+
/** @returns {Promise<string>} */
|
|
144
|
+
getHTML(): Promise<string>;
|
|
145
|
+
/**
|
|
146
|
+
* @param {string} path
|
|
147
|
+
* @returns {Promise<void>}
|
|
148
|
+
*/
|
|
149
|
+
driverVisit(path: string): Promise<void>;
|
|
150
|
+
/**
|
|
151
|
+
* @param {string} type
|
|
152
|
+
* @param {string} path
|
|
153
|
+
* @returns {Promise<void>}
|
|
154
|
+
*/
|
|
155
|
+
sendBrowserCommand(type: string, path: string): Promise<void>;
|
|
156
|
+
/**
|
|
157
|
+
* Visits a path using the injected browser helper when available, otherwise navigates directly with the driver.
|
|
158
|
+
* @param {string} path
|
|
159
|
+
* @returns {Promise<void>}
|
|
160
|
+
*/
|
|
161
|
+
visit(path: string): Promise<void>;
|
|
162
|
+
/**
|
|
163
|
+
* Dismisses to a path via the injected browser helper when available, otherwise navigates directly with the driver.
|
|
164
|
+
* @param {string} path
|
|
165
|
+
* @returns {Promise<void>}
|
|
166
|
+
*/
|
|
167
|
+
dismissTo(path: string): Promise<void>;
|
|
168
|
+
/**
|
|
169
|
+
* Formats browser logs for console output and truncates overly long output.
|
|
170
|
+
* @param {string[]} logs
|
|
171
|
+
* @param {number} [maxLines]
|
|
172
|
+
* @returns {string[]}
|
|
173
|
+
*/
|
|
174
|
+
formatBrowserLogsForConsole(logs: string[], maxLines?: number): string[];
|
|
175
|
+
/**
|
|
176
|
+
* @param {string[]} logs
|
|
177
|
+
* @returns {void}
|
|
178
|
+
*/
|
|
179
|
+
printBrowserLogsForFailure(logs: string[]): void;
|
|
180
|
+
/**
|
|
181
|
+
* Takes a screenshot, writes HTML/browser logs to disk, and returns the collected artifacts.
|
|
182
|
+
* @returns {Promise<{currentUrl: string, html: string, htmlPath: string, logs: string[], logsPath: string, screenshotPath: string}>}
|
|
183
|
+
*/
|
|
184
|
+
takeScreenshot(): Promise<{
|
|
185
|
+
currentUrl: string;
|
|
186
|
+
html: string;
|
|
187
|
+
htmlPath: string;
|
|
188
|
+
logs: string[];
|
|
189
|
+
logsPath: string;
|
|
190
|
+
screenshotPath: string;
|
|
191
|
+
}>;
|
|
192
|
+
/** @returns {Promise<void>} */
|
|
193
|
+
stopDriver(): Promise<void>;
|
|
194
|
+
}
|
|
195
|
+
export type BrowserArgs = {
|
|
196
|
+
/**
|
|
197
|
+
* Enable debug logging.
|
|
198
|
+
*/
|
|
199
|
+
debug?: boolean;
|
|
200
|
+
/**
|
|
201
|
+
* Driver configuration.
|
|
202
|
+
*/
|
|
203
|
+
driver?: BrowserDriverConfig;
|
|
204
|
+
/**
|
|
205
|
+
* Optional command communicator for helper-driven navigation.
|
|
206
|
+
*/
|
|
207
|
+
communicator?: import("./system-test-communicator.js").default;
|
|
208
|
+
/**
|
|
209
|
+
* Directory used for saved screenshots and browser artifacts.
|
|
210
|
+
*/
|
|
211
|
+
screenshotsPath?: string;
|
|
212
|
+
};
|
|
213
|
+
export type BrowserDriverConfig = {
|
|
214
|
+
/**
|
|
215
|
+
* Driver implementation to use.
|
|
216
|
+
*/
|
|
217
|
+
type?: "selenium" | "appium";
|
|
218
|
+
/**
|
|
219
|
+
* Driver-specific options.
|
|
220
|
+
*/
|
|
221
|
+
options?: Record<string, any>;
|
|
222
|
+
};
|