diff --git a/CHANGELOG.md b/CHANGELOG.md
index c4fcfc2f6..5ed328dcc 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -15,6 +15,7 @@
* Create new instance success toast no longer shows the instance HRID. Fixes UIIN-2635.
* Optimistic locking message not working for instances in non-consortial tenant. Fixes UIIN-2628.
* Add immediate warning message when a local instance is shared. Refs UIIN-2617.
+* ECS: Show info message when user in member tenant tries to view shared instance details without permission. Refs UIIN-2590.
* Show only local MARC Authorities when share local instance. Fixes UIIN-2647.
## [10.0.1](https://github.com/folio-org/ui-inventory/tree/v10.0.1) (2023-11-03)
diff --git a/src/Instance/InstanceDetails/InstanceDetails.js b/src/Instance/InstanceDetails/InstanceDetails.js
index 5f41c0c18..ee39f3d1f 100644
--- a/src/Instance/InstanceDetails/InstanceDetails.js
+++ b/src/Instance/InstanceDetails/InstanceDetails.js
@@ -19,7 +19,6 @@ import {
PaneMenu,
Row,
MessageBanner,
- Icon,
} from '@folio/stripes/components';
import { InstanceTitle } from './InstanceTitle';
@@ -71,8 +70,6 @@ const InstanceDetails = forwardRef(({
tagsEnabled,
userTenantPermissions,
isShared,
- isLoading,
- isInstanceSharing,
...rest
}, ref) => {
const intl = useIntl();
@@ -86,15 +83,12 @@ const InstanceDetails = forwardRef(({
const accordionState = useMemo(() => getAccordionState(instance, accordions), [instance]);
const [helperApp, setHelperApp] = useState();
- const isBasicPane = isInstanceSharing || isLoading;
+ const canCreateHoldings = stripes.hasPerm('ui-inventory.holdings.create');
const tags = instance?.tags?.tagList;
const isUserInCentralTenant = checkIfUserInCentralTenant(stripes);
-
- const canCreateHoldings = stripes.hasPerm('ui-inventory.holdings.create');
+ const isConsortialHoldingsVisible = instance?.shared || isInstanceShadowCopy(instance?.source);
const detailsLastMenu = useMemo(() => {
- if (isBasicPane) return null;
-
return (
{
@@ -110,15 +104,9 @@ const InstanceDetails = forwardRef(({
}
);
- }, [isBasicPane, tagsEnabled, tags, intl]);
- const detailsActionMenu = useMemo(
- () => (isBasicPane ? null : actionMenu),
- [isBasicPane, actionMenu],
- );
+ }, [tagsEnabled, tags, intl]);
const renderPaneTitle = () => {
- if (isBasicPane) return intl.formatMessage({ id: 'ui-inventory.edit' });
-
const isInstanceShared = Boolean(isShared || isInstanceShadowCopy(instance?.source));
return (
@@ -134,8 +122,6 @@ const InstanceDetails = forwardRef(({
};
const renderPaneSubtitle = () => {
- if (isBasicPane) return null;
-
return (
{
- if (isInstanceSharing) {
- return (
-
-
-
-
-
- );
- }
-
- if (isLoading) {
- return (
-
-
-
- );
- }
-
- const isConsortialHoldingsVisible = instance?.shared || isInstanceShadowCopy(instance?.source);
-
- return (
- <>
+ return (
+ <>
+ }
+ paneTitle={renderPaneTitle()}
+ paneSub={renderPaneSubtitle()}
+ actionMenu={actionMenu}
+ lastMenu={detailsLastMenu}
+ dismissible
+ onClose={onClose}
+ defaultWidth="fill"
+ >
@@ -293,25 +267,6 @@ const InstanceDetails = forwardRef(({
/>
- >
- );
- };
-
- return (
- <>
- }
- paneTitle={renderPaneTitle()}
- paneSub={renderPaneSubtitle()}
- actionMenu={detailsActionMenu}
- lastMenu={detailsLastMenu}
- dismissible
- onClose={onClose}
- defaultWidth="fill"
- >
- {renderDetails()}
{ helperApp && }
>
@@ -326,16 +281,12 @@ InstanceDetails.propTypes = {
tagsToggle: PropTypes.func,
tagsEnabled: PropTypes.bool,
userTenantPermissions: PropTypes.arrayOf(PropTypes.object),
- isLoading: PropTypes.bool,
- isInstanceSharing: PropTypes.bool,
isShared: PropTypes.bool,
};
InstanceDetails.defaultProps = {
instance: {},
tagsEnabled: false,
- isLoading: false,
- isInstanceSharing: false,
isShared: false,
};
diff --git a/src/Instance/InstanceDetails/InstanceDetails.test.js b/src/Instance/InstanceDetails/InstanceDetails.test.js
index e8cd22f7e..daf483dcd 100644
--- a/src/Instance/InstanceDetails/InstanceDetails.test.js
+++ b/src/Instance/InstanceDetails/InstanceDetails.test.js
@@ -112,13 +112,6 @@ describe('InstanceDetails', () => {
expect(screen.getByText('Instance relationship (analytics and bound-with)')).toBeInTheDocument();
});
- it('should show a correct Warning message banner when instance sharing is in progress', () => {
- renderInstanceDetails({ isInstanceSharing: true });
-
- expect(screen.getByText('Sharing this local instance will take a few moments.' +
- ' A success message and updated details will be displayed upon completion.')).toBeInTheDocument();
- });
-
it('should show a correct Warning message banner when staff suppressed', () => {
const staffSuppressedInstance = {
...instance,
diff --git a/src/Instance/InstanceDetails/InstanceLoadingPane/InstanceLoadingPane.js b/src/Instance/InstanceDetails/InstanceLoadingPane/InstanceLoadingPane.js
new file mode 100644
index 000000000..28d33c5eb
--- /dev/null
+++ b/src/Instance/InstanceDetails/InstanceLoadingPane/InstanceLoadingPane.js
@@ -0,0 +1,33 @@
+import React from 'react';
+import PropTypes from 'prop-types';
+import { useIntl } from 'react-intl';
+
+import { AppIcon } from '@folio/stripes/core';
+import {
+ Pane,
+ Icon,
+} from '@folio/stripes/components';
+
+const InstanceLoadingPane = ({ onClose }) => {
+ const intl = useIntl();
+
+ return (
+ }
+ paneTitle={intl.formatMessage({ id: 'ui-inventory.edit' })}
+ dismissible
+ onClose={onClose}
+ defaultWidth="fill"
+ >
+
+
+ );
+};
+
+InstanceLoadingPane.propTypes = { onClose: PropTypes.func.isRequired };
+
+export default InstanceLoadingPane;
diff --git a/src/Instance/InstanceDetails/InstanceLoadingPane/InstanceLoadingPane.test.js b/src/Instance/InstanceDetails/InstanceLoadingPane/InstanceLoadingPane.test.js
new file mode 100644
index 000000000..a8f1e7bc8
--- /dev/null
+++ b/src/Instance/InstanceDetails/InstanceLoadingPane/InstanceLoadingPane.test.js
@@ -0,0 +1,53 @@
+import {
+ fireEvent,
+ screen,
+} from '@folio/jest-config-stripes/testing-library/react';
+
+import '../../../../test/jest/__mock__';
+
+import { Icon } from '@folio/stripes/components';
+
+import {
+ renderWithIntl,
+ translationsProperties,
+} from '../../../../test/jest/helpers';
+
+import InstanceLoadingPane from './InstanceLoadingPane';
+
+Icon.mockClear().mockImplementation(({ icon }) => {icon});
+
+const mockOnClose = jest.fn();
+
+const renderInstanceLoadingPane = () => {
+ const component = ;
+
+ return renderWithIntl(component, translationsProperties);
+};
+
+describe('InstanceLoadingPane', () => {
+ it('should render loading spinner', () => {
+ renderInstanceLoadingPane();
+
+ expect(screen.getByText('spinner-ellipsis')).toBeInTheDocument();
+ });
+
+ it('should render correct header', () => {
+ renderInstanceLoadingPane();
+
+ expect(screen.getByText('Edit')).toBeInTheDocument();
+ });
+
+ it('should not render action menu', () => {
+ renderInstanceLoadingPane();
+
+ expect(screen.queryByRole('button', { name: 'Actions' })).not.toBeInTheDocument();
+ });
+
+ it('should call onClose cb when pane is closed', () => {
+ renderInstanceLoadingPane();
+
+ fireEvent.click(screen.getByRole('button', { name: 'Close Edit' }));
+
+ expect(mockOnClose).toHaveBeenCalled();
+ });
+});
diff --git a/src/Instance/InstanceDetails/InstanceLoadingPane/index.js b/src/Instance/InstanceDetails/InstanceLoadingPane/index.js
new file mode 100644
index 000000000..29e7c58ee
--- /dev/null
+++ b/src/Instance/InstanceDetails/InstanceLoadingPane/index.js
@@ -0,0 +1 @@
+export { default } from './InstanceLoadingPane';
diff --git a/src/Instance/InstanceDetails/InstanceWarningPane/InstanceWarningPane.js b/src/Instance/InstanceDetails/InstanceWarningPane/InstanceWarningPane.js
new file mode 100644
index 000000000..8762ddf3e
--- /dev/null
+++ b/src/Instance/InstanceDetails/InstanceWarningPane/InstanceWarningPane.js
@@ -0,0 +1,38 @@
+import React from 'react';
+import PropTypes from 'prop-types';
+import { useIntl } from 'react-intl';
+
+import { AppIcon } from '@folio/stripes/core';
+import {
+ Pane,
+ MessageBanner,
+} from '@folio/stripes/components';
+
+const InstanceWarningPane = ({
+ onClose,
+ messageBannerText,
+}) => {
+ const intl = useIntl();
+
+ return (
+ }
+ paneTitle={intl.formatMessage({ id: 'ui-inventory.edit' })}
+ dismissible
+ onClose={onClose}
+ defaultWidth="fill"
+ >
+
+ {messageBannerText}
+
+
+ );
+};
+
+InstanceWarningPane.propTypes = {
+ onClose: PropTypes.func.isRequired,
+ messageBannerText: PropTypes.oneOfType([PropTypes.element, PropTypes.string]).isRequired,
+};
+
+export default InstanceWarningPane;
diff --git a/src/Instance/InstanceDetails/InstanceWarningPane/InstanceWarningPane.test.js b/src/Instance/InstanceDetails/InstanceWarningPane/InstanceWarningPane.test.js
new file mode 100644
index 000000000..bb4603a2d
--- /dev/null
+++ b/src/Instance/InstanceDetails/InstanceWarningPane/InstanceWarningPane.test.js
@@ -0,0 +1,58 @@
+import {
+ fireEvent,
+ screen,
+} from '@folio/jest-config-stripes/testing-library/react';
+
+import '../../../../test/jest/__mock__';
+
+import { Icon } from '@folio/stripes/components';
+
+import {
+ renderWithIntl,
+ translationsProperties,
+} from '../../../../test/jest/helpers';
+
+import InstanceWarningPane from './InstanceWarningPane';
+
+Icon.mockClear().mockImplementation(({ icon }) => {icon});
+
+const mockOnClose = jest.fn();
+
+const renderInstanceLoadingPane = (messageBannerText = 'warning text') => {
+ const component = (
+
+ );
+
+ return renderWithIntl(component, translationsProperties);
+};
+
+describe('InstanceWarningPane', () => {
+ it('should render warning banner', () => {
+ renderInstanceLoadingPane();
+
+ expect(screen.getByText('warning text')).toBeInTheDocument();
+ });
+
+ it('should render correct header', () => {
+ renderInstanceLoadingPane();
+
+ expect(screen.getByText('Edit')).toBeInTheDocument();
+ });
+
+ it('should not render action menu', () => {
+ renderInstanceLoadingPane();
+
+ expect(screen.queryByRole('button', { name: 'Actions' })).not.toBeInTheDocument();
+ });
+
+ it('should call onClose cb when pane is closed', () => {
+ renderInstanceLoadingPane();
+
+ fireEvent.click(screen.getByRole('button', { name: 'Close Edit' }));
+
+ expect(mockOnClose).toHaveBeenCalled();
+ });
+});
diff --git a/src/Instance/InstanceDetails/InstanceWarningPane/index.js b/src/Instance/InstanceDetails/InstanceWarningPane/index.js
new file mode 100644
index 000000000..ab6da1637
--- /dev/null
+++ b/src/Instance/InstanceDetails/InstanceWarningPane/index.js
@@ -0,0 +1 @@
+export { default } from './InstanceWarningPane';
diff --git a/src/Instance/InstanceDetails/index.js b/src/Instance/InstanceDetails/index.js
index 065a56d8f..be1ebdbf8 100644
--- a/src/Instance/InstanceDetails/index.js
+++ b/src/Instance/InstanceDetails/index.js
@@ -11,6 +11,8 @@ export * from './InstanceTitle';
export * from './InstanceNotesView';
export * from './InstanceNewHolding';
+export { default as InstanceLoadingPane } from './InstanceLoadingPane';
+export { default as InstanceWarningPane } from './InstanceWarningPane';
export { default as SubInstanceList } from './SubInstanceList';
export { default as InstanceDetails } from './InstanceDetails';
export { default as SubInstanceGroup } from './SubInstanceGroup';
diff --git a/src/ViewInstance.js b/src/ViewInstance.js
index 4ced3f855..053d8a80f 100644
--- a/src/ViewInstance.js
+++ b/src/ViewInstance.js
@@ -41,6 +41,7 @@ import {
} from './utils';
import {
CONSORTIUM_PREFIX,
+ HTTP_RESPONSE_STATUS_CODES,
indentifierTypeNames,
INSTANCE_SHARING_STATUSES,
layers,
@@ -53,6 +54,8 @@ import {
HoldingsListContainer,
MoveItemsContext,
InstanceDetails,
+ InstanceWarningPane,
+ InstanceLoadingPane,
} from './Instance';
import {
ActionItem,
@@ -881,11 +884,18 @@ class ViewInstance extends React.Component {
isCentralTenantPermissionsLoading,
isShared,
isLoading,
+ isError,
+ error,
} = this.props;
const {
linkedAuthoritiesLength,
isInstanceSharing,
} = this.state;
+
+ const isUserLacksPermToViewSharedInstance = isError
+ && error?.response?.status === HTTP_RESPONSE_STATUS_CODES.FORBIDDEN
+ && isShared;
+
const ci = makeConnectedInstance(this.props, stripes.logger);
const instance = ci.instance();
@@ -922,6 +932,28 @@ class ViewInstance extends React.Component {
const isInstanceLoading = isLoading || !instance || isCentralTenantPermissionsLoading;
const keyInStorageToHoldingsAccsState = ['holdings'];
+ if (isUserLacksPermToViewSharedInstance) {
+ return (
+ }
+ />
+ );
+ }
+
+ if (isInstanceSharing) {
+ return (
+ }
+ />
+ );
+ }
+
+ if (isInstanceLoading) {
+ return ;
+ }
+
return (
{data => (
@@ -938,8 +970,6 @@ class ViewInstance extends React.Component {
tagsEnabled={tagsEnabled}
ref={this.accordionStatusRef}
userTenantPermissions={this.state.userTenantPermissions}
- isLoading={isInstanceLoading}
- isInstanceSharing={isInstanceSharing}
isShared={isShared}
>
{
@@ -1099,6 +1129,8 @@ ViewInstance.propTypes = {
tagsEnabled: PropTypes.bool,
updateLocation: PropTypes.func.isRequired,
isLoading: PropTypes.bool,
+ isError: PropTypes.bool,
+ error: PropTypes.object,
};
export default flowRight(
diff --git a/src/ViewInstance.test.js b/src/ViewInstance.test.js
index 8e23c34ca..91a968489 100644
--- a/src/ViewInstance.test.js
+++ b/src/ViewInstance.test.js
@@ -18,7 +18,10 @@ import {
checkIfUserInMemberTenant,
checkIfUserInCentralTenant,
} from '@folio/stripes/core';
-import { ConfirmationModal } from '@folio/stripes/components';
+import {
+ ConfirmationModal,
+ Icon,
+} from '@folio/stripes/components';
import { instances } from '../test/fixtures/instances';
import { DataContext } from './contexts';
@@ -103,6 +106,8 @@ jest
.mockImplementation(() => instance)
.mockImplementationOnce(() => {});
+Icon.mockClear().mockImplementation(({ children, icon }) => (children || {icon}));
+
ConfirmationModal.mockImplementation(({
open,
onCancel,
@@ -290,6 +295,35 @@ describe('ViewInstance', () => {
});
useStripes().hasInterface.mockReturnValue(true);
});
+
+ it('should show a correct Warning message banner when instance sharing is in progress', async () => {
+ defaultProp.mutator.shareInstance.POST.mockResolvedValue({});
+ checkIfUserInMemberTenant.mockClear().mockReturnValue(true);
+
+ renderViewInstance({ isShared: false });
+
+ fireEvent.click(screen.getByRole('button', { name: 'Actions' }));
+ fireEvent.click(screen.getByRole('button', { name: 'Share local instance' }));
+ fireEvent.click(screen.getByRole('button', { name: 'Confirm' }));
+
+ await waitFor(() => expect(screen.getByText('Sharing this local instance will take a few moments.' +
+ ' A success message and updated details will be displayed upon completion.')).toBeInTheDocument());
+ });
+ it('should show a correct Warning message banner when user lacks permission to vew shared instance', () => {
+ renderViewInstance({
+ isError: true,
+ isShared: true,
+ error: { response: { status: 403 } },
+ });
+
+ expect(screen.getByText('You do not currently have permission to access details of shared instances. ' +
+ 'Contact your FOLIO administrator for more information.')).toBeInTheDocument();
+ });
+ it('should show loading spinner when instance is loading', () => {
+ renderViewInstance({ isLoading: true });
+
+ expect(screen.getByText('spinner-ellipsis')).toBeInTheDocument();
+ });
it('should display action menu items', () => {
renderViewInstance();
expect(screen.getByText('Move items within an instance')).toBeInTheDocument();
diff --git a/src/ViewInstanceWrapper.js b/src/ViewInstanceWrapper.js
index 170ed5ef1..fddc7c78a 100644
--- a/src/ViewInstanceWrapper.js
+++ b/src/ViewInstanceWrapper.js
@@ -17,7 +17,13 @@ const ViewInstanceWrapper = (props) => {
const userId = stripes?.user?.user?.id;
const centralTenantId = stripes.user.user?.consortium?.centralTenantId;
const consortiumId = stripes.user.user?.consortium?.id;
- const { instance, isLoading, refetch } = useInstance(id);
+ const {
+ instance,
+ isLoading,
+ refetch,
+ isError,
+ error,
+ } = useInstance(id);
const isShared = Boolean(instance?.shared);
const tenantId = instance?.tenantId ?? stripes.okapi.tenant;
@@ -42,6 +48,8 @@ const ViewInstanceWrapper = (props) => {
refetchInstance={refetch}
selectedInstance={instance}
isLoading={isLoading}
+ isError={isError}
+ error={error}
centralTenantPermissions={centralTenantPermissions}
isCentralTenantPermissionsLoading={isCentralTenantPermissionsLoading}
/>
diff --git a/src/common/hooks/useInstance.js b/src/common/hooks/useInstance.js
index 686408d17..585cb136d 100644
--- a/src/common/hooks/useInstance.js
+++ b/src/common/hooks/useInstance.js
@@ -4,12 +4,20 @@ import useSearchInstanceByIdQuery from './useSearchInstanceByIdQuery';
import useInstanceQuery from './useInstanceQuery';
const useInstance = (id) => {
- const { isLoading: isSearchInstanceByIdLoading, instance: _instance } = useSearchInstanceByIdQuery(id);
+ const {
+ isLoading: isSearchInstanceByIdLoading,
+ instance: _instance,
+ } = useSearchInstanceByIdQuery(id);
const instanceTenantId = _instance?.tenantId;
const isShared = _instance?.shared;
- const { isLoading: isInstanceLoading, instance: data, refetch } = useInstanceQuery(
+ const {
+ isLoading: isInstanceLoading,
+ instance: data,
+ refetch,
+ ...rest
+ } = useInstanceQuery(
id,
{ tenantId: instanceTenantId },
{ enabled: Boolean(id && !isSearchInstanceByIdLoading) }
@@ -32,6 +40,7 @@ const useInstance = (id) => {
instance,
isLoading,
refetch,
+ ...rest,
};
};
diff --git a/src/common/hooks/useInstanceQuery/useInstanceQuery.js b/src/common/hooks/useInstanceQuery/useInstanceQuery.js
index 1dc296331..4e478348a 100644
--- a/src/common/hooks/useInstanceQuery/useInstanceQuery.js
+++ b/src/common/hooks/useInstanceQuery/useInstanceQuery.js
@@ -8,7 +8,12 @@ const useInstanceQuery = (instanceId, { tenantId = '' } = {}, options = {}) => {
const ky = useTenantKy({ tenantId });
const [namespace] = useNamespace({ key: 'instance' });
- const { isLoading, data: instance = {}, refetch } = useQuery(
+ const {
+ isLoading,
+ data: instance = {},
+ refetch,
+ ...rest
+ } = useQuery(
[namespace, instanceId, tenantId],
() => ky.get(`inventory/instances/${instanceId}`).json(),
{
@@ -21,6 +26,7 @@ const useInstanceQuery = (instanceId, { tenantId = '' } = {}, options = {}) => {
isLoading,
instance,
refetch,
+ ...rest,
});
};
diff --git a/src/constants.js b/src/constants.js
index 2aaa0852c..2897e5e19 100644
--- a/src/constants.js
+++ b/src/constants.js
@@ -677,3 +677,7 @@ export const INSTANCE_SHARING_STATUSES = {
ERROR: 'ERROR',
IN_PROGRESS: 'IN_PROGRESS',
};
+
+export const HTTP_RESPONSE_STATUS_CODES = {
+ FORBIDDEN: 403,
+};
diff --git a/test/jest/__mock__/stripesIcon.mock.js b/test/jest/__mock__/stripesIcon.mock.js
index d50082820..ffcd3589b 100644
--- a/test/jest/__mock__/stripesIcon.mock.js
+++ b/test/jest/__mock__/stripesIcon.mock.js
@@ -1,7 +1,7 @@
import React from 'react';
jest.mock('@folio/stripes-components/lib/Icon', () => {
- return props => {
+ return jest.fn(props => {
return props.children ? props.children : ;
- };
+ });
});
diff --git a/translations/ui-inventory/en.json b/translations/ui-inventory/en.json
index 91e1ac580..68a67612c 100644
--- a/translations/ui-inventory/en.json
+++ b/translations/ui-inventory/en.json
@@ -799,6 +799,7 @@
"warning.instance.staffSuppressed": "Warning: Instance is marked staff suppressed",
"warning.instance.suppressedFromDiscoveryAndStaffSuppressed": "Warning: Instance is marked suppressed from discovery and staff suppressed",
"warning.instance.sharingLocalInstance": "Sharing this local instance will take a few moments. A success message and updated details will be displayed upon completion.",
+ "warning.instance.accessSharedInstance": "You do not currently have permission to access details of shared instances. Contact your FOLIO administrator for more information.",
"warning.holdingsRecord.suppressedFromDiscovery": "Warning: Holdings is marked suppressed from discovery",
"warning.item.suppressedFromDiscovery": "Warning: Item is marked suppressed from discovery",
"discoverySuppressed": "Suppressed from discovery",