testdriverai 6.2.2 → 7.0.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 (264) 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/MIGRATION.md +389 -0
  5. package/PLUGIN_MIGRATION.md +222 -0
  6. package/PROMPT_CACHE.md +200 -0
  7. package/SDK_LOGGING.md +222 -0
  8. package/SDK_MIGRATION.md +474 -0
  9. package/SDK_README.md +1122 -0
  10. package/{testdriver → _testdriver}/acceptance/drag-and-drop.yaml +2 -2
  11. package/{testdriver → _testdriver}/acceptance/snippets/login.yaml +1 -1
  12. package/_testdriver/examples/desktop/lifecycle/prerun.yaml +0 -0
  13. package/{testdriver → _testdriver}/examples/web/lifecycle/prerun.yaml +6 -1
  14. package/{testdriver → _testdriver}/lifecycle/postrun.yaml +3 -2
  15. package/_testdriver/lifecycle/prerun.yaml +15 -0
  16. package/{testdriver → _testdriver}/lifecycle/provision.yaml +7 -2
  17. package/agent/index.js +258 -68
  18. package/agent/interface.js +15 -0
  19. package/agent/lib/cache.js +142 -0
  20. package/agent/lib/commander.js +1 -39
  21. package/agent/lib/commands.js +143 -188
  22. package/agent/lib/redraw.js +6 -3
  23. package/agent/lib/sandbox.js +19 -5
  24. package/agent/lib/sdk.js +1 -0
  25. package/agent/lib/system.js +0 -3
  26. package/agent/lib/validation.js +1 -7
  27. package/debug-locate-response.js +82 -0
  28. package/debug-screenshot-1763401388589.png +0 -0
  29. package/debugger/index.html +15 -4
  30. package/docs/ARCHITECTURE.md +424 -0
  31. package/docs/AWESOME_LOGS_QUICK_REF.md +100 -0
  32. package/docs/QUICK_START_TEST_RECORDING.md +215 -0
  33. package/docs/SDK_AWESOME_LOGS.md +468 -0
  34. package/docs/TEST_RECORDING.md +388 -0
  35. package/docs/docs.json +232 -152
  36. package/docs/sdk-browser-rendering.md +167 -0
  37. package/docs/v6/getting-started/self-hosting.mdx +407 -0
  38. package/docs/{guide → v6/guide}/dashcam.mdx +1 -1
  39. package/docs/{guide → v6/guide}/environment-variables.mdx +4 -5
  40. package/docs/{guide → v6/guide}/lifecycle.mdx +1 -1
  41. package/docs/v6/overview/comparison.mdx +101 -0
  42. package/docs/v7/README.md +135 -0
  43. package/docs/v7/api/ai.mdx +205 -0
  44. package/docs/v7/api/assert.mdx +285 -0
  45. package/docs/v7/api/assertions.mdx +403 -0
  46. package/docs/v7/api/click.mdx +287 -0
  47. package/docs/v7/api/client.mdx +322 -0
  48. package/docs/v7/api/elements.mdx +479 -0
  49. package/docs/v7/api/exec.mdx +346 -0
  50. package/docs/v7/api/find.mdx +316 -0
  51. package/docs/v7/api/focusApplication.mdx +294 -0
  52. package/docs/v7/api/hover.mdx +279 -0
  53. package/docs/v7/api/pressKeys.mdx +349 -0
  54. package/docs/v7/api/sandbox.mdx +404 -0
  55. package/docs/v7/api/scroll.mdx +300 -0
  56. package/docs/v7/api/type.mdx +314 -0
  57. package/docs/v7/commands/assert.mdx +45 -0
  58. package/docs/v7/commands/exec.mdx +282 -0
  59. package/docs/v7/commands/focus-application.mdx +44 -0
  60. package/docs/v7/commands/hover-image.mdx +69 -0
  61. package/docs/v7/commands/hover-text.mdx +47 -0
  62. package/docs/v7/commands/if.mdx +53 -0
  63. package/docs/v7/commands/match-image.mdx +67 -0
  64. package/docs/v7/commands/press-keys.mdx +87 -0
  65. package/docs/v7/commands/remember.mdx +49 -0
  66. package/docs/v7/commands/run.mdx +44 -0
  67. package/docs/v7/commands/scroll-until-image.mdx +66 -0
  68. package/docs/v7/commands/scroll-until-text.mdx +60 -0
  69. package/docs/v7/commands/scroll.mdx +69 -0
  70. package/docs/v7/commands/type.mdx +45 -0
  71. package/docs/v7/commands/wait-for-image.mdx +54 -0
  72. package/docs/v7/commands/wait-for-text.mdx +48 -0
  73. package/docs/v7/commands/wait.mdx +45 -0
  74. package/docs/v7/getting-started/quickstart.mdx +199 -0
  75. package/docs/v7/guides/migration.mdx +562 -0
  76. package/docs/{getting-started → v7/guides}/self-hosting.mdx +11 -12
  77. package/docs/v7/playwright.mdx +342 -0
  78. package/eslint.config.js +19 -1
  79. package/examples/run-tests-with-recording.sh +70 -0
  80. package/examples/screenshot-example.js +63 -0
  81. package/examples/sdk-awesome-logs-demo.js +177 -0
  82. package/examples/sdk-cache-thresholds.js +96 -0
  83. package/examples/sdk-element-properties.js +155 -0
  84. package/examples/sdk-simple-example.js +65 -0
  85. package/examples/test-recording-example.test.js +166 -0
  86. package/interfaces/cli/lib/base.js +10 -4
  87. package/interfaces/logger.js +2 -1
  88. package/interfaces/shared-test-state.mjs +69 -0
  89. package/interfaces/vitest-plugin.mjs +744 -0
  90. package/mcp-server/AI_GUIDELINES.md +57 -0
  91. package/package.json +18 -5
  92. package/schema.json +8 -29
  93. package/scripts/view-test-results.mjs +96 -0
  94. package/sdk-log-formatter.js +714 -0
  95. package/sdk.d.ts +735 -0
  96. package/sdk.js +1906 -0
  97. package/{.github/workflows/self-hosted.yml → self-hosted.yml} +13 -4
  98. package/setup/aws/cloudformation.yaml +9 -2
  99. package/test/mcp-example-test.yaml +27 -0
  100. package/test-find-api.js +73 -0
  101. package/test-prompt-cache.js +96 -0
  102. package/test-sandbox-render.js +28 -0
  103. package/test-sdk-methods.js +15 -0
  104. package/test-sdk-refactor.js +53 -0
  105. package/test-stack-trace.mjs +57 -0
  106. package/testdriver/acceptance-sdk/QUICK_REFERENCE.md +61 -0
  107. package/testdriver/acceptance-sdk/README.md +128 -0
  108. package/testdriver/acceptance-sdk/TEST_REPORTING.md +245 -0
  109. package/testdriver/acceptance-sdk/assert.test.mjs +44 -0
  110. package/testdriver/acceptance-sdk/drag-and-drop.test.mjs +70 -0
  111. package/testdriver/acceptance-sdk/element-not-found.test.mjs +38 -0
  112. package/testdriver/acceptance-sdk/exec-js.test.mjs +55 -0
  113. package/testdriver/acceptance-sdk/exec-output.test.mjs +71 -0
  114. package/testdriver/acceptance-sdk/exec-pwsh.test.mjs +69 -0
  115. package/testdriver/acceptance-sdk/focus-window.test.mjs +48 -0
  116. package/testdriver/acceptance-sdk/formatted-logging.test.mjs +41 -0
  117. package/testdriver/acceptance-sdk/hover-image.test.mjs +43 -0
  118. package/testdriver/acceptance-sdk/hover-text-with-description.test.mjs +50 -0
  119. package/testdriver/acceptance-sdk/hover-text.test.mjs +41 -0
  120. package/testdriver/acceptance-sdk/match-image.test.mjs +48 -0
  121. package/testdriver/acceptance-sdk/press-keys.test.mjs +64 -0
  122. package/testdriver/acceptance-sdk/prompt.test.mjs +45 -0
  123. package/testdriver/acceptance-sdk/scroll-keyboard.test.mjs +52 -0
  124. package/testdriver/acceptance-sdk/scroll-until-image.test.mjs +51 -0
  125. package/testdriver/acceptance-sdk/scroll-until-text.test.mjs +42 -0
  126. package/testdriver/acceptance-sdk/scroll.test.mjs +50 -0
  127. package/testdriver/acceptance-sdk/setup/globalTeardown.mjs +11 -0
  128. package/testdriver/acceptance-sdk/setup/lifecycleHelpers.mjs +239 -0
  129. package/testdriver/acceptance-sdk/setup/testHelpers.mjs +648 -0
  130. package/testdriver/acceptance-sdk/setup/vitestSetup.mjs +40 -0
  131. package/testdriver/acceptance-sdk/type-checking-demo.js +49 -0
  132. package/testdriver/acceptance-sdk/type.test.mjs +84 -0
  133. package/verify-element-api.js +89 -0
  134. package/verify-types.js +0 -0
  135. package/vitest.config.example.js +19 -0
  136. package/vitest.config.mjs +65 -0
  137. package/vitest.config.mjs.bak +44 -0
  138. package/.github/workflows/acceptance-v6.yml +0 -169
  139. package/docs/overview/comparison.mdx +0 -82
  140. package/testdriver/lifecycle/prerun.yaml +0 -17
  141. /package/{testdriver/examples/desktop/lifecycle/prerun.yaml → .env.example} +0 -0
  142. /package/{testdriver → _testdriver}/acceptance/assert.yaml +0 -0
  143. /package/{testdriver → _testdriver}/acceptance/dashcam.yaml +0 -0
  144. /package/{testdriver → _testdriver}/acceptance/embed.yaml +0 -0
  145. /package/{testdriver → _testdriver}/acceptance/exec-js.yaml +0 -0
  146. /package/{testdriver → _testdriver}/acceptance/exec-output.yaml +0 -0
  147. /package/{testdriver → _testdriver}/acceptance/exec-shell.yaml +0 -0
  148. /package/{testdriver → _testdriver}/acceptance/focus-window.yaml +0 -0
  149. /package/{testdriver → _testdriver}/acceptance/hover-image.yaml +0 -0
  150. /package/{testdriver → _testdriver}/acceptance/hover-text-with-description.yaml +0 -0
  151. /package/{testdriver → _testdriver}/acceptance/hover-text.yaml +0 -0
  152. /package/{testdriver → _testdriver}/acceptance/if-else.yaml +0 -0
  153. /package/{testdriver → _testdriver}/acceptance/match-image.yaml +0 -0
  154. /package/{testdriver → _testdriver}/acceptance/press-keys.yaml +0 -0
  155. /package/{testdriver → _testdriver}/acceptance/prompt.yaml +0 -0
  156. /package/{testdriver → _testdriver}/acceptance/remember.yaml +0 -0
  157. /package/{testdriver → _testdriver}/acceptance/screenshots/cart.png +0 -0
  158. /package/{testdriver → _testdriver}/acceptance/scroll-keyboard.yaml +0 -0
  159. /package/{testdriver → _testdriver}/acceptance/scroll-until-image.yaml +0 -0
  160. /package/{testdriver → _testdriver}/acceptance/scroll-until-text.yaml +0 -0
  161. /package/{testdriver → _testdriver}/acceptance/scroll.yaml +0 -0
  162. /package/{testdriver → _testdriver}/acceptance/snippets/match-cart.yaml +0 -0
  163. /package/{testdriver → _testdriver}/acceptance/type.yaml +0 -0
  164. /package/{testdriver → _testdriver}/behavior/failure.yaml +0 -0
  165. /package/{testdriver → _testdriver}/behavior/hover-text.yaml +0 -0
  166. /package/{testdriver → _testdriver}/behavior/lifecycle/postrun.yaml +0 -0
  167. /package/{testdriver → _testdriver}/behavior/lifecycle/prerun.yaml +0 -0
  168. /package/{testdriver → _testdriver}/behavior/lifecycle/provision.yaml +0 -0
  169. /package/{testdriver → _testdriver}/behavior/secrets.yaml +0 -0
  170. /package/{testdriver → _testdriver}/edge-cases/dashcam-chrome.yaml +0 -0
  171. /package/{testdriver → _testdriver}/edge-cases/exec-pwsh-multiline.yaml +0 -0
  172. /package/{testdriver → _testdriver}/edge-cases/js-exception.yaml +0 -0
  173. /package/{testdriver → _testdriver}/edge-cases/js-promise.yaml +0 -0
  174. /package/{testdriver → _testdriver}/edge-cases/lifecycle/postrun.yaml +0 -0
  175. /package/{testdriver → _testdriver}/edge-cases/prompt-in-middle.yaml +0 -0
  176. /package/{testdriver → _testdriver}/edge-cases/prompt-nested.yaml +0 -0
  177. /package/{testdriver → _testdriver}/edge-cases/success-test.yaml +0 -0
  178. /package/{testdriver → _testdriver}/examples/android/example.yaml +0 -0
  179. /package/{testdriver → _testdriver}/examples/android/lifecycle/postrun.yaml +0 -0
  180. /package/{testdriver → _testdriver}/examples/android/lifecycle/provision.yaml +0 -0
  181. /package/{testdriver → _testdriver}/examples/android/readme.md +0 -0
  182. /package/{testdriver → _testdriver}/examples/chrome-extension/lifecycle/provision.yaml +0 -0
  183. /package/{testdriver → _testdriver}/examples/desktop/lifecycle/provision.yaml +0 -0
  184. /package/{testdriver → _testdriver}/examples/vscode-extension/lifecycle/provision.yaml +0 -0
  185. /package/{testdriver → _testdriver}/examples/web/lifecycle/postrun.yaml +0 -0
  186. /package/docs/{account → v6/account}/dashboard.mdx +0 -0
  187. /package/docs/{account → v6/account}/enterprise.mdx +0 -0
  188. /package/docs/{account → v6/account}/pricing.mdx +0 -0
  189. /package/docs/{account → v6/account}/projects.mdx +0 -0
  190. /package/docs/{account → v6/account}/team.mdx +0 -0
  191. /package/docs/{action → v6/action}/ami.mdx +0 -0
  192. /package/docs/{action → v6/action}/performance.mdx +0 -0
  193. /package/docs/{action → v6/action}/secrets.mdx +0 -0
  194. /package/docs/{apps → v6/apps}/chrome-extensions.mdx +0 -0
  195. /package/docs/{apps → v6/apps}/desktop-apps.mdx +0 -0
  196. /package/docs/{apps → v6/apps}/mobile-apps.mdx +0 -0
  197. /package/docs/{apps → v6/apps}/static-websites.mdx +0 -0
  198. /package/docs/{apps → v6/apps}/tauri-apps.mdx +0 -0
  199. /package/docs/{bugs → v6/bugs}/jira.mdx +0 -0
  200. /package/docs/{cli → v6/cli}/overview.mdx +0 -0
  201. /package/docs/{commands → v6/commands}/assert.mdx +0 -0
  202. /package/docs/{commands → v6/commands}/exec.mdx +0 -0
  203. /package/docs/{commands → v6/commands}/focus-application.mdx +0 -0
  204. /package/docs/{commands → v6/commands}/hover-image.mdx +0 -0
  205. /package/docs/{commands → v6/commands}/hover-text.mdx +0 -0
  206. /package/docs/{commands → v6/commands}/if.mdx +0 -0
  207. /package/docs/{commands → v6/commands}/match-image.mdx +0 -0
  208. /package/docs/{commands → v6/commands}/press-keys.mdx +0 -0
  209. /package/docs/{commands → v6/commands}/remember.mdx +0 -0
  210. /package/docs/{commands → v6/commands}/run.mdx +0 -0
  211. /package/docs/{commands → v6/commands}/scroll-until-image.mdx +0 -0
  212. /package/docs/{commands → v6/commands}/scroll-until-text.mdx +0 -0
  213. /package/docs/{commands → v6/commands}/scroll.mdx +0 -0
  214. /package/docs/{commands → v6/commands}/type.mdx +0 -0
  215. /package/docs/{commands → v6/commands}/wait-for-image.mdx +0 -0
  216. /package/docs/{commands → v6/commands}/wait-for-text.mdx +0 -0
  217. /package/docs/{commands → v6/commands}/wait.mdx +0 -0
  218. /package/docs/{exporting → v6/exporting}/junit.mdx +0 -0
  219. /package/docs/{exporting → v6/exporting}/playwright.mdx +0 -0
  220. /package/docs/{features → v6/features}/auto-healing.mdx +0 -0
  221. /package/docs/{features → v6/features}/generation.mdx +0 -0
  222. /package/docs/{features → v6/features}/parallel-testing.mdx +0 -0
  223. /package/docs/{features → v6/features}/reusable-snippets.mdx +0 -0
  224. /package/docs/{features → v6/features}/selectorless.mdx +0 -0
  225. /package/docs/{features → v6/features}/visual-assertions.mdx +0 -0
  226. /package/docs/{getting-started → v6/getting-started}/ci.mdx +0 -0
  227. /package/docs/{getting-started → v6/getting-started}/cli.mdx +0 -0
  228. /package/docs/{getting-started → v6/getting-started}/editing.mdx +0 -0
  229. /package/docs/{getting-started → v6/getting-started}/playwright.mdx +0 -0
  230. /package/docs/{getting-started → v6/getting-started}/running.mdx +0 -0
  231. /package/docs/{getting-started → v6/getting-started}/vscode.mdx +0 -0
  232. /package/docs/{guide → v6/guide}/assertions.mdx +0 -0
  233. /package/docs/{guide → v6/guide}/authentication.mdx +0 -0
  234. /package/docs/{guide → v6/guide}/code.mdx +0 -0
  235. /package/docs/{guide → v6/guide}/locating.mdx +0 -0
  236. /package/docs/{guide → v6/guide}/protips.mdx +0 -0
  237. /package/docs/{guide → v6/guide}/variables.mdx +0 -0
  238. /package/docs/{guide → v6/guide}/waiting.mdx +0 -0
  239. /package/docs/{importing → v6/importing}/csv.mdx +0 -0
  240. /package/docs/{importing → v6/importing}/gherkin.mdx +0 -0
  241. /package/docs/{importing → v6/importing}/jira.mdx +0 -0
  242. /package/docs/{importing → v6/importing}/testrail.mdx +0 -0
  243. /package/docs/{integrations → v6/integrations}/electron.mdx +0 -0
  244. /package/docs/{integrations → v6/integrations}/netlify.mdx +0 -0
  245. /package/docs/{integrations → v6/integrations}/vercel.mdx +0 -0
  246. /package/docs/{interactive → v6/interactive}/explore.mdx +0 -0
  247. /package/docs/{interactive → v6/interactive}/run.mdx +0 -0
  248. /package/docs/{interactive → v6/interactive}/save.mdx +0 -0
  249. /package/docs/{overview → v6/overview}/faq.mdx +0 -0
  250. /package/docs/{overview → v6/overview}/performance.mdx +0 -0
  251. /package/docs/{overview → v6/overview}/quickstart.mdx +0 -0
  252. /package/docs/{overview → v6/overview}/what-is-testdriver.mdx +0 -0
  253. /package/docs/{scenarios → v6/scenarios}/ai-chatbot.mdx +0 -0
  254. /package/docs/{scenarios → v6/scenarios}/cookie-banner.mdx +0 -0
  255. /package/docs/{scenarios → v6/scenarios}/file-upload.mdx +0 -0
  256. /package/docs/{scenarios → v6/scenarios}/form-filling.mdx +0 -0
  257. /package/docs/{scenarios → v6/scenarios}/log-in.mdx +0 -0
  258. /package/docs/{scenarios → v6/scenarios}/pdf-generation.mdx +0 -0
  259. /package/docs/{scenarios → v6/scenarios}/spell-check.mdx +0 -0
  260. /package/docs/{security → v6/security}/action.mdx +0 -0
  261. /package/docs/{security → v6/security}/agent.mdx +0 -0
  262. /package/docs/{security → v6/security}/platform.mdx +0 -0
  263. /package/docs/{tutorials → v6/tutorials}/advanced-test.mdx +0 -0
  264. /package/docs/{tutorials → v6/tutorials}/basic-test.mdx +0 -0
