@nextsparkjs/core 0.1.0-beta.81 → 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 (376) 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/templates/next.config.mjs +5 -2
  10. package/dist/types/theme.d.ts +2 -0
  11. package/dist/types/theme.d.ts.map +1 -1
  12. package/package.json +16 -16
  13. package/scripts/build/docs-registry.mjs +0 -0
  14. package/scripts/create-theme.mjs +0 -0
  15. package/scripts/deploy/release-version.mjs +0 -0
  16. package/scripts/deploy/vercel-deploy.mjs +0 -0
  17. package/scripts/dev/watch-plugins.mjs +0 -0
  18. package/scripts/maintenance/update-core.mjs +0 -0
  19. package/scripts/setup/npm-postinstall.mjs +0 -0
  20. package/scripts/setup/setup-claude.mjs +0 -0
  21. package/scripts/validation/check-imports.sh +0 -0
  22. package/templates/next.config.mjs +5 -2
  23. package/dist/templates/app/(auth)/forgot-password/page.tsx +0 -216
  24. package/dist/templates/app/(auth)/layout.tsx +0 -51
  25. package/dist/templates/app/(auth)/login/page.tsx +0 -21
  26. package/dist/templates/app/(auth)/reset-password/page.tsx +0 -212
  27. package/dist/templates/app/(auth)/signup/page.tsx +0 -21
  28. package/dist/templates/app/(auth)/verify-email/page.tsx +0 -190
  29. package/dist/templates/app/(public)/[...slug]/page.tsx +0 -378
  30. package/dist/templates/app/(public)/docs/[section]/[page]/page.tsx +0 -90
  31. package/dist/templates/app/(public)/docs/layout.tsx +0 -25
  32. package/dist/templates/app/(public)/docs/page.tsx +0 -81
  33. package/dist/templates/app/(public)/layout.tsx +0 -41
  34. package/dist/templates/app/(public)/page.tsx +0 -19
  35. package/dist/templates/app/403/page.tsx +0 -89
  36. package/dist/templates/app/api/auth/[...all]/route.ts +0 -78
  37. package/dist/templates/app/api/cron/billing/lifecycle/route.ts +0 -98
  38. package/dist/templates/app/api/csp-report/route.ts +0 -175
  39. package/dist/templates/app/api/devtools/config/entities/route.ts +0 -108
  40. package/dist/templates/app/api/devtools/config/theme/route.ts +0 -66
  41. package/dist/templates/app/api/devtools/tests/[...path]/route.ts +0 -130
  42. package/dist/templates/app/api/devtools/tests/route.ts +0 -134
  43. package/dist/templates/app/api/health/route.ts +0 -29
  44. package/dist/templates/app/api/internal/user-metadata/route.ts +0 -36
  45. package/dist/templates/app/api/superadmin/subscriptions/route.ts +0 -310
  46. package/dist/templates/app/api/superadmin/teams/[teamId]/route.ts +0 -286
  47. package/dist/templates/app/api/superadmin/teams/route.ts +0 -188
  48. package/dist/templates/app/api/superadmin/users/[userId]/route.ts +0 -540
  49. package/dist/templates/app/api/superadmin/users/route.ts +0 -323
  50. package/dist/templates/app/api/user/delete-account/route.ts +0 -55
  51. package/dist/templates/app/api/user/plan-flags/route.ts +0 -283
  52. package/dist/templates/app/api/user/profile/route.ts +0 -133
  53. package/dist/templates/app/api/v1/[entity]/[id]/child/[childType]/[childId]/route.ts +0 -210
  54. package/dist/templates/app/api/v1/[entity]/[id]/child/[childType]/route.ts +0 -331
  55. package/dist/templates/app/api/v1/[entity]/[id]/route.ts +0 -35
  56. package/dist/templates/app/api/v1/[entity]/docs.md +0 -369
  57. package/dist/templates/app/api/v1/[entity]/presets.ts +0 -194
  58. package/dist/templates/app/api/v1/[entity]/route.ts +0 -31
  59. package/dist/templates/app/api/v1/api-keys/[id]/route.ts +0 -303
  60. package/dist/templates/app/api/v1/api-keys/docs.md +0 -101
  61. package/dist/templates/app/api/v1/api-keys/presets.ts +0 -31
  62. package/dist/templates/app/api/v1/api-keys/route.ts +0 -250
  63. package/dist/templates/app/api/v1/auth/docs.md +0 -184
  64. package/dist/templates/app/api/v1/auth/presets.ts +0 -44
  65. package/dist/templates/app/api/v1/auth/signup-with-invite/route.ts +0 -227
  66. package/dist/templates/app/api/v1/billing/cancel/route.ts +0 -206
  67. package/dist/templates/app/api/v1/billing/change-plan/route.ts +0 -97
  68. package/dist/templates/app/api/v1/billing/check-action/route.ts +0 -81
  69. package/dist/templates/app/api/v1/billing/checkout/route.ts +0 -124
  70. package/dist/templates/app/api/v1/billing/docs.md +0 -209
  71. package/dist/templates/app/api/v1/billing/plans/route.ts +0 -85
  72. package/dist/templates/app/api/v1/billing/portal/route.ts +0 -90
  73. package/dist/templates/app/api/v1/billing/presets.ts +0 -121
  74. package/dist/templates/app/api/v1/billing/webhooks/stripe/route.ts +0 -428
  75. package/dist/templates/app/api/v1/blocks/[slug]/route.ts +0 -29
  76. package/dist/templates/app/api/v1/blocks/docs.md +0 -173
  77. package/dist/templates/app/api/v1/blocks/presets.ts +0 -121
  78. package/dist/templates/app/api/v1/blocks/route.ts +0 -45
  79. package/dist/templates/app/api/v1/blocks/validate/route.ts +0 -45
  80. package/dist/templates/app/api/v1/cron/docs.md +0 -116
  81. package/dist/templates/app/api/v1/cron/presets.ts +0 -26
  82. package/dist/templates/app/api/v1/cron/process/route.ts +0 -108
  83. package/dist/templates/app/api/v1/devtools/blocks/route.ts +0 -82
  84. package/dist/templates/app/api/v1/devtools/docs/route.ts +0 -150
  85. package/dist/templates/app/api/v1/devtools/docs.md +0 -204
  86. package/dist/templates/app/api/v1/devtools/features/route.ts +0 -61
  87. package/dist/templates/app/api/v1/devtools/flows/route.ts +0 -61
  88. package/dist/templates/app/api/v1/devtools/presets.ts +0 -113
  89. package/dist/templates/app/api/v1/devtools/scheduled-actions/route.ts +0 -120
  90. package/dist/templates/app/api/v1/devtools/testing/route.ts +0 -82
  91. package/dist/templates/app/api/v1/media/docs.md +0 -117
  92. package/dist/templates/app/api/v1/media/presets.ts +0 -24
  93. package/dist/templates/app/api/v1/media/upload/route.ts +0 -150
  94. package/dist/templates/app/api/v1/patterns/[id]/usages/route.ts +0 -116
  95. package/dist/templates/app/api/v1/plugin/[...path]/route.ts +0 -373
  96. package/dist/templates/app/api/v1/plugin/docs.md +0 -79
  97. package/dist/templates/app/api/v1/plugin/presets.ts +0 -21
  98. package/dist/templates/app/api/v1/plugin/route.ts +0 -96
  99. package/dist/templates/app/api/v1/post-categories/[id]/route.ts +0 -255
  100. package/dist/templates/app/api/v1/post-categories/docs.md +0 -134
  101. package/dist/templates/app/api/v1/post-categories/presets.ts +0 -78
  102. package/dist/templates/app/api/v1/post-categories/route.ts +0 -119
  103. package/dist/templates/app/api/v1/team-invitations/[token]/accept/route.ts +0 -179
  104. package/dist/templates/app/api/v1/team-invitations/[token]/decline/route.ts +0 -120
  105. package/dist/templates/app/api/v1/team-invitations/[token]/route.ts +0 -89
  106. package/dist/templates/app/api/v1/team-invitations/docs.md +0 -88
  107. package/dist/templates/app/api/v1/team-invitations/presets.ts +0 -43
  108. package/dist/templates/app/api/v1/team-invitations/route.ts +0 -114
  109. package/dist/templates/app/api/v1/teams/[teamId]/invitations/route.ts +0 -171
  110. package/dist/templates/app/api/v1/teams/[teamId]/invoices/[invoiceNumber]/route.ts +0 -105
  111. package/dist/templates/app/api/v1/teams/[teamId]/invoices/route.ts +0 -125
  112. package/dist/templates/app/api/v1/teams/[teamId]/members/[memberId]/route.ts +0 -263
  113. package/dist/templates/app/api/v1/teams/[teamId]/members/route.ts +0 -358
  114. package/dist/templates/app/api/v1/teams/[teamId]/route.ts +0 -322
  115. package/dist/templates/app/api/v1/teams/[teamId]/subscription/route.ts +0 -50
  116. package/dist/templates/app/api/v1/teams/[teamId]/usage/[limitSlug]/route.ts +0 -91
  117. package/dist/templates/app/api/v1/teams/docs.md +0 -320
  118. package/dist/templates/app/api/v1/teams/presets.ts +0 -178
  119. package/dist/templates/app/api/v1/teams/route.ts +0 -293
  120. package/dist/templates/app/api/v1/teams/switch/route.ts +0 -88
  121. package/dist/templates/app/api/v1/theme/[...path]/route.ts +0 -361
  122. package/dist/templates/app/api/v1/theme/docs.md +0 -74
  123. package/dist/templates/app/api/v1/theme/presets.ts +0 -21
  124. package/dist/templates/app/api/v1/theme/route.ts +0 -96
  125. package/dist/templates/app/api/v1/users/[id]/meta/[key]/route.ts +0 -363
  126. package/dist/templates/app/api/v1/users/[id]/route.ts +0 -302
  127. package/dist/templates/app/api/v1/users/docs.md +0 -93
  128. package/dist/templates/app/api/v1/users/presets.ts +0 -59
  129. package/dist/templates/app/api/v1/users/route.ts +0 -197
  130. package/dist/templates/app/dashboard/(main)/[entity]/[id]/edit/page.tsx +0 -117
  131. package/dist/templates/app/dashboard/(main)/[entity]/[id]/page.tsx +0 -103
  132. package/dist/templates/app/dashboard/(main)/[entity]/create/page.tsx +0 -95
  133. package/dist/templates/app/dashboard/(main)/[entity]/error.tsx +0 -51
  134. package/dist/templates/app/dashboard/(main)/[entity]/layout.tsx +0 -113
  135. package/dist/templates/app/dashboard/(main)/[entity]/loading.tsx +0 -61
  136. package/dist/templates/app/dashboard/(main)/[entity]/page.tsx +0 -90
  137. package/dist/templates/app/dashboard/(main)/layout.tsx +0 -98
  138. package/dist/templates/app/dashboard/(main)/loading.tsx +0 -5
  139. package/dist/templates/app/dashboard/(main)/page.tsx +0 -201
  140. package/dist/templates/app/dashboard/(main)/patterns/[id]/edit/page.tsx +0 -114
  141. package/dist/templates/app/dashboard/(main)/patterns/[id]/page.tsx +0 -20
  142. package/dist/templates/app/dashboard/(main)/patterns/[id]/reports/page.tsx +0 -171
  143. package/dist/templates/app/dashboard/(main)/patterns/create/page.tsx +0 -86
  144. package/dist/templates/app/dashboard/(main)/patterns/page.tsx +0 -444
  145. package/dist/templates/app/dashboard/features/analytics/page.tsx +0 -35
  146. package/dist/templates/app/dashboard/features/automation/page.tsx +0 -35
  147. package/dist/templates/app/dashboard/features/layout.tsx +0 -13
  148. package/dist/templates/app/dashboard/features/loading.tsx +0 -5
  149. package/dist/templates/app/dashboard/features/webhooks/page.tsx +0 -35
  150. package/dist/templates/app/dashboard/layout.tsx +0 -86
  151. package/dist/templates/app/dashboard/permission-denied/page.tsx +0 -29
  152. package/dist/templates/app/dashboard/settings/api-keys/loading.tsx +0 -5
  153. package/dist/templates/app/dashboard/settings/api-keys/page.tsx +0 -513
  154. package/dist/templates/app/dashboard/settings/billing/loading.tsx +0 -5
  155. package/dist/templates/app/dashboard/settings/billing/page.tsx +0 -284
  156. package/dist/templates/app/dashboard/settings/invoices/[invoiceNumber]/page.tsx +0 -222
  157. package/dist/templates/app/dashboard/settings/invoices/loading.tsx +0 -5
  158. package/dist/templates/app/dashboard/settings/invoices/page.tsx +0 -82
  159. package/dist/templates/app/dashboard/settings/layout.tsx +0 -151
  160. package/dist/templates/app/dashboard/settings/loading.tsx +0 -5
  161. package/dist/templates/app/dashboard/settings/notifications/loading.tsx +0 -5
  162. package/dist/templates/app/dashboard/settings/notifications/page.tsx +0 -462
  163. package/dist/templates/app/dashboard/settings/page.tsx +0 -92
  164. package/dist/templates/app/dashboard/settings/password/loading.tsx +0 -5
  165. package/dist/templates/app/dashboard/settings/password/page.tsx +0 -306
  166. package/dist/templates/app/dashboard/settings/plans/loading.tsx +0 -5
  167. package/dist/templates/app/dashboard/settings/plans/page.tsx +0 -40
  168. package/dist/templates/app/dashboard/settings/profile/loading.tsx +0 -5
  169. package/dist/templates/app/dashboard/settings/profile/page.tsx +0 -686
  170. package/dist/templates/app/dashboard/settings/security/loading.tsx +0 -5
  171. package/dist/templates/app/dashboard/settings/security/page.tsx +0 -505
  172. package/dist/templates/app/dashboard/settings/teams/loading.tsx +0 -5
  173. package/dist/templates/app/dashboard/settings/teams/page.tsx +0 -272
  174. package/dist/templates/app/dashboard/settings/teams/permissions/page.tsx +0 -92
  175. package/dist/templates/app/devtools/blocks/[slug]/page.tsx +0 -39
  176. package/dist/templates/app/devtools/blocks/page.tsx +0 -31
  177. package/dist/templates/app/devtools/config/page.tsx +0 -31
  178. package/dist/templates/app/devtools/features/page.tsx +0 -31
  179. package/dist/templates/app/devtools/flows/page.tsx +0 -31
  180. package/dist/templates/app/devtools/layout.tsx +0 -58
  181. package/dist/templates/app/devtools/page.tsx +0 -121
  182. package/dist/templates/app/devtools/scheduled-actions/page.tsx +0 -157
  183. package/dist/templates/app/devtools/style/page.tsx +0 -330
  184. package/dist/templates/app/devtools/tags/page.tsx +0 -31
  185. package/dist/templates/app/devtools/tests/[[...path]]/page.tsx +0 -47
  186. package/dist/templates/app/favicon.ico +0 -0
  187. package/dist/templates/app/globals.css +0 -12
  188. package/dist/templates/app/layout.tsx +0 -96
  189. package/dist/templates/app/public/page.tsx +0 -30
  190. package/dist/templates/app/superadmin/docs/[section]/[page]/page.tsx +0 -92
  191. package/dist/templates/app/superadmin/docs/page.tsx +0 -75
  192. package/dist/templates/app/superadmin/layout.tsx +0 -67
  193. package/dist/templates/app/superadmin/page.tsx +0 -149
  194. package/dist/templates/app/superadmin/subscriptions/page.tsx +0 -655
  195. package/dist/templates/app/superadmin/team-roles/page.tsx +0 -493
  196. package/dist/templates/app/superadmin/teams/[teamId]/page.tsx +0 -687
  197. package/dist/templates/app/superadmin/teams/page.tsx +0 -302
  198. package/dist/templates/app/superadmin/users/[userId]/page.tsx +0 -548
  199. package/dist/templates/app/superadmin/users/page.tsx +0 -528
  200. package/templates/app/(auth)/forgot-password/page.tsx +0 -216
  201. package/templates/app/(auth)/layout.tsx +0 -51
  202. package/templates/app/(auth)/login/page.tsx +0 -21
  203. package/templates/app/(auth)/reset-password/page.tsx +0 -212
  204. package/templates/app/(auth)/signup/page.tsx +0 -21
  205. package/templates/app/(auth)/verify-email/page.tsx +0 -190
  206. package/templates/app/(public)/[...slug]/page.tsx +0 -378
  207. package/templates/app/(public)/docs/[section]/[page]/page.tsx +0 -90
  208. package/templates/app/(public)/docs/layout.tsx +0 -25
  209. package/templates/app/(public)/docs/page.tsx +0 -81
  210. package/templates/app/(public)/layout.tsx +0 -41
  211. package/templates/app/(public)/page.tsx +0 -19
  212. package/templates/app/403/page.tsx +0 -89
  213. package/templates/app/api/auth/[...all]/route.ts +0 -78
  214. package/templates/app/api/cron/billing/lifecycle/route.ts +0 -98
  215. package/templates/app/api/csp-report/route.ts +0 -175
  216. package/templates/app/api/devtools/config/entities/route.ts +0 -108
  217. package/templates/app/api/devtools/config/theme/route.ts +0 -66
  218. package/templates/app/api/devtools/tests/[...path]/route.ts +0 -130
  219. package/templates/app/api/devtools/tests/route.ts +0 -134
  220. package/templates/app/api/health/route.ts +0 -29
  221. package/templates/app/api/internal/user-metadata/route.ts +0 -36
  222. package/templates/app/api/superadmin/subscriptions/route.ts +0 -310
  223. package/templates/app/api/superadmin/teams/[teamId]/route.ts +0 -286
  224. package/templates/app/api/superadmin/teams/route.ts +0 -188
  225. package/templates/app/api/superadmin/users/[userId]/route.ts +0 -540
  226. package/templates/app/api/superadmin/users/route.ts +0 -323
  227. package/templates/app/api/user/delete-account/route.ts +0 -55
  228. package/templates/app/api/user/plan-flags/route.ts +0 -283
  229. package/templates/app/api/user/profile/route.ts +0 -133
  230. package/templates/app/api/v1/[entity]/[id]/child/[childType]/[childId]/route.ts +0 -210
  231. package/templates/app/api/v1/[entity]/[id]/child/[childType]/route.ts +0 -331
  232. package/templates/app/api/v1/[entity]/[id]/route.ts +0 -35
  233. package/templates/app/api/v1/[entity]/docs.md +0 -369
  234. package/templates/app/api/v1/[entity]/presets.ts +0 -194
  235. package/templates/app/api/v1/[entity]/route.ts +0 -31
  236. package/templates/app/api/v1/api-keys/[id]/route.ts +0 -303
  237. package/templates/app/api/v1/api-keys/docs.md +0 -101
  238. package/templates/app/api/v1/api-keys/presets.ts +0 -31
  239. package/templates/app/api/v1/api-keys/route.ts +0 -250
  240. package/templates/app/api/v1/auth/docs.md +0 -184
  241. package/templates/app/api/v1/auth/presets.ts +0 -44
  242. package/templates/app/api/v1/auth/signup-with-invite/route.ts +0 -227
  243. package/templates/app/api/v1/billing/cancel/route.ts +0 -206
  244. package/templates/app/api/v1/billing/change-plan/route.ts +0 -97
  245. package/templates/app/api/v1/billing/check-action/route.ts +0 -81
  246. package/templates/app/api/v1/billing/checkout/route.ts +0 -124
  247. package/templates/app/api/v1/billing/docs.md +0 -209
  248. package/templates/app/api/v1/billing/plans/route.ts +0 -85
  249. package/templates/app/api/v1/billing/portal/route.ts +0 -90
  250. package/templates/app/api/v1/billing/presets.ts +0 -121
  251. package/templates/app/api/v1/billing/webhooks/stripe/route.ts +0 -428
  252. package/templates/app/api/v1/blocks/[slug]/route.ts +0 -29
  253. package/templates/app/api/v1/blocks/docs.md +0 -173
  254. package/templates/app/api/v1/blocks/presets.ts +0 -121
  255. package/templates/app/api/v1/blocks/route.ts +0 -45
  256. package/templates/app/api/v1/blocks/validate/route.ts +0 -45
  257. package/templates/app/api/v1/cron/docs.md +0 -116
  258. package/templates/app/api/v1/cron/presets.ts +0 -26
  259. package/templates/app/api/v1/cron/process/route.ts +0 -108
  260. package/templates/app/api/v1/devtools/blocks/route.ts +0 -82
  261. package/templates/app/api/v1/devtools/docs/route.ts +0 -150
  262. package/templates/app/api/v1/devtools/docs.md +0 -204
  263. package/templates/app/api/v1/devtools/features/route.ts +0 -61
  264. package/templates/app/api/v1/devtools/flows/route.ts +0 -61
  265. package/templates/app/api/v1/devtools/presets.ts +0 -113
  266. package/templates/app/api/v1/devtools/scheduled-actions/route.ts +0 -120
  267. package/templates/app/api/v1/devtools/testing/route.ts +0 -82
  268. package/templates/app/api/v1/media/docs.md +0 -117
  269. package/templates/app/api/v1/media/presets.ts +0 -24
  270. package/templates/app/api/v1/media/upload/route.ts +0 -150
  271. package/templates/app/api/v1/patterns/[id]/usages/route.ts +0 -116
  272. package/templates/app/api/v1/plugin/[...path]/route.ts +0 -373
  273. package/templates/app/api/v1/plugin/docs.md +0 -79
  274. package/templates/app/api/v1/plugin/presets.ts +0 -21
  275. package/templates/app/api/v1/plugin/route.ts +0 -96
  276. package/templates/app/api/v1/post-categories/[id]/route.ts +0 -255
  277. package/templates/app/api/v1/post-categories/docs.md +0 -134
  278. package/templates/app/api/v1/post-categories/presets.ts +0 -78
  279. package/templates/app/api/v1/post-categories/route.ts +0 -119
  280. package/templates/app/api/v1/team-invitations/[token]/accept/route.ts +0 -179
  281. package/templates/app/api/v1/team-invitations/[token]/decline/route.ts +0 -120
  282. package/templates/app/api/v1/team-invitations/[token]/route.ts +0 -89
  283. package/templates/app/api/v1/team-invitations/docs.md +0 -88
  284. package/templates/app/api/v1/team-invitations/presets.ts +0 -43
  285. package/templates/app/api/v1/team-invitations/route.ts +0 -114
  286. package/templates/app/api/v1/teams/[teamId]/invitations/route.ts +0 -171
  287. package/templates/app/api/v1/teams/[teamId]/invoices/[invoiceNumber]/route.ts +0 -105
  288. package/templates/app/api/v1/teams/[teamId]/invoices/route.ts +0 -125
  289. package/templates/app/api/v1/teams/[teamId]/members/[memberId]/route.ts +0 -263
  290. package/templates/app/api/v1/teams/[teamId]/members/route.ts +0 -358
  291. package/templates/app/api/v1/teams/[teamId]/route.ts +0 -322
  292. package/templates/app/api/v1/teams/[teamId]/subscription/route.ts +0 -50
  293. package/templates/app/api/v1/teams/[teamId]/usage/[limitSlug]/route.ts +0 -91
  294. package/templates/app/api/v1/teams/docs.md +0 -320
  295. package/templates/app/api/v1/teams/presets.ts +0 -178
  296. package/templates/app/api/v1/teams/route.ts +0 -293
  297. package/templates/app/api/v1/teams/switch/route.ts +0 -88
  298. package/templates/app/api/v1/theme/[...path]/route.ts +0 -361
  299. package/templates/app/api/v1/theme/docs.md +0 -74
  300. package/templates/app/api/v1/theme/presets.ts +0 -21
  301. package/templates/app/api/v1/theme/route.ts +0 -96
  302. package/templates/app/api/v1/users/[id]/meta/[key]/route.ts +0 -363
  303. package/templates/app/api/v1/users/[id]/route.ts +0 -302
  304. package/templates/app/api/v1/users/docs.md +0 -93
  305. package/templates/app/api/v1/users/presets.ts +0 -59
  306. package/templates/app/api/v1/users/route.ts +0 -197
  307. package/templates/app/dashboard/(main)/[entity]/[id]/edit/page.tsx +0 -117
  308. package/templates/app/dashboard/(main)/[entity]/[id]/page.tsx +0 -103
  309. package/templates/app/dashboard/(main)/[entity]/create/page.tsx +0 -95
  310. package/templates/app/dashboard/(main)/[entity]/error.tsx +0 -51
  311. package/templates/app/dashboard/(main)/[entity]/layout.tsx +0 -113
  312. package/templates/app/dashboard/(main)/[entity]/loading.tsx +0 -61
  313. package/templates/app/dashboard/(main)/[entity]/page.tsx +0 -90
  314. package/templates/app/dashboard/(main)/layout.tsx +0 -98
  315. package/templates/app/dashboard/(main)/loading.tsx +0 -5
  316. package/templates/app/dashboard/(main)/page.tsx +0 -201
  317. package/templates/app/dashboard/(main)/patterns/[id]/edit/page.tsx +0 -114
  318. package/templates/app/dashboard/(main)/patterns/[id]/page.tsx +0 -20
  319. package/templates/app/dashboard/(main)/patterns/[id]/reports/page.tsx +0 -171
  320. package/templates/app/dashboard/(main)/patterns/create/page.tsx +0 -86
  321. package/templates/app/dashboard/(main)/patterns/page.tsx +0 -444
  322. package/templates/app/dashboard/features/analytics/page.tsx +0 -35
  323. package/templates/app/dashboard/features/automation/page.tsx +0 -35
  324. package/templates/app/dashboard/features/layout.tsx +0 -13
  325. package/templates/app/dashboard/features/loading.tsx +0 -5
  326. package/templates/app/dashboard/features/webhooks/page.tsx +0 -35
  327. package/templates/app/dashboard/layout.tsx +0 -86
  328. package/templates/app/dashboard/permission-denied/page.tsx +0 -29
  329. package/templates/app/dashboard/settings/api-keys/loading.tsx +0 -5
  330. package/templates/app/dashboard/settings/api-keys/page.tsx +0 -513
  331. package/templates/app/dashboard/settings/billing/loading.tsx +0 -5
  332. package/templates/app/dashboard/settings/billing/page.tsx +0 -284
  333. package/templates/app/dashboard/settings/invoices/[invoiceNumber]/page.tsx +0 -222
  334. package/templates/app/dashboard/settings/invoices/loading.tsx +0 -5
  335. package/templates/app/dashboard/settings/invoices/page.tsx +0 -82
  336. package/templates/app/dashboard/settings/layout.tsx +0 -151
  337. package/templates/app/dashboard/settings/loading.tsx +0 -5
  338. package/templates/app/dashboard/settings/notifications/loading.tsx +0 -5
  339. package/templates/app/dashboard/settings/notifications/page.tsx +0 -462
  340. package/templates/app/dashboard/settings/page.tsx +0 -92
  341. package/templates/app/dashboard/settings/password/loading.tsx +0 -5
  342. package/templates/app/dashboard/settings/password/page.tsx +0 -306
  343. package/templates/app/dashboard/settings/plans/loading.tsx +0 -5
  344. package/templates/app/dashboard/settings/plans/page.tsx +0 -40
  345. package/templates/app/dashboard/settings/profile/loading.tsx +0 -5
  346. package/templates/app/dashboard/settings/profile/page.tsx +0 -686
  347. package/templates/app/dashboard/settings/security/loading.tsx +0 -5
  348. package/templates/app/dashboard/settings/security/page.tsx +0 -505
  349. package/templates/app/dashboard/settings/teams/loading.tsx +0 -5
  350. package/templates/app/dashboard/settings/teams/page.tsx +0 -272
  351. package/templates/app/dashboard/settings/teams/permissions/page.tsx +0 -92
  352. package/templates/app/devtools/blocks/[slug]/page.tsx +0 -39
  353. package/templates/app/devtools/blocks/page.tsx +0 -31
  354. package/templates/app/devtools/config/page.tsx +0 -31
  355. package/templates/app/devtools/features/page.tsx +0 -31
  356. package/templates/app/devtools/flows/page.tsx +0 -31
  357. package/templates/app/devtools/layout.tsx +0 -58
  358. package/templates/app/devtools/page.tsx +0 -121
  359. package/templates/app/devtools/scheduled-actions/page.tsx +0 -157
  360. package/templates/app/devtools/style/page.tsx +0 -330
  361. package/templates/app/devtools/tags/page.tsx +0 -31
  362. package/templates/app/devtools/tests/[[...path]]/page.tsx +0 -47
  363. package/templates/app/favicon.ico +0 -0
  364. package/templates/app/globals.css +0 -12
  365. package/templates/app/layout.tsx +0 -96
  366. package/templates/app/public/page.tsx +0 -30
  367. package/templates/app/superadmin/docs/[section]/[page]/page.tsx +0 -92
  368. package/templates/app/superadmin/docs/page.tsx +0 -75
  369. package/templates/app/superadmin/layout.tsx +0 -67
  370. package/templates/app/superadmin/page.tsx +0 -149
  371. package/templates/app/superadmin/subscriptions/page.tsx +0 -655
  372. package/templates/app/superadmin/team-roles/page.tsx +0 -493
  373. package/templates/app/superadmin/teams/[teamId]/page.tsx +0 -687
  374. package/templates/app/superadmin/teams/page.tsx +0 -302
  375. package/templates/app/superadmin/users/[userId]/page.tsx +0 -548
  376. 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
- });