@alepha/ui 0.18.3 → 0.19.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 (163) hide show
  1. package/dist/admin/{AdminApiKeys-Dy_k-4Vd.js → AdminApiKeys-Bt1PjO6o.js} +3 -4
  2. package/dist/admin/{AdminApiKeys-Dy_k-4Vd.js.map → AdminApiKeys-Bt1PjO6o.js.map} +1 -1
  3. package/dist/admin/{AdminAudits-CKiFMSSU.js → AdminAudits-C7c1CN4c.js} +3 -4
  4. package/dist/admin/{AdminAudits-CKiFMSSU.js.map → AdminAudits-C7c1CN4c.js.map} +1 -1
  5. package/dist/admin/{AdminDashboard-PhC_dZqo.js → AdminDashboard-C3RXpTp6.js} +3 -4
  6. package/dist/admin/{AdminDashboard-PhC_dZqo.js.map → AdminDashboard-C3RXpTp6.js.map} +1 -1
  7. package/dist/admin/{AdminFiles-DFTjijGp.js → AdminFiles-31ivR6Wq.js} +3 -4
  8. package/dist/admin/{AdminFiles-DFTjijGp.js.map → AdminFiles-31ivR6Wq.js.map} +1 -1
  9. package/dist/admin/{AdminJobDashboard-BL8gGPDp.js → AdminJobDashboard-BABLe7hL.js} +73 -25
  10. package/dist/admin/AdminJobDashboard-BABLe7hL.js.map +1 -0
  11. package/dist/admin/{AdminJobExecutions-D9E-CS-U.js → AdminJobExecutions-D-G8RIlr.js} +3 -4
  12. package/dist/admin/{AdminJobExecutions-D9E-CS-U.js.map → AdminJobExecutions-D-G8RIlr.js.map} +1 -1
  13. package/dist/admin/{AdminJobRegistry-Ci9ue1zC.js → AdminJobRegistry-oIS3K9NX.js} +3 -4
  14. package/dist/admin/{AdminJobRegistry-Ci9ue1zC.js.map → AdminJobRegistry-oIS3K9NX.js.map} +1 -1
  15. package/dist/admin/{AdminLayout-I6TlUMPc.js → AdminLayout-BmZ9mtXh.js} +8 -25
  16. package/dist/admin/AdminLayout-BmZ9mtXh.js.map +1 -0
  17. package/dist/admin/{AdminNotifications-ZPHCYrv7.js → AdminNotifications-DHdzksww.js} +3 -4
  18. package/dist/admin/{AdminNotifications-ZPHCYrv7.js.map → AdminNotifications-DHdzksww.js.map} +1 -1
  19. package/dist/admin/{AdminParameters-CqgvhRsb.js → AdminParameters-CyZQSXnN.js} +3 -12
  20. package/dist/admin/{AdminParameters-CqgvhRsb.js.map → AdminParameters-CyZQSXnN.js.map} +1 -1
  21. package/dist/admin/{AdminSessions-Bz5NRuoW.js → AdminSessions--xwELDSO.js} +3 -4
  22. package/dist/admin/{AdminSessions-Bz5NRuoW.js.map → AdminSessions--xwELDSO.js.map} +1 -1
  23. package/dist/admin/{AdminUserLayout-lXT6I0Qq.js → AdminUserLayout-DvBTG5gd.js} +72 -111
  24. package/dist/admin/AdminUserLayout-DvBTG5gd.js.map +1 -0
  25. package/dist/admin/{AdminUserProfile-vFBLoJ3h.js → AdminUserProfile-CzsPBl6Z.js} +7 -6
  26. package/dist/admin/AdminUserProfile-CzsPBl6Z.js.map +1 -0
  27. package/dist/admin/{AdminUserSessions-CT_YDim0.js → AdminUserSessions-C-aUnhVN.js} +3 -4
  28. package/dist/admin/{AdminUserSessions-CT_YDim0.js.map → AdminUserSessions-C-aUnhVN.js.map} +1 -1
  29. package/dist/admin/{AdminUsers-D1UfGya9.js → AdminUsers-BYwei5sj.js} +4 -4
  30. package/dist/admin/AdminUsers-BYwei5sj.js.map +1 -0
  31. package/dist/admin/{AuthLayout-_frhdgOO.js → AuthLayout-CkPGLJku.js} +3 -4
  32. package/dist/admin/{AuthLayout-_frhdgOO.js.map → AuthLayout-CkPGLJku.js.map} +1 -1
  33. package/dist/{demo/IconGoogle-CSQLPYwX.js → admin/IconGoogle-8Nkx6yax.js} +2 -4
  34. package/dist/admin/{IconGoogle-Ch1m3Uzl.js.map → IconGoogle-8Nkx6yax.js.map} +1 -1
  35. package/dist/admin/{Login-xtNmQtGh.js → Login-DSBqNsZc.js} +5 -6
  36. package/dist/{auth/Login-BA1E8IZl.js.map → admin/Login-DSBqNsZc.js.map} +1 -1
  37. package/dist/admin/{Profile-_AtPUwAP.js → Profile-CDRjJo0P.js} +3 -5
  38. package/dist/{demo/Profile-DS5q4vOh.js.map → admin/Profile-CDRjJo0P.js.map} +1 -1
  39. package/dist/admin/{Register-JcCjHUUn.js → Register-4QGFOnfh.js} +5 -6
  40. package/dist/{demo/Register-B4hLBeEv.js.map → admin/Register-4QGFOnfh.js.map} +1 -1
  41. package/dist/admin/{ResetPassword-CwGBPLJO.js → ResetPassword-Gxc9L_mY.js} +4 -5
  42. package/dist/{auth/ResetPassword-DCtGcneA.js.map → admin/ResetPassword-Gxc9L_mY.js.map} +1 -1
  43. package/dist/admin/{VerifyEmail-hNxWejWf.js → VerifyEmail-D7G5NnaN.js} +4 -5
  44. package/dist/{auth/VerifyEmail-DkH7NBfn.js.map → admin/VerifyEmail-D7G5NnaN.js.map} +1 -1
  45. package/dist/admin/adminUserAtom-DCi4wf-v.js +11 -0
  46. package/dist/admin/adminUserAtom-DCi4wf-v.js.map +1 -0
  47. package/dist/admin/{core-CYaRQ8O-.js → core-D1AbU50V.js} +24 -85
  48. package/dist/admin/core-D1AbU50V.js.map +1 -0
  49. package/dist/admin/index.d.ts +59 -10
  50. package/dist/admin/index.d.ts.map +1 -1
  51. package/dist/admin/index.js +35 -36
  52. package/dist/admin/index.js.map +1 -1
  53. package/dist/admin/rolldown-runtime-CiIaOW0V.js +13 -0
  54. package/dist/{demo/AuthLayout-Brri4A-L.js → auth/AuthLayout-CfRKcTqP.js} +3 -4
  55. package/dist/auth/{AuthLayout-AvLlcLjS.js.map → AuthLayout-CfRKcTqP.js.map} +1 -1
  56. package/dist/{admin/IconGoogle-Ch1m3Uzl.js → auth/IconGoogle-8Nkx6yax.js} +2 -4
  57. package/dist/auth/{IconGoogle-Ch1m3Uzl.js.map → IconGoogle-8Nkx6yax.js.map} +1 -1
  58. package/dist/auth/{Login-BA1E8IZl.js → Login-DJyweoPS.js} +5 -6
  59. package/dist/{demo/Login-C12N4oGs.js.map → auth/Login-DJyweoPS.js.map} +1 -1
  60. package/dist/{demo/Profile-DS5q4vOh.js → auth/Profile-Cy93pNTw.js} +3 -5
  61. package/dist/auth/{Profile-YcWdeuFz.js.map → Profile-Cy93pNTw.js.map} +1 -1
  62. package/dist/auth/{Register-CPhEO5MG.js → Register-CSqzzitW.js} +5 -6
  63. package/dist/{admin/Register-JcCjHUUn.js.map → auth/Register-CSqzzitW.js.map} +1 -1
  64. package/dist/{demo/ResetPassword-D8g9ha1N.js → auth/ResetPassword-B61QPlQi.js} +4 -5
  65. package/dist/{admin/ResetPassword-CwGBPLJO.js.map → auth/ResetPassword-B61QPlQi.js.map} +1 -1
  66. package/dist/auth/{VerifyEmail-DkH7NBfn.js → VerifyEmail-CqBJ11id.js} +4 -5
  67. package/dist/{admin/VerifyEmail-hNxWejWf.js.map → auth/VerifyEmail-CqBJ11id.js.map} +1 -1
  68. package/dist/auth/{core-D5jIAVF2.js → core-C6D3pazL.js} +22 -54
  69. package/dist/auth/core-C6D3pazL.js.map +1 -0
  70. package/dist/auth/index.d.ts +0 -6
  71. package/dist/auth/index.d.ts.map +1 -1
  72. package/dist/auth/index.js +13 -18
  73. package/dist/auth/index.js.map +1 -1
  74. package/dist/auth/rolldown-runtime-CiIaOW0V.js +13 -0
  75. package/dist/core/index.d.ts +10 -4
  76. package/dist/core/index.d.ts.map +1 -1
  77. package/dist/core/index.js +21 -91
  78. package/dist/core/index.js.map +1 -1
  79. package/dist/{auth/AuthLayout-AvLlcLjS.js → demo/AuthLayout-Dq5tSLSc.js} +3 -4
  80. package/dist/demo/{AuthLayout-Brri4A-L.js.map → AuthLayout-Dq5tSLSc.js.map} +1 -1
  81. package/dist/demo/{DemoButton-wiCxZZ_L.js → DemoButton-_Ws2w-J0.js} +4 -5
  82. package/dist/demo/{DemoButton-wiCxZZ_L.js.map → DemoButton-_Ws2w-J0.js.map} +1 -1
  83. package/dist/demo/{DemoControlSelect-D7ILObVg.js → DemoControlSelect-ChP4ZOpQ.js} +4 -5
  84. package/dist/demo/{DemoControlSelect-D7ILObVg.js.map → DemoControlSelect-ChP4ZOpQ.js.map} +1 -1
  85. package/dist/demo/{DemoDataTable-DZ5Y8pFX.js → DemoDataTable-Hwf_UUni.js} +4 -5
  86. package/dist/demo/{DemoDataTable-DZ5Y8pFX.js.map → DemoDataTable-Hwf_UUni.js.map} +1 -1
  87. package/dist/demo/{DemoDialog-CUWdLHim.js → DemoDialog-B01OMVRd.js} +3 -4
  88. package/dist/demo/{DemoDialog-CUWdLHim.js.map → DemoDialog-B01OMVRd.js.map} +1 -1
  89. package/dist/demo/{DemoFlex-a8OhMMvq.js → DemoFlex-870PEl0V.js} +4 -5
  90. package/dist/demo/{DemoFlex-a8OhMMvq.js.map → DemoFlex-870PEl0V.js.map} +1 -1
  91. package/dist/demo/{DemoHeading-C13OVDfS.js → DemoHeading-C1YR27fz.js} +4 -5
  92. package/dist/demo/{DemoHeading-C13OVDfS.js.map → DemoHeading-C1YR27fz.js.map} +1 -1
  93. package/dist/demo/{DemoHome-D_De3UiT.js → DemoHome-DRbL2eGf.js} +4 -5
  94. package/dist/demo/{DemoHome-D_De3UiT.js.map → DemoHome-DRbL2eGf.js.map} +1 -1
  95. package/dist/demo/{DemoJsonViewer-B50s9aGM.js → DemoJsonViewer-DoABiqBW.js} +4 -5
  96. package/dist/demo/{DemoJsonViewer-B50s9aGM.js.map → DemoJsonViewer-DoABiqBW.js.map} +1 -1
  97. package/dist/demo/{DemoLayout-CHU8WTwO.js → DemoLayout-CN_PDCX2.js} +4 -5
  98. package/dist/demo/DemoLayout-CN_PDCX2.js.map +1 -0
  99. package/dist/demo/{DemoLogin-BBlrWpml.js → DemoLogin-B5x-ug3Q.js} +10 -11
  100. package/dist/demo/{DemoLogin-BBlrWpml.js.map → DemoLogin-B5x-ug3Q.js.map} +1 -1
  101. package/dist/demo/{DemoRegister-BuNE3_-f.js → DemoRegister-Q6sg2xuV.js} +10 -11
  102. package/dist/demo/{DemoRegister-BuNE3_-f.js.map → DemoRegister-Q6sg2xuV.js.map} +1 -1
  103. package/dist/demo/{DemoResetPassword-D_IjjjOJ.js → DemoResetPassword-DrqZfmEw.js} +10 -11
  104. package/dist/demo/{DemoResetPassword-D_IjjjOJ.js.map → DemoResetPassword-DrqZfmEw.js.map} +1 -1
  105. package/dist/demo/{DemoSidebar-Giy2HRBD.js → DemoSidebar-CfKS6w1o.js} +4 -5
  106. package/dist/demo/{DemoSidebar-Giy2HRBD.js.map → DemoSidebar-CfKS6w1o.js.map} +1 -1
  107. package/dist/demo/{DemoText-ubcw-vog.js → DemoText-pT6Gi5b5.js} +4 -5
  108. package/dist/demo/{DemoText-ubcw-vog.js.map → DemoText-pT6Gi5b5.js.map} +1 -1
  109. package/dist/demo/{DemoToast-9die_dYT.js → DemoToast-I13NBzQQ.js} +3 -4
  110. package/dist/demo/{DemoToast-9die_dYT.js.map → DemoToast-I13NBzQQ.js.map} +1 -1
  111. package/dist/demo/{DemoTypeForm-D_d6OVKL.js → DemoTypeForm-BqzcrtvN.js} +4 -5
  112. package/dist/demo/{DemoTypeForm-D_d6OVKL.js.map → DemoTypeForm-BqzcrtvN.js.map} +1 -1
  113. package/dist/demo/{DemoVerifyEmail-B43KlF4F.js → DemoVerifyEmail-HwD8xfQw.js} +10 -11
  114. package/dist/demo/{DemoVerifyEmail-B43KlF4F.js.map → DemoVerifyEmail-HwD8xfQw.js.map} +1 -1
  115. package/dist/{auth/IconGoogle-Ch1m3Uzl.js → demo/IconGoogle-CwQy4G9y.js} +2 -4
  116. package/dist/demo/{IconGoogle-CSQLPYwX.js.map → IconGoogle-CwQy4G9y.js.map} +1 -1
  117. package/dist/demo/{Login-C12N4oGs.js → Login-CqG1iJbn.js} +5 -6
  118. package/dist/{admin/Login-xtNmQtGh.js.map → demo/Login-CqG1iJbn.js.map} +1 -1
  119. package/dist/{auth/Profile-YcWdeuFz.js → demo/Profile-C0ojJCaG.js} +3 -5
  120. package/dist/{admin/Profile-_AtPUwAP.js.map → demo/Profile-C0ojJCaG.js.map} +1 -1
  121. package/dist/demo/{Register-B4hLBeEv.js → Register-KKZwr_lL.js} +5 -6
  122. package/dist/{auth/Register-CPhEO5MG.js.map → demo/Register-KKZwr_lL.js.map} +1 -1
  123. package/dist/{auth/ResetPassword-DCtGcneA.js → demo/ResetPassword-DMrLFEtr.js} +4 -5
  124. package/dist/demo/{ResetPassword-D8g9ha1N.js.map → ResetPassword-DMrLFEtr.js.map} +1 -1
  125. package/dist/demo/{Showcase-D6Fxt4X4.js → Showcase-D49Wud2v.js} +3 -5
  126. package/dist/demo/{Showcase-D6Fxt4X4.js.map → Showcase-D49Wud2v.js.map} +1 -1
  127. package/dist/demo/{VerifyEmail-BjDo0cZA.js → VerifyEmail-BFCAFz6T.js} +4 -5
  128. package/dist/demo/{VerifyEmail-BjDo0cZA.js.map → VerifyEmail-BFCAFz6T.js.map} +1 -1
  129. package/dist/demo/{auth-ByVTreDl.js → auth-D9qTZzCa.js} +18 -35
  130. package/dist/demo/{auth-ByVTreDl.js.map → auth-D9qTZzCa.js.map} +1 -1
  131. package/dist/demo/{core-DFgB3yU4.js → core-DRtQklr3.js} +23 -86
  132. package/dist/demo/core-DRtQklr3.js.map +1 -0
  133. package/dist/demo/index.js +19 -22
  134. package/dist/demo/index.js.map +1 -1
  135. package/dist/demo/rolldown-runtime-CiIaOW0V.js +13 -0
  136. package/package.json +17 -17
  137. package/src/admin/AdminRouter.tsx +18 -1
  138. package/src/admin/atoms/adminUserAtom.ts +7 -0
  139. package/src/admin/components/AdminLayout.tsx +2 -14
  140. package/src/admin/components/jobs/AdminJobDashboard.tsx +51 -20
  141. package/src/admin/components/users/AdminUserLayout.tsx +84 -127
  142. package/src/admin/components/users/AdminUserProfile.tsx +5 -2
  143. package/src/admin/components/users/AdminUsers.tsx +1 -1
  144. package/src/core/components/Flex.tsx +24 -0
  145. package/src/core/components/buttons/ActionButton.tsx +1 -0
  146. package/src/core/components/dialogs/PromptDialog.tsx +1 -1
  147. package/src/core/components/layout/Breadcrumb.tsx +2 -2
  148. package/src/core/components/layout/DashboardShell.tsx +1 -1
  149. package/src/core/services/DialogService.tsx +2 -2
  150. package/src/core/styles.css +2 -1
  151. package/src/core/table/components/DataTable.tsx +0 -1
  152. package/dist/admin/AdminJobDashboard-BL8gGPDp.js.map +0 -1
  153. package/dist/admin/AdminLayout-I6TlUMPc.js.map +0 -1
  154. package/dist/admin/AdminUserLayout-lXT6I0Qq.js.map +0 -1
  155. package/dist/admin/AdminUserProfile-vFBLoJ3h.js.map +0 -1
  156. package/dist/admin/AdminUsers-D1UfGya9.js.map +0 -1
  157. package/dist/admin/core-CYaRQ8O-.js.map +0 -1
  158. package/dist/admin/rolldown-runtime-CjeV3_4I.js +0 -18
  159. package/dist/auth/core-D5jIAVF2.js.map +0 -1
  160. package/dist/auth/rolldown-runtime-CjeV3_4I.js +0 -18
  161. package/dist/demo/DemoLayout-CHU8WTwO.js.map +0 -1
  162. package/dist/demo/core-DFgB3yU4.js.map +0 -1
  163. package/dist/demo/rolldown-runtime-CjeV3_4I.js +0 -18
