From 3cd200c79c2f852037063c43188f878717f7a143 Mon Sep 17 00:00:00 2001 From: Alejandro Visiedo Date: Thu, 30 May 2024 13:20:45 +0200 Subject: [PATCH] feat(HMS-4129): support rbac on UI controls This change add more granularity to the RBAC integration in the UI by enabling/disabling in a selective way the related controls depending on the current permissions when the user logged in. https://issues.redhat.com/browse/HMS-4129 Signed-off-by: Alejandro Visiedo --- src/Components/DomainList/DomainList.tsx | 14 +++++++++----- src/Routes/DefaultPage/DefaultPage.tsx | 5 ++++- .../Components/DetailGeneral/DetailGeneral.tsx | 5 +++++ src/Routes/DetailPage/DetailPage.tsx | 1 + 4 files changed, 19 insertions(+), 6 deletions(-) diff --git a/src/Components/DomainList/DomainList.tsx b/src/Components/DomainList/DomainList.tsx index eb7142a..77a500f 100644 --- a/src/Components/DomainList/DomainList.tsx +++ b/src/Components/DomainList/DomainList.tsx @@ -16,6 +16,7 @@ import { buildDeleteFailedNotification, buildDeleteSuccessNotification, } from '../../Routes/DetailPage/detailNotifications'; +import useIdmPermissions from '../../Hooks/useIdmPermissions'; export interface IColumnType { key: string; @@ -100,7 +101,8 @@ export const DomainList = () => { const base_url = '/api/idmsvc/v1'; const resources_api = ResourcesApiFactory(undefined, base_url, undefined); - const context = useContext(AppContext); + const appContext = useContext(AppContext); + const rbac = useIdmPermissions(); const navigate = useNavigate(); const { notifyError, notifySuccess } = useNotification(); @@ -112,7 +114,7 @@ export const DomainList = () => { // Sort direction of the currently sorted column const [activeSortDirection, setActiveSortDirection] = React.useState<'asc' | 'desc'>('asc'); - const domains = context.listDomains; + const domains = appContext.listDomains; const [isOpenAutoJoinChangeConfirm, setIsOpenAutoJoinChangeConfirm] = useState(false); const [isOpenConfirmDelete, setIsOpenConfirmDelete] = useState(false); @@ -136,7 +138,7 @@ export const DomainList = () => { // remove domain(s) matching the given uuid from the `domains` state const removeDomain = (uuid: string): void => { - context.deleteDomain(uuid); + appContext.deleteDomain(uuid); }; const showAutoJoinChangeConfirmDialog = (domain: Domain) => { @@ -161,7 +163,7 @@ export const DomainList = () => { }) .then((response) => { if (response.status == 200) { - context.updateDomain(response.data); + appContext.updateDomain(response.data); notifySuccess(buildAutoJoinToggleSuccessNotification(domain)); } else { notifyError(buildAutoJoinToggleFailedNotification(domain)); @@ -209,6 +211,7 @@ export const DomainList = () => { title: 'Enable/Disable', onClick: () => showAutoJoinChangeConfirmDialog(domain), ouiaId: 'ButtonActionEnableDisable', + isDisabled: !rbac.permissions.hasDomainsUpdate, }, { title: 'Edit', @@ -219,6 +222,7 @@ export const DomainList = () => { title: 'Delete', onClick: () => OnShowConfirmDelete(domain), ouiaId: 'ButtonActionDelete', + isDisabled: !rbac.permissions.hasDomainsDelete, }, ]; @@ -228,7 +232,7 @@ export const DomainList = () => { const onShowDetails = (domain: Domain | undefined) => { if (domain !== undefined) { - context.setEditing(domain); + appContext.setEditing(domain); navigate('/details/' + domain.domain_id); } }; diff --git a/src/Routes/DefaultPage/DefaultPage.tsx b/src/Routes/DefaultPage/DefaultPage.tsx index 4d081c9..c5d24a6 100644 --- a/src/Routes/DefaultPage/DefaultPage.tsx +++ b/src/Routes/DefaultPage/DefaultPage.tsx @@ -60,6 +60,7 @@ const Header = () => { const RegisterDomainButton = () => { const appContext = useContext(AppContext); + const rbac = useIdmPermissions(); const navigate = useNavigate(); const handleOpenWizard = () => { @@ -69,8 +70,10 @@ const RegisterDomainButton = () => { navigate('/domains/wizard', { replace: true }); }; + const isDisabled = rbac.isLoading || !rbac.permissions.hasTokenCreate || !rbac.permissions.hasDomainsUpdate; + return ( - ); diff --git a/src/Routes/DetailPage/Components/DetailGeneral/DetailGeneral.tsx b/src/Routes/DetailPage/Components/DetailGeneral/DetailGeneral.tsx index 5ad6600..310641c 100644 --- a/src/Routes/DetailPage/Components/DetailGeneral/DetailGeneral.tsx +++ b/src/Routes/DetailPage/Components/DetailGeneral/DetailGeneral.tsx @@ -26,6 +26,7 @@ import { buildTitleEditFailedNotification, buildTitleEditSuccessNotification, } from '../../detailNotifications'; +import useIdmPermissions from '../../../../Hooks/useIdmPermissions'; interface DetailGeneralProps { domain?: Domain; @@ -41,6 +42,7 @@ export const DetailGeneral = (props: DetailGeneralProps) => { const base_url = '/api/idmsvc/v1'; const resources_api = ResourcesApiFactory(undefined, base_url, undefined); + const rbac = useIdmPermissions(); const { notifyError, notifySuccess } = useNotification(); // States @@ -175,6 +177,7 @@ export const DetailGeneral = (props: DetailGeneralProps) => { return; }} ouiaId="ButtonDetailGeneralEditTitle" + isDisabled={!rbac.permissions.hasDomainsUpdate} > @@ -196,6 +199,7 @@ export const DetailGeneral = (props: DetailGeneralProps) => { setIsDescriptionModalOpen(true); }} ouiaId="ButtonDetailGeneralEditDescription" + isDisabled={!rbac.permissions.hasDomainsUpdate} > @@ -247,6 +251,7 @@ export const DetailGeneral = (props: DetailGeneralProps) => { isChecked={autoJoin} onChange={handleAutoJoin} ouiaId="ButtonDetailGeneralAutoenroll" + isDisabled={!rbac.permissions.hasDomainsUpdate} /> diff --git a/src/Routes/DetailPage/DetailPage.tsx b/src/Routes/DetailPage/DetailPage.tsx index 3fe8a51..aa08a8c 100644 --- a/src/Routes/DetailPage/DetailPage.tsx +++ b/src/Routes/DetailPage/DetailPage.tsx @@ -168,6 +168,7 @@ const DetailPage = () => { domain !== undefined && OnShowConfirmDelete(); }} ouiaId="ButtonDetailsDelete" + isDisabled={!rbac.permissions.hasDomainsDelete} > Delete