bmad-plus 0.7.2 → 0.7.4

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
+ };
@@ -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 ──
@@ -51,12 +51,12 @@ const SKIP_DIRS = new Set([
51
51
  'AppData', 'Recovery', 'PerfLogs',
52
52
  ]);
53
53
 
54
- function getProjectStatus(dir) {
54
+ function getProjectStatus(dir, activeDays = 30, pausedDays = 180) {
55
55
  try {
56
56
  const stat = fs.statSync(dir);
57
57
  const daysSince = (Date.now() - stat.mtimeMs) / (1000 * 60 * 60 * 24);
58
- if (daysSince < 30) return 'active';
59
- if (daysSince < 180) return 'paused';
58
+ if (daysSince < activeDays) return 'active';
59
+ if (daysSince < pausedDays) return 'paused';
60
60
  return 'archived';
61
61
  } catch { return 'unknown'; }
62
62
  }
@@ -86,7 +86,7 @@ function hasBmadInstalled(dir) {
86
86
  fs.existsSync(path.join(dir, '_bmad'));
87
87
  }
88
88
 
89
- function scanDirectory(rootDir, maxDepth = 4, currentDepth = 0) {
89
+ function scanDirectory(rootDir, maxDepth = 4, currentDepth = 0, activeDays = 30, pausedDays = 180) {
90
90
  const projects = [];
91
91
 
92
92
  if (currentDepth > maxDepth) return projects;
@@ -106,7 +106,7 @@ function scanDirectory(rootDir, maxDepth = 4, currentDepth = 0) {
106
106
  path: rootDir,
107
107
  name: getProjectName(rootDir),
108
108
  stack,
109
- status: getProjectStatus(rootDir),
109
+ status: getProjectStatus(rootDir, activeDays, pausedDays),
110
110
  bmad: hasBmadInstalled(rootDir),
111
111
  hasAgentsMd: fs.existsSync(path.join(rootDir, 'AGENTS.md')),
112
112
  hasGit: fs.existsSync(path.join(rootDir, '.git')),
@@ -121,7 +121,7 @@ function scanDirectory(rootDir, maxDepth = 4, currentDepth = 0) {
121
121
  path: rootDir,
122
122
  name: getProjectName(rootDir),
123
123
  stack: 'Unknown',
124
- status: getProjectStatus(rootDir),
124
+ status: getProjectStatus(rootDir, activeDays, pausedDays),
125
125
  bmad: hasBmadInstalled(rootDir),
126
126
  hasAgentsMd: fs.existsSync(path.join(rootDir, 'AGENTS.md')),
127
127
  hasGit: true,
@@ -136,7 +136,7 @@ function scanDirectory(rootDir, maxDepth = 4, currentDepth = 0) {
136
136
  if (entry.name.startsWith('.') && entry.name !== '.git') continue;
137
137
 
138
138
  const subPath = path.join(rootDir, entry.name);
139
- const subProjects = scanDirectory(subPath, maxDepth, currentDepth + 1);
139
+ const subProjects = scanDirectory(subPath, maxDepth, currentDepth + 1, activeDays, pausedDays);
140
140
  projects.push(...subProjects);
141
141
  }
142
142
 
@@ -149,11 +149,15 @@ module.exports = {
149
149
  options: [
150
150
  ['-d, --directory <path>', 'Directory to scan (default: current directory)'],
151
151
  ['--depth <n>', 'Max depth to scan (default: 4)', '4'],
152
+ ['--active-days <n>', 'Days since last modified to consider a project "active" (default: 30)', '30'],
153
+ ['--paused-days <n>', 'Days since last modified to consider a project "paused" (default: 180)', '180'],
152
154
  ['-y, --yes', 'Index all projects without prompting'],
153
155
  ],
154
156
  action: async (options) => {
155
157
  const scanDir = path.resolve(options.directory || process.cwd());
156
158
  const maxDepth = parseInt(options.depth) || 4;
159
+ const activeDays = parseInt(options.activeDays) || 30;
160
+ const pausedDays = parseInt(options.pausedDays) || 180;
157
161
 
158
162
  clack.intro(pc.bgMagenta(pc.white(' 🧠 BMAD+ Project Scanner ')));
159
163
 
@@ -168,7 +172,7 @@ module.exports = {
168
172
  const spinner = clack.spinner();
169
173
  spinner.start(`Scanning ${scanDir} (depth: ${maxDepth})...`);
170
174
 
171
- const projects = scanDirectory(scanDir, maxDepth);
175
+ const projects = scanDirectory(scanDir, maxDepth, 0, activeDays, pausedDays);
172
176
 
173
177
  if (projects.length === 0) {
174
178
  spinner.stop('No projects found.');
@@ -178,9 +182,20 @@ module.exports = {
178
182
 
179
183
  spinner.stop(`Found ${pc.bold(projects.length)} project(s)`);
180
184
 
181
- // Display table
185
+ // Display legend
186
+ const activeCount = projects.filter(p => p.status === 'active').length;
187
+ const pausedCount = projects.filter(p => p.status === 'paused').length;
188
+ const archivedCount = projects.filter(p => p.status === 'archived').length;
189
+
190
+ clack.log.info('');
191
+ clack.log.info(pc.dim(' Legend:'));
192
+ clack.log.info(` ${pc.green('●')} active modified < ${activeDays} days ago ${pc.dim(`(${activeCount} found)`)}`);
193
+ clack.log.info(` ${pc.yellow('◐')} paused modified ${activeDays}–${pausedDays} days ago ${pc.dim(`(${pausedCount} found)`)}`);
194
+ clack.log.info(` ${pc.dim('○')} archived modified > ${pausedDays} days ago ${pc.dim(`(${archivedCount} found)`)}`);
182
195
  clack.log.info('');
183
- clack.log.info(pc.bold(' # Status BMAD+ Stack Name Path'));
196
+
197
+ // Display table
198
+ clack.log.info(pc.bold(' # Status BMAD+ Stack Name Path'));
184
199
  clack.log.info(pc.dim(' ' + '─'.repeat(90)));
185
200
 
186
201
  projects.forEach((p, i) => {