rax-flow 0.1.5 → 0.1.7
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/LICENSE +21 -0
- package/dist/bin.js +27 -3
- package/dist/bin.js.map +1 -1
- package/dist/hub/__tests__/commands.test.d.ts.map +1 -0
- package/dist/hub/__tests__/commands.test.js +72 -0
- package/dist/hub/__tests__/commands.test.js.map +1 -0
- package/dist/hub/__tests__/history.test.d.ts.map +1 -0
- package/dist/hub/__tests__/history.test.js +119 -0
- package/dist/hub/__tests__/history.test.js.map +1 -0
- package/dist/hub/__tests__/parser.test.d.ts.map +1 -0
- package/dist/hub/__tests__/parser.test.js +92 -0
- package/dist/hub/__tests__/parser.test.js.map +1 -0
- package/dist/hub/chat/ChatApp.d.ts +2 -0
- package/dist/hub/chat/ChatApp.d.ts.map +1 -0
- package/dist/hub/chat/ChatApp.js +146 -0
- package/dist/hub/chat/ChatApp.js.map +1 -0
- package/dist/hub/chat/components/ChatInput.d.ts +9 -0
- package/dist/hub/chat/components/ChatInput.d.ts.map +1 -0
- package/dist/hub/chat/components/ChatInput.js +19 -0
- package/dist/hub/chat/components/ChatInput.js.map +1 -0
- package/dist/hub/chat/components/MessageList.d.ts +7 -0
- package/dist/hub/chat/components/MessageList.d.ts.map +1 -0
- package/dist/hub/chat/components/MessageList.js +6 -0
- package/dist/hub/chat/components/MessageList.js.map +1 -0
- package/dist/hub/chat/context.d.ts.map +1 -0
- package/dist/hub/chat/context.js +42 -0
- package/dist/hub/chat/context.js.map +1 -0
- package/dist/hub/chat/hooks/useChatHistory.d.ts +7 -0
- package/dist/hub/chat/hooks/useChatHistory.d.ts.map +1 -0
- package/dist/hub/chat/hooks/useChatHistory.js +31 -0
- package/dist/hub/chat/hooks/useChatHistory.js.map +1 -0
- package/dist/hub/chat/index.d.ts.map +1 -0
- package/dist/hub/chat/index.js +7 -0
- package/dist/hub/chat/index.js.map +1 -0
- package/dist/hub/chat/intent-parser.d.ts.map +1 -0
- package/dist/hub/chat/intent-parser.js +48 -0
- package/dist/hub/chat/intent-parser.js.map +1 -0
- package/dist/hub/chat/types.d.ts.map +1 -0
- package/dist/hub/chat/types.js +2 -0
- package/dist/hub/chat/types.js.map +1 -0
- package/dist/hub/commands/agents.d.ts.map +1 -0
- package/dist/hub/commands/agents.js +36 -0
- package/dist/hub/commands/agents.js.map +1 -0
- package/dist/hub/commands/index.d.ts.map +1 -0
- package/dist/hub/commands/index.js +120 -0
- package/dist/hub/commands/index.js.map +1 -0
- package/dist/hub/commands/logs.d.ts.map +1 -0
- package/dist/hub/commands/logs.js +53 -0
- package/dist/hub/commands/logs.js.map +1 -0
- package/dist/hub/commands/memory.d.ts.map +1 -0
- package/dist/hub/commands/memory.js +40 -0
- package/dist/hub/commands/memory.js.map +1 -0
- package/dist/hub/commands/metrics.d.ts.map +1 -0
- package/dist/hub/commands/metrics.js +35 -0
- package/dist/hub/commands/metrics.js.map +1 -0
- package/dist/hub/commands/providers.d.ts.map +1 -0
- package/dist/hub/commands/providers.js +26 -0
- package/dist/hub/commands/providers.js.map +1 -0
- package/dist/hub/commands/run.d.ts.map +1 -0
- package/dist/hub/commands/run.js +31 -0
- package/dist/hub/commands/run.js.map +1 -0
- package/dist/hub/commands/status.d.ts.map +1 -0
- package/dist/hub/commands/status.js +61 -0
- package/dist/hub/commands/status.js.map +1 -0
- package/dist/hub/commands/workflows.d.ts.map +1 -0
- package/dist/hub/commands/workflows.js +45 -0
- package/dist/hub/commands/workflows.js.map +1 -0
- package/dist/hub/config-loader.d.ts.map +1 -0
- package/dist/hub/config-loader.js +27 -0
- package/dist/hub/config-loader.js.map +1 -0
- package/dist/hub/event-listener.d.ts.map +1 -0
- package/dist/hub/event-listener.js +8 -0
- package/dist/hub/event-listener.js.map +1 -0
- package/dist/hub/history.d.ts.map +1 -0
- package/dist/hub/history.js +59 -0
- package/dist/hub/history.js.map +1 -0
- package/dist/hub/index.d.ts.map +1 -0
- package/dist/hub/index.js +102 -0
- package/dist/hub/index.js.map +1 -0
- package/dist/hub/parser.d.ts.map +1 -0
- package/dist/hub/parser.js +98 -0
- package/dist/hub/parser.js.map +1 -0
- package/dist/hub/styles/borders.d.ts.map +1 -0
- package/dist/hub/styles/borders.js +64 -0
- package/dist/hub/styles/borders.js.map +1 -0
- package/dist/hub/styles/colors.d.ts.map +1 -0
- package/dist/hub/styles/colors.js +115 -0
- package/dist/hub/styles/colors.js.map +1 -0
- package/dist/hub/styles/typography.d.ts.map +1 -0
- package/dist/hub/styles/typography.js +60 -0
- package/dist/hub/styles/typography.js.map +1 -0
- package/dist/hub/tui/App.d.ts +2 -0
- package/dist/hub/tui/App.d.ts.map +1 -0
- package/dist/hub/tui/App.js +53 -0
- package/dist/hub/tui/App.js.map +1 -0
- package/dist/hub/tui/components/AgentQueue.d.ts +6 -0
- package/dist/hub/tui/components/AgentQueue.d.ts.map +1 -0
- package/dist/hub/tui/components/AgentQueue.js +20 -0
- package/dist/hub/tui/components/AgentQueue.js.map +1 -0
- package/dist/hub/tui/components/DAGPanel.d.ts +16 -0
- package/dist/hub/tui/components/DAGPanel.d.ts.map +1 -0
- package/dist/hub/tui/components/DAGPanel.js +51 -0
- package/dist/hub/tui/components/DAGPanel.js.map +1 -0
- package/dist/hub/tui/components/Header.d.ts +7 -0
- package/dist/hub/tui/components/Header.d.ts.map +1 -0
- package/dist/hub/tui/components/Header.js +17 -0
- package/dist/hub/tui/components/Header.js.map +1 -0
- package/dist/hub/tui/components/LogsPanel.d.ts +6 -0
- package/dist/hub/tui/components/LogsPanel.d.ts.map +1 -0
- package/dist/hub/tui/components/LogsPanel.js +26 -0
- package/dist/hub/tui/components/LogsPanel.js.map +1 -0
- package/dist/hub/tui/components/StatusBar.d.ts +8 -0
- package/dist/hub/tui/components/StatusBar.d.ts.map +1 -0
- package/dist/hub/tui/components/StatusBar.js +7 -0
- package/dist/hub/tui/components/StatusBar.js.map +1 -0
- package/dist/hub/tui/hooks/useEvents.d.ts +2 -0
- package/dist/hub/tui/hooks/useEvents.d.ts.map +1 -0
- package/dist/hub/tui/hooks/useEvents.js +13 -0
- package/dist/hub/tui/hooks/useEvents.js.map +1 -0
- package/dist/hub/tui/index.d.ts.map +1 -0
- package/dist/hub/tui/index.js +7 -0
- package/dist/hub/tui/index.js.map +1 -0
- package/dist/hub/tui/types.d.ts.map +1 -0
- package/dist/hub/tui/types.js +2 -0
- package/dist/hub/tui/types.js.map +1 -0
- package/dist/hub/types.d.ts.map +1 -0
- package/dist/hub/types.js +2 -0
- package/dist/hub/types.js.map +1 -0
- package/dist/tui/App.d.ts +2 -0
- package/dist/tui/App.d.ts.map +1 -0
- package/dist/tui/App.js +31 -0
- package/dist/tui/App.js.map +1 -0
- package/dist/tui/components/ChatPanel.d.ts +14 -0
- package/dist/tui/components/ChatPanel.d.ts.map +1 -0
- package/dist/tui/components/ChatPanel.js +30 -0
- package/dist/tui/components/ChatPanel.js.map +1 -0
- package/dist/tui/components/Header.d.ts +9 -0
- package/dist/tui/components/Header.d.ts.map +1 -0
- package/dist/tui/components/Header.js +17 -0
- package/dist/tui/components/Header.js.map +1 -0
- package/dist/tui/components/InputBar.d.ts +8 -0
- package/dist/tui/components/InputBar.d.ts.map +1 -0
- package/dist/tui/components/InputBar.js +16 -0
- package/dist/tui/components/InputBar.js.map +1 -0
- package/dist/tui/components/StatusPanel.d.ts +20 -0
- package/dist/tui/components/StatusPanel.d.ts.map +1 -0
- package/dist/tui/components/StatusPanel.js +20 -0
- package/dist/tui/components/StatusPanel.js.map +1 -0
- package/dist/tui/hooks/useAppState.d.ts +38 -0
- package/dist/tui/hooks/useAppState.d.ts.map +1 -0
- package/dist/tui/hooks/useAppState.js +127 -0
- package/dist/tui/hooks/useAppState.js.map +1 -0
- package/dist/tui/index.d.ts +3 -0
- package/dist/tui/index.d.ts.map +1 -0
- package/dist/tui/index.js +8 -0
- package/dist/tui/index.js.map +1 -0
- package/package.json +17 -11
- package/src/bin.ts +29 -3
- package/src/hub/__tests__/commands.test.ts +84 -0
- package/src/hub/__tests__/history.test.ts +137 -0
- package/src/hub/__tests__/parser.test.ts +105 -0
- package/src/hub/commands/agents.ts +53 -0
- package/src/hub/commands/index.ts +137 -0
- package/src/hub/commands/logs.ts +70 -0
- package/src/hub/commands/memory.ts +47 -0
- package/src/hub/commands/metrics.ts +49 -0
- package/src/hub/commands/providers.ts +39 -0
- package/src/hub/commands/run.ts +37 -0
- package/src/hub/commands/status.ts +69 -0
- package/src/hub/commands/workflows.ts +64 -0
- package/src/hub/config-loader.ts +37 -0
- package/src/hub/event-listener.ts +13 -0
- package/src/hub/history.ts +66 -0
- package/src/hub/index.ts +120 -0
- package/src/hub/parser.ts +107 -0
- package/src/hub/styles/borders.ts +74 -0
- package/src/hub/styles/colors.ts +129 -0
- package/src/hub/styles/typography.ts +68 -0
- package/src/hub/types.ts +31 -0
- package/src/tui/App.tsx +61 -0
- package/src/tui/components/ChatPanel.tsx +79 -0
- package/src/tui/components/Header.tsx +53 -0
- package/src/tui/components/InputBar.tsx +55 -0
- package/src/tui/components/StatusPanel.tsx +85 -0
- package/src/tui/hooks/useAppState.ts +168 -0
- package/src/tui/index.ts +9 -0
- package/tsconfig.json +19 -3
- package/tsconfig.tsbuildinfo +1 -0
- package/dist/benchmark.d.ts +0 -10
- package/dist/bin.d.ts +0 -3
- package/dist/bootstrap.d.ts +0 -8
- package/dist/bridge-adapter-templates.d.ts +0 -4
- package/dist/bridge-test.d.ts +0 -7
- package/dist/dashboard.d.ts +0 -4
- package/dist/doctor.d.ts +0 -6
- package/dist/evolve.d.ts +0 -7
- package/dist/host-init-templates.d.ts +0 -16
- package/dist/index.d.ts +0 -11
- package/dist/init-host.d.ts +0 -10
- package/dist/install.d.ts +0 -8
- package/dist/run.d.ts +0 -16
- package/dist/styles.d.ts +0 -12
- package/dist/vendor-manifests.d.ts +0 -22
- package/package/dashboard/index.html +0 -420
- package/package/package.json +0 -28
- package/package/src/benchmark.ts +0 -156
- package/package/src/bin.ts +0 -127
- package/package/src/bootstrap.ts +0 -36
- package/package/src/bridge-adapter-templates.ts +0 -181
- package/package/src/bridge-test.ts +0 -107
- package/package/src/dashboard.ts +0 -51
- package/package/src/doctor.ts +0 -92
- package/package/src/evolve.ts +0 -74
- package/package/src/host-init-templates.ts +0 -134
- package/package/src/index.ts +0 -10
- package/package/src/init-host.ts +0 -285
- package/package/src/install.ts +0 -118
- package/package/src/run.ts +0 -317
- package/package/src/styles.ts +0 -12
- package/package/src/vendor-manifests.ts +0 -113
- package/package/src/ws-relay.ts +0 -156
- package/package/tsconfig.json +0 -12
- package/rax-flow-0.1.1.tgz +0 -0
|
@@ -0,0 +1,137 @@
|
|
|
1
|
+
import { describe, it, expect, beforeEach, afterEach } from "vitest";
|
|
2
|
+
import { CommandHistory } from "../history.js";
|
|
3
|
+
import * as fs from "fs/promises";
|
|
4
|
+
import * as path from "path";
|
|
5
|
+
import { tmpdir } from "os";
|
|
6
|
+
|
|
7
|
+
describe("CommandHistory", () => {
|
|
8
|
+
let tempDir: string;
|
|
9
|
+
let history: CommandHistory;
|
|
10
|
+
|
|
11
|
+
beforeEach(async () => {
|
|
12
|
+
// Create a temporary directory for testing
|
|
13
|
+
tempDir = path.join(tmpdir(), `rax-flow-test-${Date.now()}`);
|
|
14
|
+
await fs.mkdir(tempDir, { recursive: true });
|
|
15
|
+
history = new CommandHistory(tempDir);
|
|
16
|
+
});
|
|
17
|
+
|
|
18
|
+
afterEach(async () => {
|
|
19
|
+
// Clean up temporary directory
|
|
20
|
+
try {
|
|
21
|
+
await fs.rm(tempDir, { recursive: true });
|
|
22
|
+
} catch {
|
|
23
|
+
// Ignore cleanup errors
|
|
24
|
+
}
|
|
25
|
+
});
|
|
26
|
+
|
|
27
|
+
describe("add", () => {
|
|
28
|
+
it("should add a command to history", () => {
|
|
29
|
+
history.add("status");
|
|
30
|
+
expect(history.getAll()).toContain("status");
|
|
31
|
+
});
|
|
32
|
+
|
|
33
|
+
it("should not add duplicate consecutive commands", () => {
|
|
34
|
+
history.add("status");
|
|
35
|
+
history.add("status");
|
|
36
|
+
expect(history.getAll().filter(c => c === "status").length).toBe(1);
|
|
37
|
+
});
|
|
38
|
+
|
|
39
|
+
it("should allow same command if not consecutive", () => {
|
|
40
|
+
history.add("status");
|
|
41
|
+
history.add("agents");
|
|
42
|
+
history.add("status");
|
|
43
|
+
expect(history.getAll()).toEqual(["status", "agents", "status"]);
|
|
44
|
+
});
|
|
45
|
+
});
|
|
46
|
+
|
|
47
|
+
describe("up/down navigation", () => {
|
|
48
|
+
it("should return undefined when navigating up on empty history", () => {
|
|
49
|
+
expect(history.up()).toBeUndefined();
|
|
50
|
+
});
|
|
51
|
+
|
|
52
|
+
it("should return commands when navigating up", () => {
|
|
53
|
+
history.add("status");
|
|
54
|
+
history.add("agents");
|
|
55
|
+
history.add("run test");
|
|
56
|
+
|
|
57
|
+
expect(history.up()).toBe("run test");
|
|
58
|
+
expect(history.up()).toBe("agents");
|
|
59
|
+
expect(history.up()).toBe("status");
|
|
60
|
+
});
|
|
61
|
+
|
|
62
|
+
it("should stop at the beginning when navigating up", () => {
|
|
63
|
+
history.add("status");
|
|
64
|
+
history.up();
|
|
65
|
+
expect(history.up()).toBeUndefined();
|
|
66
|
+
});
|
|
67
|
+
|
|
68
|
+
it("should navigate down after navigating up", () => {
|
|
69
|
+
history.add("status");
|
|
70
|
+
history.add("agents");
|
|
71
|
+
history.up();
|
|
72
|
+
history.up();
|
|
73
|
+
expect(history.down()).toBe("agents");
|
|
74
|
+
});
|
|
75
|
+
|
|
76
|
+
it("should return undefined when navigating down past the end", () => {
|
|
77
|
+
history.add("status");
|
|
78
|
+
history.down();
|
|
79
|
+
expect(history.down()).toBeUndefined();
|
|
80
|
+
});
|
|
81
|
+
});
|
|
82
|
+
|
|
83
|
+
describe("save and load", () => {
|
|
84
|
+
it("should save history to file", async () => {
|
|
85
|
+
history.add("status");
|
|
86
|
+
history.add("agents");
|
|
87
|
+
await history.save();
|
|
88
|
+
|
|
89
|
+
const historyPath = path.join(tempDir, ".rax", "history.json");
|
|
90
|
+
const content = await fs.readFile(historyPath, "utf-8");
|
|
91
|
+
const saved = JSON.parse(content);
|
|
92
|
+
expect(saved).toEqual(["status", "agents"]);
|
|
93
|
+
});
|
|
94
|
+
|
|
95
|
+
it("should load history from file", async () => {
|
|
96
|
+
// Setup: Create history file
|
|
97
|
+
const raxDir = path.join(tempDir, ".rax");
|
|
98
|
+
await fs.mkdir(raxDir, { recursive: true });
|
|
99
|
+
const historyPath = path.join(raxDir, "history.json");
|
|
100
|
+
await fs.writeFile(historyPath, JSON.stringify(["status", "agents"]));
|
|
101
|
+
|
|
102
|
+
// Load and verify
|
|
103
|
+
const newHistory = new CommandHistory(tempDir);
|
|
104
|
+
await newHistory.load();
|
|
105
|
+
expect(newHistory.getAll()).toEqual(["status", "agents"]);
|
|
106
|
+
});
|
|
107
|
+
|
|
108
|
+
it("should handle missing history file gracefully", async () => {
|
|
109
|
+
await history.load();
|
|
110
|
+
expect(history.getAll()).toEqual([]);
|
|
111
|
+
});
|
|
112
|
+
|
|
113
|
+
it("should limit history to last 1000 commands", async () => {
|
|
114
|
+
// Add 1005 commands
|
|
115
|
+
for (let i = 0; i < 1005; i++) {
|
|
116
|
+
history.add(`command-${i}`);
|
|
117
|
+
}
|
|
118
|
+
await history.save();
|
|
119
|
+
|
|
120
|
+
const historyPath = path.join(tempDir, ".rax", "history.json");
|
|
121
|
+
const content = await fs.readFile(historyPath, "utf-8");
|
|
122
|
+
const saved = JSON.parse(content);
|
|
123
|
+
expect(saved.length).toBe(1000);
|
|
124
|
+
expect(saved[0]).toBe("command-5"); // First 5 should be removed
|
|
125
|
+
expect(saved[saved.length - 1]).toBe("command-1004");
|
|
126
|
+
});
|
|
127
|
+
});
|
|
128
|
+
|
|
129
|
+
describe("getAll", () => {
|
|
130
|
+
it("should return copy of history array", () => {
|
|
131
|
+
history.add("status");
|
|
132
|
+
const all = history.getAll();
|
|
133
|
+
all.push("modified");
|
|
134
|
+
expect(history.getAll()).not.toContain("modified");
|
|
135
|
+
});
|
|
136
|
+
});
|
|
137
|
+
});
|
|
@@ -0,0 +1,105 @@
|
|
|
1
|
+
import { describe, it, expect } from "vitest";
|
|
2
|
+
import { parseCommand, getCompletions } from "../parser.js";
|
|
3
|
+
import { HubContext } from "../types.js";
|
|
4
|
+
|
|
5
|
+
describe("parseCommand", () => {
|
|
6
|
+
it("should parse a simple command", () => {
|
|
7
|
+
const result = parseCommand("status");
|
|
8
|
+
expect(result.name).toBe("status");
|
|
9
|
+
expect(result.args).toEqual([]);
|
|
10
|
+
expect(result.flags).toEqual({});
|
|
11
|
+
});
|
|
12
|
+
|
|
13
|
+
it("should parse a command with arguments", () => {
|
|
14
|
+
const result = parseCommand("run Create an API");
|
|
15
|
+
expect(result.name).toBe("run");
|
|
16
|
+
expect(result.args).toEqual(["Create", "an", "API"]);
|
|
17
|
+
expect(result.flags).toEqual({});
|
|
18
|
+
});
|
|
19
|
+
|
|
20
|
+
it("should parse a command with long flags", () => {
|
|
21
|
+
const result = parseCommand("run --workflow test.json --stream Create API");
|
|
22
|
+
expect(result.name).toBe("run");
|
|
23
|
+
expect(result.args).toEqual(["Create", "API"]);
|
|
24
|
+
expect(result.flags).toEqual({
|
|
25
|
+
workflow: "test.json",
|
|
26
|
+
stream: true,
|
|
27
|
+
});
|
|
28
|
+
});
|
|
29
|
+
|
|
30
|
+
it("should parse a command with short flags", () => {
|
|
31
|
+
const result = parseCommand("logs -f --agent CodeGenerator");
|
|
32
|
+
expect(result.name).toBe("logs");
|
|
33
|
+
expect(result.args).toEqual([]);
|
|
34
|
+
expect(result.flags).toEqual({
|
|
35
|
+
f: true,
|
|
36
|
+
agent: "CodeGenerator",
|
|
37
|
+
});
|
|
38
|
+
});
|
|
39
|
+
|
|
40
|
+
it("should parse mixed flags and arguments", () => {
|
|
41
|
+
const result = parseCommand("run --json --max-parallel 6 Create API");
|
|
42
|
+
expect(result.name).toBe("run");
|
|
43
|
+
expect(result.args).toEqual(["Create", "API"]);
|
|
44
|
+
expect(result.flags).toEqual({
|
|
45
|
+
json: true,
|
|
46
|
+
"max-parallel": "6",
|
|
47
|
+
});
|
|
48
|
+
});
|
|
49
|
+
|
|
50
|
+
it("should handle empty input", () => {
|
|
51
|
+
const result = parseCommand("");
|
|
52
|
+
expect(result.name).toBe("");
|
|
53
|
+
expect(result.args).toEqual([]);
|
|
54
|
+
expect(result.flags).toEqual({});
|
|
55
|
+
});
|
|
56
|
+
|
|
57
|
+
it("should handle whitespace-only input", () => {
|
|
58
|
+
const result = parseCommand(" ");
|
|
59
|
+
expect(result.name).toBe("");
|
|
60
|
+
expect(result.args).toEqual([]);
|
|
61
|
+
expect(result.flags).toEqual({});
|
|
62
|
+
});
|
|
63
|
+
});
|
|
64
|
+
|
|
65
|
+
describe("getCompletions", () => {
|
|
66
|
+
it("should return all commands for empty input", () => {
|
|
67
|
+
const completions = getCompletions("");
|
|
68
|
+
expect(completions).toContain("run");
|
|
69
|
+
expect(completions).toContain("status");
|
|
70
|
+
expect(completions).toContain("agents");
|
|
71
|
+
expect(completions).toContain("exit");
|
|
72
|
+
expect(completions.length).toBeGreaterThan(10);
|
|
73
|
+
});
|
|
74
|
+
|
|
75
|
+
it("should filter commands by prefix", () => {
|
|
76
|
+
const completions = getCompletions("sta");
|
|
77
|
+
expect(completions).toContain("status");
|
|
78
|
+
expect(completions.every(c => c.startsWith("sta"))).toBe(true);
|
|
79
|
+
});
|
|
80
|
+
|
|
81
|
+
it("should suggest run flags after 'run' command", () => {
|
|
82
|
+
const completions = getCompletions("run ");
|
|
83
|
+
expect(completions).toContain("--workflow");
|
|
84
|
+
expect(completions).toContain("--max-parallel");
|
|
85
|
+
expect(completions).toContain("--stream");
|
|
86
|
+
expect(completions).toContain("--json");
|
|
87
|
+
});
|
|
88
|
+
|
|
89
|
+
it("should suggest log flags after 'logs' command", () => {
|
|
90
|
+
const completions = getCompletions("logs ");
|
|
91
|
+
expect(completions).toContain("-f");
|
|
92
|
+
expect(completions).toContain("--agent");
|
|
93
|
+
expect(completions).toContain("--level");
|
|
94
|
+
});
|
|
95
|
+
|
|
96
|
+
it("should return empty array for unknown command context", () => {
|
|
97
|
+
const completions = getCompletions("unknowncommand ");
|
|
98
|
+
expect(completions).toEqual([]);
|
|
99
|
+
});
|
|
100
|
+
|
|
101
|
+
it("should handle partial flag completion", () => {
|
|
102
|
+
const completions = getCompletions("run --work");
|
|
103
|
+
expect(completions).toContain("--workflow");
|
|
104
|
+
});
|
|
105
|
+
});
|
|
@@ -0,0 +1,53 @@
|
|
|
1
|
+
import { HubContext } from "../types.js";
|
|
2
|
+
import { drawBox } from "../styles/borders.js";
|
|
3
|
+
import { colorize, formatStatus, providerIndicators } from "../styles/colors.js";
|
|
4
|
+
import { padRight } from "../styles/typography.js";
|
|
5
|
+
|
|
6
|
+
interface AgentInfo {
|
|
7
|
+
name: string;
|
|
8
|
+
role: string;
|
|
9
|
+
status: string;
|
|
10
|
+
provider: string;
|
|
11
|
+
lastExecution: string;
|
|
12
|
+
avgResponseTime: string;
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
export async function listAgents(_args: string[], _flags: Record<string, string | boolean>, _context: HubContext): Promise<void> {
|
|
16
|
+
const width = 75;
|
|
17
|
+
|
|
18
|
+
// Agent definitions from official-agents
|
|
19
|
+
const agents: AgentInfo[] = [
|
|
20
|
+
{ name: "IntentClassifier", role: "Classify user intent", status: "online", provider: "host", lastExecution: "2s ago", avgResponseTime: "45ms" },
|
|
21
|
+
{ name: "Brainstorm", role: "Generate ideas", status: "online", provider: "host", lastExecution: "5m ago", avgResponseTime: "120ms" },
|
|
22
|
+
{ name: "Spec", role: "Write specifications", status: "online", provider: "host", lastExecution: "1m ago", avgResponseTime: "89ms" },
|
|
23
|
+
{ name: "Architecture", role: "Design architecture", status: "online", provider: "host", lastExecution: "3m ago", avgResponseTime: "156ms" },
|
|
24
|
+
{ name: "TaskPlanner", role: "Decompose to DAG", status: "online", provider: "host", lastExecution: "2s ago", avgResponseTime: "34ms" },
|
|
25
|
+
{ name: "CodeGenerator", role: "Generate code", status: "running", provider: "host", lastExecution: "running", avgResponseTime: "234ms" },
|
|
26
|
+
{ name: "Test", role: "Run tests", status: "queued", provider: "host", lastExecution: "--", avgResponseTime: "--" },
|
|
27
|
+
{ name: "Validator", role: "Validate outputs", status: "online", provider: "host", lastExecution: "1m ago", avgResponseTime: "67ms" },
|
|
28
|
+
{ name: "Fix", role: "Patch defects", status: "online", provider: "host", lastExecution: "2m ago", avgResponseTime: "89ms" },
|
|
29
|
+
{ name: "Optimizer", role: "Optimize performance", status: "idle", provider: "anthropic", lastExecution: "1h ago", avgResponseTime: "178ms" },
|
|
30
|
+
{ name: "Documentation", role: "Generate docs", status: "online", provider: "host", lastExecution: "5m ago", avgResponseTime: "145ms" },
|
|
31
|
+
{ name: "Benchmark", role: "Run benchmarks", status: "idle", provider: "host", lastExecution: "2h ago", avgResponseTime: "567ms" },
|
|
32
|
+
];
|
|
33
|
+
|
|
34
|
+
const lines: string[] = [];
|
|
35
|
+
|
|
36
|
+
// Header
|
|
37
|
+
lines.push(` ${colorize("AGENT", "bright")} ${colorize("ROLE", "bright")} ${colorize("STATUS", "bright")} ${colorize("PROVIDER", "bright")} ${colorize("LAST RUN", "bright")}`);
|
|
38
|
+
lines.push(" " + "─".repeat(71));
|
|
39
|
+
|
|
40
|
+
// Agent list
|
|
41
|
+
for (const agent of agents) {
|
|
42
|
+
const indicator = formatStatus(agent.status);
|
|
43
|
+
const providerTag = providerIndicators[agent.provider] || "[?]";
|
|
44
|
+
const statusColor = agent.status === "running" ? "yellow" : agent.status === "queued" ? "gray" : "green";
|
|
45
|
+
|
|
46
|
+
lines.push(` ${padRight(agent.name, 16)} ${padRight(agent.role.slice(0, 17), 18)} ${indicator} ${colorize(agent.status.toUpperCase().padEnd(8), statusColor)} ${providerTag} ${colorize(agent.lastExecution, "gray")}`);
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
lines.push("");
|
|
50
|
+
lines.push(` ${colorize("Legend:", "bright")} ● online ○ idle ▶ running ◐ queued`);
|
|
51
|
+
|
|
52
|
+
console.log(drawBox(lines, width, "AGENTS"));
|
|
53
|
+
}
|
|
@@ -0,0 +1,137 @@
|
|
|
1
|
+
import { CommandDefinition } from "../types.js";
|
|
2
|
+
import { showStatus } from "./status.js";
|
|
3
|
+
import { listAgents } from "./agents.js";
|
|
4
|
+
import { runWorkflow } from "./run.js";
|
|
5
|
+
import { listProviders } from "./providers.js";
|
|
6
|
+
import { listWorkflows } from "./workflows.js";
|
|
7
|
+
import { showLogs } from "./logs.js";
|
|
8
|
+
import { showMetrics } from "./metrics.js";
|
|
9
|
+
import { showMemory } from "./memory.js";
|
|
10
|
+
|
|
11
|
+
const commands = new Map<string, CommandDefinition>();
|
|
12
|
+
|
|
13
|
+
export function registerCommand(def: CommandDefinition): void {
|
|
14
|
+
commands.set(def.name, def);
|
|
15
|
+
if (def.aliases) {
|
|
16
|
+
for (const alias of def.aliases) {
|
|
17
|
+
commands.set(alias, def);
|
|
18
|
+
}
|
|
19
|
+
}
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
export function getCommandRegistry(): Map<string, CommandDefinition> {
|
|
23
|
+
return commands;
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
registerCommand({
|
|
27
|
+
name: "help",
|
|
28
|
+
aliases: ["?"],
|
|
29
|
+
description: "Affiche l'aide",
|
|
30
|
+
handler: async () => {
|
|
31
|
+
console.log(`
|
|
32
|
+
Commandes disponibles:
|
|
33
|
+
• run <prompt> : Exécuter un workflow
|
|
34
|
+
• status : Voir l'état du système
|
|
35
|
+
• agents : Liste des agents
|
|
36
|
+
• providers : État des providers
|
|
37
|
+
• workflows : Blueprints disponibles
|
|
38
|
+
• logs : Logs en temps réel
|
|
39
|
+
• metrics : Métriques de performance
|
|
40
|
+
• memory : Graph QSGM
|
|
41
|
+
• dashboard : Ouvrir dashboard web
|
|
42
|
+
• ui : Mode TUI plein écran
|
|
43
|
+
• chat : Mode conversationnel
|
|
44
|
+
• help : Cette aide
|
|
45
|
+
• exit : Quitter
|
|
46
|
+
`);
|
|
47
|
+
},
|
|
48
|
+
});
|
|
49
|
+
|
|
50
|
+
registerCommand({
|
|
51
|
+
name: "exit",
|
|
52
|
+
aliases: ["quit"],
|
|
53
|
+
description: "Quitte le hub",
|
|
54
|
+
handler: async () => {
|
|
55
|
+
},
|
|
56
|
+
});
|
|
57
|
+
|
|
58
|
+
registerCommand({
|
|
59
|
+
name: "status",
|
|
60
|
+
description: "Affiche le statut du système",
|
|
61
|
+
handler: showStatus,
|
|
62
|
+
});
|
|
63
|
+
|
|
64
|
+
registerCommand({
|
|
65
|
+
name: "agents",
|
|
66
|
+
description: "Liste les agents disponibles",
|
|
67
|
+
handler: listAgents,
|
|
68
|
+
});
|
|
69
|
+
|
|
70
|
+
registerCommand({
|
|
71
|
+
name: "providers",
|
|
72
|
+
description: "Liste les providers configurés",
|
|
73
|
+
handler: listProviders,
|
|
74
|
+
});
|
|
75
|
+
|
|
76
|
+
registerCommand({
|
|
77
|
+
name: "workflows",
|
|
78
|
+
description: "Liste les workflows disponibles",
|
|
79
|
+
handler: listWorkflows,
|
|
80
|
+
});
|
|
81
|
+
|
|
82
|
+
registerCommand({
|
|
83
|
+
name: "logs",
|
|
84
|
+
description: "Affiche les logs en temps réel",
|
|
85
|
+
handler: showLogs,
|
|
86
|
+
});
|
|
87
|
+
|
|
88
|
+
registerCommand({
|
|
89
|
+
name: "metrics",
|
|
90
|
+
description: "Affiche les métriques de performance",
|
|
91
|
+
handler: showMetrics,
|
|
92
|
+
});
|
|
93
|
+
|
|
94
|
+
registerCommand({
|
|
95
|
+
name: "memory",
|
|
96
|
+
description: "Affiche le statut de la mémoire",
|
|
97
|
+
handler: showMemory,
|
|
98
|
+
});
|
|
99
|
+
|
|
100
|
+
registerCommand({
|
|
101
|
+
name: "run",
|
|
102
|
+
description: "Exécute un workflow avec le prompt spécifié",
|
|
103
|
+
handler: runWorkflow,
|
|
104
|
+
});
|
|
105
|
+
|
|
106
|
+
registerCommand({
|
|
107
|
+
name: "ui",
|
|
108
|
+
description: "Lance le mode TUI plein écran",
|
|
109
|
+
handler: async () => {
|
|
110
|
+
console.log("⚠️ Mode TUI non disponible - nécessite l'installation de react et ink");
|
|
111
|
+
console.log(" npm install react ink");
|
|
112
|
+
},
|
|
113
|
+
});
|
|
114
|
+
|
|
115
|
+
registerCommand({
|
|
116
|
+
name: "chat",
|
|
117
|
+
description: "Lance le mode conversationnel",
|
|
118
|
+
handler: async () => {
|
|
119
|
+
console.log("⚠️ Mode Chat non disponible - nécessite l'installation de react et ink");
|
|
120
|
+
console.log(" npm install react ink");
|
|
121
|
+
},
|
|
122
|
+
});
|
|
123
|
+
|
|
124
|
+
const placeholderCommands = [
|
|
125
|
+
"install", "config", "doctor", "bridge-test", "benchmark",
|
|
126
|
+
"evolve", "dashboard"
|
|
127
|
+
];
|
|
128
|
+
|
|
129
|
+
for (const cmd of placeholderCommands) {
|
|
130
|
+
registerCommand({
|
|
131
|
+
name: cmd,
|
|
132
|
+
description: `Commande ${cmd}`,
|
|
133
|
+
handler: async () => {
|
|
134
|
+
console.log(`⚠️ Commande '${cmd}' sera implémentée dans une phase ultérieure`);
|
|
135
|
+
},
|
|
136
|
+
});
|
|
137
|
+
}
|
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
import { HubContext } from "../types.js";
|
|
2
|
+
import { drawBox } from "../styles/borders.js";
|
|
3
|
+
import { colorize, ansiColors } from "../styles/colors.js";
|
|
4
|
+
|
|
5
|
+
interface LogEntry {
|
|
6
|
+
timestamp: string;
|
|
7
|
+
level: "INFO" | "WARN" | "ERROR" | "SUCCESS" | "MUTATE";
|
|
8
|
+
agent: string;
|
|
9
|
+
message: string;
|
|
10
|
+
}
|
|
11
|
+
|
|
12
|
+
export async function showLogs(_args: string[], flags: Record<string, string | boolean>, _context: HubContext): Promise<void> {
|
|
13
|
+
const width = 85;
|
|
14
|
+
const followMode = flags["f"] === true || flags["follow"] === true;
|
|
15
|
+
const agentFilter = typeof flags["agent"] === "string" ? flags["agent"] : null;
|
|
16
|
+
const levelFilter = typeof flags["level"] === "string" ? flags["level"].toUpperCase() : null;
|
|
17
|
+
|
|
18
|
+
// Sample log entries
|
|
19
|
+
const logs: LogEntry[] = [
|
|
20
|
+
{ timestamp: "14:23:07.123", level: "INFO", agent: "IntentClassifier", message: 'detected: "code_generation"' },
|
|
21
|
+
{ timestamp: "14:23:07.456", level: "INFO", agent: "TaskPlanner", message: "decomposed into 5 tasks" },
|
|
22
|
+
{ timestamp: "14:23:08.012", level: "SUCCESS", agent: "CodeGenerator", message: "generating auth module" },
|
|
23
|
+
{ timestamp: "14:23:09.234", level: "SUCCESS", agent: "CodeGenerator", message: "artifacts: 3 files" },
|
|
24
|
+
{ timestamp: "14:23:09.456", level: "INFO", agent: "TestAgent", message: "running test suite" },
|
|
25
|
+
{ timestamp: "14:23:12.789", level: "WARN", agent: "TestAgent", message: "2 tests failed, triggering FixAgent" },
|
|
26
|
+
{ timestamp: "14:23:13.012", level: "INFO", agent: "FixAgent", message: "patching auth/login.ts" },
|
|
27
|
+
{ timestamp: "14:23:14.345", level: "SUCCESS", agent: "FixAgent", message: "patches applied" },
|
|
28
|
+
{ timestamp: "14:23:14.567", level: "MUTATE", agent: "MutationAgent", message: "workflow optimized (+2 validation nodes)" },
|
|
29
|
+
];
|
|
30
|
+
|
|
31
|
+
// Filter logs
|
|
32
|
+
let filteredLogs = logs;
|
|
33
|
+
if (agentFilter) {
|
|
34
|
+
filteredLogs = filteredLogs.filter(l => l.agent.toLowerCase().includes(agentFilter.toLowerCase()));
|
|
35
|
+
}
|
|
36
|
+
if (levelFilter) {
|
|
37
|
+
filteredLogs = filteredLogs.filter(l => l.level === levelFilter);
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
const lines: string[] = [];
|
|
41
|
+
|
|
42
|
+
// Header
|
|
43
|
+
lines.push(` ${colorize("TIME", "bright")} ${colorize("LEVEL", "bright")} ${colorize("AGENT", "bright")} ${colorize("MESSAGE", "bright")}`);
|
|
44
|
+
lines.push(" " + "─".repeat(81));
|
|
45
|
+
|
|
46
|
+
// Log entries
|
|
47
|
+
for (const log of filteredLogs) {
|
|
48
|
+
const levelColor: Record<string, keyof typeof ansiColors> = {
|
|
49
|
+
"INFO": "cyan",
|
|
50
|
+
"WARN": "yellow",
|
|
51
|
+
"ERROR": "red",
|
|
52
|
+
"SUCCESS": "green",
|
|
53
|
+
"MUTATE": "magenta",
|
|
54
|
+
};
|
|
55
|
+
|
|
56
|
+
lines.push(` ${log.timestamp} ${colorize(log.level.padEnd(8), levelColor[log.level] || "white")} ${log.agent.padEnd(18)} ${log.message}`);
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
if (followMode) {
|
|
60
|
+
lines.push("");
|
|
61
|
+
lines.push(colorize(" [Following... Press Ctrl+C to stop]", "gray"));
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
console.log(drawBox(lines, width, "LOGS"));
|
|
65
|
+
|
|
66
|
+
if (followMode) {
|
|
67
|
+
// In a real implementation, this would stream new logs
|
|
68
|
+
console.log(colorize("\n (Follow mode not yet implemented)", "gray"));
|
|
69
|
+
}
|
|
70
|
+
}
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
import { HubContext } from "../types.js";
|
|
2
|
+
import { drawBox } from "../styles/borders.js";
|
|
3
|
+
import { colorize, statusIndicators } from "../styles/colors.js";
|
|
4
|
+
|
|
5
|
+
export async function showMemory(_args: string[], _flags: Record<string, string | boolean>, _context: HubContext): Promise<void> {
|
|
6
|
+
const width = 70;
|
|
7
|
+
|
|
8
|
+
const stats = {
|
|
9
|
+
graphNodes: 247,
|
|
10
|
+
vectorStoreSize: "1.2 MB",
|
|
11
|
+
cacheEntries: 47,
|
|
12
|
+
lastAccess: "2s ago",
|
|
13
|
+
recentMutations: 3,
|
|
14
|
+
cacheHitRate: "94%",
|
|
15
|
+
};
|
|
16
|
+
|
|
17
|
+
const lines: string[] = [];
|
|
18
|
+
|
|
19
|
+
// Stats
|
|
20
|
+
lines.push(` ${colorize("Graph Memory Nodes:", "bright")} ${colorize(stats.graphNodes.toString(), "cyan")} ${statusIndicators.online}`);
|
|
21
|
+
lines.push(` ${colorize("Vector Store Size:", "bright")} ${colorize(stats.vectorStoreSize, "cyan")}`);
|
|
22
|
+
lines.push(` ${colorize("Cache Entries:", "bright")} ${colorize(stats.cacheEntries.toString(), "cyan")}`);
|
|
23
|
+
lines.push(` ${colorize("Last Access:", "bright")} ${colorize(stats.lastAccess, "gray")}`);
|
|
24
|
+
lines.push(` ${colorize("Recent Mutations:", "bright")} ${colorize(`+${stats.recentMutations}`, "yellow")} ${statusIndicators.mutation}`);
|
|
25
|
+
lines.push(` ${colorize("Cache Hit Rate:", "bright")} ${colorize(stats.cacheHitRate, "green")}`);
|
|
26
|
+
|
|
27
|
+
lines.push("");
|
|
28
|
+
lines.push(colorize(" Sample Graph Structure:", "bright"));
|
|
29
|
+
lines.push(" " + "─".repeat(66));
|
|
30
|
+
|
|
31
|
+
// Visual graph representation
|
|
32
|
+
lines.push("");
|
|
33
|
+
lines.push(` [action:code_gen]${statusIndicators.online}`);
|
|
34
|
+
lines.push(` │`);
|
|
35
|
+
lines.push(` ├───▶ [file:auth.ts]${statusIndicators.online}`);
|
|
36
|
+
lines.push(` │ │`);
|
|
37
|
+
lines.push(` │ ├───▶ [task:login]${statusIndicators.online}`);
|
|
38
|
+
lines.push(` │ │`);
|
|
39
|
+
lines.push(` │ └───▶ [test:auth]${statusIndicators.online}`);
|
|
40
|
+
lines.push(` │ │`);
|
|
41
|
+
lines.push(` │ │`);
|
|
42
|
+
lines.push(` └───────────────────────┴───▶ [fix:patch_001]${statusIndicators.mutation}`);
|
|
43
|
+
lines.push("");
|
|
44
|
+
lines.push(colorize(" // QUANTUM_SEMANTIC_GRAPH_MEMORY", "gray"));
|
|
45
|
+
|
|
46
|
+
console.log(drawBox(lines, width, "MEMORY"));
|
|
47
|
+
}
|
|
@@ -0,0 +1,49 @@
|
|
|
1
|
+
import { HubContext } from "../types.js";
|
|
2
|
+
import { drawBox, drawProgressBar } from "../styles/borders.js";
|
|
3
|
+
import { colorize } from "../styles/colors.js";
|
|
4
|
+
import { padRight } from "../styles/typography.js";
|
|
5
|
+
|
|
6
|
+
interface MetricInfo {
|
|
7
|
+
label: string;
|
|
8
|
+
value: string;
|
|
9
|
+
trend?: string;
|
|
10
|
+
}
|
|
11
|
+
|
|
12
|
+
export async function showMetrics(_args: string[], _flags: Record<string, string | boolean>, _context: HubContext): Promise<void> {
|
|
13
|
+
const width = 65;
|
|
14
|
+
|
|
15
|
+
const metrics: MetricInfo[] = [
|
|
16
|
+
{ label: "Sessions", value: "12", trend: "↗" },
|
|
17
|
+
{ label: "Avg Duration", value: "14.2s", trend: "▼ -12%" },
|
|
18
|
+
{ label: "Success Rate", value: "96%", trend: "▲ +3%" },
|
|
19
|
+
{ label: "Cost (USD)", value: "$0.002", trend: "→" },
|
|
20
|
+
{ label: "Cache Savings", value: "67%", trend: "↗" },
|
|
21
|
+
];
|
|
22
|
+
|
|
23
|
+
const topAgents = [
|
|
24
|
+
{ name: "CodeGenerator", usage: 45 },
|
|
25
|
+
{ name: "TestAgent", usage: 32 },
|
|
26
|
+
{ name: "FixAgent", usage: 18 },
|
|
27
|
+
{ name: "IntentClassifier", usage: 5 },
|
|
28
|
+
];
|
|
29
|
+
|
|
30
|
+
const lines: string[] = [];
|
|
31
|
+
|
|
32
|
+
// General metrics
|
|
33
|
+
for (const metric of metrics) {
|
|
34
|
+
const trendColor = metric.trend?.includes("▲") ? "green" : metric.trend?.includes("▼") ? "green" : "gray";
|
|
35
|
+
lines.push(` ${padRight(metric.label, 15)} ${colorize(metric.value.padEnd(12), "cyan")} ${colorize(metric.trend || "", trendColor)}`);
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
lines.push("");
|
|
39
|
+
lines.push(colorize(" Top Agents:", "bright"));
|
|
40
|
+
lines.push(" " + "─".repeat(61));
|
|
41
|
+
|
|
42
|
+
// Top agents with progress bars
|
|
43
|
+
for (const agent of topAgents) {
|
|
44
|
+
const bar = drawProgressBar(agent.usage, 20);
|
|
45
|
+
lines.push(` ${padRight(agent.name, 18)} ${bar} ${colorize(`${agent.usage}%`, "yellow")}`);
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
console.log(drawBox(lines, width, "METRICS"));
|
|
49
|
+
}
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
import { HubContext } from "../types.js";
|
|
2
|
+
import { drawBox } from "../styles/borders.js";
|
|
3
|
+
import { colorize, formatStatus } from "../styles/colors.js";
|
|
4
|
+
import { padRight } from "../styles/typography.js";
|
|
5
|
+
|
|
6
|
+
interface ProviderInfo {
|
|
7
|
+
name: string;
|
|
8
|
+
key: string;
|
|
9
|
+
status: string;
|
|
10
|
+
latency: string;
|
|
11
|
+
lastCall: string;
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
export async function listProviders(_args: string[], _flags: Record<string, string | boolean>, _context: HubContext): Promise<void> {
|
|
15
|
+
const width = 65;
|
|
16
|
+
|
|
17
|
+
const providers: ProviderInfo[] = [
|
|
18
|
+
{ name: "Host-Native", key: "[H]", status: "online", latency: "12ms", lastCall: "2s ago" },
|
|
19
|
+
{ name: "Claude Code", key: "[C]", status: "online", latency: "8ms", lastCall: "5s ago" },
|
|
20
|
+
{ name: "OpenCode", key: "[O]", status: "idle", latency: "--", lastCall: "1h ago" },
|
|
21
|
+
{ name: "Anthropic API", key: "[A]", status: "online", latency: "45ms", lastCall: "1m ago" },
|
|
22
|
+
{ name: "Gemini", key: "[G]", status: "idle", latency: "--", lastCall: "2h ago" },
|
|
23
|
+
{ name: "Mistral", key: "[M]", status: "idle", latency: "--", lastCall: "3h ago" },
|
|
24
|
+
{ name: "Groq", key: "[Q]", status: "idle", latency: "--", lastCall: "5h ago" },
|
|
25
|
+
{ name: "OpenAI", key: "[O]", status: "idle", latency: "--", lastCall: "1d ago" },
|
|
26
|
+
];
|
|
27
|
+
|
|
28
|
+
const lines: string[] = [];
|
|
29
|
+
lines.push(` ${colorize("PROVIDER", "bright")} ${colorize("STATUS", "bright")} ${colorize("LATENCY", "bright")} ${colorize("LAST CALL", "bright")}`);
|
|
30
|
+
lines.push(" " + "─".repeat(61));
|
|
31
|
+
|
|
32
|
+
for (const provider of providers) {
|
|
33
|
+
const indicator = formatStatus(provider.status);
|
|
34
|
+
const statusColor = provider.status === "online" ? "green" : "gray";
|
|
35
|
+
lines.push(` ${provider.key} ${padRight(provider.name, 14)} ${indicator} ${colorize(provider.status.toUpperCase().padEnd(8), statusColor)} ${padRight(provider.latency, 8)} ${colorize(provider.lastCall, "gray")}`);
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
console.log(drawBox(lines, width, "PROVIDERS"));
|
|
39
|
+
}
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
import { HubContext } from "../types.js";
|
|
2
|
+
import { runWorkflow as runWorkflowOriginal } from "../../run.js";
|
|
3
|
+
|
|
4
|
+
export async function runWorkflow(args: string[], flags: Record<string, string | boolean>, context: HubContext): Promise<void> {
|
|
5
|
+
const prompt = args.join(" ");
|
|
6
|
+
|
|
7
|
+
if (!prompt) {
|
|
8
|
+
console.error("❌ Usage: run <prompt>");
|
|
9
|
+
console.log("\nExemples:");
|
|
10
|
+
console.log(' run "Crée une API REST"');
|
|
11
|
+
console.log(' run "Génère des tests unitaires" --workflow custom.json');
|
|
12
|
+
return;
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
console.log(`▶ Lancement du workflow...`);
|
|
16
|
+
console.log(` Prompt: ${prompt}`);
|
|
17
|
+
console.log(` Mode: ${flags["workflow"] || "default"}`);
|
|
18
|
+
console.log("");
|
|
19
|
+
|
|
20
|
+
// Call the original run function
|
|
21
|
+
const code = await runWorkflowOriginal({
|
|
22
|
+
cwd: context.config.project || process.cwd(),
|
|
23
|
+
prompt,
|
|
24
|
+
workflowPath: typeof flags["workflow"] === "string" ? flags["workflow"] : undefined,
|
|
25
|
+
asJson: flags["json"] === true,
|
|
26
|
+
stream: flags["stream"] === true,
|
|
27
|
+
maxParallel: typeof flags["max-parallel"] === "string" ? Number(flags["max-parallel"]) : undefined,
|
|
28
|
+
noCachePersist: flags["no-cache-persist"] === true,
|
|
29
|
+
ui: flags["ui"] === true,
|
|
30
|
+
});
|
|
31
|
+
|
|
32
|
+
if (code !== 0) {
|
|
33
|
+
throw new Error(`Workflow failed with code ${code}`);
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
console.log("\n✅ Workflow terminé avec succès");
|
|
37
|
+
}
|