-
Notifications
You must be signed in to change notification settings - Fork 89
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Course Optimizer is a feature approved by the Openedx community that adds a "Course Optimizer" page to studio where users can run a scan of a course for broken links - links that point to pages that have a 404. Depends on backend: openedx/edx-platform#35887 - test together. This also requires adding a nav menu item to edx-platform legacy studio. That should be implemented before enabling the waffle flag on prod. Links: - [Internal JIRA ticket](https://2u-internal.atlassian.net/browse/TNL-11809) - [Course Optimizer Discovery](https://2u-internal.atlassian.net/wiki/spaces/TNL/pages/1426587703/TNL-11744+Course+Optimizer+Discovery) - [Openedx community proposal](openedx/platform-roadmap#388)
- Loading branch information
1 parent
e6bce56
commit 8385c4e
Showing
29 changed files
with
1,724 additions
and
8 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,193 @@ | ||
/* eslint-disable @typescript-eslint/no-shadow */ | ||
/* eslint-disable react/jsx-filename-extension */ | ||
import { | ||
fireEvent, render, waitFor, screen, | ||
} from '@testing-library/react'; | ||
import { IntlProvider } from '@edx/frontend-platform/i18n'; | ||
import { AppProvider } from '@edx/frontend-platform/react'; | ||
import { initializeMockApp } from '@edx/frontend-platform'; | ||
|
||
import { getAuthenticatedHttpClient } from '@edx/frontend-platform/auth'; | ||
import MockAdapter from 'axios-mock-adapter'; | ||
import initializeStore from '../store'; | ||
import messages from './messages'; | ||
import generalMessages from '../messages'; | ||
import scanResultsMessages from './scan-results/messages'; | ||
import CourseOptimizerPage, { pollLinkCheckDuringScan } from './CourseOptimizerPage'; | ||
import { postLinkCheckCourseApiUrl, getLinkCheckStatusApiUrl } from './data/api'; | ||
import mockApiResponse from './mocks/mockApiResponse'; | ||
import * as thunks from './data/thunks'; | ||
|
||
let store; | ||
let axiosMock; | ||
const courseId = '123'; | ||
const courseName = 'About Node JS'; | ||
|
||
jest.mock('../generic/model-store', () => ({ | ||
useModel: jest.fn().mockReturnValue({ | ||
name: courseName, | ||
}), | ||
})); | ||
|
||
const OptimizerPage = () => ( | ||
<AppProvider store={store}> | ||
<IntlProvider locale="en" messages={{}}> | ||
<CourseOptimizerPage courseId={courseId} /> | ||
</IntlProvider> | ||
</AppProvider> | ||
); | ||
|
||
describe('CourseOptimizerPage', () => { | ||
describe('pollLinkCheckDuringScan', () => { | ||
let mockFetchLinkCheckStatus; | ||
beforeEach(() => { | ||
mockFetchLinkCheckStatus = jest.fn(); | ||
jest.spyOn(thunks, 'fetchLinkCheckStatus').mockImplementation(mockFetchLinkCheckStatus); | ||
jest.useFakeTimers(); | ||
jest.spyOn(global, 'setInterval').mockImplementation((cb) => { cb(); return true; }); | ||
}); | ||
|
||
afterEach(() => { | ||
jest.clearAllTimers(); | ||
jest.useRealTimers(); | ||
jest.restoreAllMocks(); | ||
}); | ||
|
||
it('should start polling if linkCheckInProgress has never been started (is null)', () => { | ||
const linkCheckInProgress = null; | ||
const interval = { current: null }; | ||
const dispatch = jest.fn(); | ||
const courseId = 'course-123'; | ||
pollLinkCheckDuringScan(linkCheckInProgress, interval, dispatch, courseId); | ||
expect(interval.current).toBeTruthy(); | ||
expect(mockFetchLinkCheckStatus).toHaveBeenCalled(); | ||
}); | ||
|
||
it('should start polling if link check is in progress', () => { | ||
const linkCheckInProgress = true; | ||
const interval = { current: null }; | ||
const dispatch = jest.fn(); | ||
const courseId = 'course-123'; | ||
pollLinkCheckDuringScan(linkCheckInProgress, interval, dispatch, courseId); | ||
expect(interval.current).toBeTruthy(); | ||
}); | ||
it('should not start polling if link check is not in progress', () => { | ||
const linkCheckInProgress = false; | ||
const interval = { current: null }; | ||
const dispatch = jest.fn(); | ||
const courseId = 'course-123'; | ||
pollLinkCheckDuringScan(linkCheckInProgress, interval, dispatch, courseId); | ||
expect(interval.current).toBeFalsy(); | ||
}); | ||
it('should clear the interval if link check is finished', () => { | ||
const linkCheckInProgress = false; | ||
const interval = { current: 1 }; | ||
const dispatch = jest.fn(); | ||
const courseId = 'course-123'; | ||
pollLinkCheckDuringScan(linkCheckInProgress, interval, dispatch, courseId); | ||
expect(interval.current).toBeUndefined(); | ||
}); | ||
}); | ||
|
||
describe('CourseOptimizerPage component', () => { | ||
beforeEach(() => { | ||
jest.useRealTimers(); | ||
jest.clearAllMocks(); | ||
initializeMockApp({ | ||
authenticatedUser: { | ||
userId: 3, | ||
username: 'abc123', | ||
administrator: true, | ||
roles: [], | ||
}, | ||
}); | ||
store = initializeStore(); | ||
axiosMock = new MockAdapter(getAuthenticatedHttpClient()); | ||
axiosMock | ||
.onPost(postLinkCheckCourseApiUrl(courseId)) | ||
.reply(200, { LinkCheckStatus: 'In-Progress' }); | ||
axiosMock | ||
.onGet(getLinkCheckStatusApiUrl(courseId)) | ||
.reply(200, mockApiResponse); | ||
}); | ||
|
||
it('should render the component', () => { | ||
const { getByText, queryByText } = render(<OptimizerPage />); | ||
expect(getByText(messages.headingTitle.defaultMessage)).toBeInTheDocument(); | ||
expect(getByText(messages.buttonTitle.defaultMessage)).toBeInTheDocument(); | ||
expect(queryByText(messages.preparingStepTitle)).not.toBeInTheDocument(); | ||
}); | ||
|
||
it('should start scan after clicking the scan button', async () => { | ||
const { getByText } = render(<OptimizerPage />); | ||
expect(getByText(messages.headingTitle.defaultMessage)).toBeInTheDocument(); | ||
fireEvent.click(getByText(messages.buttonTitle.defaultMessage)); | ||
await waitFor(() => { | ||
expect(getByText(messages.preparingStepTitle.defaultMessage)).toBeInTheDocument(); | ||
}); | ||
}); | ||
|
||
it('should list broken links results', async () => { | ||
const { | ||
getByText, queryAllByText, getAllByText, container, | ||
} = render(<OptimizerPage />); | ||
expect(getByText(messages.headingTitle.defaultMessage)).toBeInTheDocument(); | ||
fireEvent.click(getByText(messages.buttonTitle.defaultMessage)); | ||
await waitFor(() => { | ||
expect(getByText('5 broken links')).toBeInTheDocument(); | ||
expect(getByText('5 locked links')).toBeInTheDocument(); | ||
}); | ||
const collapsibleTrigger = container.querySelector('.collapsible-trigger'); | ||
expect(collapsibleTrigger).toBeInTheDocument(); | ||
fireEvent.click(collapsibleTrigger); | ||
await waitFor(() => { | ||
expect(getAllByText(scanResultsMessages.brokenLinkStatus.defaultMessage)[0]).toBeInTheDocument(); | ||
expect(queryAllByText(scanResultsMessages.lockedLinkStatus.defaultMessage)[0]).toBeInTheDocument(); | ||
}); | ||
}); | ||
|
||
it('should not list locked links results when show locked links is unchecked', async () => { | ||
const { | ||
getByText, getAllByText, getByLabelText, queryAllByText, queryByText, container, | ||
} = render(<OptimizerPage />); | ||
expect(getByText(messages.headingTitle.defaultMessage)).toBeInTheDocument(); | ||
fireEvent.click(getByText(messages.buttonTitle.defaultMessage)); | ||
await waitFor(() => { | ||
expect(getByText('5 broken links')).toBeInTheDocument(); | ||
}); | ||
fireEvent.click(getByLabelText(scanResultsMessages.lockedCheckboxLabel.defaultMessage)); | ||
const collapsibleTrigger = container.querySelector('.collapsible-trigger'); | ||
expect(collapsibleTrigger).toBeInTheDocument(); | ||
fireEvent.click(collapsibleTrigger); | ||
await waitFor(() => { | ||
expect(queryByText('5 locked links')).not.toBeInTheDocument(); | ||
expect(getAllByText(scanResultsMessages.brokenLinkStatus.defaultMessage)[0]).toBeInTheDocument(); | ||
expect(queryAllByText(scanResultsMessages.lockedLinkStatus.defaultMessage)?.[0]).toBeUndefined(); | ||
}); | ||
}); | ||
|
||
it('should show no broken links found message', async () => { | ||
axiosMock | ||
.onGet(getLinkCheckStatusApiUrl(courseId)) | ||
.reply(200, { LinkCheckStatus: 'Succeeded' }); | ||
const { getByText } = render(<OptimizerPage />); | ||
expect(getByText(messages.headingTitle.defaultMessage)).toBeInTheDocument(); | ||
fireEvent.click(getByText(messages.buttonTitle.defaultMessage)); | ||
await waitFor(() => { | ||
expect(getByText(scanResultsMessages.noBrokenLinksCard.defaultMessage)).toBeInTheDocument(); | ||
}); | ||
}); | ||
|
||
it('should show error message if request does not go through', async () => { | ||
axiosMock | ||
.onPost(postLinkCheckCourseApiUrl(courseId)) | ||
.reply(500); | ||
render(<OptimizerPage />); | ||
expect(screen.getByText(messages.headingTitle.defaultMessage)).toBeInTheDocument(); | ||
fireEvent.click(screen.getByText(messages.buttonTitle.defaultMessage)); | ||
await waitFor(() => { | ||
expect(screen.getByText(generalMessages.supportText.defaultMessage)).toBeInTheDocument(); | ||
}); | ||
}); | ||
}); | ||
}); |
Oops, something went wrong.