@polymorphism-tech/morph-spec 2.2.0 → 2.4.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 +314 -1673
- package/LICENSE +72 -72
- package/README.md +515 -516
- package/bin/detect-agents.js +225 -225
- package/bin/morph-spec.js +358 -173
- package/bin/render-template.js +302 -302
- package/bin/semantic-detect-agents.js +246 -246
- package/bin/task-manager.js +429 -0
- package/bin/validate-agents-skills.js +251 -251
- package/bin/validate-agents.js +69 -69
- package/bin/validate-phase.js +263 -263
- package/bin/validate.js +369 -0
- 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-apply.md +221 -158
- package/content/.claude/commands/morph-archive.md +79 -79
- package/content/.claude/commands/morph-infra.md +209 -209
- package/content/.claude/commands/morph-preflight.md +227 -0
- package/content/.claude/commands/morph-proposal.md +122 -101
- package/content/.claude/commands/morph-status.md +86 -86
- package/content/.claude/commands/morph-troubleshoot.md +122 -0
- package/content/.claude/settings.local.json +15 -15
- package/content/.claude/skills/checklists/code-review.md +226 -0
- package/content/.claude/skills/checklists/morph-checklist.md +117 -0
- package/content/.claude/skills/checklists/simulation-checklist.md +77 -0
- package/content/.claude/skills/infra/bicep-architect.md +126 -419
- package/content/.claude/skills/infra/container-specialist.md +131 -437
- package/content/.claude/skills/infra/devops-engineer.md +119 -405
- package/content/.claude/skills/integrations/asaas-financial.md +130 -333
- package/content/.claude/skills/integrations/azure-identity.md +142 -309
- package/content/.claude/skills/integrations/clerk-auth.md +108 -290
- package/content/.claude/skills/integrations/resend-email.md +119 -0
- package/content/.claude/skills/specialists/ai-system-architect.md +192 -604
- package/content/.claude/skills/specialists/azure-architect.md +142 -142
- package/content/.claude/skills/specialists/code-analyzer.md +235 -0
- package/content/.claude/skills/specialists/dotnet-senior.md +287 -0
- package/content/.claude/skills/specialists/ef-modeler.md +113 -200
- package/content/.claude/skills/specialists/hangfire-orchestrator.md +126 -245
- package/content/.claude/skills/specialists/ms-agent-expert.md +109 -263
- package/content/.claude/skills/specialists/po-pm-advisor.md +197 -197
- package/content/.claude/skills/specialists/standards-architect.md +156 -78
- package/content/.claude/skills/specialists/testing-specialist.md +126 -0
- package/content/.claude/skills/specialists/ui-ux-designer.md +191 -1060
- package/content/.claude/skills/stacks/dotnet-blazor.md +210 -588
- package/content/.claude/skills/stacks/dotnet-nextjs.md +154 -402
- package/content/.claude/skills/workflows/morph-replicate.md +213 -0
- package/content/.claude/{commands/morph-clarify.md → skills/workflows/phase-clarify.md} +5 -58
- package/content/.claude/{commands/morph-design.md → skills/workflows/phase-design.md} +16 -86
- package/content/.claude/{commands/morph-setup.md → skills/workflows/phase-setup.md} +9 -17
- package/content/.claude/skills/workflows/phase-tasks.md +164 -0
- package/content/.claude/{commands/morph-uiux.md → skills/workflows/phase-uiux.md} +15 -88
- package/content/.morph/.morphversion +5 -5
- package/content/.morph/archive/.gitkeep +25 -25
- package/content/.morph/config/agents.json +378 -242
- package/content/.morph/config/config.template.json +89 -108
- package/content/.morph/docs/STORY-DRIVEN-DEVELOPMENT.md +392 -392
- package/content/.morph/docs/workflows/design-impl.md +37 -0
- package/content/.morph/docs/workflows/fast-track.md +29 -0
- package/content/.morph/docs/workflows/full-morph.md +76 -0
- package/content/.morph/docs/workflows/standard.md +44 -0
- package/content/.morph/docs/workflows/ui-refresh.md +39 -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 -0
- package/content/.morph/examples/scheduled-reports/proposal.md +95 -0
- package/content/.morph/examples/scheduled-reports/spec.md +267 -0
- package/content/.morph/examples/state-v3.json +188 -0
- package/content/.morph/features/.gitkeep +25 -25
- package/content/.morph/hooks/README.md +190 -239
- package/content/.morph/hooks/pre-commit-agents.sh +24 -24
- 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/project.md +160 -160
- package/content/.morph/schemas/agent.schema.json +296 -296
- package/content/.morph/schemas/tasks.schema.json +220 -0
- package/content/.morph/specs/.gitkeep +20 -20
- package/content/.morph/standards/agent-framework-blazor-ui.md +359 -0
- package/content/.morph/standards/agent-framework-production.md +410 -0
- package/content/.morph/standards/agent-framework-setup.md +413 -453
- package/content/.morph/standards/agent-framework-workflows.md +349 -0
- package/content/.morph/standards/architecture.md +325 -325
- package/content/.morph/standards/azure.md +605 -379
- package/content/.morph/standards/coding.md +377 -377
- package/content/.morph/standards/dotnet10-migration.md +520 -494
- 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/FluentDesignTheme.cs +149 -149
- package/content/.morph/templates/MudTheme.cs +281 -281
- package/content/.morph/templates/agent.cs +163 -172
- package/content/.morph/templates/clarify-questions.md +159 -0
- package/content/.morph/templates/component.razor +239 -239
- package/content/.morph/templates/contracts/Commands.cs +74 -0
- package/content/.morph/templates/contracts/Entities.cs +25 -0
- package/content/.morph/templates/contracts/Queries.cs +74 -0
- package/content/.morph/templates/contracts/README.md +74 -0
- package/content/.morph/templates/contracts.cs +217 -217
- package/content/.morph/templates/decisions.md +123 -106
- 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/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 -0
- 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/proposal.md +141 -155
- package/content/.morph/templates/recap.md +94 -105
- 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/simulation.md +353 -0
- package/content/.morph/templates/spec.md +149 -148
- package/content/.morph/templates/sprint-status.yaml +68 -68
- package/content/.morph/templates/state.template.json +222 -222
- package/content/.morph/templates/story.md +143 -143
- package/content/.morph/templates/tasks.md +257 -235
- package/content/.morph/templates/test.cs +239 -239
- package/content/.morph/templates/ui-components.md +362 -276
- 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/CLAUDE.md +150 -442
- 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/detectors/structure-detector.js +245 -250
- package/docs/README.md +144 -149
- 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/getting-started.md +301 -302
- package/docs/installation.md +361 -361
- package/docs/templates.md +418 -418
- package/docs/validation-checklist.md +265 -266
- package/package.json +80 -80
- package/scripts/postinstall.js +132 -132
- package/src/commands/advance-phase.js +183 -0
- package/src/commands/analyze-blazor-concurrency.js +193 -0
- package/src/commands/create-story.js +351 -351
- package/src/commands/detect-agents.js +139 -0
- package/src/commands/detect.js +104 -104
- package/src/commands/doctor.js +356 -280
- package/src/commands/generate.js +149 -149
- package/src/commands/init.js +258 -245
- package/src/commands/lint-fluent.js +352 -0
- package/src/commands/rollback-phase.js +185 -0
- package/src/commands/session-summary.js +291 -0
- 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/task.js +78 -0
- package/src/commands/troubleshoot.js +222 -0
- package/src/commands/update.js +192 -159
- package/src/commands/validate-blazor-state.js +210 -0
- package/src/commands/validate-blazor.js +156 -0
- package/src/commands/validate-css.js +84 -0
- package/src/commands/validate-phase.js +221 -0
- package/src/lib/blazor-concurrency-analyzer.js +288 -0
- package/src/lib/blazor-state-validator.js +291 -0
- package/src/lib/blazor-validator.js +374 -0
- package/src/lib/complexity-analyzer.js +441 -292
- package/src/lib/continuous-validator.js +421 -0
- package/src/lib/css-validator.js +352 -0
- package/src/lib/decision-constraint-loader.js +109 -0
- package/src/lib/design-system-generator.js +298 -298
- package/src/lib/learning-system.js +520 -0
- package/src/lib/mockup-generator.js +366 -0
- package/src/lib/recap-generator.js +205 -0
- package/src/lib/state-manager.js +397 -340
- package/src/lib/troubleshoot-grep.js +194 -0
- package/src/lib/troubleshoot-index.js +144 -0
- package/src/lib/ui-detector.js +350 -0
- package/src/lib/validation-runner.js +231 -0
- package/src/lib/validators/architecture-validator.js +387 -0
- package/src/lib/validators/contract-compliance-validator.js +273 -0
- package/src/lib/validators/package-validator.js +360 -0
- package/src/lib/validators/ui-contrast-validator.js +422 -0
- package/src/utils/file-copier.js +179 -139
- package/src/utils/logger.js +32 -32
- package/src/utils/version-checker.js +175 -175
- package/content/.claude/commands/morph-costs.md +0 -206
- package/content/.claude/commands/morph-tasks.md +0 -319
- package/content/.claude/skills/specialists/cost-guardian.md +0 -110
- package/content/.claude/skills/stacks/shopify.md +0 -445
- package/content/.morph/config/azure-pricing.json +0 -70
- package/content/.morph/config/azure-pricing.schema.json +0 -50
- package/content/.morph/hooks/pre-commit-costs.sh +0 -91
- package/docs/api/cost-calculator.js.html +0 -513
- package/docs/api/design-system-generator.js.html +0 -382
- package/docs/api/global.html +0 -5263
- package/docs/api/index.html +0 -96
- package/docs/api/state-manager.js.html +0 -423
- package/src/commands/cost.js +0 -181
- package/src/commands/update-pricing.js +0 -206
- package/src/lib/cost-calculator.js +0 -429
|
@@ -1,399 +1,399 @@
|
|
|
1
|
-
# API .NET + Next.js - Especificação
|
|
2
|
-
|
|
3
|
-
## 1. Visão Geral
|
|
4
|
-
|
|
5
|
-
Arquitetura full-stack moderna com API .NET e frontend Next.js, seguindo boas práticas de separação de responsabilidades.
|
|
6
|
-
|
|
7
|
-
### 1.1 Objetivos
|
|
8
|
-
|
|
9
|
-
- Backend robusto com .NET Minimal APIs
|
|
10
|
-
- Frontend moderno com Next.js App Router
|
|
11
|
-
- Autenticação unificada via Clerk
|
|
12
|
-
- Type-safety end-to-end
|
|
13
|
-
- Deploy independente de cada camada
|
|
14
|
-
|
|
15
|
-
### 1.2 Use Cases
|
|
16
|
-
|
|
17
|
-
- Dashboards administrativos
|
|
18
|
-
- Aplicações CRUD complexas
|
|
19
|
-
- Sistemas que precisam de SEO (Next.js SSR)
|
|
20
|
-
- Aplicações com múltiplos frontends (web, mobile)
|
|
21
|
-
|
|
22
|
-
## 2. Requisitos Funcionais
|
|
23
|
-
|
|
24
|
-
### 2.1 Backend (.NET API)
|
|
25
|
-
|
|
26
|
-
| ID | Requisito |
|
|
27
|
-
|----|-----------|
|
|
28
|
-
| API-001 | Minimal APIs com OpenAPI/Swagger |
|
|
29
|
-
| API-002 | Autenticação JWT via Clerk |
|
|
30
|
-
| API-003 | Validação com FluentValidation |
|
|
31
|
-
| API-004 | Paginação, filtros e ordenação |
|
|
32
|
-
| API-005 | Rate limiting por usuário |
|
|
33
|
-
| API-006 | Health checks (liveness + readiness) |
|
|
34
|
-
| API-007 | Logging estruturado |
|
|
35
|
-
| API-008 | CORS configurável |
|
|
36
|
-
|
|
37
|
-
### 2.2 Frontend (Next.js)
|
|
38
|
-
|
|
39
|
-
| ID | Requisito |
|
|
40
|
-
|----|-----------|
|
|
41
|
-
| WEB-001 | App Router com layouts aninhados |
|
|
42
|
-
| WEB-002 | Server Components para dados estáticos |
|
|
43
|
-
| WEB-003 | Client Components para interatividade |
|
|
44
|
-
| WEB-004 | React Query para cache e mutations |
|
|
45
|
-
| WEB-005 | Forms com React Hook Form + Zod |
|
|
46
|
-
| WEB-006 | UI com Tailwind + shadcn/ui |
|
|
47
|
-
| WEB-007 | Dark mode support |
|
|
48
|
-
| WEB-008 | Loading states e error boundaries |
|
|
49
|
-
|
|
50
|
-
### 2.3 Autenticação
|
|
51
|
-
|
|
52
|
-
| ID | Requisito |
|
|
53
|
-
|----|-----------|
|
|
54
|
-
| AUTH-001 | SSO via Clerk |
|
|
55
|
-
| AUTH-002 | Proteção de rotas (middleware) |
|
|
56
|
-
| AUTH-003 | Token refresh automático |
|
|
57
|
-
| AUTH-004 | Role-based access control |
|
|
58
|
-
|
|
59
|
-
## 3. Requisitos Não-Funcionais
|
|
60
|
-
|
|
61
|
-
### 3.1 Performance
|
|
62
|
-
|
|
63
|
-
| ID | Requisito |
|
|
64
|
-
|----|-----------|
|
|
65
|
-
| PERF-001 | API response < 100ms (P95) |
|
|
66
|
-
| PERF-002 | LCP < 2.5s (Core Web Vitals) |
|
|
67
|
-
| PERF-003 | FID < 100ms |
|
|
68
|
-
| PERF-004 | CLS < 0.1 |
|
|
69
|
-
|
|
70
|
-
### 3.2 Segurança
|
|
71
|
-
|
|
72
|
-
| ID | Requisito |
|
|
73
|
-
|----|-----------|
|
|
74
|
-
| SEC-001 | HTTPS obrigatório |
|
|
75
|
-
| SEC-002 | CORS restrito a domínios conhecidos |
|
|
76
|
-
| SEC-003 | Input validation em todos endpoints |
|
|
77
|
-
| SEC-004 | SQL injection prevention (EF Core) |
|
|
78
|
-
| SEC-005 | XSS prevention (React default) |
|
|
79
|
-
|
|
80
|
-
## 4. Arquitetura da API
|
|
81
|
-
|
|
82
|
-
### 4.1 Estrutura de Camadas
|
|
83
|
-
|
|
84
|
-
```
|
|
85
|
-
Api/
|
|
86
|
-
├── Endpoints/ # Minimal API endpoints
|
|
87
|
-
├── Middleware/ # Custom middlewares
|
|
88
|
-
├── Filters/ # Exception filters
|
|
89
|
-
└── Program.cs # Entry point
|
|
90
|
-
|
|
91
|
-
Application/
|
|
92
|
-
├── Services/ # Business logic
|
|
93
|
-
├── Validators/ # FluentValidation
|
|
94
|
-
└── DTOs/ # Data transfer objects
|
|
95
|
-
|
|
96
|
-
Domain/
|
|
97
|
-
├── Entities/ # Domain models
|
|
98
|
-
├── Interfaces/ # Repository contracts
|
|
99
|
-
└── Enums/ # Domain enums
|
|
100
|
-
|
|
101
|
-
Infrastructure/
|
|
102
|
-
├── Data/ # DbContext
|
|
103
|
-
├── Repositories/ # EF Core implementations
|
|
104
|
-
└── Services/ # External service clients
|
|
105
|
-
```
|
|
106
|
-
|
|
107
|
-
### 4.2 Endpoint Pattern
|
|
108
|
-
|
|
109
|
-
```csharp
|
|
110
|
-
// Endpoints/ProductEndpoints.cs
|
|
111
|
-
public static class ProductEndpoints
|
|
112
|
-
{
|
|
113
|
-
public static void MapProductEndpoints(this IEndpointRouteBuilder app)
|
|
114
|
-
{
|
|
115
|
-
var group = app.MapGroup("/api/products")
|
|
116
|
-
.WithTags("Products")
|
|
117
|
-
.RequireAuthorization();
|
|
118
|
-
|
|
119
|
-
group.MapGet("/", GetAll);
|
|
120
|
-
group.MapGet("/{id:int}", GetById);
|
|
121
|
-
group.MapPost("/", Create);
|
|
122
|
-
group.MapPut("/{id:int}", Update);
|
|
123
|
-
group.MapDelete("/{id:int}", Delete);
|
|
124
|
-
}
|
|
125
|
-
|
|
126
|
-
private static async Task<IResult> GetAll(
|
|
127
|
-
[AsParameters] PaginationQuery query,
|
|
128
|
-
IProductService service,
|
|
129
|
-
CancellationToken ct)
|
|
130
|
-
{
|
|
131
|
-
var result = await service.GetAllAsync(query, ct);
|
|
132
|
-
return Results.Ok(result);
|
|
133
|
-
}
|
|
134
|
-
}
|
|
135
|
-
```
|
|
136
|
-
|
|
137
|
-
### 4.3 Pagination
|
|
138
|
-
|
|
139
|
-
```csharp
|
|
140
|
-
public record PaginationQuery(
|
|
141
|
-
int Page = 1,
|
|
142
|
-
int PageSize = 20,
|
|
143
|
-
string? SortBy = null,
|
|
144
|
-
bool SortDesc = false,
|
|
145
|
-
string? Search = null
|
|
146
|
-
);
|
|
147
|
-
|
|
148
|
-
public record PagedResult<T>(
|
|
149
|
-
IReadOnlyList<T> Items,
|
|
150
|
-
int TotalItems,
|
|
151
|
-
int Page,
|
|
152
|
-
int PageSize,
|
|
153
|
-
int TotalPages
|
|
154
|
-
);
|
|
155
|
-
```
|
|
156
|
-
|
|
157
|
-
## 5. Arquitetura do Frontend
|
|
158
|
-
|
|
159
|
-
### 5.1 Estrutura de Pastas
|
|
160
|
-
|
|
161
|
-
```
|
|
162
|
-
app/
|
|
163
|
-
├── (auth)/
|
|
164
|
-
│ ├── sign-in/[[...sign-in]]/page.tsx
|
|
165
|
-
│ └── sign-up/[[...sign-up]]/page.tsx
|
|
166
|
-
├── (dashboard)/
|
|
167
|
-
│ ├── layout.tsx # Dashboard layout
|
|
168
|
-
│ ├── dashboard/page.tsx
|
|
169
|
-
│ ├── products/
|
|
170
|
-
│ │ ├── page.tsx # List
|
|
171
|
-
│ │ ├── [id]/page.tsx # Detail
|
|
172
|
-
│ │ └── new/page.tsx # Create
|
|
173
|
-
│ └── orders/
|
|
174
|
-
├── api/ # API routes (BFF)
|
|
175
|
-
├── layout.tsx # Root layout
|
|
176
|
-
└── page.tsx # Landing
|
|
177
|
-
|
|
178
|
-
components/
|
|
179
|
-
├── ui/ # shadcn components
|
|
180
|
-
├── forms/ # Form components
|
|
181
|
-
├── tables/ # Table components
|
|
182
|
-
└── shared/ # Shared components
|
|
183
|
-
|
|
184
|
-
hooks/
|
|
185
|
-
├── use-products.ts
|
|
186
|
-
├── use-orders.ts
|
|
187
|
-
└── use-pagination.ts
|
|
188
|
-
|
|
189
|
-
lib/
|
|
190
|
-
├── api-client.ts # Fetch wrapper
|
|
191
|
-
├── query-client.ts # React Query config
|
|
192
|
-
└── utils.ts # Utility functions
|
|
193
|
-
|
|
194
|
-
types/
|
|
195
|
-
├── api.ts # API response types
|
|
196
|
-
└── entities.ts # Domain types
|
|
197
|
-
```
|
|
198
|
-
|
|
199
|
-
### 5.2 Data Fetching Pattern
|
|
200
|
-
|
|
201
|
-
```typescript
|
|
202
|
-
// hooks/use-products.ts
|
|
203
|
-
import { useQuery, useMutation, useQueryClient } from "@tanstack/react-query";
|
|
204
|
-
import { apiClient } from "@/lib/api-client";
|
|
205
|
-
import type { Product, CreateProductRequest } from "@/types/entities";
|
|
206
|
-
|
|
207
|
-
export function useProducts(params?: PaginationParams) {
|
|
208
|
-
return useQuery({
|
|
209
|
-
queryKey: ["products", params],
|
|
210
|
-
queryFn: () => apiClient<PagedResult<Product>>("/api/products", { params }),
|
|
211
|
-
});
|
|
212
|
-
}
|
|
213
|
-
|
|
214
|
-
export function useProduct(id: number) {
|
|
215
|
-
return useQuery({
|
|
216
|
-
queryKey: ["products", id],
|
|
217
|
-
queryFn: () => apiClient<Product>(`/api/products/${id}`),
|
|
218
|
-
enabled: !!id,
|
|
219
|
-
});
|
|
220
|
-
}
|
|
221
|
-
|
|
222
|
-
export function useCreateProduct() {
|
|
223
|
-
const queryClient = useQueryClient();
|
|
224
|
-
|
|
225
|
-
return useMutation({
|
|
226
|
-
mutationFn: (data: CreateProductRequest) =>
|
|
227
|
-
apiClient<Product>("/api/products", {
|
|
228
|
-
method: "POST",
|
|
229
|
-
body: JSON.stringify(data),
|
|
230
|
-
}),
|
|
231
|
-
onSuccess: () => {
|
|
232
|
-
queryClient.invalidateQueries({ queryKey: ["products"] });
|
|
233
|
-
},
|
|
234
|
-
});
|
|
235
|
-
}
|
|
236
|
-
```
|
|
237
|
-
|
|
238
|
-
### 5.3 Form Pattern
|
|
239
|
-
|
|
240
|
-
```typescript
|
|
241
|
-
// components/forms/product-form.tsx
|
|
242
|
-
"use client";
|
|
243
|
-
|
|
244
|
-
import { useForm } from "react-hook-form";
|
|
245
|
-
import { zodResolver } from "@hookform/resolvers/zod";
|
|
246
|
-
import { z } from "zod";
|
|
247
|
-
import { useCreateProduct } from "@/hooks/use-products";
|
|
248
|
-
|
|
249
|
-
const schema = z.object({
|
|
250
|
-
name: z.string().min(1, "Name is required"),
|
|
251
|
-
price: z.number().positive("Price must be positive"),
|
|
252
|
-
description: z.string().optional(),
|
|
253
|
-
});
|
|
254
|
-
|
|
255
|
-
type FormData = z.infer<typeof schema>;
|
|
256
|
-
|
|
257
|
-
export function ProductForm() {
|
|
258
|
-
const { mutate, isPending } = useCreateProduct();
|
|
259
|
-
const form = useForm<FormData>({
|
|
260
|
-
resolver: zodResolver(schema),
|
|
261
|
-
});
|
|
262
|
-
|
|
263
|
-
const onSubmit = (data: FormData) => {
|
|
264
|
-
mutate(data);
|
|
265
|
-
};
|
|
266
|
-
|
|
267
|
-
return (
|
|
268
|
-
<form onSubmit={form.handleSubmit(onSubmit)}>
|
|
269
|
-
{/* Form fields */}
|
|
270
|
-
</form>
|
|
271
|
-
);
|
|
272
|
-
}
|
|
273
|
-
```
|
|
274
|
-
|
|
275
|
-
## 6. Comunicação API
|
|
276
|
-
|
|
277
|
-
### 6.1 API Client
|
|
278
|
-
|
|
279
|
-
```typescript
|
|
280
|
-
// lib/api-client.ts
|
|
281
|
-
import { auth } from "@clerk/nextjs/server";
|
|
282
|
-
|
|
283
|
-
const API_URL = process.env.NEXT_PUBLIC_API_URL;
|
|
284
|
-
|
|
285
|
-
interface FetchOptions extends RequestInit {
|
|
286
|
-
params?: Record<string, string | number | undefined>;
|
|
287
|
-
}
|
|
288
|
-
|
|
289
|
-
export async function apiClient<T>(
|
|
290
|
-
endpoint: string,
|
|
291
|
-
options: FetchOptions = {}
|
|
292
|
-
): Promise<T> {
|
|
293
|
-
const { params, ...init } = options;
|
|
294
|
-
|
|
295
|
-
// Build URL with query params
|
|
296
|
-
const url = new URL(`${API_URL}${endpoint}`);
|
|
297
|
-
if (params) {
|
|
298
|
-
Object.entries(params).forEach(([key, value]) => {
|
|
299
|
-
if (value !== undefined) {
|
|
300
|
-
url.searchParams.append(key, String(value));
|
|
301
|
-
}
|
|
302
|
-
});
|
|
303
|
-
}
|
|
304
|
-
|
|
305
|
-
// Get auth token
|
|
306
|
-
const { getToken } = auth();
|
|
307
|
-
const token = await getToken();
|
|
308
|
-
|
|
309
|
-
const response = await fetch(url.toString(), {
|
|
310
|
-
...init,
|
|
311
|
-
headers: {
|
|
312
|
-
"Content-Type": "application/json",
|
|
313
|
-
...(token && { Authorization: `Bearer ${token}` }),
|
|
314
|
-
...init.headers,
|
|
315
|
-
},
|
|
316
|
-
});
|
|
317
|
-
|
|
318
|
-
if (!response.ok) {
|
|
319
|
-
const error = await response.json().catch(() => ({}));
|
|
320
|
-
throw new ApiError(response.status, error.title || "Request failed", error);
|
|
321
|
-
}
|
|
322
|
-
|
|
323
|
-
return response.json();
|
|
324
|
-
}
|
|
325
|
-
|
|
326
|
-
class ApiError extends Error {
|
|
327
|
-
constructor(
|
|
328
|
-
public status: number,
|
|
329
|
-
message: string,
|
|
330
|
-
public details?: unknown
|
|
331
|
-
) {
|
|
332
|
-
super(message);
|
|
333
|
-
this.name = "ApiError";
|
|
334
|
-
}
|
|
335
|
-
}
|
|
336
|
-
```
|
|
337
|
-
|
|
338
|
-
## 7. Configurações
|
|
339
|
-
|
|
340
|
-
### 7.1 API (appsettings.json)
|
|
341
|
-
|
|
342
|
-
```json
|
|
343
|
-
{
|
|
344
|
-
"ConnectionStrings": {
|
|
345
|
-
"DefaultConnection": "Server=...;Database=...;"
|
|
346
|
-
},
|
|
347
|
-
"Clerk": {
|
|
348
|
-
"Authority": "https://your-clerk-domain.clerk.accounts.dev",
|
|
349
|
-
"SecretKey": "sk_..."
|
|
350
|
-
},
|
|
351
|
-
"Cors": {
|
|
352
|
-
"AllowedOrigins": ["http://localhost:3000", "https://your-domain.com"]
|
|
353
|
-
}
|
|
354
|
-
}
|
|
355
|
-
```
|
|
356
|
-
|
|
357
|
-
### 7.2 Frontend (.env.local)
|
|
358
|
-
|
|
359
|
-
```env
|
|
360
|
-
NEXT_PUBLIC_API_URL=http://localhost:5000
|
|
361
|
-
NEXT_PUBLIC_CLERK_PUBLISHABLE_KEY=pk_...
|
|
362
|
-
CLERK_SECRET_KEY=sk_...
|
|
363
|
-
```
|
|
364
|
-
|
|
365
|
-
## 8. Deploy
|
|
366
|
-
|
|
367
|
-
### 8.1 Docker
|
|
368
|
-
|
|
369
|
-
```dockerfile
|
|
370
|
-
# API Dockerfile
|
|
371
|
-
FROM mcr.microsoft.com/dotnet/aspnet:8.0 AS base
|
|
372
|
-
WORKDIR /app
|
|
373
|
-
EXPOSE 8080
|
|
374
|
-
|
|
375
|
-
FROM mcr.microsoft.com/dotnet/sdk:8.0 AS build
|
|
376
|
-
WORKDIR /src
|
|
377
|
-
COPY ["src/api/Api/Api.csproj", "Api/"]
|
|
378
|
-
RUN dotnet restore "Api/Api.csproj"
|
|
379
|
-
COPY src/api/ .
|
|
380
|
-
RUN dotnet build "Api/Api.csproj" -c Release -o /app/build
|
|
381
|
-
|
|
382
|
-
FROM build AS publish
|
|
383
|
-
RUN dotnet publish "Api/Api.csproj" -c Release -o /app/publish
|
|
384
|
-
|
|
385
|
-
FROM base AS final
|
|
386
|
-
WORKDIR /app
|
|
387
|
-
COPY --from=publish /app/publish .
|
|
388
|
-
ENTRYPOINT ["dotnet", "Api.dll"]
|
|
389
|
-
```
|
|
390
|
-
|
|
391
|
-
### 8.2 Container Apps
|
|
392
|
-
|
|
393
|
-
Dois Container Apps separados:
|
|
394
|
-
- `app-api`: .NET API
|
|
395
|
-
- `app-web`: Next.js (com output: standalone)
|
|
396
|
-
|
|
397
|
-
---
|
|
398
|
-
|
|
399
|
-
*MORPH-SPEC by Polymorphism Tech*
|
|
1
|
+
# API .NET + Next.js - Especificação
|
|
2
|
+
|
|
3
|
+
## 1. Visão Geral
|
|
4
|
+
|
|
5
|
+
Arquitetura full-stack moderna com API .NET e frontend Next.js, seguindo boas práticas de separação de responsabilidades.
|
|
6
|
+
|
|
7
|
+
### 1.1 Objetivos
|
|
8
|
+
|
|
9
|
+
- Backend robusto com .NET Minimal APIs
|
|
10
|
+
- Frontend moderno com Next.js App Router
|
|
11
|
+
- Autenticação unificada via Clerk
|
|
12
|
+
- Type-safety end-to-end
|
|
13
|
+
- Deploy independente de cada camada
|
|
14
|
+
|
|
15
|
+
### 1.2 Use Cases
|
|
16
|
+
|
|
17
|
+
- Dashboards administrativos
|
|
18
|
+
- Aplicações CRUD complexas
|
|
19
|
+
- Sistemas que precisam de SEO (Next.js SSR)
|
|
20
|
+
- Aplicações com múltiplos frontends (web, mobile)
|
|
21
|
+
|
|
22
|
+
## 2. Requisitos Funcionais
|
|
23
|
+
|
|
24
|
+
### 2.1 Backend (.NET API)
|
|
25
|
+
|
|
26
|
+
| ID | Requisito |
|
|
27
|
+
|----|-----------|
|
|
28
|
+
| API-001 | Minimal APIs com OpenAPI/Swagger |
|
|
29
|
+
| API-002 | Autenticação JWT via Clerk |
|
|
30
|
+
| API-003 | Validação com FluentValidation |
|
|
31
|
+
| API-004 | Paginação, filtros e ordenação |
|
|
32
|
+
| API-005 | Rate limiting por usuário |
|
|
33
|
+
| API-006 | Health checks (liveness + readiness) |
|
|
34
|
+
| API-007 | Logging estruturado |
|
|
35
|
+
| API-008 | CORS configurável |
|
|
36
|
+
|
|
37
|
+
### 2.2 Frontend (Next.js)
|
|
38
|
+
|
|
39
|
+
| ID | Requisito |
|
|
40
|
+
|----|-----------|
|
|
41
|
+
| WEB-001 | App Router com layouts aninhados |
|
|
42
|
+
| WEB-002 | Server Components para dados estáticos |
|
|
43
|
+
| WEB-003 | Client Components para interatividade |
|
|
44
|
+
| WEB-004 | React Query para cache e mutations |
|
|
45
|
+
| WEB-005 | Forms com React Hook Form + Zod |
|
|
46
|
+
| WEB-006 | UI com Tailwind + shadcn/ui |
|
|
47
|
+
| WEB-007 | Dark mode support |
|
|
48
|
+
| WEB-008 | Loading states e error boundaries |
|
|
49
|
+
|
|
50
|
+
### 2.3 Autenticação
|
|
51
|
+
|
|
52
|
+
| ID | Requisito |
|
|
53
|
+
|----|-----------|
|
|
54
|
+
| AUTH-001 | SSO via Clerk |
|
|
55
|
+
| AUTH-002 | Proteção de rotas (middleware) |
|
|
56
|
+
| AUTH-003 | Token refresh automático |
|
|
57
|
+
| AUTH-004 | Role-based access control |
|
|
58
|
+
|
|
59
|
+
## 3. Requisitos Não-Funcionais
|
|
60
|
+
|
|
61
|
+
### 3.1 Performance
|
|
62
|
+
|
|
63
|
+
| ID | Requisito |
|
|
64
|
+
|----|-----------|
|
|
65
|
+
| PERF-001 | API response < 100ms (P95) |
|
|
66
|
+
| PERF-002 | LCP < 2.5s (Core Web Vitals) |
|
|
67
|
+
| PERF-003 | FID < 100ms |
|
|
68
|
+
| PERF-004 | CLS < 0.1 |
|
|
69
|
+
|
|
70
|
+
### 3.2 Segurança
|
|
71
|
+
|
|
72
|
+
| ID | Requisito |
|
|
73
|
+
|----|-----------|
|
|
74
|
+
| SEC-001 | HTTPS obrigatório |
|
|
75
|
+
| SEC-002 | CORS restrito a domínios conhecidos |
|
|
76
|
+
| SEC-003 | Input validation em todos endpoints |
|
|
77
|
+
| SEC-004 | SQL injection prevention (EF Core) |
|
|
78
|
+
| SEC-005 | XSS prevention (React default) |
|
|
79
|
+
|
|
80
|
+
## 4. Arquitetura da API
|
|
81
|
+
|
|
82
|
+
### 4.1 Estrutura de Camadas
|
|
83
|
+
|
|
84
|
+
```
|
|
85
|
+
Api/
|
|
86
|
+
├── Endpoints/ # Minimal API endpoints
|
|
87
|
+
├── Middleware/ # Custom middlewares
|
|
88
|
+
├── Filters/ # Exception filters
|
|
89
|
+
└── Program.cs # Entry point
|
|
90
|
+
|
|
91
|
+
Application/
|
|
92
|
+
├── Services/ # Business logic
|
|
93
|
+
├── Validators/ # FluentValidation
|
|
94
|
+
└── DTOs/ # Data transfer objects
|
|
95
|
+
|
|
96
|
+
Domain/
|
|
97
|
+
├── Entities/ # Domain models
|
|
98
|
+
├── Interfaces/ # Repository contracts
|
|
99
|
+
└── Enums/ # Domain enums
|
|
100
|
+
|
|
101
|
+
Infrastructure/
|
|
102
|
+
├── Data/ # DbContext
|
|
103
|
+
├── Repositories/ # EF Core implementations
|
|
104
|
+
└── Services/ # External service clients
|
|
105
|
+
```
|
|
106
|
+
|
|
107
|
+
### 4.2 Endpoint Pattern
|
|
108
|
+
|
|
109
|
+
```csharp
|
|
110
|
+
// Endpoints/ProductEndpoints.cs
|
|
111
|
+
public static class ProductEndpoints
|
|
112
|
+
{
|
|
113
|
+
public static void MapProductEndpoints(this IEndpointRouteBuilder app)
|
|
114
|
+
{
|
|
115
|
+
var group = app.MapGroup("/api/products")
|
|
116
|
+
.WithTags("Products")
|
|
117
|
+
.RequireAuthorization();
|
|
118
|
+
|
|
119
|
+
group.MapGet("/", GetAll);
|
|
120
|
+
group.MapGet("/{id:int}", GetById);
|
|
121
|
+
group.MapPost("/", Create);
|
|
122
|
+
group.MapPut("/{id:int}", Update);
|
|
123
|
+
group.MapDelete("/{id:int}", Delete);
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
private static async Task<IResult> GetAll(
|
|
127
|
+
[AsParameters] PaginationQuery query,
|
|
128
|
+
IProductService service,
|
|
129
|
+
CancellationToken ct)
|
|
130
|
+
{
|
|
131
|
+
var result = await service.GetAllAsync(query, ct);
|
|
132
|
+
return Results.Ok(result);
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
```
|
|
136
|
+
|
|
137
|
+
### 4.3 Pagination
|
|
138
|
+
|
|
139
|
+
```csharp
|
|
140
|
+
public record PaginationQuery(
|
|
141
|
+
int Page = 1,
|
|
142
|
+
int PageSize = 20,
|
|
143
|
+
string? SortBy = null,
|
|
144
|
+
bool SortDesc = false,
|
|
145
|
+
string? Search = null
|
|
146
|
+
);
|
|
147
|
+
|
|
148
|
+
public record PagedResult<T>(
|
|
149
|
+
IReadOnlyList<T> Items,
|
|
150
|
+
int TotalItems,
|
|
151
|
+
int Page,
|
|
152
|
+
int PageSize,
|
|
153
|
+
int TotalPages
|
|
154
|
+
);
|
|
155
|
+
```
|
|
156
|
+
|
|
157
|
+
## 5. Arquitetura do Frontend
|
|
158
|
+
|
|
159
|
+
### 5.1 Estrutura de Pastas
|
|
160
|
+
|
|
161
|
+
```
|
|
162
|
+
app/
|
|
163
|
+
├── (auth)/
|
|
164
|
+
│ ├── sign-in/[[...sign-in]]/page.tsx
|
|
165
|
+
│ └── sign-up/[[...sign-up]]/page.tsx
|
|
166
|
+
├── (dashboard)/
|
|
167
|
+
│ ├── layout.tsx # Dashboard layout
|
|
168
|
+
│ ├── dashboard/page.tsx
|
|
169
|
+
│ ├── products/
|
|
170
|
+
│ │ ├── page.tsx # List
|
|
171
|
+
│ │ ├── [id]/page.tsx # Detail
|
|
172
|
+
│ │ └── new/page.tsx # Create
|
|
173
|
+
│ └── orders/
|
|
174
|
+
├── api/ # API routes (BFF)
|
|
175
|
+
├── layout.tsx # Root layout
|
|
176
|
+
└── page.tsx # Landing
|
|
177
|
+
|
|
178
|
+
components/
|
|
179
|
+
├── ui/ # shadcn components
|
|
180
|
+
├── forms/ # Form components
|
|
181
|
+
├── tables/ # Table components
|
|
182
|
+
└── shared/ # Shared components
|
|
183
|
+
|
|
184
|
+
hooks/
|
|
185
|
+
├── use-products.ts
|
|
186
|
+
├── use-orders.ts
|
|
187
|
+
└── use-pagination.ts
|
|
188
|
+
|
|
189
|
+
lib/
|
|
190
|
+
├── api-client.ts # Fetch wrapper
|
|
191
|
+
├── query-client.ts # React Query config
|
|
192
|
+
└── utils.ts # Utility functions
|
|
193
|
+
|
|
194
|
+
types/
|
|
195
|
+
├── api.ts # API response types
|
|
196
|
+
└── entities.ts # Domain types
|
|
197
|
+
```
|
|
198
|
+
|
|
199
|
+
### 5.2 Data Fetching Pattern
|
|
200
|
+
|
|
201
|
+
```typescript
|
|
202
|
+
// hooks/use-products.ts
|
|
203
|
+
import { useQuery, useMutation, useQueryClient } from "@tanstack/react-query";
|
|
204
|
+
import { apiClient } from "@/lib/api-client";
|
|
205
|
+
import type { Product, CreateProductRequest } from "@/types/entities";
|
|
206
|
+
|
|
207
|
+
export function useProducts(params?: PaginationParams) {
|
|
208
|
+
return useQuery({
|
|
209
|
+
queryKey: ["products", params],
|
|
210
|
+
queryFn: () => apiClient<PagedResult<Product>>("/api/products", { params }),
|
|
211
|
+
});
|
|
212
|
+
}
|
|
213
|
+
|
|
214
|
+
export function useProduct(id: number) {
|
|
215
|
+
return useQuery({
|
|
216
|
+
queryKey: ["products", id],
|
|
217
|
+
queryFn: () => apiClient<Product>(`/api/products/${id}`),
|
|
218
|
+
enabled: !!id,
|
|
219
|
+
});
|
|
220
|
+
}
|
|
221
|
+
|
|
222
|
+
export function useCreateProduct() {
|
|
223
|
+
const queryClient = useQueryClient();
|
|
224
|
+
|
|
225
|
+
return useMutation({
|
|
226
|
+
mutationFn: (data: CreateProductRequest) =>
|
|
227
|
+
apiClient<Product>("/api/products", {
|
|
228
|
+
method: "POST",
|
|
229
|
+
body: JSON.stringify(data),
|
|
230
|
+
}),
|
|
231
|
+
onSuccess: () => {
|
|
232
|
+
queryClient.invalidateQueries({ queryKey: ["products"] });
|
|
233
|
+
},
|
|
234
|
+
});
|
|
235
|
+
}
|
|
236
|
+
```
|
|
237
|
+
|
|
238
|
+
### 5.3 Form Pattern
|
|
239
|
+
|
|
240
|
+
```typescript
|
|
241
|
+
// components/forms/product-form.tsx
|
|
242
|
+
"use client";
|
|
243
|
+
|
|
244
|
+
import { useForm } from "react-hook-form";
|
|
245
|
+
import { zodResolver } from "@hookform/resolvers/zod";
|
|
246
|
+
import { z } from "zod";
|
|
247
|
+
import { useCreateProduct } from "@/hooks/use-products";
|
|
248
|
+
|
|
249
|
+
const schema = z.object({
|
|
250
|
+
name: z.string().min(1, "Name is required"),
|
|
251
|
+
price: z.number().positive("Price must be positive"),
|
|
252
|
+
description: z.string().optional(),
|
|
253
|
+
});
|
|
254
|
+
|
|
255
|
+
type FormData = z.infer<typeof schema>;
|
|
256
|
+
|
|
257
|
+
export function ProductForm() {
|
|
258
|
+
const { mutate, isPending } = useCreateProduct();
|
|
259
|
+
const form = useForm<FormData>({
|
|
260
|
+
resolver: zodResolver(schema),
|
|
261
|
+
});
|
|
262
|
+
|
|
263
|
+
const onSubmit = (data: FormData) => {
|
|
264
|
+
mutate(data);
|
|
265
|
+
};
|
|
266
|
+
|
|
267
|
+
return (
|
|
268
|
+
<form onSubmit={form.handleSubmit(onSubmit)}>
|
|
269
|
+
{/* Form fields */}
|
|
270
|
+
</form>
|
|
271
|
+
);
|
|
272
|
+
}
|
|
273
|
+
```
|
|
274
|
+
|
|
275
|
+
## 6. Comunicação API
|
|
276
|
+
|
|
277
|
+
### 6.1 API Client
|
|
278
|
+
|
|
279
|
+
```typescript
|
|
280
|
+
// lib/api-client.ts
|
|
281
|
+
import { auth } from "@clerk/nextjs/server";
|
|
282
|
+
|
|
283
|
+
const API_URL = process.env.NEXT_PUBLIC_API_URL;
|
|
284
|
+
|
|
285
|
+
interface FetchOptions extends RequestInit {
|
|
286
|
+
params?: Record<string, string | number | undefined>;
|
|
287
|
+
}
|
|
288
|
+
|
|
289
|
+
export async function apiClient<T>(
|
|
290
|
+
endpoint: string,
|
|
291
|
+
options: FetchOptions = {}
|
|
292
|
+
): Promise<T> {
|
|
293
|
+
const { params, ...init } = options;
|
|
294
|
+
|
|
295
|
+
// Build URL with query params
|
|
296
|
+
const url = new URL(`${API_URL}${endpoint}`);
|
|
297
|
+
if (params) {
|
|
298
|
+
Object.entries(params).forEach(([key, value]) => {
|
|
299
|
+
if (value !== undefined) {
|
|
300
|
+
url.searchParams.append(key, String(value));
|
|
301
|
+
}
|
|
302
|
+
});
|
|
303
|
+
}
|
|
304
|
+
|
|
305
|
+
// Get auth token
|
|
306
|
+
const { getToken } = auth();
|
|
307
|
+
const token = await getToken();
|
|
308
|
+
|
|
309
|
+
const response = await fetch(url.toString(), {
|
|
310
|
+
...init,
|
|
311
|
+
headers: {
|
|
312
|
+
"Content-Type": "application/json",
|
|
313
|
+
...(token && { Authorization: `Bearer ${token}` }),
|
|
314
|
+
...init.headers,
|
|
315
|
+
},
|
|
316
|
+
});
|
|
317
|
+
|
|
318
|
+
if (!response.ok) {
|
|
319
|
+
const error = await response.json().catch(() => ({}));
|
|
320
|
+
throw new ApiError(response.status, error.title || "Request failed", error);
|
|
321
|
+
}
|
|
322
|
+
|
|
323
|
+
return response.json();
|
|
324
|
+
}
|
|
325
|
+
|
|
326
|
+
class ApiError extends Error {
|
|
327
|
+
constructor(
|
|
328
|
+
public status: number,
|
|
329
|
+
message: string,
|
|
330
|
+
public details?: unknown
|
|
331
|
+
) {
|
|
332
|
+
super(message);
|
|
333
|
+
this.name = "ApiError";
|
|
334
|
+
}
|
|
335
|
+
}
|
|
336
|
+
```
|
|
337
|
+
|
|
338
|
+
## 7. Configurações
|
|
339
|
+
|
|
340
|
+
### 7.1 API (appsettings.json)
|
|
341
|
+
|
|
342
|
+
```json
|
|
343
|
+
{
|
|
344
|
+
"ConnectionStrings": {
|
|
345
|
+
"DefaultConnection": "Server=...;Database=...;"
|
|
346
|
+
},
|
|
347
|
+
"Clerk": {
|
|
348
|
+
"Authority": "https://your-clerk-domain.clerk.accounts.dev",
|
|
349
|
+
"SecretKey": "sk_..."
|
|
350
|
+
},
|
|
351
|
+
"Cors": {
|
|
352
|
+
"AllowedOrigins": ["http://localhost:3000", "https://your-domain.com"]
|
|
353
|
+
}
|
|
354
|
+
}
|
|
355
|
+
```
|
|
356
|
+
|
|
357
|
+
### 7.2 Frontend (.env.local)
|
|
358
|
+
|
|
359
|
+
```env
|
|
360
|
+
NEXT_PUBLIC_API_URL=http://localhost:5000
|
|
361
|
+
NEXT_PUBLIC_CLERK_PUBLISHABLE_KEY=pk_...
|
|
362
|
+
CLERK_SECRET_KEY=sk_...
|
|
363
|
+
```
|
|
364
|
+
|
|
365
|
+
## 8. Deploy
|
|
366
|
+
|
|
367
|
+
### 8.1 Docker
|
|
368
|
+
|
|
369
|
+
```dockerfile
|
|
370
|
+
# API Dockerfile
|
|
371
|
+
FROM mcr.microsoft.com/dotnet/aspnet:8.0 AS base
|
|
372
|
+
WORKDIR /app
|
|
373
|
+
EXPOSE 8080
|
|
374
|
+
|
|
375
|
+
FROM mcr.microsoft.com/dotnet/sdk:8.0 AS build
|
|
376
|
+
WORKDIR /src
|
|
377
|
+
COPY ["src/api/Api/Api.csproj", "Api/"]
|
|
378
|
+
RUN dotnet restore "Api/Api.csproj"
|
|
379
|
+
COPY src/api/ .
|
|
380
|
+
RUN dotnet build "Api/Api.csproj" -c Release -o /app/build
|
|
381
|
+
|
|
382
|
+
FROM build AS publish
|
|
383
|
+
RUN dotnet publish "Api/Api.csproj" -c Release -o /app/publish
|
|
384
|
+
|
|
385
|
+
FROM base AS final
|
|
386
|
+
WORKDIR /app
|
|
387
|
+
COPY --from=publish /app/publish .
|
|
388
|
+
ENTRYPOINT ["dotnet", "Api.dll"]
|
|
389
|
+
```
|
|
390
|
+
|
|
391
|
+
### 8.2 Container Apps
|
|
392
|
+
|
|
393
|
+
Dois Container Apps separados:
|
|
394
|
+
- `app-api`: .NET API
|
|
395
|
+
- `app-web`: Next.js (com output: standalone)
|
|
396
|
+
|
|
397
|
+
---
|
|
398
|
+
|
|
399
|
+
*MORPH-SPEC by Polymorphism Tech*
|