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.
- package/CHANGELOG.md +27 -0
- package/README.md +1 -1
- package/package.json +1 -1
- package/readme-international/README.de.md +106 -72
- package/readme-international/README.es.md +192 -158
- package/readme-international/README.fr.md +186 -152
- package/tools/cli/bmad-plus-cli.js +11 -0
- package/tools/cli/commands/autoconfig.js +489 -0
- package/tools/cli/commands/install.js +8 -2
- package/tools/cli/commands/scan.js +25 -10
- package/tools/cli/i18n.js +10 -10
|
@@ -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
|
-
|
|
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 <
|
|
59
|
-
if (daysSince <
|
|
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
|
|
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
|
-
|
|
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) => {
|