testdriverai 7.8.0-test.9 → 7.9.0-test.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (56) hide show
  1. package/agent/index.js +12 -0
  2. package/agent/lib/http.js +21 -3
  3. package/agent/lib/logger.js +15 -0
  4. package/agent/lib/provision-commands.js +176 -0
  5. package/agent/lib/sandbox.js +667 -118
  6. package/agent/lib/sdk.js +1 -20
  7. package/ai/skills/testdriver-find/SKILL.md +14 -20
  8. package/docs/_data/examples-manifest.json +46 -46
  9. package/docs/_scripts/extract-example-urls.js +67 -72
  10. package/docs/changelog.mdx +26 -0
  11. package/docs/docs.json +2 -1
  12. package/docs/v7/examples/ai.mdx +1 -1
  13. package/docs/v7/examples/assert.mdx +1 -1
  14. package/docs/v7/examples/captcha-api.mdx +1 -1
  15. package/docs/v7/examples/chrome-extension.mdx +1 -1
  16. package/docs/v7/examples/drag-and-drop.mdx +1 -1
  17. package/docs/v7/examples/element-not-found.mdx +1 -1
  18. package/docs/v7/examples/exec-output.mdx +1 -1
  19. package/docs/v7/examples/exec-pwsh.mdx +1 -1
  20. package/docs/v7/examples/focus-window.mdx +1 -1
  21. package/docs/v7/examples/hover-image.mdx +1 -1
  22. package/docs/v7/examples/hover-text.mdx +1 -1
  23. package/docs/v7/examples/installer.mdx +1 -1
  24. package/docs/v7/examples/launch-vscode-linux.mdx +1 -1
  25. package/docs/v7/examples/match-image.mdx +1 -1
  26. package/docs/v7/examples/press-keys.mdx +1 -1
  27. package/docs/v7/examples/scroll-keyboard.mdx +1 -1
  28. package/docs/v7/examples/scroll-until-image.mdx +1 -1
  29. package/docs/v7/examples/scroll-until-text.mdx +1 -1
  30. package/docs/v7/examples/scroll.mdx +1 -1
  31. package/docs/v7/examples/type.mdx +1 -1
  32. package/docs/v7/examples/windows-installer.mdx +1 -1
  33. package/docs/v7/find.mdx +14 -20
  34. package/docs/v7/test-results-json.mdx +258 -0
  35. package/examples/scroll-keyboard.test.mjs +1 -1
  36. package/examples/scroll.test.mjs +1 -12
  37. package/interfaces/vitest-plugin.mjs +167 -51
  38. package/lib/core/Dashcam.js +18 -31
  39. package/lib/environments.json +8 -4
  40. package/lib/github-comment.mjs +58 -40
  41. package/lib/init-project.js +5 -67
  42. package/lib/resolve-channel.js +39 -10
  43. package/lib/sentry.js +47 -23
  44. package/lib/vitest/hooks.mjs +117 -20
  45. package/manual/exec-stream-logs.test.mjs +25 -0
  46. package/mcp-server/dist/server.mjs +28 -8
  47. package/mcp-server/src/server.ts +31 -8
  48. package/package.json +2 -1
  49. package/sdk.d.ts +4 -0
  50. package/sdk.js +42 -12
  51. package/setup/aws/install-dev-runner.sh +79 -0
  52. package/setup/aws/spawn-runner.sh +165 -0
  53. package/test-sentry-span.js +35 -0
  54. package/vitest.config.mjs +7 -3
  55. package/vitest.runner.config.mjs +33 -0
  56. package/docs/v7/_drafts/core.mdx +0 -458
@@ -12,7 +12,7 @@ Watch this test execute in a real sandbox environment:
12
12
 
13
13
  {/* installer.test.mjs output */}
14
14
  <iframe
15
- src="https://api.testdriver.ai/api/v1/testdriver/testcase/69a62b570446888b52a4e1c9/replay"
15
+ src="https://api-test.testdriver.ai/api/v1/testdriver/testcase/69c47ec06f887c881af8c673/replay"
16
16
  width="100%"
17
17
  height="390"
18
18
  style={{ border: "1px solid #333", borderRadius: "8px" }}
@@ -12,7 +12,7 @@ Watch this test execute in a real sandbox environment:
12
12
 
