@alepha/devtools 0.16.0 → 0.19.1

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 (236) 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 +349 -321
  5. package/dist/index.js +293 -186
  6. package/dist/index.js.map +1 -1
  7. package/package.json +30 -23
  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 +280 -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 +55 -59
  21. package/src/ui/components/DevLayout.tsx +104 -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 +482 -0
  29. package/src/ui/components/database/DatabaseEditor.page.tsx +23 -0
  30. package/src/ui/components/database/DatabaseEditor.tsx +399 -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/explorer/DevExplorer.tsx +351 -0
  36. package/src/ui/components/explorer/ExplorerTree.tsx +178 -0
  37. package/src/ui/components/explorer/panels/DevPanelAction.tsx +499 -0
  38. package/src/ui/components/explorer/panels/DevPanelCache.tsx +73 -0
  39. package/src/ui/components/explorer/panels/DevPanelPage.tsx +96 -0
  40. package/src/ui/components/explorer/panels/DevPanelQueue.tsx +51 -0
  41. package/src/ui/components/explorer/panels/DevPanelTopic.tsx +56 -0
  42. package/src/ui/components/explorer/panels/index.ts +5 -0
  43. package/src/ui/components/graph/DevDependencyGraph.tsx +35 -60
  44. package/src/ui/components/graph/GraphControls.tsx +10 -11
  45. package/src/ui/components/graph/NodeDetails.tsx +22 -29
  46. package/src/ui/components/graph/ProviderNode.tsx +4 -4
  47. package/src/ui/components/graph/helpers.ts +1 -1
  48. package/src/ui/components/logs/DevLogs.tsx +661 -0
  49. package/src/ui/components/logs/index.ts +1 -0
  50. package/src/ui/components/shared/TreeView.tsx +189 -0
  51. package/src/ui/main.css +17 -0
  52. package/src/ui/main.ts +2 -6
  53. package/LICENSE +0 -21
  54. package/assets/devtools/actions.html +0 -21
  55. package/assets/devtools/actions.html.br +0 -0
  56. package/assets/devtools/actions.html.gz +0 -0
  57. package/assets/devtools/asset.BJSLFcNT.css +0 -1
  58. package/assets/devtools/asset.BJSLFcNT.css.br +0 -0
  59. package/assets/devtools/asset.BJSLFcNT.css.gz +0 -0
  60. package/assets/devtools/asset.BZV40eAE.css +0 -1
  61. package/assets/devtools/asset.BZV40eAE.css.br +0 -0
  62. package/assets/devtools/asset.BZV40eAE.css.gz +0 -0
  63. package/assets/devtools/atoms.html +0 -21
  64. package/assets/devtools/atoms.html.br +0 -0
  65. package/assets/devtools/atoms.html.gz +0 -0
  66. package/assets/devtools/caches.html +0 -21
  67. package/assets/devtools/caches.html.br +0 -0
  68. package/assets/devtools/caches.html.gz +0 -0
  69. package/assets/devtools/chunk.1h5GuATm.js +0 -1
  70. package/assets/devtools/chunk.1h5GuATm.js.br +0 -0
  71. package/assets/devtools/chunk.1h5GuATm.js.gz +0 -0
  72. package/assets/devtools/chunk.3PgxxOdM.js +0 -1
  73. package/assets/devtools/chunk.3PgxxOdM.js.br +0 -0
  74. package/assets/devtools/chunk.3PgxxOdM.js.gz +0 -0
  75. package/assets/devtools/chunk.A_W3H6Aa.js +0 -1
  76. package/assets/devtools/chunk.A_W3H6Aa.js.br +0 -0
  77. package/assets/devtools/chunk.A_W3H6Aa.js.gz +0 -0
  78. package/assets/devtools/chunk.B5tL0VjH.js +0 -1
  79. package/assets/devtools/chunk.B5tL0VjH.js.br +0 -0
  80. package/assets/devtools/chunk.B5tL0VjH.js.gz +0 -0
  81. package/assets/devtools/chunk.B8p_Szro.js +0 -1
  82. package/assets/devtools/chunk.B8p_Szro.js.br +0 -0
  83. package/assets/devtools/chunk.B8p_Szro.js.gz +0 -0
  84. package/assets/devtools/chunk.BANy8c2v.js +0 -1
  85. package/assets/devtools/chunk.BANy8c2v.js.br +0 -0
  86. package/assets/devtools/chunk.BANy8c2v.js.gz +0 -0
  87. package/assets/devtools/chunk.BKph0hv1.js +0 -1
  88. package/assets/devtools/chunk.BKph0hv1.js.br +0 -0
  89. package/assets/devtools/chunk.BKph0hv1.js.gz +0 -0
  90. package/assets/devtools/chunk.BUs1kuwE.js +0 -1
  91. package/assets/devtools/chunk.BUs1kuwE.js.br +0 -0
  92. package/assets/devtools/chunk.BUs1kuwE.js.gz +0 -0
  93. package/assets/devtools/chunk.BVIEr21R.js +0 -1
  94. package/assets/devtools/chunk.BVIEr21R.js.br +0 -0
  95. package/assets/devtools/chunk.BVIEr21R.js.gz +0 -0
  96. package/assets/devtools/chunk.Bb3re2d8.js +0 -1
  97. package/assets/devtools/chunk.Bb3re2d8.js.br +0 -2
  98. package/assets/devtools/chunk.Bb3re2d8.js.gz +0 -0
  99. package/assets/devtools/chunk.BjFrJKj1.js +0 -1
  100. package/assets/devtools/chunk.BjFrJKj1.js.br +0 -2
  101. package/assets/devtools/chunk.BjFrJKj1.js.gz +0 -0
  102. package/assets/devtools/chunk.BkXzz14p.js +0 -1
  103. package/assets/devtools/chunk.BkXzz14p.js.br +0 -0
  104. package/assets/devtools/chunk.BkXzz14p.js.gz +0 -0
  105. package/assets/devtools/chunk.BlqFPyLh.js +0 -1
  106. package/assets/devtools/chunk.BlqFPyLh.js.br +0 -0
  107. package/assets/devtools/chunk.BlqFPyLh.js.gz +0 -0
  108. package/assets/devtools/chunk.BymZ9jU5.js +0 -1
  109. package/assets/devtools/chunk.BymZ9jU5.js.br +0 -0
  110. package/assets/devtools/chunk.BymZ9jU5.js.gz +0 -0
  111. package/assets/devtools/chunk.C0BD3Ujz.js +0 -1
  112. package/assets/devtools/chunk.C0BD3Ujz.js.br +0 -0
  113. package/assets/devtools/chunk.C0BD3Ujz.js.gz +0 -0
  114. package/assets/devtools/chunk.C63rzhbT.js +0 -1
  115. package/assets/devtools/chunk.C63rzhbT.js.br +0 -0
  116. package/assets/devtools/chunk.C63rzhbT.js.gz +0 -0
  117. package/assets/devtools/chunk.CJrYVzjN.js +0 -9
  118. package/assets/devtools/chunk.CJrYVzjN.js.br +0 -0
  119. package/assets/devtools/chunk.CJrYVzjN.js.gz +0 -0
  120. package/assets/devtools/chunk.CPGX3Xpx.js +0 -1
  121. package/assets/devtools/chunk.CPGX3Xpx.js.br +0 -1
  122. package/assets/devtools/chunk.CPGX3Xpx.js.gz +0 -0
  123. package/assets/devtools/chunk.Cf-3skUw.js +0 -1
  124. package/assets/devtools/chunk.Cf-3skUw.js.br +0 -0
  125. package/assets/devtools/chunk.Cf-3skUw.js.gz +0 -0
  126. package/assets/devtools/chunk.D7JLxcoJ.js +0 -7
  127. package/assets/devtools/chunk.D7JLxcoJ.js.br +0 -0
  128. package/assets/devtools/chunk.D7JLxcoJ.js.gz +0 -0
  129. package/assets/devtools/chunk.D7e5mBY4.js +0 -1
  130. package/assets/devtools/chunk.D7e5mBY4.js.br +0 -0
  131. package/assets/devtools/chunk.D7e5mBY4.js.gz +0 -0
  132. package/assets/devtools/chunk.DClU9Z1_.js +0 -1
  133. package/assets/devtools/chunk.DClU9Z1_.js.br +0 -0
  134. package/assets/devtools/chunk.DClU9Z1_.js.gz +0 -0
  135. package/assets/devtools/chunk.DE_M8b3Z.js +0 -1
  136. package/assets/devtools/chunk.DE_M8b3Z.js.br +0 -0
  137. package/assets/devtools/chunk.DE_M8b3Z.js.gz +0 -0
  138. package/assets/devtools/chunk.DWASJDBE.js +0 -1
  139. package/assets/devtools/chunk.DWASJDBE.js.br +0 -0
  140. package/assets/devtools/chunk.DWASJDBE.js.gz +0 -0
  141. package/assets/devtools/chunk.DfzRLqwW.js +0 -1
  142. package/assets/devtools/chunk.DfzRLqwW.js.br +0 -0
  143. package/assets/devtools/chunk.DfzRLqwW.js.gz +0 -0
  144. package/assets/devtools/chunk.Dww1YQtc.js +0 -1
  145. package/assets/devtools/chunk.Dww1YQtc.js.br +0 -0
  146. package/assets/devtools/chunk.Dww1YQtc.js.gz +0 -0
  147. package/assets/devtools/chunk.HFLdduaf.js +0 -1
  148. package/assets/devtools/chunk.HFLdduaf.js.br +0 -0
  149. package/assets/devtools/chunk.HFLdduaf.js.gz +0 -0
  150. package/assets/devtools/chunk.J-htqECs.js +0 -1
  151. package/assets/devtools/chunk.J-htqECs.js.br +0 -2
  152. package/assets/devtools/chunk.J-htqECs.js.gz +0 -0
  153. package/assets/devtools/chunk.JjTGVewZ.js +0 -2
  154. package/assets/devtools/chunk.JjTGVewZ.js.br +0 -0
  155. package/assets/devtools/chunk.JjTGVewZ.js.gz +0 -0
  156. package/assets/devtools/chunk.OV_89czZ.js +0 -1
  157. package/assets/devtools/chunk.OV_89czZ.js.br +0 -0
  158. package/assets/devtools/chunk.OV_89czZ.js.gz +0 -0
  159. package/assets/devtools/chunk.YFkMUqFM.js +0 -1
  160. package/assets/devtools/chunk.YFkMUqFM.js.br +0 -0
  161. package/assets/devtools/chunk.YFkMUqFM.js.gz +0 -0
  162. package/assets/devtools/chunk._KdUFIrt.js +0 -1
  163. package/assets/devtools/chunk._KdUFIrt.js.br +0 -0
  164. package/assets/devtools/chunk._KdUFIrt.js.gz +0 -0
  165. package/assets/devtools/chunk.pjP6xqG8.js +0 -1
  166. package/assets/devtools/chunk.pjP6xqG8.js.br +0 -0
  167. package/assets/devtools/chunk.pjP6xqG8.js.gz +0 -0
  168. package/assets/devtools/chunk.uTFtY0ae.js +0 -2
  169. package/assets/devtools/chunk.uTFtY0ae.js.br +0 -0
  170. package/assets/devtools/chunk.uTFtY0ae.js.gz +0 -0
  171. package/assets/devtools/chunk.uyVen0u2.js +0 -1
  172. package/assets/devtools/chunk.uyVen0u2.js.br +0 -0
  173. package/assets/devtools/chunk.uyVen0u2.js.gz +0 -0
  174. package/assets/devtools/chunk.vHjNjQS8.js +0 -1
  175. package/assets/devtools/chunk.vHjNjQS8.js.br +0 -0
  176. package/assets/devtools/chunk.vHjNjQS8.js.gz +0 -0
  177. package/assets/devtools/db.html +0 -21
  178. package/assets/devtools/db.html.br +0 -0
  179. package/assets/devtools/db.html.gz +0 -0
  180. package/assets/devtools/entry.DhzNl8q_.js +0 -79
  181. package/assets/devtools/entry.DhzNl8q_.js.br +0 -0
  182. package/assets/devtools/entry.DhzNl8q_.js.gz +0 -0
  183. package/assets/devtools/env.html +0 -21
  184. package/assets/devtools/env.html.br +0 -0
  185. package/assets/devtools/env.html.gz +0 -0
  186. package/assets/devtools/graph.html +0 -22
  187. package/assets/devtools/graph.html.br +0 -0
  188. package/assets/devtools/graph.html.gz +0 -0
  189. package/assets/devtools/index.html +0 -21
  190. package/assets/devtools/index.html.br +0 -0
  191. package/assets/devtools/index.html.gz +0 -0
  192. package/assets/devtools/logs.html +0 -21
  193. package/assets/devtools/logs.html.br +0 -0
  194. package/assets/devtools/logs.html.gz +0 -0
  195. package/assets/devtools/queues.html +0 -21
  196. package/assets/devtools/queues.html.br +0 -0
  197. package/assets/devtools/queues.html.gz +0 -0
  198. package/assets/devtools/topics.html +0 -21
  199. package/assets/devtools/topics.html.br +0 -0
  200. package/assets/devtools/topics.html.gz +0 -0
  201. package/src/api/DevToolsProvider.ts +0 -157
  202. package/src/api/providers/DevToolsDatabaseProvider.ts +0 -27
  203. package/src/api/repositories/LogRepository.ts +0 -8
  204. package/src/api/schemas/DevCommandMetadata.ts +0 -9
  205. package/src/ui/components/DevAtomsViewer.tsx +0 -637
  206. package/src/ui/components/DevCacheInspector.tsx +0 -423
  207. package/src/ui/components/DevDashboard.tsx +0 -38
  208. package/src/ui/components/DevEnvExplorer.tsx +0 -462
  209. package/src/ui/components/DevLogViewer.tsx +0 -252
  210. package/src/ui/components/DevQueueMonitor.tsx +0 -51
  211. package/src/ui/components/DevTopicsViewer.tsx +0 -686
  212. package/src/ui/components/actions/ActionGroup.tsx +0 -37
  213. package/src/ui/components/actions/ActionItem.tsx +0 -138
  214. package/src/ui/components/actions/DevActionsExplorer.tsx +0 -132
  215. package/src/ui/components/actions/MethodBadge.tsx +0 -18
  216. package/src/ui/components/actions/SchemaViewer.tsx +0 -21
  217. package/src/ui/components/actions/TryItPanel.tsx +0 -140
  218. package/src/ui/components/actions/constants.ts +0 -7
  219. package/src/ui/components/actions/helpers.ts +0 -18
  220. package/src/ui/components/actions/index.ts +0 -8
  221. package/src/ui/components/db/ColumnBadge.tsx +0 -55
  222. package/src/ui/components/db/DevDbStudio.tsx +0 -485
  223. package/src/ui/components/db/constants.ts +0 -11
  224. package/src/ui/components/db/index.ts +0 -4
  225. package/src/ui/components/db/types.ts +0 -7
  226. package/src/ui/styles.css +0 -1
  227. /package/src/{api/schemas → schemas}/DevAtomMetadata.ts +0 -0
  228. /package/src/{api/schemas → schemas}/DevBucketMetadata.ts +0 -0
  229. /package/src/{api/schemas → schemas}/DevCacheMetadata.ts +0 -0
  230. /package/src/{api/schemas → schemas}/DevEnvMetadata.ts +0 -0
  231. /package/src/{api/schemas → schemas}/DevModuleMetadata.ts +0 -0
  232. /package/src/{api/schemas → schemas}/DevProviderMetadata.ts +0 -0
  233. /package/src/{api/schemas → schemas}/DevQueueMetadata.ts +0 -0
  234. /package/src/{api/schemas → schemas}/DevRealmMetadata.ts +0 -0
  235. /package/src/{api/schemas → schemas}/DevRouteMetadata.ts +0 -0
  236. /package/src/{api/schemas → schemas}/DevSchedulerMetadata.ts +0 -0