@@ -1,4 +1,4 @@
1
- import { _ as ActionButton, b as useToast, g as ClipboardButton, l as Flex$1, m as useDialog, r as DataTable, s as Text$1, u as StatCards } from "./core-CYaRQ8O-.js";
1
+ import { _ as ActionButton, b as useToast, g as ClipboardButton, l as Flex$1, m as useDialog, r as DataTable, s as Text$1, u as StatCards } from "./core-D1AbU50V.js";
2
2
  import { t } from "alepha";
3
3
  import { useI18n } from "alepha/react/i18n";
4
4
  import { Badge, Code, Tooltip } from "@mantine/core";
@@ -7,7 +7,6 @@ import { useCallback, useState } from "react";
7
7
  import { IconCheck, IconClock, IconKey, IconNetwork, IconTrash, IconUser } from "@tabler/icons-react";
8
8
  import { useRouter } from "alepha/react/router";
9
9
  import { useClient } from "alepha/react";
10
-
11
10
  //#region ../../src/admin/components/keys/AdminApiKeys.tsx
12
11
  const getKeyStatus = (key) => {
13
12
  if (key.revokedAt) return "revoked";
@@ -280,7 +279,7 @@ const AdminApiKeys = () => {
280
279
  }, refreshKey)]
281
280
  });
282
281
  };
283
-
284
282
  //#endregion
285
283
  export { AdminApiKeys as default };
