mirror of
https://github.com/openclaw/openclaw.git
synced 2026-03-12 07:20:45 +00:00
288 lines
9.0 KiB
TypeScript
288 lines
9.0 KiB
TypeScript
import fs from "node:fs/promises";
|
|
import path from "node:path";
|
|
import { afterAll, beforeAll, describe, expect, it } from "vitest";
|
|
import { withEnv } from "../test-utils/env.js";
|
|
import { createFixtureSuite } from "../test-utils/fixture-suite.js";
|
|
import { writeSkill } from "./skills.e2e-test-helpers.js";
|
|
import { buildWorkspaceSkillSnapshot, buildWorkspaceSkillsPrompt } from "./skills.js";
|
|
|
|
const fixtureSuite = createFixtureSuite("openclaw-skills-snapshot-suite-");
|
|
|
|
beforeAll(async () => {
|
|
await fixtureSuite.setup();
|
|
});
|
|
|
|
afterAll(async () => {
|
|
await fixtureSuite.cleanup();
|
|
});
|
|
|
|
function withWorkspaceHome<T>(workspaceDir: string, cb: () => T): T {
|
|
return withEnv({ HOME: workspaceDir, PATH: "" }, cb);
|
|
}
|
|
|
|
describe("buildWorkspaceSkillSnapshot", () => {
|
|
it("returns an empty snapshot when skills dirs are missing", async () => {
|
|
const workspaceDir = await fixtureSuite.createCaseDir("workspace");
|
|
|
|
const snapshot = withWorkspaceHome(workspaceDir, () =>
|
|
buildWorkspaceSkillSnapshot(workspaceDir, {
|
|
managedSkillsDir: path.join(workspaceDir, ".managed"),
|
|
bundledSkillsDir: path.join(workspaceDir, ".bundled"),
|
|
}),
|
|
);
|
|
|
|
expect(snapshot.prompt).toBe("");
|
|
expect(snapshot.skills).toEqual([]);
|
|
});
|
|
|
|
it("omits disable-model-invocation skills from the prompt", async () => {
|
|
const workspaceDir = await fixtureSuite.createCaseDir("workspace");
|
|
await writeSkill({
|
|
dir: path.join(workspaceDir, "skills", "visible-skill"),
|
|
name: "visible-skill",
|
|
description: "Visible skill",
|
|
});
|
|
await writeSkill({
|
|
dir: path.join(workspaceDir, "skills", "hidden-skill"),
|
|
name: "hidden-skill",
|
|
description: "Hidden skill",
|
|
frontmatterExtra: "disable-model-invocation: true",
|
|
});
|
|
|
|
const snapshot = withWorkspaceHome(workspaceDir, () =>
|
|
buildWorkspaceSkillSnapshot(workspaceDir, {
|
|
managedSkillsDir: path.join(workspaceDir, ".managed"),
|
|
bundledSkillsDir: path.join(workspaceDir, ".bundled"),
|
|
}),
|
|
);
|
|
|
|
expect(snapshot.prompt).toContain("visible-skill");
|
|
expect(snapshot.prompt).not.toContain("hidden-skill");
|
|
expect(snapshot.skills.map((skill) => skill.name).toSorted()).toEqual([
|
|
"hidden-skill",
|
|
"visible-skill",
|
|
]);
|
|
});
|
|
|
|
it("keeps prompt output aligned with buildWorkspaceSkillsPrompt", async () => {
|
|
const workspaceDir = await fixtureSuite.createCaseDir("workspace");
|
|
await writeSkill({
|
|
dir: path.join(workspaceDir, "skills", "visible"),
|
|
name: "visible",
|
|
description: "Visible",
|
|
});
|
|
await writeSkill({
|
|
dir: path.join(workspaceDir, "skills", "hidden"),
|
|
name: "hidden",
|
|
description: "Hidden",
|
|
frontmatterExtra: "disable-model-invocation: true",
|
|
});
|
|
const config = {
|
|
skills: {
|
|
limits: {
|
|
maxSkillsInPrompt: 1,
|
|
maxSkillsPromptChars: 200,
|
|
},
|
|
},
|
|
} as const;
|
|
const opts = {
|
|
config,
|
|
managedSkillsDir: path.join(workspaceDir, ".managed"),
|
|
bundledSkillsDir: path.join(workspaceDir, ".bundled"),
|
|
eligibility: {
|
|
remote: {
|
|
platforms: ["linux"],
|
|
hasBin: (_bin: string) => true,
|
|
hasAnyBin: (_bins: string[]) => true,
|
|
note: "Remote note",
|
|
},
|
|
},
|
|
};
|
|
|
|
const snapshot = withWorkspaceHome(workspaceDir, () =>
|
|
buildWorkspaceSkillSnapshot(workspaceDir, opts),
|
|
);
|
|
const prompt = withWorkspaceHome(workspaceDir, () =>
|
|
buildWorkspaceSkillsPrompt(workspaceDir, opts),
|
|
);
|
|
|
|
expect(snapshot.prompt).toBe(prompt);
|
|
});
|
|
|
|
it("truncates the skills prompt when it exceeds the configured char budget", async () => {
|
|
const workspaceDir = await fixtureSuite.createCaseDir("workspace");
|
|
|
|
// Keep fixture size modest while still forcing truncation logic.
|
|
for (let i = 0; i < 8; i += 1) {
|
|
const name = `skill-${String(i).padStart(2, "0")}`;
|
|
await writeSkill({
|
|
dir: path.join(workspaceDir, "skills", name),
|
|
name,
|
|
description: "x".repeat(800),
|
|
});
|
|
}
|
|
|
|
const snapshot = withWorkspaceHome(workspaceDir, () =>
|
|
buildWorkspaceSkillSnapshot(workspaceDir, {
|
|
config: {
|
|
skills: {
|
|
limits: {
|
|
maxSkillsInPrompt: 100,
|
|
maxSkillsPromptChars: 500,
|
|
},
|
|
},
|
|
},
|
|
managedSkillsDir: path.join(workspaceDir, ".managed"),
|
|
bundledSkillsDir: path.join(workspaceDir, ".bundled"),
|
|
}),
|
|
);
|
|
|
|
expect(snapshot.prompt).toContain("⚠️ Skills truncated");
|
|
expect(snapshot.prompt.length).toBeLessThan(2000);
|
|
});
|
|
|
|
it("limits discovery for nested repo-style skills roots (dir/skills/*)", async () => {
|
|
const workspaceDir = await fixtureSuite.createCaseDir("workspace");
|
|
const repoDir = await fixtureSuite.createCaseDir("skills-repo");
|
|
|
|
for (let i = 0; i < 8; i += 1) {
|
|
const name = `repo-skill-${String(i).padStart(2, "0")}`;
|
|
await writeSkill({
|
|
dir: path.join(repoDir, "skills", name),
|
|
name,
|
|
description: `Desc ${i}`,
|
|
});
|
|
}
|
|
|
|
const snapshot = withWorkspaceHome(workspaceDir, () =>
|
|
buildWorkspaceSkillSnapshot(workspaceDir, {
|
|
config: {
|
|
skills: {
|
|
load: {
|
|
extraDirs: [repoDir],
|
|
},
|
|
limits: {
|
|
maxCandidatesPerRoot: 5,
|
|
maxSkillsLoadedPerSource: 5,
|
|
},
|
|
},
|
|
},
|
|
managedSkillsDir: path.join(workspaceDir, ".managed"),
|
|
bundledSkillsDir: path.join(workspaceDir, ".bundled"),
|
|
}),
|
|
);
|
|
|
|
// We should only have loaded a small subset.
|
|
expect(snapshot.skills.length).toBeLessThanOrEqual(5);
|
|
expect(snapshot.prompt).toContain("repo-skill-00");
|
|
expect(snapshot.prompt).not.toContain("repo-skill-07");
|
|
});
|
|
|
|
it("skips skills whose SKILL.md exceeds maxSkillFileBytes", async () => {
|
|
const workspaceDir = await fixtureSuite.createCaseDir("workspace");
|
|
|
|
await writeSkill({
|
|
dir: path.join(workspaceDir, "skills", "small-skill"),
|
|
name: "small-skill",
|
|
description: "Small",
|
|
});
|
|
|
|
await writeSkill({
|
|
dir: path.join(workspaceDir, "skills", "big-skill"),
|
|
name: "big-skill",
|
|
description: "Big",
|
|
body: "x".repeat(5_000),
|
|
});
|
|
|
|
const snapshot = withWorkspaceHome(workspaceDir, () =>
|
|
buildWorkspaceSkillSnapshot(workspaceDir, {
|
|
config: {
|
|
skills: {
|
|
limits: {
|
|
maxSkillFileBytes: 1000,
|
|
},
|
|
},
|
|
},
|
|
managedSkillsDir: path.join(workspaceDir, ".managed"),
|
|
bundledSkillsDir: path.join(workspaceDir, ".bundled"),
|
|
}),
|
|
);
|
|
|
|
expect(snapshot.skills.map((s) => s.name)).toContain("small-skill");
|
|
expect(snapshot.skills.map((s) => s.name)).not.toContain("big-skill");
|
|
expect(snapshot.prompt).toContain("small-skill");
|
|
expect(snapshot.prompt).not.toContain("big-skill");
|
|
});
|
|
|
|
it("detects nested skills roots beyond the first 25 entries", async () => {
|
|
const workspaceDir = await fixtureSuite.createCaseDir("workspace");
|
|
const repoDir = await fixtureSuite.createCaseDir("skills-repo");
|
|
|
|
// Create 30 nested dirs, but only the last one is an actual skill.
|
|
for (let i = 0; i < 30; i += 1) {
|
|
await fs.mkdir(path.join(repoDir, "skills", `entry-${String(i).padStart(2, "0")}`), {
|
|
recursive: true,
|
|
});
|
|
}
|
|
|
|
await writeSkill({
|
|
dir: path.join(repoDir, "skills", "entry-29"),
|
|
name: "late-skill",
|
|
description: "Nested skill discovered late",
|
|
});
|
|
|
|
const snapshot = withWorkspaceHome(workspaceDir, () =>
|
|
buildWorkspaceSkillSnapshot(workspaceDir, {
|
|
config: {
|
|
skills: {
|
|
load: {
|
|
extraDirs: [repoDir],
|
|
},
|
|
limits: {
|
|
maxCandidatesPerRoot: 30,
|
|
maxSkillsLoadedPerSource: 30,
|
|
},
|
|
},
|
|
},
|
|
managedSkillsDir: path.join(workspaceDir, ".managed"),
|
|
bundledSkillsDir: path.join(workspaceDir, ".bundled"),
|
|
}),
|
|
);
|
|
|
|
expect(snapshot.skills.map((s) => s.name)).toContain("late-skill");
|
|
expect(snapshot.prompt).toContain("late-skill");
|
|
});
|
|
|
|
it("enforces maxSkillFileBytes for root-level SKILL.md", async () => {
|
|
const workspaceDir = await fixtureSuite.createCaseDir("workspace");
|
|
const rootSkillDir = await fixtureSuite.createCaseDir("root-skill");
|
|
|
|
await writeSkill({
|
|
dir: rootSkillDir,
|
|
name: "root-big-skill",
|
|
description: "Big",
|
|
body: "x".repeat(5_000),
|
|
});
|
|
|
|
const snapshot = withWorkspaceHome(workspaceDir, () =>
|
|
buildWorkspaceSkillSnapshot(workspaceDir, {
|
|
config: {
|
|
skills: {
|
|
load: {
|
|
extraDirs: [rootSkillDir],
|
|
},
|
|
limits: {
|
|
maxSkillFileBytes: 1000,
|
|
},
|
|
},
|
|
},
|
|
managedSkillsDir: path.join(workspaceDir, ".managed"),
|
|
bundledSkillsDir: path.join(workspaceDir, ".bundled"),
|
|
}),
|
|
);
|
|
|
|
expect(snapshot.skills.map((s) => s.name)).not.toContain("root-big-skill");
|
|
expect(snapshot.prompt).not.toContain("root-big-skill");
|
|
});
|
|
});
|