@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.
- package/.claude/settings.local.json +7 -0
- package/.env.example +4 -0
- package/.prettierignore +4 -0
- package/.prettierrc +1 -0
- package/CHANGELOG.md +953 -0
- package/README.md +81 -0
- package/agent/events.js +135 -0
- package/agent/index.js +2450 -0
- package/agent/interface.js +35 -0
- package/agent/lib/analytics.js +22 -0
- package/agent/lib/censorship.js +75 -0
- package/agent/lib/commander.js +246 -0
- package/agent/lib/commands.js +1684 -0
- package/agent/lib/config.js +60 -0
- package/agent/lib/generator.js +91 -0
- package/agent/lib/http.js +144 -0
- package/agent/lib/logger.js +56 -0
- package/agent/lib/outputs.js +29 -0
- package/agent/lib/parser.js +209 -0
- package/agent/lib/redraw.js +386 -0
- package/agent/lib/resources/cursor-2.png +0 -0
- package/agent/lib/sandbox.js +1104 -0
- package/agent/lib/sdk.js +633 -0
- package/agent/lib/session.js +25 -0
- package/agent/lib/source-mapper.js +342 -0
- package/agent/lib/subimage/index.js +77 -0
- package/agent/lib/subimage/opencv.js +69 -0
- package/agent/lib/system.js +204 -0
- package/agent/lib/theme.js +14 -0
- package/agent/lib/valid-version.js +21 -0
- package/agent/lib/validation.js +169 -0
- package/ai/.claude-plugin/plugin.json +9 -0
- package/ai/agents/testdriver.md +638 -0
- package/ai/skills/testdriver-ai/SKILL.md +204 -0
- package/ai/skills/testdriver-assert/SKILL.md +315 -0
- package/ai/skills/testdriver-aws-setup/SKILL.md +448 -0
- package/ai/skills/testdriver-cache/SKILL.md +221 -0
- package/ai/skills/testdriver-caching/SKILL.md +124 -0
- package/ai/skills/testdriver-captcha/SKILL.md +158 -0
- package/ai/skills/testdriver-ci-cd/SKILL.md +602 -0
- package/ai/skills/testdriver-click/SKILL.md +286 -0
- package/ai/skills/testdriver-client/SKILL.md +477 -0
- package/ai/skills/testdriver-cloud/SKILL.md +119 -0
- package/ai/skills/testdriver-customizing-devices/SKILL.md +319 -0
- package/ai/skills/testdriver-dashcam/SKILL.md +418 -0
- package/ai/skills/testdriver-debugging-with-screenshots/SKILL.md +401 -0
- package/ai/skills/testdriver-device-config/SKILL.md +317 -0
- package/ai/skills/testdriver-double-click/SKILL.md +102 -0
- package/ai/skills/testdriver-elements/SKILL.md +605 -0
- package/ai/skills/testdriver-enterprise/SKILL.md +114 -0
- package/ai/skills/testdriver-errors/SKILL.md +246 -0
- package/ai/skills/testdriver-events/SKILL.md +356 -0
- package/ai/skills/testdriver-examples/SKILL.md +7 -0
- package/ai/skills/testdriver-exec/SKILL.md +317 -0
- package/ai/skills/testdriver-find/SKILL.md +829 -0
- package/ai/skills/testdriver-focus-application/SKILL.md +293 -0
- package/ai/skills/testdriver-generating-tests/SKILL.md +36 -0
- package/ai/skills/testdriver-hover/SKILL.md +278 -0
- package/ai/skills/testdriver-locating-elements/SKILL.md +71 -0
- package/ai/skills/testdriver-making-assertions/SKILL.md +32 -0
- package/ai/skills/testdriver-mcp/SKILL.md +7 -0
- package/ai/skills/testdriver-mcp-workflow/SKILL.md +410 -0
- package/ai/skills/testdriver-mouse-down/SKILL.md +161 -0
- package/ai/skills/testdriver-mouse-up/SKILL.md +164 -0
- package/ai/skills/testdriver-parse/SKILL.md +236 -0
- package/ai/skills/testdriver-performing-actions/SKILL.md +54 -0
- package/ai/skills/testdriver-press-keys/SKILL.md +348 -0
- package/ai/skills/testdriver-provision/SKILL.md +331 -0
- package/ai/skills/testdriver-quickstart/SKILL.md +144 -0
- package/ai/skills/testdriver-redraw/SKILL.md +214 -0
- package/ai/skills/testdriver-reusable-code/SKILL.md +249 -0
- package/ai/skills/testdriver-right-click/SKILL.md +123 -0
- package/ai/skills/testdriver-running-tests/SKILL.md +185 -0
- package/ai/skills/testdriver-screenshot/SKILL.md +248 -0
- package/ai/skills/testdriver-screenshots/SKILL.md +184 -0
- package/ai/skills/testdriver-scroll/SKILL.md +335 -0
- package/ai/skills/testdriver-secrets/SKILL.md +115 -0
- package/ai/skills/testdriver-self-hosted/SKILL.md +65 -0
- package/ai/skills/testdriver-test-writer/SKILL.md +448 -0
- package/ai/skills/testdriver-testdriver/SKILL.md +628 -0
- package/ai/skills/testdriver-testdriver-mechanic/SKILL.md +165 -0
- package/ai/skills/testdriver-type/SKILL.md +357 -0
- package/ai/skills/testdriver-variables/SKILL.md +111 -0
- package/ai/skills/testdriver-wait/SKILL.md +50 -0
- package/ai/skills/testdriver-waiting-for-elements/SKILL.md +90 -0
- package/ai/skills/testdriver-what-is-testdriver/SKILL.md +54 -0
- package/bin/testdriverai.js +22 -0
- package/debugger/bg.png +0 -0
- package/debugger/icon.png +0 -0
- package/debugger/index.html +469 -0
- package/debugger/td.png +0 -0
- package/debugger/tray-buffered.png +0 -0
- package/debugger/tray.png +0 -0
- package/docs/GITHUB_COMMENTS.md +330 -0
- package/docs/GITHUB_COMMENTS_ANNOUNCEMENT.md +167 -0
- package/docs/QUICK-START-GITHUB-COMMENTS.md +84 -0
- package/docs/TEST-GITHUB-COMMENTS.md +129 -0
- package/docs/_data/examples-manifest.json +177 -0
- package/docs/_data/examples-manifest.schema.json +41 -0
- package/docs/_scripts/extract-example-urls.js +165 -0
- package/docs/_scripts/generate-examples.js +560 -0
- package/docs/_scripts/generate-skills.js +154 -0
- package/docs/_scripts/link-replacer.js +164 -0
- package/docs/_scripts/upload-docs-to-openai.js +284 -0
- package/docs/changelog.mdx +161 -0
- package/docs/claude-mcp-plugin.mdx +160 -0
- package/docs/docs.json +442 -0
- package/docs/github-integration-setup.md +266 -0
- package/docs/guide/best-practices-polling.mdx +174 -0
- package/docs/images/content/account/newprojectsettings.png +0 -0
- package/docs/images/content/account/projectpage.png +0 -0
- package/docs/images/content/account/projectreplays.png +0 -0
- package/docs/images/content/account/team-manage.png +0 -0
- package/docs/images/content/account/teampage.png +0 -0
- package/docs/images/content/extension/cursor.svg +1 -0
- package/docs/images/content/extension/vscode.svg +57 -0
- package/docs/images/content/extension/windsurf.svg +3 -0
- package/docs/images/content/parse/output.png +0 -0
- package/docs/images/content/self-hosted/launchtemplateid.png +0 -0
- package/docs/images/content/side-by-side.png +0 -0
- package/docs/images/content/vscode/ide-full.png +0 -0
- package/docs/images/content/vscode/running.png +0 -0
- package/docs/images/content/vscode/v7-chat.png +0 -0
- package/docs/images/content/vscode/v7-choose-agent.png +0 -0
- package/docs/images/content/vscode/v7-full.png +0 -0
- package/docs/images/content/vscode/v7-onboarding.png +0 -0
- package/docs/images/content/vscode/vscode-2-assert.png +0 -0
- package/docs/images/content/vscode/vscode-agent-preview.png +0 -0
- package/docs/images/content/vscode/vscode-copilot-ask.png +0 -0
- package/docs/images/content/vscode/vscode-file-creation.png +0 -0
- package/docs/images/content/vscode/vscode-install.png +0 -0
- package/docs/images/content/vscode/vscode-overview.png +0 -0
- package/docs/images/content/vscode/vscode-setup-walkthrough.png +0 -0
- package/docs/images/content/vscode/vscode-stopchat.png +0 -0
- package/docs/images/content/vscode/vscode-stoptest.png +0 -0
- package/docs/images/content/vscode/vscode-tdservice.png +0 -0
- package/docs/images/content/vscode/vscode-test-output.png +0 -0
- package/docs/images/content/vscode/vscode-testhistory.png +0 -0
- package/docs/images/content/vscode/vscode-testpane-runtests.png +0 -0
- package/docs/images/content/vscode/vscode-testpane.png +0 -0
- package/docs/images/template/dark.png +0 -0
- package/docs/images/template/icon.png +0 -0
- package/docs/images/template/light.png +0 -0
- package/docs/snippets/calendar-link.mdx +4 -0
- package/docs/snippets/gitignore-warning.mdx +7 -0
- package/docs/snippets/lifecycle-warning.mdx +6 -0
- package/docs/snippets/test-prereqs.mdx +12 -0
- package/docs/snippets/tests/assert-replay.mdx +7 -0
- package/docs/snippets/tests/assert-yaml.mdx +8 -0
- package/docs/snippets/tests/exec-js-replay.mdx +7 -0
- package/docs/snippets/tests/exec-js-yaml.mdx +32 -0
- package/docs/snippets/tests/exec-shell-replay.mdx +7 -0
- package/docs/snippets/tests/exec-shell-yaml.mdx +15 -0
- package/docs/snippets/tests/hover-image-replay.mdx +7 -0
- package/docs/snippets/tests/hover-image-yaml.mdx +17 -0
- package/docs/snippets/tests/hover-text-replay.mdx +7 -0
- package/docs/snippets/tests/hover-text-with-description-replay.mdx +7 -0
- package/docs/snippets/tests/hover-text-with-description-yaml.mdx +24 -0
- package/docs/snippets/tests/hover-text-yaml.mdx +14 -0
- package/docs/snippets/tests/match-image-replay.mdx +7 -0
- package/docs/snippets/tests/match-image-yaml.mdx +17 -0
- package/docs/snippets/tests/press-keys-replay.mdx +7 -0
- package/docs/snippets/tests/press-keys-yaml.mdx +36 -0
- package/docs/snippets/tests/remember-replay.mdx +7 -0
- package/docs/snippets/tests/remember-yaml.mdx +28 -0
- package/docs/snippets/tests/scroll-replay.mdx +7 -0
- package/docs/snippets/tests/scroll-until-image-replay.mdx +7 -0
- package/docs/snippets/tests/scroll-until-image-yaml.mdx +14 -0
- package/docs/snippets/tests/scroll-until-text-replay.mdx +7 -0
- package/docs/snippets/tests/scroll-until-text-yaml.mdx +17 -0
- package/docs/snippets/tests/scroll-yaml.mdx +30 -0
- package/docs/snippets/tests/type-repeated-replay.mdx +7 -0
- package/docs/snippets/tests/type-repeated-yaml.mdx +22 -0
- package/docs/snippets/tests/type-replay.mdx +7 -0
- package/docs/snippets/tests/type-yaml.mdx +28 -0
- package/docs/snippets/tests/wait-for-image-replay.mdx +7 -0
- package/docs/snippets/tests/wait-for-image-yaml.mdx +18 -0
- package/docs/snippets/tests/wait-for-text-replay.mdx +7 -0
- package/docs/snippets/tests/wait-for-text-yaml.mdx +18 -0
- package/docs/snippets/tests/wait-replay.mdx +7 -0
- package/docs/snippets/tests/wait-yaml.mdx +13 -0
- package/docs/styles.css +65 -0
- package/docs/v6/account/dashboard.mdx +16 -0
- package/docs/v6/account/enterprise.mdx +110 -0
- package/docs/v6/account/pricing.mdx +33 -0
- package/docs/v6/account/projects.mdx +33 -0
- package/docs/v6/account/team.mdx +35 -0
- package/docs/v6/action/ami.mdx +109 -0
- package/docs/v6/action/performance.mdx +105 -0
- package/docs/v6/action/secrets.mdx +93 -0
- package/docs/v6/apps/chrome-extensions.mdx +48 -0
- package/docs/v6/apps/desktop-apps.mdx +93 -0
- package/docs/v6/apps/mobile-apps.mdx +26 -0
- package/docs/v6/apps/static-websites.mdx +54 -0
- package/docs/v6/apps/tauri-apps.mdx +361 -0
- package/docs/v6/bugs/jira.mdx +232 -0
- package/docs/v6/cli/overview.mdx +66 -0
- package/docs/v6/commands/assert.mdx +45 -0
- package/docs/v6/commands/exec.mdx +276 -0
- package/docs/v6/commands/focus-application.mdx +44 -0
- package/docs/v6/commands/hover-image.mdx +69 -0
- package/docs/v6/commands/hover-text.mdx +47 -0
- package/docs/v6/commands/if.mdx +53 -0
- package/docs/v6/commands/match-image.mdx +67 -0
- package/docs/v6/commands/press-keys.mdx +87 -0
- package/docs/v6/commands/remember.mdx +49 -0
- package/docs/v6/commands/run.mdx +44 -0
- package/docs/v6/commands/scroll-until-image.mdx +66 -0
- package/docs/v6/commands/scroll-until-text.mdx +60 -0
- package/docs/v6/commands/scroll.mdx +69 -0
- package/docs/v6/commands/type.mdx +45 -0
- package/docs/v6/commands/wait-for-image.mdx +54 -0
- package/docs/v6/commands/wait-for-text.mdx +48 -0
- package/docs/v6/commands/wait.mdx +45 -0
- package/docs/v6/exporting/junit.mdx +218 -0
- package/docs/v6/exporting/playwright.mdx +197 -0
- package/docs/v6/features/auto-healing.mdx +144 -0
- package/docs/v6/features/generation.mdx +116 -0
- package/docs/v6/features/parallel-testing.mdx +151 -0
- package/docs/v6/features/reusable-snippets.mdx +131 -0
- package/docs/v6/features/selectorless.mdx +80 -0
- package/docs/v6/features/visual-assertions.mdx +139 -0
- package/docs/v6/getting-started/ci.mdx +146 -0
- package/docs/v6/getting-started/cli.mdx +91 -0
- package/docs/v6/getting-started/editing.mdx +100 -0
- package/docs/v6/getting-started/playwright.mdx +342 -0
- package/docs/v6/getting-started/running.mdx +48 -0
- package/docs/v6/getting-started/self-hosting.mdx +408 -0
- package/docs/v6/getting-started/vscode.mdx +88 -0
- package/docs/v6/guide/assertions.mdx +189 -0
- package/docs/v6/guide/authentication.mdx +136 -0
- package/docs/v6/guide/code.mdx +65 -0
- package/docs/v6/guide/dashcam.mdx +118 -0
- package/docs/v6/guide/environment-variables.mdx +26 -0
- package/docs/v6/guide/lifecycle.mdx +242 -0
- package/docs/v6/guide/locating.mdx +141 -0
- package/docs/v6/guide/protips.mdx +43 -0
- package/docs/v6/guide/variables.mdx +143 -0
- package/docs/v6/guide/waiting.mdx +130 -0
- package/docs/v6/importing/csv.mdx +196 -0
- package/docs/v6/importing/gherkin.mdx +143 -0
- package/docs/v6/importing/jira.mdx +164 -0
- package/docs/v6/importing/testrail.mdx +162 -0
- package/docs/v6/integrations/electron.mdx +146 -0
- package/docs/v6/integrations/netlify.mdx +100 -0
- package/docs/v6/integrations/vercel.mdx +125 -0
- package/docs/v6/interactive/explore.mdx +99 -0
- package/docs/v6/interactive/run.mdx +52 -0
- package/docs/v6/interactive/save.mdx +63 -0
- package/docs/v6/overview/comparison.mdx +101 -0
- package/docs/v6/overview/faq.mdx +162 -0
- package/docs/v6/overview/performance.mdx +52 -0
- package/docs/v6/overview/quickstart.mdx +137 -0
- package/docs/v6/overview/what-is-testdriver.mdx +85 -0
- package/docs/v6/scenarios/ai-chatbot.mdx +28 -0
- package/docs/v6/scenarios/cookie-banner.mdx +32 -0
- package/docs/v6/scenarios/file-upload.mdx +33 -0
- package/docs/v6/scenarios/form-filling.mdx +32 -0
- package/docs/v6/scenarios/log-in.mdx +75 -0
- package/docs/v6/scenarios/pdf-generation.mdx +25 -0
- package/docs/v6/scenarios/spell-check.mdx +22 -0
- package/docs/v6/security/action.mdx +84 -0
- package/docs/v6/security/agent.mdx +73 -0
- package/docs/v6/security/platform.mdx +77 -0
- package/docs/v6/tutorials/advanced-test.mdx +81 -0
- package/docs/v6/tutorials/basic-test.mdx +45 -0
- package/docs/v7/_drafts/agents.mdx +843 -0
- package/docs/v7/_drafts/architecture.mdx +399 -0
- package/docs/v7/_drafts/auto-cache-key.mdx +167 -0
- package/docs/v7/_drafts/awesome-logs-quick-ref.mdx +100 -0
- package/docs/v7/_drafts/best-practices.mdx +486 -0
- package/docs/v7/_drafts/caching-ai.mdx +215 -0
- package/docs/v7/_drafts/caching-selectors.mdx +424 -0
- package/docs/v7/_drafts/caching.mdx +366 -0
- package/docs/v7/_drafts/cli-to-sdk-migration.mdx +425 -0
- package/docs/v7/_drafts/commands/assert.mdx +45 -0
- package/docs/v7/_drafts/commands/exec.mdx +276 -0
- package/docs/v7/_drafts/commands/focus-application.mdx +44 -0
- package/docs/v7/_drafts/commands/hover-image.mdx +69 -0
- package/docs/v7/_drafts/commands/hover-text.mdx +47 -0
- package/docs/v7/_drafts/commands/if.mdx +53 -0
- package/docs/v7/_drafts/commands/match-image.mdx +67 -0
- package/docs/v7/_drafts/commands/press-keys.mdx +87 -0
- package/docs/v7/_drafts/commands/remember.mdx +49 -0
- package/docs/v7/_drafts/commands/run.mdx +44 -0
- package/docs/v7/_drafts/commands/scroll-until-image.mdx +66 -0
- package/docs/v7/_drafts/commands/scroll-until-text.mdx +60 -0
- package/docs/v7/_drafts/commands/scroll.mdx +69 -0
- package/docs/v7/_drafts/commands/type.mdx +45 -0
- package/docs/v7/_drafts/commands/wait-for-image.mdx +54 -0
- package/docs/v7/_drafts/commands/wait-for-text.mdx +48 -0
- package/docs/v7/_drafts/commands/wait.mdx +45 -0
- package/docs/v7/_drafts/configuration.mdx +378 -0
- package/docs/v7/_drafts/contributing.mdx +174 -0
- package/docs/v7/_drafts/core.mdx +458 -0
- package/docs/v7/_drafts/dashcam-title-feature.mdx +89 -0
- package/docs/v7/_drafts/debugging.mdx +349 -0
- package/docs/v7/_drafts/error-handling.mdx +501 -0
- package/docs/v7/_drafts/faq.mdx +393 -0
- package/docs/v7/_drafts/hooks.mdx +360 -0
- package/docs/v7/_drafts/init-command.mdx +95 -0
- package/docs/v7/_drafts/installation.mdx +420 -0
- package/docs/v7/_drafts/migration.mdx +562 -0
- package/docs/v7/_drafts/observable.mdx +604 -0
- package/docs/v7/_drafts/playwright.mdx +342 -0
- package/docs/v7/_drafts/plugin-migration.mdx +220 -0
- package/docs/v7/_drafts/powerful.mdx +419 -0
- package/docs/v7/_drafts/presets.mdx +210 -0
- package/docs/v7/_drafts/progressive-disclosure.mdx +230 -0
- package/docs/v7/_drafts/prompt-cache.mdx +200 -0
- package/docs/v7/_drafts/provision.mdx +390 -0
- package/docs/v7/_drafts/quick-start-test-recording.mdx +214 -0
- package/docs/v7/_drafts/readme.mdx +135 -0
- package/docs/v7/_drafts/reports.mdx +414 -0
- package/docs/v7/_drafts/scalable.mdx +763 -0
- package/docs/v7/_drafts/screenshot.mdx +155 -0
- package/docs/v7/_drafts/sdk-awesome-logs.mdx +468 -0
- package/docs/v7/_drafts/sdk-browser-rendering.mdx +167 -0
- package/docs/v7/_drafts/sdk-migration.mdx +474 -0
- package/docs/v7/_drafts/sdk-v7-complete.mdx +345 -0
- package/docs/v7/_drafts/self-hosting.mdx +369 -0
- package/docs/v7/_drafts/test-recording.mdx +382 -0
- package/docs/v7/_drafts/troubleshooting.mdx +526 -0
- package/docs/v7/_drafts/vitest-plugin.mdx +477 -0
- package/docs/v7/_drafts/vitest.mdx +535 -0
- package/docs/v7/_drafts/writing-tests.mdx +25 -0
- package/docs/v7/ai.mdx +205 -0
- package/docs/v7/assert.mdx +316 -0
- package/docs/v7/aws-setup.mdx +449 -0
- package/docs/v7/cache.mdx +223 -0
- package/docs/v7/caching.mdx +128 -0
- package/docs/v7/captcha.mdx +159 -0
- package/docs/v7/ci-cd.mdx +603 -0
- package/docs/v7/click.mdx +287 -0
- package/docs/v7/client.mdx +478 -0
- package/docs/v7/copilot/auto-healing.mdx +265 -0
- package/docs/v7/copilot/creating-tests.mdx +156 -0
- package/docs/v7/copilot/github.mdx +143 -0
- package/docs/v7/copilot/running-tests.mdx +149 -0
- package/docs/v7/copilot/setup.mdx +143 -0
- package/docs/v7/customizing-devices.mdx +319 -0
- package/docs/v7/dashcam.mdx +419 -0
- package/docs/v7/debugging-with-screenshots.mdx +402 -0
- package/docs/v7/device-config.mdx +317 -0
- package/docs/v7/double-click.mdx +102 -0
- package/docs/v7/elements.mdx +606 -0
- package/docs/v7/enterprise.mdx +9 -0
- package/docs/v7/errors.mdx +248 -0
- package/docs/v7/events.mdx +358 -0
- package/docs/v7/examples/ai.mdx +72 -0
- package/docs/v7/examples/assert.mdx +72 -0
- package/docs/v7/examples/captcha-api.mdx +92 -0
- package/docs/v7/examples/chrome-extension.mdx +132 -0
- package/docs/v7/examples/drag-and-drop.mdx +100 -0
- package/docs/v7/examples/element-not-found.mdx +67 -0
- package/docs/v7/examples/exec-output.mdx +85 -0
- package/docs/v7/examples/exec-pwsh.mdx +83 -0
- package/docs/v7/examples/focus-window.mdx +62 -0
- package/docs/v7/examples/hover-image.mdx +94 -0
- package/docs/v7/examples/hover-text.mdx +69 -0
- package/docs/v7/examples/installer.mdx +91 -0
- package/docs/v7/examples/launch-vscode-linux.mdx +101 -0
- package/docs/v7/examples/match-image.mdx +96 -0
- package/docs/v7/examples/press-keys.mdx +92 -0
- package/docs/v7/examples/scroll-keyboard.mdx +79 -0
- package/docs/v7/examples/scroll-until-image.mdx +81 -0
- package/docs/v7/examples/scroll-until-text.mdx +109 -0
- package/docs/v7/examples/scroll.mdx +81 -0
- package/docs/v7/examples/type.mdx +92 -0
- package/docs/v7/examples/windows-installer.mdx +89 -0
- package/docs/v7/exec.mdx +318 -0
- package/docs/v7/find.mdx +830 -0
- package/docs/v7/focus-application.mdx +294 -0
- package/docs/v7/generating-tests.mdx +36 -0
- package/docs/v7/hosted.mdx +158 -0
- package/docs/v7/hover.mdx +279 -0
- package/docs/v7/locating-elements.mdx +71 -0
- package/docs/v7/making-assertions.mdx +32 -0
- package/docs/v7/mcp.mdx +9 -0
- package/docs/v7/mouse-down.mdx +161 -0
- package/docs/v7/mouse-up.mdx +164 -0
- package/docs/v7/parse.mdx +237 -0
- package/docs/v7/performing-actions.mdx +54 -0
- package/docs/v7/press-keys.mdx +349 -0
- package/docs/v7/provision.mdx +333 -0
- package/docs/v7/quickstart.mdx +173 -0
- package/docs/v7/redraw.mdx +216 -0
- package/docs/v7/reusable-code.mdx +249 -0
- package/docs/v7/right-click.mdx +123 -0
- package/docs/v7/running-tests.mdx +185 -0
- package/docs/v7/screenshot.mdx +249 -0
- package/docs/v7/screenshots.mdx +186 -0
- package/docs/v7/scroll.mdx +336 -0
- package/docs/v7/secrets.mdx +115 -0
- package/docs/v7/self-hosted.mdx +149 -0
- package/docs/v7/type.mdx +358 -0
- package/docs/v7/variables.mdx +111 -0
- package/docs/v7/wait.mdx +52 -0
- package/docs/v7/waiting-for-elements.mdx +90 -0
- package/docs/v7/what-is-testdriver.mdx +54 -0
- package/eslint.config.js +67 -0
- package/examples/ai.test.mjs +31 -0
- package/examples/assert.test.mjs +47 -0
- package/examples/chrome-extension.test.mjs +97 -0
- package/examples/config.mjs +5 -0
- package/examples/element-not-found.test.mjs +27 -0
- package/examples/exec-output.test.mjs +60 -0
- package/examples/exec-pwsh.test.mjs +58 -0
- package/examples/findall-coffee-icons.test.mjs +42 -0
- package/examples/focus-window.test.mjs +37 -0
- package/examples/formatted-logging.test.mjs +27 -0
- package/examples/hover-image.test.mjs +53 -0
- package/examples/hover-text-with-description.test.mjs +57 -0
- package/examples/hover-text.test.mjs +28 -0
- package/examples/installer.test.mjs +50 -0
- package/examples/launch-vscode-linux.test.mjs +55 -0
- package/examples/match-image.test.mjs +55 -0
- package/examples/parse.test.mjs +19 -0
- package/examples/press-keys.test.mjs +44 -0
- package/examples/prompt.test.mjs +34 -0
- package/examples/scroll-keyboard.test.mjs +38 -0
- package/examples/scroll-until-image.test.mjs +40 -0
- package/examples/scroll.test.mjs +42 -0
- package/examples/type.test.mjs +46 -0
- package/examples/windows-installer.test.mjs +54 -0
- package/index.js +2 -0
- package/interfaces/cli/commands/init.js +438 -0
- package/interfaces/cli/commands/setup.js +382 -0
- package/interfaces/cli/lib/base.js +285 -0
- package/interfaces/cli.js +20 -0
- package/interfaces/junit-reporter.js +290 -0
- package/interfaces/logger.js +388 -0
- package/interfaces/readline.js +234 -0
- package/interfaces/shared-test-state.mjs +64 -0
- package/interfaces/vitest-plugin.d.ts +115 -0
- package/interfaces/vitest-plugin.mjs +1698 -0
- package/lib/captcha/solver.js +358 -0
- package/lib/core/Dashcam.js +533 -0
- package/lib/core/index.d.ts +172 -0
- package/lib/core/index.js +12 -0
- package/lib/environments.json +18 -0
- package/lib/github-comment-formatter.js +263 -0
- package/lib/github-comment.mjs +452 -0
- package/lib/init-project.js +575 -0
- package/lib/presets/index.mjs +331 -0
- package/lib/resolve-channel.js +46 -0
- package/lib/sentry.js +417 -0
- package/lib/vitest/hooks.d.ts +57 -0
- package/lib/vitest/hooks.mjs +674 -0
- package/lib/vitest/setup-aws.mjs +247 -0
- package/lib/vitest/setup-self-hosted.mjs +151 -0
- package/lib/vitest/setup.mjs +46 -0
- package/manual/captcha-api.test.mjs +51 -0
- package/manual/drag-and-drop.test.mjs +59 -0
- package/manual/flake-diffthreshold-001.test.mjs +9 -0
- package/manual/flake-diffthreshold-01.test.mjs +9 -0
- package/manual/flake-diffthreshold-05.test.mjs +9 -0
- package/manual/flake-noredraw-cache.test.mjs +9 -0
- package/manual/flake-noredraw-nocache.test.mjs +9 -0
- package/manual/flake-redraw-cache.test.mjs +9 -0
- package/manual/flake-redraw-nocache.test.mjs +9 -0
- package/manual/flake-rocket-match.test.mjs +30 -0
- package/manual/flake-shared.mjs +51 -0
- package/manual/no-provision.test.mjs +31 -0
- package/manual/packer-hover-image.test.mjs +176 -0
- package/manual/scroll-until-text.test.mjs +68 -0
- package/manual/test-init-command.js +223 -0
- package/mcp-server/README.md +322 -0
- package/mcp-server/dist/codegen.d.ts +9 -0
- package/mcp-server/dist/codegen.js +165 -0
- package/mcp-server/dist/mcp-app.html +114 -0
- package/mcp-server/dist/package.json +1 -0
- package/mcp-server/dist/provision-types.d.ts +290 -0
- package/mcp-server/dist/provision-types.js +174 -0
- package/mcp-server/dist/server.d.ts +6 -0
- package/mcp-server/dist/server.mjs +1925 -0
- package/mcp-server/dist/session.d.ts +85 -0
- package/mcp-server/dist/session.js +152 -0
- package/mcp-server/mcp-app.html +28 -0
- package/mcp-server/mcp-config.example.json +19 -0
- package/mcp-server/package-lock.json +4027 -0
- package/mcp-server/package.json +31 -0
- package/mcp-server/src/codegen.ts +189 -0
- package/mcp-server/src/mcp-app.css +360 -0
- package/mcp-server/src/mcp-app.ts +547 -0
- package/mcp-server/src/provision-types.ts +209 -0
- package/mcp-server/src/server.ts +2391 -0
- package/mcp-server/src/session.ts +194 -0
- package/mcp-server/tsconfig.json +16 -0
- package/mcp-server/vite.config.ts +23 -0
- package/package.json +158 -0
- package/schema.json +1046 -0
- package/scripts/generate-skills.js +94 -0
- package/sdk-log-formatter.js +1157 -0
- package/sdk.d.ts +1486 -0
- package/sdk.js +4336 -0
- package/setup/aws/cloudformation.yaml +463 -0
- package/setup/aws/disable-defender.sh +42 -0
- package/setup/aws/install-dev-runner.sh +79 -0
- package/setup/aws/spawn-runner.sh +289 -0
- package/test/captcha-solver.test.mjs +152 -0
- package/test/chrome-remote-debugging.test.mjs +66 -0
- package/test/duckduckgo/experiment.test.mjs +28 -0
- package/test/duckduckgo/setup.test.mjs +29 -0
- package/test/manual/debug-locate-response.js +82 -0
- package/test/manual/reconnect-provision.test.mjs +49 -0
- package/test/manual/test-console-logs.test.mjs +42 -0
- package/test/manual/test-find-api.js +73 -0
- package/test/manual/test-init.sh +54 -0
- package/test/manual/test-prompt-cache.js +97 -0
- package/test/manual/test-provision-auth.mjs +22 -0
- package/test/manual/test-sandbox-render.js +29 -0
- package/test/manual/test-sdk-methods.js +15 -0
- package/test/manual/test-sdk-refactor.js +53 -0
- package/test/manual/test-stack-trace.mjs +57 -0
- package/test/manual/verify-element-api.js +89 -0
- package/test/manual/verify-types.js +0 -0
- package/test/manual-unawaited-promise.test.mjs +31 -0
- package/vitest.config.mjs +58 -0
- package/vitest.runner.config.mjs +33 -0
- package/vscode-extension/.vscodeignore +12 -0
- package/vscode-extension/README.md +94 -0
- package/vscode-extension/media/icon.png +0 -0
- package/vscode-extension/package-lock.json +4126 -0
- package/vscode-extension/package.json +86 -0
- package/vscode-extension/src/extension.ts +829 -0
- package/vscode-extension/testdriverai-0.1.0.vsix +0 -0
- package/vscode-extension/tsconfig.json +16 -0
package/docs/v7/type.mdx
ADDED
|
@@ -0,0 +1,358 @@
|
|
|
1
|
+
---
|
|
2
|
+
title: "type()"
|
|
3
|
+
sidebarTitle: "type"
|
|
4
|
+
description: "Type text into focused input fields"
|
|
5
|
+
icon: "keyboard"
|
|
6
|
+
---
|
|
7
|
+
|
|
8
|
+
## Overview
|
|
9
|
+
|
|
10
|
+
Type text or numbers into the currently focused input field with optional delay between keystrokes.
|
|
11
|
+
|
|
12
|
+
## Syntax
|
|
13
|
+
|
|
14
|
+
```javascript
|
|
15
|
+
await testdriver.type(text, options)
|
|
16
|
+
```
|
|
17
|
+
|
|
18
|
+
## Parameters
|
|
19
|
+
|
|
20
|
+
<ParamField path="text" type="string | number" required>
|
|
21
|
+
Text to type (can be a string or number)
|
|
22
|
+
</ParamField>
|
|
23
|
+
|
|
24
|
+
<ParamField path="options" type="object | number">
|
|
25
|
+
Typing options (or legacy delay number)
|
|
26
|
+
|
|
27
|
+
<Expandable title="properties">
|
|
28
|
+
<ParamField path="delay" type="number" default={250}>
|
|
29
|
+
Delay between keystrokes in milliseconds
|
|
30
|
+
</ParamField>
|
|
31
|
+
|
|
32
|
+
<ParamField path="secret" type="boolean" default={false}>
|
|
33
|
+
If `true`, treats text as sensitive data (won't be logged or stored in debug info/dashcam)
|
|
34
|
+
</ParamField>
|
|
35
|
+
</Expandable>
|
|
36
|
+
</ParamField>
|
|
37
|
+
|
|
38
|
+
## Returns
|
|
39
|
+
|
|
40
|
+
`Promise<void>`
|
|
41
|
+
|
|
42
|
+
## Examples
|
|
43
|
+
|
|
44
|
+
### Basic Typing
|
|
45
|
+
|
|
46
|
+
```javascript
|
|
47
|
+
// Type text
|
|
48
|
+
await testdriver.type('hello@example.com');
|
|
49
|
+
|
|
50
|
+
// Type numbers
|
|
51
|
+
await testdriver.type(12345);
|
|
52
|
+
|
|
53
|
+
// Type with custom delay (legacy syntax)
|
|
54
|
+
await testdriver.type('slow typing', 500); // 500ms between each character
|
|
55
|
+
|
|
56
|
+
// Type with options object
|
|
57
|
+
await testdriver.type('text', { delay: 500 });
|
|
58
|
+
```
|
|
59
|
+
|
|
60
|
+
### Password/Secret Handling
|
|
61
|
+
|
|
62
|
+
```javascript
|
|
63
|
+
// ✅ SECURE - Mark as secret to prevent logging
|
|
64
|
+
const passwordField = await testdriver.find('password input');
|
|
65
|
+
await passwordField.click();
|
|
66
|
+
await testdriver.type('MySecureP@ssw0rd', { secret: true });
|
|
67
|
+
// Password NOT logged in dashcam or debug output
|
|
68
|
+
|
|
69
|
+
// ❌ INSECURE - Password will be logged
|
|
70
|
+
await testdriver.type('MySecureP@ssw0rd');
|
|
71
|
+
// Password appears in logs, dashcam replay, and debug info
|
|
72
|
+
|
|
73
|
+
// Use secret for any sensitive data
|
|
74
|
+
await testdriver.find('api key input').click();
|
|
75
|
+
await testdriver.type('sk-1234567890abcdef', { secret: true });
|
|
76
|
+
|
|
77
|
+
await testdriver.find('credit card input').click();
|
|
78
|
+
await testdriver.type('4111111111111111', { secret: true });
|
|
79
|
+
```
|
|
80
|
+
|
|
81
|
+
<Warning>
|
|
82
|
+
**Always use `secret: true` for passwords and sensitive data!**
|
|
83
|
+
|
|
84
|
+
Without this option, typed text appears in:
|
|
85
|
+
- Dashcam video replays
|
|
86
|
+
- TestDriver logs
|
|
87
|
+
- Debug screenshots
|
|
88
|
+
- Error messages
|
|
89
|
+
</Warning>
|
|
90
|
+
|
|
91
|
+
### Form Filling
|
|
92
|
+
|
|
93
|
+
```javascript
|
|
94
|
+
// Focus field and type
|
|
95
|
+
const emailField = await testdriver.find('email input');
|
|
96
|
+
await emailField.click();
|
|
97
|
+
await testdriver.type('user@example.com');
|
|
98
|
+
|
|
99
|
+
// Tab to next field and type
|
|
100
|
+
await testdriver.pressKeys(['tab']);
|
|
101
|
+
await testdriver.type('John Doe');
|
|
102
|
+
|
|
103
|
+
// Type password securely
|
|
104
|
+
await testdriver.pressKeys(['tab']);
|
|
105
|
+
await testdriver.type('MySecureP@ssw0rd', { secret: true });
|
|
106
|
+
```
|
|
107
|
+
|
|
108
|
+
### Clearing and Replacing Text
|
|
109
|
+
|
|
110
|
+
```javascript
|
|
111
|
+
const searchBox = await testdriver.find('search input');
|
|
112
|
+
await searchBox.click();
|
|
113
|
+
|
|
114
|
+
// Clear existing text
|
|
115
|
+
await testdriver.pressKeys(['ctrl', 'a']); // Select all
|
|
116
|
+
await testdriver.type('new search query');
|
|
117
|
+
```
|
|
118
|
+
|
|
119
|
+
## Best Practices
|
|
120
|
+
|
|
121
|
+
<Check>
|
|
122
|
+
**Focus the field first**
|
|
123
|
+
|
|
124
|
+
Always click the input field or navigate to it before typing:
|
|
125
|
+
|
|
126
|
+
```javascript
|
|
127
|
+
const input = await testdriver.find('username input');
|
|
128
|
+
await input.click();
|
|
129
|
+
await testdriver.type('testuser');
|
|
130
|
+
```
|
|
131
|
+
</Check>
|
|
132
|
+
|
|
133
|
+
<Check>
|
|
134
|
+
**Use Tab for navigation**
|
|
135
|
+
|
|
136
|
+
Navigate between fields using Tab instead of clicking each one:
|
|
137
|
+
|
|
138
|
+
```javascript
|
|
139
|
+
const firstField = await testdriver.find('first name');
|
|
140
|
+
await firstField.click();
|
|
141
|
+
await testdriver.type('John');
|
|
142
|
+
|
|
143
|
+
await testdriver.pressKeys(['tab']);
|
|
144
|
+
await testdriver.type('Doe');
|
|
145
|
+
|
|
146
|
+
await testdriver.pressKeys(['tab']);
|
|
147
|
+
await testdriver.type('john@example.com');
|
|
148
|
+
```
|
|
149
|
+
</Check>
|
|
150
|
+
|
|
151
|
+
<Check>
|
|
152
|
+
**Clear fields before typing**
|
|
153
|
+
|
|
154
|
+
Clear existing content to avoid appending:
|
|
155
|
+
|
|
156
|
+
```javascript
|
|
157
|
+
const input = await testdriver.find('search field');
|
|
158
|
+
await input.click();
|
|
159
|
+
await testdriver.pressKeys(['ctrl', 'a']); // Select all
|
|
160
|
+
await testdriver.type('new search');
|
|
161
|
+
```
|
|
162
|
+
</Check>
|
|
163
|
+
|
|
164
|
+
<Warning>
|
|
165
|
+
**Field must be focused**
|
|
166
|
+
|
|
167
|
+
Typing will only work if an input field is currently focused. If no field is focused, the text may be lost or trigger unexpected keyboard shortcuts.
|
|
168
|
+
</Warning>
|
|
169
|
+
|
|
170
|
+
## Use Cases
|
|
171
|
+
|
|
172
|
+
<AccordionGroup>
|
|
173
|
+
<Accordion title="Login Forms">
|
|
174
|
+
```javascript
|
|
175
|
+
await testdriver.focusApplication('Google Chrome');
|
|
176
|
+
|
|
177
|
+
const usernameField = await testdriver.find('username input');
|
|
178
|
+
await usernameField.click();
|
|
179
|
+
await testdriver.type('testuser@example.com');
|
|
180
|
+
|
|
181
|
+
await testdriver.pressKeys(['tab']);
|
|
182
|
+
await testdriver.type('MyP@ssword123', { secret: true });
|
|
183
|
+
|
|
184
|
+
await testdriver.pressKeys(['enter']);
|
|
185
|
+
```
|
|
186
|
+
</Accordion>
|
|
187
|
+
|
|
188
|
+
<Accordion title="Search Fields">
|
|
189
|
+
```javascript
|
|
190
|
+
const searchBox = await testdriver.find('search input');
|
|
191
|
+
await searchBox.click();
|
|
192
|
+
await testdriver.type('laptop computers');
|
|
193
|
+
await testdriver.pressKeys(['enter']);
|
|
194
|
+
|
|
195
|
+
// Wait for results
|
|
196
|
+
await new Promise(r => setTimeout(r, 2000));
|
|
197
|
+
```
|
|
198
|
+
</Accordion>
|
|
199
|
+
|
|
200
|
+
<Accordion title="Multi-Field Forms">
|
|
201
|
+
```javascript
|
|
202
|
+
// First field
|
|
203
|
+
const nameField = await testdriver.find('full name input');
|
|
204
|
+
await nameField.click();
|
|
205
|
+
await testdriver.type('Jane Smith');
|
|
206
|
+
|
|
207
|
+
// Navigate with Tab
|
|
208
|
+
await testdriver.pressKeys(['tab']);
|
|
209
|
+
await testdriver.type('jane.smith@example.com');
|
|
210
|
+
|
|
211
|
+
await testdriver.pressKeys(['tab']);
|
|
212
|
+
await testdriver.type('+1-555-0123');
|
|
213
|
+
|
|
214
|
+
await testdriver.pressKeys(['tab']);
|
|
215
|
+
await testdriver.type('123 Main Street');
|
|
216
|
+
```
|
|
217
|
+
</Accordion>
|
|
218
|
+
|
|
219
|
+
<Accordion title="Text Editors">
|
|
220
|
+
```javascript
|
|
221
|
+
const editor = await testdriver.find('text editor area');
|
|
222
|
+
await editor.click();
|
|
223
|
+
|
|
224
|
+
await testdriver.type('# My Document', 100);
|
|
225
|
+
await testdriver.pressKeys(['enter', 'enter']);
|
|
226
|
+
await testdriver.type('This is the first paragraph.', 50);
|
|
227
|
+
```
|
|
228
|
+
</Accordion>
|
|
229
|
+
|
|
230
|
+
<Accordion title="Numeric Input">
|
|
231
|
+
```javascript
|
|
232
|
+
const quantityField = await testdriver.find('quantity input');
|
|
233
|
+
await quantityField.click();
|
|
234
|
+
|
|
235
|
+
// Clear field
|
|
236
|
+
await testdriver.pressKeys(['ctrl', 'a']);
|
|
237
|
+
|
|
238
|
+
// Type number
|
|
239
|
+
await testdriver.type(5);
|
|
240
|
+
```
|
|
241
|
+
</Accordion>
|
|
242
|
+
</AccordionGroup>
|
|
243
|
+
|
|
244
|
+
## Typing Speed
|
|
245
|
+
|
|
246
|
+
Adjust the delay parameter based on your needs:
|
|
247
|
+
|
|
248
|
+
```javascript
|
|
249
|
+
// Fast typing (100ms delay)
|
|
250
|
+
await testdriver.type('quick entry', 100);
|
|
251
|
+
|
|
252
|
+
// Normal typing (250ms - default)
|
|
253
|
+
await testdriver.type('standard speed');
|
|
254
|
+
|
|
255
|
+
// Slow typing (500ms delay) - useful for fields with live validation
|
|
256
|
+
await testdriver.type('slow and steady', 500);
|
|
257
|
+
|
|
258
|
+
// Very slow (1000ms delay) - for problematic fields
|
|
259
|
+
await testdriver.type('one by one', 1000);
|
|
260
|
+
```
|
|
261
|
+
|
|
262
|
+
<Note>
|
|
263
|
+
Some applications with live validation or autocomplete may require slower typing speeds to avoid race conditions.
|
|
264
|
+
</Note>
|
|
265
|
+
|
|
266
|
+
## Special Characters
|
|
267
|
+
|
|
268
|
+
```javascript
|
|
269
|
+
// Email addresses
|
|
270
|
+
await testdriver.type('user@example.com');
|
|
271
|
+
|
|
272
|
+
// URLs
|
|
273
|
+
await testdriver.type('https://example.com/path?query=value');
|
|
274
|
+
|
|
275
|
+
// Passwords with special characters
|
|
276
|
+
await testdriver.type('P@ssw0rd!#$%');
|
|
277
|
+
|
|
278
|
+
// Paths
|
|
279
|
+
await testdriver.type('C:\\Users\\Documents\\file.txt');
|
|
280
|
+
|
|
281
|
+
// Multi-line text (use pressKeys for Enter)
|
|
282
|
+
await testdriver.type('Line 1');
|
|
283
|
+
await testdriver.pressKeys(['enter']);
|
|
284
|
+
await testdriver.type('Line 2');
|
|
285
|
+
```
|
|
286
|
+
|
|
287
|
+
## Complete Example
|
|
288
|
+
|
|
289
|
+
```javascript
|
|
290
|
+
import { beforeAll, afterAll, describe, it, expect } from 'vitest';
|
|
291
|
+
import TestDriver from 'testdriverai';
|
|
292
|
+
|
|
293
|
+
describe('Form Filling with Type', () => {
|
|
294
|
+
let testdriver;
|
|
295
|
+
|
|
296
|
+
beforeAll(async () => {
|
|
297
|
+
client = new TestDriver(process.env.TD_API_KEY);
|
|
298
|
+
await testdriver.auth();
|
|
299
|
+
await testdriver.connect();
|
|
300
|
+
});
|
|
301
|
+
|
|
302
|
+
afterAll(async () => {
|
|
303
|
+
await testdriver.disconnect();
|
|
304
|
+
});
|
|
305
|
+
|
|
306
|
+
it('should fill out registration form', async () => {
|
|
307
|
+
await testdriver.focusApplication('Google Chrome');
|
|
308
|
+
|
|
309
|
+
// Email field
|
|
310
|
+
const emailField = await testdriver.find('email input field');
|
|
311
|
+
await emailField.click();
|
|
312
|
+
await testdriver.type('john.doe@example.com');
|
|
313
|
+
|
|
314
|
+
// Tab through form fields
|
|
315
|
+
await testdriver.pressKeys(['tab']);
|
|
316
|
+
await testdriver.type('John');
|
|
317
|
+
|
|
318
|
+
await testdriver.pressKeys(['tab']);
|
|
319
|
+
await testdriver.type('Doe');
|
|
320
|
+
|
|
321
|
+
await testdriver.pressKeys(['tab']);
|
|
322
|
+
await testdriver.type('MySecureP@ssword123');
|
|
323
|
+
|
|
324
|
+
await testdriver.pressKeys(['tab']);
|
|
325
|
+
await testdriver.type('MySecureP@ssword123'); // Confirm password
|
|
326
|
+
|
|
327
|
+
// Verify fields were filled
|
|
328
|
+
const result = await testdriver.assert('all form fields are filled');
|
|
329
|
+
expect(result).toBeTruthy();
|
|
330
|
+
});
|
|
331
|
+
|
|
332
|
+
it('should update search query', async () => {
|
|
333
|
+
const searchBox = await testdriver.find('search input');
|
|
334
|
+
await searchBox.click();
|
|
335
|
+
|
|
336
|
+
// Type initial search
|
|
337
|
+
await testdriver.type('laptops');
|
|
338
|
+
await testdriver.pressKeys(['enter']);
|
|
339
|
+
|
|
340
|
+
await new Promise(r => setTimeout(r, 2000));
|
|
341
|
+
|
|
342
|
+
// Update search
|
|
343
|
+
await searchBox.click();
|
|
344
|
+
await testdriver.pressKeys(['ctrl', 'a']); // Select all
|
|
345
|
+
await testdriver.type('gaming laptops');
|
|
346
|
+
await testdriver.pressKeys(['enter']);
|
|
347
|
+
|
|
348
|
+
// Verify new search
|
|
349
|
+
await testdriver.assert('search results for "gaming laptops" are shown');
|
|
350
|
+
});
|
|
351
|
+
});
|
|
352
|
+
```
|
|
353
|
+
|
|
354
|
+
## Related Methods
|
|
355
|
+
|
|
356
|
+
- [`pressKeys()`](/v7/press-keys) - Press keyboard keys and shortcuts
|
|
357
|
+
- [`find()`](/v7/find) - Locate input fields
|
|
358
|
+
- [`click()`](/v7/click) - Focus input fields
|
|
@@ -0,0 +1,111 @@
|
|
|
1
|
+
---
|
|
2
|
+
title: "Using Variables"
|
|
3
|
+
description: "Use dynamic data and secure secrets in your tests"
|
|
4
|
+
icon: "square-root-variable"
|
|
5
|
+
---
|
|
6
|
+
|
|
7
|
+
Scale your testing with dynamic data and secure secrets management. Choose the right approach based on your testing needs.
|
|
8
|
+
|
|
9
|
+
## Environment Variables
|
|
10
|
+
|
|
11
|
+
Environment variables are ideal for **configuration that changes between environments** (dev, staging, production) or for **secrets that shouldn't be committed to code**. Use this approach when you need to run the same tests against different servers or with different credentials.
|
|
12
|
+
|
|
13
|
+
```javascript
|
|
14
|
+
import { test } from 'vitest';
|
|
15
|
+
import { chrome } from 'testdriverai/presets';
|
|
16
|
+
|
|
17
|
+
test('multi-environment testing', async (context) => {
|
|
18
|
+
const env = process.env.TEST_ENV || 'staging';
|
|
19
|
+
const urls = {
|
|
20
|
+
dev: 'https://dev.myapp.com',
|
|
21
|
+
staging: 'https://staging.myapp.com',
|
|
22
|
+
production: 'https://myapp.com'
|
|
23
|
+
};
|
|
24
|
+
|
|
25
|
+
const { testdriver } = await chrome(context, {
|
|
26
|
+
url: urls[env]
|
|
27
|
+
});
|
|
28
|
+
|
|
29
|
+
await testdriver.assert('app is running');
|
|
30
|
+
});
|
|
31
|
+
```
|
|
32
|
+
|
|
33
|
+
```bash
|
|
34
|
+
# Run against different environments
|
|
35
|
+
TEST_ENV=dev vitest run
|
|
36
|
+
TEST_ENV=staging vitest run
|
|
37
|
+
TEST_ENV=production vitest run
|
|
38
|
+
```
|
|
39
|
+
|
|
40
|
+
## Test Fixtures
|
|
41
|
+
|
|
42
|
+
Test fixtures work best when you have **structured, reusable test data** that needs to be shared across multiple tests. Use fixtures when testing different user roles, product catalogs, or any scenario where you want to parameterize tests with a known set of data.
|
|
43
|
+
|
|
44
|
+
```javascript test/fixtures/users.js
|
|
45
|
+
export const testUsers = [
|
|
46
|
+
{ email: 'admin@test.com', role: 'admin' },
|
|
47
|
+
{ email: 'user@test.com', role: 'user' },
|
|
48
|
+
{ email: 'guest@test.com', role: 'guest' }
|
|
49
|
+
];
|
|
50
|
+
|
|
51
|
+
export const products = [
|
|
52
|
+
{ name: 'Laptop', price: 999 },
|
|
53
|
+
{ name: 'Mouse', price: 29 },
|
|
54
|
+
{ name: 'Keyboard', price: 89 }
|
|
55
|
+
];
|
|
56
|
+
```
|
|
57
|
+
|
|
58
|
+
```javascript test/permissions.test.js
|
|
59
|
+
import { test } from 'vitest';
|
|
60
|
+
import { chrome } from 'testdriverai/presets';
|
|
61
|
+
import { testUsers } from './fixtures/users.js';
|
|
62
|
+
|
|
63
|
+
test.each(testUsers)('$role can access dashboard', async ({ email, role }, context) => {
|
|
64
|
+
const { testdriver } = await chrome(context, { url });
|
|
65
|
+
|
|
66
|
+
await testdriver.find('email input').type(email);
|
|
67
|
+
await testdriver.find('password input').type('password123');
|
|
68
|
+
await testdriver.find('login button').click();
|
|
69
|
+
|
|
70
|
+
if (role === 'admin') {
|
|
71
|
+
await testdriver.assert('admin panel is visible');
|
|
72
|
+
} else {
|
|
73
|
+
await testdriver.assert('user dashboard is visible');
|
|
74
|
+
}
|
|
75
|
+
});
|
|
76
|
+
```
|
|
77
|
+
|
|
78
|
+
## Dynamic Data Generation
|
|
79
|
+
|
|
80
|
+
Dynamic data generation is perfect for **creating unique test data on each run**, avoiding conflicts with existing records, and **testing edge cases with realistic data**. Use libraries like Faker when you need fresh emails, names, or other data that won't collide with previous test runs.
|
|
81
|
+
|
|
82
|
+
```javascript
|
|
83
|
+
import { test } from 'vitest';
|
|
84
|
+
import { chrome } from 'testdriverai/presets';
|
|
85
|
+
import { faker } from '@faker-js/faker';
|
|
86
|
+
|
|
87
|
+
test('user registration with dynamic data', async (context) => {
|
|
88
|
+
const { testdriver } = await chrome(context, { url });
|
|
89
|
+
|
|
90
|
+
// Generate unique test data for each run
|
|
91
|
+
const userData = {
|
|
92
|
+
firstName: faker.person.firstName(),
|
|
93
|
+
lastName: faker.person.lastName(),
|
|
94
|
+
email: faker.internet.email(),
|
|
95
|
+
password: faker.internet.password({ length: 12 })
|
|
96
|
+
};
|
|
97
|
+
|
|
98
|
+
await testdriver.find('first name input').type(userData.firstName);
|
|
99
|
+
await testdriver.find('last name input').type(userData.lastName);
|
|
100
|
+
await testdriver.find('email input').type(userData.email);
|
|
101
|
+
await testdriver.find('password input').type(userData.password);
|
|
102
|
+
await testdriver.find('register button').click();
|
|
103
|
+
|
|
104
|
+
await testdriver.assert('registration successful');
|
|
105
|
+
console.log('Registered user:', userData.email);
|
|
106
|
+
});
|
|
107
|
+
```
|
|
108
|
+
|
|
109
|
+
```bash
|
|
110
|
+
npm install --save-dev @faker-js/faker
|
|
111
|
+
```
|
package/docs/v7/wait.mdx
ADDED
|
@@ -0,0 +1,52 @@
|
|
|
1
|
+
---
|
|
2
|
+
title: "wait"
|
|
3
|
+
sidebarTitle: "wait"
|
|
4
|
+
description: "Pause the execution of the script for a specified duration."
|
|
5
|
+
icon: "clock"
|
|
6
|
+
"mode": "wide"
|
|
7
|
+
---
|
|
8
|
+
|
|
9
|
+
## Description
|
|
10
|
+
|
|
11
|
+
The `wait` method pauses test execution for a specified number of milliseconds before continuing. This is useful for adding delays between actions, waiting for animations to complete, or pausing for state changes to settle.
|
|
12
|
+
|
|
13
|
+
## Syntax
|
|
14
|
+
|
|
15
|
+
```javascript
|
|
16
|
+
await testdriver.wait(timeout);
|
|
17
|
+
```
|
|
18
|
+
|
|
19
|
+
## Arguments
|
|
20
|
+
|
|
21
|
+
| Argument | Type | Default | Description |
|
|
22
|
+
| --------- | -------- | ------- | ------------------------------------- |
|
|
23
|
+
| `timeout` | `number` | `3000` | The duration in milliseconds to wait. |
|
|
24
|
+
|
|
25
|
+
## Examples
|
|
26
|
+
|
|
27
|
+
```javascript
|
|
28
|
+
// Wait 2 seconds for an animation to complete
|
|
29
|
+
await testdriver.find('submit button').click();
|
|
30
|
+
await testdriver.wait(2000);
|
|
31
|
+
|
|
32
|
+
// Wait 5 seconds
|
|
33
|
+
await testdriver.wait(5000);
|
|
34
|
+
|
|
35
|
+
// Wait with default timeout (3 seconds)
|
|
36
|
+
await testdriver.wait();
|
|
37
|
+
```
|
|
38
|
+
|
|
39
|
+
## Best Practices
|
|
40
|
+
|
|
41
|
+
- **Use for simple delays** — waiting for animations, transitions, or state changes after an action.
|
|
42
|
+
- **Avoid for element waiting** — if you're waiting for a specific element to appear, use `find()` with a `timeout` option instead:
|
|
43
|
+
```javascript
|
|
44
|
+
// ✅ Better for waiting for elements
|
|
45
|
+
const element = await testdriver.find('success message', { timeout: 30000 });
|
|
46
|
+
|
|
47
|
+
// ❌ Don't do this for element waiting
|
|
48
|
+
await testdriver.wait(5000);
|
|
49
|
+
const element = await testdriver.find('success message');
|
|
50
|
+
```
|
|
51
|
+
- Avoid excessively long timeouts to keep tests efficient.
|
|
52
|
+
- Use sparingly — TestDriver's [redraw detection](/v7/waiting-for-elements) automatically waits for screen and network stability after each action.
|
|
@@ -0,0 +1,90 @@
|
|
|
1
|
+
---
|
|
2
|
+
title: "Waiting for Elements"
|
|
3
|
+
description: "Handle async operations and prevent flaky tests"
|
|
4
|
+
icon: "clock"
|
|
5
|
+
---
|
|
6
|
+
|
|
7
|
+
## Waiting for Elements
|
|
8
|
+
|
|
9
|
+
By default, `find()` automatically polls for up to 10 seconds, retrying every 5 seconds until the element is found. This means most elements that appear after short async operations will be found without any extra configuration.
|
|
10
|
+
|
|
11
|
+
For longer operations, increase the `timeout`:
|
|
12
|
+
|
|
13
|
+
```javascript
|
|
14
|
+
// Default behavior - polls for up to 10 seconds automatically
|
|
15
|
+
const element = await testdriver.find('Loading complete indicator');
|
|
16
|
+
await element.click();
|
|
17
|
+
|
|
18
|
+
// Wait up to 30 seconds for slower operations
|
|
19
|
+
const element = await testdriver.find('Loading complete indicator', { timeout: 30000 });
|
|
20
|
+
await element.click();
|
|
21
|
+
|
|
22
|
+
// Useful after actions that trigger loading states
|
|
23
|
+
await testdriver.find('submit button').click();
|
|
24
|
+
await testdriver.find('success message', { timeout: 15000 });
|
|
25
|
+
|
|
26
|
+
// Disable polling for instant checks
|
|
27
|
+
const toast = await testdriver.find('notification toast', { timeout: 0 });
|
|
28
|
+
```
|
|
29
|
+
|
|
30
|
+
## Flake Prevention
|
|
31
|
+
|
|
32
|
+
TestDriver automatically waits for the screen and network to stabilize after each action using **redraw detection**. This prevents flaky tests caused by animations, loading states, or dynamic content updates.
|
|
33
|
+
|
|
34
|
+
<Note>
|
|
35
|
+
Redraw detection adds a small delay after each action but significantly reduces test flakiness.
|
|
36
|
+
</Note>
|
|
37
|
+
|
|
38
|
+
For example, when clicking a submit button that navigates to a new page:
|
|
39
|
+
|
|
40
|
+
```javascript
|
|
41
|
+
// Click submit - TestDriver automatically waits for the new page to load
|
|
42
|
+
await testdriver.find('submit button').click();
|
|
43
|
+
|
|
44
|
+
// By the time this runs, the page has fully loaded and stabilized
|
|
45
|
+
await testdriver.assert('dashboard is displayed');
|
|
46
|
+
await testdriver.find('welcome message');
|
|
47
|
+
```
|
|
48
|
+
|
|
49
|
+
Without redraw detection, you'd need manual waits or retries to handle the page transition. TestDriver handles this automatically by detecting when the screen stops changing and network requests complete.
|
|
50
|
+
|
|
51
|
+
You can disable redraw detection or customize its behavior:
|
|
52
|
+
|
|
53
|
+
```javascript
|
|
54
|
+
// Disable redraw detection for faster tests (less reliable)
|
|
55
|
+
const testdriver = TestDriver(context, {
|
|
56
|
+
redraw: false
|
|
57
|
+
});
|
|
58
|
+
```
|
|
59
|
+
|
|
60
|
+
Here is an example of customizing redraw detection:
|
|
61
|
+
|
|
62
|
+
```javascript
|
|
63
|
+
// Fine-tune redraw detection
|
|
64
|
+
const testdriver = TestDriver(context, {
|
|
65
|
+
redraw: {
|
|
66
|
+
enabled: true,
|
|
67
|
+
diffThreshold: 0.1, // Pixel difference threshold (0-1)
|
|
68
|
+
screenRedraw: true, // Monitor screen changes
|
|
69
|
+
networkMonitor: true, // Wait for network idle
|
|
70
|
+
}
|
|
71
|
+
});
|
|
72
|
+
```
|
|
73
|
+
|
|
74
|
+
## Simple Delays with `wait()`
|
|
75
|
+
|
|
76
|
+
For simple pauses — waiting for animations, transitions, or state changes after an action — use `wait()`:
|
|
77
|
+
|
|
78
|
+
```javascript
|
|
79
|
+
// Wait for an animation to complete
|
|
80
|
+
await testdriver.find('menu toggle').click();
|
|
81
|
+
await testdriver.wait(2000);
|
|
82
|
+
|
|
83
|
+
// Wait for a page transition to settle
|
|
84
|
+
await testdriver.find('next page button').click();
|
|
85
|
+
await testdriver.wait(1000);
|
|
86
|
+
```
|
|
87
|
+
|
|
88
|
+
<Note>
|
|
89
|
+
For waiting for specific **elements** to appear, prefer `find()` with a `timeout` option. Use `wait()` only for simple time-based pauses.
|
|
90
|
+
</Note>
|
|
@@ -0,0 +1,54 @@
|
|
|
1
|
+
---
|
|
2
|
+
title: "What is TestDriver?"
|
|
3
|
+
description: "Reliably test your most difficult user flows"
|
|
4
|
+
icon: "circle-info"
|
|
5
|
+
---
|
|
6
|
+
|
|
7
|
+
## The problem with modern testing tools
|
|
8
|
+
|
|
9
|
+
Modern testing tools like Playwright are designed to test a single web application, running in a single browser tab using selectors.
|
|
10
|
+
|
|
11
|
+
However, selectors are often either unreliable or unavailable in complex scenarios, leading to brittle and flaky tests:
|
|
12
|
+
|
|
13
|
+
| Challenge | Problem | Examples |
|
|
14
|
+
|-----------|---------|----------|
|
|
15
|
+
| **Fast moving teams** | Frequently change UI structure, breaking CSS/XPath selectors | Agile teams, startups, vibe-coders |
|
|
16
|
+
| **Dynamic content** | Cannot be targeted with selectors | AI chatbots, PDFs, images, videos |
|
|
17
|
+
| **Software you don't own** | May lack proper accessibility attributes | Other websites, extensions, third-party applications |
|
|
18
|
+
| **Multi-application workflows** | Cannot be tested with web-only tools | Desktop apps, browser extensions, IDEs |
|
|
19
|
+
| **Visual states** | Impossible to verify with code-based selectors | Charts, graphs, videos, images, spelling errors, UI layout |
|
|
20
|
+
|
|
21
|
+
## The TestDriver Solution
|
|
22
|
+
|
|
23
|
+
TestDriver is a complete testing platform built specifically for handling these scenarios. It consists of a Javascript SDK, hosted infrastructure, and debugging tools that make it easy to write, run, and maintain tests for your most difficult user flows.
|
|
24
|
+
|
|
25
|
+
### Javascript SDK
|
|
26
|
+
|
|
27
|
+
Here is an example of a TestDriver test that installs a production Chrome extension from the Chrome Web Store and verifies that it appears in the extensions menu:
|
|
28
|
+
|
|
29
|
+
```javascript Installing Loom from the Chrome Web Store
|
|
30
|
+
import { describe, expect, it } from "vitest";
|
|
31
|
+
import { TestDriver } from "testdriverai/vitest/hooks";
|
|
32
|
+
|
|
33
|
+
describe("Chrome Extension Test", () => {
|
|
34
|
+
const testdriver = TestDriver(context);
|
|
35
|
+
|
|
36
|
+
// Launch Chrome with Loom loaded by its Chrome Web Store ID
|
|
37
|
+
await testdriver.provision.chromeExtension({
|
|
38
|
+
extensionId: 'liecbddmkiiihnedobmlmillhodjkdmb'
|
|
39
|
+
});
|
|
40
|
+
|
|
41
|
+
// Click on the extensions button (puzzle piece icon) in Chrome toolbar
|
|
42
|
+
const extensionsButton = await testdriver.find("The puzzle-shaped icon in the Chrome toolbar.");
|
|
43
|
+
await extensionsButton.click();
|
|
44
|
+
|
|
45
|
+
// Look for Loom in the extensions menu
|
|
46
|
+
const loomExtension = await testdriver.find("Loom extension in the extensions dropdown");
|
|
47
|
+
expect(loomExtension.found()).toBeTruthy();
|
|
48
|
+
});
|
|
49
|
+
```
|
|
50
|
+
|
|
51
|
+
|
|
52
|
+
<Tip>[vitest](https://vitest.dev/) is the preferred test runner for TestDriver.</Tip>
|
|
53
|
+
|
|
54
|
+
,
|