system-testing 1.0.69 → 1.0.71

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 CHANGED
@@ -74,8 +74,11 @@ import useSystemTest from "system-testing/build/use-system-test.js"
74
74
 
75
75
  export default function RootLayout() {
76
76
  const {enabled, systemTestBrowserHelper} = useSystemTest({
77
+ onFirstInitialize: () => {
78
+ // One-time setup the first time the helper initializes
79
+ },
77
80
  onInitialize: () => {
78
- // Reset any app state before tests run
81
+ // Reset any app state before each test run
79
82
  }
80
83
  })
81
84
 
@@ -96,7 +99,8 @@ export default function RootLayout() {
96
99
 
97
100
  Notes:
98
101
  - The hook auto-connects when the page is opened with `?systemTest=true` (as the runner does).
99
- - `onInitialize` runs once when the helper is ready; use it to reset globals/session.
102
+ - `onFirstInitialize` runs only on the first `initialize` command; use it for one-time setup.
103
+ - `onInitialize` is registered once when the helper is ready, but it runs on every `initialize` command (each `SystemTest.run`); use it to reset globals/session.
100
104
  - If you need scoundrel remote evaluation, wait for `systemTestBrowserHelper` and register your classes there, as shown in the commented snippet above.
101
105
  - Add a root wrapper with `testID="systemTestingComponent"` (and optionally `data-focussed="true"`) around your app so the runner has a stable element to detect and scope selectors against.
102
106
  - From your tests, use `await systemTest.getScoundrelClient()` to obtain the browser Scoundrel client for remote evaluation.
@@ -2,6 +2,7 @@ export default class SystemTestBrowserHelper {
2
2
  static current(): SystemTestBrowserHelper;
3
3
  communicator: SystemTestCommunicator;
4
4
  _enabled: boolean;
5
+ _hasInitialized: boolean;
5
6
  events: EventEmitter<string | symbol, any>;
6
7
  startScoundrel(): Promise<void>;
7
8
  scoundrelWs: WebSocket;
@@ -42,13 +43,9 @@ export default class SystemTestBrowserHelper {
42
43
  * @returns {void}
43
44
  */
44
45
  enableOnBrowser(): void;
45
- /**
46
- * @returns {boolean}
47
- */
46
+ /** @returns {boolean} */
48
47
  getEnabled(): boolean;
49
- /**
50
- * @returns {EventEmitter}
51
- */
48
+ /** @returns {EventEmitter} */
52
49
  getEvents(): EventEmitter;
53
50
  /**
54
51
  * @param {any[]} args
@@ -84,6 +81,12 @@ export default class SystemTestBrowserHelper {
84
81
  */
85
82
  onInitialize(callback: () => void): void;
86
83
  _onInitializeCallback: () => void;
84
+ /**
85
+ * @param {function() : void} callback
86
+ * @returns {void}
87
+ */
88
+ onFirstInitialize(callback: () => void): void;
89
+ _onFirstInitializeCallback: () => void;
87
90
  /**
88
91
  * @returns {void}
89
92
  */
@@ -43,6 +43,12 @@ export default class SystemTestBrowserHelper {
43
43
  this.onCommand = async ({ data }) => {
44
44
  if (data.type == "initialize") {
45
45
  this.events.emit("initialize");
46
+ if (!this._hasInitialized) {
47
+ this._hasInitialized = true;
48
+ if (this._onFirstInitializeCallback) {
49
+ await this._onFirstInitializeCallback();
50
+ }
51
+ }
46
52
  if (this._onInitializeCallback) {
47
53
  await this._onInitializeCallback();
48
54
  }
@@ -60,6 +66,7 @@ export default class SystemTestBrowserHelper {
60
66
  };
61
67
  this.communicator = new SystemTestCommunicator({ parent: this, onCommand: this.onCommand });
62
68
  this._enabled = false;
69
+ this._hasInitialized = false;
63
70
  this.events = new EventEmitter();
64
71
  shared.systemTestBrowserHelper = this;
65
72
  this.startScoundrel();
@@ -159,13 +166,9 @@ export default class SystemTestBrowserHelper {
159
166
  this.connectUnhandledRejection();
160
167
  this.overrideConsoleLog();
161
168
  }
162
- /**
163
- * @returns {boolean}
164
- */
169
+ /** @returns {boolean} */
165
170
  getEnabled() { return this._enabled; }
166
- /**
167
- * @returns {EventEmitter}
168
- */
171
+ /** @returns {EventEmitter} */
169
172
  getEvents() { return this.events; }
170
173
  /**
171
174
  * @param {any} arg
@@ -214,6 +217,13 @@ export default class SystemTestBrowserHelper {
214
217
  onInitialize(callback) {
215
218
  this._onInitializeCallback = callback;
216
219
  }
220
+ /**
221
+ * @param {function() : void} callback
222
+ * @returns {void}
223
+ */
224
+ onFirstInitialize(callback) {
225
+ this._onFirstInitializeCallback = callback;
226
+ }
217
227
  /**
218
228
  * @returns {void}
219
229
  */
@@ -235,4 +245,4 @@ export default class SystemTestBrowserHelper {
235
245
  return await this.communicator.sendCommand({ type: "query", sql });
236
246
  }
237
247
  }
238
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"system-test-browser-helper.js","sourceRoot":"","sources":["../src/system-test-browser-helper.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport Client from \"scoundrel-remote-eval/build/client/index.js\"\nimport ClientWebSocket from \"scoundrel-remote-eval/build/client/connections/web-socket/index.js\"\nimport {digg} from \"diggerize\"\nimport {EventEmitter} from \"eventemitter3\"\n\nimport SystemTestCommunicator from \"./system-test-communicator.js\"\n\n/** @type {{systemTestBrowserHelper: SystemTestBrowserHelper | null}} */\nconst shared = {\n  systemTestBrowserHelper: null\n}\n\nexport default class SystemTestBrowserHelper {\n  static current() {\n    if (!shared.systemTestBrowserHelper) {\n      throw new Error(\"No current SystemTestBrowserHelper set\")\n    }\n\n    return shared.systemTestBrowserHelper\n  }\n\n  constructor() {\n    this.communicator = new SystemTestCommunicator({parent: this, onCommand: this.onCommand})\n    this._enabled = false\n    this.events = new EventEmitter()\n\n    shared.systemTestBrowserHelper = this\n\n    this.startScoundrel()\n  }\n\n  async startScoundrel() {\n    this.scoundrelWs = new WebSocket(\"http://localhost:8090\")\n\n    // @ts-expect-error\n    this.scoundrelClientWebSocket = new ClientWebSocket(this.scoundrelWs)\n\n    await this.scoundrelClientWebSocket.waitForOpened()\n\n    this.scoundrelClient = new Client(this.scoundrelClientWebSocket, {enableServerControl: true})\n    this.events.emit(\"scoundrelStarted\")\n  }\n\n  waitForScoundrelStarted() {\n    return new Promise((resolve) => {\n      if (this.scoundrelClient) {\n        resolve(undefined)\n      } else {\n        this.events.once(\"scoundrelStarted\", () => {\n          resolve(undefined)\n        })\n      }\n    })\n  }\n\n  getScoundrel() {\n    if (!this.scoundrelClient) {\n      throw new Error(\"Scoundrel client is not started yet\")\n    }\n\n    return this.scoundrelClient\n  }\n\n  connectOnError() {\n    window.addEventListener(\"error\", (event) => {\n      this.handleError({\n        type: \"error\",\n        error: event.error,\n        errorClass: event.error?.name,\n        file: event.filename,\n        line: event.lineno,\n        message: event.message || \"Unknown error\",\n        url: window.location.href\n      })\n    })\n  }\n\n  connectUnhandledRejection() {\n    window.addEventListener(\"unhandledrejection\", (event) => {\n      this.handleError({\n        type: \"unhandledrejection\",\n        error: event.reason,\n        errorClass: \"UnhandledRejection\",\n        message: event.reason.message || event.reason || \"Unhandled promise rejection without a message\",\n        url: window.location.href\n      })\n    })\n  }\n\n  /**\n   * @param {object} data\n   * @param {string} [data.backtrace]\n   * @param {Error} [data.error]\n   * @param {string} [data.errorClass]\n   * @param {string} [data.file]\n   * @param {number} [data.line]\n   * @param {string} [data.message]\n   * @param {string} [data.type]\n   * @param {string} [data.url]\n   * @returns {void}\n   */\n  handleError(data) {\n    let backtrace\n\n    if (data.error && data.error.stack) {\n      backtrace = data.error.stack.split(\"\\n\")\n      backtrace.shift()\n      backtrace = backtrace.join(\"\\n\")\n    } else if (data.file) {\n      backtrace = `${data.file}:${data.line}`\n    }\n\n    data.backtrace = backtrace\n\n    this.communicator.sendCommand(data)\n  }\n\n  /**\n   * @returns {void}\n   */\n  connectWebSocket() {\n    this.ws = new WebSocket(\"ws://localhost:1985\")\n    this.communicator.ws = this.ws\n    this.ws.addEventListener(\"error\", digg(this, \"communicator\", \"onError\"))\n    this.ws.addEventListener(\"open\", digg(this, \"communicator\", \"onOpen\"))\n    this.ws.addEventListener(\"message\", (event) => this.communicator.onMessage(event.data))\n  }\n\n  /**\n   * @returns {void}\n   */\n  enableOnBrowser() {\n    this._enabled = true\n    this.connectWebSocket()\n    this.connectOnError()\n    this.connectUnhandledRejection()\n    this.overrideConsoleLog()\n  }\n\n  /**\n   * @returns {boolean}\n   */\n  getEnabled() { return this._enabled }\n\n  /**\n   * @returns {EventEmitter}\n   */\n  getEvents() { return this.events }\n\n  /**\n   * @param {any[]} args\n   * @returns {void}\n   */\n  fakeConsoleError = (...args) => {\n    this.communicator.sendCommand({type: \"console.error\", value: this.consoleLogMessage(args)})\n\n    if (this.originalConsoleError) {\n      return this.originalConsoleError(...args)\n    }\n  }\n\n  /**\n   * @param {any[]} args\n   * @returns {void}\n   */\n  fakeConsoleLog = (...args) => {\n    this.communicator.sendCommand({type: \"console.log\", value: this.consoleLogMessage(args)})\n\n    if (this.originalConsoleLog) {\n      return this.originalConsoleLog(...args)\n    }\n  }\n\n  /**\n   * @param {any} arg\n   * @param {any[]} [scannedObjects]\n   * @returns {any}\n   */\n  consoleLogMessage(arg, scannedObjects = []) {\n    if (Array.isArray(arg)) {\n      if (scannedObjects.includes(arg)) {\n        return \"[recursive]\"\n      } else {\n        scannedObjects.push(arg)\n      }\n\n      const result = []\n\n      for (const value of arg) {\n        result.push(this.consoleLogMessage(value, scannedObjects))\n      }\n\n      return result\n    } else if (Object.prototype.toString.call(arg) === '[object Object]') {\n      if (scannedObjects.includes(arg)) {\n        return \"[recursive]\"\n      } else {\n        scannedObjects.push(arg)\n      }\n\n      /** @type {Record<string, any>} */\n      const result = {}\n\n      for (const key in arg) {\n        result[key] = this.consoleLogMessage(arg[key], scannedObjects)\n      }\n\n      return result\n    } else if (typeof arg == \"object\") {\n      return `[object ${arg?.constructor?.name}]`\n    } else {\n      return arg\n    }\n  }\n\n  /**\n   * @param {{data: {path: string, type: string}}} args\n   * @returns {Promise<{result: string} | void>}\n   */\n  onCommand = async ({data}) => {\n    if (data.type == \"initialize\") {\n      this.events.emit(\"initialize\")\n\n      if (this._onInitializeCallback) {\n        await this._onInitializeCallback()\n      }\n\n      return {result: \"initialized\"}\n    } else if (data.type == \"visit\") {\n      this.events.emit(\"navigate\", {path: data.path})\n    } else if (data.type == \"dismissTo\") {\n      this.events.emit(\"dismissTo\", {path: data.path})\n    } else {\n      throw new Error(`Unknown command type for SystemTestBrowserHelper: ${data.type}`)\n    }\n  }\n\n  /**\n   * @param {function() : void} callback\n   * @returns {void}\n   */\n  onInitialize(callback) {\n    this._onInitializeCallback = callback\n  }\n\n  /**\n   * @returns {void}\n   */\n  overrideConsoleLog() {\n    if (this.originalConsoleError || this.originalConsoleLog) {\n      throw new Error(\"Console methods has already been overridden!\")\n    }\n\n    this.originalConsoleError = console.error\n    this.originalConsoleLog = console.log\n\n    console.error = this.fakeConsoleError\n    console.log = this.fakeConsoleLog\n  }\n\n  /**\n   * @param {string} sql\n   * @returns {Promise<Array<Record<string, any>>>}\n   */\n  async sendQuery(sql) {\n    // @ts-expect-error\n    return await this.communicator.sendCommand({type: \"query\", sql})\n  }\n}\n"]}
248
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"system-test-browser-helper.js","sourceRoot":"","sources":["../src/system-test-browser-helper.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport Client from \"scoundrel-remote-eval/build/client/index.js\"\nimport ClientWebSocket from \"scoundrel-remote-eval/build/client/connections/web-socket/index.js\"\nimport {digg} from \"diggerize\"\nimport {EventEmitter} from \"eventemitter3\"\n\nimport SystemTestCommunicator from \"./system-test-communicator.js\"\n\n/** @type {{systemTestBrowserHelper: SystemTestBrowserHelper | null}} */\nconst shared = {\n  systemTestBrowserHelper: null\n}\n\nexport default class SystemTestBrowserHelper {\n  static current() {\n    if (!shared.systemTestBrowserHelper) {\n      throw new Error(\"No current SystemTestBrowserHelper set\")\n    }\n\n    return shared.systemTestBrowserHelper\n  }\n\n  constructor() {\n    this.communicator = new SystemTestCommunicator({parent: this, onCommand: this.onCommand})\n    this._enabled = false\n    this._hasInitialized = false\n    this.events = new EventEmitter()\n\n    shared.systemTestBrowserHelper = this\n\n    this.startScoundrel()\n  }\n\n  async startScoundrel() {\n    this.scoundrelWs = new WebSocket(\"http://localhost:8090\")\n\n    // @ts-expect-error\n    this.scoundrelClientWebSocket = new ClientWebSocket(this.scoundrelWs)\n\n    await this.scoundrelClientWebSocket.waitForOpened()\n\n    this.scoundrelClient = new Client(this.scoundrelClientWebSocket, {enableServerControl: true})\n    this.events.emit(\"scoundrelStarted\")\n  }\n\n  waitForScoundrelStarted() {\n    return new Promise((resolve) => {\n      if (this.scoundrelClient) {\n        resolve(undefined)\n      } else {\n        this.events.once(\"scoundrelStarted\", () => {\n          resolve(undefined)\n        })\n      }\n    })\n  }\n\n  getScoundrel() {\n    if (!this.scoundrelClient) {\n      throw new Error(\"Scoundrel client is not started yet\")\n    }\n\n    return this.scoundrelClient\n  }\n\n  connectOnError() {\n    window.addEventListener(\"error\", (event) => {\n      this.handleError({\n        type: \"error\",\n        error: event.error,\n        errorClass: event.error?.name,\n        file: event.filename,\n        line: event.lineno,\n        message: event.message || \"Unknown error\",\n        url: window.location.href\n      })\n    })\n  }\n\n  connectUnhandledRejection() {\n    window.addEventListener(\"unhandledrejection\", (event) => {\n      this.handleError({\n        type: \"unhandledrejection\",\n        error: event.reason,\n        errorClass: \"UnhandledRejection\",\n        message: event.reason.message || event.reason || \"Unhandled promise rejection without a message\",\n        url: window.location.href\n      })\n    })\n  }\n\n  /**\n   * @param {object} data\n   * @param {string} [data.backtrace]\n   * @param {Error} [data.error]\n   * @param {string} [data.errorClass]\n   * @param {string} [data.file]\n   * @param {number} [data.line]\n   * @param {string} [data.message]\n   * @param {string} [data.type]\n   * @param {string} [data.url]\n   * @returns {void}\n   */\n  handleError(data) {\n    let backtrace\n\n    if (data.error && data.error.stack) {\n      backtrace = data.error.stack.split(\"\\n\")\n      backtrace.shift()\n      backtrace = backtrace.join(\"\\n\")\n    } else if (data.file) {\n      backtrace = `${data.file}:${data.line}`\n    }\n\n    data.backtrace = backtrace\n\n    this.communicator.sendCommand(data)\n  }\n\n  /**\n   * @returns {void}\n   */\n  connectWebSocket() {\n    this.ws = new WebSocket(\"ws://localhost:1985\")\n    this.communicator.ws = this.ws\n    this.ws.addEventListener(\"error\", digg(this, \"communicator\", \"onError\"))\n    this.ws.addEventListener(\"open\", digg(this, \"communicator\", \"onOpen\"))\n    this.ws.addEventListener(\"message\", (event) => this.communicator.onMessage(event.data))\n  }\n\n  /**\n   * @returns {void}\n   */\n  enableOnBrowser() {\n    this._enabled = true\n    this.connectWebSocket()\n    this.connectOnError()\n    this.connectUnhandledRejection()\n    this.overrideConsoleLog()\n  }\n\n  /** @returns {boolean} */\n  getEnabled() { return this._enabled }\n\n  /** @returns {EventEmitter} */\n  getEvents() { return this.events }\n\n  /**\n   * @param {any[]} args\n   * @returns {void}\n   */\n  fakeConsoleError = (...args) => {\n    this.communicator.sendCommand({type: \"console.error\", value: this.consoleLogMessage(args)})\n\n    if (this.originalConsoleError) {\n      return this.originalConsoleError(...args)\n    }\n  }\n\n  /**\n   * @param {any[]} args\n   * @returns {void}\n   */\n  fakeConsoleLog = (...args) => {\n    this.communicator.sendCommand({type: \"console.log\", value: this.consoleLogMessage(args)})\n\n    if (this.originalConsoleLog) {\n      return this.originalConsoleLog(...args)\n    }\n  }\n\n  /**\n   * @param {any} arg\n   * @param {any[]} [scannedObjects]\n   * @returns {any}\n   */\n  consoleLogMessage(arg, scannedObjects = []) {\n    if (Array.isArray(arg)) {\n      if (scannedObjects.includes(arg)) {\n        return \"[recursive]\"\n      } else {\n        scannedObjects.push(arg)\n      }\n\n      const result = []\n\n      for (const value of arg) {\n        result.push(this.consoleLogMessage(value, scannedObjects))\n      }\n\n      return result\n    } else if (Object.prototype.toString.call(arg) === '[object Object]') {\n      if (scannedObjects.includes(arg)) {\n        return \"[recursive]\"\n      } else {\n        scannedObjects.push(arg)\n      }\n\n      /** @type {Record<string, any>} */\n      const result = {}\n\n      for (const key in arg) {\n        result[key] = this.consoleLogMessage(arg[key], scannedObjects)\n      }\n\n      return result\n    } else if (typeof arg == \"object\") {\n      return `[object ${arg?.constructor?.name}]`\n    } else {\n      return arg\n    }\n  }\n\n  /**\n   * @param {{data: {path: string, type: string}}} args\n   * @returns {Promise<{result: string} | void>}\n   */\n  onCommand = async ({data}) => {\n    if (data.type == \"initialize\") {\n      this.events.emit(\"initialize\")\n\n      if (!this._hasInitialized) {\n        this._hasInitialized = true\n\n        if (this._onFirstInitializeCallback) {\n          await this._onFirstInitializeCallback()\n        }\n      }\n\n      if (this._onInitializeCallback) {\n        await this._onInitializeCallback()\n      }\n\n      return {result: \"initialized\"}\n    } else if (data.type == \"visit\") {\n      this.events.emit(\"navigate\", {path: data.path})\n    } else if (data.type == \"dismissTo\") {\n      this.events.emit(\"dismissTo\", {path: data.path})\n    } else {\n      throw new Error(`Unknown command type for SystemTestBrowserHelper: ${data.type}`)\n    }\n  }\n\n  /**\n   * @param {function() : void} callback\n   * @returns {void}\n   */\n  onInitialize(callback) {\n    this._onInitializeCallback = callback\n  }\n\n  /**\n   * @param {function() : void} callback\n   * @returns {void}\n   */\n  onFirstInitialize(callback) {\n    this._onFirstInitializeCallback = callback\n  }\n\n  /**\n   * @returns {void}\n   */\n  overrideConsoleLog() {\n    if (this.originalConsoleError || this.originalConsoleLog) {\n      throw new Error(\"Console methods has already been overridden!\")\n    }\n\n    this.originalConsoleError = console.error\n    this.originalConsoleLog = console.log\n\n    console.error = this.fakeConsoleError\n    console.log = this.fakeConsoleLog\n  }\n\n  /**\n   * @param {string} sql\n   * @returns {Promise<Array<Record<string, any>>>}\n   */\n  async sendQuery(sql) {\n    // @ts-expect-error\n    return await this.communicator.sendCommand({type: \"query\", sql})\n  }\n}\n"]}
@@ -1,19 +1,26 @@
1
1
  export default class SystemTestHttpServer {
2
2
  /**
3
- * @param {{host?: string, port?: number, debug?: boolean}} [args]
3
+ * @param {{host?: string, port?: number, debug?: boolean, onError?: (error: Error) => void, connectHost?: string}} [args]
4
4
  */
5
- constructor({ host, port, debug }?: {
5
+ constructor({ host, port, debug, onError, connectHost }?: {
6
6
  host?: string;
7
7
  port?: number;
8
8
  debug?: boolean;
9
+ onError?: (error: Error) => void;
10
+ connectHost?: string;
9
11
  });
10
12
  _host: string;
11
13
  _port: number;
12
14
  _debug: boolean;
15
+ _onError: (error: Error) => void;
16
+ _started: boolean;
17
+ _connectHost: string;
13
18
  /** @type {Set<import("node:net").Socket>} */
14
19
  _connections: Set<import("node:net").Socket>;
15
- /** @param {string} message */
16
- debugLog(message: string): void;
20
+ /** @param {any[]} args */
21
+ debugError(...args: any[]): void;
22
+ /** @param {any[]} args */
23
+ debugLog(...args: any[]): void;
17
24
  /** @returns {Promise<void>} */
18
25
  close(): Promise<void>;
19
26
  /**
@@ -27,5 +34,17 @@ export default class SystemTestHttpServer {
27
34
  /** @returns {Promise<void>} */
28
35
  startHttpServer(): Promise<void>;
29
36
  httpServer: http.Server<typeof http.IncomingMessage, typeof http.ServerResponse>;
37
+ /**
38
+ * @param {{timeoutMs?: number}} [args]
39
+ * @returns {Promise<void>}
40
+ */
41
+ assertReachable({ timeoutMs }?: {
42
+ timeoutMs?: number;
43
+ }): Promise<void>;
44
+ /**
45
+ * @param {unknown} error
46
+ * @returns {boolean}
47
+ */
48
+ isRetryableHealthCheckError(error: unknown): boolean;
30
49
  }
31
50
  import http from "node:http";
@@ -4,9 +4,9 @@ import http from "node:http";
4
4
  import mime from "mime";
5
5
  export default class SystemTestHttpServer {
6
6
  /**
7
- * @param {{host?: string, port?: number, debug?: boolean}} [args]
7
+ * @param {{host?: string, port?: number, debug?: boolean, onError?: (error: Error) => void, connectHost?: string}} [args]
8
8
  */
9
- constructor({ host = "localhost", port = 1984, debug = false } = {}) {
9
+ constructor({ host = "localhost", port = 1984, debug = false, onError, connectHost } = {}) {
10
10
  /**
11
11
  * @param {http.IncomingMessage} request
12
12
  * @param {http.ServerResponse} response
@@ -35,7 +35,17 @@ export default class SystemTestHttpServer {
35
35
  if (!fileExists) {
36
36
  filePath = `${process.cwd()}/dist/index.html`;
37
37
  }
38
- const fileContent = await fs.readFile(filePath);
38
+ let fileContent;
39
+ try {
40
+ fileContent = await fs.readFile(filePath);
41
+ }
42
+ catch (error) {
43
+ const errorMessage = error instanceof Error ? error.message : String(error);
44
+ this.debugError(`HTTP server read failed: ${errorMessage}`);
45
+ response.statusCode = 500;
46
+ response.end("Internal Server Error");
47
+ return;
48
+ }
39
49
  const mimeType = mime.getType(filePath);
40
50
  response.statusCode = 200;
41
51
  if (mimeType) {
@@ -46,13 +56,21 @@ export default class SystemTestHttpServer {
46
56
  this._host = host;
47
57
  this._port = port;
48
58
  this._debug = debug;
59
+ this._onError = onError;
60
+ this._started = false;
61
+ this._connectHost = connectHost ?? (host === "0.0.0.0" ? "127.0.0.1" : host);
49
62
  /** @type {Set<import("node:net").Socket>} */
50
63
  this._connections = new Set();
51
64
  }
52
- /** @param {string} message */
53
- debugLog(message) {
54
- if (this._debug)
55
- console.log(`[SystemTestHttpServer] ${message}`);
65
+ /** @param {any[]} args */
66
+ debugError(...args) {
67
+ console.log("[SystemTestHttpServer]", ...args);
68
+ }
69
+ /** @param {any[]} args */
70
+ debugLog(...args) {
71
+ if (this._debug) {
72
+ console.log("[SystemTestHttpServer]", ...args);
73
+ }
56
74
  }
57
75
  /** @returns {Promise<void>} */
58
76
  async close() {
@@ -67,10 +85,13 @@ export default class SystemTestHttpServer {
67
85
  }
68
86
  await new Promise((resolve, reject) => {
69
87
  this.httpServer.close((error) => {
70
- if (error)
88
+ if (error) {
71
89
  reject(error);
72
- else
90
+ }
91
+ else {
92
+ this._started = false;
73
93
  resolve();
94
+ }
74
95
  });
75
96
  });
76
97
  }
@@ -91,11 +112,75 @@ export default class SystemTestHttpServer {
91
112
  });
92
113
  });
93
114
  this.httpServer.on("error", (error) => {
94
- this.debugLog(`HTTP server error: ${error instanceof Error ? error.message : String(error)}`);
95
- reject(error);
115
+ const errorMessage = error instanceof Error ? error.message : String(error);
116
+ this.debugError(`HTTP server error: ${errorMessage}`);
117
+ if (this._started) {
118
+ if (this._onError) {
119
+ this._onError(error instanceof Error ? error : new Error(errorMessage));
120
+ }
121
+ }
122
+ else {
123
+ reject(error);
124
+ }
125
+ });
126
+ this.httpServer.listen(this._port, this._host, () => {
127
+ this._started = true;
128
+ resolve();
96
129
  });
97
- this.httpServer.listen(this._port, this._host, () => resolve());
98
130
  });
99
131
  }
132
+ /**
133
+ * @param {{timeoutMs?: number}} [args]
134
+ * @returns {Promise<void>}
135
+ */
136
+ async assertReachable({ timeoutMs = 5000 } = {}) {
137
+ const url = `http://${this._connectHost}:${this._port}/`;
138
+ const maxAttempts = 3;
139
+ for (let attempt = 1; attempt <= maxAttempts; attempt++) {
140
+ try {
141
+ await new Promise((resolve, reject) => {
142
+ const request = http.get(url, (response) => {
143
+ const chunks = [];
144
+ response.on("data", (chunk) => chunks.push(chunk));
145
+ response.on("end", () => {
146
+ const body = Buffer.concat(chunks).toString("utf8");
147
+ if (response.statusCode !== 200) {
148
+ reject(new Error(`HTTP server health check failed with status ${response.statusCode} for ${url}: ${body.slice(0, 200)}`));
149
+ return;
150
+ }
151
+ resolve(undefined);
152
+ });
153
+ });
154
+ request.on("error", (error) => reject(error));
155
+ request.setTimeout(timeoutMs, () => {
156
+ request.destroy(new Error(`HTTP server health check timed out after ${timeoutMs}ms for ${url}`));
157
+ });
158
+ });
159
+ return;
160
+ }
161
+ catch (error) {
162
+ if (attempt === maxAttempts || !this.isRetryableHealthCheckError(error)) {
163
+ throw error;
164
+ }
165
+ await new Promise((resolve) => setTimeout(resolve, 100));
166
+ }
167
+ }
168
+ }
169
+ /**
170
+ * @param {unknown} error
171
+ * @returns {boolean}
172
+ */
173
+ isRetryableHealthCheckError(error) {
174
+ if (error && typeof error === "object" && "code" in error) {
175
+ if (error.code === "ECONNRESET")
176
+ return true;
177
+ if (error.code === "ECONNREFUSED")
178
+ return true;
179
+ if (error.code === "ETIMEDOUT")
180
+ return true;
181
+ }
182
+ const message = error instanceof Error ? error.message : String(error);
183
+ return message.includes("socket hang up");
184
+ }
100
185
  }
101
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"system-test-http-server.js","sourceRoot":"","sources":["../src/system-test-http-server.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport fs from \"node:fs/promises\"\nimport http from \"node:http\"\nimport mime from \"mime\"\n\nexport default class SystemTestHttpServer {\n  /**\n   * @param {{host?: string, port?: number, debug?: boolean}} [args]\n   */\n  constructor({host = \"localhost\", port = 1984, debug = false} = {}) {\n    this._host = host\n    this._port = port\n    this._debug = debug\n    /** @type {Set<import(\"node:net\").Socket>} */\n    this._connections = new Set()\n  }\n\n  /** @param {string} message */\n  debugLog(message) {\n    if (this._debug) console.log(`[SystemTestHttpServer] ${message}`)\n  }\n\n  /** @returns {Promise<void>} */\n  async close() {\n    if (!this.httpServer) {\n      throw new Error(\"HTTP server is not initialized\")\n    }\n\n    if (this._connections.size > 0) {\n      for (const connection of this._connections) {\n        connection.destroy()\n      }\n      this._connections.clear()\n    }\n\n    await new Promise((resolve, reject) => {\n      this.httpServer.close((error) => {\n        if (error) reject(error)\n        else resolve()\n      })\n    })\n  }\n\n  /**\n   * @param {http.IncomingMessage} request\n   * @param {http.ServerResponse} response\n   * @returns {Promise<void>}\n   */\n  onHttpServerRequest = async (request, response) => {\n    if (!request.url) {\n      response.statusCode = 400\n      response.end(\"Bad Request\")\n      return\n    }\n\n    const baseUrl = `http://${request.headers.host || \"localhost:1984\"}`\n    const {pathname} = new URL(request.url, baseUrl)\n    let filePath = `${process.cwd()}/dist${pathname}`\n\n    if (filePath.endsWith(\"/\")) {\n      filePath += \"index.html\"\n    }\n\n    let fileExists\n\n    try {\n      await fs.stat(filePath)\n      fileExists = true\n    } catch (_error) { // eslint-disable-line no-unused-vars\n      fileExists = false\n    }\n\n    if (!fileExists) {\n      filePath = `${process.cwd()}/dist/index.html`\n    }\n\n    const fileContent = await fs.readFile(filePath)\n    const mimeType = mime.getType(filePath)\n\n    response.statusCode = 200\n\n    if (mimeType) {\n      response.setHeader(\"Content-Type\", mimeType)\n    }\n\n    response.end(fileContent)\n  }\n\n  /** @returns {Promise<void>} */\n  async start() {\n    this.debugLog(`Starting HTTP server on ${this._host}:${this._port}`)\n    await this.startHttpServer()\n    this.debugLog(`HTTP server started on ${this._host}:${this._port}`)\n  }\n\n  /** @returns {Promise<void>} */\n  startHttpServer() {\n    return new Promise((resolve, reject) => {\n      this.httpServer = http.createServer(this.onHttpServerRequest)\n      this.httpServer.on(\"connection\", (connection) => {\n        this._connections.add(connection)\n        connection.on(\"close\", () => {\n          this._connections.delete(connection)\n        })\n      })\n      this.httpServer.on(\"error\", (error) => {\n        this.debugLog(`HTTP server error: ${error instanceof Error ? error.message : String(error)}`)\n        reject(error)\n      })\n      this.httpServer.listen(this._port, this._host, () => resolve())\n    })\n  }\n}\n"]}
186
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"system-test-http-server.js","sourceRoot":"","sources":["../src/system-test-http-server.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport fs from \"node:fs/promises\"\nimport http from \"node:http\"\nimport mime from \"mime\"\n\nexport default class SystemTestHttpServer {\n  /**\n   * @param {{host?: string, port?: number, debug?: boolean, onError?: (error: Error) => void, connectHost?: string}} [args]\n   */\n  constructor({host = \"localhost\", port = 1984, debug = false, onError, connectHost} = {}) {\n    this._host = host\n    this._port = port\n    this._debug = debug\n    this._onError = onError\n    this._started = false\n    this._connectHost = connectHost ?? (host === \"0.0.0.0\" ? \"127.0.0.1\" : host)\n    /** @type {Set<import(\"node:net\").Socket>} */\n    this._connections = new Set()\n  }\n\n  /** @param {any[]} args */\n  debugError(...args) {\n    console.log(\"[SystemTestHttpServer]\", ...args)\n  }\n\n  /** @param {any[]} args */\n  debugLog(...args) {\n    if (this._debug) {\n      console.log(\"[SystemTestHttpServer]\", ...args)\n    }\n  }\n\n  /** @returns {Promise<void>} */\n  async close() {\n    if (!this.httpServer) {\n      throw new Error(\"HTTP server is not initialized\")\n    }\n\n    if (this._connections.size > 0) {\n      for (const connection of this._connections) {\n        connection.destroy()\n      }\n      this._connections.clear()\n    }\n\n    await new Promise((resolve, reject) => {\n      this.httpServer.close((error) => {\n        if (error) {\n          reject(error)\n        } else {\n          this._started = false\n          resolve()\n        }\n      })\n    })\n  }\n\n  /**\n   * @param {http.IncomingMessage} request\n   * @param {http.ServerResponse} response\n   * @returns {Promise<void>}\n   */\n  onHttpServerRequest = async (request, response) => {\n    if (!request.url) {\n      response.statusCode = 400\n      response.end(\"Bad Request\")\n      return\n    }\n\n    const baseUrl = `http://${request.headers.host || \"localhost:1984\"}`\n    const {pathname} = new URL(request.url, baseUrl)\n    let filePath = `${process.cwd()}/dist${pathname}`\n\n    if (filePath.endsWith(\"/\")) {\n      filePath += \"index.html\"\n    }\n\n    let fileExists\n\n    try {\n      await fs.stat(filePath)\n      fileExists = true\n    } catch (_error) { // eslint-disable-line no-unused-vars\n      fileExists = false\n    }\n\n    if (!fileExists) {\n      filePath = `${process.cwd()}/dist/index.html`\n    }\n\n    let fileContent\n\n    try {\n      fileContent = await fs.readFile(filePath)\n    } catch (error) {\n      const errorMessage = error instanceof Error ? error.message : String(error)\n\n      this.debugError(`HTTP server read failed: ${errorMessage}`)\n      response.statusCode = 500\n      response.end(\"Internal Server Error\")\n      return\n    }\n\n    const mimeType = mime.getType(filePath)\n\n    response.statusCode = 200\n\n    if (mimeType) {\n      response.setHeader(\"Content-Type\", mimeType)\n    }\n\n    response.end(fileContent)\n  }\n\n  /** @returns {Promise<void>} */\n  async start() {\n    this.debugLog(`Starting HTTP server on ${this._host}:${this._port}`)\n    await this.startHttpServer()\n    this.debugLog(`HTTP server started on ${this._host}:${this._port}`)\n  }\n\n  /** @returns {Promise<void>} */\n  startHttpServer() {\n    return new Promise((resolve, reject) => {\n      this.httpServer = http.createServer(this.onHttpServerRequest)\n      this.httpServer.on(\"connection\", (connection) => {\n        this._connections.add(connection)\n        connection.on(\"close\", () => {\n          this._connections.delete(connection)\n        })\n      })\n      this.httpServer.on(\"error\", (error) => {\n        const errorMessage = error instanceof Error ? error.message : String(error)\n\n        this.debugError(`HTTP server error: ${errorMessage}`)\n\n        if (this._started) {\n          if (this._onError) {\n            this._onError(error instanceof Error ? error : new Error(errorMessage))\n          }\n        } else {\n          reject(error)\n        }\n      })\n      this.httpServer.listen(this._port, this._host, () => {\n        this._started = true\n        resolve()\n      })\n    })\n  }\n\n  /**\n   * @param {{timeoutMs?: number}} [args]\n   * @returns {Promise<void>}\n   */\n  async assertReachable({timeoutMs = 5000} = {}) {\n    const url = `http://${this._connectHost}:${this._port}/`\n    const maxAttempts = 3\n\n    for (let attempt = 1; attempt <= maxAttempts; attempt++) {\n      try {\n        await new Promise((resolve, reject) => {\n          const request = http.get(url, (response) => {\n            const chunks = []\n\n            response.on(\"data\", (chunk) => chunks.push(chunk))\n            response.on(\"end\", () => {\n              const body = Buffer.concat(chunks).toString(\"utf8\")\n\n              if (response.statusCode !== 200) {\n                reject(new Error(`HTTP server health check failed with status ${response.statusCode} for ${url}: ${body.slice(0, 200)}`))\n                return\n              }\n\n              resolve(undefined)\n            })\n          })\n\n          request.on(\"error\", (error) => reject(error))\n\n          request.setTimeout(timeoutMs, () => {\n            request.destroy(new Error(`HTTP server health check timed out after ${timeoutMs}ms for ${url}`))\n          })\n        })\n\n        return\n      } catch (error) {\n        if (attempt === maxAttempts || !this.isRetryableHealthCheckError(error)) {\n          throw error\n        }\n\n        await new Promise((resolve) => setTimeout(resolve, 100))\n      }\n    }\n  }\n\n  /**\n   * @param {unknown} error\n   * @returns {boolean}\n   */\n  isRetryableHealthCheckError(error) {\n    if (error && typeof error === \"object\" && \"code\" in error) {\n      if (error.code === \"ECONNRESET\") return true\n      if (error.code === \"ECONNREFUSED\") return true\n      if (error.code === \"ETIMEDOUT\") return true\n    }\n\n    const message = error instanceof Error ? error.message : String(error)\n\n    return message.includes(\"socket hang up\")\n  }\n}\n"]}
@@ -37,6 +37,8 @@ export default class SystemTest {
37
37
  _httpPort: number;
38
38
  /** @type {(error: any) => boolean | undefined} */
39
39
  _errorFilter: (error: any) => boolean | undefined;
40
+ /** @type {Error | undefined} */
41
+ _httpServerError: Error | undefined;
40
42
  /** @type {WebSocketServer | undefined} */
41
43
  scoundrelWss: WebSocketServer | undefined;
42
44
  /** @type {WebSocketServer | undefined} */
@@ -88,12 +90,25 @@ export default class SystemTest {
88
90
  * @returns {string}
89
91
  */
90
92
  getSelector(selector: string): string;
93
+ /**
94
+ * Logs messages
95
+ * @param {any[]} args
96
+ * @returns {void}
97
+ */
98
+ debugError(...args: any[]): void;
91
99
  /**
92
100
  * Logs messages when debugging is enabled
93
- * @param {string} message
101
+ * @param {any[]} args
102
+ * @returns {void}
103
+ */
104
+ debugLog(...args: any[]): void;
105
+ /** @returns {void} */
106
+ throwIfHttpServerError(): void;
107
+ /**
108
+ * @param {Error} error
94
109
  * @returns {void}
95
110
  */
96
- debugLog(message: string): void;
111
+ onHttpServerError: (error: Error) => void;
97
112
  /**
98
113
  * Starts Scoundrel server which the browser connects to for remote evaluation in the browser
99
114
  * @returns {void}
@@ -122,7 +122,9 @@ class SystemTest {
122
122
  throw new Error("SystemTest.run requires a callback");
123
123
  }
124
124
  systemTest.debugLog("Run started - send initialize");
125
- await systemTest.getCommunicator().sendCommand({ type: "initialize" });
125
+ await timeout({ timeout: 10000, errorMessage: "Sending intialize to useSystemTest() timed out" }, async () => {
126
+ await systemTest.getCommunicator().sendCommand({ type: "initialize" });
127
+ });
126
128
  systemTest.debugLog("getRootPath");
127
129
  const rootPath = systemTest.getRootPath();
128
130
  systemTest.debugLog(`Visit rootPath with dismissTo: ${rootPath}`);
@@ -158,10 +160,21 @@ class SystemTest {
158
160
  this._httpPort = 1984;
159
161
  /** @type {(error: any) => boolean | undefined} */
160
162
  this._errorFilter = undefined;
163
+ /** @type {Error | undefined} */
164
+ this._httpServerError = undefined;
161
165
  /** @type {WebSocketServer | undefined} */
162
166
  this.scoundrelWss = undefined;
163
167
  /** @type {WebSocketServer | undefined} */
164
168
  this.clientWss = undefined;
169
+ /**
170
+ * @param {Error} error
171
+ * @returns {void}
172
+ */
173
+ this.onHttpServerError = (error) => {
174
+ const errorMessage = error instanceof Error ? error.message : String(error);
175
+ this._httpServerError = error instanceof Error ? error : new Error(errorMessage);
176
+ console.error(`HTTP server error: ${errorMessage}`);
177
+ };
165
178
  /**
166
179
  * Handles a command received from the browser
167
180
  * @param {{data: {message: string, backtrace?: string, type: string, value: any[]}}} args
@@ -248,6 +261,7 @@ class SystemTest {
248
261
  throw new Error("No this?");
249
262
  if (!this.driver)
250
263
  throw new Error("Driver hasn't been initialized yet");
264
+ this.throwIfHttpServerError();
251
265
  return this.driver;
252
266
  }
253
267
  /**
@@ -263,14 +277,30 @@ class SystemTest {
263
277
  getSelector(selector) {
264
278
  return this.getBaseSelector() ? `${this.getBaseSelector()} ${selector}` : selector;
265
279
  }
280
+ /**
281
+ * Logs messages
282
+ * @param {any[]} args
283
+ * @returns {void}
284
+ */
285
+ debugError(...args) {
286
+ if (this._debug) {
287
+ console.error("[SystemTest debug]", ...args);
288
+ }
289
+ }
266
290
  /**
267
291
  * Logs messages when debugging is enabled
268
- * @param {string} message
292
+ * @param {any[]} args
269
293
  * @returns {void}
270
294
  */
271
- debugLog(message) {
295
+ debugLog(...args) {
272
296
  if (this._debug) {
273
- console.log(`[SystemTest debug] ${message}`);
297
+ console.log("[SystemTest debug]", ...args);
298
+ }
299
+ }
300
+ /** @returns {void} */
301
+ throwIfHttpServerError() {
302
+ if (this._httpServerError) {
303
+ throw new Error(`HTTP server error: ${this._httpServerError.message}`);
274
304
  }
275
305
  }
276
306
  /**
@@ -744,9 +774,16 @@ class SystemTest {
744
774
  else if (process.env.SYSTEM_TEST_HOST == "dist") {
745
775
  this.currentUrl = `http://${this._httpHost}:${this._httpPort}`;
746
776
  this.debugLog(`Spawning HTTP server for dist on ${this._httpHost}:${this._httpPort}`);
747
- this.systemTestHttpServer = new SystemTestHttpServer({ host: this._httpHost, port: this._httpPort, debug: this._debug });
777
+ this.systemTestHttpServer = new SystemTestHttpServer({
778
+ host: this._httpHost,
779
+ port: this._httpPort,
780
+ debug: this._debug,
781
+ onError: this.onHttpServerError
782
+ });
748
783
  this.debugLog("Starting HTTP server");
749
784
  await this.systemTestHttpServer.start();
785
+ this.debugLog("Checking HTTP server health");
786
+ await this.systemTestHttpServer.assertReachable({ timeoutMs: this.getTimeouts() });
750
787
  this.debugLog("HTTP server started");
751
788
  }
752
789
  else {
@@ -886,6 +923,7 @@ class SystemTest {
886
923
  this.clientWss.on("connection", this.onWebSocketConnection);
887
924
  this.clientWss.on("close", this.onWebSocketClose);
888
925
  this.clientWss.on("error", (error) => {
926
+ this.debugError(error);
889
927
  if (this.waitForClientWebSocketPromiseReject) {
890
928
  this.waitForClientWebSocketPromiseReject(error instanceof Error ? error : new Error(String(error)));
891
929
  delete this.waitForClientWebSocketPromiseReject;
@@ -928,10 +966,10 @@ class SystemTest {
928
966
  this.ws = null;
929
967
  }
930
968
  await this.closeWebSocketServer(this.clientWss, "client WebSocket server");
931
- if (this.driver?.quit) {
969
+ if (this.driver) {
932
970
  await timeout({ timeout: this.getTimeouts(), errorMessage: "timeout while quitting WebDriver" }, async () => await this.driver.quit());
933
971
  }
934
- if (this.systemTestHttpServer?.close) {
972
+ if (this.systemTestHttpServer) {
935
973
  await timeout({ timeout: this.getTimeouts(), errorMessage: "timeout while closing HTTP server" }, async () => await this.systemTestHttpServer.close());
936
974
  }
937
975
  }
@@ -951,6 +989,7 @@ class SystemTest {
951
989
  this.server = undefined;
952
990
  this.serverWebSocket = undefined;
953
991
  this.systemTestHttpServer = undefined;
992
+ this._httpServerError = undefined;
954
993
  this.waitForClientWebSocketPromiseReject = undefined;
955
994
  this.waitForClientWebSocketPromiseResolve = undefined;
956
995
  this.communicator = new SystemTestCommunicator({ onCommand: this.onCommandReceived });
@@ -1045,4 +1084,4 @@ class SystemTest {
1045
1084
  }
1046
1085
  SystemTest.rootPath = "/blank?systemTest=true";
1047
1086
  export default SystemTest;
1048
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"system-test.js","sourceRoot":"","sources":["../src/system-test.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport {Builder, By, error as SeleniumError} from \"selenium-webdriver\"\nimport chrome from \"selenium-webdriver/chrome.js\"\nimport {digg} from \"diggerize\"\nimport fs from \"node:fs/promises\"\nimport logging from \"selenium-webdriver/lib/logging.js\"\nimport moment from \"moment\"\nimport {prettify} from \"htmlfy\"\nimport Server from \"scoundrel-remote-eval/build/server/index.js\"\nimport ServerWebSocket from \"scoundrel-remote-eval/build/server/connections/web-socket/index.js\"\nimport SystemTestCommunicator from \"./system-test-communicator.js\"\nimport SystemTestHttpServer from \"./system-test-http-server.js\"\nimport {wait, waitFor} from \"awaitery\"\nimport timeout from \"awaitery/build/timeout.js\"\nimport {WebSocketServer} from \"ws\"\n\n/**\n * @typedef {object} SystemTestArgs\n * @property {string} [host] Hostname for the app server.\n * @property {number} [port] Port for the app server.\n * @property {string} [httpHost] Hostname for the static HTTP server.\n * @property {number} [httpPort] Port for the static HTTP server.\n * @property {boolean} [debug] Enable debug logging.\n * @property {(error: any) => boolean} [errorFilter] Filter for browser errors (return false to ignore).\n * @property {Record<string, any>} [urlArgs] Query params appended to the root path.\n */\n/**\n * @typedef {object} FindArgs\n * @property {number} [timeout] Override timeout for lookup.\n * @property {boolean} [visible] Whether to require elements to be visible.\n * @property {boolean} [useBaseSelector] Whether to scope by the base selector.\n */\n/**\n * @typedef {object} WaitForNoSelectorArgs\n * @property {boolean} [useBaseSelector] Whether to scope by the base selector.\n */\n/**\n * @typedef {object} NotificationMessageArgs\n * @property {boolean} [dismiss] Whether to dismiss the notification after it appears.\n */\n\nclass ElementNotFoundError extends Error { }\nconst {WebDriverError} = SeleniumError\n\nexport default class SystemTest {\n  static rootPath = \"/blank?systemTest=true\"\n\n  /** @type {SystemTestCommunicator | undefined} */\n  communicator = undefined\n\n  /** @type {import(\"selenium-webdriver\").WebDriver | undefined} */\n  driver = undefined\n\n  _started = false\n  _driverTimeouts = 5000\n  _timeouts = 5000\n  _httpHost = \"localhost\"\n  _httpPort = 1984\n  /** @type {(error: any) => boolean | undefined} */\n  _errorFilter = undefined\n  /** @type {WebSocketServer | undefined} */\n  scoundrelWss = undefined\n  /** @type {WebSocketServer | undefined} */\n  clientWss = undefined\n\n  /**\n   * Gets the current system test instance\n   * @param {SystemTestArgs} [args]\n   * @returns {SystemTest}\n   */\n  static current(args = {}) {\n    if (!globalThis.systemTest) {\n      globalThis.systemTest = new SystemTest(args)\n    }\n\n    return globalThis.systemTest\n  }\n\n  /** @returns {SystemTestCommunicator} */\n  getCommunicator() {\n    if (!this.communicator) {\n      throw new Error(\"Communicator hasn't been initialized yet\")\n    }\n\n    return this.communicator\n  }\n\n  /**\n   * Extracts an error message if possible from the payload sent from the browser.\n   * @param {{message?: string, value?: any[]} | Error} data\n   * @returns {string | undefined}\n   */\n  extractErrorMessage(data) {\n    if (data instanceof Error) return data.message\n    if (typeof data === \"object\" && typeof data.message === \"string\") return data.message\n    if (typeof data === \"string\") return data\n\n    const firstValue = Array.isArray(data?.value) ? data.value[0] : undefined\n\n    if (firstValue instanceof Error && typeof firstValue.message === \"string\") return firstValue.message\n    if (typeof firstValue === \"string\") return firstValue\n\n    return undefined\n  }\n\n  /**\n   * Whether a browser error should be ignored based on built-in rules and an optional error filter.\n   * @param {{message?: string, value?: any[]}} data\n   * @returns {boolean}\n   */\n  shouldIgnoreError(data) {\n    const message = this.extractErrorMessage(data)\n\n    if (typeof message === \"string\") {\n      if (message.includes(\"Minified React error #418\")) return true\n      if (message.includes(\"Minified React error #419\")) return true\n    }\n\n    if (this._errorFilter && this._errorFilter(data) === false) {\n      return true\n    }\n\n    return false\n  }\n\n  /**\n   * Runs a system test\n   * @overload\n   * @param {function(SystemTest): Promise<void>} callback\n   * @returns {Promise<void>}\n   */\n  /**\n   * Runs a system test\n   * @overload\n   * @param {SystemTestArgs} args\n   * @param {function(SystemTest): Promise<void>} callback\n   * @returns {Promise<void>}\n   */\n  /**\n   * Runs a system test\n   * @param {SystemTestArgs | function(SystemTest): Promise<void>} [args]\n   * @param {function(SystemTest): Promise<void>} [callback]\n   * @returns {Promise<void>}\n   */\n  static async run(args, callback) {\n    const resolvedCallback = typeof args === \"function\" ? args : callback\n    const systemTest = this.current(typeof args === \"function\" ? {} : args)\n\n    if (!resolvedCallback) {\n      throw new Error(\"SystemTest.run requires a callback\")\n    }\n\n    systemTest.debugLog(\"Run started - send initialize\")\n    await systemTest.getCommunicator().sendCommand({type: \"initialize\"})\n\n    systemTest.debugLog(\"getRootPath\")\n    const rootPath = systemTest.getRootPath()\n\n    systemTest.debugLog(`Visit rootPath with dismissTo: ${rootPath}`)\n    await systemTest.dismissTo(rootPath)\n    systemTest.debugLog(`Dismissed to root path ${rootPath}`)\n\n    try {\n      systemTest.debugLog(\"findByTestID blankText\")\n      await systemTest.findByTestID(\"blankText\", {useBaseSelector: false})\n      systemTest.debugLog(\"Found blankText\")\n\n      systemTest.debugLog(\"resolvedCallback\")\n      await resolvedCallback(systemTest)\n      systemTest.debugLog(\"Run callback completed\")\n    } catch (error) {\n      systemTest.debugLog(\"Run error caught, taking screenshot\")\n      await systemTest.takeScreenshot()\n\n      throw error\n    }\n  }\n\n  /**\n   * Creates a new SystemTest instance\n   * @param {SystemTestArgs} [args]\n   */\n  constructor({host = \"localhost\", port = 8081, httpHost = \"localhost\", httpPort = 1984, debug = false, errorFilter, urlArgs, ...restArgs} = {host: \"localhost\", port: 8081, httpHost: \"localhost\", httpPort: 1984, debug: false}) {\n    const restArgsKeys = Object.keys(restArgs)\n\n    if (restArgsKeys.length > 0) {\n      throw new Error(`Unknown arguments: ${restArgsKeys.join(\", \")}`)\n    }\n\n    this._host = host\n    this._port = port\n    this._httpHost = httpHost\n    this._httpPort = httpPort\n    this._debug = debug\n    this._errorFilter = errorFilter\n    this._urlArgs = urlArgs\n    this._rootPath = this.buildRootPath()\n\n    /** @type {Record<number, object>} */\n    this._responses = {}\n\n    this._sendCount = 0\n    this.startScoundrel()\n    this.communicator = new SystemTestCommunicator({onCommand: this.onCommandReceived})\n  }\n\n  /**\n   * Gets the base selector for scoping element searches\n   * @returns {string | undefined}\n   */\n  getBaseSelector() { return this._baseSelector }\n\n  /** @returns {import(\"selenium-webdriver\").WebDriver} */\n  getDriver() {\n    if (!this) throw new Error(\"No this?\")\n    if (!this.driver) throw new Error(\"Driver hasn't been initialized yet\")\n\n    return this.driver\n  }\n\n  /**\n   * Sets the base selector for scoping element searches\n   * @param {string} baseSelector\n   */\n  setBaseSelector(baseSelector) { this._baseSelector = baseSelector }\n\n  /**\n   * Gets a selector scoped to the base selector\n   * @param {string} selector\n   * @returns {string}\n   */\n  getSelector(selector) {\n    return this.getBaseSelector() ? `${this.getBaseSelector()} ${selector}` : selector\n  }\n\n  /**\n   * Logs messages when debugging is enabled\n   * @param {string} message\n   * @returns {void}\n   */\n  debugLog(message) {\n    if (this._debug) {\n      console.log(`[SystemTest debug] ${message}`)\n    }\n  }\n\n  /**\n   * Starts Scoundrel server which the browser connects to for remote evaluation in the browser\n   * @returns {void}\n   */\n  startScoundrel() {\n    if (this.scoundrelWss) throw new Error(\"Scoundrel server already started\")\n\n    this.scoundrelWss = new WebSocketServer({port: 8090})\n    this.serverWebSocket = new ServerWebSocket(this.scoundrelWss)\n    this.server = new Server(this.serverWebSocket)\n  }\n\n  /**\n   * @returns {Promise<void>}\n   */\n  async stopScoundrel() {\n    if (this.server?.close) {\n      await timeout({timeout: this.getTimeouts(), errorMessage: \"timeout while waiting for Scoundrel to stop\"}, async () => await this.server.close())\n    }\n    await this.closeWebSocketServer(this.scoundrelWss, \"Scoundrel WebSocket server\")\n  }\n\n  /**\n   * Waits for the Scoundrel client (browser) to connect and returns it.\n   * @param {number} [timeoutMs]\n   * @returns {Promise<import(\"scoundrel-remote-eval/build/client/index.js\").default>}\n   */\n  async getScoundrelClient(timeoutMs = 10000) {\n    if (!this.server) {\n      throw new Error(\"Scoundrel server is not started\")\n    }\n\n    const existingClients = this.server.getClients?.()\n\n    if (existingClients && existingClients.length > 0) {\n      return existingClients[0]\n    }\n\n    if (!this.server.events?.on) {\n      throw new Error(\"Scoundrel server events are unavailable\")\n    }\n\n    let onNewClient\n    const cleanupListener = () => {\n      if (onNewClient) {\n        this.server?.events.off(\"newClient\", onNewClient)\n      }\n    }\n\n    try {\n      return await timeout({timeout: timeoutMs, errorMessage: \"Timed out waiting for Scoundrel client\"}, async () => await new Promise((resolve) => {\n        onNewClient = (client) => {\n          cleanupListener()\n          resolve(client)\n        }\n\n        this.server.events.on(\"newClient\", onNewClient)\n      }))\n    } finally {\n      cleanupListener()\n    }\n  }\n\n  /**\n   * Finds all elements by CSS selector\n   * @param {string} selector\n   * @param {FindArgs} [args]\n   * @returns {Promise<import(\"selenium-webdriver\").WebElement[]>}\n   */\n  async all(selector, args = {}) {\n    const {visible = true, timeout, useBaseSelector = true, ...restArgs} = args\n    const restArgsKeys = Object.keys(restArgs)\n    let actualTimeout\n\n    if (timeout === undefined) {\n      actualTimeout = this._driverTimeouts\n    } else {\n      actualTimeout = timeout\n    }\n\n    if (restArgsKeys.length > 0) throw new Error(`Unknown arguments: ${restArgsKeys.join(\", \")}`)\n\n    const actualSelector = useBaseSelector ? this.getSelector(selector) : selector\n    const startTime = Date.now()\n    const getTimeLeft = () => Math.max(actualTimeout - (Date.now() - startTime), 0)\n    const getElements = async () => {\n      const foundElements = await this.getDriver().findElements(By.css(actualSelector))\n\n      if (visible !== true && visible !== false) {\n        return foundElements\n      }\n\n      const filteredElements = []\n\n      for (const element of foundElements) {\n        const isDisplayed = await element.isDisplayed()\n\n        if (visible && !isDisplayed) continue\n        if (!visible && isDisplayed) continue\n\n        filteredElements.push(element)\n      }\n\n      return filteredElements\n    }\n    let elements = []\n\n    while (true) {\n      const timeLeft = actualTimeout == 0 ? 0 : getTimeLeft()\n\n      try {\n        if (timeLeft == 0) {\n          elements = await getElements()\n        } else {\n          await this.getDriver().wait(async () => {\n            elements = await getElements()\n\n            return elements.length > 0\n          }, timeLeft)\n        }\n\n        break\n      } catch (error) {\n        if (error instanceof SeleniumError.TimeoutError && getTimeLeft() > 0) {\n          continue\n        }\n\n        throw new Error(`Couldn't get elements with selector: ${actualSelector}: ${error instanceof Error ? error.message : error}`)\n      }\n    }\n    return elements\n  }\n\n  /**\n   * Clicks an element that has children which fills out the element and would otherwise have caused a ElementClickInterceptedError\n   * @param {string|import(\"selenium-webdriver\").WebElement} elementOrIdentifier\n   * @returns {Promise<void>}\n   */\n  /**\n   * Clicks an element, allowing selector args when using a CSS selector.\n   * @param {string|import(\"selenium-webdriver\").WebElement} elementOrIdentifier\n   * @param {FindArgs} [args]\n   * @returns {Promise<void>}\n   */\n  async click(elementOrIdentifier, args) {\n    let tries = 0\n\n    while (true) {\n      tries++\n\n      try {\n        const element = await this._findElement(elementOrIdentifier, args)\n        const actions = this.getDriver().actions({async: true})\n\n        await actions.move({origin: element}).click().perform()\n        break\n      } catch (error) {\n        if (error instanceof Error) {\n          if (error.constructor.name === \"ElementNotInteractableError\") {\n            if (tries >= 3) {\n              throw new Error(`Element ${elementOrIdentifier.constructor.name} click failed after ${tries} tries - ${error.constructor.name}: ${error.message}`)\n            } else {\n              await wait(50)\n            }\n          } else {\n            // Re-throw with un-corrupted stack trace\n            throw new Error(`Element ${elementOrIdentifier.constructor.name} click failed - ${error.constructor.name}: ${error.message}`)\n          }\n        } else {\n          throw new Error(`Element ${elementOrIdentifier.constructor.name} click failed - ${typeof error}: ${error}`)\n        }\n      }\n    }\n  }\n\n  /**\n   * Finds a single element by CSS selector\n   * @param {string} selector\n   * @param {FindArgs} [args]\n   * @returns {Promise<import(\"selenium-webdriver\").WebElement>}\n   */\n  async find(selector, args = {}) {\n    const startTime = Date.now()\n    let elements = []\n\n    try {\n      elements = await this.all(selector, args)\n    } catch (error) {\n      // Re-throw to recover stack trace\n      if (error instanceof Error) {\n        if (error.message.startsWith(\"Wait timed out after\")) {\n          elements = []\n        }\n\n        throw new Error(`${error.constructor.name} - ${error.message} (selector: ${this.getSelector(selector)})`)\n      } else {\n        throw new Error(`${typeof error} - ${error} (selector: ${this.getSelector(selector)})`)\n      }\n    }\n\n    if (elements.length > 1) {\n      throw new Error(`More than 1 elements (${elements.length}) was found by CSS: ${this.getSelector(selector)}`)\n    }\n\n    if (!elements[0]) {\n      const elapsedSeconds = (Date.now() - startTime) / 1000\n      throw new ElementNotFoundError(`Element couldn't be found after ${elapsedSeconds.toFixed(2)}s by CSS: ${this.getSelector(selector)}`)\n    }\n\n    return elements[0]\n  }\n\n  /**\n   * Finds a single element by test ID\n   * @param {string} testID\n   * @param {FindArgs} [args]\n   * @returns {Promise<import(\"selenium-webdriver\").WebElement>}\n   */\n  async findByTestID(testID, args) { return await this.find(`[data-testid='${testID}']`, args) }\n\n\n  /**\n   * @param {string|import(\"selenium-webdriver\").WebElement|{selector: string} & FindArgs} elementOrIdentifier\n   * @param {FindArgs} [args]\n   * @returns {Promise<import(\"selenium-webdriver\").WebElement>}\n   */\n  async _findElement(elementOrIdentifier, args) {\n    /** @type {import(\"selenium-webdriver\").WebElement} */\n    let element\n\n    if (typeof elementOrIdentifier == \"string\") {\n      element = await this.find(elementOrIdentifier, args)\n    } else if (typeof elementOrIdentifier == \"object\" && elementOrIdentifier !== null && \"selector\" in elementOrIdentifier) {\n      const {selector, ...restArgs} = elementOrIdentifier\n\n      element = await this.find(selector, restArgs)\n    } else {\n      element = /** @type {import(\"selenium-webdriver\").WebElement} */ (elementOrIdentifier)\n    }\n\n    return element\n  }\n\n  /**\n   * Finds a single element by CSS selector without waiting\n   * @param {string} selector\n   * @param {FindArgs} [args]\n   * @returns {Promise<import(\"selenium-webdriver\").WebElement>}\n   */\n  async findNoWait(selector, args = {}) {\n    await this.driverSetTimeouts(0)\n\n    try {\n      return await this.find(selector, args)\n    } finally {\n      await this.restoreTimeouts()\n    }\n  }\n\n  /**\n   * Gets browser logs\n   * @returns {Promise<string[]>}\n   */\n  async getBrowserLogs() {\n    const entries = await this.getDriver().manage().logs().get(logging.Type.BROWSER)\n    const browserLogs = []\n\n    for (const entry of entries) {\n      const messageMatch = entry.message.match(/^(.+) (\\d+):(\\d+) (.+)$/)\n      let message\n\n      if (messageMatch) {\n        message = messageMatch[4]\n      } else {\n        message = entry.message\n      }\n\n      browserLogs.push(`${entry.level.name}: ${message}`)\n    }\n\n    return browserLogs\n  }\n\n  /** @returns {Promise<string>} */\n  async getCurrentUrl() {\n    return await this.getDriver().getCurrentUrl()\n  }\n\n  /** @returns {number} */\n  getTimeouts() { return this._timeouts }\n\n  /**\n   * Interacts with an element by calling a method on it with the given arguments.\n   * Retrying on ElementNotInteractableError, ElementClickInterceptedError, or StaleElementReferenceError.\n   * @param {import(\"selenium-webdriver\").WebElement|string|{selector: string} & FindArgs} elementOrIdentifier The element or a CSS selector to find the element.\n   * @param {string} methodName The method name to call on the element.\n   * @param {...any} args Arguments to pass to the method.\n   * @returns {Promise<any>}\n   */\n  async interact(elementOrIdentifier, methodName, ...args) {\n    let tries = 0\n\n    while (true) {\n      tries++\n\n      const element = await this._findElement(elementOrIdentifier)\n\n      if (!element[methodName]) {\n        throw new Error(`${element.constructor.name} hasn't an attribute named: ${methodName}`)\n      } else if (typeof element[methodName] != \"function\") {\n        throw new Error(`${element.constructor.name}#${methodName} is not a function`)\n      }\n\n      try {\n        // Dont call with candidate, because that will bind the function wrong.\n        return await element[methodName](...args)\n      } catch (error) {\n        if (error instanceof Error) {\n          if (\n            error.constructor.name === \"ElementNotInteractableError\" ||\n            error.constructor.name === \"ElementClickInterceptedError\" ||\n            error.constructor.name === \"StaleElementReferenceError\"\n          ) {\n            // Retry finding the element and interacting with it\n            if (tries >= 3) {\n              let elementDescription\n\n              if (typeof elementOrIdentifier == \"string\") {\n                elementDescription = `CSS selector ${elementOrIdentifier}`\n              } else {\n                elementDescription = `${element.constructor.name}`\n              }\n\n              throw new Error(`${elementDescription} ${methodName} failed after ${tries} tries - ${error.constructor.name}: ${error.message}`)\n            } else {\n              await wait(50)\n            }\n          } else {\n            // Re-throw with un-corrupted stack trace\n            throw new Error(`${element.constructor.name} ${methodName} failed - ${error.constructor.name}: ${error.message}`)\n          }\n        } else {\n          throw new Error(`${element.constructor.name} ${methodName} failed - ${typeof error}: ${error}`)\n        }\n      }\n    }\n  }\n\n  /**\n   * Expects no element to be found by CSS selector\n   * @param {string} selector\n   * @param {FindArgs} [args]\n   * @returns {Promise<void>}\n   */\n  async expectNoElement(selector, args = {}) {\n    let found = false\n\n    try {\n      await this.findNoWait(selector, args)\n      found = true\n    } catch (error) {\n      if (error instanceof Error && error.message.startsWith(\"Element couldn't be found after \")) {\n        // Ignore\n      } else {\n        throw error\n      }\n    }\n\n    if (found) {\n      throw new Error(`Expected not to find: ${selector}`)\n    }\n  }\n\n  /**\n   * @param {string} selector\n   * @param {WaitForNoSelectorArgs} [args]\n   * @returns {Promise<void>}\n   */\n  async waitForNoSelector(selector, args = {}) {\n    const {useBaseSelector, ...restArgs} = args\n\n    if (Object.keys(restArgs).length > 0) {\n      throw new Error(`Unexpected args: ${Object.keys(restArgs).join(\", \")}`)\n    }\n\n    const actualSelector = useBaseSelector ? this.getSelector(selector) : selector\n\n    await this.driverSetTimeouts(0)\n\n    try {\n      await this._withRethrownErrors(async () => {\n        await this.getDriver().wait(\n          async () => {\n            const elements = await this.getDriver().findElements(By.css(actualSelector))\n\n            // Not found at all\n            if (elements.length === 0) {\n              return true\n            }\n\n            // Found but not visible\n            try {\n              const isDisplayed = await elements[0].isDisplayed()\n\n              return !isDisplayed\n            } catch (error) {\n              if (\n                error instanceof Error &&\n                (error.constructor.name === \"StaleElementReferenceError\" || error.message.includes(\"stale element reference\"))\n              ) {\n                return false\n              }\n\n              throw error\n            }\n          },\n          this.getTimeouts()\n        )\n      })\n    } finally {\n      await this.restoreTimeouts()\n    }\n  }\n\n  async _withRethrownErrors(callback) {\n    try {\n      return await callback()\n    } catch (error) {\n      if (error instanceof WebDriverError) {\n        throw new Error(`Selenium ${error.constructor.name}: ${error.message}`)\n      } else {\n        throw error\n      }\n    }\n  }\n\n  /**\n   * Gets notification messages\n   * @returns {Promise<string[]>}\n   */\n  async notificationMessages() {\n    const notificationMessageElements = await this.all(\"[data-class='notification-message']\", {useBaseSelector: false})\n    const notificationMessageTexts = []\n\n    for (const notificationMessageElement of notificationMessageElements) {\n      const text = await notificationMessageElement.getText()\n\n      notificationMessageTexts.push(text)\n    }\n\n    return notificationMessageTexts\n  }\n\n  /**\n   * Expects a notification message to appear and waits for it if necessary.\n   * @param {string} expectedNotificationMessage\n   * @param {NotificationMessageArgs} [args]\n   * @returns {Promise<void>}\n   */\n  async expectNotificationMessage(expectedNotificationMessage, args = {}) {\n    const {dismiss = true, ...restArgs} = args\n\n    if (Object.keys(restArgs).length > 0) {\n      throw new Error(`Unexpected args: ${Object.keys(restArgs).join(\", \")}`)\n    }\n\n    /** @type {string[]} */\n    const allDetectedNotificationMessages = []\n    let foundNotificationMessageElement\n    let foundNotificationMessageCount\n\n    await waitFor(async () => {\n      const notificationMessageElements = await this.all(\"[data-testid='notification-message']\", {useBaseSelector: false})\n\n      for (const notificationMessageElement of notificationMessageElements) {\n        const notificationMessage = (await this.getDriver().executeScript(\"return arguments[0].textContent;\", notificationMessageElement))?.trim() || await notificationMessageElement.getText()\n\n        if (!allDetectedNotificationMessages.includes(notificationMessage)) {\n          allDetectedNotificationMessages.push(notificationMessage)\n        }\n\n        if (notificationMessage == expectedNotificationMessage) {\n          foundNotificationMessageElement = notificationMessageElement\n          foundNotificationMessageCount = await notificationMessageElement.getAttribute(\"data-count\")\n          return\n        }\n      }\n\n      throw new Error(`Notification message ${expectedNotificationMessage} wasn't included in: ${allDetectedNotificationMessages.join(\", \")}`)\n    })\n\n    if (foundNotificationMessageElement && dismiss) {\n      await this.interact(foundNotificationMessageElement, \"click\") // Dismiss the notification message\n      if (!foundNotificationMessageCount) {\n        throw new Error(\"Expected notification message to have a data-count\")\n      }\n\n      await this.waitForNoSelector(`[data-testid='notification-message'][data-count='${foundNotificationMessageCount}']`, {useBaseSelector: false})\n    }\n  }\n\n  /** @returns {Promise<void>} */\n  async dismissNotificationMessages() {\n    const notificationMessageElements = await this.all(\"[data-class='notification-message']\", {useBaseSelector: false})\n\n    for (const notificationMessageElement of notificationMessageElements) {\n      await this.interact(notificationMessageElement, \"click\")\n    }\n\n    await this.waitForNoSelector(\"[data-class='notification-message']\", {useBaseSelector: false})\n  }\n\n  /**\n   * Indicates whether the system test has been started\n   * @returns {boolean}\n   */\n  isStarted() { return this._started }\n\n  /**\n   * Gets the HTML of the current page\n   * @returns {Promise<string>}\n   */\n  async getHTML() { return await this.getDriver().getPageSource() }\n\n  /**\n   * Starts the system test\n   * @returns {Promise<void>}\n   */\n  async start() {\n    this.debugLog(\"Start called\")\n    if (process.env.SYSTEM_TEST_HOST == \"expo-dev-server\") {\n      this.currentUrl = `http://${this._host}:${this._port}`\n      this.debugLog(`Using expo-dev-server at ${this.currentUrl}`)\n    } else if (process.env.SYSTEM_TEST_HOST == \"dist\") {\n      this.currentUrl = `http://${this._httpHost}:${this._httpPort}`\n\n      this.debugLog(`Spawning HTTP server for dist on ${this._httpHost}:${this._httpPort}`)\n      this.systemTestHttpServer = new SystemTestHttpServer({host: this._httpHost, port: this._httpPort, debug: this._debug})\n\n      this.debugLog(\"Starting HTTP server\")\n      await this.systemTestHttpServer.start()\n      this.debugLog(\"HTTP server started\")\n    } else {\n      throw new Error(\"Please set SYSTEM_TEST_HOST to 'expo-dev-server' or 'dist'\")\n    }\n\n    const options = new chrome.Options()\n\n    options.addArguments(\"--disable-dev-shm-usage\")\n    options.addArguments(\"--disable-gpu\")\n    options.addArguments(\"--headless=new\")\n    options.addArguments(\"--no-sandbox\")\n    options.addArguments(\"--window-size=1920,1080\")\n    this.debugLog(\"Chrome options configured\")\n\n    this.driver = new Builder()\n      .forBrowser(\"chrome\")\n      .setChromeOptions(options)\n      // @ts-expect-error\n      .setCapability(\"goog:loggingPrefs\", {browser: \"ALL\"})\n      .build()\n    this.debugLog(\"WebDriver built\")\n\n    await this.setTimeouts(10000)\n    this.debugLog(\"Timeouts set on driver\")\n\n    // Web socket server to communicate with browser\n    this.debugLog(\"Starting WebSocket server\")\n    await this.startWebSocketServer()\n    this.debugLog(\"WebSocket server started\")\n\n    // Visit the root page and wait for Expo to be loaded and the app to appear\n    this.debugLog(\"Visiting root path\")\n    const rootPath = this.getRootPath()\n    await this.driverVisit(rootPath)\n    this.debugLog(`Visited root path ${rootPath}`)\n\n    try {\n      await this.find(\"body > #root\", {useBaseSelector: false})\n      await this.findByTestID(\"systemTestingComponent\", {useBaseSelector: false, timeout: 30000, visible: true})\n      this.debugLog(\"Found root and systemTestingComponent\")\n    } catch (error) {\n      await this.takeScreenshot()\n      throw error\n    }\n\n    // Wait for client to connect\n    this.debugLog(\"Waiting for client WebSocket connection (opening)\")\n    this.debugLog(`WS state: ${this.ws?.readyState ?? \"none\"}`)\n    this.debugLog(\"waitForClientWebSocket\")\n    await this.waitForClientWebSocket()\n    this.debugLog(\"Client WebSocket connected\")\n\n    this._started = true\n    this.setBaseSelector(\"[data-testid='systemTestingComponent'][data-focussed='true']\")\n    this.debugLog(\"Start completed\")\n  }\n\n  /**\n   * @returns {string}\n   */\n  getRootPath() {\n    return this._rootPath ?? SystemTest.rootPath\n  }\n\n  /**\n   * @returns {string}\n   */\n  buildRootPath() {\n    if (!this._urlArgs) return SystemTest.rootPath\n\n    const url = new URL(SystemTest.rootPath, \"http://localhost\")\n    const appendParam = (key, value) => {\n      if (value === undefined || value === null) return\n      url.searchParams.append(key, String(value))\n    }\n\n    if (this._urlArgs instanceof URLSearchParams) {\n      for (const [key, value] of this._urlArgs) {\n        appendParam(key, value)\n      }\n    } else {\n      for (const [key, value] of Object.entries(this._urlArgs)) {\n        appendParam(key, value)\n      }\n    }\n\n    const rootPath =  `${url.pathname}${url.search}${url.hash}`\n\n    this.debugLog(`buildRootPath rootPath: ${rootPath}`)\n\n    return rootPath\n  }\n\n  /**\n   * Restores previously set timeouts\n   * @returns {Promise<void>}\n   */\n  async restoreTimeouts() {\n    if (!this.getTimeouts()) {\n      throw new Error(\"Timeouts haven't previously been set\")\n    }\n\n    await this.driverSetTimeouts(this.getTimeouts())\n  }\n\n  /**\n   * Sets driver timeouts\n   * @param {number} newTimeout\n   * @returns {Promise<void>}\n   */\n  async driverSetTimeouts(newTimeout) {\n    this._driverTimeouts = newTimeout\n    await this.getDriver().manage().setTimeouts({implicit: newTimeout})\n  }\n\n  /**\n   * Sets timeouts and stores the previous timeouts\n   * @param {number} newTimeout\n   * @returns {Promise<void>}\n   */\n  async setTimeouts(newTimeout) {\n    this._timeouts = newTimeout\n    await this.restoreTimeouts()\n  }\n\n  /**\n   * Waits for the client web socket to connect\n   * @returns {Promise<void>}\n   */\n  async waitForClientWebSocket() {\n    try {\n      await timeout({timeout: 30000}, () => new Promise((resolve, reject) => {\n        if (this.ws) {\n          resolve()\n          return\n        }\n\n        this.waitForClientWebSocketPromiseReject = reject\n        this.waitForClientWebSocketPromiseResolve = resolve\n      }))\n    } catch (error) {\n      delete this.waitForClientWebSocketPromiseReject\n      delete this.waitForClientWebSocketPromiseResolve\n      throw error\n    }\n  }\n\n  /**\n   * Starts the web socket server\n   * @returns {void}\n   */\n  startWebSocketServer() {\n    this.clientWss = new WebSocketServer({port: 1985})\n    this.clientWss.on(\"connection\", this.onWebSocketConnection)\n    this.clientWss.on(\"close\", this.onWebSocketClose)\n    this.clientWss.on(\"error\", (error) => {\n      if (this.waitForClientWebSocketPromiseReject) {\n        this.waitForClientWebSocketPromiseReject(error instanceof Error ? error : new Error(String(error)))\n        delete this.waitForClientWebSocketPromiseReject\n        delete this.waitForClientWebSocketPromiseResolve\n      }\n    })\n  }\n\n  /**\n   * Sets the on command callback\n   * @param {function({type: string, data: Record<string, any>}): Promise<void>} callback\n   * @returns {void}\n   */\n  onCommand(callback) {\n    this._onCommandCallback = callback\n  }\n\n  /**\n   * Handles a command received from the browser\n   * @param {{data: {message: string, backtrace?: string, type: string, value: any[]}}} args\n   * @returns {Promise<any>}\n   */\n  onCommandReceived = async ({data}) => {\n    const type = data.type\n    let result\n\n    if (type == \"console.error\") {\n      const showMessage = !this.shouldIgnoreError(data)\n\n      if (showMessage) {\n        console.error(\"Browser error\", ...data.value)\n      }\n    } else if (type == \"console.log\") {\n      console.log(\"Browser log\", ...data.value)\n    } else if (type == \"error\" || data.type == \"unhandledrejection\") {\n      this.handleError(data)\n    } else if (this._onCommandCallback) {\n      result = await this._onCommandCallback({data, type})\n    } else {\n      console.error(`onWebSocketClientMessage unknown data (type ${type})`, data)\n    }\n\n    return result\n  }\n\n  /**\n   * Handles a new web socket connection\n   * @param {WebSocket} ws\n   * @returns {Promise<void>}\n   */\n  onWebSocketConnection = async (ws) => {\n    this.ws = ws\n    this.getCommunicator().ws = ws\n    this.getCommunicator().onOpen()\n\n    // @ts-expect-error\n    this.ws.on(\"error\", digg(this, \"communicator\", \"onError\"))\n\n    // @ts-expect-error\n    this.ws.on(\"message\", digg(this, \"communicator\", \"onMessage\"))\n\n    if (this.waitForClientWebSocketPromiseResolve) {\n      this.waitForClientWebSocketPromiseResolve()\n      delete this.waitForClientWebSocketPromiseResolve\n      delete this.waitForClientWebSocketPromiseReject\n    }\n  }\n\n  /** @returns {void} */\n  onWebSocketClose = () => {\n    this.ws = null\n    this.getCommunicator().ws = null\n\n    if (this.waitForClientWebSocketPromiseReject) {\n      this.waitForClientWebSocketPromiseReject(new Error(\"Client websocket closed before connecting\"))\n      delete this.waitForClientWebSocketPromiseReject\n      delete this.waitForClientWebSocketPromiseResolve\n    }\n  }\n\n  /**\n   * Handles an error reported from the browser\n   * @param {object} data\n   * @param {string} data.message\n   * @param {string} [data.backtrace]\n   * @returns {void}\n   */\n  handleError(data) {\n    if (this.shouldIgnoreError(data)) return\n\n    const error = new Error(`Browser error: ${data.message}`)\n\n    if (data.backtrace) {\n      error.stack = `${error.message}\\n${data.backtrace}\\n\\n${error.stack}`\n    }\n\n    console.error(error)\n  }\n\n  /**\n   * Stops the system test\n   * @returns {Promise<void>}\n   */\n  async stop() {\n    await this.stopScoundrel()\n    if (this.ws) {\n      this.ws.close()\n      this.ws = null\n    }\n    await this.closeWebSocketServer(this.clientWss, \"client WebSocket server\")\n    if (this.driver?.quit) {\n      await timeout({timeout: this.getTimeouts(), errorMessage: \"timeout while quitting WebDriver\"}, async () => await this.driver.quit())\n    }\n    if (this.systemTestHttpServer?.close) {\n      await timeout({timeout: this.getTimeouts(), errorMessage: \"timeout while closing HTTP server\"}, async () => await this.systemTestHttpServer.close())\n    }\n  }\n\n  /**\n   * Fully tears down and restarts the system test instance.\n   * @returns {Promise<void>}\n   */\n  async reinitialize() {\n    await this.stop()\n\n    this._started = false\n    this._baseSelector = undefined\n    this.currentUrl = undefined\n    this.driver = undefined\n    this.ws = null\n    this.clientWss = undefined\n    this.scoundrelWss = undefined\n    this.server = undefined\n    this.serverWebSocket = undefined\n    this.systemTestHttpServer = undefined\n    this.waitForClientWebSocketPromiseReject = undefined\n    this.waitForClientWebSocketPromiseResolve = undefined\n    this.communicator = new SystemTestCommunicator({onCommand: this.onCommandReceived})\n\n    this.startScoundrel()\n    await this.start()\n  }\n\n  /**\n   * Visits a path in the browser\n   * @param {string} path\n   * @returns {Promise<void>}\n   */\n  async driverVisit(path) {\n    const url = `${this.currentUrl}${path}`\n\n    await this.getDriver().get(url)\n  }\n\n  /**\n   * Takes a screenshot, saves HTML and browser logs\n   * @returns {Promise<void>}\n   */\n  async takeScreenshot() {\n    const path = `${process.cwd()}/tmp/screenshots`\n\n    await fs.mkdir(path, {recursive: true})\n\n    const imageContent = await this.getDriver().takeScreenshot()\n    const now = new Date()\n    const screenshotPath = `${path}/${moment(now).format(\"YYYY-MM-DD-HH-MM-SS\")}.png`\n    const htmlPath = `${path}/${moment(now).format(\"YYYY-MM-DD-HH-MM-SS\")}.html`\n    const logsPath = `${path}/${moment(now).format(\"YYYY-MM-DD-HH-MM-SS\")}.logs.txt`\n    const logsText = await this.getBrowserLogs()\n    const html = await this.getHTML()\n    const htmlPretty = prettify(html)\n\n    await fs.writeFile(htmlPath, htmlPretty)\n    await fs.writeFile(logsPath, logsText.join(\"\\n\"))\n    await fs.writeFile(screenshotPath, imageContent, \"base64\")\n\n    console.log(\"Current URL:\", await this.getCurrentUrl())\n    console.log(\"Logs:\", logsPath)\n    console.log(\"Screenshot:\", screenshotPath)\n    console.log(\"HTML:\", htmlPath)\n  }\n\n  /**\n   * Visits a path in the browser\n   * @param {string} path\n   * @returns {Promise<void>}\n   */\n  async visit(path) {\n    await this.getCommunicator().sendCommand({type: \"visit\", path})\n  }\n\n  /**\n   * Dismisses to a path in the browser\n   * @param {string} path\n   * @returns {Promise<void>}\n   */\n  async dismissTo(path) {\n    await this.getCommunicator().sendCommand({type: \"dismissTo\", path})\n  }\n\n  /**\n   * @param {WebSocketServer | undefined} wss\n   * @param {string} [label]\n   * @returns {Promise<void>}\n   */\n  async closeWebSocketServer(wss, label = \"WebSocket server\") {\n    if (!wss) return\n\n    await timeout({timeout: this.getTimeouts(), errorMessage: `timeout while closing ${label}`}, async () => await new Promise((resolve, reject) => {\n      let settled = false\n      const terminateClient = (client) => {\n        try {\n          client.terminate()\n        } catch {\n          // Ignore termination errors\n        }\n      }\n      const settle = (callback, arg) => {\n        if (settled) return\n        settled = true\n        callback(arg)\n      }\n\n      wss.once(\"close\", () => settle(resolve))\n      wss.once(\"error\", (error) => settle(reject, error))\n      if (wss.clients && wss.clients.size > 0) {\n        wss.clients.forEach(terminateClient)\n      }\n      wss.close((error) => {\n        if (error) settle(reject, error)\n        else settle(resolve)\n      })\n    }))\n  }\n}\n"]}
1087
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"system-test.js","sourceRoot":"","sources":["../src/system-test.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport {Builder, By, error as SeleniumError} from \"selenium-webdriver\"\nimport chrome from \"selenium-webdriver/chrome.js\"\nimport {digg} from \"diggerize\"\nimport fs from \"node:fs/promises\"\nimport logging from \"selenium-webdriver/lib/logging.js\"\nimport moment from \"moment\"\nimport {prettify} from \"htmlfy\"\nimport Server from \"scoundrel-remote-eval/build/server/index.js\"\nimport ServerWebSocket from \"scoundrel-remote-eval/build/server/connections/web-socket/index.js\"\nimport SystemTestCommunicator from \"./system-test-communicator.js\"\nimport SystemTestHttpServer from \"./system-test-http-server.js\"\nimport {wait, waitFor} from \"awaitery\"\nimport timeout from \"awaitery/build/timeout.js\"\nimport {WebSocketServer} from \"ws\"\n\n/**\n * @typedef {object} SystemTestArgs\n * @property {string} [host] Hostname for the app server.\n * @property {number} [port] Port for the app server.\n * @property {string} [httpHost] Hostname for the static HTTP server.\n * @property {number} [httpPort] Port for the static HTTP server.\n * @property {boolean} [debug] Enable debug logging.\n * @property {(error: any) => boolean} [errorFilter] Filter for browser errors (return false to ignore).\n * @property {Record<string, any>} [urlArgs] Query params appended to the root path.\n */\n/**\n * @typedef {object} FindArgs\n * @property {number} [timeout] Override timeout for lookup.\n * @property {boolean} [visible] Whether to require elements to be visible.\n * @property {boolean} [useBaseSelector] Whether to scope by the base selector.\n */\n/**\n * @typedef {object} WaitForNoSelectorArgs\n * @property {boolean} [useBaseSelector] Whether to scope by the base selector.\n */\n/**\n * @typedef {object} NotificationMessageArgs\n * @property {boolean} [dismiss] Whether to dismiss the notification after it appears.\n */\n\nclass ElementNotFoundError extends Error { }\nconst {WebDriverError} = SeleniumError\n\nexport default class SystemTest {\n  static rootPath = \"/blank?systemTest=true\"\n\n  /** @type {SystemTestCommunicator | undefined} */\n  communicator = undefined\n\n  /** @type {import(\"selenium-webdriver\").WebDriver | undefined} */\n  driver = undefined\n\n  _started = false\n  _driverTimeouts = 5000\n  _timeouts = 5000\n  _httpHost = \"localhost\"\n  _httpPort = 1984\n  /** @type {(error: any) => boolean | undefined} */\n  _errorFilter = undefined\n  /** @type {Error | undefined} */\n  _httpServerError = undefined\n  /** @type {WebSocketServer | undefined} */\n  scoundrelWss = undefined\n  /** @type {WebSocketServer | undefined} */\n  clientWss = undefined\n\n  /**\n   * Gets the current system test instance\n   * @param {SystemTestArgs} [args]\n   * @returns {SystemTest}\n   */\n  static current(args = {}) {\n    if (!globalThis.systemTest) {\n      globalThis.systemTest = new SystemTest(args)\n    }\n\n    return globalThis.systemTest\n  }\n\n  /** @returns {SystemTestCommunicator} */\n  getCommunicator() {\n    if (!this.communicator) {\n      throw new Error(\"Communicator hasn't been initialized yet\")\n    }\n\n    return this.communicator\n  }\n\n  /**\n   * Extracts an error message if possible from the payload sent from the browser.\n   * @param {{message?: string, value?: any[]} | Error} data\n   * @returns {string | undefined}\n   */\n  extractErrorMessage(data) {\n    if (data instanceof Error) return data.message\n    if (typeof data === \"object\" && typeof data.message === \"string\") return data.message\n    if (typeof data === \"string\") return data\n\n    const firstValue = Array.isArray(data?.value) ? data.value[0] : undefined\n\n    if (firstValue instanceof Error && typeof firstValue.message === \"string\") return firstValue.message\n    if (typeof firstValue === \"string\") return firstValue\n\n    return undefined\n  }\n\n  /**\n   * Whether a browser error should be ignored based on built-in rules and an optional error filter.\n   * @param {{message?: string, value?: any[]}} data\n   * @returns {boolean}\n   */\n  shouldIgnoreError(data) {\n    const message = this.extractErrorMessage(data)\n\n    if (typeof message === \"string\") {\n      if (message.includes(\"Minified React error #418\")) return true\n      if (message.includes(\"Minified React error #419\")) return true\n    }\n\n    if (this._errorFilter && this._errorFilter(data) === false) {\n      return true\n    }\n\n    return false\n  }\n\n  /**\n   * Runs a system test\n   * @overload\n   * @param {function(SystemTest): Promise<void>} callback\n   * @returns {Promise<void>}\n   */\n  /**\n   * Runs a system test\n   * @overload\n   * @param {SystemTestArgs} args\n   * @param {function(SystemTest): Promise<void>} callback\n   * @returns {Promise<void>}\n   */\n  /**\n   * Runs a system test\n   * @param {SystemTestArgs | function(SystemTest): Promise<void>} [args]\n   * @param {function(SystemTest): Promise<void>} [callback]\n   * @returns {Promise<void>}\n   */\n  static async run(args, callback) {\n    const resolvedCallback = typeof args === \"function\" ? args : callback\n    const systemTest = this.current(typeof args === \"function\" ? {} : args)\n\n    if (!resolvedCallback) {\n      throw new Error(\"SystemTest.run requires a callback\")\n    }\n\n    systemTest.debugLog(\"Run started - send initialize\")\n    await timeout({timeout: 10_000, errorMessage: \"Sending intialize to useSystemTest() timed out\"}, async () => {\n      await systemTest.getCommunicator().sendCommand({type: \"initialize\"})\n    })\n\n    systemTest.debugLog(\"getRootPath\")\n    const rootPath = systemTest.getRootPath()\n\n    systemTest.debugLog(`Visit rootPath with dismissTo: ${rootPath}`)\n    await systemTest.dismissTo(rootPath)\n    systemTest.debugLog(`Dismissed to root path ${rootPath}`)\n\n    try {\n      systemTest.debugLog(\"findByTestID blankText\")\n      await systemTest.findByTestID(\"blankText\", {useBaseSelector: false})\n      systemTest.debugLog(\"Found blankText\")\n\n      systemTest.debugLog(\"resolvedCallback\")\n      await resolvedCallback(systemTest)\n      systemTest.debugLog(\"Run callback completed\")\n    } catch (error) {\n      systemTest.debugLog(\"Run error caught, taking screenshot\")\n      await systemTest.takeScreenshot()\n\n      throw error\n    }\n  }\n\n  /**\n   * Creates a new SystemTest instance\n   * @param {SystemTestArgs} [args]\n   */\n  constructor({host = \"localhost\", port = 8081, httpHost = \"localhost\", httpPort = 1984, debug = false, errorFilter, urlArgs, ...restArgs} = {host: \"localhost\", port: 8081, httpHost: \"localhost\", httpPort: 1984, debug: false}) {\n    const restArgsKeys = Object.keys(restArgs)\n\n    if (restArgsKeys.length > 0) {\n      throw new Error(`Unknown arguments: ${restArgsKeys.join(\", \")}`)\n    }\n\n    this._host = host\n    this._port = port\n    this._httpHost = httpHost\n    this._httpPort = httpPort\n    this._debug = debug\n    this._errorFilter = errorFilter\n    this._urlArgs = urlArgs\n    this._rootPath = this.buildRootPath()\n\n    /** @type {Record<number, object>} */\n    this._responses = {}\n\n    this._sendCount = 0\n    this.startScoundrel()\n    this.communicator = new SystemTestCommunicator({onCommand: this.onCommandReceived})\n  }\n\n  /**\n   * Gets the base selector for scoping element searches\n   * @returns {string | undefined}\n   */\n  getBaseSelector() { return this._baseSelector }\n\n  /** @returns {import(\"selenium-webdriver\").WebDriver} */\n  getDriver() {\n    if (!this) throw new Error(\"No this?\")\n    if (!this.driver) throw new Error(\"Driver hasn't been initialized yet\")\n    this.throwIfHttpServerError()\n\n    return this.driver\n  }\n\n  /**\n   * Sets the base selector for scoping element searches\n   * @param {string} baseSelector\n   */\n  setBaseSelector(baseSelector) { this._baseSelector = baseSelector }\n\n  /**\n   * Gets a selector scoped to the base selector\n   * @param {string} selector\n   * @returns {string}\n   */\n  getSelector(selector) {\n    return this.getBaseSelector() ? `${this.getBaseSelector()} ${selector}` : selector\n  }\n\n  /**\n   * Logs messages\n   * @param {any[]} args\n   * @returns {void}\n   */\n  debugError(...args) {\n    if (this._debug) {\n      console.error(\"[SystemTest debug]\", ...args)\n    }\n  }\n\n  /**\n   * Logs messages when debugging is enabled\n   * @param {any[]} args\n   * @returns {void}\n   */\n  debugLog(...args) {\n    if (this._debug) {\n      console.log(\"[SystemTest debug]\", ...args)\n    }\n  }\n\n  /** @returns {void} */\n  throwIfHttpServerError() {\n    if (this._httpServerError) {\n      throw new Error(`HTTP server error: ${this._httpServerError.message}`)\n    }\n  }\n\n  /**\n   * @param {Error} error\n   * @returns {void}\n   */\n  onHttpServerError = (error) => {\n    const errorMessage = error instanceof Error ? error.message : String(error)\n\n    this._httpServerError = error instanceof Error ? error : new Error(errorMessage)\n    console.error(`HTTP server error: ${errorMessage}`)\n  }\n\n  /**\n   * Starts Scoundrel server which the browser connects to for remote evaluation in the browser\n   * @returns {void}\n   */\n  startScoundrel() {\n    if (this.scoundrelWss) throw new Error(\"Scoundrel server already started\")\n\n    this.scoundrelWss = new WebSocketServer({port: 8090})\n    this.serverWebSocket = new ServerWebSocket(this.scoundrelWss)\n    this.server = new Server(this.serverWebSocket)\n  }\n\n  /**\n   * @returns {Promise<void>}\n   */\n  async stopScoundrel() {\n    if (this.server?.close) {\n      await timeout({timeout: this.getTimeouts(), errorMessage: \"timeout while waiting for Scoundrel to stop\"}, async () => await this.server.close())\n    }\n    await this.closeWebSocketServer(this.scoundrelWss, \"Scoundrel WebSocket server\")\n  }\n\n  /**\n   * Waits for the Scoundrel client (browser) to connect and returns it.\n   * @param {number} [timeoutMs]\n   * @returns {Promise<import(\"scoundrel-remote-eval/build/client/index.js\").default>}\n   */\n  async getScoundrelClient(timeoutMs = 10000) {\n    if (!this.server) {\n      throw new Error(\"Scoundrel server is not started\")\n    }\n\n    const existingClients = this.server.getClients?.()\n\n    if (existingClients && existingClients.length > 0) {\n      return existingClients[0]\n    }\n\n    if (!this.server.events?.on) {\n      throw new Error(\"Scoundrel server events are unavailable\")\n    }\n\n    let onNewClient\n    const cleanupListener = () => {\n      if (onNewClient) {\n        this.server?.events.off(\"newClient\", onNewClient)\n      }\n    }\n\n    try {\n      return await timeout({timeout: timeoutMs, errorMessage: \"Timed out waiting for Scoundrel client\"}, async () => await new Promise((resolve) => {\n        onNewClient = (client) => {\n          cleanupListener()\n          resolve(client)\n        }\n\n        this.server.events.on(\"newClient\", onNewClient)\n      }))\n    } finally {\n      cleanupListener()\n    }\n  }\n\n  /**\n   * Finds all elements by CSS selector\n   * @param {string} selector\n   * @param {FindArgs} [args]\n   * @returns {Promise<import(\"selenium-webdriver\").WebElement[]>}\n   */\n  async all(selector, args = {}) {\n    const {visible = true, timeout, useBaseSelector = true, ...restArgs} = args\n    const restArgsKeys = Object.keys(restArgs)\n    let actualTimeout\n\n    if (timeout === undefined) {\n      actualTimeout = this._driverTimeouts\n    } else {\n      actualTimeout = timeout\n    }\n\n    if (restArgsKeys.length > 0) throw new Error(`Unknown arguments: ${restArgsKeys.join(\", \")}`)\n\n    const actualSelector = useBaseSelector ? this.getSelector(selector) : selector\n    const startTime = Date.now()\n    const getTimeLeft = () => Math.max(actualTimeout - (Date.now() - startTime), 0)\n    const getElements = async () => {\n      const foundElements = await this.getDriver().findElements(By.css(actualSelector))\n\n      if (visible !== true && visible !== false) {\n        return foundElements\n      }\n\n      const filteredElements = []\n\n      for (const element of foundElements) {\n        const isDisplayed = await element.isDisplayed()\n\n        if (visible && !isDisplayed) continue\n        if (!visible && isDisplayed) continue\n\n        filteredElements.push(element)\n      }\n\n      return filteredElements\n    }\n    let elements = []\n\n    while (true) {\n      const timeLeft = actualTimeout == 0 ? 0 : getTimeLeft()\n\n      try {\n        if (timeLeft == 0) {\n          elements = await getElements()\n        } else {\n          await this.getDriver().wait(async () => {\n            elements = await getElements()\n\n            return elements.length > 0\n          }, timeLeft)\n        }\n\n        break\n      } catch (error) {\n        if (error instanceof SeleniumError.TimeoutError && getTimeLeft() > 0) {\n          continue\n        }\n\n        throw new Error(`Couldn't get elements with selector: ${actualSelector}: ${error instanceof Error ? error.message : error}`)\n      }\n    }\n    return elements\n  }\n\n  /**\n   * Clicks an element that has children which fills out the element and would otherwise have caused a ElementClickInterceptedError\n   * @param {string|import(\"selenium-webdriver\").WebElement} elementOrIdentifier\n   * @returns {Promise<void>}\n   */\n  /**\n   * Clicks an element, allowing selector args when using a CSS selector.\n   * @param {string|import(\"selenium-webdriver\").WebElement} elementOrIdentifier\n   * @param {FindArgs} [args]\n   * @returns {Promise<void>}\n   */\n  async click(elementOrIdentifier, args) {\n    let tries = 0\n\n    while (true) {\n      tries++\n\n      try {\n        const element = await this._findElement(elementOrIdentifier, args)\n        const actions = this.getDriver().actions({async: true})\n\n        await actions.move({origin: element}).click().perform()\n        break\n      } catch (error) {\n        if (error instanceof Error) {\n          if (error.constructor.name === \"ElementNotInteractableError\") {\n            if (tries >= 3) {\n              throw new Error(`Element ${elementOrIdentifier.constructor.name} click failed after ${tries} tries - ${error.constructor.name}: ${error.message}`)\n            } else {\n              await wait(50)\n            }\n          } else {\n            // Re-throw with un-corrupted stack trace\n            throw new Error(`Element ${elementOrIdentifier.constructor.name} click failed - ${error.constructor.name}: ${error.message}`)\n          }\n        } else {\n          throw new Error(`Element ${elementOrIdentifier.constructor.name} click failed - ${typeof error}: ${error}`)\n        }\n      }\n    }\n  }\n\n  /**\n   * Finds a single element by CSS selector\n   * @param {string} selector\n   * @param {FindArgs} [args]\n   * @returns {Promise<import(\"selenium-webdriver\").WebElement>}\n   */\n  async find(selector, args = {}) {\n    const startTime = Date.now()\n    let elements = []\n\n    try {\n      elements = await this.all(selector, args)\n    } catch (error) {\n      // Re-throw to recover stack trace\n      if (error instanceof Error) {\n        if (error.message.startsWith(\"Wait timed out after\")) {\n          elements = []\n        }\n\n        throw new Error(`${error.constructor.name} - ${error.message} (selector: ${this.getSelector(selector)})`)\n      } else {\n        throw new Error(`${typeof error} - ${error} (selector: ${this.getSelector(selector)})`)\n      }\n    }\n\n    if (elements.length > 1) {\n      throw new Error(`More than 1 elements (${elements.length}) was found by CSS: ${this.getSelector(selector)}`)\n    }\n\n    if (!elements[0]) {\n      const elapsedSeconds = (Date.now() - startTime) / 1000\n      throw new ElementNotFoundError(`Element couldn't be found after ${elapsedSeconds.toFixed(2)}s by CSS: ${this.getSelector(selector)}`)\n    }\n\n    return elements[0]\n  }\n\n  /**\n   * Finds a single element by test ID\n   * @param {string} testID\n   * @param {FindArgs} [args]\n   * @returns {Promise<import(\"selenium-webdriver\").WebElement>}\n   */\n  async findByTestID(testID, args) { return await this.find(`[data-testid='${testID}']`, args) }\n\n\n  /**\n   * @param {string|import(\"selenium-webdriver\").WebElement|{selector: string} & FindArgs} elementOrIdentifier\n   * @param {FindArgs} [args]\n   * @returns {Promise<import(\"selenium-webdriver\").WebElement>}\n   */\n  async _findElement(elementOrIdentifier, args) {\n    /** @type {import(\"selenium-webdriver\").WebElement} */\n    let element\n\n    if (typeof elementOrIdentifier == \"string\") {\n      element = await this.find(elementOrIdentifier, args)\n    } else if (typeof elementOrIdentifier == \"object\" && elementOrIdentifier !== null && \"selector\" in elementOrIdentifier) {\n      const {selector, ...restArgs} = elementOrIdentifier\n\n      element = await this.find(selector, restArgs)\n    } else {\n      element = /** @type {import(\"selenium-webdriver\").WebElement} */ (elementOrIdentifier)\n    }\n\n    return element\n  }\n\n  /**\n   * Finds a single element by CSS selector without waiting\n   * @param {string} selector\n   * @param {FindArgs} [args]\n   * @returns {Promise<import(\"selenium-webdriver\").WebElement>}\n   */\n  async findNoWait(selector, args = {}) {\n    await this.driverSetTimeouts(0)\n\n    try {\n      return await this.find(selector, args)\n    } finally {\n      await this.restoreTimeouts()\n    }\n  }\n\n  /**\n   * Gets browser logs\n   * @returns {Promise<string[]>}\n   */\n  async getBrowserLogs() {\n    const entries = await this.getDriver().manage().logs().get(logging.Type.BROWSER)\n    const browserLogs = []\n\n    for (const entry of entries) {\n      const messageMatch = entry.message.match(/^(.+) (\\d+):(\\d+) (.+)$/)\n      let message\n\n      if (messageMatch) {\n        message = messageMatch[4]\n      } else {\n        message = entry.message\n      }\n\n      browserLogs.push(`${entry.level.name}: ${message}`)\n    }\n\n    return browserLogs\n  }\n\n  /** @returns {Promise<string>} */\n  async getCurrentUrl() {\n    return await this.getDriver().getCurrentUrl()\n  }\n\n  /** @returns {number} */\n  getTimeouts() { return this._timeouts }\n\n  /**\n   * Interacts with an element by calling a method on it with the given arguments.\n   * Retrying on ElementNotInteractableError, ElementClickInterceptedError, or StaleElementReferenceError.\n   * @param {import(\"selenium-webdriver\").WebElement|string|{selector: string} & FindArgs} elementOrIdentifier The element or a CSS selector to find the element.\n   * @param {string} methodName The method name to call on the element.\n   * @param {...any} args Arguments to pass to the method.\n   * @returns {Promise<any>}\n   */\n  async interact(elementOrIdentifier, methodName, ...args) {\n    let tries = 0\n\n    while (true) {\n      tries++\n\n      const element = await this._findElement(elementOrIdentifier)\n\n      if (!element[methodName]) {\n        throw new Error(`${element.constructor.name} hasn't an attribute named: ${methodName}`)\n      } else if (typeof element[methodName] != \"function\") {\n        throw new Error(`${element.constructor.name}#${methodName} is not a function`)\n      }\n\n      try {\n        // Dont call with candidate, because that will bind the function wrong.\n        return await element[methodName](...args)\n      } catch (error) {\n        if (error instanceof Error) {\n          if (\n            error.constructor.name === \"ElementNotInteractableError\" ||\n            error.constructor.name === \"ElementClickInterceptedError\" ||\n            error.constructor.name === \"StaleElementReferenceError\"\n          ) {\n            // Retry finding the element and interacting with it\n            if (tries >= 3) {\n              let elementDescription\n\n              if (typeof elementOrIdentifier == \"string\") {\n                elementDescription = `CSS selector ${elementOrIdentifier}`\n              } else {\n                elementDescription = `${element.constructor.name}`\n              }\n\n              throw new Error(`${elementDescription} ${methodName} failed after ${tries} tries - ${error.constructor.name}: ${error.message}`)\n            } else {\n              await wait(50)\n            }\n          } else {\n            // Re-throw with un-corrupted stack trace\n            throw new Error(`${element.constructor.name} ${methodName} failed - ${error.constructor.name}: ${error.message}`)\n          }\n        } else {\n          throw new Error(`${element.constructor.name} ${methodName} failed - ${typeof error}: ${error}`)\n        }\n      }\n    }\n  }\n\n  /**\n   * Expects no element to be found by CSS selector\n   * @param {string} selector\n   * @param {FindArgs} [args]\n   * @returns {Promise<void>}\n   */\n  async expectNoElement(selector, args = {}) {\n    let found = false\n\n    try {\n      await this.findNoWait(selector, args)\n      found = true\n    } catch (error) {\n      if (error instanceof Error && error.message.startsWith(\"Element couldn't be found after \")) {\n        // Ignore\n      } else {\n        throw error\n      }\n    }\n\n    if (found) {\n      throw new Error(`Expected not to find: ${selector}`)\n    }\n  }\n\n  /**\n   * @param {string} selector\n   * @param {WaitForNoSelectorArgs} [args]\n   * @returns {Promise<void>}\n   */\n  async waitForNoSelector(selector, args = {}) {\n    const {useBaseSelector, ...restArgs} = args\n\n    if (Object.keys(restArgs).length > 0) {\n      throw new Error(`Unexpected args: ${Object.keys(restArgs).join(\", \")}`)\n    }\n\n    const actualSelector = useBaseSelector ? this.getSelector(selector) : selector\n\n    await this.driverSetTimeouts(0)\n\n    try {\n      await this._withRethrownErrors(async () => {\n        await this.getDriver().wait(\n          async () => {\n            const elements = await this.getDriver().findElements(By.css(actualSelector))\n\n            // Not found at all\n            if (elements.length === 0) {\n              return true\n            }\n\n            // Found but not visible\n            try {\n              const isDisplayed = await elements[0].isDisplayed()\n\n              return !isDisplayed\n            } catch (error) {\n              if (\n                error instanceof Error &&\n                (error.constructor.name === \"StaleElementReferenceError\" || error.message.includes(\"stale element reference\"))\n              ) {\n                return false\n              }\n\n              throw error\n            }\n          },\n          this.getTimeouts()\n        )\n      })\n    } finally {\n      await this.restoreTimeouts()\n    }\n  }\n\n  async _withRethrownErrors(callback) {\n    try {\n      return await callback()\n    } catch (error) {\n      if (error instanceof WebDriverError) {\n        throw new Error(`Selenium ${error.constructor.name}: ${error.message}`)\n      } else {\n        throw error\n      }\n    }\n  }\n\n  /**\n   * Gets notification messages\n   * @returns {Promise<string[]>}\n   */\n  async notificationMessages() {\n    const notificationMessageElements = await this.all(\"[data-class='notification-message']\", {useBaseSelector: false})\n    const notificationMessageTexts = []\n\n    for (const notificationMessageElement of notificationMessageElements) {\n      const text = await notificationMessageElement.getText()\n\n      notificationMessageTexts.push(text)\n    }\n\n    return notificationMessageTexts\n  }\n\n  /**\n   * Expects a notification message to appear and waits for it if necessary.\n   * @param {string} expectedNotificationMessage\n   * @param {NotificationMessageArgs} [args]\n   * @returns {Promise<void>}\n   */\n  async expectNotificationMessage(expectedNotificationMessage, args = {}) {\n    const {dismiss = true, ...restArgs} = args\n\n    if (Object.keys(restArgs).length > 0) {\n      throw new Error(`Unexpected args: ${Object.keys(restArgs).join(\", \")}`)\n    }\n\n    /** @type {string[]} */\n    const allDetectedNotificationMessages = []\n    let foundNotificationMessageElement\n    let foundNotificationMessageCount\n\n    await waitFor(async () => {\n      const notificationMessageElements = await this.all(\"[data-testid='notification-message']\", {useBaseSelector: false})\n\n      for (const notificationMessageElement of notificationMessageElements) {\n        const notificationMessage = (await this.getDriver().executeScript(\"return arguments[0].textContent;\", notificationMessageElement))?.trim() || await notificationMessageElement.getText()\n\n        if (!allDetectedNotificationMessages.includes(notificationMessage)) {\n          allDetectedNotificationMessages.push(notificationMessage)\n        }\n\n        if (notificationMessage == expectedNotificationMessage) {\n          foundNotificationMessageElement = notificationMessageElement\n          foundNotificationMessageCount = await notificationMessageElement.getAttribute(\"data-count\")\n          return\n        }\n      }\n\n      throw new Error(`Notification message ${expectedNotificationMessage} wasn't included in: ${allDetectedNotificationMessages.join(\", \")}`)\n    })\n\n    if (foundNotificationMessageElement && dismiss) {\n      await this.interact(foundNotificationMessageElement, \"click\") // Dismiss the notification message\n      if (!foundNotificationMessageCount) {\n        throw new Error(\"Expected notification message to have a data-count\")\n      }\n\n      await this.waitForNoSelector(`[data-testid='notification-message'][data-count='${foundNotificationMessageCount}']`, {useBaseSelector: false})\n    }\n  }\n\n  /** @returns {Promise<void>} */\n  async dismissNotificationMessages() {\n    const notificationMessageElements = await this.all(\"[data-class='notification-message']\", {useBaseSelector: false})\n\n    for (const notificationMessageElement of notificationMessageElements) {\n      await this.interact(notificationMessageElement, \"click\")\n    }\n\n    await this.waitForNoSelector(\"[data-class='notification-message']\", {useBaseSelector: false})\n  }\n\n  /**\n   * Indicates whether the system test has been started\n   * @returns {boolean}\n   */\n  isStarted() { return this._started }\n\n  /**\n   * Gets the HTML of the current page\n   * @returns {Promise<string>}\n   */\n  async getHTML() { return await this.getDriver().getPageSource() }\n\n  /**\n   * Starts the system test\n   * @returns {Promise<void>}\n   */\n  async start() {\n    this.debugLog(\"Start called\")\n    if (process.env.SYSTEM_TEST_HOST == \"expo-dev-server\") {\n      this.currentUrl = `http://${this._host}:${this._port}`\n      this.debugLog(`Using expo-dev-server at ${this.currentUrl}`)\n    } else if (process.env.SYSTEM_TEST_HOST == \"dist\") {\n      this.currentUrl = `http://${this._httpHost}:${this._httpPort}`\n\n      this.debugLog(`Spawning HTTP server for dist on ${this._httpHost}:${this._httpPort}`)\n      this.systemTestHttpServer = new SystemTestHttpServer({\n        host: this._httpHost,\n        port: this._httpPort,\n        debug: this._debug,\n        onError: this.onHttpServerError\n      })\n\n      this.debugLog(\"Starting HTTP server\")\n      await this.systemTestHttpServer.start()\n      this.debugLog(\"Checking HTTP server health\")\n      await this.systemTestHttpServer.assertReachable({timeoutMs: this.getTimeouts()})\n      this.debugLog(\"HTTP server started\")\n    } else {\n      throw new Error(\"Please set SYSTEM_TEST_HOST to 'expo-dev-server' or 'dist'\")\n    }\n\n    const options = new chrome.Options()\n\n    options.addArguments(\"--disable-dev-shm-usage\")\n    options.addArguments(\"--disable-gpu\")\n    options.addArguments(\"--headless=new\")\n    options.addArguments(\"--no-sandbox\")\n    options.addArguments(\"--window-size=1920,1080\")\n    this.debugLog(\"Chrome options configured\")\n\n    this.driver = new Builder()\n      .forBrowser(\"chrome\")\n      .setChromeOptions(options)\n      // @ts-expect-error\n      .setCapability(\"goog:loggingPrefs\", {browser: \"ALL\"})\n      .build()\n    this.debugLog(\"WebDriver built\")\n\n    await this.setTimeouts(10000)\n    this.debugLog(\"Timeouts set on driver\")\n\n    // Web socket server to communicate with browser\n    this.debugLog(\"Starting WebSocket server\")\n    await this.startWebSocketServer()\n    this.debugLog(\"WebSocket server started\")\n\n    // Visit the root page and wait for Expo to be loaded and the app to appear\n    this.debugLog(\"Visiting root path\")\n    const rootPath = this.getRootPath()\n    await this.driverVisit(rootPath)\n    this.debugLog(`Visited root path ${rootPath}`)\n\n    try {\n      await this.find(\"body > #root\", {useBaseSelector: false})\n      await this.findByTestID(\"systemTestingComponent\", {useBaseSelector: false, timeout: 30000, visible: true})\n      this.debugLog(\"Found root and systemTestingComponent\")\n    } catch (error) {\n      await this.takeScreenshot()\n      throw error\n    }\n\n    // Wait for client to connect\n    this.debugLog(\"Waiting for client WebSocket connection (opening)\")\n    this.debugLog(`WS state: ${this.ws?.readyState ?? \"none\"}`)\n    this.debugLog(\"waitForClientWebSocket\")\n    await this.waitForClientWebSocket()\n    this.debugLog(\"Client WebSocket connected\")\n\n    this._started = true\n    this.setBaseSelector(\"[data-testid='systemTestingComponent'][data-focussed='true']\")\n    this.debugLog(\"Start completed\")\n  }\n\n  /**\n   * @returns {string}\n   */\n  getRootPath() {\n    return this._rootPath ?? SystemTest.rootPath\n  }\n\n  /**\n   * @returns {string}\n   */\n  buildRootPath() {\n    if (!this._urlArgs) return SystemTest.rootPath\n\n    const url = new URL(SystemTest.rootPath, \"http://localhost\")\n    const appendParam = (key, value) => {\n      if (value === undefined || value === null) return\n      url.searchParams.append(key, String(value))\n    }\n\n    if (this._urlArgs instanceof URLSearchParams) {\n      for (const [key, value] of this._urlArgs) {\n        appendParam(key, value)\n      }\n    } else {\n      for (const [key, value] of Object.entries(this._urlArgs)) {\n        appendParam(key, value)\n      }\n    }\n\n    const rootPath =  `${url.pathname}${url.search}${url.hash}`\n\n    this.debugLog(`buildRootPath rootPath: ${rootPath}`)\n\n    return rootPath\n  }\n\n  /**\n   * Restores previously set timeouts\n   * @returns {Promise<void>}\n   */\n  async restoreTimeouts() {\n    if (!this.getTimeouts()) {\n      throw new Error(\"Timeouts haven't previously been set\")\n    }\n\n    await this.driverSetTimeouts(this.getTimeouts())\n  }\n\n  /**\n   * Sets driver timeouts\n   * @param {number} newTimeout\n   * @returns {Promise<void>}\n   */\n  async driverSetTimeouts(newTimeout) {\n    this._driverTimeouts = newTimeout\n    await this.getDriver().manage().setTimeouts({implicit: newTimeout})\n  }\n\n  /**\n   * Sets timeouts and stores the previous timeouts\n   * @param {number} newTimeout\n   * @returns {Promise<void>}\n   */\n  async setTimeouts(newTimeout) {\n    this._timeouts = newTimeout\n    await this.restoreTimeouts()\n  }\n\n  /**\n   * Waits for the client web socket to connect\n   * @returns {Promise<void>}\n   */\n  async waitForClientWebSocket() {\n    try {\n      await timeout({timeout: 30000}, () => new Promise((resolve, reject) => {\n        if (this.ws) {\n          resolve()\n          return\n        }\n\n        this.waitForClientWebSocketPromiseReject = reject\n        this.waitForClientWebSocketPromiseResolve = resolve\n      }))\n    } catch (error) {\n      delete this.waitForClientWebSocketPromiseReject\n      delete this.waitForClientWebSocketPromiseResolve\n      throw error\n    }\n  }\n\n  /**\n   * Starts the web socket server\n   * @returns {void}\n   */\n  startWebSocketServer() {\n    this.clientWss = new WebSocketServer({port: 1985})\n    this.clientWss.on(\"connection\", this.onWebSocketConnection)\n    this.clientWss.on(\"close\", this.onWebSocketClose)\n    this.clientWss.on(\"error\", (error) => {\n      this.debugError(error)\n\n      if (this.waitForClientWebSocketPromiseReject) {\n        this.waitForClientWebSocketPromiseReject(error instanceof Error ? error : new Error(String(error)))\n        delete this.waitForClientWebSocketPromiseReject\n        delete this.waitForClientWebSocketPromiseResolve\n      }\n    })\n  }\n\n  /**\n   * Sets the on command callback\n   * @param {function({type: string, data: Record<string, any>}): Promise<void>} callback\n   * @returns {void}\n   */\n  onCommand(callback) {\n    this._onCommandCallback = callback\n  }\n\n  /**\n   * Handles a command received from the browser\n   * @param {{data: {message: string, backtrace?: string, type: string, value: any[]}}} args\n   * @returns {Promise<any>}\n   */\n  onCommandReceived = async ({data}) => {\n    const type = data.type\n    let result\n\n    if (type == \"console.error\") {\n      const showMessage = !this.shouldIgnoreError(data)\n\n      if (showMessage) {\n        console.error(\"Browser error\", ...data.value)\n      }\n    } else if (type == \"console.log\") {\n      console.log(\"Browser log\", ...data.value)\n    } else if (type == \"error\" || data.type == \"unhandledrejection\") {\n      this.handleError(data)\n    } else if (this._onCommandCallback) {\n      result = await this._onCommandCallback({data, type})\n    } else {\n      console.error(`onWebSocketClientMessage unknown data (type ${type})`, data)\n    }\n\n    return result\n  }\n\n  /**\n   * Handles a new web socket connection\n   * @param {WebSocket} ws\n   * @returns {Promise<void>}\n   */\n  onWebSocketConnection = async (ws) => {\n    this.ws = ws\n    this.getCommunicator().ws = ws\n    this.getCommunicator().onOpen()\n\n    // @ts-expect-error\n    this.ws.on(\"error\", digg(this, \"communicator\", \"onError\"))\n\n    // @ts-expect-error\n    this.ws.on(\"message\", digg(this, \"communicator\", \"onMessage\"))\n\n    if (this.waitForClientWebSocketPromiseResolve) {\n      this.waitForClientWebSocketPromiseResolve()\n      delete this.waitForClientWebSocketPromiseResolve\n      delete this.waitForClientWebSocketPromiseReject\n    }\n  }\n\n  /** @returns {void} */\n  onWebSocketClose = () => {\n    this.ws = null\n    this.getCommunicator().ws = null\n\n    if (this.waitForClientWebSocketPromiseReject) {\n      this.waitForClientWebSocketPromiseReject(new Error(\"Client websocket closed before connecting\"))\n      delete this.waitForClientWebSocketPromiseReject\n      delete this.waitForClientWebSocketPromiseResolve\n    }\n  }\n\n  /**\n   * Handles an error reported from the browser\n   * @param {object} data\n   * @param {string} data.message\n   * @param {string} [data.backtrace]\n   * @returns {void}\n   */\n  handleError(data) {\n    if (this.shouldIgnoreError(data)) return\n\n    const error = new Error(`Browser error: ${data.message}`)\n\n    if (data.backtrace) {\n      error.stack = `${error.message}\\n${data.backtrace}\\n\\n${error.stack}`\n    }\n\n    console.error(error)\n  }\n\n  /**\n   * Stops the system test\n   * @returns {Promise<void>}\n   */\n  async stop() {\n    await this.stopScoundrel()\n    if (this.ws) {\n      this.ws.close()\n      this.ws = null\n    }\n    await this.closeWebSocketServer(this.clientWss, \"client WebSocket server\")\n    if (this.driver) {\n      await timeout({timeout: this.getTimeouts(), errorMessage: \"timeout while quitting WebDriver\"}, async () => await this.driver.quit())\n    }\n    if (this.systemTestHttpServer) {\n      await timeout({timeout: this.getTimeouts(), errorMessage: \"timeout while closing HTTP server\"}, async () => await this.systemTestHttpServer.close())\n    }\n  }\n\n  /**\n   * Fully tears down and restarts the system test instance.\n   * @returns {Promise<void>}\n   */\n  async reinitialize() {\n    await this.stop()\n\n    this._started = false\n    this._baseSelector = undefined\n    this.currentUrl = undefined\n    this.driver = undefined\n    this.ws = null\n    this.clientWss = undefined\n    this.scoundrelWss = undefined\n    this.server = undefined\n    this.serverWebSocket = undefined\n    this.systemTestHttpServer = undefined\n    this._httpServerError = undefined\n    this.waitForClientWebSocketPromiseReject = undefined\n    this.waitForClientWebSocketPromiseResolve = undefined\n    this.communicator = new SystemTestCommunicator({onCommand: this.onCommandReceived})\n\n    this.startScoundrel()\n    await this.start()\n  }\n\n  /**\n   * Visits a path in the browser\n   * @param {string} path\n   * @returns {Promise<void>}\n   */\n  async driverVisit(path) {\n    const url = `${this.currentUrl}${path}`\n\n    await this.getDriver().get(url)\n  }\n\n  /**\n   * Takes a screenshot, saves HTML and browser logs\n   * @returns {Promise<void>}\n   */\n  async takeScreenshot() {\n    const path = `${process.cwd()}/tmp/screenshots`\n\n    await fs.mkdir(path, {recursive: true})\n\n    const imageContent = await this.getDriver().takeScreenshot()\n    const now = new Date()\n    const screenshotPath = `${path}/${moment(now).format(\"YYYY-MM-DD-HH-MM-SS\")}.png`\n    const htmlPath = `${path}/${moment(now).format(\"YYYY-MM-DD-HH-MM-SS\")}.html`\n    const logsPath = `${path}/${moment(now).format(\"YYYY-MM-DD-HH-MM-SS\")}.logs.txt`\n    const logsText = await this.getBrowserLogs()\n    const html = await this.getHTML()\n    const htmlPretty = prettify(html)\n\n    await fs.writeFile(htmlPath, htmlPretty)\n    await fs.writeFile(logsPath, logsText.join(\"\\n\"))\n    await fs.writeFile(screenshotPath, imageContent, \"base64\")\n\n    console.log(\"Current URL:\", await this.getCurrentUrl())\n    console.log(\"Logs:\", logsPath)\n    console.log(\"Screenshot:\", screenshotPath)\n    console.log(\"HTML:\", htmlPath)\n  }\n\n  /**\n   * Visits a path in the browser\n   * @param {string} path\n   * @returns {Promise<void>}\n   */\n  async visit(path) {\n    await this.getCommunicator().sendCommand({type: \"visit\", path})\n  }\n\n  /**\n   * Dismisses to a path in the browser\n   * @param {string} path\n   * @returns {Promise<void>}\n   */\n  async dismissTo(path) {\n    await this.getCommunicator().sendCommand({type: \"dismissTo\", path})\n  }\n\n  /**\n   * @param {WebSocketServer | undefined} wss\n   * @param {string} [label]\n   * @returns {Promise<void>}\n   */\n  async closeWebSocketServer(wss, label = \"WebSocket server\") {\n    if (!wss) return\n\n    await timeout({timeout: this.getTimeouts(), errorMessage: `timeout while closing ${label}`}, async () => await new Promise((resolve, reject) => {\n      let settled = false\n      const terminateClient = (client) => {\n        try {\n          client.terminate()\n        } catch {\n          // Ignore termination errors\n        }\n      }\n      const settle = (callback, arg) => {\n        if (settled) return\n        settled = true\n        callback(arg)\n      }\n\n      wss.once(\"close\", () => settle(resolve))\n      wss.once(\"error\", (error) => settle(reject, error))\n      if (wss.clients && wss.clients.size > 0) {\n        wss.clients.forEach(terminateClient)\n      }\n      wss.close((error) => {\n        if (error) settle(reject, error)\n        else settle(resolve)\n      })\n    }))\n  }\n}\n"]}
@@ -2,10 +2,12 @@
2
2
  * A hook that provides system test capabilities.
3
3
  * @param {object} options - Options for the hook.
4
4
  * @param {function() : void} options.onInitialize - A callback function that is called when the system test browser helper is initialized.
5
+ * @param {function() : void} options.onFirstInitialize - A callback function that is called the first time the system test browser helper is initialized.
5
6
  * @returns {{enabled: boolean, systemTestBrowserHelper: SystemTestBrowserHelper}}
6
7
  */
7
- export default function useSystemTest({ onInitialize, ...restArgs }?: {
8
+ export default function useSystemTest({ onFirstInitialize, onInitialize, ...restArgs }?: {
8
9
  onInitialize: () => void;
10
+ onFirstInitialize: () => void;
9
11
  }): {
10
12
  enabled: boolean;
11
13
  systemTestBrowserHelper: SystemTestBrowserHelper;
@@ -36,9 +36,10 @@ function getSystemTestBrowserHelper() {
36
36
  * A hook that provides system test capabilities.
37
37
  * @param {object} options - Options for the hook.
38
38
  * @param {function() : void} options.onInitialize - A callback function that is called when the system test browser helper is initialized.
39
+ * @param {function() : void} options.onFirstInitialize - A callback function that is called the first time the system test browser helper is initialized.
39
40
  * @returns {{enabled: boolean, systemTestBrowserHelper: SystemTestBrowserHelper}}
40
41
  */
41
- export default function useSystemTest({ onInitialize, ...restArgs } = { onInitialize: undefined }) {
42
+ export default function useSystemTest({ onFirstInitialize, onInitialize, ...restArgs } = { onFirstInitialize: undefined, onInitialize: undefined }) {
42
43
  if (!useMemo)
43
44
  throw new Error("[useSystemTest] React.useMemo is not available");
44
45
  if (!useCallback)
@@ -78,12 +79,13 @@ export default function useSystemTest({ onInitialize, ...restArgs } = { onInitia
78
79
  useMemo(() => {
79
80
  if (enabled && !shared.initialized) {
80
81
  shared.initialized = true;
82
+ shared.systemTestBrowserHelper?.onFirstInitialize(onFirstInitialize);
81
83
  shared.systemTestBrowserHelper?.onInitialize(onInitialize);
82
84
  }
83
- }, [enabled, onInitialize, shared.systemTestBrowserHelper]);
85
+ }, [enabled, onFirstInitialize, onInitialize, shared.systemTestBrowserHelper]);
84
86
  const restArgsKeys = Object.keys(restArgs);
85
87
  if (restArgsKeys.length > 0)
86
88
  throw new Error(`Unknown arguments given to useSystemTest: ${restArgsKeys.join(", ")}`);
87
89
  return result;
88
90
  }
89
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"use-system-test.js","sourceRoot":"","sources":["../src/use-system-test.js"],"names":[],"mappings":"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","sourcesContent":["import qs from \"qs\"\nimport SystemTestBrowserHelper from \"./system-test-browser-helper.js\"\nimport {useCallback, useMemo} from \"react\"\nimport useEventEmitter from \"ya-use-event-emitter\"\nimport {useRouter} from \"expo-router\"\n\nconst shared = {\n  initialized: false,\n  systemTestBrowserHelper: null\n}\n\n/**\n * @returns {boolean}\n */\nfunction isSystemTestEnabled() {\n  let enabled = false\n  const initialUrl = globalThis.location?.href\n\n  if (initialUrl) {\n    const queryString = initialUrl.substring(initialUrl.indexOf(\"?\") + 1, initialUrl.length)\n    const queryParams = qs.parse(queryString)\n\n    if (queryParams.systemTest == \"true\") {\n      enabled = true\n    }\n  }\n\n  return enabled\n}\n\n/**\n * @returns {SystemTestBrowserHelper}\n */\nfunction getSystemTestBrowserHelper() {\n  if (!shared.systemTestBrowserHelper) {\n    shared.systemTestBrowserHelper = new SystemTestBrowserHelper()\n    shared.systemTestBrowserHelper.enableOnBrowser()\n  }\n\n  return shared.systemTestBrowserHelper\n}\n\n/**\n * A hook that provides system test capabilities.\n * @param {object} options - Options for the hook.\n * @param {function() : void} options.onInitialize - A callback function that is called when the system test browser helper is initialized.\n * @returns {{enabled: boolean, systemTestBrowserHelper: SystemTestBrowserHelper}}\n */\nexport default function useSystemTest({onInitialize, ...restArgs} = {onInitialize: undefined}) {\n  if (!useMemo) throw new Error(\"[useSystemTest] React.useMemo is not available\")\n  if (!useCallback) throw new Error(\"[useSystemTest] React.useCallback is not available\")\n\n  let router = null\n\n  try {\n    router = useRouter()\n  } catch (error) {\n    console.error(\"[useSystemTest] useRouter unavailable:\", error)\n  }\n  const enabled = useMemo(() => isSystemTestEnabled(), [])\n  const systemTestBrowserHelper = enabled ? getSystemTestBrowserHelper() : null\n  const result = useMemo(() => ({enabled, systemTestBrowserHelper}), [enabled, systemTestBrowserHelper])\n  const instanceShared = useMemo(() => ({enabled: false, router: null}), [])\n\n  instanceShared.enabled = enabled\n  instanceShared.router = router\n\n  // Resets navigation when instructed by the system test browser helper\n  const onSystemTestBrowserHelperDismissTo = useCallback(({path}) => {\n    if (instanceShared.enabled) {\n      try {\n        instanceShared.router?.dismissTo(path)\n      } catch (error) {\n        console.error(`Failed to dismiss to path \"${path}\": ${error.message}`)\n      }\n    }\n  }, [])\n\n  useEventEmitter(shared.systemTestBrowserHelper?.getEvents(), \"dismissTo\", onSystemTestBrowserHelperDismissTo)\n\n\n  // Navigates when instructed by the system test browser helper and keeping history of screens\n  const onSystemTestBrowserHelperNavigate = useCallback(({path}) => {\n    if (instanceShared.enabled) {\n      instanceShared.router?.navigate(path)\n    }\n  }, [])\n\n  useEventEmitter(shared.systemTestBrowserHelper?.getEvents(), \"navigate\", onSystemTestBrowserHelperNavigate)\n\n  useMemo(() => {\n    if (enabled && !shared.initialized) {\n      shared.initialized = true\n      shared.systemTestBrowserHelper?.onInitialize(onInitialize)\n    }\n  }, [enabled, onInitialize, shared.systemTestBrowserHelper])\n\n  const restArgsKeys = Object.keys(restArgs)\n\n  if (restArgsKeys.length > 0) throw new Error(`Unknown arguments given to useSystemTest: ${restArgsKeys.join(\", \")}`)\n\n  return result\n}\n"]}
91
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"use-system-test.js","sourceRoot":"","sources":["../src/use-system-test.js"],"names":[],"mappings":"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","sourcesContent":["import qs from \"qs\"\nimport SystemTestBrowserHelper from \"./system-test-browser-helper.js\"\nimport {useCallback, useMemo} from \"react\"\nimport useEventEmitter from \"ya-use-event-emitter\"\nimport {useRouter} from \"expo-router\"\n\nconst shared = {\n  initialized: false,\n  systemTestBrowserHelper: null\n}\n\n/**\n * @returns {boolean}\n */\nfunction isSystemTestEnabled() {\n  let enabled = false\n  const initialUrl = globalThis.location?.href\n\n  if (initialUrl) {\n    const queryString = initialUrl.substring(initialUrl.indexOf(\"?\") + 1, initialUrl.length)\n    const queryParams = qs.parse(queryString)\n\n    if (queryParams.systemTest == \"true\") {\n      enabled = true\n    }\n  }\n\n  return enabled\n}\n\n/**\n * @returns {SystemTestBrowserHelper}\n */\nfunction getSystemTestBrowserHelper() {\n  if (!shared.systemTestBrowserHelper) {\n    shared.systemTestBrowserHelper = new SystemTestBrowserHelper()\n    shared.systemTestBrowserHelper.enableOnBrowser()\n  }\n\n  return shared.systemTestBrowserHelper\n}\n\n/**\n * A hook that provides system test capabilities.\n * @param {object} options - Options for the hook.\n * @param {function() : void} options.onInitialize - A callback function that is called when the system test browser helper is initialized.\n * @param {function() : void} options.onFirstInitialize - A callback function that is called the first time the system test browser helper is initialized.\n * @returns {{enabled: boolean, systemTestBrowserHelper: SystemTestBrowserHelper}}\n */\nexport default function useSystemTest({onFirstInitialize, onInitialize, ...restArgs} = {onFirstInitialize: undefined, onInitialize: undefined}) {\n  if (!useMemo) throw new Error(\"[useSystemTest] React.useMemo is not available\")\n  if (!useCallback) throw new Error(\"[useSystemTest] React.useCallback is not available\")\n\n  let router = null\n\n  try {\n    router = useRouter()\n  } catch (error) {\n    console.error(\"[useSystemTest] useRouter unavailable:\", error)\n  }\n  const enabled = useMemo(() => isSystemTestEnabled(), [])\n  const systemTestBrowserHelper = enabled ? getSystemTestBrowserHelper() : null\n  const result = useMemo(() => ({enabled, systemTestBrowserHelper}), [enabled, systemTestBrowserHelper])\n  const instanceShared = useMemo(() => ({enabled: false, router: null}), [])\n\n  instanceShared.enabled = enabled\n  instanceShared.router = router\n\n  // Resets navigation when instructed by the system test browser helper\n  const onSystemTestBrowserHelperDismissTo = useCallback(({path}) => {\n    if (instanceShared.enabled) {\n      try {\n        instanceShared.router?.dismissTo(path)\n      } catch (error) {\n        console.error(`Failed to dismiss to path \"${path}\": ${error.message}`)\n      }\n    }\n  }, [])\n\n  useEventEmitter(shared.systemTestBrowserHelper?.getEvents(), \"dismissTo\", onSystemTestBrowserHelperDismissTo)\n\n\n  // Navigates when instructed by the system test browser helper and keeping history of screens\n  const onSystemTestBrowserHelperNavigate = useCallback(({path}) => {\n    if (instanceShared.enabled) {\n      instanceShared.router?.navigate(path)\n    }\n  }, [])\n\n  useEventEmitter(shared.systemTestBrowserHelper?.getEvents(), \"navigate\", onSystemTestBrowserHelperNavigate)\n\n  useMemo(() => {\n    if (enabled && !shared.initialized) {\n      shared.initialized = true\n      shared.systemTestBrowserHelper?.onFirstInitialize(onFirstInitialize)\n      shared.systemTestBrowserHelper?.onInitialize(onInitialize)\n    }\n  }, [enabled, onFirstInitialize, onInitialize, shared.systemTestBrowserHelper])\n\n  const restArgsKeys = Object.keys(restArgs)\n\n  if (restArgsKeys.length > 0) throw new Error(`Unknown arguments given to useSystemTest: ${restArgsKeys.join(\", \")}`)\n\n  return result\n}\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "system-testing",
3
- "version": "1.0.69",
3
+ "version": "1.0.71",
4
4
  "description": "System testing with Selenium and browsers.",
5
5
  "main": "build/index.js",
6
6
  "types": "build/index.d.ts",