@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,81 +0,0 @@
1
- import { Card, CardContent, CardDescription, CardHeader, CardTitle } from '@nextsparkjs/core/components/ui/card'
2
- import { DOCS_REGISTRY } from '@nextsparkjs/registries/docs-registry'
3
- import { sel } from '@nextsparkjs/core/selectors'
4
- import { BookOpen, Folder, FileText, ArrowRight } from 'lucide-react'
5
- import Link from 'next/link'
6
- import type { Metadata } from 'next'
7
-
8
- export const metadata: Metadata = {
9
- title: 'Documentation',
10
- description: 'Complete documentation for the application'
11
- }
12
-
13
- export default function DocsPage() {
14
- const sections = DOCS_REGISTRY.public
15
-
16
- return (
17
- <div className="container max-w-7xl mx-auto px-4 py-8">
18
- {/* Hero Section */}
19
- <div className="text-center mb-12">
20
- <div className="inline-flex items-center gap-2 bg-primary/10 text-primary px-3 py-1 rounded-full text-sm font-medium mb-4">
21
- <BookOpen className="h-4 w-4" />
22
- Documentation
23
- </div>
24
- <h1 className="text-4xl font-bold mb-4">
25
- Documentation
26
- </h1>
27
- <p className="text-xl text-muted-foreground max-w-2xl mx-auto">
28
- Guides and reference documentation to help you get the most out of the application.
29
- </p>
30
- </div>
31
-
32
- {/* Documentation Sections */}
33
- {sections.length > 0 ? (
34
- <div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-6">
35
- {sections.map((section) => (
36
- <Card key={section.slug} className="h-full hover:shadow-lg transition-shadow" data-cy={sel('public.docs.sectionCard', { slug: section.slug })}>
37
- <CardHeader>
38
- <div className="flex items-center gap-3 mb-3">
39
- <div className="p-2 bg-primary/10 rounded-lg text-primary">
40
- <Folder className="h-6 w-6" />
41
- </div>
42
- <div>
43
- <CardTitle className="text-lg">{section.title}</CardTitle>
44
- </div>
45
- </div>
46
- <CardDescription>
47
- {section.pages.length} {section.pages.length === 1 ? 'page' : 'pages'}
48
- </CardDescription>
49
- </CardHeader>
50
- <CardContent>
51
- <ul className="space-y-2">
52
- {section.pages.map((page) => (
53
- <li key={page.slug}>
54
- <Link
55
- href={`/docs/${section.slug}/${page.slug}`}
56
- className="text-sm text-muted-foreground hover:text-foreground transition-colors flex items-center gap-2 group"
57
- data-cy={sel('public.docs.pageLink', { slug: page.slug })}
58
- >
59
- <FileText className="h-3 w-3" />
60
- <span className="flex-1">{page.title}</span>
61
- <ArrowRight className="h-3 w-3 opacity-0 group-hover:opacity-100 group-hover:translate-x-0.5 transition-all" />
62
- </Link>
63
- </li>
64
- ))}
65
- </ul>
66
- </CardContent>
67
- </Card>
68
- ))}
69
- </div>
70
- ) : (
71
- <div className="text-center py-12">
72
- <BookOpen className="h-12 w-12 text-muted-foreground mx-auto mb-4" />
73
- <h2 className="text-xl font-semibold mb-2">No Documentation Available</h2>
74
- <p className="text-muted-foreground">
75
- Documentation will appear here once it is added to the project.
76
- </p>
77
- </div>
78
- )}
79
- </div>
80
- )
81
- }
@@ -1,41 +0,0 @@
1
- import type { Metadata } from "next"
2
- import { PublicNavbar } from '@nextsparkjs/core/components/app/layouts/PublicNavbar'
3
- import { PublicFooter } from '@nextsparkjs/core/components/app/layouts/PublicFooter'
4
- import { getTemplateOrDefault, getMetadataOrDefault } from "@nextsparkjs/core/lib/template-resolver"
5
-
6
- // ✅ MINIMAL GENERIC METADATA (cliente puede override con template)
7
- const defaultMetadata: Metadata = {
8
- title: {
9
- template: '%s | App',
10
- default: 'App',
11
- },
12
- description: 'Application',
13
- }
14
-
15
- export const metadata: Metadata = getMetadataOrDefault(
16
- 'app/(public)/layout.tsx',
17
- defaultMetadata
18
- )
19
-
20
- function PublicLayout({
21
- children
22
- }: {
23
- children: React.ReactNode
24
- }) {
25
- return (
26
- <div className="min-h-screen bg-background flex flex-col">
27
- {/* Public Navbar */}
28
- <PublicNavbar />
29
-
30
- {/* Main Content */}
31
- <main className="flex-1">
32
- {children}
33
- </main>
34
-
35
- {/* Public Footer */}
36
- <PublicFooter />
37
- </div>
38
- )
39
- }
40
-
41
- export default getTemplateOrDefault('app/(public)/layout.tsx', PublicLayout)
@@ -1,19 +0,0 @@
1
- import { redirect } from 'next/navigation'
2
- import { getTemplateOrDefault } from "@nextsparkjs/core/lib/template-resolver"
3
-
4
- /**
5
- * Default Public Home Page (CORE)
6
- *
7
- * This is the minimal CORE version that redirects to dashboard if no theme template override exists.
8
- * Theme templates can override this to provide custom landing pages.
9
- *
10
- * Location: app/(public)/page.tsx (CORE)
11
- * Override: contents/themes/[theme]/templates/(public)/page.tsx (THEME)
12
- */
13
- function DefaultPublicHome() {
14
- // Si no hay template override, redirect to dashboard
15
- // (Theme provides the actual landing page)
16
- redirect('/dashboard')
17
- }
18
-
19
- export default getTemplateOrDefault('app/(public)/page.tsx', DefaultPublicHome)
@@ -1,89 +0,0 @@
1
- 'use client'
2
-
3
- import { AlertTriangle, ArrowLeft, Home } from 'lucide-react'
4
- import Link from 'next/link'
5
- import { useSearchParams } from 'next/navigation'
6
- import { Suspense } from 'react'
7
- import { Button } from '@nextsparkjs/core/components/ui/button'
8
- import { Card, CardContent, CardDescription, CardHeader, CardTitle } from '@nextsparkjs/core/components/ui/card'
9
- import { getTemplateOrDefaultClient } from '@nextsparkjs/registries/template-registry.client'
10
-
11
- function ForbiddenContent() {
12
- const searchParams = useSearchParams()
13
- const reason = searchParams.get('reason') || 'No tienes permisos para acceder a este recurso'
14
- const upgrade = searchParams.get('upgrade') === 'true'
15
-
16
- return (
17
- <div className="min-h-screen flex items-center justify-center bg-gray-50 px-4">
18
- <Card className="w-full max-w-md">
19
- <CardHeader className="text-center">
20
- <div className="mx-auto mb-4 p-4 rounded-full bg-yellow-100">
21
- <AlertTriangle className="h-8 w-8 text-yellow-600" />
22
- </div>
23
- <CardTitle className="text-2xl">
24
- ⚠️ Ups, no tienes permisos para hacer esto
25
- </CardTitle>
26
- <CardDescription className="text-gray-600">
27
- {reason}
28
- </CardDescription>
29
- </CardHeader>
30
-
31
- <CardContent className="space-y-4">
32
- {upgrade && (
33
- <div className="bg-blue-50 border border-blue-200 rounded-lg p-4">
34
- <p className="text-sm text-blue-800">
35
- Esta funcionalidad requiere una mejora de tu plan actual.
36
- </p>
37
- <Link href="/dashboard/settings/billing" className="block mt-2">
38
- <Button variant="outline" size="sm" className="w-full">
39
- Ver planes
40
- </Button>
41
- </Link>
42
- </div>
43
- )}
44
-
45
- <div className="flex gap-2">
46
- <Button
47
- variant="outline"
48
- className="flex-1"
49
- onClick={() => window.history.back()}
50
- >
51
- <ArrowLeft className="h-4 w-4 mr-2" />
52
- Volver
53
- </Button>
54
-
55
- <Link href="/dashboard" className="flex-1">
56
- <Button className="w-full">
57
- <Home className="h-4 w-4 mr-2" />
58
- Dashboard
59
- </Button>
60
- </Link>
61
- </div>
62
- </CardContent>
63
- </Card>
64
- </div>
65
- )
66
- }
67
-
68
- function ForbiddenPage() {
69
- return (
70
- <Suspense fallback={
71
- <div className="min-h-screen flex items-center justify-center bg-gray-50 px-4">
72
- <Card className="w-full max-w-md">
73
- <CardHeader className="text-center">
74
- <div className="mx-auto mb-4 p-4 rounded-full bg-yellow-100">
75
- <AlertTriangle className="h-8 w-8 text-yellow-600" />
76
- </div>
77
- <CardTitle className="text-2xl">
78
- ⚠️ Ups, no tienes permisos para hacer esto
79
- </CardTitle>
80
- </CardHeader>
81
- </Card>
82
- </div>
83
- }>
84
- <ForbiddenContent />
85
- </Suspense>
86
- )
87
- }
88
-
89
- export default getTemplateOrDefaultClient('app/403/page.tsx', ForbiddenPage)
@@ -1,78 +0,0 @@
1
- import { auth } from "@nextsparkjs/core/lib/auth";
2
- import { toNextJsHandler } from "better-auth/next-js";
3
- import { NextRequest, NextResponse } from "next/server";
4
- import { TEAMS_CONFIG } from "@nextsparkjs/core/lib/config";
5
- import { isPublicSignupRestricted } from "@nextsparkjs/core/lib/teams/helpers";
6
- import { TeamService } from "@nextsparkjs/core/lib/services";
7
- import { wrapAuthHandlerWithCors, handleCorsPreflightRequest, addCorsHeaders } from "@nextsparkjs/core/lib/api/helpers";
8
-
9
- const handlers = toNextJsHandler(auth);
10
-
11
- // Handle CORS preflight requests for cross-origin auth (mobile apps, etc.)
12
- export async function OPTIONS(req: NextRequest) {
13
- return handleCorsPreflightRequest(req);
14
- }
15
-
16
- // Intercept email verification requests to redirect to UI page
17
- // eslint-disable-next-line @typescript-eslint/no-unused-vars
18
- export async function GET(req: NextRequest, context: { params: Promise<{ all: string[] }> }) {
19
- const pathname = req.nextUrl.pathname;
20
-
21
- // Check if this is an email verification request from an email link
22
- // We check for a special header to determine if it's from our UI or from an email click
23
- const isFromUI = req.headers.get('x-verify-from-ui') === 'true';
24
-
25
- if (pathname === '/api/auth/verify-email' && !isFromUI) {
26
- const token = req.nextUrl.searchParams.get('token');
27
- const callbackURL = req.nextUrl.searchParams.get('callbackURL');
28
-
29
- if (token) {
30
- // This is from an email link, redirect to the UI verification page
31
- const redirectUrl = new URL('/verify-email', req.url);
32
- redirectUrl.searchParams.set('token', token);
33
- if (callbackURL) {
34
- redirectUrl.searchParams.set('callbackURL', callbackURL);
35
- }
36
- return NextResponse.redirect(redirectUrl);
37
- }
38
- }
39
-
40
- // Wrap with CORS headers for cross-origin requests (mobile apps, etc.)
41
- return wrapAuthHandlerWithCors(() => handlers.GET(req), req);
42
- }
43
-
44
- // Intercept signup requests to validate single-tenant mode
45
- export async function POST(req: NextRequest) {
46
- const pathname = req.nextUrl.pathname;
47
-
48
- // Check if this is a signup request (email or OAuth)
49
- const isSignupRequest =
50
- pathname === '/api/auth/sign-up/email' ||
51
- pathname.includes('/api/auth/callback/'); // OAuth callbacks can create users
52
-
53
- if (isSignupRequest) {
54
- const teamsMode = TEAMS_CONFIG.mode;
55
-
56
- // In single-tenant mode, block public signup if team already exists
57
- if (isPublicSignupRestricted(teamsMode)) {
58
- const teamExists = await TeamService.hasGlobal();
59
-
60
- if (teamExists) {
61
- // Block public signup - users must be invited
62
- // Add CORS headers so mobile apps can read the error message
63
- const errorResponse = NextResponse.json(
64
- {
65
- error: 'Registration is closed',
66
- message: 'This application requires an invitation to register. Please contact an administrator.',
67
- code: 'SIGNUP_RESTRICTED',
68
- },
69
- { status: 403 }
70
- );
71
- return await addCorsHeaders(errorResponse, req);
72
- }
73
- }
74
- }
75
-
76
- // Wrap with CORS headers for cross-origin requests (mobile apps, etc.)
77
- return wrapAuthHandlerWithCors(() => handlers.POST(req), req);
78
- }
@@ -1,98 +0,0 @@
1
- /**
2
- * Billing Lifecycle Cron Job
3
- *
4
- * Scheduled job that runs automatically to manage subscription lifecycle.
5
- * Protected with CRON_SECRET to prevent unauthorized execution.
6
- *
7
- * P1: Lifecycle Management
8
- *
9
- * Configure in vercel.json:
10
- * {
11
- * "crons": [
12
- * {
13
- * "path": "/api/cron/billing/lifecycle",
14
- * "schedule": "0 0 * * *"
15
- * }
16
- * ]
17
- * }
18
- */
19
-
20
- import { NextRequest } from 'next/server'
21
- import { handlePastDueGracePeriod } from '@nextsparkjs/core/lib/billing/jobs'
22
- import { SubscriptionService, UsageService } from '@nextsparkjs/core/lib/services'
23
-
24
- export async function GET(request: NextRequest) {
25
- // Verify CRON_SECRET (MANDATORY for security)
26
- const authHeader = request.headers.get('authorization')
27
- const cronSecret = process.env.CRON_SECRET
28
-
29
- if (!cronSecret || authHeader !== `Bearer ${cronSecret}`) {
30
- console.error('[lifecycle-cron] Unauthorized attempt to access cron endpoint')
31
- return Response.json({ error: 'Unauthorized' }, { status: 401 })
32
- }
33
-
34
- const results = {
35
- expireTrials: { processed: 0, errors: [] as string[] },
36
- pastDueGrace: { processed: 0, errors: [] as string[] },
37
- resetUsage: { processed: 0, errors: [] as string[] }
38
- }
39
-
40
- try {
41
- console.log('[lifecycle-cron] Starting billing lifecycle jobs...')
42
-
43
- // 1. Expire trials (service returns count)
44
- try {
45
- const count = await SubscriptionService.processExpiredTrials()
46
- results.expireTrials = { processed: count, errors: [] }
47
- } catch (error) {
48
- results.expireTrials.errors.push(error instanceof Error ? error.message : 'Unknown error')
49
- }
50
-
51
- // 2. Handle past_due grace period (jobs.ts function returns {processed, errors})
52
- results.pastDueGrace = await handlePastDueGracePeriod()
53
-
54
- // 3. Reset monthly usage (only on first day of month)
55
- const today = new Date()
56
- if (today.getDate() === 1) {
57
- try {
58
- const count = await UsageService.processMonthlyReset()
59
- results.resetUsage = { processed: count, errors: [] }
60
- } catch (error) {
61
- results.resetUsage.errors.push(error instanceof Error ? error.message : 'Unknown error')
62
- }
63
- }
64
-
65
- const totalProcessed =
66
- results.expireTrials.processed + results.pastDueGrace.processed + results.resetUsage.processed
67
-
68
- const allErrors = [
69
- ...results.expireTrials.errors,
70
- ...results.pastDueGrace.errors,
71
- ...results.resetUsage.errors
72
- ]
73
-
74
- console.log(`[lifecycle-cron] Completed: ${totalProcessed} items processed, ${allErrors.length} errors`)
75
-
76
- return Response.json({
77
- success: true,
78
- processed: totalProcessed,
79
- errors: allErrors,
80
- details: results,
81
- timestamp: new Date().toISOString()
82
- })
83
- } catch (error) {
84
- console.error('[lifecycle-cron] Fatal error:', error)
85
- return Response.json(
86
- {
87
- success: false,
88
- error: error instanceof Error ? error.message : 'Unknown error'
89
- },
90
- { status: 500 }
91
- )
92
- }
93
- }
94
-
95
- // Also support POST for manual triggering via dashboard
96
- export async function POST(request: NextRequest) {
97
- return GET(request)
98
- }
@@ -1,175 +0,0 @@
1
- import { randomUUID } from 'node:crypto';
2
- import { NextRequest, NextResponse } from 'next/server';
3
-
4
- // Dynamic import for rate limiting - graceful fallback if not available
5
- let checkDistributedRateLimit: ((id: string, tier: string) => Promise<{ allowed: boolean; limit: number; remaining: number; resetTime: number; retryAfter?: number }>) | null = null;
6
- let createRateLimitErrorResponse: ((result: { allowed: boolean; limit: number; remaining: number; resetTime: number; retryAfter?: number }) => NextResponse) | null = null;
7
-
8
- // Try to load rate limiting functions
9
- try {
10
- const rateLimitModule = require('@nextsparkjs/core/lib/api');
11
- checkDistributedRateLimit = rateLimitModule.checkDistributedRateLimit;
12
- createRateLimitErrorResponse = rateLimitModule.createRateLimitErrorResponse;
13
- } catch {
14
- // Rate limiting not available - will skip rate limit checks
15
- console.warn('[CSP Report] Rate limiting not available - running without rate limits');
16
- }
17
-
18
- /**
19
- * CSP Violation Report Endpoint
20
- *
21
- * Receives Content-Security-Policy violation reports from browsers.
22
- * Reports are logged for monitoring and debugging CSP issues.
23
- *
24
- * Rate limiting: Uses 'api' tier (100 requests/minute per IP) to prevent abuse.
25
- * Falls back gracefully if rate limiting is not available.
26
- *
27
- * NOTE: This file exists in both apps/dev/app/api/csp-report/ and
28
- * packages/core/templates/app/api/csp-report/. The template version
29
- * is used when creating new projects from the core package.
30
- * Changes should be synchronized between both files.
31
- *
32
- * @see https://developer.mozilla.org/en-US/docs/Web/HTTP/CSP#violation_report_syntax
33
- */
34
-
35
- interface CSPViolationReport {
36
- 'csp-report'?: {
37
- 'document-uri'?: string;
38
- 'referrer'?: string;
39
- 'violated-directive'?: string;
40
- 'effective-directive'?: string;
41
- 'original-policy'?: string;
42
- 'disposition'?: string;
43
- 'blocked-uri'?: string;
44
- 'line-number'?: number;
45
- 'column-number'?: number;
46
- 'source-file'?: string;
47
- 'status-code'?: number;
48
- 'script-sample'?: string;
49
- };
50
- }
51
-
52
- // Get allowed origin from environment or use localhost fallback
53
- const getAllowedOrigin = () => {
54
- return process.env.NEXT_PUBLIC_APP_URL || 'http://localhost:3000';
55
- };
56
-
57
- /**
58
- * Get client IP address from request headers.
59
- * Handles various proxy scenarios (X-Forwarded-For, X-Real-IP, etc.)
60
- */
61
- function getClientIp(request: NextRequest): string {
62
- const forwardedFor = request.headers.get('x-forwarded-for');
63
- if (forwardedFor) {
64
- const ips = forwardedFor.split(',').map(ip => ip.trim());
65
- if (ips[0]) return ips[0];
66
- }
67
- const realIp = request.headers.get('x-real-ip');
68
- if (realIp) return realIp;
69
- const cfIp = request.headers.get('cf-connecting-ip');
70
- if (cfIp) return cfIp;
71
- return 'unknown';
72
- }
73
-
74
- export async function POST(request: NextRequest) {
75
- const requestId = randomUUID().slice(0, 8);
76
- let rateLimitHeaders: Record<string, string> = {};
77
-
78
- // Rate limiting: 100 requests per minute per IP (api tier)
79
- // Skip if rate limiting is not available
80
- if (checkDistributedRateLimit && createRateLimitErrorResponse) {
81
- try {
82
- const clientIp = getClientIp(request);
83
- const rateLimitResult = await checkDistributedRateLimit(`csp-report:ip:${clientIp}`, 'api');
84
-
85
- if (!rateLimitResult.allowed) {
86
- console.warn(`[CSP Report ${requestId}] Rate limit exceeded for IP: ${clientIp}`);
87
- return createRateLimitErrorResponse(rateLimitResult);
88
- }
89
-
90
- // Store rate limit headers to include in response
91
- rateLimitHeaders = {
92
- 'X-RateLimit-Limit': rateLimitResult.limit.toString(),
93
- 'X-RateLimit-Remaining': rateLimitResult.remaining.toString(),
94
- 'X-RateLimit-Reset': rateLimitResult.resetTime.toString(),
95
- };
96
- } catch (rateLimitError) {
97
- // Log but continue without rate limiting
98
- console.warn(`[CSP Report ${requestId}] Rate limit check failed, continuing without:`, {
99
- error: rateLimitError instanceof Error ? rateLimitError.message : 'Unknown error',
100
- });
101
- }
102
- }
103
-
104
- try {
105
- const contentType = request.headers.get('content-type') || '';
106
-
107
- // CSP reports are sent as application/csp-report or application/json
108
- if (!contentType.includes('application/csp-report') && !contentType.includes('application/json')) {
109
- console.warn(`[CSP Report ${requestId}] Invalid content-type: ${contentType}`);
110
- return new Response('Invalid content type', { status: 400 });
111
- }
112
-
113
- let rawBody: string | undefined;
114
- try {
115
- rawBody = await request.text();
116
- const report: CSPViolationReport = JSON.parse(rawBody);
117
- const violation = report['csp-report'];
118
-
119
- if (!violation) {
120
- console.warn(`[CSP Report ${requestId}] Invalid report format - missing csp-report key`, {
121
- bodyPreview: rawBody.slice(0, 200),
122
- });
123
- return new Response('Invalid report format', { status: 400 });
124
- }
125
-
126
- // Log the violation for monitoring
127
- // In production, you might want to send this to a logging service like Datadog, Sentry, etc.
128
- console.warn(`[CSP Violation ${requestId}]`, {
129
- documentUri: violation['document-uri'],
130
- blockedUri: violation['blocked-uri'],
131
- violatedDirective: violation['violated-directive'],
132
- effectiveDirective: violation['effective-directive'],
133
- sourceFile: violation['source-file'],
134
- lineNumber: violation['line-number'],
135
- columnNumber: violation['column-number'],
136
- scriptSample: violation['script-sample'],
137
- disposition: violation['disposition'],
138
- timestamp: new Date().toISOString(),
139
- });
140
-
141
- // Return 204 No Content - browsers don't expect a response body
142
- return new Response(null, {
143
- status: 204,
144
- headers: rateLimitHeaders,
145
- });
146
- } catch (parseError) {
147
- console.error(`[CSP Report ${requestId}] JSON parse error:`, {
148
- error: parseError instanceof Error ? parseError.message : 'Unknown error',
149
- bodyPreview: rawBody?.slice(0, 200),
150
- });
151
- // Still return 204 to not break browser behavior
152
- return new Response(null, { status: 204 });
153
- }
154
- } catch (error) {
155
- console.error(`[CSP Report ${requestId}] Unexpected error:`, {
156
- error: error instanceof Error ? error.message : 'Unknown error',
157
- stack: error instanceof Error ? error.stack : undefined,
158
- });
159
- // Still return 204 to not break browser behavior
160
- return new Response(null, { status: 204 });
161
- }
162
- }
163
-
164
- // Handle preflight requests for CORS
165
- // Restrict to same origin instead of allowing all origins
166
- export async function OPTIONS() {
167
- return new Response(null, {
168
- status: 204,
169
- headers: {
170
- 'Access-Control-Allow-Origin': getAllowedOrigin(),
171
- 'Access-Control-Allow-Methods': 'POST, OPTIONS',
172
- 'Access-Control-Allow-Headers': 'Content-Type',
173
- },
174
- });
175
- }