bmad-plus 0.7.3 → 0.7.5

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,489 @@
1
+ /**
2
+ * BMAD+ Autoconfig Command
3
+ * Smart project bootstrap — analyzes existing projects or guides new ones.
4
+ * Auto-detects stack, selects best packs, installs, populates memory.
5
+ *
6
+ * Author: Laurent Rochetta
7
+ */
8
+
9
+ const path = require('node:path');
10
+ const fs = require('node:fs');
11
+ const os = require('node:os');
12
+ const fsExtra = require('fs-extra');
13
+ const clack = require('@clack/prompts');
14
+ const pc = require('picocolors');
15
+
16
+ // ── Project Analysis Engine ──
17
+
18
+ function detectStack(dir) {
19
+ const result = {
20
+ language: null,
21
+ framework: null,
22
+ runtime: null,
23
+ packageManager: null,
24
+ hasTypeScript: false,
25
+ };
26
+
27
+ // Package.json analysis
28
+ const pkgPath = path.join(dir, 'package.json');
29
+ if (fs.existsSync(pkgPath)) {
30
+ try {
31
+ const pkg = JSON.parse(fs.readFileSync(pkgPath, 'utf8'));
32
+ const deps = { ...pkg.dependencies, ...pkg.devDependencies };
33
+ result.runtime = 'Node.js';
34
+ result.language = deps['typescript'] || fs.existsSync(path.join(dir, 'tsconfig.json')) ? 'TypeScript' : 'JavaScript';
35
+ result.hasTypeScript = result.language === 'TypeScript';
36
+
37
+ // Framework detection
38
+ if (deps['next']) result.framework = 'Next.js';
39
+ else if (deps['nuxt']) result.framework = 'Nuxt';
40
+ else if (deps['@angular/core']) result.framework = 'Angular';
41
+ else if (deps['react']) result.framework = 'React';
42
+ else if (deps['vue']) result.framework = 'Vue.js';
43
+ else if (deps['svelte']) result.framework = 'Svelte';
44
+ else if (deps['express']) result.framework = 'Express';
45
+ else if (deps['fastify']) result.framework = 'Fastify';
46
+ else if (deps['hono']) result.framework = 'Hono';
47
+ else if (deps['electron']) result.framework = 'Electron';
48
+ else if (deps['tauri']) result.framework = 'Tauri';
49
+ else if (deps['react-native']) result.framework = 'React Native';
50
+
51
+ // Package manager
52
+ if (fs.existsSync(path.join(dir, 'pnpm-lock.yaml'))) result.packageManager = 'pnpm';
53
+ else if (fs.existsSync(path.join(dir, 'yarn.lock'))) result.packageManager = 'yarn';
54
+ else if (fs.existsSync(path.join(dir, 'bun.lockb'))) result.packageManager = 'bun';
55
+ else result.packageManager = 'npm';
56
+ } catch {}
57
+ }
58
+
59
+ // Other languages
60
+ if (!result.runtime) {
61
+ if (fs.existsSync(path.join(dir, 'Cargo.toml'))) { result.language = 'Rust'; result.runtime = 'Rust'; }
62
+ else if (fs.existsSync(path.join(dir, 'pyproject.toml')) || fs.existsSync(path.join(dir, 'requirements.txt'))) { result.language = 'Python'; result.runtime = 'Python'; }
63
+ else if (fs.existsSync(path.join(dir, 'go.mod'))) { result.language = 'Go'; result.runtime = 'Go'; }
64
+ else if (fs.existsSync(path.join(dir, 'composer.json'))) { result.language = 'PHP'; result.runtime = 'PHP'; }
65
+ else if (fs.existsSync(path.join(dir, 'Gemfile'))) { result.language = 'Ruby'; result.runtime = 'Ruby'; }
66
+ else if (fs.existsSync(path.join(dir, 'pom.xml')) || fs.existsSync(path.join(dir, 'build.gradle'))) { result.language = 'Java'; result.runtime = 'JVM'; }
67
+ }
68
+
69
+ return result;
70
+ }
71
+
72
+ function analyzeStructure(dir) {
73
+ const structure = {
74
+ hasSrc: false,
75
+ hasTests: false,
76
+ hasDocs: false,
77
+ hasCI: false,
78
+ hasDocker: false,
79
+ hasConfig: false,
80
+ hasLicense: false,
81
+ hasReadme: false,
82
+ hasGit: false,
83
+ hasEnv: false,
84
+ hasBmad: false,
85
+ hasIdeConfigs: [],
86
+ fileCount: 0,
87
+ directories: [],
88
+ };
89
+
90
+ try {
91
+ const entries = fs.readdirSync(dir, { withFileTypes: true });
92
+
93
+ for (const entry of entries) {
94
+ if (entry.isDirectory()) {
95
+ const name = entry.name.toLowerCase();
96
+ if (name === 'src' || name === 'lib' || name === 'app') structure.hasSrc = true;
97
+ if (name === 'tests' || name === 'test' || name === '__tests__' || name === 'spec') structure.hasTests = true;
98
+ if (name === 'docs' || name === 'doc' || name === 'documentation') structure.hasDocs = true;
99
+ if (name === '.github' || name === '.gitlab-ci' || name === '.circleci') structure.hasCI = true;
100
+ if (name === '.agents' || name === '_bmad') structure.hasBmad = true;
101
+ if (!name.startsWith('.') && name !== 'node_modules') structure.directories.push(entry.name);
102
+ } else {
103
+ structure.fileCount++;
104
+ const name = entry.name;
105
+ if (name === 'Dockerfile' || name === 'docker-compose.yml' || name === 'docker-compose.yaml') structure.hasDocker = true;
106
+ if (name === 'LICENSE' || name === 'LICENSE.md') structure.hasLicense = true;
107
+ if (name === 'README.md' || name === 'readme.md') structure.hasReadme = true;
108
+ if (name === '.env' || name === '.env.local') structure.hasEnv = true;
109
+ if (name === '.gitignore') structure.hasGit = true;
110
+ if (name === 'CLAUDE.md') structure.hasIdeConfigs.push('claude-code');
111
+ if (name === 'GEMINI.md') structure.hasIdeConfigs.push('gemini-cli');
112
+ if (name === 'AGENTS.md') structure.hasIdeConfigs.push('codex-cli');
113
+ }
114
+ }
115
+
116
+ // Check for .git directory
117
+ if (fs.existsSync(path.join(dir, '.git'))) structure.hasGit = true;
118
+ } catch {}
119
+
120
+ return structure;
121
+ }
122
+
123
+ function calculateHealth(structure) {
124
+ const checks = [
125
+ { name: 'Source code', pass: structure.hasSrc, weight: 2 },
126
+ { name: 'Tests', pass: structure.hasTests, weight: 2 },
127
+ { name: 'Documentation', pass: structure.hasDocs, weight: 1 },
128
+ { name: 'CI/CD', pass: structure.hasCI, weight: 1 },
129
+ { name: 'Git', pass: structure.hasGit, weight: 1 },
130
+ { name: 'README', pass: structure.hasReadme, weight: 1 },
131
+ { name: 'License', pass: structure.hasLicense, weight: 1 },
132
+ { name: 'Docker', pass: structure.hasDocker, weight: 1 },
133
+ ];
134
+
135
+ const totalWeight = checks.reduce((sum, c) => sum + c.weight, 0);
136
+ const score = checks.reduce((sum, c) => sum + (c.pass ? c.weight : 0), 0);
137
+ const pct = Math.round((score / totalWeight) * 100);
138
+
139
+ return { pct, checks };
140
+ }
141
+
142
+ function recommendPacks(stack, structure, health) {
143
+ const packs = ['core', 'memory']; // Always
144
+ const reasons = {
145
+ core: 'Essential multi-role agents (Atlas, Forge, Sentinel, Nexus)',
146
+ memory: 'Persistent brain for context continuity across sessions',
147
+ };
148
+
149
+ // Dev Studio — if no docs or complex project
150
+ if (!structure.hasDocs || structure.directories.length > 5) {
151
+ packs.push('dev-studio');
152
+ reasons['dev-studio'] = !structure.hasDocs
153
+ ? 'No docs/ found — Huldah (Tech Writer) will help document'
154
+ : 'Complex project — full SDLC pipeline recommended';
155
+ }
156
+
157
+ // Shield — if CI/CD exists or Docker
158
+ if (structure.hasCI || structure.hasDocker) {
159
+ packs.push('shield');
160
+ reasons.shield = 'CI/CD and/or Docker detected — GRC compliance agents recommended';
161
+ }
162
+
163
+ // SEO — if it looks like a web project
164
+ if (stack.framework && ['Next.js', 'Nuxt', 'Angular', 'React', 'Vue.js', 'Svelte'].includes(stack.framework)) {
165
+ packs.push('seo');
166
+ reasons.seo = `${stack.framework} web app detected — SEO audit agents recommended`;
167
+ }
168
+
169
+ return { packs, reasons };
170
+ }
171
+
172
+ function generateRecommendations(stack, structure, health) {
173
+ const recs = [];
174
+
175
+ if (!structure.hasTests) {
176
+ recs.push({ agent: 'Sentinel', action: 'set up a test framework and write initial tests', priority: 'high' });
177
+ }
178
+ if (!structure.hasDocs) {
179
+ recs.push({ agent: 'Forge', action: 'document the project architecture and API', priority: 'medium' });
180
+ }
181
+ if (!structure.hasCI) {
182
+ recs.push({ agent: 'Forge', action: 'set up CI/CD pipeline', priority: 'medium' });
183
+ }
184
+ if (health.pct < 60) {
185
+ recs.push({ agent: 'Sentinel', action: 'review code quality and project health', priority: 'high' });
186
+ }
187
+
188
+ // Context-specific
189
+ if (structure.hasSrc) {
190
+ recs.push({ agent: 'Forge', action: 'continue developing the current module', priority: 'normal' });
191
+ }
192
+
193
+ // Memory
194
+ recs.push({ agent: 'Zecher', action: 'consolidate session memory', priority: 'normal' });
195
+
196
+ return recs;
197
+ }
198
+
199
+ function getProjectName(dir) {
200
+ try {
201
+ const pkgPath = path.join(dir, 'package.json');
202
+ if (fs.existsSync(pkgPath)) {
203
+ const pkg = JSON.parse(fs.readFileSync(pkgPath, 'utf8'));
204
+ if (pkg.name) return pkg.name;
205
+ }
206
+ } catch {}
207
+ return path.basename(dir);
208
+ }
209
+
210
+ // ── New Project Wizard ──
211
+
212
+ async function newProjectWizard() {
213
+ const projectType = await clack.select({
214
+ message: 'What type of project?',
215
+ options: [
216
+ { value: 'web', label: '🌐 Web Application', hint: 'SPA, SSR, static site' },
217
+ { value: 'api', label: '⚡ API / Backend', hint: 'REST, GraphQL, microservices' },
218
+ { value: 'cli', label: '💻 CLI Tool', hint: 'command-line application' },
219
+ { value: 'mobile', label: '📱 Mobile App', hint: 'React Native, Flutter' },
220
+ { value: 'library', label: '📦 Library / Package', hint: 'npm, PyPI, crate' },
221
+ { value: 'other', label: '🔧 Other', hint: 'custom project' },
222
+ ],
223
+ });
224
+
225
+ if (clack.isCancel(projectType)) return null;
226
+
227
+ const description = await clack.text({
228
+ message: 'Describe your project in one sentence:',
229
+ placeholder: 'A billing SaaS for freelancers with Stripe integration',
230
+ });
231
+
232
+ if (clack.isCancel(description)) return null;
233
+
234
+ return { type: projectType, description };
235
+ }
236
+
237
+ // ── Main Action ──
238
+
239
+ module.exports = {
240
+ command: 'autoconfig',
241
+ description: 'Smart project bootstrap — auto-detect, install, and configure',
242
+ options: [
243
+ ['-d, --directory <path>', 'Project directory (default: current directory)'],
244
+ ['-y, --yes', 'Accept all recommendations without prompting'],
245
+ ],
246
+ action: async (options) => {
247
+ const projectDir = path.resolve(options.directory || process.cwd());
248
+
249
+ clack.intro(pc.bgCyan(pc.black(' 🧠 BMAD+ Autoconfig ')));
250
+
251
+ // Check if directory has content
252
+ let entries = [];
253
+ try { entries = fs.readdirSync(projectDir).filter(e => !e.startsWith('.')); } catch {}
254
+
255
+ const isExistingProject = entries.length > 0;
256
+
257
+ if (isExistingProject) {
258
+ // ── MODE A: Existing Project ──
259
+ clack.log.info(pc.bold(`📂 Existing project detected: ${getProjectName(projectDir)}`));
260
+
261
+ const spinner = clack.spinner();
262
+ spinner.start('Analyzing project...');
263
+
264
+ const stack = detectStack(projectDir);
265
+ const structure = analyzeStructure(projectDir);
266
+ const health = calculateHealth(structure);
267
+ const { packs, reasons } = recommendPacks(stack, structure, health);
268
+ const recs = generateRecommendations(stack, structure, health);
269
+
270
+ spinner.stop('Analysis complete');
271
+
272
+ // Display analysis
273
+ clack.log.info('');
274
+ clack.log.info(pc.bold('📊 Project Analysis'));
275
+ clack.log.info('');
276
+
277
+ // Stack
278
+ const stackLabel = [stack.framework, stack.language, stack.runtime].filter(Boolean).join(' + ') || 'Unknown';
279
+ clack.log.info(` Stack: ${pc.cyan(stackLabel)}${stack.packageManager ? pc.dim(` (${stack.packageManager})`) : ''}`);
280
+
281
+ // Structure
282
+ const structItems = [];
283
+ if (structure.hasSrc) structItems.push(pc.green('src/'));
284
+ if (structure.hasTests) structItems.push(pc.green('tests/'));
285
+ if (structure.hasDocs) structItems.push(pc.green('docs/'));
286
+ if (structure.hasCI) structItems.push(pc.green('CI/CD'));
287
+ if (structure.hasDocker) structItems.push(pc.green('Docker'));
288
+ if (!structure.hasTests) structItems.push(pc.red('no tests'));
289
+ if (!structure.hasDocs) structItems.push(pc.red('no docs'));
290
+ clack.log.info(` Structure: ${structItems.join(' ')}`);
291
+
292
+ // Health
293
+ const healthColor = health.pct >= 80 ? pc.green : health.pct >= 50 ? pc.yellow : pc.red;
294
+ const healthBar = '█'.repeat(Math.round(health.pct / 10)) + '░'.repeat(10 - Math.round(health.pct / 10));
295
+ clack.log.info(` Health: ${healthColor(`${healthBar} ${health.pct}%`)}`);
296
+
297
+ // Existing BMAD+
298
+ if (structure.hasBmad) {
299
+ clack.log.info(` BMAD+: ${pc.green('✓ already installed')} — will update`);
300
+ }
301
+ if (structure.hasIdeConfigs.length > 0) {
302
+ clack.log.info(` IDE configs: ${pc.green('✓')} ${structure.hasIdeConfigs.length} found — will preserve`);
303
+ }
304
+
305
+ clack.log.info('');
306
+
307
+ // Pack recommendations
308
+ clack.log.info(pc.bold('📦 Recommended Packs'));
309
+ clack.log.info('');
310
+ for (const packId of packs) {
311
+ clack.log.info(` ${pc.green('✓')} ${pc.bold(packId.padEnd(12))} ${pc.dim(reasons[packId])}`);
312
+ }
313
+ clack.log.info('');
314
+
315
+ // Confirm
316
+ let confirmed = options.yes;
317
+ if (!confirmed) {
318
+ const answer = await clack.confirm({
319
+ message: `Install ${packs.length} recommended packs?`,
320
+ initialValue: true,
321
+ });
322
+ if (clack.isCancel(answer) || !answer) {
323
+ clack.cancel('Autoconfig cancelled.');
324
+ return;
325
+ }
326
+ confirmed = true;
327
+ }
328
+
329
+ // Run install
330
+ const spinner2 = clack.spinner();
331
+ spinner2.start('Installing...');
332
+
333
+ // Use the install module directly
334
+ const installModule = require('./install');
335
+ const toolsArg = structure.hasIdeConfigs.length > 0 ? 'none' : undefined;
336
+
337
+ // Build install args
338
+ try {
339
+ await installModule.action({
340
+ directory: projectDir,
341
+ packs: packs.join(','),
342
+ yes: true,
343
+ tools: toolsArg,
344
+ });
345
+ } catch (e) {
346
+ // Install may have its own output, that's fine
347
+ }
348
+
349
+ spinner2.stop('Installation complete');
350
+
351
+ // Update memory context.md
352
+ const contextPath = path.join(projectDir, '.agents', 'memory', 'context.md');
353
+ if (fs.existsSync(path.dirname(contextPath))) {
354
+ const contextContent = [
355
+ '---',
356
+ 'title: Project Context',
357
+ `last_updated: "${new Date().toISOString().slice(0, 10)}"`,
358
+ `auto_generated: true`,
359
+ '---',
360
+ '',
361
+ '# Project Context',
362
+ '',
363
+ `> Auto-generated by \`npx bmad-plus autoconfig\` — ${new Date().toISOString().slice(0, 10)}`,
364
+ '',
365
+ '## Stack',
366
+ '',
367
+ `- **Language:** ${stack.language || 'Unknown'}`,
368
+ `- **Framework:** ${stack.framework || 'None'}`,
369
+ `- **Runtime:** ${stack.runtime || 'Unknown'}`,
370
+ `- **Package Manager:** ${stack.packageManager || 'N/A'}`,
371
+ `- **TypeScript:** ${stack.hasTypeScript ? 'Yes' : 'No'}`,
372
+ '',
373
+ '## Structure',
374
+ '',
375
+ `- **Source code:** ${structure.hasSrc ? 'Yes' : 'No'}`,
376
+ `- **Tests:** ${structure.hasTests ? 'Yes' : 'No — needs setup'}`,
377
+ `- **Documentation:** ${structure.hasDocs ? 'Yes' : 'No — needs writing'}`,
378
+ `- **CI/CD:** ${structure.hasCI ? 'Yes' : 'No'}`,
379
+ `- **Docker:** ${structure.hasDocker ? 'Yes' : 'No'}`,
380
+ `- **Git:** ${structure.hasGit ? 'Yes' : 'No'}`,
381
+ '',
382
+ '## Health',
383
+ '',
384
+ `- **Score:** ${health.pct}%`,
385
+ ...health.checks.map(c => `- ${c.pass ? '✅' : '❌'} ${c.name}`),
386
+ '',
387
+ '## Key Directories',
388
+ '',
389
+ ...structure.directories.slice(0, 15).map(d => `- \`${d}/\``),
390
+ '',
391
+ '## Installed Packs',
392
+ '',
393
+ ...packs.map(p => `- ${p}`),
394
+ '',
395
+ ];
396
+
397
+ fs.writeFileSync(contextPath, contextContent.join('\n'), 'utf8');
398
+ }
399
+
400
+ // Display recommendations
401
+ clack.log.info('');
402
+ clack.log.info(pc.bold('🎯 Recommended Next Steps'));
403
+ clack.log.info('');
404
+
405
+ const priorityIcon = { high: pc.red('‼'), medium: pc.yellow('!'), normal: pc.dim('·') };
406
+ for (const rec of recs.slice(0, 5)) {
407
+ clack.log.info(` ${priorityIcon[rec.priority] || '·'} "${pc.cyan(rec.agent)}, ${rec.action}"`);
408
+ }
409
+
410
+ clack.log.info('');
411
+
412
+ } else {
413
+ // ── MODE B: New Project ──
414
+ clack.log.info(pc.bold('🆕 Empty directory — starting new project wizard'));
415
+ clack.log.info('');
416
+
417
+ const wizard = await newProjectWizard();
418
+ if (!wizard) {
419
+ clack.cancel('Autoconfig cancelled.');
420
+ return;
421
+ }
422
+
423
+ // Select packs based on project type
424
+ const typePacks = {
425
+ web: ['core', 'memory', 'dev-studio', 'seo'],
426
+ api: ['core', 'memory', 'dev-studio', 'shield'],
427
+ cli: ['core', 'memory'],
428
+ mobile: ['core', 'memory', 'dev-studio'],
429
+ library: ['core', 'memory', 'dev-studio'],
430
+ other: ['core', 'memory'],
431
+ };
432
+
433
+ const packs = typePacks[wizard.type] || ['core', 'memory'];
434
+
435
+ clack.log.info(`📦 Packs for ${wizard.type} project: ${packs.join(', ')}`);
436
+
437
+ // Run install
438
+ try {
439
+ const installModule = require('./install');
440
+ await installModule.action({
441
+ directory: projectDir,
442
+ packs: packs.join(','),
443
+ yes: true,
444
+ });
445
+ } catch {}
446
+
447
+ // Write initial context
448
+ const contextPath = path.join(projectDir, '.agents', 'memory', 'context.md');
449
+ if (fs.existsSync(path.dirname(contextPath))) {
450
+ const contextContent = [
451
+ '---',
452
+ 'title: Project Context',
453
+ `last_updated: "${new Date().toISOString().slice(0, 10)}"`,
454
+ `auto_generated: true`,
455
+ '---',
456
+ '',
457
+ '# Project Context',
458
+ '',
459
+ `> Auto-generated by \`npx bmad-plus autoconfig\` — ${new Date().toISOString().slice(0, 10)}`,
460
+ '',
461
+ '## Project Brief',
462
+ '',
463
+ `- **Type:** ${wizard.type}`,
464
+ `- **Description:** ${wizard.description}`,
465
+ `- **Status:** New — not started`,
466
+ '',
467
+ '## Installed Packs',
468
+ '',
469
+ ...packs.map(p => `- ${p}`),
470
+ '',
471
+ ];
472
+
473
+ fs.writeFileSync(contextPath, contextContent.join('\n'), 'utf8');
474
+ }
475
+
476
+ // Recommendations for new project
477
+ clack.log.info('');
478
+ clack.log.info(pc.bold('🎯 Recommended First Steps'));
479
+ clack.log.info('');
480
+ clack.log.info(` 1. "${pc.cyan('Atlas')}, I want to build: ${wizard.description}"`);
481
+ clack.log.info(` 2. "${pc.cyan('Atlas')}, create the PRD"`);
482
+ clack.log.info(` 3. "${pc.cyan('Forge')}, propose the architecture"`);
483
+ clack.log.info(` 4. Or: "${pc.cyan('autopilot')}" to let Nexus manage everything`);
484
+ clack.log.info('');
485
+ }
486
+
487
+ clack.outro(pc.green('Autoconfig complete! 🚀'));
488
+ },
489
+ };
@@ -149,6 +149,53 @@ module.exports = {
149
149
  }
