system-testing 1.0.78 → 1.0.81

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (35) hide show
  1. package/README.md +232 -6
  2. package/build/browser-command-client.d.ts +19 -0
  3. package/build/browser-command-client.js +39 -0
  4. package/build/browser-command-runner.d.ts +34 -0
  5. package/build/browser-command-runner.js +155 -0
  6. package/build/browser-daemon-constants.d.ts +2 -0
  7. package/build/browser-daemon-constants.js +3 -0
  8. package/build/browser-process.d.ts +45 -0
  9. package/build/browser-process.js +134 -0
  10. package/build/browser-registry.d.ts +44 -0
  11. package/build/browser-registry.js +191 -0
  12. package/build/browser.d.ts +240 -0
  13. package/build/browser.js +375 -0
  14. package/build/cli-helpers.d.ts +16 -0
  15. package/build/cli-helpers.js +177 -0
  16. package/build/cli.d.ts +2 -0
  17. package/build/cli.js +81 -0
  18. package/build/drivers/appium-driver.js +21 -21
  19. package/build/drivers/webdriver-driver.d.ts +4 -4
  20. package/build/drivers/webdriver-driver.js +32 -28
  21. package/build/index.d.ts +9 -1
  22. package/build/index.js +10 -3
  23. package/build/system-test-browser-helper.d.ts +6 -12
  24. package/build/system-test-browser-helper.js +12 -13
  25. package/build/system-test.d.ts +3 -189
  26. package/build/system-test.js +6 -220
  27. package/build/use-system-test-expo.d.ts +16 -0
  28. package/build/use-system-test-expo.js +34 -0
  29. package/build/use-system-test-react-native.d.ts +25 -0
  30. package/build/use-system-test-react-native.js +20 -0
  31. package/build/use-system-test-shape-hook.d.ts +35 -0
  32. package/build/use-system-test-shape-hook.js +74 -0
  33. package/build/use-system-test.d.ts +19 -10
  34. package/build/use-system-test.js +26 -72
  35. package/package.json +17 -8
@@ -0,0 +1,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
+ };