bmad-plus 0.7.5 → 0.9.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 (281) hide show
  1. package/CHANGELOG.md +479 -425
  2. package/LICENSE +21 -21
  3. package/README.md +557 -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 +584 -426
  31. package/readme-international/README.es.md +601 -518
  32. package/readme-international/README.fr.md +599 -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/data/role-triggers.yaml +209 -209
  46. package/src/bmad-plus/module-help.csv +10 -10
  47. package/src/bmad-plus/module.yaml +283 -280
  48. package/src/bmad-plus/{agents → packs}/pack-animated/animated-website-agent.md +325 -325
  49. package/src/bmad-plus/{agents → packs}/pack-animated/templates/animated-website-workflow.md +55 -55
  50. package/src/bmad-plus/{agents → packs}/pack-backup/backup-agent.md +71 -71
  51. package/src/bmad-plus/{agents → packs}/pack-backup/templates/backup-workflow.md +51 -51
  52. package/src/bmad-plus/packs/pack-dev-studio/README.md +162 -162
  53. package/src/bmad-plus/packs/pack-dev-studio/categories/analysis/analyst-agent.md +73 -73
  54. package/src/bmad-plus/packs/pack-dev-studio/categories/analysis/document-project.md +61 -61
  55. package/src/bmad-plus/packs/pack-dev-studio/categories/analysis/domain-research.md +95 -95
  56. package/src/bmad-plus/packs/pack-dev-studio/categories/analysis/market-research.md +95 -95
  57. package/src/bmad-plus/packs/pack-dev-studio/categories/analysis/prfaq.md +134 -134
  58. package/src/bmad-plus/packs/pack-dev-studio/categories/analysis/product-brief.md +80 -80
  59. package/src/bmad-plus/packs/pack-dev-studio/categories/analysis/tech-writer-agent.md +73 -73
  60. package/src/bmad-plus/packs/pack-dev-studio/categories/analysis/technical-research.md +95 -95
  61. package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/architect-agent.md +73 -73
  62. package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/create-architecture.md +73 -73
  63. package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/create-epics-stories.md +92 -92
  64. package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/generate-project-context.md +80 -80
  65. package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/implementation-readiness.md +90 -90
  66. package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/steps/step-01-init.md +153 -153
  67. package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/steps/step-01b-continue.md +173 -173
  68. package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/steps/step-02-context.md +224 -224
  69. package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/steps/step-03-starter.md +329 -329
  70. package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/steps/step-04-decisions.md +318 -318
  71. package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/steps/step-05-patterns.md +359 -359
  72. package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/steps/step-06-structure.md +379 -379
  73. package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/steps/step-07-validation.md +361 -361
  74. package/src/bmad-plus/packs/pack-dev-studio/categories/architecture/steps/step-08-complete.md +81 -81
  75. package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/checkpoint-preview.md +67 -67
  76. package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/code-review-steps/step-01-gather-context.md +85 -85
  77. package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/code-review-steps/step-02-review.md +35 -35
  78. package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/code-review-steps/step-03-triage.md +49 -49
  79. package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/code-review-steps/step-04-present.md +131 -131
  80. package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/code-review.md +89 -89
  81. package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/correct-course.md +300 -300
  82. package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/create-story.md +428 -428
  83. package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/dev-agent.md +73 -73
  84. package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/dev-story-checklist.md +80 -80
  85. package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/dev-story.md +484 -484
  86. package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/investigate.md +193 -193
  87. package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/qa-e2e-tests.md +175 -175
  88. package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/quick-dev.md +110 -110
  89. package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/retrospective.md +1511 -1511
  90. package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/sprint-planning.md +298 -298
  91. package/src/bmad-plus/packs/pack-dev-studio/categories/implementation/sprint-status.md +296 -296
  92. package/src/bmad-plus/packs/pack-dev-studio/categories/planning/create-prd.md +29 -29
  93. package/src/bmad-plus/packs/pack-dev-studio/categories/planning/create-ux-design.md +74 -74
  94. package/src/bmad-plus/packs/pack-dev-studio/categories/planning/edit-prd.md +29 -29
  95. package/src/bmad-plus/packs/pack-dev-studio/categories/planning/pm-agent.md +73 -73
  96. package/src/bmad-plus/packs/pack-dev-studio/categories/planning/prd.md +89 -89
  97. package/src/bmad-plus/packs/pack-dev-studio/categories/planning/ux-designer-agent.md +73 -73
  98. package/src/bmad-plus/packs/pack-dev-studio/categories/planning/validate-prd.md +29 -29
  99. package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/advanced-elicitation.md +141 -141
  100. package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/adversarial-review.md +37 -37
  101. package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/bmad-help.md +75 -75
  102. package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/brainstorming.md +6 -6
  103. package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/customize.md +110 -110
  104. package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/distillator.md +176 -176
  105. package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/edge-case-hunter.md +67 -67
  106. package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/editorial-review-prose.md +86 -86
  107. package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/editorial-review-structure.md +179 -179
  108. package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/index-docs.md +66 -66
  109. package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/party-mode.md +127 -127
  110. package/src/bmad-plus/packs/pack-dev-studio/categories/utilities/shard-doc.md +105 -105
  111. package/src/bmad-plus/packs/pack-dev-studio/dev-studio-orchestrator.md +120 -120
  112. package/src/bmad-plus/packs/pack-dev-studio/shared/architecture-decision-template.md +12 -12
  113. package/src/bmad-plus/packs/pack-dev-studio/shared/bwml-spec.md +328 -328
  114. package/src/bmad-plus/packs/pack-dev-studio/shared/module-help.csv +32 -32
  115. package/src/bmad-plus/packs/pack-dev-studio/upstream-sync.yaml +81 -81
  116. package/src/bmad-plus/packs/pack-memory/README.md +106 -106
  117. package/src/bmad-plus/packs/pack-memory/memory-orchestrator.md +79 -79
  118. package/src/bmad-plus/packs/pack-memory/shared/karpathy-guardrails.md +86 -86
  119. package/src/bmad-plus/packs/pack-memory/shared/memory-protocol.md +143 -143
  120. package/src/bmad-plus/packs/pack-memory/templates/context.md +39 -39
  121. package/src/bmad-plus/packs/pack-memory/templates/decisions.md +25 -25
  122. package/src/bmad-plus/packs/pack-memory/templates/identity.yaml +39 -39
  123. package/src/bmad-plus/packs/pack-memory/templates/lessons.md +31 -31
  124. package/src/bmad-plus/packs/pack-memory/templates/patterns.md +24 -24
  125. package/src/bmad-plus/packs/pack-memory/templates/session-handoff.md +25 -25
  126. package/src/bmad-plus/packs/pack-memory/zecher-agent.md +157 -157
  127. package/src/bmad-plus/{agents → packs}/pack-seo/SKILL.md +171 -171
  128. package/src/bmad-plus/packs/pack-seo/bmad-skill-manifest.yaml +13 -0
  129. package/src/bmad-plus/{agents → packs}/pack-seo/checklist.md +140 -140
  130. package/src/bmad-plus/{agents → packs}/pack-seo/pagespeed-playbook.md +320 -320
  131. package/src/bmad-plus/{agents → packs}/pack-seo/ref/audit-schema.json +187 -187
  132. package/src/bmad-plus/{agents → packs}/pack-seo/ref/cwv-thresholds.md +87 -87
  133. package/src/bmad-plus/{agents → packs}/pack-seo/ref/eeat-criteria.md +123 -123
  134. package/src/bmad-plus/{agents → packs}/pack-seo/ref/geo-signals.md +167 -167
  135. package/src/bmad-plus/{agents → packs}/pack-seo/ref/hreflang-rules.md +153 -153
  136. package/src/bmad-plus/{agents → packs}/pack-seo/ref/quality-gates.md +133 -133
  137. package/src/bmad-plus/{agents → packs}/pack-seo/ref/schema-catalog.md +91 -91
  138. package/src/bmad-plus/{agents → packs}/pack-seo/ref/schema-templates.json +356 -356
  139. package/src/bmad-plus/{agents → packs}/pack-seo/seo-chief.md +294 -294
  140. package/src/bmad-plus/{agents → packs}/pack-seo/seo-judge.md +241 -241
  141. package/src/bmad-plus/{agents → packs}/pack-seo/seo-scout.md +171 -171
  142. package/src/bmad-plus/{agents → packs}/pack-seo/templates/seo-audit-workflow.md +241 -241
  143. package/src/bmad-plus/packs/pack-shield/README.md +110 -110
  144. package/src/bmad-plus/packs/pack-shield/SKILL.md +82 -0
  145. package/src/bmad-plus/packs/pack-shield/categories/accessibility-esg/csrd-agent.md +262 -262
  146. package/src/bmad-plus/packs/pack-shield/categories/accessibility-esg/section508-agent.md +179 -179
  147. package/src/bmad-plus/packs/pack-shield/categories/accessibility-esg/wcag-agent.md +201 -201
  148. package/src/bmad-plus/packs/pack-shield/categories/ai-governance/eu-ai-act-agent.md +97 -97
  149. package/src/bmad-plus/packs/pack-shield/categories/ai-governance/iso42001-agent.md +251 -251
  150. package/src/bmad-plus/packs/pack-shield/categories/ai-governance/nist-ai-rmf-agent.md +133 -133
  151. package/src/bmad-plus/packs/pack-shield/categories/cybersecurity/cis-controls-agent.md +221 -221
  152. package/src/bmad-plus/packs/pack-shield/categories/cybersecurity/ism-agent.md +150 -150
  153. package/src/bmad-plus/packs/pack-shield/categories/cybersecurity/iso27001-agent.md +167 -167
  154. package/src/bmad-plus/packs/pack-shield/categories/cybersecurity/nis2-agent.md +83 -83
  155. package/src/bmad-plus/packs/pack-shield/categories/cybersecurity/nist-800-53-agent.md +250 -250
  156. package/src/bmad-plus/packs/pack-shield/categories/cybersecurity/nist-csf-agent.md +218 -218
  157. package/src/bmad-plus/packs/pack-shield/categories/data-privacy/ccpa-agent.md +94 -94
  158. package/src/bmad-plus/packs/pack-shield/categories/data-privacy/dpdpa-agent.md +136 -136
  159. package/src/bmad-plus/packs/pack-shield/categories/data-privacy/gdpr-agent.md +296 -296
  160. package/src/bmad-plus/packs/pack-shield/categories/data-privacy/iso27701-agent.md +134 -134
  161. package/src/bmad-plus/packs/pack-shield/categories/data-privacy/lgpd-agent.md +129 -129
  162. package/src/bmad-plus/packs/pack-shield/categories/defense-export/cmmc-agent.md +127 -127
  163. package/src/bmad-plus/packs/pack-shield/categories/defense-export/ear-agent.md +272 -272
  164. package/src/bmad-plus/packs/pack-shield/categories/defense-export/itar-agent.md +202 -202
  165. package/src/bmad-plus/packs/pack-shield/categories/defense-export/tsa-agent.md +367 -367
  166. package/src/bmad-plus/packs/pack-shield/categories/industry-compliance/dora-agent.md +510 -510
  167. package/src/bmad-plus/packs/pack-shield/categories/industry-compliance/fedramp-agent.md +247 -247
  168. package/src/bmad-plus/packs/pack-shield/categories/industry-compliance/hipaa-agent.md +173 -173
  169. package/src/bmad-plus/packs/pack-shield/categories/industry-compliance/pci-dss-agent.md +239 -239
  170. package/src/bmad-plus/packs/pack-shield/categories/industry-compliance/soc2-agent.md +266 -266
  171. package/src/bmad-plus/packs/pack-shield/categories/industry-compliance/swift-csp-agent.md +164 -164
  172. package/src/bmad-plus/packs/pack-shield/categories/workflows/ai-act-classifier.md +131 -131
  173. package/src/bmad-plus/packs/pack-shield/categories/workflows/ai-act-fria.md +155 -155
  174. package/src/bmad-plus/packs/pack-shield/categories/workflows/ai-act-incidents.md +187 -187
  175. package/src/bmad-plus/packs/pack-shield/categories/workflows/ai-act-roles.md +113 -113
  176. package/src/bmad-plus/packs/pack-shield/categories/workflows/breach-sentinel.md +197 -197
  177. package/src/bmad-plus/packs/pack-shield/categories/workflows/cookie-policy-gen.md +180 -180
  178. package/src/bmad-plus/packs/pack-shield/categories/workflows/dpia-sentinel.md +235 -235
  179. package/src/bmad-plus/packs/pack-shield/categories/workflows/legitimate-interest.md +159 -159
  180. package/src/bmad-plus/packs/pack-shield/categories/workflows/privacy-advisor.md +133 -133
  181. package/src/bmad-plus/packs/pack-shield/categories/workflows/privacy-notice-gen.md +160 -160
  182. package/src/bmad-plus/packs/pack-shield/categories/workflows/privacy-policy-gen.md +135 -135
  183. package/src/bmad-plus/packs/pack-shield/references/ccpa/ccpa-gdpr-comparison.md +117 -117
  184. package/src/bmad-plus/packs/pack-shield/references/ccpa/consumer-rights-workflows.md +177 -177
  185. package/src/bmad-plus/packs/pack-shield/references/cis-controls/framework-mappings.md +162 -162
  186. package/src/bmad-plus/packs/pack-shield/references/cis-controls/implementation-guidance.md +235 -235
  187. package/src/bmad-plus/packs/pack-shield/references/cis-controls/safeguards-detail.md +252 -252
  188. package/src/bmad-plus/packs/pack-shield/references/cmmc/cmmc-assessment.md +170 -170
  189. package/src/bmad-plus/packs/pack-shield/references/cmmc/cmmc-levels.md +113 -113
  190. package/src/bmad-plus/packs/pack-shield/references/cmmc/cmmc-practices.md +211 -211
  191. package/src/bmad-plus/packs/pack-shield/references/csrd/compliance-program.md +281 -281
  192. package/src/bmad-plus/packs/pack-shield/references/csrd/double-materiality.md +253 -253
  193. package/src/bmad-plus/packs/pack-shield/references/csrd/esrs-standards.md +401 -401
  194. package/src/bmad-plus/packs/pack-shield/references/dora/article-reference.md +441 -441
  195. package/src/bmad-plus/packs/pack-shield/references/dora/incident-classification.md +297 -297
  196. package/src/bmad-plus/packs/pack-shield/references/dora/rts-its-guide.md +306 -306
  197. package/src/bmad-plus/packs/pack-shield/references/dora/third-party-risk.md +349 -349
  198. package/src/bmad-plus/packs/pack-shield/references/dpdpa/gdpr-comparison.md +173 -173
  199. package/src/bmad-plus/packs/pack-shield/references/dpdpa/rights-and-obligations.md +426 -426
  200. package/src/bmad-plus/packs/pack-shield/references/dpdpa/rules-2025.md +599 -599
  201. package/src/bmad-plus/packs/pack-shield/references/dpdpa/sections-reference.md +319 -319
  202. package/src/bmad-plus/packs/pack-shield/references/ear/ccl-eccn-guide.md +250 -250
  203. package/src/bmad-plus/packs/pack-shield/references/ear/compliance-program.md +280 -280
  204. package/src/bmad-plus/packs/pack-shield/references/ear/license-exceptions.md +207 -207
  205. package/src/bmad-plus/packs/pack-shield/references/eu-ai-act/gpai-governance.md +267 -267
  206. package/src/bmad-plus/packs/pack-shield/references/eu-ai-act/obligations-high-risk.md +287 -287
  207. package/src/bmad-plus/packs/pack-shield/references/eu-ai-act/risk-classification.md +182 -182
  208. package/src/bmad-plus/packs/pack-shield/references/fedramp/appendices-guide.md +209 -209
  209. package/src/bmad-plus/packs/pack-shield/references/fedramp/control-families.md +281 -281
  210. package/src/bmad-plus/packs/pack-shield/references/fedramp/poam-guide.md +93 -93
  211. package/src/bmad-plus/packs/pack-shield/references/fedramp/readiness-checklist.md +134 -134
  212. package/src/bmad-plus/packs/pack-shield/references/fedramp/sap-sar-guide.md +86 -86
  213. package/src/bmad-plus/packs/pack-shield/references/fedramp/ssp-guide.md +129 -129
  214. package/src/bmad-plus/packs/pack-shield/references/gdpr-compliance/documents.md +192 -192
  215. package/src/bmad-plus/packs/pack-shield/references/gdpr-compliance/dpa-template.md +121 -121
  216. package/src/bmad-plus/packs/pack-shield/references/gdpr-compliance/privacy-notice.md +87 -87
  217. package/src/bmad-plus/packs/pack-shield/references/hipaa-compliance/breach-notification.md +293 -293
  218. package/src/bmad-plus/packs/pack-shield/references/hipaa-compliance/privacy-rule.md +276 -276
  219. package/src/bmad-plus/packs/pack-shield/references/hipaa-compliance/security-rule.md +299 -299
  220. package/src/bmad-plus/packs/pack-shield/references/hipaa-compliance/templates.md +568 -568
  221. package/src/bmad-plus/packs/pack-shield/references/ism/control-applicability.md +181 -181
  222. package/src/bmad-plus/packs/pack-shield/references/ism/guidelines-overview.md +183 -183
  223. package/src/bmad-plus/packs/pack-shield/references/iso27001/annex-a-2013.md +203 -203
  224. package/src/bmad-plus/packs/pack-shield/references/iso27001/annex-a-2022.md +132 -132
  225. package/src/bmad-plus/packs/pack-shield/references/iso27001/control-mapping.md +153 -153
  226. package/src/bmad-plus/packs/pack-shield/references/iso27701/annex-a-controls.md +195 -195
  227. package/src/bmad-plus/packs/pack-shield/references/iso27701/regulatory-mapping.md +229 -229
  228. package/src/bmad-plus/packs/pack-shield/references/iso27701/transition-guide.md +219 -219
  229. package/src/bmad-plus/packs/pack-shield/references/iso42001/iso42001-ai-risk-assessment.md +258 -258
  230. package/src/bmad-plus/packs/pack-shield/references/iso42001/iso42001-clauses-requirements.md +279 -279
  231. package/src/bmad-plus/packs/pack-shield/references/iso42001/iso42001-controls-annex-a.md +155 -155
  232. package/src/bmad-plus/packs/pack-shield/references/itar/compliance-program.md +174 -174
  233. package/src/bmad-plus/packs/pack-shield/references/itar/licensing-guide.md +146 -146
  234. package/src/bmad-plus/packs/pack-shield/references/itar/usml-categories.md +93 -93
  235. package/src/bmad-plus/packs/pack-shield/references/lgpd/anpd-enforcement.md +147 -147
  236. package/src/bmad-plus/packs/pack-shield/references/lgpd/compliance-program.md +272 -272
  237. package/src/bmad-plus/packs/pack-shield/references/lgpd/lgpd-articles.md +271 -271
  238. package/src/bmad-plus/packs/pack-shield/references/nis2/article-21-measures.md +153 -153
  239. package/src/bmad-plus/packs/pack-shield/references/nis2/iso27001-nis2-mapping.md +68 -68
  240. package/src/bmad-plus/packs/pack-shield/references/nist-800-53/assessment-rmf.md +349 -349
  241. package/src/bmad-plus/packs/pack-shield/references/nist-800-53/baselines-tailoring.md +277 -277
  242. package/src/bmad-plus/packs/pack-shield/references/nist-800-53/control-families.md +450 -450
  243. package/src/bmad-plus/packs/pack-shield/references/nist-ai-rmf/rmf-core.md +361 -361
  244. package/src/bmad-plus/packs/pack-shield/references/nist-ai-rmf/rmf-profiles.md +192 -192
  245. package/src/bmad-plus/packs/pack-shield/references/nist-csf/csf-10-to-20-mapping.md +143 -143
  246. package/src/bmad-plus/packs/pack-shield/references/nist-csf/csf-20-functions-categories.md +278 -278
  247. package/src/bmad-plus/packs/pack-shield/references/nist-csf/csf-implementation-tiers.md +135 -135
  248. package/src/bmad-plus/packs/pack-shield/references/pci-compliance/pci-dss-requirements.md +366 -366
  249. package/src/bmad-plus/packs/pack-shield/references/pci-compliance/pci-dss-saq-guide.md +217 -217
  250. package/src/bmad-plus/packs/pack-shield/references/pci-compliance/pci-dss-v4-changes.md +190 -190
  251. package/src/bmad-plus/packs/pack-shield/references/section-508/wcag-mapping.md +160 -160
  252. package/src/bmad-plus/packs/pack-shield/references/soc2/controls.md +241 -241
  253. package/src/bmad-plus/packs/pack-shield/references/soc2/evidence.md +236 -236
  254. package/src/bmad-plus/packs/pack-shield/references/soc2/policies.md +254 -254
  255. package/src/bmad-plus/packs/pack-shield/references/soc2/vendor.md +276 -276
  256. package/src/bmad-plus/packs/pack-shield/references/swift-csp/swift-assessment.md +202 -202
  257. package/src/bmad-plus/packs/pack-shield/references/swift-csp/swift-controls.md +545 -545
  258. package/src/bmad-plus/packs/pack-shield/references/tsa-compliance/tsa-crmp-requirements.md +359 -359
  259. package/src/bmad-plus/packs/pack-shield/references/tsa-compliance/tsa-directives-overview.md +187 -187
  260. package/src/bmad-plus/packs/pack-shield/references/tsa-compliance/tsa-incident-reporting.md +187 -187
  261. package/src/bmad-plus/packs/pack-shield/references/wcag/criteria-detail.md +510 -510
  262. package/src/bmad-plus/packs/pack-shield/shared/audit-report-template.md +103 -103
  263. package/src/bmad-plus/packs/pack-shield/shared/cross-framework-mapper.md +103 -103
  264. package/src/bmad-plus/packs/pack-shield/shared/gap-analysis-template.md +83 -83
  265. package/src/bmad-plus/packs/pack-shield/shield-orchestrator.md +229 -229
  266. package/src/bmad-plus/packs/pack-shield/upstream-sync.yaml +68 -68
  267. package/src/bmad-plus/skills/bmad-plus-autopilot/SKILL.md +99 -99
  268. package/src/bmad-plus/skills/bmad-plus-parallel/SKILL.md +93 -93
  269. package/src/bmad-plus/skills/bmad-plus-sync/SKILL.md +69 -69
  270. package/tools/bmad-plus-npx.js +3 -5
  271. package/tools/cli/commands/autoconfig.js +508 -489
  272. package/tools/cli/commands/doctor.js +219 -222
  273. package/tools/cli/commands/install.js +548 -739
  274. package/tools/cli/commands/memory.js +194 -194
  275. package/tools/cli/commands/scan.js +362 -350
  276. package/tools/cli/commands/uninstall.js +96 -96
  277. package/tools/cli/commands/update.js +116 -174
  278. package/tools/cli/i18n.js +845 -763
  279. package/tools/cli/lib/memory-init.js +114 -0
  280. package/tools/cli/lib/pack-copy.js +84 -0
  281. package/tools/cli/lib/packs.js +114 -0
