@nextsparkjs/core 0.1.0-beta.84 → 0.1.0-beta.85

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