@alepha/devtools 0.16.1 → 0.19.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 (238) hide show
  1. package/README.md +1 -5
  2. package/dist/index.browser.js +224 -0
  3. package/dist/index.browser.js.map +1 -0
  4. package/dist/index.d.ts +351 -323
  5. package/dist/index.js +350 -186
  6. package/dist/index.js.map +1 -1
  7. package/package.json +33 -25
  8. package/src/assets.ts +6 -0
  9. package/src/{api/entities → entities}/logs.ts +2 -2
  10. package/src/index.browser.ts +11 -0
  11. package/src/index.shared.ts +15 -0
  12. package/src/index.ts +11 -37
  13. package/src/{api/providers → providers}/DevToolsMetadataProvider.ts +84 -47
  14. package/src/providers/DevToolsProvider.ts +394 -0
  15. package/src/{api/schemas → schemas}/DevActionMetadata.ts +8 -0
  16. package/src/{api/schemas → schemas}/DevEntityMetadata.ts +3 -0
  17. package/src/{api/schemas → schemas}/DevMetadata.ts +13 -2
  18. package/src/{api/schemas → schemas}/DevPageMetadata.ts +3 -0
  19. package/src/{api/schemas → schemas}/DevTopicMetadata.ts +1 -0
  20. package/src/ui/AppRouter.tsx +73 -59
  21. package/src/ui/components/DevLayout.tsx +103 -84
  22. package/src/ui/components/configuration/ConfigAtoms.page.tsx +5 -0
  23. package/src/ui/components/configuration/ConfigAtoms.tsx +511 -0
  24. package/src/ui/components/configuration/ConfigEnv.page.tsx +5 -0
  25. package/src/ui/components/configuration/ConfigEnv.tsx +230 -0
  26. package/src/ui/components/configuration/DevConfiguration.tsx +36 -0
  27. package/src/ui/components/configuration/index.ts +3 -0
  28. package/src/ui/components/dashboard/DevDashboard.tsx +501 -0
  29. package/src/ui/components/database/DatabaseEditor.page.tsx +23 -0
  30. package/src/ui/components/database/DatabaseEditor.tsx +402 -0
  31. package/src/ui/components/database/DatabaseErd.page.tsx +28 -0
  32. package/src/ui/components/database/DatabaseErd.tsx +107 -0
  33. package/src/ui/components/database/DevDatabase.tsx +36 -0
  34. package/src/ui/components/database/EntityNode.tsx +83 -0
  35. package/src/ui/components/emails/DevEmails.tsx +250 -0
  36. package/src/ui/components/explorer/DevExplorer.tsx +351 -0
  37. package/src/ui/components/explorer/ExplorerTree.tsx +178 -0
  38. package/src/ui/components/explorer/panels/DevPanelAction.tsx +499 -0
  39. package/src/ui/components/explorer/panels/DevPanelCache.tsx +73 -0
  40. package/src/ui/components/explorer/panels/DevPanelPage.tsx +96 -0
  41. package/src/ui/components/explorer/panels/DevPanelQueue.tsx +51 -0
  42. package/src/ui/components/explorer/panels/DevPanelTopic.tsx +56 -0
  43. package/src/ui/components/explorer/panels/index.ts +5 -0
  44. package/src/ui/components/graph/DevDependencyGraph.tsx +35 -60
  45. package/src/ui/components/graph/GraphControls.tsx +10 -11
  46. package/src/ui/components/graph/NodeDetails.tsx +22 -29
  47. package/src/ui/components/graph/ProviderNode.tsx +4 -4
  48. package/src/ui/components/graph/helpers.ts +1 -1
  49. package/src/ui/components/logs/DevLogs.tsx +661 -0
  50. package/src/ui/components/logs/index.ts +1 -0
  51. package/src/ui/components/shared/TreeView.tsx +189 -0
  52. package/src/ui/components/sms/DevSms.tsx +225 -0
  53. package/src/ui/main.css +17 -0
  54. package/src/ui/main.ts +2 -6
  55. package/LICENSE +0 -21
  56. package/assets/devtools/actions.html +0 -21
  57. package/assets/devtools/actions.html.br +0 -0
  58. package/assets/devtools/actions.html.gz +0 -0
  59. package/assets/devtools/asset.BZV40eAE.css +0 -1
  60. package/assets/devtools/asset.BZV40eAE.css.br +0 -0
  61. package/assets/devtools/asset.BZV40eAE.css.gz +0 -0
  62. package/assets/devtools/asset.CBnMq2vO.css +0 -1
  63. package/assets/devtools/asset.CBnMq2vO.css.br +0 -0
  64. package/assets/devtools/asset.CBnMq2vO.css.gz +0 -0
  65. package/assets/devtools/atoms.html +0 -21
  66. package/assets/devtools/atoms.html.br +0 -0
  67. package/assets/devtools/atoms.html.gz +0 -0
  68. package/assets/devtools/caches.html +0 -21
  69. package/assets/devtools/caches.html.br +0 -0
  70. package/assets/devtools/caches.html.gz +0 -0
  71. package/assets/devtools/chunk.6INqNjF0.js +0 -1
  72. package/assets/devtools/chunk.6INqNjF0.js.br +0 -0
  73. package/assets/devtools/chunk.6INqNjF0.js.gz +0 -0
  74. package/assets/devtools/chunk.9vpWpXSF.js +0 -1
  75. package/assets/devtools/chunk.9vpWpXSF.js.br +0 -0
  76. package/assets/devtools/chunk.9vpWpXSF.js.gz +0 -0
  77. package/assets/devtools/chunk.B4peH6PS.js +0 -1
  78. package/assets/devtools/chunk.B4peH6PS.js.br +0 -0
  79. package/assets/devtools/chunk.B4peH6PS.js.gz +0 -0
  80. package/assets/devtools/chunk.B8CNjZzU.js +0 -1
  81. package/assets/devtools/chunk.B8CNjZzU.js.br +0 -0
  82. package/assets/devtools/chunk.B8CNjZzU.js.gz +0 -0
  83. package/assets/devtools/chunk.Bgd10SVI.js +0 -1
  84. package/assets/devtools/chunk.Bgd10SVI.js.br +0 -0
  85. package/assets/devtools/chunk.Bgd10SVI.js.gz +0 -0
  86. package/assets/devtools/chunk.BjFrJKj1.js +0 -1
  87. package/assets/devtools/chunk.BjFrJKj1.js.br +0 -2
  88. package/assets/devtools/chunk.BjFrJKj1.js.gz +0 -0
  89. package/assets/devtools/chunk.BlqFPyLh.js +0 -1
  90. package/assets/devtools/chunk.BlqFPyLh.js.br +0 -0
  91. package/assets/devtools/chunk.BlqFPyLh.js.gz +0 -0
  92. package/assets/devtools/chunk.BqBNmfN9.js +0 -1
  93. package/assets/devtools/chunk.BqBNmfN9.js.br +0 -0
  94. package/assets/devtools/chunk.BqBNmfN9.js.gz +0 -0
  95. package/assets/devtools/chunk.Bt0_vkJm.js +0 -2
  96. package/assets/devtools/chunk.Bt0_vkJm.js.br +0 -0
  97. package/assets/devtools/chunk.Bt0_vkJm.js.gz +0 -0
  98. package/assets/devtools/chunk.C3GuU4pz.js +0 -2
  99. package/assets/devtools/chunk.C3GuU4pz.js.br +0 -0
  100. package/assets/devtools/chunk.C3GuU4pz.js.gz +0 -0
  101. package/assets/devtools/chunk.CGwoN_Mo.js +0 -1
  102. package/assets/devtools/chunk.CGwoN_Mo.js.br +0 -0
  103. package/assets/devtools/chunk.CGwoN_Mo.js.gz +0 -0
  104. package/assets/devtools/chunk.CJCvhHA7.js +0 -1
  105. package/assets/devtools/chunk.CJCvhHA7.js.br +0 -2
  106. package/assets/devtools/chunk.CJCvhHA7.js.gz +0 -0
  107. package/assets/devtools/chunk.CKr2VE6v.js +0 -1
  108. package/assets/devtools/chunk.CKr2VE6v.js.br +0 -0
  109. package/assets/devtools/chunk.CKr2VE6v.js.gz +0 -0
  110. package/assets/devtools/chunk.CLvTwbkw.js +0 -1
  111. package/assets/devtools/chunk.CLvTwbkw.js.br +0 -0
  112. package/assets/devtools/chunk.CLvTwbkw.js.gz +0 -0
  113. package/assets/devtools/chunk.CR13dZhE.js +0 -7
  114. package/assets/devtools/chunk.CR13dZhE.js.br +0 -0
  115. package/assets/devtools/chunk.CR13dZhE.js.gz +0 -0
  116. package/assets/devtools/chunk.C_C-cVqs.js +0 -1
  117. package/assets/devtools/chunk.C_C-cVqs.js.br +0 -1
  118. package/assets/devtools/chunk.C_C-cVqs.js.gz +0 -0
  119. package/assets/devtools/chunk.CjevPbPy.js +0 -1
  120. package/assets/devtools/chunk.CjevPbPy.js.br +0 -0
  121. package/assets/devtools/chunk.CjevPbPy.js.gz +0 -0
  122. package/assets/devtools/chunk.CkNMZqAe.js +0 -1
  123. package/assets/devtools/chunk.CkNMZqAe.js.br +0 -0
  124. package/assets/devtools/chunk.CkNMZqAe.js.gz +0 -0
  125. package/assets/devtools/chunk.Cl1Mlnqx.js +0 -1
  126. package/assets/devtools/chunk.Cl1Mlnqx.js.br +0 -0
  127. package/assets/devtools/chunk.Cl1Mlnqx.js.gz +0 -0
  128. package/assets/devtools/chunk.CyY8OGdZ.js +0 -1
  129. package/assets/devtools/chunk.CyY8OGdZ.js.br +0 -0
  130. package/assets/devtools/chunk.CyY8OGdZ.js.gz +0 -0
  131. package/assets/devtools/chunk.Cyx9kLqD.js +0 -1
  132. package/assets/devtools/chunk.Cyx9kLqD.js.br +0 -0
  133. package/assets/devtools/chunk.Cyx9kLqD.js.gz +0 -0
  134. package/assets/devtools/chunk.D1MGgxUI.js +0 -1
  135. package/assets/devtools/chunk.D1MGgxUI.js.br +0 -0
  136. package/assets/devtools/chunk.D1MGgxUI.js.gz +0 -0
  137. package/assets/devtools/chunk.D5Ci-dwk.js +0 -1
  138. package/assets/devtools/chunk.D5Ci-dwk.js.br +0 -0
  139. package/assets/devtools/chunk.D5Ci-dwk.js.gz +0 -0
  140. package/assets/devtools/chunk.DFrWQW5x.js +0 -9
  141. package/assets/devtools/chunk.DFrWQW5x.js.br +0 -0
  142. package/assets/devtools/chunk.DFrWQW5x.js.gz +0 -0
  143. package/assets/devtools/chunk.DaVlli3f.js +0 -1
  144. package/assets/devtools/chunk.DaVlli3f.js.br +0 -0
  145. package/assets/devtools/chunk.DaVlli3f.js.gz +0 -0
  146. package/assets/devtools/chunk.DdyBCs50.js +0 -1
  147. package/assets/devtools/chunk.DdyBCs50.js.br +0 -0
  148. package/assets/devtools/chunk.DdyBCs50.js.gz +0 -0
  149. package/assets/devtools/chunk.Dl0THvrP.js +0 -1
  150. package/assets/devtools/chunk.Dl0THvrP.js.br +0 -0
  151. package/assets/devtools/chunk.Dl0THvrP.js.gz +0 -0
  152. package/assets/devtools/chunk.DwUNDm68.js +0 -1
  153. package/assets/devtools/chunk.DwUNDm68.js.br +0 -0
  154. package/assets/devtools/chunk.DwUNDm68.js.gz +0 -0
  155. package/assets/devtools/chunk.DzDkh4C6.js +0 -1
  156. package/assets/devtools/chunk.DzDkh4C6.js.br +0 -0
  157. package/assets/devtools/chunk.DzDkh4C6.js.gz +0 -0
  158. package/assets/devtools/chunk.QTExp4CY.js +0 -1
  159. package/assets/devtools/chunk.QTExp4CY.js.br +0 -0
  160. package/assets/devtools/chunk.QTExp4CY.js.gz +0 -0
  161. package/assets/devtools/chunk.ReCPcJln.js +0 -1
  162. package/assets/devtools/chunk.ReCPcJln.js.br +0 -0
  163. package/assets/devtools/chunk.ReCPcJln.js.gz +0 -0
  164. package/assets/devtools/chunk.UEhIKOMY.js +0 -1
  165. package/assets/devtools/chunk.UEhIKOMY.js.br +0 -0
  166. package/assets/devtools/chunk.UEhIKOMY.js.gz +0 -0
  167. package/assets/devtools/chunk.mWQqK3dU.js +0 -1
  168. package/assets/devtools/chunk.mWQqK3dU.js.br +0 -0
  169. package/assets/devtools/chunk.mWQqK3dU.js.gz +0 -0
  170. package/assets/devtools/chunk.uyVen0u2.js +0 -1
  171. package/assets/devtools/chunk.uyVen0u2.js.br +0 -0
  172. package/assets/devtools/chunk.uyVen0u2.js.gz +0 -0
  173. package/assets/devtools/chunk.yLRX_cUF.js +0 -1
  174. package/assets/devtools/chunk.yLRX_cUF.js.br +0 -0
  175. package/assets/devtools/chunk.yLRX_cUF.js.gz +0 -0
  176. package/assets/devtools/chunk.zuZxBYZg.js +0 -1
  177. package/assets/devtools/chunk.zuZxBYZg.js.br +0 -0
  178. package/assets/devtools/chunk.zuZxBYZg.js.gz +0 -0
  179. package/assets/devtools/db.html +0 -21
  180. package/assets/devtools/db.html.br +0 -0
  181. package/assets/devtools/db.html.gz +0 -0
  182. package/assets/devtools/entry.Cry3rxEI.js +0 -79
  183. package/assets/devtools/entry.Cry3rxEI.js.br +0 -0
  184. package/assets/devtools/entry.Cry3rxEI.js.gz +0 -0
  185. package/assets/devtools/env.html +0 -21
  186. package/assets/devtools/env.html.br +0 -0
  187. package/assets/devtools/env.html.gz +0 -0
  188. package/assets/devtools/graph.html +0 -22
  189. package/assets/devtools/graph.html.br +0 -0
  190. package/assets/devtools/graph.html.gz +0 -0
  191. package/assets/devtools/index.html +0 -21
  192. package/assets/devtools/index.html.br +0 -0
  193. package/assets/devtools/index.html.gz +0 -0
  194. package/assets/devtools/logs.html +0 -21
  195. package/assets/devtools/logs.html.br +0 -0
  196. package/assets/devtools/logs.html.gz +0 -0
  197. package/assets/devtools/queues.html +0 -21
  198. package/assets/devtools/queues.html.br +0 -0
  199. package/assets/devtools/queues.html.gz +0 -0
  200. package/assets/devtools/topics.html +0 -21
  201. package/assets/devtools/topics.html.br +0 -0
  202. package/assets/devtools/topics.html.gz +0 -0
  203. package/src/api/DevToolsProvider.ts +0 -157
  204. package/src/api/providers/DevToolsDatabaseProvider.ts +0 -27
  205. package/src/api/repositories/LogRepository.ts +0 -8
  206. package/src/api/schemas/DevCommandMetadata.ts +0 -9
  207. package/src/ui/components/DevAtomsViewer.tsx +0 -637
  208. package/src/ui/components/DevCacheInspector.tsx +0 -423
  209. package/src/ui/components/DevDashboard.tsx +0 -38
  210. package/src/ui/components/DevEnvExplorer.tsx +0 -462
  211. package/src/ui/components/DevLogViewer.tsx +0 -252
  212. package/src/ui/components/DevQueueMonitor.tsx +0 -51
  213. package/src/ui/components/DevTopicsViewer.tsx +0 -686
  214. package/src/ui/components/actions/ActionGroup.tsx +0 -37
  215. package/src/ui/components/actions/ActionItem.tsx +0 -138
  216. package/src/ui/components/actions/DevActionsExplorer.tsx +0 -132
  217. package/src/ui/components/actions/MethodBadge.tsx +0 -18
  218. package/src/ui/components/actions/SchemaViewer.tsx +0 -21
  219. package/src/ui/components/actions/TryItPanel.tsx +0 -140
  220. package/src/ui/components/actions/constants.ts +0 -7
  221. package/src/ui/components/actions/helpers.ts +0 -18
  222. package/src/ui/components/actions/index.ts +0 -8
  223. package/src/ui/components/db/ColumnBadge.tsx +0 -55
  224. package/src/ui/components/db/DevDbStudio.tsx +0 -485
  225. package/src/ui/components/db/constants.ts +0 -11
  226. package/src/ui/components/db/index.ts +0 -4
  227. package/src/ui/components/db/types.ts +0 -7
  228. package/src/ui/styles.css +0 -1
  229. /package/src/{api/schemas → schemas}/DevAtomMetadata.ts +0 -0
  230. /package/src/{api/schemas → schemas}/DevBucketMetadata.ts +0 -0
  231. /package/src/{api/schemas → schemas}/DevCacheMetadata.ts +0 -0
  232. /package/src/{api/schemas → schemas}/DevEnvMetadata.ts +0 -0
  233. /package/src/{api/schemas → schemas}/DevModuleMetadata.ts +0 -0
  234. /package/src/{api/schemas → schemas}/DevProviderMetadata.ts +0 -0
  235. /package/src/{api/schemas → schemas}/DevQueueMetadata.ts +0 -0
  236. /package/src/{api/schemas → schemas}/DevRealmMetadata.ts +0 -0
  237. /package/src/{api/schemas → schemas}/DevRouteMetadata.ts +0 -0
  238. /package/src/{api/schemas → schemas}/DevSchedulerMetadata.ts +0 -0
