@timetotest/cli 0.1.10 → 0.2.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/README.md +201 -190
- package/dist/bin/ttt.js +4 -2
- package/dist/bin/ttt.js.map +1 -1
- package/dist/package.json +20 -3
- package/dist/src/commands/ask/AskApp.js +121 -0
- package/dist/src/commands/ask/AskApp.js.map +1 -0
- package/dist/src/commands/ask/components/AssistantResponse.js +31 -0
- package/dist/src/commands/ask/components/AssistantResponse.js.map +1 -0
- package/dist/src/commands/ask/components/Banner.js +15 -0
- package/dist/src/commands/ask/components/Banner.js.map +1 -0
- package/dist/src/commands/ask/components/ChatInput.js +93 -0
- package/dist/src/commands/ask/components/ChatInput.js.map +1 -0
- package/dist/src/commands/ask/components/Divider.js +17 -0
- package/dist/src/commands/ask/components/Divider.js.map +1 -0
- package/dist/src/commands/ask/components/IntroTips.js +19 -0
- package/dist/src/commands/ask/components/IntroTips.js.map +1 -0
- package/dist/src/commands/ask/components/MessageBubble.js +47 -0
- package/dist/src/commands/ask/components/MessageBubble.js.map +1 -0
- package/dist/src/commands/ask/components/SessionInfo.js +20 -0
- package/dist/src/commands/ask/components/SessionInfo.js.map +1 -0
- package/dist/src/commands/ask/components/StatusIndicator.js +67 -0
- package/dist/src/commands/ask/components/StatusIndicator.js.map +1 -0
- package/dist/src/commands/ask-ink.js +380 -0
- package/dist/src/commands/ask-ink.js.map +1 -0
- package/dist/src/commands/ask.js +138 -50
- package/dist/src/commands/ask.js.map +1 -1
- package/dist/src/commands/chat/ChatApp.js +125 -0
- package/dist/src/commands/chat/ChatApp.js.map +1 -0
- package/dist/src/commands/chat-ink.js +436 -0
- package/dist/src/commands/chat-ink.js.map +1 -0
- package/dist/src/commands/chat.js +82 -0
- package/dist/src/commands/chat.js.map +1 -0
- package/dist/src/commands/login.js +6 -4
- package/dist/src/commands/login.js.map +1 -1
- package/dist/src/commands/start-test.js +62 -88
- package/dist/src/commands/start-test.js.map +1 -1
- package/dist/src/commands/stream.js +9 -9
- package/dist/src/commands/stream.js.map +1 -1
- package/dist/src/commands/test.js +58 -125
- package/dist/src/commands/test.js.map +1 -1
- package/dist/src/lib/agent-orchestrator.js +546 -0
- package/dist/src/lib/agent-orchestrator.js.map +1 -0
- package/dist/src/lib/cloudinary-service.js +65 -0
- package/dist/src/lib/cloudinary-service.js.map +1 -0
- package/dist/src/lib/config.js +3 -2
- package/dist/src/lib/config.js.map +1 -1
- package/dist/src/lib/events.js +73 -60
- package/dist/src/lib/events.js.map +1 -1
- package/dist/src/lib/http.js +34 -1
- package/dist/src/lib/http.js.map +1 -1
- package/dist/src/lib/legacy-chat-runner.js +37 -0
- package/dist/src/lib/legacy-chat-runner.js.map +1 -0
- package/dist/src/lib/local-tools/api/api-discovery.js +17 -0
- package/dist/src/lib/local-tools/api/api-discovery.js.map +1 -0
- package/dist/src/lib/local-tools/api/build-request.js +20 -0
- package/dist/src/lib/local-tools/api/build-request.js.map +1 -0
- package/dist/src/lib/local-tools/api/extract-response-fields.js +26 -0
- package/dist/src/lib/local-tools/api/extract-response-fields.js.map +1 -0
- package/dist/src/lib/local-tools/api/generate-api-test.js +20 -0
- package/dist/src/lib/local-tools/api/generate-api-test.js.map +1 -0
- package/dist/src/lib/local-tools/api/generate-curl.js +8 -0
- package/dist/src/lib/local-tools/api/generate-curl.js.map +1 -0
- package/dist/src/lib/local-tools/api/get-api-parameters.js +17 -0
- package/dist/src/lib/local-tools/api/get-api-parameters.js.map +1 -0
- package/dist/src/lib/local-tools/api/index.js +10 -0
- package/dist/src/lib/local-tools/api/index.js.map +1 -0
- package/dist/src/lib/local-tools/api/request.js +43 -0
- package/dist/src/lib/local-tools/api/request.js.map +1 -0
- package/dist/src/lib/local-tools/api/set-auth-header.js +8 -0
- package/dist/src/lib/local-tools/api/set-auth-header.js.map +1 -0
- package/dist/src/lib/local-tools/api/types.js +2 -0
- package/dist/src/lib/local-tools/api/types.js.map +1 -0
- package/dist/src/lib/local-tools/api/utils.js +33 -0
- package/dist/src/lib/local-tools/api/utils.js.map +1 -0
- package/dist/src/lib/local-tools/api/validate-response.js +41 -0
- package/dist/src/lib/local-tools/api/validate-response.js.map +1 -0
- package/dist/src/lib/local-tools/file-tools.js +45 -0
- package/dist/src/lib/local-tools/file-tools.js.map +1 -0
- package/dist/src/lib/local-tools/general/discover-local-services.js +95 -0
- package/dist/src/lib/local-tools/general/discover-local-services.js.map +1 -0
- package/dist/src/lib/local-tools/general/index.js +2 -0
- package/dist/src/lib/local-tools/general/index.js.map +1 -0
- package/dist/src/lib/local-tools/ui/click-element.js +105 -0
- package/dist/src/lib/local-tools/ui/click-element.js.map +1 -0
- package/dist/src/lib/local-tools/ui/dom-rag.js +201 -0
- package/dist/src/lib/local-tools/ui/dom-rag.js.map +1 -0
- package/dist/src/lib/local-tools/ui/find-element.js +31 -0
- package/dist/src/lib/local-tools/ui/find-element.js.map +1 -0
- package/dist/src/lib/local-tools/ui/hover-element.js +94 -0
- package/dist/src/lib/local-tools/ui/hover-element.js.map +1 -0
- package/dist/src/lib/local-tools/ui/index.js +3 -0
- package/dist/src/lib/local-tools/ui/index.js.map +1 -0
- package/dist/src/lib/local-tools/ui/manage-tab.js +65 -0
- package/dist/src/lib/local-tools/ui/manage-tab.js.map +1 -0
- package/dist/src/lib/local-tools/ui/navigate.js +35 -0
- package/dist/src/lib/local-tools/ui/navigate.js.map +1 -0
- package/dist/src/lib/local-tools/ui/page-discovery.js +32 -0
- package/dist/src/lib/local-tools/ui/page-discovery.js.map +1 -0
- package/dist/src/lib/local-tools/ui/playwright-mcp.js +217 -0
- package/dist/src/lib/local-tools/ui/playwright-mcp.js.map +1 -0
- package/dist/src/lib/local-tools/ui/screenshot.js +19 -0
- package/dist/src/lib/local-tools/ui/screenshot.js.map +1 -0
- package/dist/src/lib/local-tools/ui/search-interactive-elements.js +18 -0
- package/dist/src/lib/local-tools/ui/search-interactive-elements.js.map +1 -0
- package/dist/src/lib/local-tools/ui/selector-resolver.js +153 -0
- package/dist/src/lib/local-tools/ui/selector-resolver.js.map +1 -0
- package/dist/src/lib/local-tools/ui/snapshot-query.js +129 -0
- package/dist/src/lib/local-tools/ui/snapshot-query.js.map +1 -0
- package/dist/src/lib/local-tools/ui/type-text.js +40 -0
- package/dist/src/lib/local-tools/ui/type-text.js.map +1 -0
- package/dist/src/lib/local-tools/ui/types.js +2 -0
- package/dist/src/lib/local-tools/ui/types.js.map +1 -0
- package/dist/src/lib/local-tools/utility/finish-overall-test.js +12 -0
- package/dist/src/lib/local-tools/utility/finish-overall-test.js.map +1 -0
- package/dist/src/lib/local-tools/utility/index.js +2 -0
- package/dist/src/lib/local-tools/utility/index.js.map +1 -0
- package/dist/src/lib/prompts/builder.js +38 -0
- package/dist/src/lib/prompts/builder.js.map +1 -0
- package/dist/src/lib/prompts/index.js +7 -0
- package/dist/src/lib/prompts/index.js.map +1 -0
- package/dist/src/lib/prompts/templates.js +166 -0
- package/dist/src/lib/prompts/templates.js.map +1 -0
- package/dist/src/lib/session-manager.js +201 -0
- package/dist/src/lib/session-manager.js.map +1 -0
- package/dist/src/lib/socket.js +78 -6
- package/dist/src/lib/socket.js.map +1 -1
- package/dist/src/lib/testing-mode.js +33 -0
- package/dist/src/lib/testing-mode.js.map +1 -0
- package/dist/src/lib/tool-descriptions.js +59 -0
- package/dist/src/lib/tool-descriptions.js.map +1 -0
- package/dist/src/lib/tool-executor.js +537 -0
- package/dist/src/lib/tool-executor.js.map +1 -0
- package/dist/src/lib/tool-registry.js +803 -0
- package/dist/src/lib/tool-registry.js.map +1 -0
- package/dist/src/lib/tool-result-pruner.js +384 -0
- package/dist/src/lib/tool-result-pruner.js.map +1 -0
- package/dist/src/lib/tui/components/AskIntro.js +6 -0
- package/dist/src/lib/tui/components/AskIntro.js.map +1 -0
- package/dist/src/lib/tui/components/Banner.js +15 -0
- package/dist/src/lib/tui/components/Banner.js.map +1 -0
- package/dist/src/lib/tui/components/Divider.js +17 -0
- package/dist/src/lib/tui/components/Divider.js.map +1 -0
- package/dist/src/lib/tui/components/EventLine.js +110 -0
- package/dist/src/lib/tui/components/EventLine.js.map +1 -0
- package/dist/src/lib/tui/components/Header.js +15 -0
- package/dist/src/lib/tui/components/Header.js.map +1 -0
- package/dist/src/lib/tui/components/InputBox.js +9 -0
- package/dist/src/lib/tui/components/InputBox.js.map +1 -0
- package/dist/src/lib/tui/components/Mapping.js +8 -0
- package/dist/src/lib/tui/components/Mapping.js.map +1 -0
- package/dist/src/lib/tui/components/ProjectList.js +6 -0
- package/dist/src/lib/tui/components/ProjectList.js.map +1 -0
- package/dist/src/lib/tui/components/Spinner.js +20 -0
- package/dist/src/lib/tui/components/Spinner.js.map +1 -0
- package/dist/src/lib/tui/components/StatusBanner.js +12 -0
- package/dist/src/lib/tui/components/StatusBanner.js.map +1 -0
- package/dist/src/lib/tui/components/StatusBar.js +11 -0
- package/dist/src/lib/tui/components/StatusBar.js.map +1 -0
- package/dist/src/lib/tui/components/UserBubble.js +6 -0
- package/dist/src/lib/tui/components/UserBubble.js.map +1 -0
- package/dist/src/lib/tui/components/index.js +16 -0
- package/dist/src/lib/tui/components/index.js.map +1 -0
- package/dist/src/lib/tui/events.js +716 -76
- package/dist/src/lib/tui/events.js.map +1 -1
- package/dist/src/lib/tui/icons.js +14 -0
- package/dist/src/lib/tui/icons.js.map +1 -1
- package/dist/src/lib/tui/ink-print.js +41 -0
- package/dist/src/lib/tui/ink-print.js.map +1 -0
- package/dist/src/lib/tui/interactive-chat.js +345 -0
- package/dist/src/lib/tui/interactive-chat.js.map +1 -0
- package/dist/src/lib/tui/print.js +31 -26
- package/dist/src/lib/tui/print.js.map +1 -1
- package/dist/src/lib/tui/prompt.js +21 -18
- package/dist/src/lib/tui/prompt.js.map +1 -1
- package/dist/src/test-agent-flow.js +148 -0
- package/dist/src/test-agent-flow.js.map +1 -0
- package/dist/src/test-browser-session.js +152 -0
- package/dist/src/test-browser-session.js.map +1 -0
- package/dist/src/test-browser-snapshot.js +187 -0
- package/dist/src/test-browser-snapshot.js.map +1 -0
- package/dist/src/test-snapshot-detailed.js +219 -0
- package/dist/src/test-snapshot-detailed.js.map +1 -0
- package/dist/src/test-snapshot-simple.js +85 -0
- package/dist/src/test-snapshot-simple.js.map +1 -0
- package/dist/src/test-snapshot-tabs.js +110 -0
- package/dist/src/test-snapshot-tabs.js.map +1 -0
- package/package.json +20 -3
|
@@ -1,16 +1,40 @@
|
|
|
1
1
|
import { Command } from "commander";
|
|
2
|
-
import ora from "ora";
|
|
3
2
|
import chalk from "chalk";
|
|
4
|
-
import
|
|
5
|
-
import
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
3
|
+
import { runLegacyChatWrapper } from "../lib/legacy-chat-runner.js";
|
|
4
|
+
import { resolveTestingMode } from "../lib/testing-mode.js";
|
|
5
|
+
const buildSeedMessage = (prompt, opts) => {
|
|
6
|
+
const parts = [
|
|
7
|
+
`Start a ${String(opts.type || "ui").toUpperCase()} test based on this instruction:`,
|
|
8
|
+
`"${prompt}".`,
|
|
9
|
+
];
|
|
10
|
+
if (opts.projectId) {
|
|
11
|
+
parts.push(`Project ID: ${opts.projectId}.`);
|
|
12
|
+
}
|
|
13
|
+
if (opts.teamId) {
|
|
14
|
+
parts.push(`Team ID: ${opts.teamId}.`);
|
|
15
|
+
}
|
|
16
|
+
if (opts.docsUrl) {
|
|
17
|
+
parts.push(`API documentation: ${opts.docsUrl}.`);
|
|
18
|
+
}
|
|
19
|
+
const extras = [];
|
|
20
|
+
if (opts.execute)
|
|
21
|
+
extras.push("execute generated tests");
|
|
22
|
+
if (opts.report)
|
|
23
|
+
extras.push(`generate report${typeof opts.report === "string" ? ` to ${opts.report}` : ""}`);
|
|
24
|
+
if (opts.share)
|
|
25
|
+
extras.push("enable sharing");
|
|
26
|
+
if (opts.wait)
|
|
27
|
+
extras.push("wait for completion");
|
|
28
|
+
if (extras.length) {
|
|
29
|
+
parts.push(`Additional options: ${extras.join(", ")}.`);
|
|
30
|
+
}
|
|
31
|
+
parts.push(`Execution mode: ${String(opts.mode || "priority")}.`);
|
|
32
|
+
return parts.join("\n");
|
|
33
|
+
};
|
|
10
34
|
export const startTest = new Command("start-test")
|
|
11
35
|
.description("Start a test from a natural language prompt (no URL required)")
|
|
12
36
|
.argument("<prompt...>", "Describe what to test. URL is derived from your project.")
|
|
13
|
-
.option("--type <type>", "ui|api
|
|
37
|
+
.option("--type <type>", "ui|api", "ui")
|
|
14
38
|
.option("--execute", "execute generated test cases", false)
|
|
15
39
|
.option("--mode <mode>", "all|priority|failed_only", "priority")
|
|
16
40
|
.option("--project-id <id>")
|
|
@@ -20,94 +44,44 @@ export const startTest = new Command("start-test")
|
|
|
20
44
|
.option("--share", "Enable public sharing after completion")
|
|
21
45
|
.option("--wait", "Block until completion and set exit code based on result")
|
|
22
46
|
.action(async (promptParts, opts) => {
|
|
23
|
-
const prompt = Array.isArray(promptParts)
|
|
24
|
-
? promptParts.join(" ")
|
|
25
|
-
: String(promptParts || "");
|
|
26
|
-
const spinner = ora("Preparing test").start();
|
|
27
47
|
try {
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
if (
|
|
48
|
+
const prompt = Array.isArray(promptParts)
|
|
49
|
+
? promptParts.join(" ").trim()
|
|
50
|
+
: String(promptParts || "").trim();
|
|
51
|
+
if (!prompt) {
|
|
52
|
+
console.error(chalk.red("Prompt is required."));
|
|
53
|
+
process.exitCode = 1;
|
|
54
|
+
return;
|
|
55
|
+
}
|
|
56
|
+
const rawType = String(opts.type || "ui").toLowerCase();
|
|
57
|
+
let testingMode;
|
|
58
|
+
if (rawType === "mixed") {
|
|
59
|
+
console.log(chalk.yellow("⚠️ The 'mixed' test type is no longer supported. Defaulting to UI testing."));
|
|
60
|
+
testingMode = "ui";
|
|
61
|
+
}
|
|
62
|
+
else {
|
|
32
63
|
try {
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
return;
|
|
39
|
-
}
|
|
64
|
+
testingMode = resolveTestingMode(rawType, {
|
|
65
|
+
defaultMode: "ui",
|
|
66
|
+
strict: true,
|
|
67
|
+
contextLabel: "--type",
|
|
68
|
+
});
|
|
40
69
|
}
|
|
41
|
-
catch (
|
|
42
|
-
|
|
43
|
-
|
|
70
|
+
catch (error) {
|
|
71
|
+
console.error(chalk.red(error?.message || String(error)));
|
|
72
|
+
process.exitCode = 1;
|
|
73
|
+
return;
|
|
44
74
|
}
|
|
45
75
|
}
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
project_id: opts.projectId || undefined,
|
|
51
|
-
team_id: opts.teamId || undefined,
|
|
52
|
-
user_prompt: prompt,
|
|
53
|
-
test_metadata: {
|
|
54
|
-
docs_url: opts.docsUrl || undefined,
|
|
55
|
-
surface: "cli",
|
|
56
|
-
},
|
|
57
|
-
};
|
|
58
|
-
const params = {
|
|
59
|
-
execute_tests: !!opts.execute,
|
|
60
|
-
execution_mode: String(opts.mode || "priority"),
|
|
61
|
-
};
|
|
62
|
-
const resp = await http.post("/api/v1/start-test", body, { params });
|
|
63
|
-
spinner.succeed("Test started");
|
|
64
|
-
const testId = resp.data.test_id;
|
|
65
|
-
console.log(chalk.gray("Test ID"), chalk.white(String(testId)));
|
|
66
|
-
const socket = connectAndSubscribe(testId);
|
|
67
|
-
registerTestEventLogging(socket, (line) => console.log(line));
|
|
68
|
-
socket.on("test_completed", async (p) => {
|
|
69
|
-
const finalStatus = p?.status || "completed";
|
|
70
|
-
if (opts.report) {
|
|
71
|
-
const outPath = typeof opts.report === "string"
|
|
72
|
-
? opts.report
|
|
73
|
-
: path.resolve(process.cwd(), `report-${testId}.pdf`);
|
|
74
|
-
await downloadPdfReport(http, testId, outPath);
|
|
75
|
-
console.log(chalk.green(`Report saved: ${outPath}`));
|
|
76
|
-
}
|
|
77
|
-
if (opts.share) {
|
|
78
|
-
try {
|
|
79
|
-
await http.post(`/api/v1/tests/${testId}/share`, {
|
|
80
|
-
allow_indexing: false,
|
|
81
|
-
});
|
|
82
|
-
}
|
|
83
|
-
catch { }
|
|
84
|
-
}
|
|
85
|
-
if (opts.wait) {
|
|
86
|
-
process.exitCode = finalStatus === "passed" ? 0 : 1;
|
|
87
|
-
}
|
|
88
|
-
await socket.disconnect();
|
|
89
|
-
});
|
|
90
|
-
socket.on("test_failed", async (p) => {
|
|
91
|
-
if (opts.wait)
|
|
92
|
-
process.exitCode = 1;
|
|
93
|
-
await socket.disconnect();
|
|
76
|
+
await runLegacyChatWrapper({
|
|
77
|
+
title: "ttt start-test",
|
|
78
|
+
seedMessages: [buildSeedMessage(prompt, opts)],
|
|
79
|
+
testingMode,
|
|
94
80
|
});
|
|
95
|
-
if (!opts.wait)
|
|
96
|
-
ora().info('Use "ttt stream <test-id>" to attach to progress later.');
|
|
97
81
|
}
|
|
98
|
-
catch (
|
|
99
|
-
|
|
100
|
-
console.error(err?.response?.data || err?.message || err);
|
|
82
|
+
catch (error) {
|
|
83
|
+
console.error(chalk.red(error?.message || String(error)));
|
|
101
84
|
process.exitCode = 1;
|
|
102
85
|
}
|
|
103
86
|
});
|
|
104
|
-
async function downloadPdfReport(http, testId, outPath) {
|
|
105
|
-
const createResp = await http.post("/api/v1/reports/", { test_id: testId });
|
|
106
|
-
const reportId = createResp.data?.id || createResp.data?.report_id;
|
|
107
|
-
const pdfResp = await http.get(`/api/v1/reports/${reportId}/download`, {
|
|
108
|
-
responseType: "arraybuffer",
|
|
109
|
-
});
|
|
110
|
-
await fs.ensureDir(path.dirname(outPath));
|
|
111
|
-
await fs.writeFile(outPath, pdfResp.data);
|
|
112
|
-
}
|
|
113
87
|
//# sourceMappingURL=start-test.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"start-test.js","sourceRoot":"","sources":["../../../src/commands/start-test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAC,OAAO,EAAC,MAAM,WAAW,CAAC;AAClC,OAAO,
|
|
1
|
+
{"version":3,"file":"start-test.js","sourceRoot":"","sources":["../../../src/commands/start-test.ts"],"names":[],"mappings":"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"}
|
|
@@ -1,16 +1,16 @@
|
|
|
1
|
-
import { Command } from
|
|
2
|
-
import { connectAndSubscribe } from
|
|
3
|
-
import {
|
|
4
|
-
export const stream = new Command(
|
|
5
|
-
.description(
|
|
6
|
-
.argument(
|
|
1
|
+
import { Command } from "commander";
|
|
2
|
+
import { connectAndSubscribe } from "../lib/socket.js";
|
|
3
|
+
import { registerUnifiedEventLogging } from "../lib/events.js";
|
|
4
|
+
export const stream = new Command("stream")
|
|
5
|
+
.description("Attach to a running test and stream events")
|
|
6
|
+
.argument("<test-id>")
|
|
7
7
|
.action(async (testId) => {
|
|
8
8
|
const socket = connectAndSubscribe(testId);
|
|
9
|
-
|
|
10
|
-
socket.on(
|
|
9
|
+
registerUnifiedEventLogging(socket, (line) => console.log(line));
|
|
10
|
+
socket.on("session_completed", async () => {
|
|
11
11
|
await socket.disconnect();
|
|
12
12
|
});
|
|
13
|
-
socket.on(
|
|
13
|
+
socket.on("session_error", async () => {
|
|
14
14
|
await socket.disconnect();
|
|
15
15
|
});
|
|
16
16
|
});
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"stream.js","sourceRoot":"","sources":["../../../src/commands/stream.ts"],"names":[],"mappings":"AAAA,OAAO,
|
|
1
|
+
{"version":3,"file":"stream.js","sourceRoot":"","sources":["../../../src/commands/stream.ts"],"names":[],"mappings":"AAAA,OAAO,EAAC,OAAO,EAAC,MAAM,WAAW,CAAC;AAClC,OAAO,EAAC,mBAAmB,EAAC,MAAM,kBAAkB,CAAC;AACrD,OAAO,EAAC,2BAA2B,EAAC,MAAM,kBAAkB,CAAC;AAE7D,MAAM,CAAC,MAAM,MAAM,GAAG,IAAI,OAAO,CAAC,QAAQ,CAAC;KACxC,WAAW,CAAC,4CAA4C,CAAC;KACzD,QAAQ,CAAC,WAAW,CAAC;KACrB,MAAM,CAAC,KAAK,EAAE,MAAM,EAAE,EAAE;IACvB,MAAM,MAAM,GAAG,mBAAmB,CAAC,MAAM,CAAC,CAAC;IAC3C,2BAA2B,CAAC,MAAM,EAAE,CAAC,IAAI,EAAE,EAAE,CAAC,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC;IACjE,MAAM,CAAC,EAAE,CAAC,mBAAmB,EAAE,KAAK,IAAI,EAAE;QACxC,MAAM,MAAM,CAAC,UAAU,EAAE,CAAC;IAC5B,CAAC,CAAC,CAAC;IACH,MAAM,CAAC,EAAE,CAAC,eAAe,EAAE,KAAK,IAAI,EAAE;QACpC,MAAM,MAAM,CAAC,UAAU,EAAE,CAAC;IAC5B,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
|
|
@@ -1,17 +1,25 @@
|
|
|
1
1
|
import { Command } from "commander";
|
|
2
|
-
import ora from "ora";
|
|
3
|
-
import fs from "fs-extra";
|
|
4
|
-
import path from "path";
|
|
5
2
|
import chalk from "chalk";
|
|
6
|
-
import {
|
|
7
|
-
import {
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
3
|
+
import { runLegacyChatWrapper } from "../lib/legacy-chat-runner.js";
|
|
4
|
+
import { resolveTestingMode } from "../lib/testing-mode.js";
|
|
5
|
+
const formatFlags = (opts) => {
|
|
6
|
+
const flags = [];
|
|
7
|
+
if (opts.execute)
|
|
8
|
+
flags.push("execute generated tests");
|
|
9
|
+
if (opts.report)
|
|
10
|
+
flags.push(`generate report${typeof opts.report === "string" ? ` to ${opts.report}` : ""}`);
|
|
11
|
+
if (opts.share)
|
|
12
|
+
flags.push("enable sharing");
|
|
13
|
+
if (opts.wait)
|
|
14
|
+
flags.push("wait for completion");
|
|
15
|
+
if (opts.teamId)
|
|
16
|
+
flags.push(`team ${opts.teamId}`);
|
|
17
|
+
return flags;
|
|
18
|
+
};
|
|
11
19
|
export const test = new Command("test")
|
|
12
20
|
.description("Start a test (Frontend | API)")
|
|
13
21
|
.requiredOption("--url <ui-url>", "UI URL to test")
|
|
14
|
-
.option("--type <type>", "ui|api
|
|
22
|
+
.option("--type <type>", "ui|api", "ui")
|
|
15
23
|
.requiredOption("--prompt <text>", "Natural language prompt for this test")
|
|
16
24
|
.option("--execute", "execute generated test cases", false)
|
|
17
25
|
.option("--mode <mode>", "all|priority|failed_only", "priority")
|
|
@@ -23,131 +31,56 @@ export const test = new Command("test")
|
|
|
23
31
|
.option("--share", "Enable public sharing after completion")
|
|
24
32
|
.option("--wait", "Block until completion and set exit code based on result")
|
|
25
33
|
.action(async (opts) => {
|
|
26
|
-
const spinner = ora("Preparing test").start();
|
|
27
|
-
let teardown = null;
|
|
28
34
|
try {
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
printMapping("ngrok", url, publicUrl);
|
|
40
|
-
spinner.start("Continuing…");
|
|
41
|
-
url = publicUrl;
|
|
42
|
-
}
|
|
43
|
-
if (docsUrl) {
|
|
44
|
-
const t2 = await createTunnelIfNeeded(docsUrl);
|
|
45
|
-
if (t2) {
|
|
46
|
-
const publicUrl2 = t2.publicUrl;
|
|
47
|
-
spinner.stop();
|
|
48
|
-
printMapping("ngrok (docs)", docsUrl, publicUrl2);
|
|
49
|
-
spinner.start("Continuing…");
|
|
50
|
-
docsUrl = publicUrl2;
|
|
51
|
-
const old = teardown;
|
|
52
|
-
teardown = async () => {
|
|
53
|
-
await Promise.all([old?.(), t2.stop()].filter(Boolean));
|
|
54
|
-
};
|
|
55
|
-
}
|
|
56
|
-
}
|
|
35
|
+
const url = String(opts.url);
|
|
36
|
+
const prompt = String(opts.prompt || "");
|
|
37
|
+
const type = String(opts.type || "ui").toLowerCase();
|
|
38
|
+
const executionMode = String(opts.mode || "priority");
|
|
39
|
+
const docsUrl = opts.docsUrl ? String(opts.docsUrl) : null;
|
|
40
|
+
const projectId = opts.projectId ? String(opts.projectId) : null;
|
|
41
|
+
let testingMode;
|
|
42
|
+
if (type === "mixed") {
|
|
43
|
+
console.log(chalk.yellow("⚠️ The 'mixed' test type is no longer supported. Defaulting to UI testing."));
|
|
44
|
+
testingMode = "ui";
|
|
57
45
|
}
|
|
58
|
-
|
|
59
|
-
// Preflight: ensure background mode is enabled for selected project when using CLI
|
|
60
|
-
if (opts.projectId) {
|
|
46
|
+
else {
|
|
61
47
|
try {
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
if (teardown)
|
|
68
|
-
await teardown();
|
|
69
|
-
process.exitCode = 1;
|
|
70
|
-
return;
|
|
71
|
-
}
|
|
48
|
+
testingMode = resolveTestingMode(type, {
|
|
49
|
+
defaultMode: "ui",
|
|
50
|
+
strict: true,
|
|
51
|
+
contextLabel: "--type",
|
|
52
|
+
});
|
|
72
53
|
}
|
|
73
|
-
catch (
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
throw e;
|
|
54
|
+
catch (error) {
|
|
55
|
+
console.error(chalk.red(error?.message || String(error)));
|
|
56
|
+
process.exitCode = 1;
|
|
57
|
+
return;
|
|
78
58
|
}
|
|
79
59
|
}
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
console.log(chalk.gray("Test ID"), chalk.white(String(testId)));
|
|
100
|
-
const socket = connectAndSubscribe(testId);
|
|
101
|
-
registerTestEventLogging(socket, (line) => console.log(line));
|
|
102
|
-
socket.on("test_completed", async (p) => {
|
|
103
|
-
const finalStatus = p?.status || "completed";
|
|
104
|
-
if (opts.report) {
|
|
105
|
-
const outPath = typeof opts.report === "string"
|
|
106
|
-
? opts.report
|
|
107
|
-
: path.resolve(process.cwd(), `report-${testId}.pdf`);
|
|
108
|
-
await downloadPdfReport(http, testId, outPath);
|
|
109
|
-
console.log(chalk.green(`Report saved: ${outPath}`));
|
|
110
|
-
}
|
|
111
|
-
if (opts.share) {
|
|
112
|
-
try {
|
|
113
|
-
await http.post(`/api/v1/tests/${testId}/share`, {
|
|
114
|
-
allow_indexing: false,
|
|
115
|
-
});
|
|
116
|
-
}
|
|
117
|
-
catch { }
|
|
118
|
-
}
|
|
119
|
-
if (opts.wait) {
|
|
120
|
-
process.exitCode = finalStatus === "passed" ? 0 : 1;
|
|
121
|
-
}
|
|
122
|
-
await socket.disconnect();
|
|
123
|
-
if (teardown)
|
|
124
|
-
await teardown();
|
|
125
|
-
});
|
|
126
|
-
socket.on("test_failed", async (p) => {
|
|
127
|
-
if (opts.wait)
|
|
128
|
-
process.exitCode = 1;
|
|
129
|
-
await socket.disconnect();
|
|
130
|
-
if (teardown)
|
|
131
|
-
await teardown();
|
|
60
|
+
const lines = [
|
|
61
|
+
`Run a ${type.toUpperCase()} test against ${url}.`,
|
|
62
|
+
`User prompt: "${prompt}".`,
|
|
63
|
+
`Execution mode: ${executionMode}.`,
|
|
64
|
+
];
|
|
65
|
+
if (projectId)
|
|
66
|
+
lines.push(`Project ID: ${projectId}.`);
|
|
67
|
+
if (docsUrl)
|
|
68
|
+
lines.push(`API documentation: ${docsUrl}.`);
|
|
69
|
+
const extras = formatFlags(opts);
|
|
70
|
+
if (extras.length) {
|
|
71
|
+
lines.push(`Additional options: ${extras.join(", ")}.`);
|
|
72
|
+
}
|
|
73
|
+
await runLegacyChatWrapper({
|
|
74
|
+
title: "ttt test",
|
|
75
|
+
seedMessages: [lines.join("\n")],
|
|
76
|
+
testingMode,
|
|
77
|
+
baseUrl: testingMode === "ui" ? url : undefined,
|
|
78
|
+
apiBaseUrl: testingMode === "api" ? url : undefined,
|
|
132
79
|
});
|
|
133
|
-
if (!opts.wait)
|
|
134
|
-
spinner.info('Use "ttt stream <test-id>" to attach to progress later.');
|
|
135
80
|
}
|
|
136
|
-
catch (
|
|
137
|
-
|
|
138
|
-
console.error(err?.response?.data || err?.message || err);
|
|
81
|
+
catch (error) {
|
|
82
|
+
console.error(chalk.red(error?.message || String(error)));
|
|
139
83
|
process.exitCode = 1;
|
|
140
|
-
if (teardown)
|
|
141
|
-
await teardown();
|
|
142
84
|
}
|
|
143
85
|
});
|
|
144
|
-
async function downloadPdfReport(http, testId, outPath) {
|
|
145
|
-
const createResp = await http.post("/api/v1/reports/", { test_id: testId });
|
|
146
|
-
const reportId = createResp.data?.id || createResp.data?.report_id;
|
|
147
|
-
const pdfResp = await http.get(`/api/v1/reports/${reportId}/download`, {
|
|
148
|
-
responseType: "arraybuffer",
|
|
149
|
-
});
|
|
150
|
-
await fs.ensureDir(path.dirname(outPath));
|
|
151
|
-
await fs.writeFile(outPath, pdfResp.data);
|
|
152
|
-
}
|
|
153
86
|
//# sourceMappingURL=test.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"test.js","sourceRoot":"","sources":["../../../src/commands/test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAC,OAAO,EAAC,MAAM,WAAW,CAAC;AAClC,OAAO,
|
|
1
|
+
{"version":3,"file":"test.js","sourceRoot":"","sources":["../../../src/commands/test.ts"],"names":[],"mappings":"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"}
|