agileflow 3.4.3 → 4.0.0-alpha.2

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 (762) hide show
  1. package/CHANGELOG.md +238 -473
  2. package/README.md +22 -114
  3. package/bin/agileflow.js +15 -0
  4. package/bin/hooks/pre-bash.js +35 -0
  5. package/bin/hooks/pre-compact.js +34 -0
  6. package/bin/hooks/pre-edit.js +32 -0
  7. package/bin/hooks/pre-write.js +32 -0
  8. package/bin/hooks/session-start.js +42 -0
  9. package/bin/hooks/stop.js +34 -0
  10. package/content/plugins/ads/plugin.yaml +14 -0
  11. package/content/plugins/audit/plugin.yaml +14 -0
  12. package/content/plugins/core/hooks/babysit-mentor-injector.js +55 -0
  13. package/content/plugins/core/hooks/context-loader.js +169 -0
  14. package/content/plugins/core/hooks/damage-control-bash.js +78 -0
  15. package/content/plugins/core/hooks/damage-control-edit.js +76 -0
  16. package/content/plugins/core/hooks/damage-control-patterns.yaml +100 -0
  17. package/content/plugins/core/hooks/damage-control-write.js +72 -0
  18. package/content/plugins/core/hooks/pre-compact-state.js +90 -0
  19. package/content/plugins/core/hooks/session-welcome.js +19 -0
  20. package/content/plugins/core/plugin.yaml +82 -0
  21. package/content/plugins/core/skills/agileflow-adr/SKILL.md +179 -0
  22. package/content/plugins/core/skills/agileflow-babysit-mentor/SKILL.md +144 -0
  23. package/content/plugins/core/skills/agileflow-epic-planner/SKILL.md +179 -0
  24. package/content/plugins/core/skills/agileflow-status-updater/SKILL.md +132 -0
  25. package/content/plugins/core/skills/agileflow-story-writer/SKILL.md +200 -0
  26. package/content/plugins/council/plugin.yaml +14 -0
  27. package/content/plugins/seo/plugin.yaml +14 -0
  28. package/package.json +29 -49
  29. package/src/cli/commands/doctor.js +159 -0
  30. package/src/cli/commands/hook.js +80 -0
  31. package/src/cli/commands/setup.js +292 -0
  32. package/src/cli/commands/status.js +47 -0
  33. package/src/cli/commands/update.js +83 -0
  34. package/src/cli/index.js +73 -0
  35. package/src/cli/wizard/behaviors-picker.js +108 -0
  36. package/src/cli/wizard/ide-picker.js +57 -0
  37. package/src/cli/wizard/personalization.js +64 -0
  38. package/src/cli/wizard/plugin-picker.js +106 -0
  39. package/src/lib/hash.js +41 -0
  40. package/src/runtime/config/defaults.js +61 -0
  41. package/src/runtime/config/loader.js +117 -0
  42. package/src/runtime/config/schema.json +99 -0
  43. package/src/runtime/config/writer.js +55 -0
  44. package/src/runtime/hooks/aggregator.js +157 -0
  45. package/src/runtime/hooks/chain.js +93 -0
  46. package/src/runtime/hooks/logger.js +68 -0
  47. package/src/runtime/hooks/manifest-loader.js +228 -0
  48. package/src/runtime/hooks/orchestrator.js +322 -0
  49. package/src/runtime/ide/capabilities.js +111 -0
  50. package/src/runtime/ide/claude-code-settings.js +234 -0
  51. package/src/runtime/ide/claude-code-skills.js +202 -0
  52. package/src/runtime/installer/file-index.js +112 -0
  53. package/src/runtime/installer/install.js +329 -0
  54. package/src/runtime/installer/stash.js +61 -0
  55. package/src/runtime/installer/sync-engine.js +205 -0
  56. package/src/runtime/plugins/registry.js +132 -0
  57. package/src/runtime/plugins/resolver.js +138 -0
  58. package/src/runtime/plugins/validator.js +196 -0
  59. package/src/runtime/skills/validator.js +335 -0
  60. package/lib/README.md +0 -178
  61. package/lib/api-routes.js +0 -625
  62. package/lib/api-server.js +0 -278
  63. package/lib/cache-provider.js +0 -155
  64. package/lib/codebase-indexer.js +0 -819
  65. package/lib/colors.generated.js +0 -117
  66. package/lib/colors.js +0 -341
  67. package/lib/consent.js +0 -232
  68. package/lib/content-sanitizer.js +0 -464
  69. package/lib/correlation.js +0 -277
  70. package/lib/drivers/claude-driver.ts +0 -312
  71. package/lib/drivers/codex-driver.ts +0 -464
  72. package/lib/drivers/driver-manager.ts +0 -159
  73. package/lib/drivers/gemini-driver.ts +0 -498
  74. package/lib/drivers/index.ts +0 -17
  75. package/lib/error-codes.js +0 -590
  76. package/lib/errors.js +0 -670
  77. package/lib/feature-flags.js +0 -171
  78. package/lib/feedback.js +0 -595
  79. package/lib/file-cache.js +0 -541
  80. package/lib/flag-detection.js +0 -344
  81. package/lib/format-error.js +0 -156
  82. package/lib/gate-runner.js +0 -282
  83. package/lib/generator-factory.js +0 -333
  84. package/lib/git-operations.js +0 -266
  85. package/lib/lazy-require.js +0 -59
  86. package/lib/lock-file.js +0 -144
  87. package/lib/logger.js +0 -106
  88. package/lib/merge-operations.js +0 -1006
  89. package/lib/path-resolver.js +0 -544
  90. package/lib/path-utils.js +0 -49
  91. package/lib/paths.js +0 -291
  92. package/lib/placeholder-registry.js +0 -822
  93. package/lib/process-executor.js +0 -214
  94. package/lib/progress.js +0 -334
  95. package/lib/protocol/driver.ts +0 -354
  96. package/lib/protocol/index.ts +0 -12
  97. package/lib/protocol/ir.ts +0 -271
  98. package/lib/registry-cache.js +0 -80
  99. package/lib/registry-di.js +0 -358
  100. package/lib/result-schema.js +0 -363
  101. package/lib/result.js +0 -210
  102. package/lib/session-display.js +0 -331
  103. package/lib/session-operations.js +0 -611
  104. package/lib/session-registry.js +0 -484
  105. package/lib/session-state-machine.js +0 -465
  106. package/lib/session-switching.js +0 -191
  107. package/lib/skill-loader.js +0 -213
  108. package/lib/smart-json-file.js +0 -682
  109. package/lib/state-machine.js +0 -286
  110. package/lib/table-formatter.js +0 -519
  111. package/lib/template-loader.js +0 -143
  112. package/lib/transient-status.js +0 -374
  113. package/lib/ui-manager.js +0 -612
  114. package/lib/validate-args.js +0 -213
  115. package/lib/validate-commands.js +0 -308
  116. package/lib/validate-names.js +0 -143
  117. package/lib/validate-paths.js +0 -434
  118. package/lib/validate.js +0 -134
  119. package/lib/worktree-operations.js +0 -201
  120. package/lib/yaml-utils.js +0 -164
  121. package/scripts/README.md +0 -267
  122. package/scripts/af +0 -34
  123. package/scripts/agent-loop.js +0 -879
  124. package/scripts/agileflow-configure.js +0 -368
  125. package/scripts/agileflow-statusline.sh +0 -857
  126. package/scripts/agileflow-welcome.js +0 -2246
  127. package/scripts/api-server-runner.js +0 -177
  128. package/scripts/archive-completed-stories.sh +0 -308
  129. package/scripts/auto-self-improve.js +0 -326
  130. package/scripts/automation-run-due.js +0 -128
  131. package/scripts/babysit-clear-restore.js +0 -154
  132. package/scripts/babysit-context-restore.js +0 -89
  133. package/scripts/backfill-ideation-status.js +0 -128
  134. package/scripts/batch-pmap-loop.js +0 -551
  135. package/scripts/check-sessions.js +0 -116
  136. package/scripts/check-update.js +0 -282
  137. package/scripts/ci-summary.js +0 -294
  138. package/scripts/claude-smart.sh +0 -85
  139. package/scripts/claude-tmux.sh +0 -737
  140. package/scripts/claude-watchdog.sh +0 -225
  141. package/scripts/clear-active-command.js +0 -48
  142. package/scripts/compress-status.sh +0 -116
  143. package/scripts/context-loader.js +0 -310
  144. package/scripts/damage-control/bash-tool-damage-control.js +0 -22
  145. package/scripts/damage-control/edit-tool-damage-control.js +0 -19
  146. package/scripts/damage-control/patterns.yaml +0 -227
  147. package/scripts/damage-control/write-tool-damage-control.js +0 -19
  148. package/scripts/damage-control-bash.js +0 -51
  149. package/scripts/damage-control-edit.js +0 -48
  150. package/scripts/damage-control-multi-agent.js +0 -231
  151. package/scripts/damage-control-write.js +0 -48
  152. package/scripts/dependency-check.js +0 -311
  153. package/scripts/document-repl.js +0 -793
  154. package/scripts/expertise-metrics.sh +0 -264
  155. package/scripts/generate-all.sh +0 -77
  156. package/scripts/generate-colors.js +0 -314
  157. package/scripts/generators/agent-registry.js +0 -183
  158. package/scripts/generators/command-registry.js +0 -166
  159. package/scripts/generators/index.js +0 -85
  160. package/scripts/generators/inject-babysit.js +0 -191
  161. package/scripts/generators/inject-help.js +0 -125
  162. package/scripts/generators/inject-readme.js +0 -166
  163. package/scripts/generators/skill-registry.js +0 -188
  164. package/scripts/get-env.js +0 -225
  165. package/scripts/init.sh +0 -76
  166. package/scripts/lib/README-portable-tasks.md +0 -424
  167. package/scripts/lib/ac-test-matcher.js +0 -452
  168. package/scripts/lib/audit-cleanup.js +0 -250
  169. package/scripts/lib/audit-registry.js +0 -340
  170. package/scripts/lib/automation-registry.js +0 -544
  171. package/scripts/lib/automation-runner.js +0 -476
  172. package/scripts/lib/browser-qa-evidence.js +0 -409
  173. package/scripts/lib/browser-qa-status.js +0 -192
  174. package/scripts/lib/bus-utils.js +0 -473
  175. package/scripts/lib/colors.generated.sh +0 -82
  176. package/scripts/lib/colors.sh +0 -46
  177. package/scripts/lib/command-prereqs.js +0 -280
  178. package/scripts/lib/concurrency-limiter.js +0 -511
  179. package/scripts/lib/configure-detect.js +0 -596
  180. package/scripts/lib/configure-features.js +0 -1927
  181. package/scripts/lib/configure-repair.js +0 -327
  182. package/scripts/lib/configure-utils.js +0 -114
  183. package/scripts/lib/context-formatter.js +0 -1158
  184. package/scripts/lib/context-loader.js +0 -840
  185. package/scripts/lib/counter.js +0 -103
  186. package/scripts/lib/damage-control-utils.js +0 -619
  187. package/scripts/lib/feature-catalog.js +0 -332
  188. package/scripts/lib/file-lock.js +0 -392
  189. package/scripts/lib/file-tracking.js +0 -735
  190. package/scripts/lib/frontmatter-parser.js +0 -133
  191. package/scripts/lib/gate-enforcer.js +0 -295
  192. package/scripts/lib/hook-metrics.js +0 -324
  193. package/scripts/lib/ideation-index.js +0 -1205
  194. package/scripts/lib/json-utils.sh +0 -162
  195. package/scripts/lib/lifecycle-detector.js +0 -125
  196. package/scripts/lib/model-profiles.js +0 -118
  197. package/scripts/lib/portable-tasks-cli.js +0 -274
  198. package/scripts/lib/portable-tasks.js +0 -479
  199. package/scripts/lib/process-cleanup.js +0 -527
  200. package/scripts/lib/quality-gates.js +0 -788
  201. package/scripts/lib/scale-detector.js +0 -396
  202. package/scripts/lib/sessionRegistry.js +0 -678
  203. package/scripts/lib/signal-detectors.js +0 -867
  204. package/scripts/lib/skill-catalog.js +0 -557
  205. package/scripts/lib/skill-recommender.js +0 -311
  206. package/scripts/lib/state-migrator.js +0 -353
  207. package/scripts/lib/status-task-bridge.js +0 -522
  208. package/scripts/lib/status-writer.js +0 -255
  209. package/scripts/lib/story-claiming.js +0 -704
  210. package/scripts/lib/story-state-machine.js +0 -437
  211. package/scripts/lib/sync-ideation-status.js +0 -291
  212. package/scripts/lib/task-registry-cache.js +0 -490
  213. package/scripts/lib/task-registry.js +0 -1191
  214. package/scripts/lib/task-sync.js +0 -230
  215. package/scripts/lib/tdd-phase-manager.js +0 -455
  216. package/scripts/lib/team-events.js +0 -510
  217. package/scripts/lib/tmux-audit-monitor.js +0 -612
  218. package/scripts/lib/tmux-group-colors.js +0 -113
  219. package/scripts/lib/tool-registry.yaml +0 -241
  220. package/scripts/lib/tool-shed.js +0 -441
  221. package/scripts/lib/validation-registry.js +0 -177
  222. package/scripts/messaging-bridge.js +0 -561
  223. package/scripts/migrate-ideation-index.js +0 -553
  224. package/scripts/native-team-observer.js +0 -219
  225. package/scripts/obtain-context.js +0 -272
  226. package/scripts/pre-push-check.sh +0 -46
  227. package/scripts/precompact-context.sh +0 -306
  228. package/scripts/query-codebase.js +0 -543
  229. package/scripts/ralph-loop.js +0 -1278
  230. package/scripts/resume-session.sh +0 -121
  231. package/scripts/screenshot-verifier.js +0 -215
  232. package/scripts/session-boundary.js +0 -138
  233. package/scripts/session-coordinator.sh +0 -232
  234. package/scripts/session-manager.js +0 -546
  235. package/scripts/smart-detect.js +0 -449
  236. package/scripts/spawn-audit-sessions.js +0 -877
  237. package/scripts/spawn-parallel.js +0 -751
  238. package/scripts/strip-ai-attribution.js +0 -63
  239. package/scripts/task-completed-gate.js +0 -237
  240. package/scripts/team-manager.js +0 -596
  241. package/scripts/team-status-display.js +0 -200
  242. package/scripts/teammate-idle-gate.js +0 -237
  243. package/scripts/test-session-boundary.js +0 -80
  244. package/scripts/tmux-close-windows.sh +0 -180
  245. package/scripts/tmux-restore-window.sh +0 -67
  246. package/scripts/tmux-save-closed-window.sh +0 -35
  247. package/scripts/tui/App.js +0 -151
  248. package/scripts/tui/Dashboard.js +0 -277
  249. package/scripts/tui/blessed/data/watcher.js +0 -180
  250. package/scripts/tui/blessed/index.js +0 -244
  251. package/scripts/tui/blessed/panels/output.js +0 -101
  252. package/scripts/tui/blessed/panels/sessions.js +0 -150
  253. package/scripts/tui/blessed/panels/trace.js +0 -97
  254. package/scripts/tui/blessed/ui/help.js +0 -77
  255. package/scripts/tui/blessed/ui/screen.js +0 -52
  256. package/scripts/tui/blessed/ui/statusbar.js +0 -47
  257. package/scripts/tui/blessed/ui/tabbar.js +0 -99
  258. package/scripts/tui/index.js +0 -70
  259. package/scripts/tui/lib/crashRecovery.js +0 -304
  260. package/scripts/tui/lib/eventStream.js +0 -309
  261. package/scripts/tui/lib/keyboard.js +0 -261
  262. package/scripts/tui/lib/loopControl.js +0 -371
  263. package/scripts/tui/panels/OutputPanel.js +0 -240
  264. package/scripts/tui/panels/SessionPanel.js +0 -170
  265. package/scripts/tui/panels/TracePanel.js +0 -298
  266. package/scripts/tui/simple-tui.js +0 -510
  267. package/scripts/validate-expertise.sh +0 -263
  268. package/scripts/validate-tokens.sh +0 -73
  269. package/scripts/validators/README.md +0 -143
  270. package/scripts/validators/component-validator.js +0 -239
  271. package/scripts/validators/json-schema-validator.js +0 -186
  272. package/scripts/validators/markdown-validator.js +0 -152
  273. package/scripts/validators/migration-validator.js +0 -129
  274. package/scripts/validators/security-validator.js +0 -380
  275. package/scripts/validators/story-format-validator.js +0 -197
  276. package/scripts/validators/test-result-validator.js +0 -114
  277. package/scripts/validators/workflow-validator.js +0 -247
  278. package/scripts/welcome-deferred.js +0 -437
  279. package/scripts/worktree-create.sh +0 -111
  280. package/src/core/agents/a11y-analyzer-aria.md +0 -155
  281. package/src/core/agents/a11y-analyzer-forms.md +0 -162
  282. package/src/core/agents/a11y-analyzer-keyboard.md +0 -175
  283. package/src/core/agents/a11y-analyzer-semantic.md +0 -153
  284. package/src/core/agents/a11y-analyzer-visual.md +0 -158
  285. package/src/core/agents/a11y-consensus.md +0 -248
  286. package/src/core/agents/accessibility.md +0 -515
  287. package/src/core/agents/adr-writer.md +0 -463
  288. package/src/core/agents/ads-audit-budget.md +0 -181
  289. package/src/core/agents/ads-audit-compliance.md +0 -169
  290. package/src/core/agents/ads-audit-creative.md +0 -164
  291. package/src/core/agents/ads-audit-google.md +0 -226
  292. package/src/core/agents/ads-audit-meta.md +0 -183
  293. package/src/core/agents/ads-audit-tracking.md +0 -197
  294. package/src/core/agents/ads-consensus.md +0 -396
  295. package/src/core/agents/ads-generate.md +0 -145
  296. package/src/core/agents/ads-performance-tracker.md +0 -197
  297. package/src/core/agents/analytics.md +0 -617
  298. package/src/core/agents/api-quality-analyzer-conventions.md +0 -148
  299. package/src/core/agents/api-quality-analyzer-docs.md +0 -176
  300. package/src/core/agents/api-quality-analyzer-errors.md +0 -183
  301. package/src/core/agents/api-quality-analyzer-pagination.md +0 -171
  302. package/src/core/agents/api-quality-analyzer-versioning.md +0 -143
  303. package/src/core/agents/api-quality-consensus.md +0 -214
  304. package/src/core/agents/api-validator.md +0 -183
  305. package/src/core/agents/api.md +0 -665
  306. package/src/core/agents/arch-analyzer-circular.md +0 -148
  307. package/src/core/agents/arch-analyzer-complexity.md +0 -171
  308. package/src/core/agents/arch-analyzer-coupling.md +0 -146
  309. package/src/core/agents/arch-analyzer-layering.md +0 -151
  310. package/src/core/agents/arch-analyzer-patterns.md +0 -162
  311. package/src/core/agents/arch-consensus.md +0 -227
  312. package/src/core/agents/brainstorm-analyzer-features.md +0 -169
  313. package/src/core/agents/brainstorm-analyzer-growth.md +0 -161
  314. package/src/core/agents/brainstorm-analyzer-integration.md +0 -172
  315. package/src/core/agents/brainstorm-analyzer-market.md +0 -147
  316. package/src/core/agents/brainstorm-analyzer-ux.md +0 -167
  317. package/src/core/agents/brainstorm-consensus.md +0 -237
  318. package/src/core/agents/browser-qa.md +0 -328
  319. package/src/core/agents/ci.md +0 -511
  320. package/src/core/agents/code-reviewer.md +0 -288
  321. package/src/core/agents/codebase-query.md +0 -266
  322. package/src/core/agents/completeness-analyzer-api.md +0 -190
  323. package/src/core/agents/completeness-analyzer-conditional.md +0 -201
  324. package/src/core/agents/completeness-analyzer-handlers.md +0 -159
  325. package/src/core/agents/completeness-analyzer-imports.md +0 -159
  326. package/src/core/agents/completeness-analyzer-routes.md +0 -182
  327. package/src/core/agents/completeness-analyzer-state.md +0 -188
  328. package/src/core/agents/completeness-analyzer-stubs.md +0 -198
  329. package/src/core/agents/completeness-consensus.md +0 -286
  330. package/src/core/agents/compliance.md +0 -509
  331. package/src/core/agents/council-advocate.md +0 -206
  332. package/src/core/agents/council-analyst.md +0 -252
  333. package/src/core/agents/council-optimist.md +0 -170
  334. package/src/core/agents/database.md +0 -601
  335. package/src/core/agents/datamigration.md +0 -699
  336. package/src/core/agents/design.md +0 -525
  337. package/src/core/agents/devops.md +0 -720
  338. package/src/core/agents/documentation.md +0 -504
  339. package/src/core/agents/epic-planner.md +0 -480
  340. package/src/core/agents/error-analyzer.md +0 -201
  341. package/src/core/agents/integrations.md +0 -603
  342. package/src/core/agents/legal-analyzer-a11y.md +0 -110
  343. package/src/core/agents/legal-analyzer-ai.md +0 -117
  344. package/src/core/agents/legal-analyzer-consumer.md +0 -108
  345. package/src/core/agents/legal-analyzer-content.md +0 -113
  346. package/src/core/agents/legal-analyzer-international.md +0 -115
  347. package/src/core/agents/legal-analyzer-licensing.md +0 -115
  348. package/src/core/agents/legal-analyzer-privacy.md +0 -108
  349. package/src/core/agents/legal-analyzer-security.md +0 -112
  350. package/src/core/agents/legal-analyzer-terms.md +0 -111
  351. package/src/core/agents/legal-consensus.md +0 -242
  352. package/src/core/agents/logic-analyzer-edge.md +0 -170
  353. package/src/core/agents/logic-analyzer-flow.md +0 -253
  354. package/src/core/agents/logic-analyzer-invariant.md +0 -206
  355. package/src/core/agents/logic-analyzer-race.md +0 -266
  356. package/src/core/agents/logic-analyzer-type.md +0 -217
  357. package/src/core/agents/logic-consensus.md +0 -253
  358. package/src/core/agents/mentor.md +0 -654
  359. package/src/core/agents/mobile.md +0 -501
  360. package/src/core/agents/monitoring.md +0 -537
  361. package/src/core/agents/multi-expert.md +0 -311
  362. package/src/core/agents/orchestrator.md +0 -749
  363. package/src/core/agents/perf-analyzer-assets.md +0 -174
  364. package/src/core/agents/perf-analyzer-bundle.md +0 -165
  365. package/src/core/agents/perf-analyzer-caching.md +0 -160
  366. package/src/core/agents/perf-analyzer-compute.md +0 -165
  367. package/src/core/agents/perf-analyzer-memory.md +0 -182
  368. package/src/core/agents/perf-analyzer-network.md +0 -157
  369. package/src/core/agents/perf-analyzer-queries.md +0 -155
  370. package/src/core/agents/perf-analyzer-rendering.md +0 -156
  371. package/src/core/agents/perf-consensus.md +0 -280
  372. package/src/core/agents/performance.md +0 -492
  373. package/src/core/agents/product.md +0 -535
  374. package/src/core/agents/qa.md +0 -765
  375. package/src/core/agents/readme-updater.md +0 -579
  376. package/src/core/agents/refactor.md +0 -558
  377. package/src/core/agents/research.md +0 -453
  378. package/src/core/agents/rlm-subcore.md +0 -207
  379. package/src/core/agents/schema-validator.md +0 -454
  380. package/src/core/agents/security-analyzer-api.md +0 -199
  381. package/src/core/agents/security-analyzer-auth.md +0 -160
  382. package/src/core/agents/security-analyzer-authz.md +0 -168
  383. package/src/core/agents/security-analyzer-deps.md +0 -147
  384. package/src/core/agents/security-analyzer-infra.md +0 -176
  385. package/src/core/agents/security-analyzer-injection.md +0 -148
  386. package/src/core/agents/security-analyzer-input.md +0 -191
  387. package/src/core/agents/security-analyzer-secrets.md +0 -175
  388. package/src/core/agents/security-consensus.md +0 -276
  389. package/src/core/agents/security.md +0 -486
  390. package/src/core/agents/seo-analyzer-content.md +0 -167
  391. package/src/core/agents/seo-analyzer-images.md +0 -187
  392. package/src/core/agents/seo-analyzer-performance.md +0 -206
  393. package/src/core/agents/seo-analyzer-schema.md +0 -176
  394. package/src/core/agents/seo-analyzer-sitemap.md +0 -172
  395. package/src/core/agents/seo-analyzer-technical.md +0 -144
  396. package/src/core/agents/seo-consensus.md +0 -289
  397. package/src/core/agents/team-coordinator.md +0 -333
  398. package/src/core/agents/team-lead.md +0 -171
  399. package/src/core/agents/test-analyzer-assertions.md +0 -181
  400. package/src/core/agents/test-analyzer-coverage.md +0 -183
  401. package/src/core/agents/test-analyzer-fragility.md +0 -185
  402. package/src/core/agents/test-analyzer-integration.md +0 -155
  403. package/src/core/agents/test-analyzer-maintenance.md +0 -173
  404. package/src/core/agents/test-analyzer-mocking.md +0 -178
  405. package/src/core/agents/test-analyzer-patterns.md +0 -189
  406. package/src/core/agents/test-analyzer-structure.md +0 -177
  407. package/src/core/agents/test-consensus.md +0 -294
  408. package/src/core/agents/testing.md +0 -527
  409. package/src/core/agents/ui-validator.md +0 -331
  410. package/src/core/agents/ui.md +0 -1227
  411. package/src/core/commands/adr/list.md +0 -191
  412. package/src/core/commands/adr/update.md +0 -258
  413. package/src/core/commands/adr/view.md +0 -274
  414. package/src/core/commands/adr.md +0 -394
  415. package/src/core/commands/ads/audit.md +0 -453
  416. package/src/core/commands/ads/budget.md +0 -97
  417. package/src/core/commands/ads/competitor.md +0 -112
  418. package/src/core/commands/ads/creative.md +0 -85
  419. package/src/core/commands/ads/generate.md +0 -238
  420. package/src/core/commands/ads/google.md +0 -112
  421. package/src/core/commands/ads/health.md +0 -327
  422. package/src/core/commands/ads/landing.md +0 -119
  423. package/src/core/commands/ads/linkedin.md +0 -112
  424. package/src/core/commands/ads/meta.md +0 -91
  425. package/src/core/commands/ads/microsoft.md +0 -115
  426. package/src/core/commands/ads/plan.md +0 -321
  427. package/src/core/commands/ads/test-plan.md +0 -317
  428. package/src/core/commands/ads/tiktok.md +0 -129
  429. package/src/core/commands/ads/track.md +0 -288
  430. package/src/core/commands/ads/youtube.md +0 -124
  431. package/src/core/commands/ads.md +0 -140
  432. package/src/core/commands/agent.md +0 -256
  433. package/src/core/commands/api.md +0 -267
  434. package/src/core/commands/assign.md +0 -369
  435. package/src/core/commands/audit.md +0 -531
  436. package/src/core/commands/auto.md +0 -556
  437. package/src/core/commands/automate.md +0 -415
  438. package/src/core/commands/babysit.md +0 -643
  439. package/src/core/commands/baseline.md +0 -743
  440. package/src/core/commands/batch.md +0 -551
  441. package/src/core/commands/blockers.md +0 -602
  442. package/src/core/commands/board.md +0 -509
  443. package/src/core/commands/browser-qa.md +0 -240
  444. package/src/core/commands/changelog.md +0 -582
  445. package/src/core/commands/choose.md +0 -430
  446. package/src/core/commands/ci.md +0 -330
  447. package/src/core/commands/code/accessibility.md +0 -363
  448. package/src/core/commands/code/api.md +0 -313
  449. package/src/core/commands/code/architecture.md +0 -313
  450. package/src/core/commands/code/completeness.md +0 -519
  451. package/src/core/commands/code/legal.md +0 -509
  452. package/src/core/commands/code/logic.md +0 -432
  453. package/src/core/commands/code/performance.md +0 -506
  454. package/src/core/commands/code/security.md +0 -509
  455. package/src/core/commands/code/test.md +0 -505
  456. package/src/core/commands/compress.md +0 -408
  457. package/src/core/commands/configure.md +0 -1159
  458. package/src/core/commands/context/export.md +0 -296
  459. package/src/core/commands/context/full.md +0 -353
  460. package/src/core/commands/context/note.md +0 -380
  461. package/src/core/commands/council.md +0 -592
  462. package/src/core/commands/debt.md +0 -491
  463. package/src/core/commands/deploy.md +0 -864
  464. package/src/core/commands/deps.md +0 -728
  465. package/src/core/commands/diagnose.md +0 -404
  466. package/src/core/commands/docs.md +0 -469
  467. package/src/core/commands/epic/edit.md +0 -213
  468. package/src/core/commands/epic/list.md +0 -190
  469. package/src/core/commands/epic/view.md +0 -267
  470. package/src/core/commands/epic.md +0 -477
  471. package/src/core/commands/export.md +0 -238
  472. package/src/core/commands/feedback.md +0 -603
  473. package/src/core/commands/handoff.md +0 -386
  474. package/src/core/commands/help.md +0 -194
  475. package/src/core/commands/ideate/brief.md +0 -363
  476. package/src/core/commands/ideate/discover.md +0 -399
  477. package/src/core/commands/ideate/features.md +0 -497
  478. package/src/core/commands/ideate/history.md +0 -403
  479. package/src/core/commands/ideate/new.md +0 -900
  480. package/src/core/commands/impact.md +0 -407
  481. package/src/core/commands/install.md +0 -529
  482. package/src/core/commands/learn/explain.md +0 -118
  483. package/src/core/commands/learn/glossary.md +0 -135
  484. package/src/core/commands/learn/patterns.md +0 -138
  485. package/src/core/commands/learn/tour.md +0 -126
  486. package/src/core/commands/maintain.md +0 -558
  487. package/src/core/commands/metrics.md +0 -844
  488. package/src/core/commands/migrate/codemods.md +0 -151
  489. package/src/core/commands/migrate/plan.md +0 -131
  490. package/src/core/commands/migrate/scan.md +0 -114
  491. package/src/core/commands/migrate/validate.md +0 -119
  492. package/src/core/commands/multi-expert.md +0 -447
  493. package/src/core/commands/packages.md +0 -535
  494. package/src/core/commands/pr.md +0 -337
  495. package/src/core/commands/readme-sync.md +0 -329
  496. package/src/core/commands/research/analyze.md +0 -798
  497. package/src/core/commands/research/ask.md +0 -864
  498. package/src/core/commands/research/import.md +0 -1025
  499. package/src/core/commands/research/list.md +0 -273
  500. package/src/core/commands/research/synthesize.md +0 -928
  501. package/src/core/commands/research/view.md +0 -323
  502. package/src/core/commands/retro.md +0 -795
  503. package/src/core/commands/review.md +0 -694
  504. package/src/core/commands/rlm.md +0 -446
  505. package/src/core/commands/roadmap/analyze.md +0 -400
  506. package/src/core/commands/rpi.md +0 -633
  507. package/src/core/commands/seo/audit.md +0 -444
  508. package/src/core/commands/seo/competitor.md +0 -174
  509. package/src/core/commands/seo/content.md +0 -107
  510. package/src/core/commands/seo/geo.md +0 -229
  511. package/src/core/commands/seo/hreflang.md +0 -140
  512. package/src/core/commands/seo/images.md +0 -96
  513. package/src/core/commands/seo/page.md +0 -198
  514. package/src/core/commands/seo/plan.md +0 -163
  515. package/src/core/commands/seo/programmatic.md +0 -131
  516. package/src/core/commands/seo/references/cwv-thresholds.md +0 -64
  517. package/src/core/commands/seo/references/eeat-framework.md +0 -110
  518. package/src/core/commands/seo/references/quality-gates.md +0 -91
  519. package/src/core/commands/seo/references/schema-types.md +0 -102
  520. package/src/core/commands/seo/schema.md +0 -183
  521. package/src/core/commands/seo/sitemap.md +0 -97
  522. package/src/core/commands/seo/technical.md +0 -100
  523. package/src/core/commands/seo.md +0 -107
  524. package/src/core/commands/session/cleanup.md +0 -452
  525. package/src/core/commands/session/end.md +0 -865
  526. package/src/core/commands/session/history.md +0 -293
  527. package/src/core/commands/session/init.md +0 -210
  528. package/src/core/commands/session/new.md +0 -827
  529. package/src/core/commands/session/resume.md +0 -291
  530. package/src/core/commands/session/spawn.md +0 -205
  531. package/src/core/commands/session/status.md +0 -274
  532. package/src/core/commands/skill/list.md +0 -139
  533. package/src/core/commands/skill/recommend.md +0 -216
  534. package/src/core/commands/sprint.md +0 -714
  535. package/src/core/commands/status/undo.md +0 -191
  536. package/src/core/commands/status.md +0 -423
  537. package/src/core/commands/story/edit.md +0 -204
  538. package/src/core/commands/story/list.md +0 -199
  539. package/src/core/commands/story/view.md +0 -312
  540. package/src/core/commands/story-validate.md +0 -491
  541. package/src/core/commands/story.md +0 -465
  542. package/src/core/commands/tdd-next.md +0 -238
  543. package/src/core/commands/tdd.md +0 -211
  544. package/src/core/commands/team/guide.md +0 -688
  545. package/src/core/commands/team/list.md +0 -59
  546. package/src/core/commands/team/start.md +0 -130
  547. package/src/core/commands/team/status.md +0 -66
  548. package/src/core/commands/team/stop.md +0 -78
  549. package/src/core/commands/template.md +0 -644
  550. package/src/core/commands/tests.md +0 -731
  551. package/src/core/commands/update.md +0 -591
  552. package/src/core/commands/validate-expertise.md +0 -305
  553. package/src/core/commands/velocity.md +0 -630
  554. package/src/core/commands/verify.md +0 -534
  555. package/src/core/commands/whats-new.md +0 -201
  556. package/src/core/commands/workflow.md +0 -449
  557. package/src/core/council/sessions/.gitkeep +0 -0
  558. package/src/core/council/shared_reasoning.template.md +0 -106
  559. package/src/core/experts/README.md +0 -236
  560. package/src/core/experts/_core-expertise.yaml +0 -105
  561. package/src/core/experts/accessibility/expertise.yaml +0 -115
  562. package/src/core/experts/accessibility/question.md +0 -41
  563. package/src/core/experts/accessibility/self-improve.md +0 -45
  564. package/src/core/experts/accessibility/workflow.md +0 -59
  565. package/src/core/experts/adr-writer/expertise.yaml +0 -138
  566. package/src/core/experts/adr-writer/question.md +0 -56
  567. package/src/core/experts/adr-writer/self-improve.md +0 -106
  568. package/src/core/experts/adr-writer/workflow.md +0 -184
  569. package/src/core/experts/analytics/expertise.yaml +0 -119
  570. package/src/core/experts/analytics/question.md +0 -74
  571. package/src/core/experts/analytics/self-improve.md +0 -163
  572. package/src/core/experts/analytics/workflow.md +0 -272
  573. package/src/core/experts/api/expertise.yaml +0 -124
  574. package/src/core/experts/api/question.md +0 -74
  575. package/src/core/experts/api/self-improve.md +0 -122
  576. package/src/core/experts/api/workflow.md +0 -248
  577. package/src/core/experts/ci/expertise.yaml +0 -106
  578. package/src/core/experts/ci/question.md +0 -69
  579. package/src/core/experts/ci/self-improve.md +0 -100
  580. package/src/core/experts/ci/workflow.md +0 -145
  581. package/src/core/experts/codebase-query/expertise.yaml +0 -121
  582. package/src/core/experts/codebase-query/question.md +0 -73
  583. package/src/core/experts/codebase-query/self-improve.md +0 -105
  584. package/src/core/experts/compliance/expertise.yaml +0 -101
  585. package/src/core/experts/compliance/question.md +0 -56
  586. package/src/core/experts/compliance/self-improve.md +0 -106
  587. package/src/core/experts/compliance/workflow.md +0 -184
  588. package/src/core/experts/database/expertise.yaml +0 -109
  589. package/src/core/experts/database/question.md +0 -74
  590. package/src/core/experts/database/self-improve.md +0 -121
  591. package/src/core/experts/database/workflow.md +0 -234
  592. package/src/core/experts/datamigration/expertise.yaml +0 -141
  593. package/src/core/experts/datamigration/question.md +0 -56
  594. package/src/core/experts/datamigration/self-improve.md +0 -106
  595. package/src/core/experts/datamigration/workflow.md +0 -184
  596. package/src/core/experts/design/expertise.yaml +0 -116
  597. package/src/core/experts/design/question.md +0 -56
  598. package/src/core/experts/design/self-improve.md +0 -106
  599. package/src/core/experts/design/workflow.md +0 -184
  600. package/src/core/experts/devops/expertise.yaml +0 -116
  601. package/src/core/experts/devops/question.md +0 -68
  602. package/src/core/experts/devops/self-improve.md +0 -102
  603. package/src/core/experts/devops/workflow.md +0 -142
  604. package/src/core/experts/documentation/expertise.yaml +0 -126
  605. package/src/core/experts/documentation/question.md +0 -41
  606. package/src/core/experts/documentation/self-improve.md +0 -45
  607. package/src/core/experts/documentation/workflow.md +0 -55
  608. package/src/core/experts/epic-planner/expertise.yaml +0 -144
  609. package/src/core/experts/epic-planner/question.md +0 -56
  610. package/src/core/experts/epic-planner/self-improve.md +0 -106
  611. package/src/core/experts/epic-planner/workflow.md +0 -184
  612. package/src/core/experts/integrations/expertise.yaml +0 -113
  613. package/src/core/experts/integrations/question.md +0 -74
  614. package/src/core/experts/integrations/self-improve.md +0 -151
  615. package/src/core/experts/integrations/workflow.md +0 -246
  616. package/src/core/experts/mentor/expertise.yaml +0 -125
  617. package/src/core/experts/mentor/question.md +0 -56
  618. package/src/core/experts/mentor/self-improve.md +0 -106
  619. package/src/core/experts/mentor/workflow.md +0 -184
  620. package/src/core/experts/mobile/expertise.yaml +0 -136
  621. package/src/core/experts/mobile/question.md +0 -72
  622. package/src/core/experts/mobile/self-improve.md +0 -140
  623. package/src/core/experts/mobile/workflow.md +0 -240
  624. package/src/core/experts/monitoring/expertise.yaml +0 -132
  625. package/src/core/experts/monitoring/question.md +0 -76
  626. package/src/core/experts/monitoring/self-improve.md +0 -150
  627. package/src/core/experts/monitoring/workflow.md +0 -264
  628. package/src/core/experts/performance/expertise.yaml +0 -68
  629. package/src/core/experts/performance/question.md +0 -41
  630. package/src/core/experts/performance/self-improve.md +0 -45
  631. package/src/core/experts/performance/workflow.md +0 -61
  632. package/src/core/experts/product/expertise.yaml +0 -143
  633. package/src/core/experts/product/question.md +0 -56
  634. package/src/core/experts/product/self-improve.md +0 -106
  635. package/src/core/experts/product/workflow.md +0 -184
  636. package/src/core/experts/qa/expertise.yaml +0 -110
  637. package/src/core/experts/qa/question.md +0 -56
  638. package/src/core/experts/qa/self-improve.md +0 -106
  639. package/src/core/experts/qa/workflow.md +0 -184
  640. package/src/core/experts/readme-updater/expertise.yaml +0 -141
  641. package/src/core/experts/readme-updater/question.md +0 -56
  642. package/src/core/experts/readme-updater/self-improve.md +0 -106
  643. package/src/core/experts/readme-updater/workflow.md +0 -184
  644. package/src/core/experts/refactor/expertise.yaml +0 -135
  645. package/src/core/experts/refactor/question.md +0 -41
  646. package/src/core/experts/refactor/self-improve.md +0 -45
  647. package/src/core/experts/refactor/workflow.md +0 -57
  648. package/src/core/experts/research/expertise.yaml +0 -143
  649. package/src/core/experts/research/question.md +0 -56
  650. package/src/core/experts/research/self-improve.md +0 -106
  651. package/src/core/experts/research/workflow.md +0 -184
  652. package/src/core/experts/security/expertise.yaml +0 -117
  653. package/src/core/experts/security/question.md +0 -77
  654. package/src/core/experts/security/self-improve.md +0 -102
  655. package/src/core/experts/security/workflow.md +0 -152
  656. package/src/core/experts/templates/expertise-template.yaml +0 -67
  657. package/src/core/experts/templates/question-template.md +0 -56
  658. package/src/core/experts/templates/self-improve-template.md +0 -106
  659. package/src/core/experts/templates/workflow-template.md +0 -184
  660. package/src/core/experts/testing/expertise.yaml +0 -112
  661. package/src/core/experts/testing/question.md +0 -68
  662. package/src/core/experts/testing/self-improve.md +0 -102
  663. package/src/core/experts/testing/workflow.md +0 -143
  664. package/src/core/experts/ui/expertise.yaml +0 -133
  665. package/src/core/experts/ui/question.md +0 -74
  666. package/src/core/experts/ui/self-improve.md +0 -122
  667. package/src/core/experts/ui/workflow.md +0 -262
  668. package/src/core/knowledge/ads/ad-audit-checklist-scoring.md +0 -424
  669. package/src/core/knowledge/ads/ad-optimization-logic.md +0 -590
  670. package/src/core/knowledge/ads/ad-technical-specifications.md +0 -385
  671. package/src/core/knowledge/ads/definitive-advertising-reference-2026.md +0 -506
  672. package/src/core/knowledge/ads/paid-advertising-research-2026.md +0 -445
  673. package/src/core/profiles/COMPARISON.md +0 -170
  674. package/src/core/profiles/README.md +0 -178
  675. package/src/core/profiles/claude-code.yaml +0 -111
  676. package/src/core/profiles/codex.yaml +0 -103
  677. package/src/core/profiles/cursor.yaml +0 -134
  678. package/src/core/profiles/examples.js +0 -250
  679. package/src/core/profiles/loader.js +0 -235
  680. package/src/core/profiles/windsurf.yaml +0 -159
  681. package/src/core/skills/_learnings/README.md +0 -91
  682. package/src/core/skills/_learnings/_template.yaml +0 -106
  683. package/src/core/skills/_learnings/code-review.yaml +0 -118
  684. package/src/core/skills/_learnings/commit.yaml +0 -69
  685. package/src/core/skills/_learnings/story-writer.yaml +0 -71
  686. package/src/core/teams/backend.json +0 -41
  687. package/src/core/teams/builder-validator.json +0 -51
  688. package/src/core/teams/code-review.json +0 -41
  689. package/src/core/teams/frontend.json +0 -41
  690. package/src/core/teams/fullstack.json +0 -41
  691. package/src/core/teams/logic-audit.json +0 -53
  692. package/src/core/teams/perf-audit.json +0 -71
  693. package/src/core/teams/qa.json +0 -41
  694. package/src/core/teams/security-audit.json +0 -71
  695. package/src/core/teams/solo.json +0 -35
  696. package/src/core/teams/test-audit.json +0 -71
  697. package/src/core/templates/CONTEXT.md.example +0 -49
  698. package/src/core/templates/README-template.md +0 -16
  699. package/src/core/templates/adr-template.md +0 -28
  700. package/src/core/templates/agent-coordination-pattern.md +0 -38
  701. package/src/core/templates/agent-profile-template.md +0 -51
  702. package/src/core/templates/agileflow-metadata.json +0 -150
  703. package/src/core/templates/browser-qa-spec.yaml +0 -94
  704. package/src/core/templates/ci-workflow.yml +0 -74
  705. package/src/core/templates/claude-settings.advanced.example.json +0 -75
  706. package/src/core/templates/claude-settings.example.json +0 -26
  707. package/src/core/templates/command-documentation.md +0 -187
  708. package/src/core/templates/command-prerequisites.yaml +0 -169
  709. package/src/core/templates/comms-note-template.md +0 -24
  710. package/src/core/templates/damage-control-patterns.yaml +0 -243
  711. package/src/core/templates/environment.json +0 -18
  712. package/src/core/templates/epic-template.md +0 -27
  713. package/src/core/templates/plan-template.md +0 -125
  714. package/src/core/templates/preserve-rules-common.md +0 -107
  715. package/src/core/templates/preserve-rules.json +0 -42
  716. package/src/core/templates/proactive-action-spec.md +0 -29
  717. package/src/core/templates/product-brief.md +0 -136
  718. package/src/core/templates/quality-gate-priorities.md +0 -34
  719. package/src/core/templates/research-template.md +0 -44
  720. package/src/core/templates/session-harness-protocol.md +0 -128
  721. package/src/core/templates/session-state.json +0 -56
  722. package/src/core/templates/story-lifecycle.md +0 -213
  723. package/src/core/templates/story-template.md +0 -92
  724. package/src/core/templates/tdd-test-template.js +0 -241
  725. package/src/core/templates/worktrees-guide.md +0 -231
  726. package/tools/agileflow-npx.js +0 -52
  727. package/tools/cli/agileflow-cli.js +0 -72
  728. package/tools/cli/commands/config.js +0 -285
  729. package/tools/cli/commands/doctor.js +0 -496
  730. package/tools/cli/commands/list.js +0 -385
  731. package/tools/cli/commands/session.js +0 -1176
  732. package/tools/cli/commands/setup.js +0 -255
  733. package/tools/cli/commands/status.js +0 -101
  734. package/tools/cli/commands/tui.js +0 -56
  735. package/tools/cli/commands/uninstall.js +0 -155
  736. package/tools/cli/commands/update.js +0 -299
  737. package/tools/cli/installers/core/installer.js +0 -892
  738. package/tools/cli/installers/ide/_base-ide.js +0 -518
  739. package/tools/cli/installers/ide/_interface.js +0 -238
  740. package/tools/cli/installers/ide/claude-code.js +0 -432
  741. package/tools/cli/installers/ide/codex.js +0 -426
  742. package/tools/cli/installers/ide/cursor.js +0 -217
  743. package/tools/cli/installers/ide/manager.js +0 -222
  744. package/tools/cli/installers/ide/windsurf.js +0 -282
  745. package/tools/cli/lib/command-context.js +0 -382
  746. package/tools/cli/lib/config-manager.js +0 -446
  747. package/tools/cli/lib/content-injector.js +0 -969
  748. package/tools/cli/lib/content-transformer.js +0 -496
  749. package/tools/cli/lib/docs-setup.js +0 -464
  750. package/tools/cli/lib/error-handler.js +0 -165
  751. package/tools/cli/lib/ide-error-factory.js +0 -421
  752. package/tools/cli/lib/ide-errors.js +0 -367
  753. package/tools/cli/lib/ide-generator.js +0 -357
  754. package/tools/cli/lib/ide-health-monitor.js +0 -364
  755. package/tools/cli/lib/ide-registry.js +0 -297
  756. package/tools/cli/lib/npm-utils.js +0 -103
  757. package/tools/cli/lib/self-update.js +0 -148
  758. package/tools/cli/lib/ui.js +0 -211
  759. package/tools/cli/lib/utils.js +0 -87
  760. package/tools/cli/lib/validation-middleware.js +0 -491
  761. package/tools/cli/lib/version-checker.js +0 -95
  762. package/tools/postinstall.js +0 -190
