Files
openclaw/src/agents/skills.buildworkspaceskillstatus.test.ts
2026-03-28 03:45:56 +00:00

211 lines
6.2 KiB
TypeScript

import fs from "node:fs/promises";
import os from "node:os";
import path from "node:path";
import { afterEach, describe, expect, it } from "vitest";
import { withEnv, withEnvAsync } from "../test-utils/env.js";
import { buildWorkspaceSkillStatus } from "./skills-status.js";
import { writeSkill } from "./skills.e2e-test-helpers.js";
import { createCanonicalFixtureSkill } from "./skills.test-helpers.js";
import type { SkillEntry } from "./skills/types.js";
const tempDirs: string[] = [];
afterEach(async () => {
await Promise.all(
tempDirs.splice(0, tempDirs.length).map((dir) => fs.rm(dir, { recursive: true, force: true })),
);
});
function makeEntry(params: {
name: string;
source?: string;
os?: string[];
requires?: { bins?: string[]; env?: string[]; config?: string[] };
install?: Array<{
id: string;
kind: "brew" | "download";
bins?: string[];
formula?: string;
os?: string[];
url?: string;
label?: string;
}>;
}): SkillEntry {
const filePath = `/tmp/${params.name}/SKILL.md`;
const baseDir = `/tmp/${params.name}`;
return {
skill: createFixtureSkill({
name: params.name,
description: `desc:${params.name}`,
filePath,
baseDir,
source: params.source ?? "openclaw-workspace",
}),
frontmatter: {},
metadata: {
...(params.os ? { os: params.os } : {}),
...(params.requires ? { requires: params.requires } : {}),
...(params.install ? { install: params.install } : {}),
...(params.requires?.env?.[0] ? { primaryEnv: params.requires.env[0] } : {}),
},
};
}
function createFixtureSkill(params: {
name: string;
description: string;
filePath: string;
baseDir: string;
source: string;
}): SkillEntry["skill"] {
return createCanonicalFixtureSkill(params);
}
describe("buildWorkspaceSkillStatus", () => {
it("reports missing requirements and install options", async () => {
const entry = makeEntry({
name: "status-skill",
requires: {
bins: ["fakebin"],
env: ["ENV_KEY"],
config: ["browser.enabled"],
},
install: [
{
id: "brew",
kind: "brew",
formula: "fakebin",
bins: ["fakebin"],
label: "Install fakebin",
},
],
});
const report = withEnv({ PATH: "" }, () =>
buildWorkspaceSkillStatus("/tmp/ws", {
entries: [entry],
config: { browser: { enabled: false } },
}),
);
const skill = report.skills.find((entry) => entry.name === "status-skill");
expect(skill).toBeDefined();
expect(skill?.eligible).toBe(false);
expect(skill?.missing.bins).toContain("fakebin");
expect(skill?.missing.env).toContain("ENV_KEY");
expect(skill?.missing.config).toContain("browser.enabled");
expect(skill?.install[0]?.id).toBe("brew");
});
it("respects OS-gated skills", async () => {
const entry = makeEntry({
name: "os-skill",
os: ["darwin"],
});
const report = buildWorkspaceSkillStatus("/tmp/ws", { entries: [entry] });
const skill = report.skills.find((entry) => entry.name === "os-skill");
expect(skill).toBeDefined();
if (process.platform === "darwin") {
expect(skill?.eligible).toBe(true);
expect(skill?.missing.os).toEqual([]);
} else {
expect(skill?.eligible).toBe(false);
expect(skill?.missing.os).toEqual(["darwin"]);
}
});
it("marks bundled skills blocked by allowlist", async () => {
const entry = makeEntry({
name: "peekaboo",
source: "openclaw-bundled",
});
const report = buildWorkspaceSkillStatus("/tmp/ws", {
entries: [entry],
config: { skills: { allowBundled: ["other-skill"] } },
});
const skill = report.skills.find((reportEntry) => reportEntry.name === "peekaboo");
expect(skill).toBeDefined();
expect(skill?.blockedByAllowlist).toBe(true);
expect(skill?.eligible).toBe(false);
expect(skill?.bundled).toBe(true);
});
it("does not mark an overridden workspace skill as bundled by bundled name alone", async () => {
const bundledDir = await fs.mkdtemp(path.join(os.tmpdir(), "openclaw-bundled-"));
tempDirs.push(bundledDir);
await writeSkill({
dir: path.join(bundledDir, "peekaboo"),
name: "peekaboo",
description: "Bundled peekaboo",
});
await withEnvAsync({ OPENCLAW_BUNDLED_SKILLS_DIR: bundledDir }, async () => {
const report = buildWorkspaceSkillStatus("/tmp/ws", {
entries: [
makeEntry({
name: "peekaboo",
source: "openclaw-workspace",
}),
],
config: { skills: { allowBundled: ["other-skill"] } },
});
const skill = report.skills.find((reportEntry) => reportEntry.name === "peekaboo");
expect(skill).toBeDefined();
expect(skill?.source).toBe("openclaw-workspace");
expect(skill?.bundled).toBe(false);
expect(skill?.blockedByAllowlist).toBe(false);
expect(skill?.eligible).toBe(true);
});
});
it("filters install options by OS", async () => {
const entry = makeEntry({
name: "install-skill",
requires: {
bins: ["missing-bin"],
},
install: [
{
id: "mac",
kind: "download",
os: ["darwin"],
url: "https://example.com/mac.tar.bz2",
},
{
id: "linux",
kind: "download",
os: ["linux"],
url: "https://example.com/linux.tar.bz2",
},
{
id: "win",
kind: "download",
os: ["win32"],
url: "https://example.com/win.tar.bz2",
},
],
});
const report = withEnv({ PATH: "" }, () =>
buildWorkspaceSkillStatus("/tmp/ws", {
entries: [entry],
}),
);
const skill = report.skills.find((reportEntry) => reportEntry.name === "install-skill");
expect(skill).toBeDefined();
if (process.platform === "darwin") {
expect(skill?.install.map((opt) => opt.id)).toEqual(["mac"]);
} else if (process.platform === "linux") {
expect(skill?.install.map((opt) => opt.id)).toEqual(["linux"]);
} else if (process.platform === "win32") {
expect(skill?.install.map((opt) => opt.id)).toEqual(["win"]);
} else {
expect(skill?.install).toEqual([]);
}
});
});