mirror of
https://github.com/affaan-m/everything-claude-code.git
synced 2026-02-14 18:33:09 +08:00
- validate-agents.js: reject invalid model names in agent frontmatter - package-manager.js: validate script/binary names against shell injection - session-manager.js: reject impossible month/day values in filenames - utils.js: support options.all for replaceInFile string patterns - strategic-compact/SKILL.md: fix hook matcher syntax and script reference - install.sh: warn when overwriting existing rule customizations - Add 24 new tests covering all validation and edge cases
82 lines
2.2 KiB
JavaScript
82 lines
2.2 KiB
JavaScript
#!/usr/bin/env node
|
|
/**
|
|
* Validate agent markdown files have required frontmatter
|
|
*/
|
|
|
|
const fs = require('fs');
|
|
const path = require('path');
|
|
|
|
const AGENTS_DIR = path.join(__dirname, '../../agents');
|
|
const REQUIRED_FIELDS = ['model', 'tools'];
|
|
const VALID_MODELS = ['haiku', 'sonnet', 'opus'];
|
|
|
|
function extractFrontmatter(content) {
|
|
// Strip BOM if present (UTF-8 BOM: \uFEFF)
|
|
const cleanContent = content.replace(/^\uFEFF/, '');
|
|
// Support both LF and CRLF line endings
|
|
const match = cleanContent.match(/^---\r?\n([\s\S]*?)\r?\n---/);
|
|
if (!match) return null;
|
|
|
|
const frontmatter = {};
|
|
const lines = match[1].split(/\r?\n/);
|
|
for (const line of lines) {
|
|
const colonIdx = line.indexOf(':');
|
|
if (colonIdx > 0) {
|
|
const key = line.slice(0, colonIdx).trim();
|
|
const value = line.slice(colonIdx + 1).trim();
|
|
frontmatter[key] = value;
|
|
}
|
|
}
|
|
return frontmatter;
|
|
}
|
|
|
|
function validateAgents() {
|
|
if (!fs.existsSync(AGENTS_DIR)) {
|
|
console.log('No agents directory found, skipping validation');
|
|
process.exit(0);
|
|
}
|
|
|
|
const files = fs.readdirSync(AGENTS_DIR).filter(f => f.endsWith('.md'));
|
|
let hasErrors = false;
|
|
|
|
for (const file of files) {
|
|
const filePath = path.join(AGENTS_DIR, file);
|
|
let content;
|
|
try {
|
|
content = fs.readFileSync(filePath, 'utf-8');
|
|
} catch (err) {
|
|
console.error(`ERROR: ${file} - ${err.message}`);
|
|
hasErrors = true;
|
|
continue;
|
|
}
|
|
const frontmatter = extractFrontmatter(content);
|
|
|
|
if (!frontmatter) {
|
|
console.error(`ERROR: ${file} - Missing frontmatter`);
|
|
hasErrors = true;
|
|
continue;
|
|
}
|
|
|
|
for (const field of REQUIRED_FIELDS) {
|
|
if (!frontmatter[field]) {
|
|
console.error(`ERROR: ${file} - Missing required field: ${field}`);
|
|
hasErrors = true;
|
|
}
|
|
}
|
|
|
|
// Validate model is a known value
|
|
if (frontmatter.model && !VALID_MODELS.includes(frontmatter.model)) {
|
|
console.error(`ERROR: ${file} - Invalid model '${frontmatter.model}'. Must be one of: ${VALID_MODELS.join(', ')}`);
|
|
hasErrors = true;
|
|
}
|
|
}
|
|
|
|
if (hasErrors) {
|
|
process.exit(1);
|
|
}
|
|
|
|
console.log(`Validated ${files.length} agent files`);
|
|
}
|
|
|
|
validateAgents();
|