286
- //# sourceMappingURL=AdminApiKeys-Dy_k-4Vd.js.map
284
+
285
+ //# sourceMappingURL=AdminApiKeys-Bt1PjO6o.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"AdminApiKeys-Dy_k-4Vd.js","names":["Flex","Text"],"sources":["../../src/admin/components/keys/AdminApiKeys.tsx"],"sourcesContent":["import {\n ActionButton,\n ClipboardButton,\n DataTable,\n Flex,\n StatCards,\n Text,\n useDialog,\n useToast,\n} from \"@alepha/ui\";\nimport { Badge, Code, Tooltip } from \"@mantine/core\";\nimport {\n IconCheck,\n IconClock,\n IconKey,\n IconNetwork,\n IconTrash,\n IconUser,\n} from \"@tabler/icons-react\";\nimport { type Page, t } from \"alepha\";\nimport type { AdminApiKeyController } from \"alepha/api/keys\";\nimport { useClient } from \"alepha/react\";\nimport { useI18n } from \"alepha/react/i18n\";\nimport { useRouter } from \"alepha/react/router\";\nimport { useCallback, useState } from \"react\";\nimport type { AdminRouter } from \"../../AdminRouter.ts\";\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Types\n// ─────────────────────────────────────────────────────────────────────────────\n\ninterface ApiKeyResource {\n id: string;\n userId: string;\n name: string;\n description?: string;\n tokenPrefix: string;\n tokenSuffix: string;\n roles: string[];\n createdAt: string;\n lastUsedAt?: string;\n lastUsedIp?: string;\n expiresAt?: string;\n revokedAt?: string;\n usageCount: number;\n}\n\ninterface KeyStats {\n total: number;\n active: number;\n revoked: number;\n expired: number;\n neverUsed: number;\n}\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Utilities\n// ─────────────────────────────────────────────────────────────────────────────\n\nconst getKeyStatus = (\n key: ApiKeyResource,\n): \"active\" | \"revoked\" | \"expired\" => {\n if (key.revokedAt) return \"revoked\";\n if (key.expiresAt && new Date(key.expiresAt) < new Date()) return \"expired\";\n return \"active\";\n};\n\nconst formatKeyPreview = (prefix: string, suffix: string) => {\n return `${prefix}...${suffix}`;\n};\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Main Component\n// ─────────────────────────────────────────────────────────────────────────────\n\nconst AdminApiKeys = () => {\n const client = useClient<AdminApiKeyController>();\n const router = useRouter<AdminRouter>();\n const { l } = useI18n();\n const toast = useToast();\n const dialog = useDialog();\n\n const [stats, setStats] = useState<KeyStats>({\n total: 0,\n active: 0,\n revoked: 0,\n expired: 0,\n neverUsed: 0,\n });\n const [refreshKey, setRefreshKey] = useState(0);\n const [loading, setLoading] = useState(true);\n\n const filters = t.object({\n userId: t.optional(t.uuid()),\n includeRevoked: t.optional(t.boolean()),\n });\n\n const handleRevoke = useCallback(\n async (key: ApiKeyResource) => {\n const confirmed = await dialog.confirm({\n title: \"Revoke API Key\",\n message: `Are you sure you want to revoke \"${key.name}\"? This action cannot be undone and will immediately invalidate the key.`,\n confirmLabel: \"Revoke Key\",\n confirmColor: \"red\",\n });\n\n if (!confirmed) return;\n\n try {\n await client.revokeApiKey({ params: { id: key.id } });\n toast.success(`API key \"${key.name}\" has been revoked`);\n setRefreshKey((k) => k + 1);\n } catch (error) {\n toast.danger(`Failed to revoke API key`);\n }\n },\n [client, dialog, toast],\n );\n\n const updateStats = useCallback((keys: ApiKeyResource[]) => {\n const now = new Date();\n const newStats: KeyStats = {\n total: keys.length,\n active: 0,\n revoked: 0,\n expired: 0,\n neverUsed: 0,\n };\n\n for (const key of keys) {\n if (key.revokedAt) {\n newStats.revoked++;\n } else if (key.expiresAt && new Date(key.expiresAt) < now) {\n newStats.expired++;\n } else {\n newStats.active++;\n }\n\n if (!key.lastUsedAt) {\n newStats.neverUsed++;\n }\n }\n\n setStats(newStats);\n setLoading(false);\n }, []);\n\n return (\n <Flex p=\"md\" flex={1} direction=\"column\" gap=\"md\">\n <StatCards\n items={[\n { label: \"Total Keys\", value: stats.total, icon: IconKey },\n { label: \"Active\", value: stats.active, icon: IconCheck },\n { label: \"Revoked\", value: stats.revoked, icon: IconTrash },\n { label: \"Never Used\", value: stats.neverUsed, icon: IconClock },\n ]}\n />\n\n <DataTable<ApiKeyResource, typeof filters>\n key={refreshKey}\n submitOnInit\n defaultSize={15}\n typeFormProps={{\n skipSubmitButton: true,\n columns: 2,\n }}\n tableProps={{\n horizontalSpacing: \"sm\",\n verticalSpacing: \"sm\",\n }}\n onFilterChange={(_key, _value, form) => form.submit()}\n filters={filters}\n tableTrProps={(item) => {\n const status = getKeyStatus(item);\n if (status === \"revoked\") {\n return { opacity: 0.6 };\n }\n if (status === \"expired\") {\n return { opacity: 0.7 };\n }\n return {};\n }}\n items={async (filters) => {\n const response = await client.findApiKeys({\n query: {\n ...filters,\n includeRevoked: filters.includeRevoked ?? true,\n },\n });\n\n const allKeys = await client.findApiKeys({\n query: { includeRevoked: true, size: 100 },\n });\n updateStats(allKeys.content as ApiKeyResource[]);\n\n return response as Page<ApiKeyResource>;\n }}\n columns={{\n name: {\n label: \"Name\",\n value: (item) => (\n <Flex direction=\"column\" gap={2}>\n <Text size=\"sm\" fw={600}>\n {item.name}\n </Text>\n {item.description && (\n <Text size=\"xs\" c=\"dimmed\" lineClamp={1}>\n {item.description}\n </Text>\n )}\n </Flex>\n ),\n },\n token: {\n label: \"Key\",\n value: (item) => (\n <Flex gap={4}>\n <Code\n ff=\"monospace\"\n style={{\n fontSize: 11,\n letterSpacing: \"0.5px\",\n }}\n >\n {formatKeyPreview(item.tokenPrefix, item.tokenSuffix)}\n </Code>\n <ClipboardButton\n size=\"xs\"\n variant=\"subtle\"\n value={formatKeyPreview(item.tokenPrefix, item.tokenSuffix)}\n />\n </Flex>\n ),\n },\n status: {\n label: \"Status\",\n value: (item) => {\n const status = getKeyStatus(item);\n return (\n <Badge\n size=\"sm\"\n variant=\"light\"\n color={\n status === \"active\"\n ? \"green\"\n : status === \"expired\"\n ? \"yellow\"\n : \"red\"\n }\n >\n {status}\n </Badge>\n );\n },\n },\n roles: {\n label: \"Roles\",\n value: (item) => (\n <Flex gap={4} wrap=\"wrap\">\n {item.roles.length > 0 ? (\n item.roles.slice(0, 3).map((role) => (\n <Badge key={role} size=\"xs\" variant=\"outline\" color=\"gray\">\n {role}\n </Badge>\n ))\n ) : (\n <Text size=\"xs\" c=\"dimmed\">\n No roles\n </Text>\n )}\n {item.roles.length > 3 && (\n <Tooltip label={item.roles.slice(3).join(\", \")}>\n <Badge size=\"xs\" variant=\"light\" color=\"gray\">\n +{item.roles.length - 3}\n </Badge>\n </Tooltip>\n )}\n </Flex>\n ),\n },\n usage: {\n label: \"Usage\",\n value: (item) => (\n <Flex direction=\"column\" gap={2}>\n <Text size=\"xs\" ff=\"monospace\" fw={500}>\n {item.usageCount.toLocaleString()} calls\n </Text>\n {item.lastUsedAt ? (\n <Flex gap={4}>\n <Text size=\"xs\" c=\"dimmed\">\n {l(item.lastUsedAt, { date: \"fromNow\" })}\n </Text>\n {item.lastUsedIp && (\n <Tooltip label={`Last IP: ${item.lastUsedIp}`}>\n <IconNetwork\n size={12}\n color=\"var(--mantine-color-dimmed)\"\n />\n </Tooltip>\n )}\n </Flex>\n ) : (\n <Text size=\"xs\" c=\"dimmed\">\n Never used\n </Text>\n )}\n </Flex>\n ),\n },\n userId: {\n label: \"Owner\",\n value: (item) => (\n <ActionButton\n variant=\"subtle\"\n size=\"xs\"\n href={router.path(\"adminUserProfile\", {\n params: { userId: item.userId },\n })}\n leftSection={<IconUser size={12} />}\n >\n <Text size=\"xs\" ff=\"monospace\">\n {item.userId.slice(0, 8)}\n </Text>\n </ActionButton>\n ),\n },\n createdAt: {\n label: \"Created\",\n value: (item) => (\n <Text size=\"xs\" c=\"dimmed\">\n {l(item.createdAt, { date: \"fromNow\" })}\n </Text>\n ),\n },\n expiresAt: {\n label: \"Expires\",\n value: (item) => {\n if (!item.expiresAt) {\n return (\n <Text size=\"xs\" c=\"dimmed\">\n Never\n </Text>\n );\n }\n\n return (\n <Text size=\"xs\" c=\"dimmed\">\n {l(item.expiresAt, { date: \"fromNow\" })}\n </Text>\n );\n },\n },\n }}\n rowActions={(item) => [\n {\n label: \"Revoke key\",\n icon: IconTrash,\n color: \"red\",\n onClick: () => handleRevoke(item),\n visible: getKeyStatus(item) === \"active\",\n },\n ]}\n />\n </Flex>\n );\n};\n\nexport default AdminApiKeys;\n"],"mappings":";;;;;;;;;;;AA2DA,MAAM,gBACJ,QACqC;AACrC,KAAI,IAAI,UAAW,QAAO;AAC1B,KAAI,IAAI,aAAa,IAAI,KAAK,IAAI,UAAU,mBAAG,IAAI,MAAM,CAAE,QAAO;AAClE,QAAO;;AAGT,MAAM,oBAAoB,QAAgB,WAAmB;AAC3D,QAAO,GAAG,OAAO,KAAK;;AAOxB,MAAM,qBAAqB;CACzB,MAAM,SAAS,WAAkC;CACjD,MAAM,SAAS,WAAwB;CACvC,MAAM,EAAE,MAAM,SAAS;CACvB,MAAM,QAAQ,UAAU;CACxB,MAAM,SAAS,WAAW;CAE1B,MAAM,CAAC,OAAO,YAAY,SAAmB;EAC3C,OAAO;EACP,QAAQ;EACR,SAAS;EACT,SAAS;EACT,WAAW;EACZ,CAAC;CACF,MAAM,CAAC,YAAY,iBAAiB,SAAS,EAAE;CAC/C,MAAM,CAAC,SAAS,cAAc,SAAS,KAAK;CAE5C,MAAM,UAAU,EAAE,OAAO;EACvB,QAAQ,EAAE,SAAS,EAAE,MAAM,CAAC;EAC5B,gBAAgB,EAAE,SAAS,EAAE,SAAS,CAAC;EACxC,CAAC;CAEF,MAAM,eAAe,YACnB,OAAO,QAAwB;AAQ7B,MAAI,CAPc,MAAM,OAAO,QAAQ;GACrC,OAAO;GACP,SAAS,oCAAoC,IAAI,KAAK;GACtD,cAAc;GACd,cAAc;GACf,CAAC,CAEc;AAEhB,MAAI;AACF,SAAM,OAAO,aAAa,EAAE,QAAQ,EAAE,IAAI,IAAI,IAAI,EAAE,CAAC;AACrD,SAAM,QAAQ,YAAY,IAAI,KAAK,oBAAoB;AACvD,kBAAe,MAAM,IAAI,EAAE;WACpB,OAAO;AACd,SAAM,OAAO,2BAA2B;;IAG5C;EAAC;EAAQ;EAAQ;EAAM,CACxB;CAED,MAAM,cAAc,aAAa,SAA2B;EAC1D,MAAM,sBAAM,IAAI,MAAM;EACtB,MAAM,WAAqB;GACzB,OAAO,KAAK;GACZ,QAAQ;GACR,SAAS;GACT,SAAS;GACT,WAAW;GACZ;AAED,OAAK,MAAM,OAAO,MAAM;AACtB,OAAI,IAAI,UACN,UAAS;YACA,IAAI,aAAa,IAAI,KAAK,IAAI,UAAU,GAAG,IACpD,UAAS;OAET,UAAS;AAGX,OAAI,CAAC,IAAI,WACP,UAAS;;AAIb,WAAS,SAAS;AAClB,aAAW,MAAM;IAChB,EAAE,CAAC;AAEN,QACE,qBAACA;EAAK,GAAE;EAAK,MAAM;EAAG,WAAU;EAAS,KAAI;aAC3C,oBAAC,aACC,OAAO;GACL;IAAE,OAAO;IAAc,OAAO,MAAM;IAAO,MAAM;IAAS;GAC1D;IAAE,OAAO;IAAU,OAAO,MAAM;IAAQ,MAAM;IAAW;GACzD;IAAE,OAAO;IAAW,OAAO,MAAM;IAAS,MAAM;IAAW;GAC3D;IAAE,OAAO;IAAc,OAAO,MAAM;IAAW,MAAM;IAAW;GACjE,GACD,EAEF,oBAAC;GAEC;GACA,aAAa;GACb,eAAe;IACb,kBAAkB;IAClB,SAAS;IACV;GACD,YAAY;IACV,mBAAmB;IACnB,iBAAiB;IAClB;GACD,iBAAiB,MAAM,QAAQ,SAAS,KAAK,QAAQ;GAC5C;GACT,eAAe,SAAS;IACtB,MAAM,SAAS,aAAa,KAAK;AACjC,QAAI,WAAW,UACb,QAAO,EAAE,SAAS,IAAK;AAEzB,QAAI,WAAW,UACb,QAAO,EAAE,SAAS,IAAK;AAEzB,WAAO,EAAE;;GAEX,OAAO,OAAO,YAAY;IACxB,MAAM,WAAW,MAAM,OAAO,YAAY,EACxC,OAAO;KACL,GAAG;KACH,gBAAgB,QAAQ,kBAAkB;KAC3C,EACF,CAAC;AAKF,iBAHgB,MAAM,OAAO,YAAY,EACvC,OAAO;KAAE,gBAAgB;KAAM,MAAM;KAAK,EAC3C,CAAC,EACkB,QAA4B;AAEhD,WAAO;;GAET,SAAS;IACP,MAAM;KACJ,OAAO;KACP,QAAQ,SACN,qBAACA;MAAK,WAAU;MAAS,KAAK;iBAC5B,oBAACC;OAAK,MAAK;OAAK,IAAI;iBACjB,KAAK;QACD,EACN,KAAK,eACJ,oBAACA;OAAK,MAAK;OAAK,GAAE;OAAS,WAAW;iBACnC,KAAK;QACD;OAEJ;KAEV;IACD,OAAO;KACL,OAAO;KACP,QAAQ,SACN,qBAACD;MAAK,KAAK;iBACT,oBAAC;OACC,IAAG;OACH,OAAO;QACL,UAAU;QACV,eAAe;QAChB;iBAEA,iBAAiB,KAAK,aAAa,KAAK,YAAY;QAChD,EACP,oBAAC;OACC,MAAK;OACL,SAAQ;OACR,OAAO,iBAAiB,KAAK,aAAa,KAAK,YAAY;QAC3D;OACG;KAEV;IACD,QAAQ;KACN,OAAO;KACP,QAAQ,SAAS;MACf,MAAM,SAAS,aAAa,KAAK;AACjC,aACE,oBAAC;OACC,MAAK;OACL,SAAQ;OACR,OACE,WAAW,WACP,UACA,WAAW,YACT,WACA;iBAGP;QACK;;KAGb;IACD,OAAO;KACL,OAAO;KACP,QAAQ,SACN,qBAACA;MAAK,KAAK;MAAG,MAAK;iBAChB,KAAK,MAAM,SAAS,IACnB,KAAK,MAAM,MAAM,GAAG,EAAE,CAAC,KAAK,SAC1B,oBAAC;OAAiB,MAAK;OAAK,SAAQ;OAAU,OAAM;iBACjD;SADS,KAEJ,CACR,GAEF,oBAACC;OAAK,MAAK;OAAK,GAAE;iBAAS;QAEpB,EAER,KAAK,MAAM,SAAS,KACnB,oBAAC;OAAQ,OAAO,KAAK,MAAM,MAAM,EAAE,CAAC,KAAK,KAAK;iBAC5C,qBAAC;QAAM,MAAK;QAAK,SAAQ;QAAQ,OAAM;mBAAO,KAC1C,KAAK,MAAM,SAAS;SAChB;QACA;OAEP;KAEV;IACD,OAAO;KACL,OAAO;KACP,QAAQ,SACN,qBAACD;MAAK,WAAU;MAAS,KAAK;iBAC5B,qBAACC;OAAK,MAAK;OAAK,IAAG;OAAY,IAAI;kBAChC,KAAK,WAAW,gBAAgB,EAAC;QAC7B,EACN,KAAK,aACJ,qBAACD;OAAK,KAAK;kBACT,oBAACC;QAAK,MAAK;QAAK,GAAE;kBACf,EAAE,KAAK,YAAY,EAAE,MAAM,WAAW,CAAC;SACnC,EACN,KAAK,cACJ,oBAAC;QAAQ,OAAO,YAAY,KAAK;kBAC/B,oBAAC;SACC,MAAM;SACN,OAAM;UACN;SACM;QAEP,GAEP,oBAACA;OAAK,MAAK;OAAK,GAAE;iBAAS;QAEpB;OAEJ;KAEV;IACD,QAAQ;KACN,OAAO;KACP,QAAQ,SACN,oBAAC;MACC,SAAQ;MACR,MAAK;MACL,MAAM,OAAO,KAAK,oBAAoB,EACpC,QAAQ,EAAE,QAAQ,KAAK,QAAQ,EAChC,CAAC;MACF,aAAa,oBAAC,YAAS,MAAM,KAAM;gBAEnC,oBAACA;OAAK,MAAK;OAAK,IAAG;iBAChB,KAAK,OAAO,MAAM,GAAG,EAAE;QACnB;OACM;KAElB;IACD,WAAW;KACT,OAAO;KACP,QAAQ,SACN,oBAACA;MAAK,MAAK;MAAK,GAAE;gBACf,EAAE,KAAK,WAAW,EAAE,MAAM,WAAW,CAAC;OAClC;KAEV;IACD,WAAW;KACT,OAAO;KACP,QAAQ,SAAS;AACf,UAAI,CAAC,KAAK,UACR,QACE,oBAACA;OAAK,MAAK;OAAK,GAAE;iBAAS;QAEpB;AAIX,aACE,oBAACA;OAAK,MAAK;OAAK,GAAE;iBACf,EAAE,KAAK,WAAW,EAAE,MAAM,WAAW,CAAC;QAClC;;KAGZ;IACF;GACD,aAAa,SAAS,CACpB;IACE,OAAO;IACP,MAAM;IACN,OAAO;IACP,eAAe,aAAa,KAAK;IACjC,SAAS,aAAa,KAAK,KAAK;IACjC,CACF;KA1MI,WA2ML;GACG"}
1
+ {"version":3,"file":"AdminApiKeys-Bt1PjO6o.js","names":["Flex","Text"],"sources":["../../src/admin/components/keys/AdminApiKeys.tsx"],"sourcesContent":["import {\n ActionButton,\n ClipboardButton,\n DataTable,\n Flex,\n StatCards,\n Text,\n useDialog,\n useToast,\n} from \"@alepha/ui\";\nimport { Badge, Code, Tooltip } from \"@mantine/core\";\nimport {\n IconCheck,\n IconClock,\n IconKey,\n IconNetwork,\n IconTrash,\n IconUser,\n} from \"@tabler/icons-react\";\nimport { type Page, t } from \"alepha\";\nimport type { AdminApiKeyController } from \"alepha/api/keys\";\nimport { useClient } from \"alepha/react\";\nimport { useI18n } from \"alepha/react/i18n\";\nimport { useRouter } from \"alepha/react/router\";\nimport { useCallback, useState } from \"react\";\nimport type { AdminRouter } from \"../../AdminRouter.ts\";\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Types\n// ─────────────────────────────────────────────────────────────────────────────\n\ninterface ApiKeyResource {\n id: string;\n userId: string;\n name: string;\n description?: string;\n tokenPrefix: string;\n tokenSuffix: string;\n roles: string[];\n createdAt: string;\n lastUsedAt?: string;\n lastUsedIp?: string;\n expiresAt?: string;\n revokedAt?: string;\n usageCount: number;\n}\n\ninterface KeyStats {\n total: number;\n active: number;\n revoked: number;\n expired: number;\n neverUsed: number;\n}\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Utilities\n// ─────────────────────────────────────────────────────────────────────────────\n\nconst getKeyStatus = (\n key: ApiKeyResource,\n): \"active\" | \"revoked\" | \"expired\" => {\n if (key.revokedAt) return \"revoked\";\n if (key.expiresAt && new Date(key.expiresAt) < new Date()) return \"expired\";\n return \"active\";\n};\n\nconst formatKeyPreview = (prefix: string, suffix: string) => {\n return `${prefix}...${suffix}`;\n};\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Main Component\n// ─────────────────────────────────────────────────────────────────────────────\n\nconst AdminApiKeys = () => {\n const client = useClient<AdminApiKeyController>();\n const router = useRouter<AdminRouter>();\n const { l } = useI18n();\n const toast = useToast();\n const dialog = useDialog();\n\n const [stats, setStats] = useState<KeyStats>({\n total: 0,\n active: 0,\n revoked: 0,\n expired: 0,\n neverUsed: 0,\n });\n const [refreshKey, setRefreshKey] = useState(0);\n const [loading, setLoading] = useState(true);\n\n const filters = t.object({\n userId: t.optional(t.uuid()),\n includeRevoked: t.optional(t.boolean()),\n });\n\n const handleRevoke = useCallback(\n async (key: ApiKeyResource) => {\n const confirmed = await dialog.confirm({\n title: \"Revoke API Key\",\n message: `Are you sure you want to revoke \"${key.name}\"? This action cannot be undone and will immediately invalidate the key.`,\n confirmLabel: \"Revoke Key\",\n confirmColor: \"red\",\n });\n\n if (!confirmed) return;\n\n try {\n await client.revokeApiKey({ params: { id: key.id } });\n toast.success(`API key \"${key.name}\" has been revoked`);\n setRefreshKey((k) => k + 1);\n } catch (error) {\n toast.danger(`Failed to revoke API key`);\n }\n },\n [client, dialog, toast],\n );\n\n const updateStats = useCallback((keys: ApiKeyResource[]) => {\n const now = new Date();\n const newStats: KeyStats = {\n total: keys.length,\n active: 0,\n revoked: 0,\n expired: 0,\n neverUsed: 0,\n };\n\n for (const key of keys) {\n if (key.revokedAt) {\n newStats.revoked++;\n } else if (key.expiresAt && new Date(key.expiresAt) < now) {\n newStats.expired++;\n } else {\n newStats.active++;\n }\n\n if (!key.lastUsedAt) {\n newStats.neverUsed++;\n }\n }\n\n setStats(newStats);\n setLoading(false);\n }, []);\n\n return (\n <Flex p=\"md\" flex={1} direction=\"column\" gap=\"md\">\n <StatCards\n items={[\n { label: \"Total Keys\", value: stats.total, icon: IconKey },\n { label: \"Active\", value: stats.active, icon: IconCheck },\n { label: \"Revoked\", value: stats.revoked, icon: IconTrash },\n { label: \"Never Used\", value: stats.neverUsed, icon: IconClock },\n ]}\n />\n\n <DataTable<ApiKeyResource, typeof filters>\n key={refreshKey}\n submitOnInit\n defaultSize={15}\n typeFormProps={{\n skipSubmitButton: true,\n columns: 2,\n }}\n tableProps={{\n horizontalSpacing: \"sm\",\n verticalSpacing: \"sm\",\n }}\n onFilterChange={(_key, _value, form) => form.submit()}\n filters={filters}\n tableTrProps={(item) => {\n const status = getKeyStatus(item);\n if (status === \"revoked\") {\n return { opacity: 0.6 };\n }\n if (status === \"expired\") {\n return { opacity: 0.7 };\n }\n return {};\n }}\n items={async (filters) => {\n const response = await client.findApiKeys({\n query: {\n ...filters,\n includeRevoked: filters.includeRevoked ?? true,\n },\n });\n\n const allKeys = await client.findApiKeys({\n query: { includeRevoked: true, size: 100 },\n });\n updateStats(allKeys.content as ApiKeyResource[]);\n\n return response as Page<ApiKeyResource>;\n }}\n columns={{\n name: {\n label: \"Name\",\n value: (item) => (\n <Flex direction=\"column\" gap={2}>\n <Text size=\"sm\" fw={600}>\n {item.name}\n </Text>\n {item.description && (\n <Text size=\"xs\" c=\"dimmed\" lineClamp={1}>\n {item.description}\n </Text>\n )}\n </Flex>\n ),\n },\n token: {\n label: \"Key\",\n value: (item) => (\n <Flex gap={4}>\n <Code\n ff=\"monospace\"\n style={{\n fontSize: 11,\n letterSpacing: \"0.5px\",\n }}\n >\n {formatKeyPreview(item.tokenPrefix, item.tokenSuffix)}\n </Code>\n <ClipboardButton\n size=\"xs\"\n variant=\"subtle\"\n value={formatKeyPreview(item.tokenPrefix, item.tokenSuffix)}\n />\n </Flex>\n ),\n },\n status: {\n label: \"Status\",\n value: (item) => {\n const status = getKeyStatus(item);\n return (\n <Badge\n size=\"sm\"\n variant=\"light\"\n color={\n status === \"active\"\n ? \"green\"\n : status === \"expired\"\n ? \"yellow\"\n : \"red\"\n }\n >\n {status}\n </Badge>\n );\n },\n },\n roles: {\n label: \"Roles\",\n value: (item) => (\n <Flex gap={4} wrap=\"wrap\">\n {item.roles.length > 0 ? (\n item.roles.slice(0, 3).map((role) => (\n <Badge key={role} size=\"xs\" variant=\"outline\" color=\"gray\">\n {role}\n </Badge>\n ))\n ) : (\n <Text size=\"xs\" c=\"dimmed\">\n No roles\n </Text>\n )}\n {item.roles.length > 3 && (\n <Tooltip label={item.roles.slice(3).join(\", \")}>\n <Badge size=\"xs\" variant=\"light\" color=\"gray\">\n +{item.roles.length - 3}\n </Badge>\n </Tooltip>\n )}\n </Flex>\n ),\n },\n usage: {\n label: \"Usage\",\n value: (item) => (\n <Flex direction=\"column\" gap={2}>\n <Text size=\"xs\" ff=\"monospace\" fw={500}>\n {item.usageCount.toLocaleString()} calls\n </Text>\n {item.lastUsedAt ? (\n <Flex gap={4}>\n <Text size=\"xs\" c=\"dimmed\">\n {l(item.lastUsedAt, { date: \"fromNow\" })}\n </Text>\n {item.lastUsedIp && (\n <Tooltip label={`Last IP: ${item.lastUsedIp}`}>\n <IconNetwork\n size={12}\n color=\"var(--mantine-color-dimmed)\"\n />\n </Tooltip>\n )}\n </Flex>\n ) : (\n <Text size=\"xs\" c=\"dimmed\">\n Never used\n </Text>\n )}\n </Flex>\n ),\n },\n userId: {\n label: \"Owner\",\n value: (item) => (\n <ActionButton\n variant=\"subtle\"\n size=\"xs\"\n href={router.path(\"adminUserProfile\", {\n params: { userId: item.userId },\n })}\n leftSection={<IconUser size={12} />}\n >\n <Text size=\"xs\" ff=\"monospace\">\n {item.userId.slice(0, 8)}\n </Text>\n </ActionButton>\n ),\n },\n createdAt: {\n label: \"Created\",\n value: (item) => (\n <Text size=\"xs\" c=\"dimmed\">\n {l(item.createdAt, { date: \"fromNow\" })}\n </Text>\n ),\n },\n expiresAt: {\n label: \"Expires\",\n value: (item) => {\n if (!item.expiresAt) {\n return (\n <Text size=\"xs\" c=\"dimmed\">\n Never\n </Text>\n );\n }\n\n return (\n <Text size=\"xs\" c=\"dimmed\">\n {l(item.expiresAt, { date: \"fromNow\" })}\n </Text>\n );\n },\n },\n }}\n rowActions={(item) => [\n {\n label: \"Revoke key\",\n icon: IconTrash,\n color: \"red\",\n onClick: () => handleRevoke(item),\n visible: getKeyStatus(item) === \"active\",\n },\n ]}\n />\n </Flex>\n );\n};\n\nexport default AdminApiKeys;\n"],"mappings":"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"}
@@ -1,4 +1,4 @@
1
- import { l as Flex$1, r as DataTable, s as Text$1 } from "./core-CYaRQ8O-.js";
1
+ import { l as Flex$1, r as DataTable, s as Text$1 } from "./core-D1AbU50V.js";
2
2
  import { t } from "alepha";
