Expose adapter-discovered user-installed skills with provenance metadata, share persistent skill snapshot classification across local adapters, and render unmanaged skills as a read-only section in the agent skills UI. Co-Authored-By: Paperclip <noreply@paperclip.ing>
92 lines
3.4 KiB
TypeScript
92 lines
3.4 KiB
TypeScript
import fs from "node:fs/promises";
|
|
import os from "node:os";
|
|
import path from "node:path";
|
|
import { fileURLToPath } from "node:url";
|
|
import type {
|
|
AdapterSkillContext,
|
|
AdapterSkillSnapshot,
|
|
} from "@paperclipai/adapter-utils";
|
|
import {
|
|
buildPersistentSkillSnapshot,
|
|
ensurePaperclipSkillSymlink,
|
|
readPaperclipRuntimeSkillEntries,
|
|
readInstalledSkillTargets,
|
|
resolvePaperclipDesiredSkillNames,
|
|
} from "@paperclipai/adapter-utils/server-utils";
|
|
|
|
const __moduleDir = path.dirname(fileURLToPath(import.meta.url));
|
|
|
|
function asString(value: unknown): string | null {
|
|
return typeof value === "string" && value.trim().length > 0 ? value.trim() : null;
|
|
}
|
|
|
|
function resolvePiSkillsHome(config: Record<string, unknown>) {
|
|
const env =
|
|
typeof config.env === "object" && config.env !== null && !Array.isArray(config.env)
|
|
? (config.env as Record<string, unknown>)
|
|
: {};
|
|
const configuredHome = asString(env.HOME);
|
|
const home = configuredHome ? path.resolve(configuredHome) : os.homedir();
|
|
return path.join(home, ".pi", "agent", "skills");
|
|
}
|
|
|
|
async function buildPiSkillSnapshot(config: Record<string, unknown>): Promise<AdapterSkillSnapshot> {
|
|
const availableEntries = await readPaperclipRuntimeSkillEntries(config, __moduleDir);
|
|
const desiredSkills = resolvePaperclipDesiredSkillNames(config, availableEntries);
|
|
const skillsHome = resolvePiSkillsHome(config);
|
|
const installed = await readInstalledSkillTargets(skillsHome);
|
|
return buildPersistentSkillSnapshot({
|
|
adapterType: "pi_local",
|
|
availableEntries,
|
|
desiredSkills,
|
|
installed,
|
|
skillsHome,
|
|
locationLabel: "~/.pi/agent/skills",
|
|
missingDetail: "Configured but not currently linked into the Pi skills home.",
|
|
externalConflictDetail: "Skill name is occupied by an external installation.",
|
|
externalDetail: "Installed outside Paperclip management.",
|
|
});
|
|
}
|
|
|
|
export async function listPiSkills(ctx: AdapterSkillContext): Promise<AdapterSkillSnapshot> {
|
|
return buildPiSkillSnapshot(ctx.config);
|
|
}
|
|
|
|
export async function syncPiSkills(
|
|
ctx: AdapterSkillContext,
|
|
desiredSkills: string[],
|
|
): Promise<AdapterSkillSnapshot> {
|
|
const availableEntries = await readPaperclipRuntimeSkillEntries(ctx.config, __moduleDir);
|
|
const desiredSet = new Set([
|
|
...desiredSkills,
|
|
...availableEntries.filter((entry) => entry.required).map((entry) => entry.key),
|
|
]);
|
|
const skillsHome = resolvePiSkillsHome(ctx.config);
|
|
await fs.mkdir(skillsHome, { recursive: true });
|
|
const installed = await readInstalledSkillTargets(skillsHome);
|
|
const availableByRuntimeName = new Map(availableEntries.map((entry) => [entry.runtimeName, entry]));
|
|
|
|
for (const available of availableEntries) {
|
|
if (!desiredSet.has(available.key)) continue;
|
|
const target = path.join(skillsHome, available.runtimeName);
|
|
await ensurePaperclipSkillSymlink(available.source, target);
|
|
}
|
|
|
|
for (const [name, installedEntry] of installed.entries()) {
|
|
const available = availableByRuntimeName.get(name);
|
|
if (!available) continue;
|
|
if (desiredSet.has(available.key)) continue;
|
|
if (installedEntry.targetPath !== available.source) continue;
|
|
await fs.unlink(path.join(skillsHome, name)).catch(() => {});
|
|
}
|
|
|
|
return buildPiSkillSnapshot(ctx.config);
|
|
}
|
|
|
|
export function resolvePiDesiredSkillNames(
|
|
config: Record<string, unknown>,
|
|
availableEntries: Array<{ key: string; required?: boolean }>,
|
|
) {
|
|
return resolvePaperclipDesiredSkillNames(config, availableEntries);
|
|
}
|