import { Command } from "commander"; import { beforeEach, describe, expect, it, vi } from "vitest"; import { createCliRuntimeCapture } from "./test-runtime-capture.js"; const callGatewayFromCli = vi.fn(async (method: string, _opts: unknown, params?: unknown) => { if (method.endsWith(".get")) { return { path: "/tmp/exec-approvals.json", exists: true, hash: "hash-1", file: { version: 1, agents: {} }, }; } return { method, params }; }); const { runtimeErrors, defaultRuntime, resetRuntimeCapture } = createCliRuntimeCapture(); const localSnapshot = { path: "/tmp/local-exec-approvals.json", exists: true, raw: "{}", hash: "hash-local", file: { version: 1, agents: {} }, }; function resetLocalSnapshot() { localSnapshot.file = { version: 1, agents: {} }; } vi.mock("./gateway-rpc.js", () => ({ callGatewayFromCli: (method: string, opts: unknown, params?: unknown) => callGatewayFromCli(method, opts, params), })); vi.mock("./nodes-cli/rpc.js", async () => { const actual = await vi.importActual("./nodes-cli/rpc.js"); return { ...actual, resolveNodeId: vi.fn(async () => "node-1"), }; }); vi.mock("../runtime.js", () => ({ defaultRuntime, })); vi.mock("../infra/exec-approvals.js", async () => { const actual = await vi.importActual( "../infra/exec-approvals.js", ); return { ...actual, readExecApprovalsSnapshot: () => localSnapshot, saveExecApprovals: vi.fn(), }; }); const { registerExecApprovalsCli } = await import("./exec-approvals-cli.js"); const execApprovals = await import("../infra/exec-approvals.js"); describe("exec approvals CLI", () => { const createProgram = () => { const program = new Command(); program.exitOverride(); registerExecApprovalsCli(program); return program; }; const runApprovalsCommand = async (args: string[]) => { const program = createProgram(); await program.parseAsync(args, { from: "user" }); }; beforeEach(() => { resetLocalSnapshot(); resetRuntimeCapture(); callGatewayFromCli.mockClear(); }); it("routes get command to local, gateway, and node modes", async () => { await runApprovalsCommand(["approvals", "get"]); expect(callGatewayFromCli).not.toHaveBeenCalled(); expect(runtimeErrors).toHaveLength(0); callGatewayFromCli.mockClear(); await runApprovalsCommand(["approvals", "get", "--gateway"]); expect(callGatewayFromCli).toHaveBeenCalledWith("exec.approvals.get", expect.anything(), {}); expect(runtimeErrors).toHaveLength(0); callGatewayFromCli.mockClear(); await runApprovalsCommand(["approvals", "get", "--node", "macbook"]); expect(callGatewayFromCli).toHaveBeenCalledWith("exec.approvals.node.get", expect.anything(), { nodeId: "node-1", }); expect(runtimeErrors).toHaveLength(0); }); it("defaults allowlist add to wildcard agent", async () => { const saveExecApprovals = vi.mocked(execApprovals.saveExecApprovals); saveExecApprovals.mockClear(); await runApprovalsCommand(["approvals", "allowlist", "add", "/usr/bin/uname"]); expect(callGatewayFromCli).not.toHaveBeenCalledWith( "exec.approvals.set", expect.anything(), {}, ); expect(saveExecApprovals).toHaveBeenCalledWith( expect.objectContaining({ agents: expect.objectContaining({ "*": expect.anything(), }), }), ); }); it("removes wildcard allowlist entry and prunes empty agent", async () => { localSnapshot.file = { version: 1, agents: { "*": { allowlist: [{ pattern: "/usr/bin/uname", lastUsedAt: Date.now() }], }, }, }; const saveExecApprovals = vi.mocked(execApprovals.saveExecApprovals); saveExecApprovals.mockClear(); await runApprovalsCommand(["approvals", "allowlist", "remove", "/usr/bin/uname"]); expect(saveExecApprovals).toHaveBeenCalledWith( expect.objectContaining({ version: 1, agents: undefined, }), ); expect(runtimeErrors).toHaveLength(0); }); });