@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,310 @@
1
+ import { NextRequest, NextResponse } from 'next/server';
2
+ import { auth } from '@nextsparkjs/core/lib/auth';
3
+ import { queryWithRLS } from '@nextsparkjs/core/lib/db';
4
+
5
+ interface SubscriptionResult {
6
+ id: string;
7
+ teamId: string;
8
+ teamName: string;
9
+ ownerName: string;
10
+ ownerEmail: string;
11
+ planId: string;
12
+ planSlug: string;
13
+ planName: string;
14
+ planType: string;
15
+ status: string;
16
+ priceMonthly: number | null;
17
+ priceYearly: number | null;
18
+ billingInterval: 'monthly' | 'yearly';
19
+ currentPeriodStart: string;
20
+ currentPeriodEnd: string;
21
+ trialEndsAt: string | null;
22
+ canceledAt: string | null;
23
+ cancelAtPeriodEnd: boolean;
24
+ externalSubscriptionId: string | null;
25
+ createdAt: string;
26
+ }
27
+
28
+ interface StatsResult {
29
+ totalSubscriptions: string;
30
+ activeSubscriptions: string;
31
+ trialingSubscriptions: string;
32
+ canceledSubscriptions: string;
33
+ pastDueSubscriptions: string;
34
+ freeSubscriptions: string;
35
+ paidSubscriptions: string;
36
+ monthlySubscriptions: string;
37
+ yearlySubscriptions: string;
38
+ totalMRR: string;
39
+ totalARR: string;
40
+ }
41
+
42
+ interface PlanDistribution {
43
+ planSlug: string;
44
+ planName: string;
45
+ count: string;
46
+ }
47
+
48
+ /**
49
+ * GET /api/superadmin/subscriptions
50
+ *
51
+ * Retrieves all subscriptions with stats for superadmin overview.
52
+ * Supports filtering by status and pagination.
53
+ */
54
+ export async function GET(request: NextRequest) {
55
+ try {
56
+ // Get the current session using Better Auth
57
+ const session = await auth.api.getSession({
58
+ headers: request.headers
59
+ });
60
+
61
+ // Check if user is authenticated
62
+ if (!session?.user) {
63
+ return NextResponse.json(
64
+ { error: 'Unauthorized - No session found' },
65
+ { status: 401 }
66
+ );
67
+ }
68
+
69
+ // Check if user is superadmin or developer
70
+ if (session.user.role !== 'superadmin' && session.user.role !== 'developer') {
71
+ return NextResponse.json(
72
+ { error: 'Forbidden - Superadmin or developer access required' },
73
+ { status: 403 }
74
+ );
75
+ }
76
+
77
+ // Parse query params
78
+ const searchParams = request.nextUrl.searchParams;
79
+ const status = searchParams.get('status');
80
+ const plan = searchParams.get('plan');
81
+ const interval = searchParams.get('interval');
82
+ const search = searchParams.get('search');
83
+ const page = parseInt(searchParams.get('page') || '1', 10);
84
+ const limit = parseInt(searchParams.get('limit') || '20', 10);
85
+ const offset = (page - 1) * limit;
86
+
87
+ // Build WHERE clause
88
+ const conditions: string[] = [];
89
+ const queryParams: unknown[] = [];
90
+ let paramIndex = 1;
91
+
92
+ if (status && status !== 'all') {
93
+ conditions.push(`s.status = $${paramIndex}`);
94
+ queryParams.push(status);
95
+ paramIndex++;
96
+ }
97
+
98
+ if (plan && plan !== 'all') {
99
+ conditions.push(`p.slug = $${paramIndex}`);
100
+ queryParams.push(plan);
101
+ paramIndex++;
102
+ }
103
+
104
+ if (interval && interval !== 'all') {
105
+ conditions.push(`s."billingInterval" = $${paramIndex}`);
106
+ queryParams.push(interval);
107
+ paramIndex++;
108
+ }
109
+
110
+ if (search) {
111
+ conditions.push(`(
112
+ t.name ILIKE $${paramIndex} OR
113
+ u.email ILIKE $${paramIndex} OR
114
+ p.name ILIKE $${paramIndex}
115
+ )`);
116
+ queryParams.push(`%${search}%`);
117
+ paramIndex++;
118
+ }
119
+
120
+ const whereClause = conditions.length > 0
121
+ ? `WHERE ${conditions.join(' AND ')}`
122
+ : '';
123
+
124
+ // Query for stats
125
+ const statsQuery = `
126
+ SELECT
127
+ COUNT(*) as "totalSubscriptions",
128
+ COUNT(*) FILTER (WHERE s.status = 'active') as "activeSubscriptions",
129
+ COUNT(*) FILTER (WHERE s.status = 'trialing') as "trialingSubscriptions",
130
+ COUNT(*) FILTER (WHERE s.status = 'canceled') as "canceledSubscriptions",
131
+ COUNT(*) FILTER (WHERE s.status = 'past_due') as "pastDueSubscriptions",
132
+ COUNT(*) FILTER (WHERE p.type = 'free') as "freeSubscriptions",
133
+ COUNT(*) FILTER (WHERE p.type IN ('paid', 'enterprise')) as "paidSubscriptions",
134
+ COUNT(*) FILTER (WHERE s."billingInterval" = 'monthly') as "monthlySubscriptions",
135
+ COUNT(*) FILTER (WHERE s."billingInterval" = 'yearly') as "yearlySubscriptions",
136
+ COALESCE(SUM(p."priceMonthly") FILTER (
137
+ WHERE s.status IN ('active', 'trialing')
138
+ AND s."billingInterval" = 'monthly'
139
+ AND p."priceMonthly" > 0
140
+ ), 0) as "totalMRR",
141
+ COALESCE(SUM(p."priceYearly") FILTER (
142
+ WHERE s.status IN ('active', 'trialing')
143
+ AND s."billingInterval" = 'yearly'
144
+ AND p."priceYearly" > 0
145
+ ), 0) as "totalARR"
146
+ FROM "subscriptions" s
147
+ LEFT JOIN "plans" p ON s."planId" = p.id
148
+ `;
149
+
150
+ // Query for plan distribution
151
+ const planDistributionQuery = `
152
+ SELECT
153
+ p.slug as "planSlug",
154
+ p.name as "planName",
155
+ COUNT(*) as count
156
+ FROM "subscriptions" s
157
+ LEFT JOIN "plans" p ON s."planId" = p.id
158
+ WHERE s.status IN ('active', 'trialing')
159
+ GROUP BY p.slug, p.name
160
+ ORDER BY count DESC
161
+ `;
162
+
163
+ // Query for subscriptions list
164
+ const subscriptionsQuery = `
165
+ SELECT
166
+ s.id,
167
+ s."teamId",
168
+ t.name as "teamName",
169
+ COALESCE(u."firstName" || ' ' || u."lastName", u.email) as "ownerName",
170
+ u.email as "ownerEmail",
171
+ s."planId",
172
+ p.slug as "planSlug",
173
+ p.name as "planName",
174
+ p.type as "planType",
175
+ s.status,
176
+ p."priceMonthly",
177
+ p."priceYearly",
178
+ s."billingInterval",
179
+ s."currentPeriodStart",
180
+ s."currentPeriodEnd",
181
+ s."trialEndsAt",
182
+ s."canceledAt",
183
+ s."cancelAtPeriodEnd",
184
+ s."externalSubscriptionId",
185
+ s."createdAt"
186
+ FROM "subscriptions" s
187
+ LEFT JOIN "teams" t ON s."teamId" = t.id
188
+ LEFT JOIN "users" u ON t."ownerId" = u.id
189
+ LEFT JOIN "plans" p ON s."planId" = p.id
190
+ ${whereClause}
191
+ ORDER BY s."createdAt" DESC
192
+ LIMIT $${paramIndex} OFFSET $${paramIndex + 1}
193
+ `;
194
+
195
+ // Count query for pagination
196
+ const countQuery = `
197
+ SELECT COUNT(*) as total
198
+ FROM "subscriptions" s
199
+ LEFT JOIN "teams" t ON s."teamId" = t.id
200
+ LEFT JOIN "users" u ON t."ownerId" = u.id
201
+ LEFT JOIN "plans" p ON s."planId" = p.id
202
+ ${whereClause}
203
+ `;
204
+
205
+ // Execute all queries
206
+ const [statsResult, planDistResult, subscriptionsResult, countResult] = await Promise.all([
207
+ queryWithRLS(statsQuery, [], session.user.id) as Promise<StatsResult[]>,
208
+ queryWithRLS(planDistributionQuery, [], session.user.id) as Promise<PlanDistribution[]>,
209
+ queryWithRLS(subscriptionsQuery, [...queryParams, limit, offset], session.user.id) as Promise<SubscriptionResult[]>,
210
+ queryWithRLS(countQuery, queryParams, session.user.id) as Promise<{ total: string }[]>
211
+ ]);
212
+
213
+ const stats = statsResult[0] || {
214
+ totalSubscriptions: '0',
215
+ activeSubscriptions: '0',
216
+ trialingSubscriptions: '0',
217
+ canceledSubscriptions: '0',
218
+ pastDueSubscriptions: '0',
219
+ freeSubscriptions: '0',
220
+ paidSubscriptions: '0',
221
+ monthlySubscriptions: '0',
222
+ yearlySubscriptions: '0',
223
+ totalMRR: '0',
224
+ totalARR: '0'
225
+ };
226
+
227
+ const total = parseInt(countResult[0]?.total || '0', 10);
228
+ const totalPages = Math.ceil(total / limit);
229
+
230
+ // Format subscriptions
231
+ const subscriptions = subscriptionsResult.map(sub => ({
232
+ id: sub.id,
233
+ team: {
234
+ id: sub.teamId,
235
+ name: sub.teamName,
236
+ owner: {
237
+ name: sub.ownerName?.trim() || sub.ownerEmail,
238
+ email: sub.ownerEmail
239
+ }
240
+ },
241
+ plan: {
242
+ id: sub.planId,
243
+ slug: sub.planSlug,
244
+ name: sub.planName,
245
+ type: sub.planType,
246
+ priceMonthly: sub.priceMonthly,
247
+ priceYearly: sub.priceYearly
248
+ },
249
+ billingInterval: sub.billingInterval,
250
+ status: sub.status,
251
+ currentPeriodStart: sub.currentPeriodStart,
252
+ currentPeriodEnd: sub.currentPeriodEnd,
253
+ trialEndsAt: sub.trialEndsAt,
254
+ canceledAt: sub.canceledAt,
255
+ cancelAtPeriodEnd: sub.cancelAtPeriodEnd,
256
+ externalSubscriptionId: sub.externalSubscriptionId,
257
+ createdAt: sub.createdAt
258
+ }));
259
+
260
+ // Prepare response
261
+ const responseData = {
262
+ stats: {
263
+ total: parseInt(stats.totalSubscriptions, 10),
264
+ active: parseInt(stats.activeSubscriptions, 10),
265
+ trialing: parseInt(stats.trialingSubscriptions, 10),
266
+ canceled: parseInt(stats.canceledSubscriptions, 10),
267
+ pastDue: parseInt(stats.pastDueSubscriptions, 10),
268
+ free: parseInt(stats.freeSubscriptions, 10),
269
+ paid: parseInt(stats.paidSubscriptions, 10),
270
+ monthly: parseInt(stats.monthlySubscriptions, 10),
271
+ yearly: parseInt(stats.yearlySubscriptions, 10),
272
+ mrr: parseInt(stats.totalMRR, 10), // In cents
273
+ mrrFormatted: `$${(parseInt(stats.totalMRR, 10) / 100).toFixed(2)}`,
274
+ arr: parseInt(stats.totalARR, 10), // In cents
275
+ arrFormatted: `$${(parseInt(stats.totalARR, 10) / 100).toFixed(2)}`
276
+ },
277
+ planDistribution: planDistResult.map(p => ({
278
+ slug: p.planSlug,
279
+ name: p.planName,
280
+ count: parseInt(p.count, 10)
281
+ })),
282
+ subscriptions,
283
+ pagination: {
284
+ page,
285
+ limit,
286
+ total,
287
+ totalPages,
288
+ hasMore: page < totalPages
289
+ },
290
+ metadata: {
291
+ requestedBy: session.user.id,
292
+ requestedAt: new Date().toISOString(),
293
+ source: 'superadmin-api'
294
+ }
295
+ };
296
+
297
+ return NextResponse.json(responseData);
298
+
299
+ } catch (error) {
300
+ console.error('Error fetching subscriptions:', error);
301
+
302
+ return NextResponse.json(
303
+ {
304
+ error: 'Internal server error',
305
+ message: 'Failed to retrieve subscriptions data'
306
+ },
307
+ { status: 500 }
308
+ );
309
+ }
310
+ }
@@ -0,0 +1,286 @@
1
+ import { NextRequest, NextResponse } from 'next/server';
2
+ import { auth } from '@nextsparkjs/core/lib/auth';
3
+ import { queryWithRLS } from '@nextsparkjs/core/lib/db';
4
+
5
+ interface TeamResult {
6
+ id: string;
7
+ name: string;
8
+ ownerId: string;
9
+ ownerName: string;
10
+ ownerEmail: string;
11
+ createdAt: string;
12
+ updatedAt: string;
13
+ }
14
+
15
+ interface TeamMemberResult {
16
+ id: string;
17
+ name: string;
18
+ email: string;
19
+ role: string;
20
+ joinedAt: string;
21
+ userId: string;
22
+ }
23
+
24
+ interface SubscriptionResult {
25
+ id: string;
26
+ planId: string;
27
+ planSlug: string;
28
+ planName: string;
29
+ planType: string;
30
+ priceMonthly: number | null;
31
+ status: string;
32
+ currentPeriodStart: string;
33
+ currentPeriodEnd: string;
34
+ trialEndsAt: string | null;
35
+ canceledAt: string | null;
36
+ cancelAtPeriodEnd: boolean;
37
+ externalSubscriptionId: string | null;
38
+ externalCustomerId: string | null;
39
+ createdAt: string;
40
+ }
41
+
42
+ interface BillingEventResult {
43
+ id: string;
44
+ type: string;
45
+ status: string;
46
+ amount: number;
47
+ currency: string;
48
+ invoiceUrl: string | null;
49
+ receiptUrl: string | null;
50
+ createdAt: string;
51
+ }
52
+
53
+ interface UsageResult {
54
+ limitSlug: string;
55
+ currentValue: number;
56
+ periodKey: string;
57
+ }
58
+
59
+ /**
60
+ * GET /api/superadmin/teams/[teamId]
61
+ *
62
+ * Retrieves a single team with owner info and members.
63
+ * Only accessible by superadmin or developer users.
64
+ */
65
+ export async function GET(
66
+ request: NextRequest,
67
+ { params }: { params: Promise<{ teamId: string }> }
68
+ ) {
69
+ try {
70
+ const { teamId } = await params;
71
+
72
+ // Get the current session using Better Auth
73
+ const session = await auth.api.getSession({
74
+ headers: request.headers
75
+ });
76
+
77
+ // Check if user is authenticated
78
+ if (!session?.user) {
79
+ return NextResponse.json(
80
+ { error: 'Unauthorized - No session found' },
81
+ { status: 401 }
82
+ );
83
+ }
84
+
85
+ // Check if user is superadmin or developer
86
+ if (session.user.role !== 'superadmin' && session.user.role !== 'developer') {
87
+ return NextResponse.json(
88
+ { error: 'Forbidden - Superadmin or developer access required' },
89
+ { status: 403 }
90
+ );
91
+ }
92
+
93
+ // Query for team details
94
+ const teamQuery = `
95
+ SELECT
96
+ t.id,
97
+ t.name,
98
+ t."ownerId",
99
+ COALESCE(u."firstName" || ' ' || u."lastName", u.email) as "ownerName",
100
+ u.email as "ownerEmail",
101
+ t."createdAt",
102
+ t."updatedAt"
103
+ FROM "teams" t
104
+ LEFT JOIN "users" u ON t."ownerId" = u.id
105
+ WHERE t.id = $1
106
+ `;
107
+
108
+ // Query for team members
109
+ const membersQuery = `
110
+ SELECT
111
+ tm.id,
112
+ tm.role,
113
+ tm."joinedAt",
114
+ COALESCE(u."firstName" || ' ' || u."lastName", u.email) as name,
115
+ u.email,
116
+ u.id as "userId"
117
+ FROM "team_members" tm
118
+ LEFT JOIN "users" u ON tm."userId" = u.id
119
+ WHERE tm."teamId" = $1
120
+ ORDER BY
121
+ CASE tm.role
122
+ WHEN 'owner' THEN 1
123
+ WHEN 'admin' THEN 2
124
+ WHEN 'member' THEN 3
125
+ WHEN 'viewer' THEN 4
126
+ ELSE 5
127
+ END,
128
+ tm."joinedAt" ASC
129
+ `;
130
+
131
+ // Query for subscription
132
+ const subscriptionQuery = `
133
+ SELECT
134
+ s.id,
135
+ s."planId",
136
+ p.slug as "planSlug",
137
+ p.name as "planName",
138
+ p.type as "planType",
139
+ p."priceMonthly",
140
+ s.status,
141
+ s."currentPeriodStart",
142
+ s."currentPeriodEnd",
143
+ s."trialEndsAt",
144
+ s."canceledAt",
145
+ s."cancelAtPeriodEnd",
146
+ s."externalSubscriptionId",
147
+ s."externalCustomerId",
148
+ s."createdAt"
149
+ FROM "subscriptions" s
150
+ LEFT JOIN "plans" p ON s."planId" = p.id
151
+ WHERE s."teamId" = $1
152
+ ORDER BY s."createdAt" DESC
153
+ LIMIT 1
154
+ `;
155
+
156
+ // Query for billing events
157
+ const billingEventsQuery = `
158
+ SELECT
159
+ be.id,
160
+ be.type,
161
+ be.status,
162
+ be.amount,
163
+ be.currency,
164
+ be."invoiceUrl",
165
+ be."receiptUrl",
166
+ be."createdAt"
167
+ FROM "billing_events" be
168
+ JOIN "subscriptions" s ON be."subscriptionId" = s.id
169
+ WHERE s."teamId" = $1
170
+ ORDER BY be."createdAt" DESC
171
+ LIMIT 10
172
+ `;
173
+
174
+ // Query for usage
175
+ const usageQuery = `
176
+ SELECT
177
+ u."limitSlug",
178
+ u."currentValue",
179
+ u."periodKey"
180
+ FROM "usage" u
181
+ JOIN "subscriptions" s ON u."subscriptionId" = s.id
182
+ WHERE s."teamId" = $1
183
+ `;
184
+
185
+ // Execute queries
186
+ const [teamResult, membersResult, subscriptionResult, billingEventsResult, usageResult] = await Promise.all([
187
+ queryWithRLS(teamQuery, [teamId], session.user.id) as Promise<TeamResult[]>,
188
+ queryWithRLS(membersQuery, [teamId], session.user.id) as Promise<TeamMemberResult[]>,
189
+ queryWithRLS(subscriptionQuery, [teamId], session.user.id) as Promise<SubscriptionResult[]>,
190
+ queryWithRLS(billingEventsQuery, [teamId], session.user.id) as Promise<BillingEventResult[]>,
191
+ queryWithRLS(usageQuery, [teamId], session.user.id) as Promise<UsageResult[]>
192
+ ]);
193
+
194
+ if (!teamResult || teamResult.length === 0) {
195
+ return NextResponse.json(
196
+ { error: 'Team not found' },
197
+ { status: 404 }
198
+ );
199
+ }
200
+
201
+ const team = teamResult[0];
202
+ const subscription = subscriptionResult[0] || null;
203
+
204
+ // Prepare response data
205
+ const responseData = {
206
+ team: {
207
+ id: team.id,
208
+ name: team.name,
209
+ owner: {
210
+ id: team.ownerId,
211
+ name: team.ownerName?.trim() || team.ownerEmail,
212
+ email: team.ownerEmail
213
+ },
214
+ memberCount: membersResult.length,
215
+ createdAt: team.createdAt,
216
+ updatedAt: team.updatedAt
217
+ },
218
+ members: membersResult.map((member) => ({
219
+ id: member.id,
220
+ userId: member.userId,
221
+ name: member.name?.trim() || member.email,
222
+ email: member.email,
223
+ role: member.role,
224
+ joinedAt: member.joinedAt
225
+ })),
226
+ subscription: subscription ? {
227
+ id: subscription.id,
228
+ plan: {
229
+ id: subscription.planId,
230
+ slug: subscription.planSlug,
231
+ name: subscription.planName,
232
+ type: subscription.planType,
233
+ priceMonthly: subscription.priceMonthly,
234
+ priceFormatted: subscription.priceMonthly
235
+ ? `$${(subscription.priceMonthly / 100).toFixed(2)}/mo`
236
+ : 'Free'
237
+ },
238
+ status: subscription.status,
239
+ currentPeriodStart: subscription.currentPeriodStart,
240
+ currentPeriodEnd: subscription.currentPeriodEnd,
241
+ trialEndsAt: subscription.trialEndsAt,
242
+ canceledAt: subscription.canceledAt,
243
+ cancelAtPeriodEnd: subscription.cancelAtPeriodEnd,
244
+ externalSubscriptionId: subscription.externalSubscriptionId,
245
+ externalCustomerId: subscription.externalCustomerId,
246
+ createdAt: subscription.createdAt
247
+ } : null,
248
+ billingHistory: billingEventsResult.map((event) => ({
249
+ id: event.id,
250
+ type: event.type,
251
+ status: event.status,
252
+ amount: event.amount,
253
+ amountFormatted: `$${(event.amount / 100).toFixed(2)}`,
254
+ currency: event.currency,
255
+ invoiceUrl: event.invoiceUrl,
256
+ receiptUrl: event.receiptUrl,
257
+ createdAt: event.createdAt
258
+ })),
259
+ usage: usageResult.reduce((acc, u) => {
260
+ acc[u.limitSlug] = {
261
+ current: u.currentValue,
262
+ periodKey: u.periodKey
263
+ };
264
+ return acc;
265
+ }, {} as Record<string, { current: number; periodKey: string }>),
266
+ metadata: {
267
+ requestedBy: session.user.id,
268
+ requestedAt: new Date().toISOString(),
269
+ source: 'superadmin-api'
270
+ }
271
+ };
272
+
273
+ return NextResponse.json(responseData);
274
+
275
+ } catch (error) {
276
+ console.error('Error fetching team data:', error);
277
+
278
+ return NextResponse.json(
279
+ {
280
+ error: 'Internal server error',
281
+ message: 'Failed to retrieve team data'
282
+ },
283
+ { status: 500 }
284
+ );
285
+ }
286
+ }