@polymorphism-tech/morph-spec 2.4.0 → 3.0.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 (218) hide show
  1. package/CLAUDE.md +158 -26
  2. package/LICENSE +72 -72
  3. package/bin/detect-agents.js +225 -225
  4. package/bin/morph-spec.js +8 -0
  5. package/bin/render-template.js +302 -302
  6. package/bin/semantic-detect-agents.js +246 -246
  7. package/bin/validate-agents-skills.js +251 -251
  8. package/bin/validate-agents.js +69 -69
  9. package/bin/validate-phase.js +263 -263
  10. package/content/.azure/README.md +293 -293
  11. package/content/.azure/docs/azure-devops-setup.md +454 -454
  12. package/content/.azure/docs/branch-strategy.md +398 -398
  13. package/content/.azure/docs/local-development.md +515 -515
  14. package/content/.azure/pipelines/pipeline-variables.yml +34 -34
  15. package/content/.azure/pipelines/prod-pipeline.yml +319 -319
  16. package/content/.azure/pipelines/staging-pipeline.yml +234 -234
  17. package/content/.azure/pipelines/templates/build-dotnet.yml +75 -75
  18. package/content/.azure/pipelines/templates/deploy-app-service.yml +94 -94
  19. package/content/.azure/pipelines/templates/deploy-container-app.yml +120 -120
  20. package/content/.azure/pipelines/templates/infra-deploy.yml +90 -90
  21. package/content/.claude/commands/morph-archive.md +79 -79
  22. package/content/.claude/commands/morph-deploy.md +529 -0
  23. package/content/.claude/commands/morph-infra.md +209 -209
  24. package/content/.claude/commands/morph-preflight.md +227 -227
  25. package/content/.claude/commands/morph-troubleshoot.md +122 -122
  26. package/content/.claude/settings.local.json +15 -15
  27. package/content/.claude/skills/infra/azure-deploy-specialist.md +699 -0
  28. package/content/.claude/skills/level-0-meta/README.md +7 -0
  29. package/content/.claude/skills/{checklists → level-0-meta}/morph-checklist.md +117 -117
  30. package/content/.claude/skills/level-1-workflows/README.md +7 -0
  31. package/content/.claude/skills/{workflows → level-1-workflows}/morph-replicate.md +213 -213
  32. package/content/.claude/skills/{workflows → level-1-workflows}/phase-clarify.md +131 -131
  33. package/content/.claude/skills/{workflows → level-1-workflows}/phase-design.md +213 -205
  34. package/content/.claude/skills/{workflows → level-1-workflows}/phase-setup.md +106 -92
  35. package/content/.claude/skills/{workflows → level-1-workflows}/phase-tasks.md +164 -164
  36. package/content/.claude/skills/{workflows → level-1-workflows}/phase-uiux.md +169 -138
  37. package/content/.claude/skills/level-2-domains/README.md +14 -0
  38. package/content/.claude/skills/{specialists → level-2-domains/quality}/testing-specialist.md +126 -126
  39. package/content/.claude/skills/level-3-technologies/README.md +7 -0
  40. package/content/.claude/skills/level-4-patterns/README.md +7 -0
  41. package/content/.claude/skills/specialists/prompt-engineer.md +189 -0
  42. package/content/.claude/skills/specialists/seo-growth-hacker.md +320 -0
  43. package/content/.morph/.morphversion +5 -5
  44. package/content/.morph/archive/.gitkeep +25 -25
  45. package/content/.morph/config/agents.json +742 -358
  46. package/content/.morph/config/config.template.json +33 -0
  47. package/content/.morph/docs/STORY-DRIVEN-DEVELOPMENT.md +392 -392
  48. package/content/.morph/docs/workflows/enforcement-pipeline.md +668 -0
  49. package/content/.morph/examples/api-nextjs/README.md +241 -241
  50. package/content/.morph/examples/api-nextjs/contracts.ts +307 -307
  51. package/content/.morph/examples/api-nextjs/spec.md +399 -399
  52. package/content/.morph/examples/api-nextjs/tasks.md +168 -168
  53. package/content/.morph/examples/micro-saas/README.md +125 -125
  54. package/content/.morph/examples/micro-saas/contracts.cs +358 -358
  55. package/content/.morph/examples/micro-saas/decisions.md +246 -246
  56. package/content/.morph/examples/micro-saas/spec.md +236 -236
  57. package/content/.morph/examples/micro-saas/tasks.md +150 -150
  58. package/content/.morph/examples/multi-agent/README.md +309 -309
  59. package/content/.morph/examples/multi-agent/contracts.cs +433 -433
  60. package/content/.morph/examples/multi-agent/spec.md +479 -479
  61. package/content/.morph/examples/multi-agent/tasks.md +185 -185
  62. package/content/.morph/examples/scheduled-reports/decisions.md +158 -158
  63. package/content/.morph/examples/scheduled-reports/proposal.md +95 -95
  64. package/content/.morph/examples/scheduled-reports/spec.md +267 -267
  65. package/content/.morph/examples/state-v3.json +188 -188
  66. package/content/.morph/features/.gitkeep +25 -25
  67. package/content/.morph/hooks/README.md +158 -0
  68. package/content/.morph/hooks/pre-commit-all.sh +48 -48
  69. package/content/.morph/hooks/pre-commit-specs.sh +49 -49
  70. package/content/.morph/hooks/pre-commit-tests.sh +60 -60
  71. package/content/.morph/hooks/task-completed.js +73 -0
  72. package/content/.morph/hooks/teammate-idle.js +68 -0
  73. package/content/.morph/project.md +160 -160
  74. package/content/.morph/schemas/agent.schema.json +296 -296
  75. package/content/.morph/schemas/tasks.schema.json +220 -220
  76. package/content/.morph/specs/.gitkeep +20 -20
  77. package/content/.morph/standards/agent-teams-workflow.md +474 -0
  78. package/content/.morph/standards/coding.md +377 -377
  79. package/content/.morph/standards/fluent-ui-setup.md +590 -590
  80. package/content/.morph/standards/migration-guide.md +514 -514
  81. package/content/.morph/standards/passkeys-auth.md +423 -423
  82. package/content/.morph/standards/vector-search-rag.md +536 -536
  83. package/content/.morph/state.json +17 -17
  84. package/content/.morph/templates/CONTEXT-FEATURE.md +276 -0
  85. package/content/.morph/templates/CONTEXT.md +170 -0
  86. package/content/.morph/templates/FluentDesignTheme.cs +149 -149
  87. package/content/.morph/templates/MudTheme.cs +281 -281
  88. package/content/.morph/templates/clarify-questions.md +159 -159
  89. package/content/.morph/templates/component.razor +239 -239
  90. package/content/.morph/templates/contracts/Commands.cs +74 -74
  91. package/content/.morph/templates/contracts/Entities.cs +25 -25
  92. package/content/.morph/templates/contracts/Queries.cs +74 -74
  93. package/content/.morph/templates/contracts/README.md +74 -74
  94. package/content/.morph/templates/contracts.cs +217 -217
  95. package/content/.morph/templates/design-system.css +226 -226
  96. package/content/.morph/templates/infra/.dockerignore.example +89 -89
  97. package/content/.morph/templates/infra/Dockerfile.example +82 -82
  98. package/content/.morph/templates/infra/README.md +286 -286
  99. package/content/.morph/templates/infra/app-insights.bicep +63 -63
  100. package/content/.morph/templates/infra/app-service.bicep +164 -164
  101. package/content/.morph/templates/infra/azure-pipelines-deploy.yml +480 -0
  102. package/content/.morph/templates/infra/container-app-env.bicep +49 -49
  103. package/content/.morph/templates/infra/container-app.bicep +156 -156
  104. package/content/.morph/templates/infra/deploy-checklist.md +426 -426
  105. package/content/.morph/templates/infra/deploy.ps1 +229 -229
  106. package/content/.morph/templates/infra/deploy.sh +208 -208
  107. package/content/.morph/templates/infra/key-vault.bicep +91 -91
  108. package/content/.morph/templates/infra/main.bicep +189 -189
  109. package/content/.morph/templates/infra/parameters.dev.json +29 -29
  110. package/content/.morph/templates/infra/parameters.prod.json +29 -29
  111. package/content/.morph/templates/infra/parameters.staging.json +29 -29
  112. package/content/.morph/templates/infra/sql-database.bicep +103 -103
  113. package/content/.morph/templates/infra/storage.bicep +106 -106
  114. package/content/.morph/templates/integrations/asaas-client.cs +387 -387
  115. package/content/.morph/templates/integrations/asaas-webhook.cs +351 -351
  116. package/content/.morph/templates/integrations/azure-identity-config.cs +288 -288
  117. package/content/.morph/templates/integrations/clerk-config.cs +258 -258
  118. package/content/.morph/templates/job.cs +171 -171
  119. package/content/.morph/templates/migration.cs +83 -83
  120. package/content/.morph/templates/repository.cs +141 -141
  121. package/content/.morph/templates/saas/subscription.cs +347 -347
  122. package/content/.morph/templates/saas/tenant.cs +338 -338
  123. package/content/.morph/templates/service.cs +139 -139
  124. package/content/.morph/templates/sprint-status.yaml +68 -68
  125. package/content/.morph/templates/story.md +143 -143
  126. package/content/.morph/templates/test.cs +239 -239
  127. package/content/.morph/templates/ui-design-system.md +286 -286
  128. package/content/.morph/templates/ui-flows.md +336 -336
  129. package/content/.morph/templates/ui-mockups.md +133 -133
  130. package/content/.morph/test-infra/example.bicep +59 -59
  131. package/content/README.md +79 -79
  132. package/detectors/config-detector.js +223 -223
  133. package/detectors/conversation-analyzer.js +163 -163
  134. package/detectors/index.js +84 -84
  135. package/detectors/standards-generator.js +275 -275
  136. package/docs/api/fonts/Source-Sans-Pro/sourcesanspro-light-webfont.svg +977 -977
  137. package/docs/api/fonts/Source-Sans-Pro/sourcesanspro-regular-webfont.svg +1048 -1048
  138. package/docs/api/scripts/collapse.js +38 -38
  139. package/docs/api/scripts/commonNav.js +28 -28
  140. package/docs/api/scripts/linenumber.js +25 -25
  141. package/docs/api/scripts/nav.js +12 -12
  142. package/docs/api/scripts/polyfill.js +3 -3
  143. package/docs/api/scripts/prettify/Apache-License-2.0.txt +202 -202
  144. package/docs/api/scripts/prettify/lang-css.js +2 -2
  145. package/docs/api/scripts/prettify/prettify.js +28 -28
  146. package/docs/api/scripts/search.js +98 -98
  147. package/docs/api/styles/jsdoc.css +776 -776
  148. package/docs/api/styles/prettify.css +80 -80
  149. package/docs/examples.md +328 -328
  150. package/docs/templates.md +418 -418
  151. package/package.json +1 -1
  152. package/scripts/postinstall.js +132 -132
  153. package/src/commands/advance-phase.js +83 -0
  154. package/src/commands/analyze-blazor-concurrency.js +193 -193
  155. package/src/commands/create-story.js +351 -351
  156. package/src/commands/deploy.js +780 -0
  157. package/src/commands/detect-agents.js +34 -6
  158. package/src/commands/detect.js +104 -104
  159. package/src/commands/generate-context.js +40 -0
  160. package/src/commands/generate.js +149 -149
  161. package/src/commands/lint-fluent.js +352 -352
  162. package/src/commands/rollback-phase.js +185 -185
  163. package/src/commands/session-summary.js +291 -291
  164. package/src/commands/shard-spec.js +224 -224
  165. package/src/commands/sprint-status.js +250 -250
  166. package/src/commands/state.js +333 -333
  167. package/src/commands/sync.js +167 -167
  168. package/src/commands/troubleshoot.js +222 -222
  169. package/src/commands/validate-blazor-state.js +210 -210
  170. package/src/commands/validate-blazor.js +156 -156
  171. package/src/commands/validate-css.js +84 -84
  172. package/src/commands/validate-phase.js +221 -221
  173. package/src/lib/blazor-concurrency-analyzer.js +288 -288
  174. package/src/lib/blazor-state-validator.js +291 -291
  175. package/src/lib/blazor-validator.js +374 -374
  176. package/src/lib/context-generator.js +513 -0
  177. package/src/lib/css-validator.js +352 -352
  178. package/src/lib/design-system-detector.js +187 -0
  179. package/src/lib/design-system-generator.js +298 -298
  180. package/src/lib/design-system-scaffolder.js +299 -0
  181. package/src/lib/hook-executor.js +256 -0
  182. package/src/lib/learning-system.js +520 -520
  183. package/src/lib/mockup-generator.js +366 -366
  184. package/src/lib/spec-validator.js +258 -0
  185. package/src/lib/standards-context-injector.js +287 -0
  186. package/src/lib/team-orchestrator.js +322 -0
  187. package/src/lib/troubleshoot-grep.js +194 -194
  188. package/src/lib/troubleshoot-index.js +144 -144
  189. package/src/lib/ui-detector.js +350 -350
  190. package/src/lib/validation-runner.js +65 -13
  191. package/src/lib/validators/architecture-validator.js +387 -387
  192. package/src/lib/validators/design-system-validator.js +231 -0
  193. package/src/lib/validators/package-validator.js +360 -360
  194. package/src/lib/validators/ui-contrast-validator.js +422 -422
  195. package/src/utils/file-copier.js +9 -1
  196. package/src/utils/logger.js +32 -32
  197. package/src/utils/version-checker.js +175 -175
  198. /package/content/.claude/skills/{checklists → level-0-meta}/code-review.md +0 -0
  199. /package/content/.claude/skills/{checklists → level-0-meta}/simulation-checklist.md +0 -0
  200. /package/content/.claude/skills/{specialists → level-2-domains/ai-agents}/ai-system-architect.md +0 -0
  201. /package/content/.claude/skills/{specialists → level-2-domains/architecture}/po-pm-advisor.md +0 -0
  202. /package/content/.claude/skills/{specialists → level-2-domains/architecture}/standards-architect.md +0 -0
  203. /package/content/.claude/skills/{specialists → level-2-domains/backend}/dotnet-senior.md +0 -0
  204. /package/content/.claude/skills/{specialists → level-2-domains/backend}/ef-modeler.md +0 -0
  205. /package/content/.claude/skills/{specialists → level-2-domains/backend}/hangfire-orchestrator.md +0 -0
  206. /package/content/.claude/skills/{specialists → level-2-domains/backend}/ms-agent-expert.md +0 -0
  207. /package/content/.claude/skills/{stacks/dotnet-blazor.md → level-2-domains/frontend/blazor-builder.md} +0 -0
  208. /package/content/.claude/skills/{stacks/dotnet-nextjs.md → level-2-domains/frontend/nextjs-expert.md} +0 -0
  209. /package/content/.claude/skills/{specialists → level-2-domains/frontend}/ui-ux-designer.md +0 -0
  210. /package/content/.claude/skills/{specialists → level-2-domains/infrastructure}/azure-architect.md +0 -0
  211. /package/content/.claude/skills/{infra → level-2-domains/infrastructure}/bicep-architect.md +0 -0
  212. /package/content/.claude/skills/{infra → level-2-domains/infrastructure}/container-specialist.md +0 -0
  213. /package/content/.claude/skills/{infra → level-2-domains/infrastructure}/devops-engineer.md +0 -0
  214. /package/content/.claude/skills/{integrations → level-2-domains/integrations}/asaas-financial.md +0 -0
  215. /package/content/.claude/skills/{integrations → level-2-domains/integrations}/azure-identity.md +0 -0
  216. /package/content/.claude/skills/{integrations → level-2-domains/integrations}/clerk-auth.md +0 -0
  217. /package/content/.claude/skills/{integrations → level-2-domains/integrations}/resend-email.md +0 -0
  218. /package/content/.claude/skills/{specialists → level-2-domains/quality}/code-analyzer.md +0 -0