13
13
  {/* launch-vscode-linux.test.mjs output */}
14
14
  <iframe
15
- src="https://api.testdriver.ai/api/v1/testdriver/testcase/69a62b91113035da665496a6/replay"
15
+ src="https://api-test.testdriver.ai/api/v1/testdriver/testcase/69c47eb85d526a093f1e6c8c/replay"
16
16
  width="100%"
17
17
  height="390"
18
18
  style={{ border: "1px solid #333", borderRadius: "8px" }}
@@ -12,7 +12,7 @@ Watch this test execute in a real sandbox environment:
12
12
 
13
13
  {/* match-image.test.mjs output */}
14
14
  <iframe
15
- src="https://api.testdriver.ai/api/v1/testdriver/testcase/69a62b42258e8885264fc704/replay"
15
+ src="https://api-test.testdriver.ai/api/v1/testdriver/testcase/69c47eb56f887c881af8c66b/replay"
16
16
  width="100%"
17
17
  height="390"
18
18
  style={{ border: "1px solid #333", borderRadius: "8px" }}
@@ -12,7 +12,7 @@ Watch this test execute in a real sandbox environment:
12
12
 
13
13
  {/* press-keys.test.mjs output */}
14
14
  <iframe
15
- src="https://api.testdriver.ai/api/v1/testdriver/testcase/69a62b800d4a5265e44e7d86/replay"
15
+ src="https://api-test.testdriver.ai/api/v1/testdriver/testcase/69c47ec3d5e55524a4d6ca04/replay"
16
16
  width="100%"
17
17
  height="390"
18
18
  style={{ border: "1px solid #333", borderRadius: "8px" }}
@@ -12,7 +12,7 @@ Watch this test execute in a real sandbox environment:
12
12
 
13
13
  {/* scroll-keyboard.test.mjs output */}
14
14
  <iframe
15
- src="https://api.testdriver.ai/api/v1/testdriver/testcase/69a62b82258e8885264fc728/replay"
15
+ src="https://api-test.testdriver.ai/api/v1/testdriver/testcase/69c47ec85d526a093f1e6c9b/replay"
16
16
  width="100%"
17
17
  height="390"
18
18
  style={{ border: "1px solid #333", borderRadius: "8px" }}
@@ -12,7 +12,7 @@ Watch this test execute in a real sandbox environment:
12
12
 
13
13
  {/* scroll-until-image.test.mjs output */}
14
14
  <iframe
15
- src="https://api.testdriver.ai/api/v1/testdriver/testcase/69a62b4549845ced0b71e2b0/replay"
15
+ src="https://api-test.testdriver.ai/api/v1/testdriver/testcase/69c47ecad5e55524a4d6ca0b/replay"
16
16
  width="100%"
17
17
  height="390"
18
18
  style={{ border: "1px solid #333", borderRadius: "8px" }}
@@ -12,7 +12,7 @@ Watch this test execute in a real sandbox environment:
12
12
 
13
13
  {/* scroll-until-text.test.mjs output */}
14
14
  <iframe
15
- src="https://api.testdriver.ai/api/v1/testdriver/testcase/69a62b99dc33133fc0da9440/replay"
15
+ src="https://api-test.testdriver.ai/api/v1/testdriver/testcase/69a62b99dc33133fc0da9440/replay"
16
16
  width="100%"
17
17
  height="390"
18
18
  style={{ border: "1px solid #333", borderRadius: "8px" }}
@@ -12,7 +12,7 @@ Watch this test execute in a real sandbox environment:
12
12
 
13
13
  {/* scroll.test.mjs output */}
14
14
  <iframe
15
- src="https://api.testdriver.ai/api/v1/testdriver/testcase/69a62b83113035da665496a3/replay"
15
+ src="https://api-test.testdriver.ai/api/v1/testdriver/testcase/69c47ece5d526a093f1e6ca3/replay"
16
16
  width="100%"
17
17
  height="390"
18
18
  style={{ border: "1px solid #333", borderRadius: "8px" }}
@@ -12,7 +12,7 @@ Watch this test execute in a real sandbox environment:
12
12
 
13
13
  {/* type.test.mjs output */}
14
14
  <iframe
