{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 (
+
+
+
+
+
+
+
+
+ {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({
|
-
+
|
|