@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,407 +0,0 @@
1
- # Azure Storage Services Standard
2
-
3
- > **Scope:** blazor-azure
4
- > **Layer:** 2 (on keyword)
5
- > **Keywords:** azure storage, blob, queue, table, file storage
6
- > **Load When:** azure storage keywords detected
7
-
8
- Azure Storage services patterns for Blob, Queue, Table, and File storage.
9
-
10
- ---
11
-
12
- ## Overview
13
-
14
- Azure Storage provides:
15
- - **Blob Storage**: Object storage for unstructured data (files, images, videos)
16
- - **Queue Storage**: Message queuing for async processing
17
- - **Table Storage**: NoSQL key-value store
18
- - **File Storage**: SMB file shares
19
-
20
- **Stack:** .NET 10 + Blazor Server
21
-
22
- ---
23
-
24
- ## Core Principles
25
-
26
- 1. **Managed Identity**: Use Managed Identity over connection strings
27
- 2. **SAS Tokens**: Use SAS for temporary access delegation
28
- 3. **Retry Policies**: Configure exponential backoff for transient failures
29
- 4. **Lifecycle Management**: Auto-delete old blobs with lifecycle policies
30
- 5. **Cost Optimization**: Use appropriate storage tiers (Hot/Cool/Archive)
31
-
32
- ---
33
-
34
- ## Installation
35
-
36
- ```bash
37
- dotnet add package Azure.Storage.Blobs
38
- dotnet add package Azure.Storage.Queues
39
- dotnet add package Azure.Identity
40
- ```
41
-
42
- ---
43
-
44
- ## Blob Storage
45
-
46
- ### DI Configuration
47
-
48
- ```csharp
49
- // Program.cs
50
- using Azure.Identity;
51
- using Azure.Storage.Blobs;
52
-
53
- builder.Services.AddSingleton(sp =>
54
- {
55
- var blobServiceUri = new Uri(builder.Configuration["AzureStorage:BlobServiceUri"]!);
56
- return new BlobServiceClient(blobServiceUri, new DefaultAzureCredential());
57
- });
58
- ```
59
-
60
- ### Upload Blob
61
-
62
- ```csharp
63
- public class FileUploadService
64
- {
65
- private readonly BlobServiceClient _blobServiceClient;
66
-
67
- public FileUploadService(BlobServiceClient blobServiceClient)
68
- {
69
- _blobServiceClient = blobServiceClient;
70
- }
71
-
72
- public async Task<string> UploadFileAsync(Stream fileStream, string fileName, string containerName)
73
- {
74
- var containerClient = _blobServiceClient.GetBlobContainerClient(containerName);
75
- await containerClient.CreateIfNotExistsAsync();
76
-
77
- var blobClient = containerClient.GetBlobClient(fileName);
78
-
79
- await blobClient.UploadAsync(fileStream, overwrite: true);
80
-
81
- return blobClient.Uri.ToString();
82
- }
83
- }
84
- ```
85
-
86
- ### Download Blob
87
-
88
- ```csharp
89
- public async Task<Stream> DownloadFileAsync(string fileName, string containerName)
90
- {
91
- var containerClient = _blobServiceClient.GetBlobContainerClient(containerName);
92
- var blobClient = containerClient.GetBlobClient(fileName);
93
-
94
- var response = await blobClient.DownloadAsync();
95
- return response.Value.Content;
96
- }
97
- ```
98
-
99
- ### Generate SAS Token
100
-
101
- ```csharp
102
- using Azure.Storage.Sas;
103
-
104
- public string GenerateSasUri(string containerName, string blobName, TimeSpan validity)
105
- {
106
- var containerClient = _blobServiceClient.GetBlobContainerClient(containerName);
107
- var blobClient = containerClient.GetBlobClient(blobName);
108
-
109
- var sasBuilder = new BlobSasBuilder
110
- {
111
- BlobContainerName = containerName,
112
- BlobName = blobName,
113
- Resource = "b", // Blob
114
- StartsOn = DateTimeOffset.UtcNow,
115
- ExpiresOn = DateTimeOffset.UtcNow.Add(validity)
116
- };
117
-
118
- sasBuilder.SetPermissions(BlobSasPermissions.Read);
119
-
120
- var sasUri = blobClient.GenerateSasUri(sasBuilder);
121
- return sasUri.ToString();
122
- }
123
- ```
124
-
125
- ---
126
-
127
- ## Queue Storage
128
-
129
- ### DI Configuration
130
-
131
- ```csharp
132
- // Program.cs
133
- builder.Services.AddSingleton(sp =>
134
- {
135
- var queueServiceUri = new Uri(builder.Configuration["AzureStorage:QueueServiceUri"]!);
136
- return new QueueServiceClient(queueServiceUri, new DefaultAzureCredential());
137
- });
138
- ```
139
-
140
- ### Send Message
141
-
142
- ```csharp
143
- public class QueueService
144
- {
145
- private readonly QueueServiceClient _queueServiceClient;
146
-
147
- public QueueService(QueueServiceClient queueServiceClient)
148
- {
149
- _queueServiceClient = queueServiceClient;
150
- }
151
-
152
- public async Task SendMessageAsync(string queueName, string message)
153
- {
154
- var queueClient = _queueServiceClient.GetQueueClient(queueName);
155
- await queueClient.CreateIfNotExistsAsync();
156
-
157
- await queueClient.SendMessageAsync(message);
158
- }
159
-
160
- public async Task SendMessageWithDelayAsync(string queueName, string message, TimeSpan delay)
161
- {
162
- var queueClient = _queueServiceClient.GetQueueClient(queueName);
163
- await queueClient.CreateIfNotExistsAsync();
164
-
165
- await queueClient.SendMessageAsync(message, visibilityTimeout: delay);
166
- }
167
- }
168
- ```
169
-
170
- ### Process Messages
171
-
172
- ```csharp
173
- using Azure.Storage.Queues.Models;
174
-
175
- public async Task ProcessMessagesAsync(string queueName, CancellationToken cancellationToken)
176
- {
177
- var queueClient = _queueServiceClient.GetQueueClient(queueName);
178
-
179
- while (!cancellationToken.IsCancellationRequested)
180
- {
181
- QueueMessage[] messages = await queueClient.ReceiveMessagesAsync(maxMessages: 32);
182
-
183
- foreach (var message in messages)
184
- {
185
- try
186
- {
187
- // Process message
188
- await ProcessMessageAsync(message.MessageText);
189
-
190
- // Delete message after successful processing
191
- await queueClient.DeleteMessageAsync(message.MessageId, message.PopReceipt);
192
- }
193
- catch (Exception ex)
194
- {
195
- // Log error, message will become visible again after timeout
196
- _logger.LogError(ex, "Failed to process message {MessageId}", message.MessageId);
197
- }
198
- }
199
-
200
- if (messages.Length == 0)
201
- {
202
- await Task.Delay(TimeSpan.FromSeconds(5), cancellationToken);
203
- }
204
- }
205
- }
206
- ```
207
-
208
- ---
209
-
210
- ## Table Storage
211
-
212
- ### DI Configuration
213
-
214
- ```csharp
215
- dotnet add package Azure.Data.Tables
216
-
217
- // Program.cs
218
- builder.Services.AddSingleton(sp =>
219
- {
220
- var tableServiceUri = new Uri(builder.Configuration["AzureStorage:TableServiceUri"]!);
221
- return new TableServiceClient(tableServiceUri, new DefaultAzureCredential());
222
- });
223
- ```
224
-
225
- ### Entity Model
226
-
227
- ```csharp
228
- using Azure;
229
- using Azure.Data.Tables;
230
-
231
- public class ProductEntity : ITableEntity
232
- {
233
- public string PartitionKey { get; set; } = default!; // Category
234
- public string RowKey { get; set; } = default!; // ProductId
235
- public DateTimeOffset? Timestamp { get; set; }
236
- public ETag ETag { get; set; }
237
-
238
- public string Name { get; set; } = string.Empty;
239
- public decimal Price { get; set; }
240
- public int Stock { get; set; }
241
- }
242
- ```
243
-
244
- ### CRUD Operations
245
-
246
- ```csharp
247
- public class ProductTableService
248
- {
249
- private readonly TableClient _tableClient;
250
-
251
- public ProductTableService(TableServiceClient tableServiceClient)
252
- {
253
- _tableClient = tableServiceClient.GetTableClient("products");
254
- _tableClient.CreateIfNotExists();
255
- }
256
-
257
- public async Task AddProductAsync(ProductEntity product)
258
- {
259
- await _tableClient.AddEntityAsync(product);
260
- }
261
-
262
- public async Task<ProductEntity?> GetProductAsync(string category, string productId)
263
- {
264
- try
265
- {
266
- return await _tableClient.GetEntityAsync<ProductEntity>(category, productId);
267
- }
268
- catch (RequestFailedException ex) when (ex.Status == 404)
269
- {
270
- return null;
271
- }
272
- }
273
-
274
- public async Task<List<ProductEntity>> QueryProductsAsync(string category)
275
- {
276
- var query = _tableClient.QueryAsync<ProductEntity>(p => p.PartitionKey == category);
277
-
278
- var results = new List<ProductEntity>();
279
- await foreach (var product in query)
280
- {
281
- results.Add(product);
282
- }
283
-
284
- return results;
285
- }
286
-
287
- public async Task UpdateProductAsync(ProductEntity product)
288
- {
289
- await _tableClient.UpdateEntityAsync(product, ETag.All, TableUpdateMode.Replace);
290
- }
291
-
292
- public async Task DeleteProductAsync(string category, string productId)
293
- {
294
- await _tableClient.DeleteEntityAsync(category, productId);
295
- }
296
- }
297
- ```
298
-
299
- ---
300
-
301
- ## Best Practices
302
-
303
- ### Retry Policy
304
-
305
- ```csharp
306
- using Azure.Core;
307
-
308
- var options = new BlobClientOptions
309
- {
310
- Retry =
311
- {
312
- Mode = RetryMode.Exponential,
313
- MaxRetries = 3,
314
- Delay = TimeSpan.FromSeconds(2),
315
- MaxDelay = TimeSpan.FromSeconds(10)
316
- }
317
- };
318
-
319
- var blobServiceClient = new BlobServiceClient(blobServiceUri, credential, options);
320
- ```
321
-
322
- ### Blob Metadata
323
-
324
- ```csharp
325
- var metadata = new Dictionary<string, string>
326
- {
327
- { "uploadedBy", userId },
328
- { "fileType", "invoice" },
329
- { "department", "finance" }
330
- };
331
-
332
- await blobClient.UploadAsync(stream, metadata: metadata);
333
-
334
- // Retrieve metadata
335
- var properties = await blobClient.GetPropertiesAsync();
336
- var uploadedBy = properties.Value.Metadata["uploadedBy"];
337
- ```
338
-
339
- ### Lifecycle Management
340
-
341
- ```json
342
- // Configure in Azure Portal or via Bicep
343
- {
344
- "rules": [
345
- {
346
- "name": "deleteOldBackups",
347
- "enabled": true,
348
- "type": "Lifecycle",
349
- "definition": {
350
- "filters": {
351
- "blobTypes": ["blockBlob"],
352
- "prefixMatch": ["backups/"]
353
- },
354
- "actions": {
355
- "baseBlob": {
356
- "delete": {
357
- "daysAfterModificationGreaterThan": 30
358
- }
359
- }
360
- }
361
- }
362
- }
363
- ]
364
- }
365
- ```
366
-
367
- ---
368
-
369
- ## Bicep Configuration
370
-
371
- ```bicep
372
- resource storageAccount 'Microsoft.Storage/storageAccounts@2023-01-01' = {
373
- name: storageAccountName
374
- location: location
375
- sku: {
376
- name: 'Standard_LRS'
377
- }
378
- kind: 'StorageV2'
379
- properties: {
380
- allowBlobPublicAccess: false
381
- minimumTlsVersion: 'TLS1_2'
382
- supportsHttpsTrafficOnly: true
383
- }
384
- }
385
-
386
- // Grant Container App Blob access
387
- resource blobRoleAssignment 'Microsoft.Authorization/roleAssignments@2022-04-01' = {
388
- scope: storageAccount
389
- name: guid(storageAccount.id, principalId, 'Storage Blob Data Contributor')
390
- properties: {
391
- roleDefinitionId: subscriptionResourceId('Microsoft.Authorization/roleDefinitions', 'ba92f5b4-2d11-453d-a403-e96b0029c9fe')
392
- principalId: principalId
393
- }
394
- }
395
- ```
396
-
397
- ---
398
-
399
- ## References
400
-
401
- - [Azure Blob Storage](https://learn.microsoft.com/azure/storage/blobs/)
402
- - [Azure Queue Storage](https://learn.microsoft.com/azure/storage/queues/)
403
- - [Azure Table Storage](https://learn.microsoft.com/azure/storage/tables/)
404
-
405
- ---
406
-
407
- *MORPH-SPEC by Polymorphism Tech*
@@ -1,196 +0,0 @@
1
- # EasyPanel Deployment Standard
2
-
3
- > **Scope:** nextjs-supabase
4
- > **Layer:** 2 (on keyword)
5
- > **Keywords:** easypanel, docker, deploy, deployment, container
6
- > **Load When:** easypanel or docker deployment keywords detected
7
-
8
- Stack: Next.js 15 + Supabase + .NET Backend
9
-
10
- ## Core Rules
11
-
12
- - ALWAYS use multi-stage Docker builds for minimal image size
13
- - ALWAYS configure health checks for zero-downtime deploys
14
- - NEVER hardcode secrets in Dockerfiles -- use EasyPanel environment variables
15
- - ALWAYS use `.dockerignore` to exclude node_modules, .git, .env files
16
- - SSL is automatic via Let's Encrypt -- no manual certificate management
17
-
18
- ## .NET 10 Dockerfile
19
-
20
- ```dockerfile
21
- FROM mcr.microsoft.com/dotnet/sdk:10.0 AS build
22
- WORKDIR /src
23
- COPY *.csproj .
24
- RUN dotnet restore
25
- COPY . .
26
- RUN dotnet publish -c Release -o /app/publish --no-restore
27
-
28
- FROM mcr.microsoft.com/dotnet/aspnet:10.0 AS runtime
29
- WORKDIR /app
30
- RUN adduser --disabled-password --gecos "" appuser
31
- USER appuser
32
- COPY --from=build /app/publish .
33
- ENV ASPNETCORE_URLS=http://+:8080
34
- EXPOSE 8080
35
- HEALTHCHECK --interval=30s --timeout=5s --start-period=10s --retries=3 \
36
- CMD curl -f http://localhost:8080/health || exit 1
37
- ENTRYPOINT ["dotnet", "MyApp.Api.dll"]
38
- ```
39
-
40
- ## Next.js Standalone Dockerfile
41
-
42
- ```dockerfile
43
- FROM node:22-alpine AS base
44
-
45
- FROM base AS builder
46
- WORKDIR /app
47
- COPY package.json package-lock.json ./
48
- RUN npm ci
49
- COPY . .
50
- ENV NEXT_TELEMETRY_DISABLED=1
51
- RUN npm run build
52
-
53
- FROM base AS runner
54
- WORKDIR /app
55
- ENV NODE_ENV=production NEXT_TELEMETRY_DISABLED=1
56
- RUN addgroup --system --gid 1001 nodejs && adduser --system --uid 1001 nextjs
57
- COPY --from=builder /app/public ./public
58
- COPY --from=builder --chown=nextjs:nodejs /app/.next/standalone ./
59
- COPY --from=builder --chown=nextjs:nodejs /app/.next/static ./.next/static
60
- USER nextjs
61
- EXPOSE 3000
62
- ENV PORT=3000 HOSTNAME="0.0.0.0"
63
- HEALTHCHECK --interval=30s --timeout=5s --start-period=10s --retries=3 \
64
- CMD wget --no-verbose --tries=1 --spider http://localhost:3000/api/health || exit 1
65
- CMD ["node", "server.js"]
66
- ```
67
-
68
- Required: `output: "standalone"` in `next.config.ts`.
69
-
70
- ## .dockerignore
71
-
72
- ```
73
- node_modules
74
- .next
75
- .git
76
- .env*
77
- *.md
78
- .vscode
79
- coverage
80
- test
81
- __tests__
82
- ```
83
-
84
- ## EasyPanel Service Config
85
-
86
- | Setting | .NET Backend | Next.js Frontend |
87
- |---------|-------------|-----------------|
88
- | Source | GitHub | GitHub |
89
- | Build method | Dockerfile | Dockerfile |
90
- | Dockerfile path | `./backend/Dockerfile` | `./frontend/Dockerfile` |
91
- | Port | 8080 | 3000 |
92
- | Domain | api.example.com | app.example.com |
93
-
94
- ### GitHub Integration
95
-
96
- 1. Generate GitHub PAT with `repo` scope
97
- 2. EasyPanel: Settings > GitHub > Add token
98
- 3. Select repository and branch
99
- 4. Enable "Auto Deploy" for webhook-triggered deploys
100
-
101
- ### Domain and SSL
102
-
103
- DNS setup (CNAME to EasyPanel server):
104
- ```
105
- app.example.com CNAME your-server.easypanel.host
106
- api.example.com CNAME your-server.easypanel.host
107
- ```
108
-
109
- SSL via Let's Encrypt is automatic. Force HTTPS enabled by default.
110
-
111
- ## Environment Variables
112
-
113
- ### .NET Backend
114
-
115
- ```env
116
- ASPNETCORE_ENVIRONMENT=Production
117
- ASPNETCORE_URLS=http://+:8080
118
- ConnectionStrings__DefaultConnection=Host=...;Database=...;Username=...;Password=...
119
- Supabase__Url=https://xxx.supabase.co
120
- Supabase__ServiceRoleKey=eyJ...
121
- Supabase__JwtSecret=your-jwt-secret
122
- ```
123
-
124
- ### Next.js Frontend
125
-
126
- ```env
127
- NEXT_PUBLIC_SUPABASE_URL=https://xxx.supabase.co
128
- NEXT_PUBLIC_SUPABASE_ANON_KEY=eyJ...
129
- NEXT_PUBLIC_API_URL=https://api.example.com
130
- ```
131
-
132
- ## Health Check Endpoints
133
-
134
- ```csharp
135
- // .NET: Program.cs
136
- builder.Services.AddHealthChecks()
137
- .AddNpgSql(connectionString, name: "database");
138
- app.MapHealthChecks("/health");
139
- ```
140
-
141
- ```ts
142
- // Next.js: app/api/health/route.ts
143
- export async function GET() {
144
- return Response.json({ status: "healthy", timestamp: new Date().toISOString() });
145
- }
146
- ```
147
-
148
- ## Zero-Downtime Deploys
149
-
150
- EasyPanel uses Docker HEALTHCHECK to determine container readiness:
151
-
152
- 1. New container starts alongside old container
153
- 2. Health check passes after start-period + retries
154
- 3. Traffic shifts to new container
155
- 4. Old container stopped
156
-
157
- | HEALTHCHECK Param | Value | Purpose |
158
- |-------------------|-------|---------|
159
- | `--interval` | 30s | Time between checks |
160
- | `--timeout` | 5s | Max response wait |
161
- | `--start-period` | 10s | Startup grace period |
162
- | `--retries` | 3 | Failures before unhealthy |
163
-
164
- ## Monitoring
165
-
166
- - **Logs**: Real-time in EasyPanel UI
167
- - **Metrics**: CPU, memory, network via dashboard
168
- - **Restart**: Auto-restart on crash (default)
169
-
170
- Structured logging:
171
-
172
- ```csharp
173
- // .NET: Serilog with JSON output
174
- builder.Host.UseSerilog((ctx, cfg) => cfg
175
- .ReadFrom.Configuration(ctx.Configuration)
176
- .WriteTo.Console(new JsonFormatter()));
177
- ```
178
-
179
- ```ts
180
- // Next.js: pino
181
- import pino from "pino";
182
- const logger = pino({ level: process.env.LOG_LEVEL ?? "info" });
183
- ```
184
-
185
- ## Deployment Checklist
186
-
187
- | Step | Action |
188
- |------|--------|
189
- | 1 | Verify `output: "standalone"` in next.config.ts |
190
- | 2 | Test Docker build locally |
191
- | 3 | Test health endpoint |
192
- | 4 | Configure env vars in EasyPanel |
193
- | 5 | Set up custom domain + DNS |
194
- | 6 | Verify SSL certificate |
195
- | 7 | Enable auto-deploy from GitHub |
196
- | 8 | Push to main, verify deployment |