@polymorphism-tech/morph-spec 4.7.0 → 4.7.2

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 (232) hide show
  1. package/.morph/.morphversion +5 -0
  2. package/.morph/analytics/threads-log.jsonl +5 -0
  3. package/.morph/config/config.json +8 -0
  4. package/.morph/framework/agents.json +1815 -0
  5. package/.morph/framework/hooks/README.md +205 -0
  6. package/.morph/framework/hooks/claude-code/notification/approval-reminder.js +54 -0
  7. package/.morph/framework/hooks/claude-code/post-tool-use/dispatch.js +83 -0
  8. package/.morph/framework/hooks/claude-code/post-tool-use/handle-tool-failure.js +42 -0
  9. package/.morph/framework/hooks/claude-code/pre-compact/save-morph-context.js +61 -0
  10. package/.morph/framework/hooks/claude-code/pre-tool-use/enforce-phase-writes.js +71 -0
  11. package/.morph/framework/hooks/claude-code/pre-tool-use/protect-readonly-files.js +58 -0
  12. package/.morph/framework/hooks/claude-code/pre-tool-use/protect-spec-files.js +64 -0
  13. package/.morph/framework/hooks/claude-code/session-start/inject-morph-context.js +94 -0
  14. package/.morph/framework/hooks/claude-code/statusline.py +538 -0
  15. package/.morph/framework/hooks/claude-code/statusline.sh +7 -0
  16. package/.morph/framework/hooks/claude-code/stop/validate-completion.js +88 -0
  17. package/.morph/framework/hooks/claude-code/user-prompt/enrich-prompt.js +91 -0
  18. package/.morph/framework/hooks/git/commit-msg/conventional-commits.sh +33 -0
  19. package/.morph/framework/hooks/git/pre-commit/agents.sh +25 -0
  20. package/.morph/framework/hooks/git/pre-commit/orchestrator.sh +64 -0
  21. package/.morph/framework/hooks/git/pre-commit/specs.sh +50 -0
  22. package/.morph/framework/hooks/git/pre-push/run-tests.sh +44 -0
  23. package/.morph/framework/hooks/shared/hook-response.js +45 -0
  24. package/.morph/framework/hooks/shared/phase-utils.js +129 -0
  25. package/.morph/framework/hooks/shared/state-reader.js +138 -0
  26. package/.morph/framework/hooks/shared/stdin-reader.js +26 -0
  27. package/.morph/framework/standards/STANDARDS.json +933 -0
  28. package/.morph/framework/standards/ai-agents/blazor-ui.md +364 -0
  29. package/.morph/framework/standards/ai-agents/production.md +415 -0
  30. package/.morph/framework/standards/ai-agents/setup.md +418 -0
  31. package/.morph/framework/standards/ai-agents/team-orchestration.md +479 -0
  32. package/.morph/framework/standards/ai-agents/workflows.md +354 -0
  33. package/.morph/framework/standards/architecture/ddd/aggregates.md +120 -0
  34. package/.morph/framework/standards/architecture/ddd/bounded-contexts.md +105 -0
  35. package/.morph/framework/standards/architecture/ddd/complexity-levels.md +108 -0
  36. package/.morph/framework/standards/architecture/ddd/entities.md +99 -0
  37. package/.morph/framework/standards/architecture/ddd/ubiquitous-language.md +58 -0
  38. package/.morph/framework/standards/architecture/ddd/value-objects.md +124 -0
  39. package/.morph/framework/standards/backend/api/minimal-api.md +494 -0
  40. package/.morph/framework/standards/backend/api/rest.md +492 -0
  41. package/.morph/framework/standards/backend/api/validation.md +88 -0
  42. package/.morph/framework/standards/backend/authentication/passkeys.md +428 -0
  43. package/.morph/framework/standards/backend/database/ef-core.md +199 -0
  44. package/.morph/framework/standards/backend/database/migrations.md +393 -0
  45. package/.morph/framework/standards/backend/database/postgresql/database.md +352 -0
  46. package/.morph/framework/standards/backend/database/repository-patterns.md +528 -0
  47. package/.morph/framework/standards/backend/database/vector-search-rag.md +541 -0
  48. package/.morph/framework/standards/backend/dotnet/async.md +366 -0
  49. package/.morph/framework/standards/backend/dotnet/core.md +117 -0
  50. package/.morph/framework/standards/backend/dotnet/di.md +439 -0
  51. package/.morph/framework/standards/backend/dotnet/program-cs-checklist.md +92 -0
  52. package/.morph/framework/standards/backend/integrations/asaas/asaas-api.md +216 -0
  53. package/.morph/framework/standards/backend/integrations/clerk/clerk-auth.md +290 -0
  54. package/.morph/framework/standards/backend/integrations/hangfire/hangfire-jobs.md +350 -0
  55. package/.morph/framework/standards/backend/integrations/resend/resend-email.md +385 -0
  56. package/.morph/framework/standards/context/analytics.md +96 -0
  57. package/.morph/framework/standards/context/bundles.md +110 -0
  58. package/.morph/framework/standards/context/priming.md +78 -0
  59. package/.morph/framework/standards/core/architecture.md +185 -0
  60. package/.morph/framework/standards/core/coding.md +214 -0
  61. package/.morph/framework/standards/core/git-branching-strategy.md +403 -0
  62. package/.morph/framework/standards/core/git.md +185 -0
  63. package/.morph/framework/standards/core/testing.md +295 -0
  64. package/.morph/framework/standards/data/nosql/blob-storage.md +102 -0
  65. package/.morph/framework/standards/data/nosql/cache/redis.md +97 -0
  66. package/.morph/framework/standards/data/nosql/cosmos-db.md +118 -0
  67. package/.morph/framework/standards/data/vector-search/azure-ai-search.md +121 -0
  68. package/.morph/framework/standards/data/vector-search/rag-chunking.md +104 -0
  69. package/.morph/framework/standards/frontend/blazor/design-checklist.md +222 -0
  70. package/.morph/framework/standards/frontend/blazor/fluent-ui-setup.md +595 -0
  71. package/.morph/framework/standards/frontend/blazor/fluent-ui.md +137 -0
  72. package/.morph/framework/standards/frontend/blazor/html-conversion.md +184 -0
  73. package/.morph/framework/standards/frontend/blazor/lifecycle.md +195 -0
  74. package/.morph/framework/standards/frontend/blazor/pitfalls.md +198 -0
  75. package/.morph/framework/standards/frontend/blazor/state.md +191 -0
  76. package/.morph/framework/standards/frontend/design-system/animations.md +151 -0
  77. package/.morph/framework/standards/frontend/design-system/naming.md +64 -0
  78. package/.morph/framework/standards/frontend/nextjs/app-router.md +123 -0
  79. package/.morph/framework/standards/frontend/nextjs/components.md +132 -0
  80. package/.morph/framework/standards/frontend/nextjs/data-fetching.md +126 -0
  81. package/.morph/framework/standards/frontend/nextjs/forms.md +128 -0
  82. package/.morph/framework/standards/frontend/nextjs/naming-conventions.md +67 -0
  83. package/.morph/framework/standards/frontend/nextjs/nextjs-patterns.md +215 -0
  84. package/.morph/framework/standards/frontend/nextjs/project-structure.md +102 -0
  85. package/.morph/framework/standards/frontend/nextjs/state-management.md +72 -0
  86. package/.morph/framework/standards/frontend/nextjs/testing.md +111 -0
  87. package/.morph/framework/standards/infrastructure/azure/azure.md +624 -0
  88. package/.morph/framework/standards/infrastructure/azure/bicep/bicep-patterns.md +422 -0
  89. package/.morph/framework/standards/infrastructure/azure/devops/azure-devops-setup.md +516 -0
  90. package/.morph/framework/standards/infrastructure/azure/devops/local-development.md +520 -0
  91. package/.morph/framework/standards/infrastructure/azure/services/functions.md +486 -0
  92. package/.morph/framework/standards/infrastructure/azure/services/service-bus.md +459 -0
  93. package/.morph/framework/standards/infrastructure/azure/services/storage.md +407 -0
  94. package/.morph/framework/standards/infrastructure/docker/easypanel-deploy.md +196 -0
  95. package/.morph/framework/standards/infrastructure/supabase/mcp-setup.md +252 -0
  96. package/.morph/framework/standards/infrastructure/supabase/supabase-auth.md +176 -0
  97. package/.morph/framework/standards/infrastructure/supabase/supabase-pgvector.md +169 -0
  98. package/.morph/framework/standards/infrastructure/supabase/supabase-rls.md +184 -0
  99. package/.morph/framework/standards/infrastructure/supabase/supabase-storage.md +153 -0
  100. package/.morph/framework/standards/integration/api/graphql.md +91 -0
  101. package/.morph/framework/standards/integration/api/grpc.md +114 -0
  102. package/.morph/framework/standards/integration/api/rest-design.md +95 -0
  103. package/.morph/framework/standards/integration/event-driven/cqrs.md +101 -0
  104. package/.morph/framework/standards/integration/event-driven/event-sourcing.md +124 -0
  105. package/.morph/framework/standards/integration/event-driven/service-bus.md +95 -0
  106. package/.morph/framework/standards/integration/mcp/mcp-tools.md +384 -0
  107. package/.morph/framework/standards/observability/logging.md +131 -0
  108. package/.morph/framework/standards/observability/metrics.md +121 -0
  109. package/.morph/framework/standards/observability/monitoring.md +114 -0
  110. package/.morph/framework/standards/observability/tracing.md +132 -0
  111. package/.morph/framework/standards/workflows/parallel-execution.md +112 -0
  112. package/.morph/framework/standards/workflows/thread-management.md +113 -0
  113. package/.morph/framework/templates/.idea/morph-templates.xml +92 -0
  114. package/.morph/framework/templates/.vscode/morph-templates.code-snippets +186 -0
  115. package/.morph/framework/templates/IDE-SNIPPETS.md +266 -0
  116. package/.morph/framework/templates/README.md +814 -0
  117. package/.morph/framework/templates/REGISTRY.json +1888 -0
  118. package/.morph/framework/templates/code/dotnet/backend/repository.cs +141 -0
  119. package/.morph/framework/templates/code/dotnet/backend/service.cs +139 -0
  120. package/.morph/framework/templates/code/dotnet/contracts/Commands.cs +74 -0
  121. package/.morph/framework/templates/code/dotnet/contracts/Entities.cs +25 -0
  122. package/.morph/framework/templates/code/dotnet/contracts/Queries.cs +74 -0
  123. package/.morph/framework/templates/code/dotnet/contracts/README.md +74 -0
  124. package/.morph/framework/templates/code/dotnet/contracts/api-contracts.cs +173 -0
  125. package/.morph/framework/templates/code/dotnet/contracts/contracts-level1.cs +69 -0
  126. package/.morph/framework/templates/code/dotnet/contracts/contracts-level2.cs +86 -0
  127. package/.morph/framework/templates/code/dotnet/contracts/contracts-level3.cs +41 -0
  128. package/.morph/framework/templates/code/dotnet/database/migration.cs +83 -0
  129. package/.morph/framework/templates/code/dotnet/frontend/component.razor +239 -0
  130. package/.morph/framework/templates/code/dotnet/jobs/agent.cs +163 -0
  131. package/.morph/framework/templates/code/dotnet/jobs/job.cs +171 -0
  132. package/.morph/framework/templates/code/dotnet/test.cs +239 -0
  133. package/.morph/framework/templates/code/sql/rls-policy.sql +57 -0
  134. package/.morph/framework/templates/code/sql/supabase-migration.sql +100 -0
  135. package/.morph/framework/templates/code/sql/supabase-migration.template.sql +113 -0
  136. package/.morph/framework/templates/code/typescript/contracts.ts +168 -0
  137. package/.morph/framework/templates/context/CONTEXT-FEATURE.md +276 -0
  138. package/.morph/framework/templates/context/CONTEXT.md +181 -0
  139. package/.morph/framework/templates/docs/clarifications.md +253 -0
  140. package/.morph/framework/templates/docs/onboarding.md +123 -0
  141. package/.morph/framework/templates/docs/proposal.md +182 -0
  142. package/.morph/framework/templates/docs/schema-analysis.md +119 -0
  143. package/.morph/framework/templates/docs/spec.md +198 -0
  144. package/.morph/framework/templates/docs/ui-components.md +124 -0
  145. package/.morph/framework/templates/docs/ui-design-system.md +76 -0
  146. package/.morph/framework/templates/docs/ui-flows.md +167 -0
  147. package/.morph/framework/templates/docs/ui-mockups.md +98 -0
  148. package/.morph/framework/templates/docs/user-stories.md +34 -0
  149. package/.morph/framework/templates/examples/design-system-examples.md +357 -0
  150. package/.morph/framework/templates/examples/spec-examples.md +90 -0
  151. package/.morph/framework/templates/feature/decisions.md +187 -0
  152. package/.morph/framework/templates/feature/recap.md +146 -0
  153. package/.morph/framework/templates/feature/tasks.md +199 -0
  154. package/.morph/framework/templates/frontend/nextjs/Dockerfile.nextjs.hbs +43 -0
  155. package/.morph/framework/templates/frontend/nextjs/client-component.tsx.hbs +26 -0
  156. package/.morph/framework/templates/frontend/nextjs/env.mjs.hbs +32 -0
  157. package/.morph/framework/templates/frontend/nextjs/feature-form.tsx.hbs +56 -0
  158. package/.morph/framework/templates/frontend/nextjs/page.tsx.hbs +22 -0
  159. package/.morph/framework/templates/frontend/nextjs/tsconfig.json.hbs +26 -0
  160. package/.morph/framework/templates/frontend/nextjs/use-feature.ts.hbs +54 -0
  161. package/.morph/framework/templates/infrastructure/azure/Dockerfile.example +82 -0
  162. package/.morph/framework/templates/infrastructure/azure/README.md +286 -0
  163. package/.morph/framework/templates/infrastructure/azure/app-insights.bicep +63 -0
  164. package/.morph/framework/templates/infrastructure/azure/app-service.bicep +164 -0
  165. package/.morph/framework/templates/infrastructure/azure/container-app-env.bicep +49 -0
  166. package/.morph/framework/templates/infrastructure/azure/container-app.bicep +156 -0
  167. package/.morph/framework/templates/infrastructure/azure/deploy-checklist.md +426 -0
  168. package/.morph/framework/templates/infrastructure/azure/deploy.ps1 +229 -0
  169. package/.morph/framework/templates/infrastructure/azure/deploy.sh +208 -0
  170. package/.morph/framework/templates/infrastructure/azure/key-vault.bicep +91 -0
  171. package/.morph/framework/templates/infrastructure/azure/main.bicep +189 -0
  172. package/.morph/framework/templates/infrastructure/azure/parameters.dev.json +29 -0
  173. package/.morph/framework/templates/infrastructure/azure/parameters.prod.json +29 -0
  174. package/.morph/framework/templates/infrastructure/azure/parameters.staging.json +29 -0
  175. package/.morph/framework/templates/infrastructure/azure/sql-database.bicep +103 -0
  176. package/.morph/framework/templates/infrastructure/azure/storage.bicep +106 -0
  177. package/.morph/framework/templates/infrastructure/docker/Dockerfile.template +58 -0
  178. package/.morph/framework/templates/infrastructure/docker/docker-compose.template.yml +67 -0
  179. package/.morph/framework/templates/infrastructure/docker/dockerfile-api.dockerfile +38 -0
  180. package/.morph/framework/templates/infrastructure/docker/dockerfile-web.dockerfile +48 -0
  181. package/.morph/framework/templates/infrastructure/docker/easypanel.template.json +54 -0
  182. package/.morph/framework/templates/infrastructure/github/README.md +593 -0
  183. package/.morph/framework/templates/infrastructure/github/actions/azure-auth/action.yml.hbs +22 -0
  184. package/.morph/framework/templates/infrastructure/github/actions/docker-build-push/action.yml.hbs +45 -0
  185. package/.morph/framework/templates/infrastructure/github/actions/health-check/action.yml.hbs +27 -0
  186. package/.morph/framework/templates/infrastructure/github/workflows/deploy-azure-app-service.yml.hbs +61 -0
  187. package/.morph/framework/templates/infrastructure/github/workflows/deploy-easypanel.yml.hbs +31 -0
  188. package/.morph/framework/templates/infrastructure/github/workflows/docker-build-push.yml.hbs +59 -0
  189. package/.morph/framework/templates/infrastructure/github/workflows/dotnet-build.yml.hbs +39 -0
  190. package/.morph/framework/templates/integrations/asaas-client.cs +387 -0
  191. package/.morph/framework/templates/integrations/asaas-webhook.cs +351 -0
  192. package/.morph/framework/templates/integrations/azure-identity-config.cs +288 -0
  193. package/.morph/framework/templates/integrations/clerk-config.cs +258 -0
  194. package/.morph/framework/templates/meta-prompts/fusion/fusion-agent.md +76 -0
  195. package/.morph/framework/templates/meta-prompts/fusion/fusion-aggregator.md +100 -0
  196. package/.morph/framework/templates/meta-prompts/hops/hop-retry.md +78 -0
  197. package/.morph/framework/templates/meta-prompts/hops/hop-validation.md +97 -0
  198. package/.morph/framework/templates/meta-prompts/hops/hop-wrapper.md +36 -0
  199. package/.morph/framework/templates/meta-prompts/parallel-workers/parallel-coordinator.md +113 -0
  200. package/.morph/framework/templates/meta-prompts/parallel-workers/parallel-worker.md +80 -0
  201. package/.morph/framework/templates/meta-prompts/squad-leaders/backend-squad.md +90 -0
  202. package/.morph/framework/templates/meta-prompts/squad-leaders/frontend-squad.md +126 -0
  203. package/.morph/framework/templates/meta-prompts/squad-leaders/squad-leader.md +43 -0
  204. package/.morph/framework/templates/meta-prompts/validators/checkpoint-validator.md +107 -0
  205. package/.morph/framework/templates/meta-prompts/validators/pre-commit-validator.md +95 -0
  206. package/.morph/framework/templates/project-structure/dotnet-ddd.md +70 -0
  207. package/.morph/framework/templates/saas/subscription.cs +347 -0
  208. package/.morph/framework/templates/saas/tenant.cs +338 -0
  209. package/.morph/framework/templates/state.template.json +17 -0
  210. package/.morph/framework/templates/ui/FluentDesignTheme.cs +149 -0
  211. package/.morph/framework/templates/ui/MudTheme.cs +281 -0
  212. package/.morph/framework/templates/ui/design-system.css +226 -0
  213. package/.morph/logs/tool-failures.log +17 -0
  214. package/.morph/memory/pre-compact-2026-02-24T17-43-30-049Z.json +16 -0
  215. package/.morph/plans/eager-watching-bunny.md +105 -0
  216. package/.morph/plans/temporal-seeking-nebula.md +45 -0
  217. package/.morph/state.json +48 -0
  218. package/CLAUDE.md +1 -1
  219. package/README.md +119 -99
  220. package/bin/morph-spec.js +0 -9
  221. package/framework/CLAUDE.md +1 -1
  222. package/framework/hooks/README.md +10 -6
  223. package/framework/hooks/claude-code/notification/approval-reminder.js +2 -0
  224. package/framework/hooks/claude-code/post-tool-use/dispatch.js +1 -1
  225. package/framework/hooks/claude-code/stop/validate-completion.js +1 -1
  226. package/framework/hooks/claude-code/user-prompt/enrich-prompt.js +1 -1
  227. package/package.json +1 -1
  228. package/src/commands/project/init.js +15 -42
  229. package/src/commands/project/update.js +22 -37
  230. package/src/lib/installers/mcp-installer.js +18 -3
  231. package/src/utils/hooks-installer.js +5 -15
  232. package/src/commands/project/detect.js +0 -114
