mirror of
https://github.com/affaan-m/everything-claude-code.git
synced 2026-04-30 22:13:28 +08:00
test: cover pre-bash commit quality edges
This commit is contained in:
@@ -40,6 +40,48 @@ function inTempRepo(fn) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function captureConsoleError(fn) {
|
||||||
|
const previousError = console.error;
|
||||||
|
const lines = [];
|
||||||
|
console.error = (...args) => {
|
||||||
|
lines.push(args.join(' '));
|
||||||
|
};
|
||||||
|
|
||||||
|
try {
|
||||||
|
const result = fn();
|
||||||
|
return { result, stderr: lines.join('\n') };
|
||||||
|
} finally {
|
||||||
|
console.error = previousError;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function writeAndStage(repoDir, relativePath, content) {
|
||||||
|
const filePath = path.join(repoDir, relativePath);
|
||||||
|
fs.mkdirSync(path.dirname(filePath), { recursive: true });
|
||||||
|
fs.writeFileSync(filePath, content, 'utf8');
|
||||||
|
spawnSync('git', ['add', relativePath], { cwd: repoDir, stdio: 'pipe', encoding: 'utf8' });
|
||||||
|
}
|
||||||
|
|
||||||
|
function withEnv(overrides, fn) {
|
||||||
|
const previous = {};
|
||||||
|
for (const key of Object.keys(overrides)) {
|
||||||
|
previous[key] = process.env[key];
|
||||||
|
process.env[key] = overrides[key];
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
return fn();
|
||||||
|
} finally {
|
||||||
|
for (const key of Object.keys(overrides)) {
|
||||||
|
if (typeof previous[key] === 'string') {
|
||||||
|
process.env[key] = previous[key];
|
||||||
|
} else {
|
||||||
|
delete process.env[key];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
let passed = 0;
|
let passed = 0;
|
||||||
let failed = 0;
|
let failed = 0;
|
||||||
|
|
||||||
@@ -77,5 +119,159 @@ if (test('evaluate inspects staged snapshot instead of newer working tree conten
|
|||||||
});
|
});
|
||||||
})) passed++; else failed++;
|
})) passed++; else failed++;
|
||||||
|
|
||||||
|
if (test('passes through non-commit amend malformed JSON and run wrapper paths', () => {
|
||||||
|
const readInput = JSON.stringify({ tool_input: { command: 'git status --short' } });
|
||||||
|
assert.deepStrictEqual(hook.evaluate(readInput), { output: readInput, exitCode: 0 });
|
||||||
|
|
||||||
|
const amendInput = JSON.stringify({ tool_input: { command: 'git commit --amend -m "fix: update"' } });
|
||||||
|
assert.deepStrictEqual(hook.evaluate(amendInput), { output: amendInput, exitCode: 0 });
|
||||||
|
|
||||||
|
const malformed = 'not json {{{';
|
||||||
|
const malformedResult = captureConsoleError(() => hook.run(malformed));
|
||||||
|
assert.deepStrictEqual(malformedResult.result, { stdout: malformed, exitCode: 0 });
|
||||||
|
assert.ok(malformedResult.stderr.includes('[Hook] Error:'), 'should log JSON parse errors without blocking');
|
||||||
|
})) passed++; else failed++;
|
||||||
|
|
||||||
|
if (test('allows git commit when no files are staged', () => {
|
||||||
|
inTempRepo(() => {
|
||||||
|
const input = JSON.stringify({ tool_input: { command: 'git commit -m "fix: no staged files"' } });
|
||||||
|
const { result, stderr } = captureConsoleError(() => hook.evaluate(input));
|
||||||
|
|
||||||
|
assert.strictEqual(result.output, input);
|
||||||
|
assert.strictEqual(result.exitCode, 0);
|
||||||
|
assert.ok(stderr.includes('No staged files found'), `expected no-staged warning, got: ${stderr}`);
|
||||||
|
});
|
||||||
|
})) passed++; else failed++;
|
||||||
|
|
||||||
|
if (test('allows warning-only issues while reporting console TODO and message warnings', () => {
|
||||||
|
inTempRepo(repoDir => {
|
||||||
|
writeAndStage(repoDir, 'index.js', [
|
||||||
|
'console.log("debug only");',
|
||||||
|
'// TODO: clean this up',
|
||||||
|
'// TODO: tracked in issue #123',
|
||||||
|
'// console.log("commented out");',
|
||||||
|
'* console.log("doc comment");',
|
||||||
|
'const ok = true;',
|
||||||
|
''
|
||||||
|
].join('\n'));
|
||||||
|
|
||||||
|
const input = JSON.stringify({
|
||||||
|
tool_input: {
|
||||||
|
command: 'git commit -m "fix: Uppercase subject."'
|
||||||
|
}
|
||||||
|
});
|
||||||
|
const { result, stderr } = captureConsoleError(() => hook.evaluate(input));
|
||||||
|
|
||||||
|
assert.strictEqual(result.output, input);
|
||||||
|
assert.strictEqual(result.exitCode, 0, 'warning-only issues should not block');
|
||||||
|
assert.ok(stderr.includes('WARNING Line 1'), `expected console warning, got: ${stderr}`);
|
||||||
|
assert.ok(stderr.includes('INFO Line 2'), `expected TODO info warning, got: ${stderr}`);
|
||||||
|
assert.ok(stderr.includes('Subject should start with lowercase'), `expected capitalization warning, got: ${stderr}`);
|
||||||
|
assert.ok(stderr.includes('should not end with a period'), `expected punctuation warning, got: ${stderr}`);
|
||||||
|
assert.ok(stderr.includes('Warnings found'), `expected warning summary, got: ${stderr}`);
|
||||||
|
});
|
||||||
|
})) passed++; else failed++;
|
||||||
|
|
||||||
|
if (test('reports invalid and long commit messages without blocking when files are clean', () => {
|
||||||
|
inTempRepo(repoDir => {
|
||||||
|
writeAndStage(repoDir, 'index.js', 'const clean = true;\n');
|
||||||
|
|
||||||
|
const longMessage = `Bad message ${'x'.repeat(80)}`;
|
||||||
|
const input = JSON.stringify({
|
||||||
|
tool_input: {
|
||||||
|
command: `git commit --message="${longMessage}"`
|
||||||
|
}
|
||||||
|
});
|
||||||
|
const { result, stderr } = captureConsoleError(() => hook.evaluate(input));
|
||||||
|
|
||||||
|
assert.strictEqual(result.output, input);
|
||||||
|
assert.strictEqual(result.exitCode, 0);
|
||||||
|
assert.ok(stderr.includes('does not follow conventional commit format'), `expected format warning, got: ${stderr}`);
|
||||||
|
assert.ok(stderr.includes('Commit message too long'), `expected length warning, got: ${stderr}`);
|
||||||
|
});
|
||||||
|
})) passed++; else failed++;
|
||||||
|
|
||||||
|
if (test('blocks commits with staged secret patterns across checkable files', () => {
|
||||||
|
inTempRepo(repoDir => {
|
||||||
|
writeAndStage(repoDir, 'index.js', [
|
||||||
|
"const openai = 'sk-abcdefghijklmnopqrstuvwxyz';",
|
||||||
|
"const token = 'ghp_abcdefghijklmnopqrstuvwxyzABCDEFGHIJ';",
|
||||||
|
''
|
||||||
|
].join('\n'));
|
||||||
|
writeAndStage(repoDir, 'app.py', [
|
||||||
|
'aws = "AKIAABCDEFGHIJKLMNOP"',
|
||||||
|
'api_key = "secret-value"',
|
||||||
|
''
|
||||||
|
].join('\n'));
|
||||||
|
|
||||||
|
const input = JSON.stringify({ tool_input: { command: 'git commit -m "fix: block secrets"' } });
|
||||||
|
const { result, stderr } = captureConsoleError(() => hook.evaluate(input));
|
||||||
|
|
||||||
|
assert.strictEqual(result.output, input);
|
||||||
|
assert.strictEqual(result.exitCode, 2);
|
||||||
|
assert.ok(stderr.includes('Potential OpenAI API key'), `expected OpenAI secret warning, got: ${stderr}`);
|
||||||
|
assert.ok(stderr.includes('Potential GitHub PAT'), `expected GitHub PAT warning, got: ${stderr}`);
|
||||||
|
assert.ok(stderr.includes('Potential AWS Access Key'), `expected AWS key warning, got: ${stderr}`);
|
||||||
|
assert.ok(stderr.includes('Potential API key'), `expected generic API key warning, got: ${stderr}`);
|
||||||
|
});
|
||||||
|
})) passed++; else failed++;
|
||||||
|
|
||||||
|
if (test('reports eslint pylint and golint failures from staged files', () => {
|
||||||
|
inTempRepo(repoDir => {
|
||||||
|
writeAndStage(repoDir, 'index.js', 'const lint = true;\n');
|
||||||
|
writeAndStage(repoDir, 'app.py', 'print("lint")\n');
|
||||||
|
writeAndStage(repoDir, 'main.go', 'package main\n');
|
||||||
|
|
||||||
|
const eslintPath = path.join(repoDir, 'node_modules', '.bin', process.platform === 'win32' ? 'eslint.cmd' : 'eslint');
|
||||||
|
fs.mkdirSync(path.dirname(eslintPath), { recursive: true });
|
||||||
|
fs.writeFileSync(eslintPath, '#!/bin/sh\necho "eslint failed"\nexit 1\n', 'utf8');
|
||||||
|
fs.chmodSync(eslintPath, 0o755);
|
||||||
|
|
||||||
|
const binDir = path.join(repoDir, 'fake-bin');
|
||||||
|
fs.mkdirSync(binDir, { recursive: true });
|
||||||
|
const pylintPath = path.join(binDir, 'pylint');
|
||||||
|
const golintPath = path.join(binDir, 'golint');
|
||||||
|
fs.writeFileSync(pylintPath, '#!/bin/sh\necho "pylint failed"\nexit 1\n', 'utf8');
|
||||||
|
fs.writeFileSync(golintPath, '#!/bin/sh\necho "main.go:1: lint failed"\nexit 0\n', 'utf8');
|
||||||
|
fs.chmodSync(pylintPath, 0o755);
|
||||||
|
fs.chmodSync(golintPath, 0o755);
|
||||||
|
|
||||||
|
withEnv({ PATH: `${binDir}${path.delimiter}${process.env.PATH || ''}` }, () => {
|
||||||
|
const input = JSON.stringify({ tool_input: { command: 'git commit -m "fix: lint failures"' } });
|
||||||
|
const { result, stderr } = captureConsoleError(() => hook.evaluate(input));
|
||||||
|
|
||||||
|
assert.strictEqual(result.output, input);
|
||||||
|
assert.strictEqual(result.exitCode, 2);
|
||||||
|
assert.ok(stderr.includes('ESLint Issues'), `expected ESLint output, got: ${stderr}`);
|
||||||
|
assert.ok(stderr.includes('eslint failed'), `expected ESLint failure text, got: ${stderr}`);
|
||||||
|
assert.ok(stderr.includes('Pylint Issues'), `expected Pylint output, got: ${stderr}`);
|
||||||
|
assert.ok(stderr.includes('pylint failed'), `expected Pylint failure text, got: ${stderr}`);
|
||||||
|
assert.ok(stderr.includes('golint Issues'), `expected golint output, got: ${stderr}`);
|
||||||
|
assert.ok(stderr.includes('main.go:1: lint failed'), `expected golint failure text, got: ${stderr}`);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
})) passed++; else failed++;
|
||||||
|
|
||||||
|
if (test('stdin entry point truncates oversized input and preserves pass-through output', () => {
|
||||||
|
const oversized = JSON.stringify({
|
||||||
|
tool_input: {
|
||||||
|
command: 'git status',
|
||||||
|
filler: 'x'.repeat(1024 * 1024 + 1024)
|
||||||
|
}
|
||||||
|
});
|
||||||
|
const result = spawnSync('node', [path.join(__dirname, '..', '..', 'scripts', 'hooks', 'pre-bash-commit-quality.js')], {
|
||||||
|
input: oversized,
|
||||||
|
encoding: 'utf8',
|
||||||
|
stdio: ['pipe', 'pipe', 'pipe'],
|
||||||
|
timeout: 10000
|
||||||
|
});
|
||||||
|
|
||||||
|
assert.strictEqual(result.status, 0);
|
||||||
|
assert.ok(result.stdout.length > 0, 'expected truncated payload to pass through');
|
||||||
|
assert.ok(result.stdout.length <= 1024 * 1024, 'expected stdout to stay within hook input limit');
|
||||||
|
assert.strictEqual(result.stdout, oversized.slice(0, result.stdout.length));
|
||||||
|
assert.ok(result.stderr.includes('[Hook] Error:'), 'truncated JSON should be logged and allowed');
|
||||||
|
})) passed++; else failed++;
|
||||||
|
|
||||||
console.log(`\nResults: Passed: ${passed}, Failed: ${failed}`);
|
console.log(`\nResults: Passed: ${passed}, Failed: ${failed}`);
|
||||||
process.exit(failed > 0 ? 1 : 0);
|
process.exit(failed > 0 ? 1 : 0);
|
||||||
|
|||||||
Reference in New Issue
Block a user