Skip to content

Commit

Permalink
feat: added waffle flag state for Courseware Search
Browse files Browse the repository at this point in the history
  • Loading branch information
rijuma committed Oct 10, 2023
1 parent 5604def commit 84cc806
Show file tree
Hide file tree
Showing 10 changed files with 157 additions and 0 deletions.
26 changes: 26 additions & 0 deletions src/course-home/courseware-search/CoursewareSearch.jsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
import React from 'react';
import { useParams } from 'react-router-dom';
import { injectIntl, intlShape } from '@edx/frontend-platform/i18n';
import { Button, Icon } from '@edx/paragon';
import { Search } from '@edx/paragon/icons';
import { useModel } from '../../generic/model-store';
import messages from './messages';

const CoursewareSearch = ({ intl, ...rest }) => {
const { courseId } = useParams();
const { enabled } = useModel('coursewareSearch', courseId);

if (!enabled) { return null; }

return (
<Button variant="tertiary" size="sm" className="p-1 mt-2 mr-2 rounded-lg" aria-label={intl.formatMessage(messages.searchOpenAction)} data-testid="courseware-search-button" {...rest}>
<Icon src={Search} />
</Button>
);
};

CoursewareSearch.propTypes = {
intl: intlShape.isRequired,
};

export default injectIntl(CoursewareSearch);
49 changes: 49 additions & 0 deletions src/course-home/courseware-search/CoursewareSearch.test.jsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,49 @@
import React from 'react';
import {
initializeMockApp,
render,
screen,
} from '../../setupTest';
import { useModel } from '../../generic/model-store';
import { CoursewareSearch } from './index';

jest.mock('../../generic/model-store', () => ({
useModel: jest.fn(),
}));

function renderComponent() {
const { container } = render(<CoursewareSearch />);
return container;
}

describe('CoursewareSearch', () => {
beforeAll(async () => {
initializeMockApp();
});

beforeEach(() => {
jest.clearAllMocks();
});

describe('When rendered by default', () => {
beforeEach(() => {
useModel.mockReturnValue({ enabled: false });
renderComponent();
});

it('Should not show', () => {
expect(screen.queryByTestId('courseware-search-button')).not.toBeInTheDocument();
});
});

describe('When rendered with the waffle flag enabled', () => {
beforeEach(() => {
useModel.mockReturnValue({ enabled: true });
renderComponent();
});

it('Should show', () => {
expect(screen.getByTestId('courseware-search-button')).toBeInTheDocument();
});
});
});
2 changes: 2 additions & 0 deletions src/course-home/courseware-search/index.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
/* eslint-disable import/prefer-default-export */
export { default as CoursewareSearch } from './CoursewareSearch';
11 changes: 11 additions & 0 deletions src/course-home/courseware-search/messages.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
import { defineMessages } from '@edx/frontend-platform/i18n';

const messages = defineMessages({
searchOpenAction: {
id: 'learn.coursewareSerch.openAction',
defaultMessage: 'Search within this course',
description: 'Aria-label for a button that will pop up Courseware Search.',
},
});