@@ -0,0 +1,402 @@
1
+ import { ActionButton, Flex, TypeForm, ui } from "@alepha/ui";
2
+ import {
3
+ Badge,
4
+ Loader,
5
+ ScrollArea,
6
+ Text,
7
+ TextInput,
8
+ UnstyledButton,
9
+ } from "@mantine/core";
10
+ import {
11
+ IconDatabase,
12
+ IconPlus,
13
+ IconSearch,
14
+ IconTrash,
15
+ } from "@tabler/icons-react";
16
+ import { jsonSchemaToTypeBox, t } from "alepha";
17
+ import { useInject } from "alepha/react";
18
+ import { useForm } from "alepha/react/form";
19
+ import { useRouter, useRouterState } from "alepha/react/router";
20
+ import { HttpClient } from "alepha/server";
21
+ import { useCallback, useEffect, useMemo, useState } from "react";
22
+ import { TreeView, type TreeViewNode } from "../shared/TreeView.tsx";
23
+
24
+ const EMPTY_SCHEMA = t.object({});
25
+
26
+ const toTypeBoxSchema = (jsonSchema: any): any => {
27
+ if (!jsonSchema) return null;
28
+ try {
29
+ const converted = jsonSchemaToTypeBox(jsonSchema);
30
+ if (converted?.properties) return converted;
31
+ } catch {
32
+ // Schema conversion failed
33
+ }
34
+ return null;
35
+ };
36
+
37
+ const RecordForm = ({
38
+ entity,
39
+ record,
40
+ isNew,
41
+ onSave,
42
+ onDelete,
43
+ pkColumn,
44
+ }: {
45
+ entity: any;
46
+ record: any;
47
+ isNew: boolean;
48
+ onSave: (values: any) => void;
49
+ onDelete: () => void;
50
+ pkColumn: string;
51
+ }) => {
52
+ const schema = useMemo(() => {
53
+ const jsonSchema = isNew ? entity.insertSchema : entity.updateSchema;
54
+ return toTypeBoxSchema(jsonSchema) ?? EMPTY_SCHEMA;
55
+ }, [entity, isNew]);
56
+
57
+ const form = useForm(
58
+ {
59
+ schema,
60
+ handler: () => {},
61
+ initialValues: isNew ? undefined : record,
62
+ },
63
+ [schema, record, isNew],
64
+ );
65
+
66
+ return (
67
+ <Flex direction="column" gap="md">
68
+ <Flex justify="space-between">
69
+ <Text fz="sm" fw={600}>
70
+ {isNew
71
+ ? "New Record"
72
+ : `Edit Record (${pkColumn}: ${record?.[pkColumn]})`}
73
+ </Text>
74
+ <Flex gap="xs">
75
+ <ActionButton size="xs" onClick={() => onSave(form.currentValues)}>
76
+ {isNew ? "Create" : "Save"}
77
+ </ActionButton>
78
+ {!isNew && (
79
+ <ActionButton
80
+ size="sm"
81
+ variant="light"
82
+ intent="danger"
83
+ onClick={onDelete}
84
+ icon={<IconTrash size={14} />}
85
+ />
86
+ )}
87
+ </Flex>
88
+ </Flex>
89
+ <TypeForm form={form} skipSubmitButton skipFormElement columns={1} />
90
+ </Flex>
91
+ );
92
+ };
93
+
94
+ /** Parse /db/editor/:table/:id from pathname */
95
+ const parseEditorPath = (pathname: string) => {
96
+ const prefix = "/db/editor/";
97
+ if (!pathname.startsWith(prefix)) return { table: "", recordId: "" };
98
+ const rest = pathname.slice(prefix.length);
99
+ const slashIdx = rest.indexOf("/");
100
+ if (slashIdx === -1) return { table: decodeURIComponent(rest), recordId: "" };
101
+ return {
102
+ table: decodeURIComponent(rest.slice(0, slashIdx)),
103
+ recordId: decodeURIComponent(rest.slice(slashIdx + 1)),
104
+ };
105
+ };
106
+
107
+ export const DatabaseEditor = ({ entities }: { entities: any[] }) => {
108
+ const http = useInject(HttpClient);
109
+ const router = useRouter();
110
+ const state = useRouterState();
111
+ const [records, setRecords] = useState<any[]>([]);
112
+ const [loading, setLoading] = useState(false);
113
+ const [search, setSearch] = useState("");
114
+ const [pageInfo, setPageInfo] = useState<any>(null);
115
+
116
+ const { table: selectedEntity, recordId } = parseEditorPath(
117
+ state.url.pathname,
118
+ );
119
+ const isNew = recordId === "new";
120
+
121
+ const entity = entities.find((e) => e.name === selectedEntity);
122
+ const pkColumn =
123
+ entity?.columns?.find((c: any) => c.primaryKey)?.name ?? "id";
124
+
125
+ const fetchRecords = useCallback(async () => {
126
+ if (!selectedEntity) return;
127
+ setLoading(true);
128
+ try {
129
+ const res = await http.fetch(
130
+ `/__devtools/api/db/${selectedEntity}/records?size=50`,
131
+ );
132
+ setRecords((res.data as any)?.content ?? []);
133
+ setPageInfo((res.data as any)?.page);
134
+ } catch {
135
+ setRecords([]);
136
+ } finally {
137
+ setLoading(false);
138
+ }
139
+ }, [http, selectedEntity]);
140
+
141
+ useEffect(() => {
142
+ fetchRecords();
143
+ }, [selectedEntity, fetchRecords]);
144
+
145
+ const selectedRecord = useMemo(() => {
146
+ if (!recordId || isNew) return null;
147
+ return records.find((r) => String(r[pkColumn]) === recordId) ?? null;
148
+ }, [records, recordId, isNew, pkColumn]);
149
+
150
+ const navigateToEntity = useCallback(
151
+ (name: string) => {
152
+ router.push(`/db/editor/${encodeURIComponent(name)}`);
153
+ },
154
+ [router],
155
+ );
156
+
157
+ const navigateToRecord = useCallback(
158
+ (record: any) => {
159
+ const id = record[pkColumn];
160
+ router.push(
161
+ `/db/editor/${encodeURIComponent(selectedEntity)}/${encodeURIComponent(String(id))}`,
162
+ );
163
+ },
164
+ [router, selectedEntity, pkColumn],
165
+ );
166
+
167
+ const navigateToNew = useCallback(() => {
168
+ router.push(`/db/editor/${encodeURIComponent(selectedEntity)}/new`);
169
+ }, [router, selectedEntity]);
170
+
171
+ const handleSave = async (data: any) => {
172
+ try {
173
+ if (isNew) {
174
+ await http.fetch(`/__devtools/api/db/${selectedEntity}/records`, {
175
+ method: "POST",
176
+ body: JSON.stringify(data),
177
+ headers: { "Content-Type": "application/json" },
178
+ });
179
+ router.push(`/db/editor/${encodeURIComponent(selectedEntity)}`);
180
+ } else if (selectedRecord) {
181
+ const idValue = selectedRecord[pkColumn];
182
+ await http.fetch(
183
+ `/__devtools/api/db/${selectedEntity}/records/${idValue}`,
184
+ {
185
+ method: "PUT",
186
+ body: JSON.stringify(data),
187
+ headers: { "Content-Type": "application/json" },
188
+ },
189
+ );
190
+ }
191
+ await fetchRecords();
192
+ } catch {
193
+ // handle error
194
+ }
195
+ };
196
+
197
+ const handleDelete = async () => {
198
+ if (!selectedRecord) return;
199
+ const idValue = selectedRecord[pkColumn];
200
+ try {
201
+ await http.fetch(
202
+ `/__devtools/api/db/${selectedEntity}/records/${idValue}`,
203
+ { method: "DELETE" },
204
+ );
205
+ router.push(`/db/editor/${encodeURIComponent(selectedEntity)}`);
206
+ await fetchRecords();
207
+ } catch {
208
+ // handle error
209
+ }
210
+ };
211
+
212
+ const entityNodes: TreeViewNode[] = useMemo(() => {
213
+ const filtered = entities.filter((e) =>
214
+ e.name.toLowerCase().includes(search.toLowerCase()),
215
+ );
216
+ return filtered.map((e) => ({
217
+ id: `entity:${e.name}`,
218
+ label: e.name,
219
+ icon: (
220
+ <IconDatabase
221
+ size={13}
222
+ color="var(--mantine-color-blue-text)"
223
+ style={{ flexShrink: 0 }}
224
+ />
225
+ ),
226
+ badge: (
227
+ <Badge size="xs" variant="light" color="gray">
228
+ {e.columns?.length ?? 0}
229
+ </Badge>
230
+ ),
231
+ }));
232
+ }, [entities, search]);
233
+
234
+ const handleEntitySelect = useCallback(
235
+ (id: string) => {
236
+ const name = id.replace(/^entity:/, "");
237
+ navigateToEntity(name);
238
+ },
239
+ [navigateToEntity],
240
+ );
241
+
242
+ const emptyOpenNodes = useMemo(() => new Set<string>(), []);
243
+ const selectedEntityId = selectedEntity ? `entity:${selectedEntity}` : "";
244
+
245
+ return (
246
+ <Flex style={{ flex: 1, overflow: "hidden" }}>
247
+ {/* Entity list */}
248
+ <Flex
249
+ w={200}
250
+ style={{
251
+ borderRight: `1px solid ${ui.colors.border}`,
252
+ flexShrink: 0,
253
+ display: "flex",
254
+ flexDirection: "column",
255
+ }}
256
+ >
257
+ <Flex p="xs">
258
+ <TextInput
259
+ size="xs"
260
+ placeholder="Filter tables..."
261
+ leftSection={<IconSearch size={14} />}
262
+ value={search}
263
+ onChange={(e) => setSearch(e.currentTarget.value)}
264
+ />
265
+ </Flex>
266
+ <ScrollArea style={{ flex: 1 }} px="xs">
267
+ <TreeView
268
+ nodes={entityNodes}
269
+ selectedId={selectedEntityId}
270
+ openNodes={emptyOpenNodes}
271
+ onSelect={handleEntitySelect}
272
+ onToggle={() => {}}
273
+ showLeafCount={false}
274
+ />
275
+ </ScrollArea>
276
+ </Flex>
277
+
278
+ {/* Records list */}
279
+ {selectedEntity && (
280
+ <Flex
281
+ w={200}
282
+ style={{
283
+ borderRight: `1px solid ${ui.colors.border}`,
284
+ flexShrink: 0,
285
+ display: "flex",
286
+ flexDirection: "column",
287
+ }}
288
+ >
289
+ <Flex
290
+ px="xs"
291
+ py="xs"
292
+ style={{
293
+ borderBottom: `1px solid ${ui.colors.border}`,
294
+ flexShrink: 0,
295
+ }}
296
+ >
297
+ <Flex gap="xs" justify="space-between">
298
+ <Text fz={10} c="dimmed" tt="uppercase" fw={600} lts={0.5}>
299
+ Records{" "}
300
+ {pageInfo?.totalElements != null &&
301
+ `(${pageInfo.totalElements})`}
302
+ </Text>
303
+ <ActionButton
304
+ size="xs"
305
+ variant="subtle"
306
+ color="teal"
307
+ onClick={navigateToNew}
308
+ icon={<IconPlus size={14} />}
309
+ />
310
+ </Flex>
311
+ </Flex>
312
+ <ScrollArea style={{ flex: 1 }} px="xs" py="xs">
313
+ <UnstyledButton
314
+ w="100%"
315
+ py={4}
316
+ className="devtools-tree-node"
317
+ data-selected={isNew || undefined}
318
+ style={{
319
+ borderRadius: 6,
320
+ paddingLeft: 8,
321
+ paddingRight: 8,
322
+ transition: "background 100ms ease",
323
+ }}
324
+ onClick={navigateToNew}
325
+ >
326
+ <Flex gap={8} wrap="nowrap">
327
+ <IconPlus
328
+ size={13}
329
+ color="var(--mantine-color-teal-text)"
330
+ style={{ flexShrink: 0 }}
331
+ />
332
+ <Text fz={12} c="teal">
333
+ New Record
334
+ </Text>
335
+ </Flex>
336
+ </UnstyledButton>
337
+ {loading ? (
338
+ <Flex justify="center" py="md">
339
+ <Loader size="xs" />
340
+ </Flex>
341
+ ) : (
342
+ records.map((record, i) => {
343
+ const idVal = record[pkColumn] ?? i;
344
+ const isActive = !isNew && String(idVal) === recordId;
345
+ return (
346
+ <UnstyledButton
347
+ key={String(idVal)}
348
+ w="100%"
349
+ py={4}
350
+ className="devtools-tree-node"
351
+ data-selected={isActive || undefined}
352
+ style={{
353
+ borderRadius: 6,
354
+ paddingLeft: 8,
355
+ paddingRight: 8,
356
+ transition: "background 100ms ease",
357
+ }}
358
+ onClick={() => navigateToRecord(record)}
359
+ >
360
+ <Text fz={12} ff="monospace" truncate>
361
+ {pkColumn}: {String(idVal)}
362
+ </Text>
363
+ </UnstyledButton>
364
+ );
365
+ })
366
+ )}
367
+ </ScrollArea>
368
+ </Flex>
369
+ )}
370
+
371
+ {/* Editor panel */}
372
+ <Flex style={{ flex: 1, overflow: "auto" }} p="md">
373
+ {!selectedEntity && (
374
+ <Flex align="center" justify="center" h="100%">
375
+ <Text c="dimmed" fz="sm">
376
+ Select a table to browse records
377
+ </Text>
378
+ </Flex>
379
+ )}
380
+
381
+ {selectedEntity && !selectedRecord && !isNew && (
382
+ <Flex align="center" justify="center" h="100%">
383
+ <Text c="dimmed" fz="sm">
384
+ Select a record or create a new one
385
+ </Text>
386
+ </Flex>
387
+ )}
388
+
389
+ {(selectedRecord || isNew) && entity && (
390
+ <RecordForm
391
+ entity={entity}
392
+ record={selectedRecord}
393
+ isNew={isNew}
394
+ onSave={handleSave}
395
+ onDelete={handleDelete}
396
+ pkColumn={pkColumn}
397
+ />
398
+ )}
399
+ </Flex>
400
+ </Flex>
401
+ );
402
+ };
@@ -0,0 +1,28 @@
1
+ import { devMetadataSchema } from "@alepha/devtools";
2
+ import { Flex } from "@mantine/core";
3
+ import { useInject } from "alepha/react";
4
+ import { HttpClient } from "alepha/server";
5
+ import { useEffect, useState } from "react";
6
+ import { DatabaseErd } from "./DatabaseErd.tsx";
7
+
8
+ const DatabaseErdPage = () => {
9
+ const http = useInject(HttpClient);
10
+ const [entities, setEntities] = useState<any[]>([]);
11
+
12
+ useEffect(() => {
13
+ http
14
+ .fetch("/__devtools/api/metadata", {
15
+ schema: { response: devMetadataSchema },
16
+ })
17
+ .then((res) => setEntities(res.data.entities ?? []))
18
+ .catch(() => {});
19
+ }, [http]);
20
+
21
+ return (
22
+ <Flex flex={1} style={{ position: "relative" }}>
23
+ <DatabaseErd entities={entities} />
24
+ </Flex>
25
+ );
26
+ };
27
+
28
+ export default DatabaseErdPage;
@@ -0,0 +1,107 @@
1
+ import { ui } from "@alepha/ui";
2
+ import { Flex } from "@mantine/core";
3
+ import {
4
+ Background,
5
+ BackgroundVariant,
6
+ Controls,
7
+ type Edge,
8
+ MiniMap,
9
+ type Node,
10
+ ReactFlow,
11
+ ReactFlowProvider,
12
+ useEdgesState,
13
+ useNodesState,
14
+ } from "@xyflow/react";
15
+ import "@xyflow/react/dist/style.css";
16
+ import { useEffect, useMemo } from "react";
17
+ import { EntityNode } from "./EntityNode.tsx";
18
+
19
+ const nodeTypes = { entity: EntityNode };
20
+
21
+ const buildGraph = (entities: any[]): { nodes: Node[]; edges: Edge[] } => {
22
+ const nodes: Node[] = entities.map((entity, i) => {
23
+ const cols = Math.ceil(Math.sqrt(entities.length));
24
+ const row = Math.floor(i / cols);
25
+ const col = i % cols;
26
+ return {
27
+ id: entity.name,
28
+ type: "entity",
29
+ position: { x: col * 320, y: row * 300 },
30
+ data: entity,
31
+ };
32
+ });
33
+
34
+ const edges: Edge[] = [];
35
+ for (const entity of entities) {
36
+ for (const column of entity.columns ?? []) {
37
+ if (column.ref) {
38
+ edges.push({
39
+ id: `${entity.name}.${column.name}->${column.ref.entity}`,
40
+ source: entity.name,
41
+ target: column.ref.entity,
42
+ label: `${column.name} → ${column.ref.column}`,
43
+ style: { stroke: "var(--mantine-color-blue-6)", strokeWidth: 1.5 },
44
+ labelStyle: { fontSize: 10, fill: "var(--mantine-color-dimmed)" },
45
+ animated: true,
46
+ });
47
+ }
48
+ }
49
+ }
50
+
51
+ return { nodes, edges };
52
+ };
53
+
54
+ const ErdFlow = ({ entities }: { entities: any[] }) => {
55
+ const graph = useMemo(() => buildGraph(entities), [entities]);
56
+ const [nodes, setNodes, onNodesChange] = useNodesState(graph.nodes);
57
+ const [edges, setEdges, onEdgesChange] = useEdgesState(graph.edges);
58
+
59
+ useEffect(() => {
60
+ setNodes(graph.nodes);
61
+ setEdges(graph.edges);
62
+ }, [graph, setNodes, setEdges]);
63
+
64
+ return (
65
+ <ReactFlow
66
+ nodes={nodes}
67
+ edges={edges}
68
+ onNodesChange={onNodesChange}
69
+ onEdgesChange={onEdgesChange}
70
+ nodeTypes={nodeTypes}
71
+ fitView
72
+ fitViewOptions={{ padding: 0.2 }}
73
+ minZoom={0.1}
74
+ maxZoom={2}
75
+ proOptions={{ hideAttribution: true }}
76
+ style={{ background: ui.colors.background }}
77
+ >
78
+ <Background variant={BackgroundVariant.Dots} gap={20} size={1} />
79
+ <Controls />
80
+ <MiniMap
81
+ nodeColor="var(--mantine-color-dark-4)"
82
+ maskColor="rgba(0,0,0,0.6)"
83
+ style={{ backgroundColor: ui.colors.surface }}
84
+ />
85
+ </ReactFlow>
86
+ );
87
+ };
88
+
89
+ export const DatabaseErd = ({ entities }: { entities: any[] }) => {
90
+ return (
91
+ <Flex
92
+ flex={1}
93
+ h="100%"
94
+ style={{
95
+ position: "absolute",
96
+ top: 0,
97
+ left: 0,
98
+ right: 0,
99
+ bottom: 0,
100
+ }}
101
+ >
102
+ <ReactFlowProvider>
103
+ <ErdFlow entities={entities} />
104
+ </ReactFlowProvider>
105
+ </Flex>
106
+ );
107
+ };
@@ -0,0 +1,36 @@
1
+ import { Flex, ui } from "@alepha/ui";
2
+ import { SegmentedControl } from "@mantine/core";
3
+ import { NestedView, useRouter, useRouterState } from "alepha/react/router";
4
+
5
+ export const DevDatabase = () => {
6
+ const router = useRouter();
7
+ const state = useRouterState();
8
+ const tab = state.url.pathname.startsWith("/db/editor") ? "editor" : "erd";
9
+
10
+ const handleTabChange = (value: string) => {
11
+ router.push(value === "editor" ? "/db/editor" : "/db/erd");
12
+ };
13
+
14
+ return (
15
+ <Flex direction="column" style={{ flex: 1 }}>
16
+ <Flex
17
+ px="md"
18
+ py="xs"
19
+ style={{ borderBottom: `1px solid ${ui.colors.border}` }}
20
+ >
21
+ <SegmentedControl
22
+ size="xs"
23
+ value={tab}
24
+ onChange={handleTabChange}
25
+ data={[
26
+ { label: "ERD", value: "erd" },
27
+ { label: "Editor", value: "editor" },
28
+ ]}
29
+ />
30
+ </Flex>
31
+ <NestedView />
32
+ </Flex>
33
+ );
34
+ };
35
+
36
+ export default DevDatabase;
@@ -0,0 +1,83 @@
1
+ import { Flex, ui } from "@alepha/ui";
2
+ import { Badge, Text } from "@mantine/core";
3
+ import { IconKey, IconLink } from "@tabler/icons-react";
4
+ import { Handle, type NodeProps, Position } from "@xyflow/react";
5
+
6
+ export const EntityNode = ({ data }: NodeProps) => {
7
+ const entity = data as any;
8
+
9
+ return (
10
+ <div
11
+ style={{
12
+ background: ui.colors.surface,
13
+ border: `1px solid ${ui.colors.border}`,
14
+ borderRadius: 8,
15
+ minWidth: 240,
16
+ overflow: "hidden",
17
+ }}
18
+ >
19
+ <Handle
20
+ type="target"
21
+ position={Position.Top}
22
+ style={{ background: "var(--mantine-color-blue-6)" }}
23
+ />
24
+
25
+ {/* Header */}
26
+ <div
27
+ style={{
28
+ padding: "8px 12px",
29
+ borderBottom: `1px solid ${ui.colors.border}`,
30
+ background: ui.colors.elevated,
31
+ }}
32
+ >
33
+ <Flex gap="xs">
34
+ <Text fz="sm" fw={700}>
35
+ {entity.name}
36
+ </Text>
37
+ <Badge size="xs" variant="light" color="gray">
38
+ {entity.provider}
39
+ </Badge>
40
+ </Flex>
41
+ </div>
42
+
43
+ {/* Columns */}
44
+ <Flex direction="column" gap={0} p={0}>
45
+ {(entity.columns ?? []).map((col: any) => (
46
+ <Flex
47
+ key={col.name}
48
+ gap="xs"
49
+ px="sm"
50
+ py={4}
51
+ style={{ borderBottom: `1px solid ${ui.colors.border}20` }}
52
+ wrap="nowrap"
53
+ >
54
+ {col.primaryKey && (
55
+ <IconKey size={10} color="var(--mantine-color-yellow-6)" />
56
+ )}
57
+ {col.ref && (
58
+ <IconLink size={10} color="var(--mantine-color-blue-6)" />
59
+ )}
60
+ {!col.primaryKey && !col.ref && <span style={{ width: 10 }} />}
61
+ <Text fz={11} ff="monospace" style={{ flex: 1 }}>
62
+ {col.name}
63
+ </Text>
64
+ <Text fz={10} c="dimmed" ff="monospace">
65
+ {col.type}
66
+ </Text>
67
+ {col.nullable && (
68
+ <Text fz={9} c="dimmed">
69
+ ?
70
+ </Text>
71
+ )}
72
+ </Flex>
73
+ ))}
74
+ </Flex>
75
+
76
+ <Handle
77
+ type="source"
78
+ position={Position.Bottom}
79
+ style={{ background: "var(--mantine-color-blue-6)" }}
80
+ />
81
+ </div>
82
+ );
83
+ };