- notification-store: schema v2 — repeatCount/lastTs merge for non-blocking notices; NOTICE_KIND enum (SYSTEM_NOTICE, TOOL_NOTICE, BLOCKING_NOTICE, USER_VISIBLE) for renderer classification without message parsing - sf-db: remove gate_runs and audit_events tables (replaced by uok audit.js and trace-writer); schema reduced by ~370 lines - notify-interceptor: tag auto-mode system notices with NOTICE_KIND.SYSTEM_NOTICE - auto-prompts, guided-flow, system-context: use NOTICE_KIND on emit calls - cli-status: expanded headless status surface + test coverage - headless-types: new status fields - Makefile/justfile: dev workflow improvements - record-promoter, requirement-promoter: minor cleanup - sf-db-migration tests: updated for dropped tables - uok-gate-runner, uok-metrics, uok-outcome, uok-status tests: updated Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com>
168 lines
3.8 KiB
TypeScript
168 lines
3.8 KiB
TypeScript
import assert from "node:assert/strict";
|
|
import { mkdirSync, mkdtempSync, rmSync, writeFileSync } from "node:fs";
|
|
import { tmpdir } from "node:os";
|
|
import { join } from "node:path";
|
|
import { afterEach, test } from "vitest";
|
|
import {
|
|
renderLiveStatus,
|
|
resolveRecoveryPick,
|
|
runStatusCli,
|
|
} from "../cli-status.ts";
|
|
|
|
const tempDirs: string[] = [];
|
|
|
|
function makeProject(): string {
|
|
const dir = mkdtempSync(join(tmpdir(), "sf-cli-status-"));
|
|
tempDirs.push(dir);
|
|
return dir;
|
|
}
|
|
|
|
afterEach(() => {
|
|
for (const dir of tempDirs) rmSync(dir, { recursive: true, force: true });
|
|
tempDirs.length = 0;
|
|
});
|
|
|
|
function snapshot() {
|
|
return {
|
|
state: {
|
|
activeMilestone: { id: "M001", title: "Milestone" },
|
|
activeSlice: { id: "S01", title: "Slice" },
|
|
activeTask: { id: "T01", title: "Task" },
|
|
phase: "executing",
|
|
nextAction: "execute",
|
|
},
|
|
next: {
|
|
action: "dispatch",
|
|
unitType: "execute-task",
|
|
unitId: "M001/S01/T01",
|
|
},
|
|
cost: { total: 0, workers: [] },
|
|
};
|
|
}
|
|
|
|
test("resolveRecoveryPick_auto_selects_most_recent_row", () => {
|
|
const picked = resolveRecoveryPick(
|
|
"/tmp",
|
|
[
|
|
{
|
|
unitType: "execute-task",
|
|
unitId: "M001/S01/T01",
|
|
updatedAt: 100,
|
|
},
|
|
{
|
|
unitType: "discuss-milestone",
|
|
unitId: "M001-X",
|
|
updatedAt: 200,
|
|
},
|
|
],
|
|
undefined,
|
|
() => null,
|
|
);
|
|
assert.deepEqual(picked, {
|
|
unitType: "discuss-milestone",
|
|
unitId: "M001-X",
|
|
});
|
|
});
|
|
|
|
test("resolveRecoveryPick_explicit_unitId_uses_matching_unitType", () => {
|
|
const picked = resolveRecoveryPick(
|
|
"/tmp",
|
|
[
|
|
{
|
|
unitType: "discuss-milestone",
|
|
unitId: "M001-X",
|
|
updatedAt: 500,
|
|
},
|
|
{
|
|
unitType: "execute-task",
|
|
unitId: "M001/S01/T01",
|
|
updatedAt: 900,
|
|
},
|
|
],
|
|
"M001-X",
|
|
() => null,
|
|
);
|
|
assert.deepEqual(picked, {
|
|
unitType: "discuss-milestone",
|
|
unitId: "M001-X",
|
|
});
|
|
});
|
|
|
|
test("resolveRecoveryPick_fallback_to_execute_task_when_not_in_list_but_on_disk", () => {
|
|
const picked = resolveRecoveryPick(
|
|
"/tmp",
|
|
[],
|
|
"M001/S01/T01",
|
|
(_base, ut, uid) =>
|
|
ut === "execute-task" && uid === "M001/S01/T01" ? { status: "x" } : null,
|
|
);
|
|
assert.deepEqual(picked, {
|
|
unitType: "execute-task",
|
|
unitId: "M001/S01/T01",
|
|
});
|
|
});
|
|
|
|
test("runStatusCli_recovery_when_newest_runtime_is_non_execute_task_succeeds", async () => {
|
|
const project = makeProject();
|
|
const { writeUnitRuntimeRecord } = await import(
|
|
"../resources/extensions/sf/uok/unit-runtime.js"
|
|
);
|
|
const tOld = Date.now() - 5_000;
|
|
writeUnitRuntimeRecord(project, "execute-task", "M001/S01/T01", tOld, {
|
|
status: "running",
|
|
});
|
|
writeUnitRuntimeRecord(project, "discuss-milestone", "M002-Y", Date.now(), {
|
|
status: "failed",
|
|
});
|
|
let out = "";
|
|
let err = "";
|
|
const code = await runStatusCli(["status", "recovery"], {
|
|
basePath: project,
|
|
stdout: {
|
|
write(s: string) {
|
|
out += s;
|
|
},
|
|
isTTY: false,
|
|
},
|
|
stderr: {
|
|
write(s: string) {
|
|
err += s;
|
|
},
|
|
},
|
|
});
|
|
assert.equal(code, 0);
|
|
assert.match(out, /discuss-milestone\s+M002-Y/);
|
|
assert.match(out, /failed/);
|
|
assert.equal(err, "");
|
|
});
|
|
|
|
test("renderLiveStatus_when_solver_state_exists_includes_solver_line", () => {
|
|
const project = makeProject();
|
|
const solverDir = join(project, ".sf/runtime/autonomous-solver");
|
|
mkdirSync(solverDir, { recursive: true });
|
|
writeFileSync(
|
|
join(solverDir, "active.json"),
|
|
`${JSON.stringify({
|
|
unitType: "execute-task",
|
|
unitId: "M001/S01/T01",
|
|
iteration: 3,
|
|
maxIterations: 12,
|
|
latestCheckpoint: {
|
|
outcome: "continue",
|
|
summary: "Implementation remains.",
|
|
remainingItems: ["Finish enforcement", "Run build"],
|
|
},
|
|
})}\n`,
|
|
);
|
|
|
|
const rendered = renderLiveStatus(snapshot() as any, {
|
|
basePath: project,
|
|
model: null,
|
|
recentEvents: [],
|
|
});
|
|
|
|
assert.match(rendered, /Solver:/);
|
|
assert.match(rendered, /execute-task M001\/S01\/T01/);
|
|
assert.match(rendered, /iter 3\/12/);
|
|
assert.match(rendered, /2 remaining/);
|
|
});
|