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.
Files changed (361) hide show
  1. package/.claude/hooks/mind-clone-governance.py +212 -212
  2. package/.claude/hooks/read-protection.py +152 -152
  3. package/.claude/hooks/slug-validation.py +175 -175
  4. package/.claude/hooks/sql-governance.py +183 -183
  5. package/.claude/rules/documentation-first.md +1 -1
  6. package/.claude/rules/hook-governance.md +1 -1
  7. package/.claude/rules/mandatory-delegation.md +1 -1
  8. package/.claude/rules/project-intelligence.md +1 -1
  9. package/.codex/agents/analyst.md +4 -371
  10. package/.codex/agents/animations-orqx.md +4 -57
  11. package/.codex/agents/architect.md +4 -560
  12. package/.codex/agents/brand-orqx.md +4 -95
  13. package/.codex/agents/claude-mastery-chief.md +4 -0
  14. package/.codex/agents/cloning-orqx.md +4 -70
  15. package/.codex/agents/commercial-orqx.md +4 -67
  16. package/.codex/agents/config-engineer.md +2 -2
  17. package/.codex/agents/content-orqx.md +4 -77
  18. package/.codex/agents/copy-orqx.md +4 -65
  19. package/.codex/agents/cost-optimizer.md +4 -0
  20. package/.codex/agents/council-orqx.md +4 -68
  21. package/.codex/agents/courses-orqx.md +4 -64
  22. package/.codex/agents/cro-persuasion.md +4 -0
  23. package/.codex/agents/cyber-orqx.md +4 -67
  24. package/.codex/agents/data-engineer.md +4 -542
  25. package/.codex/agents/design-orqx.md +4 -65
  26. package/.codex/agents/design-system.md +4 -210
  27. package/.codex/agents/developer.md +4 -666
  28. package/.codex/agents/devops.md +4 -668
  29. package/.codex/agents/finance-orqx.md +4 -57
  30. package/.codex/agents/fiscal-compliance-br.md +4 -0
  31. package/.codex/agents/forecast-strategist.md +4 -0
  32. package/.codex/agents/growth-orqx.md +4 -75
  33. package/.codex/agents/hooks-architect.md +2 -2
  34. package/.codex/agents/mcp-integrator.md +2 -2
  35. package/.codex/agents/paidmedia-orqx.md +4 -67
  36. package/.codex/agents/platform-aesthetic-director.md +4 -0
  37. package/.codex/agents/premium-packaging-strategist.md +4 -0
  38. package/.codex/agents/product-lead.md +4 -371
  39. package/.codex/agents/product-orqx.md +4 -57
  40. package/.codex/agents/product-surface-director.md +4 -0
  41. package/.codex/agents/project-integrator.md +2 -2
  42. package/.codex/agents/project-lead.md +4 -414
  43. package/.codex/agents/quality-gate.md +4 -547
  44. package/.codex/agents/research-orqx.md +4 -67
  45. package/.codex/agents/roadmap-sentinel.md +2 -2
  46. package/.codex/agents/skill-craftsman.md +2 -2
  47. package/.codex/agents/snps-orqx.md +4 -684
  48. package/.codex/agents/sop-extractor.md +4 -61
  49. package/.codex/agents/sprint-lead.md +4 -324
  50. package/.codex/agents/squad-creator.md +4 -402
  51. package/.codex/agents/storytelling-orqx.md +4 -65
  52. package/.codex/agents/swarm-orqx.md +4 -64
  53. package/.codex/agents/ux-design-expert.md +4 -532
  54. package/.codex/agents/ux-designer.md +4 -124
  55. package/.codex/command-registry.json +9 -9
  56. package/.codex/delegation-matrix.json +375 -839
  57. package/.codex/delegation-parity.json +658 -0
  58. package/.codex/handoff-packet.parity.schema.json +148 -0
  59. package/.codex/handoff-packet.template.json +26 -0
  60. package/.codex/instructions.md +8 -8
  61. package/.codex/scripts/resolve-codex-agent.js +482 -0
  62. package/.codex/scripts/resolve-codex-command.js +75 -12
  63. package/.codex/scripts/resolve-codex-delegation.js +131 -92
  64. package/.codex/skills/sinapse-claude/SKILL.md +3 -3
  65. package/.codex/skills/sinapse-po/SKILL.md +1 -1
  66. package/.codex/tasks/resolve-sinapse-conflict.md +1 -1
  67. package/.sinapse-ai/constitution.md +5 -5
  68. package/.sinapse-ai/core/doctor/checks/git-hooks.js +163 -19
  69. package/.sinapse-ai/core/events/dashboard-emitter.js +30 -9
  70. package/.sinapse-ai/core/execution/subagent-dispatcher.js +1 -1
  71. package/.sinapse-ai/core/synapse/engine.js +15 -0
  72. package/.sinapse-ai/core/ui/observability-panel.js +240 -0
  73. package/.sinapse-ai/core-config.yaml +0 -20
  74. package/.sinapse-ai/data/entity-registry.yaml +185 -236
  75. package/.sinapse-ai/development/agents/snps-orqx.md +16 -26
  76. package/.sinapse-ai/development/tasks/build-autonomous.md +11 -1
  77. package/.sinapse-ai/development/tasks/build-resume.md +8 -0
  78. package/.sinapse-ai/development/tasks/build-status.md +8 -0
  79. package/.sinapse-ai/development/tasks/build.md +8 -0
  80. package/.sinapse-ai/development/tasks/cleanup-worktrees.md +8 -1
  81. package/.sinapse-ai/development/tasks/gotcha.md +8 -0
  82. package/.sinapse-ai/development/tasks/gotchas.md +8 -0
  83. package/.sinapse-ai/development/tasks/ids-health.md +14 -6
  84. package/.sinapse-ai/development/tasks/list-mcps.md +15 -0
  85. package/.sinapse-ai/development/tasks/merge-worktree.md +8 -1
  86. package/.sinapse-ai/development/tasks/qa-review-build.md +18 -0
  87. package/.sinapse-ai/development/tasks/remove-mcp.md +8 -1
  88. package/.sinapse-ai/development/tasks/validate-agents.md +26 -14
  89. package/.sinapse-ai/development/templates/service-template/README.md.hbs +159 -159
  90. package/.sinapse-ai/development/templates/service-template/__tests__/index.test.ts.hbs +238 -238
  91. package/.sinapse-ai/development/templates/service-template/client.ts.hbs +404 -404
  92. package/.sinapse-ai/development/templates/service-template/errors.ts.hbs +183 -183
  93. package/.sinapse-ai/development/templates/service-template/index.ts.hbs +121 -121
  94. package/.sinapse-ai/development/templates/service-template/package.json.hbs +88 -88
  95. package/.sinapse-ai/development/templates/service-template/types.ts.hbs +146 -146
  96. package/.sinapse-ai/development/templates/squad-template/LICENSE +22 -22
  97. package/.sinapse-ai/git-hooks/lib/framework-guard.js +258 -0
  98. package/.sinapse-ai/git-hooks/lib/secret-scanner-core.js +355 -0
  99. package/.sinapse-ai/git-hooks/lib/staged-secret-scan.js +179 -0
  100. package/.sinapse-ai/git-hooks/lib/staged-sql-guard.js +204 -0
  101. package/.sinapse-ai/git-hooks/post-commit +28 -0
  102. package/.sinapse-ai/git-hooks/pre-commit +81 -0
  103. package/.sinapse-ai/git-hooks/pre-push +83 -0
  104. package/.sinapse-ai/hooks/ids-post-commit.js +13 -11
  105. package/.sinapse-ai/hooks/ids-pre-push.js +9 -7
  106. package/.sinapse-ai/infrastructure/scripts/codex-parity/resolve.js +161 -0
  107. package/.sinapse-ai/infrastructure/scripts/dashboard-status-writer.js +6 -2
  108. package/.sinapse-ai/infrastructure/scripts/ide-sync/index.js +65 -68
  109. package/.sinapse-ai/infrastructure/scripts/sync-codex-local-first.js +156 -1
  110. package/.sinapse-ai/infrastructure/scripts/validate-codex-delegation.js +1 -4
  111. package/.sinapse-ai/infrastructure/scripts/validate-codex-integration.js +41 -5
  112. package/.sinapse-ai/infrastructure/templates/coderabbit.yaml.template +280 -280
  113. package/.sinapse-ai/infrastructure/templates/config/env.example +16 -16
  114. package/.sinapse-ai/infrastructure/templates/config/gitignore-additions.tmpl +59 -59
  115. package/.sinapse-ai/infrastructure/templates/github/CODEOWNERS.template +12 -12
  116. package/.sinapse-ai/infrastructure/templates/github-workflows/ci.yml.template +170 -170
  117. package/.sinapse-ai/infrastructure/templates/github-workflows/pr-automation.yml.template +331 -331
  118. package/.sinapse-ai/infrastructure/templates/github-workflows/release.yml.template +197 -197
  119. package/.sinapse-ai/infrastructure/templates/gitignore/gitignore-brownfield-merge.tmpl +19 -19
  120. package/.sinapse-ai/infrastructure/templates/gitignore/gitignore-node.tmpl +86 -86
  121. package/.sinapse-ai/infrastructure/templates/gitignore/gitignore-python.tmpl +146 -146
  122. package/.sinapse-ai/infrastructure/templates/gitignore/gitignore-sinapse-base.tmpl +64 -64
  123. package/.sinapse-ai/infrastructure/templates/safe-collab/CODEOWNERS.template +16 -16
  124. package/.sinapse-ai/infrastructure/templates/sinapse-sync.yaml.template +183 -183
  125. package/.sinapse-ai/install-manifest.yaml +112 -164
  126. package/.sinapse-ai/local-config.yaml.template +65 -65
  127. package/.sinapse-ai/product/templates/adr.hbs +126 -126
  128. package/.sinapse-ai/product/templates/dbdr.hbs +242 -242
  129. package/.sinapse-ai/product/templates/epic.hbs +213 -213
  130. package/.sinapse-ai/product/templates/ide-rules/codex-rules.md +30 -0
  131. package/.sinapse-ai/product/templates/pmdr.hbs +187 -187
  132. package/.sinapse-ai/product/templates/prd-v2.0.hbs +217 -217
  133. package/.sinapse-ai/product/templates/prd.hbs +202 -202
  134. package/.sinapse-ai/product/templates/statusline/statusline-script.js +31 -8
  135. package/.sinapse-ai/product/templates/statusline/track-agent-clear.cjs +79 -0
  136. package/.sinapse-ai/product/templates/statusline/track-agent.cjs +218 -0
  137. package/.sinapse-ai/product/templates/story.hbs +264 -264
  138. package/.sinapse-ai/product/templates/task.hbs +171 -171
  139. package/.sinapse-ai/product/templates/tmpl-comment-on-examples.sql +159 -159
  140. package/.sinapse-ai/product/templates/tmpl-migration-script.sql +92 -92
  141. package/.sinapse-ai/product/templates/tmpl-rls-granular-policies.sql +105 -105
  142. package/.sinapse-ai/product/templates/tmpl-rls-kiss-policy.sql +11 -11
  143. package/.sinapse-ai/product/templates/tmpl-rls-roles.sql +136 -136
  144. package/.sinapse-ai/product/templates/tmpl-rls-simple.sql +78 -78
  145. package/.sinapse-ai/product/templates/tmpl-rls-tenant.sql +153 -153
  146. package/.sinapse-ai/product/templates/tmpl-rollback-script.sql +78 -78
  147. package/.sinapse-ai/product/templates/tmpl-seed-data.sql +141 -141
  148. package/.sinapse-ai/product/templates/tmpl-smoke-test.sql +17 -17
  149. package/.sinapse-ai/product/templates/tmpl-staging-copy-merge.sql +140 -140
  150. package/.sinapse-ai/product/templates/tmpl-stored-proc.sql +141 -141
  151. package/.sinapse-ai/product/templates/tmpl-trigger.sql +153 -153
  152. package/.sinapse-ai/product/templates/tmpl-view-materialized.sql +134 -134
  153. package/.sinapse-ai/product/templates/tmpl-view.sql +178 -178
  154. package/AGENTS.md +193 -0
  155. package/CHANGELOG.md +1247 -0
  156. package/LICENSE +63 -63
  157. package/README.en.md +17 -18
  158. package/README.md +18 -19
  159. package/bin/cli.js +1 -1
  160. package/bin/commands/install.js +194 -22
  161. package/bin/commands/status.js +14 -1
  162. package/bin/commands/uninstall.js +2 -2
  163. package/bin/commands/update.js +52 -0
  164. package/bin/lib/setup-statusline.js +191 -0
  165. package/bin/sinapse-init.js +11 -83
  166. package/bin/utils/framework-guard.js +17 -4
  167. package/bin/utils/secret-scanner-core.js +109 -7
  168. package/bin/utils/staged-sql-guard.js +204 -0
  169. package/bin/utils/validate-publish.js +63 -0
  170. package/docs/agent-reference-guide.md +5 -7
  171. package/docs/framework/agent-prefix-convention.md +58 -0
  172. package/docs/framework/architecture-overview.md +4 -4
  173. package/docs/framework/collaboration-activation.md +45 -0
  174. package/docs/framework/guiding-principles.md +9 -9
  175. package/docs/getting-started.md +1 -1
  176. package/docs/guides/agent-reference.md +1 -1
  177. package/docs/guides/codex-config.md +4 -5
  178. package/docs/pt/architecture/sub-orqx-pattern.md +20 -18
  179. package/docs/security/overview.md +1 -1
  180. package/package.json +16 -12
  181. package/packages/installer/src/index.js +26 -0
  182. package/packages/installer/src/installer/git-hooks-installer.js +211 -47
  183. package/packages/installer/src/installer/sinapse-ai-installer.js +71 -0
  184. package/packages/installer/src/wizard/feedback.js +1 -1
  185. package/packages/installer/src/wizard/ide-config-generator.js +26 -26
  186. package/packages/installer/src/wizard/index.js +53 -4
  187. package/packages/sinapse-install/bin/edmcp.js +0 -0
  188. package/packages/sinapse-install/bin/sinapse-install.js +0 -0
  189. package/scripts/audit-tasks.cjs +112 -91
  190. package/scripts/check-markdown-links.py +352 -352
  191. package/scripts/prepare-hooks.js +58 -0
  192. package/scripts/regenerate-orqx-stubs.ps1 +2 -3
  193. package/scripts/sync-counts.js +10 -2
  194. package/scripts/sync-squad-yaml-components.js +108 -6
  195. package/scripts/validate-agents-md.js +128 -0
  196. package/scripts/validate-all.js +1 -0
  197. package/scripts/validate-squad-orqx.js +19 -9
  198. package/sinapse/agents/sinapse-orqx.md +16 -26
  199. package/sinapse/agents/snps-orqx.md +15 -25
  200. package/sinapse/knowledge-base/routing-catalog.md +1 -1
  201. package/sinapse/tasks/diagnose-and-route.md +1 -1
  202. package/sinapse/tasks/squad-status-report.md +1 -1
  203. package/squads/claude-code-mastery/agents/claude-mastery-chief.md +1 -1
  204. package/squads/claude-code-mastery/agents/hooks-architect.md +60 -68
  205. package/squads/claude-code-mastery/knowledge-base/swarm-orchestration-patterns.md +1 -1
  206. package/squads/claude-code-mastery/squad.yaml +8 -0
  207. package/squads/claude-code-mastery/tasks/audit-setup.md +1 -1
  208. package/squads/claude-code-mastery/workflows/optimization-cycle.yaml +4 -4
  209. package/squads/claude-code-mastery/workflows/project-setup-cycle.yaml +4 -4
  210. package/squads/squad-animations/README.md +1 -1
  211. package/squads/squad-animations/squad.yaml +1 -1
  212. package/squads/squad-brand/squad.yaml +1 -1
  213. package/squads/squad-cloning/README.md +1 -1
  214. package/squads/squad-cloning/squad.yaml +1 -1
  215. package/squads/squad-commercial/README.md +1 -1
  216. package/squads/squad-commercial/squad.yaml +2 -3
  217. package/squads/squad-content/README.md +1 -1
  218. package/squads/squad-content/squad.yaml +1 -1
  219. package/squads/squad-copy/README.md +1 -1
  220. package/squads/squad-copy/squad.yaml +2 -3
  221. package/squads/squad-council/README.md +1 -1
  222. package/squads/squad-courses/README.md +1 -1
  223. package/squads/squad-courses/squad.yaml +1 -1
  224. package/squads/squad-cybersecurity/README.md +1 -1
  225. package/squads/squad-cybersecurity/squad.yaml +2 -3
  226. package/squads/squad-design/README.md +1 -1
  227. package/squads/{squad-artdir → squad-design}/agents/cro-persuasion.md +1 -1
  228. package/squads/{squad-artdir → squad-design}/agents/platform-aesthetic-director.md +2 -2
  229. package/squads/{squad-artdir → squad-design}/agents/premium-packaging-strategist.md +2 -2
  230. package/squads/{squad-artdir → squad-design}/agents/product-surface-director.md +3 -3
  231. package/squads/squad-design/squad.yaml +6 -3
  232. package/squads/squad-finance/README.md +1 -1
  233. package/squads/squad-finance/squad.yaml +7 -1
  234. package/squads/squad-growth/README.md +1 -1
  235. package/squads/squad-growth/squad.yaml +1 -1
  236. package/squads/squad-paidmedia/README.md +1 -1
  237. package/squads/squad-paidmedia/squad.yaml +2 -3
  238. package/squads/squad-product/README.md +1 -1
  239. package/squads/squad-product/squad.yaml +1 -1
  240. package/squads/squad-research/README.md +1 -1
  241. package/squads/squad-research/squad.yaml +2 -3
  242. package/squads/squad-storytelling/README.md +1 -1
  243. package/squads/squad-storytelling/squad.yaml +2 -3
  244. package/.codex/agents/brad-frost.md +0 -46
  245. package/.codex/agents/claude-orqx.md +0 -72
  246. package/.codex/agents/copy-chief.md +0 -162
  247. package/.codex/agents/cyber-chief.md +0 -169
  248. package/.codex/agents/dan-mall.md +0 -43
  249. package/.codex/agents/data-chief.md +0 -198
  250. package/.codex/agents/dave-malouf.md +0 -43
  251. package/.codex/agents/db-sage.md +0 -152
  252. package/.codex/agents/design-chief.md +0 -226
  253. package/.codex/agents/dev.md +0 -102
  254. package/.codex/agents/legal-chief.md +0 -199
  255. package/.codex/agents/nano-banana-generator.md +0 -42
  256. package/.codex/agents/pm.md +0 -81
  257. package/.codex/agents/po.md +0 -85
  258. package/.codex/agents/qa.md +0 -98
  259. package/.codex/agents/sm.md +0 -77
  260. package/.codex/agents/squad-chief.md +0 -1553
  261. package/.codex/agents/squad.md +0 -66
  262. package/.codex/agents/story-chief.md +0 -180
  263. package/.codex/agents/tools-orqx.md +0 -219
  264. package/.codex/agents/traffic-masters-chief.md +0 -211
  265. package/.sinapse-ai/core/memory/__tests__/active-modules.verify.js +0 -265
  266. package/.sinapse-ai/core/permissions/__tests__/permission-mode.test.js +0 -293
  267. package/.sinapse-ai/data/registry-update-log.jsonl +0 -158
  268. package/.sinapse-ai/infrastructure/scripts/ide-sync/gemini-commands.js +0 -298
  269. package/.sinapse-ai/infrastructure/scripts/ide-sync/transformers/antigravity.js +0 -121
  270. package/.sinapse-ai/infrastructure/scripts/ide-sync/transformers/cursor.js +0 -119
  271. package/.sinapse-ai/infrastructure/scripts/ide-sync/transformers/github-copilot.js +0 -191
  272. package/.sinapse-ai/infrastructure/scripts/ide-sync/transformers/kimi.js +0 -448
  273. package/.sinapse-ai/infrastructure/tests/project-status-loader.test.js +0 -569
  274. package/.sinapse-ai/infrastructure/tests/regression-suite-v2.md +0 -622
  275. package/.sinapse-ai/infrastructure/tests/validate-module.js +0 -98
  276. package/.sinapse-ai/infrastructure/tests/worktree-manager.test.js +0 -620
  277. package/.sinapse-ai/monitor/hooks/lib/__init__.py +0 -2
  278. package/.sinapse-ai/monitor/hooks/lib/enrich.py +0 -59
  279. package/.sinapse-ai/monitor/hooks/lib/send_event.py +0 -48
  280. package/.sinapse-ai/monitor/hooks/notification.py +0 -30
  281. package/.sinapse-ai/monitor/hooks/post_tool_use.py +0 -46
  282. package/.sinapse-ai/monitor/hooks/pre_compact.py +0 -30
  283. package/.sinapse-ai/monitor/hooks/pre_tool_use.py +0 -41
  284. package/.sinapse-ai/monitor/hooks/stop.py +0 -30
  285. package/.sinapse-ai/monitor/hooks/subagent_stop.py +0 -30
  286. package/.sinapse-ai/monitor/hooks/user_prompt_submit.py +0 -39
  287. package/.sinapse-ai/product/templates/statusline/track-agent.sh +0 -69
  288. package/.sinapse-ai/workflow-intelligence/__tests__/confidence-scorer.test.js +0 -335
  289. package/.sinapse-ai/workflow-intelligence/__tests__/integration.test.js +0 -340
  290. package/.sinapse-ai/workflow-intelligence/__tests__/suggestion-engine.test.js +0 -438
  291. package/.sinapse-ai/workflow-intelligence/__tests__/wave-analyzer.test.js +0 -448
  292. package/.sinapse-ai/workflow-intelligence/__tests__/workflow-registry.test.js +0 -303
  293. package/bin/sinapse-graph.js +0 -19
  294. package/docs/codex-integration-process.md +0 -22
  295. package/docs/codex-parity-program.md +0 -27
  296. package/packages/installer/src/__tests__/performance-benchmark.js +0 -383
  297. package/packages/installer/tests/integration/environment-configuration.test.js +0 -332
  298. package/packages/installer/tests/integration/wizard-detection.test.js +0 -352
  299. package/packages/installer/tests/unit/artifact-copy-pipeline/artifact-copy-pipeline.test.js +0 -383
  300. package/packages/installer/tests/unit/claude-md-template-v5/claude-md-template-v5.test.js +0 -193
  301. package/packages/installer/tests/unit/config-validator.test.js +0 -315
  302. package/packages/installer/tests/unit/detection/detect-project-type.test.js +0 -539
  303. package/packages/installer/tests/unit/doctor/doctor-checks.test.js +0 -636
  304. package/packages/installer/tests/unit/doctor/doctor-orchestrator.test.js +0 -192
  305. package/packages/installer/tests/unit/entity-registry-bootstrap.test.js +0 -186
  306. package/packages/installer/tests/unit/env-template.test.js +0 -187
  307. package/packages/installer/tests/unit/generate-settings-json/generate-settings-json.test.js +0 -310
  308. package/packages/installer/tests/unit/git-hooks-installer.test.js +0 -262
  309. package/packages/installer/tests/unit/ide-sync-integration/ide-sync-integration.test.js +0 -231
  310. package/packages/installer/tests/unit/merger/env-merger.test.js +0 -191
  311. package/packages/installer/tests/unit/merger/markdown-merger.test.js +0 -262
  312. package/packages/installer/tests/unit/merger/strategies.test.js +0 -154
  313. package/packages/installer/tests/unit/merger/yaml-merger.test.js +0 -328
  314. package/packages/sinapse-install/tests/unit/chrome-brain.smoke.test.js +0 -66
  315. package/scripts/install-monitor-hooks.sh +0 -82
  316. package/squads/squad-artdir/README.md +0 -90
  317. package/squads/squad-artdir/agents/accessibility-guardian.md +0 -184
  318. package/squads/squad-artdir/agents/artdir-orqx.md +0 -222
  319. package/squads/squad-artdir/agents/color-psychologist.md +0 -166
  320. package/squads/squad-artdir/agents/design-system-architect.md +0 -100
  321. package/squads/squad-artdir/agents/ia-architect.md +0 -169
  322. package/squads/squad-artdir/agents/interaction-designer.md +0 -162
  323. package/squads/squad-artdir/agents/layout-engineer.md +0 -163
  324. package/squads/squad-artdir/agents/motion-architect.md +0 -185
  325. package/squads/squad-artdir/agents/type-systemist.md +0 -138
  326. package/squads/squad-artdir/agents/visual-strategist.md +0 -127
  327. package/squads/squad-artdir/checklists/seven-pillars-validation-checklist.md +0 -172
  328. package/squads/squad-artdir/knowledge-base/case-nyo-ia-reference.md +0 -289
  329. package/squads/squad-artdir/knowledge-base/deliverables-templates.md +0 -457
  330. package/squads/squad-artdir/knowledge-base/motion-technique-catalog.md +0 -247
  331. package/squads/squad-artdir/knowledge-base/premium-packaging-principles.md +0 -133
  332. package/squads/squad-artdir/knowledge-base/psychological-toolkit.md +0 -229
  333. package/squads/squad-artdir/knowledge-base/saas-art-direction-canon.md +0 -242
  334. package/squads/squad-artdir/knowledge-base/seven-pillars-framework.md +0 -289
  335. package/squads/squad-artdir/knowledge-base/ten-pillars-framework.md +0 -221
  336. package/squads/squad-artdir/package.json +0 -20
  337. package/squads/squad-artdir/squad.yaml +0 -299
  338. package/squads/squad-artdir/tasks/audit-conversion.md +0 -97
  339. package/squads/squad-artdir/tasks/audit-drift-multi-surface.md +0 -55
  340. package/squads/squad-artdir/tasks/consult-saas-canon.md +0 -54
  341. package/squads/squad-artdir/tasks/create-art-direction-brief.md +0 -110
  342. package/squads/squad-artdir/tasks/create-premium-packaging-brief.md +0 -61
  343. package/squads/squad-artdir/tasks/create-wireflow.md +0 -84
  344. package/squads/squad-artdir/tasks/design-color-system.md +0 -81
  345. package/squads/squad-artdir/tasks/design-product-surface.md +0 -60
  346. package/squads/squad-artdir/tasks/design-token-system.md +0 -58
  347. package/squads/squad-artdir/tasks/diagnose-visual-language.md +0 -92
  348. package/squads/squad-artdir/tasks/first-5-minutes-choreography.md +0 -65
  349. package/squads/squad-artdir/tasks/specify-motion-system.md +0 -84
  350. package/squads/squad-artdir/tasks/validate-against-pillars.md +0 -143
  351. package/squads/squad-artdir/templates/art-direction-brief-template.md +0 -215
  352. package/squads/squad-artdir/workflows/conversion-audit-cycle.yaml +0 -142
  353. package/squads/squad-artdir/workflows/full-art-direction-cycle.yaml +0 -179
  354. package/squads/squad-artdir/workflows/saas-platform-art-direction-cycle.yaml +0 -338
  355. package/squads/squad-commercial/agents/legal-chief.md +0 -199
  356. package/squads/squad-copy/agents/copy-chief.md +0 -162
  357. package/squads/squad-cybersecurity/agents/cyber-chief.md +0 -169
  358. package/squads/squad-design/agents/design-chief.md +0 -226
  359. package/squads/squad-paidmedia/agents/traffic-masters-chief.md +0 -211
  360. package/squads/squad-research/agents/data-chief.md +0 -198
  361. 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
-