@nextsparkjs/core 0.1.0-beta.83 → 0.1.0-beta.85
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/dist/styles/classes.json +1 -1
- package/dist/templates/app/(auth)/forgot-password/page.tsx +216 -0
- package/dist/templates/app/(auth)/layout.tsx +51 -0
- package/dist/templates/app/(auth)/login/page.tsx +21 -0
- package/dist/templates/app/(auth)/reset-password/page.tsx +212 -0
- package/dist/templates/app/(auth)/signup/page.tsx +21 -0
- package/dist/templates/app/(auth)/verify-email/page.tsx +190 -0
- package/dist/templates/app/(public)/[...slug]/page.tsx +378 -0
- package/dist/templates/app/(public)/docs/[section]/[page]/page.tsx +90 -0
- package/dist/templates/app/(public)/docs/layout.tsx +25 -0
- package/dist/templates/app/(public)/docs/page.tsx +81 -0
- package/dist/templates/app/(public)/layout.tsx +41 -0
- package/dist/templates/app/(public)/page.tsx +19 -0
- package/dist/templates/app/403/page.tsx +89 -0
- package/dist/templates/app/api/auth/[...all]/route.ts +78 -0
- package/dist/templates/app/api/cron/billing/lifecycle/route.ts +98 -0
- package/dist/templates/app/api/csp-report/route.ts +175 -0
- package/dist/templates/app/api/devtools/config/entities/route.ts +108 -0
- package/dist/templates/app/api/devtools/config/theme/route.ts +66 -0
- package/dist/templates/app/api/devtools/tests/[...path]/route.ts +130 -0
- package/dist/templates/app/api/devtools/tests/route.ts +134 -0
- package/dist/templates/app/api/health/route.ts +29 -0
- package/dist/templates/app/api/internal/user-metadata/route.ts +36 -0
- package/dist/templates/app/api/superadmin/subscriptions/route.ts +310 -0
- package/dist/templates/app/api/superadmin/teams/[teamId]/route.ts +286 -0
- package/dist/templates/app/api/superadmin/teams/route.ts +188 -0
- package/dist/templates/app/api/superadmin/users/[userId]/route.ts +540 -0
- package/dist/templates/app/api/superadmin/users/route.ts +323 -0
- package/dist/templates/app/api/user/delete-account/route.ts +55 -0
- package/dist/templates/app/api/user/plan-flags/route.ts +283 -0
- package/dist/templates/app/api/user/profile/route.ts +133 -0
- package/dist/templates/app/api/v1/[entity]/[id]/child/[childType]/[childId]/route.ts +210 -0
- package/dist/templates/app/api/v1/[entity]/[id]/child/[childType]/route.ts +331 -0
- package/dist/templates/app/api/v1/[entity]/[id]/route.ts +35 -0
- package/dist/templates/app/api/v1/[entity]/docs.md +369 -0
- package/dist/templates/app/api/v1/[entity]/presets.ts +194 -0
- package/dist/templates/app/api/v1/[entity]/route.ts +31 -0
- package/dist/templates/app/api/v1/api-keys/[id]/route.ts +303 -0
- package/dist/templates/app/api/v1/api-keys/docs.md +101 -0
- package/dist/templates/app/api/v1/api-keys/presets.ts +31 -0
- package/dist/templates/app/api/v1/api-keys/route.ts +250 -0
- package/dist/templates/app/api/v1/auth/docs.md +184 -0
- package/dist/templates/app/api/v1/auth/presets.ts +44 -0
- package/dist/templates/app/api/v1/auth/signup-with-invite/route.ts +227 -0
- package/dist/templates/app/api/v1/billing/cancel/route.ts +206 -0
- package/dist/templates/app/api/v1/billing/change-plan/route.ts +97 -0
- package/dist/templates/app/api/v1/billing/check-action/route.ts +81 -0
- package/dist/templates/app/api/v1/billing/checkout/route.ts +124 -0
- package/dist/templates/app/api/v1/billing/docs.md +209 -0
- package/dist/templates/app/api/v1/billing/plans/route.ts +85 -0
- package/dist/templates/app/api/v1/billing/portal/route.ts +90 -0
- package/dist/templates/app/api/v1/billing/presets.ts +121 -0
- package/dist/templates/app/api/v1/billing/webhooks/stripe/route.ts +428 -0
- package/dist/templates/app/api/v1/blocks/[slug]/route.ts +29 -0
- package/dist/templates/app/api/v1/blocks/docs.md +173 -0
- package/dist/templates/app/api/v1/blocks/presets.ts +121 -0
- package/dist/templates/app/api/v1/blocks/route.ts +45 -0
- package/dist/templates/app/api/v1/blocks/validate/route.ts +45 -0
- package/dist/templates/app/api/v1/cron/docs.md +116 -0
- package/dist/templates/app/api/v1/cron/presets.ts +26 -0
- package/dist/templates/app/api/v1/cron/process/route.ts +108 -0
- package/dist/templates/app/api/v1/devtools/blocks/route.ts +82 -0
- package/dist/templates/app/api/v1/devtools/docs/route.ts +150 -0
- package/dist/templates/app/api/v1/devtools/docs.md +204 -0
- package/dist/templates/app/api/v1/devtools/features/route.ts +61 -0
- package/dist/templates/app/api/v1/devtools/flows/route.ts +61 -0
- package/dist/templates/app/api/v1/devtools/presets.ts +113 -0
- package/dist/templates/app/api/v1/devtools/scheduled-actions/route.ts +120 -0
- package/dist/templates/app/api/v1/devtools/testing/route.ts +82 -0
- package/dist/templates/app/api/v1/media/docs.md +117 -0
- package/dist/templates/app/api/v1/media/presets.ts +24 -0
- package/dist/templates/app/api/v1/media/upload/route.ts +150 -0
- package/dist/templates/app/api/v1/patterns/[id]/usages/route.ts +116 -0
- package/dist/templates/app/api/v1/plugin/[...path]/route.ts +373 -0
- package/dist/templates/app/api/v1/plugin/docs.md +79 -0
- package/dist/templates/app/api/v1/plugin/presets.ts +21 -0
- package/dist/templates/app/api/v1/plugin/route.ts +96 -0
- package/dist/templates/app/api/v1/post-categories/[id]/route.ts +255 -0
- package/dist/templates/app/api/v1/post-categories/docs.md +134 -0
- package/dist/templates/app/api/v1/post-categories/presets.ts +78 -0
- package/dist/templates/app/api/v1/post-categories/route.ts +119 -0
- package/dist/templates/app/api/v1/team-invitations/[token]/accept/route.ts +179 -0
- package/dist/templates/app/api/v1/team-invitations/[token]/decline/route.ts +120 -0
- package/dist/templates/app/api/v1/team-invitations/[token]/route.ts +89 -0
- package/dist/templates/app/api/v1/team-invitations/docs.md +88 -0
- package/dist/templates/app/api/v1/team-invitations/presets.ts +43 -0
- package/dist/templates/app/api/v1/team-invitations/route.ts +114 -0
- package/dist/templates/app/api/v1/teams/[teamId]/invitations/route.ts +171 -0
- package/dist/templates/app/api/v1/teams/[teamId]/invoices/[invoiceNumber]/route.ts +105 -0
- package/dist/templates/app/api/v1/teams/[teamId]/invoices/route.ts +125 -0
- package/dist/templates/app/api/v1/teams/[teamId]/members/[memberId]/route.ts +263 -0
- package/dist/templates/app/api/v1/teams/[teamId]/members/route.ts +358 -0
- package/dist/templates/app/api/v1/teams/[teamId]/route.ts +322 -0
- package/dist/templates/app/api/v1/teams/[teamId]/subscription/route.ts +50 -0
- package/dist/templates/app/api/v1/teams/[teamId]/usage/[limitSlug]/route.ts +91 -0
- package/dist/templates/app/api/v1/teams/docs.md +320 -0
- package/dist/templates/app/api/v1/teams/presets.ts +178 -0
- package/dist/templates/app/api/v1/teams/route.ts +293 -0
- package/dist/templates/app/api/v1/teams/switch/route.ts +88 -0
- package/dist/templates/app/api/v1/theme/[...path]/route.ts +361 -0
- package/dist/templates/app/api/v1/theme/docs.md +74 -0
- package/dist/templates/app/api/v1/theme/presets.ts +21 -0
- package/dist/templates/app/api/v1/theme/route.ts +96 -0
- package/dist/templates/app/api/v1/users/[id]/meta/[key]/route.ts +363 -0
- package/dist/templates/app/api/v1/users/[id]/route.ts +302 -0
- package/dist/templates/app/api/v1/users/docs.md +93 -0
- package/dist/templates/app/api/v1/users/presets.ts +59 -0
- package/dist/templates/app/api/v1/users/route.ts +197 -0
- package/dist/templates/app/dashboard/(main)/[entity]/[id]/edit/page.tsx +117 -0
- package/dist/templates/app/dashboard/(main)/[entity]/[id]/page.tsx +103 -0
- package/dist/templates/app/dashboard/(main)/[entity]/create/page.tsx +95 -0
- package/dist/templates/app/dashboard/(main)/[entity]/error.tsx +51 -0
- package/dist/templates/app/dashboard/(main)/[entity]/layout.tsx +113 -0
- package/dist/templates/app/dashboard/(main)/[entity]/loading.tsx +61 -0
- package/dist/templates/app/dashboard/(main)/[entity]/page.tsx +90 -0
- package/dist/templates/app/dashboard/(main)/layout.tsx +98 -0
- package/dist/templates/app/dashboard/(main)/loading.tsx +5 -0
- package/dist/templates/app/dashboard/(main)/page.tsx +201 -0
- package/dist/templates/app/dashboard/(main)/patterns/[id]/edit/page.tsx +114 -0
- package/dist/templates/app/dashboard/(main)/patterns/[id]/page.tsx +20 -0
- package/dist/templates/app/dashboard/(main)/patterns/[id]/reports/page.tsx +171 -0
- package/dist/templates/app/dashboard/(main)/patterns/create/page.tsx +86 -0
- package/dist/templates/app/dashboard/(main)/patterns/page.tsx +444 -0
- package/dist/templates/app/dashboard/features/analytics/page.tsx +35 -0
- package/dist/templates/app/dashboard/features/automation/page.tsx +35 -0
- package/dist/templates/app/dashboard/features/layout.tsx +13 -0
- package/dist/templates/app/dashboard/features/loading.tsx +5 -0
- package/dist/templates/app/dashboard/features/webhooks/page.tsx +35 -0
- package/dist/templates/app/dashboard/layout.tsx +86 -0
- package/dist/templates/app/dashboard/permission-denied/page.tsx +29 -0
- package/dist/templates/app/dashboard/settings/api-keys/loading.tsx +5 -0
- package/dist/templates/app/dashboard/settings/api-keys/page.tsx +513 -0
- package/dist/templates/app/dashboard/settings/billing/loading.tsx +5 -0
- package/dist/templates/app/dashboard/settings/billing/page.tsx +284 -0
- package/dist/templates/app/dashboard/settings/invoices/[invoiceNumber]/page.tsx +222 -0
- package/dist/templates/app/dashboard/settings/invoices/loading.tsx +5 -0
- package/dist/templates/app/dashboard/settings/invoices/page.tsx +82 -0
- package/dist/templates/app/dashboard/settings/layout.tsx +151 -0
- package/dist/templates/app/dashboard/settings/loading.tsx +5 -0
- package/dist/templates/app/dashboard/settings/notifications/loading.tsx +5 -0
- package/dist/templates/app/dashboard/settings/notifications/page.tsx +462 -0
- package/dist/templates/app/dashboard/settings/page.tsx +92 -0
- package/dist/templates/app/dashboard/settings/password/loading.tsx +5 -0
- package/dist/templates/app/dashboard/settings/password/page.tsx +306 -0
- package/dist/templates/app/dashboard/settings/plans/loading.tsx +5 -0
- package/dist/templates/app/dashboard/settings/plans/page.tsx +40 -0
- package/dist/templates/app/dashboard/settings/profile/loading.tsx +5 -0
- package/dist/templates/app/dashboard/settings/profile/page.tsx +686 -0
- package/dist/templates/app/dashboard/settings/security/loading.tsx +5 -0
- package/dist/templates/app/dashboard/settings/security/page.tsx +505 -0
- package/dist/templates/app/dashboard/settings/teams/loading.tsx +5 -0
- package/dist/templates/app/dashboard/settings/teams/page.tsx +272 -0
- package/dist/templates/app/dashboard/settings/teams/permissions/page.tsx +92 -0
- package/dist/templates/app/devtools/blocks/[slug]/page.tsx +39 -0
- package/dist/templates/app/devtools/blocks/page.tsx +31 -0
- package/dist/templates/app/devtools/config/page.tsx +31 -0
- package/dist/templates/app/devtools/features/page.tsx +31 -0
- package/dist/templates/app/devtools/flows/page.tsx +31 -0
- package/dist/templates/app/devtools/layout.tsx +58 -0
- package/dist/templates/app/devtools/page.tsx +121 -0
- package/dist/templates/app/devtools/scheduled-actions/page.tsx +157 -0
- package/dist/templates/app/devtools/style/page.tsx +330 -0
- package/dist/templates/app/devtools/tags/page.tsx +31 -0
- package/dist/templates/app/devtools/tests/[[...path]]/page.tsx +47 -0
- package/dist/templates/app/favicon.ico +0 -0
- package/dist/templates/app/globals.css +12 -0
- package/dist/templates/app/layout.tsx +96 -0
- package/dist/templates/app/public/page.tsx +30 -0
- package/dist/templates/app/superadmin/docs/[section]/[page]/page.tsx +92 -0
- package/dist/templates/app/superadmin/docs/page.tsx +75 -0
- package/dist/templates/app/superadmin/layout.tsx +67 -0
- package/dist/templates/app/superadmin/page.tsx +149 -0
- package/dist/templates/app/superadmin/subscriptions/page.tsx +655 -0
- package/dist/templates/app/superadmin/team-roles/page.tsx +493 -0
- package/dist/templates/app/superadmin/teams/[teamId]/page.tsx +687 -0
- package/dist/templates/app/superadmin/teams/page.tsx +302 -0
- package/dist/templates/app/superadmin/users/[userId]/page.tsx +548 -0
- package/dist/templates/app/superadmin/users/page.tsx +528 -0
- package/package.json +15 -15
- package/scripts/build/docs-registry.mjs +0 -0
- package/scripts/create-theme.mjs +0 -0
- package/scripts/deploy/release-version.mjs +0 -0
- package/scripts/deploy/vercel-deploy.mjs +0 -0
- package/scripts/dev/watch-plugins.mjs +0 -0
- package/scripts/maintenance/update-core.mjs +0 -0
- package/scripts/setup/npm-postinstall.mjs +0 -0
- package/scripts/setup/setup-claude.mjs +0 -0
- package/scripts/validation/check-imports.sh +0 -0
- package/templates/app/(auth)/forgot-password/page.tsx +216 -0
- package/templates/app/(auth)/layout.tsx +51 -0
- package/templates/app/(auth)/login/page.tsx +21 -0
- package/templates/app/(auth)/reset-password/page.tsx +212 -0
- package/templates/app/(auth)/signup/page.tsx +21 -0
- package/templates/app/(auth)/verify-email/page.tsx +190 -0
- package/templates/app/(public)/[...slug]/page.tsx +378 -0
- package/templates/app/(public)/docs/[section]/[page]/page.tsx +90 -0
- package/templates/app/(public)/docs/layout.tsx +25 -0
- package/templates/app/(public)/docs/page.tsx +81 -0
- package/templates/app/(public)/layout.tsx +41 -0
- package/templates/app/(public)/page.tsx +19 -0
- package/templates/app/403/page.tsx +89 -0
- package/templates/app/api/auth/[...all]/route.ts +78 -0
- package/templates/app/api/cron/billing/lifecycle/route.ts +98 -0
- package/templates/app/api/csp-report/route.ts +175 -0
- package/templates/app/api/devtools/config/entities/route.ts +108 -0
- package/templates/app/api/devtools/config/theme/route.ts +66 -0
- package/templates/app/api/devtools/tests/[...path]/route.ts +130 -0
- package/templates/app/api/devtools/tests/route.ts +134 -0
- package/templates/app/api/health/route.ts +29 -0
- package/templates/app/api/internal/user-metadata/route.ts +36 -0
- package/templates/app/api/superadmin/subscriptions/route.ts +310 -0
- package/templates/app/api/superadmin/teams/[teamId]/route.ts +286 -0
- package/templates/app/api/superadmin/teams/route.ts +188 -0
- package/templates/app/api/superadmin/users/[userId]/route.ts +540 -0
- package/templates/app/api/superadmin/users/route.ts +323 -0
- package/templates/app/api/user/delete-account/route.ts +55 -0
- package/templates/app/api/user/plan-flags/route.ts +283 -0
- package/templates/app/api/user/profile/route.ts +133 -0
- package/templates/app/api/v1/[entity]/[id]/child/[childType]/[childId]/route.ts +210 -0
- package/templates/app/api/v1/[entity]/[id]/child/[childType]/route.ts +331 -0
- package/templates/app/api/v1/[entity]/[id]/route.ts +35 -0
- package/templates/app/api/v1/[entity]/docs.md +369 -0
- package/templates/app/api/v1/[entity]/presets.ts +194 -0
- package/templates/app/api/v1/[entity]/route.ts +31 -0
- package/templates/app/api/v1/api-keys/[id]/route.ts +303 -0
- package/templates/app/api/v1/api-keys/docs.md +101 -0
- package/templates/app/api/v1/api-keys/presets.ts +31 -0
- package/templates/app/api/v1/api-keys/route.ts +250 -0
- package/templates/app/api/v1/auth/docs.md +184 -0
- package/templates/app/api/v1/auth/presets.ts +44 -0
- package/templates/app/api/v1/auth/signup-with-invite/route.ts +227 -0
- package/templates/app/api/v1/billing/cancel/route.ts +206 -0
- package/templates/app/api/v1/billing/change-plan/route.ts +97 -0
- package/templates/app/api/v1/billing/check-action/route.ts +81 -0
- package/templates/app/api/v1/billing/checkout/route.ts +124 -0
- package/templates/app/api/v1/billing/docs.md +209 -0
- package/templates/app/api/v1/billing/plans/route.ts +85 -0
- package/templates/app/api/v1/billing/portal/route.ts +90 -0
- package/templates/app/api/v1/billing/presets.ts +121 -0
- package/templates/app/api/v1/billing/webhooks/stripe/route.ts +428 -0
- package/templates/app/api/v1/blocks/[slug]/route.ts +29 -0
- package/templates/app/api/v1/blocks/docs.md +173 -0
- package/templates/app/api/v1/blocks/presets.ts +121 -0
- package/templates/app/api/v1/blocks/route.ts +45 -0
- package/templates/app/api/v1/blocks/validate/route.ts +45 -0
- package/templates/app/api/v1/cron/docs.md +116 -0
- package/templates/app/api/v1/cron/presets.ts +26 -0
- package/templates/app/api/v1/cron/process/route.ts +108 -0
- package/templates/app/api/v1/devtools/blocks/route.ts +82 -0
- package/templates/app/api/v1/devtools/docs/route.ts +150 -0
- package/templates/app/api/v1/devtools/docs.md +204 -0
- package/templates/app/api/v1/devtools/features/route.ts +61 -0
- package/templates/app/api/v1/devtools/flows/route.ts +61 -0
- package/templates/app/api/v1/devtools/presets.ts +113 -0
- package/templates/app/api/v1/devtools/scheduled-actions/route.ts +120 -0
- package/templates/app/api/v1/devtools/testing/route.ts +82 -0
- package/templates/app/api/v1/media/docs.md +117 -0
- package/templates/app/api/v1/media/presets.ts +24 -0
- package/templates/app/api/v1/media/upload/route.ts +150 -0
- package/templates/app/api/v1/patterns/[id]/usages/route.ts +116 -0
- package/templates/app/api/v1/plugin/[...path]/route.ts +373 -0
- package/templates/app/api/v1/plugin/docs.md +79 -0
- package/templates/app/api/v1/plugin/presets.ts +21 -0
- package/templates/app/api/v1/plugin/route.ts +96 -0
- package/templates/app/api/v1/post-categories/[id]/route.ts +255 -0
- package/templates/app/api/v1/post-categories/docs.md +134 -0
- package/templates/app/api/v1/post-categories/presets.ts +78 -0
- package/templates/app/api/v1/post-categories/route.ts +119 -0
- package/templates/app/api/v1/team-invitations/[token]/accept/route.ts +179 -0
- package/templates/app/api/v1/team-invitations/[token]/decline/route.ts +120 -0
- package/templates/app/api/v1/team-invitations/[token]/route.ts +89 -0
- package/templates/app/api/v1/team-invitations/docs.md +88 -0
- package/templates/app/api/v1/team-invitations/presets.ts +43 -0
- package/templates/app/api/v1/team-invitations/route.ts +114 -0
- package/templates/app/api/v1/teams/[teamId]/invitations/route.ts +171 -0
- package/templates/app/api/v1/teams/[teamId]/invoices/[invoiceNumber]/route.ts +105 -0
- package/templates/app/api/v1/teams/[teamId]/invoices/route.ts +125 -0
- package/templates/app/api/v1/teams/[teamId]/members/[memberId]/route.ts +263 -0
- package/templates/app/api/v1/teams/[teamId]/members/route.ts +358 -0
- package/templates/app/api/v1/teams/[teamId]/route.ts +322 -0
- package/templates/app/api/v1/teams/[teamId]/subscription/route.ts +50 -0
- package/templates/app/api/v1/teams/[teamId]/usage/[limitSlug]/route.ts +91 -0
- package/templates/app/api/v1/teams/docs.md +320 -0
- package/templates/app/api/v1/teams/presets.ts +178 -0
- package/templates/app/api/v1/teams/route.ts +293 -0
- package/templates/app/api/v1/teams/switch/route.ts +88 -0
- package/templates/app/api/v1/theme/[...path]/route.ts +361 -0
- package/templates/app/api/v1/theme/docs.md +74 -0
- package/templates/app/api/v1/theme/presets.ts +21 -0
- package/templates/app/api/v1/theme/route.ts +96 -0
- package/templates/app/api/v1/users/[id]/meta/[key]/route.ts +363 -0
- package/templates/app/api/v1/users/[id]/route.ts +302 -0
- package/templates/app/api/v1/users/docs.md +93 -0
- package/templates/app/api/v1/users/presets.ts +59 -0
- package/templates/app/api/v1/users/route.ts +197 -0
- package/templates/app/dashboard/(main)/[entity]/[id]/edit/page.tsx +117 -0
- package/templates/app/dashboard/(main)/[entity]/[id]/page.tsx +103 -0
- package/templates/app/dashboard/(main)/[entity]/create/page.tsx +95 -0
- package/templates/app/dashboard/(main)/[entity]/error.tsx +51 -0
- package/templates/app/dashboard/(main)/[entity]/layout.tsx +113 -0
- package/templates/app/dashboard/(main)/[entity]/loading.tsx +61 -0
- package/templates/app/dashboard/(main)/[entity]/page.tsx +90 -0
- package/templates/app/dashboard/(main)/layout.tsx +98 -0
- package/templates/app/dashboard/(main)/loading.tsx +5 -0
- package/templates/app/dashboard/(main)/page.tsx +201 -0
- package/templates/app/dashboard/(main)/patterns/[id]/edit/page.tsx +114 -0
- package/templates/app/dashboard/(main)/patterns/[id]/page.tsx +20 -0
- package/templates/app/dashboard/(main)/patterns/[id]/reports/page.tsx +171 -0
- package/templates/app/dashboard/(main)/patterns/create/page.tsx +86 -0
- package/templates/app/dashboard/(main)/patterns/page.tsx +444 -0
- package/templates/app/dashboard/features/analytics/page.tsx +35 -0
- package/templates/app/dashboard/features/automation/page.tsx +35 -0
- package/templates/app/dashboard/features/layout.tsx +13 -0
- package/templates/app/dashboard/features/loading.tsx +5 -0
- package/templates/app/dashboard/features/webhooks/page.tsx +35 -0
- package/templates/app/dashboard/layout.tsx +86 -0
- package/templates/app/dashboard/permission-denied/page.tsx +29 -0
- package/templates/app/dashboard/settings/api-keys/loading.tsx +5 -0
- package/templates/app/dashboard/settings/api-keys/page.tsx +513 -0
- package/templates/app/dashboard/settings/billing/loading.tsx +5 -0
- package/templates/app/dashboard/settings/billing/page.tsx +284 -0
- package/templates/app/dashboard/settings/invoices/[invoiceNumber]/page.tsx +222 -0
- package/templates/app/dashboard/settings/invoices/loading.tsx +5 -0
- package/templates/app/dashboard/settings/invoices/page.tsx +82 -0
- package/templates/app/dashboard/settings/layout.tsx +151 -0
- package/templates/app/dashboard/settings/loading.tsx +5 -0
- package/templates/app/dashboard/settings/notifications/loading.tsx +5 -0
- package/templates/app/dashboard/settings/notifications/page.tsx +462 -0
- package/templates/app/dashboard/settings/page.tsx +92 -0
- package/templates/app/dashboard/settings/password/loading.tsx +5 -0
- package/templates/app/dashboard/settings/password/page.tsx +306 -0
- package/templates/app/dashboard/settings/plans/loading.tsx +5 -0
- package/templates/app/dashboard/settings/plans/page.tsx +40 -0
- package/templates/app/dashboard/settings/profile/loading.tsx +5 -0
- package/templates/app/dashboard/settings/profile/page.tsx +686 -0
- package/templates/app/dashboard/settings/security/loading.tsx +5 -0
- package/templates/app/dashboard/settings/security/page.tsx +505 -0
- package/templates/app/dashboard/settings/teams/loading.tsx +5 -0
- package/templates/app/dashboard/settings/teams/page.tsx +272 -0
- package/templates/app/dashboard/settings/teams/permissions/page.tsx +92 -0
- package/templates/app/devtools/blocks/[slug]/page.tsx +39 -0
- package/templates/app/devtools/blocks/page.tsx +31 -0
- package/templates/app/devtools/config/page.tsx +31 -0
- package/templates/app/devtools/features/page.tsx +31 -0
- package/templates/app/devtools/flows/page.tsx +31 -0
- package/templates/app/devtools/layout.tsx +58 -0
- package/templates/app/devtools/page.tsx +121 -0
- package/templates/app/devtools/scheduled-actions/page.tsx +157 -0
- package/templates/app/devtools/style/page.tsx +330 -0
- package/templates/app/devtools/tags/page.tsx +31 -0
- package/templates/app/devtools/tests/[[...path]]/page.tsx +47 -0
- package/templates/app/favicon.ico +0 -0
- package/templates/app/globals.css +12 -0
- package/templates/app/layout.tsx +96 -0
- package/templates/app/public/page.tsx +30 -0
- package/templates/app/superadmin/docs/[section]/[page]/page.tsx +92 -0
- package/templates/app/superadmin/docs/page.tsx +75 -0
- package/templates/app/superadmin/layout.tsx +67 -0
- package/templates/app/superadmin/page.tsx +149 -0
- package/templates/app/superadmin/subscriptions/page.tsx +655 -0
- package/templates/app/superadmin/team-roles/page.tsx +493 -0
- package/templates/app/superadmin/teams/[teamId]/page.tsx +687 -0
- package/templates/app/superadmin/teams/page.tsx +302 -0
- package/templates/app/superadmin/users/[userId]/page.tsx +548 -0
- package/templates/app/superadmin/users/page.tsx +528 -0
|
@@ -0,0 +1,363 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* User Metadata Individual Endpoint
|
|
3
|
+
*
|
|
4
|
+
* RESTful API for managing individual user metadata keys.
|
|
5
|
+
* More efficient than bulk metadata operations when working with single values.
|
|
6
|
+
*
|
|
7
|
+
* Endpoints:
|
|
8
|
+
* - GET /api/v1/users/:id/meta/:key - Get specific metadata value
|
|
9
|
+
* - PUT /api/v1/users/:id/meta/:key - Create or update metadata value
|
|
10
|
+
* - DELETE /api/v1/users/:id/meta/:key - Delete metadata value
|
|
11
|
+
*
|
|
12
|
+
* Features:
|
|
13
|
+
* - Dual authentication (API Key + Session)
|
|
14
|
+
* - RLS (Row Level Security)
|
|
15
|
+
* - Efficient single-meta operations
|
|
16
|
+
* - CORS support
|
|
17
|
+
* - Rate limiting
|
|
18
|
+
*
|
|
19
|
+
* @module api/v1/users/[id]/meta/[key]
|
|
20
|
+
*/
|
|
21
|
+
|
|
22
|
+
import { NextRequest, NextResponse } from 'next/server'
|
|
23
|
+
import { UserService } from '@nextsparkjs/core/lib/services'
|
|
24
|
+
import {
|
|
25
|
+
createApiResponse,
|
|
26
|
+
createApiError,
|
|
27
|
+
withApiLogging,
|
|
28
|
+
handleCorsPreflightRequest,
|
|
29
|
+
addCorsHeaders,
|
|
30
|
+
} from '@nextsparkjs/core/lib/api/helpers'
|
|
31
|
+
import { authenticateRequest, hasRequiredScope } from '@nextsparkjs/core/lib/api/auth/dual-auth'
|
|
32
|
+
import { z } from 'zod'
|
|
33
|
+
|
|
34
|
+
// Validation schema for metadata value
|
|
35
|
+
const metadataValueSchema = z.object({
|
|
36
|
+
value: z.any(), // Accept any JSON value
|
|
37
|
+
isPublic: z.boolean().optional().default(false),
|
|
38
|
+
isSearchable: z.boolean().optional().default(false),
|
|
39
|
+
dataType: z.enum(['string', 'number', 'boolean', 'json', 'array']).optional().default('json'),
|
|
40
|
+
})
|
|
41
|
+
|
|
42
|
+
// Handle CORS preflight
|
|
43
|
+
export async function OPTIONS() {
|
|
44
|
+
return handleCorsPreflightRequest()
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
/**
|
|
48
|
+
* GET /api/v1/users/:id/meta/:key
|
|
49
|
+
* Retrieve a specific user metadata value
|
|
50
|
+
*
|
|
51
|
+
* @param id - User ID or email
|
|
52
|
+
* @param key - Metadata key to retrieve
|
|
53
|
+
* @returns Metadata value or 404 if not found
|
|
54
|
+
*
|
|
55
|
+
* @example
|
|
56
|
+
* GET /api/v1/users/user-123/meta/theme
|
|
57
|
+
* Response: { success: true, data: { key: "theme", value: "dark" } }
|
|
58
|
+
*/
|
|
59
|
+
export const GET = withApiLogging(async (
|
|
60
|
+
req: NextRequest,
|
|
61
|
+
{ params }: { params: Promise<{ id: string; key: string }> }
|
|
62
|
+
): Promise<NextResponse> => {
|
|
63
|
+
try {
|
|
64
|
+
// Authenticate using dual auth
|
|
65
|
+
const authResult = await authenticateRequest(req)
|
|
66
|
+
|
|
67
|
+
if (!authResult.success || !authResult.user) {
|
|
68
|
+
return NextResponse.json(
|
|
69
|
+
createApiError('Authentication required', 401, null, 'AUTHENTICATION_FAILED'),
|
|
70
|
+
{ status: 401 }
|
|
71
|
+
)
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
if (authResult.rateLimitResponse) {
|
|
75
|
+
return authResult.rateLimitResponse as NextResponse
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
// Check required permissions
|
|
79
|
+
const hasPermission =
|
|
80
|
+
authResult.type === 'session' ||
|
|
81
|
+
(authResult.type === 'api-key' && hasRequiredScope(authResult, 'users:read'))
|
|
82
|
+
|
|
83
|
+
if (!hasPermission) {
|
|
84
|
+
const response = createApiError(
|
|
85
|
+
'Insufficient permissions. Admin access required for user metadata.',
|
|
86
|
+
403
|
|
87
|
+
)
|
|
88
|
+
return addCorsHeaders(response)
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
const { id, key } = await params
|
|
92
|
+
|
|
93
|
+
// Validate parameters
|
|
94
|
+
if (!id || id.trim() === '') {
|
|
95
|
+
const response = createApiError('User ID is required', 400, null, 'MISSING_USER_ID')
|
|
96
|
+
return addCorsHeaders(response)
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
if (!key || key.trim() === '') {
|
|
100
|
+
const response = createApiError('Metadata key is required', 400, null, 'MISSING_META_KEY')
|
|
101
|
+
return addCorsHeaders(response)
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
// Validate key length
|
|
105
|
+
if (key.length > 100) {
|
|
106
|
+
const response = createApiError(
|
|
107
|
+
'Metadata key too long (max 100 characters)',
|
|
108
|
+
400,
|
|
109
|
+
null,
|
|
110
|
+
'INVALID_META_KEY'
|
|
111
|
+
)
|
|
112
|
+
return addCorsHeaders(response)
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
// Get user metadata value using UserService
|
|
116
|
+
const metaValue = await UserService.getUserMeta(id, key, authResult.user.id)
|
|
117
|
+
|
|
118
|
+
// Return 404 if metadata key doesn't exist
|
|
119
|
+
if (metaValue === null || metaValue === undefined) {
|
|
120
|
+
const response = createApiError(
|
|
121
|
+
`Metadata key '${key}' not found for user`,
|
|
122
|
+
404,
|
|
123
|
+
null,
|
|
124
|
+
'META_NOT_FOUND'
|
|
125
|
+
)
|
|
126
|
+
return addCorsHeaders(response)
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
// Return metadata value
|
|
130
|
+
const response = createApiResponse({
|
|
131
|
+
key,
|
|
132
|
+
value: metaValue,
|
|
133
|
+
})
|
|
134
|
+
return addCorsHeaders(response)
|
|
135
|
+
} catch (error) {
|
|
136
|
+
const response = createApiError(
|
|
137
|
+
'Failed to fetch user metadata',
|
|
138
|
+
500,
|
|
139
|
+
error instanceof Error ? error.message : undefined
|
|
140
|
+
)
|
|
141
|
+
return addCorsHeaders(response)
|
|
142
|
+
}
|
|
143
|
+
})
|
|
144
|
+
|
|
145
|
+
/**
|
|
146
|
+
* PUT /api/v1/users/:id/meta/:key
|
|
147
|
+
* Create or update a specific user metadata value
|
|
148
|
+
*
|
|
149
|
+
* Body:
|
|
150
|
+
* {
|
|
151
|
+
* value: any, // Required - The metadata value (any JSON type)
|
|
152
|
+
* isPublic?: boolean, // Optional - Whether metadata is public (default: false)
|
|
153
|
+
* isSearchable?: boolean, // Optional - Whether metadata is searchable (default: false)
|
|
154
|
+
* dataType?: string // Optional - Data type hint (default: "json")
|
|
155
|
+
* }
|
|
156
|
+
*
|
|
157
|
+
* @example
|
|
158
|
+
* PUT /api/v1/users/user-123/meta/theme
|
|
159
|
+
* Body: { value: "dark", isPublic: false }
|
|
160
|
+
* Response: { success: true, data: { key: "theme", value: "dark", updated: true } }
|
|
161
|
+
*/
|
|
162
|
+
export const PUT = withApiLogging(async (
|
|
163
|
+
req: NextRequest,
|
|
164
|
+
{ params }: { params: Promise<{ id: string; key: string }> }
|
|
165
|
+
): Promise<NextResponse> => {
|
|
166
|
+
try {
|
|
167
|
+
// Authenticate using dual auth
|
|
168
|
+
const authResult = await authenticateRequest(req)
|
|
169
|
+
|
|
170
|
+
if (!authResult.success || !authResult.user) {
|
|
171
|
+
return NextResponse.json(
|
|
172
|
+
createApiError('Authentication required', 401, null, 'AUTHENTICATION_FAILED'),
|
|
173
|
+
{ status: 401 }
|
|
174
|
+
)
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
if (authResult.rateLimitResponse) {
|
|
178
|
+
return authResult.rateLimitResponse as NextResponse
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
// Check required permissions
|
|
182
|
+
const hasPermission =
|
|
183
|
+
authResult.type === 'session' ||
|
|
184
|
+
(authResult.type === 'api-key' && hasRequiredScope(authResult, 'users:write'))
|
|
185
|
+
|
|
186
|
+
if (!hasPermission) {
|
|
187
|
+
const response = createApiError(
|
|
188
|
+
'Insufficient permissions. Admin access required for user metadata.',
|
|
189
|
+
403
|
|
190
|
+
)
|
|
191
|
+
return addCorsHeaders(response)
|
|
192
|
+
}
|
|
193
|
+
|
|
194
|
+
const { id, key } = await params
|
|
195
|
+
|
|
196
|
+
// Validate parameters
|
|
197
|
+
if (!id || id.trim() === '') {
|
|
198
|
+
const response = createApiError('User ID is required', 400, null, 'MISSING_USER_ID')
|
|
199
|
+
return addCorsHeaders(response)
|
|
200
|
+
}
|
|
201
|
+
|
|
202
|
+
if (!key || key.trim() === '') {
|
|
203
|
+
const response = createApiError('Metadata key is required', 400, null, 'MISSING_META_KEY')
|
|
204
|
+
return addCorsHeaders(response)
|
|
205
|
+
}
|
|
206
|
+
|
|
207
|
+
// Validate key length
|
|
208
|
+
if (key.length > 100) {
|
|
209
|
+
const response = createApiError(
|
|
210
|
+
'Metadata key too long (max 100 characters)',
|
|
211
|
+
400,
|
|
212
|
+
null,
|
|
213
|
+
'INVALID_META_KEY'
|
|
214
|
+
)
|
|
215
|
+
return addCorsHeaders(response)
|
|
216
|
+
}
|
|
217
|
+
|
|
218
|
+
// Parse and validate request body
|
|
219
|
+
const body = await req.json()
|
|
220
|
+
const validatedData = metadataValueSchema.parse(body)
|
|
221
|
+
|
|
222
|
+
// Validate value size (max 1MB)
|
|
223
|
+
const jsonString = JSON.stringify(validatedData.value)
|
|
224
|
+
if (new TextEncoder().encode(jsonString).length > 1048576) {
|
|
225
|
+
const response = createApiError(
|
|
226
|
+
'Metadata value too large (max 1MB)',
|
|
227
|
+
400,
|
|
228
|
+
null,
|
|
229
|
+
'VALUE_TOO_LARGE'
|
|
230
|
+
)
|
|
231
|
+
return addCorsHeaders(response)
|
|
232
|
+
}
|
|
233
|
+
|
|
234
|
+
// Update metadata using UserService
|
|
235
|
+
await UserService.updateUserMeta(
|
|
236
|
+
id,
|
|
237
|
+
key,
|
|
238
|
+
validatedData.value,
|
|
239
|
+
authResult.user.id,
|
|
240
|
+
{
|
|
241
|
+
isPublic: validatedData.isPublic,
|
|
242
|
+
isSearchable: validatedData.isSearchable,
|
|
243
|
+
dataType: validatedData.dataType,
|
|
244
|
+
}
|
|
245
|
+
)
|
|
246
|
+
|
|
247
|
+
// Return success response
|
|
248
|
+
const response = createApiResponse({
|
|
249
|
+
key,
|
|
250
|
+
value: validatedData.value,
|
|
251
|
+
updated: true,
|
|
252
|
+
})
|
|
253
|
+
return addCorsHeaders(response)
|
|
254
|
+
} catch (error) {
|
|
255
|
+
// Handle validation errors
|
|
256
|
+
if (error instanceof z.ZodError) {
|
|
257
|
+
const response = createApiError(
|
|
258
|
+
'Invalid request body',
|
|
259
|
+
400,
|
|
260
|
+
error.issues,
|
|
261
|
+
'VALIDATION_ERROR'
|
|
262
|
+
)
|
|
263
|
+
return addCorsHeaders(response)
|
|
264
|
+
}
|
|
265
|
+
|
|
266
|
+
const response = createApiError(
|
|
267
|
+
'Failed to update user metadata',
|
|
268
|
+
500,
|
|
269
|
+
error instanceof Error ? error.message : undefined
|
|
270
|
+
)
|
|
271
|
+
return addCorsHeaders(response)
|
|
272
|
+
}
|
|
273
|
+
})
|
|
274
|
+
|
|
275
|
+
/**
|
|
276
|
+
* PATCH /api/v1/users/:id/meta/:key
|
|
277
|
+
* Alias for PUT - Create or update a specific user metadata value
|
|
278
|
+
* Some clients prefer PATCH for partial updates
|
|
279
|
+
*/
|
|
280
|
+
export const PATCH = PUT
|
|
281
|
+
|
|
282
|
+
/**
|
|
283
|
+
* DELETE /api/v1/users/:id/meta/:key
|
|
284
|
+
* Delete a specific user metadata value
|
|
285
|
+
*
|
|
286
|
+
* @example
|
|
287
|
+
* DELETE /api/v1/users/user-123/meta/theme
|
|
288
|
+
* Response: { success: true, data: { key: "theme", deleted: true } }
|
|
289
|
+
*/
|
|
290
|
+
export const DELETE = withApiLogging(async (
|
|
291
|
+
req: NextRequest,
|
|
292
|
+
{ params }: { params: Promise<{ id: string; key: string }> }
|
|
293
|
+
): Promise<NextResponse> => {
|
|
294
|
+
try {
|
|
295
|
+
// Authenticate using dual auth
|
|
296
|
+
const authResult = await authenticateRequest(req)
|
|
297
|
+
|
|
298
|
+
if (!authResult.success || !authResult.user) {
|
|
299
|
+
return NextResponse.json(
|
|
300
|
+
createApiError('Authentication required', 401, null, 'AUTHENTICATION_FAILED'),
|
|
301
|
+
{ status: 401 }
|
|
302
|
+
)
|
|
303
|
+
}
|
|
304
|
+
|
|
305
|
+
if (authResult.rateLimitResponse) {
|
|
306
|
+
return authResult.rateLimitResponse as NextResponse
|
|
307
|
+
}
|
|
308
|
+
|
|
309
|
+
// Check required permissions
|
|
310
|
+
const hasPermission =
|
|
311
|
+
authResult.type === 'session' ||
|
|
312
|
+
(authResult.type === 'api-key' && hasRequiredScope(authResult, 'users:write'))
|
|
313
|
+
|
|
314
|
+
if (!hasPermission) {
|
|
315
|
+
const response = createApiError(
|
|
316
|
+
'Insufficient permissions. Admin access required for user metadata.',
|
|
317
|
+
403
|
|
318
|
+
)
|
|
319
|
+
return addCorsHeaders(response)
|
|
320
|
+
}
|
|
321
|
+
|
|
322
|
+
const { id, key } = await params
|
|
323
|
+
|
|
324
|
+
// Validate parameters
|
|
325
|
+
if (!id || id.trim() === '') {
|
|
326
|
+
const response = createApiError('User ID is required', 400, null, 'MISSING_USER_ID')
|
|
327
|
+
return addCorsHeaders(response)
|
|
328
|
+
}
|
|
329
|
+
|
|
330
|
+
if (!key || key.trim() === '') {
|
|
331
|
+
const response = createApiError('Metadata key is required', 400, null, 'MISSING_META_KEY')
|
|
332
|
+
return addCorsHeaders(response)
|
|
333
|
+
}
|
|
334
|
+
|
|
335
|
+
// Validate key length
|
|
336
|
+
if (key.length > 100) {
|
|
337
|
+
const response = createApiError(
|
|
338
|
+
'Metadata key too long (max 100 characters)',
|
|
339
|
+
400,
|
|
340
|
+
null,
|
|
341
|
+
'INVALID_META_KEY'
|
|
342
|
+
)
|
|
343
|
+
return addCorsHeaders(response)
|
|
344
|
+
}
|
|
345
|
+
|
|
346
|
+
// Delete metadata using UserService
|
|
347
|
+
await UserService.deleteUserMeta(id, key, authResult.user.id)
|
|
348
|
+
|
|
349
|
+
// Return success response
|
|
350
|
+
const response = createApiResponse({
|
|
351
|
+
key,
|
|
352
|
+
deleted: true,
|
|
353
|
+
})
|
|
354
|
+
return addCorsHeaders(response)
|
|
355
|
+
} catch (error) {
|
|
356
|
+
const response = createApiError(
|
|
357
|
+
'Failed to delete user metadata',
|
|
358
|
+
500,
|
|
359
|
+
error instanceof Error ? error.message : undefined
|
|
360
|
+
)
|
|
361
|
+
return addCorsHeaders(response)
|
|
362
|
+
}
|
|
363
|
+
})
|
|
@@ -0,0 +1,302 @@
|
|
|
1
|
+
import { NextRequest, NextResponse } from 'next/server';
|
|
2
|
+
import { queryOneWithRLS, mutateWithRLS } from '@nextsparkjs/core/lib/db';
|
|
3
|
+
import {
|
|
4
|
+
createApiResponse,
|
|
5
|
+
createApiError,
|
|
6
|
+
withApiLogging,
|
|
7
|
+
handleCorsPreflightRequest,
|
|
8
|
+
addCorsHeaders,
|
|
9
|
+
parseMetaParams,
|
|
10
|
+
includeEntityMetadataForSingle,
|
|
11
|
+
handleEntityMetadataInResponse,
|
|
12
|
+
processEntityMetadata
|
|
13
|
+
} from '@nextsparkjs/core/lib/api/helpers';
|
|
14
|
+
import { authenticateRequest, hasRequiredScope } from '@nextsparkjs/core/lib/api/auth/dual-auth';
|
|
15
|
+
import { z } from 'zod';
|
|
16
|
+
|
|
17
|
+
const updateUserSchema = z.object({
|
|
18
|
+
firstName: z.string().min(1).optional(),
|
|
19
|
+
lastName: z.string().optional(),
|
|
20
|
+
language: z.string().optional(),
|
|
21
|
+
role: z.enum(['member', 'superadmin']).optional(),
|
|
22
|
+
metas: z.record(z.string(), z.any()).optional()
|
|
23
|
+
});
|
|
24
|
+
|
|
25
|
+
// Handle CORS preflight
|
|
26
|
+
export async function OPTIONS() {
|
|
27
|
+
return handleCorsPreflightRequest();
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
// GET /api/v1/users/:id - Get specific user
|
|
31
|
+
export const GET = withApiLogging(async (
|
|
32
|
+
req: NextRequest,
|
|
33
|
+
{ params }: { params: Promise<{ id: string }> }
|
|
34
|
+
): Promise<NextResponse> => {
|
|
35
|
+
try {
|
|
36
|
+
// Authenticate using dual auth
|
|
37
|
+
const authResult = await authenticateRequest(req);
|
|
38
|
+
|
|
39
|
+
if (!authResult.success) {
|
|
40
|
+
return NextResponse.json(
|
|
41
|
+
{ success: false, error: 'Authentication required', code: 'AUTHENTICATION_FAILED' },
|
|
42
|
+
{ status: 401 }
|
|
43
|
+
);
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
if (authResult.rateLimitResponse) {
|
|
47
|
+
return authResult.rateLimitResponse as NextResponse;
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
// Check required permissions - session users need admin role, API key users need specific scope
|
|
51
|
+
const hasPermission = authResult.type === 'session' ||
|
|
52
|
+
(authResult.type === 'api-key' && hasRequiredScope(authResult, 'users:read'));
|
|
53
|
+
|
|
54
|
+
if (!hasPermission) {
|
|
55
|
+
const response = createApiError('Insufficient permissions. Admin access required for user management.', 403);
|
|
56
|
+
return addCorsHeaders(response);
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
const { id } = await params;
|
|
60
|
+
|
|
61
|
+
// Validate that id is not empty
|
|
62
|
+
if (!id || id.trim() === '') {
|
|
63
|
+
const response = createApiError('User ID or email is required', 400, null, 'MISSING_IDENTIFIER');
|
|
64
|
+
return addCorsHeaders(response);
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
// Search by ID or email
|
|
68
|
+
const user = await queryOneWithRLS(
|
|
69
|
+
'SELECT id, email, name, "firstName", "lastName", image, country, timezone, language, role, "emailVerified", "createdAt", "updatedAt" FROM "users" WHERE id = $1 OR email = $1',
|
|
70
|
+
[id],
|
|
71
|
+
authResult.user!.id
|
|
72
|
+
);
|
|
73
|
+
|
|
74
|
+
if (!user) {
|
|
75
|
+
const response = createApiError('User not found', 404, null, 'USER_NOT_FOUND');
|
|
76
|
+
return addCorsHeaders(response);
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
// Handle metadata if requested (usando helper compartido)
|
|
80
|
+
const metaParams = parseMetaParams(req);
|
|
81
|
+
const userWithMeta = await includeEntityMetadataForSingle('user', user as { id: string }, metaParams, authResult.user!.id);
|
|
82
|
+
|
|
83
|
+
const response = createApiResponse(userWithMeta);
|
|
84
|
+
return addCorsHeaders(response);
|
|
85
|
+
} catch (error) {
|
|
86
|
+
console.error('Error fetching user:', error);
|
|
87
|
+
const response = createApiError('Internal server error', 500);
|
|
88
|
+
return addCorsHeaders(response);
|
|
89
|
+
}
|
|
90
|
+
});
|
|
91
|
+
|
|
92
|
+
// PATCH /api/v1/users/:id - Update user
|
|
93
|
+
export const PATCH = withApiLogging(async (
|
|
94
|
+
req: NextRequest,
|
|
95
|
+
{ params }: { params: Promise<{ id: string }> }
|
|
96
|
+
): Promise<NextResponse> => {
|
|
97
|
+
try {
|
|
98
|
+
// Authenticate using dual auth
|
|
99
|
+
const authResult = await authenticateRequest(req);
|
|
100
|
+
|
|
101
|
+
if (!authResult.success) {
|
|
102
|
+
return NextResponse.json(
|
|
103
|
+
{ success: false, error: 'Authentication required', code: 'AUTHENTICATION_FAILED' },
|
|
104
|
+
{ status: 401 }
|
|
105
|
+
);
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
if (authResult.rateLimitResponse) {
|
|
109
|
+
return authResult.rateLimitResponse as NextResponse;
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
// Check required permissions - session users need admin role, API key users need specific scope
|
|
113
|
+
const hasPermission = authResult.type === 'session' ||
|
|
114
|
+
(authResult.type === 'api-key' && hasRequiredScope(authResult, 'users:write'));
|
|
115
|
+
|
|
116
|
+
if (!hasPermission) {
|
|
117
|
+
const response = createApiError('Insufficient permissions. Admin access required for user management.', 403);
|
|
118
|
+
return addCorsHeaders(response);
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
const { id } = await params;
|
|
122
|
+
|
|
123
|
+
// Validate that id is not empty
|
|
124
|
+
if (!id || id.trim() === '') {
|
|
125
|
+
const response = createApiError('User ID or email is required', 400, null, 'MISSING_IDENTIFIER');
|
|
126
|
+
return addCorsHeaders(response);
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
const body = await req.json();
|
|
130
|
+
const { metas, ...userData } = body;
|
|
131
|
+
const validatedData = updateUserSchema.parse(userData);
|
|
132
|
+
|
|
133
|
+
// Build dynamic update query
|
|
134
|
+
const updates = [];
|
|
135
|
+
const values = [];
|
|
136
|
+
let paramCount = 1;
|
|
137
|
+
|
|
138
|
+
if (validatedData.firstName !== undefined) {
|
|
139
|
+
updates.push(`"firstName" = $${paramCount++}`);
|
|
140
|
+
values.push(validatedData.firstName);
|
|
141
|
+
}
|
|
142
|
+
|
|
143
|
+
if (validatedData.lastName !== undefined) {
|
|
144
|
+
updates.push(`"lastName" = $${paramCount++}`);
|
|
145
|
+
values.push(validatedData.lastName);
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
if (validatedData.language !== undefined) {
|
|
149
|
+
updates.push(`language = $${paramCount++}`);
|
|
150
|
+
values.push(validatedData.language);
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
if (validatedData.role !== undefined) {
|
|
154
|
+
updates.push(`role = $${paramCount++}`);
|
|
155
|
+
values.push(validatedData.role);
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
// Verificar si hay algo para actualizar (campos de entidad O metadatos)
|
|
159
|
+
const hasEntityFieldsToUpdate = updates.length > 0;
|
|
160
|
+
const hasMetadataToUpdate = metas && typeof metas === 'object' && Object.keys(metas).length > 0;
|
|
161
|
+
|
|
162
|
+
if (!hasEntityFieldsToUpdate && !hasMetadataToUpdate) {
|
|
163
|
+
const response = createApiError('No fields to update', 400, null, 'NO_FIELDS');
|
|
164
|
+
return addCorsHeaders(response);
|
|
165
|
+
}
|
|
166
|
+
|
|
167
|
+
let updatedUser;
|
|
168
|
+
|
|
169
|
+
if (hasEntityFieldsToUpdate) {
|
|
170
|
+
// Solo actualizar campos de entidad si hay campos para actualizar
|
|
171
|
+
updates.push(`"updatedAt" = CURRENT_TIMESTAMP`);
|
|
172
|
+
values.push(id);
|
|
173
|
+
|
|
174
|
+
const query = `
|
|
175
|
+
UPDATE "users"
|
|
176
|
+
SET ${updates.join(", ")}
|
|
177
|
+
WHERE id = $${paramCount} OR email = $${paramCount}
|
|
178
|
+
RETURNING id, email, name, "firstName", "lastName", image, country, timezone, language, role, "emailVerified", "createdAt", "updatedAt"
|
|
179
|
+
`;
|
|
180
|
+
|
|
181
|
+
const result = await mutateWithRLS(query, values, authResult.user!.id);
|
|
182
|
+
|
|
183
|
+
if (result.rows.length === 0) {
|
|
184
|
+
const response = createApiError('User not found', 404, null, 'USER_NOT_FOUND');
|
|
185
|
+
return addCorsHeaders(response);
|
|
186
|
+
}
|
|
187
|
+
|
|
188
|
+
updatedUser = result.rows[0];
|
|
189
|
+
} else {
|
|
190
|
+
// Solo metadata a actualizar - obtener usuario existente
|
|
191
|
+
const user = await queryOneWithRLS(
|
|
192
|
+
'SELECT id, email, name, "firstName", "lastName", image, country, timezone, language, role, "emailVerified", "createdAt", "updatedAt" FROM "users" WHERE id = $1 OR email = $1',
|
|
193
|
+
[id],
|
|
194
|
+
authResult.user!.id
|
|
195
|
+
);
|
|
196
|
+
|
|
197
|
+
if (!user) {
|
|
198
|
+
const response = createApiError('User not found', 404, null, 'USER_NOT_FOUND');
|
|
199
|
+
return addCorsHeaders(response);
|
|
200
|
+
}
|
|
201
|
+
|
|
202
|
+
updatedUser = user;
|
|
203
|
+
}
|
|
204
|
+
|
|
205
|
+
// Handle metadata if provided (usando helper compartido)
|
|
206
|
+
const metadataWasProvided = metas && typeof metas === 'object' && Object.keys(metas).length > 0;
|
|
207
|
+
|
|
208
|
+
if (metadataWasProvided) {
|
|
209
|
+
await processEntityMetadata('user', (updatedUser as { id: string }).id, metas, authResult.user!.id);
|
|
210
|
+
}
|
|
211
|
+
|
|
212
|
+
// Crear respuesta según criterio: incluir metadata solo si se envió en el payload
|
|
213
|
+
const responseData = await handleEntityMetadataInResponse('user', updatedUser as { id: string }, metadataWasProvided, authResult.user!.id);
|
|
214
|
+
|
|
215
|
+
const response = createApiResponse(responseData);
|
|
216
|
+
return addCorsHeaders(response);
|
|
217
|
+
} catch (error) {
|
|
218
|
+
if (error instanceof z.ZodError) {
|
|
219
|
+
const response = createApiError('Validation error', 400, error.issues, 'VALIDATION_ERROR');
|
|
220
|
+
return addCorsHeaders(response);
|
|
221
|
+
}
|
|
222
|
+
|
|
223
|
+
console.error('Error updating user:', error);
|
|
224
|
+
const response = createApiError('Internal server error', 500);
|
|
225
|
+
return addCorsHeaders(response);
|
|
226
|
+
}
|
|
227
|
+
});
|
|
228
|
+
|
|
229
|
+
// DELETE /api/v1/users/:id - Delete user
|
|
230
|
+
export const DELETE = withApiLogging(async (
|
|
231
|
+
req: NextRequest,
|
|
232
|
+
{ params }: { params: Promise<{ id: string }> }
|
|
233
|
+
): Promise<NextResponse> => {
|
|
234
|
+
try {
|
|
235
|
+
// Authenticate using dual auth
|
|
236
|
+
const authResult = await authenticateRequest(req);
|
|
237
|
+
|
|
238
|
+
if (!authResult.success) {
|
|
239
|
+
return NextResponse.json(
|
|
240
|
+
{ success: false, error: 'Authentication required', code: 'AUTHENTICATION_FAILED' },
|
|
241
|
+
{ status: 401 }
|
|
242
|
+
);
|
|
243
|
+
}
|
|
244
|
+
|
|
245
|
+
if (authResult.rateLimitResponse) {
|
|
246
|
+
return authResult.rateLimitResponse as NextResponse;
|
|
247
|
+
}
|
|
248
|
+
|
|
249
|
+
// Check required permissions - session users need admin role, API key users need specific scope
|
|
250
|
+
const hasPermission = authResult.type === 'session' ||
|
|
251
|
+
(authResult.type === 'api-key' && hasRequiredScope(authResult, 'users:delete'));
|
|
252
|
+
|
|
253
|
+
if (!hasPermission) {
|
|
254
|
+
const response = createApiError('Insufficient permissions. Admin access required for user management.', 403);
|
|
255
|
+
return addCorsHeaders(response);
|
|
256
|
+
}
|
|
257
|
+
|
|
258
|
+
const { id } = await params;
|
|
259
|
+
|
|
260
|
+
// Validate that id is not empty
|
|
261
|
+
if (!id || id.trim() === '') {
|
|
262
|
+
const response = createApiError('User ID or email is required', 400, null, 'MISSING_IDENTIFIER');
|
|
263
|
+
return addCorsHeaders(response);
|
|
264
|
+
}
|
|
265
|
+
|
|
266
|
+
// First, get the user to check if it exists and prevent self-deletion
|
|
267
|
+
const targetUser = await queryOneWithRLS(
|
|
268
|
+
'SELECT id, email FROM "users" WHERE id = $1 OR email = $1',
|
|
269
|
+
[id],
|
|
270
|
+
authResult.user!.id
|
|
271
|
+
);
|
|
272
|
+
|
|
273
|
+
if (!targetUser) {
|
|
274
|
+
const response = createApiError('User not found', 404, null, 'USER_NOT_FOUND');
|
|
275
|
+
return addCorsHeaders(response);
|
|
276
|
+
}
|
|
277
|
+
|
|
278
|
+
// Prevent self-deletion
|
|
279
|
+
if ((targetUser as Record<string, unknown>).id === authResult.user!.id) {
|
|
280
|
+
const response = createApiError('Cannot delete your own account via API', 403, null, 'SELF_DELETE_FORBIDDEN');
|
|
281
|
+
return addCorsHeaders(response);
|
|
282
|
+
}
|
|
283
|
+
|
|
284
|
+
const result = await mutateWithRLS(
|
|
285
|
+
'DELETE FROM "users" WHERE id = $1 RETURNING id',
|
|
286
|
+
[(targetUser as Record<string, unknown>).id],
|
|
287
|
+
authResult.user!.id
|
|
288
|
+
);
|
|
289
|
+
|
|
290
|
+
if (result.rows.length === 0) {
|
|
291
|
+
const response = createApiError('User not found', 404, null, 'USER_NOT_FOUND');
|
|
292
|
+
return addCorsHeaders(response);
|
|
293
|
+
}
|
|
294
|
+
|
|
295
|
+
const response = createApiResponse({ deleted: true, id });
|
|
296
|
+
return addCorsHeaders(response);
|
|
297
|
+
} catch (error) {
|
|
298
|
+
console.error('Error deleting user:', error);
|
|
299
|
+
const response = createApiError('Internal server error', 500);
|
|
300
|
+
return addCorsHeaders(response);
|
|
301
|
+
}
|
|
302
|
+
});
|