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