Compare commits

..

1 Commits

Author SHA1 Message Date
don-petry 38cb927e2e
Merge 9924dc6344 into 47991536c5 2026-04-06 11:36:30 -07:00
8 changed files with 129 additions and 420 deletions

View File

@ -1,17 +0,0 @@
# BMad Method - Skill Removal List
# Entries listed here will be removed from IDE skill directories during install/update.
# One entry per line. Lines starting with # are comments.
# Each entry is a skill directory name (canonicalId) that was removed or renamed.
# Removed agents (v6.2.0 - v6.2.2)
bmad-agent-sm
bmad-agent-qa
bmad-agent-quick-flow-solo-dev
# Removed skills (v6.2.0 - v6.2.2)
bmad-create-product-brief
bmad-product-brief-preview
bmad-quick-spec
bmad-quick-flow
bmad-quick-dev-new-preview
bmad-init

View File

@ -1301,14 +1301,6 @@ async function runTests() {
'---\nname: bmad-architect\ndescription: Architect\n---\nOld skill content\n', '---\nname: bmad-architect\ndescription: Architect\n---\nOld skill content\n',
); );
// Add bmad-architect to the existing skill-manifest.csv so cleanup knows it was previously installed
const configDir27 = path.join(installedBmadDir27, '_config');
const existingCsv27 = await fs.readFile(path.join(configDir27, 'skill-manifest.csv'), 'utf8');
await fs.writeFile(
path.join(configDir27, 'skill-manifest.csv'),
existingCsv27.trimEnd() + '\n"bmad-architect","bmad-architect","Architect","bmm","_bmad/bmm/agents/bmad-architect/SKILL.md","true"\n',
);
// Run Claude Code setup (which triggers cleanup then install) // Run Claude Code setup (which triggers cleanup then install)
const ideManager27 = new IdeManager(); const ideManager27 = new IdeManager();
await ideManager27.ensureInitialized(); await ideManager27.ensureInitialized();

View File

