@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,513 @@
1
+ "use client";
2
+
3
+ import { useState } from 'react';
4
+ import { useQuery, useMutation, useQueryClient } from '@tanstack/react-query';
5
+ import { Button } from '@nextsparkjs/core/components/ui/button';
6
+ import { Card, CardContent, CardDescription, CardHeader, CardTitle } from '@nextsparkjs/core/components/ui/card';
7
+ import { Badge } from '@nextsparkjs/core/components/ui/badge';
8
+ import { Alert, AlertDescription } from '@nextsparkjs/core/components/ui/alert';
9
+ import { Skeleton } from '@nextsparkjs/core/components/ui/skeleton';
10
+ import {
11
+ Dialog,
12
+ DialogContent,
13
+ DialogDescription,
14
+ DialogHeader,
15
+ DialogTitle
16
+ } from '@nextsparkjs/core/components/ui/dialog';
17
+ import {
18
+ DropdownMenu,
19
+ DropdownMenuContent,
20
+ DropdownMenuItem,
21
+ DropdownMenuTrigger
22
+ } from '@nextsparkjs/core/components/ui/dropdown-menu';
23
+ import {
24
+ AlertTriangle,
25
+ Copy,
26
+ Eye,
27
+ EyeOff,
28
+ Key,
29
+ MoreVertical,
30
+ Plus,
31
+ Trash2,
32
+ Activity
33
+ } from 'lucide-react';
34
+ import { CreateApiKeyDialog } from '@nextsparkjs/core/components/api/keys/CreateApiKeyDialog';
35
+ import { ApiKeyDisplay } from '@nextsparkjs/core/components/api/keys/ApiKeyDisplay';
36
+ import { toast } from 'sonner';
37
+ import { getTemplateOrDefaultClient } from '@nextsparkjs/registries/template-registry.client'
38
+ import { sel } from '@nextsparkjs/core/selectors'
39
+
40
+ interface ApiKey {
41
+ id: string;
42
+ keyPrefix: string;
43
+ name: string;
44
+ scopes: string[];
45
+ status: 'active' | 'inactive' | 'expired';
46
+ lastUsedAt: string | null;
47
+ expiresAt: string | null;
48
+ createdAt: string;
49
+ usage_stats: {
50
+ total_requests: number;
51
+ last_24h: number;
52
+ avg_response_time: number | null;
53
+ };
54
+ }
55
+
56
+ interface NewApiKeyResponse {
57
+ id: string;
58
+ name: string;
59
+ key: string;
60
+ scopes: string[];
61
+ warning: string;
62
+ }
63
+
64
+ function ApiKeysPage() {
65
+ const [showCreateDialog, setShowCreateDialog] = useState(false);
66
+ const [newApiKey, setNewApiKey] = useState<NewApiKeyResponse | null>(null);
67
+ const [selectedKey, setSelectedKey] = useState<string | null>(null);
68
+ const queryClient = useQueryClient();
69
+
70
+ // Fetch API keys
71
+ const { data: apiKeys, isLoading, error } = useQuery<ApiKey[]>({
72
+ queryKey: ['api-keys'],
73
+ queryFn: async () => {
74
+ const response = await fetch('/api/v1/api-keys');
75
+ if (!response.ok) {
76
+ const errorData = await response.json();
77
+ throw new Error(errorData.error || 'Failed to fetch API keys');
78
+ }
79
+ const result = await response.json();
80
+ return result.data;
81
+ }
82
+ });
83
+
84
+ // Revoke API key mutation
85
+ const revokeApiKey = useMutation({
86
+ mutationFn: async (keyId: string) => {
87
+ const response = await fetch(`/api/v1/api-keys/${keyId}`, {
88
+ method: 'DELETE'
89
+ });
90
+ if (!response.ok) {
91
+ const errorData = await response.json();
92
+ throw new Error(errorData.error || 'Failed to revoke API key');
93
+ }
94
+ return response.json();
95
+ },
96
+ onSuccess: (data) => {
97
+ toast.success(`API key "${data.data.name}" has been revoked`);
98
+ queryClient.invalidateQueries({ queryKey: ['api-keys'] });
99
+ },
100
+ onError: (error) => {
101
+ toast.error(`Failed to revoke API key: ${error.message}`);
102
+ }
103
+ });
104
+
105
+ // Toggle API key status
106
+ const toggleApiKey = useMutation({
107
+ mutationFn: async ({ keyId, status }: { keyId: string; status: 'active' | 'inactive' }) => {
108
+ const response = await fetch(`/api/v1/api-keys/${keyId}`, {
109
+ method: 'PATCH',
110
+ headers: { 'Content-Type': 'application/json' },
111
+ body: JSON.stringify({ status })
112
+ });
113
+ if (!response.ok) {
114
+ const errorData = await response.json();
115
+ throw new Error(errorData.error || 'Failed to update API key');
116
+ }
117
+ return response.json();
118
+ },
119
+ onSuccess: (data) => {
120
+ const statusText = data.data.status === 'active' ? 'activated' : 'deactivated';
121
+ toast.success(`API key "${data.data.name}" has been ${statusText}`);
122
+ queryClient.invalidateQueries({ queryKey: ['api-keys'] });
123
+ },
124
+ onError: (error) => {
125
+ toast.error(`Failed to update API key: ${error.message}`);
126
+ }
127
+ });
128
+
129
+ const handleCreateSuccess = (apiKeyData: NewApiKeyResponse) => {
130
+ setNewApiKey(apiKeyData);
131
+ setShowCreateDialog(false);
132
+ queryClient.invalidateQueries({ queryKey: ['api-keys'] });
133
+ };
134
+
135
+ const copyToClipboard = (text: string) => {
136
+ navigator.clipboard.writeText(text);
137
+ toast.success('Copied to clipboard');
138
+ };
139
+
140
+ const formatDate = (dateString: string) => {
141
+ return new Date(dateString).toLocaleDateString('es-ES', {
142
+ year: 'numeric',
143
+ month: 'short',
144
+ day: 'numeric',
145
+ hour: '2-digit',
146
+ minute: '2-digit'
147
+ });
148
+ };
149
+
150
+ const getScopeColor = (scope: string) => {
151
+ if (scope === '*') return 'destructive';
152
+ if (scope.includes('admin')) return 'secondary';
153
+ if (scope.includes('write') || scope.includes('delete')) return 'default';
154
+ return 'outline';
155
+ };
156
+
157
+ if (error) {
158
+ const isPermissionError = error.message.includes('Insufficient permissions');
159
+
160
+ return (
161
+ <div className="space-y-6">
162
+ <div>
163
+ <h1 className="text-2xl font-bold">API Keys</h1>
164
+ <p className="text-muted-foreground">
165
+ Gestiona las API keys para integración externa
166
+ </p>
167
+ </div>
168
+ <Alert variant={isPermissionError ? "default" : "destructive"}>
169
+ <AlertTriangle className="h-4 w-4" />
170
+ <AlertDescription>
171
+ {isPermissionError ? (
172
+ <div>
173
+ <strong>Acceso restringido</strong>
174
+ <p className="mt-1">
175
+ Solo los administradores pueden gestionar API keys.
176
+ Contacta a un administrador si necesitas acceso a esta funcionalidad.
177
+ </p>
178
+ </div>
179
+ ) : (
180
+ `Error loading API keys: ${error.message}`
181
+ )}
182
+ </AlertDescription>
183
+ </Alert>
184
+ </div>
185
+ );
186
+ }
187
+
188
+ return (
189
+ <div className="space-y-6" data-cy={sel('settings.apiKeys.container')}>
190
+ <div className="flex justify-between items-center">
191
+ <div>
192
+ <h1 className="text-2xl font-bold" data-cy={sel('settings.apiKeys.header')}>API Keys</h1>
193
+ <p className="text-muted-foreground">
194
+ Gestiona las API keys para integración externa
195
+ </p>
196
+ </div>
197
+ <Button onClick={() => setShowCreateDialog(true)} data-cy={sel('settings.apiKeys.createButton')}>
198
+ <Plus className="h-4 w-4 mr-2" />
199
+ Crear API Key
200
+ </Button>
201
+ </div>
202
+
203
+ {/* New API Key Display */}
204
+ {newApiKey && (
205
+ <ApiKeyDisplay
206
+ apiKey={newApiKey.key}
207
+ keyName={newApiKey.name}
208
+ warning={newApiKey.warning}
209
+ onClose={() => setNewApiKey(null)}
210
+ />
211
+ )}
212
+
213
+ {/* API Keys List */}
214
+ <div className="grid gap-4" data-cy={sel('settings.apiKeys.list.container')}>
215
+ {isLoading ? (
216
+ // Loading skeletons
217
+ Array.from({ length: 3 }).map((_, i) => (
218
+ <Card key={i} data-cy={sel('settings.apiKeys.list.skeleton')}>
219
+ <CardHeader>
220
+ <div className="flex justify-between items-start">
221
+ <div className="space-y-2">
222
+ <Skeleton className="h-5 w-32" />
223
+ <Skeleton className="h-4 w-64" />
224
+ </div>
225
+ <Skeleton className="h-8 w-20" />
226
+ </div>
227
+ </CardHeader>
228
+ <CardContent>
229
+ <div className="space-y-2">
230
+ <Skeleton className="h-4 w-48" />
231
+ <Skeleton className="h-4 w-32" />
232
+ </div>
233
+ </CardContent>
234
+ </Card>
235
+ ))
236
+ ) : apiKeys?.length === 0 ? (
237
+ <Card data-cy={sel('settings.apiKeys.list.empty')}>
238
+ <CardContent className="flex flex-col items-center justify-center py-12">
239
+ <Key className="h-12 w-12 text-muted-foreground mb-4" />
240
+ <h3 className="text-lg font-semibold mb-2">No API Keys</h3>
241
+ <p className="text-muted-foreground text-center mb-4">
242
+ No tienes API keys creadas. Crea una para empezar a usar la API externa.
243
+ </p>
244
+ <Button onClick={() => setShowCreateDialog(true)} data-cy={sel('settings.apiKeys.createButton')}>
245
+ <Plus className="h-4 w-4 mr-2" />
246
+ Crear primera API Key
247
+ </Button>
248
+ </CardContent>
249
+ </Card>
250
+ ) : (
251
+ apiKeys?.map((apiKey) => (
252
+ <Card key={apiKey.id} data-cy={sel('settings.apiKeys.row.container', { id: apiKey.id })}>
253
+ <CardHeader>
254
+ <div className="flex justify-between items-start">
255
+ <div>
256
+ <CardTitle className="text-lg flex items-center gap-2" data-cy={sel('settings.apiKeys.row.name', { id: apiKey.id })}>
257
+ {apiKey.name}
258
+ {apiKey.status !== 'active' && (
259
+ <Badge variant="secondary" data-cy={sel('settings.apiKeys.row.status', { id: apiKey.id })}>
260
+ {apiKey.status === 'inactive' ? 'Inactiva' : 'Expirada'}
261
+ </Badge>
262
+ )}
263
+ </CardTitle>
264
+ <CardDescription className="font-mono text-xs" data-cy={sel('settings.apiKeys.row.prefix', { id: apiKey.id })}>
265
+ {apiKey.keyPrefix}••••••••••••••••••••••••••••••••••••••••••••••••••••
266
+ <Button
267
+ variant="ghost"
268
+ size="sm"
269
+ className="ml-2 h-6 w-6 p-0"
270
+ onClick={() => copyToClipboard(apiKey.keyPrefix)}
271
+ data-cy={sel('settings.apiKeys.row.copyPrefix', { id: apiKey.id })}
272
+ >
273
+ <Copy className="h-3 w-3" />
274
+ </Button>
275
+ </CardDescription>
276
+ </div>
277
+ <div className="flex items-center gap-2">
278
+ <Badge variant={apiKey.status === 'active' ? "default" : "secondary"} data-cy={sel('settings.apiKeys.row.status', { id: apiKey.id })}>
279
+ {apiKey.status === 'active' ? 'Activa' : apiKey.status === 'inactive' ? 'Inactiva' : 'Expirada'}
280
+ </Badge>
281
+ <DropdownMenu>
282
+ <DropdownMenuTrigger asChild>
283
+ <Button variant="ghost" size="sm" data-cy={sel('settings.apiKeys.row.menu.trigger', { id: apiKey.id })}>
284
+ <MoreVertical className="h-4 w-4" />
285
+ </Button>
286
+ </DropdownMenuTrigger>
287
+ <DropdownMenuContent align="end" data-cy={sel('settings.apiKeys.row.menu.content', { id: apiKey.id })}>
288
+ <DropdownMenuItem
289
+ onClick={() => setSelectedKey(apiKey.id)}
290
+ data-cy={sel('settings.apiKeys.row.menu.viewDetails', { id: apiKey.id })}
291
+ >
292
+ <Eye className="h-4 w-4 mr-2" />
293
+ Ver detalles
294
+ </DropdownMenuItem>
295
+ <DropdownMenuItem
296
+ onClick={() => toggleApiKey.mutate({
297
+ keyId: apiKey.id,
298
+ status: apiKey.status === 'active' ? 'inactive' : 'active'
299
+ })}
300
+ disabled={toggleApiKey.isPending}
301
+ data-cy={sel('settings.apiKeys.row.menu.toggle', { id: apiKey.id })}
302
+ >
303
+ {apiKey.status === 'active' ? (
304
+ <>
305
+ <EyeOff className="h-4 w-4 mr-2" />
306
+ Desactivar
307
+ </>
308
+ ) : (
309
+ <>
310
+ <Eye className="h-4 w-4 mr-2" />
311
+ Activar
312
+ </>
313
+ )}
314
+ </DropdownMenuItem>
315
+ <DropdownMenuItem
316
+ onClick={() => revokeApiKey.mutate(apiKey.id)}
317
+ disabled={revokeApiKey.isPending}
318
+ className="text-destructive"
319
+ data-cy={sel('settings.apiKeys.row.menu.revoke', { id: apiKey.id })}
320
+ >
321
+ <Trash2 className="h-4 w-4 mr-2" />
322
+ Revocar
323
+ </DropdownMenuItem>
324
+ </DropdownMenuContent>
325
+ </DropdownMenu>
326
+ </div>
327
+ </div>
328
+ </CardHeader>
329
+ <CardContent>
330
+ <div className="space-y-4">
331
+ {/* Scopes */}
332
+ <div data-cy={sel('settings.apiKeys.row.scopes', { id: apiKey.id })}>
333
+ <strong className="text-sm">Permisos:</strong>
334
+ <div className="flex flex-wrap gap-1 mt-1">
335
+ {apiKey.scopes.map((scope) => (
336
+ <Badge
337
+ key={scope}
338
+ variant={getScopeColor(scope)}
339
+ className="text-xs"
340
+ data-cy={sel('settings.apiKeys.row.scope', { id: apiKey.id, scope })}
341
+ >
342
+ {scope}
343
+ </Badge>
344
+ ))}
345
+ </div>
346
+ </div>
347
+
348
+ {/* Usage Stats */}
349
+ <div className="grid grid-cols-3 gap-4 text-sm" data-cy={sel('settings.apiKeys.row.stats.container', { id: apiKey.id })}>
350
+ <div data-cy={sel('settings.apiKeys.row.stats.totalRequests', { id: apiKey.id })}>
351
+ <div className="text-muted-foreground">Total requests</div>
352
+ <div className="font-semibold">{apiKey.usage_stats.total_requests.toLocaleString()}</div>
353
+ </div>
354
+ <div data-cy={sel('settings.apiKeys.row.stats.last24h', { id: apiKey.id })}>
355
+ <div className="text-muted-foreground">Últimas 24h</div>
356
+ <div className="font-semibold">{apiKey.usage_stats.last_24h.toLocaleString()}</div>
357
+ </div>
358
+ <div data-cy={sel('settings.apiKeys.row.stats.avgTime', { id: apiKey.id })}>
359
+ <div className="text-muted-foreground">Tiempo promedio</div>
360
+ <div className="font-semibold">
361
+ {apiKey.usage_stats.avg_response_time
362
+ ? `${Math.round(apiKey.usage_stats.avg_response_time)}ms`
363
+ : 'N/A'
364
+ }
365
+ </div>
366
+ </div>
367
+ </div>
368
+
369
+ {/* Metadata */}
370
+ <div className="text-sm text-muted-foreground space-y-1" data-cy={sel('settings.apiKeys.row.metadata.container', { id: apiKey.id })}>
371
+ <div data-cy={sel('settings.apiKeys.row.metadata.createdAt', { id: apiKey.id })}>Creada: {formatDate(apiKey.createdAt)}</div>
372
+ {apiKey.lastUsedAt && (
373
+ <div data-cy={sel('settings.apiKeys.row.metadata.lastUsed', { id: apiKey.id })}>Ultimo uso: {formatDate(apiKey.lastUsedAt)}</div>
374
+ )}
375
+ {apiKey.expiresAt && (
376
+ <div data-cy={sel('settings.apiKeys.row.metadata.expiresAt', { id: apiKey.id })}>Expira: {formatDate(apiKey.expiresAt)}</div>
377
+ )}
378
+ </div>
379
+ </div>
380
+ </CardContent>
381
+ </Card>
382
+ ))
383
+ )}
384
+ </div>
385
+
386
+ {/* Create API Key Dialog */}
387
+ <CreateApiKeyDialog
388
+ open={showCreateDialog}
389
+ onClose={() => setShowCreateDialog(false)}
390
+ onSuccess={handleCreateSuccess}
391
+ />
392
+
393
+ {/* API Key Details Dialog */}
394
+ {selectedKey && (
395
+ <ApiKeyDetailsDialog
396
+ keyId={selectedKey}
397
+ open={!!selectedKey}
398
+ onClose={() => setSelectedKey(null)}
399
+ />
400
+ )}
401
+ </div>
402
+ );
403
+ }
404
+
405
+ // Componente para mostrar detalles de API key
406
+ function ApiKeyDetailsDialog({
407
+ keyId,
408
+ open,
409
+ onClose
410
+ }: {
411
+ keyId: string;
412
+ open: boolean;
413
+ onClose: () => void;
414
+ }) {
415
+ const { data: keyDetails, isLoading } = useQuery({
416
+ queryKey: ['api-key-details', keyId],
417
+ queryFn: async () => {
418
+ const response = await fetch(`/api/v1/api-keys/${keyId}`);
419
+ if (!response.ok) throw new Error('Failed to fetch API key details');
420
+ const result = await response.json();
421
+ return result.data;
422
+ },
423
+ enabled: open
424
+ });
425
+
426
+ return (
427
+ <Dialog open={open} onOpenChange={onClose}>
428
+ <DialogContent className="max-w-2xl" data-cy={sel('settings.apiKeys.detailsDialog.container')}>
429
+ <DialogHeader>
430
+ <DialogTitle data-cy={sel('settings.apiKeys.detailsDialog.title')}>Detalles de API Key</DialogTitle>
431
+ <DialogDescription>
432
+ Estadísticas de uso y configuración
433
+ </DialogDescription>
434
+ </DialogHeader>
435
+
436
+ {isLoading ? (
437
+ <div className="space-y-4" data-cy={sel('settings.apiKeys.detailsDialog.loading')}>
438
+ <Skeleton className="h-4 w-full" />
439
+ <Skeleton className="h-4 w-3/4" />
440
+ <Skeleton className="h-32 w-full" />
441
+ </div>
442
+ ) : keyDetails ? (
443
+ <div className="space-y-6" data-cy={sel('settings.apiKeys.detailsDialog.content')}>
444
+ {/* Basic Info */}
445
+ <div data-cy={sel('settings.apiKeys.detailsDialog.basicInfo.container')}>
446
+ <h4 className="font-semibold mb-2">Información básica</h4>
447
+ <div className="grid grid-cols-2 gap-4 text-sm">
448
+ <div>
449
+ <span className="text-muted-foreground">Nombre:</span>
450
+ <div className="font-medium" data-cy={sel('settings.apiKeys.detailsDialog.basicInfo.name')}>{keyDetails.name}</div>
451
+ </div>
452
+ <div>
453
+ <span className="text-muted-foreground">Estado:</span>
454
+ <div>
455
+ <Badge variant={keyDetails.status === 'active' ? "default" : "secondary"} data-cy={sel('settings.apiKeys.detailsDialog.basicInfo.status')}>
456
+ {keyDetails.status === 'active' ? 'Activa' : keyDetails.status === 'inactive' ? 'Inactiva' : 'Expirada'}
457
+ </Badge>
458
+ </div>
459
+ </div>
460
+ </div>
461
+ </div>
462
+
463
+ {/* Usage Statistics */}
464
+ <div data-cy={sel('settings.apiKeys.detailsDialog.stats.container')}>
465
+ <h4 className="font-semibold mb-2 flex items-center gap-2">
466
+ <Activity className="h-4 w-4" />
467
+ Estadísticas de uso
468
+ </h4>
469
+ <div className="grid grid-cols-2 gap-4 text-sm">
470
+ <div>
471
+ <span className="text-muted-foreground">Total de requests:</span>
472
+ <div className="font-medium" data-cy={sel('settings.apiKeys.detailsDialog.stats.totalRequests')}>{keyDetails.usage_stats.total_requests.toLocaleString()}</div>
473
+ </div>
474
+ <div>
475
+ <span className="text-muted-foreground">Ultimas 24 horas:</span>
476
+ <div className="font-medium" data-cy={sel('settings.apiKeys.detailsDialog.stats.last24h')}>{keyDetails.usage_stats.last_24h.toLocaleString()}</div>
477
+ </div>
478
+ <div>
479
+ <span className="text-muted-foreground">Ultimos 7 dias:</span>
480
+ <div className="font-medium" data-cy={sel('settings.apiKeys.detailsDialog.stats.last7d')}>{keyDetails.usage_stats.last_7d.toLocaleString()}</div>
481
+ </div>
482
+ <div>
483
+ <span className="text-muted-foreground">Ultimos 30 dias:</span>
484
+ <div className="font-medium" data-cy={sel('settings.apiKeys.detailsDialog.stats.last30d')}>{keyDetails.usage_stats.last_30d.toLocaleString()}</div>
485
+ </div>
486
+ <div>
487
+ <span className="text-muted-foreground">Tiempo promedio:</span>
488
+ <div className="font-medium" data-cy={sel('settings.apiKeys.detailsDialog.stats.avgTime')}>
489
+ {keyDetails.usage_stats.avg_response_time
490
+ ? `${Math.round(keyDetails.usage_stats.avg_response_time)}ms`
491
+ : 'N/A'
492
+ }
493
+ </div>
494
+ </div>
495
+ <div>
496
+ <span className="text-muted-foreground">Tasa de exito:</span>
497
+ <div className="font-medium" data-cy={sel('settings.apiKeys.detailsDialog.stats.successRate')}>
498
+ {keyDetails.usage_stats.success_rate
499
+ ? `${Math.round(keyDetails.usage_stats.success_rate)}%`
500
+ : 'N/A'
501
+ }
502
+ </div>
503
+ </div>
504
+ </div>
505
+ </div>
506
+ </div>
507
+ ) : null}
508
+ </DialogContent>
509
+ </Dialog>
510
+ );
511
+ }
512
+
513
+ export default getTemplateOrDefaultClient('app/dashboard/settings/api-keys/page.tsx', ApiKeysPage)
@@ -0,0 +1,5 @@
1
+ import { SkeletonBillingPage } from '@nextsparkjs/core/components/ui/skeleton-settings'
2
+
3
+ export default function BillingLoading() {
4
+ return <SkeletonBillingPage />
5
+ }