@jaimevalasek/aioson 1.7.2 → 1.9.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (1049) hide show
  1. package/CHANGELOG.md +595 -560
  2. package/CODE_OF_CONDUCT.md +12 -12
  3. package/CONTRIBUTING.md +13 -13
  4. package/LICENSE +661 -661
  5. package/README.md +919 -776
  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 -409
  41. package/docs/en/5-reference/i18n.md +52 -0
  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 -116
  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 -57
  64. package/docs/en/{schemas → 5-reference/schemas}/parallel-doctor.schema.json +122 -86
  65. package/docs/en/5-reference/schemas/parallel-guard.schema.json +63 -0
  66. package/docs/en/{schemas → 5-reference/schemas}/parallel-init.schema.json +53 -53
  67. package/docs/en/5-reference/schemas/parallel-merge.schema.json +84 -0
  68. package/docs/en/5-reference/schemas/parallel-status.schema.json +184 -0
  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 -0
  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 -286
  149. package/docs/pt/{comandos-cli.md → 5-referencia/comandos-cli.md} +1823 -1634
  150. package/docs/pt/5-referencia/compress-agents.md +304 -0
  151. package/docs/pt/5-referencia/design-docs-governance.md +59 -0
  152. package/docs/pt/{devlog-pipeline.md → 5-referencia/devlog-pipeline.md} +270 -270
  153. package/docs/pt/5-referencia/feature-archive.md +199 -0
  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 -296
  157. package/docs/pt/5-referencia/genome-distribution.md +232 -0
  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/5-referencia/motor-hardening.md +493 -0
  163. package/docs/pt/{output-strategy-delivery.md → 5-referencia/output-strategy-delivery.md} +655 -655
  164. package/docs/pt/5-referencia/runner-system.md +113 -0
  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 -116
  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 -250
  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 -672
  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 -50
  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 -0
  221. package/src/agents.js +92 -72
  222. package/src/autonomy-policy.js +163 -0
  223. package/src/backup-local.js +74 -74
  224. package/src/backup-provider.js +303 -303
  225. package/src/brain-query.js +171 -0
  226. package/src/cli.js +1450 -1099
  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 -160
  231. package/src/commands/artifact-validate.js +218 -189
  232. package/src/commands/auth.js +298 -0
  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 -0
  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 -0
  239. package/src/commands/classify.js +256 -256
  240. package/src/commands/cloud.js +1767 -1767
  241. package/src/commands/commit-prepare.js +610 -0
  242. package/src/commands/compress-agents.js +416 -0
  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 -175
  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 -177
  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 -292
  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 -0
  263. package/src/commands/feature-archive.js +513 -0
  264. package/src/commands/feature-close.js +554 -165
  265. package/src/commands/gate-approve.js +198 -0
  266. package/src/commands/gate-check.js +247 -228
  267. package/src/commands/genome-doctor.js +489 -41
  268. package/src/commands/genome-migrate.js +49 -49
  269. package/src/commands/git-guard.js +170 -0
  270. package/src/commands/harness.js +307 -0
  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 -340
  276. package/src/commands/info.js +41 -41
  277. package/src/commands/init.js +120 -116
  278. package/src/commands/install.js +162 -107
  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 -1641
  285. package/src/commands/locale-apply.js +54 -51
  286. package/src/commands/locale-diff.js +25 -126
  287. package/src/commands/mcp-doctor.js +407 -406
  288. package/src/commands/mcp-init.js +373 -379
  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 -0
  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 -403
  298. package/src/commands/parallel-doctor.js +850 -437
  299. package/src/commands/parallel-guard.js +241 -0
  300. package/src/commands/parallel-init.js +311 -249
  301. package/src/commands/parallel-merge.js +299 -0
  302. package/src/commands/parallel-status.js +434 -290
  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 -208
  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 -161
  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 -0
  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 -0
  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 -0
  327. package/src/commands/security-scan.js +376 -0
  328. package/src/commands/self-implement-loop.js +306 -256
  329. package/src/commands/session-guard.js +218 -218
  330. package/src/commands/setup-context.js +699 -698
  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 -788
  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 -261
  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 -329
  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 -55
  360. package/src/commands/squad-score.js +311 -250
  361. package/src/commands/squad-status.js +481 -460
  362. package/src/commands/squad-tool-register.js +157 -157
  363. package/src/commands/squad-validate.js +438 -347
  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 +122 -122
  368. package/src/commands/store-genome.js +667 -0
  369. package/src/commands/store-skill.js +247 -0
  370. package/src/commands/store-squad.js +431 -0
  371. package/src/commands/store-system.js +392 -0
  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 -0
  375. package/src/commands/tool-registry-cmd.js +232 -232
  376. package/src/commands/update.js +64 -64
  377. package/src/commands/verify-gate.js +612 -572
  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 -241
  381. package/src/commands/workflow-harden.js +231 -0
  382. package/src/commands/workflow-heal.js +136 -0
  383. package/src/commands/workflow-next.js +1279 -601
  384. package/src/commands/workflow-plan.js +108 -108
  385. package/src/commands/workflow-status.js +440 -250
  386. package/src/commands/workspace.js +144 -0
  387. package/src/constants.js +413 -417
  388. package/src/context-cache.js +159 -159
  389. package/src/context-memory.js +975 -837
  390. package/src/context-parse-reason.js +22 -22
  391. package/src/context-search.js +326 -326
  392. package/src/context-writer.js +197 -196
  393. package/src/context.js +247 -217
  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 +760 -289
  398. package/src/dossier/codemap-store.js +267 -0
  399. package/src/dossier/dossier-bootstrap.js +222 -0
  400. package/src/dossier/dossier-compact.js +159 -0
  401. package/src/dossier/lock.js +128 -0
  402. package/src/dossier/research-index-store.js +233 -0
  403. package/src/dossier/revision-store.js +313 -0
  404. package/src/dossier/schema.js +162 -0
  405. package/src/dossier/scout-section.js +127 -0
  406. package/src/dossier/store.js +406 -0
  407. package/src/execution-gateway.js +464 -461
  408. package/src/friction-scanner.js +202 -0
  409. package/src/genome-files.js +198 -198
  410. package/src/genome-format.js +442 -442
  411. package/src/genome-schema.js +238 -215
  412. package/src/genomes/bindings.js +281 -281
  413. package/src/genomes.js +500 -467
  414. package/src/handoff-contract.js +417 -0
  415. package/src/handoff-validator.js +45 -0
  416. package/src/harness/circuit-breaker.js +135 -0
  417. package/src/i18n/index.js +103 -103
  418. package/src/i18n/messages/en.js +1541 -1139
  419. package/src/i18n/messages/es.js +1325 -980
  420. package/src/i18n/messages/fr.js +1333 -987
  421. package/src/i18n/messages/pt-BR.js +1561 -1166
  422. package/src/i18n/scaffold.js +64 -64
  423. package/src/install-animation.js +260 -260
  424. package/src/install-profile.js +127 -143
  425. package/src/install-wizard.js +475 -475
  426. package/src/installer-config-merge.js +207 -0
  427. package/src/installer.js +449 -294
  428. package/src/learning-loop-archive.js +595 -0
  429. package/src/learning-loop-doctor.js +217 -0
  430. package/src/learning-loop-engine.js +254 -0
  431. package/src/learning-loop-fts5.js +132 -0
  432. package/src/learning-loop-migration.js +163 -0
  433. package/src/lib/dev-resume.js +140 -0
  434. package/src/lib/dossier-telemetry.js +36 -0
  435. package/src/lib/genomes/compat.js +206 -206
  436. package/src/lib/genomes/migrate.js +90 -90
  437. package/src/lib/git-commit-guard.js +751 -0
  438. package/src/lib/health-check.js +158 -158
  439. package/src/lib/hook-protocol.js +76 -76
  440. package/src/lib/llm-content-sanitizer.js +44 -0
  441. package/src/lib/security/artifact-reader.js +167 -0
  442. package/src/lib/security/exit-codes.js +51 -0
  443. package/src/lib/security/findings-writer.js +176 -0
  444. package/src/lib/security/runtime-events.js +77 -0
  445. package/src/lib/security/secrets-regex.js +115 -0
  446. package/src/lib/squads/genome-repair.js +49 -49
  447. package/src/lib/store/security-scan.js +175 -0
  448. package/src/lib/terminal-checkbox.js +135 -0
  449. package/src/lib/terminal-picker.js +447 -0
  450. package/src/lib/tmux-launcher.js +163 -0
  451. package/src/lib/tool-capabilities.js +102 -0
  452. package/src/lib/webhook-server.js +328 -328
  453. package/src/locales.js +88 -84
  454. package/src/mcp/apps/squad-dashboard/app.js +163 -163
  455. package/src/mcp/apps/squad-dashboard/index.html +261 -261
  456. package/src/mcp/apps/squad-dashboard/mcp-manifest.json +23 -23
  457. package/src/mcp/resources/squad-state.js +130 -130
  458. package/src/mcp-connectors/registry.js +602 -602
  459. package/src/memory-reflect-engine.js +359 -0
  460. package/src/notify-renderer.js +32 -0
  461. package/src/onboarding.js +305 -305
  462. package/src/parallel-workspace.js +756 -0
  463. package/src/parser.js +66 -59
  464. package/src/path-guard.js +47 -0
  465. package/src/permissions-generator.js +400 -0
  466. package/src/preflight-engine.js +654 -443
  467. package/src/prompt-tool.js +20 -20
  468. package/src/qa-html-report.js +472 -472
  469. package/src/recovery-context-session.js +154 -154
  470. package/src/runner/cascade.js +97 -97
  471. package/src/runner/cli-launcher.js +109 -109
  472. package/src/runner/plan-importer.js +63 -63
  473. package/src/runner/queue-store.js +159 -159
  474. package/src/runtime-store.js +2720 -2676
  475. package/src/sandbox.js +194 -177
  476. package/src/self-healing.js +142 -0
  477. package/src/session-handoff.js +295 -77
  478. package/src/squad/agent-teams-adapter.js +270 -264
  479. package/src/squad/brief-validator.js +350 -350
  480. package/src/squad/bus-bridge.js +140 -140
  481. package/src/squad/context-compactor.js +265 -265
  482. package/src/squad/cross-ai-synthesizer.js +250 -250
  483. package/src/squad/external-session.js +180 -180
  484. package/src/squad/hooks-generator.js +196 -196
  485. package/src/squad/inter-squad-events.js +175 -175
  486. package/src/squad/inter-squad.js +74 -74
  487. package/src/squad/intra-bus.js +345 -345
  488. package/src/squad/learning-extractor.js +213 -213
  489. package/src/squad/pattern-detector.js +365 -365
  490. package/src/squad/preflight-context.js +296 -296
  491. package/src/squad/recovery-context.js +372 -372
  492. package/src/squad/reflection.js +365 -365
  493. package/src/squad/squad-scaffold.js +341 -177
  494. package/src/squad/state-manager.js +310 -310
  495. package/src/squad/task-decomposer.js +652 -652
  496. package/src/squad/verify-gate.js +303 -303
  497. package/src/squad/worktree-manager.js +114 -114
  498. package/src/squad-daemon.js +490 -490
  499. package/src/squad-dashboard/api.js +223 -223
  500. package/src/squad-dashboard/attachment-handler.js +93 -93
  501. package/src/squad-dashboard/context-monitor.js +157 -157
  502. package/src/squad-dashboard/execution-logs.js +115 -115
  503. package/src/squad-dashboard/hunk-review.js +209 -209
  504. package/src/squad-dashboard/metrics.js +133 -133
  505. package/src/squad-dashboard/process-monitor.js +125 -125
  506. package/src/squad-dashboard/renderer.js +858 -858
  507. package/src/squad-dashboard/server.js +232 -232
  508. package/src/squad-dashboard/styles.js +525 -525
  509. package/src/squad-dashboard/token-tracker.js +99 -99
  510. package/src/squads/apply-genome.js +21 -21
  511. package/src/squads/genome-binding-service.js +154 -154
  512. package/src/sub-task-engine.js +415 -0
  513. package/src/sub-task-schemas.js +150 -0
  514. package/src/sub-task-state.js +152 -0
  515. package/src/sub-task-telemetry.js +69 -0
  516. package/src/test-briefing.js +226 -0
  517. package/src/tool-executor.js +94 -94
  518. package/src/updater.js +39 -39
  519. package/src/utils.js +49 -46
  520. package/src/version.js +50 -50
  521. package/src/web.js +284 -284
  522. package/src/worker-runner.js +541 -524
  523. package/src/workflow-gates.js +185 -0
  524. package/template/.aioson/advisors/.gitkeep +1 -1
  525. package/template/.aioson/agents/analyst.md +333 -372
  526. package/template/.aioson/agents/architect.md +325 -338
  527. package/template/.aioson/agents/briefing.md +264 -0
  528. package/template/.aioson/agents/committer.md +161 -0
  529. package/template/.aioson/agents/copywriter.md +937 -463
  530. package/template/.aioson/agents/design-hybrid-forge.md +141 -141
  531. package/template/.aioson/agents/dev.md +295 -779
  532. package/template/.aioson/agents/deyvin.md +198 -290
  533. package/template/.aioson/agents/discover.md +235 -0
  534. package/template/.aioson/agents/discovery-design-doc.md +56 -264
  535. package/template/.aioson/agents/genome.md +1904 -314
  536. package/template/.aioson/agents/manifests/analyst.manifest.json +26 -0
  537. package/template/.aioson/agents/manifests/architect.manifest.json +23 -0
  538. package/template/.aioson/agents/manifests/committer.manifest.json +23 -0
  539. package/template/.aioson/agents/manifests/dev.manifest.json +54 -0
  540. package/template/.aioson/agents/manifests/deyvin.manifest.json +41 -0
  541. package/template/.aioson/agents/manifests/orchestrator.manifest.json +30 -0
  542. package/template/.aioson/agents/manifests/pentester.manifest.json +39 -0
  543. package/template/.aioson/agents/manifests/pm.manifest.json +26 -0
  544. package/template/.aioson/agents/manifests/product.manifest.json +23 -0
  545. package/template/.aioson/agents/manifests/qa.manifest.json +41 -0
  546. package/template/.aioson/agents/manifests/setup.manifest.json +20 -0
  547. package/template/.aioson/agents/manifests/ux-ui.manifest.json +24 -0
  548. package/template/.aioson/agents/neo.md +341 -233
  549. package/template/.aioson/agents/orache.md +430 -434
  550. package/template/.aioson/agents/orchestrator.md +274 -364
  551. package/template/.aioson/agents/pair.md +5 -5
  552. package/template/.aioson/agents/pentester.md +289 -0
  553. package/template/.aioson/agents/pm.md +141 -194
  554. package/template/.aioson/agents/product.md +351 -518
  555. package/template/.aioson/agents/profiler-enricher.md +331 -280
  556. package/template/.aioson/agents/profiler-forge.md +212 -202
  557. package/template/.aioson/agents/profiler-researcher.md +282 -259
  558. package/template/.aioson/agents/qa.md +432 -688
  559. package/template/.aioson/agents/setup.md +423 -649
  560. package/template/.aioson/agents/sheldon.md +259 -829
  561. package/template/.aioson/agents/site-forge.md +281 -1753
  562. package/template/.aioson/agents/squad.md +160 -2027
  563. package/template/.aioson/agents/tester.md +536 -463
  564. package/template/.aioson/agents/ux-ui.md +195 -870
  565. package/template/.aioson/agents/validator.md +101 -0
  566. package/template/.aioson/brains/README.md +132 -128
  567. package/template/.aioson/brains/_archived/.gitkeep +0 -0
  568. package/template/.aioson/brains/_index.json +34 -16
  569. package/template/.aioson/brains/dev/patterns.brain.json +79 -0
  570. package/template/.aioson/brains/scripts/query.js +107 -103
  571. package/template/.aioson/brains/sheldon/architecture-decisions.brain.json +79 -0
  572. package/template/.aioson/brains/site-forge/visual-patterns.brain.json +205 -205
  573. package/template/.aioson/config/autonomy-protocol.json +125 -0
  574. package/template/.aioson/config/learning-loop.json +10 -0
  575. package/template/.aioson/config/scout-engine.json +1 -0
  576. package/template/.aioson/config.md +410 -382
  577. package/template/.aioson/constitution.md +36 -33
  578. package/template/.aioson/context/_archived/.gitkeep +0 -0
  579. package/template/.aioson/context/design-doc.md +136 -0
  580. package/template/.aioson/context/project-map.md +57 -0
  581. package/template/.aioson/context/project-pulse.md +34 -34
  582. package/template/.aioson/context/seeds/seed-example.md +27 -27
  583. package/template/.aioson/context/spec.md.template +54 -54
  584. package/template/.aioson/context/user-profile.md +42 -42
  585. package/template/.aioson/design-docs/code-reuse.md +48 -0
  586. package/template/.aioson/design-docs/componentization.md +47 -0
  587. package/template/.aioson/design-docs/file-size.md +52 -0
  588. package/template/.aioson/design-docs/folder-structure.md +51 -0
  589. package/template/.aioson/design-docs/naming.md +54 -0
  590. package/template/.aioson/docs/LAYERS.md +89 -79
  591. package/template/.aioson/docs/README.md +76 -76
  592. package/template/.aioson/docs/autonomy-protocol.md +80 -0
  593. package/template/.aioson/docs/briefing/briefing-craft.md +237 -0
  594. package/template/.aioson/docs/dev/execution-discipline.md +106 -0
  595. package/template/.aioson/docs/dev/stack-conventions.md +83 -0
  596. package/template/.aioson/docs/deyvin/continuity-recovery.md +57 -0
  597. package/template/.aioson/docs/deyvin/debugging-escalation.md +30 -0
  598. package/template/.aioson/docs/deyvin/pair-execution.md +44 -0
  599. package/template/.aioson/docs/deyvin/runtime-handoffs.md +36 -0
  600. package/template/.aioson/docs/example-external-api-context.md +72 -72
  601. package/template/.aioson/docs/pentester/app-playbooks.md +206 -0
  602. package/template/.aioson/docs/pentester/llm-supplychain.md +165 -0
  603. package/template/.aioson/docs/product/conversation-playbook.md +116 -0
  604. package/template/.aioson/docs/product/prd-contract.md +107 -0
  605. package/template/.aioson/docs/product/quality-lens.md +57 -0
  606. package/template/.aioson/docs/product/research-loop.md +65 -0
  607. package/template/.aioson/docs/sheldon/enrichment-paths.md +134 -0
  608. package/template/.aioson/docs/sheldon/harness-contract.md +118 -0
  609. package/template/.aioson/docs/sheldon/quality-lens.md +57 -0
  610. package/template/.aioson/docs/sheldon/research-loop.md +56 -0
  611. package/template/.aioson/docs/sheldon/web-intelligence.md +75 -0
  612. package/template/.aioson/docs/site-forge-build.md +195 -0
  613. package/template/.aioson/docs/site-forge-extraction.md +135 -0
  614. package/template/.aioson/docs/site-forge-qa.md +155 -0
  615. package/template/.aioson/docs/site-forge-recon.md +434 -0
  616. package/template/.aioson/docs/site-forge-transform.md +249 -0
  617. package/template/.aioson/docs/squad/content-output.md +91 -0
  618. package/template/.aioson/docs/squad/creation-flow.md +149 -0
  619. package/template/.aioson/docs/squad/domain-breadth.md +322 -0
  620. package/template/.aioson/docs/squad/domain-classification.md +117 -0
  621. package/template/.aioson/docs/squad/genome-bindings.md +47 -0
  622. package/template/.aioson/docs/squad/package-contract.md +260 -0
  623. package/template/.aioson/docs/squad/quality-lens.md +60 -0
  624. package/template/.aioson/docs/squad/research-loop.md +59 -0
  625. package/template/.aioson/docs/squad/session-operations.md +117 -0
  626. package/template/.aioson/docs/squad/workflow-quality.md +165 -0
  627. package/template/.aioson/docs/tester/coverage-quality.md +351 -0
  628. package/template/.aioson/docs/ux-ui/accessibility-audit.md +55 -0
  629. package/template/.aioson/docs/ux-ui/audit-mode.md +86 -0
  630. package/template/.aioson/docs/ux-ui/component-map.md +35 -0
  631. package/template/.aioson/docs/ux-ui/design-execution.md +111 -0
  632. package/template/.aioson/docs/ux-ui/design-gate.md +27 -0
  633. package/template/.aioson/docs/ux-ui/research-mode.md +39 -0
  634. package/template/.aioson/docs/ux-ui/site-delivery.md +156 -0
  635. package/template/.aioson/docs/ux-ui/token-contract.md +57 -0
  636. package/template/.aioson/genomes/INDEX.md +195 -0
  637. package/template/.aioson/genomes/copywriting/SKILL.md +137 -0
  638. package/template/.aioson/genomes/copywriting/manifest.json +140 -0
  639. package/template/.aioson/genomes/copywriting/references/application-notes.md +145 -0
  640. package/template/.aioson/genomes/copywriting/references/decision-weights.md +45 -0
  641. package/template/.aioson/genomes/copywriting/references/frameworks/5-act-narrative.md +184 -0
  642. package/template/.aioson/genomes/copywriting/references/frameworks/classical-formulas.md +164 -0
  643. package/template/.aioson/genomes/copywriting/references/frameworks/offer-stack.md +195 -0
  644. package/template/.aioson/genomes/copywriting/references/frameworks/one-belief.md +135 -0
  645. package/template/.aioson/genomes/copywriting/references/frameworks/pms-research.md +211 -0
  646. package/template/.aioson/genomes/copywriting/references/frameworks/two-paths-close.md +190 -0
  647. package/template/.aioson/genomes/copywriting/references/heuristics.md +114 -0
  648. package/template/.aioson/genomes/copywriting/references/meta-axioms.md +68 -0
  649. package/template/.aioson/genomes/copywriting/references/methodology.md +115 -0
  650. package/template/.aioson/genomes/copywriting-brunson/SKILL.md +133 -0
  651. package/template/.aioson/genomes/copywriting-brunson/manifest.json +152 -0
  652. package/template/.aioson/genomes/copywriting-brunson/references/application-notes.md +113 -0
  653. package/template/.aioson/genomes/copywriting-brunson/references/decision-weights.md +33 -0
  654. package/template/.aioson/genomes/copywriting-brunson/references/evidence-and-attribution.md +81 -0
  655. package/template/.aioson/genomes/copywriting-brunson/references/frameworks/6-part-structure.md +136 -0
  656. package/template/.aioson/genomes/copywriting-brunson/references/frameworks/origin-story.md +121 -0
  657. package/template/.aioson/genomes/copywriting-brunson/references/frameworks/perfect-webinar-script.md +139 -0
  658. package/template/.aioson/genomes/copywriting-brunson/references/frameworks/persuasive-storytelling-5-structures.md +164 -0
  659. package/template/.aioson/genomes/copywriting-brunson/references/frameworks/value-stack.md +136 -0
  660. package/template/.aioson/genomes/copywriting-brunson/references/frameworks/who-what-why-how.md +110 -0
  661. package/template/.aioson/genomes/copywriting-brunson/references/meta-axioms.md +36 -0
  662. package/template/.aioson/genomes/copywriting-brunson/references/methodology.md +112 -0
  663. package/template/.aioson/git-guard.json +12 -0
  664. package/template/.aioson/mcp/servers.md +23 -24
  665. package/template/.aioson/profiler-reports/.gitkeep +1 -1
  666. package/template/.aioson/rules/README.md +69 -69
  667. package/template/.aioson/rules/_archived/.gitkeep +0 -0
  668. package/template/.aioson/rules/agent-language-policy.md +93 -0
  669. package/template/.aioson/rules/aioson-context-boundary.md +63 -0
  670. package/template/.aioson/rules/canonical-path-contract.md +47 -0
  671. package/template/.aioson/rules/data-format-convention.md +74 -136
  672. package/template/.aioson/rules/disk-first-artifacts.md +44 -0
  673. package/template/.aioson/rules/example-monetary-values.md +30 -30
  674. package/template/.aioson/rules/output-brevity.md +44 -0
  675. package/template/.aioson/rules/prd-section-ownership.md +49 -0
  676. package/template/.aioson/rules/security-baseline.md +139 -0
  677. package/template/.aioson/rules/spec-level-ownership.md +61 -0
  678. package/template/.aioson/rules/squad/README.md +50 -50
  679. package/template/.aioson/rules/squad-driver-pattern.md +81 -0
  680. package/template/.aioson/schemas/content-blueprint.schema.json +30 -30
  681. package/template/.aioson/schemas/genome-meta.schema.json +150 -150
  682. package/template/.aioson/schemas/genome.schema.json +115 -115
  683. package/template/.aioson/schemas/readiness.schema.json +27 -27
  684. package/template/.aioson/schemas/squad-blueprint.schema.json +228 -204
  685. package/template/.aioson/schemas/squad-manifest.schema.json +874 -830
  686. package/template/.aioson/skills/design/aurora-command-ui/SKILL.md +243 -243
  687. package/template/.aioson/skills/design/aurora-command-ui/references/art-direction.md +293 -293
  688. package/template/.aioson/skills/design/aurora-command-ui/references/components.md +827 -827
  689. package/template/.aioson/skills/design/aurora-command-ui/references/dashboards.md +250 -250
  690. package/template/.aioson/skills/design/aurora-command-ui/references/design-tokens.md +585 -585
  691. package/template/.aioson/skills/design/aurora-command-ui/references/motion.md +365 -365
  692. package/template/.aioson/skills/design/aurora-command-ui/references/patterns.md +482 -482
  693. package/template/.aioson/skills/design/aurora-command-ui/references/websites.md +387 -387
  694. package/template/.aioson/skills/design/bold-editorial-ui/SKILL.md +205 -205
  695. package/template/.aioson/skills/design/bold-editorial-ui/references/art-direction.md +338 -338
  696. package/template/.aioson/skills/design/bold-editorial-ui/references/components.md +977 -977
  697. package/template/.aioson/skills/design/bold-editorial-ui/references/dashboards.md +218 -218
  698. package/template/.aioson/skills/design/bold-editorial-ui/references/design-tokens.md +326 -326
  699. package/template/.aioson/skills/design/bold-editorial-ui/references/motion.md +461 -461
  700. package/template/.aioson/skills/design/bold-editorial-ui/references/patterns.md +293 -293
  701. package/template/.aioson/skills/design/bold-editorial-ui/references/websites.md +352 -352
  702. package/template/.aioson/skills/design/clean-saas-ui/SKILL.md +210 -210
  703. package/template/.aioson/skills/design/clean-saas-ui/references/art-direction.md +319 -319
  704. package/template/.aioson/skills/design/clean-saas-ui/references/components.md +365 -365
  705. package/template/.aioson/skills/design/clean-saas-ui/references/dashboards.md +196 -196
  706. package/template/.aioson/skills/design/clean-saas-ui/references/design-tokens.md +244 -244
  707. package/template/.aioson/skills/design/clean-saas-ui/references/motion.md +235 -235
  708. package/template/.aioson/skills/design/clean-saas-ui/references/patterns.md +215 -215
  709. package/template/.aioson/skills/design/clean-saas-ui/references/websites.md +295 -295
  710. package/template/.aioson/skills/design/cognitive-core-ui/SKILL.md +203 -203
  711. package/template/.aioson/skills/design/cognitive-core-ui/references/art-direction.md +339 -339
  712. package/template/.aioson/skills/design/cognitive-core-ui/references/components.md +407 -407
  713. package/template/.aioson/skills/design/cognitive-core-ui/references/dashboards.md +272 -272
  714. package/template/.aioson/skills/design/cognitive-core-ui/references/design-tokens.md +524 -524
  715. package/template/.aioson/skills/design/cognitive-core-ui/references/motion.md +279 -279
  716. package/template/.aioson/skills/design/cognitive-core-ui/references/patterns.md +289 -289
  717. package/template/.aioson/skills/design/cognitive-core-ui/references/websites.md +437 -437
  718. package/template/.aioson/skills/design/glassmorphism-ui/SKILL.md +222 -222
  719. package/template/.aioson/skills/design/glassmorphism-ui/references/art-direction.md +159 -159
  720. package/template/.aioson/skills/design/glassmorphism-ui/references/components.md +498 -498
  721. package/template/.aioson/skills/design/glassmorphism-ui/references/dashboards.md +236 -236
  722. package/template/.aioson/skills/design/glassmorphism-ui/references/design-tokens.md +274 -274
  723. package/template/.aioson/skills/design/glassmorphism-ui/references/motion.md +355 -355
  724. package/template/.aioson/skills/design/glassmorphism-ui/references/patterns.md +198 -198
  725. package/template/.aioson/skills/design/glassmorphism-ui/references/websites.md +307 -307
  726. package/template/.aioson/skills/design/interface-design/SKILL.md +47 -47
  727. package/template/.aioson/skills/design/interface-design/references/components-and-states.md +105 -105
  728. package/template/.aioson/skills/design/interface-design/references/design-directions.md +101 -101
  729. package/template/.aioson/skills/design/interface-design/references/handoff-and-quality.md +71 -71
  730. package/template/.aioson/skills/design/interface-design/references/intent-and-domain.md +74 -74
  731. package/template/.aioson/skills/design/interface-design/references/tokens-and-depth.md +173 -173
  732. package/template/.aioson/skills/design/neo-brutalist-ui/SKILL.md +213 -213
  733. package/template/.aioson/skills/design/neo-brutalist-ui/references/art-direction.md +228 -228
  734. package/template/.aioson/skills/design/neo-brutalist-ui/references/components.md +855 -855
  735. package/template/.aioson/skills/design/neo-brutalist-ui/references/dashboards.md +334 -334
  736. package/template/.aioson/skills/design/neo-brutalist-ui/references/design-tokens.md +342 -342
  737. package/template/.aioson/skills/design/neo-brutalist-ui/references/motion.md +286 -286
  738. package/template/.aioson/skills/design/neo-brutalist-ui/references/patterns.md +458 -458
  739. package/template/.aioson/skills/design/neo-brutalist-ui/references/websites.md +723 -723
  740. package/template/.aioson/skills/design/premium-command-center-ui/SKILL.md +62 -62
  741. package/template/.aioson/skills/design/premium-command-center-ui/references/operations.md +74 -74
  742. package/template/.aioson/skills/design/premium-command-center-ui/references/patterns.md +116 -116
  743. package/template/.aioson/skills/design/premium-command-center-ui/references/validation.md +47 -47
  744. package/template/.aioson/skills/design/premium-command-center-ui/references/visual-system.md +215 -215
  745. package/template/.aioson/skills/design/pt.squarespace.com/.skill-meta.json +31 -31
  746. package/template/.aioson/skills/design/pt.squarespace.com/SKILL.md +66 -66
  747. package/template/.aioson/skills/design/pt.squarespace.com/references/components.md +368 -368
  748. package/template/.aioson/skills/design/pt.squarespace.com/references/design-tokens.md +150 -150
  749. package/template/.aioson/skills/design/pt.squarespace.com/references/motion.md +270 -270
  750. package/template/.aioson/skills/design/pt.squarespace.com/references/patterns.md +189 -189
  751. package/template/.aioson/skills/design/pt.squarespace.com/references/websites.md +165 -165
  752. package/template/.aioson/skills/design/warm-craft-ui/SKILL.md +209 -209
  753. package/template/.aioson/skills/design/warm-craft-ui/references/art-direction.md +324 -324
  754. package/template/.aioson/skills/design/warm-craft-ui/references/components.md +508 -508
  755. package/template/.aioson/skills/design/warm-craft-ui/references/dashboards.md +223 -223
  756. package/template/.aioson/skills/design/warm-craft-ui/references/design-tokens.md +374 -374
  757. package/template/.aioson/skills/design/warm-craft-ui/references/motion.md +356 -356
  758. package/template/.aioson/skills/design/warm-craft-ui/references/patterns.md +288 -288
  759. package/template/.aioson/skills/design/warm-craft-ui/references/websites.md +289 -289
  760. package/template/.aioson/skills/design-system/SKILL.md +92 -92
  761. package/template/.aioson/skills/design-system/components/SKILL.md +274 -274
  762. package/template/.aioson/skills/design-system/dashboards/SKILL.md +184 -184
  763. package/template/.aioson/skills/design-system/foundations/SKILL.md +250 -250
  764. package/template/.aioson/skills/design-system/motion/SKILL.md +197 -197
  765. package/template/.aioson/skills/design-system/patterns/SKILL.md +231 -231
  766. package/template/.aioson/skills/dynamic/README.md +30 -30
  767. package/template/.aioson/skills/dynamic/cardano-docs.md +16 -16
  768. package/template/.aioson/skills/dynamic/ethereum-docs.md +17 -17
  769. package/template/.aioson/skills/dynamic/flux-ui-docs.md +13 -13
  770. package/template/.aioson/skills/dynamic/laravel-docs.md +41 -41
  771. package/template/.aioson/skills/dynamic/npm-packages.md +16 -16
  772. package/template/.aioson/skills/dynamic/solana-docs.md +16 -16
  773. package/template/.aioson/skills/marketing/references/anti-patterns.md +254 -254
  774. package/template/.aioson/skills/marketing/references/cta-matrix.md +361 -0
  775. package/template/.aioson/skills/marketing/references/fascinations.md +192 -192
  776. package/template/.aioson/skills/marketing/references/five-acts.md +248 -248
  777. package/template/.aioson/skills/marketing/references/headline-matrix.md +358 -0
  778. package/template/.aioson/skills/marketing/references/market-intelligence.md +198 -198
  779. package/template/.aioson/skills/marketing/references/offer-structure.md +203 -203
  780. package/template/.aioson/skills/marketing/references/one-belief.md +149 -149
  781. package/template/.aioson/skills/marketing/references/patterns.md +218 -218
  782. package/template/.aioson/skills/marketing/references/platform-constraints.md +337 -0
  783. package/template/.aioson/skills/marketing/references/pms-research.md +193 -193
  784. package/template/.aioson/skills/marketing/vsl-craft.md +385 -385
  785. package/template/.aioson/skills/premium-visual-design/SKILL.md +83 -83
  786. package/template/.aioson/skills/premium-visual-design/components/agent-badge.md +92 -92
  787. package/template/.aioson/skills/premium-visual-design/components/dependency-node.md +102 -102
  788. package/template/.aioson/skills/premium-visual-design/components/mention-autocomplete.md +136 -136
  789. package/template/.aioson/skills/premium-visual-design/components/notification-center.md +136 -136
  790. package/template/.aioson/skills/premium-visual-design/components/review-action-bar.md +188 -188
  791. package/template/.aioson/skills/premium-visual-design/components/team-switcher.md +131 -131
  792. package/template/.aioson/skills/premium-visual-design/patterns/agent-message-thread.md +198 -198
  793. package/template/.aioson/skills/premium-visual-design/patterns/notification-panel.md +275 -275
  794. package/template/.aioson/skills/premium-visual-design/patterns/review-workflow-ui.md +234 -234
  795. package/template/.aioson/skills/premium-visual-design/patterns/task-dependency-graph.md +147 -147
  796. package/template/.aioson/skills/premium-visual-design/tokens/status-extended.md +142 -142
  797. package/template/.aioson/skills/process/aioson-spec-driven/SKILL.md +46 -46
  798. package/template/.aioson/skills/process/aioson-spec-driven/references/analyst.md +30 -30
  799. package/template/.aioson/skills/process/aioson-spec-driven/references/approval-gates.md +109 -109
  800. package/template/.aioson/skills/process/aioson-spec-driven/references/architect.md +23 -23
  801. package/template/.aioson/skills/process/aioson-spec-driven/references/artifact-map.md +44 -44
  802. package/template/.aioson/skills/process/aioson-spec-driven/references/classification-map.md +37 -37
  803. package/template/.aioson/skills/process/aioson-spec-driven/references/dev.md +47 -47
  804. package/template/.aioson/skills/process/aioson-spec-driven/references/deyvin.md +27 -27
  805. package/template/.aioson/skills/process/aioson-spec-driven/references/hardening-lane.md +49 -49
  806. package/template/.aioson/skills/process/aioson-spec-driven/references/maintenance-and-state.md +101 -101
  807. package/template/.aioson/skills/process/aioson-spec-driven/references/pm.md +30 -0
  808. package/template/.aioson/skills/process/aioson-spec-driven/references/product.md +25 -25
  809. package/template/.aioson/skills/process/aioson-spec-driven/references/qa.md +30 -30
  810. package/template/.aioson/skills/process/aioson-spec-driven/references/sheldon.md +25 -25
  811. package/template/.aioson/skills/process/aioson-spec-driven/references/ui-language.md +75 -75
  812. package/template/.aioson/skills/process/design-hybrid-forge/SKILL.md +147 -147
  813. package/template/.aioson/skills/process/design-hybrid-forge/references/crossover-protocol.md +221 -221
  814. package/template/.aioson/skills/process/design-hybrid-forge/references/naming-registry.md +88 -88
  815. package/template/.aioson/skills/process/design-hybrid-forge/references/output-contract.md +306 -306
  816. package/template/.aioson/skills/process/design-hybrid-forge/references/pair-compatibility.md +149 -149
  817. package/template/.aioson/skills/process/design-hybrid-forge/references/quality-gates.md +208 -208
  818. package/template/.aioson/skills/process/design-hybrid-forge/references/variation-library.md +125 -125
  819. package/template/.aioson/skills/process/secure-tdd/SKILL.md +97 -0
  820. package/template/.aioson/skills/process/simplify/SKILL.md +173 -173
  821. package/template/.aioson/skills/references/premium-command-center-ui/master-application-prompt.md +79 -79
  822. package/template/.aioson/skills/references/premium-command-center-ui/operational-ux-playbook.md +253 -253
  823. package/template/.aioson/skills/references/premium-command-center-ui/quality-validation-checklist.md +82 -82
  824. package/template/.aioson/skills/references/premium-command-center-ui/visual-system-and-component-patterns.md +270 -270
  825. package/template/.aioson/skills/squad/SKILL.md +58 -58
  826. package/template/.aioson/skills/squad/formats/catalog.json +15 -15
  827. package/template/.aioson/skills/squad/formats/content/blog-post.md +47 -47
  828. package/template/.aioson/skills/squad/formats/content/newsletter.md +47 -47
  829. package/template/.aioson/skills/squad/formats/creative/podcast-script.md +43 -43
  830. package/template/.aioson/skills/squad/formats/creative/video-script.md +41 -41
  831. package/template/.aioson/skills/squad/formats/social/instagram-feed.md +42 -42
  832. package/template/.aioson/skills/squad/formats/social/linkedin-post.md +42 -42
  833. package/template/.aioson/skills/squad/formats/social/tiktok.md +39 -39
  834. package/template/.aioson/skills/squad/formats/social/twitter-thread.md +39 -39
  835. package/template/.aioson/skills/squad/formats/social/youtube-long.md +47 -47
  836. package/template/.aioson/skills/squad/formats/social/youtube-shorts.md +39 -39
  837. package/template/.aioson/skills/squad/patterns/multi-platform-pattern.md +108 -108
  838. package/template/.aioson/skills/squad/patterns/persona-based-pattern.md +98 -98
  839. package/template/.aioson/skills/squad/patterns/pipeline-pattern.md +106 -106
  840. package/template/.aioson/skills/squad/patterns/review-loop-pattern.md +81 -81
  841. package/template/.aioson/skills/squad/references/checklist-templates.md +122 -122
  842. package/template/.aioson/skills/squad/references/executor-archetypes.md +123 -123
  843. package/template/.aioson/skills/squad/references/workflow-templates.md +169 -169
  844. package/template/.aioson/skills/static/context-budget-guide.md +46 -46
  845. package/template/.aioson/skills/static/debugging-protocol.md +42 -42
  846. package/template/.aioson/skills/static/django-patterns.md +342 -342
  847. package/template/.aioson/skills/static/fastapi-patterns.md +344 -344
  848. package/template/.aioson/skills/static/filament-patterns.md +267 -267
  849. package/template/.aioson/skills/static/flux-ui-components.md +262 -262
  850. package/template/.aioson/skills/static/git-conventions.md +227 -227
  851. package/template/.aioson/skills/static/git-worktrees.md +36 -36
  852. package/template/.aioson/skills/static/harness-sensors.md +74 -74
  853. package/template/.aioson/skills/static/harness-validate/SKILL.md +46 -0
  854. package/template/.aioson/skills/static/jetstream-setup.md +200 -200
  855. package/template/.aioson/skills/static/landing-page-deploy.md +192 -192
  856. package/template/.aioson/skills/static/landing-page-forge.md +730 -730
  857. package/template/.aioson/skills/static/laravel-conventions.md +491 -491
  858. package/template/.aioson/skills/static/multi-agent-patterns.md +43 -43
  859. package/template/.aioson/skills/static/nextjs-patterns.md +321 -321
  860. package/template/.aioson/skills/static/node-express-patterns.md +317 -317
  861. package/template/.aioson/skills/static/node-typescript-patterns.md +282 -282
  862. package/template/.aioson/skills/static/rails-conventions.md +307 -307
  863. package/template/.aioson/skills/static/react-motion-patterns.md +599 -599
  864. package/template/.aioson/skills/static/static-html-patterns/checklists.md +43 -43
  865. package/template/.aioson/skills/static/static-html-patterns/css-tokens.md +609 -609
  866. package/template/.aioson/skills/static/static-html-patterns/motion.md +193 -193
  867. package/template/.aioson/skills/static/static-html-patterns/premium.md +711 -711
  868. package/template/.aioson/skills/static/static-html-patterns/structure.md +209 -209
  869. package/template/.aioson/skills/static/static-html-patterns/utilities.md +190 -190
  870. package/template/.aioson/skills/static/static-html-patterns.md +80 -80
  871. package/template/.aioson/skills/static/tall-stack-patterns.md +286 -286
  872. package/template/.aioson/skills/static/threejs-patterns.md +929 -929
  873. package/template/.aioson/skills/static/ui-ux-modern.md +76 -76
  874. package/template/.aioson/skills/static/web-research-cache.md +115 -112
  875. package/template/.aioson/skills/static/web3-cardano-patterns.md +337 -337
  876. package/template/.aioson/skills/static/web3-ethereum-patterns.md +310 -310
  877. package/template/.aioson/skills/static/web3-security-checklist.md +284 -284
  878. package/template/.aioson/skills/static/web3-solana-patterns.md +324 -324
  879. package/template/.aioson/squads/memory.md +5 -5
  880. package/template/.aioson/tasks/implementation-plan.md +327 -327
  881. package/template/.aioson/tasks/squad-analyze.md +83 -83
  882. package/template/.aioson/tasks/squad-create.md +148 -121
  883. package/template/.aioson/tasks/squad-design.md +206 -158
  884. package/template/.aioson/tasks/squad-execution-plan.md +279 -279
  885. package/template/.aioson/tasks/squad-export.md +20 -20
  886. package/template/.aioson/tasks/squad-extend.md +68 -68
  887. package/template/.aioson/tasks/squad-investigate.md +57 -44
  888. package/template/.aioson/tasks/squad-learning-review.md +44 -44
  889. package/template/.aioson/tasks/squad-output-config.md +177 -177
  890. package/template/.aioson/tasks/squad-pipeline.md +122 -122
  891. package/template/.aioson/tasks/squad-profile.md +48 -48
  892. package/template/.aioson/tasks/squad-refresh.md +236 -0
  893. package/template/.aioson/tasks/squad-repair.md +85 -85
  894. package/template/.aioson/tasks/squad-review.md +61 -61
  895. package/template/.aioson/tasks/squad-task-decompose.md +66 -66
  896. package/template/.aioson/tasks/squad-validate.md +58 -58
  897. package/template/.aioson/templates/reflect-prompts/current-state.md +36 -0
  898. package/template/.aioson/templates/reflect-prompts/how-it-works.md +23 -0
  899. package/template/.aioson/templates/reflect-prompts/what-it-does.md +21 -0
  900. package/template/.aioson/templates/squads/content-basic/template.json +21 -21
  901. package/template/.aioson/templates/squads/digital-marketing-agency/template.json +96 -96
  902. package/template/.aioson/templates/squads/media-channel/template.json +24 -24
  903. package/template/.aioson/templates/squads/research-analysis/template.json +22 -22
  904. package/template/.aioson/templates/squads/software-delivery/template.json +21 -21
  905. package/template/.claude/commands/aioson/agent/analyst.md +5 -5
  906. package/template/.claude/commands/aioson/agent/architect.md +5 -5
  907. package/template/.claude/commands/aioson/agent/briefing.md +5 -0
  908. package/template/.claude/commands/aioson/agent/committer.md +5 -0
  909. package/template/.claude/commands/aioson/agent/copywriter.md +5 -0
  910. package/template/.claude/commands/aioson/agent/design-hybrid-forge.md +5 -5
  911. package/template/.claude/commands/aioson/agent/dev.md +5 -5
  912. package/template/.claude/commands/aioson/agent/deyvin.md +5 -5
  913. package/template/.claude/commands/aioson/agent/discover.md +5 -0
  914. package/template/.claude/commands/aioson/agent/discovery-design-doc.md +5 -5
  915. package/template/.claude/commands/aioson/agent/genome.md +5 -5
  916. package/template/.claude/commands/aioson/agent/neo.md +5 -5
  917. package/template/.claude/commands/aioson/agent/orache.md +5 -5
  918. package/template/.claude/commands/aioson/agent/orchestrator.md +5 -5
  919. package/template/.claude/commands/aioson/agent/pair.md +5 -0
  920. package/template/.claude/commands/aioson/agent/pentester.md +5 -0
  921. package/template/.claude/commands/aioson/agent/pm.md +5 -5
  922. package/template/.claude/commands/aioson/agent/product.md +5 -5
  923. package/template/.claude/commands/aioson/agent/profiler-enricher.md +5 -5
  924. package/template/.claude/commands/aioson/agent/profiler-forge.md +5 -5
  925. package/template/.claude/commands/aioson/agent/profiler-researcher.md +5 -5
  926. package/template/.claude/commands/aioson/agent/qa.md +5 -5
  927. package/template/.claude/commands/aioson/agent/setup.md +5 -5
  928. package/template/.claude/commands/aioson/agent/sheldon.md +5 -5
  929. package/template/.claude/commands/aioson/agent/site-forge.md +5 -5
  930. package/template/.claude/commands/aioson/agent/squad.md +5 -5
  931. package/template/.claude/commands/aioson/agent/tester.md +5 -5
  932. package/template/.claude/commands/aioson/agent/ux-ui.md +5 -5
  933. package/template/.claude/commands/aioson/agent/validator.md +5 -0
  934. package/template/.gemini/GEMINI.md +13 -13
  935. package/template/.gemini/commands/aios-analyst.toml +7 -4
  936. package/template/.gemini/commands/aios-architect.toml +8 -7
  937. package/template/.gemini/commands/aios-committer.toml +7 -0
  938. package/template/.gemini/commands/aios-copywriter.toml +7 -0
  939. package/template/.gemini/commands/aios-cypher.toml +7 -0
  940. package/template/.gemini/commands/aios-dev.toml +9 -8
  941. package/template/.gemini/commands/aios-deyvin.toml +7 -6
  942. package/template/.gemini/commands/aios-discover.toml +6 -0
  943. package/template/.gemini/commands/aios-discovery-design-doc.toml +7 -4
  944. package/template/.gemini/commands/aios-genome.toml +7 -0
  945. package/template/.gemini/commands/aios-neo.toml +6 -4
  946. package/template/.gemini/commands/aios-orache.toml +7 -0
  947. package/template/.gemini/commands/aios-orchestrator.toml +9 -8
  948. package/template/.gemini/commands/aios-pair.toml +7 -6
  949. package/template/.gemini/commands/aios-pm.toml +9 -8
  950. package/template/.gemini/commands/aios-product.toml +6 -4
  951. package/template/.gemini/commands/aios-qa.toml +7 -6
  952. package/template/.gemini/commands/aios-setup.toml +6 -3
  953. package/template/.gemini/commands/aios-sheldon.toml +7 -0
  954. package/template/.gemini/commands/aios-site-forge.toml +7 -0
  955. package/template/.gemini/commands/aios-squad.toml +7 -0
  956. package/template/.gemini/commands/aios-tester.toml +7 -6
  957. package/template/.gemini/commands/aios-ux-ui.toml +9 -8
  958. package/template/.gemini/commands/aios-validator.toml +7 -0
  959. package/template/AGENTS.md +184 -172
  960. package/template/CLAUDE.md +98 -93
  961. package/template/OPENCODE.md +35 -34
  962. package/template/aioson-models.json +40 -40
  963. package/docs/en/i18n.md +0 -52
  964. package/docs/en/schemas/parallel-status.schema.json +0 -94
  965. package/template/.aioson/genomes/copywriting.md +0 -204
  966. package/template/.aioson/locales/en/agents/analyst.md +0 -244
  967. package/template/.aioson/locales/en/agents/architect.md +0 -245
  968. package/template/.aioson/locales/en/agents/dev.md +0 -397
  969. package/template/.aioson/locales/en/agents/deyvin.md +0 -137
  970. package/template/.aioson/locales/en/agents/discovery-design-doc.md +0 -27
  971. package/template/.aioson/locales/en/agents/genome.md +0 -212
  972. package/template/.aioson/locales/en/agents/neo.md +0 -8
  973. package/template/.aioson/locales/en/agents/orache.md +0 -6
  974. package/template/.aioson/locales/en/agents/orchestrator.md +0 -189
  975. package/template/.aioson/locales/en/agents/pair.md +0 -5
  976. package/template/.aioson/locales/en/agents/pm.md +0 -84
  977. package/template/.aioson/locales/en/agents/product.md +0 -378
  978. package/template/.aioson/locales/en/agents/profiler-enricher.md +0 -5
  979. package/template/.aioson/locales/en/agents/profiler-forge.md +0 -5
  980. package/template/.aioson/locales/en/agents/profiler-researcher.md +0 -5
  981. package/template/.aioson/locales/en/agents/qa.md +0 -270
  982. package/template/.aioson/locales/en/agents/setup.md +0 -421
  983. package/template/.aioson/locales/en/agents/sheldon.md +0 -455
  984. package/template/.aioson/locales/en/agents/squad.md +0 -449
  985. package/template/.aioson/locales/en/agents/tester.md +0 -6
  986. package/template/.aioson/locales/en/agents/ux-ui.md +0 -668
  987. package/template/.aioson/locales/es/agents/analyst.md +0 -225
  988. package/template/.aioson/locales/es/agents/architect.md +0 -245
  989. package/template/.aioson/locales/es/agents/dev.md +0 -370
  990. package/template/.aioson/locales/es/agents/deyvin.md +0 -99
  991. package/template/.aioson/locales/es/agents/discovery-design-doc.md +0 -21
  992. package/template/.aioson/locales/es/agents/genome.md +0 -104
  993. package/template/.aioson/locales/es/agents/neo.md +0 -50
  994. package/template/.aioson/locales/es/agents/orache.md +0 -105
  995. package/template/.aioson/locales/es/agents/orchestrator.md +0 -194
  996. package/template/.aioson/locales/es/agents/pair.md +0 -7
  997. package/template/.aioson/locales/es/agents/pm.md +0 -90
  998. package/template/.aioson/locales/es/agents/product.md +0 -372
  999. package/template/.aioson/locales/es/agents/profiler-enricher.md +0 -7
  1000. package/template/.aioson/locales/es/agents/profiler-forge.md +0 -7
  1001. package/template/.aioson/locales/es/agents/profiler-researcher.md +0 -7
  1002. package/template/.aioson/locales/es/agents/qa.md +0 -198
  1003. package/template/.aioson/locales/es/agents/setup.md +0 -405
  1004. package/template/.aioson/locales/es/agents/sheldon.md +0 -309
  1005. package/template/.aioson/locales/es/agents/squad.md +0 -532
  1006. package/template/.aioson/locales/es/agents/tester.md +0 -9
  1007. package/template/.aioson/locales/es/agents/ux-ui.md +0 -212
  1008. package/template/.aioson/locales/fr/agents/analyst.md +0 -225
  1009. package/template/.aioson/locales/fr/agents/architect.md +0 -245
  1010. package/template/.aioson/locales/fr/agents/dev.md +0 -370
  1011. package/template/.aioson/locales/fr/agents/deyvin.md +0 -99
  1012. package/template/.aioson/locales/fr/agents/discovery-design-doc.md +0 -21
  1013. package/template/.aioson/locales/fr/agents/genome.md +0 -104
  1014. package/template/.aioson/locales/fr/agents/neo.md +0 -50
  1015. package/template/.aioson/locales/fr/agents/orache.md +0 -106
  1016. package/template/.aioson/locales/fr/agents/orchestrator.md +0 -194
  1017. package/template/.aioson/locales/fr/agents/pair.md +0 -7
  1018. package/template/.aioson/locales/fr/agents/pm.md +0 -90
  1019. package/template/.aioson/locales/fr/agents/product.md +0 -372
  1020. package/template/.aioson/locales/fr/agents/profiler-enricher.md +0 -7
  1021. package/template/.aioson/locales/fr/agents/profiler-forge.md +0 -7
  1022. package/template/.aioson/locales/fr/agents/profiler-researcher.md +0 -7
  1023. package/template/.aioson/locales/fr/agents/qa.md +0 -198
  1024. package/template/.aioson/locales/fr/agents/setup.md +0 -405
  1025. package/template/.aioson/locales/fr/agents/sheldon.md +0 -309
  1026. package/template/.aioson/locales/fr/agents/squad.md +0 -532
  1027. package/template/.aioson/locales/fr/agents/tester.md +0 -9
  1028. package/template/.aioson/locales/fr/agents/ux-ui.md +0 -212
  1029. package/template/.aioson/locales/pt-BR/agents/analyst.md +0 -319
  1030. package/template/.aioson/locales/pt-BR/agents/architect.md +0 -284
  1031. package/template/.aioson/locales/pt-BR/agents/dev.md +0 -483
  1032. package/template/.aioson/locales/pt-BR/agents/deyvin.md +0 -184
  1033. package/template/.aioson/locales/pt-BR/agents/discovery-design-doc.md +0 -198
  1034. package/template/.aioson/locales/pt-BR/agents/genome.md +0 -297
  1035. package/template/.aioson/locales/pt-BR/agents/neo.md +0 -208
  1036. package/template/.aioson/locales/pt-BR/agents/orache.md +0 -137
  1037. package/template/.aioson/locales/pt-BR/agents/orchestrator.md +0 -324
  1038. package/template/.aioson/locales/pt-BR/agents/pair.md +0 -5
  1039. package/template/.aioson/locales/pt-BR/agents/pm.md +0 -182
  1040. package/template/.aioson/locales/pt-BR/agents/product.md +0 -466
  1041. package/template/.aioson/locales/pt-BR/agents/profiler-enricher.md +0 -5
  1042. package/template/.aioson/locales/pt-BR/agents/profiler-forge.md +0 -5
  1043. package/template/.aioson/locales/pt-BR/agents/profiler-researcher.md +0 -5
  1044. package/template/.aioson/locales/pt-BR/agents/qa.md +0 -300
  1045. package/template/.aioson/locales/pt-BR/agents/setup.md +0 -533
  1046. package/template/.aioson/locales/pt-BR/agents/sheldon.md +0 -323
  1047. package/template/.aioson/locales/pt-BR/agents/squad.md +0 -1330
  1048. package/template/.aioson/locales/pt-BR/agents/tester.md +0 -449
  1049. package/template/.aioson/locales/pt-BR/agents/ux-ui.md +0 -669
