diff --git a/src/course-checklist/CourseChecklist.test.jsx b/src/course-checklist/CourseChecklist.test.jsx
index 53d52af77e..69057abf70 100644
--- a/src/course-checklist/CourseChecklist.test.jsx
+++ b/src/course-checklist/CourseChecklist.test.jsx
@@ -149,5 +149,20 @@ describe('CourseChecklistPage', () => {
});
});
});
+
+ it('displays an alert and sets status to DENIED when API responds with 403', async () => {
+ const courseLaunchApiUrl = getCourseLaunchApiUrl({
+ courseId, gradedOnly: true, validateOras: true, all: true,
+ });
+ axiosMock.onGet(courseLaunchApiUrl).reply(403);
+
+ renderComponent();
+
+ await waitFor(() => {
+ const { launchChecklistStatus } = store.getState().courseChecklist.loadingStatus;
+ expect(launchChecklistStatus).toEqual(RequestStatus.DENIED);
+ expect(screen.getByRole('alert')).toBeInTheDocument();
+ });
+ });
});
});
diff --git a/src/course-outline/CourseOutline.test.jsx b/src/course-outline/CourseOutline.test.jsx
index b7f8332eeb..bfe6c07705 100644
--- a/src/course-outline/CourseOutline.test.jsx
+++ b/src/course-outline/CourseOutline.test.jsx
@@ -2291,4 +2291,18 @@ describe('', () => {
expect(await screen.findByText('Please wait. Creating export file for course tags...')).toBeInTheDocument();
expect(await screen.findByText('An error has occurred creating the file')).toBeInTheDocument();
});
+
+ it('displays an alert and sets status to DENIED when API responds with 403', async () => {
+ axiosMock
+ .onGet(getCourseOutlineIndexApiUrl(courseId))
+ .reply(403);
+
+ const { getByRole } = render();
+
+ await waitFor(() => {
+ expect(getByRole('alert')).toBeInTheDocument();
+ const { outlineIndexLoadingStatus } = store.getState().courseOutline.loadingStatus;
+ expect(outlineIndexLoadingStatus).toEqual(RequestStatus.DENIED);
+ });
+ });
});
diff --git a/src/course-team/CourseTeam.test.jsx b/src/course-team/CourseTeam.test.jsx
index 4f33788744..046e8ef95f 100644
--- a/src/course-team/CourseTeam.test.jsx
+++ b/src/course-team/CourseTeam.test.jsx
@@ -1,4 +1,3 @@
-import React from 'react';
import {
render,
fireEvent,
@@ -18,6 +17,7 @@ import CourseTeam from './CourseTeam';
import messages from './messages';
import { USER_ROLES } from '../constants';
import { executeThunk } from '../utils';
+import { RequestStatus } from '../data/constants';
import { changeRoleTeamUserQuery, deleteCourseTeamQuery } from './data/thunk';
let axiosMock;
@@ -219,4 +219,31 @@ describe('', () => {
await executeThunk(changeRoleTeamUserQuery(courseId, 'staff@example.com', { role: USER_ROLES.admin }), store.dispatch);
expect(getAllByText('Admin')).toHaveLength(1);
});
+
+ it('displays an alert and sets status to DENIED when API responds with 403', async () => {
+ axiosMock
+ .onGet(getCourseTeamApiUrl(courseId))
+ .reply(403);
+
+ const { getByRole } = render();
+
+ await waitFor(() => {
+ expect(getByRole('alert')).toBeInTheDocument();
+ const { loadingCourseTeamStatus } = store.getState().courseTeam;
+ expect(loadingCourseTeamStatus).toEqual(RequestStatus.DENIED);
+ });
+ });
+
+ it('sets loading status to FAILED upon receiving a 404 response from the API', async () => {
+ axiosMock
+ .onGet(getCourseTeamApiUrl(courseId))
+ .reply(404);
+
+ render();
+
+ await waitFor(() => {
+ const { loadingCourseTeamStatus } = store.getState().courseTeam;
+ expect(loadingCourseTeamStatus).toEqual(RequestStatus.FAILED);
+ });
+ });
});
diff --git a/src/course-updates/CourseUpdates.test.jsx b/src/course-updates/CourseUpdates.test.jsx
index 387d3b3c26..ab8e1a8022 100644
--- a/src/course-updates/CourseUpdates.test.jsx
+++ b/src/course-updates/CourseUpdates.test.jsx
@@ -1,5 +1,6 @@
-import React from 'react';
-import { render, waitFor, fireEvent } from '@testing-library/react';
+import {
+ render, waitFor, fireEvent,
+} from '@testing-library/react';
import { IntlProvider } from '@edx/frontend-platform/i18n';
import { AppProvider } from '@edx/frontend-platform/react';
import { initializeMockApp } from '@edx/frontend-platform';
@@ -19,6 +20,7 @@ import {
} from './data/thunk';
import initializeStore from '../store';
import { executeThunk } from '../utils';
+import { RequestStatus } from '../data/constants';
import { courseUpdatesMock, courseHandoutsMock } from './__mocks__';
import CourseUpdates from './CourseUpdates';
import messages from './messages';
@@ -278,6 +280,24 @@ describe('', () => {
expect(getByTestId('course-handouts-edit-button')).toBeDisabled();
});
});
+
+ it('displays an alert and sets status to DENIED when API responds with 403', async () => {
+ axiosMock
+ .onGet(getCourseUpdatesApiUrl(courseId))
+ .reply(403, courseUpdatesMock);
+ axiosMock
+ .onGet(getCourseHandoutApiUrl(courseId))
+ .reply(403);
+
+ const { getByTestId } = render();
+
+ await waitFor(() => {
+ expect(getByTestId('connectionErrorAlert')).toBeInTheDocument();
+ const { loadingStatuses } = store.getState().courseUpdates;
+ Object.values(loadingStatuses)
+ .some(status => expect(status).toEqual(RequestStatus.DENIED));
+ });
+ });
});
describe('saving failure API responses', () => {
diff --git a/src/export-page/CourseExportPage.test.jsx b/src/export-page/CourseExportPage.test.jsx
index 05db07d5bf..4c0a01d5e6 100644
--- a/src/export-page/CourseExportPage.test.jsx
+++ b/src/export-page/CourseExportPage.test.jsx
@@ -1,4 +1,3 @@
-import React from 'react';
import { getConfig, initializeMockApp } from '@edx/frontend-platform';
import { getAuthenticatedHttpClient } from '@edx/frontend-platform/auth';
import { IntlProvider, injectIntl } from '@edx/frontend-platform/i18n';
@@ -6,9 +5,10 @@ import { AppProvider } from '@edx/frontend-platform/react';
import { fireEvent, render, waitFor } from '@testing-library/react';
import MockAdapter from 'axios-mock-adapter';
import { Helmet } from 'react-helmet';
-
import Cookies from 'universal-cookie';
+
import initializeStore from '../store';
+import { RequestStatus } from '../data/constants';
import stepperMessages from './export-stepper/messages';
import modalErrorMessages from './export-modal-error/messages';
import { getExportStatusApiUrl, postExportCourseApiUrl } from './data/api';
@@ -137,4 +137,30 @@ describe('', () => {
expect(downloadButton).toBeInTheDocument();
expect(downloadButton.getAttribute('href')).toEqual('http://test-download-path.test');
});
+ it('displays an alert and sets status to DENIED when API responds with 403', async () => {
+ axiosMock
+ .onGet(getExportStatusApiUrl(courseId))
+ .reply(403);
+ const { getByRole, container } = render();
+ const startExportButton = container.querySelector('.btn-primary');
+ fireEvent.click(startExportButton);
+ // eslint-disable-next-line no-promise-executor-return
+ await new Promise((r) => setTimeout(r, 3500));
+ expect(getByRole('alert')).toBeInTheDocument();
+ const { loadingStatus } = store.getState().courseExport;
+ expect(loadingStatus).toEqual(RequestStatus.DENIED);
+ });
+
+ it('sets loading status to FAILED upon receiving a 404 response from the API', async () => {
+ axiosMock
+ .onGet(getExportStatusApiUrl(courseId))
+ .reply(404);
+ const { container } = render();
+ const startExportButton = container.querySelector('.btn-primary');
+ fireEvent.click(startExportButton);
+ // eslint-disable-next-line no-promise-executor-return
+ await new Promise((r) => setTimeout(r, 3500));
+ const { loadingStatus } = store.getState().courseExport;
+ expect(loadingStatus).toEqual(RequestStatus.FAILED);
+ });
});
diff --git a/src/grading-settings/GradingSettings.jsx b/src/grading-settings/GradingSettings.jsx
index 8ddd7a9bb5..bcb3a8f089 100644
--- a/src/grading-settings/GradingSettings.jsx
+++ b/src/grading-settings/GradingSettings.jsx
@@ -8,7 +8,6 @@ import {
useGradingSettings,
useGradingSettingUpdater,
} from 'CourseAuthoring/grading-settings/data/apiHooks';
-import { RequestStatus } from 'CourseAuthoring/data/constants';
import PropTypes from 'prop-types';
import React, { useEffect, useState } from 'react';
import { Helmet } from 'react-helmet';
@@ -34,10 +33,12 @@ const GradingSettings = ({ courseId }) => {
const {
data: gradingSettings,
isLoading: isGradingSettingsLoading,
+ isError: isGradingSettingsError,
} = useGradingSettings(courseId);
const {
data: courseSettingsData,
isLoading: isCourseSettingsLoading,
+ isError: isCourseSettingsError,
} = useCourseSettings(courseId);
const {
mutate: updateGradingSettings,
@@ -48,7 +49,7 @@ const GradingSettings = ({ courseId }) => {
const courseAssignmentLists = gradingSettings?.courseAssignmentLists;
const courseGradingDetails = gradingSettings?.courseDetails;
- const isLoadingDenied = isCourseSettingsLoading === RequestStatus.DENIED;
+ const isLoadingDenied = isGradingSettingsError || isCourseSettingsError;
const [showSuccessAlert, setShowSuccessAlert] = useState(false);
const isLoading = isCourseSettingsLoading || isGradingSettingsLoading;
const [isQueryPending, setIsQueryPending] = useState(false);
diff --git a/src/group-configurations/GroupConfigurations.test.jsx b/src/group-configurations/GroupConfigurations.test.jsx
index 34486c368b..303b557f2c 100644
--- a/src/group-configurations/GroupConfigurations.test.jsx
+++ b/src/group-configurations/GroupConfigurations.test.jsx
@@ -103,4 +103,21 @@ describe('', () => {
RequestStatus.FAILED,
);
});
+
+ it('displays an alert and sets status to DENIED when API responds with 403', async () => {
+ axiosMock
+ .onGet(getContentStoreApiUrl(courseId))
+ .reply(403);
+
+ await executeThunk(fetchGroupConfigurationsQuery(courseId), store.dispatch);
+
+ const { getByTestId } = renderComponent();
+
+ await waitFor(() => {
+ expect(getByTestId('connectionErrorAlert')).toBeInTheDocument();
+ expect(store.getState().groupConfigurations.loadingStatus).toBe(
+ RequestStatus.DENIED,
+ );
+ });
+ });
});
diff --git a/src/import-page/CourseImportPage.test.jsx b/src/import-page/CourseImportPage.test.jsx
index 8359312692..646eadbe6e 100644
--- a/src/import-page/CourseImportPage.test.jsx
+++ b/src/import-page/CourseImportPage.test.jsx
@@ -1,14 +1,14 @@
-import React from 'react';
import { initializeMockApp } from '@edx/frontend-platform';
import { IntlProvider, injectIntl } from '@edx/frontend-platform/i18n';
import { AppProvider } from '@edx/frontend-platform/react';
import { render, waitFor } from '@testing-library/react';
import { Helmet } from 'react-helmet';
-
import MockAdapter from 'axios-mock-adapter';
import { getAuthenticatedHttpClient } from '@edx/frontend-platform/auth';
+
import Cookies from 'universal-cookie';
import initializeStore from '../store';
+import { RequestStatus } from '../data/constants';
import messages from './messages';
import CourseImportPage from './CourseImportPage';
import { getImportStatusApiUrl } from './data/api';
@@ -108,4 +108,29 @@ describe('', () => {
await new Promise((r) => setTimeout(r, 3500));
expect(getByText(stepperMessages.viewOutlineButton.defaultMessage)).toBeInTheDocument();
});
+
+ it('displays an alert and sets status to DENIED when API responds with 403', async () => {
+ axiosMock
+ .onGet(getImportStatusApiUrl(courseId, 'testFileName.tar.gz'))
+ .reply(403);
+ cookies.get.mockReturnValue({ date: 1679787000, completed: false, fileName: 'testFileName.tar.gz' });
+ const { getByRole } = render();
+ // eslint-disable-next-line no-promise-executor-return
+ await new Promise((r) => setTimeout(r, 3500));
+ expect(getByRole('alert')).toBeInTheDocument();
+ const { loadingStatus } = store.getState().courseImport;
+ expect(loadingStatus).toEqual(RequestStatus.DENIED);
+ });
+
+ it('sets loading status to FAILED upon receiving a 404 response from the API', async () => {
+ axiosMock
+ .onGet(getImportStatusApiUrl(courseId, 'testFileName.tar.gz'))
+ .reply(404);
+ cookies.get.mockReturnValue({ date: 1679787000, completed: false, fileName: 'testFileName.tar.gz' });
+ render();
+ // eslint-disable-next-line no-promise-executor-return
+ await new Promise((r) => setTimeout(r, 3500));
+ const { loadingStatus } = store.getState().courseImport;
+ expect(loadingStatus).toEqual(RequestStatus.FAILED);
+ });
});
diff --git a/src/import-page/data/thunks.js b/src/import-page/data/thunks.js
index 95acb291ec..028fea2da2 100644
--- a/src/import-page/data/thunks.js
+++ b/src/import-page/data/thunks.js
@@ -1,4 +1,3 @@
-/* eslint-disable import/prefer-default-export */
import Cookies from 'universal-cookie';
import moment from 'moment';
@@ -32,7 +31,11 @@ export function fetchImportStatus(courseId, fileName) {
dispatch(updateLoadingStatus(RequestStatus.SUCCESSFUL));
return true;
} catch (error) {
- dispatch(updateLoadingStatus(RequestStatus.FAILED));
+ if (error.response && error.response.status === 403) {
+ dispatch(updateLoadingStatus(RequestStatus.DENIED));
+ } else {
+ dispatch(updateLoadingStatus(RequestStatus.FAILED));
+ }
return false;
}
};
diff --git a/src/textbooks/Textbook.test.jsx b/src/textbooks/Textbook.test.jsx
index b8437f114f..60af1e3026 100644
--- a/src/textbooks/Textbook.test.jsx
+++ b/src/textbooks/Textbook.test.jsx
@@ -6,6 +6,7 @@ import { initializeMockApp } from '@edx/frontend-platform';
import { getAuthenticatedHttpClient } from '@edx/frontend-platform/auth';
import userEvent from '@testing-library/user-event';
+import { RequestStatus } from '../data/constants';
import initializeStore from '../store';
import { executeThunk } from '../utils';
import { getTextbooksApiUrl } from './data/api';
@@ -84,4 +85,19 @@ describe('', () => {
expect(queryAllByTestId('textbook-card')).toHaveLength(0);
});
});
+
+ it('displays an alert and sets status to FAILED when API responds with 403', async () => {
+ axiosMock
+ .onGet(getTextbooksApiUrl(courseId))
+ .reply(403);
+ await executeThunk(fetchTextbooksQuery(courseId), store.dispatch);
+ const { getByTestId } = renderComponent();
+
+ await waitFor(() => {
+ expect(getByTestId('connectionErrorAlert')).toBeInTheDocument();
+ expect(store.getState().textbooks.loadingStatus).toBe(
+ RequestStatus.FAILED,
+ );
+ });
+ });
});