@@ -0,0 +1,428 @@
1
+ # Passkeys/WebAuthn - Autenticação Sem Senhas (.NET 10)
2
+
3
+ > **Scope:** blazor-azure
4
+ > **Layer:** 2 (on keyword)
5
+ > **Keywords:** passkeys, azure entra, fido2, webauthn, authentication
6
+ > **Load When:** passkeys or authentication keywords detected
7
+
8
+ **Novidade .NET 10:** Suporte nativo a Passkeys/WebAuthn integrado no ASP.NET Core Identity.
9
+
10
+ ---
11
+
12
+ ## 🎯 O Que São Passkeys?
13
+
14
+ **Passkeys** são credenciais criptográficas que substituem senhas tradicionais.
15
+
16
+ ### Características
17
+
18
+ | Aspecto | Descrição |
19
+ |---------|-----------|
20
+ | **Segurança** | Resistentes a phishing, não podem ser roubadas |
21
+ | **Usabilidade** | Login com biometria ou PIN do dispositivo |
22
+ | **Privacidade** | Chave privada nunca sai do dispositivo |
23
+ | **Padrão** | WebAuthn (W3C) + FIDO2 |
24
+
25
+ ### Como Funciona
26
+
27
+ ```
28
+ 1. Usuário solicita criação de passkey
29
+ 2. Sistema gera par de chaves (pública/privada)
30
+ 3. Chave pública → Servidor
31
+ 4. Chave privada → Authenticator (Windows Hello, smartphone, security key)
32
+ 5. Login: Servidor desafia → Authenticator assina → Servidor verifica
33
+ ```
34
+
35
+ **Authenticators suportados:**
36
+ - Windows Hello
37
+ - Face ID / Touch ID (Apple)
38
+ - Biometria Android
39
+ - Security keys (YubiKey, etc.)
40
+
41
+ ---
42
+
43
+ ## 🚀 Setup em Blazor Web App
44
+
45
+ ### 1. Criar Projeto com Identity
46
+
47
+ ```bash
48
+ dotnet new blazor --auth Individual -o MyApp
49
+ cd MyApp
50
+ ```
51
+
52
+ **Importante:** Escolher "Individual User Accounts" habilita passkeys automaticamente.
53
+
54
+ ### 2. Estrutura Gerada
55
+
56
+ ```
57
+ MyApp/
58
+ ├── Components/
59
+ │ └── Account/
60
+ │ ├── Pages/
61
+ │ │ ├── Manage/
62
+ │ │ │ └── Passkeys.razor ← Gerenciamento de passkeys
63
+ │ │ ├── Login.razor ← Login com passkey
64
+ │ │ └── Register.razor
65
+ │ └── IdentityUserAccessor.cs
66
+ ├── Data/
67
+ │ ├── ApplicationDbContext.cs
68
+ │ └── ApplicationUser.cs
69
+ └── Program.cs
70
+ ```
71
+
72
+ ### 3. Configuração Automática no Program.cs
73
+
74
+ ```csharp
75
+ // Já vem configurado no template
76
+ builder.Services.AddIdentityCore<ApplicationUser>(options =>
77
+ {
78
+ options.SignIn.RequireConfirmedAccount = true;
79
+
80
+ // Passkeys habilitados por padrão
81
+ options.Stores.MaxLengthForKeys = 128;
82
+ })
83
+ .AddEntityFrameworkStores<ApplicationDbContext>()
84
+ .AddSignInManager()
85
+ .AddDefaultTokenProviders();
86
+
87
+ // WebAuthn/Passkeys configurado automaticamente
88
+ builder.Services.AddAuthentication(options =>
89
+ {
90
+ options.DefaultScheme = IdentityConstants.ApplicationScheme;
91
+ options.DefaultSignInScheme = IdentityConstants.ExternalScheme;
92
+ })
93
+ .AddIdentityCookies();
94
+ ```
95
+
96
+ ---
97
+
98
+ ## 📱 Fluxo de Uso
99
+
100
+ ### Criar Passkey
101
+
102
+ 1. Usuário loga com email/senha (primeira vez)
103
+ 2. Acessa perfil → Aba "Passkeys"
104
+ 3. Clica "Add Passkey"
105
+ 4. Sistema solicita authenticator (Windows Hello, celular, etc.)
106
+ 5. Usuário confirma identidade (biometria/PIN)
107
+ 6. Passkey é criada e nomeada
108
+ 7. Salva no banco de dados
109
+
110
+ ### Login com Passkey
111
+
112
+ 1. Usuário acessa página de login
113
+ 2. Clica "Login with Passkey"
114
+ 3. Sistema envia desafio
115
+ 4. Authenticator assina desafio
116
+ 5. Servidor verifica assinatura
117
+ 6. Usuário autenticado
118
+
119
+ ---
120
+
121
+ ## 💻 Implementação Customizada
122
+
123
+ ### Adicionar Passkeys a Projeto Existente
124
+
125
+ Se você tem um projeto Blazor sem passkeys, use o **scaffolder**:
126
+
127
+ ```bash
128
+ dotnet tool install -g dotnet-aspnet-codegenerator
129
+
130
+ dotnet aspnet-codegenerator identity \
131
+ --useDefaultUI \
132
+ --dbContext ApplicationDbContext \
133
+ --files "Account.Manage.Passkeys;Account.Login"
134
+ ```
135
+
136
+ Isso adiciona:
137
+ - `Components/Account/Pages/Manage/Passkeys.razor`
138
+ - `Components/Account/Pages/Login.razor` (atualizado)
139
+
140
+ ### Verificar Suporte a Passkeys
141
+
142
+ ```csharp
143
+ @inject IPasskeyService PasskeyService
144
+
145
+ @code {
146
+ private bool _supportsPasskeys;
147
+
148
+ protected override async Task OnInitializedAsync()
149
+ {
150
+ _supportsPasskeys = await PasskeyService.IsSupportedAsync();
151
+ }
152
+ }
153
+ ```
154
+
155
+ ### Listar Passkeys do Usuário
156
+
157
+ ```csharp
158
+ @page "/manage/passkeys"
159
+ @inject IPasskeyService PasskeyService
160
+ @inject UserManager<ApplicationUser> UserManager
161
+
162
+ <h3>Minhas Passkeys</h3>
163
+
164
+ @if (_passkeys is null)
165
+ {
166
+ <p>Carregando...</p>
167
+ }
168
+ else if (!_passkeys.Any())
169
+ {
170
+ <p>Você não tem passkeys configuradas.</p>
171
+ }
172
+ else
173
+ {
174
+ <ul>
175
+ @foreach (var passkey in _passkeys)
176
+ {
177
+ <li>
178
+ @passkey.Name (@passkey.CreatedAt.ToString("dd/MM/yyyy"))
179
+ <button @onclick="() => RemovePasskey(passkey.Id)">Remover</button>
180
+ </li>
181
+ }
182
+ </ul>
183
+ }
184
+
185
+ <button @onclick="AddPasskey">Adicionar Passkey</button>
186
+
187
+ @code {
188
+ private List<Passkey>? _passkeys;
189
+
190
+ protected override async Task OnInitializedAsync()
191
+ {
192
+ var user = await UserManager.GetUserAsync(User);
193
+ _passkeys = await PasskeyService.GetUserPasskeysAsync(user!.Id);
194
+ }
195
+
196
+ private async Task AddPasskey()
197
+ {
198
+ var user = await UserManager.GetUserAsync(User);
199
+ await PasskeyService.CreatePasskeyAsync(user!.Id);
200
+ await OnInitializedAsync(); // Recarregar lista
201
+ }
202
+
203
+ private async Task RemovePasskey(string passkeyId)
204
+ {
205
+ await PasskeyService.RemovePasskeyAsync(passkeyId);
206
+ await OnInitializedAsync(); // Recarregar lista
207
+ }
208
+ }
209
+ ```
210
+
211
+ ---
212
+
213
+ ## 🗄️ Modelo de Dados
214
+
215
+ ### Entidade Passkey
216
+
217
+ ```csharp
218
+ public class Passkey
219
+ {
220
+ public string Id { get; set; } = Guid.NewGuid().ToString();
221
+ public string UserId { get; set; } = null!;
222
+ public string Name { get; set; } = "Passkey";
223
+ public byte[] CredentialId { get; set; } = Array.Empty<byte>();
224
+ public byte[] PublicKey { get; set; } = Array.Empty<byte>();
225
+ public int SignatureCounter { get; set; }
226
+ public DateTime CreatedAt { get; set; } = DateTime.UtcNow;
227
+ public DateTime? LastUsedAt { get; set; }
228
+
229
+ // Relacionamento
230
+ public ApplicationUser User { get; set; } = null!;
231
+ }
232
+ ```
233
+
234
+ ### DbContext
235
+
236
+ ```csharp
237
+ public class ApplicationDbContext : IdentityDbContext<ApplicationUser>
238
+ {
239
+ public DbSet<Passkey> Passkeys { get; set; }
240
+
241
+ protected override void OnModelCreating(ModelBuilder builder)
242
+ {
243
+ base.OnModelCreating(builder);
244
+
245
+ builder.Entity<Passkey>(entity =>
246
+ {
247
+ entity.HasKey(p => p.Id);
248
+ entity.HasIndex(p => p.UserId);
249
+ entity.HasIndex(p => p.CredentialId).IsUnique();
250
+
251
+ entity.HasOne(p => p.User)
252
+ .WithMany()
253
+ .HasForeignKey(p => p.UserId)
254
+ .OnDelete(DeleteBehavior.Cascade);
255
+ });
256
+ }
257
+ }
258
+ ```
259
+
260
+ ### Migration
261
+
262
+ ```bash
263
+ dotnet ef migrations add AddPasskeys
264
+ dotnet ef database update
265
+ ```
266
+
267
+ ---
268
+
269
+ ## 🔒 Segurança
270
+
271
+ ### Configurações Recomendadas
272
+
273
+ ```csharp
274
+ builder.Services.AddAuthentication()
275
+ .AddWebAuthn(options =>
276
+ {
277
+ // Nome do site (aparece no authenticator)
278
+ options.RelyingPartyId = "myapp.com";
279
+ options.RelyingPartyName = "My Application";
280
+
281
+ // Origem permitida
282
+ options.Origins = new[] { "https://myapp.com" };
283
+
284
+ // Timeout de autenticação
285
+ options.Timeout = TimeSpan.FromSeconds(60);
286
+
287
+ // Tipo de authenticator
288
+ options.AuthenticatorSelection = new()
289
+ {
290
+ RequireResidentKey = false,
291
+ UserVerification = UserVerificationRequirement.Preferred
292
+ };
293
+ });
294
+ ```
295
+
296
+ ### User Verification
297
+
298
+ | Modo | Descrição | Quando Usar |
299
+ |------|-----------|-------------|
300
+ | `Required` | Exige biometria/PIN sempre | Alto risco |
301
+ | `Preferred` | Prefere biometria mas aceita alternativa | Padrão recomendado |
302
+ | `Discouraged` | Não solicita verificação | Não use |
303
+
304
+ ### Attestation
305
+
306
+ ```csharp
307
+ options.Attestation = AttestationConveyancePreference.None;
308
+ ```
309
+
310
+ **Valores:**
311
+ - `None`: Não requer attestation (mais compatível)
312
+ - `Indirect`: Valida authenticator via CA
313
+ - `Direct`: Requer attestation direta (mais restritivo)
314
+
315
+ **Recomendação:** Use `None` para máxima compatibilidade.
316
+
317
+ ---
318
+
319
+ ## 🧪 Testando Passkeys
320
+
321
+ ### Ambiente de Desenvolvimento
322
+
323
+ 1. **HTTPS obrigatório:** WebAuthn só funciona com HTTPS (ou localhost)
324
+ 2. **Windows Hello:** Habilite no Windows
325
+ 3. **Chrome DevTools:** Use Virtual Authenticator para testes
326
+
327
+ ### Virtual Authenticator (Chrome)
328
+
329
+ 1. F12 → More Tools → WebAuthn
330
+ 2. Enable virtual authenticator environment
331
+ 3. Add authenticator (escolha tipo: USB, NFC, Internal)
332
+ 4. Teste criação e login
333
+
334
+ ### Smartphone como Authenticator
335
+
336
+ 1. Use HTTPS (não localhost)
337
+ 2. Escaneie QR code gerado
338
+ 3. Confirme com biometria do celular
339
+
340
+ ---
341
+
342
+ ## 🐛 Troubleshooting
343
+
344
+ ### Erro: "WebAuthn not supported"
345
+
346
+ **Causa:** Navegador antigo ou HTTP (não HTTPS)
347
+
348
+ **Solução:**
349
+ - Use HTTPS em produção
350
+ - Localhost funciona com HTTP (somente dev)
351
+ - Atualize navegador
352
+
353
+ ### Erro: "User verification failed"
354
+
355
+ **Causa:** Authenticator não configurado ou cancelado
356
+
357
+ **Solução:**
358
+ - Configure Windows Hello / Touch ID
359
+ - Tente outro authenticator
360
+ - Verifique `UserVerification = Preferred`
361
+
362
+ ### Passkey não aparece em outro dispositivo
363
+
364
+ **Causa:** Passkeys não sincronizam automaticamente (depende do authenticator)
365
+
366
+ **Solução:**
367
+ - Use authenticator com sync (ex: Google Password Manager)
368
+ - Ou crie passkey separada por dispositivo
369
+
370
+ ---
371
+
372
+ ## 📊 Migração Gradual
373
+
374
+ ### Permitir Email/Senha + Passkeys
375
+
376
+ ```csharp
377
+ // Login.razor
378
+ <EditForm Model="Input" OnValidSubmit="LoginAsync">
379
+ <InputText @bind-Value="Input.Email" />
380
+ <InputText @bind-Value="Input.Password" type="password" />
381
+ <button type="submit">Login with Password</button>
382
+ </EditForm>
383
+
384
+ <hr />
385
+
386
+ <button @onclick="LoginWithPasskey">Login with Passkey</button>
387
+
388
+ @code {
389
+ private async Task LoginAsync()
390
+ {
391
+ // Login tradicional com senha
392
+ }
393
+
394
+ private async Task LoginWithPasskey()
395
+ {
396
+ // Login com WebAuthn
397
+ }
398
+ }
399
+ ```
400
+
401
+ **Estratégia:** Mantenha senha como fallback, incentive passkeys.
402
+
403
+ ---
404
+
405
+ ## ✅ Checklist de Implementação
406
+
407
+ - [ ] Projeto criado com `--auth Individual` ou scaffolder usado
408
+ - [ ] DbContext inclui tabela `Passkeys`
409
+ - [ ] Migration executada
410
+ - [ ] HTTPS habilitado (dev e prod)
411
+ - [ ] Página de gerenciamento de passkeys funcional
412
+ - [ ] Login com passkey funcional
413
+ - [ ] Fallback para email/senha disponível
414
+ - [ ] Testado com Windows Hello ou smartphone
415
+ - [ ] `RelyingPartyId` configurado corretamente
416
+
417
+ ---
418
+
419
+ ## 📚 Referências
420
+
421
+ - [WebAuthn Specification (W3C)](https://w3c.github.io/webauthn/)
422
+ - [FIDO Alliance](https://fidoalliance.org/)
423
+ - [ASP.NET Core Identity - Passkeys](https://learn.microsoft.com/aspnet/core/security/authentication/identity/passkeys)
424
+ - [Chrome WebAuthn DevTools](https://developer.chrome.com/docs/devtools/webauthn/)
425
+
426
+ ---
427
+
428
+ *MORPH-SPEC by Polymorphism Tech*
@@ -0,0 +1,199 @@
1
+ # Blazor Server + EF Core
2
+
3
+ > **Scope:** blazor-azure,nextjs-supabase
4
+ > **Layer:** 1 (on domain)
5
+ > **Keywords:** efcore, entity framework, dbcontext, dbfactory, concurrency
6
+ > **Load When:** database work
7
+
8
+ Patterns for Blazor Server + EF Core concurrency and background operations.
9
+
10
+ ---
11
+
12
+ ## Core Problem: DbContext Concurrency
13
+
14
+ Blazor Server scoped services = circuit lifecycle (SignalR), NOT HTTP requests.
15
+ `Task.Run` or fire-and-forget share the same scoped DbContext → concurrent access violations.
16
+
17
+ **Rule:** Separate thread from HTTP request = isolated DbContext via `IDbContextFactory`.
18
+
19
+ | Scenario | Pattern |
20
+ |----------|---------|
21
+ | Regular component ops | Scoped DbContext/Repository |
22
+ | `Task.Run` / fire-and-forget | IDbContextFactory |
23
+ | Hangfire / background jobs | IDbContextFactory |
24
+ | SignalR hub handlers | IDbContextFactory |
25
+ | Hosted services | IDbContextFactory |
26
+
27
+ ---
28
+
29
+ ## Pattern: Repository Factory
30
+
31
+ ```csharp
32
+ // Interfaces
33
+ public interface IOrderRepositoryBase
34
+ {
35
+ Task<Order?> GetByIdAsync(Guid id, CancellationToken ct = default);
36
+ Task UpdateAsync(Order order, CancellationToken ct = default);
37
+ }
38
+
39
+ public interface IOrderRepository : IOrderRepositoryBase
40
+ {
41
+ Task<List<Order>> GetPendingAsync(CancellationToken ct = default);
42
+ }
43
+
44
+ public interface IScopedOrderRepository : IOrderRepositoryBase, IAsyncDisposable { }
45
+
46
+ public interface IOrderRepositoryFactory
47
+ {
48
+ IScopedOrderRepository CreateScoped();
49
+ }
50
+
51
+ // Implementation
52
+ public abstract class OrderRepositoryBase(AppDbContext context)
53
+ {
54
+ protected readonly AppDbContext _context = context;
55
+
56
+ public async Task<Order?> GetByIdAsync(Guid id, CancellationToken ct = default)
57
+ => await _context.Orders.Include(o => o.Items).FirstOrDefaultAsync(o => o.Id == id, ct);
58
+
59
+ public async Task UpdateAsync(Order order, CancellationToken ct = default)
60
+ {
61
+ _context.Orders.Update(order);
62
+ await _context.SaveChangesAsync(ct);
63
+ }
64
+ }
65
+
66
+ public class ScopedOrderRepository(AppDbContext context) : OrderRepositoryBase(context), IScopedOrderRepository
67
+ {
68
+ public async ValueTask DisposeAsync() => await _context.DisposeAsync();
69
+ }
70
+
71
+ public class OrderRepositoryFactory(IDbContextFactory<AppDbContext> contextFactory) : IOrderRepositoryFactory
72
+ {
73
+ public IScopedOrderRepository CreateScoped() => new ScopedOrderRepository(contextFactory.CreateDbContext());
74
+ }
75
+
76
+ // DI
77
+ builder.Services.AddDbContextFactory<AppDbContext>(o => o.UseSqlServer(conn));
78
+ builder.Services.AddScoped<IOrderRepository, OrderRepository>();
79
+ builder.Services.AddSingleton<IOrderRepositoryFactory, OrderRepositoryFactory>();
80
+
81
+ // Usage
82
+ _ = Task.Run(async () =>
83
+ {
84
+ await using var repo = _repoFactory.CreateScoped();
85
+ var order = await repo.GetByIdAsync(orderId);
86
+ await repo.UpdateAsync(order);
87
+ });
88
+ ```
89
+
90
+ ---
91
+
92
+ ## Background Operations Decision Matrix
93
+
94
+ | Option | When | Cost | Complexity | Resilience |
95
+ |--------|------|------|------------|------------|
96
+ | Task.Run + RepositoryFactory | < 5min, low volume, non-critical | $0 | Low | Low |
97
+ | Background Service + Queue | Long, needs retry, medium volume | $0-16/mo | Medium | Medium |
98
+ | Hangfire | Scheduled/recurring, needs dashboard | $0 (OSS) | Medium | High |
99
+ | Azure Functions Durable | High volume, spikes, orchestration | Pay-per-use | Med-High | High |
100
+
101
+ ### Hangfire Quick Setup
102
+
103
+ ```csharp
104
+ builder.Services.AddHangfire(c => c.UseSqlServerStorage(conn));
105
+ builder.Services.AddHangfireServer();
106
+
107
+ public class OrderJob(IOrderRepositoryFactory repoFactory)
108
+ {
109
+ [AutomaticRetry(Attempts = 3)]
110
+ public async Task ProcessAsync(Guid id, CancellationToken ct)
111
+ {
112
+ await using var repo = repoFactory.CreateScoped();
113
+ var order = await repo.GetByIdAsync(id, ct);
114
+ }
115
+ }
116
+
117
+ BackgroundJob.Enqueue<OrderJob>(x => x.ProcessAsync(orderId, CancellationToken.None));
118
+ RecurringJob.AddOrUpdate<ReportJob>("daily", x => x.GenerateAsync(), Cron.Daily);
119
+ ```
120
+
121
+ ---
122
+
123
+ ## EF Core Configuration
124
+
125
+ ### Navigation Properties
126
+
127
+ ```csharp
128
+ // WRONG - Include() silently fails
129
+ builder.HasMany<GeneratedArt>().WithOne(x => x.Order).HasForeignKey(x => x.OrderId);
130
+
131
+ // CORRECT - Specify navigation property
132
+ builder.HasMany(x => x.GeneratedArts).WithOne(x => x.Order).HasForeignKey(x => x.OrderId);
133
+ ```
134
+
135
+ **Rule:** `ICollection<T>` in entity → MUST appear in `HasMany(x => x.Property)`.
136
+
137
+ ### Migrations (.NET 10)
138
+
139
+ ```bash
140
+ dotnet ef migrations add <Name> --project src/Infrastructure --startup-project src/Web
141
+ dotnet ef database update --project src/Infrastructure --startup-project src/Web
142
+ ```
143
+
144
+ | Environment | Auto-Migration? | Why |
145
+ |-------------|----------------|-----|
146
+ | Development | OK | Fast iteration |
147
+ | Staging | Caution | Test before prod |
148
+ | Production | NO | Use CI/CD pipeline |
149
+
150
+ ---
151
+
152
+ ## Anti-Patterns
153
+
154
+ | Anti-Pattern | Fix |
155
+ |-------------|-----|
156
+ | Scoped DbContext in `Task.Run` | Use `IDbContextFactory` |
157
+ | Scoped service in singleton | Use factory pattern |
158
+ | Missing `await using` | Always use with scoped repo |
159
+ | `IScopedRepo` inherits full `IRepo` | Inherit only `IBase` |
160
+
161
+ ---
162
+
163
+ ## Checklist
164
+
165
+ - [ ] Background ops use `IDbContextFactory` / Repository Factory?
166
+ - [ ] Duration/volume assessed against decision matrix?
167
+ - [ ] `HasMany` uses navigation property lambda?
168
+ - [ ] Migration created before `database update`?
169
+ - [ ] `await using` for scoped repositories?
170
+ - [ ] `AsNoTracking()` for read-only queries?
171
+ - [ ] CancellationToken propagated?
172
+
173
+ ---
174
+
175
+ *MORPH-SPEC by Polymorphism Tech*
176
+
177
+ ---
178
+
179
+ ## Code Review Checklist
180
+
181
+ Use this checklist during code review for EF Core in Blazor:
182
+
183
+ ### DbContext and Concurrency
184
+
185
+ - [ ] **No scoped DbContext used in `Task.Run` or background operations?**
186
+ - [ ] **`IDbContextFactory<T>` used for background/parallel operations?**
187
+ - [ ] **Factory-created contexts are properly disposed (`await using`)?**
188
+ - [ ] **No shared DbContext between concurrent operations?**
189
+
190
+ ### File Upload (IBrowserFile)
191
+
192
+ - [ ] **`IBrowserFile` is read only ONCE and bytes stored?**
193
+ - [ ] **`maxAllowedSize` specified in `OpenReadStream()`?**
194
+ - [ ] **File size validated BEFORE opening stream?**
195
+ - [ ] **File extension/type validated?**
196
+
197
+ ---
198
+
199
+ *MORPH-SPEC by Polymorphism Tech*