@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,540 @@
|
|
|
1
|
+
import { NextRequest, NextResponse } from 'next/server';
|
|
2
|
+
import { auth } from '@nextsparkjs/core/lib/auth';
|
|
3
|
+
import { queryWithRLS } from '@nextsparkjs/core/lib/db';
|
|
4
|
+
|
|
5
|
+
interface UserResult {
|
|
6
|
+
id: string;
|
|
7
|
+
firstName: string | null;
|
|
8
|
+
lastName: string | null;
|
|
9
|
+
email: string;
|
|
10
|
+
role: string;
|
|
11
|
+
emailVerified: boolean;
|
|
12
|
+
createdAt: string;
|
|
13
|
+
updatedAt: string;
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
interface TeamMembershipResult {
|
|
17
|
+
teamId: string;
|
|
18
|
+
teamName: string;
|
|
19
|
+
role: string;
|
|
20
|
+
joinedAt: string;
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
interface UserMetaResult {
|
|
24
|
+
id: string;
|
|
25
|
+
metaKey: string;
|
|
26
|
+
metaValue: unknown;
|
|
27
|
+
dataType: string | null;
|
|
28
|
+
isPublic: boolean;
|
|
29
|
+
isSearchable: boolean;
|
|
30
|
+
createdAt: string;
|
|
31
|
+
updatedAt: string;
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
interface RouteParams {
|
|
35
|
+
params: Promise<{ userId: string }>;
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
/**
|
|
39
|
+
* GET /api/superadmin/users/[userId]
|
|
40
|
+
*
|
|
41
|
+
* Retrieves detailed information about a specific user.
|
|
42
|
+
* Only accessible by superadmin or developer users.
|
|
43
|
+
*
|
|
44
|
+
* Returns:
|
|
45
|
+
* - user: User details
|
|
46
|
+
* - teams: Array of team memberships
|
|
47
|
+
* - stats: User statistics
|
|
48
|
+
*/
|
|
49
|
+
export async function GET(request: NextRequest, { params }: RouteParams) {
|
|
50
|
+
try {
|
|
51
|
+
// Get the current session using Better Auth
|
|
52
|
+
const session = await auth.api.getSession({
|
|
53
|
+
headers: request.headers
|
|
54
|
+
});
|
|
55
|
+
|
|
56
|
+
// Check if user is authenticated
|
|
57
|
+
if (!session?.user) {
|
|
58
|
+
return NextResponse.json(
|
|
59
|
+
{ error: 'Unauthorized - No session found' },
|
|
60
|
+
{ status: 401 }
|
|
61
|
+
);
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
// Check if user is superadmin or developer
|
|
65
|
+
if (session.user.role !== 'superadmin' && session.user.role !== 'developer') {
|
|
66
|
+
return NextResponse.json(
|
|
67
|
+
{ error: 'Forbidden - Superadmin or developer access required' },
|
|
68
|
+
{ status: 403 }
|
|
69
|
+
);
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
const { userId } = await params;
|
|
73
|
+
|
|
74
|
+
// Validate userId
|
|
75
|
+
if (!userId || typeof userId !== 'string') {
|
|
76
|
+
return NextResponse.json(
|
|
77
|
+
{ error: 'Invalid user ID' },
|
|
78
|
+
{ status: 400 }
|
|
79
|
+
);
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
// Query user details
|
|
83
|
+
const userQuery = `
|
|
84
|
+
SELECT
|
|
85
|
+
id,
|
|
86
|
+
"firstName",
|
|
87
|
+
"lastName",
|
|
88
|
+
email,
|
|
89
|
+
role,
|
|
90
|
+
"emailVerified",
|
|
91
|
+
"createdAt",
|
|
92
|
+
"updatedAt"
|
|
93
|
+
FROM "users"
|
|
94
|
+
WHERE id = $1
|
|
95
|
+
`;
|
|
96
|
+
|
|
97
|
+
// Query user's team memberships
|
|
98
|
+
const teamsQuery = `
|
|
99
|
+
SELECT
|
|
100
|
+
t.id as "teamId",
|
|
101
|
+
t.name as "teamName",
|
|
102
|
+
tm.role,
|
|
103
|
+
tm."createdAt" as "joinedAt"
|
|
104
|
+
FROM "team_members" tm
|
|
105
|
+
JOIN "teams" t ON tm."teamId" = t.id
|
|
106
|
+
WHERE tm."userId" = $1
|
|
107
|
+
ORDER BY tm."createdAt" DESC
|
|
108
|
+
`;
|
|
109
|
+
|
|
110
|
+
// Query for counts
|
|
111
|
+
const countsQuery = `
|
|
112
|
+
SELECT
|
|
113
|
+
(SELECT COUNT(*)::int FROM "team_members" WHERE "userId" = $1) as "totalTeams",
|
|
114
|
+
(SELECT COUNT(*)::int FROM "team_members" WHERE "userId" = $1 AND role = 'owner') as "ownedTeams"
|
|
115
|
+
`;
|
|
116
|
+
|
|
117
|
+
// Query for user metadata
|
|
118
|
+
const metasQuery = `
|
|
119
|
+
SELECT
|
|
120
|
+
id,
|
|
121
|
+
"metaKey",
|
|
122
|
+
"metaValue",
|
|
123
|
+
"dataType",
|
|
124
|
+
"isPublic",
|
|
125
|
+
"isSearchable",
|
|
126
|
+
"createdAt",
|
|
127
|
+
"updatedAt"
|
|
128
|
+
FROM "users_metas"
|
|
129
|
+
WHERE "userId" = $1
|
|
130
|
+
ORDER BY "metaKey"
|
|
131
|
+
`;
|
|
132
|
+
|
|
133
|
+
// Execute all queries in parallel
|
|
134
|
+
const [userResults, teamsResults, countsResults, metasResults] = await Promise.all([
|
|
135
|
+
queryWithRLS(userQuery, [userId], session.user.id) as Promise<UserResult[]>,
|
|
136
|
+
queryWithRLS(teamsQuery, [userId], session.user.id) as Promise<TeamMembershipResult[]>,
|
|
137
|
+
queryWithRLS(countsQuery, [userId], session.user.id) as Promise<{
|
|
138
|
+
totalTeams: number;
|
|
139
|
+
ownedTeams: number;
|
|
140
|
+
}[]>,
|
|
141
|
+
queryWithRLS(metasQuery, [userId], session.user.id) as Promise<UserMetaResult[]>
|
|
142
|
+
]);
|
|
143
|
+
|
|
144
|
+
// Check if user exists
|
|
145
|
+
if (!userResults || userResults.length === 0) {
|
|
146
|
+
return NextResponse.json(
|
|
147
|
+
{ error: 'User not found' },
|
|
148
|
+
{ status: 404 }
|
|
149
|
+
);
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
const user = userResults[0];
|
|
153
|
+
const counts = countsResults[0] || { totalTeams: 0, ownedTeams: 0 };
|
|
154
|
+
|
|
155
|
+
// Format user data
|
|
156
|
+
const formattedUser = {
|
|
157
|
+
id: user.id,
|
|
158
|
+
firstName: user.firstName || '',
|
|
159
|
+
lastName: user.lastName || '',
|
|
160
|
+
fullName: [user.firstName, user.lastName].filter(Boolean).join(' ') || user.email,
|
|
161
|
+
email: user.email,
|
|
162
|
+
role: user.role,
|
|
163
|
+
emailVerified: user.emailVerified,
|
|
164
|
+
createdAt: user.createdAt,
|
|
165
|
+
updatedAt: user.updatedAt
|
|
166
|
+
};
|
|
167
|
+
|
|
168
|
+
// Format team memberships
|
|
169
|
+
const formattedTeams = teamsResults.map(team => ({
|
|
170
|
+
teamId: team.teamId,
|
|
171
|
+
teamName: team.teamName,
|
|
172
|
+
role: team.role,
|
|
173
|
+
joinedAt: team.joinedAt
|
|
174
|
+
}));
|
|
175
|
+
|
|
176
|
+
// Format user metas
|
|
177
|
+
const formattedMetas = (metasResults || []).map(meta => ({
|
|
178
|
+
id: meta.id,
|
|
179
|
+
metaKey: meta.metaKey,
|
|
180
|
+
metaValue: meta.metaValue,
|
|
181
|
+
dataType: meta.dataType,
|
|
182
|
+
isPublic: meta.isPublic,
|
|
183
|
+
isSearchable: meta.isSearchable,
|
|
184
|
+
createdAt: meta.createdAt,
|
|
185
|
+
updatedAt: meta.updatedAt
|
|
186
|
+
}));
|
|
187
|
+
|
|
188
|
+
// Prepare response data
|
|
189
|
+
const responseData = {
|
|
190
|
+
user: formattedUser,
|
|
191
|
+
teams: formattedTeams,
|
|
192
|
+
userMetas: formattedMetas,
|
|
193
|
+
stats: {
|
|
194
|
+
totalTeams: counts.totalTeams,
|
|
195
|
+
ownedTeams: counts.ownedTeams
|
|
196
|
+
},
|
|
197
|
+
metadata: {
|
|
198
|
+
requestedBy: session.user.id,
|
|
199
|
+
requestedAt: new Date().toISOString(),
|
|
200
|
+
source: 'superadmin-api'
|
|
201
|
+
}
|
|
202
|
+
};
|
|
203
|
+
|
|
204
|
+
return NextResponse.json(responseData);
|
|
205
|
+
|
|
206
|
+
} catch (error) {
|
|
207
|
+
console.error('Error fetching user details:', error);
|
|
208
|
+
|
|
209
|
+
return NextResponse.json(
|
|
210
|
+
{
|
|
211
|
+
error: 'Internal server error',
|
|
212
|
+
message: 'Failed to retrieve user details'
|
|
213
|
+
},
|
|
214
|
+
{ status: 500 }
|
|
215
|
+
);
|
|
216
|
+
}
|
|
217
|
+
}
|
|
218
|
+
|
|
219
|
+
interface UserActionBody {
|
|
220
|
+
action: 'change-role' | 'suspend' | 'unsuspend' | 'verify-email';
|
|
221
|
+
role?: string;
|
|
222
|
+
}
|
|
223
|
+
|
|
224
|
+
/**
|
|
225
|
+
* PATCH /api/superadmin/users/[userId]
|
|
226
|
+
*
|
|
227
|
+
* Performs actions on a specific user.
|
|
228
|
+
* Only accessible by superadmin or developer users.
|
|
229
|
+
*
|
|
230
|
+
* Actions:
|
|
231
|
+
* - change-role: Change user's role (requires 'role' field)
|
|
232
|
+
* - suspend: Suspend the user (sets role to 'suspended')
|
|
233
|
+
* - unsuspend: Restore user's role to 'member'
|
|
234
|
+
* - verify-email: Manually verify user's email
|
|
235
|
+
*/
|
|
236
|
+
export async function PATCH(request: NextRequest, { params }: RouteParams) {
|
|
237
|
+
try {
|
|
238
|
+
// Get the current session using Better Auth
|
|
239
|
+
const session = await auth.api.getSession({
|
|
240
|
+
headers: request.headers
|
|
241
|
+
});
|
|
242
|
+
|
|
243
|
+
// Check if user is authenticated
|
|
244
|
+
if (!session?.user) {
|
|
245
|
+
return NextResponse.json(
|
|
246
|
+
{ error: 'Unauthorized - No session found' },
|
|
247
|
+
{ status: 401 }
|
|
248
|
+
);
|
|
249
|
+
}
|
|
250
|
+
|
|
251
|
+
// Check if user is superadmin or developer
|
|
252
|
+
if (session.user.role !== 'superadmin' && session.user.role !== 'developer') {
|
|
253
|
+
return NextResponse.json(
|
|
254
|
+
{ error: 'Forbidden - Superadmin or developer access required' },
|
|
255
|
+
{ status: 403 }
|
|
256
|
+
);
|
|
257
|
+
}
|
|
258
|
+
|
|
259
|
+
const { userId } = await params;
|
|
260
|
+
|
|
261
|
+
// Validate userId
|
|
262
|
+
if (!userId || typeof userId !== 'string') {
|
|
263
|
+
return NextResponse.json(
|
|
264
|
+
{ error: 'Invalid user ID' },
|
|
265
|
+
{ status: 400 }
|
|
266
|
+
);
|
|
267
|
+
}
|
|
268
|
+
|
|
269
|
+
// Prevent self-modification
|
|
270
|
+
if (userId === session.user.id) {
|
|
271
|
+
return NextResponse.json(
|
|
272
|
+
{ error: 'Cannot modify your own account' },
|
|
273
|
+
{ status: 400 }
|
|
274
|
+
);
|
|
275
|
+
}
|
|
276
|
+
|
|
277
|
+
// Parse request body
|
|
278
|
+
const body: UserActionBody = await request.json();
|
|
279
|
+
const { action, role } = body;
|
|
280
|
+
|
|
281
|
+
// Validate action
|
|
282
|
+
if (!action || !['change-role', 'suspend', 'unsuspend', 'verify-email'].includes(action)) {
|
|
283
|
+
return NextResponse.json(
|
|
284
|
+
{ error: 'Invalid action. Must be one of: change-role, suspend, unsuspend, verify-email' },
|
|
285
|
+
{ status: 400 }
|
|
286
|
+
);
|
|
287
|
+
}
|
|
288
|
+
|
|
289
|
+
// Check if target user exists and get their current role
|
|
290
|
+
const checkUserQuery = `
|
|
291
|
+
SELECT id, role FROM "users" WHERE id = $1
|
|
292
|
+
`;
|
|
293
|
+
const userCheck = await queryWithRLS(checkUserQuery, [userId], session.user.id) as { id: string; role: string }[];
|
|
294
|
+
|
|
295
|
+
if (!userCheck || userCheck.length === 0) {
|
|
296
|
+
return NextResponse.json(
|
|
297
|
+
{ error: 'User not found' },
|
|
298
|
+
{ status: 404 }
|
|
299
|
+
);
|
|
300
|
+
}
|
|
301
|
+
|
|
302
|
+
const targetUser = userCheck[0];
|
|
303
|
+
|
|
304
|
+
// Prevent modifying other superadmins
|
|
305
|
+
if (targetUser.role === 'superadmin') {
|
|
306
|
+
return NextResponse.json(
|
|
307
|
+
{ error: 'Cannot modify other superadmin accounts' },
|
|
308
|
+
{ status: 403 }
|
|
309
|
+
);
|
|
310
|
+
}
|
|
311
|
+
|
|
312
|
+
let updateQuery = '';
|
|
313
|
+
let updateParams: (string | boolean)[] = [];
|
|
314
|
+
|
|
315
|
+
switch (action) {
|
|
316
|
+
case 'change-role':
|
|
317
|
+
// Validate role
|
|
318
|
+
if (!role || !['member', 'colaborator', 'admin'].includes(role)) {
|
|
319
|
+
return NextResponse.json(
|
|
320
|
+
{ error: 'Invalid role. Must be one of: member, colaborator, admin' },
|
|
321
|
+
{ status: 400 }
|
|
322
|
+
);
|
|
323
|
+
}
|
|
324
|
+
updateQuery = `
|
|
325
|
+
UPDATE "users"
|
|
326
|
+
SET role = $1, "updatedAt" = NOW()
|
|
327
|
+
WHERE id = $2
|
|
328
|
+
RETURNING id, role
|
|
329
|
+
`;
|
|
330
|
+
updateParams = [role, userId];
|
|
331
|
+
break;
|
|
332
|
+
|
|
333
|
+
case 'suspend':
|
|
334
|
+
updateQuery = `
|
|
335
|
+
UPDATE "users"
|
|
336
|
+
SET role = 'suspended', "updatedAt" = NOW()
|
|
337
|
+
WHERE id = $1
|
|
338
|
+
RETURNING id, role
|
|
339
|
+
`;
|
|
340
|
+
updateParams = [userId];
|
|
341
|
+
break;
|
|
342
|
+
|
|
343
|
+
case 'unsuspend':
|
|
344
|
+
updateQuery = `
|
|
345
|
+
UPDATE "users"
|
|
346
|
+
SET role = 'member', "updatedAt" = NOW()
|
|
347
|
+
WHERE id = $1
|
|
348
|
+
RETURNING id, role
|
|
349
|
+
`;
|
|
350
|
+
updateParams = [userId];
|
|
351
|
+
break;
|
|
352
|
+
|
|
353
|
+
case 'verify-email':
|
|
354
|
+
updateQuery = `
|
|
355
|
+
UPDATE "users"
|
|
356
|
+
SET "emailVerified" = true, "updatedAt" = NOW()
|
|
357
|
+
WHERE id = $1
|
|
358
|
+
RETURNING id, "emailVerified"
|
|
359
|
+
`;
|
|
360
|
+
updateParams = [userId];
|
|
361
|
+
break;
|
|
362
|
+
}
|
|
363
|
+
|
|
364
|
+
// Execute update
|
|
365
|
+
const result = await queryWithRLS(updateQuery, updateParams, session.user.id) as { id: string }[];
|
|
366
|
+
|
|
367
|
+
if (!result || result.length === 0) {
|
|
368
|
+
return NextResponse.json(
|
|
369
|
+
{ error: 'Failed to update user' },
|
|
370
|
+
{ status: 500 }
|
|
371
|
+
);
|
|
372
|
+
}
|
|
373
|
+
|
|
374
|
+
return NextResponse.json({
|
|
375
|
+
success: true,
|
|
376
|
+
action,
|
|
377
|
+
userId,
|
|
378
|
+
message: `User ${action} successful`,
|
|
379
|
+
metadata: {
|
|
380
|
+
performedBy: session.user.id,
|
|
381
|
+
performedAt: new Date().toISOString(),
|
|
382
|
+
source: 'superadmin-api'
|
|
383
|
+
}
|
|
384
|
+
});
|
|
385
|
+
|
|
386
|
+
} catch (error) {
|
|
387
|
+
console.error('Error performing user action:', error);
|
|
388
|
+
|
|
389
|
+
return NextResponse.json(
|
|
390
|
+
{
|
|
391
|
+
error: 'Internal server error',
|
|
392
|
+
message: 'Failed to perform user action'
|
|
393
|
+
},
|
|
394
|
+
{ status: 500 }
|
|
395
|
+
);
|
|
396
|
+
}
|
|
397
|
+
}
|
|
398
|
+
|
|
399
|
+
/**
|
|
400
|
+
* DELETE /api/superadmin/users/[userId]
|
|
401
|
+
*
|
|
402
|
+
* Deletes a specific user.
|
|
403
|
+
* Only accessible by superadmin or developer users.
|
|
404
|
+
*
|
|
405
|
+
* This will:
|
|
406
|
+
* - Remove user from all teams
|
|
407
|
+
* - Delete user's personal team
|
|
408
|
+
* - Delete the user account
|
|
409
|
+
*/
|
|
410
|
+
export async function DELETE(request: NextRequest, { params }: RouteParams) {
|
|
411
|
+
try {
|
|
412
|
+
// Get the current session using Better Auth
|
|
413
|
+
const session = await auth.api.getSession({
|
|
414
|
+
headers: request.headers
|
|
415
|
+
});
|
|
416
|
+
|
|
417
|
+
// Check if user is authenticated
|
|
418
|
+
if (!session?.user) {
|
|
419
|
+
return NextResponse.json(
|
|
420
|
+
{ error: 'Unauthorized - No session found' },
|
|
421
|
+
{ status: 401 }
|
|
422
|
+
);
|
|
423
|
+
}
|
|
424
|
+
|
|
425
|
+
// Check if user is superadmin or developer
|
|
426
|
+
if (session.user.role !== 'superadmin' && session.user.role !== 'developer') {
|
|
427
|
+
return NextResponse.json(
|
|
428
|
+
{ error: 'Forbidden - Superadmin or developer access required' },
|
|
429
|
+
{ status: 403 }
|
|
430
|
+
);
|
|
431
|
+
}
|
|
432
|
+
|
|
433
|
+
const { userId } = await params;
|
|
434
|
+
|
|
435
|
+
// Validate userId
|
|
436
|
+
if (!userId || typeof userId !== 'string') {
|
|
437
|
+
return NextResponse.json(
|
|
438
|
+
{ error: 'Invalid user ID' },
|
|
439
|
+
{ status: 400 }
|
|
440
|
+
);
|
|
441
|
+
}
|
|
442
|
+
|
|
443
|
+
// Prevent self-deletion
|
|
444
|
+
if (userId === session.user.id) {
|
|
445
|
+
return NextResponse.json(
|
|
446
|
+
{ error: 'Cannot delete your own account' },
|
|
447
|
+
{ status: 400 }
|
|
448
|
+
);
|
|
449
|
+
}
|
|
450
|
+
|
|
451
|
+
// Check if target user exists and get their role
|
|
452
|
+
const checkUserQuery = `
|
|
453
|
+
SELECT id, role, email FROM "users" WHERE id = $1
|
|
454
|
+
`;
|
|
455
|
+
const userCheck = await queryWithRLS(checkUserQuery, [userId], session.user.id) as { id: string; role: string; email: string }[];
|
|
456
|
+
|
|
457
|
+
if (!userCheck || userCheck.length === 0) {
|
|
458
|
+
return NextResponse.json(
|
|
459
|
+
{ error: 'User not found' },
|
|
460
|
+
{ status: 404 }
|
|
461
|
+
);
|
|
462
|
+
}
|
|
463
|
+
|
|
464
|
+
const targetUser = userCheck[0];
|
|
465
|
+
|
|
466
|
+
// Prevent deleting other superadmins
|
|
467
|
+
if (targetUser.role === 'superadmin') {
|
|
468
|
+
return NextResponse.json(
|
|
469
|
+
{ error: 'Cannot delete other superadmin accounts' },
|
|
470
|
+
{ status: 403 }
|
|
471
|
+
);
|
|
472
|
+
}
|
|
473
|
+
|
|
474
|
+
// Delete in order to respect foreign key constraints:
|
|
475
|
+
// 1. Delete user's team memberships
|
|
476
|
+
await queryWithRLS(
|
|
477
|
+
`DELETE FROM "team_members" WHERE "userId" = $1`,
|
|
478
|
+
[userId],
|
|
479
|
+
session.user.id
|
|
480
|
+
);
|
|
481
|
+
|
|
482
|
+
// 2. Delete user's owned teams (and their members)
|
|
483
|
+
// For simplicity, we'll delete all teams owned by this user (in production, you'd want to transfer ownership)
|
|
484
|
+
await queryWithRLS(
|
|
485
|
+
`DELETE FROM "team_members" WHERE "teamId" IN (SELECT id FROM "teams" WHERE "ownerId" = $1)`,
|
|
486
|
+
[userId],
|
|
487
|
+
session.user.id
|
|
488
|
+
);
|
|
489
|
+
await queryWithRLS(
|
|
490
|
+
`DELETE FROM "teams" WHERE "ownerId" = $1`,
|
|
491
|
+
[userId],
|
|
492
|
+
session.user.id
|
|
493
|
+
);
|
|
494
|
+
|
|
495
|
+
// 4. Delete user sessions
|
|
496
|
+
await queryWithRLS(
|
|
497
|
+
`DELETE FROM "session" WHERE "userId" = $1`,
|
|
498
|
+
[userId],
|
|
499
|
+
session.user.id
|
|
500
|
+
);
|
|
501
|
+
|
|
502
|
+
// 5. Delete user accounts (OAuth)
|
|
503
|
+
await queryWithRLS(
|
|
504
|
+
`DELETE FROM "account" WHERE "userId" = $1`,
|
|
505
|
+
[userId],
|
|
506
|
+
session.user.id
|
|
507
|
+
);
|
|
508
|
+
|
|
509
|
+
// 6. Delete the user
|
|
510
|
+
await queryWithRLS(
|
|
511
|
+
`DELETE FROM "users" WHERE id = $1`,
|
|
512
|
+
[userId],
|
|
513
|
+
session.user.id
|
|
514
|
+
);
|
|
515
|
+
|
|
516
|
+
return NextResponse.json({
|
|
517
|
+
success: true,
|
|
518
|
+
action: 'delete',
|
|
519
|
+
userId,
|
|
520
|
+
userEmail: targetUser.email,
|
|
521
|
+
message: 'User deleted successfully',
|
|
522
|
+
metadata: {
|
|
523
|
+
performedBy: session.user.id,
|
|
524
|
+
performedAt: new Date().toISOString(),
|
|
525
|
+
source: 'superadmin-api'
|
|
526
|
+
}
|
|
527
|
+
});
|
|
528
|
+
|
|
529
|
+
} catch (error) {
|
|
530
|
+
console.error('Error deleting user:', error);
|
|
531
|
+
|
|
532
|
+
return NextResponse.json(
|
|
533
|
+
{
|
|
534
|
+
error: 'Internal server error',
|
|
535
|
+
message: 'Failed to delete user'
|
|
536
|
+
},
|
|
537
|
+
{ status: 500 }
|
|
538
|
+
);
|
|
539
|
+
}
|
|
540
|
+
}
|