@alepha/ui 0.18.3 → 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.
- package/dist/admin/{AdminApiKeys-Dy_k-4Vd.js → AdminApiKeys-C2ze85eD.js} +3 -4
- package/dist/admin/{AdminApiKeys-Dy_k-4Vd.js.map → AdminApiKeys-C2ze85eD.js.map} +1 -1
- package/dist/admin/{AdminAudits-CKiFMSSU.js → AdminAudits-BIj81e4k.js} +3 -4
- package/dist/admin/{AdminAudits-CKiFMSSU.js.map → AdminAudits-BIj81e4k.js.map} +1 -1
- package/dist/admin/{AdminDashboard-PhC_dZqo.js → AdminDashboard-PMVzrwSu.js} +3 -4
- package/dist/admin/{AdminDashboard-PhC_dZqo.js.map → AdminDashboard-PMVzrwSu.js.map} +1 -1
- package/dist/admin/AdminFiles-Bq03BLt-.js +189 -0
- package/dist/admin/AdminFiles-Bq03BLt-.js.map +1 -0
- package/dist/admin/{AdminJobExecutions-D9E-CS-U.js → AdminJobs-D1_QGCDy.js} +401 -358
- package/dist/admin/AdminJobs-D1_QGCDy.js.map +1 -0
- package/dist/admin/{AdminLayout-I6TlUMPc.js → AdminLayout-BNiwiw2D.js} +8 -25
- package/dist/admin/AdminLayout-BNiwiw2D.js.map +1 -0
- package/dist/admin/{AdminNotifications-ZPHCYrv7.js → AdminNotifications-DSKQtUfn.js} +85 -124
- package/dist/admin/AdminNotifications-DSKQtUfn.js.map +1 -0
- package/dist/admin/{AdminParameters-CqgvhRsb.js → AdminParameters-CoB7EhyM.js} +3 -12
- package/dist/admin/{AdminParameters-CqgvhRsb.js.map → AdminParameters-CoB7EhyM.js.map} +1 -1
- package/dist/admin/{AdminSessions-Bz5NRuoW.js → AdminSessions-DFbFcrJQ.js} +3 -4
- package/dist/admin/{AdminSessions-Bz5NRuoW.js.map → AdminSessions-DFbFcrJQ.js.map} +1 -1
- package/dist/admin/{AdminUserLayout-lXT6I0Qq.js → AdminUserLayout-fSfi3KMm.js} +72 -111
- package/dist/admin/AdminUserLayout-fSfi3KMm.js.map +1 -0
- package/dist/admin/{AdminUserProfile-vFBLoJ3h.js → AdminUserProfile-_C-h8vUK.js} +7 -6
- package/dist/admin/AdminUserProfile-_C-h8vUK.js.map +1 -0
- package/dist/admin/{AdminUserSessions-CT_YDim0.js → AdminUserSessions-KpJHIeQo.js} +3 -4
- package/dist/admin/{AdminUserSessions-CT_YDim0.js.map → AdminUserSessions-KpJHIeQo.js.map} +1 -1
- package/dist/admin/{AdminUsers-D1UfGya9.js → AdminUsers-DcVrzdQP.js} +4 -4
- package/dist/admin/AdminUsers-DcVrzdQP.js.map +1 -0
- package/dist/admin/{AuthLayout-_frhdgOO.js → AuthLayout-CazfLzcf.js} +3 -4
- package/dist/admin/{AuthLayout-_frhdgOO.js.map → AuthLayout-CazfLzcf.js.map} +1 -1
- package/dist/{demo/IconGoogle-CSQLPYwX.js → admin/IconGoogle-8Nkx6yax.js} +2 -4
- package/dist/admin/{IconGoogle-Ch1m3Uzl.js.map → IconGoogle-8Nkx6yax.js.map} +1 -1
- package/dist/admin/{Login-xtNmQtGh.js → Login-CaMjUrDP.js} +5 -6
- package/dist/{auth/Login-BA1E8IZl.js.map → admin/Login-CaMjUrDP.js.map} +1 -1
- package/dist/admin/{Profile-_AtPUwAP.js → Profile-Ca4fZX15.js} +3 -5
- package/dist/{demo/Profile-DS5q4vOh.js.map → admin/Profile-Ca4fZX15.js.map} +1 -1
- package/dist/admin/{Register-JcCjHUUn.js → Register-C5DyKWPO.js} +5 -6
- package/dist/{demo/Register-B4hLBeEv.js.map → admin/Register-C5DyKWPO.js.map} +1 -1
- package/dist/admin/{ResetPassword-CwGBPLJO.js → ResetPassword-BA5sAgXo.js} +4 -5
- package/dist/{auth/ResetPassword-DCtGcneA.js.map → admin/ResetPassword-BA5sAgXo.js.map} +1 -1
- package/dist/admin/{VerifyEmail-hNxWejWf.js → VerifyEmail-DKNXROj_.js} +4 -5
- package/dist/{auth/VerifyEmail-DkH7NBfn.js.map → admin/VerifyEmail-DKNXROj_.js.map} +1 -1
- package/dist/admin/adminUserAtom-BLNc7XbT.js +11 -0
- package/dist/admin/adminUserAtom-BLNc7XbT.js.map +1 -0
- package/dist/admin/{core-CYaRQ8O-.js → core-CJCEx18C.js} +132 -86
- package/dist/admin/core-CJCEx18C.js.map +1 -0
- package/dist/admin/index.d.ts +80 -13
- package/dist/admin/index.d.ts.map +1 -1
- package/dist/admin/index.js +38 -68
- package/dist/admin/index.js.map +1 -1
- package/dist/admin/rolldown-runtime-CiIaOW0V.js +13 -0
- package/dist/{demo/AuthLayout-Brri4A-L.js → auth/AuthLayout-vXPcCVzp.js} +3 -4
- package/dist/auth/{AuthLayout-AvLlcLjS.js.map → AuthLayout-vXPcCVzp.js.map} +1 -1
- package/dist/{admin/IconGoogle-Ch1m3Uzl.js → auth/IconGoogle-8Nkx6yax.js} +2 -4
- package/dist/auth/{IconGoogle-Ch1m3Uzl.js.map → IconGoogle-8Nkx6yax.js.map} +1 -1
- package/dist/auth/{Login-BA1E8IZl.js → Login-Dg08QR20.js} +5 -6
- package/dist/{demo/Login-C12N4oGs.js.map → auth/Login-Dg08QR20.js.map} +1 -1
- package/dist/{demo/Profile-DS5q4vOh.js → auth/Profile-Bb5O1yeh.js} +3 -5
- package/dist/auth/{Profile-YcWdeuFz.js.map → Profile-Bb5O1yeh.js.map} +1 -1
- package/dist/auth/{Register-CPhEO5MG.js → Register-B2AN71NC.js} +5 -6
- package/dist/{admin/Register-JcCjHUUn.js.map → auth/Register-B2AN71NC.js.map} +1 -1
- package/dist/{demo/ResetPassword-D8g9ha1N.js → auth/ResetPassword-BLxwzbDj.js} +4 -5
- package/dist/{admin/ResetPassword-CwGBPLJO.js.map → auth/ResetPassword-BLxwzbDj.js.map} +1 -1
- package/dist/auth/{VerifyEmail-DkH7NBfn.js → VerifyEmail-CSDOk3Zm.js} +4 -5
- package/dist/{admin/VerifyEmail-hNxWejWf.js.map → auth/VerifyEmail-CSDOk3Zm.js.map} +1 -1
- package/dist/auth/{core-D5jIAVF2.js → core-DuGkjPiU.js} +23 -54
- package/dist/auth/core-DuGkjPiU.js.map +1 -0
- package/dist/auth/index.d.ts +20 -6
- package/dist/auth/index.d.ts.map +1 -1
- package/dist/auth/index.js +13 -18
- package/dist/auth/index.js.map +1 -1
- package/dist/auth/rolldown-runtime-CiIaOW0V.js +13 -0
- package/dist/core/index.d.ts +78 -20
- package/dist/core/index.d.ts.map +1 -1
- package/dist/core/index.js +130 -98
- package/dist/core/index.js.map +1 -1
- package/dist/{auth/AuthLayout-AvLlcLjS.js → demo/AuthLayout-DPsOOG4u.js} +3 -4
- package/dist/demo/{AuthLayout-Brri4A-L.js.map → AuthLayout-DPsOOG4u.js.map} +1 -1
- package/dist/demo/{DemoButton-wiCxZZ_L.js → DemoButton-wzcqGk4u.js} +4 -5
- package/dist/demo/{DemoButton-wiCxZZ_L.js.map → DemoButton-wzcqGk4u.js.map} +1 -1
- package/dist/demo/{DemoControlSelect-D7ILObVg.js → DemoControlSelect-CMWvQ6Gm.js} +4 -5
- package/dist/demo/{DemoControlSelect-D7ILObVg.js.map → DemoControlSelect-CMWvQ6Gm.js.map} +1 -1
- package/dist/demo/{DemoDataTable-DZ5Y8pFX.js → DemoDataTable-CHsAP3e2.js} +4 -5
- package/dist/demo/{DemoDataTable-DZ5Y8pFX.js.map → DemoDataTable-CHsAP3e2.js.map} +1 -1
- package/dist/demo/{DemoDialog-CUWdLHim.js → DemoDialog-Co2IePxX.js} +3 -4
- package/dist/demo/{DemoDialog-CUWdLHim.js.map → DemoDialog-Co2IePxX.js.map} +1 -1
- package/dist/demo/{DemoFlex-a8OhMMvq.js → DemoFlex-OEwQt5do.js} +4 -5
- package/dist/demo/{DemoFlex-a8OhMMvq.js.map → DemoFlex-OEwQt5do.js.map} +1 -1
- package/dist/demo/DemoHeading-Db-XkQIK.js +69 -0
- package/dist/demo/DemoHeading-Db-XkQIK.js.map +1 -0
- package/dist/demo/{DemoHome-D_De3UiT.js → DemoHome-Cyp29ygy.js} +4 -5
- package/dist/demo/{DemoHome-D_De3UiT.js.map → DemoHome-Cyp29ygy.js.map} +1 -1
- package/dist/demo/{DemoJsonViewer-B50s9aGM.js → DemoJsonViewer-DXtCeMzH.js} +4 -5
- package/dist/demo/{DemoJsonViewer-B50s9aGM.js.map → DemoJsonViewer-DXtCeMzH.js.map} +1 -1
- package/dist/demo/{DemoLayout-CHU8WTwO.js → DemoLayout-hh9VmZQP.js} +4 -5
- package/dist/demo/DemoLayout-hh9VmZQP.js.map +1 -0
- package/dist/demo/{DemoLogin-BBlrWpml.js → DemoLogin-DX7mnmkh.js} +15 -11
- package/dist/demo/{DemoLogin-BBlrWpml.js.map → DemoLogin-DX7mnmkh.js.map} +1 -1
- package/dist/demo/{DemoRegister-BuNE3_-f.js → DemoRegister-DVcZl04m.js} +15 -11
- package/dist/demo/{DemoRegister-BuNE3_-f.js.map → DemoRegister-DVcZl04m.js.map} +1 -1
- package/dist/demo/{DemoResetPassword-D_IjjjOJ.js → DemoResetPassword-CPENlZH5.js} +15 -11
- package/dist/demo/{DemoResetPassword-D_IjjjOJ.js.map → DemoResetPassword-CPENlZH5.js.map} +1 -1
- package/dist/demo/{DemoSidebar-Giy2HRBD.js → DemoSidebar-CGu7DZeM.js} +4 -5
- package/dist/demo/{DemoSidebar-Giy2HRBD.js.map → DemoSidebar-CGu7DZeM.js.map} +1 -1
- package/dist/demo/{DemoText-ubcw-vog.js → DemoText-DYUJ7bY_.js} +4 -5
- package/dist/demo/{DemoText-ubcw-vog.js.map → DemoText-DYUJ7bY_.js.map} +1 -1
- package/dist/demo/{DemoToast-9die_dYT.js → DemoToast-CgdnZNvx.js} +3 -4
- package/dist/demo/{DemoToast-9die_dYT.js.map → DemoToast-CgdnZNvx.js.map} +1 -1
- package/dist/demo/{DemoTypeForm-D_d6OVKL.js → DemoTypeForm-Pims-cGa.js} +4 -5
- package/dist/demo/{DemoTypeForm-D_d6OVKL.js.map → DemoTypeForm-Pims-cGa.js.map} +1 -1
- package/dist/demo/{DemoVerifyEmail-B43KlF4F.js → DemoVerifyEmail-C7B3xxch.js} +10 -11
- package/dist/demo/{DemoVerifyEmail-B43KlF4F.js.map → DemoVerifyEmail-C7B3xxch.js.map} +1 -1
- package/dist/{auth/IconGoogle-Ch1m3Uzl.js → demo/IconGoogle-CwQy4G9y.js} +2 -4
- package/dist/demo/{IconGoogle-CSQLPYwX.js.map → IconGoogle-CwQy4G9y.js.map} +1 -1
- package/dist/demo/{Login-C12N4oGs.js → Login-pwMF4TUj.js} +5 -6
- package/dist/{admin/Login-xtNmQtGh.js.map → demo/Login-pwMF4TUj.js.map} +1 -1
- package/dist/{auth/Profile-YcWdeuFz.js → demo/Profile-BliZapZS.js} +3 -5
- package/dist/{admin/Profile-_AtPUwAP.js.map → demo/Profile-BliZapZS.js.map} +1 -1
- package/dist/demo/{Register-B4hLBeEv.js → Register-CiwAT7Hy.js} +5 -6
- package/dist/{auth/Register-CPhEO5MG.js.map → demo/Register-CiwAT7Hy.js.map} +1 -1
- package/dist/{auth/ResetPassword-DCtGcneA.js → demo/ResetPassword-l9Vg4JE-.js} +4 -5
- package/dist/demo/{ResetPassword-D8g9ha1N.js.map → ResetPassword-l9Vg4JE-.js.map} +1 -1
- package/dist/demo/{Showcase-D6Fxt4X4.js → Showcase-CX6bDgwe.js} +3 -5
- package/dist/demo/{Showcase-D6Fxt4X4.js.map → Showcase-CX6bDgwe.js.map} +1 -1
- package/dist/demo/{VerifyEmail-BjDo0cZA.js → VerifyEmail-CAB-OS7i.js} +4 -5
- package/dist/demo/{VerifyEmail-BjDo0cZA.js.map → VerifyEmail-CAB-OS7i.js.map} +1 -1
- package/dist/demo/{auth-ByVTreDl.js → auth-uegJAdKu.js} +18 -35
- package/dist/demo/{auth-ByVTreDl.js.map → auth-uegJAdKu.js.map} +1 -1
- package/dist/demo/{core-DFgB3yU4.js → core-B4LVHzPn.js} +132 -93
- package/dist/demo/core-B4LVHzPn.js.map +1 -0
- package/dist/demo/index.js +20 -23
- package/dist/demo/index.js.map +1 -1
- package/dist/demo/rolldown-runtime-CiIaOW0V.js +13 -0
- package/package.json +17 -20
- package/src/admin/AdminRouter.tsx +23 -38
- package/src/admin/atoms/adminUserAtom.ts +7 -0
- package/src/admin/components/AdminLayout.tsx +2 -14
- package/src/admin/components/files/AdminFiles.tsx +123 -1
- package/src/admin/components/jobs/{AdminJobExecutions.tsx → AdminJobs.tsx} +450 -317
- package/src/admin/components/notifications/AdminNotifications.tsx +11 -25
- package/src/admin/components/users/AdminUserLayout.tsx +84 -127
- package/src/admin/components/users/AdminUserProfile.tsx +5 -2
- package/src/admin/components/users/AdminUsers.tsx +1 -1
- package/src/core/components/Flex.tsx +24 -0
- package/src/core/components/Section.tsx +109 -0
- package/src/core/components/SectionHeader.tsx +106 -0
- package/src/core/components/buttons/ActionButton.tsx +1 -0
- package/src/core/components/dialogs/PromptDialog.tsx +1 -1
- package/src/core/components/layout/Breadcrumb.tsx +2 -2
- package/src/core/components/layout/DashboardShell.tsx +1 -1
- package/src/core/index.ts +4 -1
- package/src/core/services/DialogService.tsx +2 -2
- package/src/core/styles.css +2 -1
- package/src/core/table/components/DataTable.tsx +5 -2
- package/src/demo/DemoRouter.ts +1 -1
- package/src/demo/components/auth/DemoLogin.tsx +5 -0
- package/src/demo/components/auth/DemoRegister.tsx +5 -0
- package/src/demo/components/auth/DemoResetPassword.tsx +5 -0
- package/src/demo/components/core/DemoHeading.tsx +56 -3
- package/dist/admin/AdminFiles-DFTjijGp.js +0 -111
- package/dist/admin/AdminFiles-DFTjijGp.js.map +0 -1
- package/dist/admin/AdminJobDashboard-BL8gGPDp.js +0 -354
- package/dist/admin/AdminJobDashboard-BL8gGPDp.js.map +0 -1
- package/dist/admin/AdminJobExecutions-D9E-CS-U.js.map +0 -1
- package/dist/admin/AdminJobRegistry-Ci9ue1zC.js +0 -270
- package/dist/admin/AdminJobRegistry-Ci9ue1zC.js.map +0 -1
- package/dist/admin/AdminLayout-I6TlUMPc.js.map +0 -1
- package/dist/admin/AdminNotifications-ZPHCYrv7.js.map +0 -1
- package/dist/admin/AdminUserLayout-lXT6I0Qq.js.map +0 -1
- package/dist/admin/AdminUserProfile-vFBLoJ3h.js.map +0 -1
- package/dist/admin/AdminUsers-D1UfGya9.js.map +0 -1
- package/dist/admin/core-CYaRQ8O-.js.map +0 -1
- package/dist/admin/rolldown-runtime-CjeV3_4I.js +0 -18
- package/dist/auth/core-D5jIAVF2.js.map +0 -1
- package/dist/auth/rolldown-runtime-CjeV3_4I.js +0 -18
- package/dist/demo/DemoHeading-C13OVDfS.js +0 -18
- package/dist/demo/DemoHeading-C13OVDfS.js.map +0 -1
- package/dist/demo/DemoLayout-CHU8WTwO.js.map +0 -1
- package/dist/demo/core-DFgB3yU4.js.map +0 -1
- package/dist/demo/rolldown-runtime-CjeV3_4I.js +0 -18
- package/src/admin/components/jobs/AdminJobDashboard.tsx +0 -349
- package/src/admin/components/jobs/AdminJobRegistry.tsx +0 -301
- package/src/core/components/Heading.tsx +0 -19
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { _ as
|
|
1
|
+
import { _ as ClipboardButton, c as Text$1, d as StatCards, h as useDialog, r as DataTable, u as Flex$1, v as ActionButton, x as useToast } from "./core-CJCEx18C.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
|
-
|
|
284
|
+
|
|
285
|
+
//# sourceMappingURL=AdminApiKeys-C2ze85eD.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":"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"}
|
|
1
|
+
{"version":3,"file":"AdminApiKeys-C2ze85eD.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 {
|
|
1
|
+
import { c as Text$1, r as DataTable, u as Flex$1 } from "./core-CJCEx18C.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
|
-
|
|
182
|
+
|
|
183
|
+
//# sourceMappingURL=AdminAudits-BIj81e4k.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"AdminAudits-
|
|
1
|
+
{"version":3,"file":"AdminAudits-BIj81e4k.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 {
|
|
1
|
+
import { c as Text, u as Flex } from "./core-CJCEx18C.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
|
-
|
|
65
|
+
|
|
66
|
+
//# sourceMappingURL=AdminDashboard-PMVzrwSu.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"AdminDashboard-
|
|
1
|
+
{"version":3,"file":"AdminDashboard-PMVzrwSu.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"}
|
|
@@ -0,0 +1,189 @@
|
|
|
1
|
+
import { c as Text$1, h as useDialog, r as DataTable, u as Flex$1, x as useToast } from "./core-CJCEx18C.js";
|
|
2
|
+
import { t } from "alepha";
|
|
3
|
+
import { useI18n } from "alepha/react/i18n";
|
|
4
|
+
import { HoverCard } from "@mantine/core";
|
|
5
|
+
import { jsx, jsxs } from "react/jsx-runtime";
|
|
6
|
+
import { IconDownload, IconFile, IconFileCode, IconFileMusic, IconFileSpreadsheet, IconFileText, IconFileZip, IconPdf, IconPhoto, IconTrash, IconVideo } from "@tabler/icons-react";
|
|
7
|
+
import { useClient } from "alepha/react";
|
|
8
|
+
import { files } from "alepha/api/files";
|
|
9
|
+
//#region ../../src/admin/components/files/AdminFiles.tsx
|
|
10
|
+
const AdminFiles = () => {
|
|
11
|
+
const client = useClient();
|
|
12
|
+
const { l } = useI18n();
|
|
13
|
+
const dialog = useDialog();
|
|
14
|
+
const toast = useToast();
|
|
15
|
+
const filters = t.object({
|
|
16
|
+
bucket: t.optional(t.string()),
|
|
17
|
+
name: t.optional(t.string({ $control: { query: t.pick(files.schema, [
|
|
18
|
+
"name",
|
|
19
|
+
"bucket",
|
|
20
|
+
"mimeType"
|
|
21
|
+
]) } }))
|
|
22
|
+
});
|
|
23
|
+
const isImage = (mime) => mime.startsWith("image/");
|
|
24
|
+
const getFileIcon = (mime) => {
|
|
25
|
+
if (isImage(mime)) return IconPhoto;
|
|
26
|
+
if (mime === "application/pdf") return IconPdf;
|
|
27
|
+
if (mime.startsWith("video/")) return IconVideo;
|
|
28
|
+
if (mime.startsWith("audio/")) return IconFileMusic;
|
|
29
|
+
if (mime.startsWith("text/csv") || mime.includes("spreadsheet")) return IconFileSpreadsheet;
|
|
30
|
+
if (mime.includes("zip") || mime.includes("tar") || mime.includes("gzip") || mime.includes("compress")) return IconFileZip;
|
|
31
|
+
if (mime.includes("javascript") || mime.includes("json") || mime.includes("xml") || mime.includes("html") || mime.includes("css")) return IconFileCode;
|
|
32
|
+
if (mime.startsWith("text/")) return IconFileText;
|
|
33
|
+
return IconFile;
|
|
34
|
+
};
|
|
35
|
+
const formatFileSize = (bytes) => {
|
|
36
|
+
if (bytes === 0) return "0 B";
|
|
37
|
+
const k = 1024;
|
|
38
|
+
const sizes = [
|
|
39
|
+
"B",
|
|
40
|
+
"KB",
|
|
41
|
+
"MB",
|
|
42
|
+
"GB"
|
|
43
|
+
];
|
|
44
|
+
const i = Math.floor(Math.log(bytes) / Math.log(k));
|
|
45
|
+
return `${Number.parseFloat((bytes / k ** i).toFixed(1))} ${sizes[i]}`;
|
|
46
|
+
};
|
|
47
|
+
return /* @__PURE__ */ jsx(Flex$1, {
|
|
48
|
+
p: "md",
|
|
49
|
+
flex: 1,
|
|
50
|
+
direction: "column",
|
|
51
|
+
children: /* @__PURE__ */ jsx(DataTable, {
|
|
52
|
+
submitOnInit: true,
|
|
53
|
+
defaultSize: 10,
|
|
54
|
+
typeFormProps: {
|
|
55
|
+
skipSubmitButton: true,
|
|
56
|
+
columns: 3
|
|
57
|
+
},
|
|
58
|
+
tableProps: {
|
|
59
|
+
horizontalSpacing: "xs",
|
|
60
|
+
verticalSpacing: "xs"
|
|
61
|
+
},
|
|
62
|
+
onFilterChange: (key, _value, form) => {
|
|
63
|
+
if (key === "name" || key === "bucket") return form.submit();
|
|
64
|
+
},
|
|
65
|
+
filters,
|
|
66
|
+
items: async (filters) => {
|
|
67
|
+
return await client.findFiles({ query: filters });
|
|
68
|
+
},
|
|
69
|
+
columns: {
|
|
70
|
+
preview: {
|
|
71
|
+
label: "",
|
|
72
|
+
fit: true,
|
|
73
|
+
value: (item) => {
|
|
74
|
+
const Icon = getFileIcon(item.mimeType);
|
|
75
|
+
const url = `/api/files/${item.id}`;
|
|
76
|
+
if (isImage(item.mimeType)) return /* @__PURE__ */ jsxs(HoverCard, {
|
|
77
|
+
width: 512,
|
|
78
|
+
position: "right",
|
|
79
|
+
shadow: "lg",
|
|
80
|
+
openDelay: 200,
|
|
81
|
+
children: [/* @__PURE__ */ jsx(HoverCard.Target, { children: /* @__PURE__ */ jsx("img", {
|
|
82
|
+
src: url,
|
|
83
|
+
alt: item.name,
|
|
84
|
+
style: {
|
|
85
|
+
width: 32,
|
|
86
|
+
height: 32,
|
|
87
|
+
objectFit: "cover",
|
|
88
|
+
borderRadius: 4,
|
|
89
|
+
display: "block"
|
|
90
|
+
}
|
|
91
|
+
}) }), /* @__PURE__ */ jsx(HoverCard.Dropdown, {
|
|
92
|
+
p: 4,
|
|
93
|
+
children: /* @__PURE__ */ jsx("img", {
|
|
94
|
+
src: url,
|
|
95
|
+
alt: item.name,
|
|
96
|
+
style: {
|
|
97
|
+
width: 512,
|
|
98
|
+
height: 512,
|
|
99
|
+
objectFit: "contain",
|
|
100
|
+
display: "block"
|
|
101
|
+
}
|
|
102
|
+
})
|
|
103
|
+
})]
|
|
104
|
+
});
|
|
105
|
+
return /* @__PURE__ */ jsx(Icon, {
|
|
106
|
+
size: 20,
|
|
107
|
+
color: "var(--mantine-color-dimmed)"
|
|
108
|
+
});
|
|
109
|
+
}
|
|
110
|
+
},
|
|
111
|
+
name: {
|
|
112
|
+
label: "Name",
|
|
113
|
+
value: (item) => /* @__PURE__ */ jsx(Text$1, {
|
|
114
|
+
size: "sm",
|
|
115
|
+
fw: 500,
|
|
116
|
+
lineClamp: 1,
|
|
117
|
+
children: item.name
|
|
118
|
+
})
|
|
119
|
+
},
|
|
120
|
+
bucket: {
|
|
121
|
+
label: "Bucket",
|
|
122
|
+
value: (item) => /* @__PURE__ */ jsx(Text$1, {
|
|
123
|
+
size: "xs",
|
|
124
|
+
ff: "monospace",
|
|
125
|
+
children: item.bucket
|
|
126
|
+
})
|
|
127
|
+
},
|
|
128
|
+
mimeType: {
|
|
129
|
+
label: "Type",
|
|
130
|
+
value: (item) => /* @__PURE__ */ jsx(Text$1, {
|
|
131
|
+
size: "xs",
|
|
132
|
+
c: "dimmed",
|
|
133
|
+
children: item.mimeType
|
|
134
|
+
})
|
|
135
|
+
},
|
|
136
|
+
size: {
|
|
137
|
+
label: "Size",
|
|
138
|
+
value: (item) => /* @__PURE__ */ jsx(Text$1, {
|
|
139
|
+
size: "xs",
|
|
140
|
+
c: "dimmed",
|
|
141
|
+
children: formatFileSize(item.size)
|
|
142
|
+
})
|
|
143
|
+
},
|
|
144
|
+
creatorName: {
|
|
145
|
+
label: "Creator",
|
|
146
|
+
value: (item) => /* @__PURE__ */ jsx(Text$1, {
|
|
147
|
+
size: "xs",
|
|
148
|
+
c: "dimmed",
|
|
149
|
+
children: item.creatorName || "-"
|
|
150
|
+
})
|
|
151
|
+
},
|
|
152
|
+
createdAt: {
|
|
153
|
+
label: "Created",
|
|
154
|
+
value: (item) => /* @__PURE__ */ jsx(Text$1, {
|
|
155
|
+
size: "xs",
|
|
156
|
+
c: "dimmed",
|
|
157
|
+
children: l(item.createdAt, { date: "fromNow" })
|
|
158
|
+
})
|
|
159
|
+
}
|
|
160
|
+
},
|
|
161
|
+
rowActions: (item) => [{
|
|
162
|
+
label: "Download",
|
|
163
|
+
icon: IconDownload,
|
|
164
|
+
skipRefresh: true,
|
|
165
|
+
onClick: () => {
|
|
166
|
+
window.open(`/api/files/${item.id}`, "_blank");
|
|
167
|
+
}
|
|
168
|
+
}],
|
|
169
|
+
checkboxActions: [{
|
|
170
|
+
label: "Delete selected",
|
|
171
|
+
icon: /* @__PURE__ */ jsx(IconTrash, { size: 14 }),
|
|
172
|
+
intent: "danger",
|
|
173
|
+
onClick: async ({ selectedItems, clearSelection }) => {
|
|
174
|
+
if (!await dialog.confirm({
|
|
175
|
+
title: "Delete files",
|
|
176
|
+
message: `Permanently delete ${selectedItems.length} file(s)? This action cannot be undone.`
|
|
177
|
+
})) return;
|
|
178
|
+
for (const file of selectedItems) await client.deleteFile({ params: { id: file.id } });
|
|
179
|
+
toast.success({ message: `${selectedItems.length} file(s) deleted` });
|
|
180
|
+
clearSelection();
|
|
181
|
+
}
|
|
182
|
+
}]
|
|
183
|
+
})
|
|
184
|
+
});
|
|
185
|
+
};
|
|
186
|
+
//#endregion
|
|
187
|
+
export { AdminFiles as default };
|
|
188
|
+
|
|
189
|
+
//# sourceMappingURL=AdminFiles-Bq03BLt-.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"AdminFiles-Bq03BLt-.js","names":["Flex","Text"],"sources":["../../src/admin/components/files/AdminFiles.tsx"],"sourcesContent":["import { DataTable, Flex, Text, useDialog, useToast } from \"@alepha/ui\";\nimport { HoverCard } from \"@mantine/core\";\nimport {\n IconDownload,\n IconFile,\n IconFileCode,\n IconFileMusic,\n IconFileSpreadsheet,\n IconFileText,\n IconFileZip,\n IconPdf,\n IconPhoto,\n IconTrash,\n IconVideo,\n} from \"@tabler/icons-react\";\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 const dialog = useDialog();\n const toast = useToast();\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 isImage = (mime: string) => mime.startsWith(\"image/\");\n\n const getFileIcon = (mime: string) => {\n if (isImage(mime)) return IconPhoto;\n if (mime === \"application/pdf\") return IconPdf;\n if (mime.startsWith(\"video/\")) return IconVideo;\n if (mime.startsWith(\"audio/\")) return IconFileMusic;\n if (mime.startsWith(\"text/csv\") || mime.includes(\"spreadsheet\"))\n return IconFileSpreadsheet;\n if (\n mime.includes(\"zip\") ||\n mime.includes(\"tar\") ||\n mime.includes(\"gzip\") ||\n mime.includes(\"compress\")\n )\n return IconFileZip;\n if (\n mime.includes(\"javascript\") ||\n mime.includes(\"json\") ||\n mime.includes(\"xml\") ||\n mime.includes(\"html\") ||\n mime.includes(\"css\")\n )\n return IconFileCode;\n if (mime.startsWith(\"text/\")) return IconFileText;\n return IconFile;\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 preview: {\n label: \"\",\n fit: true,\n value: (item) => {\n const Icon = getFileIcon(item.mimeType);\n const url = `/api/files/${item.id}`;\n\n if (isImage(item.mimeType)) {\n return (\n <HoverCard\n width={512}\n position=\"right\"\n shadow=\"lg\"\n openDelay={200}\n >\n <HoverCard.Target>\n <img\n src={url}\n alt={item.name}\n style={{\n width: 32,\n height: 32,\n objectFit: \"cover\",\n borderRadius: 4,\n display: \"block\",\n }}\n />\n </HoverCard.Target>\n <HoverCard.Dropdown p={4}>\n <img\n src={url}\n alt={item.name}\n style={{\n width: 512,\n height: 512,\n objectFit: \"contain\",\n display: \"block\",\n }}\n />\n </HoverCard.Dropdown>\n </HoverCard>\n );\n }\n\n return <Icon size={20} color=\"var(--mantine-color-dimmed)\" />;\n },\n },\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 rowActions={(item) => [\n {\n label: \"Download\",\n icon: IconDownload,\n skipRefresh: true,\n onClick: () => {\n window.open(`/api/files/${item.id}`, \"_blank\");\n },\n },\n ]}\n checkboxActions={[\n {\n label: \"Delete selected\",\n icon: <IconTrash size={14} />,\n intent: \"danger\",\n onClick: async ({ selectedItems, clearSelection }) => {\n const confirmed = await dialog.confirm({\n title: \"Delete files\",\n message: `Permanently delete ${selectedItems.length} file(s)? This action cannot be undone.`,\n });\n if (!confirmed) return;\n for (const file of selectedItems) {\n await client.deleteFile({ params: { id: file.id } });\n }\n toast.success({\n message: `${selectedItems.length} file(s) deleted`,\n });\n clearSelection();\n },\n },\n ]}\n />\n </Flex>\n );\n};\n\nexport default AdminFiles;\n"],"mappings":"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"}
|