@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,81 @@
|
|
|
1
|
+
import { Card, CardContent, CardDescription, CardHeader, CardTitle } from '@nextsparkjs/core/components/ui/card'
|
|
2
|
+
import { DOCS_REGISTRY } from '@nextsparkjs/registries/docs-registry'
|
|
3
|
+
import { sel } from '@nextsparkjs/core/selectors'
|
|
4
|
+
import { BookOpen, Folder, FileText, ArrowRight } from 'lucide-react'
|
|
5
|
+
import Link from 'next/link'
|
|
6
|
+
import type { Metadata } from 'next'
|
|
7
|
+
|
|
8
|
+
export const metadata: Metadata = {
|
|
9
|
+
title: 'Documentation',
|
|
10
|
+
description: 'Complete documentation for the application'
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
export default function DocsPage() {
|
|
14
|
+
const sections = DOCS_REGISTRY.public
|
|
15
|
+
|
|
16
|
+
return (
|
|
17
|
+
<div className="container max-w-7xl mx-auto px-4 py-8">
|
|
18
|
+
{/* Hero Section */}
|
|
19
|
+
<div className="text-center mb-12">
|
|
20
|
+
<div className="inline-flex items-center gap-2 bg-primary/10 text-primary px-3 py-1 rounded-full text-sm font-medium mb-4">
|
|
21
|
+
<BookOpen className="h-4 w-4" />
|
|
22
|
+
Documentation
|
|
23
|
+
</div>
|
|
24
|
+
<h1 className="text-4xl font-bold mb-4">
|
|
25
|
+
Documentation
|
|
26
|
+
</h1>
|
|
27
|
+
<p className="text-xl text-muted-foreground max-w-2xl mx-auto">
|
|
28
|
+
Guides and reference documentation to help you get the most out of the application.
|
|
29
|
+
</p>
|
|
30
|
+
</div>
|
|
31
|
+
|
|
32
|
+
{/* Documentation Sections */}
|
|
33
|
+
{sections.length > 0 ? (
|
|
34
|
+
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-6">
|
|
35
|
+
{sections.map((section) => (
|
|
36
|
+
<Card key={section.slug} className="h-full hover:shadow-lg transition-shadow" data-cy={sel('public.docs.sectionCard', { slug: section.slug })}>
|
|
37
|
+
<CardHeader>
|
|
38
|
+
<div className="flex items-center gap-3 mb-3">
|
|
39
|
+
<div className="p-2 bg-primary/10 rounded-lg text-primary">
|
|
40
|
+
<Folder className="h-6 w-6" />
|
|
41
|
+
</div>
|
|
42
|
+
<div>
|
|
43
|
+
<CardTitle className="text-lg">{section.title}</CardTitle>
|
|
44
|
+
</div>
|
|
45
|
+
</div>
|
|
46
|
+
<CardDescription>
|
|
47
|
+
{section.pages.length} {section.pages.length === 1 ? 'page' : 'pages'}
|
|
48
|
+
</CardDescription>
|
|
49
|
+
</CardHeader>
|
|
50
|
+
<CardContent>
|
|
51
|
+
<ul className="space-y-2">
|
|
52
|
+
{section.pages.map((page) => (
|
|
53
|
+
<li key={page.slug}>
|
|
54
|
+
<Link
|
|
55
|
+
href={`/docs/${section.slug}/${page.slug}`}
|
|
56
|
+
className="text-sm text-muted-foreground hover:text-foreground transition-colors flex items-center gap-2 group"
|
|
57
|
+
data-cy={sel('public.docs.pageLink', { slug: page.slug })}
|
|
58
|
+
>
|
|
59
|
+
<FileText className="h-3 w-3" />
|
|
60
|
+
<span className="flex-1">{page.title}</span>
|
|
61
|
+
<ArrowRight className="h-3 w-3 opacity-0 group-hover:opacity-100 group-hover:translate-x-0.5 transition-all" />
|
|
62
|
+
</Link>
|
|
63
|
+
</li>
|
|
64
|
+
))}
|
|
65
|
+
</ul>
|
|
66
|
+
</CardContent>
|
|
67
|
+
</Card>
|
|
68
|
+
))}
|
|
69
|
+
</div>
|
|
70
|
+
) : (
|
|
71
|
+
<div className="text-center py-12">
|
|
72
|
+
<BookOpen className="h-12 w-12 text-muted-foreground mx-auto mb-4" />
|
|
73
|
+
<h2 className="text-xl font-semibold mb-2">No Documentation Available</h2>
|
|
74
|
+
<p className="text-muted-foreground">
|
|
75
|
+
Documentation will appear here once it is added to the project.
|
|
76
|
+
</p>
|
|
77
|
+
</div>
|
|
78
|
+
)}
|
|
79
|
+
</div>
|
|
80
|
+
)
|
|
81
|
+
}
|
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
import type { Metadata } from "next"
|
|
2
|
+
import { PublicNavbar } from '@nextsparkjs/core/components/app/layouts/PublicNavbar'
|
|
3
|
+
import { PublicFooter } from '@nextsparkjs/core/components/app/layouts/PublicFooter'
|
|
4
|
+
import { getTemplateOrDefault, getMetadataOrDefault } from "@nextsparkjs/core/lib/template-resolver"
|
|
5
|
+
|
|
6
|
+
// ✅ MINIMAL GENERIC METADATA (cliente puede override con template)
|
|
7
|
+
const defaultMetadata: Metadata = {
|
|
8
|
+
title: {
|
|
9
|
+
template: '%s | App',
|
|
10
|
+
default: 'App',
|
|
11
|
+
},
|
|
12
|
+
description: 'Application',
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
export const metadata: Metadata = getMetadataOrDefault(
|
|
16
|
+
'app/(public)/layout.tsx',
|
|
17
|
+
defaultMetadata
|
|
18
|
+
)
|
|
19
|
+
|
|
20
|
+
function PublicLayout({
|
|
21
|
+
children
|
|
22
|
+
}: {
|
|
23
|
+
children: React.ReactNode
|
|
24
|
+
}) {
|
|
25
|
+
return (
|
|
26
|
+
<div className="min-h-screen bg-background flex flex-col">
|
|
27
|
+
{/* Public Navbar */}
|
|
28
|
+
<PublicNavbar />
|
|
29
|
+
|
|
30
|
+
{/* Main Content */}
|
|
31
|
+
<main className="flex-1">
|
|
32
|
+
{children}
|
|
33
|
+
</main>
|
|
34
|
+
|
|
35
|
+
{/* Public Footer */}
|
|
36
|
+
<PublicFooter />
|
|
37
|
+
</div>
|
|
38
|
+
)
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
export default getTemplateOrDefault('app/(public)/layout.tsx', PublicLayout)
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
import { redirect } from 'next/navigation'
|
|
2
|
+
import { getTemplateOrDefault } from "@nextsparkjs/core/lib/template-resolver"
|
|
3
|
+
|
|
4
|
+
/**
|
|
5
|
+
* Default Public Home Page (CORE)
|
|
6
|
+
*
|
|
7
|
+
* This is the minimal CORE version that redirects to dashboard if no theme template override exists.
|
|
8
|
+
* Theme templates can override this to provide custom landing pages.
|
|
9
|
+
*
|
|
10
|
+
* Location: app/(public)/page.tsx (CORE)
|
|
11
|
+
* Override: contents/themes/[theme]/templates/(public)/page.tsx (THEME)
|
|
12
|
+
*/
|
|
13
|
+
function DefaultPublicHome() {
|
|
14
|
+
// Si no hay template override, redirect to dashboard
|
|
15
|
+
// (Theme provides the actual landing page)
|
|
16
|
+
redirect('/dashboard')
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
export default getTemplateOrDefault('app/(public)/page.tsx', DefaultPublicHome)
|
|
@@ -0,0 +1,89 @@
|
|
|
1
|
+
'use client'
|
|
2
|
+
|
|
3
|
+
import { AlertTriangle, ArrowLeft, Home } from 'lucide-react'
|
|
4
|
+
import Link from 'next/link'
|
|
5
|
+
import { useSearchParams } from 'next/navigation'
|
|
6
|
+
import { Suspense } from 'react'
|
|
7
|
+
import { Button } from '@nextsparkjs/core/components/ui/button'
|
|
8
|
+
import { Card, CardContent, CardDescription, CardHeader, CardTitle } from '@nextsparkjs/core/components/ui/card'
|
|
9
|
+
import { getTemplateOrDefaultClient } from '@nextsparkjs/registries/template-registry.client'
|
|
10
|
+
|
|
11
|
+
function ForbiddenContent() {
|
|
12
|
+
const searchParams = useSearchParams()
|
|
13
|
+
const reason = searchParams.get('reason') || 'No tienes permisos para acceder a este recurso'
|
|
14
|
+
const upgrade = searchParams.get('upgrade') === 'true'
|
|
15
|
+
|
|
16
|
+
return (
|
|
17
|
+
<div className="min-h-screen flex items-center justify-center bg-gray-50 px-4">
|
|
18
|
+
<Card className="w-full max-w-md">
|
|
19
|
+
<CardHeader className="text-center">
|
|
20
|
+
<div className="mx-auto mb-4 p-4 rounded-full bg-yellow-100">
|
|
21
|
+
<AlertTriangle className="h-8 w-8 text-yellow-600" />
|
|
22
|
+
</div>
|
|
23
|
+
<CardTitle className="text-2xl">
|
|
24
|
+
⚠️ Ups, no tienes permisos para hacer esto
|
|
25
|
+
</CardTitle>
|
|
26
|
+
<CardDescription className="text-gray-600">
|
|
27
|
+
{reason}
|
|
28
|
+
</CardDescription>
|
|
29
|
+
</CardHeader>
|
|
30
|
+
|
|
31
|
+
<CardContent className="space-y-4">
|
|
32
|
+
{upgrade && (
|
|
33
|
+
<div className="bg-blue-50 border border-blue-200 rounded-lg p-4">
|
|
34
|
+
<p className="text-sm text-blue-800">
|
|
35
|
+
Esta funcionalidad requiere una mejora de tu plan actual.
|
|
36
|
+
</p>
|
|
37
|
+
<Link href="/dashboard/settings/billing" className="block mt-2">
|
|
38
|
+
<Button variant="outline" size="sm" className="w-full">
|
|
39
|
+
Ver planes
|
|
40
|
+
</Button>
|
|
41
|
+
</Link>
|
|
42
|
+
</div>
|
|
43
|
+
)}
|
|
44
|
+
|
|
45
|
+
<div className="flex gap-2">
|
|
46
|
+
<Button
|
|
47
|
+
variant="outline"
|
|
48
|
+
className="flex-1"
|
|
49
|
+
onClick={() => window.history.back()}
|
|
50
|
+
>
|
|
51
|
+
<ArrowLeft className="h-4 w-4 mr-2" />
|
|
52
|
+
Volver
|
|
53
|
+
</Button>
|
|
54
|
+
|
|
55
|
+
<Link href="/dashboard" className="flex-1">
|
|
56
|
+
<Button className="w-full">
|
|
57
|
+
<Home className="h-4 w-4 mr-2" />
|
|
58
|
+
Dashboard
|
|
59
|
+
</Button>
|
|
60
|
+
</Link>
|
|
61
|
+
</div>
|
|
62
|
+
</CardContent>
|
|
63
|
+
</Card>
|
|
64
|
+
</div>
|
|
65
|
+
)
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
function ForbiddenPage() {
|
|
69
|
+
return (
|
|
70
|
+
<Suspense fallback={
|
|
71
|
+
<div className="min-h-screen flex items-center justify-center bg-gray-50 px-4">
|
|
72
|
+
<Card className="w-full max-w-md">
|
|
73
|
+
<CardHeader className="text-center">
|
|
74
|
+
<div className="mx-auto mb-4 p-4 rounded-full bg-yellow-100">
|
|
75
|
+
<AlertTriangle className="h-8 w-8 text-yellow-600" />
|
|
76
|
+
</div>
|
|
77
|
+
<CardTitle className="text-2xl">
|
|
78
|
+
⚠️ Ups, no tienes permisos para hacer esto
|
|
79
|
+
</CardTitle>
|
|
80
|
+
</CardHeader>
|
|
81
|
+
</Card>
|
|
82
|
+
</div>
|
|
83
|
+
}>
|
|
84
|
+
<ForbiddenContent />
|
|
85
|
+
</Suspense>
|
|
86
|
+
)
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
export default getTemplateOrDefaultClient('app/403/page.tsx', ForbiddenPage)
|
|
@@ -0,0 +1,78 @@
|
|
|
1
|
+
import { auth } from "@nextsparkjs/core/lib/auth";
|
|
2
|
+
import { toNextJsHandler } from "better-auth/next-js";
|
|
3
|
+
import { NextRequest, NextResponse } from "next/server";
|
|
4
|
+
import { TEAMS_CONFIG } from "@nextsparkjs/core/lib/config";
|
|
5
|
+
import { isPublicSignupRestricted } from "@nextsparkjs/core/lib/teams/helpers";
|
|
6
|
+
import { TeamService } from "@nextsparkjs/core/lib/services";
|
|
7
|
+
import { wrapAuthHandlerWithCors, handleCorsPreflightRequest, addCorsHeaders } from "@nextsparkjs/core/lib/api/helpers";
|
|
8
|
+
|
|
9
|
+
const handlers = toNextJsHandler(auth);
|
|
10
|
+
|
|
11
|
+
// Handle CORS preflight requests for cross-origin auth (mobile apps, etc.)
|
|
12
|
+
export async function OPTIONS(req: NextRequest) {
|
|
13
|
+
return handleCorsPreflightRequest(req);
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
// Intercept email verification requests to redirect to UI page
|
|
17
|
+
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
18
|
+
export async function GET(req: NextRequest, context: { params: Promise<{ all: string[] }> }) {
|
|
19
|
+
const pathname = req.nextUrl.pathname;
|
|
20
|
+
|
|
21
|
+
// Check if this is an email verification request from an email link
|
|
22
|
+
// We check for a special header to determine if it's from our UI or from an email click
|
|
23
|
+
const isFromUI = req.headers.get('x-verify-from-ui') === 'true';
|
|
24
|
+
|
|
25
|
+
if (pathname === '/api/auth/verify-email' && !isFromUI) {
|
|
26
|
+
const token = req.nextUrl.searchParams.get('token');
|
|
27
|
+
const callbackURL = req.nextUrl.searchParams.get('callbackURL');
|
|
28
|
+
|
|
29
|
+
if (token) {
|
|
30
|
+
// This is from an email link, redirect to the UI verification page
|
|
31
|
+
const redirectUrl = new URL('/verify-email', req.url);
|
|
32
|
+
redirectUrl.searchParams.set('token', token);
|
|
33
|
+
if (callbackURL) {
|
|
34
|
+
redirectUrl.searchParams.set('callbackURL', callbackURL);
|
|
35
|
+
}
|
|
36
|
+
return NextResponse.redirect(redirectUrl);
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
// Wrap with CORS headers for cross-origin requests (mobile apps, etc.)
|
|
41
|
+
return wrapAuthHandlerWithCors(() => handlers.GET(req), req);
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
// Intercept signup requests to validate single-tenant mode
|
|
45
|
+
export async function POST(req: NextRequest) {
|
|
46
|
+
const pathname = req.nextUrl.pathname;
|
|
47
|
+
|
|
48
|
+
// Check if this is a signup request (email or OAuth)
|
|
49
|
+
const isSignupRequest =
|
|
50
|
+
pathname === '/api/auth/sign-up/email' ||
|
|
51
|
+
pathname.includes('/api/auth/callback/'); // OAuth callbacks can create users
|
|
52
|
+
|
|
53
|
+
if (isSignupRequest) {
|
|
54
|
+
const teamsMode = TEAMS_CONFIG.mode;
|
|
55
|
+
|
|
56
|
+
// In single-tenant mode, block public signup if team already exists
|
|
57
|
+
if (isPublicSignupRestricted(teamsMode)) {
|
|
58
|
+
const teamExists = await TeamService.hasGlobal();
|
|
59
|
+
|
|
60
|
+
if (teamExists) {
|
|
61
|
+
// Block public signup - users must be invited
|
|
62
|
+
// Add CORS headers so mobile apps can read the error message
|
|
63
|
+
const errorResponse = NextResponse.json(
|
|
64
|
+
{
|
|
65
|
+
error: 'Registration is closed',
|
|
66
|
+
message: 'This application requires an invitation to register. Please contact an administrator.',
|
|
67
|
+
code: 'SIGNUP_RESTRICTED',
|
|
68
|
+
},
|
|
69
|
+
{ status: 403 }
|
|
70
|
+
);
|
|
71
|
+
return await addCorsHeaders(errorResponse, req);
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
// Wrap with CORS headers for cross-origin requests (mobile apps, etc.)
|
|
77
|
+
return wrapAuthHandlerWithCors(() => handlers.POST(req), req);
|
|
78
|
+
}
|
|
@@ -0,0 +1,98 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Billing Lifecycle Cron Job
|
|
3
|
+
*
|
|
4
|
+
* Scheduled job that runs automatically to manage subscription lifecycle.
|
|
5
|
+
* Protected with CRON_SECRET to prevent unauthorized execution.
|
|
6
|
+
*
|
|
7
|
+
* P1: Lifecycle Management
|
|
8
|
+
*
|
|
9
|
+
* Configure in vercel.json:
|
|
10
|
+
* {
|
|
11
|
+
* "crons": [
|
|
12
|
+
* {
|
|
13
|
+
* "path": "/api/cron/billing/lifecycle",
|
|
14
|
+
* "schedule": "0 0 * * *"
|
|
15
|
+
* }
|
|
16
|
+
* ]
|
|
17
|
+
* }
|
|
18
|
+
*/
|
|
19
|
+
|
|
20
|
+
import { NextRequest } from 'next/server'
|
|
21
|
+
import { handlePastDueGracePeriod } from '@nextsparkjs/core/lib/billing/jobs'
|
|
22
|
+
import { SubscriptionService, UsageService } from '@nextsparkjs/core/lib/services'
|
|
23
|
+
|
|
24
|
+
export async function GET(request: NextRequest) {
|
|
25
|
+
// Verify CRON_SECRET (MANDATORY for security)
|
|
26
|
+
const authHeader = request.headers.get('authorization')
|
|
27
|
+
const cronSecret = process.env.CRON_SECRET
|
|
28
|
+
|
|
29
|
+
if (!cronSecret || authHeader !== `Bearer ${cronSecret}`) {
|
|
30
|
+
console.error('[lifecycle-cron] Unauthorized attempt to access cron endpoint')
|
|
31
|
+
return Response.json({ error: 'Unauthorized' }, { status: 401 })
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
const results = {
|
|
35
|
+
expireTrials: { processed: 0, errors: [] as string[] },
|
|
36
|
+
pastDueGrace: { processed: 0, errors: [] as string[] },
|
|
37
|
+
resetUsage: { processed: 0, errors: [] as string[] }
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
try {
|
|
41
|
+
console.log('[lifecycle-cron] Starting billing lifecycle jobs...')
|
|
42
|
+
|
|
43
|
+
// 1. Expire trials (service returns count)
|
|
44
|
+
try {
|
|
45
|
+
const count = await SubscriptionService.processExpiredTrials()
|
|
46
|
+
results.expireTrials = { processed: count, errors: [] }
|
|
47
|
+
} catch (error) {
|
|
48
|
+
results.expireTrials.errors.push(error instanceof Error ? error.message : 'Unknown error')
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
// 2. Handle past_due grace period (jobs.ts function returns {processed, errors})
|
|
52
|
+
results.pastDueGrace = await handlePastDueGracePeriod()
|
|
53
|
+
|
|
54
|
+
// 3. Reset monthly usage (only on first day of month)
|
|
55
|
+
const today = new Date()
|
|
56
|
+
if (today.getDate() === 1) {
|
|
57
|
+
try {
|
|
58
|
+
const count = await UsageService.processMonthlyReset()
|
|
59
|
+
results.resetUsage = { processed: count, errors: [] }
|
|
60
|
+
} catch (error) {
|
|
61
|
+
results.resetUsage.errors.push(error instanceof Error ? error.message : 'Unknown error')
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
const totalProcessed =
|
|
66
|
+
results.expireTrials.processed + results.pastDueGrace.processed + results.resetUsage.processed
|
|
67
|
+
|
|
68
|
+
const allErrors = [
|
|
69
|
+
...results.expireTrials.errors,
|
|
70
|
+
...results.pastDueGrace.errors,
|
|
71
|
+
...results.resetUsage.errors
|
|
72
|
+
]
|
|
73
|
+
|
|
74
|
+
console.log(`[lifecycle-cron] Completed: ${totalProcessed} items processed, ${allErrors.length} errors`)
|
|
75
|
+
|
|
76
|
+
return Response.json({
|
|
77
|
+
success: true,
|
|
78
|
+
processed: totalProcessed,
|
|
79
|
+
errors: allErrors,
|
|
80
|
+
details: results,
|
|
81
|
+
timestamp: new Date().toISOString()
|
|
82
|
+
})
|
|
83
|
+
} catch (error) {
|
|
84
|
+
console.error('[lifecycle-cron] Fatal error:', error)
|
|
85
|
+
return Response.json(
|
|
86
|
+
{
|
|
87
|
+
success: false,
|
|
88
|
+
error: error instanceof Error ? error.message : 'Unknown error'
|
|
89
|
+
},
|
|
90
|
+
{ status: 500 }
|
|
91
|
+
)
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
// Also support POST for manual triggering via dashboard
|
|
96
|
+
export async function POST(request: NextRequest) {
|
|
97
|
+
return GET(request)
|
|
98
|
+
}
|
|
@@ -0,0 +1,175 @@
|
|
|
1
|
+
import { randomUUID } from 'node:crypto';
|
|
2
|
+
import { NextRequest, NextResponse } from 'next/server';
|
|
3
|
+
|
|
4
|
+
// Dynamic import for rate limiting - graceful fallback if not available
|
|
5
|
+
let checkDistributedRateLimit: ((id: string, tier: string) => Promise<{ allowed: boolean; limit: number; remaining: number; resetTime: number; retryAfter?: number }>) | null = null;
|
|
6
|
+
let createRateLimitErrorResponse: ((result: { allowed: boolean; limit: number; remaining: number; resetTime: number; retryAfter?: number }) => NextResponse) | null = null;
|
|
7
|
+
|
|
8
|
+
// Try to load rate limiting functions
|
|
9
|
+
try {
|
|
10
|
+
const rateLimitModule = require('@nextsparkjs/core/lib/api');
|
|
11
|
+
checkDistributedRateLimit = rateLimitModule.checkDistributedRateLimit;
|
|
12
|
+
createRateLimitErrorResponse = rateLimitModule.createRateLimitErrorResponse;
|
|
13
|
+
} catch {
|
|
14
|
+
// Rate limiting not available - will skip rate limit checks
|
|
15
|
+
console.warn('[CSP Report] Rate limiting not available - running without rate limits');
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
/**
|
|
19
|
+
* CSP Violation Report Endpoint
|
|
20
|
+
*
|
|
21
|
+
* Receives Content-Security-Policy violation reports from browsers.
|
|
22
|
+
* Reports are logged for monitoring and debugging CSP issues.
|
|
23
|
+
*
|
|
24
|
+
* Rate limiting: Uses 'api' tier (100 requests/minute per IP) to prevent abuse.
|
|
25
|
+
* Falls back gracefully if rate limiting is not available.
|
|
26
|
+
*
|
|
27
|
+
* NOTE: This file exists in both apps/dev/app/api/csp-report/ and
|
|
28
|
+
* packages/core/templates/app/api/csp-report/. The template version
|
|
29
|
+
* is used when creating new projects from the core package.
|
|
30
|
+
* Changes should be synchronized between both files.
|
|
31
|
+
*
|
|
32
|
+
* @see https://developer.mozilla.org/en-US/docs/Web/HTTP/CSP#violation_report_syntax
|
|
33
|
+
*/
|
|
34
|
+
|
|
35
|
+
interface CSPViolationReport {
|
|
36
|
+
'csp-report'?: {
|
|
37
|
+
'document-uri'?: string;
|
|
38
|
+
'referrer'?: string;
|
|
39
|
+
'violated-directive'?: string;
|
|
40
|
+
'effective-directive'?: string;
|
|
41
|
+
'original-policy'?: string;
|
|
42
|
+
'disposition'?: string;
|
|
43
|
+
'blocked-uri'?: string;
|
|
44
|
+
'line-number'?: number;
|
|
45
|
+
'column-number'?: number;
|
|
46
|
+
'source-file'?: string;
|
|
47
|
+
'status-code'?: number;
|
|
48
|
+
'script-sample'?: string;
|
|
49
|
+
};
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
// Get allowed origin from environment or use localhost fallback
|
|
53
|
+
const getAllowedOrigin = () => {
|
|
54
|
+
return process.env.NEXT_PUBLIC_APP_URL || 'http://localhost:3000';
|
|
55
|
+
};
|
|
56
|
+
|
|
57
|
+
/**
|
|
58
|
+
* Get client IP address from request headers.
|
|
59
|
+
* Handles various proxy scenarios (X-Forwarded-For, X-Real-IP, etc.)
|
|
60
|
+
*/
|
|
61
|
+
function getClientIp(request: NextRequest): string {
|
|
62
|
+
const forwardedFor = request.headers.get('x-forwarded-for');
|
|
63
|
+
if (forwardedFor) {
|
|
64
|
+
const ips = forwardedFor.split(',').map(ip => ip.trim());
|
|
65
|
+
if (ips[0]) return ips[0];
|
|
66
|
+
}
|
|
67
|
+
const realIp = request.headers.get('x-real-ip');
|
|
68
|
+
if (realIp) return realIp;
|
|
69
|
+
const cfIp = request.headers.get('cf-connecting-ip');
|
|
70
|
+
if (cfIp) return cfIp;
|
|
71
|
+
return 'unknown';
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
export async function POST(request: NextRequest) {
|
|
75
|
+
const requestId = randomUUID().slice(0, 8);
|
|
76
|
+
let rateLimitHeaders: Record<string, string> = {};
|
|
77
|
+
|
|
78
|
+
// Rate limiting: 100 requests per minute per IP (api tier)
|
|
79
|
+
// Skip if rate limiting is not available
|
|
80
|
+
if (checkDistributedRateLimit && createRateLimitErrorResponse) {
|
|
81
|
+
try {
|
|
82
|
+
const clientIp = getClientIp(request);
|
|
83
|
+
const rateLimitResult = await checkDistributedRateLimit(`csp-report:ip:${clientIp}`, 'api');
|
|
84
|
+
|
|
85
|
+
if (!rateLimitResult.allowed) {
|
|
86
|
+
console.warn(`[CSP Report ${requestId}] Rate limit exceeded for IP: ${clientIp}`);
|
|
87
|
+
return createRateLimitErrorResponse(rateLimitResult);
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
// Store rate limit headers to include in response
|
|
91
|
+
rateLimitHeaders = {
|
|
92
|
+
'X-RateLimit-Limit': rateLimitResult.limit.toString(),
|
|
93
|
+
'X-RateLimit-Remaining': rateLimitResult.remaining.toString(),
|
|
94
|
+
'X-RateLimit-Reset': rateLimitResult.resetTime.toString(),
|
|
95
|
+
};
|
|
96
|
+
} catch (rateLimitError) {
|
|
97
|
+
// Log but continue without rate limiting
|
|
98
|
+
console.warn(`[CSP Report ${requestId}] Rate limit check failed, continuing without:`, {
|
|
99
|
+
error: rateLimitError instanceof Error ? rateLimitError.message : 'Unknown error',
|
|
100
|
+
});
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
try {
|
|
105
|
+
const contentType = request.headers.get('content-type') || '';
|
|
106
|
+
|
|
107
|
+
// CSP reports are sent as application/csp-report or application/json
|
|
108
|
+
if (!contentType.includes('application/csp-report') && !contentType.includes('application/json')) {
|
|
109
|
+
console.warn(`[CSP Report ${requestId}] Invalid content-type: ${contentType}`);
|
|
110
|
+
return new Response('Invalid content type', { status: 400 });
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
let rawBody: string | undefined;
|
|
114
|
+
try {
|
|
115
|
+
rawBody = await request.text();
|
|
116
|
+
const report: CSPViolationReport = JSON.parse(rawBody);
|
|
117
|
+
const violation = report['csp-report'];
|
|
118
|
+
|
|
119
|
+
if (!violation) {
|
|
120
|
+
console.warn(`[CSP Report ${requestId}] Invalid report format - missing csp-report key`, {
|
|
121
|
+
bodyPreview: rawBody.slice(0, 200),
|
|
122
|
+
});
|
|
123
|
+
return new Response('Invalid report format', { status: 400 });
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
// Log the violation for monitoring
|
|
127
|
+
// In production, you might want to send this to a logging service like Datadog, Sentry, etc.
|
|
128
|
+
console.warn(`[CSP Violation ${requestId}]`, {
|
|
129
|
+
documentUri: violation['document-uri'],
|
|
130
|
+
blockedUri: violation['blocked-uri'],
|
|
131
|
+
violatedDirective: violation['violated-directive'],
|
|
132
|
+
effectiveDirective: violation['effective-directive'],
|
|
133
|
+
sourceFile: violation['source-file'],
|
|
134
|
+
lineNumber: violation['line-number'],
|
|
135
|
+
columnNumber: violation['column-number'],
|
|
136
|
+
scriptSample: violation['script-sample'],
|
|
137
|
+
disposition: violation['disposition'],
|
|
138
|
+
timestamp: new Date().toISOString(),
|
|
139
|
+
});
|
|
140
|
+
|
|
141
|
+
// Return 204 No Content - browsers don't expect a response body
|
|
142
|
+
return new Response(null, {
|
|
143
|
+
status: 204,
|
|
144
|
+
headers: rateLimitHeaders,
|
|
145
|
+
});
|
|
146
|
+
} catch (parseError) {
|
|
147
|
+
console.error(`[CSP Report ${requestId}] JSON parse error:`, {
|
|
148
|
+
error: parseError instanceof Error ? parseError.message : 'Unknown error',
|
|
149
|
+
bodyPreview: rawBody?.slice(0, 200),
|
|
150
|
+
});
|
|
151
|
+
// Still return 204 to not break browser behavior
|
|
152
|
+
return new Response(null, { status: 204 });
|
|
153
|
+
}
|
|
154
|
+
} catch (error) {
|
|
155
|
+
console.error(`[CSP Report ${requestId}] Unexpected error:`, {
|
|
156
|
+
error: error instanceof Error ? error.message : 'Unknown error',
|
|
157
|
+
stack: error instanceof Error ? error.stack : undefined,
|
|
158
|
+
});
|
|
159
|
+
// Still return 204 to not break browser behavior
|
|
160
|
+
return new Response(null, { status: 204 });
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
// Handle preflight requests for CORS
|
|
165
|
+
// Restrict to same origin instead of allowing all origins
|
|
166
|
+
export async function OPTIONS() {
|
|
167
|
+
return new Response(null, {
|
|
168
|
+
status: 204,
|
|
169
|
+
headers: {
|
|
170
|
+
'Access-Control-Allow-Origin': getAllowedOrigin(),
|
|
171
|
+
'Access-Control-Allow-Methods': 'POST, OPTIONS',
|
|
172
|
+
'Access-Control-Allow-Headers': 'Content-Type',
|
|
173
|
+
},
|
|
174
|
+
});
|
|
175
|
+
}
|