@ -19,33 +19,24 @@ const CLIUtils = {
* Display BMAD logo and version using @clack intro + box * Display BMAD logo and version using @clack intro + box
*/ */
async displayLogo() { async displayLogo() {
const version = this.getVersion();
const color = await prompts.getColor(); const color = await prompts.getColor();
const termWidth = process.stdout.columns || 80;
// Full "BMad Method" logo for wide terminals, "BMad" only for narrow // ASCII art logo
const logoWide = [ const logo = [
' ██████╗ ███╗ ███╗ █████╗ ██████╗ ███╗ ███╗███████╗████████╗██╗ ██╗ ██████╗ ██████╗ ™',
'██╔══██╗████╗ ████║██╔══██╗██╔══██╗ ████╗ ████║██╔════╝╚══██╔══╝██║ ██║██╔═══██╗██╔══██╗',
'██████╔╝██╔████╔██║███████║██║ ██║ ██╔████╔██║█████╗ ██║ ███████║██║ ██║██║ ██║',
'██╔══██╗██║╚██╔╝██║██╔══██║██║ ██║ ██║╚██╔╝██║██╔══╝ ██║ ██╔══██║██║ ██║██║ ██║',
'██████╔╝██║ ╚═╝ ██║██║ ██║██████╔╝ ██║ ╚═╝ ██║███████╗ ██║ ██║ ██║╚██████╔╝██████╔╝',
'╚═════╝ ╚═╝ ╚═╝╚═╝ ╚═╝╚═════╝ ╚═╝ ╚═╝╚══════╝ ╚═╝ ╚═╝ ╚═╝ ╚═════╝ ╚═════╝ ',
];
const logoNarrow = [
' ██████╗ ███╗ ███╗ █████╗ ██████╗ ™', ' ██████╗ ███╗ ███╗ █████╗ ██████╗ ™',
' ██╔══██╗████╗ ████║██╔══██╗██╔══██╗', ' ██╔══██╗████╗ ████║██╔══██╗██╔══██╗',
' ██████╔╝██╔████╔██║███████║██║ ██║', ' ██████╔╝██╔████╔██║███████║██║ ██║',
' ██╔══██╗██║╚██╔╝██║██╔══██║██║ ██║', ' ██╔══██╗██║╚██╔╝██║██╔══██║██║ ██║',
' ██████╔╝██║ ╚═╝ ██║██║ ██║██████╔╝', ' ██████╔╝██║ ╚═╝ ██║██║ ██║██████╔╝',
' ╚═════╝ ╚═╝ ╚═╝╚═╝ ╚═╝╚═════╝', ' ╚═════╝ ╚═╝ ╚═╝╚═╝ ╚═╝╚═════╝',
]; ]
.map((line) => color.yellow(line))
.join('\n');
const logoLines = termWidth >= 95 ? logoWide : logoNarrow; const tagline = ' Build More, Architect Dreams';
const logo = logoLines.map((line) => color.blue(line)).join('\n');
const tagline = color.white(' Build More, Architect Dreams\n © BMad Code');
await prompts.box(`${logo}\n${tagline}`, '', { await prompts.box(`${logo}\n${tagline}`, `v${version}`, {
contentAlign: 'center', contentAlign: 'center',
rounded: true, rounded: true,
formatBorder: color.blue, formatBorder: color.blue,

View File

@ -26,44 +26,6 @@ class Installer {
this.bmadFolderName = BMAD_FOLDER_NAME; this.bmadFolderName = BMAD_FOLDER_NAME;
} }
/**
* Read the module version from .claude-plugin/marketplace.json
* Walks up from sourcePath looking for .claude-plugin/marketplace.json
* @param {string} sourcePath - Module source directory
* @returns {string} Version string or empty string
*/
async _getMarketplaceVersion(sourcePath) {
let dir = sourcePath;
for (let i = 0; i < 5; i++) {
const marketplacePath = path.join(dir, '.claude-plugin', 'marketplace.json');
if (await fs.pathExists(marketplacePath)) {
try {
const data = JSON.parse(await fs.readFile(marketplacePath, 'utf8'));
return this._extractMarketplaceVersion(data);
} catch {
return '';
}
}
const parent = path.dirname(dir);
if (parent === dir) break;
dir = parent;
}
return '';
}
/**
* Extract the highest version from marketplace.json plugins array
*/
_extractMarketplaceVersion(data) {
const plugins = data?.plugins;
if (!Array.isArray(plugins) || plugins.length === 0) return '';
let best = '';
for (const p of plugins) {
if (p.version && (!best || p.version > best)) best = p.version;
}
return best;
}
/** /**
* Main installation method * Main installation method
* @param {Object} config - Installation configuration * @param {Object} config - Installation configuration
@ -90,36 +52,9 @@ class Installer {
await this._validateIdeSelection(config); await this._validateIdeSelection(config);
// Capture pre-install module versions for from→to display
const preInstallVersions = new Map();
if (existingInstall.installed) {
const existingModules = await this.manifest.getAllModuleVersions(paths.bmadDir);
for (const mod of existingModules) {
if (mod.name && mod.version) {
preInstallVersions.set(mod.name, mod.version);
}
}
}
// Results collector for consolidated summary // Results collector for consolidated summary
const results = []; const results = [];
const addResult = (step, status, detail = '', meta = {}) => results.push({ step, status, detail, ...meta }); const addResult = (step, status, detail = '') => results.push({ step, status, detail });
// Capture previously installed skill IDs before they get overwritten
const previousSkillIds = new Set();
const prevCsvPath = path.join(paths.bmadDir, '_config', 'skill-manifest.csv');
if (await fs.pathExists(prevCsvPath)) {
try {
const csvParse = require('csv-parse/sync');
const content = await fs.readFile(prevCsvPath, 'utf8');
const records = csvParse.parse(content, { columns: true, skip_empty_lines: true });
for (const r of records) {
if (r.canonicalId) previousSkillIds.add(r.canonicalId);
}
} catch (error) {
await prompts.log.warn(`Failed to parse skill-manifest.csv: ${error.message}`);
}
}
await this._cacheCustomModules(paths, addResult); await this._cacheCustomModules(paths, addResult);
@ -130,7 +65,7 @@ class Installer {
await this._installAndConfigure(config, originalConfig, paths, officialModuleIds, allModules, addResult, officialModules); await this._installAndConfigure(config, originalConfig, paths, officialModuleIds, allModules, addResult, officialModules);
await this._setupIdes(config, allModules, paths, addResult, previousSkillIds); await this._setupIdes(config, allModules, paths, addResult);
const restoreResult = await this._restoreUserFiles(paths, updateState); const restoreResult = await this._restoreUserFiles(paths, updateState);
@ -141,7 +76,6 @@ class Installer {
ides: config.ides, ides: config.ides,
customFiles: restoreResult.customFiles.length > 0 ? restoreResult.customFiles : undefined, customFiles: restoreResult.customFiles.length > 0 ? restoreResult.customFiles : undefined,
modifiedFiles: restoreResult.modifiedFiles.length > 0 ? restoreResult.modifiedFiles : undefined, modifiedFiles: restoreResult.modifiedFiles.length > 0 ? restoreResult.modifiedFiles : undefined,
preInstallVersions,
}); });
return { return {
@ -387,7 +321,7 @@ class Installer {
/** /**
* Set up IDE integrations for each selected IDE. * Set up IDE integrations for each selected IDE.
*/ */
async _setupIdes(config, allModules, paths, addResult, previousSkillIds = new Set()) { async _setupIdes(config, allModules, paths, addResult) {
if (config.skipIde || !config.ides || config.ides.length === 0) return; if (config.skipIde || !config.ides || config.ides.length === 0) return;
await this.ideManager.ensureInitialized(); await this.ideManager.ensureInitialized();
@ -402,7 +336,6 @@ class Installer {
const setupResult = await this.ideManager.setup(ide, paths.projectRoot, paths.bmadDir, { const setupResult = await this.ideManager.setup(ide, paths.projectRoot, paths.bmadDir, {
selectedModules: allModules || [], selectedModules: allModules || [],
verbose: config.verbose, verbose: config.verbose,
previousSkillIds,
}); });
if (setupResult.success) { if (setupResult.success) {
@ -623,7 +556,7 @@ class Installer {
message(`${isQuickUpdate ? 'Updating' : 'Installing'} ${moduleName}...`); message(`${isQuickUpdate ? 'Updating' : 'Installing'} ${moduleName}...`);
const moduleConfig = officialModules.moduleConfigs[moduleName] || {}; const moduleConfig = officialModules.moduleConfigs[moduleName] || {};
const installResult = await officialModules.install( await officialModules.install(
moduleName, moduleName,
paths.bmadDir, paths.bmadDir,
(filePath) => { (filePath) => {
@ -637,12 +570,7 @@ class Installer {
}, },
); );
// Get display name from source module.yaml; version from marketplace.json addResult(`Module: ${moduleName}`, 'ok', isQuickUpdate ? 'updated' : 'installed');
const sourcePath = await officialModules.findModuleSource(moduleName, { silent: true });
const moduleInfo = sourcePath ? await officialModules.getModuleInfo(sourcePath, moduleName, '') : null;
const displayName = moduleInfo?.name || moduleName;
const version = sourcePath ? await this._getMarketplaceVersion(sourcePath) : '';
addResult(displayName, 'ok', '', { moduleCode: moduleName, newVersion: version });
} }
} }
@ -670,11 +598,7 @@ class Installer {
[moduleName]: { ...config.coreConfig, ...result.moduleConfig, ...collectedModuleConfig }, [moduleName]: { ...config.coreConfig, ...result.moduleConfig, ...collectedModuleConfig },
}); });
// Get display name from source module.yaml; version from marketplace.json addResult(`Module: ${moduleName}`, 'ok', isQuickUpdate ? 'updated' : 'installed');
const moduleInfo = await officialModules.getModuleInfo(sourcePath, moduleName, '');
const displayName = moduleInfo?.name || moduleName;
const version = await this._getMarketplaceVersion(sourcePath);
addResult(displayName, 'ok', '', { moduleCode: moduleName, newVersion: version });
} }
} }
@ -1138,10 +1062,23 @@ class Installer {
const selectedIdes = new Set((context.ides || []).map((ide) => String(ide).toLowerCase())); const selectedIdes = new Set((context.ides || []).map((ide) => String(ide).toLowerCase()));
// Build step lines with status indicators // Build step lines with status indicators
const preVersions = context.preInstallVersions || new Map();
const lines = []; const lines = [];
for (const r of results) { for (const r of results) {
const stepLabel = r.step; let stepLabel = null;
if (r.status !== 'ok') {
stepLabel = r.step;
} else if (r.step === 'Core') {
stepLabel = 'BMAD';
} else if (r.step.startsWith('Module: ')) {
stepLabel = r.step;
} else if (selectedIdes.has(String(r.step).toLowerCase())) {
stepLabel = r.step;
}
if (!stepLabel) {
continue;
}
let icon; let icon;
if (r.status === 'ok') { if (r.status === 'ok') {
@ -1151,32 +1088,18 @@ class Installer {
} else { } else {
icon = color.red('\u2717'); icon = color.red('\u2717');
} }
const detail = r.detail ? color.dim(` (${r.detail})`) : '';
// Build version detail for module results
let detail = '';
if (r.moduleCode && r.newVersion) {
const oldVersion = preVersions.get(r.moduleCode);
if (oldVersion && oldVersion === r.newVersion) {
detail = ` (v${r.newVersion}, no change)`;
} else if (oldVersion) {
detail = ` (v${oldVersion} → v${r.newVersion})`;
} else {
detail = ` (v${r.newVersion}, installed)`;
}
} else if (r.detail) {
detail = ` (${r.detail})`;
}
lines.push(` ${icon} ${stepLabel}${detail}`); lines.push(` ${icon} ${stepLabel}${detail}`);
} }
if ((context.ides || []).length === 0) { if ((context.ides || []).length === 0) {
lines.push(` ${color.green('\u2713')} No IDE selected (installed in _bmad only)`); lines.push(` ${color.green('\u2713')} No IDE selected ${color.dim('(installed in _bmad only)')}`);
} }
// Context and warnings // Context and warnings
lines.push(''); lines.push('');
if (context.bmadDir) { if (context.bmadDir) {
lines.push(` Installed to: ${context.bmadDir}`); lines.push(` Installed to: ${color.dim(context.bmadDir)}`);
} }
if (context.customFiles && context.customFiles.length > 0) { if (context.customFiles && context.customFiles.length > 0) {
lines.push(` ${color.cyan(`Custom files preserved: ${context.customFiles.length}`)}`); lines.push(` ${color.cyan(`Custom files preserved: ${context.customFiles.length}`)}`);
@ -1188,18 +1111,17 @@ class Installer {
// Next steps // Next steps
lines.push( lines.push(
'', '',
' Get started:', ' Next steps:',
` 1. Launch your AI agent from your project folder`, ` Read our new Docs Site: ${color.dim('https://docs.bmad-method.org/')}`,
` 2. Not sure what to do? Invoke the ${color.cyan('bmad-help')} skill and ask it what to do!`, ` Join our Discord: ${color.dim('https://discord.gg/gk8jAdXWmj')}`,
'', ` Star us on GitHub: ${color.dim('https://github.com/bmad-code-org/BMAD-METHOD/')}`,
` Blog, Docs and Guides: ${color.blue('https://bmadcode.com/')}`, ` Subscribe on YouTube: ${color.dim('https://www.youtube.com/@BMadCode')}`,
` Community: ${color.blue('https://discord.gg/gk8jAdXWmj')}`,
); );
if (context.ides && context.ides.length > 0) {
lines.push(` Invoke the ${color.cyan('bmad-help')} skill in your IDE Agent to get started`);
}
await prompts.box(lines.join('\n'), 'BMAD is ready to use!', { await prompts.note(lines.join('\n'), 'BMAD is ready to use!');
rounded: true,
formatBorder: color.green,
});
} }
/** /**
@ -1309,7 +1231,6 @@ class Installer {
} }
for (const moduleName of modulesToUpdate) { for (const moduleName of modulesToUpdate) {
if (moduleName === 'core') continue; // Already collected above
const modulePrompted = await quickModules.collectModuleConfigQuick(moduleName, projectDir, true); const modulePrompted = await quickModules.collectModuleConfigQuick(moduleName, projectDir, true);
if (modulePrompted) { if (modulePrompted) {
promptedForNewFields = true; promptedForNewFields = true;

View File

@ -837,13 +837,14 @@ class Manifest {
* @returns {Object} Version info object with version, source, npmPackage, repoUrl * @returns {Object} Version info object with version, source, npmPackage, repoUrl
*/ */
async getModuleVersionInfo(moduleName, bmadDir, moduleSourcePath = null) { async getModuleVersionInfo(moduleName, bmadDir, moduleSourcePath = null) {
const os = require('node:os');
const yaml = require('yaml'); const yaml = require('yaml');
// Resolve source type first, then read version with the correct path context // Built-in modules use BMad version (only core and bmm are in BMAD-METHOD repo)
if (['core', 'bmm'].includes(moduleName)) { if (['core', 'bmm'].includes(moduleName)) {
const version = await this._readMarketplaceVersion(moduleName, moduleSourcePath); const bmadVersion = require(path.join(getProjectRoot(), 'package.json')).version;
return { return {
version, version: bmadVersion,
source: 'built-in', source: 'built-in',
npmPackage: null, npmPackage: null,
repoUrl: null, repoUrl: null,
@ -856,20 +857,42 @@ class Manifest {
const moduleInfo = await extMgr.getModuleByCode(moduleName); const moduleInfo = await extMgr.getModuleByCode(moduleName);
if (moduleInfo) { if (moduleInfo) {
// External module: use moduleSourcePath if provided, otherwise fall back to cache // External module - try to get version from npm registry first, then fall back to cache
const version = await this._readMarketplaceVersion(moduleName, moduleSourcePath); let version = null;
if (moduleInfo.npmPackage) {
// Fetch version from npm registry
try {
version = await this.fetchNpmVersion(moduleInfo.npmPackage);
} catch {
// npm fetch failed, try cache as fallback
}
}
// If npm didn't work, try reading from cached repo's package.json
if (!version) {
const cacheDir = path.join(os.homedir(), '.bmad', 'cache', 'external-modules', moduleName);
const packageJsonPath = path.join(cacheDir, 'package.json');
if (await fs.pathExists(packageJsonPath)) {
try {
const pkg = require(packageJsonPath);
version = pkg.version;
} catch (error) {
await prompts.log.warn(`Failed to read package.json for ${moduleName}: ${error.message}`);
}
}
}
return { return {
version, version: version,
source: 'external', source: 'external',
npmPackage: moduleInfo.npmPackage || null, npmPackage: moduleInfo.npmPackage || null,
repoUrl: moduleInfo.url || null, repoUrl: moduleInfo.url || null,
}; };
} }
// Custom module: resolve path from source or cache before reading version // Custom module - check cache directory
const customSourcePath = moduleSourcePath || path.join(bmadDir, '_config', 'custom', moduleName);
const version = await this._readMarketplaceVersion(moduleName, customSourcePath);
const cacheDir = path.join(bmadDir, '_config', 'custom', moduleName); const cacheDir = path.join(bmadDir, '_config', 'custom', moduleName);
const moduleYamlPath = path.join(cacheDir, 'module.yaml'); const moduleYamlPath = path.join(cacheDir, 'module.yaml');
@ -878,7 +901,7 @@ class Manifest {
const yamlContent = await fs.readFile(moduleYamlPath, 'utf8'); const yamlContent = await fs.readFile(moduleYamlPath, 'utf8');
const moduleConfig = yaml.parse(yamlContent); const moduleConfig = yaml.parse(yamlContent);
return { return {
version: version || moduleConfig.version || null, version: moduleConfig.version || null,
source: 'custom', source: 'custom',
npmPackage: moduleConfig.npmPackage || null, npmPackage: moduleConfig.npmPackage || null,
repoUrl: moduleConfig.repoUrl || null, repoUrl: moduleConfig.repoUrl || null,
@ -890,62 +913,13 @@ class Manifest {
// Unknown module // Unknown module
return { return {
version, version: null,
source: 'unknown', source: 'unknown',
npmPackage: null, npmPackage: null,
repoUrl: null, repoUrl: null,
}; };
} }
/**
* Read version from .claude-plugin/marketplace.json for a module
* @param {string} moduleName - Module code
* @returns {string|null} Version or null
*/
async _readMarketplaceVersion(moduleName, moduleSourcePath = null) {
const os = require('node:os');
let marketplacePath;
if (['core', 'bmm'].includes(moduleName)) {
marketplacePath = path.join(getProjectRoot(), '.claude-plugin', 'marketplace.json');
} else if (moduleSourcePath) {
// Walk up from source path to find marketplace.json
let dir = moduleSourcePath;
for (let i = 0; i < 5; i++) {
const candidate = path.join(dir, '.claude-plugin', 'marketplace.json');
if (await fs.pathExists(candidate)) {
marketplacePath = candidate;
break;
}
const parent = path.dirname(dir);
if (parent === dir) break;
dir = parent;
}
}
// Fallback to external module cache
if (!marketplacePath) {
const cacheDir = path.join(os.homedir(), '.bmad', 'cache', 'external-modules', moduleName);
marketplacePath = path.join(cacheDir, '.claude-plugin', 'marketplace.json');
}
try {
if (await fs.pathExists(marketplacePath)) {
const data = JSON.parse(await fs.readFile(marketplacePath, 'utf8'));
const plugins = data?.plugins;
if (!Array.isArray(plugins) || plugins.length === 0) return null;
let best = null;
for (const p of plugins) {
if (p.version && (!best || p.version > best)) best = p.version;
}
return best;
}
} catch {
// ignore
}
return null;
}
/** /**
* Fetch latest version from npm for a package * Fetch latest version from npm for a package
* @param {string} packageName - npm package name * @param {string} packageName - npm package name

View File

@ -86,7 +86,7 @@ class ConfigDrivenIdeSetup {
if (!options.silent) await prompts.log.info(`Setting up ${this.name}...`); if (!options.silent) await prompts.log.info(`Setting up ${this.name}...`);
// Clean up any old BMAD installation first // Clean up any old BMAD installation first
await this.cleanup(projectDir, options, bmadDir); await this.cleanup(projectDir, options);
if (!this.installerConfig) { if (!this.installerConfig) {
return { success: false, reason: 'no-config' }; return { success: false, reason: 'no-config' };
@ -215,34 +215,15 @@ class ConfigDrivenIdeSetup {
* Cleanup IDE configuration * Cleanup IDE configuration
* @param {string} projectDir - Project directory * @param {string} projectDir - Project directory
*/ */
async cleanup(projectDir, options = {}, bmadDir = null) { async cleanup(projectDir, options = {}) {
const resolvedBmadDir = bmadDir || (await this._findBmadDir(projectDir));
// Build removal set: previously installed skills + removals.txt entries
let removalSet;
if (options.previousSkillIds && options.previousSkillIds.size > 0) {
// Install/update flow: use pre-captured skill IDs (before manifest was overwritten)
removalSet = new Set(options.previousSkillIds);
if (resolvedBmadDir) {
const removals = await this.loadRemovalLists(resolvedBmadDir);
for (const entry of removals) removalSet.add(entry);
}
} else if (resolvedBmadDir) {
// Uninstall flow: read from current skill-manifest.csv + removals.txt
removalSet = await this._buildUninstallSet(resolvedBmadDir);
} else {
removalSet = new Set();
}
// Migrate legacy target directories (e.g. .opencode/agent → .opencode/agents) // Migrate legacy target directories (e.g. .opencode/agent → .opencode/agents)
// Legacy dirs are abandoned entirely, so use prefix matching (null removalSet)
if (this.installerConfig?.legacy_targets) { if (this.installerConfig?.legacy_targets) {
if (!options.silent) await prompts.log.message(' Migrating legacy directories...'); if (!options.silent) await prompts.log.message(' Migrating legacy directories...');
for (const legacyDir of this.installerConfig.legacy_targets) { for (const legacyDir of this.installerConfig.legacy_targets) {
if (this.isGlobalPath(legacyDir)) { if (this.isGlobalPath(legacyDir)) {
await this.warnGlobalLegacy(legacyDir, options); await this.warnGlobalLegacy(legacyDir, options);
} else { } else {
await this.cleanupTarget(projectDir, legacyDir, options, null); await this.cleanupTarget(projectDir, legacyDir, options);
await this.removeEmptyParents(projectDir, legacyDir); await this.removeEmptyParents(projectDir, legacyDir);
} }
} }
@ -263,9 +244,9 @@ class ConfigDrivenIdeSetup {
await this.cleanupRovoDevPrompts(projectDir, options); await this.cleanupRovoDevPrompts(projectDir, options);
} }
// Clean current target directory // Clean target directory
if (this.installerConfig?.target_dir) { if (this.installerConfig?.target_dir) {
await this.cleanupTarget(projectDir, this.installerConfig.target_dir, options, removalSet); await this.cleanupTarget(projectDir, this.installerConfig.target_dir, options);
} }
} }
@ -305,117 +286,23 @@ class ConfigDrivenIdeSetup {
} }
/** /**
* Find the _bmad directory in a project * Cleanup a specific target directory
* @param {string} projectDir - Project directory
* @returns {string|null} Path to bmad dir or null
*/
async _findBmadDir(projectDir) {
const bmadDir = path.join(projectDir, BMAD_FOLDER_NAME);
return (await fs.pathExists(bmadDir)) ? bmadDir : null;
}
/**
* Build the full set of entries to remove for uninstall.
* Reads skill-manifest.csv to know exactly what was installed, plus removal lists.
* @param {string} bmadDir - BMAD installation directory
* @returns {Set<string>} Set of entries to remove
*/
async _buildUninstallSet(bmadDir) {
const removals = await this.loadRemovalLists(bmadDir);
// Also add all currently installed skills from skill-manifest.csv
const csvPath = path.join(bmadDir, '_config', 'skill-manifest.csv');
try {
if (await fs.pathExists(csvPath)) {
const content = await fs.readFile(csvPath, 'utf8');
const records = csv.parse(content, { columns: true, skip_empty_lines: true });
for (const record of records) {
if (record.canonicalId) {
removals.add(record.canonicalId);
}
}
}
} catch {
// If we can't read the manifest, we still have the removal lists
}
return removals;
}
/**
* Load removal lists from all module sources in the bmad directory.
* Each module can have an optional removals.txt listing entries to remove.
* @param {string} bmadDir - BMAD installation directory
* @returns {Set<string>} Set of entries to remove
*/
async loadRemovalLists(bmadDir) {
const removals = new Set();
const { getProjectRoot } = require('../project-root');
// Read project-level removals.txt (covers core and bmm)
const projectRemovalsPath = path.join(getProjectRoot(), 'removals.txt');
await this._readRemovalFile(projectRemovalsPath, removals);
// Read per-module removals.txt from installed module directories
try {
const entries = await fs.readdir(bmadDir);
for (const entry of entries) {
if (entry.startsWith('_')) continue;
const removalPath = path.join(bmadDir, entry, 'removals.txt');
await this._readRemovalFile(removalPath, removals);
}
} catch {
// bmadDir may not exist yet on fresh install
}
return removals;
}
/**
* Read a removals.txt file and add entries to the set
* @param {string} filePath - Path to removals.txt
* @param {Set<string>} removals - Set to add entries to
*/
async _readRemovalFile(filePath, removals) {
try {
if (await fs.pathExists(filePath)) {
const content = await fs.readFile(filePath, 'utf8');
for (const line of content.split('\n')) {
const trimmed = line.trim();
if (trimmed && !trimmed.startsWith('#')) {
removals.add(trimmed);
}
}
}
} catch {
// Optional file — ignore errors
}
}
/**
* Cleanup a specific target directory.
* When removalSet is provided, only removes entries in that set.
* When removalSet is null (legacy dirs), removes all bmad-prefixed entries.
* @param {string} projectDir - Project directory * @param {string} projectDir - Project directory
* @param {string} targetDir - Target directory to clean * @param {string} targetDir - Target directory to clean
* @param {Object} options - Cleanup options
* @param {Set<string>|null} removalSet - Entries to remove, or null for legacy prefix matching
*/ */
async cleanupTarget(projectDir, targetDir, options = {}, removalSet = new Set()) { async cleanupTarget(projectDir, targetDir, options = {}) {
const targetPath = path.join(projectDir, targetDir); const targetPath = path.join(projectDir, targetDir);
if (!(await fs.pathExists(targetPath))) { if (!(await fs.pathExists(targetPath))) {
return; return;
} }
if (removalSet && removalSet.size === 0) { // Remove all bmad* files
return;
}
let entries; let entries;
try { try {
entries = await fs.readdir(targetPath); entries = await fs.readdir(targetPath);
} catch { } catch {
// Directory exists but can't be read - skip cleanup
return; return;
} }
@ -426,26 +313,23 @@ class ConfigDrivenIdeSetup {
let removedCount = 0; let removedCount = 0;
for (const entry of entries) { for (const entry of entries) {
if (!entry || typeof entry !== 'string') continue; if (!entry || typeof entry !== 'string') {
continue;
// Always preserve bmad-os-* utility skills regardless of cleanup mode }
if (entry.startsWith('bmad-os-')) continue; if (entry.startsWith('bmad') && !entry.startsWith('bmad-os-')) {
const entryPath = path.join(targetPath, entry);
// Surgical removal from set, or legacy prefix matching when set is null
const shouldRemove = removalSet ? removalSet.has(entry) : entry.startsWith('bmad');
if (shouldRemove) {
try { try {
await fs.remove(path.join(targetPath, entry)); await fs.remove(entryPath);
removedCount++; removedCount++;
} catch { } catch {
// Skip entries that can't be removed // Skip entries that can't be removed (broken symlinks, permission errors)
} }
} }
} }
// Only log cleanup when it's not a routine reinstall (legacy dir cleanup or actual removals) if (removedCount > 0 && !options.silent) {
// Suppress for current target_dir since it's always cleaned before a fresh write await prompts.log.message(` Cleaned ${removedCount} BMAD files from ${targetDir}`);
}
// Remove empty directory after cleanup // Remove empty directory after cleanup
if (removedCount > 0) { if (removedCount > 0) {
@ -455,7 +339,7 @@ class ConfigDrivenIdeSetup {
await fs.remove(targetPath); await fs.remove(targetPath);
} }
} catch { } catch {
// Directory may already be gone or in use // Directory may already be gone or in use — skip
} }
} }
} }

View File

@ -6,25 +6,32 @@
startMessage: | startMessage: |
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
Agile AI-Driven Development. Powered by BMad Core and a growing module ecosystem. 🎉 V6 IS HERE! Welcome to BMad Method V6 - Official Stable Release!
Install official and community modules during setup to customize your experience.
🌟 100% free. 100% open source. Always. The BMad Method is now a Platform powered by the BMad Method Core and Module Ecosystem!
No paywalls. No gated content. Knowledge shared, not sold. - Select and install modules during setup - customize your experience
- New BMad Method for Agile AI-Driven Development (the evolution of V4)
- Exciting new modules available during installation, with community modules coming soon
- Documentation: https://docs.bmad-method.org
🌐 CONNECT: 🌟 BMad is 100% free and open source.
Website: https://bmadcode.com/ - No gated Discord. No paywalls. No gated content.
Discord: https://discord.gg/gk8jAdXWmj - We believe in empowering everyone, not just those who can pay.
YouTube: https://www.youtube.com/@BMadCode - Knowledge should be shared, not sold.
X: https://x.com/BMadCode
Facebook: https://facebook.com/@BMadCode
⭐ SUPPORT THE PROJECT: 🎤 SPEAKING & MEDIA:
Star us: https://github.com/bmad-code-org/BMAD-METHOD/ - Available for conferences, podcasts, and media appearances
Donate: https://buymeacoffee.com/bmad - Topics: AI-Native Transformation, Spec and Context Engineering, BMad Method
Corporate sponsorship and speaking inquiries: contact@bmadcode.com - For speaking inquiries or interviews, reach out to BMad on Discord!
Docs, blog, and latest updates: https://bmadcode.com/ ⭐ HELP US GROW:
- Star us on GitHub: https://github.com/bmad-code-org/BMAD-METHOD/
- Subscribe on YouTube: https://www.youtube.com/@BMadCode
- Free Community and Support: https://discord.gg/gk8jAdXWmj
- Donate: https://buymeacoffee.com/bmad
- Corporate Sponsorship available
Latest updates: https://github.com/bmad-code-org/BMAD-METHOD/blob/main/CHANGELOG.md
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━

View File

@ -4,50 +4,8 @@ const fs = require('fs-extra');
const { CLIUtils } = require('./cli-utils'); const { CLIUtils } = require('./cli-utils');
const { CustomHandler } = require('./custom-handler'); const { CustomHandler } = require('./custom-handler');
const { ExternalModuleManager } = require('./modules/external-manager'); const { ExternalModuleManager } = require('./modules/external-manager');
const { getProjectRoot } = require('./project-root');
const prompts = require('./prompts'); const prompts = require('./prompts');
/**
* Read module version from .claude-plugin/marketplace.json
* @param {string} moduleCode - Module code (e.g., 'core', 'bmm', 'cis')
* @returns {string} Version string or empty string
*/
async function getMarketplaceVersion(moduleCode) {
let marketplacePath;
if (moduleCode === 'core' || moduleCode === 'bmm') {
marketplacePath = path.join(getProjectRoot(), '.claude-plugin', 'marketplace.json');
} else {
const cacheDir = path.join(os.homedir(), '.bmad', 'cache', 'external-modules', moduleCode);
marketplacePath = path.join(cacheDir, '.claude-plugin', 'marketplace.json');
}
try {
if (await fs.pathExists(marketplacePath)) {
const data = JSON.parse(await fs.readFile(marketplacePath, 'utf8'));
return _extractMarketplaceVersion(data);
}
} catch {
// ignore
}
return '';
}
/**
* Extract the highest version from marketplace.json plugins array.
* Handles multiple plugins per file safely.
* @param {Object} data - Parsed marketplace.json
* @returns {string} Version string or empty string
*/
function _extractMarketplaceVersion(data) {
const plugins = data?.plugins;
if (!Array.isArray(plugins) || plugins.length === 0) return '';
// Use the highest version across all plugins in the file
let best = '';
for (const p of plugins) {
if (p.version && (!best || p.version > best)) best = p.version;
}
return best;
}
// Separator class for visual grouping in select/multiselect prompts // Separator class for visual grouping in select/multiselect prompts
// Note: @clack/prompts doesn't support separators natively, they are filtered out // Note: @clack/prompts doesn't support separators natively, they are filtered out
class Separator { class Separator {
@ -112,14 +70,17 @@ class UI {
if (hasExistingInstall) { if (hasExistingInstall) {
// Get version information // Get version information
const { existingInstall, bmadDir } = await this.getExistingInstallation(confirmedDirectory); const { existingInstall, bmadDir } = await this.getExistingInstallation(confirmedDirectory);
const packageJsonPath = path.join(__dirname, '../../package.json');
const currentVersion = require(packageJsonPath).version;
const installedVersion = existingInstall.installed ? existingInstall.version || 'unknown' : 'unknown';
// Build menu choices dynamically // Build menu choices dynamically
const choices = []; const choices = [];
// Always show Quick Update first (allows refreshing installation even on same version) // Always show Quick Update first (allows refreshing installation even on same version)
if (existingInstall.installed) { if (installedVersion !== 'unknown') {
choices.push({ choices.push({
name: 'Quick Update', name: `Quick Update (v${installedVersion} → v${currentVersion})`,
value: 'quick-update', value: 'quick-update',
}); });
} }
@ -919,18 +880,14 @@ class UI {
const lockedValues = ['core']; const lockedValues = ['core'];
// Core module is always installed — show it locked at the top // Core module is always installed — show it locked at the top
const coreVersion = await getMarketplaceVersion('core'); allOptions.push({ label: 'BMad Core Module', value: 'core', hint: 'Core configuration and shared resources' });
const coreLabel = coreVersion ? `BMad Core Module (v${coreVersion})` : 'BMad Core Module';
allOptions.push({ label: coreLabel, value: 'core', hint: 'Core configuration and shared resources' });
initialValues.push('core'); initialValues.push('core');
// Helper to build module entry with proper sorting and selection // Helper to build module entry with proper sorting and selection
const buildModuleEntry = async (mod, value, group) => { const buildModuleEntry = (mod, value, group) => {
const isInstalled = installedModuleIds.has(value); const isInstalled = installedModuleIds.has(value);
const version = await getMarketplaceVersion(value);
const label = version ? `${mod.name} (v${version})` : mod.name;
return { return {
label, label: mod.name,
value, value,
hint: mod.description || group, hint: mod.description || group,
// Pre-select only if already installed (not on fresh install) // Pre-select only if already installed (not on fresh install)
@ -942,7 +899,7 @@ class UI {
const localEntries = []; const localEntries = [];
for (const mod of localModules) { for (const mod of localModules) {
if (!mod.isCustom && mod.id !== 'core') { if (!mod.isCustom && mod.id !== 'core') {
const entry = await buildModuleEntry(mod, mod.id, 'Local'); const entry = buildModuleEntry(mod, mod.id, 'Local');
localEntries.push(entry); localEntries.push(entry);
if (entry.selected) { if (entry.selected) {
initialValues.push(mod.id); initialValues.push(mod.id);
@ -955,7 +912,7 @@ class UI {
const officialModules = []; const officialModules = [];
for (const mod of externalModules) { for (const mod of externalModules) {
if (mod.type === 'bmad-org') { if (mod.type === 'bmad-org') {
const entry = await buildModuleEntry(mod, mod.code, 'Official'); const entry = buildModuleEntry(mod, mod.code, 'Official');
officialModules.push(entry); officialModules.push(entry);
if (entry.selected) { if (entry.selected) {
initialValues.push(mod.code); initialValues.push(mod.code);
@ -968,7 +925,7 @@ class UI {
const communityModules = []; const communityModules = [];
for (const mod of externalModules) { for (const mod of externalModules) {
if (mod.type === 'community') { if (mod.type === 'community') {
const entry = await buildModuleEntry(mod, mod.code, 'Community'); const entry = buildModuleEntry(mod, mod.code, 'Community');
communityModules.push(entry); communityModules.push(entry);
if (entry.selected) { if (entry.selected) {
initialValues.push(mod.code); initialValues.push(mod.code);