mirror of
https://github.com/affaan-m/everything-claude-code.git
synced 2026-03-30 13:43:26 +08:00
Implements three roadmap features: - Agent description compression (#491): New `agent-compress` module with catalog/summary/full compression modes and lazy-loading. Reduces ~26k token agent descriptions to ~2-3k catalog entries for context efficiency. - Inspection logic (#485): New `inspection` module that detects recurring failure patterns in skill_runs. Groups by skill + normalized failure reason, generates structured reports with suggested remediation actions. Configurable threshold (default: 3 failures). - Governance event capture hook (#482): PreToolUse/PostToolUse hook that detects secrets, policy violations, approval-required commands, and elevated privilege usage. Gated behind ECC_GOVERNANCE_CAPTURE=1 flag. Writes to governance_events table via JSON-line stderr output. 59 new tests (16 + 16 + 27), all passing.
231 lines
5.6 KiB
JavaScript
231 lines
5.6 KiB
JavaScript
'use strict';
|
|
|
|
const fs = require('fs');
|
|
const path = require('path');
|
|
|
|
/**
|
|
* Parse YAML frontmatter from a markdown string.
|
|
* Returns { frontmatter: {}, body: string }.
|
|
*/
|
|
function parseFrontmatter(content) {
|
|
const match = content.match(/^---\r?\n([\s\S]*?)\r?\n---\r?\n([\s\S]*)$/);
|
|
if (!match) {
|
|
return { frontmatter: {}, body: content };
|
|
}
|
|
|
|
const frontmatter = {};
|
|
for (const line of match[1].split('\n')) {
|
|
const colonIdx = line.indexOf(':');
|
|
if (colonIdx === -1) continue;
|
|
|
|
const key = line.slice(0, colonIdx).trim();
|
|
let value = line.slice(colonIdx + 1).trim();
|
|
|
|
// Handle JSON arrays (e.g. tools: ["Read", "Grep"])
|
|
if (value.startsWith('[') && value.endsWith(']')) {
|
|
try {
|
|
value = JSON.parse(value);
|
|
} catch {
|
|
// keep as string
|
|
}
|
|
}
|
|
|
|
// Strip surrounding quotes
|
|
if (typeof value === 'string' && value.startsWith('"') && value.endsWith('"')) {
|
|
value = value.slice(1, -1);
|
|
}
|
|
|
|
frontmatter[key] = value;
|
|
}
|
|
|
|
return { frontmatter, body: match[2] };
|
|
}
|
|
|
|
/**
|
|
* Extract the first meaningful paragraph from agent body as a summary.
|
|
* Skips headings and blank lines, returns up to maxSentences sentences.
|
|
*/
|
|
function extractSummary(body, maxSentences = 1) {
|
|
const lines = body.split('\n');
|
|
const paragraphs = [];
|
|
let current = [];
|
|
|
|
for (const line of lines) {
|
|
const trimmed = line.trim();
|
|
|
|
if (trimmed === '') {
|
|
if (current.length > 0) {
|
|
paragraphs.push(current.join(' '));
|
|
current = [];
|
|
}
|
|
continue;
|
|
}
|
|
|
|
// Skip headings
|
|
if (trimmed.startsWith('#')) {
|
|
if (current.length > 0) {
|
|
paragraphs.push(current.join(' '));
|
|
current = [];
|
|
}
|
|
continue;
|
|
}
|
|
|
|
// Skip list items, code blocks, etc.
|
|
if (trimmed.startsWith('```') || trimmed.startsWith('- **') || trimmed.startsWith('|')) {
|
|
continue;
|
|
}
|
|
|
|
current.push(trimmed);
|
|
}
|
|
if (current.length > 0) {
|
|
paragraphs.push(current.join(' '));
|
|
}
|
|
|
|
// Find first non-empty paragraph
|
|
const firstParagraph = paragraphs.find(p => p.length > 0);
|
|
if (!firstParagraph) {
|
|
return '';
|
|
}
|
|
|
|
// Extract up to maxSentences sentences
|
|
const sentences = firstParagraph.match(/[^.!?]+[.!?]+/g) || [firstParagraph];
|
|
return sentences.slice(0, maxSentences).join(' ').trim();
|
|
}
|
|
|
|
/**
|
|
* Load and parse a single agent file.
|
|
* Returns the full agent object with frontmatter and body.
|
|
*/
|
|
function loadAgent(filePath) {
|
|
const content = fs.readFileSync(filePath, 'utf8');
|
|
const { frontmatter, body } = parseFrontmatter(content);
|
|
const fileName = path.basename(filePath, '.md');
|
|
|
|
return {
|
|
fileName,
|
|
name: frontmatter.name || fileName,
|
|
description: frontmatter.description || '',
|
|
tools: Array.isArray(frontmatter.tools) ? frontmatter.tools : [],
|
|
model: frontmatter.model || 'sonnet',
|
|
body,
|
|
byteSize: Buffer.byteLength(content, 'utf8'),
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Load all agents from a directory.
|
|
*/
|
|
function loadAgents(agentsDir) {
|
|
if (!fs.existsSync(agentsDir)) {
|
|
return [];
|
|
}
|
|
|
|
return fs.readdirSync(agentsDir)
|
|
.filter(f => f.endsWith('.md'))
|
|
.sort()
|
|
.map(f => loadAgent(path.join(agentsDir, f)));
|
|
}
|
|
|
|
/**
|
|
* Compress an agent to its catalog entry (metadata only).
|
|
* This is the minimal representation needed for agent selection.
|
|
*/
|
|
function compressToCatalog(agent) {
|
|
return {
|
|
name: agent.name,
|
|
description: agent.description,
|
|
tools: agent.tools,
|
|
model: agent.model,
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Compress an agent to a summary entry (metadata + first paragraph).
|
|
* More context than catalog, less than full body.
|
|
*/
|
|
function compressToSummary(agent) {
|
|
return {
|
|
name: agent.name,
|
|
description: agent.description,
|
|
tools: agent.tools,
|
|
model: agent.model,
|
|
summary: extractSummary(agent.body),
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Build a full compressed catalog from a directory of agents.
|
|
*
|
|
* Modes:
|
|
* - 'catalog': name, description, tools, model only (~2-3k tokens for 27 agents)
|
|
* - 'summary': catalog + first paragraph summary (~4-5k tokens)
|
|
* - 'full': no compression, full body included
|
|
*
|
|
* Returns { agents: [], stats: { totalAgents, originalBytes, compressedTokenEstimate } }
|
|
*/
|
|
function buildAgentCatalog(agentsDir, options = {}) {
|
|
const mode = options.mode || 'catalog';
|
|
const filter = options.filter || null;
|
|
|
|
let agents = loadAgents(agentsDir);
|
|
|
|
if (typeof filter === 'function') {
|
|
agents = agents.filter(filter);
|
|
}
|
|
|
|
const originalBytes = agents.reduce((sum, a) => sum + a.byteSize, 0);
|
|
|
|
let compressed;
|
|
if (mode === 'catalog') {
|
|
compressed = agents.map(compressToCatalog);
|
|
} else if (mode === 'summary') {
|
|
compressed = agents.map(compressToSummary);
|
|
} else {
|
|
compressed = agents.map(a => ({
|
|
name: a.name,
|
|
description: a.description,
|
|
tools: a.tools,
|
|
model: a.model,
|
|
body: a.body,
|
|
}));
|
|
}
|
|
|
|
const compressedJson = JSON.stringify(compressed);
|
|
// Rough token estimate: ~4 chars per token for English text
|
|
const compressedTokenEstimate = Math.ceil(compressedJson.length / 4);
|
|
|
|
return {
|
|
agents: compressed,
|
|
stats: {
|
|
totalAgents: agents.length,
|
|
originalBytes,
|
|
compressedBytes: Buffer.byteLength(compressedJson, 'utf8'),
|
|
compressedTokenEstimate,
|
|
mode,
|
|
},
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Lazy-load a single agent's full content by name from a directory.
|
|
* Returns null if not found.
|
|
*/
|
|
function lazyLoadAgent(agentsDir, agentName) {
|
|
const filePath = path.join(agentsDir, `${agentName}.md`);
|
|
if (!fs.existsSync(filePath)) {
|
|
return null;
|
|
}
|
|
return loadAgent(filePath);
|
|
}
|
|
|
|
module.exports = {
|
|
buildAgentCatalog,
|
|
compressToCatalog,
|
|
compressToSummary,
|
|
extractSummary,
|
|
lazyLoadAgent,
|
|
loadAgent,
|
|
loadAgents,
|
|
parseFrontmatter,
|
|
};
|