15
- src="https://api.testdriver.ai/api/v1/testdriver/testcase/69a62b8d49845ced0b71e2bf/replay"
15
+ src="https://api-test.testdriver.ai/api/v1/testdriver/testcase/69c47ec25d526a093f1e6c92/replay"
16
16
  width="100%"
17
17
  height="390"
18
18
  style={{ border: "1px solid #333", borderRadius: "8px" }}
@@ -12,7 +12,7 @@ Watch this test execute in a real sandbox environment:
12
12
 
13
13
  {/* windows-installer.test.mjs output */}
14
14
  <iframe
15
- src="https://api.testdriver.ai/api/v1/testdriver/testcase/69a62b42565d339e8065f17f/replay"
15
+ src="https://api-test.testdriver.ai/api/v1/testdriver/testcase/69c47eba6ebfe4f295e78011/replay"
16
16
  width="100%"
17
17
  height="390"
18
18
  style={{ border: "1px solid #333", borderRadius: "8px" }}
package/docs/v7/find.mdx CHANGED
@@ -50,8 +50,8 @@ const element = await testdriver.find(description, options)
50
50
  - `"any"` — No wrapping, uses the description as-is (default behavior)
51
51
  </ParamField>
52
52
 
53
- <ParamField path="zoom" type="boolean" default={false}>
54
- Enable two-phase zoom mode for better precision in crowded UIs with many similar elements.
53
+ <ParamField path="zoom" type="boolean" default={true}>
54
+ Two-phase zoom mode for better precision in crowded UIs with many similar elements. Enabled by default.
55
55
  </ParamField>
56
56
 
57
57
  <ParamField path="ai" type="object">
@@ -333,14 +333,19 @@ The `timeout` option:
333
333
  - Returns the element (check `element.found()` if not throwing on failure)
334
334
  - Set to `0` to disable polling and make a single attempt
335
335
 
336
- ## Zoom Mode for Crowded UIs
336
+ ## Zoom Mode
337
337
 
338
- When dealing with many similar icons or elements clustered together (like browser toolbars), enable `zoom` mode for better precision:
338
+ Zoom mode is **enabled by default**. It uses a two-phase approach for better precision when locating elements, especially in crowded UIs with many similar elements.
339
+
340
+ To disable zoom for a specific find call, pass `zoom: false`:
339
341
 
340
342
  ```javascript
341
- // Enable zoom for better precision in crowded UIs
342
- const extensionsBtn = await testdriver.find('extensions puzzle icon in Chrome toolbar', { zoom: true });
343
+ // Zoom is on by default no option needed
344
+ const extensionsBtn = await testdriver.find('extensions puzzle icon in Chrome toolbar');
343
345
  await extensionsBtn.click();
346
+
347
+ // Disable zoom for a specific call if needed
348
+ const largeButton = await testdriver.find('big hero button', { zoom: false });
344
349
  ```
345
350
 
346
351
  ### How Zoom Mode Works
@@ -353,22 +358,11 @@ await extensionsBtn.click();
353
358
  This two-phase approach gives the AI a higher-resolution view of the target area, improving accuracy when multiple similar elements are close together.
354
359
 
355
360
  <Tip>
356
- Use `zoom: true` when:
357
- - Clicking small icons in toolbars
358
- - Selecting from a grid of similar items
359
- - Targeting elements in dense UI areas
360
- - The default locate is clicking the wrong similar element
361
- - You get an AI verification rejection like "The crosshair is located in the empty space of the browser's tab bar/title bar area" — this means the initial locate was imprecise and zoom will help the AI pinpoint the correct element
361
+ You may want to disable zoom with `zoom: false` when:
362
+ - Targeting large, isolated elements where the extra precision isn't needed
363
+ - You want to speed up find calls in simple UIs
362
364
  </Tip>
363
365
 
364
- ```javascript
365
- // Without zoom - may click wrong icon in toolbar
366
- const icon = await testdriver.find('settings icon');
367
-
368
- // With zoom - better precision for crowded areas
369
- const icon = await testdriver.find('settings icon', { zoom: true });
370
- ```
371
-
372
366
  ## Cache Options
373
367
 
374
368
  Control caching behavior to optimize performance, especially when using dynamic variables in prompts.
