@polymorphism-tech/morph-spec 3.0.0 → 3.0.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 (160) hide show
  1. package/CLAUDE.md +75 -371
  2. package/LICENSE +72 -72
  3. package/bin/detect-agents.js +225 -225
  4. package/bin/render-template.js +302 -302
  5. package/bin/semantic-detect-agents.js +246 -246
  6. package/bin/validate-agents-skills.js +251 -251
  7. package/bin/validate-agents.js +69 -69
  8. package/bin/validate-phase.js +263 -263
  9. package/content/.azure/README.md +293 -293
  10. package/content/.azure/docs/azure-devops-setup.md +454 -454
  11. package/content/.azure/docs/branch-strategy.md +398 -398
  12. package/content/.azure/docs/local-development.md +515 -515
  13. package/content/.azure/pipelines/pipeline-variables.yml +34 -34
  14. package/content/.azure/pipelines/prod-pipeline.yml +319 -319
  15. package/content/.azure/pipelines/staging-pipeline.yml +234 -234
  16. package/content/.azure/pipelines/templates/build-dotnet.yml +75 -75
  17. package/content/.azure/pipelines/templates/deploy-app-service.yml +94 -94
  18. package/content/.azure/pipelines/templates/deploy-container-app.yml +120 -120
  19. package/content/.azure/pipelines/templates/infra-deploy.yml +90 -90
  20. package/content/.claude/commands/morph-archive.md +79 -79
  21. package/content/.claude/commands/morph-deploy.md +529 -529
  22. package/content/.claude/commands/morph-infra.md +209 -209
  23. package/content/.claude/commands/morph-preflight.md +227 -227
  24. package/content/.claude/commands/morph-troubleshoot.md +122 -122
  25. package/content/.claude/settings.local.json +15 -15
  26. package/content/.claude/skills/{specialists → level-2-domains/architecture}/prompt-engineer.md +189 -189
  27. package/content/.claude/skills/{specialists → level-2-domains/architecture}/seo-growth-hacker.md +320 -320
  28. package/content/.claude/skills/{infra → level-2-domains/infrastructure}/azure-deploy-specialist.md +699 -699
  29. package/content/.morph/.morphversion +5 -5
  30. package/content/.morph/archive/.gitkeep +25 -25
  31. package/content/.morph/config/agents.json +7 -5
  32. package/content/.morph/docs/STORY-DRIVEN-DEVELOPMENT.md +392 -392
  33. package/content/.morph/examples/api-nextjs/README.md +241 -241
  34. package/content/.morph/examples/api-nextjs/contracts.ts +307 -307
  35. package/content/.morph/examples/api-nextjs/spec.md +399 -399
  36. package/content/.morph/examples/api-nextjs/tasks.md +168 -168
  37. package/content/.morph/examples/micro-saas/README.md +125 -125
  38. package/content/.morph/examples/micro-saas/contracts.cs +358 -358
  39. package/content/.morph/examples/micro-saas/decisions.md +246 -246
  40. package/content/.morph/examples/micro-saas/spec.md +236 -236
  41. package/content/.morph/examples/micro-saas/tasks.md +150 -150
  42. package/content/.morph/examples/multi-agent/README.md +309 -309
  43. package/content/.morph/examples/multi-agent/contracts.cs +433 -433
  44. package/content/.morph/examples/multi-agent/spec.md +479 -479
  45. package/content/.morph/examples/multi-agent/tasks.md +185 -185
  46. package/content/.morph/examples/state-v3.json +188 -188
  47. package/content/.morph/features/.gitkeep +25 -25
  48. package/content/.morph/hooks/pre-commit-all.sh +48 -48
  49. package/content/.morph/hooks/pre-commit-specs.sh +49 -49
  50. package/content/.morph/hooks/pre-commit-tests.sh +60 -60
  51. package/content/.morph/project.md +160 -160
  52. package/content/.morph/schemas/agent.schema.json +296 -296
  53. package/content/.morph/specs/.gitkeep +20 -20
  54. package/content/.morph/standards/coding.md +377 -377
  55. package/content/.morph/standards/fluent-ui-setup.md +590 -590
  56. package/content/.morph/standards/migration-guide.md +514 -514
  57. package/content/.morph/standards/passkeys-auth.md +423 -423
  58. package/content/.morph/standards/vector-search-rag.md +536 -536
  59. package/content/.morph/state.json +17 -17
  60. package/content/.morph/templates/FluentDesignTheme.cs +149 -149
  61. package/content/.morph/templates/MudTheme.cs +281 -281
  62. package/content/.morph/templates/component.razor +239 -239
  63. package/content/.morph/templates/contracts.cs +217 -217
  64. package/content/.morph/templates/design-system.css +226 -226
  65. package/content/.morph/templates/infra/.dockerignore.example +89 -89
  66. package/content/.morph/templates/infra/Dockerfile.example +82 -82
  67. package/content/.morph/templates/infra/README.md +286 -286
  68. package/content/.morph/templates/infra/app-insights.bicep +63 -63
  69. package/content/.morph/templates/infra/app-service.bicep +164 -164
  70. package/content/.morph/templates/infra/azure-pipelines-deploy.yml +480 -480
  71. package/content/.morph/templates/infra/container-app-env.bicep +49 -49
  72. package/content/.morph/templates/infra/container-app.bicep +156 -156
  73. package/content/.morph/templates/infra/deploy-checklist.md +426 -426
  74. package/content/.morph/templates/infra/deploy.ps1 +229 -229
  75. package/content/.morph/templates/infra/deploy.sh +208 -208
  76. package/content/.morph/templates/infra/key-vault.bicep +91 -91
  77. package/content/.morph/templates/infra/main.bicep +189 -189
  78. package/content/.morph/templates/infra/parameters.dev.json +29 -29
  79. package/content/.morph/templates/infra/parameters.prod.json +29 -29
  80. package/content/.morph/templates/infra/parameters.staging.json +29 -29
  81. package/content/.morph/templates/infra/sql-database.bicep +103 -103
  82. package/content/.morph/templates/infra/storage.bicep +106 -106
  83. package/content/.morph/templates/integrations/asaas-client.cs +387 -387
  84. package/content/.morph/templates/integrations/asaas-webhook.cs +351 -351
  85. package/content/.morph/templates/integrations/azure-identity-config.cs +288 -288
  86. package/content/.morph/templates/integrations/clerk-config.cs +258 -258
  87. package/content/.morph/templates/job.cs +171 -171
  88. package/content/.morph/templates/migration.cs +83 -83
  89. package/content/.morph/templates/repository.cs +141 -141
  90. package/content/.morph/templates/saas/subscription.cs +347 -347
  91. package/content/.morph/templates/saas/tenant.cs +338 -338
  92. package/content/.morph/templates/service.cs +139 -139
  93. package/content/.morph/templates/sprint-status.yaml +68 -68
  94. package/content/.morph/templates/story.md +143 -143
  95. package/content/.morph/templates/test.cs +239 -239
  96. package/content/.morph/templates/ui-design-system.md +286 -286
  97. package/content/.morph/templates/ui-flows.md +336 -336
  98. package/content/.morph/templates/ui-mockups.md +133 -133
  99. package/content/.morph/test-infra/example.bicep +59 -59
  100. package/content/README.md +79 -79
  101. package/docs/api/fonts/Source-Sans-Pro/sourcesanspro-light-webfont.svg +977 -977
  102. package/docs/api/fonts/Source-Sans-Pro/sourcesanspro-regular-webfont.svg +1048 -1048
  103. package/docs/api/scripts/collapse.js +38 -38
  104. package/docs/api/scripts/commonNav.js +28 -28
  105. package/docs/api/scripts/linenumber.js +25 -25
  106. package/docs/api/scripts/nav.js +12 -12
  107. package/docs/api/scripts/polyfill.js +3 -3
  108. package/docs/api/scripts/prettify/Apache-License-2.0.txt +202 -202
  109. package/docs/api/scripts/prettify/lang-css.js +2 -2
  110. package/docs/api/scripts/prettify/prettify.js +28 -28
  111. package/docs/api/scripts/search.js +98 -98
  112. package/docs/api/styles/jsdoc.css +776 -776
  113. package/docs/api/styles/prettify.css +80 -80
  114. package/docs/examples.md +328 -328
  115. package/docs/templates.md +418 -418
  116. package/package.json +1 -2
  117. package/scripts/postinstall.js +132 -132
  118. package/scripts/reorganize-skills.cjs +175 -0
  119. package/scripts/validate-agents-structure.cjs +52 -0
  120. package/scripts/validate-skills.cjs +180 -0
  121. package/src/commands/analyze-blazor-concurrency.js +193 -193
  122. package/src/commands/create-story.js +351 -351
  123. package/src/commands/deploy.js +780 -780
  124. package/src/commands/detect-agents.js +9 -0
  125. package/src/commands/detect.js +104 -104
  126. package/src/commands/generate.js +149 -149
  127. package/src/commands/lint-fluent.js +352 -352
  128. package/src/commands/rollback-phase.js +185 -185
  129. package/src/commands/session-summary.js +291 -291
  130. package/src/commands/shard-spec.js +224 -224
  131. package/src/commands/sprint-status.js +250 -250
  132. package/src/commands/state.js +334 -333
  133. package/src/commands/sync.js +167 -167
  134. package/src/commands/troubleshoot.js +222 -222
  135. package/src/commands/update.js +13 -1
  136. package/src/commands/validate-blazor-state.js +210 -210
  137. package/src/commands/validate-blazor.js +156 -156
  138. package/src/commands/validate-css.js +84 -84
  139. package/src/commands/validate-phase.js +221 -221
  140. package/src/lib/blazor-concurrency-analyzer.js +288 -288
  141. package/src/lib/blazor-state-validator.js +291 -291
  142. package/src/lib/blazor-validator.js +374 -374
  143. package/src/lib/css-validator.js +352 -352
  144. package/src/lib/design-system-generator.js +298 -298
  145. package/{detectors → src/lib/detectors}/config-detector.js +223 -223
  146. package/{detectors → src/lib/detectors}/conversation-analyzer.js +163 -163
  147. package/{detectors → src/lib/detectors}/index.js +84 -84
  148. package/{detectors → src/lib/detectors}/standards-generator.js +275 -275
  149. package/src/lib/learning-system.js +520 -520
  150. package/src/lib/mockup-generator.js +366 -366
  151. package/src/lib/state-manager.js +21 -4
  152. package/src/lib/troubleshoot-grep.js +194 -194
  153. package/src/lib/troubleshoot-index.js +144 -144
  154. package/src/lib/ui-detector.js +350 -350
  155. package/src/lib/validators/architecture-validator.js +387 -387
  156. package/src/lib/validators/package-validator.js +360 -360
  157. package/src/lib/validators/ui-contrast-validator.js +422 -422
  158. package/src/utils/logger.js +32 -32
  159. package/src/utils/version-checker.js +175 -175
  160. /package/{detectors → src/lib/detectors}/structure-detector.js +0 -0
