mirror of
https://github.com/getcompanion-ai/co-mono.git
synced 2026-04-16 03:01:56 +00:00
437 lines
12 KiB
TypeScript
437 lines
12 KiB
TypeScript
import { existsSync, readdirSync, readFileSync, realpathSync, statSync } from "fs";
|
|
import { minimatch } from "minimatch";
|
|
import { homedir } from "os";
|
|
import { basename, dirname, join, resolve } from "path";
|
|
import { CONFIG_DIR_NAME, getAgentDir } from "../config.js";
|
|
import type { SkillsSettings } from "./settings-manager.js";
|
|
|
|
/**
|
|
* Standard frontmatter fields per Agent Skills spec.
|
|
* See: https://agentskills.io/specification#frontmatter-required
|
|
*/
|
|
const ALLOWED_FRONTMATTER_FIELDS = new Set([
|
|
"name",
|
|
"description",
|
|
"license",
|
|
"compatibility",
|
|
"metadata",
|
|
"allowed-tools",
|
|
]);
|
|
|
|
/** Max name length per spec */
|
|
const MAX_NAME_LENGTH = 64;
|
|
|
|
/** Max description length per spec */
|
|
const MAX_DESCRIPTION_LENGTH = 1024;
|
|
|
|
export interface SkillFrontmatter {
|
|
name?: string;
|
|
description?: string;
|
|
[key: string]: unknown;
|
|
}
|
|
|
|
export interface Skill {
|
|
name: string;
|
|
description: string;
|
|
filePath: string;
|
|
baseDir: string;
|
|
source: string;
|
|
}
|
|
|
|
export interface SkillWarning {
|
|
skillPath: string;
|
|
message: string;
|
|
}
|
|
|
|
export interface LoadSkillsResult {
|
|
skills: Skill[];
|
|
warnings: SkillWarning[];
|
|
}
|
|
|
|
type SkillFormat = "recursive" | "claude";
|
|
|
|
function stripQuotes(value: string): string {
|
|
if ((value.startsWith('"') && value.endsWith('"')) || (value.startsWith("'") && value.endsWith("'"))) {
|
|
return value.slice(1, -1);
|
|
}
|
|
return value;
|
|
}
|
|
|
|
function parseFrontmatter(content: string): { frontmatter: SkillFrontmatter; body: string; allKeys: string[] } {
|
|
const frontmatter: SkillFrontmatter = {};
|
|
const allKeys: string[] = [];
|
|
|
|
const normalizedContent = content.replace(/\r\n/g, "\n").replace(/\r/g, "\n");
|
|
|
|
if (!normalizedContent.startsWith("---")) {
|
|
return { frontmatter, body: normalizedContent, allKeys };
|
|
}
|
|
|
|
const endIndex = normalizedContent.indexOf("\n---", 3);
|
|
if (endIndex === -1) {
|
|
return { frontmatter, body: normalizedContent, allKeys };
|
|
}
|
|
|
|
const frontmatterBlock = normalizedContent.slice(4, endIndex);
|
|
const body = normalizedContent.slice(endIndex + 4).trim();
|
|
|
|
for (const line of frontmatterBlock.split("\n")) {
|
|
const match = line.match(/^(\w[\w-]*):\s*(.*)$/);
|
|
if (match) {
|
|
const key = match[1];
|
|
const value = stripQuotes(match[2].trim());
|
|
allKeys.push(key);
|
|
if (key === "name") {
|
|
frontmatter.name = value;
|
|
} else if (key === "description") {
|
|
frontmatter.description = value;
|
|
}
|
|
}
|
|
}
|
|
|
|
return { frontmatter, body, allKeys };
|
|
}
|
|
|
|
/**
|
|
* Validate skill name per Agent Skills spec.
|
|
* Returns array of validation error messages (empty if valid).
|
|
*/
|
|
function validateName(name: string, parentDirName: string): string[] {
|
|
const errors: string[] = [];
|
|
|
|
if (name !== parentDirName) {
|
|
errors.push(`name "${name}" does not match parent directory "${parentDirName}"`);
|
|
}
|
|
|
|
if (name.length > MAX_NAME_LENGTH) {
|
|
errors.push(`name exceeds ${MAX_NAME_LENGTH} characters (${name.length})`);
|
|
}
|
|
|
|
if (!/^[a-z0-9-]+$/.test(name)) {
|
|
errors.push(`name contains invalid characters (must be lowercase a-z, 0-9, hyphens only)`);
|
|
}
|
|
|
|
if (name.startsWith("-") || name.endsWith("-")) {
|
|
errors.push(`name must not start or end with a hyphen`);
|
|
}
|
|
|
|
if (name.includes("--")) {
|
|
errors.push(`name must not contain consecutive hyphens`);
|
|
}
|
|
|
|
return errors;
|
|
}
|
|
|
|
/**
|
|
* Validate description per Agent Skills spec.
|
|
*/
|
|
function validateDescription(description: string | undefined): string[] {
|
|
const errors: string[] = [];
|
|
|
|
if (!description || description.trim() === "") {
|
|
errors.push(`description is required`);
|
|
} else if (description.length > MAX_DESCRIPTION_LENGTH) {
|
|
errors.push(`description exceeds ${MAX_DESCRIPTION_LENGTH} characters (${description.length})`);
|
|
}
|
|
|
|
return errors;
|
|
}
|
|
|
|
/**
|
|
* Check for unknown frontmatter fields.
|
|
*/
|
|
function validateFrontmatterFields(keys: string[]): string[] {
|
|
const errors: string[] = [];
|
|
for (const key of keys) {
|
|
if (!ALLOWED_FRONTMATTER_FIELDS.has(key)) {
|
|
errors.push(`unknown frontmatter field "${key}"`);
|
|
}
|
|
}
|
|
return errors;
|
|
}
|
|
|
|
export interface LoadSkillsFromDirOptions {
|
|
/** Directory to scan for skills */
|
|
dir: string;
|
|
/** Source identifier for these skills */
|
|
source: string;
|
|
}
|
|
|
|
/**
|
|
* Load skills from a directory recursively.
|
|
* Skills are directories containing a SKILL.md file with frontmatter including a description.
|
|
*/
|
|
export function loadSkillsFromDir(options: LoadSkillsFromDirOptions): LoadSkillsResult {
|
|
const { dir, source } = options;
|
|
return loadSkillsFromDirInternal(dir, source, "recursive");
|
|
}
|
|
|
|
function loadSkillsFromDirInternal(dir: string, source: string, format: SkillFormat): LoadSkillsResult {
|
|
const skills: Skill[] = [];
|
|
const warnings: SkillWarning[] = [];
|
|
|
|
if (!existsSync(dir)) {
|
|
return { skills, warnings };
|
|
}
|
|
|
|
try {
|
|
const entries = readdirSync(dir, { withFileTypes: true });
|
|
|
|
for (const entry of entries) {
|
|
if (entry.name.startsWith(".")) {
|
|
continue;
|
|
}
|
|
|
|
// Skip node_modules to avoid scanning dependencies
|
|
if (entry.name === "node_modules") {
|
|
continue;
|
|
}
|
|
|
|
const fullPath = join(dir, entry.name);
|
|
|
|
// For symlinks, check if they point to a directory and follow them
|
|
let isDirectory = entry.isDirectory();
|
|
let isFile = entry.isFile();
|
|
if (entry.isSymbolicLink()) {
|
|
try {
|
|
const stats = statSync(fullPath);
|
|
isDirectory = stats.isDirectory();
|
|
isFile = stats.isFile();
|
|
} catch {
|
|
// Broken symlink, skip it
|
|
continue;
|
|
}
|
|
}
|
|
|
|
if (format === "recursive") {
|
|
// Recursive format: scan directories, look for SKILL.md files
|
|
if (isDirectory) {
|
|
const subResult = loadSkillsFromDirInternal(fullPath, source, format);
|
|
skills.push(...subResult.skills);
|
|
warnings.push(...subResult.warnings);
|
|
} else if (isFile && entry.name === "SKILL.md") {
|
|
const result = loadSkillFromFile(fullPath, source);
|
|
if (result.skill) {
|
|
skills.push(result.skill);
|
|
}
|
|
warnings.push(...result.warnings);
|
|
}
|
|
} else if (format === "claude") {
|
|
// Claude format: only one level deep, each directory must contain SKILL.md
|
|
if (!isDirectory) {
|
|
continue;
|
|
}
|
|
|
|
const skillFile = join(fullPath, "SKILL.md");
|
|
if (!existsSync(skillFile)) {
|
|
continue;
|
|
}
|
|
|
|
const result = loadSkillFromFile(skillFile, source);
|
|
if (result.skill) {
|
|
skills.push(result.skill);
|
|
}
|
|
warnings.push(...result.warnings);
|
|
}
|
|
}
|
|
} catch {}
|
|
|
|
return { skills, warnings };
|
|
}
|
|
|
|
function loadSkillFromFile(filePath: string, source: string): { skill: Skill | null; warnings: SkillWarning[] } {
|
|
const warnings: SkillWarning[] = [];
|
|
|
|
try {
|
|
const rawContent = readFileSync(filePath, "utf-8");
|
|
const { frontmatter, allKeys } = parseFrontmatter(rawContent);
|
|
const skillDir = dirname(filePath);
|
|
const parentDirName = basename(skillDir);
|
|
|
|
// Validate frontmatter fields
|
|
const fieldErrors = validateFrontmatterFields(allKeys);
|
|
for (const error of fieldErrors) {
|
|
warnings.push({ skillPath: filePath, message: error });
|
|
}
|
|
|
|
// Validate description
|
|
const descErrors = validateDescription(frontmatter.description);
|
|
for (const error of descErrors) {
|
|
warnings.push({ skillPath: filePath, message: error });
|
|
}
|
|
|
|
// Use name from frontmatter, or fall back to parent directory name
|
|
const name = frontmatter.name || parentDirName;
|
|
|
|
// Validate name
|
|
const nameErrors = validateName(name, parentDirName);
|
|
for (const error of nameErrors) {
|
|
warnings.push({ skillPath: filePath, message: error });
|
|
}
|
|
|
|
// Still load the skill even with warnings (unless description is completely missing)
|
|
if (!frontmatter.description || frontmatter.description.trim() === "") {
|
|
return { skill: null, warnings };
|
|
}
|
|
|
|
return {
|
|
skill: {
|
|
name,
|
|
description: frontmatter.description,
|
|
filePath,
|
|
baseDir: skillDir,
|
|
source,
|
|
},
|
|
warnings,
|
|
};
|
|
} catch {
|
|
return { skill: null, warnings };
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Format skills for inclusion in a system prompt.
|
|
* Uses XML format per Agent Skills standard.
|
|
* See: https://agentskills.io/integrate-skills
|
|
*/
|
|
export function formatSkillsForPrompt(skills: Skill[]): string {
|
|
if (skills.length === 0) {
|
|
return "";
|
|
}
|
|
|
|
const lines = [
|
|
"\n\nThe following skills provide specialized instructions for specific tasks.",
|
|
"Use the read tool to load a skill's file when the task matches its description.",
|
|
"",
|
|
"<available_skills>",
|
|
];
|
|
|
|
for (const skill of skills) {
|
|
lines.push(" <skill>");
|
|
lines.push(` <name>${escapeXml(skill.name)}</name>`);
|
|
lines.push(` <description>${escapeXml(skill.description)}</description>`);
|
|
lines.push(` <location>${escapeXml(skill.filePath)}</location>`);
|
|
lines.push(" </skill>");
|
|
}
|
|
|
|
lines.push("</available_skills>");
|
|
|
|
return lines.join("\n");
|
|
}
|
|
|
|
function escapeXml(str: string): string {
|
|
return str
|
|
.replace(/&/g, "&")
|
|
.replace(/</g, "<")
|
|
.replace(/>/g, ">")
|
|
.replace(/"/g, """)
|
|
.replace(/'/g, "'");
|
|
}
|
|
|
|
export interface LoadSkillsOptions extends SkillsSettings {
|
|
/** Working directory for project-local skills. Default: process.cwd() */
|
|
cwd?: string;
|
|
/** Agent config directory for global skills. Default: ~/.pi/agent */
|
|
agentDir?: string;
|
|
}
|
|
|
|
/**
|
|
* Load skills from all configured locations.
|
|
* Returns skills and any validation warnings.
|
|
*/
|
|
export function loadSkills(options: LoadSkillsOptions = {}): LoadSkillsResult {
|
|
const {
|
|
cwd = process.cwd(),
|
|
agentDir,
|
|
enableCodexUser = true,
|
|
enableClaudeUser = true,
|
|
enableClaudeProject = true,
|
|
enablePiUser = true,
|
|
enablePiProject = true,
|
|
customDirectories = [],
|
|
ignoredSkills = [],
|
|
includeSkills = [],
|
|
} = options;
|
|
|
|
// Resolve agentDir - if not provided, use default from config
|
|
const resolvedAgentDir = agentDir ?? getAgentDir();
|
|
|
|
const skillMap = new Map<string, Skill>();
|
|
const realPathSet = new Set<string>();
|
|
const allWarnings: SkillWarning[] = [];
|
|
const collisionWarnings: SkillWarning[] = [];
|
|
|
|
// Check if skill name matches any of the include patterns
|
|
function matchesIncludePatterns(name: string): boolean {
|
|
if (includeSkills.length === 0) return true; // No filter = include all
|
|
return includeSkills.some((pattern) => minimatch(name, pattern));
|
|
}
|
|
|
|
// Check if skill name matches any of the ignore patterns
|
|
function matchesIgnorePatterns(name: string): boolean {
|
|
if (ignoredSkills.length === 0) return false;
|
|
return ignoredSkills.some((pattern) => minimatch(name, pattern));
|
|
}
|
|
|
|
function addSkills(result: LoadSkillsResult) {
|
|
allWarnings.push(...result.warnings);
|
|
for (const skill of result.skills) {
|
|
// Apply ignore filter (glob patterns) - takes precedence over include
|
|
if (matchesIgnorePatterns(skill.name)) {
|
|
continue;
|
|
}
|
|
// Apply include filter (glob patterns)
|
|
if (!matchesIncludePatterns(skill.name)) {
|
|
continue;
|
|
}
|
|
|
|
// Resolve symlinks to detect duplicate files
|
|
let realPath: string;
|
|
try {
|
|
realPath = realpathSync(skill.filePath);
|
|
} catch {
|
|
realPath = skill.filePath;
|
|
}
|
|
|
|
// Skip silently if we've already loaded this exact file (via symlink)
|
|
if (realPathSet.has(realPath)) {
|
|
continue;
|
|
}
|
|
|
|
const existing = skillMap.get(skill.name);
|
|
if (existing) {
|
|
collisionWarnings.push({
|
|
skillPath: skill.filePath,
|
|
message: `name collision: "${skill.name}" already loaded from ${existing.filePath}, skipping this one`,
|
|
});
|
|
} else {
|
|
skillMap.set(skill.name, skill);
|
|
realPathSet.add(realPath);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (enableCodexUser) {
|
|
addSkills(loadSkillsFromDirInternal(join(homedir(), ".codex", "skills"), "codex-user", "recursive"));
|
|
}
|
|
if (enableClaudeUser) {
|
|
addSkills(loadSkillsFromDirInternal(join(homedir(), ".claude", "skills"), "claude-user", "claude"));
|
|
}
|
|
if (enableClaudeProject) {
|
|
addSkills(loadSkillsFromDirInternal(resolve(cwd, ".claude", "skills"), "claude-project", "claude"));
|
|
}
|
|
if (enablePiUser) {
|
|
addSkills(loadSkillsFromDirInternal(join(resolvedAgentDir, "skills"), "user", "recursive"));
|
|
}
|
|
if (enablePiProject) {
|
|
addSkills(loadSkillsFromDirInternal(resolve(cwd, CONFIG_DIR_NAME, "skills"), "project", "recursive"));
|
|
}
|
|
for (const customDir of customDirectories) {
|
|
addSkills(loadSkillsFromDirInternal(customDir.replace(/^~(?=$|[\\/])/, homedir()), "custom", "recursive"));
|
|
}
|
|
|
|
return {
|
|
skills: Array.from(skillMap.values()),
|
|
warnings: [...allWarnings, ...collisionWarnings],
|
|
};
|
|
}
|