testdriverai 6.2.2 → 7.1.0

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 (300) hide show
  1. package/.github/workflows/acceptance-linux.yml +75 -0
  2. package/.github/workflows/acceptance-sdk-tests.yml +133 -0
  3. package/.vscode/settings.json +5 -1
  4. package/AGENTS.md +550 -0
  5. package/CODEOWNERS +0 -1
  6. package/README.md +126 -0
  7. package/{testdriver → _testdriver}/acceptance/drag-and-drop.yaml +2 -2
  8. package/{testdriver → _testdriver}/acceptance/snippets/login.yaml +1 -1
  9. package/_testdriver/examples/desktop/lifecycle/prerun.yaml +0 -0
  10. package/{testdriver → _testdriver}/examples/web/lifecycle/prerun.yaml +6 -1
  11. package/{testdriver → _testdriver}/lifecycle/postrun.yaml +3 -2
  12. package/_testdriver/lifecycle/prerun.yaml +15 -0
  13. package/{testdriver → _testdriver}/lifecycle/provision.yaml +7 -2
  14. package/agent/index.js +300 -85
  15. package/agent/interface.js +15 -0
  16. package/agent/lib/cache.js +142 -0
  17. package/agent/lib/commander.js +1 -39
  18. package/agent/lib/commands.js +910 -296
  19. package/agent/lib/redraw.js +129 -41
  20. package/agent/lib/sandbox.js +29 -6
  21. package/agent/lib/sdk.js +22 -0
  22. package/agent/lib/system.js +0 -3
  23. package/agent/lib/validation.js +1 -7
  24. package/debug-locate-response.js +82 -0
  25. package/debugger/index.html +15 -4
  26. package/docs/ARCHITECTURE.md +424 -0
  27. package/docs/AWESOME_LOGS_QUICK_REF.md +100 -0
  28. package/docs/MIGRATION.md +425 -0
  29. package/docs/PRESETS.md +210 -0
  30. package/docs/QUICK_START_TEST_RECORDING.md +215 -0
  31. package/docs/SDK_AWESOME_LOGS.md +468 -0
  32. package/docs/TEST_RECORDING.md +388 -0
  33. package/docs/docs.json +286 -152
  34. package/docs/guide/best-practices-polling.mdx +154 -0
  35. package/docs/sdk-browser-rendering.md +167 -0
  36. package/docs/v6/getting-started/self-hosting.mdx +407 -0
  37. package/docs/{guide → v6/guide}/dashcam.mdx +1 -1
  38. package/docs/{guide → v6/guide}/environment-variables.mdx +4 -5
  39. package/docs/{guide → v6/guide}/lifecycle.mdx +1 -1
  40. package/docs/v6/overview/comparison.mdx +101 -0
  41. package/docs/v7/README.md +135 -0
  42. package/docs/v7/api/ai.mdx +205 -0
  43. package/docs/v7/api/assert.mdx +285 -0
  44. package/docs/v7/api/assertions.mdx +403 -0
  45. package/docs/v7/api/click.mdx +287 -0
  46. package/docs/v7/api/client.mdx +322 -0
  47. package/docs/v7/api/dashcam.mdx +497 -0
  48. package/docs/v7/api/doubleClick.mdx +102 -0
  49. package/docs/v7/api/elements.mdx +479 -0
  50. package/docs/v7/api/exec.mdx +346 -0
  51. package/docs/v7/api/find.mdx +316 -0
  52. package/docs/v7/api/focusApplication.mdx +294 -0
  53. package/docs/v7/api/hover.mdx +279 -0
  54. package/docs/v7/api/mouseDown.mdx +161 -0
  55. package/docs/v7/api/mouseUp.mdx +164 -0
  56. package/docs/v7/api/pressKeys.mdx +349 -0
  57. package/docs/v7/api/rightClick.mdx +123 -0
  58. package/docs/v7/api/sandbox.mdx +404 -0
  59. package/docs/v7/api/scroll.mdx +300 -0
  60. package/docs/v7/api/type.mdx +314 -0
  61. package/docs/v7/commands/assert.mdx +45 -0
  62. package/docs/v7/commands/exec.mdx +282 -0
  63. package/docs/v7/commands/focus-application.mdx +44 -0
  64. package/docs/v7/commands/hover-image.mdx +69 -0
  65. package/docs/v7/commands/hover-text.mdx +47 -0
  66. package/docs/v7/commands/if.mdx +53 -0
  67. package/docs/v7/commands/match-image.mdx +67 -0
  68. package/docs/v7/commands/press-keys.mdx +87 -0
  69. package/docs/v7/commands/remember.mdx +49 -0
  70. package/docs/v7/commands/run.mdx +44 -0
  71. package/docs/v7/commands/scroll-until-image.mdx +66 -0
  72. package/docs/v7/commands/scroll-until-text.mdx +60 -0
  73. package/docs/v7/commands/scroll.mdx +69 -0
  74. package/docs/v7/commands/type.mdx +45 -0
  75. package/docs/v7/commands/wait-for-image.mdx +54 -0
  76. package/docs/v7/commands/wait-for-text.mdx +48 -0
  77. package/docs/v7/commands/wait.mdx +45 -0
  78. package/docs/v7/getting-started/configuration.mdx +380 -0
  79. package/docs/v7/getting-started/quickstart.mdx +332 -0
  80. package/docs/v7/guides/best-practices.mdx +486 -0
  81. package/docs/v7/guides/caching-ai.mdx +215 -0
  82. package/docs/v7/guides/caching-selectors.mdx +292 -0
  83. package/docs/v7/guides/caching.mdx +366 -0
  84. package/docs/v7/guides/ci-cd/azure.mdx +587 -0
  85. package/docs/v7/guides/ci-cd/circleci.mdx +523 -0
  86. package/docs/v7/guides/ci-cd/github-actions.mdx +457 -0
  87. package/docs/v7/guides/ci-cd/gitlab.mdx +498 -0
  88. package/docs/v7/guides/ci-cd/jenkins.mdx +664 -0
  89. package/docs/v7/guides/ci-cd/travis.mdx +438 -0
  90. package/docs/v7/guides/debugging.mdx +349 -0
  91. package/docs/v7/guides/faq.mdx +393 -0
  92. package/docs/v7/guides/migration.mdx +562 -0
  93. package/docs/v7/guides/performance.mdx +517 -0
  94. package/docs/{getting-started → v7/guides}/self-hosting.mdx +11 -12
  95. package/docs/v7/guides/troubleshooting.mdx +526 -0
  96. package/docs/v7/guides/vitest-plugin.mdx +477 -0
  97. package/docs/v7/guides/vitest.mdx +535 -0
  98. package/docs/v7/platforms/linux.mdx +308 -0
  99. package/docs/v7/platforms/macos.mdx +433 -0
  100. package/docs/v7/platforms/windows.mdx +430 -0
  101. package/docs/v7/playwright.mdx +342 -0
  102. package/docs/v7/presets/chrome-extension.mdx +223 -0
  103. package/docs/v7/presets/chrome.mdx +287 -0
  104. package/docs/v7/presets/electron.mdx +435 -0
  105. package/docs/v7/presets/vscode.mdx +398 -0
  106. package/docs/v7/presets/webapp.mdx +396 -0
  107. package/docs/v7/progressive-apis/CORE.md +459 -0
  108. package/docs/v7/progressive-apis/HOOKS.md +360 -0
  109. package/docs/v7/progressive-apis/PROGRESSIVE_DISCLOSURE.md +230 -0
  110. package/docs/v7/progressive-apis/PROVISION.md +266 -0
  111. package/eslint.config.js +19 -1
  112. package/interfaces/cli/lib/base.js +10 -4
  113. package/interfaces/logger.js +2 -1
  114. package/interfaces/shared-test-state.mjs +69 -0
  115. package/interfaces/vitest-plugin.mjs +830 -0
  116. package/package.json +29 -5
  117. package/schema.json +8 -29
  118. package/scripts/view-test-results.mjs +96 -0
  119. package/sdk-log-formatter.js +714 -0
  120. package/sdk.d.ts +1028 -0
  121. package/sdk.js +2567 -0
  122. package/{.github/workflows/self-hosted.yml → self-hosted.yml} +13 -4
  123. package/setup/aws/cloudformation.yaml +9 -2
  124. package/src/core/Dashcam.js +469 -0
  125. package/src/core/index.d.ts +150 -0
  126. package/src/core/index.js +12 -0
  127. package/src/presets/index.mjs +331 -0
  128. package/src/vitest/extended.mjs +108 -0
  129. package/src/vitest/hooks.d.ts +119 -0
  130. package/src/vitest/hooks.mjs +298 -0
  131. package/src/vitest/index.mjs +64 -0
  132. package/src/vitest/lifecycle.mjs +277 -0
  133. package/src/vitest/utils.mjs +150 -0
  134. package/test/dashcam.test.js +137 -0
  135. package/test/mcp-example-test.yaml +27 -0
  136. package/testdriver/acceptance-sdk/QUICK_REFERENCE.md +61 -0
  137. package/testdriver/acceptance-sdk/README.md +128 -0
  138. package/testdriver/acceptance-sdk/TEST_REPORTING.md +245 -0
  139. package/testdriver/acceptance-sdk/assert.test.mjs +26 -0
  140. package/testdriver/acceptance-sdk/auto-cache-key-demo.test.mjs +56 -0
  141. package/testdriver/acceptance-sdk/chrome-extension.test.mjs +89 -0
  142. package/testdriver/acceptance-sdk/drag-and-drop.test.mjs +58 -0
  143. package/testdriver/acceptance-sdk/element-not-found.test.mjs +25 -0
  144. package/testdriver/acceptance-sdk/exec-js.test.mjs +43 -0
  145. package/testdriver/acceptance-sdk/exec-output.test.mjs +59 -0
  146. package/testdriver/acceptance-sdk/exec-pwsh.test.mjs +57 -0
  147. package/testdriver/acceptance-sdk/focus-window.test.mjs +36 -0
  148. package/testdriver/acceptance-sdk/formatted-logging.test.mjs +26 -0
  149. package/testdriver/acceptance-sdk/hooks-example.test.mjs +38 -0
  150. package/testdriver/acceptance-sdk/hover-image.test.mjs +34 -0
  151. package/testdriver/acceptance-sdk/hover-text-with-description.test.mjs +38 -0
  152. package/testdriver/acceptance-sdk/hover-text.test.mjs +27 -0
  153. package/testdriver/acceptance-sdk/match-image.test.mjs +36 -0
  154. package/testdriver/acceptance-sdk/presets-example.test.mjs +87 -0
  155. package/testdriver/acceptance-sdk/press-keys.test.mjs +50 -0
  156. package/testdriver/acceptance-sdk/prompt.test.mjs +33 -0
  157. package/testdriver/acceptance-sdk/scroll-keyboard.test.mjs +38 -0
  158. package/testdriver/acceptance-sdk/scroll-until-image.test.mjs +39 -0
  159. package/testdriver/acceptance-sdk/scroll-until-text.test.mjs +28 -0
  160. package/testdriver/acceptance-sdk/scroll.test.mjs +41 -0
  161. package/testdriver/acceptance-sdk/setup/globalTeardown.mjs +11 -0
  162. package/testdriver/acceptance-sdk/setup/testHelpers.mjs +420 -0
  163. package/testdriver/acceptance-sdk/setup/vitestSetup.mjs +40 -0
  164. package/testdriver/acceptance-sdk/sully-ai.test.mjs +234 -0
  165. package/testdriver/acceptance-sdk/test-console-logs.test.mjs +42 -0
  166. package/testdriver/acceptance-sdk/type-checking-demo.js +49 -0
  167. package/testdriver/acceptance-sdk/type.test.mjs +45 -0
  168. package/verify-element-api.js +89 -0
  169. package/verify-types.js +0 -0
  170. package/vitest.config.example.js +19 -0
  171. package/vitest.config.mjs +66 -0
  172. package/vitest.config.mjs.bak +44 -0
  173. package/.github/workflows/acceptance-v6.yml +0 -169
  174. package/.vscode/mcp.json +0 -9
  175. package/docs/overview/comparison.mdx +0 -82
  176. package/testdriver/lifecycle/prerun.yaml +0 -17
  177. /package/{testdriver/examples/desktop/lifecycle/prerun.yaml → .env.example} +0 -0
  178. /package/{testdriver → _testdriver}/acceptance/assert.yaml +0 -0
  179. /package/{testdriver → _testdriver}/acceptance/dashcam.yaml +0 -0
  180. /package/{testdriver → _testdriver}/acceptance/embed.yaml +0 -0
  181. /package/{testdriver → _testdriver}/acceptance/exec-js.yaml +0 -0
  182. /package/{testdriver → _testdriver}/acceptance/exec-output.yaml +0 -0
  183. /package/{testdriver → _testdriver}/acceptance/exec-shell.yaml +0 -0
  184. /package/{testdriver → _testdriver}/acceptance/focus-window.yaml +0 -0
  185. /package/{testdriver → _testdriver}/acceptance/hover-image.yaml +0 -0
  186. /package/{testdriver → _testdriver}/acceptance/hover-text-with-description.yaml +0 -0
  187. /package/{testdriver → _testdriver}/acceptance/hover-text.yaml +0 -0
  188. /package/{testdriver → _testdriver}/acceptance/if-else.yaml +0 -0
  189. /package/{testdriver → _testdriver}/acceptance/match-image.yaml +0 -0
  190. /package/{testdriver → _testdriver}/acceptance/press-keys.yaml +0 -0
  191. /package/{testdriver → _testdriver}/acceptance/prompt.yaml +0 -0
  192. /package/{testdriver → _testdriver}/acceptance/remember.yaml +0 -0
  193. /package/{testdriver → _testdriver}/acceptance/screenshots/cart.png +0 -0
  194. /package/{testdriver → _testdriver}/acceptance/scroll-keyboard.yaml +0 -0
  195. /package/{testdriver → _testdriver}/acceptance/scroll-until-image.yaml +0 -0
  196. /package/{testdriver → _testdriver}/acceptance/scroll-until-text.yaml +0 -0
  197. /package/{testdriver → _testdriver}/acceptance/scroll.yaml +0 -0
  198. /package/{testdriver → _testdriver}/acceptance/snippets/match-cart.yaml +0 -0
  199. /package/{testdriver → _testdriver}/acceptance/type.yaml +0 -0
  200. /package/{testdriver → _testdriver}/behavior/failure.yaml +0 -0
  201. /package/{testdriver → _testdriver}/behavior/hover-text.yaml +0 -0
  202. /package/{testdriver → _testdriver}/behavior/lifecycle/postrun.yaml +0 -0
  203. /package/{testdriver → _testdriver}/behavior/lifecycle/prerun.yaml +0 -0
  204. /package/{testdriver → _testdriver}/behavior/lifecycle/provision.yaml +0 -0
  205. /package/{testdriver → _testdriver}/behavior/secrets.yaml +0 -0
  206. /package/{testdriver → _testdriver}/edge-cases/dashcam-chrome.yaml +0 -0
  207. /package/{testdriver → _testdriver}/edge-cases/exec-pwsh-multiline.yaml +0 -0
  208. /package/{testdriver → _testdriver}/edge-cases/js-exception.yaml +0 -0
  209. /package/{testdriver → _testdriver}/edge-cases/js-promise.yaml +0 -0
  210. /package/{testdriver → _testdriver}/edge-cases/lifecycle/postrun.yaml +0 -0
  211. /package/{testdriver → _testdriver}/edge-cases/prompt-in-middle.yaml +0 -0
  212. /package/{testdriver → _testdriver}/edge-cases/prompt-nested.yaml +0 -0
  213. /package/{testdriver → _testdriver}/edge-cases/success-test.yaml +0 -0
  214. /package/{testdriver → _testdriver}/examples/android/example.yaml +0 -0
  215. /package/{testdriver → _testdriver}/examples/android/lifecycle/postrun.yaml +0 -0
  216. /package/{testdriver → _testdriver}/examples/android/lifecycle/provision.yaml +0 -0
  217. /package/{testdriver → _testdriver}/examples/android/readme.md +0 -0
  218. /package/{testdriver → _testdriver}/examples/chrome-extension/lifecycle/provision.yaml +0 -0
  219. /package/{testdriver → _testdriver}/examples/desktop/lifecycle/provision.yaml +0 -0
  220. /package/{testdriver → _testdriver}/examples/vscode-extension/lifecycle/provision.yaml +0 -0
  221. /package/{testdriver → _testdriver}/examples/web/lifecycle/postrun.yaml +0 -0
  222. /package/docs/{account → v6/account}/dashboard.mdx +0 -0
  223. /package/docs/{account → v6/account}/enterprise.mdx +0 -0
  224. /package/docs/{account → v6/account}/pricing.mdx +0 -0
  225. /package/docs/{account → v6/account}/projects.mdx +0 -0
  226. /package/docs/{account → v6/account}/team.mdx +0 -0
  227. /package/docs/{action → v6/action}/ami.mdx +0 -0
  228. /package/docs/{action → v6/action}/performance.mdx +0 -0
  229. /package/docs/{action → v6/action}/secrets.mdx +0 -0
  230. /package/docs/{apps → v6/apps}/chrome-extensions.mdx +0 -0
  231. /package/docs/{apps → v6/apps}/desktop-apps.mdx +0 -0
  232. /package/docs/{apps → v6/apps}/mobile-apps.mdx +0 -0
  233. /package/docs/{apps → v6/apps}/static-websites.mdx +0 -0
  234. /package/docs/{apps → v6/apps}/tauri-apps.mdx +0 -0
  235. /package/docs/{bugs → v6/bugs}/jira.mdx +0 -0
  236. /package/docs/{cli → v6/cli}/overview.mdx +0 -0
  237. /package/docs/{commands → v6/commands}/assert.mdx +0 -0
  238. /package/docs/{commands → v6/commands}/exec.mdx +0 -0
  239. /package/docs/{commands → v6/commands}/focus-application.mdx +0 -0
  240. /package/docs/{commands → v6/commands}/hover-image.mdx +0 -0
  241. /package/docs/{commands → v6/commands}/hover-text.mdx +0 -0
  242. /package/docs/{commands → v6/commands}/if.mdx +0 -0
  243. /package/docs/{commands → v6/commands}/match-image.mdx +0 -0
  244. /package/docs/{commands → v6/commands}/press-keys.mdx +0 -0
  245. /package/docs/{commands → v6/commands}/remember.mdx +0 -0
  246. /package/docs/{commands → v6/commands}/run.mdx +0 -0
  247. /package/docs/{commands → v6/commands}/scroll-until-image.mdx +0 -0
  248. /package/docs/{commands → v6/commands}/scroll-until-text.mdx +0 -0
  249. /package/docs/{commands → v6/commands}/scroll.mdx +0 -0
  250. /package/docs/{commands → v6/commands}/type.mdx +0 -0
  251. /package/docs/{commands → v6/commands}/wait-for-image.mdx +0 -0
  252. /package/docs/{commands → v6/commands}/wait-for-text.mdx +0 -0
  253. /package/docs/{commands → v6/commands}/wait.mdx +0 -0
  254. /package/docs/{exporting → v6/exporting}/junit.mdx +0 -0
  255. /package/docs/{exporting → v6/exporting}/playwright.mdx +0 -0
  256. /package/docs/{features → v6/features}/auto-healing.mdx +0 -0
  257. /package/docs/{features → v6/features}/generation.mdx +0 -0
  258. /package/docs/{features → v6/features}/parallel-testing.mdx +0 -0
  259. /package/docs/{features → v6/features}/reusable-snippets.mdx +0 -0
  260. /package/docs/{features → v6/features}/selectorless.mdx +0 -0
  261. /package/docs/{features → v6/features}/visual-assertions.mdx +0 -0
  262. /package/docs/{getting-started → v6/getting-started}/ci.mdx +0 -0
  263. /package/docs/{getting-started → v6/getting-started}/cli.mdx +0 -0
  264. /package/docs/{getting-started → v6/getting-started}/editing.mdx +0 -0
  265. /package/docs/{getting-started → v6/getting-started}/playwright.mdx +0 -0
  266. /package/docs/{getting-started → v6/getting-started}/running.mdx +0 -0
  267. /package/docs/{getting-started → v6/getting-started}/vscode.mdx +0 -0
  268. /package/docs/{guide → v6/guide}/assertions.mdx +0 -0
  269. /package/docs/{guide → v6/guide}/authentication.mdx +0 -0
  270. /package/docs/{guide → v6/guide}/code.mdx +0 -0
  271. /package/docs/{guide → v6/guide}/locating.mdx +0 -0
  272. /package/docs/{guide → v6/guide}/protips.mdx +0 -0
  273. /package/docs/{guide → v6/guide}/variables.mdx +0 -0
  274. /package/docs/{guide → v6/guide}/waiting.mdx +0 -0
  275. /package/docs/{importing → v6/importing}/csv.mdx +0 -0
  276. /package/docs/{importing → v6/importing}/gherkin.mdx +0 -0
  277. /package/docs/{importing → v6/importing}/jira.mdx +0 -0
  278. /package/docs/{importing → v6/importing}/testrail.mdx +0 -0
  279. /package/docs/{integrations → v6/integrations}/electron.mdx +0 -0
  280. /package/docs/{integrations → v6/integrations}/netlify.mdx +0 -0
  281. /package/docs/{integrations → v6/integrations}/vercel.mdx +0 -0
  282. /package/docs/{interactive → v6/interactive}/explore.mdx +0 -0
  283. /package/docs/{interactive → v6/interactive}/run.mdx +0 -0
  284. /package/docs/{interactive → v6/interactive}/save.mdx +0 -0
  285. /package/docs/{overview → v6/overview}/faq.mdx +0 -0
  286. /package/docs/{overview → v6/overview}/performance.mdx +0 -0
  287. /package/docs/{overview → v6/overview}/quickstart.mdx +0 -0
  288. /package/docs/{overview → v6/overview}/what-is-testdriver.mdx +0 -0
  289. /package/docs/{scenarios → v6/scenarios}/ai-chatbot.mdx +0 -0
  290. /package/docs/{scenarios → v6/scenarios}/cookie-banner.mdx +0 -0
  291. /package/docs/{scenarios → v6/scenarios}/file-upload.mdx +0 -0
  292. /package/docs/{scenarios → v6/scenarios}/form-filling.mdx +0 -0
  293. /package/docs/{scenarios → v6/scenarios}/log-in.mdx +0 -0
  294. /package/docs/{scenarios → v6/scenarios}/pdf-generation.mdx +0 -0
  295. /package/docs/{scenarios → v6/scenarios}/spell-check.mdx +0 -0
  296. /package/docs/{security → v6/security}/action.mdx +0 -0
  297. /package/docs/{security → v6/security}/agent.mdx +0 -0
  298. /package/docs/{security → v6/security}/platform.mdx +0 -0
  299. /package/docs/{tutorials → v6/tutorials}/advanced-test.mdx +0 -0
  300. /package/docs/{tutorials → v6/tutorials}/basic-test.mdx +0 -0
