@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
|
@@ -0,0 +1,438 @@
|
|
|
1
|
+
const BaseCommand = require("../lib/base.js");
|
|
2
|
+
const { createCommandDefinitions } = require("../../../agent/interface.js");
|
|
3
|
+
const { initProject } = require("../../../lib/init-project.js");
|
|
4
|
+
const fs = require("fs");
|
|
5
|
+
const path = require("path");
|
|
6
|
+
const chalk = require("chalk");
|
|
7
|
+
const readline = require("readline");
|
|
8
|
+
const os = require("os");
|
|
9
|
+
const { execSync } = require("child_process");
|
|
10
|
+
|
|
11
|
+
// Load .env file for CLI usage (TD_API_ROOT, etc.)
|
|
12
|
+
require("dotenv").config();
|
|
13
|
+
|
|
14
|
+
// API configuration
|
|
15
|
+
const channelConfig = require("../../../lib/resolve-channel.js");
|
|
16
|
+
const API_BASE_URL = process.env.TD_API_ROOT || channelConfig.channels[channelConfig.active];
|
|
17
|
+
const POLL_INTERVAL = 5000; // 5 seconds
|
|
18
|
+
const POLL_TIMEOUT = 900000; // 15 minutes
|
|
19
|
+
|
|
20
|
+
/**
|
|
21
|
+
* Init command - scaffolds Vitest SDK example tests for TestDriver
|
|
22
|
+
*/
|
|
23
|
+
class InitCommand extends BaseCommand {
|
|
24
|
+
async run() {
|
|
25
|
+
await this.parse(InitCommand);
|
|
26
|
+
|
|
27
|
+
console.log(chalk.cyan("\n🚀 Initializing TestDriver project...\n"));
|
|
28
|
+
|
|
29
|
+
// Prompt for API key first
|
|
30
|
+
const apiKey = await this.promptForApiKey();
|
|
31
|
+
|
|
32
|
+
// Helper to print progress messages with appropriate colors
|
|
33
|
+
const printProgress = (msg) => {
|
|
34
|
+
if (msg.startsWith("✓")) {
|
|
35
|
+
console.log(chalk.green(` ${msg}`));
|
|
36
|
+
} else if (msg.startsWith("⚠") || msg.startsWith("ℹ")) {
|
|
37
|
+
console.log(chalk.yellow(` ${msg}`));
|
|
38
|
+
} else if (msg.startsWith("⊘")) {
|
|
39
|
+
console.log(chalk.gray(` ${msg}`));
|
|
40
|
+
} else {
|
|
41
|
+
console.log(` ${msg}`);
|
|
42
|
+
}
|
|
43
|
+
};
|
|
44
|
+
|
|
45
|
+
// Run the shared init logic with real-time progress output
|
|
46
|
+
const result = await initProject({
|
|
47
|
+
targetDir: process.cwd(),
|
|
48
|
+
apiKey: apiKey,
|
|
49
|
+
skipInstall: false,
|
|
50
|
+
onProgress: printProgress,
|
|
51
|
+
});
|
|
52
|
+
|
|
53
|
+
// Print errors if any
|
|
54
|
+
for (const err of result.errors) {
|
|
55
|
+
console.log(chalk.yellow(` ⚠️ ${err}`));
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
// Handle shell profile for API key (CLI-specific feature)
|
|
59
|
+
if (apiKey && apiKey.trim()) {
|
|
60
|
+
this.addToShellProfile("TD_API_KEY", apiKey.trim());
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
if (result.success) {
|
|
64
|
+
console.log(chalk.green("\n✅ Project initialized successfully!\n"));
|
|
65
|
+
this.printNextSteps();
|
|
66
|
+
process.exit(0);
|
|
67
|
+
} else {
|
|
68
|
+
console.log(chalk.red("\n❌ Project initialization completed with errors.\n"));
|
|
69
|
+
process.exit(1);
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
/**
|
|
74
|
+
* Prompt user for API key and save to .env
|
|
75
|
+
* @returns {Promise<string|null>} The API key or null if skipped
|
|
76
|
+
*/
|
|
77
|
+
async promptForApiKey() {
|
|
78
|
+
const envPath = path.join(process.cwd(), ".env");
|
|
79
|
+
|
|
80
|
+
// Check if .env already exists with a valid TD_API_KEY value
|
|
81
|
+
if (fs.existsSync(envPath)) {
|
|
82
|
+
const envContent = fs.readFileSync(envPath, "utf8");
|
|
83
|
+
// Match TD_API_KEY= that's not commented out and has a real value (not empty or placeholder)
|
|
84
|
+
const apiKeyMatch = envContent.match(/^TD_API_KEY=(.+)$/m);
|
|
85
|
+
if (apiKeyMatch) {
|
|
86
|
+
const value = apiKeyMatch[1].trim();
|
|
87
|
+
// Skip only if there's a real value (not empty or placeholder text)
|
|
88
|
+
if (value && value !== "your_api_key" && !value.startsWith("<") && !value.startsWith("$")) {
|
|
89
|
+
console.log(
|
|
90
|
+
chalk.gray("\n API key already configured in .env, skipping...\n"),
|
|
91
|
+
);
|
|
92
|
+
return null;
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
console.log(chalk.cyan(" Setting up your TestDriver API key...\n"));
|
|
98
|
+
|
|
99
|
+
// Ask user how they want to authenticate
|
|
100
|
+
const choice = await this.askChoice(
|
|
101
|
+
" How would you like to authenticate?\n",
|
|
102
|
+
[
|
|
103
|
+
{ key: "1", label: "Login with browser", description: "(recommended)" },
|
|
104
|
+
{ key: "2", label: "Enter API key manually", description: "" },
|
|
105
|
+
],
|
|
106
|
+
);
|
|
107
|
+
|
|
108
|
+
if (choice === "1") {
|
|
109
|
+
// Browser login flow
|
|
110
|
+
try {
|
|
111
|
+
const apiKey = await this.browserLogin();
|
|
112
|
+
if (apiKey) {
|
|
113
|
+
console.log(chalk.green("\n ✓ Logged in successfully!\n"));
|
|
114
|
+
return apiKey;
|
|
115
|
+
}
|
|
116
|
+
} catch (error) {
|
|
117
|
+
console.log(chalk.yellow(`\n ⚠️ Browser login failed: ${error.message}\n`));
|
|
118
|
+
console.log(chalk.gray(" Falling back to manual API key entry...\n"));
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
// Manual API key entry
|
|
123
|
+
console.log(
|
|
124
|
+
chalk.gray(" Get your API key from: https://console.testdriver.ai/team\n"),
|
|
125
|
+
);
|
|
126
|
+
|
|
127
|
+
// Ask if user wants to open the browser
|
|
128
|
+
const shouldOpen = await this.askYesNo(
|
|
129
|
+
" Open API keys page in browser? (Y/n): ",
|
|
130
|
+
);
|
|
131
|
+
if (shouldOpen) {
|
|
132
|
+
try {
|
|
133
|
+
const open = (await import("open")).default;
|
|
134
|
+
await open("https://console.testdriver.ai/team");
|
|
135
|
+
console.log(chalk.gray(" Opening browser...\n"));
|
|
136
|
+
} catch (error) {
|
|
137
|
+
console.log(
|
|
138
|
+
chalk.yellow(" ⚠️ Could not open browser automatically\n"),
|
|
139
|
+
);
|
|
140
|
+
}
|
|
141
|
+
}
|
|
142
|
+
|
|
143
|
+
// Prompt for API key with hidden input
|
|
144
|
+
const apiKey = await this.promptHidden(
|
|
145
|
+
" Enter your API key (input will be hidden): ",
|
|
146
|
+
);
|
|
147
|
+
|
|
148
|
+
if (apiKey && apiKey.trim()) {
|
|
149
|
+
console.log(chalk.green("\n ✓ API key will be saved\n"));
|
|
150
|
+
return apiKey.trim();
|
|
151
|
+
} else {
|
|
152
|
+
console.log(
|
|
153
|
+
chalk.yellow(
|
|
154
|
+
"\n ⚠️ No API key entered. You can add it later to .env:\n",
|
|
155
|
+
),
|
|
156
|
+
);
|
|
157
|
+
console.log(chalk.gray(" TD_API_KEY=your_api_key\n"));
|
|
158
|
+
return null;
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
/**
|
|
163
|
+
* Browser-based login flow using device code
|
|
164
|
+
* @returns {Promise<string>} The API key
|
|
165
|
+
*/
|
|
166
|
+
async browserLogin() {
|
|
167
|
+
// Step 1: Create device code
|
|
168
|
+
process.stdout.write(chalk.gray(" Requesting authorization code..."));
|
|
169
|
+
|
|
170
|
+
const createResponse = await fetch(`${API_BASE_URL}/auth/device/code`, {
|
|
171
|
+
method: "POST",
|
|
172
|
+
headers: { "Content-Type": "application/json" },
|
|
173
|
+
});
|
|
174
|
+
|
|
175
|
+
if (!createResponse.ok) {
|
|
176
|
+
throw new Error("Failed to create device code");
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
const { device_code, verification_uri, expires_in, interval } = await createResponse.json();
|
|
180
|
+
console.log(chalk.green(" done\n"));
|
|
181
|
+
|
|
182
|
+
// Step 2: Open browser
|
|
183
|
+
console.log(chalk.cyan(` Opening browser to authorize CLI...\n`));
|
|
184
|
+
console.log(chalk.gray(` If browser doesn't open, visit:\n ${verification_uri}\n`));
|
|
185
|
+
|
|
186
|
+
try {
|
|
187
|
+
const open = (await import("open")).default;
|
|
188
|
+
await open(verification_uri);
|
|
189
|
+
} catch (error) {
|
|
190
|
+
// Browser didn't open, user can use the URL manually
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
// Step 3: Poll for token
|
|
194
|
+
const pollInterval = (interval || 5) * 1000;
|
|
195
|
+
const timeout = (expires_in || 900) * 1000;
|
|
196
|
+
const startTime = Date.now();
|
|
197
|
+
|
|
198
|
+
process.stdout.write(chalk.gray(" Waiting for authorization..."));
|
|
199
|
+
|
|
200
|
+
// Start spinner
|
|
201
|
+
const spinnerFrames = ["⠋", "⠙", "⠹", "⠸", "⠼", "⠴", "⠦", "⠧", "⠇", "⠏"];
|
|
202
|
+
let spinnerIndex = 0;
|
|
203
|
+
const spinnerInterval = setInterval(() => {
|
|
204
|
+
process.stdout.write(`\r Waiting for authorization... ${spinnerFrames[spinnerIndex]}`);
|
|
205
|
+
spinnerIndex = (spinnerIndex + 1) % spinnerFrames.length;
|
|
206
|
+
}, 100);
|
|
207
|
+
|
|
208
|
+
try {
|
|
209
|
+
while (Date.now() - startTime < timeout) {
|
|
210
|
+
await this.sleep(pollInterval);
|
|
211
|
+
|
|
212
|
+
const tokenResponse = await fetch(`${API_BASE_URL}/auth/device/token`, {
|
|
213
|
+
method: "POST",
|
|
214
|
+
headers: { "Content-Type": "application/json" },
|
|
215
|
+
body: JSON.stringify({ deviceCode: device_code }),
|
|
216
|
+
});
|
|
217
|
+
|
|
218
|
+
const data = await tokenResponse.json();
|
|
219
|
+
|
|
220
|
+
if (tokenResponse.ok && data.apiKey) {
|
|
221
|
+
clearInterval(spinnerInterval);
|
|
222
|
+
process.stdout.write("\r Waiting for authorization... " + chalk.green("✓") + "\n");
|
|
223
|
+
return data.apiKey;
|
|
224
|
+
}
|
|
225
|
+
|
|
226
|
+
if (data.error === "expired_token") {
|
|
227
|
+
clearInterval(spinnerInterval);
|
|
228
|
+
throw new Error("Authorization timed out. Please try again.");
|
|
229
|
+
}
|
|
230
|
+
|
|
231
|
+
// authorization_pending - continue polling
|
|
232
|
+
}
|
|
233
|
+
|
|
234
|
+
clearInterval(spinnerInterval);
|
|
235
|
+
throw new Error("Authorization timed out. Please try again.");
|
|
236
|
+
} catch (error) {
|
|
237
|
+
clearInterval(spinnerInterval);
|
|
238
|
+
process.stdout.write("\n");
|
|
239
|
+
throw error;
|
|
240
|
+
}
|
|
241
|
+
}
|
|
242
|
+
|
|
243
|
+
/**
|
|
244
|
+
* Ask user to choose from a list of options
|
|
245
|
+
*/
|
|
246
|
+
async askChoice(question, options) {
|
|
247
|
+
return new Promise((resolve) => {
|
|
248
|
+
const rl = readline.createInterface({
|
|
249
|
+
input: process.stdin,
|
|
250
|
+
output: process.stdout,
|
|
251
|
+
});
|
|
252
|
+
|
|
253
|
+
console.log(question);
|
|
254
|
+
for (const opt of options) {
|
|
255
|
+
const desc = opt.description ? chalk.gray(` ${opt.description}`) : "";
|
|
256
|
+
console.log(` ${chalk.cyan(opt.key)}. ${opt.label}${desc}`);
|
|
257
|
+
}
|
|
258
|
+
console.log("");
|
|
259
|
+
|
|
260
|
+
rl.question(" Enter choice [1]: ", (answer) => {
|
|
261
|
+
rl.close();
|
|
262
|
+
const normalized = answer.trim() || "1";
|
|
263
|
+
resolve(normalized);
|
|
264
|
+
});
|
|
265
|
+
});
|
|
266
|
+
}
|
|
267
|
+
|
|
268
|
+
/**
|
|
269
|
+
* Sleep for a given number of milliseconds
|
|
270
|
+
*/
|
|
271
|
+
sleep(ms) {
|
|
272
|
+
return new Promise((resolve) => setTimeout(resolve, ms));
|
|
273
|
+
}
|
|
274
|
+
|
|
275
|
+
/**
|
|
276
|
+
* Prompt for hidden input (like password)
|
|
277
|
+
*/
|
|
278
|
+
async promptHidden(question) {
|
|
279
|
+
return new Promise((resolve) => {
|
|
280
|
+
process.stdout.write(question);
|
|
281
|
+
|
|
282
|
+
const stdin = process.stdin;
|
|
283
|
+
const wasRaw = stdin.isRaw;
|
|
284
|
+
stdin.setRawMode(true);
|
|
285
|
+
stdin.resume();
|
|
286
|
+
stdin.setEncoding("utf8");
|
|
287
|
+
|
|
288
|
+
let input = "";
|
|
289
|
+
|
|
290
|
+
const onData = (char) => {
|
|
291
|
+
// Handle Ctrl+C
|
|
292
|
+
if (char === "\u0003") {
|
|
293
|
+
stdin.setRawMode(wasRaw);
|
|
294
|
+
process.exit();
|
|
295
|
+
}
|
|
296
|
+
// Handle Enter
|
|
297
|
+
if (char === "\r" || char === "\n") {
|
|
298
|
+
stdin.setRawMode(wasRaw);
|
|
299
|
+
stdin.removeListener("data", onData);
|
|
300
|
+
stdin.pause();
|
|
301
|
+
console.log(""); // New line after hidden input
|
|
302
|
+
resolve(input);
|
|
303
|
+
return;
|
|
304
|
+
}
|
|
305
|
+
// Handle Backspace
|
|
306
|
+
if (char === "\u007F" || char === "\b") {
|
|
307
|
+
input = input.slice(0, -1);
|
|
308
|
+
return;
|
|
309
|
+
}
|
|
310
|
+
// Add character to input (but don't echo it)
|
|
311
|
+
input += char;
|
|
312
|
+
};
|
|
313
|
+
|
|
314
|
+
stdin.on("data", onData);
|
|
315
|
+
});
|
|
316
|
+
}
|
|
317
|
+
|
|
318
|
+
/**
|
|
319
|
+
* Add an environment variable export to the user's shell profile
|
|
320
|
+
*/
|
|
321
|
+
addToShellProfile(key, value) {
|
|
322
|
+
if (process.platform === "win32") {
|
|
323
|
+
// On Windows, set a persistent user environment variable via setx
|
|
324
|
+
try {
|
|
325
|
+
execSync(`setx ${key} "${value}"`, { stdio: "ignore" });
|
|
326
|
+
console.log(
|
|
327
|
+
chalk.green(` ✓ Set ${key} as user environment variable`),
|
|
328
|
+
);
|
|
329
|
+
console.log(
|
|
330
|
+
chalk.gray(` Restart your terminal for changes to take effect\n`),
|
|
331
|
+
);
|
|
332
|
+
} catch (error) {
|
|
333
|
+
console.log(
|
|
334
|
+
chalk.yellow(` ⚠️ Could not set ${key} via setx. You can set it manually:\n`),
|
|
335
|
+
);
|
|
336
|
+
console.log(chalk.gray(` setx ${key} "your_api_key"\n`));
|
|
337
|
+
}
|
|
338
|
+
return;
|
|
339
|
+
}
|
|
340
|
+
|
|
341
|
+
// Unix: append export to shell profile
|
|
342
|
+
const shell = process.env.SHELL || "/bin/bash";
|
|
343
|
+
const home = os.homedir();
|
|
344
|
+
let profilePath;
|
|
345
|
+
|
|
346
|
+
if (shell.includes("zsh")) {
|
|
347
|
+
profilePath = path.join(home, ".zshrc");
|
|
348
|
+
} else {
|
|
349
|
+
profilePath = path.join(home, ".bashrc");
|
|
350
|
+
}
|
|
351
|
+
|
|
352
|
+
const exportLine = `export ${key}="${value}"`;
|
|
353
|
+
|
|
354
|
+
// Check if already present
|
|
355
|
+
if (fs.existsSync(profilePath)) {
|
|
356
|
+
const content = fs.readFileSync(profilePath, "utf8");
|
|
357
|
+
if (content.includes(`export ${key}=`)) {
|
|
358
|
+
// Replace existing line
|
|
359
|
+
const updated = content.replace(
|
|
360
|
+
new RegExp(`^export ${key}=.*$`, "m"),
|
|
361
|
+
exportLine,
|
|
362
|
+
);
|
|
363
|
+
fs.writeFileSync(profilePath, updated);
|
|
364
|
+
console.log(
|
|
365
|
+
chalk.green(` ✓ Updated ${key} in ${profilePath}`),
|
|
366
|
+
);
|
|
367
|
+
console.log(
|
|
368
|
+
chalk.gray(` Run: source ${profilePath} (or open a new terminal)\n`),
|
|
369
|
+
);
|
|
370
|
+
return;
|
|
371
|
+
}
|
|
372
|
+
}
|
|
373
|
+
|
|
374
|
+
// Append to profile
|
|
375
|
+
fs.appendFileSync(profilePath, `\n${exportLine}\n`);
|
|
376
|
+
console.log(
|
|
377
|
+
chalk.green(` ✓ Added ${key} to ${profilePath}`),
|
|
378
|
+
);
|
|
379
|
+
console.log(
|
|
380
|
+
chalk.gray(` Run: source ${profilePath} (or open a new terminal)\n`),
|
|
381
|
+
);
|
|
382
|
+
}
|
|
383
|
+
|
|
384
|
+
/**
|
|
385
|
+
* Ask a yes/no question
|
|
386
|
+
*/
|
|
387
|
+
async askYesNo(question) {
|
|
388
|
+
return new Promise((resolve) => {
|
|
389
|
+
const rl = readline.createInterface({
|
|
390
|
+
input: process.stdin,
|
|
391
|
+
output: process.stdout,
|
|
392
|
+
});
|
|
393
|
+
|
|
394
|
+
rl.question(question, (answer) => {
|
|
395
|
+
rl.close();
|
|
396
|
+
const normalized = answer.toLowerCase().trim();
|
|
397
|
+
resolve(
|
|
398
|
+
normalized === "" || normalized === "y" || normalized === "yes",
|
|
399
|
+
);
|
|
400
|
+
});
|
|
401
|
+
});
|
|
402
|
+
}
|
|
403
|
+
|
|
404
|
+
/**
|
|
405
|
+
* Print next steps
|
|
406
|
+
*/
|
|
407
|
+
printNextSteps() {
|
|
408
|
+
console.log(chalk.cyan("Next steps:\n"));
|
|
409
|
+
console.log(" 1. Run your tests:");
|
|
410
|
+
console.log(chalk.gray(" vitest run\n"));
|
|
411
|
+
console.log(" 2. Use AI agents to write tests:");
|
|
412
|
+
console.log(chalk.gray(" Open VSCode/Cursor and use @testdriver agent\n"));
|
|
413
|
+
console.log(" 3. MCP server configured:");
|
|
414
|
+
console.log(chalk.gray(" TestDriver tools available via MCP in .vscode/mcp.json\n"));
|
|
415
|
+
console.log(
|
|
416
|
+
" 4. For CI/CD, add TD_API_KEY to your GitHub repository secrets",
|
|
417
|
+
);
|
|
418
|
+
console.log(
|
|
419
|
+
chalk.gray(" Settings → Secrets → Actions → New repository secret\n"),
|
|
420
|
+
);
|
|
421
|
+
console.log(
|
|
422
|
+
chalk.cyan(
|
|
423
|
+
"Learn more at https://docs.testdriver.ai/v7/getting-started/\n",
|
|
424
|
+
),
|
|
425
|
+
);
|
|
426
|
+
}
|
|
427
|
+
}
|
|
428
|
+
|
|
429
|
+
// Get command definition from interface.js
|
|
430
|
+
const tempAgent = { workingDir: process.cwd() };
|
|
431
|
+
const definitions = createCommandDefinitions(tempAgent);
|
|
432
|
+
const commandDef = definitions["init"];
|
|
433
|
+
|
|
434
|
+
InitCommand.description = commandDef?.description || "";
|
|
435
|
+
InitCommand.args = commandDef?.args || {};
|
|
436
|
+
InitCommand.flags = commandDef?.flags || {};
|
|
437
|
+
|
|
438
|
+
module.exports = InitCommand;
|