@polymorphism-tech/morph-spec 2.2.0 → 2.4.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.
- package/CLAUDE.md +314 -1673
- package/LICENSE +72 -72
- package/README.md +515 -516
- package/bin/detect-agents.js +225 -225
- package/bin/morph-spec.js +358 -173
- package/bin/render-template.js +302 -302
- package/bin/semantic-detect-agents.js +246 -246
- package/bin/task-manager.js +429 -0
- package/bin/validate-agents-skills.js +251 -251
- package/bin/validate-agents.js +69 -69
- package/bin/validate-phase.js +263 -263
- package/bin/validate.js +369 -0
- package/content/.azure/README.md +293 -293
- package/content/.azure/docs/azure-devops-setup.md +454 -454
- package/content/.azure/docs/branch-strategy.md +398 -398
- package/content/.azure/docs/local-development.md +515 -515
- package/content/.azure/pipelines/pipeline-variables.yml +34 -34
- package/content/.azure/pipelines/prod-pipeline.yml +319 -319
- package/content/.azure/pipelines/staging-pipeline.yml +234 -234
- package/content/.azure/pipelines/templates/build-dotnet.yml +75 -75
- package/content/.azure/pipelines/templates/deploy-app-service.yml +94 -94
- package/content/.azure/pipelines/templates/deploy-container-app.yml +120 -120
- package/content/.azure/pipelines/templates/infra-deploy.yml +90 -90
- package/content/.claude/commands/morph-apply.md +221 -158
- package/content/.claude/commands/morph-archive.md +79 -79
- package/content/.claude/commands/morph-infra.md +209 -209
- package/content/.claude/commands/morph-preflight.md +227 -0
- package/content/.claude/commands/morph-proposal.md +122 -101
- package/content/.claude/commands/morph-status.md +86 -86
- package/content/.claude/commands/morph-troubleshoot.md +122 -0
- package/content/.claude/settings.local.json +15 -15
- package/content/.claude/skills/checklists/code-review.md +226 -0
- package/content/.claude/skills/checklists/morph-checklist.md +117 -0
- package/content/.claude/skills/checklists/simulation-checklist.md +77 -0
- package/content/.claude/skills/infra/bicep-architect.md +126 -419
- package/content/.claude/skills/infra/container-specialist.md +131 -437
- package/content/.claude/skills/infra/devops-engineer.md +119 -405
- package/content/.claude/skills/integrations/asaas-financial.md +130 -333
- package/content/.claude/skills/integrations/azure-identity.md +142 -309
- package/content/.claude/skills/integrations/clerk-auth.md +108 -290
- package/content/.claude/skills/integrations/resend-email.md +119 -0
- package/content/.claude/skills/specialists/ai-system-architect.md +192 -604
- package/content/.claude/skills/specialists/azure-architect.md +142 -142
- package/content/.claude/skills/specialists/code-analyzer.md +235 -0
- package/content/.claude/skills/specialists/dotnet-senior.md +287 -0
- package/content/.claude/skills/specialists/ef-modeler.md +113 -200
- package/content/.claude/skills/specialists/hangfire-orchestrator.md +126 -245
- package/content/.claude/skills/specialists/ms-agent-expert.md +109 -263
- package/content/.claude/skills/specialists/po-pm-advisor.md +197 -197
- package/content/.claude/skills/specialists/standards-architect.md +156 -78
- package/content/.claude/skills/specialists/testing-specialist.md +126 -0
- package/content/.claude/skills/specialists/ui-ux-designer.md +191 -1060
- package/content/.claude/skills/stacks/dotnet-blazor.md +210 -588
- package/content/.claude/skills/stacks/dotnet-nextjs.md +154 -402
- package/content/.claude/skills/workflows/morph-replicate.md +213 -0
- package/content/.claude/{commands/morph-clarify.md → skills/workflows/phase-clarify.md} +5 -58
- package/content/.claude/{commands/morph-design.md → skills/workflows/phase-design.md} +16 -86
- package/content/.claude/{commands/morph-setup.md → skills/workflows/phase-setup.md} +9 -17
- package/content/.claude/skills/workflows/phase-tasks.md +164 -0
- package/content/.claude/{commands/morph-uiux.md → skills/workflows/phase-uiux.md} +15 -88
- package/content/.morph/.morphversion +5 -5
- package/content/.morph/archive/.gitkeep +25 -25
- package/content/.morph/config/agents.json +378 -242
- package/content/.morph/config/config.template.json +89 -108
- package/content/.morph/docs/STORY-DRIVEN-DEVELOPMENT.md +392 -392
- package/content/.morph/docs/workflows/design-impl.md +37 -0
- package/content/.morph/docs/workflows/fast-track.md +29 -0
- package/content/.morph/docs/workflows/full-morph.md +76 -0
- package/content/.morph/docs/workflows/standard.md +44 -0
- package/content/.morph/docs/workflows/ui-refresh.md +39 -0
- package/content/.morph/examples/api-nextjs/README.md +241 -241
- package/content/.morph/examples/api-nextjs/contracts.ts +307 -307
- package/content/.morph/examples/api-nextjs/spec.md +399 -399
- package/content/.morph/examples/api-nextjs/tasks.md +168 -168
- package/content/.morph/examples/micro-saas/README.md +125 -125
- package/content/.morph/examples/micro-saas/contracts.cs +358 -358
- package/content/.morph/examples/micro-saas/decisions.md +246 -246
- package/content/.morph/examples/micro-saas/spec.md +236 -236
- package/content/.morph/examples/micro-saas/tasks.md +150 -150
- package/content/.morph/examples/multi-agent/README.md +309 -309
- package/content/.morph/examples/multi-agent/contracts.cs +433 -433
- package/content/.morph/examples/multi-agent/spec.md +479 -479
- package/content/.morph/examples/multi-agent/tasks.md +185 -185
- package/content/.morph/examples/scheduled-reports/decisions.md +158 -0
- package/content/.morph/examples/scheduled-reports/proposal.md +95 -0
- package/content/.morph/examples/scheduled-reports/spec.md +267 -0
- package/content/.morph/examples/state-v3.json +188 -0
- package/content/.morph/features/.gitkeep +25 -25
- package/content/.morph/hooks/README.md +190 -239
- package/content/.morph/hooks/pre-commit-agents.sh +24 -24
- package/content/.morph/hooks/pre-commit-all.sh +48 -48
- package/content/.morph/hooks/pre-commit-specs.sh +49 -49
- package/content/.morph/hooks/pre-commit-tests.sh +60 -60
- package/content/.morph/project.md +160 -160
- package/content/.morph/schemas/agent.schema.json +296 -296
- package/content/.morph/schemas/tasks.schema.json +220 -0
- package/content/.morph/specs/.gitkeep +20 -20
- package/content/.morph/standards/agent-framework-blazor-ui.md +359 -0
- package/content/.morph/standards/agent-framework-production.md +410 -0
- package/content/.morph/standards/agent-framework-setup.md +413 -453
- package/content/.morph/standards/agent-framework-workflows.md +349 -0
- package/content/.morph/standards/architecture.md +325 -325
- package/content/.morph/standards/azure.md +605 -379
- package/content/.morph/standards/coding.md +377 -377
- package/content/.morph/standards/dotnet10-migration.md +520 -494
- package/content/.morph/standards/fluent-ui-setup.md +590 -590
- package/content/.morph/standards/migration-guide.md +514 -514
- package/content/.morph/standards/passkeys-auth.md +423 -423
- package/content/.morph/standards/vector-search-rag.md +536 -536
- package/content/.morph/state.json +17 -17
- package/content/.morph/templates/FluentDesignTheme.cs +149 -149
- package/content/.morph/templates/MudTheme.cs +281 -281
- package/content/.morph/templates/agent.cs +163 -172
- package/content/.morph/templates/clarify-questions.md +159 -0
- package/content/.morph/templates/component.razor +239 -239
- package/content/.morph/templates/contracts/Commands.cs +74 -0
- package/content/.morph/templates/contracts/Entities.cs +25 -0
- package/content/.morph/templates/contracts/Queries.cs +74 -0
- package/content/.morph/templates/contracts/README.md +74 -0
- package/content/.morph/templates/contracts.cs +217 -217
- package/content/.morph/templates/decisions.md +123 -106
- package/content/.morph/templates/design-system.css +226 -226
- package/content/.morph/templates/infra/.dockerignore.example +89 -89
- package/content/.morph/templates/infra/Dockerfile.example +82 -82
- package/content/.morph/templates/infra/README.md +286 -286
- package/content/.morph/templates/infra/app-insights.bicep +63 -63
- package/content/.morph/templates/infra/app-service.bicep +164 -164
- package/content/.morph/templates/infra/container-app-env.bicep +49 -49
- package/content/.morph/templates/infra/container-app.bicep +156 -156
- package/content/.morph/templates/infra/deploy-checklist.md +426 -0
- package/content/.morph/templates/infra/deploy.ps1 +229 -229
- package/content/.morph/templates/infra/deploy.sh +208 -208
- package/content/.morph/templates/infra/key-vault.bicep +91 -91
- package/content/.morph/templates/infra/main.bicep +189 -189
- package/content/.morph/templates/infra/parameters.dev.json +29 -29
- package/content/.morph/templates/infra/parameters.prod.json +29 -29
- package/content/.morph/templates/infra/parameters.staging.json +29 -29
- package/content/.morph/templates/infra/sql-database.bicep +103 -103
- package/content/.morph/templates/infra/storage.bicep +106 -106
- package/content/.morph/templates/integrations/asaas-client.cs +387 -387
- package/content/.morph/templates/integrations/asaas-webhook.cs +351 -351
- package/content/.morph/templates/integrations/azure-identity-config.cs +288 -288
- package/content/.morph/templates/integrations/clerk-config.cs +258 -258
- package/content/.morph/templates/job.cs +171 -171
- package/content/.morph/templates/migration.cs +83 -83
- package/content/.morph/templates/proposal.md +141 -155
- package/content/.morph/templates/recap.md +94 -105
- package/content/.morph/templates/repository.cs +141 -141
- package/content/.morph/templates/saas/subscription.cs +347 -347
- package/content/.morph/templates/saas/tenant.cs +338 -338
- package/content/.morph/templates/service.cs +139 -139
- package/content/.morph/templates/simulation.md +353 -0
- package/content/.morph/templates/spec.md +149 -148
- package/content/.morph/templates/sprint-status.yaml +68 -68
- package/content/.morph/templates/state.template.json +222 -222
- package/content/.morph/templates/story.md +143 -143
- package/content/.morph/templates/tasks.md +257 -235
- package/content/.morph/templates/test.cs +239 -239
- package/content/.morph/templates/ui-components.md +362 -276
- package/content/.morph/templates/ui-design-system.md +286 -286
- package/content/.morph/templates/ui-flows.md +336 -336
- package/content/.morph/templates/ui-mockups.md +133 -133
- package/content/.morph/test-infra/example.bicep +59 -59
- package/content/CLAUDE.md +150 -442
- package/content/README.md +79 -79
- package/detectors/config-detector.js +223 -223
- package/detectors/conversation-analyzer.js +163 -163
- package/detectors/index.js +84 -84
- package/detectors/standards-generator.js +275 -275
- package/detectors/structure-detector.js +245 -250
- package/docs/README.md +144 -149
- package/docs/api/fonts/Source-Sans-Pro/sourcesanspro-light-webfont.svg +977 -977
- package/docs/api/fonts/Source-Sans-Pro/sourcesanspro-regular-webfont.svg +1048 -1048
- package/docs/api/scripts/collapse.js +38 -38
- package/docs/api/scripts/commonNav.js +28 -28
- package/docs/api/scripts/linenumber.js +25 -25
- package/docs/api/scripts/nav.js +12 -12
- package/docs/api/scripts/polyfill.js +3 -3
- package/docs/api/scripts/prettify/Apache-License-2.0.txt +202 -202
- package/docs/api/scripts/prettify/lang-css.js +2 -2
- package/docs/api/scripts/prettify/prettify.js +28 -28
- package/docs/api/scripts/search.js +98 -98
- package/docs/api/styles/jsdoc.css +776 -776
- package/docs/api/styles/prettify.css +80 -80
- package/docs/examples.md +328 -328
- package/docs/getting-started.md +301 -302
- package/docs/installation.md +361 -361
- package/docs/templates.md +418 -418
- package/docs/validation-checklist.md +265 -266
- package/package.json +80 -80
- package/scripts/postinstall.js +132 -132
- package/src/commands/advance-phase.js +183 -0
- package/src/commands/analyze-blazor-concurrency.js +193 -0
- package/src/commands/create-story.js +351 -351
- package/src/commands/detect-agents.js +139 -0
- package/src/commands/detect.js +104 -104
- package/src/commands/doctor.js +356 -280
- package/src/commands/generate.js +149 -149
- package/src/commands/init.js +258 -245
- package/src/commands/lint-fluent.js +352 -0
- package/src/commands/rollback-phase.js +185 -0
- package/src/commands/session-summary.js +291 -0
- package/src/commands/shard-spec.js +224 -224
- package/src/commands/sprint-status.js +250 -250
- package/src/commands/state.js +333 -333
- package/src/commands/sync.js +167 -167
- package/src/commands/task.js +78 -0
- package/src/commands/troubleshoot.js +222 -0
- package/src/commands/update.js +192 -159
- package/src/commands/validate-blazor-state.js +210 -0
- package/src/commands/validate-blazor.js +156 -0
- package/src/commands/validate-css.js +84 -0
- package/src/commands/validate-phase.js +221 -0
- package/src/lib/blazor-concurrency-analyzer.js +288 -0
- package/src/lib/blazor-state-validator.js +291 -0
- package/src/lib/blazor-validator.js +374 -0
- package/src/lib/complexity-analyzer.js +441 -292
- package/src/lib/continuous-validator.js +421 -0
- package/src/lib/css-validator.js +352 -0
- package/src/lib/decision-constraint-loader.js +109 -0
- package/src/lib/design-system-generator.js +298 -298
- package/src/lib/learning-system.js +520 -0
- package/src/lib/mockup-generator.js +366 -0
- package/src/lib/recap-generator.js +205 -0
- package/src/lib/state-manager.js +397 -340
- package/src/lib/troubleshoot-grep.js +194 -0
- package/src/lib/troubleshoot-index.js +144 -0
- package/src/lib/ui-detector.js +350 -0
- package/src/lib/validation-runner.js +231 -0
- package/src/lib/validators/architecture-validator.js +387 -0
- package/src/lib/validators/contract-compliance-validator.js +273 -0
- package/src/lib/validators/package-validator.js +360 -0
- package/src/lib/validators/ui-contrast-validator.js +422 -0
- package/src/utils/file-copier.js +179 -139
- package/src/utils/logger.js +32 -32
- package/src/utils/version-checker.js +175 -175
- package/content/.claude/commands/morph-costs.md +0 -206
- package/content/.claude/commands/morph-tasks.md +0 -319
- package/content/.claude/skills/specialists/cost-guardian.md +0 -110
- package/content/.claude/skills/stacks/shopify.md +0 -445
- package/content/.morph/config/azure-pricing.json +0 -70
- package/content/.morph/config/azure-pricing.schema.json +0 -50
- package/content/.morph/hooks/pre-commit-costs.sh +0 -91
- package/docs/api/cost-calculator.js.html +0 -513
- package/docs/api/design-system-generator.js.html +0 -382
- package/docs/api/global.html +0 -5263
- package/docs/api/index.html +0 -96
- package/docs/api/state-manager.js.html +0 -423
- package/src/commands/cost.js +0 -181
- package/src/commands/update-pricing.js +0 -206
- package/src/lib/cost-calculator.js +0 -429
|
@@ -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
|
+
}
|