@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,515 +1,515 @@
1
- # Desenvolvimento Local - MORPH-SPEC
2
-
3
- > Como rodar o projeto localmente e acessar recursos Azure do Staging
4
-
5
- ---
6
-
7
- ## 📋 Conceito
8
-
9
- Developers **não precisam de ambiente próprio**. Ao invés disso:
10
-
11
- - ✅ **Rodam o projeto LOCAL** (Visual Studio, Rider, VS Code)
12
- - ✅ **Acessam recursos REMOTOS** do ambiente Staging (SQL, Storage, Key Vault)
13
- - ✅ **Testam localmente** antes de commitar
14
- - ✅ **Criam PRs** para Staging quando pronto
15
-
16
- **Vantagens:**
17
- - 💰 **Custo zero** para ambiente dev
18
- - ⚡ **Desenvolvimento mais rápido** (hot reload, debug local)
19
- - 🔄 **Dados compartilhados** (todos testam com mesmos dados)
20
- - 🎯 **Ambiente staging sempre atualizado**
21
-
22
- ---
23
-
24
- ## ⚙️ Setup Inicial (Uma vez)
25
-
26
- ### 1. Instalar Ferramentas
27
-
28
- ```bash
29
- # .NET SDK
30
- winget install Microsoft.DotNet.SDK.8
31
-
32
- # Azure CLI
33
- winget install Microsoft.AzureCLI
34
-
35
- # Git
36
- winget install Git.Git
37
-
38
- # Visual Studio ou VS Code
39
- winget install Microsoft.VisualStudio.2022.Community
40
- # ou
41
- winget install Microsoft.VisualStudioCode
42
- ```
43
-
44
- ### 2. Login no Azure
45
-
46
- ```bash
47
- # Login com sua conta corporativa
48
- az login
49
-
50
- # Verificar subscription
51
- az account show
52
-
53
- # Se necessário, trocar subscription
54
- az account set --subscription "<SUBSCRIPTION_ID>"
55
- ```
56
-
57
- ### 3. Obter Permissões Staging
58
-
59
- Solicite ao tech lead para adicionar você ao ambiente staging:
60
-
61
- ```bash
62
- # Tech lead executa:
63
- az role assignment create \
64
- --assignee <developer-email> \
65
- --role "Contributor" \
66
- --scope "/subscriptions/<SUB_ID>/resourceGroups/rg-myapp-staging"
67
-
68
- # Permissão SQL Database
69
- az sql server ad-admin set \
70
- --resource-group rg-myapp-staging \
71
- --server-name sql-myapp-staging \
72
- --display-name "<developer-name>" \
73
- --object-id <developer-object-id>
74
- ```
75
-
76
- ---
77
-
78
- ## 🔧 Configuração do Projeto
79
-
80
- ### 1. Clonar Repositório
81
-
82
- ```bash
83
- git clone https://dev.azure.com/org/project/_git/repo
84
- cd repo
85
- ```
86
-
87
- ### 2. Criar appsettings.Development.json
88
-
89
- ```json
90
- {
91
- "ConnectionStrings": {
92
- "DefaultConnection": "Server=sql-myapp-staging.database.windows.net;Database=myapp;Authentication=Active Directory Default;"
93
- },
94
- "Azure": {
95
- "KeyVault": "https://kv-myapp-staging.vault.azure.net/",
96
- "Storage": {
97
- "AccountName": "stmyappstaging",
98
- "BlobEndpoint": "https://stmyappstaging.blob.core.windows.net"
99
- }
100
- },
101
- "ApplicationInsights": {
102
- "ConnectionString": "<copiar-do-azure-portal-staging>"
103
- },
104
- "Logging": {
105
- "LogLevel": {
106
- "Default": "Information",
107
- "Microsoft.AspNetCore": "Warning"
108
- }
109
- }
110
- }
111
- ```
112
-
113
- **⚠️ IMPORTANTE:**
114
- - ❌ **NUNCA commitar** `appsettings.Development.json`
115
- - ✅ Já está no `.gitignore`
116
- - ✅ Cada developer tem seu próprio arquivo local
117
-
118
- ### 3. Configurar DefaultAzureCredential
119
-
120
- No `Program.cs`:
121
-
122
- ```csharp
123
- builder.Services.AddDbContext<ApplicationDbContext>(options =>
124
- {
125
- var connectionString = builder.Configuration.GetConnectionString("DefaultConnection");
126
- options.UseSqlServer(connectionString, sqlOptions =>
127
- {
128
- sqlOptions.EnableRetryOnFailure(
129
- maxRetryCount: 3,
130
- maxRetryDelay: TimeSpan.FromSeconds(30),
131
- errorNumbersToAdd: null);
132
- });
133
- });
134
-
135
- // Azure Key Vault
136
- if (!builder.Environment.IsDevelopment())
137
- {
138
- var keyVaultUrl = builder.Configuration["Azure:KeyVault"];
139
- builder.Configuration.AddAzureKeyVault(
140
- new Uri(keyVaultUrl),
141
- new DefaultAzureCredential());
142
- }
143
-
144
- // Azure Storage
145
- builder.Services.AddSingleton(x =>
146
- {
147
- var accountName = builder.Configuration["Azure:Storage:AccountName"];
148
- var blobServiceClient = new BlobServiceClient(
149
- new Uri($"https://{accountName}.blob.core.windows.net"),
150
- new DefaultAzureCredential());
151
- return blobServiceClient;
152
- });
153
- ```
154
-
155
- **DefaultAzureCredential** tenta autenticar nesta ordem:
156
- 1. Environment variables (CI/CD)
157
- 2. Managed Identity (Azure)
158
- 3. **Azure CLI** (← usado localmente)
159
- 4. Visual Studio
160
- 5. VS Code
161
-
162
- ---
163
-
164
- ## 🚀 Rodando Localmente
165
-
166
- ### Opção 1: Visual Studio
167
-
168
- 1. Abrir `.sln`
169
- 2. F5 (Debug) ou Ctrl+F5 (Run)
170
- 3. Projeto abrirá em `https://localhost:7001`
171
-
172
- ### Opção 2: Rider
173
-
174
- 1. Abrir `.sln`
175
- 2. Shift+F10 (Run) ou Shift+F9 (Debug)
176
-
177
- ### Opção 3: VS Code
178
-
179
- ```bash
180
- # Restaurar dependências
181
- dotnet restore
182
-
183
- # Rodar projeto
184
- dotnet run --project src/YourApp/YourApp.csproj
185
-
186
- # Ou com watch (hot reload)
187
- dotnet watch --project src/YourApp/YourApp.csproj
188
- ```
189
-
190
- ### Opção 4: CLI direto
191
-
192
- ```bash
193
- cd src/YourApp
194
- dotnet run
195
- ```
196
-
197
- ---
198
-
199
- ## 🗄️ Acesso ao Banco de Dados
200
-
201
- ### Via Azure Data Studio
202
-
203
- 1. Download: https://aka.ms/azuredatastudio
204
- 2. Conectar:
205
- - **Server**: `sql-myapp-staging.database.windows.net`
206
- - **Database**: `myapp`
207
- - **Authentication**: `Azure Active Directory - Universal with MFA`
208
- - **Account**: Sua conta corporativa
209
-
210
- ### Via SQL Server Management Studio (SSMS)
211
-
212
- 1. Download: https://aka.ms/ssmsfullsetup
213
- 2. Conectar:
214
- - **Server**: `sql-myapp-staging.database.windows.net`
215
- - **Authentication**: `Azure Active Directory - Universal with MFA`
216
- - **Login**: Sua conta
217
-
218
- ### Via Entity Framework Migrations
219
-
220
- ```bash
221
- # Criar migration
222
- dotnet ef migrations add NomeDaMigration
223
-
224
- # Aplicar migration no staging (CUIDADO!)
225
- dotnet ef database update
226
-
227
- # Reverter migration
228
- dotnet ef database update PreviousMigration
229
- ```
230
-
231
- ⚠️ **ATENÇÃO**: Migrations aplicadas localmente afetam **TODOS** os developers!
232
-
233
- **Melhor prática:**
234
- - Criar migration localmente
235
- - Commitar migration
236
- - Deixar pipeline aplicar automaticamente
237
-
238
- ---
239
-
240
- ## 📦 Acesso ao Azure Storage
241
-
242
- ### Via Azure Storage Explorer
243
-
244
- 1. Download: https://aka.ms/storageexplorer
245
- 2. Login com sua conta
246
- 3. Navegue até: `stmyappstaging`
247
- 4. Pode fazer upload/download de blobs
248
-
249
- ### Via Código
250
-
251
- ```csharp
252
- // Upload blob
253
- var blobClient = blobServiceClient
254
- .GetBlobContainerClient("uploads")
255
- .GetBlobClient("test.txt");
256
-
257
- await blobClient.UploadAsync(
258
- BinaryData.FromString("Hello from local!"),
259
- overwrite: true);
260
-
261
- // Download blob
262
- var downloadedData = await blobClient.DownloadContentAsync();
263
- Console.WriteLine(downloadedData.Value.Content.ToString());
264
- ```
265
-
266
- ---
267
-
268
- ## 🔐 Acesso ao Key Vault
269
-
270
- ### Via Azure CLI
271
-
272
- ```bash
273
- # Listar secrets
274
- az keyvault secret list \
275
- --vault-name kv-myapp-staging
276
-
277
- # Obter secret
278
- az keyvault secret show \
279
- --vault-name kv-myapp-staging \
280
- --name "ConnectionStrings--DefaultConnection"
281
- ```
282
-
283
- ### Via Código
284
-
285
- ```csharp
286
- // Program.cs já configurou automaticamente
287
- // Acesse via IConfiguration normalmente:
288
-
289
- var secret = builder.Configuration["MySecretKey"];
290
- ```
291
-
292
- ---
293
-
294
- ## 🧪 Testes Locais
295
-
296
- ### Unit Tests
297
-
298
- ```bash
299
- # Rodar todos os testes
300
- dotnet test
301
-
302
- # Rodar com cobertura
303
- dotnet test --collect:"XPlat Code Coverage"
304
-
305
- # Apenas testes de uma classe
306
- dotnet test --filter "FullyQualifiedName~MyTests"
307
- ```
308
-
309
- ### Integration Tests
310
-
311
- ```bash
312
- # Integration tests usam staging database
313
- dotnet test --filter "Category=Integration"
314
- ```
315
-
316
- ⚠️ **Cuidado**: Integration tests modificam dados staging!
317
-
318
- **Melhor prática:**
319
- - Use transações que fazem rollback
320
- - Ou crie/delete dados de teste
321
- - Ou use database separado para testes
322
-
323
- ---
324
-
325
- ## 🔄 Workflow Diário
326
-
327
- ### 1. Começar Nova Feature
328
-
329
- ```bash
330
- # Atualizar staging
331
- git checkout staging
332
- git pull origin staging
333
-
334
- # Criar feature branch
335
- git checkout -b feature/minha-funcionalidade
336
-
337
- # Rodar projeto local
338
- dotnet run
339
- ```
340
-
341
- ### 2. Desenvolver
342
-
343
- ```bash
344
- # Fazer mudanças
345
- # Testar localmente (conecta em staging resources)
346
- # Commitar frequentemente
347
-
348
- git add .
349
- git commit -m "feat: adiciona funcionalidade X"
350
- ```
351
-
352
- ### 3. Testar
353
-
354
- ```bash
355
- # Unit tests
356
- dotnet test
357
-
358
- # Teste manual no browser
359
- # http://localhost:5000
360
- ```
361
-
362
- ### 4. Push e PR
363
-
364
- ```bash
365
- # Push da feature branch
366
- git push origin feature/minha-funcionalidade
367
-
368
- # Criar PR para staging no Azure DevOps
369
- # 1. Vá em Repos → Pull Requests → New Pull Request
370
- # 2. Source: feature/minha-funcionalidade
371
- # 3. Target: staging
372
- # 4. Adicionar reviewers
373
- # 5. Create
374
- ```
375
-
376
- ### 5. Code Review e Merge
377
-
378
- ```
379
- # Reviewer faz code review
380
- # Após aprovação, merge
381
- # Pipeline staging triggará automaticamente
382
- ```
383
-
384
- ### 6. Testar em Staging Deployado
385
-
386
- ```bash
387
- # Acessar URL staging
388
- https://ca-myapp-staging.<region>.azurecontainerapps.io
389
-
390
- # Validar funcionalidade
391
- # Se OK, criar PR staging → main para prod
392
- ```
393
-
394
- ---
395
-
396
- ## 🆘 Troubleshooting
397
-
398
- ### Erro: "Login failed for user"
399
-
400
- **Causa**: Azure CLI não autenticado ou token expirado.
401
-
402
- **Solução**:
403
- ```bash
404
- az logout
405
- az login
406
- az account show
407
- ```
408
-
409
- ### Erro: "Cannot connect to SQL Server"
410
-
411
- **Causa**: Firewall SQL bloqueando seu IP.
412
-
413
- **Solução**:
414
- ```bash
415
- # Adicionar seu IP ao firewall
416
- az sql server firewall-rule create \
417
- --resource-group rg-myapp-staging \
418
- --server sql-myapp-staging \
419
- --name "DevIP-$(whoami)" \
420
- --start-ip-address <SEU_IP> \
421
- --end-ip-address <SEU_IP>
422
-
423
- # Obter seu IP público
424
- curl https://api.ipify.org
425
- ```
426
-
427
- ### Erro: "Access denied to Key Vault"
428
-
429
- **Causa**: Sem permissão no Key Vault.
430
-
431
- **Solução** (Tech lead):
432
- ```bash
433
- az keyvault set-policy \
434
- --name kv-myapp-staging \
435
- --upn <developer-email> \
436
- --secret-permissions get list
437
- ```
438
-
439
- ### Erro: "Blob not found"
440
-
441
- **Causa**: Container não existe ou sem permissão.
442
-
443
- **Solução** (Tech lead):
444
- ```bash
445
- # Dar permissão Storage Blob Data Contributor
446
- az role assignment create \
447
- --assignee <developer-email> \
448
- --role "Storage Blob Data Contributor" \
449
- --scope "/subscriptions/<SUB_ID>/resourceGroups/rg-myapp-staging/providers/Microsoft.Storage/storageAccounts/stmyappstaging"
450
- ```
451
-
452
- ### Erro: "DefaultAzureCredential failed"
453
-
454
- **Causa**: Nenhum método de autenticação funcionou.
455
-
456
- **Solução**:
457
- 1. Feche Visual Studio/Rider/VS Code
458
- 2. `az logout && az login`
459
- 3. Reabra IDE
460
- 4. Rode novamente
461
-
462
- ### Migration falhou
463
-
464
- **Causa**: Conflito com migration de outro developer.
465
-
466
- **Solução**:
467
- ```bash
468
- # Pull latest
469
- git pull origin staging
470
-
471
- # Deletar sua migration
472
- dotnet ef migrations remove
473
-
474
- # Aplicar migrations remotas
475
- dotnet ef database update
476
-
477
- # Recriar sua migration
478
- dotnet ef migrations add SuaMigration
479
- ```
480
-
481
- ---
482
-
483
- ## ✅ Checklist de Setup
484
-
485
- Para novos developers:
486
-
487
- - [ ] Instalar .NET SDK 8
488
- - [ ] Instalar Azure CLI
489
- - [ ] Instalar Git
490
- - [ ] Instalar IDE (VS/Rider/VS Code)
491
- - [ ] `az login` com conta corporativa
492
- - [ ] Solicitar permissões staging (tech lead)
493
- - [ ] Clonar repositório
494
- - [ ] Criar `appsettings.Development.json`
495
- - [ ] Testar conexão SQL (Azure Data Studio)
496
- - [ ] Rodar projeto local (`dotnet run`)
497
- - [ ] Acessar `https://localhost:7001`
498
- - [ ] Rodar testes (`dotnet test`)
499
- - [ ] Criar feature branch de teste
500
- - [ ] Fazer commit de teste
501
- - [ ] Criar PR de teste
502
- - [ ] Deletar PR/branch de teste
503
-
504
- ---
505
-
506
- ## 📚 Referências
507
-
508
- - [DefaultAzureCredential](https://learn.microsoft.com/dotnet/api/azure.identity.defaultazurecredential)
509
- - [Azure SQL with Azure AD](https://learn.microsoft.com/azure/azure-sql/database/authentication-aad-configure)
510
- - [Azure Storage SDK](https://learn.microsoft.com/azure/storage/blobs/storage-quickstart-blobs-dotnet)
511
- - [Entity Framework Migrations](https://learn.microsoft.com/ef/core/managing-schemas/migrations/)
512
-
513
- ---
514
-
515
- *MORPH-SPEC by Polymorphism Tech*
1
+ # Desenvolvimento Local - MORPH-SPEC
2
+
3
+ > Como rodar o projeto localmente e acessar recursos Azure do Staging
4
+
5
+ ---
6
+
7
+ ## 📋 Conceito
8
+
9
+ Developers **não precisam de ambiente próprio**. Ao invés disso:
10
+
11
+ - ✅ **Rodam o projeto LOCAL** (Visual Studio, Rider, VS Code)
12
+ - ✅ **Acessam recursos REMOTOS** do ambiente Staging (SQL, Storage, Key Vault)
13
+ - ✅ **Testam localmente** antes de commitar
14
+ - ✅ **Criam PRs** para Staging quando pronto
15
+
16
+ **Vantagens:**
17
+ - 💰 **Custo zero** para ambiente dev
18
+ - ⚡ **Desenvolvimento mais rápido** (hot reload, debug local)
19
+ - 🔄 **Dados compartilhados** (todos testam com mesmos dados)
20
+ - 🎯 **Ambiente staging sempre atualizado**
21
+
22
+ ---
23
+
24
+ ## ⚙️ Setup Inicial (Uma vez)
25
+
26
+ ### 1. Instalar Ferramentas
27
+
28
+ ```bash
29
+ # .NET SDK
30
+ winget install Microsoft.DotNet.SDK.8
31
+
32
+ # Azure CLI
33
+ winget install Microsoft.AzureCLI
34
+
35
+ # Git
36
+ winget install Git.Git
37
+
38
+ # Visual Studio ou VS Code
39
+ winget install Microsoft.VisualStudio.2022.Community
40
+ # ou
41
+ winget install Microsoft.VisualStudioCode
42
+ ```
43
+
44
+ ### 2. Login no Azure
45
+
46
+ ```bash
47
+ # Login com sua conta corporativa
48
+ az login
49
+
50
+ # Verificar subscription
51
+ az account show
52
+
53
+ # Se necessário, trocar subscription
54
+ az account set --subscription "<SUBSCRIPTION_ID>"
55
+ ```
56
+
57
+ ### 3. Obter Permissões Staging
58
+
59
+ Solicite ao tech lead para adicionar você ao ambiente staging:
60
+
61
+ ```bash
62
+ # Tech lead executa:
63
+ az role assignment create \
64
+ --assignee <developer-email> \
65
+ --role "Contributor" \
66
+ --scope "/subscriptions/<SUB_ID>/resourceGroups/rg-myapp-staging"
67
+
68
+ # Permissão SQL Database
69
+ az sql server ad-admin set \
70
+ --resource-group rg-myapp-staging \
71
+ --server-name sql-myapp-staging \
72
+ --display-name "<developer-name>" \
73
+ --object-id <developer-object-id>
74
+ ```
75
+
76
+ ---
77
+
78
+ ## 🔧 Configuração do Projeto
79
+
80
+ ### 1. Clonar Repositório
81
+
82
+ ```bash
83
+ git clone https://dev.azure.com/org/project/_git/repo
84
+ cd repo
85
+ ```
86
+
87
+ ### 2. Criar appsettings.Development.json
88
+
89
+ ```json
90
+ {
91
+ "ConnectionStrings": {
92
+ "DefaultConnection": "Server=sql-myapp-staging.database.windows.net;Database=myapp;Authentication=Active Directory Default;"
93
+ },
94
+ "Azure": {
95
+ "KeyVault": "https://kv-myapp-staging.vault.azure.net/",
96
+ "Storage": {
97
+ "AccountName": "stmyappstaging",
98
+ "BlobEndpoint": "https://stmyappstaging.blob.core.windows.net"
99
+ }
100
+ },
101
+ "ApplicationInsights": {
102
+ "ConnectionString": "<copiar-do-azure-portal-staging>"
103
+ },
104
+ "Logging": {
105
+ "LogLevel": {
106
+ "Default": "Information",
107
+ "Microsoft.AspNetCore": "Warning"
108
+ }
109
+ }
110
+ }
111
+ ```
112
+
113
+ **⚠️ IMPORTANTE:**
114
+ - ❌ **NUNCA commitar** `appsettings.Development.json`
115
+ - ✅ Já está no `.gitignore`
116
+ - ✅ Cada developer tem seu próprio arquivo local
117
+
118
+ ### 3. Configurar DefaultAzureCredential
119
+
120
+ No `Program.cs`:
121
+
122
+ ```csharp
123
+ builder.Services.AddDbContext<ApplicationDbContext>(options =>
124
+ {
125
+ var connectionString = builder.Configuration.GetConnectionString("DefaultConnection");
126
+ options.UseSqlServer(connectionString, sqlOptions =>
127
+ {
128
+ sqlOptions.EnableRetryOnFailure(
129
+ maxRetryCount: 3,
130
+ maxRetryDelay: TimeSpan.FromSeconds(30),
131
+ errorNumbersToAdd: null);
132
+ });
133
+ });
134
+
135
+ // Azure Key Vault
136
+ if (!builder.Environment.IsDevelopment())
137
+ {
138
+ var keyVaultUrl = builder.Configuration["Azure:KeyVault"];
139
+ builder.Configuration.AddAzureKeyVault(
140
+ new Uri(keyVaultUrl),
141
+ new DefaultAzureCredential());
142
+ }
143
+
144
+ // Azure Storage
145
+ builder.Services.AddSingleton(x =>
146
+ {
147
+ var accountName = builder.Configuration["Azure:Storage:AccountName"];
148
+ var blobServiceClient = new BlobServiceClient(
149
+ new Uri($"https://{accountName}.blob.core.windows.net"),
150
+ new DefaultAzureCredential());
151
+ return blobServiceClient;
152
+ });
153
+ ```
154
+
155
+ **DefaultAzureCredential** tenta autenticar nesta ordem:
156
+ 1. Environment variables (CI/CD)
157
+ 2. Managed Identity (Azure)
158
+ 3. **Azure CLI** (← usado localmente)
159
+ 4. Visual Studio
160
+ 5. VS Code
161
+
162
+ ---
163
+
164
+ ## 🚀 Rodando Localmente
165
+
166
+ ### Opção 1: Visual Studio
167
+
168
+ 1. Abrir `.sln`
169
+ 2. F5 (Debug) ou Ctrl+F5 (Run)
170
+ 3. Projeto abrirá em `https://localhost:7001`
171
+
172
+ ### Opção 2: Rider
173
+
174
+ 1. Abrir `.sln`
175
+ 2. Shift+F10 (Run) ou Shift+F9 (Debug)
176
+
177
+ ### Opção 3: VS Code
178
+
179
+ ```bash
180
+ # Restaurar dependências
181
+ dotnet restore
182
+
183
+ # Rodar projeto
184
+ dotnet run --project src/YourApp/YourApp.csproj
185
+
186
+ # Ou com watch (hot reload)
187
+ dotnet watch --project src/YourApp/YourApp.csproj
188
+ ```
189
+
190
+ ### Opção 4: CLI direto
191
+
192
+ ```bash
193
+ cd src/YourApp
194
+ dotnet run
195
+ ```
196
+
197
+ ---
198
+
199
+ ## 🗄️ Acesso ao Banco de Dados
200
+
201
+ ### Via Azure Data Studio
202
+
203
+ 1. Download: https://aka.ms/azuredatastudio
204
+ 2. Conectar:
205
+ - **Server**: `sql-myapp-staging.database.windows.net`
206
+ - **Database**: `myapp`
207
+ - **Authentication**: `Azure Active Directory - Universal with MFA`
208
+ - **Account**: Sua conta corporativa
209
+
210
+ ### Via SQL Server Management Studio (SSMS)
211
+
212
+ 1. Download: https://aka.ms/ssmsfullsetup
213
+ 2. Conectar:
214
+ - **Server**: `sql-myapp-staging.database.windows.net`
215
+ - **Authentication**: `Azure Active Directory - Universal with MFA`
216
+ - **Login**: Sua conta
217
+
218
+ ### Via Entity Framework Migrations
219
+
220
+ ```bash
221
+ # Criar migration
222
+ dotnet ef migrations add NomeDaMigration
223
+
224
+ # Aplicar migration no staging (CUIDADO!)
225
+ dotnet ef database update
226
+
227
+ # Reverter migration
228
+ dotnet ef database update PreviousMigration
229
+ ```
230
+
231
+ ⚠️ **ATENÇÃO**: Migrations aplicadas localmente afetam **TODOS** os developers!
232
+
233
+ **Melhor prática:**
234
+ - Criar migration localmente
235
+ - Commitar migration
236
+ - Deixar pipeline aplicar automaticamente
237
+
238
+ ---
239
+
240
+ ## 📦 Acesso ao Azure Storage
241
+
242
+ ### Via Azure Storage Explorer
243
+
244
+ 1. Download: https://aka.ms/storageexplorer
245
+ 2. Login com sua conta
246
+ 3. Navegue até: `stmyappstaging`
247
+ 4. Pode fazer upload/download de blobs
248
+
249
+ ### Via Código
250
+
251
+ ```csharp
252
+ // Upload blob
253
+ var blobClient = blobServiceClient
254
+ .GetBlobContainerClient("uploads")
255
+ .GetBlobClient("test.txt");
256
+
257
+ await blobClient.UploadAsync(
258
+ BinaryData.FromString("Hello from local!"),
259
+ overwrite: true);
260
+
261
+ // Download blob
262
+ var downloadedData = await blobClient.DownloadContentAsync();
263
+ Console.WriteLine(downloadedData.Value.Content.ToString());
264
+ ```
265
+
266
+ ---
267
+
268
+ ## 🔐 Acesso ao Key Vault
269
+
270
+ ### Via Azure CLI
271
+
272
+ ```bash
273
+ # Listar secrets
274
+ az keyvault secret list \
275
+ --vault-name kv-myapp-staging
276
+
277
+ # Obter secret
278
+ az keyvault secret show \
279
+ --vault-name kv-myapp-staging \
280
+ --name "ConnectionStrings--DefaultConnection"
281
+ ```
282
+
283
+ ### Via Código
284
+
285
+ ```csharp
286
+ // Program.cs já configurou automaticamente
287
+ // Acesse via IConfiguration normalmente:
288
+
289
+ var secret = builder.Configuration["MySecretKey"];
290
+ ```
291
+
292
+ ---
293
+
294
+ ## 🧪 Testes Locais
295
+
296
+ ### Unit Tests
297
+
298
+ ```bash
299
+ # Rodar todos os testes
300
+ dotnet test
301
+
302
+ # Rodar com cobertura
303
+ dotnet test --collect:"XPlat Code Coverage"
304
+
305
+ # Apenas testes de uma classe
306
+ dotnet test --filter "FullyQualifiedName~MyTests"
307
+ ```
308
+
309
+ ### Integration Tests
310
+
311
+ ```bash
312
+ # Integration tests usam staging database
313
+ dotnet test --filter "Category=Integration"
314
+ ```
315
+
316
+ ⚠️ **Cuidado**: Integration tests modificam dados staging!
317
+
318
+ **Melhor prática:**
319
+ - Use transações que fazem rollback
320
+ - Ou crie/delete dados de teste
321
+ - Ou use database separado para testes
322
+
323
+ ---
324
+
325
+ ## 🔄 Workflow Diário
326
+
327
+ ### 1. Começar Nova Feature
328
+
329
+ ```bash
330
+ # Atualizar staging
331
+ git checkout staging
332
+ git pull origin staging
333
+
334
+ # Criar feature branch
335
+ git checkout -b feature/minha-funcionalidade
336
+
337
+ # Rodar projeto local
338
+ dotnet run
339
+ ```
340
+
341
+ ### 2. Desenvolver
342
+
343
+ ```bash
344
+ # Fazer mudanças
345
+ # Testar localmente (conecta em staging resources)
346
+ # Commitar frequentemente
347
+
348
+ git add .
349
+ git commit -m "feat: adiciona funcionalidade X"
350
+ ```
351
+
352
+ ### 3. Testar
353
+
354
+ ```bash
355
+ # Unit tests
356
+ dotnet test
357
+
358
+ # Teste manual no browser
359
+ # http://localhost:5000
360
+ ```
361
+
362
+ ### 4. Push e PR
363
+
364
+ ```bash
365
+ # Push da feature branch
366
+ git push origin feature/minha-funcionalidade
367
+
368
+ # Criar PR para staging no Azure DevOps
369
+ # 1. Vá em Repos → Pull Requests → New Pull Request
370
+ # 2. Source: feature/minha-funcionalidade
371
+ # 3. Target: staging
372
+ # 4. Adicionar reviewers
373
+ # 5. Create
374
+ ```
375
+
376
+ ### 5. Code Review e Merge
377
+
378
+ ```
379
+ # Reviewer faz code review
380
+ # Após aprovação, merge
381
+ # Pipeline staging triggará automaticamente
382
+ ```
383
+
384
+ ### 6. Testar em Staging Deployado
385
+
386
+ ```bash
387
+ # Acessar URL staging
388
+ https://ca-myapp-staging.<region>.azurecontainerapps.io
389
+
390
+ # Validar funcionalidade
391
+ # Se OK, criar PR staging → main para prod
392
+ ```
393
+
394
+ ---
395
+
396
+ ## 🆘 Troubleshooting
397
+
398
+ ### Erro: "Login failed for user"
399
+
400
+ **Causa**: Azure CLI não autenticado ou token expirado.
401
+
402
+ **Solução**:
403
+ ```bash
404
+ az logout
405
+ az login
406
+ az account show
407
+ ```
408
+
409
+ ### Erro: "Cannot connect to SQL Server"
410
+
411
+ **Causa**: Firewall SQL bloqueando seu IP.
412
+
413
+ **Solução**:
414
+ ```bash
415
+ # Adicionar seu IP ao firewall
416
+ az sql server firewall-rule create \
417
+ --resource-group rg-myapp-staging \
418
+ --server sql-myapp-staging \
419
+ --name "DevIP-$(whoami)" \
420
+ --start-ip-address <SEU_IP> \
421
+ --end-ip-address <SEU_IP>
422
+
423
+ # Obter seu IP público
424
+ curl https://api.ipify.org
425
+ ```
426
+
427
+ ### Erro: "Access denied to Key Vault"
428
+
429
+ **Causa**: Sem permissão no Key Vault.
430
+
431
+ **Solução** (Tech lead):
432
+ ```bash
433
+ az keyvault set-policy \
434
+ --name kv-myapp-staging \
435
+ --upn <developer-email> \
436
+ --secret-permissions get list
437
+ ```
438
+
439
+ ### Erro: "Blob not found"
440
+
441
+ **Causa**: Container não existe ou sem permissão.
442
+
443
+ **Solução** (Tech lead):
444
+ ```bash
445
+ # Dar permissão Storage Blob Data Contributor
446
+ az role assignment create \
447
+ --assignee <developer-email> \
448
+ --role "Storage Blob Data Contributor" \
449
+ --scope "/subscriptions/<SUB_ID>/resourceGroups/rg-myapp-staging/providers/Microsoft.Storage/storageAccounts/stmyappstaging"
450
+ ```
451
+
452
+ ### Erro: "DefaultAzureCredential failed"
453
+
454
+ **Causa**: Nenhum método de autenticação funcionou.
455
+
456
+ **Solução**:
457
+ 1. Feche Visual Studio/Rider/VS Code
458
+ 2. `az logout && az login`
459
+ 3. Reabra IDE
460
+ 4. Rode novamente
461
+
462
+ ### Migration falhou
463
+
464
+ **Causa**: Conflito com migration de outro developer.
465
+
466
+ **Solução**:
467
+ ```bash
468
+ # Pull latest
469
+ git pull origin staging
470
+
471
+ # Deletar sua migration
472
+ dotnet ef migrations remove
473
+
474
+ # Aplicar migrations remotas
475
+ dotnet ef database update
476
+
477
+ # Recriar sua migration
478
+ dotnet ef migrations add SuaMigration
479
+ ```
480
+
481
+ ---
482
+
483
+ ## ✅ Checklist de Setup
484
+
485
+ Para novos developers:
486
+
487
+ - [ ] Instalar .NET SDK 8
488
+ - [ ] Instalar Azure CLI
489
+ - [ ] Instalar Git
490
+ - [ ] Instalar IDE (VS/Rider/VS Code)
491
+ - [ ] `az login` com conta corporativa
492
+ - [ ] Solicitar permissões staging (tech lead)
493
+ - [ ] Clonar repositório
494
+ - [ ] Criar `appsettings.Development.json`
495
+ - [ ] Testar conexão SQL (Azure Data Studio)
496
+ - [ ] Rodar projeto local (`dotnet run`)
497
+ - [ ] Acessar `https://localhost:7001`
498
+ - [ ] Rodar testes (`dotnet test`)
499
+ - [ ] Criar feature branch de teste
500
+ - [ ] Fazer commit de teste
501
+ - [ ] Criar PR de teste
502
+ - [ ] Deletar PR/branch de teste
503
+
504
+ ---
505
+
506
+ ## 📚 Referências
507
+
508
+ - [DefaultAzureCredential](https://learn.microsoft.com/dotnet/api/azure.identity.defaultazurecredential)
509
+ - [Azure SQL with Azure AD](https://learn.microsoft.com/azure/azure-sql/database/authentication-aad-configure)
510
+ - [Azure Storage SDK](https://learn.microsoft.com/azure/storage/blobs/storage-quickstart-blobs-dotnet)
511
+ - [Entity Framework Migrations](https://learn.microsoft.com/ef/core/managing-schemas/migrations/)
512
+
513
+ ---
514
+
515
+ *MORPH-SPEC by Polymorphism Tech*