Files
get-shit-done/tests/workflow-guard-registration.test.cjs
Tom Boucher 0b6ef6fa24 fix: register gsd-workflow-guard.js in settings.json during install (#1772)
The hook was built, copied to hooks/dist/, and installed to disk — but
never registered as a PreToolUse entry in settings.json, making the
hooks.workflow_guard config flag permanently inert.

Adds the registration block following the same pattern as the other
community hooks (prompt-guard, read-guard, validate-commit, etc.).

Includes regression test that verifies every JS hook in gsdHooks has a
corresponding command construction and registration block.

Fixes #1767

Co-authored-by: Claude Opus 4.6 <noreply@anthropic.com>
2026-04-05 12:30:24 -04:00

102 lines
4.0 KiB
JavaScript

/**
* Regression guard for #1767: gsd-workflow-guard.js must be registered in settings.json
*
* The hook file is built, copied, and installed — but was never registered as a
* PreToolUse hook entry in install.js. This test ensures the registration block
* exists with the correct structure.
*
* Also tests the broader anti-pattern: every hook in gsdHooks that is a JS
* PreToolUse/PostToolUse hook should have a corresponding registration block.
*/
const { describe, test } = require('node:test');
const assert = require('node:assert/strict');
const fs = require('fs');
const path = require('path');
const INSTALL_JS = path.join(__dirname, '..', 'bin', 'install.js');
describe('workflow-guard hook registration (#1767)', () => {
test('install.js constructs a command path variable for gsd-workflow-guard.js', () => {
const content = fs.readFileSync(INSTALL_JS, 'utf-8');
const lines = content.split('\n');
// Every registered JS hook has a command variable constructed via
// buildHookCommand() or string concatenation. Filter out references
// that are only in the cleanup/uninstall arrays.
const commandConstructionLines = lines.filter(line =>
line.includes('gsd-workflow-guard.js') &&
(line.includes('buildHookCommand') || line.includes("'node '"))
);
assert.ok(
commandConstructionLines.length > 0,
[
'install.js must construct a command path for gsd-workflow-guard.js',
'(e.g. buildHookCommand or node + dirName pattern).',
'Currently only referenced in gsdHooks cleanup array.',
].join(' ')
);
});
test('install.js has a hasWorkflowGuardHook dedup check', () => {
const content = fs.readFileSync(INSTALL_JS, 'utf-8');
// Every registered hook has a dedup check: hasXxxHook = settings.hooks[...].some(...)
const hasDedup = content.includes('hasWorkflowGuardHook') ||
content.includes('hasWorkflowGuard');
assert.ok(
hasDedup,
'install.js must have a dedup check variable for workflow-guard (like hasPromptGuardHook)'
);
});
test('install.js pushes workflow-guard entry with correct matcher', () => {
const content = fs.readFileSync(INSTALL_JS, 'utf-8');
// Extract the section between "workflow-guard" command construction
// and the next console.log confirmation. The push block should have:
// matcher: 'Write|Edit' and command referencing workflow-guard
const workflowGuardSection = content.match(
/workflowGuardCommand[\s\S]*?console\.log\([^)]*workflow.guard/i
);
assert.ok(
workflowGuardSection,
'install.js must have a push block for workflow-guard with a console.log confirmation'
);
});
});
describe('hook registration completeness anti-pattern guard', () => {
test('every JS hook in gsdHooks has a command construction in install.js', () => {
const content = fs.readFileSync(INSTALL_JS, 'utf-8');
// Extract gsdHooks array entries
const hooksMatch = content.match(/gsdHooks\s*=\s*\[([^\]]+)\]/);
assert.ok(hooksMatch, 'gsdHooks array must exist in install.js');
const hookNames = hooksMatch[1]
.match(/'([^']+)'/g)
.map(h => h.replace(/'/g, ''));
const jsHooks = hookNames.filter(h => h.endsWith('.js'));
const missing = [];
for (const hook of jsHooks) {
// Each JS hook should have a buildHookCommand or 'node ' command construction
// that references the hook filename (not just the gsdHooks array or uninstall filter)
const hookBase = hook.replace('.js', '');
const lines = content.split('\n').filter(line =>
line.includes(hook) &&
(line.includes('buildHookCommand') || line.includes("'node '"))
);
if (lines.length === 0) {
missing.push(hook);
}
}
assert.strictEqual(
missing.length, 0,
[
'Every JS hook in gsdHooks must have a command construction in install.js.',
'Missing registration for:',
...missing.map(h => ` - ${h}`),
].join('\n')
);
});
});