mindforge-cc 9.0.0 → 10.0.1

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 (717) hide show
  1. package/.mindforge/config.json +1 -1
  2. package/.mindforge/personas/a11y-architect.md +190 -0
  3. package/.mindforge/personas/accessibility-tester.md +108 -0
  4. package/.mindforge/personas/api-designer.md +190 -0
  5. package/.mindforge/personas/api-gateway-architect.md +168 -0
  6. package/.mindforge/personas/api-load-tester.md +144 -0
  7. package/.mindforge/personas/authentication-architect.md +163 -0
  8. package/.mindforge/personas/backup-recovery-specialist.md +181 -0
  9. package/.mindforge/personas/browser-extension-architect.md +96 -0
  10. package/.mindforge/personas/build-optimizer.md +160 -0
  11. package/.mindforge/personas/caching-strategist.md +180 -0
  12. package/.mindforge/personas/chaos-engineer.md +207 -0
  13. package/.mindforge/personas/cli-designer.md +151 -0
  14. package/.mindforge/personas/cloud-architect.md +229 -0
  15. package/.mindforge/personas/code-archeologist.md +176 -0
  16. package/.mindforge/personas/code-explorer.md +144 -0
  17. package/.mindforge/personas/compliance-auditor.md +190 -0
  18. package/.mindforge/personas/concurrency-expert.md +310 -0
  19. package/.mindforge/personas/config-management-expert.md +277 -0
  20. package/.mindforge/personas/contract-tester.md +224 -0
  21. package/.mindforge/personas/cost-analyst.md +209 -0
  22. package/.mindforge/personas/data-engineer.md +235 -0
  23. package/.mindforge/personas/data-privacy-engineer.md +187 -0
  24. package/.mindforge/personas/database-expert.md +223 -0
  25. package/.mindforge/personas/dependency-auditor.md +181 -0
  26. package/.mindforge/personas/design-system-engineer.md +115 -0
  27. package/.mindforge/personas/devops-engineer.md +561 -0
  28. package/.mindforge/personas/domain-modeler.md +127 -0
  29. package/.mindforge/personas/email-systems-engineer.md +119 -0
  30. package/.mindforge/personas/error-handling-architect.md +246 -0
  31. package/.mindforge/personas/event-driven-architect.md +134 -0
  32. package/.mindforge/personas/frontend-architect.md +107 -0
  33. package/.mindforge/personas/git-forensics.md +146 -0
  34. package/.mindforge/personas/git-workflow-expert.md +161 -0
  35. package/.mindforge/personas/go-specialist.md +249 -0
  36. package/.mindforge/personas/graphql-specialist.md +195 -0
  37. package/.mindforge/personas/incident-commander.md +214 -0
  38. package/.mindforge/personas/internationalization-expert.md +164 -0
  39. package/.mindforge/personas/java-specialist.md +271 -0
  40. package/.mindforge/personas/kubernetes-debugger.md +175 -0
  41. package/.mindforge/personas/logging-architect.md +200 -0
  42. package/.mindforge/personas/migration-specialist.md +237 -0
  43. package/.mindforge/personas/ml-engineer.md +312 -0
  44. package/.mindforge/personas/mobile-engineer.md +183 -0
  45. package/.mindforge/personas/monorepo-architect.md +323 -0
  46. package/.mindforge/personas/observability-engineer.md +217 -0
  47. package/.mindforge/personas/onboarding-guide.md +265 -0
  48. package/.mindforge/personas/performance-optimizer.md +293 -0
  49. package/.mindforge/personas/product-manager.md +105 -0
  50. package/.mindforge/personas/prompt-engineer.md +200 -0
  51. package/.mindforge/personas/python-specialist.md +277 -0
  52. package/.mindforge/personas/queue-architect.md +136 -0
  53. package/.mindforge/personas/react-specialist.md +97 -0
  54. package/.mindforge/personas/real-time-engineer.md +121 -0
  55. package/.mindforge/personas/refactoring-expert.md +117 -0
  56. package/.mindforge/personas/regex-craftsman.md +130 -0
  57. package/.mindforge/personas/rust-specialist.md +262 -0
  58. package/.mindforge/personas/sdk-designer.md +185 -0
  59. package/.mindforge/personas/search-engineer.md +290 -0
  60. package/.mindforge/personas/senior-reviewer.md +372 -0
  61. package/.mindforge/personas/seo-specialist.md +99 -0
  62. package/.mindforge/personas/spec-reviewer.md +172 -0
  63. package/.mindforge/personas/state-machine-designer.md +172 -0
  64. package/.mindforge/personas/swarm-templates.json +72 -18
  65. package/.mindforge/personas/tailwind-specialist.md +95 -0
  66. package/.mindforge/personas/tech-debt-analyst.md +200 -0
  67. package/.mindforge/personas/tech-stack-selector.md +118 -0
  68. package/.mindforge/personas/technical-interviewer.md +158 -0
  69. package/.mindforge/personas/test-data-engineer.md +169 -0
  70. package/.mindforge/personas/typescript-wizard.md +247 -0
  71. package/.mindforge/personas/ux-auditor.md +251 -0
  72. package/.mindforge/personas/webhook-designer.md +161 -0
  73. package/CHANGELOG.md +82 -0
  74. package/LICENSE +1 -1
  75. package/MINDFORGE.md +18 -5
  76. package/README.md +7 -1
  77. package/RELEASENOTES.md +121 -193
  78. package/SECURITY.md +108 -2
  79. package/bin/autonomous/audit-writer.js +90 -0
  80. package/bin/autonomous/auto-runner.js +209 -431
  81. package/bin/autonomous/state-manager.js +116 -0
  82. package/bin/autonomous/task-dispatcher.js +114 -0
  83. package/bin/autonomous/wave-executor.js +169 -0
  84. package/bin/browser/browser-daemon.js +31 -1
  85. package/bin/change-classifier.js +3 -3
  86. package/bin/dashboard/api-router.js +21 -22
  87. package/bin/dashboard/metrics-aggregator.js +44 -3
  88. package/bin/dashboard/server.js +54 -10
  89. package/bin/dashboard/sse-bridge.js +51 -5
  90. package/bin/engine/learning-manager.js +1 -1
  91. package/bin/engine/mesh-syncer.js +26 -22
  92. package/bin/engine/nexus-tracer.js +6 -5
  93. package/bin/engine/orbital-guardian.js +13 -10
  94. package/bin/engine/skill-evolver.js +12 -14
  95. package/bin/engine/temporal-hub.js +25 -1
  96. package/bin/governance/policy-engine.js +5 -5
  97. package/bin/governance/policy-gate-hardened.js +2 -2
  98. package/bin/governance/quantum-crypto.js +10 -2
  99. package/bin/memory/identity-synthesizer.js +9 -11
  100. package/bin/memory/knowledge-store.js +62 -6
  101. package/bin/memory/semantic-hub.js +26 -30
  102. package/bin/memory/vector-hub.js +377 -193
  103. package/bin/migrations/v8-sqlite-migration.js +22 -21
  104. package/bin/migrations/v9-unified-memory.js +1 -1
  105. package/bin/mindforge-cli.js +47 -11
  106. package/bin/models/model-broker.js +2 -0
  107. package/bin/revops/remediation-queue.js +16 -55
  108. package/bin/utils/errors.js +40 -0
  109. package/bin/utils/file-io.js +102 -0
  110. package/bin/utils/index.js +6 -0
  111. package/bin/utils/paths.js +33 -0
  112. package/docs/commands-reference.md +38 -2
  113. package/docs/getting-started.md +16 -6
  114. package/docs/sdk-reference.md +374 -24
  115. package/docs/troubleshooting.md +4 -4
  116. package/docs/user-guide.md +31 -11
  117. package/examples/sdk-integration/README.md +24 -0
  118. package/examples/sdk-integration/index.js +24 -0
  119. package/examples/starter-project/MINDFORGE.md +8 -8
  120. package/package.json +30 -15
  121. package/.agent/bin/lib/commands.cjs +0 -959
  122. package/.agent/bin/lib/config.cjs +0 -421
  123. package/.agent/bin/lib/core.cjs +0 -1166
  124. package/.agent/bin/lib/frontmatter.cjs +0 -307
  125. package/.agent/bin/lib/init.cjs +0 -1336
  126. package/.agent/bin/lib/milestone.cjs +0 -252
  127. package/.agent/bin/lib/model-profiles.cjs +0 -68
  128. package/.agent/bin/lib/phase.cjs +0 -888
  129. package/.agent/bin/lib/profile-output.cjs +0 -952
  130. package/.agent/bin/lib/profile-pipeline.cjs +0 -539
  131. package/.agent/bin/lib/roadmap.cjs +0 -329
  132. package/.agent/bin/lib/security.cjs +0 -356
  133. package/.agent/bin/lib/state.cjs +0 -969
  134. package/.agent/bin/lib/template.cjs +0 -222
  135. package/.agent/bin/lib/uat.cjs +0 -189
  136. package/.agent/bin/lib/verify.cjs +0 -851
  137. package/.agent/bin/lib/workstream.cjs +0 -491
  138. package/.agent/bin/mindforge-tools.cjs +0 -897
  139. package/.agent/file-manifest.json +0 -219
  140. package/.agent/forge/help.md +0 -11
  141. package/.agent/forge/init-project.md +0 -36
  142. package/.agent/forge/plan-phase.md +0 -34
  143. package/.agent/mindforge/add-backlog.md +0 -32
  144. package/.agent/mindforge/agent.md +0 -31
  145. package/.agent/mindforge/approve.md +0 -22
  146. package/.agent/mindforge/audit.md +0 -34
  147. package/.agent/mindforge/auto.md +0 -26
  148. package/.agent/mindforge/benchmark.md +0 -37
  149. package/.agent/mindforge/browse.md +0 -30
  150. package/.agent/mindforge/complete-milestone.md +0 -22
  151. package/.agent/mindforge/costs.md +0 -15
  152. package/.agent/mindforge/cross-review.md +0 -21
  153. package/.agent/mindforge/dashboard.md +0 -102
  154. package/.agent/mindforge/debug.md +0 -133
  155. package/.agent/mindforge/discuss-phase.md +0 -142
  156. package/.agent/mindforge/do.md +0 -31
  157. package/.agent/mindforge/execute-phase.md +0 -200
  158. package/.agent/mindforge/health.md +0 -31
  159. package/.agent/mindforge/help.md +0 -33
  160. package/.agent/mindforge/init-org.md +0 -135
  161. package/.agent/mindforge/init-project.md +0 -170
  162. package/.agent/mindforge/install-skill.md +0 -28
  163. package/.agent/mindforge/learn.md +0 -147
  164. package/.agent/mindforge/learning.md +0 -20
  165. package/.agent/mindforge/map-codebase.md +0 -302
  166. package/.agent/mindforge/marketplace.md +0 -124
  167. package/.agent/mindforge/metrics.md +0 -26
  168. package/.agent/mindforge/migrate.md +0 -44
  169. package/.agent/mindforge/milestone.md +0 -16
  170. package/.agent/mindforge/new-runtime.md +0 -23
  171. package/.agent/mindforge/next.md +0 -109
  172. package/.agent/mindforge/note.md +0 -35
  173. package/.agent/mindforge/plan-phase.md +0 -131
  174. package/.agent/mindforge/plant-seed.md +0 -31
  175. package/.agent/mindforge/plugins.md +0 -44
  176. package/.agent/mindforge/pr-review.md +0 -45
  177. package/.agent/mindforge/profile-team.md +0 -27
  178. package/.agent/mindforge/publish-skill.md +0 -23
  179. package/.agent/mindforge/qa.md +0 -20
  180. package/.agent/mindforge/quick.md +0 -139
  181. package/.agent/mindforge/record-learning.md +0 -22
  182. package/.agent/mindforge/release.md +0 -14
  183. package/.agent/mindforge/remember.md +0 -30
  184. package/.agent/mindforge/research.md +0 -16
  185. package/.agent/mindforge/retrospective.md +0 -31
  186. package/.agent/mindforge/review-backlog.md +0 -34
  187. package/.agent/mindforge/review.md +0 -161
  188. package/.agent/mindforge/security-scan.md +0 -242
  189. package/.agent/mindforge/session-report.md +0 -39
  190. package/.agent/mindforge/ship.md +0 -111
  191. package/.agent/mindforge/skills.md +0 -145
  192. package/.agent/mindforge/status.md +0 -113
  193. package/.agent/mindforge/steer.md +0 -17
  194. package/.agent/mindforge/sync-confluence.md +0 -15
  195. package/.agent/mindforge/sync-jira.md +0 -16
  196. package/.agent/mindforge/tokens.md +0 -12
  197. package/.agent/mindforge/ui-phase.md +0 -34
  198. package/.agent/mindforge/ui-review.md +0 -36
  199. package/.agent/mindforge/update.md +0 -46
  200. package/.agent/mindforge/validate-phase.md +0 -31
  201. package/.agent/mindforge/verify-phase.md +0 -66
  202. package/.agent/mindforge/workspace.md +0 -33
  203. package/.agent/mindforge/workstreams.md +0 -35
  204. package/.agent/settings.json +0 -42
  205. package/.agent/skills/mindforge-add-backlog/SKILL.md +0 -72
  206. package/.agent/skills/mindforge-add-phase/SKILL.md +0 -39
  207. package/.agent/skills/mindforge-add-tests/SKILL.md +0 -28
  208. package/.agent/skills/mindforge-add-todo/SKILL.md +0 -42
  209. package/.agent/skills/mindforge-audit-milestone/SKILL.md +0 -29
  210. package/.agent/skills/mindforge-audit-uat/SKILL.md +0 -20
  211. package/.agent/skills/mindforge-autonomous/SKILL.md +0 -33
  212. package/.agent/skills/mindforge-brainstorming/SKILL.md +0 -164
  213. package/.agent/skills/mindforge-brainstorming/scripts/frame-template.html +0 -214
  214. package/.agent/skills/mindforge-brainstorming/scripts/helper.js +0 -88
  215. package/.agent/skills/mindforge-brainstorming/scripts/server.cjs +0 -354
  216. package/.agent/skills/mindforge-brainstorming/scripts/start-server.sh +0 -148
  217. package/.agent/skills/mindforge-brainstorming/scripts/stop-server.sh +0 -56
  218. package/.agent/skills/mindforge-brainstorming/spec-document-reviewer-prompt.md +0 -49
  219. package/.agent/skills/mindforge-brainstorming/visual-companion.md +0 -287
  220. package/.agent/skills/mindforge-check-todos/SKILL.md +0 -40
  221. package/.agent/skills/mindforge-cleanup/SKILL.md +0 -19
  222. package/.agent/skills/mindforge-complete-milestone/SKILL.md +0 -131
  223. package/.agent/skills/mindforge-debug/SKILL.md +0 -163
  224. package/.agent/skills/mindforge-debug_extended/CREATION-LOG.md +0 -119
  225. package/.agent/skills/mindforge-debug_extended/SKILL.md +0 -296
  226. package/.agent/skills/mindforge-debug_extended/condition-based-waiting-example.ts +0 -158
  227. package/.agent/skills/mindforge-debug_extended/condition-based-waiting.md +0 -115
  228. package/.agent/skills/mindforge-debug_extended/defense-in-depth.md +0 -122
  229. package/.agent/skills/mindforge-debug_extended/find-polluter.sh +0 -63
  230. package/.agent/skills/mindforge-debug_extended/root-cause-tracing.md +0 -169
  231. package/.agent/skills/mindforge-debug_extended/test-academic.md +0 -14
  232. package/.agent/skills/mindforge-debug_extended/test-pressure-1.md +0 -58
  233. package/.agent/skills/mindforge-debug_extended/test-pressure-2.md +0 -68
  234. package/.agent/skills/mindforge-debug_extended/test-pressure-3.md +0 -69
  235. package/.agent/skills/mindforge-discuss-phase/SKILL.md +0 -54
  236. package/.agent/skills/mindforge-do/SKILL.md +0 -26
  237. package/.agent/skills/mindforge-execute-phase/SKILL.md +0 -49
  238. package/.agent/skills/mindforge-execute-phase_extended/SKILL.md +0 -70
  239. package/.agent/skills/mindforge-fast/SKILL.md +0 -23
  240. package/.agent/skills/mindforge-forensics/SKILL.md +0 -49
  241. package/.agent/skills/mindforge-health/SKILL.md +0 -17
  242. package/.agent/skills/mindforge-help/SKILL.md +0 -23
  243. package/.agent/skills/mindforge-insert-phase/SKILL.md +0 -28
  244. package/.agent/skills/mindforge-join-discord/SKILL.md +0 -19
  245. package/.agent/skills/mindforge-list-phase-assumptions/SKILL.md +0 -41
  246. package/.agent/skills/mindforge-list-workspaces/SKILL.md +0 -17
  247. package/.agent/skills/mindforge-manager/SKILL.md +0 -32
  248. package/.agent/skills/mindforge-map-codebase/SKILL.md +0 -64
  249. package/.agent/skills/mindforge-milestone-summary/SKILL.md +0 -44
  250. package/.agent/skills/mindforge-neural-orchestrator/SKILL.md +0 -115
  251. package/.agent/skills/mindforge-neural-orchestrator/references/codex-tools.md +0 -100
  252. package/.agent/skills/mindforge-neural-orchestrator/references/gemini-tools.md +0 -33
  253. package/.agent/skills/mindforge-new-milestone/SKILL.md +0 -38
  254. package/.agent/skills/mindforge-new-project/SKILL.md +0 -36
  255. package/.agent/skills/mindforge-new-workspace/SKILL.md +0 -39
  256. package/.agent/skills/mindforge-next/SKILL.md +0 -19
  257. package/.agent/skills/mindforge-note/SKILL.md +0 -29
  258. package/.agent/skills/mindforge-parallel-mesh_extended/SKILL.md +0 -182
  259. package/.agent/skills/mindforge-pause-work/SKILL.md +0 -35
  260. package/.agent/skills/mindforge-plan-milestone-gaps/SKILL.md +0 -28
  261. package/.agent/skills/mindforge-plan-phase/SKILL.md +0 -38
  262. package/.agent/skills/mindforge-plan-phase_extended/SKILL.md +0 -152
  263. package/.agent/skills/mindforge-plan-phase_extended/plan-document-reviewer-prompt.md +0 -49
  264. package/.agent/skills/mindforge-plant-seed/SKILL.md +0 -22
  265. package/.agent/skills/mindforge-pr-branch/SKILL.md +0 -21
  266. package/.agent/skills/mindforge-profile-user/SKILL.md +0 -38
  267. package/.agent/skills/mindforge-progress/SKILL.md +0 -19
  268. package/.agent/skills/mindforge-quick/SKILL.md +0 -38
  269. package/.agent/skills/mindforge-reapply-patches/SKILL.md +0 -124
  270. package/.agent/skills/mindforge-remove-phase/SKILL.md +0 -26
  271. package/.agent/skills/mindforge-remove-workspace/SKILL.md +0 -22
  272. package/.agent/skills/mindforge-research-phase/SKILL.md +0 -186
  273. package/.agent/skills/mindforge-resume-work/SKILL.md +0 -35
  274. package/.agent/skills/mindforge-review/SKILL.md +0 -31
  275. package/.agent/skills/mindforge-review-backlog/SKILL.md +0 -58
  276. package/.agent/skills/mindforge-review-inbound/SKILL.md +0 -213
  277. package/.agent/skills/mindforge-review-request/SKILL.md +0 -105
  278. package/.agent/skills/mindforge-review-request/code-reviewer.md +0 -146
  279. package/.agent/skills/mindforge-session-report/SKILL.md +0 -16
  280. package/.agent/skills/mindforge-set-profile/SKILL.md +0 -9
  281. package/.agent/skills/mindforge-settings/SKILL.md +0 -32
  282. package/.agent/skills/mindforge-ship/SKILL.md +0 -16
  283. package/.agent/skills/mindforge-ship_extended/SKILL.md +0 -200
  284. package/.agent/skills/mindforge-skill-creation/SKILL.md +0 -655
  285. package/.agent/skills/mindforge-skill-creation/anthropic-best-practices.md +0 -1150
  286. package/.agent/skills/mindforge-skill-creation/examples/CLAUDE_MD_TESTING.md +0 -189
  287. package/.agent/skills/mindforge-skill-creation/graphviz-conventions.dot +0 -172
  288. package/.agent/skills/mindforge-skill-creation/persuasion-principles.md +0 -187
  289. package/.agent/skills/mindforge-skill-creation/render-graphs.js +0 -168
  290. package/.agent/skills/mindforge-skill-creation/testing-skills-with-subagents.md +0 -384
  291. package/.agent/skills/mindforge-stats/SKILL.md +0 -16
  292. package/.agent/skills/mindforge-swarm-execution/SKILL.md +0 -277
  293. package/.agent/skills/mindforge-swarm-execution/code-quality-reviewer-prompt.md +0 -26
  294. package/.agent/skills/mindforge-swarm-execution/implementer-prompt.md +0 -113
  295. package/.agent/skills/mindforge-swarm-execution/spec-reviewer-prompt.md +0 -61
  296. package/.agent/skills/mindforge-system-architecture/SKILL.md +0 -136
  297. package/.agent/skills/mindforge-system-architecture/examples.md +0 -120
  298. package/.agent/skills/mindforge-system-architecture/scaling-checklist.md +0 -76
  299. package/.agent/skills/mindforge-tdd/SKILL.md +0 -112
  300. package/.agent/skills/mindforge-tdd/deep-modules.md +0 -21
  301. package/.agent/skills/mindforge-tdd/interface-design.md +0 -22
  302. package/.agent/skills/mindforge-tdd/mocking.md +0 -24
  303. package/.agent/skills/mindforge-tdd/refactoring.md +0 -21
  304. package/.agent/skills/mindforge-tdd/tests.md +0 -28
  305. package/.agent/skills/mindforge-tdd_extended/SKILL.md +0 -371
  306. package/.agent/skills/mindforge-tdd_extended/testing-anti-patterns.md +0 -299
  307. package/.agent/skills/mindforge-thread/SKILL.md +0 -123
  308. package/.agent/skills/mindforge-ui-phase/SKILL.md +0 -24
  309. package/.agent/skills/mindforge-ui-review/SKILL.md +0 -24
  310. package/.agent/skills/mindforge-update/SKILL.md +0 -35
  311. package/.agent/skills/mindforge-validate-phase/SKILL.md +0 -26
  312. package/.agent/skills/mindforge-verify-work/SKILL.md +0 -30
  313. package/.agent/skills/mindforge-verify-work_extended/SKILL.md +0 -139
  314. package/.agent/skills/mindforge-workspace-isolated/SKILL.md +0 -218
  315. package/.agent/skills/mindforge-workstreams/SKILL.md +0 -65
  316. package/.agent/workflows/forge:help.md +0 -10
  317. package/.agent/workflows/forge:init-project.md +0 -35
  318. package/.agent/workflows/forge:plan-phase.md +0 -33
  319. package/.agent/workflows/mindforge-add-phase.md +0 -112
  320. package/.agent/workflows/mindforge-add-tests.md +0 -351
  321. package/.agent/workflows/mindforge-add-todo.md +0 -158
  322. package/.agent/workflows/mindforge-audit-milestone.md +0 -332
  323. package/.agent/workflows/mindforge-audit-uat.md +0 -109
  324. package/.agent/workflows/mindforge-autonomous.md +0 -815
  325. package/.agent/workflows/mindforge-check-todos.md +0 -177
  326. package/.agent/workflows/mindforge-cleanup.md +0 -152
  327. package/.agent/workflows/mindforge-complete-milestone.md +0 -766
  328. package/.agent/workflows/mindforge-diagnose-issues.md +0 -220
  329. package/.agent/workflows/mindforge-discovery-phase.md +0 -289
  330. package/.agent/workflows/mindforge-discuss-phase-assumptions.md +0 -645
  331. package/.agent/workflows/mindforge-discuss-phase.md +0 -1047
  332. package/.agent/workflows/mindforge-do.md +0 -104
  333. package/.agent/workflows/mindforge-execute-phase.md +0 -838
  334. package/.agent/workflows/mindforge-execute-plan.md +0 -509
  335. package/.agent/workflows/mindforge-fast.md +0 -105
  336. package/.agent/workflows/mindforge-forensics.md +0 -265
  337. package/.agent/workflows/mindforge-health.md +0 -181
  338. package/.agent/workflows/mindforge-help.md +0 -606
  339. package/.agent/workflows/mindforge-insert-phase.md +0 -130
  340. package/.agent/workflows/mindforge-list-phase-assumptions.md +0 -178
  341. package/.agent/workflows/mindforge-list-workspaces.md +0 -56
  342. package/.agent/workflows/mindforge-manager.md +0 -360
  343. package/.agent/workflows/mindforge-map-codebase.md +0 -370
  344. package/.agent/workflows/mindforge-milestone-summary.md +0 -223
  345. package/.agent/workflows/mindforge-new-milestone.md +0 -469
  346. package/.agent/workflows/mindforge-new-project.md +0 -1226
  347. package/.agent/workflows/mindforge-new-workspace.md +0 -237
  348. package/.agent/workflows/mindforge-next.md +0 -97
  349. package/.agent/workflows/mindforge-node-repair.md +0 -92
  350. package/.agent/workflows/mindforge-note.md +0 -156
  351. package/.agent/workflows/mindforge-pause-work.md +0 -176
  352. package/.agent/workflows/mindforge-plan-milestone-gaps.md +0 -273
  353. package/.agent/workflows/mindforge-plan-phase.md +0 -877
  354. package/.agent/workflows/mindforge-plant-seed.md +0 -169
  355. package/.agent/workflows/mindforge-pr-branch.md +0 -129
  356. package/.agent/workflows/mindforge-profile-user.md +0 -450
  357. package/.agent/workflows/mindforge-progress.md +0 -507
  358. package/.agent/workflows/mindforge-quick.md +0 -732
  359. package/.agent/workflows/mindforge-remove-phase.md +0 -155
  360. package/.agent/workflows/mindforge-remove-workspace.md +0 -90
  361. package/.agent/workflows/mindforge-research-phase.md +0 -74
  362. package/.agent/workflows/mindforge-resume-project.md +0 -325
  363. package/.agent/workflows/mindforge-review.md +0 -228
  364. package/.agent/workflows/mindforge-session-report.md +0 -146
  365. package/.agent/workflows/mindforge-settings.md +0 -283
  366. package/.agent/workflows/mindforge-ship.md +0 -228
  367. package/.agent/workflows/mindforge-stats.md +0 -60
  368. package/.agent/workflows/mindforge-transition.md +0 -671
  369. package/.agent/workflows/mindforge-ui-phase.md +0 -290
  370. package/.agent/workflows/mindforge-ui-review.md +0 -157
  371. package/.agent/workflows/mindforge-update.md +0 -323
  372. package/.agent/workflows/mindforge-validate-phase.md +0 -167
  373. package/.agent/workflows/mindforge-verify-phase.md +0 -254
  374. package/.agent/workflows/mindforge-verify-work.md +0 -628
  375. package/.agent/workflows/mindforge:add-backlog.md +0 -24
  376. package/.agent/workflows/mindforge:agent.md +0 -25
  377. package/.agent/workflows/mindforge:approve.md +0 -21
  378. package/.agent/workflows/mindforge:architecture.md +0 -40
  379. package/.agent/workflows/mindforge:audit.md +0 -33
  380. package/.agent/workflows/mindforge:auto.md +0 -25
  381. package/.agent/workflows/mindforge:benchmark.md +0 -36
  382. package/.agent/workflows/mindforge:brainstorming.md +0 -16
  383. package/.agent/workflows/mindforge:browse.md +0 -29
  384. package/.agent/workflows/mindforge:complete-milestone.md +0 -21
  385. package/.agent/workflows/mindforge:costs.md +0 -14
  386. package/.agent/workflows/mindforge:cross-review.md +0 -20
  387. package/.agent/workflows/mindforge:dashboard.md +0 -101
  388. package/.agent/workflows/mindforge:debug.md +0 -131
  389. package/.agent/workflows/mindforge:discuss-phase.md +0 -141
  390. package/.agent/workflows/mindforge:do.md +0 -25
  391. package/.agent/workflows/mindforge:execute-phase.md +0 -205
  392. package/.agent/workflows/mindforge:executor.md +0 -18
  393. package/.agent/workflows/mindforge:health.md +0 -24
  394. package/.agent/workflows/mindforge:help.md +0 -26
  395. package/.agent/workflows/mindforge:identity.md +0 -18
  396. package/.agent/workflows/mindforge:init-org.md +0 -134
  397. package/.agent/workflows/mindforge:init-project.md +0 -185
  398. package/.agent/workflows/mindforge:install-skill.md +0 -27
  399. package/.agent/workflows/mindforge:learn.md +0 -146
  400. package/.agent/workflows/mindforge:map-codebase.md +0 -301
  401. package/.agent/workflows/mindforge:marketplace.md +0 -123
  402. package/.agent/workflows/mindforge:memory.md +0 -18
  403. package/.agent/workflows/mindforge:metrics.md +0 -25
  404. package/.agent/workflows/mindforge:migrate.md +0 -43
  405. package/.agent/workflows/mindforge:milestone.md +0 -15
  406. package/.agent/workflows/mindforge:new-runtime.md +0 -22
  407. package/.agent/workflows/mindforge:next.md +0 -108
  408. package/.agent/workflows/mindforge:note.md +0 -27
  409. package/.agent/workflows/mindforge:plan-phase.md +0 -139
  410. package/.agent/workflows/mindforge:planner.md +0 -18
  411. package/.agent/workflows/mindforge:plant-seed.md +0 -24
  412. package/.agent/workflows/mindforge:plugins.md +0 -43
  413. package/.agent/workflows/mindforge:pr-review.md +0 -44
  414. package/.agent/workflows/mindforge:profile-team.md +0 -26
  415. package/.agent/workflows/mindforge:publish-skill.md +0 -22
  416. package/.agent/workflows/mindforge:qa.md +0 -19
  417. package/.agent/workflows/mindforge:quick.md +0 -138
  418. package/.agent/workflows/mindforge:release.md +0 -13
  419. package/.agent/workflows/mindforge:remember.md +0 -29
  420. package/.agent/workflows/mindforge:research.md +0 -15
  421. package/.agent/workflows/mindforge:researcher.md +0 -18
  422. package/.agent/workflows/mindforge:retrospective.md +0 -29
  423. package/.agent/workflows/mindforge:review-backlog.md +0 -26
  424. package/.agent/workflows/mindforge:review.md +0 -160
  425. package/.agent/workflows/mindforge:reviewer.md +0 -18
  426. package/.agent/workflows/mindforge:security-scan.md +0 -236
  427. package/.agent/workflows/mindforge:session-report.md +0 -31
  428. package/.agent/workflows/mindforge:ship.md +0 -108
  429. package/.agent/workflows/mindforge:skills.md +0 -144
  430. package/.agent/workflows/mindforge:soul.md +0 -54
  431. package/.agent/workflows/mindforge:status.md +0 -107
  432. package/.agent/workflows/mindforge:steer.md +0 -16
  433. package/.agent/workflows/mindforge:sync-confluence.md +0 -14
  434. package/.agent/workflows/mindforge:sync-jira.md +0 -15
  435. package/.agent/workflows/mindforge:tdd.md +0 -46
  436. package/.agent/workflows/mindforge:tokens.md +0 -11
  437. package/.agent/workflows/mindforge:tool.md +0 -18
  438. package/.agent/workflows/mindforge:ui-phase.md +0 -27
  439. package/.agent/workflows/mindforge:ui-review.md +0 -28
  440. package/.agent/workflows/mindforge:update.md +0 -45
  441. package/.agent/workflows/mindforge:validate-phase.md +0 -25
  442. package/.agent/workflows/mindforge:verify-phase.md +0 -65
  443. package/.agent/workflows/mindforge:workspace.md +0 -32
  444. package/.agent/workflows/mindforge:workstreams.md +0 -27
  445. package/.agent/workflows/publish-release.md +0 -36
  446. package/.claude/CLAUDE.md +0 -102
  447. package/.claude/commands/forge/help.md +0 -7
  448. package/.claude/commands/forge/init-project.md +0 -32
  449. package/.claude/commands/forge/plan-phase.md +0 -30
  450. package/.claude/commands/mindforge/add-backlog.md +0 -32
  451. package/.claude/commands/mindforge/agent.md +0 -31
  452. package/.claude/commands/mindforge/approve.md +0 -22
  453. package/.claude/commands/mindforge/audit.md +0 -34
  454. package/.claude/commands/mindforge/auto.md +0 -26
  455. package/.claude/commands/mindforge/benchmark.md +0 -37
  456. package/.claude/commands/mindforge/browse.md +0 -30
  457. package/.claude/commands/mindforge/complete-milestone.md +0 -22
  458. package/.claude/commands/mindforge/costs.md +0 -15
  459. package/.claude/commands/mindforge/cross-review.md +0 -21
  460. package/.claude/commands/mindforge/dashboard.md +0 -102
  461. package/.claude/commands/mindforge/debug.md +0 -133
  462. package/.claude/commands/mindforge/discuss-phase.md +0 -142
  463. package/.claude/commands/mindforge/do.md +0 -31
  464. package/.claude/commands/mindforge/execute-phase.md +0 -200
  465. package/.claude/commands/mindforge/health.md +0 -31
  466. package/.claude/commands/mindforge/help.md +0 -33
  467. package/.claude/commands/mindforge/init-org.md +0 -135
  468. package/.claude/commands/mindforge/init-project.md +0 -170
  469. package/.claude/commands/mindforge/install-skill.md +0 -28
  470. package/.claude/commands/mindforge/learn.md +0 -147
  471. package/.claude/commands/mindforge/learning.md +0 -20
  472. package/.claude/commands/mindforge/map-codebase.md +0 -302
  473. package/.claude/commands/mindforge/marketplace.md +0 -124
  474. package/.claude/commands/mindforge/metrics.md +0 -26
  475. package/.claude/commands/mindforge/migrate.md +0 -44
  476. package/.claude/commands/mindforge/milestone.md +0 -16
  477. package/.claude/commands/mindforge/new-runtime.md +0 -23
  478. package/.claude/commands/mindforge/next.md +0 -109
  479. package/.claude/commands/mindforge/note.md +0 -35
  480. package/.claude/commands/mindforge/plan-phase.md +0 -131
  481. package/.claude/commands/mindforge/plant-seed.md +0 -31
  482. package/.claude/commands/mindforge/plugins.md +0 -44
  483. package/.claude/commands/mindforge/pr-review.md +0 -45
  484. package/.claude/commands/mindforge/profile-team.md +0 -27
  485. package/.claude/commands/mindforge/publish-skill.md +0 -23
  486. package/.claude/commands/mindforge/qa.md +0 -20
  487. package/.claude/commands/mindforge/quick.md +0 -139
  488. package/.claude/commands/mindforge/record-learning.md +0 -22
  489. package/.claude/commands/mindforge/release.md +0 -14
  490. package/.claude/commands/mindforge/remember.md +0 -30
  491. package/.claude/commands/mindforge/research.md +0 -16
  492. package/.claude/commands/mindforge/retrospective.md +0 -31
  493. package/.claude/commands/mindforge/review-backlog.md +0 -34
  494. package/.claude/commands/mindforge/review.md +0 -161
  495. package/.claude/commands/mindforge/security-scan.md +0 -242
  496. package/.claude/commands/mindforge/session-report.md +0 -39
  497. package/.claude/commands/mindforge/ship.md +0 -111
  498. package/.claude/commands/mindforge/skills.md +0 -145
  499. package/.claude/commands/mindforge/status.md +0 -113
  500. package/.claude/commands/mindforge/steer.md +0 -17
  501. package/.claude/commands/mindforge/sync-confluence.md +0 -15
  502. package/.claude/commands/mindforge/sync-jira.md +0 -16
  503. package/.claude/commands/mindforge/tokens.md +0 -12
  504. package/.claude/commands/mindforge/ui-phase.md +0 -34
  505. package/.claude/commands/mindforge/ui-review.md +0 -36
  506. package/.claude/commands/mindforge/update.md +0 -46
  507. package/.claude/commands/mindforge/validate-phase.md +0 -31
  508. package/.claude/commands/mindforge/verify-phase.md +0 -66
  509. package/.claude/commands/mindforge/workspace.md +0 -33
  510. package/.claude/commands/mindforge/workstreams.md +0 -35
  511. package/.claude/settings.local.json +0 -16
  512. package/.mindforge/audit/AUDIT-SCHEMA.md +0 -470
  513. package/.mindforge/browser/daemon-protocol.md +0 -24
  514. package/.mindforge/browser/qa-engine.md +0 -16
  515. package/.mindforge/browser/session-manager.md +0 -18
  516. package/.mindforge/browser/visual-verify-spec.md +0 -31
  517. package/.mindforge/celestial.db +0 -0
  518. package/.mindforge/ci/ci-config-schema.md +0 -21
  519. package/.mindforge/ci/ci-mode.md +0 -179
  520. package/.mindforge/ci/github-actions-adapter.md +0 -224
  521. package/.mindforge/ci/gitlab-ci-adapter.md +0 -31
  522. package/.mindforge/ci/jenkins-adapter.md +0 -44
  523. package/.mindforge/dashboard/api-reference.md +0 -122
  524. package/.mindforge/dashboard/dashboard-spec.md +0 -96
  525. package/.mindforge/distribution/marketplace.md +0 -53
  526. package/.mindforge/distribution/registry-client.md +0 -166
  527. package/.mindforge/distribution/registry-schema.md +0 -96
  528. package/.mindforge/distribution/skill-publisher.md +0 -44
  529. package/.mindforge/distribution/skill-validator.md +0 -74
  530. package/.mindforge/governance/GOVERNANCE-CONFIG.md +0 -17
  531. package/.mindforge/governance/approval-workflow.md +0 -37
  532. package/.mindforge/governance/change-classifier.md +0 -63
  533. package/.mindforge/governance/compliance-gates.md +0 -31
  534. package/.mindforge/governance/policies/sovereign-default.json +0 -16
  535. package/.mindforge/integrations/confluence.md +0 -27
  536. package/.mindforge/integrations/connection-manager.md +0 -163
  537. package/.mindforge/integrations/github.md +0 -25
  538. package/.mindforge/integrations/gitlab.md +0 -13
  539. package/.mindforge/integrations/jira.md +0 -102
  540. package/.mindforge/integrations/slack.md +0 -41
  541. package/.mindforge/intelligence/antipattern-detector.md +0 -75
  542. package/.mindforge/intelligence/difficulty-scorer.md +0 -55
  543. package/.mindforge/intelligence/health-engine.md +0 -208
  544. package/.mindforge/intelligence/skill-gap-analyser.md +0 -40
  545. package/.mindforge/intelligence/smart-compaction.md +0 -71
  546. package/.mindforge/memory/MEMORY-SCHEMA.md +0 -155
  547. package/.mindforge/memory/engine/capture-protocol.md +0 -36
  548. package/.mindforge/memory/engine/global-sync-spec.md +0 -42
  549. package/.mindforge/memory/engine/retrieval-spec.md +0 -44
  550. package/.mindforge/memory/sync-manifest.json +0 -6
  551. package/.mindforge/metrics/METRICS-SCHEMA.md +0 -42
  552. package/.mindforge/metrics/quality-tracker.md +0 -32
  553. package/.mindforge/models/model-registry.md +0 -48
  554. package/.mindforge/models/model-router.md +0 -30
  555. package/.mindforge/monorepo/cross-package-planner.md +0 -114
  556. package/.mindforge/monorepo/dependency-graph-builder.md +0 -32
  557. package/.mindforge/monorepo/workspace-detector.md +0 -129
  558. package/.mindforge/org/CONVENTIONS.md +0 -62
  559. package/.mindforge/org/ORG.md +0 -51
  560. package/.mindforge/org/SECURITY.md +0 -50
  561. package/.mindforge/org/TOOLS.md +0 -53
  562. package/.mindforge/org/integrations/INTEGRATIONS-CONFIG.md +0 -58
  563. package/.mindforge/org/skills/MANIFEST.md +0 -15
  564. package/.mindforge/plugins/PLUGINS-MANIFEST.md +0 -23
  565. package/.mindforge/plugins/plugin-loader.md +0 -93
  566. package/.mindforge/plugins/plugin-registry.md +0 -44
  567. package/.mindforge/plugins/plugin-schema.md +0 -68
  568. package/.mindforge/pr-review/ai-reviewer.md +0 -266
  569. package/.mindforge/pr-review/finding-formatter.md +0 -46
  570. package/.mindforge/pr-review/review-prompt-templates.md +0 -44
  571. package/.mindforge/production/compatibility-layer.md +0 -39
  572. package/.mindforge/production/migration-engine.md +0 -52
  573. package/.mindforge/production/production-checklist.md +0 -76
  574. package/.mindforge/production/token-optimiser.md +0 -68
  575. package/.mindforge/skills-builder/auto-capture-protocol.md +0 -88
  576. package/.mindforge/skills-builder/learn-protocol.md +0 -161
  577. package/.mindforge/skills-builder/quality-scoring.md +0 -120
  578. package/.mindforge/team/TEAM-PROFILE.md +0 -42
  579. package/.mindforge/team/multi-handoff.md +0 -23
  580. package/.mindforge/team/profiles/README.md +0 -13
  581. package/.mindforge/team/session-merger.md +0 -18
  582. package/.planning/ARCHITECTURE.md +0 -0
  583. package/.planning/HANDOFF.json +0 -8
  584. package/.planning/PROJECT.md +0 -33
  585. package/.planning/RELEASE-CHECKLIST.md +0 -68
  586. package/.planning/REQUIREMENTS.md +0 -35
  587. package/.planning/ROADMAP.md +0 -12
  588. package/.planning/STATE.md +0 -36
  589. package/.planning/approvals/.gitkeep +0 -1
  590. package/.planning/archive/.gitkeep +0 -1
  591. package/.planning/audit-archive/.gitkeep +0 -1
  592. package/.planning/decisions/.gitkeep +0 -0
  593. package/.planning/jira-sync.json +0 -1
  594. package/.planning/milestones/.gitkeep +0 -1
  595. package/.planning/phases/.gitkeep +0 -0
  596. package/.planning/research/.gitkeep +0 -0
  597. package/.planning/screenshots/.gitkeep +0 -0
  598. package/.planning/slack-threads.json +0 -1
  599. package/docs/CAPABILITIES-MANIFEST.md +0 -64
  600. package/docs/Context/Master-Context.md +0 -731
  601. package/docs/INTELLIGENCE-MESH.md +0 -37
  602. package/docs/MIND-FORGE-REFERENCE-V6.md +0 -96
  603. package/docs/PERSONAS.md +0 -960
  604. package/docs/References/audit-events.md +0 -59
  605. package/docs/References/checkpoints.md +0 -778
  606. package/docs/References/commands.md +0 -107
  607. package/docs/References/config-reference.md +0 -81
  608. package/docs/References/continuation-format.md +0 -249
  609. package/docs/References/decimal-phase-calculation.md +0 -64
  610. package/docs/References/git-integration.md +0 -295
  611. package/docs/References/git-planning-commit.md +0 -38
  612. package/docs/References/model-profile-resolution.md +0 -36
  613. package/docs/References/model-profiles.md +0 -139
  614. package/docs/References/phase-argument-parsing.md +0 -61
  615. package/docs/References/planning-config.md +0 -202
  616. package/docs/References/questioning.md +0 -162
  617. package/docs/References/sdk-api.md +0 -53
  618. package/docs/References/skills-api.md +0 -57
  619. package/docs/References/tdd.md +0 -263
  620. package/docs/References/ui-brand.md +0 -160
  621. package/docs/References/user-profiling.md +0 -681
  622. package/docs/References/verification-patterns.md +0 -612
  623. package/docs/References/workstream-flag.md +0 -58
  624. package/docs/Templates/Agents/CLAUDE-MD.md +0 -122
  625. package/docs/Templates/Agents/COPILOT-INSTRUCTIONS.md +0 -7
  626. package/docs/Templates/Agents/DEBUGGER-PROMPT.md +0 -91
  627. package/docs/Templates/Agents/PLANNER-PROMPT.md +0 -117
  628. package/docs/Templates/Codebase/architecture.md +0 -255
  629. package/docs/Templates/Codebase/concerns.md +0 -310
  630. package/docs/Templates/Codebase/conventions.md +0 -307
  631. package/docs/Templates/Codebase/integrations.md +0 -280
  632. package/docs/Templates/Codebase/stack.md +0 -186
  633. package/docs/Templates/Codebase/structure.md +0 -285
  634. package/docs/Templates/Codebase/testing.md +0 -480
  635. package/docs/Templates/Execution/CONTINUE-HERE.md +0 -78
  636. package/docs/Templates/Execution/DISCUSSION-LOG.md +0 -63
  637. package/docs/Templates/Execution/PHASE-PROMPT.md +0 -610
  638. package/docs/Templates/Execution/STATE.md +0 -176
  639. package/docs/Templates/Execution/SUMMARY-COMPLEX.md +0 -59
  640. package/docs/Templates/Execution/SUMMARY-MINIMAL.md +0 -41
  641. package/docs/Templates/Execution/SUMMARY-STANDARD.md +0 -48
  642. package/docs/Templates/Execution/SUMMARY.md +0 -248
  643. package/docs/Templates/Profile/DEV-PREFERENCES.md +0 -21
  644. package/docs/Templates/Profile/USER-PROFILE.md +0 -146
  645. package/docs/Templates/Profile/USER-SETUP.md +0 -311
  646. package/docs/Templates/Project/AGENTS_LEARNING.md +0 -88
  647. package/docs/Templates/Project/DISCOVERY.md +0 -146
  648. package/docs/Templates/Project/MILESTONE-ARCHIVE.md +0 -123
  649. package/docs/Templates/Project/MILESTONE.md +0 -115
  650. package/docs/Templates/Project/PROJECT.md +0 -206
  651. package/docs/Templates/Project/REQUIREMENTS.md +0 -231
  652. package/docs/Templates/Project/RETROSPECTIVE.md +0 -54
  653. package/docs/Templates/Project/ROADMAP.md +0 -202
  654. package/docs/Templates/Quality/DEBUG.md +0 -164
  655. package/docs/Templates/Quality/UAT.md +0 -280
  656. package/docs/Templates/Quality/UI-SPEC.md +0 -100
  657. package/docs/Templates/Quality/VALIDATION.md +0 -76
  658. package/docs/Templates/Quality/VERIFICATION-REPORT.md +0 -322
  659. package/docs/Templates/Research/ARCHITECTURE.md +0 -204
  660. package/docs/Templates/Research/FEATURES.md +0 -147
  661. package/docs/Templates/Research/PITFALLS.md +0 -200
  662. package/docs/Templates/Research/STACK.md +0 -120
  663. package/docs/Templates/Research/SUMMARY.md +0 -170
  664. package/docs/Templates/System/CONFIG.json +0 -43
  665. package/docs/Templates/System/CONTEXT.md +0 -352
  666. package/docs/adr/ADR-024-browser-localhost-only.md +0 -17
  667. package/docs/adr/ADR-025-visual-verify-failure-treatment.md +0 -19
  668. package/docs/adr/ADR-026-session-persistence-security.md +0 -20
  669. package/docs/adr/ADR-042-ads-protocol.md +0 -30
  670. package/docs/architecture/NEXUS-DASHBOARD.md +0 -35
  671. package/docs/architecture/PAR-ZTS-SURVEY.md +0 -43
  672. package/docs/architecture/README.md +0 -78
  673. package/docs/architecture/V3-CORE.md +0 -52
  674. package/docs/architecture/V4-SWARM-MESH.md +0 -77
  675. package/docs/architecture/V5-ENTERPRISE.md +0 -131
  676. package/docs/architecture/V6-SOVEREIGN.md +0 -43
  677. package/docs/architecture/V8-SRE.md +0 -88
  678. package/docs/architecture/V9-BEDROCK.md +0 -162
  679. package/docs/architecture/adr-039-multi-runtime-support.md +0 -20
  680. package/docs/architecture/adr-040-additive-schema-migration.md +0 -21
  681. package/docs/architecture/adr-041-stable-runtime-interface-contract.md +0 -20
  682. package/docs/architecture/decision-records-index.md +0 -29
  683. package/docs/ci-cd-integration.md +0 -30
  684. package/docs/ci-cd.md +0 -92
  685. package/docs/ci-quickstart.md +0 -78
  686. package/docs/commands-skills/DISCOVERED_SKILLS.md +0 -21
  687. package/docs/contributing/CONTRIBUTING.md +0 -38
  688. package/docs/contributing/plugin-authoring.md +0 -50
  689. package/docs/contributing/skill-authoring.md +0 -41
  690. package/docs/enterprise-setup.md +0 -25
  691. package/docs/feature-dashboard.md +0 -63
  692. package/docs/governance-guide.md +0 -134
  693. package/docs/monorepo-guide.md +0 -26
  694. package/docs/persona-customisation.md +0 -56
  695. package/docs/publishing-guide.md +0 -43
  696. package/docs/quick-verify.md +0 -33
  697. package/docs/registry/AGENTS.md +0 -37
  698. package/docs/registry/COMMANDS.md +0 -87
  699. package/docs/registry/HOOKS.md +0 -38
  700. package/docs/registry/PERSONAS.md +0 -64
  701. package/docs/registry/README.md +0 -27
  702. package/docs/registry/SKILLS.md +0 -142
  703. package/docs/registry/WORKFLOWS.md +0 -72
  704. package/docs/release-checklist-guide.md +0 -37
  705. package/docs/requirements.md +0 -29
  706. package/docs/security/SECURITY.md +0 -55
  707. package/docs/security/ZTAI-OVERVIEW.md +0 -37
  708. package/docs/security/penetration-test-results.md +0 -31
  709. package/docs/security/threat-model.md +0 -142
  710. package/docs/skills-authoring-guide.md +0 -176
  711. package/docs/skills-publishing-guide.md +0 -22
  712. package/docs/team-setup-guide.md +0 -21
  713. package/docs/testing-current-version.md +0 -130
  714. package/docs/tutorial.md +0 -162
  715. package/docs/upgrade.md +0 -58
  716. package/docs/usp-features.md +0 -102
  717. package/docs/workflow-atlas.md +0 -57
