system-testing 1.0.85 → 1.0.86
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.
|
@@ -1,3 +1,17 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Ensures Chrome Appium capabilities use a caller-owned user-data directory.
|
|
3
|
+
* This avoids repeated sessions leaking temp profiles under `/tmp` in CI.
|
|
4
|
+
* @param {object} args
|
|
5
|
+
* @param {string} [args.browserName]
|
|
6
|
+
* @param {Record<string, any>} args.capabilities
|
|
7
|
+
* @param {string} [args.tempRootDir]
|
|
8
|
+
* @returns {Promise<string | undefined>}
|
|
9
|
+
*/
|
|
10
|
+
export function ensureChromeUserDataDirCapability({ browserName, capabilities, tempRootDir }: {
|
|
11
|
+
browserName?: string;
|
|
12
|
+
capabilities: Record<string, any>;
|
|
13
|
+
tempRootDir?: string;
|
|
14
|
+
}): Promise<string | undefined>;
|
|
1
15
|
/**
|
|
2
16
|
* @typedef {object} AppiumDriverOptions
|
|
3
17
|
* @property {string} [serverUrl] Remote Appium server URL to connect to.
|
|
@@ -19,8 +33,12 @@
|
|
|
19
33
|
* Appium driver backed by the Appium server package.
|
|
20
34
|
*/
|
|
21
35
|
export default class AppiumDriver extends WebDriverDriver {
|
|
36
|
+
/** @type {string | undefined} */
|
|
37
|
+
chromeUserDataDir: string | undefined;
|
|
22
38
|
serverUrl: any;
|
|
23
39
|
appiumServer: import("@appium/types").AppiumServer;
|
|
40
|
+
/** @returns {Promise<void>} */
|
|
41
|
+
cleanupChromeUserDataDir(): Promise<void>;
|
|
24
42
|
/**
|
|
25
43
|
* @param {string} testId
|
|
26
44
|
* @param {FindArgs} [args]
|
|
@@ -1,3 +1,5 @@
|
|
|
1
|
+
import fs from "node:fs/promises";
|
|
2
|
+
import path from "node:path";
|
|
1
3
|
import { Builder, By } from "selenium-webdriver";
|
|
2
4
|
import timeout from "awaitery/build/timeout.js";
|
|
3
5
|
import WebDriverDriver from "./webdriver-driver.js";
|
|
@@ -11,6 +13,34 @@ function errorWithCause(message, cause) {
|
|
|
11
13
|
error.cause = cause;
|
|
12
14
|
return error;
|
|
13
15
|
}
|
|
16
|
+
/**
|
|
17
|
+
* Ensures Chrome Appium capabilities use a caller-owned user-data directory.
|
|
18
|
+
* This avoids repeated sessions leaking temp profiles under `/tmp` in CI.
|
|
19
|
+
* @param {object} args
|
|
20
|
+
* @param {string} [args.browserName]
|
|
21
|
+
* @param {Record<string, any>} args.capabilities
|
|
22
|
+
* @param {string} [args.tempRootDir]
|
|
23
|
+
* @returns {Promise<string | undefined>}
|
|
24
|
+
*/
|
|
25
|
+
export async function ensureChromeUserDataDirCapability({ browserName, capabilities, tempRootDir = path.join(process.cwd(), "tmp", "appium-chrome-user-data") }) {
|
|
26
|
+
const resolvedBrowserName = capabilities.browserName ?? browserName;
|
|
27
|
+
if (typeof resolvedBrowserName !== "string" || resolvedBrowserName.toLowerCase() !== "chrome") {
|
|
28
|
+
return undefined;
|
|
29
|
+
}
|
|
30
|
+
const chromeOptions = capabilities["goog:chromeOptions"];
|
|
31
|
+
const args = Array.isArray(chromeOptions?.args) ? [...chromeOptions.args] : [];
|
|
32
|
+
if (args.some((arg) => typeof arg === "string" && arg.startsWith("--user-data-dir="))) {
|
|
33
|
+
return undefined;
|
|
34
|
+
}
|
|
35
|
+
await fs.mkdir(tempRootDir, { recursive: true });
|
|
36
|
+
const userDataDir = await fs.mkdtemp(path.join(tempRootDir, "profile-"));
|
|
37
|
+
args.push(`--user-data-dir=${userDataDir}`);
|
|
38
|
+
capabilities["goog:chromeOptions"] = {
|
|
39
|
+
...(chromeOptions ?? {}),
|
|
40
|
+
args
|
|
41
|
+
};
|
|
42
|
+
return userDataDir;
|
|
43
|
+
}
|
|
14
44
|
/**
|
|
15
45
|
* @typedef {object} AppiumDriverOptions
|
|
16
46
|
* @property {string} [serverUrl] Remote Appium server URL to connect to.
|
|
@@ -32,6 +62,11 @@ function errorWithCause(message, cause) {
|
|
|
32
62
|
* Appium driver backed by the Appium server package.
|
|
33
63
|
*/
|
|
34
64
|
export default class AppiumDriver extends WebDriverDriver {
|
|
65
|
+
constructor() {
|
|
66
|
+
super(...arguments);
|
|
67
|
+
/** @type {string | undefined} */
|
|
68
|
+
this.chromeUserDataDir = undefined;
|
|
69
|
+
}
|
|
35
70
|
/**
|
|
36
71
|
* @returns {Promise<void>}
|
|
37
72
|
*/
|
|
@@ -59,12 +94,22 @@ export default class AppiumDriver extends WebDriverDriver {
|
|
|
59
94
|
if (capabilities.browserName === undefined) {
|
|
60
95
|
capabilities.browserName = "";
|
|
61
96
|
}
|
|
97
|
+
this.chromeUserDataDir = await ensureChromeUserDataDirCapability({
|
|
98
|
+
browserName: this.options.browserName,
|
|
99
|
+
capabilities
|
|
100
|
+
});
|
|
62
101
|
const builder = new Builder().usingServer(this.serverUrl);
|
|
63
102
|
if (Object.keys(capabilities).length > 0) {
|
|
64
103
|
builder.withCapabilities(capabilities);
|
|
65
104
|
}
|
|
66
|
-
|
|
67
|
-
|
|
105
|
+
try {
|
|
106
|
+
const webDriver = await builder.build();
|
|
107
|
+
this.setWebDriver(webDriver);
|
|
108
|
+
}
|
|
109
|
+
catch (error) {
|
|
110
|
+
await this.cleanupChromeUserDataDir();
|
|
111
|
+
throw error;
|
|
112
|
+
}
|
|
68
113
|
}
|
|
69
114
|
/**
|
|
70
115
|
* @returns {Promise<void>}
|
|
@@ -78,8 +123,17 @@ export default class AppiumDriver extends WebDriverDriver {
|
|
|
78
123
|
await timeout({ timeout: this.getTimeouts(), errorMessage: "timeout while closing Appium server" }, async () => await this.appiumServer.close());
|
|
79
124
|
}
|
|
80
125
|
this.appiumServer = undefined;
|
|
126
|
+
await this.cleanupChromeUserDataDir();
|
|
81
127
|
}
|
|
82
128
|
}
|
|
129
|
+
/** @returns {Promise<void>} */
|
|
130
|
+
async cleanupChromeUserDataDir() {
|
|
131
|
+
if (!this.chromeUserDataDir)
|
|
132
|
+
return;
|
|
133
|
+
const chromeUserDataDir = this.chromeUserDataDir;
|
|
134
|
+
this.chromeUserDataDir = undefined;
|
|
135
|
+
await fs.rm(chromeUserDataDir, { recursive: true, force: true });
|
|
136
|
+
}
|
|
83
137
|
/**
|
|
84
138
|
* @returns {Promise<string[]>}
|
|
85
139
|
*/
|
|
@@ -315,4 +369,4 @@ export default class AppiumDriver extends WebDriverDriver {
|
|
|
315
369
|
return `http://${address}:${port}${basePath}`;
|
|
316
370
|
}
|
|
317
371
|
}
|
|
318
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"appium-driver.js","sourceRoot":"","sources":["../../src/drivers/appium-driver.js"],"names":[],"mappings":"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","sourcesContent":["import {Builder, By} from \"selenium-webdriver\"\nimport timeout from \"awaitery/build/timeout.js\"\nimport WebDriverDriver from \"./webdriver-driver.js\"\n\n/**\n * @param {string} message\n * @param {unknown} cause\n * @returns {Error & {cause: unknown}}\n */\nfunction errorWithCause(message, cause) {\n  const error = /** @type {Error & {cause: unknown}} */ (new Error(message))\n  error.cause = cause\n  return error\n}\n\n/**\n * @typedef {object} AppiumDriverOptions\n * @property {string} [serverUrl] Remote Appium server URL to connect to.\n * @property {Record<string, any>} [serverArgs] Options passed to the Appium server.\n * @property {string[]} [useDrivers] Appium driver names to load when starting the server.\n * @property {Record<string, any>} [capabilities] Desired capabilities for the session.\n * @property {string} [browserName] Browser name for web sessions.\n * @property {\"accessibilityId\"|\"css\"|\"id\"} [testIdStrategy] Strategy for resolving test IDs.\n * @property {string} [testIdAttribute] Attribute name when using the CSS test ID strategy.\n */\n/**\n * @typedef {object} FindArgs\n * @property {number} [timeout] Override timeout for lookup.\n * @property {boolean | null} [visible] Whether to require elements to be visible (`true`) or hidden (`false`). Use `null` to disable visibility filtering.\n * @property {boolean} [scrollTo] Whether to scroll found elements into view before returning them.\n * @property {boolean} [useBaseSelector] Whether to scope by the base selector.\n */\n\n/**\n * Appium driver backed by the Appium server package.\n */\nexport default class AppiumDriver extends WebDriverDriver {\n  /**\n   * @returns {Promise<void>}\n   */\n  async start() {\n    const serverArgs = {...(this.options.serverArgs ?? {})}\n\n    if (this.options.useDrivers && !serverArgs.useDrivers) {\n      serverArgs.useDrivers = this.options.useDrivers\n    }\n\n    if (this.options.serverUrl) {\n      this.serverUrl = this.options.serverUrl\n    } else {\n      const appiumModule = await import(\"appium\")\n      const appiumMain = appiumModule.main ?? appiumModule.default?.main\n\n      if (!appiumMain) {\n        throw new Error(\"Appium main() is unavailable from the appium package\")\n      }\n\n      this.appiumServer = await appiumMain(serverArgs)\n      this.serverUrl = this.buildServerUrl(serverArgs)\n    }\n\n    const capabilities = this.options.capabilities ? {...this.options.capabilities} : {}\n\n    if (this.options.browserName && !capabilities.browserName) {\n      capabilities.browserName = this.options.browserName\n    }\n    if (capabilities.browserName === undefined) {\n      capabilities.browserName = \"\"\n    }\n\n    const builder = new Builder().usingServer(this.serverUrl)\n\n    if (Object.keys(capabilities).length > 0) {\n      builder.withCapabilities(capabilities)\n    }\n\n    const webDriver = await builder.build()\n\n    this.setWebDriver(webDriver)\n  }\n\n  /**\n   * @returns {Promise<void>}\n   */\n  async stop() {\n    try {\n      await super.stop()\n    } finally {\n      if (this.appiumServer?.close) {\n        await timeout({timeout: this.getTimeouts(), errorMessage: \"timeout while closing Appium server\"}, async () => await this.appiumServer.close())\n      }\n\n      this.appiumServer = undefined\n    }\n  }\n\n  /**\n   * @returns {Promise<string[]>}\n   */\n  async getBrowserLogs() {\n    const platformName = this.options.capabilities?.platformName\n    const browserName = this.options.capabilities?.browserName ?? this.options.browserName\n    const isAndroid = typeof platformName === \"string\" && platformName.toLowerCase() === \"android\"\n\n    if (isAndroid && !browserName) {\n      let entries\n\n      try {\n        entries = await this.getWebDriver().manage().logs().get(\"logcat\")\n      } catch {\n        return []\n      }\n\n      const logcatLogs = []\n\n      for (const entry of entries) {\n        const messageMatch = entry.message.match(/^(.+) (\\d+):(\\d+) (.+)$/)\n        const message = messageMatch ? messageMatch[4] : entry.message\n\n        logcatLogs.push(`${entry.level.name}: ${message}`)\n      }\n\n      return logcatLogs\n    }\n\n    return await super.getBrowserLogs()\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) {\n    const testIdStrategy = this.options.testIdStrategy ?? \"accessibilityId\"\n\n    if (testIdStrategy === \"css\") {\n      const testIdAttribute = this.options.testIdAttribute ?? \"data-testid\"\n      return await this.find(`[${testIdAttribute}='${testID}']`, args)\n    }\n    if (testIdStrategy === \"id\") {\n      return await this.findById(testID, args)\n    }\n\n    return await this.findByAccessibilityId(testID, args)\n  }\n\n  /**\n   * @param {string} testId\n   * @param {FindArgs} [args]\n   * @returns {Promise<import(\"selenium-webdriver\").WebElement>}\n   */\n  async findByAccessibilityId(testId, args = {}) {\n    const startTime = Date.now()\n    /** @type {import(\"selenium-webdriver\").WebElement[]} */\n    let elements\n\n    try {\n      elements = await this.allByAccessibilityId(testId, args)\n    } catch (error) {\n      if (error instanceof Error) {\n        throw errorWithCause(`${error.constructor.name} - ${error.message} (accessibility id: ${testId})`, error)\n      }\n\n      throw errorWithCause(`${typeof error} - ${error} (accessibility id: ${testId})`, error)\n    }\n\n    if (elements.length > 1) {\n      throw new Error(`More than 1 elements (${elements.length}) was found by accessibility id: ${testId}`)\n    }\n\n    if (!elements[0]) {\n      const elapsedSeconds = (Date.now() - startTime) / 1000\n      throw new Error(`Element couldn't be found after ${elapsedSeconds.toFixed(2)}s by accessibility id: ${testId}`)\n    }\n\n    return elements[0]\n  }\n\n  /**\n   * @param {string} testId\n   * @param {FindArgs} [args]\n   * @returns {Promise<import(\"selenium-webdriver\").WebElement[]>}\n   */\n  async allByAccessibilityId(testId, args = {}) {\n    const {scrollTo = false, visible = true, timeout, ...restArgs} = args\n    const restArgsKeys = Object.keys(restArgs).filter((key) => key !== \"useBaseSelector\")\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 startTime = Date.now()\n    const getTimeLeft = () => Math.max(actualTimeout - (Date.now() - startTime), 0)\n    const getElements = async () => {\n      const foundElements = await this.getWebDriver().findElements(new By(\"accessibility id\", testId))\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.getWebDriver().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 Error && error.constructor.name === \"TimeoutError\" && getTimeLeft() > 0) {\n          continue\n        }\n\n        throw errorWithCause(`Couldn't get elements with accessibility id: ${testId}: ${error instanceof Error ? error.message : error}`, error)\n      }\n    }\n\n    if (scrollTo) {\n      for (const element of elements) {\n        await this.scrollElementIntoView(element)\n      }\n    }\n\n    return elements\n  }\n\n  /**\n   * @param {string} testId\n   * @param {FindArgs} [args]\n   * @returns {Promise<import(\"selenium-webdriver\").WebElement>}\n   */\n  async findById(testId, args = {}) {\n    const startTime = Date.now()\n    /** @type {import(\"selenium-webdriver\").WebElement[]} */\n    let elements\n\n    try {\n      elements = await this.allById(testId, args)\n    } catch (error) {\n      if (error instanceof Error) {\n        throw errorWithCause(`${error.constructor.name} - ${error.message} (id: ${testId})`, error)\n      }\n\n      throw errorWithCause(`${typeof error} - ${error} (id: ${testId})`, error)\n    }\n\n    if (elements.length > 1) {\n      throw new Error(`More than 1 elements (${elements.length}) was found by id: ${testId}`)\n    }\n\n    if (!elements[0]) {\n      const elapsedSeconds = (Date.now() - startTime) / 1000\n      throw new Error(`Element couldn't be found after ${elapsedSeconds.toFixed(2)}s by id: ${testId}`)\n    }\n\n    return elements[0]\n  }\n\n  /**\n   * @param {string} testId\n   * @param {FindArgs} [args]\n   * @returns {Promise<import(\"selenium-webdriver\").WebElement[]>}\n   */\n  async allById(testId, args = {}) {\n    const {scrollTo = false, visible = true, timeout, ...restArgs} = args\n    const restArgsKeys = Object.keys(restArgs).filter((key) => key !== \"useBaseSelector\")\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 startTime = Date.now()\n    const getTimeLeft = () => Math.max(actualTimeout - (Date.now() - startTime), 0)\n    const getElements = async () => {\n      const foundElements = await this.getWebDriver().findElements(By.id(testId))\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.getWebDriver().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 Error && error.constructor.name === \"TimeoutError\" && getTimeLeft() > 0) {\n          continue\n        }\n\n        throw errorWithCause(`Couldn't get elements with id: ${testId}: ${error instanceof Error ? error.message : error}`, error)\n      }\n    }\n\n    if (scrollTo) {\n      for (const element of elements) {\n        await this.scrollElementIntoView(element)\n      }\n    }\n\n    return elements\n  }\n\n  /**\n   * @param {Record<string, any>} serverArgs\n   * @returns {string}\n   */\n  buildServerUrl(serverArgs) {\n    const address = serverArgs.address ?? \"127.0.0.1\"\n    const port = serverArgs.port ?? 4723\n    const basePath = serverArgs.basePath ? (serverArgs.basePath.startsWith(\"/\") ? serverArgs.basePath : `/${serverArgs.basePath}`) : \"\"\n\n    return `http://${address}:${port}${basePath}`\n  }\n}\n"]}
|
|
372
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"appium-driver.js","sourceRoot":"","sources":["../../src/drivers/appium-driver.js"],"names":[],"mappings":"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","sourcesContent":["import fs from \"node:fs/promises\"\nimport path from \"node:path\"\nimport {Builder, By} from \"selenium-webdriver\"\nimport timeout from \"awaitery/build/timeout.js\"\nimport WebDriverDriver from \"./webdriver-driver.js\"\n\n/**\n * @param {string} message\n * @param {unknown} cause\n * @returns {Error & {cause: unknown}}\n */\nfunction errorWithCause(message, cause) {\n  const error = /** @type {Error & {cause: unknown}} */ (new Error(message))\n  error.cause = cause\n  return error\n}\n\n/**\n * Ensures Chrome Appium capabilities use a caller-owned user-data directory.\n * This avoids repeated sessions leaking temp profiles under `/tmp` in CI.\n * @param {object} args\n * @param {string} [args.browserName]\n * @param {Record<string, any>} args.capabilities\n * @param {string} [args.tempRootDir]\n * @returns {Promise<string | undefined>}\n */\nexport async function ensureChromeUserDataDirCapability({browserName, capabilities, tempRootDir = path.join(process.cwd(), \"tmp\", \"appium-chrome-user-data\")}) {\n  const resolvedBrowserName = capabilities.browserName ?? browserName\n\n  if (typeof resolvedBrowserName !== \"string\" || resolvedBrowserName.toLowerCase() !== \"chrome\") {\n    return undefined\n  }\n\n  const chromeOptions = capabilities[\"goog:chromeOptions\"]\n  const args = Array.isArray(chromeOptions?.args) ? [...chromeOptions.args] : []\n\n  if (args.some((arg) => typeof arg === \"string\" && arg.startsWith(\"--user-data-dir=\"))) {\n    return undefined\n  }\n\n  await fs.mkdir(tempRootDir, {recursive: true})\n\n  const userDataDir = await fs.mkdtemp(path.join(tempRootDir, \"profile-\"))\n\n  args.push(`--user-data-dir=${userDataDir}`)\n  capabilities[\"goog:chromeOptions\"] = {\n    ...(chromeOptions ?? {}),\n    args\n  }\n\n  return userDataDir\n}\n\n/**\n * @typedef {object} AppiumDriverOptions\n * @property {string} [serverUrl] Remote Appium server URL to connect to.\n * @property {Record<string, any>} [serverArgs] Options passed to the Appium server.\n * @property {string[]} [useDrivers] Appium driver names to load when starting the server.\n * @property {Record<string, any>} [capabilities] Desired capabilities for the session.\n * @property {string} [browserName] Browser name for web sessions.\n * @property {\"accessibilityId\"|\"css\"|\"id\"} [testIdStrategy] Strategy for resolving test IDs.\n * @property {string} [testIdAttribute] Attribute name when using the CSS test ID strategy.\n */\n/**\n * @typedef {object} FindArgs\n * @property {number} [timeout] Override timeout for lookup.\n * @property {boolean | null} [visible] Whether to require elements to be visible (`true`) or hidden (`false`). Use `null` to disable visibility filtering.\n * @property {boolean} [scrollTo] Whether to scroll found elements into view before returning them.\n * @property {boolean} [useBaseSelector] Whether to scope by the base selector.\n */\n\n/**\n * Appium driver backed by the Appium server package.\n */\nexport default class AppiumDriver extends WebDriverDriver {\n  /** @type {string | undefined} */\n  chromeUserDataDir = undefined\n\n  /**\n   * @returns {Promise<void>}\n   */\n  async start() {\n    const serverArgs = {...(this.options.serverArgs ?? {})}\n\n    if (this.options.useDrivers && !serverArgs.useDrivers) {\n      serverArgs.useDrivers = this.options.useDrivers\n    }\n\n    if (this.options.serverUrl) {\n      this.serverUrl = this.options.serverUrl\n    } else {\n      const appiumModule = await import(\"appium\")\n      const appiumMain = appiumModule.main ?? appiumModule.default?.main\n\n      if (!appiumMain) {\n        throw new Error(\"Appium main() is unavailable from the appium package\")\n      }\n\n      this.appiumServer = await appiumMain(serverArgs)\n      this.serverUrl = this.buildServerUrl(serverArgs)\n    }\n\n    const capabilities = this.options.capabilities ? {...this.options.capabilities} : {}\n\n    if (this.options.browserName && !capabilities.browserName) {\n      capabilities.browserName = this.options.browserName\n    }\n    if (capabilities.browserName === undefined) {\n      capabilities.browserName = \"\"\n    }\n\n    this.chromeUserDataDir = await ensureChromeUserDataDirCapability({\n      browserName: this.options.browserName,\n      capabilities\n    })\n\n    const builder = new Builder().usingServer(this.serverUrl)\n\n    if (Object.keys(capabilities).length > 0) {\n      builder.withCapabilities(capabilities)\n    }\n\n    try {\n      const webDriver = await builder.build()\n\n      this.setWebDriver(webDriver)\n    } catch (error) {\n      await this.cleanupChromeUserDataDir()\n      throw error\n    }\n  }\n\n  /**\n   * @returns {Promise<void>}\n   */\n  async stop() {\n    try {\n      await super.stop()\n    } finally {\n      if (this.appiumServer?.close) {\n        await timeout({timeout: this.getTimeouts(), errorMessage: \"timeout while closing Appium server\"}, async () => await this.appiumServer.close())\n      }\n\n      this.appiumServer = undefined\n      await this.cleanupChromeUserDataDir()\n    }\n  }\n\n  /** @returns {Promise<void>} */\n  async cleanupChromeUserDataDir() {\n    if (!this.chromeUserDataDir) return\n\n    const chromeUserDataDir = this.chromeUserDataDir\n\n    this.chromeUserDataDir = undefined\n    await fs.rm(chromeUserDataDir, {recursive: true, force: true})\n  }\n\n  /**\n   * @returns {Promise<string[]>}\n   */\n  async getBrowserLogs() {\n    const platformName = this.options.capabilities?.platformName\n    const browserName = this.options.capabilities?.browserName ?? this.options.browserName\n    const isAndroid = typeof platformName === \"string\" && platformName.toLowerCase() === \"android\"\n\n    if (isAndroid && !browserName) {\n      let entries\n\n      try {\n        entries = await this.getWebDriver().manage().logs().get(\"logcat\")\n      } catch {\n        return []\n      }\n\n      const logcatLogs = []\n\n      for (const entry of entries) {\n        const messageMatch = entry.message.match(/^(.+) (\\d+):(\\d+) (.+)$/)\n        const message = messageMatch ? messageMatch[4] : entry.message\n\n        logcatLogs.push(`${entry.level.name}: ${message}`)\n      }\n\n      return logcatLogs\n    }\n\n    return await super.getBrowserLogs()\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) {\n    const testIdStrategy = this.options.testIdStrategy ?? \"accessibilityId\"\n\n    if (testIdStrategy === \"css\") {\n      const testIdAttribute = this.options.testIdAttribute ?? \"data-testid\"\n      return await this.find(`[${testIdAttribute}='${testID}']`, args)\n    }\n    if (testIdStrategy === \"id\") {\n      return await this.findById(testID, args)\n    }\n\n    return await this.findByAccessibilityId(testID, args)\n  }\n\n  /**\n   * @param {string} testId\n   * @param {FindArgs} [args]\n   * @returns {Promise<import(\"selenium-webdriver\").WebElement>}\n   */\n  async findByAccessibilityId(testId, args = {}) {\n    const startTime = Date.now()\n    /** @type {import(\"selenium-webdriver\").WebElement[]} */\n    let elements\n\n    try {\n      elements = await this.allByAccessibilityId(testId, args)\n    } catch (error) {\n      if (error instanceof Error) {\n        throw errorWithCause(`${error.constructor.name} - ${error.message} (accessibility id: ${testId})`, error)\n      }\n\n      throw errorWithCause(`${typeof error} - ${error} (accessibility id: ${testId})`, error)\n    }\n\n    if (elements.length > 1) {\n      throw new Error(`More than 1 elements (${elements.length}) was found by accessibility id: ${testId}`)\n    }\n\n    if (!elements[0]) {\n      const elapsedSeconds = (Date.now() - startTime) / 1000\n      throw new Error(`Element couldn't be found after ${elapsedSeconds.toFixed(2)}s by accessibility id: ${testId}`)\n    }\n\n    return elements[0]\n  }\n\n  /**\n   * @param {string} testId\n   * @param {FindArgs} [args]\n   * @returns {Promise<import(\"selenium-webdriver\").WebElement[]>}\n   */\n  async allByAccessibilityId(testId, args = {}) {\n    const {scrollTo = false, visible = true, timeout, ...restArgs} = args\n    const restArgsKeys = Object.keys(restArgs).filter((key) => key !== \"useBaseSelector\")\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 startTime = Date.now()\n    const getTimeLeft = () => Math.max(actualTimeout - (Date.now() - startTime), 0)\n    const getElements = async () => {\n      const foundElements = await this.getWebDriver().findElements(new By(\"accessibility id\", testId))\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.getWebDriver().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 Error && error.constructor.name === \"TimeoutError\" && getTimeLeft() > 0) {\n          continue\n        }\n\n        throw errorWithCause(`Couldn't get elements with accessibility id: ${testId}: ${error instanceof Error ? error.message : error}`, error)\n      }\n    }\n\n    if (scrollTo) {\n      for (const element of elements) {\n        await this.scrollElementIntoView(element)\n      }\n    }\n\n    return elements\n  }\n\n  /**\n   * @param {string} testId\n   * @param {FindArgs} [args]\n   * @returns {Promise<import(\"selenium-webdriver\").WebElement>}\n   */\n  async findById(testId, args = {}) {\n    const startTime = Date.now()\n    /** @type {import(\"selenium-webdriver\").WebElement[]} */\n    let elements\n\n    try {\n      elements = await this.allById(testId, args)\n    } catch (error) {\n      if (error instanceof Error) {\n        throw errorWithCause(`${error.constructor.name} - ${error.message} (id: ${testId})`, error)\n      }\n\n      throw errorWithCause(`${typeof error} - ${error} (id: ${testId})`, error)\n    }\n\n    if (elements.length > 1) {\n      throw new Error(`More than 1 elements (${elements.length}) was found by id: ${testId}`)\n    }\n\n    if (!elements[0]) {\n      const elapsedSeconds = (Date.now() - startTime) / 1000\n      throw new Error(`Element couldn't be found after ${elapsedSeconds.toFixed(2)}s by id: ${testId}`)\n    }\n\n    return elements[0]\n  }\n\n  /**\n   * @param {string} testId\n   * @param {FindArgs} [args]\n   * @returns {Promise<import(\"selenium-webdriver\").WebElement[]>}\n   */\n  async allById(testId, args = {}) {\n    const {scrollTo = false, visible = true, timeout, ...restArgs} = args\n    const restArgsKeys = Object.keys(restArgs).filter((key) => key !== \"useBaseSelector\")\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 startTime = Date.now()\n    const getTimeLeft = () => Math.max(actualTimeout - (Date.now() - startTime), 0)\n    const getElements = async () => {\n      const foundElements = await this.getWebDriver().findElements(By.id(testId))\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.getWebDriver().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 Error && error.constructor.name === \"TimeoutError\" && getTimeLeft() > 0) {\n          continue\n        }\n\n        throw errorWithCause(`Couldn't get elements with id: ${testId}: ${error instanceof Error ? error.message : error}`, error)\n      }\n    }\n\n    if (scrollTo) {\n      for (const element of elements) {\n        await this.scrollElementIntoView(element)\n      }\n    }\n\n    return elements\n  }\n\n  /**\n   * @param {Record<string, any>} serverArgs\n   * @returns {string}\n   */\n  buildServerUrl(serverArgs) {\n    const address = serverArgs.address ?? \"127.0.0.1\"\n    const port = serverArgs.port ?? 4723\n    const basePath = serverArgs.basePath ? (serverArgs.basePath.startsWith(\"/\") ? serverArgs.basePath : `/${serverArgs.basePath}`) : \"\"\n\n    return `http://${address}:${port}${basePath}`\n  }\n}\n"]}
|
|
@@ -2,6 +2,12 @@ export default class SystemTestBrowserHelper {
|
|
|
2
2
|
/** @returns {SystemTestBrowserHelper | null} */
|
|
3
3
|
static getCurrent(): SystemTestBrowserHelper | null;
|
|
4
4
|
static current(): SystemTestBrowserHelper;
|
|
5
|
+
/**
|
|
6
|
+
* @param {string} parameterName
|
|
7
|
+
* @param {number} fallback
|
|
8
|
+
* @returns {number}
|
|
9
|
+
*/
|
|
10
|
+
getSystemTestPort(parameterName: string, fallback: number): number;
|
|
5
11
|
communicator: SystemTestCommunicator;
|
|
6
12
|
_enabled: boolean;
|
|
7
13
|
_hasInitialized: boolean;
|
|
@@ -9,6 +9,24 @@ const shared = {
|
|
|
9
9
|
systemTestBrowserHelper: null
|
|
10
10
|
};
|
|
11
11
|
export default class SystemTestBrowserHelper {
|
|
12
|
+
/**
|
|
13
|
+
* @param {string} parameterName
|
|
14
|
+
* @param {number} fallback
|
|
15
|
+
* @returns {number}
|
|
16
|
+
*/
|
|
17
|
+
getSystemTestPort(parameterName, fallback) {
|
|
18
|
+
const location = globalThis.location;
|
|
19
|
+
const search = location?.search;
|
|
20
|
+
if (!search) {
|
|
21
|
+
return fallback;
|
|
22
|
+
}
|
|
23
|
+
const value = new URLSearchParams(search).get(parameterName);
|
|
24
|
+
if (!value) {
|
|
25
|
+
return fallback;
|
|
26
|
+
}
|
|
27
|
+
const parsed = Number(value);
|
|
28
|
+
return Number.isInteger(parsed) && parsed > 0 ? parsed : fallback;
|
|
29
|
+
}
|
|
12
30
|
/** @returns {SystemTestBrowserHelper | null} */
|
|
13
31
|
static getCurrent() {
|
|
14
32
|
return shared.systemTestBrowserHelper;
|
|
@@ -83,7 +101,8 @@ export default class SystemTestBrowserHelper {
|
|
|
83
101
|
}
|
|
84
102
|
async startScoundrel() {
|
|
85
103
|
const host = this.getSystemTestHost();
|
|
86
|
-
|
|
104
|
+
const scoundrelPort = this.getSystemTestPort("systemTestScoundrelPort", 8090);
|
|
105
|
+
this.scoundrelWs = new WebSocket(`ws://${host}:${scoundrelPort}`);
|
|
87
106
|
// @ts-expect-error
|
|
88
107
|
this.scoundrelClientWebSocket = new ClientWebSocket(this.scoundrelWs);
|
|
89
108
|
await this.scoundrelClientWebSocket.waitForOpened();
|
|
@@ -164,7 +183,8 @@ export default class SystemTestBrowserHelper {
|
|
|
164
183
|
/** @returns {void} */
|
|
165
184
|
connectWebSocket() {
|
|
166
185
|
const host = this.getSystemTestHost();
|
|
167
|
-
|
|
186
|
+
const clientWsPort = this.getSystemTestPort("systemTestClientWsPort", 1985);
|
|
187
|
+
this.ws = new WebSocket(`ws://${host}:${clientWsPort}`);
|
|
168
188
|
this.communicator.ws = this.ws;
|
|
169
189
|
this.ws.addEventListener("error", digg(this, "communicator", "onError"));
|
|
170
190
|
this.ws.addEventListener("open", digg(this, "communicator", "onOpen"));
|
|
@@ -287,4 +307,4 @@ export default class SystemTestBrowserHelper {
|
|
|
287
307
|
return await this.communicator.sendCommand({ type: "query", sql });
|
|
288
308
|
}
|
|
289
309
|
}
|
|
290
|
-
//# 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  /** @returns {SystemTestBrowserHelper | null} */\n  static getCurrent() {\n    return shared.systemTestBrowserHelper\n  }\n\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    const host = this.getSystemTestHost()\n    this.scoundrelWs = new WebSocket(`ws://${host}: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    if (!window?.addEventListener) return\n\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    if (!window?.addEventListener) return\n\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  /** @returns {void} */\n  connectWebSocket() {\n    const host = this.getSystemTestHost()\n    this.ws = new WebSocket(`ws://${host}: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  /** @returns {string} */\n  getSystemTestHost() {\n    const location = globalThis.location\n    const defaultHost = location?.hostname || \"localhost\"\n    const search = location?.search\n    const envHost = process.env.EXPO_PUBLIC_SYSTEM_TEST_HOST\n\n    if (envHost) return envHost\n\n    if (!search) return defaultHost\n\n    const params = new URLSearchParams(search)\n    const host = params.get(\"systemTestHost\")\n\n    const resolvedHost = host || defaultHost\n\n    if (resolvedHost === \"0.0.0.0\") return \"127.0.0.1\"\n\n    return resolvedHost\n  }\n\n  /** @returns {void} */\n  enableOnBrowser() {\n    if (this._enabled) {\n      return\n    }\n\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   * Emits a command event and throws when no listener handled it.\n   * @param {string} eventName\n   * @param {{path: string}} payload\n   * @returns {void}\n   */\n  emitCommandEvent(eventName, payload) {\n    const didEmit = this.events.emit(eventName, payload)\n\n    if (!didEmit) {\n      throw new Error(`No listener registered for command event: ${eventName} (${payload.path})`)\n    }\n  }\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 == \"waitForScoundrel\") {\n      await this.waitForScoundrelStarted()\n      return {result: \"scoundrel-ready\"}\n    } else if (data.type == \"visit\") {\n      this.emitCommandEvent(\"navigate\", {path: data.path})\n      return {result: \"visited\"}\n    } else if (data.type == \"dismissTo\") {\n      this.emitCommandEvent(\"dismissTo\", {path: data.path})\n      return {result: \"dismissed\"}\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  /** @returns {void} */\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"]}
|
|
310
|
+
//# 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  /**\n   * @param {string} parameterName\n   * @param {number} fallback\n   * @returns {number}\n   */\n  getSystemTestPort(parameterName, fallback) {\n    const location = globalThis.location\n    const search = location?.search\n\n    if (!search) {\n      return fallback\n    }\n\n    const value = new URLSearchParams(search).get(parameterName)\n\n    if (!value) {\n      return fallback\n    }\n\n    const parsed = Number(value)\n\n    return Number.isInteger(parsed) && parsed > 0 ? parsed : fallback\n  }\n\n  /** @returns {SystemTestBrowserHelper | null} */\n  static getCurrent() {\n    return shared.systemTestBrowserHelper\n  }\n\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    const host = this.getSystemTestHost()\n    const scoundrelPort = this.getSystemTestPort(\"systemTestScoundrelPort\", 8090)\n\n    this.scoundrelWs = new WebSocket(`ws://${host}:${scoundrelPort}`)\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    if (!window?.addEventListener) return\n\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    if (!window?.addEventListener) return\n\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  /** @returns {void} */\n  connectWebSocket() {\n    const host = this.getSystemTestHost()\n    const clientWsPort = this.getSystemTestPort(\"systemTestClientWsPort\", 1985)\n\n    this.ws = new WebSocket(`ws://${host}:${clientWsPort}`)\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  /** @returns {string} */\n  getSystemTestHost() {\n    const location = globalThis.location\n    const defaultHost = location?.hostname || \"localhost\"\n    const search = location?.search\n    const envHost = process.env.EXPO_PUBLIC_SYSTEM_TEST_HOST\n\n    if (envHost) return envHost\n\n    if (!search) return defaultHost\n\n    const params = new URLSearchParams(search)\n    const host = params.get(\"systemTestHost\")\n\n    const resolvedHost = host || defaultHost\n\n    if (resolvedHost === \"0.0.0.0\") return \"127.0.0.1\"\n\n    return resolvedHost\n  }\n\n  /** @returns {void} */\n  enableOnBrowser() {\n    if (this._enabled) {\n      return\n    }\n\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   * Emits a command event and throws when no listener handled it.\n   * @param {string} eventName\n   * @param {{path: string}} payload\n   * @returns {void}\n   */\n  emitCommandEvent(eventName, payload) {\n    const didEmit = this.events.emit(eventName, payload)\n\n    if (!didEmit) {\n      throw new Error(`No listener registered for command event: ${eventName} (${payload.path})`)\n    }\n  }\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 == \"waitForScoundrel\") {\n      await this.waitForScoundrelStarted()\n      return {result: \"scoundrel-ready\"}\n    } else if (data.type == \"visit\") {\n      this.emitCommandEvent(\"navigate\", {path: data.path})\n      return {result: \"visited\"}\n    } else if (data.type == \"dismissTo\") {\n      this.emitCommandEvent(\"dismissTo\", {path: data.path})\n      return {result: \"dismissed\"}\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  /** @returns {void} */\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"]}
|
package/build/system-test.d.ts
CHANGED
|
@@ -7,6 +7,8 @@
|
|
|
7
7
|
* @property {string} [httpConnectHost] Hostname used by the driver to reach the HTTP server.
|
|
8
8
|
* @property {boolean} [debug] Enable debug logging.
|
|
9
9
|
* @property {(error: any) => boolean} [errorFilter] Filter for browser errors (return false to ignore).
|
|
10
|
+
* @property {number} [clientWsPort] Port for the browser-command WebSocket server.
|
|
11
|
+
* @property {number} [scoundrelPort] Port for the Scoundrel WebSocket server.
|
|
10
12
|
* @property {Record<string, any>} [urlArgs] Query params appended to the root path.
|
|
11
13
|
* @property {SystemTestDriverConfig} [driver] Driver configuration.
|
|
12
14
|
*/
|
|
@@ -60,12 +62,14 @@ export default class SystemTest extends Browser {
|
|
|
60
62
|
* Creates a new SystemTest instance
|
|
61
63
|
* @param {SystemTestArgs} [args]
|
|
62
64
|
*/
|
|
63
|
-
constructor({ host, port, httpHost, httpPort, httpConnectHost, debug, errorFilter, urlArgs, driver, ...restArgs }?: SystemTestArgs);
|
|
65
|
+
constructor({ clientWsPort, host, port, httpHost, httpPort, httpConnectHost, debug, errorFilter, scoundrelPort, urlArgs, driver, ...restArgs }?: SystemTestArgs);
|
|
64
66
|
_started: boolean;
|
|
67
|
+
_clientWsPort: number;
|
|
65
68
|
_httpHost: string;
|
|
66
69
|
_httpPort: number;
|
|
67
70
|
/** @type {(error: any) => boolean | undefined} */
|
|
68
71
|
_errorFilter: (error: any) => boolean | undefined;
|
|
72
|
+
_scoundrelPort: number;
|
|
69
73
|
/** @type {WebSocketServer | undefined} */
|
|
70
74
|
scoundrelWss: WebSocketServer | undefined;
|
|
71
75
|
/** @type {WebSocketServer | undefined} */
|
|
@@ -253,6 +257,14 @@ export type SystemTestArgs = {
|
|
|
253
257
|
* Filter for browser errors (return false to ignore).
|
|
254
258
|
*/
|
|
255
259
|
errorFilter?: (error: any) => boolean;
|
|
260
|
+
/**
|
|
261
|
+
* Port for the browser-command WebSocket server.
|
|
262
|
+
*/
|
|
263
|
+
clientWsPort?: number;
|
|
264
|
+
/**
|
|
265
|
+
* Port for the Scoundrel WebSocket server.
|
|
266
|
+
*/
|
|
267
|
+
scoundrelPort?: number;
|
|
256
268
|
/**
|
|
257
269
|
* Query params appended to the root path.
|
|
258
270
|
*/
|
package/build/system-test.js
CHANGED
|
@@ -17,6 +17,8 @@ import Browser from "./browser.js";
|
|
|
17
17
|
* @property {string} [httpConnectHost] Hostname used by the driver to reach the HTTP server.
|
|
18
18
|
* @property {boolean} [debug] Enable debug logging.
|
|
19
19
|
* @property {(error: any) => boolean} [errorFilter] Filter for browser errors (return false to ignore).
|
|
20
|
+
* @property {number} [clientWsPort] Port for the browser-command WebSocket server.
|
|
21
|
+
* @property {number} [scoundrelPort] Port for the Scoundrel WebSocket server.
|
|
20
22
|
* @property {Record<string, any>} [urlArgs] Query params appended to the root path.
|
|
21
23
|
* @property {SystemTestDriverConfig} [driver] Driver configuration.
|
|
22
24
|
*/
|
|
@@ -151,15 +153,17 @@ class SystemTest extends Browser {
|
|
|
151
153
|
* Creates a new SystemTest instance
|
|
152
154
|
* @param {SystemTestArgs} [args]
|
|
153
155
|
*/
|
|
154
|
-
constructor({ host = "localhost", port = 8081, httpHost = "localhost", httpPort = 1984, httpConnectHost, debug = false, errorFilter, urlArgs, driver, ...restArgs } = { host: "localhost", port: 8081, httpHost: "localhost", httpPort: 1984, debug: false }) {
|
|
156
|
+
constructor({ clientWsPort = 1985, host = "localhost", port = 8081, httpHost = "localhost", httpPort = 1984, httpConnectHost, debug = false, errorFilter, scoundrelPort = 8090, urlArgs, driver, ...restArgs } = { host: "localhost", port: 8081, httpHost: "localhost", httpPort: 1984, debug: false }) {
|
|
155
157
|
super({ debug, driver });
|
|
156
158
|
/** @type {SystemTestCommunicator | undefined} */
|
|
157
159
|
this.communicator = undefined;
|
|
158
160
|
this._started = false;
|
|
161
|
+
this._clientWsPort = 1985;
|
|
159
162
|
this._httpHost = "localhost";
|
|
160
163
|
this._httpPort = 1984;
|
|
161
164
|
/** @type {(error: any) => boolean | undefined} */
|
|
162
165
|
this._errorFilter = undefined;
|
|
166
|
+
this._scoundrelPort = 8090;
|
|
163
167
|
/** @type {WebSocketServer | undefined} */
|
|
164
168
|
this.scoundrelWss = undefined;
|
|
165
169
|
/** @type {WebSocketServer | undefined} */
|
|
@@ -227,11 +231,13 @@ class SystemTest extends Browser {
|
|
|
227
231
|
}
|
|
228
232
|
this._host = host;
|
|
229
233
|
this._port = port;
|
|
234
|
+
this._clientWsPort = clientWsPort;
|
|
230
235
|
this._httpHost = httpHost;
|
|
231
236
|
this._httpPort = httpPort;
|
|
232
237
|
this._httpConnectHost = httpConnectHost;
|
|
233
238
|
this._debug = debug;
|
|
234
239
|
this._errorFilter = errorFilter;
|
|
240
|
+
this._scoundrelPort = scoundrelPort;
|
|
235
241
|
this._urlArgs = urlArgs;
|
|
236
242
|
this._rootPath = this.buildRootPath();
|
|
237
243
|
/** @type {Record<number, object>} */
|
|
@@ -248,7 +254,7 @@ class SystemTest extends Browser {
|
|
|
248
254
|
startScoundrel() {
|
|
249
255
|
if (this.scoundrelWss)
|
|
250
256
|
throw new Error("Scoundrel server already started");
|
|
251
|
-
this.scoundrelWss = new WebSocketServer({ port:
|
|
257
|
+
this.scoundrelWss = new WebSocketServer({ port: this._scoundrelPort });
|
|
252
258
|
this.serverWebSocket = new ServerWebSocket(this.scoundrelWss);
|
|
253
259
|
this.server = new Server(this.serverWebSocket);
|
|
254
260
|
}
|
|
@@ -573,24 +579,30 @@ class SystemTest extends Browser {
|
|
|
573
579
|
* @returns {string}
|
|
574
580
|
*/
|
|
575
581
|
buildRootPath() {
|
|
576
|
-
if (!this._urlArgs)
|
|
577
|
-
return SystemTest.rootPath;
|
|
578
582
|
const url = new URL(SystemTest.rootPath, "http://localhost");
|
|
579
583
|
const appendParam = (key, value) => {
|
|
580
584
|
if (value === undefined || value === null)
|
|
581
585
|
return;
|
|
582
586
|
url.searchParams.append(key, String(value));
|
|
583
587
|
};
|
|
584
|
-
if (this._urlArgs
|
|
585
|
-
|
|
586
|
-
|
|
588
|
+
if (this._urlArgs) {
|
|
589
|
+
if (this._urlArgs instanceof URLSearchParams) {
|
|
590
|
+
for (const [key, value] of this._urlArgs) {
|
|
591
|
+
appendParam(key, value);
|
|
592
|
+
}
|
|
587
593
|
}
|
|
588
|
-
|
|
589
|
-
|
|
590
|
-
|
|
591
|
-
|
|
594
|
+
else {
|
|
595
|
+
for (const [key, value] of Object.entries(this._urlArgs)) {
|
|
596
|
+
appendParam(key, value);
|
|
597
|
+
}
|
|
592
598
|
}
|
|
593
599
|
}
|
|
600
|
+
if (!url.searchParams.has("systemTestClientWsPort") && this._clientWsPort !== 1985) {
|
|
601
|
+
appendParam("systemTestClientWsPort", this._clientWsPort);
|
|
602
|
+
}
|
|
603
|
+
if (!url.searchParams.has("systemTestScoundrelPort") && this._scoundrelPort !== 8090) {
|
|
604
|
+
appendParam("systemTestScoundrelPort", this._scoundrelPort);
|
|
605
|
+
}
|
|
594
606
|
const rootPath = `${url.pathname}${url.search}${url.hash}`;
|
|
595
607
|
this.debugLog(`buildRootPath rootPath: ${rootPath}`);
|
|
596
608
|
return rootPath;
|
|
@@ -621,7 +633,7 @@ class SystemTest extends Browser {
|
|
|
621
633
|
* @returns {void}
|
|
622
634
|
*/
|
|
623
635
|
startWebSocketServer() {
|
|
624
|
-
this.clientWss = new WebSocketServer({ port:
|
|
636
|
+
this.clientWss = new WebSocketServer({ port: this._clientWsPort });
|
|
625
637
|
this.clientWss.on("connection", this.onWebSocketConnection);
|
|
626
638
|
this.clientWss.on("close", this.onWebSocketClose);
|
|
627
639
|
this.clientWss.on("error", (error) => {
|
|
@@ -740,4 +752,4 @@ class SystemTest extends Browser {
|
|
|
740
752
|
}
|
|
741
753
|
SystemTest.rootPath = "/blank?systemTest=true";
|
|
742
754
|
export default SystemTest;
|
|
743
|
-
//# 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 {digg} from \"diggerize\"\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 {waitFor} from \"awaitery\"\nimport timeout from \"awaitery/build/timeout.js\"\nimport {WebSocketServer} from \"ws\"\nimport Browser from \"./browser.js\"\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 {string} [httpConnectHost] Hostname used by the driver to reach the 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 * @property {SystemTestDriverConfig} [driver] Driver configuration.\n */\n/**\n * @typedef {object} SystemTestDriverConfig\n * @property {\"selenium\"|\"appium\"} [type] Driver implementation to use.\n * @property {Record<string, any>} [options] Driver-specific options.\n */\n/**\n * @typedef {object} FindArgs\n * @property {number} [timeout] Override timeout for lookup.\n * @property {boolean | null} [visible] Whether to require elements to be visible (`true`) or hidden (`false`). Use `null` to disable visibility filtering.\n * @property {boolean} [scrollTo] Whether to scroll found elements into view before returning them.\n * @property {boolean} [useBaseSelector] Whether to scope by the base selector.\n */\n/**\n * @typedef {FindArgs & {withFallback?: boolean}} InteractArgs\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\nexport default class SystemTest extends Browser {\n  static rootPath = \"/blank?systemTest=true\"\n\n  /** @type {SystemTestCommunicator | undefined} */\n  communicator = undefined\n\n  _started = false\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 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: ${error instanceof Error ? error.message : error}`)\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, httpConnectHost, debug = false, errorFilter, urlArgs, driver, ...restArgs} = {host: \"localhost\", port: 8081, httpHost: \"localhost\", httpPort: 1984, debug: false}) {\n    super({debug, driver})\n\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._httpConnectHost = httpConnectHost\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    this.setCommunicator(this.communicator)\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    this.debugLog(\"getScoundrelClient: waiting for browser Scoundrel initialization\")\n    await timeout({timeout: timeoutMs, errorMessage: \"Timed out waiting for Scoundrel to initialize\"}, async () => {\n      await this.getCommunicator().sendCommand({type: \"waitForScoundrel\"})\n    })\n    this.debugLog(\"getScoundrelClient: browser reported Scoundrel initialized\")\n\n    /**\n     * @param {any} client\n     * @returns {boolean}\n     */\n    const isOpenClient = (client) => client?.backend?.ws?.readyState === 1\n\n    const existingClients = this.server.getClients?.()\n    const openExistingClients = existingClients?.filter(isOpenClient)\n\n    if (openExistingClients && openExistingClients.length > 0) {\n      this.debugLog(`getScoundrelClient: using existing open client (${openExistingClients.length} available)`)\n      return openExistingClients[openExistingClients.length - 1]\n    }\n    this.debugLog(`getScoundrelClient: no open cached clients, waiting for new connection (cached total: ${existingClients?.length ?? 0})`)\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          if (!isOpenClient(client)) return\n          cleanupListener()\n          this.debugLog(\"getScoundrelClient: received new open client\")\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    return await this.getDriverAdapter().all(selector, args)\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    await this.getDriverAdapter().click(elementOrIdentifier, args)\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    return await this.getDriverAdapter().find(selector, args)\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) {\n    return await this.getDriverAdapter().findByTestID(testID, args)\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    return await this.getDriverAdapter().findNoWait(selector, args)\n  }\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} & InteractArgs} 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    return await this.getDriverAdapter().interact(elementOrIdentifier, methodName, ...args)\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    await this.getDriverAdapter().waitForNoSelector(selector, args)\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   * Starts the system test\n   * @returns {Promise<void>}\n   */\n  async start() {\n    this.debugLog(\"Start called\")\n    const isNativeHost = process.env.SYSTEM_TEST_HOST === \"native\"\n\n    if (isNativeHost) {\n      this.currentUrl = \"native://\"\n      this.debugLog(\"Using native app host\")\n    } else 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      const connectHost = this._httpConnectHost ?? this._httpHost\n      this.currentUrl = `http://${connectHost}:${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', 'dist', or 'native'\")\n    }\n\n    this.driverAdapter.setBaseUrl(this.currentUrl)\n    this.debugLog(\"Starting driver\")\n    await this.driverAdapter.start()\n    this.debugLog(\"Driver started\")\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    if (!isNativeHost) {\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        this.debugLog(\"Finding root element body > #root\")\n        await this.find(\"body > #root\", {useBaseSelector: false})\n        this.debugLog(\"Found root element body > #root\")\n\n        this.debugLog(\"Finding systemTestingComponent\")\n        await this.findByTestID(\"systemTestingComponent\", {useBaseSelector: false, timeout: 30000, visible: true})\n        this.debugLog(\"Found systemTestingComponent\")\n        this.debugLog(\"Found root and systemTestingComponent\")\n      } catch (error) {\n        this.debugLog(\"Error while finding root/systemTestingComponent, taking screenshot\")\n        await this.takeScreenshot()\n        this.debugLog(\"Screenshot captured after root/systemTestingComponent lookup failure\")\n        throw error\n      }\n    } else {\n      try {\n        this.debugLog(\"Finding systemTestingComponent for native app\")\n        await this.findByTestID(\"systemTestingComponent\", {useBaseSelector: false, timeout: 30000, visible: true})\n        this.debugLog(\"Found systemTestingComponent for native app\")\n      } catch (error) {\n        this.debugLog(\"Error while finding native systemTestingComponent, taking screenshot\")\n        await this.takeScreenshot()\n        this.debugLog(\"Screenshot captured after native systemTestingComponent lookup failure\")\n        throw error\n      }\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.debugLog(\"Marked system test as started\")\n    if (!isNativeHost) {\n      this.debugLog(\"Setting base selector to focused systemTestingComponent\")\n      this.setBaseSelector(\"[data-testid='systemTestingComponent'][data-focussed='true']\")\n      this.debugLog(\"Base selector set\")\n    }\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   * Waits for the client web socket to connect\n   * @returns {Promise<void>}\n   */\n  async waitForClientWebSocket() {\n    try {\n      await timeout({timeout: 30000, errorMessage: \"timeout while waiting for client WebSocket connection\"}, () => 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.driverAdapter) {\n      await this.driverAdapter.stop()\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.driverAdapter = this.createDriver(this._driverConfig)\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    this.setCommunicator(this.communicator)\n\n    this.startScoundrel()\n    await this.start()\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"]}
|
|
755
|
+
//# 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 {digg} from \"diggerize\"\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 {waitFor} from \"awaitery\"\nimport timeout from \"awaitery/build/timeout.js\"\nimport {WebSocketServer} from \"ws\"\nimport Browser from \"./browser.js\"\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 {string} [httpConnectHost] Hostname used by the driver to reach the 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 {number} [clientWsPort] Port for the browser-command WebSocket server.\n * @property {number} [scoundrelPort] Port for the Scoundrel WebSocket server.\n * @property {Record<string, any>} [urlArgs] Query params appended to the root path.\n * @property {SystemTestDriverConfig} [driver] Driver configuration.\n */\n/**\n * @typedef {object} SystemTestDriverConfig\n * @property {\"selenium\"|\"appium\"} [type] Driver implementation to use.\n * @property {Record<string, any>} [options] Driver-specific options.\n */\n/**\n * @typedef {object} FindArgs\n * @property {number} [timeout] Override timeout for lookup.\n * @property {boolean | null} [visible] Whether to require elements to be visible (`true`) or hidden (`false`). Use `null` to disable visibility filtering.\n * @property {boolean} [scrollTo] Whether to scroll found elements into view before returning them.\n * @property {boolean} [useBaseSelector] Whether to scope by the base selector.\n */\n/**\n * @typedef {FindArgs & {withFallback?: boolean}} InteractArgs\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\nexport default class SystemTest extends Browser {\n  static rootPath = \"/blank?systemTest=true\"\n\n  /** @type {SystemTestCommunicator | undefined} */\n  communicator = undefined\n\n  _started = false\n  _clientWsPort = 1985\n  _httpHost = \"localhost\"\n  _httpPort = 1984\n  /** @type {(error: any) => boolean | undefined} */\n  _errorFilter = undefined\n  _scoundrelPort = 8090\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: ${error instanceof Error ? error.message : error}`)\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({clientWsPort = 1985, host = \"localhost\", port = 8081, httpHost = \"localhost\", httpPort = 1984, httpConnectHost, debug = false, errorFilter, scoundrelPort = 8090, urlArgs, driver, ...restArgs} = {host: \"localhost\", port: 8081, httpHost: \"localhost\", httpPort: 1984, debug: false}) {\n    super({debug, driver})\n\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._clientWsPort = clientWsPort\n    this._httpHost = httpHost\n    this._httpPort = httpPort\n    this._httpConnectHost = httpConnectHost\n    this._debug = debug\n    this._errorFilter = errorFilter\n    this._scoundrelPort = scoundrelPort\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    this.setCommunicator(this.communicator)\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: this._scoundrelPort})\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    this.debugLog(\"getScoundrelClient: waiting for browser Scoundrel initialization\")\n    await timeout({timeout: timeoutMs, errorMessage: \"Timed out waiting for Scoundrel to initialize\"}, async () => {\n      await this.getCommunicator().sendCommand({type: \"waitForScoundrel\"})\n    })\n    this.debugLog(\"getScoundrelClient: browser reported Scoundrel initialized\")\n\n    /**\n     * @param {any} client\n     * @returns {boolean}\n     */\n    const isOpenClient = (client) => client?.backend?.ws?.readyState === 1\n\n    const existingClients = this.server.getClients?.()\n    const openExistingClients = existingClients?.filter(isOpenClient)\n\n    if (openExistingClients && openExistingClients.length > 0) {\n      this.debugLog(`getScoundrelClient: using existing open client (${openExistingClients.length} available)`)\n      return openExistingClients[openExistingClients.length - 1]\n    }\n    this.debugLog(`getScoundrelClient: no open cached clients, waiting for new connection (cached total: ${existingClients?.length ?? 0})`)\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          if (!isOpenClient(client)) return\n          cleanupListener()\n          this.debugLog(\"getScoundrelClient: received new open client\")\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    return await this.getDriverAdapter().all(selector, args)\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    await this.getDriverAdapter().click(elementOrIdentifier, args)\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    return await this.getDriverAdapter().find(selector, args)\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) {\n    return await this.getDriverAdapter().findByTestID(testID, args)\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    return await this.getDriverAdapter().findNoWait(selector, args)\n  }\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} & InteractArgs} 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    return await this.getDriverAdapter().interact(elementOrIdentifier, methodName, ...args)\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    await this.getDriverAdapter().waitForNoSelector(selector, args)\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   * Starts the system test\n   * @returns {Promise<void>}\n   */\n  async start() {\n    this.debugLog(\"Start called\")\n    const isNativeHost = process.env.SYSTEM_TEST_HOST === \"native\"\n\n    if (isNativeHost) {\n      this.currentUrl = \"native://\"\n      this.debugLog(\"Using native app host\")\n    } else 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      const connectHost = this._httpConnectHost ?? this._httpHost\n      this.currentUrl = `http://${connectHost}:${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', 'dist', or 'native'\")\n    }\n\n    this.driverAdapter.setBaseUrl(this.currentUrl)\n    this.debugLog(\"Starting driver\")\n    await this.driverAdapter.start()\n    this.debugLog(\"Driver started\")\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    if (!isNativeHost) {\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        this.debugLog(\"Finding root element body > #root\")\n        await this.find(\"body > #root\", {useBaseSelector: false})\n        this.debugLog(\"Found root element body > #root\")\n\n        this.debugLog(\"Finding systemTestingComponent\")\n        await this.findByTestID(\"systemTestingComponent\", {useBaseSelector: false, timeout: 30000, visible: true})\n        this.debugLog(\"Found systemTestingComponent\")\n        this.debugLog(\"Found root and systemTestingComponent\")\n      } catch (error) {\n        this.debugLog(\"Error while finding root/systemTestingComponent, taking screenshot\")\n        await this.takeScreenshot()\n        this.debugLog(\"Screenshot captured after root/systemTestingComponent lookup failure\")\n        throw error\n      }\n    } else {\n      try {\n        this.debugLog(\"Finding systemTestingComponent for native app\")\n        await this.findByTestID(\"systemTestingComponent\", {useBaseSelector: false, timeout: 30000, visible: true})\n        this.debugLog(\"Found systemTestingComponent for native app\")\n      } catch (error) {\n        this.debugLog(\"Error while finding native systemTestingComponent, taking screenshot\")\n        await this.takeScreenshot()\n        this.debugLog(\"Screenshot captured after native systemTestingComponent lookup failure\")\n        throw error\n      }\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.debugLog(\"Marked system test as started\")\n    if (!isNativeHost) {\n      this.debugLog(\"Setting base selector to focused systemTestingComponent\")\n      this.setBaseSelector(\"[data-testid='systemTestingComponent'][data-focussed='true']\")\n      this.debugLog(\"Base selector set\")\n    }\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    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) {\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\n    if (!url.searchParams.has(\"systemTestClientWsPort\") && this._clientWsPort !== 1985) {\n      appendParam(\"systemTestClientWsPort\", this._clientWsPort)\n    }\n\n    if (!url.searchParams.has(\"systemTestScoundrelPort\") && this._scoundrelPort !== 8090) {\n      appendParam(\"systemTestScoundrelPort\", this._scoundrelPort)\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   * Waits for the client web socket to connect\n   * @returns {Promise<void>}\n   */\n  async waitForClientWebSocket() {\n    try {\n      await timeout({timeout: 30000, errorMessage: \"timeout while waiting for client WebSocket connection\"}, () => 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: this._clientWsPort})\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.driverAdapter) {\n      await this.driverAdapter.stop()\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.driverAdapter = this.createDriver(this._driverConfig)\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    this.setCommunicator(this.communicator)\n\n    this.startScoundrel()\n    await this.start()\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"]}
|