186 lines
5.6 KiB
TypeScript
186 lines
5.6 KiB
TypeScript
import { describe, expect, it, vi } from "vitest";
|
|
import type { OpenClawConfig } from "../config/config.js";
|
|
import type { RuntimeEnv } from "../runtime.js";
|
|
import type { WizardPrompter } from "../wizard/prompts.js";
|
|
|
|
// Module under test imports these at module scope.
|
|
vi.mock("../agents/skills-status.js", () => ({
|
|
buildWorkspaceSkillStatus: vi.fn(),
|
|
}));
|
|
vi.mock("../agents/skills-install.js", () => ({
|
|
installSkill: vi.fn(),
|
|
}));
|
|
vi.mock("./onboard-helpers.js", () => ({
|
|
detectBinary: vi.fn(),
|
|
resolveNodeManagerOptions: vi.fn(() => [
|
|
{ value: "npm", label: "npm" },
|
|
{ value: "pnpm", label: "pnpm" },
|
|
{ value: "bun", label: "bun" },
|
|
]),
|
|
}));
|
|
|
|
import { installSkill } from "../agents/skills-install.js";
|
|
import { buildWorkspaceSkillStatus } from "../agents/skills-status.js";
|
|
import { detectBinary } from "./onboard-helpers.js";
|
|
import { setupSkills } from "./onboard-skills.js";
|
|
|
|
function createBundledSkill(params: {
|
|
name: string;
|
|
description: string;
|
|
bins: string[];
|
|
os?: string[];
|
|
installLabel: string;
|
|
}): {
|
|
name: string;
|
|
description: string;
|
|
source: string;
|
|
bundled: boolean;
|
|
filePath: string;
|
|
baseDir: string;
|
|
skillKey: string;
|
|
always: boolean;
|
|
disabled: boolean;
|
|
blockedByAllowlist: boolean;
|
|
eligible: boolean;
|
|
requirements: {
|
|
bins: string[];
|
|
anyBins: string[];
|
|
env: string[];
|
|
config: string[];
|
|
os: string[];
|
|
};
|
|
missing: { bins: string[]; anyBins: string[]; env: string[]; config: string[]; os: string[] };
|
|
configChecks: [];
|
|
install: Array<{ id: string; kind: string; label: string; bins: string[] }>;
|
|
} {
|
|
return {
|
|
name: params.name,
|
|
description: params.description,
|
|
source: "openclaw-bundled",
|
|
bundled: true,
|
|
filePath: `/tmp/skills/${params.name}`,
|
|
baseDir: `/tmp/skills/${params.name}`,
|
|
skillKey: params.name,
|
|
always: false,
|
|
disabled: false,
|
|
blockedByAllowlist: false,
|
|
eligible: false,
|
|
requirements: { bins: params.bins, anyBins: [], env: [], config: [], os: params.os ?? [] },
|
|
missing: { bins: params.bins, anyBins: [], env: [], config: [], os: params.os ?? [] },
|
|
configChecks: [],
|
|
install: [{ id: "brew", kind: "brew", label: params.installLabel, bins: params.bins }],
|
|
};
|
|
}
|
|
|
|
function mockMissingBrewStatus(skills: Array<ReturnType<typeof createBundledSkill>>): void {
|
|
vi.mocked(detectBinary).mockResolvedValue(false);
|
|
vi.mocked(installSkill).mockResolvedValue({
|
|
ok: true,
|
|
message: "Installed",
|
|
stdout: "",
|
|
stderr: "",
|
|
code: 0,
|
|
});
|
|
vi.mocked(buildWorkspaceSkillStatus).mockReturnValue({
|
|
workspaceDir: "/tmp/ws",
|
|
managedSkillsDir: "/tmp/managed",
|
|
skills,
|
|
} as never);
|
|
}
|
|
|
|
function createPrompter(params: {
|
|
configure?: boolean;
|
|
showBrewInstall?: boolean;
|
|
multiselect?: string[];
|
|
}): { prompter: WizardPrompter; notes: Array<{ title?: string; message: string }> } {
|
|
const notes: Array<{ title?: string; message: string }> = [];
|
|
|
|
const confirmAnswers: boolean[] = [];
|
|
confirmAnswers.push(params.configure ?? true);
|
|
|
|
const prompter: WizardPrompter = {
|
|
intro: vi.fn(async () => {}),
|
|
outro: vi.fn(async () => {}),
|
|
note: vi.fn(async (message: string, title?: string) => {
|
|
notes.push({ title, message });
|
|
}),
|
|
select: vi.fn(async () => "npm") as unknown as WizardPrompter["select"],
|
|
multiselect: vi.fn(
|
|
async () => params.multiselect ?? ["__skip__"],
|
|
) as unknown as WizardPrompter["multiselect"],
|
|
text: vi.fn(async () => ""),
|
|
confirm: vi.fn(async ({ message }) => {
|
|
if (message === "Show Homebrew install command?") {
|
|
return params.showBrewInstall ?? false;
|
|
}
|
|
return confirmAnswers.shift() ?? false;
|
|
}),
|
|
progress: vi.fn(() => ({ update: vi.fn(), stop: vi.fn() })),
|
|
};
|
|
|
|
return { prompter, notes };
|
|
}
|
|
|
|
const runtime: RuntimeEnv = {
|
|
log: vi.fn(),
|
|
error: vi.fn(),
|
|
exit: ((code: number) => {
|
|
throw new Error(`unexpected exit ${code}`);
|
|
}) as RuntimeEnv["exit"],
|
|
};
|
|
|
|
describe("setupSkills", () => {
|
|
it("does not recommend Homebrew when user skips installing brew-backed deps", async () => {
|
|
if (process.platform === "win32") {
|
|
return;
|
|
}
|
|
|
|
mockMissingBrewStatus([
|
|
createBundledSkill({
|
|
name: "apple-reminders",
|
|
description: "macOS-only",
|
|
bins: ["remindctl"],
|
|
os: ["darwin"],
|
|
installLabel: "Install remindctl (brew)",
|
|
}),
|
|
createBundledSkill({
|
|
name: "video-frames",
|
|
description: "ffmpeg",
|
|
bins: ["ffmpeg"],
|
|
installLabel: "Install ffmpeg (brew)",
|
|
}),
|
|
]);
|
|
|
|
const { prompter, notes } = createPrompter({ multiselect: ["__skip__"] });
|
|
await setupSkills({} as OpenClawConfig, "/tmp/ws", runtime, prompter);
|
|
|
|
// OS-mismatched skill should be counted as unsupported, not installable/missing.
|
|
const status = notes.find((n) => n.title === "Skills status")?.message ?? "";
|
|
expect(status).toContain("Unsupported on this OS: 1");
|
|
|
|
const brewNote = notes.find((n) => n.title === "Homebrew recommended");
|
|
expect(brewNote).toBeUndefined();
|
|
});
|
|
|
|
it("recommends Homebrew when user selects a brew-backed install and brew is missing", async () => {
|
|
if (process.platform === "win32") {
|
|
return;
|
|
}
|
|
|
|
mockMissingBrewStatus([
|
|
createBundledSkill({
|
|
name: "video-frames",
|
|
description: "ffmpeg",
|
|
bins: ["ffmpeg"],
|
|
installLabel: "Install ffmpeg (brew)",
|
|
}),
|
|
]);
|
|
|
|
const { prompter, notes } = createPrompter({ multiselect: ["video-frames"] });
|
|
await setupSkills({} as OpenClawConfig, "/tmp/ws", runtime, prompter);
|
|
|
|
const brewNote = notes.find((n) => n.title === "Homebrew recommended");
|
|
expect(brewNote).toBeDefined();
|
|
});
|
|
});
|