@@ -0,0 +1,205 @@
1
+ ---
2
+ title: "ai()"
3
+ sidebarTitle: "ai"
4
+ description: "Execute natural language tasks using AI"
5
+ icon: "wand-magic-sparkles"
6
+ ---
7
+
8
+ ## Overview
9
+
10
+ The `ai()` method allows you to execute complex tasks using natural language descriptions. TestDriver's AI will figure out the steps needed to accomplish the task.
11
+
12
+ ## Syntax
13
+
14
+ ```javascript
15
+ await testdriver.ai(task, options)
16
+ ```
17
+
18
+ ## Parameters
19
+
20
+ <ParamField path="task" type="string" required>
21
+ Natural language description of what to do
22
+ </ParamField>
23
+
24
+ <ParamField path="options" type="object">
25
+ Execution options
26
+
27
+ <Expandable title="properties">
28
+ <ParamField path="validateAndLoop" type="boolean" default="false">
29
+ Whether to validate completion and retry if incomplete
30
+ </ParamField>
31
+ </Expandable>
32
+ </ParamField>
33
+
34
+ ## Returns
35
+
36
+ `Promise<string | void>` - Final AI response if `validateAndLoop` is true
37
+
38
+ ## Examples
39
+
40
+ ### Basic Usage
41
+
42
+ ```javascript
43
+ // Simple task execution
44
+ await testdriver.ai('Click the submit button');
45
+
46
+ // Complex multi-step task
47
+ await testdriver.ai('Fill out the contact form and submit it');
48
+
49
+ // Navigation task
50
+ await testdriver.ai('Go to the settings page and enable notifications');
51
+ ```
52
+
53
+ ### With Validation
54
+
55
+ ```javascript
56
+ // AI will verify the task completed successfully
57
+ const result = await testdriver.ai('Complete the checkout process', {
58
+ validateAndLoop: true
59
+ });
60
+
61
+ console.log('Task result:', result);
62
+ ```
63
+
64
+ ### Multi-Step Workflows
65
+
66
+ ```javascript
67
+ // The AI will break down complex tasks
68
+ await testdriver.ai('Search for "laptop", add the first result to cart, and proceed to checkout');
69
+
70
+ // UI exploration
71
+ await testdriver.ai('Find and click all menu items to explore the application');
72
+ ```
73
+
74
+ ## Use Cases
75
+
76
+ <AccordionGroup>
77
+ <Accordion title="Exploratory Testing">
78
+ Use AI to explore unfamiliar applications:
79
+
80
+ ```javascript
81
+ await testdriver.ai('Explore the main navigation menu');
82
+ await testdriver.ai('Try to find the user profile settings');
83
+ ```
84
+ </Accordion>
85
+
86
+ <Accordion title="Complex Workflows">
87
+ Let AI handle multi-step processes:
88
+
89
+ ```javascript
90
+ await testdriver.ai('Complete the multi-step registration form');
91
+ await testdriver.ai('Configure all the advanced settings to default values');
92
+ ```
93
+ </Accordion>
94
+
95
+ <Accordion title="Flexible Interactions">
96
+ When exact element locations aren't critical:
97
+
98
+ ```javascript
99
+ await testdriver.ai('Close any popup dialogs');
100
+ await testdriver.ai('Accept the cookie consent if it appears');
101
+ ```
102
+ </Accordion>
103
+ </AccordionGroup>
104
+
105
+ ## Best Practices
106
+
107
+ <Check>
108
+ **Be specific but flexible**: Give enough context without over-constraining the AI
109
+
110
+ ```javascript
111
+ // ✅ Good
112
+ await testdriver.ai('Add the first product to the shopping cart');
113
+
114
+ // ❌ Too vague
115
+ await testdriver.ai('do something');
116
+
117
+ // ❌ Too specific (defeats the purpose)
118
+ await testdriver.ai('click at coordinates 500, 300');
119
+ ```
120
+ </Check>
121
+
122
+ <Check>
123
+ **Use for exploration, not precision**: For critical assertions, use explicit methods
124
+
125
+ ```javascript
126
+ // Use AI for setup
127
+ await testdriver.ai('Navigate to the login page');
128
+
129
+ // Use explicit methods for critical steps
130
+ const usernameField = await testdriver.find('username input');
131
+ await usernameField.click();
132
+ await testdriver.type('testuser');
133
+
134
+ await testdriver.assert('login page is displayed');
135
+ ```
136
+ </Check>
137
+
138
+ <Warning>
139
+ **AI tasks may be slower**: The AI needs to analyze the screen and plan actions. For performance-critical tests, use explicit methods.
140
+ </Warning>
141
+
142
+ ## When to Use AI vs Explicit Methods
143
+
144
+ ### Use `ai()` when:
145
+ - Exploring unfamiliar applications
146
+ - Handling optional UI elements (popups, cookies, etc.)
147
+ - Prototyping tests quickly
148
+ - Tasks where exact steps may vary
149
+
150
+ ### Use explicit methods when:
151
+ - Performance is critical
152
+ - You need precise control
153
+ - Making assertions
154
+ - Debugging test failures
155
+ - Repetitive, predictable actions
156
+
157
+ ## Complete Example
158
+
159
+ ```javascript
160
+ import { beforeAll, afterAll, describe, it } from 'vitest';
161
+ import TestDriver from 'testdriverai';
162
+
163
+ describe('E-commerce Flow with AI', () => {
164
+ let testdriver;
165
+
166
+ beforeAll(async () => {
167
+ client = new TestDriver(process.env.TD_API_KEY);
168
+ await testdriver.auth();
169
+ await testdriver.connect({ newSandbox: true });
170
+ });
171
+
172
+ afterAll(async () => {
173
+ await testdriver.disconnect();
174
+ });
175
+
176
+ it('should complete shopping flow using AI assistance', async () => {
177
+ await testdriver.focusApplication('Google Chrome');
178
+
179
+ // Use AI for navigation and exploration
180
+ await testdriver.ai('Browse to the electronics section');
181
+ await testdriver.ai('Find and add a laptop to the cart');
182
+
183
+ // Use explicit methods for critical steps
184
+ const cartIcon = await testdriver.find('shopping cart icon');
185
+ await cartIcon.click();
186
+
187
+ const total = await testdriver.remember('the cart total amount');
188
+ console.log('Cart total:', total);
189
+
190
+ // Use AI for checkout flow
191
+ await testdriver.ai('Proceed to checkout and fill in shipping details', {
192
+ validateAndLoop: true
193
+ });
194
+
195
+ // Explicit assertion
196
+ await testdriver.assert('order confirmation page is displayed');
197
+ });
198
+ });
199
+ ```
200
+
201
+ ## Related Methods
202
+
203
+ - [`find()`](/v7/api/find) - Locate specific elements
204
+ - [`assert()`](/v7/api/assert) - Make assertions
205
+ - [`remember()`](/v7/api/remember) - Extract information
@@ -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