@testdriverai/agent 7.8.0-canary.10

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 (528) hide show
  1. package/.claude/settings.local.json +7 -0
  2. package/.env.example +4 -0
  3. package/.prettierignore +4 -0
  4. package/.prettierrc +1 -0
  5. package/CHANGELOG.md +953 -0
  6. package/README.md +81 -0
  7. package/agent/events.js +135 -0
  8. package/agent/index.js +2450 -0
  9. package/agent/interface.js +35 -0
  10. package/agent/lib/analytics.js +22 -0
  11. package/agent/lib/censorship.js +75 -0
  12. package/agent/lib/commander.js +246 -0
  13. package/agent/lib/commands.js +1684 -0
  14. package/agent/lib/config.js +60 -0
  15. package/agent/lib/generator.js +91 -0
  16. package/agent/lib/http.js +144 -0
  17. package/agent/lib/logger.js +56 -0
  18. package/agent/lib/outputs.js +29 -0
  19. package/agent/lib/parser.js +209 -0
  20. package/agent/lib/redraw.js +386 -0
  21. package/agent/lib/resources/cursor-2.png +0 -0
  22. package/agent/lib/sandbox.js +1104 -0
  23. package/agent/lib/sdk.js +633 -0
  24. package/agent/lib/session.js +25 -0
  25. package/agent/lib/source-mapper.js +342 -0
  26. package/agent/lib/subimage/index.js +77 -0
  27. package/agent/lib/subimage/opencv.js +69 -0
  28. package/agent/lib/system.js +204 -0
  29. package/agent/lib/theme.js +14 -0
  30. package/agent/lib/valid-version.js +21 -0
  31. package/agent/lib/validation.js +169 -0
  32. package/ai/.claude-plugin/plugin.json +9 -0
  33. package/ai/agents/testdriver.md +638 -0
  34. package/ai/skills/testdriver-ai/SKILL.md +204 -0
  35. package/ai/skills/testdriver-assert/SKILL.md +315 -0
  36. package/ai/skills/testdriver-aws-setup/SKILL.md +448 -0
  37. package/ai/skills/testdriver-cache/SKILL.md +221 -0
  38. package/ai/skills/testdriver-caching/SKILL.md +124 -0
  39. package/ai/skills/testdriver-captcha/SKILL.md +158 -0
  40. package/ai/skills/testdriver-ci-cd/SKILL.md +602 -0
  41. package/ai/skills/testdriver-click/SKILL.md +286 -0
  42. package/ai/skills/testdriver-client/SKILL.md +477 -0
  43. package/ai/skills/testdriver-cloud/SKILL.md +119 -0
  44. package/ai/skills/testdriver-customizing-devices/SKILL.md +319 -0
  45. package/ai/skills/testdriver-dashcam/SKILL.md +418 -0
  46. package/ai/skills/testdriver-debugging-with-screenshots/SKILL.md +401 -0
  47. package/ai/skills/testdriver-device-config/SKILL.md +317 -0
  48. package/ai/skills/testdriver-double-click/SKILL.md +102 -0
  49. package/ai/skills/testdriver-elements/SKILL.md +605 -0
  50. package/ai/skills/testdriver-enterprise/SKILL.md +114 -0
  51. package/ai/skills/testdriver-errors/SKILL.md +246 -0
  52. package/ai/skills/testdriver-events/SKILL.md +356 -0
  53. package/ai/skills/testdriver-examples/SKILL.md +7 -0
  54. package/ai/skills/testdriver-exec/SKILL.md +317 -0
  55. package/ai/skills/testdriver-find/SKILL.md +829 -0
  56. package/ai/skills/testdriver-focus-application/SKILL.md +293 -0
  57. package/ai/skills/testdriver-generating-tests/SKILL.md +36 -0
  58. package/ai/skills/testdriver-hover/SKILL.md +278 -0
  59. package/ai/skills/testdriver-locating-elements/SKILL.md +71 -0
  60. package/ai/skills/testdriver-making-assertions/SKILL.md +32 -0
  61. package/ai/skills/testdriver-mcp/SKILL.md +7 -0
  62. package/ai/skills/testdriver-mcp-workflow/SKILL.md +410 -0
  63. package/ai/skills/testdriver-mouse-down/SKILL.md +161 -0
  64. package/ai/skills/testdriver-mouse-up/SKILL.md +164 -0
  65. package/ai/skills/testdriver-parse/SKILL.md +236 -0
  66. package/ai/skills/testdriver-performing-actions/SKILL.md +54 -0
  67. package/ai/skills/testdriver-press-keys/SKILL.md +348 -0
  68. package/ai/skills/testdriver-provision/SKILL.md +331 -0
  69. package/ai/skills/testdriver-quickstart/SKILL.md +144 -0
  70. package/ai/skills/testdriver-redraw/SKILL.md +214 -0
  71. package/ai/skills/testdriver-reusable-code/SKILL.md +249 -0
  72. package/ai/skills/testdriver-right-click/SKILL.md +123 -0
  73. package/ai/skills/testdriver-running-tests/SKILL.md +185 -0
  74. package/ai/skills/testdriver-screenshot/SKILL.md +248 -0
  75. package/ai/skills/testdriver-screenshots/SKILL.md +184 -0
  76. package/ai/skills/testdriver-scroll/SKILL.md +335 -0
  77. package/ai/skills/testdriver-secrets/SKILL.md +115 -0
  78. package/ai/skills/testdriver-self-hosted/SKILL.md +65 -0
  79. package/ai/skills/testdriver-test-writer/SKILL.md +448 -0
  80. package/ai/skills/testdriver-testdriver/SKILL.md +628 -0
  81. package/ai/skills/testdriver-testdriver-mechanic/SKILL.md +165 -0
  82. package/ai/skills/testdriver-type/SKILL.md +357 -0
  83. package/ai/skills/testdriver-variables/SKILL.md +111 -0
  84. package/ai/skills/testdriver-wait/SKILL.md +50 -0
  85. package/ai/skills/testdriver-waiting-for-elements/SKILL.md +90 -0
  86. package/ai/skills/testdriver-what-is-testdriver/SKILL.md +54 -0
  87. package/bin/testdriverai.js +22 -0
  88. package/debugger/bg.png +0 -0
  89. package/debugger/icon.png +0 -0
  90. package/debugger/index.html +469 -0
  91. package/debugger/td.png +0 -0
  92. package/debugger/tray-buffered.png +0 -0
  93. package/debugger/tray.png +0 -0
  94. package/docs/GITHUB_COMMENTS.md +330 -0
  95. package/docs/GITHUB_COMMENTS_ANNOUNCEMENT.md +167 -0
  96. package/docs/QUICK-START-GITHUB-COMMENTS.md +84 -0
  97. package/docs/TEST-GITHUB-COMMENTS.md +129 -0
  98. package/docs/_data/examples-manifest.json +177 -0
  99. package/docs/_data/examples-manifest.schema.json +41 -0
  100. package/docs/_scripts/extract-example-urls.js +165 -0
  101. package/docs/_scripts/generate-examples.js +560 -0
  102. package/docs/_scripts/generate-skills.js +154 -0
  103. package/docs/_scripts/link-replacer.js +164 -0
  104. package/docs/_scripts/upload-docs-to-openai.js +284 -0
  105. package/docs/changelog.mdx +161 -0
  106. package/docs/claude-mcp-plugin.mdx +160 -0
  107. package/docs/docs.json +442 -0
  108. package/docs/github-integration-setup.md +266 -0
  109. package/docs/guide/best-practices-polling.mdx +174 -0
  110. package/docs/images/content/account/newprojectsettings.png +0 -0
  111. package/docs/images/content/account/projectpage.png +0 -0
  112. package/docs/images/content/account/projectreplays.png +0 -0
  113. package/docs/images/content/account/team-manage.png +0 -0
  114. package/docs/images/content/account/teampage.png +0 -0
  115. package/docs/images/content/extension/cursor.svg +1 -0
  116. package/docs/images/content/extension/vscode.svg +57 -0
  117. package/docs/images/content/extension/windsurf.svg +3 -0
  118. package/docs/images/content/parse/output.png +0 -0
  119. package/docs/images/content/self-hosted/launchtemplateid.png +0 -0
  120. package/docs/images/content/side-by-side.png +0 -0
  121. package/docs/images/content/vscode/ide-full.png +0 -0
  122. package/docs/images/content/vscode/running.png +0 -0
  123. package/docs/images/content/vscode/v7-chat.png +0 -0
  124. package/docs/images/content/vscode/v7-choose-agent.png +0 -0
  125. package/docs/images/content/vscode/v7-full.png +0 -0
  126. package/docs/images/content/vscode/v7-onboarding.png +0 -0
  127. package/docs/images/content/vscode/vscode-2-assert.png +0 -0
  128. package/docs/images/content/vscode/vscode-agent-preview.png +0 -0
  129. package/docs/images/content/vscode/vscode-copilot-ask.png +0 -0
  130. package/docs/images/content/vscode/vscode-file-creation.png +0 -0
  131. package/docs/images/content/vscode/vscode-install.png +0 -0
  132. package/docs/images/content/vscode/vscode-overview.png +0 -0
  133. package/docs/images/content/vscode/vscode-setup-walkthrough.png +0 -0
  134. package/docs/images/content/vscode/vscode-stopchat.png +0 -0
  135. package/docs/images/content/vscode/vscode-stoptest.png +0 -0
  136. package/docs/images/content/vscode/vscode-tdservice.png +0 -0
  137. package/docs/images/content/vscode/vscode-test-output.png +0 -0
  138. package/docs/images/content/vscode/vscode-testhistory.png +0 -0
  139. package/docs/images/content/vscode/vscode-testpane-runtests.png +0 -0
  140. package/docs/images/content/vscode/vscode-testpane.png +0 -0
  141. package/docs/images/template/dark.png +0 -0
  142. package/docs/images/template/icon.png +0 -0
  143. package/docs/images/template/light.png +0 -0
  144. package/docs/snippets/calendar-link.mdx +4 -0
  145. package/docs/snippets/gitignore-warning.mdx +7 -0
  146. package/docs/snippets/lifecycle-warning.mdx +6 -0
  147. package/docs/snippets/test-prereqs.mdx +12 -0
  148. package/docs/snippets/tests/assert-replay.mdx +7 -0
  149. package/docs/snippets/tests/assert-yaml.mdx +8 -0
  150. package/docs/snippets/tests/exec-js-replay.mdx +7 -0
  151. package/docs/snippets/tests/exec-js-yaml.mdx +32 -0
  152. package/docs/snippets/tests/exec-shell-replay.mdx +7 -0
  153. package/docs/snippets/tests/exec-shell-yaml.mdx +15 -0
  154. package/docs/snippets/tests/hover-image-replay.mdx +7 -0
  155. package/docs/snippets/tests/hover-image-yaml.mdx +17 -0
  156. package/docs/snippets/tests/hover-text-replay.mdx +7 -0
  157. package/docs/snippets/tests/hover-text-with-description-replay.mdx +7 -0
  158. package/docs/snippets/tests/hover-text-with-description-yaml.mdx +24 -0
  159. package/docs/snippets/tests/hover-text-yaml.mdx +14 -0
  160. package/docs/snippets/tests/match-image-replay.mdx +7 -0
  161. package/docs/snippets/tests/match-image-yaml.mdx +17 -0
  162. package/docs/snippets/tests/press-keys-replay.mdx +7 -0
  163. package/docs/snippets/tests/press-keys-yaml.mdx +36 -0
  164. package/docs/snippets/tests/remember-replay.mdx +7 -0
  165. package/docs/snippets/tests/remember-yaml.mdx +28 -0
  166. package/docs/snippets/tests/scroll-replay.mdx +7 -0
  167. package/docs/snippets/tests/scroll-until-image-replay.mdx +7 -0
  168. package/docs/snippets/tests/scroll-until-image-yaml.mdx +14 -0
  169. package/docs/snippets/tests/scroll-until-text-replay.mdx +7 -0
  170. package/docs/snippets/tests/scroll-until-text-yaml.mdx +17 -0
  171. package/docs/snippets/tests/scroll-yaml.mdx +30 -0
  172. package/docs/snippets/tests/type-repeated-replay.mdx +7 -0
  173. package/docs/snippets/tests/type-repeated-yaml.mdx +22 -0
  174. package/docs/snippets/tests/type-replay.mdx +7 -0
  175. package/docs/snippets/tests/type-yaml.mdx +28 -0
  176. package/docs/snippets/tests/wait-for-image-replay.mdx +7 -0
  177. package/docs/snippets/tests/wait-for-image-yaml.mdx +18 -0
  178. package/docs/snippets/tests/wait-for-text-replay.mdx +7 -0
  179. package/docs/snippets/tests/wait-for-text-yaml.mdx +18 -0
  180. package/docs/snippets/tests/wait-replay.mdx +7 -0
  181. package/docs/snippets/tests/wait-yaml.mdx +13 -0
  182. package/docs/styles.css +65 -0
  183. package/docs/v6/account/dashboard.mdx +16 -0
  184. package/docs/v6/account/enterprise.mdx +110 -0
  185. package/docs/v6/account/pricing.mdx +33 -0
  186. package/docs/v6/account/projects.mdx +33 -0
  187. package/docs/v6/account/team.mdx +35 -0
  188. package/docs/v6/action/ami.mdx +109 -0
  189. package/docs/v6/action/performance.mdx +105 -0
  190. package/docs/v6/action/secrets.mdx +93 -0
  191. package/docs/v6/apps/chrome-extensions.mdx +48 -0
  192. package/docs/v6/apps/desktop-apps.mdx +93 -0
  193. package/docs/v6/apps/mobile-apps.mdx +26 -0
  194. package/docs/v6/apps/static-websites.mdx +54 -0
  195. package/docs/v6/apps/tauri-apps.mdx +361 -0
  196. package/docs/v6/bugs/jira.mdx +232 -0
  197. package/docs/v6/cli/overview.mdx +66 -0
  198. package/docs/v6/commands/assert.mdx +45 -0
  199. package/docs/v6/commands/exec.mdx +276 -0
  200. package/docs/v6/commands/focus-application.mdx +44 -0
  201. package/docs/v6/commands/hover-image.mdx +69 -0
  202. package/docs/v6/commands/hover-text.mdx +47 -0
  203. package/docs/v6/commands/if.mdx +53 -0
  204. package/docs/v6/commands/match-image.mdx +67 -0
  205. package/docs/v6/commands/press-keys.mdx +87 -0
  206. package/docs/v6/commands/remember.mdx +49 -0
  207. package/docs/v6/commands/run.mdx +44 -0
  208. package/docs/v6/commands/scroll-until-image.mdx +66 -0
  209. package/docs/v6/commands/scroll-until-text.mdx +60 -0
  210. package/docs/v6/commands/scroll.mdx +69 -0
  211. package/docs/v6/commands/type.mdx +45 -0
  212. package/docs/v6/commands/wait-for-image.mdx +54 -0
  213. package/docs/v6/commands/wait-for-text.mdx +48 -0
  214. package/docs/v6/commands/wait.mdx +45 -0
  215. package/docs/v6/exporting/junit.mdx +218 -0
  216. package/docs/v6/exporting/playwright.mdx +197 -0
  217. package/docs/v6/features/auto-healing.mdx +144 -0
  218. package/docs/v6/features/generation.mdx +116 -0
  219. package/docs/v6/features/parallel-testing.mdx +151 -0
  220. package/docs/v6/features/reusable-snippets.mdx +131 -0
  221. package/docs/v6/features/selectorless.mdx +80 -0
  222. package/docs/v6/features/visual-assertions.mdx +139 -0
  223. package/docs/v6/getting-started/ci.mdx +146 -0
  224. package/docs/v6/getting-started/cli.mdx +91 -0
  225. package/docs/v6/getting-started/editing.mdx +100 -0
  226. package/docs/v6/getting-started/playwright.mdx +342 -0
  227. package/docs/v6/getting-started/running.mdx +48 -0
  228. package/docs/v6/getting-started/self-hosting.mdx +408 -0
  229. package/docs/v6/getting-started/vscode.mdx +88 -0
  230. package/docs/v6/guide/assertions.mdx +189 -0
  231. package/docs/v6/guide/authentication.mdx +136 -0
  232. package/docs/v6/guide/code.mdx +65 -0
  233. package/docs/v6/guide/dashcam.mdx +118 -0
  234. package/docs/v6/guide/environment-variables.mdx +26 -0
  235. package/docs/v6/guide/lifecycle.mdx +242 -0
  236. package/docs/v6/guide/locating.mdx +141 -0
  237. package/docs/v6/guide/protips.mdx +43 -0
  238. package/docs/v6/guide/variables.mdx +143 -0
  239. package/docs/v6/guide/waiting.mdx +130 -0
  240. package/docs/v6/importing/csv.mdx +196 -0
  241. package/docs/v6/importing/gherkin.mdx +143 -0
  242. package/docs/v6/importing/jira.mdx +164 -0
  243. package/docs/v6/importing/testrail.mdx +162 -0
  244. package/docs/v6/integrations/electron.mdx +146 -0
  245. package/docs/v6/integrations/netlify.mdx +100 -0
  246. package/docs/v6/integrations/vercel.mdx +125 -0
  247. package/docs/v6/interactive/explore.mdx +99 -0
  248. package/docs/v6/interactive/run.mdx +52 -0
  249. package/docs/v6/interactive/save.mdx +63 -0
  250. package/docs/v6/overview/comparison.mdx +101 -0
  251. package/docs/v6/overview/faq.mdx +162 -0
  252. package/docs/v6/overview/performance.mdx +52 -0
  253. package/docs/v6/overview/quickstart.mdx +137 -0
  254. package/docs/v6/overview/what-is-testdriver.mdx +85 -0
  255. package/docs/v6/scenarios/ai-chatbot.mdx +28 -0
  256. package/docs/v6/scenarios/cookie-banner.mdx +32 -0
  257. package/docs/v6/scenarios/file-upload.mdx +33 -0
  258. package/docs/v6/scenarios/form-filling.mdx +32 -0
  259. package/docs/v6/scenarios/log-in.mdx +75 -0
  260. package/docs/v6/scenarios/pdf-generation.mdx +25 -0
  261. package/docs/v6/scenarios/spell-check.mdx +22 -0
  262. package/docs/v6/security/action.mdx +84 -0
  263. package/docs/v6/security/agent.mdx +73 -0
  264. package/docs/v6/security/platform.mdx +77 -0
  265. package/docs/v6/tutorials/advanced-test.mdx +81 -0
  266. package/docs/v6/tutorials/basic-test.mdx +45 -0
  267. package/docs/v7/_drafts/agents.mdx +843 -0
  268. package/docs/v7/_drafts/architecture.mdx +399 -0
  269. package/docs/v7/_drafts/auto-cache-key.mdx +167 -0
  270. package/docs/v7/_drafts/awesome-logs-quick-ref.mdx +100 -0
  271. package/docs/v7/_drafts/best-practices.mdx +486 -0
  272. package/docs/v7/_drafts/caching-ai.mdx +215 -0
  273. package/docs/v7/_drafts/caching-selectors.mdx +424 -0
  274. package/docs/v7/_drafts/caching.mdx +366 -0
  275. package/docs/v7/_drafts/cli-to-sdk-migration.mdx +425 -0
  276. package/docs/v7/_drafts/commands/assert.mdx +45 -0
  277. package/docs/v7/_drafts/commands/exec.mdx +276 -0
  278. package/docs/v7/_drafts/commands/focus-application.mdx +44 -0
  279. package/docs/v7/_drafts/commands/hover-image.mdx +69 -0
  280. package/docs/v7/_drafts/commands/hover-text.mdx +47 -0
  281. package/docs/v7/_drafts/commands/if.mdx +53 -0
  282. package/docs/v7/_drafts/commands/match-image.mdx +67 -0
  283. package/docs/v7/_drafts/commands/press-keys.mdx +87 -0
  284. package/docs/v7/_drafts/commands/remember.mdx +49 -0
  285. package/docs/v7/_drafts/commands/run.mdx +44 -0
  286. package/docs/v7/_drafts/commands/scroll-until-image.mdx +66 -0
  287. package/docs/v7/_drafts/commands/scroll-until-text.mdx +60 -0
  288. package/docs/v7/_drafts/commands/scroll.mdx +69 -0
  289. package/docs/v7/_drafts/commands/type.mdx +45 -0
  290. package/docs/v7/_drafts/commands/wait-for-image.mdx +54 -0
  291. package/docs/v7/_drafts/commands/wait-for-text.mdx +48 -0
  292. package/docs/v7/_drafts/commands/wait.mdx +45 -0
  293. package/docs/v7/_drafts/configuration.mdx +378 -0
  294. package/docs/v7/_drafts/contributing.mdx +174 -0
  295. package/docs/v7/_drafts/core.mdx +458 -0
  296. package/docs/v7/_drafts/dashcam-title-feature.mdx +89 -0
  297. package/docs/v7/_drafts/debugging.mdx +349 -0
  298. package/docs/v7/_drafts/error-handling.mdx +501 -0
  299. package/docs/v7/_drafts/faq.mdx +393 -0
  300. package/docs/v7/_drafts/hooks.mdx +360 -0
  301. package/docs/v7/_drafts/init-command.mdx +95 -0
  302. package/docs/v7/_drafts/installation.mdx +420 -0
  303. package/docs/v7/_drafts/migration.mdx +562 -0
  304. package/docs/v7/_drafts/observable.mdx +604 -0
  305. package/docs/v7/_drafts/playwright.mdx +342 -0
  306. package/docs/v7/_drafts/plugin-migration.mdx +220 -0
  307. package/docs/v7/_drafts/powerful.mdx +419 -0
  308. package/docs/v7/_drafts/presets.mdx +210 -0
  309. package/docs/v7/_drafts/progressive-disclosure.mdx +230 -0
  310. package/docs/v7/_drafts/prompt-cache.mdx +200 -0
  311. package/docs/v7/_drafts/provision.mdx +390 -0
  312. package/docs/v7/_drafts/quick-start-test-recording.mdx +214 -0
  313. package/docs/v7/_drafts/readme.mdx +135 -0
  314. package/docs/v7/_drafts/reports.mdx +414 -0
  315. package/docs/v7/_drafts/scalable.mdx +763 -0
  316. package/docs/v7/_drafts/screenshot.mdx +155 -0
  317. package/docs/v7/_drafts/sdk-awesome-logs.mdx +468 -0
  318. package/docs/v7/_drafts/sdk-browser-rendering.mdx +167 -0
  319. package/docs/v7/_drafts/sdk-migration.mdx +474 -0
  320. package/docs/v7/_drafts/sdk-v7-complete.mdx +345 -0
  321. package/docs/v7/_drafts/self-hosting.mdx +369 -0
  322. package/docs/v7/_drafts/test-recording.mdx +382 -0
  323. package/docs/v7/_drafts/troubleshooting.mdx +526 -0
  324. package/docs/v7/_drafts/vitest-plugin.mdx +477 -0
  325. package/docs/v7/_drafts/vitest.mdx +535 -0
  326. package/docs/v7/_drafts/writing-tests.mdx +25 -0
  327. package/docs/v7/ai.mdx +205 -0
  328. package/docs/v7/assert.mdx +316 -0
  329. package/docs/v7/aws-setup.mdx +449 -0
  330. package/docs/v7/cache.mdx +223 -0
  331. package/docs/v7/caching.mdx +128 -0
  332. package/docs/v7/captcha.mdx +159 -0
  333. package/docs/v7/ci-cd.mdx +603 -0
  334. package/docs/v7/click.mdx +287 -0
  335. package/docs/v7/client.mdx +478 -0
  336. package/docs/v7/copilot/auto-healing.mdx +265 -0
  337. package/docs/v7/copilot/creating-tests.mdx +156 -0
  338. package/docs/v7/copilot/github.mdx +143 -0
  339. package/docs/v7/copilot/running-tests.mdx +149 -0
  340. package/docs/v7/copilot/setup.mdx +143 -0
  341. package/docs/v7/customizing-devices.mdx +319 -0
  342. package/docs/v7/dashcam.mdx +419 -0
  343. package/docs/v7/debugging-with-screenshots.mdx +402 -0
  344. package/docs/v7/device-config.mdx +317 -0
  345. package/docs/v7/double-click.mdx +102 -0
  346. package/docs/v7/elements.mdx +606 -0
  347. package/docs/v7/enterprise.mdx +9 -0
  348. package/docs/v7/errors.mdx +248 -0
  349. package/docs/v7/events.mdx +358 -0
  350. package/docs/v7/examples/ai.mdx +72 -0
  351. package/docs/v7/examples/assert.mdx +72 -0
  352. package/docs/v7/examples/captcha-api.mdx +92 -0
  353. package/docs/v7/examples/chrome-extension.mdx +132 -0
  354. package/docs/v7/examples/drag-and-drop.mdx +100 -0
  355. package/docs/v7/examples/element-not-found.mdx +67 -0
  356. package/docs/v7/examples/exec-output.mdx +85 -0
  357. package/docs/v7/examples/exec-pwsh.mdx +83 -0
  358. package/docs/v7/examples/focus-window.mdx +62 -0
  359. package/docs/v7/examples/hover-image.mdx +94 -0
  360. package/docs/v7/examples/hover-text.mdx +69 -0
  361. package/docs/v7/examples/installer.mdx +91 -0
  362. package/docs/v7/examples/launch-vscode-linux.mdx +101 -0
  363. package/docs/v7/examples/match-image.mdx +96 -0
  364. package/docs/v7/examples/press-keys.mdx +92 -0
  365. package/docs/v7/examples/scroll-keyboard.mdx +79 -0
  366. package/docs/v7/examples/scroll-until-image.mdx +81 -0
  367. package/docs/v7/examples/scroll-until-text.mdx +109 -0
  368. package/docs/v7/examples/scroll.mdx +81 -0
  369. package/docs/v7/examples/type.mdx +92 -0
  370. package/docs/v7/examples/windows-installer.mdx +89 -0
  371. package/docs/v7/exec.mdx +318 -0
  372. package/docs/v7/find.mdx +830 -0
  373. package/docs/v7/focus-application.mdx +294 -0
  374. package/docs/v7/generating-tests.mdx +36 -0
  375. package/docs/v7/hosted.mdx +158 -0
  376. package/docs/v7/hover.mdx +279 -0
  377. package/docs/v7/locating-elements.mdx +71 -0
  378. package/docs/v7/making-assertions.mdx +32 -0
  379. package/docs/v7/mcp.mdx +9 -0
  380. package/docs/v7/mouse-down.mdx +161 -0
  381. package/docs/v7/mouse-up.mdx +164 -0
  382. package/docs/v7/parse.mdx +237 -0
  383. package/docs/v7/performing-actions.mdx +54 -0
  384. package/docs/v7/press-keys.mdx +349 -0
  385. package/docs/v7/provision.mdx +333 -0
  386. package/docs/v7/quickstart.mdx +173 -0
  387. package/docs/v7/redraw.mdx +216 -0
  388. package/docs/v7/reusable-code.mdx +249 -0
  389. package/docs/v7/right-click.mdx +123 -0
  390. package/docs/v7/running-tests.mdx +185 -0
  391. package/docs/v7/screenshot.mdx +249 -0
  392. package/docs/v7/screenshots.mdx +186 -0
  393. package/docs/v7/scroll.mdx +336 -0
  394. package/docs/v7/secrets.mdx +115 -0
  395. package/docs/v7/self-hosted.mdx +149 -0
  396. package/docs/v7/type.mdx +358 -0
  397. package/docs/v7/variables.mdx +111 -0
  398. package/docs/v7/wait.mdx +52 -0
  399. package/docs/v7/waiting-for-elements.mdx +90 -0
  400. package/docs/v7/what-is-testdriver.mdx +54 -0
  401. package/eslint.config.js +67 -0
  402. package/examples/ai.test.mjs +31 -0
  403. package/examples/assert.test.mjs +47 -0
  404. package/examples/chrome-extension.test.mjs +97 -0
  405. package/examples/config.mjs +5 -0
  406. package/examples/element-not-found.test.mjs +27 -0
  407. package/examples/exec-output.test.mjs +60 -0
  408. package/examples/exec-pwsh.test.mjs +58 -0
  409. package/examples/findall-coffee-icons.test.mjs +42 -0
  410. package/examples/focus-window.test.mjs +37 -0
  411. package/examples/formatted-logging.test.mjs +27 -0
  412. package/examples/hover-image.test.mjs +53 -0
  413. package/examples/hover-text-with-description.test.mjs +57 -0
  414. package/examples/hover-text.test.mjs +28 -0
  415. package/examples/installer.test.mjs +50 -0
  416. package/examples/launch-vscode-linux.test.mjs +55 -0
  417. package/examples/match-image.test.mjs +55 -0
  418. package/examples/parse.test.mjs +19 -0
  419. package/examples/press-keys.test.mjs +44 -0
  420. package/examples/prompt.test.mjs +34 -0
  421. package/examples/scroll-keyboard.test.mjs +38 -0
  422. package/examples/scroll-until-image.test.mjs +40 -0
  423. package/examples/scroll.test.mjs +42 -0
  424. package/examples/type.test.mjs +46 -0
  425. package/examples/windows-installer.test.mjs +54 -0
  426. package/index.js +2 -0
  427. package/interfaces/cli/commands/init.js +438 -0
  428. package/interfaces/cli/commands/setup.js +382 -0
  429. package/interfaces/cli/lib/base.js +285 -0
  430. package/interfaces/cli.js +20 -0
  431. package/interfaces/junit-reporter.js +290 -0
  432. package/interfaces/logger.js +388 -0
  433. package/interfaces/readline.js +234 -0
  434. package/interfaces/shared-test-state.mjs +64 -0
  435. package/interfaces/vitest-plugin.d.ts +115 -0
  436. package/interfaces/vitest-plugin.mjs +1698 -0
  437. package/lib/captcha/solver.js +358 -0
  438. package/lib/core/Dashcam.js +533 -0
  439. package/lib/core/index.d.ts +172 -0
  440. package/lib/core/index.js +12 -0
  441. package/lib/environments.json +18 -0
  442. package/lib/github-comment-formatter.js +263 -0
  443. package/lib/github-comment.mjs +452 -0
  444. package/lib/init-project.js +575 -0
  445. package/lib/presets/index.mjs +331 -0
  446. package/lib/resolve-channel.js +46 -0
  447. package/lib/sentry.js +417 -0
  448. package/lib/vitest/hooks.d.ts +57 -0
  449. package/lib/vitest/hooks.mjs +674 -0
  450. package/lib/vitest/setup-aws.mjs +247 -0
  451. package/lib/vitest/setup-self-hosted.mjs +151 -0
  452. package/lib/vitest/setup.mjs +46 -0
  453. package/manual/captcha-api.test.mjs +51 -0
  454. package/manual/drag-and-drop.test.mjs +59 -0
  455. package/manual/flake-diffthreshold-001.test.mjs +9 -0
  456. package/manual/flake-diffthreshold-01.test.mjs +9 -0
  457. package/manual/flake-diffthreshold-05.test.mjs +9 -0
  458. package/manual/flake-noredraw-cache.test.mjs +9 -0
  459. package/manual/flake-noredraw-nocache.test.mjs +9 -0
  460. package/manual/flake-redraw-cache.test.mjs +9 -0
  461. package/manual/flake-redraw-nocache.test.mjs +9 -0
  462. package/manual/flake-rocket-match.test.mjs +30 -0
  463. package/manual/flake-shared.mjs +51 -0
  464. package/manual/no-provision.test.mjs +31 -0
  465. package/manual/packer-hover-image.test.mjs +176 -0
  466. package/manual/scroll-until-text.test.mjs +68 -0
  467. package/manual/test-init-command.js +223 -0
  468. package/mcp-server/README.md +322 -0
  469. package/mcp-server/dist/codegen.d.ts +9 -0
  470. package/mcp-server/dist/codegen.js +165 -0
  471. package/mcp-server/dist/mcp-app.html +114 -0
  472. package/mcp-server/dist/package.json +1 -0
  473. package/mcp-server/dist/provision-types.d.ts +290 -0
  474. package/mcp-server/dist/provision-types.js +174 -0
  475. package/mcp-server/dist/server.d.ts +6 -0
  476. package/mcp-server/dist/server.mjs +1925 -0
  477. package/mcp-server/dist/session.d.ts +85 -0
  478. package/mcp-server/dist/session.js +152 -0
  479. package/mcp-server/mcp-app.html +28 -0
  480. package/mcp-server/mcp-config.example.json +19 -0
  481. package/mcp-server/package-lock.json +4027 -0
  482. package/mcp-server/package.json +31 -0
  483. package/mcp-server/src/codegen.ts +189 -0
  484. package/mcp-server/src/mcp-app.css +360 -0
  485. package/mcp-server/src/mcp-app.ts +547 -0
  486. package/mcp-server/src/provision-types.ts +209 -0
  487. package/mcp-server/src/server.ts +2391 -0
  488. package/mcp-server/src/session.ts +194 -0
  489. package/mcp-server/tsconfig.json +16 -0
  490. package/mcp-server/vite.config.ts +23 -0
  491. package/package.json +158 -0
  492. package/schema.json +1046 -0
  493. package/scripts/generate-skills.js +94 -0
  494. package/sdk-log-formatter.js +1157 -0
  495. package/sdk.d.ts +1486 -0
  496. package/sdk.js +4336 -0
  497. package/setup/aws/cloudformation.yaml +463 -0
  498. package/setup/aws/disable-defender.sh +42 -0
  499. package/setup/aws/install-dev-runner.sh +79 -0
  500. package/setup/aws/spawn-runner.sh +289 -0
  501. package/test/captcha-solver.test.mjs +152 -0
  502. package/test/chrome-remote-debugging.test.mjs +66 -0
  503. package/test/duckduckgo/experiment.test.mjs +28 -0
  504. package/test/duckduckgo/setup.test.mjs +29 -0
  505. package/test/manual/debug-locate-response.js +82 -0
  506. package/test/manual/reconnect-provision.test.mjs +49 -0
  507. package/test/manual/test-console-logs.test.mjs +42 -0
  508. package/test/manual/test-find-api.js +73 -0
  509. package/test/manual/test-init.sh +54 -0
  510. package/test/manual/test-prompt-cache.js +97 -0
  511. package/test/manual/test-provision-auth.mjs +22 -0
  512. package/test/manual/test-sandbox-render.js +29 -0
  513. package/test/manual/test-sdk-methods.js +15 -0
  514. package/test/manual/test-sdk-refactor.js +53 -0
  515. package/test/manual/test-stack-trace.mjs +57 -0
  516. package/test/manual/verify-element-api.js +89 -0
  517. package/test/manual/verify-types.js +0 -0
  518. package/test/manual-unawaited-promise.test.mjs +31 -0
  519. package/vitest.config.mjs +58 -0
  520. package/vitest.runner.config.mjs +33 -0
  521. package/vscode-extension/.vscodeignore +12 -0
  522. package/vscode-extension/README.md +94 -0
  523. package/vscode-extension/media/icon.png +0 -0
  524. package/vscode-extension/package-lock.json +4126 -0
  525. package/vscode-extension/package.json +86 -0
  526. package/vscode-extension/src/extension.ts +829 -0
  527. package/vscode-extension/testdriverai-0.1.0.vsix +0 -0
  528. package/vscode-extension/tsconfig.json +16 -0
