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,729 @@
1
+ import { Check, Copy, Key, Loader2, Plus, Trash2 } from "lucide-react";
2
+ import { useCallback, useEffect, useState } from "react";
3
+ import type { SerializableConfig } from "../../../types";
4
+ import { api } from "../../api-client";
5
+ import type { ToastType } from "../../stores/ui";
6
+
7
+ interface ApiKey {
8
+ id: string;
9
+ name: string;
10
+ prefix: string;
11
+ createdAt: string | number | Date;
12
+ permissions?: Record<string, string[]>;
13
+ }
14
+
15
+ export interface SettingsViewProps {
16
+ notify?: (message: string, type?: ToastType) => void;
17
+ config: SerializableConfig;
18
+ }
19
+
20
+ export function SettingsView({ notify, config }: SettingsViewProps) {
21
+ const [apiKeys, setApiKeys] = useState<ApiKey[]>([]);
22
+ const [loading, setLoading] = useState(true);
23
+ const [generating, setGenerating] = useState(false);
24
+ const [newKey, setNewKey] = useState<string | null>(null);
25
+ const [keyName, setKeyName] = useState("");
26
+ const [expiresIn, setExpiresIn] = useState<string>("31536000"); // Default: 1 year in seconds
27
+ const [copied, setCopied] = useState(false);
28
+ const [selectedPermissions, setSelectedPermissions] = useState<Record<string, string[]>>({});
29
+
30
+ const fetchApiKeys = useCallback(async () => {
31
+ try {
32
+ setLoading(true);
33
+ const res = await api.get("api/auth/api-key/list").json<{ apiKeys: ApiKey[] }>();
34
+ setApiKeys(res.apiKeys || []);
35
+ } catch (err: unknown) {
36
+ console.error(err);
37
+ notify?.("Failed to load API keys", "error");
38
+ } finally {
39
+ setLoading(false);
40
+ }
41
+ }, [notify]);
42
+
43
+ useEffect(() => {
44
+ fetchApiKeys();
45
+ }, [fetchApiKeys]);
46
+
47
+ const togglePermission = (collectionSlug: string, permission: string) => {
48
+ setSelectedPermissions((prev) => {
49
+ const current = prev[collectionSlug] || [];
50
+ const newPermissions = { ...prev };
51
+
52
+ if (current.includes(permission)) {
53
+ const filtered = current.filter((p: string) => p !== permission);
54
+ if (filtered.length === 0) {
55
+ delete newPermissions[collectionSlug];
56
+ } else {
57
+ newPermissions[collectionSlug] = filtered;
58
+ }
59
+ } else {
60
+ newPermissions[collectionSlug] = [...current, permission];
61
+ }
62
+ return newPermissions;
63
+ });
64
+ };
65
+
66
+ const handleSelectAll = (collection: string) => {
67
+ setSelectedPermissions((prev) => ({
68
+ ...prev,
69
+ [collection]: ACTIONS,
70
+ }));
71
+ };
72
+
73
+ const handleClearAll = (collection: string) => {
74
+ setSelectedPermissions((prev) => {
75
+ const next = { ...prev };
76
+ delete next[collection];
77
+ return next;
78
+ });
79
+ };
80
+
81
+ const handleGenerateKey = async (e: React.FormEvent) => {
82
+ e.preventDefault();
83
+ if (!keyName.trim()) return;
84
+
85
+ try {
86
+ setGenerating(true);
87
+ const res = await api
88
+ .post("api/__admin/api-key/create", {
89
+ json: {
90
+ name: keyName,
91
+ expiresIn: expiresIn ? Number(expiresIn) : undefined,
92
+ permissions: selectedPermissions,
93
+ },
94
+ })
95
+ .json<{ key: string }>();
96
+
97
+ setNewKey(res.key);
98
+ setKeyName("");
99
+ setExpiresIn("31536000");
100
+ setSelectedPermissions({});
101
+ await fetchApiKeys();
102
+ notify?.("API Key generated successfully");
103
+ } catch (err: unknown) {
104
+ console.error("API Key Creation Error:", err);
105
+ const errorData = (err as any).response?.json
106
+ ? await (err as any).response.json().catch(() => ({}))
107
+ : {};
108
+ notify?.(
109
+ errorData.message || (err as Error).message || "Failed to generate API key",
110
+ "error",
111
+ );
112
+ } finally {
113
+ setGenerating(false);
114
+ }
115
+ };
116
+
117
+ const handleDeleteKey = async (keyId: string) => {
118
+ if (!confirm("Are you sure you want to delete this API key? This action cannot be undone."))
119
+ return;
120
+
121
+ try {
122
+ await api.post("api/auth/api-key/delete", {
123
+ json: { keyId },
124
+ });
125
+ await fetchApiKeys();
126
+ notify?.("API Key deleted successfully");
127
+ } catch (err: unknown) {
128
+ console.error(err);
129
+ notify?.("Failed to delete API key", "error");
130
+ }
131
+ };
132
+
133
+ const copyToClipboard = () => {
134
+ if (newKey) {
135
+ navigator.clipboard.writeText(newKey);
136
+ setCopied(true);
137
+ setTimeout(() => setCopied(false), 2000);
138
+ notify?.("API Key copied to clipboard");
139
+ }
140
+ };
141
+
142
+ const ACTIONS = ["read", "create", "update", "delete"];
143
+
144
+ return (
145
+ <div className="opaca-admin-form">
146
+ <div className="opaca-header">
147
+ <div>
148
+ <h1 className="opaca-title">Settings</h1>
149
+ <p className="opaca-subtitle">Manage project settings and API Keys.</p>
150
+ </div>
151
+ </div>
152
+
153
+ <div style={{ display: "grid", gap: "2.5rem" }}>
154
+ {/* Create API Key Section */}
155
+ <div
156
+ className="opaca-card"
157
+ style={{
158
+ padding: "2.5rem",
159
+ border: "1px solid var(--opaca-border-strong)",
160
+ }}
161
+ >
162
+ <div
163
+ style={{
164
+ display: "flex",
165
+ alignItems: "center",
166
+ gap: "0.75rem",
167
+ marginBottom: "1rem",
168
+ }}
169
+ >
170
+ <div
171
+ style={{
172
+ width: "32px",
173
+ height: "32px",
174
+ borderRadius: "8px",
175
+ background: "rgba(var(--opaca-accent-rgb), 0.1)",
176
+ display: "flex",
177
+ alignItems: "center",
178
+ justifyContent: "center",
179
+ }}
180
+ >
181
+ <Plus size={18} color="var(--opaca-accent)" />
182
+ </div>
183
+ <h2 style={{ fontSize: "1.25rem", fontWeight: "600", margin: 0 }}>
184
+ Create New API Key
185
+ </h2>
186
+ </div>
187
+
188
+ <p
189
+ style={{
190
+ fontSize: "0.925rem",
191
+ color: "var(--opaca-text-dim)",
192
+ marginBottom: "2.5rem",
193
+ maxWidth: "600px",
194
+ }}
195
+ >
196
+ Generate a secure token for external systems. Define granular permissions to control
197
+ which data can be accessed or modified.
198
+ </p>
199
+
200
+ <form onSubmit={handleGenerateKey} style={{ display: "grid", gap: "2rem" }}>
201
+ <div
202
+ style={{
203
+ display: "grid",
204
+ gridTemplateColumns: "1fr 1fr",
205
+ gap: "1rem",
206
+ }}
207
+ >
208
+ <div className="opaca-form-group" style={{ margin: 0 }}>
209
+ <label className="opaca-label" htmlFor="key-name">
210
+ Key Name
211
+ </label>
212
+ <input
213
+ id="key-name"
214
+ type="text"
215
+ className="opaca-input"
216
+ style={{ fontSize: "1rem", padding: "0.875rem 1rem" }}
217
+ value={keyName}
218
+ onChange={(e) => setKeyName(e.target.value)}
219
+ placeholder="e.g. Production Mobile App"
220
+ required
221
+ />
222
+ <span
223
+ style={{
224
+ fontSize: "0.75rem",
225
+ color: "var(--opaca-text-muted)",
226
+ marginTop: "0.5rem",
227
+ display: "block",
228
+ }}
229
+ >
230
+ A descriptive name to help you identify where this key is used.
231
+ </span>
232
+ </div>
233
+ <div className="opaca-form-group" style={{ margin: 0 }}>
234
+ <label className="opaca-label" htmlFor="expires-in">
235
+ Expiration
236
+ </label>
237
+ <select
238
+ id="expires-in"
239
+ className="opaca-input"
240
+ style={{
241
+ fontSize: "1rem",
242
+ padding: "0.875rem 1rem",
243
+ height: "auto",
244
+ }}
245
+ value={expiresIn}
246
+ onChange={(e) => setExpiresIn(e.target.value)}
247
+ >
248
+ <option value="2592000">30 days</option>
249
+ <option value="7776000">90 days</option>
250
+ <option value="31536000">1 year</option>
251
+ <option value="">Never expire</option>
252
+ </select>
253
+ <span
254
+ style={{
255
+ fontSize: "0.75rem",
256
+ color: "var(--opaca-text-muted)",
257
+ marginTop: "0.5rem",
258
+ display: "block",
259
+ }}
260
+ >
261
+ How long until this key automatically expires.
262
+ </span>
263
+ </div>
264
+ </div>
265
+
266
+ <div
267
+ style={{
268
+ background: "rgba(var(--opaca-card-bg-rgb), 0.5)",
269
+ border: "1px solid var(--opaca-border)",
270
+ borderRadius: "12px",
271
+ overflow: "hidden",
272
+ }}
273
+ >
274
+ <div
275
+ style={{
276
+ padding: "1rem 1.25rem",
277
+ background: "rgba(0,0,0,0.2)",
278
+ borderBottom: "1px solid var(--opaca-border)",
279
+ display: "flex",
280
+ justifyContent: "space-between",
281
+ alignItems: "center",
282
+ }}
283
+ >
284
+ <label
285
+ htmlFor="permissions-matrix"
286
+ className="opaca-label"
287
+ style={{ margin: 0, fontWeight: "600" }}
288
+ >
289
+ Permissions Matrix
290
+ </label>
291
+ <span
292
+ style={{
293
+ fontSize: "0.75rem",
294
+ color: "var(--opaca-text-muted)",
295
+ }}
296
+ >
297
+ Actions for {config.collections.length}{" "}
298
+ {config.collections.length === 1 ? "collection" : "collections"}
299
+ </span>
300
+ </div>
301
+
302
+ <div style={{ overflowX: "auto" }}>
303
+ <table
304
+ style={{
305
+ width: "100%",
306
+ borderCollapse: "collapse",
307
+ textAlign: "left",
308
+ }}
309
+ >
310
+ <thead>
311
+ <tr
312
+ style={{
313
+ borderBottom: "1px solid var(--opaca-border)",
314
+ background: "rgba(255,255,255,0.02)",
315
+ }}
316
+ >
317
+ <th
318
+ style={{
319
+ padding: "1rem 1.25rem",
320
+ fontSize: "0.75rem",
321
+ fontWeight: "600",
322
+ color: "var(--opaca-text-muted)",
323
+ textTransform: "uppercase",
324
+ }}
325
+ >
326
+ Collection
327
+ </th>
328
+ {ACTIONS.map((action) => (
329
+ <th
330
+ key={action}
331
+ style={{
332
+ padding: "1rem",
333
+ fontSize: "0.75rem",
334
+ fontWeight: "600",
335
+ color: "var(--opaca-text-muted)",
336
+ textAlign: "center",
337
+ textTransform: "uppercase",
338
+ }}
339
+ >
340
+ {action}
341
+ </th>
342
+ ))}
343
+ <th
344
+ style={{
345
+ padding: "1rem 1.25rem",
346
+ fontSize: "0.75rem",
347
+ fontWeight: "600",
348
+ color: "var(--opaca-text-muted)",
349
+ textAlign: "right",
350
+ textTransform: "uppercase",
351
+ }}
352
+ >
353
+ Full Access
354
+ </th>
355
+ </tr>
356
+ </thead>
357
+ <tbody>
358
+ {config.collections.map((col) => {
359
+ const selectedForCol = selectedPermissions[col.slug] || [];
360
+ const isFullAccess = ACTIONS.every((a) => selectedForCol.includes(a));
361
+
362
+ return (
363
+ <tr
364
+ key={col.slug}
365
+ style={{
366
+ borderBottom: "1px solid var(--opaca-border)",
367
+ transition: "background 0.2s",
368
+ }}
369
+ >
370
+ <td style={{ padding: "1rem 1.25rem" }}>
371
+ <span
372
+ style={{
373
+ fontSize: "0.875rem",
374
+ fontWeight: "600",
375
+ }}
376
+ >
377
+ {col.slug}
378
+ </span>
379
+ </td>
380
+ {ACTIONS.map((action) => {
381
+ const isSelected = selectedForCol.includes(action);
382
+ return (
383
+ <td
384
+ key={action}
385
+ style={{
386
+ padding: "0.5rem",
387
+ textAlign: "center",
388
+ }}
389
+ >
390
+ <button
391
+ type="button"
392
+ onClick={() => togglePermission(col.slug, action)}
393
+ style={{
394
+ width: "32px",
395
+ height: "32px",
396
+ borderRadius: "6px",
397
+ border: "1px solid",
398
+ borderColor: isSelected
399
+ ? "var(--opaca-accent)"
400
+ : "var(--opaca-border)",
401
+ background: isSelected
402
+ ? "rgba(var(--opaca-accent-rgb), 0.15)"
403
+ : "transparent",
404
+ color: isSelected
405
+ ? "var(--opaca-accent)"
406
+ : "var(--opaca-text-muted)",
407
+ display: "inline-flex",
408
+ alignItems: "center",
409
+ justifyContent: "center",
410
+ cursor: "pointer",
411
+ transition: "all 0.2s",
412
+ }}
413
+ title={`${action.charAt(0).toUpperCase() + action.slice(1)} ${col.slug}`}
414
+ >
415
+ <Check
416
+ size={14}
417
+ style={{ opacity: isSelected ? 1 : 0.1 }}
418
+ strokeWidth={3}
419
+ />
420
+ </button>
421
+ </td>
422
+ );
423
+ })}
424
+ <td
425
+ style={{
426
+ padding: "1rem 1.25rem",
427
+ textAlign: "right",
428
+ }}
429
+ >
430
+ <button
431
+ type="button"
432
+ onClick={() =>
433
+ isFullAccess ? handleClearAll(col.slug) : handleSelectAll(col.slug)
434
+ }
435
+ style={{
436
+ fontSize: "0.75rem",
437
+ color: isFullAccess
438
+ ? "var(--opaca-accent)"
439
+ : "var(--opaca-text-dim)",
440
+ background: "transparent",
441
+ border: "1px solid",
442
+ borderColor: isFullAccess
443
+ ? "var(--opaca-accent)"
444
+ : "var(--opaca-border)",
445
+ padding: "0.3rem 0.75rem",
446
+ borderRadius: "6px",
447
+ cursor: "pointer",
448
+ transition: "all 0.2s",
449
+ }}
450
+ >
451
+ {isFullAccess ? "All Actions" : "Full Access?"}
452
+ </button>
453
+ </td>
454
+ </tr>
455
+ );
456
+ })}
457
+ </tbody>
458
+ </table>
459
+ </div>
460
+
461
+ {config.collections.length === 0 && (
462
+ <div style={{ padding: "3rem 1rem", textAlign: "center" }}>
463
+ <p
464
+ style={{
465
+ fontSize: "0.875rem",
466
+ color: "var(--opaca-text-dim)",
467
+ }}
468
+ >
469
+ No collections available to configure.
470
+ </p>
471
+ </div>
472
+ )}
473
+ </div>
474
+
475
+ <button
476
+ type="submit"
477
+ className="opaca-btn opaca-btn-primary"
478
+ disabled={generating || !keyName.trim()}
479
+ style={{
480
+ width: "100%",
481
+ height: "48px",
482
+ fontSize: "1rem",
483
+ fontWeight: "600",
484
+ boxShadow: "0 4px 12px rgba(var(--opaca-accent-rgb), 0.2)",
485
+ marginTop: "0.5rem",
486
+ }}
487
+ >
488
+ {generating ? (
489
+ <Loader2 size={18} className="opaca-spin" />
490
+ ) : (
491
+ <Plus size={18} strokeWidth={2.5} />
492
+ )}
493
+ Generate API Key
494
+ </button>
495
+ </form>
496
+ </div>
497
+
498
+ {/* Existing API Keys Section */}
499
+ <div style={{ padding: "0 0.5rem" }}>
500
+ <div
501
+ style={{
502
+ display: "flex",
503
+ alignItems: "center",
504
+ gap: "0.75rem",
505
+ marginBottom: "1.5rem",
506
+ }}
507
+ >
508
+ <Key size={18} color="var(--opaca-text-dim)" />
509
+ <h2 style={{ fontSize: "1.125rem", fontWeight: "500", margin: 0 }}>Existing Keys</h2>
510
+ <div
511
+ style={{
512
+ height: "1px",
513
+ flex: 1,
514
+ background: "var(--opaca-border)",
515
+ marginLeft: "0.5rem",
516
+ }}
517
+ />
518
+ </div>
519
+
520
+ {newKey && (
521
+ <div
522
+ style={{
523
+ marginBottom: "2.5rem",
524
+ padding: "1.75rem",
525
+ background:
526
+ "linear-gradient(135deg, rgba(52, 211, 153, 0.1) 0%, rgba(52, 211, 153, 0.02) 100%)",
527
+ border: "1px solid rgba(52, 211, 153, 0.3)",
528
+ borderRadius: "16px",
529
+ position: "relative",
530
+ overflow: "hidden",
531
+ }}
532
+ >
533
+ <div
534
+ style={{
535
+ position: "absolute",
536
+ top: 0,
537
+ left: 0,
538
+ width: "4px",
539
+ height: "100%",
540
+ background: "var(--opaca-success)",
541
+ }}
542
+ />
543
+ <div
544
+ style={{
545
+ display: "flex",
546
+ justifyContent: "space-between",
547
+ alignItems: "flex-start",
548
+ marginBottom: "1.25rem",
549
+ }}
550
+ >
551
+ <div>
552
+ <h3
553
+ style={{
554
+ fontSize: "1rem",
555
+ fontWeight: "600",
556
+ color: "var(--opaca-success)",
557
+ margin: "0 0 0.4rem 0",
558
+ }}
559
+ >
560
+ Key Generated Successfully
561
+ </h3>
562
+ <p
563
+ style={{
564
+ fontSize: "0.875rem",
565
+ color: "var(--opaca-text-dim)",
566
+ margin: 0,
567
+ }}
568
+ >
569
+ Copy this key now. For security reasons, it cannot be displayed again.
570
+ </p>
571
+ </div>
572
+ <div
573
+ style={{
574
+ padding: "0.4rem 0.8rem",
575
+ borderRadius: "20px",
576
+ background: "rgba(52, 211, 153, 0.2)",
577
+ color: "var(--opaca-success)",
578
+ fontSize: "0.7rem",
579
+ fontWeight: "700",
580
+ textTransform: "uppercase",
581
+ }}
582
+ >
583
+ Confidential
584
+ </div>
585
+ </div>
586
+ <div style={{ display: "flex", gap: "0.75rem" }}>
587
+ <code
588
+ style={{
589
+ flex: 1,
590
+ padding: "1rem 1.25rem",
591
+ background: "rgba(0,0,0,0.4)",
592
+ border: "1px solid rgba(255,255,255,0.1)",
593
+ borderRadius: "10px",
594
+ fontFamily: "monospace",
595
+ fontSize: "1rem",
596
+ color: "#fff",
597
+ letterSpacing: "0.5px",
598
+ }}
599
+ >
600
+ {newKey}
601
+ </code>
602
+ <button
603
+ type="button"
604
+ onClick={copyToClipboard}
605
+ className="opaca-btn"
606
+ style={{
607
+ background: "white",
608
+ color: "#000",
609
+ width: "48px",
610
+ height: "48px",
611
+ borderRadius: "10px",
612
+ }}
613
+ >
614
+ {copied ? <Check size={18} /> : <Copy size={18} />}
615
+ </button>
616
+ </div>
617
+ </div>
618
+ )}
619
+
620
+ {loading ? (
621
+ <div
622
+ style={{
623
+ display: "flex",
624
+ justifyContent: "center",
625
+ padding: "2rem",
626
+ }}
627
+ >
628
+ <Loader2 size={24} className="opaca-spin" color="var(--opaca-text-muted)" />
629
+ </div>
630
+ ) : apiKeys.length === 0 ? (
631
+ <div
632
+ style={{
633
+ textAlign: "center",
634
+ padding: "2rem",
635
+ color: "var(--opaca-text-muted)",
636
+ fontSize: "0.875rem",
637
+ background: "rgba(0,0,0,0.1)",
638
+ borderRadius: "var(--opaca-radius)",
639
+ }}
640
+ >
641
+ No API keys generated yet.
642
+ </div>
643
+ ) : (
644
+ <div style={{ display: "grid", gap: "0.5rem" }}>
645
+ {apiKeys.map((key) => (
646
+ <div
647
+ key={key.id}
648
+ style={{
649
+ display: "flex",
650
+ alignItems: "center",
651
+ justifyContent: "space-between",
652
+ padding: "1rem",
653
+ background: "var(--opaca-card-bg)",
654
+ border: "1px solid var(--opaca-border)",
655
+ borderRadius: "var(--opaca-radius)",
656
+ }}
657
+ >
658
+ <div>
659
+ <div
660
+ style={{
661
+ fontWeight: "500",
662
+ fontSize: "0.875rem",
663
+ marginBottom: "0.25rem",
664
+ }}
665
+ >
666
+ {key.name}
667
+ </div>
668
+ <div
669
+ style={{
670
+ display: "flex",
671
+ gap: "1rem",
672
+ fontSize: "0.75rem",
673
+ color: "var(--opaca-text-dim)",
674
+ }}
675
+ >
676
+ <span>Created: {new Date(key.createdAt).toLocaleDateString()}</span>
677
+ <span>Prefix: {key.prefix}...</span>
678
+ </div>
679
+ {key.permissions && Object.keys(key.permissions).length > 0 && (
680
+ <div
681
+ style={{
682
+ marginTop: "0.75rem",
683
+ display: "flex",
684
+ flexWrap: "wrap",
685
+ gap: "0.375rem",
686
+ }}
687
+ >
688
+ {Object.entries(key.permissions).map(([col, perms]) => (
689
+ <div
690
+ key={col}
691
+ style={{
692
+ fontSize: "10px",
693
+ background: "rgba(255,255,255,0.05)",
694
+ padding: "0.125rem 0.5rem",
695
+ borderRadius: "4px",
696
+ border: "1px solid var(--opaca-border)",
697
+ color: "var(--opaca-text-dim)",
698
+ }}
699
+ >
700
+ <strong style={{ color: "var(--opaca-text)" }}>{col}:</strong>{" "}
701
+ {(perms as string[]).join(", ")}
702
+ </div>
703
+ ))}
704
+ </div>
705
+ )}
706
+ </div>
707
+ <button
708
+ type="button"
709
+ onClick={() => handleDeleteKey(key.id)}
710
+ className="opaca-btn"
711
+ style={{
712
+ color: "var(--opaca-error)",
713
+ background: "transparent",
714
+ border: "none",
715
+ padding: "0.5rem",
716
+ }}
717
+ title="Revoke Key"
718
+ >
719
+ <Trash2 size={16} />
720
+ </button>
721
+ </div>
722
+ ))}
723
+ </div>
724
+ )}
725
+ </div>
726
+ </div>
727
+ </div>
728
+ );
729
+ }