@@ -1,1166 +0,0 @@
1
- /**
2
- * Core — Shared utilities, constants, and internal helpers
3
- */
4
-
5
- const fs = require('fs');
6
- const path = require('path');
7
- const { execSync, execFileSync, spawnSync } = require('child_process');
8
- const { MODEL_PROFILES } = require('./model-profiles.cjs');
9
-
10
- // ─── Path helpers ────────────────────────────────────────────────────────────
11
-
12
- /** Normalize a relative path to always use forward slashes (cross-platform). */
13
- function toPosixPath(p) {
14
- return p.split(path.sep).join('/');
15
- }
16
-
17
- /**
18
- * Scan immediate child directories for separate git repos.
19
- * Returns a sorted array of directory names that have their own `.git`.
20
- * Excludes hidden directories and node_modules.
21
- */
22
- function detectSubRepos(cwd) {
23
- const results = [];
24
- try {
25
- const entries = fs.readdirSync(cwd, { withFileTypes: true });
26
- for (const entry of entries) {
27
- if (!entry.isDirectory()) continue;
28
- if (entry.name.startsWith('.') || entry.name === 'node_modules') continue;
29
- const gitPath = path.join(cwd, entry.name, '.git');
30
- try {
31
- if (fs.existsSync(gitPath)) {
32
- results.push(entry.name);
33
- }
34
- } catch {}
35
- }
36
- } catch {}
37
- return results.sort();
38
- }
39
-
40
- /**
41
- * Walk up from `startDir` to find the project root that owns `.planning/`.
42
- *
43
- * In multi-repo workspaces, the agent may open inside a sub-repo (e.g. `backend/`)
44
- * instead of the project root. This function prevents `.planning/` from being
45
- * created inside the sub-repo by locating the nearest ancestor that already has
46
- * a `.planning/` directory.
47
- *
48
- * Detection strategy (checked in order for each ancestor):
49
- * 1. Parent has `.planning/config.json` with `sub_repos` listing this directory
50
- * 2. Parent has `.planning/config.json` with `multiRepo: true` (legacy format)
51
- * 3. Parent has `.planning/` and current dir has its own `.git` (heuristic)
52
- *
53
- * Returns `startDir` unchanged when no ancestor `.planning/` is found (first-run
54
- * or single-repo projects).
55
- */
56
- function findProjectRoot(startDir) {
57
- const resolved = path.resolve(startDir);
58
- const root = path.parse(resolved).root;
59
- const homedir = require('os').homedir();
60
-
61
- // Check if startDir or any of its ancestors (up to but not including a
62
- // candidate project root) contains a .git directory. This handles both
63
- // `backend/` (direct sub-repo) and `backend/src/modules/` (nested inside).
64
- function isInsideGitRepo(candidateParent) {
65
- let d = resolved;
66
- while (d !== candidateParent && d !== root) {
67
- if (fs.existsSync(path.join(d, '.git'))) return true;
68
- d = path.dirname(d);
69
- }
70
- return false;
71
- }
72
-
73
- let dir = resolved;
74
- while (dir !== root) {
75
- const parent = path.dirname(dir);
76
- if (parent === dir) break; // filesystem root
77
- if (parent === homedir) break; // never go above home
78
-
79
- const parentPlanning = path.join(parent, '.planning');
80
- if (fs.existsSync(parentPlanning) && fs.statSync(parentPlanning).isDirectory()) {
81
- const configPath = path.join(parentPlanning, 'config.json');
82
- try {
83
- const config = JSON.parse(fs.readFileSync(configPath, 'utf-8'));
84
- const subRepos = config.sub_repos || config.planning?.sub_repos || [];
85
-
86
- // Check explicit sub_repos list
87
- if (Array.isArray(subRepos) && subRepos.length > 0) {
88
- const relPath = path.relative(parent, resolved);
89
- const topSegment = relPath.split(path.sep)[0];
90
- if (subRepos.includes(topSegment)) {
91
- return parent;
92
- }
93
- }
94
-
95
- // Check legacy multiRepo flag
96
- if (config.multiRepo === true && isInsideGitRepo(parent)) {
97
- return parent;
98
- }
99
- } catch {
100
- // config.json missing or malformed — fall back to .git heuristic
101
- }
102
-
103
- // Heuristic: parent has .planning/ and we're inside a git repo
104
- if (isInsideGitRepo(parent)) {
105
- return parent;
106
- }
107
- }
108
- dir = parent;
109
- }
110
- return startDir;
111
- }
112
-
113
- // ─── Output helpers ───────────────────────────────────────────────────────────
114
-
115
- /**
116
- * Remove stale mindforge-* temp files/dirs older than maxAgeMs (default: 5 minutes).
117
- * Runs opportunistically before each new temp file write to prevent unbounded accumulation.
118
- * @param {string} prefix - filename prefix to match (e.g., 'mindforge-')
119
- * @param {object} opts
120
- * @param {number} opts.maxAgeMs - max age in ms before removal (default: 5 min)
121
- * @param {boolean} opts.dirsOnly - if true, only remove directories (default: false)
122
- */
123
- function reapStaleTempFiles(prefix = 'mindforge-', { maxAgeMs = 5 * 60 * 1000, dirsOnly = false } = {}) {
124
- try {
125
- const tmpDir = require('os').tmpdir();
126
- const now = Date.now();
127
- const entries = fs.readdirSync(tmpDir);
128
- for (const entry of entries) {
129
- if (!entry.startsWith(prefix)) continue;
130
- const fullPath = path.join(tmpDir, entry);
131
- try {
132
- const stat = fs.statSync(fullPath);
133
- if (now - stat.mtimeMs > maxAgeMs) {
134
- if (stat.isDirectory()) {
135
- fs.rmSync(fullPath, { recursive: true, force: true });
136
- } else if (!dirsOnly) {
137
- fs.unlinkSync(fullPath);
138
- }
139
- }
140
- } catch {
141
- // File may have been removed between readdir and stat — ignore
142
- }
143
- }
144
- } catch {
145
- // Non-critical — don't let cleanup failures break output
146
- }
147
- }
148
-
149
- function output(result, raw, rawValue) {
150
- let data;
151
- if (raw && rawValue !== undefined) {
152
- data = String(rawValue);
153
- } else {
154
- const json = JSON.stringify(result, null, 2);
155
- // Large payloads exceed Claude Code's Bash tool buffer (~50KB).
156
- // Write to tmpfile and output the path prefixed with @file: so callers can detect it.
157
- if (json.length > 50000) {
158
- reapStaleTempFiles();
159
- const tmpPath = path.join(require('os').tmpdir(), `mindforge-${Date.now()}.json`);
160
- fs.writeFileSync(tmpPath, json, 'utf-8');
161
- data = '@file:' + tmpPath;
162
- } else {
163
- data = json;
164
- }
165
- }
166
- // process.stdout.write() is async when stdout is a pipe — process.exit()
167
- // can tear down the process before the reader consumes the buffer.
168
- // fs.writeSync(1, ...) blocks until the kernel accepts the bytes, and
169
- // skipping process.exit() lets the event loop drain naturally.
170
- fs.writeSync(1, data);
171
- }
172
-
173
- function error(message) {
174
- fs.writeSync(2, 'Error: ' + message + '\n');
175
- process.exit(1);
176
- }
177
-
178
- // ─── File & Config utilities ──────────────────────────────────────────────────
179
-
180
- function safeReadFile(filePath) {
181
- try {
182
- return fs.readFileSync(filePath, 'utf-8');
183
- } catch {
184
- return null;
185
- }
186
- }
187
-
188
- function loadConfig(cwd) {
189
- const configPath = path.join(cwd, '.planning', 'config.json');
190
- const defaults = {
191
- model_profile: 'balanced',
192
- commit_docs: true,
193
- search_gitignored: false,
194
- branching_strategy: 'none',
195
- phase_branch_template: 'mindforge/phase-{phase}-{slug}',
196
- milestone_branch_template: 'mindforge/{milestone}-{slug}',
197
- quick_branch_template: null,
198
- research: true,
199
- plan_checker: true,
200
- verifier: true,
201
- nyquist_validation: true,
202
- parallelization: true,
203
- brave_search: false,
204
- firecrawl: false,
205
- exa_search: false,
206
- text_mode: false, // when true, use plain-text numbered lists instead of AskUserQuestion menus
207
- sub_repos: [],
208
- resolve_model_ids: false, // false: return alias as-is | true: map to full the agent model ID | "omit": return '' (runtime uses its default)
209
- context_window: 200000, // default 200k; set to 1000000 for Opus/Sonnet 4.6 1M models
210
- phase_naming: 'sequential', // 'sequential' (default, auto-increment) or 'custom' (arbitrary string IDs)
211
- };
212
-
213
- try {
214
- const raw = fs.readFileSync(configPath, 'utf-8');
215
- const parsed = JSON.parse(raw);
216
-
217
- // Migrate deprecated "depth" key to "granularity" with value mapping
218
- if ('depth' in parsed && !('granularity' in parsed)) {
219
- const depthToGranularity = { quick: 'coarse', standard: 'standard', comprehensive: 'fine' };
220
- parsed.granularity = depthToGranularity[parsed.depth] || parsed.depth;
221
- delete parsed.depth;
222
- try { fs.writeFileSync(configPath, JSON.stringify(parsed, null, 2), 'utf-8'); } catch { /* intentionally empty */ }
223
- }
224
-
225
- // Auto-detect and sync sub_repos: scan for child directories with .git
226
- let configDirty = false;
227
-
228
- // Migrate legacy "multiRepo: true" boolean → sub_repos array
229
- if (parsed.multiRepo === true && !parsed.sub_repos && !parsed.planning?.sub_repos) {
230
- const detected = detectSubRepos(cwd);
231
- if (detected.length > 0) {
232
- parsed.sub_repos = detected;
233
- if (!parsed.planning) parsed.planning = {};
234
- parsed.planning.commit_docs = false;
235
- delete parsed.multiRepo;
236
- configDirty = true;
237
- }
238
- }
239
-
240
- // Keep sub_repos in sync with actual filesystem
241
- const currentSubRepos = parsed.sub_repos || parsed.planning?.sub_repos || [];
242
- if (Array.isArray(currentSubRepos) && currentSubRepos.length > 0) {
243
- const detected = detectSubRepos(cwd);
244
- if (detected.length > 0) {
245
- const sorted = [...currentSubRepos].sort();
246
- if (JSON.stringify(sorted) !== JSON.stringify(detected)) {
247
- parsed.sub_repos = detected;
248
- configDirty = true;
249
- }
250
- }
251
- }
252
-
253
- // Persist sub_repos changes (migration or sync)
254
- if (configDirty) {
255
- try { fs.writeFileSync(configPath, JSON.stringify(parsed, null, 2), 'utf-8'); } catch {}
256
- }
257
-
258
- const get = (key, nested) => {
259
- if (parsed[key] !== undefined) return parsed[key];
260
- if (nested && parsed[nested.section] && parsed[nested.section][nested.field] !== undefined) {
261
- return parsed[nested.section][nested.field];
262
- }
263
- return undefined;
264
- };
265
-
266
- const parallelization = (() => {
267
- const val = get('parallelization');
268
- if (typeof val === 'boolean') return val;
269
- if (typeof val === 'object' && val !== null && 'enabled' in val) return val.enabled;
270
- return defaults.parallelization;
271
- })();
272
-
273
- return {
274
- model_profile: get('model_profile') ?? defaults.model_profile,
275
- commit_docs: (() => {
276
- const explicit = get('commit_docs', { section: 'planning', field: 'commit_docs' });
277
- // If explicitly set in config, respect the user's choice
278
- if (explicit !== undefined) return explicit;
279
- // Auto-detection: when no explicit value and .planning/ is gitignored,
280
- // default to false instead of true
281
- if (isGitIgnored(cwd, '.planning/')) return false;
282
- return defaults.commit_docs;
283
- })(),
284
- search_gitignored: get('search_gitignored', { section: 'planning', field: 'search_gitignored' }) ?? defaults.search_gitignored,
285
- branching_strategy: get('branching_strategy', { section: 'git', field: 'branching_strategy' }) ?? defaults.branching_strategy,
286
- phase_branch_template: get('phase_branch_template', { section: 'git', field: 'phase_branch_template' }) ?? defaults.phase_branch_template,
287
- milestone_branch_template: get('milestone_branch_template', { section: 'git', field: 'milestone_branch_template' }) ?? defaults.milestone_branch_template,
288
- quick_branch_template: get('quick_branch_template', { section: 'git', field: 'quick_branch_template' }) ?? defaults.quick_branch_template,
289
- research: get('research', { section: 'workflow', field: 'research' }) ?? defaults.research,
290
- plan_checker: get('plan_checker', { section: 'workflow', field: 'plan_check' }) ?? defaults.plan_checker,
291
- verifier: get('verifier', { section: 'workflow', field: 'verifier' }) ?? defaults.verifier,
292
- nyquist_validation: get('nyquist_validation', { section: 'workflow', field: 'nyquist_validation' }) ?? defaults.nyquist_validation,
293
- parallelization,
294
- brave_search: get('brave_search') ?? defaults.brave_search,
295
- firecrawl: get('firecrawl') ?? defaults.firecrawl,
296
- exa_search: get('exa_search') ?? defaults.exa_search,
297
- text_mode: get('text_mode', { section: 'workflow', field: 'text_mode' }) ?? defaults.text_mode,
298
- sub_repos: get('sub_repos', { section: 'planning', field: 'sub_repos' }) ?? defaults.sub_repos,
299
- resolve_model_ids: get('resolve_model_ids') ?? defaults.resolve_model_ids,
300
- context_window: get('context_window') ?? defaults.context_window,
301
- phase_naming: get('phase_naming') ?? defaults.phase_naming,
302
- model_overrides: parsed.model_overrides || null,
303
- };
304
- } catch {
305
- return defaults;
306
- }
307
- }
308
-
309
- // ─── Git utilities ────────────────────────────────────────────────────────────
310
-
311
- function isGitIgnored(cwd, targetPath) {
312
- try {
313
- // --no-index checks .gitignore rules regardless of whether the file is tracked.
314
- // Without it, git check-ignore returns "not ignored" for tracked files even when
315
- // .gitignore explicitly lists them — a common source of confusion when .planning/
316
- // was committed before being added to .gitignore.
317
- // Use execFileSync (array args) to prevent shell interpretation of special characters
318
- // in file paths — avoids command injection via crafted path names.
319
- execFileSync('git', ['check-ignore', '-q', '--no-index', '--', targetPath], {
320
- cwd,
321
- stdio: 'pipe',
322
- });
323
- return true;
324
- } catch {
325
- return false;
326
- }
327
- }
328
-
329
- // ─── Markdown normalization ─────────────────────────────────────────────────
330
-
331
- /**
332
- * Normalize markdown to fix common markdownlint violations.
333
- * Applied at write points so MindForge-generated .planning/ files are IDE-friendly.
334
- *
335
- * Rules enforced:
336
- * MD022 — Blank lines around headings
337
- * MD031 — Blank lines around fenced code blocks
338
- * MD032 — Blank lines around lists
339
- * MD012 — No multiple consecutive blank lines (collapsed to 2 max)
340
- * MD047 — Files end with a single newline
341
- */
342
- function normalizeMd(content) {
343
- if (!content || typeof content !== 'string') return content;
344
-
345
- // Normalize line endings to LF for consistent processing
346
- let text = content.replace(/\r\n/g, '\n');
347
-
348
- const lines = text.split('\n');
349
- const result = [];
350
-
351
- for (let i = 0; i < lines.length; i++) {
352
- const line = lines[i];
353
- const prev = i > 0 ? lines[i - 1] : '';
354
- const prevTrimmed = prev.trimEnd();
355
- const trimmed = line.trimEnd();
356
-
357
- // MD022: Blank line before headings (skip first line and frontmatter delimiters)
358
- if (/^#{1,6}\s/.test(trimmed) && i > 0 && prevTrimmed !== '' && prevTrimmed !== '---') {
359
- result.push('');
360
- }
361
-
362
- // MD031: Blank line before fenced code blocks
363
- if (/^```/.test(trimmed) && i > 0 && prevTrimmed !== '' && !isInsideFencedBlock(lines, i)) {
364
- result.push('');
365
- }
366
-
367
- // MD032: Blank line before lists (- item, * item, N. item, - [ ] item)
368
- if (/^(\s*[-*+]\s|\s*\d+\.\s)/.test(line) && i > 0 &&
369
- prevTrimmed !== '' && !/^(\s*[-*+]\s|\s*\d+\.\s)/.test(prev) &&
370
- prevTrimmed !== '---') {
371
- result.push('');
372
- }
373
-
374
- result.push(line);
375
-
376
- // MD022: Blank line after headings
377
- if (/^#{1,6}\s/.test(trimmed) && i < lines.length - 1) {
378
- const next = lines[i + 1];
379
- if (next !== undefined && next.trimEnd() !== '') {
380
- result.push('');
381
- }
382
- }
383
-
384
- // MD031: Blank line after closing fenced code blocks
385
- if (/^```\s*$/.test(trimmed) && isClosingFence(lines, i) && i < lines.length - 1) {
386
- const next = lines[i + 1];
387
- if (next !== undefined && next.trimEnd() !== '') {
388
- result.push('');
389
- }
390
- }
391
-
392
- // MD032: Blank line after last list item in a block
393
- if (/^(\s*[-*+]\s|\s*\d+\.\s)/.test(line) && i < lines.length - 1) {
394
- const next = lines[i + 1];
395
- if (next !== undefined && next.trimEnd() !== '' &&
396
- !/^(\s*[-*+]\s|\s*\d+\.\s)/.test(next) &&
397
- !/^\s/.test(next)) {
398
- // Only add blank line if next line is not a continuation/indented line
399
- result.push('');
400
- }
401
- }
402
- }
403
-
404
- text = result.join('\n');
405
-
406
- // MD012: Collapse 3+ consecutive blank lines to 2
407
- text = text.replace(/\n{3,}/g, '\n\n');
408
-
409
- // MD047: Ensure file ends with exactly one newline
410
- text = text.replace(/\n*$/, '\n');
411
-
412
- return text;
413
- }
414
-
415
- /** Check if line index i is inside an already-open fenced code block */
416
- function isInsideFencedBlock(lines, i) {
417
- let fenceCount = 0;
418
- for (let j = 0; j < i; j++) {
419
- if (/^```/.test(lines[j].trimEnd())) fenceCount++;
420
- }
421
- return fenceCount % 2 === 1;
422
- }
423
-
424
- /** Check if a ``` line is a closing fence (odd number of fences up to and including this one) */
425
- function isClosingFence(lines, i) {
426
- let fenceCount = 0;
427
- for (let j = 0; j <= i; j++) {
428
- if (/^```/.test(lines[j].trimEnd())) fenceCount++;
429
- }
430
- return fenceCount % 2 === 0;
431
- }
432
-
433
- function execGit(cwd, args) {
434
- const result = spawnSync('git', args, {
435
- cwd,
436
- stdio: 'pipe',
437
- encoding: 'utf-8',
438
- });
439
- return {
440
- exitCode: result.status ?? 1,
441
- stdout: (result.stdout ?? '').toString().trim(),
442
- stderr: (result.stderr ?? '').toString().trim(),
443
- };
444
- }
445
-
446
- // ─── Common path helpers ──────────────────────────────────────────────────────
447
-
448
- /**
449
- * Resolve the main worktree root when running inside a git worktree.
450
- * In a linked worktree, .planning/ lives in the main worktree, not in the linked one.
451
- * Returns the main worktree path, or cwd if not in a worktree.
452
- */
453
- function resolveWorktreeRoot(cwd) {
454
- // If the current directory already has its own .planning/, respect it.
455
- // This handles linked worktrees with independent planning state (e.g., Conductor workspaces).
456
- if (fs.existsSync(path.join(cwd, '.planning'))) {
457
- return cwd;
458
- }
459
-
460
- // Check if we're in a linked worktree
461
- const gitDir = execGit(cwd, ['rev-parse', '--git-dir']);
462
- const commonDir = execGit(cwd, ['rev-parse', '--git-common-dir']);
463
-
464
- if (gitDir.exitCode !== 0 || commonDir.exitCode !== 0) return cwd;
465
-
466
- // In a linked worktree, .git is a file pointing to .git/worktrees/<name>
467
- // and git-common-dir points to the main repo's .git directory
468
- const gitDirResolved = path.resolve(cwd, gitDir.stdout);
469
- const commonDirResolved = path.resolve(cwd, commonDir.stdout);
470
-
471
- if (gitDirResolved !== commonDirResolved) {
472
- // We're in a linked worktree — resolve main worktree root
473
- // The common dir is the main repo's .git, so its parent is the main worktree root
474
- return path.dirname(commonDirResolved);
475
- }
476
-
477
- return cwd;
478
- }
479
-
480
- /**
481
- * Acquire a file-based lock for .planning/ writes.
482
- * Prevents concurrent worktrees from corrupting shared planning files.
483
- * Lock is auto-released after the callback completes.
484
- */
485
- function withPlanningLock(cwd, fn) {
486
- const lockPath = path.join(planningDir(cwd), '.lock');
487
- const lockTimeout = 10000; // 10 seconds
488
- const retryDelay = 100;
489
- const start = Date.now();
490
-
491
- // Ensure .planning/ exists
492
- try { fs.mkdirSync(planningDir(cwd), { recursive: true }); } catch { /* ok */ }
493
-
494
- while (Date.now() - start < lockTimeout) {
495
- try {
496
- // Atomic create — fails if file exists
497
- fs.writeFileSync(lockPath, JSON.stringify({
498
- pid: process.pid,
499
- cwd,
500
- acquired: new Date().toISOString(),
501
- }), { flag: 'wx' });
502
-
503
- // Lock acquired — run the function
504
- try {
505
- return fn();
506
- } finally {
507
- try { fs.unlinkSync(lockPath); } catch { /* already released */ }
508
- }
509
- } catch (err) {
510
- if (err.code === 'EEXIST') {
511
- // Lock exists — check if stale (>30s old)
512
- try {
513
- const stat = fs.statSync(lockPath);
514
- if (Date.now() - stat.mtimeMs > 30000) {
515
- fs.unlinkSync(lockPath);
516
- continue; // retry
517
- }
518
- } catch { continue; }
519
-
520
- // Wait and retry
521
- spawnSync('sleep', ['0.1'], { stdio: 'ignore' });
522
- continue;
523
- }
524
- throw err;
525
- }
526
- }
527
- // Timeout — force acquire (stale lock recovery)
528
- try { fs.unlinkSync(lockPath); } catch { /* ok */ }
529
- return fn();
530
- }
531
-
532
- /**
533
- * Get the .planning directory path, workstream-aware.
534
- * When a workstream is active (via explicit ws arg or MindForge_WORKSTREAM env var),
535
- * returns `.planning/workstreams/{ws}/`. Otherwise returns `.planning/`.
536
- *
537
- * @param {string} cwd - project root
538
- * @param {string} [ws] - explicit workstream name; if omitted, checks MindForge_WORKSTREAM env var
539
- */
540
- function planningDir(cwd, ws) {
541
- if (ws === undefined) ws = process.env.MindForge_WORKSTREAM || null;
542
- if (!ws) return path.join(cwd, '.planning');
543
- return path.join(cwd, '.planning', 'workstreams', ws);
544
- }
545
-
546
- /** Always returns the root .planning/ path, ignoring workstreams. For shared resources. */
547
- function planningRoot(cwd) {
548
- return path.join(cwd, '.planning');
549
- }
550
-
551
- /**
552
- * Get common .planning file paths, workstream-aware.
553
- * Scoped paths (state, roadmap, phases, requirements) resolve to the active workstream.
554
- * Shared paths (project, config) always resolve to the root .planning/.
555
- */
556
- function planningPaths(cwd, ws) {
557
- const base = planningDir(cwd, ws);
558
- const root = path.join(cwd, '.planning');
559
- return {
560
- planning: base,
561
- state: path.join(base, 'STATE.md'),
562
- roadmap: path.join(base, 'ROADMAP.md'),
563
- project: path.join(root, 'PROJECT.md'),
564
- config: path.join(root, 'config.json'),
565
- phases: path.join(base, 'phases'),
566
- requirements: path.join(base, 'REQUIREMENTS.md'),
567
- };
568
- }
569
-
570
- // ─── Active Workstream Detection ─────────────────────────────────────────────
571
-
572
- /**
573
- * Get the active workstream name from .planning/active-workstream file.
574
- * Returns null if no active workstream or file doesn't exist.
575
- */
576
- function getActiveWorkstream(cwd) {
577
- const filePath = path.join(planningRoot(cwd), 'active-workstream');
578
- try {
579
- const name = fs.readFileSync(filePath, 'utf-8').trim();
580
- if (!name || !/^[a-zA-Z0-9_-]+$/.test(name)) return null;
581
- const wsDir = path.join(planningRoot(cwd), 'workstreams', name);
582
- if (!fs.existsSync(wsDir)) return null;
583
- return name;
584
- } catch {
585
- return null;
586
- }
587
- }
588
-
589
- /**
590
- * Set the active workstream. Pass null to clear.
591
- */
592
- function setActiveWorkstream(cwd, name) {
593
- const filePath = path.join(planningRoot(cwd), 'active-workstream');
594
- if (!name) {
595
- try { fs.unlinkSync(filePath); } catch {}
596
- return;
597
- }
598
- if (!/^[a-zA-Z0-9_-]+$/.test(name)) {
599
- throw new Error('Invalid workstream name: must be alphanumeric, hyphens, and underscores only');
600
- }
601
- fs.writeFileSync(filePath, name + '\n', 'utf-8');
602
- }
603
-
604
- // ─── Phase utilities ──────────────────────────────────────────────────────────
605
-
606
- function escapeRegex(value) {
607
- return String(value).replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
608
- }
609
-
610
- function normalizePhaseName(phase) {
611
- const str = String(phase);
612
- // Standard numeric phases: 1, 01, 12A, 12.1
613
- const match = str.match(/^(\d+)([A-Z])?((?:\.\d+)*)/i);
614
- if (match) {
615
- const padded = match[1].padStart(2, '0');
616
- const letter = match[2] ? match[2].toUpperCase() : '';
617
- const decimal = match[3] || '';
618
- return padded + letter + decimal;
619
- }
620
- // Custom phase IDs (e.g. PROJ-42, AUTH-101): return as-is
621
- return str;
622
- }
623
-
624
- function comparePhaseNum(a, b) {
625
- const pa = String(a).match(/^(\d+)([A-Z])?((?:\.\d+)*)/i);
626
- const pb = String(b).match(/^(\d+)([A-Z])?((?:\.\d+)*)/i);
627
- // If either is non-numeric (custom ID), fall back to string comparison
628
- if (!pa || !pb) return String(a).localeCompare(String(b));
629
- const intDiff = parseInt(pa[1], 10) - parseInt(pb[1], 10);
630
- if (intDiff !== 0) return intDiff;
631
- // No letter sorts before letter: 12 < 12A < 12B
632
- const la = (pa[2] || '').toUpperCase();
633
- const lb = (pb[2] || '').toUpperCase();
634
- if (la !== lb) {
635
- if (!la) return -1;
636
- if (!lb) return 1;
637
- return la < lb ? -1 : 1;
638
- }
639
- // Segment-by-segment decimal comparison: 12A < 12A.1 < 12A.1.2 < 12A.2
640
- const aDecParts = pa[3] ? pa[3].slice(1).split('.').map(p => parseInt(p, 10)) : [];
641
- const bDecParts = pb[3] ? pb[3].slice(1).split('.').map(p => parseInt(p, 10)) : [];
642
- const maxLen = Math.max(aDecParts.length, bDecParts.length);
643
- if (aDecParts.length === 0 && bDecParts.length > 0) return -1;
644
- if (bDecParts.length === 0 && aDecParts.length > 0) return 1;
645
- for (let i = 0; i < maxLen; i++) {
646
- const av = Number.isFinite(aDecParts[i]) ? aDecParts[i] : 0;
647
- const bv = Number.isFinite(bDecParts[i]) ? bDecParts[i] : 0;
648
- if (av !== bv) return av - bv;
649
- }
650
- return 0;
651
- }
652
-
653
- function searchPhaseInDir(baseDir, relBase, normalized) {
654
- try {
655
- const dirs = readSubdirectories(baseDir, true);
656
- // Match: starts with normalized (numeric) OR contains normalized as prefix segment (custom ID)
657
- const match = dirs.find(d => {
658
- if (d.startsWith(normalized)) return true;
659
- // For custom IDs like PROJ-42, match case-insensitively
660
- if (d.toUpperCase().startsWith(normalized.toUpperCase())) return true;
661
- return false;
662
- });
663
- if (!match) return null;
664
-
665
- // Extract phase number and name — supports both numeric (01-name) and custom (PROJ-42-name)
666
- const dirMatch = match.match(/^(\d+[A-Z]?(?:\.\d+)*)-?(.*)/i)
667
- || match.match(/^([A-Z][A-Z0-9]*(?:-[A-Z0-9]+)*)-(.+)/i)
668
- || [null, match, null];
669
- const phaseNumber = dirMatch ? dirMatch[1] : normalized;
670
- const phaseName = dirMatch && dirMatch[2] ? dirMatch[2] : null;
671
- const phaseDir = path.join(baseDir, match);
672
- const { plans: unsortedPlans, summaries: unsortedSummaries, hasResearch, hasContext, hasVerification, hasReviews } = getPhaseFileStats(phaseDir);
673
- const plans = unsortedPlans.sort();
674
- const summaries = unsortedSummaries.sort();
675
-
676
- const completedPlanIds = new Set(
677
- summaries.map(s => s.replace('-SUMMARY.md', '').replace('SUMMARY.md', ''))
678
- );
679
- const incompletePlans = plans.filter(p => {
680
- const planId = p.replace('-PLAN.md', '').replace('PLAN.md', '');
681
- return !completedPlanIds.has(planId);
682
- });
683
-
684
- return {
685
- found: true,
686
- directory: toPosixPath(path.join(relBase, match)),
687
- phase_number: phaseNumber,
688
- phase_name: phaseName,
689
- phase_slug: phaseName ? phaseName.toLowerCase().replace(/[^a-z0-9]+/g, '-').replace(/^-+|-+$/g, '') : null,
690
- plans,
691
- summaries,
692
- incomplete_plans: incompletePlans,
693
- has_research: hasResearch,
694
- has_context: hasContext,
695
- has_verification: hasVerification,
696
- has_reviews: hasReviews,
697
- };
698
- } catch {
699
- return null;
700
- }
701
- }
702
-
703
- function findPhaseInternal(cwd, phase) {
704
- if (!phase) return null;
705
-
706
- const phasesDir = path.join(planningDir(cwd), 'phases');
707
- const normalized = normalizePhaseName(phase);
708
-
709
- // Search current phases first
710
- const relPhasesDir = toPosixPath(path.relative(cwd, phasesDir));
711
- const current = searchPhaseInDir(phasesDir, relPhasesDir, normalized);
712
- if (current) return current;
713
-
714
- // Search archived milestone phases (newest first)
715
- const milestonesDir = path.join(cwd, '.planning', 'milestones');
716
- if (!fs.existsSync(milestonesDir)) return null;
717
-
718
- try {
719
- const milestoneEntries = fs.readdirSync(milestonesDir, { withFileTypes: true });
720
- const archiveDirs = milestoneEntries
721
- .filter(e => e.isDirectory() && /^v[\d.]+-phases$/.test(e.name))
722
- .map(e => e.name)
723
- .sort()
724
- .reverse();
725
-
726
- for (const archiveName of archiveDirs) {
727
- const version = archiveName.match(/^(v[\d.]+)-phases$/)[1];
728
- const archivePath = path.join(milestonesDir, archiveName);
729
- const relBase = '.planning/milestones/' + archiveName;
730
- const result = searchPhaseInDir(archivePath, relBase, normalized);
731
- if (result) {
732
- result.archived = version;
733
- return result;
734
- }
735
- }
736
- } catch { /* intentionally empty */ }
737
-
738
- return null;
739
- }
740
-
741
- function getArchivedPhaseDirs(cwd) {
742
- const milestonesDir = path.join(cwd, '.planning', 'milestones');
743
- const results = [];
744
-
745
- if (!fs.existsSync(milestonesDir)) return results;
746
-
747
- try {
748
- const milestoneEntries = fs.readdirSync(milestonesDir, { withFileTypes: true });
749
- // Find v*-phases directories, sort newest first
750
- const phaseDirs = milestoneEntries
751
- .filter(e => e.isDirectory() && /^v[\d.]+-phases$/.test(e.name))
752
- .map(e => e.name)
753
- .sort()
754
- .reverse();
755
-
756
- for (const archiveName of phaseDirs) {
757
- const version = archiveName.match(/^(v[\d.]+)-phases$/)[1];
758
- const archivePath = path.join(milestonesDir, archiveName);
759
- const dirs = readSubdirectories(archivePath, true);
760
-
761
- for (const dir of dirs) {
762
- results.push({
763
- name: dir,
764
- milestone: version,
765
- basePath: path.join('.planning', 'milestones', archiveName),
766
- fullPath: path.join(archivePath, dir),
767
- });
768
- }
769
- }
770
- } catch { /* intentionally empty */ }
771
-
772
- return results;
773
- }
774
-
775
- // ─── Roadmap milestone scoping ───────────────────────────────────────────────
776
-
777
- /**
778
- * Strip shipped milestone content wrapped in <details> blocks.
779
- * Used to isolate current milestone phases when searching ROADMAP.md
780
- * for phase headings or checkboxes — prevents matching archived milestone
781
- * phases that share the same numbers as current milestone phases.
782
- */
783
- function stripShippedMilestones(content) {
784
- return content.replace(/<details>[\s\S]*?<\/details>/gi, '');
785
- }
786
-
787
- /**
788
- * Extract the current milestone section from ROADMAP.md by positive lookup.
789
- *
790
- * Instead of stripping <details> blocks (negative heuristic that breaks if
791
- * agents wrap the current milestone in <details>), this finds the section
792
- * matching the current milestone version and returns only that content.
793
- *
794
- * Falls back to stripShippedMilestones() if:
795
- * - cwd is not provided
796
- * - STATE.md doesn't exist or has no milestone field
797
- * - Version can't be found in ROADMAP.md
798
- *
799
- * @param {string} content - Full ROADMAP.md content
800
- * @param {string} [cwd] - Working directory for reading STATE.md
801
- * @returns {string} Content scoped to current milestone
802
- */
803
- function extractCurrentMilestone(content, cwd) {
804
- if (!cwd) return stripShippedMilestones(content);
805
-
806
- // 1. Get current milestone version from STATE.md frontmatter
807
- let version = null;
808
- try {
809
- const statePath = path.join(planningDir(cwd), 'STATE.md');
810
- if (fs.existsSync(statePath)) {
811
- const stateRaw = fs.readFileSync(statePath, 'utf-8');
812
- const milestoneMatch = stateRaw.match(/^milestone:\s*(.+)/m);
813
- if (milestoneMatch) {
814
- version = milestoneMatch[1].trim();
815
- }
816
- }
817
- } catch {}
818
-
819
- // 2. Fallback: derive version from getMilestoneInfo pattern in ROADMAP.md itself
820
- if (!version) {
821
- // Check for 🚧 in-progress marker
822
- const inProgressMatch = content.match(/🚧\s*\*\*v(\d+\.\d+)\s/);
823
- if (inProgressMatch) {
824
- version = 'v' + inProgressMatch[1];
825
- }
826
- }
827
-
828
- if (!version) return stripShippedMilestones(content);
829
-
830
- // 3. Find the section matching this version
831
- // Match headings like: ## Roadmap v3.0: Name, ## v3.0 Name, etc.
832
- const escapedVersion = escapeRegex(version);
833
- const sectionPattern = new RegExp(
834
- `(^#{1,3}\\s+.*${escapedVersion}[^\\n]*)`,
835
- 'mi'
836
- );
837
- const sectionMatch = content.match(sectionPattern);
838
-
839
- if (!sectionMatch) return stripShippedMilestones(content);
840
-
841
- const sectionStart = sectionMatch.index;
842
-
843
- // Find the end: next milestone heading at same or higher level, or EOF
844
- // Milestone headings look like: ## v2.0, ## Roadmap v2.0, ## ✅ v1.0, etc.
845
- const headingLevel = sectionMatch[1].match(/^(#{1,3})\s/)[1].length;
846
- const restContent = content.slice(sectionStart + sectionMatch[0].length);
847
- const nextMilestonePattern = new RegExp(
848
- `^#{1,${headingLevel}}\\s+(?:.*v\\d+\\.\\d+|✅|📋|🚧)`,
849
- 'mi'
850
- );
851
- const nextMatch = restContent.match(nextMilestonePattern);
852
-
853
- let sectionEnd;
854
- if (nextMatch) {
855
- sectionEnd = sectionStart + sectionMatch[0].length + nextMatch.index;
856
- } else {
857
- sectionEnd = content.length;
858
- }
859
-
860
- // Return everything before the current milestone section (non-milestone content
861
- // like title, overview) plus the current milestone section
862
- const beforeMilestones = content.slice(0, sectionStart);
863
- const currentSection = content.slice(sectionStart, sectionEnd);
864
-
865
- // Also include any content before the first milestone heading (title, overview, etc.)
866
- // but strip any <details> blocks in it (these are definitely shipped)
867
- const preamble = beforeMilestones.replace(/<details>[\s\S]*?<\/details>/gi, '');
868
-
869
- return preamble + currentSection;
870
- }
871
-
872
- /**
873
- * Replace a pattern only in the current milestone section of ROADMAP.md
874
- * (everything after the last </details> close tag). Used for write operations
875
- * that must not accidentally modify archived milestone checkboxes/tables.
876
- */
877
- function replaceInCurrentMilestone(content, pattern, replacement) {
878
- const lastDetailsClose = content.lastIndexOf('</details>');
879
- if (lastDetailsClose === -1) {
880
- return content.replace(pattern, replacement);
881
- }
882
- const offset = lastDetailsClose + '</details>'.length;
883
- const before = content.slice(0, offset);
884
- const after = content.slice(offset);
885
- return before + after.replace(pattern, replacement);
886
- }
887
-
888
- // ─── Roadmap & model utilities ────────────────────────────────────────────────
889
-
890
- function getRoadmapPhaseInternal(cwd, phaseNum) {
891
- if (!phaseNum) return null;
892
- const roadmapPath = path.join(planningDir(cwd), 'ROADMAP.md');
893
- if (!fs.existsSync(roadmapPath)) return null;
894
-
895
- try {
896
- const content = extractCurrentMilestone(fs.readFileSync(roadmapPath, 'utf-8'), cwd);
897
- const escapedPhase = escapeRegex(phaseNum.toString());
898
- // Match both numeric (Phase 1:) and custom (Phase PROJ-42:) headers
899
- const phasePattern = new RegExp(`#{2,4}\\s*Phase\\s+${escapedPhase}:\\s*([^\\n]+)`, 'i');
900
- const headerMatch = content.match(phasePattern);
901
- if (!headerMatch) return null;
902
-
903
- const phaseName = headerMatch[1].trim();
904
- const headerIndex = headerMatch.index;
905
- const restOfContent = content.slice(headerIndex);
906
- const nextHeaderMatch = restOfContent.match(/\n#{2,4}\s+Phase\s+[\w]/i);
907
- const sectionEnd = nextHeaderMatch ? headerIndex + nextHeaderMatch.index : content.length;
908
- const section = content.slice(headerIndex, sectionEnd).trim();
909
-
910
- const goalMatch = section.match(/\*\*Goal(?:\*\*:|\*?\*?:\*\*)\s*([^\n]+)/i);
911
- const goal = goalMatch ? goalMatch[1].trim() : null;
912
-
913
- return {
914
- found: true,
915
- phase_number: phaseNum.toString(),
916
- phase_name: phaseName,
917
- goal,
918
- section,
919
- };
920
- } catch {
921
- return null;
922
- }
923
- }
924
-
925
- // ─── Model alias resolution ───────────────────────────────────────────────────
926
-
927
- /**
928
- * Map short model aliases to full model IDs.
929
- * Updated each release to match current model versions.
930
- * Users can override with model_overrides in config.json for custom/latest models.
931
- */
932
- const MODEL_ALIAS_MAP = {
933
- 'opus': 'claude-opus-4-0',
934
- 'sonnet': 'claude-sonnet-4-5',
935
- 'haiku': 'claude-haiku-3-5',
936
- };
937
-
938
- function resolveModelInternal(cwd, agentType) {
939
- const config = loadConfig(cwd);
940
-
941
- // Check per-agent override first — always respected regardless of resolve_model_ids.
942
- // Users who set fully-qualified model IDs (e.g., "openai/gpt-5.4") get exactly that.
943
- const override = config.model_overrides?.[agentType];
944
- if (override) {
945
- return override;
946
- }
947
-
948
- // resolve_model_ids: "omit" — return empty string so the runtime uses its configured
949
- // default model. For non-the agent runtimes (OpenCode, Codex, etc.) that don't recognize
950
- // the agent aliases (opus/sonnet/haiku/inherit). Set automatically during install. See #1156.
951
- if (config.resolve_model_ids === 'omit') {
952
- return '';
953
- }
954
-
955
- // Fall back to profile lookup
956
- const profile = String(config.model_profile || 'balanced').toLowerCase();
957
- const agentModels = MODEL_PROFILES[agentType];
958
- if (!agentModels) return 'sonnet';
959
- if (profile === 'inherit') return 'inherit';
960
- const alias = agentModels[profile] || agentModels['balanced'] || 'sonnet';
961
-
962
- // resolve_model_ids: true — map alias to full the agent model ID
963
- // Prevents 404s when the Task tool passes aliases directly to the API
964
- if (config.resolve_model_ids) {
965
- return MODEL_ALIAS_MAP[alias] || alias;
966
- }
967
-
968
- return alias;
969
- }
970
-
971
- // ─── Summary body helpers ─────────────────────────────────────────────────
972
-
973
- /**
974
- * Extract a one-liner from the summary body when it's not in frontmatter.
975
- * The summary template defines one-liner as a bold markdown line after the heading:
976
- * # Phase X: Name Summary
977
- * **[substantive one-liner text]**
978
- */
979
- function extractOneLinerFromBody(content) {
980
- if (!content) return null;
981
- // Strip frontmatter first
982
- const body = content.replace(/^---\n[\s\S]*?\n---\n*/, '');
983
- // Find the first **...** line after a # heading
984
- const match = body.match(/^#[^\n]*\n+\*\*([^*]+)\*\*/m);
985
- return match ? match[1].trim() : null;
986
- }
987
-
988
- // ─── Misc utilities ───────────────────────────────────────────────────────────
989
-
990
- function pathExistsInternal(cwd, targetPath) {
991
- const fullPath = path.isAbsolute(targetPath) ? targetPath : path.join(cwd, targetPath);
992
- try {
993
- fs.statSync(fullPath);
994
- return true;
995
- } catch {
996
- return false;
997
- }
998
- }
999
-
1000
- function generateSlugInternal(text) {
1001
- if (!text) return null;
1002
- return text.toLowerCase().replace(/[^a-z0-9]+/g, '-').replace(/^-+|-+$/g, '');
1003
- }
1004
-
1005
- function getMilestoneInfo(cwd) {
1006
- try {
1007
- const roadmap = fs.readFileSync(path.join(planningDir(cwd), 'ROADMAP.md'), 'utf-8');
1008
-
1009
- // First: check for list-format roadmaps using 🚧 (in-progress) marker
1010
- // e.g. "- 🚧 **v2.1 Belgium** — Phases 24-28 (in progress)"
1011
- // e.g. "- 🚧 **v1.2.1 Tech Debt** — Phases 1-8 (in progress)"
1012
- const inProgressMatch = roadmap.match(/🚧\s*\*\*v(\d+(?:\.\d+)+)\s+([^*]+)\*\*/);
1013
- if (inProgressMatch) {
1014
- return {
1015
- version: 'v' + inProgressMatch[1],
1016
- name: inProgressMatch[2].trim(),
1017
- };
1018
- }
1019
-
1020
- // Second: heading-format roadmaps — strip shipped milestones in <details> blocks
1021
- const cleaned = stripShippedMilestones(roadmap);
1022
- // Extract version and name from the same ## heading for consistency
1023
- // Supports 2+ segment versions: v1.2, v1.2.1, v2.0.1, etc.
1024
- const headingMatch = cleaned.match(/## .*v(\d+(?:\.\d+)+)[:\s]+([^\n(]+)/);
1025
- if (headingMatch) {
1026
- return {
1027
- version: 'v' + headingMatch[1],
1028
- name: headingMatch[2].trim(),
1029
- };
1030
- }
1031
- // Fallback: try bare version match (greedy — capture longest version string)
1032
- const versionMatch = cleaned.match(/v(\d+(?:\.\d+)+)/);
1033
- return {
1034
- version: versionMatch ? versionMatch[0] : 'v1.0',
1035
- name: 'milestone',
1036
- };
1037
- } catch {
1038
- return { version: 'v1.0', name: 'milestone' };
1039
- }
1040
- }
1041
-
1042
- /**
1043
- * Returns a filter function that checks whether a phase directory belongs
1044
- * to the current milestone based on ROADMAP.md phase headings.
1045
- * If no ROADMAP exists or no phases are listed, returns a pass-all filter.
1046
- */
1047
- function getMilestonePhaseFilter(cwd) {
1048
- const milestonePhaseNums = new Set();
1049
- try {
1050
- const roadmap = extractCurrentMilestone(fs.readFileSync(path.join(planningDir(cwd), 'ROADMAP.md'), 'utf-8'), cwd);
1051
- // Match both numeric phases (Phase 1:) and custom IDs (Phase PROJ-42:)
1052
- const phasePattern = /#{2,4}\s*Phase\s+([\w][\w.-]*)\s*:/gi;
1053
- let m;
1054
- while ((m = phasePattern.exec(roadmap)) !== null) {
1055
- milestonePhaseNums.add(m[1]);
1056
- }
1057
- } catch { /* intentionally empty */ }
1058
-
1059
- if (milestonePhaseNums.size === 0) {
1060
- const passAll = () => true;
1061
- passAll.phaseCount = 0;
1062
- return passAll;
1063
- }
1064
-
1065
- const normalized = new Set(
1066
- [...milestonePhaseNums].map(n => (n.replace(/^0+/, '') || '0').toLowerCase())
1067
- );
1068
-
1069
- function isDirInMilestone(dirName) {
1070
- // Try numeric match first
1071
- const m = dirName.match(/^0*(\d+[A-Za-z]?(?:\.\d+)*)/);
1072
- if (m && normalized.has(m[1].toLowerCase())) return true;
1073
- // Try custom ID match (e.g. PROJ-42-description → PROJ-42)
1074
- const customMatch = dirName.match(/^([A-Za-z][A-Za-z0-9]*(?:-[A-Za-z0-9]+)*)/);
1075
- if (customMatch && normalized.has(customMatch[1].toLowerCase())) return true;
1076
- return false;
1077
- }
1078
- isDirInMilestone.phaseCount = milestonePhaseNums.size;
1079
- return isDirInMilestone;
1080
- }
1081
-
1082
- // ─── Phase file helpers ──────────────────────────────────────────────────────
1083
-
1084
- /** Filter a file list to just PLAN.md / *-PLAN.md entries. */
1085
- function filterPlanFiles(files) {
1086
- return files.filter(f => f.endsWith('-PLAN.md') || f === 'PLAN.md');
1087
- }
1088
-
1089
- /** Filter a file list to just SUMMARY.md / *-SUMMARY.md entries. */
1090
- function filterSummaryFiles(files) {
1091
- return files.filter(f => f.endsWith('-SUMMARY.md') || f === 'SUMMARY.md');
1092
- }
1093
-
1094
- /**
1095
- * Read a phase directory and return counts/flags for common file types.
1096
- * Returns an object with plans[], summaries[], and boolean flags for
1097
- * research/context/verification files.
1098
- */
1099
- function getPhaseFileStats(phaseDir) {
1100
- const files = fs.readdirSync(phaseDir);
1101
- return {
1102
- plans: filterPlanFiles(files),
1103
- summaries: filterSummaryFiles(files),
1104
- hasResearch: files.some(f => f.endsWith('-RESEARCH.md') || f === 'RESEARCH.md'),
1105
- hasContext: files.some(f => f.endsWith('-CONTEXT.md') || f === 'CONTEXT.md'),
1106
- hasVerification: files.some(f => f.endsWith('-VERIFICATION.md') || f === 'VERIFICATION.md'),
1107
- hasReviews: files.some(f => f.endsWith('-REVIEWS.md') || f === 'REVIEWS.md'),
1108
- };
1109
- }
1110
-
1111
- /**
1112
- * Read immediate child directories from a path.
1113
- * Returns [] if the path doesn't exist or can't be read.
1114
- * Pass sort=true to apply comparePhaseNum ordering.
1115
- */
1116
- function readSubdirectories(dirPath, sort = false) {
1117
- try {
1118
- const entries = fs.readdirSync(dirPath, { withFileTypes: true });
1119
- const dirs = entries.filter(e => e.isDirectory()).map(e => e.name);
1120
- return sort ? dirs.sort((a, b) => comparePhaseNum(a, b)) : dirs;
1121
- } catch {
1122
- return [];
1123
- }
1124
- }
1125
-
1126
- module.exports = {
1127
- output,
1128
- error,
1129
- safeReadFile,
1130
- loadConfig,
1131
- isGitIgnored,
1132
- execGit,
1133
- normalizeMd,
1134
- escapeRegex,
1135
- normalizePhaseName,
1136
- comparePhaseNum,
1137
- searchPhaseInDir,
1138
- findPhaseInternal,
1139
- getArchivedPhaseDirs,
1140
- getRoadmapPhaseInternal,
1141
- resolveModelInternal,
1142
- pathExistsInternal,
1143
- generateSlugInternal,
1144
- getMilestoneInfo,
1145
- getMilestonePhaseFilter,
1146
- stripShippedMilestones,
1147
- extractCurrentMilestone,
1148
- replaceInCurrentMilestone,
1149
- toPosixPath,
1150
- extractOneLinerFromBody,
1151
- resolveWorktreeRoot,
1152
- withPlanningLock,
1153
- findProjectRoot,
1154
- detectSubRepos,
1155
- reapStaleTempFiles,
1156
- MODEL_ALIAS_MAP,
1157
- planningDir,
1158
- planningRoot,
1159
- planningPaths,
1160
- getActiveWorkstream,
1161
- setActiveWorkstream,
1162
- filterPlanFiles,
1163
- filterSummaryFiles,
1164
- getPhaseFileStats,
1165
- readSubdirectories,
1166
- };