@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,120 @@
1
+ /**
2
+ * DevTools Scheduled Actions API
3
+ *
4
+ * GET /api/v1/devtools/scheduled-actions
5
+ *
6
+ * Returns scheduled actions with filtering and pagination.
7
+ * Requires superadmin or developer user role.
8
+ */
9
+
10
+ import { NextRequest, NextResponse } from 'next/server'
11
+ import { authenticateRequest } from '@nextsparkjs/core/lib/api/auth/dual-auth'
12
+ import {
13
+ canAccessDevtoolsApi,
14
+ createDevtoolsAccessDeniedResponse,
15
+ createDevtoolsUnauthorizedResponse,
16
+ } from '@nextsparkjs/core/lib/api/auth/devtools-auth'
17
+ import { queryWithRLS } from '@nextsparkjs/core/lib/db'
18
+ import type { ScheduledAction, ScheduledActionStatus } from '@nextsparkjs/core/lib/scheduled-actions/types'
19
+ import { getAllRegisteredActions } from '@nextsparkjs/core/lib/scheduled-actions/registry'
20
+
21
+ export async function GET(request: NextRequest) {
22
+ // Authenticate request
23
+ const authResult = await authenticateRequest(request)
24
+
25
+ if (!authResult.success) {
26
+ return createDevtoolsUnauthorizedResponse()
27
+ }
28
+
29
+ // Check DevTools access permission
30
+ if (!canAccessDevtoolsApi(authResult)) {
31
+ return createDevtoolsAccessDeniedResponse()
32
+ }
33
+
34
+ // Extract query parameters
35
+ const searchParams = request.nextUrl.searchParams
36
+ const status = searchParams.get('status') as ScheduledActionStatus | null
37
+ const actionType = searchParams.get('action_type')
38
+ const page = parseInt(searchParams.get('page') || '1', 10)
39
+ const limit = parseInt(searchParams.get('limit') || '20', 10)
40
+
41
+ // Build query
42
+ const conditions: string[] = []
43
+ const params: unknown[] = []
44
+ let paramIndex = 1
45
+
46
+ if (status) {
47
+ conditions.push(`status = $${paramIndex++}`)
48
+ params.push(status)
49
+ }
50
+
51
+ if (actionType) {
52
+ conditions.push(`"actionType" = $${paramIndex++}`)
53
+ params.push(actionType)
54
+ }
55
+
56
+ const whereClause = conditions.length > 0 ? `WHERE ${conditions.join(' AND ')}` : ''
57
+
58
+ // Query total count
59
+ const countQuery = `
60
+ SELECT COUNT(*) as count
61
+ FROM "scheduled_actions"
62
+ ${whereClause}
63
+ `
64
+ const countResult = await queryWithRLS<{ count: string }>(countQuery, params, null)
65
+ const total = parseInt(countResult[0]?.count || '0', 10)
66
+
67
+ // Query actions with pagination
68
+ const offset = (page - 1) * limit
69
+ const actionsQuery = `
70
+ SELECT
71
+ id,
72
+ "actionType",
73
+ status,
74
+ payload,
75
+ "teamId",
76
+ "scheduledAt",
77
+ "startedAt",
78
+ "completedAt",
79
+ "errorMessage",
80
+ attempts,
81
+ "recurringInterval",
82
+ "createdAt",
83
+ "updatedAt"
84
+ FROM "scheduled_actions"
85
+ ${whereClause}
86
+ ORDER BY "createdAt" DESC
87
+ LIMIT $${paramIndex++}
88
+ OFFSET $${paramIndex}
89
+ `
90
+ const actionsResult = await queryWithRLS<ScheduledAction>(actionsQuery, [...params, limit, offset], null)
91
+
92
+ // Get all registered action types
93
+ const registeredActions = getAllRegisteredActions()
94
+
95
+ return NextResponse.json({
96
+ success: true,
97
+ data: {
98
+ actions: actionsResult,
99
+ pagination: {
100
+ total,
101
+ page,
102
+ limit,
103
+ totalPages: Math.ceil(total / limit),
104
+ },
105
+ meta: {
106
+ registeredActionTypes: registeredActions,
107
+ },
108
+ },
109
+ })
110
+ }
111
+
112
+ export async function OPTIONS() {
113
+ return new NextResponse(null, {
114
+ status: 204,
115
+ headers: {
116
+ 'Access-Control-Allow-Methods': 'GET, OPTIONS',
117
+ 'Access-Control-Allow-Headers': 'Content-Type, Authorization, x-api-key',
118
+ },
119
+ })
120
+ }
@@ -0,0 +1,82 @@
1
+ /**
2
+ * DevTools Testing API
3
+ *
4
+ * GET /api/v1/devtools/testing
5
+ *
6
+ * Returns the complete tags registry with test coverage statistics.
7
+ * Requires superadmin or developer user role.
8
+ */
9
+
10
+ import { NextRequest, NextResponse } from 'next/server'
11
+ import { authenticateRequest } from '@nextsparkjs/core/lib/api/auth/dual-auth'
12
+ import {
13
+ canAccessDevtoolsApi,
14
+ createDevtoolsAccessDeniedResponse,
15
+ createDevtoolsUnauthorizedResponse,
16
+ } from '@nextsparkjs/core/lib/api/auth/devtools-auth'
17
+ import {
18
+ TAGS_REGISTRY,
19
+ COVERAGE_SUMMARY,
20
+ FEATURE_REGISTRY,
21
+ FLOW_REGISTRY,
22
+ } from '@nextsparkjs/registries/testing-registry'
23
+
24
+ export async function GET(request: NextRequest) {
25
+ // Authenticate request
26
+ const authResult = await authenticateRequest(request)
27
+
28
+ if (!authResult.success) {
29
+ return createDevtoolsUnauthorizedResponse()
30
+ }
31
+
32
+ // Check DevTools access permission
33
+ if (!canAccessDevtoolsApi(authResult)) {
34
+ return createDevtoolsAccessDeniedResponse()
35
+ }
36
+
37
+ // Count total tags across all categories
38
+ let totalTags = 0
39
+ const categorySummary: Record<string, number> = {}
40
+
41
+ for (const [category, tags] of Object.entries(TAGS_REGISTRY)) {
42
+ const tagCount = Object.keys(tags).length
43
+ totalTags += tagCount
44
+ categorySummary[category] = tagCount
45
+ }
46
+
47
+ return NextResponse.json({
48
+ success: true,
49
+ data: {
50
+ tags: TAGS_REGISTRY,
51
+ summary: {
52
+ totalTags,
53
+ testFiles: COVERAGE_SUMMARY.tags.testFiles,
54
+ byCategory: categorySummary,
55
+ features: {
56
+ total: Object.keys(FEATURE_REGISTRY).length,
57
+ withTests: COVERAGE_SUMMARY.features.withTests,
58
+ withoutTests: COVERAGE_SUMMARY.features.withoutTests,
59
+ },
60
+ flows: {
61
+ total: Object.keys(FLOW_REGISTRY).length,
62
+ withTests: COVERAGE_SUMMARY.flows.withTests,
63
+ withoutTests: COVERAGE_SUMMARY.flows.withoutTests,
64
+ },
65
+ },
66
+ meta: {
67
+ theme: COVERAGE_SUMMARY.theme,
68
+ generatedAt: COVERAGE_SUMMARY.generatedAt,
69
+ },
70
+ },
71
+ })
72
+ }
73
+
74
+ export async function OPTIONS() {
75
+ return new NextResponse(null, {
76
+ status: 204,
77
+ headers: {
78
+ 'Access-Control-Allow-Methods': 'GET, OPTIONS',
79
+ 'Access-Control-Allow-Headers': 'Content-Type, Authorization, x-api-key',
80
+ },
81
+ })
82
+ }
@@ -0,0 +1,117 @@
1
+ # Media API
2
+
3
+ Upload and manage media files.
4
+
5
+ ## Overview
6
+
7
+ The Media API provides endpoints for uploading images and videos to Vercel Blob storage. All operations require authentication and appropriate scopes.
8
+
9
+ ## Authentication
10
+
11
+ All endpoints require authentication via:
12
+ - **Session cookie** (for browser-based requests)
13
+ - **API Key** header with `media:write` or `media:read` scope
14
+
15
+ ## Endpoints
16
+
17
+ ### Upload Files
18
+ `POST /api/v1/media/upload`
19
+
20
+ Upload one or multiple media files.
21
+
22
+ **Required Scope:** `media:write`
23
+
24
+ **Request:**
25
+ - Content-Type: `multipart/form-data`
26
+ - Field name: `files`
27
+
28
+ **Example (cURL):**
29
+ ```bash
30
+ curl -X POST /api/v1/media/upload \
31
+ -H "x-api-key: sk_live_xxx" \
32
+ -F "files=@image1.jpg" \
33
+ -F "files=@image2.png"
34
+ ```
35
+
36
+ **Success Response:**
37
+ ```json
38
+ {
39
+ "success": true,
40
+ "data": {
41
+ "message": "Files uploaded successfully",
42
+ "urls": [
43
+ "https://xxx.public.blob.vercel-storage.com/uploads/temp/123_abc.jpg",
44
+ "https://xxx.public.blob.vercel-storage.com/uploads/temp/124_def.png"
45
+ ],
46
+ "count": 2
47
+ }
48
+ }
49
+ ```
50
+
51
+ ### Get Upload Info
52
+ `GET /api/v1/media/upload`
53
+
54
+ Get information about the upload endpoint configuration.
55
+
56
+ **Required Scope:** `media:read`
57
+
58
+ **Response:**
59
+ ```json
60
+ {
61
+ "success": true,
62
+ "data": {
63
+ "message": "Media upload endpoint is active",
64
+ "storage": "Vercel Blob",
65
+ "uploadPath": "uploads/temp/",
66
+ "supportedTypes": ["image/jpeg", "image/png", "image/gif", "image/webp", "video/mp4", "video/webm"],
67
+ "maxFileSize": "10MB"
68
+ }
69
+ }
70
+ ```
71
+
72
+ ## Supported File Types
73
+
74
+ ### Images
75
+ - JPEG (`.jpg`, `.jpeg`)
76
+ - PNG (`.png`)
77
+ - GIF (`.gif`)
78
+ - WebP (`.webp`)
79
+
80
+ ### Videos
81
+ - MP4 (`.mp4`)
82
+ - MPEG (`.mpeg`)
83
+ - QuickTime (`.mov`)
84
+ - WebM (`.webm`)
85
+
86
+ ## Limitations
87
+
88
+ | Limit | Value |
89
+ |-------|-------|
90
+ | Maximum file size | 10 MB |
91
+ | Maximum files per request | No limit |
92
+ | Storage | Vercel Blob (public access) |
93
+
94
+ ## File Naming
95
+
96
+ Files are renamed on upload using the pattern:
97
+ ```
98
+ {timestamp}_{randomString}.{extension}
99
+ ```
100
+
101
+ Example: `1704816000000_a1b2c3d4e5f.jpg`
102
+
103
+ ## Error Responses
104
+
105
+ | Status | Description |
106
+ |--------|-------------|
107
+ | 400 | Bad Request - No files uploaded or invalid file type |
108
+ | 401 | Unauthorized - Authentication required |
109
+ | 403 | Forbidden - Insufficient permissions |
110
+ | 500 | Server Error - Upload to Vercel Blob failed |
111
+
112
+ ## Usage Notes
113
+
114
+ - Files are uploaded to `uploads/temp/` directory
115
+ - All uploaded files are publicly accessible
116
+ - Consider implementing cleanup for temporary files
117
+ - The API validates file types by MIME type, not just extension
@@ -0,0 +1,24 @@
1
+ /**
2
+ * API Presets for Media
3
+ *
4
+ * These presets appear in the DevTools API Explorer's "Presets" tab.
5
+ * Note: File upload presets cannot include actual files.
6
+ */
7
+
8
+ import { defineApiEndpoint } from '@nextsparkjs/core/types/api-presets'
9
+
10
+ export default defineApiEndpoint({
11
+ endpoint: '/api/v1/media',
12
+ summary: 'Upload and manage media files',
13
+ presets: [
14
+ // Get upload info
15
+ {
16
+ id: 'get-upload-info',
17
+ title: 'Get Upload Info',
18
+ description: 'Get media upload endpoint configuration',
19
+ method: 'GET',
20
+ path: '/upload',
21
+ tags: ['read', 'info']
22
+ }
23
+ ]
24
+ })
@@ -0,0 +1,150 @@
1
+ import { NextRequest } from 'next/server'
2
+ import { put } from '@vercel/blob'
3
+ import { authenticateRequest, hasRequiredScope } from '@nextsparkjs/core/lib/api/auth/dual-auth'
4
+ import { createApiResponse, createApiError } from '@nextsparkjs/core/lib/api/helpers'
5
+
6
+ export async function POST(request: NextRequest) {
7
+ try {
8
+ // 1. Dual Authentication (API Key OR Session)
9
+ const authResult = await authenticateRequest(request)
10
+
11
+ if (!authResult.success) {
12
+ return createApiError('Unauthorized', 401)
13
+ }
14
+
15
+ // 2. Check permissions for media upload
16
+ const hasPermission = hasRequiredScope(authResult, 'media:write')
17
+
18
+ if (!hasPermission) {
19
+ return createApiError('Insufficient permissions - media:write scope required', 403)
20
+ }
21
+
22
+ const formData = await request.formData()
23
+ const files = formData.getAll('files') as File[]
24
+
25
+ if (!files || files.length === 0) {
26
+ return createApiError('No files uploaded', 400)
27
+ }
28
+
29
+ const uploadedUrls: string[] = []
30
+
31
+ for (const file of files) {
32
+ if (!file.size) {
33
+ continue // Skip empty files
34
+ }
35
+
36
+ // Validate file type (images and videos)
37
+ const allowedTypes = [
38
+ 'image/jpeg',
39
+ 'image/jpg',
40
+ 'image/png',
41
+ 'image/gif',
42
+ 'image/webp',
43
+ 'video/mp4',
44
+ 'video/mpeg',
45
+ 'video/quicktime',
46
+ 'video/webm'
47
+ ]
48
+
49
+ if (!allowedTypes.includes(file.type)) {
50
+ return createApiError(
51
+ `File type ${file.type} not allowed. Only images and videos are supported.`,
52
+ 400,
53
+ { allowedTypes }
54
+ )
55
+ }
56
+
57
+ // Validate file size (max 10MB)
58
+ const maxSize = 10 * 1024 * 1024 // 10MB
59
+ if (file.size > maxSize) {
60
+ return createApiError(
61
+ `File ${file.name} is too large. Maximum size is 10MB.`,
62
+ 400,
63
+ { maxSize: '10MB', fileSize: `${(file.size / 1024 / 1024).toFixed(2)}MB` }
64
+ )
65
+ }
66
+
67
+ // Generate unique filename
68
+ const timestamp = Date.now()
69
+ const randomString = Math.random().toString(36).substring(2, 15)
70
+ const extension = file.name.split('.').pop()
71
+ const fileName = `${timestamp}_${randomString}.${extension}`
72
+
73
+ try {
74
+ // Upload to Vercel Blob
75
+ const blob = await put(`uploads/temp/${fileName}`, file, {
76
+ access: 'public',
77
+ addRandomSuffix: false
78
+ })
79
+
80
+ uploadedUrls.push(blob.url)
81
+ } catch (fileError) {
82
+ console.error(`❌ Failed to upload ${file.name} to Vercel Blob:`)
83
+ console.error(`❌ Error details:`, fileError)
84
+
85
+ const errorMessage = fileError instanceof Error ? fileError.message : String(fileError)
86
+ console.error(`❌ Error message:`, errorMessage)
87
+
88
+ if (fileError instanceof Error && fileError.stack) {
89
+ console.error(`❌ Error stack:`, fileError.stack)
90
+ }
91
+
92
+ return createApiError(
93
+ `Failed to upload file ${file.name}`,
94
+ 500,
95
+ { fileName: file.name, error: errorMessage }
96
+ )
97
+ }
98
+ }
99
+
100
+ return createApiResponse({
101
+ message: 'Files uploaded successfully',
102
+ urls: uploadedUrls,
103
+ count: uploadedUrls.length
104
+ })
105
+
106
+ } catch (error) {
107
+ console.error('Error uploading files:', error)
108
+ return createApiError(
109
+ 'Failed to upload files',
110
+ 500,
111
+ { error: error instanceof Error ? error.message : String(error) }
112
+ )
113
+ }
114
+ }
115
+
116
+ // Optional: Add a GET endpoint to get upload info
117
+ export async function GET(request: NextRequest) {
118
+ try {
119
+ // 1. Dual Authentication (API Key OR Session)
120
+ const authResult = await authenticateRequest(request)
121
+
122
+ if (!authResult.success) {
123
+ return createApiError('Unauthorized', 401)
124
+ }
125
+
126
+ // 2. Check permissions for media read
127
+ const hasPermission = hasRequiredScope(authResult, 'media:read')
128
+
129
+ if (!hasPermission) {
130
+ return createApiError('Insufficient permissions - media:read scope required', 403)
131
+ }
132
+
133
+ // This could be used for cleanup or management
134
+ return createApiResponse({
135
+ message: 'Media upload endpoint is active',
136
+ storage: 'Vercel Blob',
137
+ uploadPath: 'uploads/temp/',
138
+ supportedTypes: ['image/jpeg', 'image/png', 'image/gif', 'image/webp', 'video/mp4', 'video/webm'],
139
+ maxFileSize: '10MB'
140
+ })
141
+
142
+ } catch (error) {
143
+ console.error('Error in media upload GET:', error)
144
+ return createApiError(
145
+ 'Failed to get upload info',
146
+ 500,
147
+ { error: error instanceof Error ? error.message : String(error) }
148
+ )
149
+ }
150
+ }
@@ -0,0 +1,116 @@
1
+ /**
2
+ * Pattern Usages API
3
+ *
4
+ * GET /api/v1/patterns/[id]/usages - Get detailed usage information for a pattern
5
+ *
6
+ * Query Parameters:
7
+ * - entityType: Filter by entity type (e.g., 'pages', 'posts')
8
+ * - limit: Number of results per page (default: 50)
9
+ * - page: Page number (default: 1)
10
+ *
11
+ * Response includes:
12
+ * - usages: Array of usage records with entity info
13
+ * - counts: Array of counts by entity type
14
+ * - total: Total number of usages
15
+ */
16
+
17
+ import { NextRequest, NextResponse } from 'next/server'
18
+ import {
19
+ createApiResponse,
20
+ createApiError,
21
+ createPaginationMeta,
22
+ withApiLogging,
23
+ handleCorsPreflightRequest,
24
+ addCorsHeaders,
25
+ } from '@nextsparkjs/core/lib/api/helpers'
26
+ import { authenticateRequest, hasRequiredScope } from '@nextsparkjs/core/lib/api/auth/dual-auth'
27
+ import { PatternUsageService } from '@nextsparkjs/core/lib/services'
28
+ import { queryOneWithRLS } from '@nextsparkjs/core/lib/db'
29
+
30
+ // Handle CORS preflight
31
+ export async function OPTIONS() {
32
+ return handleCorsPreflightRequest()
33
+ }
34
+
35
+ // GET /api/v1/patterns/:id/usages - Get pattern usages
36
+ export const GET = withApiLogging(
37
+ async (req: NextRequest, { params }: { params: Promise<{ id: string }> }): Promise<NextResponse> => {
38
+ try {
39
+ // Authenticate using dual auth
40
+ const authResult = await authenticateRequest(req)
41
+
42
+ if (!authResult.success) {
43
+ return NextResponse.json(
44
+ { success: false, error: 'Authentication required', code: 'AUTHENTICATION_FAILED' },
45
+ { status: 401 }
46
+ )
47
+ }
48
+
49
+ if (authResult.rateLimitResponse) {
50
+ return authResult.rateLimitResponse as NextResponse
51
+ }
52
+
53
+ // Check required permissions
54
+ if (!hasRequiredScope(authResult, 'patterns:read')) {
55
+ const response = createApiError('Insufficient permissions', 403)
56
+ return addCorsHeaders(response)
57
+ }
58
+
59
+ const { id: patternId } = await params
60
+ const userId = authResult.user!.id
61
+
62
+ // Validate that patternId is not empty
63
+ if (!patternId || patternId.trim() === '') {
64
+ const response = createApiError('Pattern ID is required', 400, null, 'MISSING_PATTERN_ID')
65
+ return addCorsHeaders(response)
66
+ }
67
+
68
+ // Verify pattern exists and user has access
69
+ const pattern = await queryOneWithRLS<{ id: string }>(
70
+ 'SELECT id FROM "patterns" WHERE id = $1',
71
+ [patternId],
72
+ userId
73
+ )
74
+
75
+ if (!pattern) {
76
+ const response = createApiError('Pattern not found', 404, null, 'PATTERN_NOT_FOUND')
77
+ return addCorsHeaders(response)
78
+ }
79
+
80
+ // Parse query parameters
81
+ const url = new URL(req.url)
82
+ const entityType = url.searchParams.get('entityType') || undefined
83
+ const limit = Math.min(parseInt(url.searchParams.get('limit') || '50', 10), 100)
84
+ const page = Math.max(parseInt(url.searchParams.get('page') || '1', 10), 1)
85
+ const offset = (page - 1) * limit
86
+
87
+ // Get usages with entity info
88
+ const result = await PatternUsageService.getUsagesWithEntityInfo(
89
+ patternId,
90
+ userId,
91
+ { entityType, limit, offset }
92
+ )
93
+
94
+ // Create pagination meta
95
+ const totalPages = Math.ceil(result.total / limit)
96
+ const paginationMeta = createPaginationMeta(page, limit, result.total)
97
+
98
+ const response = createApiResponse({
99
+ usages: result.usages,
100
+ counts: result.counts,
101
+ total: result.total,
102
+ }, {
103
+ ...paginationMeta,
104
+ totalPages,
105
+ patternId,
106
+ entityType: entityType || 'all',
107
+ })
108
+
109
+ return addCorsHeaders(response)
110
+ } catch (error) {
111
+ console.error('Error fetching pattern usages:', error)
112
+ const response = createApiError('Internal server error', 500)
113
+ return addCorsHeaders(response)
114
+ }
115
+ }
116
+ )