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.
- package/CHANGELOG.md +33 -0
- package/LICENSE +21 -0
- package/README.md +58 -57
- 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/doctor.js +47 -0
- package/tools/cli/commands/install.js +8 -2
- 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
|
+
};
|
|
@@ -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
|
-
|
|
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.
|
|
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.
|
|
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.
|
|
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.
|
|
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.
|
|
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.
|
|
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.
|
|
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.
|
|
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.
|
|
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.
|
|
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)',
|