@@ -0,0 +1,285 @@
1
+ ---
2
+ title: "assert()"
3
+ sidebarTitle: "assert"
4
+ description: "Make AI-powered assertions about screen state"
5
+ icon: "check-circle"
6
+ ---
7
+
8
+ ## Overview
9
+
10
+ Make AI-powered assertions about the current screen state using natural language. The AI analyzes the screen and verifies that your assertion is true.
11
+
12
+ ## Syntax
13
+
14
+ ```javascript
15
+ await testdriver.assert(assertion)
16
+ ```
17
+
18
+ ## Parameters
19
+
20
+ <ParamField path="assertion" type="string" required>
21
+ Natural language description of what should be true
22
+ </ParamField>
23
+
24
+ ## Returns
25
+
26
+ `Promise<boolean>` - `true` if assertion passes, throws error if assertion fails
27
+
28
+ ## Examples
29
+
30
+ ### Basic Assertions
31
+
32
+ ```javascript
33
+ // Verify page elements
34
+ await testdriver.assert('the login page is displayed');
35
+ await testdriver.assert('submit button is visible');
36
+ await testdriver.assert('error message is shown');
37
+
38
+ // Verify text content
39
+ await testdriver.assert('the page title is "Welcome"');
40
+ await testdriver.assert('username field contains "john.doe"');
41
+ await testdriver.assert('success message says "Account created"');
42
+
43
+ // Verify states
44
+ await testdriver.assert('the form is empty');
45
+ await testdriver.assert('the checkbox is checked');
46
+ await testdriver.assert('the dropdown shows "United States"');
47
+
48
+ // Verify visual appearance
49
+ await testdriver.assert('the button is blue');
50
+ await testdriver.assert('the loading spinner is displayed');
51
+ await testdriver.assert('the modal dialog is open');
52
+ ```
53
+
54
+ ## Best Practices
55
+
56
+ <Check>
57
+ **Be specific in assertions**
58
+
59
+ More specific assertions are more reliable:
60
+
61
+ ```javascript
62
+ // ❌ Too vague
63
+ await testdriver.assert('button is visible');
64
+
65
+ // ✅ Specific
66
+ await testdriver.assert('blue submit button is visible below the form');
67
+ ```
68
+ </Check>
69
+
70
+ <Check>
71
+ **Assert state changes**
72
+
73
+ Verify state before and after actions:
74
+
75
+ ```javascript
76
+ // Before
77
+ await testdriver.assert('cart is empty');
78
+
79
+ // Action
80
+ const addBtn = await testdriver.find('add to cart');
81
+ await addBtn.click();
82
+
83
+ // After
84
+ await testdriver.assert('cart contains 1 item');
85
+ ```
86
+ </Check>
87
+
88
+ <Check>
89
+ **Use with test framework assertions**
90
+
91
+ Combine AI assertions with traditional test assertions:
92
+
93
+ ```javascript
94
+ // AI assertion
95
+ const result = await testdriver.assert('success message is displayed');
96
+
97
+ // Framework assertion
98
+ expect(result).toBeTruthy();
99
+
100
+ // Extract for detailed comparison
101
+ const message = await testdriver.remember('the success message text');
102
+ expect(message).toContain('successfully');
103
+ ```
104
+ </Check>
105
+
106
+ ## Polling Assertions
107
+
108
+ For conditions that may take time to become true:
109
+
110
+ ```javascript
111
+ async function waitForAssertion(testdriver, assertion, timeout = 30000) {
112
+ const startTime = Date.now();
113
+
114
+ while (Date.now() - startTime < timeout) {
115
+ try {
116
+ await testdriver.assert(assertion);
117
+ return true; // Assertion passed
118
+ } catch (error) {
119
+ // Assertion failed, wait and retry
120
+ await new Promise(r => setTimeout(r, 1000));
121
+ }
122
+ }
123
+
124
+ throw new Error(`Assertion timeout: "${assertion}"`);
125
+ }
126
+
127
+ // Usage
128
+ await waitForAssertion(testdriver, 'page has finished loading', 30000);
129
+ await waitForAssertion(testdriver, 'results are displayed', 10000);
130
+ ```
131
+
132
+ ## Use Cases
133
+
134
+ <AccordionGroup>
135
+ <Accordion title="Form Validation">
136
+ ```javascript
137
+ // Try to submit empty form
138
+ const submitBtn = await testdriver.find('submit button');
139
+ await submitBtn.click();
140
+
141
+ // Verify validation errors
142
+ await testdriver.assert('email field shows "required" error');
143
+ await testdriver.assert('password field shows "required" error');
144
+
145
+ // Verify form not submitted
146
+ await testdriver.assert('still on the form page');
147
+ ```
148
+ </Accordion>
149
+
150
+ <Accordion title="Page Navigation">
151
+ ```javascript
152
+ const loginBtn = await testdriver.find('login button');
153
+ await loginBtn.click();
154
+
155
+ // Verify navigation
156
+ await testdriver.assert('user dashboard is displayed');
157
+ await testdriver.assert('welcome message shows user name');
158
+ await testdriver.assert('logout button is visible');
159
+ ```
160
+ </Accordion>
161
+
162
+ <Accordion title="Dynamic Content">
163
+ ```javascript
164
+ const loadBtn = await testdriver.find('load more button');
165
+ await loadBtn.click();
166
+
167
+ // Poll for content using helper
168
+ await waitForAssertion(testdriver, 'more than 10 items are shown', 10000);
169
+ await testdriver.assert('load more button is still visible');
170
+ ```
171
+ </Accordion>
172
+
173
+ <Accordion title="Visual States">
174
+ ```javascript
175
+ // Verify hover effect
176
+ const button = await testdriver.find('primary button');
177
+ await button.hover();
178
+
179
+ await testdriver.assert('button background is darker');
180
+
181
+ // Verify button is enabled
182
+ await testdriver.assert('submit button is enabled');
183
+ ```
184
+ </Accordion>
185
+
186
+ <Accordion title="Multi-Step Workflows">
187
+ ```javascript
188
+ // Step 1
189
+ await testdriver.assert('step 1 is active');
190
+ const nextBtn = await testdriver.find('next button');
191
+ await nextBtn.click();
192
+
193
+ // Step 2
194
+ await testdriver.assert('step 2 is active');
195
+ await testdriver.assert('step 1 is completed');
196
+ await nextBtn.click();
197
+
198
+ // Step 3
199
+ await testdriver.assert('step 3 is active');
200
+ await testdriver.assert('step 2 is completed');
201
+ ```
202
+ </Accordion>
203
+ </AccordionGroup>
204
+
205
+ ## Complete Example
206
+
207
+ ```javascript
208
+ import { beforeAll, afterAll, describe, it, expect } from 'vitest';
209
+ import TestDriver from 'testdriverai';
210
+
211
+ describe('Assertions', () => {
212
+ let testdriver;
213
+
214
+ beforeAll(async () => {
215
+ client = new TestDriver(process.env.TD_API_KEY);
216
+ await testdriver.auth();
217
+ await testdriver.connect({ newSandbox: true });
218
+ });
219
+
220
+ afterAll(async () => {
221
+ await testdriver.disconnect();
222
+ });
223
+
224
+ it('should validate login flow', async () => {
225
+ await testdriver.focusApplication('Google Chrome');
226
+
227
+ // Verify initial state
228
+ await testdriver.assert('the login page is displayed');
229
+ await testdriver.assert('username field is empty');
230
+ await testdriver.assert('password field is empty');
231
+
232
+ // Fill form
233
+ const usernameField = await testdriver.find('username input');
234
+ await usernameField.click();
235
+ await testdriver.type('testuser');
236
+
237
+ await testdriver.pressKeys(['tab']);
238
+ await testdriver.type('password123');
239
+
240
+ // Verify fields filled
241
+ await testdriver.assert('username field contains "testuser"');
242
+ await testdriver.assert('password field is not empty');
243
+
244
+ // Submit
245
+ await testdriver.pressKeys(['enter']);
246
+
247
+ // Poll for success
248
+ await waitForAssertion(testdriver, 'user dashboard is displayed', 10000);
249
+
250
+ // Verify logged in state
251
+ await testdriver.assert('welcome message is shown');
252
+ await testdriver.assert('logout button is visible');
253
+
254
+ // Verify login page is gone
255
+ await testdriver.assert('login form is displayed', false, true);
256
+ });
257
+
258
+ it('should show validation errors', async () => {
259
+ // Try empty submission
260
+ const submitBtn = await testdriver.find('submit button');
261
+ await submitBtn.click();
262
+
263
+ // Verify multiple errors
264
+ const errors = [
265
+ 'email field shows error',
266
+ 'name field shows error',
267
+ 'password field shows error'
268
+ ];
269
+
270
+ for (const error of errors) {
271
+ const result = await testdriver.assert(error);
272
+ expect(result).toBeTruthy();
273
+ }
274
+
275
+ // Verify not submitted
276
+ await testdriver.assert('confirmation page is shown', false, true);
277
+ });
278
+ });
279
+ ```
280
+
281
+ ## Related Methods
282
+
283
+ - [`remember()`](/v7/api/remember) - Extract information for detailed assertions
284
+ - [`find()`](/v7/api/find) - Locate elements to verify
285
+ - [`ai()`](/v7/api/ai) - Complex AI-driven tasks
@@ -0,0 +1,403 @@
1
+ ---
2
+ title: "Assertions & Testing"
3
+ sidebarTitle: "Assertions"
4
+ description: "AI-powered assertions and validation"
5
+ icon: "check-circle"
6
+ ---
7
+
8
+ ## Overview
9
+
10
+ TestDriver provides AI-powered assertions that can verify complex visual states and conditions on screen without requiring explicit element locators or coordinates.
11
+
12
+ ## AI Assertions
13
+
14
+ ### assert()
15
+
16
+ Make an AI-powered assertion about the current screen state.
17
+
18
+ ```javascript
19
+ await testdriver.assert(assertion)
20
+ ```
21
+
22
+ **Parameters:**
23
+ - `assertion` (string) - Natural language description of what should be true
24
+
25
+ **Returns:** `Promise<boolean>` - `true` if assertion passes, throws error if fails
26
+
27
+ ### Basic Assertions
28
+
29
+ ```javascript
30
+ // Verify page elements
31
+ await testdriver.assert('the login page is displayed');
32
+ await testdriver.assert('submit button is visible');
33
+ await testdriver.assert('error message is shown');
34
+
35
+ // Verify text content
36
+ await testdriver.assert('the page title is "Welcome"');
37
+ await testdriver.assert('username field contains "john.doe"');
38
+ await testdriver.assert('success message says "Account created"');
39
+
40
+ // Verify states
41
+ await testdriver.assert('the form is empty');
42
+ await testdriver.assert('the checkbox is checked');
43
+ await testdriver.assert('the dropdown shows "United States"');
44
+
45
+ // Verify visual appearance
46
+ await testdriver.assert('the button is blue');
47
+ await testdriver.assert('the loading spinner is displayed');
48
+ await testdriver.assert('the modal dialog is open');
49
+ ```
50
+
51
+ ## Pattern: Polling Assertions
52
+
53
+ For conditions that may take time to become true, use polling:
54
+
55
+ ```javascript
56
+ // Poll until condition is true
57
+ async function waitForAssertion(testdriver, assertion, timeout = 30000) {
58
+ const startTime = Date.now();
59
+
60
+ while (Date.now() - startTime < timeout) {
61
+ try {
62
+ await testdriver.assert(assertion);
63
+ return true; // Assertion passed
64
+ } catch (error) {
65
+ // Assertion failed, wait and retry
66
+ await new Promise(resolve => setTimeout(resolve, 1000));
67
+ }
68
+ }
69
+
70
+ throw new Error(`Assertion timeout: "${assertion}"`);
71
+ }
72
+
73
+ // Usage
74
+ await waitForAssertion(testdriver, 'the page has finished loading', 30000);
75
+ await waitForAssertion(testdriver, 'results are displayed', 10000);
76
+ ```
77
+
78
+ ## Extracting Information
79
+
80
+ ### remember()
81
+
82
+ Extract and remember information from the screen using AI.
83
+
84
+ ```javascript
85
+ await testdriver.remember(description)
86
+ ```
87
+
88
+ **Parameters:**
89
+ - `description` (string) - What information to extract from the screen
90
+
91
+ **Returns:** `Promise<string>` - Extracted information
92
+
93
+ ### Examples
94
+
95
+ ```javascript
96
+ // Extract text from screen
97
+ const password = await testdriver.remember('the password displayed on screen');
98
+ const total = await testdriver.remember('the order total amount');
99
+ const errorMessage = await testdriver.remember('the error message text');
100
+
101
+ // Extract structured data
102
+ const email = await testdriver.remember('the email address in the confirmation');
103
+ const orderId = await testdriver.remember('the order ID number');
104
+ const phoneNumber = await testdriver.remember('the phone number');
105
+
106
+ // Use extracted data
107
+ const password = await testdriver.remember('the password for standard_user');
108
+ const passwordField = await testdriver.find('password input');
109
+ await passwordField.click();
110
+ await testdriver.type(password);
111
+ ```
112
+
113
+ ### Use Cases
114
+
115
+ **Dynamic Content:**
116
+ ```javascript
117
+ // Remember generated values
118
+ const confirmationCode = await testdriver.remember('the 6-digit confirmation code');
119
+ console.log('Code:', confirmationCode);
120
+
121
+ // Use it later in the test
122
+ await testdriver.type(confirmationCode);
123
+ ```
124
+
125
+ **Verification:**
126
+ ```javascript
127
+ // Extract and verify
128
+ const displayedTotal = await testdriver.remember('the cart total');
129
+ const expectedTotal = '$99.99';
130
+
131
+ expect(displayedTotal).toBe(expectedTotal);
132
+ ```
133
+
134
+ **Conditional Logic:**
135
+ ```javascript
136
+ // Extract state and make decisions
137
+ const status = await testdriver.remember('the order status');
138
+
139
+ if (status.includes('Pending')) {
140
+ // Wait for processing
141
+ await new Promise(resolve => setTimeout(resolve, 5000));
142
+ } else if (status.includes('Completed')) {
143
+ // Continue with test
144
+ const downloadButton = await testdriver.find('download receipt button');
145
+ await downloadButton.click();
146
+ }
147
+ ```
148
+
149
+ ## Testing Patterns
150
+
151
+ ### Test-Driven Assertions
152
+
153
+ Structure tests with clear arrange-act-assert pattern:
154
+
155
+ ```javascript
156
+ it('should display validation error for empty email', async () => {
157
+ // Arrange
158
+ await testdriver.focusApplication('Google Chrome');
159
+
160
+ // Act
161
+ const submitButton = await testdriver.find('submit button');
162
+ await submitButton.click();
163
+
164
+ // Assert
165
+ const result = await testdriver.assert('validation error says "Email is required"');
166
+ expect(result).toBeTruthy();
167
+ });
168
+ ```
169
+
170
+ ### Multi-Step Assertions
171
+
172
+ Verify state at each step of a workflow:
173
+
174
+ ```javascript
175
+ it('should complete checkout flow', async () => {
176
+ // Add item to cart
177
+ const addButton = await testdriver.find('add to cart button');
178
+ await addButton.click();
179
+
180
+ // Verify item added
181
+ await testdriver.assert('cart shows 1 item');
182
+
183
+ // Go to checkout
184
+ const cartIcon = await testdriver.find('cart icon');
185
+ await cartIcon.click();
186
+
187
+ // Verify cart page
188
+ await testdriver.assert('shopping cart page is displayed');
189
+ await testdriver.assert('cart contains the selected item');
190
+
191
+ // Proceed to checkout
192
+ const checkoutButton = await testdriver.find('proceed to checkout button');
193
+ await checkoutButton.click();
194
+
195
+ // Verify checkout page
196
+ await testdriver.assert('checkout form is displayed');
197
+ });
198
+ ```
199
+
200
+ ### Soft Assertions
201
+
202
+ Collect multiple assertion failures without stopping:
203
+
204
+ ```javascript
205
+ it('should validate all form fields', async () => {
206
+ const errors = [];
207
+
208
+ // Try multiple assertions
209
+ try {
210
+ await testdriver.assert('username field has error border');
211
+ } catch (e) {
212
+ errors.push('Username validation failed');
213
+ }
214
+
215
+ try {
216
+ await testdriver.assert('email field has error border');
217
+ } catch (e) {
218
+ errors.push('Email validation failed');
219
+ }
220
+
221
+ try {
222
+ await testdriver.assert('password field has error border');
223
+ } catch (e) {
224
+ errors.push('Password validation failed');
225
+ }
226
+
227
+ // Report all errors
228
+ if (errors.length > 0) {
229
+ throw new Error(`Validation errors:\n${errors.join('\n')}`);
230
+ }
231
+ });
232
+ ```
233
+
234
+ ## Complete Example
235
+
236
+ ```javascript
237
+ import { beforeAll, afterAll, describe, it, expect } from 'vitest';
238
+ import TestDriver from 'testdriverai';
239
+
240
+ describe('E2E Shopping Flow', () => {
241
+ let testdriver;
242
+
243
+ beforeAll(async () => {
244
+ client = new TestDriver(process.env.TD_API_KEY);
245
+ await testdriver.auth();
246
+ await testdriver.connect({ newSandbox: true });
247
+ });
248
+
249
+ afterAll(async () => {
250
+ await testdriver.disconnect();
251
+ });
252
+
253
+ it('should complete a purchase', async () => {
254
+ await testdriver.focusApplication('Google Chrome');
255
+
256
+ // Verify initial state
257
+ await testdriver.assert('the product listing page is displayed');
258
+ await testdriver.assert('cart is empty');
259
+
260
+ // Add product to cart
261
+ const addButton = await testdriver.find('add to cart button for first product');
262
+ await addButton.click();
263
+
264
+ // Verify cart updated
265
+ await testdriver.assert('cart badge shows 1 item');
266
+
267
+ // Remember product details
268
+ const productName = await testdriver.remember('the name of the product just added');
269
+ const price = await testdriver.remember('the price of the product');
270
+
271
+ console.log(`Added ${productName} at ${price}`);
272
+
273
+ // Go to cart
274
+ const cartIcon = await testdriver.find('shopping cart icon');
275
+ await cartIcon.click();
276
+
277
+ // Verify cart contents
278
+ await testdriver.assert(`cart contains ${productName}`);
279
+ await testdriver.assert(`cart total is ${price}`);
280
+
281
+ // Proceed to checkout
282
+ const checkoutButton = await testdriver.find('checkout button');
283
+ await checkoutButton.click();
284
+
285
+ // Fill shipping info
286
+ const emailField = await testdriver.find('email input');
287
+ await emailField.click();
288
+ await testdriver.type('test@example.com');
289
+
290
+ await testdriver.pressKeys(['tab']);
291
+ await testdriver.type('John Doe');
292
+
293
+ await testdriver.pressKeys(['tab']);
294
+ await testdriver.type('123 Main St');
295
+
296
+ // Submit order
297
+ const submitButton = await testdriver.find('place order button');
298
+ await submitButton.click();
299
+
300
+ // Verify success
301
+ await testdriver.assert('order confirmation page is displayed');
302
+ await testdriver.assert('success message is shown');
303
+
304
+ // Extract order number
305
+ const orderNumber = await testdriver.remember('the order number');
306
+ console.log('Order placed:', orderNumber);
307
+ expect(orderNumber).toBeTruthy();
308
+ });
309
+
310
+ it('should show validation errors', async () => {
311
+ // Try to submit empty form
312
+ const submitButton = await testdriver.find('submit button');
313
+ await submitButton.click();
314
+
315
+ // Verify multiple validation errors
316
+ await testdriver.assert('email field shows "required" error');
317
+ await testdriver.assert('name field shows "required" error');
318
+ await testdriver.assert('address field shows "required" error');
319
+
320
+ // Verify form not submitted
321
+ await testdriver.assert('still on checkout page', false, false);
322
+ await testdriver.assert('order confirmation page is shown', false, true); // Inverted
323
+ });
324
+ });
325
+ ```
326
+
327
+ ## Best Practices
328
+
329
+ <AccordionGroup>
330
+ <Accordion title="Be specific in assertions">
331
+ More specific assertions are more reliable:
332
+
333
+ ```javascript
334
+ // ❌ Too vague
335
+ await testdriver.assert('button is visible');
336
+
337
+ // ✅ Specific
338
+ await testdriver.assert('blue submit button is visible below the form');
339
+ ```
340
+ </Accordion>
341
+
342
+ <Accordion title="Assert state changes">
343
+ Verify state before and after actions:
344
+
345
+ ```javascript
346
+ // Before
347
+ await testdriver.assert('cart is empty');
348
+
349
+ // Action
350
+ const addButton = await testdriver.find('add to cart');
351
+ await addButton.click();
352
+
353
+ // After
354
+ await testdriver.assert('cart contains 1 item');
355
+ ```
356
+ </Accordion>
357
+
358
+ <Accordion title="Use polling for dynamic content">
359
+ Some assertions may need time to become true:
360
+
361
+ ```javascript
362
+ async function waitFor(assertion, timeout = 10000) {
363
+ const start = Date.now();
364
+ while (Date.now() - start < timeout) {
365
+ try {
366
+ await testdriver.assert(assertion);
367
+ return;
368
+ } catch {
369
+ await new Promise(r => setTimeout(r, 1000));
370
+ }
371
+ }
372
+ throw new Error(`Timeout: ${assertion}`);
373
+ }
374
+
375
+ await waitFor('loading complete');
376
+ ```
377
+ </Accordion>
378
+
379
+ <Accordion title="Extract before comparing">
380
+ Use `remember()` to extract values for detailed comparisons:
381
+
382
+ ```javascript
383
+ const actualTotal = await testdriver.remember('the cart total');
384
+ const expectedTotal = '$99.99';
385
+
386
+ expect(actualTotal).toBe(expectedTotal);
387
+ ```
388
+ </Accordion>
389
+
390
+ <Accordion title="Combine with traditional assertions">
391
+ Mix AI assertions with framework assertions for comprehensive testing:
392
+
393
+ ```javascript
394
+ // AI assertion for UI state
395
+ await testdriver.assert('success message is displayed');
396
+
397
+ // Extract and use traditional assertion for exact values
398
+ const message = await testdriver.remember('the success message text');
399
+ expect(message).toContain('successfully');
400
+ expect(message).toMatch(/order #\d+/i);
401
+ ```
402
+ </Accordion>
403
+ </AccordionGroup>