@webdevtoday/claude-agents
Version:
AI-powered development shop with 15 specialized agents for Claude Code. Features concurrent execution, shared memory, context-forge integration, and web dashboard for 80% faster development.
180 lines (152 loc) • 6.3 kB
JavaScript
import chalk from 'chalk';
import ora from 'ora';
import { writeFileSync, copyFileSync, existsSync, mkdirSync } from 'fs';
import { join, dirname } from 'path';
import { fileURLToPath } from 'url';
import {
getAgentsDir,
getCommandsDir,
ensureDirectories,
ensureProjectDirectories
} from '../utils/paths.js';
import {
selectAgents,
confirmAction,
selectInstallScope,
selectHookOptions
} from '../utils/prompts.js';
import {
addInstalledAgent,
getInstalledAgents
} from '../utils/config.js';
import {
getAvailableAgents,
getAgentDetails,
formatAgentForInstall
} from '../utils/agents.js';
import { detectContextForge } from '../utils/contextForgeDetector.js';
const __filename = fileURLToPath(import.meta.url);
const __dirname = dirname(__filename);
export async function installCommand(options) {
const spinner = ora();
try {
// Detect context-forge project
const contextForgeInfo = detectContextForge();
if (contextForgeInfo.hasContextForge) {
console.log(chalk.cyan('🛠️ Context-Forge project detected!'));
console.log(chalk.gray(' Sub-agents will be integrated with your existing setup.\n'));
}
// Ensure directories exist
ensureDirectories();
// Get available agents
spinner.start('Loading available agents...');
const availableAgents = getAvailableAgents();
spinner.stop();
if (availableAgents.length === 0) {
console.log(chalk.yellow('No agents available to install.'));
return;
}
// Get already installed agents
const installedAgents = getInstalledAgents();
const installedNames = Object.keys(installedAgents);
// Filter out already installed agents
const installableAgents = availableAgents.filter(
agent => !installedNames.includes(agent.name)
);
if (installableAgents.length === 0) {
console.log(chalk.yellow('All available agents are already installed.'));
console.log(chalk.gray('Use "claude-agents list" to see installed agents.'));
return;
}
// Select agents to install
let selectedAgents;
if (options.all) {
selectedAgents = installableAgents.map(a => a.name);
} else {
selectedAgents = await selectAgents(installableAgents);
}
if (selectedAgents.length === 0) {
console.log(chalk.yellow('No agents selected for installation.'));
return;
}
// Select installation scope
const scope = options.project ? 'project' : await selectInstallScope();
const isProject = scope === 'project';
if (isProject) {
ensureProjectDirectories();
}
const agentsDir = getAgentsDir(isProject);
const commandsDir = getCommandsDir(isProject);
// Confirm installation
const confirmMessage = `Install ${selectedAgents.length} agent(s) to ${scope} directory?`;
if (!await confirmAction(confirmMessage)) {
console.log(chalk.yellow('Installation cancelled.'));
return;
}
// Install each selected agent
console.log('');
for (const agentName of selectedAgents) {
spinner.start(`Installing ${chalk.bold(agentName)}...`);
try {
const agentDetails = getAgentDetails(agentName);
if (!agentDetails) {
spinner.fail(`Failed to load agent ${agentName}`);
continue;
}
// Write agent file
const agentPath = join(agentsDir, `${agentName}.md`);
const formattedContent = formatAgentForInstall(agentDetails);
writeFileSync(agentPath, formattedContent);
// Copy associated slash commands if they exist
if (agentDetails.commands && agentDetails.commands.length > 0) {
// If context-forge project, put commands in sub-agents folder to avoid conflicts
const targetCommandsDir = contextForgeInfo.hasContextForge && isProject
? join(commandsDir, 'agents')
: commandsDir;
// Ensure sub-agents command directory exists
if (contextForgeInfo.hasContextForge && isProject && !existsSync(targetCommandsDir)) {
mkdirSync(targetCommandsDir, { recursive: true });
}
for (const command of agentDetails.commands) {
const srcPath = join(__dirname, '..', '..', 'commands', `${command}.md`);
if (existsSync(srcPath)) {
// Prefix command name if in context-forge project to avoid conflicts
const commandName = contextForgeInfo.hasContextForge && isProject
? `agent-${command}.md`
: `${command}.md`;
const destPath = join(targetCommandsDir, commandName);
copyFileSync(srcPath, destPath);
}
}
}
// Add to config
addInstalledAgent(agentName, agentDetails, isProject);
spinner.succeed(`Installed ${chalk.bold(agentName)}`);
// Ask about hooks configuration
if (agentDetails.hooks?.recommended || agentDetails.hooks?.optional) {
const hooks = await selectHookOptions();
if (hooks && hooks.length > 0) {
console.log(chalk.gray(` Configure hooks manually in your settings.json`));
}
}
} catch (error) {
spinner.fail(`Failed to install ${agentName}: ${error.message}`);
}
}
console.log('');
console.log(chalk.green('✓ Installation complete!'));
console.log(chalk.gray('Use "claude-agents list" to see your installed agents.'));
console.log(chalk.gray('Agents are automatically enabled. Use "claude-agents disable <agent>" to disable.'));
if (contextForgeInfo.hasContextForge && isProject) {
console.log('');
console.log(chalk.cyan('📝 Context-Forge Integration:'));
console.log(chalk.gray(' • Agent commands are in .claude/commands/agents/ to avoid conflicts'));
console.log(chalk.gray(' • Agents can work with your existing PRPs'));
console.log(chalk.gray(' • Use Task("agent-name: description") in Claude Code'));
}
} catch (error) {
spinner.fail('Installation failed');
console.error(chalk.red('Error:'), error.message);
process.exit(1);
}
}