import { useState } from "react"; import { useTranslation } from "react-i18next"; import { AlertVariant, Badge, Button, ButtonVariant, Checkbox, ToolbarItem, } from "@patternfly/react-core"; import { cellWidth } from "@patternfly/react-table"; import type ClientRepresentation from "@keycloak/keycloak-admin-client/lib/defs/clientRepresentation"; import type RoleRepresentation from "@keycloak/keycloak-admin-client/lib/defs/roleRepresentation"; import type KeycloakAdminClient from "@keycloak/keycloak-admin-client"; import { AddRoleMappingModal } from "./AddRoleMappingModal"; import { KeycloakDataTable } from "../table-toolbar/KeycloakDataTable"; import { emptyFormatter, upperCaseFormatter } from "../../util"; import { useAlerts } from "../alert/Alerts"; import { useConfirmDialog } from "../confirm-dialog/ConfirmDialog"; import { useAdminClient } from "../../context/auth/AdminClient"; import { ListEmptyState } from "../list-empty-state/ListEmptyState"; import { deleteMapping, getEffectiveRoles, getMapping } from "./queries"; import { getEffectiveClientRoles } from "./resource"; import "./role-mapping.css"; export type CompositeRole = RoleRepresentation & { parent: RoleRepresentation; isInherited?: boolean; }; export type Row = { client?: ClientRepresentation; role: RoleRepresentation | CompositeRole; }; export const mapRoles = ( assignedRoles: Row[], effectiveRoles: Row[], hide: boolean ) => [ ...(hide ? assignedRoles.map((row) => ({ ...row, role: { ...row.role, isInherited: false, }, })) : effectiveRoles.map((row) => ({ ...row, role: { ...row.role, isInherited: assignedRoles.find((r) => r.role.id === row.role.id) === undefined, }, }))), ]; export const ServiceRole = ({ role, client }: Row) => ( <> {client?.clientId && ( {client.clientId} )} {role.name} ); export type ResourcesKey = keyof KeycloakAdminClient; type RoleMappingProps = { name: string; id: string; type: ResourcesKey; isManager?: boolean; save: (rows: Row[]) => Promise; }; export const RoleMapping = ({ name, id, type, isManager = true, save, }: RoleMappingProps) => { const { t } = useTranslation(type); const { adminClient } = useAdminClient(); const { addAlert, addError } = useAlerts(); const [key, setKey] = useState(0); const refresh = () => setKey(key + 1); const [hide, setHide] = useState(true); const [showAssign, setShowAssign] = useState(false); const [selected, setSelected] = useState([]); const assignRoles = async (rows: Row[]) => { await save(rows); refresh(); }; const loader = async () => { let effectiveRoles: Row[] = []; let effectiveClientRoles: Row[] = []; if (!hide) { effectiveRoles = await getEffectiveRoles(adminClient, type, id); effectiveClientRoles = ( await getEffectiveClientRoles({ adminClient, type, id, }) ).map((e) => ({ client: { clientId: e.client, id: e.clientId }, role: { id: e.id, name: e.role, description: e.description }, })); } const roles = await getMapping(adminClient, type, id); const realmRolesMapping = roles.realmMappings?.map((role) => ({ role })) || []; const clientMapping = Object.values(roles.clientMappings || {}) .map((client) => client.mappings.map((role: RoleRepresentation) => ({ client: { clientId: client.client, ...client }, role, })) ) .flat(); return [ ...mapRoles( [...realmRolesMapping, ...clientMapping], [...effectiveClientRoles, ...effectiveRoles], hide ), ]; }; const [toggleDeleteDialog, DeleteConfirm] = useConfirmDialog({ titleKey: "clients:removeMappingTitle", messageKey: t("clients:removeMappingConfirm", { count: selected.length }), continueButtonLabel: "common:remove", continueButtonVariant: ButtonVariant.danger, onConfirm: async () => { try { await Promise.all(deleteMapping(adminClient, type, id, selected)); addAlert(t("clients:clientScopeRemoveSuccess"), AlertVariant.success); refresh(); } catch (error) { addError("clients:clientScopeRemoveError", error); } }, }); const ManagerToolbarItems = () => { if (!isManager) return ; return ( <> ); }; return ( <> {showAssign && ( setShowAssign(false)} /> )} setSelected(rows)} searchPlaceholderKey="clients:searchByName" ariaLabelKey="clients:clientScopeList" isRowDisabled={(value) => (value.role as CompositeRole).isInherited || false } toolbarItem={ <> { setHide(check); refresh(); }} /> } actions={ isManager ? [ { title: t("common:unAssignRole"), onRowClick: async (role) => { setSelected([role]); toggleDeleteDialog(); return false; }, }, ] : [] } columns={[ { name: "role.name", displayKey: t("common:name"), transforms: [cellWidth(30)], cellRenderer: ServiceRole, }, { name: "role.isInherited", displayKey: t("common:inherent"), cellFormatters: [upperCaseFormatter(), emptyFormatter()], }, { name: "role.description", displayKey: t("common:description"), cellFormatters: [emptyFormatter()], }, ]} emptyState={ setShowAssign(true)} /> } /> ); };