claude-flow
Version:
Ruflo - Enterprise AI agent orchestration for Claude Code. Deploy 60+ specialized agents in coordinated swarms with self-learning, fault-tolerant consensus, vector memory, and MCP integration
130 lines (113 loc) • 5.2 kB
JavaScript
/**
* Regression guard for ruvnet/ruflo#1859 + #1862.
*
* Drives each PostToolUse hook command from `hooks/hooks.json` with synthetic
* Claude-Code-style stdin against a locally built CLI, asserting:
*
* - Exit code 0 (no parser errors like "Invalid value for --format")
* - Output records the *intended* value (the file path / command), not a
* stray boolean like "true" — the symptom that #1859 reported
*
* The script substitutes `npx ruflo@alpha` → the local CLI binary, so
* we exercise the same flag wiring users hit in production but pinned to
* the build under test.
*
* Usage (from repo root):
* node plugins/ruflo-core/scripts/test-hooks.mjs <path-to-cli-binary>
*
* Wired into .github/workflows/v3-ci.yml as the `plugin-hooks-smoke` job.
*/
import { readFileSync } from 'node:fs';
import { spawnSync } from 'node:child_process';
import { join, dirname } from 'node:path';
import { fileURLToPath } from 'node:url';
const __dirname = dirname(fileURLToPath(import.meta.url));
const HOOKS_JSON = join(__dirname, '..', 'hooks', 'hooks.json');
// `cliInvoke` is the literal token-string that should run the CLI — caller
// passes the full thing so this script doesn't need to guess shebangs:
// - local node script: "node /abs/path/to/bin/cli.js"
// - shell wrapper: "/abs/path/to/wrapper.sh"
// - npx fallthrough: "npx --yes @claude-flow/cli@latest"
const cliInvoke = process.argv[2];
if (!cliInvoke) {
console.error('Usage: node test-hooks.mjs "<cli-invocation-string>"');
console.error('Examples:');
console.error(' node test-hooks.mjs "node $PWD/v3/@claude-flow/cli/bin/cli.js"');
console.error(' node test-hooks.mjs "npx --yes @claude-flow/cli@latest"');
process.exit(2);
}
const hooks = JSON.parse(readFileSync(HOOKS_JSON, 'utf8'));
const post = hooks.hooks?.PostToolUse ?? [];
const findHook = (matcher) => {
const hit = post.find(h => h.matcher === matcher);
if (!hit) throw new Error(`No PostToolUse hook with matcher=${matcher}`);
return hit.hooks[0].command
// legacy form: `npx ruflo@alpha hooks …`
.replace(/npx ruflo@alpha/g, cliInvoke)
// #1921 form: hook subcommands go through scripts/ruflo-hook.sh (which
// prepends `hooks`). Bypass the shim here and call the built CLI directly
// so the test exercises the same flag wiring users hit, pinned to the
// build under test. Also drop the shim's `|| true` so exit codes are
// still asserted (the shim makes failures non-fatal in production).
.replace(/"\$\{CLAUDE_PLUGIN_ROOT\}\/scripts\/ruflo-hook\.sh"/g, `${cliInvoke} hooks`)
.replace(/\s*\|\|\s*true(\s*')/g, '$1');
};
const cmdBash = findHook('Bash');
const cmdEdit = findHook('Write|Edit|MultiEdit');
let failed = 0;
const cases = [];
const run = (name, cmd, stdin, assertions) => {
const r = spawnSync('bash', ['-c', cmd], { input: stdin, encoding: 'utf8' });
const combined = (r.stdout ?? '') + (r.stderr ?? '');
const errors = [];
if (r.status !== 0) errors.push(`exit ${r.status} (expected 0)`);
for (const a of assertions) {
if (a.contains && !combined.includes(a.contains)) errors.push(`missing "${a.contains}" in output`);
if (a.absent && combined.includes(a.absent)) errors.push(`unexpected "${a.absent}" in output`);
}
if (errors.length === 0) {
console.log(`ok: ${name}`);
} else {
console.error(`FAIL: ${name}`);
for (const e of errors) console.error(` - ${e}`);
if (combined.trim()) {
console.error(' output:');
for (const line of combined.split('\n').slice(0, 8)) console.error(` ${line}`);
}
failed++;
}
cases.push(name);
};
// --- Edit hook ---
run('Edit hook records file_path (regression #1859: was "true")',
cmdEdit,
'{"tool_input":{"file_path":"/tmp/foo.ts"}}',
[{ contains: '/tmp/foo.ts' }, { absent: 'Recording outcome for: true' }, { absent: 'Invalid value' }]);
run('Edit hook records legacy "path" field',
cmdEdit,
'{"tool_input":{"path":"/tmp/bar.ts"}}',
[{ contains: '/tmp/bar.ts' }, { absent: 'Invalid value' }]);
run('Edit hook silently no-ops when no path present',
cmdEdit,
'{"tool_input":{}}',
[]);
// --- Bash hook ---
run('Bash hook records simple command',
cmdBash,
'{"tool_input":{"command":"echo hi"},"tool_response":{"exit_code":0}}',
[{ contains: 'echo hi' }, { absent: 'Required option missing' }, { absent: 'Invalid value' }]);
run('Bash hook records multi-line heredoc (regression #1859)',
cmdBash,
'{"tool_input":{"command":"cat <<EOF\\nline1\\nline2\\nEOF"},"tool_response":{"exit_code":0}}',
[{ contains: 'cat <<EOF' }, { absent: 'Required option missing' }]);
run('Bash hook records non-zero exit (distinct from -s value)',
cmdBash,
'{"tool_input":{"command":"echo failing-cmd"},"tool_response":{"exit_code":1}}',
[{ contains: 'echo failing-cmd' }, { absent: 'Recording command outcome: false' }, { absent: 'Recording command outcome: true' }]);
run('Bash hook silently no-ops when no command present',
cmdBash,
'{"tool_input":{},"tool_response":{}}',
[]);
console.log(`\n${cases.length - failed}/${cases.length} passed`);
process.exit(failed === 0 ? 0 : 1);