Skip to content

Commit

Permalink
Merge remote-tracking branch 'rhecosystem/master' into pull-in-rhecos…
Browse files Browse the repository at this point in the history
…ystem-changes
  • Loading branch information
florkbr committed Jul 17, 2024
2 parents 81d1f66 + 663a88b commit 8298e2f
Show file tree
Hide file tree
Showing 17 changed files with 1,174 additions and 55 deletions.
107 changes: 107 additions & 0 deletions src/Messages.js
Original file line number Diff line number Diff line change
@@ -1,6 +1,93 @@
import { defineMessages } from 'react-intl';

export default defineMessages({
inviteUsers: {
id: 'inviteUsers',
description: 'Invite users',
defaultMessage: 'Invite users',
},
inviteUsersTitle: {
id: 'inviteUsersTitle',
description: 'Invite users modal title',
defaultMessage: 'Invite New Users',
},
inviteUsersDescription: {
id: 'inviteUsersDescription',
description: 'Invite users modal description',
defaultMessage:
'Invite users to create a Red Hat login with your organization. Your e-mail address will be included in the invite as a point of contact.',
},
inviteUsersFormIsAdminFieldTitle: {
id: 'inviteUsersFormIsAdminFieldTitle',
description: 'Invite users form is admin field title',
defaultMessage: 'Organization Administrators',
},
inviteUsersFormIsAdminFieldDescription: {
id: 'inviteUsersFormIsAdminFieldDescription',
description: 'Invite users form is admin field description',
defaultMessage:
'The organization administrator role is the highest permission level with full access to content and features. This is the only role that can manage users.',
},
inviteUsersFormEmailsFieldTitle: {
id: 'inviteUsersFormEmailsFieldTitle',
description: 'Invite users form emails field title',
defaultMessage: 'Enter the e-mail addresses of the users you would like to invite',
},
inviteUsersFormEmailsFieldDescription: {
id: 'inviteUsersFormEmailsFieldDescription',
description: 'Invite users form emails field description',
defaultMessage: 'Enter up to 50 email addresses separated by commas or returns.',
},
inviteUsersCancelled: {
id: 'inviteUsersCancelled',
description: 'Invite users cancelled notification description',
defaultMessage: 'Invite users process was canceled by the user.',
},
inviteUsersButton: {
id: 'inviteUsersButton',
description: 'Invite users button text',
defaultMessage: 'Invite new users',
},
inviteUsersErrorTitle: {
id: 'inviteUsersErrorTitle',
description: 'Invite users error notification title',
defaultMessage: 'Failed inviting all users',
},
inviteUsersErrorDescription: {
id: 'inviteUsersErrorDescription',
description: 'Invite users error notification description',
defaultMessage: 'Failed inviting users.',
},
activateUsersButton: {
id: 'activateUsersButton',
description: 'activate users button text',
defaultMessage: 'Activate users',
},
deactivateUsersButton: {
id: 'deactivateUsersButton',
description: 'deactivate users button text',
defaultMessage: 'Deactivate users',
},
deactivateUsersConfirmationModalTitle: {
id: 'deactivateUsersConfirmationModalTitle',
description: 'deactivate users confirmation modal title text',
defaultMessage: 'Deactivate users',
},
deactivateUsersConfirmationModalDescription: {
id: 'deactivateUsersConfirmationModalDescription',
description: 'deactivate users confirmation modal description text',
defaultMessage: 'Are you sure you want to deactivate the user(s) below from your Red Hat organization?',
},
deactivateUsersConfirmationModalCheckboxText: {
id: 'deactivateUsersConfirmationModalCheckboxText',
description: 'deactivate users confirmation modal checkbox text',
defaultMessage: 'Yes, I confirm that I want to remove these users',
},
deactivateUsersConfirmationButton: {
id: 'deactivateUsersConfirmationButton',
description: 'deactivate users confirmation button text',
defaultMessage: 'Deactivate user(s)',
},
notApplicable: {
id: 'notApplicable',
description: 'Not applicable text for resource definitions',
Expand Down Expand Up @@ -222,6 +309,26 @@ export default defineMessages({
description: 'Edit group error notification description',
defaultMessage: 'The group was not updated successfuly.',
},
editUserSuccessTitle: {
id: 'editUserSuccessTitle',
description: 'Edit user success notification title',
defaultMessage: 'Success updating user',
},
editUserSuccessDescription: {
id: 'editUserSuccessDescription',
description: 'Edit user success notification description',
defaultMessage: 'The user was updated successfully.',
},
editUserErrorTitle: {
id: 'editUserErrorTitle',
description: 'Edit user error notification title',
defaultMessage: 'Failed updating user',
},
editUserErrorDescription: {
id: 'editUserErrorDescription',
description: 'Edit user error notification description',
defaultMessage: 'The user was not updated successfuly.',
},
removeGroupSuccess: {
id: 'removeGroupSuccess',
description: 'Remove group success notification title',
Expand Down
16 changes: 16 additions & 0 deletions src/helpers/shared/helpers.js
Original file line number Diff line number Diff line change
Expand Up @@ -79,3 +79,19 @@ export const getDateFormat = (date) => {
const monthAgo = new Date(Date.now());
return Date.parse(date) < monthAgo.setMonth(monthAgo.getMonth() - 1) ? 'onlyDate' : 'relative';
};

export const isExternalIdp = (token = '') => {
let roles = [''];
let tokenArray = token.split('.');
if (tokenArray.length > 1) {
let token1 = window.atob(tokenArray[1]);
if (token1) {
roles = JSON.parse(token1)?.realm_access?.roles;
if (roles.includes('external-idp')) {
return true;
}
}
}

return false;
};
140 changes: 140 additions & 0 deletions src/helpers/user/user-helper.js
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
import { getLastPageOffset, isOffsetValid } from '../shared/pagination';
import { getPrincipalApi } from '../shared/user-login';
import { isInt, isStage, isITLessProd } from '../../itLessConfig';

const principalApi = getPrincipalApi();

Expand All @@ -8,6 +9,145 @@ const principalStatusApiMap = {
Inactive: 'disabled',
All: 'all',
};

const getBaseUrl = (url) => {
if (isInt) {
return url.int;
} else if (isStage) {
return url.stage;
} else if (isITLessProd) {
return url.prod;
} else {
return '';
}
};

async function fetchBaseUrl() {
try {
const response = await fetch(`${insights.chrome.isBeta() ? '/beta' : ''}/apps/rbac/env.json`);
const jsonData = await response.json();
return jsonData;
} catch (error) {
console.log(error);
}
}

export async function addUsers(usersData = { emails: [], isAdmin: undefined }) {
const token = await insights.chrome.auth.getToken();
const requestOpts = {
method: 'POST',
headers: {
'Content-Type': 'application/json',
Accept: 'application/json',
Authorization: `Bearer ${token}`,
},
body: JSON.stringify({
emails: usersData.emails,
isAdmin: usersData.isAdmin,
}),
};
const url = await fetchBaseUrl();
const baseUrl = getBaseUrl(url);
let promise = new Promise((resolve, reject) => {
return fetch(`${baseUrl}/user/invite`, requestOpts)
.then(
(response) => {
if (response.ok && response.status !== 206) {
resolve(response);
} else if (response.ok && response.status === 206) {
response.json().then((body) => {
reject(body);
});
} else {
reject(response);
}
},
(error) => {
reject(new Error(error.message));
}
)
.catch((err) => {
reject(new Error(err.message));
});
});

return promise;
}

export async function updateUserIsOrgAdminStatus(user) {
const token = await insights.chrome.auth.getToken();
let requestOpts = {
method: 'PUT',
headers: {
'Content-Type': 'application/json',
Accept: 'application/json',
Authorization: `Bearer ${token}`,
},
};

const url = await fetchBaseUrl();
const baseUrl = getBaseUrl(url);

let promise = new Promise((resolve, reject) => {
return fetch(`${baseUrl}/user/${user.id}/admin/${user.is_org_admin}`, requestOpts)
.then(
(response) => {
if (response.ok) {
resolve(response);
} else {
reject(response);
}
},
(error) => {
reject(new Error(error.message));
}
)
.catch((err) => {
reject(new Error(err.message));
});
});

return promise;
}

export async function updateUsers(users) {
// await principalApi.updateUser(user.uuid, user);
const token = await insights.chrome.auth.getToken();
let requestOpts = {
method: 'PUT',
headers: {
'Content-Type': 'application/json',
Accept: 'application/json',
Authorization: `Bearer ${token}`,
},
body: JSON.stringify({ users: users }),
};

const url = await fetchBaseUrl();
const baseUrl = getBaseUrl(url);

let promise = new Promise((resolve, reject) => {
return fetch(`${baseUrl}/change-users-status`, requestOpts)
.then(
(response) => {
if (response.ok) {
resolve(response);
} else {
reject(response);
}
},
(error) => {
reject(new Error(error.message));
}
)
.catch((err) => {
reject(new Error(err.message));
});
});

return promise;
}

export async function fetchUsers({ limit, offset = 0, orderBy, filters = {}, usesMetaInURL, matchCriteria = 'partial' }) {
const { username, email, status = [] } = filters;
const sortOrder = orderBy === '-username' ? 'desc' : 'asc';
Expand Down
4 changes: 4 additions & 0 deletions src/itLessConfig.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
export const isEphem = insights.chrome.getEnvironment() === 'ephem';
export const isInt = insights.chrome.getEnvironment() === 'int';
export const isStage = insights.chrome.getEnvironment() === 'frhStage';
export const isITLessProd = insights.chrome.getEnvironment() === 'frh';
2 changes: 1 addition & 1 deletion src/presentational-components/shared/ActiveUsers.js
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,7 @@ const ActiveUser = ({ linkDescription, linkTitle }) => {
const env = chrome.getEnvironment();
const prefix = chrome.isProd() ? '' : `${env}.`;
const { orgAdmin } = useContext(PermissionsContext);
return orgAdmin ? (
return !chrome.isFedramp && orgAdmin ? (
<Text className="pf-v5-u-mt-0" component={TextVariants.h7}>
{`${intl.formatMessage(messages.usersDescription)} `}
{linkDescription}
Expand Down
8 changes: 7 additions & 1 deletion src/presentational-components/shared/table-toolbar-view.js
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ import propTypes from 'prop-types';
import messages from '../../Messages';
import { TableVariant } from '@patternfly/react-table';
import { Table, TableHeader, TableBody } from '@patternfly/react-table/deprecated';
import { useChrome } from '@redhat-cloud-services/frontend-components/useChrome';
import TableToolbar from '@redhat-cloud-services/frontend-components/TableToolbar';
import SkeletonTable from '@patternfly/react-component-groups/dist/dynamic/SkeletonTable';
import { Button, Pagination, EmptyStateActions } from '@patternfly/react-core';
Expand Down Expand Up @@ -59,8 +60,10 @@ export const TableToolbarView = ({
tableId,
containerRef,
textFilterRef,
toolbarChildren,
}) => {
const intl = useIntl();
const chrome = useChrome();
const renderEmpty = () => ({
title: (
<EmptyWithAction
Expand Down Expand Up @@ -130,6 +133,7 @@ export const TableToolbarView = ({
tableId={tableId}
containerRef={containerRef}
textFilterRef={textFilterRef}
toolbarChildren={toolbarChildren}
/>
{isLoading ? (
<SkeletonTable
Expand All @@ -146,7 +150,7 @@ export const TableToolbarView = ({
{...(isSelectable &&
rows?.length > 0 && {
onSelect: (_e, isSelected, _idx, { uuid, cells: [name], requires }) =>
setCheckedItems(selectedRows([{ uuid, name, requires }], isSelected)),
setCheckedItems(selectedRows([{ uuid, name, requires, ...(chrome.isFedramp && { username: data[_idx]?.username }) }], isSelected)),
})}
{...(isExpandable && { onExpand })}
rows={rows?.length > 0 ? rows : [{ fullWidth: true, cells: [renderEmpty()] }]}
Expand Down Expand Up @@ -234,6 +238,7 @@ TableToolbarView.propTypes = {
noDataDescription: propTypes.arrayOf(propTypes.node),
filters: propTypes.array,
tableId: propTypes.string.isRequired,
toolbarChildren: propTypes.func,
};

TableToolbarView.defaultProps = {
Expand All @@ -245,4 +250,5 @@ TableToolbarView.defaultProps = {
hideFilterChips: false,
checkedRows: [],
hideHeader: false,
toolbarChildren: () => null,
};
Loading

0 comments on commit 8298e2f

Please sign in to comment.