@timetotest/cli 0.2.3 → 0.2.4
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/dist/package.json +8 -5
- package/dist/src/commands/chat/ChatApp.js +30 -42
- package/dist/src/commands/chat/ChatApp.js.map +1 -1
- package/dist/src/commands/chat/components/Banner.js +1 -1
- package/dist/src/commands/chat/components/ChatInput.js +39 -17
- package/dist/src/commands/chat/components/ChatInput.js.map +1 -1
- package/dist/src/commands/chat/components/MessageBubble.js +2 -1
- package/dist/src/commands/chat/components/MessageBubble.js.map +1 -1
- package/dist/src/commands/chat-ink.js +118 -9
- package/dist/src/commands/chat-ink.js.map +1 -1
- package/dist/src/commands/start-test.js +61 -0
- package/dist/src/commands/start-test.js.map +1 -1
- package/dist/src/commands/stream/StreamApp.js +127 -0
- package/dist/src/commands/stream/StreamApp.js.map +1 -0
- package/dist/src/commands/stream.js +43 -8
- package/dist/src/commands/stream.js.map +1 -1
- package/dist/src/commands/test/TestRunApp.js +183 -0
- package/dist/src/commands/test/TestRunApp.js.map +1 -0
- package/dist/src/commands/test.js +97 -0
- package/dist/src/commands/test.js.map +1 -1
- package/dist/src/lib/agent-orchestrator.js +5 -0
- package/dist/src/lib/agent-orchestrator.js.map +1 -1
- package/dist/src/lib/local-tools/ui/playwright-mcp.js +1 -1
- package/dist/src/lib/tui/ink/theme.js +21 -0
- package/dist/src/lib/tui/ink/theme.js.map +1 -0
- package/package.json +8 -5
- package/dist/src/commands/ask/AskApp.js +0 -121
- package/dist/src/commands/ask/AskApp.js.map +0 -1
- package/dist/src/commands/ask/components/AssistantResponse.js +0 -31
- package/dist/src/commands/ask/components/AssistantResponse.js.map +0 -1
- package/dist/src/commands/ask/components/Banner.js +0 -15
- package/dist/src/commands/ask/components/Banner.js.map +0 -1
- package/dist/src/commands/ask/components/ChatInput.js +0 -93
- package/dist/src/commands/ask/components/ChatInput.js.map +0 -1
- package/dist/src/commands/ask/components/Divider.js +0 -17
- package/dist/src/commands/ask/components/Divider.js.map +0 -1
- package/dist/src/commands/ask/components/IntroTips.js +0 -19
- package/dist/src/commands/ask/components/IntroTips.js.map +0 -1
- package/dist/src/commands/ask/components/MessageBubble.js +0 -47
- package/dist/src/commands/ask/components/MessageBubble.js.map +0 -1
- package/dist/src/commands/ask/components/SessionInfo.js +0 -20
- package/dist/src/commands/ask/components/SessionInfo.js.map +0 -1
- package/dist/src/commands/ask/components/StatusIndicator.js +0 -67
- package/dist/src/commands/ask/components/StatusIndicator.js.map +0 -1
- package/dist/src/commands/ask-ink.js +0 -380
- package/dist/src/commands/ask-ink.js.map +0 -1
- package/dist/src/commands/ask.js +0 -991
- package/dist/src/commands/ask.js.map +0 -1
- package/dist/src/commands/chat/components/Divider.js +0 -7
- package/dist/src/commands/chat/components/Divider.js.map +0 -1
- package/dist/src/commands/chat/components/SessionInfo.js +0 -11
- package/dist/src/commands/chat/components/SessionInfo.js.map +0 -1
- package/dist/src/commands/chat.js +0 -82
- package/dist/src/commands/chat.js.map +0 -1
- package/dist/src/lib/legacy-chat-runner.js +0 -37
- package/dist/src/lib/legacy-chat-runner.js.map +0 -1
- package/dist/src/lib/local-tools/ui/click-element.js +0 -105
- package/dist/src/lib/local-tools/ui/click-element.js.map +0 -1
- package/dist/src/lib/local-tools/ui/dom-rag.js +0 -201
- package/dist/src/lib/local-tools/ui/dom-rag.js.map +0 -1
- package/dist/src/lib/local-tools/ui/find-element.js +0 -31
- package/dist/src/lib/local-tools/ui/find-element.js.map +0 -1
- package/dist/src/lib/local-tools/ui/hover-element.js +0 -94
- package/dist/src/lib/local-tools/ui/hover-element.js.map +0 -1
- package/dist/src/lib/local-tools/ui/manage-tab.js +0 -65
- package/dist/src/lib/local-tools/ui/manage-tab.js.map +0 -1
- package/dist/src/lib/local-tools/ui/navigate.js +0 -35
- package/dist/src/lib/local-tools/ui/navigate.js.map +0 -1
- package/dist/src/lib/local-tools/ui/page-discovery.js +0 -32
- package/dist/src/lib/local-tools/ui/page-discovery.js.map +0 -1
- package/dist/src/lib/local-tools/ui/screenshot.js +0 -19
- package/dist/src/lib/local-tools/ui/screenshot.js.map +0 -1
- package/dist/src/lib/local-tools/ui/search-interactive-elements.js +0 -18
- package/dist/src/lib/local-tools/ui/search-interactive-elements.js.map +0 -1
- package/dist/src/lib/local-tools/ui/selector-resolver.js +0 -153
- package/dist/src/lib/local-tools/ui/selector-resolver.js.map +0 -1
- package/dist/src/lib/local-tools/ui/type-text.js +0 -40
- package/dist/src/lib/local-tools/ui/type-text.js.map +0 -1
- package/dist/src/lib/tui/components/AskIntro.js +0 -6
- package/dist/src/lib/tui/components/AskIntro.js.map +0 -1
- package/dist/src/lib/tui/components/Banner.js +0 -15
- package/dist/src/lib/tui/components/Banner.js.map +0 -1
- package/dist/src/lib/tui/components/Divider.js +0 -17
- package/dist/src/lib/tui/components/Divider.js.map +0 -1
- package/dist/src/lib/tui/components/EventLine.js +0 -110
- package/dist/src/lib/tui/components/EventLine.js.map +0 -1
- package/dist/src/lib/tui/components/Header.js +0 -15
- package/dist/src/lib/tui/components/Header.js.map +0 -1
- package/dist/src/lib/tui/components/InputBox.js +0 -9
- package/dist/src/lib/tui/components/InputBox.js.map +0 -1
- package/dist/src/lib/tui/components/Mapping.js +0 -8
- package/dist/src/lib/tui/components/Mapping.js.map +0 -1
- package/dist/src/lib/tui/components/ProjectList.js +0 -6
- package/dist/src/lib/tui/components/ProjectList.js.map +0 -1
- package/dist/src/lib/tui/components/Spinner.js +0 -20
- package/dist/src/lib/tui/components/Spinner.js.map +0 -1
- package/dist/src/lib/tui/components/StatusBanner.js +0 -12
- package/dist/src/lib/tui/components/StatusBanner.js.map +0 -1
- package/dist/src/lib/tui/components/StatusBar.js +0 -11
- package/dist/src/lib/tui/components/StatusBar.js.map +0 -1
- package/dist/src/lib/tui/components/UserBubble.js +0 -6
- package/dist/src/lib/tui/components/UserBubble.js.map +0 -1
- package/dist/src/lib/tui/components/index.js +0 -16
- package/dist/src/lib/tui/components/index.js.map +0 -1
- package/dist/src/lib/tui/ink-print.js +0 -41
- package/dist/src/lib/tui/ink-print.js.map +0 -1
- package/dist/src/test-agent-flow.js +0 -148
- package/dist/src/test-agent-flow.js.map +0 -1
- package/dist/src/test-browser-session.js +0 -152
- package/dist/src/test-browser-session.js.map +0 -1
- package/dist/src/test-browser-snapshot.js +0 -187
- package/dist/src/test-browser-snapshot.js.map +0 -1
- package/dist/src/test-snapshot-detailed.js +0 -219
- package/dist/src/test-snapshot-detailed.js.map +0 -1
- package/dist/src/test-snapshot-simple.js +0 -85
- package/dist/src/test-snapshot-simple.js.map +0 -1
- package/dist/src/test-snapshot-tabs.js +0 -110
- package/dist/src/test-snapshot-tabs.js.map +0 -1
|
@@ -1,10 +1,13 @@
|
|
|
1
1
|
import { Command } from "commander";
|
|
2
2
|
import chalk from "chalk";
|
|
3
|
+
import React from "react";
|
|
4
|
+
import { render } from "ink";
|
|
3
5
|
import { resolveTestingMode } from "../lib/testing-mode.js";
|
|
4
6
|
import { createAgentHttpClient } from "../lib/http.js";
|
|
5
7
|
import { connectAndSubscribe } from "../lib/socket.js";
|
|
6
8
|
import { registerUnifiedEventLogging } from "../lib/events.js";
|
|
7
9
|
import { getAuthToken } from "../lib/config.js";
|
|
10
|
+
import { TestRunApp } from "./test/TestRunApp.js";
|
|
8
11
|
export const startTest = new Command("start-test")
|
|
9
12
|
.description("Start a test from a natural language prompt (no URL required)")
|
|
10
13
|
.argument("<prompt...>", "Describe what to test. URL is derived from your project.")
|
|
@@ -17,6 +20,8 @@ export const startTest = new Command("start-test")
|
|
|
17
20
|
.option("--report [path]", "Generate and download PDF report to path")
|
|
18
21
|
.option("--share", "Enable public sharing after completion")
|
|
19
22
|
.option("--wait", "Block until completion and set exit code based on result")
|
|
23
|
+
.option("--ui", "Show Ink UI (TTY only)", false)
|
|
24
|
+
.option("--plain", "Disable Ink UI (plain output)", false)
|
|
20
25
|
.action(async (promptParts, opts) => {
|
|
21
26
|
try {
|
|
22
27
|
const prompt = Array.isArray(promptParts)
|
|
@@ -56,6 +61,62 @@ export const startTest = new Command("start-test")
|
|
|
56
61
|
process.exitCode = 1;
|
|
57
62
|
return;
|
|
58
63
|
}
|
|
64
|
+
const shouldUseInk = !opts.plain &&
|
|
65
|
+
process.stdout.isTTY &&
|
|
66
|
+
(Boolean(opts.ui) || Boolean(opts.wait));
|
|
67
|
+
if (shouldUseInk) {
|
|
68
|
+
let exitCode = 0;
|
|
69
|
+
const { waitUntilExit } = render(React.createElement(TestRunApp, {
|
|
70
|
+
title: "ttt start-test",
|
|
71
|
+
wait: Boolean(opts.wait),
|
|
72
|
+
onExitCode: (code) => {
|
|
73
|
+
exitCode = code;
|
|
74
|
+
},
|
|
75
|
+
run: async ({ update, signal }) => {
|
|
76
|
+
if (signal.aborted) {
|
|
77
|
+
throw new Error("Cancelled");
|
|
78
|
+
}
|
|
79
|
+
update({ phase: "auth", message: "Checking authentication…" });
|
|
80
|
+
if (!getAuthToken()) {
|
|
81
|
+
throw new Error("Not logged in. Please run 'ttt login' first.");
|
|
82
|
+
}
|
|
83
|
+
update({ phase: "start", message: "Starting test on backend…" });
|
|
84
|
+
const testMetadata = {};
|
|
85
|
+
if (docsUrl) {
|
|
86
|
+
testMetadata.docs_url = docsUrl;
|
|
87
|
+
}
|
|
88
|
+
const payload = {
|
|
89
|
+
user_prompt: prompt,
|
|
90
|
+
test_type: testingMode,
|
|
91
|
+
project_id: projectId,
|
|
92
|
+
team_id: teamId,
|
|
93
|
+
test_metadata: testMetadata,
|
|
94
|
+
};
|
|
95
|
+
const client = createAgentHttpClient({});
|
|
96
|
+
const response = await client.startTest(payload, {
|
|
97
|
+
execute_tests: opts.execute,
|
|
98
|
+
execution_mode: executionMode,
|
|
99
|
+
});
|
|
100
|
+
update({
|
|
101
|
+
testId: response.test_id,
|
|
102
|
+
phase: "connect",
|
|
103
|
+
message: "Connecting to stream…",
|
|
104
|
+
});
|
|
105
|
+
const socket = connectAndSubscribe(response.test_id);
|
|
106
|
+
const cleanup = async () => {
|
|
107
|
+
try {
|
|
108
|
+
socket.disconnect();
|
|
109
|
+
}
|
|
110
|
+
catch { }
|
|
111
|
+
};
|
|
112
|
+
return { testId: response.test_id, socket, cleanup };
|
|
113
|
+
},
|
|
114
|
+
}));
|
|
115
|
+
await waitUntilExit();
|
|
116
|
+
if (exitCode)
|
|
117
|
+
process.exitCode = exitCode;
|
|
118
|
+
return;
|
|
119
|
+
}
|
|
59
120
|
// Prepare Request
|
|
60
121
|
const testMetadata = {};
|
|
61
122
|
if (docsUrl) {
|
|
@@ -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,KAAK,MAAM,OAAO,CAAC;AAC1B,OAAO,EAAC,kBAAkB,EAAmB,MAAM,wBAAwB,CAAC;AAC5E,OAAO,EAAC,qBAAqB,EAAC,MAAM,gBAAgB,CAAC;AACrD,OAAO,EAAC,mBAAmB,EAAC,MAAM,kBAAkB,CAAC;AACrD,OAAO,EAAC,2BAA2B,EAAC,MAAM,kBAAkB,CAAC;AAE7D,OAAO,EAAC,YAAY,EAAC,MAAM,kBAAkB,CAAC;
|
|
1
|
+
{"version":3,"file":"start-test.js","sourceRoot":"","sources":["../../../src/commands/start-test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,127 @@
|
|
|
1
|
+
import React, { useEffect, useRef, useState } from "react";
|
|
2
|
+
import { Box, Text, Static, useApp, useInput } from "ink";
|
|
3
|
+
import { formatEventLines } from "../../lib/tui/events.js";
|
|
4
|
+
import { inkTheme } from "../../lib/tui/ink/theme.js";
|
|
5
|
+
export function StreamApp(props) {
|
|
6
|
+
const { socket, testId, onExit } = props;
|
|
7
|
+
const { exit } = useApp();
|
|
8
|
+
const [connectionStatus, setConnectionStatus] = useState("connecting");
|
|
9
|
+
const [events, setEvents] = useState([]);
|
|
10
|
+
const [completed, setCompleted] = useState(null);
|
|
11
|
+
const counterRef = useRef(0);
|
|
12
|
+
const socketRef = useRef(socket);
|
|
13
|
+
socketRef.current = socket;
|
|
14
|
+
const pushLines = (lines) => {
|
|
15
|
+
if (!lines.length)
|
|
16
|
+
return;
|
|
17
|
+
setEvents((prev) => {
|
|
18
|
+
const next = [
|
|
19
|
+
...prev,
|
|
20
|
+
{
|
|
21
|
+
id: `e-${counterRef.current++}`,
|
|
22
|
+
lines,
|
|
23
|
+
},
|
|
24
|
+
];
|
|
25
|
+
// Prevent unbounded growth for long sessions.
|
|
26
|
+
return next.length > 500 ? next.slice(-500) : next;
|
|
27
|
+
});
|
|
28
|
+
};
|
|
29
|
+
const handleExit = () => {
|
|
30
|
+
try {
|
|
31
|
+
if (socketRef.current.connected) {
|
|
32
|
+
socketRef.current.disconnect();
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
catch { }
|
|
36
|
+
try {
|
|
37
|
+
onExit();
|
|
38
|
+
}
|
|
39
|
+
catch { }
|
|
40
|
+
exit();
|
|
41
|
+
};
|
|
42
|
+
useInput((input, key) => {
|
|
43
|
+
if (input === "q" || input === "Q" || key.escape) {
|
|
44
|
+
handleExit();
|
|
45
|
+
}
|
|
46
|
+
});
|
|
47
|
+
useEffect(() => {
|
|
48
|
+
const onConnect = () => setConnectionStatus("connected");
|
|
49
|
+
const onDisconnect = () => setConnectionStatus("disconnected");
|
|
50
|
+
const onConnectError = () => setConnectionStatus("error");
|
|
51
|
+
const onAny = (eventType, payload) => {
|
|
52
|
+
try {
|
|
53
|
+
const lines = formatEventLines(String(eventType), payload);
|
|
54
|
+
pushLines(lines.filter(Boolean));
|
|
55
|
+
}
|
|
56
|
+
catch { }
|
|
57
|
+
};
|
|
58
|
+
const onSessionCompleted = () => {
|
|
59
|
+
setCompleted({ ok: true });
|
|
60
|
+
setTimeout(() => handleExit(), 1000);
|
|
61
|
+
};
|
|
62
|
+
const onSessionError = (payload) => {
|
|
63
|
+
setConnectionStatus("error");
|
|
64
|
+
setCompleted({
|
|
65
|
+
ok: false,
|
|
66
|
+
message: payload?.message ||
|
|
67
|
+
payload?.data?.message ||
|
|
68
|
+
payload?.data?.error ||
|
|
69
|
+
payload?.error ||
|
|
70
|
+
undefined,
|
|
71
|
+
});
|
|
72
|
+
setTimeout(() => handleExit(), 1200);
|
|
73
|
+
};
|
|
74
|
+
socket.on("connect", onConnect);
|
|
75
|
+
socket.on("disconnect", onDisconnect);
|
|
76
|
+
socket.on("connect_error", onConnectError);
|
|
77
|
+
socket.on("session_completed", onSessionCompleted);
|
|
78
|
+
socket.on("session_error", onSessionError);
|
|
79
|
+
socket.onAny?.(onAny);
|
|
80
|
+
// Initial state
|
|
81
|
+
if (socket.connected) {
|
|
82
|
+
setConnectionStatus("connected");
|
|
83
|
+
}
|
|
84
|
+
return () => {
|
|
85
|
+
socket.off("connect", onConnect);
|
|
86
|
+
socket.off("disconnect", onDisconnect);
|
|
87
|
+
socket.off("connect_error", onConnectError);
|
|
88
|
+
socket.off("session_completed", onSessionCompleted);
|
|
89
|
+
socket.off("session_error", onSessionError);
|
|
90
|
+
socket.offAny?.(onAny);
|
|
91
|
+
};
|
|
92
|
+
}, [socket]);
|
|
93
|
+
const statusColor = connectionStatus === "connected"
|
|
94
|
+
? "green"
|
|
95
|
+
: connectionStatus === "error"
|
|
96
|
+
? "red"
|
|
97
|
+
: connectionStatus === "disconnected"
|
|
98
|
+
? "gray"
|
|
99
|
+
: "yellow";
|
|
100
|
+
const statusText = connectionStatus === "connected"
|
|
101
|
+
? "Connected"
|
|
102
|
+
: connectionStatus === "error"
|
|
103
|
+
? "Error"
|
|
104
|
+
: connectionStatus === "disconnected"
|
|
105
|
+
? "Disconnected"
|
|
106
|
+
: "Connecting…";
|
|
107
|
+
return (React.createElement(Box, { flexDirection: "column", height: "100%" },
|
|
108
|
+
React.createElement(Box, { paddingX: 1, paddingY: 1, borderStyle: "single", borderColor: inkTheme.colors.border.default, flexDirection: "column" },
|
|
109
|
+
React.createElement(Box, { flexDirection: "row", justifyContent: "space-between" },
|
|
110
|
+
React.createElement(Text, { bold: true }, "Streaming Test Events"),
|
|
111
|
+
React.createElement(Text, { color: statusColor }, statusText)),
|
|
112
|
+
React.createElement(Box, { marginTop: 1 },
|
|
113
|
+
React.createElement(Text, { color: "gray" }, "Test ID: "),
|
|
114
|
+
React.createElement(Text, { color: "white" }, String(testId))),
|
|
115
|
+
completed && (React.createElement(Box, { marginTop: 1 }, completed.ok ? (React.createElement(Text, { color: "green" }, "Session completed")) : (React.createElement(Text, { color: "red" },
|
|
116
|
+
"Session ended with error",
|
|
117
|
+
completed.message ? `: ${completed.message}` : ""))))),
|
|
118
|
+
React.createElement(Box, { flexDirection: "column", flexGrow: 1, paddingX: 1, paddingY: 1 }, events.length === 0 ? (React.createElement(Text, { color: "gray", dimColor: true }, "Waiting for events\u2026")) : (React.createElement(Static, { items: events }, (event) => (React.createElement(Box, { key: event.id, flexDirection: "column" }, event.lines.map((line, idx) => (React.createElement(Text, { key: `${event.id}-${idx}` }, line)))))))),
|
|
119
|
+
React.createElement(Box, { paddingX: 1, paddingY: 1, borderStyle: "single", borderColor: inkTheme.colors.border.default },
|
|
120
|
+
React.createElement(Text, { color: inkTheme.colors.text.muted, dimColor: true },
|
|
121
|
+
"Press ",
|
|
122
|
+
React.createElement(Text, { bold: true }, "q"),
|
|
123
|
+
" or ",
|
|
124
|
+
React.createElement(Text, { bold: true }, "ESC"),
|
|
125
|
+
" to exit"))));
|
|
126
|
+
}
|
|
127
|
+
//# sourceMappingURL=StreamApp.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"StreamApp.js","sourceRoot":"","sources":["../../../../src/commands/stream/StreamApp.tsx"],"names":[],"mappings":"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"}
|
|
@@ -1,17 +1,52 @@
|
|
|
1
1
|
import { Command } from "commander";
|
|
2
|
+
import chalk from "chalk";
|
|
3
|
+
import React from "react";
|
|
4
|
+
import { render } from "ink";
|
|
2
5
|
import { connectAndSubscribe } from "../lib/socket.js";
|
|
3
6
|
import { registerUnifiedEventLogging } from "../lib/events.js";
|
|
7
|
+
import { getAuthToken } from "../lib/config.js";
|
|
8
|
+
import { StreamApp } from "./stream/StreamApp.js";
|
|
4
9
|
export const stream = new Command("stream")
|
|
5
10
|
.description("Attach to a running test and stream events")
|
|
6
11
|
.argument("<test-id>")
|
|
7
12
|
.action(async (testId) => {
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
13
|
+
try {
|
|
14
|
+
if (!getAuthToken()) {
|
|
15
|
+
console.error(chalk.red("Not logged in. Please run 'ttt login' first."));
|
|
16
|
+
process.exitCode = 1;
|
|
17
|
+
return;
|
|
18
|
+
}
|
|
19
|
+
const socket = connectAndSubscribe(testId);
|
|
20
|
+
// Non-interactive fallback (CI / piping).
|
|
21
|
+
if (!process.stdout.isTTY) {
|
|
22
|
+
registerUnifiedEventLogging(socket, (line) => console.log(line));
|
|
23
|
+
socket.on("session_completed", async () => {
|
|
24
|
+
await socket.disconnect();
|
|
25
|
+
});
|
|
26
|
+
socket.on("session_error", async () => {
|
|
27
|
+
await socket.disconnect();
|
|
28
|
+
});
|
|
29
|
+
return;
|
|
30
|
+
}
|
|
31
|
+
const onExit = () => {
|
|
32
|
+
try {
|
|
33
|
+
if (socket.connected)
|
|
34
|
+
socket.disconnect();
|
|
35
|
+
}
|
|
36
|
+
catch { }
|
|
37
|
+
};
|
|
38
|
+
const cleanup = () => {
|
|
39
|
+
onExit();
|
|
40
|
+
process.exit(0);
|
|
41
|
+
};
|
|
42
|
+
process.on("SIGINT", cleanup);
|
|
43
|
+
process.on("SIGTERM", cleanup);
|
|
44
|
+
const { waitUntilExit } = render(React.createElement(StreamApp, { socket, testId, onExit }));
|
|
45
|
+
await waitUntilExit();
|
|
46
|
+
}
|
|
47
|
+
catch (err) {
|
|
48
|
+
console.error(chalk.red(err?.message || String(err)));
|
|
49
|
+
process.exitCode = 1;
|
|
50
|
+
}
|
|
16
51
|
});
|
|
17
52
|
//# sourceMappingURL=stream.js.map
|
|
@@ -1 +1 @@
|
|
|
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;
|
|
1
|
+
{"version":3,"file":"stream.js","sourceRoot":"","sources":["../../../src/commands/stream.ts"],"names":[],"mappings":"AAAA,OAAO,EAAC,OAAO,EAAC,MAAM,WAAW,CAAC;AAClC,OAAO,KAAK,MAAM,OAAO,CAAC;AAC1B,OAAO,KAAK,MAAM,OAAO,CAAC;AAC1B,OAAO,EAAC,MAAM,EAAC,MAAM,KAAK,CAAC;AAC3B,OAAO,EAAC,mBAAmB,EAAC,MAAM,kBAAkB,CAAC;AACrD,OAAO,EAAC,2BAA2B,EAAC,MAAM,kBAAkB,CAAC;AAC7D,OAAO,EAAC,YAAY,EAAC,MAAM,kBAAkB,CAAC;AAC9C,OAAO,EAAC,SAAS,EAAC,MAAM,uBAAuB,CAAC;AAEhD,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,IAAI,CAAC;QACH,IAAI,CAAC,YAAY,EAAE,EAAE,CAAC;YACpB,OAAO,CAAC,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,8CAA8C,CAAC,CAAC,CAAC;YACzE,OAAO,CAAC,QAAQ,GAAG,CAAC,CAAC;YACrB,OAAO;QACT,CAAC;QAED,MAAM,MAAM,GAAG,mBAAmB,CAAC,MAAM,CAAC,CAAC;QAE3C,0CAA0C;QAC1C,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC;YAC1B,2BAA2B,CAAC,MAAM,EAAE,CAAC,IAAI,EAAE,EAAE,CAAC,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC;YACjE,MAAM,CAAC,EAAE,CAAC,mBAAmB,EAAE,KAAK,IAAI,EAAE;gBACxC,MAAM,MAAM,CAAC,UAAU,EAAE,CAAC;YAC5B,CAAC,CAAC,CAAC;YACH,MAAM,CAAC,EAAE,CAAC,eAAe,EAAE,KAAK,IAAI,EAAE;gBACpC,MAAM,MAAM,CAAC,UAAU,EAAE,CAAC;YAC5B,CAAC,CAAC,CAAC;YACH,OAAO;QACT,CAAC;QAED,MAAM,MAAM,GAAG,GAAG,EAAE;YAClB,IAAI,CAAC;gBACH,IAAI,MAAM,CAAC,SAAS;oBAAE,MAAM,CAAC,UAAU,EAAE,CAAC;YAC5C,CAAC;YAAC,MAAM,CAAC,CAAA,CAAC;QACZ,CAAC,CAAC;QAEF,MAAM,OAAO,GAAG,GAAG,EAAE;YACnB,MAAM,EAAE,CAAC;YACT,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QAClB,CAAC,CAAC;QAEF,OAAO,CAAC,EAAE,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;QAC9B,OAAO,CAAC,EAAE,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC;QAE/B,MAAM,EAAC,aAAa,EAAC,GAAG,MAAM,CAC5B,KAAK,CAAC,aAAa,CAAC,SAAS,EAAE,EAAC,MAAM,EAAE,MAAM,EAAE,MAAM,EAAC,CAAC,CACzD,CAAC;QACF,MAAM,aAAa,EAAE,CAAC;IACxB,CAAC;IAAC,OAAO,GAAQ,EAAE,CAAC;QAClB,OAAO,CAAC,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,GAAG,EAAE,OAAO,IAAI,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;QACtD,OAAO,CAAC,QAAQ,GAAG,CAAC,CAAC;IACvB,CAAC;AACH,CAAC,CAAC,CAAC"}
|
|
@@ -0,0 +1,183 @@
|
|
|
1
|
+
import React, { useEffect, useMemo, useRef, useState } from "react";
|
|
2
|
+
import { Box, Text, Static, useApp, useInput } from "ink";
|
|
3
|
+
import { formatEventLines } from "../../lib/tui/events.js";
|
|
4
|
+
import { inkTheme } from "../../lib/tui/ink/theme.js";
|
|
5
|
+
export function TestRunApp(props) {
|
|
6
|
+
const { title, wait, run, onExitCode } = props;
|
|
7
|
+
const { exit } = useApp();
|
|
8
|
+
const [phase, setPhase] = useState({
|
|
9
|
+
phase: "auth",
|
|
10
|
+
message: "Preparing…",
|
|
11
|
+
});
|
|
12
|
+
const [events, setEvents] = useState([]);
|
|
13
|
+
const [done, setDone] = useState(null);
|
|
14
|
+
const abortRef = useRef(null);
|
|
15
|
+
const socketRef = useRef(null);
|
|
16
|
+
const cleanupRef = useRef(null);
|
|
17
|
+
const counterRef = useRef(0);
|
|
18
|
+
const update = (patch) => {
|
|
19
|
+
setPhase((prev) => ({ ...prev, ...patch }));
|
|
20
|
+
};
|
|
21
|
+
const pushLines = (lines) => {
|
|
22
|
+
if (!lines.length)
|
|
23
|
+
return;
|
|
24
|
+
setEvents((prev) => {
|
|
25
|
+
const next = [
|
|
26
|
+
...prev,
|
|
27
|
+
{ id: `e-${counterRef.current++}`, lines: lines.filter(Boolean) },
|
|
28
|
+
];
|
|
29
|
+
return next.length > 800 ? next.slice(-800) : next;
|
|
30
|
+
});
|
|
31
|
+
};
|
|
32
|
+
const cleanupOnly = async (code) => {
|
|
33
|
+
if (typeof code === "number") {
|
|
34
|
+
onExitCode?.(code);
|
|
35
|
+
}
|
|
36
|
+
try {
|
|
37
|
+
abortRef.current?.abort();
|
|
38
|
+
}
|
|
39
|
+
catch { }
|
|
40
|
+
try {
|
|
41
|
+
await cleanupRef.current?.();
|
|
42
|
+
}
|
|
43
|
+
catch { }
|
|
44
|
+
try {
|
|
45
|
+
if (socketRef.current?.connected) {
|
|
46
|
+
socketRef.current.disconnect();
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
catch { }
|
|
50
|
+
};
|
|
51
|
+
const gracefulExit = async (code) => {
|
|
52
|
+
await cleanupOnly(code);
|
|
53
|
+
exit();
|
|
54
|
+
};
|
|
55
|
+
useInput((input, key) => {
|
|
56
|
+
if (input === "q" || input === "Q" || key.escape) {
|
|
57
|
+
void gracefulExit();
|
|
58
|
+
}
|
|
59
|
+
});
|
|
60
|
+
useEffect(() => {
|
|
61
|
+
const abort = new AbortController();
|
|
62
|
+
abortRef.current = abort;
|
|
63
|
+
let didUnmount = false;
|
|
64
|
+
const start = async () => {
|
|
65
|
+
try {
|
|
66
|
+
const result = await run({ update, signal: abort.signal });
|
|
67
|
+
if (didUnmount)
|
|
68
|
+
return;
|
|
69
|
+
update({ testId: result.testId, phase: "connect", message: "Connected" });
|
|
70
|
+
socketRef.current = result.socket;
|
|
71
|
+
cleanupRef.current = result.cleanup ?? null;
|
|
72
|
+
const onAny = (eventType, payload) => {
|
|
73
|
+
try {
|
|
74
|
+
pushLines(formatEventLines(String(eventType), payload));
|
|
75
|
+
}
|
|
76
|
+
catch { }
|
|
77
|
+
};
|
|
78
|
+
const onCompleted = async () => {
|
|
79
|
+
setDone({ ok: true });
|
|
80
|
+
update({ phase: "completed", message: "Session completed" });
|
|
81
|
+
if (wait) {
|
|
82
|
+
setTimeout(() => void gracefulExit(0), 800);
|
|
83
|
+
}
|
|
84
|
+
};
|
|
85
|
+
const onError = async () => {
|
|
86
|
+
setDone({ ok: false });
|
|
87
|
+
update({ phase: "error", message: "Session ended with error" });
|
|
88
|
+
if (wait) {
|
|
89
|
+
setTimeout(() => void gracefulExit(1), 1200);
|
|
90
|
+
}
|
|
91
|
+
};
|
|
92
|
+
result.socket.on("session_completed", onCompleted);
|
|
93
|
+
result.socket.on("session_error", onError);
|
|
94
|
+
result.socket.onAny?.(onAny);
|
|
95
|
+
update({ phase: "streaming", message: wait ? "Waiting for completion…" : "Streaming…" });
|
|
96
|
+
// Cleanup listeners when unmounting/exiting.
|
|
97
|
+
const previousCleanup = cleanupRef.current;
|
|
98
|
+
cleanupRef.current = async () => {
|
|
99
|
+
try {
|
|
100
|
+
result.socket.off("session_completed", onCompleted);
|
|
101
|
+
result.socket.off("session_error", onError);
|
|
102
|
+
result.socket.offAny?.(onAny);
|
|
103
|
+
}
|
|
104
|
+
catch { }
|
|
105
|
+
if (previousCleanup) {
|
|
106
|
+
await previousCleanup();
|
|
107
|
+
}
|
|
108
|
+
};
|
|
109
|
+
}
|
|
110
|
+
catch (err) {
|
|
111
|
+
if (didUnmount)
|
|
112
|
+
return;
|
|
113
|
+
setDone({ ok: false });
|
|
114
|
+
update({
|
|
115
|
+
phase: "error",
|
|
116
|
+
message: err?.message || String(err) || "Unknown error",
|
|
117
|
+
});
|
|
118
|
+
onExitCode?.(1);
|
|
119
|
+
if (wait) {
|
|
120
|
+
setTimeout(() => void gracefulExit(1), 1500);
|
|
121
|
+
}
|
|
122
|
+
}
|
|
123
|
+
};
|
|
124
|
+
void start();
|
|
125
|
+
return () => {
|
|
126
|
+
didUnmount = true;
|
|
127
|
+
void cleanupOnly();
|
|
128
|
+
};
|
|
129
|
+
// `run` is stable per render() call.
|
|
130
|
+
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
131
|
+
}, []);
|
|
132
|
+
const phaseLabel = useMemo(() => {
|
|
133
|
+
switch (phase.phase) {
|
|
134
|
+
case "auth":
|
|
135
|
+
return "Authenticating";
|
|
136
|
+
case "tunnel":
|
|
137
|
+
return "Creating tunnel";
|
|
138
|
+
case "start":
|
|
139
|
+
return "Starting test";
|
|
140
|
+
case "connect":
|
|
141
|
+
return "Connecting";
|
|
142
|
+
case "streaming":
|
|
143
|
+
return "Streaming";
|
|
144
|
+
case "completed":
|
|
145
|
+
return "Completed";
|
|
146
|
+
case "error":
|
|
147
|
+
return "Error";
|
|
148
|
+
default:
|
|
149
|
+
return phase.phase;
|
|
150
|
+
}
|
|
151
|
+
}, [phase.phase]);
|
|
152
|
+
const phaseColor = phase.phase === "error"
|
|
153
|
+
? "red"
|
|
154
|
+
: phase.phase === "completed"
|
|
155
|
+
? "green"
|
|
156
|
+
: phase.phase === "streaming"
|
|
157
|
+
? "cyan"
|
|
158
|
+
: "yellow";
|
|
159
|
+
return (React.createElement(Box, { flexDirection: "column", height: "100%" },
|
|
160
|
+
React.createElement(Box, { paddingX: 1, paddingY: 1, borderStyle: "single", borderColor: inkTheme.colors.border.default, flexDirection: "column" },
|
|
161
|
+
React.createElement(Box, { flexDirection: "row", justifyContent: "space-between" },
|
|
162
|
+
React.createElement(Text, { bold: true }, title),
|
|
163
|
+
React.createElement(Text, { color: phaseColor },
|
|
164
|
+
phaseLabel,
|
|
165
|
+
done ? (done.ok ? " ✓" : " ✕") : "")),
|
|
166
|
+
phase.testId !== undefined && (React.createElement(Box, { marginTop: 1 },
|
|
167
|
+
React.createElement(Text, { color: "gray" }, "Test ID: "),
|
|
168
|
+
React.createElement(Text, { color: "white" }, String(phase.testId)))),
|
|
169
|
+
phase.tunnelUrl && (React.createElement(Box, { marginTop: 0 },
|
|
170
|
+
React.createElement(Text, { color: "gray" }, "Tunnel: "),
|
|
171
|
+
React.createElement(Text, { color: "white" }, phase.tunnelUrl))),
|
|
172
|
+
phase.message && (React.createElement(Box, { marginTop: 1 },
|
|
173
|
+
React.createElement(Text, { color: "gray" }, phase.message)))),
|
|
174
|
+
React.createElement(Box, { flexDirection: "column", flexGrow: 1, paddingX: 1, paddingY: 1 }, events.length === 0 ? (React.createElement(Text, { color: "gray", dimColor: true }, "Waiting for events\u2026")) : (React.createElement(Static, { items: events }, (event) => (React.createElement(Box, { key: event.id, flexDirection: "column" }, event.lines.map((line, idx) => (React.createElement(Text, { key: `${event.id}-${idx}` }, line)))))))),
|
|
175
|
+
React.createElement(Box, { paddingX: 1, paddingY: 1, borderStyle: "single", borderColor: inkTheme.colors.border.default },
|
|
176
|
+
React.createElement(Text, { color: inkTheme.colors.text.muted, dimColor: true },
|
|
177
|
+
"Press ",
|
|
178
|
+
React.createElement(Text, { bold: true }, "q"),
|
|
179
|
+
" or ",
|
|
180
|
+
React.createElement(Text, { bold: true }, "ESC"),
|
|
181
|
+
" to exit"))));
|
|
182
|
+
}
|
|
183
|
+
//# sourceMappingURL=TestRunApp.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"TestRunApp.js","sourceRoot":"","sources":["../../../../src/commands/test/TestRunApp.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,EAAC,SAAS,EAAE,OAAO,EAAE,MAAM,EAAE,QAAQ,EAAC,MAAM,OAAO,CAAC;AAClE,OAAO,EAAC,GAAG,EAAE,IAAI,EAAE,MAAM,EAAE,MAAM,EAAE,QAAQ,EAAC,MAAM,KAAK,CAAC;AAExD,OAAO,EAAC,gBAAgB,EAAC,MAAM,yBAAyB,CAAC;AACzD,OAAO,EAAC,QAAQ,EAAC,MAAM,4BAA4B,CAAC;AAkCpD,MAAM,UAAU,UAAU,CAAC,KAK1B;IACC,MAAM,EAAC,KAAK,EAAE,IAAI,EAAE,GAAG,EAAE,UAAU,EAAC,GAAG,KAAK,CAAC;IAC7C,MAAM,EAAC,IAAI,EAAC,GAAG,MAAM,EAAE,CAAC;IAExB,MAAM,CAAC,KAAK,EAAE,QAAQ,CAAC,GAAG,QAAQ,CAAa;QAC7C,KAAK,EAAE,MAAM;QACb,OAAO,EAAE,YAAY;KACtB,CAAC,CAAC;IACH,MAAM,CAAC,MAAM,EAAE,SAAS,CAAC,GAAG,QAAQ,CAAkB,EAAE,CAAC,CAAC;IAC1D,MAAM,CAAC,IAAI,EAAE,OAAO,CAAC,GAAG,QAAQ,CAAkC,IAAI,CAAC,CAAC;IAExE,MAAM,QAAQ,GAAG,MAAM,CAAyB,IAAI,CAAC,CAAC;IACtD,MAAM,SAAS,GAAG,MAAM,CAAgB,IAAI,CAAC,CAAC;IAC9C,MAAM,UAAU,GAAG,MAAM,CAAsC,IAAI,CAAC,CAAC;IACrE,MAAM,UAAU,GAAG,MAAM,CAAC,CAAC,CAAC,CAAC;IAE7B,MAAM,MAAM,GAAG,CAAC,KAA0B,EAAE,EAAE;QAC5C,QAAQ,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC,EAAC,GAAG,IAAI,EAAE,GAAG,KAAK,EAAC,CAAC,CAAC,CAAC;IAC5C,CAAC,CAAC;IAEF,MAAM,SAAS,GAAG,CAAC,KAAe,EAAE,EAAE;QACpC,IAAI,CAAC,KAAK,CAAC,MAAM;YAAE,OAAO;QAC1B,SAAS,CAAC,CAAC,IAAI,EAAE,EAAE;YACjB,MAAM,IAAI,GAAG;gBACX,GAAG,IAAI;gBACP,EAAC,EAAE,EAAE,KAAK,UAAU,CAAC,OAAO,EAAE,EAAE,EAAE,KAAK,EAAE,KAAK,CAAC,MAAM,CAAC,OAAO,CAAC,EAAC;aAChE,CAAC;YACF,OAAO,IAAI,CAAC,MAAM,GAAG,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;QACrD,CAAC,CAAC,CAAC;IACL,CAAC,CAAC;IAEF,MAAM,WAAW,GAAG,KAAK,EAAE,IAAa,EAAE,EAAE;QAC1C,IAAI,OAAO,IAAI,KAAK,QAAQ,EAAE,CAAC;YAC7B,UAAU,EAAE,CAAC,IAAI,CAAC,CAAC;QACrB,CAAC;QACD,IAAI,CAAC;YACH,QAAQ,CAAC,OAAO,EAAE,KAAK,EAAE,CAAC;QAC5B,CAAC;QAAC,MAAM,CAAC,CAAA,CAAC;QACV,IAAI,CAAC;YACH,MAAM,UAAU,CAAC,OAAO,EAAE,EAAE,CAAC;QAC/B,CAAC;QAAC,MAAM,CAAC,CAAA,CAAC;QACV,IAAI,CAAC;YACH,IAAI,SAAS,CAAC,OAAO,EAAE,SAAS,EAAE,CAAC;gBACjC,SAAS,CAAC,OAAO,CAAC,UAAU,EAAE,CAAC;YACjC,CAAC;QACH,CAAC;QAAC,MAAM,CAAC,CAAA,CAAC;IACZ,CAAC,CAAC;IAEF,MAAM,YAAY,GAAG,KAAK,EAAE,IAAa,EAAE,EAAE;QAC3C,MAAM,WAAW,CAAC,IAAI,CAAC,CAAC;QACxB,IAAI,EAAE,CAAC;IACT,CAAC,CAAC;IAEF,QAAQ,CAAC,CAAC,KAAK,EAAE,GAAG,EAAE,EAAE;QACtB,IAAI,KAAK,KAAK,GAAG,IAAI,KAAK,KAAK,GAAG,IAAI,GAAG,CAAC,MAAM,EAAE,CAAC;YACjD,KAAK,YAAY,EAAE,CAAC;QACtB,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,SAAS,CAAC,GAAG,EAAE;QACb,MAAM,KAAK,GAAG,IAAI,eAAe,EAAE,CAAC;QACpC,QAAQ,CAAC,OAAO,GAAG,KAAK,CAAC;QAEzB,IAAI,UAAU,GAAG,KAAK,CAAC;QAEvB,MAAM,KAAK,GAAG,KAAK,IAAI,EAAE;YACvB,IAAI,CAAC;gBACH,MAAM,MAAM,GAAG,MAAM,GAAG,CAAC,EAAC,MAAM,EAAE,MAAM,EAAE,KAAK,CAAC,MAAM,EAAC,CAAC,CAAC;gBACzD,IAAI,UAAU;oBAAE,OAAO;gBAEvB,MAAM,CAAC,EAAC,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,KAAK,EAAE,SAAS,EAAE,OAAO,EAAE,WAAW,EAAC,CAAC,CAAC;gBACxE,SAAS,CAAC,OAAO,GAAG,MAAM,CAAC,MAAM,CAAC;gBAClC,UAAU,CAAC,OAAO,GAAG,MAAM,CAAC,OAAO,IAAI,IAAI,CAAC;gBAE5C,MAAM,KAAK,GAAG,CAAC,SAAiB,EAAE,OAAY,EAAE,EAAE;oBAChD,IAAI,CAAC;wBACH,SAAS,CAAC,gBAAgB,CAAC,MAAM,CAAC,SAAS,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC;oBAC1D,CAAC;oBAAC,MAAM,CAAC,CAAA,CAAC;gBACZ,CAAC,CAAC;gBAEF,MAAM,WAAW,GAAG,KAAK,IAAI,EAAE;oBAC7B,OAAO,CAAC,EAAC,EAAE,EAAE,IAAI,EAAC,CAAC,CAAC;oBACpB,MAAM,CAAC,EAAC,KAAK,EAAE,WAAW,EAAE,OAAO,EAAE,mBAAmB,EAAC,CAAC,CAAC;oBAC3D,IAAI,IAAI,EAAE,CAAC;wBACT,UAAU,CAAC,GAAG,EAAE,CAAC,KAAK,YAAY,CAAC,CAAC,CAAC,EAAE,GAAG,CAAC,CAAC;oBAC9C,CAAC;gBACH,CAAC,CAAC;gBAEF,MAAM,OAAO,GAAG,KAAK,IAAI,EAAE;oBACzB,OAAO,CAAC,EAAC,EAAE,EAAE,KAAK,EAAC,CAAC,CAAC;oBACrB,MAAM,CAAC,EAAC,KAAK,EAAE,OAAO,EAAE,OAAO,EAAE,0BAA0B,EAAC,CAAC,CAAC;oBAC9D,IAAI,IAAI,EAAE,CAAC;wBACT,UAAU,CAAC,GAAG,EAAE,CAAC,KAAK,YAAY,CAAC,CAAC,CAAC,EAAE,IAAI,CAAC,CAAC;oBAC/C,CAAC;gBACH,CAAC,CAAC;gBAEF,MAAM,CAAC,MAAM,CAAC,EAAE,CAAC,mBAAmB,EAAE,WAAW,CAAC,CAAC;gBACnD,MAAM,CAAC,MAAM,CAAC,EAAE,CAAC,eAAe,EAAE,OAAO,CAAC,CAAC;gBAC1C,MAAM,CAAC,MAAc,CAAC,KAAK,EAAE,CAAC,KAAK,CAAC,CAAC;gBAEtC,MAAM,CAAC,EAAC,KAAK,EAAE,WAAW,EAAE,OAAO,EAAE,IAAI,CAAC,CAAC,CAAC,yBAAyB,CAAC,CAAC,CAAC,YAAY,EAAC,CAAC,CAAC;gBAEvF,6CAA6C;gBAC7C,MAAM,eAAe,GAAG,UAAU,CAAC,OAAO,CAAC;gBAC3C,UAAU,CAAC,OAAO,GAAG,KAAK,IAAI,EAAE;oBAC9B,IAAI,CAAC;wBACH,MAAM,CAAC,MAAM,CAAC,GAAG,CAAC,mBAAmB,EAAE,WAAW,CAAC,CAAC;wBACpD,MAAM,CAAC,MAAM,CAAC,GAAG,CAAC,eAAe,EAAE,OAAO,CAAC,CAAC;wBAC3C,MAAM,CAAC,MAAc,CAAC,MAAM,EAAE,CAAC,KAAK,CAAC,CAAC;oBACzC,CAAC;oBAAC,MAAM,CAAC,CAAA,CAAC;oBACV,IAAI,eAAe,EAAE,CAAC;wBACpB,MAAM,eAAe,EAAE,CAAC;oBAC1B,CAAC;gBACH,CAAC,CAAC;YACJ,CAAC;YAAC,OAAO,GAAQ,EAAE,CAAC;gBAClB,IAAI,UAAU;oBAAE,OAAO;gBACvB,OAAO,CAAC,EAAC,EAAE,EAAE,KAAK,EAAC,CAAC,CAAC;gBACrB,MAAM,CAAC;oBACL,KAAK,EAAE,OAAO;oBACd,OAAO,EAAE,GAAG,EAAE,OAAO,IAAI,MAAM,CAAC,GAAG,CAAC,IAAI,eAAe;iBACxD,CAAC,CAAC;gBACH,UAAU,EAAE,CAAC,CAAC,CAAC,CAAC;gBAChB,IAAI,IAAI,EAAE,CAAC;oBACT,UAAU,CAAC,GAAG,EAAE,CAAC,KAAK,YAAY,CAAC,CAAC,CAAC,EAAE,IAAI,CAAC,CAAC;gBAC/C,CAAC;YACH,CAAC;QACH,CAAC,CAAC;QAEF,KAAK,KAAK,EAAE,CAAC;QAEb,OAAO,GAAG,EAAE;YACV,UAAU,GAAG,IAAI,CAAC;YAClB,KAAK,WAAW,EAAE,CAAC;QACrB,CAAC,CAAC;QACF,qCAAqC;QACrC,uDAAuD;IACzD,CAAC,EAAE,EAAE,CAAC,CAAC;IAEP,MAAM,UAAU,GAAG,OAAO,CAAC,GAAG,EAAE;QAC9B,QAAQ,KAAK,CAAC,KAAK,EAAE,CAAC;YACpB,KAAK,MAAM;gBACT,OAAO,gBAAgB,CAAC;YAC1B,KAAK,QAAQ;gBACX,OAAO,iBAAiB,CAAC;YAC3B,KAAK,OAAO;gBACV,OAAO,eAAe,CAAC;YACzB,KAAK,SAAS;gBACZ,OAAO,YAAY,CAAC;YACtB,KAAK,WAAW;gBACd,OAAO,WAAW,CAAC;YACrB,KAAK,WAAW;gBACd,OAAO,WAAW,CAAC;YACrB,KAAK,OAAO;gBACV,OAAO,OAAO,CAAC;YACjB;gBACE,OAAO,KAAK,CAAC,KAAK,CAAC;QACvB,CAAC;IACH,CAAC,EAAE,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC;IAElB,MAAM,UAAU,GACd,KAAK,CAAC,KAAK,KAAK,OAAO;QACrB,CAAC,CAAC,KAAK;QACP,CAAC,CAAC,KAAK,CAAC,KAAK,KAAK,WAAW;YAC3B,CAAC,CAAC,OAAO;YACT,CAAC,CAAC,KAAK,CAAC,KAAK,KAAK,WAAW;gBAC3B,CAAC,CAAC,MAAM;gBACR,CAAC,CAAC,QAAQ,CAAC;IAEnB,OAAO,CACL,oBAAC,GAAG,IAAC,aAAa,EAAC,QAAQ,EAAC,MAAM,EAAC,MAAM;QACvC,oBAAC,GAAG,IACF,QAAQ,EAAE,CAAC,EACX,QAAQ,EAAE,CAAC,EACX,WAAW,EAAC,QAAQ,EACpB,WAAW,EAAE,QAAQ,CAAC,MAAM,CAAC,MAAM,CAAC,OAAO,EAC3C,aAAa,EAAC,QAAQ;YAEtB,oBAAC,GAAG,IAAC,aAAa,EAAC,KAAK,EAAC,cAAc,EAAC,eAAe;gBACrD,oBAAC,IAAI,IAAC,IAAI,UAAE,KAAK,CAAQ;gBACzB,oBAAC,IAAI,IAAC,KAAK,EAAE,UAAU;oBACpB,UAAU;oBACV,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,EAAE,CAC/B,CACH;YACL,KAAK,CAAC,MAAM,KAAK,SAAS,IAAI,CAC7B,oBAAC,GAAG,IAAC,SAAS,EAAE,CAAC;gBACf,oBAAC,IAAI,IAAC,KAAK,EAAC,MAAM,gBAAiB;gBACnC,oBAAC,IAAI,IAAC,KAAK,EAAC,OAAO,IAAE,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,CAAQ,CAC7C,CACP;YACA,KAAK,CAAC,SAAS,IAAI,CAClB,oBAAC,GAAG,IAAC,SAAS,EAAE,CAAC;gBACf,oBAAC,IAAI,IAAC,KAAK,EAAC,MAAM,eAAgB;gBAClC,oBAAC,IAAI,IAAC,KAAK,EAAC,OAAO,IAAE,KAAK,CAAC,SAAS,CAAQ,CACxC,CACP;YACA,KAAK,CAAC,OAAO,IAAI,CAChB,oBAAC,GAAG,IAAC,SAAS,EAAE,CAAC;gBACf,oBAAC,IAAI,IAAC,KAAK,EAAC,MAAM,IAAE,KAAK,CAAC,OAAO,CAAQ,CACrC,CACP,CACG;QAEN,oBAAC,GAAG,IAAC,aAAa,EAAC,QAAQ,EAAC,QAAQ,EAAE,CAAC,EAAE,QAAQ,EAAE,CAAC,EAAE,QAAQ,EAAE,CAAC,IAC9D,MAAM,CAAC,MAAM,KAAK,CAAC,CAAC,CAAC,CAAC,CACrB,oBAAC,IAAI,IAAC,KAAK,EAAC,MAAM,EAAC,QAAQ,qCAEpB,CACR,CAAC,CAAC,CAAC,CACF,oBAAC,MAAM,IAAC,KAAK,EAAE,MAAM,IAClB,CAAC,KAAK,EAAE,EAAE,CAAC,CACV,oBAAC,GAAG,IAAC,GAAG,EAAE,KAAK,CAAC,EAAE,EAAE,aAAa,EAAC,QAAQ,IACvC,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,GAAG,EAAE,EAAE,CAAC,CAC9B,oBAAC,IAAI,IAAC,GAAG,EAAE,GAAG,KAAK,CAAC,EAAE,IAAI,GAAG,EAAE,IAAG,IAAI,CAAQ,CAC/C,CAAC,CACE,CACP,CACM,CACV,CACG;QAEN,oBAAC,GAAG,IACF,QAAQ,EAAE,CAAC,EACX,QAAQ,EAAE,CAAC,EACX,WAAW,EAAC,QAAQ,EACpB,WAAW,EAAE,QAAQ,CAAC,MAAM,CAAC,MAAM,CAAC,OAAO;YAE3C,oBAAC,IAAI,IAAC,KAAK,EAAE,QAAQ,CAAC,MAAM,CAAC,IAAI,CAAC,KAAK,EAAE,QAAQ;;gBACzC,oBAAC,IAAI,IAAC,IAAI,cAAS;;gBAAI,oBAAC,IAAI,IAAC,IAAI,gBAAW;2BAC7C,CACH,CACF,CACP,CAAC;AACJ,CAAC"}
|
|
@@ -1,11 +1,14 @@
|
|
|
1
1
|
import { Command } from "commander";
|
|
2
2
|
import chalk from "chalk";
|
|
3
|
+
import React from "react";
|
|
4
|
+
import { render } from "ink";
|
|
3
5
|
import { resolveTestingMode } from "../lib/testing-mode.js";
|
|
4
6
|
import { createAgentHttpClient } from "../lib/http.js";
|
|
5
7
|
import { createTunnelIfNeeded } from "../lib/ngrok.js";
|
|
6
8
|
import { connectAndSubscribe } from "../lib/socket.js";
|
|
7
9
|
import { registerUnifiedEventLogging } from "../lib/events.js";
|
|
8
10
|
import { getAuthToken } from "../lib/config.js";
|
|
11
|
+
import { TestRunApp } from "./test/TestRunApp.js";
|
|
9
12
|
const formatFlags = (opts) => {
|
|
10
13
|
const flags = [];
|
|
11
14
|
if (opts.execute)
|
|
@@ -34,6 +37,8 @@ export const test = new Command("test")
|
|
|
34
37
|
.option("--report [path]", "Generate and download PDF report to path")
|
|
35
38
|
.option("--share", "Enable public sharing after completion")
|
|
36
39
|
.option("--wait", "Block until completion and set exit code based on result")
|
|
40
|
+
.option("--ui", "Show Ink UI (TTY only)", false)
|
|
41
|
+
.option("--plain", "Disable Ink UI (plain output)", false)
|
|
37
42
|
.action(async (promptParts, opts) => {
|
|
38
43
|
let tunnel = null;
|
|
39
44
|
try {
|
|
@@ -100,6 +105,98 @@ export const test = new Command("test")
|
|
|
100
105
|
process.exitCode = 1;
|
|
101
106
|
return;
|
|
102
107
|
}
|
|
108
|
+
const shouldUseInk = !opts.plain &&
|
|
109
|
+
process.stdout.isTTY &&
|
|
110
|
+
(Boolean(opts.ui) || Boolean(opts.wait));
|
|
111
|
+
if (shouldUseInk) {
|
|
112
|
+
let exitCode = 0;
|
|
113
|
+
const { waitUntilExit } = render(React.createElement(TestRunApp, {
|
|
114
|
+
title: "ttt test",
|
|
115
|
+
wait: Boolean(opts.wait),
|
|
116
|
+
onExitCode: (code) => {
|
|
117
|
+
exitCode = code;
|
|
118
|
+
},
|
|
119
|
+
run: async ({ update, signal }) => {
|
|
120
|
+
if (signal.aborted) {
|
|
121
|
+
throw new Error("Cancelled");
|
|
122
|
+
}
|
|
123
|
+
update({ phase: "auth", message: "Checking authentication…" });
|
|
124
|
+
if (!getAuthToken()) {
|
|
125
|
+
throw new Error("Not logged in. Please run 'ttt login' first.");
|
|
126
|
+
}
|
|
127
|
+
let tunnelLocal = null;
|
|
128
|
+
if (url && !opts.noTunnel) {
|
|
129
|
+
update({ phase: "tunnel", message: `Creating tunnel for ${url}…` });
|
|
130
|
+
try {
|
|
131
|
+
tunnelLocal = await createTunnelIfNeeded(url);
|
|
132
|
+
if (tunnelLocal) {
|
|
133
|
+
update({
|
|
134
|
+
tunnelUrl: tunnelLocal.publicUrl,
|
|
135
|
+
message: `Tunnel active at ${tunnelLocal.publicUrl}`,
|
|
136
|
+
});
|
|
137
|
+
}
|
|
138
|
+
else {
|
|
139
|
+
update({ message: "No tunnel needed" });
|
|
140
|
+
}
|
|
141
|
+
}
|
|
142
|
+
catch (err) {
|
|
143
|
+
update({
|
|
144
|
+
phase: "tunnel",
|
|
145
|
+
message: `Tunnel creation failed: ${err?.message || String(err)}`,
|
|
146
|
+
});
|
|
147
|
+
}
|
|
148
|
+
}
|
|
149
|
+
update({ phase: "start", message: "Starting test on backend…" });
|
|
150
|
+
const testMetadata = {};
|
|
151
|
+
if (tunnelLocal) {
|
|
152
|
+
testMetadata.tunnel = {
|
|
153
|
+
original_url: url,
|
|
154
|
+
public_url: tunnelLocal.publicUrl,
|
|
155
|
+
provider: "ngrok",
|
|
156
|
+
};
|
|
157
|
+
}
|
|
158
|
+
if (docsUrl) {
|
|
159
|
+
testMetadata.docs_url = docsUrl;
|
|
160
|
+
}
|
|
161
|
+
const payload = {
|
|
162
|
+
url: tunnelLocal ? tunnelLocal.publicUrl : url,
|
|
163
|
+
user_prompt: prompt,
|
|
164
|
+
test_type: testingMode,
|
|
165
|
+
project_id: projectId,
|
|
166
|
+
team_id: teamId,
|
|
167
|
+
test_metadata: testMetadata,
|
|
168
|
+
};
|
|
169
|
+
const client = createAgentHttpClient({});
|
|
170
|
+
const response = await client.startTest(payload, {
|
|
171
|
+
execute_tests: opts.execute,
|
|
172
|
+
execution_mode: executionMode,
|
|
173
|
+
});
|
|
174
|
+
update({
|
|
175
|
+
testId: response.test_id,
|
|
176
|
+
phase: "connect",
|
|
177
|
+
message: "Connecting to stream…",
|
|
178
|
+
});
|
|
179
|
+
const socket = connectAndSubscribe(response.test_id);
|
|
180
|
+
const cleanup = async () => {
|
|
181
|
+
try {
|
|
182
|
+
socket.disconnect();
|
|
183
|
+
}
|
|
184
|
+
catch { }
|
|
185
|
+
try {
|
|
186
|
+
if (tunnelLocal) {
|
|
187
|
+
await tunnelLocal.stop();
|
|
188
|
+
}
|
|
189
|
+
}
|
|
190
|
+
catch { }
|
|
191
|
+
};
|
|
192
|
+
return { testId: response.test_id, socket, cleanup };
|
|
193
|
+
},
|
|
194
|
+
}));
|
|
195
|
+
await waitUntilExit();
|
|
196
|
+
if (exitCode)
|
|
197
|
+
process.exitCode = exitCode;
|
|
198
|
+
return;
|
|
199
|
+
}
|
|
103
200
|
// 1. Setup Tunnel if needed
|
|
104
201
|
// Only if URL is provided and is local
|
|
105
202
|
if (url && !opts.noTunnel) {
|