@jaimevalasek/aioson 1.8.0 → 1.9.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 (974) hide show
  1. package/CHANGELOG.md +595 -595
  2. package/CODE_OF_CONDUCT.md +12 -12
  3. package/CONTRIBUTING.md +13 -13
  4. package/LICENSE +661 -661
  5. package/README.md +919 -919
  6. package/bin/aioson.js +4 -4
  7. package/docs/design-previews/aurora-command-ui-website.html +884 -884
  8. package/docs/design-previews/aurora-command-ui.html +682 -682
  9. package/docs/design-previews/bold-editorial-ui-website.html +658 -658
  10. package/docs/design-previews/bold-editorial-ui.html +717 -717
  11. package/docs/design-previews/clean-saas-ui-website.html +1202 -1202
  12. package/docs/design-previews/clean-saas-ui.html +549 -549
  13. package/docs/design-previews/cognitive-core-ui-website.html +1009 -1009
  14. package/docs/design-previews/cognitive-core-ui.html +463 -463
  15. package/docs/design-previews/glassmorphism-ui-website.html +572 -572
  16. package/docs/design-previews/glassmorphism-ui.html +886 -886
  17. package/docs/design-previews/index.html +699 -699
  18. package/docs/design-previews/interface-design-website.html +1187 -1187
  19. package/docs/design-previews/interface-design.html +513 -513
  20. package/docs/design-previews/neo-brutalist-ui-website.html +621 -621
  21. package/docs/design-previews/neo-brutalist-ui.html +797 -797
  22. package/docs/design-previews/premium-command-center-ui-website.html +1217 -1217
  23. package/docs/design-previews/premium-command-center-ui.html +552 -552
  24. package/docs/design-previews/pt.squarespace.com-homepage.html +889 -889
  25. package/docs/design-previews/warm-craft-ui-website.html +684 -684
  26. package/docs/design-previews/warm-craft-ui.html +739 -739
  27. package/docs/en/1-understand/ecosystem-map.md +228 -0
  28. package/docs/en/1-understand/glossary.md +288 -0
  29. package/docs/en/1-understand/what-is-aioson.md +94 -0
  30. package/docs/en/1-understand/why-it-exists.md +106 -0
  31. package/docs/en/2-start/existing-project.md +246 -0
  32. package/docs/en/2-start/first-project.md +307 -0
  33. package/docs/en/2-start/initial-decisions.md +223 -0
  34. package/docs/en/3-recipes/README.md +28 -0
  35. package/docs/en/3-recipes/continuity-between-sessions.md +303 -0
  36. package/docs/en/3-recipes/from-idea-to-prd-via-briefing.md +235 -0
  37. package/docs/en/3-recipes/full-feature-with-sheldon.md +338 -0
  38. package/docs/en/4-agents/README.md +56 -0
  39. package/docs/en/5-reference/README.md +60 -0
  40. package/docs/en/{cli-reference.md → 5-reference/cli-reference.md} +639 -464
  41. package/docs/en/{i18n.md → 5-reference/i18n.md} +52 -52
  42. package/docs/en/{json-schemas.md → 5-reference/json-schemas.md} +41 -41
  43. package/docs/en/{mcp.md → 5-reference/mcp.md} +56 -56
  44. package/docs/en/{parallel.md → 5-reference/parallel.md} +82 -82
  45. package/docs/en/{qa-browser.md → 5-reference/qa-browser.md} +339 -339
  46. package/docs/en/{release-flow.md → 5-reference/release-flow.md} +22 -22
  47. package/docs/en/{release-notes-template.md → 5-reference/release-notes-template.md} +41 -41
  48. package/docs/en/{release.md → 5-reference/release.md} +28 -28
  49. package/docs/en/{schemas → 5-reference/schemas}/agent-prompt.schema.json +17 -17
  50. package/docs/en/{schemas → 5-reference/schemas}/agents.schema.json +32 -32
  51. package/docs/en/{schemas → 5-reference/schemas}/context-validate.schema.json +36 -36
  52. package/docs/en/{schemas → 5-reference/schemas}/doctor.schema.json +89 -89
  53. package/docs/en/{schemas → 5-reference/schemas}/error.schema.json +24 -24
  54. package/docs/en/{schemas → 5-reference/schemas}/i18n-add.schema.json +15 -15
  55. package/docs/en/{schemas → 5-reference/schemas}/index.json +126 -126
  56. package/docs/en/{schemas → 5-reference/schemas}/info.schema.json +39 -39
  57. package/docs/en/{schemas → 5-reference/schemas}/init.schema.json +48 -48
  58. package/docs/en/{schemas → 5-reference/schemas}/install.schema.json +60 -60
  59. package/docs/en/{schemas → 5-reference/schemas}/locale-apply.schema.json +30 -30
  60. package/docs/en/{schemas → 5-reference/schemas}/mcp-doctor.schema.json +95 -95
  61. package/docs/en/{schemas → 5-reference/schemas}/mcp-init.schema.json +122 -122
  62. package/docs/en/{schemas → 5-reference/schemas}/package-test.schema.json +24 -24
  63. package/docs/en/{schemas → 5-reference/schemas}/parallel-assign.schema.json +66 -66
  64. package/docs/en/{schemas → 5-reference/schemas}/parallel-doctor.schema.json +122 -122
  65. package/docs/en/{schemas → 5-reference/schemas}/parallel-guard.schema.json +63 -63
  66. package/docs/en/{schemas → 5-reference/schemas}/parallel-init.schema.json +53 -53
  67. package/docs/en/{schemas → 5-reference/schemas}/parallel-merge.schema.json +84 -84
  68. package/docs/en/{schemas → 5-reference/schemas}/parallel-status.schema.json +184 -184
  69. package/docs/en/{schemas → 5-reference/schemas}/setup-context.schema.json +39 -39
  70. package/docs/en/{schemas → 5-reference/schemas}/smoke.schema.json +23 -23
  71. package/docs/en/{schemas → 5-reference/schemas}/update.schema.json +48 -48
  72. package/docs/en/{schemas → 5-reference/schemas}/workflow-plan.schema.json +30 -30
  73. package/docs/en/{squad-dashboard.md → 5-reference/squad-dashboard.md} +372 -372
  74. package/docs/en/{web3.md → 5-reference/web3.md} +54 -54
  75. package/docs/en/README.md +115 -0
  76. package/docs/en/active-learning-loop/README.md +117 -0
  77. package/docs/en/active-learning-loop/active-learning-loop.md +117 -0
  78. package/docs/en/active-learning-loop/cli-commands.md +320 -0
  79. package/docs/en/active-learning-loop/diagrams.md +225 -0
  80. package/docs/en/active-learning-loop/doctor-checks.md +151 -0
  81. package/docs/en/active-learning-loop/how-to-use.md +313 -0
  82. package/docs/en/active-learning-loop/troubleshooting.md +283 -0
  83. package/docs/en/deyvin-subtask-scout/README.md +109 -0
  84. package/docs/en/deyvin-subtask-scout/cli-commands.md +248 -0
  85. package/docs/en/deyvin-subtask-scout/diagrams.md +124 -0
  86. package/docs/en/deyvin-subtask-scout/how-to-use.md +221 -0
  87. package/docs/en/deyvin-subtask-scout/sub-task-scout.md +115 -0
  88. package/docs/en/deyvin-subtask-scout/troubleshooting.md +184 -0
  89. package/docs/integrations/apps-publish-marketplace.md +94 -94
  90. package/docs/integrations/sdlc-genius-boundary.md +76 -76
  91. package/docs/integrations/sdlc-genius-eval-matrix.md +75 -75
  92. package/docs/integrations/sdlc-genius-install-checklist.md +93 -93
  93. package/docs/integrations/sdlc-genius-review-samples.md +86 -86
  94. package/docs/openclaw-bridge.md +308 -308
  95. package/docs/pt/1-entender/glossario.md +288 -0
  96. package/docs/pt/1-entender/mapa-do-ecossistema.md +228 -0
  97. package/docs/pt/1-entender/o-que-e-aioson.md +94 -0
  98. package/docs/pt/1-entender/por-que-existe.md +107 -0
  99. package/docs/pt/2-comecar/decisoes-iniciais.md +223 -0
  100. package/docs/pt/2-comecar/primeiro-projeto.md +307 -0
  101. package/docs/pt/2-comecar/projeto-existente.md +245 -0
  102. package/docs/pt/3-receitas/README.md +28 -0
  103. package/docs/pt/3-receitas/app-saas-do-zero.md +324 -0
  104. package/docs/pt/3-receitas/auditoria-seguranca.md +254 -0
  105. package/docs/pt/3-receitas/clonar-design-de-site.md +211 -0
  106. package/docs/pt/3-receitas/continuidade-entre-sessoes.md +303 -0
  107. package/docs/pt/3-receitas/da-ideia-ao-prd-via-briefing.md +234 -0
  108. package/docs/pt/3-receitas/feature-completa-com-sheldon.md +338 -0
  109. package/docs/pt/3-receitas/integracao-em-codebase-grande.md +243 -0
  110. package/docs/pt/3-receitas/landing-page.md +281 -0
  111. package/docs/pt/3-receitas/plans-externos-para-product.md +191 -0
  112. package/docs/pt/3-receitas/publicar-no-aioson-com.md +219 -0
  113. package/docs/pt/3-receitas/refatoracao-grande.md +251 -0
  114. package/docs/pt/4-agentes/README.md +65 -0
  115. package/docs/pt/4-agentes/analyst.md +111 -0
  116. package/docs/pt/4-agentes/architect.md +113 -0
  117. package/docs/pt/4-agentes/briefing.md +95 -0
  118. package/docs/pt/4-agentes/committer.md +108 -0
  119. package/docs/pt/4-agentes/copywriter.md +279 -0
  120. package/docs/pt/4-agentes/design-hybrid-forge.md +116 -0
  121. package/docs/pt/4-agentes/dev.md +136 -0
  122. package/docs/pt/4-agentes/deyvin.md +99 -0
  123. package/docs/pt/4-agentes/discover.md +122 -0
  124. package/docs/pt/4-agentes/discovery-design-doc.md +91 -0
  125. package/docs/pt/4-agentes/genome.md +115 -0
  126. package/docs/pt/4-agentes/neo.md +93 -0
  127. package/docs/pt/4-agentes/orache.md +107 -0
  128. package/docs/pt/4-agentes/orchestrator.md +118 -0
  129. package/docs/pt/4-agentes/pentester.md +131 -0
  130. package/docs/pt/4-agentes/pm.md +97 -0
  131. package/docs/pt/4-agentes/product.md +114 -0
  132. package/docs/pt/4-agentes/profiler-enricher.md +93 -0
  133. package/docs/pt/4-agentes/profiler-forge.md +93 -0
  134. package/docs/pt/4-agentes/profiler-researcher.md +98 -0
  135. package/docs/pt/4-agentes/qa.md +124 -0
  136. package/docs/pt/4-agentes/setup.md +104 -0
  137. package/docs/pt/4-agentes/sheldon.md +95 -0
  138. package/docs/pt/4-agentes/site-forge.md +104 -0
  139. package/docs/pt/4-agentes/squad.md +127 -0
  140. package/docs/pt/4-agentes/tester.md +105 -0
  141. package/docs/pt/4-agentes/ux-ui.md +110 -0
  142. package/docs/pt/4-agentes/validator.md +118 -0
  143. package/docs/pt/5-referencia/README.md +88 -0
  144. package/docs/pt/5-referencia/agent-chain-continuity.md +124 -0
  145. package/docs/pt/{agent-sharding.md → 5-referencia/agent-sharding.md} +132 -132
  146. package/docs/pt/5-referencia/aioson-com-store.md +119 -0
  147. package/docs/pt/{automacao-squads.md → 5-referencia/automacao-squads.md} +407 -407
  148. package/docs/pt/{clientes-ai.md → 5-referencia/clientes-ai.md} +300 -290
  149. package/docs/pt/{comandos-cli.md → 5-referencia/comandos-cli.md} +1823 -1781
  150. package/docs/pt/{compress-agents.md → 5-referencia/compress-agents.md} +304 -304
  151. package/docs/pt/{design-docs-governance.md → 5-referencia/design-docs-governance.md} +59 -59
  152. package/docs/pt/{devlog-pipeline.md → 5-referencia/devlog-pipeline.md} +270 -270
  153. package/docs/pt/{feature-archive.md → 5-referencia/feature-archive.md} +199 -191
  154. package/docs/pt/5-referencia/feature-dossier.md +121 -0
  155. package/docs/pt/{fluxo-artefatos.md → 5-referencia/fluxo-artefatos.md} +179 -178
  156. package/docs/pt/{genome-3.0-spec.md → 5-referencia/genome-4.0-spec.md} +407 -407
  157. package/docs/pt/{genome-distribution.md → 5-referencia/genome-distribution.md} +232 -232
  158. package/docs/pt/{hooks-session-guard.md → 5-referencia/hooks-session-guard.md} +454 -454
  159. package/docs/pt/{inteligencia-adaptativa.md → 5-referencia/inteligencia-adaptativa.md} +324 -324
  160. package/docs/pt/5-referencia/live-sessions.md +144 -0
  161. package/docs/pt/5-referencia/memoria-e-contexto.md +340 -0
  162. package/docs/pt/{motor-hardening.md → 5-referencia/motor-hardening.md} +493 -492
  163. package/docs/pt/{output-strategy-delivery.md → 5-referencia/output-strategy-delivery.md} +655 -655
  164. package/docs/pt/{runner-system.md → 5-referencia/runner-system.md} +113 -113
  165. package/docs/pt/{runtime-observability.md → 5-referencia/runtime-observability.md} +76 -76
  166. package/docs/pt/{sandbox.md → 5-referencia/sandbox.md} +125 -125
  167. package/docs/pt/{sdd-automation-scripts.md → 5-referencia/sdd-automation-scripts.md} +559 -557
  168. package/docs/pt/5-referencia/sdd-framework.md +115 -0
  169. package/docs/pt/5-referencia/sdd-planos-e-estrutura.md +321 -0
  170. package/docs/pt/5-referencia/secure-by-default.md +117 -0
  171. package/docs/pt/{skills.md → 5-referencia/skills.md} +275 -267
  172. package/docs/pt/{spec-learnings-pipeline.md → 5-referencia/spec-learnings-pipeline.md} +265 -265
  173. package/docs/pt/{squad-dashboard.md → 5-referencia/squad-dashboard.md} +373 -373
  174. package/docs/pt/{web3.md → 5-referencia/web3.md} +797 -797
  175. package/docs/pt/README.md +111 -125
  176. package/docs/pt/_arquivo/README.md +130 -0
  177. package/docs/pt/{advisor-spec.md → _arquivo/advisor-spec.md} +343 -335
  178. package/docs/pt/{agentes-customizados.md → _arquivo/agentes-customizados.md} +678 -670
  179. package/docs/pt/{busca-de-contexto.md → _arquivo/busca-de-contexto.md} +136 -129
  180. package/docs/pt/{cache-de-contexto.md → _arquivo/cache-de-contexto.md} +163 -156
  181. package/docs/pt/{cenarios.md → _arquivo/cenarios.md} +1282 -1274
  182. package/docs/pt/{design-hybrid-forge.md → _arquivo/design-hybrid-forge.md} +365 -356
  183. package/docs/pt/{deyvin.md → _arquivo/deyvin.md} +123 -115
  184. package/docs/pt/{guia-engineer.md → _arquivo/guia-engineer.md} +234 -226
  185. package/docs/pt/{inicio-rapido.md → _arquivo/inicio-rapido.md} +261 -251
  186. package/docs/pt/{memoria-contexto.md → _arquivo/memoria-contexto.md} +262 -255
  187. package/docs/pt/{monitor-de-contexto.md → _arquivo/monitor-de-contexto.md} +165 -158
  188. package/docs/pt/{profiler-system.md → _arquivo/profiler-system.md} +222 -214
  189. package/docs/pt/{recuperacao-de-sessao.md → _arquivo/recuperacao-de-sessao.md} +134 -125
  190. package/docs/pt/{site-forge.md → _arquivo/site-forge.md} +318 -309
  191. package/docs/pt/{squad-genome.md → _arquivo/squad-genome.md} +793 -783
  192. package/docs/pt/active-learning-loop/README.md +117 -0
  193. package/docs/pt/active-learning-loop/ativo-learning-loop.md +117 -0
  194. package/docs/pt/active-learning-loop/comandos-cli.md +320 -0
  195. package/docs/pt/active-learning-loop/como-usar.md +313 -0
  196. package/docs/pt/active-learning-loop/diagramas.md +225 -0
  197. package/docs/pt/active-learning-loop/doctor-checks.md +151 -0
  198. package/docs/pt/active-learning-loop/troubleshooting.md +283 -0
  199. package/docs/pt/agentes.md +996 -993
  200. package/docs/pt/deyvin-subtask-scout/README.md +109 -0
  201. package/docs/pt/deyvin-subtask-scout/comandos-cli.md +248 -0
  202. package/docs/pt/deyvin-subtask-scout/como-usar.md +221 -0
  203. package/docs/pt/deyvin-subtask-scout/diagramas.md +124 -0
  204. package/docs/pt/deyvin-subtask-scout/sub-task-scout.md +113 -0
  205. package/docs/pt/deyvin-subtask-scout/troubleshooting.md +184 -0
  206. package/docs/pt/living-memory/README.md +81 -0
  207. package/docs/pt/living-memory/autonomy-contract.md +206 -0
  208. package/docs/pt/living-memory/diagramas.md +365 -0
  209. package/docs/pt/living-memory/memoria-viva.md +141 -0
  210. package/docs/pt/living-memory/notificacoes-info.md +142 -0
  211. package/docs/pt/living-memory/reflexao-in-harness.md +218 -0
  212. package/docs/pt/living-memory/troubleshooting.md +286 -0
  213. package/docs/testing/genome-2.0-manual-regression.md +23 -23
  214. package/docs/testing/genome-2.0-matrix.md +36 -36
  215. package/docs/testing/genome-2.0-rollout.md +184 -184
  216. package/package.json +51 -51
  217. package/src/a2a/client.js +165 -165
  218. package/src/a2a/server.js +223 -223
  219. package/src/agent-loader.js +280 -280
  220. package/src/agent-manifests.js +86 -66
  221. package/src/agents.js +92 -92
  222. package/src/autonomy-policy.js +163 -139
  223. package/src/backup-local.js +74 -74
  224. package/src/backup-provider.js +303 -303
  225. package/src/brain-query.js +171 -161
  226. package/src/cli.js +85 -5
  227. package/src/commands/agent-audit.js +397 -397
  228. package/src/commands/agent-export-skill.js +229 -229
  229. package/src/commands/agent-loader.js +85 -85
  230. package/src/commands/agents.js +273 -255
  231. package/src/commands/artifact-validate.js +218 -218
  232. package/src/commands/auth.js +298 -272
  233. package/src/commands/backup-local-cmd.js +25 -25
  234. package/src/commands/backup.js +533 -533
  235. package/src/commands/brain-query.js +44 -44
  236. package/src/commands/brief-gen.js +405 -405
  237. package/src/commands/brief-validate.js +65 -65
  238. package/src/commands/briefing.js +344 -344
  239. package/src/commands/classify.js +256 -256
  240. package/src/commands/cloud.js +1767 -1767
  241. package/src/commands/commit-prepare.js +610 -547
  242. package/src/commands/compress-agents.js +416 -416
  243. package/src/commands/config.js +90 -90
  244. package/src/commands/context-cache.js +90 -90
  245. package/src/commands/context-compact.js +49 -49
  246. package/src/commands/context-health.js +187 -177
  247. package/src/commands/context-load.js +219 -0
  248. package/src/commands/context-monitor.js +163 -163
  249. package/src/commands/context-pack.js +45 -45
  250. package/src/commands/context-search.js +66 -66
  251. package/src/commands/context-trim.js +183 -183
  252. package/src/commands/context-validate.js +91 -91
  253. package/src/commands/design-hybrid-options.js +385 -385
  254. package/src/commands/detect-test-runner.js +55 -55
  255. package/src/commands/dev-resume.js +32 -0
  256. package/src/commands/devlog-export-brains.js +27 -27
  257. package/src/commands/devlog-process.js +294 -294
  258. package/src/commands/devlog-watch.js +131 -131
  259. package/src/commands/doctor.js +123 -123
  260. package/src/commands/dossier-add-research.js +114 -0
  261. package/src/commands/dossier-audit.js +222 -0
  262. package/src/commands/dossier.js +423 -423
  263. package/src/commands/feature-archive.js +513 -513
  264. package/src/commands/feature-close.js +554 -270
  265. package/src/commands/gate-approve.js +198 -198
  266. package/src/commands/gate-check.js +247 -247
  267. package/src/commands/genome-doctor.js +489 -198
  268. package/src/commands/genome-migrate.js +49 -49
  269. package/src/commands/git-guard.js +170 -170
  270. package/src/commands/harness.js +307 -121
  271. package/src/commands/health.js +214 -214
  272. package/src/commands/hooks-emit.js +253 -253
  273. package/src/commands/hooks-install.js +347 -347
  274. package/src/commands/i18n-add.js +56 -56
  275. package/src/commands/implementation-plan.js +367 -367
  276. package/src/commands/info.js +41 -41
  277. package/src/commands/init.js +120 -120
  278. package/src/commands/install.js +162 -111
  279. package/src/commands/learning-auto-promote.js +197 -195
  280. package/src/commands/learning-evolve.js +364 -364
  281. package/src/commands/learning-export.js +103 -103
  282. package/src/commands/learning-rollback.js +164 -164
  283. package/src/commands/learning.js +134 -134
  284. package/src/commands/live.js +2101 -2082
  285. package/src/commands/locale-apply.js +54 -54
  286. package/src/commands/locale-diff.js +25 -25
  287. package/src/commands/mcp-doctor.js +407 -407
  288. package/src/commands/mcp-init.js +373 -373
  289. package/src/commands/memory-archive.js +193 -0
  290. package/src/commands/memory-reflect-commit.js +148 -0
  291. package/src/commands/memory-reflect-prepare.js +97 -0
  292. package/src/commands/memory-restore.js +177 -0
  293. package/src/commands/memory-search.js +135 -0
  294. package/src/commands/memory.js +299 -234
  295. package/src/commands/notify.js +68 -0
  296. package/src/commands/package-e2e.js +273 -273
  297. package/src/commands/parallel-assign.js +483 -483
  298. package/src/commands/parallel-doctor.js +850 -850
  299. package/src/commands/parallel-guard.js +241 -241
  300. package/src/commands/parallel-init.js +311 -311
  301. package/src/commands/parallel-merge.js +299 -299
  302. package/src/commands/parallel-status.js +434 -434
  303. package/src/commands/pattern-detect.js +33 -33
  304. package/src/commands/preflight-context.js +30 -30
  305. package/src/commands/preflight.js +267 -267
  306. package/src/commands/pulse-update.js +130 -130
  307. package/src/commands/qa-doctor.js +185 -185
  308. package/src/commands/qa-init.js +166 -166
  309. package/src/commands/qa-report.js +58 -58
  310. package/src/commands/qa-run.js +873 -873
  311. package/src/commands/qa-scan.js +337 -337
  312. package/src/commands/recovery.js +43 -43
  313. package/src/commands/revision.js +235 -235
  314. package/src/commands/runner-daemon.js +274 -274
  315. package/src/commands/runner-plan.js +70 -70
  316. package/src/commands/runner-queue-from-plan.js +166 -166
  317. package/src/commands/runner-queue.js +189 -189
  318. package/src/commands/runner-run.js +129 -129
  319. package/src/commands/runtime.js +2086 -2067
  320. package/src/commands/sandbox.js +37 -37
  321. package/src/commands/scaffold-complete.js +188 -188
  322. package/src/commands/scan-project.js +1371 -1371
  323. package/src/commands/scout-commit.js +163 -0
  324. package/src/commands/scout-prep.js +214 -0
  325. package/src/commands/scout-validate.js +112 -0
  326. package/src/commands/security-audit.js +275 -275
  327. package/src/commands/security-scan.js +376 -376
  328. package/src/commands/self-implement-loop.js +306 -300
  329. package/src/commands/session-guard.js +218 -218
  330. package/src/commands/setup-context.js +699 -699
  331. package/src/commands/setup.js +178 -178
  332. package/src/commands/sizing.js +165 -165
  333. package/src/commands/skill.js +670 -670
  334. package/src/commands/smoke.js +426 -426
  335. package/src/commands/spec-checkpoint.js +177 -177
  336. package/src/commands/spec-status.js +79 -79
  337. package/src/commands/spec-sync.js +190 -190
  338. package/src/commands/spec-tasks.js +288 -288
  339. package/src/commands/squad-agent-create.js +830 -830
  340. package/src/commands/squad-autorun.js +1220 -1220
  341. package/src/commands/squad-bus.js +217 -217
  342. package/src/commands/squad-card.js +149 -149
  343. package/src/commands/squad-daemon.js +343 -343
  344. package/src/commands/squad-dashboard.js +39 -39
  345. package/src/commands/squad-dependency-graph.js +164 -164
  346. package/src/commands/squad-deploy.js +64 -64
  347. package/src/commands/squad-doctor.js +460 -460
  348. package/src/commands/squad-export.js +77 -46
  349. package/src/commands/squad-investigate.js +314 -314
  350. package/src/commands/squad-learning.js +209 -209
  351. package/src/commands/squad-mcp.js +270 -270
  352. package/src/commands/squad-pipeline.js +343 -343
  353. package/src/commands/squad-plan.js +361 -361
  354. package/src/commands/squad-processes.js +56 -56
  355. package/src/commands/squad-recovery.js +42 -42
  356. package/src/commands/squad-repair-genomes.js +39 -39
  357. package/src/commands/squad-review.js +106 -106
  358. package/src/commands/squad-roi.js +291 -291
  359. package/src/commands/squad-scaffold.js +56 -56
  360. package/src/commands/squad-score.js +311 -307
  361. package/src/commands/squad-status.js +481 -481
  362. package/src/commands/squad-tool-register.js +157 -157
  363. package/src/commands/squad-validate.js +438 -438
  364. package/src/commands/squad-webhook.js +160 -160
  365. package/src/commands/squad-worker.js +191 -191
  366. package/src/commands/squad-worktrees.js +75 -75
  367. package/src/commands/state-save.js +220 -122
  368. package/src/commands/store-genome.js +667 -304
  369. package/src/commands/store-skill.js +247 -247
  370. package/src/commands/store-squad.js +431 -431
  371. package/src/commands/store-system.js +392 -392
  372. package/src/commands/sync-agents-preflight.js +176 -0
  373. package/src/commands/test-agents.js +199 -199
  374. package/src/commands/tool-capabilities.js +63 -63
  375. package/src/commands/tool-registry-cmd.js +232 -232
  376. package/src/commands/update.js +68 -64
  377. package/src/commands/verify-gate.js +612 -612
  378. package/src/commands/web-map.js +70 -70
  379. package/src/commands/web-scrape.js +71 -71
  380. package/src/commands/workflow-execute.js +730 -730
  381. package/src/commands/workflow-harden.js +231 -231
  382. package/src/commands/workflow-heal.js +136 -136
  383. package/src/commands/workflow-next.js +1279 -1039
  384. package/src/commands/workflow-plan.js +108 -108
  385. package/src/commands/workflow-status.js +440 -440
  386. package/src/commands/workspace.js +144 -144
  387. package/src/constants.js +417 -384
  388. package/src/context-cache.js +159 -159
  389. package/src/context-memory.js +975 -966
  390. package/src/context-parse-reason.js +22 -22
  391. package/src/context-search.js +326 -326
  392. package/src/context-writer.js +197 -197
  393. package/src/context.js +247 -247
  394. package/src/delivery-runner.js +319 -319
  395. package/src/design-variation-catalog.js +503 -503
  396. package/src/detector.js +261 -261
  397. package/src/doctor.js +812 -329
  398. package/src/dossier/codemap-store.js +267 -267
  399. package/src/dossier/dossier-bootstrap.js +222 -222
  400. package/src/dossier/dossier-compact.js +159 -159
  401. package/src/dossier/lock.js +128 -128
  402. package/src/dossier/research-index-store.js +233 -0
  403. package/src/dossier/revision-store.js +313 -313
  404. package/src/dossier/schema.js +162 -155
  405. package/src/dossier/scout-section.js +127 -0
  406. package/src/dossier/store.js +406 -400
  407. package/src/execution-gateway.js +464 -464
  408. package/src/friction-scanner.js +202 -202
  409. package/src/gateway-pointer-merge.js +101 -0
  410. package/src/genome-files.js +198 -198
  411. package/src/genome-format.js +442 -442
  412. package/src/genome-schema.js +238 -238
  413. package/src/genomes/bindings.js +281 -281
  414. package/src/genomes.js +500 -500
  415. package/src/handoff-contract.js +417 -363
  416. package/src/handoff-validator.js +45 -45
  417. package/src/harness/circuit-breaker.js +135 -135
  418. package/src/i18n/index.js +103 -103
  419. package/src/i18n/messages/en.js +1548 -1434
  420. package/src/i18n/messages/es.js +1332 -1221
  421. package/src/i18n/messages/fr.js +1340 -1229
  422. package/src/i18n/messages/pt-BR.js +1568 -1457
  423. package/src/i18n/scaffold.js +64 -64
  424. package/src/install-animation.js +260 -260
  425. package/src/install-profile.js +127 -127
  426. package/src/install-wizard.js +475 -475
  427. package/src/installer-config-merge.js +207 -0
  428. package/src/installer.js +487 -358
  429. package/src/jargon-leak-doctor.js +257 -0
  430. package/src/learning-loop-archive.js +595 -0
  431. package/src/learning-loop-doctor.js +217 -0
  432. package/src/learning-loop-engine.js +254 -0
  433. package/src/learning-loop-fts5.js +132 -0
  434. package/src/learning-loop-migration.js +163 -0
  435. package/src/lib/dev-resume.js +140 -0
  436. package/src/lib/dossier-telemetry.js +36 -0
  437. package/src/lib/genomes/compat.js +206 -206
  438. package/src/lib/genomes/migrate.js +90 -90
  439. package/src/lib/git-commit-guard.js +751 -691
  440. package/src/lib/health-check.js +158 -158
  441. package/src/lib/hook-protocol.js +76 -76
  442. package/src/lib/llm-content-sanitizer.js +44 -0
  443. package/src/lib/security/artifact-reader.js +167 -167
  444. package/src/lib/security/exit-codes.js +51 -51
  445. package/src/lib/security/findings-writer.js +176 -176
  446. package/src/lib/security/runtime-events.js +77 -77
  447. package/src/lib/security/secrets-regex.js +115 -115
  448. package/src/lib/squads/genome-repair.js +49 -49
  449. package/src/lib/store/security-scan.js +175 -173
  450. package/src/lib/terminal-checkbox.js +135 -130
  451. package/src/lib/terminal-picker.js +447 -0
  452. package/src/lib/tmux-launcher.js +163 -163
  453. package/src/lib/tool-capabilities.js +102 -102
  454. package/src/lib/webhook-server.js +328 -328
  455. package/src/locales.js +88 -88
  456. package/src/mcp/apps/squad-dashboard/app.js +163 -163
  457. package/src/mcp/apps/squad-dashboard/index.html +261 -261
  458. package/src/mcp/apps/squad-dashboard/mcp-manifest.json +23 -23
  459. package/src/mcp/resources/squad-state.js +130 -130
  460. package/src/mcp-connectors/registry.js +602 -602
  461. package/src/memory-reflect-engine.js +359 -0
  462. package/src/migrations/profile-rename.js +66 -0
  463. package/src/notify-renderer.js +32 -0
  464. package/src/onboarding.js +307 -305
  465. package/src/parallel-workspace.js +756 -756
  466. package/src/parser.js +74 -66
  467. package/src/path-guard.js +47 -47
  468. package/src/permissions-generator.js +400 -0
  469. package/src/preflight-engine.js +654 -654
  470. package/src/prompt-tool.js +20 -20
  471. package/src/qa-html-report.js +472 -472
  472. package/src/recovery-context-session.js +154 -154
  473. package/src/runner/cascade.js +97 -97
  474. package/src/runner/cli-launcher.js +109 -109
  475. package/src/runner/plan-importer.js +63 -63
  476. package/src/runner/queue-store.js +159 -159
  477. package/src/runtime-store.js +2720 -2676
  478. package/src/sandbox.js +194 -177
  479. package/src/self-healing.js +142 -142
  480. package/src/session-handoff.js +295 -187
  481. package/src/squad/agent-teams-adapter.js +270 -264
  482. package/src/squad/brief-validator.js +350 -350
  483. package/src/squad/bus-bridge.js +140 -140
  484. package/src/squad/context-compactor.js +265 -265
  485. package/src/squad/cross-ai-synthesizer.js +250 -250
  486. package/src/squad/external-session.js +180 -180
  487. package/src/squad/hooks-generator.js +196 -196
  488. package/src/squad/inter-squad-events.js +175 -175
  489. package/src/squad/inter-squad.js +74 -74
  490. package/src/squad/intra-bus.js +345 -345
  491. package/src/squad/learning-extractor.js +213 -213
  492. package/src/squad/pattern-detector.js +365 -365
  493. package/src/squad/preflight-context.js +296 -296
  494. package/src/squad/recovery-context.js +372 -372
  495. package/src/squad/reflection.js +365 -365
  496. package/src/squad/squad-scaffold.js +341 -341
  497. package/src/squad/state-manager.js +310 -310
  498. package/src/squad/task-decomposer.js +652 -652
  499. package/src/squad/verify-gate.js +303 -303
  500. package/src/squad/worktree-manager.js +114 -114
  501. package/src/squad-daemon.js +490 -490
  502. package/src/squad-dashboard/api.js +223 -223
  503. package/src/squad-dashboard/attachment-handler.js +93 -93
  504. package/src/squad-dashboard/context-monitor.js +157 -157
  505. package/src/squad-dashboard/execution-logs.js +115 -115
  506. package/src/squad-dashboard/hunk-review.js +209 -209
  507. package/src/squad-dashboard/metrics.js +133 -133
  508. package/src/squad-dashboard/process-monitor.js +125 -125
  509. package/src/squad-dashboard/renderer.js +858 -858
  510. package/src/squad-dashboard/server.js +232 -232
  511. package/src/squad-dashboard/styles.js +525 -525
  512. package/src/squad-dashboard/token-tracker.js +99 -99
  513. package/src/squads/apply-genome.js +21 -21
  514. package/src/squads/genome-binding-service.js +154 -154
  515. package/src/sub-task-engine.js +415 -0
  516. package/src/sub-task-schemas.js +150 -0
  517. package/src/sub-task-state.js +152 -0
  518. package/src/sub-task-telemetry.js +69 -0
  519. package/src/test-briefing.js +226 -226
  520. package/src/tool-executor.js +94 -94
  521. package/src/updater.js +52 -39
  522. package/src/utils.js +49 -49
  523. package/src/version.js +50 -50
  524. package/src/web.js +284 -284
  525. package/src/worker-runner.js +541 -524
  526. package/src/workflow-gates.js +185 -185
  527. package/template/.aioson/advisors/.gitkeep +1 -1
  528. package/template/.aioson/agents/analyst.md +345 -318
  529. package/template/.aioson/agents/architect.md +325 -305
  530. package/template/.aioson/agents/{cypher.md → briefing.md} +264 -252
  531. package/template/.aioson/agents/committer.md +161 -161
  532. package/template/.aioson/agents/copywriter.md +937 -463
  533. package/template/.aioson/agents/design-hybrid-forge.md +141 -141
  534. package/template/.aioson/agents/dev.md +298 -263
  535. package/template/.aioson/agents/deyvin.md +200 -87
  536. package/template/.aioson/agents/discover.md +235 -235
  537. package/template/.aioson/agents/discovery-design-doc.md +56 -29
  538. package/template/.aioson/agents/genome.md +1904 -364
  539. package/template/.aioson/agents/manifests/analyst.manifest.json +26 -26
  540. package/template/.aioson/agents/manifests/architect.manifest.json +23 -23
  541. package/template/.aioson/agents/manifests/committer.manifest.json +23 -23
  542. package/template/.aioson/agents/manifests/dev.manifest.json +54 -37
  543. package/template/.aioson/agents/manifests/deyvin.manifest.json +41 -0
  544. package/template/.aioson/agents/manifests/orchestrator.manifest.json +30 -30
  545. package/template/.aioson/agents/manifests/pentester.manifest.json +39 -39
  546. package/template/.aioson/agents/manifests/pm.manifest.json +26 -26
  547. package/template/.aioson/agents/manifests/product.manifest.json +23 -23
  548. package/template/.aioson/agents/manifests/qa.manifest.json +41 -25
  549. package/template/.aioson/agents/manifests/setup.manifest.json +20 -20
  550. package/template/.aioson/agents/manifests/ux-ui.manifest.json +24 -24
  551. package/template/.aioson/agents/neo.md +356 -231
  552. package/template/.aioson/agents/orache.md +430 -430
  553. package/template/.aioson/agents/orchestrator.md +274 -263
  554. package/template/.aioson/agents/pair.md +5 -5
  555. package/template/.aioson/agents/pentester.md +289 -235
  556. package/template/.aioson/agents/pm.md +141 -130
  557. package/template/.aioson/agents/product.md +367 -273
  558. package/template/.aioson/agents/profiler-enricher.md +331 -331
  559. package/template/.aioson/agents/profiler-forge.md +212 -212
  560. package/template/.aioson/agents/profiler-researcher.md +282 -282
  561. package/template/.aioson/agents/qa.md +432 -342
  562. package/template/.aioson/agents/setup.md +425 -423
  563. package/template/.aioson/agents/sheldon.md +259 -197
  564. package/template/.aioson/agents/site-forge.md +281 -281
  565. package/template/.aioson/agents/squad.md +160 -156
  566. package/template/.aioson/agents/tester.md +536 -473
  567. package/template/.aioson/agents/ux-ui.md +195 -162
  568. package/template/.aioson/agents/validator.md +101 -69
  569. package/template/.aioson/brains/README.md +132 -128
  570. package/template/.aioson/brains/_archived/.gitkeep +0 -0
  571. package/template/.aioson/brains/_index.json +34 -16
  572. package/template/.aioson/brains/dev/patterns.brain.json +79 -0
  573. package/template/.aioson/brains/scripts/query.js +107 -107
  574. package/template/.aioson/brains/sheldon/architecture-decisions.brain.json +79 -0
  575. package/template/.aioson/brains/site-forge/visual-patterns.brain.json +205 -205
  576. package/template/.aioson/config/autonomy-protocol.json +125 -43
  577. package/template/.aioson/config/learning-loop.json +10 -0
  578. package/template/.aioson/config/scout-engine.json +1 -0
  579. package/template/.aioson/config.md +410 -410
  580. package/template/.aioson/context/_archived/.gitkeep +0 -0
  581. package/template/.aioson/context/design-doc.md +136 -136
  582. package/template/.aioson/context/project-map.md +57 -57
  583. package/template/.aioson/context/project-pulse.md +34 -34
  584. package/template/.aioson/context/seeds/seed-example.md +27 -27
  585. package/template/.aioson/context/spec.md.template +54 -54
  586. package/template/.aioson/context/user-profile.md +42 -42
  587. package/template/.aioson/design-docs/code-reuse.md +48 -48
  588. package/template/.aioson/design-docs/componentization.md +47 -47
  589. package/template/.aioson/design-docs/file-size.md +52 -52
  590. package/template/.aioson/design-docs/folder-structure.md +51 -51
  591. package/template/.aioson/design-docs/naming.md +54 -54
  592. package/template/.aioson/docs/LAYERS.md +89 -89
  593. package/template/.aioson/docs/README.md +76 -76
  594. package/template/.aioson/docs/autonomy-protocol.md +80 -0
  595. package/template/.aioson/docs/briefing/briefing-craft.md +237 -0
  596. package/template/.aioson/docs/dev/execution-discipline.md +106 -106
  597. package/template/.aioson/docs/dev/stack-conventions.md +83 -83
  598. package/template/.aioson/docs/deyvin/continuity-recovery.md +57 -57
  599. package/template/.aioson/docs/deyvin/debugging-escalation.md +30 -30
  600. package/template/.aioson/docs/deyvin/pair-execution.md +44 -44
  601. package/template/.aioson/docs/deyvin/runtime-handoffs.md +42 -36
  602. package/template/.aioson/docs/example-external-api-context.md +72 -72
  603. package/template/.aioson/docs/handoff-persistence.md +94 -0
  604. package/template/.aioson/docs/pentester/app-playbooks.md +206 -0
  605. package/template/.aioson/docs/pentester/llm-supplychain.md +165 -0
  606. package/template/.aioson/docs/product/conversation-playbook.md +116 -116
  607. package/template/.aioson/docs/product/prd-contract.md +107 -107
  608. package/template/.aioson/docs/product/quality-lens.md +57 -57
  609. package/template/.aioson/docs/product/research-loop.md +65 -65
  610. package/template/.aioson/docs/sheldon/enrichment-paths.md +134 -134
  611. package/template/.aioson/docs/sheldon/harness-contract.md +118 -0
  612. package/template/.aioson/docs/sheldon/quality-lens.md +57 -57
  613. package/template/.aioson/docs/sheldon/research-loop.md +56 -56
  614. package/template/.aioson/docs/sheldon/web-intelligence.md +75 -75
  615. package/template/.aioson/docs/site-forge-build.md +195 -195
  616. package/template/.aioson/docs/site-forge-extraction.md +135 -135
  617. package/template/.aioson/docs/site-forge-qa.md +155 -155
  618. package/template/.aioson/docs/site-forge-recon.md +434 -434
  619. package/template/.aioson/docs/site-forge-transform.md +249 -249
  620. package/template/.aioson/docs/squad/content-output.md +91 -91
  621. package/template/.aioson/docs/squad/creation-flow.md +149 -135
  622. package/template/.aioson/docs/squad/domain-breadth.md +322 -0
  623. package/template/.aioson/docs/squad/domain-classification.md +117 -117
  624. package/template/.aioson/docs/squad/genome-bindings.md +47 -47
  625. package/template/.aioson/docs/squad/package-contract.md +260 -234
  626. package/template/.aioson/docs/squad/quality-lens.md +60 -56
  627. package/template/.aioson/docs/squad/research-loop.md +59 -59
  628. package/template/.aioson/docs/squad/session-operations.md +117 -117
  629. package/template/.aioson/docs/squad/workflow-quality.md +165 -165
  630. package/template/.aioson/docs/tester/coverage-quality.md +351 -0
  631. package/template/.aioson/docs/ux-ui/accessibility-audit.md +55 -55
  632. package/template/.aioson/docs/ux-ui/audit-mode.md +86 -86
  633. package/template/.aioson/docs/ux-ui/component-map.md +35 -35
  634. package/template/.aioson/docs/ux-ui/design-execution.md +111 -111
  635. package/template/.aioson/docs/ux-ui/design-gate.md +27 -27
  636. package/template/.aioson/docs/ux-ui/research-mode.md +39 -39
  637. package/template/.aioson/docs/ux-ui/site-delivery.md +156 -156
  638. package/template/.aioson/docs/ux-ui/token-contract.md +57 -57
  639. package/template/.aioson/genomes/INDEX.md +195 -0
  640. package/template/.aioson/genomes/copywriting/SKILL.md +137 -0
  641. package/template/.aioson/genomes/copywriting/manifest.json +140 -0
  642. package/template/.aioson/genomes/copywriting/references/application-notes.md +145 -0
  643. package/template/.aioson/genomes/copywriting/references/decision-weights.md +45 -0
  644. package/template/.aioson/genomes/copywriting/references/frameworks/5-act-narrative.md +184 -0
  645. package/template/.aioson/genomes/copywriting/references/frameworks/classical-formulas.md +164 -0
  646. package/template/.aioson/genomes/copywriting/references/frameworks/offer-stack.md +195 -0
  647. package/template/.aioson/genomes/copywriting/references/frameworks/one-belief.md +135 -0
  648. package/template/.aioson/genomes/copywriting/references/frameworks/pms-research.md +211 -0
  649. package/template/.aioson/genomes/copywriting/references/frameworks/two-paths-close.md +190 -0
  650. package/template/.aioson/genomes/copywriting/references/heuristics.md +114 -0
  651. package/template/.aioson/genomes/copywriting/references/meta-axioms.md +68 -0
  652. package/template/.aioson/genomes/copywriting/references/methodology.md +115 -0
  653. package/template/.aioson/genomes/copywriting-brunson/SKILL.md +133 -0
  654. package/template/.aioson/genomes/copywriting-brunson/manifest.json +152 -0
  655. package/template/.aioson/genomes/copywriting-brunson/references/application-notes.md +113 -0
  656. package/template/.aioson/genomes/copywriting-brunson/references/decision-weights.md +33 -0
  657. package/template/.aioson/genomes/copywriting-brunson/references/evidence-and-attribution.md +81 -0
  658. package/template/.aioson/genomes/copywriting-brunson/references/frameworks/6-part-structure.md +136 -0
  659. package/template/.aioson/genomes/copywriting-brunson/references/frameworks/origin-story.md +121 -0
  660. package/template/.aioson/genomes/copywriting-brunson/references/frameworks/perfect-webinar-script.md +139 -0
  661. package/template/.aioson/genomes/copywriting-brunson/references/frameworks/persuasive-storytelling-5-structures.md +164 -0
  662. package/template/.aioson/genomes/copywriting-brunson/references/frameworks/value-stack.md +136 -0
  663. package/template/.aioson/genomes/copywriting-brunson/references/frameworks/who-what-why-how.md +110 -0
  664. package/template/.aioson/genomes/copywriting-brunson/references/meta-axioms.md +36 -0
  665. package/template/.aioson/genomes/copywriting-brunson/references/methodology.md +112 -0
  666. package/template/.aioson/git-guard.json +12 -11
  667. package/template/.aioson/mcp/servers.md +23 -23
  668. package/template/.aioson/profiler-reports/.gitkeep +1 -1
  669. package/template/.aioson/rules/README.md +69 -69
  670. package/template/.aioson/rules/_archived/.gitkeep +0 -0
  671. package/template/.aioson/rules/agent-language-policy.md +93 -93
  672. package/template/.aioson/rules/aioson-context-boundary.md +63 -63
  673. package/template/.aioson/rules/canonical-path-contract.md +47 -47
  674. package/template/.aioson/rules/data-format-convention.md +74 -74
  675. package/template/.aioson/rules/disk-first-artifacts.md +44 -44
  676. package/template/.aioson/rules/example-monetary-values.md +30 -30
  677. package/template/.aioson/rules/output-brevity.md +44 -44
  678. package/template/.aioson/rules/prd-section-ownership.md +49 -49
  679. package/template/.aioson/rules/security-baseline.md +139 -139
  680. package/template/.aioson/rules/spec-level-ownership.md +61 -61
  681. package/template/.aioson/rules/squad/README.md +50 -50
  682. package/template/.aioson/rules/squad-driver-pattern.md +81 -81
  683. package/template/.aioson/schemas/content-blueprint.schema.json +30 -30
  684. package/template/.aioson/schemas/genome-meta.schema.json +150 -150
  685. package/template/.aioson/schemas/genome.schema.json +115 -115
  686. package/template/.aioson/schemas/readiness.schema.json +27 -27
  687. package/template/.aioson/schemas/squad-blueprint.schema.json +228 -228
  688. package/template/.aioson/schemas/squad-manifest.schema.json +874 -874
  689. package/template/.aioson/skills/design/aurora-command-ui/SKILL.md +243 -243
  690. package/template/.aioson/skills/design/aurora-command-ui/references/art-direction.md +293 -293
  691. package/template/.aioson/skills/design/aurora-command-ui/references/components.md +827 -827
  692. package/template/.aioson/skills/design/aurora-command-ui/references/dashboards.md +250 -250
  693. package/template/.aioson/skills/design/aurora-command-ui/references/design-tokens.md +585 -585
  694. package/template/.aioson/skills/design/aurora-command-ui/references/motion.md +365 -365
  695. package/template/.aioson/skills/design/aurora-command-ui/references/patterns.md +482 -482
  696. package/template/.aioson/skills/design/aurora-command-ui/references/websites.md +387 -387
  697. package/template/.aioson/skills/design/bold-editorial-ui/SKILL.md +205 -205
  698. package/template/.aioson/skills/design/bold-editorial-ui/references/art-direction.md +338 -338
  699. package/template/.aioson/skills/design/bold-editorial-ui/references/components.md +977 -977
  700. package/template/.aioson/skills/design/bold-editorial-ui/references/dashboards.md +218 -218
  701. package/template/.aioson/skills/design/bold-editorial-ui/references/design-tokens.md +326 -326
  702. package/template/.aioson/skills/design/bold-editorial-ui/references/motion.md +461 -461
  703. package/template/.aioson/skills/design/bold-editorial-ui/references/patterns.md +293 -293
  704. package/template/.aioson/skills/design/bold-editorial-ui/references/websites.md +352 -352
  705. package/template/.aioson/skills/design/clean-saas-ui/SKILL.md +210 -210
  706. package/template/.aioson/skills/design/clean-saas-ui/references/art-direction.md +319 -319
  707. package/template/.aioson/skills/design/clean-saas-ui/references/components.md +365 -365
  708. package/template/.aioson/skills/design/clean-saas-ui/references/dashboards.md +196 -196
  709. package/template/.aioson/skills/design/clean-saas-ui/references/design-tokens.md +244 -244
  710. package/template/.aioson/skills/design/clean-saas-ui/references/motion.md +235 -235
  711. package/template/.aioson/skills/design/clean-saas-ui/references/patterns.md +215 -215
  712. package/template/.aioson/skills/design/clean-saas-ui/references/websites.md +295 -295
  713. package/template/.aioson/skills/design/cognitive-core-ui/SKILL.md +203 -203
  714. package/template/.aioson/skills/design/cognitive-core-ui/references/art-direction.md +339 -339
  715. package/template/.aioson/skills/design/cognitive-core-ui/references/components.md +407 -407
  716. package/template/.aioson/skills/design/cognitive-core-ui/references/dashboards.md +272 -272
  717. package/template/.aioson/skills/design/cognitive-core-ui/references/design-tokens.md +524 -524
  718. package/template/.aioson/skills/design/cognitive-core-ui/references/motion.md +279 -279
  719. package/template/.aioson/skills/design/cognitive-core-ui/references/patterns.md +289 -289
  720. package/template/.aioson/skills/design/cognitive-core-ui/references/websites.md +437 -437
  721. package/template/.aioson/skills/design/glassmorphism-ui/SKILL.md +222 -222
  722. package/template/.aioson/skills/design/glassmorphism-ui/references/art-direction.md +159 -159
  723. package/template/.aioson/skills/design/glassmorphism-ui/references/components.md +498 -498
  724. package/template/.aioson/skills/design/glassmorphism-ui/references/dashboards.md +236 -236
  725. package/template/.aioson/skills/design/glassmorphism-ui/references/design-tokens.md +274 -274
  726. package/template/.aioson/skills/design/glassmorphism-ui/references/motion.md +355 -355
  727. package/template/.aioson/skills/design/glassmorphism-ui/references/patterns.md +198 -198
  728. package/template/.aioson/skills/design/glassmorphism-ui/references/websites.md +307 -307
  729. package/template/.aioson/skills/design/interface-design/SKILL.md +47 -47
  730. package/template/.aioson/skills/design/interface-design/references/components-and-states.md +105 -105
  731. package/template/.aioson/skills/design/interface-design/references/design-directions.md +101 -101
  732. package/template/.aioson/skills/design/interface-design/references/handoff-and-quality.md +71 -71
  733. package/template/.aioson/skills/design/interface-design/references/intent-and-domain.md +74 -74
  734. package/template/.aioson/skills/design/interface-design/references/tokens-and-depth.md +173 -173
  735. package/template/.aioson/skills/design/neo-brutalist-ui/SKILL.md +213 -213
  736. package/template/.aioson/skills/design/neo-brutalist-ui/references/art-direction.md +228 -228
  737. package/template/.aioson/skills/design/neo-brutalist-ui/references/components.md +855 -855
  738. package/template/.aioson/skills/design/neo-brutalist-ui/references/dashboards.md +334 -334
  739. package/template/.aioson/skills/design/neo-brutalist-ui/references/design-tokens.md +342 -342
  740. package/template/.aioson/skills/design/neo-brutalist-ui/references/motion.md +286 -286
  741. package/template/.aioson/skills/design/neo-brutalist-ui/references/patterns.md +458 -458
  742. package/template/.aioson/skills/design/neo-brutalist-ui/references/websites.md +723 -723
  743. package/template/.aioson/skills/design/premium-command-center-ui/SKILL.md +62 -62
  744. package/template/.aioson/skills/design/premium-command-center-ui/references/operations.md +74 -74
  745. package/template/.aioson/skills/design/premium-command-center-ui/references/patterns.md +116 -116
  746. package/template/.aioson/skills/design/premium-command-center-ui/references/validation.md +47 -47
  747. package/template/.aioson/skills/design/premium-command-center-ui/references/visual-system.md +215 -215
  748. package/template/.aioson/skills/design/pt.squarespace.com/.skill-meta.json +31 -31
  749. package/template/.aioson/skills/design/pt.squarespace.com/SKILL.md +66 -66
  750. package/template/.aioson/skills/design/pt.squarespace.com/references/components.md +368 -368
  751. package/template/.aioson/skills/design/pt.squarespace.com/references/design-tokens.md +150 -150
  752. package/template/.aioson/skills/design/pt.squarespace.com/references/motion.md +270 -270
  753. package/template/.aioson/skills/design/pt.squarespace.com/references/patterns.md +189 -189
  754. package/template/.aioson/skills/design/pt.squarespace.com/references/websites.md +165 -165
  755. package/template/.aioson/skills/design/warm-craft-ui/SKILL.md +209 -209
  756. package/template/.aioson/skills/design/warm-craft-ui/references/art-direction.md +324 -324
  757. package/template/.aioson/skills/design/warm-craft-ui/references/components.md +508 -508
  758. package/template/.aioson/skills/design/warm-craft-ui/references/dashboards.md +223 -223
  759. package/template/.aioson/skills/design/warm-craft-ui/references/design-tokens.md +374 -374
  760. package/template/.aioson/skills/design/warm-craft-ui/references/motion.md +356 -356
  761. package/template/.aioson/skills/design/warm-craft-ui/references/patterns.md +288 -288
  762. package/template/.aioson/skills/design/warm-craft-ui/references/websites.md +289 -289
  763. package/template/.aioson/skills/design-system/SKILL.md +92 -92
  764. package/template/.aioson/skills/design-system/components/SKILL.md +274 -274
  765. package/template/.aioson/skills/design-system/dashboards/SKILL.md +184 -184
  766. package/template/.aioson/skills/design-system/foundations/SKILL.md +250 -250
  767. package/template/.aioson/skills/design-system/motion/SKILL.md +197 -197
  768. package/template/.aioson/skills/design-system/patterns/SKILL.md +231 -231
  769. package/template/.aioson/skills/dynamic/README.md +30 -30
  770. package/template/.aioson/skills/dynamic/cardano-docs.md +16 -16
  771. package/template/.aioson/skills/dynamic/ethereum-docs.md +17 -17
  772. package/template/.aioson/skills/dynamic/flux-ui-docs.md +13 -13
  773. package/template/.aioson/skills/dynamic/laravel-docs.md +41 -41
  774. package/template/.aioson/skills/dynamic/npm-packages.md +16 -16
  775. package/template/.aioson/skills/dynamic/solana-docs.md +16 -16
  776. package/template/.aioson/skills/marketing/references/anti-patterns.md +254 -254
  777. package/template/.aioson/skills/marketing/references/cta-matrix.md +361 -0
  778. package/template/.aioson/skills/marketing/references/fascinations.md +192 -192
  779. package/template/.aioson/skills/marketing/references/five-acts.md +248 -248
  780. package/template/.aioson/skills/marketing/references/headline-matrix.md +358 -0
  781. package/template/.aioson/skills/marketing/references/market-intelligence.md +198 -198
  782. package/template/.aioson/skills/marketing/references/offer-structure.md +203 -203
  783. package/template/.aioson/skills/marketing/references/one-belief.md +149 -149
  784. package/template/.aioson/skills/marketing/references/patterns.md +218 -218
  785. package/template/.aioson/skills/marketing/references/platform-constraints.md +337 -0
  786. package/template/.aioson/skills/marketing/references/pms-research.md +193 -193
  787. package/template/.aioson/skills/marketing/vsl-craft.md +385 -385
  788. package/template/.aioson/skills/premium-visual-design/SKILL.md +83 -83
  789. package/template/.aioson/skills/premium-visual-design/components/agent-badge.md +92 -92
  790. package/template/.aioson/skills/premium-visual-design/components/dependency-node.md +102 -102
  791. package/template/.aioson/skills/premium-visual-design/components/mention-autocomplete.md +136 -136
  792. package/template/.aioson/skills/premium-visual-design/components/notification-center.md +136 -136
  793. package/template/.aioson/skills/premium-visual-design/components/review-action-bar.md +188 -188
  794. package/template/.aioson/skills/premium-visual-design/components/team-switcher.md +131 -131
  795. package/template/.aioson/skills/premium-visual-design/patterns/agent-message-thread.md +198 -198
  796. package/template/.aioson/skills/premium-visual-design/patterns/notification-panel.md +275 -275
  797. package/template/.aioson/skills/premium-visual-design/patterns/review-workflow-ui.md +234 -234
  798. package/template/.aioson/skills/premium-visual-design/patterns/task-dependency-graph.md +147 -147
  799. package/template/.aioson/skills/premium-visual-design/tokens/status-extended.md +142 -142
  800. package/template/.aioson/skills/process/aioson-spec-driven/SKILL.md +46 -46
  801. package/template/.aioson/skills/process/aioson-spec-driven/references/analyst.md +30 -30
  802. package/template/.aioson/skills/process/aioson-spec-driven/references/approval-gates.md +109 -109
  803. package/template/.aioson/skills/process/aioson-spec-driven/references/architect.md +23 -23
  804. package/template/.aioson/skills/process/aioson-spec-driven/references/artifact-map.md +44 -44
  805. package/template/.aioson/skills/process/aioson-spec-driven/references/classification-map.md +37 -37
  806. package/template/.aioson/skills/process/aioson-spec-driven/references/dev.md +47 -47
  807. package/template/.aioson/skills/process/aioson-spec-driven/references/deyvin.md +27 -27
  808. package/template/.aioson/skills/process/aioson-spec-driven/references/hardening-lane.md +49 -49
  809. package/template/.aioson/skills/process/aioson-spec-driven/references/maintenance-and-state.md +101 -101
  810. package/template/.aioson/skills/process/aioson-spec-driven/references/pm.md +30 -30
  811. package/template/.aioson/skills/process/aioson-spec-driven/references/product.md +25 -25
  812. package/template/.aioson/skills/process/aioson-spec-driven/references/qa.md +30 -30
  813. package/template/.aioson/skills/process/aioson-spec-driven/references/sheldon.md +25 -25
  814. package/template/.aioson/skills/process/aioson-spec-driven/references/ui-language.md +75 -75
  815. package/template/.aioson/skills/process/decision-presentation/SKILL.md +119 -0
  816. package/template/.aioson/skills/process/decision-presentation/references/jargon-map.en.yaml +108 -0
  817. package/template/.aioson/skills/process/decision-presentation/references/jargon-map.pt-BR.yaml +108 -0
  818. package/template/.aioson/skills/process/design-hybrid-forge/SKILL.md +147 -147
  819. package/template/.aioson/skills/process/design-hybrid-forge/references/crossover-protocol.md +221 -221
  820. package/template/.aioson/skills/process/design-hybrid-forge/references/naming-registry.md +88 -88
  821. package/template/.aioson/skills/process/design-hybrid-forge/references/output-contract.md +306 -306
  822. package/template/.aioson/skills/process/design-hybrid-forge/references/pair-compatibility.md +149 -149
  823. package/template/.aioson/skills/process/design-hybrid-forge/references/quality-gates.md +208 -208
  824. package/template/.aioson/skills/process/design-hybrid-forge/references/variation-library.md +125 -125
  825. package/template/.aioson/skills/process/secure-tdd/SKILL.md +97 -97
  826. package/template/.aioson/skills/process/simplify/SKILL.md +173 -173
  827. package/template/.aioson/skills/references/premium-command-center-ui/master-application-prompt.md +79 -79
  828. package/template/.aioson/skills/references/premium-command-center-ui/operational-ux-playbook.md +253 -253
  829. package/template/.aioson/skills/references/premium-command-center-ui/quality-validation-checklist.md +82 -82
  830. package/template/.aioson/skills/references/premium-command-center-ui/visual-system-and-component-patterns.md +270 -270
  831. package/template/.aioson/skills/squad/SKILL.md +58 -58
  832. package/template/.aioson/skills/squad/formats/catalog.json +15 -15
  833. package/template/.aioson/skills/squad/formats/content/blog-post.md +47 -47
  834. package/template/.aioson/skills/squad/formats/content/newsletter.md +47 -47
  835. package/template/.aioson/skills/squad/formats/creative/podcast-script.md +43 -43
  836. package/template/.aioson/skills/squad/formats/creative/video-script.md +41 -41
  837. package/template/.aioson/skills/squad/formats/social/instagram-feed.md +42 -42
  838. package/template/.aioson/skills/squad/formats/social/linkedin-post.md +42 -42
  839. package/template/.aioson/skills/squad/formats/social/tiktok.md +39 -39
  840. package/template/.aioson/skills/squad/formats/social/twitter-thread.md +39 -39
  841. package/template/.aioson/skills/squad/formats/social/youtube-long.md +47 -47
  842. package/template/.aioson/skills/squad/formats/social/youtube-shorts.md +39 -39
  843. package/template/.aioson/skills/squad/patterns/multi-platform-pattern.md +108 -108
  844. package/template/.aioson/skills/squad/patterns/persona-based-pattern.md +98 -98
  845. package/template/.aioson/skills/squad/patterns/pipeline-pattern.md +106 -106
  846. package/template/.aioson/skills/squad/patterns/review-loop-pattern.md +81 -81
  847. package/template/.aioson/skills/squad/references/checklist-templates.md +122 -122
  848. package/template/.aioson/skills/squad/references/executor-archetypes.md +123 -123
  849. package/template/.aioson/skills/squad/references/workflow-templates.md +169 -169
  850. package/template/.aioson/skills/static/context-budget-guide.md +46 -46
  851. package/template/.aioson/skills/static/debugging-protocol.md +42 -42
  852. package/template/.aioson/skills/static/django-patterns.md +342 -342
  853. package/template/.aioson/skills/static/fastapi-patterns.md +344 -344
  854. package/template/.aioson/skills/static/filament-patterns.md +267 -267
  855. package/template/.aioson/skills/static/flux-ui-components.md +262 -262
  856. package/template/.aioson/skills/static/git-conventions.md +227 -227
  857. package/template/.aioson/skills/static/git-worktrees.md +36 -36
  858. package/template/.aioson/skills/static/harness-sensors.md +74 -74
  859. package/template/.aioson/skills/static/harness-validate/SKILL.md +46 -46
  860. package/template/.aioson/skills/static/jetstream-setup.md +200 -200
  861. package/template/.aioson/skills/static/landing-page-deploy.md +192 -192
  862. package/template/.aioson/skills/static/landing-page-forge.md +730 -730
  863. package/template/.aioson/skills/static/laravel-conventions.md +491 -491
  864. package/template/.aioson/skills/static/multi-agent-patterns.md +43 -43
  865. package/template/.aioson/skills/static/nextjs-patterns.md +321 -321
  866. package/template/.aioson/skills/static/node-express-patterns.md +317 -317
  867. package/template/.aioson/skills/static/node-typescript-patterns.md +282 -282
  868. package/template/.aioson/skills/static/rails-conventions.md +307 -307
  869. package/template/.aioson/skills/static/react-motion-patterns.md +599 -599
  870. package/template/.aioson/skills/static/static-html-patterns/checklists.md +43 -43
  871. package/template/.aioson/skills/static/static-html-patterns/css-tokens.md +609 -609
  872. package/template/.aioson/skills/static/static-html-patterns/motion.md +193 -193
  873. package/template/.aioson/skills/static/static-html-patterns/premium.md +711 -711
  874. package/template/.aioson/skills/static/static-html-patterns/structure.md +209 -209
  875. package/template/.aioson/skills/static/static-html-patterns/utilities.md +190 -190
  876. package/template/.aioson/skills/static/static-html-patterns.md +80 -80
  877. package/template/.aioson/skills/static/tall-stack-patterns.md +286 -286
  878. package/template/.aioson/skills/static/threejs-patterns.md +929 -929
  879. package/template/.aioson/skills/static/ui-ux-modern.md +76 -76
  880. package/template/.aioson/skills/static/web-research-cache.md +115 -115
  881. package/template/.aioson/skills/static/web3-cardano-patterns.md +337 -337
  882. package/template/.aioson/skills/static/web3-ethereum-patterns.md +310 -310
  883. package/template/.aioson/skills/static/web3-security-checklist.md +284 -284
  884. package/template/.aioson/skills/static/web3-solana-patterns.md +324 -324
  885. package/template/.aioson/squads/memory.md +5 -5
  886. package/template/.aioson/tasks/implementation-plan.md +327 -327
  887. package/template/.aioson/tasks/squad-analyze.md +83 -83
  888. package/template/.aioson/tasks/squad-create.md +148 -148
  889. package/template/.aioson/tasks/squad-design.md +206 -206
  890. package/template/.aioson/tasks/squad-execution-plan.md +279 -279
  891. package/template/.aioson/tasks/squad-export.md +20 -20
  892. package/template/.aioson/tasks/squad-extend.md +68 -68
  893. package/template/.aioson/tasks/squad-investigate.md +57 -57
  894. package/template/.aioson/tasks/squad-learning-review.md +44 -44
  895. package/template/.aioson/tasks/squad-output-config.md +177 -177
  896. package/template/.aioson/tasks/squad-pipeline.md +122 -122
  897. package/template/.aioson/tasks/squad-profile.md +48 -48
  898. package/template/.aioson/tasks/squad-refresh.md +236 -0
  899. package/template/.aioson/tasks/squad-repair.md +85 -85
  900. package/template/.aioson/tasks/squad-review.md +61 -61
  901. package/template/.aioson/tasks/squad-task-decompose.md +66 -66
  902. package/template/.aioson/tasks/squad-validate.md +58 -58
  903. package/template/.aioson/templates/reflect-prompts/current-state.md +36 -0
  904. package/template/.aioson/templates/reflect-prompts/how-it-works.md +23 -0
  905. package/template/.aioson/templates/reflect-prompts/what-it-does.md +21 -0
  906. package/template/.aioson/templates/squads/content-basic/template.json +21 -21
  907. package/template/.aioson/templates/squads/digital-marketing-agency/template.json +96 -96
  908. package/template/.aioson/templates/squads/media-channel/template.json +24 -24
  909. package/template/.aioson/templates/squads/research-analysis/template.json +22 -22
  910. package/template/.aioson/templates/squads/software-delivery/template.json +21 -21
  911. package/template/.claude/commands/aioson/agent/analyst.md +5 -5
  912. package/template/.claude/commands/aioson/agent/architect.md +5 -5
  913. package/template/.claude/commands/aioson/agent/briefing.md +5 -0
  914. package/template/.claude/commands/aioson/agent/committer.md +5 -5
  915. package/template/.claude/commands/aioson/agent/copywriter.md +5 -5
  916. package/template/.claude/commands/aioson/agent/design-hybrid-forge.md +5 -5
  917. package/template/.claude/commands/aioson/agent/dev.md +5 -5
  918. package/template/.claude/commands/aioson/agent/deyvin.md +5 -5
  919. package/template/.claude/commands/aioson/agent/discover.md +5 -0
  920. package/template/.claude/commands/aioson/agent/discovery-design-doc.md +5 -5
  921. package/template/.claude/commands/aioson/agent/genome.md +5 -5
  922. package/template/.claude/commands/aioson/agent/neo.md +5 -5
  923. package/template/.claude/commands/aioson/agent/orache.md +5 -5
  924. package/template/.claude/commands/aioson/agent/orchestrator.md +5 -5
  925. package/template/.claude/commands/aioson/agent/pair.md +5 -5
  926. package/template/.claude/commands/aioson/agent/pentester.md +5 -0
  927. package/template/.claude/commands/aioson/agent/pm.md +5 -5
  928. package/template/.claude/commands/aioson/agent/product.md +5 -5
  929. package/template/.claude/commands/aioson/agent/profiler-enricher.md +5 -5
  930. package/template/.claude/commands/aioson/agent/profiler-forge.md +5 -5
  931. package/template/.claude/commands/aioson/agent/profiler-researcher.md +5 -5
  932. package/template/.claude/commands/aioson/agent/qa.md +5 -5
  933. package/template/.claude/commands/aioson/agent/setup.md +5 -5
  934. package/template/.claude/commands/aioson/agent/sheldon.md +5 -5
  935. package/template/.claude/commands/aioson/agent/site-forge.md +5 -5
  936. package/template/.claude/commands/aioson/agent/squad.md +5 -5
  937. package/template/.claude/commands/aioson/agent/tester.md +5 -5
  938. package/template/.claude/commands/aioson/agent/ux-ui.md +5 -5
  939. package/template/.claude/commands/aioson/agent/validator.md +5 -5
  940. package/template/.gemini/GEMINI.md +13 -13
  941. package/template/.gemini/commands/aios-analyst.toml +7 -7
  942. package/template/.gemini/commands/aios-architect.toml +8 -8
  943. package/template/.gemini/commands/aios-committer.toml +7 -7
  944. package/template/.gemini/commands/aios-copywriter.toml +7 -7
  945. package/template/.gemini/commands/aios-cypher.toml +7 -7
  946. package/template/.gemini/commands/aios-dev.toml +9 -9
  947. package/template/.gemini/commands/aios-deyvin.toml +7 -7
  948. package/template/.gemini/commands/aios-discover.toml +6 -0
  949. package/template/.gemini/commands/aios-discovery-design-doc.toml +7 -7
  950. package/template/.gemini/commands/aios-genome.toml +7 -7
  951. package/template/.gemini/commands/aios-neo.toml +6 -6
  952. package/template/.gemini/commands/aios-orache.toml +7 -7
  953. package/template/.gemini/commands/aios-orchestrator.toml +9 -9
  954. package/template/.gemini/commands/aios-pair.toml +7 -7
  955. package/template/.gemini/commands/aios-pm.toml +9 -9
  956. package/template/.gemini/commands/aios-product.toml +6 -6
  957. package/template/.gemini/commands/aios-qa.toml +7 -7
  958. package/template/.gemini/commands/aios-setup.toml +6 -6
  959. package/template/.gemini/commands/aios-sheldon.toml +7 -7
  960. package/template/.gemini/commands/aios-site-forge.toml +7 -7
  961. package/template/.gemini/commands/aios-squad.toml +7 -7
  962. package/template/.gemini/commands/aios-tester.toml +7 -7
  963. package/template/.gemini/commands/aios-ux-ui.toml +9 -9
  964. package/template/.gemini/commands/aios-validator.toml +7 -7
  965. package/template/AGENTS.md +184 -183
  966. package/template/CLAUDE.md +98 -97
  967. package/template/OPENCODE.md +35 -34
  968. package/template/aioson-models.json +40 -40
  969. package/template/.aioson/genomes/copywriting.md +0 -204
  970. package/template/.aioson/genomes/copywriting.meta.json +0 -48
  971. package/template/.aioson/skills/process/secure-tdd/references/nextjs.md +0 -81
  972. package/template/.aioson/skills/process/secure-tdd/references/node-express.md +0 -91
  973. package/template/.aioson/skills/process/secure-tdd/references/planned-stacks.md +0 -33
  974. package/template/.claude/commands/aioson/agent/cypher.md +0 -5
