Files
openclaw/src/agents/skills.loadworkspaceskillentries.test.ts
Eugene 5341b5c71c Diffs: Migrate tool usage guidance from before_prompt_build to a plugin skill (#32630)
Merged via squash.

Prepared head SHA: 585697a4e1556baa2cd79a7b449b120c4fd87e17
Co-authored-by: sircrumpet <4436535+sircrumpet@users.noreply.github.com>
Co-authored-by: gumadeiras <5599352+gumadeiras@users.noreply.github.com>
Reviewed-by: @gumadeiras
2026-03-03 01:50:59 -05:00

159 lines
4.9 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 { loadWorkspaceSkillEntries } from "./skills.js";
const tempDirs: string[] = [];
async function createTempWorkspaceDir() {
const workspaceDir = await fs.mkdtemp(path.join(os.tmpdir(), "openclaw-"));
tempDirs.push(workspaceDir);
return workspaceDir;
}
afterEach(async () => {
await Promise.all(
tempDirs.splice(0, tempDirs.length).map((dir) => fs.rm(dir, { recursive: true, force: true })),
);
});
async function setupWorkspaceWithProsePlugin() {
const workspaceDir = await createTempWorkspaceDir();
const managedDir = path.join(workspaceDir, ".managed");
const bundledDir = path.join(workspaceDir, ".bundled");
const pluginRoot = path.join(workspaceDir, ".openclaw", "extensions", "open-prose");
await fs.mkdir(path.join(pluginRoot, "skills", "prose"), { recursive: true });
await fs.writeFile(
path.join(pluginRoot, "openclaw.plugin.json"),
JSON.stringify(
{
id: "open-prose",
skills: ["./skills"],
configSchema: { type: "object", additionalProperties: false, properties: {} },
},
null,
2,
),
"utf-8",
);
await fs.writeFile(path.join(pluginRoot, "index.ts"), "export {};\n", "utf-8");
await fs.writeFile(
path.join(pluginRoot, "skills", "prose", "SKILL.md"),
`---\nname: prose\ndescription: test\n---\n`,
"utf-8",
);
return { workspaceDir, managedDir, bundledDir };
}
async function setupWorkspaceWithDiffsPlugin() {
const workspaceDir = await createTempWorkspaceDir();
const managedDir = path.join(workspaceDir, ".managed");
const bundledDir = path.join(workspaceDir, ".bundled");
const pluginRoot = path.join(workspaceDir, ".openclaw", "extensions", "diffs");
await fs.mkdir(path.join(pluginRoot, "skills", "diffs"), { recursive: true });
await fs.writeFile(
path.join(pluginRoot, "openclaw.plugin.json"),
JSON.stringify(
{
id: "diffs",
skills: ["./skills"],
configSchema: { type: "object", additionalProperties: false, properties: {} },
},
null,
2,
),
"utf-8",
);
await fs.writeFile(path.join(pluginRoot, "index.ts"), "export {};\n", "utf-8");
await fs.writeFile(
path.join(pluginRoot, "skills", "diffs", "SKILL.md"),
`---\nname: diffs\ndescription: test\n---\n`,
"utf-8",
);
return { workspaceDir, managedDir, bundledDir };
}
describe("loadWorkspaceSkillEntries", () => {
it("handles an empty managed skills dir without throwing", async () => {
const workspaceDir = await createTempWorkspaceDir();
const managedDir = path.join(workspaceDir, ".managed");
await fs.mkdir(managedDir, { recursive: true });
const entries = loadWorkspaceSkillEntries(workspaceDir, {
managedSkillsDir: managedDir,
bundledSkillsDir: path.join(workspaceDir, ".bundled"),
});
expect(entries).toEqual([]);
});
it("includes plugin-shipped skills when the plugin is enabled", async () => {
const { workspaceDir, managedDir, bundledDir } = await setupWorkspaceWithProsePlugin();
const entries = loadWorkspaceSkillEntries(workspaceDir, {
config: {
plugins: {
entries: { "open-prose": { enabled: true } },
},
},
managedSkillsDir: managedDir,
bundledSkillsDir: bundledDir,
});
expect(entries.map((entry) => entry.skill.name)).toContain("prose");
});
it("excludes plugin-shipped skills when the plugin is not allowed", async () => {
const { workspaceDir, managedDir, bundledDir } = await setupWorkspaceWithProsePlugin();
const entries = loadWorkspaceSkillEntries(workspaceDir, {
config: {
plugins: {
allow: ["something-else"],
},
},
managedSkillsDir: managedDir,
bundledSkillsDir: bundledDir,
});
expect(entries.map((entry) => entry.skill.name)).not.toContain("prose");
});
it("includes diffs plugin skill when the plugin is enabled", async () => {
const { workspaceDir, managedDir, bundledDir } = await setupWorkspaceWithDiffsPlugin();
const entries = loadWorkspaceSkillEntries(workspaceDir, {
config: {
plugins: {
entries: { diffs: { enabled: true } },
},
},
managedSkillsDir: managedDir,
bundledSkillsDir: bundledDir,
});
expect(entries.map((entry) => entry.skill.name)).toContain("diffs");
});
it("excludes diffs plugin skill when the plugin is disabled", async () => {
const { workspaceDir, managedDir, bundledDir } = await setupWorkspaceWithDiffsPlugin();
const entries = loadWorkspaceSkillEntries(workspaceDir, {
config: {
plugins: {
entries: { diffs: { enabled: false } },
},
},
managedSkillsDir: managedDir,
bundledSkillsDir: bundledDir,
});
expect(entries.map((entry) => entry.skill.name)).not.toContain("diffs");
});
});