export default messages;
18 changes: 18 additions & 0 deletions src/course-home/data/__snapshots__/redux.test.js.snap
Original file line number Diff line number Diff line change
Expand Up @@ -88,6 +88,12 @@ Object {
},
},
},
"coursewareSearch": Object {
"course-v1:edX+DemoX+Demo_Course": Object {
"enabled": false,
"id": "course-v1:edX+DemoX+Demo_Course",
},
},
"dates": Object {
"course-v1:edX+DemoX+Demo_Course": Object {
"courseDateBlocks": Array [
Expand Down Expand Up @@ -409,6 +415,12 @@ Object {
},
},
},
"coursewareSearch": Object {
"course-v1:edX+DemoX+Demo_Course": Object {
"enabled": true,
"id": "course-v1:edX+DemoX+Demo_Course",
},
},
"outline": Object {
"course-v1:edX+DemoX+Demo_Course": Object {
"accessExpiration": null,
Expand Down Expand Up @@ -608,6 +620,12 @@ Object {
},
},
},
"coursewareSearch": Object {
"course-v1:edX+DemoX+Demo_Course": Object {
"enabled": true,
"id": "course-v1:edX+DemoX+Demo_Course",
},
},
"progress": Object {
"course-v1:edX+DemoX+Demo_Course": Object {
"accessExpiration": null,
Expand Down
6 changes: 6 additions & 0 deletions src/course-home/data/api.js
Original file line number Diff line number Diff line change
Expand Up @@ -445,3 +445,9 @@ export async function unsubscribeFromCourseGoal(token) {
return getAuthenticatedHttpClient().post(url.href)
.then(res => camelCaseObject(res));
}

export async function getCoursewareSearchEnabledFlag(courseId) {
const url = new URL(`${getConfig().LMS_BASE_URL}/courses/${courseId}/courseware-search/enabled/`);
const { data } = await getAuthenticatedHttpClient().get(url.href);
return { enabled: data.enabled || false };
}
11 changes: 11 additions & 0 deletions src/course-home/data/redux.test.js
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@ describe('Data layer integration tests', () => {
const { id: courseId } = courseHomeMetadata;
let courseMetadataUrl = `${getConfig().LMS_BASE_URL}/api/course_home/course_metadata/${courseId}`;
courseMetadataUrl = appendBrowserTimezoneToUrl(courseMetadataUrl);
const courseSearchEnabledUrl = `${getConfig().LMS_BASE_URL}/courses/${courseId}/courseware-search/enabled/`;

const courseHomeAccessDeniedMetadata = Factory.build(
'courseHomeMetadata',
Expand Down Expand Up @@ -48,6 +49,7 @@ describe('Data layer integration tests', () => {
it('Should fail to fetch if error occurs', async () => {
axiosMock.onGet(courseMetadataUrl).networkError();
axiosMock.onGet(`${datesBaseUrl}/${courseId}`).networkError();
axiosMock.onGet(courseSearchEnabledUrl).networkError();

await executeThunk(thunks.fetchDatesTab(courseId), store.dispatch);

Expand All @@ -62,6 +64,7 @@ describe('Data layer integration tests', () => {

axiosMock.onGet(courseMetadataUrl).reply(200, courseHomeMetadata);
axiosMock.onGet(datesUrl).reply(200, datesTabData);
axiosMock.onGet(courseSearchEnabledUrl).reply(200, { enabled: false });

await executeThunk(thunks.fetchDatesTab(courseId), store.dispatch);

Expand All @@ -82,6 +85,7 @@ describe('Data layer integration tests', () => {
async (errorStatus) => {
axiosMock.onGet(courseMetadataUrl).reply(200, courseHomeAccessDeniedMetadata);
axiosMock.onGet(`${datesBaseUrl}/${courseId}`).reply(errorStatus, {});
axiosMock.onGet(courseSearchEnabledUrl).reply(200, { enabled: false });

await executeThunk(thunks.fetchDatesTab(courseId), store.dispatch);

Expand All @@ -101,6 +105,7 @@ describe('Data layer integration tests', () => {
it('Should result in fetch failure if error occurs', async () => {
axiosMock.onGet(courseMetadataUrl).networkError();
axiosMock.onGet(outlineUrl).networkError();
axiosMock.onGet(courseSearchEnabledUrl).networkError();

await executeThunk(thunks.fetchOutlineTab(courseId), store.dispatch);

Expand All @@ -113,6 +118,7 @@ describe('Data layer integration tests', () => {

axiosMock.onGet(courseMetadataUrl).reply(200, courseHomeMetadata);
axiosMock.onGet(outlineUrl).reply(200, outlineTabData);
axiosMock.onGet(courseSearchEnabledUrl).reply(200, { enabled: true });

await executeThunk(thunks.fetchOutlineTab(courseId), store.dispatch);

Expand All @@ -133,6 +139,7 @@ describe('Data layer integration tests', () => {
async (errorStatus) => {
axiosMock.onGet(courseMetadataUrl).reply(200, courseHomeAccessDeniedMetadata);
axiosMock.onGet(outlineUrl).reply(errorStatus, {});
axiosMock.onGet(courseSearchEnabledUrl).reply(200, { enabled: false });

await executeThunk(thunks.fetchOutlineTab(courseId), store.dispatch);

Expand All @@ -151,6 +158,7 @@ describe('Data layer integration tests', () => {
it('Should result in fetch failure if error occurs', async () => {
axiosMock.onGet(courseMetadataUrl).networkError();
axiosMock.onGet(`${progressBaseUrl}/${courseId}`).networkError();
axiosMock.onGet(courseSearchEnabledUrl).networkError();

await executeThunk(thunks.fetchProgressTab(courseId), store.dispatch);

Expand All @@ -165,6 +173,7 @@ describe('Data layer integration tests', () => {

axiosMock.onGet(courseMetadataUrl).reply(200, courseHomeMetadata);
axiosMock.onGet(progressUrl).reply(200, progressTabData);
axiosMock.onGet(courseSearchEnabledUrl).reply(200, { enabled: true });

await executeThunk(thunks.fetchProgressTab(courseId), store.dispatch);

Expand All @@ -187,6 +196,7 @@ describe('Data layer integration tests', () => {

axiosMock.onGet(courseMetadataUrl).reply(200, courseHomeMetadata);
axiosMock.onGet(progressUrl).reply(200, progressTabData);
axiosMock.onGet(courseSearchEnabledUrl).reply(200, { enabled: false });

await executeThunk(thunks.fetchProgressTab(courseId, 2), store.dispatch);

Expand All @@ -200,6 +210,7 @@ describe('Data layer integration tests', () => {
const progressUrl = `${progressBaseUrl}/${courseId}`;
axiosMock.onGet(courseMetadataUrl).reply(200, courseHomeAccessDeniedMetadata);
axiosMock.onGet(progressUrl).reply(errorStatus, {});
axiosMock.onGet(courseSearchEnabledUrl).reply(200, { enabled: false });

await executeThunk(thunks.fetchProgressTab(courseId), store.dispatch);

Expand Down
11 changes: 11 additions & 0 deletions src/course-home/data/thunks.js
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ import {
postDismissWelcomeMessage,
postRequestCert,
getLiveTabIframe,
getCoursewareSearchEnabledFlag,
} from './api';

import {
Expand Down Expand Up @@ -52,6 +53,16 @@ export function fetchTab(courseId, tab, getTabData, targetUserId) {
},
}));
}

const { enabled } = await getCoursewareSearchEnabledFlag(courseId);
dispatch(addModel({
modelType: 'coursewareSearch',
model: {
id: courseId,
enabled,
},
}));

// Disable the access-denied path for now - it caused a regression
if (!courseHomeCourseMetadata.courseAccess.hasAccess) {
dispatch(fetchTabDenied({ courseId }));
Expand Down
4 changes: 4 additions & 0 deletions src/course-tabs/CourseTabsNavigation.jsx
Original file line number Diff line number Diff line change
Expand Up @@ -5,11 +5,15 @@ import classNames from 'classnames';

import messages from './messages';
import Tabs from '../generic/tabs/Tabs';
import { CoursewareSearch } from '../course-home/courseware-search';

const CourseTabsNavigation = ({
activeTabSlug, className, tabs, intl,
}) => (
<div id="courseTabsNavigation" className={classNames('course-tabs-navigation', className)}>
<div className="float-right">
<CoursewareSearch data-testid="courseware-search" />
</div>
<div className="container-xl">
<Tabs
className="nav-underline-tabs"
Expand Down
19 changes: 19 additions & 0 deletions src/course-tabs/CourseTabsNavigation.test.jsx
Original file line number Diff line number Diff line change
@@ -1,9 +1,15 @@
import React from 'react';
import { initializeMockApp, render, screen } from '../setupTest';
import { CourseTabsNavigation } from './index';
import { useModel } from '../generic/model-store';

jest.mock('../generic/model-store', () => ({
useModel: jest.fn(),
}));

describe('Course Tabs Navigation', () => {
beforeAll(async () => {
useModel.mockReturnValue({ enabled: false });
initializeMockApp();
});

Expand All @@ -29,4 +35,17 @@ describe('Course Tabs Navigation', () => {
expect(screen.getByRole('link', { name: tabs[1].title })).toHaveAttribute('href', tabs[1].url);
expect(screen.getByRole('link', { name: tabs[1].title })).not.toHaveClass('active');
});

it('does not render the CoursewareSearch component by default', () => {
render(<CourseTabsNavigation tabs={[]} />);

expect(screen.queryByTestId('courseware-search')).not.toBeInTheDocument();
});

it('renders the CoursewareSearch component if the waffle flag is enabled', () => {
useModel.mockReturnValue({ enabled: true });
render(<CourseTabsNavigation tabs={[]} />);

expect(screen.getByTestId('courseware-search')).toBeInTheDocument();
});
});

0 comments on commit 84cc806

Please sign in to comment.