opacacms 0.1.0

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 (399) hide show
  1. package/bun.lock +34 -0
  2. package/dist/admin/api-client.d.ts +8 -0
  3. package/dist/admin/auth-client.d.ts +940 -0
  4. package/dist/admin/custom-field.d.ts +71 -0
  5. package/dist/admin/index.d.ts +11 -0
  6. package/dist/admin/react.d.ts +3 -0
  7. package/dist/admin/router.d.ts +7 -0
  8. package/dist/admin/stores/admin-queries.d.ts +32 -0
  9. package/dist/admin/stores/auth.d.ts +33 -0
  10. package/dist/admin/stores/column-visibility.d.ts +21 -0
  11. package/dist/admin/stores/config.d.ts +7 -0
  12. package/dist/admin/stores/media.d.ts +44 -0
  13. package/dist/admin/stores/query.d.ts +4 -0
  14. package/dist/admin/stores/ui.d.ts +11 -0
  15. package/dist/admin/ui/admin-client.d.ts +7 -0
  16. package/dist/admin/ui/admin-layout.d.ts +14 -0
  17. package/dist/admin/ui/components/ColumnVisibilityToggle.d.ts +10 -0
  18. package/dist/admin/ui/components/DataDetailSheet.d.ts +13 -0
  19. package/dist/admin/ui/components/DataDetailView.d.ts +9 -0
  20. package/dist/admin/ui/components/Table.d.ts +10 -0
  21. package/dist/admin/ui/components/fields/ArrayField.d.ts +13 -0
  22. package/dist/admin/ui/components/fields/BlocksField.d.ts +17 -0
  23. package/dist/admin/ui/components/fields/BooleanField.d.ts +13 -0
  24. package/dist/admin/ui/components/fields/CollapsibleField.d.ts +16 -0
  25. package/dist/admin/ui/components/fields/DateField.d.ts +13 -0
  26. package/dist/admin/ui/components/fields/FileField.d.ts +23 -0
  27. package/dist/admin/ui/components/fields/GroupField.d.ts +13 -0
  28. package/dist/admin/ui/components/fields/JoinField.d.ts +15 -0
  29. package/dist/admin/ui/components/fields/NumberField.d.ts +14 -0
  30. package/dist/admin/ui/components/fields/RadioField.d.ts +17 -0
  31. package/dist/admin/ui/components/fields/RelationshipField.d.ts +16 -0
  32. package/dist/admin/ui/components/fields/RowField.d.ts +12 -0
  33. package/dist/admin/ui/components/fields/SelectField.d.ts +18 -0
  34. package/dist/admin/ui/components/fields/TabsField.d.ts +15 -0
  35. package/dist/admin/ui/components/fields/TextAreaField.d.ts +14 -0
  36. package/dist/admin/ui/components/fields/TextField.d.ts +14 -0
  37. package/dist/admin/ui/components/fields/VirtualField.d.ts +8 -0
  38. package/dist/admin/ui/components/fields/index.d.ts +28 -0
  39. package/dist/admin/ui/components/fields/richtext-editor/index.d.ts +10 -0
  40. package/dist/admin/ui/components/fields/richtext-editor/nodes/ImageComponent.d.ts +7 -0
  41. package/dist/admin/ui/components/fields/richtext-editor/nodes/ImageNode.d.ts +27 -0
  42. package/dist/admin/ui/components/fields/richtext-editor/plugins/ComponentPickerPlugin.d.ts +1 -0
  43. package/dist/admin/ui/components/fields/richtext-editor/plugins/EditableSyncPlugin.d.ts +5 -0
  44. package/dist/admin/ui/components/fields/richtext-editor/plugins/NotionToolbarPlugin.d.ts +1 -0
  45. package/dist/admin/ui/components/fields/richtext-editor/plugins/SimpleToolbarPlugin.d.ts +1 -0
  46. package/dist/admin/ui/components/fields/richtext-editor/plugins/ValueSyncPlugin.d.ts +5 -0
  47. package/dist/admin/ui/components/fields/utils.d.ts +1 -0
  48. package/dist/admin/ui/components/link.d.ts +8 -0
  49. package/dist/admin/ui/components/media/AssetManagerModal.d.ts +17 -0
  50. package/dist/admin/ui/components/toast.d.ts +10 -0
  51. package/dist/admin/ui/components/ui/accordion.d.ts +11 -0
  52. package/dist/admin/ui/components/ui/alert-dialog.d.ts +12 -0
  53. package/dist/admin/ui/components/ui/blocks.d.ts +5 -0
  54. package/dist/admin/ui/components/ui/breadcrumbs.d.ts +7 -0
  55. package/dist/admin/ui/components/ui/button.d.ts +7 -0
  56. package/dist/admin/ui/components/ui/collapsible.d.ts +16 -0
  57. package/dist/admin/ui/components/ui/dialog.d.ts +27 -0
  58. package/dist/admin/ui/components/ui/group.d.ts +6 -0
  59. package/dist/admin/ui/components/ui/index.d.ts +17 -0
  60. package/dist/admin/ui/components/ui/input.d.ts +5 -0
  61. package/dist/admin/ui/components/ui/join.d.ts +7 -0
  62. package/dist/admin/ui/components/ui/label.d.ts +3 -0
  63. package/dist/admin/ui/components/ui/radio-group.d.ts +13 -0
  64. package/dist/admin/ui/components/ui/relationship-detail-sheet.d.ts +9 -0
  65. package/dist/admin/ui/components/ui/relationship.d.ts +8 -0
  66. package/dist/admin/ui/components/ui/scroll-area.d.ts +7 -0
  67. package/dist/admin/ui/components/ui/select.d.ts +37 -0
  68. package/dist/admin/ui/components/ui/separator.d.ts +8 -0
  69. package/dist/admin/ui/components/ui/sheet.d.ts +28 -0
  70. package/dist/admin/ui/components/ui/tabs.d.ts +17 -0
  71. package/dist/admin/ui/components/ui/utils.d.ts +1 -0
  72. package/dist/admin/ui/hooks/use-debounce.d.ts +1 -0
  73. package/dist/admin/ui/views/collection-list-view.d.ts +5 -0
  74. package/dist/admin/ui/views/dashboard-view.d.ts +10 -0
  75. package/dist/admin/ui/views/document-edit-view.d.ts +7 -0
  76. package/dist/admin/ui/views/global-edit-view.d.ts +19 -0
  77. package/dist/admin/ui/views/init-view.d.ts +4 -0
  78. package/dist/admin/ui/views/login-view.d.ts +4 -0
  79. package/dist/admin/ui/views/media-registry-view.d.ts +7 -0
  80. package/dist/admin/ui/views/settings-view.d.ts +7 -0
  81. package/dist/admin/webcomponent.d.ts +1 -0
  82. package/dist/api.d.ts +6 -0
  83. package/dist/auth/index.d.ts +2107 -0
  84. package/dist/auth/migrations.d.ts +5 -0
  85. package/dist/auth/premissions.d.ts +6 -0
  86. package/dist/chunk-16vgcf3k.js +88 -0
  87. package/dist/chunk-2zm8cy1w.js +9482 -0
  88. package/dist/chunk-5gvbp2qa.js +167 -0
  89. package/dist/chunk-62ev8gnc.js +41 -0
  90. package/dist/chunk-6dhs73zq.js +126 -0
  91. package/dist/chunk-6ew02s0c.js +472 -0
  92. package/dist/chunk-7a9kn0np.js +116 -0
  93. package/dist/chunk-8gkhn1d4.js +309 -0
  94. package/dist/chunk-8sqjbsgt.js +42 -0
  95. package/dist/chunk-9kxpbcb1.js +85 -0
  96. package/dist/chunk-cvdd4eqh.js +110 -0
  97. package/dist/chunk-d3ffeqp9.js +87 -0
  98. package/dist/chunk-dy5t83hr.js +261 -0
  99. package/dist/chunk-f3nvxn63.js +17 -0
  100. package/dist/chunk-hmhcense.js +1352 -0
  101. package/dist/chunk-j4d50hrx.js +20 -0
  102. package/dist/chunk-jwjk85ze.js +15 -0
  103. package/dist/chunk-kwp83w8b.js +250 -0
  104. package/dist/chunk-s8mqwnm1.js +14 -0
  105. package/dist/chunk-srsac177.js +85 -0
  106. package/dist/chunk-v521d72w.js +10 -0
  107. package/dist/chunk-xa7rjsn2.js +20 -0
  108. package/dist/chunk-xg35h5a3.js +15 -0
  109. package/dist/chunk-ybbbqj63.js +130 -0
  110. package/dist/chunk-zvwb67nd.js +332 -0
  111. package/dist/cli/commands/generate-types.d.ts +1 -0
  112. package/dist/cli/commands/init.d.ts +1 -0
  113. package/dist/cli/commands/migrate-commands.d.ts +5 -0
  114. package/dist/cli/commands/seed-command.d.ts +2 -0
  115. package/dist/cli/d1-mock.d.ts +30 -0
  116. package/dist/cli/index.d.ts +5 -0
  117. package/dist/cli/index.test.d.ts +1 -0
  118. package/dist/cli/r2-mock.d.ts +46 -0
  119. package/dist/cli/seeding.d.ts +17 -0
  120. package/dist/client.d.ts +51 -0
  121. package/dist/config-utils.d.ts +6 -0
  122. package/dist/config.d.ts +10 -0
  123. package/dist/db/adapter.d.ts +34 -0
  124. package/dist/db/better-sqlite.d.ts +40 -0
  125. package/dist/db/bun-sqlite.d.ts +40 -0
  126. package/dist/db/d1.d.ts +42 -0
  127. package/dist/db/kysely/data-mapper.d.ts +6 -0
  128. package/dist/db/kysely/field-mapper.d.ts +22 -0
  129. package/dist/db/kysely/migration-generator.d.ts +9 -0
  130. package/dist/db/kysely/query-builder.d.ts +9 -0
  131. package/dist/db/kysely/schema-builder.d.ts +15 -0
  132. package/dist/db/kysely/sql-utils.d.ts +1 -0
  133. package/dist/db/postgres.d.ts +51 -0
  134. package/dist/db/sqlite.d.ts +41 -0
  135. package/dist/db/system-schema.d.ts +2 -0
  136. package/dist/index.d.ts +6 -0
  137. package/dist/runtimes/bun.d.ts +17 -0
  138. package/dist/runtimes/cloudflare-workers.d.ts +10 -0
  139. package/dist/runtimes/next.d.ts +16 -0
  140. package/dist/runtimes/node.d.ts +18 -0
  141. package/dist/schema/collection.d.ts +100 -0
  142. package/dist/schema/fields/base.d.ts +83 -0
  143. package/dist/schema/fields/index.d.ts +135 -0
  144. package/dist/schema/global.d.ts +82 -0
  145. package/dist/schema/index.d.ts +4 -0
  146. package/dist/schema/infer.d.ts +55 -0
  147. package/dist/server/admin-router.d.ts +9 -0
  148. package/dist/server/admin.d.ts +18 -0
  149. package/dist/server/assets.d.ts +47 -0
  150. package/dist/server/collection-router.d.ts +14 -0
  151. package/dist/server/handlers.d.ts +76 -0
  152. package/dist/server/middlewares/admin.d.ts +6 -0
  153. package/dist/server/middlewares/auth.d.ts +16 -0
  154. package/dist/server/middlewares/context.d.ts +9 -0
  155. package/dist/server/middlewares/cors.d.ts +3 -0
  156. package/dist/server/middlewares/database-init.d.ts +11 -0
  157. package/dist/server/middlewares/rate-limit.d.ts +3 -0
  158. package/dist/server/router.d.ts +7 -0
  159. package/dist/server/setup-middlewares.d.ts +17 -0
  160. package/dist/server/system-router.d.ts +9 -0
  161. package/dist/server.d.ts +6 -0
  162. package/dist/src/admin/index.css +47 -0
  163. package/dist/src/admin/index.js +176 -0
  164. package/dist/src/admin/webcomponent.js +19 -0
  165. package/dist/src/api.js +27 -0
  166. package/dist/src/cli/index.js +157 -0
  167. package/dist/src/client.js +9 -0
  168. package/dist/src/db/bun-sqlite.js +523 -0
  169. package/dist/src/db/d1.js +568 -0
  170. package/dist/src/db/postgres.js +520 -0
  171. package/dist/src/db/sqlite.js +534 -0
  172. package/dist/src/index.js +20 -0
  173. package/dist/src/runtimes/bun.js +36 -0
  174. package/dist/src/runtimes/cloudflare-workers.js +29 -0
  175. package/dist/src/runtimes/next.js +26 -0
  176. package/dist/src/runtimes/node.js +38 -0
  177. package/dist/src/server.js +27 -0
  178. package/dist/src/storage/index.js +355 -0
  179. package/dist/storage/adapters/cloudflare-r2.d.ts +6 -0
  180. package/dist/storage/adapters/local.d.ts +6 -0
  181. package/dist/storage/adapters/s3.d.ts +13 -0
  182. package/dist/storage/errors.d.ts +12 -0
  183. package/dist/storage/index.d.ts +5 -0
  184. package/dist/storage/types.d.ts +31 -0
  185. package/dist/types.d.ts +484 -0
  186. package/dist/utils/lexical.d.ts +5 -0
  187. package/dist/utils/logger.d.ts +35 -0
  188. package/dist/validation.d.ts +300 -0
  189. package/dist/validator.d.ts +9 -0
  190. package/global.d.ts +11 -0
  191. package/package.json +151 -0
  192. package/src/admin/api-client.ts +63 -0
  193. package/src/admin/auth-client.ts +40 -0
  194. package/src/admin/custom-field.ts +179 -0
  195. package/src/admin/index.ts +15 -0
  196. package/src/admin/react.tsx +72 -0
  197. package/src/admin/router.ts +9 -0
  198. package/src/admin/stores/admin-queries.ts +121 -0
  199. package/src/admin/stores/auth.ts +61 -0
  200. package/src/admin/stores/column-visibility.ts +67 -0
  201. package/src/admin/stores/config.ts +15 -0
  202. package/src/admin/stores/media.ts +95 -0
  203. package/src/admin/stores/query.ts +13 -0
  204. package/src/admin/stores/ui.ts +29 -0
  205. package/src/admin/ui/admin-client.tsx +283 -0
  206. package/src/admin/ui/admin-layout.tsx +276 -0
  207. package/src/admin/ui/components/ColumnVisibilityToggle.tsx +141 -0
  208. package/src/admin/ui/components/DataDetailSheet.tsx +141 -0
  209. package/src/admin/ui/components/DataDetailView.tsx +175 -0
  210. package/src/admin/ui/components/Table.tsx +67 -0
  211. package/src/admin/ui/components/fields/ArrayField.tsx +166 -0
  212. package/src/admin/ui/components/fields/BlocksField.tsx +202 -0
  213. package/src/admin/ui/components/fields/BooleanField.tsx +50 -0
  214. package/src/admin/ui/components/fields/CollapsibleField.tsx +75 -0
  215. package/src/admin/ui/components/fields/DateField.tsx +45 -0
  216. package/src/admin/ui/components/fields/FileField.tsx +322 -0
  217. package/src/admin/ui/components/fields/GroupField.tsx +50 -0
  218. package/src/admin/ui/components/fields/JoinField.tsx +23 -0
  219. package/src/admin/ui/components/fields/NumberField.tsx +46 -0
  220. package/src/admin/ui/components/fields/RadioField.tsx +62 -0
  221. package/src/admin/ui/components/fields/RelationshipField.tsx +278 -0
  222. package/src/admin/ui/components/fields/RowField.tsx +40 -0
  223. package/src/admin/ui/components/fields/SelectField.tsx +59 -0
  224. package/src/admin/ui/components/fields/TabsField.tsx +101 -0
  225. package/src/admin/ui/components/fields/TextAreaField.tsx +54 -0
  226. package/src/admin/ui/components/fields/TextField.tsx +49 -0
  227. package/src/admin/ui/components/fields/VirtualField.tsx +53 -0
  228. package/src/admin/ui/components/fields/index.tsx +371 -0
  229. package/src/admin/ui/components/fields/richtext-editor/index.tsx +211 -0
  230. package/src/admin/ui/components/fields/richtext-editor/nodes/ImageComponent.tsx +142 -0
  231. package/src/admin/ui/components/fields/richtext-editor/nodes/ImageNode.tsx +95 -0
  232. package/src/admin/ui/components/fields/richtext-editor/plugins/ComponentPickerPlugin.tsx +226 -0
  233. package/src/admin/ui/components/fields/richtext-editor/plugins/EditableSyncPlugin.tsx +16 -0
  234. package/src/admin/ui/components/fields/richtext-editor/plugins/NotionToolbarPlugin.tsx +184 -0
  235. package/src/admin/ui/components/fields/richtext-editor/plugins/SimpleToolbarPlugin.tsx +240 -0
  236. package/src/admin/ui/components/fields/richtext-editor/plugins/ValueSyncPlugin.tsx +40 -0
  237. package/src/admin/ui/components/fields/utils.ts +1 -0
  238. package/src/admin/ui/components/link.tsx +41 -0
  239. package/src/admin/ui/components/media/AssetManagerModal.tsx +334 -0
  240. package/src/admin/ui/components/toast.tsx +72 -0
  241. package/src/admin/ui/components/ui/accordion.tsx +51 -0
  242. package/src/admin/ui/components/ui/alert-dialog.tsx +98 -0
  243. package/src/admin/ui/components/ui/blocks.tsx +32 -0
  244. package/src/admin/ui/components/ui/breadcrumbs.tsx +59 -0
  245. package/src/admin/ui/components/ui/button.tsx +26 -0
  246. package/src/admin/ui/components/ui/collapsible.tsx +124 -0
  247. package/src/admin/ui/components/ui/dialog.tsx +79 -0
  248. package/src/admin/ui/components/ui/group.tsx +20 -0
  249. package/src/admin/ui/components/ui/index.ts +17 -0
  250. package/src/admin/ui/components/ui/input.tsx +12 -0
  251. package/src/admin/ui/components/ui/join.tsx +53 -0
  252. package/src/admin/ui/components/ui/label.tsx +11 -0
  253. package/src/admin/ui/components/ui/radio-group.tsx +75 -0
  254. package/src/admin/ui/components/ui/relationship-detail-sheet.tsx +122 -0
  255. package/src/admin/ui/components/ui/relationship.tsx +58 -0
  256. package/src/admin/ui/components/ui/scroll-area.tsx +19 -0
  257. package/src/admin/ui/components/ui/select.tsx +187 -0
  258. package/src/admin/ui/components/ui/separator.tsx +21 -0
  259. package/src/admin/ui/components/ui/sheet.tsx +106 -0
  260. package/src/admin/ui/components/ui/tabs.tsx +116 -0
  261. package/src/admin/ui/components/ui/utils.ts +3 -0
  262. package/src/admin/ui/hooks/use-debounce.ts +15 -0
  263. package/src/admin/ui/styles/_locale-switcher.scss +33 -0
  264. package/src/admin/ui/styles/accordion.scss +60 -0
  265. package/src/admin/ui/styles/animations.scss +41 -0
  266. package/src/admin/ui/styles/asset-manager.scss +547 -0
  267. package/src/admin/ui/styles/badge.scss +13 -0
  268. package/src/admin/ui/styles/base.scss +22 -0
  269. package/src/admin/ui/styles/button.scss +161 -0
  270. package/src/admin/ui/styles/card.scss +13 -0
  271. package/src/admin/ui/styles/collapsible.scss +75 -0
  272. package/src/admin/ui/styles/data-detail.scss +92 -0
  273. package/src/admin/ui/styles/dialog.scss +102 -0
  274. package/src/admin/ui/styles/empty-state.scss +22 -0
  275. package/src/admin/ui/styles/group.scss +19 -0
  276. package/src/admin/ui/styles/index.scss +33 -0
  277. package/src/admin/ui/styles/input.scss +80 -0
  278. package/src/admin/ui/styles/label.scss +12 -0
  279. package/src/admin/ui/styles/layout.scss +56 -0
  280. package/src/admin/ui/styles/lexical.scss +469 -0
  281. package/src/admin/ui/styles/loading.scss +102 -0
  282. package/src/admin/ui/styles/media-registry.scss +597 -0
  283. package/src/admin/ui/styles/pagination.scss +20 -0
  284. package/src/admin/ui/styles/radio-group.scss +66 -0
  285. package/src/admin/ui/styles/row.scss +17 -0
  286. package/src/admin/ui/styles/scrollbar.scss +36 -0
  287. package/src/admin/ui/styles/select.scss +121 -0
  288. package/src/admin/ui/styles/separator.scss +14 -0
  289. package/src/admin/ui/styles/sheet.scss +152 -0
  290. package/src/admin/ui/styles/sidebar.scss +148 -0
  291. package/src/admin/ui/styles/switch.scss +59 -0
  292. package/src/admin/ui/styles/table.scss +207 -0
  293. package/src/admin/ui/styles/tabs.scss +62 -0
  294. package/src/admin/ui/styles/toast.scss +45 -0
  295. package/src/admin/ui/styles/variables.scss +24 -0
  296. package/src/admin/ui/views/collection-list-view.tsx +720 -0
  297. package/src/admin/ui/views/dashboard-view.tsx +263 -0
  298. package/src/admin/ui/views/document-edit-view.tsx +384 -0
  299. package/src/admin/ui/views/global-edit-view.tsx +226 -0
  300. package/src/admin/ui/views/init-view.tsx +182 -0
  301. package/src/admin/ui/views/login-view.tsx +123 -0
  302. package/src/admin/ui/views/media-registry-view.tsx +1104 -0
  303. package/src/admin/ui/views/settings-view.tsx +729 -0
  304. package/src/admin/webcomponent.tsx +15 -0
  305. package/src/api.ts +9 -0
  306. package/src/auth/index.ts +194 -0
  307. package/src/auth/migrations.ts +87 -0
  308. package/src/auth/premissions.ts +46 -0
  309. package/src/cli/commands/generate-types.ts +116 -0
  310. package/src/cli/commands/init.ts +95 -0
  311. package/src/cli/commands/migrate-commands.ts +160 -0
  312. package/src/cli/commands/seed-command.ts +11 -0
  313. package/src/cli/d1-mock.ts +101 -0
  314. package/src/cli/index.test.ts +84 -0
  315. package/src/cli/index.ts +183 -0
  316. package/src/cli/r2-mock.ts +217 -0
  317. package/src/cli/seeding.ts +405 -0
  318. package/src/client.ts +181 -0
  319. package/src/config-utils.ts +102 -0
  320. package/src/config.ts +49 -0
  321. package/src/db/adapter.ts +53 -0
  322. package/src/db/better-sqlite.ts +630 -0
  323. package/src/db/bun-sqlite.ts +646 -0
  324. package/src/db/d1.ts +711 -0
  325. package/src/db/kysely/data-mapper.ts +142 -0
  326. package/src/db/kysely/field-mapper.ts +148 -0
  327. package/src/db/kysely/migration-generator.ts +223 -0
  328. package/src/db/kysely/query-builder.ts +92 -0
  329. package/src/db/kysely/schema-builder.ts +439 -0
  330. package/src/db/kysely/sql-utils.ts +13 -0
  331. package/src/db/postgres.ts +621 -0
  332. package/src/db/sqlite.ts +658 -0
  333. package/src/db/system-schema.ts +121 -0
  334. package/src/index.ts +13 -0
  335. package/src/runtimes/README.md +59 -0
  336. package/src/runtimes/bun.ts +49 -0
  337. package/src/runtimes/cloudflare-workers.ts +38 -0
  338. package/src/runtimes/next.ts +26 -0
  339. package/src/runtimes/node.ts +52 -0
  340. package/src/schema/collection.ts +184 -0
  341. package/src/schema/fields/base.ts +164 -0
  342. package/src/schema/fields/index.ts +427 -0
  343. package/src/schema/global.ts +145 -0
  344. package/src/schema/index.ts +4 -0
  345. package/src/schema/infer.ts +72 -0
  346. package/src/server/admin-router.ts +20 -0
  347. package/src/server/admin.ts +142 -0
  348. package/src/server/assets.ts +306 -0
  349. package/src/server/collection-router.ts +55 -0
  350. package/src/server/handlers.ts +722 -0
  351. package/src/server/middlewares/admin.ts +27 -0
  352. package/src/server/middlewares/auth.ts +89 -0
  353. package/src/server/middlewares/context.ts +17 -0
  354. package/src/server/middlewares/cors.ts +24 -0
  355. package/src/server/middlewares/database-init.ts +74 -0
  356. package/src/server/middlewares/rate-limit.ts +71 -0
  357. package/src/server/router.ts +47 -0
  358. package/src/server/setup-middlewares.ts +58 -0
  359. package/src/server/system-router.ts +35 -0
  360. package/src/server.ts +9 -0
  361. package/src/storage/adapters/cloudflare-r2.ts +136 -0
  362. package/src/storage/adapters/local.ts +146 -0
  363. package/src/storage/adapters/s3.ts +186 -0
  364. package/src/storage/errors.ts +46 -0
  365. package/src/storage/index.ts +5 -0
  366. package/src/storage/types.ts +39 -0
  367. package/src/types.ts +577 -0
  368. package/src/utils/lexical.ts +37 -0
  369. package/src/utils/logger.ts +73 -0
  370. package/src/validation.ts +429 -0
  371. package/src/validator.ts +179 -0
  372. package/test/admin-custom-field.test.ts +162 -0
  373. package/test/admin-react-field.test.tsx +134 -0
  374. package/test/api-features.test.ts +78 -0
  375. package/test/api.test.ts +178 -0
  376. package/test/auth.test.ts +62 -0
  377. package/test/cli-integration.test.ts +146 -0
  378. package/test/cli.test.ts +25 -0
  379. package/test/db/postgres.test.ts +95 -0
  380. package/test/db/sqlite-filter.test.ts +53 -0
  381. package/test/db/sqlite.test.ts +82 -0
  382. package/test/engine-features.test.ts +79 -0
  383. package/test/globals.test.ts +74 -0
  384. package/test/integration-tmp/db-app/opacacms.config.ts +15 -0
  385. package/test/integration-tmp/my-sqlite-app/opacacms.config.ts +25 -0
  386. package/test/integration-tmp/my-test-app/index.ts +8 -0
  387. package/test/integration-tmp/my-test-app/opacacms.config.ts +16 -0
  388. package/test/integration-tmp/my-test-app/package.json +12 -0
  389. package/test/populate.test.ts +79 -0
  390. package/test/runtimes.test.ts +43 -0
  391. package/test/schema-builder.test.ts +107 -0
  392. package/test/schema-features.test.ts +63 -0
  393. package/test/seeding.test.ts +68 -0
  394. package/test/storage/local.test.ts +72 -0
  395. package/test/storage/s3.test.ts +60 -0
  396. package/test/structural-data.test.ts +100 -0
  397. package/test/test-setup.ts +11 -0
  398. package/test/validation.test.ts +162 -0
  399. package/tsconfig.json +42 -0