@@ -1,347 +1,347 @@
1
- // ==============================================================================
2
- // MORPH-SPEC - Subscription Model Template
3
- // Modelo de assinatura para SaaS
4
- // ==============================================================================
5
-
6
- using System.ComponentModel.DataAnnotations;
7
-
8
- namespace {{Namespace}}.Domain.Entities;
9
-
10
- // ==============================================================================
11
- // SUBSCRIPTION ENTITY
12
- // ==============================================================================
13
-
14
- public class Subscription : BaseEntity
15
- {
16
- public int TenantId { get; private set; }
17
- public Tenant Tenant { get; private set; } = null!;
18
-
19
- public int PlanId { get; private set; }
20
- public Plan Plan { get; private set; } = null!;
21
-
22
- public SubscriptionStatus Status { get; private set; }
23
- public BillingCycle Cycle { get; private set; }
24
-
25
- public DateTime StartDate { get; private set; }
26
- public DateTime? EndDate { get; private set; }
27
- public DateTime? CancelledAt { get; private set; }
28
- public DateTime NextBillingDate { get; private set; }
29
-
30
- public decimal CurrentPrice { get; private set; }
31
- public string? ExternalSubscriptionId { get; private set; } // Asaas subscription ID
32
-
33
- public ICollection<SubscriptionPayment> Payments { get; private set; } = new List<SubscriptionPayment>();
34
-
35
- // =========================================================================
36
- // FACTORY
37
- // =========================================================================
38
-
39
- public static Subscription Create(int tenantId, int planId, BillingCycle cycle, decimal price)
40
- {
41
- return new Subscription
42
- {
43
- TenantId = tenantId,
44
- PlanId = planId,
45
- Status = SubscriptionStatus.PendingPayment,
46
- Cycle = cycle,
47
- StartDate = DateTime.UtcNow,
48
- NextBillingDate = DateTime.UtcNow,
49
- CurrentPrice = price,
50
- CreatedAt = DateTime.UtcNow
51
- };
52
- }
53
-
54
- // =========================================================================
55
- // BEHAVIORS
56
- // =========================================================================
57
-
58
- public void Activate(string externalSubscriptionId)
59
- {
60
- Status = SubscriptionStatus.Active;
61
- ExternalSubscriptionId = externalSubscriptionId;
62
- UpdatedAt = DateTime.UtcNow;
63
- }
64
-
65
- public void Suspend()
66
- {
67
- Status = SubscriptionStatus.Suspended;
68
- UpdatedAt = DateTime.UtcNow;
69
- }
70
-
71
- public void Cancel(DateTime? endDate = null)
72
- {
73
- Status = SubscriptionStatus.Cancelled;
74
- CancelledAt = DateTime.UtcNow;
75
- EndDate = endDate ?? DateTime.UtcNow;
76
- UpdatedAt = DateTime.UtcNow;
77
- }
78
-
79
- public void Renew()
80
- {
81
- NextBillingDate = CalculateNextBillingDate();
82
- UpdatedAt = DateTime.UtcNow;
83
- }
84
-
85
- public void ChangePlan(int newPlanId, decimal newPrice)
86
- {
87
- PlanId = newPlanId;
88
- CurrentPrice = newPrice;
89
- UpdatedAt = DateTime.UtcNow;
90
- }
91
-
92
- public void AddPayment(SubscriptionPayment payment)
93
- {
94
- Payments.Add(payment);
95
- }
96
-
97
- // =========================================================================
98
- // QUERIES
99
- // =========================================================================
100
-
101
- public bool IsActive => Status == SubscriptionStatus.Active;
102
- public bool IsCancelled => Status == SubscriptionStatus.Cancelled;
103
- public bool IsSuspended => Status == SubscriptionStatus.Suspended;
104
- public bool IsExpired => EndDate.HasValue && EndDate.Value < DateTime.UtcNow;
105
-
106
- public int DaysUntilRenewal => (NextBillingDate - DateTime.UtcNow).Days;
107
- public bool IsNearRenewal => DaysUntilRenewal <= 3;
108
-
109
- private DateTime CalculateNextBillingDate()
110
- {
111
- return Cycle switch
112
- {
113
- BillingCycle.Monthly => NextBillingDate.AddMonths(1),
114
- BillingCycle.Quarterly => NextBillingDate.AddMonths(3),
115
- BillingCycle.Yearly => NextBillingDate.AddYears(1),
116
- _ => NextBillingDate.AddMonths(1)
117
- };
118
- }
119
- }
120
-
121
- // ==============================================================================
122
- // SUBSCRIPTION PAYMENT ENTITY
123
- // ==============================================================================
124
-
125
- public class SubscriptionPayment : BaseEntity
126
- {
127
- public int SubscriptionId { get; private set; }
128
- public Subscription Subscription { get; private set; } = null!;
129
-
130
- public decimal Amount { get; private set; }
131
- public PaymentStatus Status { get; private set; }
132
- public PaymentMethod Method { get; private set; }
133
-
134
- public DateTime DueDate { get; private set; }
135
- public DateTime? PaidAt { get; private set; }
136
-
137
- public string? ExternalPaymentId { get; private set; } // Asaas payment ID
138
- public string? InvoiceUrl { get; private set; }
139
- public string? BoletoUrl { get; private set; }
140
- public string? PixQrCode { get; private set; }
141
- public string? PixPayload { get; private set; }
142
-
143
- // =========================================================================
144
- // FACTORY
145
- // =========================================================================
146
-
147
- public static SubscriptionPayment Create(
148
- int subscriptionId,
149
- decimal amount,
150
- PaymentMethod method,
151
- DateTime dueDate)
152
- {
153
- return new SubscriptionPayment
154
- {
155
- SubscriptionId = subscriptionId,
156
- Amount = amount,
157
- Status = PaymentStatus.Pending,
158
- Method = method,
159
- DueDate = dueDate,
160
- CreatedAt = DateTime.UtcNow
161
- };
162
- }
163
-
164
- // =========================================================================
165
- // BEHAVIORS
166
- // =========================================================================
167
-
168
- public void SetExternalId(string externalPaymentId)
169
- {
170
- ExternalPaymentId = externalPaymentId;
171
- UpdatedAt = DateTime.UtcNow;
172
- }
173
-
174
- public void SetPaymentUrls(string? invoiceUrl, string? boletoUrl)
175
- {
176
- InvoiceUrl = invoiceUrl;
177
- BoletoUrl = boletoUrl;
178
- UpdatedAt = DateTime.UtcNow;
179
- }
180
-
181
- public void SetPixData(string qrCode, string payload)
182
- {
183
- PixQrCode = qrCode;
184
- PixPayload = payload;
185
- UpdatedAt = DateTime.UtcNow;
186
- }
187
-
188
- public void MarkAsPaid()
189
- {
190
- Status = PaymentStatus.Paid;
191
- PaidAt = DateTime.UtcNow;
192
- UpdatedAt = DateTime.UtcNow;
193
- }
194
-
195
- public void MarkAsOverdue()
196
- {
197
- Status = PaymentStatus.Overdue;
198
- UpdatedAt = DateTime.UtcNow;
199
- }
200
-
201
- public void MarkAsRefunded()
202
- {
203
- Status = PaymentStatus.Refunded;
204
- UpdatedAt = DateTime.UtcNow;
205
- }
206
- }
207
-
208
- // ==============================================================================
209
- // PLAN ENTITY
210
- // ==============================================================================
211
-
212
- public class Plan : BaseEntity
213
- {
214
- [Required]
215
- [MaxLength(100)]
216
- public string Name { get; private set; } = string.Empty;
217
-
218
- [MaxLength(500)]
219
- public string? Description { get; private set; }
220
-
221
- public decimal MonthlyPrice { get; private set; }
222
- public decimal YearlyPrice { get; private set; }
223
-
224
- public int MaxUsers { get; private set; }
225
- public long MaxStorageBytes { get; private set; }
226
-
227
- public bool IsActive { get; private set; }
228
- public int SortOrder { get; private set; }
229
-
230
- public ICollection<PlanFeature> Features { get; private set; } = new List<PlanFeature>();
231
-
232
- // =========================================================================
233
- // FACTORY
234
- // =========================================================================
235
-
236
- public static Plan Create(
237
- string name,
238
- decimal monthlyPrice,
239
- decimal yearlyPrice,
240
- int maxUsers,
241
- long maxStorageBytes)
242
- {
243
- return new Plan
244
- {
245
- Name = name,
246
- MonthlyPrice = monthlyPrice,
247
- YearlyPrice = yearlyPrice,
248
- MaxUsers = maxUsers,
249
- MaxStorageBytes = maxStorageBytes,
250
- IsActive = true,
251
- CreatedAt = DateTime.UtcNow
252
- };
253
- }
254
-
255
- // =========================================================================
256
- // QUERIES
257
- // =========================================================================
258
-
259
- public decimal GetPrice(BillingCycle cycle) => cycle switch
260
- {
261
- BillingCycle.Monthly => MonthlyPrice,
262
- BillingCycle.Quarterly => MonthlyPrice * 3 * 0.95m, // 5% discount
263
- BillingCycle.Yearly => YearlyPrice,
264
- _ => MonthlyPrice
265
- };
266
-
267
- public decimal YearlySavings => (MonthlyPrice * 12) - YearlyPrice;
268
- public int YearlySavingsPercent => (int)((YearlySavings / (MonthlyPrice * 12)) * 100);
269
- }
270
-
271
- // ==============================================================================
272
- // PLAN FEATURE ENTITY
273
- // ==============================================================================
274
-
275
- public class PlanFeature
276
- {
277
- public int Id { get; private set; }
278
- public int PlanId { get; private set; }
279
- public Plan Plan { get; private set; } = null!;
280
-
281
- [Required]
282
- [MaxLength(100)]
283
- public string Name { get; private set; } = string.Empty;
284
-
285
- [MaxLength(200)]
286
- public string? Description { get; private set; }
287
-
288
- public bool IsIncluded { get; private set; }
289
- public int? Limit { get; private set; } // null = unlimited
290
-
291
- public static PlanFeature Create(string name, bool isIncluded, int? limit = null)
292
- {
293
- return new PlanFeature
294
- {
295
- Name = name,
296
- IsIncluded = isIncluded,
297
- Limit = limit
298
- };
299
- }
300
- }
301
-
302
- // ==============================================================================
303
- // ENUMS
304
- // ==============================================================================
305
-
306
- public enum SubscriptionStatus
307
- {
308
- PendingPayment,
309
- Active,
310
- Suspended,
311
- Cancelled,
312
- Expired
313
- }
314
-
315
- public enum BillingCycle
316
- {
317
- Monthly,
318
- Quarterly,
319
- Yearly
320
- }
321
-
322
- public enum PaymentStatus
323
- {
324
- Pending,
325
- Paid,
326
- Overdue,
327
- Refunded,
328
- Cancelled
329
- }
330
-
331
- public enum PaymentMethod
332
- {
333
- Pix,
334
- Boleto,
335
- CreditCard
336
- }
337
-
338
- // ==============================================================================
339
- // BASE ENTITY
340
- // ==============================================================================
341
-
342
- public abstract class BaseEntity
343
- {
344
- public int Id { get; protected set; }
345
- public DateTime CreatedAt { get; protected set; }
346
- public DateTime? UpdatedAt { get; protected set; }
347
- }
1
+ // ==============================================================================
2
+ // MORPH-SPEC - Subscription Model Template
3
+ // Modelo de assinatura para SaaS
4
+ // ==============================================================================
5
+
6
+ using System.ComponentModel.DataAnnotations;
7
+
8
+ namespace {{Namespace}}.Domain.Entities;
9
+
10
+ // ==============================================================================
11
+ // SUBSCRIPTION ENTITY
12
+ // ==============================================================================
13
+
14
+ public class Subscription : BaseEntity
15
+ {
16
+ public int TenantId { get; private set; }
17
+ public Tenant Tenant { get; private set; } = null!;
18
+
19
+ public int PlanId { get; private set; }
20
+ public Plan Plan { get; private set; } = null!;
21
+
22
+ public SubscriptionStatus Status { get; private set; }
23
+ public BillingCycle Cycle { get; private set; }
24
+
25
+ public DateTime StartDate { get; private set; }
26
+ public DateTime? EndDate { get; private set; }
27
+ public DateTime? CancelledAt { get; private set; }
28
+ public DateTime NextBillingDate { get; private set; }
29
+
30
+ public decimal CurrentPrice { get; private set; }
31
+ public string? ExternalSubscriptionId { get; private set; } // Asaas subscription ID
32
+
33
+ public ICollection<SubscriptionPayment> Payments { get; private set; } = new List<SubscriptionPayment>();
34
+
35
+ // =========================================================================
36
+ // FACTORY
37
+ // =========================================================================
38
+
39
+ public static Subscription Create(int tenantId, int planId, BillingCycle cycle, decimal price)
40
+ {
41
+ return new Subscription
42
+ {
43
+ TenantId = tenantId,
44
+ PlanId = planId,
45
+ Status = SubscriptionStatus.PendingPayment,
46
+ Cycle = cycle,
47
+ StartDate = DateTime.UtcNow,
48
+ NextBillingDate = DateTime.UtcNow,
49
+ CurrentPrice = price,
50
+ CreatedAt = DateTime.UtcNow
51
+ };
52
+ }
53
+
54
+ // =========================================================================
55
+ // BEHAVIORS
56
+ // =========================================================================
57
+
58
+ public void Activate(string externalSubscriptionId)
59
+ {
60
+ Status = SubscriptionStatus.Active;
61
+ ExternalSubscriptionId = externalSubscriptionId;
62
+ UpdatedAt = DateTime.UtcNow;
63
+ }
64
+
65
+ public void Suspend()
66
+ {
67
+ Status = SubscriptionStatus.Suspended;
68
+ UpdatedAt = DateTime.UtcNow;
69
+ }
70
+
71
+ public void Cancel(DateTime? endDate = null)
72
+ {
73
+ Status = SubscriptionStatus.Cancelled;
74
+ CancelledAt = DateTime.UtcNow;
75
+ EndDate = endDate ?? DateTime.UtcNow;
76
+ UpdatedAt = DateTime.UtcNow;
77
+ }
78
+
79
+ public void Renew()
80
+ {
81
+ NextBillingDate = CalculateNextBillingDate();
82
+ UpdatedAt = DateTime.UtcNow;
83
+ }
84
+
85
+ public void ChangePlan(int newPlanId, decimal newPrice)
86
+ {
87
+ PlanId = newPlanId;
88
+ CurrentPrice = newPrice;
89
+ UpdatedAt = DateTime.UtcNow;
90
+ }
91
+
92
+ public void AddPayment(SubscriptionPayment payment)
93
+ {
94
+ Payments.Add(payment);
95
+ }
96
+
97
+ // =========================================================================
98
+ // QUERIES
99
+ // =========================================================================
100
+
101
+ public bool IsActive => Status == SubscriptionStatus.Active;
102
+ public bool IsCancelled => Status == SubscriptionStatus.Cancelled;
103
+ public bool IsSuspended => Status == SubscriptionStatus.Suspended;
104
+ public bool IsExpired => EndDate.HasValue && EndDate.Value < DateTime.UtcNow;
105
+
106
+ public int DaysUntilRenewal => (NextBillingDate - DateTime.UtcNow).Days;
107
+ public bool IsNearRenewal => DaysUntilRenewal <= 3;
108
+
109
+ private DateTime CalculateNextBillingDate()
110
+ {
111
+ return Cycle switch
112
+ {
113
+ BillingCycle.Monthly => NextBillingDate.AddMonths(1),
114
+ BillingCycle.Quarterly => NextBillingDate.AddMonths(3),
115
+ BillingCycle.Yearly => NextBillingDate.AddYears(1),
116
+ _ => NextBillingDate.AddMonths(1)
117
+ };
118
+ }
119
+ }
120
+
121
+ // ==============================================================================
122
+ // SUBSCRIPTION PAYMENT ENTITY
123
+ // ==============================================================================
124
+
125
+ public class SubscriptionPayment : BaseEntity
126
+ {
127
+ public int SubscriptionId { get; private set; }
128
+ public Subscription Subscription { get; private set; } = null!;
129
+
130
+ public decimal Amount { get; private set; }
131
+ public PaymentStatus Status { get; private set; }
132
+ public PaymentMethod Method { get; private set; }
133
+
134
+ public DateTime DueDate { get; private set; }
135
+ public DateTime? PaidAt { get; private set; }
136
+
137
+ public string? ExternalPaymentId { get; private set; } // Asaas payment ID
138
+ public string? InvoiceUrl { get; private set; }
139
+ public string? BoletoUrl { get; private set; }
140
+ public string? PixQrCode { get; private set; }
141
+ public string? PixPayload { get; private set; }
142
+
143
+ // =========================================================================
144
+ // FACTORY
145
+ // =========================================================================
146
+
147
+ public static SubscriptionPayment Create(
148
+ int subscriptionId,
149
+ decimal amount,
150
+ PaymentMethod method,
151
+ DateTime dueDate)
152
+ {
153
+ return new SubscriptionPayment
154
+ {
155
+ SubscriptionId = subscriptionId,
156
+ Amount = amount,
157
+ Status = PaymentStatus.Pending,
158
+ Method = method,
159
+ DueDate = dueDate,
160
+ CreatedAt = DateTime.UtcNow
161
+ };
162
+ }
163
+
164
+ // =========================================================================
165
+ // BEHAVIORS
166
+ // =========================================================================
167
+
168
+ public void SetExternalId(string externalPaymentId)
169
+ {
170
+ ExternalPaymentId = externalPaymentId;
171
+ UpdatedAt = DateTime.UtcNow;
172
+ }
173
+
174
+ public void SetPaymentUrls(string? invoiceUrl, string? boletoUrl)
175
+ {
176
+ InvoiceUrl = invoiceUrl;
177
+ BoletoUrl = boletoUrl;
178
+ UpdatedAt = DateTime.UtcNow;
179
+ }
180
+
181
+ public void SetPixData(string qrCode, string payload)
182
+ {
183
+ PixQrCode = qrCode;
184
+ PixPayload = payload;
185
+ UpdatedAt = DateTime.UtcNow;
186
+ }
187
+
188
+ public void MarkAsPaid()
189
+ {
190
+ Status = PaymentStatus.Paid;
191
+ PaidAt = DateTime.UtcNow;
192
+ UpdatedAt = DateTime.UtcNow;
193
+ }
194
+
195
+ public void MarkAsOverdue()
196
+ {
197
+ Status = PaymentStatus.Overdue;
198
+ UpdatedAt = DateTime.UtcNow;
199
+ }
200
+
201
+ public void MarkAsRefunded()
202
+ {
203
+ Status = PaymentStatus.Refunded;
204
+ UpdatedAt = DateTime.UtcNow;
205
+ }
206
+ }
207
+
208
+ // ==============================================================================
209
+ // PLAN ENTITY
210
+ // ==============================================================================
211
+
212
+ public class Plan : BaseEntity
213
+ {
214
+ [Required]
215
+ [MaxLength(100)]
216
+ public string Name { get; private set; } = string.Empty;
217
+
218
+ [MaxLength(500)]
219
+ public string? Description { get; private set; }
220
+
221
+ public decimal MonthlyPrice { get; private set; }
222
+ public decimal YearlyPrice { get; private set; }
223
+
224
+ public int MaxUsers { get; private set; }
225
+ public long MaxStorageBytes { get; private set; }
226
+
227
+ public bool IsActive { get; private set; }
228
+ public int SortOrder { get; private set; }
229
+
230
+ public ICollection<PlanFeature> Features { get; private set; } = new List<PlanFeature>();
231
+
232
+ // =========================================================================
233
+ // FACTORY
234
+ // =========================================================================
235
+
236
+ public static Plan Create(
237
+ string name,
238
+ decimal monthlyPrice,
239
+ decimal yearlyPrice,
240
+ int maxUsers,
241
+ long maxStorageBytes)
242
+ {
243
+ return new Plan
244
+ {
245
+ Name = name,
246
+ MonthlyPrice = monthlyPrice,
247
+ YearlyPrice = yearlyPrice,
248
+ MaxUsers = maxUsers,
249
+ MaxStorageBytes = maxStorageBytes,
250
+ IsActive = true,
251
+ CreatedAt = DateTime.UtcNow
252
+ };
253
+ }
254
+
255
+ // =========================================================================
256
+ // QUERIES
257
+ // =========================================================================
258
+
259
+ public decimal GetPrice(BillingCycle cycle) => cycle switch
260
+ {
261
+ BillingCycle.Monthly => MonthlyPrice,
262
+ BillingCycle.Quarterly => MonthlyPrice * 3 * 0.95m, // 5% discount
263
+ BillingCycle.Yearly => YearlyPrice,
264
+ _ => MonthlyPrice
265
+ };
266
+
267
+ public decimal YearlySavings => (MonthlyPrice * 12) - YearlyPrice;
268
+ public int YearlySavingsPercent => (int)((YearlySavings / (MonthlyPrice * 12)) * 100);
269
+ }
270
+
271
+ // ==============================================================================
272
+ // PLAN FEATURE ENTITY
273
+ // ==============================================================================
274
+
275
+ public class PlanFeature
276
+ {
277
+ public int Id { get; private set; }
278
+ public int PlanId { get; private set; }
279
+ public Plan Plan { get; private set; } = null!;
280
+
281
+ [Required]
282
+ [MaxLength(100)]
283
+ public string Name { get; private set; } = string.Empty;
284
+
285
+ [MaxLength(200)]
286
+ public string? Description { get; private set; }
287
+
288
+ public bool IsIncluded { get; private set; }
289
+ public int? Limit { get; private set; } // null = unlimited
290
+
291
+ public static PlanFeature Create(string name, bool isIncluded, int? limit = null)
292
+ {
293
+ return new PlanFeature
294
+ {
295
+ Name = name,
296
+ IsIncluded = isIncluded,
297
+ Limit = limit
298
+ };
299
+ }
300
+ }
301
+
302
+ // ==============================================================================
303
+ // ENUMS
304
+ // ==============================================================================
305
+
306
+ public enum SubscriptionStatus
307
+ {
308
+ PendingPayment,
309
+ Active,
310
+ Suspended,
311
+ Cancelled,
312
+ Expired
313
+ }
314
+
315
+ public enum BillingCycle
316
+ {
317
+ Monthly,
318
+ Quarterly,
319
+ Yearly
320
+ }
321
+
322
+ public enum PaymentStatus
323
+ {
324
+ Pending,
325
+ Paid,
326
+ Overdue,
327
+ Refunded,
328
+ Cancelled
329
+ }
330
+
331
+ public enum PaymentMethod
332
+ {
333
+ Pix,
334
+ Boleto,
335
+ CreditCard
336
+ }
337
+
338
+ // ==============================================================================
339
+ // BASE ENTITY
340
+ // ==============================================================================
341
+
342
+ public abstract class BaseEntity
343
+ {
344
+ public int Id { get; protected set; }
345
+ public DateTime CreatedAt { get; protected set; }
346
+ public DateTime? UpdatedAt { get; protected set; }
347
+ }