@vellumai/cli 0.8.12-dev.202606150322.9e174d5 → 0.8.12-dev.202606150824.4ddebe7
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/package.json +1 -1
- package/src/commands/flags.ts +1 -22
- package/src/commands/workflows.ts +301 -0
- package/src/index.ts +3 -0
- package/src/lib/arg-utils.ts +48 -0
package/package.json
CHANGED
package/src/commands/flags.ts
CHANGED
|
@@ -1,3 +1,4 @@
|
|
|
1
|
+
import { extractAssistantFlag } from "../lib/arg-utils.js";
|
|
1
2
|
import { AssistantClient } from "../lib/assistant-client.js";
|
|
2
3
|
import {
|
|
3
4
|
formatAssistantLookupError,
|
|
@@ -199,28 +200,6 @@ async function setFlag(
|
|
|
199
200
|
console.log(`Flag "${key}" set to ${value}`);
|
|
200
201
|
}
|
|
201
202
|
|
|
202
|
-
/**
|
|
203
|
-
* Strip `--assistant <name>` from argv and return the captured value.
|
|
204
|
-
*
|
|
205
|
-
* Mutates the input array so positional parsing downstream sees a clean
|
|
206
|
-
* shape (subcommand + key + value). Returns `undefined` if the flag is
|
|
207
|
-
* absent. Error-reports a missing value so the user gets a clear message
|
|
208
|
-
* rather than the flag being silently swallowed as a positional.
|
|
209
|
-
*/
|
|
210
|
-
function extractAssistantFlag(args: string[]): string | undefined {
|
|
211
|
-
for (let i = 0; i < args.length; i++) {
|
|
212
|
-
if (args[i] !== "--assistant") continue;
|
|
213
|
-
const value = args[i + 1];
|
|
214
|
-
if (!value || value.startsWith("-")) {
|
|
215
|
-
console.error("Missing value for --assistant <name>");
|
|
216
|
-
process.exit(1);
|
|
217
|
-
}
|
|
218
|
-
args.splice(i, 2);
|
|
219
|
-
return value;
|
|
220
|
-
}
|
|
221
|
-
return undefined;
|
|
222
|
-
}
|
|
223
|
-
|
|
224
203
|
export async function flags(): Promise<void> {
|
|
225
204
|
const args = process.argv.slice(3);
|
|
226
205
|
|
|
@@ -0,0 +1,301 @@
|
|
|
1
|
+
import { extractAssistantFlag, extractValueFlag } from "../lib/arg-utils.js";
|
|
2
|
+
import { AssistantClient } from "../lib/assistant-client.js";
|
|
3
|
+
import {
|
|
4
|
+
formatAssistantLookupError,
|
|
5
|
+
lookupAssistantByIdentifier,
|
|
6
|
+
} from "../lib/assistant-config.js";
|
|
7
|
+
|
|
8
|
+
/**
|
|
9
|
+
* Client-side mirror of the server's wire-run projection
|
|
10
|
+
* (`WorkflowRunWire` from `assistant/src/runtime/routes/workflow-routes.ts`).
|
|
11
|
+
* The CLI is an independent build unit and deliberately does NOT import from
|
|
12
|
+
* `assistant/` (see `cli/src/shared/provider-env-vars.ts`), so the shape is
|
|
13
|
+
* mirrored here. Only the fields the CLI renders are declared — the server may
|
|
14
|
+
* send a superset. Keep in sync with `workflowRunSchema`.
|
|
15
|
+
*/
|
|
16
|
+
type WorkflowRun = {
|
|
17
|
+
id: string;
|
|
18
|
+
name: string | null;
|
|
19
|
+
status: string;
|
|
20
|
+
agentsSpawned: number;
|
|
21
|
+
inputTokens: number;
|
|
22
|
+
outputTokens: number;
|
|
23
|
+
error: string | null;
|
|
24
|
+
createdAt: number | null;
|
|
25
|
+
finishedAt: number | null;
|
|
26
|
+
};
|
|
27
|
+
|
|
28
|
+
type SavedWorkflow = {
|
|
29
|
+
name: string;
|
|
30
|
+
description: string;
|
|
31
|
+
path: string;
|
|
32
|
+
};
|
|
33
|
+
|
|
34
|
+
function pad(s: string, w: number): string {
|
|
35
|
+
return s + " ".repeat(Math.max(0, w - s.length));
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
function fmtTime(ms: number | null): string {
|
|
39
|
+
return ms == null ? "-" : new Date(ms).toISOString();
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
function printRunsTable(runs: WorkflowRun[]): void {
|
|
43
|
+
const headers = {
|
|
44
|
+
id: "ID",
|
|
45
|
+
name: "NAME",
|
|
46
|
+
status: "STATUS",
|
|
47
|
+
agents: "AGENTS",
|
|
48
|
+
created: "CREATED",
|
|
49
|
+
};
|
|
50
|
+
const rows = runs.map((r) => ({
|
|
51
|
+
id: r.id,
|
|
52
|
+
name: r.name ?? "-",
|
|
53
|
+
status: r.status,
|
|
54
|
+
agents: String(r.agentsSpawned),
|
|
55
|
+
created: fmtTime(r.createdAt),
|
|
56
|
+
}));
|
|
57
|
+
const all = [headers, ...rows];
|
|
58
|
+
const w = {
|
|
59
|
+
id: Math.max(...all.map((r) => r.id.length)),
|
|
60
|
+
name: Math.max(...all.map((r) => r.name.length)),
|
|
61
|
+
status: Math.max(...all.map((r) => r.status.length)),
|
|
62
|
+
agents: Math.max(...all.map((r) => r.agents.length)),
|
|
63
|
+
created: Math.max(...all.map((r) => r.created.length)),
|
|
64
|
+
};
|
|
65
|
+
const formatRow = (r: typeof headers) =>
|
|
66
|
+
`${pad(r.id, w.id)} ${pad(r.name, w.name)} ${pad(r.status, w.status)} ${pad(r.agents, w.agents)} ${r.created}`;
|
|
67
|
+
console.log(formatRow(headers));
|
|
68
|
+
console.log(
|
|
69
|
+
`${"-".repeat(w.id)} ${"-".repeat(w.name)} ${"-".repeat(w.status)} ${"-".repeat(w.agents)} ${"-".repeat(w.created)}`,
|
|
70
|
+
);
|
|
71
|
+
for (const row of rows) console.log(formatRow(row));
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
function printSavedTable(workflows: SavedWorkflow[]): void {
|
|
75
|
+
const headers = { name: "NAME", description: "DESCRIPTION" };
|
|
76
|
+
const rows = workflows.map((w) => ({
|
|
77
|
+
name: w.name,
|
|
78
|
+
description: w.description,
|
|
79
|
+
}));
|
|
80
|
+
const all = [headers, ...rows];
|
|
81
|
+
const w = {
|
|
82
|
+
name: Math.max(...all.map((r) => r.name.length)),
|
|
83
|
+
description: Math.max(...all.map((r) => r.description.length)),
|
|
84
|
+
};
|
|
85
|
+
const formatRow = (r: typeof headers) =>
|
|
86
|
+
`${pad(r.name, w.name)} ${r.description}`;
|
|
87
|
+
console.log(formatRow(headers));
|
|
88
|
+
console.log(`${"-".repeat(w.name)} ${"-".repeat(w.description)}`);
|
|
89
|
+
for (const row of rows) console.log(formatRow(row));
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
function printHelp(): void {
|
|
93
|
+
console.log("Usage: vellum workflows <subcommand> [options]");
|
|
94
|
+
console.log("");
|
|
95
|
+
console.log("Inspect and control workflow runs on the active assistant.");
|
|
96
|
+
console.log("");
|
|
97
|
+
console.log("Subcommands:");
|
|
98
|
+
console.log(" list List saved (named) workflows");
|
|
99
|
+
console.log(" runs List recent workflow runs");
|
|
100
|
+
console.log(" show <run-id> Show details for a single run");
|
|
101
|
+
console.log(" abort <run-id> Abort an in-flight run");
|
|
102
|
+
console.log(
|
|
103
|
+
" resume <run-id> Resume an interrupted run (orphaned by a restart)",
|
|
104
|
+
);
|
|
105
|
+
console.log("");
|
|
106
|
+
console.log("Options:");
|
|
107
|
+
console.log(
|
|
108
|
+
" --assistant <name> Target a specific assistant (display name or ID)",
|
|
109
|
+
);
|
|
110
|
+
console.log(" --limit <n> (runs) Max runs to list");
|
|
111
|
+
console.log(" --status <status> (runs) Filter by run status");
|
|
112
|
+
console.log(" --help, -h Show this help");
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
function createClient(assistantName?: string): AssistantClient {
|
|
116
|
+
let assistantId: string | undefined;
|
|
117
|
+
if (assistantName) {
|
|
118
|
+
const result = lookupAssistantByIdentifier(assistantName);
|
|
119
|
+
if (result.status !== "found") {
|
|
120
|
+
throw new Error(formatAssistantLookupError(assistantName, result));
|
|
121
|
+
}
|
|
122
|
+
assistantId = result.entry.assistantId;
|
|
123
|
+
}
|
|
124
|
+
try {
|
|
125
|
+
return new AssistantClient(assistantId ? { assistantId } : undefined);
|
|
126
|
+
} catch {
|
|
127
|
+
throw new Error(
|
|
128
|
+
assistantName
|
|
129
|
+
? `No assistant found matching '${assistantName}'.`
|
|
130
|
+
: "No assistant found. Hatch one with 'vellum hatch' first.",
|
|
131
|
+
);
|
|
132
|
+
}
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
function rethrowFetchError(err: unknown): never {
|
|
136
|
+
if (
|
|
137
|
+
err instanceof TypeError &&
|
|
138
|
+
(err.message.includes("fetch") || err.message.includes("connect"))
|
|
139
|
+
) {
|
|
140
|
+
throw new Error(
|
|
141
|
+
"Could not reach the assistant. Is it running? Try 'vellum wake'.",
|
|
142
|
+
);
|
|
143
|
+
}
|
|
144
|
+
throw err;
|
|
145
|
+
}
|
|
146
|
+
|
|
147
|
+
async function requestJson<T>(
|
|
148
|
+
client: AssistantClient,
|
|
149
|
+
method: "get" | "post",
|
|
150
|
+
path: string,
|
|
151
|
+
query?: Record<string, string>,
|
|
152
|
+
): Promise<T> {
|
|
153
|
+
let res: Response;
|
|
154
|
+
try {
|
|
155
|
+
res =
|
|
156
|
+
method === "get"
|
|
157
|
+
? await client.get(path, query ? { query } : undefined)
|
|
158
|
+
: await client.post(path, undefined);
|
|
159
|
+
} catch (err) {
|
|
160
|
+
rethrowFetchError(err);
|
|
161
|
+
}
|
|
162
|
+
if (!res.ok) {
|
|
163
|
+
const body = await res.text().catch(() => "");
|
|
164
|
+
throw new Error(`Request failed: HTTP ${res.status} ${body}`.trim());
|
|
165
|
+
}
|
|
166
|
+
return (await res.json()) as T;
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
async function listSaved(assistantName?: string): Promise<void> {
|
|
170
|
+
const client = createClient(assistantName);
|
|
171
|
+
const data = await requestJson<{ workflows: SavedWorkflow[] }>(
|
|
172
|
+
client,
|
|
173
|
+
"get",
|
|
174
|
+
"/workflows",
|
|
175
|
+
);
|
|
176
|
+
if (data.workflows.length === 0) {
|
|
177
|
+
console.log("No saved workflows found.");
|
|
178
|
+
return;
|
|
179
|
+
}
|
|
180
|
+
printSavedTable(data.workflows);
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
async function listRuns(
|
|
184
|
+
opts: { limit?: string; status?: string },
|
|
185
|
+
assistantName?: string,
|
|
186
|
+
): Promise<void> {
|
|
187
|
+
const client = createClient(assistantName);
|
|
188
|
+
const query: Record<string, string> = {};
|
|
189
|
+
if (opts.limit) query.limit = opts.limit;
|
|
190
|
+
if (opts.status) query.status = opts.status;
|
|
191
|
+
const data = await requestJson<{ runs: WorkflowRun[] }>(
|
|
192
|
+
client,
|
|
193
|
+
"get",
|
|
194
|
+
"/workflows/runs",
|
|
195
|
+
Object.keys(query).length ? query : undefined,
|
|
196
|
+
);
|
|
197
|
+
if (data.runs.length === 0) {
|
|
198
|
+
console.log("No workflow runs found.");
|
|
199
|
+
return;
|
|
200
|
+
}
|
|
201
|
+
printRunsTable(data.runs);
|
|
202
|
+
}
|
|
203
|
+
|
|
204
|
+
async function showRun(runId: string, assistantName?: string): Promise<void> {
|
|
205
|
+
const client = createClient(assistantName);
|
|
206
|
+
const run = await requestJson<WorkflowRun>(
|
|
207
|
+
client,
|
|
208
|
+
"get",
|
|
209
|
+
`/workflows/runs/${runId}`,
|
|
210
|
+
);
|
|
211
|
+
console.log(`ID: ${run.id}`);
|
|
212
|
+
console.log(`Name: ${run.name ?? "(unnamed)"}`);
|
|
213
|
+
console.log(`Status: ${run.status}`);
|
|
214
|
+
console.log(`Agents spawned: ${run.agentsSpawned}`);
|
|
215
|
+
console.log(
|
|
216
|
+
`Tokens: ${run.inputTokens} in / ${run.outputTokens} out`,
|
|
217
|
+
);
|
|
218
|
+
console.log(`Created: ${fmtTime(run.createdAt)}`);
|
|
219
|
+
console.log(`Finished: ${fmtTime(run.finishedAt)}`);
|
|
220
|
+
if (run.error) console.log(`Error: ${run.error}`);
|
|
221
|
+
}
|
|
222
|
+
|
|
223
|
+
async function abortRun(runId: string, assistantName?: string): Promise<void> {
|
|
224
|
+
const client = createClient(assistantName);
|
|
225
|
+
await requestJson<{ ok: boolean; runId: string }>(
|
|
226
|
+
client,
|
|
227
|
+
"post",
|
|
228
|
+
`/workflows/runs/${runId}/abort`,
|
|
229
|
+
);
|
|
230
|
+
console.log(`Abort signalled for workflow run ${runId}.`);
|
|
231
|
+
}
|
|
232
|
+
|
|
233
|
+
async function resumeRun(runId: string, assistantName?: string): Promise<void> {
|
|
234
|
+
const client = createClient(assistantName);
|
|
235
|
+
await requestJson<{ ok: boolean; runId: string }>(
|
|
236
|
+
client,
|
|
237
|
+
"post",
|
|
238
|
+
`/workflows/runs/${runId}/resume`,
|
|
239
|
+
);
|
|
240
|
+
console.log(
|
|
241
|
+
`Resumed workflow run ${runId}. It replays its completed steps and continues from where it was interrupted.`,
|
|
242
|
+
);
|
|
243
|
+
}
|
|
244
|
+
|
|
245
|
+
export async function workflows(): Promise<void> {
|
|
246
|
+
const args = process.argv.slice(3);
|
|
247
|
+
|
|
248
|
+
if (args.includes("--help") || args.includes("-h")) {
|
|
249
|
+
printHelp();
|
|
250
|
+
return;
|
|
251
|
+
}
|
|
252
|
+
|
|
253
|
+
const assistantName = extractAssistantFlag(args);
|
|
254
|
+
const limit = extractValueFlag(args, "limit");
|
|
255
|
+
const status = extractValueFlag(args, "status");
|
|
256
|
+
const subcommand = args[0];
|
|
257
|
+
|
|
258
|
+
if (!subcommand || subcommand === "list") {
|
|
259
|
+
await listSaved(assistantName);
|
|
260
|
+
return;
|
|
261
|
+
}
|
|
262
|
+
|
|
263
|
+
if (subcommand === "runs") {
|
|
264
|
+
await listRuns({ limit, status }, assistantName);
|
|
265
|
+
return;
|
|
266
|
+
}
|
|
267
|
+
|
|
268
|
+
if (subcommand === "show") {
|
|
269
|
+
const runId = args[1];
|
|
270
|
+
if (!runId) {
|
|
271
|
+
console.error("Usage: vellum workflows show <run-id>");
|
|
272
|
+
process.exit(1);
|
|
273
|
+
}
|
|
274
|
+
await showRun(runId, assistantName);
|
|
275
|
+
return;
|
|
276
|
+
}
|
|
277
|
+
|
|
278
|
+
if (subcommand === "abort") {
|
|
279
|
+
const runId = args[1];
|
|
280
|
+
if (!runId) {
|
|
281
|
+
console.error("Usage: vellum workflows abort <run-id>");
|
|
282
|
+
process.exit(1);
|
|
283
|
+
}
|
|
284
|
+
await abortRun(runId, assistantName);
|
|
285
|
+
return;
|
|
286
|
+
}
|
|
287
|
+
|
|
288
|
+
if (subcommand === "resume") {
|
|
289
|
+
const runId = args[1];
|
|
290
|
+
if (!runId) {
|
|
291
|
+
console.error("Usage: vellum workflows resume <run-id>");
|
|
292
|
+
process.exit(1);
|
|
293
|
+
}
|
|
294
|
+
await resumeRun(runId, assistantName);
|
|
295
|
+
return;
|
|
296
|
+
}
|
|
297
|
+
|
|
298
|
+
console.error(`Unknown subcommand: ${subcommand}`);
|
|
299
|
+
printHelp();
|
|
300
|
+
process.exit(1);
|
|
301
|
+
}
|
package/src/index.ts
CHANGED
|
@@ -33,6 +33,7 @@ import { unpair } from "./commands/unpair";
|
|
|
33
33
|
import { upgrade } from "./commands/upgrade";
|
|
34
34
|
import { use } from "./commands/use";
|
|
35
35
|
import { wake } from "./commands/wake";
|
|
36
|
+
import { workflows } from "./commands/workflows";
|
|
36
37
|
import { resolveAssistant, setActiveAssistant } from "./lib/assistant-config";
|
|
37
38
|
import { loadGuardianToken } from "./lib/guardian-token";
|
|
38
39
|
import { checkHealth } from "./lib/health-check";
|
|
@@ -72,6 +73,7 @@ const commands = {
|
|
|
72
73
|
use,
|
|
73
74
|
wake,
|
|
74
75
|
whoami,
|
|
76
|
+
workflows,
|
|
75
77
|
} as const;
|
|
76
78
|
|
|
77
79
|
type CommandName = keyof typeof commands;
|
|
@@ -126,6 +128,7 @@ function printHelp(): void {
|
|
|
126
128
|
console.log(" use Set the active assistant for commands");
|
|
127
129
|
console.log(" wake Start the assistant and gateway");
|
|
128
130
|
console.log(" whoami Show current logged-in user");
|
|
131
|
+
console.log(" workflows Inspect and control workflow runs");
|
|
129
132
|
console.log("");
|
|
130
133
|
console.log("Options:");
|
|
131
134
|
console.log(
|
package/src/lib/arg-utils.ts
CHANGED
|
@@ -11,3 +11,51 @@ export function extractFlag(
|
|
|
11
11
|
const remaining = [...args.slice(0, idx), ...args.slice(idx + 2)];
|
|
12
12
|
return [value, remaining];
|
|
13
13
|
}
|
|
14
|
+
|
|
15
|
+
/**
|
|
16
|
+
* Strip `--<name> <value>` from argv and return the captured value.
|
|
17
|
+
*
|
|
18
|
+
* Mutates the input array so positional parsing downstream sees a clean shape.
|
|
19
|
+
* Returns `undefined` if the flag is absent. Error-reports a missing value (and
|
|
20
|
+
* exits) so the user gets a clear message rather than the flag being silently
|
|
21
|
+
* swallowed as a positional.
|
|
22
|
+
*/
|
|
23
|
+
export function extractValueFlag(
|
|
24
|
+
args: string[],
|
|
25
|
+
name: string,
|
|
26
|
+
): string | undefined {
|
|
27
|
+
for (let i = 0; i < args.length; i++) {
|
|
28
|
+
if (args[i] !== `--${name}`) continue;
|
|
29
|
+
const value = args[i + 1];
|
|
30
|
+
if (!value || value.startsWith("-")) {
|
|
31
|
+
console.error(`Missing value for --${name} <value>`);
|
|
32
|
+
process.exit(1);
|
|
33
|
+
}
|
|
34
|
+
args.splice(i, 2);
|
|
35
|
+
return value;
|
|
36
|
+
}
|
|
37
|
+
return undefined;
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
/**
|
|
41
|
+
* Strip `--assistant <name>` from argv and return the captured value.
|
|
42
|
+
*
|
|
43
|
+
* Mutates the input array so positional parsing downstream sees a clean shape
|
|
44
|
+
* (subcommand + key + value). Returns `undefined` if the flag is absent.
|
|
45
|
+
* Error-reports a missing value so the user gets a clear message rather than
|
|
46
|
+
* the flag being silently swallowed as a positional. (Kept distinct from
|
|
47
|
+
* {@link extractValueFlag} only for its `<name>` wording in the error string.)
|
|
48
|
+
*/
|
|
49
|
+
export function extractAssistantFlag(args: string[]): string | undefined {
|
|
50
|
+
for (let i = 0; i < args.length; i++) {
|
|
51
|
+
if (args[i] !== "--assistant") continue;
|
|
52
|
+
const value = args[i + 1];
|
|
53
|
+
if (!value || value.startsWith("-")) {
|
|
54
|
+
console.error("Missing value for --assistant <name>");
|
|
55
|
+
process.exit(1);
|
|
56
|
+
}
|
|
57
|
+
args.splice(i, 2);
|
|
58
|
+
return value;
|
|
59
|
+
}
|
|
60
|
+
return undefined;
|
|
61
|
+
}
|