@h-rig/cli 0.0.6-alpha.3 → 0.0.6-alpha.30
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/bin/rig.js +3606 -1172
- package/dist/src/commands/_authority-runs.js +1 -0
- package/dist/src/commands/_cli-format.js +369 -0
- package/dist/src/commands/_connection-state.js +1 -3
- package/dist/src/commands/_doctor-checks.js +13 -27
- package/dist/src/commands/_help-catalog.js +388 -0
- package/dist/src/commands/_operator-surface.js +204 -0
- package/dist/src/commands/_operator-view.js +861 -56
- package/dist/src/commands/_parsers.js +0 -2
- package/dist/src/commands/_pi-frontend.js +841 -0
- package/dist/src/commands/_pi-install.js +4 -3
- package/dist/src/commands/_pi-worker-bridge-extension.js +759 -0
- package/dist/src/commands/_policy.js +0 -2
- package/dist/src/commands/_preflight.js +32 -109
- package/dist/src/commands/_run-driver-helpers.js +0 -2
- package/dist/src/commands/_server-client.js +161 -31
- package/dist/src/commands/_snapshot-upload.js +8 -23
- package/dist/src/commands/_task-picker.js +44 -16
- package/dist/src/commands/agent.js +9 -9
- package/dist/src/commands/browser.js +4 -6
- package/dist/src/commands/connect.js +132 -25
- package/dist/src/commands/dist.js +4 -6
- package/dist/src/commands/doctor.js +13 -27
- package/dist/src/commands/github.js +10 -25
- package/dist/src/commands/inbox.js +351 -31
- package/dist/src/commands/init.js +298 -71
- package/dist/src/commands/inspect.js +10 -12
- package/dist/src/commands/inspector.js +2 -4
- package/dist/src/commands/plugin.js +76 -22
- package/dist/src/commands/profile-and-review.js +8 -10
- package/dist/src/commands/queue.js +2 -3
- package/dist/src/commands/remote.js +18 -20
- package/dist/src/commands/repo-git-harness.js +6 -8
- package/dist/src/commands/run.js +1157 -122
- package/dist/src/commands/server.js +217 -33
- package/dist/src/commands/setup.js +17 -37
- package/dist/src/commands/task-report-bug.js +5 -7
- package/dist/src/commands/task-run-driver.js +660 -73
- package/dist/src/commands/task.js +1542 -252
- package/dist/src/commands/test.js +3 -5
- package/dist/src/commands/workspace.js +4 -6
- package/dist/src/commands.js +3599 -1159
- package/dist/src/index.js +3646 -1215
- package/dist/src/launcher.js +5 -3
- package/dist/src/report-bug.js +3 -3
- package/dist/src/runner.js +5 -19
- package/package.json +6 -4
|
@@ -67,6 +67,7 @@ function upsertAgentAuthorityRun(projectRoot, input) {
|
|
|
67
67
|
const runtimeAdapter = normalizeRuntimeAdapter(input.runtimeAdapter ?? existing?.runtimeAdapter ?? "claude-code");
|
|
68
68
|
const title = resolveTaskTitleForAuthorityRun(projectRoot, input.taskId) ?? input.taskId;
|
|
69
69
|
const next = {
|
|
70
|
+
...existing ?? {},
|
|
70
71
|
runId: input.runId,
|
|
71
72
|
projectRoot,
|
|
72
73
|
workspaceId: existing?.workspaceId ?? RIG_WORKSPACE_ID,
|
|
@@ -0,0 +1,369 @@
|
|
|
1
|
+
// @bun
|
|
2
|
+
// packages/cli/src/commands/_cli-format.ts
|
|
3
|
+
import { log, note } from "@clack/prompts";
|
|
4
|
+
import pc from "picocolors";
|
|
5
|
+
function stringField(record, key, fallback = "") {
|
|
6
|
+
const value = record[key];
|
|
7
|
+
return typeof value === "string" && value.trim() ? value.trim() : fallback;
|
|
8
|
+
}
|
|
9
|
+
function numberField(record, key) {
|
|
10
|
+
const value = record[key];
|
|
11
|
+
return typeof value === "number" && Number.isFinite(value) ? value : null;
|
|
12
|
+
}
|
|
13
|
+
function arrayField(record, key) {
|
|
14
|
+
const value = record[key];
|
|
15
|
+
return Array.isArray(value) ? value.flatMap((entry) => typeof entry === "string" && entry.trim() ? [entry.trim()] : []) : [];
|
|
16
|
+
}
|
|
17
|
+
function rawObject(record) {
|
|
18
|
+
const raw = record.raw;
|
|
19
|
+
return raw && typeof raw === "object" && !Array.isArray(raw) ? raw : {};
|
|
20
|
+
}
|
|
21
|
+
function truncate(value, width) {
|
|
22
|
+
if (value.length <= width)
|
|
23
|
+
return value;
|
|
24
|
+
if (width <= 1)
|
|
25
|
+
return "\u2026";
|
|
26
|
+
return `${value.slice(0, width - 1)}\u2026`;
|
|
27
|
+
}
|
|
28
|
+
function pad(value, width) {
|
|
29
|
+
return value.length >= width ? value : `${value}${" ".repeat(width - value.length)}`;
|
|
30
|
+
}
|
|
31
|
+
function statusColor(status) {
|
|
32
|
+
const normalized = status.toLowerCase();
|
|
33
|
+
if (["completed", "merged", "closed", "done", "accepted", "pass", "selected", "approved"].includes(normalized))
|
|
34
|
+
return pc.green;
|
|
35
|
+
if (["failed", "needs_attention", "needs-attention", "blocked", "error", "rejected"].includes(normalized))
|
|
36
|
+
return pc.red;
|
|
37
|
+
if (["running", "reviewing", "validating", "in_progress", "in-progress", "remote"].includes(normalized))
|
|
38
|
+
return pc.cyan;
|
|
39
|
+
if (["ready", "open", "queued", "created", "preparing", "local", "pending"].includes(normalized))
|
|
40
|
+
return pc.yellow;
|
|
41
|
+
return pc.dim;
|
|
42
|
+
}
|
|
43
|
+
function compactDate(value) {
|
|
44
|
+
if (!value.trim())
|
|
45
|
+
return "";
|
|
46
|
+
const parsed = Date.parse(value);
|
|
47
|
+
if (!Number.isFinite(parsed))
|
|
48
|
+
return value;
|
|
49
|
+
return new Date(parsed).toISOString().replace("T", " ").replace(/\.\d{3}Z$/, "Z");
|
|
50
|
+
}
|
|
51
|
+
function compactValue(value) {
|
|
52
|
+
if (value === null || value === undefined)
|
|
53
|
+
return "";
|
|
54
|
+
if (typeof value === "string")
|
|
55
|
+
return value;
|
|
56
|
+
if (typeof value === "number" || typeof value === "boolean")
|
|
57
|
+
return String(value);
|
|
58
|
+
if (Array.isArray(value))
|
|
59
|
+
return value.map(compactValue).filter(Boolean).join(", ");
|
|
60
|
+
return JSON.stringify(value);
|
|
61
|
+
}
|
|
62
|
+
function firstString(record, keys, fallback = "") {
|
|
63
|
+
for (const key of keys) {
|
|
64
|
+
const value = stringField(record, key);
|
|
65
|
+
if (value)
|
|
66
|
+
return value;
|
|
67
|
+
}
|
|
68
|
+
return fallback;
|
|
69
|
+
}
|
|
70
|
+
function runIdOf(run) {
|
|
71
|
+
return firstString(run, ["runId", "id"], "(unknown-run)");
|
|
72
|
+
}
|
|
73
|
+
function taskIdOf(run) {
|
|
74
|
+
return firstString(run, ["taskId", "task", "task_id"]);
|
|
75
|
+
}
|
|
76
|
+
function runTitleOf(run) {
|
|
77
|
+
return firstString(run, ["title", "summary", "name"], taskIdOf(run) || "(untitled)");
|
|
78
|
+
}
|
|
79
|
+
function requestIdOf(entry) {
|
|
80
|
+
return firstString(entry, ["requestId", "id", "approvalId", "inputId"], "(unknown-request)");
|
|
81
|
+
}
|
|
82
|
+
function shouldUseClackOutput() {
|
|
83
|
+
return Boolean(process.stdout.isTTY) && process.env.RIG_CLI_PLAIN_HELP !== "1";
|
|
84
|
+
}
|
|
85
|
+
function printFormattedOutput(message, options = {}) {
|
|
86
|
+
if (!shouldUseClackOutput()) {
|
|
87
|
+
console.log(message);
|
|
88
|
+
return;
|
|
89
|
+
}
|
|
90
|
+
if (options.title)
|
|
91
|
+
note(message, options.title);
|
|
92
|
+
else
|
|
93
|
+
log.message(message);
|
|
94
|
+
}
|
|
95
|
+
function formatStatusPill(status) {
|
|
96
|
+
const label = status || "unknown";
|
|
97
|
+
return statusColor(label)(`\u25CF ${label}`);
|
|
98
|
+
}
|
|
99
|
+
function formatSection(title, subtitle) {
|
|
100
|
+
return `${pc.bold(pc.cyan("\u25C6"))} ${pc.bold(title)}${subtitle ? pc.dim(` \u2014 ${subtitle}`) : ""}`;
|
|
101
|
+
}
|
|
102
|
+
function formatSuccessCard(title, rows = []) {
|
|
103
|
+
const body = rows.filter(([, value]) => value !== undefined && value !== null && String(value).length > 0).map(([key, value]) => `${pc.dim("\u2502")} ${pc.dim(key.padEnd(12))} ${value}`);
|
|
104
|
+
return [formatSection(title), ...body].join(`
|
|
105
|
+
`);
|
|
106
|
+
}
|
|
107
|
+
function formatNextSteps(steps) {
|
|
108
|
+
if (steps.length === 0)
|
|
109
|
+
return [];
|
|
110
|
+
return [pc.bold("Next"), ...steps.map((step) => `${pc.dim("\u203A")} ${step}`)];
|
|
111
|
+
}
|
|
112
|
+
function formatTaskList(tasks, options = {}) {
|
|
113
|
+
if (options.raw)
|
|
114
|
+
return tasks.map((task) => JSON.stringify(task)).join(`
|
|
115
|
+
`);
|
|
116
|
+
if (tasks.length === 0)
|
|
117
|
+
return [formatSection("Tasks", "none found"), ...formatNextSteps(["Try `rig server status` to confirm the selected server.", "Relax filters or run `rig task run --title ... --initial-prompt ...` for ad hoc work."])].join(`
|
|
118
|
+
`);
|
|
119
|
+
const rows = tasks.map((task) => {
|
|
120
|
+
const raw = rawObject(task);
|
|
121
|
+
const id = stringField(task, "id", "<unknown>");
|
|
122
|
+
const status = stringField(task, "status", "unknown");
|
|
123
|
+
const title = stringField(task, "title", "Untitled task");
|
|
124
|
+
const source = stringField(task, "source", stringField(raw, "source", ""));
|
|
125
|
+
const labels = arrayField(task, "labels").length > 0 ? arrayField(task, "labels") : arrayField(raw, "labels");
|
|
126
|
+
return { id, status, title, source, labels };
|
|
127
|
+
});
|
|
128
|
+
const idWidth = Math.min(18, Math.max(4, ...rows.map((row) => row.id.length)));
|
|
129
|
+
const statusWidth = Math.min(16, Math.max(6, ...rows.map((row) => row.status.length)));
|
|
130
|
+
const header = `${pc.bold(pad("TASK", idWidth))} ${pc.bold(pad("STATUS", statusWidth))} ${pc.bold("TITLE")}`;
|
|
131
|
+
const body = rows.map((row) => {
|
|
132
|
+
const labels = row.labels.length > 0 ? pc.dim(` ${row.labels.slice(0, 4).map((label) => `#${label}`).join(" ")}`) : "";
|
|
133
|
+
const source = row.source ? pc.dim(` ${row.source}`) : "";
|
|
134
|
+
return [
|
|
135
|
+
pc.bold(pad(truncate(row.id, idWidth), idWidth)),
|
|
136
|
+
statusColor(row.status)(pad(truncate(row.status, statusWidth), statusWidth)),
|
|
137
|
+
`${row.title}${labels}${source}`
|
|
138
|
+
].join(" ");
|
|
139
|
+
});
|
|
140
|
+
return [formatSection("Tasks", `${rows.length} shown`), header, ...body, "", ...formatNextSteps(["Run one: `rig task run <id>` or `rig task run --next`", "Attach later: `rig run attach <run-id> --follow`"])].join(`
|
|
141
|
+
`);
|
|
142
|
+
}
|
|
143
|
+
function formatTaskCard(task, options = {}) {
|
|
144
|
+
const raw = rawObject(task);
|
|
145
|
+
const id = stringField(task, "id", stringField(raw, "id", "<unknown>"));
|
|
146
|
+
const status = stringField(task, "status", stringField(raw, "status", "unknown"));
|
|
147
|
+
const title = stringField(task, "title", stringField(raw, "title", "Untitled task"));
|
|
148
|
+
const source = stringField(task, "source", stringField(raw, "source", ""));
|
|
149
|
+
const url = stringField(task, "url", stringField(raw, "url", ""));
|
|
150
|
+
const number = numberField(task, "number") ?? numberField(raw, "number");
|
|
151
|
+
const labels = arrayField(task, "labels").length > 0 ? arrayField(task, "labels") : arrayField(raw, "labels");
|
|
152
|
+
const assignees = arrayField(task, "assignees").length > 0 ? arrayField(task, "assignees") : arrayField(raw, "assignees");
|
|
153
|
+
const readiness = compactValue(task.readiness ?? raw.readiness);
|
|
154
|
+
const validators = compactValue(task.validators ?? raw.validators ?? task.validation ?? raw.validation);
|
|
155
|
+
const rows = [
|
|
156
|
+
["task", pc.bold(id)],
|
|
157
|
+
["status", formatStatusPill(status)],
|
|
158
|
+
["title", title],
|
|
159
|
+
["source", source],
|
|
160
|
+
["number", number],
|
|
161
|
+
["labels", labels.length ? labels.map((label) => `#${label}`).join(" ") : ""],
|
|
162
|
+
["assignees", assignees.join(", ")],
|
|
163
|
+
["readiness", readiness],
|
|
164
|
+
["validators", validators],
|
|
165
|
+
["url", url]
|
|
166
|
+
];
|
|
167
|
+
return [
|
|
168
|
+
formatSuccessCard(options.title ?? (options.selected ? "Selected task" : "Task"), rows),
|
|
169
|
+
"",
|
|
170
|
+
...formatNextSteps([`Start: \`rig task run ${id}\``, `Details: \`rig task show ${id} --raw\``])
|
|
171
|
+
].join(`
|
|
172
|
+
`);
|
|
173
|
+
}
|
|
174
|
+
function formatTaskDetails(task) {
|
|
175
|
+
return formatTaskCard(task, { title: "Task details" });
|
|
176
|
+
}
|
|
177
|
+
function formatRunList(runs, options = {}) {
|
|
178
|
+
if (runs.length === 0) {
|
|
179
|
+
return [
|
|
180
|
+
formatSection("Runs", "none recorded"),
|
|
181
|
+
options.source === "server" ? pc.dim("No runs recorded on the selected Rig server.") : pc.dim("No runs recorded in .rig/runs."),
|
|
182
|
+
"",
|
|
183
|
+
...formatNextSteps(["Start one: `rig task run --next`", "Check server: `rig server status`"])
|
|
184
|
+
].join(`
|
|
185
|
+
`);
|
|
186
|
+
}
|
|
187
|
+
const rows = runs.map((run) => {
|
|
188
|
+
const runId = stringField(run, "runId", stringField(run, "id", "(unknown-run)"));
|
|
189
|
+
const status = stringField(run, "status", "unknown");
|
|
190
|
+
const taskId = stringField(run, "taskId", "");
|
|
191
|
+
const title = stringField(run, "title", taskId || "(untitled)");
|
|
192
|
+
const runtime = stringField(run, "runtimeAdapter", "");
|
|
193
|
+
return { runId, status, title, runtime };
|
|
194
|
+
});
|
|
195
|
+
const idWidth = Math.min(36, Math.max(6, ...rows.map((row) => row.runId.length)));
|
|
196
|
+
const statusWidth = Math.min(16, Math.max(6, ...rows.map((row) => row.status.length)));
|
|
197
|
+
const header = `${pc.bold(pad("RUN", idWidth))} ${pc.bold(pad("STATUS", statusWidth))} ${pc.bold("TITLE")}`;
|
|
198
|
+
const body = rows.map((row) => [
|
|
199
|
+
pc.bold(pad(truncate(row.runId, idWidth), idWidth)),
|
|
200
|
+
statusColor(row.status)(pad(truncate(row.status, statusWidth), statusWidth)),
|
|
201
|
+
`${row.title}${row.runtime ? pc.dim(` ${row.runtime}`) : ""}`
|
|
202
|
+
].join(" "));
|
|
203
|
+
return [formatSection("Runs", options.source === "server" ? "selected server" : "local state"), header, ...body, "", ...formatNextSteps(["Follow live: `rig run attach <run-id> --follow`", "Details: `rig run show <run-id>`"])].join(`
|
|
204
|
+
`);
|
|
205
|
+
}
|
|
206
|
+
function formatSubmittedRun(input) {
|
|
207
|
+
const rows = [["run", pc.bold(input.runId)]];
|
|
208
|
+
if (input.task) {
|
|
209
|
+
const id = stringField(input.task, "id", "<unknown>");
|
|
210
|
+
const status = stringField(input.task, "status", "unknown");
|
|
211
|
+
const title = stringField(input.task, "title", "Untitled task");
|
|
212
|
+
rows.push(["task", `${pc.bold(id)} ${formatStatusPill(status)} ${title}`]);
|
|
213
|
+
}
|
|
214
|
+
const runtime = [input.runtimeAdapter || "pi", input.runtimeMode || "full-access", input.interactionMode || "default"].filter(Boolean).join(" \xB7 ");
|
|
215
|
+
rows.push(["runtime", runtime]);
|
|
216
|
+
return [
|
|
217
|
+
formatSuccessCard("Run submitted", rows),
|
|
218
|
+
"",
|
|
219
|
+
...formatNextSteps([
|
|
220
|
+
`Attach: \`rig run attach ${input.runId} --follow\``,
|
|
221
|
+
`Inspect: \`rig run show ${input.runId}\``,
|
|
222
|
+
input.detached ? "Submitted detached; attach when you are ready." : "Interactive mode opens the native bundled Pi frontend."
|
|
223
|
+
])
|
|
224
|
+
].join(`
|
|
225
|
+
`);
|
|
226
|
+
}
|
|
227
|
+
function formatRunCard(run, options = {}) {
|
|
228
|
+
const raw = rawObject(run);
|
|
229
|
+
const merged = { ...raw, ...run };
|
|
230
|
+
const runId = runIdOf(merged);
|
|
231
|
+
const status = firstString(merged, ["status"], "unknown");
|
|
232
|
+
const taskId = taskIdOf(merged);
|
|
233
|
+
const title = runTitleOf(merged);
|
|
234
|
+
const runtime = firstString(merged, ["runtimeAdapter", "runtime", "adapter"]);
|
|
235
|
+
const mode = firstString(merged, ["runtimeMode", "mode"]);
|
|
236
|
+
const interaction = firstString(merged, ["interactionMode"]);
|
|
237
|
+
const created = compactDate(firstString(merged, ["createdAt"]));
|
|
238
|
+
const started = compactDate(firstString(merged, ["startedAt"]));
|
|
239
|
+
const updated = compactDate(firstString(merged, ["updatedAt"]));
|
|
240
|
+
const completed = compactDate(firstString(merged, ["completedAt", "finishedAt"]));
|
|
241
|
+
const worktree = firstString(merged, ["worktreePath", "cwd", "projectRoot"]);
|
|
242
|
+
const piSession = merged.piSession && typeof merged.piSession === "object" && !Array.isArray(merged.piSession) ? firstString(merged.piSession, ["sessionId", "id"]) : "";
|
|
243
|
+
const timeline = Array.isArray(merged.timeline) ? merged.timeline.length : null;
|
|
244
|
+
const approvals = Array.isArray(merged.approvals) ? merged.approvals.length : null;
|
|
245
|
+
const inputs = Array.isArray(merged.userInputs) ? merged.userInputs.length : null;
|
|
246
|
+
const rows = [
|
|
247
|
+
["run", pc.bold(runId)],
|
|
248
|
+
["status", formatStatusPill(status)],
|
|
249
|
+
["task", taskId],
|
|
250
|
+
["title", title],
|
|
251
|
+
["runtime", [runtime, mode, interaction].filter(Boolean).join(" \xB7 ")],
|
|
252
|
+
["created", created],
|
|
253
|
+
["started", started],
|
|
254
|
+
["updated", updated],
|
|
255
|
+
["completed", completed],
|
|
256
|
+
["worktree", worktree],
|
|
257
|
+
["pi", piSession],
|
|
258
|
+
["timeline", timeline],
|
|
259
|
+
["approvals", approvals],
|
|
260
|
+
["inputs", inputs]
|
|
261
|
+
];
|
|
262
|
+
return [
|
|
263
|
+
formatSuccessCard(options.title ?? "Run details", rows),
|
|
264
|
+
"",
|
|
265
|
+
...formatNextSteps([`Follow live: \`rig run attach ${runId} --follow\``, `Raw payload: \`rig run show ${runId} --raw\``])
|
|
266
|
+
].join(`
|
|
267
|
+
`);
|
|
268
|
+
}
|
|
269
|
+
function formatRunStatus(summary, options = {}) {
|
|
270
|
+
const activeRuns = summary.activeRuns ?? [];
|
|
271
|
+
const recentRuns = summary.recentRuns ?? [];
|
|
272
|
+
const lines = [formatSection("Run status", options.source === "server" ? "selected server" : "local state")];
|
|
273
|
+
lines.push("", pc.bold(`Active runs (${activeRuns.length})`));
|
|
274
|
+
if (activeRuns.length === 0) {
|
|
275
|
+
lines.push(pc.dim("No active runs."));
|
|
276
|
+
} else {
|
|
277
|
+
for (const run of activeRuns) {
|
|
278
|
+
lines.push(formatRunSummaryLine(run));
|
|
279
|
+
}
|
|
280
|
+
}
|
|
281
|
+
lines.push("", pc.bold(`Recent runs (${recentRuns.length})`));
|
|
282
|
+
if (recentRuns.length === 0) {
|
|
283
|
+
lines.push(pc.dim("No recent terminal runs."));
|
|
284
|
+
} else {
|
|
285
|
+
for (const run of recentRuns.slice(0, 10)) {
|
|
286
|
+
lines.push(formatRunSummaryLine(run));
|
|
287
|
+
}
|
|
288
|
+
}
|
|
289
|
+
lines.push("", ...formatNextSteps(["Start work: `rig task run --next`", "Attach: `rig run attach <run-id> --follow`", "Details: `rig run show <run-id>`"]));
|
|
290
|
+
return lines.join(`
|
|
291
|
+
`);
|
|
292
|
+
}
|
|
293
|
+
function formatRunSummaryLine(run) {
|
|
294
|
+
const record = run;
|
|
295
|
+
const runId = runIdOf(record);
|
|
296
|
+
const status = firstString(record, ["status"], "unknown");
|
|
297
|
+
const taskId = taskIdOf(record);
|
|
298
|
+
const title = runTitleOf(record);
|
|
299
|
+
const runtime = firstString(record, ["runtimeAdapter", "runtime", "adapter"]);
|
|
300
|
+
const descriptor = [taskId, title].filter(Boolean).join(" \xB7 ");
|
|
301
|
+
return `${pc.dim("\u2502")} ${pc.bold(runId)} ${formatStatusPill(status)} ${descriptor}${runtime ? pc.dim(` ${runtime}`) : ""}`;
|
|
302
|
+
}
|
|
303
|
+
function formatInboxList(kind, entries) {
|
|
304
|
+
const title = kind === "approvals" ? "Approval inbox" : "Input inbox";
|
|
305
|
+
if (entries.length === 0) {
|
|
306
|
+
return [
|
|
307
|
+
formatSection(title, "empty"),
|
|
308
|
+
pc.dim(kind === "approvals" ? "No pending approvals." : "No pending user-input requests."),
|
|
309
|
+
"",
|
|
310
|
+
...formatNextSteps(["Check runs: `rig run status`", "Start work: `rig task run --next`"])
|
|
311
|
+
].join(`
|
|
312
|
+
`);
|
|
313
|
+
}
|
|
314
|
+
const lines = [formatSection(title, `${entries.length} pending`)];
|
|
315
|
+
for (const entry of entries) {
|
|
316
|
+
const record = entry.record && typeof entry.record === "object" && !Array.isArray(entry.record) ? entry.record : entry;
|
|
317
|
+
const runId = firstString(entry, ["runId"], firstString(record, ["runId"]));
|
|
318
|
+
const taskId = firstString(entry, ["taskId"], firstString(record, ["taskId", "task"]));
|
|
319
|
+
const requestId = requestIdOf(record);
|
|
320
|
+
const status = firstString(record, ["status", "state"], "pending");
|
|
321
|
+
const prompt = firstString(record, ["prompt", "message", "reason", "title", "summary"], kind === "approvals" ? "Approval requested" : "Input requested");
|
|
322
|
+
lines.push(`${pc.dim("\u2502")} ${pc.bold(requestId)} ${formatStatusPill(status)} ${prompt}`);
|
|
323
|
+
lines.push(`${pc.dim("\u2502")} ${pc.dim("run ")} ${runId || "(unknown-run)"}${taskId ? pc.dim(` task ${taskId}`) : ""}`);
|
|
324
|
+
}
|
|
325
|
+
lines.push("", ...formatNextSteps(kind === "approvals" ? ["Resolve: `rig inbox approve --run <run-id> --request <request-id> --decision approve|reject`", "Rejoin: `rig run attach <run-id> --follow`"] : ["Respond: `rig inbox respond --run <run-id> --request <request-id> --answer key=value`", "Rejoin: `rig run attach <run-id> --follow`"]));
|
|
326
|
+
return lines.join(`
|
|
327
|
+
`);
|
|
328
|
+
}
|
|
329
|
+
function formatConnectionList(connections) {
|
|
330
|
+
const rows = [["local", { kind: "local", mode: "auto" }], ...Object.entries(connections)];
|
|
331
|
+
const aliasWidth = Math.min(24, Math.max(5, ...rows.map(([alias]) => alias.length)));
|
|
332
|
+
const lines = rows.map(([alias, connection]) => [
|
|
333
|
+
pc.bold(pad(truncate(alias, aliasWidth), aliasWidth)),
|
|
334
|
+
formatStatusPill(connection.kind),
|
|
335
|
+
connection.kind === "remote" ? connection.baseUrl ?? "" : connection.mode ?? "local"
|
|
336
|
+
].join(" "));
|
|
337
|
+
return [formatSection("Rig servers", `${rows.length} available`), `${pc.bold(pad("ALIAS", aliasWidth))} ${pc.bold("KIND")} ${pc.bold("TARGET")}`, ...lines, "", ...formatNextSteps(["Select one: `rig server use <alias|local>`"])].join(`
|
|
338
|
+
`);
|
|
339
|
+
}
|
|
340
|
+
function formatConnectionStatus(selected, connections) {
|
|
341
|
+
const connection = selected === "local" ? { kind: "local", mode: "auto" } : connections[selected];
|
|
342
|
+
const target = !connection ? "not configured" : connection.kind === "remote" ? connection.baseUrl : "local";
|
|
343
|
+
return [
|
|
344
|
+
formatSection("Rig server", "selected for this repo"),
|
|
345
|
+
`${pc.dim("\u2502")} ${pc.dim("selected ")} ${pc.bold(selected)}`,
|
|
346
|
+
`${pc.dim("\u2502")} ${pc.dim("kind ")} ${formatStatusPill(connection?.kind ?? "unknown")}`,
|
|
347
|
+
`${pc.dim("\u2502")} ${pc.dim("target ")} ${target ?? "not configured"}`,
|
|
348
|
+
"",
|
|
349
|
+
...formatNextSteps(["Change: `rig server use <alias|local>`", "List saved servers: `rig server list`"])
|
|
350
|
+
].join(`
|
|
351
|
+
`);
|
|
352
|
+
}
|
|
353
|
+
export {
|
|
354
|
+
printFormattedOutput,
|
|
355
|
+
formatTaskList,
|
|
356
|
+
formatTaskDetails,
|
|
357
|
+
formatTaskCard,
|
|
358
|
+
formatSuccessCard,
|
|
359
|
+
formatSubmittedRun,
|
|
360
|
+
formatStatusPill,
|
|
361
|
+
formatSection,
|
|
362
|
+
formatRunStatus,
|
|
363
|
+
formatRunList,
|
|
364
|
+
formatRunCard,
|
|
365
|
+
formatNextSteps,
|
|
366
|
+
formatInboxList,
|
|
367
|
+
formatConnectionStatus,
|
|
368
|
+
formatConnectionList
|
|
369
|
+
};
|
|
@@ -8,8 +8,6 @@ import { dirname, resolve } from "path";
|
|
|
8
8
|
import { EventBus } from "@rig/runtime/control-plane/runtime/events";
|
|
9
9
|
import { CliError } from "@rig/runtime/control-plane/errors";
|
|
10
10
|
import { evaluate, loadPolicy, resolveAction } from "@rig/runtime/control-plane/runtime/guard";
|
|
11
|
-
import { PluginManager } from "@rig/runtime/control-plane/runtime/plugins";
|
|
12
|
-
import { loadRuntimeContextFromEnv } from "@rig/runtime/control-plane/runtime/context";
|
|
13
11
|
import { buildBinary } from "@rig/runtime/control-plane/runtime/isolation";
|
|
14
12
|
import { CliError as CliError2 } from "@rig/runtime/control-plane/errors";
|
|
15
13
|
|
|
@@ -107,7 +105,7 @@ function resolveSelectedConnection(projectRoot, options = {}) {
|
|
|
107
105
|
const global = readGlobalConnections(options);
|
|
108
106
|
const connection = global.connections[repo.selected];
|
|
109
107
|
if (!connection) {
|
|
110
|
-
throw new CliError2(`Selected Rig
|
|
108
|
+
throw new CliError2(`Selected Rig server "${repo.selected}" was not found. Run \`rig server list\` or \`rig server use local\`.`, 1);
|
|
111
109
|
}
|
|
112
110
|
return { alias: repo.selected, connection };
|
|
113
111
|
}
|
|
@@ -9,8 +9,6 @@ import { resolve as resolve4 } from "path";
|
|
|
9
9
|
import { EventBus } from "@rig/runtime/control-plane/runtime/events";
|
|
10
10
|
import { CliError } from "@rig/runtime/control-plane/errors";
|
|
11
11
|
import { evaluate, loadPolicy, resolveAction } from "@rig/runtime/control-plane/runtime/guard";
|
|
12
|
-
import { PluginManager } from "@rig/runtime/control-plane/runtime/plugins";
|
|
13
|
-
import { loadRuntimeContextFromEnv } from "@rig/runtime/control-plane/runtime/context";
|
|
14
12
|
import { buildBinary } from "@rig/runtime/control-plane/runtime/isolation";
|
|
15
13
|
import { CliError as CliError2 } from "@rig/runtime/control-plane/errors";
|
|
16
14
|
|
|
@@ -94,17 +92,16 @@ function resolveSelectedConnection(projectRoot, options = {}) {
|
|
|
94
92
|
const global = readGlobalConnections(options);
|
|
95
93
|
const connection = global.connections[repo.selected];
|
|
96
94
|
if (!connection) {
|
|
97
|
-
throw new CliError2(`Selected Rig
|
|
95
|
+
throw new CliError2(`Selected Rig server "${repo.selected}" was not found. Run \`rig server list\` or \`rig server use local\`.`, 1);
|
|
98
96
|
}
|
|
99
97
|
return { alias: repo.selected, connection };
|
|
100
98
|
}
|
|
101
99
|
|
|
102
100
|
// packages/cli/src/commands/_server-client.ts
|
|
103
|
-
import { spawnSync } from "child_process";
|
|
104
101
|
import { existsSync as existsSync2, readFileSync as readFileSync2 } from "fs";
|
|
105
102
|
import { resolve as resolve2 } from "path";
|
|
106
103
|
import { ensureLocalRigServerConnection } from "@rig/runtime/local-server";
|
|
107
|
-
var
|
|
104
|
+
var scopedGitHubBearerTokens = new Map;
|
|
108
105
|
function cleanToken(value) {
|
|
109
106
|
const trimmed = value?.trim();
|
|
110
107
|
return trimmed ? trimmed : null;
|
|
@@ -121,25 +118,13 @@ function readPrivateRemoteSessionToken(projectRoot) {
|
|
|
121
118
|
}
|
|
122
119
|
}
|
|
123
120
|
function readGitHubBearerTokenForRemote(projectRoot) {
|
|
124
|
-
|
|
125
|
-
|
|
121
|
+
const scopedKey = resolve2(projectRoot);
|
|
122
|
+
if (scopedGitHubBearerTokens.has(scopedKey))
|
|
123
|
+
return scopedGitHubBearerTokens.get(scopedKey) ?? null;
|
|
126
124
|
const privateSession = readPrivateRemoteSessionToken(projectRoot);
|
|
127
|
-
if (privateSession)
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
}
|
|
131
|
-
const envToken = cleanToken(process.env.RIG_GITHUB_TOKEN) ?? cleanToken(process.env.GITHUB_TOKEN) ?? cleanToken(process.env.GH_TOKEN);
|
|
132
|
-
if (envToken) {
|
|
133
|
-
cachedGitHubBearerToken = envToken;
|
|
134
|
-
return cachedGitHubBearerToken;
|
|
135
|
-
}
|
|
136
|
-
const result = spawnSync("gh", ["auth", "token"], {
|
|
137
|
-
encoding: "utf8",
|
|
138
|
-
timeout: 5000,
|
|
139
|
-
stdio: ["ignore", "pipe", "ignore"]
|
|
140
|
-
});
|
|
141
|
-
cachedGitHubBearerToken = result.status === 0 ? cleanToken(result.stdout) : null;
|
|
142
|
-
return cachedGitHubBearerToken;
|
|
125
|
+
if (privateSession)
|
|
126
|
+
return privateSession;
|
|
127
|
+
return cleanToken(process.env.RIG_SERVER_AUTH_TOKEN) ?? cleanToken(process.env.RIG_REMOTE_AUTH_TOKEN);
|
|
143
128
|
}
|
|
144
129
|
async function ensureServerForCli(projectRoot) {
|
|
145
130
|
try {
|
|
@@ -222,7 +207,8 @@ async function loadRigConfigOrNull(projectRoot) {
|
|
|
222
207
|
import { existsSync as existsSync3, readFileSync as readFileSync3, rmSync } from "fs";
|
|
223
208
|
import { homedir as homedir2 } from "os";
|
|
224
209
|
import { resolve as resolve3 } from "path";
|
|
225
|
-
var PI_RIG_PACKAGE_NAME = "@rig/pi-rig";
|
|
210
|
+
var PI_RIG_PACKAGE_NAME = "@h-rig/pi-rig";
|
|
211
|
+
var LEGACY_PI_RIG_PACKAGE_NAME = "@rig/pi-rig";
|
|
226
212
|
async function defaultCommandRunner(command, options = {}) {
|
|
227
213
|
const proc = Bun.spawn(command, { cwd: options.cwd, stdout: "pipe", stderr: "pipe" });
|
|
228
214
|
const [stdout, stderr, exitCode] = await Promise.all([
|
|
@@ -241,7 +227,7 @@ function resolvePiHomeDir(inputHomeDir) {
|
|
|
241
227
|
function piListContainsPiRig(output) {
|
|
242
228
|
return output.split(/\r?\n/).some((line) => {
|
|
243
229
|
const normalized = line.trim();
|
|
244
|
-
return normalized.includes(PI_RIG_PACKAGE_NAME) || /(?:^|[\\/])packages[\\/]pi-rig(?:$|\s)/.test(normalized);
|
|
230
|
+
return normalized.includes(PI_RIG_PACKAGE_NAME) || normalized.includes(LEGACY_PI_RIG_PACKAGE_NAME) || /(?:^|[\\/])packages[\\/]pi-rig(?:$|\s)/.test(normalized);
|
|
245
231
|
});
|
|
246
232
|
}
|
|
247
233
|
async function safeRun(runner, command, options) {
|
|
@@ -417,7 +403,7 @@ async function runRigDoctorChecks(options) {
|
|
|
417
403
|
const taskSourceKind = config?.taskSource?.kind;
|
|
418
404
|
checks.push(taskSourceKind ? check("task-source", "task source configured", "pass", taskSourceKind) : check("task-source", "task source configured", "fail", "missing taskSource", "Configure taskSource in rig.config.ts."));
|
|
419
405
|
const repo = readRepoConnection(projectRoot);
|
|
420
|
-
checks.push(repo ? check("project-link", "repo selected Rig
|
|
406
|
+
checks.push(repo ? check("project-link", "repo selected Rig server", repo.project ? "pass" : "warn", `${repo.selected}${repo.project ? ` -> ${repo.project}` : ""}`, "Run `rig init --yes --repo owner/repo` to link this checkout to a GitHub repo slug.") : check("project-link", "repo selected Rig server", "fail", "missing .rig/state/connection.json", "Run `rig init` or `rig server use <alias|local>`."));
|
|
421
407
|
const selected = (() => {
|
|
422
408
|
try {
|
|
423
409
|
return resolveSelectedConnection(projectRoot);
|
|
@@ -425,7 +411,7 @@ async function runRigDoctorChecks(options) {
|
|
|
425
411
|
return null;
|
|
426
412
|
}
|
|
427
413
|
})();
|
|
428
|
-
checks.push(selected ? check("connection", "selected server connection", "pass", selected.connection.kind === "remote" ? selected.connection.baseUrl : "local auto") : check("connection", "selected server
|
|
414
|
+
checks.push(selected ? check("connection", "selected server connection", "pass", selected.connection.kind === "remote" ? selected.connection.baseUrl : "local auto") : check("connection", "selected server", repo ? "fail" : "warn", repo ? "selected alias is missing" : "will auto-start local server", repo ? "Run `rig server list` and `rig server use <alias|local>`." : undefined));
|
|
429
415
|
let server = null;
|
|
430
416
|
try {
|
|
431
417
|
server = await (options.resolveServer ?? ensureServerForCli)(projectRoot);
|