bmad-plus 0.7.5 → 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.
Files changed (294) hide show
  1. package/CHANGELOG.md +450 -425
  2. package/LICENSE +21 -21
  3. package/README.md +555 -447
  4. package/osint-agent-package/README.md +88 -88
  5. package/osint-agent-package/SETUP_KEYS.md +108 -108
  6. package/osint-agent-package/agents/osint-investigator.md +80 -80
  7. package/osint-agent-package/install.ps1 +87 -87
  8. package/osint-agent-package/install.sh +76 -76
  9. package/osint-agent-package/skills/bmad-osint-investigate/SKILL.md +147 -147
  10. package/osint-agent-package/skills/bmad-osint-investigate/osint/SKILL.md +452 -452
  11. package/osint-agent-package/skills/bmad-osint-investigate/osint/assets/dossier-template.md +116 -116
  12. package/osint-agent-package/skills/bmad-osint-investigate/osint/references/content-extraction.md +100 -100
  13. package/osint-agent-package/skills/bmad-osint-investigate/osint/references/enrichment-databases-fr.md +148 -148
  14. package/osint-agent-package/skills/bmad-osint-investigate/osint/references/platforms.md +130 -130
  15. package/osint-agent-package/skills/bmad-osint-investigate/osint/references/psychoprofile.md +69 -69
  16. package/osint-agent-package/skills/bmad-osint-investigate/osint/references/tools.md +281 -281
  17. package/osint-agent-package/skills/bmad-osint-investigate/osint/scripts/_http.py +101 -101
  18. package/osint-agent-package/skills/bmad-osint-investigate/osint/scripts/apify.py +266 -260
  19. package/osint-agent-package/skills/bmad-osint-investigate/osint/scripts/brightdata.py +101 -101
  20. package/osint-agent-package/skills/bmad-osint-investigate/osint/scripts/diagnose.py +141 -141
  21. package/osint-agent-package/skills/bmad-osint-investigate/osint/scripts/exa.py +79 -79
  22. package/osint-agent-package/skills/bmad-osint-investigate/osint/scripts/jina.py +71 -71
  23. package/osint-agent-package/skills/bmad-osint-investigate/osint/scripts/mcp-client.py +136 -136
  24. package/osint-agent-package/skills/bmad-osint-investigate/osint/scripts/parallel.py +85 -85
  25. package/osint-agent-package/skills/bmad-osint-investigate/osint/scripts/perplexity.py +102 -102
  26. package/osint-agent-package/skills/bmad-osint-investigate/osint/scripts/tavily.py +72 -72
  27. package/osint-agent-package/skills/bmad-osint-investigate/osint/scripts/volley.py +208 -208
  28. package/osint-agent-package/skills/bmad-osint-investigator/SKILL.md +15 -15
  29. package/package.json +62 -57
  30. package/readme-international/README.de.md +576 -426
  31. package/readme-international/README.es.md +578 -518
  32. package/readme-international/README.fr.md +576 -516
  33. package/src/bmad-plus/agents/agent-architect-dev/SKILL.md +96 -96
  34. package/src/bmad-plus/agents/agent-architect-dev/bmad-skill-manifest.yaml +13 -13
  35. package/src/bmad-plus/agents/agent-maker/SKILL.md +201 -201
  36. package/src/bmad-plus/agents/agent-maker/bmad-skill-manifest.yaml +13 -13
  37. package/src/bmad-plus/agents/agent-orchestrator/SKILL.md +137 -137
  38. package/src/bmad-plus/agents/agent-orchestrator/bmad-skill-manifest.yaml +13 -13
  39. package/src/bmad-plus/agents/agent-quality/SKILL.md +83 -83
  40. package/src/bmad-plus/agents/agent-quality/bmad-skill-manifest.yaml +13 -13
  41. package/src/bmad-plus/agents/agent-shadow/SKILL.md +71 -71
  42. package/src/bmad-plus/agents/agent-shadow/bmad-skill-manifest.yaml +13 -13
  43. package/src/bmad-plus/agents/agent-strategist/SKILL.md +80 -80
  44. package/src/bmad-plus/agents/agent-strategist/bmad-skill-manifest.yaml +13 -13
  45. package/src/bmad-plus/agents/pack-animated/animated-website-agent.md +325 -325
  46. package/src/bmad-plus/agents/pack-animated/templates/animated-website-workflow.md +55 -55
  47. package/src/bmad-plus/agents/pack-backup/backup-agent.md +71 -71
  48. package/src/bmad-plus/agents/pack-backup/templates/backup-workflow.md +51 -51
  49. package/src/bmad-plus/agents/pack-seo/SKILL.md +171 -171
  50. package/src/bmad-plus/agents/pack-seo/checklist.md +140 -140
  51. package/src/bmad-plus/agents/pack-seo/pagespeed-playbook.md +320 -320
  52. package/src/bmad-plus/agents/pack-seo/ref/audit-schema.json +187 -187
  53. package/src/bmad-plus/agents/pack-seo/ref/cwv-thresholds.md +87 -87
  54. package/src/bmad-plus/agents/pack-seo/ref/eeat-criteria.md +123 -123
  55. package/src/bmad-plus/agents/pack-seo/ref/geo-signals.md +167 -167
  56. package/src/bmad-plus/agents/pack-seo/ref/hreflang-rules.md +153 -153
  57. package/src/bmad-plus/agents/pack-seo/ref/quality-gates.md +133 -133
  58. package/src/bmad-plus/agents/pack-seo/ref/schema-catalog.md +91 -91
  59. package/src/bmad-plus/agents/pack-seo/ref/schema-templates.json +356 -356
  60. package/src/bmad-plus/agents/pack-seo/seo-chief.md +294 -294
  61. package/src/bmad-plus/agents/pack-seo/seo-judge.md +241 -241
  62. package/src/bmad-plus/agents/pack-seo/seo-scout.md +171 -171
  63. package/src/bmad-plus/agents/pack-seo/templates/seo-audit-workflow.md +241 -241
  64. package/src/bmad-plus/data/role-triggers.yaml +209 -209
  65. package/src/bmad-plus/module-help.csv +10 -10
  66. package/src/bmad-plus/module.yaml +283 -280
  67. package/src/bmad-plus/packs/pack-animated/animated-website-agent.md +325 -0
  68. package/src/bmad-plus/packs/pack-animated/templates/animated-website-workflow.md +55 -0
  69. package/src/bmad-plus/packs/pack-backup/backup-agent.md +71 -0
  70. package/src/bmad-plus/packs/pack-backup/templates/backup-workflow.md +51 -0
  71. package/src/bmad-plus/packs/pack-dev-studio/README.md +162 -162
  72. package/src/bmad-plus/packs/pack-dev-studio/categories/analysis/analyst-agent.md +73 -73
  73. package/src/bmad-plus/packs/pack-dev-studio/categories/analysis/document-project.md +61 -61
  74. package/src/bmad-plus/packs/pack-dev-studio/categories/analysis/domain-research.md +95 -95
  75. package/src/bmad-plus/packs/pack-dev-studio/categories/analysis/market-research.md +95 -95
  76. package/src/bmad-plus/packs/pack-dev-studio/categories/analysis/prfaq.md +134 -134
  77. package/src/bmad-plus/packs/pack-dev-studio/categories/analysis/product-brief.md +80 -80
  78. package/src/bmad-plus/packs/pack-dev-studio/categories/analysis/tech-writer-agent.md +73 -73
  79. package/src/bmad-plus/packs/pack-dev-studio/categories/analysis/technical-research.md +95 -95
  80. package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/architect-agent.md +73 -73
  81. package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/create-architecture.md +73 -73
  82. package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/create-epics-stories.md +92 -92
  83. package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/generate-project-context.md +80 -80
  84. package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/implementation-readiness.md +90 -90
  85. package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/steps/step-01-init.md +153 -153
  86. package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/steps/step-01b-continue.md +173 -173
  87. package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/steps/step-02-context.md +224 -224
  88. package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/steps/step-03-starter.md +329 -329
  89. package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/steps/step-04-decisions.md +318 -318
  90. package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/steps/step-05-patterns.md +359 -359
  91. package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/steps/step-06-structure.md +379 -379
  92. package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/steps/step-07-validation.md +361 -361
  93. package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/steps/step-08-complete.md +81 -81
  94. package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/checkpoint-preview.md +67 -67
  95. package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/code-review-steps/step-01-gather-context.md +85 -85
  96. package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/code-review-steps/step-02-review.md +35 -35
  97. package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/code-review-steps/step-03-triage.md +49 -49
  98. package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/code-review-steps/step-04-present.md +131 -131
  99. package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/code-review.md +89 -89
  100. package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/correct-course.md +300 -300
  101. package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/create-story.md +428 -428
  102. package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/dev-agent.md +73 -73
  103. package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/dev-story-checklist.md +80 -80
  104. package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/dev-story.md +484 -484
  105. package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/investigate.md +193 -193
  106. package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/qa-e2e-tests.md +175 -175
  107. package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/quick-dev.md +110 -110
  108. package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/retrospective.md +1511 -1511
  109. package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/sprint-planning.md +298 -298
  110. package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/sprint-status.md +296 -296
  111. package/src/bmad-plus/packs/pack-dev-studio/categories/planning/create-prd.md +29 -29
  112. package/src/bmad-plus/packs/pack-dev-studio/categories/planning/create-ux-design.md +74 -74
  113. package/src/bmad-plus/packs/pack-dev-studio/categories/planning/edit-prd.md +29 -29
  114. package/src/bmad-plus/packs/pack-dev-studio/categories/planning/pm-agent.md +73 -73
  115. package/src/bmad-plus/packs/pack-dev-studio/categories/planning/prd.md +89 -89
  116. package/src/bmad-plus/packs/pack-dev-studio/categories/planning/ux-designer-agent.md +73 -73
  117. package/src/bmad-plus/packs/pack-dev-studio/categories/planning/validate-prd.md +29 -29
  118. package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/advanced-elicitation.md +141 -141
  119. package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/adversarial-review.md +37 -37
  120. package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/bmad-help.md +75 -75
  121. package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/brainstorming.md +6 -6
  122. package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/customize.md +110 -110
  123. package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/distillator.md +176 -176
  124. package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/edge-case-hunter.md +67 -67
  125. package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/editorial-review-prose.md +86 -86
  126. package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/editorial-review-structure.md +179 -179
  127. package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/index-docs.md +66 -66
  128. package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/party-mode.md +127 -127
  129. package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/shard-doc.md +105 -105
  130. package/src/bmad-plus/packs/pack-dev-studio/dev-studio-orchestrator.md +120 -120
  131. package/src/bmad-plus/packs/pack-dev-studio/shared/architecture-decision-template.md +12 -12
  132. package/src/bmad-plus/packs/pack-dev-studio/shared/bwml-spec.md +328 -328
  133. package/src/bmad-plus/packs/pack-dev-studio/shared/module-help.csv +32 -32
  134. package/src/bmad-plus/packs/pack-dev-studio/upstream-sync.yaml +81 -81
  135. package/src/bmad-plus/packs/pack-memory/README.md +106 -106
  136. package/src/bmad-plus/packs/pack-memory/memory-orchestrator.md +79 -79
  137. package/src/bmad-plus/packs/pack-memory/shared/karpathy-guardrails.md +86 -86
  138. package/src/bmad-plus/packs/pack-memory/shared/memory-protocol.md +143 -143
  139. package/src/bmad-plus/packs/pack-memory/templates/context.md +39 -39
  140. package/src/bmad-plus/packs/pack-memory/templates/decisions.md +25 -25
  141. package/src/bmad-plus/packs/pack-memory/templates/identity.yaml +39 -39
  142. package/src/bmad-plus/packs/pack-memory/templates/lessons.md +31 -31
  143. package/src/bmad-plus/packs/pack-memory/templates/patterns.md +24 -24
  144. package/src/bmad-plus/packs/pack-memory/templates/session-handoff.md +25 -25
  145. package/src/bmad-plus/packs/pack-memory/zecher-agent.md +157 -157
  146. package/src/bmad-plus/packs/pack-seo/SKILL.md +171 -0
  147. package/src/bmad-plus/packs/pack-seo/checklist.md +140 -0
  148. package/src/bmad-plus/packs/pack-seo/pagespeed-playbook.md +320 -0
  149. package/src/bmad-plus/packs/pack-seo/ref/audit-schema.json +187 -0
  150. package/src/bmad-plus/packs/pack-seo/ref/cwv-thresholds.md +87 -0
  151. package/src/bmad-plus/packs/pack-seo/ref/eeat-criteria.md +123 -0
  152. package/src/bmad-plus/packs/pack-seo/ref/geo-signals.md +167 -0
  153. package/src/bmad-plus/packs/pack-seo/ref/hreflang-rules.md +153 -0
  154. package/src/bmad-plus/packs/pack-seo/ref/quality-gates.md +133 -0
  155. package/src/bmad-plus/packs/pack-seo/ref/schema-catalog.md +91 -0
  156. package/src/bmad-plus/packs/pack-seo/ref/schema-templates.json +356 -0
  157. package/src/bmad-plus/packs/pack-seo/seo-chief.md +294 -0
  158. package/src/bmad-plus/packs/pack-seo/seo-judge.md +241 -0
  159. package/src/bmad-plus/packs/pack-seo/seo-scout.md +171 -0
  160. package/src/bmad-plus/packs/pack-seo/templates/seo-audit-workflow.md +241 -0
  161. package/src/bmad-plus/packs/pack-shield/README.md +110 -110
  162. package/src/bmad-plus/packs/pack-shield/categories/accessibility-esg/csrd-agent.md +262 -262
  163. package/src/bmad-plus/packs/pack-shield/categories/accessibility-esg/section508-agent.md +179 -179
  164. package/src/bmad-plus/packs/pack-shield/categories/accessibility-esg/wcag-agent.md +201 -201
  165. package/src/bmad-plus/packs/pack-shield/categories/ai-governance/eu-ai-act-agent.md +97 -97
  166. package/src/bmad-plus/packs/pack-shield/categories/ai-governance/iso42001-agent.md +251 -251
  167. package/src/bmad-plus/packs/pack-shield/categories/ai-governance/nist-ai-rmf-agent.md +133 -133
  168. package/src/bmad-plus/packs/pack-shield/categories/cybersecurity/cis-controls-agent.md +221 -221
  169. package/src/bmad-plus/packs/pack-shield/categories/cybersecurity/ism-agent.md +150 -150
  170. package/src/bmad-plus/packs/pack-shield/categories/cybersecurity/iso27001-agent.md +167 -167
  171. package/src/bmad-plus/packs/pack-shield/categories/cybersecurity/nis2-agent.md +83 -83
  172. package/src/bmad-plus/packs/pack-shield/categories/cybersecurity/nist-800-53-agent.md +250 -250
  173. package/src/bmad-plus/packs/pack-shield/categories/cybersecurity/nist-csf-agent.md +218 -218
  174. package/src/bmad-plus/packs/pack-shield/categories/data-privacy/ccpa-agent.md +94 -94
  175. package/src/bmad-plus/packs/pack-shield/categories/data-privacy/dpdpa-agent.md +136 -136
  176. package/src/bmad-plus/packs/pack-shield/categories/data-privacy/gdpr-agent.md +296 -296
  177. package/src/bmad-plus/packs/pack-shield/categories/data-privacy/iso27701-agent.md +134 -134
  178. package/src/bmad-plus/packs/pack-shield/categories/data-privacy/lgpd-agent.md +129 -129
  179. package/src/bmad-plus/packs/pack-shield/categories/defense-export/cmmc-agent.md +127 -127
  180. package/src/bmad-plus/packs/pack-shield/categories/defense-export/ear-agent.md +272 -272
  181. package/src/bmad-plus/packs/pack-shield/categories/defense-export/itar-agent.md +202 -202
  182. package/src/bmad-plus/packs/pack-shield/categories/defense-export/tsa-agent.md +367 -367
  183. package/src/bmad-plus/packs/pack-shield/categories/industry-compliance/dora-agent.md +510 -510
  184. package/src/bmad-plus/packs/pack-shield/categories/industry-compliance/fedramp-agent.md +247 -247
  185. package/src/bmad-plus/packs/pack-shield/categories/industry-compliance/hipaa-agent.md +173 -173
  186. package/src/bmad-plus/packs/pack-shield/categories/industry-compliance/pci-dss-agent.md +239 -239
  187. package/src/bmad-plus/packs/pack-shield/categories/industry-compliance/soc2-agent.md +266 -266
  188. package/src/bmad-plus/packs/pack-shield/categories/industry-compliance/swift-csp-agent.md +164 -164
  189. package/src/bmad-plus/packs/pack-shield/categories/workflows/ai-act-classifier.md +131 -131
  190. package/src/bmad-plus/packs/pack-shield/categories/workflows/ai-act-fria.md +155 -155
  191. package/src/bmad-plus/packs/pack-shield/categories/workflows/ai-act-incidents.md +187 -187
  192. package/src/bmad-plus/packs/pack-shield/categories/workflows/ai-act-roles.md +113 -113
  193. package/src/bmad-plus/packs/pack-shield/categories/workflows/breach-sentinel.md +197 -197
  194. package/src/bmad-plus/packs/pack-shield/categories/workflows/cookie-policy-gen.md +180 -180
  195. package/src/bmad-plus/packs/pack-shield/categories/workflows/dpia-sentinel.md +235 -235
  196. package/src/bmad-plus/packs/pack-shield/categories/workflows/legitimate-interest.md +159 -159
  197. package/src/bmad-plus/packs/pack-shield/categories/workflows/privacy-advisor.md +133 -133
  198. package/src/bmad-plus/packs/pack-shield/categories/workflows/privacy-notice-gen.md +160 -160
  199. package/src/bmad-plus/packs/pack-shield/categories/workflows/privacy-policy-gen.md +135 -135
  200. package/src/bmad-plus/packs/pack-shield/references/ccpa/ccpa-gdpr-comparison.md +117 -117
  201. package/src/bmad-plus/packs/pack-shield/references/ccpa/consumer-rights-workflows.md +177 -177
  202. package/src/bmad-plus/packs/pack-shield/references/cis-controls/framework-mappings.md +162 -162
  203. package/src/bmad-plus/packs/pack-shield/references/cis-controls/implementation-guidance.md +235 -235
  204. package/src/bmad-plus/packs/pack-shield/references/cis-controls/safeguards-detail.md +252 -252
  205. package/src/bmad-plus/packs/pack-shield/references/cmmc/cmmc-assessment.md +170 -170
  206. package/src/bmad-plus/packs/pack-shield/references/cmmc/cmmc-levels.md +113 -113
  207. package/src/bmad-plus/packs/pack-shield/references/cmmc/cmmc-practices.md +211 -211
  208. package/src/bmad-plus/packs/pack-shield/references/csrd/compliance-program.md +281 -281
  209. package/src/bmad-plus/packs/pack-shield/references/csrd/double-materiality.md +253 -253
  210. package/src/bmad-plus/packs/pack-shield/references/csrd/esrs-standards.md +401 -401
  211. package/src/bmad-plus/packs/pack-shield/references/dora/article-reference.md +441 -441
  212. package/src/bmad-plus/packs/pack-shield/references/dora/incident-classification.md +297 -297
  213. package/src/bmad-plus/packs/pack-shield/references/dora/rts-its-guide.md +306 -306
  214. package/src/bmad-plus/packs/pack-shield/references/dora/third-party-risk.md +349 -349
  215. package/src/bmad-plus/packs/pack-shield/references/dpdpa/gdpr-comparison.md +173 -173
  216. package/src/bmad-plus/packs/pack-shield/references/dpdpa/rights-and-obligations.md +426 -426
  217. package/src/bmad-plus/packs/pack-shield/references/dpdpa/rules-2025.md +599 -599
  218. package/src/bmad-plus/packs/pack-shield/references/dpdpa/sections-reference.md +319 -319
  219. package/src/bmad-plus/packs/pack-shield/references/ear/ccl-eccn-guide.md +250 -250
  220. package/src/bmad-plus/packs/pack-shield/references/ear/compliance-program.md +280 -280
  221. package/src/bmad-plus/packs/pack-shield/references/ear/license-exceptions.md +207 -207
  222. package/src/bmad-plus/packs/pack-shield/references/eu-ai-act/gpai-governance.md +267 -267
  223. package/src/bmad-plus/packs/pack-shield/references/eu-ai-act/obligations-high-risk.md +287 -287
  224. package/src/bmad-plus/packs/pack-shield/references/eu-ai-act/risk-classification.md +182 -182
  225. package/src/bmad-plus/packs/pack-shield/references/fedramp/appendices-guide.md +209 -209
  226. package/src/bmad-plus/packs/pack-shield/references/fedramp/control-families.md +281 -281
  227. package/src/bmad-plus/packs/pack-shield/references/fedramp/poam-guide.md +93 -93
  228. package/src/bmad-plus/packs/pack-shield/references/fedramp/readiness-checklist.md +134 -134
  229. package/src/bmad-plus/packs/pack-shield/references/fedramp/sap-sar-guide.md +86 -86
  230. package/src/bmad-plus/packs/pack-shield/references/fedramp/ssp-guide.md +129 -129
  231. package/src/bmad-plus/packs/pack-shield/references/gdpr-compliance/documents.md +192 -192
  232. package/src/bmad-plus/packs/pack-shield/references/gdpr-compliance/dpa-template.md +121 -121
  233. package/src/bmad-plus/packs/pack-shield/references/gdpr-compliance/privacy-notice.md +87 -87
  234. package/src/bmad-plus/packs/pack-shield/references/hipaa-compliance/breach-notification.md +293 -293
  235. package/src/bmad-plus/packs/pack-shield/references/hipaa-compliance/privacy-rule.md +276 -276
  236. package/src/bmad-plus/packs/pack-shield/references/hipaa-compliance/security-rule.md +299 -299
  237. package/src/bmad-plus/packs/pack-shield/references/hipaa-compliance/templates.md +568 -568
  238. package/src/bmad-plus/packs/pack-shield/references/ism/control-applicability.md +181 -181
  239. package/src/bmad-plus/packs/pack-shield/references/ism/guidelines-overview.md +183 -183
  240. package/src/bmad-plus/packs/pack-shield/references/iso27001/annex-a-2013.md +203 -203
  241. package/src/bmad-plus/packs/pack-shield/references/iso27001/annex-a-2022.md +132 -132
  242. package/src/bmad-plus/packs/pack-shield/references/iso27001/control-mapping.md +153 -153
  243. package/src/bmad-plus/packs/pack-shield/references/iso27701/annex-a-controls.md +195 -195
  244. package/src/bmad-plus/packs/pack-shield/references/iso27701/regulatory-mapping.md +229 -229
  245. package/src/bmad-plus/packs/pack-shield/references/iso27701/transition-guide.md +219 -219
  246. package/src/bmad-plus/packs/pack-shield/references/iso42001/iso42001-ai-risk-assessment.md +258 -258
  247. package/src/bmad-plus/packs/pack-shield/references/iso42001/iso42001-clauses-requirements.md +279 -279
  248. package/src/bmad-plus/packs/pack-shield/references/iso42001/iso42001-controls-annex-a.md +155 -155
  249. package/src/bmad-plus/packs/pack-shield/references/itar/compliance-program.md +174 -174
  250. package/src/bmad-plus/packs/pack-shield/references/itar/licensing-guide.md +146 -146
  251. package/src/bmad-plus/packs/pack-shield/references/itar/usml-categories.md +93 -93
  252. package/src/bmad-plus/packs/pack-shield/references/lgpd/anpd-enforcement.md +147 -147
  253. package/src/bmad-plus/packs/pack-shield/references/lgpd/compliance-program.md +272 -272
  254. package/src/bmad-plus/packs/pack-shield/references/lgpd/lgpd-articles.md +271 -271
  255. package/src/bmad-plus/packs/pack-shield/references/nis2/article-21-measures.md +153 -153
  256. package/src/bmad-plus/packs/pack-shield/references/nis2/iso27001-nis2-mapping.md +68 -68
  257. package/src/bmad-plus/packs/pack-shield/references/nist-800-53/assessment-rmf.md +349 -349
  258. package/src/bmad-plus/packs/pack-shield/references/nist-800-53/baselines-tailoring.md +277 -277
  259. package/src/bmad-plus/packs/pack-shield/references/nist-800-53/control-families.md +450 -450
  260. package/src/bmad-plus/packs/pack-shield/references/nist-ai-rmf/rmf-core.md +361 -361
  261. package/src/bmad-plus/packs/pack-shield/references/nist-ai-rmf/rmf-profiles.md +192 -192
  262. package/src/bmad-plus/packs/pack-shield/references/nist-csf/csf-10-to-20-mapping.md +143 -143
  263. package/src/bmad-plus/packs/pack-shield/references/nist-csf/csf-20-functions-categories.md +278 -278
  264. package/src/bmad-plus/packs/pack-shield/references/nist-csf/csf-implementation-tiers.md +135 -135
  265. package/src/bmad-plus/packs/pack-shield/references/pci-compliance/pci-dss-requirements.md +366 -366
  266. package/src/bmad-plus/packs/pack-shield/references/pci-compliance/pci-dss-saq-guide.md +217 -217
  267. package/src/bmad-plus/packs/pack-shield/references/pci-compliance/pci-dss-v4-changes.md +190 -190
  268. package/src/bmad-plus/packs/pack-shield/references/section-508/wcag-mapping.md +160 -160
  269. package/src/bmad-plus/packs/pack-shield/references/soc2/controls.md +241 -241
  270. package/src/bmad-plus/packs/pack-shield/references/soc2/evidence.md +236 -236
  271. package/src/bmad-plus/packs/pack-shield/references/soc2/policies.md +254 -254
  272. package/src/bmad-plus/packs/pack-shield/references/soc2/vendor.md +276 -276
  273. package/src/bmad-plus/packs/pack-shield/references/swift-csp/swift-assessment.md +202 -202
  274. package/src/bmad-plus/packs/pack-shield/references/swift-csp/swift-controls.md +545 -545
  275. package/src/bmad-plus/packs/pack-shield/references/tsa-compliance/tsa-crmp-requirements.md +359 -359
  276. package/src/bmad-plus/packs/pack-shield/references/tsa-compliance/tsa-directives-overview.md +187 -187
  277. package/src/bmad-plus/packs/pack-shield/references/tsa-compliance/tsa-incident-reporting.md +187 -187
  278. package/src/bmad-plus/packs/pack-shield/references/wcag/criteria-detail.md +510 -510
  279. package/src/bmad-plus/packs/pack-shield/shared/audit-report-template.md +103 -103
  280. package/src/bmad-plus/packs/pack-shield/shared/cross-framework-mapper.md +103 -103
  281. package/src/bmad-plus/packs/pack-shield/shared/gap-analysis-template.md +83 -83
  282. package/src/bmad-plus/packs/pack-shield/shield-orchestrator.md +229 -229
  283. package/src/bmad-plus/packs/pack-shield/upstream-sync.yaml +68 -68
  284. package/src/bmad-plus/skills/bmad-plus-autopilot/SKILL.md +99 -99
  285. package/src/bmad-plus/skills/bmad-plus-parallel/SKILL.md +93 -93
  286. package/src/bmad-plus/skills/bmad-plus-sync/SKILL.md +69 -69
  287. package/tools/cli/commands/autoconfig.js +498 -489
  288. package/tools/cli/commands/doctor.js +222 -222
  289. package/tools/cli/commands/install.js +739 -739
  290. package/tools/cli/commands/memory.js +194 -194
  291. package/tools/cli/commands/scan.js +360 -350
  292. package/tools/cli/commands/uninstall.js +96 -96
  293. package/tools/cli/commands/update.js +174 -174
  294. 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
+ };