3
3
  import { useI18n } from "alepha/react/i18n";
4
4
  import { Badge, Tooltip } from "@mantine/core";
@@ -6,7 +6,6 @@ import { jsx, jsxs } from "react/jsx-runtime";
6
6
  import { IconUser } from "@tabler/icons-react";
7
7
  import { useRouter } from "alepha/react/router";
8
8
  import { useClient } from "alepha/react";
9
-
10
9
  //#region ../../src/admin/components/audits/AdminAudits.tsx
11
10
  const AdminAudits = (props) => {
12
11
  const client = useClient();
@@ -178,7 +177,7 @@ const AdminAudits = (props) => {
178
177
  })
179
178
  });
180
179
  };
181
-
182
180
  //#endregion
183
181
  export { AdminAudits as default };
184
- //# sourceMappingURL=AdminAudits-CKiFMSSU.js.map
182
+
183
+ //# sourceMappingURL=AdminAudits-C7c1CN4c.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"AdminAudits-CKiFMSSU.js","names":["Flex","Text"],"sources":["../../src/admin/components/audits/AdminAudits.tsx"],"sourcesContent":["import { DataTable, Flex, Text } from \"@alepha/ui\";\nimport { Badge, Tooltip } from \"@mantine/core\";\nimport { IconUser } from \"@tabler/icons-react\";\nimport { type Page, t } from \"alepha\";\nimport type { AdminAuditController, AuditEntity } from \"alepha/api/audits\";\nimport { useClient } from \"alepha/react\";\nimport { useI18n } from \"alepha/react/i18n\";\nimport { useRouter } from \"alepha/react/router\";\nimport type { AdminRouter } from \"../../AdminRouter.ts\";\n\nexport interface AdminAuditsProps {\n userRealmName?: string;\n}\n\nconst AdminAudits = (props: AdminAuditsProps) => {\n const client = useClient<AdminAuditController>();\n const router = useRouter<AdminRouter>();\n const { l } = useI18n();\n\n const filters = t.object({\n type: t.optional(t.text()),\n action: t.optional(t.text()),\n severity: t.optional(t.enum([\"info\", \"warning\", \"critical\"])),\n success: t.optional(t.boolean()),\n resourceType: t.optional(t.text()),\n search: t.optional(t.text()),\n from: t.optional(t.datetime()),\n to: t.optional(t.datetime()),\n });\n\n return (\n <Flex p=\"md\" flex={1} direction=\"column\">\n <DataTable<AuditEntity, typeof filters>\n submitOnInit\n defaultSize={20}\n defaultFilters={[\"search\", \"severity\"]}\n typeFormProps={{\n skipSubmitButton: true,\n columns: 4,\n }}\n tableProps={{\n horizontalSpacing: \"xs\",\n verticalSpacing: \"xs\",\n }}\n onFilterChange={(_key, _value, form) => form.submit()}\n filters={filters}\n tableTrProps={(item) => ({\n style: {\n cursor: item.userId ? \"pointer\" : \"default\",\n opacity: item.success ? 1 : 0.85,\n },\n onClick: () => {\n if (item.userId) {\n router.push(\"adminUserProfile\", {\n params: { userId: item.userId },\n });\n }\n },\n })}\n items={async (query) => {\n const response = await client.findAudits({\n query: {\n ...query,\n userRealm: props.userRealmName,\n },\n });\n return response as Page<AuditEntity>;\n }}\n columns={{\n type: {\n label: \"Type\",\n value: (item) => (\n <Badge size=\"sm\" variant=\"default\">\n {item.type}\n </Badge>\n ),\n },\n action: {\n label: \"Action\",\n value: (item) => (\n <Badge size=\"sm\" variant=\"default\">\n {item.action}\n </Badge>\n ),\n },\n severity: {\n label: \"Severity\",\n value: (item) => (\n <Badge\n size=\"sm\"\n variant=\"light\"\n color={\n item.severity === \"critical\"\n ? \"red\"\n : item.severity === \"warning\"\n ? \"yellow\"\n : \"gray\"\n }\n >\n {item.severity}\n </Badge>\n ),\n },\n user: {\n label: \"User\",\n value: (item) =>\n item.userId ? (\n <Tooltip\n label={\n <Flex direction=\"column\" gap={2}>\n <Text size=\"xs\">{item.userEmail || \"No email\"}</Text>\n <Text size=\"xs\" c=\"dimmed\">\n {item.userRealm || \"default\"}\n </Text>\n </Flex>\n }\n >\n <Flex gap={4}>\n <IconUser size={12} />\n <Text size=\"xs\" lineClamp={1} maw={100}>\n {item.userEmail?.split(\"@\")[0] || item.userId.slice(0, 8)}\n </Text>\n </Flex>\n </Tooltip>\n ) : (\n <Text size=\"xs\" c=\"dimmed\">\n System\n </Text>\n ),\n },\n description: {\n label: \"Description\",\n value: (item) => (\n <Text size=\"sm\" lineClamp={1}>\n {item.description || \"-\"}\n </Text>\n ),\n },\n resource: {\n label: \"Resource\",\n value: (item) =>\n item.resourceType ? (\n <Tooltip label={`${item.resourceType}: ${item.resourceId}`}>\n <Text size=\"xs\" ff=\"monospace\">\n {item.resourceType}\n </Text>\n </Tooltip>\n ) : (\n <Text size=\"xs\" c=\"dimmed\">\n -\n </Text>\n ),\n },\n success: {\n label: \"Status\",\n value: (item) =>\n item.success ? (\n <Badge size=\"sm\" variant=\"light\" color=\"green\">\n OK\n </Badge>\n ) : (\n <Tooltip label={item.errorMessage || \"Failed\"}>\n <Badge size=\"sm\" variant=\"light\" color=\"red\">\n Failed\n </Badge>\n </Tooltip>\n ),\n },\n ipAddress: {\n label: \"IP\",\n value: (item) => (\n <Text size=\"xs\" c=\"dimmed\" ff=\"monospace\">\n {item.ipAddress || \"-\"}\n </Text>\n ),\n },\n createdAt: {\n label: \"Time\",\n value: (item) => (\n <Tooltip label={l(item.createdAt, { date: \"medium\" })}>\n <Text size=\"xs\" c=\"dimmed\">\n {l(item.createdAt, { date: \"fromNow\" })}\n </Text>\n </Tooltip>\n ),\n },\n }}\n />\n </Flex>\n );\n};\n\nexport default AdminAudits;\n"],"mappings":"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"}
1
+ {"version":3,"file":"AdminAudits-C7c1CN4c.js","names":["Flex","Text"],"sources":["../../src/admin/components/audits/AdminAudits.tsx"],"sourcesContent":["import { DataTable, Flex, Text } from \"@alepha/ui\";\nimport { Badge, Tooltip } from \"@mantine/core\";\nimport { IconUser } from \"@tabler/icons-react\";\nimport { type Page, t } from \"alepha\";\nimport type { AdminAuditController, AuditEntity } from \"alepha/api/audits\";\nimport { useClient } from \"alepha/react\";\nimport { useI18n } from \"alepha/react/i18n\";\nimport { useRouter } from \"alepha/react/router\";\nimport type { AdminRouter } from \"../../AdminRouter.ts\";\n\nexport interface AdminAuditsProps {\n userRealmName?: string;\n}\n\nconst AdminAudits = (props: AdminAuditsProps) => {\n const client = useClient<AdminAuditController>();\n const router = useRouter<AdminRouter>();\n const { l } = useI18n();\n\n const filters = t.object({\n type: t.optional(t.text()),\n action: t.optional(t.text()),\n severity: t.optional(t.enum([\"info\", \"warning\", \"critical\"])),\n success: t.optional(t.boolean()),\n resourceType: t.optional(t.text()),\n search: t.optional(t.text()),\n from: t.optional(t.datetime()),\n to: t.optional(t.datetime()),\n });\n\n return (\n <Flex p=\"md\" flex={1} direction=\"column\">\n <DataTable<AuditEntity, typeof filters>\n submitOnInit\n defaultSize={20}\n defaultFilters={[\"search\", \"severity\"]}\n typeFormProps={{\n skipSubmitButton: true,\n columns: 4,\n }}\n tableProps={{\n horizontalSpacing: \"xs\",\n verticalSpacing: \"xs\",\n }}\n onFilterChange={(_key, _value, form) => form.submit()}\n filters={filters}\n tableTrProps={(item) => ({\n style: {\n cursor: item.userId ? \"pointer\" : \"default\",\n opacity: item.success ? 1 : 0.85,\n },\n onClick: () => {\n if (item.userId) {\n router.push(\"adminUserProfile\", {\n params: { userId: item.userId },\n });\n }\n },\n })}\n items={async (query) => {\n const response = await client.findAudits({\n query: {\n ...query,\n userRealm: props.userRealmName,\n },\n });\n return response as Page<AuditEntity>;\n }}\n columns={{\n type: {\n label: \"Type\",\n value: (item) => (\n <Badge size=\"sm\" variant=\"default\">\n {item.type}\n </Badge>\n ),\n },\n action: {\n label: \"Action\",\n value: (item) => (\n <Badge size=\"sm\" variant=\"default\">\n {item.action}\n </Badge>\n ),\n },\n severity: {\n label: \"Severity\",\n value: (item) => (\n <Badge\n size=\"sm\"\n variant=\"light\"\n color={\n item.severity === \"critical\"\n ? \"red\"\n : item.severity === \"warning\"\n ? \"yellow\"\n : \"gray\"\n }\n >\n {item.severity}\n </Badge>\n ),\n },\n user: {\n label: \"User\",\n value: (item) =>\n item.userId ? (\n <Tooltip\n label={\n <Flex direction=\"column\" gap={2}>\n <Text size=\"xs\">{item.userEmail || \"No email\"}</Text>\n <Text size=\"xs\" c=\"dimmed\">\n {item.userRealm || \"default\"}\n </Text>\n </Flex>\n }\n >\n <Flex gap={4}>\n <IconUser size={12} />\n <Text size=\"xs\" lineClamp={1} maw={100}>\n {item.userEmail?.split(\"@\")[0] || item.userId.slice(0, 8)}\n </Text>\n </Flex>\n </Tooltip>\n ) : (\n <Text size=\"xs\" c=\"dimmed\">\n System\n </Text>\n ),\n },\n description: {\n label: \"Description\",\n value: (item) => (\n <Text size=\"sm\" lineClamp={1}>\n {item.description || \"-\"}\n </Text>\n ),\n },\n resource: {\n label: \"Resource\",\n value: (item) =>\n item.resourceType ? (\n <Tooltip label={`${item.resourceType}: ${item.resourceId}`}>\n <Text size=\"xs\" ff=\"monospace\">\n {item.resourceType}\n </Text>\n </Tooltip>\n ) : (\n <Text size=\"xs\" c=\"dimmed\">\n -\n </Text>\n ),\n },\n success: {\n label: \"Status\",\n value: (item) =>\n item.success ? (\n <Badge size=\"sm\" variant=\"light\" color=\"green\">\n OK\n </Badge>\n ) : (\n <Tooltip label={item.errorMessage || \"Failed\"}>\n <Badge size=\"sm\" variant=\"light\" color=\"red\">\n Failed\n </Badge>\n </Tooltip>\n ),\n },\n ipAddress: {\n label: \"IP\",\n value: (item) => (\n <Text size=\"xs\" c=\"dimmed\" ff=\"monospace\">\n {item.ipAddress || \"-\"}\n </Text>\n ),\n },\n createdAt: {\n label: \"Time\",\n value: (item) => (\n <Tooltip label={l(item.createdAt, { date: \"medium\" })}>\n <Text size=\"xs\" c=\"dimmed\">\n {l(item.createdAt, { date: \"fromNow\" })}\n </Text>\n </Tooltip>\n ),\n },\n }}\n />\n </Flex>\n );\n};\n\nexport default AdminAudits;\n"],"mappings":"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"}
@@ -1,7 +1,6 @@
1
- import { l as Flex, s as Text } from "./core-CYaRQ8O-.js";
1
+ import { l as Flex, s as Text } from "./core-D1AbU50V.js";
2
2
  import { jsx, jsxs } from "react/jsx-runtime";
