Skip to content

Commit

Permalink
feat: reduce unnecessary api calls (#129)
Browse files Browse the repository at this point in the history
* feat: reduce unnecessary api calls

* test: fix broken test
  • Loading branch information
Zacharis278 committed Jan 8, 2024
1 parent 8457727 commit 7f6f442
Show file tree
Hide file tree
Showing 2 changed files with 57 additions and 5 deletions.
5 changes: 4 additions & 1 deletion src/exam/ExamWrapper.jsx
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,10 @@ const ExamWrapper = ({ children, ...props }) => {
const isGated = sequence && sequence.gatedContent !== undefined && sequence.gatedContent.gated;

useEffect(() => {
loadInitialData();
// fetch exam data on exam sequences or if no exam data has been fetched yet
if (sequence.isTimeLimited || state.isLoading) {
loadInitialData();
}
// eslint-disable-next-line react-hooks/exhaustive-deps
}, []);

Expand Down
57 changes: 53 additions & 4 deletions src/exam/ExamWrapper.test.jsx
Original file line number Diff line number Diff line change
Expand Up @@ -2,16 +2,28 @@ import '@testing-library/jest-dom';
import { Factory } from 'rosie';
import React from 'react';
import SequenceExamWrapper from './ExamWrapper';
import { store, getExamAttemptsData, startTimedExam } from '../data';
import { render } from '../setupTest';
import { store, startTimedExam } from '../data';
import { getExamAttemptsData } from '../data/thunks';
import { render, waitFor } from '../setupTest';
import ExamStateProvider from '../core/ExamStateProvider';
import { ExamStatus, ExamType } from '../constants';

jest.mock('../data', () => ({
store: {},
getExamAttemptsData: jest.fn(),
startTimedExam: jest.fn(),
}));

// because of the way ExamStateProvider and other locations inconsistantly import from
// thunks directly instead of using the data module we need to mock the underlying
// thunk file. It would be nice to clean this up in the future.
jest.mock('../data/thunks', () => {
const originalModule = jest.requireActual('../data/thunks');
return {
...originalModule,
getExamAttemptsData: jest.fn(),
};
});

getExamAttemptsData.mockReturnValue(jest.fn());
startTimedExam.mockReturnValue(jest.fn());
store.subscribe = jest.fn();
Expand All @@ -24,10 +36,15 @@ describe('SequenceExamWrapper', () => {
};
const courseId = 'course-v1:test+test+test';

it('is successfully rendered and shows instructions if the user is not staff', () => {
beforeEach(() => {
jest.clearAllMocks();
store.getState = () => ({
examState: Factory.build('examState'),
isLoading: false,
});
});

it('is successfully rendered and shows instructions if the user is not staff', () => {
const { queryByTestId } = render(
<ExamStateProvider>
<SequenceExamWrapper sequence={sequence} courseId={courseId}>
Expand Down Expand Up @@ -114,6 +131,38 @@ describe('SequenceExamWrapper', () => {
expect(queryByTestId('exam-api-error-component')).not.toBeInTheDocument();
});

it('does not fetch exam data if already loaded and the sequence is not an exam', async () => {
render(
<ExamStateProvider>
<SequenceExamWrapper sequence={{ ...sequence, isTimeLimited: false }} courseId={courseId}>
<div>children</div>
</SequenceExamWrapper>
</ExamStateProvider>,
{ store },
);
// assert the exam data is not fetched
await expect(waitFor(() => expect(getExamAttemptsData).toHaveBeenCalled())).rejects.toThrow();
});

it('does fetch exam data for non exam sequences if not already loaded', async () => {
// this would only occur if the user deeplinks directly to a non-exam sequence
store.getState = () => ({
examState: Factory.build('examState', {
isLoading: true,
}),
});

render(
<ExamStateProvider>
<SequenceExamWrapper sequence={{ ...sequence, isTimeLimited: false }} courseId={courseId}>
<div>children</div>
</SequenceExamWrapper>
</ExamStateProvider>,
{ store },
);
await waitFor(() => expect(getExamAttemptsData).toHaveBeenCalled());
});

it('does not take any actions if sequence item is not exam', () => {
const { getByTestId } = render(
<ExamStateProvider>
Expand Down

0 comments on commit 7f6f442

Please sign in to comment.