@nextsparkjs/core 0.1.0-beta.82 → 0.1.0-beta.83
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/components/entities/wrappers/EntityDetailWrapper.d.ts.map +1 -1
- package/dist/components/entities/wrappers/EntityDetailWrapper.js +11 -39
- package/dist/hooks/useEntityQuery.d.ts.map +1 -1
- package/dist/hooks/useEntityQuery.js +21 -3
- package/dist/lib/theme/get-default-theme-mode.d.ts +11 -0
- package/dist/lib/theme/get-default-theme-mode.d.ts.map +1 -1
- package/dist/lib/theme/get-default-theme-mode.js +42 -25
- package/dist/styles/classes.json +1 -1
- package/dist/types/theme.d.ts +2 -0
- package/dist/types/theme.d.ts.map +1 -1
- package/package.json +16 -16
- 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/dist/templates/app/(auth)/forgot-password/page.tsx +0 -216
- package/dist/templates/app/(auth)/layout.tsx +0 -51
- package/dist/templates/app/(auth)/login/page.tsx +0 -21
- package/dist/templates/app/(auth)/reset-password/page.tsx +0 -212
- package/dist/templates/app/(auth)/signup/page.tsx +0 -21
- package/dist/templates/app/(auth)/verify-email/page.tsx +0 -190
- package/dist/templates/app/(public)/[...slug]/page.tsx +0 -378
- package/dist/templates/app/(public)/docs/[section]/[page]/page.tsx +0 -90
- package/dist/templates/app/(public)/docs/layout.tsx +0 -25
- package/dist/templates/app/(public)/docs/page.tsx +0 -81
- package/dist/templates/app/(public)/layout.tsx +0 -41
- package/dist/templates/app/(public)/page.tsx +0 -19
- package/dist/templates/app/403/page.tsx +0 -89
- package/dist/templates/app/api/auth/[...all]/route.ts +0 -78
- package/dist/templates/app/api/cron/billing/lifecycle/route.ts +0 -98
- package/dist/templates/app/api/csp-report/route.ts +0 -175
- package/dist/templates/app/api/devtools/config/entities/route.ts +0 -108
- package/dist/templates/app/api/devtools/config/theme/route.ts +0 -66
- package/dist/templates/app/api/devtools/tests/[...path]/route.ts +0 -130
- package/dist/templates/app/api/devtools/tests/route.ts +0 -134
- package/dist/templates/app/api/health/route.ts +0 -29
- package/dist/templates/app/api/internal/user-metadata/route.ts +0 -36
- package/dist/templates/app/api/superadmin/subscriptions/route.ts +0 -310
- package/dist/templates/app/api/superadmin/teams/[teamId]/route.ts +0 -286
- package/dist/templates/app/api/superadmin/teams/route.ts +0 -188
- package/dist/templates/app/api/superadmin/users/[userId]/route.ts +0 -540
- package/dist/templates/app/api/superadmin/users/route.ts +0 -323
- package/dist/templates/app/api/user/delete-account/route.ts +0 -55
- package/dist/templates/app/api/user/plan-flags/route.ts +0 -283
- package/dist/templates/app/api/user/profile/route.ts +0 -133
- package/dist/templates/app/api/v1/[entity]/[id]/child/[childType]/[childId]/route.ts +0 -210
- package/dist/templates/app/api/v1/[entity]/[id]/child/[childType]/route.ts +0 -331
- package/dist/templates/app/api/v1/[entity]/[id]/route.ts +0 -35
- package/dist/templates/app/api/v1/[entity]/docs.md +0 -369
- package/dist/templates/app/api/v1/[entity]/presets.ts +0 -194
- package/dist/templates/app/api/v1/[entity]/route.ts +0 -31
- package/dist/templates/app/api/v1/api-keys/[id]/route.ts +0 -303
- package/dist/templates/app/api/v1/api-keys/docs.md +0 -101
- package/dist/templates/app/api/v1/api-keys/presets.ts +0 -31
- package/dist/templates/app/api/v1/api-keys/route.ts +0 -250
- package/dist/templates/app/api/v1/auth/docs.md +0 -184
- package/dist/templates/app/api/v1/auth/presets.ts +0 -44
- package/dist/templates/app/api/v1/auth/signup-with-invite/route.ts +0 -227
- package/dist/templates/app/api/v1/billing/cancel/route.ts +0 -206
- package/dist/templates/app/api/v1/billing/change-plan/route.ts +0 -97
- package/dist/templates/app/api/v1/billing/check-action/route.ts +0 -81
- package/dist/templates/app/api/v1/billing/checkout/route.ts +0 -124
- package/dist/templates/app/api/v1/billing/docs.md +0 -209
- package/dist/templates/app/api/v1/billing/plans/route.ts +0 -85
- package/dist/templates/app/api/v1/billing/portal/route.ts +0 -90
- package/dist/templates/app/api/v1/billing/presets.ts +0 -121
- package/dist/templates/app/api/v1/billing/webhooks/stripe/route.ts +0 -428
- package/dist/templates/app/api/v1/blocks/[slug]/route.ts +0 -29
- package/dist/templates/app/api/v1/blocks/docs.md +0 -173
- package/dist/templates/app/api/v1/blocks/presets.ts +0 -121
- package/dist/templates/app/api/v1/blocks/route.ts +0 -45
- package/dist/templates/app/api/v1/blocks/validate/route.ts +0 -45
- package/dist/templates/app/api/v1/cron/docs.md +0 -116
- package/dist/templates/app/api/v1/cron/presets.ts +0 -26
- package/dist/templates/app/api/v1/cron/process/route.ts +0 -108
- package/dist/templates/app/api/v1/devtools/blocks/route.ts +0 -82
- package/dist/templates/app/api/v1/devtools/docs/route.ts +0 -150
- package/dist/templates/app/api/v1/devtools/docs.md +0 -204
- package/dist/templates/app/api/v1/devtools/features/route.ts +0 -61
- package/dist/templates/app/api/v1/devtools/flows/route.ts +0 -61
- package/dist/templates/app/api/v1/devtools/presets.ts +0 -113
- package/dist/templates/app/api/v1/devtools/scheduled-actions/route.ts +0 -120
- package/dist/templates/app/api/v1/devtools/testing/route.ts +0 -82
- package/dist/templates/app/api/v1/media/docs.md +0 -117
- package/dist/templates/app/api/v1/media/presets.ts +0 -24
- package/dist/templates/app/api/v1/media/upload/route.ts +0 -150
- package/dist/templates/app/api/v1/patterns/[id]/usages/route.ts +0 -116
- package/dist/templates/app/api/v1/plugin/[...path]/route.ts +0 -373
- package/dist/templates/app/api/v1/plugin/docs.md +0 -79
- package/dist/templates/app/api/v1/plugin/presets.ts +0 -21
- package/dist/templates/app/api/v1/plugin/route.ts +0 -96
- package/dist/templates/app/api/v1/post-categories/[id]/route.ts +0 -255
- package/dist/templates/app/api/v1/post-categories/docs.md +0 -134
- package/dist/templates/app/api/v1/post-categories/presets.ts +0 -78
- package/dist/templates/app/api/v1/post-categories/route.ts +0 -119
- package/dist/templates/app/api/v1/team-invitations/[token]/accept/route.ts +0 -179
- package/dist/templates/app/api/v1/team-invitations/[token]/decline/route.ts +0 -120
- package/dist/templates/app/api/v1/team-invitations/[token]/route.ts +0 -89
- package/dist/templates/app/api/v1/team-invitations/docs.md +0 -88
- package/dist/templates/app/api/v1/team-invitations/presets.ts +0 -43
- package/dist/templates/app/api/v1/team-invitations/route.ts +0 -114
- package/dist/templates/app/api/v1/teams/[teamId]/invitations/route.ts +0 -171
- package/dist/templates/app/api/v1/teams/[teamId]/invoices/[invoiceNumber]/route.ts +0 -105
- package/dist/templates/app/api/v1/teams/[teamId]/invoices/route.ts +0 -125
- package/dist/templates/app/api/v1/teams/[teamId]/members/[memberId]/route.ts +0 -263
- package/dist/templates/app/api/v1/teams/[teamId]/members/route.ts +0 -358
- package/dist/templates/app/api/v1/teams/[teamId]/route.ts +0 -322
- package/dist/templates/app/api/v1/teams/[teamId]/subscription/route.ts +0 -50
- package/dist/templates/app/api/v1/teams/[teamId]/usage/[limitSlug]/route.ts +0 -91
- package/dist/templates/app/api/v1/teams/docs.md +0 -320
- package/dist/templates/app/api/v1/teams/presets.ts +0 -178
- package/dist/templates/app/api/v1/teams/route.ts +0 -293
- package/dist/templates/app/api/v1/teams/switch/route.ts +0 -88
- package/dist/templates/app/api/v1/theme/[...path]/route.ts +0 -361
- package/dist/templates/app/api/v1/theme/docs.md +0 -74
- package/dist/templates/app/api/v1/theme/presets.ts +0 -21
- package/dist/templates/app/api/v1/theme/route.ts +0 -96
- package/dist/templates/app/api/v1/users/[id]/meta/[key]/route.ts +0 -363
- package/dist/templates/app/api/v1/users/[id]/route.ts +0 -302
- package/dist/templates/app/api/v1/users/docs.md +0 -93
- package/dist/templates/app/api/v1/users/presets.ts +0 -59
- package/dist/templates/app/api/v1/users/route.ts +0 -197
- package/dist/templates/app/dashboard/(main)/[entity]/[id]/edit/page.tsx +0 -117
- package/dist/templates/app/dashboard/(main)/[entity]/[id]/page.tsx +0 -103
- package/dist/templates/app/dashboard/(main)/[entity]/create/page.tsx +0 -95
- package/dist/templates/app/dashboard/(main)/[entity]/error.tsx +0 -51
- package/dist/templates/app/dashboard/(main)/[entity]/layout.tsx +0 -113
- package/dist/templates/app/dashboard/(main)/[entity]/loading.tsx +0 -61
- package/dist/templates/app/dashboard/(main)/[entity]/page.tsx +0 -90
- package/dist/templates/app/dashboard/(main)/layout.tsx +0 -98
- package/dist/templates/app/dashboard/(main)/loading.tsx +0 -5
- package/dist/templates/app/dashboard/(main)/page.tsx +0 -201
- package/dist/templates/app/dashboard/(main)/patterns/[id]/edit/page.tsx +0 -114
- package/dist/templates/app/dashboard/(main)/patterns/[id]/page.tsx +0 -20
- package/dist/templates/app/dashboard/(main)/patterns/[id]/reports/page.tsx +0 -171
- package/dist/templates/app/dashboard/(main)/patterns/create/page.tsx +0 -86
- package/dist/templates/app/dashboard/(main)/patterns/page.tsx +0 -444
- package/dist/templates/app/dashboard/features/analytics/page.tsx +0 -35
- package/dist/templates/app/dashboard/features/automation/page.tsx +0 -35
- package/dist/templates/app/dashboard/features/layout.tsx +0 -13
- package/dist/templates/app/dashboard/features/loading.tsx +0 -5
- package/dist/templates/app/dashboard/features/webhooks/page.tsx +0 -35
- package/dist/templates/app/dashboard/layout.tsx +0 -86
- package/dist/templates/app/dashboard/permission-denied/page.tsx +0 -29
- package/dist/templates/app/dashboard/settings/api-keys/loading.tsx +0 -5
- package/dist/templates/app/dashboard/settings/api-keys/page.tsx +0 -513
- package/dist/templates/app/dashboard/settings/billing/loading.tsx +0 -5
- package/dist/templates/app/dashboard/settings/billing/page.tsx +0 -284
- package/dist/templates/app/dashboard/settings/invoices/[invoiceNumber]/page.tsx +0 -222
- package/dist/templates/app/dashboard/settings/invoices/loading.tsx +0 -5
- package/dist/templates/app/dashboard/settings/invoices/page.tsx +0 -82
- package/dist/templates/app/dashboard/settings/layout.tsx +0 -151
- package/dist/templates/app/dashboard/settings/loading.tsx +0 -5
- package/dist/templates/app/dashboard/settings/notifications/loading.tsx +0 -5
- package/dist/templates/app/dashboard/settings/notifications/page.tsx +0 -462
- package/dist/templates/app/dashboard/settings/page.tsx +0 -92
- package/dist/templates/app/dashboard/settings/password/loading.tsx +0 -5
- package/dist/templates/app/dashboard/settings/password/page.tsx +0 -306
- package/dist/templates/app/dashboard/settings/plans/loading.tsx +0 -5
- package/dist/templates/app/dashboard/settings/plans/page.tsx +0 -40
- package/dist/templates/app/dashboard/settings/profile/loading.tsx +0 -5
- package/dist/templates/app/dashboard/settings/profile/page.tsx +0 -686
- package/dist/templates/app/dashboard/settings/security/loading.tsx +0 -5
- package/dist/templates/app/dashboard/settings/security/page.tsx +0 -505
- package/dist/templates/app/dashboard/settings/teams/loading.tsx +0 -5
- package/dist/templates/app/dashboard/settings/teams/page.tsx +0 -272
- package/dist/templates/app/dashboard/settings/teams/permissions/page.tsx +0 -92
- package/dist/templates/app/devtools/blocks/[slug]/page.tsx +0 -39
- package/dist/templates/app/devtools/blocks/page.tsx +0 -31
- package/dist/templates/app/devtools/config/page.tsx +0 -31
- package/dist/templates/app/devtools/features/page.tsx +0 -31
- package/dist/templates/app/devtools/flows/page.tsx +0 -31
- package/dist/templates/app/devtools/layout.tsx +0 -58
- package/dist/templates/app/devtools/page.tsx +0 -121
- package/dist/templates/app/devtools/scheduled-actions/page.tsx +0 -157
- package/dist/templates/app/devtools/style/page.tsx +0 -330
- package/dist/templates/app/devtools/tags/page.tsx +0 -31
- package/dist/templates/app/devtools/tests/[[...path]]/page.tsx +0 -47
- package/dist/templates/app/favicon.ico +0 -0
- package/dist/templates/app/globals.css +0 -12
- package/dist/templates/app/layout.tsx +0 -96
- package/dist/templates/app/public/page.tsx +0 -30
- package/dist/templates/app/superadmin/docs/[section]/[page]/page.tsx +0 -92
- package/dist/templates/app/superadmin/docs/page.tsx +0 -75
- package/dist/templates/app/superadmin/layout.tsx +0 -67
- package/dist/templates/app/superadmin/page.tsx +0 -149
- package/dist/templates/app/superadmin/subscriptions/page.tsx +0 -655
- package/dist/templates/app/superadmin/team-roles/page.tsx +0 -493
- package/dist/templates/app/superadmin/teams/[teamId]/page.tsx +0 -687
- package/dist/templates/app/superadmin/teams/page.tsx +0 -302
- package/dist/templates/app/superadmin/users/[userId]/page.tsx +0 -548
- package/dist/templates/app/superadmin/users/page.tsx +0 -528
- package/templates/app/(auth)/forgot-password/page.tsx +0 -216
- package/templates/app/(auth)/layout.tsx +0 -51
- package/templates/app/(auth)/login/page.tsx +0 -21
- package/templates/app/(auth)/reset-password/page.tsx +0 -212
- package/templates/app/(auth)/signup/page.tsx +0 -21
- package/templates/app/(auth)/verify-email/page.tsx +0 -190
- package/templates/app/(public)/[...slug]/page.tsx +0 -378
- package/templates/app/(public)/docs/[section]/[page]/page.tsx +0 -90
- package/templates/app/(public)/docs/layout.tsx +0 -25
- package/templates/app/(public)/docs/page.tsx +0 -81
- package/templates/app/(public)/layout.tsx +0 -41
- package/templates/app/(public)/page.tsx +0 -19
- package/templates/app/403/page.tsx +0 -89
- package/templates/app/api/auth/[...all]/route.ts +0 -78
- package/templates/app/api/cron/billing/lifecycle/route.ts +0 -98
- package/templates/app/api/csp-report/route.ts +0 -175
- package/templates/app/api/devtools/config/entities/route.ts +0 -108
- package/templates/app/api/devtools/config/theme/route.ts +0 -66
- package/templates/app/api/devtools/tests/[...path]/route.ts +0 -130
- package/templates/app/api/devtools/tests/route.ts +0 -134
- package/templates/app/api/health/route.ts +0 -29
- package/templates/app/api/internal/user-metadata/route.ts +0 -36
- package/templates/app/api/superadmin/subscriptions/route.ts +0 -310
- package/templates/app/api/superadmin/teams/[teamId]/route.ts +0 -286
- package/templates/app/api/superadmin/teams/route.ts +0 -188
- package/templates/app/api/superadmin/users/[userId]/route.ts +0 -540
- package/templates/app/api/superadmin/users/route.ts +0 -323
- package/templates/app/api/user/delete-account/route.ts +0 -55
- package/templates/app/api/user/plan-flags/route.ts +0 -283
- package/templates/app/api/user/profile/route.ts +0 -133
- package/templates/app/api/v1/[entity]/[id]/child/[childType]/[childId]/route.ts +0 -210
- package/templates/app/api/v1/[entity]/[id]/child/[childType]/route.ts +0 -331
- package/templates/app/api/v1/[entity]/[id]/route.ts +0 -35
- package/templates/app/api/v1/[entity]/docs.md +0 -369
- package/templates/app/api/v1/[entity]/presets.ts +0 -194
- package/templates/app/api/v1/[entity]/route.ts +0 -31
- package/templates/app/api/v1/api-keys/[id]/route.ts +0 -303
- package/templates/app/api/v1/api-keys/docs.md +0 -101
- package/templates/app/api/v1/api-keys/presets.ts +0 -31
- package/templates/app/api/v1/api-keys/route.ts +0 -250
- package/templates/app/api/v1/auth/docs.md +0 -184
- package/templates/app/api/v1/auth/presets.ts +0 -44
- package/templates/app/api/v1/auth/signup-with-invite/route.ts +0 -227
- package/templates/app/api/v1/billing/cancel/route.ts +0 -206
- package/templates/app/api/v1/billing/change-plan/route.ts +0 -97
- package/templates/app/api/v1/billing/check-action/route.ts +0 -81
- package/templates/app/api/v1/billing/checkout/route.ts +0 -124
- package/templates/app/api/v1/billing/docs.md +0 -209
- package/templates/app/api/v1/billing/plans/route.ts +0 -85
- package/templates/app/api/v1/billing/portal/route.ts +0 -90
- package/templates/app/api/v1/billing/presets.ts +0 -121
- package/templates/app/api/v1/billing/webhooks/stripe/route.ts +0 -428
- package/templates/app/api/v1/blocks/[slug]/route.ts +0 -29
- package/templates/app/api/v1/blocks/docs.md +0 -173
- package/templates/app/api/v1/blocks/presets.ts +0 -121
- package/templates/app/api/v1/blocks/route.ts +0 -45
- package/templates/app/api/v1/blocks/validate/route.ts +0 -45
- package/templates/app/api/v1/cron/docs.md +0 -116
- package/templates/app/api/v1/cron/presets.ts +0 -26
- package/templates/app/api/v1/cron/process/route.ts +0 -108
- package/templates/app/api/v1/devtools/blocks/route.ts +0 -82
- package/templates/app/api/v1/devtools/docs/route.ts +0 -150
- package/templates/app/api/v1/devtools/docs.md +0 -204
- package/templates/app/api/v1/devtools/features/route.ts +0 -61
- package/templates/app/api/v1/devtools/flows/route.ts +0 -61
- package/templates/app/api/v1/devtools/presets.ts +0 -113
- package/templates/app/api/v1/devtools/scheduled-actions/route.ts +0 -120
- package/templates/app/api/v1/devtools/testing/route.ts +0 -82
- package/templates/app/api/v1/media/docs.md +0 -117
- package/templates/app/api/v1/media/presets.ts +0 -24
- package/templates/app/api/v1/media/upload/route.ts +0 -150
- package/templates/app/api/v1/patterns/[id]/usages/route.ts +0 -116
- package/templates/app/api/v1/plugin/[...path]/route.ts +0 -373
- package/templates/app/api/v1/plugin/docs.md +0 -79
- package/templates/app/api/v1/plugin/presets.ts +0 -21
- package/templates/app/api/v1/plugin/route.ts +0 -96
- package/templates/app/api/v1/post-categories/[id]/route.ts +0 -255
- package/templates/app/api/v1/post-categories/docs.md +0 -134
- package/templates/app/api/v1/post-categories/presets.ts +0 -78
- package/templates/app/api/v1/post-categories/route.ts +0 -119
- package/templates/app/api/v1/team-invitations/[token]/accept/route.ts +0 -179
- package/templates/app/api/v1/team-invitations/[token]/decline/route.ts +0 -120
- package/templates/app/api/v1/team-invitations/[token]/route.ts +0 -89
- package/templates/app/api/v1/team-invitations/docs.md +0 -88
- package/templates/app/api/v1/team-invitations/presets.ts +0 -43
- package/templates/app/api/v1/team-invitations/route.ts +0 -114
- package/templates/app/api/v1/teams/[teamId]/invitations/route.ts +0 -171
- package/templates/app/api/v1/teams/[teamId]/invoices/[invoiceNumber]/route.ts +0 -105
- package/templates/app/api/v1/teams/[teamId]/invoices/route.ts +0 -125
- package/templates/app/api/v1/teams/[teamId]/members/[memberId]/route.ts +0 -263
- package/templates/app/api/v1/teams/[teamId]/members/route.ts +0 -358
- package/templates/app/api/v1/teams/[teamId]/route.ts +0 -322
- package/templates/app/api/v1/teams/[teamId]/subscription/route.ts +0 -50
- package/templates/app/api/v1/teams/[teamId]/usage/[limitSlug]/route.ts +0 -91
- package/templates/app/api/v1/teams/docs.md +0 -320
- package/templates/app/api/v1/teams/presets.ts +0 -178
- package/templates/app/api/v1/teams/route.ts +0 -293
- package/templates/app/api/v1/teams/switch/route.ts +0 -88
- package/templates/app/api/v1/theme/[...path]/route.ts +0 -361
- package/templates/app/api/v1/theme/docs.md +0 -74
- package/templates/app/api/v1/theme/presets.ts +0 -21
- package/templates/app/api/v1/theme/route.ts +0 -96
- package/templates/app/api/v1/users/[id]/meta/[key]/route.ts +0 -363
- package/templates/app/api/v1/users/[id]/route.ts +0 -302
- package/templates/app/api/v1/users/docs.md +0 -93
- package/templates/app/api/v1/users/presets.ts +0 -59
- package/templates/app/api/v1/users/route.ts +0 -197
- package/templates/app/dashboard/(main)/[entity]/[id]/edit/page.tsx +0 -117
- package/templates/app/dashboard/(main)/[entity]/[id]/page.tsx +0 -103
- package/templates/app/dashboard/(main)/[entity]/create/page.tsx +0 -95
- package/templates/app/dashboard/(main)/[entity]/error.tsx +0 -51
- package/templates/app/dashboard/(main)/[entity]/layout.tsx +0 -113
- package/templates/app/dashboard/(main)/[entity]/loading.tsx +0 -61
- package/templates/app/dashboard/(main)/[entity]/page.tsx +0 -90
- package/templates/app/dashboard/(main)/layout.tsx +0 -98
- package/templates/app/dashboard/(main)/loading.tsx +0 -5
- package/templates/app/dashboard/(main)/page.tsx +0 -201
- package/templates/app/dashboard/(main)/patterns/[id]/edit/page.tsx +0 -114
- package/templates/app/dashboard/(main)/patterns/[id]/page.tsx +0 -20
- package/templates/app/dashboard/(main)/patterns/[id]/reports/page.tsx +0 -171
- package/templates/app/dashboard/(main)/patterns/create/page.tsx +0 -86
- package/templates/app/dashboard/(main)/patterns/page.tsx +0 -444
- package/templates/app/dashboard/features/analytics/page.tsx +0 -35
- package/templates/app/dashboard/features/automation/page.tsx +0 -35
- package/templates/app/dashboard/features/layout.tsx +0 -13
- package/templates/app/dashboard/features/loading.tsx +0 -5
- package/templates/app/dashboard/features/webhooks/page.tsx +0 -35
- package/templates/app/dashboard/layout.tsx +0 -86
- package/templates/app/dashboard/permission-denied/page.tsx +0 -29
- package/templates/app/dashboard/settings/api-keys/loading.tsx +0 -5
- package/templates/app/dashboard/settings/api-keys/page.tsx +0 -513
- package/templates/app/dashboard/settings/billing/loading.tsx +0 -5
- package/templates/app/dashboard/settings/billing/page.tsx +0 -284
- package/templates/app/dashboard/settings/invoices/[invoiceNumber]/page.tsx +0 -222
- package/templates/app/dashboard/settings/invoices/loading.tsx +0 -5
- package/templates/app/dashboard/settings/invoices/page.tsx +0 -82
- package/templates/app/dashboard/settings/layout.tsx +0 -151
- package/templates/app/dashboard/settings/loading.tsx +0 -5
- package/templates/app/dashboard/settings/notifications/loading.tsx +0 -5
- package/templates/app/dashboard/settings/notifications/page.tsx +0 -462
- package/templates/app/dashboard/settings/page.tsx +0 -92
- package/templates/app/dashboard/settings/password/loading.tsx +0 -5
- package/templates/app/dashboard/settings/password/page.tsx +0 -306
- package/templates/app/dashboard/settings/plans/loading.tsx +0 -5
- package/templates/app/dashboard/settings/plans/page.tsx +0 -40
- package/templates/app/dashboard/settings/profile/loading.tsx +0 -5
- package/templates/app/dashboard/settings/profile/page.tsx +0 -686
- package/templates/app/dashboard/settings/security/loading.tsx +0 -5
- package/templates/app/dashboard/settings/security/page.tsx +0 -505
- package/templates/app/dashboard/settings/teams/loading.tsx +0 -5
- package/templates/app/dashboard/settings/teams/page.tsx +0 -272
- package/templates/app/dashboard/settings/teams/permissions/page.tsx +0 -92
- package/templates/app/devtools/blocks/[slug]/page.tsx +0 -39
- package/templates/app/devtools/blocks/page.tsx +0 -31
- package/templates/app/devtools/config/page.tsx +0 -31
- package/templates/app/devtools/features/page.tsx +0 -31
- package/templates/app/devtools/flows/page.tsx +0 -31
- package/templates/app/devtools/layout.tsx +0 -58
- package/templates/app/devtools/page.tsx +0 -121
- package/templates/app/devtools/scheduled-actions/page.tsx +0 -157
- package/templates/app/devtools/style/page.tsx +0 -330
- package/templates/app/devtools/tags/page.tsx +0 -31
- package/templates/app/devtools/tests/[[...path]]/page.tsx +0 -47
- package/templates/app/favicon.ico +0 -0
- package/templates/app/globals.css +0 -12
- package/templates/app/layout.tsx +0 -96
- package/templates/app/public/page.tsx +0 -30
- package/templates/app/superadmin/docs/[section]/[page]/page.tsx +0 -92
- package/templates/app/superadmin/docs/page.tsx +0 -75
- package/templates/app/superadmin/layout.tsx +0 -67
- package/templates/app/superadmin/page.tsx +0 -149
- package/templates/app/superadmin/subscriptions/page.tsx +0 -655
- package/templates/app/superadmin/team-roles/page.tsx +0 -493
- package/templates/app/superadmin/teams/[teamId]/page.tsx +0 -687
- package/templates/app/superadmin/teams/page.tsx +0 -302
- package/templates/app/superadmin/users/[userId]/page.tsx +0 -548
- package/templates/app/superadmin/users/page.tsx +0 -528
|
@@ -1,133 +0,0 @@
|
|
|
1
|
-
import { NextResponse } from 'next/server'
|
|
2
|
-
import { auth } from '@nextsparkjs/core/lib/auth'
|
|
3
|
-
import { headers } from 'next/headers'
|
|
4
|
-
import { queryOneWithRLS, mutateWithRLS, queryOne } from '@nextsparkjs/core/lib/db'
|
|
5
|
-
import { profileSchema } from '@nextsparkjs/core/lib/validation'
|
|
6
|
-
import { MetaService } from '@nextsparkjs/core/lib/services/meta.service'
|
|
7
|
-
|
|
8
|
-
export async function GET(request: Request) {
|
|
9
|
-
const url = new URL(request.url)
|
|
10
|
-
const includeMeta = url.searchParams.get('includeMeta') === 'true'
|
|
11
|
-
try {
|
|
12
|
-
const sessionHeaders = await headers()
|
|
13
|
-
const session = await auth.api.getSession({ headers: sessionHeaders })
|
|
14
|
-
|
|
15
|
-
if (!session?.user) {
|
|
16
|
-
return NextResponse.json({ error: 'Unauthorized' }, { status: 401 })
|
|
17
|
-
}
|
|
18
|
-
|
|
19
|
-
// Get user profile data including the new fields
|
|
20
|
-
const user = await queryOneWithRLS(
|
|
21
|
-
'SELECT id, email, "firstName", "lastName", country, timezone, language, image, "emailVerified", "createdAt", "updatedAt" FROM "users" WHERE id = $1',
|
|
22
|
-
[session.user.id],
|
|
23
|
-
session.user.id
|
|
24
|
-
)
|
|
25
|
-
|
|
26
|
-
if (!user) {
|
|
27
|
-
return NextResponse.json({ error: 'User not found' }, { status: 404 })
|
|
28
|
-
}
|
|
29
|
-
|
|
30
|
-
// Get auth method (check if user has password or uses OAuth) - using direct query for auth table
|
|
31
|
-
const account = await queryOne<{ providerId: string }>(
|
|
32
|
-
'SELECT "providerId" FROM "account" WHERE "userId" = $1',
|
|
33
|
-
[session.user.id]
|
|
34
|
-
)
|
|
35
|
-
|
|
36
|
-
const authMethod = account
|
|
37
|
-
? account.providerId === 'credential' ? 'Email' : 'Google'
|
|
38
|
-
: 'Email'
|
|
39
|
-
|
|
40
|
-
let result: Record<string, unknown> = {
|
|
41
|
-
...user,
|
|
42
|
-
authMethod
|
|
43
|
-
}
|
|
44
|
-
|
|
45
|
-
// Include metadata if requested
|
|
46
|
-
if (includeMeta) {
|
|
47
|
-
const metadata = await MetaService.getEntityMetas('user', session.user.id, session.user.id, true)
|
|
48
|
-
result = { ...result, meta: metadata }
|
|
49
|
-
}
|
|
50
|
-
|
|
51
|
-
return NextResponse.json(result)
|
|
52
|
-
} catch (error) {
|
|
53
|
-
console.error('Error fetching user profile:', error)
|
|
54
|
-
return NextResponse.json({ error: 'Internal server error' }, { status: 500 })
|
|
55
|
-
}
|
|
56
|
-
}
|
|
57
|
-
|
|
58
|
-
export async function PATCH(request: Request) {
|
|
59
|
-
try {
|
|
60
|
-
const sessionHeaders = await headers()
|
|
61
|
-
const session = await auth.api.getSession({ headers: sessionHeaders })
|
|
62
|
-
|
|
63
|
-
if (!session?.user) {
|
|
64
|
-
return NextResponse.json({ error: 'Unauthorized' }, { status: 401 })
|
|
65
|
-
}
|
|
66
|
-
|
|
67
|
-
const body = await request.json()
|
|
68
|
-
const { meta, ...profileData } = body
|
|
69
|
-
|
|
70
|
-
// Handle metadata-only updates (formato anidado)
|
|
71
|
-
if (meta && Object.keys(profileData).length === 0) {
|
|
72
|
-
// Procesar cada grupo de metadata por separado
|
|
73
|
-
for (const [metaKey, metaValue] of Object.entries(meta)) {
|
|
74
|
-
if (metaValue && typeof metaValue === 'object') {
|
|
75
|
-
await MetaService.setEntityMeta('user', session.user.id, metaKey, metaValue, session.user.id)
|
|
76
|
-
}
|
|
77
|
-
}
|
|
78
|
-
|
|
79
|
-
return NextResponse.json({
|
|
80
|
-
message: 'Settings updated successfully',
|
|
81
|
-
success: true
|
|
82
|
-
})
|
|
83
|
-
}
|
|
84
|
-
|
|
85
|
-
// Handle profile data updates
|
|
86
|
-
if (Object.keys(profileData).length > 0) {
|
|
87
|
-
// Validate the request body
|
|
88
|
-
const validationResult = profileSchema.safeParse(profileData)
|
|
89
|
-
if (!validationResult.success) {
|
|
90
|
-
return NextResponse.json({
|
|
91
|
-
error: 'Invalid data',
|
|
92
|
-
details: validationResult.error.issues
|
|
93
|
-
}, { status: 400 })
|
|
94
|
-
}
|
|
95
|
-
|
|
96
|
-
const { firstName, lastName, country, timezone, language } = validationResult.data
|
|
97
|
-
|
|
98
|
-
// Update user profile
|
|
99
|
-
const result = await mutateWithRLS(
|
|
100
|
-
`UPDATE "users"
|
|
101
|
-
SET "firstName" = $1, "lastName" = $2, country = $3, timezone = $4, language = $5, "updatedAt" = CURRENT_TIMESTAMP
|
|
102
|
-
WHERE id = $6
|
|
103
|
-
RETURNING id, email, "firstName", "lastName", country, timezone, language, image, "emailVerified", "createdAt", "updatedAt"`,
|
|
104
|
-
[firstName, lastName, country, timezone, language, session.user.id],
|
|
105
|
-
session.user.id
|
|
106
|
-
)
|
|
107
|
-
|
|
108
|
-
const updatedUser = result.rows[0]
|
|
109
|
-
|
|
110
|
-
if (!updatedUser) {
|
|
111
|
-
return NextResponse.json({ error: 'User not found' }, { status: 404 })
|
|
112
|
-
}
|
|
113
|
-
}
|
|
114
|
-
|
|
115
|
-
// Handle metadata updates if provided (formato anidado)
|
|
116
|
-
if (meta) {
|
|
117
|
-
// Procesar cada grupo de metadata por separado
|
|
118
|
-
for (const [metaKey, metaValue] of Object.entries(meta)) {
|
|
119
|
-
if (metaValue && typeof metaValue === 'object') {
|
|
120
|
-
await MetaService.setEntityMeta('user', session.user.id, metaKey, metaValue, session.user.id)
|
|
121
|
-
}
|
|
122
|
-
}
|
|
123
|
-
}
|
|
124
|
-
|
|
125
|
-
return NextResponse.json({
|
|
126
|
-
message: 'Profile updated successfully',
|
|
127
|
-
success: true
|
|
128
|
-
})
|
|
129
|
-
} catch (error) {
|
|
130
|
-
console.error('Error updating user profile:', error)
|
|
131
|
-
return NextResponse.json({ error: 'Internal server error' }, { status: 500 })
|
|
132
|
-
}
|
|
133
|
-
}
|
|
@@ -1,210 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* API endpoint for individual child entity operations
|
|
3
|
-
* PUT /api/v1/[entity]/[id]/child/[childType]/[childId]
|
|
4
|
-
* DELETE /api/v1/[entity]/[id]/child/[childType]/[childId]
|
|
5
|
-
*/
|
|
6
|
-
|
|
7
|
-
// CRITICAL: Initialize entity registry for API routes
|
|
8
|
-
// This import is processed by webpack which resolves the @nextsparkjs alias
|
|
9
|
-
// The setEntityRegistry call happens at module load time
|
|
10
|
-
import { setEntityRegistry, isRegistryInitialized, getChildEntities, getEntity } from '@nextsparkjs/core/lib/entities/queries'
|
|
11
|
-
import { ENTITY_REGISTRY, ENTITY_METADATA } from '@nextsparkjs/registries/entity-registry'
|
|
12
|
-
if (!isRegistryInitialized()) {
|
|
13
|
-
setEntityRegistry(ENTITY_REGISTRY, ENTITY_METADATA)
|
|
14
|
-
}
|
|
15
|
-
|
|
16
|
-
import { NextRequest, NextResponse } from 'next/server'
|
|
17
|
-
import { queryWithRLS } from '@nextsparkjs/core/lib/db'
|
|
18
|
-
import { resolveEntityFromUrl } from '@nextsparkjs/core/lib/api/entity/resolver'
|
|
19
|
-
|
|
20
|
-
interface RouteParams {
|
|
21
|
-
entity: string
|
|
22
|
-
id: string
|
|
23
|
-
childType: string
|
|
24
|
-
childId: string
|
|
25
|
-
}
|
|
26
|
-
|
|
27
|
-
export async function PUT(
|
|
28
|
-
request: NextRequest,
|
|
29
|
-
{ params }: { params: Promise<RouteParams> }
|
|
30
|
-
) {
|
|
31
|
-
const { entity, id, childType, childId } = await params
|
|
32
|
-
|
|
33
|
-
try {
|
|
34
|
-
const body = await request.json()
|
|
35
|
-
|
|
36
|
-
// Resolve entity from URL (handles plural to singular conversion)
|
|
37
|
-
const resolution = await resolveEntityFromUrl(request.nextUrl.pathname)
|
|
38
|
-
if (!resolution.isValidEntity || !resolution.entityConfig) {
|
|
39
|
-
return NextResponse.json(
|
|
40
|
-
{ error: `Entity "${entity}" not found` },
|
|
41
|
-
{ status: 404 }
|
|
42
|
-
)
|
|
43
|
-
}
|
|
44
|
-
|
|
45
|
-
// Get child entities for this parent entity from registry
|
|
46
|
-
const childEntities = getChildEntities(resolution.entityName as string)
|
|
47
|
-
const childEntity = childEntities.find(child => child.name === childType)
|
|
48
|
-
|
|
49
|
-
if (!childEntity) {
|
|
50
|
-
return NextResponse.json(
|
|
51
|
-
{ error: `Child entity "${childType}" not found for "${entity}"` },
|
|
52
|
-
{ status: 404 }
|
|
53
|
-
)
|
|
54
|
-
}
|
|
55
|
-
|
|
56
|
-
// Get child entity configuration from registry
|
|
57
|
-
const childConfig = getEntity(childType as string)
|
|
58
|
-
if (!childConfig) {
|
|
59
|
-
return NextResponse.json(
|
|
60
|
-
{ error: `Child entity configuration "${childType}" not found` },
|
|
61
|
-
{ status: 404 }
|
|
62
|
-
)
|
|
63
|
-
}
|
|
64
|
-
|
|
65
|
-
// Prepare data
|
|
66
|
-
const childTable = childEntity.tableName
|
|
67
|
-
const parentIdColumn = 'parentId' // Use consistent naming
|
|
68
|
-
|
|
69
|
-
// Extract field names and values for update (exclude system fields)
|
|
70
|
-
const systemFields = ['id', 'parentId', 'createdAt', 'updatedAt']
|
|
71
|
-
const userFields = childConfig.fields.filter(field => !systemFields.includes(field.name))
|
|
72
|
-
const fieldNames = userFields.map(field => field.name)
|
|
73
|
-
const updateClauses = fieldNames
|
|
74
|
-
.filter(name => body[name] !== undefined)
|
|
75
|
-
.map(name => {
|
|
76
|
-
const value = body[name]
|
|
77
|
-
if (value === null) {
|
|
78
|
-
return `"${name}" = NULL`
|
|
79
|
-
}
|
|
80
|
-
|
|
81
|
-
const field = childConfig.fields.find(f => f.name === name)
|
|
82
|
-
|
|
83
|
-
// Handle relation fields specially - extract single ID from array or object
|
|
84
|
-
if (field?.type === 'relation') {
|
|
85
|
-
if (Array.isArray(value) && value.length > 0) {
|
|
86
|
-
const firstItem = value[0]
|
|
87
|
-
const relationId = typeof firstItem === 'object' && firstItem && 'id' in firstItem ? firstItem.id : firstItem
|
|
88
|
-
return `"${name}" = '${String(relationId).replace(/'/g, "''")}'`
|
|
89
|
-
} else if (typeof value === 'object' && value && 'id' in value) {
|
|
90
|
-
return `"${name}" = '${String((value as { id: unknown }).id).replace(/'/g, "''")}'`
|
|
91
|
-
} else if (typeof value === 'string') {
|
|
92
|
-
return `"${name}" = '${String(value).replace(/'/g, "''")}'`
|
|
93
|
-
}
|
|
94
|
-
return `"${name}" = NULL`
|
|
95
|
-
}
|
|
96
|
-
|
|
97
|
-
// Handle relation-multi fields - store as JSONB array of IDs
|
|
98
|
-
if (field?.type === 'relation-multi') {
|
|
99
|
-
let relationIds = []
|
|
100
|
-
if (Array.isArray(value)) {
|
|
101
|
-
relationIds = value.map(item =>
|
|
102
|
-
typeof item === 'object' && item && 'id' in item ? item.id : item
|
|
103
|
-
).filter(id => id && String(id).trim() !== '')
|
|
104
|
-
}
|
|
105
|
-
return `"${name}" = '${JSON.stringify(relationIds).replace(/'/g, "''")}'::jsonb`
|
|
106
|
-
}
|
|
107
|
-
|
|
108
|
-
// Handle multiselect, user, and other complex types as JSONB
|
|
109
|
-
if (field?.type === 'multiselect' || field?.type === 'user' ||
|
|
110
|
-
Array.isArray(value) || (typeof value === 'object' && value !== null)) {
|
|
111
|
-
return `"${name}" = '${JSON.stringify(value).replace(/'/g, "''")}'::jsonb`
|
|
112
|
-
}
|
|
113
|
-
|
|
114
|
-
return `"${name}" = '${String(value).replace(/'/g, "''")}'`
|
|
115
|
-
})
|
|
116
|
-
|
|
117
|
-
// Add updatedAt
|
|
118
|
-
updateClauses.push(`"updatedAt" = NOW()`)
|
|
119
|
-
|
|
120
|
-
const query = `
|
|
121
|
-
UPDATE "${childTable}"
|
|
122
|
-
SET ${updateClauses.join(', ')}
|
|
123
|
-
WHERE "id" = $1 AND "${parentIdColumn}" = $2
|
|
124
|
-
RETURNING *
|
|
125
|
-
`
|
|
126
|
-
|
|
127
|
-
const result = await queryWithRLS<Record<string, unknown>>(query, [childId, id], 'system')
|
|
128
|
-
|
|
129
|
-
if (result.length === 0) {
|
|
130
|
-
return NextResponse.json(
|
|
131
|
-
{ error: 'Child entity not found' },
|
|
132
|
-
{ status: 404 }
|
|
133
|
-
)
|
|
134
|
-
}
|
|
135
|
-
|
|
136
|
-
return NextResponse.json({
|
|
137
|
-
success: true,
|
|
138
|
-
data: result[0],
|
|
139
|
-
info: {
|
|
140
|
-
timestamp: new Date().toISOString()
|
|
141
|
-
}
|
|
142
|
-
})
|
|
143
|
-
} catch (error) {
|
|
144
|
-
console.error(`Error updating child entity:`, error)
|
|
145
|
-
return NextResponse.json(
|
|
146
|
-
{ error: 'Internal server error' },
|
|
147
|
-
{ status: 500 }
|
|
148
|
-
)
|
|
149
|
-
}
|
|
150
|
-
}
|
|
151
|
-
|
|
152
|
-
export async function DELETE(
|
|
153
|
-
request: NextRequest,
|
|
154
|
-
{ params }: { params: Promise<RouteParams> }
|
|
155
|
-
) {
|
|
156
|
-
try {
|
|
157
|
-
const { entity, id, childType, childId } = await params
|
|
158
|
-
|
|
159
|
-
// Resolve entity from URL (handles plural to singular conversion)
|
|
160
|
-
const resolution = await resolveEntityFromUrl(request.nextUrl.pathname)
|
|
161
|
-
if (!resolution.isValidEntity || !resolution.entityConfig) {
|
|
162
|
-
return NextResponse.json(
|
|
163
|
-
{ error: `Entity "${entity}" not found` },
|
|
164
|
-
{ status: 404 }
|
|
165
|
-
)
|
|
166
|
-
}
|
|
167
|
-
|
|
168
|
-
// Get child entities for this parent entity from registry
|
|
169
|
-
const childEntities = getChildEntities(resolution.entityName as string)
|
|
170
|
-
const childEntity = childEntities.find(child => child.name === childType)
|
|
171
|
-
|
|
172
|
-
if (!childEntity) {
|
|
173
|
-
return NextResponse.json(
|
|
174
|
-
{ error: `Child entity "${childType}" not found for "${entity}"` },
|
|
175
|
-
{ status: 404 }
|
|
176
|
-
)
|
|
177
|
-
}
|
|
178
|
-
|
|
179
|
-
// Delete child entity
|
|
180
|
-
const childTable = childEntity.tableName
|
|
181
|
-
const parentIdColumn = 'parentId' // Use consistent naming
|
|
182
|
-
|
|
183
|
-
const query = `
|
|
184
|
-
DELETE FROM "${childTable}"
|
|
185
|
-
WHERE "id" = $1 AND "${parentIdColumn}" = $2
|
|
186
|
-
RETURNING id
|
|
187
|
-
`
|
|
188
|
-
|
|
189
|
-
const result = await queryWithRLS<{ id: string }>(query, [childId, id], 'system')
|
|
190
|
-
|
|
191
|
-
if (result.length === 0) {
|
|
192
|
-
return NextResponse.json(
|
|
193
|
-
{ error: 'Child entity not found' },
|
|
194
|
-
{ status: 404 }
|
|
195
|
-
)
|
|
196
|
-
}
|
|
197
|
-
|
|
198
|
-
return NextResponse.json({
|
|
199
|
-
success: true,
|
|
200
|
-
message: 'Child entity deleted successfully',
|
|
201
|
-
id: result[0].id
|
|
202
|
-
})
|
|
203
|
-
} catch (error) {
|
|
204
|
-
console.error(`Error deleting child entity:`, error)
|
|
205
|
-
return NextResponse.json(
|
|
206
|
-
{ error: 'Internal server error' },
|
|
207
|
-
{ status: 500 }
|
|
208
|
-
)
|
|
209
|
-
}
|
|
210
|
-
}
|
|
@@ -1,331 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* API endpoint for child entities
|
|
3
|
-
* GET /api/v1/[entity]/[id]/child/[childType]
|
|
4
|
-
* POST /api/v1/[entity]/[id]/child/[childType]
|
|
5
|
-
*/
|
|
6
|
-
|
|
7
|
-
// CRITICAL: Initialize entity registry for API routes
|
|
8
|
-
// This import is processed by webpack which resolves the @nextsparkjs alias
|
|
9
|
-
// The setEntityRegistry call happens at module load time
|
|
10
|
-
import { setEntityRegistry, isRegistryInitialized, getChildEntities, getEntity } from '@nextsparkjs/core/lib/entities/queries'
|
|
11
|
-
import { ENTITY_REGISTRY, ENTITY_METADATA } from '@nextsparkjs/registries/entity-registry'
|
|
12
|
-
if (!isRegistryInitialized()) {
|
|
13
|
-
setEntityRegistry(ENTITY_REGISTRY, ENTITY_METADATA)
|
|
14
|
-
}
|
|
15
|
-
|
|
16
|
-
import { NextRequest, NextResponse } from 'next/server'
|
|
17
|
-
import { queryWithRLS } from '@nextsparkjs/core/lib/db'
|
|
18
|
-
import { resolveEntityFromUrl } from '@nextsparkjs/core/lib/api/entity/resolver'
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
interface RouteParams {
|
|
22
|
-
entity: string
|
|
23
|
-
id: string
|
|
24
|
-
childType: string
|
|
25
|
-
}
|
|
26
|
-
|
|
27
|
-
export async function GET(
|
|
28
|
-
request: NextRequest,
|
|
29
|
-
{ params }: { params: Promise<RouteParams> }
|
|
30
|
-
) {
|
|
31
|
-
const { entity, id, childType } = await params
|
|
32
|
-
|
|
33
|
-
try {
|
|
34
|
-
|
|
35
|
-
// Resolve entity from URL (handles plural to singular conversion)
|
|
36
|
-
const resolution = await resolveEntityFromUrl(request.nextUrl.pathname)
|
|
37
|
-
if (!resolution.isValidEntity || !resolution.entityConfig) {
|
|
38
|
-
return NextResponse.json(
|
|
39
|
-
{ error: `Entity "${entity}" not found` },
|
|
40
|
-
{ status: 404 }
|
|
41
|
-
)
|
|
42
|
-
}
|
|
43
|
-
|
|
44
|
-
// Check if child entity exists using the new registry system
|
|
45
|
-
const childEntities = getChildEntities(resolution.entityName as string)
|
|
46
|
-
const childEntity = childEntities.find(child => child.name === childType)
|
|
47
|
-
|
|
48
|
-
if (!childEntity) {
|
|
49
|
-
return NextResponse.json(
|
|
50
|
-
{ error: `Child entity "${childType}" not found for "${entity}"` },
|
|
51
|
-
{ status: 404 }
|
|
52
|
-
)
|
|
53
|
-
}
|
|
54
|
-
|
|
55
|
-
// Get child config from registry
|
|
56
|
-
const childConfig = getEntity(childType as string)
|
|
57
|
-
if (!childConfig) {
|
|
58
|
-
return NextResponse.json(
|
|
59
|
-
{ error: `Child entity configuration "${childType}" not found` },
|
|
60
|
-
{ status: 404 }
|
|
61
|
-
)
|
|
62
|
-
}
|
|
63
|
-
|
|
64
|
-
// Query child entities
|
|
65
|
-
const childTable = childEntity.tableName
|
|
66
|
-
const parentIdColumn = 'parentId' // Child entities use 'parentId' as foreign key
|
|
67
|
-
|
|
68
|
-
const query = `
|
|
69
|
-
SELECT * FROM "${childTable}"
|
|
70
|
-
WHERE "${parentIdColumn}" = $1
|
|
71
|
-
ORDER BY "createdAt" DESC
|
|
72
|
-
`
|
|
73
|
-
|
|
74
|
-
const childRows = await queryWithRLS<Record<string, unknown>>(query, [id], 'system')
|
|
75
|
-
|
|
76
|
-
return NextResponse.json({
|
|
77
|
-
success: true,
|
|
78
|
-
data: childRows,
|
|
79
|
-
info: {
|
|
80
|
-
timestamp: new Date().toISOString()
|
|
81
|
-
}
|
|
82
|
-
})
|
|
83
|
-
} catch (error) {
|
|
84
|
-
console.error(`Error loading child entities for ${entity}/${id}/${childType}:`, error)
|
|
85
|
-
return NextResponse.json(
|
|
86
|
-
{ error: 'Internal server error' },
|
|
87
|
-
{ status: 500 }
|
|
88
|
-
)
|
|
89
|
-
}
|
|
90
|
-
}
|
|
91
|
-
|
|
92
|
-
export async function POST(
|
|
93
|
-
request: NextRequest,
|
|
94
|
-
{ params }: { params: Promise<RouteParams> }
|
|
95
|
-
) {
|
|
96
|
-
const { entity, id, childType } = await params
|
|
97
|
-
|
|
98
|
-
try {
|
|
99
|
-
console.log(`[ChildAPI] POST ${entity}/${id}/child/${childType}`)
|
|
100
|
-
const body = await request.json()
|
|
101
|
-
console.log(`[ChildAPI] Request body:`, body)
|
|
102
|
-
|
|
103
|
-
// Resolve entity from URL (handles plural to singular conversion)
|
|
104
|
-
const resolution = await resolveEntityFromUrl(request.nextUrl.pathname)
|
|
105
|
-
console.log(`[ChildAPI] Resolution:`, resolution)
|
|
106
|
-
if (!resolution.isValidEntity || !resolution.entityConfig) {
|
|
107
|
-
return NextResponse.json(
|
|
108
|
-
{ error: `Entity "${entity}" not found` },
|
|
109
|
-
{ status: 404 }
|
|
110
|
-
)
|
|
111
|
-
}
|
|
112
|
-
|
|
113
|
-
// Check if child entity exists using the new registry system
|
|
114
|
-
const childEntities = getChildEntities(resolution.entityName as string)
|
|
115
|
-
const childEntity = childEntities.find(child => child.name === childType)
|
|
116
|
-
console.log(`[ChildAPI] Found child entities for ${entity}:`, childEntities.map(c => c.name))
|
|
117
|
-
console.log(`[ChildAPI] Looking for childType:`, childType)
|
|
118
|
-
console.log(`[ChildAPI] Found child entity:`, childEntity)
|
|
119
|
-
|
|
120
|
-
if (!childEntity) {
|
|
121
|
-
return NextResponse.json(
|
|
122
|
-
{ error: `Child entity "${childType}" not found for "${entity}"` },
|
|
123
|
-
{ status: 404 }
|
|
124
|
-
)
|
|
125
|
-
}
|
|
126
|
-
|
|
127
|
-
// Get child config from registry
|
|
128
|
-
const childConfig = getEntity(childType as string)
|
|
129
|
-
console.log(`[ChildAPI] Child config for ${childType}:`, childConfig)
|
|
130
|
-
if (!childConfig) {
|
|
131
|
-
return NextResponse.json(
|
|
132
|
-
{ error: `Child entity configuration "${childType}" not found` },
|
|
133
|
-
{ status: 404 }
|
|
134
|
-
)
|
|
135
|
-
}
|
|
136
|
-
|
|
137
|
-
// Prepare data
|
|
138
|
-
const childTable = childEntity.tableName
|
|
139
|
-
const parentIdColumn = 'parentId' // Child entities use 'parentId' as foreign key
|
|
140
|
-
|
|
141
|
-
// Extract field names and values (exclude system fields that we'll add separately)
|
|
142
|
-
const systemFields = ['id', 'parentId', 'createdAt', 'updatedAt']
|
|
143
|
-
const userFields = childConfig.fields.filter(field => !systemFields.includes(field.name))
|
|
144
|
-
const fieldNames = userFields.map(field => field.name)
|
|
145
|
-
const fieldValues = fieldNames.map(name => body[name])
|
|
146
|
-
|
|
147
|
-
// Determine ID generation strategy (default: uuid)
|
|
148
|
-
const idStrategy = childConfig.idStrategy?.type || 'uuid'
|
|
149
|
-
|
|
150
|
-
// Add parent ID and system fields based on ID strategy
|
|
151
|
-
let allFields: string[]
|
|
152
|
-
let allValues: string[]
|
|
153
|
-
|
|
154
|
-
if (idStrategy === 'serial') {
|
|
155
|
-
// SERIAL: Let database generate ID via DEFAULT/SERIAL
|
|
156
|
-
allFields = [parentIdColumn, ...fieldNames, 'createdAt', 'updatedAt']
|
|
157
|
-
allValues = [
|
|
158
|
-
`'${id}'`, // parentId
|
|
159
|
-
...fieldValues.map((value, index) => {
|
|
160
|
-
|
|
161
|
-
if (value === null || value === undefined) {
|
|
162
|
-
return 'NULL'
|
|
163
|
-
}
|
|
164
|
-
|
|
165
|
-
const fieldName = fieldNames[index]
|
|
166
|
-
const field = childConfig.fields.find(f => f.name === fieldName)
|
|
167
|
-
|
|
168
|
-
// Handle number fields - convert empty strings to NULL
|
|
169
|
-
if (field?.type === 'number') {
|
|
170
|
-
if (typeof value === 'string' && value.trim() === '') {
|
|
171
|
-
return 'NULL'
|
|
172
|
-
}
|
|
173
|
-
// Convert to number and validate
|
|
174
|
-
const numValue = Number(value)
|
|
175
|
-
if (isNaN(numValue)) {
|
|
176
|
-
return 'NULL'
|
|
177
|
-
}
|
|
178
|
-
return String(numValue)
|
|
179
|
-
}
|
|
180
|
-
|
|
181
|
-
// Handle relation fields specially - extract single ID from array or object
|
|
182
|
-
if (field?.type === 'relation') {
|
|
183
|
-
if (Array.isArray(value) && value.length > 0) {
|
|
184
|
-
const firstItem = value[0]
|
|
185
|
-
const relationId = typeof firstItem === 'object' && firstItem && 'id' in firstItem ? firstItem.id : firstItem
|
|
186
|
-
// Check if relationId is empty string or null/undefined
|
|
187
|
-
if (!relationId || String(relationId).trim() === '') {
|
|
188
|
-
return 'NULL'
|
|
189
|
-
}
|
|
190
|
-
return `'${String(relationId).replace(/'/g, "''")}'`
|
|
191
|
-
} else if (typeof value === 'object' && value && 'id' in value) {
|
|
192
|
-
const relationId = (value as { id: unknown }).id
|
|
193
|
-
// Check if relationId is empty string or null/undefined
|
|
194
|
-
if (!relationId || String(relationId).trim() === '') {
|
|
195
|
-
return 'NULL'
|
|
196
|
-
}
|
|
197
|
-
return `'${String(relationId).replace(/'/g, "''")}'`
|
|
198
|
-
} else if (typeof value === 'string') {
|
|
199
|
-
// Check if string is empty or just whitespace
|
|
200
|
-
if (value.trim() === '') {
|
|
201
|
-
return 'NULL'
|
|
202
|
-
}
|
|
203
|
-
return `'${String(value).replace(/'/g, "''")}'`
|
|
204
|
-
}
|
|
205
|
-
return 'NULL'
|
|
206
|
-
}
|
|
207
|
-
|
|
208
|
-
// Handle relation-multi fields - store as JSONB array of IDs
|
|
209
|
-
if (field?.type === 'relation-multi') {
|
|
210
|
-
let relationIds = []
|
|
211
|
-
if (Array.isArray(value)) {
|
|
212
|
-
relationIds = value.map(item =>
|
|
213
|
-
typeof item === 'object' && item && 'id' in item ? item.id : item
|
|
214
|
-
).filter(id => id && String(id).trim() !== '')
|
|
215
|
-
}
|
|
216
|
-
return `'${JSON.stringify(relationIds).replace(/'/g, "''")}'::jsonb`
|
|
217
|
-
}
|
|
218
|
-
|
|
219
|
-
// Handle multiselect, user, and other complex types as JSONB
|
|
220
|
-
if (field?.type === 'multiselect' || field?.type === 'user' ||
|
|
221
|
-
Array.isArray(value) || (typeof value === 'object' && value !== null)) {
|
|
222
|
-
return `'${JSON.stringify(value).replace(/'/g, "''")}'::jsonb`
|
|
223
|
-
}
|
|
224
|
-
|
|
225
|
-
return `'${String(value).replace(/'/g, "''")}'`
|
|
226
|
-
}), // properly handle null values and complex types
|
|
227
|
-
'NOW()', // createdAt
|
|
228
|
-
'NOW()' // updatedAt
|
|
229
|
-
]
|
|
230
|
-
} else {
|
|
231
|
-
// UUID: Generate ID and include in INSERT
|
|
232
|
-
allFields = ['id', parentIdColumn, ...fieldNames, 'createdAt', 'updatedAt']
|
|
233
|
-
allValues = [
|
|
234
|
-
`gen_random_uuid()::TEXT`, // id
|
|
235
|
-
`'${id}'`, // parentId
|
|
236
|
-
...fieldValues.map((value, index) => {
|
|
237
|
-
if (value === null || value === undefined) {
|
|
238
|
-
return 'NULL'
|
|
239
|
-
}
|
|
240
|
-
|
|
241
|
-
const fieldName = fieldNames[index]
|
|
242
|
-
const field = childConfig.fields.find(f => f.name === fieldName)
|
|
243
|
-
|
|
244
|
-
// Handle number fields - convert empty strings to NULL
|
|
245
|
-
if (field?.type === 'number') {
|
|
246
|
-
if (typeof value === 'string' && value.trim() === '') {
|
|
247
|
-
return 'NULL'
|
|
248
|
-
}
|
|
249
|
-
// Convert to number and validate
|
|
250
|
-
const numValue = Number(value)
|
|
251
|
-
if (isNaN(numValue)) {
|
|
252
|
-
return 'NULL'
|
|
253
|
-
}
|
|
254
|
-
return String(numValue)
|
|
255
|
-
}
|
|
256
|
-
|
|
257
|
-
// Handle relation fields specially - extract single ID from array or object
|
|
258
|
-
if (field?.type === 'relation') {
|
|
259
|
-
if (Array.isArray(value) && value.length > 0) {
|
|
260
|
-
const firstItem = value[0]
|
|
261
|
-
const relationId = typeof firstItem === 'object' && firstItem && 'id' in firstItem ? firstItem.id : firstItem
|
|
262
|
-
// Check if relationId is empty string or null/undefined
|
|
263
|
-
if (!relationId || String(relationId).trim() === '') {
|
|
264
|
-
return 'NULL'
|
|
265
|
-
}
|
|
266
|
-
return `'${String(relationId).replace(/'/g, "''")}'`
|
|
267
|
-
} else if (typeof value === 'object' && value && 'id' in value) {
|
|
268
|
-
const relationId = (value as { id: unknown }).id
|
|
269
|
-
// Check if relationId is empty string or null/undefined
|
|
270
|
-
if (!relationId || String(relationId).trim() === '') {
|
|
271
|
-
return 'NULL'
|
|
272
|
-
}
|
|
273
|
-
return `'${String(relationId).replace(/'/g, "''")}'`
|
|
274
|
-
} else if (typeof value === 'string') {
|
|
275
|
-
// Check if string is empty or just whitespace
|
|
276
|
-
if (value.trim() === '') {
|
|
277
|
-
return 'NULL'
|
|
278
|
-
}
|
|
279
|
-
return `'${String(value).replace(/'/g, "''")}'`
|
|
280
|
-
}
|
|
281
|
-
return 'NULL'
|
|
282
|
-
}
|
|
283
|
-
|
|
284
|
-
// Handle relation-multi fields - store as JSONB array of IDs
|
|
285
|
-
if (field?.type === 'relation-multi') {
|
|
286
|
-
let relationIds = []
|
|
287
|
-
if (Array.isArray(value)) {
|
|
288
|
-
relationIds = value.map(item =>
|
|
289
|
-
typeof item === 'object' && item && 'id' in item ? item.id : item
|
|
290
|
-
).filter(id => id && String(id).trim() !== '')
|
|
291
|
-
}
|
|
292
|
-
return `'${JSON.stringify(relationIds).replace(/'/g, "''")}'::jsonb`
|
|
293
|
-
}
|
|
294
|
-
|
|
295
|
-
// Handle multiselect, user, and other complex types as JSONB
|
|
296
|
-
if (field?.type === 'multiselect' || field?.type === 'user' ||
|
|
297
|
-
Array.isArray(value) || (typeof value === 'object' && value !== null)) {
|
|
298
|
-
return `'${JSON.stringify(value).replace(/'/g, "''")}'::jsonb`
|
|
299
|
-
}
|
|
300
|
-
|
|
301
|
-
return `'${String(value).replace(/'/g, "''")}'`
|
|
302
|
-
}), // properly handle null values and complex types
|
|
303
|
-
'NOW()', // createdAt
|
|
304
|
-
'NOW()' // updatedAt
|
|
305
|
-
]
|
|
306
|
-
}
|
|
307
|
-
|
|
308
|
-
const query = `
|
|
309
|
-
INSERT INTO "${childTable}" (${allFields.map(f => `"${f}"`).join(', ')})
|
|
310
|
-
VALUES (${allValues.join(', ')})
|
|
311
|
-
RETURNING *
|
|
312
|
-
`
|
|
313
|
-
|
|
314
|
-
const result = await queryWithRLS<Record<string, unknown>>(query, [], 'system')
|
|
315
|
-
|
|
316
|
-
return NextResponse.json({
|
|
317
|
-
success: true,
|
|
318
|
-
data: result[0],
|
|
319
|
-
info: {
|
|
320
|
-
timestamp: new Date().toISOString()
|
|
321
|
-
}
|
|
322
|
-
})
|
|
323
|
-
} catch (error) {
|
|
324
|
-
console.error(`[ChildAPI] Error creating child entity for ${entity}/${id}/${childType}:`, error)
|
|
325
|
-
console.error(`[ChildAPI] Error stack:`, error instanceof Error ? error.stack : 'No stack available')
|
|
326
|
-
return NextResponse.json(
|
|
327
|
-
{ error: 'Internal server error', details: error instanceof Error ? error.message : String(error) },
|
|
328
|
-
{ status: 500 }
|
|
329
|
-
)
|
|
330
|
-
}
|
|
331
|
-
}
|