@@ -1,1039 +1,1279 @@
1
- 'use strict';
2
-
3
- const fs = require('node:fs/promises');
4
- const path = require('node:path');
5
- const { getAgentDefinition, resolveInstructionPath, buildAgentPrompt } = require('../agents');
6
- const { normalizeInteractionLanguage } = require('../locales');
7
- const { validateProjectContextFile, getInteractionLanguage } = require('../context');
8
- const { exists, ensureDir } = require('../utils');
9
- const { syncWorkflowRuntime } = require('../execution-gateway');
10
- const { writeHandoff, buildWorkflowHandoff, buildWorkflowHandoffProtocol } = require('../session-handoff');
11
- const { runTechnicalGate, formatGateError } = require('../workflow-gates');
12
- const { buildTestBriefing } = require('../test-briefing');
13
- const { validateHandoffContract, formatContractError, getBlockingRevisions } = require('../handoff-contract');
14
- const { buildPathGuardBlock } = require('../path-guard');
15
- const { logError, buildHealingPrompt } = require('../self-healing');
16
- const { validateHandoffProtocol } = require('../handoff-validator');
17
- const { readAutonomyProtocol, resolveEffectiveMode } = require('../autonomy-policy');
18
- const { readAgentManifest, buildAgentCapabilitySummary } = require('../agent-manifests');
19
- const { inspectStagedChanges } = require('../lib/git-commit-guard');
20
- const { emitSecurityRuntimeEvent } = require('../lib/security/runtime-events');
21
- const { runSecurityAudit } = require('./security-audit');
22
-
23
- const STATE_RELATIVE_PATH = '.aioson/context/workflow.state.json';
24
- const CONFIG_RELATIVE_PATH = '.aioson/context/workflow.config.json';
25
- const EVENTS_RELATIVE_PATH = '.aioson/context/workflow.events.jsonl';
26
-
27
- const DEFAULT_FEATURE_WORKFLOW_BY_CLASSIFICATION = {
28
- MICRO: ['product', 'dev', 'qa'],
29
- SMALL: ['product', 'analyst', 'dev', 'qa'],
30
- MEDIUM: ['product', 'analyst', 'dev', 'pentester', 'qa']
31
- };
32
-
33
- function normalizeAgentName(input) {
34
- return String(input || '')
35
- .trim()
36
- .toLowerCase()
37
- .replace(/^@/, '');
38
- }
39
-
40
- function normalizeClassification(value, fallback = 'MICRO') {
41
- const text = String(value || '').trim().toUpperCase();
42
- if (text === 'MICRO' || text === 'SMALL' || text === 'MEDIUM') return text;
43
- return fallback;
44
- }
45
-
46
- function buildDefaultWorkflowConfig() {
47
- return {
48
- version: 1,
49
- project: {
50
- MICRO: ['setup', 'dev'],
51
- SMALL: ['setup', 'product', 'analyst', 'architect', 'dev', 'qa'],
52
- MEDIUM: ['setup', 'product', 'analyst', 'architect', 'ux-ui', 'pm', 'orchestrator', 'dev', 'qa']
53
- },
54
- feature: DEFAULT_FEATURE_WORKFLOW_BY_CLASSIFICATION,
55
- rules: {
56
- required: ['dev'],
57
- allowDetours: true
58
- }
59
- };
60
- }
61
-
62
- function parseFeaturesMarkdown(markdown) {
63
- return String(markdown || '')
64
- .split(/\r?\n/)
65
- .slice(3)
66
- .map((line) => line.trim())
67
- .filter(Boolean)
68
- .filter((line) => line.startsWith('|'))
69
- .map((line) => line.split('|').map((part) => part.trim()))
70
- .filter((parts) => parts.length >= 5)
71
- .map((parts) => ({
72
- slug: parts[1],
73
- status: parts[2],
74
- started: parts[3],
75
- completed: parts[4]
76
- }))
77
- .filter((row) => row.slug && row.slug !== 'slug')
78
- .filter((row) => !/^-+$/ .test(row.slug));
79
- }
80
-
81
- async function readJsonIfExists(filePath) {
82
- if (!(await exists(filePath))) return null;
83
- const content = await fs.readFile(filePath, 'utf8');
84
- return JSON.parse(content);
85
- }
86
-
87
- async function writeJson(filePath, payload) {
88
- await ensureDir(path.dirname(filePath));
89
- await fs.writeFile(filePath, `${JSON.stringify(payload, null, 2)}\n`, 'utf8');
90
- }
91
-
92
- async function appendJsonLine(filePath, payload) {
93
- await ensureDir(path.dirname(filePath));
94
- await fs.appendFile(filePath, `${JSON.stringify(payload)}\n`, 'utf8');
95
- }
96
-
97
- function buildWorkflowEventMessage({ created, state, activation, completedStage, options }) {
98
- const requestedAgent = options.requestedAgent ? normalizeAgentName(options.requestedAgent) : null;
99
- if (requestedAgent && activation.agent && requestedAgent !== activation.agent) {
100
- return `Workflow enforced @${activation.agent} after direct request for @${requestedAgent}.`;
101
- }
102
- if (completedStage && activation.agent) {
103
- return `Completed @${completedStage}. Next stage ready: @${activation.agent}.`;
104
- }
105
- if (completedStage && !activation.agent) {
106
- return `Completed @${completedStage}. Workflow has no pending stage.`;
107
- }
108
- if (state.detour && state.detour.active) {
109
- return `Detour started with @${state.detour.agent}. Return to ${
110
- state.detour.returnTo ? `@${state.detour.returnTo}` : 'the main flow'
111
- }.`;
112
- }
113
- if (options.skip && activation.agent) {
114
- return `Workflow advanced to @${activation.agent} after skip.`;
115
- }
116
- if (activation.agent) {
117
- return created
118
- ? `Workflow initialized at @${activation.agent}.`
119
- : `Stage @${activation.agent} is active.`;
120
- }
121
- return 'Workflow has no pending stage.';
122
- }
123
-
124
- function buildWorkflowEventType({ completedStage, state, activation, options }) {
125
- const requestedAgent = options.requestedAgent ? normalizeAgentName(options.requestedAgent) : null;
126
- if (requestedAgent && activation.agent && requestedAgent !== activation.agent) return 'routed';
127
- if (completedStage) return 'completed';
128
- if (state.detour && state.detour.active) return 'workflow';
129
- if (options.skip) return 'workflow';
130
- if (activation.agent) return 'start';
131
- return 'workflow';
132
- }
133
-
134
- async function appendWorkflowEvent(targetDir, payload) {
135
- const eventsPath = path.join(targetDir, EVENTS_RELATIVE_PATH);
136
- await appendJsonLine(eventsPath, payload);
137
- return eventsPath;
138
- }
139
-
140
- async function readWorkflowConfig(targetDir) {
141
- const configPath = path.join(targetDir, CONFIG_RELATIVE_PATH);
142
- const userConfig = await readJsonIfExists(configPath);
143
- const base = buildDefaultWorkflowConfig();
144
- if (!userConfig || typeof userConfig !== 'object') {
145
- return { configPath, config: base, exists: false };
146
- }
147
-
148
- const merged = {
149
- ...base,
150
- ...userConfig,
151
- project: {
152
- ...base.project,
153
- ...(userConfig.project || {})
154
- },
155
- feature: {
156
- ...base.feature,
157
- ...(userConfig.feature || {})
158
- },
159
- rules: {
160
- ...base.rules,
161
- ...(userConfig.rules || {})
162
- }
163
- };
164
-
165
- return { configPath, config: merged, exists: true };
166
- }
167
-
168
- async function resolveLocaleForTarget(targetDir, options) {
169
- const fromOption = options.language || options.lang;
170
- if (fromOption) return normalizeInteractionLanguage(fromOption);
171
-
172
- const context = await validateProjectContextFile(targetDir);
173
- if (context.parsed && context.data) {
174
- return getInteractionLanguage(context.data, 'en');
175
- }
176
-
177
- return 'en';
178
- }
179
-
180
- async function resolveExistingInstructionPath(targetDir, agent, locale) {
181
- const candidate = resolveInstructionPath(agent, locale);
182
- const candidateAbs = path.join(targetDir, candidate);
183
- if (await exists(candidateAbs)) return candidate;
184
- return agent.path;
185
- }
186
-
187
- async function detectWorkflowMode(targetDir) {
188
- const prdPath = path.join(targetDir, '.aioson/context/prd.md');
189
- const featuresPath = path.join(targetDir, '.aioson/context/features.md');
190
- const hasProjectPrd = await exists(prdPath);
191
- const featuresMarkdown = await fs.readFile(featuresPath, 'utf8').catch(() => '');
192
- const features = parseFeaturesMarkdown(featuresMarkdown);
193
- const activeFeature = features.find((feature) => feature.status === 'in_progress') || null;
194
-
195
- if (activeFeature) {
196
- return {
197
- mode: 'feature',
198
- featureSlug: activeFeature.slug,
199
- features
200
- };
201
- }
202
-
203
- return {
204
- mode: hasProjectPrd ? 'project' : 'project',
205
- featureSlug: null,
206
- features
207
- };
208
- }
209
-
210
- function getSequenceForMode(config, mode, classification) {
211
- const group = mode === 'feature' ? config.feature : config.project;
212
- const sequence = group[normalizeClassification(classification, 'MICRO')];
213
- return Array.isArray(sequence) && sequence.length > 0 ? [...sequence] : [];
214
- }
215
-
216
- async function validateStageArtifacts(targetDir, state, stage) {
217
- const base = path.join(targetDir, '.aioson/context');
218
- const slug = state.featureSlug;
219
-
220
- if (stage === 'setup') {
221
- const context = await validateProjectContextFile(targetDir);
222
- return context.valid;
223
- }
224
-
225
- if (stage === 'product') {
226
- if (state.mode === 'feature' && slug) {
227
- const prdFeature = path.join(base, `prd-${slug}.md`);
228
- const prdFix = path.join(base, `prd-${slug}-fix.md`);
229
- return (await exists(prdFeature)) || (await exists(prdFix));
230
- }
231
- return await exists(path.join(base, 'prd.md'));
232
- }
233
-
234
- if (stage === 'analyst') {
235
- if (state.mode === 'feature' && slug) {
236
- const requirements = path.join(base, `requirements-${slug}.md`);
237
- const spec = path.join(base, `spec-${slug}.md`);
238
- return (await exists(requirements)) && (await exists(spec));
239
- }
240
- return await exists(path.join(base, 'discovery.md'));
241
- }
242
-
243
- if (stage === 'architect') {
244
- return await exists(path.join(base, 'architecture.md'));
245
- }
246
-
247
- if (stage === 'ux-ui') {
248
- return await exists(path.join(base, 'ui-spec.md'));
249
- }
250
-
251
- if (stage === 'orchestrator') {
252
- return await exists(path.join(base, 'parallel'));
253
- }
254
-
255
- return true;
256
- }
257
-
258
- function isRequiredAgent(config, agentName) {
259
- return Array.isArray(config.rules?.required)
260
- ? config.rules.required.map(normalizeAgentName).includes(agentName)
261
- : false;
262
- }
263
-
264
- function buildStatePayload(input) {
265
- return {
266
- version: 1,
267
- mode: input.mode,
268
- classification: input.classification,
269
- sequence: input.sequence,
270
- current: input.current || null,
271
- next: input.next || null,
272
- completed: Array.isArray(input.completed) ? input.completed : [],
273
- skipped: Array.isArray(input.skipped) ? input.skipped : [],
274
- featureSlug: input.featureSlug || null,
275
- detour: input.detour || null,
276
- updatedAt: new Date().toISOString()
277
- };
278
- }
279
-
280
- function findNextFromSequence(sequence, completed, skipped) {
281
- const done = new Set([...(completed || []), ...(skipped || [])].map(normalizeAgentName));
282
- return sequence.find((stage) => !done.has(normalizeAgentName(stage))) || null;
283
- }
284
-
285
- function reconcileWorkflowState(state) {
286
- if (!state || typeof state !== 'object' || !Array.isArray(state.sequence)) {
287
- return { state, changed: false };
288
- }
289
-
290
- const sequence = state.sequence.map(normalizeAgentName);
291
- const completed = Array.from(new Set((state.completed || []).map(normalizeAgentName).filter(Boolean)));
292
- const skippedSet = new Set((state.skipped || []).map(normalizeAgentName).filter(Boolean));
293
- const detour = state.detour && typeof state.detour === 'object'
294
- ? {
295
- ...state.detour,
296
- agent: normalizeAgentName(state.detour.agent),
297
- returnTo: normalizeAgentName(state.detour.returnTo)
298
- }
299
- : null;
300
- let changed = false;
301
-
302
- // If a later stage is already completed, any unresolved earlier stage was
303
- // effectively bypassed outside the workflow and must not remain "active".
304
- const furthestCompletedIndex = sequence.reduce((max, stage, index) => (
305
- completed.includes(stage) ? Math.max(max, index) : max
306
- ), -1);
307
-
308
- if (furthestCompletedIndex >= 0) {
309
- for (let index = 0; index < furthestCompletedIndex; index += 1) {
310
- const stage = sequence[index];
311
- if (!completed.includes(stage) && !skippedSet.has(stage)) {
312
- skippedSet.add(stage);
313
- changed = true;
314
- }
315
- }
316
- }
317
-
318
- const skipped = sequence.filter((stage) => skippedSet.has(stage));
319
- const resolved = new Set([...completed, ...skipped]);
320
- let current = state.current ? normalizeAgentName(state.current) : null;
321
- let next = state.next ? normalizeAgentName(state.next) : null;
322
- const currentIsActiveDetour = Boolean(
323
- detour &&
324
- detour.active &&
325
- current &&
326
- current === detour.agent
327
- );
328
-
329
- if (current && ((!sequence.includes(current) && !currentIsActiveDetour) || resolved.has(current))) {
330
- current = null;
331
- changed = true;
332
- }
333
-
334
- if (!detour || !detour.active) {
335
- if (current) {
336
- const currentIndex = sequence.indexOf(current);
337
- const expectedQueuedNext = sequence.find(
338
- (stage, index) => index > currentIndex && !resolved.has(stage)
339
- ) || null;
340
- if (next && next !== current && next !== expectedQueuedNext) {
341
- next = expectedQueuedNext || current;
342
- changed = true;
343
- } else if (!next) {
344
- next = expectedQueuedNext || current;
345
- changed = true;
346
- }
347
- } else {
348
- const inferredNext = findNextFromSequence(sequence, completed, skipped);
349
- if (next !== inferredNext) {
350
- next = inferredNext;
351
- changed = true;
352
- }
353
- }
354
- }
355
-
356
- if (!changed) {
357
- return { state, changed: false };
358
- }
359
-
360
- return {
361
- changed: true,
362
- state: buildStatePayload({
363
- ...state,
364
- sequence,
365
- completed,
366
- skipped,
367
- current,
368
- next,
369
- detour
370
- })
371
- };
372
- }
373
-
374
- function isInferableStage(stage) {
375
- return ['setup', 'product', 'analyst', 'architect', 'ux-ui', 'orchestrator'].includes(
376
- normalizeAgentName(stage)
377
- );
378
- }
379
-
380
- function isSecurityGateBlocked(contractCheck, state, stageName) {
381
- if (normalizeAgentName(stageName) !== 'qa' || state.mode !== 'feature' || !state.featureSlug) {
382
- return false;
383
- }
384
- return contractCheck.missing.some((item) =>
385
- item.includes('security:') ||
386
- item.includes(`security-findings-${state.featureSlug}.json`)
387
- );
388
- }
389
-
390
- function buildQaSecurityAuditBriefing(result, targetDir) {
391
- if (!result) return '';
392
-
393
- if (result.ok === false && result.reason) {
394
- return [
395
- '## Secure by Default audit',
396
- `- Auto-run failed before QA review: ${result.reason}.`,
397
- '- Gate D will remain blocked until a valid `security-findings-{slug}.json` artifact exists.',
398
- '- If CLI is unavailable in your client, use the fallback checklist and record the limitation explicitly in the QA report and `project-pulse.md`.'
399
- ].join('\n');
400
- }
401
-
402
- return [
403
- '## Secure by Default audit',
404
- `- Auto-ran \`security:audit\` for feature \`${result.slug}\` at QA activation.`,
405
- `- Exit code: ${result.exitCode}. Findings: ${result.findingsCount}.`,
406
- `- Summary: critical=${result.summary.critical}, high=${result.summary.high}, medium=${result.summary.medium}, low=${result.summary.low}, inconclusive=${result.summary.inconclusive}.`,
407
- `- Artifact: \`${path.relative(targetDir, result.artifactPath)}\`.`,
408
- '- If the audit or manual heuristics indicate auth, money, or ownership risk, invoke `@pentester` with `--mode=app_target --feature=<slug> --scope=<target>` before final Gate D sign-off.',
409
- '- If CLI is unavailable in your client, use the fallback checklist and record the limitation explicitly in the QA report and `project-pulse.md`.'
410
- ].join('\n');
411
- }
412
-
413
- async function inferCompletedStages(targetDir, draftState) {
414
- const completed = [];
415
- for (const stage of draftState.sequence) {
416
- if (!isInferableStage(stage)) break;
417
- const valid = await validateStageArtifacts(targetDir, draftState, stage);
418
- if (!valid) break;
419
- completed.push(normalizeAgentName(stage));
420
- }
421
- return completed;
422
- }
423
-
424
- async function loadOrCreateState(targetDir, options = {}) {
425
- const statePath = path.join(targetDir, STATE_RELATIVE_PATH);
426
- const existing = await readJsonIfExists(statePath);
427
- if (existing && typeof existing === 'object' && Array.isArray(existing.sequence)) {
428
- const reconciled = reconcileWorkflowState(existing);
429
- if (reconciled.changed) {
430
- await writeJson(statePath, reconciled.state);
431
- }
432
- return { statePath, state: reconciled.state, created: false };
433
- }
434
-
435
- const context = await validateProjectContextFile(targetDir);
436
- const classification = normalizeClassification(
437
- options.classification || (context.data && context.data.classification) || 'MICRO',
438
- 'MICRO'
439
- );
440
- const modeInfo = await detectWorkflowMode(targetDir);
441
- const { config } = await readWorkflowConfig(targetDir);
442
- const sequence = getSequenceForMode(config, modeInfo.mode, classification);
443
- const draftState = buildStatePayload({
444
- mode: modeInfo.mode,
445
- classification,
446
- sequence,
447
- current: null,
448
- next: null,
449
- completed: [],
450
- skipped: [],
451
- featureSlug: modeInfo.featureSlug,
452
- detour: null
453
- });
454
- const completed = await inferCompletedStages(targetDir, draftState);
455
- const next = findNextFromSequence(sequence, completed, []);
456
- const state = buildStatePayload({
457
- mode: modeInfo.mode,
458
- classification,
459
- sequence,
460
- current: null,
461
- next,
462
- completed,
463
- skipped: [],
464
- featureSlug: modeInfo.featureSlug,
465
- detour: null
466
- });
467
-
468
- await writeJson(statePath, state);
469
- return { statePath, state, created: true };
470
- }
471
-
472
- async function persistState(targetDir, nextState) {
473
- const statePath = path.join(targetDir, STATE_RELATIVE_PATH);
474
- await writeJson(statePath, nextState);
475
- return statePath;
476
- }
477
-
478
- function ensureAgentInSequence(state, agentName) {
479
- if (state.sequence.includes(agentName)) return;
480
- throw new Error(`Agent ${agentName} is not part of the active workflow sequence.`);
481
- }
482
-
483
- function ensureSkippableTarget(config, state, targetAgent) {
484
- const normalizedTarget = normalizeAgentName(targetAgent);
485
- ensureAgentInSequence(state, normalizedTarget);
486
-
487
- const currentIndex = state.next ? state.sequence.indexOf(state.next) : -1;
488
- const targetIndex = state.sequence.indexOf(normalizedTarget);
489
- const devIndex = state.sequence.indexOf('dev');
490
-
491
- if (currentIndex === -1) {
492
- throw new Error('No next stage is available to skip from.');
493
- }
494
- if (targetIndex === -1 || targetIndex < currentIndex) {
495
- throw new Error(`Cannot skip backwards to ${targetAgent}.`);
496
- }
497
- if (normalizedTarget === 'dev') return;
498
- if (devIndex !== -1 && targetIndex > devIndex) {
499
- throw new Error('Cannot skip past @dev because @dev is mandatory.');
500
- }
501
- if (isRequiredAgent(config, normalizedTarget) && normalizedTarget !== 'dev') {
502
- return;
503
- }
504
- }
505
-
506
- async function finalizeCurrentStage(targetDir, config, state, stageName) {
507
- const normalizedStage = normalizeAgentName(stageName || state.current || state.next);
508
- if (!normalizedStage) {
509
- throw new Error('No stage is active to complete.');
510
- }
511
-
512
- // ── Harness Done Gate ───────────────────────────────────────────────────
513
- if (state.mode === 'feature' && state.featureSlug) {
514
- const contractPath = path.join(targetDir, '.aioson', 'plans', state.featureSlug, 'harness-contract.json');
515
- const progressPath = path.join(targetDir, '.aioson', 'plans', state.featureSlug, 'progress.json');
516
-
517
- // Se contrato existe, verificamos o progresso
518
- const fs = require('node:fs');
519
- if (fs.existsSync(contractPath) && fs.existsSync(progressPath)) {
520
- try {
521
- const progress = JSON.parse(fs.readFileSync(progressPath, 'utf8'));
522
- // Bloqueia se não estiver pronto para o gate E o estágio for crítico (dev/qa)
523
- if (!progress.ready_for_done_gate && (normalizedStage === 'dev' || normalizedStage === 'qa')) {
524
- throw new Error(`[Harness Block] A feature "${state.featureSlug}" não passou na validação contratual. Execute 'aioson harness:validate' e resolva os problemas antes de concluir o estágio @${normalizedStage}.`);
525
- }
526
- } catch (err) {
527
- if (err.message.includes('[Harness Block]')) throw err;
528
- // Se erro de parse, ignoramos para não quebrar o workflow por corrupção
529
- }
530
- }
531
- }
532
-
533
- if (state.detour && state.detour.active && normalizeAgentName(state.detour.agent) === normalizedStage) {
534
- const validDetour = await validateStageArtifacts(targetDir, state, normalizedStage);
535
- if (!validDetour) {
536
- throw new Error(`Cannot complete detour ${normalizedStage}; expected artifacts are missing.`);
537
- }
538
- const nextState = buildStatePayload({
539
- ...state,
540
- current: null,
541
- next: state.detour.returnTo,
542
- detour: null
543
- });
544
- return { state: nextState, completedStage: normalizedStage };
545
- }
546
-
547
- ensureAgentInSequence(state, normalizedStage);
548
- const valid = await validateStageArtifacts(targetDir, state, normalizedStage);
549
- if (!valid) {
550
- throw new Error(`Cannot complete ${normalizedStage}; expected artifacts are missing.`);
551
- }
552
-
553
- // ── Handoff Contract Gate ───────────────────────────────────────────────
554
- const contractCheck = await validateHandoffContract(targetDir, state, normalizedStage);
555
- if (!contractCheck.ok) {
556
- if (isSecurityGateBlocked(contractCheck, state, normalizedStage)) {
557
- await emitSecurityRuntimeEvent({
558
- targetDir,
559
- eventType: 'security_gate_blocked',
560
- message: `Gate D blocked for ${state.featureSlug} at @qa`,
561
- status: 'failed',
562
- agentName: 'qa',
563
- source: 'workflow',
564
- workflowState: state,
565
- workflowStage: 'qa',
566
- payload: {
567
- feature_slug: state.featureSlug,
568
- classification: state.classification,
569
- blockers: contractCheck.missing
570
- }
571
- });
572
- }
573
- const errMsg = formatContractError(contractCheck);
574
- await logError(targetDir, normalizedStage, errMsg, 'contract');
575
- throw new Error(errMsg);
576
- }
577
-
578
- // ── Revision Gate (Phase 2) ─────────────────────────────────────────────
579
- const blockingRevisions = await getBlockingRevisions(targetDir, state.featureSlug);
580
- if (blockingRevisions.length > 0) {
581
- const ids = blockingRevisions.map((r) => r.id).join(', ');
582
- const errMsg = [
583
- `[Revision Gate BLOCKED]`,
584
- `Feature: ${state.featureSlug}`,
585
- ``,
586
- `Pending blocking revision(s): ${ids}`,
587
- ``,
588
- `Resolve each revision before completing this stage:`,
589
- ...blockingRevisions.map((r) => ` aioson revision:resolve . --slug=${state.featureSlug} --rev-id=${r.id} --approve|--reject`)
590
- ].join('\n');
591
- await logError(targetDir, normalizedStage, errMsg, 'revision');
592
- throw new Error(errMsg);
593
- }
594
-
595
- // ── Technical Compilation/Test Gate ─────────────────────────────────────
596
- const techGate = await runTechnicalGate(targetDir, normalizedStage);
597
- if (!techGate.ok) {
598
- const errMsg = formatGateError(techGate);
599
- await logError(targetDir, normalizedStage, errMsg, 'technical');
600
- throw new Error(errMsg);
601
- }
602
-
603
- const completed = Array.from(new Set([...(state.completed || []), normalizedStage]));
604
- const next = findNextFromSequence(state.sequence, completed, state.skipped || []);
605
- const nextState = buildStatePayload({
606
- ...state,
607
- completed,
608
- current: null,
609
- next,
610
- detour: null
611
- });
612
-
613
- return { state: nextState, completedStage: normalizedStage };
614
- }
615
-
616
- function applySkip(config, state, target) {
617
- const normalizedTarget = normalizeAgentName(target);
618
- ensureSkippableTarget(config, state, normalizedTarget);
619
- const currentIndex = state.sequence.indexOf(state.next);
620
- const targetIndex = state.sequence.indexOf(normalizedTarget);
621
- const toSkip = state.sequence.slice(currentIndex, targetIndex);
622
- if (toSkip.some((agent) => normalizeAgentName(agent) === 'dev')) {
623
- throw new Error('Cannot skip @dev because it is mandatory.');
624
- }
625
-
626
- const skipped = Array.from(new Set([...(state.skipped || []), ...toSkip]));
627
- return buildStatePayload({
628
- ...state,
629
- skipped,
630
- current: null,
631
- next: normalizedTarget
632
- });
633
- }
634
-
635
- async function activateStage(targetDir, state, locale, tool, explicitAgent = null, requestedMode = null) {
636
- const stageName = normalizeAgentName(explicitAgent || state.current || state.next);
637
- if (!stageName) {
638
- return {
639
- state,
640
- agent: null,
641
- instructionPath: null,
642
- prompt: null
643
- };
644
- }
645
-
646
- // ── Committer Safety Gate ───────────────────────────────────────────────
647
- if (stageName === 'committer') {
648
- const guard = await inspectStagedChanges(targetDir, { allowWarnings: false });
649
- if (guard.summary.stagedCount === 0) {
650
- throw new Error(
651
- `[Committer Gate BLOCKED] Nenhum arquivo no stage para commit. ` +
652
- `Execute primeiro: aioson commit:prepare . --agent-safe --staged-only --mode=headless`
653
- );
654
- }
655
- if (!guard.ok) {
656
- throw new Error(
657
- `[Committer Gate BLOCKED] Arquivos proibidos detectados no stage ` +
658
- `(node_modules, build artifacts, secrets, etc.). ` +
659
- `Execute 'aioson git:guard .' para ver detalhes, corrija e rode 'aioson commit:prepare . --agent-safe --staged-only --mode=headless' antes de ativar @committer.`
660
- );
661
- }
662
- }
663
-
664
- // ── Test Briefing Injection for qa/tester ───────────────────────────────
665
- let testBriefing = '';
666
- let securityAuditBriefing = '';
667
- if (stageName === 'qa' || stageName === 'tester') {
668
- try {
669
- testBriefing = await buildTestBriefing(targetDir);
670
- } catch {
671
- // Non-fatal: if briefing generation fails, proceed without it
672
- testBriefing = '';
673
- }
674
- }
675
-
676
- if (
677
- stageName === 'qa' &&
678
- state.mode === 'feature' &&
679
- state.classification === 'MEDIUM' &&
680
- state.featureSlug
681
- ) {
682
- try {
683
- const auditResult = await runSecurityAudit({
684
- args: [targetDir],
685
- options: {
686
- slug: state.featureSlug,
687
- json: true,
688
- runtimeAgentName: 'qa',
689
- runtimeSource: 'workflow',
690
- runtimeState: state,
691
- runtimeWorkflowStage: 'qa'
692
- },
693
- logger: { log() {}, error() {}, warn() {} }
694
- });
695
- securityAuditBriefing = buildQaSecurityAuditBriefing(auditResult, targetDir);
696
- } catch {
697
- securityAuditBriefing = buildQaSecurityAuditBriefing({
698
- ok: false,
699
- reason: 'audit_runtime_failure'
700
- }, targetDir);
701
- }
702
- }
703
-
704
- // ── Path Guard Injection for implementation agents ────────────────────────
705
- let pathGuardBlock = '';
706
- if (['dev', 'architect', 'ux-ui', 'pentester', 'qa', 'tester', 'committer'].includes(stageName)) {
707
- try {
708
- pathGuardBlock = await buildPathGuardBlock(targetDir);
709
- } catch {
710
- pathGuardBlock = '';
711
- }
712
- }
713
-
714
- const agent = getAgentDefinition(stageName);
715
- if (!agent) {
716
- throw new Error(`Unknown agent: ${stageName}`);
717
- }
718
-
719
- const autonomyProtocol = await readAutonomyProtocol(targetDir);
720
- const agentManifest = await readAgentManifest(targetDir, agent.id);
721
- const effectiveMode = resolveEffectiveMode({
722
- protocol: autonomyProtocol,
723
- tool,
724
- agentId: agent.id,
725
- manifest: agentManifest,
726
- requestedMode
727
- });
728
-
729
- const instructionPath = await resolveExistingInstructionPath(targetDir, agent, locale);
730
- let prompt = buildAgentPrompt(agent, tool, {
731
- instructionPath,
732
- targetDir,
733
- interactionLanguage: locale,
734
- autonomyMode: effectiveMode,
735
- capabilitySummary: buildAgentCapabilitySummary(agentManifest, tool)
736
- });
737
-
738
- if (testBriefing) {
739
- prompt += '\n\n' + testBriefing;
740
- }
741
-
742
- if (securityAuditBriefing) {
743
- prompt += '\n\n' + securityAuditBriefing;
744
- }
745
-
746
- if (pathGuardBlock) {
747
- prompt += '\n\n' + pathGuardBlock;
748
- }
749
-
750
- let nextState = state;
751
- if (explicitAgent && stageName !== normalizeAgentName(state.next)) {
752
- nextState = buildStatePayload({
753
- ...state,
754
- current: stageName,
755
- detour: {
756
- active: true,
757
- agent: stageName,
758
- returnTo: state.next
759
- }
760
- });
761
- } else {
762
- nextState = buildStatePayload({
763
- ...state,
764
- current: stageName
765
- });
766
- }
767
-
768
- return {
769
- state: nextState,
770
- agent: stageName,
771
- instructionPath,
772
- prompt,
773
- effectiveMode
774
- };
775
- }
776
-
777
- async function runWorkflowNext({ args, options, logger, t }) {
778
- if (options.status || options.suggest) {
779
- const { runWorkflowStatus } = require('./workflow-status');
780
- return runWorkflowStatus({ args, options, logger, t });
781
- }
782
-
783
- const logErrorLine = typeof logger.error === 'function'
784
- ? logger.error.bind(logger)
785
- : typeof logger.log === 'function'
786
- ? logger.log.bind(logger)
787
- : () => {};
788
-
789
- const targetDir = path.resolve(process.cwd(), args[0] || '.');
790
- const tool = options.tool || 'codex';
791
- const locale = await resolveLocaleForTarget(targetDir, options);
792
- const { config } = await readWorkflowConfig(targetDir);
793
- const loaded = await loadOrCreateState(targetDir, options);
794
- let state = loaded.state;
795
- let completedStage = null;
796
-
797
- if (options.complete || options['complete-current']) {
798
- let finalized;
799
- try {
800
- finalized = await finalizeCurrentStage(
801
- targetDir,
802
- config,
803
- state,
804
- options.complete === true ? state.current || state.next : options.complete
805
- );
806
- } catch (err) {
807
- // ── Auto-heal intercept ───────────────────────────────────────────────
808
- const autoHeal = Boolean(options['auto-heal'] || options.autoHeal);
809
- const isHealabled = autoHeal && (
810
- err.message.includes('[Technical Gate BLOCKED]') ||
811
- err.message.includes('[Handoff Contract BLOCKED]')
812
- );
813
- if (isHealabled) {
814
- const failedStage = normalizeAgentName(options.complete === true ? state.current || state.next : options.complete);
815
- await logError(targetDir, failedStage, err.message, 'technical');
816
- const retryCount = await require('../self-healing').getRetryCount(targetDir, failedStage);
817
- if (retryCount < require('../self-healing').MAX_RETRIES) {
818
- await require('../self-healing').incrementRetryCount(targetDir, failedStage, err.message.substring(0, 200));
819
- // Build healing activation
820
- const baseActivation = await activateStage(targetDir, state, locale, tool, failedStage, options.mode || null);
821
- const healingPrompt = buildHealingPrompt(
822
- baseActivation.prompt || '',
823
- failedStage,
824
- { error: err.message },
825
- retryCount + 1
826
- );
827
- const healedState = {
828
- ...baseActivation.state,
829
- current: failedStage,
830
- detour: null
831
- };
832
- await persistState(targetDir, healedState);
833
- const eventPayload = {
834
- id: Date.now(),
835
- kind: 'workflow',
836
- createdAt: new Date().toISOString(),
837
- eventType: 'heal',
838
- message: `Auto-heal @${failedStage} — retry ${retryCount + 1}/3`,
839
- mode: state.mode,
840
- classification: state.classification,
841
- featureSlug: state.featureSlug,
842
- current: failedStage,
843
- next: state.next,
844
- completed: state.completed,
845
- skipped: state.skipped,
846
- sequence: state.sequence,
847
- healing: true,
848
- retryCount: retryCount + 1,
849
- autonomyMode: baseActivation.effectiveMode || null
850
- };
851
- await appendWorkflowEvent(targetDir, eventPayload);
852
- const runtime = await syncWorkflowRuntime(targetDir, {
853
- state: healedState,
854
- eventPayload,
855
- activationAgent: failedStage,
856
- completedStage: null
857
- });
858
- const healingHandoff = buildWorkflowHandoff(healedState, null, failedStage);
859
- healingHandoff.protocol = buildWorkflowHandoffProtocol(healedState, null, failedStage, {
860
- autonomyMode: baseActivation.effectiveMode || null,
861
- handoffContractOk: true,
862
- technicalGateOk: false,
863
- artifactUris: []
864
- });
865
- const healingValidation = await validateHandoffProtocol(targetDir, healingHandoff.protocol);
866
- if (!healingValidation.ok) {
867
- logErrorLine('Handoff protocol warning:');
868
- for (const err of healingValidation.errors) logErrorLine(` - ${err}`);
869
- }
870
- await writeHandoff(targetDir, healingHandoff);
871
- logger.log(t('workflow_heal.title', { stage: `@${failedStage}`, count: retryCount + 1 }));
872
- logger.log(healingPrompt);
873
- return {
874
- ok: true,
875
- targetDir,
876
- locale,
877
- tool,
878
- statePath: STATE_RELATIVE_PATH,
879
- configPath: CONFIG_RELATIVE_PATH,
880
- created: loaded.created,
881
- mode: state.mode,
882
- classification: state.classification,
883
- current: healedState.current,
884
- next: healedState.next,
885
- detour: healedState.detour,
886
- completed: healedState.completed,
887
- skipped: healedState.skipped,
888
- completedStage: null,
889
- featureSlug: state.featureSlug,
890
- runtime,
891
- agent: failedStage,
892
- instructionPath: baseActivation.instructionPath,
893
- prompt: healingPrompt,
894
- autoHealed: true,
895
- effectiveMode: baseActivation.effectiveMode || null
896
- };
897
- }
898
- }
899
- throw err;
900
- }
901
- state = finalized.state;
902
- completedStage = finalized.completedStage;
903
- await require('../self-healing').incrementRetryCount(targetDir, completedStage, '');
904
- const { getRetryCount } = require('../self-healing');
905
- const retries = await getRetryCount(targetDir, completedStage);
906
- if (retries > 0) {
907
- // Reset retry count on successful completion after healing
908
- const retriesPath = path.join(targetDir, '.aioson/context/pipeline-retries', `${completedStage}.json`);
909
- try { await fs.unlink(retriesPath); } catch { /* ignore */ }
910
- }
911
- }
912
-
913
- if (options.skip) {
914
- state = applySkip(config, state, options.skip);
915
- }
916
-
917
- const requestedAgent = options.agent ? normalizeAgentName(options.agent) : null;
918
- const activation = await activateStage(targetDir, state, locale, tool, requestedAgent, options.mode || null);
919
- state = activation.state;
920
- const statePath = await persistState(targetDir, state);
921
- const eventPayload = {
922
- id: Date.now(),
923
- kind: 'workflow',
924
- createdAt: new Date().toISOString(),
925
- eventType: buildWorkflowEventType({ completedStage, state, activation, options }),
926
- message: buildWorkflowEventMessage({
927
- created: loaded.created,
928
- state,
929
- activation,
930
- completedStage,
931
- options
932
- }),
933
- mode: state.mode,
934
- classification: state.classification,
935
- featureSlug: state.featureSlug,
936
- current: state.current,
937
- next: state.detour && state.detour.active ? state.detour.returnTo : state.next,
938
- completedStage,
939
- detour: state.detour,
940
- requestedAgent: options.requestedAgent ? normalizeAgentName(options.requestedAgent) : null,
941
- completed: state.completed,
942
- skipped: state.skipped,
943
- sequence: state.sequence,
944
- autonomyMode: activation.effectiveMode || null
945
- };
946
- await appendWorkflowEvent(targetDir, eventPayload);
947
- const runtime = await syncWorkflowRuntime(targetDir, {
948
- state,
949
- eventPayload,
950
- activationAgent: activation.agent,
951
- completedStage
952
- });
953
-
954
- // Generate session handoff when a stage completes or workflow finishes
955
- if (completedStage || !activation.agent) {
956
- const handoffData = buildWorkflowHandoff(state, completedStage, activation.agent);
957
- handoffData.autonomyMode = activation.effectiveMode || null;
958
- handoffData.protocol = buildWorkflowHandoffProtocol(state, completedStage, activation.agent, {
959
- autonomyMode: activation.effectiveMode || null,
960
- handoffContractOk: true,
961
- technicalGateOk: true,
962
- artifactUris: []
963
- });
964
- const handoffValidation = await validateHandoffProtocol(targetDir, handoffData.protocol);
965
- if (!handoffValidation.ok) {
966
- logErrorLine('Handoff protocol warning:');
967
- for (const err of handoffValidation.errors) logErrorLine(` - ${err}`);
968
- }
969
- await writeHandoff(targetDir, handoffData);
970
- }
971
-
972
- const payload = {
973
- ok: true,
974
- targetDir,
975
- locale,
976
- tool,
977
- statePath: STATE_RELATIVE_PATH,
978
- configPath: CONFIG_RELATIVE_PATH,
979
- created: loaded.created,
980
- mode: state.mode,
981
- classification: state.classification,
982
- current: state.current,
983
- next: state.detour && state.detour.active ? state.detour.returnTo : state.next,
984
- detour: state.detour,
985
- completed: state.completed,
986
- skipped: state.skipped,
987
- completedStage,
988
- featureSlug: state.featureSlug,
989
- runtime,
990
- agent: activation.agent,
991
- effectiveMode: activation.effectiveMode || null,
992
- instructionPath: activation.instructionPath,
993
- prompt: activation.prompt
994
- };
995
-
996
- logger.log(t('workflow_next.title', {
997
- mode: state.mode,
998
- classification: state.classification
999
- }));
1000
- if (completedStage) {
1001
- logger.log(t('workflow_next.completed', { agent: `@${completedStage}` }));
1002
- }
1003
- if (state.detour && state.detour.active) {
1004
- logger.log(
1005
- t('workflow_next.detour', {
1006
- agent: `@${state.detour.agent}`,
1007
- returnTo: `@${state.detour.returnTo}`
1008
- })
1009
- );
1010
- }
1011
- if (activation.agent) {
1012
- logger.log(t('workflow_next.current_agent', { agent: `@${activation.agent}` }));
1013
- if (payload.next) {
1014
- logger.log(t('workflow_next.next_agent', { agent: `@${payload.next}` }));
1015
- }
1016
- logger.log(activation.prompt);
1017
- } else {
1018
- logger.log(t('workflow_next.done'));
1019
- }
1020
- logger.log(t('workflow_next.state_file', { path: STATE_RELATIVE_PATH }));
1021
-
1022
- return payload;
1023
- }
1024
-
1025
- module.exports = {
1026
- STATE_RELATIVE_PATH,
1027
- CONFIG_RELATIVE_PATH,
1028
- EVENTS_RELATIVE_PATH,
1029
- buildDefaultWorkflowConfig,
1030
- parseFeaturesMarkdown,
1031
- readWorkflowConfig,
1032
- detectWorkflowMode,
1033
- loadOrCreateState,
1034
- reconcileWorkflowState,
1035
- finalizeCurrentStage,
1036
- applySkip,
1037
- activateStage,
1038
- runWorkflowNext
1039
- };
1
+ 'use strict';
2
+
3
+ const fs = require('node:fs/promises');
4
+ const path = require('node:path');
5
+ const { getAgentDefinition, resolveInstructionPath, buildAgentPrompt } = require('../agents');
6
+ const { normalizeInteractionLanguage } = require('../locales');
7
+ const { validateProjectContextFile, getInteractionLanguage } = require('../context');
8
+ const { exists, ensureDir } = require('../utils');
9
+ const { syncWorkflowRuntime } = require('../execution-gateway');
10
+ const { writeHandoff, buildWorkflowHandoff, buildWorkflowHandoffProtocol } = require('../session-handoff');
11
+ const { runTechnicalGate, formatGateError } = require('../workflow-gates');
12
+ const { buildTestBriefing } = require('../test-briefing');
13
+ const { validateHandoffContract, formatContractError, getBlockingRevisions, parseFrontmatterValue } = require('../handoff-contract');
14
+ const { buildPathGuardBlock } = require('../path-guard');
15
+ const { logError, buildHealingPrompt } = require('../self-healing');
16
+ const { validateHandoffProtocol } = require('../handoff-validator');
17
+ const { readAutonomyProtocol, resolveEffectiveMode } = require('../autonomy-policy');
18
+ const { readAgentManifest, buildAgentCapabilitySummary } = require('../agent-manifests');
19
+ const { runMemoryReflectPrepare } = require('./memory-reflect-prepare');
20
+ const { inspectStagedChanges } = require('../lib/git-commit-guard');
21
+ const { emitSecurityRuntimeEvent } = require('../lib/security/runtime-events');
22
+ const { runSecurityAudit } = require('./security-audit');
23
+ const dossierBootstrap = require('../dossier/dossier-bootstrap');
24
+ const dossierStore = require('../dossier/store');
25
+ const { emitDossierEvent } = require('../lib/dossier-telemetry');
26
+
27
+ const STATE_RELATIVE_PATH = '.aioson/context/workflow.state.json';
28
+ const CONFIG_RELATIVE_PATH = '.aioson/context/workflow.config.json';
29
+ const EVENTS_RELATIVE_PATH = '.aioson/context/workflow.events.jsonl';
30
+
31
+ const DEFAULT_FEATURE_WORKFLOW_BY_CLASSIFICATION = {
32
+ MICRO: ['product', 'dev', 'qa'],
33
+ SMALL: ['product', 'analyst', 'dev', 'qa'],
34
+ MEDIUM: ['product', 'analyst', 'dev', 'pentester', 'qa']
35
+ };
36
+
37
+ function normalizeAgentName(input) {
38
+ return String(input || '')
39
+ .trim()
40
+ .toLowerCase()
41
+ .replace(/^@/, '');
42
+ }
43
+
44
+ function normalizeClassification(value, fallback = 'MICRO') {
45
+ const text = String(value || '').trim().toUpperCase();
46
+ if (text === 'MICRO' || text === 'SMALL' || text === 'MEDIUM') return text;
47
+ return fallback;
48
+ }
49
+
50
+ function buildDefaultWorkflowConfig() {
51
+ return {
52
+ version: 1,
53
+ project: {
54
+ MICRO: ['setup', 'dev'],
55
+ SMALL: ['setup', 'product', 'analyst', 'architect', 'dev', 'qa'],
56
+ MEDIUM: ['setup', 'product', 'analyst', 'architect', 'ux-ui', 'pm', 'orchestrator', 'dev', 'qa']
57
+ },
58
+ feature: DEFAULT_FEATURE_WORKFLOW_BY_CLASSIFICATION,
59
+ rules: {
60
+ required: ['dev'],
61
+ allowDetours: true
62
+ }
63
+ };
64
+ }
65
+
66
+ function parseFeaturesMarkdown(markdown) {
67
+ return String(markdown || '')
68
+ .split(/\r?\n/)
69
+ .slice(3)
70
+ .map((line) => line.trim())
71
+ .filter(Boolean)
72
+ .filter((line) => line.startsWith('|'))
73
+ .map((line) => line.split('|').map((part) => part.trim()))
74
+ .filter((parts) => parts.length >= 5)
75
+ .map((parts) => ({
76
+ slug: parts[1],
77
+ status: parts[2],
78
+ started: parts[3],
79
+ completed: parts[4]
80
+ }))
81
+ .filter((row) => row.slug && row.slug !== 'slug')
82
+ .filter((row) => !/^-+$/ .test(row.slug));
83
+ }
84
+
85
+ async function readJsonIfExists(filePath) {
86
+ if (!(await exists(filePath))) return null;
87
+ const content = await fs.readFile(filePath, 'utf8');
88
+ return JSON.parse(content);
89
+ }
90
+
91
+ async function writeJson(filePath, payload) {
92
+ await ensureDir(path.dirname(filePath));
93
+ await fs.writeFile(filePath, `${JSON.stringify(payload, null, 2)}\n`, 'utf8');
94
+ }
95
+
96
+ async function appendJsonLine(filePath, payload) {
97
+ await ensureDir(path.dirname(filePath));
98
+ await fs.appendFile(filePath, `${JSON.stringify(payload)}\n`, 'utf8');
99
+ }
100
+
101
+ function buildWorkflowEventMessage({ created, state, activation, completedStage, options }) {
102
+ const requestedAgent = options.requestedAgent ? normalizeAgentName(options.requestedAgent) : null;
103
+ if (requestedAgent && activation.agent && requestedAgent !== activation.agent) {
104
+ return `Workflow enforced @${activation.agent} after direct request for @${requestedAgent}.`;
105
+ }
106
+ if (completedStage && activation.agent) {
107
+ return `Completed @${completedStage}. Next stage ready: @${activation.agent}.`;
108
+ }
109
+ if (completedStage && !activation.agent) {
110
+ return `Completed @${completedStage}. Workflow has no pending stage.`;
111
+ }
112
+ if (state.detour && state.detour.active) {
113
+ return `Detour started with @${state.detour.agent}. Return to ${
114
+ state.detour.returnTo ? `@${state.detour.returnTo}` : 'the main flow'
115
+ }.`;
116
+ }
117
+ if (options.skip && activation.agent) {
118
+ return `Workflow advanced to @${activation.agent} after skip.`;
119
+ }
120
+ if (activation.agent) {
121
+ return created
122
+ ? `Workflow initialized at @${activation.agent}.`
123
+ : `Stage @${activation.agent} is active.`;
124
+ }
125
+ return 'Workflow has no pending stage.';
126
+ }
127
+
128
+ function buildWorkflowEventType({ completedStage, state, activation, options }) {
129
+ const requestedAgent = options.requestedAgent ? normalizeAgentName(options.requestedAgent) : null;
130
+ if (requestedAgent && activation.agent && requestedAgent !== activation.agent) return 'routed';
131
+ if (completedStage) return 'completed';
132
+ if (state.detour && state.detour.active) return 'workflow';
133
+ if (options.skip) return 'workflow';
134
+ if (activation.agent) return 'start';
135
+ return 'workflow';
136
+ }
137
+
138
+ async function appendWorkflowEvent(targetDir, payload) {
139
+ const eventsPath = path.join(targetDir, EVENTS_RELATIVE_PATH);
140
+ await appendJsonLine(eventsPath, payload);
141
+ return eventsPath;
142
+ }
143
+
144
+ async function readWorkflowConfig(targetDir) {
145
+ const configPath = path.join(targetDir, CONFIG_RELATIVE_PATH);
146
+ const userConfig = await readJsonIfExists(configPath);
147
+ const base = buildDefaultWorkflowConfig();
148
+ if (!userConfig || typeof userConfig !== 'object') {
149
+ return { configPath, config: base, exists: false };
150
+ }
151
+
152
+ const merged = {
153
+ ...base,
154
+ ...userConfig,
155
+ project: {
156
+ ...base.project,
157
+ ...(userConfig.project || {})
158
+ },
159
+ feature: {
160
+ ...base.feature,
161
+ ...(userConfig.feature || {})
162
+ },
163
+ rules: {
164
+ ...base.rules,
165
+ ...(userConfig.rules || {})
166
+ }
167
+ };
168
+
169
+ return { configPath, config: merged, exists: true };
170
+ }
171
+
172
+ async function resolveLocaleForTarget(targetDir, options) {
173
+ const fromOption = options.language || options.lang;
174
+ if (fromOption) return normalizeInteractionLanguage(fromOption);
175
+
176
+ const context = await validateProjectContextFile(targetDir);
177
+ if (context.parsed && context.data) {
178
+ return getInteractionLanguage(context.data, 'en');
179
+ }
180
+
181
+ return 'en';
182
+ }
183
+
184
+ async function resolveExistingInstructionPath(targetDir, agent, locale) {
185
+ const candidate = resolveInstructionPath(agent, locale);
186
+ const candidateAbs = path.join(targetDir, candidate);
187
+ if (await exists(candidateAbs)) return candidate;
188
+ return agent.path;
189
+ }
190
+
191
+ async function detectWorkflowMode(targetDir) {
192
+ const prdPath = path.join(targetDir, '.aioson/context/prd.md');
193
+ const featuresPath = path.join(targetDir, '.aioson/context/features.md');
194
+ const hasProjectPrd = await exists(prdPath);
195
+ const featuresMarkdown = await fs.readFile(featuresPath, 'utf8').catch(() => '');
196
+ const features = parseFeaturesMarkdown(featuresMarkdown);
197
+ const activeFeature = features.find((feature) => feature.status === 'in_progress') || null;
198
+
199
+ if (activeFeature) {
200
+ return {
201
+ mode: 'feature',
202
+ featureSlug: activeFeature.slug,
203
+ features
204
+ };
205
+ }
206
+
207
+ return {
208
+ mode: hasProjectPrd ? 'project' : 'project',
209
+ featureSlug: null,
210
+ features
211
+ };
212
+ }
213
+
214
+ function getSequenceForMode(config, mode, classification) {
215
+ const group = mode === 'feature' ? config.feature : config.project;
216
+ const sequence = group[normalizeClassification(classification, 'MICRO')];
217
+ return Array.isArray(sequence) && sequence.length > 0 ? [...sequence] : [];
218
+ }
219
+
220
+ async function validateStageArtifacts(targetDir, state, stage) {
221
+ const base = path.join(targetDir, '.aioson/context');
222
+ const slug = state.featureSlug;
223
+
224
+ if (stage === 'setup') {
225
+ const context = await validateProjectContextFile(targetDir);
226
+ return context.valid;
227
+ }
228
+
229
+ if (stage === 'product') {
230
+ if (state.mode === 'feature' && slug) {
231
+ const prdFeature = path.join(base, `prd-${slug}.md`);
232
+ const prdFix = path.join(base, `prd-${slug}-fix.md`);
233
+ return (await exists(prdFeature)) || (await exists(prdFix));
234
+ }
235
+ return await exists(path.join(base, 'prd.md'));
236
+ }
237
+
238
+ if (stage === 'analyst') {
239
+ if (state.mode === 'feature' && slug) {
240
+ const requirements = path.join(base, `requirements-${slug}.md`);
241
+ const spec = path.join(base, `spec-${slug}.md`);
242
+ return (await exists(requirements)) && (await exists(spec));
243
+ }
244
+ return await exists(path.join(base, 'discovery.md'));
245
+ }
246
+
247
+ if (stage === 'architect') {
248
+ return await exists(path.join(base, 'architecture.md'));
249
+ }
250
+
251
+ if (stage === 'ux-ui') {
252
+ return await exists(path.join(base, 'ui-spec.md'));
253
+ }
254
+
255
+ if (stage === 'orchestrator') {
256
+ return await exists(path.join(base, 'parallel'));
257
+ }
258
+
259
+ return true;
260
+ }
261
+
262
+ function isRequiredAgent(config, agentName) {
263
+ return Array.isArray(config.rules?.required)
264
+ ? config.rules.required.map(normalizeAgentName).includes(agentName)
265
+ : false;
266
+ }
267
+
268
+ function buildStatePayload(input) {
269
+ return {
270
+ version: 1,
271
+ mode: input.mode,
272
+ classification: input.classification,
273
+ sequence: input.sequence,
274
+ current: input.current || null,
275
+ next: input.next || null,
276
+ completed: Array.isArray(input.completed) ? input.completed : [],
277
+ skipped: Array.isArray(input.skipped) ? input.skipped : [],
278
+ featureSlug: input.featureSlug || null,
279
+ detour: input.detour || null,
280
+ updatedAt: new Date().toISOString()
281
+ };
282
+ }
283
+
284
+ function findNextFromSequence(sequence, completed, skipped) {
285
+ const done = new Set([...(completed || []), ...(skipped || [])].map(normalizeAgentName));
286
+ return sequence.find((stage) => !done.has(normalizeAgentName(stage))) || null;
287
+ }
288
+
289
+ function reconcileWorkflowState(state) {
290
+ if (!state || typeof state !== 'object' || !Array.isArray(state.sequence)) {
291
+ return { state, changed: false };
292
+ }
293
+
294
+ const sequence = state.sequence.map(normalizeAgentName);
295
+ const completed = Array.from(new Set((state.completed || []).map(normalizeAgentName).filter(Boolean)));
296
+ const skippedSet = new Set((state.skipped || []).map(normalizeAgentName).filter(Boolean));
297
+ const detour = state.detour && typeof state.detour === 'object'
298
+ ? {
299
+ ...state.detour,
300
+ agent: normalizeAgentName(state.detour.agent),
301
+ returnTo: normalizeAgentName(state.detour.returnTo)
302
+ }
303
+ : null;
304
+ let changed = false;
305
+
306
+ // If a later stage is already completed, any unresolved earlier stage was
307
+ // effectively bypassed outside the workflow and must not remain "active".
308
+ const furthestCompletedIndex = sequence.reduce((max, stage, index) => (
309
+ completed.includes(stage) ? Math.max(max, index) : max
310
+ ), -1);
311
+
312
+ if (furthestCompletedIndex >= 0) {
313
+ for (let index = 0; index < furthestCompletedIndex; index += 1) {
314
+ const stage = sequence[index];
315
+ if (!completed.includes(stage) && !skippedSet.has(stage)) {
316
+ skippedSet.add(stage);
317
+ changed = true;
318
+ }
319
+ }
320
+ }
321
+
322
+ const skipped = sequence.filter((stage) => skippedSet.has(stage));
323
+ const resolved = new Set([...completed, ...skipped]);
324
+ let current = state.current ? normalizeAgentName(state.current) : null;
325
+ let next = state.next ? normalizeAgentName(state.next) : null;
326
+ const currentIsActiveDetour = Boolean(
327
+ detour &&
328
+ detour.active &&
329
+ current &&
330
+ current === detour.agent
331
+ );
332
+
333
+ if (current && ((!sequence.includes(current) && !currentIsActiveDetour) || resolved.has(current))) {
334
+ current = null;
335
+ changed = true;
336
+ }
337
+
338
+ if (!detour || !detour.active) {
339
+ if (current) {
340
+ const currentIndex = sequence.indexOf(current);
341
+ const expectedQueuedNext = sequence.find(
342
+ (stage, index) => index > currentIndex && !resolved.has(stage)
343
+ ) || null;
344
+ if (next && next !== current && next !== expectedQueuedNext) {
345
+ next = expectedQueuedNext || current;
346
+ changed = true;
347
+ } else if (!next) {
348
+ next = expectedQueuedNext || current;
349
+ changed = true;
350
+ }
351
+ } else {
352
+ const inferredNext = findNextFromSequence(sequence, completed, skipped);
353
+ if (next !== inferredNext) {
354
+ next = inferredNext;
355
+ changed = true;
356
+ }
357
+ }
358
+ }
359
+
360
+ if (!changed) {
361
+ return { state, changed: false };
362
+ }
363
+
364
+ return {
365
+ changed: true,
366
+ state: buildStatePayload({
367
+ ...state,
368
+ sequence,
369
+ completed,
370
+ skipped,
371
+ current,
372
+ next,
373
+ detour
374
+ })
375
+ };
376
+ }
377
+
378
+ function isInferableStage(stage) {
379
+ return ['setup', 'product', 'analyst', 'architect', 'ux-ui', 'orchestrator'].includes(
380
+ normalizeAgentName(stage)
381
+ );
382
+ }
383
+
384
+ function isSecurityGateBlocked(contractCheck, state, stageName) {
385
+ if (normalizeAgentName(stageName) !== 'qa' || state.mode !== 'feature' || !state.featureSlug) {
386
+ return false;
387
+ }
388
+ return contractCheck.missing.some((item) =>
389
+ item.includes('security:') ||
390
+ item.includes(`security-findings-${state.featureSlug}.json`)
391
+ );
392
+ }
393
+
394
+ function buildQaSecurityAuditBriefing(result, targetDir) {
395
+ if (!result) return '';
396
+
397
+ if (result.ok === false && result.reason) {
398
+ return [
399
+ '## Secure by Default audit',
400
+ `- Auto-run failed before QA review: ${result.reason}.`,
401
+ '- Gate D will remain blocked until a valid `security-findings-{slug}.json` artifact exists.',
402
+ '- If CLI is unavailable in your client, use the fallback checklist and record the limitation explicitly in the QA report and `project-pulse.md`.'
403
+ ].join('\n');
404
+ }
405
+
406
+ return [
407
+ '## Secure by Default audit',
408
+ `- Auto-ran \`security:audit\` for feature \`${result.slug}\` at QA activation.`,
409
+ `- Exit code: ${result.exitCode}. Findings: ${result.findingsCount}.`,
410
+ `- Summary: critical=${result.summary.critical}, high=${result.summary.high}, medium=${result.summary.medium}, low=${result.summary.low}, inconclusive=${result.summary.inconclusive}.`,
411
+ `- Artifact: \`${path.relative(targetDir, result.artifactPath)}\`.`,
412
+ '- If the audit or manual heuristics indicate auth, money, or ownership risk, invoke `@pentester` with `--mode=app_target --feature=<slug> --scope=<target>` before final Gate D sign-off.',
413
+ '- If CLI is unavailable in your client, use the fallback checklist and record the limitation explicitly in the QA report and `project-pulse.md`.'
414
+ ].join('\n');
415
+ }
416
+
417
+ async function inferCompletedStages(targetDir, draftState) {
418
+ const completed = [];
419
+ for (const stage of draftState.sequence) {
420
+ if (!isInferableStage(stage)) break;
421
+ const valid = await validateStageArtifacts(targetDir, draftState, stage);
422
+ if (!valid) break;
423
+ completed.push(normalizeAgentName(stage));
424
+ }
425
+ return completed;
426
+ }
427
+
428
+ // SF-project-18: cross-check workflow.state.json#completed against runtime
429
+ // telemetry. Stages claimed as completed without a corresponding agent_done
430
+ // event in .aioson/runtime/aios.sqlite are surfaced as a warning. Detection
431
+ // is best-effort — if the runtime DB is unavailable, the check is silently
432
+ // skipped (the framework still works in environments without telemetry).
433
+ async function detectUnsubstantiatedCompletions(targetDir, completedStages, logger = null) {
434
+ if (!Array.isArray(completedStages) || completedStages.length === 0) return [];
435
+ let runtimeStore;
436
+ try {
437
+ runtimeStore = require('../runtime-store');
438
+ } catch {
439
+ return [];
440
+ }
441
+ if (!runtimeStore.runtimeStoreExists) return [];
442
+ let dbExists;
443
+ try { dbExists = await runtimeStore.runtimeStoreExists(targetDir); } catch { return []; }
444
+ if (!dbExists) return [];
445
+ let db;
446
+ try {
447
+ db = await runtimeStore.openRuntimeDb(targetDir);
448
+ } catch {
449
+ return [];
450
+ }
451
+ if (!db || typeof db.prepare !== 'function') {
452
+ try { if (db && typeof db.close === 'function') db.close(); } catch { /* ignore */ }
453
+ return [];
454
+ }
455
+ const unsubstantiated = [];
456
+ try {
457
+ let stmt;
458
+ try {
459
+ stmt = db.prepare(
460
+ "SELECT 1 FROM agent_events WHERE agent = ? AND event_type IN ('agent_done', 'stage_completed') LIMIT 1"
461
+ );
462
+ } catch {
463
+ // schema differences across versions — abort the cross-check.
464
+ return [];
465
+ }
466
+ for (const stage of completedStages) {
467
+ try {
468
+ const row = stmt.get(stage);
469
+ if (!row) unsubstantiated.push(stage);
470
+ } catch {
471
+ return [];
472
+ }
473
+ }
474
+ } finally {
475
+ try { db.close(); } catch { /* ignore */ }
476
+ }
477
+ if (unsubstantiated.length > 0 && logger && typeof logger.warn === 'function') {
478
+ logger.warn(
479
+ `[workflow:next] state-file integrity warning — completed stages without agent_done telemetry: ${unsubstantiated.join(', ')}. ` +
480
+ `If you did not just edit workflow.state.json by hand, this may indicate tampering.`
481
+ );
482
+ }
483
+ return unsubstantiated;
484
+ }
485
+
486
+ async function loadOrCreateState(targetDir, options = {}) {
487
+ const statePath = path.join(targetDir, STATE_RELATIVE_PATH);
488
+ let existing = await readJsonIfExists(statePath);
489
+
490
+ // Feature-transition guard: if the persisted state belongs to a different
491
+ // feature than the currently active one (per features.md), it is stale —
492
+ // left over from a closed/abandoned feature. Discard it and rebuild from
493
+ // current features.md. Without this, opening a new feature inherits the
494
+ // previous feature's slug, classification, and completion record, blocking
495
+ // the new workflow on gates that don't apply.
496
+ if (existing && existing.mode === 'feature' && existing.featureSlug) {
497
+ const modeInfo = await detectWorkflowMode(targetDir);
498
+ if (modeInfo.featureSlug && existing.featureSlug !== modeInfo.featureSlug) {
499
+ existing = null;
500
+ }
501
+ }
502
+
503
+ if (existing && typeof existing === 'object' && Array.isArray(existing.sequence)) {
504
+ // SF-project-18: warn-on-mismatch only, never refuse — preserves
505
+ // backwards-compat with environments that lack runtime telemetry.
506
+ if (Array.isArray(existing.completed) && existing.completed.length > 0 && options.logger) {
507
+ await detectUnsubstantiatedCompletions(targetDir, existing.completed, options.logger);
508
+ }
509
+ const reconciled = reconcileWorkflowState(existing);
510
+ if (reconciled.changed) {
511
+ await writeJson(statePath, reconciled.state);
512
+ }
513
+ return { statePath, state: reconciled.state, created: false };
514
+ }
515
+
516
+ const context = await validateProjectContextFile(targetDir);
517
+ const modeInfo = await detectWorkflowMode(targetDir);
518
+
519
+ // Feature classification (from prd-{slug}.md frontmatter) takes precedence
520
+ // over the project classification. A MICRO feature inside a MEDIUM project
521
+ // must be sequenced and gated as MICRO.
522
+ let featurePrdClassification = null;
523
+ if (modeInfo.mode === 'feature' && modeInfo.featureSlug) {
524
+ const prdPath = path.join(targetDir, '.aioson/context', `prd-${modeInfo.featureSlug}.md`);
525
+ const prdContent = await fs.readFile(prdPath, 'utf8').catch(() => '');
526
+ if (prdContent) {
527
+ const parsed = parseFrontmatterValue(prdContent, 'classification');
528
+ if (parsed) featurePrdClassification = parsed;
529
+ }
530
+ }
531
+
532
+ const classification = normalizeClassification(
533
+ options.classification
534
+ || featurePrdClassification
535
+ || (context.data && context.data.classification)
536
+ || 'MICRO',
537
+ 'MICRO'
538
+ );
539
+ const { config } = await readWorkflowConfig(targetDir);
540
+ const sequence = getSequenceForMode(config, modeInfo.mode, classification);
541
+ const draftState = buildStatePayload({
542
+ mode: modeInfo.mode,
543
+ classification,
544
+ sequence,
545
+ current: null,
546
+ next: null,
547
+ completed: [],
548
+ skipped: [],
549
+ featureSlug: modeInfo.featureSlug,
550
+ detour: null
551
+ });
552
+ const completed = await inferCompletedStages(targetDir, draftState);
553
+ const next = findNextFromSequence(sequence, completed, []);
554
+ const state = buildStatePayload({
555
+ mode: modeInfo.mode,
556
+ classification,
557
+ sequence,
558
+ current: null,
559
+ next,
560
+ completed,
561
+ skipped: [],
562
+ featureSlug: modeInfo.featureSlug,
563
+ detour: null
564
+ });
565
+
566
+ await writeJson(statePath, state);
567
+ return { statePath, state, created: true };
568
+ }
569
+
570
+ async function persistState(targetDir, nextState) {
571
+ const statePath = path.join(targetDir, STATE_RELATIVE_PATH);
572
+ await writeJson(statePath, nextState);
573
+ return statePath;
574
+ }
575
+
576
+ function ensureAgentInSequence(state, agentName) {
577
+ if (state.sequence.includes(agentName)) return;
578
+ throw new Error(`Agent ${agentName} is not part of the active workflow sequence.`);
579
+ }
580
+
581
+ function ensureSkippableTarget(config, state, targetAgent) {
582
+ const normalizedTarget = normalizeAgentName(targetAgent);
583
+ ensureAgentInSequence(state, normalizedTarget);
584
+
585
+ const currentIndex = state.next ? state.sequence.indexOf(state.next) : -1;
586
+ const targetIndex = state.sequence.indexOf(normalizedTarget);
587
+ const devIndex = state.sequence.indexOf('dev');
588
+
589
+ if (currentIndex === -1) {
590
+ throw new Error('No next stage is available to skip from.');
591
+ }
592
+ if (targetIndex === -1 || targetIndex < currentIndex) {
593
+ throw new Error(`Cannot skip backwards to ${targetAgent}.`);
594
+ }
595
+ if (normalizedTarget === 'dev') return;
596
+ if (devIndex !== -1 && targetIndex > devIndex) {
597
+ throw new Error('Cannot skip past @dev because @dev is mandatory.');
598
+ }
599
+ if (isRequiredAgent(config, normalizedTarget) && normalizedTarget !== 'dev') {
600
+ return;
601
+ }
602
+ }
603
+
604
+ async function finalizeCurrentStage(targetDir, config, state, stageName) {
605
+ const normalizedStage = normalizeAgentName(stageName || state.current || state.next);
606
+ if (!normalizedStage) {
607
+ throw new Error('No stage is active to complete.');
608
+ }
609
+
610
+ // ── Harness Done Gate ───────────────────────────────────────────────────
611
+ if (state.mode === 'feature' && state.featureSlug) {
612
+ const contractPath = path.join(targetDir, '.aioson', 'plans', state.featureSlug, 'harness-contract.json');
613
+ const progressPath = path.join(targetDir, '.aioson', 'plans', state.featureSlug, 'progress.json');
614
+
615
+ // Se contrato existe, verificamos o progresso
616
+ const fs = require('node:fs');
617
+ if (fs.existsSync(contractPath) && fs.existsSync(progressPath)) {
618
+ try {
619
+ const progress = JSON.parse(fs.readFileSync(progressPath, 'utf8'));
620
+ // Bloqueia se não estiver pronto para o gate E o estágio for crítico (dev/qa)
621
+ if (!progress.ready_for_done_gate && (normalizedStage === 'dev' || normalizedStage === 'qa')) {
622
+ throw new Error(`[Harness Block] A feature "${state.featureSlug}" não passou na validação contratual. Execute 'aioson harness:validate' e resolva os problemas antes de concluir o estágio @${normalizedStage}.`);
623
+ }
624
+ } catch (err) {
625
+ if (err.message.includes('[Harness Block]')) throw err;
626
+ // Se erro de parse, ignoramos para não quebrar o workflow por corrupção
627
+ }
628
+ }
629
+ }
630
+
631
+ if (state.detour && state.detour.active && normalizeAgentName(state.detour.agent) === normalizedStage) {
632
+ const validDetour = await validateStageArtifacts(targetDir, state, normalizedStage);
633
+ if (!validDetour) {
634
+ throw new Error(`Cannot complete detour ${normalizedStage}; expected artifacts are missing.`);
635
+ }
636
+ const nextState = buildStatePayload({
637
+ ...state,
638
+ current: null,
639
+ next: state.detour.returnTo,
640
+ detour: null
641
+ });
642
+ return { state: nextState, completedStage: normalizedStage };
643
+ }
644
+
645
+ ensureAgentInSequence(state, normalizedStage);
646
+ const valid = await validateStageArtifacts(targetDir, state, normalizedStage);
647
+ if (!valid) {
648
+ throw new Error(`Cannot complete ${normalizedStage}; expected artifacts are missing.`);
649
+ }
650
+
651
+ // ── Handoff Contract Gate ───────────────────────────────────────────────
652
+ const contractCheck = await validateHandoffContract(targetDir, state, normalizedStage);
653
+ if (!contractCheck.ok) {
654
+ if (isSecurityGateBlocked(contractCheck, state, normalizedStage)) {
655
+ await emitSecurityRuntimeEvent({
656
+ targetDir,
657
+ eventType: 'security_gate_blocked',
658
+ message: `Gate D blocked for ${state.featureSlug} at @qa`,
659
+ status: 'failed',
660
+ agentName: 'qa',
661
+ source: 'workflow',
662
+ workflowState: state,
663
+ workflowStage: 'qa',
664
+ payload: {
665
+ feature_slug: state.featureSlug,
666
+ classification: state.classification,
667
+ blockers: contractCheck.missing
668
+ }
669
+ });
670
+ }
671
+ const errMsg = formatContractError(contractCheck);
672
+ await logError(targetDir, normalizedStage, errMsg, 'contract');
673
+ throw new Error(errMsg);
674
+ }
675
+
676
+ // ── Revision Gate (Phase 2) ─────────────────────────────────────────────
677
+ const blockingRevisions = await getBlockingRevisions(targetDir, state.featureSlug);
678
+ if (blockingRevisions.length > 0) {
679
+ const ids = blockingRevisions.map((r) => r.id).join(', ');
680
+ const errMsg = [
681
+ `[Revision Gate BLOCKED]`,
682
+ `Feature: ${state.featureSlug}`,
683
+ ``,
684
+ `Pending blocking revision(s): ${ids}`,
685
+ ``,
686
+ `Resolve each revision before completing this stage:`,
687
+ ...blockingRevisions.map((r) => ` aioson revision:resolve . --slug=${state.featureSlug} --rev-id=${r.id} --approve|--reject`)
688
+ ].join('\n');
689
+ await logError(targetDir, normalizedStage, errMsg, 'revision');
690
+ throw new Error(errMsg);
691
+ }
692
+
693
+ // ── Technical Compilation/Test Gate ─────────────────────────────────────
694
+ const techGate = await runTechnicalGate(targetDir, normalizedStage);
695
+ if (!techGate.ok) {
696
+ const errMsg = formatGateError(techGate);
697
+ await logError(targetDir, normalizedStage, errMsg, 'technical');
698
+ throw new Error(errMsg);
699
+ }
700
+
701
+ const completed = Array.from(new Set([...(state.completed || []), normalizedStage]));
702
+ const next = findNextFromSequence(state.sequence, completed, state.skipped || []);
703
+ const nextState = buildStatePayload({
704
+ ...state,
705
+ completed,
706
+ current: null,
707
+ next,
708
+ detour: null
709
+ });
710
+
711
+ return { state: nextState, completedStage: normalizedStage };
712
+ }
713
+
714
+ /**
715
+ * Detects whether the current feature has a harness contract awaiting
716
+ * validation. Used by runWorkflowNext to route to @validator (as a detour)
717
+ * before any other agent. Implements AC-HD-14 of harness-driven-aioson.
718
+ *
719
+ * Returns true only when ALL of the following hold:
720
+ * - state.mode === 'feature' AND state.featureSlug is set
721
+ * - .aioson/plans/<slug>/harness-contract.json exists
722
+ * - .aioson/plans/<slug>/progress.json exists, parses, and reports
723
+ * status === 'waiting_validation'
724
+ *
725
+ * Without these conditions the function returns false and the workflow
726
+ * routing proceeds exactly as before (zero behavior change for MICRO/SMALL
727
+ * or any MEDIUM feature without a contract).
728
+ */
729
+ function shouldRouteToValidator(targetDir, state) {
730
+ if (!state || state.mode !== 'feature' || !state.featureSlug) return false;
731
+ const fsLocal = require('node:fs');
732
+ const planDir = path.join(targetDir, '.aioson', 'plans', state.featureSlug);
733
+ const contractPath = path.join(planDir, 'harness-contract.json');
734
+ const progressPath = path.join(planDir, 'progress.json');
735
+ if (!fsLocal.existsSync(contractPath) || !fsLocal.existsSync(progressPath)) return false;
736
+ try {
737
+ const progress = JSON.parse(fsLocal.readFileSync(progressPath, 'utf8'));
738
+ return progress && progress.status === 'waiting_validation';
739
+ } catch {
740
+ // Corrupted progress: do NOT override routing — fail safe to default flow.
741
+ return false;
742
+ }
743
+ }
744
+
745
+ function applySkip(config, state, target) {
746
+ const normalizedTarget = normalizeAgentName(target);
747
+ ensureSkippableTarget(config, state, normalizedTarget);
748
+ const currentIndex = state.sequence.indexOf(state.next);
749
+ const targetIndex = state.sequence.indexOf(normalizedTarget);
750
+ const toSkip = state.sequence.slice(currentIndex, targetIndex);
751
+ if (toSkip.some((agent) => normalizeAgentName(agent) === 'dev')) {
752
+ throw new Error('Cannot skip @dev because it is mandatory.');
753
+ }
754
+
755
+ const skipped = Array.from(new Set([...(state.skipped || []), ...toSkip]));
756
+ return buildStatePayload({
757
+ ...state,
758
+ skipped,
759
+ current: null,
760
+ next: normalizedTarget
761
+ });
762
+ }
763
+
764
+ async function ensureFeatureDossier(targetDir, state) {
765
+ if (state.mode !== 'feature' || !state.featureSlug) return;
766
+ const classification = String(state.classification || '').toUpperCase();
767
+ if (classification !== 'SMALL' && classification !== 'MEDIUM') return;
768
+
769
+ const ctxDir = path.join(targetDir, '.aioson', 'context');
770
+ const dossierFile = path.join(ctxDir, 'features', state.featureSlug, 'dossier.md');
771
+ try {
772
+ await fs.access(dossierFile);
773
+ return;
774
+ } catch {
775
+ // proceed to create
776
+ }
777
+
778
+ let mode = null;
779
+ try {
780
+ await dossierBootstrap.initFromExisting({
781
+ slug: state.featureSlug,
782
+ contextDir: ctxDir,
783
+ classification,
784
+ targetDir
785
+ });
786
+ mode = 'from-existing';
787
+ } catch (err) {
788
+ if (err && err.code === 'EBOOTSTRAPEMPTY') {
789
+ try {
790
+ await dossierStore.init({
791
+ slug: state.featureSlug,
792
+ contextDir: ctxDir,
793
+ classification,
794
+ whyText: '(auto-init by workflow:next; no source artifacts yet)',
795
+ whatText: '(auto-init by workflow:next; no source artifacts yet)'
796
+ });
797
+ mode = 'minimal-fallback';
798
+ } catch {
799
+ return;
800
+ }
801
+ } else if (err && err.code === 'EDOSSIEREXISTS') {
802
+ return;
803
+ } else {
804
+ return;
805
+ }
806
+ }
807
+
808
+ if (mode) {
809
+ await emitDossierEvent(targetDir, {
810
+ agent: 'workflow-next',
811
+ type: 'dossier_auto_initialized',
812
+ summary: `${state.featureSlug} ${mode}`,
813
+ meta: {
814
+ feature_slug: state.featureSlug,
815
+ classification,
816
+ trigger_source: 'workflow_next_pre_stage',
817
+ mode
818
+ }
819
+ });
820
+ }
821
+ }
822
+
823
+ async function activateStage(targetDir, state, locale, tool, explicitAgent = null, requestedMode = null) {
824
+ const stageName = normalizeAgentName(explicitAgent || state.current || state.next);
825
+ if (!stageName) {
826
+ return {
827
+ state,
828
+ agent: null,
829
+ instructionPath: null,
830
+ prompt: null
831
+ };
832
+ }
833
+
834
+ // Pre-stage hook: ensure feature dossier exists for SMALL/MEDIUM features.
835
+ // Silent (no logging); idempotent (no-op if dossier already exists).
836
+ // Defense-in-depth alongside the @product prompt instruction.
837
+ await ensureFeatureDossier(targetDir, state);
838
+
839
+ // ── Committer Safety Gate ───────────────────────────────────────────────
840
+ if (stageName === 'committer') {
841
+ const guard = await inspectStagedChanges(targetDir, { allowWarnings: false });
842
+ if (guard.summary.stagedCount === 0) {
843
+ throw new Error(
844
+ `[Committer Gate BLOCKED] Nenhum arquivo no stage para commit. ` +
845
+ `Execute primeiro: aioson commit:prepare . --agent-safe --staged-only --mode=headless`
846
+ );
847
+ }
848
+ if (!guard.ok) {
849
+ throw new Error(
850
+ `[Committer Gate BLOCKED] Arquivos proibidos detectados no stage ` +
851
+ `(node_modules, build artifacts, secrets, etc.). ` +
852
+ `Execute 'aioson git:guard .' para ver detalhes, corrija e rode 'aioson commit:prepare . --agent-safe --staged-only --mode=headless' antes de ativar @committer.`
853
+ );
854
+ }
855
+ }
856
+
857
+ // ── Test Briefing Injection for qa/tester ───────────────────────────────
858
+ let testBriefing = '';
859
+ let securityAuditBriefing = '';
860
+ if (stageName === 'qa' || stageName === 'tester') {
861
+ try {
862
+ testBriefing = await buildTestBriefing(targetDir);
863
+ } catch {
864
+ // Non-fatal: if briefing generation fails, proceed without it
865
+ testBriefing = '';
866
+ }
867
+ }
868
+
869
+ if (
870
+ stageName === 'qa' &&
871
+ state.mode === 'feature' &&
872
+ state.classification === 'MEDIUM' &&
873
+ state.featureSlug
874
+ ) {
875
+ try {
876
+ const auditResult = await runSecurityAudit({
877
+ args: [targetDir],
878
+ options: {
879
+ slug: state.featureSlug,
880
+ json: true,
881
+ runtimeAgentName: 'qa',
882
+ runtimeSource: 'workflow',
883
+ runtimeState: state,
884
+ runtimeWorkflowStage: 'qa'
885
+ },
886
+ logger: { log() {}, error() {}, warn() {} }
887
+ });
888
+ securityAuditBriefing = buildQaSecurityAuditBriefing(auditResult, targetDir);
889
+ } catch {
890
+ securityAuditBriefing = buildQaSecurityAuditBriefing({
891
+ ok: false,
892
+ reason: 'audit_runtime_failure'
893
+ }, targetDir);
894
+ }
895
+ }
896
+
897
+ // ── Path Guard Injection for implementation agents ────────────────────────
898
+ let pathGuardBlock = '';
899
+ if (['dev', 'architect', 'ux-ui', 'pentester', 'qa', 'tester', 'committer'].includes(stageName)) {
900
+ try {
901
+ pathGuardBlock = await buildPathGuardBlock(targetDir);
902
+ } catch {
903
+ pathGuardBlock = '';
904
+ }
905
+ }
906
+
907
+ const agent = getAgentDefinition(stageName);
908
+ if (!agent) {
909
+ throw new Error(`Unknown agent: ${stageName}`);
910
+ }
911
+
912
+ const autonomyProtocol = await readAutonomyProtocol(targetDir);
913
+ const agentManifest = await readAgentManifest(targetDir, agent.id);
914
+ const effectiveMode = resolveEffectiveMode({
915
+ protocol: autonomyProtocol,
916
+ tool,
917
+ agentId: agent.id,
918
+ manifest: agentManifest,
919
+ requestedMode
920
+ });
921
+
922
+ const instructionPath = await resolveExistingInstructionPath(targetDir, agent, locale);
923
+ let prompt = buildAgentPrompt(agent, tool, {
924
+ instructionPath,
925
+ targetDir,
926
+ interactionLanguage: locale,
927
+ autonomyMode: effectiveMode,
928
+ capabilitySummary: buildAgentCapabilitySummary(agentManifest, tool)
929
+ });
930
+
931
+ if (testBriefing) {
932
+ prompt += '\n\n' + testBriefing;
933
+ }
934
+
935
+ if (securityAuditBriefing) {
936
+ prompt += '\n\n' + securityAuditBriefing;
937
+ }
938
+
939
+ if (pathGuardBlock) {
940
+ prompt += '\n\n' + pathGuardBlock;
941
+ }
942
+
943
+ let nextState = state;
944
+ if (explicitAgent && stageName !== normalizeAgentName(state.next)) {
945
+ nextState = buildStatePayload({
946
+ ...state,
947
+ current: stageName,
948
+ detour: {
949
+ active: true,
950
+ agent: stageName,
951
+ returnTo: state.next
952
+ }
953
+ });
954
+ } else {
955
+ nextState = buildStatePayload({
956
+ ...state,
957
+ current: stageName
958
+ });
959
+ }
960
+
961
+ return {
962
+ state: nextState,
963
+ agent: stageName,
964
+ instructionPath,
965
+ prompt,
966
+ effectiveMode
967
+ };
968
+ }
969
+
970
+ async function runWorkflowNext({ args, options, logger, t }) {
971
+ if (options.status || options.suggest) {
972
+ const { runWorkflowStatus } = require('./workflow-status');
973
+ return runWorkflowStatus({ args, options, logger, t });
974
+ }
975
+
976
+ const logErrorLine = typeof logger.error === 'function'
977
+ ? logger.error.bind(logger)
978
+ : typeof logger.log === 'function'
979
+ ? logger.log.bind(logger)
980
+ : () => {};
981
+
982
+ const targetDir = path.resolve(process.cwd(), args[0] || '.');
983
+ const tool = options.tool || 'codex';
984
+ const locale = await resolveLocaleForTarget(targetDir, options);
985
+ const { config } = await readWorkflowConfig(targetDir);
986
+ const loaded = await loadOrCreateState(targetDir, options);
987
+ let state = loaded.state;
988
+ let completedStage = null;
989
+
990
+ if (options.complete || options['complete-current']) {
991
+ let finalized;
992
+ try {
993
+ finalized = await finalizeCurrentStage(
994
+ targetDir,
995
+ config,
996
+ state,
997
+ options.complete === true ? state.current || state.next : options.complete
998
+ );
999
+ } catch (err) {
1000
+ // ── Auto-heal intercept ───────────────────────────────────────────────
1001
+ const autoHeal = Boolean(options['auto-heal'] || options.autoHeal);
1002
+ const isHealabled = autoHeal && (
1003
+ err.message.includes('[Technical Gate BLOCKED]') ||
1004
+ err.message.includes('[Handoff Contract BLOCKED]')
1005
+ );
1006
+ if (isHealabled) {
1007
+ const failedStage = normalizeAgentName(options.complete === true ? state.current || state.next : options.complete);
1008
+ await logError(targetDir, failedStage, err.message, 'technical');
1009
+ const retryCount = await require('../self-healing').getRetryCount(targetDir, failedStage);
1010
+ if (retryCount < require('../self-healing').MAX_RETRIES) {
1011
+ await require('../self-healing').incrementRetryCount(targetDir, failedStage, err.message.substring(0, 200));
1012
+ // Build healing activation
1013
+ const baseActivation = await activateStage(targetDir, state, locale, tool, failedStage, options.mode || null);
1014
+ const healingPrompt = buildHealingPrompt(
1015
+ baseActivation.prompt || '',
1016
+ failedStage,
1017
+ { error: err.message },
1018
+ retryCount + 1
1019
+ );
1020
+ const healedState = {
1021
+ ...baseActivation.state,
1022
+ current: failedStage,
1023
+ detour: null
1024
+ };
1025
+ await persistState(targetDir, healedState);
1026
+ const eventPayload = {
1027
+ id: Date.now(),
1028
+ kind: 'workflow',
1029
+ createdAt: new Date().toISOString(),
1030
+ eventType: 'heal',
1031
+ message: `Auto-heal @${failedStage} — retry ${retryCount + 1}/3`,
1032
+ mode: state.mode,
1033
+ classification: state.classification,
1034
+ featureSlug: state.featureSlug,
1035
+ current: failedStage,
1036
+ next: state.next,
1037
+ completed: state.completed,
1038
+ skipped: state.skipped,
1039
+ sequence: state.sequence,
1040
+ healing: true,
1041
+ retryCount: retryCount + 1,
1042
+ autonomyMode: baseActivation.effectiveMode || null
1043
+ };
1044
+ await appendWorkflowEvent(targetDir, eventPayload);
1045
+ const runtime = await syncWorkflowRuntime(targetDir, {
1046
+ state: healedState,
1047
+ eventPayload,
1048
+ activationAgent: failedStage,
1049
+ completedStage: null
1050
+ });
1051
+ const healingHandoff = buildWorkflowHandoff(healedState, null, failedStage);
1052
+ healingHandoff.protocol = buildWorkflowHandoffProtocol(healedState, null, failedStage, {
1053
+ autonomyMode: baseActivation.effectiveMode || null,
1054
+ handoffContractOk: true,
1055
+ technicalGateOk: false,
1056
+ artifactUris: []
1057
+ });
1058
+ const healingValidation = await validateHandoffProtocol(targetDir, healingHandoff.protocol);
1059
+ if (!healingValidation.ok) {
1060
+ // SF-project-17: the current validator error set is intentionally
1061
+ // soft (missing manifests / unknown capabilities are common during
1062
+ // bootstrap), so we WARN and continue. Blocking would require a
1063
+ // validator that distinguishes warnings from hard contract
1064
+ // violations — see SF-17 dev_session_note for the deferred fix.
1065
+ logErrorLine('Handoff protocol warning:');
1066
+ for (const err of healingValidation.errors) logErrorLine(` - ${err}`);
1067
+ }
1068
+ await writeHandoff(targetDir, healingHandoff);
1069
+ logger.log(t('workflow_heal.title', { stage: `@${failedStage}`, count: retryCount + 1 }));
1070
+ logger.log(healingPrompt);
1071
+ return {
1072
+ ok: true,
1073
+ targetDir,
1074
+ locale,
1075
+ tool,
1076
+ statePath: STATE_RELATIVE_PATH,
1077
+ configPath: CONFIG_RELATIVE_PATH,
1078
+ created: loaded.created,
1079
+ mode: state.mode,
1080
+ classification: state.classification,
1081
+ current: healedState.current,
1082
+ next: healedState.next,
1083
+ detour: healedState.detour,
1084
+ completed: healedState.completed,
1085
+ skipped: healedState.skipped,
1086
+ completedStage: null,
1087
+ featureSlug: state.featureSlug,
1088
+ runtime,
1089
+ agent: failedStage,
1090
+ instructionPath: baseActivation.instructionPath,
1091
+ prompt: healingPrompt,
1092
+ autoHealed: true,
1093
+ effectiveMode: baseActivation.effectiveMode || null
1094
+ };
1095
+ }
1096
+ }
1097
+ throw err;
1098
+ }
1099
+ state = finalized.state;
1100
+ completedStage = finalized.completedStage;
1101
+ await require('../self-healing').incrementRetryCount(targetDir, completedStage, '');
1102
+ const { getRetryCount } = require('../self-healing');
1103
+ const retries = await getRetryCount(targetDir, completedStage);
1104
+ if (retries > 0) {
1105
+ // Reset retry count on successful completion after healing
1106
+ const retriesPath = path.join(targetDir, '.aioson/context/pipeline-retries', `${completedStage}.json`);
1107
+ try { await fs.unlink(retriesPath); } catch { /* ignore */ }
1108
+ }
1109
+
1110
+ // ── Living Memory: reflect bootstrap if the completed stage produced
1111
+ // a relevant diff (routes/models/contracts/volume). Best-effort —
1112
+ // never fail the workflow on reflection errors.
1113
+ try {
1114
+ await runMemoryReflectPrepare({
1115
+ args: [targetDir],
1116
+ options: { agent: completedStage, json: true },
1117
+ logger: { log: () => {}, error: () => {} }
1118
+ });
1119
+ } catch { /* reflection is advisory; never block the workflow */ }
1120
+ }
1121
+
1122
+ if (options.skip) {
1123
+ state = applySkip(config, state, options.skip);
1124
+ }
1125
+
1126
+ let requestedAgent = options.agent ? normalizeAgentName(options.agent) : null;
1127
+
1128
+ // ── Harness Validator Routing (AC-HD-14) ────────────────────────────────
1129
+ // When the active feature has a harness-contract and progress is
1130
+ // `waiting_validation`, route to @validator as a detour. Explicit user
1131
+ // override (--agent=…) is preserved; auto-routing only fires when no agent
1132
+ // was requested. Without contract or in MICRO/SMALL, this is a no-op.
1133
+ if (!requestedAgent && shouldRouteToValidator(targetDir, state)) {
1134
+ requestedAgent = 'validator';
1135
+ }
1136
+
1137
+ const activation = await activateStage(targetDir, state, locale, tool, requestedAgent, options.mode || null);
1138
+ state = activation.state;
1139
+
1140
+ // ── Living Memory: if a reflect manifest is pending (created above by the
1141
+ // completed agent), prepend a one-line instruction so the next agent
1142
+ // consumes it before any other action.
1143
+ try {
1144
+ const reflectPath = path.join(targetDir, '.aioson/runtime/reflect-prompt.json');
1145
+ if (await exists(reflectPath) && activation.prompt) {
1146
+ activation.prompt =
1147
+ 'ℹ [memory] reflect-prompt.json pending — before any other action, read .aioson/runtime/reflect-prompt.json and run `aioson memory:reflect-commit . --agent=' + (activation.agent || 'dev') + ' --output=<path>` per your Memory Reflection section.\n\n' +
1148
+ activation.prompt;
1149
+ }
1150
+ } catch { /* ignore */ }
1151
+ const statePath = await persistState(targetDir, state);
1152
+ const eventPayload = {
1153
+ id: Date.now(),
1154
+ kind: 'workflow',
1155
+ createdAt: new Date().toISOString(),
1156
+ eventType: buildWorkflowEventType({ completedStage, state, activation, options }),
1157
+ message: buildWorkflowEventMessage({
1158
+ created: loaded.created,
1159
+ state,
1160
+ activation,
1161
+ completedStage,
1162
+ options
1163
+ }),
1164
+ mode: state.mode,
1165
+ classification: state.classification,
1166
+ featureSlug: state.featureSlug,
1167
+ current: state.current,
1168
+ next: state.detour && state.detour.active ? state.detour.returnTo : state.next,
1169
+ completedStage,
1170
+ detour: state.detour,
1171
+ requestedAgent: options.requestedAgent ? normalizeAgentName(options.requestedAgent) : null,
1172
+ completed: state.completed,
1173
+ skipped: state.skipped,
1174
+ sequence: state.sequence,
1175
+ autonomyMode: activation.effectiveMode || null
1176
+ };
1177
+ await appendWorkflowEvent(targetDir, eventPayload);
1178
+ const runtime = await syncWorkflowRuntime(targetDir, {
1179
+ state,
1180
+ eventPayload,
1181
+ activationAgent: activation.agent,
1182
+ completedStage
1183
+ });
1184
+
1185
+ // Generate session handoff when a stage completes or workflow finishes
1186
+ if (completedStage || !activation.agent) {
1187
+ const handoffData = buildWorkflowHandoff(state, completedStage, activation.agent);
1188
+ handoffData.autonomyMode = activation.effectiveMode || null;
1189
+ handoffData.protocol = buildWorkflowHandoffProtocol(state, completedStage, activation.agent, {
1190
+ autonomyMode: activation.effectiveMode || null,
1191
+ handoffContractOk: true,
1192
+ technicalGateOk: true,
1193
+ artifactUris: []
1194
+ });
1195
+ const handoffValidation = await validateHandoffProtocol(targetDir, handoffData.protocol);
1196
+ if (!handoffValidation.ok) {
1197
+ // SF-project-17: the validator currently returns errors for soft
1198
+ // conditions (missing manifest, unknown capability) that occur during
1199
+ // normal bootstrap. Treating them as blockers would break the workflow
1200
+ // for any agent whose manifest has not been committed yet. Until the
1201
+ // validator is refactored to separate warnings from hard contract
1202
+ // violations, this caller emits a warning and continues — see
1203
+ // SF-project-17 dev_session_note.
1204
+ logErrorLine('Handoff protocol warning:');
1205
+ for (const err of handoffValidation.errors) logErrorLine(` - ${err}`);
1206
+ }
1207
+ await writeHandoff(targetDir, handoffData);
1208
+ }
1209
+
1210
+ const payload = {
1211
+ ok: true,
1212
+ targetDir,
1213
+ locale,
1214
+ tool,
1215
+ statePath: STATE_RELATIVE_PATH,
1216
+ configPath: CONFIG_RELATIVE_PATH,
1217
+ created: loaded.created,
1218
+ mode: state.mode,
1219
+ classification: state.classification,
1220
+ current: state.current,
1221
+ next: state.detour && state.detour.active ? state.detour.returnTo : state.next,
1222
+ detour: state.detour,
1223
+ completed: state.completed,
1224
+ skipped: state.skipped,
1225
+ completedStage,
1226
+ featureSlug: state.featureSlug,
1227
+ runtime,
1228
+ agent: activation.agent,
1229
+ effectiveMode: activation.effectiveMode || null,
1230
+ instructionPath: activation.instructionPath,
1231
+ prompt: activation.prompt
1232
+ };
1233
+
1234
+ logger.log(t('workflow_next.title', {
1235
+ mode: state.mode,
1236
+ classification: state.classification
1237
+ }));
1238
+ if (completedStage) {
1239
+ logger.log(t('workflow_next.completed', { agent: `@${completedStage}` }));
1240
+ }
1241
+ if (state.detour && state.detour.active) {
1242
+ logger.log(
1243
+ t('workflow_next.detour', {
1244
+ agent: `@${state.detour.agent}`,
1245
+ returnTo: `@${state.detour.returnTo}`
1246
+ })
1247
+ );
1248
+ }
1249
+ if (activation.agent) {
1250
+ logger.log(t('workflow_next.current_agent', { agent: `@${activation.agent}` }));
1251
+ if (payload.next) {
1252
+ logger.log(t('workflow_next.next_agent', { agent: `@${payload.next}` }));
1253
+ }
1254
+ logger.log(activation.prompt);
1255
+ } else {
1256
+ logger.log(t('workflow_next.done'));
1257
+ }
1258
+ logger.log(t('workflow_next.state_file', { path: STATE_RELATIVE_PATH }));
1259
+
1260
+ return payload;
1261
+ }
1262
+
1263
+ module.exports = {
1264
+ STATE_RELATIVE_PATH,
1265
+ CONFIG_RELATIVE_PATH,
1266
+ EVENTS_RELATIVE_PATH,
1267
+ buildDefaultWorkflowConfig,
1268
+ parseFeaturesMarkdown,
1269
+ readWorkflowConfig,
1270
+ detectWorkflowMode,
1271
+ loadOrCreateState,
1272
+ reconcileWorkflowState,
1273
+ finalizeCurrentStage,
1274
+ applySkip,
1275
+ activateStage,
1276
+ runWorkflowNext,
1277
+ shouldRouteToValidator,
1278
+ detectUnsubstantiatedCompletions
1279
+ };