@@ -0,0 +1,482 @@
1
+ import { devMetadataSchema } from "@alepha/devtools";
2
+ import { Flex, ui } from "@alepha/ui";
3
+ import {
4
+ Badge,
5
+ Card,
6
+ Grid,
7
+ ScrollArea,
8
+ SimpleGrid,
9
+ Text,
10
+ Title,
11
+ } from "@mantine/core";
12
+ import {
13
+ IconApi,
14
+ IconAtom,
15
+ IconBrandNodejs,
16
+ IconBroadcast,
17
+ IconClock,
18
+ IconCpu,
19
+ IconDatabase,
20
+ IconFileText,
21
+ IconPlug,
22
+ IconRoute,
23
+ IconServer,
24
+ IconSettings,
25
+ IconStack2,
26
+ IconVariable,
27
+ } from "@tabler/icons-react";
28
+ import { useInject } from "alepha/react";
29
+ import { useRouter } from "alepha/react/router";
30
+ import { HttpClient } from "alepha/server";
31
+ import { useCallback, useEffect, useMemo, useState } from "react";
32
+
33
+ interface LogEntry {
34
+ level: string;
35
+ message: string;
36
+ module: string;
37
+ service: string;
38
+ data?: any;
39
+ timestamp: number;
40
+ }
41
+
42
+ const formatUptime = (seconds: number): string => {
43
+ if (seconds < 60) return `${Math.floor(seconds)}s`;
44
+ if (seconds < 3600)
45
+ return `${Math.floor(seconds / 60)}m ${Math.floor(seconds % 60)}s`;
46
+ const h = Math.floor(seconds / 3600);
47
+ const m = Math.floor((seconds % 3600) / 60);
48
+ return `${h}h ${m}m`;
49
+ };
50
+
51
+ const formatBytes = (bytes: number): string => {
52
+ if (bytes < 1024) return `${bytes} B`;
53
+ if (bytes < 1024 * 1024) return `${(bytes / 1024).toFixed(1)} KB`;
54
+ return `${(bytes / (1024 * 1024)).toFixed(1)} MB`;
55
+ };
56
+
57
+ const levelColors: Record<string, string> = {
58
+ ERROR: "red",
59
+ WARN: "yellow",
60
+ INFO: "blue",
61
+ DEBUG: "gray",
62
+ TRACE: "dimmed",
63
+ };
64
+
65
+ const eventTypeColors: Record<string, string> = {
66
+ http: "teal",
67
+ db: "violet",
68
+ error: "red",
69
+ };
70
+
71
+ const detectEventType = (data: any): string | undefined => {
72
+ if (!data || typeof data !== "object") return undefined;
73
+ if (data.status && data.method && data.path && data.duration) return "http";
74
+ if (data.type === "db:query") return "db";
75
+ return undefined;
76
+ };
77
+
78
+ const formatEvent = (entry: LogEntry): string => {
79
+ const data = entry.data;
80
+ const eventType = detectEventType(data);
81
+ if (eventType === "http") {
82
+ return `${data.method} ${data.path} → ${data.status} (${data.duration}ms)`;
83
+ }
84
+ if (eventType === "db") {
85
+ return `${data.operation} (${Math.round(data.duration)}ms)`;
86
+ }
87
+ return entry.message;
88
+ };
89
+
90
+ export const DevDashboard = () => {
91
+ const http = useInject(HttpClient);
92
+ const router = useRouter();
93
+ const [metadata, setMetadata] = useState<any>(null);
94
+ const [logs, setLogs] = useState<LogEntry[]>([]);
95
+ const [events, setEvents] = useState<LogEntry[]>([]);
96
+
97
+ const fetchData = useCallback(async () => {
98
+ if (document.visibilityState !== "visible") return;
99
+ try {
100
+ const [metaRes, logsRes, eventsRes] = await Promise.all([
101
+ http.fetch("/__devtools/api/metadata", {
102
+ schema: { response: devMetadataSchema },
103
+ }),
104
+ http.fetch("/__devtools/api/logs?limit=10&level=DEBUG"),
105
+ http.fetch("/__devtools/api/logs?limit=20&type=http,db"),
106
+ ]);
107
+ setMetadata(metaRes.data);
108
+ setLogs((logsRes.data as any)?.logs ?? []);
109
+ setEvents((eventsRes.data as any)?.logs ?? []);
110
+ } catch {
111
+ // silently fail
112
+ }
113
+ }, [http]);
114
+
115
+ useEffect(() => {
116
+ fetchData();
117
+ const interval = setInterval(fetchData, 10_000);
118
+ return () => clearInterval(interval);
119
+ }, [fetchData]);
120
+
121
+ const system = metadata?.system;
122
+
123
+ const primitiveStats = useMemo(() => {
124
+ if (!metadata) return [];
125
+ return [
126
+ {
127
+ label: "Actions",
128
+ count: metadata.actions?.length ?? 0,
129
+ icon: IconApi,
130
+ color: "teal",
131
+ },
132
+ {
133
+ label: "Pages",
134
+ count: metadata.pages?.length ?? 0,
135
+ icon: IconFileText,
136
+ color: "blue",
137
+ },
138
+ {
139
+ label: "Entities",
140
+ count: metadata.entities?.length ?? 0,
141
+ icon: IconDatabase,
142
+ color: "violet",
143
+ },
144
+ {
145
+ label: "Queues",
146
+ count: metadata.queues?.length ?? 0,
147
+ icon: IconStack2,
148
+ color: "orange",
149
+ },
150
+ {
151
+ label: "Topics",
152
+ count: metadata.topics?.length ?? 0,
153
+ icon: IconBroadcast,
154
+ color: "pink",
155
+ },
156
+ {
157
+ label: "Caches",
158
+ count: metadata.caches?.length ?? 0,
159
+ icon: IconRoute,
160
+ color: "cyan",
161
+ },
162
+ {
163
+ label: "Providers",
164
+ count: metadata.providers?.length ?? 0,
165
+ icon: IconPlug,
166
+ color: "grape",
167
+ },
168
+ {
169
+ label: "Atoms",
170
+ count: metadata.atoms?.length ?? 0,
171
+ icon: IconAtom,
172
+ color: "lime",
173
+ },
174
+ {
175
+ label: "Env Vars",
176
+ count: metadata.envs?.length ?? 0,
177
+ icon: IconVariable,
178
+ color: "yellow",
179
+ },
180
+ ];
181
+ }, [metadata]);
182
+
183
+ return (
184
+ <Flex
185
+ p="lg"
186
+ w={"100%"}
187
+ flex={1}
188
+ direction="column"
189
+ style={{ overflow: "auto" }}
190
+ >
191
+ <Flex direction="column" gap="lg">
192
+ {/* App Stats */}
193
+ <div>
194
+ <Title
195
+ order={5}
196
+ mb="sm"
197
+ c="dimmed"
198
+ tt="uppercase"
199
+ fz="xs"
200
+ fw={600}
201
+ lts={1}
202
+ >
203
+ System
204
+ </Title>
205
+ <SimpleGrid cols={{ base: 2, sm: 3, md: 4, lg: 6 }} spacing="sm">
206
+ {system && (
207
+ <>
208
+ <StatCard
209
+ icon={IconServer}
210
+ label="Runtime"
211
+ value={system.runtime === "bun" ? "Bun" : "Node.js"}
212
+ />
213
+ <StatCard
214
+ icon={IconBrandNodejs}
215
+ label="Version"
216
+ value={system.nodeVersion}
217
+ />
218
+ <StatCard
219
+ icon={IconSettings}
220
+ label="Mode"
221
+ value={system.mode}
222
+ />
223
+ <StatCard
224
+ icon={IconRoute}
225
+ label="Port"
226
+ value={String(system.port)}
227
+ />
228
+ <StatCard
229
+ icon={IconClock}
230
+ label="Uptime"
231
+ value={formatUptime(system.uptime)}
232
+ />
233
+ <StatCard
234
+ icon={IconCpu}
235
+ label="Memory"
236
+ value={formatBytes(system.memoryUsage)}
237
+ />
238
+ </>
239
+ )}
240
+ </SimpleGrid>
241
+ </div>
242
+
243
+ <Grid gutter="lg">
244
+ {/* Recent Events */}
245
+ <Grid.Col span={{ base: 12, md: 7 }}>
246
+ <Card
247
+ padding="md"
248
+ radius="md"
249
+ style={{
250
+ background: ui.colors.surface,
251
+ border: `1px solid ${ui.colors.border}`,
252
+ }}
253
+ >
254
+ <Title
255
+ order={5}
256
+ mb="sm"
257
+ c="dimmed"
258
+ tt="uppercase"
259
+ fz="xs"
260
+ fw={600}
261
+ lts={1}
262
+ >
263
+ Recent Events
264
+ </Title>
265
+ <ScrollArea h={280}>
266
+ <Flex direction="column" gap={4}>
267
+ {events.length === 0 && (
268
+ <Text c="dimmed" fz="sm" ta="center" py="xl">
269
+ No events yet
270
+ </Text>
271
+ )}
272
+ {events.map((entry, i) => {
273
+ const eventType = detectEventType(entry.data) ?? "log";
274
+ return (
275
+ <Flex
276
+ key={`${entry.timestamp}-${i}`}
277
+ gap="xs"
278
+ px="xs"
279
+ py={4}
280
+ style={{
281
+ borderRadius: 4,
282
+ cursor: "pointer",
283
+ transition: "background 150ms",
284
+ }}
285
+ onMouseEnter={(e) => {
286
+ (e.currentTarget as HTMLElement).style.background =
287
+ `${ui.colors.elevated}`;
288
+ }}
289
+ onMouseLeave={(e) => {
290
+ (e.currentTarget as HTMLElement).style.background =
291
+ "transparent";
292
+ }}
293
+ >
294
+ <Text
295
+ fz="xs"
296
+ c="dimmed"
297
+ ff="monospace"
298
+ w={55}
299
+ style={{ flexShrink: 0 }}
300
+ >
301
+ {new Date(entry.timestamp).toLocaleTimeString("en", {
302
+ hour12: false,
303
+ })}
304
+ </Text>
305
+ <Badge
306
+ size="xs"
307
+ variant="light"
308
+ color={eventTypeColors[eventType] ?? "gray"}
309
+ style={{ flexShrink: 0 }}
310
+ >
311
+ {eventType === "http"
312
+ ? "HTTP"
313
+ : eventType === "db"
314
+ ? "DB"
315
+ : "LOG"}
316
+ </Badge>
317
+ <Text
318
+ fz="xs"
319
+ ff="monospace"
320
+ truncate
321
+ style={{ flex: 1 }}
322
+ >
323
+ {formatEvent(entry)}
324
+ </Text>
325
+ </Flex>
326
+ );
327
+ })}
328
+ </Flex>
329
+ </ScrollArea>
330
+ </Card>
331
+ </Grid.Col>
332
+
333
+ {/* Recent Logs */}
334
+ <Grid.Col span={{ base: 12, md: 5 }}>
335
+ <Card
336
+ padding="md"
337
+ radius="md"
338
+ style={{
339
+ background: ui.colors.surface,
340
+ border: `1px solid ${ui.colors.border}`,
341
+ }}
342
+ >
343
+ <Flex justify="space-between" mb="sm">
344
+ <Title
345
+ order={5}
346
+ c="dimmed"
347
+ tt="uppercase"
348
+ fz="xs"
349
+ fw={600}
350
+ lts={1}
351
+ >
352
+ Recent Logs
353
+ </Title>
354
+ <Text
355
+ fz="xs"
356
+ c="blue"
357
+ style={{ cursor: "pointer" }}
358
+ onClick={() => router.push("/logs")}
359
+ >
360
+ View all
361
+ </Text>
362
+ </Flex>
363
+ <ScrollArea h={280}>
364
+ <Flex direction="column" gap={4}>
365
+ {logs.length === 0 && (
366
+ <Text c="dimmed" fz="sm" ta="center" py="xl">
367
+ No logs yet
368
+ </Text>
369
+ )}
370
+ {logs.map((entry, i) => (
371
+ <Flex
372
+ key={`${entry.timestamp}-${i}`}
373
+ gap="xs"
374
+ px="xs"
375
+ py={4}
376
+ style={{ borderRadius: 4 }}
377
+ >
378
+ <Badge
379
+ size="xs"
380
+ variant="light"
381
+ color={levelColors[entry.level] ?? "gray"}
382
+ w={50}
383
+ style={{ flexShrink: 0 }}
384
+ >
385
+ {entry.level}
386
+ </Badge>
387
+ <Text
388
+ fz="xs"
389
+ c="dimmed"
390
+ w={80}
391
+ truncate
392
+ style={{ flexShrink: 0 }}
393
+ >
394
+ {entry.module}
395
+ </Text>
396
+ <Text fz="xs" ff="monospace" truncate style={{ flex: 1 }}>
397
+ {entry.message}
398
+ </Text>
399
+ </Flex>
400
+ ))}
401
+ </Flex>
402
+ </ScrollArea>
403
+ </Card>
404
+ </Grid.Col>
405
+ </Grid>
406
+
407
+ {/* Primitives Stats */}
408
+ <div>
409
+ <Title
410
+ order={5}
411
+ mb="sm"
412
+ c="dimmed"
413
+ tt="uppercase"
414
+ fz="xs"
415
+ fw={600}
416
+ lts={1}
417
+ >
418
+ Primitives
419
+ </Title>
420
+ <SimpleGrid cols={{ base: 2, sm: 3, md: 5 }} spacing="sm">
421
+ {primitiveStats.map((stat) => (
422
+ <Card
423
+ key={stat.label}
424
+ padding="sm"
425
+ radius="md"
426
+ style={{
427
+ background: ui.colors.surface,
428
+ border: `1px solid ${ui.colors.border}`,
429
+ }}
430
+ >
431
+ <Flex gap="sm">
432
+ <stat.icon size={18} opacity={0.5} />
433
+ <div style={{ flex: 1 }}>
434
+ <Text fz="xs" c="dimmed">
435
+ {stat.label}
436
+ </Text>
437
+ <Text fz="lg" fw={700} lh={1.2}>
438
+ {stat.count}
439
+ </Text>
440
+ </div>
441
+ </Flex>
442
+ </Card>
443
+ ))}
444
+ </SimpleGrid>
445
+ </div>
446
+ </Flex>
447
+ </Flex>
448
+ );
449
+ };
450
+
451
+ const StatCard = ({
452
+ icon: Icon,
453
+ label,
454
+ value,
455
+ }: {
456
+ icon: any;
457
+ label: string;
458
+ value: string;
459
+ }) => (
460
+ <Card
461
+ padding="sm"
462
+ radius="md"
463
+ style={{
464
+ background: ui.colors.surface,
465
+ border: `1px solid ${ui.colors.border}`,
466
+ }}
467
+ >
468
+ <Flex gap="xs">
469
+ <Icon size={16} opacity={0.4} />
470
+ <div>
471
+ <Text fz={10} c="dimmed" tt="uppercase" lts={0.5}>
472
+ {label}
473
+ </Text>
474
+ <Text fz="sm" fw={600} ff="monospace" lh={1.3}>
475
+ {value}
476
+ </Text>
477
+ </div>
478
+ </Flex>
479
+ </Card>
480
+ );
481
+
482
+ export default DevDashboard;
@@ -0,0 +1,23 @@
1
+ import { devMetadataSchema } from "@alepha/devtools";
2
+ import { useInject } from "alepha/react";
3
+ import { HttpClient } from "alepha/server";
4
+ import { useEffect, useState } from "react";
5
+ import { DatabaseEditor } from "./DatabaseEditor.tsx";
6
+
7
+ const DatabaseEditorPage = () => {
8
+ const http = useInject(HttpClient);
9
+ const [entities, setEntities] = useState<any[]>([]);
10
+
11
+ useEffect(() => {
12
+ http
13
+ .fetch("/__devtools/api/metadata", {
14
+ schema: { response: devMetadataSchema },
15
+ })
16
+ .then((res) => setEntities(res.data.entities ?? []))
17
+ .catch(() => {});
18
+ }, [http]);
19
+
20
+ return <DatabaseEditor entities={entities} />;
21
+ };
22
+
23
+ export default DatabaseEditorPage;