mirror of
https://github.com/affaan-m/everything-claude-code.git
synced 2026-02-16 03:13:08 +08:00
- Add try-catch around readFileSync in validate-agents, validate-commands, validate-skills to handle TOCTOU races and file read errors - Add validate-hooks.js and all test suites to package.json test script (was only running 4/5 validators and 0/4 test files) - Fix shell variable injection in observe.sh: use os.environ instead of interpolating $timestamp/$OBSERVATIONS_FILE into Python string literals - Fix $? always being 0 in start-observer.sh: capture exit code before conditional since `if !` inverts the status - Add OLD_VERSION validation in release.sh and use pipe delimiter in sed to avoid issues with slash-containing values - Add jq dependency check in evaluate-session.sh before parsing config - Sync .cursor/ copies of all modified shell scripts
75 lines
1.9 KiB
JavaScript
75 lines
1.9 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'];
|
|
|
|
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('\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;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (hasErrors) {
|
|
process.exit(1);
|
|
}
|
|
|
|
console.log(`Validated ${files.length} agent files`);
|
|
}
|
|
|
|
validateAgents();
|