@polymorphism-tech/morph-spec 4.6.0 → 4.7.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (239) hide show
  1. package/README.md +414 -700
  2. package/docs/ARCHITECTURE.md +331 -0
  3. package/docs/CHEATSHEET.md +221 -0
  4. package/docs/COMMAND-FLOWS.md +368 -0
  5. package/docs/QUICKSTART.md +212 -0
  6. package/docs/examples/order-management/contracts.cs +84 -0
  7. package/docs/examples/order-management/proposal.md +24 -0
  8. package/docs/examples/order-management/spec.md +162 -0
  9. package/docs/plans/2026-02-23-ddd-architecture-refactor.md +1153 -0
  10. package/docs/plans/2026-02-23-ddd-nextsteps.md +682 -0
  11. package/docs/plans/2026-02-23-infra-architect-refactor.md +437 -0
  12. package/docs/plans/2026-02-23-nextjs-code-review-design.md +156 -0
  13. package/docs/plans/2026-02-23-nextjs-code-review-impl.md +1254 -0
  14. package/docs/plans/2026-02-23-nextjs-standards-design.md +149 -0
  15. package/docs/plans/2026-02-23-nextjs-standards-impl.md +1846 -0
  16. package/framework/agents/README.md +14 -14
  17. package/framework/agents/architecture/standards-architect.md +159 -159
  18. package/framework/agents/frontend/nextjs-expert.md +87 -127
  19. package/framework/agents/infrastructure/azure-architect.md +147 -147
  20. package/framework/agents/infrastructure/infra-architect.md +45 -0
  21. package/framework/agents.json +1145 -278
  22. package/framework/rules/frontend-standards.md +0 -3
  23. package/framework/rules/nextjs-standards.md +17 -0
  24. package/framework/skills/level-0-meta/code-review-nextjs/SKILL.md +147 -0
  25. package/framework/skills/level-0-meta/code-review-nextjs/references/review-example-nextjs.md +254 -0
  26. package/framework/skills/level-0-meta/tool-usage-guide/SKILL.md +3 -3
  27. package/framework/skills/level-1-workflows/phase-design/SKILL.md +45 -9
  28. package/framework/skills/level-1-workflows/phase-tasks/SKILL.md +38 -0
  29. package/framework/standards/STANDARDS.json +121 -0
  30. package/framework/standards/architecture/ddd/bounded-contexts.md +105 -0
  31. package/framework/standards/architecture/ddd/complexity-levels.md +108 -0
  32. package/framework/standards/architecture/ddd/ubiquitous-language.md +58 -0
  33. package/framework/standards/frontend/nextjs/app-router.md +123 -0
  34. package/framework/standards/frontend/nextjs/components.md +132 -0
  35. package/framework/standards/frontend/nextjs/data-fetching.md +126 -0
  36. package/framework/standards/frontend/nextjs/forms.md +128 -0
  37. package/framework/standards/frontend/nextjs/naming-conventions.md +67 -0
  38. package/framework/standards/frontend/nextjs/project-structure.md +102 -0
  39. package/framework/standards/frontend/nextjs/state-management.md +72 -0
  40. package/framework/standards/frontend/nextjs/testing.md +111 -0
  41. package/framework/templates/REGISTRY.json +538 -142
  42. package/framework/templates/code/dotnet/contracts/contracts-level1.cs +69 -0
  43. package/framework/templates/code/dotnet/contracts/contracts-level2.cs +86 -0
  44. package/framework/templates/code/dotnet/contracts/contracts-level3.cs +41 -0
  45. package/framework/templates/docs/spec.md +49 -0
  46. package/framework/templates/frontend/nextjs/Dockerfile.nextjs.hbs +43 -0
  47. package/framework/templates/frontend/nextjs/client-component.tsx.hbs +26 -0
  48. package/framework/templates/frontend/nextjs/env.mjs.hbs +32 -0
  49. package/framework/templates/frontend/nextjs/feature-form.tsx.hbs +56 -0
  50. package/framework/templates/frontend/nextjs/page.tsx.hbs +22 -0
  51. package/framework/templates/frontend/nextjs/tsconfig.json.hbs +26 -0
  52. package/framework/templates/frontend/nextjs/use-feature.ts.hbs +54 -0
  53. package/framework/templates/project-structure/dotnet-ddd.md +70 -0
  54. package/framework/workflows/docs/enforcement-pipeline.md +2 -1
  55. package/package.json +1 -1
  56. package/scripts/scan-nextjs.mjs +169 -0
  57. package/src/commands/project/doctor.js +52 -1
  58. package/src/commands/project/init.js +15 -1
  59. package/src/commands/project/update.js +6 -1
  60. package/src/lib/standards/standards-context-injector.js +5 -0
  61. package/src/lib/validators/nextjs/index.js +6 -0
  62. package/src/lib/validators/nextjs/next-component-validator.js +181 -0
  63. package/src/lib/validators/validation-runner.js +5 -0
  64. package/src/utils/agents-installer.js +14 -2
  65. package/.morph/.morphversion +0 -5
  66. package/.morph/analytics/threads-log.jsonl +0 -6
  67. package/.morph/config/config.json +0 -8
  68. package/.morph/framework/agents.json +0 -948
  69. package/.morph/framework/standards/STANDARDS.json +0 -812
  70. package/.morph/framework/standards/ai-agents/blazor-ui.md +0 -364
  71. package/.morph/framework/standards/ai-agents/production.md +0 -415
  72. package/.morph/framework/standards/ai-agents/setup.md +0 -418
  73. package/.morph/framework/standards/ai-agents/team-orchestration.md +0 -479
  74. package/.morph/framework/standards/ai-agents/workflows.md +0 -354
  75. package/.morph/framework/standards/architecture/ddd/aggregates.md +0 -120
  76. package/.morph/framework/standards/architecture/ddd/entities.md +0 -99
  77. package/.morph/framework/standards/architecture/ddd/value-objects.md +0 -124
  78. package/.morph/framework/standards/backend/api/minimal-api.md +0 -494
  79. package/.morph/framework/standards/backend/api/rest.md +0 -492
  80. package/.morph/framework/standards/backend/api/validation.md +0 -88
  81. package/.morph/framework/standards/backend/authentication/passkeys.md +0 -428
  82. package/.morph/framework/standards/backend/database/ef-core.md +0 -199
  83. package/.morph/framework/standards/backend/database/migrations.md +0 -393
  84. package/.morph/framework/standards/backend/database/postgresql/database.md +0 -352
  85. package/.morph/framework/standards/backend/database/repository-patterns.md +0 -528
  86. package/.morph/framework/standards/backend/database/vector-search-rag.md +0 -541
  87. package/.morph/framework/standards/backend/dotnet/async.md +0 -366
  88. package/.morph/framework/standards/backend/dotnet/core.md +0 -117
  89. package/.morph/framework/standards/backend/dotnet/di.md +0 -439
  90. package/.morph/framework/standards/backend/dotnet/program-cs-checklist.md +0 -92
  91. package/.morph/framework/standards/backend/integrations/asaas/asaas-api.md +0 -216
  92. package/.morph/framework/standards/backend/integrations/clerk/clerk-auth.md +0 -290
  93. package/.morph/framework/standards/backend/integrations/hangfire/hangfire-jobs.md +0 -350
  94. package/.morph/framework/standards/backend/integrations/resend/resend-email.md +0 -385
  95. package/.morph/framework/standards/context/analytics.md +0 -96
  96. package/.morph/framework/standards/context/bundles.md +0 -110
  97. package/.morph/framework/standards/context/priming.md +0 -78
  98. package/.morph/framework/standards/core/architecture.md +0 -185
  99. package/.morph/framework/standards/core/coding.md +0 -214
  100. package/.morph/framework/standards/core/git-branching-strategy.md +0 -403
  101. package/.morph/framework/standards/core/git.md +0 -185
  102. package/.morph/framework/standards/core/testing.md +0 -295
  103. package/.morph/framework/standards/data/nosql/blob-storage.md +0 -102
  104. package/.morph/framework/standards/data/nosql/cache/redis.md +0 -97
  105. package/.morph/framework/standards/data/nosql/cosmos-db.md +0 -118
  106. package/.morph/framework/standards/data/vector-search/azure-ai-search.md +0 -121
  107. package/.morph/framework/standards/data/vector-search/rag-chunking.md +0 -104
  108. package/.morph/framework/standards/frontend/blazor/design-checklist.md +0 -222
  109. package/.morph/framework/standards/frontend/blazor/fluent-ui-setup.md +0 -595
  110. package/.morph/framework/standards/frontend/blazor/fluent-ui.md +0 -137
  111. package/.morph/framework/standards/frontend/blazor/html-conversion.md +0 -184
  112. package/.morph/framework/standards/frontend/blazor/lifecycle.md +0 -195
  113. package/.morph/framework/standards/frontend/blazor/pitfalls.md +0 -198
  114. package/.morph/framework/standards/frontend/blazor/state.md +0 -191
  115. package/.morph/framework/standards/frontend/design-system/animations.md +0 -151
  116. package/.morph/framework/standards/frontend/design-system/naming.md +0 -64
  117. package/.morph/framework/standards/frontend/nextjs/nextjs-patterns.md +0 -215
  118. package/.morph/framework/standards/infrastructure/azure/azure.md +0 -624
  119. package/.morph/framework/standards/infrastructure/azure/bicep/bicep-patterns.md +0 -422
  120. package/.morph/framework/standards/infrastructure/azure/devops/azure-devops-setup.md +0 -516
  121. package/.morph/framework/standards/infrastructure/azure/devops/local-development.md +0 -520
  122. package/.morph/framework/standards/infrastructure/azure/services/functions.md +0 -486
  123. package/.morph/framework/standards/infrastructure/azure/services/service-bus.md +0 -459
  124. package/.morph/framework/standards/infrastructure/azure/services/storage.md +0 -407
  125. package/.morph/framework/standards/infrastructure/docker/easypanel-deploy.md +0 -196
  126. package/.morph/framework/standards/infrastructure/supabase/mcp-setup.md +0 -252
  127. package/.morph/framework/standards/infrastructure/supabase/supabase-auth.md +0 -176
  128. package/.morph/framework/standards/infrastructure/supabase/supabase-pgvector.md +0 -169
  129. package/.morph/framework/standards/infrastructure/supabase/supabase-rls.md +0 -184
  130. package/.morph/framework/standards/infrastructure/supabase/supabase-storage.md +0 -153
  131. package/.morph/framework/standards/integration/api/graphql.md +0 -91
  132. package/.morph/framework/standards/integration/api/grpc.md +0 -114
  133. package/.morph/framework/standards/integration/api/rest-design.md +0 -95
  134. package/.morph/framework/standards/integration/event-driven/cqrs.md +0 -101
  135. package/.morph/framework/standards/integration/event-driven/event-sourcing.md +0 -124
  136. package/.morph/framework/standards/integration/event-driven/service-bus.md +0 -95
  137. package/.morph/framework/standards/integration/mcp/mcp-tools.md +0 -384
  138. package/.morph/framework/standards/observability/logging.md +0 -131
  139. package/.morph/framework/standards/observability/metrics.md +0 -121
  140. package/.morph/framework/standards/observability/monitoring.md +0 -114
  141. package/.morph/framework/standards/observability/tracing.md +0 -132
  142. package/.morph/framework/standards/workflows/parallel-execution.md +0 -112
  143. package/.morph/framework/standards/workflows/thread-management.md +0 -113
  144. package/.morph/framework/templates/.idea/morph-templates.xml +0 -92
  145. package/.morph/framework/templates/.vscode/morph-templates.code-snippets +0 -186
  146. package/.morph/framework/templates/IDE-SNIPPETS.md +0 -266
  147. package/.morph/framework/templates/README.md +0 -814
  148. package/.morph/framework/templates/REGISTRY.json +0 -1492
  149. package/.morph/framework/templates/code/dotnet/backend/repository.cs +0 -141
  150. package/.morph/framework/templates/code/dotnet/backend/service.cs +0 -139
  151. package/.morph/framework/templates/code/dotnet/contracts/Commands.cs +0 -74
  152. package/.morph/framework/templates/code/dotnet/contracts/Entities.cs +0 -25
  153. package/.morph/framework/templates/code/dotnet/contracts/Queries.cs +0 -74
  154. package/.morph/framework/templates/code/dotnet/contracts/README.md +0 -74
  155. package/.morph/framework/templates/code/dotnet/contracts/api-contracts.cs +0 -173
  156. package/.morph/framework/templates/code/dotnet/contracts/contracts.cs +0 -217
  157. package/.morph/framework/templates/code/dotnet/contracts/contracts.cs.hbs +0 -172
  158. package/.morph/framework/templates/code/dotnet/database/migration.cs +0 -83
  159. package/.morph/framework/templates/code/dotnet/frontend/component.razor +0 -239
  160. package/.morph/framework/templates/code/dotnet/jobs/agent.cs +0 -163
  161. package/.morph/framework/templates/code/dotnet/jobs/job.cs +0 -171
  162. package/.morph/framework/templates/code/dotnet/test.cs +0 -239
  163. package/.morph/framework/templates/code/sql/rls-policy.sql +0 -57
  164. package/.morph/framework/templates/code/sql/supabase-migration.sql +0 -100
  165. package/.morph/framework/templates/code/sql/supabase-migration.template.sql +0 -113
  166. package/.morph/framework/templates/code/typescript/contracts.ts +0 -168
  167. package/.morph/framework/templates/context/CONTEXT-FEATURE.md +0 -276
  168. package/.morph/framework/templates/context/CONTEXT.md +0 -181
  169. package/.morph/framework/templates/docs/clarifications.md +0 -253
  170. package/.morph/framework/templates/docs/onboarding.md +0 -123
  171. package/.morph/framework/templates/docs/proposal.md +0 -182
  172. package/.morph/framework/templates/docs/schema-analysis.md +0 -119
  173. package/.morph/framework/templates/docs/spec.md +0 -149
  174. package/.morph/framework/templates/docs/ui-components.md +0 -124
  175. package/.morph/framework/templates/docs/ui-design-system.md +0 -76
  176. package/.morph/framework/templates/docs/ui-flows.md +0 -167
  177. package/.morph/framework/templates/docs/ui-mockups.md +0 -98
  178. package/.morph/framework/templates/docs/user-stories.md +0 -34
  179. package/.morph/framework/templates/examples/design-system-examples.md +0 -357
  180. package/.morph/framework/templates/examples/spec-examples.md +0 -90
  181. package/.morph/framework/templates/feature/decisions.md +0 -187
  182. package/.morph/framework/templates/feature/recap.md +0 -146
  183. package/.morph/framework/templates/feature/tasks.md +0 -199
  184. package/.morph/framework/templates/infrastructure/azure/Dockerfile.example +0 -82
  185. package/.morph/framework/templates/infrastructure/azure/README.md +0 -286
  186. package/.morph/framework/templates/infrastructure/azure/app-insights.bicep +0 -63
  187. package/.morph/framework/templates/infrastructure/azure/app-service.bicep +0 -164
  188. package/.morph/framework/templates/infrastructure/azure/container-app-env.bicep +0 -49
  189. package/.morph/framework/templates/infrastructure/azure/container-app.bicep +0 -156
  190. package/.morph/framework/templates/infrastructure/azure/deploy-checklist.md +0 -426
  191. package/.morph/framework/templates/infrastructure/azure/deploy.ps1 +0 -229
  192. package/.morph/framework/templates/infrastructure/azure/deploy.sh +0 -208
  193. package/.morph/framework/templates/infrastructure/azure/key-vault.bicep +0 -91
  194. package/.morph/framework/templates/infrastructure/azure/main.bicep +0 -189
  195. package/.morph/framework/templates/infrastructure/azure/parameters.dev.json +0 -29
  196. package/.morph/framework/templates/infrastructure/azure/parameters.prod.json +0 -29
  197. package/.morph/framework/templates/infrastructure/azure/parameters.staging.json +0 -29
  198. package/.morph/framework/templates/infrastructure/azure/sql-database.bicep +0 -103
  199. package/.morph/framework/templates/infrastructure/azure/storage.bicep +0 -106
  200. package/.morph/framework/templates/infrastructure/docker/Dockerfile.template +0 -58
  201. package/.morph/framework/templates/infrastructure/docker/docker-compose.template.yml +0 -67
  202. package/.morph/framework/templates/infrastructure/docker/dockerfile-api.dockerfile +0 -38
  203. package/.morph/framework/templates/infrastructure/docker/dockerfile-web.dockerfile +0 -48
  204. package/.morph/framework/templates/infrastructure/docker/easypanel.template.json +0 -54
  205. package/.morph/framework/templates/infrastructure/github/README.md +0 -593
  206. package/.morph/framework/templates/infrastructure/github/actions/azure-auth/action.yml.hbs +0 -22
  207. package/.morph/framework/templates/infrastructure/github/actions/docker-build-push/action.yml.hbs +0 -45
  208. package/.morph/framework/templates/infrastructure/github/actions/health-check/action.yml.hbs +0 -27
  209. package/.morph/framework/templates/infrastructure/github/workflows/deploy-azure-app-service.yml.hbs +0 -61
  210. package/.morph/framework/templates/infrastructure/github/workflows/deploy-easypanel.yml.hbs +0 -31
  211. package/.morph/framework/templates/infrastructure/github/workflows/docker-build-push.yml.hbs +0 -59
  212. package/.morph/framework/templates/infrastructure/github/workflows/dotnet-build.yml.hbs +0 -39
  213. package/.morph/framework/templates/integrations/asaas-client.cs +0 -387
  214. package/.morph/framework/templates/integrations/asaas-webhook.cs +0 -351
  215. package/.morph/framework/templates/integrations/azure-identity-config.cs +0 -288
  216. package/.morph/framework/templates/integrations/clerk-config.cs +0 -258
  217. package/.morph/framework/templates/meta-prompts/fusion/fusion-agent.md +0 -76
  218. package/.morph/framework/templates/meta-prompts/fusion/fusion-aggregator.md +0 -100
  219. package/.morph/framework/templates/meta-prompts/hops/hop-retry.md +0 -78
  220. package/.morph/framework/templates/meta-prompts/hops/hop-validation.md +0 -97
  221. package/.morph/framework/templates/meta-prompts/hops/hop-wrapper.md +0 -36
  222. package/.morph/framework/templates/meta-prompts/parallel-workers/parallel-coordinator.md +0 -113
  223. package/.morph/framework/templates/meta-prompts/parallel-workers/parallel-worker.md +0 -80
  224. package/.morph/framework/templates/meta-prompts/squad-leaders/backend-squad.md +0 -90
  225. package/.morph/framework/templates/meta-prompts/squad-leaders/frontend-squad.md +0 -126
  226. package/.morph/framework/templates/meta-prompts/squad-leaders/squad-leader.md +0 -43
  227. package/.morph/framework/templates/meta-prompts/validators/checkpoint-validator.md +0 -107
  228. package/.morph/framework/templates/meta-prompts/validators/pre-commit-validator.md +0 -95
  229. package/.morph/framework/templates/saas/subscription.cs +0 -347
  230. package/.morph/framework/templates/saas/tenant.cs +0 -338
  231. package/.morph/framework/templates/state.template.json +0 -17
  232. package/.morph/framework/templates/ui/FluentDesignTheme.cs +0 -149
  233. package/.morph/framework/templates/ui/MudTheme.cs +0 -281
  234. package/.morph/framework/templates/ui/design-system.css +0 -226
  235. package/.morph/logs/tool-failures.log +0 -7
  236. package/.morph/memory/pre-compact-2026-02-23T15-43-03-521Z.json +0 -16
  237. package/.morph/state.json +0 -48
  238. package/framework/templates/code/dotnet/contracts/contracts.cs +0 -217
  239. package/framework/templates/code/dotnet/contracts/contracts.cs.hbs +0 -172
