sinapse-ai 1.8.0 → 1.9.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.claude/hooks/mind-clone-governance.py +212 -212
- package/.claude/hooks/read-protection.py +152 -152
- package/.claude/hooks/slug-validation.py +175 -175
- package/.claude/hooks/sql-governance.py +183 -183
- package/.claude/rules/documentation-first.md +1 -1
- package/.claude/rules/hook-governance.md +1 -1
- package/.claude/rules/mandatory-delegation.md +1 -1
- package/.claude/rules/project-intelligence.md +1 -1
- package/.codex/agents/analyst.md +4 -371
- package/.codex/agents/animations-orqx.md +4 -57
- package/.codex/agents/architect.md +4 -560
- package/.codex/agents/brand-orqx.md +4 -95
- package/.codex/agents/claude-mastery-chief.md +4 -0
- package/.codex/agents/cloning-orqx.md +4 -70
- package/.codex/agents/commercial-orqx.md +4 -67
- package/.codex/agents/config-engineer.md +2 -2
- package/.codex/agents/content-orqx.md +4 -77
- package/.codex/agents/copy-orqx.md +4 -65
- package/.codex/agents/cost-optimizer.md +4 -0
- package/.codex/agents/council-orqx.md +4 -68
- package/.codex/agents/courses-orqx.md +4 -64
- package/.codex/agents/cro-persuasion.md +4 -0
- package/.codex/agents/cyber-orqx.md +4 -67
- package/.codex/agents/data-engineer.md +4 -542
- package/.codex/agents/design-orqx.md +4 -65
- package/.codex/agents/design-system.md +4 -210
- package/.codex/agents/developer.md +4 -666
- package/.codex/agents/devops.md +4 -668
- package/.codex/agents/finance-orqx.md +4 -57
- package/.codex/agents/fiscal-compliance-br.md +4 -0
- package/.codex/agents/forecast-strategist.md +4 -0
- package/.codex/agents/growth-orqx.md +4 -75
- package/.codex/agents/hooks-architect.md +2 -2
- package/.codex/agents/mcp-integrator.md +2 -2
- package/.codex/agents/paidmedia-orqx.md +4 -67
- package/.codex/agents/platform-aesthetic-director.md +4 -0
- package/.codex/agents/premium-packaging-strategist.md +4 -0
- package/.codex/agents/product-lead.md +4 -371
- package/.codex/agents/product-orqx.md +4 -57
- package/.codex/agents/product-surface-director.md +4 -0
- package/.codex/agents/project-integrator.md +2 -2
- package/.codex/agents/project-lead.md +4 -414
- package/.codex/agents/quality-gate.md +4 -547
- package/.codex/agents/research-orqx.md +4 -67
- package/.codex/agents/roadmap-sentinel.md +2 -2
- package/.codex/agents/skill-craftsman.md +2 -2
- package/.codex/agents/snps-orqx.md +4 -684
- package/.codex/agents/sop-extractor.md +4 -61
- package/.codex/agents/sprint-lead.md +4 -324
- package/.codex/agents/squad-creator.md +4 -402
- package/.codex/agents/storytelling-orqx.md +4 -65
- package/.codex/agents/swarm-orqx.md +4 -64
- package/.codex/agents/ux-design-expert.md +4 -532
- package/.codex/agents/ux-designer.md +4 -124
- package/.codex/command-registry.json +9 -9
- package/.codex/delegation-matrix.json +375 -839
- package/.codex/delegation-parity.json +658 -0
- package/.codex/handoff-packet.parity.schema.json +148 -0
- package/.codex/handoff-packet.template.json +26 -0
- package/.codex/instructions.md +8 -8
- package/.codex/scripts/resolve-codex-agent.js +482 -0
- package/.codex/scripts/resolve-codex-command.js +75 -12
- package/.codex/scripts/resolve-codex-delegation.js +131 -92
- package/.codex/skills/sinapse-claude/SKILL.md +3 -3
- package/.codex/skills/sinapse-po/SKILL.md +1 -1
- package/.codex/tasks/resolve-sinapse-conflict.md +1 -1
- package/.sinapse-ai/constitution.md +5 -5
- package/.sinapse-ai/core/doctor/checks/git-hooks.js +163 -19
- package/.sinapse-ai/core/events/dashboard-emitter.js +30 -9
- package/.sinapse-ai/core/execution/subagent-dispatcher.js +1 -1
- package/.sinapse-ai/core/synapse/engine.js +15 -0
- package/.sinapse-ai/core/ui/observability-panel.js +240 -0
- package/.sinapse-ai/core-config.yaml +0 -20
- package/.sinapse-ai/data/entity-registry.yaml +185 -236
- package/.sinapse-ai/development/agents/snps-orqx.md +16 -26
- package/.sinapse-ai/development/tasks/build-autonomous.md +11 -1
- package/.sinapse-ai/development/tasks/build-resume.md +8 -0
- package/.sinapse-ai/development/tasks/build-status.md +8 -0
- package/.sinapse-ai/development/tasks/build.md +8 -0
- package/.sinapse-ai/development/tasks/cleanup-worktrees.md +8 -1
- package/.sinapse-ai/development/tasks/gotcha.md +8 -0
- package/.sinapse-ai/development/tasks/gotchas.md +8 -0
- package/.sinapse-ai/development/tasks/ids-health.md +14 -6
- package/.sinapse-ai/development/tasks/list-mcps.md +15 -0
- package/.sinapse-ai/development/tasks/merge-worktree.md +8 -1
- package/.sinapse-ai/development/tasks/qa-review-build.md +18 -0
- package/.sinapse-ai/development/tasks/remove-mcp.md +8 -1
- package/.sinapse-ai/development/tasks/validate-agents.md +26 -14
- package/.sinapse-ai/development/templates/service-template/README.md.hbs +159 -159
- package/.sinapse-ai/development/templates/service-template/__tests__/index.test.ts.hbs +238 -238
- package/.sinapse-ai/development/templates/service-template/client.ts.hbs +404 -404
- package/.sinapse-ai/development/templates/service-template/errors.ts.hbs +183 -183
- package/.sinapse-ai/development/templates/service-template/index.ts.hbs +121 -121
- package/.sinapse-ai/development/templates/service-template/package.json.hbs +88 -88
- package/.sinapse-ai/development/templates/service-template/types.ts.hbs +146 -146
- package/.sinapse-ai/development/templates/squad-template/LICENSE +22 -22
- package/.sinapse-ai/git-hooks/lib/framework-guard.js +258 -0
- package/.sinapse-ai/git-hooks/lib/secret-scanner-core.js +355 -0
- package/.sinapse-ai/git-hooks/lib/staged-secret-scan.js +179 -0
- package/.sinapse-ai/git-hooks/lib/staged-sql-guard.js +204 -0
- package/.sinapse-ai/git-hooks/post-commit +28 -0
- package/.sinapse-ai/git-hooks/pre-commit +81 -0
- package/.sinapse-ai/git-hooks/pre-push +83 -0
- package/.sinapse-ai/hooks/ids-post-commit.js +13 -11
- package/.sinapse-ai/hooks/ids-pre-push.js +9 -7
- package/.sinapse-ai/infrastructure/scripts/codex-parity/resolve.js +161 -0
- package/.sinapse-ai/infrastructure/scripts/dashboard-status-writer.js +6 -2
- package/.sinapse-ai/infrastructure/scripts/ide-sync/index.js +65 -68
- package/.sinapse-ai/infrastructure/scripts/sync-codex-local-first.js +156 -1
- package/.sinapse-ai/infrastructure/scripts/validate-codex-delegation.js +1 -4
- package/.sinapse-ai/infrastructure/scripts/validate-codex-integration.js +41 -5
- package/.sinapse-ai/infrastructure/templates/coderabbit.yaml.template +280 -280
- package/.sinapse-ai/infrastructure/templates/config/env.example +16 -16
- package/.sinapse-ai/infrastructure/templates/config/gitignore-additions.tmpl +59 -59
- package/.sinapse-ai/infrastructure/templates/github/CODEOWNERS.template +12 -12
- package/.sinapse-ai/infrastructure/templates/github-workflows/ci.yml.template +170 -170
- package/.sinapse-ai/infrastructure/templates/github-workflows/pr-automation.yml.template +331 -331
- package/.sinapse-ai/infrastructure/templates/github-workflows/release.yml.template +197 -197
- package/.sinapse-ai/infrastructure/templates/gitignore/gitignore-brownfield-merge.tmpl +19 -19
- package/.sinapse-ai/infrastructure/templates/gitignore/gitignore-node.tmpl +86 -86
- package/.sinapse-ai/infrastructure/templates/gitignore/gitignore-python.tmpl +146 -146
- package/.sinapse-ai/infrastructure/templates/gitignore/gitignore-sinapse-base.tmpl +64 -64
- package/.sinapse-ai/infrastructure/templates/safe-collab/CODEOWNERS.template +16 -16
- package/.sinapse-ai/infrastructure/templates/sinapse-sync.yaml.template +183 -183
- package/.sinapse-ai/install-manifest.yaml +112 -164
- package/.sinapse-ai/local-config.yaml.template +65 -65
- package/.sinapse-ai/product/templates/adr.hbs +126 -126
- package/.sinapse-ai/product/templates/dbdr.hbs +242 -242
- package/.sinapse-ai/product/templates/epic.hbs +213 -213
- package/.sinapse-ai/product/templates/ide-rules/codex-rules.md +30 -0
- package/.sinapse-ai/product/templates/pmdr.hbs +187 -187
- package/.sinapse-ai/product/templates/prd-v2.0.hbs +217 -217
- package/.sinapse-ai/product/templates/prd.hbs +202 -202
- package/.sinapse-ai/product/templates/statusline/statusline-script.js +31 -8
- package/.sinapse-ai/product/templates/statusline/track-agent-clear.cjs +79 -0
- package/.sinapse-ai/product/templates/statusline/track-agent.cjs +218 -0
- package/.sinapse-ai/product/templates/story.hbs +264 -264
- package/.sinapse-ai/product/templates/task.hbs +171 -171
- package/.sinapse-ai/product/templates/tmpl-comment-on-examples.sql +159 -159
- package/.sinapse-ai/product/templates/tmpl-migration-script.sql +92 -92
- package/.sinapse-ai/product/templates/tmpl-rls-granular-policies.sql +105 -105
- package/.sinapse-ai/product/templates/tmpl-rls-kiss-policy.sql +11 -11
- package/.sinapse-ai/product/templates/tmpl-rls-roles.sql +136 -136
- package/.sinapse-ai/product/templates/tmpl-rls-simple.sql +78 -78
- package/.sinapse-ai/product/templates/tmpl-rls-tenant.sql +153 -153
- package/.sinapse-ai/product/templates/tmpl-rollback-script.sql +78 -78
- package/.sinapse-ai/product/templates/tmpl-seed-data.sql +141 -141
- package/.sinapse-ai/product/templates/tmpl-smoke-test.sql +17 -17
- package/.sinapse-ai/product/templates/tmpl-staging-copy-merge.sql +140 -140
- package/.sinapse-ai/product/templates/tmpl-stored-proc.sql +141 -141
- package/.sinapse-ai/product/templates/tmpl-trigger.sql +153 -153
- package/.sinapse-ai/product/templates/tmpl-view-materialized.sql +134 -134
- package/.sinapse-ai/product/templates/tmpl-view.sql +178 -178
- package/AGENTS.md +193 -0
- package/CHANGELOG.md +1247 -0
- package/LICENSE +63 -63
- package/README.en.md +17 -18
- package/README.md +18 -19
- package/bin/cli.js +1 -1
- package/bin/commands/install.js +194 -22
- package/bin/commands/status.js +14 -1
- package/bin/commands/uninstall.js +2 -2
- package/bin/commands/update.js +52 -0
- package/bin/lib/setup-statusline.js +191 -0
- package/bin/sinapse-init.js +11 -83
- package/bin/utils/framework-guard.js +17 -4
- package/bin/utils/secret-scanner-core.js +109 -7
- package/bin/utils/staged-sql-guard.js +204 -0
- package/bin/utils/validate-publish.js +63 -0
- package/docs/agent-reference-guide.md +5 -7
- package/docs/framework/agent-prefix-convention.md +58 -0
- package/docs/framework/architecture-overview.md +4 -4
- package/docs/framework/collaboration-activation.md +45 -0
- package/docs/framework/guiding-principles.md +9 -9
- package/docs/getting-started.md +1 -1
- package/docs/guides/agent-reference.md +1 -1
- package/docs/guides/codex-config.md +4 -5
- package/docs/pt/architecture/sub-orqx-pattern.md +20 -18
- package/docs/security/overview.md +1 -1
- package/package.json +16 -12
- package/packages/installer/src/index.js +26 -0
- package/packages/installer/src/installer/git-hooks-installer.js +211 -47
- package/packages/installer/src/installer/sinapse-ai-installer.js +71 -0
- package/packages/installer/src/wizard/feedback.js +1 -1
- package/packages/installer/src/wizard/ide-config-generator.js +26 -26
- package/packages/installer/src/wizard/index.js +53 -4
- package/packages/sinapse-install/bin/edmcp.js +0 -0
- package/packages/sinapse-install/bin/sinapse-install.js +0 -0
- package/scripts/audit-tasks.cjs +112 -91
- package/scripts/check-markdown-links.py +352 -352
- package/scripts/prepare-hooks.js +58 -0
- package/scripts/regenerate-orqx-stubs.ps1 +2 -3
- package/scripts/sync-counts.js +10 -2
- package/scripts/sync-squad-yaml-components.js +108 -6
- package/scripts/validate-agents-md.js +128 -0
- package/scripts/validate-all.js +1 -0
- package/scripts/validate-squad-orqx.js +19 -9
- package/sinapse/agents/sinapse-orqx.md +16 -26
- package/sinapse/agents/snps-orqx.md +15 -25
- package/sinapse/knowledge-base/routing-catalog.md +1 -1
- package/sinapse/tasks/diagnose-and-route.md +1 -1
- package/sinapse/tasks/squad-status-report.md +1 -1
- package/squads/claude-code-mastery/agents/claude-mastery-chief.md +1 -1
- package/squads/claude-code-mastery/agents/hooks-architect.md +60 -68
- package/squads/claude-code-mastery/knowledge-base/swarm-orchestration-patterns.md +1 -1
- package/squads/claude-code-mastery/squad.yaml +8 -0
- package/squads/claude-code-mastery/tasks/audit-setup.md +1 -1
- package/squads/claude-code-mastery/workflows/optimization-cycle.yaml +4 -4
- package/squads/claude-code-mastery/workflows/project-setup-cycle.yaml +4 -4
- package/squads/squad-animations/README.md +1 -1
- package/squads/squad-animations/squad.yaml +1 -1
- package/squads/squad-brand/squad.yaml +1 -1
- package/squads/squad-cloning/README.md +1 -1
- package/squads/squad-cloning/squad.yaml +1 -1
- package/squads/squad-commercial/README.md +1 -1
- package/squads/squad-commercial/squad.yaml +2 -3
- package/squads/squad-content/README.md +1 -1
- package/squads/squad-content/squad.yaml +1 -1
- package/squads/squad-copy/README.md +1 -1
- package/squads/squad-copy/squad.yaml +2 -3
- package/squads/squad-council/README.md +1 -1
- package/squads/squad-courses/README.md +1 -1
- package/squads/squad-courses/squad.yaml +1 -1
- package/squads/squad-cybersecurity/README.md +1 -1
- package/squads/squad-cybersecurity/squad.yaml +2 -3
- package/squads/squad-design/README.md +1 -1
- package/squads/{squad-artdir → squad-design}/agents/cro-persuasion.md +1 -1
- package/squads/{squad-artdir → squad-design}/agents/platform-aesthetic-director.md +2 -2
- package/squads/{squad-artdir → squad-design}/agents/premium-packaging-strategist.md +2 -2
- package/squads/{squad-artdir → squad-design}/agents/product-surface-director.md +3 -3
- package/squads/squad-design/squad.yaml +6 -3
- package/squads/squad-finance/README.md +1 -1
- package/squads/squad-finance/squad.yaml +7 -1
- package/squads/squad-growth/README.md +1 -1
- package/squads/squad-growth/squad.yaml +1 -1
- package/squads/squad-paidmedia/README.md +1 -1
- package/squads/squad-paidmedia/squad.yaml +2 -3
- package/squads/squad-product/README.md +1 -1
- package/squads/squad-product/squad.yaml +1 -1
- package/squads/squad-research/README.md +1 -1
- package/squads/squad-research/squad.yaml +2 -3
- package/squads/squad-storytelling/README.md +1 -1
- package/squads/squad-storytelling/squad.yaml +2 -3
- package/.codex/agents/brad-frost.md +0 -46
- package/.codex/agents/claude-orqx.md +0 -72
- package/.codex/agents/copy-chief.md +0 -162
- package/.codex/agents/cyber-chief.md +0 -169
- package/.codex/agents/dan-mall.md +0 -43
- package/.codex/agents/data-chief.md +0 -198
- package/.codex/agents/dave-malouf.md +0 -43
- package/.codex/agents/db-sage.md +0 -152
- package/.codex/agents/design-chief.md +0 -226
- package/.codex/agents/dev.md +0 -102
- package/.codex/agents/legal-chief.md +0 -199
- package/.codex/agents/nano-banana-generator.md +0 -42
- package/.codex/agents/pm.md +0 -81
- package/.codex/agents/po.md +0 -85
- package/.codex/agents/qa.md +0 -98
- package/.codex/agents/sm.md +0 -77
- package/.codex/agents/squad-chief.md +0 -1553
- package/.codex/agents/squad.md +0 -66
- package/.codex/agents/story-chief.md +0 -180
- package/.codex/agents/tools-orqx.md +0 -219
- package/.codex/agents/traffic-masters-chief.md +0 -211
- package/.sinapse-ai/core/memory/__tests__/active-modules.verify.js +0 -265
- package/.sinapse-ai/core/permissions/__tests__/permission-mode.test.js +0 -293
- package/.sinapse-ai/data/registry-update-log.jsonl +0 -158
- package/.sinapse-ai/infrastructure/scripts/ide-sync/gemini-commands.js +0 -298
- package/.sinapse-ai/infrastructure/scripts/ide-sync/transformers/antigravity.js +0 -121
- package/.sinapse-ai/infrastructure/scripts/ide-sync/transformers/cursor.js +0 -119
- package/.sinapse-ai/infrastructure/scripts/ide-sync/transformers/github-copilot.js +0 -191
- package/.sinapse-ai/infrastructure/scripts/ide-sync/transformers/kimi.js +0 -448
- package/.sinapse-ai/infrastructure/tests/project-status-loader.test.js +0 -569
- package/.sinapse-ai/infrastructure/tests/regression-suite-v2.md +0 -622
- package/.sinapse-ai/infrastructure/tests/validate-module.js +0 -98
- package/.sinapse-ai/infrastructure/tests/worktree-manager.test.js +0 -620
- package/.sinapse-ai/monitor/hooks/lib/__init__.py +0 -2
- package/.sinapse-ai/monitor/hooks/lib/enrich.py +0 -59
- package/.sinapse-ai/monitor/hooks/lib/send_event.py +0 -48
- package/.sinapse-ai/monitor/hooks/notification.py +0 -30
- package/.sinapse-ai/monitor/hooks/post_tool_use.py +0 -46
- package/.sinapse-ai/monitor/hooks/pre_compact.py +0 -30
- package/.sinapse-ai/monitor/hooks/pre_tool_use.py +0 -41
- package/.sinapse-ai/monitor/hooks/stop.py +0 -30
- package/.sinapse-ai/monitor/hooks/subagent_stop.py +0 -30
- package/.sinapse-ai/monitor/hooks/user_prompt_submit.py +0 -39
- package/.sinapse-ai/product/templates/statusline/track-agent.sh +0 -69
- package/.sinapse-ai/workflow-intelligence/__tests__/confidence-scorer.test.js +0 -335
- package/.sinapse-ai/workflow-intelligence/__tests__/integration.test.js +0 -340
- package/.sinapse-ai/workflow-intelligence/__tests__/suggestion-engine.test.js +0 -438
- package/.sinapse-ai/workflow-intelligence/__tests__/wave-analyzer.test.js +0 -448
- package/.sinapse-ai/workflow-intelligence/__tests__/workflow-registry.test.js +0 -303
- package/bin/sinapse-graph.js +0 -19
- package/docs/codex-integration-process.md +0 -22
- package/docs/codex-parity-program.md +0 -27
- package/packages/installer/src/__tests__/performance-benchmark.js +0 -383
- package/packages/installer/tests/integration/environment-configuration.test.js +0 -332
- package/packages/installer/tests/integration/wizard-detection.test.js +0 -352
- package/packages/installer/tests/unit/artifact-copy-pipeline/artifact-copy-pipeline.test.js +0 -383
- package/packages/installer/tests/unit/claude-md-template-v5/claude-md-template-v5.test.js +0 -193
- package/packages/installer/tests/unit/config-validator.test.js +0 -315
- package/packages/installer/tests/unit/detection/detect-project-type.test.js +0 -539
- package/packages/installer/tests/unit/doctor/doctor-checks.test.js +0 -636
- package/packages/installer/tests/unit/doctor/doctor-orchestrator.test.js +0 -192
- package/packages/installer/tests/unit/entity-registry-bootstrap.test.js +0 -186
- package/packages/installer/tests/unit/env-template.test.js +0 -187
- package/packages/installer/tests/unit/generate-settings-json/generate-settings-json.test.js +0 -310
- package/packages/installer/tests/unit/git-hooks-installer.test.js +0 -262
- package/packages/installer/tests/unit/ide-sync-integration/ide-sync-integration.test.js +0 -231
- package/packages/installer/tests/unit/merger/env-merger.test.js +0 -191
- package/packages/installer/tests/unit/merger/markdown-merger.test.js +0 -262
- package/packages/installer/tests/unit/merger/strategies.test.js +0 -154
- package/packages/installer/tests/unit/merger/yaml-merger.test.js +0 -328
- package/packages/sinapse-install/tests/unit/chrome-brain.smoke.test.js +0 -66
- package/scripts/install-monitor-hooks.sh +0 -82
- package/squads/squad-artdir/README.md +0 -90
- package/squads/squad-artdir/agents/accessibility-guardian.md +0 -184
- package/squads/squad-artdir/agents/artdir-orqx.md +0 -222
- package/squads/squad-artdir/agents/color-psychologist.md +0 -166
- package/squads/squad-artdir/agents/design-system-architect.md +0 -100
- package/squads/squad-artdir/agents/ia-architect.md +0 -169
- package/squads/squad-artdir/agents/interaction-designer.md +0 -162
- package/squads/squad-artdir/agents/layout-engineer.md +0 -163
- package/squads/squad-artdir/agents/motion-architect.md +0 -185
- package/squads/squad-artdir/agents/type-systemist.md +0 -138
- package/squads/squad-artdir/agents/visual-strategist.md +0 -127
- package/squads/squad-artdir/checklists/seven-pillars-validation-checklist.md +0 -172
- package/squads/squad-artdir/knowledge-base/case-nyo-ia-reference.md +0 -289
- package/squads/squad-artdir/knowledge-base/deliverables-templates.md +0 -457
- package/squads/squad-artdir/knowledge-base/motion-technique-catalog.md +0 -247
- package/squads/squad-artdir/knowledge-base/premium-packaging-principles.md +0 -133
- package/squads/squad-artdir/knowledge-base/psychological-toolkit.md +0 -229
- package/squads/squad-artdir/knowledge-base/saas-art-direction-canon.md +0 -242
- package/squads/squad-artdir/knowledge-base/seven-pillars-framework.md +0 -289
- package/squads/squad-artdir/knowledge-base/ten-pillars-framework.md +0 -221
- package/squads/squad-artdir/package.json +0 -20
- package/squads/squad-artdir/squad.yaml +0 -299
- package/squads/squad-artdir/tasks/audit-conversion.md +0 -97
- package/squads/squad-artdir/tasks/audit-drift-multi-surface.md +0 -55
- package/squads/squad-artdir/tasks/consult-saas-canon.md +0 -54
- package/squads/squad-artdir/tasks/create-art-direction-brief.md +0 -110
- package/squads/squad-artdir/tasks/create-premium-packaging-brief.md +0 -61
- package/squads/squad-artdir/tasks/create-wireflow.md +0 -84
- package/squads/squad-artdir/tasks/design-color-system.md +0 -81
- package/squads/squad-artdir/tasks/design-product-surface.md +0 -60
- package/squads/squad-artdir/tasks/design-token-system.md +0 -58
- package/squads/squad-artdir/tasks/diagnose-visual-language.md +0 -92
- package/squads/squad-artdir/tasks/first-5-minutes-choreography.md +0 -65
- package/squads/squad-artdir/tasks/specify-motion-system.md +0 -84
- package/squads/squad-artdir/tasks/validate-against-pillars.md +0 -143
- package/squads/squad-artdir/templates/art-direction-brief-template.md +0 -215
- package/squads/squad-artdir/workflows/conversion-audit-cycle.yaml +0 -142
- package/squads/squad-artdir/workflows/full-art-direction-cycle.yaml +0 -179
- package/squads/squad-artdir/workflows/saas-platform-art-direction-cycle.yaml +0 -338
- package/squads/squad-commercial/agents/legal-chief.md +0 -199
- package/squads/squad-copy/agents/copy-chief.md +0 -162
- package/squads/squad-cybersecurity/agents/cyber-chief.md +0 -169
- package/squads/squad-design/agents/design-chief.md +0 -226
- package/squads/squad-paidmedia/agents/traffic-masters-chief.md +0 -211
- package/squads/squad-research/agents/data-chief.md +0 -198
- package/squads/squad-storytelling/agents/story-chief.md +0 -180
|
@@ -1,383 +0,0 @@
|
|
|
1
|
-
#!/usr/bin/env node
|
|
2
|
-
|
|
3
|
-
/**
|
|
4
|
-
* SINAPSE Installer Performance Benchmark
|
|
5
|
-
* Story INS-2: Installer Performance Optimization
|
|
6
|
-
*
|
|
7
|
-
* Measures baseline performance metrics for the installer to track optimization progress.
|
|
8
|
-
*
|
|
9
|
-
* Usage:
|
|
10
|
-
* node performance-benchmark.js [--output <file>] [--runs <n>]
|
|
11
|
-
*
|
|
12
|
-
* Output:
|
|
13
|
-
* JSON report with phase timings and statistics
|
|
14
|
-
*/
|
|
15
|
-
|
|
16
|
-
const fs = require('fs');
|
|
17
|
-
const fse = require('fs-extra');
|
|
18
|
-
const path = require('path');
|
|
19
|
-
const crypto = require('crypto');
|
|
20
|
-
const { performance } = require('perf_hooks');
|
|
21
|
-
|
|
22
|
-
// Configuration
|
|
23
|
-
const CONFIG = {
|
|
24
|
-
runs: 3, // Number of runs for averaging
|
|
25
|
-
outputFile: null, // Output file path (null = stdout)
|
|
26
|
-
testProjectSize: 1000, // Number of files for test project
|
|
27
|
-
verbose: false,
|
|
28
|
-
};
|
|
29
|
-
|
|
30
|
-
// Parse CLI arguments
|
|
31
|
-
process.argv.slice(2).forEach((arg, i, arr) => {
|
|
32
|
-
if (arg === '--output' && arr[i + 1]) CONFIG.outputFile = arr[i + 1];
|
|
33
|
-
if (arg === '--runs' && arr[i + 1]) CONFIG.runs = parseInt(arr[i + 1], 10);
|
|
34
|
-
if (arg === '--verbose' || arg === '-v') CONFIG.verbose = true;
|
|
35
|
-
if (arg === '--help' || arg === '-h') {
|
|
36
|
-
console.log(`
|
|
37
|
-
SINAPSE Installer Performance Benchmark
|
|
38
|
-
|
|
39
|
-
Usage: node performance-benchmark.js [options]
|
|
40
|
-
|
|
41
|
-
Options:
|
|
42
|
-
--output <file> Save JSON report to file (default: stdout)
|
|
43
|
-
--runs <n> Number of benchmark runs (default: 3)
|
|
44
|
-
--verbose, -v Show detailed progress
|
|
45
|
-
--help, -h Show this help
|
|
46
|
-
|
|
47
|
-
Example:
|
|
48
|
-
node performance-benchmark.js --output baseline.json --runs 5
|
|
49
|
-
`);
|
|
50
|
-
process.exit(0);
|
|
51
|
-
}
|
|
52
|
-
});
|
|
53
|
-
|
|
54
|
-
// Benchmark results structure
|
|
55
|
-
const results = {
|
|
56
|
-
timestamp: new Date().toISOString(),
|
|
57
|
-
system: {
|
|
58
|
-
platform: process.platform,
|
|
59
|
-
arch: process.arch,
|
|
60
|
-
nodeVersion: process.version,
|
|
61
|
-
cpus: require('os').cpus().length,
|
|
62
|
-
totalMemory: Math.round(require('os').totalmem() / 1024 / 1024) + ' MB',
|
|
63
|
-
},
|
|
64
|
-
config: { ...CONFIG },
|
|
65
|
-
phases: {},
|
|
66
|
-
summary: {},
|
|
67
|
-
};
|
|
68
|
-
|
|
69
|
-
/**
|
|
70
|
-
* Timer utility for measuring phase durations
|
|
71
|
-
*/
|
|
72
|
-
class Timer {
|
|
73
|
-
constructor(name) {
|
|
74
|
-
this.name = name;
|
|
75
|
-
this.start = null;
|
|
76
|
-
this.end = null;
|
|
77
|
-
this.runs = [];
|
|
78
|
-
}
|
|
79
|
-
|
|
80
|
-
begin() {
|
|
81
|
-
this.start = performance.now();
|
|
82
|
-
}
|
|
83
|
-
|
|
84
|
-
stop() {
|
|
85
|
-
this.end = performance.now();
|
|
86
|
-
const duration = this.end - this.start;
|
|
87
|
-
this.runs.push(duration);
|
|
88
|
-
return duration;
|
|
89
|
-
}
|
|
90
|
-
|
|
91
|
-
getStats() {
|
|
92
|
-
if (this.runs.length === 0) return null;
|
|
93
|
-
const sorted = [...this.runs].sort((a, b) => a - b);
|
|
94
|
-
return {
|
|
95
|
-
min: Math.round(sorted[0]),
|
|
96
|
-
max: Math.round(sorted[sorted.length - 1]),
|
|
97
|
-
avg: Math.round(this.runs.reduce((a, b) => a + b, 0) / this.runs.length),
|
|
98
|
-
median: Math.round(sorted[Math.floor(sorted.length / 2)]),
|
|
99
|
-
runs: this.runs.map((r) => Math.round(r)),
|
|
100
|
-
unit: 'ms',
|
|
101
|
-
};
|
|
102
|
-
}
|
|
103
|
-
}
|
|
104
|
-
|
|
105
|
-
// Phase timers
|
|
106
|
-
const timers = {
|
|
107
|
-
directoryRead: new Timer('Directory Read (readdirSync)'),
|
|
108
|
-
directoryReadWithTypes: new Timer('Directory Read (withFileTypes)'),
|
|
109
|
-
statLoop: new Timer('Stat Loop (statSync per file)'),
|
|
110
|
-
realpathSingle: new Timer('Realpath (single call)'),
|
|
111
|
-
realpathDouble: new Timer('Realpath (double call - current)'),
|
|
112
|
-
hashSequential: new Timer('Hash Files (sequential)'),
|
|
113
|
-
hashParallelBatch: new Timer('Hash Files (parallel batch)'),
|
|
114
|
-
fileCopySequential: new Timer('File Copy (sequential)'),
|
|
115
|
-
fileCopyParallel: new Timer('File Copy (parallel)'),
|
|
116
|
-
totalInstallSimulation: new Timer('Total Install Simulation'),
|
|
117
|
-
};
|
|
118
|
-
|
|
119
|
-
/**
|
|
120
|
-
* Log if verbose mode is enabled
|
|
121
|
-
*/
|
|
122
|
-
function log(msg) {
|
|
123
|
-
if (CONFIG.verbose) console.log(`[benchmark] ${msg}`);
|
|
124
|
-
}
|
|
125
|
-
|
|
126
|
-
/**
|
|
127
|
-
* Get the .sinapse-ai directory for benchmarking
|
|
128
|
-
*/
|
|
129
|
-
function getSinapseCoreDir() {
|
|
130
|
-
const projectRoot = path.resolve(__dirname, '../../../../');
|
|
131
|
-
return path.join(projectRoot, '.sinapse-ai');
|
|
132
|
-
}
|
|
133
|
-
|
|
134
|
-
/**
|
|
135
|
-
* Benchmark: Directory read comparison
|
|
136
|
-
*/
|
|
137
|
-
async function benchmarkDirectoryRead(dir) {
|
|
138
|
-
const files = fs.readdirSync(dir);
|
|
139
|
-
|
|
140
|
-
// Method 1: readdirSync + statSync for each
|
|
141
|
-
timers.statLoop.begin();
|
|
142
|
-
for (const file of files) {
|
|
143
|
-
const fullPath = path.join(dir, file);
|
|
144
|
-
fs.statSync(fullPath).isDirectory();
|
|
145
|
-
}
|
|
146
|
-
timers.statLoop.stop();
|
|
147
|
-
|
|
148
|
-
// Method 2: readdirSync with withFileTypes
|
|
149
|
-
timers.directoryReadWithTypes.begin();
|
|
150
|
-
const entries = fs.readdirSync(dir, { withFileTypes: true });
|
|
151
|
-
for (const entry of entries) {
|
|
152
|
-
entry.isDirectory();
|
|
153
|
-
}
|
|
154
|
-
timers.directoryReadWithTypes.stop();
|
|
155
|
-
|
|
156
|
-
return files.length;
|
|
157
|
-
}
|
|
158
|
-
|
|
159
|
-
/**
|
|
160
|
-
* Benchmark: Realpath comparison
|
|
161
|
-
*/
|
|
162
|
-
async function benchmarkRealpath(files) {
|
|
163
|
-
const sampleFiles = files.slice(0, 100); // Sample 100 files
|
|
164
|
-
|
|
165
|
-
// Method 1: Single realpath call
|
|
166
|
-
timers.realpathSingle.begin();
|
|
167
|
-
for (const file of sampleFiles) {
|
|
168
|
-
fs.realpathSync(file);
|
|
169
|
-
}
|
|
170
|
-
timers.realpathSingle.stop();
|
|
171
|
-
|
|
172
|
-
// Method 2: Double realpath call (current behavior)
|
|
173
|
-
timers.realpathDouble.begin();
|
|
174
|
-
for (const file of sampleFiles) {
|
|
175
|
-
fs.realpathSync(file);
|
|
176
|
-
fs.realpathSync(path.dirname(file)); // Simulates the duplicate call
|
|
177
|
-
}
|
|
178
|
-
timers.realpathDouble.stop();
|
|
179
|
-
}
|
|
180
|
-
|
|
181
|
-
/**
|
|
182
|
-
* Benchmark: File hashing comparison
|
|
183
|
-
*/
|
|
184
|
-
async function benchmarkHashing(files) {
|
|
185
|
-
const sampleFiles = files.slice(0, 200); // Sample 200 files for hashing
|
|
186
|
-
|
|
187
|
-
// Method 1: Sequential hashing
|
|
188
|
-
timers.hashSequential.begin();
|
|
189
|
-
for (const file of sampleFiles) {
|
|
190
|
-
try {
|
|
191
|
-
const content = fs.readFileSync(file);
|
|
192
|
-
crypto.createHash('sha256').update(content).digest('hex');
|
|
193
|
-
} catch {
|
|
194
|
-
// Skip files that can't be read
|
|
195
|
-
}
|
|
196
|
-
}
|
|
197
|
-
timers.hashSequential.stop();
|
|
198
|
-
|
|
199
|
-
// Method 2: Parallel batch hashing
|
|
200
|
-
timers.hashParallelBatch.begin();
|
|
201
|
-
const batchSize = 50;
|
|
202
|
-
for (let i = 0; i < sampleFiles.length; i += batchSize) {
|
|
203
|
-
const batch = sampleFiles.slice(i, i + batchSize);
|
|
204
|
-
await Promise.all(
|
|
205
|
-
batch.map(async (file) => {
|
|
206
|
-
try {
|
|
207
|
-
const content = await fse.readFile(file);
|
|
208
|
-
crypto.createHash('sha256').update(content).digest('hex');
|
|
209
|
-
} catch {
|
|
210
|
-
// Skip files that can't be read
|
|
211
|
-
}
|
|
212
|
-
}),
|
|
213
|
-
);
|
|
214
|
-
}
|
|
215
|
-
timers.hashParallelBatch.stop();
|
|
216
|
-
}
|
|
217
|
-
|
|
218
|
-
/**
|
|
219
|
-
* Collect all files recursively
|
|
220
|
-
*/
|
|
221
|
-
function collectFiles(dir, maxFiles = 1000) {
|
|
222
|
-
const files = [];
|
|
223
|
-
|
|
224
|
-
function walk(currentDir) {
|
|
225
|
-
if (files.length >= maxFiles) return;
|
|
226
|
-
|
|
227
|
-
try {
|
|
228
|
-
const entries = fs.readdirSync(currentDir, { withFileTypes: true });
|
|
229
|
-
for (const entry of entries) {
|
|
230
|
-
if (files.length >= maxFiles) break;
|
|
231
|
-
|
|
232
|
-
const fullPath = path.join(currentDir, entry.name);
|
|
233
|
-
if (entry.isDirectory()) {
|
|
234
|
-
walk(fullPath);
|
|
235
|
-
} else if (entry.isFile()) {
|
|
236
|
-
files.push(fullPath);
|
|
237
|
-
}
|
|
238
|
-
}
|
|
239
|
-
} catch {
|
|
240
|
-
// Skip directories we can't read
|
|
241
|
-
}
|
|
242
|
-
}
|
|
243
|
-
|
|
244
|
-
walk(dir);
|
|
245
|
-
return files;
|
|
246
|
-
}
|
|
247
|
-
|
|
248
|
-
/**
|
|
249
|
-
* Run all benchmarks
|
|
250
|
-
*/
|
|
251
|
-
async function runBenchmarks() {
|
|
252
|
-
const sinapseCoreDir = getSinapseCoreDir();
|
|
253
|
-
|
|
254
|
-
if (!fs.existsSync(sinapseCoreDir)) {
|
|
255
|
-
console.error(`Error: .sinapse-ai directory not found at ${sinapseCoreDir}`);
|
|
256
|
-
process.exit(1);
|
|
257
|
-
}
|
|
258
|
-
|
|
259
|
-
log(`Starting benchmark with ${CONFIG.runs} runs`);
|
|
260
|
-
log(`Using .sinapse-ai at: ${sinapseCoreDir}`);
|
|
261
|
-
|
|
262
|
-
// Collect files for benchmarking
|
|
263
|
-
log('Collecting files...');
|
|
264
|
-
const allFiles = collectFiles(sinapseCoreDir, CONFIG.testProjectSize);
|
|
265
|
-
log(`Collected ${allFiles.length} files`);
|
|
266
|
-
|
|
267
|
-
results.fileCount = allFiles.length;
|
|
268
|
-
|
|
269
|
-
// Run benchmarks multiple times
|
|
270
|
-
for (let run = 1; run <= CONFIG.runs; run++) {
|
|
271
|
-
log(`\n--- Run ${run}/${CONFIG.runs} ---`);
|
|
272
|
-
|
|
273
|
-
// Directory read benchmarks
|
|
274
|
-
const agentsDir = path.join(sinapseCoreDir, 'development', 'agents');
|
|
275
|
-
if (fs.existsSync(agentsDir)) {
|
|
276
|
-
log('Benchmarking directory read...');
|
|
277
|
-
await benchmarkDirectoryRead(agentsDir);
|
|
278
|
-
}
|
|
279
|
-
|
|
280
|
-
// Realpath benchmarks
|
|
281
|
-
log('Benchmarking realpath...');
|
|
282
|
-
await benchmarkRealpath(allFiles);
|
|
283
|
-
|
|
284
|
-
// Hashing benchmarks
|
|
285
|
-
log('Benchmarking file hashing...');
|
|
286
|
-
await benchmarkHashing(allFiles);
|
|
287
|
-
|
|
288
|
-
// Total simulation
|
|
289
|
-
log('Running total install simulation...');
|
|
290
|
-
timers.totalInstallSimulation.begin();
|
|
291
|
-
|
|
292
|
-
// Simulate full install: read dirs + hash files
|
|
293
|
-
const devDir = path.join(sinapseCoreDir, 'development');
|
|
294
|
-
if (fs.existsSync(devDir)) {
|
|
295
|
-
const subdirs = fs.readdirSync(devDir, { withFileTypes: true });
|
|
296
|
-
for (const subdir of subdirs) {
|
|
297
|
-
if (subdir.isDirectory()) {
|
|
298
|
-
const fullSubdir = path.join(devDir, subdir.name);
|
|
299
|
-
fs.readdirSync(fullSubdir);
|
|
300
|
-
}
|
|
301
|
-
}
|
|
302
|
-
}
|
|
303
|
-
|
|
304
|
-
// Simulate sequential file processing
|
|
305
|
-
for (const file of allFiles.slice(0, 500)) {
|
|
306
|
-
try {
|
|
307
|
-
fs.statSync(file);
|
|
308
|
-
fs.readFileSync(file);
|
|
309
|
-
} catch {
|
|
310
|
-
// Skip
|
|
311
|
-
}
|
|
312
|
-
}
|
|
313
|
-
|
|
314
|
-
timers.totalInstallSimulation.stop();
|
|
315
|
-
}
|
|
316
|
-
|
|
317
|
-
// Compile results
|
|
318
|
-
log('\nCompiling results...');
|
|
319
|
-
|
|
320
|
-
for (const [name, timer] of Object.entries(timers)) {
|
|
321
|
-
const stats = timer.getStats();
|
|
322
|
-
if (stats) {
|
|
323
|
-
results.phases[name] = {
|
|
324
|
-
description: timer.name,
|
|
325
|
-
...stats,
|
|
326
|
-
};
|
|
327
|
-
}
|
|
328
|
-
}
|
|
329
|
-
|
|
330
|
-
// Calculate summary
|
|
331
|
-
const hashSeq = results.phases.hashSequential?.avg || 0;
|
|
332
|
-
const hashPar = results.phases.hashParallelBatch?.avg || 0;
|
|
333
|
-
const realpathSingle = results.phases.realpathSingle?.avg || 0;
|
|
334
|
-
const realpathDouble = results.phases.realpathDouble?.avg || 0;
|
|
335
|
-
const statLoop = results.phases.statLoop?.avg || 0;
|
|
336
|
-
const withTypes = results.phases.directoryReadWithTypes?.avg || 0;
|
|
337
|
-
|
|
338
|
-
results.summary = {
|
|
339
|
-
totalFiles: results.fileCount,
|
|
340
|
-
hashingSpeedup: hashSeq > 0 ? `${(hashSeq / hashPar).toFixed(2)}x` : 'N/A',
|
|
341
|
-
realpathSavings: realpathDouble > 0 ? `${Math.round(((realpathDouble - realpathSingle) / realpathDouble) * 100)}%` : 'N/A',
|
|
342
|
-
statLoopSavings: statLoop > 0 ? `${Math.round(((statLoop - withTypes) / statLoop) * 100)}%` : 'N/A',
|
|
343
|
-
estimatedTotalTime: results.phases.totalInstallSimulation?.avg || 0,
|
|
344
|
-
target: '<30000ms for 1000 files',
|
|
345
|
-
baseline: `${results.phases.totalInstallSimulation?.avg || 'TBD'}ms`,
|
|
346
|
-
};
|
|
347
|
-
|
|
348
|
-
// Output results
|
|
349
|
-
const output = JSON.stringify(results, null, 2);
|
|
350
|
-
|
|
351
|
-
if (CONFIG.outputFile) {
|
|
352
|
-
fs.writeFileSync(CONFIG.outputFile, output);
|
|
353
|
-
console.log(`Benchmark results saved to: ${CONFIG.outputFile}`);
|
|
354
|
-
} else {
|
|
355
|
-
console.log(output);
|
|
356
|
-
}
|
|
357
|
-
|
|
358
|
-
// Print summary to stderr for visibility
|
|
359
|
-
console.error('\n' + '='.repeat(60));
|
|
360
|
-
console.error('SINAPSE Installer Performance Baseline');
|
|
361
|
-
console.error('='.repeat(60));
|
|
362
|
-
console.error(`Files analyzed: ${results.fileCount}`);
|
|
363
|
-
console.error(`Runs: ${CONFIG.runs}`);
|
|
364
|
-
console.error('');
|
|
365
|
-
console.error('Phase Results (avg ms):');
|
|
366
|
-
console.error(` Directory stat loop: ${statLoop}ms`);
|
|
367
|
-
console.error(` Directory withFileTypes: ${withTypes}ms (${results.summary.statLoopSavings} faster)`);
|
|
368
|
-
console.error(` Realpath single: ${realpathSingle}ms`);
|
|
369
|
-
console.error(` Realpath double: ${realpathDouble}ms (${results.summary.realpathSavings} overhead)`);
|
|
370
|
-
console.error(` Hash sequential: ${hashSeq}ms`);
|
|
371
|
-
console.error(` Hash parallel: ${hashPar}ms (${results.summary.hashingSpeedup} faster)`);
|
|
372
|
-
console.error('');
|
|
373
|
-
console.error(`Total Install Simulation: ${results.summary.baseline}`);
|
|
374
|
-
console.error(`Target: ${results.summary.target}`);
|
|
375
|
-
console.error('='.repeat(60));
|
|
376
|
-
}
|
|
377
|
-
|
|
378
|
-
// Run benchmarks
|
|
379
|
-
runBenchmarks().catch((err) => {
|
|
380
|
-
console.error('Benchmark failed:', err);
|
|
381
|
-
process.exit(1);
|
|
382
|
-
});
|
|
383
|
-
|
|
@@ -1,332 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Integration Tests: Environment Configuration
|
|
3
|
-
* Story 1.6: Environment Configuration
|
|
4
|
-
*
|
|
5
|
-
* Tests for configure-environment.js with file system operations
|
|
6
|
-
*/
|
|
7
|
-
|
|
8
|
-
const fs = require('fs-extra');
|
|
9
|
-
const path = require('path');
|
|
10
|
-
const os = require('os');
|
|
11
|
-
const { configureEnvironment, updateGitignore } = require('../../src/config/configure-environment');
|
|
12
|
-
|
|
13
|
-
/**
|
|
14
|
-
* Cleanup helper with retry logic for flaky file system operations
|
|
15
|
-
* Handles ENOTEMPTY and EBUSY errors common in CI environments
|
|
16
|
-
* @param {string} dir - Directory to remove
|
|
17
|
-
* @param {number} maxRetries - Maximum retry attempts
|
|
18
|
-
* @param {number} retryDelay - Delay between retries in ms
|
|
19
|
-
*/
|
|
20
|
-
async function cleanupWithRetry(dir, maxRetries = 5, retryDelay = 100) {
|
|
21
|
-
for (let attempt = 1; attempt <= maxRetries; attempt++) {
|
|
22
|
-
try {
|
|
23
|
-
if (await fs.pathExists(dir)) {
|
|
24
|
-
await fs.remove(dir);
|
|
25
|
-
}
|
|
26
|
-
return;
|
|
27
|
-
} catch (error) {
|
|
28
|
-
const isRetryable = error.code && ['ENOTEMPTY', 'EBUSY', 'EPERM', 'EACCES'].includes(error.code);
|
|
29
|
-
if (attempt === maxRetries || !isRetryable) {
|
|
30
|
-
// Last attempt failed or non-retryable error, log but don't throw
|
|
31
|
-
console.warn(`Warning: Failed to cleanup ${dir} after ${attempt} attempts:`, error.code);
|
|
32
|
-
return;
|
|
33
|
-
}
|
|
34
|
-
// Linear backoff (100ms, 200ms, 300ms...)
|
|
35
|
-
await new Promise(resolve => setTimeout(resolve, retryDelay * attempt));
|
|
36
|
-
}
|
|
37
|
-
}
|
|
38
|
-
}
|
|
39
|
-
|
|
40
|
-
describe('Environment Configuration Integration', () => {
|
|
41
|
-
let testDir;
|
|
42
|
-
let testId;
|
|
43
|
-
|
|
44
|
-
beforeEach(async () => {
|
|
45
|
-
// Create unique temporary test directory with random suffix to avoid collisions
|
|
46
|
-
testId = `${Date.now()}-${Math.random().toString(36).substring(2, 8)}`;
|
|
47
|
-
testDir = path.join(os.tmpdir(), `sinapse-env-test-${testId}`);
|
|
48
|
-
await fs.ensureDir(testDir);
|
|
49
|
-
});
|
|
50
|
-
|
|
51
|
-
afterEach(async () => {
|
|
52
|
-
// Small delay to allow file handles to close
|
|
53
|
-
await new Promise(resolve => setTimeout(resolve, 50));
|
|
54
|
-
// Cleanup test directory with retry logic
|
|
55
|
-
await cleanupWithRetry(testDir);
|
|
56
|
-
});
|
|
57
|
-
|
|
58
|
-
describe('configureEnvironment', () => {
|
|
59
|
-
it('should create .env file with skip prompts', async () => {
|
|
60
|
-
const result = await configureEnvironment({
|
|
61
|
-
targetDir: testDir,
|
|
62
|
-
skipPrompts: true,
|
|
63
|
-
projectType: 'GREENFIELD',
|
|
64
|
-
selectedIDEs: ['vscode'],
|
|
65
|
-
mcpServers: [],
|
|
66
|
-
});
|
|
67
|
-
|
|
68
|
-
expect(result.envCreated).toBe(true);
|
|
69
|
-
|
|
70
|
-
const envPath = path.join(testDir, '.env');
|
|
71
|
-
expect(await fs.pathExists(envPath)).toBe(true);
|
|
72
|
-
|
|
73
|
-
const content = await fs.readFile(envPath, 'utf8');
|
|
74
|
-
expect(content).toContain('NODE_ENV=development');
|
|
75
|
-
// Version-agnostic: check SINAPSE_VERSION exists with valid semver format
|
|
76
|
-
expect(content).toMatch(/SINAPSE_VERSION=\d+\.\d+\.\d+/);
|
|
77
|
-
});
|
|
78
|
-
|
|
79
|
-
it('should create .env.example file', async () => {
|
|
80
|
-
const result = await configureEnvironment({
|
|
81
|
-
targetDir: testDir,
|
|
82
|
-
skipPrompts: true,
|
|
83
|
-
});
|
|
84
|
-
|
|
85
|
-
expect(result.envExampleCreated).toBe(true);
|
|
86
|
-
|
|
87
|
-
const envExamplePath = path.join(testDir, '.env.example');
|
|
88
|
-
expect(await fs.pathExists(envExamplePath)).toBe(true);
|
|
89
|
-
|
|
90
|
-
const content = await fs.readFile(envExamplePath, 'utf8');
|
|
91
|
-
expect(content).toContain('OPENAI_API_KEY=');
|
|
92
|
-
expect(content).not.toMatch(/sk-[a-zA-Z0-9]{20,}/);
|
|
93
|
-
});
|
|
94
|
-
|
|
95
|
-
it('should create core-config.yaml', async () => {
|
|
96
|
-
const result = await configureEnvironment({
|
|
97
|
-
targetDir: testDir,
|
|
98
|
-
skipPrompts: true,
|
|
99
|
-
projectType: 'BROWNFIELD',
|
|
100
|
-
selectedIDEs: ['vscode', 'cursor'],
|
|
101
|
-
mcpServers: [{ name: 'github' }, { name: 'exa' }],
|
|
102
|
-
});
|
|
103
|
-
|
|
104
|
-
expect(result.coreConfigCreated).toBe(true);
|
|
105
|
-
|
|
106
|
-
const configPath = path.join(testDir, '.sinapse-ai', 'core-config.yaml');
|
|
107
|
-
expect(await fs.pathExists(configPath)).toBe(true);
|
|
108
|
-
|
|
109
|
-
const content = await fs.readFile(configPath, 'utf8');
|
|
110
|
-
expect(content).toContain('type: BROWNFIELD');
|
|
111
|
-
expect(content).toContain('- vscode');
|
|
112
|
-
expect(content).toContain('- cursor');
|
|
113
|
-
expect(content).toContain('- github');
|
|
114
|
-
expect(content).toContain('- exa');
|
|
115
|
-
});
|
|
116
|
-
|
|
117
|
-
it('should update .gitignore', async () => {
|
|
118
|
-
const result = await configureEnvironment({
|
|
119
|
-
targetDir: testDir,
|
|
120
|
-
skipPrompts: true,
|
|
121
|
-
});
|
|
122
|
-
|
|
123
|
-
expect(result.gitignoreUpdated).toBe(true);
|
|
124
|
-
|
|
125
|
-
const gitignorePath = path.join(testDir, '.gitignore');
|
|
126
|
-
expect(await fs.pathExists(gitignorePath)).toBe(true);
|
|
127
|
-
|
|
128
|
-
const content = await fs.readFile(gitignorePath, 'utf8');
|
|
129
|
-
expect(content).toContain('.env');
|
|
130
|
-
});
|
|
131
|
-
|
|
132
|
-
it('should set .env file permissions on Unix', async () => {
|
|
133
|
-
if (process.platform === 'win32') {
|
|
134
|
-
// Skip on Windows
|
|
135
|
-
return;
|
|
136
|
-
}
|
|
137
|
-
|
|
138
|
-
await configureEnvironment({
|
|
139
|
-
targetDir: testDir,
|
|
140
|
-
skipPrompts: true,
|
|
141
|
-
});
|
|
142
|
-
|
|
143
|
-
const envPath = path.join(testDir, '.env');
|
|
144
|
-
const stats = await fs.stat(envPath);
|
|
145
|
-
|
|
146
|
-
// Check permissions are 0600 (owner read/write only)
|
|
147
|
-
const mode = stats.mode & 0o777;
|
|
148
|
-
expect(mode).toBe(0o600);
|
|
149
|
-
});
|
|
150
|
-
|
|
151
|
-
it('should preserve existing .env values via merge (Story 10.38)', async () => {
|
|
152
|
-
// Create existing .env with a user-defined value
|
|
153
|
-
const envPath = path.join(testDir, '.env');
|
|
154
|
-
await fs.writeFile(envPath, 'EXISTING_KEY=existing_value', 'utf8');
|
|
155
|
-
|
|
156
|
-
// Story 10.38: merge-only policy — existing values are ALWAYS preserved.
|
|
157
|
-
await configureEnvironment({
|
|
158
|
-
targetDir: testDir,
|
|
159
|
-
skipPrompts: true,
|
|
160
|
-
});
|
|
161
|
-
|
|
162
|
-
const content = await fs.readFile(envPath, 'utf8');
|
|
163
|
-
// User value is preserved
|
|
164
|
-
expect(content).toContain('EXISTING_KEY=existing_value');
|
|
165
|
-
// SINAPSE variables are added
|
|
166
|
-
expect(content).toMatch(/OPENROUTER_API_KEY|ANTHROPIC_API_KEY|OPENAI_API_KEY/);
|
|
167
|
-
});
|
|
168
|
-
|
|
169
|
-
it('should handle errors gracefully', async () => {
|
|
170
|
-
// Try to write to a read-only directory (simulate permission error)
|
|
171
|
-
const readOnlyDir = path.join(testDir, 'readonly');
|
|
172
|
-
await fs.ensureDir(readOnlyDir);
|
|
173
|
-
|
|
174
|
-
// Note: This test is platform-dependent and may not work on all systems
|
|
175
|
-
// Just verify the function handles errors
|
|
176
|
-
try {
|
|
177
|
-
await configureEnvironment({
|
|
178
|
-
targetDir: '/invalid/path/that/does/not/exist',
|
|
179
|
-
skipPrompts: true,
|
|
180
|
-
});
|
|
181
|
-
fail('Should have thrown an error');
|
|
182
|
-
} catch (error) {
|
|
183
|
-
expect(error).toBeDefined();
|
|
184
|
-
}
|
|
185
|
-
});
|
|
186
|
-
});
|
|
187
|
-
|
|
188
|
-
describe('updateGitignore', () => {
|
|
189
|
-
it('should create .gitignore if not exists', async () => {
|
|
190
|
-
await updateGitignore(testDir);
|
|
191
|
-
|
|
192
|
-
const gitignorePath = path.join(testDir, '.gitignore');
|
|
193
|
-
expect(await fs.pathExists(gitignorePath)).toBe(true);
|
|
194
|
-
|
|
195
|
-
const content = await fs.readFile(gitignorePath, 'utf8');
|
|
196
|
-
expect(content).toContain('.env');
|
|
197
|
-
});
|
|
198
|
-
|
|
199
|
-
it('should append to existing .gitignore', async () => {
|
|
200
|
-
const gitignorePath = path.join(testDir, '.gitignore');
|
|
201
|
-
await fs.writeFile(gitignorePath, 'node_modules\n*.log\n', 'utf8');
|
|
202
|
-
|
|
203
|
-
await updateGitignore(testDir);
|
|
204
|
-
|
|
205
|
-
const content = await fs.readFile(gitignorePath, 'utf8');
|
|
206
|
-
expect(content).toContain('node_modules');
|
|
207
|
-
expect(content).toContain('*.log');
|
|
208
|
-
expect(content).toContain('.env');
|
|
209
|
-
});
|
|
210
|
-
|
|
211
|
-
it('should not duplicate .env entry', async () => {
|
|
212
|
-
const gitignorePath = path.join(testDir, '.gitignore');
|
|
213
|
-
await fs.writeFile(gitignorePath, '.env\n', 'utf8');
|
|
214
|
-
|
|
215
|
-
await updateGitignore(testDir);
|
|
216
|
-
|
|
217
|
-
const content = await fs.readFile(gitignorePath, 'utf8');
|
|
218
|
-
const lines = content.split('\n').filter(line => line.trim() === '.env' || line.trim() === '/.env');
|
|
219
|
-
expect(lines.length).toBe(1);
|
|
220
|
-
});
|
|
221
|
-
|
|
222
|
-
it('should recognize existing .env entry with slash', async () => {
|
|
223
|
-
const gitignorePath = path.join(testDir, '.gitignore');
|
|
224
|
-
await fs.writeFile(gitignorePath, '/.env\n', 'utf8');
|
|
225
|
-
|
|
226
|
-
await updateGitignore(testDir);
|
|
227
|
-
|
|
228
|
-
const content = await fs.readFile(gitignorePath, 'utf8');
|
|
229
|
-
const lines = content.split('\n').filter(line => line.trim() === '.env' || line.trim() === '/.env');
|
|
230
|
-
expect(lines.length).toBe(1);
|
|
231
|
-
});
|
|
232
|
-
});
|
|
233
|
-
|
|
234
|
-
describe('Wizard Integration', () => {
|
|
235
|
-
it('should integrate with wizard state from previous stories', async () => {
|
|
236
|
-
// Simulate wizard state from Stories 1.3, 1.4, 1.5
|
|
237
|
-
const wizardState = {
|
|
238
|
-
projectType: 'GREENFIELD',
|
|
239
|
-
selectedIDEs: ['vscode', 'cursor'],
|
|
240
|
-
mcpServers: [
|
|
241
|
-
{ name: 'github', id: 'github' },
|
|
242
|
-
{ name: 'exa', id: 'exa' },
|
|
243
|
-
],
|
|
244
|
-
};
|
|
245
|
-
|
|
246
|
-
const result = await configureEnvironment({
|
|
247
|
-
targetDir: testDir,
|
|
248
|
-
skipPrompts: true,
|
|
249
|
-
...wizardState,
|
|
250
|
-
});
|
|
251
|
-
|
|
252
|
-
// Verify all files created
|
|
253
|
-
expect(result.envCreated).toBe(true);
|
|
254
|
-
expect(result.coreConfigCreated).toBe(true);
|
|
255
|
-
expect(result.gitignoreUpdated).toBe(true);
|
|
256
|
-
|
|
257
|
-
// Verify core-config includes wizard state
|
|
258
|
-
const configPath = path.join(testDir, '.sinapse-ai', 'core-config.yaml');
|
|
259
|
-
const configContent = await fs.readFile(configPath, 'utf8');
|
|
260
|
-
|
|
261
|
-
expect(configContent).toContain('GREENFIELD');
|
|
262
|
-
expect(configContent).toContain('vscode');
|
|
263
|
-
expect(configContent).toContain('cursor');
|
|
264
|
-
expect(configContent).toContain('github');
|
|
265
|
-
expect(configContent).toContain('exa');
|
|
266
|
-
});
|
|
267
|
-
});
|
|
268
|
-
|
|
269
|
-
describe('Cross-Platform Compatibility', () => {
|
|
270
|
-
it('should handle Windows paths', async () => {
|
|
271
|
-
const result = await configureEnvironment({
|
|
272
|
-
targetDir: testDir,
|
|
273
|
-
skipPrompts: true,
|
|
274
|
-
});
|
|
275
|
-
|
|
276
|
-
expect(result.envCreated).toBe(true);
|
|
277
|
-
expect(result.coreConfigCreated).toBe(true);
|
|
278
|
-
});
|
|
279
|
-
|
|
280
|
-
it('should create valid YAML on all platforms', async () => {
|
|
281
|
-
await configureEnvironment({
|
|
282
|
-
targetDir: testDir,
|
|
283
|
-
skipPrompts: true,
|
|
284
|
-
projectType: 'GREENFIELD',
|
|
285
|
-
});
|
|
286
|
-
|
|
287
|
-
const yaml = require('js-yaml');
|
|
288
|
-
const configPath = path.join(testDir, '.sinapse-ai', 'core-config.yaml');
|
|
289
|
-
const content = await fs.readFile(configPath, 'utf8');
|
|
290
|
-
|
|
291
|
-
// Should parse without errors
|
|
292
|
-
const parsed = yaml.load(content);
|
|
293
|
-
expect(parsed).toBeDefined();
|
|
294
|
-
expect(parsed.project.type).toBe('GREENFIELD');
|
|
295
|
-
});
|
|
296
|
-
});
|
|
297
|
-
|
|
298
|
-
describe('File Structure', () => {
|
|
299
|
-
it('should create correct directory structure', async () => {
|
|
300
|
-
await configureEnvironment({
|
|
301
|
-
targetDir: testDir,
|
|
302
|
-
skipPrompts: true,
|
|
303
|
-
});
|
|
304
|
-
|
|
305
|
-
// Check all expected files exist
|
|
306
|
-
const expectedFiles = [
|
|
307
|
-
'.env',
|
|
308
|
-
'.env.example',
|
|
309
|
-
'.gitignore',
|
|
310
|
-
'.sinapse-ai/core-config.yaml',
|
|
311
|
-
];
|
|
312
|
-
|
|
313
|
-
for (const file of expectedFiles) {
|
|
314
|
-
const filePath = path.join(testDir, file);
|
|
315
|
-
expect(await fs.pathExists(filePath)).toBe(true);
|
|
316
|
-
}
|
|
317
|
-
});
|
|
318
|
-
|
|
319
|
-
it('should create .sinapse-ai directory if missing', async () => {
|
|
320
|
-
const sinapsecoreDir = path.join(testDir, '.sinapse-ai');
|
|
321
|
-
expect(await fs.pathExists(sinapsecoreDir)).toBe(false);
|
|
322
|
-
|
|
323
|
-
await configureEnvironment({
|
|
324
|
-
targetDir: testDir,
|
|
325
|
-
skipPrompts: true,
|
|
326
|
-
});
|
|
327
|
-
|
|
328
|
-
expect(await fs.pathExists(sinapsecoreDir)).toBe(true);
|
|
329
|
-
});
|
|
330
|
-
});
|
|
331
|
-
});
|
|
332
|
-
|