@polymorphism-tech/morph-spec 2.4.0 → 3.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CLAUDE.md +158 -26
- package/LICENSE +72 -72
- package/bin/detect-agents.js +225 -225
- package/bin/morph-spec.js +8 -0
- package/bin/render-template.js +302 -302
- package/bin/semantic-detect-agents.js +246 -246
- package/bin/validate-agents-skills.js +251 -251
- package/bin/validate-agents.js +69 -69
- package/bin/validate-phase.js +263 -263
- package/content/.azure/README.md +293 -293
- package/content/.azure/docs/azure-devops-setup.md +454 -454
- package/content/.azure/docs/branch-strategy.md +398 -398
- package/content/.azure/docs/local-development.md +515 -515
- package/content/.azure/pipelines/pipeline-variables.yml +34 -34
- package/content/.azure/pipelines/prod-pipeline.yml +319 -319
- package/content/.azure/pipelines/staging-pipeline.yml +234 -234
- package/content/.azure/pipelines/templates/build-dotnet.yml +75 -75
- package/content/.azure/pipelines/templates/deploy-app-service.yml +94 -94
- package/content/.azure/pipelines/templates/deploy-container-app.yml +120 -120
- package/content/.azure/pipelines/templates/infra-deploy.yml +90 -90
- package/content/.claude/commands/morph-archive.md +79 -79
- package/content/.claude/commands/morph-deploy.md +529 -0
- package/content/.claude/commands/morph-infra.md +209 -209
- package/content/.claude/commands/morph-preflight.md +227 -227
- package/content/.claude/commands/morph-troubleshoot.md +122 -122
- package/content/.claude/settings.local.json +15 -15
- package/content/.claude/skills/infra/azure-deploy-specialist.md +699 -0
- package/content/.claude/skills/level-0-meta/README.md +7 -0
- package/content/.claude/skills/{checklists → level-0-meta}/morph-checklist.md +117 -117
- package/content/.claude/skills/level-1-workflows/README.md +7 -0
- package/content/.claude/skills/{workflows → level-1-workflows}/morph-replicate.md +213 -213
- package/content/.claude/skills/{workflows → level-1-workflows}/phase-clarify.md +131 -131
- package/content/.claude/skills/{workflows → level-1-workflows}/phase-design.md +213 -205
- package/content/.claude/skills/{workflows → level-1-workflows}/phase-setup.md +106 -92
- package/content/.claude/skills/{workflows → level-1-workflows}/phase-tasks.md +164 -164
- package/content/.claude/skills/{workflows → level-1-workflows}/phase-uiux.md +169 -138
- package/content/.claude/skills/level-2-domains/README.md +14 -0
- package/content/.claude/skills/{specialists → level-2-domains/quality}/testing-specialist.md +126 -126
- package/content/.claude/skills/level-3-technologies/README.md +7 -0
- package/content/.claude/skills/level-4-patterns/README.md +7 -0
- package/content/.claude/skills/specialists/prompt-engineer.md +189 -0
- package/content/.claude/skills/specialists/seo-growth-hacker.md +320 -0
- package/content/.morph/.morphversion +5 -5
- package/content/.morph/archive/.gitkeep +25 -25
- package/content/.morph/config/agents.json +742 -358
- package/content/.morph/config/config.template.json +33 -0
- package/content/.morph/docs/STORY-DRIVEN-DEVELOPMENT.md +392 -392
- package/content/.morph/docs/workflows/enforcement-pipeline.md +668 -0
- package/content/.morph/examples/api-nextjs/README.md +241 -241
- package/content/.morph/examples/api-nextjs/contracts.ts +307 -307
- package/content/.morph/examples/api-nextjs/spec.md +399 -399
- package/content/.morph/examples/api-nextjs/tasks.md +168 -168
- package/content/.morph/examples/micro-saas/README.md +125 -125
- package/content/.morph/examples/micro-saas/contracts.cs +358 -358
- package/content/.morph/examples/micro-saas/decisions.md +246 -246
- package/content/.morph/examples/micro-saas/spec.md +236 -236
- package/content/.morph/examples/micro-saas/tasks.md +150 -150
- package/content/.morph/examples/multi-agent/README.md +309 -309
- package/content/.morph/examples/multi-agent/contracts.cs +433 -433
- package/content/.morph/examples/multi-agent/spec.md +479 -479
- package/content/.morph/examples/multi-agent/tasks.md +185 -185
- package/content/.morph/examples/scheduled-reports/decisions.md +158 -158
- package/content/.morph/examples/scheduled-reports/proposal.md +95 -95
- package/content/.morph/examples/scheduled-reports/spec.md +267 -267
- package/content/.morph/examples/state-v3.json +188 -188
- package/content/.morph/features/.gitkeep +25 -25
- package/content/.morph/hooks/README.md +158 -0
- package/content/.morph/hooks/pre-commit-all.sh +48 -48
- package/content/.morph/hooks/pre-commit-specs.sh +49 -49
- package/content/.morph/hooks/pre-commit-tests.sh +60 -60
- package/content/.morph/hooks/task-completed.js +73 -0
- package/content/.morph/hooks/teammate-idle.js +68 -0
- package/content/.morph/project.md +160 -160
- package/content/.morph/schemas/agent.schema.json +296 -296
- package/content/.morph/schemas/tasks.schema.json +220 -220
- package/content/.morph/specs/.gitkeep +20 -20
- package/content/.morph/standards/agent-teams-workflow.md +474 -0
- package/content/.morph/standards/coding.md +377 -377
- package/content/.morph/standards/fluent-ui-setup.md +590 -590
- package/content/.morph/standards/migration-guide.md +514 -514
- package/content/.morph/standards/passkeys-auth.md +423 -423
- package/content/.morph/standards/vector-search-rag.md +536 -536
- package/content/.morph/state.json +17 -17
- package/content/.morph/templates/CONTEXT-FEATURE.md +276 -0
- package/content/.morph/templates/CONTEXT.md +170 -0
- package/content/.morph/templates/FluentDesignTheme.cs +149 -149
- package/content/.morph/templates/MudTheme.cs +281 -281
- package/content/.morph/templates/clarify-questions.md +159 -159
- package/content/.morph/templates/component.razor +239 -239
- package/content/.morph/templates/contracts/Commands.cs +74 -74
- package/content/.morph/templates/contracts/Entities.cs +25 -25
- package/content/.morph/templates/contracts/Queries.cs +74 -74
- package/content/.morph/templates/contracts/README.md +74 -74
- package/content/.morph/templates/contracts.cs +217 -217
- package/content/.morph/templates/design-system.css +226 -226
- package/content/.morph/templates/infra/.dockerignore.example +89 -89
- package/content/.morph/templates/infra/Dockerfile.example +82 -82
- package/content/.morph/templates/infra/README.md +286 -286
- package/content/.morph/templates/infra/app-insights.bicep +63 -63
- package/content/.morph/templates/infra/app-service.bicep +164 -164
- package/content/.morph/templates/infra/azure-pipelines-deploy.yml +480 -0
- package/content/.morph/templates/infra/container-app-env.bicep +49 -49
- package/content/.morph/templates/infra/container-app.bicep +156 -156
- package/content/.morph/templates/infra/deploy-checklist.md +426 -426
- package/content/.morph/templates/infra/deploy.ps1 +229 -229
- package/content/.morph/templates/infra/deploy.sh +208 -208
- package/content/.morph/templates/infra/key-vault.bicep +91 -91
- package/content/.morph/templates/infra/main.bicep +189 -189
- package/content/.morph/templates/infra/parameters.dev.json +29 -29
- package/content/.morph/templates/infra/parameters.prod.json +29 -29
- package/content/.morph/templates/infra/parameters.staging.json +29 -29
- package/content/.morph/templates/infra/sql-database.bicep +103 -103
- package/content/.morph/templates/infra/storage.bicep +106 -106
- package/content/.morph/templates/integrations/asaas-client.cs +387 -387
- package/content/.morph/templates/integrations/asaas-webhook.cs +351 -351
- package/content/.morph/templates/integrations/azure-identity-config.cs +288 -288
- package/content/.morph/templates/integrations/clerk-config.cs +258 -258
- package/content/.morph/templates/job.cs +171 -171
- package/content/.morph/templates/migration.cs +83 -83
- package/content/.morph/templates/repository.cs +141 -141
- package/content/.morph/templates/saas/subscription.cs +347 -347
- package/content/.morph/templates/saas/tenant.cs +338 -338
- package/content/.morph/templates/service.cs +139 -139
- package/content/.morph/templates/sprint-status.yaml +68 -68
- package/content/.morph/templates/story.md +143 -143
- package/content/.morph/templates/test.cs +239 -239
- package/content/.morph/templates/ui-design-system.md +286 -286
- package/content/.morph/templates/ui-flows.md +336 -336
- package/content/.morph/templates/ui-mockups.md +133 -133
- package/content/.morph/test-infra/example.bicep +59 -59
- package/content/README.md +79 -79
- package/detectors/config-detector.js +223 -223
- package/detectors/conversation-analyzer.js +163 -163
- package/detectors/index.js +84 -84
- package/detectors/standards-generator.js +275 -275
- package/docs/api/fonts/Source-Sans-Pro/sourcesanspro-light-webfont.svg +977 -977
- package/docs/api/fonts/Source-Sans-Pro/sourcesanspro-regular-webfont.svg +1048 -1048
- package/docs/api/scripts/collapse.js +38 -38
- package/docs/api/scripts/commonNav.js +28 -28
- package/docs/api/scripts/linenumber.js +25 -25
- package/docs/api/scripts/nav.js +12 -12
- package/docs/api/scripts/polyfill.js +3 -3
- package/docs/api/scripts/prettify/Apache-License-2.0.txt +202 -202
- package/docs/api/scripts/prettify/lang-css.js +2 -2
- package/docs/api/scripts/prettify/prettify.js +28 -28
- package/docs/api/scripts/search.js +98 -98
- package/docs/api/styles/jsdoc.css +776 -776
- package/docs/api/styles/prettify.css +80 -80
- package/docs/examples.md +328 -328
- package/docs/templates.md +418 -418
- package/package.json +1 -1
- package/scripts/postinstall.js +132 -132
- package/src/commands/advance-phase.js +83 -0
- package/src/commands/analyze-blazor-concurrency.js +193 -193
- package/src/commands/create-story.js +351 -351
- package/src/commands/deploy.js +780 -0
- package/src/commands/detect-agents.js +34 -6
- package/src/commands/detect.js +104 -104
- package/src/commands/generate-context.js +40 -0
- package/src/commands/generate.js +149 -149
- package/src/commands/lint-fluent.js +352 -352
- package/src/commands/rollback-phase.js +185 -185
- package/src/commands/session-summary.js +291 -291
- package/src/commands/shard-spec.js +224 -224
- package/src/commands/sprint-status.js +250 -250
- package/src/commands/state.js +333 -333
- package/src/commands/sync.js +167 -167
- package/src/commands/troubleshoot.js +222 -222
- package/src/commands/validate-blazor-state.js +210 -210
- package/src/commands/validate-blazor.js +156 -156
- package/src/commands/validate-css.js +84 -84
- package/src/commands/validate-phase.js +221 -221
- package/src/lib/blazor-concurrency-analyzer.js +288 -288
- package/src/lib/blazor-state-validator.js +291 -291
- package/src/lib/blazor-validator.js +374 -374
- package/src/lib/context-generator.js +513 -0
- package/src/lib/css-validator.js +352 -352
- package/src/lib/design-system-detector.js +187 -0
- package/src/lib/design-system-generator.js +298 -298
- package/src/lib/design-system-scaffolder.js +299 -0
- package/src/lib/hook-executor.js +256 -0
- package/src/lib/learning-system.js +520 -520
- package/src/lib/mockup-generator.js +366 -366
- package/src/lib/spec-validator.js +258 -0
- package/src/lib/standards-context-injector.js +287 -0
- package/src/lib/team-orchestrator.js +322 -0
- package/src/lib/troubleshoot-grep.js +194 -194
- package/src/lib/troubleshoot-index.js +144 -144
- package/src/lib/ui-detector.js +350 -350
- package/src/lib/validation-runner.js +65 -13
- package/src/lib/validators/architecture-validator.js +387 -387
- package/src/lib/validators/design-system-validator.js +231 -0
- package/src/lib/validators/package-validator.js +360 -360
- package/src/lib/validators/ui-contrast-validator.js +422 -422
- package/src/utils/file-copier.js +9 -1
- package/src/utils/logger.js +32 -32
- package/src/utils/version-checker.js +175 -175
- /package/content/.claude/skills/{checklists → level-0-meta}/code-review.md +0 -0
- /package/content/.claude/skills/{checklists → level-0-meta}/simulation-checklist.md +0 -0
- /package/content/.claude/skills/{specialists → level-2-domains/ai-agents}/ai-system-architect.md +0 -0
- /package/content/.claude/skills/{specialists → level-2-domains/architecture}/po-pm-advisor.md +0 -0
- /package/content/.claude/skills/{specialists → level-2-domains/architecture}/standards-architect.md +0 -0
- /package/content/.claude/skills/{specialists → level-2-domains/backend}/dotnet-senior.md +0 -0
- /package/content/.claude/skills/{specialists → level-2-domains/backend}/ef-modeler.md +0 -0
- /package/content/.claude/skills/{specialists → level-2-domains/backend}/hangfire-orchestrator.md +0 -0
- /package/content/.claude/skills/{specialists → level-2-domains/backend}/ms-agent-expert.md +0 -0
- /package/content/.claude/skills/{stacks/dotnet-blazor.md → level-2-domains/frontend/blazor-builder.md} +0 -0
- /package/content/.claude/skills/{stacks/dotnet-nextjs.md → level-2-domains/frontend/nextjs-expert.md} +0 -0
- /package/content/.claude/skills/{specialists → level-2-domains/frontend}/ui-ux-designer.md +0 -0
- /package/content/.claude/skills/{specialists → level-2-domains/infrastructure}/azure-architect.md +0 -0
- /package/content/.claude/skills/{infra → level-2-domains/infrastructure}/bicep-architect.md +0 -0
- /package/content/.claude/skills/{infra → level-2-domains/infrastructure}/container-specialist.md +0 -0
- /package/content/.claude/skills/{infra → level-2-domains/infrastructure}/devops-engineer.md +0 -0
- /package/content/.claude/skills/{integrations → level-2-domains/integrations}/asaas-financial.md +0 -0
- /package/content/.claude/skills/{integrations → level-2-domains/integrations}/azure-identity.md +0 -0
- /package/content/.claude/skills/{integrations → level-2-domains/integrations}/clerk-auth.md +0 -0
- /package/content/.claude/skills/{integrations → level-2-domains/integrations}/resend-email.md +0 -0
- /package/content/.claude/skills/{specialists → level-2-domains/quality}/code-analyzer.md +0 -0
|
@@ -1,258 +1,258 @@
|
|
|
1
|
-
// ==============================================================================
|
|
2
|
-
// MORPH-SPEC - Clerk Authentication Configuration Template
|
|
3
|
-
// Configuração de autenticação com Clerk para .NET
|
|
4
|
-
// ==============================================================================
|
|
5
|
-
|
|
6
|
-
using Microsoft.AspNetCore.Authentication;
|
|
7
|
-
using Microsoft.AspNetCore.Components.Authorization;
|
|
8
|
-
using System.Security.Claims;
|
|
9
|
-
|
|
10
|
-
namespace {{Namespace}}.Infrastructure.Auth;
|
|
11
|
-
|
|
12
|
-
// ==============================================================================
|
|
13
|
-
// OPTIONS
|
|
14
|
-
// ==============================================================================
|
|
15
|
-
|
|
16
|
-
public class ClerkOptions
|
|
17
|
-
{
|
|
18
|
-
public const string SectionName = "Clerk";
|
|
19
|
-
|
|
20
|
-
/// <summary>
|
|
21
|
-
/// Clerk Secret Key (starts with sk_)
|
|
22
|
-
/// </summary>
|
|
23
|
-
public string SecretKey { get; set; } = string.Empty;
|
|
24
|
-
|
|
25
|
-
/// <summary>
|
|
26
|
-
/// Clerk Publishable Key (starts with pk_)
|
|
27
|
-
/// </summary>
|
|
28
|
-
public string PublishableKey { get; set; } = string.Empty;
|
|
29
|
-
|
|
30
|
-
/// <summary>
|
|
31
|
-
/// Clerk Frontend API URL
|
|
32
|
-
/// </summary>
|
|
33
|
-
public string FrontendApi { get; set; } = string.Empty;
|
|
34
|
-
}
|
|
35
|
-
|
|
36
|
-
// ==============================================================================
|
|
37
|
-
// SERVICE EXTENSIONS
|
|
38
|
-
// ==============================================================================
|
|
39
|
-
|
|
40
|
-
public static class ClerkServiceExtensions
|
|
41
|
-
{
|
|
42
|
-
/// <summary>
|
|
43
|
-
/// Adiciona autenticação Clerk ao projeto
|
|
44
|
-
/// Requer: Clerk.Net.AspNetCore.Security
|
|
45
|
-
/// </summary>
|
|
46
|
-
public static IServiceCollection AddClerkAuthentication(
|
|
47
|
-
this IServiceCollection services,
|
|
48
|
-
IConfiguration configuration)
|
|
49
|
-
{
|
|
50
|
-
services.Configure<ClerkOptions>(configuration.GetSection(ClerkOptions.SectionName));
|
|
51
|
-
|
|
52
|
-
// Opção 1: Usando Clerk.Net SDK oficial
|
|
53
|
-
// services.AddClerk(configuration);
|
|
54
|
-
|
|
55
|
-
// Opção 2: Configuração manual com JWT
|
|
56
|
-
services.AddAuthentication("Clerk")
|
|
57
|
-
.AddJwtBearer("Clerk", options =>
|
|
58
|
-
{
|
|
59
|
-
var clerkOptions = configuration.GetSection(ClerkOptions.SectionName).Get<ClerkOptions>()!;
|
|
60
|
-
|
|
61
|
-
options.Authority = clerkOptions.FrontendApi;
|
|
62
|
-
options.TokenValidationParameters = new()
|
|
63
|
-
{
|
|
64
|
-
ValidateIssuer = true,
|
|
65
|
-
ValidateAudience = false,
|
|
66
|
-
ValidateLifetime = true,
|
|
67
|
-
ValidateIssuerSigningKey = true,
|
|
68
|
-
NameClaimType = ClaimTypes.NameIdentifier
|
|
69
|
-
};
|
|
70
|
-
});
|
|
71
|
-
|
|
72
|
-
services.AddAuthorization();
|
|
73
|
-
|
|
74
|
-
return services;
|
|
75
|
-
}
|
|
76
|
-
|
|
77
|
-
/// <summary>
|
|
78
|
-
/// Adiciona Clerk para Blazor Server
|
|
79
|
-
/// </summary>
|
|
80
|
-
public static IServiceCollection AddClerkBlazor(
|
|
81
|
-
this IServiceCollection services,
|
|
82
|
-
IConfiguration configuration)
|
|
83
|
-
{
|
|
84
|
-
services.AddClerkAuthentication(configuration);
|
|
85
|
-
services.AddScoped<AuthenticationStateProvider, ClerkAuthenticationStateProvider>();
|
|
86
|
-
services.AddCascadingAuthenticationState();
|
|
87
|
-
|
|
88
|
-
return services;
|
|
89
|
-
}
|
|
90
|
-
}
|
|
91
|
-
|
|
92
|
-
// ==============================================================================
|
|
93
|
-
// AUTHENTICATION STATE PROVIDER (Blazor)
|
|
94
|
-
// ==============================================================================
|
|
95
|
-
|
|
96
|
-
public class ClerkAuthenticationStateProvider : AuthenticationStateProvider
|
|
97
|
-
{
|
|
98
|
-
private readonly IHttpContextAccessor _httpContextAccessor;
|
|
99
|
-
|
|
100
|
-
public ClerkAuthenticationStateProvider(IHttpContextAccessor httpContextAccessor)
|
|
101
|
-
{
|
|
102
|
-
_httpContextAccessor = httpContextAccessor;
|
|
103
|
-
}
|
|
104
|
-
|
|
105
|
-
public override Task<AuthenticationState> GetAuthenticationStateAsync()
|
|
106
|
-
{
|
|
107
|
-
var user = _httpContextAccessor.HttpContext?.User ?? new ClaimsPrincipal();
|
|
108
|
-
return Task.FromResult(new AuthenticationState(user));
|
|
109
|
-
}
|
|
110
|
-
|
|
111
|
-
public void NotifyAuthenticationStateChanged()
|
|
112
|
-
{
|
|
113
|
-
NotifyAuthenticationStateChanged(GetAuthenticationStateAsync());
|
|
114
|
-
}
|
|
115
|
-
}
|
|
116
|
-
|
|
117
|
-
// ==============================================================================
|
|
118
|
-
// CLERK USER SERVICE
|
|
119
|
-
// ==============================================================================
|
|
120
|
-
|
|
121
|
-
public interface IClerkUserService
|
|
122
|
-
{
|
|
123
|
-
Task<ClerkUser?> GetCurrentUserAsync(CancellationToken ct = default);
|
|
124
|
-
Task<ClerkUser?> GetUserByIdAsync(string userId, CancellationToken ct = default);
|
|
125
|
-
Task UpdateUserMetadataAsync(string userId, Dictionary<string, object> metadata, CancellationToken ct = default);
|
|
126
|
-
}
|
|
127
|
-
|
|
128
|
-
public class ClerkUserService : IClerkUserService
|
|
129
|
-
{
|
|
130
|
-
private readonly HttpClient _httpClient;
|
|
131
|
-
private readonly IHttpContextAccessor _httpContextAccessor;
|
|
132
|
-
private readonly ILogger<ClerkUserService> _logger;
|
|
133
|
-
|
|
134
|
-
public ClerkUserService(
|
|
135
|
-
HttpClient httpClient,
|
|
136
|
-
IHttpContextAccessor httpContextAccessor,
|
|
137
|
-
ILogger<ClerkUserService> logger)
|
|
138
|
-
{
|
|
139
|
-
_httpClient = httpClient;
|
|
140
|
-
_httpContextAccessor = httpContextAccessor;
|
|
141
|
-
_logger = logger;
|
|
142
|
-
}
|
|
143
|
-
|
|
144
|
-
public async Task<ClerkUser?> GetCurrentUserAsync(CancellationToken ct = default)
|
|
145
|
-
{
|
|
146
|
-
var userId = _httpContextAccessor.HttpContext?.User.FindFirstValue(ClaimTypes.NameIdentifier);
|
|
147
|
-
if (string.IsNullOrEmpty(userId)) return null;
|
|
148
|
-
|
|
149
|
-
return await GetUserByIdAsync(userId, ct);
|
|
150
|
-
}
|
|
151
|
-
|
|
152
|
-
public async Task<ClerkUser?> GetUserByIdAsync(string userId, CancellationToken ct = default)
|
|
153
|
-
{
|
|
154
|
-
try
|
|
155
|
-
{
|
|
156
|
-
var response = await _httpClient.GetAsync($"users/{userId}", ct);
|
|
157
|
-
|
|
158
|
-
if (!response.IsSuccessStatusCode)
|
|
159
|
-
{
|
|
160
|
-
_logger.LogWarning("Failed to get Clerk user {UserId}: {Status}", userId, response.StatusCode);
|
|
161
|
-
return null;
|
|
162
|
-
}
|
|
163
|
-
|
|
164
|
-
return await response.Content.ReadFromJsonAsync<ClerkUser>(ct);
|
|
165
|
-
}
|
|
166
|
-
catch (Exception ex)
|
|
167
|
-
{
|
|
168
|
-
_logger.LogError(ex, "Error getting Clerk user {UserId}", userId);
|
|
169
|
-
return null;
|
|
170
|
-
}
|
|
171
|
-
}
|
|
172
|
-
|
|
173
|
-
public async Task UpdateUserMetadataAsync(string userId, Dictionary<string, object> metadata, CancellationToken ct = default)
|
|
174
|
-
{
|
|
175
|
-
var request = new { public_metadata = metadata };
|
|
176
|
-
var response = await _httpClient.PatchAsJsonAsync($"users/{userId}", request, ct);
|
|
177
|
-
response.EnsureSuccessStatusCode();
|
|
178
|
-
}
|
|
179
|
-
}
|
|
180
|
-
|
|
181
|
-
// ==============================================================================
|
|
182
|
-
// CLERK USER MODEL
|
|
183
|
-
// ==============================================================================
|
|
184
|
-
|
|
185
|
-
public record ClerkUser
|
|
186
|
-
{
|
|
187
|
-
public string Id { get; init; } = string.Empty;
|
|
188
|
-
public string? FirstName { get; init; }
|
|
189
|
-
public string? LastName { get; init; }
|
|
190
|
-
public string? ImageUrl { get; init; }
|
|
191
|
-
public List<ClerkEmailAddress>? EmailAddresses { get; init; }
|
|
192
|
-
public string? PrimaryEmailAddressId { get; init; }
|
|
193
|
-
public Dictionary<string, object>? PublicMetadata { get; init; }
|
|
194
|
-
public long CreatedAt { get; init; }
|
|
195
|
-
public long UpdatedAt { get; init; }
|
|
196
|
-
|
|
197
|
-
public string? PrimaryEmail => EmailAddresses?
|
|
198
|
-
.FirstOrDefault(e => e.Id == PrimaryEmailAddressId)?.EmailAddress;
|
|
199
|
-
|
|
200
|
-
public string FullName => $"{FirstName} {LastName}".Trim();
|
|
201
|
-
}
|
|
202
|
-
|
|
203
|
-
public record ClerkEmailAddress
|
|
204
|
-
{
|
|
205
|
-
public string Id { get; init; } = string.Empty;
|
|
206
|
-
public string EmailAddress { get; init; } = string.Empty;
|
|
207
|
-
public bool Verified { get; init; }
|
|
208
|
-
}
|
|
209
|
-
|
|
210
|
-
// ==============================================================================
|
|
211
|
-
// MIDDLEWARE PIPELINE
|
|
212
|
-
// ==============================================================================
|
|
213
|
-
|
|
214
|
-
public static class ClerkMiddlewareExtensions
|
|
215
|
-
{
|
|
216
|
-
public static IApplicationBuilder UseClerkAuthentication(this IApplicationBuilder app)
|
|
217
|
-
{
|
|
218
|
-
app.UseAuthentication();
|
|
219
|
-
app.UseAuthorization();
|
|
220
|
-
return app;
|
|
221
|
-
}
|
|
222
|
-
}
|
|
223
|
-
|
|
224
|
-
// ==============================================================================
|
|
225
|
-
// APPSETTINGS EXAMPLE
|
|
226
|
-
// ==============================================================================
|
|
227
|
-
|
|
228
|
-
/*
|
|
229
|
-
{
|
|
230
|
-
"Clerk": {
|
|
231
|
-
"SecretKey": "sk_test_xxxxx",
|
|
232
|
-
"PublishableKey": "pk_test_xxxxx",
|
|
233
|
-
"FrontendApi": "https://your-app.clerk.accounts.dev"
|
|
234
|
-
}
|
|
235
|
-
}
|
|
236
|
-
*/
|
|
237
|
-
|
|
238
|
-
// ==============================================================================
|
|
239
|
-
// PROGRAM.CS EXAMPLE
|
|
240
|
-
// ==============================================================================
|
|
241
|
-
|
|
242
|
-
/*
|
|
243
|
-
// Program.cs
|
|
244
|
-
var builder = WebApplication.CreateBuilder(args);
|
|
245
|
-
|
|
246
|
-
// Add Clerk authentication
|
|
247
|
-
builder.Services.AddClerkAuthentication(builder.Configuration);
|
|
248
|
-
// Or for Blazor:
|
|
249
|
-
// builder.Services.AddClerkBlazor(builder.Configuration);
|
|
250
|
-
|
|
251
|
-
var app = builder.Build();
|
|
252
|
-
|
|
253
|
-
app.UseClerkAuthentication();
|
|
254
|
-
|
|
255
|
-
app.MapControllers().RequireAuthorization();
|
|
256
|
-
|
|
257
|
-
app.Run();
|
|
258
|
-
*/
|
|
1
|
+
// ==============================================================================
|
|
2
|
+
// MORPH-SPEC - Clerk Authentication Configuration Template
|
|
3
|
+
// Configuração de autenticação com Clerk para .NET
|
|
4
|
+
// ==============================================================================
|
|
5
|
+
|
|
6
|
+
using Microsoft.AspNetCore.Authentication;
|
|
7
|
+
using Microsoft.AspNetCore.Components.Authorization;
|
|
8
|
+
using System.Security.Claims;
|
|
9
|
+
|
|
10
|
+
namespace {{Namespace}}.Infrastructure.Auth;
|
|
11
|
+
|
|
12
|
+
// ==============================================================================
|
|
13
|
+
// OPTIONS
|
|
14
|
+
// ==============================================================================
|
|
15
|
+
|
|
16
|
+
public class ClerkOptions
|
|
17
|
+
{
|
|
18
|
+
public const string SectionName = "Clerk";
|
|
19
|
+
|
|
20
|
+
/// <summary>
|
|
21
|
+
/// Clerk Secret Key (starts with sk_)
|
|
22
|
+
/// </summary>
|
|
23
|
+
public string SecretKey { get; set; } = string.Empty;
|
|
24
|
+
|
|
25
|
+
/// <summary>
|
|
26
|
+
/// Clerk Publishable Key (starts with pk_)
|
|
27
|
+
/// </summary>
|
|
28
|
+
public string PublishableKey { get; set; } = string.Empty;
|
|
29
|
+
|
|
30
|
+
/// <summary>
|
|
31
|
+
/// Clerk Frontend API URL
|
|
32
|
+
/// </summary>
|
|
33
|
+
public string FrontendApi { get; set; } = string.Empty;
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
// ==============================================================================
|
|
37
|
+
// SERVICE EXTENSIONS
|
|
38
|
+
// ==============================================================================
|
|
39
|
+
|
|
40
|
+
public static class ClerkServiceExtensions
|
|
41
|
+
{
|
|
42
|
+
/// <summary>
|
|
43
|
+
/// Adiciona autenticação Clerk ao projeto
|
|
44
|
+
/// Requer: Clerk.Net.AspNetCore.Security
|
|
45
|
+
/// </summary>
|
|
46
|
+
public static IServiceCollection AddClerkAuthentication(
|
|
47
|
+
this IServiceCollection services,
|
|
48
|
+
IConfiguration configuration)
|
|
49
|
+
{
|
|
50
|
+
services.Configure<ClerkOptions>(configuration.GetSection(ClerkOptions.SectionName));
|
|
51
|
+
|
|
52
|
+
// Opção 1: Usando Clerk.Net SDK oficial
|
|
53
|
+
// services.AddClerk(configuration);
|
|
54
|
+
|
|
55
|
+
// Opção 2: Configuração manual com JWT
|
|
56
|
+
services.AddAuthentication("Clerk")
|
|
57
|
+
.AddJwtBearer("Clerk", options =>
|
|
58
|
+
{
|
|
59
|
+
var clerkOptions = configuration.GetSection(ClerkOptions.SectionName).Get<ClerkOptions>()!;
|
|
60
|
+
|
|
61
|
+
options.Authority = clerkOptions.FrontendApi;
|
|
62
|
+
options.TokenValidationParameters = new()
|
|
63
|
+
{
|
|
64
|
+
ValidateIssuer = true,
|
|
65
|
+
ValidateAudience = false,
|
|
66
|
+
ValidateLifetime = true,
|
|
67
|
+
ValidateIssuerSigningKey = true,
|
|
68
|
+
NameClaimType = ClaimTypes.NameIdentifier
|
|
69
|
+
};
|
|
70
|
+
});
|
|
71
|
+
|
|
72
|
+
services.AddAuthorization();
|
|
73
|
+
|
|
74
|
+
return services;
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
/// <summary>
|
|
78
|
+
/// Adiciona Clerk para Blazor Server
|
|
79
|
+
/// </summary>
|
|
80
|
+
public static IServiceCollection AddClerkBlazor(
|
|
81
|
+
this IServiceCollection services,
|
|
82
|
+
IConfiguration configuration)
|
|
83
|
+
{
|
|
84
|
+
services.AddClerkAuthentication(configuration);
|
|
85
|
+
services.AddScoped<AuthenticationStateProvider, ClerkAuthenticationStateProvider>();
|
|
86
|
+
services.AddCascadingAuthenticationState();
|
|
87
|
+
|
|
88
|
+
return services;
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
// ==============================================================================
|
|
93
|
+
// AUTHENTICATION STATE PROVIDER (Blazor)
|
|
94
|
+
// ==============================================================================
|
|
95
|
+
|
|
96
|
+
public class ClerkAuthenticationStateProvider : AuthenticationStateProvider
|
|
97
|
+
{
|
|
98
|
+
private readonly IHttpContextAccessor _httpContextAccessor;
|
|
99
|
+
|
|
100
|
+
public ClerkAuthenticationStateProvider(IHttpContextAccessor httpContextAccessor)
|
|
101
|
+
{
|
|
102
|
+
_httpContextAccessor = httpContextAccessor;
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
public override Task<AuthenticationState> GetAuthenticationStateAsync()
|
|
106
|
+
{
|
|
107
|
+
var user = _httpContextAccessor.HttpContext?.User ?? new ClaimsPrincipal();
|
|
108
|
+
return Task.FromResult(new AuthenticationState(user));
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
public void NotifyAuthenticationStateChanged()
|
|
112
|
+
{
|
|
113
|
+
NotifyAuthenticationStateChanged(GetAuthenticationStateAsync());
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
|
|
117
|
+
// ==============================================================================
|
|
118
|
+
// CLERK USER SERVICE
|
|
119
|
+
// ==============================================================================
|
|
120
|
+
|
|
121
|
+
public interface IClerkUserService
|
|
122
|
+
{
|
|
123
|
+
Task<ClerkUser?> GetCurrentUserAsync(CancellationToken ct = default);
|
|
124
|
+
Task<ClerkUser?> GetUserByIdAsync(string userId, CancellationToken ct = default);
|
|
125
|
+
Task UpdateUserMetadataAsync(string userId, Dictionary<string, object> metadata, CancellationToken ct = default);
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
public class ClerkUserService : IClerkUserService
|
|
129
|
+
{
|
|
130
|
+
private readonly HttpClient _httpClient;
|
|
131
|
+
private readonly IHttpContextAccessor _httpContextAccessor;
|
|
132
|
+
private readonly ILogger<ClerkUserService> _logger;
|
|
133
|
+
|
|
134
|
+
public ClerkUserService(
|
|
135
|
+
HttpClient httpClient,
|
|
136
|
+
IHttpContextAccessor httpContextAccessor,
|
|
137
|
+
ILogger<ClerkUserService> logger)
|
|
138
|
+
{
|
|
139
|
+
_httpClient = httpClient;
|
|
140
|
+
_httpContextAccessor = httpContextAccessor;
|
|
141
|
+
_logger = logger;
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
public async Task<ClerkUser?> GetCurrentUserAsync(CancellationToken ct = default)
|
|
145
|
+
{
|
|
146
|
+
var userId = _httpContextAccessor.HttpContext?.User.FindFirstValue(ClaimTypes.NameIdentifier);
|
|
147
|
+
if (string.IsNullOrEmpty(userId)) return null;
|
|
148
|
+
|
|
149
|
+
return await GetUserByIdAsync(userId, ct);
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
public async Task<ClerkUser?> GetUserByIdAsync(string userId, CancellationToken ct = default)
|
|
153
|
+
{
|
|
154
|
+
try
|
|
155
|
+
{
|
|
156
|
+
var response = await _httpClient.GetAsync($"users/{userId}", ct);
|
|
157
|
+
|
|
158
|
+
if (!response.IsSuccessStatusCode)
|
|
159
|
+
{
|
|
160
|
+
_logger.LogWarning("Failed to get Clerk user {UserId}: {Status}", userId, response.StatusCode);
|
|
161
|
+
return null;
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
return await response.Content.ReadFromJsonAsync<ClerkUser>(ct);
|
|
165
|
+
}
|
|
166
|
+
catch (Exception ex)
|
|
167
|
+
{
|
|
168
|
+
_logger.LogError(ex, "Error getting Clerk user {UserId}", userId);
|
|
169
|
+
return null;
|
|
170
|
+
}
|
|
171
|
+
}
|
|
172
|
+
|
|
173
|
+
public async Task UpdateUserMetadataAsync(string userId, Dictionary<string, object> metadata, CancellationToken ct = default)
|
|
174
|
+
{
|
|
175
|
+
var request = new { public_metadata = metadata };
|
|
176
|
+
var response = await _httpClient.PatchAsJsonAsync($"users/{userId}", request, ct);
|
|
177
|
+
response.EnsureSuccessStatusCode();
|
|
178
|
+
}
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
// ==============================================================================
|
|
182
|
+
// CLERK USER MODEL
|
|
183
|
+
// ==============================================================================
|
|
184
|
+
|
|
185
|
+
public record ClerkUser
|
|
186
|
+
{
|
|
187
|
+
public string Id { get; init; } = string.Empty;
|
|
188
|
+
public string? FirstName { get; init; }
|
|
189
|
+
public string? LastName { get; init; }
|
|
190
|
+
public string? ImageUrl { get; init; }
|
|
191
|
+
public List<ClerkEmailAddress>? EmailAddresses { get; init; }
|
|
192
|
+
public string? PrimaryEmailAddressId { get; init; }
|
|
193
|
+
public Dictionary<string, object>? PublicMetadata { get; init; }
|
|
194
|
+
public long CreatedAt { get; init; }
|
|
195
|
+
public long UpdatedAt { get; init; }
|
|
196
|
+
|
|
197
|
+
public string? PrimaryEmail => EmailAddresses?
|
|
198
|
+
.FirstOrDefault(e => e.Id == PrimaryEmailAddressId)?.EmailAddress;
|
|
199
|
+
|
|
200
|
+
public string FullName => $"{FirstName} {LastName}".Trim();
|
|
201
|
+
}
|
|
202
|
+
|
|
203
|
+
public record ClerkEmailAddress
|
|
204
|
+
{
|
|
205
|
+
public string Id { get; init; } = string.Empty;
|
|
206
|
+
public string EmailAddress { get; init; } = string.Empty;
|
|
207
|
+
public bool Verified { get; init; }
|
|
208
|
+
}
|
|
209
|
+
|
|
210
|
+
// ==============================================================================
|
|
211
|
+
// MIDDLEWARE PIPELINE
|
|
212
|
+
// ==============================================================================
|
|
213
|
+
|
|
214
|
+
public static class ClerkMiddlewareExtensions
|
|
215
|
+
{
|
|
216
|
+
public static IApplicationBuilder UseClerkAuthentication(this IApplicationBuilder app)
|
|
217
|
+
{
|
|
218
|
+
app.UseAuthentication();
|
|
219
|
+
app.UseAuthorization();
|
|
220
|
+
return app;
|
|
221
|
+
}
|
|
222
|
+
}
|
|
223
|
+
|
|
224
|
+
// ==============================================================================
|
|
225
|
+
// APPSETTINGS EXAMPLE
|
|
226
|
+
// ==============================================================================
|
|
227
|
+
|
|
228
|
+
/*
|
|
229
|
+
{
|
|
230
|
+
"Clerk": {
|
|
231
|
+
"SecretKey": "sk_test_xxxxx",
|
|
232
|
+
"PublishableKey": "pk_test_xxxxx",
|
|
233
|
+
"FrontendApi": "https://your-app.clerk.accounts.dev"
|
|
234
|
+
}
|
|
235
|
+
}
|
|
236
|
+
*/
|
|
237
|
+
|
|
238
|
+
// ==============================================================================
|
|
239
|
+
// PROGRAM.CS EXAMPLE
|
|
240
|
+
// ==============================================================================
|
|
241
|
+
|
|
242
|
+
/*
|
|
243
|
+
// Program.cs
|
|
244
|
+
var builder = WebApplication.CreateBuilder(args);
|
|
245
|
+
|
|
246
|
+
// Add Clerk authentication
|
|
247
|
+
builder.Services.AddClerkAuthentication(builder.Configuration);
|
|
248
|
+
// Or for Blazor:
|
|
249
|
+
// builder.Services.AddClerkBlazor(builder.Configuration);
|
|
250
|
+
|
|
251
|
+
var app = builder.Build();
|
|
252
|
+
|
|
253
|
+
app.UseClerkAuthentication();
|
|
254
|
+
|
|
255
|
+
app.MapControllers().RequireAuthorization();
|
|
256
|
+
|
|
257
|
+
app.Run();
|
|
258
|
+
*/
|