mirror of
https://github.com/github/awesome-copilot.git
synced 2026-02-20 18:35:14 +00:00
- Move plugin manifests from .github/plugin/ to .claude-plugin/ - Convert items[] to Claude Code spec fields (agents, commands, skills) - Rename tags to keywords, drop display/featured/instructions from plugins - Delete all symlinks and materialized files from plugin directories - Add eng/materialize-plugins.mjs to copy source files into plugin dirs at publish time - Add .github/workflows/publish.yml for staged->main publishing - Update CI triggers to target staged branch - Update validation, creation, marketplace, and README generation scripts - Update CONTRIBUTING.md and AGENTS.md documentation - Include all new content from main (polyglot-test-agent, gem-browser-tester, fabric-lakehouse, fluentui-blazor, quasi-coder, transloadit-media-processing, make-repo-contribution hardening, website logo/gradient changes) Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com>
230 lines
6.6 KiB
JavaScript
Executable File
230 lines
6.6 KiB
JavaScript
Executable File
#!/usr/bin/env node
|
|
|
|
import fs from "fs";
|
|
import path from "path";
|
|
import { ROOT_FOLDER } from "./constants.mjs";
|
|
|
|
const PLUGINS_DIR = path.join(ROOT_FOLDER, "plugins");
|
|
|
|
// Validation functions
|
|
function validateName(name, folderName) {
|
|
const errors = [];
|
|
if (!name || typeof name !== "string") {
|
|
errors.push("name is required and must be a string");
|
|
return errors;
|
|
}
|
|
if (name.length < 1 || name.length > 50) {
|
|
errors.push("name must be between 1 and 50 characters");
|
|
}
|
|
if (!/^[a-z0-9-]+$/.test(name)) {
|
|
errors.push("name must contain only lowercase letters, numbers, and hyphens");
|
|
}
|
|
if (name !== folderName) {
|
|
errors.push(`name "${name}" must match folder name "${folderName}"`);
|
|
}
|
|
return errors;
|
|
}
|
|
|
|
function validateDescription(description) {
|
|
if (!description || typeof description !== "string") {
|
|
return "description is required and must be a string";
|
|
}
|
|
if (description.length < 1 || description.length > 500) {
|
|
return "description must be between 1 and 500 characters";
|
|
}
|
|
return null;
|
|
}
|
|
|
|
function validateVersion(version) {
|
|
if (!version || typeof version !== "string") {
|
|
return "version is required and must be a string";
|
|
}
|
|
return null;
|
|
}
|
|
|
|
function validateKeywords(keywords) {
|
|
if (keywords === undefined) return null;
|
|
if (!Array.isArray(keywords)) {
|
|
return "keywords must be an array";
|
|
}
|
|
if (keywords.length > 10) {
|
|
return "maximum 10 keywords allowed";
|
|
}
|
|
for (const keyword of keywords) {
|
|
if (typeof keyword !== "string") {
|
|
return "all keywords must be strings";
|
|
}
|
|
if (!/^[a-z0-9-]+$/.test(keyword)) {
|
|
return `keyword "${keyword}" must contain only lowercase letters, numbers, and hyphens`;
|
|
}
|
|
if (keyword.length < 1 || keyword.length > 30) {
|
|
return `keyword "${keyword}" must be between 1 and 30 characters`;
|
|
}
|
|
}
|
|
return null;
|
|
}
|
|
|
|
function validateSpecPaths(plugin) {
|
|
const errors = [];
|
|
const specs = {
|
|
agents: { prefix: "./agents/", suffix: ".md", repoDir: "agents", repoSuffix: ".agent.md" },
|
|
commands: { prefix: "./commands/", suffix: ".md", repoDir: "prompts", repoSuffix: ".prompt.md" },
|
|
skills: { prefix: "./skills/", suffix: "/", repoDir: "skills", repoFile: "SKILL.md" },
|
|
};
|
|
|
|
for (const [field, spec] of Object.entries(specs)) {
|
|
const arr = plugin[field];
|
|
if (arr === undefined) continue;
|
|
if (!Array.isArray(arr)) {
|
|
errors.push(`${field} must be an array`);
|
|
continue;
|
|
}
|
|
for (let i = 0; i < arr.length; i++) {
|
|
const p = arr[i];
|
|
if (typeof p !== "string") {
|
|
errors.push(`${field}[${i}] must be a string`);
|
|
continue;
|
|
}
|
|
if (!p.startsWith("./")) {
|
|
errors.push(`${field}[${i}] must start with "./"`);
|
|
continue;
|
|
}
|
|
if (!p.startsWith(spec.prefix)) {
|
|
errors.push(`${field}[${i}] must start with "${spec.prefix}"`);
|
|
continue;
|
|
}
|
|
if (!p.endsWith(spec.suffix)) {
|
|
errors.push(`${field}[${i}] must end with "${spec.suffix}"`);
|
|
continue;
|
|
}
|
|
// Validate the source file exists at repo root
|
|
const basename = p.slice(spec.prefix.length, p.length - spec.suffix.length);
|
|
if (field === "skills") {
|
|
const skillDir = path.join(ROOT_FOLDER, spec.repoDir, basename);
|
|
const skillFile = path.join(skillDir, spec.repoFile);
|
|
if (!fs.existsSync(skillFile)) {
|
|
errors.push(`${field}[${i}] source not found: ${spec.repoDir}/${basename}/SKILL.md`);
|
|
}
|
|
} else {
|
|
const srcFile = path.join(ROOT_FOLDER, spec.repoDir, basename + spec.repoSuffix);
|
|
if (!fs.existsSync(srcFile)) {
|
|
errors.push(`${field}[${i}] source not found: ${spec.repoDir}/${basename}${spec.repoSuffix}`);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return errors;
|
|
}
|
|
|
|
function validatePlugin(folderName) {
|
|
const pluginDir = path.join(PLUGINS_DIR, folderName);
|
|
const errors = [];
|
|
|
|
// Rule 1: Must have .github/plugin/plugin.json
|
|
const pluginJsonPath = path.join(pluginDir, ".github/plugin", "plugin.json");
|
|
if (!fs.existsSync(pluginJsonPath)) {
|
|
errors.push("missing required file: .github/plugin/plugin.json");
|
|
return errors;
|
|
}
|
|
|
|
// Rule 2: Must have README.md
|
|
const readmePath = path.join(pluginDir, "README.md");
|
|
if (!fs.existsSync(readmePath)) {
|
|
errors.push("missing required file: README.md");
|
|
}
|
|
|
|
// Parse plugin.json
|
|
let plugin;
|
|
try {
|
|
const raw = fs.readFileSync(pluginJsonPath, "utf-8");
|
|
plugin = JSON.parse(raw);
|
|
} catch (err) {
|
|
errors.push(`failed to parse plugin.json: ${err.message}`);
|
|
return errors;
|
|
}
|
|
|
|
// Rule 3 & 4: name, description, version
|
|
const nameErrors = validateName(plugin.name, folderName);
|
|
errors.push(...nameErrors);
|
|
|
|
const descError = validateDescription(plugin.description);
|
|
if (descError) errors.push(descError);
|
|
|
|
const versionError = validateVersion(plugin.version);
|
|
if (versionError) errors.push(versionError);
|
|
|
|
// Rule 5: keywords (or tags for backward compat)
|
|
const keywordsError = validateKeywords(plugin.keywords ?? plugin.tags);
|
|
if (keywordsError) errors.push(keywordsError);
|
|
|
|
// Rule 6: agents, commands, skills paths
|
|
const specErrors = validateSpecPaths(plugin);
|
|
errors.push(...specErrors);
|
|
|
|
return errors;
|
|
}
|
|
|
|
// Main validation function
|
|
function validatePlugins() {
|
|
if (!fs.existsSync(PLUGINS_DIR)) {
|
|
console.log("No plugins directory found - validation skipped");
|
|
return true;
|
|
}
|
|
|
|
const pluginDirs = fs
|
|
.readdirSync(PLUGINS_DIR, { withFileTypes: true })
|
|
.filter((d) => d.isDirectory())
|
|
.map((d) => d.name);
|
|
|
|
if (pluginDirs.length === 0) {
|
|
console.log("No plugin directories found - validation skipped");
|
|
return true;
|
|
}
|
|
|
|
console.log(`Validating ${pluginDirs.length} plugins...\n`);
|
|
|
|
let hasErrors = false;
|
|
const seenNames = new Set();
|
|
|
|
for (const dir of pluginDirs) {
|
|
console.log(`Validating ${dir}...`);
|
|
|
|
const errors = validatePlugin(dir);
|
|
|
|
if (errors.length > 0) {
|
|
console.error(`❌ ${dir}:`);
|
|
errors.forEach((e) => console.error(` - ${e}`));
|
|
hasErrors = true;
|
|
} else {
|
|
console.log(`✅ ${dir} is valid`);
|
|
}
|
|
|
|
// Rule 10: duplicate names
|
|
if (seenNames.has(dir)) {
|
|
console.error(`❌ Duplicate plugin name "${dir}"`);
|
|
hasErrors = true;
|
|
} else {
|
|
seenNames.add(dir);
|
|
}
|
|
}
|
|
|
|
if (!hasErrors) {
|
|
console.log(`\n✅ All ${pluginDirs.length} plugins are valid`);
|
|
}
|
|
|
|
return !hasErrors;
|
|
}
|
|
|
|
// Run validation
|
|
try {
|
|
const isValid = validatePlugins();
|
|
if (!isValid) {
|
|
console.error("\n❌ Plugin validation failed");
|
|
process.exit(1);
|
|
}
|
|
console.log("\n🎉 Plugin validation passed");
|
|
} catch (error) {
|
|
console.error(`Error during validation: ${error.message}`);
|
|
process.exit(1);
|
|
}
|