system-testing 1.0.78 → 1.0.81
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +232 -6
- package/build/browser-command-client.d.ts +19 -0
- package/build/browser-command-client.js +39 -0
- package/build/browser-command-runner.d.ts +34 -0
- package/build/browser-command-runner.js +155 -0
- package/build/browser-daemon-constants.d.ts +2 -0
- package/build/browser-daemon-constants.js +3 -0
- package/build/browser-process.d.ts +45 -0
- package/build/browser-process.js +134 -0
- package/build/browser-registry.d.ts +44 -0
- package/build/browser-registry.js +191 -0
- package/build/browser.d.ts +240 -0
- package/build/browser.js +375 -0
- package/build/cli-helpers.d.ts +16 -0
- package/build/cli-helpers.js +177 -0
- package/build/cli.d.ts +2 -0
- package/build/cli.js +81 -0
- package/build/drivers/appium-driver.js +21 -21
- package/build/drivers/webdriver-driver.d.ts +4 -4
- package/build/drivers/webdriver-driver.js +32 -28
- package/build/index.d.ts +9 -1
- package/build/index.js +10 -3
- package/build/system-test-browser-helper.d.ts +6 -12
- package/build/system-test-browser-helper.js +12 -13
- package/build/system-test.d.ts +3 -189
- package/build/system-test.js +6 -220
- package/build/use-system-test-expo.d.ts +16 -0
- package/build/use-system-test-expo.js +34 -0
- package/build/use-system-test-react-native.d.ts +25 -0
- package/build/use-system-test-react-native.js +20 -0
- package/build/use-system-test-shape-hook.d.ts +35 -0
- package/build/use-system-test-shape-hook.js +74 -0
- package/build/use-system-test.d.ts +19 -10
- package/build/use-system-test.js +26 -72
- package/package.json +17 -8
package/README.md
CHANGED
|
@@ -8,6 +8,34 @@ Rails inspired system testing for Expo apps.
|
|
|
8
8
|
npm install --save-dev system-testing
|
|
9
9
|
```
|
|
10
10
|
|
|
11
|
+
## Choose the right layer
|
|
12
|
+
|
|
13
|
+
This package has three main entry points:
|
|
14
|
+
|
|
15
|
+
- `SystemTest`: full app-oriented system testing with selector helpers, app bootstrapping, WebSocket communication, screenshots, logs, and Scoundrel support.
|
|
16
|
+
- `Browser`: lower-level driver session for opening URLs, taking screenshots, and reading HTML/logs without the rest of the system-test flow.
|
|
17
|
+
- `system-testing` CLI browser daemon: a long-running named browser process that can be controlled from CLI commands or WebSocket messages.
|
|
18
|
+
- `useSystemTest*` hooks: browser-side integration that lets your app respond to `visit` / `dismissTo` commands from `SystemTest`.
|
|
19
|
+
|
|
20
|
+
Use `SystemTest` if you are testing your app. Use `Browser` if you just want a Selenium/Appium-backed browser session.
|
|
21
|
+
|
|
22
|
+
## Getting started
|
|
23
|
+
|
|
24
|
+
1. Add one of the browser-side hooks to your app:
|
|
25
|
+
`useSystemTestExpo` for Expo Router, or `useSystemTest` / `useSystemTestReactNative` for your own navigation stack.
|
|
26
|
+
2. Wrap your app in a root element with `testID="systemTestingComponent"`.
|
|
27
|
+
3. Make sure your root test route renders an element with `testID="blankText"`, or change `SystemTest.rootPath`.
|
|
28
|
+
4. Start tests with `SystemTest.run(...)` for app flows, or instantiate `Browser` directly for ordinary browsing/capture.
|
|
29
|
+
|
|
30
|
+
Minimal app-side requirements:
|
|
31
|
+
|
|
32
|
+
```jsx
|
|
33
|
+
<View testID="systemTestingComponent" dataSet={{focussed: "true"}}>
|
|
34
|
+
<Text testID="blankText">Blank</Text>
|
|
35
|
+
{children}
|
|
36
|
+
</View>
|
|
37
|
+
```
|
|
38
|
+
|
|
11
39
|
## Usage
|
|
12
40
|
|
|
13
41
|
```js
|
|
@@ -90,9 +118,160 @@ await SystemTest.run({
|
|
|
90
118
|
|
|
91
119
|
If you already run an Appium server, provide `serverUrl` instead of `serverArgs`. By default, `findByTestID` uses the Appium `accessibility id` strategy. To use CSS instead (for web contexts), set `options.testIdStrategy` to `"css"` and optionally `options.testIdAttribute` (defaults to `"data-testid"`).
|
|
92
120
|
|
|
93
|
-
|
|
121
|
+
For local or CI web runs against Chrome, `npm run test:appium:web` now resolves and downloads a matching Chrome for Testing `chromedriver` binary before it starts Appium. That keeps the Appium web path reproducible even when the installed Chrome patch version changes.
|
|
122
|
+
|
|
123
|
+
### Generic browser usage
|
|
124
|
+
|
|
125
|
+
`Browser` is the lower-level browser/session class behind `SystemTest`. Use it when you want driver-backed browsing, screenshots, logs, and HTML capture without the rest of the system-test bootstrapping.
|
|
126
|
+
|
|
127
|
+
```js
|
|
128
|
+
import {Browser} from "system-testing/build/index.js"
|
|
129
|
+
|
|
130
|
+
const browser = new Browser()
|
|
131
|
+
|
|
132
|
+
browser.getDriverAdapter().setBaseUrl("https://example.com")
|
|
133
|
+
await browser.getDriverAdapter().start()
|
|
134
|
+
await browser.setTimeouts(10000)
|
|
135
|
+
|
|
136
|
+
await browser.visit("/")
|
|
137
|
+
|
|
138
|
+
const html = await browser.getHTML()
|
|
139
|
+
const logs = await browser.getBrowserLogs()
|
|
140
|
+
const screenshot = await browser.takeScreenshot()
|
|
141
|
+
|
|
142
|
+
await browser.stopDriver()
|
|
143
|
+
```
|
|
94
144
|
|
|
95
|
-
`
|
|
145
|
+
If `visit()`/`dismissTo()` should drive in-app navigation through the browser-side helper instead of direct URL loads, inject a communicator when constructing `Browser`. Without one, it falls back to direct driver navigation, which makes it usable for ordinary website browsing as well.
|
|
146
|
+
|
|
147
|
+
Common `Browser` flow:
|
|
148
|
+
|
|
149
|
+
1. Create the browser with the desired driver config.
|
|
150
|
+
2. Set the base URL on the driver adapter.
|
|
151
|
+
3. Start the driver and set timeouts.
|
|
152
|
+
4. Call `visit()`.
|
|
153
|
+
5. Read `getHTML()`, `getBrowserLogs()`, `getCurrentUrl()`, or `takeScreenshot()`.
|
|
154
|
+
6. Call `stopDriver()` during teardown.
|
|
155
|
+
|
|
156
|
+
Useful browser methods:
|
|
157
|
+
|
|
158
|
+
- `visit(pathOrUrl)`: uses the helper communicator if present, otherwise loads directly through Selenium/Appium.
|
|
159
|
+
- `dismissTo(pathOrUrl)`: same fallback behavior as `visit()`.
|
|
160
|
+
- `getHTML()`: returns the current page source.
|
|
161
|
+
- `getBrowserLogs()`: returns collected browser logs, or Appium logcat output for Android native runs.
|
|
162
|
+
- `takeScreenshot()`: writes screenshot, HTML, and logs to disk and returns the artifact paths.
|
|
163
|
+
|
|
164
|
+
If you want app-level navigation instead of direct URL loads, keep `Browser` for the driver/session side and use one of the `useSystemTest*` hooks in the app so the communicator has something to talk to.
|
|
165
|
+
|
|
166
|
+
`react` and `expo-router` are optional peer dependencies. Install them only in apps that import the React/Expo hook helpers; CLI/browser-daemon consumers should not need React just to use `system-testing`.
|
|
167
|
+
|
|
168
|
+
### Browser daemon CLI
|
|
169
|
+
|
|
170
|
+
If you want an external agent to drive a reusable browser process, start the browser daemon:
|
|
171
|
+
|
|
172
|
+
```bash
|
|
173
|
+
npx system-testing browser my-browser
|
|
174
|
+
```
|
|
175
|
+
|
|
176
|
+
Optional arguments:
|
|
177
|
+
|
|
178
|
+
- `--port 1991`: use a fixed WebSocket port instead of an ephemeral one
|
|
179
|
+
- `--base-url https://example.com`: set the browser base URL so relative `visit` paths work
|
|
180
|
+
- `--driver selenium|appium`: choose the driver type
|
|
181
|
+
- `--debug`: enable browser debug logging
|
|
182
|
+
|
|
183
|
+
The process stays running until you stop it. On start it prints JSON with at least the browser `name`, `pid`, and `port`.
|
|
184
|
+
|
|
185
|
+
List running browser daemons:
|
|
186
|
+
|
|
187
|
+
```bash
|
|
188
|
+
npx system-testing browser-list
|
|
189
|
+
```
|
|
190
|
+
|
|
191
|
+
This prints one line per browser with the name and port. Use `--json` if you want machine-readable output.
|
|
192
|
+
|
|
193
|
+
Stop a running browser daemon:
|
|
194
|
+
|
|
195
|
+
```bash
|
|
196
|
+
npx system-testing browser-stop --name my-browser
|
|
197
|
+
```
|
|
198
|
+
|
|
199
|
+
If only one browser daemon is running, `browser-stop` can omit `--name`.
|
|
200
|
+
|
|
201
|
+
Send commands from the CLI:
|
|
202
|
+
|
|
203
|
+
```bash
|
|
204
|
+
npx system-testing browser-command --name my-browser --visit=https://example.com/path
|
|
205
|
+
npx system-testing browser-command --name my-browser --find-by-test-id saveButton
|
|
206
|
+
npx system-testing browser-command --name my-browser --find-by-test-id saveButton --timeout 15
|
|
207
|
+
npx system-testing browser-command --name my-browser --click='[data-testid="saveButton"]'
|
|
208
|
+
npx system-testing browser-command --name my-browser --get-html
|
|
209
|
+
npx system-testing browser-command --name my-browser --get-browser-logs
|
|
210
|
+
npx system-testing browser-command --name my-browser --take-screenshot
|
|
211
|
+
```
|
|
212
|
+
|
|
213
|
+
If only one browser daemon is running, `browser-command` can omit `--name`. Results are printed as JSON so automation tools can parse them easily.
|
|
214
|
+
|
|
215
|
+
CLI `--timeout` values are supported on navigation and selector-based commands. Bare numbers are interpreted as seconds, and explicit `ms` / `s` suffixes are also accepted.
|
|
216
|
+
|
|
217
|
+
Generic commands are also supported:
|
|
218
|
+
|
|
219
|
+
```bash
|
|
220
|
+
npx system-testing browser-command \
|
|
221
|
+
--name my-browser \
|
|
222
|
+
--command=interact \
|
|
223
|
+
--selector='[data-testid="emailInput"]' \
|
|
224
|
+
--method=sendKeys \
|
|
225
|
+
--arg='user@example.com'
|
|
226
|
+
```
|
|
227
|
+
|
|
228
|
+
The browser daemon is intended for agent-style development workflows where an AI or script needs to open the app, inspect HTML, locate elements, click controls, and read logs while validating layout or behavior changes.
|
|
229
|
+
|
|
230
|
+
### Browser daemon WebSocket protocol
|
|
231
|
+
|
|
232
|
+
The daemon also accepts WebSocket commands on its configured port. Send JSON payloads like:
|
|
233
|
+
|
|
234
|
+
```json
|
|
235
|
+
{"type":"browser-command","command":"visit","url":"https://example.com/path"}
|
|
236
|
+
```
|
|
237
|
+
|
|
238
|
+
Another example:
|
|
239
|
+
|
|
240
|
+
```json
|
|
241
|
+
{"type":"browser-command","command":"findByTestID","args":{"testID":"saveButton"}}
|
|
242
|
+
```
|
|
243
|
+
|
|
244
|
+
The server responds with JSON:
|
|
245
|
+
|
|
246
|
+
```json
|
|
247
|
+
{"ok":true,"requestId":"...","type":"browser-command-result","result":{"ok":true}}
|
|
248
|
+
```
|
|
249
|
+
|
|
250
|
+
If the command fails:
|
|
251
|
+
|
|
252
|
+
```json
|
|
253
|
+
{"ok":false,"requestId":"...","type":"browser-command-result","error":"..."}
|
|
254
|
+
```
|
|
255
|
+
|
|
256
|
+
Supported daemon commands currently include:
|
|
257
|
+
|
|
258
|
+
- `visit`
|
|
259
|
+
- `dismissTo`
|
|
260
|
+
- `setBaseSelector`
|
|
261
|
+
- `getCurrentUrl`
|
|
262
|
+
- `getHTML`
|
|
263
|
+
- `getBrowserLogs`
|
|
264
|
+
- `takeScreenshot`
|
|
265
|
+
- `find`
|
|
266
|
+
- `findByTestID`
|
|
267
|
+
- `click`
|
|
268
|
+
- `waitForNoSelector`
|
|
269
|
+
- `expectNoElement`
|
|
270
|
+
- `interact`
|
|
271
|
+
|
|
272
|
+
### Using `useSystemTestExpo` in your Expo app
|
|
273
|
+
|
|
274
|
+
`useSystemTestExpo` wires your Expo app to the system-testing runner: it listens for WebSocket commands, initializes the browser helper, and lets tests navigate or reset state. Add it near the root layout of your Expo Router app (for example in `_layout.tsx` or a top-level provider component).
|
|
96
275
|
|
|
97
276
|
To enable system tests in native builds, set `EXPO_PUBLIC_SYSTEM_TEST=true` at build time (and optionally `EXPO_PUBLIC_SYSTEM_TEST_HOST` to reach the test runner from a device/emulator). For native Appium runs, set `SYSTEM_TEST_HOST=native` in the test environment and point Appium at your APK.
|
|
98
277
|
|
|
@@ -100,10 +279,12 @@ Minimal example:
|
|
|
100
279
|
|
|
101
280
|
```jsx
|
|
102
281
|
import {Stack} from "expo-router"
|
|
103
|
-
import
|
|
282
|
+
import useSystemTestExpo from "system-testing/build/use-system-test-expo.js"
|
|
104
283
|
|
|
105
284
|
export default function RootLayout() {
|
|
106
|
-
const {enabled, systemTestBrowserHelper} =
|
|
285
|
+
const {enabled, systemTestBrowserHelper} = useSystemTestExpo({
|
|
286
|
+
// Optional: inject your own helper instance instead of using the shared default
|
|
287
|
+
// browserHelper: mySystemTestBrowserHelper,
|
|
107
288
|
onFirstInitialize: () => {
|
|
108
289
|
// One-time setup the first time the helper initializes
|
|
109
290
|
},
|
|
@@ -129,12 +310,51 @@ export default function RootLayout() {
|
|
|
129
310
|
|
|
130
311
|
Notes:
|
|
131
312
|
- The hook auto-connects when the page is opened with `?systemTest=true` (as the runner does).
|
|
313
|
+
- Pass `browserHelper` if you want to inject a prebuilt `SystemTestBrowserHelper`; otherwise the hook creates and enables a shared default instance.
|
|
132
314
|
- `onFirstInitialize` runs only on the first `initialize` command; use it for one-time setup.
|
|
133
315
|
- `onInitialize` is registered once when the helper is ready, but it runs on every `initialize` command (each `SystemTest.run`); use it to reset globals/session.
|
|
134
316
|
- If you need scoundrel remote evaluation, wait for `systemTestBrowserHelper` and register your classes there, as shown in the commented snippet above.
|
|
135
317
|
- Add a root wrapper with `testID="systemTestingComponent"` (and optionally `data-focussed="true"`) around your app so the runner has a stable element to detect and scope selectors against.
|
|
136
318
|
- From your tests, use `await systemTest.getScoundrelClient()` to obtain the browser Scoundrel client for remote evaluation.
|
|
137
|
-
- `
|
|
319
|
+
- `useSystemTestExpo` calls `useRouter()` from `expo-router`.
|
|
320
|
+
|
|
321
|
+
### Using `useSystemTest` or `useSystemTestReactNative` without Expo Router
|
|
322
|
+
|
|
323
|
+
`useSystemTest` is the generic runtime-agnostic hook. Provide `onNavigate` and `onDismissTo` callbacks for your own navigation stack. `useSystemTestReactNative` is a convenience wrapper around the same generic API for non-Expo React Native apps.
|
|
324
|
+
|
|
325
|
+
Use these when:
|
|
326
|
+
|
|
327
|
+
- you are not using Expo Router
|
|
328
|
+
- you want to inject your own navigation behavior
|
|
329
|
+
- you want to share the same app-side helper integration across different routing setups
|
|
330
|
+
|
|
331
|
+
```js
|
|
332
|
+
import useSystemTestReactNative from "system-testing/build/use-system-test-react-native.js"
|
|
333
|
+
|
|
334
|
+
export default function App({navigation}) {
|
|
335
|
+
useSystemTestReactNative({
|
|
336
|
+
onDismissTo: ({path}) => {
|
|
337
|
+
navigation.reset({
|
|
338
|
+
index: 0,
|
|
339
|
+
routes: [{name: path}]
|
|
340
|
+
})
|
|
341
|
+
},
|
|
342
|
+
onNavigate: ({path}) => {
|
|
343
|
+
navigation.navigate(path)
|
|
344
|
+
}
|
|
345
|
+
})
|
|
346
|
+
|
|
347
|
+
return <Navigator />
|
|
348
|
+
}
|
|
349
|
+
```
|
|
350
|
+
|
|
351
|
+
The generic hook options are:
|
|
352
|
+
|
|
353
|
+
- `browserHelper`: inject an existing `SystemTestBrowserHelper` instance instead of using the shared default
|
|
354
|
+
- `onFirstInitialize`: one-time setup callback
|
|
355
|
+
- `onInitialize`: callback that runs on every `initialize` command
|
|
356
|
+
- `onNavigate`: handler for `visit(...)`
|
|
357
|
+
- `onDismissTo`: handler for `dismissTo(...)`
|
|
138
358
|
|
|
139
359
|
### Root path and `blankText`
|
|
140
360
|
|
|
@@ -208,4 +428,10 @@ This tears down the browser, servers, and sockets, then starts them again so sub
|
|
|
208
428
|
|
|
209
429
|
## Dummy Expo app
|
|
210
430
|
|
|
211
|
-
A ready-to-run Expo Router dummy app that uses `system-testing` lives in `spec/dummy`.
|
|
431
|
+
A ready-to-run Expo Router dummy app that uses `system-testing` lives in `spec/dummy`.
|
|
432
|
+
|
|
433
|
+
Useful commands from the package root:
|
|
434
|
+
|
|
435
|
+
- `npm run export:web`: build the dummy Expo app for web
|
|
436
|
+
- `SYSTEM_TEST_HOST=dist npx jasmine spec/system-test.spec.js`: run the sample system specs against the exported bundle
|
|
437
|
+
- `SYSTEM_TEST_HOST=dist npx jasmine spec/system-test-logging.spec.js`: run the browser-log capture spec
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
/** Sends browser commands to a running browser daemon. */
|
|
2
|
+
export default class BrowserCommandClient {
|
|
3
|
+
/**
|
|
4
|
+
* @param {object} args
|
|
5
|
+
* @param {string} [args.name]
|
|
6
|
+
* @param {number} [args.port]
|
|
7
|
+
*/
|
|
8
|
+
constructor({ name, port }?: {
|
|
9
|
+
name?: string;
|
|
10
|
+
port?: number;
|
|
11
|
+
});
|
|
12
|
+
name: string;
|
|
13
|
+
port: number;
|
|
14
|
+
/**
|
|
15
|
+
* @param {Record<string, any>} payload
|
|
16
|
+
* @returns {Promise<any>}
|
|
17
|
+
*/
|
|
18
|
+
send(payload: Record<string, any>): Promise<any>;
|
|
19
|
+
}
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
import BrowserRegistry from "./browser-registry.js";
|
|
2
|
+
import WebSocket from "ws";
|
|
3
|
+
/** Sends browser commands to a running browser daemon. */
|
|
4
|
+
export default class BrowserCommandClient {
|
|
5
|
+
/**
|
|
6
|
+
* @param {object} args
|
|
7
|
+
* @param {string} [args.name]
|
|
8
|
+
* @param {number} [args.port]
|
|
9
|
+
*/
|
|
10
|
+
constructor({ name, port } = {}) {
|
|
11
|
+
this.name = name;
|
|
12
|
+
this.port = port;
|
|
13
|
+
}
|
|
14
|
+
/**
|
|
15
|
+
* @param {Record<string, any>} payload
|
|
16
|
+
* @returns {Promise<any>}
|
|
17
|
+
*/
|
|
18
|
+
async send(payload) {
|
|
19
|
+
const resolvedPort = this.port ?? (await BrowserRegistry.resolve(this.name)).port;
|
|
20
|
+
const ws = new WebSocket(`ws://127.0.0.1:${resolvedPort}`);
|
|
21
|
+
return await new Promise((resolve, reject) => {
|
|
22
|
+
ws.on("open", () => {
|
|
23
|
+
ws.send(JSON.stringify(payload));
|
|
24
|
+
});
|
|
25
|
+
ws.on("message", (rawData) => {
|
|
26
|
+
const response = JSON.parse(rawData.toString());
|
|
27
|
+
ws.close();
|
|
28
|
+
if (response.ok) {
|
|
29
|
+
resolve(response.result);
|
|
30
|
+
}
|
|
31
|
+
else {
|
|
32
|
+
reject(new Error(response.error));
|
|
33
|
+
}
|
|
34
|
+
});
|
|
35
|
+
ws.on("error", reject);
|
|
36
|
+
});
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYnJvd3Nlci1jb21tYW5kLWNsaWVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uL3NyYy9icm93c2VyLWNvbW1hbmQtY2xpZW50LmpzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sZUFBZSxNQUFNLHVCQUF1QixDQUFBO0FBQ25ELE9BQU8sU0FBUyxNQUFNLElBQUksQ0FBQTtBQUUxQiwwREFBMEQ7QUFDMUQsTUFBTSxDQUFDLE9BQU8sT0FBTyxvQkFBb0I7SUFDdkM7Ozs7T0FJRztJQUNILFlBQVksRUFBQyxJQUFJLEVBQUUsSUFBSSxFQUFDLEdBQUcsRUFBRTtRQUMzQixJQUFJLENBQUMsSUFBSSxHQUFHLElBQUksQ0FBQTtRQUNoQixJQUFJLENBQUMsSUFBSSxHQUFHLElBQUksQ0FBQTtJQUNsQixDQUFDO0lBRUQ7OztPQUdHO0lBQ0gsS0FBSyxDQUFDLElBQUksQ0FBQyxPQUFPO1FBQ2hCLE1BQU0sWUFBWSxHQUFHLElBQUksQ0FBQyxJQUFJLElBQUksQ0FBQyxNQUFNLGVBQWUsQ0FBQyxPQUFPLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFBO1FBQ2pGLE1BQU0sRUFBRSxHQUFHLElBQUksU0FBUyxDQUFDLGtCQUFrQixZQUFZLEVBQUUsQ0FBQyxDQUFBO1FBRTFELE9BQU8sTUFBTSxJQUFJLE9BQU8sQ0FBQyxDQUFDLE9BQU8sRUFBRSxNQUFNLEVBQUUsRUFBRTtZQUMzQyxFQUFFLENBQUMsRUFBRSxDQUFDLE1BQU0sRUFBRSxHQUFHLEVBQUU7Z0JBQ2pCLEVBQUUsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFBO1lBQ2xDLENBQUMsQ0FBQyxDQUFBO1lBRUYsRUFBRSxDQUFDLEVBQUUsQ0FBQyxTQUFTLEVBQUUsQ0FBQyxPQUFPLEVBQUUsRUFBRTtnQkFDM0IsTUFBTSxRQUFRLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyxPQUFPLENBQUMsUUFBUSxFQUFFLENBQUMsQ0FBQTtnQkFFL0MsRUFBRSxDQUFDLEtBQUssRUFBRSxDQUFBO2dCQUVWLElBQUksUUFBUSxDQUFDLEVBQUUsRUFBRSxDQUFDO29CQUNoQixPQUFPLENBQUMsUUFBUSxDQUFDLE1BQU0sQ0FBQyxDQUFBO2dCQUMxQixDQUFDO3FCQUFNLENBQUM7b0JBQ04sTUFBTSxDQUFDLElBQUksS0FBSyxDQUFDLFFBQVEsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFBO2dCQUNuQyxDQUFDO1lBQ0gsQ0FBQyxDQUFDLENBQUE7WUFFRixFQUFFLENBQUMsRUFBRSxDQUFDLE9BQU8sRUFBRSxNQUFNLENBQUMsQ0FBQTtRQUN4QixDQUFDLENBQUMsQ0FBQTtJQUNKLENBQUM7Q0FDRiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBCcm93c2VyUmVnaXN0cnkgZnJvbSBcIi4vYnJvd3Nlci1yZWdpc3RyeS5qc1wiXG5pbXBvcnQgV2ViU29ja2V0IGZyb20gXCJ3c1wiXG5cbi8qKiBTZW5kcyBicm93c2VyIGNvbW1hbmRzIHRvIGEgcnVubmluZyBicm93c2VyIGRhZW1vbi4gKi9cbmV4cG9ydCBkZWZhdWx0IGNsYXNzIEJyb3dzZXJDb21tYW5kQ2xpZW50IHtcbiAgLyoqXG4gICAqIEBwYXJhbSB7b2JqZWN0fSBhcmdzXG4gICAqIEBwYXJhbSB7c3RyaW5nfSBbYXJncy5uYW1lXVxuICAgKiBAcGFyYW0ge251bWJlcn0gW2FyZ3MucG9ydF1cbiAgICovXG4gIGNvbnN0cnVjdG9yKHtuYW1lLCBwb3J0fSA9IHt9KSB7XG4gICAgdGhpcy5uYW1lID0gbmFtZVxuICAgIHRoaXMucG9ydCA9IHBvcnRcbiAgfVxuXG4gIC8qKlxuICAgKiBAcGFyYW0ge1JlY29yZDxzdHJpbmcsIGFueT59IHBheWxvYWRcbiAgICogQHJldHVybnMge1Byb21pc2U8YW55Pn1cbiAgICovXG4gIGFzeW5jIHNlbmQocGF5bG9hZCkge1xuICAgIGNvbnN0IHJlc29sdmVkUG9ydCA9IHRoaXMucG9ydCA/PyAoYXdhaXQgQnJvd3NlclJlZ2lzdHJ5LnJlc29sdmUodGhpcy5uYW1lKSkucG9ydFxuICAgIGNvbnN0IHdzID0gbmV3IFdlYlNvY2tldChgd3M6Ly8xMjcuMC4wLjE6JHtyZXNvbHZlZFBvcnR9YClcblxuICAgIHJldHVybiBhd2FpdCBuZXcgUHJvbWlzZSgocmVzb2x2ZSwgcmVqZWN0KSA9PiB7XG4gICAgICB3cy5vbihcIm9wZW5cIiwgKCkgPT4ge1xuICAgICAgICB3cy5zZW5kKEpTT04uc3RyaW5naWZ5KHBheWxvYWQpKVxuICAgICAgfSlcblxuICAgICAgd3Mub24oXCJtZXNzYWdlXCIsIChyYXdEYXRhKSA9PiB7XG4gICAgICAgIGNvbnN0IHJlc3BvbnNlID0gSlNPTi5wYXJzZShyYXdEYXRhLnRvU3RyaW5nKCkpXG5cbiAgICAgICAgd3MuY2xvc2UoKVxuXG4gICAgICAgIGlmIChyZXNwb25zZS5vaykge1xuICAgICAgICAgIHJlc29sdmUocmVzcG9uc2UucmVzdWx0KVxuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIHJlamVjdChuZXcgRXJyb3IocmVzcG9uc2UuZXJyb3IpKVxuICAgICAgICB9XG4gICAgICB9KVxuXG4gICAgICB3cy5vbihcImVycm9yXCIsIHJlamVjdClcbiAgICB9KVxuICB9XG59XG4iXX0=
|
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
/** Runs browser commands across CLI and WebSocket transports. */
|
|
2
|
+
export default class BrowserCommandRunner {
|
|
3
|
+
/**
|
|
4
|
+
* @param {object} args
|
|
5
|
+
* @param {import("./browser.js").default} args.browser
|
|
6
|
+
*/
|
|
7
|
+
constructor({ browser }: {
|
|
8
|
+
browser: import("./browser.js").default;
|
|
9
|
+
});
|
|
10
|
+
browser: import("./browser.js").default;
|
|
11
|
+
/**
|
|
12
|
+
* @param {Record<string, any>} commandArgs
|
|
13
|
+
* @returns {{timeout?: number}}
|
|
14
|
+
*/
|
|
15
|
+
normalizeTimeoutArgs(commandArgs: Record<string, any>): {
|
|
16
|
+
timeout?: number;
|
|
17
|
+
};
|
|
18
|
+
/**
|
|
19
|
+
* @param {Record<string, any>} commandArgs
|
|
20
|
+
* @returns {import("./system-test.js").FindArgs}
|
|
21
|
+
*/
|
|
22
|
+
normalizeFindArgs(commandArgs: Record<string, any>): import("./system-test.js").FindArgs;
|
|
23
|
+
/**
|
|
24
|
+
* @param {import("selenium-webdriver").WebElement} element
|
|
25
|
+
* @returns {Promise<Record<string, any>>}
|
|
26
|
+
*/
|
|
27
|
+
serializeElement(element: import("selenium-webdriver").WebElement): Promise<Record<string, any>>;
|
|
28
|
+
/**
|
|
29
|
+
* @param {string} command
|
|
30
|
+
* @param {Record<string, any>} commandArgs
|
|
31
|
+
* @returns {Promise<any>}
|
|
32
|
+
*/
|
|
33
|
+
run(command: string, commandArgs?: Record<string, any>): Promise<any>;
|
|
34
|
+
}
|
|
@@ -0,0 +1,155 @@
|
|
|
1
|
+
/** Runs browser commands across CLI and WebSocket transports. */
|
|
2
|
+
export default class BrowserCommandRunner {
|
|
3
|
+
/**
|
|
4
|
+
* @param {object} args
|
|
5
|
+
* @param {import("./browser.js").default} args.browser
|
|
6
|
+
*/
|
|
7
|
+
constructor({ browser }) {
|
|
8
|
+
this.browser = browser;
|
|
9
|
+
}
|
|
10
|
+
/**
|
|
11
|
+
* @param {Record<string, any>} commandArgs
|
|
12
|
+
* @returns {{timeout?: number}}
|
|
13
|
+
*/
|
|
14
|
+
normalizeTimeoutArgs(commandArgs) {
|
|
15
|
+
const normalizedArgs = {};
|
|
16
|
+
if ("timeout" in commandArgs && commandArgs.timeout !== undefined) {
|
|
17
|
+
normalizedArgs.timeout = Number(commandArgs.timeout);
|
|
18
|
+
if (Number.isNaN(normalizedArgs.timeout)) {
|
|
19
|
+
throw new Error(`Invalid timeout: ${commandArgs.timeout}`);
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
return normalizedArgs;
|
|
23
|
+
}
|
|
24
|
+
/**
|
|
25
|
+
* @param {Record<string, any>} commandArgs
|
|
26
|
+
* @returns {import("./system-test.js").FindArgs}
|
|
27
|
+
*/
|
|
28
|
+
normalizeFindArgs(commandArgs) {
|
|
29
|
+
const findArgs = /** @type {import("./system-test.js").FindArgs} */ (this.normalizeTimeoutArgs(commandArgs));
|
|
30
|
+
if ("visible" in commandArgs && commandArgs.visible !== undefined) {
|
|
31
|
+
if (commandArgs.visible === null || commandArgs.visible === "null") {
|
|
32
|
+
findArgs.visible = null;
|
|
33
|
+
}
|
|
34
|
+
else if (typeof commandArgs.visible === "boolean") {
|
|
35
|
+
findArgs.visible = commandArgs.visible;
|
|
36
|
+
}
|
|
37
|
+
else {
|
|
38
|
+
findArgs.visible = commandArgs.visible === "true";
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
if ("useBaseSelector" in commandArgs && commandArgs.useBaseSelector !== undefined) {
|
|
42
|
+
if (typeof commandArgs.useBaseSelector === "boolean") {
|
|
43
|
+
findArgs.useBaseSelector = commandArgs.useBaseSelector;
|
|
44
|
+
}
|
|
45
|
+
else {
|
|
46
|
+
findArgs.useBaseSelector = commandArgs.useBaseSelector === "true";
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
return findArgs;
|
|
50
|
+
}
|
|
51
|
+
/**
|
|
52
|
+
* @param {import("selenium-webdriver").WebElement} element
|
|
53
|
+
* @returns {Promise<Record<string, any>>}
|
|
54
|
+
*/
|
|
55
|
+
async serializeElement(element) {
|
|
56
|
+
const text = await element.getText();
|
|
57
|
+
const tagName = await element.getTagName();
|
|
58
|
+
const displayed = await element.isDisplayed();
|
|
59
|
+
return { displayed, tagName, text };
|
|
60
|
+
}
|
|
61
|
+
/**
|
|
62
|
+
* @param {string} command
|
|
63
|
+
* @param {Record<string, any>} commandArgs
|
|
64
|
+
* @returns {Promise<any>}
|
|
65
|
+
*/
|
|
66
|
+
async run(command, commandArgs = {}) {
|
|
67
|
+
if (command === "visit") {
|
|
68
|
+
const path = commandArgs.path ?? commandArgs.url;
|
|
69
|
+
if (!path) {
|
|
70
|
+
throw new Error("visit requires path or url");
|
|
71
|
+
}
|
|
72
|
+
await this.browser.visit(path, this.normalizeTimeoutArgs(commandArgs));
|
|
73
|
+
return { ok: true };
|
|
74
|
+
}
|
|
75
|
+
if (command === "dismissTo") {
|
|
76
|
+
const path = commandArgs.path ?? commandArgs.url;
|
|
77
|
+
if (!path) {
|
|
78
|
+
throw new Error("dismissTo requires path or url");
|
|
79
|
+
}
|
|
80
|
+
await this.browser.dismissTo(path, this.normalizeTimeoutArgs(commandArgs));
|
|
81
|
+
return { ok: true };
|
|
82
|
+
}
|
|
83
|
+
if (command === "setBaseSelector") {
|
|
84
|
+
if (!commandArgs.selector) {
|
|
85
|
+
throw new Error("setBaseSelector requires selector");
|
|
86
|
+
}
|
|
87
|
+
this.browser.setBaseSelector(commandArgs.selector);
|
|
88
|
+
return { ok: true };
|
|
89
|
+
}
|
|
90
|
+
if (command === "getCurrentUrl") {
|
|
91
|
+
return { currentUrl: await this.browser.getCurrentUrl() };
|
|
92
|
+
}
|
|
93
|
+
if (command === "getHTML") {
|
|
94
|
+
return { html: await this.browser.getHTML() };
|
|
95
|
+
}
|
|
96
|
+
if (command === "getBrowserLogs") {
|
|
97
|
+
return { logs: await this.browser.getBrowserLogs() };
|
|
98
|
+
}
|
|
99
|
+
if (command === "takeScreenshot") {
|
|
100
|
+
return await this.browser.takeScreenshot();
|
|
101
|
+
}
|
|
102
|
+
if (command === "find") {
|
|
103
|
+
if (!commandArgs.selector) {
|
|
104
|
+
throw new Error("find requires selector");
|
|
105
|
+
}
|
|
106
|
+
const element = await this.browser.find(commandArgs.selector, this.normalizeFindArgs(commandArgs));
|
|
107
|
+
return { element: await this.serializeElement(element) };
|
|
108
|
+
}
|
|
109
|
+
if (command === "findByTestID") {
|
|
110
|
+
const testID = commandArgs.testID ?? commandArgs.testId;
|
|
111
|
+
if (!testID) {
|
|
112
|
+
throw new Error("findByTestID requires testID");
|
|
113
|
+
}
|
|
114
|
+
const element = await this.browser.findByTestID(testID, this.normalizeFindArgs(commandArgs));
|
|
115
|
+
return { element: await this.serializeElement(element) };
|
|
116
|
+
}
|
|
117
|
+
if (command === "click") {
|
|
118
|
+
const selector = commandArgs.selector;
|
|
119
|
+
if (!selector) {
|
|
120
|
+
throw new Error("click requires selector");
|
|
121
|
+
}
|
|
122
|
+
await this.browser.click(selector, this.normalizeFindArgs(commandArgs));
|
|
123
|
+
return { ok: true };
|
|
124
|
+
}
|
|
125
|
+
if (command === "waitForNoSelector") {
|
|
126
|
+
if (!commandArgs.selector) {
|
|
127
|
+
throw new Error("waitForNoSelector requires selector");
|
|
128
|
+
}
|
|
129
|
+
await this.browser.waitForNoSelector(commandArgs.selector, this.normalizeFindArgs(commandArgs));
|
|
130
|
+
return { ok: true };
|
|
131
|
+
}
|
|
132
|
+
if (command === "expectNoElement") {
|
|
133
|
+
if (!commandArgs.selector) {
|
|
134
|
+
throw new Error("expectNoElement requires selector");
|
|
135
|
+
}
|
|
136
|
+
await this.browser.expectNoElement(commandArgs.selector, this.normalizeFindArgs(commandArgs));
|
|
137
|
+
return { ok: true };
|
|
138
|
+
}
|
|
139
|
+
if (command === "interact") {
|
|
140
|
+
const selector = commandArgs.selector;
|
|
141
|
+
const methodName = commandArgs.methodName ?? commandArgs.method;
|
|
142
|
+
const methodArgs = Array.isArray(commandArgs.args) ? commandArgs.args : [];
|
|
143
|
+
if (!selector) {
|
|
144
|
+
throw new Error("interact requires selector");
|
|
145
|
+
}
|
|
146
|
+
if (!methodName) {
|
|
147
|
+
throw new Error("interact requires methodName");
|
|
148
|
+
}
|
|
149
|
+
const result = await this.browser.interact({ selector, ...this.normalizeFindArgs(commandArgs) }, methodName, ...methodArgs);
|
|
150
|
+
return { result };
|
|
151
|
+
}
|
|
152
|
+
throw new Error(`Unknown browser command: ${command}`);
|
|
153
|
+
}
|
|
154
|
+
}
|
|
155
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"browser-command-runner.js","sourceRoot":"","sources":["../src/browser-command-runner.js"],"names":[],"mappings":"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","sourcesContent":["/** Runs browser commands across CLI and WebSocket transports. */\nexport default class BrowserCommandRunner {\n  /**\n   * @param {object} args\n   * @param {import(\"./browser.js\").default} args.browser\n   */\n  constructor({browser}) {\n    this.browser = browser\n  }\n\n  /**\n   * @param {Record<string, any>} commandArgs\n   * @returns {{timeout?: number}}\n   */\n  normalizeTimeoutArgs(commandArgs) {\n    const normalizedArgs = {}\n\n    if (\"timeout\" in commandArgs && commandArgs.timeout !== undefined) {\n      normalizedArgs.timeout = Number(commandArgs.timeout)\n\n      if (Number.isNaN(normalizedArgs.timeout)) {\n        throw new Error(`Invalid timeout: ${commandArgs.timeout}`)\n      }\n    }\n\n    return normalizedArgs\n  }\n\n  /**\n   * @param {Record<string, any>} commandArgs\n   * @returns {import(\"./system-test.js\").FindArgs}\n   */\n  normalizeFindArgs(commandArgs) {\n    const findArgs = /** @type {import(\"./system-test.js\").FindArgs} */ (this.normalizeTimeoutArgs(commandArgs))\n\n    if (\"visible\" in commandArgs && commandArgs.visible !== undefined) {\n      if (commandArgs.visible === null || commandArgs.visible === \"null\") {\n        findArgs.visible = null\n      } else if (typeof commandArgs.visible === \"boolean\") {\n        findArgs.visible = commandArgs.visible\n      } else {\n        findArgs.visible = commandArgs.visible === \"true\"\n      }\n    }\n\n    if (\"useBaseSelector\" in commandArgs && commandArgs.useBaseSelector !== undefined) {\n      if (typeof commandArgs.useBaseSelector === \"boolean\") {\n        findArgs.useBaseSelector = commandArgs.useBaseSelector\n      } else {\n        findArgs.useBaseSelector = commandArgs.useBaseSelector === \"true\"\n      }\n    }\n\n    return findArgs\n  }\n\n  /**\n   * @param {import(\"selenium-webdriver\").WebElement} element\n   * @returns {Promise<Record<string, any>>}\n   */\n  async serializeElement(element) {\n    const text = await element.getText()\n    const tagName = await element.getTagName()\n    const displayed = await element.isDisplayed()\n\n    return {displayed, tagName, text}\n  }\n\n  /**\n   * @param {string} command\n   * @param {Record<string, any>} commandArgs\n   * @returns {Promise<any>}\n   */\n  async run(command, commandArgs = {}) {\n    if (command === \"visit\") {\n      const path = commandArgs.path ?? commandArgs.url\n\n      if (!path) {\n        throw new Error(\"visit requires path or url\")\n      }\n\n      await this.browser.visit(path, this.normalizeTimeoutArgs(commandArgs))\n      return {ok: true}\n    }\n\n    if (command === \"dismissTo\") {\n      const path = commandArgs.path ?? commandArgs.url\n\n      if (!path) {\n        throw new Error(\"dismissTo requires path or url\")\n      }\n\n      await this.browser.dismissTo(path, this.normalizeTimeoutArgs(commandArgs))\n      return {ok: true}\n    }\n\n    if (command === \"setBaseSelector\") {\n      if (!commandArgs.selector) {\n        throw new Error(\"setBaseSelector requires selector\")\n      }\n\n      this.browser.setBaseSelector(commandArgs.selector)\n      return {ok: true}\n    }\n\n    if (command === \"getCurrentUrl\") {\n      return {currentUrl: await this.browser.getCurrentUrl()}\n    }\n\n    if (command === \"getHTML\") {\n      return {html: await this.browser.getHTML()}\n    }\n\n    if (command === \"getBrowserLogs\") {\n      return {logs: await this.browser.getBrowserLogs()}\n    }\n\n    if (command === \"takeScreenshot\") {\n      return await this.browser.takeScreenshot()\n    }\n\n    if (command === \"find\") {\n      if (!commandArgs.selector) {\n        throw new Error(\"find requires selector\")\n      }\n\n      const element = await this.browser.find(commandArgs.selector, this.normalizeFindArgs(commandArgs))\n\n      return {element: await this.serializeElement(element)}\n    }\n\n    if (command === \"findByTestID\") {\n      const testID = commandArgs.testID ?? commandArgs.testId\n\n      if (!testID) {\n        throw new Error(\"findByTestID requires testID\")\n      }\n\n      const element = await this.browser.findByTestID(testID, this.normalizeFindArgs(commandArgs))\n\n      return {element: await this.serializeElement(element)}\n    }\n\n    if (command === \"click\") {\n      const selector = commandArgs.selector\n\n      if (!selector) {\n        throw new Error(\"click requires selector\")\n      }\n\n      await this.browser.click(selector, this.normalizeFindArgs(commandArgs))\n      return {ok: true}\n    }\n\n    if (command === \"waitForNoSelector\") {\n      if (!commandArgs.selector) {\n        throw new Error(\"waitForNoSelector requires selector\")\n      }\n\n      await this.browser.waitForNoSelector(commandArgs.selector, this.normalizeFindArgs(commandArgs))\n      return {ok: true}\n    }\n\n    if (command === \"expectNoElement\") {\n      if (!commandArgs.selector) {\n        throw new Error(\"expectNoElement requires selector\")\n      }\n\n      await this.browser.expectNoElement(commandArgs.selector, this.normalizeFindArgs(commandArgs))\n      return {ok: true}\n    }\n\n    if (command === \"interact\") {\n      const selector = commandArgs.selector\n      const methodName = commandArgs.methodName ?? commandArgs.method\n      const methodArgs = Array.isArray(commandArgs.args) ? commandArgs.args : []\n\n      if (!selector) {\n        throw new Error(\"interact requires selector\")\n      }\n\n      if (!methodName) {\n        throw new Error(\"interact requires methodName\")\n      }\n\n      const result = await this.browser.interact({selector, ...this.normalizeFindArgs(commandArgs)}, methodName, ...methodArgs)\n\n      return {result}\n    }\n\n    throw new Error(`Unknown browser command: ${command}`)\n  }\n}\n"]}
|
|
@@ -0,0 +1,3 @@
|
|
|
1
|
+
export const browserDaemonStopTimeoutMs = 10000;
|
|
2
|
+
export const browserDaemonVerifyTimeoutMs = 1000;
|
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYnJvd3Nlci1kYWVtb24tY29uc3RhbnRzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vc3JjL2Jyb3dzZXItZGFlbW9uLWNvbnN0YW50cy5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxNQUFNLENBQUMsTUFBTSwwQkFBMEIsR0FBRyxLQUFLLENBQUE7QUFDL0MsTUFBTSxDQUFDLE1BQU0sNEJBQTRCLEdBQUcsSUFBSSxDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGNvbnN0IGJyb3dzZXJEYWVtb25TdG9wVGltZW91dE1zID0gMTAwMDBcbmV4cG9ydCBjb25zdCBicm93c2VyRGFlbW9uVmVyaWZ5VGltZW91dE1zID0gMTAwMFxuIl19
|
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
/** Long-running browser daemon exposing browser commands over WebSocket. */
|
|
2
|
+
export default class BrowserProcess {
|
|
3
|
+
/**
|
|
4
|
+
* @param {object} args
|
|
5
|
+
* @param {string} args.name
|
|
6
|
+
* @param {Browser} [args.browser]
|
|
7
|
+
* @param {Record<string, any>} [args.browserArgs]
|
|
8
|
+
* @param {string} [args.baseUrl]
|
|
9
|
+
* @param {boolean} [args.debug]
|
|
10
|
+
* @param {number} [args.port]
|
|
11
|
+
*/
|
|
12
|
+
constructor({ name, browser, browserArgs, baseUrl, debug, port }: {
|
|
13
|
+
name: string;
|
|
14
|
+
browser?: Browser;
|
|
15
|
+
browserArgs?: Record<string, any>;
|
|
16
|
+
baseUrl?: string;
|
|
17
|
+
debug?: boolean;
|
|
18
|
+
port?: number;
|
|
19
|
+
});
|
|
20
|
+
name: string;
|
|
21
|
+
browser: Browser;
|
|
22
|
+
baseUrl: string;
|
|
23
|
+
debug: boolean;
|
|
24
|
+
requestRunner: BrowserCommandRunner;
|
|
25
|
+
requestCount: number;
|
|
26
|
+
port: number;
|
|
27
|
+
/** @returns {Promise<void>} */
|
|
28
|
+
start(): Promise<void>;
|
|
29
|
+
wss: import("ws").Server<typeof import("ws").default, typeof import("node:http").IncomingMessage>;
|
|
30
|
+
/** @returns {Promise<void>} */
|
|
31
|
+
stop(): Promise<void>;
|
|
32
|
+
stopped: boolean;
|
|
33
|
+
/**
|
|
34
|
+
* @param {import("ws").WebSocket} ws
|
|
35
|
+
* @returns {void}
|
|
36
|
+
*/
|
|
37
|
+
onConnection: (ws: import("ws").WebSocket) => void;
|
|
38
|
+
/**
|
|
39
|
+
* @param {Record<string, any>} payload
|
|
40
|
+
* @returns {Promise<any>}
|
|
41
|
+
*/
|
|
42
|
+
handlePayload(payload: Record<string, any>): Promise<any>;
|
|
43
|
+
}
|
|
44
|
+
import Browser from "./browser.js";
|
|
45
|
+
import BrowserCommandRunner from "./browser-command-runner.js";
|