@@ -1,350 +1,362 @@
1
- /**
2
- * BMAD+ Scan Command
3
- * Scan directories to discover projects, detect stacks, and index them in the global brain.
4
- * Interactive validation — user confirms each project before indexing.
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 crypto = require('node:crypto');
13
- const clack = require('@clack/prompts');
14
- const pc = require('picocolors');
15
-
16
- // Project detection markers (priority order)
17
- const PROJECT_MARKERS = [
18
- { file: 'package.json', stack: 'Node.js', detect: (dir) => {
19
- try {
20
- const pkg = JSON.parse(fs.readFileSync(path.join(dir, 'package.json'), 'utf8'));
21
- const deps = { ...pkg.dependencies, ...pkg.devDependencies };
22
- if (deps['next']) return 'Next.js';
23
- if (deps['nuxt']) return 'Nuxt';
24
- if (deps['react']) return 'React';
25
- if (deps['vue']) return 'Vue.js';
26
- if (deps['svelte']) return 'Svelte';
27
- if (deps['express']) return 'Express';
28
- if (deps['fastify']) return 'Fastify';
29
- if (deps['electron']) return 'Electron';
30
- if (deps['tauri']) return 'Tauri';
31
- return 'Node.js';
32
- } catch { return 'Node.js'; }
33
- }},
34
- { file: 'Cargo.toml', stack: 'Rust' },
35
- { file: 'pyproject.toml', stack: 'Python' },
36
- { file: 'requirements.txt', stack: 'Python' },
37
- { file: 'go.mod', stack: 'Go' },
38
- { file: 'composer.json', stack: 'PHP' },
39
- { file: 'Gemfile', stack: 'Ruby' },
40
- { file: 'pom.xml', stack: 'Java' },
41
- { file: 'build.gradle', stack: 'Java/Kotlin' },
42
- ];
43
-
44
- // Directories to skip during scanning
45
- const SKIP_DIRS = new Set([
46
- 'node_modules', '.git', 'vendor', '__pycache__', 'dist', 'build',
47
- '.next', '.nuxt', '.svelte-kit', 'target', '.venv', 'venv',
48
- '.cache', '.output', 'coverage', '.turbo', '.angular',
49
- '$RECYCLE.BIN', 'System Volume Information', 'Windows',
50
- 'Program Files', 'Program Files (x86)', 'ProgramData',
51
- 'AppData', 'Recovery', 'PerfLogs',
52
- ]);
53
-
54
- function getProjectStatus(dir, activeDays = 30, pausedDays = 180) {
55
- try {
56
- const stat = fs.statSync(dir);
57
- const daysSince = (Date.now() - stat.mtimeMs) / (1000 * 60 * 60 * 24);
58
- if (daysSince < activeDays) return 'active';
59
- if (daysSince < pausedDays) return 'paused';
60
- return 'archived';
61
- } catch { return 'unknown'; }
62
- }
63
-
64
- function getStatusIcon(status) {
65
- switch (status) {
66
- case 'active': return pc.green('●');
67
- case 'paused': return pc.yellow('◐');
68
- case 'archived': return pc.dim('○');
69
- default: return pc.dim('?');
70
- }
71
- }
72
-
73
- function getProjectName(dir) {
74
- try {
75
- const pkgPath = path.join(dir, 'package.json');
76
- if (fs.existsSync(pkgPath)) {
77
- const pkg = JSON.parse(fs.readFileSync(pkgPath, 'utf8'));
78
- if (pkg.name) return pkg.name;
79
- }
80
- } catch {}
81
- return path.basename(dir);
82
- }
83
-
84
- function hasBmadInstalled(dir) {
85
- return fs.existsSync(path.join(dir, '.agents')) ||
86
- fs.existsSync(path.join(dir, '_bmad'));
87
- }
88
-
89
- function scanDirectory(rootDir, maxDepth = 4, currentDepth = 0, activeDays = 30, pausedDays = 180) {
90
- const projects = [];
91
-
92
- if (currentDepth > maxDepth) return projects;
93
-
94
- let entries;
95
- try {
96
- entries = fs.readdirSync(rootDir, { withFileTypes: true });
97
- } catch {
98
- return projects; // Permission denied or inaccessible
99
- }
100
-
101
- // Check if current dir is a project
102
- for (const marker of PROJECT_MARKERS) {
103
- if (fs.existsSync(path.join(rootDir, marker.file))) {
104
- const stack = marker.detect ? marker.detect(rootDir) : marker.stack;
105
- projects.push({
106
- path: rootDir,
107
- name: getProjectName(rootDir),
108
- stack,
109
- status: getProjectStatus(rootDir, activeDays, pausedDays),
110
- bmad: hasBmadInstalled(rootDir),
111
- hasAgentsMd: fs.existsSync(path.join(rootDir, 'AGENTS.md')),
112
- hasGit: fs.existsSync(path.join(rootDir, '.git')),
113
- });
114
- return projects; // Don't recurse into project subdirs
115
- }
116
- }
117
-
118
- // Also detect by .git alone (any project with version control)
119
- if (fs.existsSync(path.join(rootDir, '.git')) && currentDepth > 0) {
120
- projects.push({
121
- path: rootDir,
122
- name: getProjectName(rootDir),
123
- stack: 'Unknown',
124
- status: getProjectStatus(rootDir, activeDays, pausedDays),
125
- bmad: hasBmadInstalled(rootDir),
126
- hasAgentsMd: fs.existsSync(path.join(rootDir, 'AGENTS.md')),
127
- hasGit: true,
128
- });
129
- return projects;
130
- }
131
-
132
- // Recurse into subdirectories
133
- for (const entry of entries) {
134
- if (!entry.isDirectory()) continue;
135
- if (SKIP_DIRS.has(entry.name)) continue;
136
- if (entry.name.startsWith('.') && entry.name !== '.git') continue;
137
-
138
- const subPath = path.join(rootDir, entry.name);
139
- const subProjects = scanDirectory(subPath, maxDepth, currentDepth + 1, activeDays, pausedDays);
140
- projects.push(...subProjects);
141
- }
142
-
143
- return projects;
144
- }
145
-
146
- module.exports = {
147
- command: 'scan',
148
- description: 'Scan directories to discover and index projects in the global brain',
149
- options: [
150
- ['-d, --directory <path>', 'Directory to scan (default: current directory)'],
151
- ['--depth <n>', 'Max depth to scan (default: 4)', '4'],
152
- ['--active-days <n>', 'Days since last modified to consider a project "active" (default: 30)', '30'],
153
- ['--paused-days <n>', 'Days since last modified to consider a project "paused" (default: 180)', '180'],
154
- ['-y, --yes', 'Index all projects without prompting'],
155
- ],
156
- action: async (options) => {
157
- const scanDir = path.resolve(options.directory || process.cwd());
158
- const maxDepth = parseInt(options.depth) || 4;
159
- const activeDays = parseInt(options.activeDays) || 30;
160
- const pausedDays = parseInt(options.pausedDays) || 180;
161
-
162
- clack.intro(pc.bgMagenta(pc.white(' 🧠 BMAD+ Project Scanner ')));
163
-
164
- // Verify directory exists
165
- if (!fs.existsSync(scanDir)) {
166
- clack.log.error(`Directory not found: ${scanDir}`);
167
- clack.outro(pc.red('Scan failed.'));
168
- return;
169
- }
170
-
171
- // Scan
172
- const spinner = clack.spinner();
173
- spinner.start(`Scanning ${scanDir} (depth: ${maxDepth})...`);
174
-
175
- const projects = scanDirectory(scanDir, maxDepth, 0, activeDays, pausedDays);
176
-
177
- if (projects.length === 0) {
178
- spinner.stop('No projects found.');
179
- clack.outro('Try scanning a different directory or increasing --depth');
180
- return;
181
- }
182
-
183
- spinner.stop(`Found ${pc.bold(projects.length)} project(s)`);
184
-
185
- // Display legend
186
- const activeCount = projects.filter(p => p.status === 'active').length;
187
- const pausedCount = projects.filter(p => p.status === 'paused').length;
188
- const archivedCount = projects.filter(p => p.status === 'archived').length;
189
-
190
- clack.log.info('');
191
- clack.log.info(pc.dim(' Legend:'));
192
- clack.log.info(` ${pc.green('●')} active modified < ${activeDays} days ago ${pc.dim(`(${activeCount} found)`)}`);
193
- clack.log.info(` ${pc.yellow('◐')} paused modified ${activeDays}–${pausedDays} days ago ${pc.dim(`(${pausedCount} found)`)}`);
194
- clack.log.info(` ${pc.dim('○')} archived modified > ${pausedDays} days ago ${pc.dim(`(${archivedCount} found)`)}`);
195
- clack.log.info('');
196
-
197
- // Display table
198
- clack.log.info(pc.bold(' # Status BMAD+ Stack Name Path'));
199
- clack.log.info(pc.dim(' ' + '─'.repeat(90)));
200
-
201
- projects.forEach((p, i) => {
202
- const num = String(i + 1).padStart(3);
203
- const status = getStatusIcon(p.status) + ' ' + p.status.padEnd(8);
204
- const bmad = p.bmad ? pc.green('✓') : pc.dim('·');
205
- const stack = p.stack.padEnd(16);
206
- const name = p.name.substring(0, 20).padEnd(20);
207
- const projPath = p.path.length > 40 ? '...' + p.path.slice(-37) : p.path;
208
- clack.log.info(` ${num} ${status} ${bmad} ${stack} ${name} ${pc.dim(projPath)}`);
209
- });
210
-
211
- clack.log.info('');
212
-
213
- // Interactive validation
214
- const globalBrainDir = path.join(os.homedir(), '.bmad-plus', 'brain', 'projects');
215
-
216
- if (options.yes) {
217
- // Auto-index all
218
- const fsExtra = require('fs-extra');
219
- fsExtra.ensureDirSync(globalBrainDir);
220
-
221
- let indexed = 0;
222
- for (const proj of projects) {
223
- const hash = crypto.createHash('sha256').update(proj.path).digest('hex').slice(0, 8);
224
- const meta = {
225
- path: proj.path,
226
- name: proj.name,
227
- hash,
228
- stack: proj.stack,
229
- status: proj.status,
230
- bmad_installed: proj.bmad,
231
- has_git: proj.hasGit,
232
- last_scanned: new Date().toISOString().slice(0, 10),
233
- };
234
- fs.writeFileSync(
235
- path.join(globalBrainDir, `${hash}.yaml`),
236
- Object.entries(meta).map(([k, v]) => `${k}: ${JSON.stringify(v)}`).join('\n'),
237
- 'utf8'
238
- );
239
- indexed++;
240
- }
241
- clack.log.success(`✅ ${indexed} project(s) indexed in ${globalBrainDir}`);
242
- } else {
243
- // Interactive mode
244
- const action = await clack.select({
245
- message: `${projects.length} project(s) found. What to do?`,
246
- options: [
247
- { value: 'all', label: `✅ Index all ${projects.length} projects` },
248
- { value: 'select', label: '✏️ Select which to index' },
249
- { value: 'none', label: '⏭️ Skip — don\'t index anything' },
250
- ],
251
- });
252
-
253
- if (clack.isCancel(action) || action === 'none') {
254
- clack.cancel('Scan cancelled.');
255
- return;
256
- }
257
-
258
- const fsExtra = require('fs-extra');
259
- fsExtra.ensureDirSync(globalBrainDir);
260
-
261
- let toIndex = projects;
262
-
263
- if (action === 'select') {
264
- const selected = await clack.multiselect({
265
- message: 'Select projects to index:',
266
- options: projects.map((p, i) => ({
267
- value: i,
268
- label: `${p.name} (${p.stack})`,
269
- hint: `${p.status} ${p.path}`,
270
- })),
271
- required: false,
272
- });
273
-
274
- if (clack.isCancel(selected)) {
275
- clack.cancel('Scan cancelled.');
276
- return;
277
- }
278
-
279
- toIndex = selected.map(i => projects[i]);
280
- }
281
-
282
- let indexed = 0;
283
- for (const proj of toIndex) {
284
- const hash = crypto.createHash('sha256').update(proj.path).digest('hex').slice(0, 8);
285
- const meta = {
286
- path: proj.path,
287
- name: proj.name,
288
- hash,
289
- stack: proj.stack,
290
- status: proj.status,
291
- bmad_installed: proj.bmad,
292
- has_git: proj.hasGit,
293
- last_scanned: new Date().toISOString().slice(0, 10),
294
- };
295
- fs.writeFileSync(
296
- path.join(globalBrainDir, `${hash}.yaml`),
297
- Object.entries(meta).map(([k, v]) => `${k}: ${JSON.stringify(v)}`).join('\n'),
298
- 'utf8'
299
- );
300
- indexed++;
301
- }
302
-
303
- clack.log.success(`✅ ${indexed} project(s) indexed in global brain`);
304
- }
305
-
306
- // Generate human-readable index
307
- const indexPath = path.join(os.homedir(), '.bmad-plus', 'brain', 'projects-index.md');
308
- const existingProjects = [];
309
- if (fs.existsSync(globalBrainDir)) {
310
- for (const f of fs.readdirSync(globalBrainDir)) {
311
- if (!f.endsWith('.yaml')) continue;
312
- try {
313
- const content = fs.readFileSync(path.join(globalBrainDir, f), 'utf8');
314
- const meta = {};
315
- for (const line of content.split('\n')) {
316
- const m = line.match(/^(\w+):\s*(.+)$/);
317
- if (m) {
318
- try { meta[m[1]] = JSON.parse(m[2]); } catch { meta[m[1]] = m[2]; }
319
- }
320
- }
321
- existingProjects.push(meta);
322
- } catch {}
323
- }
324
- }
325
-
326
- const indexContent = [
327
- '---',
328
- 'title: Project Index',
329
- `last_updated: "${new Date().toISOString().slice(0, 10)}"`,
330
- `total_projects: ${existingProjects.length}`,
331
- '---',
332
- '',
333
- '# Project Index',
334
- '',
335
- `> Auto-generated by \`npx bmad-plus scan\` — ${new Date().toISOString().slice(0, 10)}`,
336
- '',
337
- '| Status | Name | Stack | BMAD+ | Path |',
338
- '|--------|------|-------|-------|------|',
339
- ...existingProjects.map(p =>
340
- `| ${p.status || '?'} | ${p.name || '?'} | ${p.stack || '?'} | ${p.bmad_installed ? '✓' : '·'} | \`${p.path || '?'}\` |`
341
- ),
342
- '',
343
- ];
344
-
345
- fs.writeFileSync(indexPath, indexContent.join('\n'), 'utf8');
346
- clack.log.info(`📋 Project index updated: ${indexPath}`);
347
-
348
- clack.outro(pc.green('Scan complete! 🧠'));
349
- },
350
- };
1
+ /**
2
+ * BMAD+ Scan Command
3
+ * Scan directories to discover projects, detect stacks, and index them in the global brain.
4
+ * Interactive validation — user confirms each project before indexing.
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 crypto = require('node:crypto');
13
+ const clack = require('@clack/prompts');
14
+ const pc = require('picocolors');
15
+
16
+ // Project detection markers (priority order)
17
+ const PROJECT_MARKERS = [
18
+ { file: 'package.json', stack: 'Node.js', detect: (dir) => {
19
+ try {
20
+ const pkg = JSON.parse(fs.readFileSync(path.join(dir, 'package.json'), 'utf8'));
21
+ const deps = { ...pkg.dependencies, ...pkg.devDependencies };
22
+ if (deps['next']) return 'Next.js';
23
+ if (deps['nuxt']) return 'Nuxt';
24
+ if (deps['react']) return 'React';
25
+ if (deps['vue']) return 'Vue.js';
26
+ if (deps['svelte']) return 'Svelte';
27
+ if (deps['express']) return 'Express';
28
+ if (deps['fastify']) return 'Fastify';
29
+ if (deps['electron']) return 'Electron';
30
+ if (deps['tauri']) return 'Tauri';
31
+ return 'Node.js';
32
+ } catch { return 'Node.js'; }
33
+ }},
34
+ { file: 'Cargo.toml', stack: 'Rust' },
35
+ { file: 'pyproject.toml', stack: 'Python' },
36
+ { file: 'requirements.txt', stack: 'Python' },
37
+ { file: 'go.mod', stack: 'Go' },
38
+ { file: 'composer.json', stack: 'PHP' },
39
+ { file: 'Gemfile', stack: 'Ruby' },
40
+ { file: 'pom.xml', stack: 'Java' },
41
+ { file: 'build.gradle', stack: 'Java/Kotlin' },
42
+ ];
43
+
44
+ // Directories to skip during scanning
45
+ const SKIP_DIRS = new Set([
46
+ 'node_modules', '.git', 'vendor', '__pycache__', 'dist', 'build',
47
+ '.next', '.nuxt', '.svelte-kit', 'target', '.venv', 'venv',
48
+ '.cache', '.output', 'coverage', '.turbo', '.angular',
49
+ '$RECYCLE.BIN', 'System Volume Information', 'Windows',
50
+ 'Program Files', 'Program Files (x86)', 'ProgramData',
51
+ 'AppData', 'Recovery', 'PerfLogs',
52
+ ]);
53
+
54
+ function getProjectStatus(dir, activeDays = 30, pausedDays = 180) {
55
+ try {
56
+ const stat = fs.statSync(dir);
57
+ const daysSince = (Date.now() - stat.mtimeMs) / (1000 * 60 * 60 * 24);
58
+ if (daysSince < activeDays) return 'active';
59
+ if (daysSince < pausedDays) return 'paused';
60
+ return 'archived';
61
+ } catch { return 'unknown'; }
62
+ }
63
+
64
+ function getStatusIcon(status) {
65
+ switch (status) {
66
+ case 'active': return pc.green('●');
67
+ case 'paused': return pc.yellow('◐');
68
+ case 'archived': return pc.dim('○');
69
+ default: return pc.dim('?');
70
+ }
71
+ }
72
+
73
+ function getProjectName(dir) {
74
+ try {
75
+ const pkgPath = path.join(dir, 'package.json');
76
+ if (fs.existsSync(pkgPath)) {
77
+ const pkg = JSON.parse(fs.readFileSync(pkgPath, 'utf8'));
78
+ if (pkg.name) return pkg.name;
79
+ }
80
+ } catch {}
81
+ return path.basename(dir);
82
+ }
83
+
84
+ function hasBmadInstalled(dir) {
85
+ return fs.existsSync(path.join(dir, '.agents')) ||
86
+ fs.existsSync(path.join(dir, '_bmad'));
87
+ }
88
+
89
+ function scanDirectory(rootDir, maxDepth = 4, currentDepth = 0, activeDays = 30, pausedDays = 180) {
90
+ const projects = [];
91
+
92
+ if (currentDepth > maxDepth) return projects;
93
+
94
+ let entries;
95
+ try {
96
+ entries = fs.readdirSync(rootDir, { withFileTypes: true });
97
+ } catch {
98
+ return projects; // Permission denied or inaccessible
99
+ }
100
+
101
+ // Check if current dir is a project
102
+ for (const marker of PROJECT_MARKERS) {
103
+ if (fs.existsSync(path.join(rootDir, marker.file))) {
104
+ const stack = marker.detect ? marker.detect(rootDir) : marker.stack;
105
+ projects.push({
106
+ path: rootDir,
107
+ name: getProjectName(rootDir),
108
+ stack,
109
+ status: getProjectStatus(rootDir, activeDays, pausedDays),
110
+ bmad: hasBmadInstalled(rootDir),
111
+ hasAgentsMd: fs.existsSync(path.join(rootDir, 'AGENTS.md')),
112
+ hasGit: fs.existsSync(path.join(rootDir, '.git')),
113
+ });
114
+ return projects; // Don't recurse into project subdirs
115
+ }
116
+ }
117
+
118
+ // Also detect by .git alone (any project with version control)
119
+ if (fs.existsSync(path.join(rootDir, '.git')) && currentDepth > 0) {
120
+ projects.push({
121
+ path: rootDir,
122
+ name: getProjectName(rootDir),
123
+ stack: 'Unknown',
124
+ status: getProjectStatus(rootDir, activeDays, pausedDays),
125
+ bmad: hasBmadInstalled(rootDir),
126
+ hasAgentsMd: fs.existsSync(path.join(rootDir, 'AGENTS.md')),
127
+ hasGit: true,
128
+ });
129
+ return projects;
130
+ }
131
+
132
+ // Recurse into subdirectories
133
+ for (const entry of entries) {
134
+ if (!entry.isDirectory()) continue;
135
+ if (SKIP_DIRS.has(entry.name)) continue;
136
+ if (entry.name.startsWith('.') && entry.name !== '.git') continue;
137
+
138
+ const subPath = path.join(rootDir, entry.name);
139
+ const subProjects = scanDirectory(subPath, maxDepth, currentDepth + 1, activeDays, pausedDays);
140
+ projects.push(...subProjects);
141
+ }
142
+
143
+ return projects;
144
+ }
145
+
146
+ /**
147
+ * Index a single project in the global brain by writing its metadata YAML file.
148
+ * @param {object} project - Project metadata object
149
+ * @param {string} globalBrainDir - Path to the brain projects directory
150
+ * @returns {void}
151
+ */
152
+ function indexProject(project, globalBrainDir) {
153
+ const hash = crypto.createHash('sha256').update(project.path).digest('hex').slice(0, 8);
154
+ const meta = {
155
+ path: project.path,
156
+ name: project.name,
157
+ hash,
158
+ stack: project.stack,
159
+ status: project.status,
160
+ bmad_installed: project.bmad,
161
+ has_git: project.hasGit,
162
+ last_scanned: new Date().toISOString().slice(0, 10),
163
+ };
164
+ fs.writeFileSync(
165
+ path.join(globalBrainDir, `${hash}.yaml`),
166
+ Object.entries(meta).map(([k, v]) => `${k}: ${JSON.stringify(v)}`).join('\n'),
167
+ 'utf8'
168
+ );
169
+ }
170
+
171
+ module.exports = {
172
+ command: 'scan',
173
+ description: 'Scan directories to discover and index projects in the global brain',
174
+ options: [
175
+ ['-d, --directory <path>', 'Directory to scan (default: current directory)'],
176
+ ['--depth <n>', 'Max depth to scan (default: 4)', '4'],
177
+ ['--active-days <n>', 'Days since last modified to consider a project "active" (default: 30)', '30'],
178
+ ['--paused-days <n>', 'Days since last modified to consider a project "paused" (default: 180)', '180'],
179
+ ['-y, --yes', 'Index all projects without prompting'],
180
+ ],
181
+ action: async (options) => {
182
+ const scanDir = path.resolve(options.directory || process.cwd());
183
+
184
+ const rawDepth = parseInt(options.depth, 10);
185
+ const maxDepth = (!isNaN(rawDepth) && rawDepth > 0) ? rawDepth : (() => { clack.log.warn(`Invalid --depth "${options.depth}", defaulting to 4`); return 4; })();
186
+
187
+ const rawActive = parseInt(options.activeDays, 10);
188
+ const activeDays = (!isNaN(rawActive) && rawActive > 0) ? rawActive : (() => { clack.log.warn(`Invalid --active-days "${options.activeDays}", defaulting to 30`); return 30; })();
189
+
190
+ const rawPaused = parseInt(options.pausedDays, 10);
191
+ const pausedDays = (!isNaN(rawPaused) && rawPaused > 0) ? rawPaused : (() => { clack.log.warn(`Invalid --paused-days "${options.pausedDays}", defaulting to 180`); return 180; })();
192
+
193
+ clack.intro(pc.bgMagenta(pc.white(' 🧠 BMAD+ Project Scanner ')));
194
+
195
+ // Verify directory exists
196
+ if (!fs.existsSync(scanDir)) {
197
+ clack.log.error(`Directory not found: ${scanDir}`);
198
+ clack.outro(pc.red('Scan failed.'));
199
+ return;
200
+ }
201
+
202
+ // Scan
203
+ const spinner = clack.spinner();
204
+ spinner.start(`Scanning ${scanDir} (depth: ${maxDepth})...`);
205
+
206
+ const projects = scanDirectory(scanDir, maxDepth, 0, activeDays, pausedDays);
207
+
208
+ if (projects.length === 0) {
209
+ spinner.stop('No projects found.');
210
+ clack.outro('Try scanning a different directory or increasing --depth');
211
+ return;
212
+ }
213
+
214
+ spinner.stop(`Found ${pc.bold(projects.length)} project(s)`);
215
+
216
+ // Display legend
217
+ const activeCount = projects.filter(p => p.status === 'active').length;
218
+ const pausedCount = projects.filter(p => p.status === 'paused').length;
219
+ const archivedCount = projects.filter(p => p.status === 'archived').length;
220
+
221
+ clack.log.info('');
222
+ clack.log.info(pc.dim(' Legend:'));
223
+ clack.log.info(` ${pc.green('')} active modified < ${activeDays} days ago ${pc.dim(`(${activeCount} found)`)}`);
224
+ clack.log.info(` ${pc.yellow('◐')} paused modified ${activeDays}–${pausedDays} days ago ${pc.dim(`(${pausedCount} found)`)}`);
225
+ clack.log.info(` ${pc.dim('○')} archived modified > ${pausedDays} days ago ${pc.dim(`(${archivedCount} found)`)}`);
226
+ clack.log.info('');
227
+
228
+ // Display table
229
+ clack.log.info(pc.bold(' # Status BMAD+ Stack Name Path'));
230
+ clack.log.info(pc.dim(' ' + '─'.repeat(90)));
231
+
232
+ projects.forEach((p, i) => {
233
+ const num = String(i + 1).padStart(3);
234
+ const status = getStatusIcon(p.status) + ' ' + p.status.padEnd(8);
235
+ const bmad = p.bmad ? pc.green('✓') : pc.dim('·');
236
+ const stack = p.stack.padEnd(16);
237
+ const name = p.name.substring(0, 20).padEnd(20);
238
+ const projPath = p.path.length > 40 ? '...' + p.path.slice(-37) : p.path;
239
+ clack.log.info(` ${num} ${status} ${bmad} ${stack} ${name} ${pc.dim(projPath)}`);
240
+ });
241
+
242
+ clack.log.info('');
243
+
244
+ // Interactive validation
245
+ const globalBrainDir = path.join(os.homedir(), '.bmad-plus', 'brain', 'projects');
246
+
247
+ if (options.yes) {
248
+ // Auto-index all
249
+ const fsExtra = require('fs-extra');
250
+ fsExtra.ensureDirSync(globalBrainDir);
251
+
252
+ let indexed = 0;
253
+ for (const proj of projects) {
254
+ indexProject(proj, globalBrainDir);
255
+ indexed++;
256
+ }
257
+ clack.log.success(`✅ ${indexed} project(s) indexed in ${globalBrainDir}`);
258
+ } else {
259
+ // Interactive mode
260
+ const action = await clack.select({
261
+ message: `${projects.length} project(s) found. What to do?`,
262
+ options: [
263
+ { value: 'all', label: `✅ Index all ${projects.length} projects` },
264
+ { value: 'select', label: '✏️ Select which to index' },
265
+ { value: 'none', label: '⏭️ Skip — don\'t index anything' },
266
+ ],
267
+ });
268
+
269
+ if (clack.isCancel(action) || action === 'none') {
270
+ clack.cancel('Scan cancelled.');
271
+ return;
272
+ }
273
+
274
+ const fsExtra = require('fs-extra');
275
+ fsExtra.ensureDirSync(globalBrainDir);
276
+
277
+ let toIndex = projects;
278
+
279
+ if (action === 'select') {
280
+ const selected = await clack.multiselect({
281
+ message: 'Select projects to index:',
282
+ options: projects.map((p, i) => ({
283
+ value: i,
284
+ label: `${p.name} (${p.stack})`,
285
+ hint: `${p.status} ${p.path}`,
286
+ })),
287
+ required: false,
288
+ });
289
+
290
+ if (clack.isCancel(selected)) {
291
+ clack.cancel('Scan cancelled.');
292
+ return;
293
+ }
294
+
295
+ toIndex = selected.map(i => projects[i]);
296
+ }
297
+
298
+ let indexed = 0;
299
+ for (const proj of toIndex) {
300
+ indexProject(proj, globalBrainDir);
301
+ indexed++;
302
+ }
303
+
304
+ clack.log.success(`✅ ${indexed} project(s) indexed in global brain`);
305
+ }
306
+
307
+ // Generate human-readable index
308
+ const indexPath = path.join(os.homedir(), '.bmad-plus', 'brain', 'projects-index.md');
309
+ const existingProjects = [];
310
+ if (fs.existsSync(globalBrainDir)) {
311
+ for (const f of fs.readdirSync(globalBrainDir)) {
312
+ if (!f.endsWith('.yaml')) continue;
313
+ try {
314
+ const content = fs.readFileSync(path.join(globalBrainDir, f), 'utf8');
315
+ const meta = {};
316
+ for (const line of content.split('\n')) {
317
+ const m = line.match(/^(\w+):\s*(.+)$/);
318
+ if (m) {
319
+ try { meta[m[1]] = JSON.parse(m[2]); } catch { meta[m[1]] = m[2]; }
320
+ }
321
+ }
322
+ existingProjects.push(meta);
323
+ } catch {}
324
+ }
325
+ }
326
+
327
+ const indexContent = [
328
+ '---',
329
+ 'title: Project Index',
330
+ `last_updated: "${new Date().toISOString().slice(0, 10)}"`,
331
+ `total_projects: ${existingProjects.length}`,
332
+ '---',
333
+ '',
334
+ '# Project Index',
335
+ '',
336
+ `> Auto-generated by \`npx bmad-plus scan\` — ${new Date().toISOString().slice(0, 10)}`,
337
+ '',
338
+ '| Status | Name | Stack | BMAD+ | Path |',
339
+ '|--------|------|-------|-------|------|',
340
+ ...existingProjects.map(p =>
341
+ `| ${p.status || '?'} | ${p.name || '?'} | ${p.stack || '?'} | ${p.bmad_installed ? '✓' : '·'} | \`${p.path || '?'}\` |`
342
+ ),
343
+ '',
344
+ ];
345
+
346
+ fs.writeFileSync(indexPath, indexContent.join('\n'), 'utf8');
347
+ clack.log.info(`📋 Project index updated: ${indexPath}`);
348
+
349
+ clack.outro(pc.green('Scan complete! 🧠'));
350
+ },
351
+ // Exported for functional testing (not part of the public API)
352
+ _internal: {
353
+ PROJECT_MARKERS,
354
+ SKIP_DIRS,
355
+ getProjectStatus,
356
+ getStatusIcon,
357
+ getProjectName,
358
+ hasBmadInstalled,
359
+ scanDirectory,
360
+ indexProject,
361
+ },
362
+ };