@@ -1,514 +1,514 @@
1
- # Guia de Migração: App Service → Container Apps
2
-
3
- > **MORPH-SPEC Framework**
4
- > Quando seu projeto crescer além do Free Tier
5
-
6
- ---
7
-
8
- ## 📊 Quando Migrar?
9
-
10
- ### Sinais de que é hora de migrar
11
-
12
- | Indicador | Descrição | Ação |
13
- |-----------|-----------|------|
14
- | 🔴 **CPU Quota Esgotada** | Erro `CPU quota exceeded` frequente | Migrar imediatamente |
15
- | 🔴 **Cold Starts Críticos** | App dormindo afeta UX/negócio | Avaliar impacto → Migrar |
16
- | 🟡 **Tráfego Crescente** | >500 req/dia consistente | Planejar migração |
17
- | 🟡 **SSL Customizado** | Necessita domínio próprio com SSL | Avaliar alternativas |
18
- | 🟢 **Background Jobs** | Precisa rodar Hangfire 24/7 | Migrar ou separar worker |
19
-
20
- ### Matriz de Decisão
21
-
22
- ```
23
- Tráfego < 100 req/dia → App Service Free F1
24
- Tráfego 100-1000 req/dia → App Service B1 ($13/mês)
25
- Tráfego > 1000 req/dia → Container Apps (~$5-15/mês)
26
- Background Jobs 24/7 → Container Apps com minReplicas: 1
27
- SSL Customizado → Container Apps ou App Service B1+
28
- ```
29
-
30
- ---
31
-
32
- ## 🗺️ Roadmap de Migração
33
-
34
- ### Fase 1: Preparação (1-2 dias)
35
-
36
- #### 1.1. Auditoria do Código
37
-
38
- **Checklist:**
39
- - [ ] App usa configuração via `IConfiguration`
40
- - [ ] Não há paths hardcoded (usar `IWebHostEnvironment`)
41
- - [ ] Logging para stdout/stderr (não arquivos locais)
42
- - [ ] Health checks implementados (`/health`, `/health/ready`)
43
- - [ ] Secrets em Key Vault (não `appsettings.json`)
44
-
45
- **Verificar:**
46
- ```csharp
47
- // ✅ BOM: Configuração via IConfiguration
48
- var connectionString = configuration["ConnectionStrings:Default"];
49
-
50
- // ❌ RUIM: Hardcoded
51
- var path = "C:\\Data\\uploads";
52
-
53
- // ✅ BOM: Usar IWebHostEnvironment
54
- var path = Path.Combine(webHostEnv.ContentRootPath, "uploads");
55
- ```
56
-
57
- #### 1.2. Criar Dockerfile
58
-
59
- ```dockerfile
60
- # ============================================
61
- # MORPH-SPEC - .NET 8 Blazor Container
62
- # ============================================
63
-
64
- FROM mcr.microsoft.com/dotnet/aspnet:8.0 AS base
65
- WORKDIR /app
66
- EXPOSE 8080
67
- EXPOSE 8081
68
-
69
- # Create non-root user
70
- RUN adduser --disabled-password --gecos '' appuser && chown -R appuser /app
71
- USER appuser
72
-
73
- FROM mcr.microsoft.com/dotnet/sdk:8.0 AS build
74
- WORKDIR /src
75
- COPY ["YourApp/YourApp.csproj", "YourApp/"]
76
- RUN dotnet restore "YourApp/YourApp.csproj"
77
- COPY . .
78
- WORKDIR "/src/YourApp"
79
- RUN dotnet build "YourApp.csproj" -c Release -o /app/build
80
-
81
- FROM build AS publish
82
- RUN dotnet publish "YourApp.csproj" -c Release -o /app/publish /p:UseAppHost=false
83
-
84
- FROM base AS final
85
- WORKDIR /app
86
- COPY --from=publish /app/publish .
87
-
88
- # Health check
89
- HEALTHCHECK --interval=30s --timeout=3s --start-period=5s --retries=3 \
90
- CMD curl -f http://localhost:8080/health || exit 1
91
-
92
- ENTRYPOINT ["dotnet", "YourApp.dll"]
93
- ```
94
-
95
- #### 1.3. Implementar Health Checks
96
-
97
- ```csharp
98
- // Program.cs
99
- builder.Services.AddHealthChecks()
100
- .AddSqlServer(
101
- connectionString: builder.Configuration["ConnectionStrings:Default"],
102
- name: "sql",
103
- timeout: TimeSpan.FromSeconds(3))
104
- .AddCheck("self", () => HealthCheckResult.Healthy());
105
-
106
- var app = builder.Build();
107
-
108
- app.MapHealthChecks("/health", new HealthCheckOptions
109
- {
110
- Predicate = _ => true,
111
- ResponseWriter = UIResponseWriter.WriteHealthCheckUIResponse
112
- });
113
-
114
- app.MapHealthChecks("/health/ready", new HealthCheckOptions
115
- {
116
- Predicate = check => check.Tags.Contains("ready")
117
- });
118
-
119
- app.MapHealthChecks("/health/live", new HealthCheckOptions
120
- {
121
- Predicate = _ => false
122
- });
123
- ```
124
-
125
- ---
126
-
127
- ### Fase 2: Setup Azure Container Registry (ACR)
128
-
129
- #### 2.1. Criar ACR via Bicep
130
-
131
- ```bicep
132
- // acr.bicep
133
- resource acr 'Microsoft.ContainerRegistry/registries@2023-01-01-preview' = {
134
- name: 'acr${uniqueString(resourceGroup().id)}'
135
- location: location
136
- sku: {
137
- name: 'Basic' // $5/mês
138
- }
139
- properties: {
140
- adminUserEnabled: false // Usar Managed Identity
141
- }
142
- }
143
- ```
144
-
145
- #### 2.2. Deploy ACR
146
-
147
- ```bash
148
- # Via Azure CLI
149
- az deployment group create \
150
- --resource-group rg-myapp-dev \
151
- --template-file acr.bicep
152
-
153
- # Obter ACR login server
154
- ACR_NAME=$(az acr list -g rg-myapp-dev --query "[0].name" -o tsv)
155
- ACR_LOGIN_SERVER=$(az acr show -n $ACR_NAME --query loginServer -o tsv)
156
- ```
157
-
158
- #### 2.3. Build e Push Imagem
159
-
160
- **Via Azure DevOps Pipeline:**
161
- ```yaml
162
- # staging-pipeline.yml
163
- - task: Docker@2
164
- displayName: 'Build and push container'
165
- inputs:
166
- containerRegistry: 'ACR-Connection'
167
- repository: 'myapp'
168
- command: 'buildAndPush'
169
- Dockerfile: 'Dockerfile'
170
- tags: |
171
- v1.0.0
172
- staging-latest
173
- ```
174
-
175
- **Via Azure CLI (local):**
176
- ```bash
177
- # Build local e push para ACR
178
- az acr build \
179
- --registry $ACR_NAME \
180
- --image myapp:latest \
181
- --image myapp:v1.0.0 \
182
- --file Dockerfile \
183
- .
184
- ```
185
-
186
- ---
187
-
188
- ### Fase 3: Deploy Container Apps
189
-
190
- #### 3.1. Atualizar Parameters
191
-
192
- **parameters.dev.json**
193
- ```json
194
- {
195
- "hostingType": {
196
- "value": "containerapp"
197
- },
198
- "containerImage": {
199
- "value": "acrXXXX.azurecr.io/myapp:latest"
200
- }
201
- }
202
- ```
203
-
204
- #### 3.2. Deploy Infraestrutura
205
-
206
- **Via Azure DevOps Pipeline:**
207
- ```yaml
208
- # Use o template pronto
209
- - template: templates/infra-deploy.yml
210
- parameters:
211
- azureSubscription: 'Azure-Staging-Connection'
212
- resourceGroupName: 'rg-myapp-staging'
213
- bicepTemplateFile: 'content/.morph/templates/infra/main.bicep'
214
- parametersFile: 'content/.morph/templates/infra/parameters.staging.json'
215
- ```
216
-
217
- **Via Azure CLI (local):**
218
- ```bash
219
- # Deploy via Azure CLI
220
- az deployment group create \
221
- --resource-group rg-myapp-dev \
222
- --template-file main.bicep \
223
- --parameters @parameters.dev.json
224
- ```
225
-
226
- #### 3.3. Configurar Managed Identity para ACR
227
-
228
- ```bash
229
- # Obter Container App Managed Identity
230
- CA_PRINCIPAL_ID=$(az containerapp show \
231
- -n ca-myapp-dev \
232
- -g rg-myapp-dev \
233
- --query identity.principalId -o tsv)
234
-
235
- # Dar permissão AcrPull
236
- az role assignment create \
237
- --assignee $CA_PRINCIPAL_ID \
238
- --role AcrPull \
239
- --scope $(az acr show -n $ACR_NAME --query id -o tsv)
240
- ```
241
-
242
- ---
243
-
244
- ### Fase 4: Teste e Validação
245
-
246
- #### 4.1. Smoke Tests
247
-
248
- ```bash
249
- # Obter URL do Container App
250
- CA_URL=$(az containerapp show \
251
- -n ca-myapp-dev \
252
- -g rg-myapp-dev \
253
- --query properties.configuration.ingress.fqdn -o tsv)
254
-
255
- # Testar health checks
256
- curl https://$CA_URL/health
257
- curl https://$CA_URL/health/ready
258
-
259
- # Testar funcionalidade principal
260
- curl https://$CA_URL/api/status
261
- ```
262
-
263
- #### 4.2. Load Test (opcional)
264
-
265
- ```bash
266
- # Instalar hey (HTTP load generator)
267
- go install github.com/rakyll/hey@latest
268
-
269
- # Simular carga
270
- hey -z 30s -c 10 https://$CA_URL/
271
- ```
272
-
273
- #### 4.3. Validar Auto-Scaling
274
-
275
- ```bash
276
- # Gerar carga
277
- hey -z 2m -c 50 https://$CA_URL/
278
-
279
- # Monitorar réplicas
280
- watch -n 2 "az containerapp revision list \
281
- -n ca-myapp-dev \
282
- -g rg-myapp-dev \
283
- --query '[0].properties.replicas' -o tsv"
284
- ```
285
-
286
- ---
287
-
288
- ### Fase 5: Cutover (Mudança de DNS)
289
-
290
- #### 5.1. Estratégia Blue-Green
291
-
292
- ```bash
293
- # Manter App Service rodando (blue)
294
- # Container Apps já rodando (green)
295
-
296
- # Testar green profundamente
297
- # Quando confiante, mudar DNS para green
298
- # Aguardar 24h
299
- # Se OK, desligar blue (App Service)
300
- ```
301
-
302
- #### 5.2. Atualizar DNS
303
-
304
- Se usando domínio customizado:
305
-
306
- ```
307
- # Old (App Service)
308
- myapp.azurewebsites.net → CNAME → myapp.com
309
-
310
- # New (Container Apps)
311
- myapp.region.azurecontainerapps.io → CNAME → myapp.com
312
- ```
313
-
314
- ---
315
-
316
- ## 💰 Comparativo de Custos
317
-
318
- ### Antes (App Service Free F1)
319
- ```
320
- App Service F1: $0/mês
321
- Azure SQL Free: $0/mês
322
- App Insights Free: $0/mês
323
- ────────────────────────────
324
- TOTAL: $0/mês
325
- ```
326
-
327
- ### Depois (Container Apps)
328
- ```
329
- Container Apps: $0-8/mês (scale-to-zero, uso variável)
330
- ACR Basic: $5/mês
331
- Azure SQL Free: $0/mês
332
- App Insights Free: $0/mês
333
- ────────────────────────────
334
- TOTAL: $5-13/mês
335
- ```
336
-
337
- ### Alternativa (App Service B1)
338
- ```
339
- App Service B1: $13/mês (always-on)
340
- Azure SQL Free: $0/mês
341
- App Insights Free: $0/mês
342
- ────────────────────────────
343
- TOTAL: $13/mês
344
- ```
345
-
346
- **Análise:**
347
- - Container Apps mais econômico que B1 para tráfego variável
348
- - Container Apps melhor auto-scaling que App Service
349
- - Container Apps suporta múltiplos containers (arquitetura futura)
350
-
351
- ---
352
-
353
- ## 🔄 Rollback Plan
354
-
355
- ### Se algo der errado
356
-
357
- #### Opção 1: Rollback via DNS (mais rápido)
358
- ```bash
359
- # Reverter CNAME para App Service
360
- # Tempo: ~5 minutos (propagação DNS)
361
- ```
362
-
363
- #### Opção 2: Rollback via código
364
- ```bash
365
- # Atualizar parameters.dev.json
366
- {
367
- "hostingType": {
368
- "value": "appservice"
369
- }
370
- }
371
-
372
- # Redeploy
373
- az deployment group create \
374
- --resource-group rg-myapp-dev \
375
- --template-file main.bicep \
376
- --parameters @parameters.dev.json
377
- ```
378
-
379
- ---
380
-
381
- ## ✅ Checklist Final
382
-
383
- ### Antes de Desligar App Service
384
-
385
- - [ ] Container Apps rodando estável por 7+ dias
386
- - [ ] Todos os health checks passando
387
- - [ ] Monitoramento no App Insights funcionando
388
- - [ ] Logs sendo coletados corretamente
389
- - [ ] Performance igual ou melhor que App Service
390
- - [ ] Custo dentro do esperado
391
- - [ ] Backup do código e configuração
392
- - [ ] Equipe treinada no novo ambiente
393
-
394
- ### Pós-Migração
395
-
396
- - [ ] Deletar App Service Plan (economizar recursos)
397
- - [ ] Deletar deployment slots antigos
398
- - [ ] Atualizar documentação interna
399
- - [ ] Atualizar `.morph/project.md` com nova infra
400
- - [ ] Arquivar configurações antigas
401
-
402
- ---
403
-
404
- ## 🔄 Automatizar com CI/CD
405
-
406
- ### Azure DevOps Pipelines
407
-
408
- Após migração manual bem-sucedida, automatize com pipelines:
409
-
410
- **1. Configurar Workload Identity:**
411
- ```bash
412
- # Ver guia completo em: .azure/docs/azure-devops-setup.md
413
-
414
- # Criar App Registration
415
- az ad app create --display-name "myapp-staging-pipeline"
416
-
417
- # Configurar federated credential
418
- az ad app federated-credential create \
419
- --id <APP_ID> \
420
- --parameters @federated-credential.json
421
- ```
422
-
423
- **2. Usar Pipeline Pronto:**
424
- ```bash
425
- # Copiar pipeline para seu projeto
426
- cp .azure/pipelines/staging-pipeline.yml azure-pipelines.yml
427
-
428
- # Configurar variáveis no Azure DevOps:
429
- # - ACR_NAME
430
- # - APP_NAME
431
- # - SUBSCRIPTION_ID
432
-
433
- # Commit e push
434
- git add azure-pipelines.yml
435
- git commit -m "ci: add staging pipeline"
436
- git push origin main
437
- ```
438
-
439
- **3. Deploy Automático:**
440
- - Dev: Triggado automaticamente em push para `develop`
441
- - Staging: Triggado automaticamente em push para `main`
442
- - Prod: Manual ou via tag `v*`
443
-
444
- **Pipelines inclusos no framework:**
445
- - `.azure/pipelines/dev-pipeline.yml` - App Service Free F1
446
- - `.azure/pipelines/staging-pipeline.yml` - Container Apps (scale-to-zero)
447
- - `.azure/pipelines/prod-pipeline.yml` - Container Apps (always-on, com approval)
448
-
449
- ---
450
-
451
- ## 📚 Referências
452
-
453
- - [Azure Container Apps Docs](https://learn.microsoft.com/azure/container-apps/)
454
- - [ACR Best Practices](https://learn.microsoft.com/azure/container-registry/container-registry-best-practices)
455
- - [Dockerfile Best Practices](https://docs.docker.com/develop/dev-best-practices/)
456
- - [Health Checks ASP.NET](https://learn.microsoft.com/aspnet/core/host-and-deploy/health-checks)
457
- - [Azure DevOps Workload Identity](https://learn.microsoft.com/azure/devops/pipelines/library/connect-to-azure)
458
- - [MORPH-SPEC Azure DevOps Setup](.azure/docs/azure-devops-setup.md)
459
-
460
- ---
461
-
462
- ## 🆘 Troubleshooting
463
-
464
- ### Container não inicia
465
-
466
- ```bash
467
- # Ver logs do container
468
- az containerapp logs show \
469
- -n ca-myapp-dev \
470
- -g rg-myapp-dev \
471
- --follow
472
-
473
- # Ver eventos do container
474
- az containerapp revision list \
475
- -n ca-myapp-dev \
476
- -g rg-myapp-dev \
477
- --query '[0].properties.provisioningState'
478
- ```
479
-
480
- ### Erro de autenticação ACR
481
-
482
- ```bash
483
- # Verificar permissões Managed Identity
484
- az role assignment list \
485
- --assignee $CA_PRINCIPAL_ID \
486
- --scope $(az acr show -n $ACR_NAME --query id -o tsv)
487
-
488
- # Recriar permissão se necessário
489
- az role assignment create \
490
- --assignee $CA_PRINCIPAL_ID \
491
- --role AcrPull \
492
- --scope $(az acr show -n $ACR_NAME --query id -o tsv)
493
- ```
494
-
495
- ### App lento ou timeout
496
-
497
- ```bash
498
- # Verificar recursos
499
- az containerapp show \
500
- -n ca-myapp-dev \
501
- -g rg-myapp-dev \
502
- --query 'properties.template.containers[0].resources'
503
-
504
- # Aumentar CPU/memória se necessário
505
- az containerapp update \
506
- -n ca-myapp-dev \
507
- -g rg-myapp-dev \
508
- --cpu 0.5 \
509
- --memory 1Gi
510
- ```
511
-
512
- ---
513
-
514
- *MORPH-SPEC by Polymorphism Tech*
1
+ # Guia de Migração: App Service → Container Apps
2
+
3
+ > **MORPH-SPEC Framework**
4
+ > Quando seu projeto crescer além do Free Tier
5
+
6
+ ---
7
+
8
+ ## 📊 Quando Migrar?
9
+
10
+ ### Sinais de que é hora de migrar
11
+
12
+ | Indicador | Descrição | Ação |
13
+ |-----------|-----------|------|
14
+ | 🔴 **CPU Quota Esgotada** | Erro `CPU quota exceeded` frequente | Migrar imediatamente |
15
+ | 🔴 **Cold Starts Críticos** | App dormindo afeta UX/negócio | Avaliar impacto → Migrar |
16
+ | 🟡 **Tráfego Crescente** | >500 req/dia consistente | Planejar migração |
17
+ | 🟡 **SSL Customizado** | Necessita domínio próprio com SSL | Avaliar alternativas |
18
+ | 🟢 **Background Jobs** | Precisa rodar Hangfire 24/7 | Migrar ou separar worker |
19
+
20
+ ### Matriz de Decisão
21
+
22
+ ```
23
+ Tráfego < 100 req/dia → App Service Free F1
24
+ Tráfego 100-1000 req/dia → App Service B1 ($13/mês)
25
+ Tráfego > 1000 req/dia → Container Apps (~$5-15/mês)
26
+ Background Jobs 24/7 → Container Apps com minReplicas: 1
27
+ SSL Customizado → Container Apps ou App Service B1+
28
+ ```
29
+
30
+ ---
31
+
32
+ ## 🗺️ Roadmap de Migração
33
+
34
+ ### Fase 1: Preparação (1-2 dias)
35
+
36
+ #### 1.1. Auditoria do Código
37
+
38
+ **Checklist:**
39
+ - [ ] App usa configuração via `IConfiguration`
40
+ - [ ] Não há paths hardcoded (usar `IWebHostEnvironment`)
41
+ - [ ] Logging para stdout/stderr (não arquivos locais)
42
+ - [ ] Health checks implementados (`/health`, `/health/ready`)
43
+ - [ ] Secrets em Key Vault (não `appsettings.json`)
44
+
45
+ **Verificar:**
46
+ ```csharp
47
+ // ✅ BOM: Configuração via IConfiguration
48
+ var connectionString = configuration["ConnectionStrings:Default"];
49
+
50
+ // ❌ RUIM: Hardcoded
51
+ var path = "C:\\Data\\uploads";
52
+
53
+ // ✅ BOM: Usar IWebHostEnvironment
54
+ var path = Path.Combine(webHostEnv.ContentRootPath, "uploads");
55
+ ```
56
+
57
+ #### 1.2. Criar Dockerfile
58
+
59
+ ```dockerfile
60
+ # ============================================
61
+ # MORPH-SPEC - .NET 8 Blazor Container
62
+ # ============================================
63
+
64
+ FROM mcr.microsoft.com/dotnet/aspnet:8.0 AS base
65
+ WORKDIR /app
66
+ EXPOSE 8080
67
+ EXPOSE 8081
68
+
69
+ # Create non-root user
70
+ RUN adduser --disabled-password --gecos '' appuser && chown -R appuser /app
71
+ USER appuser
72
+
73
+ FROM mcr.microsoft.com/dotnet/sdk:8.0 AS build
74
+ WORKDIR /src
75
+ COPY ["YourApp/YourApp.csproj", "YourApp/"]
76
+ RUN dotnet restore "YourApp/YourApp.csproj"
77
+ COPY . .
78
+ WORKDIR "/src/YourApp"
79
+ RUN dotnet build "YourApp.csproj" -c Release -o /app/build
80
+
81
+ FROM build AS publish
82
+ RUN dotnet publish "YourApp.csproj" -c Release -o /app/publish /p:UseAppHost=false
83
+
84
+ FROM base AS final
85
+ WORKDIR /app
86
+ COPY --from=publish /app/publish .
87
+
88
+ # Health check
89
+ HEALTHCHECK --interval=30s --timeout=3s --start-period=5s --retries=3 \
90
+ CMD curl -f http://localhost:8080/health || exit 1
91
+
92
+ ENTRYPOINT ["dotnet", "YourApp.dll"]
93
+ ```
94
+
95
+ #### 1.3. Implementar Health Checks
96
+
97
+ ```csharp
98
+ // Program.cs
99
+ builder.Services.AddHealthChecks()
100
+ .AddSqlServer(
101
+ connectionString: builder.Configuration["ConnectionStrings:Default"],
102
+ name: "sql",
103
+ timeout: TimeSpan.FromSeconds(3))
104
+ .AddCheck("self", () => HealthCheckResult.Healthy());
105
+
106
+ var app = builder.Build();
107
+
108
+ app.MapHealthChecks("/health", new HealthCheckOptions
109
+ {
110
+ Predicate = _ => true,
111
+ ResponseWriter = UIResponseWriter.WriteHealthCheckUIResponse
112
+ });
113
+
114
+ app.MapHealthChecks("/health/ready", new HealthCheckOptions
115
+ {
116
+ Predicate = check => check.Tags.Contains("ready")
117
+ });
118
+
119
+ app.MapHealthChecks("/health/live", new HealthCheckOptions
120
+ {
121
+ Predicate = _ => false
122
+ });
123
+ ```
124
+
125
+ ---
126
+
127
+ ### Fase 2: Setup Azure Container Registry (ACR)
128
+
129
+ #### 2.1. Criar ACR via Bicep
130
+
131
+ ```bicep
132
+ // acr.bicep
133
+ resource acr 'Microsoft.ContainerRegistry/registries@2023-01-01-preview' = {
134
+ name: 'acr${uniqueString(resourceGroup().id)}'
135
+ location: location
136
+ sku: {
137
+ name: 'Basic' // $5/mês
138
+ }
139
+ properties: {
140
+ adminUserEnabled: false // Usar Managed Identity
141
+ }
142
+ }
143
+ ```
144
+
145
+ #### 2.2. Deploy ACR
146
+
147
+ ```bash
148
+ # Via Azure CLI
149
+ az deployment group create \
150
+ --resource-group rg-myapp-dev \
151
+ --template-file acr.bicep
152
+
153
+ # Obter ACR login server
154
+ ACR_NAME=$(az acr list -g rg-myapp-dev --query "[0].name" -o tsv)
155
+ ACR_LOGIN_SERVER=$(az acr show -n $ACR_NAME --query loginServer -o tsv)
156
+ ```
157
+
158
+ #### 2.3. Build e Push Imagem
159
+
160
+ **Via Azure DevOps Pipeline:**
161
+ ```yaml
162
+ # staging-pipeline.yml
163
+ - task: Docker@2
164
+ displayName: 'Build and push container'
165
+ inputs:
166
+ containerRegistry: 'ACR-Connection'
167
+ repository: 'myapp'
168
+ command: 'buildAndPush'
169
+ Dockerfile: 'Dockerfile'
170
+ tags: |
171
+ v1.0.0
172
+ staging-latest
173
+ ```
174
+
175
+ **Via Azure CLI (local):**
176
+ ```bash
177
+ # Build local e push para ACR
178
+ az acr build \
179
+ --registry $ACR_NAME \
180
+ --image myapp:latest \
181
+ --image myapp:v1.0.0 \
182
+ --file Dockerfile \
183
+ .
184
+ ```
185
+
186
+ ---
187
+
188
+ ### Fase 3: Deploy Container Apps
189
+
190
+ #### 3.1. Atualizar Parameters
191
+
192
+ **parameters.dev.json**
193
+ ```json
194
+ {
195
+ "hostingType": {
196
+ "value": "containerapp"
197
+ },
198
+ "containerImage": {
199
+ "value": "acrXXXX.azurecr.io/myapp:latest"
200
+ }
201
+ }
202
+ ```
203
+
204
+ #### 3.2. Deploy Infraestrutura
205
+
206
+ **Via Azure DevOps Pipeline:**
207
+ ```yaml
208
+ # Use o template pronto
209
+ - template: templates/infra-deploy.yml
210
+ parameters:
211
+ azureSubscription: 'Azure-Staging-Connection'
212
+ resourceGroupName: 'rg-myapp-staging'
213
+ bicepTemplateFile: 'content/.morph/templates/infra/main.bicep'
214
+ parametersFile: 'content/.morph/templates/infra/parameters.staging.json'
215
+ ```
216
+
217
+ **Via Azure CLI (local):**
218
+ ```bash
219
+ # Deploy via Azure CLI
220
+ az deployment group create \
221
+ --resource-group rg-myapp-dev \
222
+ --template-file main.bicep \
223
+ --parameters @parameters.dev.json
224
+ ```
225
+
226
+ #### 3.3. Configurar Managed Identity para ACR
227
+
228
+ ```bash
229
+ # Obter Container App Managed Identity
230
+ CA_PRINCIPAL_ID=$(az containerapp show \
231
+ -n ca-myapp-dev \
232
+ -g rg-myapp-dev \
233
+ --query identity.principalId -o tsv)
234
+
235
+ # Dar permissão AcrPull
236
+ az role assignment create \
237
+ --assignee $CA_PRINCIPAL_ID \
238
+ --role AcrPull \
239
+ --scope $(az acr show -n $ACR_NAME --query id -o tsv)
240
+ ```
241
+
242
+ ---
243
+
244
+ ### Fase 4: Teste e Validação
245
+
246
+ #### 4.1. Smoke Tests
247
+
248
+ ```bash
249
+ # Obter URL do Container App
250
+ CA_URL=$(az containerapp show \
251
+ -n ca-myapp-dev \
252
+ -g rg-myapp-dev \
253
+ --query properties.configuration.ingress.fqdn -o tsv)
254
+
255
+ # Testar health checks
256
+ curl https://$CA_URL/health
257
+ curl https://$CA_URL/health/ready
258
+
259
+ # Testar funcionalidade principal
260
+ curl https://$CA_URL/api/status
261
+ ```
262
+
263
+ #### 4.2. Load Test (opcional)
264
+
265
+ ```bash
266
+ # Instalar hey (HTTP load generator)
267
+ go install github.com/rakyll/hey@latest
268
+
269
+ # Simular carga
270
+ hey -z 30s -c 10 https://$CA_URL/
271
+ ```
272
+
273
+ #### 4.3. Validar Auto-Scaling
274
+
275
+ ```bash
276
+ # Gerar carga
277
+ hey -z 2m -c 50 https://$CA_URL/
278
+
279
+ # Monitorar réplicas
280
+ watch -n 2 "az containerapp revision list \
281
+ -n ca-myapp-dev \
282
+ -g rg-myapp-dev \
283
+ --query '[0].properties.replicas' -o tsv"
284
+ ```
285
+
286
+ ---
287
+
288
+ ### Fase 5: Cutover (Mudança de DNS)
289
+
290
+ #### 5.1. Estratégia Blue-Green
291
+
292
+ ```bash
293
+ # Manter App Service rodando (blue)
294
+ # Container Apps já rodando (green)
295
+
296
+ # Testar green profundamente
297
+ # Quando confiante, mudar DNS para green
298
+ # Aguardar 24h
299
+ # Se OK, desligar blue (App Service)
300
+ ```
301
+
302
+ #### 5.2. Atualizar DNS
303
+
304
+ Se usando domínio customizado:
305
+
306
+ ```
307
+ # Old (App Service)
308
+ myapp.azurewebsites.net → CNAME → myapp.com
309
+
310
+ # New (Container Apps)
311
+ myapp.region.azurecontainerapps.io → CNAME → myapp.com
312
+ ```
313
+
314
+ ---
315
+
316
+ ## 💰 Comparativo de Custos
317
+
318
+ ### Antes (App Service Free F1)
319
+ ```
320
+ App Service F1: $0/mês
321
+ Azure SQL Free: $0/mês
322
+ App Insights Free: $0/mês
323
+ ────────────────────────────
324
+ TOTAL: $0/mês
325
+ ```
326
+
327
+ ### Depois (Container Apps)
328
+ ```
329
+ Container Apps: $0-8/mês (scale-to-zero, uso variável)
330
+ ACR Basic: $5/mês
331
+ Azure SQL Free: $0/mês
332
+ App Insights Free: $0/mês
333
+ ────────────────────────────
334
+ TOTAL: $5-13/mês
335
+ ```
336
+
337
+ ### Alternativa (App Service B1)
338
+ ```
339
+ App Service B1: $13/mês (always-on)
340
+ Azure SQL Free: $0/mês
341
+ App Insights Free: $0/mês
342
+ ────────────────────────────
343
+ TOTAL: $13/mês
344
+ ```
345
+
346
+ **Análise:**
347
+ - Container Apps mais econômico que B1 para tráfego variável
348
+ - Container Apps melhor auto-scaling que App Service
349
+ - Container Apps suporta múltiplos containers (arquitetura futura)
350
+
351
+ ---
352
+
353
+ ## 🔄 Rollback Plan
354
+
355
+ ### Se algo der errado
356
+
357
+ #### Opção 1: Rollback via DNS (mais rápido)
358
+ ```bash
359
+ # Reverter CNAME para App Service
360
+ # Tempo: ~5 minutos (propagação DNS)
361
+ ```
362
+
363
+ #### Opção 2: Rollback via código
364
+ ```bash
365
+ # Atualizar parameters.dev.json
366
+ {
367
+ "hostingType": {
368
+ "value": "appservice"
369
+ }
370
+ }
371
+
372
+ # Redeploy
373
+ az deployment group create \
374
+ --resource-group rg-myapp-dev \
375
+ --template-file main.bicep \
376
+ --parameters @parameters.dev.json
377
+ ```
378
+
379
+ ---
380
+
381
+ ## ✅ Checklist Final
382
+
383
+ ### Antes de Desligar App Service
384
+
385
+ - [ ] Container Apps rodando estável por 7+ dias
386
+ - [ ] Todos os health checks passando
387
+ - [ ] Monitoramento no App Insights funcionando
388
+ - [ ] Logs sendo coletados corretamente
389
+ - [ ] Performance igual ou melhor que App Service
390
+ - [ ] Custo dentro do esperado
391
+ - [ ] Backup do código e configuração
392
+ - [ ] Equipe treinada no novo ambiente
393
+
394
+ ### Pós-Migração
395
+
396
+ - [ ] Deletar App Service Plan (economizar recursos)
397
+ - [ ] Deletar deployment slots antigos
398
+ - [ ] Atualizar documentação interna
399
+ - [ ] Atualizar `.morph/project.md` com nova infra
400
+ - [ ] Arquivar configurações antigas
401
+
402
+ ---
403
+
404
+ ## 🔄 Automatizar com CI/CD
405
+
406
+ ### Azure DevOps Pipelines
407
+
408
+ Após migração manual bem-sucedida, automatize com pipelines:
409
+
410
+ **1. Configurar Workload Identity:**
411
+ ```bash
412
+ # Ver guia completo em: .azure/docs/azure-devops-setup.md
413
+
414
+ # Criar App Registration
415
+ az ad app create --display-name "myapp-staging-pipeline"
416
+
417
+ # Configurar federated credential
418
+ az ad app federated-credential create \
419
+ --id <APP_ID> \
420
+ --parameters @federated-credential.json
421
+ ```
422
+
423
+ **2. Usar Pipeline Pronto:**
424
+ ```bash
425
+ # Copiar pipeline para seu projeto
426
+ cp .azure/pipelines/staging-pipeline.yml azure-pipelines.yml
427
+
428
+ # Configurar variáveis no Azure DevOps:
429
+ # - ACR_NAME
430
+ # - APP_NAME
431
+ # - SUBSCRIPTION_ID
432
+
433
+ # Commit e push
434
+ git add azure-pipelines.yml
435
+ git commit -m "ci: add staging pipeline"
436
+ git push origin main
437
+ ```
438
+
439
+ **3. Deploy Automático:**
440
+ - Dev: Triggado automaticamente em push para `develop`
441
+ - Staging: Triggado automaticamente em push para `main`
442
+ - Prod: Manual ou via tag `v*`
443
+
444
+ **Pipelines inclusos no framework:**
445
+ - `.azure/pipelines/dev-pipeline.yml` - App Service Free F1
446
+ - `.azure/pipelines/staging-pipeline.yml` - Container Apps (scale-to-zero)
447
+ - `.azure/pipelines/prod-pipeline.yml` - Container Apps (always-on, com approval)
448
+
449
+ ---
450
+
451
+ ## 📚 Referências
452
+
453
+ - [Azure Container Apps Docs](https://learn.microsoft.com/azure/container-apps/)
454
+ - [ACR Best Practices](https://learn.microsoft.com/azure/container-registry/container-registry-best-practices)
455
+ - [Dockerfile Best Practices](https://docs.docker.com/develop/dev-best-practices/)
456
+ - [Health Checks ASP.NET](https://learn.microsoft.com/aspnet/core/host-and-deploy/health-checks)
457
+ - [Azure DevOps Workload Identity](https://learn.microsoft.com/azure/devops/pipelines/library/connect-to-azure)
458
+ - [MORPH-SPEC Azure DevOps Setup](.azure/docs/azure-devops-setup.md)
459
+
460
+ ---
461
+
462
+ ## 🆘 Troubleshooting
463
+
464
+ ### Container não inicia
465
+
466
+ ```bash
467
+ # Ver logs do container
468
+ az containerapp logs show \
469
+ -n ca-myapp-dev \
470
+ -g rg-myapp-dev \
471
+ --follow
472
+
473
+ # Ver eventos do container
474
+ az containerapp revision list \
475
+ -n ca-myapp-dev \
476
+ -g rg-myapp-dev \
477
+ --query '[0].properties.provisioningState'
478
+ ```
479
+
480
+ ### Erro de autenticação ACR
481
+
482
+ ```bash
483
+ # Verificar permissões Managed Identity
484
+ az role assignment list \
485
+ --assignee $CA_PRINCIPAL_ID \
486
+ --scope $(az acr show -n $ACR_NAME --query id -o tsv)
487
+
488
+ # Recriar permissão se necessário
489
+ az role assignment create \
490
+ --assignee $CA_PRINCIPAL_ID \
491
+ --role AcrPull \
492
+ --scope $(az acr show -n $ACR_NAME --query id -o tsv)
493
+ ```
494
+
495
+ ### App lento ou timeout
496
+
497
+ ```bash
498
+ # Verificar recursos
499
+ az containerapp show \
500
+ -n ca-myapp-dev \
501
+ -g rg-myapp-dev \
502
+ --query 'properties.template.containers[0].resources'
503
+
504
+ # Aumentar CPU/memória se necessário
505
+ az containerapp update \
506
+ -n ca-myapp-dev \
507
+ -g rg-myapp-dev \
508
+ --cpu 0.5 \
509
+ --memory 1Gi
510
+ ```
511
+
512
+ ---
513
+
514
+ *MORPH-SPEC by Polymorphism Tech*