@@ -0,0 +1,258 @@
1
+ ---
2
+ title: "Test Result JSON"
3
+ sidebarTitle: "Test Result JSON"
4
+ description: "Per-test JSON result files with metadata, versions, and infrastructure details"
5
+ icon: "file-code"
6
+ ---
7
+
8
+ ## Overview
9
+
10
+ TestDriver automatically writes a JSON result file for each test case after it finishes. These files contain comprehensive metadata about the test run, including SDK and runner versions, infrastructure details, interaction statistics, and links to recordings.
11
+
12
+ Result files are written to:
13
+
14
+ ```
15
+ .testdriver/results/<testFile>/<testName>.json
16
+ ```
17
+
18
+ For example, a test file `tests/login.test.mjs` with a test named `"should log in"` produces:
19
+
20
+ ```
21
+ .testdriver/results/tests/login.test.mjs/should_log_in.json
22
+ ```
23
+
24
+ <Note>
25
+ Test names are sanitized for filesystem use — special characters are replaced with underscores and names are truncated to 200 characters.
26
+ </Note>
27
+
28
+ ## Enabling
29
+
30
+ No configuration is required. The JSON files are written automatically by the TestDriver Vitest reporter plugin whenever tests run.
31
+
32
+ ## JSON Schema
33
+
34
+ Each result file is organized into logical groups:
35
+
36
+ ### `versions`
37
+
38
+ | Field | Type | Description |
39
+ |---|---|---|
40
+ | `versions.sdk` | `string \| null` | TestDriver SDK version (e.g. `"7.8.0"`) |
41
+ | `versions.vitest` | `string \| null` | Vitest version used to run the test |
42
+ | `versions.api` | `string \| null` | TestDriver API server version |
43
+ | `versions.runnerBefore` | `string \| null` | Runner version at sandbox start |
44
+ | `versions.runnerAfter` | `string \| null` | Runner version after auto-update |
45
+ | `versions.runnerWasUpdated` | `boolean` | Whether the runner was auto-updated during provisioning |
46
+
47
+ ### `test`
48
+
49
+ | Field | Type | Description |
50
+ |---|---|---|
51
+ | `test.file` | `string \| null` | Relative path to the test file |
52
+ | `test.name` | `string \| null` | Name of the test case |
53
+ | `test.suite` | `string \| null` | Name of the parent `describe` block |
54
+ | `test.passed` | `boolean` | Whether the test passed |
55
+ | `test.caseId` | `string \| null` | Database ID for this test case |
56
+ | `test.runId` | `string \| null` | Database ID for the overall test run |
57
+ | `test.error` | `string \| null` | Error message if the test failed |
58
+ | `test.errorStack` | `string \| null` | Error stack trace if the test failed |
59
+
60
+ ### `urls`
61
+
62
+ | Field | Type | Description |
63
+ |---|---|---|
64
+ | `urls.api` | `string \| null` | API root URL used for this test |
65
+ | `urls.console` | `string \| null` | TestDriver console base URL |
66
+ | `urls.vnc` | `string \| null` | VNC URL for the sandbox |
67
+ | `urls.testRun` | `string \| null` | Direct link to this test case in the console |
68
+
69
+ ### `replay`
70
+
71
+ The `replay` object contains the recording replay URL and derived embed links. The `gifUrl` and `embedUrl` are generated automatically from the replay URL.
72
+
73
+ | Field | Type | Description |
74
+ |---|---|---|
75
+ | `replay.url` | `string \| null` | Recording replay URL |
76
+ | `replay.gifUrl` | `string \| null` | Animated GIF thumbnail of the recording |
77
+ | `replay.embedUrl` | `string \| null` | Embeddable replay URL (appends `&embed=true`) |
78
+ | `replay.markdown` | `string \| null` | Ready-to-use Markdown embed with GIF linking to the replay |
79
+
80
+ The `replay.markdown` field produces a clickable GIF badge you can paste directly into PR comments, README files, or issue descriptions:
81
+
82
+ ```markdown
83
+ [![Test Recording](https://api.testdriver.ai/replay/abc123/gif?shareKey=xyz)](https://console.testdriver.ai/replay/abc123?share=xyz)
84
+ ```
85
+
86
+ ### `date`
87
+
88
+ | Field | Type | Description |
89
+ |---|---|---|
90
+ | `date` | `string` | ISO 8601 timestamp when the test finished |
91
+
92
+ ### `team`
93
+
94
+ | Field | Type | Description |
95
+ |---|---|---|
96
+ | `team.id` | `string \| null` | Team ID from the sandbox |
97
+ | `team.sessionId` | `string \| null` | SDK session ID |
98
+
99
+ ### `infrastructure`
100
+
101
+ | Field | Type | Description |
102
+ |---|---|---|
103
+ | `infrastructure.sandboxId` | `string \| null` | Sandbox instance ID |
104
+ | `infrastructure.instanceId` | `string \| null` | Instance ID |
105
+ | `infrastructure.os` | `string \| null` | Operating system of the sandbox (`"linux"` or `"windows"`) |
106
+ | `infrastructure.amiId` | `string \| null` | AWS AMI ID used for provisioning |
107
+ | `infrastructure.e2bTemplateId` | `string \| null` | E2B template ID used for provisioning |
108
+ | `infrastructure.imageVersion` | `string \| null` | Sandbox image version |
109
+
110
+ ### `realtime`
111
+
112
+ | Field | Type | Description |
113
+ |---|---|---|
114
+ | `realtime.channel` | `string \| null` | Ably channel name used for communication |
115
+ | `realtime.messageCount` | `number` | Number of messages published to the realtime channel |
116
+
117
+ ### `interactions`
118
+
119
+ | Field | Type | Description |
120
+ |---|---|---|
121
+ | `interactions.total` | `number` | Total number of interactions recorded |
122
+ | `interactions.cached` | `number` | Number of interactions served from cache |
123
+ | `interactions.byType` | `object` | Breakdown of interactions by type (e.g. `find`, `click`, `assert`) |
124
+
125
+ ## Example Output
126
+
127
+ ```json
128
+ {
129
+ "sdkVersion": "7.8.0",
130
+ "vitestVersion": "4.0.0",
131
+ "apiVersion": "1.45.0",
132
+ "runnerVersionBefore": "2.1.0",
133
+ "runnerVersionAfter": "2.1.1",
134
+ "wasUpdated": true,
135
+ "apiUrl": "https://api.testdriver.ai",
136
+ "consoleUrl": "https://console.testdriver.ai",
137
+ "testRunLink": "https://console.testdriver.ai/runs/abc123/def456",
138
+ "dashcamUrl": "https://app.dashcam.io/replay/abc123",
139
+ "vncUrl": "wss://sandbox-123.testdriver.ai/vnc",
140
+ "date": "2025-01-15T14:30:00.000Z",
141
+ "team": {
142
+ "id": "team_abc123",
143
+ "sessionId": "sess_xyz789"
144
+ },
145
+ "infrastructure": {
146
+ "sandboxId": "sandbox-123",
147
+ "instanceId": "i-abc123",
148
+ "os": "linux",
149
+ "amiId": "ami-0abc123",
150
+ "e2bTemplateId": null,
151
+ "imageVersion": "v2.1.0"
152
+ },
153
+ "realtime": {
154
+ "channel": "sandbox:sandbox-123",
155
+ "messageCount": 42
156
+ },
157
+ "interactions": {
158
+ "total": 15,
159
+ "cached": 3,
160
+ "byType": {
161
+ "find": 8,
162
+ "click": 5,
163
+ "assert": 2
164
+ }
165
+ }
166
+ }
167
+ ```
168
+
169
+ ## Using Result Files in CI
170
+
171
+ Result files are useful for extracting test metadata in CI pipelines without parsing log output.
172
+
173
+ ### GitHub Actions Example
174
+
175
+ Use `fromJSON` to parse a result file into a GitHub Actions expression you can reference in subsequent steps:
176
+
177
+ ```yaml
178
+ - name: Run tests
179
+ run: npx vitest run tests/login.test.mjs
180
+
181
+ - name: Parse result
182
+ id: result
183
+ run: |
184
+ # Read the first JSON result file
185
+ FILE=$(find .testdriver/results -name '*.json' | head -n 1)
186
+ echo "json=$(cat "$FILE")" >> "$GITHUB_OUTPUT"
187
+
188
+ - name: Comment on PR
189
+ if: fromJSON(steps.result.outputs.json).test.passed == false
190
+ uses: actions/github-script@v7
191
+ with:
192
+ script: |
193
+ const result = ${{ steps.result.outputs.json }};
194
+ await github.rest.issues.createComment({
195
+ owner: context.repo.owner,
196
+ repo: context.repo.repo,
197
+ issue_number: context.issue.number,
198
+ body: [
199
+ `❌ **${result.test.name}** failed`,
200
+ ``,
201
+ `Error: ${result.test.error}`,
202
+ ``,
203
+ result.replay.markdown,
204
+ ``,
205
+ `[View full recording](${result.urls.testRun})`
206
+ ].join('\n')
207
+ });
208
+ ```
209
+
210
+ You can also load all results into a matrix or iterate over them:
211
+
212
+ ```yaml
213
+ - name: Run tests
214
+ run: npx vitest run tests/*.test.mjs
215
+
216
+ - name: Collect results
217
+ id: results
218
+ run: |
219
+ # Merge all result files into a JSON array
220
+ echo "json=$(find .testdriver/results -name '*.json' -exec cat {} + | jq -s '.')" >> "$GITHUB_OUTPUT"
221
+
222
+ - name: Summary
223
+ run: |
224
+ echo '## Test Results' >> $GITHUB_STEP_SUMMARY
225
+ RESULTS='${{ steps.results.outputs.json }}'
226
+ echo "$RESULTS" | jq -r '.[] | "| \(.test.name) | \(if .test.passed then "✅" else "❌" end) | \(.urls.testRun) |"' >> $GITHUB_STEP_SUMMARY
227
+ ```
228
+
229
+ ### Reading Results Programmatically
230
+
231
+ ```javascript
232
+ import fs from "fs";
233
+ import path from "path";
234
+
235
+ const resultsDir = ".testdriver/results";
236
+
237
+ function readResults(dir) {
238
+ const results = [];
239
+ for (const testDir of fs.readdirSync(dir, { recursive: true })) {
240
+ const fullPath = path.join(dir, testDir);
241
+ if (fullPath.endsWith(".json") && fs.statSync(fullPath).isFile()) {
242
+ results.push(JSON.parse(fs.readFileSync(fullPath, "utf-8")));
243
+ }
244
+ }
245
+ return results;
246
+ }
247
+
248
+ const results = readResults(resultsDir);
249
+ const passed = results.filter(r => r.test.passed);
250
+ const failed = results.filter(r => !r.test.passed);
251
+
252
+ console.log(`${passed.length} passed, ${failed.length} failed`);
253
+ for (const r of failed) {
254
+ console.log(` FAIL: ${r.test.name} — ${r.test.error}`);
255
+ console.log(` Recording: ${r.urls.testRun}`);
256
+ console.log(` Embed: ${r.replay.markdown}`);
257
+ }
258
+ ```
@@ -16,7 +16,7 @@ describe("Scroll Keyboard Test", () => {
16
16
  // Navigate to https://www.webhamster.com/
17
17
  await testdriver.focusApplication("Google Chrome");
18
18
  const urlBar = await testdriver.find(
19
- "testdriver-sandbox.vercel.app/login, the URL in the omnibox showing the current page", {zoom: true}
19
+ "the URL in the omnibox", {zoom: true}
20
20
  );
21
21
  await urlBar.click();
22
22
  await testdriver.pressKeys(["ctrl", "a"]);
@@ -12,19 +12,8 @@ import { getDefaults } from "./config.mjs";
12
12
  describe("Scroll Test", () => {
13
13
  it("should navigate and scroll down the page", async (context) => {
14
14
  const testdriver = TestDriver(context, { ...getDefaults(context), headless: true });
15
- await testdriver.provision.chrome({ url: 'http://testdriver-sandbox.vercel.app/login' });
15
+ await testdriver.provision.chrome({ url: 'https://www.webhamster.com/' });
16
16
 
17
- // Give Chrome a moment to fully render the UI
18
- await new Promise(resolve => setTimeout(resolve, 2000));
19
-
20
- // Navigate to webhamster.com - just look for the domain, not the full path
21
- const urlBar = await testdriver.find(
22
- "testdriver-sandbox.vercel.app, the URL in the address bar",
23
- );
24
- await urlBar.click();
25
- await testdriver.pressKeys(["ctrl", "a"]);
26
- await testdriver.type("https://www.webhamster.com/");
27
- await testdriver.pressKeys(["enter"]);
28
17
 
29
18
  // Wait for page to load and click heading
30
19
  const heading = await testdriver.find(