system-testing 1.0.77 → 1.0.79

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