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