/*
* Copyright OpenSearch Contributors
*
* Licensed under the Apache License, Version 2.0 (the "License").
* You may not use this file except in compliance with the License.
* A copy of the License is located at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* or in the "license" file accompanying this file. This file is distributed
* on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either
* express or implied. See the License for the specific language governing
* permissions and limitations under the License.
*/
import React from 'react';
import { mount, shallow } from 'enzyme';
import { RoleView } from '../role-view';
import { ClusterPermissionPanel } from '../../role-view/cluster-permission-panel';
import { IndexPermissionPanel } from '../index-permission-panel';
import { TenantsPanel } from '../tenants-panel';
import { EuiTabbedContent } from '@elastic/eui';
import {
getRoleMappingData,
transformRoleMappingData,
updateRoleMapping,
} from '../../../utils/role-mapping-utils';
import { fetchActionGroups } from '../../../utils/action-groups-utils';
import { getRoleDetail } from '../../../utils/role-detail-utils';
import { transformRoleIndexPermissions } from '../../../utils/index-permission-utils';
import { useDeleteConfirmState } from '../../../utils/delete-confirm-modal-utils';
import { requestDeleteRoles } from '../../../utils/role-list-utils';
import { Action, ResourceType, SubAction } from '../../../types';
import { buildHashUrl } from '../../../utils/url-builder';
import { createUnknownErrorToast } from '../../../utils/toast-utils';
jest.mock('../../../utils/role-mapping-utils', () => ({
getRoleMappingData: jest.fn().mockReturnValue({ backend_roles: [], hosts: [], users: [] }),
transformRoleMappingData: jest.fn().mockReturnValue({
userName: '',
userType: '',
}),
updateRoleMapping: jest.fn(),
}));
jest.mock('../../../utils/action-groups-utils', () => ({
fetchActionGroups: jest.fn().mockReturnValue({}),
}));
jest.mock('../../../utils/role-detail-utils', () => ({
getRoleDetail: jest.fn().mockReturnValue({
cluster_permissions: [],
index_permissions: [],
tenant_permissions: [],
reserved: false,
}),
}));
jest.mock('../../../utils/delete-confirm-modal-utils', () => ({
useDeleteConfirmState: jest.fn().mockReturnValue([jest.fn(), '']),
}));
jest.mock('../../../utils/index-permission-utils');
jest.mock('../../../utils/tenant-utils');
jest.mock('../../../utils/role-list-utils', () => ({
requestDeleteRoles: jest.fn(),
}));
jest.mock('../../../utils/context-menu', () => ({
useContextMenuState: jest
.fn()
.mockImplementation((buttonText, buttonProps, children) => [children, jest.fn()]),
}));
jest.mock('../../../utils/toast-utils', () => ({
createErrorToast: jest.fn(),
createUnknownErrorToast: jest.fn(),
useToastState: jest.fn().mockReturnValue([[], jest.fn(), jest.fn()]),
}));
describe('Role view', () => {
const setState = jest.fn();
const sampleRole = 'role';
const mockCoreStart = {
http: 1,
};
const buildBreadcrumbs = jest.fn();
const useEffect = jest.spyOn(React, 'useEffect');
const useState = jest.spyOn(React, 'useState');
beforeEach(() => {
useEffect.mockImplementationOnce((f) => f());
useState.mockImplementation((initialValue) => [initialValue, setState]);
});
it('basic rendering when permission tab is selected', () => {
const component = shallow(
);
expect(buildBreadcrumbs).toBeCalledTimes(1);
expect(component.find(EuiTabbedContent).length).toBe(1);
const tabs = component.find(EuiTabbedContent).dive();
expect(tabs.find(ClusterPermissionPanel).length).toBe(1);
expect(tabs.find(IndexPermissionPanel).length).toBe(1);
expect(tabs.find(TenantsPanel).length).toBe(1);
expect(component).toMatchSnapshot();
});
it('renders when mapped user tab is selected', () => {
const component = shallow(
);
expect(component).toMatchSnapshot();
});
it('should render to map user page when click on Map users', () => {
const wrapper = shallow(
);
const tabs = wrapper.find(EuiTabbedContent).dive();
const roleMappingList = tabs.find('[data-test-subj="role-mapping-list"]').dive();
const Wrapper = mount(<>{roleMappingList}>);
Wrapper.find('[data-test-subj="map-users"]').first().simulate('click');
expect(window.location.hash).toBe(
buildHashUrl(ResourceType.roles, Action.edit, sampleRole, SubAction.mapuser)
);
});
it('should render to map user page when click on Manage Mapping', () => {
const wrapper = shallow(
);
const tabs = wrapper.find(EuiTabbedContent).dive();
tabs.find('[data-test-subj="manage-mapping"]').first().simulate('click');
expect(window.location.hash).toBe(
buildHashUrl(ResourceType.roles, Action.edit, sampleRole, SubAction.mapuser)
);
});
it('fetch data', (done) => {
const component = shallow(
);
process.nextTick(() => {
expect(getRoleMappingData).toHaveBeenCalledTimes(1);
expect(transformRoleMappingData).toHaveBeenCalledTimes(1);
expect(fetchActionGroups).toHaveBeenCalledTimes(1);
expect(getRoleDetail).toHaveBeenCalledTimes(1);
expect(transformRoleIndexPermissions).toHaveBeenCalledTimes(1);
done();
});
});
it('fetch data error', () => {
getRoleMappingData.mockImplementationOnce(() => {
throw new Error();
});
// Hide the error message
jest.spyOn(console, 'log').mockImplementationOnce(() => {});
shallow(
);
expect(setState).toHaveBeenCalledWith(true);
});
it('delete role mapping', (done) => {
shallow(
);
const deleteFunc = useDeleteConfirmState.mock.calls[0][0];
deleteFunc();
process.nextTick(() => {
expect(updateRoleMapping).toBeCalled();
done();
});
});
it('should capture error by console.log if error occurred while deleting role mapping', (done) => {
(updateRoleMapping as jest.Mock).mockImplementationOnce(() => {
throw new Error();
});
const spy = jest.spyOn(console, 'log').mockImplementationOnce(() => {});
shallow(
);
const deleteFunc = useDeleteConfirmState.mock.calls[0][0];
deleteFunc();
process.nextTick(() => {
expect(spy).toBeCalled();
done();
});
});
it('delete role', () => {
const component = shallow(
);
component.find('[data-test-subj="delete"]').simulate('click');
expect(requestDeleteRoles).toBeCalled();
});
it('error occurred while deleting the role', () => {
(requestDeleteRoles as jest.Mock).mockImplementationOnce(() => {
throw new Error();
});
const component = shallow(
);
component.find('[data-test-subj="delete"]').simulate('click');
expect(createUnknownErrorToast).toBeCalled();
});
});