3
3
  import { useAuth } from "alepha/react/auth";
4
-
5
4
  //#region ../../src/admin/components/AdminDashboard.tsx
6
5
  const AdminDashboard = () => {
7
6
  const auth = useAuth();
@@ -61,7 +60,7 @@ const AdminDashboard = () => {
61
60
  })]
62
61
  });
63
62
  };
64
-
65
63
  //#endregion
66
64
  export { AdminDashboard as default };
67
- //# sourceMappingURL=AdminDashboard-PhC_dZqo.js.map
65
+
66
+ //# sourceMappingURL=AdminDashboard-C3RXpTp6.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"AdminDashboard-PhC_dZqo.js","names":[],"sources":["../../src/admin/components/AdminDashboard.tsx"],"sourcesContent":["import { Flex, Text } from \"@alepha/ui\";\nimport { useAuth } from \"alepha/react/auth\";\n\n// ─────────────────────────────────────────────────────────────────────────────\n\nconst AdminDashboard = () => {\n const auth = useAuth();\n\n return (\n <Flex elevated fill col pos=\"relative\" style={{ overflow: \"hidden\" }}>\n <Flex\n bg=\"yellow\"\n style={{\n position: \"absolute\",\n top: 0,\n left: 0,\n width: 256,\n height: 256,\n borderRadius: \"50%\",\n transform: \"translate(-50%, -50%)\",\n pointerEvents: \"none\",\n }}\n />\n\n {/* Header */}\n <Flex gap={4} p={128} pb=\"xl\" style={{ zIndex: 1 }}>\n <Flex col pos={\"relative\"}>\n <Text muted bold small uppercase>\n Account Home\n </Text>\n <Text\n top={10}\n pos={\"absolute\"}\n left={-20}\n fz={42}\n c={\"yellow\"}\n style={{ transform: \"rotate(-15deg)\" }}\n >\n !\n </Text>\n <Text fz={36}>Hello, {auth.user?.name ?? \"Admin\"}</Text>\n <Text muted>\n This is your admin dashboard. You can manage resources from the\n sidebar menu.\n </Text>\n </Flex>\n </Flex>\n </Flex>\n );\n};\n\nexport default AdminDashboard;\n"],"mappings":";;;;;AAKA,MAAM,uBAAuB;CAC3B,MAAM,OAAO,SAAS;AAEtB,QACE,qBAAC;EAAK;EAAS;EAAK;EAAI,KAAI;EAAW,OAAO,EAAE,UAAU,UAAU;aAClE,oBAAC;GACC,IAAG;GACH,OAAO;IACL,UAAU;IACV,KAAK;IACL,MAAM;IACN,OAAO;IACP,QAAQ;IACR,cAAc;IACd,WAAW;IACX,eAAe;IAChB;IACD,EAGF,oBAAC;GAAK,KAAK;GAAG,GAAG;GAAK,IAAG;GAAK,OAAO,EAAE,QAAQ,GAAG;aAChD,qBAAC;IAAK;IAAI,KAAK;;KACb,oBAAC;MAAK;MAAM;MAAK;MAAM;gBAAU;OAE1B;KACP,oBAAC;MACC,KAAK;MACL,KAAK;MACL,MAAM;MACN,IAAI;MACJ,GAAG;MACH,OAAO,EAAE,WAAW,kBAAkB;gBACvC;OAEM;KACP,qBAAC;MAAK,IAAI;iBAAI,WAAQ,KAAK,MAAM,QAAQ;OAAe;KACxD,oBAAC;MAAK;gBAAM;OAGL;;KACF;IACF;GACF"}
1
+ {"version":3,"file":"AdminDashboard-C3RXpTp6.js","names":[],"sources":["../../src/admin/components/AdminDashboard.tsx"],"sourcesContent":["import { Flex, Text } from \"@alepha/ui\";\nimport { useAuth } from \"alepha/react/auth\";\n\n// ─────────────────────────────────────────────────────────────────────────────\n\nconst AdminDashboard = () => {\n const auth = useAuth();\n\n return (\n <Flex elevated fill col pos=\"relative\" style={{ overflow: \"hidden\" }}>\n <Flex\n bg=\"yellow\"\n style={{\n position: \"absolute\",\n top: 0,\n left: 0,\n width: 256,\n height: 256,\n borderRadius: \"50%\",\n transform: \"translate(-50%, -50%)\",\n pointerEvents: \"none\",\n }}\n />\n\n {/* Header */}\n <Flex gap={4} p={128} pb=\"xl\" style={{ zIndex: 1 }}>\n <Flex col pos={\"relative\"}>\n <Text muted bold small uppercase>\n Account Home\n </Text>\n <Text\n top={10}\n pos={\"absolute\"}\n left={-20}\n fz={42}\n c={\"yellow\"}\n style={{ transform: \"rotate(-15deg)\" }}\n >\n !\n </Text>\n <Text fz={36}>Hello, {auth.user?.name ?? \"Admin\"}</Text>\n <Text muted>\n This is your admin dashboard. You can manage resources from the\n sidebar menu.\n </Text>\n </Flex>\n </Flex>\n </Flex>\n );\n};\n\nexport default AdminDashboard;\n"],"mappings":";;;;AAKA,MAAM,uBAAuB;CAC3B,MAAM,OAAO,SAAS;AAEtB,QACE,qBAAC,MAAD;EAAM,UAAA;EAAS,MAAA;EAAK,KAAA;EAAI,KAAI;EAAW,OAAO,EAAE,UAAU,UAAU;YAApE,CACE,oBAAC,MAAD;GACE,IAAG;GACH,OAAO;IACL,UAAU;IACV,KAAK;IACL,MAAM;IACN,OAAO;IACP,QAAQ;IACR,cAAc;IACd,WAAW;IACX,eAAe;IAChB;GACD,CAAA,EAGF,oBAAC,MAAD;GAAM,KAAK;GAAG,GAAG;GAAK,IAAG;GAAK,OAAO,EAAE,QAAQ,GAAG;aAChD,qBAAC,MAAD;IAAM,KAAA;IAAI,KAAK;cAAf;KACE,oBAAC,MAAD;MAAM,OAAA;MAAM,MAAA;MAAK,OAAA;MAAM,WAAA;gBAAU;MAE1B,CAAA;KACP,oBAAC,MAAD;MACE,KAAK;MACL,KAAK;MACL,MAAM;MACN,IAAI;MACJ,GAAG;MACH,OAAO,EAAE,WAAW,kBAAkB;gBACvC;MAEM,CAAA;KACP,qBAAC,MAAD;MAAM,IAAI;gBAAV,CAAc,WAAQ,KAAK,MAAM,QAAQ,QAAe;;KACxD,oBAAC,MAAD;MAAM,OAAA;gBAAM;MAGL,CAAA;KACF;;GACF,CAAA,CACF"}
@@ -1,10 +1,9 @@
1
- import { l as Flex, r as DataTable, s as Text } from "./core-CYaRQ8O-.js";
1
+ import { l as Flex, r as DataTable, s as Text } from "./core-D1AbU50V.js";
2
2
  import { t } from "alepha";
3
3
  import { useI18n } from "alepha/react/i18n";
4
4
  import { jsx } from "react/jsx-runtime";
5
5
  import { useClient } from "alepha/react";
6
6
  import { files } from "alepha/api/files";
7
-
8
7
  //#region ../../src/admin/components/files/AdminFiles.tsx
9
8
  const AdminFiles = () => {
10
9
  const client = useClient();
@@ -105,7 +104,7 @@ const AdminFiles = () => {
105
104
  })
106
105
  });
107
106
  };
108
-
109
107
  //#endregion
110
108
  export { AdminFiles as default };
