opacacms 0.1.1 → 0.1.2

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 (212) hide show
  1. package/package.json +36 -1
  2. package/bun.lock +0 -34
  3. package/global.d.ts +0 -11
  4. package/src/admin/api-client.ts +0 -63
  5. package/src/admin/auth-client.ts +0 -40
  6. package/src/admin/custom-field.ts +0 -179
  7. package/src/admin/index.ts +0 -15
  8. package/src/admin/react.tsx +0 -72
  9. package/src/admin/router.ts +0 -9
  10. package/src/admin/stores/admin-queries.ts +0 -121
  11. package/src/admin/stores/auth.ts +0 -61
  12. package/src/admin/stores/column-visibility.ts +0 -67
  13. package/src/admin/stores/config.ts +0 -15
  14. package/src/admin/stores/media.ts +0 -95
  15. package/src/admin/stores/query.ts +0 -13
  16. package/src/admin/stores/ui.ts +0 -29
  17. package/src/admin/ui/admin-client.tsx +0 -283
  18. package/src/admin/ui/admin-layout.tsx +0 -276
  19. package/src/admin/ui/components/ColumnVisibilityToggle.tsx +0 -141
  20. package/src/admin/ui/components/DataDetailSheet.tsx +0 -141
  21. package/src/admin/ui/components/DataDetailView.tsx +0 -175
  22. package/src/admin/ui/components/Table.tsx +0 -67
  23. package/src/admin/ui/components/fields/ArrayField.tsx +0 -166
  24. package/src/admin/ui/components/fields/BlocksField.tsx +0 -202
  25. package/src/admin/ui/components/fields/BooleanField.tsx +0 -50
  26. package/src/admin/ui/components/fields/CollapsibleField.tsx +0 -75
  27. package/src/admin/ui/components/fields/DateField.tsx +0 -45
  28. package/src/admin/ui/components/fields/FileField.tsx +0 -322
  29. package/src/admin/ui/components/fields/GroupField.tsx +0 -50
  30. package/src/admin/ui/components/fields/JoinField.tsx +0 -23
  31. package/src/admin/ui/components/fields/NumberField.tsx +0 -46
  32. package/src/admin/ui/components/fields/RadioField.tsx +0 -62
  33. package/src/admin/ui/components/fields/RelationshipField.tsx +0 -278
  34. package/src/admin/ui/components/fields/RowField.tsx +0 -40
  35. package/src/admin/ui/components/fields/SelectField.tsx +0 -59
  36. package/src/admin/ui/components/fields/TabsField.tsx +0 -101
  37. package/src/admin/ui/components/fields/TextAreaField.tsx +0 -54
  38. package/src/admin/ui/components/fields/TextField.tsx +0 -49
  39. package/src/admin/ui/components/fields/VirtualField.tsx +0 -53
  40. package/src/admin/ui/components/fields/index.tsx +0 -371
  41. package/src/admin/ui/components/fields/richtext-editor/index.tsx +0 -211
  42. package/src/admin/ui/components/fields/richtext-editor/nodes/ImageComponent.tsx +0 -142
  43. package/src/admin/ui/components/fields/richtext-editor/nodes/ImageNode.tsx +0 -95
  44. package/src/admin/ui/components/fields/richtext-editor/plugins/ComponentPickerPlugin.tsx +0 -226
  45. package/src/admin/ui/components/fields/richtext-editor/plugins/EditableSyncPlugin.tsx +0 -16
  46. package/src/admin/ui/components/fields/richtext-editor/plugins/NotionToolbarPlugin.tsx +0 -184
  47. package/src/admin/ui/components/fields/richtext-editor/plugins/SimpleToolbarPlugin.tsx +0 -240
  48. package/src/admin/ui/components/fields/richtext-editor/plugins/ValueSyncPlugin.tsx +0 -40
  49. package/src/admin/ui/components/fields/utils.ts +0 -1
  50. package/src/admin/ui/components/link.tsx +0 -41
  51. package/src/admin/ui/components/media/AssetManagerModal.tsx +0 -334
  52. package/src/admin/ui/components/toast.tsx +0 -72
  53. package/src/admin/ui/components/ui/accordion.tsx +0 -51
  54. package/src/admin/ui/components/ui/alert-dialog.tsx +0 -98
  55. package/src/admin/ui/components/ui/blocks.tsx +0 -32
  56. package/src/admin/ui/components/ui/breadcrumbs.tsx +0 -59
  57. package/src/admin/ui/components/ui/button.tsx +0 -26
  58. package/src/admin/ui/components/ui/collapsible.tsx +0 -124
  59. package/src/admin/ui/components/ui/dialog.tsx +0 -79
  60. package/src/admin/ui/components/ui/group.tsx +0 -20
  61. package/src/admin/ui/components/ui/index.ts +0 -17
  62. package/src/admin/ui/components/ui/input.tsx +0 -12
  63. package/src/admin/ui/components/ui/join.tsx +0 -53
  64. package/src/admin/ui/components/ui/label.tsx +0 -11
  65. package/src/admin/ui/components/ui/radio-group.tsx +0 -75
  66. package/src/admin/ui/components/ui/relationship-detail-sheet.tsx +0 -122
  67. package/src/admin/ui/components/ui/relationship.tsx +0 -58
  68. package/src/admin/ui/components/ui/scroll-area.tsx +0 -19
  69. package/src/admin/ui/components/ui/select.tsx +0 -187
  70. package/src/admin/ui/components/ui/separator.tsx +0 -21
  71. package/src/admin/ui/components/ui/sheet.tsx +0 -106
  72. package/src/admin/ui/components/ui/tabs.tsx +0 -116
  73. package/src/admin/ui/components/ui/utils.ts +0 -3
  74. package/src/admin/ui/hooks/use-debounce.ts +0 -15
  75. package/src/admin/ui/styles/_locale-switcher.scss +0 -33
  76. package/src/admin/ui/styles/accordion.scss +0 -60
  77. package/src/admin/ui/styles/animations.scss +0 -41
  78. package/src/admin/ui/styles/asset-manager.scss +0 -547
  79. package/src/admin/ui/styles/badge.scss +0 -13
  80. package/src/admin/ui/styles/base.scss +0 -22
  81. package/src/admin/ui/styles/button.scss +0 -161
  82. package/src/admin/ui/styles/card.scss +0 -13
  83. package/src/admin/ui/styles/collapsible.scss +0 -75
  84. package/src/admin/ui/styles/data-detail.scss +0 -92
  85. package/src/admin/ui/styles/dialog.scss +0 -102
  86. package/src/admin/ui/styles/empty-state.scss +0 -22
  87. package/src/admin/ui/styles/group.scss +0 -19
  88. package/src/admin/ui/styles/index.scss +0 -33
  89. package/src/admin/ui/styles/input.scss +0 -80
  90. package/src/admin/ui/styles/label.scss +0 -12
  91. package/src/admin/ui/styles/layout.scss +0 -56
  92. package/src/admin/ui/styles/lexical.scss +0 -469
  93. package/src/admin/ui/styles/loading.scss +0 -102
  94. package/src/admin/ui/styles/media-registry.scss +0 -597
  95. package/src/admin/ui/styles/pagination.scss +0 -20
  96. package/src/admin/ui/styles/radio-group.scss +0 -66
  97. package/src/admin/ui/styles/row.scss +0 -17
  98. package/src/admin/ui/styles/scrollbar.scss +0 -36
  99. package/src/admin/ui/styles/select.scss +0 -121
  100. package/src/admin/ui/styles/separator.scss +0 -14
  101. package/src/admin/ui/styles/sheet.scss +0 -152
  102. package/src/admin/ui/styles/sidebar.scss +0 -148
  103. package/src/admin/ui/styles/switch.scss +0 -59
  104. package/src/admin/ui/styles/table.scss +0 -207
  105. package/src/admin/ui/styles/tabs.scss +0 -62
  106. package/src/admin/ui/styles/toast.scss +0 -45
  107. package/src/admin/ui/styles/variables.scss +0 -24
  108. package/src/admin/ui/views/collection-list-view.tsx +0 -720
  109. package/src/admin/ui/views/dashboard-view.tsx +0 -263
  110. package/src/admin/ui/views/document-edit-view.tsx +0 -384
  111. package/src/admin/ui/views/global-edit-view.tsx +0 -226
  112. package/src/admin/ui/views/init-view.tsx +0 -182
  113. package/src/admin/ui/views/login-view.tsx +0 -123
  114. package/src/admin/ui/views/media-registry-view.tsx +0 -1104
  115. package/src/admin/ui/views/settings-view.tsx +0 -729
  116. package/src/admin/webcomponent.tsx +0 -15
  117. package/src/auth/index.ts +0 -194
  118. package/src/auth/migrations.ts +0 -87
  119. package/src/auth/premissions.ts +0 -46
  120. package/src/cli/commands/generate-types.ts +0 -116
  121. package/src/cli/commands/init.ts +0 -95
  122. package/src/cli/commands/migrate-commands.ts +0 -160
  123. package/src/cli/commands/seed-command.ts +0 -11
  124. package/src/cli/d1-mock.ts +0 -101
  125. package/src/cli/index.test.ts +0 -84
  126. package/src/cli/index.ts +0 -183
  127. package/src/cli/r2-mock.ts +0 -217
  128. package/src/cli/seeding.ts +0 -409
  129. package/src/client.ts +0 -181
  130. package/src/config-utils.ts +0 -102
  131. package/src/config.ts +0 -49
  132. package/src/db/adapter.ts +0 -53
  133. package/src/db/better-sqlite.ts +0 -630
  134. package/src/db/bun-sqlite.ts +0 -646
  135. package/src/db/d1.ts +0 -711
  136. package/src/db/index.ts +0 -2
  137. package/src/db/kysely/data-mapper.ts +0 -142
  138. package/src/db/kysely/field-mapper.ts +0 -148
  139. package/src/db/kysely/migration-generator.ts +0 -223
  140. package/src/db/kysely/query-builder.ts +0 -92
  141. package/src/db/kysely/schema-builder.ts +0 -439
  142. package/src/db/kysely/sql-utils.ts +0 -13
  143. package/src/db/migration.ts +0 -40
  144. package/src/db/postgres.ts +0 -621
  145. package/src/db/sqlite.ts +0 -658
  146. package/src/db/system-schema.ts +0 -121
  147. package/src/index.ts +0 -11
  148. package/src/runtimes/README.md +0 -59
  149. package/src/runtimes/bun.ts +0 -49
  150. package/src/runtimes/cloudflare-workers.ts +0 -38
  151. package/src/runtimes/next.ts +0 -26
  152. package/src/runtimes/node.ts +0 -52
  153. package/src/schema/collection.ts +0 -184
  154. package/src/schema/fields/base.ts +0 -164
  155. package/src/schema/fields/index.ts +0 -427
  156. package/src/schema/global.ts +0 -145
  157. package/src/schema/index.ts +0 -4
  158. package/src/schema/infer.ts +0 -72
  159. package/src/server/admin-router.ts +0 -20
  160. package/src/server/admin.ts +0 -142
  161. package/src/server/assets.ts +0 -306
  162. package/src/server/collection-router.ts +0 -55
  163. package/src/server/handlers.ts +0 -722
  164. package/src/server/middlewares/admin.ts +0 -27
  165. package/src/server/middlewares/auth.ts +0 -89
  166. package/src/server/middlewares/context.ts +0 -17
  167. package/src/server/middlewares/cors.ts +0 -24
  168. package/src/server/middlewares/database-init.ts +0 -74
  169. package/src/server/middlewares/rate-limit.ts +0 -71
  170. package/src/server/router.ts +0 -47
  171. package/src/server/setup-middlewares.ts +0 -58
  172. package/src/server/system-router.ts +0 -35
  173. package/src/server.ts +0 -9
  174. package/src/storage/adapters/cloudflare-r2.ts +0 -136
  175. package/src/storage/adapters/local.ts +0 -146
  176. package/src/storage/adapters/s3.ts +0 -186
  177. package/src/storage/errors.ts +0 -46
  178. package/src/storage/index.ts +0 -6
  179. package/src/storage/types.ts +0 -39
  180. package/src/types.ts +0 -605
  181. package/src/utils/lexical.ts +0 -37
  182. package/src/utils/logger.ts +0 -73
  183. package/src/validation.ts +0 -429
  184. package/src/validator.ts +0 -179
  185. package/test/admin-custom-field.test.ts +0 -162
  186. package/test/admin-react-field.test.tsx +0 -134
  187. package/test/api-features.test.ts +0 -78
  188. package/test/api.test.ts +0 -178
  189. package/test/auth.test.ts +0 -62
  190. package/test/cli-integration.test.ts +0 -148
  191. package/test/cli.test.ts +0 -25
  192. package/test/db/postgres.test.ts +0 -95
  193. package/test/db/sqlite-filter.test.ts +0 -53
  194. package/test/db/sqlite.test.ts +0 -82
  195. package/test/engine-features.test.ts +0 -79
  196. package/test/globals.test.ts +0 -74
  197. package/test/integration-tmp/db-app/opacacms.config.ts +0 -15
  198. package/test/integration-tmp/my-sqlite-app/opacacms.config.ts +0 -25
  199. package/test/integration-tmp/my-test-app/index.ts +0 -8
  200. package/test/integration-tmp/my-test-app/opacacms.config.ts +0 -16
  201. package/test/integration-tmp/my-test-app/package.json +0 -12
  202. package/test/populate.test.ts +0 -79
  203. package/test/runtimes.test.ts +0 -43
  204. package/test/schema-builder.test.ts +0 -107
  205. package/test/schema-features.test.ts +0 -63
  206. package/test/seeding.test.ts +0 -68
  207. package/test/storage/local.test.ts +0 -72
  208. package/test/storage/s3.test.ts +0 -60
  209. package/test/structural-data.test.ts +0 -100
  210. package/test/test-setup.ts +0 -11
  211. package/test/validation.test.ts +0 -162
  212. package/tsconfig.json +0 -42
@@ -1,729 +0,0 @@
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
- }