@@ -1,1767 +1,1767 @@
1
- 'use strict';
2
-
3
- const fs = require('node:fs/promises');
4
- const path = require('node:path');
5
- const { ensureDir, exists, nowStamp, toRelativeSafe } = require('../utils');
6
- const { openRuntimeDb, upsertSquadManifest } = require('../runtime-store');
7
- const {
8
- attachBindingsToExecutors,
9
- flattenGenomeBindings,
10
- mergeGenomeBindings,
11
- normalizeBinding,
12
- normalizeGenomeBindings,
13
- resolveExecutorGenomes
14
- } = require('../genomes/bindings');
15
-
16
- function sanitizeSegment(value, fallback) {
17
- const normalized = String(value || fallback || '')
18
- .trim()
19
- .toLowerCase()
20
- .replace(/[^a-z0-9._-]+/g, '-')
21
- .replace(/^-+|-+$/g, '')
22
- .slice(0, 120);
23
-
24
- return normalized || fallback;
25
- }
26
-
27
- function cloudImportsRoot(projectDir) {
28
- return path.join(projectDir, '.aioson', 'cloud-imports');
29
- }
30
-
31
- function squadImportFilePath(projectDir, slug, versionNumber) {
32
- const safeSlug = sanitizeSegment(slug, 'squad');
33
- const safeVersion = sanitizeSegment(versionNumber, 'latest');
34
- return path.join(cloudImportsRoot(projectDir), 'squads', safeSlug, `${safeVersion}.json`);
35
- }
36
-
37
- function genomeImportFilePath(projectDir, slug, versionNumber) {
38
- const safeSlug = sanitizeSegment(slug, 'genome');
39
- const safeVersion = sanitizeSegment(versionNumber, 'latest');
40
- return path.join(cloudImportsRoot(projectDir), 'genomes', safeSlug, `${safeVersion}.json`);
41
- }
42
-
43
- function historyImportFilePath(projectDir, slug, versionNumber) {
44
- const safeSlug = sanitizeSegment(slug, 'squad');
45
- const safeVersion = sanitizeSegment(versionNumber, 'latest');
46
- return path.join(
47
- cloudImportsRoot(projectDir),
48
- 'history',
49
- 'squads',
50
- safeSlug,
51
- `${safeVersion}--${nowStamp()}.json`
52
- );
53
- }
54
-
55
- function genomeHistoryImportFilePath(projectDir, slug, versionNumber) {
56
- const safeSlug = sanitizeSegment(slug, 'genome');
57
- const safeVersion = sanitizeSegment(versionNumber, 'latest');
58
- return path.join(
59
- cloudImportsRoot(projectDir),
60
- 'history',
61
- 'genomes',
62
- safeSlug,
63
- `${safeVersion}--${nowStamp()}.json`
64
- );
65
- }
66
-
67
- function installedRoot(projectDir) {
68
- return path.join(cloudImportsRoot(projectDir), 'installed');
69
- }
70
-
71
- function installedManifestPath(projectDir, slug) {
72
- return path.join(installedRoot(projectDir), 'squads', sanitizeSegment(slug, 'squad'), 'manifest.json');
73
- }
74
-
75
- function installedGenomeManifestPath(projectDir, slug) {
76
- return path.join(installedRoot(projectDir), 'genomes', sanitizeSegment(slug, 'genome'), 'manifest.json');
77
- }
78
-
79
- function localSquadPackageDir(projectDir, slug) {
80
- return path.join(projectDir, '.aioson', 'squads', sanitizeSegment(slug, 'squad'));
81
- }
82
-
83
- function localSquadSummaryPath(projectDir, slug) {
84
- return path.join(localSquadPackageDir(projectDir, slug), 'squad.md');
85
- }
86
-
87
- function localLegacySquadMetadataPath(projectDir, slug) {
88
- return path.join(projectDir, '.aioson', 'squads', `${sanitizeSegment(slug, 'squad')}.md`);
89
- }
90
-
91
- function localSquadAgentsDir(projectDir, slug) {
92
- return path.join(localSquadPackageDir(projectDir, slug), 'agents');
93
- }
94
-
95
- function localSquadSkillsDir(projectDir, slug) {
96
- return path.join(localSquadPackageDir(projectDir, slug), 'skills');
97
- }
98
-
99
- function localSquadTemplatesDir(projectDir, slug) {
100
- return path.join(localSquadPackageDir(projectDir, slug), 'templates');
101
- }
102
-
103
- function localSquadDocsDir(projectDir, slug) {
104
- return path.join(localSquadPackageDir(projectDir, slug), 'docs');
105
- }
106
-
107
- function localLegacySquadAgentsDir(projectDir, slug) {
108
- return path.join(projectDir, 'agents', sanitizeSegment(slug, 'squad'));
109
- }
110
-
111
- function localSquadOutputDir(projectDir, slug) {
112
- return path.join(projectDir, 'output', sanitizeSegment(slug, 'squad'));
113
- }
114
-
115
- function localSquadLogsDir(projectDir, slug) {
116
- return path.join(projectDir, 'aioson-logs', sanitizeSegment(slug, 'squad'));
117
- }
118
-
119
- function localSquadMediaDir(projectDir, slug) {
120
- return path.join(projectDir, 'media', sanitizeSegment(slug, 'squad'));
121
- }
122
-
123
- function localSquadTextManifestPath(projectDir, slug) {
124
- return path.join(localSquadAgentsDir(projectDir, slug), 'agents.md');
125
- }
126
-
127
- function localSquadJsonManifestPath(projectDir, slug) {
128
- return path.join(localSquadPackageDir(projectDir, slug), 'squad.manifest.json');
129
- }
130
-
131
- function localSquadDesignDocPath(projectDir, slug) {
132
- return path.join(localSquadDocsDir(projectDir, slug), 'design-doc.md');
133
- }
134
-
135
- function localSquadReadinessPath(projectDir, slug) {
136
- return path.join(localSquadDocsDir(projectDir, slug), 'readiness.md');
137
- }
138
-
139
- function localSquadRulesDocPath(projectDir, slug) {
140
- return path.join(localSquadDocsDir(projectDir, slug), 'squad-rules.md');
141
- }
142
-
143
- function localSquadOutputContractsPath(projectDir, slug) {
144
- return path.join(localSquadDocsDir(projectDir, slug), 'output-contracts.md');
145
- }
146
-
147
- function localLegacySquadTextManifestPath(projectDir, slug) {
148
- return path.join(localLegacySquadAgentsDir(projectDir, slug), 'agents.md');
149
- }
150
-
151
- function localLegacySquadJsonManifestPath(projectDir, slug) {
152
- return path.join(localLegacySquadAgentsDir(projectDir, slug), 'squad.manifest.json');
153
- }
154
-
155
- function localLegacySquadDesignDocPath(projectDir, slug) {
156
- return path.join(localLegacySquadAgentsDir(projectDir, slug), 'design-doc.md');
157
- }
158
-
159
- function localLegacySquadReadinessPath(projectDir, slug) {
160
- return path.join(localLegacySquadAgentsDir(projectDir, slug), 'readiness.md');
161
- }
162
-
163
- function localGenomeFilePath(projectDir, slug) {
164
- return path.join(projectDir, '.aioson', 'genomes', `${sanitizeSegment(slug, 'genome')}.md`);
165
- }
166
-
167
- function findPrimaryHeading(markdown, fallback) {
168
- const match = String(markdown || '').match(/^#\s+(.+)$/m);
169
- return match ? String(match[1]).trim() : fallback;
170
- }
171
-
172
- function firstParagraph(markdown) {
173
- const blocks = String(markdown || '')
174
- .split(/\n\s*\n/)
175
- .map((block) => block.trim())
176
- .filter(Boolean);
177
-
178
- for (const block of blocks) {
179
- if (block.startsWith('#')) continue;
180
- return block.replace(/\n+/g, ' ').trim();
181
- }
182
-
183
- return null;
184
- }
185
-
186
- function extractField(content, ...labels) {
187
- for (const label of labels) {
188
- const regex = new RegExp(`^(?:${label}):\\s*(.+)$`, 'im');
189
- const match = String(content || '').match(regex);
190
- if (match) return String(match[1]).trim();
191
- }
192
- return null;
193
- }
194
-
195
- function parseListSection(content, heading) {
196
- const lines = String(content || '').split(/\r?\n/);
197
- const startIndex = lines.findIndex((line) => line.trim() === `${heading}:`);
198
- if (startIndex === -1) return [];
199
-
200
- const values = [];
201
- for (let i = startIndex + 1; i < lines.length; i += 1) {
202
- const line = lines[i];
203
- if (/^\S.+:$/.test(line.trim())) break;
204
- const match = line.match(/^\s*-\s+(.+?)\s*$/);
205
- if (match) values.push(match[1].trim());
206
- }
207
- return values;
208
- }
209
-
210
- function normalizeRel(relPath) {
211
- return String(relPath || '')
212
- .replace(/\\/g, '/')
213
- .replace(/^\.\//, '')
214
- .replace(/\/+$/, '');
215
- }
216
-
217
- function parseAgentGenomeEntry(entry) {
218
- const text = String(entry || '').trim();
219
- const index = text.indexOf(':');
220
- if (index === -1) return null;
221
- return {
222
- agentSlug: normalizeAgentSlug(text.slice(0, index).trim()),
223
- genomePath: text.slice(index + 1).trim()
224
- };
225
- }
226
-
227
- function guessAgentBody(agent) {
228
- return (
229
- agent.promptText ||
230
- agent.prompt ||
231
- agent.content ||
232
- agent.markdown ||
233
- agent.body ||
234
- agent.fileContent ||
235
- null
236
- );
237
- }
238
-
239
- function normalizeAgentSlug(value) {
240
- return sanitizeSegment(String(value || '').replace(/^@/, ''), 'agent');
241
- }
242
-
243
- function normalizeAgentsManifest(agentsManifestJson) {
244
- if (!agentsManifestJson) return [];
245
-
246
- const source =
247
- Array.isArray(agentsManifestJson)
248
- ? agentsManifestJson
249
- : Array.isArray(agentsManifestJson.agents)
250
- ? agentsManifestJson.agents
251
- : typeof agentsManifestJson === 'object'
252
- ? Object.entries(agentsManifestJson).map(([key, value]) => ({
253
- slug: key,
254
- ...(value && typeof value === 'object' ? value : { content: value })
255
- }))
256
- : [];
257
-
258
- return source
259
- .map((agent, index) => {
260
- const slug = normalizeAgentSlug(agent.slug || agent.name || agent.id || `agent-${index + 1}`);
261
- const title = agent.name || agent.title || agent.roleTitle || slug;
262
- const description = agent.description || agent.summary || agent.role || null;
263
- const body = guessAgentBody(agent);
264
- return {
265
- slug,
266
- title: String(title),
267
- description: description ? String(description) : null,
268
- body: body ? String(body) : null
269
- };
270
- })
271
- .filter((agent) => Boolean(agent.slug));
272
- }
273
-
274
- async function loadLocalSquadManifest(projectDir, slug) {
275
- const preferredPath = localSquadJsonManifestPath(projectDir, slug);
276
- const manifestPath = (await exists(preferredPath))
277
- ? preferredPath
278
- : localLegacySquadJsonManifestPath(projectDir, slug);
279
- if (!(await exists(manifestPath))) {
280
- return null;
281
- }
282
-
283
- const raw = await fs.readFile(manifestPath, 'utf8').catch(() => null);
284
- if (!raw) return null;
285
-
286
- try {
287
- return JSON.parse(raw);
288
- } catch {
289
- return null;
290
- }
291
- }
292
-
293
- function deriveFallbackSkills(snapshot) {
294
- const domain = String(snapshot?.squad?.name || snapshot?.squad?.slug || 'squad');
295
- return [
296
- {
297
- slug: 'structured-domain-output',
298
- title: 'Structured domain output',
299
- description: `Produce structured outputs for ${domain}.`
300
- },
301
- {
302
- slug: 'critical-synthesis',
303
- title: 'Critical synthesis',
304
- description: 'Consolidate specialist reasoning into a practical next step.'
305
- }
306
- ];
307
- }
308
-
309
- function deriveFallbackMcps(snapshot) {
310
- const items = [{ slug: 'filesystem', required: true, purpose: 'Persist local drafts, manifests, outputs, logs, and media.' }];
311
- if ((snapshot?.squad?.visibility || '').toUpperCase() === 'FREE') {
312
- items.push({ slug: 'web-search', required: false, purpose: 'Optional external research when the task requires current references.' });
313
- }
314
- return items;
315
- }
316
-
317
- function normalizeContentBlueprints(value) {
318
- if (!Array.isArray(value)) return [];
319
-
320
- return value
321
- .map((item, index) => {
322
- if (!item || typeof item !== 'object') return null;
323
-
324
- const slug = sanitizeSegment(item.slug || `blueprint-${index + 1}`, `blueprint-${index + 1}`);
325
- const contentType = String(item.contentType || 'content').trim() || 'content';
326
- const layoutType = String(item.layoutType || 'document').trim() || 'document';
327
- const description = item.description ? String(item.description).trim() : null;
328
- const sections = Array.isArray(item.sections)
329
- ? item.sections
330
- .map((section, sectionIndex) => {
331
- if (!section || typeof section !== 'object') return null;
332
-
333
- const key = sanitizeSegment(section.key || `section-${sectionIndex + 1}`, `section-${sectionIndex + 1}`);
334
- const label = String(section.label || section.key || key).trim();
335
- const blockTypes = Array.isArray(section.blockTypes)
336
- ? section.blockTypes.map((blockType) => String(blockType).trim()).filter(Boolean)
337
- : [];
338
-
339
- if (!key || !label) return null;
340
- return { key, label, blockTypes };
341
- })
342
- .filter(Boolean)
343
- : [];
344
-
345
- return { slug, contentType, layoutType, description, sections };
346
- })
347
- .filter(Boolean);
348
- }
349
-
350
- function buildBindingsFromAppliedGenomes(appliedGenomes = []) {
351
- const draft = {
352
- squad: [],
353
- executors: {}
354
- };
355
-
356
- for (const item of appliedGenomes) {
357
- const binding = normalizeBinding({
358
- slug: item?.genome?.slug,
359
- type: item?.genome?.type || item?.version?.manifestJson?.type,
360
- source: item?.genome?.sourceKind ? String(item.genome.sourceKind).toLowerCase() : 'cloud',
361
- priority: item?.priority,
362
- version: item?.version?.versionNumber,
363
- evidenceMode:
364
- item?.version?.manifestJson?.evidenceMode ||
365
- item?.version?.manifestJson?.evidence_mode ||
366
- item?.genome?.evidenceMode
367
- });
368
- if (!binding) continue;
369
-
370
- if (String(item?.scopeType || 'SQUAD').toUpperCase() === 'SQUAD') {
371
- draft.squad.push(binding);
372
- continue;
373
- }
374
-
375
- const executorSlug = normalizeAgentSlug(item?.agentSlug);
376
- if (!executorSlug) continue;
377
- draft.executors[executorSlug] = draft.executors[executorSlug] || [];
378
- draft.executors[executorSlug].push(binding);
379
- }
380
-
381
- return normalizeGenomeBindings(draft);
382
- }
383
-
384
- function buildLocalSquadManifest(snapshot, agents) {
385
- const source = snapshot?.version?.manifestJson && typeof snapshot.version.manifestJson === 'object'
386
- ? snapshot.version.manifestJson
387
- : {};
388
- const slug = sanitizeSegment(snapshot.squad.slug, 'squad');
389
- const explicitMode = typeof source.mode === 'string' ? source.mode : null;
390
- const mode = String(explicitMode || (source.storagePolicy?.primary === 'files' ? 'builder' : 'content')).trim();
391
- const sourceContext = source.context && typeof source.context === 'object' ? source.context : {};
392
- const packageRoot = `.aioson/squads/${slug}`;
393
- const sourceBindings = mergeGenomeBindings({
394
- blueprintBindings: source.genomeBindings,
395
- manifestBindings: source.genomeBindings || source.genomes,
396
- legacyExecutors: source.executors
397
- });
398
- const importedBindings = buildBindingsFromAppliedGenomes(snapshot.appliedGenomes || []);
399
- const genomeBindings = mergeGenomeBindings({
400
- blueprintBindings: sourceBindings,
401
- manifestBindings: importedBindings
402
- });
403
- const executorSource = Array.isArray(source.executors) && source.executors.length > 0
404
- ? source.executors
405
- : agents.map((agent) => ({
406
- slug: agent.slug,
407
- title: agent.title,
408
- role: agent.description || (agent.slug === 'orquestrador' ? 'Coordinates the squad and publishes the final HTML.' : null),
409
- file: `${packageRoot}/agents/${agent.slug}.md`,
410
- skills: agent.slug === 'orquestrador' ? [] : ['structured-domain-output'],
411
- genomes: resolveExecutorGenomes(agent.slug, genomeBindings)
412
- }));
413
- const executors = attachBindingsToExecutors(executorSource, genomeBindings);
414
-
415
- return {
416
- schemaVersion: String(source.schemaVersion || snapshot?.version?.schemaVersion || '1.0.0'),
417
- packageVersion: String(source.packageVersion || snapshot?.version?.versionNumber || '1.0.0'),
418
- slug,
419
- name: String(source.name || snapshot.squad.name),
420
- mode,
421
- mission: String(source.mission || snapshot.squad.description || `Operate the ${snapshot.squad.name} squad.`),
422
- goal: String(source.goal || snapshot.squad.goal || snapshot.squad.description || 'Imported from AIOSON Cloud'),
423
- visibility: String(source.visibility || String(snapshot.squad.visibility || 'PRIVATE').toLowerCase()),
424
- aiosLiteCompatibility: String(
425
- source.aiosLiteCompatibility ||
426
- snapshot?.version?.compatibilityMin ||
427
- '^1.1.0'
428
- ),
429
- rules: {
430
- outputsDir: `output/${slug}`,
431
- logsDir: `aioson-logs/${slug}`,
432
- mediaDir: `media/${slug}`,
433
- reviewPolicy: Array.isArray(source?.rules?.reviewPolicy)
434
- ? source.rules.reviewPolicy
435
- : ['clarity', 'density', 'consistency', 'next-step']
436
- },
437
- storagePolicy:
438
- source.storagePolicy && typeof source.storagePolicy === 'object'
439
- ? source.storagePolicy
440
- : {
441
- primary: mode === 'builder' ? 'files' : 'sqlite',
442
- artifacts: mode === 'builder' ? 'files+sqlite' : 'sqlite-json',
443
- exports: { html: true, markdown: true, json: true }
444
- },
445
- package: {
446
- rootDir: packageRoot,
447
- agentsDir: `${packageRoot}/agents`,
448
- skillsDir: `${packageRoot}/skills`,
449
- templatesDir: `${packageRoot}/templates`,
450
- docsDir: `${packageRoot}/docs`
451
- },
452
- baseRoles: Array.isArray(source.baseRoles) && source.baseRoles.length > 0
453
- ? source.baseRoles
454
- : ['orchestrator', 'discovery-lead', 'design-doc-lead', 'planner', 'implementer', 'reviewer', 'docs-maintainer'],
455
- skills: Array.isArray(source.skills) && source.skills.length > 0 ? source.skills : deriveFallbackSkills(snapshot),
456
- mcps: Array.isArray(source.mcps) && source.mcps.length > 0 ? source.mcps : deriveFallbackMcps(snapshot),
457
- subagents: source.subagents && typeof source.subagents === 'object'
458
- ? source.subagents
459
- : {
460
- allowed: true,
461
- when: ['broad research', 'comparison', 'large-context summarization', 'parallel analysis']
462
- },
463
- contentBlueprints: normalizeContentBlueprints(source.contentBlueprints),
464
- context: {
465
- mode: String(
466
- sourceContext.mode ||
467
- (snapshot?.version?.designDocMarkdown && /feature mode|modo feature/i.test(snapshot.version.designDocMarkdown)
468
- ? 'feature'
469
- : 'project')
470
- ),
471
- summary: String(
472
- sourceContext.summary ||
473
- source.goal ||
474
- snapshot.squad.goal ||
475
- snapshot.squad.description ||
476
- 'Imported squad context.'
477
- ),
478
- designDocPath: `${packageRoot}/docs/design-doc.md`,
479
- readinessPath: `${packageRoot}/docs/readiness.md`,
480
- docsPackage: Array.isArray(sourceContext.docsPackage)
481
- ? sourceContext.docsPackage
482
- : ['project.context.md', 'design-doc.md', 'readiness.md'],
483
- readiness: sourceContext.readiness && typeof sourceContext.readiness === 'object'
484
- ? sourceContext.readiness
485
- : null
486
- },
487
- executors,
488
- genomes: genomeBindings,
489
- genomeBindings
490
- };
491
- }
492
-
493
- function buildSquadTextManifest(snapshot, manifest) {
494
- const lines = [
495
- `# Squad ${manifest.name}`,
496
- '',
497
- '## Mission',
498
- manifest.mission,
499
- '',
500
- '## Does',
501
- `- Deliver outputs for the domain: ${snapshot.squad.name}`,
502
- `- Target goal: ${manifest.goal}`,
503
- '- Coordinate specialists through the local orchestrator',
504
- '',
505
- '## Does not do',
506
- '- Replace the AIOSON official agents',
507
- '- Use subagents as a substitute for permanent executors or skills',
508
- '',
509
- '## Permanent executors'
510
- ];
511
-
512
- for (const executor of manifest.executors || []) {
513
- lines.push(`- @${executor.slug} — ${executor.role || executor.title || 'Specialist executor'}`);
514
- }
515
-
516
- lines.push('', '## Squad skills');
517
- for (const skill of manifest.skills || []) {
518
- lines.push(`- ${skill.slug} — ${skill.description || skill.title || 'Reusable capability'}`);
519
- }
520
-
521
- lines.push('', '## Squad MCPs');
522
- for (const mcp of manifest.mcps || []) {
523
- lines.push(`- ${mcp.slug} — ${mcp.purpose || 'External integration'}`);
524
- }
525
-
526
- lines.push(
527
- '',
528
- '## Subagent policy',
529
- '- Use subagents only for isolated investigation, broad reading, comparison, or parallel work.',
530
- '- Do not use subagents as a substitute for permanent executors or reusable skills.',
531
- '',
532
- '## Outputs and review',
533
- `- Drafts: \`output/${manifest.slug}/\``,
534
- `- Final HTML: \`output/${manifest.slug}/{session-id}.html\``,
535
- `- Logs: \`aioson-logs/${manifest.slug}/\``,
536
- `- Media: \`media/${manifest.slug}/\``,
537
- `- Package root: \`.aioson/squads/${manifest.slug}/\``,
538
- `- Design doc: \`.aioson/squads/${manifest.slug}/docs/design-doc.md\``,
539
- `- Readiness: \`.aioson/squads/${manifest.slug}/docs/readiness.md\``,
540
- '- Final outputs should include recommendation, reasoning, tradeoff, and next step.'
541
- );
542
-
543
- return `${lines.join('\n')}\n`;
544
- }
545
-
546
- function buildAgentStub(snapshot, agent) {
547
- const lines = [
548
- `# ${agent.title}`,
549
- '',
550
- '> Imported from AIOSON Cloud.',
551
- '',
552
- '## Origin',
553
- '',
554
- `- Squad: ${snapshot.squad.name}`,
555
- `- Slug: ${snapshot.squad.slug}`,
556
- `- Version: ${snapshot.version.versionNumber}`,
557
- `- Owner: ${snapshot.squad.ownerUsername}`
558
- ];
559
-
560
- if (agent.description) {
561
- lines.push('', '## Role', '', agent.description);
562
- }
563
-
564
- lines.push(
565
- '',
566
- '## Import Notice',
567
- '',
568
- 'The cloud snapshot did not include a full prompt body for this agent.',
569
- 'Regenerate or enrich this agent locally before using it as a production specialist.'
570
- );
571
-
572
- return `${lines.join('\n')}\n`;
573
- }
574
-
575
- function buildSquadMetadata(snapshot, options = {}) {
576
- const slug = sanitizeSegment(snapshot.squad.slug, 'squad');
577
- const installedAt = new Date().toISOString();
578
- const packageRoot = `.aioson/squads/${slug}`;
579
- const lines = [
580
- `Squad: ${snapshot.squad.name}`,
581
- `Mode: ${snapshot?.version?.manifestJson?.mode || 'CloudImport'}`,
582
- `Goal: ${snapshot.squad.goal || snapshot.squad.description || 'Imported from AIOSON Cloud'}`,
583
- `Package: ${packageRoot}/`,
584
- `Agents: ${packageRoot}/agents/`,
585
- `Skills: ${packageRoot}/skills/`,
586
- `Templates: ${packageRoot}/templates/`,
587
- `Docs: ${packageRoot}/docs/`,
588
- `Output: output/${slug}/`,
589
- `Logs: aioson-logs/${slug}/`,
590
- `Media: media/${slug}/`,
591
- `DesignDoc: ${packageRoot}/docs/design-doc.md`,
592
- `Readiness: ${packageRoot}/docs/readiness.md`,
593
- `LatestSession: output/${slug}/latest.html`,
594
- `SourceUrl: ${options.sourceUrl || '—'}`,
595
- `SourceVersion: ${snapshot.version.versionNumber}`,
596
- `ImportedAt: ${installedAt}`,
597
- '',
598
- 'Genomes:'
599
- ];
600
-
601
- const shared = Array.isArray(snapshot.appliedGenomes)
602
- ? snapshot.appliedGenomes.filter((item) => String(item.scopeType || 'SQUAD').toUpperCase() === 'SQUAD')
603
- : [];
604
- for (const genome of shared) {
605
- lines.push(`- .aioson/genomes/${sanitizeSegment(genome.genome.slug, 'genome')}.md`);
606
- }
607
-
608
- lines.push('', 'AgentGenomes:');
609
- const scoped = Array.isArray(snapshot.appliedGenomes)
610
- ? snapshot.appliedGenomes.filter(
611
- (item) => String(item.scopeType || 'SQUAD').toUpperCase() !== 'SQUAD' && item.agentSlug
612
- )
613
- : [];
614
-
615
- for (const genome of scoped) {
616
- lines.push(
617
- `- ${normalizeAgentSlug(genome.agentSlug)}: .aioson/genomes/${sanitizeSegment(genome.genome.slug, 'genome')}.md`
618
- );
619
- }
620
-
621
- return `${lines.join('\n')}\n`;
622
- }
623
-
624
- function buildInstalledManifest(snapshot, sourceUrl, agents) {
625
- return {
626
- kind: 'aiosforge.local-installed-squad',
627
- installVersion: 1,
628
- installedAt: new Date().toISOString(),
629
- sourceUrl,
630
- squad: snapshot.squad,
631
- version: {
632
- versionNumber: snapshot.version.versionNumber,
633
- compatibilityMin: snapshot.version.compatibilityMin,
634
- compatibilityMax: snapshot.version.compatibilityMax,
635
- schemaVersion: snapshot.version.schemaVersion
636
- },
637
- packageRoot: `.aioson/squads/${sanitizeSegment(snapshot.squad.slug, 'squad')}`,
638
- agents: agents.map((agent) => ({
639
- slug: agent.slug,
640
- title: agent.title,
641
- hasBody: Boolean(agent.body)
642
- })),
643
- appliedGenomes: snapshot.appliedGenomes || []
644
- };
645
- }
646
-
647
- function buildSquadDesignDoc(snapshot, manifest) {
648
- if (snapshot?.version?.designDocMarkdown) {
649
- const content = String(snapshot.version.designDocMarkdown);
650
- return content.endsWith('\n') ? content : `${content}\n`;
651
- }
652
-
653
- return [
654
- `# Design Doc - ${manifest.name}`,
655
- '',
656
- '## Context and motivation',
657
- snapshot.squad.description || manifest.goal || 'Imported squad context.',
658
- '',
659
- '## Objective',
660
- manifest.goal || 'Operate the imported squad for the target domain.',
661
- '',
662
- '## Scope',
663
- '- Materialize the squad locally with manifest, executors, outputs, logs, and media.',
664
- '- Preserve the operational and cognitive blueprint published in the cloud.',
665
- '',
666
- '## Out of scope',
667
- '- Rewriting imported executors automatically beyond the published snapshot.',
668
- '- Inventing undocumented MCPs or genomes.'
669
- ].join('\n') + '\n';
670
- }
671
-
672
- function buildSquadReadiness(snapshot, manifest) {
673
- if (snapshot?.version?.readinessMarkdown) {
674
- const content = String(snapshot.version.readinessMarkdown);
675
- return content.endsWith('\n') ? content : `${content}\n`;
676
- }
677
-
678
- const readiness = manifest?.context?.readiness && typeof manifest.context.readiness === 'object'
679
- ? manifest.context.readiness
680
- : {
681
- level: 'medium',
682
- totalScore: 15,
683
- maxScore: 25
684
- };
685
-
686
- return [
687
- `# Readiness - ${manifest.name}`,
688
- '',
689
- `- Readiness score total: ${readiness.totalScore ?? 15}`,
690
- `- Readiness score maximo: ${readiness.maxScore ?? 25}`,
691
- `- Readiness level: ${readiness.level || 'medium'}`,
692
- '',
693
- '## What is already clear',
694
- '- The squad structure and executors are defined in the imported snapshot.',
695
- '- Outputs, logs, media, and genomes can be materialized locally.',
696
- '',
697
- '## What is still missing',
698
- '- Local project-specific refinements after import.',
699
- '- Any additional feature context not present in the published snapshot.'
700
- ].join('\n') + '\n';
701
- }
702
-
703
- function buildSkillMarkdown(skill) {
704
- const title = String(skill?.title || skill?.slug || 'Skill').trim();
705
- const description = skill?.description ? String(skill.description).trim() : 'Reusable squad capability.';
706
- return [`# ${title}`, '', description, ''].join('\n');
707
- }
708
-
709
- function buildTemplateJson(type, slug, manifest) {
710
- if (type === 'discovery') {
711
- return {
712
- squad: manifest.slug,
713
- mode: manifest.mode,
714
- summary: '',
715
- goals: [],
716
- constraints: [],
717
- references: []
718
- };
719
- }
720
-
721
- if (type === 'design-doc') {
722
- return {
723
- squad: manifest.slug,
724
- mode: manifest.mode,
725
- problem: '',
726
- scope: [],
727
- outOfScope: [],
728
- deliverables: []
729
- };
730
- }
731
-
732
- return {
733
- squad: manifest.slug,
734
- blueprint: slug,
735
- contentType: 'content',
736
- layoutType: 'document',
737
- title: '',
738
- blocks: []
739
- };
740
- }
741
-
742
- function buildRulesDoc(manifest) {
743
- return [
744
- `# Squad Rules - ${manifest.name}`,
745
- '',
746
- '## Mission',
747
- manifest.mission,
748
- '',
749
- '## Operating mode',
750
- `- Mode: ${manifest.mode}`,
751
- '- Keep agents light and load skills on demand.',
752
- '- Use discovery and design doc before implementation or heavy generation.',
753
- '',
754
- '## Persistence',
755
- `- Primary: ${manifest.storagePolicy?.primary || 'sqlite'}`,
756
- `- Artifacts: ${manifest.storagePolicy?.artifacts || 'sqlite-json'}`,
757
- ''
758
- ].join('\n');
759
- }
760
-
761
- function buildOutputContractsDoc(manifest) {
762
- const blueprints = Array.isArray(manifest.contentBlueprints) ? manifest.contentBlueprints : [];
763
- const lines = [
764
- `# Output Contracts - ${manifest.name}`,
765
- '',
766
- '## Persistence rule',
767
- `- Mode: ${manifest.mode}`,
768
- `- Primary storage: ${manifest.storagePolicy?.primary || 'sqlite'}`,
769
- '',
770
- '## Blueprints'
771
- ];
772
-
773
- if (blueprints.length === 0) {
774
- lines.push('- No explicit content blueprints were declared.');
775
- } else {
776
- for (const blueprint of blueprints) {
777
- lines.push(`- ${blueprint.slug} (${blueprint.contentType} / ${blueprint.layoutType})`);
778
- }
779
- }
780
-
781
- lines.push('');
782
- return lines.join('\n');
783
- }
784
-
785
- async function materializeImportedSquad(projectDir, payload, sourceUrl, force) {
786
- const slug = sanitizeSegment(payload.squad.slug, 'squad');
787
- const agents = normalizeAgentsManifest(payload.version.agentsManifestJson);
788
- const squadManifest = buildLocalSquadManifest(payload, agents);
789
- const packageDir = localSquadPackageDir(projectDir, slug);
790
- const metadataPath = localSquadSummaryPath(projectDir, slug);
791
- const agentsDir = localSquadAgentsDir(projectDir, slug);
792
- const skillsDir = localSquadSkillsDir(projectDir, slug);
793
- const templatesDir = localSquadTemplatesDir(projectDir, slug);
794
- const docsDir = localSquadDocsDir(projectDir, slug);
795
- const textManifestPath = localSquadTextManifestPath(projectDir, slug);
796
- const jsonManifestPath = localSquadJsonManifestPath(projectDir, slug);
797
- const outputDir = localSquadOutputDir(projectDir, slug);
798
- const logsDir = localSquadLogsDir(projectDir, slug);
799
- const mediaDir = localSquadMediaDir(projectDir, slug);
800
- const designDocPath = localSquadDesignDocPath(projectDir, slug);
801
- const readinessPath = localSquadReadinessPath(projectDir, slug);
802
- const rulesDocPath = localSquadRulesDocPath(projectDir, slug);
803
- const outputContractsPath = localSquadOutputContractsPath(projectDir, slug);
804
- const manifestPath = installedManifestPath(projectDir, slug);
805
-
806
- if (!force && (await exists(metadataPath))) {
807
- throw new Error(`Imported squad already materialized: ${metadataPath}`);
808
- }
809
-
810
- await ensureDir(packageDir);
811
- await ensureDir(agentsDir);
812
- await ensureDir(skillsDir);
813
- await ensureDir(templatesDir);
814
- await ensureDir(docsDir);
815
- await ensureDir(outputDir);
816
- await ensureDir(logsDir);
817
- await ensureDir(mediaDir);
818
- await ensureDir(path.dirname(manifestPath));
819
-
820
- const metadata = buildSquadMetadata(payload, { sourceUrl });
821
- await fs.writeFile(metadataPath, metadata, 'utf8');
822
- await fs.writeFile(textManifestPath, buildSquadTextManifest(payload, squadManifest), 'utf8');
823
- await fs.writeFile(jsonManifestPath, `${JSON.stringify(squadManifest, null, 2)}\n`, 'utf8');
824
- await fs.writeFile(designDocPath, buildSquadDesignDoc(payload, squadManifest), 'utf8');
825
- await fs.writeFile(readinessPath, buildSquadReadiness(payload, squadManifest), 'utf8');
826
- await fs.writeFile(rulesDocPath, buildRulesDoc(squadManifest), 'utf8');
827
- await fs.writeFile(outputContractsPath, buildOutputContractsDoc(squadManifest), 'utf8');
828
- await fs.writeFile(
829
- manifestPath,
830
- `${JSON.stringify(buildInstalledManifest(payload, sourceUrl, agents), null, 2)}\n`,
831
- 'utf8'
832
- );
833
-
834
- const writtenAgents = [];
835
- for (const agent of agents) {
836
- const filePath = path.join(agentsDir, `${agent.slug}.md`);
837
- const body = agent.body || buildAgentStub(payload, agent);
838
- await fs.writeFile(filePath, body.endsWith('\n') ? body : `${body}\n`, 'utf8');
839
- writtenAgents.push(filePath);
840
- }
841
-
842
- const writtenSkills = [];
843
- for (const skill of squadManifest.skills || []) {
844
- const filePath = path.join(skillsDir, `${sanitizeSegment(skill.slug, 'skill')}.md`);
845
- await fs.writeFile(filePath, buildSkillMarkdown(skill), 'utf8');
846
- writtenSkills.push(filePath);
847
- }
848
-
849
- const templateEntries = [
850
- { slug: 'discovery', filePath: path.join(templatesDir, 'discovery.template.json') },
851
- { slug: 'design-doc', filePath: path.join(templatesDir, 'design-doc.template.json') },
852
- { slug: 'artifact', filePath: path.join(templatesDir, 'artifact.template.json') }
853
- ];
854
- const writtenTemplates = [];
855
- for (const template of templateEntries) {
856
- await fs.writeFile(
857
- template.filePath,
858
- `${JSON.stringify(buildTemplateJson(template.slug, template.slug, squadManifest), null, 2)}\n`,
859
- 'utf8'
860
- );
861
- writtenTemplates.push(template.filePath);
862
- }
863
-
864
- const writtenGenomes = [];
865
- for (const genome of payload.appliedGenomes || []) {
866
- const genomePath = localGenomeFilePath(projectDir, genome.genome.slug);
867
- if (!force && (await exists(genomePath))) continue;
868
- await ensureDir(path.dirname(genomePath));
869
- const content = genome.version.contentMarkdown
870
- ? String(genome.version.contentMarkdown)
871
- : [
872
- `# ${genome.genome.name}`,
873
- '',
874
- '> Imported from AIOSON Cloud.',
875
- '',
876
- `Version: ${genome.version.versionNumber}`,
877
- `Scope: ${genome.scopeType}`,
878
- genome.agentSlug ? `Agent: ${normalizeAgentSlug(genome.agentSlug)}` : null
879
- ]
880
- .filter(Boolean)
881
- .join('\n');
882
- await fs.writeFile(genomePath, content.endsWith('\n') ? content : `${content}\n`, 'utf8');
883
- writtenGenomes.push(genomePath);
884
- }
885
-
886
- const runtimeHandle = await openRuntimeDb(projectDir);
887
- try {
888
- upsertSquadManifest(runtimeHandle.db, {
889
- slug,
890
- name: squadManifest.name,
891
- mode: squadManifest.mode,
892
- mission: squadManifest.mission,
893
- goal: squadManifest.goal,
894
- visibility: squadManifest.visibility,
895
- status: 'active',
896
- manifest: squadManifest,
897
- context: squadManifest.context,
898
- packageDir: `.aioson/squads/${slug}`,
899
- agentsDir: `.aioson/squads/${slug}/agents`,
900
- outputDir: `output/${slug}`,
901
- logsDir: `aioson-logs/${slug}`,
902
- mediaDir: `media/${slug}`,
903
- latestSessionPath: `output/${slug}/latest.html`
904
- });
905
- } finally {
906
- runtimeHandle.db.close();
907
- }
908
-
909
- return {
910
- metadataPath,
911
- textManifestPath,
912
- jsonManifestPath,
913
- designDocPath,
914
- readinessPath,
915
- rulesDocPath,
916
- outputContractsPath,
917
- manifestPath,
918
- packageDir,
919
- agentsDir,
920
- skillsDir,
921
- templatesDir,
922
- docsDir,
923
- outputDir,
924
- logsDir,
925
- mediaDir,
926
- writtenAgents,
927
- writtenSkills,
928
- writtenTemplates,
929
- writtenGenomes
930
- };
931
- }
932
-
933
- async function ensureProjectDir(targetDir, t) {
934
- const absolute = path.resolve(process.cwd(), targetDir || '.');
935
- const stat = await fs.stat(absolute).catch(() => null);
936
- if (!stat || !stat.isDirectory()) {
937
- throw new Error(t('cloud.project_missing', { path: absolute }));
938
- }
939
- return absolute;
940
- }
941
-
942
- async function fetchJson(url) {
943
- const response = await fetch(url, {
944
- headers: {
945
- accept: 'application/json'
946
- }
947
- });
948
-
949
- const text = await response.text();
950
- let parsed = null;
951
-
952
- try {
953
- parsed = text ? JSON.parse(text) : null;
954
- } catch {
955
- parsed = null;
956
- }
957
-
958
- if (!response.ok) {
959
- const detail =
960
- parsed && typeof parsed === 'object' && parsed.error ? String(parsed.error) : `${response.status} ${response.statusText}`;
961
- throw new Error(`HTTP ${response.status}: ${detail}`);
962
- }
963
-
964
- if (!parsed || typeof parsed !== 'object') {
965
- throw new Error('Invalid JSON response.');
966
- }
967
-
968
- return parsed;
969
- }
970
-
971
- async function postJson(url, payload, fetchImpl = fetch) {
972
- const response = await fetchImpl(url, {
973
- method: 'POST',
974
- headers: {
975
- accept: 'application/json',
976
- 'content-type': 'application/json'
977
- },
978
- body: JSON.stringify(payload)
979
- });
980
-
981
- const text = await response.text();
982
- let parsed = null;
983
-
984
- try {
985
- parsed = text ? JSON.parse(text) : null;
986
- } catch {
987
- parsed = null;
988
- }
989
-
990
- if (!response.ok) {
991
- const detail =
992
- parsed && typeof parsed === 'object' && parsed.error ? String(parsed.error) : `${response.status} ${response.statusText}`;
993
- throw new Error(`HTTP ${response.status}: ${detail}`);
994
- }
995
-
996
- if (!parsed || typeof parsed !== 'object') {
997
- throw new Error('Invalid JSON response.');
998
- }
999
-
1000
- return parsed;
1001
- }
1002
-
1003
- function validateSquadSnapshot(payload) {
1004
- if (!payload || typeof payload !== 'object') {
1005
- throw new Error('Invalid squad snapshot payload.');
1006
- }
1007
-
1008
- if (payload.kind !== 'aiosforge.squad') {
1009
- throw new Error('Unsupported snapshot kind.');
1010
- }
1011
-
1012
- if (payload.exportVersion !== 1) {
1013
- throw new Error('Unsupported snapshot export version.');
1014
- }
1015
-
1016
- if (!payload.squad || typeof payload.squad !== 'object' || !payload.squad.slug) {
1017
- throw new Error('Snapshot is missing squad metadata.');
1018
- }
1019
-
1020
- if (!payload.version || typeof payload.version !== 'object' || !payload.version.versionNumber) {
1021
- throw new Error('Snapshot is missing version metadata.');
1022
- }
1023
-
1024
- return {
1025
- slug: String(payload.squad.slug),
1026
- versionNumber: String(payload.version.versionNumber)
1027
- };
1028
- }
1029
-
1030
- function validateGenomeSnapshot(payload) {
1031
- if (!payload || typeof payload !== 'object') {
1032
- throw new Error('Invalid genome snapshot payload.');
1033
- }
1034
-
1035
- if (payload.kind !== 'aiosforge.genome') {
1036
- throw new Error('Unsupported genome snapshot kind.');
1037
- }
1038
-
1039
- if (payload.exportVersion !== 1) {
1040
- throw new Error('Unsupported genome snapshot export version.');
1041
- }
1042
-
1043
- if (!payload.genome || typeof payload.genome !== 'object' || !payload.genome.slug) {
1044
- throw new Error('Snapshot is missing genome metadata.');
1045
- }
1046
-
1047
- if (!payload.version || typeof payload.version !== 'object' || !payload.version.versionNumber) {
1048
- throw new Error('Snapshot is missing genome version metadata.');
1049
- }
1050
-
1051
- return {
1052
- slug: String(payload.genome.slug),
1053
- versionNumber: String(payload.version.versionNumber)
1054
- };
1055
- }
1056
-
1057
- async function writeImportSnapshot(projectDir, payload, slug, versionNumber, force) {
1058
- const latestPath = squadImportFilePath(projectDir, slug, versionNumber);
1059
- const archivePath = historyImportFilePath(projectDir, slug, versionNumber);
1060
-
1061
- if (!force && (await exists(latestPath))) {
1062
- throw new Error(`Snapshot already exists: ${latestPath}`);
1063
- }
1064
-
1065
- await ensureDir(path.dirname(latestPath));
1066
- await ensureDir(path.dirname(archivePath));
1067
- const json = `${JSON.stringify(payload, null, 2)}\n`;
1068
- await fs.writeFile(latestPath, json, 'utf8');
1069
- await fs.writeFile(archivePath, json, 'utf8');
1070
-
1071
- return {
1072
- latestPath,
1073
- archivePath
1074
- };
1075
- }
1076
-
1077
- async function writeGenomeImportSnapshot(projectDir, payload, slug, versionNumber, force) {
1078
- const latestPath = genomeImportFilePath(projectDir, slug, versionNumber);
1079
- const archivePath = genomeHistoryImportFilePath(projectDir, slug, versionNumber);
1080
-
1081
- if (!force && (await exists(latestPath))) {
1082
- throw new Error(`Snapshot already exists: ${latestPath}`);
1083
- }
1084
-
1085
- await ensureDir(path.dirname(latestPath));
1086
- await ensureDir(path.dirname(archivePath));
1087
- const json = `${JSON.stringify(payload, null, 2)}\n`;
1088
- await fs.writeFile(latestPath, json, 'utf8');
1089
- await fs.writeFile(archivePath, json, 'utf8');
1090
-
1091
- return {
1092
- latestPath,
1093
- archivePath
1094
- };
1095
- }
1096
-
1097
- function buildInstalledGenomeManifest(snapshot, sourceUrl) {
1098
- return {
1099
- kind: 'aiosforge.local-installed-genome',
1100
- installVersion: 1,
1101
- installedAt: new Date().toISOString(),
1102
- sourceUrl,
1103
- genome: snapshot.genome,
1104
- version: {
1105
- versionNumber: snapshot.version.versionNumber,
1106
- schemaVersion: snapshot.version.schemaVersion
1107
- }
1108
- };
1109
- }
1110
-
1111
- async function materializeImportedGenome(projectDir, payload, sourceUrl, force) {
1112
- const slug = sanitizeSegment(payload.genome.slug, 'genome');
1113
- const genomePath = localGenomeFilePath(projectDir, slug);
1114
- const manifestPath = installedGenomeManifestPath(projectDir, slug);
1115
-
1116
- if (!force && (await exists(genomePath))) {
1117
- throw new Error(`Imported genome already materialized: ${genomePath}`);
1118
- }
1119
-
1120
- await ensureDir(path.dirname(genomePath));
1121
- await ensureDir(path.dirname(manifestPath));
1122
-
1123
- const content = payload.version.contentMarkdown
1124
- ? String(payload.version.contentMarkdown)
1125
- : [
1126
- `# ${payload.genome.name}`,
1127
- '',
1128
- '> Imported from AIOSON Cloud.',
1129
- '',
1130
- `Version: ${payload.version.versionNumber}`,
1131
- payload.version.summary ? '' : null,
1132
- payload.version.summary || null
1133
- ]
1134
- .filter(Boolean)
1135
- .join('\n');
1136
-
1137
- await fs.writeFile(genomePath, content.endsWith('\n') ? content : `${content}\n`, 'utf8');
1138
- await fs.writeFile(
1139
- manifestPath,
1140
- `${JSON.stringify(buildInstalledGenomeManifest(payload, sourceUrl), null, 2)}\n`,
1141
- 'utf8'
1142
- );
1143
-
1144
- return {
1145
- genomePath,
1146
- manifestPath
1147
- };
1148
- }
1149
-
1150
- async function runCloudImportSquad({ args, options = {}, logger, t }) {
1151
- const projectDir = await ensureProjectDir(args[0] || '.', t);
1152
- const url = String(options.url || '').trim();
1153
- if (!url) {
1154
- throw new Error(t('cloud.url_required'));
1155
- }
1156
-
1157
- const dryRun = Boolean(options['dry-run']);
1158
- const force = Boolean(options.force);
1159
- const payload = await fetchJson(url);
1160
- const snapshot = validateSquadSnapshot(payload);
1161
-
1162
- const planned = {
1163
- ok: true,
1164
- resource: 'squad',
1165
- url,
1166
- dryRun,
1167
- force,
1168
- projectDir,
1169
- slug: snapshot.slug,
1170
- versionNumber: snapshot.versionNumber,
1171
- importDir: path.dirname(squadImportFilePath(projectDir, snapshot.slug, snapshot.versionNumber)),
1172
- latestFile: squadImportFilePath(projectDir, snapshot.slug, snapshot.versionNumber),
1173
- materialized: !Boolean(options['snapshots-only'])
1174
- };
1175
-
1176
- if (dryRun) {
1177
- logger.log(
1178
- t('cloud.import_squad_dry_run', {
1179
- slug: snapshot.slug,
1180
- version: snapshot.versionNumber
1181
- })
1182
- );
1183
- return planned;
1184
- }
1185
-
1186
- const written = await writeImportSnapshot(projectDir, payload, snapshot.slug, snapshot.versionNumber, force);
1187
- let materialized = null;
1188
- if (!options['snapshots-only']) {
1189
- materialized = await materializeImportedSquad(projectDir, payload, url, force);
1190
- }
1191
- logger.log(
1192
- t('cloud.import_squad_done', {
1193
- slug: snapshot.slug,
1194
- version: snapshot.versionNumber
1195
- })
1196
- );
1197
-
1198
- return {
1199
- ...planned,
1200
- latestFile: written.latestPath,
1201
- archiveFile: written.archivePath,
1202
- relativeLatestFile: toRelativeSafe(projectDir, written.latestPath),
1203
- relativeArchiveFile: toRelativeSafe(projectDir, written.archivePath),
1204
- materializedMetadataFile: materialized ? toRelativeSafe(projectDir, materialized.metadataPath) : null,
1205
- materializedPackageDir: materialized ? toRelativeSafe(projectDir, materialized.packageDir) : null,
1206
- materializedManifestFile: materialized ? toRelativeSafe(projectDir, materialized.manifestPath) : null,
1207
- materializedAgentsDir: materialized ? toRelativeSafe(projectDir, materialized.agentsDir) : null,
1208
- materializedSkillsDir: materialized ? toRelativeSafe(projectDir, materialized.skillsDir) : null,
1209
- materializedTemplatesDir: materialized ? toRelativeSafe(projectDir, materialized.templatesDir) : null,
1210
- materializedDocsDir: materialized ? toRelativeSafe(projectDir, materialized.docsDir) : null,
1211
- materializedOutputDir: materialized ? toRelativeSafe(projectDir, materialized.outputDir) : null,
1212
- materializedLogsDir: materialized ? toRelativeSafe(projectDir, materialized.logsDir) : null,
1213
- materializedDesignDocFile: materialized ? toRelativeSafe(projectDir, materialized.designDocPath) : null,
1214
- materializedReadinessFile: materialized ? toRelativeSafe(projectDir, materialized.readinessPath) : null,
1215
- writtenAgents: materialized ? materialized.writtenAgents.map((file) => toRelativeSafe(projectDir, file)) : [],
1216
- writtenSkills: materialized ? materialized.writtenSkills.map((file) => toRelativeSafe(projectDir, file)) : [],
1217
- writtenTemplates: materialized ? materialized.writtenTemplates.map((file) => toRelativeSafe(projectDir, file)) : [],
1218
- writtenGenomes: materialized ? materialized.writtenGenomes.map((file) => toRelativeSafe(projectDir, file)) : []
1219
- };
1220
- }
1221
-
1222
- async function runCloudImportGenome({ args, options = {}, logger, t }) {
1223
- const projectDir = await ensureProjectDir(args[0] || '.', t);
1224
- const url = String(options.url || '').trim();
1225
- if (!url) {
1226
- throw new Error(t('cloud.url_required'));
1227
- }
1228
-
1229
- const dryRun = Boolean(options['dry-run']);
1230
- const force = Boolean(options.force);
1231
- const payload = await fetchJson(url);
1232
- const snapshot = validateGenomeSnapshot(payload);
1233
-
1234
- const planned = {
1235
- ok: true,
1236
- resource: 'genome',
1237
- url,
1238
- dryRun,
1239
- force,
1240
- projectDir,
1241
- slug: snapshot.slug,
1242
- versionNumber: snapshot.versionNumber,
1243
- importDir: path.dirname(genomeImportFilePath(projectDir, snapshot.slug, snapshot.versionNumber)),
1244
- latestFile: genomeImportFilePath(projectDir, snapshot.slug, snapshot.versionNumber),
1245
- materialized: !Boolean(options['snapshots-only'])
1246
- };
1247
-
1248
- if (dryRun) {
1249
- logger.log(
1250
- t('cloud.import_genome_dry_run', {
1251
- slug: snapshot.slug,
1252
- version: snapshot.versionNumber
1253
- })
1254
- );
1255
- return planned;
1256
- }
1257
-
1258
- const written = await writeGenomeImportSnapshot(projectDir, payload, snapshot.slug, snapshot.versionNumber, force);
1259
- let materialized = null;
1260
- if (!options['snapshots-only']) {
1261
- materialized = await materializeImportedGenome(projectDir, payload, url, force);
1262
- }
1263
-
1264
- logger.log(
1265
- t('cloud.import_genome_done', {
1266
- slug: snapshot.slug,
1267
- version: snapshot.versionNumber
1268
- })
1269
- );
1270
-
1271
- return {
1272
- ...planned,
1273
- latestFile: written.latestPath,
1274
- archiveFile: written.archivePath,
1275
- relativeLatestFile: toRelativeSafe(projectDir, written.latestPath),
1276
- relativeArchiveFile: toRelativeSafe(projectDir, written.archivePath),
1277
- materializedGenomeFile: materialized ? toRelativeSafe(projectDir, materialized.genomePath) : null,
1278
- materializedManifestFile: materialized ? toRelativeSafe(projectDir, materialized.manifestPath) : null
1279
- };
1280
- }
1281
-
1282
- async function loadLocalGenomeSnapshot(projectDir, slug, options = {}) {
1283
- const filePath = localGenomeFilePath(projectDir, slug);
1284
- if (!(await exists(filePath))) {
1285
- throw new Error(`Genome file not found: ${filePath}`);
1286
- }
1287
-
1288
- const markdown = await fs.readFile(filePath, 'utf8');
1289
- const versionNumber = String(options['resource-version'] || '').trim();
1290
- if (!versionNumber) {
1291
- throw new Error('Missing required --resource-version for genome publish.');
1292
- }
1293
-
1294
- const genomeName = findPrimaryHeading(markdown, slug);
1295
- return {
1296
- kind: 'aiosforge.genome',
1297
- exportVersion: 1,
1298
- genome: {
1299
- id: null,
1300
- name: genomeName,
1301
- slug: sanitizeSegment(slug, 'genome'),
1302
- description: options.description ? String(options.description).trim() : firstParagraph(markdown),
1303
- visibility: String(options.visibility || 'PRIVATE').toUpperCase(),
1304
- status: 'PUBLISHED',
1305
- sourceKind: String(options['source-kind'] || 'LOCAL').toUpperCase(),
1306
- ownerUsername: String(options.owner || 'local')
1307
- },
1308
- version: {
1309
- id: null,
1310
- versionNumber,
1311
- versionCode: 1,
1312
- title: options.title ? String(options.title).trim() : genomeName,
1313
- summary: options.summary ? String(options.summary).trim() : firstParagraph(markdown),
1314
- schemaVersion: options['schema-version'] ? String(options['schema-version']).trim() : '1',
1315
- isCurrent: true,
1316
- createdAt: new Date().toISOString(),
1317
- contentMarkdown: markdown,
1318
- manifestJson: null
1319
- }
1320
- };
1321
- }
1322
-
1323
- async function listAgentFiles(agentsDir) {
1324
- const entries = await fs.readdir(agentsDir).catch(() => []);
1325
- const files = [];
1326
- for (const entry of entries) {
1327
- if (!entry.endsWith('.md')) continue;
1328
- const absPath = path.join(agentsDir, entry);
1329
- const stat = await fs.stat(absPath).catch(() => null);
1330
- if (!stat?.isFile()) continue;
1331
- files.push(absPath);
1332
- }
1333
- files.sort();
1334
- return files;
1335
- }
1336
-
1337
- async function buildAppliedGenomesFromMetadata(projectDir, metadataContent, options = {}) {
1338
- const structured = await buildAppliedGenomesFromBindings(
1339
- projectDir,
1340
- options.genomeBindings || options.manifestBindings || options.manifest?.genomes,
1341
- options
1342
- );
1343
- if (structured.length > 0) {
1344
- return structured;
1345
- }
1346
-
1347
- const linkedVersion = String(options['linked-genome-version'] || options['resource-version'] || '1.0.0').trim();
1348
- const shared = parseListSection(metadataContent, 'Genomes');
1349
- const scoped = parseListSection(metadataContent, 'AgentGenomes').map(parseAgentGenomeEntry).filter(Boolean);
1350
- const items = [];
1351
-
1352
- for (const genomeRelPath of shared) {
1353
- const genomeAbsPath = path.join(projectDir, normalizeRel(genomeRelPath));
1354
- const markdown = await fs.readFile(genomeAbsPath, 'utf8').catch(() => null);
1355
- if (!markdown) continue;
1356
- const slug = sanitizeSegment(path.basename(genomeAbsPath, '.md'), 'genome');
1357
- items.push({
1358
- scopeType: 'SQUAD',
1359
- agentSlug: null,
1360
- priority: 0,
1361
- genome: {
1362
- id: null,
1363
- name: findPrimaryHeading(markdown, slug),
1364
- slug,
1365
- visibility: String(options.visibility || 'PRIVATE').toUpperCase(),
1366
- status: 'PUBLISHED',
1367
- sourceKind: 'LOCAL'
1368
- },
1369
- version: {
1370
- id: null,
1371
- versionNumber: linkedVersion,
1372
- versionCode: 1,
1373
- title: findPrimaryHeading(markdown, slug),
1374
- summary: firstParagraph(markdown),
1375
- schemaVersion: '1',
1376
- contentMarkdown: markdown,
1377
- manifestJson: null,
1378
- createdAt: new Date().toISOString()
1379
- }
1380
- });
1381
- }
1382
-
1383
- for (const entry of scoped) {
1384
- const genomeAbsPath = path.join(projectDir, normalizeRel(entry.genomePath));
1385
- const markdown = await fs.readFile(genomeAbsPath, 'utf8').catch(() => null);
1386
- if (!markdown) continue;
1387
- const slug = sanitizeSegment(path.basename(genomeAbsPath, '.md'), 'genome');
1388
- items.push({
1389
- scopeType: 'AGENT',
1390
- agentSlug: entry.agentSlug,
1391
- priority: 0,
1392
- genome: {
1393
- id: null,
1394
- name: findPrimaryHeading(markdown, slug),
1395
- slug,
1396
- visibility: String(options.visibility || 'PRIVATE').toUpperCase(),
1397
- status: 'PUBLISHED',
1398
- sourceKind: 'LOCAL'
1399
- },
1400
- version: {
1401
- id: null,
1402
- versionNumber: linkedVersion,
1403
- versionCode: 1,
1404
- title: findPrimaryHeading(markdown, slug),
1405
- summary: firstParagraph(markdown),
1406
- schemaVersion: '1',
1407
- contentMarkdown: markdown,
1408
- manifestJson: null,
1409
- createdAt: new Date().toISOString()
1410
- }
1411
- });
1412
- }
1413
-
1414
- return items;
1415
- }
1416
-
1417
- async function buildAppliedGenomesFromBindings(projectDir, genomeBindings, options = {}) {
1418
- const linkedVersion = String(options['linked-genome-version'] || options['resource-version'] || '1.0.0').trim();
1419
- const flattened = flattenGenomeBindings(genomeBindings);
1420
- const items = [];
1421
-
1422
- for (const binding of flattened) {
1423
- const genomeAbsPath = path.join(projectDir, '.aioson', 'genomes', `${binding.slug}.md`);
1424
- const markdown = await fs.readFile(genomeAbsPath, 'utf8').catch(() => null);
1425
- if (!markdown) continue;
1426
-
1427
- items.push({
1428
- scopeType: binding.scope === 'squad' ? 'SQUAD' : 'AGENT',
1429
- agentSlug: binding.agentSlug || null,
1430
- priority: Number.isFinite(binding.priority) ? binding.priority : 0,
1431
- genome: {
1432
- id: null,
1433
- name: findPrimaryHeading(markdown, binding.slug),
1434
- slug: binding.slug,
1435
- type: binding.type || null,
1436
- visibility: String(options.visibility || 'PRIVATE').toUpperCase(),
1437
- status: 'PUBLISHED',
1438
- sourceKind: String(binding.source || 'LOCAL').toUpperCase()
1439
- },
1440
- version: {
1441
- id: null,
1442
- versionNumber: binding.version || linkedVersion,
1443
- versionCode: 1,
1444
- title: findPrimaryHeading(markdown, binding.slug),
1445
- summary: firstParagraph(markdown),
1446
- schemaVersion: '1',
1447
- contentMarkdown: markdown,
1448
- manifestJson: {
1449
- type: binding.type,
1450
- evidenceMode: binding.evidenceMode
1451
- },
1452
- createdAt: new Date().toISOString()
1453
- }
1454
- });
1455
- }
1456
-
1457
- return items;
1458
- }
1459
-
1460
- async function loadLocalSquadSnapshot(projectDir, slug, options = {}) {
1461
- const packageSummaryPath = localSquadSummaryPath(projectDir, slug);
1462
- const metadataPath = (await exists(packageSummaryPath))
1463
- ? packageSummaryPath
1464
- : localLegacySquadMetadataPath(projectDir, slug);
1465
- if (!(await exists(metadataPath))) {
1466
- throw new Error(`Squad metadata not found: ${metadataPath}`);
1467
- }
1468
-
1469
- const content = await fs.readFile(metadataPath, 'utf8');
1470
- const versionNumber = String(options['resource-version'] || '').trim();
1471
- if (!versionNumber) {
1472
- throw new Error('Missing required --resource-version for squad publish.');
1473
- }
1474
-
1475
- const squadName = extractField(content, 'Squad') || slug;
1476
- const goal = extractField(content, 'Goal', 'Objetivo') || firstParagraph(content) || null;
1477
- const packageRootRel = `.aioson/squads/${slug}`;
1478
- const agentsDirRel = normalizeRel(extractField(content, 'Agents') || `${packageRootRel}/agents`);
1479
- const outputDirRel = normalizeRel(extractField(content, 'Output') || `output/${slug}`);
1480
- const logsDirRel = normalizeRel(extractField(content, 'Logs') || `aioson-logs/${slug}`);
1481
- const mediaDirRel = normalizeRel(extractField(content, 'Media') || `media/${slug}`);
1482
- const agentsDirAbs = path.join(projectDir, agentsDirRel);
1483
- const localManifest = (await loadLocalSquadManifest(projectDir, slug)) || {};
1484
- const packageDirRel = normalizeRel(localManifest?.package?.rootDir || packageRootRel);
1485
- const textManifestPath = (await exists(localSquadTextManifestPath(projectDir, slug)))
1486
- ? localSquadTextManifestPath(projectDir, slug)
1487
- : localLegacySquadTextManifestPath(projectDir, slug);
1488
- const designDocPath = (await exists(localSquadDesignDocPath(projectDir, slug)))
1489
- ? localSquadDesignDocPath(projectDir, slug)
1490
- : localLegacySquadDesignDocPath(projectDir, slug);
1491
- const readinessPath = (await exists(localSquadReadinessPath(projectDir, slug)))
1492
- ? localSquadReadinessPath(projectDir, slug)
1493
- : localLegacySquadReadinessPath(projectDir, slug);
1494
- const textManifest = (await fs.readFile(textManifestPath, 'utf8').catch(() => null)) || null;
1495
- const designDocMarkdown = (await fs.readFile(designDocPath, 'utf8').catch(() => null)) || null;
1496
- const readinessMarkdown = (await fs.readFile(readinessPath, 'utf8').catch(() => null)) || null;
1497
- const agentFiles = await listAgentFiles(agentsDirAbs);
1498
- const agentsManifestJson = [];
1499
-
1500
- for (const filePath of agentFiles) {
1501
- const baseName = path.basename(filePath);
1502
- if (
1503
- baseName === 'agents.md' ||
1504
- baseName === 'squad.manifest.json' ||
1505
- baseName === 'design-doc.md' ||
1506
- baseName === 'readiness.md'
1507
- ) continue;
1508
- const markdown = await fs.readFile(filePath, 'utf8');
1509
- const agentSlug = sanitizeSegment(path.basename(filePath, '.md'), 'agent');
1510
- agentsManifestJson.push({
1511
- slug: agentSlug,
1512
- name: findPrimaryHeading(markdown, agentSlug),
1513
- description: firstParagraph(markdown),
1514
- content: markdown
1515
- });
1516
- }
1517
-
1518
- const manifestBindings = mergeGenomeBindings({
1519
- blueprintBindings: localManifest?.genomeBindings,
1520
- manifestBindings: localManifest?.genomeBindings || localManifest?.genomes,
1521
- legacyExecutors: localManifest?.executors
1522
- });
1523
- const normalizedManifest =
1524
- localManifest && typeof localManifest === 'object'
1525
- ? {
1526
- ...localManifest,
1527
- slug: sanitizeSegment(localManifest.slug || slug, 'squad'),
1528
- name: String(localManifest.name || squadName),
1529
- packageVersion: String(localManifest.packageVersion || versionNumber),
1530
- mode: String(localManifest.mode || 'content'),
1531
- mission: String(localManifest.mission || extractField(content, 'Description', 'Descricao') || goal || squadName),
1532
- goal: String(localManifest.goal || goal || ''),
1533
- visibility: String(localManifest.visibility || options.visibility || 'private').toLowerCase(),
1534
- aiosLiteCompatibility: String(
1535
- localManifest.aiosLiteCompatibility ||
1536
- options['compatibility-min'] ||
1537
- '^1.1.0'
1538
- ),
1539
- rules: {
1540
- ...(localManifest.rules && typeof localManifest.rules === 'object' ? localManifest.rules : {}),
1541
- outputsDir: localManifest?.rules?.outputsDir || outputDirRel,
1542
- logsDir: localManifest?.rules?.logsDir || logsDirRel,
1543
- mediaDir: localManifest?.rules?.mediaDir || mediaDirRel
1544
- },
1545
- storagePolicy:
1546
- localManifest.storagePolicy && typeof localManifest.storagePolicy === 'object'
1547
- ? localManifest.storagePolicy
1548
- : {
1549
- primary: String(localManifest.mode || 'content') === 'builder' ? 'files' : 'sqlite',
1550
- artifacts: String(localManifest.mode || 'content') === 'builder' ? 'files+sqlite' : 'sqlite-json',
1551
- exports: { html: true, markdown: true, json: true }
1552
- },
1553
- package:
1554
- localManifest.package && typeof localManifest.package === 'object'
1555
- ? localManifest.package
1556
- : {
1557
- rootDir: packageDirRel,
1558
- agentsDir: `${packageDirRel}/agents`,
1559
- skillsDir: `${packageDirRel}/skills`,
1560
- templatesDir: `${packageDirRel}/templates`,
1561
- docsDir: `${packageDirRel}/docs`
1562
- },
1563
- baseRoles: Array.isArray(localManifest.baseRoles)
1564
- ? localManifest.baseRoles
1565
- : ['orchestrator', 'discovery-lead', 'design-doc-lead', 'planner', 'implementer', 'reviewer', 'docs-maintainer'],
1566
- skills: Array.isArray(localManifest.skills) ? localManifest.skills : [],
1567
- mcps: Array.isArray(localManifest.mcps) ? localManifest.mcps : [],
1568
- subagents:
1569
- localManifest.subagents && typeof localManifest.subagents === 'object'
1570
- ? localManifest.subagents
1571
- : {
1572
- allowed: true,
1573
- when: ['broad research', 'comparison', 'large-context summarization', 'parallel analysis']
1574
- },
1575
- contentBlueprints: normalizeContentBlueprints(localManifest.contentBlueprints),
1576
- context:
1577
- localManifest.context && typeof localManifest.context === 'object'
1578
- ? {
1579
- ...localManifest.context,
1580
- designDocPath: localManifest.context.designDocPath || `${packageDirRel}/docs/design-doc.md`,
1581
- readinessPath: localManifest.context.readinessPath || `${packageDirRel}/docs/readiness.md`,
1582
- docsPackage: Array.isArray(localManifest.context.docsPackage)
1583
- ? localManifest.context.docsPackage
1584
- : ['project.context.md', 'design-doc.md', 'readiness.md']
1585
- }
1586
- : {
1587
- mode: 'project',
1588
- summary: goal || squadName,
1589
- designDocPath: `${packageDirRel}/docs/design-doc.md`,
1590
- readinessPath: `${packageDirRel}/docs/readiness.md`,
1591
- docsPackage: ['project.context.md', 'design-doc.md', 'readiness.md']
1592
- },
1593
- executors: Array.isArray(localManifest.executors)
1594
- ? attachBindingsToExecutors(localManifest.executors, manifestBindings)
1595
- : agentsManifestJson.map((agent) => ({
1596
- slug: agent.slug,
1597
- title: agent.name,
1598
- role: agent.description,
1599
- file: `${packageDirRel}/agents/${agent.slug}.md`,
1600
- skills: [],
1601
- genomes: []
1602
- })),
1603
- genomes: manifestBindings,
1604
- genomeBindings: manifestBindings
1605
- }
1606
- : null;
1607
-
1608
- return {
1609
- kind: 'aiosforge.squad',
1610
- exportVersion: 1,
1611
- squad: {
1612
- id: null,
1613
- name: normalizedManifest?.name || squadName,
1614
- slug: sanitizeSegment(slug, 'squad'),
1615
- description: extractField(content, 'Description', 'Descricao') || normalizedManifest?.mission || null,
1616
- goal: normalizedManifest?.goal || goal,
1617
- visibility: String(options.visibility || normalizedManifest?.visibility || 'PRIVATE').toUpperCase(),
1618
- status: 'PUBLISHED',
1619
- ownerUsername: String(options.owner || 'local'),
1620
- projectName: null
1621
- },
1622
- version: {
1623
- id: null,
1624
- versionNumber,
1625
- versionCode: 1,
1626
- title: options.title ? String(options.title).trim() : squadName,
1627
- summary: options.summary ? String(options.summary).trim() : goal,
1628
- changeLog: options['change-log'] ? String(options['change-log']).trim() : null,
1629
- compatibilityMin: options['compatibility-min'] ? String(options['compatibility-min']).trim() : null,
1630
- compatibilityMax: options['compatibility-max'] ? String(options['compatibility-max']).trim() : null,
1631
- schemaVersion: options['schema-version'] ? String(options['schema-version']).trim() : '1',
1632
- sourceType: 'local_publish',
1633
- isCurrent: true,
1634
- createdAt: new Date().toISOString(),
1635
- designDocMarkdown,
1636
- readinessMarkdown,
1637
- manifestJson:
1638
- normalizedManifest || {
1639
- packageVersion: versionNumber,
1640
- mode: 'content',
1641
- metadataPath: normalizeRel(path.relative(projectDir, metadataPath)),
1642
- textManifestPath: normalizeRel(path.relative(projectDir, textManifestPath)),
1643
- package: {
1644
- rootDir: packageDirRel,
1645
- agentsDir: `${packageDirRel}/agents`,
1646
- skillsDir: `${packageDirRel}/skills`,
1647
- templatesDir: `${packageDirRel}/templates`,
1648
- docsDir: `${packageDirRel}/docs`
1649
- },
1650
- storagePolicy: {
1651
- primary: 'sqlite',
1652
- artifacts: 'sqlite-json',
1653
- exports: { html: true, markdown: true, json: true }
1654
- },
1655
- context: {
1656
- mode: 'project',
1657
- summary: goal || squadName,
1658
- designDocPath: normalizeRel(path.relative(projectDir, designDocPath)),
1659
- readinessPath: normalizeRel(path.relative(projectDir, readinessPath)),
1660
- docsPackage: ['project.context.md', 'design-doc.md', 'readiness.md']
1661
- },
1662
- outputDir: outputDirRel,
1663
- logsDir: logsDirRel,
1664
- mediaDir: mediaDirRel
1665
- },
1666
- agentsManifestJson,
1667
- genomesManifestJson: {
1668
- textManifestPath: textManifest
1669
- ? normalizeRel(path.relative(projectDir, textManifestPath))
1670
- : null,
1671
- genomes: normalizedManifest?.genomes || { squad: [], executors: {} }
1672
- }
1673
- },
1674
- appliedGenomes: await buildAppliedGenomesFromMetadata(projectDir, content, {
1675
- ...options,
1676
- genomeBindings: normalizedManifest?.genomes
1677
- })
1678
- };
1679
- }
1680
-
1681
- function resolvePublishUrl(options, resource) {
1682
- if (options.url) return String(options.url).trim();
1683
- const baseUrl = String(options['base-url'] || '').trim().replace(/\/+$/, '');
1684
- if (!baseUrl) return '';
1685
- return `${baseUrl}/api/publish/${resource}s`;
1686
- }
1687
-
1688
- async function runCloudPublishGenome({ args, options = {}, logger, t, dependencies = {} }) {
1689
- const projectDir = await ensureProjectDir(args[0] || '.', t);
1690
- const slug = String(options.slug || '').trim();
1691
- if (!slug) {
1692
- throw new Error('Missing required --slug for genome publish.');
1693
- }
1694
- const url = resolvePublishUrl(options, 'genome');
1695
- if (!url) {
1696
- throw new Error('Provide --url or --base-url for genome publish.');
1697
- }
1698
-
1699
- const payload = await loadLocalGenomeSnapshot(projectDir, slug, options);
1700
- const dryRun = Boolean(options['dry-run']);
1701
- const planned = {
1702
- ok: true,
1703
- resource: 'genome',
1704
- dryRun,
1705
- url,
1706
- slug: payload.genome.slug,
1707
- versionNumber: payload.version.versionNumber,
1708
- projectDir
1709
- };
1710
-
1711
- if (dryRun) {
1712
- logger.log(t('cloud.publish_genome_dry_run', { slug: payload.genome.slug, version: payload.version.versionNumber }));
1713
- return planned;
1714
- }
1715
-
1716
- const response = await postJson(url, payload, dependencies.fetchImpl || fetch);
1717
- logger.log(t('cloud.publish_genome_done', { slug: payload.genome.slug, version: payload.version.versionNumber }));
1718
- return {
1719
- ...planned,
1720
- response
1721
- };
1722
- }
1723
-
1724
- async function runCloudPublishSquad({ args, options = {}, logger, t, dependencies = {} }) {
1725
- const projectDir = await ensureProjectDir(args[0] || '.', t);
1726
- const slug = String(options.slug || '').trim();
1727
- if (!slug) {
1728
- throw new Error('Missing required --slug for squad publish.');
1729
- }
1730
- const url = resolvePublishUrl(options, 'squad');
1731
- if (!url) {
1732
- throw new Error('Provide --url or --base-url for squad publish.');
1733
- }
1734
-
1735
- const payload = await loadLocalSquadSnapshot(projectDir, slug, options);
1736
- const dryRun = Boolean(options['dry-run']);
1737
- const planned = {
1738
- ok: true,
1739
- resource: 'squad',
1740
- dryRun,
1741
- url,
1742
- slug: payload.squad.slug,
1743
- versionNumber: payload.version.versionNumber,
1744
- projectDir,
1745
- agentCount: Array.isArray(payload.version.agentsManifestJson) ? payload.version.agentsManifestJson.length : 0,
1746
- genomeCount: Array.isArray(payload.appliedGenomes) ? payload.appliedGenomes.length : 0
1747
- };
1748
-
1749
- if (dryRun) {
1750
- logger.log(t('cloud.publish_squad_dry_run', { slug: payload.squad.slug, version: payload.version.versionNumber }));
1751
- return planned;
1752
- }
1753
-
1754
- const response = await postJson(url, payload, dependencies.fetchImpl || fetch);
1755
- logger.log(t('cloud.publish_squad_done', { slug: payload.squad.slug, version: payload.version.versionNumber }));
1756
- return {
1757
- ...planned,
1758
- response
1759
- };
1760
- }
1761
-
1762
- module.exports = {
1763
- runCloudImportSquad,
1764
- runCloudImportGenome,
1765
- runCloudPublishGenome,
1766
- runCloudPublishSquad
1767
- };
1
+ 'use strict';
2
+
3
+ const fs = require('node:fs/promises');
4
+ const path = require('node:path');
5
+ const { ensureDir, exists, nowStamp, toRelativeSafe } = require('../utils');
6
+ const { openRuntimeDb, upsertSquadManifest } = require('../runtime-store');
7
+ const {
8
+ attachBindingsToExecutors,
9
+ flattenGenomeBindings,
10
+ mergeGenomeBindings,
11
+ normalizeBinding,
12
+ normalizeGenomeBindings,
13
+ resolveExecutorGenomes
14
+ } = require('../genomes/bindings');
15
+
16
+ function sanitizeSegment(value, fallback) {
17
+ const normalized = String(value || fallback || '')
18
+ .trim()
19
+ .toLowerCase()
20
+ .replace(/[^a-z0-9._-]+/g, '-')
21
+ .replace(/^-+|-+$/g, '')
22
+ .slice(0, 120);
23
+
24
+ return normalized || fallback;
25
+ }
26
+
27
+ function cloudImportsRoot(projectDir) {
28
+ return path.join(projectDir, '.aioson', 'cloud-imports');
29
+ }
30
+
31
+ function squadImportFilePath(projectDir, slug, versionNumber) {
32
+ const safeSlug = sanitizeSegment(slug, 'squad');
33
+ const safeVersion = sanitizeSegment(versionNumber, 'latest');
34
+ return path.join(cloudImportsRoot(projectDir), 'squads', safeSlug, `${safeVersion}.json`);
35
+ }
36
+
37
+ function genomeImportFilePath(projectDir, slug, versionNumber) {
38
+ const safeSlug = sanitizeSegment(slug, 'genome');
39
+ const safeVersion = sanitizeSegment(versionNumber, 'latest');
40
+ return path.join(cloudImportsRoot(projectDir), 'genomes', safeSlug, `${safeVersion}.json`);
41
+ }
42
+
43
+ function historyImportFilePath(projectDir, slug, versionNumber) {
44
+ const safeSlug = sanitizeSegment(slug, 'squad');
45
+ const safeVersion = sanitizeSegment(versionNumber, 'latest');
46
+ return path.join(
47
+ cloudImportsRoot(projectDir),
48
+ 'history',
49
+ 'squads',
50
+ safeSlug,
51
+ `${safeVersion}--${nowStamp()}.json`
52
+ );
53
+ }
54
+
55
+ function genomeHistoryImportFilePath(projectDir, slug, versionNumber) {
56
+ const safeSlug = sanitizeSegment(slug, 'genome');
57
+ const safeVersion = sanitizeSegment(versionNumber, 'latest');
58
+ return path.join(
59
+ cloudImportsRoot(projectDir),
60
+ 'history',
61
+ 'genomes',
62
+ safeSlug,
63
+ `${safeVersion}--${nowStamp()}.json`
64
+ );
65
+ }
66
+
67
+ function installedRoot(projectDir) {
68
+ return path.join(cloudImportsRoot(projectDir), 'installed');
69
+ }
70
+
71
+ function installedManifestPath(projectDir, slug) {
72
+ return path.join(installedRoot(projectDir), 'squads', sanitizeSegment(slug, 'squad'), 'manifest.json');
73
+ }
74
+
75
+ function installedGenomeManifestPath(projectDir, slug) {
76
+ return path.join(installedRoot(projectDir), 'genomes', sanitizeSegment(slug, 'genome'), 'manifest.json');
77
+ }
78
+
79
+ function localSquadPackageDir(projectDir, slug) {
80
+ return path.join(projectDir, '.aioson', 'squads', sanitizeSegment(slug, 'squad'));
81
+ }
82
+
83
+ function localSquadSummaryPath(projectDir, slug) {
84
+ return path.join(localSquadPackageDir(projectDir, slug), 'squad.md');
85
+ }
86
+
87
+ function localLegacySquadMetadataPath(projectDir, slug) {
88
+ return path.join(projectDir, '.aioson', 'squads', `${sanitizeSegment(slug, 'squad')}.md`);
89
+ }
90
+
91
+ function localSquadAgentsDir(projectDir, slug) {
92
+ return path.join(localSquadPackageDir(projectDir, slug), 'agents');
93
+ }
94
+
95
+ function localSquadSkillsDir(projectDir, slug) {
96
+ return path.join(localSquadPackageDir(projectDir, slug), 'skills');
97
+ }
98
+
99
+ function localSquadTemplatesDir(projectDir, slug) {
100
+ return path.join(localSquadPackageDir(projectDir, slug), 'templates');
101
+ }
102
+
103
+ function localSquadDocsDir(projectDir, slug) {
104
+ return path.join(localSquadPackageDir(projectDir, slug), 'docs');
105
+ }
106
+
107
+ function localLegacySquadAgentsDir(projectDir, slug) {
108
+ return path.join(projectDir, 'agents', sanitizeSegment(slug, 'squad'));
109
+ }
110
+
111
+ function localSquadOutputDir(projectDir, slug) {
112
+ return path.join(projectDir, 'output', sanitizeSegment(slug, 'squad'));
113
+ }
114
+
115
+ function localSquadLogsDir(projectDir, slug) {
116
+ return path.join(projectDir, 'aioson-logs', sanitizeSegment(slug, 'squad'));
117
+ }
118
+
119
+ function localSquadMediaDir(projectDir, slug) {
120
+ return path.join(projectDir, 'media', sanitizeSegment(slug, 'squad'));
121
+ }
122
+
123
+ function localSquadTextManifestPath(projectDir, slug) {
124
+ return path.join(localSquadAgentsDir(projectDir, slug), 'agents.md');
125
+ }
126
+
127
+ function localSquadJsonManifestPath(projectDir, slug) {
128
+ return path.join(localSquadPackageDir(projectDir, slug), 'squad.manifest.json');
129
+ }
130
+
131
+ function localSquadDesignDocPath(projectDir, slug) {
132
+ return path.join(localSquadDocsDir(projectDir, slug), 'design-doc.md');
133
+ }
134
+
135
+ function localSquadReadinessPath(projectDir, slug) {
136
+ return path.join(localSquadDocsDir(projectDir, slug), 'readiness.md');
137
+ }
138
+
139
+ function localSquadRulesDocPath(projectDir, slug) {
140
+ return path.join(localSquadDocsDir(projectDir, slug), 'squad-rules.md');
141
+ }
142
+
143
+ function localSquadOutputContractsPath(projectDir, slug) {
144
+ return path.join(localSquadDocsDir(projectDir, slug), 'output-contracts.md');
145
+ }
146
+
147
+ function localLegacySquadTextManifestPath(projectDir, slug) {
148
+ return path.join(localLegacySquadAgentsDir(projectDir, slug), 'agents.md');
149
+ }
150
+
151
+ function localLegacySquadJsonManifestPath(projectDir, slug) {
152
+ return path.join(localLegacySquadAgentsDir(projectDir, slug), 'squad.manifest.json');
153
+ }
154
+
155
+ function localLegacySquadDesignDocPath(projectDir, slug) {
156
+ return path.join(localLegacySquadAgentsDir(projectDir, slug), 'design-doc.md');
157
+ }
158
+
159
+ function localLegacySquadReadinessPath(projectDir, slug) {
160
+ return path.join(localLegacySquadAgentsDir(projectDir, slug), 'readiness.md');
161
+ }
162
+
163
+ function localGenomeFilePath(projectDir, slug) {
164
+ return path.join(projectDir, '.aioson', 'genomes', `${sanitizeSegment(slug, 'genome')}.md`);
165
+ }
166
+
167
+ function findPrimaryHeading(markdown, fallback) {
168
+ const match = String(markdown || '').match(/^#\s+(.+)$/m);
169
+ return match ? String(match[1]).trim() : fallback;
170
+ }
171
+
172
+ function firstParagraph(markdown) {
173
+ const blocks = String(markdown || '')
174
+ .split(/\n\s*\n/)
175
+ .map((block) => block.trim())
176
+ .filter(Boolean);
177
+
178
+ for (const block of blocks) {
179
+ if (block.startsWith('#')) continue;
180
+ return block.replace(/\n+/g, ' ').trim();
181
+ }
182
+
183
+ return null;
184
+ }
185
+
186
+ function extractField(content, ...labels) {
187
+ for (const label of labels) {
188
+ const regex = new RegExp(`^(?:${label}):\\s*(.+)$`, 'im');
189
+ const match = String(content || '').match(regex);
190
+ if (match) return String(match[1]).trim();
191
+ }
192
+ return null;
193
+ }
194
+
195
+ function parseListSection(content, heading) {
196
+ const lines = String(content || '').split(/\r?\n/);
197
+ const startIndex = lines.findIndex((line) => line.trim() === `${heading}:`);
198
+ if (startIndex === -1) return [];
199
+
200
+ const values = [];
201
+ for (let i = startIndex + 1; i < lines.length; i += 1) {
202
+ const line = lines[i];
203
+ if (/^\S.+:$/.test(line.trim())) break;
204
+ const match = line.match(/^\s*-\s+(.+?)\s*$/);
205
+ if (match) values.push(match[1].trim());
206
+ }
207
+ return values;
208
+ }
209
+
210
+ function normalizeRel(relPath) {
211
+ return String(relPath || '')
212
+ .replace(/\\/g, '/')
213
+ .replace(/^\.\//, '')
214
+ .replace(/\/+$/, '');
215
+ }
216
+
217
+ function parseAgentGenomeEntry(entry) {
218
+ const text = String(entry || '').trim();
219
+ const index = text.indexOf(':');
220
+ if (index === -1) return null;
221
+ return {
222
+ agentSlug: normalizeAgentSlug(text.slice(0, index).trim()),
223
+ genomePath: text.slice(index + 1).trim()
224
+ };
225
+ }
226
+
227
+ function guessAgentBody(agent) {
228
+ return (
229
+ agent.promptText ||
230
+ agent.prompt ||
231
+ agent.content ||
232
+ agent.markdown ||
233
+ agent.body ||
234
+ agent.fileContent ||
235
+ null
236
+ );
237
+ }
238
+
239
+ function normalizeAgentSlug(value) {
240
+ return sanitizeSegment(String(value || '').replace(/^@/, ''), 'agent');
241
+ }
242
+
243
+ function normalizeAgentsManifest(agentsManifestJson) {
244
+ if (!agentsManifestJson) return [];
245
+
246
+ const source =
247
+ Array.isArray(agentsManifestJson)
248
+ ? agentsManifestJson
249
+ : Array.isArray(agentsManifestJson.agents)
250
+ ? agentsManifestJson.agents
251
+ : typeof agentsManifestJson === 'object'
252
+ ? Object.entries(agentsManifestJson).map(([key, value]) => ({
253
+ slug: key,
254
+ ...(value && typeof value === 'object' ? value : { content: value })
255
+ }))
256
+ : [];
257
+
258
+ return source
259
+ .map((agent, index) => {
260
+ const slug = normalizeAgentSlug(agent.slug || agent.name || agent.id || `agent-${index + 1}`);
261
+ const title = agent.name || agent.title || agent.roleTitle || slug;
262
+ const description = agent.description || agent.summary || agent.role || null;
263
+ const body = guessAgentBody(agent);
264
+ return {
265
+ slug,
266
+ title: String(title),
267
+ description: description ? String(description) : null,
268
+ body: body ? String(body) : null
269
+ };
270
+ })
271
+ .filter((agent) => Boolean(agent.slug));
272
+ }
273
+
274
+ async function loadLocalSquadManifest(projectDir, slug) {
275
+ const preferredPath = localSquadJsonManifestPath(projectDir, slug);
276
+ const manifestPath = (await exists(preferredPath))
277
+ ? preferredPath
278
+ : localLegacySquadJsonManifestPath(projectDir, slug);
279
+ if (!(await exists(manifestPath))) {
280
+ return null;
281
+ }
282
+
283
+ const raw = await fs.readFile(manifestPath, 'utf8').catch(() => null);
284
+ if (!raw) return null;
285
+
286
+ try {
287
+ return JSON.parse(raw);
288
+ } catch {
289
+ return null;
290
+ }
291
+ }
292
+
293
+ function deriveFallbackSkills(snapshot) {
294
+ const domain = String(snapshot?.squad?.name || snapshot?.squad?.slug || 'squad');
295
+ return [
296
+ {
297
+ slug: 'structured-domain-output',
298
+ title: 'Structured domain output',
299
+ description: `Produce structured outputs for ${domain}.`
300
+ },
301
+ {
302
+ slug: 'critical-synthesis',
303
+ title: 'Critical synthesis',
304
+ description: 'Consolidate specialist reasoning into a practical next step.'
305
+ }
306
+ ];
307
+ }
308
+
309
+ function deriveFallbackMcps(snapshot) {
310
+ const items = [{ slug: 'filesystem', required: true, purpose: 'Persist local drafts, manifests, outputs, logs, and media.' }];
311
+ if ((snapshot?.squad?.visibility || '').toUpperCase() === 'FREE') {
312
+ items.push({ slug: 'web-search', required: false, purpose: 'Optional external research when the task requires current references.' });
313
+ }
314
+ return items;
315
+ }
316
+
317
+ function normalizeContentBlueprints(value) {
318
+ if (!Array.isArray(value)) return [];
319
+
320
+ return value
321
+ .map((item, index) => {
322
+ if (!item || typeof item !== 'object') return null;
323
+
324
+ const slug = sanitizeSegment(item.slug || `blueprint-${index + 1}`, `blueprint-${index + 1}`);
325
+ const contentType = String(item.contentType || 'content').trim() || 'content';
326
+ const layoutType = String(item.layoutType || 'document').trim() || 'document';
327
+ const description = item.description ? String(item.description).trim() : null;
328
+ const sections = Array.isArray(item.sections)
329
+ ? item.sections
330
+ .map((section, sectionIndex) => {
331
+ if (!section || typeof section !== 'object') return null;
332
+
333
+ const key = sanitizeSegment(section.key || `section-${sectionIndex + 1}`, `section-${sectionIndex + 1}`);
334
+ const label = String(section.label || section.key || key).trim();
335
+ const blockTypes = Array.isArray(section.blockTypes)
336
+ ? section.blockTypes.map((blockType) => String(blockType).trim()).filter(Boolean)
337
+ : [];
338
+
339
+ if (!key || !label) return null;
340
+ return { key, label, blockTypes };
341
+ })
342
+ .filter(Boolean)
343
+ : [];
344
+
345
+ return { slug, contentType, layoutType, description, sections };
346
+ })
347
+ .filter(Boolean);
348
+ }
349
+
350
+ function buildBindingsFromAppliedGenomes(appliedGenomes = []) {
351
+ const draft = {
352
+ squad: [],
353
+ executors: {}
354
+ };
355
+
356
+ for (const item of appliedGenomes) {
357
+ const binding = normalizeBinding({
358
+ slug: item?.genome?.slug,
359
+ type: item?.genome?.type || item?.version?.manifestJson?.type,
360
+ source: item?.genome?.sourceKind ? String(item.genome.sourceKind).toLowerCase() : 'cloud',
361
+ priority: item?.priority,
362
+ version: item?.version?.versionNumber,
363
+ evidenceMode:
364
+ item?.version?.manifestJson?.evidenceMode ||
365
+ item?.version?.manifestJson?.evidence_mode ||
366
+ item?.genome?.evidenceMode
367
+ });
368
+ if (!binding) continue;
369
+
370
+ if (String(item?.scopeType || 'SQUAD').toUpperCase() === 'SQUAD') {
371
+ draft.squad.push(binding);
372
+ continue;
373
+ }
374
+
375
+ const executorSlug = normalizeAgentSlug(item?.agentSlug);
376
+ if (!executorSlug) continue;
377
+ draft.executors[executorSlug] = draft.executors[executorSlug] || [];
378
+ draft.executors[executorSlug].push(binding);
379
+ }
380
+
381
+ return normalizeGenomeBindings(draft);
382
+ }
383
+
384
+ function buildLocalSquadManifest(snapshot, agents) {
385
+ const source = snapshot?.version?.manifestJson && typeof snapshot.version.manifestJson === 'object'
386
+ ? snapshot.version.manifestJson
387
+ : {};
388
+ const slug = sanitizeSegment(snapshot.squad.slug, 'squad');
389
+ const explicitMode = typeof source.mode === 'string' ? source.mode : null;
390
+ const mode = String(explicitMode || (source.storagePolicy?.primary === 'files' ? 'builder' : 'content')).trim();
391
+ const sourceContext = source.context && typeof source.context === 'object' ? source.context : {};
392
+ const packageRoot = `.aioson/squads/${slug}`;
393
+ const sourceBindings = mergeGenomeBindings({
394
+ blueprintBindings: source.genomeBindings,
395
+ manifestBindings: source.genomeBindings || source.genomes,
396
+ legacyExecutors: source.executors
397
+ });
398
+ const importedBindings = buildBindingsFromAppliedGenomes(snapshot.appliedGenomes || []);
399
+ const genomeBindings = mergeGenomeBindings({
400
+ blueprintBindings: sourceBindings,
401
+ manifestBindings: importedBindings
402
+ });
403
+ const executorSource = Array.isArray(source.executors) && source.executors.length > 0
404
+ ? source.executors
405
+ : agents.map((agent) => ({
406
+ slug: agent.slug,
407
+ title: agent.title,
408
+ role: agent.description || (agent.slug === 'orquestrador' ? 'Coordinates the squad and publishes the final HTML.' : null),
409
+ file: `${packageRoot}/agents/${agent.slug}.md`,
410
+ skills: agent.slug === 'orquestrador' ? [] : ['structured-domain-output'],
411
+ genomes: resolveExecutorGenomes(agent.slug, genomeBindings)
412
+ }));
413
+ const executors = attachBindingsToExecutors(executorSource, genomeBindings);
414
+
415
+ return {
416
+ schemaVersion: String(source.schemaVersion || snapshot?.version?.schemaVersion || '1.0.0'),
417
+ packageVersion: String(source.packageVersion || snapshot?.version?.versionNumber || '1.0.0'),
418
+ slug,
419
+ name: String(source.name || snapshot.squad.name),
420
+ mode,
421
+ mission: String(source.mission || snapshot.squad.description || `Operate the ${snapshot.squad.name} squad.`),
422
+ goal: String(source.goal || snapshot.squad.goal || snapshot.squad.description || 'Imported from AIOSON Cloud'),
423
+ visibility: String(source.visibility || String(snapshot.squad.visibility || 'PRIVATE').toLowerCase()),
424
+ aiosLiteCompatibility: String(
425
+ source.aiosLiteCompatibility ||
426
+ snapshot?.version?.compatibilityMin ||
427
+ '^1.1.0'
428
+ ),
429
+ rules: {
430
+ outputsDir: `output/${slug}`,
431
+ logsDir: `aioson-logs/${slug}`,
432
+ mediaDir: `media/${slug}`,
433
+ reviewPolicy: Array.isArray(source?.rules?.reviewPolicy)
434
+ ? source.rules.reviewPolicy
435
+ : ['clarity', 'density', 'consistency', 'next-step']
436
+ },
437
+ storagePolicy:
438
+ source.storagePolicy && typeof source.storagePolicy === 'object'
439
+ ? source.storagePolicy
440
+ : {
441
+ primary: mode === 'builder' ? 'files' : 'sqlite',
442
+ artifacts: mode === 'builder' ? 'files+sqlite' : 'sqlite-json',
443
+ exports: { html: true, markdown: true, json: true }
444
+ },
445
+ package: {
446
+ rootDir: packageRoot,
447
+ agentsDir: `${packageRoot}/agents`,
448
+ skillsDir: `${packageRoot}/skills`,
449
+ templatesDir: `${packageRoot}/templates`,
450
+ docsDir: `${packageRoot}/docs`
451
+ },
452
+ baseRoles: Array.isArray(source.baseRoles) && source.baseRoles.length > 0
453
+ ? source.baseRoles
454
+ : ['orchestrator', 'discovery-lead', 'design-doc-lead', 'planner', 'implementer', 'reviewer', 'docs-maintainer'],
455
+ skills: Array.isArray(source.skills) && source.skills.length > 0 ? source.skills : deriveFallbackSkills(snapshot),
456
+ mcps: Array.isArray(source.mcps) && source.mcps.length > 0 ? source.mcps : deriveFallbackMcps(snapshot),
457
+ subagents: source.subagents && typeof source.subagents === 'object'
458
+ ? source.subagents
459
+ : {
460
+ allowed: true,
461
+ when: ['broad research', 'comparison', 'large-context summarization', 'parallel analysis']
462
+ },
463
+ contentBlueprints: normalizeContentBlueprints(source.contentBlueprints),
464
+ context: {
465
+ mode: String(
466
+ sourceContext.mode ||
467
+ (snapshot?.version?.designDocMarkdown && /feature mode|modo feature/i.test(snapshot.version.designDocMarkdown)
468
+ ? 'feature'
469
+ : 'project')
470
+ ),
471
+ summary: String(
472
+ sourceContext.summary ||
473
+ source.goal ||
474
+ snapshot.squad.goal ||
475
+ snapshot.squad.description ||
476
+ 'Imported squad context.'
477
+ ),
478
+ designDocPath: `${packageRoot}/docs/design-doc.md`,
479
+ readinessPath: `${packageRoot}/docs/readiness.md`,
480
+ docsPackage: Array.isArray(sourceContext.docsPackage)
481
+ ? sourceContext.docsPackage
482
+ : ['project.context.md', 'design-doc.md', 'readiness.md'],
483
+ readiness: sourceContext.readiness && typeof sourceContext.readiness === 'object'
484
+ ? sourceContext.readiness
485
+ : null
486
+ },
487
+ executors,
488
+ genomes: genomeBindings,
489
+ genomeBindings
490
+ };
491
+ }
492
+
493
+ function buildSquadTextManifest(snapshot, manifest) {
494
+ const lines = [
495
+ `# Squad ${manifest.name}`,
496
+ '',
497
+ '## Mission',
498
+ manifest.mission,
499
+ '',
500
+ '## Does',
501
+ `- Deliver outputs for the domain: ${snapshot.squad.name}`,
502
+ `- Target goal: ${manifest.goal}`,
503
+ '- Coordinate specialists through the local orchestrator',
504
+ '',
505
+ '## Does not do',
506
+ '- Replace the AIOSON official agents',
507
+ '- Use subagents as a substitute for permanent executors or skills',
508
+ '',
509
+ '## Permanent executors'
510
+ ];
511
+
512
+ for (const executor of manifest.executors || []) {
513
+ lines.push(`- @${executor.slug} — ${executor.role || executor.title || 'Specialist executor'}`);
514
+ }
515
+
516
+ lines.push('', '## Squad skills');
517
+ for (const skill of manifest.skills || []) {
518
+ lines.push(`- ${skill.slug} — ${skill.description || skill.title || 'Reusable capability'}`);
519
+ }
520
+
521
+ lines.push('', '## Squad MCPs');
522
+ for (const mcp of manifest.mcps || []) {
523
+ lines.push(`- ${mcp.slug} — ${mcp.purpose || 'External integration'}`);
524
+ }
525
+
526
+ lines.push(
527
+ '',
528
+ '## Subagent policy',
529
+ '- Use subagents only for isolated investigation, broad reading, comparison, or parallel work.',
530
+ '- Do not use subagents as a substitute for permanent executors or reusable skills.',
531
+ '',
532
+ '## Outputs and review',
533
+ `- Drafts: \`output/${manifest.slug}/\``,
534
+ `- Final HTML: \`output/${manifest.slug}/{session-id}.html\``,
535
+ `- Logs: \`aioson-logs/${manifest.slug}/\``,
536
+ `- Media: \`media/${manifest.slug}/\``,
537
+ `- Package root: \`.aioson/squads/${manifest.slug}/\``,
538
+ `- Design doc: \`.aioson/squads/${manifest.slug}/docs/design-doc.md\``,
539
+ `- Readiness: \`.aioson/squads/${manifest.slug}/docs/readiness.md\``,
540
+ '- Final outputs should include recommendation, reasoning, tradeoff, and next step.'
541
+ );
542
+
543
+ return `${lines.join('\n')}\n`;
544
+ }
545
+
546
+ function buildAgentStub(snapshot, agent) {
547
+ const lines = [
548
+ `# ${agent.title}`,
549
+ '',
550
+ '> Imported from AIOSON Cloud.',
551
+ '',
552
+ '## Origin',
553
+ '',
554
+ `- Squad: ${snapshot.squad.name}`,
555
+ `- Slug: ${snapshot.squad.slug}`,
556
+ `- Version: ${snapshot.version.versionNumber}`,
557
+ `- Owner: ${snapshot.squad.ownerUsername}`
558
+ ];
559
+
560
+ if (agent.description) {
561
+ lines.push('', '## Role', '', agent.description);
562
+ }
563
+
564
+ lines.push(
565
+ '',
566
+ '## Import Notice',
567
+ '',
568
+ 'The cloud snapshot did not include a full prompt body for this agent.',
569
+ 'Regenerate or enrich this agent locally before using it as a production specialist.'
570
+ );
571
+
572
+ return `${lines.join('\n')}\n`;
573
+ }
574
+
575
+ function buildSquadMetadata(snapshot, options = {}) {
576
+ const slug = sanitizeSegment(snapshot.squad.slug, 'squad');
577
+ const installedAt = new Date().toISOString();
578
+ const packageRoot = `.aioson/squads/${slug}`;
579
+ const lines = [
580
+ `Squad: ${snapshot.squad.name}`,
581
+ `Mode: ${snapshot?.version?.manifestJson?.mode || 'CloudImport'}`,
582
+ `Goal: ${snapshot.squad.goal || snapshot.squad.description || 'Imported from AIOSON Cloud'}`,
583
+ `Package: ${packageRoot}/`,
584
+ `Agents: ${packageRoot}/agents/`,
585
+ `Skills: ${packageRoot}/skills/`,
586
+ `Templates: ${packageRoot}/templates/`,
587
+ `Docs: ${packageRoot}/docs/`,
588
+ `Output: output/${slug}/`,
589
+ `Logs: aioson-logs/${slug}/`,
590
+ `Media: media/${slug}/`,
591
+ `DesignDoc: ${packageRoot}/docs/design-doc.md`,
592
+ `Readiness: ${packageRoot}/docs/readiness.md`,
593
+ `LatestSession: output/${slug}/latest.html`,
594
+ `SourceUrl: ${options.sourceUrl || '—'}`,
595
+ `SourceVersion: ${snapshot.version.versionNumber}`,
596
+ `ImportedAt: ${installedAt}`,
597
+ '',
598
+ 'Genomes:'
599
+ ];
600
+
601
+ const shared = Array.isArray(snapshot.appliedGenomes)
602
+ ? snapshot.appliedGenomes.filter((item) => String(item.scopeType || 'SQUAD').toUpperCase() === 'SQUAD')
603
+ : [];
604
+ for (const genome of shared) {
605
+ lines.push(`- .aioson/genomes/${sanitizeSegment(genome.genome.slug, 'genome')}.md`);
606
+ }
607
+
608
+ lines.push('', 'AgentGenomes:');
609
+ const scoped = Array.isArray(snapshot.appliedGenomes)
610
+ ? snapshot.appliedGenomes.filter(
611
+ (item) => String(item.scopeType || 'SQUAD').toUpperCase() !== 'SQUAD' && item.agentSlug
612
+ )
613
+ : [];
614
+
615
+ for (const genome of scoped) {
616
+ lines.push(
617
+ `- ${normalizeAgentSlug(genome.agentSlug)}: .aioson/genomes/${sanitizeSegment(genome.genome.slug, 'genome')}.md`
618
+ );
619
+ }
620
+
621
+ return `${lines.join('\n')}\n`;
622
+ }
623
+
624
+ function buildInstalledManifest(snapshot, sourceUrl, agents) {
625
+ return {
626
+ kind: 'aiosforge.local-installed-squad',
627
+ installVersion: 1,
628
+ installedAt: new Date().toISOString(),
629
+ sourceUrl,
630
+ squad: snapshot.squad,
631
+ version: {
632
+ versionNumber: snapshot.version.versionNumber,
633
+ compatibilityMin: snapshot.version.compatibilityMin,
634
+ compatibilityMax: snapshot.version.compatibilityMax,
635
+ schemaVersion: snapshot.version.schemaVersion
636
+ },
637
+ packageRoot: `.aioson/squads/${sanitizeSegment(snapshot.squad.slug, 'squad')}`,
638
+ agents: agents.map((agent) => ({
639
+ slug: agent.slug,
640
+ title: agent.title,
641
+ hasBody: Boolean(agent.body)
642
+ })),
643
+ appliedGenomes: snapshot.appliedGenomes || []
644
+ };
645
+ }
646
+
647
+ function buildSquadDesignDoc(snapshot, manifest) {
648
+ if (snapshot?.version?.designDocMarkdown) {
649
+ const content = String(snapshot.version.designDocMarkdown);
650
+ return content.endsWith('\n') ? content : `${content}\n`;
651
+ }
652
+
653
+ return [
654
+ `# Design Doc - ${manifest.name}`,
655
+ '',
656
+ '## Context and motivation',
657
+ snapshot.squad.description || manifest.goal || 'Imported squad context.',
658
+ '',
659
+ '## Objective',
660
+ manifest.goal || 'Operate the imported squad for the target domain.',
661
+ '',
662
+ '## Scope',
663
+ '- Materialize the squad locally with manifest, executors, outputs, logs, and media.',
664
+ '- Preserve the operational and cognitive blueprint published in the cloud.',
665
+ '',
666
+ '## Out of scope',
667
+ '- Rewriting imported executors automatically beyond the published snapshot.',
668
+ '- Inventing undocumented MCPs or genomes.'
669
+ ].join('\n') + '\n';
670
+ }
671
+
672
+ function buildSquadReadiness(snapshot, manifest) {
673
+ if (snapshot?.version?.readinessMarkdown) {
674
+ const content = String(snapshot.version.readinessMarkdown);
675
+ return content.endsWith('\n') ? content : `${content}\n`;
676
+ }
677
+
678
+ const readiness = manifest?.context?.readiness && typeof manifest.context.readiness === 'object'
679
+ ? manifest.context.readiness
680
+ : {
681
+ level: 'medium',
682
+ totalScore: 15,
683
+ maxScore: 25
684
+ };
685
+
686
+ return [
687
+ `# Readiness - ${manifest.name}`,
688
+ '',
689
+ `- Readiness score total: ${readiness.totalScore ?? 15}`,
690
+ `- Readiness score maximo: ${readiness.maxScore ?? 25}`,
691
+ `- Readiness level: ${readiness.level || 'medium'}`,
692
+ '',
693
+ '## What is already clear',
694
+ '- The squad structure and executors are defined in the imported snapshot.',
695
+ '- Outputs, logs, media, and genomes can be materialized locally.',
696
+ '',
697
+ '## What is still missing',
698
+ '- Local project-specific refinements after import.',
699
+ '- Any additional feature context not present in the published snapshot.'
700
+ ].join('\n') + '\n';
701
+ }
702
+
703
+ function buildSkillMarkdown(skill) {
704
+ const title = String(skill?.title || skill?.slug || 'Skill').trim();
705
+ const description = skill?.description ? String(skill.description).trim() : 'Reusable squad capability.';
706
+ return [`# ${title}`, '', description, ''].join('\n');
707
+ }
708
+
709
+ function buildTemplateJson(type, slug, manifest) {
710
+ if (type === 'discovery') {
711
+ return {
712
+ squad: manifest.slug,
713
+ mode: manifest.mode,
714
+ summary: '',
715
+ goals: [],
716
+ constraints: [],
717
+ references: []
718
+ };
719
+ }
720
+
721
+ if (type === 'design-doc') {
722
+ return {
723
+ squad: manifest.slug,
724
+ mode: manifest.mode,
725
+ problem: '',
726
+ scope: [],
727
+ outOfScope: [],
728
+ deliverables: []
729
+ };
730
+ }
731
+
732
+ return {
733
+ squad: manifest.slug,
734
+ blueprint: slug,
735
+ contentType: 'content',
736
+ layoutType: 'document',
737
+ title: '',
738
+ blocks: []
739
+ };
740
+ }
741
+
742
+ function buildRulesDoc(manifest) {
743
+ return [
744
+ `# Squad Rules - ${manifest.name}`,
745
+ '',
746
+ '## Mission',
747
+ manifest.mission,
748
+ '',
749
+ '## Operating mode',
750
+ `- Mode: ${manifest.mode}`,
751
+ '- Keep agents light and load skills on demand.',
752
+ '- Use discovery and design doc before implementation or heavy generation.',
753
+ '',
754
+ '## Persistence',
755
+ `- Primary: ${manifest.storagePolicy?.primary || 'sqlite'}`,
756
+ `- Artifacts: ${manifest.storagePolicy?.artifacts || 'sqlite-json'}`,
757
+ ''
758
+ ].join('\n');
759
+ }
760
+
761
+ function buildOutputContractsDoc(manifest) {
762
+ const blueprints = Array.isArray(manifest.contentBlueprints) ? manifest.contentBlueprints : [];
763
+ const lines = [
764
+ `# Output Contracts - ${manifest.name}`,
765
+ '',
766
+ '## Persistence rule',
767
+ `- Mode: ${manifest.mode}`,
768
+ `- Primary storage: ${manifest.storagePolicy?.primary || 'sqlite'}`,
769
+ '',
770
+ '## Blueprints'
771
+ ];
772
+
773
+ if (blueprints.length === 0) {
774
+ lines.push('- No explicit content blueprints were declared.');
775
+ } else {
776
+ for (const blueprint of blueprints) {
777
+ lines.push(`- ${blueprint.slug} (${blueprint.contentType} / ${blueprint.layoutType})`);
778
+ }
779
+ }
780
+
781
+ lines.push('');
782
+ return lines.join('\n');
783
+ }
784
+
785
+ async function materializeImportedSquad(projectDir, payload, sourceUrl, force) {
786
+ const slug = sanitizeSegment(payload.squad.slug, 'squad');
787
+ const agents = normalizeAgentsManifest(payload.version.agentsManifestJson);
788
+ const squadManifest = buildLocalSquadManifest(payload, agents);
789
+ const packageDir = localSquadPackageDir(projectDir, slug);
790
+ const metadataPath = localSquadSummaryPath(projectDir, slug);
791
+ const agentsDir = localSquadAgentsDir(projectDir, slug);
792
+ const skillsDir = localSquadSkillsDir(projectDir, slug);
793
+ const templatesDir = localSquadTemplatesDir(projectDir, slug);
794
+ const docsDir = localSquadDocsDir(projectDir, slug);
795
+ const textManifestPath = localSquadTextManifestPath(projectDir, slug);
796
+ const jsonManifestPath = localSquadJsonManifestPath(projectDir, slug);
797
+ const outputDir = localSquadOutputDir(projectDir, slug);
798
+ const logsDir = localSquadLogsDir(projectDir, slug);
799
+ const mediaDir = localSquadMediaDir(projectDir, slug);
800
+ const designDocPath = localSquadDesignDocPath(projectDir, slug);
801
+ const readinessPath = localSquadReadinessPath(projectDir, slug);
802
+ const rulesDocPath = localSquadRulesDocPath(projectDir, slug);
803
+ const outputContractsPath = localSquadOutputContractsPath(projectDir, slug);
804
+ const manifestPath = installedManifestPath(projectDir, slug);
805
+
806
+ if (!force && (await exists(metadataPath))) {
807
+ throw new Error(`Imported squad already materialized: ${metadataPath}`);
808
+ }
809
+
810
+ await ensureDir(packageDir);
811
+ await ensureDir(agentsDir);
812
+ await ensureDir(skillsDir);
813
+ await ensureDir(templatesDir);
814
+ await ensureDir(docsDir);
815
+ await ensureDir(outputDir);
816
+ await ensureDir(logsDir);
817
+ await ensureDir(mediaDir);
818
+ await ensureDir(path.dirname(manifestPath));
819
+
820
+ const metadata = buildSquadMetadata(payload, { sourceUrl });
821
+ await fs.writeFile(metadataPath, metadata, 'utf8');
822
+ await fs.writeFile(textManifestPath, buildSquadTextManifest(payload, squadManifest), 'utf8');
823
+ await fs.writeFile(jsonManifestPath, `${JSON.stringify(squadManifest, null, 2)}\n`, 'utf8');
824
+ await fs.writeFile(designDocPath, buildSquadDesignDoc(payload, squadManifest), 'utf8');
825
+ await fs.writeFile(readinessPath, buildSquadReadiness(payload, squadManifest), 'utf8');
826
+ await fs.writeFile(rulesDocPath, buildRulesDoc(squadManifest), 'utf8');
827
+ await fs.writeFile(outputContractsPath, buildOutputContractsDoc(squadManifest), 'utf8');
828
+ await fs.writeFile(
829
+ manifestPath,
830
+ `${JSON.stringify(buildInstalledManifest(payload, sourceUrl, agents), null, 2)}\n`,
831
+ 'utf8'
832
+ );
833
+
834
+ const writtenAgents = [];
835
+ for (const agent of agents) {
836
+ const filePath = path.join(agentsDir, `${agent.slug}.md`);
837
+ const body = agent.body || buildAgentStub(payload, agent);
838
+ await fs.writeFile(filePath, body.endsWith('\n') ? body : `${body}\n`, 'utf8');
839
+ writtenAgents.push(filePath);
840
+ }
841
+
842
+ const writtenSkills = [];
843
+ for (const skill of squadManifest.skills || []) {
844
+ const filePath = path.join(skillsDir, `${sanitizeSegment(skill.slug, 'skill')}.md`);
845
+ await fs.writeFile(filePath, buildSkillMarkdown(skill), 'utf8');
846
+ writtenSkills.push(filePath);
847
+ }
848
+
849
+ const templateEntries = [
850
+ { slug: 'discovery', filePath: path.join(templatesDir, 'discovery.template.json') },
851
+ { slug: 'design-doc', filePath: path.join(templatesDir, 'design-doc.template.json') },
852
+ { slug: 'artifact', filePath: path.join(templatesDir, 'artifact.template.json') }
853
+ ];
854
+ const writtenTemplates = [];
855
+ for (const template of templateEntries) {
856
+ await fs.writeFile(
857
+ template.filePath,
858
+ `${JSON.stringify(buildTemplateJson(template.slug, template.slug, squadManifest), null, 2)}\n`,
859
+ 'utf8'
860
+ );
861
+ writtenTemplates.push(template.filePath);
862
+ }
863
+
864
+ const writtenGenomes = [];
865
+ for (const genome of payload.appliedGenomes || []) {
866
+ const genomePath = localGenomeFilePath(projectDir, genome.genome.slug);
867
+ if (!force && (await exists(genomePath))) continue;
868
+ await ensureDir(path.dirname(genomePath));
869
+ const content = genome.version.contentMarkdown
870
+ ? String(genome.version.contentMarkdown)
871
+ : [
872
+ `# ${genome.genome.name}`,
873
+ '',
874
+ '> Imported from AIOSON Cloud.',
875
+ '',
876
+ `Version: ${genome.version.versionNumber}`,
877
+ `Scope: ${genome.scopeType}`,
878
+ genome.agentSlug ? `Agent: ${normalizeAgentSlug(genome.agentSlug)}` : null
879
+ ]
880
+ .filter(Boolean)
881
+ .join('\n');
882
+ await fs.writeFile(genomePath, content.endsWith('\n') ? content : `${content}\n`, 'utf8');
883
+ writtenGenomes.push(genomePath);
884
+ }
885
+
886
+ const runtimeHandle = await openRuntimeDb(projectDir);
887
+ try {
888
+ upsertSquadManifest(runtimeHandle.db, {
889
+ slug,
890
+ name: squadManifest.name,
891
+ mode: squadManifest.mode,
892
+ mission: squadManifest.mission,
893
+ goal: squadManifest.goal,
894
+ visibility: squadManifest.visibility,
895
+ status: 'active',
896
+ manifest: squadManifest,
897
+ context: squadManifest.context,
898
+ packageDir: `.aioson/squads/${slug}`,
899
+ agentsDir: `.aioson/squads/${slug}/agents`,
900
+ outputDir: `output/${slug}`,
901
+ logsDir: `aioson-logs/${slug}`,
902
+ mediaDir: `media/${slug}`,
903
+ latestSessionPath: `output/${slug}/latest.html`
904
+ });
905
+ } finally {
906
+ runtimeHandle.db.close();
907
+ }
908
+
909
+ return {
910
+ metadataPath,
911
+ textManifestPath,
912
+ jsonManifestPath,
913
+ designDocPath,
914
+ readinessPath,
915
+ rulesDocPath,
916
+ outputContractsPath,
917
+ manifestPath,
918
+ packageDir,
919
+ agentsDir,
920
+ skillsDir,
921
+ templatesDir,
922
+ docsDir,
923
+ outputDir,
924
+ logsDir,
925
+ mediaDir,
926
+ writtenAgents,
927
+ writtenSkills,
928
+ writtenTemplates,
929
+ writtenGenomes
930
+ };
931
+ }
932
+
933
+ async function ensureProjectDir(targetDir, t) {
934
+ const absolute = path.resolve(process.cwd(), targetDir || '.');
935
+ const stat = await fs.stat(absolute).catch(() => null);
936
+ if (!stat || !stat.isDirectory()) {
937
+ throw new Error(t('cloud.project_missing', { path: absolute }));
938
+ }
939
+ return absolute;
940
+ }
941
+
942
+ async function fetchJson(url) {
943
+ const response = await fetch(url, {
944
+ headers: {
945
+ accept: 'application/json'
946
+ }
947
+ });
948
+
949
+ const text = await response.text();
950
+ let parsed = null;
951
+
952
+ try {
953
+ parsed = text ? JSON.parse(text) : null;
954
+ } catch {
955
+ parsed = null;
956
+ }
957
+
958
+ if (!response.ok) {
959
+ const detail =
960
+ parsed && typeof parsed === 'object' && parsed.error ? String(parsed.error) : `${response.status} ${response.statusText}`;
961
+ throw new Error(`HTTP ${response.status}: ${detail}`);
962
+ }
963
+
964
+ if (!parsed || typeof parsed !== 'object') {
965
+ throw new Error('Invalid JSON response.');
966
+ }
967
+
968
+ return parsed;
969
+ }
970
+
971
+ async function postJson(url, payload, fetchImpl = fetch) {
972
+ const response = await fetchImpl(url, {
973
+ method: 'POST',
974
+ headers: {
975
+ accept: 'application/json',
976
+ 'content-type': 'application/json'
977
+ },
978
+ body: JSON.stringify(payload)
979
+ });
980
+
981
+ const text = await response.text();
982
+ let parsed = null;
983
+
984
+ try {
985
+ parsed = text ? JSON.parse(text) : null;
986
+ } catch {
987
+ parsed = null;
988
+ }
989
+
990
+ if (!response.ok) {
991
+ const detail =
992
+ parsed && typeof parsed === 'object' && parsed.error ? String(parsed.error) : `${response.status} ${response.statusText}`;
993
+ throw new Error(`HTTP ${response.status}: ${detail}`);
994
+ }
995
+
996
+ if (!parsed || typeof parsed !== 'object') {
997
+ throw new Error('Invalid JSON response.');
998
+ }
999
+
1000
+ return parsed;
1001
+ }
1002
+
1003
+ function validateSquadSnapshot(payload) {
1004
+ if (!payload || typeof payload !== 'object') {
1005
+ throw new Error('Invalid squad snapshot payload.');
1006
+ }
1007
+
1008
+ if (payload.kind !== 'aiosforge.squad') {
1009
+ throw new Error('Unsupported snapshot kind.');
1010
+ }
1011
+
1012
+ if (payload.exportVersion !== 1) {
1013
+ throw new Error('Unsupported snapshot export version.');
1014
+ }
1015
+
1016
+ if (!payload.squad || typeof payload.squad !== 'object' || !payload.squad.slug) {
1017
+ throw new Error('Snapshot is missing squad metadata.');
1018
+ }
1019
+
1020
+ if (!payload.version || typeof payload.version !== 'object' || !payload.version.versionNumber) {
1021
+ throw new Error('Snapshot is missing version metadata.');
1022
+ }
1023
+
1024
+ return {
1025
+ slug: String(payload.squad.slug),
1026
+ versionNumber: String(payload.version.versionNumber)
1027
+ };
1028
+ }
1029
+
1030
+ function validateGenomeSnapshot(payload) {
1031
+ if (!payload || typeof payload !== 'object') {
1032
+ throw new Error('Invalid genome snapshot payload.');
1033
+ }
1034
+
1035
+ if (payload.kind !== 'aiosforge.genome') {
1036
+ throw new Error('Unsupported genome snapshot kind.');
1037
+ }
1038
+
1039
+ if (payload.exportVersion !== 1) {
1040
+ throw new Error('Unsupported genome snapshot export version.');
1041
+ }
1042
+
1043
+ if (!payload.genome || typeof payload.genome !== 'object' || !payload.genome.slug) {
1044
+ throw new Error('Snapshot is missing genome metadata.');
1045
+ }
1046
+
1047
+ if (!payload.version || typeof payload.version !== 'object' || !payload.version.versionNumber) {
1048
+ throw new Error('Snapshot is missing genome version metadata.');
1049
+ }
1050
+
1051
+ return {
1052
+ slug: String(payload.genome.slug),
1053
+ versionNumber: String(payload.version.versionNumber)
1054
+ };
1055
+ }
1056
+
1057
+ async function writeImportSnapshot(projectDir, payload, slug, versionNumber, force) {
1058
+ const latestPath = squadImportFilePath(projectDir, slug, versionNumber);
1059
+ const archivePath = historyImportFilePath(projectDir, slug, versionNumber);
1060
+
1061
+ if (!force && (await exists(latestPath))) {
1062
+ throw new Error(`Snapshot already exists: ${latestPath}`);
1063
+ }
1064
+
1065
+ await ensureDir(path.dirname(latestPath));
1066
+ await ensureDir(path.dirname(archivePath));
1067
+ const json = `${JSON.stringify(payload, null, 2)}\n`;
1068
+ await fs.writeFile(latestPath, json, 'utf8');
1069
+ await fs.writeFile(archivePath, json, 'utf8');
1070
+
1071
+ return {
1072
+ latestPath,
1073
+ archivePath
1074
+ };
1075
+ }
1076
+
1077
+ async function writeGenomeImportSnapshot(projectDir, payload, slug, versionNumber, force) {
1078
+ const latestPath = genomeImportFilePath(projectDir, slug, versionNumber);
1079
+ const archivePath = genomeHistoryImportFilePath(projectDir, slug, versionNumber);
1080
+
1081
+ if (!force && (await exists(latestPath))) {
1082
+ throw new Error(`Snapshot already exists: ${latestPath}`);
1083
+ }
1084
+
1085
+ await ensureDir(path.dirname(latestPath));
1086
+ await ensureDir(path.dirname(archivePath));
1087
+ const json = `${JSON.stringify(payload, null, 2)}\n`;
1088
+ await fs.writeFile(latestPath, json, 'utf8');
1089
+ await fs.writeFile(archivePath, json, 'utf8');
1090
+
1091
+ return {
1092
+ latestPath,
1093
+ archivePath
1094
+ };
1095
+ }
1096
+
1097
+ function buildInstalledGenomeManifest(snapshot, sourceUrl) {
1098
+ return {
1099
+ kind: 'aiosforge.local-installed-genome',
1100
+ installVersion: 1,
1101
+ installedAt: new Date().toISOString(),
1102
+ sourceUrl,
1103
+ genome: snapshot.genome,
1104
+ version: {
1105
+ versionNumber: snapshot.version.versionNumber,
1106
+ schemaVersion: snapshot.version.schemaVersion
1107
+ }
1108
+ };
1109
+ }
1110
+
1111
+ async function materializeImportedGenome(projectDir, payload, sourceUrl, force) {
1112
+ const slug = sanitizeSegment(payload.genome.slug, 'genome');
1113
+ const genomePath = localGenomeFilePath(projectDir, slug);
1114
+ const manifestPath = installedGenomeManifestPath(projectDir, slug);
1115
+
1116
+ if (!force && (await exists(genomePath))) {
1117
+ throw new Error(`Imported genome already materialized: ${genomePath}`);
1118
+ }
1119
+
1120
+ await ensureDir(path.dirname(genomePath));
1121
+ await ensureDir(path.dirname(manifestPath));
1122
+
1123
+ const content = payload.version.contentMarkdown
1124
+ ? String(payload.version.contentMarkdown)
1125
+ : [
1126
+ `# ${payload.genome.name}`,
1127
+ '',
1128
+ '> Imported from AIOSON Cloud.',
1129
+ '',
1130
+ `Version: ${payload.version.versionNumber}`,
1131
+ payload.version.summary ? '' : null,
1132
+ payload.version.summary || null
1133
+ ]
1134
+ .filter(Boolean)
1135
+ .join('\n');
1136
+
1137
+ await fs.writeFile(genomePath, content.endsWith('\n') ? content : `${content}\n`, 'utf8');
1138
+ await fs.writeFile(
1139
+ manifestPath,
1140
+ `${JSON.stringify(buildInstalledGenomeManifest(payload, sourceUrl), null, 2)}\n`,
1141
+ 'utf8'
1142
+ );
1143
+
1144
+ return {
1145
+ genomePath,
1146
+ manifestPath
1147
+ };
1148
+ }
1149
+
1150
+ async function runCloudImportSquad({ args, options = {}, logger, t }) {
1151
+ const projectDir = await ensureProjectDir(args[0] || '.', t);
1152
+ const url = String(options.url || '').trim();
1153
+ if (!url) {
1154
+ throw new Error(t('cloud.url_required'));
1155
+ }
1156
+
1157
+ const dryRun = Boolean(options['dry-run']);
1158
+ const force = Boolean(options.force);
1159
+ const payload = await fetchJson(url);
1160
+ const snapshot = validateSquadSnapshot(payload);
1161
+
1162
+ const planned = {
1163
+ ok: true,
1164
+ resource: 'squad',
1165
+ url,
1166
+ dryRun,
1167
+ force,
1168
+ projectDir,
1169
+ slug: snapshot.slug,
1170
+ versionNumber: snapshot.versionNumber,
1171
+ importDir: path.dirname(squadImportFilePath(projectDir, snapshot.slug, snapshot.versionNumber)),
1172
+ latestFile: squadImportFilePath(projectDir, snapshot.slug, snapshot.versionNumber),
1173
+ materialized: !Boolean(options['snapshots-only'])
1174
+ };
1175
+
1176
+ if (dryRun) {
1177
+ logger.log(
1178
+ t('cloud.import_squad_dry_run', {
1179
+ slug: snapshot.slug,
1180
+ version: snapshot.versionNumber
1181
+ })
1182
+ );
1183
+ return planned;
1184
+ }
1185
+
1186
+ const written = await writeImportSnapshot(projectDir, payload, snapshot.slug, snapshot.versionNumber, force);
1187
+ let materialized = null;
1188
+ if (!options['snapshots-only']) {
1189
+ materialized = await materializeImportedSquad(projectDir, payload, url, force);
1190
+ }
1191
+ logger.log(
1192
+ t('cloud.import_squad_done', {
1193
+ slug: snapshot.slug,
1194
+ version: snapshot.versionNumber
1195
+ })
1196
+ );
1197
+
1198
+ return {
1199
+ ...planned,
1200
+ latestFile: written.latestPath,
1201
+ archiveFile: written.archivePath,
1202
+ relativeLatestFile: toRelativeSafe(projectDir, written.latestPath),
1203
+ relativeArchiveFile: toRelativeSafe(projectDir, written.archivePath),
1204
+ materializedMetadataFile: materialized ? toRelativeSafe(projectDir, materialized.metadataPath) : null,
1205
+ materializedPackageDir: materialized ? toRelativeSafe(projectDir, materialized.packageDir) : null,
1206
+ materializedManifestFile: materialized ? toRelativeSafe(projectDir, materialized.manifestPath) : null,
1207
+ materializedAgentsDir: materialized ? toRelativeSafe(projectDir, materialized.agentsDir) : null,
1208
+ materializedSkillsDir: materialized ? toRelativeSafe(projectDir, materialized.skillsDir) : null,
1209
+ materializedTemplatesDir: materialized ? toRelativeSafe(projectDir, materialized.templatesDir) : null,
1210
+ materializedDocsDir: materialized ? toRelativeSafe(projectDir, materialized.docsDir) : null,
1211
+ materializedOutputDir: materialized ? toRelativeSafe(projectDir, materialized.outputDir) : null,
1212
+ materializedLogsDir: materialized ? toRelativeSafe(projectDir, materialized.logsDir) : null,
1213
+ materializedDesignDocFile: materialized ? toRelativeSafe(projectDir, materialized.designDocPath) : null,
1214
+ materializedReadinessFile: materialized ? toRelativeSafe(projectDir, materialized.readinessPath) : null,
1215
+ writtenAgents: materialized ? materialized.writtenAgents.map((file) => toRelativeSafe(projectDir, file)) : [],
1216
+ writtenSkills: materialized ? materialized.writtenSkills.map((file) => toRelativeSafe(projectDir, file)) : [],
1217
+ writtenTemplates: materialized ? materialized.writtenTemplates.map((file) => toRelativeSafe(projectDir, file)) : [],
1218
+ writtenGenomes: materialized ? materialized.writtenGenomes.map((file) => toRelativeSafe(projectDir, file)) : []
1219
+ };
1220
+ }
1221
+
1222
+ async function runCloudImportGenome({ args, options = {}, logger, t }) {
1223
+ const projectDir = await ensureProjectDir(args[0] || '.', t);
1224
+ const url = String(options.url || '').trim();
1225
+ if (!url) {
1226
+ throw new Error(t('cloud.url_required'));
1227
+ }
1228
+
1229
+ const dryRun = Boolean(options['dry-run']);
1230
+ const force = Boolean(options.force);
1231
+ const payload = await fetchJson(url);
1232
+ const snapshot = validateGenomeSnapshot(payload);
1233
+
1234
+ const planned = {
1235
+ ok: true,
1236
+ resource: 'genome',
1237
+ url,
1238
+ dryRun,
1239
+ force,
1240
+ projectDir,
1241
+ slug: snapshot.slug,
1242
+ versionNumber: snapshot.versionNumber,
1243
+ importDir: path.dirname(genomeImportFilePath(projectDir, snapshot.slug, snapshot.versionNumber)),
1244
+ latestFile: genomeImportFilePath(projectDir, snapshot.slug, snapshot.versionNumber),
1245
+ materialized: !Boolean(options['snapshots-only'])
1246
+ };
1247
+
1248
+ if (dryRun) {
1249
+ logger.log(
1250
+ t('cloud.import_genome_dry_run', {
1251
+ slug: snapshot.slug,
1252
+ version: snapshot.versionNumber
1253
+ })
1254
+ );
1255
+ return planned;
1256
+ }
1257
+
1258
+ const written = await writeGenomeImportSnapshot(projectDir, payload, snapshot.slug, snapshot.versionNumber, force);
1259
+ let materialized = null;
1260
+ if (!options['snapshots-only']) {
1261
+ materialized = await materializeImportedGenome(projectDir, payload, url, force);
1262
+ }
1263
+
1264
+ logger.log(
1265
+ t('cloud.import_genome_done', {
1266
+ slug: snapshot.slug,
1267
+ version: snapshot.versionNumber
1268
+ })
1269
+ );
1270
+
1271
+ return {
1272
+ ...planned,
1273
+ latestFile: written.latestPath,
1274
+ archiveFile: written.archivePath,
1275
+ relativeLatestFile: toRelativeSafe(projectDir, written.latestPath),
1276
+ relativeArchiveFile: toRelativeSafe(projectDir, written.archivePath),
1277
+ materializedGenomeFile: materialized ? toRelativeSafe(projectDir, materialized.genomePath) : null,
1278
+ materializedManifestFile: materialized ? toRelativeSafe(projectDir, materialized.manifestPath) : null
1279
+ };
1280
+ }
1281
+
1282
+ async function loadLocalGenomeSnapshot(projectDir, slug, options = {}) {
1283
+ const filePath = localGenomeFilePath(projectDir, slug);
1284
+ if (!(await exists(filePath))) {
1285
+ throw new Error(`Genome file not found: ${filePath}`);
1286
+ }
1287
+
1288
+ const markdown = await fs.readFile(filePath, 'utf8');
1289
+ const versionNumber = String(options['resource-version'] || '').trim();
1290
+ if (!versionNumber) {
1291
+ throw new Error('Missing required --resource-version for genome publish.');
1292
+ }
1293
+
1294
+ const genomeName = findPrimaryHeading(markdown, slug);
1295
+ return {
1296
+ kind: 'aiosforge.genome',
1297
+ exportVersion: 1,
1298
+ genome: {
1299
+ id: null,
1300
+ name: genomeName,
1301
+ slug: sanitizeSegment(slug, 'genome'),
1302
+ description: options.description ? String(options.description).trim() : firstParagraph(markdown),
1303
+ visibility: String(options.visibility || 'PRIVATE').toUpperCase(),
1304
+ status: 'PUBLISHED',
1305
+ sourceKind: String(options['source-kind'] || 'LOCAL').toUpperCase(),
1306
+ ownerUsername: String(options.owner || 'local')
1307
+ },
1308
+ version: {
1309
+ id: null,
1310
+ versionNumber,
1311
+ versionCode: 1,
1312
+ title: options.title ? String(options.title).trim() : genomeName,
1313
+ summary: options.summary ? String(options.summary).trim() : firstParagraph(markdown),
1314
+ schemaVersion: options['schema-version'] ? String(options['schema-version']).trim() : '1',
1315
+ isCurrent: true,
1316
+ createdAt: new Date().toISOString(),
1317
+ contentMarkdown: markdown,
1318
+ manifestJson: null
1319
+ }
1320
+ };
1321
+ }
1322
+
1323
+ async function listAgentFiles(agentsDir) {
1324
+ const entries = await fs.readdir(agentsDir).catch(() => []);
1325
+ const files = [];
1326
+ for (const entry of entries) {
1327
+ if (!entry.endsWith('.md')) continue;
1328
+ const absPath = path.join(agentsDir, entry);
1329
+ const stat = await fs.stat(absPath).catch(() => null);
1330
+ if (!stat?.isFile()) continue;
1331
+ files.push(absPath);
1332
+ }
1333
+ files.sort();
1334
+ return files;
1335
+ }
1336
+
1337
+ async function buildAppliedGenomesFromMetadata(projectDir, metadataContent, options = {}) {
1338
+ const structured = await buildAppliedGenomesFromBindings(
1339
+ projectDir,
1340
+ options.genomeBindings || options.manifestBindings || options.manifest?.genomes,
1341
+ options
1342
+ );
1343
+ if (structured.length > 0) {
1344
+ return structured;
1345
+ }
1346
+
1347
+ const linkedVersion = String(options['linked-genome-version'] || options['resource-version'] || '1.0.0').trim();
1348
+ const shared = parseListSection(metadataContent, 'Genomes');
1349
+ const scoped = parseListSection(metadataContent, 'AgentGenomes').map(parseAgentGenomeEntry).filter(Boolean);
1350
+ const items = [];
1351
+
1352
+ for (const genomeRelPath of shared) {
1353
+ const genomeAbsPath = path.join(projectDir, normalizeRel(genomeRelPath));
1354
+ const markdown = await fs.readFile(genomeAbsPath, 'utf8').catch(() => null);
1355
+ if (!markdown) continue;
1356
+ const slug = sanitizeSegment(path.basename(genomeAbsPath, '.md'), 'genome');
1357
+ items.push({
1358
+ scopeType: 'SQUAD',
1359
+ agentSlug: null,
1360
+ priority: 0,
1361
+ genome: {
1362
+ id: null,
1363
+ name: findPrimaryHeading(markdown, slug),
1364
+ slug,
1365
+ visibility: String(options.visibility || 'PRIVATE').toUpperCase(),
1366
+ status: 'PUBLISHED',
1367
+ sourceKind: 'LOCAL'
1368
+ },
1369
+ version: {
1370
+ id: null,
1371
+ versionNumber: linkedVersion,
1372
+ versionCode: 1,
1373
+ title: findPrimaryHeading(markdown, slug),
1374
+ summary: firstParagraph(markdown),
1375
+ schemaVersion: '1',
1376
+ contentMarkdown: markdown,
1377
+ manifestJson: null,
1378
+ createdAt: new Date().toISOString()
1379
+ }
1380
+ });
1381
+ }
1382
+
1383
+ for (const entry of scoped) {
1384
+ const genomeAbsPath = path.join(projectDir, normalizeRel(entry.genomePath));
1385
+ const markdown = await fs.readFile(genomeAbsPath, 'utf8').catch(() => null);
1386
+ if (!markdown) continue;
1387
+ const slug = sanitizeSegment(path.basename(genomeAbsPath, '.md'), 'genome');
1388
+ items.push({
1389
+ scopeType: 'AGENT',
1390
+ agentSlug: entry.agentSlug,
1391
+ priority: 0,
1392
+ genome: {
1393
+ id: null,
1394
+ name: findPrimaryHeading(markdown, slug),
1395
+ slug,
1396
+ visibility: String(options.visibility || 'PRIVATE').toUpperCase(),
1397
+ status: 'PUBLISHED',
1398
+ sourceKind: 'LOCAL'
1399
+ },
1400
+ version: {
1401
+ id: null,
1402
+ versionNumber: linkedVersion,
1403
+ versionCode: 1,
1404
+ title: findPrimaryHeading(markdown, slug),
1405
+ summary: firstParagraph(markdown),
1406
+ schemaVersion: '1',
1407
+ contentMarkdown: markdown,
1408
+ manifestJson: null,
1409
+ createdAt: new Date().toISOString()
1410
+ }
1411
+ });
1412
+ }
1413
+
1414
+ return items;
1415
+ }
1416
+
1417
+ async function buildAppliedGenomesFromBindings(projectDir, genomeBindings, options = {}) {
1418
+ const linkedVersion = String(options['linked-genome-version'] || options['resource-version'] || '1.0.0').trim();
1419
+ const flattened = flattenGenomeBindings(genomeBindings);
1420
+ const items = [];
1421
+
1422
+ for (const binding of flattened) {
1423
+ const genomeAbsPath = path.join(projectDir, '.aioson', 'genomes', `${binding.slug}.md`);
1424
+ const markdown = await fs.readFile(genomeAbsPath, 'utf8').catch(() => null);
1425
+ if (!markdown) continue;
1426
+
1427
+ items.push({
1428
+ scopeType: binding.scope === 'squad' ? 'SQUAD' : 'AGENT',
1429
+ agentSlug: binding.agentSlug || null,
1430
+ priority: Number.isFinite(binding.priority) ? binding.priority : 0,
1431
+ genome: {
1432
+ id: null,
1433
+ name: findPrimaryHeading(markdown, binding.slug),
1434
+ slug: binding.slug,
1435
+ type: binding.type || null,
1436
+ visibility: String(options.visibility || 'PRIVATE').toUpperCase(),
1437
+ status: 'PUBLISHED',
1438
+ sourceKind: String(binding.source || 'LOCAL').toUpperCase()
1439
+ },
1440
+ version: {
1441
+ id: null,
1442
+ versionNumber: binding.version || linkedVersion,
1443
+ versionCode: 1,
1444
+ title: findPrimaryHeading(markdown, binding.slug),
1445
+ summary: firstParagraph(markdown),
1446
+ schemaVersion: '1',
1447
+ contentMarkdown: markdown,
1448
+ manifestJson: {
1449
+ type: binding.type,
1450
+ evidenceMode: binding.evidenceMode
1451
+ },
1452
+ createdAt: new Date().toISOString()
1453
+ }
1454
+ });
1455
+ }
1456
+
1457
+ return items;
1458
+ }
1459
+
1460
+ async function loadLocalSquadSnapshot(projectDir, slug, options = {}) {
1461
+ const packageSummaryPath = localSquadSummaryPath(projectDir, slug);
1462
+ const metadataPath = (await exists(packageSummaryPath))
1463
+ ? packageSummaryPath
1464
+ : localLegacySquadMetadataPath(projectDir, slug);
1465
+ if (!(await exists(metadataPath))) {
1466
+ throw new Error(`Squad metadata not found: ${metadataPath}`);
1467
+ }
1468
+
1469
+ const content = await fs.readFile(metadataPath, 'utf8');
1470
+ const versionNumber = String(options['resource-version'] || '').trim();
1471
+ if (!versionNumber) {
1472
+ throw new Error('Missing required --resource-version for squad publish.');
1473
+ }
1474
+
1475
+ const squadName = extractField(content, 'Squad') || slug;
1476
+ const goal = extractField(content, 'Goal', 'Objetivo') || firstParagraph(content) || null;
1477
+ const packageRootRel = `.aioson/squads/${slug}`;
1478
+ const agentsDirRel = normalizeRel(extractField(content, 'Agents') || `${packageRootRel}/agents`);
1479
+ const outputDirRel = normalizeRel(extractField(content, 'Output') || `output/${slug}`);
1480
+ const logsDirRel = normalizeRel(extractField(content, 'Logs') || `aioson-logs/${slug}`);
1481
+ const mediaDirRel = normalizeRel(extractField(content, 'Media') || `media/${slug}`);
1482
+ const agentsDirAbs = path.join(projectDir, agentsDirRel);
1483
+ const localManifest = (await loadLocalSquadManifest(projectDir, slug)) || {};
1484
+ const packageDirRel = normalizeRel(localManifest?.package?.rootDir || packageRootRel);
1485
+ const textManifestPath = (await exists(localSquadTextManifestPath(projectDir, slug)))
1486
+ ? localSquadTextManifestPath(projectDir, slug)
1487
+ : localLegacySquadTextManifestPath(projectDir, slug);
1488
+ const designDocPath = (await exists(localSquadDesignDocPath(projectDir, slug)))
1489
+ ? localSquadDesignDocPath(projectDir, slug)
1490
+ : localLegacySquadDesignDocPath(projectDir, slug);
1491
+ const readinessPath = (await exists(localSquadReadinessPath(projectDir, slug)))
1492
+ ? localSquadReadinessPath(projectDir, slug)
1493
+ : localLegacySquadReadinessPath(projectDir, slug);
1494
+ const textManifest = (await fs.readFile(textManifestPath, 'utf8').catch(() => null)) || null;
1495
+ const designDocMarkdown = (await fs.readFile(designDocPath, 'utf8').catch(() => null)) || null;
1496
+ const readinessMarkdown = (await fs.readFile(readinessPath, 'utf8').catch(() => null)) || null;
1497
+ const agentFiles = await listAgentFiles(agentsDirAbs);
1498
+ const agentsManifestJson = [];
1499
+
1500
+ for (const filePath of agentFiles) {
1501
+ const baseName = path.basename(filePath);
1502
+ if (
1503
+ baseName === 'agents.md' ||
1504
+ baseName === 'squad.manifest.json' ||
1505
+ baseName === 'design-doc.md' ||
1506
+ baseName === 'readiness.md'
1507
+ ) continue;
1508
+ const markdown = await fs.readFile(filePath, 'utf8');
1509
+ const agentSlug = sanitizeSegment(path.basename(filePath, '.md'), 'agent');
1510
+ agentsManifestJson.push({
1511
+ slug: agentSlug,
1512
+ name: findPrimaryHeading(markdown, agentSlug),
1513
+ description: firstParagraph(markdown),
1514
+ content: markdown
1515
+ });
1516
+ }
1517
+
1518
+ const manifestBindings = mergeGenomeBindings({
1519
+ blueprintBindings: localManifest?.genomeBindings,
1520
+ manifestBindings: localManifest?.genomeBindings || localManifest?.genomes,
1521
+ legacyExecutors: localManifest?.executors
1522
+ });
1523
+ const normalizedManifest =
1524
+ localManifest && typeof localManifest === 'object'
1525
+ ? {
1526
+ ...localManifest,
1527
+ slug: sanitizeSegment(localManifest.slug || slug, 'squad'),
1528
+ name: String(localManifest.name || squadName),
1529
+ packageVersion: String(localManifest.packageVersion || versionNumber),
1530
+ mode: String(localManifest.mode || 'content'),
1531
+ mission: String(localManifest.mission || extractField(content, 'Description', 'Descricao') || goal || squadName),
1532
+ goal: String(localManifest.goal || goal || ''),
1533
+ visibility: String(localManifest.visibility || options.visibility || 'private').toLowerCase(),
1534
+ aiosLiteCompatibility: String(
1535
+ localManifest.aiosLiteCompatibility ||
1536
+ options['compatibility-min'] ||
1537
+ '^1.1.0'
1538
+ ),
1539
+ rules: {
1540
+ ...(localManifest.rules && typeof localManifest.rules === 'object' ? localManifest.rules : {}),
1541
+ outputsDir: localManifest?.rules?.outputsDir || outputDirRel,
1542
+ logsDir: localManifest?.rules?.logsDir || logsDirRel,
1543
+ mediaDir: localManifest?.rules?.mediaDir || mediaDirRel
1544
+ },
1545
+ storagePolicy:
1546
+ localManifest.storagePolicy && typeof localManifest.storagePolicy === 'object'
1547
+ ? localManifest.storagePolicy
1548
+ : {
1549
+ primary: String(localManifest.mode || 'content') === 'builder' ? 'files' : 'sqlite',
1550
+ artifacts: String(localManifest.mode || 'content') === 'builder' ? 'files+sqlite' : 'sqlite-json',
1551
+ exports: { html: true, markdown: true, json: true }
1552
+ },
1553
+ package:
1554
+ localManifest.package && typeof localManifest.package === 'object'
1555
+ ? localManifest.package
1556
+ : {
1557
+ rootDir: packageDirRel,
1558
+ agentsDir: `${packageDirRel}/agents`,
1559
+ skillsDir: `${packageDirRel}/skills`,
1560
+ templatesDir: `${packageDirRel}/templates`,
1561
+ docsDir: `${packageDirRel}/docs`
1562
+ },
1563
+ baseRoles: Array.isArray(localManifest.baseRoles)
1564
+ ? localManifest.baseRoles
1565
+ : ['orchestrator', 'discovery-lead', 'design-doc-lead', 'planner', 'implementer', 'reviewer', 'docs-maintainer'],
1566
+ skills: Array.isArray(localManifest.skills) ? localManifest.skills : [],
1567
+ mcps: Array.isArray(localManifest.mcps) ? localManifest.mcps : [],
1568
+ subagents:
1569
+ localManifest.subagents && typeof localManifest.subagents === 'object'
1570
+ ? localManifest.subagents
1571
+ : {
1572
+ allowed: true,
1573
+ when: ['broad research', 'comparison', 'large-context summarization', 'parallel analysis']
1574
+ },
1575
+ contentBlueprints: normalizeContentBlueprints(localManifest.contentBlueprints),
1576
+ context:
1577
+ localManifest.context && typeof localManifest.context === 'object'
1578
+ ? {
1579
+ ...localManifest.context,
1580
+ designDocPath: localManifest.context.designDocPath || `${packageDirRel}/docs/design-doc.md`,
1581
+ readinessPath: localManifest.context.readinessPath || `${packageDirRel}/docs/readiness.md`,
1582
+ docsPackage: Array.isArray(localManifest.context.docsPackage)
1583
+ ? localManifest.context.docsPackage
1584
+ : ['project.context.md', 'design-doc.md', 'readiness.md']
1585
+ }
1586
+ : {
1587
+ mode: 'project',
1588
+ summary: goal || squadName,
1589
+ designDocPath: `${packageDirRel}/docs/design-doc.md`,
1590
+ readinessPath: `${packageDirRel}/docs/readiness.md`,
1591
+ docsPackage: ['project.context.md', 'design-doc.md', 'readiness.md']
1592
+ },
1593
+ executors: Array.isArray(localManifest.executors)
1594
+ ? attachBindingsToExecutors(localManifest.executors, manifestBindings)
1595
+ : agentsManifestJson.map((agent) => ({
1596
+ slug: agent.slug,
1597
+ title: agent.name,
1598
+ role: agent.description,
1599
+ file: `${packageDirRel}/agents/${agent.slug}.md`,
1600
+ skills: [],
1601
+ genomes: []
1602
+ })),
1603
+ genomes: manifestBindings,
1604
+ genomeBindings: manifestBindings
1605
+ }
1606
+ : null;
1607
+
1608
+ return {
1609
+ kind: 'aiosforge.squad',
1610
+ exportVersion: 1,
1611
+ squad: {
1612
+ id: null,
1613
+ name: normalizedManifest?.name || squadName,
1614
+ slug: sanitizeSegment(slug, 'squad'),
1615
+ description: extractField(content, 'Description', 'Descricao') || normalizedManifest?.mission || null,
1616
+ goal: normalizedManifest?.goal || goal,
1617
+ visibility: String(options.visibility || normalizedManifest?.visibility || 'PRIVATE').toUpperCase(),
1618
+ status: 'PUBLISHED',
1619
+ ownerUsername: String(options.owner || 'local'),
1620
+ projectName: null
1621
+ },
1622
+ version: {
1623
+ id: null,
1624
+ versionNumber,
1625
+ versionCode: 1,
1626
+ title: options.title ? String(options.title).trim() : squadName,
1627
+ summary: options.summary ? String(options.summary).trim() : goal,
1628
+ changeLog: options['change-log'] ? String(options['change-log']).trim() : null,
1629
+ compatibilityMin: options['compatibility-min'] ? String(options['compatibility-min']).trim() : null,
1630
+ compatibilityMax: options['compatibility-max'] ? String(options['compatibility-max']).trim() : null,
1631
+ schemaVersion: options['schema-version'] ? String(options['schema-version']).trim() : '1',
1632
+ sourceType: 'local_publish',
1633
+ isCurrent: true,
1634
+ createdAt: new Date().toISOString(),
1635
+ designDocMarkdown,
1636
+ readinessMarkdown,
1637
+ manifestJson:
1638
+ normalizedManifest || {
1639
+ packageVersion: versionNumber,
1640
+ mode: 'content',
1641
+ metadataPath: normalizeRel(path.relative(projectDir, metadataPath)),
1642
+ textManifestPath: normalizeRel(path.relative(projectDir, textManifestPath)),
1643
+ package: {
1644
+ rootDir: packageDirRel,
1645
+ agentsDir: `${packageDirRel}/agents`,
1646
+ skillsDir: `${packageDirRel}/skills`,
1647
+ templatesDir: `${packageDirRel}/templates`,
1648
+ docsDir: `${packageDirRel}/docs`
1649
+ },
1650
+ storagePolicy: {
1651
+ primary: 'sqlite',
1652
+ artifacts: 'sqlite-json',
1653
+ exports: { html: true, markdown: true, json: true }
1654
+ },
1655
+ context: {
1656
+ mode: 'project',
1657
+ summary: goal || squadName,
1658
+ designDocPath: normalizeRel(path.relative(projectDir, designDocPath)),
1659
+ readinessPath: normalizeRel(path.relative(projectDir, readinessPath)),
1660
+ docsPackage: ['project.context.md', 'design-doc.md', 'readiness.md']
1661
+ },
1662
+ outputDir: outputDirRel,
1663
+ logsDir: logsDirRel,
1664
+ mediaDir: mediaDirRel
1665
+ },
1666
+ agentsManifestJson,
1667
+ genomesManifestJson: {
1668
+ textManifestPath: textManifest
1669
+ ? normalizeRel(path.relative(projectDir, textManifestPath))
1670
+ : null,
1671
+ genomes: normalizedManifest?.genomes || { squad: [], executors: {} }
1672
+ }
1673
+ },
1674
+ appliedGenomes: await buildAppliedGenomesFromMetadata(projectDir, content, {
1675
+ ...options,
1676
+ genomeBindings: normalizedManifest?.genomes
1677
+ })
1678
+ };
1679
+ }
1680
+
1681
+ function resolvePublishUrl(options, resource) {
1682
+ if (options.url) return String(options.url).trim();
1683
+ const baseUrl = String(options['base-url'] || '').trim().replace(/\/+$/, '');
1684
+ if (!baseUrl) return '';
1685
+ return `${baseUrl}/api/publish/${resource}s`;
1686
+ }
1687
+
1688
+ async function runCloudPublishGenome({ args, options = {}, logger, t, dependencies = {} }) {
1689
+ const projectDir = await ensureProjectDir(args[0] || '.', t);
1690
+ const slug = String(options.slug || '').trim();
1691
+ if (!slug) {
1692
+ throw new Error('Missing required --slug for genome publish.');
1693
+ }
1694
+ const url = resolvePublishUrl(options, 'genome');
1695
+ if (!url) {
1696
+ throw new Error('Provide --url or --base-url for genome publish.');
1697
+ }
1698
+
1699
+ const payload = await loadLocalGenomeSnapshot(projectDir, slug, options);
1700
+ const dryRun = Boolean(options['dry-run']);
1701
+ const planned = {
1702
+ ok: true,
1703
+ resource: 'genome',
1704
+ dryRun,
1705
+ url,
1706
+ slug: payload.genome.slug,
1707
+ versionNumber: payload.version.versionNumber,
1708
+ projectDir
1709
+ };
1710
+
1711
+ if (dryRun) {
1712
+ logger.log(t('cloud.publish_genome_dry_run', { slug: payload.genome.slug, version: payload.version.versionNumber }));
1713
+ return planned;
1714
+ }
1715
+
1716
+ const response = await postJson(url, payload, dependencies.fetchImpl || fetch);
1717
+ logger.log(t('cloud.publish_genome_done', { slug: payload.genome.slug, version: payload.version.versionNumber }));
1718
+ return {
1719
+ ...planned,
1720
+ response
1721
+ };
1722
+ }
1723
+
1724
+ async function runCloudPublishSquad({ args, options = {}, logger, t, dependencies = {} }) {
1725
+ const projectDir = await ensureProjectDir(args[0] || '.', t);
1726
+ const slug = String(options.slug || '').trim();
1727
+ if (!slug) {
1728
+ throw new Error('Missing required --slug for squad publish.');
1729
+ }
1730
+ const url = resolvePublishUrl(options, 'squad');
1731
+ if (!url) {
1732
+ throw new Error('Provide --url or --base-url for squad publish.');
1733
+ }
1734
+
1735
+ const payload = await loadLocalSquadSnapshot(projectDir, slug, options);
1736
+ const dryRun = Boolean(options['dry-run']);
1737
+ const planned = {
1738
+ ok: true,
1739
+ resource: 'squad',
1740
+ dryRun,
1741
+ url,
1742
+ slug: payload.squad.slug,
1743
+ versionNumber: payload.version.versionNumber,
1744
+ projectDir,
1745
+ agentCount: Array.isArray(payload.version.agentsManifestJson) ? payload.version.agentsManifestJson.length : 0,
1746
+ genomeCount: Array.isArray(payload.appliedGenomes) ? payload.appliedGenomes.length : 0
1747
+ };
1748
+
1749
+ if (dryRun) {
1750
+ logger.log(t('cloud.publish_squad_dry_run', { slug: payload.squad.slug, version: payload.version.versionNumber }));
1751
+ return planned;
1752
+ }
1753
+
1754
+ const response = await postJson(url, payload, dependencies.fetchImpl || fetch);
1755
+ logger.log(t('cloud.publish_squad_done', { slug: payload.squad.slug, version: payload.version.versionNumber }));
1756
+ return {
1757
+ ...planned,
1758
+ response
1759
+ };
1760
+ }
1761
+
1762
+ module.exports = {
1763
+ runCloudImportSquad,
1764
+ runCloudImportGenome,
1765
+ runCloudPublishGenome,
1766
+ runCloudPublishSquad
1767
+ };