@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,338 @@
1
+ // ==============================================================================
2
+ // MORPH-SPEC - Multi-Tenant Model Template
3
+ // Modelo de multi-tenancy para SaaS
4
+ // ==============================================================================
5
+
6
+ using System.ComponentModel.DataAnnotations;
7
+ using Microsoft.EntityFrameworkCore;
8
+
9
+ namespace {{Namespace}}.Domain.Entities;
10
+
11
+ // ==============================================================================
12
+ // TENANT ENTITY
13
+ // ==============================================================================
14
+
15
+ public class Tenant : BaseEntity
16
+ {
17
+ [Required]
18
+ [MaxLength(100)]
19
+ public string Name { get; private set; } = string.Empty;
20
+
21
+ [Required]
22
+ [MaxLength(50)]
23
+ public string Slug { get; private set; } = string.Empty; // Unique identifier (URL-friendly)
24
+
25
+ [MaxLength(500)]
26
+ public string? Description { get; private set; }
27
+
28
+ [MaxLength(200)]
29
+ public string? LogoUrl { get; private set; }
30
+
31
+ public TenantStatus Status { get; private set; }
32
+
33
+ // Settings
34
+ public string? CustomDomain { get; private set; }
35
+ public string? TimeZone { get; private set; }
36
+ public string? Locale { get; private set; }
37
+
38
+ // Billing
39
+ public string? AsaasCustomerId { get; private set; }
40
+
41
+ // Navigation
42
+ public ICollection<TenantUser> Users { get; private set; } = new List<TenantUser>();
43
+ public Subscription? CurrentSubscription { get; private set; }
44
+
45
+ // =========================================================================
46
+ // FACTORY
47
+ // =========================================================================
48
+
49
+ public static Tenant Create(string name, string slug)
50
+ {
51
+ return new Tenant
52
+ {
53
+ Name = name,
54
+ Slug = slug.ToLowerInvariant(),
55
+ Status = TenantStatus.Active,
56
+ TimeZone = "E. South America Standard Time",
57
+ Locale = "pt-BR",
58
+ CreatedAt = DateTime.UtcNow
59
+ };
60
+ }
61
+
62
+ // =========================================================================
63
+ // BEHAVIORS
64
+ // =========================================================================
65
+
66
+ public void UpdateInfo(string name, string? description, string? logoUrl)
67
+ {
68
+ Name = name;
69
+ Description = description;
70
+ LogoUrl = logoUrl;
71
+ UpdatedAt = DateTime.UtcNow;
72
+ }
73
+
74
+ public void SetCustomDomain(string? domain)
75
+ {
76
+ CustomDomain = domain;
77
+ UpdatedAt = DateTime.UtcNow;
78
+ }
79
+
80
+ public void SetAsaasCustomerId(string customerId)
81
+ {
82
+ AsaasCustomerId = customerId;
83
+ UpdatedAt = DateTime.UtcNow;
84
+ }
85
+
86
+ public void Suspend()
87
+ {
88
+ Status = TenantStatus.Suspended;
89
+ UpdatedAt = DateTime.UtcNow;
90
+ }
91
+
92
+ public void Activate()
93
+ {
94
+ Status = TenantStatus.Active;
95
+ UpdatedAt = DateTime.UtcNow;
96
+ }
97
+
98
+ public void Delete()
99
+ {
100
+ Status = TenantStatus.Deleted;
101
+ UpdatedAt = DateTime.UtcNow;
102
+ }
103
+
104
+ // =========================================================================
105
+ // QUERIES
106
+ // =========================================================================
107
+
108
+ public bool IsActive => Status == TenantStatus.Active;
109
+ public bool IsSuspended => Status == TenantStatus.Suspended;
110
+ public bool IsDeleted => Status == TenantStatus.Deleted;
111
+ }
112
+
113
+ // ==============================================================================
114
+ // TENANT USER ENTITY (Association between Users and Tenants)
115
+ // ==============================================================================
116
+
117
+ public class TenantUser : BaseEntity
118
+ {
119
+ public int TenantId { get; private set; }
120
+ public Tenant Tenant { get; private set; } = null!;
121
+
122
+ [Required]
123
+ [MaxLength(100)]
124
+ public string ExternalUserId { get; private set; } = string.Empty; // Clerk/Azure AD user ID
125
+
126
+ [Required]
127
+ [MaxLength(200)]
128
+ public string Email { get; private set; } = string.Empty;
129
+
130
+ [MaxLength(100)]
131
+ public string? Name { get; private set; }
132
+
133
+ public TenantUserRole Role { get; private set; }
134
+ public TenantUserStatus Status { get; private set; }
135
+
136
+ public DateTime? LastLoginAt { get; private set; }
137
+
138
+ // =========================================================================
139
+ // FACTORY
140
+ // =========================================================================
141
+
142
+ public static TenantUser Create(
143
+ int tenantId,
144
+ string externalUserId,
145
+ string email,
146
+ string? name,
147
+ TenantUserRole role = TenantUserRole.Member)
148
+ {
149
+ return new TenantUser
150
+ {
151
+ TenantId = tenantId,
152
+ ExternalUserId = externalUserId,
153
+ Email = email,
154
+ Name = name,
155
+ Role = role,
156
+ Status = TenantUserStatus.Active,
157
+ CreatedAt = DateTime.UtcNow
158
+ };
159
+ }
160
+
161
+ // =========================================================================
162
+ // BEHAVIORS
163
+ // =========================================================================
164
+
165
+ public void UpdateRole(TenantUserRole role)
166
+ {
167
+ Role = role;
168
+ UpdatedAt = DateTime.UtcNow;
169
+ }
170
+
171
+ public void RecordLogin()
172
+ {
173
+ LastLoginAt = DateTime.UtcNow;
174
+ }
175
+
176
+ public void Deactivate()
177
+ {
178
+ Status = TenantUserStatus.Inactive;
179
+ UpdatedAt = DateTime.UtcNow;
180
+ }
181
+
182
+ public void Activate()
183
+ {
184
+ Status = TenantUserStatus.Active;
185
+ UpdatedAt = DateTime.UtcNow;
186
+ }
187
+
188
+ // =========================================================================
189
+ // QUERIES
190
+ // =========================================================================
191
+
192
+ public bool IsOwner => Role == TenantUserRole.Owner;
193
+ public bool IsAdmin => Role == TenantUserRole.Owner || Role == TenantUserRole.Admin;
194
+ public bool IsActive => Status == TenantUserStatus.Active;
195
+ }
196
+
197
+ // ==============================================================================
198
+ // ENUMS
199
+ // ==============================================================================
200
+
201
+ public enum TenantStatus
202
+ {
203
+ Active,
204
+ Suspended,
205
+ Deleted
206
+ }
207
+
208
+ public enum TenantUserRole
209
+ {
210
+ Owner,
211
+ Admin,
212
+ Member,
213
+ Viewer
214
+ }
215
+
216
+ public enum TenantUserStatus
217
+ {
218
+ Pending,
219
+ Active,
220
+ Inactive
221
+ }
222
+
223
+ // ==============================================================================
224
+ // TENANT CONTEXT (Current Tenant Resolution)
225
+ // ==============================================================================
226
+
227
+ public interface ITenantContext
228
+ {
229
+ int? TenantId { get; }
230
+ Tenant? CurrentTenant { get; }
231
+ Task<Tenant?> GetCurrentTenantAsync(CancellationToken ct = default);
232
+ }
233
+
234
+ public class TenantContext : ITenantContext
235
+ {
236
+ private readonly IHttpContextAccessor _httpContextAccessor;
237
+ private readonly ITenantRepository _tenantRepository;
238
+ private Tenant? _currentTenant;
239
+ private bool _loaded;
240
+
241
+ public TenantContext(
242
+ IHttpContextAccessor httpContextAccessor,
243
+ ITenantRepository tenantRepository)
244
+ {
245
+ _httpContextAccessor = httpContextAccessor;
246
+ _tenantRepository = tenantRepository;
247
+ }
248
+
249
+ public int? TenantId => _currentTenant?.Id;
250
+ public Tenant? CurrentTenant => _currentTenant;
251
+
252
+ public async Task<Tenant?> GetCurrentTenantAsync(CancellationToken ct = default)
253
+ {
254
+ if (_loaded) return _currentTenant;
255
+
256
+ var tenantSlug = ResolveTenantSlug();
257
+ if (string.IsNullOrEmpty(tenantSlug))
258
+ {
259
+ _loaded = true;
260
+ return null;
261
+ }
262
+
263
+ _currentTenant = await _tenantRepository.GetBySlugAsync(tenantSlug, ct);
264
+ _loaded = true;
265
+
266
+ return _currentTenant;
267
+ }
268
+
269
+ private string? ResolveTenantSlug()
270
+ {
271
+ var httpContext = _httpContextAccessor.HttpContext;
272
+ if (httpContext is null) return null;
273
+
274
+ // Strategy 1: From route (e.g., /tenant/{slug}/...)
275
+ if (httpContext.Request.RouteValues.TryGetValue("tenantSlug", out var routeSlug))
276
+ return routeSlug?.ToString();
277
+
278
+ // Strategy 2: From header (e.g., X-Tenant-ID)
279
+ if (httpContext.Request.Headers.TryGetValue("X-Tenant-ID", out var headerSlug))
280
+ return headerSlug.FirstOrDefault();
281
+
282
+ // Strategy 3: From subdomain (e.g., acme.app.com)
283
+ var host = httpContext.Request.Host.Host;
284
+ var parts = host.Split('.');
285
+ if (parts.Length >= 3)
286
+ return parts[0];
287
+
288
+ // Strategy 4: From claim
289
+ var claimSlug = httpContext.User.FindFirst("tenant_id")?.Value;
290
+ if (!string.IsNullOrEmpty(claimSlug))
291
+ return claimSlug;
292
+
293
+ return null;
294
+ }
295
+ }
296
+
297
+ // ==============================================================================
298
+ // TENANT REPOSITORY INTERFACE
299
+ // ==============================================================================
300
+
301
+ public interface ITenantRepository
302
+ {
303
+ Task<Tenant?> GetByIdAsync(int id, CancellationToken ct = default);
304
+ Task<Tenant?> GetBySlugAsync(string slug, CancellationToken ct = default);
305
+ Task<bool> SlugExistsAsync(string slug, CancellationToken ct = default);
306
+ Task AddAsync(Tenant tenant, CancellationToken ct = default);
307
+ void Update(Tenant tenant);
308
+ }
309
+
310
+ // ==============================================================================
311
+ // TENANT QUERY FILTER (EF Core)
312
+ // ==============================================================================
313
+
314
+ /*
315
+ // In DbContext:
316
+ protected override void OnModelCreating(ModelBuilder modelBuilder)
317
+ {
318
+ // Apply tenant filter to all tenant-scoped entities
319
+ modelBuilder.Entity<Order>().HasQueryFilter(o => o.TenantId == _tenantContext.TenantId);
320
+ modelBuilder.Entity<Customer>().HasQueryFilter(c => c.TenantId == _tenantContext.TenantId);
321
+ // ... other entities
322
+ }
323
+ */
324
+
325
+ // ==============================================================================
326
+ // DEPENDENCY INJECTION
327
+ // ==============================================================================
328
+
329
+ public static class TenantServiceExtensions
330
+ {
331
+ public static IServiceCollection AddMultiTenancy(this IServiceCollection services)
332
+ {
333
+ services.AddScoped<ITenantContext, TenantContext>();
334
+ services.AddScoped<ITenantRepository, TenantRepository>();
335
+
336
+ return services;
337
+ }
338
+ }
@@ -0,0 +1,17 @@
1
+ {
2
+ "version": "3.0.0",
3
+ "project": {
4
+ "name": "{{PROJECT_NAME}}",
5
+ "type": "{{PROJECT_TYPE}}",
6
+ "createdAt": "{{TIMESTAMP}}",
7
+ "updatedAt": "{{TIMESTAMP}}"
8
+ },
9
+ "features": {},
10
+ "threads": {},
11
+ "metadata": {
12
+ "totalFeatures": 0,
13
+ "completedFeatures": 0,
14
+ "totalTimeSpent": 0,
15
+ "lastUpdated": "{{TIMESTAMP}}"
16
+ }
17
+ }
@@ -0,0 +1,149 @@
1
+ using Microsoft.FluentUI.AspNetCore.Components;
2
+
3
+ namespace {Namespace}.Themes;
4
+
5
+ /// <summary>
6
+ /// Fluent UI Design Theme customizado para {Project Name}.
7
+ /// Gerado automaticamente baseado em ui-design-system.md
8
+ /// </summary>
9
+ /// <remarks>
10
+ /// Template MORPH-SPEC v2.1.1 by Polymorphism Tech
11
+ /// </remarks>
12
+ public static class FluentDesignTheme
13
+ {
14
+ /// <summary>
15
+ /// Paleta de cores do tema
16
+ /// </summary>
17
+ public static DesignThemePalette GetPalette() => new()
18
+ {
19
+ // Cores Principais
20
+ Primary = "#{hex da cor primária}", // Ex: #3b82f6
21
+ Secondary = "#{hex da cor secundária}", // Ex: #6b7280
22
+
23
+ // Cores de Estado
24
+ Success = "#{hex}", // Ex: #10b981
25
+ Error = "#{hex}", // Ex: #ef4444
26
+ Warning = "#{hex}", // Ex: #f59e0b
27
+ Info = "#{hex}", // Ex: #06b6d4
28
+
29
+ // Neutros
30
+ NeutralLight = "#{hex do gray-50}", // Ex: #f9fafb
31
+ NeutralLighter = "#{hex do gray-100}", // Ex: #f3f4f6
32
+ NeutralDark = "#{hex do gray-800}", // Ex: #1f2937
33
+ NeutralDarker = "#{hex do gray-900}", // Ex: #111827
34
+
35
+ // Background e Foreground
36
+ Background = "#ffffff",
37
+ Foreground = "#{hex do gray-900}", // Texto principal
38
+
39
+ // Accent (geralmente = Primary)
40
+ Accent = "#{hex da cor primária}",
41
+ };
42
+
43
+ /// <summary>
44
+ /// Configuração de tipografia
45
+ /// </summary>
46
+ public static DesignThemeTypography GetTypography() => new()
47
+ {
48
+ FontFamily = "'{Font Name}', -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif",
49
+
50
+ // Tamanhos
51
+ FontSizeBase = "1rem", // 16px
52
+ FontSizeSmall = "0.875rem", // 14px
53
+ FontSizeLarge = "1.125rem", // 18px
54
+
55
+ // Weights
56
+ FontWeightRegular = 400,
57
+ FontWeightSemibold = 600,
58
+ FontWeightBold = 700,
59
+
60
+ // Line Heights
61
+ LineHeightBase = 1.5,
62
+ };
63
+
64
+ /// <summary>
65
+ /// Configuração de espaçamento e layout
66
+ /// </summary>
67
+ public static DesignThemeLayout GetLayout() => new()
68
+ {
69
+ // Espaçamento (baseado em múltiplos de {base}px - ex: 4px)
70
+ SpacingBase = "0.25rem", // 4px
71
+ SpacingSmall = "0.5rem", // 8px
72
+ SpacingMedium = "1rem", // 16px
73
+ SpacingLarge = "1.5rem", // 24px
74
+
75
+ // Border Radius
76
+ BorderRadiusSmall = "0.125rem", // 2px
77
+ BorderRadiusMedium = "0.375rem", // 6px
78
+ BorderRadiusLarge = "0.5rem", // 8px
79
+
80
+ // Shadows (depende da biblioteca - ajustar conforme API)
81
+ Elevation1 = "0 1px 2px 0 rgba(0, 0, 0, 0.05)",
82
+ Elevation2 = "0 4px 6px -1px rgba(0, 0, 0, 0.1)",
83
+ Elevation3 = "0 10px 15px -3px rgba(0, 0, 0, 0.1)",
84
+ };
85
+
86
+ /// <summary>
87
+ /// Aplica o tema customizado ao FluentUI
88
+ /// </summary>
89
+ /// <param name="services">Service collection</param>
90
+ public static void AddFluentUIWithCustomTheme(this IServiceCollection services)
91
+ {
92
+ services.AddFluentUIComponents(options =>
93
+ {
94
+ // Aplicar paleta customizada
95
+ options.DesignThemePalette = GetPalette();
96
+
97
+ // Outras configurações do Fluent UI
98
+ // ...
99
+ });
100
+ }
101
+ }
102
+
103
+ /// <summary>
104
+ /// Classe de configuração de tipografia
105
+ /// (Ajustar conforme API do Fluent UI Blazor v4.x)
106
+ /// </summary>
107
+ public class DesignThemeTypography
108
+ {
109
+ public string FontFamily { get; set; } = string.Empty;
110
+ public string FontSizeBase { get; set; } = string.Empty;
111
+ public string FontSizeSmall { get; set; } = string.Empty;
112
+ public string FontSizeLarge { get; set; } = string.Empty;
113
+ public int FontWeightRegular { get; set; }
114
+ public int FontWeightSemibold { get; set; }
115
+ public int FontWeightBold { get; set; }
116
+ public double LineHeightBase { get; set; }
117
+ }
118
+
119
+ /// <summary>
120
+ /// Classe de configuração de layout
121
+ /// (Ajustar conforme API do Fluent UI Blazor v4.x)
122
+ /// </summary>
123
+ public class DesignThemeLayout
124
+ {
125
+ public string SpacingBase { get; set; } = string.Empty;
126
+ public string SpacingSmall { get; set; } = string.Empty;
127
+ public string SpacingMedium { get; set; } = string.Empty;
128
+ public string SpacingLarge { get; set; } = string.Empty;
129
+
130
+ public string BorderRadiusSmall { get; set; } = string.Empty;
131
+ public string BorderRadiusMedium { get; set; } = string.Empty;
132
+ public string BorderRadiusLarge { get; set; } = string.Empty;
133
+
134
+ public string Elevation1 { get; set; } = string.Empty;
135
+ public string Elevation2 { get; set; } = string.Empty;
136
+ public string Elevation3 { get; set; } = string.Empty;
137
+ }
138
+
139
+ /// <summary>
140
+ /// EXEMPLO DE USO:
141
+ ///
142
+ /// // Program.cs
143
+ /// builder.Services.AddFluentUIWithCustomTheme();
144
+ ///
145
+ /// // App.razor ou MainLayout.razor
146
+ /// <FluentDesignTheme Mode="DesignThemeMode.Light" />
147
+ ///
148
+ /// // Componentes podem acessar cores via CSS variables ou propriedades do tema
149
+ /// </summary>