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