@@ -0,0 +1,309 @@
1
+ import {
2
+ __require
3
+ } from "./chunk-8sqjbsgt.js";
4
+
5
+ // src/cli/seeding.ts
6
+ import { faker } from "@faker-js/faker";
7
+ var defaultFieldGenerators = {
8
+ text: () => faker.lorem.words(3),
9
+ textarea: () => faker.lorem.paragraph(),
10
+ number: () => faker.number.int({ min: 1, max: 1000 }),
11
+ richtext: () => JSON.stringify({
12
+ root: {
13
+ children: [
14
+ {
15
+ children: [
16
+ {
17
+ detail: 0,
18
+ format: 0,
19
+ mode: "normal",
20
+ style: "",
21
+ text: faker.lorem.sentence(),
22
+ type: "text",
23
+ version: 1
24
+ }
25
+ ],
26
+ direction: "ltr",
27
+ format: "",
28
+ indent: 0,
29
+ type: "heading",
30
+ tag: "h1",
31
+ version: 1
32
+ },
33
+ {
34
+ children: [
35
+ {
36
+ detail: 0,
37
+ format: 0,
38
+ mode: "normal",
39
+ style: "",
40
+ text: faker.lorem.paragraphs(2),
41
+ type: "text",
42
+ version: 1
43
+ }
44
+ ],
45
+ direction: "ltr",
46
+ format: "",
47
+ indent: 0,
48
+ type: "paragraph",
49
+ version: 1
50
+ }
51
+ ],
52
+ direction: "ltr",
53
+ format: "",
54
+ indent: 0,
55
+ type: "root",
56
+ version: 1
57
+ }
58
+ }),
59
+ boolean: () => faker.datatype.boolean(),
60
+ date: () => faker.date.recent().toISOString(),
61
+ email: () => faker.internet.email(),
62
+ json: () => ({ [faker.lorem.word()]: faker.lorem.sentence() }),
63
+ select: () => faker.lorem.word(),
64
+ radio: () => faker.lorem.word(),
65
+ blocks: () => []
66
+ };
67
+ async function getRandomIds(db, relationTo, count = 1) {
68
+ try {
69
+ const result = await db.find(relationTo, {}, { limit: 50 });
70
+ const docs = result.docs || [];
71
+ if (docs.length === 0) {
72
+ if (!relationTo.startsWith("_")) {
73
+ console.warn(`[Seeding] No documents found in '${relationTo}' for relationship.`);
74
+ }
75
+ return [];
76
+ }
77
+ const shuffled = [...docs].sort(() => 0.5 - Math.random());
78
+ const selected = shuffled.slice(0, count).map((doc) => doc.id);
79
+ return selected;
80
+ } catch (e) {
81
+ console.error(`[Seeding] Failed to fetch random IDs for ${relationTo}:`, e);
82
+ return [];
83
+ }
84
+ }
85
+ async function generateDataForFields(fields, db, locales = []) {
86
+ const record = {};
87
+ for (const field of fields) {
88
+ if (field.type === "row" || field.type === "collapsible") {
89
+ Object.assign(record, await generateDataForFields(field.fields, db, locales));
90
+ continue;
91
+ }
92
+ if (field.type === "tabs") {
93
+ for (const tab of field.tabs) {
94
+ Object.assign(record, await generateDataForFields(tab.fields, db, locales));
95
+ }
96
+ continue;
97
+ }
98
+ if (field.type === "group" && field.name) {
99
+ record[field.name] = await generateDataForFields(field.fields, db, locales);
100
+ continue;
101
+ }
102
+ if (field.name) {
103
+ const isLocalized = !!field.localized && locales.length > 0;
104
+ const generateFieldValue = async () => {
105
+ if (field.type === "relationship" && "relationTo" in field) {
106
+ if (field.hasMany) {
107
+ const count = faker.number.int({ min: 1, max: 3 });
108
+ return await getRandomIds(db, field.relationTo, count);
109
+ }
110
+ const ids = await getRandomIds(db, field.relationTo, 1);
111
+ return ids[0] || null;
112
+ }
113
+ if (field.type === "blocks" && field.blocks) {
114
+ const blockCount = faker.number.int({ min: 1, max: 3 });
115
+ const generatedBlocks = [];
116
+ for (let i = 0;i < blockCount; i++) {
117
+ const blockType = field.blocks[faker.number.int({ min: 0, max: field.blocks.length - 1 })];
118
+ if (!blockType)
119
+ continue;
120
+ const blockData = await generateDataForFields(blockType.fields, db, locales);
121
+ generatedBlocks.push({
122
+ ...blockData,
123
+ blockType: blockType.slug,
124
+ id: faker.string.uuid()
125
+ });
126
+ }
127
+ return generatedBlocks;
128
+ }
129
+ const generator = defaultFieldGenerators[field.type];
130
+ if (generator) {
131
+ if (field.type === "select" || field.type === "radio") {
132
+ const options = field.options;
133
+ const choices = options?.choices || [];
134
+ if (choices.length > 0) {
135
+ const choice = choices[Math.floor(Math.random() * choices.length)];
136
+ return typeof choice === "string" ? choice : choice.value;
137
+ }
138
+ }
139
+ return generator();
140
+ }
141
+ return null;
142
+ };
143
+ if (isLocalized) {
144
+ const localizedValue = {};
145
+ for (const locale of locales) {
146
+ localizedValue[locale] = await generateFieldValue();
147
+ }
148
+ record[field.name] = localizedValue;
149
+ } else {
150
+ record[field.name] = await generateFieldValue();
151
+ }
152
+ }
153
+ }
154
+ return record;
155
+ }
156
+ async function generateRecord(db, collection, locales = []) {
157
+ return generateDataForFields(collection.fields, db, locales);
158
+ }
159
+ function sortCollections(collections) {
160
+ const sorted = [];
161
+ const visited = new Set;
162
+ const visiting = new Set;
163
+ const visit = (collection) => {
164
+ if (visited.has(collection.slug))
165
+ return;
166
+ if (visiting.has(collection.slug)) {
167
+ throw new Error(`Circular dependency detected: ${collection.slug}`);
168
+ }
169
+ visiting.add(collection.slug);
170
+ const deps = [];
171
+ const findDeps = (fields) => {
172
+ for (const f of fields) {
173
+ if (f.type === "relationship") {
174
+ deps.push(f.relationTo);
175
+ } else if (f.type === "blocks" && f.blocks) {
176
+ f.blocks.forEach((b) => findDeps(b.fields));
177
+ } else if ("fields" in f && f.fields) {
178
+ findDeps(f.fields);
179
+ } else if ("tabs" in f && f.tabs) {
180
+ f.tabs.forEach((t) => findDeps(t.fields));
181
+ }
182
+ }
183
+ };
184
+ findDeps(collection.fields);
185
+ for (const depSlug of deps) {
186
+ const depColl = collections.find((c) => c.slug === depSlug);
187
+ if (depColl)
188
+ visit(depColl);
189
+ }
190
+ visiting.delete(collection.slug);
191
+ visited.add(collection.slug);
192
+ sorted.push(collection);
193
+ };
194
+ for (const collection of collections) {
195
+ visit(collection);
196
+ }
197
+ return sorted;
198
+ }
199
+ async function autoSeed(config, countPerCollection = 10, reset = false, type = "all") {
200
+ const { collections, db, globals, storages, serverURL } = config;
201
+ console.log(`\uD83C\uDF31 Starting automatic seed (${countPerCollection} records per collection)...`);
202
+ const { getSystemCollections } = await import("./chunk-v521d72w.js");
203
+ const systemCollections = getSystemCollections().filter((c) => c.slug === "_opaca_assets");
204
+ const allCollections = [...systemCollections, ...collections];
205
+ let collectionsToSeed = sortCollections(allCollections);
206
+ if (type === "assets") {
207
+ collectionsToSeed = collectionsToSeed.filter((c) => c.slug === "_opaca_assets");
208
+ console.log("\uD83D\uDCC1 Seeding only assets...");
209
+ } else if (type === "collections") {
210
+ collectionsToSeed = collectionsToSeed.filter((c) => c.slug !== "_opaca_assets");
211
+ console.log("\uD83D\uDCDA Seeding only user collections...");
212
+ }
213
+ await db.connect();
214
+ await db.migrate(allCollections, globals || []);
215
+ try {
216
+ if (reset) {
217
+ console.log("\uD83E\uDDF9 Resetting data (deleting existing records)...");
218
+ const reversed = [...collectionsToSeed].reverse();
219
+ for (const collection of reversed) {
220
+ console.log(`Cleaning ${collection.slug}...`);
221
+ await db.deleteMany?.(collection.slug, {});
222
+ }
223
+ }
224
+ const storageAdapter = storages?.default || storages;
225
+ const locales = config.i18n?.locales || [];
226
+ for (const collection of collectionsToSeed) {
227
+ console.log(`Seeding ${collection.slug}...`);
228
+ const isAssetCollection = collection.slug === "_opaca_assets";
229
+ for (let i = 0;i < countPerCollection; i++) {
230
+ let data;
231
+ if (isAssetCollection) {
232
+ const id = faker.string.uuid();
233
+ const width = faker.number.int({ min: 400, max: 1200 });
234
+ const height = faker.number.int({ min: 300, max: 800 });
235
+ const color = faker.color.rgb({ prefix: "" });
236
+ const textColor = faker.color.rgb({ prefix: "" });
237
+ const usePicsum = Math.random() > 0.5;
238
+ let imageUrl = "";
239
+ if (usePicsum) {
240
+ const categories = ["nature", "city", "tech", "people", "animals", "architecture"];
241
+ const category = faker.helpers.arrayElement(categories);
242
+ imageUrl = `https://picsum.photos/seed/${category}-${id}/${width}/${height}`;
243
+ } else {
244
+ imageUrl = `https://placehold.co/${width}x${height}/${color}/${textColor}.png?text=Seed+${i}`;
245
+ }
246
+ const res = await fetch(imageUrl);
247
+ if (!res.ok) {
248
+ throw new Error(`Failed to fetch placeholder image: ${imageUrl}`);
249
+ }
250
+ const arrayBuffer = await res.arrayBuffer();
251
+ const mime_type = res.headers.get("content-type")?.split(";")[0] || "image/png";
252
+ const extMap = {
253
+ "image/jpeg": "jpg",
254
+ "image/png": "png",
255
+ "image/webp": "webp",
256
+ "image/gif": "gif",
257
+ "image/svg+xml": "svg"
258
+ };
259
+ const ext = extMap[mime_type] || "png";
260
+ const fileRecord = {
261
+ filename: `seed-image-${i}.${ext}`,
262
+ original_filename: `seed-image-${i}.${ext}`,
263
+ mime_type,
264
+ filesize: arrayBuffer.byteLength,
265
+ buffer: new Uint8Array(arrayBuffer)
266
+ };
267
+ if (!storageAdapter || typeof storageAdapter.upload !== "function") {
268
+ throw new Error("Storage adapter is required for seeding assets and must have an 'upload' method.");
269
+ }
270
+ const uploaded = await storageAdapter.upload(fileRecord, {
271
+ generateUniqueName: true,
272
+ customMetadata: {
273
+ sourceUrl: imageUrl
274
+ }
275
+ });
276
+ data = {
277
+ id,
278
+ key: uploaded.filename,
279
+ filename: uploaded.filename,
280
+ originalFilename: fileRecord.original_filename,
281
+ mimeType: uploaded.mime_type,
282
+ filesize: uploaded.filesize,
283
+ width,
284
+ height,
285
+ bucket: "default",
286
+ url: uploaded.url,
287
+ thumbnailUrl: uploaded.url,
288
+ altText: faker.lorem.sentence()
289
+ };
290
+ const baseURL = serverURL || "http://localhost:8787";
291
+ console.log(`[Asset] Source: ${imageUrl}`);
292
+ console.log(`[Asset] Seeded: ${baseURL}/api/assets/${id}/view (${uploaded.filename})`);
293
+ } else {
294
+ data = await generateRecord(db, collection, locales);
295
+ }
296
+ await db.create(collection.slug, data);
297
+ }
298
+ }
299
+ console.log("✅ Seeding completed.");
300
+ } finally {
301
+ await db.disconnect();
302
+ }
303
+ }
304
+ export {
305
+ sortCollections,
306
+ generateRecord,
307
+ defaultFieldGenerators,
308
+ autoSeed
309
+ };
@@ -0,0 +1,42 @@
1
+ import { createRequire } from "node:module";
2
+ var __defProp = Object.defineProperty;
3
+ var __getOwnPropNames = Object.getOwnPropertyNames;
4
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
5
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
6
+ function __accessProp(key) {
7
+ return this[key];
8
+ }
9
+ var __toCommonJS = (from) => {
10
+ var entry = (__moduleCache ??= new WeakMap).get(from), desc;
11
+ if (entry)
12
+ return entry;
13
+ entry = __defProp({}, "__esModule", { value: true });
14
+ if (from && typeof from === "object" || typeof from === "function") {
15
+ for (var key of __getOwnPropNames(from))
16
+ if (!__hasOwnProp.call(entry, key))
17
+ __defProp(entry, key, {
18
+ get: __accessProp.bind(from, key),
19
+ enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable
20
+ });
21
+ }
22
+ __moduleCache.set(from, entry);
23
+ return entry;
24
+ };
25
+ var __moduleCache;
26
+ var __returnValue = (v) => v;
27
+ function __exportSetter(name, newValue) {
28
+ this[name] = __returnValue.bind(null, newValue);
29
+ }
30
+ var __export = (target, all) => {
31
+ for (var name in all)
32
+ __defProp(target, name, {
33
+ get: all[name],
34
+ enumerable: true,
35
+ configurable: true,
36
+ set: __exportSetter.bind(all, name)
37
+ });
38
+ };
39
+ var __esm = (fn, res) => () => (fn && (res = fn(fn = 0)), res);
40
+ var __require = /* @__PURE__ */ createRequire(import.meta.url);
41
+
42
+ export { __toCommonJS, __export, __esm, __require };
@@ -0,0 +1,85 @@
1
+ import"./chunk-8sqjbsgt.js";
2
+
3
+ // src/cli/d1-mock.ts
4
+ import { Database } from "bun:sqlite";
5
+ import fs from "node:fs";
6
+ import path from "node:path";
7
+ function createD1Mock(dbPath = ".opaca/local-d1.db") {
8
+ const wranglerD1Dir = path.resolve(process.cwd(), ".wrangler/state/v3/d1/miniflare-D1DatabaseObject");
9
+ if (fs.existsSync(wranglerD1Dir)) {
10
+ const files = fs.readdirSync(wranglerD1Dir);
11
+ const sqliteFile = files.find((f) => f.endsWith(".sqlite"));
12
+ if (sqliteFile) {
13
+ const wranglerPath = path.join(wranglerD1Dir, sqliteFile);
14
+ try {
15
+ const sqlite = new Database(wranglerPath);
16
+ console.log(`[OpacaCMS] Using Wrangler D1 local state: ${sqliteFile}`);
17
+ return wrapSqlite(sqlite);
18
+ } catch (e) {
19
+ console.warn(`[OpacaCMS] Failed to open Wrangler D1 state at ${wranglerPath}. Falling back...`);
20
+ }
21
+ }
22
+ }
23
+ const absolutePath = path.isAbsolute(dbPath) ? dbPath : path.resolve(process.cwd(), dbPath);
24
+ const dir = path.dirname(absolutePath);
25
+ try {
26
+ if (!fs.existsSync(dir)) {
27
+ fs.mkdirSync(dir, { recursive: true });
28
+ }
29
+ const sqlite = new Database(absolutePath);
30
+ console.log(`[OpacaCMS] Using local CLI database: ${absolutePath}`);
31
+ return wrapSqlite(sqlite);
32
+ } catch (err) {
33
+ console.warn(`[OpacaCMS] Warning: Could not open database at ${absolutePath}. Using memory database.`);
34
+ return wrapSqlite(new Database(":memory:"));
35
+ }
36
+ }
37
+ function wrapSqlite(sqlite) {
38
+ return {
39
+ prepare(query) {
40
+ const stmt = sqlite.prepare(query);
41
+ const bridge = {
42
+ bind(...params) {
43
+ bridge._params = params;
44
+ return bridge;
45
+ },
46
+ async first() {
47
+ const params = bridge._params || [];
48
+ return stmt.get(...params);
49
+ },
50
+ async all() {
51
+ const params = bridge._params || [];
52
+ const results = stmt.all(...params);
53
+ return { results, success: true, meta: {} };
54
+ },
55
+ async run() {
56
+ const params = bridge._params || [];
57
+ const result = stmt.run(...params);
58
+ return {
59
+ success: true,
60
+ meta: { changes: result.changes, last_row_id: result.lastInsertRowid }
61
+ };
62
+ },
63
+ async raw() {
64
+ const params = bridge._params || [];
65
+ return stmt.all(...params);
66
+ }
67
+ };
68
+ return bridge;
69
+ },
70
+ async batch(statements) {
71
+ const results = [];
72
+ for (const stmt of statements) {
73
+ results.push(await stmt.all());
74
+ }
75
+ return results;
76
+ },
77
+ async exec(query) {
78
+ sqlite.exec(query);
79
+ return { count: 0, duration: 0 };
80
+ }
81
+ };
82
+ }
83
+ export {
84
+ createD1Mock
85
+ };
@@ -0,0 +1,110 @@
1
+ // src/db/kysely/field-mapper.ts
2
+ function toSnakeCase(str) {
3
+ return str.replace(/([A-Z])/g, "_$1").toLowerCase().replace(/^_/, "");
4
+ }
5
+ function mapFieldToPostgresType(field) {
6
+ switch (field.type) {
7
+ case "text":
8
+ case "richtext":
9
+ case "select":
10
+ case "radio":
11
+ case "relationship":
12
+ return "text";
13
+ case "number":
14
+ return "double precision";
15
+ case "boolean":
16
+ return "boolean";
17
+ case "date":
18
+ return "timestamp with time zone";
19
+ case "json":
20
+ case "file":
21
+ return "jsonb";
22
+ default:
23
+ return "text";
24
+ }
25
+ }
26
+ function mapFieldToSQLiteType(field) {
27
+ switch (field.type) {
28
+ case "text":
29
+ case "richtext":
30
+ case "select":
31
+ case "radio":
32
+ case "relationship":
33
+ case "date":
34
+ case "json":
35
+ case "file":
36
+ return "text";
37
+ case "number":
38
+ return "numeric";
39
+ case "boolean":
40
+ return "integer";
41
+ default:
42
+ return "text";
43
+ }
44
+ }
45
+ function flattenFields(fields, prefix = "") {
46
+ const result = [];
47
+ for (const field of fields) {
48
+ if (field.type === "join" || field.type === "virtual")
49
+ continue;
50
+ const currentName = field.name ? `${prefix}${field.name}` : undefined;
51
+ if (field.type === "group") {
52
+ if (field.fields && Array.isArray(field.fields)) {
53
+ const nextPrefix = currentName ? `${currentName}__` : "";
54
+ result.push(...flattenFields(field.fields, nextPrefix));
55
+ }
56
+ continue;
57
+ }
58
+ if (field.type === "blocks") {
59
+ continue;
60
+ }
61
+ if (field.type === "relationship" && "hasMany" in field && field.hasMany) {
62
+ continue;
63
+ }
64
+ if (currentName) {
65
+ result.push({ ...field, name: currentName });
66
+ }
67
+ if (field.type === "row" || field.type === "collapsible") {
68
+ if (field.fields && Array.isArray(field.fields)) {
69
+ result.push(...flattenFields(field.fields, prefix));
70
+ }
71
+ }
72
+ if (field.type === "tabs" && field.tabs && Array.isArray(field.tabs)) {
73
+ for (const tab of field.tabs) {
74
+ if (tab.fields && Array.isArray(tab.fields)) {
75
+ result.push(...flattenFields(tab.fields, prefix));
76
+ }
77
+ }
78
+ }
79
+ }
80
+ return result;
81
+ }
82
+ function getRelationalFields(fields, prefix = "") {
83
+ const result = [];
84
+ for (const field of fields) {
85
+ const currentName = field.name ? `${prefix}${field.name}` : undefined;
86
+ if (field.type === "relationship" && "hasMany" in field && field.hasMany || field.type === "blocks") {
87
+ if (currentName) {
88
+ result.push({ ...field, name: currentName });
89
+ }
90
+ continue;
91
+ }
92
+ if (field.type === "group" || field.type === "row" || field.type === "collapsible") {
93
+ if (field.fields && Array.isArray(field.fields)) {
94
+ const nextPrefix = field.type === "group" && field.name ? `${currentName}__` : prefix;
95
+ result.push(...getRelationalFields(field.fields, nextPrefix));
96
+ }
97
+ continue;
98
+ }
99
+ if (field.type === "tabs" && field.tabs && Array.isArray(field.tabs)) {
100
+ for (const tab of field.tabs) {
101
+ if (tab.fields && Array.isArray(tab.fields)) {
102
+ result.push(...getRelationalFields(tab.fields, prefix));
103
+ }
104
+ }
105
+ }
106
+ }
107
+ return result;
108
+ }
109
+
110
+ export { toSnakeCase, mapFieldToPostgresType, mapFieldToSQLiteType, flattenFields, getRelationalFields };
@@ -0,0 +1,87 @@
1
+ // src/client.ts
2
+ class OpacaError extends Error {
3
+ status;
4
+ data;
5
+ constructor(message, status, data = {}) {
6
+ super(message);
7
+ this.name = "OpacaError";
8
+ this.status = status;
9
+ this.data = data;
10
+ }
11
+ }
12
+ function createClient(configOrOptions) {
13
+ const options = configOrOptions.serverURL ? { baseURL: configOrOptions.serverURL } : configOrOptions;
14
+ const baseURL = options.baseURL.replace(/\/$/, "");
15
+ const getHeaders = () => {
16
+ const headers = {
17
+ "Content-Type": "application/json"
18
+ };
19
+ if (options.token) {
20
+ headers.Authorization = `Bearer ${options.token}`;
21
+ }
22
+ return headers;
23
+ };
24
+ const fetcher = async (path, init) => {
25
+ const res = await fetch(`${baseURL}${path}`, {
26
+ ...init,
27
+ credentials: "include",
28
+ headers: {
29
+ ...getHeaders(),
30
+ ...init?.headers
31
+ }
32
+ });
33
+ if (!res.ok) {
34
+ const errorData = await res.json().catch(() => ({}));
35
+ throw new OpacaError(errorData.message || `API Error: ${res.status}`, res.status, errorData);
36
+ }
37
+ return res.json();
38
+ };
39
+ const client = {
40
+ getMetadata: () => fetcher("/api/__admin/metadata"),
41
+ getCollections: () => fetcher("/api/__admin/collections"),
42
+ getSetupStatus: () => fetcher("/api/__admin/setup"),
43
+ collections: new Proxy({}, {
44
+ get(_, collectionSlug) {
45
+ return {
46
+ find: (query) => {
47
+ const searchParams = new URLSearchParams;
48
+ if (query) {
49
+ Object.entries(query).forEach(([key, value]) => {
50
+ searchParams.append(key, typeof value === "object" ? JSON.stringify(value) : String(value));
51
+ });
52
+ }
53
+ const queryString = searchParams.toString();
54
+ return fetcher(`/api/${String(collectionSlug)}${queryString ? `?${queryString}` : ""}`);
55
+ },
56
+ findOne: (id) => fetcher(`/api/${String(collectionSlug)}/${id}`),
57
+ list: () => fetcher(`/api/${String(collectionSlug)}`),
58
+ create: (data) => fetcher(`/api/${String(collectionSlug)}`, {
59
+ method: "POST",
60
+ body: JSON.stringify(data)
61
+ }),
62
+ update: (id, data) => fetcher(`/api/${String(collectionSlug)}/${id}`, {
63
+ method: "PATCH",
64
+ body: JSON.stringify(data)
65
+ }),
66
+ delete: (id) => fetcher(`/api/${String(collectionSlug)}/${id}`, {
67
+ method: "DELETE"
68
+ })
69
+ };
70
+ }
71
+ }),
72
+ globals: new Proxy({}, {
73
+ get(_, globalSlug) {
74
+ return {
75
+ get: () => fetcher(`/api/globals/${String(globalSlug)}`),
76
+ update: (data) => fetcher(`/api/globals/${String(globalSlug)}`, {
77
+ method: "PATCH",
78
+ body: JSON.stringify(data)
79
+ })
80
+ };
81
+ }
82
+ })
83
+ };
84
+ return client;
85
+ }
86
+
87
+ export { OpacaError, createClient };