111
- //# sourceMappingURL=AdminFiles-DFTjijGp.js.map
109
+
110
+ //# sourceMappingURL=AdminFiles-31ivR6Wq.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"AdminFiles-DFTjijGp.js","names":[],"sources":["../../src/admin/components/files/AdminFiles.tsx"],"sourcesContent":["import { DataTable, Flex, Text } from \"@alepha/ui\";\nimport { type Page, t } from \"alepha\";\nimport { type FileController, type FileEntity, files } from \"alepha/api/files\";\nimport { useClient } from \"alepha/react\";\nimport { useI18n } from \"alepha/react/i18n\";\n\nconst AdminFiles = () => {\n const client = useClient<FileController>();\n const { l } = useI18n();\n\n const filters = t.object({\n bucket: t.optional(t.string()),\n name: t.optional(\n t.string({\n $control: {\n query: t.pick(files.schema, [\"name\", \"bucket\", \"mimeType\"]),\n },\n }),\n ),\n });\n\n const formatFileSize = (bytes: number) => {\n if (bytes === 0) return \"0 B\";\n const k = 1024;\n const sizes = [\"B\", \"KB\", \"MB\", \"GB\"];\n const i = Math.floor(Math.log(bytes) / Math.log(k));\n return `${Number.parseFloat((bytes / k ** i).toFixed(1))} ${sizes[i]}`;\n };\n\n return (\n <Flex p=\"md\" flex={1} direction={\"column\"}>\n <DataTable<FileEntity, typeof filters>\n submitOnInit\n defaultSize={10}\n typeFormProps={{\n skipSubmitButton: true,\n columns: 3,\n }}\n tableProps={{\n horizontalSpacing: \"xs\",\n verticalSpacing: \"xs\",\n }}\n onFilterChange={(key, _value, form) => {\n if (key === \"name\" || key === \"bucket\") {\n return form.submit();\n }\n }}\n filters={filters}\n items={async (filters) => {\n const response = await client.findFiles({\n query: filters,\n });\n\n return response as Page<FileEntity>;\n }}\n columns={{\n name: {\n label: \"Name\",\n value: (item) => (\n <Text size=\"sm\" fw={500} lineClamp={1}>\n {item.name}\n </Text>\n ),\n },\n bucket: {\n label: \"Bucket\",\n value: (item) => (\n <Text size=\"xs\" ff=\"monospace\">\n {item.bucket}\n </Text>\n ),\n },\n mimeType: {\n label: \"Type\",\n value: (item) => (\n <Text size=\"xs\" c=\"dimmed\">\n {item.mimeType}\n </Text>\n ),\n },\n size: {\n label: \"Size\",\n value: (item) => (\n <Text size=\"xs\" c=\"dimmed\">\n {formatFileSize(item.size)}\n </Text>\n ),\n },\n creatorName: {\n label: \"Creator\",\n value: (item) => (\n <Text size=\"xs\" c=\"dimmed\">\n {item.creatorName || \"-\"}\n </Text>\n ),\n },\n createdAt: {\n label: \"Created\",\n value: (item) => (\n <Text size=\"xs\" c=\"dimmed\">\n {l(item.createdAt, { date: \"fromNow\" })}\n </Text>\n ),\n },\n }}\n />\n </Flex>\n );\n};\n\nexport default AdminFiles;\n"],"mappings":";;;;;;;;AAMA,MAAM,mBAAmB;CACvB,MAAM,SAAS,WAA2B;CAC1C,MAAM,EAAE,MAAM,SAAS;CAEvB,MAAM,UAAU,EAAE,OAAO;EACvB,QAAQ,EAAE,SAAS,EAAE,QAAQ,CAAC;EAC9B,MAAM,EAAE,SACN,EAAE,OAAO,EACP,UAAU,EACR,OAAO,EAAE,KAAK,MAAM,QAAQ;GAAC;GAAQ;GAAU;GAAW,CAAC,EAC5D,EACF,CAAC,CACH;EACF,CAAC;CAEF,MAAM,kBAAkB,UAAkB;AACxC,MAAI,UAAU,EAAG,QAAO;EACxB,MAAM,IAAI;EACV,MAAM,QAAQ;GAAC;GAAK;GAAM;GAAM;GAAK;EACrC,MAAM,IAAI,KAAK,MAAM,KAAK,IAAI,MAAM,GAAG,KAAK,IAAI,EAAE,CAAC;AACnD,SAAO,GAAG,OAAO,YAAY,QAAQ,KAAK,GAAG,QAAQ,EAAE,CAAC,CAAC,GAAG,MAAM;;AAGpE,QACE,oBAAC;EAAK,GAAE;EAAK,MAAM;EAAG,WAAW;YAC/B,oBAAC;GACC;GACA,aAAa;GACb,eAAe;IACb,kBAAkB;IAClB,SAAS;IACV;GACD,YAAY;IACV,mBAAmB;IACnB,iBAAiB;IAClB;GACD,iBAAiB,KAAK,QAAQ,SAAS;AACrC,QAAI,QAAQ,UAAU,QAAQ,SAC5B,QAAO,KAAK,QAAQ;;GAGf;GACT,OAAO,OAAO,YAAY;AAKxB,WAJiB,MAAM,OAAO,UAAU,EACtC,OAAO,SACR,CAAC;;GAIJ,SAAS;IACP,MAAM;KACJ,OAAO;KACP,QAAQ,SACN,oBAAC;MAAK,MAAK;MAAK,IAAI;MAAK,WAAW;gBACjC,KAAK;OACD;KAEV;IACD,QAAQ;KACN,OAAO;KACP,QAAQ,SACN,oBAAC;MAAK,MAAK;MAAK,IAAG;gBAChB,KAAK;OACD;KAEV;IACD,UAAU;KACR,OAAO;KACP,QAAQ,SACN,oBAAC;MAAK,MAAK;MAAK,GAAE;gBACf,KAAK;OACD;KAEV;IACD,MAAM;KACJ,OAAO;KACP,QAAQ,SACN,oBAAC;MAAK,MAAK;MAAK,GAAE;gBACf,eAAe,KAAK,KAAK;OACrB;KAEV;IACD,aAAa;KACX,OAAO;KACP,QAAQ,SACN,oBAAC;MAAK,MAAK;MAAK,GAAE;gBACf,KAAK,eAAe;OAChB;KAEV;IACD,WAAW;KACT,OAAO;KACP,QAAQ,SACN,oBAAC;MAAK,MAAK;MAAK,GAAE;gBACf,EAAE,KAAK,WAAW,EAAE,MAAM,WAAW,CAAC;OAClC;KAEV;IACF;IACD;GACG"}
1
+ {"version":3,"file":"AdminFiles-31ivR6Wq.js","names":[],"sources":["../../src/admin/components/files/AdminFiles.tsx"],"sourcesContent":["import { DataTable, Flex, Text } from \"@alepha/ui\";\nimport { type Page, t } from \"alepha\";\nimport { type FileController, type FileEntity, files } from \"alepha/api/files\";\nimport { useClient } from \"alepha/react\";\nimport { useI18n } from \"alepha/react/i18n\";\n\nconst AdminFiles = () => {\n const client = useClient<FileController>();\n const { l } = useI18n();\n\n const filters = t.object({\n bucket: t.optional(t.string()),\n name: t.optional(\n t.string({\n $control: {\n query: t.pick(files.schema, [\"name\", \"bucket\", \"mimeType\"]),\n },\n }),\n ),\n });\n\n const formatFileSize = (bytes: number) => {\n if (bytes === 0) return \"0 B\";\n const k = 1024;\n const sizes = [\"B\", \"KB\", \"MB\", \"GB\"];\n const i = Math.floor(Math.log(bytes) / Math.log(k));\n return `${Number.parseFloat((bytes / k ** i).toFixed(1))} ${sizes[i]}`;\n };\n\n return (\n <Flex p=\"md\" flex={1} direction={\"column\"}>\n <DataTable<FileEntity, typeof filters>\n submitOnInit\n defaultSize={10}\n typeFormProps={{\n skipSubmitButton: true,\n columns: 3,\n }}\n tableProps={{\n horizontalSpacing: \"xs\",\n verticalSpacing: \"xs\",\n }}\n onFilterChange={(key, _value, form) => {\n if (key === \"name\" || key === \"bucket\") {\n return form.submit();\n }\n }}\n filters={filters}\n items={async (filters) => {\n const response = await client.findFiles({\n query: filters,\n });\n\n return response as Page<FileEntity>;\n }}\n columns={{\n name: {\n label: \"Name\",\n value: (item) => (\n <Text size=\"sm\" fw={500} lineClamp={1}>\n {item.name}\n </Text>\n ),\n },\n bucket: {\n label: \"Bucket\",\n value: (item) => (\n <Text size=\"xs\" ff=\"monospace\">\n {item.bucket}\n </Text>\n ),\n },\n mimeType: {\n label: \"Type\",\n value: (item) => (\n <Text size=\"xs\" c=\"dimmed\">\n {item.mimeType}\n </Text>\n ),\n },\n size: {\n label: \"Size\",\n value: (item) => (\n <Text size=\"xs\" c=\"dimmed\">\n {formatFileSize(item.size)}\n </Text>\n ),\n },\n creatorName: {\n label: \"Creator\",\n value: (item) => (\n <Text size=\"xs\" c=\"dimmed\">\n {item.creatorName || \"-\"}\n </Text>\n ),\n },\n createdAt: {\n label: \"Created\",\n value: (item) => (\n <Text size=\"xs\" c=\"dimmed\">\n {l(item.createdAt, { date: \"fromNow\" })}\n </Text>\n ),\n },\n }}\n />\n </Flex>\n );\n};\n\nexport default AdminFiles;\n"],"mappings":";;;;;;;AAMA,MAAM,mBAAmB;CACvB,MAAM,SAAS,WAA2B;CAC1C,MAAM,EAAE,MAAM,SAAS;CAEvB,MAAM,UAAU,EAAE,OAAO;EACvB,QAAQ,EAAE,SAAS,EAAE,QAAQ,CAAC;EAC9B,MAAM,EAAE,SACN,EAAE,OAAO,EACP,UAAU,EACR,OAAO,EAAE,KAAK,MAAM,QAAQ;GAAC;GAAQ;GAAU;GAAW,CAAC,EAC5D,EACF,CAAC,CACH;EACF,CAAC;CAEF,MAAM,kBAAkB,UAAkB;AACxC,MAAI,UAAU,EAAG,QAAO;EACxB,MAAM,IAAI;EACV,MAAM,QAAQ;GAAC;GAAK;GAAM;GAAM;GAAK;EACrC,MAAM,IAAI,KAAK,MAAM,KAAK,IAAI,MAAM,GAAG,KAAK,IAAI,EAAE,CAAC;AACnD,SAAO,GAAG,OAAO,YAAY,QAAQ,KAAK,GAAG,QAAQ,EAAE,CAAC,CAAC,GAAG,MAAM;;AAGpE,QACE,oBAAC,MAAD;EAAM,GAAE;EAAK,MAAM;EAAG,WAAW;YAC/B,oBAAC,WAAD;GACE,cAAA;GACA,aAAa;GACb,eAAe;IACb,kBAAkB;IAClB,SAAS;IACV;GACD,YAAY;IACV,mBAAmB;IACnB,iBAAiB;IAClB;GACD,iBAAiB,KAAK,QAAQ,SAAS;AACrC,QAAI,QAAQ,UAAU,QAAQ,SAC5B,QAAO,KAAK,QAAQ;;GAGf;GACT,OAAO,OAAO,YAAY;AAKxB,WAJiB,MAAM,OAAO,UAAU,EACtC,OAAO,SACR,CAAC;;GAIJ,SAAS;IACP,MAAM;KACJ,OAAO;KACP,QAAQ,SACN,oBAAC,MAAD;MAAM,MAAK;MAAK,IAAI;MAAK,WAAW;gBACjC,KAAK;MACD,CAAA;KAEV;IACD,QAAQ;KACN,OAAO;KACP,QAAQ,SACN,oBAAC,MAAD;MAAM,MAAK;MAAK,IAAG;gBAChB,KAAK;MACD,CAAA;KAEV;IACD,UAAU;KACR,OAAO;KACP,QAAQ,SACN,oBAAC,MAAD;MAAM,MAAK;MAAK,GAAE;gBACf,KAAK;MACD,CAAA;KAEV;IACD,MAAM;KACJ,OAAO;KACP,QAAQ,SACN,oBAAC,MAAD;MAAM,MAAK;MAAK,GAAE;gBACf,eAAe,KAAK,KAAK;MACrB,CAAA;KAEV;IACD,aAAa;KACX,OAAO;KACP,QAAQ,SACN,oBAAC,MAAD;MAAM,MAAK;MAAK,GAAE;gBACf,KAAK,eAAe;MAChB,CAAA;KAEV;IACD,WAAW;KACT,OAAO;KACP,QAAQ,SACN,oBAAC,MAAD;MAAM,MAAK;MAAK,GAAE;gBACf,EAAE,KAAK,WAAW,EAAE,MAAM,WAAW,CAAC;MAClC,CAAA;KAEV;IACF;GACD,CAAA;EACG,CAAA"}
@@ -1,13 +1,30 @@
1
- import { _ as ActionButton, b as useToast, l as Flex$1, s as Text$1, u as StatCards } from "./core-CYaRQ8O-.js";
1
+ import { _ as ActionButton, b as useToast, l as Flex$1, s as Text$1, u as StatCards } from "./core-D1AbU50V.js";
2
2
  import { useI18n } from "alepha/react/i18n";
3
- import { Paper, SimpleGrid, Table } from "@mantine/core";
3
+ import { Paper, SegmentedControl, SimpleGrid, Table } from "@mantine/core";
4
4
  import { jsx, jsxs } from "react/jsx-runtime";
5
5
  import { useCallback, useEffect, useState } from "react";
6
6
  import { IconAlertTriangle, IconCircleCheck, IconPlayerPlay, IconRefresh, IconTerminal2 } from "@tabler/icons-react";
7
7
  import { useClient } from "alepha/react";
8
8
  import { AreaChart, BarChart, DonutChart } from "@mantine/charts";
9
-
10
9
  //#region ../../src/admin/components/jobs/AdminJobDashboard.tsx
10
+ const TIME_RANGES = [
11
+ {
12
+ label: "24h",
13
+ value: "1"
14
+ },
15
+ {
16
+ label: "7d",
17
+ value: "7"
18
+ },
19
+ {
20
+ label: "14d",
21
+ value: "14"
22
+ },
23
+ {
24
+ label: "30d",
25
+ value: "30"
26
+ }
27
+ ];
11
28
  const formatDuration = (start, end) => {
12
29
  const startTime = new Date(start).getTime();
13
30
  const duration = (end ? new Date(end).getTime() : Date.now()) - startTime;
@@ -20,21 +37,23 @@ const AdminJobDashboard = () => {
20
37
  const client = useClient();
21
38
  const { l } = useI18n();
22
39
  const toast = useToast();
40
+ const [days, setDays] = useState("7");
23
41
  const [stats, setStats] = useState(null);
24
42
  const [recent, setRecent] = useState([]);
25
43
  const [failures, setFailures] = useState([]);
26
44
  const [activity, setActivity] = useState([]);
27
45
  const [queueDepth, setQueueDepth] = useState([]);
46
+ const daysNum = Number(days);
28
47
  const loadData = useCallback(async () => {
29
48
  try {
30
49
  const [statsData, recentData, failureData, activityData, queueData] = await Promise.all([
31
- client.getJobStats(),
50
+ client.getJobStats({ query: { days: daysNum } }),
32
51
  client.findJobExecutions({ query: {
33
52
  sort: "-createdAt",
34
53
  size: 10
35
54
  } }),
36
- client.getJobTopFailures(),
37
- client.getJobActivity({ query: { days: 14 } }),
55
+ client.getJobTopFailures({ query: { days: daysNum } }),
56
+ client.getJobActivity({ query: { days: daysNum } }),
38
57
  client.getJobQueueDepth()
39
58
  ]);
40
59
  setStats(statsData);
@@ -45,7 +64,11 @@ const AdminJobDashboard = () => {
45
64
  } catch {
46
65
  toast.danger("Failed to load dashboard data");
47
66
  }
48
- }, [client, toast]);
67
+ }, [
68
+ client,
69
+ toast,
70
+ daysNum
71
+ ]);
49
72
  useEffect(() => {
50
73
  loadData();
51
74
  }, [loadData]);
@@ -92,6 +115,7 @@ const AdminJobDashboard = () => {
92
115
  color: "red"
93
116
  }
94
117
  ].filter((d) => d.value > 0) : [];
118
+ const rangeLabel = TIME_RANGES.find((r) => r.value === days)?.label ?? `${days}d`;
95
119
  return /* @__PURE__ */ jsxs(Flex$1, {
96
120
  flex: 1,
97
121
  direction: "column",
@@ -105,12 +129,28 @@ const AdminJobDashboard = () => {
105
129
  size: "lg",
106
130
  fw: 600,
107
131
  children: "Jobs Dashboard"
108
- }), /* @__PURE__ */ jsx(ActionButton, {
109
- tooltip: "Refresh",
110
- variant: "light",
111
- size: "sm",
112
- icon: IconRefresh,
113
- onClick: loadData
132
+ }), /* @__PURE__ */ jsxs(Flex$1, {
133
+ gap: "sm",
134
+ align: "center",
135
+ py: "xs",
136
+ px: "sm",
137
+ style: {
138
+ backgroundColor: "var(--mantine-color-body)",
139
+ borderRadius: "var(--mantine-radius-md)",
140
+ border: "1px solid var(--mantine-color-default-border)"
141
+ },
142
+ children: [/* @__PURE__ */ jsx(SegmentedControl, {
143
+ size: "xs",
144
+ data: TIME_RANGES,
145
+ value: days,
146
+ onChange: setDays
147
+ }), /* @__PURE__ */ jsx(ActionButton, {
148
+ tooltip: "Refresh",
149
+ variant: "minimal",
150
+ size: "sm",
151
+ icon: IconRefresh,
152
+ onClick: loadData
153
+ })]
114
154
  })]
115
155
  }),
116
156
  stats && /* @__PURE__ */ jsx(StatCards, { items: [
@@ -125,13 +165,13 @@ const AdminJobDashboard = () => {
125
165
  icon: IconPlayerPlay
126
166
  },
127
167
  {
128
- label: "Completed 24h",
129
- value: stats.completed24h,
168
+ label: `Completed (${rangeLabel})`,
169
+ value: stats.completed,
130
170
  icon: IconCircleCheck
131
171
  },
132
172
  {
133
- label: "Failed 24h",
134
- value: stats.failed24h,
173
+ label: `Failed (${rangeLabel})`,
174
+ value: stats.failed,
135
175
  icon: IconAlertTriangle
136
176
  }
137
177
  ] }),
