@polymorphism-tech/morph-spec 4.7.0 → 4.7.2
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/.morph/.morphversion +5 -0
- package/.morph/analytics/threads-log.jsonl +5 -0
- package/.morph/config/config.json +8 -0
- package/.morph/framework/agents.json +1815 -0
- package/.morph/framework/hooks/README.md +205 -0
- package/.morph/framework/hooks/claude-code/notification/approval-reminder.js +54 -0
- package/.morph/framework/hooks/claude-code/post-tool-use/dispatch.js +83 -0
- package/.morph/framework/hooks/claude-code/post-tool-use/handle-tool-failure.js +42 -0
- package/.morph/framework/hooks/claude-code/pre-compact/save-morph-context.js +61 -0
- package/.morph/framework/hooks/claude-code/pre-tool-use/enforce-phase-writes.js +71 -0
- package/.morph/framework/hooks/claude-code/pre-tool-use/protect-readonly-files.js +58 -0
- package/.morph/framework/hooks/claude-code/pre-tool-use/protect-spec-files.js +64 -0
- package/.morph/framework/hooks/claude-code/session-start/inject-morph-context.js +94 -0
- package/.morph/framework/hooks/claude-code/statusline.py +538 -0
- package/.morph/framework/hooks/claude-code/statusline.sh +7 -0
- package/.morph/framework/hooks/claude-code/stop/validate-completion.js +88 -0
- package/.morph/framework/hooks/claude-code/user-prompt/enrich-prompt.js +91 -0
- package/.morph/framework/hooks/git/commit-msg/conventional-commits.sh +33 -0
- package/.morph/framework/hooks/git/pre-commit/agents.sh +25 -0
- package/.morph/framework/hooks/git/pre-commit/orchestrator.sh +64 -0
- package/.morph/framework/hooks/git/pre-commit/specs.sh +50 -0
- package/.morph/framework/hooks/git/pre-push/run-tests.sh +44 -0
- package/.morph/framework/hooks/shared/hook-response.js +45 -0
- package/.morph/framework/hooks/shared/phase-utils.js +129 -0
- package/.morph/framework/hooks/shared/state-reader.js +138 -0
- package/.morph/framework/hooks/shared/stdin-reader.js +26 -0
- package/.morph/framework/standards/STANDARDS.json +933 -0
- package/.morph/framework/standards/ai-agents/blazor-ui.md +364 -0
- package/.morph/framework/standards/ai-agents/production.md +415 -0
- package/.morph/framework/standards/ai-agents/setup.md +418 -0
- package/.morph/framework/standards/ai-agents/team-orchestration.md +479 -0
- package/.morph/framework/standards/ai-agents/workflows.md +354 -0
- package/.morph/framework/standards/architecture/ddd/aggregates.md +120 -0
- package/.morph/framework/standards/architecture/ddd/bounded-contexts.md +105 -0
- package/.morph/framework/standards/architecture/ddd/complexity-levels.md +108 -0
- package/.morph/framework/standards/architecture/ddd/entities.md +99 -0
- package/.morph/framework/standards/architecture/ddd/ubiquitous-language.md +58 -0
- package/.morph/framework/standards/architecture/ddd/value-objects.md +124 -0
- package/.morph/framework/standards/backend/api/minimal-api.md +494 -0
- package/.morph/framework/standards/backend/api/rest.md +492 -0
- package/.morph/framework/standards/backend/api/validation.md +88 -0
- package/.morph/framework/standards/backend/authentication/passkeys.md +428 -0
- package/.morph/framework/standards/backend/database/ef-core.md +199 -0
- package/.morph/framework/standards/backend/database/migrations.md +393 -0
- package/.morph/framework/standards/backend/database/postgresql/database.md +352 -0
- package/.morph/framework/standards/backend/database/repository-patterns.md +528 -0
- package/.morph/framework/standards/backend/database/vector-search-rag.md +541 -0
- package/.morph/framework/standards/backend/dotnet/async.md +366 -0
- package/.morph/framework/standards/backend/dotnet/core.md +117 -0
- package/.morph/framework/standards/backend/dotnet/di.md +439 -0
- package/.morph/framework/standards/backend/dotnet/program-cs-checklist.md +92 -0
- package/.morph/framework/standards/backend/integrations/asaas/asaas-api.md +216 -0
- package/.morph/framework/standards/backend/integrations/clerk/clerk-auth.md +290 -0
- package/.morph/framework/standards/backend/integrations/hangfire/hangfire-jobs.md +350 -0
- package/.morph/framework/standards/backend/integrations/resend/resend-email.md +385 -0
- package/.morph/framework/standards/context/analytics.md +96 -0
- package/.morph/framework/standards/context/bundles.md +110 -0
- package/.morph/framework/standards/context/priming.md +78 -0
- package/.morph/framework/standards/core/architecture.md +185 -0
- package/.morph/framework/standards/core/coding.md +214 -0
- package/.morph/framework/standards/core/git-branching-strategy.md +403 -0
- package/.morph/framework/standards/core/git.md +185 -0
- package/.morph/framework/standards/core/testing.md +295 -0
- package/.morph/framework/standards/data/nosql/blob-storage.md +102 -0
- package/.morph/framework/standards/data/nosql/cache/redis.md +97 -0
- package/.morph/framework/standards/data/nosql/cosmos-db.md +118 -0
- package/.morph/framework/standards/data/vector-search/azure-ai-search.md +121 -0
- package/.morph/framework/standards/data/vector-search/rag-chunking.md +104 -0
- package/.morph/framework/standards/frontend/blazor/design-checklist.md +222 -0
- package/.morph/framework/standards/frontend/blazor/fluent-ui-setup.md +595 -0
- package/.morph/framework/standards/frontend/blazor/fluent-ui.md +137 -0
- package/.morph/framework/standards/frontend/blazor/html-conversion.md +184 -0
- package/.morph/framework/standards/frontend/blazor/lifecycle.md +195 -0
- package/.morph/framework/standards/frontend/blazor/pitfalls.md +198 -0
- package/.morph/framework/standards/frontend/blazor/state.md +191 -0
- package/.morph/framework/standards/frontend/design-system/animations.md +151 -0
- package/.morph/framework/standards/frontend/design-system/naming.md +64 -0
- package/.morph/framework/standards/frontend/nextjs/app-router.md +123 -0
- package/.morph/framework/standards/frontend/nextjs/components.md +132 -0
- package/.morph/framework/standards/frontend/nextjs/data-fetching.md +126 -0
- package/.morph/framework/standards/frontend/nextjs/forms.md +128 -0
- package/.morph/framework/standards/frontend/nextjs/naming-conventions.md +67 -0
- package/.morph/framework/standards/frontend/nextjs/nextjs-patterns.md +215 -0
- package/.morph/framework/standards/frontend/nextjs/project-structure.md +102 -0
- package/.morph/framework/standards/frontend/nextjs/state-management.md +72 -0
- package/.morph/framework/standards/frontend/nextjs/testing.md +111 -0
- package/.morph/framework/standards/infrastructure/azure/azure.md +624 -0
- package/.morph/framework/standards/infrastructure/azure/bicep/bicep-patterns.md +422 -0
- package/.morph/framework/standards/infrastructure/azure/devops/azure-devops-setup.md +516 -0
- package/.morph/framework/standards/infrastructure/azure/devops/local-development.md +520 -0
- package/.morph/framework/standards/infrastructure/azure/services/functions.md +486 -0
- package/.morph/framework/standards/infrastructure/azure/services/service-bus.md +459 -0
- package/.morph/framework/standards/infrastructure/azure/services/storage.md +407 -0
- package/.morph/framework/standards/infrastructure/docker/easypanel-deploy.md +196 -0
- package/.morph/framework/standards/infrastructure/supabase/mcp-setup.md +252 -0
- package/.morph/framework/standards/infrastructure/supabase/supabase-auth.md +176 -0
- package/.morph/framework/standards/infrastructure/supabase/supabase-pgvector.md +169 -0
- package/.morph/framework/standards/infrastructure/supabase/supabase-rls.md +184 -0
- package/.morph/framework/standards/infrastructure/supabase/supabase-storage.md +153 -0
- package/.morph/framework/standards/integration/api/graphql.md +91 -0
- package/.morph/framework/standards/integration/api/grpc.md +114 -0
- package/.morph/framework/standards/integration/api/rest-design.md +95 -0
- package/.morph/framework/standards/integration/event-driven/cqrs.md +101 -0
- package/.morph/framework/standards/integration/event-driven/event-sourcing.md +124 -0
- package/.morph/framework/standards/integration/event-driven/service-bus.md +95 -0
- package/.morph/framework/standards/integration/mcp/mcp-tools.md +384 -0
- package/.morph/framework/standards/observability/logging.md +131 -0
- package/.morph/framework/standards/observability/metrics.md +121 -0
- package/.morph/framework/standards/observability/monitoring.md +114 -0
- package/.morph/framework/standards/observability/tracing.md +132 -0
- package/.morph/framework/standards/workflows/parallel-execution.md +112 -0
- package/.morph/framework/standards/workflows/thread-management.md +113 -0
- package/.morph/framework/templates/.idea/morph-templates.xml +92 -0
- package/.morph/framework/templates/.vscode/morph-templates.code-snippets +186 -0
- package/.morph/framework/templates/IDE-SNIPPETS.md +266 -0
- package/.morph/framework/templates/README.md +814 -0
- package/.morph/framework/templates/REGISTRY.json +1888 -0
- package/.morph/framework/templates/code/dotnet/backend/repository.cs +141 -0
- package/.morph/framework/templates/code/dotnet/backend/service.cs +139 -0
- package/.morph/framework/templates/code/dotnet/contracts/Commands.cs +74 -0
- package/.morph/framework/templates/code/dotnet/contracts/Entities.cs +25 -0
- package/.morph/framework/templates/code/dotnet/contracts/Queries.cs +74 -0
- package/.morph/framework/templates/code/dotnet/contracts/README.md +74 -0
- package/.morph/framework/templates/code/dotnet/contracts/api-contracts.cs +173 -0
- package/.morph/framework/templates/code/dotnet/contracts/contracts-level1.cs +69 -0
- package/.morph/framework/templates/code/dotnet/contracts/contracts-level2.cs +86 -0
- package/.morph/framework/templates/code/dotnet/contracts/contracts-level3.cs +41 -0
- package/.morph/framework/templates/code/dotnet/database/migration.cs +83 -0
- package/.morph/framework/templates/code/dotnet/frontend/component.razor +239 -0
- package/.morph/framework/templates/code/dotnet/jobs/agent.cs +163 -0
- package/.morph/framework/templates/code/dotnet/jobs/job.cs +171 -0
- package/.morph/framework/templates/code/dotnet/test.cs +239 -0
- package/.morph/framework/templates/code/sql/rls-policy.sql +57 -0
- package/.morph/framework/templates/code/sql/supabase-migration.sql +100 -0
- package/.morph/framework/templates/code/sql/supabase-migration.template.sql +113 -0
- package/.morph/framework/templates/code/typescript/contracts.ts +168 -0
- package/.morph/framework/templates/context/CONTEXT-FEATURE.md +276 -0
- package/.morph/framework/templates/context/CONTEXT.md +181 -0
- package/.morph/framework/templates/docs/clarifications.md +253 -0
- package/.morph/framework/templates/docs/onboarding.md +123 -0
- package/.morph/framework/templates/docs/proposal.md +182 -0
- package/.morph/framework/templates/docs/schema-analysis.md +119 -0
- package/.morph/framework/templates/docs/spec.md +198 -0
- package/.morph/framework/templates/docs/ui-components.md +124 -0
- package/.morph/framework/templates/docs/ui-design-system.md +76 -0
- package/.morph/framework/templates/docs/ui-flows.md +167 -0
- package/.morph/framework/templates/docs/ui-mockups.md +98 -0
- package/.morph/framework/templates/docs/user-stories.md +34 -0
- package/.morph/framework/templates/examples/design-system-examples.md +357 -0
- package/.morph/framework/templates/examples/spec-examples.md +90 -0
- package/.morph/framework/templates/feature/decisions.md +187 -0
- package/.morph/framework/templates/feature/recap.md +146 -0
- package/.morph/framework/templates/feature/tasks.md +199 -0
- package/.morph/framework/templates/frontend/nextjs/Dockerfile.nextjs.hbs +43 -0
- package/.morph/framework/templates/frontend/nextjs/client-component.tsx.hbs +26 -0
- package/.morph/framework/templates/frontend/nextjs/env.mjs.hbs +32 -0
- package/.morph/framework/templates/frontend/nextjs/feature-form.tsx.hbs +56 -0
- package/.morph/framework/templates/frontend/nextjs/page.tsx.hbs +22 -0
- package/.morph/framework/templates/frontend/nextjs/tsconfig.json.hbs +26 -0
- package/.morph/framework/templates/frontend/nextjs/use-feature.ts.hbs +54 -0
- package/.morph/framework/templates/infrastructure/azure/Dockerfile.example +82 -0
- package/.morph/framework/templates/infrastructure/azure/README.md +286 -0
- package/.morph/framework/templates/infrastructure/azure/app-insights.bicep +63 -0
- package/.morph/framework/templates/infrastructure/azure/app-service.bicep +164 -0
- package/.morph/framework/templates/infrastructure/azure/container-app-env.bicep +49 -0
- package/.morph/framework/templates/infrastructure/azure/container-app.bicep +156 -0
- package/.morph/framework/templates/infrastructure/azure/deploy-checklist.md +426 -0
- package/.morph/framework/templates/infrastructure/azure/deploy.ps1 +229 -0
- package/.morph/framework/templates/infrastructure/azure/deploy.sh +208 -0
- package/.morph/framework/templates/infrastructure/azure/key-vault.bicep +91 -0
- package/.morph/framework/templates/infrastructure/azure/main.bicep +189 -0
- package/.morph/framework/templates/infrastructure/azure/parameters.dev.json +29 -0
- package/.morph/framework/templates/infrastructure/azure/parameters.prod.json +29 -0
- package/.morph/framework/templates/infrastructure/azure/parameters.staging.json +29 -0
- package/.morph/framework/templates/infrastructure/azure/sql-database.bicep +103 -0
- package/.morph/framework/templates/infrastructure/azure/storage.bicep +106 -0
- package/.morph/framework/templates/infrastructure/docker/Dockerfile.template +58 -0
- package/.morph/framework/templates/infrastructure/docker/docker-compose.template.yml +67 -0
- package/.morph/framework/templates/infrastructure/docker/dockerfile-api.dockerfile +38 -0
- package/.morph/framework/templates/infrastructure/docker/dockerfile-web.dockerfile +48 -0
- package/.morph/framework/templates/infrastructure/docker/easypanel.template.json +54 -0
- package/.morph/framework/templates/infrastructure/github/README.md +593 -0
- package/.morph/framework/templates/infrastructure/github/actions/azure-auth/action.yml.hbs +22 -0
- package/.morph/framework/templates/infrastructure/github/actions/docker-build-push/action.yml.hbs +45 -0
- package/.morph/framework/templates/infrastructure/github/actions/health-check/action.yml.hbs +27 -0
- package/.morph/framework/templates/infrastructure/github/workflows/deploy-azure-app-service.yml.hbs +61 -0
- package/.morph/framework/templates/infrastructure/github/workflows/deploy-easypanel.yml.hbs +31 -0
- package/.morph/framework/templates/infrastructure/github/workflows/docker-build-push.yml.hbs +59 -0
- package/.morph/framework/templates/infrastructure/github/workflows/dotnet-build.yml.hbs +39 -0
- package/.morph/framework/templates/integrations/asaas-client.cs +387 -0
- package/.morph/framework/templates/integrations/asaas-webhook.cs +351 -0
- package/.morph/framework/templates/integrations/azure-identity-config.cs +288 -0
- package/.morph/framework/templates/integrations/clerk-config.cs +258 -0
- package/.morph/framework/templates/meta-prompts/fusion/fusion-agent.md +76 -0
- package/.morph/framework/templates/meta-prompts/fusion/fusion-aggregator.md +100 -0
- package/.morph/framework/templates/meta-prompts/hops/hop-retry.md +78 -0
- package/.morph/framework/templates/meta-prompts/hops/hop-validation.md +97 -0
- package/.morph/framework/templates/meta-prompts/hops/hop-wrapper.md +36 -0
- package/.morph/framework/templates/meta-prompts/parallel-workers/parallel-coordinator.md +113 -0
- package/.morph/framework/templates/meta-prompts/parallel-workers/parallel-worker.md +80 -0
- package/.morph/framework/templates/meta-prompts/squad-leaders/backend-squad.md +90 -0
- package/.morph/framework/templates/meta-prompts/squad-leaders/frontend-squad.md +126 -0
- package/.morph/framework/templates/meta-prompts/squad-leaders/squad-leader.md +43 -0
- package/.morph/framework/templates/meta-prompts/validators/checkpoint-validator.md +107 -0
- package/.morph/framework/templates/meta-prompts/validators/pre-commit-validator.md +95 -0
- package/.morph/framework/templates/project-structure/dotnet-ddd.md +70 -0
- package/.morph/framework/templates/saas/subscription.cs +347 -0
- package/.morph/framework/templates/saas/tenant.cs +338 -0
- package/.morph/framework/templates/state.template.json +17 -0
- package/.morph/framework/templates/ui/FluentDesignTheme.cs +149 -0
- package/.morph/framework/templates/ui/MudTheme.cs +281 -0
- package/.morph/framework/templates/ui/design-system.css +226 -0
- package/.morph/logs/tool-failures.log +17 -0
- package/.morph/memory/pre-compact-2026-02-24T17-43-30-049Z.json +16 -0
- package/.morph/plans/eager-watching-bunny.md +105 -0
- package/.morph/plans/temporal-seeking-nebula.md +45 -0
- package/.morph/state.json +48 -0
- package/CLAUDE.md +1 -1
- package/README.md +119 -99
- package/bin/morph-spec.js +0 -9
- package/framework/CLAUDE.md +1 -1
- package/framework/hooks/README.md +10 -6
- package/framework/hooks/claude-code/notification/approval-reminder.js +2 -0
- package/framework/hooks/claude-code/post-tool-use/dispatch.js +1 -1
- package/framework/hooks/claude-code/stop/validate-completion.js +1 -1
- package/framework/hooks/claude-code/user-prompt/enrich-prompt.js +1 -1
- package/package.json +1 -1
- package/src/commands/project/init.js +15 -42
- package/src/commands/project/update.js +22 -37
- package/src/lib/installers/mcp-installer.js +18 -3
- package/src/utils/hooks-installer.js +5 -15
- package/src/commands/project/detect.js +0 -114
|
@@ -0,0 +1,99 @@
|
|
|
1
|
+
# Architecture Standard: DDD Entities
|
|
2
|
+
|
|
3
|
+
## Overview
|
|
4
|
+
Entities have identity — two entities with the same ID are the same, regardless of their attribute values.
|
|
5
|
+
|
|
6
|
+
## Entity Base Class
|
|
7
|
+
```csharp
|
|
8
|
+
public abstract class Entity
|
|
9
|
+
{
|
|
10
|
+
public Guid Id { get; protected set; } = Guid.NewGuid();
|
|
11
|
+
public DateTime CreatedAt { get; protected set; } = DateTime.UtcNow;
|
|
12
|
+
public DateTime? UpdatedAt { get; protected set; }
|
|
13
|
+
public bool IsDeleted { get; private set; }
|
|
14
|
+
public DateTime? DeletedAt { get; private set; }
|
|
15
|
+
|
|
16
|
+
// Identity equality (not attribute equality)
|
|
17
|
+
public override bool Equals(object? obj)
|
|
18
|
+
=> obj is Entity other && Id == other.Id;
|
|
19
|
+
|
|
20
|
+
public override int GetHashCode() => Id.GetHashCode();
|
|
21
|
+
public static bool operator ==(Entity? a, Entity? b) => a?.Equals(b) ?? b is null;
|
|
22
|
+
public static bool operator !=(Entity? a, Entity? b) => !(a == b);
|
|
23
|
+
|
|
24
|
+
// Soft delete support
|
|
25
|
+
public void SoftDelete()
|
|
26
|
+
{
|
|
27
|
+
IsDeleted = true;
|
|
28
|
+
DeletedAt = DateTime.UtcNow;
|
|
29
|
+
}
|
|
30
|
+
}
|
|
31
|
+
```
|
|
32
|
+
|
|
33
|
+
## Entity Lifecycle
|
|
34
|
+
```csharp
|
|
35
|
+
// Entities are created through factory methods (never bare constructors from outside the aggregate)
|
|
36
|
+
public class OrderItem : Entity
|
|
37
|
+
{
|
|
38
|
+
private OrderItem() { } // Private for EF Core
|
|
39
|
+
|
|
40
|
+
internal static OrderItem Create(Guid productId, string productName, decimal price, int quantity)
|
|
41
|
+
{
|
|
42
|
+
if (price <= 0) throw new DomainException("Price must be positive");
|
|
43
|
+
if (quantity <= 0) throw new DomainException("Quantity must be positive");
|
|
44
|
+
|
|
45
|
+
return new OrderItem
|
|
46
|
+
{
|
|
47
|
+
ProductId = productId,
|
|
48
|
+
ProductName = productName,
|
|
49
|
+
UnitPrice = price,
|
|
50
|
+
Quantity = quantity
|
|
51
|
+
};
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
public Guid ProductId { get; private set; }
|
|
55
|
+
public string ProductName { get; private set; } = string.Empty;
|
|
56
|
+
public decimal UnitPrice { get; private set; }
|
|
57
|
+
public int Quantity { get; private set; }
|
|
58
|
+
public decimal Subtotal => UnitPrice * Quantity;
|
|
59
|
+
|
|
60
|
+
internal void IncreaseQuantity(int additionalQuantity)
|
|
61
|
+
{
|
|
62
|
+
if (additionalQuantity <= 0) throw new DomainException("Quantity must be positive");
|
|
63
|
+
Quantity += additionalQuantity;
|
|
64
|
+
UpdatedAt = DateTime.UtcNow;
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
```
|
|
68
|
+
|
|
69
|
+
## EF Core Mapping
|
|
70
|
+
|
|
71
|
+
### Required Conventions
|
|
72
|
+
```csharp
|
|
73
|
+
builder.HasKey(e => e.Id);
|
|
74
|
+
builder.Property(e => e.CreatedAt).IsRequired();
|
|
75
|
+
|
|
76
|
+
// Soft delete filter (apply globally)
|
|
77
|
+
builder.HasQueryFilter(e => !e.IsDeleted);
|
|
78
|
+
|
|
79
|
+
// All string properties: max length
|
|
80
|
+
builder.Property(e => e.Name)
|
|
81
|
+
.IsRequired()
|
|
82
|
+
.HasMaxLength(200);
|
|
83
|
+
```
|
|
84
|
+
|
|
85
|
+
### Inheritance (TPH by default)
|
|
86
|
+
```csharp
|
|
87
|
+
// Table Per Hierarchy (single table with discriminator)
|
|
88
|
+
builder.HasDiscriminator<string>("Type")
|
|
89
|
+
.HasValue<ProductOrder>("product")
|
|
90
|
+
.HasValue<ServiceOrder>("service");
|
|
91
|
+
```
|
|
92
|
+
|
|
93
|
+
## Entity vs Value Object Decision
|
|
94
|
+
| Use Entity When | Use Value Object When |
|
|
95
|
+
|----------------|----------------------|
|
|
96
|
+
| Has lifecycle (created, updated, deleted) | Immutable by definition |
|
|
97
|
+
| Identity matters (two objects with same data are different) | Equality by value |
|
|
98
|
+
| Needs to be tracked in a repository | Owned by an entity (no separate table) |
|
|
99
|
+
| Examples: User, Order, Product | Examples: Money, Address, DateRange |
|
|
@@ -0,0 +1,58 @@
|
|
|
1
|
+
# DDD Ubiquitous Language
|
|
2
|
+
|
|
3
|
+
> **Scope:** universal
|
|
4
|
+
> **Layer:** 2
|
|
5
|
+
> **Keywords:** ubiquitous-language, ddd, glossary, domain-terms, naming
|
|
6
|
+
|
|
7
|
+
Linguagem Ubíqua é o vocabulário compartilhado entre devs e domínio de negócio.
|
|
8
|
+
Elimina ambiguidade e alinha código com a linguagem do negócio.
|
|
9
|
+
|
|
10
|
+
---
|
|
11
|
+
|
|
12
|
+
## Formato do Glossário
|
|
13
|
+
|
|
14
|
+
```markdown
|
|
15
|
+
# Ubiquitous Language: {Feature/BoundedContext}
|
|
16
|
+
|
|
17
|
+
## Termos do Domínio
|
|
18
|
+
|
|
19
|
+
| Termo | Definição | Contexto | Código |
|
|
20
|
+
|-------|-----------|----------|--------|
|
|
21
|
+
| Subscription | Contrato ativo de uso por um tenant | Billing | `Subscription` (AggregateRoot) |
|
|
22
|
+
| PlanTier | Nível do plano (Free/Pro/Enterprise) | Billing | `PlanTier` (ValueObject) |
|
|
23
|
+
| Activation | Ato de ativar Subscription após pagamento | Billing | `Subscription.Activate()` |
|
|
24
|
+
|
|
25
|
+
## Estados e Transições
|
|
26
|
+
|
|
27
|
+
| Estado | Descrição | Transições |
|
|
28
|
+
|--------|-----------|-----------|
|
|
29
|
+
| Draft | Criada, aguardando pagamento | → Active |
|
|
30
|
+
| Active | Paga e em uso | → Suspended, Cancelled |
|
|
31
|
+
| Cancelled | Encerrada definitivamente | (terminal) |
|
|
32
|
+
|
|
33
|
+
## Invariants em Linguagem Natural
|
|
34
|
+
|
|
35
|
+
- "Uma Subscription só pode ser Cancelada se estiver Active ou Suspended"
|
|
36
|
+
- "Um Tenant não pode ter duas Subscriptions Active simultaneamente"
|
|
37
|
+
```
|
|
38
|
+
|
|
39
|
+
---
|
|
40
|
+
|
|
41
|
+
## Regras de Nomenclatura
|
|
42
|
+
|
|
43
|
+
1. Use termos do glossário como nomes de classe/método
|
|
44
|
+
2. Eventos no passado: `OrderConfirmed`, não `OrderConfirmation`
|
|
45
|
+
3. Commands no imperativo: `CreateOrder`, não `OrderCreation`
|
|
46
|
+
4. Value Objects no substantivo: `Money`, `Email`
|
|
47
|
+
5. Métodos do Aggregate espelham ações do negócio: `order.Confirm()`, não `order.SetStatusToConfirmed()`
|
|
48
|
+
|
|
49
|
+
---
|
|
50
|
+
|
|
51
|
+
## Quando Gerar
|
|
52
|
+
|
|
53
|
+
- **Nível 2:** Seção inline no `spec.md` (Aggregate Blueprint)
|
|
54
|
+
- **Nível 3:** Arquivo separado `1-design/ubiquitous-language.md` por BC
|
|
55
|
+
|
|
56
|
+
---
|
|
57
|
+
|
|
58
|
+
*MORPH-SPEC by Polymorphism Tech*
|
|
@@ -0,0 +1,124 @@
|
|
|
1
|
+
# Architecture Standard: DDD Value Objects
|
|
2
|
+
|
|
3
|
+
## Overview
|
|
4
|
+
Value objects have no identity — equality is based on their attribute values. They are immutable.
|
|
5
|
+
|
|
6
|
+
## Value Object Base Class
|
|
7
|
+
```csharp
|
|
8
|
+
public abstract class ValueObject
|
|
9
|
+
{
|
|
10
|
+
protected abstract IEnumerable<object?> GetEqualityComponents();
|
|
11
|
+
|
|
12
|
+
public override bool Equals(object? obj)
|
|
13
|
+
{
|
|
14
|
+
if (obj is null || obj.GetType() != GetType()) return false;
|
|
15
|
+
return GetEqualityComponents().SequenceEqual(((ValueObject)obj).GetEqualityComponents());
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
public override int GetHashCode()
|
|
19
|
+
=> GetEqualityComponents().Aggregate(0, (hash, c) => HashCode.Combine(hash, c?.GetHashCode() ?? 0));
|
|
20
|
+
|
|
21
|
+
public static bool operator ==(ValueObject? a, ValueObject? b) => a?.Equals(b) ?? b is null;
|
|
22
|
+
public static bool operator !=(ValueObject? a, ValueObject? b) => !(a == b);
|
|
23
|
+
}
|
|
24
|
+
```
|
|
25
|
+
|
|
26
|
+
## Money Value Object
|
|
27
|
+
```csharp
|
|
28
|
+
public sealed class Money : ValueObject
|
|
29
|
+
{
|
|
30
|
+
public decimal Amount { get; }
|
|
31
|
+
public string Currency { get; }
|
|
32
|
+
|
|
33
|
+
private Money(decimal amount, string currency)
|
|
34
|
+
{
|
|
35
|
+
Amount = amount;
|
|
36
|
+
Currency = currency.ToUpper();
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
public static Money Of(decimal amount, string currency = "BRL")
|
|
40
|
+
{
|
|
41
|
+
if (amount < 0) throw new DomainException("Money amount cannot be negative");
|
|
42
|
+
if (string.IsNullOrWhiteSpace(currency)) throw new DomainException("Currency required");
|
|
43
|
+
return new Money(amount, currency);
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
public Money Add(Money other)
|
|
47
|
+
{
|
|
48
|
+
if (Currency != other.Currency)
|
|
49
|
+
throw new DomainException($"Cannot add {Currency} and {other.Currency}");
|
|
50
|
+
return new Money(Amount + other.Amount, Currency);
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
public Money Multiply(decimal factor)
|
|
54
|
+
=> new(Amount * factor, Currency);
|
|
55
|
+
|
|
56
|
+
public override string ToString() => $"{Amount:F2} {Currency}";
|
|
57
|
+
|
|
58
|
+
protected override IEnumerable<object?> GetEqualityComponents()
|
|
59
|
+
{
|
|
60
|
+
yield return Amount;
|
|
61
|
+
yield return Currency;
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
```
|
|
65
|
+
|
|
66
|
+
## Address Value Object
|
|
67
|
+
```csharp
|
|
68
|
+
public sealed class Address : ValueObject
|
|
69
|
+
{
|
|
70
|
+
public string Street { get; }
|
|
71
|
+
public string City { get; }
|
|
72
|
+
public string State { get; }
|
|
73
|
+
public string ZipCode { get; }
|
|
74
|
+
public string Country { get; }
|
|
75
|
+
|
|
76
|
+
public Address(string street, string city, string state, string zipCode, string country)
|
|
77
|
+
{
|
|
78
|
+
Street = street ?? throw new DomainException("Street required");
|
|
79
|
+
City = city ?? throw new DomainException("City required");
|
|
80
|
+
State = state ?? throw new DomainException("State required");
|
|
81
|
+
ZipCode = zipCode ?? throw new DomainException("ZipCode required");
|
|
82
|
+
Country = country ?? throw new DomainException("Country required");
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
protected override IEnumerable<object?> GetEqualityComponents()
|
|
86
|
+
{
|
|
87
|
+
yield return Street;
|
|
88
|
+
yield return City;
|
|
89
|
+
yield return State;
|
|
90
|
+
yield return ZipCode;
|
|
91
|
+
yield return Country;
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
```
|
|
95
|
+
|
|
96
|
+
## EF Core Configuration (Owned Types)
|
|
97
|
+
```csharp
|
|
98
|
+
public class OrderConfiguration : IEntityTypeConfiguration<Order>
|
|
99
|
+
{
|
|
100
|
+
public void Configure(EntityTypeBuilder<Order> builder)
|
|
101
|
+
{
|
|
102
|
+
// Embedded in same table (no separate table)
|
|
103
|
+
builder.OwnsOne(o => o.ShippingAddress, addr =>
|
|
104
|
+
{
|
|
105
|
+
addr.Property(a => a.Street).HasMaxLength(200).IsRequired();
|
|
106
|
+
addr.Property(a => a.City).HasMaxLength(100).IsRequired();
|
|
107
|
+
addr.Property(a => a.ZipCode).HasMaxLength(20).IsRequired();
|
|
108
|
+
});
|
|
109
|
+
|
|
110
|
+
// Money: stored as separate Amount + Currency columns
|
|
111
|
+
builder.OwnsOne(o => o.Total, money =>
|
|
112
|
+
{
|
|
113
|
+
money.Property(m => m.Amount).HasColumnType("decimal(18,2)").IsRequired();
|
|
114
|
+
money.Property(m => m.Currency).HasMaxLength(3).IsRequired();
|
|
115
|
+
});
|
|
116
|
+
}
|
|
117
|
+
}
|
|
118
|
+
```
|
|
119
|
+
|
|
120
|
+
## Rules
|
|
121
|
+
- Value objects are always immutable — no setters, return new instance from methods
|
|
122
|
+
- Use `record` types only when they don't need to inherit from `ValueObject` base class
|
|
123
|
+
- Never use value objects as EF Core entity (no `Id` column)
|
|
124
|
+
- Never create value objects with invalid state — validate in constructor/factory
|