@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,173 @@
|
|
|
1
|
+
# Blocks API
|
|
2
|
+
|
|
3
|
+
Access and validate page builder blocks registered in the application.
|
|
4
|
+
|
|
5
|
+
## Overview
|
|
6
|
+
|
|
7
|
+
The Blocks API provides read-only access to block metadata from the block registry. Blocks are reusable content components used in the page builder. This API allows you to list available blocks, retrieve individual block details, and validate block properties against their schemas.
|
|
8
|
+
|
|
9
|
+
## Authentication
|
|
10
|
+
|
|
11
|
+
These endpoints are **public** (no authentication required) as block metadata is not sensitive.
|
|
12
|
+
|
|
13
|
+
## Endpoints
|
|
14
|
+
|
|
15
|
+
### List Blocks
|
|
16
|
+
`GET /api/v1/blocks`
|
|
17
|
+
|
|
18
|
+
Returns all registered blocks with their metadata.
|
|
19
|
+
|
|
20
|
+
**Query Parameters:**
|
|
21
|
+
- `category` (string, optional): Filter by block category (e.g., "hero", "content", "cta")
|
|
22
|
+
- `scope` (string, optional): Filter by allowed scope (e.g., "pages", "posts")
|
|
23
|
+
|
|
24
|
+
**Example Response:**
|
|
25
|
+
```json
|
|
26
|
+
{
|
|
27
|
+
"success": true,
|
|
28
|
+
"data": [
|
|
29
|
+
{
|
|
30
|
+
"slug": "hero-simple",
|
|
31
|
+
"name": "Simple Hero",
|
|
32
|
+
"description": "A simple hero block with title and subtitle",
|
|
33
|
+
"category": "hero",
|
|
34
|
+
"icon": "Layout",
|
|
35
|
+
"thumbnail": "/blocks/hero-simple.png",
|
|
36
|
+
"scope": ["pages"],
|
|
37
|
+
"fieldDefinitions": [
|
|
38
|
+
{
|
|
39
|
+
"name": "title",
|
|
40
|
+
"type": "text",
|
|
41
|
+
"label": "Title",
|
|
42
|
+
"required": true
|
|
43
|
+
},
|
|
44
|
+
{
|
|
45
|
+
"name": "subtitle",
|
|
46
|
+
"type": "text",
|
|
47
|
+
"label": "Subtitle"
|
|
48
|
+
}
|
|
49
|
+
]
|
|
50
|
+
}
|
|
51
|
+
],
|
|
52
|
+
"meta": {
|
|
53
|
+
"categories": ["hero", "content", "cta", "testimonials", "faq"],
|
|
54
|
+
"total": 15
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
```
|
|
58
|
+
|
|
59
|
+
### Get Block by Slug
|
|
60
|
+
`GET /api/v1/blocks/[slug]`
|
|
61
|
+
|
|
62
|
+
Returns detailed metadata for a specific block.
|
|
63
|
+
|
|
64
|
+
**Path Parameters:**
|
|
65
|
+
- `slug` (string, required): The block identifier (e.g., "hero-simple", "faq-accordion")
|
|
66
|
+
|
|
67
|
+
**Example Response:**
|
|
68
|
+
```json
|
|
69
|
+
{
|
|
70
|
+
"slug": "faq-accordion",
|
|
71
|
+
"name": "FAQ Accordion",
|
|
72
|
+
"description": "Expandable FAQ section with questions and answers",
|
|
73
|
+
"category": "faq",
|
|
74
|
+
"icon": "HelpCircle",
|
|
75
|
+
"thumbnail": "/blocks/faq-accordion.png",
|
|
76
|
+
"fieldDefinitions": [
|
|
77
|
+
{
|
|
78
|
+
"name": "title",
|
|
79
|
+
"type": "text",
|
|
80
|
+
"label": "Section Title"
|
|
81
|
+
},
|
|
82
|
+
{
|
|
83
|
+
"name": "items",
|
|
84
|
+
"type": "repeater",
|
|
85
|
+
"label": "FAQ Items",
|
|
86
|
+
"fields": [
|
|
87
|
+
{ "name": "question", "type": "text", "label": "Question" },
|
|
88
|
+
{ "name": "answer", "type": "richtext", "label": "Answer" }
|
|
89
|
+
]
|
|
90
|
+
}
|
|
91
|
+
]
|
|
92
|
+
}
|
|
93
|
+
```
|
|
94
|
+
|
|
95
|
+
### Validate Block Props
|
|
96
|
+
`POST /api/v1/blocks/validate`
|
|
97
|
+
|
|
98
|
+
Validates block properties against the block's Zod schema.
|
|
99
|
+
|
|
100
|
+
**Request Body:**
|
|
101
|
+
```json
|
|
102
|
+
{
|
|
103
|
+
"blockSlug": "hero-simple",
|
|
104
|
+
"props": {
|
|
105
|
+
"title": "Welcome to Our Site",
|
|
106
|
+
"subtitle": "Discover what we offer"
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
```
|
|
110
|
+
|
|
111
|
+
**Success Response:**
|
|
112
|
+
```json
|
|
113
|
+
{
|
|
114
|
+
"valid": true
|
|
115
|
+
}
|
|
116
|
+
```
|
|
117
|
+
|
|
118
|
+
**Validation Error Response (400):**
|
|
119
|
+
```json
|
|
120
|
+
{
|
|
121
|
+
"valid": false,
|
|
122
|
+
"errors": [
|
|
123
|
+
{
|
|
124
|
+
"code": "invalid_type",
|
|
125
|
+
"expected": "string",
|
|
126
|
+
"received": "undefined",
|
|
127
|
+
"path": ["title"],
|
|
128
|
+
"message": "Required"
|
|
129
|
+
}
|
|
130
|
+
]
|
|
131
|
+
}
|
|
132
|
+
```
|
|
133
|
+
|
|
134
|
+
## Error Responses
|
|
135
|
+
|
|
136
|
+
| Status | Description |
|
|
137
|
+
|--------|-------------|
|
|
138
|
+
| 400 | Bad Request - Invalid request body or validation failed |
|
|
139
|
+
| 404 | Not Found - Block with specified slug not found |
|
|
140
|
+
| 500 | Server Error - Internal error or missing schema |
|
|
141
|
+
|
|
142
|
+
## Block Categories
|
|
143
|
+
|
|
144
|
+
Common block categories include:
|
|
145
|
+
- `hero` - Hero sections and page headers
|
|
146
|
+
- `content` - General content blocks (text, images, galleries)
|
|
147
|
+
- `cta` - Call-to-action sections
|
|
148
|
+
- `testimonials` - Customer testimonials and reviews
|
|
149
|
+
- `faq` - Frequently asked questions
|
|
150
|
+
- `pricing` - Pricing tables and plans
|
|
151
|
+
- `features` - Feature lists and grids
|
|
152
|
+
- `contact` - Contact forms and information
|
|
153
|
+
|
|
154
|
+
## Field Types
|
|
155
|
+
|
|
156
|
+
Block field definitions use these types:
|
|
157
|
+
- `text` - Single line text input
|
|
158
|
+
- `textarea` - Multi-line text
|
|
159
|
+
- `richtext` - Rich text editor (HTML)
|
|
160
|
+
- `number` - Numeric input
|
|
161
|
+
- `select` - Dropdown selection
|
|
162
|
+
- `checkbox` - Boolean toggle
|
|
163
|
+
- `image` - Image upload/selection
|
|
164
|
+
- `link` - URL with label
|
|
165
|
+
- `repeater` - Repeatable group of fields
|
|
166
|
+
- `group` - Nested field group
|
|
167
|
+
|
|
168
|
+
## Usage Notes
|
|
169
|
+
|
|
170
|
+
- Block slugs are unique identifiers in kebab-case
|
|
171
|
+
- Field definitions describe the editable properties of each block
|
|
172
|
+
- The `scope` property indicates where blocks can be used (pages, posts, etc.)
|
|
173
|
+
- Use the validate endpoint before saving block content to ensure data integrity
|
|
@@ -0,0 +1,121 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* API Presets for Blocks
|
|
3
|
+
*
|
|
4
|
+
* These presets appear in the DevTools API Explorer's "Presets" tab.
|
|
5
|
+
*/
|
|
6
|
+
|
|
7
|
+
import { defineApiEndpoint } from '@nextsparkjs/core/types/api-presets'
|
|
8
|
+
|
|
9
|
+
export default defineApiEndpoint({
|
|
10
|
+
endpoint: '/api/v1/blocks',
|
|
11
|
+
summary: 'Access and validate page builder blocks',
|
|
12
|
+
presets: [
|
|
13
|
+
// List blocks
|
|
14
|
+
{
|
|
15
|
+
id: 'list-all',
|
|
16
|
+
title: 'List All Blocks',
|
|
17
|
+
description: 'Fetch all registered blocks with metadata',
|
|
18
|
+
method: 'GET',
|
|
19
|
+
tags: ['read', 'list']
|
|
20
|
+
},
|
|
21
|
+
{
|
|
22
|
+
id: 'list-hero-blocks',
|
|
23
|
+
title: 'List Hero Blocks',
|
|
24
|
+
description: 'Filter blocks by hero category',
|
|
25
|
+
method: 'GET',
|
|
26
|
+
queryParams: {
|
|
27
|
+
category: 'hero'
|
|
28
|
+
},
|
|
29
|
+
tags: ['read', 'list', 'filter']
|
|
30
|
+
},
|
|
31
|
+
{
|
|
32
|
+
id: 'list-content-blocks',
|
|
33
|
+
title: 'List Content Blocks',
|
|
34
|
+
description: 'Filter blocks by content category',
|
|
35
|
+
method: 'GET',
|
|
36
|
+
queryParams: {
|
|
37
|
+
category: 'content'
|
|
38
|
+
},
|
|
39
|
+
tags: ['read', 'list', 'filter']
|
|
40
|
+
},
|
|
41
|
+
{
|
|
42
|
+
id: 'list-page-blocks',
|
|
43
|
+
title: 'List Page Blocks',
|
|
44
|
+
description: 'Filter blocks available for pages',
|
|
45
|
+
method: 'GET',
|
|
46
|
+
queryParams: {
|
|
47
|
+
scope: 'pages'
|
|
48
|
+
},
|
|
49
|
+
tags: ['read', 'list', 'filter']
|
|
50
|
+
},
|
|
51
|
+
|
|
52
|
+
// Get specific block
|
|
53
|
+
{
|
|
54
|
+
id: 'get-hero-simple',
|
|
55
|
+
title: 'Get Hero Simple',
|
|
56
|
+
description: 'Fetch hero-simple block metadata',
|
|
57
|
+
method: 'GET',
|
|
58
|
+
path: '/hero-simple',
|
|
59
|
+
tags: ['read', 'single']
|
|
60
|
+
},
|
|
61
|
+
{
|
|
62
|
+
id: 'get-faq-accordion',
|
|
63
|
+
title: 'Get FAQ Accordion',
|
|
64
|
+
description: 'Fetch faq-accordion block metadata',
|
|
65
|
+
method: 'GET',
|
|
66
|
+
path: '/faq-accordion',
|
|
67
|
+
tags: ['read', 'single']
|
|
68
|
+
},
|
|
69
|
+
|
|
70
|
+
// Validate
|
|
71
|
+
{
|
|
72
|
+
id: 'validate-hero',
|
|
73
|
+
title: 'Validate Hero Props',
|
|
74
|
+
description: 'Validate hero block properties',
|
|
75
|
+
method: 'POST',
|
|
76
|
+
path: '/validate',
|
|
77
|
+
payload: {
|
|
78
|
+
blockSlug: 'hero-simple',
|
|
79
|
+
props: {
|
|
80
|
+
title: 'Welcome to Our Site',
|
|
81
|
+
subtitle: 'Discover amazing features'
|
|
82
|
+
}
|
|
83
|
+
},
|
|
84
|
+
tags: ['write', 'validate']
|
|
85
|
+
},
|
|
86
|
+
{
|
|
87
|
+
id: 'validate-faq',
|
|
88
|
+
title: 'Validate FAQ Props',
|
|
89
|
+
description: 'Validate FAQ block with items',
|
|
90
|
+
method: 'POST',
|
|
91
|
+
path: '/validate',
|
|
92
|
+
payload: {
|
|
93
|
+
blockSlug: 'faq-accordion',
|
|
94
|
+
props: {
|
|
95
|
+
title: 'Frequently Asked Questions',
|
|
96
|
+
items: [
|
|
97
|
+
{
|
|
98
|
+
question: 'How does it work?',
|
|
99
|
+
answer: 'It works seamlessly with your existing workflow.'
|
|
100
|
+
}
|
|
101
|
+
]
|
|
102
|
+
}
|
|
103
|
+
},
|
|
104
|
+
tags: ['write', 'validate']
|
|
105
|
+
},
|
|
106
|
+
{
|
|
107
|
+
id: 'validate-invalid',
|
|
108
|
+
title: 'Validate Invalid Props',
|
|
109
|
+
description: 'Test validation error response (missing required field)',
|
|
110
|
+
method: 'POST',
|
|
111
|
+
path: '/validate',
|
|
112
|
+
payload: {
|
|
113
|
+
blockSlug: 'hero-simple',
|
|
114
|
+
props: {
|
|
115
|
+
subtitle: 'Missing required title field'
|
|
116
|
+
}
|
|
117
|
+
},
|
|
118
|
+
tags: ['write', 'validate', 'error']
|
|
119
|
+
}
|
|
120
|
+
]
|
|
121
|
+
})
|
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
import { NextRequest, NextResponse } from 'next/server'
|
|
2
|
+
import { BLOCK_REGISTRY, BLOCK_CATEGORIES } from '@nextsparkjs/registries/block-registry'
|
|
3
|
+
|
|
4
|
+
export async function GET(request: NextRequest) {
|
|
5
|
+
try {
|
|
6
|
+
const { searchParams } = new URL(request.url)
|
|
7
|
+
const category = searchParams.get('category')
|
|
8
|
+
const scope = searchParams.get('scope')
|
|
9
|
+
|
|
10
|
+
let blocks = Object.values(BLOCK_REGISTRY)
|
|
11
|
+
|
|
12
|
+
// Filter by category
|
|
13
|
+
if (category) {
|
|
14
|
+
blocks = blocks.filter(block => block.category === category)
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
// Filter by scope (pages, posts, etc.)
|
|
18
|
+
if (scope) {
|
|
19
|
+
blocks = blocks.filter(block => block.scope?.includes(scope))
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
const blocksMetadata = blocks.map(block => ({
|
|
23
|
+
slug: block.slug,
|
|
24
|
+
name: block.name,
|
|
25
|
+
description: block.description,
|
|
26
|
+
category: block.category,
|
|
27
|
+
icon: block.icon,
|
|
28
|
+
thumbnail: block.thumbnail,
|
|
29
|
+
fieldDefinitions: block.fieldDefinitions,
|
|
30
|
+
scope: block.scope
|
|
31
|
+
}))
|
|
32
|
+
|
|
33
|
+
return NextResponse.json({
|
|
34
|
+
success: true,
|
|
35
|
+
data: blocksMetadata,
|
|
36
|
+
meta: {
|
|
37
|
+
categories: BLOCK_CATEGORIES,
|
|
38
|
+
total: blocksMetadata.length
|
|
39
|
+
}
|
|
40
|
+
})
|
|
41
|
+
} catch (err) {
|
|
42
|
+
console.error('Error listing blocks:', err)
|
|
43
|
+
return NextResponse.json({ success: false, error: 'Internal server error' }, { status: 500 })
|
|
44
|
+
}
|
|
45
|
+
}
|
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
import { NextRequest, NextResponse } from 'next/server'
|
|
2
|
+
import { BLOCK_REGISTRY } from '@nextsparkjs/registries/block-registry'
|
|
3
|
+
import { z } from 'zod'
|
|
4
|
+
|
|
5
|
+
const requestSchema = z.object({
|
|
6
|
+
blockSlug: z.string(),
|
|
7
|
+
props: z.record(z.string(), z.unknown())
|
|
8
|
+
})
|
|
9
|
+
|
|
10
|
+
export async function POST(request: NextRequest) {
|
|
11
|
+
try {
|
|
12
|
+
const body = await request.json()
|
|
13
|
+
const { blockSlug, props } = requestSchema.parse(body)
|
|
14
|
+
|
|
15
|
+
const block = BLOCK_REGISTRY[blockSlug]
|
|
16
|
+
|
|
17
|
+
if (!block) {
|
|
18
|
+
return NextResponse.json({ error: 'Block not found' }, { status: 404 })
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
if (!block.schemaPath) {
|
|
22
|
+
return NextResponse.json({ error: 'Block schema not found' }, { status: 500 })
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
try {
|
|
26
|
+
// Dynamic import is allowed here (server-side validation, not registry loading)
|
|
27
|
+
const schemaModule = await import(block.schemaPath)
|
|
28
|
+
const schema = schemaModule.schema
|
|
29
|
+
|
|
30
|
+
schema.parse(props)
|
|
31
|
+
return NextResponse.json({ valid: true })
|
|
32
|
+
} catch (error) {
|
|
33
|
+
if (error instanceof z.ZodError) {
|
|
34
|
+
return NextResponse.json({
|
|
35
|
+
valid: false,
|
|
36
|
+
errors: error.issues
|
|
37
|
+
}, { status: 400 })
|
|
38
|
+
}
|
|
39
|
+
throw error
|
|
40
|
+
}
|
|
41
|
+
} catch (err) {
|
|
42
|
+
console.error('Error validating block:', err)
|
|
43
|
+
return NextResponse.json({ error: 'Internal server error' }, { status: 500 })
|
|
44
|
+
}
|
|
45
|
+
}
|
|
@@ -0,0 +1,116 @@
|
|
|
1
|
+
# Cron API
|
|
2
|
+
|
|
3
|
+
Process scheduled actions via external cron service.
|
|
4
|
+
|
|
5
|
+
## Overview
|
|
6
|
+
|
|
7
|
+
The Cron API provides an endpoint for processing scheduled actions. This endpoint is designed to be called by an external cron service (e.g., cron-job.org, Vercel Cron, GitHub Actions) at regular intervals.
|
|
8
|
+
|
|
9
|
+
## Authentication
|
|
10
|
+
|
|
11
|
+
The endpoint is protected by `CRON_SECRET` header validation. This prevents unauthorized access while allowing external cron services to trigger processing.
|
|
12
|
+
|
|
13
|
+
**Required Header:**
|
|
14
|
+
```
|
|
15
|
+
x-cron-secret: your_cron_secret_value
|
|
16
|
+
```
|
|
17
|
+
|
|
18
|
+
## Endpoints
|
|
19
|
+
|
|
20
|
+
### Process Scheduled Actions
|
|
21
|
+
`GET /api/v1/cron/process`
|
|
22
|
+
|
|
23
|
+
Process pending scheduled actions and clean up old completed actions.
|
|
24
|
+
|
|
25
|
+
**Request Headers:**
|
|
26
|
+
```
|
|
27
|
+
x-cron-secret: {CRON_SECRET}
|
|
28
|
+
```
|
|
29
|
+
|
|
30
|
+
**Success Response:**
|
|
31
|
+
```json
|
|
32
|
+
{
|
|
33
|
+
"success": true,
|
|
34
|
+
"data": {
|
|
35
|
+
"processing": {
|
|
36
|
+
"processed": 5,
|
|
37
|
+
"succeeded": 4,
|
|
38
|
+
"failed": 1,
|
|
39
|
+
"details": [
|
|
40
|
+
{
|
|
41
|
+
"actionId": "action_123",
|
|
42
|
+
"actionType": "send-email",
|
|
43
|
+
"status": "completed",
|
|
44
|
+
"duration": 150
|
|
45
|
+
}
|
|
46
|
+
]
|
|
47
|
+
},
|
|
48
|
+
"cleanup": {
|
|
49
|
+
"deletedCount": 12
|
|
50
|
+
},
|
|
51
|
+
"executionTime": 2345
|
|
52
|
+
},
|
|
53
|
+
"info": {
|
|
54
|
+
"timestamp": "2024-01-15T10:00:00.000Z"
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
```
|
|
58
|
+
|
|
59
|
+
## Configuration
|
|
60
|
+
|
|
61
|
+
### Environment Variables
|
|
62
|
+
|
|
63
|
+
| Variable | Description | Required |
|
|
64
|
+
|----------|-------------|----------|
|
|
65
|
+
| `CRON_SECRET` | Secret token for authentication | Yes |
|
|
66
|
+
|
|
67
|
+
### Recommended Setup
|
|
68
|
+
|
|
69
|
+
**Vercel Cron (vercel.json):**
|
|
70
|
+
```json
|
|
71
|
+
{
|
|
72
|
+
"crons": [
|
|
73
|
+
{
|
|
74
|
+
"path": "/api/v1/cron/process",
|
|
75
|
+
"schedule": "* * * * *"
|
|
76
|
+
}
|
|
77
|
+
]
|
|
78
|
+
}
|
|
79
|
+
```
|
|
80
|
+
|
|
81
|
+
**External Cron Service:**
|
|
82
|
+
- URL: `https://your-domain.com/api/v1/cron/process`
|
|
83
|
+
- Method: GET
|
|
84
|
+
- Header: `x-cron-secret: {your_secret}`
|
|
85
|
+
- Frequency: Every 1 minute
|
|
86
|
+
|
|
87
|
+
## Processing Details
|
|
88
|
+
|
|
89
|
+
Each execution:
|
|
90
|
+
1. Validates `CRON_SECRET` header
|
|
91
|
+
2. Processes up to 10 pending actions
|
|
92
|
+
3. Cleans up actions older than 7 days
|
|
93
|
+
4. Returns processing results
|
|
94
|
+
|
|
95
|
+
## Scheduled Actions
|
|
96
|
+
|
|
97
|
+
Actions are registered in the `scheduled_actions` table and processed based on:
|
|
98
|
+
- `status = 'pending'`
|
|
99
|
+
- `scheduledAt <= now()`
|
|
100
|
+
|
|
101
|
+
Each action type has a registered handler that executes the business logic.
|
|
102
|
+
|
|
103
|
+
## Error Responses
|
|
104
|
+
|
|
105
|
+
| Status | Code | Description |
|
|
106
|
+
|--------|------|-------------|
|
|
107
|
+
| 401 | INVALID_CRON_SECRET | Missing or invalid cron secret |
|
|
108
|
+
| 500 | CRON_SECRET_NOT_CONFIGURED | CRON_SECRET env var not set |
|
|
109
|
+
| 500 | PROCESSING_ERROR | Error during action processing |
|
|
110
|
+
|
|
111
|
+
## Security Notes
|
|
112
|
+
|
|
113
|
+
- Never expose `CRON_SECRET` in client-side code
|
|
114
|
+
- Use HTTPS for the cron endpoint
|
|
115
|
+
- Consider IP allowlisting for additional security
|
|
116
|
+
- Monitor failed authentication attempts
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* API Presets for Cron
|
|
3
|
+
*
|
|
4
|
+
* These presets appear in the DevTools API Explorer's "Presets" tab.
|
|
5
|
+
* Note: These require CRON_SECRET header which should be configured in environment.
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
import { defineApiEndpoint } from '@nextsparkjs/core/types/api-presets'
|
|
9
|
+
|
|
10
|
+
export default defineApiEndpoint({
|
|
11
|
+
endpoint: '/api/v1/cron',
|
|
12
|
+
summary: 'Process scheduled actions via cron',
|
|
13
|
+
presets: [
|
|
14
|
+
{
|
|
15
|
+
id: 'process-actions',
|
|
16
|
+
title: 'Process Scheduled Actions',
|
|
17
|
+
description: 'Process pending actions (requires CRON_SECRET header)',
|
|
18
|
+
method: 'GET',
|
|
19
|
+
path: '/process',
|
|
20
|
+
headers: {
|
|
21
|
+
'x-cron-secret': '{{CRON_SECRET}}'
|
|
22
|
+
},
|
|
23
|
+
tags: ['write', 'cron']
|
|
24
|
+
}
|
|
25
|
+
]
|
|
26
|
+
})
|
|
@@ -0,0 +1,108 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Cron Endpoint - Process Scheduled Actions
|
|
3
|
+
* GET /api/v1/cron/process
|
|
4
|
+
*
|
|
5
|
+
* This endpoint is designed to be called by an external cron service
|
|
6
|
+
* (e.g., cron-job.org, Vercel Cron, GitHub Actions)
|
|
7
|
+
*
|
|
8
|
+
* Authentication: Requires CRON_SECRET header
|
|
9
|
+
* Frequency: Recommended every 1 minute
|
|
10
|
+
*/
|
|
11
|
+
|
|
12
|
+
import { NextRequest, NextResponse } from 'next/server'
|
|
13
|
+
import {
|
|
14
|
+
processPendingActions,
|
|
15
|
+
cleanupOldActions,
|
|
16
|
+
initializeScheduledActions
|
|
17
|
+
} from '@nextsparkjs/core/lib/scheduled-actions'
|
|
18
|
+
import type { ProcessResult } from '@nextsparkjs/core/lib/scheduled-actions'
|
|
19
|
+
|
|
20
|
+
/**
|
|
21
|
+
* Process pending scheduled actions
|
|
22
|
+
* Protected by CRON_SECRET for security
|
|
23
|
+
*/
|
|
24
|
+
export async function GET(request: NextRequest): Promise<NextResponse> {
|
|
25
|
+
const startTime = Date.now()
|
|
26
|
+
|
|
27
|
+
// Ensure action handlers are registered in this context
|
|
28
|
+
// (initializeScheduledActions has its own guard against duplicates)
|
|
29
|
+
initializeScheduledActions()
|
|
30
|
+
|
|
31
|
+
try {
|
|
32
|
+
// Validate CRON_SECRET
|
|
33
|
+
const cronSecret = request.headers.get('x-cron-secret')
|
|
34
|
+
const expectedSecret = process.env.CRON_SECRET
|
|
35
|
+
|
|
36
|
+
if (!expectedSecret) {
|
|
37
|
+
console.error('[Cron] CRON_SECRET environment variable is not set')
|
|
38
|
+
return NextResponse.json(
|
|
39
|
+
{
|
|
40
|
+
success: false,
|
|
41
|
+
error: 'Server configuration error',
|
|
42
|
+
code: 'CRON_SECRET_NOT_CONFIGURED'
|
|
43
|
+
},
|
|
44
|
+
{ status: 500 }
|
|
45
|
+
)
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
if (!cronSecret || cronSecret !== expectedSecret) {
|
|
49
|
+
console.warn('[Cron] Unauthorized cron request attempt')
|
|
50
|
+
return NextResponse.json(
|
|
51
|
+
{
|
|
52
|
+
success: false,
|
|
53
|
+
error: 'Unauthorized',
|
|
54
|
+
code: 'INVALID_CRON_SECRET'
|
|
55
|
+
},
|
|
56
|
+
{ status: 401 }
|
|
57
|
+
)
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
console.log('[Cron] Starting scheduled actions processing')
|
|
61
|
+
|
|
62
|
+
// Process pending actions (batch of 10 max)
|
|
63
|
+
const processResult: ProcessResult = await processPendingActions(10)
|
|
64
|
+
|
|
65
|
+
// Optionally run cleanup (every 24th run if called every hour)
|
|
66
|
+
// For now, run cleanup every time (can be optimized later)
|
|
67
|
+
const cleanupCount = await cleanupOldActions(7)
|
|
68
|
+
|
|
69
|
+
const executionTime = Date.now() - startTime
|
|
70
|
+
|
|
71
|
+
console.log(`[Cron] Processing complete in ${executionTime}ms`)
|
|
72
|
+
|
|
73
|
+
return NextResponse.json(
|
|
74
|
+
{
|
|
75
|
+
success: true,
|
|
76
|
+
data: {
|
|
77
|
+
processing: processResult,
|
|
78
|
+
cleanup: {
|
|
79
|
+
deletedCount: cleanupCount
|
|
80
|
+
},
|
|
81
|
+
executionTime
|
|
82
|
+
},
|
|
83
|
+
info: {
|
|
84
|
+
timestamp: new Date().toISOString()
|
|
85
|
+
}
|
|
86
|
+
},
|
|
87
|
+
{ status: 200 }
|
|
88
|
+
)
|
|
89
|
+
} catch (error) {
|
|
90
|
+
const executionTime = Date.now() - startTime
|
|
91
|
+
|
|
92
|
+
console.error('[Cron] Error processing scheduled actions:', error)
|
|
93
|
+
|
|
94
|
+
return NextResponse.json(
|
|
95
|
+
{
|
|
96
|
+
success: false,
|
|
97
|
+
error: 'Internal server error',
|
|
98
|
+
code: 'PROCESSING_ERROR',
|
|
99
|
+
details: error instanceof Error ? error.message : 'Unknown error',
|
|
100
|
+
info: {
|
|
101
|
+
timestamp: new Date().toISOString(),
|
|
102
|
+
executionTime
|
|
103
|
+
}
|
|
104
|
+
},
|
|
105
|
+
{ status: 500 }
|
|
106
|
+
)
|
|
107
|
+
}
|
|
108
|
+
}
|