@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.
Files changed (118) hide show
  1. package/dist/package.json +8 -5
  2. package/dist/src/commands/chat/ChatApp.js +30 -42
  3. package/dist/src/commands/chat/ChatApp.js.map +1 -1
  4. package/dist/src/commands/chat/components/Banner.js +1 -1
  5. package/dist/src/commands/chat/components/ChatInput.js +39 -17
  6. package/dist/src/commands/chat/components/ChatInput.js.map +1 -1
  7. package/dist/src/commands/chat/components/MessageBubble.js +2 -1
  8. package/dist/src/commands/chat/components/MessageBubble.js.map +1 -1
  9. package/dist/src/commands/chat-ink.js +118 -9
  10. package/dist/src/commands/chat-ink.js.map +1 -1
  11. package/dist/src/commands/start-test.js +61 -0
  12. package/dist/src/commands/start-test.js.map +1 -1
  13. package/dist/src/commands/stream/StreamApp.js +127 -0
  14. package/dist/src/commands/stream/StreamApp.js.map +1 -0
  15. package/dist/src/commands/stream.js +43 -8
  16. package/dist/src/commands/stream.js.map +1 -1
  17. package/dist/src/commands/test/TestRunApp.js +183 -0
  18. package/dist/src/commands/test/TestRunApp.js.map +1 -0
  19. package/dist/src/commands/test.js +97 -0
  20. package/dist/src/commands/test.js.map +1 -1
  21. package/dist/src/lib/agent-orchestrator.js +5 -0
  22. package/dist/src/lib/agent-orchestrator.js.map +1 -1
  23. package/dist/src/lib/local-tools/ui/playwright-mcp.js +1 -1
  24. package/dist/src/lib/tui/ink/theme.js +21 -0
  25. package/dist/src/lib/tui/ink/theme.js.map +1 -0
  26. package/package.json +8 -5
  27. package/dist/src/commands/ask/AskApp.js +0 -121
  28. package/dist/src/commands/ask/AskApp.js.map +0 -1
  29. package/dist/src/commands/ask/components/AssistantResponse.js +0 -31
  30. package/dist/src/commands/ask/components/AssistantResponse.js.map +0 -1
  31. package/dist/src/commands/ask/components/Banner.js +0 -15
  32. package/dist/src/commands/ask/components/Banner.js.map +0 -1
  33. package/dist/src/commands/ask/components/ChatInput.js +0 -93
  34. package/dist/src/commands/ask/components/ChatInput.js.map +0 -1
  35. package/dist/src/commands/ask/components/Divider.js +0 -17
  36. package/dist/src/commands/ask/components/Divider.js.map +0 -1
  37. package/dist/src/commands/ask/components/IntroTips.js +0 -19
  38. package/dist/src/commands/ask/components/IntroTips.js.map +0 -1
  39. package/dist/src/commands/ask/components/MessageBubble.js +0 -47
  40. package/dist/src/commands/ask/components/MessageBubble.js.map +0 -1
  41. package/dist/src/commands/ask/components/SessionInfo.js +0 -20
  42. package/dist/src/commands/ask/components/SessionInfo.js.map +0 -1
  43. package/dist/src/commands/ask/components/StatusIndicator.js +0 -67
  44. package/dist/src/commands/ask/components/StatusIndicator.js.map +0 -1
  45. package/dist/src/commands/ask-ink.js +0 -380
  46. package/dist/src/commands/ask-ink.js.map +0 -1
  47. package/dist/src/commands/ask.js +0 -991
  48. package/dist/src/commands/ask.js.map +0 -1
  49. package/dist/src/commands/chat/components/Divider.js +0 -7
  50. package/dist/src/commands/chat/components/Divider.js.map +0 -1
  51. package/dist/src/commands/chat/components/SessionInfo.js +0 -11
  52. package/dist/src/commands/chat/components/SessionInfo.js.map +0 -1
  53. package/dist/src/commands/chat.js +0 -82
  54. package/dist/src/commands/chat.js.map +0 -1
  55. package/dist/src/lib/legacy-chat-runner.js +0 -37
  56. package/dist/src/lib/legacy-chat-runner.js.map +0 -1
  57. package/dist/src/lib/local-tools/ui/click-element.js +0 -105
  58. package/dist/src/lib/local-tools/ui/click-element.js.map +0 -1
  59. package/dist/src/lib/local-tools/ui/dom-rag.js +0 -201
  60. package/dist/src/lib/local-tools/ui/dom-rag.js.map +0 -1
  61. package/dist/src/lib/local-tools/ui/find-element.js +0 -31
  62. package/dist/src/lib/local-tools/ui/find-element.js.map +0 -1
  63. package/dist/src/lib/local-tools/ui/hover-element.js +0 -94
  64. package/dist/src/lib/local-tools/ui/hover-element.js.map +0 -1
  65. package/dist/src/lib/local-tools/ui/manage-tab.js +0 -65
  66. package/dist/src/lib/local-tools/ui/manage-tab.js.map +0 -1
  67. package/dist/src/lib/local-tools/ui/navigate.js +0 -35
  68. package/dist/src/lib/local-tools/ui/navigate.js.map +0 -1
  69. package/dist/src/lib/local-tools/ui/page-discovery.js +0 -32
  70. package/dist/src/lib/local-tools/ui/page-discovery.js.map +0 -1
  71. package/dist/src/lib/local-tools/ui/screenshot.js +0 -19
  72. package/dist/src/lib/local-tools/ui/screenshot.js.map +0 -1
  73. package/dist/src/lib/local-tools/ui/search-interactive-elements.js +0 -18
  74. package/dist/src/lib/local-tools/ui/search-interactive-elements.js.map +0 -1
  75. package/dist/src/lib/local-tools/ui/selector-resolver.js +0 -153
  76. package/dist/src/lib/local-tools/ui/selector-resolver.js.map +0 -1
  77. package/dist/src/lib/local-tools/ui/type-text.js +0 -40
  78. package/dist/src/lib/local-tools/ui/type-text.js.map +0 -1
  79. package/dist/src/lib/tui/components/AskIntro.js +0 -6
  80. package/dist/src/lib/tui/components/AskIntro.js.map +0 -1
  81. package/dist/src/lib/tui/components/Banner.js +0 -15
  82. package/dist/src/lib/tui/components/Banner.js.map +0 -1
  83. package/dist/src/lib/tui/components/Divider.js +0 -17
  84. package/dist/src/lib/tui/components/Divider.js.map +0 -1
  85. package/dist/src/lib/tui/components/EventLine.js +0 -110
  86. package/dist/src/lib/tui/components/EventLine.js.map +0 -1
  87. package/dist/src/lib/tui/components/Header.js +0 -15
  88. package/dist/src/lib/tui/components/Header.js.map +0 -1
  89. package/dist/src/lib/tui/components/InputBox.js +0 -9
  90. package/dist/src/lib/tui/components/InputBox.js.map +0 -1
  91. package/dist/src/lib/tui/components/Mapping.js +0 -8
  92. package/dist/src/lib/tui/components/Mapping.js.map +0 -1
  93. package/dist/src/lib/tui/components/ProjectList.js +0 -6
  94. package/dist/src/lib/tui/components/ProjectList.js.map +0 -1
  95. package/dist/src/lib/tui/components/Spinner.js +0 -20
  96. package/dist/src/lib/tui/components/Spinner.js.map +0 -1
  97. package/dist/src/lib/tui/components/StatusBanner.js +0 -12
  98. package/dist/src/lib/tui/components/StatusBanner.js.map +0 -1
  99. package/dist/src/lib/tui/components/StatusBar.js +0 -11
  100. package/dist/src/lib/tui/components/StatusBar.js.map +0 -1
  101. package/dist/src/lib/tui/components/UserBubble.js +0 -6
  102. package/dist/src/lib/tui/components/UserBubble.js.map +0 -1
  103. package/dist/src/lib/tui/components/index.js +0 -16
  104. package/dist/src/lib/tui/components/index.js.map +0 -1
  105. package/dist/src/lib/tui/ink-print.js +0 -41
  106. package/dist/src/lib/tui/ink-print.js.map +0 -1
  107. package/dist/src/test-agent-flow.js +0 -148
  108. package/dist/src/test-agent-flow.js.map +0 -1
  109. package/dist/src/test-browser-session.js +0 -152
  110. package/dist/src/test-browser-session.js.map +0 -1
  111. package/dist/src/test-browser-snapshot.js +0 -187
  112. package/dist/src/test-browser-snapshot.js.map +0 -1
  113. package/dist/src/test-snapshot-detailed.js +0 -219
  114. package/dist/src/test-snapshot-detailed.js.map +0 -1
  115. package/dist/src/test-snapshot-simple.js +0 -85
  116. package/dist/src/test-snapshot-simple.js.map +0 -1
  117. package/dist/src/test-snapshot-tabs.js +0 -110
  118. 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":"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"}
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
- const socket = connectAndSubscribe(testId);
9
- registerUnifiedEventLogging(socket, (line) => console.log(line));
10
- socket.on("session_completed", async () => {
11
- await socket.disconnect();
12
- });
13
- socket.on("session_error", async () => {
14
- await socket.disconnect();
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;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
+ {"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":"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"}
@@ -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) {