ktg-plugin-marketplace/plugins/config-audit/tests/scanners/manifest.test.mjs
Kjell Tore Guttormsen 0420b8cc4a feat(config-audit): /config-audit manifest command (v5 N2) [skip-docs]
New scanners/manifest.mjs CLI + commands/manifest.md slash command.
Reads activeConfig and produces a flat, ranked list of every token
source (CLAUDE.md cascade entries, plugins, skills, MCP servers, hooks)
sorted DESC by estimated_tokens.

CLAUDE.md per-file tokens are derived by distributing
claudeMd.estimatedTokens across the cascade proportional to bytes.

Tests cover both real-config (plugin root) and fixture (rich-repo with
patched HOME containing 2 plugins + 3 skills + .mcp.json) paths, plus
error handling (nonexistent path → exit 3, --output-file).

Builds on readActiveConfig from M1 (v5 alpha.2).

[skip-docs] reason: v5 plan fences off README/CLAUDE.md badge updates
to Session 5; Forgejo pre-commit-docs-gate hook requires this tag on
feat commits without doc changes.

Tests: 593 → 604 (+11).

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-01 07:32:54 +02:00

201 lines
7.8 KiB
JavaScript

import { describe, it, before, after } from 'node:test';
import assert from 'node:assert/strict';
import { spawnSync } from 'node:child_process';
import { resolve, join, dirname } from 'node:path';
import { fileURLToPath } from 'node:url';
import { mkdir, writeFile, rm } from 'node:fs/promises';
import { tmpdir } from 'node:os';
const __dirname = fileURLToPath(new URL('.', import.meta.url));
const PLUGIN_ROOT = resolve(__dirname, '../..');
const CLI = join(PLUGIN_ROOT, 'scanners', 'manifest.mjs');
function uniqueDir(suffix) {
return join(tmpdir(), `config-audit-manifest-${suffix}-${Date.now()}-${Math.random().toString(36).slice(2, 8)}`);
}
function runCli(args, env = {}) {
const proc = spawnSync('node', [CLI, ...args], {
encoding: 'utf-8',
env: { ...process.env, ...env },
});
return proc;
}
describe('manifest CLI — real-config path (plugin root)', () => {
let output;
before(() => {
const proc = runCli([PLUGIN_ROOT, '--json']);
assert.equal(proc.status, 0, `expected exit 0, got ${proc.status}; stderr: ${proc.stderr}`);
output = JSON.parse(proc.stdout);
});
it('emits non-empty sources array', () => {
assert.ok(Array.isArray(output.sources));
assert.ok(output.sources.length > 0,
`expected sources.length > 0, got ${output.sources.length}`);
});
it('sources are sorted DESC by estimated_tokens', () => {
for (let i = 1; i < output.sources.length; i++) {
const prev = output.sources[i - 1].estimated_tokens;
const curr = output.sources[i].estimated_tokens;
assert.ok(prev >= curr,
`sources[${i - 1}] (${prev}) should be >= sources[${i}] (${curr})`);
}
});
it('total ≈ sum(sources.estimated_tokens) (within rounding tolerance)', () => {
const sum = output.sources.reduce((s, x) => s + (x.estimated_tokens || 0), 0);
assert.ok(output.total >= sum - 1 && output.total <= sum + 1,
`expected total ≈ ${sum}, got ${output.total}`);
});
it('every source has kind/name/source/estimated_tokens', () => {
for (const s of output.sources) {
assert.ok(typeof s.kind === 'string' && s.kind.length > 0, 's.kind missing');
assert.ok(typeof s.name === 'string' && s.name.length > 0, 's.name missing');
assert.ok(typeof s.source === 'string', 's.source missing');
assert.equal(typeof s.estimated_tokens, 'number', 's.estimated_tokens not a number');
}
});
it('meta.repoPath matches the requested path', () => {
assert.equal(output.meta.repoPath, PLUGIN_ROOT);
});
});
describe('manifest CLI — fixture path (rich-repo with patched HOME)', () => {
let fixture;
before(async () => {
fixture = await buildRichManifestRepo(uniqueDir('rich'));
});
after(async () => {
if (fixture) await rm(fixture.root, { recursive: true, force: true });
});
it('discovers ≥5 sources (CLAUDE.md cascade + plugins + skills + MCP)', () => {
const proc = runCli([fixture.root, '--json'], { HOME: fixture.fakeHome });
assert.equal(proc.status, 0, `stderr: ${proc.stderr}`);
const out = JSON.parse(proc.stdout);
assert.ok(out.sources.length >= 5,
`expected sources.length >= 5, got ${out.sources.length}: ${out.sources.map(s => `${s.kind}:${s.name}`).join(', ')}`);
});
it('includes both plugins (manifest-plugin-a + manifest-plugin-b)', () => {
const proc = runCli([fixture.root, '--json'], { HOME: fixture.fakeHome });
const out = JSON.parse(proc.stdout);
const pluginNames = out.sources.filter(s => s.kind === 'plugin').map(s => s.name);
assert.ok(pluginNames.includes('manifest-plugin-a'),
`expected manifest-plugin-a in plugins; got: ${pluginNames.join(', ')}`);
assert.ok(pluginNames.includes('manifest-plugin-b'),
`expected manifest-plugin-b in plugins; got: ${pluginNames.join(', ')}`);
});
it('includes 3 fixture skills (alpha-skill, beta-skill, gamma-skill)', () => {
const proc = runCli([fixture.root, '--json'], { HOME: fixture.fakeHome });
const out = JSON.parse(proc.stdout);
const skillNames = out.sources.filter(s => s.kind === 'skill').map(s => s.name);
for (const expected of ['alpha-skill', 'beta-skill', 'gamma-skill']) {
assert.ok(skillNames.includes(expected),
`expected skill ${expected}; got skills: ${skillNames.join(', ')}`);
}
});
it('includes the project .mcp.json server (manifest-mcp)', () => {
const proc = runCli([fixture.root, '--json'], { HOME: fixture.fakeHome });
const out = JSON.parse(proc.stdout);
const mcpNames = out.sources.filter(s => s.kind === 'mcp-server').map(s => s.name);
assert.ok(mcpNames.includes('manifest-mcp'),
`expected manifest-mcp in mcp-servers; got: ${mcpNames.join(', ')}`);
});
});
describe('manifest CLI — error handling', () => {
it('exits 3 for nonexistent path', () => {
const proc = runCli(['/nonexistent/path/should/not/exist', '--json']);
assert.equal(proc.status, 3);
});
it('--output-file writes JSON to the path', async () => {
const outPath = join(tmpdir(), `manifest-output-${Date.now()}.json`);
try {
const proc = runCli([PLUGIN_ROOT, '--output-file', outPath]);
assert.equal(proc.status, 0);
const { readFile } = await import('node:fs/promises');
const content = await readFile(outPath, 'utf-8');
const parsed = JSON.parse(content);
assert.ok(Array.isArray(parsed.sources));
} finally {
await rm(outPath, { force: true });
}
});
});
/**
* Build a richer fixture for manifest tests: 2 plugins + 3 skills + project
* .mcp.json. Mirrors buildRichRepo from active-config-reader.test.mjs but
* gives every plugin/skill a unique, recognizable name so assertions can be
* substring-based instead of count-based.
*/
async function buildRichManifestRepo(root) {
const fakeHome = join(root, 'fake-home');
await mkdir(join(root, '.git'), { recursive: true });
await writeFile(join(root, '.git', 'HEAD'), 'ref: refs/heads/main\n');
await writeFile(
join(root, 'CLAUDE.md'),
'# Project\n\nManifest fixture.\n',
);
await mkdir(join(fakeHome, '.claude'), { recursive: true });
await writeFile(
join(fakeHome, '.claude', 'CLAUDE.md'),
'# User\n\nFake home for manifest tests.\n',
);
await writeFile(
join(root, '.mcp.json'),
JSON.stringify({
mcpServers: {
'manifest-mcp': { command: 'npx', args: ['fake-pkg'] },
},
}, null, 2),
);
// Plugin A — has alpha-skill + beta-skill
const pluginA = join(fakeHome, '.claude', 'plugins', 'marketplaces', 'mp', 'plugins', 'manifest-plugin-a');
await mkdir(join(pluginA, '.claude-plugin'), { recursive: true });
await writeFile(
join(pluginA, '.claude-plugin', 'plugin.json'),
JSON.stringify({ name: 'manifest-plugin-a', version: '1.0.0', description: 'plugin a' }, null, 2),
);
await mkdir(join(pluginA, 'skills', 'alpha-skill'), { recursive: true });
await writeFile(
join(pluginA, 'skills', 'alpha-skill', 'SKILL.md'),
'---\nname: alpha-skill\ndescription: alpha skill from plugin a\n---\n\nAlpha body.\n',
);
await mkdir(join(pluginA, 'skills', 'beta-skill'), { recursive: true });
await writeFile(
join(pluginA, 'skills', 'beta-skill', 'SKILL.md'),
'---\nname: beta-skill\ndescription: beta skill from plugin a\n---\n\nBeta body.\n',
);
// Plugin B — has gamma-skill
const pluginB = join(fakeHome, '.claude', 'plugins', 'marketplaces', 'mp', 'plugins', 'manifest-plugin-b');
await mkdir(join(pluginB, '.claude-plugin'), { recursive: true });
await writeFile(
join(pluginB, '.claude-plugin', 'plugin.json'),
JSON.stringify({ name: 'manifest-plugin-b', version: '1.0.0', description: 'plugin b' }, null, 2),
);
await mkdir(join(pluginB, 'skills', 'gamma-skill'), { recursive: true });
await writeFile(
join(pluginB, 'skills', 'gamma-skill', 'SKILL.md'),
'---\nname: gamma-skill\ndescription: gamma skill from plugin b\n---\n\nGamma body.\n',
);
return { root, fakeHome };
}