150
150
  }
151
151
 
152
+ // ── Check 9: PACKS ↔ module.yaml sync ──
153
+ checks++;
154
+ try {
155
+ const yaml = require('js-yaml');
156
+ const installModule = require('./install');
157
+ const installSrc = fs.readFileSync(path.join(__dirname, 'install.js'), 'utf8');
158
+ const packsMatch = installSrc.match(/const PACKS\s*=\s*\{/);
159
+
160
+ const moduleYamlSrc = path.join(__dirname, '..', '..', '..', 'src', 'bmad-plus', 'module.yaml');
161
+ if (packsMatch && fs.existsSync(moduleYamlSrc)) {
162
+ const moduleContent = yaml.load(fs.readFileSync(moduleYamlSrc, 'utf8'));
163
+ const modulePackIds = Object.keys(moduleContent.packs || {});
164
+
165
+ // Extract PACKS keys from install.js via require
166
+ // The PACKS keys are the pack IDs available in the CLI menu
167
+ // We compare against module.yaml packs
168
+ const installContent = fs.readFileSync(path.join(__dirname, 'install.js'), 'utf8');
169
+ const packKeyMatches = installContent.match(/^\s+'?([a-z][-a-z]*)'?\s*:\s*\{/gm);
170
+ const installPackIds = packKeyMatches
171
+ ? packKeyMatches.map(m => m.trim().replace(/[':{ ]/g, '').replace(/-/g, '-'))
172
+ : [];
173
+
174
+ // Find mismatches
175
+ const missingInInstall = modulePackIds.filter(p => !installPackIds.includes(p));
176
+ const missingInModule = installPackIds.filter(p => !modulePackIds.includes(p));
177
+
178
+ if (missingInInstall.length === 0 && missingInModule.length === 0) {
179
+ clack.log.success(`✅ PACKS ↔ module.yaml in sync (${modulePackIds.length} packs)`);
180
+ passed++;
181
+ } else {
182
+ if (missingInInstall.length > 0) {
183
+ clack.log.warn(`⚠️ Packs in module.yaml but missing from install.js PACKS: ${missingInInstall.join(', ')}`);
184
+ }
185
+ if (missingInModule.length > 0) {
186
+ clack.log.warn(`⚠️ Packs in install.js PACKS but missing from module.yaml: ${missingInModule.join(', ')}`);
187
+ }
188
+ warnings++;
189
+ }
190
+ } else {
191
+ clack.log.info(pc.dim('ℹ️ PACKS↔module.yaml check skipped (source not available in npx context)'));
192
+ passed++; // Not a failure — just unavailable
193
+ }
194
+ } catch (e) {
195
+ clack.log.info(pc.dim(`ℹ️ PACKS↔module.yaml check skipped: ${e.message}`));
196
+ passed++; // Graceful skip
197
+ }
198
+
152
199
  // ── Summary ──
153
200
  const summaryColor = errors > 0 ? pc.red : warnings > 0 ? pc.yellow : pc.green;
154
201
  const summaryIcon = errors > 0 ? '❌' : warnings > 0 ? '⚠️' : '✅';
@@ -177,7 +177,7 @@ module.exports = {
177
177
  if (requested.includes('all')) {
178
178
  selectedPacks = Object.keys(PACKS).filter(k => !PACKS[k].disabled);
179
179
  } else {
180
- selectedPacks = ['core', ...requested.filter(p => PACKS[p] && !PACKS[p].disabled)];
180
+ selectedPacks = [...new Set(['core', ...requested.filter(p => PACKS[p] && !PACKS[p].disabled)])];
181
181
  }
182
182
  } else if (!options.yes) {
183
183
  const packChoice = await clack.multiselect({
@@ -207,7 +207,11 @@ module.exports = {
207
207
  let detectedIDEs = [];
208
208
 
209
209
  if (options.tools) {
210
- detectedIDEs = options.tools.split(',').map(t => t.trim());
210
+ if (options.tools === 'none' || options.tools === 'skip') {
211
+ detectedIDEs = [];
212
+ } else {
213
+ detectedIDEs = options.tools.split(',').map(t => t.trim()).filter(t => IDE_CONFIGS[t]);
214
+ }
211
215
  } else {
212
216
  // Auto-detect
213
217
  for (const [id, ide] of Object.entries(IDE_CONFIGS)) {
@@ -243,6 +247,8 @@ module.exports = {
243
247
 
244
248
  if (detectedIDEs.length > 0) {
245
249
  clack.log.info(`${i.detected_ides}: ${detectedIDEs.map(id => IDE_CONFIGS[id].name).join(', ')}`);
250
+ } else if (options.tools === 'none' || options.tools === 'skip') {
251
+ clack.log.info(pc.dim('⏭️ IDE config skipped (--tools none) — existing configs preserved'));
246
252
  }
247
253
 
248
254
  // ── Step 3: User Config ──
package/tools/cli/i18n.js CHANGED
@@ -10,7 +10,7 @@ const LANGUAGES = {
10
10
  flag: '🇬🇧',
11
11
  name: 'English',
12
12
  locale: 'en',
13
- installer_title: ' BMAD+ Installer v0.7.3 ',
13
+ installer_title: ' BMAD+ Installer v0.7.5 ',
14
14
  select_language: 'Select your language',
15
15
  installing_to: 'Installing to',
16
16
  select_packs: 'Which packs to install? (Core is always included)',
@@ -92,7 +92,7 @@ const LANGUAGES = {
92
92
  flag: '🇫🇷',
93
93
  name: 'Français',
94
94
  locale: 'fr',
95
- installer_title: ' BMAD+ Installeur v0.7.3 ',
95
+ installer_title: ' BMAD+ Installeur v0.7.5 ',
96
96
  select_language: 'Choisissez votre langue',
97
97
  installing_to: 'Installation dans',
98
98
  select_packs: 'Quels packs installer ? (Core est toujours inclus)',
@@ -172,7 +172,7 @@ const LANGUAGES = {
172
172
  flag: '🇪🇸',
173
173
  name: 'Español',
174
174
  locale: 'es',
175
- installer_title: ' BMAD+ Instalador v0.7.3 ',
175
+ installer_title: ' BMAD+ Instalador v0.7.5 ',
176
176
  select_language: 'Seleccione su idioma',
177
177
  installing_to: 'Instalando en',
178
178
  select_packs: '¿Qué packs instalar? (Core siempre está incluido)',
@@ -242,7 +242,7 @@ const LANGUAGES = {
242
242
  flag: '🇩🇪',
243
243
  name: 'Deutsch',
244
244
  locale: 'de',
245
- installer_title: ' BMAD+ Installer v0.7.3 ',
245
+ installer_title: ' BMAD+ Installer v0.7.5 ',
246
246
  select_language: 'Wählen Sie Ihre Sprache',
247
247
  installing_to: 'Installiere in',
248
248
  select_packs: 'Welche Packs installieren? (Core ist immer enthalten)',
@@ -312,7 +312,7 @@ const LANGUAGES = {
312
312
  flag: '🇧🇷',
313
313
  name: 'Português (Brasil)',
314
314
  locale: 'pt-BR',
315
- installer_title: ' BMAD+ Instalador v0.7.3 ',
315
+ installer_title: ' BMAD+ Instalador v0.7.5 ',
316
316
  select_language: 'Selecione seu idioma',
317
317
  installing_to: 'Instalando em',
318
318
  select_packs: 'Quais packs instalar? (Core sempre está incluído)',
@@ -382,7 +382,7 @@ const LANGUAGES = {
382
382
  flag: '🇷🇺',
383
383
  name: 'Русский',
384
384
  locale: 'ru',
385
- installer_title: ' BMAD+ Установщик v0.7.3 ',
385
+ installer_title: ' BMAD+ Установщик v0.7.5 ',
386
386
  select_language: 'Выберите язык',
387
387
  installing_to: 'Установка в',
388
388
  select_packs: 'Какие пакеты установить? (Core всегда включён)',
@@ -452,7 +452,7 @@ const LANGUAGES = {
452
452
  flag: '🇨🇳',
453
453
  name: '中文 (简体)',
454
454
  locale: 'zh-CN',
455
- installer_title: ' BMAD+ 安装程序 v0.7.3 ',
455
+ installer_title: ' BMAD+ 安装程序 v0.7.5 ',
456
456
  select_language: '选择您的语言',
457
457
  installing_to: '安装到',
458
458
  select_packs: '安装哪些包?(Core 始终包含)',
@@ -522,7 +522,7 @@ const LANGUAGES = {
522
522
  flag: '🇮🇱',
523
523
  name: 'עברית',
524
524
  locale: 'he',
525
- installer_title: ' BMAD+ מתקין v0.7.3 ',
525
+ installer_title: ' BMAD+ מתקין v0.7.5 ',
526
526
  select_language: 'בחר את השפה שלך',
527
527
  installing_to: 'מתקין ב',
528
528
  select_packs: 'אילו חבילות להתקין? (Core תמיד כלול)',
@@ -592,7 +592,7 @@ const LANGUAGES = {
592
592
  flag: '🇯🇵',
593
593
  name: '日本語',
594
594
  locale: 'ja',
595
- installer_title: ' BMAD+ インストーラー v0.7.3 ',
595
+ installer_title: ' BMAD+ インストーラー v0.7.5 ',
596
596
  select_language: '言語を選択してください',
597
597
  installing_to: 'インストール先',
598
598
  select_packs: 'どのパックをインストールしますか?(Coreは常に含まれます)',
@@ -662,7 +662,7 @@ const LANGUAGES = {
662
662
  flag: '🇮🇹',
663
663
  name: 'Italiano',
664
664
  locale: 'it',
665
- installer_title: ' BMAD+ Installatore v0.7.3 ',
665
+ installer_title: ' BMAD+ Installatore v0.7.5 ',
666
666
  select_language: 'Seleziona la tua lingua',
667
667
  installing_to: 'Installazione in',
668
668
  select_packs: 'Quali pack installare? (Core è sempre incluso)',