@polymorphism-tech/morph-spec 4.6.0 → 4.7.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 (239) hide show
  1. package/README.md +394 -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,459 +0,0 @@
1
- # Azure Service Bus Standard
2
-
3
- > **Scope:** blazor-azure
4
- > **Layer:** 2 (on keyword)
5
- > **Keywords:** service bus, messaging, queue, topic, subscription
6
- > **Load When:** service bus or azure messaging keywords detected
7
-
8
- Enterprise messaging patterns with Azure Service Bus.
9
-
10
- ---
11
-
12
- ## Overview
13
-
14
- Azure Service Bus provides:
15
- - **Queues**: Point-to-point messaging with FIFO guarantee
16
- - **Topics/Subscriptions**: Publish-subscribe patterns
17
- - **Dead-letter queues**: Failed message handling
18
- - **Message sessions**: Ordered processing
19
- - **Scheduled messages**: Delayed delivery
20
-
21
- **Use Cases:** Event-driven architectures, microservices communication, async workflows
22
-
23
- ---
24
-
25
- ## Core Principles
26
-
27
- 1. **At-Least-Once Delivery**: Messages delivered at least once
28
- 2. **Managed Identity**: Use Managed Identity over connection strings
29
- 3. **Idempotent Handlers**: Handle duplicate messages gracefully
30
- 4. **Dead-Letter Management**: Monitor and reprocess failed messages
31
- 5. **Sessions for Ordering**: Use sessions when message order matters
32
-
33
- ---
34
-
35
- ## Installation
36
-
37
- ```bash
38
- dotnet add package Azure.Messaging.ServiceBus
39
- dotnet add package Azure.Identity
40
- ```
41
-
42
- ---
43
-
44
- ## Queue Patterns
45
-
46
- ### DI Configuration
47
-
48
- ```csharp
49
- // Program.cs
50
- using Azure.Identity;
51
- using Azure.Messaging.ServiceBus;
52
-
53
- builder.Services.AddSingleton(sp =>
54
- {
55
- var fullyQualifiedNamespace = builder.Configuration["ServiceBus:FullyQualifiedNamespace"]!;
56
- return new ServiceBusClient(fullyQualifiedNamespace, new DefaultAzureCredential());
57
- });
58
- ```
59
-
60
- ### Send Message
61
-
62
- ```csharp
63
- public class OrderQueueService
64
- {
65
- private readonly ServiceBusClient _serviceBusClient;
66
- private readonly string _queueName = "orders";
67
-
68
- public OrderQueueService(ServiceBusClient serviceBusClient)
69
- {
70
- _serviceBusClient = serviceBusClient;
71
- }
72
-
73
- public async Task SendOrderAsync(string orderId, string orderData)
74
- {
75
- await using var sender = _serviceBusClient.CreateSender(_queueName);
76
-
77
- var message = new ServiceBusMessage(orderData)
78
- {
79
- MessageId = orderId,
80
- ContentType = "application/json",
81
- Subject = "OrderCreated"
82
- };
83
-
84
- // Add custom properties
85
- message.ApplicationProperties.Add("OrderId", orderId);
86
- message.ApplicationProperties.Add("Priority", "High");
87
-
88
- await sender.SendMessageAsync(message);
89
- }
90
-
91
- public async Task ScheduleOrderAsync(string orderId, string orderData, DateTimeOffset scheduleTime)
92
- {
93
- await using var sender = _serviceBusClient.CreateSender(_queueName);
94
-
95
- var message = new ServiceBusMessage(orderData)
96
- {
97
- MessageId = orderId
98
- };
99
-
100
- await sender.ScheduleMessageAsync(message, scheduleTime);
101
- }
102
- }
103
- ```
104
-
105
- ### Receive Messages
106
-
107
- ```csharp
108
- public class OrderProcessorService : BackgroundService
109
- {
110
- private readonly ServiceBusClient _serviceBusClient;
111
- private readonly ILogger<OrderProcessorService> _logger;
112
- private readonly string _queueName = "orders";
113
-
114
- public OrderProcessorService(
115
- ServiceBusClient serviceBusClient,
116
- ILogger<OrderProcessorService> logger)
117
- {
118
- _serviceBusClient = serviceBusClient;
119
- _logger = logger;
120
- }
121
-
122
- protected override async Task ExecuteAsync(CancellationToken stoppingToken)
123
- {
124
- await using var processor = _serviceBusClient.CreateProcessor(_queueName, new ServiceBusProcessorOptions
125
- {
126
- MaxConcurrentCalls = 10,
127
- AutoCompleteMessages = false,
128
- PrefetchCount = 20
129
- });
130
-
131
- processor.ProcessMessageAsync += ProcessMessageAsync;
132
- processor.ProcessErrorAsync += ProcessErrorAsync;
133
-
134
- await processor.StartProcessingAsync(stoppingToken);
135
-
136
- // Wait until cancellation
137
- await Task.Delay(Timeout.Infinite, stoppingToken);
138
-
139
- await processor.StopProcessingAsync();
140
- }
141
-
142
- private async Task ProcessMessageAsync(ProcessMessageEventArgs args)
143
- {
144
- var body = args.Message.Body.ToString();
145
- var orderId = args.Message.ApplicationProperties["OrderId"].ToString();
146
-
147
- try
148
- {
149
- _logger.LogInformation("Processing order {OrderId}", orderId);
150
-
151
- // Process order
152
- await ProcessOrderAsync(body);
153
-
154
- // Complete the message
155
- await args.CompleteMessageAsync(args.Message);
156
- }
157
- catch (Exception ex)
158
- {
159
- _logger.LogError(ex, "Failed to process order {OrderId}", orderId);
160
-
161
- // Dead-letter the message
162
- await args.DeadLetterMessageAsync(args.Message, "ProcessingFailed", ex.Message);
163
- }
164
- }
165
-
166
- private Task ProcessErrorAsync(ProcessErrorEventArgs args)
167
- {
168
- _logger.LogError(args.Exception, "Service Bus error: {ErrorSource}", args.ErrorSource);
169
- return Task.CompletedTask;
170
- }
171
-
172
- private Task ProcessOrderAsync(string orderData)
173
- {
174
- // Implementation...
175
- return Task.CompletedTask;
176
- }
177
- }
178
- ```
179
-
180
- ---
181
-
182
- ## Topic/Subscription Patterns
183
-
184
- ### Send to Topic
185
-
186
- ```csharp
187
- public class EventPublisherService
188
- {
189
- private readonly ServiceBusClient _serviceBusClient;
190
- private readonly string _topicName = "order-events";
191
-
192
- public EventPublisherService(ServiceBusClient serviceBusClient)
193
- {
194
- _serviceBusClient = serviceBusClient;
195
- }
196
-
197
- public async Task PublishOrderCreatedAsync(string orderId)
198
- {
199
- await using var sender = _serviceBusClient.CreateSender(_topicName);
200
-
201
- var message = new ServiceBusMessage(JsonSerializer.Serialize(new
202
- {
203
- OrderId = orderId,
204
- Timestamp = DateTime.UtcNow
205
- }))
206
- {
207
- Subject = "OrderCreated",
208
- CorrelationId = Guid.NewGuid().ToString()
209
- };
210
-
211
- await sender.SendMessageAsync(message);
212
- }
213
- }
214
- ```
215
-
216
- ### Subscribe to Topic
217
-
218
- ```csharp
219
- public class NotificationSubscriberService : BackgroundService
220
- {
221
- private readonly ServiceBusClient _serviceBusClient;
222
- private readonly string _topicName = "order-events";
223
- private readonly string _subscriptionName = "notification-service";
224
-
225
- protected override async Task ExecuteAsync(CancellationToken stoppingToken)
226
- {
227
- await using var processor = _serviceBusClient.CreateProcessor(
228
- _topicName,
229
- _subscriptionName,
230
- new ServiceBusProcessorOptions
231
- {
232
- MaxConcurrentCalls = 5,
233
- AutoCompleteMessages = false
234
- });
235
-
236
- processor.ProcessMessageAsync += async args =>
237
- {
238
- if (args.Message.Subject == "OrderCreated")
239
- {
240
- await SendNotificationAsync(args.Message.Body.ToString());
241
- await args.CompleteMessageAsync(args.Message);
242
- }
243
- };
244
-
245
- processor.ProcessErrorAsync += args =>
246
- {
247
- _logger.LogError(args.Exception, "Notification subscriber error");
248
- return Task.CompletedTask;
249
- };
250
-
251
- await processor.StartProcessingAsync(stoppingToken);
252
- await Task.Delay(Timeout.Infinite, stoppingToken);
253
- await processor.StopProcessingAsync();
254
- }
255
-
256
- private Task SendNotificationAsync(string orderData)
257
- {
258
- // Send notification...
259
- return Task.CompletedTask;
260
- }
261
- }
262
- ```
263
-
264
- ---
265
-
266
- ## Message Sessions (Ordered Processing)
267
-
268
- ```csharp
269
- public async Task SendSessionMessageAsync(string sessionId, string message)
270
- {
271
- await using var sender = _serviceBusClient.CreateSender("orders");
272
-
273
- var serviceBusMessage = new ServiceBusMessage(message)
274
- {
275
- SessionId = sessionId // All messages with same SessionId processed in order
276
- };
277
-
278
- await sender.SendMessageAsync(serviceBusMessage);
279
- }
280
-
281
- // Process with sessions
282
- var processor = _serviceBusClient.CreateSessionProcessor("orders", new ServiceBusSessionProcessorOptions
283
- {
284
- MaxConcurrentSessions = 5,
285
- SessionIdleTimeout = TimeSpan.FromMinutes(1)
286
- });
287
-
288
- processor.ProcessMessageAsync += async args =>
289
- {
290
- _logger.LogInformation("Processing message in session {SessionId}", args.SessionId);
291
- await ProcessMessageAsync(args.Message.Body.ToString());
292
- await args.CompleteMessageAsync(args.Message);
293
- };
294
- ```
295
-
296
- ---
297
-
298
- ## Dead-Letter Queue Processing
299
-
300
- ```csharp
301
- public async Task ReprocessDeadLetterMessagesAsync()
302
- {
303
- var deadLetterQueuePath = $"orders/$DeadLetterQueue";
304
-
305
- await using var receiver = _serviceBusClient.CreateReceiver(deadLetterQueuePath);
306
-
307
- var messages = await receiver.ReceiveMessagesAsync(maxMessages: 100);
308
-
309
- foreach (var message in messages)
310
- {
311
- try
312
- {
313
- // Attempt to reprocess
314
- await ProcessMessageAsync(message.Body.ToString());
315
-
316
- // Complete from dead-letter queue
317
- await receiver.CompleteMessageAsync(message);
318
-
319
- // Optionally, resend to main queue
320
- await using var sender = _serviceBusClient.CreateSender("orders");
321
- await sender.SendMessageAsync(new ServiceBusMessage(message.Body)
322
- {
323
- MessageId = message.MessageId
324
- });
325
- }
326
- catch (Exception ex)
327
- {
328
- _logger.LogError(ex, "Failed to reprocess dead-letter message {MessageId}", message.MessageId);
329
- }
330
- }
331
- }
332
- ```
333
-
334
- ---
335
-
336
- ## Bicep Configuration
337
-
338
- ```bicep
339
- resource serviceBusNamespace 'Microsoft.ServiceBus/namespaces@2022-10-01-preview' = {
340
- name: serviceBusNamespaceName
341
- location: location
342
- sku: {
343
- name: 'Standard'
344
- tier: 'Standard'
345
- }
346
- }
347
-
348
- resource queue 'Microsoft.ServiceBus/namespaces/queues@2022-10-01-preview' = {
349
- parent: serviceBusNamespace
350
- name: 'orders'
351
- properties: {
352
- maxDeliveryCount: 10
353
- deadLetteringOnMessageExpiration: true
354
- defaultMessageTimeToLive: 'P14D' // 14 days
355
- lockDuration: 'PT5M' // 5 minutes
356
- }
357
- }
358
-
359
- resource topic 'Microsoft.ServiceBus/namespaces/topics@2022-10-01-preview' = {
360
- parent: serviceBusNamespace
361
- name: 'order-events'
362
- properties: {
363
- defaultMessageTimeToLive: 'P7D'
364
- }
365
- }
366
-
367
- resource subscription 'Microsoft.ServiceBus/namespaces/topics/subscriptions@2022-10-01-preview' = {
368
- parent: topic
369
- name: 'notification-service'
370
- properties: {
371
- deadLetteringOnMessageExpiration: true
372
- maxDeliveryCount: 5
373
- }
374
- }
375
-
376
- // Grant Container App send/receive access
377
- resource serviceBusRole 'Microsoft.Authorization/roleAssignments@2022-04-01' = {
378
- scope: serviceBusNamespace
379
- name: guid(serviceBusNamespace.id, principalId, 'Azure Service Bus Data Owner')
380
- properties: {
381
- roleDefinitionId: subscriptionResourceId('Microsoft.Authorization/roleDefinitions', '090c5cfd-751d-490a-894a-3ce6f1109419')
382
- principalId: principalId
383
- }
384
- }
385
- ```
386
-
387
- ---
388
-
389
- ## Best Practices
390
-
391
- ### Idempotent Message Handling
392
-
393
- ```csharp
394
- private async Task<bool> IsMessageProcessedAsync(string messageId)
395
- {
396
- await using var db = await _dbFactory.CreateDbContextAsync();
397
- return await db.ProcessedMessages.AnyAsync(m => m.MessageId == messageId);
398
- }
399
-
400
- private async Task ProcessMessageAsync(ProcessMessageEventArgs args)
401
- {
402
- if (await IsMessageProcessedAsync(args.Message.MessageId))
403
- {
404
- // Already processed, complete and skip
405
- await args.CompleteMessageAsync(args.Message);
406
- return;
407
- }
408
-
409
- // Process message...
410
-
411
- // Mark as processed
412
- await using var db = await _dbFactory.CreateDbContextAsync();
413
- db.ProcessedMessages.Add(new ProcessedMessage
414
- {
415
- MessageId = args.Message.MessageId,
416
- ProcessedAt = DateTime.UtcNow
417
- });
418
- await db.SaveChangesAsync();
419
-
420
- await args.CompleteMessageAsync(args.Message);
421
- }
422
- ```
423
-
424
- ### Exponential Backoff on Errors
425
-
426
- ```csharp
427
- private async Task ProcessWithRetryAsync(ServiceBusReceivedMessage message)
428
- {
429
- int maxRetries = 3;
430
- int attempt = 0;
431
-
432
- while (attempt < maxRetries)
433
- {
434
- try
435
- {
436
- await ProcessMessageAsync(message.Body.ToString());
437
- return;
438
- }
439
- catch (Exception ex)
440
- {
441
- attempt++;
442
- if (attempt >= maxRetries) throw;
443
-
444
- await Task.Delay(TimeSpan.FromSeconds(Math.Pow(2, attempt)));
445
- }
446
- }
447
- }
448
- ```
449
-
450
- ---
451
-
452
- ## References
453
-
454
- - [Azure Service Bus Documentation](https://learn.microsoft.com/azure/service-bus-messaging/)
455
- - [Service Bus .NET SDK](https://learn.microsoft.com/dotnet/api/overview/azure/messaging.servicebus-readme)
456
-
457
- ---
458
-
459
- *MORPH-SPEC by Polymorphism Tech*