diff --git a/js/apps/admin-ui/cypress/support/pages/admin-ui/manage/clients/client_details/ClientDetailsPage.ts b/js/apps/admin-ui/cypress/support/pages/admin-ui/manage/clients/client_details/ClientDetailsPage.ts index 6d6b95cd013d..20b38dd73c10 100644 --- a/js/apps/admin-ui/cypress/support/pages/admin-ui/manage/clients/client_details/ClientDetailsPage.ts +++ b/js/apps/admin-ui/cypress/support/pages/admin-ui/manage/clients/client_details/ClientDetailsPage.ts @@ -19,6 +19,7 @@ export enum ClientsDetailsTab { ServiceAccountsRoles = "Service accounts roles", Advanced = "Advanced", Scope = "Scope", + UserEvents = "Events", } export default class ClientDetailsPage extends CommonPage { diff --git a/js/apps/admin-ui/src/clients/ClientDetails.tsx b/js/apps/admin-ui/src/clients/ClientDetails.tsx index 64a47361085b..d53039dfeff9 100644 --- a/js/apps/admin-ui/src/clients/ClientDetails.tsx +++ b/js/apps/admin-ui/src/clients/ClientDetails.tsx @@ -72,6 +72,7 @@ import { ClientScopes } from "./scopes/ClientScopes"; import { EvaluateScopes } from "./scopes/EvaluateScopes"; import { ServiceAccount } from "./service-account/ServiceAccount"; import { getProtocolName, isRealmClient } from "./utils"; +import { UserEvents } from "../events/UserEvents"; type ClientDetailHeaderProps = { onChange: (value: boolean) => void; @@ -190,7 +191,7 @@ export default function ClientDetails() { const { t } = useTranslation(); const { addAlert, addError } = useAlerts(); - const { realm } = useRealm(); + const { realm, realmRepresentation } = useRealm(); const { hasAccess } = useAccess(); const isFeatureEnabled = useIsFeatureEnabled(); @@ -225,54 +226,65 @@ export default function ClientDetails() { return sortBy(roles, (role) => role.name?.toUpperCase()); }; - const useTab = (tab: ClientTab) => - useRoutableTab( - toClient({ - realm, - clientId, - tab, - }), - ); - - const settingsTab = useTab("settings"); - const keysTab = useTab("keys"); - const credentialsTab = useTab("credentials"); - const rolesTab = useTab("roles"); - const clientScopesTab = useTab("clientScopes"); - const authorizationTab = useTab("authorization"); - const serviceAccountTab = useTab("serviceAccount"); - const sessionsTab = useTab("sessions"); - const permissionsTab = useTab("permissions"); - const advancedTab = useTab("advanced"); - - const useClientScopesTab = (tab: ClientScopesTab) => - useRoutableTab( - toClientScopesTab({ - realm, - clientId, - tab, - }), - ); + const tab = (tab: ClientTab) => + toClient({ + realm, + clientId, + tab, + }); + + const settingsTab = useRoutableTab(tab("settings")); + const keysTab = useRoutableTab(tab("keys")); + const credentialsTab = useRoutableTab(tab("credentials")); + const rolesTab = useRoutableTab(tab("roles")); + const clientScopesTab = useRoutableTab(tab("clientScopes")); + const authorizationTab = useRoutableTab(tab("authorization")); + const serviceAccountTab = useRoutableTab(tab("serviceAccount")); + const sessionsTab = useRoutableTab(tab("sessions")); + const permissionsTab = useRoutableTab(tab("permissions")); + const advancedTab = useRoutableTab(tab("advanced")); + const userEventsTab = useRoutableTab(tab("user-events")); + + const clientScopesTabRoute = (tab: ClientScopesTab) => + toClientScopesTab({ + realm, + clientId, + tab, + }); + + const clientScopesSetupTab = useRoutableTab(clientScopesTabRoute("setup")); + const clientScopesEvaluateTab = useRoutableTab( + clientScopesTabRoute("evaluate"), + ); - const clientScopesSetupTab = useClientScopesTab("setup"); - const clientScopesEvaluateTab = useClientScopesTab("evaluate"); + const authorizationTabRoute = (tab: AuthorizationTab) => + toAuthorizationTab({ + realm, + clientId, + tab, + }); - const useAuthorizationTab = (tab: AuthorizationTab) => - useRoutableTab( - toAuthorizationTab({ - realm, - clientId, - tab, - }), - ); - - const authorizationSettingsTab = useAuthorizationTab("settings"); - const authorizationResourcesTab = useAuthorizationTab("resources"); - const authorizationScopesTab = useAuthorizationTab("scopes"); - const authorizationPoliciesTab = useAuthorizationTab("policies"); - const authorizationPermissionsTab = useAuthorizationTab("permissions"); - const authorizationEvaluateTab = useAuthorizationTab("evaluate"); - const authorizationExportTab = useAuthorizationTab("export"); + const authorizationSettingsTab = useRoutableTab( + authorizationTabRoute("settings"), + ); + const authorizationResourcesTab = useRoutableTab( + authorizationTabRoute("resources"), + ); + const authorizationScopesTab = useRoutableTab( + authorizationTabRoute("scopes"), + ); + const authorizationPoliciesTab = useRoutableTab( + authorizationTabRoute("policies"), + ); + const authorizationPermissionsTab = useRoutableTab( + authorizationTabRoute("permissions"), + ); + const authorizationEvaluateTab = useRoutableTab( + authorizationTabRoute("evaluate"), + ); + const authorizationExportTab = useRoutableTab( + authorizationTabRoute("export"), + ); const [toggleDeleteDialog, DeleteConfirm] = useConfirmDialog({ titleKey: "clientDeleteConfirmTitle", @@ -663,6 +675,15 @@ export default function ClientDetails() { > + {hasAccess("view-events") && realmRepresentation?.eventsEnabled && ( + {t("events")}} + {...userEventsTab} + > + + + )} diff --git a/js/apps/admin-ui/src/clients/routes/Client.tsx b/js/apps/admin-ui/src/clients/routes/Client.tsx index 2701af4f0c00..871f95e934a8 100644 --- a/js/apps/admin-ui/src/clients/routes/Client.tsx +++ b/js/apps/admin-ui/src/clients/routes/Client.tsx @@ -14,7 +14,8 @@ export type ClientTab = | "authorization" | "serviceAccount" | "permissions" - | "sessions"; + | "sessions" + | "user-events"; export type ClientParams = { realm: string; diff --git a/js/apps/admin-ui/src/events/EventsSection.tsx b/js/apps/admin-ui/src/events/EventsSection.tsx index 1de72d6f0852..8ec321ec5f9f 100644 --- a/js/apps/admin-ui/src/events/EventsSection.tsx +++ b/js/apps/admin-ui/src/events/EventsSection.tsx @@ -1,44 +1,6 @@ -import type EventRepresentation from "@keycloak/keycloak-admin-client/lib/defs/eventRepresentation"; -import type EventType from "@keycloak/keycloak-admin-client/lib/defs/eventTypes"; -import type { RealmEventsConfigRepresentation } from "@keycloak/keycloak-admin-client/lib/defs/realmEventsConfigRepresentation"; -import { - KeycloakDataTable, - KeycloakSelect, - ListEmptyState, - SelectVariant, - TextControl, - useFetch, -} from "@keycloak/keycloak-ui-shared"; -import { - ActionGroup, - Button, - Chip, - ChipGroup, - DatePicker, - DescriptionList, - DescriptionListDescription, - DescriptionListGroup, - DescriptionListTerm, - Flex, - FlexItem, - Form, - FormGroup, - Icon, - PageSection, - SelectOption, - Tab, - TabTitleText, - Tooltip, -} from "@patternfly/react-core"; -import { CheckCircleIcon, WarningTriangleIcon } from "@patternfly/react-icons"; -import { cellWidth } from "@patternfly/react-table"; -import { pickBy } from "lodash-es"; -import { useState } from "react"; -import { Controller, FormProvider, useForm } from "react-hook-form"; +import { PageSection, Tab, TabTitleText } from "@patternfly/react-core"; import { Trans, useTranslation } from "react-i18next"; import { Link } from "react-router-dom"; -import { useAdminClient } from "../admin-client"; -import DropdownPanel from "../components/dropdown-panel/DropdownPanel"; import { RoutableTabs, useRoutableTab, @@ -47,383 +9,21 @@ import { ViewHeader } from "../components/view-header/ViewHeader"; import { useRealm } from "../context/realm-context/RealmContext"; import helpUrls from "../help-urls"; import { toRealmSettings } from "../realm-settings/routes/RealmSettings"; -import { toUser } from "../user/routes/User"; -import useFormatDate, { FORMAT_DATE_AND_TIME } from "../utils/useFormatDate"; import { AdminEvents } from "./AdminEvents"; -import { EventsTab, toEvents } from "./routes/Events"; +import { UserEvents } from "./UserEvents"; +import { toEvents } from "./routes/Events"; import "./events.css"; -type UserEventSearchForm = { - client: string; - dateFrom: string; - dateTo: string; - user: string; - type: EventType[]; - ipAddress: string; -}; - -const defaultValues: UserEventSearchForm = { - client: "", - dateFrom: "", - dateTo: "", - user: "", - type: [], - ipAddress: "", -}; - -const StatusRow = (event: EventRepresentation) => - !event.error ? ( - - - - - {event.type} - - ) : ( - - - - - - {event.type} - - - ); - -const DetailCell = (event: EventRepresentation) => ( - - {event.details && - Object.entries(event.details).map(([key, value]) => ( - - {key} - {value} - - ))} - {event.error && ( - - error - {event.error} - - )} - -); - -const UserDetailLink = (event: EventRepresentation) => { - const { t } = useTranslation(); - const { realm } = useRealm(); - - return ( - <> - {event.userId && ( - - {event.userId} - - )} - {!event.userId && t("noUserDetails")} - - ); -}; - export default function EventsSection() { - const { adminClient } = useAdminClient(); - const { t } = useTranslation(); const { realm } = useRealm(); - const formatDate = useFormatDate(); - const [key, setKey] = useState(0); - const [searchDropdownOpen, setSearchDropdownOpen] = useState(false); - const [selectOpen, setSelectOpen] = useState(false); - const [events, setEvents] = useState(); - const [activeFilters, setActiveFilters] = useState< - Partial - >({}); - - const filterLabels: Record = { - client: t("client"), - dateFrom: t("dateFrom"), - dateTo: t("dateTo"), - user: t("userId"), - type: t("eventType"), - ipAddress: t("ipAddress"), - }; - const form = useForm({ - mode: "onChange", - defaultValues, - }); - - const { - getValues, - reset, - formState: { isDirty }, - control, - handleSubmit, - } = form; - - useFetch( - () => adminClient.realms.getConfigEvents({ realm }), - (events) => setEvents(events), - [], + const userEventsTab = useRoutableTab(toEvents({ realm, tab: "user-events" })); + const adminEventsTab = useRoutableTab( + toEvents({ realm, tab: "admin-events" }), ); - function loader(first?: number, max?: number) { - return adminClient.realms.findEvents({ - // The admin client wants 'dateFrom' and 'dateTo' to be Date objects, however it cannot actually handle them so we need to cast to any. - ...(activeFilters as any), - realm, - first, - max, - }); - } - - const useTab = (tab: EventsTab) => useRoutableTab(toEvents({ realm, tab })); - - const userEventsTab = useTab("user-events"); - const adminEventsTab = useTab("admin-events"); - - function onSubmit() { - setSearchDropdownOpen(false); - commitFilters(); - } - - function resetSearch() { - reset(); - commitFilters(); - } - - function removeFilter(key: keyof UserEventSearchForm) { - const formValues: UserEventSearchForm = { ...getValues() }; - delete formValues[key]; - - reset({ ...defaultValues, ...formValues }); - commitFilters(); - } - - function removeFilterValue( - key: keyof UserEventSearchForm, - valueToRemove: EventType, - ) { - const formValues = getValues(); - const fieldValue = formValues[key]; - const newFieldValue = Array.isArray(fieldValue) - ? fieldValue.filter((val) => val !== valueToRemove) - : fieldValue; - - reset({ ...formValues, [key]: newFieldValue }); - commitFilters(); - } - - function commitFilters() { - const newFilters: Partial = pickBy( - getValues(), - (value) => value !== "" || (Array.isArray(value) && value.length > 0), - ); - - setActiveFilters(newFilters); - setKey(key + 1); - } - - const userEventSearchFormDisplay = () => { - return ( - - - - -
- - - ( - setSelectOpen(isOpen)} - selections={field.value} - onSelect={(selectedValue) => { - const option = selectedValue.toString() as EventType; - const changedValue = field.value.includes(option) - ? field.value.filter((item) => item !== option) - : [...field.value, option]; - - field.onChange(changedValue); - }} - onClear={() => { - field.onChange([]); - }} - isOpen={selectOpen} - aria-labelledby={"eventType"} - chipGroupComponent={ - - {field.value.map((chip) => ( - { - event.stopPropagation(); - field.onChange( - field.value.filter((val) => val !== chip), - ); - }} - > - {t(`eventTypes.${chip}.name`)} - - ))} - - } - > - {events?.enabledEventTypes?.map((option) => ( - - {t(`eventTypes.${option}.name`)} - - ))} - - )} - /> - - - - ( - field.onChange(value)} - inputProps={{ id: "kc-dateFrom" }} - /> - )} - /> - - - ( - field.onChange(value)} - inputProps={{ id: "kc-dateTo" }} - /> - )} - /> - - - - - - - -
-
- - {Object.entries(activeFilters).length > 0 && ( -
- {Object.entries(activeFilters).map((filter) => { - const [key, value] = filter as [ - keyof UserEventSearchForm, - string | EventType[], - ]; - - return ( - removeFilter(key)} - > - {typeof value === "string" ? ( - {value} - ) : ( - value.map((entry) => ( - removeFilterValue(key, entry)} - > - {t(`eventTypes.${entry}.name`)} - - )) - )} - - ); - })} -
- )} -
-
-
- ); - }; - return ( <> {t("userEvents")}} {...userEventsTab} > -
- event.details !== undefined, - cellRenderer: DetailCell, - }, - ]} - isPaginated - ariaLabelKey="titleEvents" - toolbarItem={userEventSearchFormDisplay()} - columns={[ - { - name: "time", - displayKey: "time", - cellRenderer: (row) => - formatDate(new Date(row.time!), FORMAT_DATE_AND_TIME), - }, - { - name: "userId", - displayKey: "user", - cellRenderer: UserDetailLink, - }, - { - name: "type", - displayKey: "eventType", - cellRenderer: StatusRow, - }, - { - name: "ipAddress", - displayKey: "ipAddress", - transforms: [cellWidth(10)], - }, - { - name: "clientId", - displayKey: "client", - }, - ]} - emptyState={ - - } - isSearching={Object.keys(activeFilters).length > 0} - /> -
+ {t("adminEvents")}} diff --git a/js/apps/admin-ui/src/events/UserEvents.tsx b/js/apps/admin-ui/src/events/UserEvents.tsx new file mode 100644 index 000000000000..257f75fac77c --- /dev/null +++ b/js/apps/admin-ui/src/events/UserEvents.tsx @@ -0,0 +1,488 @@ +import type EventRepresentation from "@keycloak/keycloak-admin-client/lib/defs/eventRepresentation"; +import type EventType from "@keycloak/keycloak-admin-client/lib/defs/eventTypes"; +import type { RealmEventsConfigRepresentation } from "@keycloak/keycloak-admin-client/lib/defs/realmEventsConfigRepresentation"; +import { + KeycloakDataTable, + KeycloakSelect, + ListEmptyState, + SelectVariant, + TextControl, + useFetch, +} from "@keycloak/keycloak-ui-shared"; +import { + ActionGroup, + Button, + Chip, + ChipGroup, + DatePicker, + DescriptionList, + DescriptionListDescription, + DescriptionListGroup, + DescriptionListTerm, + Flex, + FlexItem, + Form, + FormGroup, + Icon, + SelectOption, + Tooltip, +} from "@patternfly/react-core"; +import { CheckCircleIcon, WarningTriangleIcon } from "@patternfly/react-icons"; +import { cellWidth } from "@patternfly/react-table"; +import { pickBy } from "lodash-es"; +import { useState } from "react"; +import { Controller, FormProvider, useForm } from "react-hook-form"; +import { useTranslation } from "react-i18next"; +import { Link } from "react-router-dom"; +import { useAdminClient } from "../admin-client"; +import DropdownPanel from "../components/dropdown-panel/DropdownPanel"; +import { useRealm } from "../context/realm-context/RealmContext"; +import { toUser } from "../user/routes/User"; +import useFormatDate, { FORMAT_DATE_AND_TIME } from "../utils/useFormatDate"; + +import "./events.css"; + +type UserEventSearchForm = { + client: string; + dateFrom: string; + dateTo: string; + user: string; + type: EventType[]; + ipAddress: string; +}; + +const StatusRow = (event: EventRepresentation) => + !event.error ? ( + + + + + {event.type} + + ) : ( + + + + + + {event.type} + + + ); + +const DetailCell = (event: EventRepresentation) => ( + + {event.details && + Object.entries(event.details).map(([key, value]) => ( + + {key} + {value} + + ))} + {event.error && ( + + error + {event.error} + + )} + +); + +const UserDetailLink = (event: EventRepresentation) => { + const { t } = useTranslation(); + const { realm } = useRealm(); + + return ( + <> + {event.userId && ( + + {event.userId} + + )} + {!event.userId && t("noUserDetails")} + + ); +}; + +type UserEventsProps = { + user?: string; + client?: string; +}; + +export const UserEvents = ({ user, client }: UserEventsProps) => { + const { adminClient } = useAdminClient(); + + const { t } = useTranslation(); + const { realm } = useRealm(); + const formatDate = useFormatDate(); + const [key, setKey] = useState(0); + const [searchDropdownOpen, setSearchDropdownOpen] = useState(false); + const [selectOpen, setSelectOpen] = useState(false); + const [events, setEvents] = useState(); + const [activeFilters, setActiveFilters] = useState< + Partial + >({}); + + const defaultValues: UserEventSearchForm = { + client: client ? client : "", + dateFrom: "", + dateTo: "", + user: user ? user : "", + type: [], + ipAddress: "", + }; + + const filterLabels: Record = { + client: t("client"), + dateFrom: t("dateFrom"), + dateTo: t("dateTo"), + user: t("userId"), + type: t("eventType"), + ipAddress: t("ipAddress"), + }; + + const form = useForm({ + mode: "onChange", + defaultValues, + }); + + const { + getValues, + reset, + formState: { isDirty }, + control, + handleSubmit, + } = form; + + useFetch( + () => adminClient.realms.getConfigEvents({ realm }), + (events) => setEvents(events), + [], + ); + + function loader(first?: number, max?: number) { + return adminClient.realms.findEvents({ + // The admin client wants 'dateFrom' and 'dateTo' to be Date objects, however it cannot actually handle them so we need to cast to any. + ...(activeFilters as any), + client, + user, + realm, + first, + max, + }); + } + + function onSubmit() { + setSearchDropdownOpen(false); + commitFilters(); + } + + function resetSearch() { + reset(); + commitFilters(); + } + + function removeFilter(key: keyof UserEventSearchForm) { + const formValues: UserEventSearchForm = { ...getValues() }; + delete formValues[key]; + + reset({ ...defaultValues, ...formValues }); + commitFilters(); + } + + function removeFilterValue( + key: keyof UserEventSearchForm, + valueToRemove: EventType, + ) { + const formValues = getValues(); + const fieldValue = formValues[key]; + const newFieldValue = Array.isArray(fieldValue) + ? fieldValue.filter((val) => val !== valueToRemove) + : fieldValue; + + reset({ ...formValues, [key]: newFieldValue }); + commitFilters(); + } + + function commitFilters() { + const newFilters: Partial = pickBy( + getValues(), + (value) => value !== "" || (Array.isArray(value) && value.length > 0), + ); + + if (user) { + delete newFilters.user; + } + + if (client) { + delete newFilters.client; + } + + setActiveFilters(newFilters); + setKey(key + 1); + } + + const userEventSearchFormDisplay = () => { + return ( + + + + +
+ {!user && ( + + )} + + ( + setSelectOpen(isOpen)} + selections={field.value} + onSelect={(selectedValue) => { + const option = selectedValue.toString() as EventType; + const changedValue = field.value.includes(option) + ? field.value.filter((item) => item !== option) + : [...field.value, option]; + + field.onChange(changedValue); + }} + onClear={() => { + field.onChange([]); + }} + isOpen={selectOpen} + aria-labelledby={"eventType"} + chipGroupComponent={ + + {field.value.map((chip) => ( + { + event.stopPropagation(); + field.onChange( + field.value.filter((val) => val !== chip), + ); + }} + > + {t(`eventTypes.${chip}.name`)} + + ))} + + } + > + {events?.enabledEventTypes?.map((option) => ( + + {t(`eventTypes.${option}.name`)} + + ))} + + )} + /> + + {!client && ( + + )} + + ( + field.onChange(value)} + inputProps={{ id: "kc-dateFrom" }} + /> + )} + /> + + + ( + field.onChange(value)} + inputProps={{ id: "kc-dateTo" }} + /> + )} + /> + + + + + + + +
+
+ + {Object.entries(activeFilters).length > 0 && ( +
+ {Object.entries(activeFilters).map((filter) => { + const [key, value] = filter as [ + keyof UserEventSearchForm, + string | EventType[], + ]; + + return ( + removeFilter(key)} + isClosable + > + {typeof value === "string" ? ( + {value} + ) : ( + value.map((entry) => ( + removeFilterValue(key, entry)} + > + {t(`eventTypes.${entry}.name`)} + + )) + )} + + ); + })} +
+ )} +
+
+
+ ); + }; + + return ( +
+ event.details !== undefined, + cellRenderer: DetailCell, + }, + ]} + isPaginated + ariaLabelKey="titleEvents" + toolbarItem={userEventSearchFormDisplay()} + columns={[ + { + name: "time", + displayKey: "time", + cellRenderer: (row) => + formatDate(new Date(row.time!), FORMAT_DATE_AND_TIME), + }, + ...(!user + ? [ + { + name: "userId", + cellRenderer: UserDetailLink, + }, + ] + : []), + { + name: "type", + displayKey: "eventType", + cellRenderer: StatusRow, + }, + { + name: "ipAddress", + displayKey: "ipAddress", + transforms: [cellWidth(10)], + }, + ...(!client + ? [ + { + name: "clientId", + displayKey: "client", + }, + ] + : []), + ]} + emptyState={ + + } + isSearching={Object.keys(activeFilters).length > 0} + /> +
+ ); +}; diff --git a/js/apps/admin-ui/src/realm-settings/event-config/EventsTab.tsx b/js/apps/admin-ui/src/realm-settings/event-config/EventsTab.tsx index 8e3bdd223881..0d3e4edd2c02 100644 --- a/js/apps/admin-ui/src/realm-settings/event-config/EventsTab.tsx +++ b/js/apps/admin-ui/src/realm-settings/event-config/EventsTab.tsx @@ -111,7 +111,6 @@ export const EventsTab = ({ realm }: EventsTabProps) => { }, ); } - refreshRealm(); try { await adminClient.realms.updateConfigEvents( @@ -133,6 +132,8 @@ export const EventsTab = ({ realm }: EventsTabProps) => { error, ); } + + refreshRealm(); }; const addEventTypes = async (eventTypes: EventType[]) => { diff --git a/js/apps/admin-ui/src/user/EditUser.tsx b/js/apps/admin-ui/src/user/EditUser.tsx index 0af116efad2d..857ad75188e4 100644 --- a/js/apps/admin-ui/src/user/EditUser.tsx +++ b/js/apps/admin-ui/src/user/EditUser.tsx @@ -47,6 +47,7 @@ import { UserGroups } from "./UserGroups"; import { UserIdentityProviderLinks } from "./UserIdentityProviderLinks"; import { UserRoleMapping } from "./UserRoleMapping"; import { UserSessions } from "./UserSessions"; +import { UserEvents } from "../events/UserEvents"; import { UIUserRepresentation, UserFormFields, @@ -101,17 +102,18 @@ export default function EditUser() { tab, }); - const useTab = (tab: UserTab) => useRoutableTab(toTab(tab)); - - const settingsTab = useTab("settings"); - const attributesTab = useTab("attributes"); - const credentialsTab = useTab("credentials"); - const roleMappingTab = useTab("role-mapping"); - const groupsTab = useTab("groups"); - const organizationsTab = useTab("organizations"); - const consentsTab = useTab("consents"); - const identityProviderLinksTab = useTab("identity-provider-links"); - const sessionsTab = useTab("sessions"); + const settingsTab = useRoutableTab(toTab("settings")); + const attributesTab = useRoutableTab(toTab("attributes")); + const credentialsTab = useRoutableTab(toTab("credentials")); + const roleMappingTab = useRoutableTab(toTab("role-mapping")); + const groupsTab = useRoutableTab(toTab("groups")); + const organizationsTab = useRoutableTab(toTab("organizations")); + const consentsTab = useRoutableTab(toTab("consents")); + const identityProviderLinksTab = useRoutableTab( + toTab("identity-provider-links"), + ); + const sessionsTab = useRoutableTab(toTab("sessions")); + const userEventsTab = useRoutableTab(toTab("user-events")); useFetch( async () => @@ -424,6 +426,15 @@ export default function EditUser() { >
+ {hasAccess("view-events") && realm?.eventsEnabled && ( + {t("events")}} + {...userEventsTab} + > + + + )} diff --git a/js/apps/admin-ui/src/user/routes/User.tsx b/js/apps/admin-ui/src/user/routes/User.tsx index f72ae3c23f69..5bc9c7d9be5e 100644 --- a/js/apps/admin-ui/src/user/routes/User.tsx +++ b/js/apps/admin-ui/src/user/routes/User.tsx @@ -12,7 +12,8 @@ export type UserTab = | "sessions" | "credentials" | "role-mapping" - | "identity-provider-links"; + | "identity-provider-links" + | "user-events"; export type UserParams = { realm: string; diff --git a/js/libs/ui-shared/src/controls/table/KeycloakDataTable.tsx b/js/libs/ui-shared/src/controls/table/KeycloakDataTable.tsx index 91909de5cd45..24b11c6ae194 100644 --- a/js/libs/ui-shared/src/controls/table/KeycloakDataTable.tsx +++ b/js/libs/ui-shared/src/controls/table/KeycloakDataTable.tsx @@ -76,13 +76,18 @@ type CellRendererProps = { row: IRow; }; -const CellRenderer = ({ row }: CellRendererProps) => { - const isRow = (c: ReactNode | IRowCell): c is IRowCell => - !!c && (c as IRowCell).title !== undefined; - return row.cells!.map((c, i) => ( +const isRow = (c: ReactNode | IRowCell): c is IRowCell => + !!c && (c as IRowCell).title !== undefined; + +const CellRenderer = ({ row }: CellRendererProps) => + row.cells!.map((c, i) => ( {(isRow(c) ? c.title : c) as ReactNode} )); -}; + +const ExpandableRowRenderer = ({ row }: CellRendererProps) => + row.cells!.map((c, i) => ( +
{(isRow(c) ? c.title : c) as ReactNode}
+ )); function DataTable({ columns, @@ -130,9 +135,10 @@ function DataTable({ > - {onCollapse && } + {onCollapse && } {canSelectAll && ( ({ {columns.map((column) => ( {t(column.displayKey || column.name)} @@ -212,7 +219,7 @@ function DataTable({ - +