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