From 49357a4e872539f215383d89c55256664126ded5 Mon Sep 17 00:00:00 2001 From: "Adolfo R. Brandes" Date: Mon, 15 May 2023 17:48:49 -0300 Subject: [PATCH 1/3] feat: Support runtime configuration frontend-platform supports runtime configuration since 2.5.0 (see the PR that introduced it[1], but it requires MFE cooperation. This implements just that: by avoiding making configuration values constant, it should now be possible to change them after initialization. Almost all changes here relate to the `LMS_BASE_URL` setting, which in most places was treated as a constant. [1] openedx/frontend-platform#335 --- .../ListView/ListViewBreadcrumb.jsx | 2 +- .../ListView/ListViewBreadcrumb.test.jsx | 2 +- src/data/constants/app.js | 4 +-- src/data/constants/app.test.js | 4 +-- src/data/redux/thunkActions/app.js | 2 +- src/data/redux/thunkActions/app.test.js | 2 +- src/data/redux/thunkActions/download.js | 2 +- src/data/services/lms/api.js | 32 +++++++++---------- src/data/services/lms/api.test.js | 22 ++++++------- src/data/services/lms/urls.js | 24 +++++++------- src/setupTest.js | 2 +- 11 files changed, 49 insertions(+), 49 deletions(-) diff --git a/src/containers/ListView/ListViewBreadcrumb.jsx b/src/containers/ListView/ListViewBreadcrumb.jsx index f6b33151f..6202f2a13 100644 --- a/src/containers/ListView/ListViewBreadcrumb.jsx +++ b/src/containers/ListView/ListViewBreadcrumb.jsx @@ -22,7 +22,7 @@ export const ListViewBreadcrumb = ({ courseId, oraName }) => (

{oraName} - +

diff --git a/src/containers/ListView/ListViewBreadcrumb.test.jsx b/src/containers/ListView/ListViewBreadcrumb.test.jsx index 02c45b82c..7547420ef 100644 --- a/src/containers/ListView/ListViewBreadcrumb.test.jsx +++ b/src/containers/ListView/ListViewBreadcrumb.test.jsx @@ -50,7 +50,7 @@ describe('ListViewBreadcrumb component', () => { test('ora destination', () => { expect( el.find(Hyperlink).at(1).props().destination, - ).toEqual(urls.ora(props.courseId, constants.locationId)); + ).toEqual(urls.ora(props.courseId, constants.locationId())); }); }); describe('mapStateToProps', () => { diff --git a/src/data/constants/app.js b/src/data/constants/app.js index dda06d5f9..58e11be65 100644 --- a/src/data/constants/app.js +++ b/src/data/constants/app.js @@ -1,4 +1,4 @@ import { getConfig } from '@edx/frontend-platform'; -export const routePath = `${getConfig().PUBLIC_PATH}:courseId`; -export const locationId = window.location.pathname.replace(getConfig().PUBLIC_PATH, ''); +export const routePath = () => `${getConfig().PUBLIC_PATH}:courseId`; +export const locationId = () => window.location.pathname.replace(getConfig().PUBLIC_PATH, ''); diff --git a/src/data/constants/app.test.js b/src/data/constants/app.test.js index 17fb772bc..b36a0e47d 100644 --- a/src/data/constants/app.test.js +++ b/src/data/constants/app.test.js @@ -13,12 +13,12 @@ jest.mock('@edx/frontend-platform', () => { describe('app constants', () => { test('route path draws from public path and adds courseId', () => { - expect(constants.routePath).toEqual(`${platform.PUBLIC_PATH}:courseId`); + expect(constants.routePath()).toEqual(`${platform.PUBLIC_PATH}:courseId`); }); test('locationId returns trimmed pathname', () => { const old = window.location; window.location = { pathName: `${platform.PUBLIC_PATH}somePath.jpg` }; - expect(constants.locationId).toEqual(window.location.pathname.replace(platform.PUBLIC_PATH, '')); + expect(constants.locationId()).toEqual(window.location.pathname.replace(platform.PUBLIC_PATH, '')); window.location = old; }); }); diff --git a/src/data/redux/thunkActions/app.js b/src/data/redux/thunkActions/app.js index 2bc691fae..00b9a636e 100644 --- a/src/data/redux/thunkActions/app.js +++ b/src/data/redux/thunkActions/app.js @@ -15,7 +15,7 @@ import * as module from './app'; */ export const initialize = () => (dispatch) => { dispatch(initializeApp({ - locationId, + locationId: locationId(), onSuccess: (response) => { dispatch(actions.app.loadIsEnabled(response.isEnabled)); dispatch(actions.app.loadOraMetadata(response.oraMetadata)); diff --git a/src/data/redux/thunkActions/app.test.js b/src/data/redux/thunkActions/app.test.js index 18765bd72..08b16b80c 100644 --- a/src/data/redux/thunkActions/app.test.js +++ b/src/data/redux/thunkActions/app.test.js @@ -25,7 +25,7 @@ describe('app thunkActions', () => { [[dispatchedAction]] = dispatch.mock.calls; }); it('dispatches initializeApp with locationId and onSuccess', () => { - expect(dispatchedAction.initializeApp.locationId).toEqual(locationId); + expect(dispatchedAction.initializeApp.locationId).toEqual(locationId()); expect(typeof dispatchedAction.initializeApp.onSuccess).toEqual('function'); }); describe('on success', () => { diff --git a/src/data/redux/thunkActions/download.js b/src/data/redux/thunkActions/download.js index 15b2f5b41..d24a96713 100644 --- a/src/data/redux/thunkActions/download.js +++ b/src/data/redux/thunkActions/download.js @@ -47,7 +47,7 @@ export const zipFiles = async (files, blobs, username) => { } const zipFile = await zipWriter.close(); - const zipName = `${username}-${locationId}.zip`; + const zipName = `${username}-${locationId()}.zip`; FileSaver.saveAs(zipFile, zipName); }; diff --git a/src/data/services/lms/api.js b/src/data/services/lms/api.js index 994095dcb..425ceab0e 100644 --- a/src/data/services/lms/api.js +++ b/src/data/services/lms/api.js @@ -32,8 +32,8 @@ import { * } */ const initializeApp = () => get( - stringifyUrl(urls.oraInitializeUrl, { - [paramKeys.oraLocation]: locationId, + stringifyUrl(urls.oraInitializeUrl(), { + [paramKeys.oraLocation]: locationId(), }), ).then(response => response.data); @@ -48,8 +48,8 @@ const initializeApp = () => get( * } */ const fetchSubmission = (submissionUUID) => get( - stringifyUrl(urls.fetchSubmissionUrl, { - [paramKeys.oraLocation]: locationId, + stringifyUrl(urls.fetchSubmissionUrl(), { + [paramKeys.oraLocation]: locationId(), [paramKeys.submissionUUID]: submissionUUID, }), ).then(response => response.data); @@ -61,8 +61,8 @@ const fetchSubmission = (submissionUUID) => get( * } */ const fetchSubmissionFiles = (submissionUUID) => get( - stringifyUrl(urls.fetchSubmissionFilesUrl, { - [paramKeys.oraLocation]: locationId, + stringifyUrl(urls.fetchSubmissionFilesUrl(), { + [paramKeys.oraLocation]: locationId(), [paramKeys.submissionUUID]: submissionUUID, }), ).then(response => response.data); @@ -78,8 +78,8 @@ const fetchSubmissionFiles = (submissionUUID) => get( * } */ const fetchSubmissionStatus = (submissionUUID) => get( - stringifyUrl(urls.fetchSubmissionStatusUrl, { - [paramKeys.oraLocation]: locationId, + stringifyUrl(urls.fetchSubmissionStatusUrl(), { + [paramKeys.oraLocation]: locationId(), [paramKeys.submissionUUID]: submissionUUID, }), ).then(response => response.data); @@ -89,8 +89,8 @@ const fetchSubmissionStatus = (submissionUUID) => get( * @param {string} submissionUUID */ const lockSubmission = (submissionUUID) => post( - stringifyUrl(urls.fetchSubmissionLockUrl, { - [paramKeys.oraLocation]: locationId, + stringifyUrl(urls.fetchSubmissionLockUrl(), { + [paramKeys.oraLocation]: locationId(), [paramKeys.submissionUUID]: submissionUUID, }), ).then(response => response.data); @@ -100,8 +100,8 @@ const lockSubmission = (submissionUUID) => post( * @param {string} submissionUUID */ const unlockSubmission = (submissionUUID) => client().delete( - stringifyUrl(urls.fetchSubmissionLockUrl, { - [paramKeys.oraLocation]: locationId, + stringifyUrl(urls.fetchSubmissionLockUrl(), { + [paramKeys.oraLocation]: locationId(), [paramKeys.submissionUUID]: submissionUUID, }), ).then(response => response.data); @@ -112,8 +112,8 @@ const unlockSubmission = (submissionUUID) => client().delete( */ const batchUnlockSubmissions = (submissionUUIDs) => post( stringifyUrl( - urls.batchUnlockSubmissionsUrl, - { [paramKeys.oraLocation]: locationId }, + urls.batchUnlockSubmissionsUrl(), + { [paramKeys.oraLocation]: locationId() }, ), { submissionUUIDs }, ).then(response => response.data); @@ -123,8 +123,8 @@ const batchUnlockSubmissions = (submissionUUIDs) => post( * @param {object} gradeData - full grading submission data */ const updateGrade = (submissionUUID, gradeData) => post( - stringifyUrl(urls.updateSubmissionGradeUrl, { - [paramKeys.oraLocation]: locationId, + stringifyUrl(urls.updateSubmissionGradeUrl(), { + [paramKeys.oraLocation]: locationId(), [paramKeys.submissionUUID]: submissionUUID, }), gradeData, diff --git a/src/data/services/lms/api.test.js b/src/data/services/lms/api.test.js index 89f85623e..ce2dbb252 100644 --- a/src/data/services/lms/api.test.js +++ b/src/data/services/lms/api.test.js @@ -13,7 +13,7 @@ jest.mock('./utils', () => ({ })); jest.mock('data/constants/app', () => ({ - locationId: 'test-location-id', + locationId: () => 'test-location-id', })); const gradeData = 'test-grade-data'; @@ -37,10 +37,10 @@ const testAPI = ({ ...otherExpected }, }) => { - it(`returns ${method}(${urlKey}) with correct args and reoslves with response data`, () => ( + it(`returns ${method}(${urlKey}) with correct args and resolves with response data`, () => ( promise.then((data) => { expect(data[method]).toEqual({ - url: stringifyUrl(urls[urlKey], urlParams), + url: stringifyUrl(urls[urlKey](), urlParams), ...otherExpected, }); }) @@ -54,7 +54,7 @@ describe('lms service api methods', () => { method: methodKeys.get, expected: { urlKey: urlKeys.oraInitializeUrl, - urlParams: { [paramKeys.oraLocation]: locationId }, + urlParams: { [paramKeys.oraLocation]: locationId() }, }, }); }); @@ -65,7 +65,7 @@ describe('lms service api methods', () => { expected: { urlKey: urlKeys.fetchSubmissionUrl, urlParams: { - [paramKeys.oraLocation]: locationId, + [paramKeys.oraLocation]: locationId(), [paramKeys.submissionUUID]: submissionUUID, }, }, @@ -78,7 +78,7 @@ describe('lms service api methods', () => { expected: { urlKey: urlKeys.fetchSubmissionFilesUrl, urlParams: { - [paramKeys.oraLocation]: locationId, + [paramKeys.oraLocation]: locationId(), [paramKeys.submissionUUID]: submissionUUID, }, }, @@ -91,7 +91,7 @@ describe('lms service api methods', () => { expected: { urlKey: urlKeys.fetchSubmissionStatusUrl, urlParams: { - [paramKeys.oraLocation]: locationId, + [paramKeys.oraLocation]: locationId(), [paramKeys.submissionUUID]: submissionUUID, }, }, @@ -104,7 +104,7 @@ describe('lms service api methods', () => { expected: { urlKey: urlKeys.fetchSubmissionLockUrl, urlParams: { - [paramKeys.oraLocation]: locationId, + [paramKeys.oraLocation]: locationId(), [paramKeys.submissionUUID]: submissionUUID, }, }, @@ -117,7 +117,7 @@ describe('lms service api methods', () => { expected: { urlKey: urlKeys.fetchSubmissionLockUrl, urlParams: { - [paramKeys.oraLocation]: locationId, + [paramKeys.oraLocation]: locationId(), [paramKeys.submissionUUID]: submissionUUID, }, }, @@ -130,7 +130,7 @@ describe('lms service api methods', () => { expected: { urlKey: urlKeys.batchUnlockSubmissionsUrl, urlParams: { - [paramKeys.oraLocation]: locationId, + [paramKeys.oraLocation]: locationId(), }, data: { submissionUUIDs }, }, @@ -143,7 +143,7 @@ describe('lms service api methods', () => { expected: { urlKey: urlKeys.updateSubmissionGradeUrl, urlParams: { - [paramKeys.oraLocation]: locationId, + [paramKeys.oraLocation]: locationId(), [paramKeys.submissionUUID]: submissionUUID, }, data: gradeData, diff --git a/src/data/services/lms/urls.js b/src/data/services/lms/urls.js index f4ec13811..de95808bf 100644 --- a/src/data/services/lms/urls.js +++ b/src/data/services/lms/urls.js @@ -1,20 +1,20 @@ import { StrictDict } from 'utils'; -import { configuration } from 'config'; +import { getConfig } from '@edx/frontend-platform'; -const baseUrl = `${configuration.LMS_BASE_URL}`; +const baseUrl = () => getConfig().LMS_BASE_URL; -const api = `${baseUrl}/api/`; -const baseEsgUrl = `${api}ora_staff_grader/`; +const api = () => `${baseUrl()}/api/`; +const baseEsgUrl = () => `${api()}ora_staff_grader/`; -const oraInitializeUrl = `${baseEsgUrl}initialize`; -const fetchSubmissionUrl = `${baseEsgUrl}submission`; -const fetchSubmissionFilesUrl = `${baseEsgUrl}submission/files`; -const fetchSubmissionStatusUrl = `${baseEsgUrl}submission/status`; -const fetchSubmissionLockUrl = `${baseEsgUrl}submission/lock`; -const batchUnlockSubmissionsUrl = `${baseEsgUrl}submission/batch/unlock`; -const updateSubmissionGradeUrl = `${baseEsgUrl}submission/grade`; +const oraInitializeUrl = () => `${baseEsgUrl()}initialize`; +const fetchSubmissionUrl = () => `${baseEsgUrl()}submission`; +const fetchSubmissionFilesUrl = () => `${baseEsgUrl()}submission/files`; +const fetchSubmissionStatusUrl = () => `${baseEsgUrl()}submission/status`; +const fetchSubmissionLockUrl = () => `${baseEsgUrl()}submission/lock`; +const batchUnlockSubmissionsUrl = () => `${baseEsgUrl()}submission/batch/unlock`; +const updateSubmissionGradeUrl = () => `${baseEsgUrl()}submission/grade`; -const course = (courseId) => `${baseUrl}/courses/${courseId}`; +const course = (courseId) => `${baseUrl()}/courses/${courseId}`; const openResponse = (courseId) => ( `${course(courseId)}/instructor#view-open_response_assessment` diff --git a/src/setupTest.js b/src/setupTest.js index 914910bd1..d2f525bc1 100755 --- a/src/setupTest.js +++ b/src/setupTest.js @@ -113,7 +113,7 @@ jest.mock('@edx/paragon/icons', () => ({ })); jest.mock('data/constants/app', () => ({ - locationId: 'fake-location-id', + locationId: () => 'fake-location-id', })); jest.mock('hooks', () => ({ From 0bb5f50917151bd782a5f58f054d6940d4319221 Mon Sep 17 00:00:00 2001 From: "Adolfo R. Brandes" Date: Wed, 31 May 2023 12:32:09 -0300 Subject: [PATCH 2/3] refactor: use getConfig --- src/config/index.js | 16 ---------------- src/segment.js | 4 ++-- 2 files changed, 2 insertions(+), 18 deletions(-) delete mode 100644 src/config/index.js diff --git a/src/config/index.js b/src/config/index.js deleted file mode 100644 index 94e3119f3..000000000 --- a/src/config/index.js +++ /dev/null @@ -1,16 +0,0 @@ -const configuration = { - // BASE_URL: process.env.BASE_URL, - LMS_BASE_URL: process.env.LMS_BASE_URL, - // LOGIN_URL: process.env.LOGIN_URL, - // LOGOUT_URL: process.env.LOGOUT_URL, - // CSRF_TOKEN_API_PATH: process.env.CSRF_TOKEN_API_PATH, - // REFRESH_ACCESS_TOKEN_ENDPOINT: process.env.REFRESH_ACCESS_TOKEN_ENDPOINT, - // DATA_API_BASE_URL: process.env.DATA_API_BASE_URL, - // SECURE_COOKIES: process.env.NODE_ENV !== 'development', - // SEGMENT_KEY: process.env.SEGMENT_KEY, - // ACCESS_TOKEN_COOKIE_NAME: process.env.ACCESS_TOKEN_COOKIE_NAME, -}; - -const features = {}; - -export { configuration, features }; diff --git a/src/segment.js b/src/segment.js index c2eb562a7..309b20b0d 100644 --- a/src/segment.js +++ b/src/segment.js @@ -1,6 +1,6 @@ // The code in this file is from Segment's website: // https://segment.com/docs/sources/website/analytics.js/quickstart/ -import { configuration } from './config'; +import { getConfig } from '@edx/frontend-platform'; (function () { // Create a queue, but don't obliterate an existing one! @@ -81,5 +81,5 @@ import { configuration } from './config'; // Load Analytics.js with your key, which will automatically // load the tools you've enabled for your account. Boosh! - analytics.load(configuration.SEGMENT_KEY); + analytics.load(getConfig().SEGMENT_KEY); }()); From a07d6f9b8072d40f1076f28ff2dff33b47913d24 Mon Sep 17 00:00:00 2001 From: Leangseu Kim Date: Mon, 5 Jun 2023 12:43:30 -0400 Subject: [PATCH 3/3] fix: when there is no options, the validation should be valid --- src/data/redux/grading/selectors/validation.js | 2 +- src/data/redux/grading/selectors/validation.test.js | 12 ++++++++++-- 2 files changed, 11 insertions(+), 3 deletions(-) diff --git a/src/data/redux/grading/selectors/validation.js b/src/data/redux/grading/selectors/validation.js index 6ed033e09..db6603ff7 100644 --- a/src/data/redux/grading/selectors/validation.js +++ b/src/data/redux/grading/selectors/validation.js @@ -32,7 +32,7 @@ validation.criteria = createSelector( criterion.feedback === feedbackRequirement.required && gradingData.criteria[index].feedback === '' ), - selectedOption: gradingData.criteria[index].selectedOption !== '', + selectedOption: rubricConfig.criteria[index].options.length === 0 || gradingData.criteria[index].selectedOption !== '', })), ); diff --git a/src/data/redux/grading/selectors/validation.test.js b/src/data/redux/grading/selectors/validation.test.js index 1b4c4984b..80526c883 100644 --- a/src/data/redux/grading/selectors/validation.test.js +++ b/src/data/redux/grading/selectors/validation.test.js @@ -64,8 +64,8 @@ describe('validation grading selectors unit tests', () => { expect(preSelectors).toEqual([selected.gradingData, appSelectors.rubric.config]); }); describe('returned object', () => { - const feedbackOptional = { feedback: feedbackRequirement.optional }; - const feedbackRequired = { feedback: feedbackRequirement.required }; + const feedbackOptional = { feedback: feedbackRequirement.optional, options: [1] }; + const feedbackRequired = { feedback: feedbackRequirement.required, options: [1] }; describe('feedback', () => { const validFeedback = { feedback: 'Fair' }; const emptyFeedback = { feedback: '' }; @@ -85,6 +85,14 @@ describe('validation grading selectors unit tests', () => { ); expect(output.map(({ selectedOption }) => selectedOption)).toEqual([true, false]); }); + it('returns true criteria options are empty', () => { + const emptyOptionsFeedback = { ...feedbackOptional, options: [] }; + const output = cb( + { criteria: [{ selectedOption: '' }, { selectedOption: 'Invalid' }] }, + { criteria: [emptyOptionsFeedback, emptyOptionsFeedback] }, + ); + expect(output.map(({ selectedOption }) => selectedOption)).toEqual([true, true]); + }); }); }); });