@@ -0,0 +1,184 @@
1
+ ---
2
+ name: testdriver:screenshots
3
+ description: Capture and manage screenshots during test execution
4
+ ---
5
+ <!-- Generated from screenshots.mdx. DO NOT EDIT. -->
6
+
7
+ ## Overview
8
+
9
+ TestDriver can capture screenshots manually at any point during a test, or automatically before and after every command. Screenshots are saved to a structured directory for easy debugging.
10
+
11
+ ## Manual Screenshots
12
+
13
+ Use `testdriver.screenshot()` to capture the current screen:
14
+
15
+ ```javascript
16
+ const path = await testdriver.screenshot();
17
+ console.log('Saved to:', path);
18
+ // .testdriver/screenshots/my-test/screenshot-1719849312345.png
19
+ ```
20
+
21
+ ### Options
22
+
23
+ ```javascript
24
+ await testdriver.screenshot(filename?)
25
+ ```
26
+
27
+ <ParamField path="filename" type="string">
28
+ Custom filename for the screenshot. `.png` is appended automatically if missing. If omitted, defaults to `screenshot-<timestamp>.png`.
29
+ </ParamField>
30
+
31
+ **Returns:** `Promise<string>` — the absolute file path of the saved screenshot.
32
+
33
+ ```javascript
34
+ // Default filename
35
+ await testdriver.screenshot();
36
+ // → .testdriver/screenshots/my-test/screenshot-1719849312345.png
37
+
38
+ // Custom filename
39
+ await testdriver.screenshot('login-page');
40
+ // → .testdriver/screenshots/my-test/login-page.png
41
+
42
+ // With .png extension
43
+ await testdriver.screenshot('dashboard-loaded.png');
44
+ // → .testdriver/screenshots/my-test/dashboard-loaded.png
45
+ ```
46
+
47
+ ## Auto Screenshots
48
+
49
+ Enable automatic screenshots before and after every command:
50
+
51
+ ```javascript
52
+ const testdriver = new TestDriver({
53
+ autoScreenshots: true,
54
+ });
55
+ ```
56
+
57
+ <ParamField path="autoScreenshots" type="boolean" default={false}>
58
+ When `true`, captures a screenshot before and after every SDK command (`click`, `type`, `find`, `scroll`, `hover`, `pressKeys`, `assert`, `exec`, etc.). On error, an error-phase screenshot replaces the after-phase screenshot.
59
+ </ParamField>
60
+
61
+ ### Filename Format
62
+
63
+ Auto-screenshots follow this naming convention:
64
+
65
+ ```
66
+ <seq>-<action>-<phase>-L<line>-<description>.png
67
+ ```
68
+
69
+ | Part | Description | Example |
70
+ |---|---|---|
71
+ | `seq` | 3-digit zero-padded sequence number | `001` |
72
+ | `action` | Command name | `click`, `type`, `find` |
73
+ | `phase` | `before`, `after`, or `error` | `before` |
74
+ | `L<line>` | Source line number from your test file | `L42` |
75
+ | `description` | Sanitized from command arguments (max 30 chars) | `submit-button` |
76
+
77
+ **Examples:**
78
+ ```
79
+ 001-find-before-L15-login-button.png
80
+ 002-find-after-L15-login-button.png
81
+ 003-click-before-L16-login-button.png
82
+ 004-click-after-L16-login-button.png
83
+ 005-type-before-L18-username-field.png
84
+ 006-type-error-L18-username-field.png
85
+ ```
86
+
87
+ ### Phases
88
+
89
+ | Phase | When | Description |
90
+ |---|---|---|
91
+ | `before` | Before command executes | Captures the screen state before the action |
92
+ | `after` | After successful command | Captures the result of the action |
93
+ | `error` | After failed command | Captures the screen at the point of failure (replaces `after`) |
94
+
95
+ ## Screenshot Directory
96
+
97
+ Screenshots are saved to:
98
+
99
+ ```
100
+ <cwd>/.testdriver/screenshots/<testFileName>/
101
+ ```
102
+
103
+ Where `<testFileName>` is the test file name without its extension. For example, a test at `tests/login.test.mjs` saves screenshots to `.testdriver/screenshots/login.test/`.
104
+
105
+ ### Directory Cleanup
106
+
107
+ The screenshot directory for each test file is **automatically cleaned** at the start of a test run. This happens once per process per test file to prevent concurrent tests from the same file from interfering with each other.
108
+
109
+ ## Debug Screenshots
110
+
111
+ Elements have a `saveDebugScreenshot()` method for debugging element detection:
112
+
113
+ ```javascript
114
+ const el = await testdriver.find('submit button');
115
+
116
+ // Save the screenshot that was used to detect this element
117
+ const debugPath = await el.saveDebugScreenshot();
118
+ console.log('Debug screenshot:', debugPath);
119
+ // → ./debug-screenshot-1719849312345.png
120
+
121
+ // Custom path
122
+ await el.saveDebugScreenshot('./my-debug.png');
123
+ ```
124
+
125
+ This saves the screenshot that was captured during the `find()` call, which can be useful for understanding what the AI "saw" when locating the element.
126
+
127
+ ## Complete Example
128
+
129
+ ```javascript
130
+ import { describe, it, beforeAll, afterAll } from 'vitest';
131
+ import TestDriver from 'testdriverai';
132
+
133
+ describe('Screenshot Example', () => {
134
+ let testdriver;
135
+
136
+ beforeAll(async () => {
137
+ testdriver = new TestDriver({
138
+ autoScreenshots: true, // capture every step
139
+ });
140
+ await testdriver.ready();
141
+ await testdriver.provision.chrome({ url: 'https://example.com' });
142
+ });
143
+
144
+ afterAll(async () => {
145
+ await testdriver.disconnect();
146
+ });
147
+
148
+ it('captures the login flow', async () => {
149
+ // Auto-screenshots capture before/after each command
150
+
151
+ // Manual screenshot for a specific moment
152
+ await testdriver.screenshot('initial-page-load');
153
+
154
+ const username = await testdriver.find('username input');
155
+ await username.click();
156
+ await testdriver.type('testuser@example.com');
157
+
158
+ await testdriver.screenshot('after-username-entry');
159
+
160
+ const password = await testdriver.find('password input');
161
+ await password.click();
162
+ await testdriver.type('password123');
163
+
164
+ await testdriver.find('login button').click();
165
+
166
+ await testdriver.screenshot('after-login-click');
167
+ });
168
+ });
169
+ ```
170
+
171
+ After running, your screenshot directory will contain:
172
+ ```
173
+ .testdriver/screenshots/login-flow.test/
174
+ ├── initial-page-load.png
175
+ ├── 001-find-before-L18-username-input.png
176
+ ├── 002-find-after-L18-username-input.png
177
+ ├── 003-click-before-L19-username-input.png
178
+ ├── 004-click-after-L19-username-input.png
179
+ ├── 005-type-before-L20-testuser-example-com.png
180
+ ├── 006-type-after-L20-testuser-example-com.png
181
+ ├── after-username-entry.png
182
+ ├── 007-find-before-L24-password-input.png
183
+ ├── ...
184
+ ```
@@ -0,0 +1,335 @@
1
+ ---
2
+ name: testdriver:scroll
3
+ description: Scroll pages and elements
4
+ ---
5
+ <!-- Generated from scroll.mdx. DO NOT EDIT. -->
6
+
7
+ ## Overview
8
+
9
+ Scroll the page or active element in any direction using mouse wheel or keyboard.
10
+
11
+ <Warning>
12
+ **Focus Requirements**
13
+
14
+ Scrolling requires the page or a frame to be focused. If an input field or other interactive element has focus, scroll commands may not work as expected. Before scrolling, ensure focus is on the page by:
15
+ - Clicking on a non-interactive area (e.g., page background)
16
+ - Pressing the Escape key to unfocus interactive elements
17
+ - Clicking outside of input fields or text areas
18
+
19
+ **If scroll is still not working**, try using Page Down/Page Up keys directly:
20
+ ```javascript
21
+ await testdriver.pressKeys(['pagedown']); // Scroll down
22
+ await testdriver.pressKeys(['pageup']); // Scroll up
23
+ ```
24
+ </Warning>
25
+
26
+ ## Syntax
27
+
28
+ ```javascript
29
+ await testdriver.scroll(direction, amount, method)
30
+ ```
31
+
32
+ ## Parameters
33
+
34
+ <ParamField path="direction" type="string" default="down">
35
+ Direction to scroll: `'up'`, `'down'`, `'left'`, `'right'`
36
+ </ParamField>
37
+
38
+ <ParamField path="amount" type="number" default="300">
39
+ Amount to scroll in pixels
40
+ </ParamField>
41
+
42
+ <ParamField path="method" type="string" default="mouse">
43
+ Scroll method: `'mouse'` or `'keyboard'`
44
+ </ParamField>
45
+
46
+ ## Returns
47
+
48
+ `Promise<void>`
49
+
50
+ ## Examples
51
+
52
+ ### Basic Scrolling
53
+
54
+ ```javascript
55
+ // Scroll down (default)
56
+ await testdriver.scroll();
57
+
58
+ // Scroll down 500 pixels
59
+ await testdriver.scroll('down', 500);
60
+
61
+ // Scroll up
62
+ await testdriver.scroll('up');
63
+
64
+ // Scroll up 200 pixels
65
+ await testdriver.scroll('up', 200);
66
+ ```
67
+
68
+ ### Horizontal Scrolling
69
+
70
+ ```javascript
71
+ // Scroll right
72
+ await testdriver.scroll('right', 300);
73
+
74
+ // Scroll left
75
+ await testdriver.scroll('left', 300);
76
+ ```
77
+
78
+ ### Scroll Methods
79
+
80
+ ```javascript
81
+ // Mouse wheel scroll (smooth, pixel-precise)
82
+ await testdriver.scroll('down', 300, 'mouse');
83
+
84
+ // Keyboard scroll (uses Page Down/Up, more compatible)
85
+ await testdriver.scroll('down', 300, 'keyboard');
86
+ ```
87
+
88
+ ## Scroll Until Found
89
+
90
+ ### scrollUntilText()
91
+
92
+ Scroll until specific text appears on screen.
93
+
94
+ ```javascript
95
+ await testdriver.scrollUntilText(text, direction, maxDistance, textMatchMethod, method, invert)
96
+ ```
97
+
98
+ **Parameters:**
99
+ - `text` (string) - Text to find
100
+ - `direction` (string) - Scroll direction (default: `'down'`)
101
+ - `maxDistance` (number) - Max pixels to scroll (default: 10000)
102
+ - `textMatchMethod` (string) - `'turbo'` or `'ai'` (default: `'turbo'`)
103
+ - `method` (string) - `'keyboard'` or `'mouse'` (default: `'keyboard'`)
104
+ - `invert` (boolean) - Scroll until text disappears (default: false)
105
+
106
+ **Examples:**
107
+ ```javascript
108
+ // Scroll down until "Contact Us" appears
109
+ await testdriver.scrollUntilText('Contact Us');
110
+
111
+ // Scroll up to find text
112
+ await testdriver.scrollUntilText('Header', 'up');
113
+
114
+ // Scroll until text disappears
115
+ await testdriver.scrollUntilText('Loading...', 'down', 5000, 'turbo', 'keyboard', true);
116
+
117
+ // Use AI matching for fuzzy text
118
+ await testdriver.scrollUntilText('footer content', 'down', 10000, 'ai');
119
+ ```
120
+
121
+ ### scrollUntilImage()
122
+
123
+ Scroll until a visual element appears.
124
+
125
+ ```javascript
126
+ await testdriver.scrollUntilImage(description, direction, maxDistance, method, path, invert)
127
+ ```
128
+
129
+ **Parameters:**
130
+ - `description` (string) - Description of the image/element
131
+ - `direction` (string) - Scroll direction (default: `'down'`)
132
+ - `maxDistance` (number) - Max pixels to scroll (default: 10000)
133
+ - `method` (string) - `'keyboard'` or `'mouse'` (default: `'keyboard'`)
134
+ - `path` (string | null) - Path to image template (optional)
135
+ - `invert` (boolean) - Scroll until image disappears (default: false)
136
+
137
+ **Examples:**
138
+ ```javascript
139
+ // Scroll until visual element appears
140
+ await testdriver.scrollUntilImage('red subscribe button');
141
+
142
+ // Scroll using image template
143
+ await testdriver.scrollUntilImage('', 'down', 10000, 'keyboard', './footer-logo.png');
144
+
145
+ // Scroll until image disappears
146
+ await testdriver.scrollUntilImage('loading spinner', 'down', 5000, 'keyboard', null, true);
147
+ ```
148
+
149
+ ## Best Practices
150
+
151
+ <Check>
152
+ **Ensure page has focus before scrolling**
153
+
154
+ ```javascript
155
+ // After typing in an input, unfocus it first
156
+ await testdriver.find('email input').click();
157
+ await testdriver.type('user@example.com');
158
+
159
+ // Click elsewhere or press Escape before scrolling
160
+ await testdriver.pressKeys(['escape']);
161
+ // Or click a non-interactive area
162
+ // await testdriver.find('page background').click();
163
+
164
+ // Now scroll will work properly
165
+ await testdriver.scroll('down', 300);
166
+
167
+ // If scroll still doesn't work, use Page Down directly
168
+ // await testdriver.pressKeys(['pagedown']);
169
+ ```
170
+ </Check>
171
+
172
+ <Check>
173
+ **Choose the right scroll method**
174
+
175
+ ```javascript
176
+ // For web pages, mouse scroll is usually smoother
177
+ await testdriver.scroll('down', 300, 'mouse');
178
+
179
+ // For desktop apps or when mouse doesn't work
180
+ await testdriver.scroll('down', 300, 'keyboard');
181
+ ```
182
+ </Check>
183
+
184
+ <Check>
185
+ **Use scrollUntil for dynamic content**
186
+
187
+ ```javascript
188
+ // Instead of guessing scroll amount
189
+ await testdriver.scrollUntilText('Load More button');
190
+
191
+ const loadMoreBtn = await testdriver.find('Load More button');
192
+ await loadMoreBtn.click();
193
+ ```
194
+ </Check>
195
+
196
+ <Check>
197
+ **Set reasonable max distance**
198
+
199
+ ```javascript
200
+ // Avoid infinite scrolling
201
+ await testdriver.scrollUntilText('Footer', 'down', 5000); // Max 5000px
202
+ ```
203
+ </Check>
204
+
205
+ <Warning>
206
+ **Keyboard scroll uses Page Down/Up**
207
+
208
+ Keyboard scrolling typically moves by one "page" at a time, which may be more than the specified pixel amount. It's more compatible but less precise than mouse scrolling.
209
+ </Warning>
210
+
211
+ ## Use Cases
212
+
213
+ <AccordionGroup>
214
+ <Accordion title="Navigate to Footer">
215
+ ```javascript
216
+ // Scroll to bottom of page
217
+ await testdriver.scrollUntilText('Contact Us');
218
+
219
+ const contactLink = await testdriver.find('Contact Us link');
220
+ await contactLink.click();
221
+ ```
222
+ </Accordion>
223
+
224
+ <Accordion title="Load More Results">
225
+ ```javascript
226
+ // Scroll to load more button
227
+ await testdriver.scrollUntilText('Load More');
228
+
229
+ const loadBtn = await testdriver.find('Load More button');
230
+ await loadBtn.click();
231
+
232
+ await new Promise(r => setTimeout(r, 2000));
233
+ ```
234
+ </Accordion>
235
+
236
+ <Accordion title="Find Element in Long List">
237
+ ```javascript
238
+ // Scroll through list to find item
239
+ await testdriver.scrollUntilText('Product #42');
240
+
241
+ const product = await testdriver.find('Product #42');
242
+ await product.click();
243
+ ```
244
+ </Accordion>
245
+
246
+ <Accordion title="Infinite Scroll">
247
+ ```javascript
248
+ // Scroll multiple times for infinite scroll
249
+ for (let i = 0; i < 5; i++) {
250
+ await testdriver.scroll('down', 500);
251
+ await new Promise(r => setTimeout(r, 1000)); // Wait for load
252
+ }
253
+ ```
254
+ </Accordion>
255
+
256
+ <Accordion title="Horizontal Gallery">
257
+ ```javascript
258
+ // Navigate horizontal carousel
259
+ await testdriver.scroll('right', 300);
260
+ await new Promise(r => setTimeout(r, 500));
261
+
262
+ const nextImage = await testdriver.find('next image in carousel');
263
+ await nextImage.click();
264
+ ```
265
+ </Accordion>
266
+ </AccordionGroup>
267
+
268
+ ## Complete Example
269
+
270
+ ```javascript
271
+ import { beforeAll, afterAll, describe, it } from 'vitest';
272
+ import TestDriver from 'testdriverai';
273
+
274
+ describe('Scrolling', () => {
275
+ let testdriver;
276
+
277
+ beforeAll(async () => {
278
+ client = new TestDriver(process.env.TD_API_KEY);
279
+ await testdriver.auth();
280
+ await testdriver.connect();
281
+ });
282
+
283
+ afterAll(async () => {
284
+ await testdriver.disconnect();
285
+ });
286
+
287
+ it('should scroll to find elements', async () => {
288
+ await testdriver.focusApplication('Google Chrome');
289
+
290
+ // Scroll to footer
291
+ await testdriver.scrollUntilText('Contact Information');
292
+
293
+ // Click footer link
294
+ const privacyLink = await testdriver.find('Privacy Policy link');
295
+ await privacyLink.click();
296
+
297
+ await testdriver.assert('privacy policy page is displayed');
298
+ });
299
+
300
+ it('should handle infinite scroll', async () => {
301
+ await testdriver.focusApplication('Google Chrome');
302
+
303
+ // Scroll multiple times to load content
304
+ for (let i = 0; i < 3; i++) {
305
+ await testdriver.scroll('down', 500);
306
+ await new Promise(r => setTimeout(r, 1500)); // Wait for load
307
+ }
308
+
309
+ // Verify content loaded
310
+ await testdriver.assert('more than 10 items are visible');
311
+ });
312
+
313
+ it('should scroll until loading completes', async () => {
314
+ // Scroll until loading spinner disappears
315
+ await testdriver.scrollUntilImage(
316
+ 'loading spinner',
317
+ 'down',
318
+ 5000,
319
+ 'keyboard',
320
+ null,
321
+ true // invert - wait for it to disappear
322
+ );
323
+
324
+ // Now interact with loaded content
325
+ const firstResult = await testdriver.find('first search result');
326
+ await firstResult.click();
327
+ });
328
+ });
329
+ ```
330
+
331
+ ## Related Methods
332
+
333
+ - [`find()`](/v7/find) - Locate elements after scrolling
334
+ - [`pressKeys()`](/v7/press-keys) - Use Page Down/Up keys
335
+ - [`wait()`](/v7/wait) - Wait after scrolling
@@ -0,0 +1,115 @@
1
+ ---
2
+ name: testdriver:secrets
3
+ description: Securely manage passwords and sensitive data in your tests
4
+ ---
5
+ <!-- Generated from secrets.mdx. DO NOT EDIT. -->
6
+
7
+ Protect sensitive information like passwords, API keys, and tokens in your TestDriver tests.
8
+
9
+ ## Typing Secrets Securely
10
+
11
+ When typing sensitive information like passwords, use the `secret: true` option to prevent the value from being logged or stored:
12
+
13
+ ```javascript
14
+ import { test } from 'vitest';
15
+ import { chrome } from 'testdriverai/presets';
16
+
17
+ test('login with secure password', async (context) => {
18
+ const { testdriver } = await chrome(context, {
19
+ url: 'https://myapp.com/login'
20
+ });
21
+
22
+ await testdriver.find('email input').click();
23
+ await testdriver.type(process.env.TD_USERNAME);
24
+
25
+ await testdriver.find('password input').click();
26
+ // Password is masked in logs and recordings
27
+ await testdriver.type(process.env.TD_PASSWORD, { secret: true });
28
+
29
+ await testdriver.find('login button').click();
30
+ await testdriver.assert('dashboard is visible');
31
+ });
32
+ ```
33
+
34
+ <Note>
35
+ When `secret: true` is set, the typed text appears as `****` in all logs, recordings, and dashcam output.
36
+ </Note>
37
+
38
+ ## Storing Secrets in GitHub
39
+
40
+ Store sensitive credentials as GitHub repository secrets so they're never exposed in your code:
41
+
42
+ <Steps>
43
+ <Step title="Navigate to Repository Settings">
44
+ Go to your GitHub repository → **Settings** → **Secrets and variables** → **Actions**
45
+ </Step>
46
+ <Step title="Add Repository Secrets">
47
+ Click **New repository secret** and add your secrets:
48
+ - `TD_API_KEY` - Your TestDriver API key
49
+ - `TD_USERNAME` - Test account username
50
+ - `TD_PASSWORD` - Test account password
51
+ </Step>
52
+ <Step title="Use in GitHub Actions">
53
+ Reference secrets in your workflow file:
54
+ ```yaml .github/workflows/test.yml
55
+ - name: Run TestDriver tests
56
+ env:
57
+ TD_API_KEY: ${{ secrets.TD_API_KEY }}
58
+ TD_USERNAME: ${{ secrets.TD_USERNAME }}
59
+ TD_PASSWORD: ${{ secrets.TD_PASSWORD }}
60
+ run: vitest run
61
+ ```
62
+ </Step>
63
+ </Steps>
64
+
65
+ ## Local Development
66
+
67
+ For local development, store secrets in a `.env` file:
68
+
69
+ ```bash .env
70
+ TD_API_KEY=your_api_key_here
71
+ TD_USERNAME=testuser@example.com
72
+ TD_PASSWORD=your_secure_password
73
+ ```
74
+
75
+ <Warning>
76
+ Never commit `.env` files to version control. Add `.env` to your `.gitignore` file.
77
+ </Warning>
78
+
79
+ ## Complete Example
80
+
81
+ Here's a full login test with proper secrets handling:
82
+
83
+ ```javascript tests/login.test.js
84
+ import { test, expect } from 'vitest';
85
+ import { chrome } from 'testdriverai/presets';
86
+
87
+ test('secure login flow', async (context) => {
88
+ const { testdriver } = await chrome(context, {
89
+ url: process.env.TD_WEBSITE || 'https://staging.myapp.com'
90
+ });
91
+
92
+ // Enter username (not sensitive)
93
+ await testdriver.find('email input').click();
94
+ await testdriver.type(process.env.TD_USERNAME);
95
+
96
+ // Enter password securely
97
+ await testdriver.find('password input').click();
98
+ await testdriver.type(process.env.TD_PASSWORD, { secret: true });
99
+
100
+ // Submit login
101
+ await testdriver.find('login button').click();
102
+
103
+ // Verify successful login
104
+ const loggedIn = await testdriver.assert('user is logged in');
105
+ expect(loggedIn).toBeTruthy();
106
+ });
107
+ ```
108
+
109
+ <Card title="Secrets Best Practices" icon="shield-check">
110
+ - **Always use `secret: true`** when typing passwords, tokens, or sensitive data
111
+ - **Use environment variables** to keep secrets out of code
112
+ - **Store secrets in your CI provider** (GitHub Actions, GitLab CI, etc.)
113
+ - **Never commit secrets** to version control
114
+ - **Rotate secrets regularly** to maintain security
115
+ </Card>
@@ -0,0 +1,65 @@
1
+ ---
2
+ name: testdriver:self-hosted
3
+ description: Unlimited test execution, complete privacy, and the ability to customize everything — all for a predictable flat license fee.
4
+ ---
5
+ <!-- Generated from self-hosted.mdx. DO NOT EDIT. -->
6
+
7
+ Self-hosted pricing is based on **parallel test capacity**: the number of tests you can run simultaneously on **your infrastructure**.
8
+
9
+ With self-hosting, you get:.
10
+
11
+ - **Flat license fee** per parallel test slot
12
+ - **Unlimited test execution** — run as many tests as you want
13
+ - **No device-second metering** — predictable monthly costs
14
+ - **Use your own AI keys** — control data usage with your own OpenAI, Anthropic, or other AI provider keys
15
+ - **Custom hardware & software** — choose instance types, resolution, install specific software, and configure networking as needed
16
+ - **Debug & Customize** — RDP into test machines, install custom software, modify the AMI, and debug issues directly. No black boxes.
17
+
18
+ ## Get Started
19
+
20
+ Ready to self-host? Follow our comprehensive AWS setup guide:
21
+
22
+ <Card
23
+ title="AWS Setup Guide"
24
+ icon="aws"
25
+ href="/v7/aws-setup"
26
+ >
27
+ Step-by-step instructions for deploying TestDriver on your AWS infrastructure using CloudFormation.
28
+ </Card>
29
+
30
+
31
+ ## Who Should Self-Host?
32
+
33
+ Self-hosting is ideal for teams that:
34
+
35
+ - **Run high test volumes** — Flat pricing becomes more economical at scale
36
+ - **Want infrastructure control** — Custom hardware, specific software dependencies, or network configurations
37
+ - **Prefer predictable costs** — Budget with confidence using flat monthly fees
38
+
39
+
40
+ ## How It Works
41
+
42
+ With self-hosting, you run test sandboxes on your own AWS infrastructure. TestDriver still provides:
43
+
44
+ - **Dashboard** — View test results, analytics, and reports at [console.testdriver.ai](https://console.testdriver.ai)
45
+ - **API** — Orchestration and AI-powered test execution
46
+ - **License Management** — Your parallel test capacity
47
+
48
+ You provide:
49
+
50
+ - **AWS Infrastructure** — EC2 instances running in your account
51
+ - **AI API Keys** — Use your own OpenAI, Anthropic, or other AI provider keys
52
+ - **Custom Configuration** — Hardware specs, networking, installed software
53
+
54
+ ## Comparison vs Cloud
55
+
56
+ | Feature | Cloud | Self-Hosted |
57
+ |---------|-------|-------------|
58
+ | **Setup Time** | Minutes | Hours |
59
+ | **Pricing Model** | Device-seconds | Flat license fee |
60
+ | **Infrastructure Management** | TestDriver | You |
61
+ | **Device Location** | TestDriver cloud | Your AWS account |
62
+ | **AI API Keys** | TestDriver's | Your own |
63
+ | **Custom Software** | Limited | Full control |
64
+ | **Hardware Selection** | Standard | Your choice |
65
+ | **Debugging Access** | Replays only | Full RDP access |