bmad-plus 0.7.4 → 0.8.0
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 +450 -407
- package/LICENSE +21 -0
- package/README.md +555 -446
- package/osint-agent-package/README.md +88 -88
- package/osint-agent-package/SETUP_KEYS.md +108 -108
- package/osint-agent-package/agents/osint-investigator.md +80 -80
- package/osint-agent-package/install.ps1 +87 -87
- package/osint-agent-package/install.sh +76 -76
- package/osint-agent-package/skills/bmad-osint-investigate/SKILL.md +147 -147
- package/osint-agent-package/skills/bmad-osint-investigate/osint/SKILL.md +452 -452
- package/osint-agent-package/skills/bmad-osint-investigate/osint/assets/dossier-template.md +116 -116
- package/osint-agent-package/skills/bmad-osint-investigate/osint/references/content-extraction.md +100 -100
- package/osint-agent-package/skills/bmad-osint-investigate/osint/references/enrichment-databases-fr.md +148 -148
- package/osint-agent-package/skills/bmad-osint-investigate/osint/references/platforms.md +130 -130
- package/osint-agent-package/skills/bmad-osint-investigate/osint/references/psychoprofile.md +69 -69
- package/osint-agent-package/skills/bmad-osint-investigate/osint/references/tools.md +281 -281
- package/osint-agent-package/skills/bmad-osint-investigate/osint/scripts/_http.py +101 -101
- package/osint-agent-package/skills/bmad-osint-investigate/osint/scripts/apify.py +266 -260
- package/osint-agent-package/skills/bmad-osint-investigate/osint/scripts/brightdata.py +101 -101
- package/osint-agent-package/skills/bmad-osint-investigate/osint/scripts/diagnose.py +141 -141
- package/osint-agent-package/skills/bmad-osint-investigate/osint/scripts/exa.py +79 -79
- package/osint-agent-package/skills/bmad-osint-investigate/osint/scripts/jina.py +71 -71
- package/osint-agent-package/skills/bmad-osint-investigate/osint/scripts/mcp-client.py +136 -136
- package/osint-agent-package/skills/bmad-osint-investigate/osint/scripts/parallel.py +85 -85
- package/osint-agent-package/skills/bmad-osint-investigate/osint/scripts/perplexity.py +102 -102
- package/osint-agent-package/skills/bmad-osint-investigate/osint/scripts/tavily.py +72 -72
- package/osint-agent-package/skills/bmad-osint-investigate/osint/scripts/volley.py +208 -208
- package/osint-agent-package/skills/bmad-osint-investigator/SKILL.md +15 -15
- package/package.json +62 -57
- package/readme-international/README.de.md +576 -426
- package/readme-international/README.es.md +578 -518
- package/readme-international/README.fr.md +576 -516
- package/src/bmad-plus/agents/agent-architect-dev/SKILL.md +96 -96
- package/src/bmad-plus/agents/agent-architect-dev/bmad-skill-manifest.yaml +13 -13
- package/src/bmad-plus/agents/agent-maker/SKILL.md +201 -201
- package/src/bmad-plus/agents/agent-maker/bmad-skill-manifest.yaml +13 -13
- package/src/bmad-plus/agents/agent-orchestrator/SKILL.md +137 -137
- package/src/bmad-plus/agents/agent-orchestrator/bmad-skill-manifest.yaml +13 -13
- package/src/bmad-plus/agents/agent-quality/SKILL.md +83 -83
- package/src/bmad-plus/agents/agent-quality/bmad-skill-manifest.yaml +13 -13
- package/src/bmad-plus/agents/agent-shadow/SKILL.md +71 -71
- package/src/bmad-plus/agents/agent-shadow/bmad-skill-manifest.yaml +13 -13
- package/src/bmad-plus/agents/agent-strategist/SKILL.md +80 -80
- package/src/bmad-plus/agents/agent-strategist/bmad-skill-manifest.yaml +13 -13
- package/src/bmad-plus/agents/pack-animated/animated-website-agent.md +325 -325
- package/src/bmad-plus/agents/pack-animated/templates/animated-website-workflow.md +55 -55
- package/src/bmad-plus/agents/pack-backup/backup-agent.md +71 -71
- package/src/bmad-plus/agents/pack-backup/templates/backup-workflow.md +51 -51
- package/src/bmad-plus/agents/pack-seo/SKILL.md +171 -171
- package/src/bmad-plus/agents/pack-seo/checklist.md +140 -140
- package/src/bmad-plus/agents/pack-seo/pagespeed-playbook.md +320 -320
- package/src/bmad-plus/agents/pack-seo/ref/audit-schema.json +187 -187
- package/src/bmad-plus/agents/pack-seo/ref/cwv-thresholds.md +87 -87
- package/src/bmad-plus/agents/pack-seo/ref/eeat-criteria.md +123 -123
- package/src/bmad-plus/agents/pack-seo/ref/geo-signals.md +167 -167
- package/src/bmad-plus/agents/pack-seo/ref/hreflang-rules.md +153 -153
- package/src/bmad-plus/agents/pack-seo/ref/quality-gates.md +133 -133
- package/src/bmad-plus/agents/pack-seo/ref/schema-catalog.md +91 -91
- package/src/bmad-plus/agents/pack-seo/ref/schema-templates.json +356 -356
- package/src/bmad-plus/agents/pack-seo/seo-chief.md +294 -294
- package/src/bmad-plus/agents/pack-seo/seo-judge.md +241 -241
- package/src/bmad-plus/agents/pack-seo/seo-scout.md +171 -171
- package/src/bmad-plus/agents/pack-seo/templates/seo-audit-workflow.md +241 -241
- package/src/bmad-plus/data/role-triggers.yaml +209 -209
- package/src/bmad-plus/module-help.csv +10 -10
- package/src/bmad-plus/module.yaml +283 -280
- package/src/bmad-plus/packs/pack-animated/animated-website-agent.md +325 -0
- package/src/bmad-plus/packs/pack-animated/templates/animated-website-workflow.md +55 -0
- package/src/bmad-plus/packs/pack-backup/backup-agent.md +71 -0
- package/src/bmad-plus/packs/pack-backup/templates/backup-workflow.md +51 -0
- package/src/bmad-plus/packs/pack-dev-studio/README.md +162 -162
- package/src/bmad-plus/packs/pack-dev-studio/categories/analysis/analyst-agent.md +73 -73
- package/src/bmad-plus/packs/pack-dev-studio/categories/analysis/document-project.md +61 -61
- package/src/bmad-plus/packs/pack-dev-studio/categories/analysis/domain-research.md +95 -95
- package/src/bmad-plus/packs/pack-dev-studio/categories/analysis/market-research.md +95 -95
- package/src/bmad-plus/packs/pack-dev-studio/categories/analysis/prfaq.md +134 -134
- package/src/bmad-plus/packs/pack-dev-studio/categories/analysis/product-brief.md +80 -80
- package/src/bmad-plus/packs/pack-dev-studio/categories/analysis/tech-writer-agent.md +73 -73
- package/src/bmad-plus/packs/pack-dev-studio/categories/analysis/technical-research.md +95 -95
- package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/architect-agent.md +73 -73
- package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/create-architecture.md +73 -73
- package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/create-epics-stories.md +92 -92
- package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/generate-project-context.md +80 -80
- package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/implementation-readiness.md +90 -90
- package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/steps/step-01-init.md +153 -153
- package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/steps/step-01b-continue.md +173 -173
- package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/steps/step-02-context.md +224 -224
- package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/steps/step-03-starter.md +329 -329
- package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/steps/step-04-decisions.md +318 -318
- package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/steps/step-05-patterns.md +359 -359
- package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/steps/step-06-structure.md +379 -379
- package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/steps/step-07-validation.md +361 -361
- package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/steps/step-08-complete.md +81 -81
- package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/checkpoint-preview.md +67 -67
- package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/code-review-steps/step-01-gather-context.md +85 -85
- package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/code-review-steps/step-02-review.md +35 -35
- package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/code-review-steps/step-03-triage.md +49 -49
- package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/code-review-steps/step-04-present.md +131 -131
- package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/code-review.md +89 -89
- package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/correct-course.md +300 -300
- package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/create-story.md +428 -428
- package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/dev-agent.md +73 -73
- package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/dev-story-checklist.md +80 -80
- package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/dev-story.md +484 -484
- package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/investigate.md +193 -193
- package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/qa-e2e-tests.md +175 -175
- package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/quick-dev.md +110 -110
- package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/retrospective.md +1511 -1511
- package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/sprint-planning.md +298 -298
- package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/sprint-status.md +296 -296
- package/src/bmad-plus/packs/pack-dev-studio/categories/planning/create-prd.md +29 -29
- package/src/bmad-plus/packs/pack-dev-studio/categories/planning/create-ux-design.md +74 -74
- package/src/bmad-plus/packs/pack-dev-studio/categories/planning/edit-prd.md +29 -29
- package/src/bmad-plus/packs/pack-dev-studio/categories/planning/pm-agent.md +73 -73
- package/src/bmad-plus/packs/pack-dev-studio/categories/planning/prd.md +89 -89
- package/src/bmad-plus/packs/pack-dev-studio/categories/planning/ux-designer-agent.md +73 -73
- package/src/bmad-plus/packs/pack-dev-studio/categories/planning/validate-prd.md +29 -29
- package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/advanced-elicitation.md +141 -141
- package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/adversarial-review.md +37 -37
- package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/bmad-help.md +75 -75
- package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/brainstorming.md +6 -6
- package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/customize.md +110 -110
- package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/distillator.md +176 -176
- package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/edge-case-hunter.md +67 -67
- package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/editorial-review-prose.md +86 -86
- package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/editorial-review-structure.md +179 -179
- package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/index-docs.md +66 -66
- package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/party-mode.md +127 -127
- package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/shard-doc.md +105 -105
- package/src/bmad-plus/packs/pack-dev-studio/dev-studio-orchestrator.md +120 -120
- package/src/bmad-plus/packs/pack-dev-studio/shared/architecture-decision-template.md +12 -12
- package/src/bmad-plus/packs/pack-dev-studio/shared/bwml-spec.md +328 -328
- package/src/bmad-plus/packs/pack-dev-studio/shared/module-help.csv +32 -32
- package/src/bmad-plus/packs/pack-dev-studio/upstream-sync.yaml +81 -81
- package/src/bmad-plus/packs/pack-memory/README.md +106 -106
- package/src/bmad-plus/packs/pack-memory/memory-orchestrator.md +79 -79
- package/src/bmad-plus/packs/pack-memory/shared/karpathy-guardrails.md +86 -86
- package/src/bmad-plus/packs/pack-memory/shared/memory-protocol.md +143 -143
- package/src/bmad-plus/packs/pack-memory/templates/context.md +39 -39
- package/src/bmad-plus/packs/pack-memory/templates/decisions.md +25 -25
- package/src/bmad-plus/packs/pack-memory/templates/identity.yaml +39 -39
- package/src/bmad-plus/packs/pack-memory/templates/lessons.md +31 -31
- package/src/bmad-plus/packs/pack-memory/templates/patterns.md +24 -24
- package/src/bmad-plus/packs/pack-memory/templates/session-handoff.md +25 -25
- package/src/bmad-plus/packs/pack-memory/zecher-agent.md +157 -157
- package/src/bmad-plus/packs/pack-seo/SKILL.md +171 -0
- package/src/bmad-plus/packs/pack-seo/checklist.md +140 -0
- package/src/bmad-plus/packs/pack-seo/pagespeed-playbook.md +320 -0
- package/src/bmad-plus/packs/pack-seo/ref/audit-schema.json +187 -0
- package/src/bmad-plus/packs/pack-seo/ref/cwv-thresholds.md +87 -0
- package/src/bmad-plus/packs/pack-seo/ref/eeat-criteria.md +123 -0
- package/src/bmad-plus/packs/pack-seo/ref/geo-signals.md +167 -0
- package/src/bmad-plus/packs/pack-seo/ref/hreflang-rules.md +153 -0
- package/src/bmad-plus/packs/pack-seo/ref/quality-gates.md +133 -0
- package/src/bmad-plus/packs/pack-seo/ref/schema-catalog.md +91 -0
- package/src/bmad-plus/packs/pack-seo/ref/schema-templates.json +356 -0
- package/src/bmad-plus/packs/pack-seo/seo-chief.md +294 -0
- package/src/bmad-plus/packs/pack-seo/seo-judge.md +241 -0
- package/src/bmad-plus/packs/pack-seo/seo-scout.md +171 -0
- package/src/bmad-plus/packs/pack-seo/templates/seo-audit-workflow.md +241 -0
- package/src/bmad-plus/packs/pack-shield/README.md +110 -110
- package/src/bmad-plus/packs/pack-shield/categories/accessibility-esg/csrd-agent.md +262 -262
- package/src/bmad-plus/packs/pack-shield/categories/accessibility-esg/section508-agent.md +179 -179
- package/src/bmad-plus/packs/pack-shield/categories/accessibility-esg/wcag-agent.md +201 -201
- package/src/bmad-plus/packs/pack-shield/categories/ai-governance/eu-ai-act-agent.md +97 -97
- package/src/bmad-plus/packs/pack-shield/categories/ai-governance/iso42001-agent.md +251 -251
- package/src/bmad-plus/packs/pack-shield/categories/ai-governance/nist-ai-rmf-agent.md +133 -133
- package/src/bmad-plus/packs/pack-shield/categories/cybersecurity/cis-controls-agent.md +221 -221
- package/src/bmad-plus/packs/pack-shield/categories/cybersecurity/ism-agent.md +150 -150
- package/src/bmad-plus/packs/pack-shield/categories/cybersecurity/iso27001-agent.md +167 -167
- package/src/bmad-plus/packs/pack-shield/categories/cybersecurity/nis2-agent.md +83 -83
- package/src/bmad-plus/packs/pack-shield/categories/cybersecurity/nist-800-53-agent.md +250 -250
- package/src/bmad-plus/packs/pack-shield/categories/cybersecurity/nist-csf-agent.md +218 -218
- package/src/bmad-plus/packs/pack-shield/categories/data-privacy/ccpa-agent.md +94 -94
- package/src/bmad-plus/packs/pack-shield/categories/data-privacy/dpdpa-agent.md +136 -136
- package/src/bmad-plus/packs/pack-shield/categories/data-privacy/gdpr-agent.md +296 -296
- package/src/bmad-plus/packs/pack-shield/categories/data-privacy/iso27701-agent.md +134 -134
- package/src/bmad-plus/packs/pack-shield/categories/data-privacy/lgpd-agent.md +129 -129
- package/src/bmad-plus/packs/pack-shield/categories/defense-export/cmmc-agent.md +127 -127
- package/src/bmad-plus/packs/pack-shield/categories/defense-export/ear-agent.md +272 -272
- package/src/bmad-plus/packs/pack-shield/categories/defense-export/itar-agent.md +202 -202
- package/src/bmad-plus/packs/pack-shield/categories/defense-export/tsa-agent.md +367 -367
- package/src/bmad-plus/packs/pack-shield/categories/industry-compliance/dora-agent.md +510 -510
- package/src/bmad-plus/packs/pack-shield/categories/industry-compliance/fedramp-agent.md +247 -247
- package/src/bmad-plus/packs/pack-shield/categories/industry-compliance/hipaa-agent.md +173 -173
- package/src/bmad-plus/packs/pack-shield/categories/industry-compliance/pci-dss-agent.md +239 -239
- package/src/bmad-plus/packs/pack-shield/categories/industry-compliance/soc2-agent.md +266 -266
- package/src/bmad-plus/packs/pack-shield/categories/industry-compliance/swift-csp-agent.md +164 -164
- package/src/bmad-plus/packs/pack-shield/categories/workflows/ai-act-classifier.md +131 -131
- package/src/bmad-plus/packs/pack-shield/categories/workflows/ai-act-fria.md +155 -155
- package/src/bmad-plus/packs/pack-shield/categories/workflows/ai-act-incidents.md +187 -187
- package/src/bmad-plus/packs/pack-shield/categories/workflows/ai-act-roles.md +113 -113
- package/src/bmad-plus/packs/pack-shield/categories/workflows/breach-sentinel.md +197 -197
- package/src/bmad-plus/packs/pack-shield/categories/workflows/cookie-policy-gen.md +180 -180
- package/src/bmad-plus/packs/pack-shield/categories/workflows/dpia-sentinel.md +235 -235
- package/src/bmad-plus/packs/pack-shield/categories/workflows/legitimate-interest.md +159 -159
- package/src/bmad-plus/packs/pack-shield/categories/workflows/privacy-advisor.md +133 -133
- package/src/bmad-plus/packs/pack-shield/categories/workflows/privacy-notice-gen.md +160 -160
- package/src/bmad-plus/packs/pack-shield/categories/workflows/privacy-policy-gen.md +135 -135
- package/src/bmad-plus/packs/pack-shield/references/ccpa/ccpa-gdpr-comparison.md +117 -117
- package/src/bmad-plus/packs/pack-shield/references/ccpa/consumer-rights-workflows.md +177 -177
- package/src/bmad-plus/packs/pack-shield/references/cis-controls/framework-mappings.md +162 -162
- package/src/bmad-plus/packs/pack-shield/references/cis-controls/implementation-guidance.md +235 -235
- package/src/bmad-plus/packs/pack-shield/references/cis-controls/safeguards-detail.md +252 -252
- package/src/bmad-plus/packs/pack-shield/references/cmmc/cmmc-assessment.md +170 -170
- package/src/bmad-plus/packs/pack-shield/references/cmmc/cmmc-levels.md +113 -113
- package/src/bmad-plus/packs/pack-shield/references/cmmc/cmmc-practices.md +211 -211
- package/src/bmad-plus/packs/pack-shield/references/csrd/compliance-program.md +281 -281
- package/src/bmad-plus/packs/pack-shield/references/csrd/double-materiality.md +253 -253
- package/src/bmad-plus/packs/pack-shield/references/csrd/esrs-standards.md +401 -401
- package/src/bmad-plus/packs/pack-shield/references/dora/article-reference.md +441 -441
- package/src/bmad-plus/packs/pack-shield/references/dora/incident-classification.md +297 -297
- package/src/bmad-plus/packs/pack-shield/references/dora/rts-its-guide.md +306 -306
- package/src/bmad-plus/packs/pack-shield/references/dora/third-party-risk.md +349 -349
- package/src/bmad-plus/packs/pack-shield/references/dpdpa/gdpr-comparison.md +173 -173
- package/src/bmad-plus/packs/pack-shield/references/dpdpa/rights-and-obligations.md +426 -426
- package/src/bmad-plus/packs/pack-shield/references/dpdpa/rules-2025.md +599 -599
- package/src/bmad-plus/packs/pack-shield/references/dpdpa/sections-reference.md +319 -319
- package/src/bmad-plus/packs/pack-shield/references/ear/ccl-eccn-guide.md +250 -250
- package/src/bmad-plus/packs/pack-shield/references/ear/compliance-program.md +280 -280
- package/src/bmad-plus/packs/pack-shield/references/ear/license-exceptions.md +207 -207
- package/src/bmad-plus/packs/pack-shield/references/eu-ai-act/gpai-governance.md +267 -267
- package/src/bmad-plus/packs/pack-shield/references/eu-ai-act/obligations-high-risk.md +287 -287
- package/src/bmad-plus/packs/pack-shield/references/eu-ai-act/risk-classification.md +182 -182
- package/src/bmad-plus/packs/pack-shield/references/fedramp/appendices-guide.md +209 -209
- package/src/bmad-plus/packs/pack-shield/references/fedramp/control-families.md +281 -281
- package/src/bmad-plus/packs/pack-shield/references/fedramp/poam-guide.md +93 -93
- package/src/bmad-plus/packs/pack-shield/references/fedramp/readiness-checklist.md +134 -134
- package/src/bmad-plus/packs/pack-shield/references/fedramp/sap-sar-guide.md +86 -86
- package/src/bmad-plus/packs/pack-shield/references/fedramp/ssp-guide.md +129 -129
- package/src/bmad-plus/packs/pack-shield/references/gdpr-compliance/documents.md +192 -192
- package/src/bmad-plus/packs/pack-shield/references/gdpr-compliance/dpa-template.md +121 -121
- package/src/bmad-plus/packs/pack-shield/references/gdpr-compliance/privacy-notice.md +87 -87
- package/src/bmad-plus/packs/pack-shield/references/hipaa-compliance/breach-notification.md +293 -293
- package/src/bmad-plus/packs/pack-shield/references/hipaa-compliance/privacy-rule.md +276 -276
- package/src/bmad-plus/packs/pack-shield/references/hipaa-compliance/security-rule.md +299 -299
- package/src/bmad-plus/packs/pack-shield/references/hipaa-compliance/templates.md +568 -568
- package/src/bmad-plus/packs/pack-shield/references/ism/control-applicability.md +181 -181
- package/src/bmad-plus/packs/pack-shield/references/ism/guidelines-overview.md +183 -183
- package/src/bmad-plus/packs/pack-shield/references/iso27001/annex-a-2013.md +203 -203
- package/src/bmad-plus/packs/pack-shield/references/iso27001/annex-a-2022.md +132 -132
- package/src/bmad-plus/packs/pack-shield/references/iso27001/control-mapping.md +153 -153
- package/src/bmad-plus/packs/pack-shield/references/iso27701/annex-a-controls.md +195 -195
- package/src/bmad-plus/packs/pack-shield/references/iso27701/regulatory-mapping.md +229 -229
- package/src/bmad-plus/packs/pack-shield/references/iso27701/transition-guide.md +219 -219
- package/src/bmad-plus/packs/pack-shield/references/iso42001/iso42001-ai-risk-assessment.md +258 -258
- package/src/bmad-plus/packs/pack-shield/references/iso42001/iso42001-clauses-requirements.md +279 -279
- package/src/bmad-plus/packs/pack-shield/references/iso42001/iso42001-controls-annex-a.md +155 -155
- package/src/bmad-plus/packs/pack-shield/references/itar/compliance-program.md +174 -174
- package/src/bmad-plus/packs/pack-shield/references/itar/licensing-guide.md +146 -146
- package/src/bmad-plus/packs/pack-shield/references/itar/usml-categories.md +93 -93
- package/src/bmad-plus/packs/pack-shield/references/lgpd/anpd-enforcement.md +147 -147
- package/src/bmad-plus/packs/pack-shield/references/lgpd/compliance-program.md +272 -272
- package/src/bmad-plus/packs/pack-shield/references/lgpd/lgpd-articles.md +271 -271
- package/src/bmad-plus/packs/pack-shield/references/nis2/article-21-measures.md +153 -153
- package/src/bmad-plus/packs/pack-shield/references/nis2/iso27001-nis2-mapping.md +68 -68
- package/src/bmad-plus/packs/pack-shield/references/nist-800-53/assessment-rmf.md +349 -349
- package/src/bmad-plus/packs/pack-shield/references/nist-800-53/baselines-tailoring.md +277 -277
- package/src/bmad-plus/packs/pack-shield/references/nist-800-53/control-families.md +450 -450
- package/src/bmad-plus/packs/pack-shield/references/nist-ai-rmf/rmf-core.md +361 -361
- package/src/bmad-plus/packs/pack-shield/references/nist-ai-rmf/rmf-profiles.md +192 -192
- package/src/bmad-plus/packs/pack-shield/references/nist-csf/csf-10-to-20-mapping.md +143 -143
- package/src/bmad-plus/packs/pack-shield/references/nist-csf/csf-20-functions-categories.md +278 -278
- package/src/bmad-plus/packs/pack-shield/references/nist-csf/csf-implementation-tiers.md +135 -135
- package/src/bmad-plus/packs/pack-shield/references/pci-compliance/pci-dss-requirements.md +366 -366
- package/src/bmad-plus/packs/pack-shield/references/pci-compliance/pci-dss-saq-guide.md +217 -217
- package/src/bmad-plus/packs/pack-shield/references/pci-compliance/pci-dss-v4-changes.md +190 -190
- package/src/bmad-plus/packs/pack-shield/references/section-508/wcag-mapping.md +160 -160
- package/src/bmad-plus/packs/pack-shield/references/soc2/controls.md +241 -241
- package/src/bmad-plus/packs/pack-shield/references/soc2/evidence.md +236 -236
- package/src/bmad-plus/packs/pack-shield/references/soc2/policies.md +254 -254
- package/src/bmad-plus/packs/pack-shield/references/soc2/vendor.md +276 -276
- package/src/bmad-plus/packs/pack-shield/references/swift-csp/swift-assessment.md +202 -202
- package/src/bmad-plus/packs/pack-shield/references/swift-csp/swift-controls.md +545 -545
- package/src/bmad-plus/packs/pack-shield/references/tsa-compliance/tsa-crmp-requirements.md +359 -359
- package/src/bmad-plus/packs/pack-shield/references/tsa-compliance/tsa-directives-overview.md +187 -187
- package/src/bmad-plus/packs/pack-shield/references/tsa-compliance/tsa-incident-reporting.md +187 -187
- package/src/bmad-plus/packs/pack-shield/references/wcag/criteria-detail.md +510 -510
- package/src/bmad-plus/packs/pack-shield/shared/audit-report-template.md +103 -103
- package/src/bmad-plus/packs/pack-shield/shared/cross-framework-mapper.md +103 -103
- package/src/bmad-plus/packs/pack-shield/shared/gap-analysis-template.md +83 -83
- package/src/bmad-plus/packs/pack-shield/shield-orchestrator.md +229 -229
- package/src/bmad-plus/packs/pack-shield/upstream-sync.yaml +68 -68
- package/src/bmad-plus/skills/bmad-plus-autopilot/SKILL.md +99 -99
- package/src/bmad-plus/skills/bmad-plus-parallel/SKILL.md +93 -93
- package/src/bmad-plus/skills/bmad-plus-sync/SKILL.md +69 -69
- package/tools/cli/commands/autoconfig.js +498 -489
- package/tools/cli/commands/doctor.js +222 -175
- package/tools/cli/commands/install.js +739 -739
- package/tools/cli/commands/memory.js +194 -194
- package/tools/cli/commands/scan.js +360 -350
- package/tools/cli/commands/uninstall.js +96 -96
- package/tools/cli/commands/update.js +174 -174
- package/tools/cli/i18n.js +763 -763
|
@@ -1,489 +1,498 @@
|
|
|
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
|
-
|
|
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
|
+
// Exported for functional testing (not part of the public API)
|
|
490
|
+
_internal: {
|
|
491
|
+
detectStack,
|
|
492
|
+
analyzeStructure,
|
|
493
|
+
calculateHealth,
|
|
494
|
+
recommendPacks,
|
|
495
|
+
generateRecommendations,
|
|
496
|
+
getProjectName,
|
|
497
|
+
},
|
|
498
|
+
};
|