mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-01 08:50:23 +00:00
211 lines
6.2 KiB
TypeScript
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([]);
|
|
}
|
|
});
|
|
});
|