@@ -145,11 +185,15 @@ const AdminJobDashboard = () => {
145
185
  p: "md",
146
186
  radius: "md",
147
187
  withBorder: true,
148
- children: [/* @__PURE__ */ jsx(Text$1, {
188
+ children: [/* @__PURE__ */ jsxs(Text$1, {
149
189
  size: "sm",
150
190
  fw: 600,
151
191
  mb: "sm",
152
- children: "Activity (14d)"
192
+ children: [
193
+ "Activity (",
194
+ rangeLabel,
195
+ ")"
196
+ ]
153
197
  }), activityChartData.length > 0 ? /* @__PURE__ */ jsx(AreaChart, {
154
198
  h: 220,
155
199
  data: activityChartData,
@@ -303,11 +347,15 @@ const AdminJobDashboard = () => {
303
347
  p: "md",
304
348
  radius: "md",
305
349
  withBorder: true,
306
- children: [/* @__PURE__ */ jsx(Text$1, {
350
+ children: [/* @__PURE__ */ jsxs(Text$1, {
307
351
  size: "sm",
308
352
  fw: 600,
309
353
  mb: "sm",
310
- children: "Top Failures (7d)"
354
+ children: [
355
+ "Top Failures (",
356
+ rangeLabel,
357
+ ")"
358
+ ]
311
359
  }), failures.length > 0 ? /* @__PURE__ */ jsxs(Table, { children: [/* @__PURE__ */ jsx(Table.Thead, { children: /* @__PURE__ */ jsxs(Table.Tr, { children: [
312
360
  /* @__PURE__ */ jsx(Table.Th, { children: "Job" }),
313
361
  /* @__PURE__ */ jsx(Table.Th, { children: "Count" }),
@@ -337,10 +385,10 @@ const AdminJobDashboard = () => {
337
385
  h: 100,
338
386
  align: "center",
339
387
  justify: "center",
340
- children: /* @__PURE__ */ jsx(Text$1, {
388
+ children: /* @__PURE__ */ jsxs(Text$1, {
341
389
  size: "sm",
342
390
  c: "dimmed",
343
- children: "No failures in the last 7 days"
391
+ children: ["No failures in the last ", rangeLabel]
344
392
  })
345
393
  })]
346
394
  })]
@@ -348,7 +396,7 @@ const AdminJobDashboard = () => {
348
396
  ]
349
397
  });
350
398
  };
351
-
352
399
  //#endregion
353
400
  export { AdminJobDashboard as default };
354
- //# sourceMappingURL=AdminJobDashboard-BL8gGPDp.js.map
401
+
402
+ //# sourceMappingURL=AdminJobDashboard-BABLe7hL.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"AdminJobDashboard-BABLe7hL.js","names":["Flex","Text"],"sources":["../../src/admin/components/jobs/AdminJobDashboard.tsx"],"sourcesContent":["import { ActionButton, Flex, StatCards, Text, useToast } from \"@alepha/ui\";\nimport { AreaChart, BarChart, DonutChart } from \"@mantine/charts\";\nimport { Paper, SegmentedControl, SimpleGrid, Table } from \"@mantine/core\";\nimport {\n IconAlertTriangle,\n IconCircleCheck,\n IconPlayerPlay,\n IconRefresh,\n IconTerminal2,\n} from \"@tabler/icons-react\";\nimport type {\n AdminJobController,\n JobActivityPoint,\n JobFailure,\n JobQueueDepth,\n JobStats,\n} from \"alepha/api/jobs\";\nimport { useClient } from \"alepha/react\";\nimport { useI18n } from \"alepha/react/i18n\";\nimport { useCallback, useEffect, useState } from \"react\";\n\n// ─────────────────────────────────────────────────────────────────────────────\n\nconst TIME_RANGES = [\n { label: \"24h\", value: \"1\" },\n { label: \"7d\", value: \"7\" },\n { label: \"14d\", value: \"14\" },\n { label: \"30d\", value: \"30\" },\n];\n\nconst formatDuration = (\n start: Date | string,\n end?: Date | string | null,\n): string => {\n const startTime = new Date(start).getTime();\n const endTime = end ? new Date(end).getTime() : Date.now();\n const duration = endTime - startTime;\n\n if (duration < 1000) return `${duration}ms`;\n if (duration < 60000) return `${(duration / 1000).toFixed(1)}s`;\n if (duration < 3600000)\n return `${Math.floor(duration / 60000)}m ${Math.floor((duration % 60000) / 1000)}s`;\n return `${Math.floor(duration / 3600000)}h ${Math.floor((duration % 3600000) / 60000)}m`;\n};\n\n// ─────────────────────────────────────────────────────────────────────────────\n\ninterface RecentExecution {\n id: string;\n jobName: string;\n status: string;\n startedAt?: string;\n completedAt?: string;\n}\n\n// ─────────────────────────────────────────────────────────────────────────────\n\nconst AdminJobDashboard = () => {\n const client = useClient<AdminJobController>();\n const { l } = useI18n();\n const toast = useToast();\n\n const [days, setDays] = useState(\"7\");\n const [stats, setStats] = useState<JobStats | null>(null);\n const [recent, setRecent] = useState<RecentExecution[]>([]);\n const [failures, setFailures] = useState<JobFailure[]>([]);\n const [activity, setActivity] = useState<JobActivityPoint[]>([]);\n const [queueDepth, setQueueDepth] = useState<JobQueueDepth[]>([]);\n\n const daysNum = Number(days);\n\n const loadData = useCallback(async () => {\n try {\n const [statsData, recentData, failureData, activityData, queueData] =\n await Promise.all([\n client.getJobStats({ query: { days: daysNum } }),\n client.findJobExecutions({ query: { sort: \"-createdAt\", size: 10 } }),\n client.getJobTopFailures({ query: { days: daysNum } }),\n client.getJobActivity({ query: { days: daysNum } }),\n client.getJobQueueDepth(),\n ]);\n setStats(statsData);\n setRecent(recentData.content as RecentExecution[]);\n setFailures(failureData);\n setActivity(activityData);\n setQueueDepth(queueData);\n } catch {\n toast.danger(\"Failed to load dashboard data\");\n }\n }, [client, toast, daysNum]);\n\n useEffect(() => {\n loadData();\n }, [loadData]);\n\n // Prepare chart data\n const activityChartData = activity.map((point) => ({\n date: new Date(point.date).toLocaleDateString(\"en-US\", {\n month: \"short\",\n day: \"numeric\",\n }),\n completed: point.completed,\n failed: point.failed,\n }));\n\n const queueChartData = queueDepth\n .filter(\n (q) => q.pending + q.running + q.scheduled + q.retrying + q.dead > 0,\n )\n .map((q) => ({\n job:\n q.jobName.length > 20\n ? `...${q.jobName.slice(q.jobName.length - 18)}`\n : q.jobName,\n pending: q.pending,\n running: q.running,\n scheduled: q.scheduled,\n retrying: q.retrying,\n dead: q.dead,\n }));\n\n const statusDonutData = stats\n ? [\n { name: \"Running\", value: stats.running, color: \"blue\" },\n { name: \"Pending\", value: stats.pending, color: \"gray\" },\n { name: \"Scheduled\", value: stats.scheduled, color: \"violet\" },\n { name: \"Retrying\", value: stats.retrying, color: \"yellow\" },\n { name: \"Dead\", value: stats.dead, color: \"red\" },\n ].filter((d) => d.value > 0)\n : [];\n\n const rangeLabel =\n TIME_RANGES.find((r) => r.value === days)?.label ?? `${days}d`;\n\n return (\n <Flex flex={1} direction=\"column\" gap=\"md\" p=\"md\">\n <Flex justify=\"space-between\" align=\"center\">\n <Text size=\"lg\" fw={600}>\n Jobs Dashboard\n </Text>\n <Flex\n gap=\"sm\"\n align=\"center\"\n py=\"xs\"\n px=\"sm\"\n style={{\n backgroundColor: \"var(--mantine-color-body)\",\n borderRadius: \"var(--mantine-radius-md)\",\n border: \"1px solid var(--mantine-color-default-border)\",\n }}\n >\n <SegmentedControl\n size=\"xs\"\n data={TIME_RANGES}\n value={days}\n onChange={setDays}\n />\n <ActionButton\n tooltip=\"Refresh\"\n variant=\"minimal\"\n size=\"sm\"\n icon={IconRefresh}\n onClick={loadData}\n />\n </Flex>\n </Flex>\n\n {/* Stats Cards */}\n {stats && (\n <StatCards\n items={[\n {\n label: \"Registered\",\n value: stats.registered,\n icon: IconTerminal2,\n },\n {\n label: \"Running\",\n value: stats.running,\n icon: IconPlayerPlay,\n },\n {\n label: `Completed (${rangeLabel})`,\n value: stats.completed,\n icon: IconCircleCheck,\n },\n {\n label: `Failed (${rangeLabel})`,\n value: stats.failed,\n icon: IconAlertTriangle,\n },\n ]}\n />\n )}\n\n {/* Charts Row */}\n <SimpleGrid cols={{ base: 1, md: 2 }} spacing=\"md\">\n {/* Activity Timeline */}\n <Paper p=\"md\" radius=\"md\" withBorder>\n <Text size=\"sm\" fw={600} mb=\"sm\">\n Activity ({rangeLabel})\n </Text>\n {activityChartData.length > 0 ? (\n <AreaChart\n h={220}\n data={activityChartData}\n dataKey=\"date\"\n series={[\n { name: \"completed\", label: \"Completed\", color: \"teal.6\" },\n { name: \"failed\", label: \"Failed\", color: \"red.6\" },\n ]}\n curveType=\"monotone\"\n withGradient\n withTooltip\n withDots={false}\n />\n ) : (\n <Flex h={220} align=\"center\" justify=\"center\">\n <Text size=\"sm\" c=\"dimmed\">\n No activity data\n </Text>\n </Flex>\n )}\n </Paper>\n\n {/* Current Status Donut */}\n <Paper p=\"md\" radius=\"md\" withBorder>\n <Text size=\"sm\" fw={600} mb=\"sm\">\n Active Executions\n </Text>\n {statusDonutData.length > 0 ? (\n <DonutChart\n h={220}\n data={statusDonutData}\n withTooltip\n tooltipDataSource=\"segment\"\n chartLabel={String(\n statusDonutData.reduce((sum, d) => sum + d.value, 0),\n )}\n />\n ) : (\n <Flex h={220} align=\"center\" justify=\"center\">\n <Text size=\"sm\" c=\"dimmed\">\n No active executions\n </Text>\n </Flex>\n )}\n </Paper>\n </SimpleGrid>\n\n {/* Queue Depth Chart */}\n {queueChartData.length > 0 && (\n <Paper p=\"md\" radius=\"md\" withBorder>\n <Text size=\"sm\" fw={600} mb=\"sm\">\n Queue Depth by Job\n </Text>\n <BarChart\n h={200}\n data={queueChartData}\n dataKey=\"job\"\n type=\"stacked\"\n series={[\n { name: \"running\", label: \"Running\", color: \"blue.6\" },\n { name: \"pending\", label: \"Pending\", color: \"gray.5\" },\n { name: \"scheduled\", label: \"Scheduled\", color: \"violet.5\" },\n { name: \"retrying\", label: \"Retrying\", color: \"yellow.5\" },\n { name: \"dead\", label: \"Dead\", color: \"red.6\" },\n ]}\n withTooltip\n withLegend\n />\n </Paper>\n )}\n\n {/* Recent Activity + Top Failures side by side */}\n <SimpleGrid cols={{ base: 1, md: 2 }} spacing=\"md\">\n {/* Recent Activity */}\n <Paper p=\"md\" radius=\"md\" withBorder>\n <Text size=\"sm\" fw={600} mb=\"sm\">\n Recent Executions\n </Text>\n <Table highlightOnHover>\n <Table.Thead>\n <Table.Tr>\n <Table.Th>Job</Table.Th>\n <Table.Th>Status</Table.Th>\n <Table.Th>Duration</Table.Th>\n </Table.Tr>\n </Table.Thead>\n <Table.Tbody>\n {recent.map((exec) => (\n <Table.Tr key={exec.id}>\n <Table.Td>\n <Text size=\"xs\" fw={500} ff=\"monospace\" lineClamp={1}>\n {exec.jobName}\n </Text>\n </Table.Td>\n <Table.Td>\n <Text size=\"xs\" ff=\"monospace\">\n {exec.status}\n </Text>\n </Table.Td>\n <Table.Td>\n <Text size=\"xs\" c=\"dimmed\" ff=\"monospace\">\n {exec.startedAt &&\n (exec.completedAt || exec.status === \"running\")\n ? formatDuration(exec.startedAt, exec.completedAt)\n : \"\\u2014\"}\n </Text>\n </Table.Td>\n </Table.Tr>\n ))}\n {recent.length === 0 && (\n <Table.Tr>\n <Table.Td colSpan={3}>\n <Text size=\"sm\" c=\"dimmed\" ta=\"center\">\n No recent executions\n </Text>\n </Table.Td>\n </Table.Tr>\n )}\n </Table.Tbody>\n </Table>\n </Paper>\n\n {/* Top Failures */}\n <Paper p=\"md\" radius=\"md\" withBorder>\n <Text size=\"sm\" fw={600} mb=\"sm\">\n Top Failures ({rangeLabel})\n </Text>\n {failures.length > 0 ? (\n <Table>\n <Table.Thead>\n <Table.Tr>\n <Table.Th>Job</Table.Th>\n <Table.Th>Count</Table.Th>\n <Table.Th>Last Error</Table.Th>\n </Table.Tr>\n </Table.Thead>\n <Table.Tbody>\n {failures.map((f) => (\n <Table.Tr key={f.jobName}>\n <Table.Td>\n <Text size=\"xs\" fw={500} ff=\"monospace\" lineClamp={1}>\n {f.jobName}\n </Text>\n </Table.Td>\n <Table.Td>\n <Text size=\"xs\" fw={600} ff=\"monospace\">\n {f.failures}\n </Text>\n </Table.Td>\n <Table.Td>\n <Text\n size=\"xs\"\n c=\"dimmed\"\n lineClamp={1}\n style={{ maxWidth: 200 }}\n >\n {f.lastError ?? \"\\u2014\"}\n </Text>\n </Table.Td>\n </Table.Tr>\n ))}\n </Table.Tbody>\n </Table>\n ) : (\n <Flex h={100} align=\"center\" justify=\"center\">\n <Text size=\"sm\" c=\"dimmed\">\n No failures in the last {rangeLabel}\n </Text>\n </Flex>\n )}\n </Paper>\n </SimpleGrid>\n </Flex>\n );\n};\n\nexport default AdminJobDashboard;\n"],"mappings":";;;;;;;;;AAuBA,MAAM,cAAc;CAClB;EAAE,OAAO;EAAO,OAAO;EAAK;CAC5B;EAAE,OAAO;EAAM,OAAO;EAAK;CAC3B;EAAE,OAAO;EAAO,OAAO;EAAM;CAC7B;EAAE,OAAO;EAAO,OAAO;EAAM;CAC9B;AAED,MAAM,kBACJ,OACA,QACW;CACX,MAAM,YAAY,IAAI,KAAK,MAAM,CAAC,SAAS;CAE3C,MAAM,YADU,MAAM,IAAI,KAAK,IAAI,CAAC,SAAS,GAAG,KAAK,KAAK,IAC/B;AAE3B,KAAI,WAAW,IAAM,QAAO,GAAG,SAAS;AACxC,KAAI,WAAW,IAAO,QAAO,IAAI,WAAW,KAAM,QAAQ,EAAE,CAAC;AAC7D,KAAI,WAAW,KACb,QAAO,GAAG,KAAK,MAAM,WAAW,IAAM,CAAC,IAAI,KAAK,MAAO,WAAW,MAAS,IAAK,CAAC;AACnF,QAAO,GAAG,KAAK,MAAM,WAAW,KAAQ,CAAC,IAAI,KAAK,MAAO,WAAW,OAAW,IAAM,CAAC;;AAexF,MAAM,0BAA0B;CAC9B,MAAM,SAAS,WAA+B;CAC9C,MAAM,EAAE,MAAM,SAAS;CACvB,MAAM,QAAQ,UAAU;CAExB,MAAM,CAAC,MAAM,WAAW,SAAS,IAAI;CACrC,MAAM,CAAC,OAAO,YAAY,SAA0B,KAAK;CACzD,MAAM,CAAC,QAAQ,aAAa,SAA4B,EAAE,CAAC;CAC3D,MAAM,CAAC,UAAU,eAAe,SAAuB,EAAE,CAAC;CAC1D,MAAM,CAAC,UAAU,eAAe,SAA6B,EAAE,CAAC;CAChE,MAAM,CAAC,YAAY,iBAAiB,SAA0B,EAAE,CAAC;CAEjE,MAAM,UAAU,OAAO,KAAK;CAE5B,MAAM,WAAW,YAAY,YAAY;AACvC,MAAI;GACF,MAAM,CAAC,WAAW,YAAY,aAAa,cAAc,aACvD,MAAM,QAAQ,IAAI;IAChB,OAAO,YAAY,EAAE,OAAO,EAAE,MAAM,SAAS,EAAE,CAAC;IAChD,OAAO,kBAAkB,EAAE,OAAO;KAAE,MAAM;KAAc,MAAM;KAAI,EAAE,CAAC;IACrE,OAAO,kBAAkB,EAAE,OAAO,EAAE,MAAM,SAAS,EAAE,CAAC;IACtD,OAAO,eAAe,EAAE,OAAO,EAAE,MAAM,SAAS,EAAE,CAAC;IACnD,OAAO,kBAAkB;IAC1B,CAAC;AACJ,YAAS,UAAU;AACnB,aAAU,WAAW,QAA6B;AAClD,eAAY,YAAY;AACxB,eAAY,aAAa;AACzB,iBAAc,UAAU;UAClB;AACN,SAAM,OAAO,gCAAgC;;IAE9C;EAAC;EAAQ;EAAO;EAAQ,CAAC;AAE5B,iBAAgB;AACd,YAAU;IACT,CAAC,SAAS,CAAC;CAGd,MAAM,oBAAoB,SAAS,KAAK,WAAW;EACjD,MAAM,IAAI,KAAK,MAAM,KAAK,CAAC,mBAAmB,SAAS;GACrD,OAAO;GACP,KAAK;GACN,CAAC;EACF,WAAW,MAAM;EACjB,QAAQ,MAAM;EACf,EAAE;CAEH,MAAM,iBAAiB,WACpB,QACE,MAAM,EAAE,UAAU,EAAE,UAAU,EAAE,YAAY,EAAE,WAAW,EAAE,OAAO,EACpE,CACA,KAAK,OAAO;EACX,KACE,EAAE,QAAQ,SAAS,KACf,MAAM,EAAE,QAAQ,MAAM,EAAE,QAAQ,SAAS,GAAG,KAC5C,EAAE;EACR,SAAS,EAAE;EACX,SAAS,EAAE;EACX,WAAW,EAAE;EACb,UAAU,EAAE;EACZ,MAAM,EAAE;EACT,EAAE;CAEL,MAAM,kBAAkB,QACpB;EACE;GAAE,MAAM;GAAW,OAAO,MAAM;GAAS,OAAO;GAAQ;EACxD;GAAE,MAAM;GAAW,OAAO,MAAM;GAAS,OAAO;GAAQ;EACxD;GAAE,MAAM;GAAa,OAAO,MAAM;GAAW,OAAO;GAAU;EAC9D;GAAE,MAAM;GAAY,OAAO,MAAM;GAAU,OAAO;GAAU;EAC5D;GAAE,MAAM;GAAQ,OAAO,MAAM;GAAM,OAAO;GAAO;EAClD,CAAC,QAAQ,MAAM,EAAE,QAAQ,EAAE,GAC5B,EAAE;CAEN,MAAM,aACJ,YAAY,MAAM,MAAM,EAAE,UAAU,KAAK,EAAE,SAAS,GAAG,KAAK;AAE9D,QACE,qBAACA,QAAD;EAAM,MAAM;EAAG,WAAU;EAAS,KAAI;EAAK,GAAE;YAA7C;GACE,qBAACA,QAAD;IAAM,SAAQ;IAAgB,OAAM;cAApC,CACE,oBAACC,QAAD;KAAM,MAAK;KAAK,IAAI;eAAK;KAElB,CAAA,EACP,qBAACD,QAAD;KACE,KAAI;KACJ,OAAM;KACN,IAAG;KACH,IAAG;KACH,OAAO;MACL,iBAAiB;MACjB,cAAc;MACd,QAAQ;MACT;eATH,CAWE,oBAAC,kBAAD;MACE,MAAK;MACL,MAAM;MACN,OAAO;MACP,UAAU;MACV,CAAA,EACF,oBAAC,cAAD;MACE,SAAQ;MACR,SAAQ;MACR,MAAK;MACL,MAAM;MACN,SAAS;MACT,CAAA,CACG;OACF;;GAGN,SACC,oBAAC,WAAD,EACE,OAAO;IACL;KACE,OAAO;KACP,OAAO,MAAM;KACb,MAAM;KACP;IACD;KACE,OAAO;KACP,OAAO,MAAM;KACb,MAAM;KACP;IACD;KACE,OAAO,cAAc,WAAW;KAChC,OAAO,MAAM;KACb,MAAM;KACP;IACD;KACE,OAAO,WAAW,WAAW;KAC7B,OAAO,MAAM;KACb,MAAM;KACP;IACF,EACD,CAAA;GAIJ,qBAAC,YAAD;IAAY,MAAM;KAAE,MAAM;KAAG,IAAI;KAAG;IAAE,SAAQ;cAA9C,CAEE,qBAAC,OAAD;KAAO,GAAE;KAAK,QAAO;KAAK,YAAA;eAA1B,CACE,qBAACC,QAAD;MAAM,MAAK;MAAK,IAAI;MAAK,IAAG;gBAA5B;OAAiC;OACpB;OAAW;OACjB;SACN,kBAAkB,SAAS,IAC1B,oBAAC,WAAD;MACE,GAAG;MACH,MAAM;MACN,SAAQ;MACR,QAAQ,CACN;OAAE,MAAM;OAAa,OAAO;OAAa,OAAO;OAAU,EAC1D;OAAE,MAAM;OAAU,OAAO;OAAU,OAAO;OAAS,CACpD;MACD,WAAU;MACV,cAAA;MACA,aAAA;MACA,UAAU;MACV,CAAA,GAEF,oBAACD,QAAD;MAAM,GAAG;MAAK,OAAM;MAAS,SAAQ;gBACnC,oBAACC,QAAD;OAAM,MAAK;OAAK,GAAE;iBAAS;OAEpB,CAAA;MACF,CAAA,CAEH;QAGR,qBAAC,OAAD;KAAO,GAAE;KAAK,QAAO;KAAK,YAAA;eAA1B,CACE,oBAACA,QAAD;MAAM,MAAK;MAAK,IAAI;MAAK,IAAG;gBAAK;MAE1B,CAAA,EACN,gBAAgB,SAAS,IACxB,oBAAC,YAAD;MACE,GAAG;MACH,MAAM;MACN,aAAA;MACA,mBAAkB;MAClB,YAAY,OACV,gBAAgB,QAAQ,KAAK,MAAM,MAAM,EAAE,OAAO,EAAE,CACrD;MACD,CAAA,GAEF,oBAACD,QAAD;MAAM,GAAG;MAAK,OAAM;MAAS,SAAQ;gBACnC,oBAACC,QAAD;OAAM,MAAK;OAAK,GAAE;iBAAS;OAEpB,CAAA;MACF,CAAA,CAEH;OACG;;GAGZ,eAAe,SAAS,KACvB,qBAAC,OAAD;IAAO,GAAE;IAAK,QAAO;IAAK,YAAA;cAA1B,CACE,oBAACA,QAAD;KAAM,MAAK;KAAK,IAAI;KAAK,IAAG;eAAK;KAE1B,CAAA,EACP,oBAAC,UAAD;KACE,GAAG;KACH,MAAM;KACN,SAAQ;KACR,MAAK;KACL,QAAQ;MACN;OAAE,MAAM;OAAW,OAAO;OAAW,OAAO;OAAU;MACtD;OAAE,MAAM;OAAW,OAAO;OAAW,OAAO;OAAU;MACtD;OAAE,MAAM;OAAa,OAAO;OAAa,OAAO;OAAY;MAC5D;OAAE,MAAM;OAAY,OAAO;OAAY,OAAO;OAAY;MAC1D;OAAE,MAAM;OAAQ,OAAO;OAAQ,OAAO;OAAS;MAChD;KACD,aAAA;KACA,YAAA;KACA,CAAA,CACI;;GAIV,qBAAC,YAAD;IAAY,MAAM;KAAE,MAAM;KAAG,IAAI;KAAG;IAAE,SAAQ;cAA9C,CAEE,qBAAC,OAAD;KAAO,GAAE;KAAK,QAAO;KAAK,YAAA;eAA1B,CACE,oBAACA,QAAD;MAAM,MAAK;MAAK,IAAI;MAAK,IAAG;gBAAK;MAE1B,CAAA,EACP,qBAAC,OAAD;MAAO,kBAAA;gBAAP,CACE,oBAAC,MAAM,OAAP,EAAA,UACE,qBAAC,MAAM,IAAP,EAAA,UAAA;OACE,oBAAC,MAAM,IAAP,EAAA,UAAU,OAAc,CAAA;OACxB,oBAAC,MAAM,IAAP,EAAA,UAAU,UAAiB,CAAA;OAC3B,oBAAC,MAAM,IAAP,EAAA,UAAU,YAAmB,CAAA;OACpB,EAAA,CAAA,EACC,CAAA,EACd,qBAAC,MAAM,OAAP,EAAA,UAAA,CACG,OAAO,KAAK,SACX,qBAAC,MAAM,IAAP,EAAA,UAAA;OACE,oBAAC,MAAM,IAAP,EAAA,UACE,oBAACA,QAAD;QAAM,MAAK;QAAK,IAAI;QAAK,IAAG;QAAY,WAAW;kBAChD,KAAK;QACD,CAAA,EACE,CAAA;OACX,oBAAC,MAAM,IAAP,EAAA,UACE,oBAACA,QAAD;QAAM,MAAK;QAAK,IAAG;kBAChB,KAAK;QACD,CAAA,EACE,CAAA;OACX,oBAAC,MAAM,IAAP,EAAA,UACE,oBAACA,QAAD;QAAM,MAAK;QAAK,GAAE;QAAS,IAAG;kBAC3B,KAAK,cACL,KAAK,eAAe,KAAK,WAAW,aACjC,eAAe,KAAK,WAAW,KAAK,YAAY,GAChD;QACC,CAAA,EACE,CAAA;OACF,EAAA,EAnBI,KAAK,GAmBT,CACX,EACD,OAAO,WAAW,KACjB,oBAAC,MAAM,IAAP,EAAA,UACE,oBAAC,MAAM,IAAP;OAAU,SAAS;iBACjB,oBAACA,QAAD;QAAM,MAAK;QAAK,GAAE;QAAS,IAAG;kBAAS;QAEhC,CAAA;OACE,CAAA,EACF,CAAA,CAED,EAAA,CAAA,CACR;QACF;QAGR,qBAAC,OAAD;KAAO,GAAE;KAAK,QAAO;KAAK,YAAA;eAA1B,CACE,qBAACA,QAAD;MAAM,MAAK;MAAK,IAAI;MAAK,IAAG;gBAA5B;OAAiC;OAChB;OAAW;OACrB;SACN,SAAS,SAAS,IACjB,qBAAC,OAAD,EAAA,UAAA,CACE,oBAAC,MAAM,OAAP,EAAA,UACE,qBAAC,MAAM,IAAP,EAAA,UAAA;MACE,oBAAC,MAAM,IAAP,EAAA,UAAU,OAAc,CAAA;MACxB,oBAAC,MAAM,IAAP,EAAA,UAAU,SAAgB,CAAA;MAC1B,oBAAC,MAAM,IAAP,EAAA,UAAU,cAAqB,CAAA;MACtB,EAAA,CAAA,EACC,CAAA,EACd,oBAAC,MAAM,OAAP,EAAA,UACG,SAAS,KAAK,MACb,qBAAC,MAAM,IAAP,EAAA,UAAA;MACE,oBAAC,MAAM,IAAP,EAAA,UACE,oBAACA,QAAD;OAAM,MAAK;OAAK,IAAI;OAAK,IAAG;OAAY,WAAW;iBAChD,EAAE;OACE,CAAA,EACE,CAAA;MACX,oBAAC,MAAM,IAAP,EAAA,UACE,oBAACA,QAAD;OAAM,MAAK;OAAK,IAAI;OAAK,IAAG;iBACzB,EAAE;OACE,CAAA,EACE,CAAA;MACX,oBAAC,MAAM,IAAP,EAAA,UACE,oBAACA,QAAD;OACE,MAAK;OACL,GAAE;OACF,WAAW;OACX,OAAO,EAAE,UAAU,KAAK;iBAEvB,EAAE,aAAa;OACX,CAAA,EACE,CAAA;MACF,EAAA,EArBI,EAAE,QAqBN,CACX,EACU,CAAA,CACR,EAAA,CAAA,GAER,oBAACD,QAAD;MAAM,GAAG;MAAK,OAAM;MAAS,SAAQ;gBACnC,qBAACC,QAAD;OAAM,MAAK;OAAK,GAAE;iBAAlB,CAA2B,4BACA,WACpB;;MACF,CAAA,CAEH;OACG;;GACR"}
@@ -1,4 +1,4 @@
1
- import { _ as ActionButton, b as useToast, d as DetailList, l as Flex$1, m as useDialog, r as DataTable, s as Text$1 } from "./core-CYaRQ8O-.js";
1
+ import { _ as ActionButton, b as useToast, d as DetailList, l as Flex$1, m as useDialog, r as DataTable, s as Text$1 } from "./core-D1AbU50V.js";
2
2
  import { t } from "alepha";
3
3
  import { useI18n } from "alepha/react/i18n";
4
4
  import { Badge, Code, Paper, Table } from "@mantine/core";
@@ -6,7 +6,6 @@ import { jsx, jsxs } from "react/jsx-runtime";
6
6
  import { useCallback, useEffect, useState } from "react";
7
7
  import { IconCircleX, IconRefresh } from "@tabler/icons-react";
8
8
  import { useClient } from "alepha/react";
9
-
10
9
  //#region ../../src/admin/components/jobs/AdminJobExecutions.tsx
11
10
  const PRIORITY_LABELS = {
12
11
  0: "Critical",
@@ -624,7 +623,7 @@ const ExecutionDetailContent = ({ item, onRetry, onCancel }) => {
624
623
  ]
625
624
  });
626
625
  };
627
-
628
626
  //#endregion
629
627
  export { AdminJobExecutions as default };
630
- //# sourceMappingURL=AdminJobExecutions-D9E-CS-U.js.map
628
+
629
+ //# sourceMappingURL=AdminJobExecutions-D-G8RIlr.js.map