@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.
- package/CHANGELOG.md +595 -560
- package/CODE_OF_CONDUCT.md +12 -12
- package/CONTRIBUTING.md +13 -13
- package/LICENSE +661 -661
- package/README.md +919 -776
- package/bin/aioson.js +4 -4
- package/docs/design-previews/aurora-command-ui-website.html +884 -884
- package/docs/design-previews/aurora-command-ui.html +682 -682
- package/docs/design-previews/bold-editorial-ui-website.html +658 -658
- package/docs/design-previews/bold-editorial-ui.html +717 -717
- package/docs/design-previews/clean-saas-ui-website.html +1202 -1202
- package/docs/design-previews/clean-saas-ui.html +549 -549
- package/docs/design-previews/cognitive-core-ui-website.html +1009 -1009
- package/docs/design-previews/cognitive-core-ui.html +463 -463
- package/docs/design-previews/glassmorphism-ui-website.html +572 -572
- package/docs/design-previews/glassmorphism-ui.html +886 -886
- package/docs/design-previews/index.html +699 -699
- package/docs/design-previews/interface-design-website.html +1187 -1187
- package/docs/design-previews/interface-design.html +513 -513
- package/docs/design-previews/neo-brutalist-ui-website.html +621 -621
- package/docs/design-previews/neo-brutalist-ui.html +797 -797
- package/docs/design-previews/premium-command-center-ui-website.html +1217 -1217
- package/docs/design-previews/premium-command-center-ui.html +552 -552
- package/docs/design-previews/pt.squarespace.com-homepage.html +889 -889
- package/docs/design-previews/warm-craft-ui-website.html +684 -684
- package/docs/design-previews/warm-craft-ui.html +739 -739
- package/docs/en/1-understand/ecosystem-map.md +228 -0
- package/docs/en/1-understand/glossary.md +288 -0
- package/docs/en/1-understand/what-is-aioson.md +94 -0
- package/docs/en/1-understand/why-it-exists.md +106 -0
- package/docs/en/2-start/existing-project.md +246 -0
- package/docs/en/2-start/first-project.md +307 -0
- package/docs/en/2-start/initial-decisions.md +223 -0
- package/docs/en/3-recipes/README.md +28 -0
- package/docs/en/3-recipes/continuity-between-sessions.md +303 -0
- package/docs/en/3-recipes/from-idea-to-prd-via-briefing.md +235 -0
- package/docs/en/3-recipes/full-feature-with-sheldon.md +338 -0
- package/docs/en/4-agents/README.md +56 -0
- package/docs/en/5-reference/README.md +60 -0
- package/docs/en/{cli-reference.md → 5-reference/cli-reference.md} +639 -409
- package/docs/en/5-reference/i18n.md +52 -0
- package/docs/en/{json-schemas.md → 5-reference/json-schemas.md} +41 -41
- package/docs/en/{mcp.md → 5-reference/mcp.md} +56 -56
- package/docs/en/{parallel.md → 5-reference/parallel.md} +82 -82
- package/docs/en/{qa-browser.md → 5-reference/qa-browser.md} +339 -339
- package/docs/en/{release-flow.md → 5-reference/release-flow.md} +22 -22
- package/docs/en/{release-notes-template.md → 5-reference/release-notes-template.md} +41 -41
- package/docs/en/{release.md → 5-reference/release.md} +28 -28
- package/docs/en/{schemas → 5-reference/schemas}/agent-prompt.schema.json +17 -17
- package/docs/en/{schemas → 5-reference/schemas}/agents.schema.json +32 -32
- package/docs/en/{schemas → 5-reference/schemas}/context-validate.schema.json +36 -36
- package/docs/en/{schemas → 5-reference/schemas}/doctor.schema.json +89 -89
- package/docs/en/{schemas → 5-reference/schemas}/error.schema.json +24 -24
- package/docs/en/{schemas → 5-reference/schemas}/i18n-add.schema.json +15 -15
- package/docs/en/{schemas → 5-reference/schemas}/index.json +126 -116
- package/docs/en/{schemas → 5-reference/schemas}/info.schema.json +39 -39
- package/docs/en/{schemas → 5-reference/schemas}/init.schema.json +48 -48
- package/docs/en/{schemas → 5-reference/schemas}/install.schema.json +60 -60
- package/docs/en/{schemas → 5-reference/schemas}/locale-apply.schema.json +30 -30
- package/docs/en/{schemas → 5-reference/schemas}/mcp-doctor.schema.json +95 -95
- package/docs/en/{schemas → 5-reference/schemas}/mcp-init.schema.json +122 -122
- package/docs/en/{schemas → 5-reference/schemas}/package-test.schema.json +24 -24
- package/docs/en/{schemas → 5-reference/schemas}/parallel-assign.schema.json +66 -57
- package/docs/en/{schemas → 5-reference/schemas}/parallel-doctor.schema.json +122 -86
- package/docs/en/5-reference/schemas/parallel-guard.schema.json +63 -0
- package/docs/en/{schemas → 5-reference/schemas}/parallel-init.schema.json +53 -53
- package/docs/en/5-reference/schemas/parallel-merge.schema.json +84 -0
- package/docs/en/5-reference/schemas/parallel-status.schema.json +184 -0
- package/docs/en/{schemas → 5-reference/schemas}/setup-context.schema.json +39 -39
- package/docs/en/{schemas → 5-reference/schemas}/smoke.schema.json +23 -23
- package/docs/en/{schemas → 5-reference/schemas}/update.schema.json +48 -48
- package/docs/en/{schemas → 5-reference/schemas}/workflow-plan.schema.json +30 -30
- package/docs/en/{squad-dashboard.md → 5-reference/squad-dashboard.md} +372 -372
- package/docs/en/{web3.md → 5-reference/web3.md} +54 -54
- package/docs/en/README.md +115 -0
- package/docs/en/active-learning-loop/README.md +117 -0
- package/docs/en/active-learning-loop/active-learning-loop.md +117 -0
- package/docs/en/active-learning-loop/cli-commands.md +320 -0
- package/docs/en/active-learning-loop/diagrams.md +225 -0
- package/docs/en/active-learning-loop/doctor-checks.md +151 -0
- package/docs/en/active-learning-loop/how-to-use.md +313 -0
- package/docs/en/active-learning-loop/troubleshooting.md +283 -0
- package/docs/en/deyvin-subtask-scout/README.md +109 -0
- package/docs/en/deyvin-subtask-scout/cli-commands.md +248 -0
- package/docs/en/deyvin-subtask-scout/diagrams.md +124 -0
- package/docs/en/deyvin-subtask-scout/how-to-use.md +221 -0
- package/docs/en/deyvin-subtask-scout/sub-task-scout.md +115 -0
- package/docs/en/deyvin-subtask-scout/troubleshooting.md +184 -0
- package/docs/integrations/apps-publish-marketplace.md +94 -0
- package/docs/integrations/sdlc-genius-boundary.md +76 -76
- package/docs/integrations/sdlc-genius-eval-matrix.md +75 -75
- package/docs/integrations/sdlc-genius-install-checklist.md +93 -93
- package/docs/integrations/sdlc-genius-review-samples.md +86 -86
- package/docs/openclaw-bridge.md +308 -308
- package/docs/pt/1-entender/glossario.md +288 -0
- package/docs/pt/1-entender/mapa-do-ecossistema.md +228 -0
- package/docs/pt/1-entender/o-que-e-aioson.md +94 -0
- package/docs/pt/1-entender/por-que-existe.md +107 -0
- package/docs/pt/2-comecar/decisoes-iniciais.md +223 -0
- package/docs/pt/2-comecar/primeiro-projeto.md +307 -0
- package/docs/pt/2-comecar/projeto-existente.md +245 -0
- package/docs/pt/3-receitas/README.md +28 -0
- package/docs/pt/3-receitas/app-saas-do-zero.md +324 -0
- package/docs/pt/3-receitas/auditoria-seguranca.md +254 -0
- package/docs/pt/3-receitas/clonar-design-de-site.md +211 -0
- package/docs/pt/3-receitas/continuidade-entre-sessoes.md +303 -0
- package/docs/pt/3-receitas/da-ideia-ao-prd-via-briefing.md +234 -0
- package/docs/pt/3-receitas/feature-completa-com-sheldon.md +338 -0
- package/docs/pt/3-receitas/integracao-em-codebase-grande.md +243 -0
- package/docs/pt/3-receitas/landing-page.md +281 -0
- package/docs/pt/3-receitas/plans-externos-para-product.md +191 -0
- package/docs/pt/3-receitas/publicar-no-aioson-com.md +219 -0
- package/docs/pt/3-receitas/refatoracao-grande.md +251 -0
- package/docs/pt/4-agentes/README.md +65 -0
- package/docs/pt/4-agentes/analyst.md +111 -0
- package/docs/pt/4-agentes/architect.md +113 -0
- package/docs/pt/4-agentes/briefing.md +95 -0
- package/docs/pt/4-agentes/committer.md +108 -0
- package/docs/pt/4-agentes/copywriter.md +279 -0
- package/docs/pt/4-agentes/design-hybrid-forge.md +116 -0
- package/docs/pt/4-agentes/dev.md +136 -0
- package/docs/pt/4-agentes/deyvin.md +99 -0
- package/docs/pt/4-agentes/discover.md +122 -0
- package/docs/pt/4-agentes/discovery-design-doc.md +91 -0
- package/docs/pt/4-agentes/genome.md +115 -0
- package/docs/pt/4-agentes/neo.md +93 -0
- package/docs/pt/4-agentes/orache.md +107 -0
- package/docs/pt/4-agentes/orchestrator.md +118 -0
- package/docs/pt/4-agentes/pentester.md +131 -0
- package/docs/pt/4-agentes/pm.md +97 -0
- package/docs/pt/4-agentes/product.md +114 -0
- package/docs/pt/4-agentes/profiler-enricher.md +93 -0
- package/docs/pt/4-agentes/profiler-forge.md +93 -0
- package/docs/pt/4-agentes/profiler-researcher.md +98 -0
- package/docs/pt/4-agentes/qa.md +124 -0
- package/docs/pt/4-agentes/setup.md +104 -0
- package/docs/pt/4-agentes/sheldon.md +95 -0
- package/docs/pt/4-agentes/site-forge.md +104 -0
- package/docs/pt/4-agentes/squad.md +127 -0
- package/docs/pt/4-agentes/tester.md +105 -0
- package/docs/pt/4-agentes/ux-ui.md +110 -0
- package/docs/pt/4-agentes/validator.md +118 -0
- package/docs/pt/5-referencia/README.md +88 -0
- package/docs/pt/5-referencia/agent-chain-continuity.md +124 -0
- package/docs/pt/{agent-sharding.md → 5-referencia/agent-sharding.md} +132 -132
- package/docs/pt/5-referencia/aioson-com-store.md +119 -0
- package/docs/pt/{automacao-squads.md → 5-referencia/automacao-squads.md} +407 -407
- package/docs/pt/{clientes-ai.md → 5-referencia/clientes-ai.md} +300 -286
- package/docs/pt/{comandos-cli.md → 5-referencia/comandos-cli.md} +1823 -1634
- package/docs/pt/5-referencia/compress-agents.md +304 -0
- package/docs/pt/5-referencia/design-docs-governance.md +59 -0
- package/docs/pt/{devlog-pipeline.md → 5-referencia/devlog-pipeline.md} +270 -270
- package/docs/pt/5-referencia/feature-archive.md +199 -0
- package/docs/pt/5-referencia/feature-dossier.md +121 -0
- package/docs/pt/{fluxo-artefatos.md → 5-referencia/fluxo-artefatos.md} +179 -178
- package/docs/pt/{genome-3.0-spec.md → 5-referencia/genome-4.0-spec.md} +407 -296
- package/docs/pt/5-referencia/genome-distribution.md +232 -0
- package/docs/pt/{hooks-session-guard.md → 5-referencia/hooks-session-guard.md} +454 -454
- package/docs/pt/{inteligencia-adaptativa.md → 5-referencia/inteligencia-adaptativa.md} +324 -324
- package/docs/pt/5-referencia/live-sessions.md +144 -0
- package/docs/pt/5-referencia/memoria-e-contexto.md +340 -0
- package/docs/pt/5-referencia/motor-hardening.md +493 -0
- package/docs/pt/{output-strategy-delivery.md → 5-referencia/output-strategy-delivery.md} +655 -655
- package/docs/pt/5-referencia/runner-system.md +113 -0
- package/docs/pt/{runtime-observability.md → 5-referencia/runtime-observability.md} +76 -76
- package/docs/pt/{sandbox.md → 5-referencia/sandbox.md} +125 -125
- package/docs/pt/{sdd-automation-scripts.md → 5-referencia/sdd-automation-scripts.md} +559 -557
- package/docs/pt/5-referencia/sdd-framework.md +115 -0
- package/docs/pt/5-referencia/sdd-planos-e-estrutura.md +321 -0
- package/docs/pt/5-referencia/secure-by-default.md +117 -0
- package/docs/pt/{skills.md → 5-referencia/skills.md} +275 -267
- package/docs/pt/{spec-learnings-pipeline.md → 5-referencia/spec-learnings-pipeline.md} +265 -265
- package/docs/pt/{squad-dashboard.md → 5-referencia/squad-dashboard.md} +373 -373
- package/docs/pt/{web3.md → 5-referencia/web3.md} +797 -797
- package/docs/pt/README.md +111 -116
- package/docs/pt/_arquivo/README.md +130 -0
- package/docs/pt/{advisor-spec.md → _arquivo/advisor-spec.md} +343 -335
- package/docs/pt/{agentes-customizados.md → _arquivo/agentes-customizados.md} +678 -670
- package/docs/pt/{busca-de-contexto.md → _arquivo/busca-de-contexto.md} +136 -129
- package/docs/pt/{cache-de-contexto.md → _arquivo/cache-de-contexto.md} +163 -156
- package/docs/pt/{cenarios.md → _arquivo/cenarios.md} +1282 -1274
- package/docs/pt/{design-hybrid-forge.md → _arquivo/design-hybrid-forge.md} +365 -356
- package/docs/pt/{deyvin.md → _arquivo/deyvin.md} +123 -115
- package/docs/pt/{guia-engineer.md → _arquivo/guia-engineer.md} +234 -226
- package/docs/pt/{inicio-rapido.md → _arquivo/inicio-rapido.md} +261 -250
- package/docs/pt/{memoria-contexto.md → _arquivo/memoria-contexto.md} +262 -255
- package/docs/pt/{monitor-de-contexto.md → _arquivo/monitor-de-contexto.md} +165 -158
- package/docs/pt/{profiler-system.md → _arquivo/profiler-system.md} +222 -214
- package/docs/pt/{recuperacao-de-sessao.md → _arquivo/recuperacao-de-sessao.md} +134 -125
- package/docs/pt/{site-forge.md → _arquivo/site-forge.md} +318 -309
- package/docs/pt/{squad-genome.md → _arquivo/squad-genome.md} +793 -783
- package/docs/pt/active-learning-loop/README.md +117 -0
- package/docs/pt/active-learning-loop/ativo-learning-loop.md +117 -0
- package/docs/pt/active-learning-loop/comandos-cli.md +320 -0
- package/docs/pt/active-learning-loop/como-usar.md +313 -0
- package/docs/pt/active-learning-loop/diagramas.md +225 -0
- package/docs/pt/active-learning-loop/doctor-checks.md +151 -0
- package/docs/pt/active-learning-loop/troubleshooting.md +283 -0
- package/docs/pt/agentes.md +996 -672
- package/docs/pt/deyvin-subtask-scout/README.md +109 -0
- package/docs/pt/deyvin-subtask-scout/comandos-cli.md +248 -0
- package/docs/pt/deyvin-subtask-scout/como-usar.md +221 -0
- package/docs/pt/deyvin-subtask-scout/diagramas.md +124 -0
- package/docs/pt/deyvin-subtask-scout/sub-task-scout.md +113 -0
- package/docs/pt/deyvin-subtask-scout/troubleshooting.md +184 -0
- package/docs/pt/living-memory/README.md +81 -0
- package/docs/pt/living-memory/autonomy-contract.md +206 -0
- package/docs/pt/living-memory/diagramas.md +365 -0
- package/docs/pt/living-memory/memoria-viva.md +141 -0
- package/docs/pt/living-memory/notificacoes-info.md +142 -0
- package/docs/pt/living-memory/reflexao-in-harness.md +218 -0
- package/docs/pt/living-memory/troubleshooting.md +286 -0
- package/docs/testing/genome-2.0-manual-regression.md +23 -23
- package/docs/testing/genome-2.0-matrix.md +36 -36
- package/docs/testing/genome-2.0-rollout.md +184 -184
- package/package.json +51 -50
- package/src/a2a/client.js +165 -165
- package/src/a2a/server.js +223 -223
- package/src/agent-loader.js +280 -280
- package/src/agent-manifests.js +86 -0
- package/src/agents.js +92 -72
- package/src/autonomy-policy.js +163 -0
- package/src/backup-local.js +74 -74
- package/src/backup-provider.js +303 -303
- package/src/brain-query.js +171 -0
- package/src/cli.js +1450 -1099
- package/src/commands/agent-audit.js +397 -397
- package/src/commands/agent-export-skill.js +229 -229
- package/src/commands/agent-loader.js +85 -85
- package/src/commands/agents.js +273 -160
- package/src/commands/artifact-validate.js +218 -189
- package/src/commands/auth.js +298 -0
- package/src/commands/backup-local-cmd.js +25 -25
- package/src/commands/backup.js +533 -533
- package/src/commands/brain-query.js +44 -0
- package/src/commands/brief-gen.js +405 -405
- package/src/commands/brief-validate.js +65 -65
- package/src/commands/briefing.js +344 -0
- package/src/commands/classify.js +256 -256
- package/src/commands/cloud.js +1767 -1767
- package/src/commands/commit-prepare.js +610 -0
- package/src/commands/compress-agents.js +416 -0
- package/src/commands/config.js +90 -90
- package/src/commands/context-cache.js +90 -90
- package/src/commands/context-compact.js +49 -49
- package/src/commands/context-health.js +187 -175
- package/src/commands/context-load.js +219 -0
- package/src/commands/context-monitor.js +163 -163
- package/src/commands/context-pack.js +45 -45
- package/src/commands/context-search.js +66 -66
- package/src/commands/context-trim.js +183 -177
- package/src/commands/context-validate.js +91 -91
- package/src/commands/design-hybrid-options.js +385 -385
- package/src/commands/detect-test-runner.js +55 -55
- package/src/commands/dev-resume.js +32 -0
- package/src/commands/devlog-export-brains.js +27 -27
- package/src/commands/devlog-process.js +294 -292
- package/src/commands/devlog-watch.js +131 -131
- package/src/commands/doctor.js +123 -123
- package/src/commands/dossier-add-research.js +114 -0
- package/src/commands/dossier-audit.js +222 -0
- package/src/commands/dossier.js +423 -0
- package/src/commands/feature-archive.js +513 -0
- package/src/commands/feature-close.js +554 -165
- package/src/commands/gate-approve.js +198 -0
- package/src/commands/gate-check.js +247 -228
- package/src/commands/genome-doctor.js +489 -41
- package/src/commands/genome-migrate.js +49 -49
- package/src/commands/git-guard.js +170 -0
- package/src/commands/harness.js +307 -0
- package/src/commands/health.js +214 -214
- package/src/commands/hooks-emit.js +253 -253
- package/src/commands/hooks-install.js +347 -347
- package/src/commands/i18n-add.js +56 -56
- package/src/commands/implementation-plan.js +367 -340
- package/src/commands/info.js +41 -41
- package/src/commands/init.js +120 -116
- package/src/commands/install.js +162 -107
- package/src/commands/learning-auto-promote.js +197 -195
- package/src/commands/learning-evolve.js +364 -364
- package/src/commands/learning-export.js +103 -103
- package/src/commands/learning-rollback.js +164 -164
- package/src/commands/learning.js +134 -134
- package/src/commands/live.js +2101 -1641
- package/src/commands/locale-apply.js +54 -51
- package/src/commands/locale-diff.js +25 -126
- package/src/commands/mcp-doctor.js +407 -406
- package/src/commands/mcp-init.js +373 -379
- package/src/commands/memory-archive.js +193 -0
- package/src/commands/memory-reflect-commit.js +148 -0
- package/src/commands/memory-reflect-prepare.js +97 -0
- package/src/commands/memory-restore.js +177 -0
- package/src/commands/memory-search.js +135 -0
- package/src/commands/memory.js +299 -0
- package/src/commands/notify.js +68 -0
- package/src/commands/package-e2e.js +273 -273
- package/src/commands/parallel-assign.js +483 -403
- package/src/commands/parallel-doctor.js +850 -437
- package/src/commands/parallel-guard.js +241 -0
- package/src/commands/parallel-init.js +311 -249
- package/src/commands/parallel-merge.js +299 -0
- package/src/commands/parallel-status.js +434 -290
- package/src/commands/pattern-detect.js +33 -33
- package/src/commands/preflight-context.js +30 -30
- package/src/commands/preflight.js +267 -208
- package/src/commands/pulse-update.js +130 -130
- package/src/commands/qa-doctor.js +185 -185
- package/src/commands/qa-init.js +166 -161
- package/src/commands/qa-report.js +58 -58
- package/src/commands/qa-run.js +873 -873
- package/src/commands/qa-scan.js +337 -337
- package/src/commands/recovery.js +43 -43
- package/src/commands/revision.js +235 -0
- package/src/commands/runner-daemon.js +274 -274
- package/src/commands/runner-plan.js +70 -70
- package/src/commands/runner-queue-from-plan.js +166 -166
- package/src/commands/runner-queue.js +189 -189
- package/src/commands/runner-run.js +129 -129
- package/src/commands/runtime.js +2086 -2067
- package/src/commands/sandbox.js +37 -37
- package/src/commands/scaffold-complete.js +188 -0
- package/src/commands/scan-project.js +1371 -1371
- package/src/commands/scout-commit.js +163 -0
- package/src/commands/scout-prep.js +214 -0
- package/src/commands/scout-validate.js +112 -0
- package/src/commands/security-audit.js +275 -0
- package/src/commands/security-scan.js +376 -0
- package/src/commands/self-implement-loop.js +306 -256
- package/src/commands/session-guard.js +218 -218
- package/src/commands/setup-context.js +699 -698
- package/src/commands/setup.js +178 -178
- package/src/commands/sizing.js +165 -165
- package/src/commands/skill.js +670 -670
- package/src/commands/smoke.js +426 -426
- package/src/commands/spec-checkpoint.js +177 -177
- package/src/commands/spec-status.js +79 -79
- package/src/commands/spec-sync.js +190 -190
- package/src/commands/spec-tasks.js +288 -288
- package/src/commands/squad-agent-create.js +830 -788
- package/src/commands/squad-autorun.js +1220 -1220
- package/src/commands/squad-bus.js +217 -217
- package/src/commands/squad-card.js +149 -149
- package/src/commands/squad-daemon.js +343 -343
- package/src/commands/squad-dashboard.js +39 -39
- package/src/commands/squad-dependency-graph.js +164 -164
- package/src/commands/squad-deploy.js +64 -64
- package/src/commands/squad-doctor.js +460 -460
- package/src/commands/squad-export.js +77 -46
- package/src/commands/squad-investigate.js +314 -261
- package/src/commands/squad-learning.js +209 -209
- package/src/commands/squad-mcp.js +270 -270
- package/src/commands/squad-pipeline.js +343 -343
- package/src/commands/squad-plan.js +361 -329
- package/src/commands/squad-processes.js +56 -56
- package/src/commands/squad-recovery.js +42 -42
- package/src/commands/squad-repair-genomes.js +39 -39
- package/src/commands/squad-review.js +106 -106
- package/src/commands/squad-roi.js +291 -291
- package/src/commands/squad-scaffold.js +56 -55
- package/src/commands/squad-score.js +311 -250
- package/src/commands/squad-status.js +481 -460
- package/src/commands/squad-tool-register.js +157 -157
- package/src/commands/squad-validate.js +438 -347
- package/src/commands/squad-webhook.js +160 -160
- package/src/commands/squad-worker.js +191 -191
- package/src/commands/squad-worktrees.js +75 -75
- package/src/commands/state-save.js +122 -122
- package/src/commands/store-genome.js +667 -0
- package/src/commands/store-skill.js +247 -0
- package/src/commands/store-squad.js +431 -0
- package/src/commands/store-system.js +392 -0
- package/src/commands/sync-agents-preflight.js +176 -0
- package/src/commands/test-agents.js +199 -199
- package/src/commands/tool-capabilities.js +63 -0
- package/src/commands/tool-registry-cmd.js +232 -232
- package/src/commands/update.js +64 -64
- package/src/commands/verify-gate.js +612 -572
- package/src/commands/web-map.js +70 -70
- package/src/commands/web-scrape.js +71 -71
- package/src/commands/workflow-execute.js +730 -241
- package/src/commands/workflow-harden.js +231 -0
- package/src/commands/workflow-heal.js +136 -0
- package/src/commands/workflow-next.js +1279 -601
- package/src/commands/workflow-plan.js +108 -108
- package/src/commands/workflow-status.js +440 -250
- package/src/commands/workspace.js +144 -0
- package/src/constants.js +413 -417
- package/src/context-cache.js +159 -159
- package/src/context-memory.js +975 -837
- package/src/context-parse-reason.js +22 -22
- package/src/context-search.js +326 -326
- package/src/context-writer.js +197 -196
- package/src/context.js +247 -217
- package/src/delivery-runner.js +319 -319
- package/src/design-variation-catalog.js +503 -503
- package/src/detector.js +261 -261
- package/src/doctor.js +760 -289
- package/src/dossier/codemap-store.js +267 -0
- package/src/dossier/dossier-bootstrap.js +222 -0
- package/src/dossier/dossier-compact.js +159 -0
- package/src/dossier/lock.js +128 -0
- package/src/dossier/research-index-store.js +233 -0
- package/src/dossier/revision-store.js +313 -0
- package/src/dossier/schema.js +162 -0
- package/src/dossier/scout-section.js +127 -0
- package/src/dossier/store.js +406 -0
- package/src/execution-gateway.js +464 -461
- package/src/friction-scanner.js +202 -0
- package/src/genome-files.js +198 -198
- package/src/genome-format.js +442 -442
- package/src/genome-schema.js +238 -215
- package/src/genomes/bindings.js +281 -281
- package/src/genomes.js +500 -467
- package/src/handoff-contract.js +417 -0
- package/src/handoff-validator.js +45 -0
- package/src/harness/circuit-breaker.js +135 -0
- package/src/i18n/index.js +103 -103
- package/src/i18n/messages/en.js +1541 -1139
- package/src/i18n/messages/es.js +1325 -980
- package/src/i18n/messages/fr.js +1333 -987
- package/src/i18n/messages/pt-BR.js +1561 -1166
- package/src/i18n/scaffold.js +64 -64
- package/src/install-animation.js +260 -260
- package/src/install-profile.js +127 -143
- package/src/install-wizard.js +475 -475
- package/src/installer-config-merge.js +207 -0
- package/src/installer.js +449 -294
- package/src/learning-loop-archive.js +595 -0
- package/src/learning-loop-doctor.js +217 -0
- package/src/learning-loop-engine.js +254 -0
- package/src/learning-loop-fts5.js +132 -0
- package/src/learning-loop-migration.js +163 -0
- package/src/lib/dev-resume.js +140 -0
- package/src/lib/dossier-telemetry.js +36 -0
- package/src/lib/genomes/compat.js +206 -206
- package/src/lib/genomes/migrate.js +90 -90
- package/src/lib/git-commit-guard.js +751 -0
- package/src/lib/health-check.js +158 -158
- package/src/lib/hook-protocol.js +76 -76
- package/src/lib/llm-content-sanitizer.js +44 -0
- package/src/lib/security/artifact-reader.js +167 -0
- package/src/lib/security/exit-codes.js +51 -0
- package/src/lib/security/findings-writer.js +176 -0
- package/src/lib/security/runtime-events.js +77 -0
- package/src/lib/security/secrets-regex.js +115 -0
- package/src/lib/squads/genome-repair.js +49 -49
- package/src/lib/store/security-scan.js +175 -0
- package/src/lib/terminal-checkbox.js +135 -0
- package/src/lib/terminal-picker.js +447 -0
- package/src/lib/tmux-launcher.js +163 -0
- package/src/lib/tool-capabilities.js +102 -0
- package/src/lib/webhook-server.js +328 -328
- package/src/locales.js +88 -84
- package/src/mcp/apps/squad-dashboard/app.js +163 -163
- package/src/mcp/apps/squad-dashboard/index.html +261 -261
- package/src/mcp/apps/squad-dashboard/mcp-manifest.json +23 -23
- package/src/mcp/resources/squad-state.js +130 -130
- package/src/mcp-connectors/registry.js +602 -602
- package/src/memory-reflect-engine.js +359 -0
- package/src/notify-renderer.js +32 -0
- package/src/onboarding.js +305 -305
- package/src/parallel-workspace.js +756 -0
- package/src/parser.js +66 -59
- package/src/path-guard.js +47 -0
- package/src/permissions-generator.js +400 -0
- package/src/preflight-engine.js +654 -443
- package/src/prompt-tool.js +20 -20
- package/src/qa-html-report.js +472 -472
- package/src/recovery-context-session.js +154 -154
- package/src/runner/cascade.js +97 -97
- package/src/runner/cli-launcher.js +109 -109
- package/src/runner/plan-importer.js +63 -63
- package/src/runner/queue-store.js +159 -159
- package/src/runtime-store.js +2720 -2676
- package/src/sandbox.js +194 -177
- package/src/self-healing.js +142 -0
- package/src/session-handoff.js +295 -77
- package/src/squad/agent-teams-adapter.js +270 -264
- package/src/squad/brief-validator.js +350 -350
- package/src/squad/bus-bridge.js +140 -140
- package/src/squad/context-compactor.js +265 -265
- package/src/squad/cross-ai-synthesizer.js +250 -250
- package/src/squad/external-session.js +180 -180
- package/src/squad/hooks-generator.js +196 -196
- package/src/squad/inter-squad-events.js +175 -175
- package/src/squad/inter-squad.js +74 -74
- package/src/squad/intra-bus.js +345 -345
- package/src/squad/learning-extractor.js +213 -213
- package/src/squad/pattern-detector.js +365 -365
- package/src/squad/preflight-context.js +296 -296
- package/src/squad/recovery-context.js +372 -372
- package/src/squad/reflection.js +365 -365
- package/src/squad/squad-scaffold.js +341 -177
- package/src/squad/state-manager.js +310 -310
- package/src/squad/task-decomposer.js +652 -652
- package/src/squad/verify-gate.js +303 -303
- package/src/squad/worktree-manager.js +114 -114
- package/src/squad-daemon.js +490 -490
- package/src/squad-dashboard/api.js +223 -223
- package/src/squad-dashboard/attachment-handler.js +93 -93
- package/src/squad-dashboard/context-monitor.js +157 -157
- package/src/squad-dashboard/execution-logs.js +115 -115
- package/src/squad-dashboard/hunk-review.js +209 -209
- package/src/squad-dashboard/metrics.js +133 -133
- package/src/squad-dashboard/process-monitor.js +125 -125
- package/src/squad-dashboard/renderer.js +858 -858
- package/src/squad-dashboard/server.js +232 -232
- package/src/squad-dashboard/styles.js +525 -525
- package/src/squad-dashboard/token-tracker.js +99 -99
- package/src/squads/apply-genome.js +21 -21
- package/src/squads/genome-binding-service.js +154 -154
- package/src/sub-task-engine.js +415 -0
- package/src/sub-task-schemas.js +150 -0
- package/src/sub-task-state.js +152 -0
- package/src/sub-task-telemetry.js +69 -0
- package/src/test-briefing.js +226 -0
- package/src/tool-executor.js +94 -94
- package/src/updater.js +39 -39
- package/src/utils.js +49 -46
- package/src/version.js +50 -50
- package/src/web.js +284 -284
- package/src/worker-runner.js +541 -524
- package/src/workflow-gates.js +185 -0
- package/template/.aioson/advisors/.gitkeep +1 -1
- package/template/.aioson/agents/analyst.md +333 -372
- package/template/.aioson/agents/architect.md +325 -338
- package/template/.aioson/agents/briefing.md +264 -0
- package/template/.aioson/agents/committer.md +161 -0
- package/template/.aioson/agents/copywriter.md +937 -463
- package/template/.aioson/agents/design-hybrid-forge.md +141 -141
- package/template/.aioson/agents/dev.md +295 -779
- package/template/.aioson/agents/deyvin.md +198 -290
- package/template/.aioson/agents/discover.md +235 -0
- package/template/.aioson/agents/discovery-design-doc.md +56 -264
- package/template/.aioson/agents/genome.md +1904 -314
- package/template/.aioson/agents/manifests/analyst.manifest.json +26 -0
- package/template/.aioson/agents/manifests/architect.manifest.json +23 -0
- package/template/.aioson/agents/manifests/committer.manifest.json +23 -0
- package/template/.aioson/agents/manifests/dev.manifest.json +54 -0
- package/template/.aioson/agents/manifests/deyvin.manifest.json +41 -0
- package/template/.aioson/agents/manifests/orchestrator.manifest.json +30 -0
- package/template/.aioson/agents/manifests/pentester.manifest.json +39 -0
- package/template/.aioson/agents/manifests/pm.manifest.json +26 -0
- package/template/.aioson/agents/manifests/product.manifest.json +23 -0
- package/template/.aioson/agents/manifests/qa.manifest.json +41 -0
- package/template/.aioson/agents/manifests/setup.manifest.json +20 -0
- package/template/.aioson/agents/manifests/ux-ui.manifest.json +24 -0
- package/template/.aioson/agents/neo.md +341 -233
- package/template/.aioson/agents/orache.md +430 -434
- package/template/.aioson/agents/orchestrator.md +274 -364
- package/template/.aioson/agents/pair.md +5 -5
- package/template/.aioson/agents/pentester.md +289 -0
- package/template/.aioson/agents/pm.md +141 -194
- package/template/.aioson/agents/product.md +351 -518
- package/template/.aioson/agents/profiler-enricher.md +331 -280
- package/template/.aioson/agents/profiler-forge.md +212 -202
- package/template/.aioson/agents/profiler-researcher.md +282 -259
- package/template/.aioson/agents/qa.md +432 -688
- package/template/.aioson/agents/setup.md +423 -649
- package/template/.aioson/agents/sheldon.md +259 -829
- package/template/.aioson/agents/site-forge.md +281 -1753
- package/template/.aioson/agents/squad.md +160 -2027
- package/template/.aioson/agents/tester.md +536 -463
- package/template/.aioson/agents/ux-ui.md +195 -870
- package/template/.aioson/agents/validator.md +101 -0
- package/template/.aioson/brains/README.md +132 -128
- package/template/.aioson/brains/_archived/.gitkeep +0 -0
- package/template/.aioson/brains/_index.json +34 -16
- package/template/.aioson/brains/dev/patterns.brain.json +79 -0
- package/template/.aioson/brains/scripts/query.js +107 -103
- package/template/.aioson/brains/sheldon/architecture-decisions.brain.json +79 -0
- package/template/.aioson/brains/site-forge/visual-patterns.brain.json +205 -205
- package/template/.aioson/config/autonomy-protocol.json +125 -0
- package/template/.aioson/config/learning-loop.json +10 -0
- package/template/.aioson/config/scout-engine.json +1 -0
- package/template/.aioson/config.md +410 -382
- package/template/.aioson/constitution.md +36 -33
- package/template/.aioson/context/_archived/.gitkeep +0 -0
- package/template/.aioson/context/design-doc.md +136 -0
- package/template/.aioson/context/project-map.md +57 -0
- package/template/.aioson/context/project-pulse.md +34 -34
- package/template/.aioson/context/seeds/seed-example.md +27 -27
- package/template/.aioson/context/spec.md.template +54 -54
- package/template/.aioson/context/user-profile.md +42 -42
- package/template/.aioson/design-docs/code-reuse.md +48 -0
- package/template/.aioson/design-docs/componentization.md +47 -0
- package/template/.aioson/design-docs/file-size.md +52 -0
- package/template/.aioson/design-docs/folder-structure.md +51 -0
- package/template/.aioson/design-docs/naming.md +54 -0
- package/template/.aioson/docs/LAYERS.md +89 -79
- package/template/.aioson/docs/README.md +76 -76
- package/template/.aioson/docs/autonomy-protocol.md +80 -0
- package/template/.aioson/docs/briefing/briefing-craft.md +237 -0
- package/template/.aioson/docs/dev/execution-discipline.md +106 -0
- package/template/.aioson/docs/dev/stack-conventions.md +83 -0
- package/template/.aioson/docs/deyvin/continuity-recovery.md +57 -0
- package/template/.aioson/docs/deyvin/debugging-escalation.md +30 -0
- package/template/.aioson/docs/deyvin/pair-execution.md +44 -0
- package/template/.aioson/docs/deyvin/runtime-handoffs.md +36 -0
- package/template/.aioson/docs/example-external-api-context.md +72 -72
- package/template/.aioson/docs/pentester/app-playbooks.md +206 -0
- package/template/.aioson/docs/pentester/llm-supplychain.md +165 -0
- package/template/.aioson/docs/product/conversation-playbook.md +116 -0
- package/template/.aioson/docs/product/prd-contract.md +107 -0
- package/template/.aioson/docs/product/quality-lens.md +57 -0
- package/template/.aioson/docs/product/research-loop.md +65 -0
- package/template/.aioson/docs/sheldon/enrichment-paths.md +134 -0
- package/template/.aioson/docs/sheldon/harness-contract.md +118 -0
- package/template/.aioson/docs/sheldon/quality-lens.md +57 -0
- package/template/.aioson/docs/sheldon/research-loop.md +56 -0
- package/template/.aioson/docs/sheldon/web-intelligence.md +75 -0
- package/template/.aioson/docs/site-forge-build.md +195 -0
- package/template/.aioson/docs/site-forge-extraction.md +135 -0
- package/template/.aioson/docs/site-forge-qa.md +155 -0
- package/template/.aioson/docs/site-forge-recon.md +434 -0
- package/template/.aioson/docs/site-forge-transform.md +249 -0
- package/template/.aioson/docs/squad/content-output.md +91 -0
- package/template/.aioson/docs/squad/creation-flow.md +149 -0
- package/template/.aioson/docs/squad/domain-breadth.md +322 -0
- package/template/.aioson/docs/squad/domain-classification.md +117 -0
- package/template/.aioson/docs/squad/genome-bindings.md +47 -0
- package/template/.aioson/docs/squad/package-contract.md +260 -0
- package/template/.aioson/docs/squad/quality-lens.md +60 -0
- package/template/.aioson/docs/squad/research-loop.md +59 -0
- package/template/.aioson/docs/squad/session-operations.md +117 -0
- package/template/.aioson/docs/squad/workflow-quality.md +165 -0
- package/template/.aioson/docs/tester/coverage-quality.md +351 -0
- package/template/.aioson/docs/ux-ui/accessibility-audit.md +55 -0
- package/template/.aioson/docs/ux-ui/audit-mode.md +86 -0
- package/template/.aioson/docs/ux-ui/component-map.md +35 -0
- package/template/.aioson/docs/ux-ui/design-execution.md +111 -0
- package/template/.aioson/docs/ux-ui/design-gate.md +27 -0
- package/template/.aioson/docs/ux-ui/research-mode.md +39 -0
- package/template/.aioson/docs/ux-ui/site-delivery.md +156 -0
- package/template/.aioson/docs/ux-ui/token-contract.md +57 -0
- package/template/.aioson/genomes/INDEX.md +195 -0
- package/template/.aioson/genomes/copywriting/SKILL.md +137 -0
- package/template/.aioson/genomes/copywriting/manifest.json +140 -0
- package/template/.aioson/genomes/copywriting/references/application-notes.md +145 -0
- package/template/.aioson/genomes/copywriting/references/decision-weights.md +45 -0
- package/template/.aioson/genomes/copywriting/references/frameworks/5-act-narrative.md +184 -0
- package/template/.aioson/genomes/copywriting/references/frameworks/classical-formulas.md +164 -0
- package/template/.aioson/genomes/copywriting/references/frameworks/offer-stack.md +195 -0
- package/template/.aioson/genomes/copywriting/references/frameworks/one-belief.md +135 -0
- package/template/.aioson/genomes/copywriting/references/frameworks/pms-research.md +211 -0
- package/template/.aioson/genomes/copywriting/references/frameworks/two-paths-close.md +190 -0
- package/template/.aioson/genomes/copywriting/references/heuristics.md +114 -0
- package/template/.aioson/genomes/copywriting/references/meta-axioms.md +68 -0
- package/template/.aioson/genomes/copywriting/references/methodology.md +115 -0
- package/template/.aioson/genomes/copywriting-brunson/SKILL.md +133 -0
- package/template/.aioson/genomes/copywriting-brunson/manifest.json +152 -0
- package/template/.aioson/genomes/copywriting-brunson/references/application-notes.md +113 -0
- package/template/.aioson/genomes/copywriting-brunson/references/decision-weights.md +33 -0
- package/template/.aioson/genomes/copywriting-brunson/references/evidence-and-attribution.md +81 -0
- package/template/.aioson/genomes/copywriting-brunson/references/frameworks/6-part-structure.md +136 -0
- package/template/.aioson/genomes/copywriting-brunson/references/frameworks/origin-story.md +121 -0
- package/template/.aioson/genomes/copywriting-brunson/references/frameworks/perfect-webinar-script.md +139 -0
- package/template/.aioson/genomes/copywriting-brunson/references/frameworks/persuasive-storytelling-5-structures.md +164 -0
- package/template/.aioson/genomes/copywriting-brunson/references/frameworks/value-stack.md +136 -0
- package/template/.aioson/genomes/copywriting-brunson/references/frameworks/who-what-why-how.md +110 -0
- package/template/.aioson/genomes/copywriting-brunson/references/meta-axioms.md +36 -0
- package/template/.aioson/genomes/copywriting-brunson/references/methodology.md +112 -0
- package/template/.aioson/git-guard.json +12 -0
- package/template/.aioson/mcp/servers.md +23 -24
- package/template/.aioson/profiler-reports/.gitkeep +1 -1
- package/template/.aioson/rules/README.md +69 -69
- package/template/.aioson/rules/_archived/.gitkeep +0 -0
- package/template/.aioson/rules/agent-language-policy.md +93 -0
- package/template/.aioson/rules/aioson-context-boundary.md +63 -0
- package/template/.aioson/rules/canonical-path-contract.md +47 -0
- package/template/.aioson/rules/data-format-convention.md +74 -136
- package/template/.aioson/rules/disk-first-artifacts.md +44 -0
- package/template/.aioson/rules/example-monetary-values.md +30 -30
- package/template/.aioson/rules/output-brevity.md +44 -0
- package/template/.aioson/rules/prd-section-ownership.md +49 -0
- package/template/.aioson/rules/security-baseline.md +139 -0
- package/template/.aioson/rules/spec-level-ownership.md +61 -0
- package/template/.aioson/rules/squad/README.md +50 -50
- package/template/.aioson/rules/squad-driver-pattern.md +81 -0
- package/template/.aioson/schemas/content-blueprint.schema.json +30 -30
- package/template/.aioson/schemas/genome-meta.schema.json +150 -150
- package/template/.aioson/schemas/genome.schema.json +115 -115
- package/template/.aioson/schemas/readiness.schema.json +27 -27
- package/template/.aioson/schemas/squad-blueprint.schema.json +228 -204
- package/template/.aioson/schemas/squad-manifest.schema.json +874 -830
- package/template/.aioson/skills/design/aurora-command-ui/SKILL.md +243 -243
- package/template/.aioson/skills/design/aurora-command-ui/references/art-direction.md +293 -293
- package/template/.aioson/skills/design/aurora-command-ui/references/components.md +827 -827
- package/template/.aioson/skills/design/aurora-command-ui/references/dashboards.md +250 -250
- package/template/.aioson/skills/design/aurora-command-ui/references/design-tokens.md +585 -585
- package/template/.aioson/skills/design/aurora-command-ui/references/motion.md +365 -365
- package/template/.aioson/skills/design/aurora-command-ui/references/patterns.md +482 -482
- package/template/.aioson/skills/design/aurora-command-ui/references/websites.md +387 -387
- package/template/.aioson/skills/design/bold-editorial-ui/SKILL.md +205 -205
- package/template/.aioson/skills/design/bold-editorial-ui/references/art-direction.md +338 -338
- package/template/.aioson/skills/design/bold-editorial-ui/references/components.md +977 -977
- package/template/.aioson/skills/design/bold-editorial-ui/references/dashboards.md +218 -218
- package/template/.aioson/skills/design/bold-editorial-ui/references/design-tokens.md +326 -326
- package/template/.aioson/skills/design/bold-editorial-ui/references/motion.md +461 -461
- package/template/.aioson/skills/design/bold-editorial-ui/references/patterns.md +293 -293
- package/template/.aioson/skills/design/bold-editorial-ui/references/websites.md +352 -352
- package/template/.aioson/skills/design/clean-saas-ui/SKILL.md +210 -210
- package/template/.aioson/skills/design/clean-saas-ui/references/art-direction.md +319 -319
- package/template/.aioson/skills/design/clean-saas-ui/references/components.md +365 -365
- package/template/.aioson/skills/design/clean-saas-ui/references/dashboards.md +196 -196
- package/template/.aioson/skills/design/clean-saas-ui/references/design-tokens.md +244 -244
- package/template/.aioson/skills/design/clean-saas-ui/references/motion.md +235 -235
- package/template/.aioson/skills/design/clean-saas-ui/references/patterns.md +215 -215
- package/template/.aioson/skills/design/clean-saas-ui/references/websites.md +295 -295
- package/template/.aioson/skills/design/cognitive-core-ui/SKILL.md +203 -203
- package/template/.aioson/skills/design/cognitive-core-ui/references/art-direction.md +339 -339
- package/template/.aioson/skills/design/cognitive-core-ui/references/components.md +407 -407
- package/template/.aioson/skills/design/cognitive-core-ui/references/dashboards.md +272 -272
- package/template/.aioson/skills/design/cognitive-core-ui/references/design-tokens.md +524 -524
- package/template/.aioson/skills/design/cognitive-core-ui/references/motion.md +279 -279
- package/template/.aioson/skills/design/cognitive-core-ui/references/patterns.md +289 -289
- package/template/.aioson/skills/design/cognitive-core-ui/references/websites.md +437 -437
- package/template/.aioson/skills/design/glassmorphism-ui/SKILL.md +222 -222
- package/template/.aioson/skills/design/glassmorphism-ui/references/art-direction.md +159 -159
- package/template/.aioson/skills/design/glassmorphism-ui/references/components.md +498 -498
- package/template/.aioson/skills/design/glassmorphism-ui/references/dashboards.md +236 -236
- package/template/.aioson/skills/design/glassmorphism-ui/references/design-tokens.md +274 -274
- package/template/.aioson/skills/design/glassmorphism-ui/references/motion.md +355 -355
- package/template/.aioson/skills/design/glassmorphism-ui/references/patterns.md +198 -198
- package/template/.aioson/skills/design/glassmorphism-ui/references/websites.md +307 -307
- package/template/.aioson/skills/design/interface-design/SKILL.md +47 -47
- package/template/.aioson/skills/design/interface-design/references/components-and-states.md +105 -105
- package/template/.aioson/skills/design/interface-design/references/design-directions.md +101 -101
- package/template/.aioson/skills/design/interface-design/references/handoff-and-quality.md +71 -71
- package/template/.aioson/skills/design/interface-design/references/intent-and-domain.md +74 -74
- package/template/.aioson/skills/design/interface-design/references/tokens-and-depth.md +173 -173
- package/template/.aioson/skills/design/neo-brutalist-ui/SKILL.md +213 -213
- package/template/.aioson/skills/design/neo-brutalist-ui/references/art-direction.md +228 -228
- package/template/.aioson/skills/design/neo-brutalist-ui/references/components.md +855 -855
- package/template/.aioson/skills/design/neo-brutalist-ui/references/dashboards.md +334 -334
- package/template/.aioson/skills/design/neo-brutalist-ui/references/design-tokens.md +342 -342
- package/template/.aioson/skills/design/neo-brutalist-ui/references/motion.md +286 -286
- package/template/.aioson/skills/design/neo-brutalist-ui/references/patterns.md +458 -458
- package/template/.aioson/skills/design/neo-brutalist-ui/references/websites.md +723 -723
- package/template/.aioson/skills/design/premium-command-center-ui/SKILL.md +62 -62
- package/template/.aioson/skills/design/premium-command-center-ui/references/operations.md +74 -74
- package/template/.aioson/skills/design/premium-command-center-ui/references/patterns.md +116 -116
- package/template/.aioson/skills/design/premium-command-center-ui/references/validation.md +47 -47
- package/template/.aioson/skills/design/premium-command-center-ui/references/visual-system.md +215 -215
- package/template/.aioson/skills/design/pt.squarespace.com/.skill-meta.json +31 -31
- package/template/.aioson/skills/design/pt.squarespace.com/SKILL.md +66 -66
- package/template/.aioson/skills/design/pt.squarespace.com/references/components.md +368 -368
- package/template/.aioson/skills/design/pt.squarespace.com/references/design-tokens.md +150 -150
- package/template/.aioson/skills/design/pt.squarespace.com/references/motion.md +270 -270
- package/template/.aioson/skills/design/pt.squarespace.com/references/patterns.md +189 -189
- package/template/.aioson/skills/design/pt.squarespace.com/references/websites.md +165 -165
- package/template/.aioson/skills/design/warm-craft-ui/SKILL.md +209 -209
- package/template/.aioson/skills/design/warm-craft-ui/references/art-direction.md +324 -324
- package/template/.aioson/skills/design/warm-craft-ui/references/components.md +508 -508
- package/template/.aioson/skills/design/warm-craft-ui/references/dashboards.md +223 -223
- package/template/.aioson/skills/design/warm-craft-ui/references/design-tokens.md +374 -374
- package/template/.aioson/skills/design/warm-craft-ui/references/motion.md +356 -356
- package/template/.aioson/skills/design/warm-craft-ui/references/patterns.md +288 -288
- package/template/.aioson/skills/design/warm-craft-ui/references/websites.md +289 -289
- package/template/.aioson/skills/design-system/SKILL.md +92 -92
- package/template/.aioson/skills/design-system/components/SKILL.md +274 -274
- package/template/.aioson/skills/design-system/dashboards/SKILL.md +184 -184
- package/template/.aioson/skills/design-system/foundations/SKILL.md +250 -250
- package/template/.aioson/skills/design-system/motion/SKILL.md +197 -197
- package/template/.aioson/skills/design-system/patterns/SKILL.md +231 -231
- package/template/.aioson/skills/dynamic/README.md +30 -30
- package/template/.aioson/skills/dynamic/cardano-docs.md +16 -16
- package/template/.aioson/skills/dynamic/ethereum-docs.md +17 -17
- package/template/.aioson/skills/dynamic/flux-ui-docs.md +13 -13
- package/template/.aioson/skills/dynamic/laravel-docs.md +41 -41
- package/template/.aioson/skills/dynamic/npm-packages.md +16 -16
- package/template/.aioson/skills/dynamic/solana-docs.md +16 -16
- package/template/.aioson/skills/marketing/references/anti-patterns.md +254 -254
- package/template/.aioson/skills/marketing/references/cta-matrix.md +361 -0
- package/template/.aioson/skills/marketing/references/fascinations.md +192 -192
- package/template/.aioson/skills/marketing/references/five-acts.md +248 -248
- package/template/.aioson/skills/marketing/references/headline-matrix.md +358 -0
- package/template/.aioson/skills/marketing/references/market-intelligence.md +198 -198
- package/template/.aioson/skills/marketing/references/offer-structure.md +203 -203
- package/template/.aioson/skills/marketing/references/one-belief.md +149 -149
- package/template/.aioson/skills/marketing/references/patterns.md +218 -218
- package/template/.aioson/skills/marketing/references/platform-constraints.md +337 -0
- package/template/.aioson/skills/marketing/references/pms-research.md +193 -193
- package/template/.aioson/skills/marketing/vsl-craft.md +385 -385
- package/template/.aioson/skills/premium-visual-design/SKILL.md +83 -83
- package/template/.aioson/skills/premium-visual-design/components/agent-badge.md +92 -92
- package/template/.aioson/skills/premium-visual-design/components/dependency-node.md +102 -102
- package/template/.aioson/skills/premium-visual-design/components/mention-autocomplete.md +136 -136
- package/template/.aioson/skills/premium-visual-design/components/notification-center.md +136 -136
- package/template/.aioson/skills/premium-visual-design/components/review-action-bar.md +188 -188
- package/template/.aioson/skills/premium-visual-design/components/team-switcher.md +131 -131
- package/template/.aioson/skills/premium-visual-design/patterns/agent-message-thread.md +198 -198
- package/template/.aioson/skills/premium-visual-design/patterns/notification-panel.md +275 -275
- package/template/.aioson/skills/premium-visual-design/patterns/review-workflow-ui.md +234 -234
- package/template/.aioson/skills/premium-visual-design/patterns/task-dependency-graph.md +147 -147
- package/template/.aioson/skills/premium-visual-design/tokens/status-extended.md +142 -142
- package/template/.aioson/skills/process/aioson-spec-driven/SKILL.md +46 -46
- package/template/.aioson/skills/process/aioson-spec-driven/references/analyst.md +30 -30
- package/template/.aioson/skills/process/aioson-spec-driven/references/approval-gates.md +109 -109
- package/template/.aioson/skills/process/aioson-spec-driven/references/architect.md +23 -23
- package/template/.aioson/skills/process/aioson-spec-driven/references/artifact-map.md +44 -44
- package/template/.aioson/skills/process/aioson-spec-driven/references/classification-map.md +37 -37
- package/template/.aioson/skills/process/aioson-spec-driven/references/dev.md +47 -47
- package/template/.aioson/skills/process/aioson-spec-driven/references/deyvin.md +27 -27
- package/template/.aioson/skills/process/aioson-spec-driven/references/hardening-lane.md +49 -49
- package/template/.aioson/skills/process/aioson-spec-driven/references/maintenance-and-state.md +101 -101
- package/template/.aioson/skills/process/aioson-spec-driven/references/pm.md +30 -0
- package/template/.aioson/skills/process/aioson-spec-driven/references/product.md +25 -25
- package/template/.aioson/skills/process/aioson-spec-driven/references/qa.md +30 -30
- package/template/.aioson/skills/process/aioson-spec-driven/references/sheldon.md +25 -25
- package/template/.aioson/skills/process/aioson-spec-driven/references/ui-language.md +75 -75
- package/template/.aioson/skills/process/design-hybrid-forge/SKILL.md +147 -147
- package/template/.aioson/skills/process/design-hybrid-forge/references/crossover-protocol.md +221 -221
- package/template/.aioson/skills/process/design-hybrid-forge/references/naming-registry.md +88 -88
- package/template/.aioson/skills/process/design-hybrid-forge/references/output-contract.md +306 -306
- package/template/.aioson/skills/process/design-hybrid-forge/references/pair-compatibility.md +149 -149
- package/template/.aioson/skills/process/design-hybrid-forge/references/quality-gates.md +208 -208
- package/template/.aioson/skills/process/design-hybrid-forge/references/variation-library.md +125 -125
- package/template/.aioson/skills/process/secure-tdd/SKILL.md +97 -0
- package/template/.aioson/skills/process/simplify/SKILL.md +173 -173
- package/template/.aioson/skills/references/premium-command-center-ui/master-application-prompt.md +79 -79
- package/template/.aioson/skills/references/premium-command-center-ui/operational-ux-playbook.md +253 -253
- package/template/.aioson/skills/references/premium-command-center-ui/quality-validation-checklist.md +82 -82
- package/template/.aioson/skills/references/premium-command-center-ui/visual-system-and-component-patterns.md +270 -270
- package/template/.aioson/skills/squad/SKILL.md +58 -58
- package/template/.aioson/skills/squad/formats/catalog.json +15 -15
- package/template/.aioson/skills/squad/formats/content/blog-post.md +47 -47
- package/template/.aioson/skills/squad/formats/content/newsletter.md +47 -47
- package/template/.aioson/skills/squad/formats/creative/podcast-script.md +43 -43
- package/template/.aioson/skills/squad/formats/creative/video-script.md +41 -41
- package/template/.aioson/skills/squad/formats/social/instagram-feed.md +42 -42
- package/template/.aioson/skills/squad/formats/social/linkedin-post.md +42 -42
- package/template/.aioson/skills/squad/formats/social/tiktok.md +39 -39
- package/template/.aioson/skills/squad/formats/social/twitter-thread.md +39 -39
- package/template/.aioson/skills/squad/formats/social/youtube-long.md +47 -47
- package/template/.aioson/skills/squad/formats/social/youtube-shorts.md +39 -39
- package/template/.aioson/skills/squad/patterns/multi-platform-pattern.md +108 -108
- package/template/.aioson/skills/squad/patterns/persona-based-pattern.md +98 -98
- package/template/.aioson/skills/squad/patterns/pipeline-pattern.md +106 -106
- package/template/.aioson/skills/squad/patterns/review-loop-pattern.md +81 -81
- package/template/.aioson/skills/squad/references/checklist-templates.md +122 -122
- package/template/.aioson/skills/squad/references/executor-archetypes.md +123 -123
- package/template/.aioson/skills/squad/references/workflow-templates.md +169 -169
- package/template/.aioson/skills/static/context-budget-guide.md +46 -46
- package/template/.aioson/skills/static/debugging-protocol.md +42 -42
- package/template/.aioson/skills/static/django-patterns.md +342 -342
- package/template/.aioson/skills/static/fastapi-patterns.md +344 -344
- package/template/.aioson/skills/static/filament-patterns.md +267 -267
- package/template/.aioson/skills/static/flux-ui-components.md +262 -262
- package/template/.aioson/skills/static/git-conventions.md +227 -227
- package/template/.aioson/skills/static/git-worktrees.md +36 -36
- package/template/.aioson/skills/static/harness-sensors.md +74 -74
- package/template/.aioson/skills/static/harness-validate/SKILL.md +46 -0
- package/template/.aioson/skills/static/jetstream-setup.md +200 -200
- package/template/.aioson/skills/static/landing-page-deploy.md +192 -192
- package/template/.aioson/skills/static/landing-page-forge.md +730 -730
- package/template/.aioson/skills/static/laravel-conventions.md +491 -491
- package/template/.aioson/skills/static/multi-agent-patterns.md +43 -43
- package/template/.aioson/skills/static/nextjs-patterns.md +321 -321
- package/template/.aioson/skills/static/node-express-patterns.md +317 -317
- package/template/.aioson/skills/static/node-typescript-patterns.md +282 -282
- package/template/.aioson/skills/static/rails-conventions.md +307 -307
- package/template/.aioson/skills/static/react-motion-patterns.md +599 -599
- package/template/.aioson/skills/static/static-html-patterns/checklists.md +43 -43
- package/template/.aioson/skills/static/static-html-patterns/css-tokens.md +609 -609
- package/template/.aioson/skills/static/static-html-patterns/motion.md +193 -193
- package/template/.aioson/skills/static/static-html-patterns/premium.md +711 -711
- package/template/.aioson/skills/static/static-html-patterns/structure.md +209 -209
- package/template/.aioson/skills/static/static-html-patterns/utilities.md +190 -190
- package/template/.aioson/skills/static/static-html-patterns.md +80 -80
- package/template/.aioson/skills/static/tall-stack-patterns.md +286 -286
- package/template/.aioson/skills/static/threejs-patterns.md +929 -929
- package/template/.aioson/skills/static/ui-ux-modern.md +76 -76
- package/template/.aioson/skills/static/web-research-cache.md +115 -112
- package/template/.aioson/skills/static/web3-cardano-patterns.md +337 -337
- package/template/.aioson/skills/static/web3-ethereum-patterns.md +310 -310
- package/template/.aioson/skills/static/web3-security-checklist.md +284 -284
- package/template/.aioson/skills/static/web3-solana-patterns.md +324 -324
- package/template/.aioson/squads/memory.md +5 -5
- package/template/.aioson/tasks/implementation-plan.md +327 -327
- package/template/.aioson/tasks/squad-analyze.md +83 -83
- package/template/.aioson/tasks/squad-create.md +148 -121
- package/template/.aioson/tasks/squad-design.md +206 -158
- package/template/.aioson/tasks/squad-execution-plan.md +279 -279
- package/template/.aioson/tasks/squad-export.md +20 -20
- package/template/.aioson/tasks/squad-extend.md +68 -68
- package/template/.aioson/tasks/squad-investigate.md +57 -44
- package/template/.aioson/tasks/squad-learning-review.md +44 -44
- package/template/.aioson/tasks/squad-output-config.md +177 -177
- package/template/.aioson/tasks/squad-pipeline.md +122 -122
- package/template/.aioson/tasks/squad-profile.md +48 -48
- package/template/.aioson/tasks/squad-refresh.md +236 -0
- package/template/.aioson/tasks/squad-repair.md +85 -85
- package/template/.aioson/tasks/squad-review.md +61 -61
- package/template/.aioson/tasks/squad-task-decompose.md +66 -66
- package/template/.aioson/tasks/squad-validate.md +58 -58
- package/template/.aioson/templates/reflect-prompts/current-state.md +36 -0
- package/template/.aioson/templates/reflect-prompts/how-it-works.md +23 -0
- package/template/.aioson/templates/reflect-prompts/what-it-does.md +21 -0
- package/template/.aioson/templates/squads/content-basic/template.json +21 -21
- package/template/.aioson/templates/squads/digital-marketing-agency/template.json +96 -96
- package/template/.aioson/templates/squads/media-channel/template.json +24 -24
- package/template/.aioson/templates/squads/research-analysis/template.json +22 -22
- package/template/.aioson/templates/squads/software-delivery/template.json +21 -21
- package/template/.claude/commands/aioson/agent/analyst.md +5 -5
- package/template/.claude/commands/aioson/agent/architect.md +5 -5
- package/template/.claude/commands/aioson/agent/briefing.md +5 -0
- package/template/.claude/commands/aioson/agent/committer.md +5 -0
- package/template/.claude/commands/aioson/agent/copywriter.md +5 -0
- package/template/.claude/commands/aioson/agent/design-hybrid-forge.md +5 -5
- package/template/.claude/commands/aioson/agent/dev.md +5 -5
- package/template/.claude/commands/aioson/agent/deyvin.md +5 -5
- package/template/.claude/commands/aioson/agent/discover.md +5 -0
- package/template/.claude/commands/aioson/agent/discovery-design-doc.md +5 -5
- package/template/.claude/commands/aioson/agent/genome.md +5 -5
- package/template/.claude/commands/aioson/agent/neo.md +5 -5
- package/template/.claude/commands/aioson/agent/orache.md +5 -5
- package/template/.claude/commands/aioson/agent/orchestrator.md +5 -5
- package/template/.claude/commands/aioson/agent/pair.md +5 -0
- package/template/.claude/commands/aioson/agent/pentester.md +5 -0
- package/template/.claude/commands/aioson/agent/pm.md +5 -5
- package/template/.claude/commands/aioson/agent/product.md +5 -5
- package/template/.claude/commands/aioson/agent/profiler-enricher.md +5 -5
- package/template/.claude/commands/aioson/agent/profiler-forge.md +5 -5
- package/template/.claude/commands/aioson/agent/profiler-researcher.md +5 -5
- package/template/.claude/commands/aioson/agent/qa.md +5 -5
- package/template/.claude/commands/aioson/agent/setup.md +5 -5
- package/template/.claude/commands/aioson/agent/sheldon.md +5 -5
- package/template/.claude/commands/aioson/agent/site-forge.md +5 -5
- package/template/.claude/commands/aioson/agent/squad.md +5 -5
- package/template/.claude/commands/aioson/agent/tester.md +5 -5
- package/template/.claude/commands/aioson/agent/ux-ui.md +5 -5
- package/template/.claude/commands/aioson/agent/validator.md +5 -0
- package/template/.gemini/GEMINI.md +13 -13
- package/template/.gemini/commands/aios-analyst.toml +7 -4
- package/template/.gemini/commands/aios-architect.toml +8 -7
- package/template/.gemini/commands/aios-committer.toml +7 -0
- package/template/.gemini/commands/aios-copywriter.toml +7 -0
- package/template/.gemini/commands/aios-cypher.toml +7 -0
- package/template/.gemini/commands/aios-dev.toml +9 -8
- package/template/.gemini/commands/aios-deyvin.toml +7 -6
- package/template/.gemini/commands/aios-discover.toml +6 -0
- package/template/.gemini/commands/aios-discovery-design-doc.toml +7 -4
- package/template/.gemini/commands/aios-genome.toml +7 -0
- package/template/.gemini/commands/aios-neo.toml +6 -4
- package/template/.gemini/commands/aios-orache.toml +7 -0
- package/template/.gemini/commands/aios-orchestrator.toml +9 -8
- package/template/.gemini/commands/aios-pair.toml +7 -6
- package/template/.gemini/commands/aios-pm.toml +9 -8
- package/template/.gemini/commands/aios-product.toml +6 -4
- package/template/.gemini/commands/aios-qa.toml +7 -6
- package/template/.gemini/commands/aios-setup.toml +6 -3
- package/template/.gemini/commands/aios-sheldon.toml +7 -0
- package/template/.gemini/commands/aios-site-forge.toml +7 -0
- package/template/.gemini/commands/aios-squad.toml +7 -0
- package/template/.gemini/commands/aios-tester.toml +7 -6
- package/template/.gemini/commands/aios-ux-ui.toml +9 -8
- package/template/.gemini/commands/aios-validator.toml +7 -0
- package/template/AGENTS.md +184 -172
- package/template/CLAUDE.md +98 -93
- package/template/OPENCODE.md +35 -34
- package/template/aioson-models.json +40 -40
- package/docs/en/i18n.md +0 -52
- package/docs/en/schemas/parallel-status.schema.json +0 -94
- package/template/.aioson/genomes/copywriting.md +0 -204
- package/template/.aioson/locales/en/agents/analyst.md +0 -244
- package/template/.aioson/locales/en/agents/architect.md +0 -245
- package/template/.aioson/locales/en/agents/dev.md +0 -397
- package/template/.aioson/locales/en/agents/deyvin.md +0 -137
- package/template/.aioson/locales/en/agents/discovery-design-doc.md +0 -27
- package/template/.aioson/locales/en/agents/genome.md +0 -212
- package/template/.aioson/locales/en/agents/neo.md +0 -8
- package/template/.aioson/locales/en/agents/orache.md +0 -6
- package/template/.aioson/locales/en/agents/orchestrator.md +0 -189
- package/template/.aioson/locales/en/agents/pair.md +0 -5
- package/template/.aioson/locales/en/agents/pm.md +0 -84
- package/template/.aioson/locales/en/agents/product.md +0 -378
- package/template/.aioson/locales/en/agents/profiler-enricher.md +0 -5
- package/template/.aioson/locales/en/agents/profiler-forge.md +0 -5
- package/template/.aioson/locales/en/agents/profiler-researcher.md +0 -5
- package/template/.aioson/locales/en/agents/qa.md +0 -270
- package/template/.aioson/locales/en/agents/setup.md +0 -421
- package/template/.aioson/locales/en/agents/sheldon.md +0 -455
- package/template/.aioson/locales/en/agents/squad.md +0 -449
- package/template/.aioson/locales/en/agents/tester.md +0 -6
- package/template/.aioson/locales/en/agents/ux-ui.md +0 -668
- package/template/.aioson/locales/es/agents/analyst.md +0 -225
- package/template/.aioson/locales/es/agents/architect.md +0 -245
- package/template/.aioson/locales/es/agents/dev.md +0 -370
- package/template/.aioson/locales/es/agents/deyvin.md +0 -99
- package/template/.aioson/locales/es/agents/discovery-design-doc.md +0 -21
- package/template/.aioson/locales/es/agents/genome.md +0 -104
- package/template/.aioson/locales/es/agents/neo.md +0 -50
- package/template/.aioson/locales/es/agents/orache.md +0 -105
- package/template/.aioson/locales/es/agents/orchestrator.md +0 -194
- package/template/.aioson/locales/es/agents/pair.md +0 -7
- package/template/.aioson/locales/es/agents/pm.md +0 -90
- package/template/.aioson/locales/es/agents/product.md +0 -372
- package/template/.aioson/locales/es/agents/profiler-enricher.md +0 -7
- package/template/.aioson/locales/es/agents/profiler-forge.md +0 -7
- package/template/.aioson/locales/es/agents/profiler-researcher.md +0 -7
- package/template/.aioson/locales/es/agents/qa.md +0 -198
- package/template/.aioson/locales/es/agents/setup.md +0 -405
- package/template/.aioson/locales/es/agents/sheldon.md +0 -309
- package/template/.aioson/locales/es/agents/squad.md +0 -532
- package/template/.aioson/locales/es/agents/tester.md +0 -9
- package/template/.aioson/locales/es/agents/ux-ui.md +0 -212
- package/template/.aioson/locales/fr/agents/analyst.md +0 -225
- package/template/.aioson/locales/fr/agents/architect.md +0 -245
- package/template/.aioson/locales/fr/agents/dev.md +0 -370
- package/template/.aioson/locales/fr/agents/deyvin.md +0 -99
- package/template/.aioson/locales/fr/agents/discovery-design-doc.md +0 -21
- package/template/.aioson/locales/fr/agents/genome.md +0 -104
- package/template/.aioson/locales/fr/agents/neo.md +0 -50
- package/template/.aioson/locales/fr/agents/orache.md +0 -106
- package/template/.aioson/locales/fr/agents/orchestrator.md +0 -194
- package/template/.aioson/locales/fr/agents/pair.md +0 -7
- package/template/.aioson/locales/fr/agents/pm.md +0 -90
- package/template/.aioson/locales/fr/agents/product.md +0 -372
- package/template/.aioson/locales/fr/agents/profiler-enricher.md +0 -7
- package/template/.aioson/locales/fr/agents/profiler-forge.md +0 -7
- package/template/.aioson/locales/fr/agents/profiler-researcher.md +0 -7
- package/template/.aioson/locales/fr/agents/qa.md +0 -198
- package/template/.aioson/locales/fr/agents/setup.md +0 -405
- package/template/.aioson/locales/fr/agents/sheldon.md +0 -309
- package/template/.aioson/locales/fr/agents/squad.md +0 -532
- package/template/.aioson/locales/fr/agents/tester.md +0 -9
- package/template/.aioson/locales/fr/agents/ux-ui.md +0 -212
- package/template/.aioson/locales/pt-BR/agents/analyst.md +0 -319
- package/template/.aioson/locales/pt-BR/agents/architect.md +0 -284
- package/template/.aioson/locales/pt-BR/agents/dev.md +0 -483
- package/template/.aioson/locales/pt-BR/agents/deyvin.md +0 -184
- package/template/.aioson/locales/pt-BR/agents/discovery-design-doc.md +0 -198
- package/template/.aioson/locales/pt-BR/agents/genome.md +0 -297
- package/template/.aioson/locales/pt-BR/agents/neo.md +0 -208
- package/template/.aioson/locales/pt-BR/agents/orache.md +0 -137
- package/template/.aioson/locales/pt-BR/agents/orchestrator.md +0 -324
- package/template/.aioson/locales/pt-BR/agents/pair.md +0 -5
- package/template/.aioson/locales/pt-BR/agents/pm.md +0 -182
- package/template/.aioson/locales/pt-BR/agents/product.md +0 -466
- package/template/.aioson/locales/pt-BR/agents/profiler-enricher.md +0 -5
- package/template/.aioson/locales/pt-BR/agents/profiler-forge.md +0 -5
- package/template/.aioson/locales/pt-BR/agents/profiler-researcher.md +0 -5
- package/template/.aioson/locales/pt-BR/agents/qa.md +0 -300
- package/template/.aioson/locales/pt-BR/agents/setup.md +0 -533
- package/template/.aioson/locales/pt-BR/agents/sheldon.md +0 -323
- package/template/.aioson/locales/pt-BR/agents/squad.md +0 -1330
- package/template/.aioson/locales/pt-BR/agents/tester.md +0 -449
- package/template/.aioson/locales/pt-BR/agents/ux-ui.md +0 -669
package/src/commands/cloud.js
CHANGED
|
@@ -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
|
+
};
|