@polymorphism-tech/morph-spec 4.7.1 → 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 +2 -2
- 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,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>
|