@polymorphism-tech/morph-spec 3.0.0 → 3.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (240) hide show
  1. package/CLAUDE.md +68 -400
  2. package/README.md +198 -76
  3. package/bin/detect-agents.js +227 -225
  4. package/bin/morph-spec.js +10 -0
  5. package/bin/render-template.js +303 -302
  6. package/bin/semantic-detect-agents.js +247 -246
  7. package/bin/{task-manager.js → task-manager.cjs} +12 -1
  8. package/bin/validate-agents-skills.js +257 -251
  9. package/bin/validate-agents.js +70 -69
  10. package/bin/validate-phase.js +263 -263
  11. package/docs/getting-started.md +3 -3
  12. package/package.json +3 -4
  13. package/scripts/reorganize-skills.cjs +175 -0
  14. package/scripts/validate-agents-structure.cjs +52 -0
  15. package/scripts/validate-skills.cjs +180 -0
  16. package/src/commands/create-story.js +354 -351
  17. package/src/commands/detect-agents.js +13 -2
  18. package/src/commands/detect.js +104 -104
  19. package/src/commands/state.js +334 -333
  20. package/src/commands/sync.js +167 -167
  21. package/src/commands/task.js +1 -1
  22. package/src/commands/update.js +13 -1
  23. package/src/lib/context-generator.js +7 -4
  24. package/{detectors → src/lib/detectors}/config-detector.js +223 -223
  25. package/{detectors → src/lib/detectors}/conversation-analyzer.js +163 -163
  26. package/{detectors → src/lib/detectors}/index.js +84 -84
  27. package/{detectors → src/lib/detectors}/standards-generator.js +275 -275
  28. package/src/lib/hook-executor.js +2 -1
  29. package/src/lib/stack-resolver.js +148 -0
  30. package/src/lib/standards-context-injector.js +4 -3
  31. package/src/lib/state-manager.js +21 -4
  32. package/src/lib/team-orchestrator.js +2 -1
  33. package/src/lib/troubleshoot-grep.js +13 -3
  34. package/src/lib/validation-runner.js +2 -1
  35. package/src/utils/file-copier.js +3 -1
  36. package/{content → stacks/blazor-azure}/.azure/README.md +293 -293
  37. package/{content → stacks/blazor-azure}/.azure/docs/azure-devops-setup.md +454 -454
  38. package/{content → stacks/blazor-azure}/.azure/docs/branch-strategy.md +398 -398
  39. package/{content → stacks/blazor-azure}/.azure/docs/local-development.md +515 -515
  40. package/{content → stacks/blazor-azure}/.azure/pipelines/pipeline-variables.yml +34 -34
  41. package/{content → stacks/blazor-azure}/.azure/pipelines/prod-pipeline.yml +319 -319
  42. package/{content → stacks/blazor-azure}/.azure/pipelines/staging-pipeline.yml +234 -234
  43. package/{content → stacks/blazor-azure}/.azure/pipelines/templates/build-dotnet.yml +75 -75
  44. package/{content → stacks/blazor-azure}/.azure/pipelines/templates/deploy-app-service.yml +94 -94
  45. package/{content → stacks/blazor-azure}/.azure/pipelines/templates/deploy-container-app.yml +120 -120
  46. package/{content → stacks/blazor-azure}/.azure/pipelines/templates/infra-deploy.yml +90 -90
  47. package/{content → stacks/blazor-azure}/.claude/commands/morph-archive.md +79 -79
  48. package/{content → stacks/blazor-azure}/.claude/commands/morph-deploy.md +529 -529
  49. package/{content → stacks/blazor-azure}/.claude/commands/morph-infra.md +209 -209
  50. package/{content → stacks/blazor-azure}/.claude/commands/morph-troubleshoot.md +1 -1
  51. package/{content → stacks/blazor-azure}/.claude/settings.local.json +15 -15
  52. package/{content → stacks/blazor-azure}/.claude/skills/level-1-workflows/phase-setup.md +1 -1
  53. package/{content/.claude/skills/specialists → stacks/blazor-azure/.claude/skills/level-2-domains/architecture}/prompt-engineer.md +189 -189
  54. package/{content/.claude/skills/specialists → stacks/blazor-azure/.claude/skills/level-2-domains/architecture}/seo-growth-hacker.md +320 -320
  55. package/{content/.claude/skills/infra → stacks/blazor-azure/.claude/skills/level-2-domains/infrastructure}/azure-deploy-specialist.md +699 -699
  56. package/{content → stacks/blazor-azure}/.morph/.morphversion +5 -5
  57. package/{content → stacks/blazor-azure}/.morph/archive/.gitkeep +25 -25
  58. package/{content → stacks/blazor-azure}/.morph/config/agents.json +7 -5
  59. package/{content → stacks/blazor-azure}/.morph/docs/STORY-DRIVEN-DEVELOPMENT.md +392 -392
  60. package/{content → stacks/blazor-azure}/.morph/docs/workflows/enforcement-pipeline.md +3 -3
  61. package/{content → stacks/blazor-azure}/.morph/examples/api-nextjs/README.md +241 -241
  62. package/{content → stacks/blazor-azure}/.morph/examples/api-nextjs/contracts.ts +307 -307
  63. package/{content → stacks/blazor-azure}/.morph/examples/api-nextjs/spec.md +399 -399
  64. package/{content → stacks/blazor-azure}/.morph/examples/api-nextjs/tasks.md +168 -168
  65. package/{content → stacks/blazor-azure}/.morph/examples/micro-saas/README.md +125 -125
  66. package/{content → stacks/blazor-azure}/.morph/examples/micro-saas/contracts.cs +358 -358
  67. package/{content → stacks/blazor-azure}/.morph/examples/micro-saas/decisions.md +246 -246
  68. package/{content → stacks/blazor-azure}/.morph/examples/micro-saas/spec.md +236 -236
  69. package/{content → stacks/blazor-azure}/.morph/examples/micro-saas/tasks.md +150 -150
  70. package/{content → stacks/blazor-azure}/.morph/examples/multi-agent/README.md +309 -309
  71. package/{content → stacks/blazor-azure}/.morph/examples/multi-agent/contracts.cs +433 -433
  72. package/{content → stacks/blazor-azure}/.morph/examples/multi-agent/spec.md +479 -479
  73. package/{content → stacks/blazor-azure}/.morph/examples/multi-agent/tasks.md +185 -185
  74. package/{content → stacks/blazor-azure}/.morph/examples/state-v3.json +188 -188
  75. package/{content → stacks/blazor-azure}/.morph/features/.gitkeep +25 -25
  76. package/{content → stacks/blazor-azure}/.morph/hooks/README.md +12 -12
  77. package/{content → stacks/blazor-azure}/.morph/hooks/pre-commit-all.sh +48 -48
  78. package/{content → stacks/blazor-azure}/.morph/hooks/pre-commit-specs.sh +49 -49
  79. package/{content → stacks/blazor-azure}/.morph/hooks/pre-commit-tests.sh +60 -60
  80. package/{content → stacks/blazor-azure}/.morph/project.md +160 -160
  81. package/{content → stacks/blazor-azure}/.morph/schemas/agent.schema.json +296 -296
  82. package/{content → stacks/blazor-azure}/.morph/specs/.gitkeep +20 -20
  83. package/{content → stacks/blazor-azure}/.morph/standards/agent-teams-workflow.md +2 -2
  84. package/{content → stacks/blazor-azure}/.morph/standards/coding.md +377 -377
  85. package/{content → stacks/blazor-azure}/.morph/standards/fluent-ui-setup.md +590 -590
  86. package/{content → stacks/blazor-azure}/.morph/standards/migration-guide.md +514 -514
  87. package/{content → stacks/blazor-azure}/.morph/standards/passkeys-auth.md +423 -423
  88. package/{content → stacks/blazor-azure}/.morph/standards/vector-search-rag.md +536 -536
  89. package/{content → stacks/blazor-azure}/.morph/state.json +17 -17
  90. package/{content → stacks/blazor-azure}/.morph/templates/FluentDesignTheme.cs +149 -149
  91. package/{content → stacks/blazor-azure}/.morph/templates/MudTheme.cs +281 -281
  92. package/{content → stacks/blazor-azure}/.morph/templates/component.razor +239 -239
  93. package/{content → stacks/blazor-azure}/.morph/templates/contracts.cs +217 -217
  94. package/{content → stacks/blazor-azure}/.morph/templates/design-system.css +226 -226
  95. package/{content → stacks/blazor-azure}/.morph/templates/infra/.dockerignore.example +89 -89
  96. package/{content → stacks/blazor-azure}/.morph/templates/infra/Dockerfile.example +82 -82
  97. package/{content → stacks/blazor-azure}/.morph/templates/infra/README.md +286 -286
  98. package/{content → stacks/blazor-azure}/.morph/templates/infra/app-insights.bicep +63 -63
  99. package/{content → stacks/blazor-azure}/.morph/templates/infra/app-service.bicep +164 -164
  100. package/{content → stacks/blazor-azure}/.morph/templates/infra/azure-pipelines-deploy.yml +480 -480
  101. package/{content → stacks/blazor-azure}/.morph/templates/infra/container-app-env.bicep +49 -49
  102. package/{content → stacks/blazor-azure}/.morph/templates/infra/container-app.bicep +156 -156
  103. package/{content → stacks/blazor-azure}/.morph/templates/infra/deploy.ps1 +229 -229
  104. package/{content → stacks/blazor-azure}/.morph/templates/infra/deploy.sh +208 -208
  105. package/{content → stacks/blazor-azure}/.morph/templates/infra/key-vault.bicep +91 -91
  106. package/{content → stacks/blazor-azure}/.morph/templates/infra/main.bicep +189 -189
  107. package/{content → stacks/blazor-azure}/.morph/templates/infra/parameters.dev.json +29 -29
  108. package/{content → stacks/blazor-azure}/.morph/templates/infra/parameters.prod.json +29 -29
  109. package/{content → stacks/blazor-azure}/.morph/templates/infra/parameters.staging.json +29 -29
  110. package/{content → stacks/blazor-azure}/.morph/templates/infra/sql-database.bicep +103 -103
  111. package/{content → stacks/blazor-azure}/.morph/templates/infra/storage.bicep +106 -106
  112. package/{content → stacks/blazor-azure}/.morph/templates/integrations/asaas-client.cs +387 -387
  113. package/{content → stacks/blazor-azure}/.morph/templates/integrations/asaas-webhook.cs +351 -351
  114. package/{content → stacks/blazor-azure}/.morph/templates/integrations/azure-identity-config.cs +288 -288
  115. package/{content → stacks/blazor-azure}/.morph/templates/integrations/clerk-config.cs +258 -258
  116. package/{content → stacks/blazor-azure}/.morph/templates/job.cs +171 -171
  117. package/{content → stacks/blazor-azure}/.morph/templates/migration.cs +83 -83
  118. package/{content → stacks/blazor-azure}/.morph/templates/repository.cs +141 -141
  119. package/{content → stacks/blazor-azure}/.morph/templates/saas/subscription.cs +347 -347
  120. package/{content → stacks/blazor-azure}/.morph/templates/saas/tenant.cs +338 -338
  121. package/{content → stacks/blazor-azure}/.morph/templates/service.cs +139 -139
  122. package/{content → stacks/blazor-azure}/.morph/templates/sprint-status.yaml +68 -68
  123. package/{content → stacks/blazor-azure}/.morph/templates/story.md +143 -143
  124. package/{content → stacks/blazor-azure}/.morph/templates/test.cs +239 -239
  125. package/{content → stacks/blazor-azure}/.morph/templates/ui-design-system.md +286 -286
  126. package/{content → stacks/blazor-azure}/.morph/templates/ui-flows.md +336 -336
  127. package/{content → stacks/blazor-azure}/.morph/templates/ui-mockups.md +133 -133
  128. package/{content → stacks/blazor-azure}/.morph/test-infra/example.bicep +59 -59
  129. package/{content → stacks/blazor-azure}/README.md +79 -79
  130. package/stacks/nextjs-supabase/.claude/skills/level-2-domains/backend/dotnet-supabase.md +244 -0
  131. package/stacks/nextjs-supabase/.claude/skills/level-2-domains/frontend/nextjs-supabase.md +335 -0
  132. package/stacks/nextjs-supabase/.claude/skills/level-2-domains/infrastructure/easypanel-deployer.md +189 -0
  133. package/stacks/nextjs-supabase/.claude/skills/level-2-domains/integrations/supabase-expert.md +170 -0
  134. package/stacks/nextjs-supabase/.morph/config/agents.json +345 -0
  135. package/stacks/nextjs-supabase/.morph/config/config.template.json +92 -0
  136. package/stacks/nextjs-supabase/.morph/docs/easypanel-setup.md +169 -0
  137. package/stacks/nextjs-supabase/.morph/docs/supabase-mcp-setup.md +247 -0
  138. package/stacks/nextjs-supabase/.morph/examples/crud-nextjs-supabase/README.md +697 -0
  139. package/stacks/nextjs-supabase/.morph/examples/crud-nextjs-supabase/spec.md +85 -0
  140. package/stacks/nextjs-supabase/.morph/examples/crud-nextjs-supabase/tasks.md +86 -0
  141. package/stacks/nextjs-supabase/.morph/examples/saas-nextjs-supabase/README.md +498 -0
  142. package/stacks/nextjs-supabase/.morph/examples/saas-nextjs-supabase/decisions.md +121 -0
  143. package/stacks/nextjs-supabase/.morph/examples/saas-nextjs-supabase/spec.md +138 -0
  144. package/stacks/nextjs-supabase/.morph/examples/saas-nextjs-supabase/tasks.md +162 -0
  145. package/stacks/nextjs-supabase/.morph/project.md +168 -0
  146. package/stacks/nextjs-supabase/.morph/standards/easypanel-deploy.md +191 -0
  147. package/stacks/nextjs-supabase/.morph/standards/nextjs-patterns.md +193 -0
  148. package/stacks/nextjs-supabase/.morph/standards/supabase-auth.md +171 -0
  149. package/stacks/nextjs-supabase/.morph/standards/supabase-pgvector.md +164 -0
  150. package/stacks/nextjs-supabase/.morph/standards/supabase-rls.md +179 -0
  151. package/stacks/nextjs-supabase/.morph/standards/supabase-storage.md +148 -0
  152. package/stacks/nextjs-supabase/.morph/templates/contracts.cs +173 -0
  153. package/stacks/nextjs-supabase/.morph/templates/contracts.ts +168 -0
  154. package/stacks/nextjs-supabase/.morph/templates/decisions.md +115 -0
  155. package/stacks/nextjs-supabase/.morph/templates/dockerfile-api.dockerfile +38 -0
  156. package/stacks/nextjs-supabase/.morph/templates/dockerfile-web.dockerfile +48 -0
  157. package/stacks/nextjs-supabase/.morph/templates/proposal.md +145 -0
  158. package/stacks/nextjs-supabase/.morph/templates/recap.md +134 -0
  159. package/stacks/nextjs-supabase/.morph/templates/rls-policy.sql +57 -0
  160. package/stacks/nextjs-supabase/.morph/templates/spec.md +231 -0
  161. package/stacks/nextjs-supabase/.morph/templates/supabase-migration.sql +100 -0
  162. package/stacks/nextjs-supabase/.morph/templates/tasks.md +257 -0
  163. package/stacks/nextjs-supabase/CLAUDE.md +149 -0
  164. package/stacks/nextjs-supabase/README.md +112 -0
  165. /package/{detectors → src/lib/detectors}/structure-detector.js +0 -0
  166. /package/{content → stacks/blazor-azure}/.claude/commands/morph-apply.md +0 -0
  167. /package/{content → stacks/blazor-azure}/.claude/commands/morph-preflight.md +0 -0
  168. /package/{content → stacks/blazor-azure}/.claude/commands/morph-proposal.md +0 -0
  169. /package/{content → stacks/blazor-azure}/.claude/commands/morph-status.md +0 -0
  170. /package/{content → stacks/blazor-azure}/.claude/skills/level-0-meta/README.md +0 -0
  171. /package/{content → stacks/blazor-azure}/.claude/skills/level-0-meta/code-review.md +0 -0
  172. /package/{content → stacks/blazor-azure}/.claude/skills/level-0-meta/morph-checklist.md +0 -0
  173. /package/{content → stacks/blazor-azure}/.claude/skills/level-0-meta/simulation-checklist.md +0 -0
  174. /package/{content → stacks/blazor-azure}/.claude/skills/level-1-workflows/README.md +0 -0
  175. /package/{content → stacks/blazor-azure}/.claude/skills/level-1-workflows/morph-replicate.md +0 -0
  176. /package/{content → stacks/blazor-azure}/.claude/skills/level-1-workflows/phase-clarify.md +0 -0
  177. /package/{content → stacks/blazor-azure}/.claude/skills/level-1-workflows/phase-design.md +0 -0
  178. /package/{content → stacks/blazor-azure}/.claude/skills/level-1-workflows/phase-tasks.md +0 -0
  179. /package/{content → stacks/blazor-azure}/.claude/skills/level-1-workflows/phase-uiux.md +0 -0
  180. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/README.md +0 -0
  181. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/ai-agents/ai-system-architect.md +0 -0
  182. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/architecture/po-pm-advisor.md +0 -0
  183. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/architecture/standards-architect.md +0 -0
  184. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/backend/dotnet-senior.md +0 -0
  185. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/backend/ef-modeler.md +0 -0
  186. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/backend/hangfire-orchestrator.md +0 -0
  187. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/backend/ms-agent-expert.md +0 -0
  188. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/frontend/blazor-builder.md +0 -0
  189. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/frontend/nextjs-expert.md +0 -0
  190. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/frontend/ui-ux-designer.md +0 -0
  191. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/infrastructure/azure-architect.md +0 -0
  192. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/infrastructure/bicep-architect.md +0 -0
  193. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/infrastructure/container-specialist.md +0 -0
  194. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/infrastructure/devops-engineer.md +0 -0
  195. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/integrations/asaas-financial.md +0 -0
  196. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/integrations/azure-identity.md +0 -0
  197. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/integrations/clerk-auth.md +0 -0
  198. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/integrations/resend-email.md +0 -0
  199. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/quality/code-analyzer.md +0 -0
  200. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/quality/testing-specialist.md +0 -0
  201. /package/{content → stacks/blazor-azure}/.claude/skills/level-3-technologies/README.md +0 -0
  202. /package/{content → stacks/blazor-azure}/.claude/skills/level-4-patterns/README.md +0 -0
  203. /package/{content → stacks/blazor-azure}/.morph/config/config.template.json +0 -0
  204. /package/{content → stacks/blazor-azure}/.morph/docs/workflows/design-impl.md +0 -0
  205. /package/{content → stacks/blazor-azure}/.morph/docs/workflows/fast-track.md +0 -0
  206. /package/{content → stacks/blazor-azure}/.morph/docs/workflows/full-morph.md +0 -0
  207. /package/{content → stacks/blazor-azure}/.morph/docs/workflows/standard.md +0 -0
  208. /package/{content → stacks/blazor-azure}/.morph/docs/workflows/ui-refresh.md +0 -0
  209. /package/{content → stacks/blazor-azure}/.morph/examples/scheduled-reports/decisions.md +0 -0
  210. /package/{content → stacks/blazor-azure}/.morph/examples/scheduled-reports/proposal.md +0 -0
  211. /package/{content → stacks/blazor-azure}/.morph/examples/scheduled-reports/spec.md +0 -0
  212. /package/{content → stacks/blazor-azure}/.morph/hooks/pre-commit-agents.sh +0 -0
  213. /package/{content → stacks/blazor-azure}/.morph/hooks/task-completed.js +0 -0
  214. /package/{content → stacks/blazor-azure}/.morph/hooks/teammate-idle.js +0 -0
  215. /package/{content → stacks/blazor-azure}/.morph/schemas/tasks.schema.json +0 -0
  216. /package/{content → stacks/blazor-azure}/.morph/standards/agent-framework-blazor-ui.md +0 -0
  217. /package/{content → stacks/blazor-azure}/.morph/standards/agent-framework-production.md +0 -0
  218. /package/{content → stacks/blazor-azure}/.morph/standards/agent-framework-setup.md +0 -0
  219. /package/{content → stacks/blazor-azure}/.morph/standards/agent-framework-workflows.md +0 -0
  220. /package/{content → stacks/blazor-azure}/.morph/standards/architecture.md +0 -0
  221. /package/{content → stacks/blazor-azure}/.morph/standards/azure.md +0 -0
  222. /package/{content → stacks/blazor-azure}/.morph/standards/dotnet10-migration.md +0 -0
  223. /package/{content → stacks/blazor-azure}/.morph/templates/CONTEXT-FEATURE.md +0 -0
  224. /package/{content → stacks/blazor-azure}/.morph/templates/CONTEXT.md +0 -0
  225. /package/{content → stacks/blazor-azure}/.morph/templates/agent.cs +0 -0
  226. /package/{content → stacks/blazor-azure}/.morph/templates/clarify-questions.md +0 -0
  227. /package/{content → stacks/blazor-azure}/.morph/templates/contracts/Commands.cs +0 -0
  228. /package/{content → stacks/blazor-azure}/.morph/templates/contracts/Entities.cs +0 -0
  229. /package/{content → stacks/blazor-azure}/.morph/templates/contracts/Queries.cs +0 -0
  230. /package/{content → stacks/blazor-azure}/.morph/templates/contracts/README.md +0 -0
  231. /package/{content → stacks/blazor-azure}/.morph/templates/decisions.md +0 -0
  232. /package/{content → stacks/blazor-azure}/.morph/templates/infra/deploy-checklist.md +0 -0
  233. /package/{content → stacks/blazor-azure}/.morph/templates/proposal.md +0 -0
  234. /package/{content → stacks/blazor-azure}/.morph/templates/recap.md +0 -0
  235. /package/{content → stacks/blazor-azure}/.morph/templates/simulation.md +0 -0
  236. /package/{content → stacks/blazor-azure}/.morph/templates/spec.md +0 -0
  237. /package/{content → stacks/blazor-azure}/.morph/templates/state.template.json +0 -0
  238. /package/{content → stacks/blazor-azure}/.morph/templates/tasks.md +0 -0
  239. /package/{content → stacks/blazor-azure}/.morph/templates/ui-components.md +0 -0
  240. /package/{content → stacks/blazor-azure}/CLAUDE.md +0 -0
