@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.
Files changed (232) hide show
  1. package/.morph/.morphversion +5 -0
  2. package/.morph/analytics/threads-log.jsonl +5 -0
  3. package/.morph/config/config.json +8 -0
  4. package/.morph/framework/agents.json +1815 -0
  5. package/.morph/framework/hooks/README.md +205 -0
  6. package/.morph/framework/hooks/claude-code/notification/approval-reminder.js +54 -0
  7. package/.morph/framework/hooks/claude-code/post-tool-use/dispatch.js +83 -0
  8. package/.morph/framework/hooks/claude-code/post-tool-use/handle-tool-failure.js +42 -0
  9. package/.morph/framework/hooks/claude-code/pre-compact/save-morph-context.js +61 -0
  10. package/.morph/framework/hooks/claude-code/pre-tool-use/enforce-phase-writes.js +71 -0
  11. package/.morph/framework/hooks/claude-code/pre-tool-use/protect-readonly-files.js +58 -0
  12. package/.morph/framework/hooks/claude-code/pre-tool-use/protect-spec-files.js +64 -0
  13. package/.morph/framework/hooks/claude-code/session-start/inject-morph-context.js +94 -0
  14. package/.morph/framework/hooks/claude-code/statusline.py +538 -0
  15. package/.morph/framework/hooks/claude-code/statusline.sh +7 -0
  16. package/.morph/framework/hooks/claude-code/stop/validate-completion.js +88 -0
  17. package/.morph/framework/hooks/claude-code/user-prompt/enrich-prompt.js +91 -0
  18. package/.morph/framework/hooks/git/commit-msg/conventional-commits.sh +33 -0
  19. package/.morph/framework/hooks/git/pre-commit/agents.sh +25 -0
  20. package/.morph/framework/hooks/git/pre-commit/orchestrator.sh +64 -0
  21. package/.morph/framework/hooks/git/pre-commit/specs.sh +50 -0
  22. package/.morph/framework/hooks/git/pre-push/run-tests.sh +44 -0
  23. package/.morph/framework/hooks/shared/hook-response.js +45 -0
  24. package/.morph/framework/hooks/shared/phase-utils.js +129 -0
  25. package/.morph/framework/hooks/shared/state-reader.js +138 -0
  26. package/.morph/framework/hooks/shared/stdin-reader.js +26 -0
  27. package/.morph/framework/standards/STANDARDS.json +933 -0
  28. package/.morph/framework/standards/ai-agents/blazor-ui.md +364 -0
  29. package/.morph/framework/standards/ai-agents/production.md +415 -0
  30. package/.morph/framework/standards/ai-agents/setup.md +418 -0
  31. package/.morph/framework/standards/ai-agents/team-orchestration.md +479 -0
  32. package/.morph/framework/standards/ai-agents/workflows.md +354 -0
  33. package/.morph/framework/standards/architecture/ddd/aggregates.md +120 -0
  34. package/.morph/framework/standards/architecture/ddd/bounded-contexts.md +105 -0
  35. package/.morph/framework/standards/architecture/ddd/complexity-levels.md +108 -0
  36. package/.morph/framework/standards/architecture/ddd/entities.md +99 -0
  37. package/.morph/framework/standards/architecture/ddd/ubiquitous-language.md +58 -0
  38. package/.morph/framework/standards/architecture/ddd/value-objects.md +124 -0
  39. package/.morph/framework/standards/backend/api/minimal-api.md +494 -0
  40. package/.morph/framework/standards/backend/api/rest.md +492 -0
  41. package/.morph/framework/standards/backend/api/validation.md +88 -0
  42. package/.morph/framework/standards/backend/authentication/passkeys.md +428 -0
  43. package/.morph/framework/standards/backend/database/ef-core.md +199 -0
  44. package/.morph/framework/standards/backend/database/migrations.md +393 -0
  45. package/.morph/framework/standards/backend/database/postgresql/database.md +352 -0
  46. package/.morph/framework/standards/backend/database/repository-patterns.md +528 -0
  47. package/.morph/framework/standards/backend/database/vector-search-rag.md +541 -0
  48. package/.morph/framework/standards/backend/dotnet/async.md +366 -0
  49. package/.morph/framework/standards/backend/dotnet/core.md +117 -0
  50. package/.morph/framework/standards/backend/dotnet/di.md +439 -0
  51. package/.morph/framework/standards/backend/dotnet/program-cs-checklist.md +92 -0
  52. package/.morph/framework/standards/backend/integrations/asaas/asaas-api.md +216 -0
  53. package/.morph/framework/standards/backend/integrations/clerk/clerk-auth.md +290 -0
  54. package/.morph/framework/standards/backend/integrations/hangfire/hangfire-jobs.md +350 -0
  55. package/.morph/framework/standards/backend/integrations/resend/resend-email.md +385 -0
  56. package/.morph/framework/standards/context/analytics.md +96 -0
  57. package/.morph/framework/standards/context/bundles.md +110 -0
  58. package/.morph/framework/standards/context/priming.md +78 -0
  59. package/.morph/framework/standards/core/architecture.md +185 -0
  60. package/.morph/framework/standards/core/coding.md +214 -0
  61. package/.morph/framework/standards/core/git-branching-strategy.md +403 -0
  62. package/.morph/framework/standards/core/git.md +185 -0
  63. package/.morph/framework/standards/core/testing.md +295 -0
  64. package/.morph/framework/standards/data/nosql/blob-storage.md +102 -0
  65. package/.morph/framework/standards/data/nosql/cache/redis.md +97 -0
  66. package/.morph/framework/standards/data/nosql/cosmos-db.md +118 -0
  67. package/.morph/framework/standards/data/vector-search/azure-ai-search.md +121 -0
  68. package/.morph/framework/standards/data/vector-search/rag-chunking.md +104 -0
  69. package/.morph/framework/standards/frontend/blazor/design-checklist.md +222 -0
  70. package/.morph/framework/standards/frontend/blazor/fluent-ui-setup.md +595 -0
  71. package/.morph/framework/standards/frontend/blazor/fluent-ui.md +137 -0
  72. package/.morph/framework/standards/frontend/blazor/html-conversion.md +184 -0
  73. package/.morph/framework/standards/frontend/blazor/lifecycle.md +195 -0
  74. package/.morph/framework/standards/frontend/blazor/pitfalls.md +198 -0
  75. package/.morph/framework/standards/frontend/blazor/state.md +191 -0
  76. package/.morph/framework/standards/frontend/design-system/animations.md +151 -0
  77. package/.morph/framework/standards/frontend/design-system/naming.md +64 -0
  78. package/.morph/framework/standards/frontend/nextjs/app-router.md +123 -0
  79. package/.morph/framework/standards/frontend/nextjs/components.md +132 -0
  80. package/.morph/framework/standards/frontend/nextjs/data-fetching.md +126 -0
  81. package/.morph/framework/standards/frontend/nextjs/forms.md +128 -0
  82. package/.morph/framework/standards/frontend/nextjs/naming-conventions.md +67 -0
  83. package/.morph/framework/standards/frontend/nextjs/nextjs-patterns.md +215 -0
  84. package/.morph/framework/standards/frontend/nextjs/project-structure.md +102 -0
  85. package/.morph/framework/standards/frontend/nextjs/state-management.md +72 -0
  86. package/.morph/framework/standards/frontend/nextjs/testing.md +111 -0
  87. package/.morph/framework/standards/infrastructure/azure/azure.md +624 -0
  88. package/.morph/framework/standards/infrastructure/azure/bicep/bicep-patterns.md +422 -0
  89. package/.morph/framework/standards/infrastructure/azure/devops/azure-devops-setup.md +516 -0
  90. package/.morph/framework/standards/infrastructure/azure/devops/local-development.md +520 -0
  91. package/.morph/framework/standards/infrastructure/azure/services/functions.md +486 -0
  92. package/.morph/framework/standards/infrastructure/azure/services/service-bus.md +459 -0
  93. package/.morph/framework/standards/infrastructure/azure/services/storage.md +407 -0
  94. package/.morph/framework/standards/infrastructure/docker/easypanel-deploy.md +196 -0
  95. package/.morph/framework/standards/infrastructure/supabase/mcp-setup.md +252 -0
  96. package/.morph/framework/standards/infrastructure/supabase/supabase-auth.md +176 -0
  97. package/.morph/framework/standards/infrastructure/supabase/supabase-pgvector.md +169 -0
  98. package/.morph/framework/standards/infrastructure/supabase/supabase-rls.md +184 -0
  99. package/.morph/framework/standards/infrastructure/supabase/supabase-storage.md +153 -0
  100. package/.morph/framework/standards/integration/api/graphql.md +91 -0
  101. package/.morph/framework/standards/integration/api/grpc.md +114 -0
  102. package/.morph/framework/standards/integration/api/rest-design.md +95 -0
  103. package/.morph/framework/standards/integration/event-driven/cqrs.md +101 -0
  104. package/.morph/framework/standards/integration/event-driven/event-sourcing.md +124 -0
  105. package/.morph/framework/standards/integration/event-driven/service-bus.md +95 -0
  106. package/.morph/framework/standards/integration/mcp/mcp-tools.md +384 -0
  107. package/.morph/framework/standards/observability/logging.md +131 -0
  108. package/.morph/framework/standards/observability/metrics.md +121 -0
  109. package/.morph/framework/standards/observability/monitoring.md +114 -0
  110. package/.morph/framework/standards/observability/tracing.md +132 -0
  111. package/.morph/framework/standards/workflows/parallel-execution.md +112 -0
  112. package/.morph/framework/standards/workflows/thread-management.md +113 -0
  113. package/.morph/framework/templates/.idea/morph-templates.xml +92 -0
  114. package/.morph/framework/templates/.vscode/morph-templates.code-snippets +186 -0
  115. package/.morph/framework/templates/IDE-SNIPPETS.md +266 -0
  116. package/.morph/framework/templates/README.md +814 -0
  117. package/.morph/framework/templates/REGISTRY.json +1888 -0
  118. package/.morph/framework/templates/code/dotnet/backend/repository.cs +141 -0
  119. package/.morph/framework/templates/code/dotnet/backend/service.cs +139 -0
  120. package/.morph/framework/templates/code/dotnet/contracts/Commands.cs +74 -0
  121. package/.morph/framework/templates/code/dotnet/contracts/Entities.cs +25 -0
  122. package/.morph/framework/templates/code/dotnet/contracts/Queries.cs +74 -0
  123. package/.morph/framework/templates/code/dotnet/contracts/README.md +74 -0
  124. package/.morph/framework/templates/code/dotnet/contracts/api-contracts.cs +173 -0
  125. package/.morph/framework/templates/code/dotnet/contracts/contracts-level1.cs +69 -0
  126. package/.morph/framework/templates/code/dotnet/contracts/contracts-level2.cs +86 -0
  127. package/.morph/framework/templates/code/dotnet/contracts/contracts-level3.cs +41 -0
  128. package/.morph/framework/templates/code/dotnet/database/migration.cs +83 -0
  129. package/.morph/framework/templates/code/dotnet/frontend/component.razor +239 -0
  130. package/.morph/framework/templates/code/dotnet/jobs/agent.cs +163 -0
  131. package/.morph/framework/templates/code/dotnet/jobs/job.cs +171 -0
  132. package/.morph/framework/templates/code/dotnet/test.cs +239 -0
  133. package/.morph/framework/templates/code/sql/rls-policy.sql +57 -0
  134. package/.morph/framework/templates/code/sql/supabase-migration.sql +100 -0
  135. package/.morph/framework/templates/code/sql/supabase-migration.template.sql +113 -0
  136. package/.morph/framework/templates/code/typescript/contracts.ts +168 -0
  137. package/.morph/framework/templates/context/CONTEXT-FEATURE.md +276 -0
  138. package/.morph/framework/templates/context/CONTEXT.md +181 -0
  139. package/.morph/framework/templates/docs/clarifications.md +253 -0
  140. package/.morph/framework/templates/docs/onboarding.md +123 -0
  141. package/.morph/framework/templates/docs/proposal.md +182 -0
  142. package/.morph/framework/templates/docs/schema-analysis.md +119 -0
  143. package/.morph/framework/templates/docs/spec.md +198 -0
  144. package/.morph/framework/templates/docs/ui-components.md +124 -0
  145. package/.morph/framework/templates/docs/ui-design-system.md +76 -0
  146. package/.morph/framework/templates/docs/ui-flows.md +167 -0
  147. package/.morph/framework/templates/docs/ui-mockups.md +98 -0
  148. package/.morph/framework/templates/docs/user-stories.md +34 -0
  149. package/.morph/framework/templates/examples/design-system-examples.md +357 -0
  150. package/.morph/framework/templates/examples/spec-examples.md +90 -0
  151. package/.morph/framework/templates/feature/decisions.md +187 -0
  152. package/.morph/framework/templates/feature/recap.md +146 -0
  153. package/.morph/framework/templates/feature/tasks.md +199 -0
  154. package/.morph/framework/templates/frontend/nextjs/Dockerfile.nextjs.hbs +43 -0
  155. package/.morph/framework/templates/frontend/nextjs/client-component.tsx.hbs +26 -0
  156. package/.morph/framework/templates/frontend/nextjs/env.mjs.hbs +32 -0
  157. package/.morph/framework/templates/frontend/nextjs/feature-form.tsx.hbs +56 -0
  158. package/.morph/framework/templates/frontend/nextjs/page.tsx.hbs +22 -0
  159. package/.morph/framework/templates/frontend/nextjs/tsconfig.json.hbs +26 -0
  160. package/.morph/framework/templates/frontend/nextjs/use-feature.ts.hbs +54 -0
  161. package/.morph/framework/templates/infrastructure/azure/Dockerfile.example +82 -0
  162. package/.morph/framework/templates/infrastructure/azure/README.md +286 -0
  163. package/.morph/framework/templates/infrastructure/azure/app-insights.bicep +63 -0
  164. package/.morph/framework/templates/infrastructure/azure/app-service.bicep +164 -0
  165. package/.morph/framework/templates/infrastructure/azure/container-app-env.bicep +49 -0
  166. package/.morph/framework/templates/infrastructure/azure/container-app.bicep +156 -0
  167. package/.morph/framework/templates/infrastructure/azure/deploy-checklist.md +426 -0
  168. package/.morph/framework/templates/infrastructure/azure/deploy.ps1 +229 -0
  169. package/.morph/framework/templates/infrastructure/azure/deploy.sh +208 -0
  170. package/.morph/framework/templates/infrastructure/azure/key-vault.bicep +91 -0
  171. package/.morph/framework/templates/infrastructure/azure/main.bicep +189 -0
  172. package/.morph/framework/templates/infrastructure/azure/parameters.dev.json +29 -0
  173. package/.morph/framework/templates/infrastructure/azure/parameters.prod.json +29 -0
  174. package/.morph/framework/templates/infrastructure/azure/parameters.staging.json +29 -0
  175. package/.morph/framework/templates/infrastructure/azure/sql-database.bicep +103 -0
  176. package/.morph/framework/templates/infrastructure/azure/storage.bicep +106 -0
  177. package/.morph/framework/templates/infrastructure/docker/Dockerfile.template +58 -0
  178. package/.morph/framework/templates/infrastructure/docker/docker-compose.template.yml +67 -0
  179. package/.morph/framework/templates/infrastructure/docker/dockerfile-api.dockerfile +38 -0
  180. package/.morph/framework/templates/infrastructure/docker/dockerfile-web.dockerfile +48 -0
  181. package/.morph/framework/templates/infrastructure/docker/easypanel.template.json +54 -0
  182. package/.morph/framework/templates/infrastructure/github/README.md +593 -0
  183. package/.morph/framework/templates/infrastructure/github/actions/azure-auth/action.yml.hbs +22 -0
  184. package/.morph/framework/templates/infrastructure/github/actions/docker-build-push/action.yml.hbs +45 -0
  185. package/.morph/framework/templates/infrastructure/github/actions/health-check/action.yml.hbs +27 -0
  186. package/.morph/framework/templates/infrastructure/github/workflows/deploy-azure-app-service.yml.hbs +61 -0
  187. package/.morph/framework/templates/infrastructure/github/workflows/deploy-easypanel.yml.hbs +31 -0
  188. package/.morph/framework/templates/infrastructure/github/workflows/docker-build-push.yml.hbs +59 -0
  189. package/.morph/framework/templates/infrastructure/github/workflows/dotnet-build.yml.hbs +39 -0
  190. package/.morph/framework/templates/integrations/asaas-client.cs +387 -0
  191. package/.morph/framework/templates/integrations/asaas-webhook.cs +351 -0
  192. package/.morph/framework/templates/integrations/azure-identity-config.cs +288 -0
  193. package/.morph/framework/templates/integrations/clerk-config.cs +258 -0
  194. package/.morph/framework/templates/meta-prompts/fusion/fusion-agent.md +76 -0
  195. package/.morph/framework/templates/meta-prompts/fusion/fusion-aggregator.md +100 -0
  196. package/.morph/framework/templates/meta-prompts/hops/hop-retry.md +78 -0
  197. package/.morph/framework/templates/meta-prompts/hops/hop-validation.md +97 -0
  198. package/.morph/framework/templates/meta-prompts/hops/hop-wrapper.md +36 -0
  199. package/.morph/framework/templates/meta-prompts/parallel-workers/parallel-coordinator.md +113 -0
  200. package/.morph/framework/templates/meta-prompts/parallel-workers/parallel-worker.md +80 -0
  201. package/.morph/framework/templates/meta-prompts/squad-leaders/backend-squad.md +90 -0
  202. package/.morph/framework/templates/meta-prompts/squad-leaders/frontend-squad.md +126 -0
  203. package/.morph/framework/templates/meta-prompts/squad-leaders/squad-leader.md +43 -0
  204. package/.morph/framework/templates/meta-prompts/validators/checkpoint-validator.md +107 -0
  205. package/.morph/framework/templates/meta-prompts/validators/pre-commit-validator.md +95 -0
  206. package/.morph/framework/templates/project-structure/dotnet-ddd.md +70 -0
  207. package/.morph/framework/templates/saas/subscription.cs +347 -0
  208. package/.morph/framework/templates/saas/tenant.cs +338 -0
  209. package/.morph/framework/templates/state.template.json +17 -0
  210. package/.morph/framework/templates/ui/FluentDesignTheme.cs +149 -0
  211. package/.morph/framework/templates/ui/MudTheme.cs +281 -0
  212. package/.morph/framework/templates/ui/design-system.css +226 -0
  213. package/.morph/logs/tool-failures.log +17 -0
  214. package/.morph/memory/pre-compact-2026-02-24T17-43-30-049Z.json +16 -0
  215. package/.morph/plans/eager-watching-bunny.md +105 -0
  216. package/.morph/plans/temporal-seeking-nebula.md +45 -0
  217. package/.morph/state.json +48 -0
  218. package/CLAUDE.md +1 -1
  219. package/README.md +119 -99
  220. package/bin/morph-spec.js +0 -9
  221. package/framework/CLAUDE.md +1 -1
  222. package/framework/hooks/README.md +10 -6
  223. package/framework/hooks/claude-code/notification/approval-reminder.js +2 -0
  224. package/framework/hooks/claude-code/post-tool-use/dispatch.js +1 -1
  225. package/framework/hooks/claude-code/stop/validate-completion.js +1 -1
  226. package/framework/hooks/claude-code/user-prompt/enrich-prompt.js +1 -1
  227. package/package.json +1 -1
  228. package/src/commands/project/init.js +15 -42
  229. package/src/commands/project/update.js +22 -37
  230. package/src/lib/installers/mcp-installer.js +18 -3
  231. package/src/utils/hooks-installer.js +5 -15
  232. 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