@@ -1,439 +0,0 @@
1
- # .NET Dependency Injection Standards
2
-
3
- > **Scope:** universal
4
- > **Layer:** 0 (always load)
5
- > **Keywords:** di, dependency injection, ioc, lifetime, scoped, singleton, transient
6
- > **Load When:** always
7
-
8
- Dependency Injection patterns and lifetimes for .NET applications
9
-
10
- ---
11
-
12
- ## Service Lifetimes
13
-
14
- ### Transient
15
-
16
- **Created every time they're requested**
17
-
18
- Use for: Lightweight, stateless services
19
-
20
- ```csharp
21
- // Program.cs
22
- builder.Services.AddTransient<IEmailService, EmailService>();
23
- builder.Services.AddTransient<IPdfGenerator, PdfGenerator>();
24
- ```
25
-
26
- ### Scoped
27
-
28
- **Created once per request (HTTP request in web apps)**
29
-
30
- Use for: DbContext, repositories, request-specific services
31
-
32
- ```csharp
33
- // Program.cs
34
- builder.Services.AddScoped<IUserRepository, UserRepository>();
35
- builder.Services.AddScoped<IOrderService, OrderService>();
36
- builder.Services.AddDbContext<AppDbContext>(); // Scoped by default
37
- ```
38
-
39
- ### Singleton
40
-
41
- **Created once for application lifetime**
42
-
43
- Use for: Caching, configuration, thread-safe services
44
-
45
- ```csharp
46
- // Program.cs
47
- builder.Services.AddSingleton<IMemoryCache, MemoryCache>();
48
- builder.Services.AddSingleton<IConfiguration>(builder.Configuration);
49
- ```
50
-
51
- ---
52
-
53
- ## Registration Patterns
54
-
55
- ### Interface + Implementation
56
-
57
- ```csharp
58
- // ✅ CORRECT
59
- builder.Services.AddScoped<IUserService, UserService>();
60
- builder.Services.AddScoped<IOrderRepository, OrderRepository>();
61
-
62
- // ❌ WRONG (concrete types in constructor)
63
- public class OrderController
64
- {
65
- private readonly OrderRepository _repo; // Don't inject concrete types!
66
-
67
- public OrderController(OrderRepository repo) { }
68
- }
69
- ```
70
-
71
- ### Multiple Implementations
72
-
73
- ```csharp
74
- // Register multiple implementations
75
- builder.Services.AddScoped<INotificationService, EmailNotificationService>();
76
- builder.Services.AddScoped<INotificationService, SmsNotificationService>();
77
-
78
- // Inject IEnumerable<T>
79
- public class NotificationManager
80
- {
81
- private readonly IEnumerable<INotificationService> _services;
82
-
83
- public NotificationManager(IEnumerable<INotificationService> services)
84
- {
85
- _services = services;
86
- }
87
-
88
- public async Task NotifyAllAsync(string message)
89
- {
90
- foreach (var service in _services)
91
- {
92
- await service.SendAsync(message);
93
- }
94
- }
95
- }
96
- ```
97
-
98
- ### Factory Pattern
99
-
100
- ```csharp
101
- // For services that need per-operation instances
102
- builder.Services.AddScoped<IUserRepositoryFactory, UserRepositoryFactory>();
103
-
104
- public interface IUserRepositoryFactory
105
- {
106
- IScopedUserRepository CreateScoped();
107
- }
108
-
109
- public class UserRepositoryFactory : IUserRepositoryFactory
110
- {
111
- private readonly IDbContextFactory<AppDbContext> _contextFactory;
112
-
113
- public UserRepositoryFactory(IDbContextFactory<AppDbContext> contextFactory)
114
- {
115
- _contextFactory = contextFactory;
116
- }
117
-
118
- public IScopedUserRepository CreateScoped()
119
- {
120
- var context = _contextFactory.CreateDbContext();
121
- return new ScopedUserRepository(context);
122
- }
123
- }
124
- ```
125
-
126
- ---
127
-
128
- ## Constructor Injection
129
-
130
- ### Primary Constructors (.NET 8+)
131
-
132
- ```csharp
133
- // ✅ CORRECT (modern syntax)
134
- public class UserService(
135
- IUserRepository repository,
136
- ILogger<UserService> logger,
137
- IEmailService emailService) : IUserService
138
- {
139
- public async Task<User> GetUserAsync(int id)
140
- {
141
- logger.LogInformation("Getting user {UserId}", id);
142
- return await repository.GetByIdAsync(id);
143
- }
144
- }
145
- ```
146
-
147
- ### Traditional Constructors
148
-
149
- ```csharp
150
- // ✅ CORRECT
151
- public class UserService : IUserService
152
- {
153
- private readonly IUserRepository _repository;
154
- private readonly ILogger<UserService> _logger;
155
-
156
- public UserService(IUserRepository repository, ILogger<UserService> logger)
157
- {
158
- _repository = repository;
159
- _logger = logger;
160
- }
161
- }
162
- ```
163
-
164
- ---
165
-
166
- ## Blazor-Specific DI
167
-
168
- ### DbContext in Blazor Components
169
-
170
- **❌ WRONG: Direct DbContext Injection**
171
-
172
- ```csharp
173
- // ❌ WRONG (DbContext lifecycle issue)
174
- @inject AppDbContext DbContext
175
-
176
- @code {
177
- protected override async Task OnInitializedAsync()
178
- {
179
- users = await DbContext.Users.ToListAsync(); // May be disposed!
180
- }
181
- }
182
- ```
183
-
184
- **✅ CORRECT: Use IDbContextFactory**
185
-
186
- ```csharp
187
- // Program.cs
188
- builder.Services.AddDbContextFactory<AppDbContext>(options =>
189
- options.UseSqlServer(connectionString));
190
-
191
- // Component
192
- @inject IDbContextFactory<AppDbContext> DbFactory
193
-
194
- @code {
195
- protected override async Task OnInitializedAsync()
196
- {
197
- await using var context = await DbFactory.CreateDbContextAsync();
198
- users = await context.Users.ToListAsync();
199
- }
200
- }
201
- ```
202
-
203
- ### Service Injection in Components
204
-
205
- ```csharp
206
- @inject IUserService UserService
207
- @inject NavigationManager Navigation
208
- @inject ILogger<UserList> Logger
209
-
210
- @code {
211
- private List<User> users = new();
212
-
213
- protected override async Task OnInitializedAsync()
214
- {
215
- try
216
- {
217
- users = await UserService.GetAllAsync();
218
- }
219
- catch (Exception ex)
220
- {
221
- Logger.LogError(ex, "Failed to load users");
222
- }
223
- }
224
- }
225
- ```
226
-
227
- ---
228
-
229
- ## Lifetime Conflicts
230
-
231
- ### Problem: Scoped in Singleton
232
-
233
- ```csharp
234
- // ❌ WRONG (Scoped DbContext injected in Singleton)
235
- builder.Services.AddSingleton<CacheService>(); // Singleton
236
- builder.Services.AddDbContext<AppDbContext>(); // Scoped
237
-
238
- public class CacheService
239
- {
240
- private readonly AppDbContext _context; // ❌ WRONG!
241
-
242
- public CacheService(AppDbContext context)
243
- {
244
- _context = context; // Scoped injected in Singleton!
245
- }
246
- }
247
- ```
248
-
249
- **✅ CORRECT: Use IServiceProvider or Factory**
250
-
251
- ```csharp
252
- public class CacheService
253
- {
254
- private readonly IDbContextFactory<AppDbContext> _factory;
255
-
256
- public CacheService(IDbContextFactory<AppDbContext> factory)
257
- {
258
- _factory = factory;
259
- }
260
-
261
- public async Task<User> GetUserAsync(int id)
262
- {
263
- await using var context = await _factory.CreateDbContextAsync();
264
- return await context.Users.FindAsync(id);
265
- }
266
- }
267
- ```
268
-
269
- ---
270
-
271
- ## Registration Order
272
-
273
- ### Program.cs Order
274
-
275
- ```csharp
276
- var builder = WebApplication.CreateBuilder(args);
277
-
278
- // 1. Configuration
279
- var config = builder.Configuration;
280
-
281
- // 2. Logging
282
- builder.Services.AddLogging();
283
-
284
- // 3. Database
285
- builder.Services.AddDbContextFactory<AppDbContext>(options =>
286
- options.UseSqlServer(config.GetConnectionString("DefaultConnection")));
287
-
288
- // 4. Repositories (depend on DbContext)
289
- builder.Services.AddScoped<IUserRepository, UserRepository>();
290
- builder.Services.AddScoped<IOrderRepository, OrderRepository>();
291
-
292
- // 5. Services (depend on repositories)
293
- builder.Services.AddScoped<IUserService, UserService>();
294
- builder.Services.AddScoped<IOrderService, OrderService>();
295
-
296
- // 6. Infrastructure
297
- builder.Services.AddScoped<IEmailService, EmailService>();
298
- builder.Services.AddSingleton<IMemoryCache, MemoryCache>();
299
-
300
- // 7. External integrations
301
- builder.Services.AddHttpClient<IPaymentGateway, StripePaymentGateway>();
302
-
303
- // 8. Framework services (last)
304
- builder.Services.AddControllers();
305
- builder.Services.AddRazorComponents();
306
-
307
- var app = builder.Build();
308
- ```
309
-
310
- ---
311
-
312
- ## Best Practices
313
-
314
- ### DO
315
-
316
- ✅ **Inject interfaces, not concrete types**
317
-
318
- ```csharp
319
- // ✅ CORRECT
320
- public class OrderService(IOrderRepository repository) { }
321
-
322
- // ❌ WRONG
323
- public class OrderService(OrderRepository repository) { }
324
- ```
325
-
326
- ✅ **Use primary constructors (.NET 8+)**
327
-
328
- ```csharp
329
- // ✅ CORRECT (concise)
330
- public class UserService(IUserRepository repository, ILogger<UserService> logger) : IUserService
331
- {
332
- public async Task<User> GetAsync(int id) =>
333
- await repository.GetByIdAsync(id);
334
- }
335
- ```
336
-
337
- ✅ **Use factories for DbContext in singletons**
338
-
339
- ```csharp
340
- // ✅ CORRECT
341
- public class CacheService(IDbContextFactory<AppDbContext> factory)
342
- {
343
- public async Task LoadAsync()
344
- {
345
- await using var context = await factory.CreateDbContextAsync();
346
- // Use context
347
- }
348
- }
349
- ```
350
-
351
- ### DON'T
352
-
353
- ❌ **Don't inject scoped services in singletons**
354
-
355
- ```csharp
356
- // ❌ WRONG
357
- builder.Services.AddSingleton<CacheService>();
358
- builder.Services.AddScoped<IUserRepository, UserRepository>();
359
-
360
- public class CacheService(IUserRepository repository) // ❌ Scoped in Singleton!
361
- {
362
- }
363
- ```
364
-
365
- ❌ **Don't use service locator pattern**
366
-
367
- ```csharp
368
- // ❌ WRONG (anti-pattern)
369
- public class OrderService
370
- {
371
- public async Task ProcessAsync(int orderId)
372
- {
373
- var serviceProvider = ...; // Don't do this!
374
- var repo = serviceProvider.GetService<IOrderRepository>();
375
- }
376
- }
377
-
378
- // ✅ CORRECT (constructor injection)
379
- public class OrderService(IOrderRepository repository)
380
- {
381
- public async Task ProcessAsync(int orderId)
382
- {
383
- var order = await repository.GetByIdAsync(orderId);
384
- }
385
- }
386
- ```
387
-
388
- ❌ **Don't create instances with `new`**
389
-
390
- ```csharp
391
- // ❌ WRONG
392
- public class OrderService
393
- {
394
- public async Task ProcessAsync()
395
- {
396
- var emailService = new EmailService(); // Don't create manually!
397
- await emailService.SendAsync(...);
398
- }
399
- }
400
-
401
- // ✅ CORRECT
402
- public class OrderService(IEmailService emailService)
403
- {
404
- public async Task ProcessAsync()
405
- {
406
- await emailService.SendAsync(...);
407
- }
408
- }
409
- ```
410
-
411
- ---
412
-
413
- ## Testing with DI
414
-
415
- ### Mock Dependencies
416
-
417
- ```csharp
418
- [Fact]
419
- public async Task GetUser_ValidId_ReturnsUser()
420
- {
421
- // Arrange
422
- var mockRepo = new Mock<IUserRepository>();
423
- mockRepo.Setup(r => r.GetByIdAsync(1))
424
- .ReturnsAsync(new User { Id = 1, Name = "John" });
425
-
426
- var service = new UserService(mockRepo.Object, Mock.Of<ILogger<UserService>>());
427
-
428
- // Act
429
- var user = await service.GetUserAsync(1);
430
-
431
- // Assert
432
- Assert.NotNull(user);
433
- Assert.Equal("John", user.Name);
434
- }
435
- ```
436
-
437
- ---
438
-
439
- *MORPH-SPEC by Polymorphism Tech*
@@ -1,92 +0,0 @@
1
- # Program.cs Setup Checklist (Blazor Server)
2
-
3
- > **Scope:** blazor-azure
4
- > **Layer:** 2 (on keyword)
5
- > **Keywords:** program.cs, startup, configuration, blazor, setup
6
- > **Load When:** blazor project setup
7
-
8
- ## Template
9
-
10
- ```csharp
11
- var builder = WebApplication.CreateBuilder(args);
12
-
13
- // === SERVICES ===
14
- builder.Services.AddRazorComponents().AddInteractiveServerComponents();
15
- builder.Services.AddHttpClient();
16
- builder.Services.AddHttpContextAccessor();
17
- builder.Services.AddScoped(sp =>
18
- {
19
- var client = sp.GetRequiredService<IHttpClientFactory>().CreateClient();
20
- var ctx = sp.GetRequiredService<IHttpContextAccessor>().HttpContext;
21
- if (ctx != null) client.BaseAddress = new Uri($"{ctx.Request.Scheme}://{ctx.Request.Host}");
22
- return client;
23
- });
24
- builder.Services.AddDbContext<AppDbContext>(o =>
25
- o.UseSqlServer(builder.Configuration.GetConnectionString("DefaultConnection")));
26
- // builder.Services.AddFluentUIComponents(); // or AddMudServices()
27
- // builder.Services.AddHangfire(...); builder.Services.AddHangfireServer();
28
-
29
- var app = builder.Build();
30
-
31
- // === MIDDLEWARE (ORDER CRITICAL!) ===
32
- if (!app.Environment.IsDevelopment()) { app.UseExceptionHandler("/Error"); app.UseHsts(); }
33
- app.UseHttpsRedirection();
34
- app.UseStaticFiles(); // MUST be before UseAntiforgery!
35
- app.UseAntiforgery();
36
- app.MapRazorComponents<App>().AddInteractiveServerRenderMode();
37
- // app.MapControllers();
38
- // app.MapHangfireDashboard("/hangfire");
39
- app.Run();
40
- ```
41
-
42
- ## Services Checklist
43
-
44
- | Service | Required | Error if Missing |
45
- |---------|----------|------------------|
46
- | `AddRazorComponents()` + `AddInteractiveServerComponents()` | Yes | Components don't render / static only |
47
- | `AddHttpClient()` + `AddHttpContextAccessor()` + BaseAddress | **Critical** | DI fails / "Invalid URI" errors |
48
- | `AddDbContext()` | If EF Core | Can't inject DbContext |
49
- | `AddFluentUIComponents()` / `AddMudServices()` | If UI lib | Components unstyled |
50
- | `AddHangfire()` + `AddHangfireServer()` | If background jobs | Can't enqueue jobs |
51
-
52
- ## Middleware Order
53
-
54
- | # | Middleware | Required | Error if Wrong |
55
- |---|-----------|----------|----------------|
56
- | 1 | `UseExceptionHandler()` | Prod only | Unhandled exceptions crash |
57
- | 2 | `UseHsts()` | Prod only | HTTP allowed in prod |
58
- | 3 | `UseHttpsRedirection()` | Yes | Security risk |
59
- | 4 | **`UseStaticFiles()`** | **Critical** | Static files 404 |
60
- | 5 | `UseAntiforgery()` | Yes | Form submissions fail |
61
- | 6 | `MapRazorComponents()` | Yes | App doesn't load |
62
- | 7 | `MapControllers()` | If API | API 404 |
63
-
64
- ## Common Mistakes
65
-
66
- | Mistake | Symptom | Fix |
67
- |---------|---------|-----|
68
- | Missing `UseStaticFiles()` | CSS/JS don't apply | Add before `UseAntiforgery()` |
69
- | HttpClient without BaseAddress | `InvalidOperationException: Invalid URI` | Set via `IHttpContextAccessor` (see template) |
70
- | Wrong middleware order | Static 404, forms fail | Follow order above exactly |
71
-
72
- ## UI Library Setup
73
-
74
- **Fluent UI:** `builder.Services.AddFluentUIComponents()` → wrap in `<FluentProvider Theme="@theme">`
75
- **MudBlazor:** `builder.Services.AddMudServices()` → add `<MudThemeProvider/>` + `<MudDialogProvider/>` + `<MudSnackbarProvider/>`
76
- **Hangfire:** `AddHangfire(c => c.UseSqlServerStorage(...))` + `AddHangfireServer()` → `MapHangfireDashboard("/hangfire")`
77
-
78
- ## Quick Checklist
79
-
80
- - [ ] `AddRazorComponents()` + `AddInteractiveServerComponents()`
81
- - [ ] `AddHttpClient()` + `AddHttpContextAccessor()` + BaseAddress
82
- - [ ] `AddDbContext()` (if EF Core)
83
- - [ ] UI library registered
84
- - [ ] `UseStaticFiles()` BEFORE `UseAntiforgery()`
85
- - [ ] Middleware in correct order
86
- - [ ] `MapRazorComponents<App>().AddInteractiveServerRenderMode()`
87
-
88
- > **Ref:** [blazor-pitfalls.md](./blazor-pitfalls.md) | [blazor-lifecycle.md](./blazor-lifecycle.md) | [dotnet10-compatibility.md](./dotnet10-compatibility.md)
89
-
90
- ---
91
-
92
- *MORPH-SPEC by Polymorphism Tech*