mirror of
https://github.com/affaan-m/everything-claude-code.git
synced 2026-02-15 10:53:18 +08:00
Bug fixes:
- utils.js: glob-to-regex conversion now escapes all regex special chars
(+, ^, $, |, (), {}, [], \) before converting * and ? wildcards
- validate-hooks.js: escape sequence processing order corrected —
\\\\ now processed before \\n and \\t to prevent double-processing
- 6 hooks: added process.stdin.setEncoding('utf8') to prevent
multi-byte UTF-8 character corruption at chunk boundaries
(check-console-log, post-edit-format, post-edit-typecheck,
post-edit-console-warn, session-end, evaluate-session)
New tests (37):
- CI validator test suite (tests/ci/validators.test.js):
- validate-agents: 9 tests (real project, frontmatter parsing,
BOM/CRLF, colons in values, missing fields, non-md skip)
- validate-hooks: 13 tests (real project, invalid JSON, invalid
event types, missing fields, async/timeout validation, inline JS
syntax, array commands, legacy format)
- validate-skills: 6 tests (real project, missing SKILL.md, empty
files, non-directory entries)
- validate-commands: 5 tests (real project, empty files, non-md skip)
- validate-rules: 4 tests (real project, empty files)
Total test count: 228 (up from 191)
149 lines
4.6 KiB
JavaScript
149 lines
4.6 KiB
JavaScript
#!/usr/bin/env node
|
|
/**
|
|
* Validate hooks.json schema
|
|
*/
|
|
|
|
const fs = require('fs');
|
|
const path = require('path');
|
|
const vm = require('vm');
|
|
|
|
const HOOKS_FILE = path.join(__dirname, '../../hooks/hooks.json');
|
|
const VALID_EVENTS = ['PreToolUse', 'PostToolUse', 'PreCompact', 'SessionStart', 'SessionEnd', 'Stop', 'Notification', 'SubagentStop'];
|
|
|
|
/**
|
|
* Validate a single hook entry has required fields and valid inline JS
|
|
* @param {object} hook - Hook object with type and command fields
|
|
* @param {string} label - Label for error messages (e.g., "PreToolUse[0].hooks[1]")
|
|
* @returns {boolean} true if errors were found
|
|
*/
|
|
function validateHookEntry(hook, label) {
|
|
let hasErrors = false;
|
|
|
|
if (!hook.type || typeof hook.type !== 'string') {
|
|
console.error(`ERROR: ${label} missing or invalid 'type' field`);
|
|
hasErrors = true;
|
|
}
|
|
|
|
// Validate optional async and timeout fields
|
|
if ('async' in hook && typeof hook.async !== 'boolean') {
|
|
console.error(`ERROR: ${label} 'async' must be a boolean`);
|
|
hasErrors = true;
|
|
}
|
|
if ('timeout' in hook && (typeof hook.timeout !== 'number' || hook.timeout < 0)) {
|
|
console.error(`ERROR: ${label} 'timeout' must be a non-negative number`);
|
|
hasErrors = true;
|
|
}
|
|
|
|
if (!hook.command || (typeof hook.command !== 'string' && !Array.isArray(hook.command))) {
|
|
console.error(`ERROR: ${label} missing or invalid 'command' field`);
|
|
hasErrors = true;
|
|
} else if (typeof hook.command === 'string') {
|
|
// Validate inline JS syntax in node -e commands
|
|
const nodeEMatch = hook.command.match(/^node -e "(.*)"$/s);
|
|
if (nodeEMatch) {
|
|
try {
|
|
new vm.Script(nodeEMatch[1].replace(/\\\\/g, '\\').replace(/\\"/g, '"').replace(/\\n/g, '\n').replace(/\\t/g, '\t'));
|
|
} catch (syntaxErr) {
|
|
console.error(`ERROR: ${label} has invalid inline JS: ${syntaxErr.message}`);
|
|
hasErrors = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
return hasErrors;
|
|
}
|
|
|
|
function validateHooks() {
|
|
if (!fs.existsSync(HOOKS_FILE)) {
|
|
console.log('No hooks.json found, skipping validation');
|
|
process.exit(0);
|
|
}
|
|
|
|
let data;
|
|
try {
|
|
data = JSON.parse(fs.readFileSync(HOOKS_FILE, 'utf-8'));
|
|
} catch (e) {
|
|
console.error(`ERROR: Invalid JSON in hooks.json: ${e.message}`);
|
|
process.exit(1);
|
|
}
|
|
|
|
// Support both object format { hooks: {...} } and array format
|
|
const hooks = data.hooks || data;
|
|
let hasErrors = false;
|
|
let totalMatchers = 0;
|
|
|
|
if (typeof hooks === 'object' && !Array.isArray(hooks)) {
|
|
// Object format: { EventType: [matchers] }
|
|
for (const [eventType, matchers] of Object.entries(hooks)) {
|
|
if (!VALID_EVENTS.includes(eventType)) {
|
|
console.error(`ERROR: Invalid event type: ${eventType}`);
|
|
hasErrors = true;
|
|
continue;
|
|
}
|
|
|
|
if (!Array.isArray(matchers)) {
|
|
console.error(`ERROR: ${eventType} must be an array`);
|
|
hasErrors = true;
|
|
continue;
|
|
}
|
|
|
|
for (let i = 0; i < matchers.length; i++) {
|
|
const matcher = matchers[i];
|
|
if (typeof matcher !== 'object' || matcher === null) {
|
|
console.error(`ERROR: ${eventType}[${i}] is not an object`);
|
|
hasErrors = true;
|
|
continue;
|
|
}
|
|
if (!matcher.matcher) {
|
|
console.error(`ERROR: ${eventType}[${i}] missing 'matcher' field`);
|
|
hasErrors = true;
|
|
}
|
|
if (!matcher.hooks || !Array.isArray(matcher.hooks)) {
|
|
console.error(`ERROR: ${eventType}[${i}] missing 'hooks' array`);
|
|
hasErrors = true;
|
|
} else {
|
|
// Validate each hook entry
|
|
for (let j = 0; j < matcher.hooks.length; j++) {
|
|
if (validateHookEntry(matcher.hooks[j], `${eventType}[${i}].hooks[${j}]`)) {
|
|
hasErrors = true;
|
|
}
|
|
}
|
|
}
|
|
totalMatchers++;
|
|
}
|
|
}
|
|
} else if (Array.isArray(hooks)) {
|
|
// Array format (legacy)
|
|
for (let i = 0; i < hooks.length; i++) {
|
|
const hook = hooks[i];
|
|
if (!hook.matcher) {
|
|
console.error(`ERROR: Hook ${i} missing 'matcher' field`);
|
|
hasErrors = true;
|
|
}
|
|
if (!hook.hooks || !Array.isArray(hook.hooks)) {
|
|
console.error(`ERROR: Hook ${i} missing 'hooks' array`);
|
|
hasErrors = true;
|
|
} else {
|
|
// Validate each hook entry
|
|
for (let j = 0; j < hook.hooks.length; j++) {
|
|
if (validateHookEntry(hook.hooks[j], `Hook ${i}.hooks[${j}]`)) {
|
|
hasErrors = true;
|
|
}
|
|
}
|
|
}
|
|
totalMatchers++;
|
|
}
|
|
} else {
|
|
console.error('ERROR: hooks.json must be an object or array');
|
|
process.exit(1);
|
|
}
|
|
|
|
if (hasErrors) {
|
|
process.exit(1);
|
|
}
|
|
|
|
console.log(`Validated ${totalMatchers} hook matchers`);
|
|
}
|
|
|
|
validateHooks();
|