@@ -1,1191 +0,0 @@
1
- /**
2
- * task-registry.js - Multi-Agent Task Orchestration System
3
- *
4
- * Provides CRUD operations for tasks with:
5
- * - State machine with valid transitions (queued→running→completed/failed/blocked)
6
- * - DAG validation to prevent circular dependencies
7
- * - Atomic file writes (temp + rename) for crash recovery
8
- * - Lock file for concurrent access
9
- * - Event emissions for observability
10
- *
11
- * Task States:
12
- * - queued: Task is waiting to be executed
13
- * - running: Task is currently being executed by an agent
14
- * - completed: Task finished successfully
15
- * - failed: Task failed with an error
16
- * - blocked: Task cannot proceed due to unmet dependencies
17
- *
18
- * Valid Transitions:
19
- * - queued → running, blocked, cancelled
20
- * - running → completed, failed, blocked
21
- * - blocked → queued, running, cancelled
22
- * - completed → (terminal)
23
- * - failed → queued (retry), cancelled
24
- * - cancelled → (terminal)
25
- */
26
-
27
- 'use strict';
28
-
29
- const EventEmitter = require('events');
30
- const fs = require('fs');
31
- const path = require('path');
32
- const crypto = require('crypto');
33
- const os = require('os');
34
-
35
- // ============================================================================
36
- // Constants
37
- // ============================================================================
38
-
39
- // Valid task states - SINGLE SOURCE OF TRUTH
40
- const TASK_STATES = ['queued', 'running', 'completed', 'failed', 'blocked', 'cancelled'];
41
-
42
- // Terminal states (no transitions out)
43
- const TERMINAL_STATES = ['completed', 'cancelled'];
44
-
45
- // States indicating work is done (for metrics)
46
- const DONE_STATES = ['completed', 'failed', 'cancelled'];
47
-
48
- // Valid state transitions
49
- // Key = from state, Value = array of valid "to" states
50
- const TRANSITIONS = {
51
- queued: ['running', 'blocked', 'cancelled'],
52
- running: ['completed', 'failed', 'blocked'],
53
- blocked: ['queued', 'running', 'cancelled'],
54
- completed: [], // Terminal
55
- failed: ['queued', 'cancelled'], // Can retry
56
- cancelled: [], // Terminal
57
- };
58
-
59
- // Join strategies for parallel task groups
60
- const JOIN_STRATEGIES = ['all', 'first', 'any', 'any-N', 'majority'];
61
-
62
- // Failure policies
63
- const FAILURE_POLICIES = ['fail-fast', 'continue', 'ignore'];
64
-
65
- // Default configuration
66
- const DEFAULT_STATE_PATH = '.agileflow/state/task-dependencies.json';
67
- const LOCK_FILE_NAME = 'task-registry.lock';
68
- const LOCK_TIMEOUT_MS = 30000; // 30 seconds
69
- const LOCK_STALE_MS = 60000; // 1 minute - consider lock stale if older
70
-
71
- // ============================================================================
72
- // Utility Functions
73
- // ============================================================================
74
-
75
- /**
76
- * Generate a unique task ID
77
- * @returns {string} Unique task ID (e.g., "task-abc12345")
78
- */
79
- function generateTaskId() {
80
- const timestamp = Date.now().toString(36);
81
- const random = crypto.randomBytes(4).toString('hex');
82
- return `task-${timestamp}-${random}`;
83
- }
84
-
85
- /**
86
- * Find project root by looking for .agileflow directory
87
- * @returns {string} Project root path or current working directory
88
- */
89
- function findProjectRoot() {
90
- let dir = process.cwd();
91
- while (dir !== '/' && dir !== path.parse(dir).root) {
92
- if (fs.existsSync(path.join(dir, '.agileflow'))) {
93
- return dir;
94
- }
95
- dir = path.dirname(dir);
96
- }
97
- return process.cwd();
98
- }
99
-
100
- /**
101
- * Atomic file write using temp file + rename
102
- * @param {string} filePath - Target file path
103
- * @param {string} content - Content to write
104
- */
105
- function atomicWrite(filePath, content) {
106
- const dir = path.dirname(filePath);
107
- if (!fs.existsSync(dir)) {
108
- fs.mkdirSync(dir, { recursive: true });
109
- }
110
-
111
- // Write to temp file first
112
- const tempPath = `${filePath}.tmp.${process.pid}.${Date.now()}`;
113
- fs.writeFileSync(tempPath, content, 'utf8');
114
-
115
- // Atomic rename
116
- fs.renameSync(tempPath, filePath);
117
- }
118
-
119
- /**
120
- * Safe JSON parse with default
121
- * @param {string} content - JSON string
122
- * @param {*} defaultValue - Default value if parse fails
123
- * @returns {*} Parsed value or default
124
- */
125
- function safeJSONParse(content, defaultValue = null) {
126
- try {
127
- return JSON.parse(content);
128
- } catch (e) {
129
- return defaultValue;
130
- }
131
- }
132
-
133
- // ============================================================================
134
- // DAG Validation
135
- // ============================================================================
136
-
137
- /**
138
- * Detect circular dependencies using DFS
139
- * @param {Object} tasks - Map of task ID to task object
140
- * @param {string} startId - Task ID to start from
141
- * @param {Set} [visited] - Already visited nodes
142
- * @param {Set} [recursionStack] - Current recursion path
143
- * @returns {{ hasCycle: boolean, cycle: string[] | null }}
144
- */
145
- function detectCycle(tasks, startId, visited = new Set(), recursionStack = new Set()) {
146
- visited.add(startId);
147
- recursionStack.add(startId);
148
-
149
- const task = tasks[startId];
150
- if (!task) {
151
- return { hasCycle: false, cycle: null };
152
- }
153
-
154
- // Check all dependencies (blockedBy)
155
- const deps = task.blockedBy || [];
156
- for (const depId of deps) {
157
- if (!visited.has(depId)) {
158
- const result = detectCycle(tasks, depId, visited, recursionStack);
159
- if (result.hasCycle) {
160
- return result;
161
- }
162
- } else if (recursionStack.has(depId)) {
163
- // Found cycle - reconstruct it
164
- const cycleNodes = [depId, startId];
165
- return { hasCycle: true, cycle: cycleNodes };
166
- }
167
- }
168
-
169
- recursionStack.delete(startId);
170
- return { hasCycle: false, cycle: null };
171
- }
172
-
173
- /**
174
- * Validate entire task graph for cycles
175
- * @param {Object} tasks - Map of task ID to task object
176
- * @returns {{ valid: boolean, cycles: string[][] }}
177
- */
178
- function validateDAG(tasks) {
179
- const visited = new Set();
180
- const cycles = [];
181
-
182
- for (const taskId of Object.keys(tasks)) {
183
- if (!visited.has(taskId)) {
184
- const result = detectCycle(tasks, taskId, visited, new Set());
185
- if (result.hasCycle && result.cycle) {
186
- cycles.push(result.cycle);
187
- }
188
- }
189
- }
190
-
191
- return {
192
- valid: cycles.length === 0,
193
- cycles,
194
- };
195
- }
196
-
197
- /**
198
- * Get topological sort of tasks (dependency order)
199
- * @param {Object} tasks - Map of task ID to task object
200
- * @returns {{ sorted: string[], valid: boolean }}
201
- */
202
- function topologicalSort(tasks) {
203
- const inDegree = {};
204
- const adjList = {};
205
-
206
- // Initialize
207
- for (const taskId of Object.keys(tasks)) {
208
- inDegree[taskId] = 0;
209
- adjList[taskId] = [];
210
- }
211
-
212
- // Build adjacency list and in-degrees
213
- for (const [taskId, task] of Object.entries(tasks)) {
214
- const deps = task.blockedBy || [];
215
- for (const depId of deps) {
216
- if (adjList[depId]) {
217
- adjList[depId].push(taskId);
218
- inDegree[taskId]++;
219
- }
220
- }
221
- }
222
-
223
- // Kahn's algorithm
224
- const queue = [];
225
- for (const taskId of Object.keys(tasks)) {
226
- if (inDegree[taskId] === 0) {
227
- queue.push(taskId);
228
- }
229
- }
230
-
231
- const sorted = [];
232
- while (queue.length > 0) {
233
- const node = queue.shift();
234
- sorted.push(node);
235
-
236
- for (const neighbor of adjList[node] || []) {
237
- inDegree[neighbor]--;
238
- if (inDegree[neighbor] === 0) {
239
- queue.push(neighbor);
240
- }
241
- }
242
- }
243
-
244
- // If sorted doesn't include all tasks, there's a cycle
245
- const valid = sorted.length === Object.keys(tasks).length;
246
-
247
- return { sorted, valid };
248
- }
249
-
250
- // ============================================================================
251
- // State Machine
252
- // ============================================================================
253
-
254
- /**
255
- * Check if a state is valid
256
- * @param {string} state - State to check
257
- * @returns {boolean}
258
- */
259
- function isValidState(state) {
260
- return TASK_STATES.includes(state);
261
- }
262
-
263
- /**
264
- * Check if a transition is valid
265
- * @param {string} fromState - Current state
266
- * @param {string} toState - Target state
267
- * @returns {boolean}
268
- */
269
- function isValidTransition(fromState, toState) {
270
- if (fromState === toState) return true; // No-op
271
- if (!TRANSITIONS[fromState]) return false;
272
- return TRANSITIONS[fromState].includes(toState);
273
- }
274
-
275
- /**
276
- * Get valid transitions from a state
277
- * @param {string} fromState - Current state
278
- * @returns {string[]}
279
- */
280
- function getValidTransitions(fromState) {
281
- return TRANSITIONS[fromState] || [];
282
- }
283
-
284
- /**
285
- * Check if state is terminal
286
- * @param {string} state - State to check
287
- * @returns {boolean}
288
- */
289
- function isTerminalState(state) {
290
- return TERMINAL_STATES.includes(state);
291
- }
292
-
293
- // ============================================================================
294
- // Lock Management
295
- // ============================================================================
296
-
297
- /**
298
- * Simple file-based lock for concurrent access
299
- */
300
- class FileLock {
301
- constructor(lockPath, options = {}) {
302
- this.lockPath = lockPath;
303
- this.timeout = options.timeout || LOCK_TIMEOUT_MS;
304
- this.staleMs = options.staleMs || LOCK_STALE_MS;
305
- this.held = false;
306
- }
307
-
308
- /**
309
- * Acquire lock with timeout
310
- * @returns {boolean} True if lock acquired
311
- */
312
- acquire() {
313
- const startTime = Date.now();
314
-
315
- while (Date.now() - startTime < this.timeout) {
316
- // Check if existing lock is stale
317
- if (fs.existsSync(this.lockPath)) {
318
- try {
319
- const stat = fs.statSync(this.lockPath);
320
- const age = Date.now() - stat.mtimeMs;
321
- if (age > this.staleMs) {
322
- // Stale lock - remove it
323
- fs.unlinkSync(this.lockPath);
324
- } else {
325
- // Lock held by another process - wait
326
- this._sleep(50);
327
- continue;
328
- }
329
- } catch (e) {
330
- // Lock file gone, continue to try acquiring
331
- }
332
- }
333
-
334
- // Try to acquire
335
- let fd = null;
336
- try {
337
- const dir = path.dirname(this.lockPath);
338
- if (!fs.existsSync(dir)) {
339
- fs.mkdirSync(dir, { recursive: true });
340
- }
341
-
342
- // Use exclusive flag to prevent race
343
- fd = fs.openSync(this.lockPath, 'wx');
344
- const lockInfo = {
345
- pid: process.pid,
346
- hostname: os.hostname(),
347
- acquired: new Date().toISOString(),
348
- };
349
- fs.writeSync(fd, JSON.stringify(lockInfo));
350
- fs.closeSync(fd);
351
- fd = null;
352
- this.held = true;
353
- return true;
354
- } catch (e) {
355
- // Clean up fd if still open
356
- if (fd !== null) {
357
- try {
358
- fs.closeSync(fd);
359
- } catch (_) {
360
- /* ignore */
361
- }
362
- }
363
- if (e.code === 'EEXIST') {
364
- // Another process got the lock
365
- this._sleep(50);
366
- continue;
367
- }
368
- throw e;
369
- }
370
- }
371
-
372
- return false;
373
- }
374
-
375
- /**
376
- * Release lock
377
- */
378
- release() {
379
- if (this.held && fs.existsSync(this.lockPath)) {
380
- try {
381
- fs.unlinkSync(this.lockPath);
382
- } catch (e) {
383
- // Ignore - lock may already be removed
384
- }
385
- this.held = false;
386
- }
387
- }
388
-
389
- /**
390
- * Synchronous sleep
391
- * @param {number} ms - Milliseconds to sleep
392
- */
393
- _sleep(ms) {
394
- const end = Date.now() + ms;
395
- while (Date.now() < end) {
396
- // Busy wait - not ideal but works for short durations
397
- }
398
- }
399
- }
400
-
401
- // ============================================================================
402
- // Task Registry
403
- // ============================================================================
404
-
405
- /**
406
- * TaskRegistry - Event-driven task management with state machine
407
- */
408
- class TaskRegistry extends EventEmitter {
409
- constructor(options = {}) {
410
- super();
411
-
412
- this.rootDir = options.rootDir || findProjectRoot();
413
- this.statePath = path.join(this.rootDir, options.statePath || DEFAULT_STATE_PATH);
414
- this.lockPath = path.join(path.dirname(this.statePath), LOCK_FILE_NAME);
415
-
416
- // Cache
417
- this._cache = null;
418
- this._dirty = false;
419
- }
420
-
421
- // --------------------------------------------------------------------------
422
- // State Persistence
423
- // --------------------------------------------------------------------------
424
-
425
- /**
426
- * Load task state from disk
427
- * @returns {Object} Task state
428
- */
429
- load() {
430
- if (this._cache && !this._dirty) {
431
- return this._cache;
432
- }
433
-
434
- const defaultState = this._createDefaultState();
435
-
436
- if (!fs.existsSync(this.statePath)) {
437
- this._cache = defaultState;
438
- this.save();
439
- return this._cache;
440
- }
441
-
442
- try {
443
- const content = fs.readFileSync(this.statePath, 'utf8');
444
- this._cache = safeJSONParse(content, defaultState);
445
- this._dirty = false;
446
- this.emit('loaded', { state: this._cache });
447
- return this._cache;
448
- } catch (e) {
449
- this.emit('error', { type: 'load', error: e.message });
450
- this._cache = defaultState;
451
- return this._cache;
452
- }
453
- }
454
-
455
- /**
456
- * Save task state to disk (atomic)
457
- */
458
- save() {
459
- if (!this._cache) return;
460
-
461
- this._cache.updated_at = new Date().toISOString();
462
- const content = JSON.stringify(this._cache, null, 2) + '\n';
463
-
464
- try {
465
- atomicWrite(this.statePath, content);
466
- this._dirty = false;
467
- this.emit('saved', { state: this._cache });
468
- } catch (e) {
469
- this.emit('error', { type: 'save', error: e.message });
470
- throw e;
471
- }
472
- }
473
-
474
- /**
475
- * Create default state structure
476
- * @returns {Object}
477
- */
478
- _createDefaultState() {
479
- return {
480
- schema_version: '1.0.0',
481
- created_at: new Date().toISOString(),
482
- updated_at: new Date().toISOString(),
483
- tasks: {},
484
- task_groups: {},
485
- audit_trail: [],
486
- };
487
- }
488
-
489
- // --------------------------------------------------------------------------
490
- // CRUD Operations
491
- // --------------------------------------------------------------------------
492
-
493
- /**
494
- * Create a new task
495
- * @param {Object} taskData - Task data
496
- * @returns {{ success: boolean, task?: Object, error?: string }}
497
- */
498
- create(taskData) {
499
- const lock = new FileLock(this.lockPath);
500
- if (!lock.acquire()) {
501
- return { success: false, error: 'Could not acquire lock' };
502
- }
503
-
504
- try {
505
- const state = this.load();
506
-
507
- // Generate ID if not provided
508
- const taskId = taskData.id || generateTaskId();
509
-
510
- // Check for duplicate ID
511
- if (state.tasks[taskId]) {
512
- return { success: false, error: `Task ${taskId} already exists` };
513
- }
514
-
515
- // Validate dependencies won't create cycle
516
- if (taskData.blockedBy && taskData.blockedBy.length > 0) {
517
- // Temporarily add task to check for cycles
518
- const testTasks = {
519
- ...state.tasks,
520
- [taskId]: { ...taskData, blockedBy: taskData.blockedBy },
521
- };
522
- const dagResult = validateDAG(testTasks);
523
- if (!dagResult.valid) {
524
- return {
525
- success: false,
526
- error: `Adding this task would create circular dependency: ${dagResult.cycles[0].join(' → ')}`,
527
- };
528
- }
529
- }
530
-
531
- // Create task object
532
- const task = {
533
- id: taskId,
534
- description: taskData.description || '',
535
- prompt: taskData.prompt || '',
536
- subagent_type: taskData.subagent_type || null,
537
- state: 'queued',
538
- created_at: new Date().toISOString(),
539
- updated_at: new Date().toISOString(),
540
- // Dependencies
541
- blockedBy: taskData.blockedBy || [],
542
- blocks: taskData.blocks || [],
543
- // Execution config
544
- join_strategy: taskData.join_strategy || 'all',
545
- on_failure: taskData.on_failure || 'fail-fast',
546
- run_in_background: taskData.run_in_background || false,
547
- // Link to story (optional)
548
- story_id: taskData.story_id || null,
549
- // Results (populated on completion)
550
- result: null,
551
- error: null,
552
- // Metadata
553
- metadata: taskData.metadata || {},
554
- };
555
-
556
- // Automatically set to blocked if has unmet dependencies
557
- if (task.blockedBy.length > 0) {
558
- const unmetDeps = task.blockedBy.filter(depId => {
559
- const dep = state.tasks[depId];
560
- return !dep || dep.state !== 'completed';
561
- });
562
- if (unmetDeps.length > 0) {
563
- task.state = 'blocked';
564
- }
565
- }
566
-
567
- // Update reverse dependencies (blocks)
568
- for (const blockerId of task.blockedBy) {
569
- if (state.tasks[blockerId]) {
570
- if (!state.tasks[blockerId].blocks) {
571
- state.tasks[blockerId].blocks = [];
572
- }
573
- if (!state.tasks[blockerId].blocks.includes(taskId)) {
574
- state.tasks[blockerId].blocks.push(taskId);
575
- }
576
- }
577
- }
578
-
579
- // Store task
580
- state.tasks[taskId] = task;
581
- this._dirty = true;
582
- this.save();
583
-
584
- // Emit event
585
- this.emit('created', { task });
586
-
587
- return { success: true, task };
588
- } finally {
589
- lock.release();
590
- }
591
- }
592
-
593
- /**
594
- * Get a task by ID
595
- * @param {string} taskId - Task ID
596
- * @returns {Object|null}
597
- */
598
- get(taskId) {
599
- const state = this.load();
600
- return state.tasks[taskId] || null;
601
- }
602
-
603
- /**
604
- * Get all tasks
605
- * @param {Object} [filter] - Optional filter
606
- * @returns {Object[]}
607
- */
608
- getAll(filter = {}) {
609
- const state = this.load();
610
- let tasks = Object.values(state.tasks);
611
-
612
- // Apply filters
613
- if (filter.state) {
614
- tasks = tasks.filter(t => t.state === filter.state);
615
- }
616
- if (filter.story_id) {
617
- tasks = tasks.filter(t => t.story_id === filter.story_id);
618
- }
619
- if (filter.subagent_type) {
620
- tasks = tasks.filter(t => t.subagent_type === filter.subagent_type);
621
- }
622
-
623
- return tasks;
624
- }
625
-
626
- /**
627
- * Update a task
628
- * @param {string} taskId - Task ID
629
- * @param {Object} updates - Fields to update
630
- * @returns {{ success: boolean, task?: Object, error?: string }}
631
- */
632
- update(taskId, updates) {
633
- const lock = new FileLock(this.lockPath);
634
- if (!lock.acquire()) {
635
- return { success: false, error: 'Could not acquire lock' };
636
- }
637
-
638
- try {
639
- const state = this.load();
640
- const task = state.tasks[taskId];
641
-
642
- if (!task) {
643
- return { success: false, error: `Task ${taskId} not found` };
644
- }
645
-
646
- // Handle state transition
647
- if (updates.state && updates.state !== task.state) {
648
- if (!isValidTransition(task.state, updates.state)) {
649
- return {
650
- success: false,
651
- error: `Invalid transition: ${task.state} → ${updates.state}. Valid: ${getValidTransitions(task.state).join(', ') || 'none'}`,
652
- };
653
- }
654
-
655
- // Log to audit trail
656
- state.audit_trail.push({
657
- task_id: taskId,
658
- from_state: task.state,
659
- to_state: updates.state,
660
- at: new Date().toISOString(),
661
- reason: updates.reason || null,
662
- });
663
- }
664
-
665
- // Handle dependency updates
666
- if (updates.blockedBy) {
667
- // Validate no cycles
668
- const testTask = { ...task, blockedBy: updates.blockedBy };
669
- const testTasks = { ...state.tasks, [taskId]: testTask };
670
- const dagResult = validateDAG(testTasks);
671
- if (!dagResult.valid) {
672
- return {
673
- success: false,
674
- error: `Update would create circular dependency: ${dagResult.cycles[0].join(' → ')}`,
675
- };
676
- }
677
- }
678
-
679
- // Apply updates
680
- const changedFields = [];
681
- for (const [key, value] of Object.entries(updates)) {
682
- if (key === 'reason') continue; // reason is for audit only
683
- if (JSON.stringify(task[key]) !== JSON.stringify(value)) {
684
- task[key] = value;
685
- changedFields.push(key);
686
- }
687
- }
688
-
689
- if (changedFields.length > 0) {
690
- task.updated_at = new Date().toISOString();
691
-
692
- // If transitioning to completed, unblock dependent tasks AFTER state is applied
693
- if (changedFields.includes('state') && task.state === 'completed') {
694
- this._unblockDependents(state, taskId);
695
- }
696
-
697
- this._dirty = true;
698
- this.save();
699
-
700
- this.emit('updated', { task, changes: changedFields });
701
- }
702
-
703
- return { success: true, task };
704
- } finally {
705
- lock.release();
706
- }
707
- }
708
-
709
- /**
710
- * Delete a task
711
- * @param {string} taskId - Task ID
712
- * @returns {{ success: boolean, error?: string }}
713
- */
714
- delete(taskId) {
715
- const lock = new FileLock(this.lockPath);
716
- if (!lock.acquire()) {
717
- return { success: false, error: 'Could not acquire lock' };
718
- }
719
-
720
- try {
721
- const state = this.load();
722
- const task = state.tasks[taskId];
723
-
724
- if (!task) {
725
- return { success: false, error: `Task ${taskId} not found` };
726
- }
727
-
728
- // Remove from blockedBy lists of other tasks
729
- for (const otherTask of Object.values(state.tasks)) {
730
- if (otherTask.blockedBy) {
731
- otherTask.blockedBy = otherTask.blockedBy.filter(id => id !== taskId);
732
- }
733
- if (otherTask.blocks) {
734
- otherTask.blocks = otherTask.blocks.filter(id => id !== taskId);
735
- }
736
- }
737
-
738
- delete state.tasks[taskId];
739
- this._dirty = true;
740
- this.save();
741
-
742
- this.emit('deleted', { taskId });
743
-
744
- return { success: true };
745
- } finally {
746
- lock.release();
747
- }
748
- }
749
-
750
- // --------------------------------------------------------------------------
751
- // State Transitions
752
- // --------------------------------------------------------------------------
753
-
754
- /**
755
- * Transition a task to a new state
756
- * @param {string} taskId - Task ID
757
- * @param {string} toState - Target state
758
- * @param {Object} [options] - Options
759
- * @returns {{ success: boolean, task?: Object, error?: string }}
760
- */
761
- transition(taskId, toState, options = {}) {
762
- const { reason = null, result = null, error = null } = options;
763
-
764
- const updates = { state: toState, reason };
765
- if (result !== null) updates.result = result;
766
- if (error !== null) updates.error = error;
767
-
768
- return this.update(taskId, updates);
769
- }
770
-
771
- /**
772
- * Mark task as running
773
- * @param {string} taskId - Task ID
774
- * @returns {{ success: boolean, task?: Object, error?: string }}
775
- */
776
- start(taskId) {
777
- return this.transition(taskId, 'running');
778
- }
779
-
780
- /**
781
- * Mark task as completed
782
- * @param {string} taskId - Task ID
783
- * @param {*} result - Task result
784
- * @returns {{ success: boolean, task?: Object, error?: string }}
785
- */
786
- complete(taskId, result = null) {
787
- return this.transition(taskId, 'completed', { result });
788
- }
789
-
790
- /**
791
- * Mark task as failed
792
- * @param {string} taskId - Task ID
793
- * @param {string} error - Error message
794
- * @returns {{ success: boolean, task?: Object, error?: string }}
795
- */
796
- fail(taskId, error) {
797
- return this.transition(taskId, 'failed', { error });
798
- }
799
-
800
- /**
801
- * Mark task as blocked
802
- * @param {string} taskId - Task ID
803
- * @param {string} reason - Reason for blocking
804
- * @returns {{ success: boolean, task?: Object, error?: string }}
805
- */
806
- block(taskId, reason) {
807
- return this.transition(taskId, 'blocked', { reason });
808
- }
809
-
810
- /**
811
- * Cancel a task
812
- * @param {string} taskId - Task ID
813
- * @param {string} reason - Reason for cancellation
814
- * @returns {{ success: boolean, task?: Object, error?: string }}
815
- */
816
- cancel(taskId, reason) {
817
- return this.transition(taskId, 'cancelled', { reason });
818
- }
819
-
820
- /**
821
- * Retry a failed task
822
- * @param {string} taskId - Task ID
823
- * @returns {{ success: boolean, task?: Object, error?: string }}
824
- */
825
- retry(taskId) {
826
- const task = this.get(taskId);
827
- if (!task) {
828
- return { success: false, error: `Task ${taskId} not found` };
829
- }
830
- if (task.state !== 'failed') {
831
- return { success: false, error: `Can only retry failed tasks, current state: ${task.state}` };
832
- }
833
- return this.transition(taskId, 'queued', { reason: 'retry' });
834
- }
835
-
836
- // --------------------------------------------------------------------------
837
- // Dependency Management
838
- // --------------------------------------------------------------------------
839
-
840
- /**
841
- * Add a dependency (blockedBy)
842
- * @param {string} taskId - Task that will be blocked
843
- * @param {string} blockerId - Task that blocks
844
- * @returns {{ success: boolean, error?: string }}
845
- */
846
- addDependency(taskId, blockerId) {
847
- const task = this.get(taskId);
848
- if (!task) {
849
- return { success: false, error: `Task ${taskId} not found` };
850
- }
851
-
852
- const blockedBy = [...(task.blockedBy || [])];
853
- if (!blockedBy.includes(blockerId)) {
854
- blockedBy.push(blockerId);
855
- }
856
-
857
- return this.update(taskId, { blockedBy });
858
- }
859
-
860
- /**
861
- * Remove a dependency
862
- * @param {string} taskId - Task ID
863
- * @param {string} blockerId - Blocker to remove
864
- * @returns {{ success: boolean, error?: string }}
865
- */
866
- removeDependency(taskId, blockerId) {
867
- const task = this.get(taskId);
868
- if (!task) {
869
- return { success: false, error: `Task ${taskId} not found` };
870
- }
871
-
872
- const blockedBy = (task.blockedBy || []).filter(id => id !== blockerId);
873
- return this.update(taskId, { blockedBy });
874
- }
875
-
876
- /**
877
- * Get tasks that are ready to run (queued with no unmet dependencies)
878
- * @returns {Object[]}
879
- */
880
- getReadyTasks() {
881
- const state = this.load();
882
- const ready = [];
883
-
884
- for (const task of Object.values(state.tasks)) {
885
- if (task.state !== 'queued') continue;
886
-
887
- // Check all dependencies are complete
888
- const unmetDeps = (task.blockedBy || []).filter(depId => {
889
- const dep = state.tasks[depId];
890
- return !dep || dep.state !== 'completed';
891
- });
892
-
893
- if (unmetDeps.length === 0) {
894
- ready.push(task);
895
- }
896
- }
897
-
898
- return ready;
899
- }
900
-
901
- /**
902
- * Get the dependency graph for visualization
903
- * @returns {{ nodes: Object[], edges: Object[] }}
904
- */
905
- getDependencyGraph() {
906
- const state = this.load();
907
- const nodes = [];
908
- const edges = [];
909
-
910
- for (const task of Object.values(state.tasks)) {
911
- nodes.push({
912
- id: task.id,
913
- label: task.description || task.id,
914
- state: task.state,
915
- subagent_type: task.subagent_type,
916
- });
917
-
918
- for (const blockerId of task.blockedBy || []) {
919
- edges.push({
920
- from: blockerId,
921
- to: task.id,
922
- });
923
- }
924
- }
925
-
926
- return { nodes, edges };
927
- }
928
-
929
- /**
930
- * Unblock dependent tasks when a task completes
931
- * @param {Object} state - Current state
932
- * @param {string} completedTaskId - ID of completed task
933
- */
934
- _unblockDependents(state, completedTaskId) {
935
- const completedTask = state.tasks[completedTaskId];
936
- if (!completedTask) return;
937
-
938
- // Find tasks that were blocked by this one
939
- for (const dependentId of completedTask.blocks || []) {
940
- const dependent = state.tasks[dependentId];
941
- if (!dependent || dependent.state !== 'blocked') continue;
942
-
943
- // Check if all blockers are now complete
944
- const unmetDeps = (dependent.blockedBy || []).filter(depId => {
945
- const dep = state.tasks[depId];
946
- return !dep || dep.state !== 'completed';
947
- });
948
-
949
- if (unmetDeps.length === 0) {
950
- dependent.state = 'queued';
951
- dependent.updated_at = new Date().toISOString();
952
- state.audit_trail.push({
953
- task_id: dependentId,
954
- from_state: 'blocked',
955
- to_state: 'queued',
956
- at: new Date().toISOString(),
957
- reason: `Unblocked: ${completedTaskId} completed`,
958
- });
959
- this.emit('unblocked', { task: dependent, unblockedBy: completedTaskId });
960
- }
961
- }
962
- }
963
-
964
- // --------------------------------------------------------------------------
965
- // Task Groups
966
- // --------------------------------------------------------------------------
967
-
968
- /**
969
- * Create a task group for parallel execution
970
- * @param {Object} groupData - Group configuration
971
- * @returns {{ success: boolean, group?: Object, error?: string }}
972
- */
973
- createGroup(groupData) {
974
- const lock = new FileLock(this.lockPath);
975
- if (!lock.acquire()) {
976
- return { success: false, error: 'Could not acquire lock' };
977
- }
978
-
979
- try {
980
- const state = this.load();
981
-
982
- const groupId = groupData.id || `group-${Date.now().toString(36)}`;
983
-
984
- if (state.task_groups[groupId]) {
985
- return { success: false, error: `Group ${groupId} already exists` };
986
- }
987
-
988
- const group = {
989
- id: groupId,
990
- name: groupData.name || groupId,
991
- task_ids: groupData.task_ids || [],
992
- join_strategy: groupData.join_strategy || 'all',
993
- on_failure: groupData.on_failure || 'fail-fast',
994
- created_at: new Date().toISOString(),
995
- state: 'pending', // pending, running, completed, failed
996
- };
997
-
998
- state.task_groups[groupId] = group;
999
- this._dirty = true;
1000
- this.save();
1001
-
1002
- this.emit('group_created', { group });
1003
-
1004
- return { success: true, group };
1005
- } finally {
1006
- lock.release();
1007
- }
1008
- }
1009
-
1010
- /**
1011
- * Get group status
1012
- * @param {string} groupId - Group ID
1013
- * @returns {Object|null}
1014
- */
1015
- getGroupStatus(groupId) {
1016
- const state = this.load();
1017
- const group = state.task_groups[groupId];
1018
- if (!group) return null;
1019
-
1020
- const tasks = group.task_ids.map(id => state.tasks[id]).filter(Boolean);
1021
- const completed = tasks.filter(t => t.state === 'completed').length;
1022
- const failed = tasks.filter(t => t.state === 'failed').length;
1023
- const running = tasks.filter(t => t.state === 'running').length;
1024
-
1025
- return {
1026
- ...group,
1027
- total: tasks.length,
1028
- completed,
1029
- failed,
1030
- running,
1031
- pending: tasks.length - completed - failed - running,
1032
- };
1033
- }
1034
-
1035
- // --------------------------------------------------------------------------
1036
- // Metrics & Queries
1037
- // --------------------------------------------------------------------------
1038
-
1039
- /**
1040
- * Get task statistics
1041
- * @returns {Object}
1042
- */
1043
- getStats() {
1044
- const state = this.load();
1045
- const tasks = Object.values(state.tasks);
1046
-
1047
- const byState = {};
1048
- for (const s of TASK_STATES) {
1049
- byState[s] = tasks.filter(t => t.state === s).length;
1050
- }
1051
-
1052
- const byAgent = {};
1053
- for (const task of tasks) {
1054
- const agent = task.subagent_type || 'unknown';
1055
- byAgent[agent] = (byAgent[agent] || 0) + 1;
1056
- }
1057
-
1058
- return {
1059
- total: tasks.length,
1060
- by_state: byState,
1061
- by_agent: byAgent,
1062
- groups: Object.keys(state.task_groups).length,
1063
- };
1064
- }
1065
-
1066
- /**
1067
- * Get tasks for a story
1068
- * @param {string} storyId - Story ID
1069
- * @returns {Object[]}
1070
- */
1071
- getTasksForStory(storyId) {
1072
- return this.getAll({ story_id: storyId });
1073
- }
1074
-
1075
- /**
1076
- * Get audit trail
1077
- * @param {Object} [filter] - Optional filter
1078
- * @returns {Object[]}
1079
- */
1080
- getAuditTrail(filter = {}) {
1081
- const state = this.load();
1082
- let trail = [...state.audit_trail];
1083
-
1084
- if (filter.task_id) {
1085
- trail = trail.filter(e => e.task_id === filter.task_id);
1086
- }
1087
-
1088
- return trail;
1089
- }
1090
-
1091
- /**
1092
- * Clear completed/cancelled tasks older than threshold
1093
- * @param {number} [maxAgeMs] - Max age in milliseconds (default: 7 days)
1094
- * @returns {{ cleared: number }}
1095
- */
1096
- cleanup(maxAgeMs = 7 * 24 * 60 * 60 * 1000) {
1097
- const lock = new FileLock(this.lockPath);
1098
- if (!lock.acquire()) {
1099
- return { cleared: 0 };
1100
- }
1101
-
1102
- try {
1103
- const state = this.load();
1104
- const now = Date.now();
1105
- let cleared = 0;
1106
-
1107
- for (const [taskId, task] of Object.entries(state.tasks)) {
1108
- if (!DONE_STATES.includes(task.state)) continue;
1109
-
1110
- const updatedAt = new Date(task.updated_at).getTime();
1111
- if (now - updatedAt > maxAgeMs) {
1112
- delete state.tasks[taskId];
1113
- cleared++;
1114
- }
1115
- }
1116
-
1117
- if (cleared > 0) {
1118
- this._dirty = true;
1119
- this.save();
1120
- this.emit('cleanup', { cleared });
1121
- }
1122
-
1123
- return { cleared };
1124
- } finally {
1125
- lock.release();
1126
- }
1127
- }
1128
- }
1129
-
1130
- // ============================================================================
1131
- // Singleton & Factory
1132
- // ============================================================================
1133
-
1134
- let _instance = null;
1135
-
1136
- /**
1137
- * Get singleton registry instance
1138
- * @param {Object} [options] - Options
1139
- * @returns {TaskRegistry}
1140
- */
1141
- function getTaskRegistry(options = {}) {
1142
- if (!_instance || options.forceNew) {
1143
- _instance = new TaskRegistry(options);
1144
- }
1145
- return _instance;
1146
- }
1147
-
1148
- /**
1149
- * Reset singleton (for testing)
1150
- */
1151
- function resetTaskRegistry() {
1152
- _instance = null;
1153
- }
1154
-
1155
- // ============================================================================
1156
- // Exports
1157
- // ============================================================================
1158
-
1159
- module.exports = {
1160
- // Constants
1161
- TASK_STATES,
1162
- TERMINAL_STATES,
1163
- DONE_STATES,
1164
- TRANSITIONS,
1165
- JOIN_STRATEGIES,
1166
- FAILURE_POLICIES,
1167
-
1168
- // State machine
1169
- isValidState,
1170
- isValidTransition,
1171
- getValidTransitions,
1172
- isTerminalState,
1173
-
1174
- // DAG validation
1175
- detectCycle,
1176
- validateDAG,
1177
- topologicalSort,
1178
-
1179
- // Utilities
1180
- generateTaskId,
1181
- findProjectRoot,
1182
- atomicWrite,
1183
-
1184
- // Classes
1185
- TaskRegistry,
1186
- FileLock,
1187
-
1188
- // Factory
1189
- getTaskRegistry,
1190
- resetTaskRegistry,
1191
- };