267 lines
9.0 KiB
JavaScript
Raw Normal View History

const path = require('node:path');
const fs = require('fs-extra');
const chalk = require('chalk');
const yaml = require('js-yaml');
const { FileOps } = require('../../../lib/file-ops');
/**
* Handler for custom content (custom.yaml)
* Installs custom agents and workflows without requiring a full module structure
*/
class CustomHandler {
constructor() {
this.fileOps = new FileOps();
}
/**
* Find all custom.yaml files in the project
* @param {string} projectRoot - Project root directory
* @returns {Array} List of custom content paths
*/
async findCustomContent(projectRoot) {
const customPaths = [];
// Helper function to recursively scan directories
async function scanDirectory(dir, excludePaths = []) {
try {
const entries = await fs.readdir(dir, { withFileTypes: true });
for (const entry of entries) {
const fullPath = path.join(dir, entry.name);
// Skip hidden directories and common exclusions
if (
entry.name.startsWith('.') ||
entry.name === 'node_modules' ||
entry.name === 'dist' ||
entry.name === 'build' ||
entry.name === '.git' ||
entry.name === 'bmad'
) {
continue;
}
// Skip excluded paths
if (excludePaths.some((exclude) => fullPath.startsWith(exclude))) {
continue;
}
if (entry.isDirectory()) {
// Recursively scan subdirectories
await scanDirectory(fullPath, excludePaths);
} else if (entry.name === 'custom.yaml') {
// Found a custom.yaml file
customPaths.push(fullPath);
}
}
} catch {
// Ignore errors (e.g., permission denied)
}
}
// Scan the entire project, but exclude source directories
await scanDirectory(projectRoot, [path.join(projectRoot, 'src'), path.join(projectRoot, 'tools'), path.join(projectRoot, 'test')]);
return customPaths;
}
/**
* Get custom content info from a custom.yaml file
* @param {string} customYamlPath - Path to custom.yaml file
* @returns {Object|null} Custom content info
*/
async getCustomInfo(customYamlPath) {
try {
const configContent = await fs.readFile(customYamlPath, 'utf8');
// Try to parse YAML with error handling
let config;
try {
config = yaml.load(configContent);
} catch (parseError) {
console.warn(chalk.yellow(`Warning: YAML parse error in ${customYamlPath}:`, parseError.message));
return null;
}
const customDir = path.dirname(customYamlPath);
const relativePath = path.relative(process.cwd(), customDir);
return {
id: config.code || path.basename(customDir),
name: config.name || `Custom: ${path.basename(customDir)}`,
description: config.description || 'Custom agents and workflows',
path: customDir,
relativePath: relativePath,
defaultSelected: config.default_selected === true,
config: config,
};
} catch (error) {
console.warn(chalk.yellow(`Warning: Failed to read ${customYamlPath}:`, error.message));
return null;
}
}
/**
* Install custom content
* @param {string} customPath - Path to custom content directory
* @param {string} bmadDir - Target bmad directory
* @param {Object} config - Configuration from custom.yaml
* @param {Function} fileTrackingCallback - Optional callback to track installed files
* @returns {Object} Installation result
*/
async install(customPath, bmadDir, config, fileTrackingCallback = null) {
const results = {
agentsInstalled: 0,
workflowsInstalled: 0,
filesCopied: 0,
preserved: 0,
errors: [],
};
try {
// Create custom directories in bmad
const bmadCustomDir = path.join(bmadDir, 'custom');
const bmadAgentsDir = path.join(bmadCustomDir, 'agents');
const bmadWorkflowsDir = path.join(bmadCustomDir, 'workflows');
await fs.ensureDir(bmadCustomDir);
await fs.ensureDir(bmadAgentsDir);
await fs.ensureDir(bmadWorkflowsDir);
// Process agents - copy entire agents directory structure
const agentsDir = path.join(customPath, 'agents');
if (await fs.pathExists(agentsDir)) {
await this.copyDirectory(agentsDir, bmadAgentsDir, results, fileTrackingCallback, config);
// Count agent files
const agentFiles = await this.findFilesRecursively(agentsDir, ['.agent.yaml', '.md']);
results.agentsInstalled = agentFiles.length;
}
// Process workflows - copy entire workflows directory structure
const workflowsDir = path.join(customPath, 'workflows');
if (await fs.pathExists(workflowsDir)) {
await this.copyDirectory(workflowsDir, bmadWorkflowsDir, results, fileTrackingCallback, config);
// Count workflow files
const workflowFiles = await this.findFilesRecursively(workflowsDir, ['.md']);
results.workflowsInstalled = workflowFiles.length;
}
// Process any additional files at root
const entries = await fs.readdir(customPath, { withFileTypes: true });
for (const entry of entries) {
if (entry.isFile() && entry.name !== 'custom.yaml' && !entry.name.startsWith('.') && !entry.name.endsWith('.md')) {
// Skip .md files at root as they're likely docs
const sourcePath = path.join(customPath, entry.name);
const targetPath = path.join(bmadCustomDir, entry.name);
try {
// Check if file already exists
if (await fs.pathExists(targetPath)) {
// File already exists, preserve it
results.preserved = (results.preserved || 0) + 1;
} else {
await fs.copy(sourcePath, targetPath);
results.filesCopied++;
if (fileTrackingCallback) {
fileTrackingCallback(targetPath);
}
}
} catch (error) {
results.errors.push(`Failed to copy file ${entry.name}: ${error.message}`);
}
}
}
} catch (error) {
results.errors.push(`Installation failed: ${error.message}`);
}
return results;
}
/**
* Find all files with specific extensions recursively
* @param {string} dir - Directory to search
* @param {Array} extensions - File extensions to match
* @returns {Array} List of matching files
*/
async findFilesRecursively(dir, extensions) {
const files = [];
async function search(currentDir) {
const entries = await fs.readdir(currentDir, { withFileTypes: true });
for (const entry of entries) {
const fullPath = path.join(currentDir, entry.name);
if (entry.isDirectory()) {
await search(fullPath);
} else if (extensions.some((ext) => entry.name.endsWith(ext))) {
files.push(fullPath);
}
}
}
await search(dir);
return files;
}
/**
* Recursively copy a directory
* @param {string} sourceDir - Source directory
* @param {string} targetDir - Target directory
* @param {Object} results - Results object to update
* @param {Function} fileTrackingCallback - Optional callback
* @param {Object} config - Configuration for placeholder replacement
*/
async copyDirectory(sourceDir, targetDir, results, fileTrackingCallback, config) {
await fs.ensureDir(targetDir);
const entries = await fs.readdir(sourceDir, { withFileTypes: true });
for (const entry of entries) {
const sourcePath = path.join(sourceDir, entry.name);
const targetPath = path.join(targetDir, entry.name);
if (entry.isDirectory()) {
await this.copyDirectory(sourcePath, targetPath, results, fileTrackingCallback, config);
} else {
try {
// Check if file already exists
if (await fs.pathExists(targetPath)) {
// File already exists, preserve it
results.preserved = (results.preserved || 0) + 1;
} else {
// Copy with placeholder replacement for text files
const textExtensions = ['.md', '.yaml', '.yml', '.txt', '.json'];
if (textExtensions.some((ext) => entry.name.endsWith(ext))) {
await this.fileOps.copyFile(sourcePath, targetPath, {
bmadFolder: config.bmad_folder || 'bmad',
userName: config.user_name || 'User',
communicationLanguage: config.communication_language || 'English',
outputFolder: config.output_folder || 'docs',
});
} else {
await fs.copy(sourcePath, targetPath);
}
results.filesCopied++;
if (fileTrackingCallback) {
fileTrackingCallback(targetPath);
}
}
if (entry.name.endsWith('.md')) {
results.workflowsInstalled++;
}
} catch (error) {
results.errors.push(`Failed to copy ${entry.name}: ${error.message}`);
}
}
}
}
}
module.exports = { CustomHandler };