@@ -1,319 +1,319 @@
1
- # ==============================================================================
2
- # MORPH-SPEC - Production Pipeline
3
- # Deployment to Azure Container Apps with manual approval and always-on
4
- # ==============================================================================
5
-
6
- trigger:
7
- branches:
8
- include:
9
- - main
10
- - master
11
-
12
- pr: none # No PR builds for prod
13
-
14
- variables:
15
- - template: pipeline-variables.yml
16
- - name: environment
17
- value: 'prod'
18
- - name: resourceGroupName
19
- value: 'rg-$(APP_NAME)-prod'
20
- - name: containerAppName
21
- value: 'ca-$(APP_NAME)-prod'
22
- - name: hostingType
23
- value: 'containerapp'
24
- - name: parametersFile
25
- value: 'content/.morph/templates/infra/parameters.prod.json'
26
- - name: imageName
27
- value: '$(APP_NAME)'
28
- - name: imageFullName
29
- value: '$(containerRegistry)/$(imageName):$(imageTag)'
30
-
31
- stages:
32
- # ===========================================================================
33
- # STAGE 1: Build & Test
34
- # ===========================================================================
35
- - stage: Build
36
- displayName: 'Build & Test'
37
- jobs:
38
- - job: BuildJob
39
- displayName: 'Build .NET Application'
40
- pool:
41
- vmImage: 'ubuntu-latest'
42
- steps:
43
- - checkout: self
44
- fetchDepth: 1
45
-
46
- - template: templates/build-dotnet.yml
47
- parameters:
48
- dotnetVersion: $(dotnetVersion)
49
- buildConfiguration: $(buildConfiguration)
50
- runTests: true
51
- publishArtifact: false
52
-
53
- # ===========================================================================
54
- # STAGE 2: Security Scan
55
- # ===========================================================================
56
- - stage: SecurityScan
57
- displayName: 'Security Scan'
58
- dependsOn: Build
59
- condition: succeeded()
60
- jobs:
61
- - job: SecurityScanJob
62
- displayName: 'Run Security Scans'
63
- pool:
64
- vmImage: 'ubuntu-latest'
65
- steps:
66
- - checkout: self
67
- fetchDepth: 1
68
-
69
- - task: DotNetCoreCLI@2
70
- displayName: 'Restore packages'
71
- inputs:
72
- command: 'restore'
73
-
74
- - task: PowerShell@2
75
- displayName: 'Check for vulnerable packages'
76
- inputs:
77
- targetType: 'inline'
78
- script: |
79
- Write-Host "🔍 Scanning for vulnerable NuGet packages..."
80
- dotnet list package --vulnerable --include-transitive
81
-
82
- if ($LASTEXITCODE -ne 0) {
83
- Write-Warning "⚠️ Vulnerable packages found. Review before deploying to production."
84
- }
85
-
86
- # ===========================================================================
87
- # STAGE 3: Deploy Infrastructure
88
- # ===========================================================================
89
- - stage: DeployInfra
90
- displayName: 'Deploy Infrastructure'
91
- dependsOn: SecurityScan
92
- condition: succeeded()
93
- jobs:
94
- - deployment: DeployInfraJob
95
- displayName: 'Deploy Bicep Templates'
96
- pool:
97
- vmImage: 'ubuntu-latest'
98
- environment: 'production' # Requires manual approval in Azure DevOps
99
- strategy:
100
- runOnce:
101
- deploy:
102
- steps:
103
- - checkout: self
104
- fetchDepth: 1
105
-
106
- - template: templates/infra-deploy.yml
107
- parameters:
108
- azureSubscription: 'Azure-Prod-Connection'
109
- resourceGroupName: $(resourceGroupName)
110
- location: $(azureLocation)
111
- environment: $(environment)
112
- appName: $(APP_NAME)
113
- hostingType: $(hostingType)
114
- bicepTemplateFile: $(bicepTemplateFile)
115
- parametersFile: $(parametersFile)
116
-
117
- # ===========================================================================
118
- # STAGE 4: Build and Push Container
119
- # ===========================================================================
120
- - stage: BuildContainer
121
- displayName: 'Build Container'
122
- dependsOn: DeployInfra
123
- condition: succeeded()
124
- jobs:
125
- - job: BuildContainerJob
126
- displayName: 'Build and Push Docker Image'
127
- pool:
128
- vmImage: 'ubuntu-latest'
129
- steps:
130
- - checkout: self
131
- fetchDepth: 1
132
-
133
- - task: Docker@2
134
- displayName: 'Build and push container'
135
- inputs:
136
- containerRegistry: 'ACR-Connection'
137
- repository: '$(imageName)'
138
- command: 'buildAndPush'
139
- Dockerfile: '$(dockerfilePath)'
140
- tags: |
141
- $(imageTag)
142
- prod-latest
143
- $(Build.BuildId)
144
-
145
- - task: AzureCLI@2
146
- displayName: 'Scan image for vulnerabilities (Microsoft Defender)'
147
- continueOnError: false # Fail on vulnerabilities in prod
148
- inputs:
149
- azureSubscription: 'Azure-Prod-Connection'
150
- scriptType: 'bash'
151
- scriptLocation: 'inlineScript'
152
- inlineScript: |
153
- echo "🔍 Scanning image for vulnerabilities..."
154
- # Add your security scanning tool here (e.g., Trivy, Defender for Containers)
155
- # az acr scan --name $(ACR_NAME) --image $(imageName):$(imageTag)
156
-
157
- # ===========================================================================
158
- # STAGE 5: Deploy to Production (Blue-Green)
159
- # ===========================================================================
160
- - stage: DeployApp
161
- displayName: 'Deploy to Production'
162
- dependsOn: BuildContainer
163
- condition: succeeded()
164
- jobs:
165
- - deployment: DeployAppJob
166
- displayName: 'Deploy Container App'
167
- pool:
168
- vmImage: 'ubuntu-latest'
169
- environment: 'production' # Requires manual approval
170
- strategy:
171
- runOnce:
172
- deploy:
173
- steps:
174
- - checkout: self
175
- fetchDepth: 1
176
-
177
- # Create new revision
178
- - template: templates/deploy-container-app.yml
179
- parameters:
180
- azureSubscription: 'Azure-Prod-Connection'
181
- containerAppName: $(containerAppName)
182
- resourceGroupName: $(resourceGroupName)
183
- containerRegistry: $(containerRegistry)
184
- imageName: $(imageName)
185
- imageTag: $(imageTag)
186
- acrServiceConnection: 'ACR-Connection'
187
- healthCheckUrl: '/health'
188
- healthCheckTimeout: 300
189
-
190
- # Wait before activating
191
- - task: PowerShell@2
192
- displayName: 'Monitor new revision (5 min)'
193
- inputs:
194
- targetType: 'inline'
195
- script: |
196
- Write-Host "⏳ Monitoring new revision for 5 minutes before activating..."
197
- Start-Sleep -Seconds 300
198
- Write-Host "✅ Monitoring period complete"
199
-
200
- # ===========================================================================
201
- # STAGE 6: Smoke Tests in Production
202
- # ===========================================================================
203
- - stage: SmokeTests
204
- displayName: 'Production Smoke Tests'
205
- dependsOn: DeployApp
206
- condition: succeeded()
207
- jobs:
208
- - job: SmokeTestsJob
209
- displayName: 'Run Production Smoke Tests'
210
- pool:
211
- vmImage: 'ubuntu-latest'
212
- steps:
213
- - task: AzureCLI@2
214
- displayName: 'Get Container App URL'
215
- name: getUrl
216
- inputs:
217
- azureSubscription: 'Azure-Prod-Connection'
218
- scriptType: 'bash'
219
- scriptLocation: 'inlineScript'
220
- inlineScript: |
221
- FQDN=$(az containerapp show \
222
- --name $(containerAppName) \
223
- --resource-group $(resourceGroupName) \
224
- --query properties.configuration.ingress.fqdn -o tsv)
225
-
226
- APP_URL="https://$FQDN"
227
- echo "##vso[task.setvariable variable=appUrl]$APP_URL"
228
- echo "Production URL: $APP_URL"
229
-
230
- - task: PowerShell@2
231
- displayName: 'Critical smoke tests'
232
- inputs:
233
- targetType: 'inline'
234
- script: |
235
- $appUrl = "$(appUrl)"
236
-
237
- Write-Host "🧪 Running CRITICAL smoke tests in PRODUCTION"
238
- Write-Host "URL: $appUrl"
239
-
240
- $criticalEndpoints = @(
241
- @{Path="/health"; Description="Health Check"},
242
- @{Path="/health/ready"; Description="Readiness Check"},
243
- @{Path="/"; Description="Home Page"}
244
- )
245
-
246
- $failed = $false
247
- foreach ($endpoint in $criticalEndpoints) {
248
- $url = "$appUrl$($endpoint.Path)"
249
- try {
250
- $response = Invoke-WebRequest -Uri $url -UseBasicParsing -TimeoutSec 10
251
- Write-Host "✅ $($endpoint.Description): $($response.StatusCode)"
252
- }
253
- catch {
254
- Write-Error "❌ $($endpoint.Description) FAILED: $_"
255
- $failed = $true
256
- }
257
- }
258
-
259
- if ($failed) {
260
- Write-Error "❌ CRITICAL: Smoke tests failed in production!"
261
- exit 1
262
- }
263
-
264
- Write-Host "✅ All critical smoke tests passed!"
265
-
266
- - task: AzureCLI@2
267
- displayName: 'Monitor metrics'
268
- inputs:
269
- azureSubscription: 'Azure-Prod-Connection'
270
- scriptType: 'bash'
271
- scriptLocation: 'inlineScript'
272
- inlineScript: |
273
- echo "📊 Production Deployment Metrics:"
274
-
275
- # Get replica count
276
- REPLICAS=$(az containerapp revision list \
277
- --name $(containerAppName) \
278
- --resource-group $(resourceGroupName) \
279
- --query "[?properties.active].properties.replicas" -o tsv)
280
-
281
- echo "Active Replicas: $REPLICAS"
282
-
283
- # Show active revisions
284
- az containerapp revision list \
285
- --name $(containerAppName) \
286
- --resource-group $(resourceGroupName) \
287
- --query "[?properties.active].{Name:name, Traffic:properties.trafficWeight, Replicas:properties.replicas}" \
288
- --output table
289
-
290
- - task: AzureCLI@2
291
- displayName: 'Deployment summary'
292
- inputs:
293
- azureSubscription: 'Azure-Prod-Connection'
294
- scriptType: 'bash'
295
- scriptLocation: 'inlineScript'
296
- inlineScript: |
297
- echo "╔════════════════════════════════════════════════════════════════╗"
298
- echo "║ PRODUCTION DEPLOYMENT SUCCESSFUL ║"
299
- echo "╚════════════════════════════════════════════════════════════════╝"
300
- echo ""
301
- echo "🌐 Application URL: $(appUrl)"
302
- echo "📊 Environment: $(environment)"
303
- echo "🐳 Container Image: $(imageFullName)"
304
- echo "💰 Hosting: Container Apps (always-on) - ~$10-20/month"
305
- echo "📦 Resource Group: $(resourceGroupName)"
306
- echo "🏷️ Version: $(imageTag)"
307
- echo ""
308
- echo "⚠️ IMPORTANT:"
309
- echo " 1. Monitor Application Insights for errors"
310
- echo " 2. Watch for performance degradation"
311
- echo " 3. Have rollback plan ready"
312
- echo " 4. Monitor costs in Azure Portal"
313
- echo ""
314
- echo "🔄 Rollback command (if needed):"
315
- echo " az containerapp revision activate \\"
316
- echo " --name $(containerAppName) \\"
317
- echo " --resource-group $(resourceGroupName) \\"
318
- echo " --revision <PREVIOUS_REVISION_NAME>"
319
- echo ""
1
+ # ==============================================================================
2
+ # MORPH-SPEC - Production Pipeline
3
+ # Deployment to Azure Container Apps with manual approval and always-on
4
+ # ==============================================================================
5
+
6
+ trigger:
7
+ branches:
8
+ include:
9
+ - main
10
+ - master
11
+
12
+ pr: none # No PR builds for prod
13
+
14
+ variables:
15
+ - template: pipeline-variables.yml
16
+ - name: environment
17
+ value: 'prod'
18
+ - name: resourceGroupName
19
+ value: 'rg-$(APP_NAME)-prod'
20
+ - name: containerAppName
21
+ value: 'ca-$(APP_NAME)-prod'
22
+ - name: hostingType
23
+ value: 'containerapp'
24
+ - name: parametersFile
25
+ value: '.morph/templates/infra/parameters.prod.json'
26
+ - name: imageName
27
+ value: '$(APP_NAME)'
28
+ - name: imageFullName
29
+ value: '$(containerRegistry)/$(imageName):$(imageTag)'
30
+
31
+ stages:
32
+ # ===========================================================================
33
+ # STAGE 1: Build & Test
34
+ # ===========================================================================
35
+ - stage: Build
36
+ displayName: 'Build & Test'
37
+ jobs:
38
+ - job: BuildJob
39
+ displayName: 'Build .NET Application'
40
+ pool:
41
+ vmImage: 'ubuntu-latest'
42
+ steps:
43
+ - checkout: self
44
+ fetchDepth: 1
45
+
46
+ - template: templates/build-dotnet.yml
47
+ parameters:
48
+ dotnetVersion: $(dotnetVersion)
49
+ buildConfiguration: $(buildConfiguration)
50
+ runTests: true
51
+ publishArtifact: false
52
+
53
+ # ===========================================================================
54
+ # STAGE 2: Security Scan
55
+ # ===========================================================================
56
+ - stage: SecurityScan
57
+ displayName: 'Security Scan'
58
+ dependsOn: Build
59
+ condition: succeeded()
60
+ jobs:
61
+ - job: SecurityScanJob
62
+ displayName: 'Run Security Scans'
63
+ pool:
64
+ vmImage: 'ubuntu-latest'
65
+ steps:
66
+ - checkout: self
67
+ fetchDepth: 1
68
+
69
+ - task: DotNetCoreCLI@2
70
+ displayName: 'Restore packages'
71
+ inputs:
72
+ command: 'restore'
73
+
74
+ - task: PowerShell@2
75
+ displayName: 'Check for vulnerable packages'
76
+ inputs:
77
+ targetType: 'inline'
78
+ script: |
79
+ Write-Host "🔍 Scanning for vulnerable NuGet packages..."
80
+ dotnet list package --vulnerable --include-transitive
81
+
82
+ if ($LASTEXITCODE -ne 0) {
83
+ Write-Warning "⚠️ Vulnerable packages found. Review before deploying to production."
84
+ }
85
+
86
+ # ===========================================================================
87
+ # STAGE 3: Deploy Infrastructure
88
+ # ===========================================================================
89
+ - stage: DeployInfra
90
+ displayName: 'Deploy Infrastructure'
91
+ dependsOn: SecurityScan
92
+ condition: succeeded()
93
+ jobs:
94
+ - deployment: DeployInfraJob
95
+ displayName: 'Deploy Bicep Templates'
96
+ pool:
97
+ vmImage: 'ubuntu-latest'
98
+ environment: 'production' # Requires manual approval in Azure DevOps
99
+ strategy:
100
+ runOnce:
101
+ deploy:
102
+ steps:
103
+ - checkout: self
104
+ fetchDepth: 1
105
+
106
+ - template: templates/infra-deploy.yml
107
+ parameters:
108
+ azureSubscription: 'Azure-Prod-Connection'
109
+ resourceGroupName: $(resourceGroupName)
110
+ location: $(azureLocation)
111
+ environment: $(environment)
112
+ appName: $(APP_NAME)
113
+ hostingType: $(hostingType)
114
+ bicepTemplateFile: $(bicepTemplateFile)
115
+ parametersFile: $(parametersFile)
116
+
117
+ # ===========================================================================
118
+ # STAGE 4: Build and Push Container
119
+ # ===========================================================================
120
+ - stage: BuildContainer
121
+ displayName: 'Build Container'
122
+ dependsOn: DeployInfra
123
+ condition: succeeded()
124
+ jobs:
125
+ - job: BuildContainerJob
126
+ displayName: 'Build and Push Docker Image'
127
+ pool:
128
+ vmImage: 'ubuntu-latest'
129
+ steps:
130
+ - checkout: self
131
+ fetchDepth: 1
132
+
133
+ - task: Docker@2
134
+ displayName: 'Build and push container'
135
+ inputs:
136
+ containerRegistry: 'ACR-Connection'
137
+ repository: '$(imageName)'
138
+ command: 'buildAndPush'
139
+ Dockerfile: '$(dockerfilePath)'
140
+ tags: |
141
+ $(imageTag)
142
+ prod-latest
143
+ $(Build.BuildId)
144
+
145
+ - task: AzureCLI@2
146
+ displayName: 'Scan image for vulnerabilities (Microsoft Defender)'
147
+ continueOnError: false # Fail on vulnerabilities in prod
148
+ inputs:
149
+ azureSubscription: 'Azure-Prod-Connection'
150
+ scriptType: 'bash'
151
+ scriptLocation: 'inlineScript'
152
+ inlineScript: |
153
+ echo "🔍 Scanning image for vulnerabilities..."
154
+ # Add your security scanning tool here (e.g., Trivy, Defender for Containers)
155
+ # az acr scan --name $(ACR_NAME) --image $(imageName):$(imageTag)
156
+
157
+ # ===========================================================================
158
+ # STAGE 5: Deploy to Production (Blue-Green)
159
+ # ===========================================================================
160
+ - stage: DeployApp
161
+ displayName: 'Deploy to Production'
162
+ dependsOn: BuildContainer
163
+ condition: succeeded()
164
+ jobs:
165
+ - deployment: DeployAppJob
166
+ displayName: 'Deploy Container App'
167
+ pool:
168
+ vmImage: 'ubuntu-latest'
169
+ environment: 'production' # Requires manual approval
170
+ strategy:
171
+ runOnce:
172
+ deploy:
173
+ steps:
174
+ - checkout: self
175
+ fetchDepth: 1
176
+
177
+ # Create new revision
178
+ - template: templates/deploy-container-app.yml
179
+ parameters:
180
+ azureSubscription: 'Azure-Prod-Connection'
181
+ containerAppName: $(containerAppName)
182
+ resourceGroupName: $(resourceGroupName)
183
+ containerRegistry: $(containerRegistry)
184
+ imageName: $(imageName)
185
+ imageTag: $(imageTag)
186
+ acrServiceConnection: 'ACR-Connection'
187
+ healthCheckUrl: '/health'
188
+ healthCheckTimeout: 300
189
+
190
+ # Wait before activating
191
+ - task: PowerShell@2
192
+ displayName: 'Monitor new revision (5 min)'
193
+ inputs:
194
+ targetType: 'inline'
195
+ script: |
196
+ Write-Host "⏳ Monitoring new revision for 5 minutes before activating..."
197
+ Start-Sleep -Seconds 300
198
+ Write-Host "✅ Monitoring period complete"
199
+
200
+ # ===========================================================================
201
+ # STAGE 6: Smoke Tests in Production
202
+ # ===========================================================================
203
+ - stage: SmokeTests
204
+ displayName: 'Production Smoke Tests'
205
+ dependsOn: DeployApp
206
+ condition: succeeded()
207
+ jobs:
208
+ - job: SmokeTestsJob
209
+ displayName: 'Run Production Smoke Tests'
210
+ pool:
211
+ vmImage: 'ubuntu-latest'
212
+ steps:
213
+ - task: AzureCLI@2
214
+ displayName: 'Get Container App URL'
215
+ name: getUrl
216
+ inputs:
217
+ azureSubscription: 'Azure-Prod-Connection'
218
+ scriptType: 'bash'
219
+ scriptLocation: 'inlineScript'
220
+ inlineScript: |
221
+ FQDN=$(az containerapp show \
222
+ --name $(containerAppName) \
223
+ --resource-group $(resourceGroupName) \
224
+ --query properties.configuration.ingress.fqdn -o tsv)
225
+
226
+ APP_URL="https://$FQDN"
227
+ echo "##vso[task.setvariable variable=appUrl]$APP_URL"
228
+ echo "Production URL: $APP_URL"
229
+
230
+ - task: PowerShell@2
231
+ displayName: 'Critical smoke tests'
232
+ inputs:
233
+ targetType: 'inline'
234
+ script: |
235
+ $appUrl = "$(appUrl)"
236
+
237
+ Write-Host "🧪 Running CRITICAL smoke tests in PRODUCTION"
238
+ Write-Host "URL: $appUrl"
239
+
240
+ $criticalEndpoints = @(
241
+ @{Path="/health"; Description="Health Check"},
242
+ @{Path="/health/ready"; Description="Readiness Check"},
243
+ @{Path="/"; Description="Home Page"}
244
+ )
245
+
246
+ $failed = $false
247
+ foreach ($endpoint in $criticalEndpoints) {
248
+ $url = "$appUrl$($endpoint.Path)"
249
+ try {
250
+ $response = Invoke-WebRequest -Uri $url -UseBasicParsing -TimeoutSec 10
251
+ Write-Host "✅ $($endpoint.Description): $($response.StatusCode)"
252
+ }
253
+ catch {
254
+ Write-Error "❌ $($endpoint.Description) FAILED: $_"
255
+ $failed = $true
256
+ }
257
+ }
258
+
259
+ if ($failed) {
260
+ Write-Error "❌ CRITICAL: Smoke tests failed in production!"
261
+ exit 1
262
+ }
263
+
264
+ Write-Host "✅ All critical smoke tests passed!"
265
+
266
+ - task: AzureCLI@2
267
+ displayName: 'Monitor metrics'
268
+ inputs:
269
+ azureSubscription: 'Azure-Prod-Connection'
270
+ scriptType: 'bash'
271
+ scriptLocation: 'inlineScript'
272
+ inlineScript: |
273
+ echo "📊 Production Deployment Metrics:"
274
+
275
+ # Get replica count
276
+ REPLICAS=$(az containerapp revision list \
277
+ --name $(containerAppName) \
278
+ --resource-group $(resourceGroupName) \
279
+ --query "[?properties.active].properties.replicas" -o tsv)
280
+
281
+ echo "Active Replicas: $REPLICAS"
282
+
283
+ # Show active revisions
284
+ az containerapp revision list \
285
+ --name $(containerAppName) \
286
+ --resource-group $(resourceGroupName) \
287
+ --query "[?properties.active].{Name:name, Traffic:properties.trafficWeight, Replicas:properties.replicas}" \
288
+ --output table
289
+
290
+ - task: AzureCLI@2
291
+ displayName: 'Deployment summary'
292
+ inputs:
293
+ azureSubscription: 'Azure-Prod-Connection'
294
+ scriptType: 'bash'
295
+ scriptLocation: 'inlineScript'
296
+ inlineScript: |
297
+ echo "╔════════════════════════════════════════════════════════════════╗"
298
+ echo "║ PRODUCTION DEPLOYMENT SUCCESSFUL ║"
299
+ echo "╚════════════════════════════════════════════════════════════════╝"
300
+ echo ""
301
+ echo "🌐 Application URL: $(appUrl)"
302
+ echo "📊 Environment: $(environment)"
303
+ echo "🐳 Container Image: $(imageFullName)"
304
+ echo "💰 Hosting: Container Apps (always-on) - ~$10-20/month"
305
+ echo "📦 Resource Group: $(resourceGroupName)"
306
+ echo "🏷️ Version: $(imageTag)"
307
+ echo ""
308
+ echo "⚠️ IMPORTANT:"
309
+ echo " 1. Monitor Application Insights for errors"
310
+ echo " 2. Watch for performance degradation"
311
+ echo " 3. Have rollback plan ready"
312
+ echo " 4. Monitor costs in Azure Portal"
313
+ echo ""
314
+ echo "🔄 Rollback command (if needed):"
315
+ echo " az containerapp revision activate \\"
316
+ echo " --name $(containerAppName) \\"
317
+ echo " --resource-group $(resourceGroupName) \\"
318
+ echo " --revision <PREVIOUS_REVISION_NAME>"
319
+ echo ""