Features: - 30+ Custom Skills (cognitive, development, UI/UX, autonomous agents) - RalphLoop autonomous agent integration - Multi-AI consultation (Qwen) - Agent management system with sync capabilities - Custom hooks for session management - MCP servers integration - Plugin marketplace setup - Comprehensive installation script Components: - Skills: always-use-superpowers, ralph, brainstorming, ui-ux-pro-max, etc. - Agents: 100+ agents across engineering, marketing, product, etc. - Hooks: session-start-superpowers, qwen-consult, ralph-auto-trigger - Commands: /brainstorm, /write-plan, /execute-plan - MCP Servers: zai-mcp-server, web-search-prime, web-reader, zread - Binaries: ralphloop wrapper Installation: ./supercharge.sh
169 lines
4.7 KiB
JavaScript
Executable File
169 lines
4.7 KiB
JavaScript
Executable File
#!/usr/bin/env node
|
|
|
|
/**
|
|
* Render graphviz diagrams from a skill's SKILL.md to SVG files.
|
|
*
|
|
* Usage:
|
|
* ./render-graphs.js <skill-directory> # Render each diagram separately
|
|
* ./render-graphs.js <skill-directory> --combine # Combine all into one diagram
|
|
*
|
|
* Extracts all ```dot blocks from SKILL.md and renders to SVG.
|
|
* Useful for helping your human partner visualize the process flows.
|
|
*
|
|
* Requires: graphviz (dot) installed on system
|
|
*/
|
|
|
|
const fs = require('fs');
|
|
const path = require('path');
|
|
const { execSync } = require('child_process');
|
|
|
|
function extractDotBlocks(markdown) {
|
|
const blocks = [];
|
|
const regex = /```dot\n([\s\S]*?)```/g;
|
|
let match;
|
|
|
|
while ((match = regex.exec(markdown)) !== null) {
|
|
const content = match[1].trim();
|
|
|
|
// Extract digraph name
|
|
const nameMatch = content.match(/digraph\s+(\w+)/);
|
|
const name = nameMatch ? nameMatch[1] : `graph_${blocks.length + 1}`;
|
|
|
|
blocks.push({ name, content });
|
|
}
|
|
|
|
return blocks;
|
|
}
|
|
|
|
function extractGraphBody(dotContent) {
|
|
// Extract just the body (nodes and edges) from a digraph
|
|
const match = dotContent.match(/digraph\s+\w+\s*\{([\s\S]*)\}/);
|
|
if (!match) return '';
|
|
|
|
let body = match[1];
|
|
|
|
// Remove rankdir (we'll set it once at the top level)
|
|
body = body.replace(/^\s*rankdir\s*=\s*\w+\s*;?\s*$/gm, '');
|
|
|
|
return body.trim();
|
|
}
|
|
|
|
function combineGraphs(blocks, skillName) {
|
|
const bodies = blocks.map((block, i) => {
|
|
const body = extractGraphBody(block.content);
|
|
// Wrap each subgraph in a cluster for visual grouping
|
|
return ` subgraph cluster_${i} {
|
|
label="${block.name}";
|
|
${body.split('\n').map(line => ' ' + line).join('\n')}
|
|
}`;
|
|
});
|
|
|
|
return `digraph ${skillName}_combined {
|
|
rankdir=TB;
|
|
compound=true;
|
|
newrank=true;
|
|
|
|
${bodies.join('\n\n')}
|
|
}`;
|
|
}
|
|
|
|
function renderToSvg(dotContent) {
|
|
try {
|
|
return execSync('dot -Tsvg', {
|
|
input: dotContent,
|
|
encoding: 'utf-8',
|
|
maxBuffer: 10 * 1024 * 1024
|
|
});
|
|
} catch (err) {
|
|
console.error('Error running dot:', err.message);
|
|
if (err.stderr) console.error(err.stderr.toString());
|
|
return null;
|
|
}
|
|
}
|
|
|
|
function main() {
|
|
const args = process.argv.slice(2);
|
|
const combine = args.includes('--combine');
|
|
const skillDirArg = args.find(a => !a.startsWith('--'));
|
|
|
|
if (!skillDirArg) {
|
|
console.error('Usage: render-graphs.js <skill-directory> [--combine]');
|
|
console.error('');
|
|
console.error('Options:');
|
|
console.error(' --combine Combine all diagrams into one SVG');
|
|
console.error('');
|
|
console.error('Example:');
|
|
console.error(' ./render-graphs.js ../subagent-driven-development');
|
|
console.error(' ./render-graphs.js ../subagent-driven-development --combine');
|
|
process.exit(1);
|
|
}
|
|
|
|
const skillDir = path.resolve(skillDirArg);
|
|
const skillFile = path.join(skillDir, 'SKILL.md');
|
|
const skillName = path.basename(skillDir).replace(/-/g, '_');
|
|
|
|
if (!fs.existsSync(skillFile)) {
|
|
console.error(`Error: ${skillFile} not found`);
|
|
process.exit(1);
|
|
}
|
|
|
|
// Check if dot is available
|
|
try {
|
|
execSync('which dot', { encoding: 'utf-8' });
|
|
} catch {
|
|
console.error('Error: graphviz (dot) not found. Install with:');
|
|
console.error(' brew install graphviz # macOS');
|
|
console.error(' apt install graphviz # Linux');
|
|
process.exit(1);
|
|
}
|
|
|
|
const markdown = fs.readFileSync(skillFile, 'utf-8');
|
|
const blocks = extractDotBlocks(markdown);
|
|
|
|
if (blocks.length === 0) {
|
|
console.log('No ```dot blocks found in', skillFile);
|
|
process.exit(0);
|
|
}
|
|
|
|
console.log(`Found ${blocks.length} diagram(s) in ${path.basename(skillDir)}/SKILL.md`);
|
|
|
|
const outputDir = path.join(skillDir, 'diagrams');
|
|
if (!fs.existsSync(outputDir)) {
|
|
fs.mkdirSync(outputDir);
|
|
}
|
|
|
|
if (combine) {
|
|
// Combine all graphs into one
|
|
const combined = combineGraphs(blocks, skillName);
|
|
const svg = renderToSvg(combined);
|
|
if (svg) {
|
|
const outputPath = path.join(outputDir, `${skillName}_combined.svg`);
|
|
fs.writeFileSync(outputPath, svg);
|
|
console.log(` Rendered: ${skillName}_combined.svg`);
|
|
|
|
// Also write the dot source for debugging
|
|
const dotPath = path.join(outputDir, `${skillName}_combined.dot`);
|
|
fs.writeFileSync(dotPath, combined);
|
|
console.log(` Source: ${skillName}_combined.dot`);
|
|
} else {
|
|
console.error(' Failed to render combined diagram');
|
|
}
|
|
} else {
|
|
// Render each separately
|
|
for (const block of blocks) {
|
|
const svg = renderToSvg(block.content);
|
|
if (svg) {
|
|
const outputPath = path.join(outputDir, `${block.name}.svg`);
|
|
fs.writeFileSync(outputPath, svg);
|
|
console.log(` Rendered: ${block.name}.svg`);
|
|
} else {
|
|
console.error(` Failed: ${block.name}`);
|
|
}
|
|
}
|
|
}
|
|
|
|
console.log(`\nOutput: ${outputDir}/`);
|
|
}
|
|
|
|
main();
|