2026-01-03 16:24:36 -08:00
|
|
|
/**
|
|
|
|
|
* @license
|
|
|
|
|
* Copyright 2025 Google LLC
|
|
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
*/
|
|
|
|
|
|
2026-01-12 15:44:08 -08:00
|
|
|
import * as path from 'node:path';
|
|
|
|
|
import { fileURLToPath } from 'node:url';
|
2026-01-03 16:24:36 -08:00
|
|
|
import { Storage } from '../config/storage.js';
|
|
|
|
|
import { type SkillDefinition, loadSkillsFromDir } from './skillLoader.js';
|
2026-01-04 14:45:07 -08:00
|
|
|
import type { GeminiCLIExtension } from '../config/config.js';
|
|
|
|
|
|
|
|
|
|
export { type SkillDefinition };
|
2026-01-03 16:24:36 -08:00
|
|
|
|
|
|
|
|
export class SkillManager {
|
|
|
|
|
private skills: SkillDefinition[] = [];
|
|
|
|
|
private activeSkillNames: Set<string> = new Set();
|
2026-01-13 23:40:23 -08:00
|
|
|
private adminSkillsEnabled = true;
|
2026-01-03 16:24:36 -08:00
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Clears all discovered skills.
|
|
|
|
|
*/
|
|
|
|
|
clearSkills(): void {
|
|
|
|
|
this.skills = [];
|
|
|
|
|
}
|
|
|
|
|
|
2026-01-13 23:40:23 -08:00
|
|
|
/**
|
|
|
|
|
* Sets administrative settings for skills.
|
|
|
|
|
*/
|
|
|
|
|
setAdminSettings(enabled: boolean): void {
|
|
|
|
|
this.adminSkillsEnabled = enabled;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Returns true if skills are enabled by the admin.
|
|
|
|
|
*/
|
|
|
|
|
isAdminEnabled(): boolean {
|
|
|
|
|
return this.adminSkillsEnabled;
|
|
|
|
|
}
|
|
|
|
|
|
2026-01-03 16:24:36 -08:00
|
|
|
/**
|
2026-01-04 14:45:07 -08:00
|
|
|
* Discovers skills from standard user and project locations, as well as extensions.
|
|
|
|
|
* Precedence: Extensions (lowest) -> User -> Project (highest).
|
2026-01-03 16:24:36 -08:00
|
|
|
*/
|
2026-01-04 14:45:07 -08:00
|
|
|
async discoverSkills(
|
|
|
|
|
storage: Storage,
|
|
|
|
|
extensions: GeminiCLIExtension[] = [],
|
|
|
|
|
): Promise<void> {
|
2026-01-03 16:24:36 -08:00
|
|
|
this.clearSkills();
|
|
|
|
|
|
2026-01-09 22:26:58 -08:00
|
|
|
// 1. Built-in skills (lowest precedence)
|
|
|
|
|
await this.discoverBuiltinSkills();
|
|
|
|
|
|
|
|
|
|
// 2. Extension skills
|
2026-01-04 14:45:07 -08:00
|
|
|
for (const extension of extensions) {
|
|
|
|
|
if (extension.isActive && extension.skills) {
|
|
|
|
|
this.addSkillsWithPrecedence(extension.skills);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2026-01-09 22:26:58 -08:00
|
|
|
// 3. User skills
|
2026-01-03 16:24:36 -08:00
|
|
|
const userSkills = await loadSkillsFromDir(Storage.getUserSkillsDir());
|
|
|
|
|
this.addSkillsWithPrecedence(userSkills);
|
|
|
|
|
|
2026-01-09 22:26:58 -08:00
|
|
|
// 4. Project skills (highest precedence)
|
2026-01-03 16:24:36 -08:00
|
|
|
const projectSkills = await loadSkillsFromDir(
|
|
|
|
|
storage.getProjectSkillsDir(),
|
|
|
|
|
);
|
|
|
|
|
this.addSkillsWithPrecedence(projectSkills);
|
|
|
|
|
}
|
|
|
|
|
|
2026-01-09 22:26:58 -08:00
|
|
|
/**
|
|
|
|
|
* Discovers built-in skills.
|
|
|
|
|
*/
|
|
|
|
|
private async discoverBuiltinSkills(): Promise<void> {
|
2026-01-12 15:44:08 -08:00
|
|
|
const __dirname = path.dirname(fileURLToPath(import.meta.url));
|
|
|
|
|
const builtinDir = path.join(__dirname, 'builtin');
|
|
|
|
|
|
|
|
|
|
const builtinSkills = await loadSkillsFromDir(builtinDir);
|
|
|
|
|
|
|
|
|
|
for (const skill of builtinSkills) {
|
|
|
|
|
skill.isBuiltin = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
this.addSkillsWithPrecedence(builtinSkills);
|
2026-01-09 22:26:58 -08:00
|
|
|
}
|
|
|
|
|
|
2026-01-03 16:24:36 -08:00
|
|
|
private addSkillsWithPrecedence(newSkills: SkillDefinition[]): void {
|
|
|
|
|
const skillMap = new Map<string, SkillDefinition>();
|
|
|
|
|
for (const skill of [...this.skills, ...newSkills]) {
|
|
|
|
|
skillMap.set(skill.name, skill);
|
|
|
|
|
}
|
|
|
|
|
this.skills = Array.from(skillMap.values());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Returns the list of enabled discovered skills.
|
|
|
|
|
*/
|
|
|
|
|
getSkills(): SkillDefinition[] {
|
|
|
|
|
return this.skills.filter((s) => !s.disabled);
|
|
|
|
|
}
|
|
|
|
|
|
2026-01-09 22:26:58 -08:00
|
|
|
/**
|
|
|
|
|
* Returns the list of enabled discovered skills that should be displayed in the UI.
|
|
|
|
|
* This excludes built-in skills.
|
|
|
|
|
*/
|
|
|
|
|
getDisplayableSkills(): SkillDefinition[] {
|
|
|
|
|
return this.skills.filter((s) => !s.disabled && !s.isBuiltin);
|
|
|
|
|
}
|
|
|
|
|
|
2026-01-03 16:24:36 -08:00
|
|
|
/**
|
|
|
|
|
* Returns all discovered skills, including disabled ones.
|
|
|
|
|
*/
|
|
|
|
|
getAllSkills(): SkillDefinition[] {
|
|
|
|
|
return this.skills;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Filters discovered skills by name.
|
|
|
|
|
*/
|
|
|
|
|
filterSkills(predicate: (skill: SkillDefinition) => boolean): void {
|
|
|
|
|
this.skills = this.skills.filter(predicate);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Sets the list of disabled skill names.
|
|
|
|
|
*/
|
|
|
|
|
setDisabledSkills(disabledNames: string[]): void {
|
2026-01-09 22:26:58 -08:00
|
|
|
const lowercaseDisabledNames = disabledNames.map((n) => n.toLowerCase());
|
2026-01-03 16:24:36 -08:00
|
|
|
for (const skill of this.skills) {
|
2026-01-09 22:26:58 -08:00
|
|
|
skill.disabled = lowercaseDisabledNames.includes(
|
|
|
|
|
skill.name.toLowerCase(),
|
|
|
|
|
);
|
2026-01-03 16:24:36 -08:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Reads the full content (metadata + body) of a skill by name.
|
|
|
|
|
*/
|
|
|
|
|
getSkill(name: string): SkillDefinition | null {
|
2026-01-09 22:26:58 -08:00
|
|
|
const lowercaseName = name.toLowerCase();
|
|
|
|
|
return (
|
|
|
|
|
this.skills.find((s) => s.name.toLowerCase() === lowercaseName) ?? null
|
|
|
|
|
);
|
2026-01-03 16:24:36 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Activates a skill by name.
|
|
|
|
|
*/
|
|
|
|
|
activateSkill(name: string): void {
|
|
|
|
|
this.activeSkillNames.add(name);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Checks if a skill is active.
|
|
|
|
|
*/
|
|
|
|
|
isSkillActive(name: string): boolean {
|
|
|
|
|
return this.activeSkillNames.has(name);
|
|
|
|
|
}
|
|
|
|
|
}
|