studiokit-scaffolding-js 7.0.12-next.1.3 → 7.0.12-next.2.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/lib/components/ActionList.js +164 -37
- package/lib/components/AlertDialog.js +128 -12
- package/lib/components/AlertWithIcon.js +88 -29
- package/lib/components/ConnectedModal.js +35 -12
- package/lib/components/Dropdowns/GroupsDropdown.js +63 -45
- package/lib/components/Dropdowns/ManagedNavDropdown.js +92 -67
- package/lib/components/Dropdowns/UserDropdown.js +105 -24
- package/lib/components/Dropdowns/index.js +4 -10
- package/lib/components/EntityOwnerList.js +47 -21
- package/lib/components/Error.js +101 -12
- package/lib/components/ErrorBoundary.js +127 -38
- package/lib/components/ErrorMessage.js +39 -12
- package/lib/components/Forms/DateField.js +56 -45
- package/lib/components/Forms/TimeField.js +76 -45
- package/lib/components/Forms/index.js +3 -5
- package/lib/components/Groups/CreateEditCopySaveButtons.js +109 -14
- package/lib/components/Groups/ExternalGroups/Attach.js +206 -151
- package/lib/components/Groups/ExternalGroups/Table.js +176 -48
- package/lib/components/Groups/GroupCreateOrEditCommonProps.js +2 -2
- package/lib/components/Groups/RosterSyncInfo.js +142 -23
- package/lib/components/HOC/AccessibleAppComponent.js +88 -72
- package/lib/components/HOC/ActivityRequiredComponent.js +68 -33
- package/lib/components/HOC/AsyncComponent.js +49 -41
- package/lib/components/HOC/AuthenticatedComponent.js +55 -44
- package/lib/components/HOC/CollectionComponent.js +154 -104
- package/lib/components/HOC/CollectionFirstItemComponent.js +45 -40
- package/lib/components/HOC/CollectionItemComponent.js +152 -100
- package/lib/components/HOC/ConnectedModalComponent.js +87 -69
- package/lib/components/HOC/DataDependentComponent.js +26 -27
- package/lib/components/HOC/EntityComponent.js +57 -53
- package/lib/components/HOC/FullscreenModalComponent.js +139 -108
- package/lib/components/HOC/GroupActivityRequiredComponent.js +27 -20
- package/lib/components/HOC/GuidComponent.js +20 -20
- package/lib/components/HOC/ModelContextDependencyVerifyComponent.js +32 -29
- package/lib/components/HOC/ModelErrorRedirectComponent.js +37 -39
- package/lib/components/HOC/SearchPersistorComponent.js +237 -173
- package/lib/components/HOC/UnauthenticatedComponent.js +32 -30
- package/lib/components/HOC/UserComponent.js +6 -8
- package/lib/components/Icons/IconAlphaList.js +28 -8
- package/lib/components/Icons/IconExternalUser.js +28 -8
- package/lib/components/Icons/IconImpersonation.js +28 -8
- package/lib/components/Icons/IconStopImpersonating.js +28 -8
- package/lib/components/Icons/IconTable.js +29 -9
- package/lib/components/Icons/IconTableDeleteCol.js +28 -8
- package/lib/components/Icons/IconTableDeleteRow.js +28 -8
- package/lib/components/Icons/IconTableInsertCol.js +28 -8
- package/lib/components/Icons/IconTableInsertRow.js +28 -8
- package/lib/components/Impersonation/Button.js +71 -16
- package/lib/components/Impersonation/Link.js +72 -16
- package/lib/components/Impersonation/UserDetail.js +60 -11
- package/lib/components/Loading.js +23 -8
- package/lib/components/LockDownBrowser/Check.js +188 -51
- package/lib/components/LockDownBrowser/ExitButton.js +22 -13
- package/lib/components/LockDownBrowser/Launch.js +64 -64
- package/lib/components/Lti/Confirm.js +147 -14
- package/lib/components/Lti/CreateNonLtiGroupAlertDialog.js +165 -36
- package/lib/components/Lti/Launch.js +99 -25
- package/lib/components/Lti/LaunchGroup.js +81 -16
- package/lib/components/ManageTable.js +304 -90
- package/lib/components/ManageTableNoDataComponent.js +38 -7
- package/lib/components/NewVersionAlert.js +76 -49
- package/lib/components/NotFound.js +81 -11
- package/lib/components/Notifications.js +179 -129
- package/lib/components/PaginationNextButton.js +28 -9
- package/lib/components/PaginationPreviousButton.js +28 -9
- package/lib/components/Quill/CustomToolbar.js +427 -222
- package/lib/components/Quill/Formats/Image.js +67 -67
- package/lib/components/Quill/Formats/List.js +38 -47
- package/lib/components/Quill/Formats/Video.js +23 -26
- package/lib/components/Quill/ImageDropModule.js +136 -114
- package/lib/components/Quill/ImageWarning.js +41 -12
- package/lib/components/Quill/ImageWithAltTextModal.js +420 -89
- package/lib/components/Quill/Specs/CustomImageSpec.js +32 -31
- package/lib/components/Quill/Specs/CustomVideoSpec.js +22 -23
- package/lib/components/Quill/TableModule/Blots/BaseTableBlot.js +89 -97
- package/lib/components/Quill/TableModule/Blots/TableBlot.js +47 -50
- package/lib/components/Quill/TableModule/Blots/TableBodyBlot.js +48 -51
- package/lib/components/Quill/TableModule/Blots/TableCellBlot.js +219 -224
- package/lib/components/Quill/TableModule/Blots/TableContainer.js +75 -86
- package/lib/components/Quill/TableModule/Blots/TableRowBlot.js +70 -73
- package/lib/components/Quill/TableModule/constants.js +40 -42
- package/lib/components/Quill/TableModule/index.js +357 -305
- package/lib/components/Quill/TableModule/utils.js +39 -48
- package/lib/components/Quill/accessibilityFix.js +219 -223
- package/lib/components/Quill/index.js +30 -33
- package/lib/components/RefreshIndicator/Bordered.js +44 -10
- package/lib/components/RefreshIndicator/Inline.js +43 -12
- package/lib/components/RefreshIndicator/index.js +257 -62
- package/lib/components/SearchControls.js +211 -14
- package/lib/components/SentryRoute.js +5 -7
- package/lib/components/Tables/RoleFilter.js +66 -38
- package/lib/components/Tables/TextFilter.js +58 -18
- package/lib/components/UserRoles/Add.js +193 -99
- package/lib/components/UserRoles/Context.js +3 -6
- package/lib/components/UserRoles/RoleCell.js +176 -75
- package/lib/components/UserRoles/Select.js +151 -20
- package/lib/components/UserRoles/Table.js +215 -82
- package/lib/components/UserRoles/index.js +526 -386
- package/lib/config/eslint/index.js +26 -29
- package/lib/config/eslint/lib/order.js +21 -28
- package/lib/config/eslint/lib/prettier.js +15 -19
- package/lib/config/eslint/lib/typescript.js +87 -113
- package/lib/config/eslint/react.js +18 -15
- package/lib/constants/baseActivity.js +26 -28
- package/lib/constants/baseRole.js +10 -12
- package/lib/constants/configuration.js +43 -55
- package/lib/constants/externalProviderType.js +6 -8
- package/lib/constants/fetchErrorData.js +10 -12
- package/lib/constants/index.js +13 -15
- package/lib/constants/lockDownBrowser.js +23 -25
- package/lib/constants/mockData.js +370 -300
- package/lib/constants/modelStatus.js +11 -13
- package/lib/constants/notificationType.js +8 -10
- package/lib/constants/operatingSystem.js +8 -10
- package/lib/constants/shard.js +7 -9
- package/lib/constants/table.js +18 -22
- package/lib/constants/tier.js +8 -10
- package/lib/constants/userRole.js +11 -8
- package/lib/endpointMappings.js +191 -182
- package/lib/hooks/useCollection.js +79 -65
- package/lib/hooks/useCollectionConfiguration.js +220 -80
- package/lib/hooks/useCollectionItem.js +151 -57
- package/lib/hooks/useGuid.js +16 -9
- package/lib/hooks/usePrevious.js +14 -13
- package/lib/index.js +11 -26
- package/lib/redux/actionCreator.js +44 -35
- package/lib/redux/actions/AuthAction.js +45 -32
- package/lib/redux/actions/ModalAction.js +6 -8
- package/lib/redux/actions/ModelAction.js +95 -43
- package/lib/redux/actions/NotificationAction.js +6 -8
- package/lib/redux/actions/SearchAction.js +5 -7
- package/lib/redux/actions/index.js +6 -8
- package/lib/redux/configureReducers.js +48 -46
- package/lib/redux/configureStore.js +77 -91
- package/lib/redux/helpers.js +2 -5
- package/lib/redux/reducers/authReducer.js +44 -43
- package/lib/redux/reducers/index.js +7 -14
- package/lib/redux/reducers/modalsReducer.js +43 -31
- package/lib/redux/reducers/modelsReducer.js +131 -137
- package/lib/redux/reducers/notificationsReducer.js +20 -20
- package/lib/redux/reducers/searchReducer.js +13 -13
- package/lib/redux/sagas/appInsightsSaga.js +19 -21
- package/lib/redux/sagas/authSaga.js +248 -234
- package/lib/redux/sagas/caliperSaga.js +142 -131
- package/lib/redux/sagas/clockOffsetSaga.js +29 -32
- package/lib/redux/sagas/configurationSaga.js +8 -10
- package/lib/redux/sagas/downtimeApiErrorSaga.js +16 -19
- package/lib/redux/sagas/errorSaga.js +23 -24
- package/lib/redux/sagas/googleAnalyticsSaga.js +24 -27
- package/lib/redux/sagas/identityProviderSaga.js +19 -21
- package/lib/redux/sagas/initialDataLoadSaga.js +34 -31
- package/lib/redux/sagas/lockDownBrowserErrorSaga.js +25 -22
- package/lib/redux/sagas/modelFetchSaga.js +302 -286
- package/lib/redux/sagas/noStoreSaga.js +60 -61
- package/lib/redux/sagas/postLoginDataSaga.js +37 -32
- package/lib/redux/sagas/postLoginRedirectSaga.js +22 -27
- package/lib/redux/sagas/rootSaga.js +77 -60
- package/lib/redux/sagas/sentrySaga.js +25 -28
- package/lib/redux/sagas/userIdSaga.js +13 -15
- package/lib/services/codeProviderService.js +21 -21
- package/lib/services/dateService.js +6 -8
- package/lib/services/documentService.js +10 -11
- package/lib/services/fetchService.js +103 -95
- package/lib/services/persistenceService.js +27 -30
- package/lib/services/ticketProviderService.js +25 -25
- package/lib/services/tokenPersistenceService.js +8 -10
- package/lib/services/windowService.js +14 -16
- package/lib/startup.js +110 -101
- package/lib/types/AppConfiguration.js +2 -2
- package/lib/types/Artifact.js +7 -9
- package/lib/types/BaseReduxState.js +2 -2
- package/lib/types/Client.js +2 -2
- package/lib/types/Collection.js +2 -2
- package/lib/types/Configuration.js +2 -2
- package/lib/types/DeepLinkingResponseRequest.js +2 -2
- package/lib/types/DeletableModel.js +2 -2
- package/lib/types/Event.js +2 -2
- package/lib/types/ExternalGroup.js +2 -2
- package/lib/types/ExternalProvider.js +2 -2
- package/lib/types/ExternalTerm.js +2 -2
- package/lib/types/Group.js +2 -2
- package/lib/types/IdentityProvider.js +2 -2
- package/lib/types/LtiLaunch.js +2 -2
- package/lib/types/NameOnlyEntity.js +2 -2
- package/lib/types/Notification.js +2 -2
- package/lib/types/OptionalRecord.js +2 -2
- package/lib/types/OwnerSchedule.js +2 -2
- package/lib/types/PropertyOfType.js +2 -2
- package/lib/types/Quill.js +2 -2
- package/lib/types/RoleDescription.js +2 -2
- package/lib/types/Search.js +2 -2
- package/lib/types/SimpleLocation.js +2 -2
- package/lib/types/UniTime.js +2 -2
- package/lib/types/User.js +2 -2
- package/lib/types/UserRole.js +2 -2
- package/lib/types/auth/AuthState.js +2 -2
- package/lib/types/auth/CasV1LoginRequestBody.js +2 -2
- package/lib/types/auth/ClientCredentials.js +2 -2
- package/lib/types/auth/CodeProviderService.js +2 -2
- package/lib/types/auth/LocalLoginRequestBody.js +2 -2
- package/lib/types/auth/TicketProviderService.js +2 -2
- package/lib/types/auth/TokenPersistenceService.js +2 -2
- package/lib/types/auth/index.js +8 -10
- package/lib/types/externals.d.js +2 -0
- package/lib/types/index.js +29 -31
- package/lib/types/net/EndpointConfig.js +2 -2
- package/lib/types/net/EndpointMapping.js +2 -2
- package/lib/types/net/EndpointMappings.js +2 -2
- package/lib/types/net/ErrorHandler.js +2 -2
- package/lib/types/net/FetchConfig.js +2 -2
- package/lib/types/net/FetchErrorData.js +6 -8
- package/lib/types/net/FetchResult.js +2 -2
- package/lib/types/net/HTTPMethod.js +2 -2
- package/lib/types/net/HTTPStatusCode.js +12 -14
- package/lib/types/net/Metadata.js +2 -2
- package/lib/types/net/Model.js +2 -2
- package/lib/types/net/ModelCollection.js +2 -2
- package/lib/types/net/ModelsState.js +2 -2
- package/lib/types/net/OAuthToken.js +2 -2
- package/lib/types/net/OAuthTokenOrNull.js +2 -2
- package/lib/types/net/TokenAccessFunction.js +2 -2
- package/lib/types/net/index.js +17 -19
- package/lib/utils/baseActivity.js +83 -85
- package/lib/utils/baseRole.js +32 -36
- package/lib/utils/collection.js +403 -297
- package/lib/utils/cookies.js +19 -23
- package/lib/utils/date.js +188 -205
- package/lib/utils/dom.js +130 -131
- package/lib/utils/domainIdentifier.js +4 -8
- package/lib/utils/entityUserRole.js +2 -5
- package/lib/utils/error.js +14 -19
- package/lib/utils/events.js +32 -31
- package/lib/utils/externalGroup.js +20 -25
- package/lib/utils/externalProviders.js +4 -7
- package/lib/utils/externalTerms.js +6 -6
- package/lib/utils/fetch.js +168 -176
- package/lib/utils/group.js +14 -11
- package/lib/utils/groupDates.js +38 -46
- package/lib/utils/groupRoles.js +23 -32
- package/lib/utils/lockDownBrowser.js +12 -15
- package/lib/utils/logger.js +23 -28
- package/lib/utils/lti.js +4 -7
- package/lib/utils/model.js +28 -43
- package/lib/utils/number.js +9 -13
- package/lib/utils/promise.js +23 -26
- package/lib/utils/quill.js +55 -60
- package/lib/utils/route.js +52 -60
- package/lib/utils/search.js +72 -87
- package/lib/utils/shard.js +33 -42
- package/lib/utils/sort.js +47 -50
- package/lib/utils/string.js +10 -12
- package/lib/utils/table.js +29 -33
- package/lib/utils/timezone.js +7 -12
- package/lib/utils/url.js +130 -144
- package/lib/utils/user.js +54 -64
- package/lib/utils/userAgent.js +7 -14
- package/lib/utils/userRole.js +36 -39
- package/package.json +17 -3
|
@@ -1,342 +1,353 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
const dateService_1 = require("../../services/dateService");
|
|
15
|
-
const fetchService_1 = require("../../services/fetchService");
|
|
16
|
-
const windowService_1 = require("../../services/windowService");
|
|
17
|
-
const types_1 = require("../../types");
|
|
18
|
-
const error_1 = require("../../utils/error");
|
|
19
|
-
const fetch_1 = require("../../utils/fetch");
|
|
20
|
-
const logger_1 = require("../../utils/logger");
|
|
21
|
-
const actions_1 = require("../actions");
|
|
1
|
+
import { isNil, isPlainObject, merge } from 'lodash';
|
|
2
|
+
import moment from 'moment-timezone';
|
|
3
|
+
import { call, cancel, cancelled, delay, fork, put, take, takeEvery } from 'redux-saga/effects';
|
|
4
|
+
import { networkOfflineErrorData, unknownErrorData } from '../../constants/fetchErrorData';
|
|
5
|
+
import { getLastActivityDate } from '../../services/dateService';
|
|
6
|
+
import { sendFetch, setApiRoot } from '../../services/fetchService';
|
|
7
|
+
import { windowService } from '../../services/windowService';
|
|
8
|
+
import { HTTP_STATUS_CODE } from '../../types';
|
|
9
|
+
import { constructErrorFromCaughtError } from '../../utils/error';
|
|
10
|
+
import { isFetchErrorData, prepareFetch } from '../../utils/fetch';
|
|
11
|
+
import { getLogger } from '../../utils/logger';
|
|
12
|
+
import { MODEL_FETCH_ERROR_ACTION_TYPE, MODEL_FETCH_REQUEST_ACTION_TYPE, MODEL_FETCH_RESULT_ACTION_TYPE, MODEL_FETCH_START_ACTION_TYPE, MODEL_REMOVE_KEY_ACTION_TYPE, PERIODIC_MODEL_FETCH_TERMINATION_ACTION_TYPE } from '../actions';
|
|
13
|
+
|
|
22
14
|
/** The total number of tries for `fetchData`, including the initial request. */
|
|
23
|
-
|
|
15
|
+
export const TRY_LIMIT = 5;
|
|
16
|
+
|
|
24
17
|
//#region Helpers
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
exports.takeMatchesTerminationAction = takeMatchesTerminationAction;
|
|
18
|
+
|
|
19
|
+
export const matchesTerminationAction = (incomingAction, dataRequestAction) => incomingAction.type === PERIODIC_MODEL_FETCH_TERMINATION_ACTION_TYPE.TERMINATE && incomingAction.taskId === dataRequestAction.taskId;
|
|
20
|
+
export const takeMatchesTerminationAction = dataRequestAction => incomingAction => matchesTerminationAction(incomingAction, dataRequestAction);
|
|
21
|
+
|
|
30
22
|
/* istanbul ignore next */
|
|
31
23
|
/* eslint-disable-next-line require-yield */
|
|
32
|
-
const defaultTokenAccessFunction = function* () {
|
|
33
|
-
|
|
24
|
+
export const defaultTokenAccessFunction = function* () {
|
|
25
|
+
return null;
|
|
34
26
|
};
|
|
35
|
-
|
|
27
|
+
|
|
36
28
|
/* istanbul ignore next */
|
|
37
|
-
const defaultErrorHandler = () => {
|
|
38
|
-
|
|
29
|
+
export const defaultErrorHandler = () => {
|
|
30
|
+
return;
|
|
39
31
|
};
|
|
40
|
-
|
|
41
|
-
const
|
|
42
|
-
|
|
43
|
-
if (!value || !Object.prototype.hasOwnProperty.call(value, '_config')) {
|
|
44
|
-
return out;
|
|
45
|
-
}
|
|
46
|
-
const em = value;
|
|
47
|
-
if (!!em._config && !!em._config.isCollection) {
|
|
48
|
-
out.push(key);
|
|
49
|
-
}
|
|
32
|
+
const getChildCollectionKeys = endpointMapping => Object.keys(endpointMapping).reduce((out, key) => {
|
|
33
|
+
const value = endpointMapping[key];
|
|
34
|
+
if (!value || !Object.prototype.hasOwnProperty.call(value, '_config')) {
|
|
50
35
|
return out;
|
|
36
|
+
}
|
|
37
|
+
const em = value;
|
|
38
|
+
if (!!em._config && !!em._config.isCollection) {
|
|
39
|
+
out.push(key);
|
|
40
|
+
}
|
|
41
|
+
return out;
|
|
51
42
|
}, []);
|
|
52
43
|
const convertCollections = (endpointMapping, prepareFetchResult, data) => {
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
}
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
44
|
+
const {
|
|
45
|
+
endpointConfig,
|
|
46
|
+
fetchConfig,
|
|
47
|
+
isCollectionItemCreate,
|
|
48
|
+
isCollectionItemFetch
|
|
49
|
+
} = prepareFetchResult;
|
|
50
|
+
const childCollectionKeys = getChildCollectionKeys(endpointMapping);
|
|
51
|
+
const hasChildCollections = childCollectionKeys.length > 0;
|
|
52
|
+
// no collections, return as-is
|
|
53
|
+
if (!endpointConfig.isCollection && !hasChildCollections) {
|
|
54
|
+
return data;
|
|
55
|
+
}
|
|
56
|
+
// delete returns no data
|
|
57
|
+
if (fetchConfig.method === 'DELETE') {
|
|
58
|
+
return {};
|
|
59
|
+
}
|
|
60
|
+
// data is a collection item, convert any child collections in item response
|
|
61
|
+
if (isCollectionItemFetch || isCollectionItemCreate) {
|
|
62
|
+
return convertDataCollectionItem(data, childCollectionKeys, endpointMapping);
|
|
63
|
+
}
|
|
64
|
+
// data is a collection, convert collection response, and any child collections in each item
|
|
65
|
+
return endpointConfig.isCollection ? convertDataToCollection(data, childCollectionKeys, endpointMapping) : convertDataCollectionItem(data, childCollectionKeys, endpointMapping);
|
|
72
66
|
};
|
|
73
67
|
const convertDataCollectionItem = (data, collectionKeys, endpointMapping) => {
|
|
74
|
-
|
|
75
|
-
|
|
68
|
+
if (isNil(data)) {
|
|
69
|
+
return data;
|
|
70
|
+
}
|
|
71
|
+
return Object.keys(data).reduce((out, key) => {
|
|
72
|
+
let value = data[key];
|
|
73
|
+
if (collectionKeys.includes(key)) {
|
|
74
|
+
const childEndpointMapping = endpointMapping[key];
|
|
75
|
+
const childCollectionKeys = getChildCollectionKeys(childEndpointMapping);
|
|
76
|
+
value = convertDataToCollection(value, childCollectionKeys, childEndpointMapping);
|
|
76
77
|
}
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
const childEndpointMapping = endpointMapping[key];
|
|
81
|
-
const childCollectionKeys = getChildCollectionKeys(childEndpointMapping);
|
|
82
|
-
value = convertDataToCollection(value, childCollectionKeys, childEndpointMapping);
|
|
83
|
-
}
|
|
84
|
-
out[key] = value;
|
|
85
|
-
return out;
|
|
86
|
-
}, {});
|
|
78
|
+
out[key] = value;
|
|
79
|
+
return out;
|
|
80
|
+
}, {});
|
|
87
81
|
};
|
|
88
82
|
const convertDataToCollection = (data, collectionKeys, endpointMapping) => {
|
|
89
|
-
|
|
90
|
-
|
|
83
|
+
if (isNil(data)) {
|
|
84
|
+
return data;
|
|
85
|
+
}
|
|
86
|
+
return Object.keys(data).reduce((out, key) => {
|
|
87
|
+
const value = convertDataCollectionItem(data[key], collectionKeys, endpointMapping);
|
|
88
|
+
/* istanbul ignore else */
|
|
89
|
+
if (value && value.id) {
|
|
90
|
+
out[value.id] = value;
|
|
91
91
|
}
|
|
92
|
-
return
|
|
93
|
-
|
|
94
|
-
/* istanbul ignore else */
|
|
95
|
-
if (value && value.id) {
|
|
96
|
-
out[value.id] = value;
|
|
97
|
-
}
|
|
98
|
-
return out;
|
|
99
|
-
}, {});
|
|
92
|
+
return out;
|
|
93
|
+
}, {});
|
|
100
94
|
};
|
|
95
|
+
|
|
101
96
|
/**
|
|
102
97
|
* Using the `lastActivityDate` set in local storage by "studiokit-caliper-js",
|
|
103
98
|
* determine if the user has been idle, e.g. has had no activity in the last 15 minutes.
|
|
104
99
|
*/
|
|
105
100
|
function isUserIdle() {
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
101
|
+
const lastActivityDateString = getLastActivityDate();
|
|
102
|
+
if (!lastActivityDateString) {
|
|
103
|
+
return false;
|
|
104
|
+
}
|
|
105
|
+
const lastActivityDateUtc = moment.utc(lastActivityDateString);
|
|
106
|
+
const nowUtc = moment.utc();
|
|
107
|
+
const idleMinutesThreshold = 1000 * 60 * 15; // 15 minutes
|
|
108
|
+
|
|
109
|
+
return nowUtc.diff(lastActivityDateUtc) >= idleMinutesThreshold;
|
|
114
110
|
}
|
|
111
|
+
|
|
115
112
|
//#endregion Helpers
|
|
113
|
+
|
|
116
114
|
//#region Local Variables
|
|
115
|
+
|
|
117
116
|
let endpointMappings;
|
|
118
117
|
let tokenAccessFunction;
|
|
119
118
|
let errorHandler;
|
|
120
119
|
let logger;
|
|
120
|
+
|
|
121
121
|
//#endregion Local Variables
|
|
122
|
+
|
|
122
123
|
/** Set the shared EndpointMappings variable, exposed for testability */
|
|
123
|
-
function setEndpointMappings(endpointMappingsParam) {
|
|
124
|
-
|
|
124
|
+
export function setEndpointMappings(endpointMappingsParam) {
|
|
125
|
+
endpointMappings = endpointMappingsParam;
|
|
125
126
|
}
|
|
127
|
+
|
|
126
128
|
/**
|
|
127
129
|
* Construct a request based on the provided dataRequestAction, make a request with a configurable retry,
|
|
128
130
|
* and handle errors, logging and dispatching all steps.
|
|
129
131
|
*
|
|
130
132
|
* @param modelFetchRequestAction A model fetch request action with the request configuration
|
|
131
133
|
*/
|
|
132
|
-
function* modelFetch(modelFetchRequestAction) {
|
|
133
|
-
|
|
134
|
+
export function* modelFetch(modelFetchRequestAction) {
|
|
135
|
+
let prepareFetchResult = undefined;
|
|
136
|
+
try {
|
|
137
|
+
prepareFetchResult = prepareFetch(modelFetchRequestAction, endpointMappings);
|
|
138
|
+
} catch (error) {
|
|
139
|
+
// Note: `prepareFetch` already ensures that `error` is an actual `Error`
|
|
140
|
+
logger.error(error);
|
|
141
|
+
errorHandler(error, modelFetchRequestAction);
|
|
142
|
+
return;
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
// get prepared variables
|
|
146
|
+
const {
|
|
147
|
+
modelPath,
|
|
148
|
+
fetchConfig,
|
|
149
|
+
endpointMapping,
|
|
150
|
+
isCollectionItemCreate,
|
|
151
|
+
isCollectionItemDelete
|
|
152
|
+
} = prepareFetchResult;
|
|
153
|
+
|
|
154
|
+
// Configure retry
|
|
155
|
+
const tryLimit = modelFetchRequestAction.noRetry ? 1 : TRY_LIMIT;
|
|
156
|
+
let tryCount = 0;
|
|
157
|
+
let didFail = false;
|
|
158
|
+
let lastFetchResult;
|
|
159
|
+
let lastError;
|
|
160
|
+
let lastErrorData;
|
|
161
|
+
const fetchResultUndefinedErrorMessage = "'fetchResult' is undefined";
|
|
162
|
+
const fetchResultNotOkErrorMessage = "'fetchResult.ok' is false";
|
|
163
|
+
|
|
164
|
+
// Run retry loop
|
|
165
|
+
do {
|
|
166
|
+
didFail = false;
|
|
167
|
+
tryCount++;
|
|
168
|
+
|
|
169
|
+
// dispatch that the fetch request started, which updates `_metadata` and `guid`, if provided, at the target redux `modelPath`
|
|
170
|
+
yield put({
|
|
171
|
+
type: modelFetchRequestAction.noStore ? MODEL_FETCH_START_ACTION_TYPE.TRANSIENT_FETCH_START : MODEL_FETCH_START_ACTION_TYPE.FETCH_START,
|
|
172
|
+
modelPath,
|
|
173
|
+
guid: modelFetchRequestAction.guid
|
|
174
|
+
});
|
|
175
|
+
let fetchResult = undefined;
|
|
134
176
|
try {
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
errorHandler(error, modelFetchRequestAction);
|
|
141
|
-
return;
|
|
142
|
-
}
|
|
143
|
-
// get prepared variables
|
|
144
|
-
const { modelPath, fetchConfig, endpointMapping, isCollectionItemCreate, isCollectionItemDelete } = prepareFetchResult;
|
|
145
|
-
// Configure retry
|
|
146
|
-
const tryLimit = modelFetchRequestAction.noRetry ? 1 : exports.TRY_LIMIT;
|
|
147
|
-
let tryCount = 0;
|
|
148
|
-
let didFail = false;
|
|
149
|
-
let lastFetchResult;
|
|
150
|
-
let lastError;
|
|
151
|
-
let lastErrorData;
|
|
152
|
-
const fetchResultUndefinedErrorMessage = "'fetchResult' is undefined";
|
|
153
|
-
const fetchResultNotOkErrorMessage = "'fetchResult.ok' is false";
|
|
154
|
-
// Run retry loop
|
|
155
|
-
do {
|
|
156
|
-
didFail = false;
|
|
157
|
-
tryCount++;
|
|
158
|
-
// dispatch that the fetch request started, which updates `_metadata` and `guid`, if provided, at the target redux `modelPath`
|
|
159
|
-
yield (0, effects_1.put)({
|
|
160
|
-
type: modelFetchRequestAction.noStore
|
|
161
|
-
? actions_1.MODEL_FETCH_START_ACTION_TYPE.TRANSIENT_FETCH_START
|
|
162
|
-
: actions_1.MODEL_FETCH_START_ACTION_TYPE.FETCH_START,
|
|
163
|
-
modelPath,
|
|
164
|
-
guid: modelFetchRequestAction.guid
|
|
177
|
+
// get the OAuth Token, if any, and call `fetch`
|
|
178
|
+
const oauthToken = yield call(tokenAccessFunction, modelFetchRequestAction.modelName);
|
|
179
|
+
if (oauthToken?.access_token) {
|
|
180
|
+
fetchConfig.headers = merge({}, fetchConfig.headers, {
|
|
181
|
+
Authorization: `Bearer ${oauthToken.access_token}`
|
|
165
182
|
});
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
modelPath: `${modelPathLevels.join('.')}.${data.id}`,
|
|
201
|
-
guid: modelFetchRequestAction.guid,
|
|
202
|
-
replaceValue: modelFetchRequestAction.replaceValue,
|
|
203
|
-
data
|
|
204
|
-
});
|
|
205
|
-
// remove temp item under guid key
|
|
206
|
-
yield (0, effects_1.put)({
|
|
207
|
-
type: actions_1.MODEL_REMOVE_KEY_ACTION_TYPE,
|
|
208
|
-
modelPath: modelPath
|
|
209
|
-
});
|
|
210
|
-
}
|
|
211
|
-
// DELETE collection item
|
|
212
|
-
else if (isCollectionItemDelete) {
|
|
213
|
-
yield (0, effects_1.put)({
|
|
214
|
-
type: actions_1.MODEL_REMOVE_KEY_ACTION_TYPE,
|
|
215
|
-
modelPath: modelPath
|
|
216
|
-
});
|
|
217
|
-
}
|
|
218
|
-
else {
|
|
219
|
-
yield (0, effects_1.put)({
|
|
220
|
-
type: resultActionType,
|
|
221
|
-
modelPath: modelPath,
|
|
222
|
-
guid: modelFetchRequestAction.guid,
|
|
223
|
-
replaceValue: modelFetchRequestAction.replaceValue,
|
|
224
|
-
data
|
|
225
|
-
});
|
|
226
|
-
}
|
|
227
|
-
}
|
|
228
|
-
catch (caughtError) {
|
|
229
|
-
// construct `Error` using `caughtError`
|
|
230
|
-
const baseErrorMessage = `'modelFetch' failed for '${modelPath}'`;
|
|
231
|
-
const error = (0, error_1.constructErrorFromCaughtError)(caughtError, 'ModelFetchError', baseErrorMessage);
|
|
232
|
-
// create a default `errorData` object
|
|
233
|
-
const isNetworkOnline = windowService_1.windowService.getIsOnline();
|
|
234
|
-
let errorData = isNetworkOnline ? fetchErrorData_1.unknownErrorData : fetchErrorData_1.networkOfflineErrorData;
|
|
235
|
-
// update `errorData` from the `fetchResult.data`
|
|
236
|
-
// for when the fetch got a result, but it was an unsuccessful status
|
|
237
|
-
if ((fetchResult === null || fetchResult === void 0 ? void 0 : fetchResult.data) && (0, fetch_1.isFetchErrorData)(fetchResult.data)) {
|
|
238
|
-
errorData = fetchResult.data;
|
|
239
|
-
// update the error's message to include info from the fetch result
|
|
240
|
-
error.message = `${baseErrorMessage}.\n${fetchConfig.method || 'GET'} ${fetchConfig.path} failed.\n${errorData.title} (${errorData.status})`;
|
|
241
|
-
if (errorData.detail) {
|
|
242
|
-
error.message += `: ${errorData.detail}`;
|
|
243
|
-
}
|
|
244
|
-
}
|
|
245
|
-
// track `errorData` and `error` for after the while-loop
|
|
246
|
-
lastErrorData = errorData;
|
|
247
|
-
lastError = error;
|
|
248
|
-
// dispatch that this try failed
|
|
249
|
-
yield (0, effects_1.put)({
|
|
250
|
-
type: actions_1.MODEL_FETCH_ERROR_ACTION_TYPE.TRY_FETCH_FAILED,
|
|
251
|
-
modelPath: modelPath,
|
|
252
|
-
guid: modelFetchRequestAction.guid,
|
|
253
|
-
errorData
|
|
254
|
-
});
|
|
255
|
-
// log to the console
|
|
256
|
-
logger.error(error);
|
|
257
|
-
didFail = true;
|
|
258
|
-
// Do not retry if the response is between 400-500 (except 408: request timeout)
|
|
259
|
-
if (errorData.status >= types_1.HTTP_STATUS_CODE.BAD_REQUEST &&
|
|
260
|
-
errorData.status < types_1.HTTP_STATUS_CODE.INTERNAL_SERVER_ERROR &&
|
|
261
|
-
errorData.status !== types_1.HTTP_STATUS_CODE.REQUEST_TIMEOUT) {
|
|
262
|
-
tryCount = tryLimit;
|
|
263
|
-
}
|
|
264
|
-
// if there are tries remaining, perform an exponential backoff
|
|
265
|
-
if (tryCount < tryLimit) {
|
|
266
|
-
yield (0, effects_1.delay)(Math.pow(2, (tryCount - 1)) * 100); // 100, 200, 400, 800
|
|
267
|
-
}
|
|
268
|
-
}
|
|
269
|
-
} while (tryCount < tryLimit && didFail);
|
|
270
|
-
// if fetch failed completely after all retries
|
|
271
|
-
if (tryCount === tryLimit && didFail) {
|
|
272
|
-
// these will always exist
|
|
273
|
-
const errorData = lastErrorData;
|
|
274
|
-
const error = lastError;
|
|
275
|
-
// dispatch that the fetch failed, which updates `_metadata` (and `guid`, if provided) at the target redux `modelPath`
|
|
276
|
-
yield (0, effects_1.put)({
|
|
277
|
-
type: modelFetchRequestAction.noStore
|
|
278
|
-
? actions_1.MODEL_FETCH_ERROR_ACTION_TYPE.TRANSIENT_FETCH_FAILED
|
|
279
|
-
: actions_1.MODEL_FETCH_ERROR_ACTION_TYPE.FETCH_FAILED,
|
|
280
|
-
modelPath: modelPath,
|
|
281
|
-
guid: modelFetchRequestAction.guid,
|
|
282
|
-
errorData
|
|
183
|
+
}
|
|
184
|
+
fetchResult = yield call(sendFetch, fetchConfig);
|
|
185
|
+
// store `fetchResult` for reference after the do-while loop
|
|
186
|
+
lastFetchResult = fetchResult;
|
|
187
|
+
|
|
188
|
+
// throw on unsuccessful result to short-circuit and trigger catch+retry
|
|
189
|
+
if (!fetchResult) throw new Error(fetchResultUndefinedErrorMessage);
|
|
190
|
+
if (!fetchResult.ok) throw new Error(fetchResultNotOkErrorMessage);
|
|
191
|
+
|
|
192
|
+
// handle success
|
|
193
|
+
const resultActionType = modelFetchRequestAction.noStore ? MODEL_FETCH_RESULT_ACTION_TYPE.TRANSIENT_FETCH_RESULT_RECEIVED : MODEL_FETCH_RESULT_ACTION_TYPE.FETCH_RESULT_RECEIVED;
|
|
194
|
+
const data = convertCollections(endpointMapping, prepareFetchResult, fetchResult.data);
|
|
195
|
+
// attach guid to result data if it is an object
|
|
196
|
+
if (modelFetchRequestAction.guid && isPlainObject(data)) {
|
|
197
|
+
data.guid = modelFetchRequestAction.guid;
|
|
198
|
+
}
|
|
199
|
+
|
|
200
|
+
// POST new collection item
|
|
201
|
+
if (isCollectionItemCreate) {
|
|
202
|
+
const modelPathLevels = modelPath.split('.');
|
|
203
|
+
// remove guid
|
|
204
|
+
modelPathLevels.pop();
|
|
205
|
+
// add by new result's id
|
|
206
|
+
yield put({
|
|
207
|
+
type: resultActionType,
|
|
208
|
+
modelPath: `${modelPathLevels.join('.')}.${data.id}`,
|
|
209
|
+
guid: modelFetchRequestAction.guid,
|
|
210
|
+
replaceValue: modelFetchRequestAction.replaceValue,
|
|
211
|
+
data
|
|
212
|
+
});
|
|
213
|
+
// remove temp item under guid key
|
|
214
|
+
yield put({
|
|
215
|
+
type: MODEL_REMOVE_KEY_ACTION_TYPE,
|
|
216
|
+
modelPath: modelPath
|
|
283
217
|
});
|
|
284
|
-
|
|
285
|
-
|
|
286
|
-
|
|
218
|
+
}
|
|
219
|
+
// DELETE collection item
|
|
220
|
+
else if (isCollectionItemDelete) {
|
|
221
|
+
yield put({
|
|
222
|
+
type: MODEL_REMOVE_KEY_ACTION_TYPE,
|
|
223
|
+
modelPath: modelPath
|
|
224
|
+
});
|
|
225
|
+
} else {
|
|
226
|
+
yield put({
|
|
227
|
+
type: resultActionType,
|
|
228
|
+
modelPath: modelPath,
|
|
229
|
+
guid: modelFetchRequestAction.guid,
|
|
230
|
+
replaceValue: modelFetchRequestAction.replaceValue,
|
|
231
|
+
data
|
|
232
|
+
});
|
|
233
|
+
}
|
|
234
|
+
} catch (caughtError) {
|
|
235
|
+
// construct `Error` using `caughtError`
|
|
236
|
+
const baseErrorMessage = `'modelFetch' failed for '${modelPath}'`;
|
|
237
|
+
const error = constructErrorFromCaughtError(caughtError, 'ModelFetchError', baseErrorMessage);
|
|
238
|
+
// create a default `errorData` object
|
|
239
|
+
const isNetworkOnline = windowService.getIsOnline();
|
|
240
|
+
let errorData = isNetworkOnline ? unknownErrorData : networkOfflineErrorData;
|
|
241
|
+
// update `errorData` from the `fetchResult.data`
|
|
242
|
+
// for when the fetch got a result, but it was an unsuccessful status
|
|
243
|
+
if (fetchResult?.data && isFetchErrorData(fetchResult.data)) {
|
|
244
|
+
errorData = fetchResult.data;
|
|
245
|
+
// update the error's message to include info from the fetch result
|
|
246
|
+
error.message = `${baseErrorMessage}.\n${fetchConfig.method || 'GET'} ${fetchConfig.path} failed.\n${errorData.title} (${errorData.status})`;
|
|
247
|
+
if (errorData.detail) {
|
|
248
|
+
error.message += `: ${errorData.detail}`;
|
|
287
249
|
}
|
|
250
|
+
}
|
|
251
|
+
|
|
252
|
+
// track `errorData` and `error` for after the while-loop
|
|
253
|
+
lastErrorData = errorData;
|
|
254
|
+
lastError = error;
|
|
255
|
+
|
|
256
|
+
// dispatch that this try failed
|
|
257
|
+
yield put({
|
|
258
|
+
type: MODEL_FETCH_ERROR_ACTION_TYPE.TRY_FETCH_FAILED,
|
|
259
|
+
modelPath: modelPath,
|
|
260
|
+
guid: modelFetchRequestAction.guid,
|
|
261
|
+
errorData
|
|
262
|
+
});
|
|
263
|
+
|
|
264
|
+
// log to the console
|
|
265
|
+
logger.error(error);
|
|
266
|
+
didFail = true;
|
|
267
|
+
|
|
268
|
+
// Do not retry if the response is between 400-500 (except 408: request timeout)
|
|
269
|
+
if (errorData.status >= HTTP_STATUS_CODE.BAD_REQUEST && errorData.status < HTTP_STATUS_CODE.INTERNAL_SERVER_ERROR && errorData.status !== HTTP_STATUS_CODE.REQUEST_TIMEOUT) {
|
|
270
|
+
tryCount = tryLimit;
|
|
271
|
+
}
|
|
272
|
+
|
|
273
|
+
// if there are tries remaining, perform an exponential backoff
|
|
274
|
+
if (tryCount < tryLimit) {
|
|
275
|
+
yield delay(2 ** (tryCount - 1) * 100); // 100, 200, 400, 800
|
|
276
|
+
}
|
|
277
|
+
}
|
|
278
|
+
} while (tryCount < tryLimit && didFail);
|
|
279
|
+
|
|
280
|
+
// if fetch failed completely after all retries
|
|
281
|
+
if (tryCount === tryLimit && didFail) {
|
|
282
|
+
// these will always exist
|
|
283
|
+
const errorData = lastErrorData;
|
|
284
|
+
const error = lastError;
|
|
285
|
+
|
|
286
|
+
// dispatch that the fetch failed, which updates `_metadata` (and `guid`, if provided) at the target redux `modelPath`
|
|
287
|
+
yield put({
|
|
288
|
+
type: modelFetchRequestAction.noStore ? MODEL_FETCH_ERROR_ACTION_TYPE.TRANSIENT_FETCH_FAILED : MODEL_FETCH_ERROR_ACTION_TYPE.FETCH_FAILED,
|
|
289
|
+
modelPath: modelPath,
|
|
290
|
+
guid: modelFetchRequestAction.guid,
|
|
291
|
+
errorData
|
|
292
|
+
});
|
|
293
|
+
|
|
294
|
+
// Send error to error handler, except for 401s
|
|
295
|
+
if (errorData.status !== HTTP_STATUS_CODE.UNAUTHORIZED) {
|
|
296
|
+
errorHandler(error, modelFetchRequestAction, fetchConfig, lastFetchResult, errorData);
|
|
288
297
|
}
|
|
298
|
+
}
|
|
289
299
|
}
|
|
300
|
+
|
|
290
301
|
/**
|
|
291
302
|
* The loop saga that makes the request every {action.period} milliseconds until cancelled
|
|
292
303
|
*
|
|
293
304
|
* @param action An action with the request configuration
|
|
294
305
|
*/
|
|
295
|
-
function* modelFetchLoop(action) {
|
|
296
|
-
|
|
297
|
-
|
|
298
|
-
|
|
299
|
-
|
|
300
|
-
|
|
301
|
-
|
|
302
|
-
|
|
303
|
-
|
|
304
|
-
|
|
305
|
-
|
|
306
|
-
|
|
307
|
-
|
|
308
|
-
|
|
309
|
-
|
|
310
|
-
|
|
311
|
-
|
|
312
|
-
|
|
313
|
-
|
|
314
|
-
|
|
315
|
-
|
|
316
|
-
|
|
317
|
-
|
|
318
|
-
taskId: action.taskId
|
|
319
|
-
});
|
|
320
|
-
}
|
|
306
|
+
export function* modelFetchLoop(action) {
|
|
307
|
+
try {
|
|
308
|
+
let hasFetched = false;
|
|
309
|
+
do {
|
|
310
|
+
// call the fetch if this is the first loop iteration,
|
|
311
|
+
// or if the user is not idle
|
|
312
|
+
if (!hasFetched || !isUserIdle()) {
|
|
313
|
+
yield call(modelFetch, action);
|
|
314
|
+
hasFetched = true;
|
|
315
|
+
}
|
|
316
|
+
yield delay(action.period);
|
|
317
|
+
} while (true);
|
|
318
|
+
} catch (caughtError) {
|
|
319
|
+
const error = constructErrorFromCaughtError(caughtError, 'ModelFetchLoopError', 'modelFetchLoop failed');
|
|
320
|
+
logger.error(error);
|
|
321
|
+
// catch and log any unexpected errors not caught inside `fetchData`
|
|
322
|
+
errorHandler(error, action);
|
|
323
|
+
} finally {
|
|
324
|
+
if (yield cancelled()) {
|
|
325
|
+
yield put({
|
|
326
|
+
type: PERIODIC_MODEL_FETCH_TERMINATION_ACTION_TYPE.SUCCEEDED,
|
|
327
|
+
taskId: action.taskId
|
|
328
|
+
});
|
|
321
329
|
}
|
|
330
|
+
}
|
|
322
331
|
}
|
|
332
|
+
|
|
323
333
|
/**
|
|
324
334
|
* Call the fetchData saga every {action.period} milliseconds. This saga requires the 'period' and 'taskId' properties
|
|
325
335
|
* on the action parameter.
|
|
326
336
|
*
|
|
327
337
|
* @param action An action with the request configuration
|
|
328
338
|
*/
|
|
329
|
-
function* modelFetchRecurring(action) {
|
|
330
|
-
|
|
331
|
-
|
|
332
|
-
|
|
333
|
-
|
|
334
|
-
|
|
335
|
-
|
|
336
|
-
|
|
337
|
-
|
|
338
|
-
|
|
339
|
+
export function* modelFetchRecurring(action) {
|
|
340
|
+
if (!action.period) {
|
|
341
|
+
throw new Error("'period' is required");
|
|
342
|
+
}
|
|
343
|
+
if (!action.taskId) {
|
|
344
|
+
throw new Error("'taskId' is required");
|
|
345
|
+
}
|
|
346
|
+
const bgSyncTask = yield fork(modelFetchLoop, action);
|
|
347
|
+
yield take(takeMatchesTerminationAction(action));
|
|
348
|
+
yield cancel(bgSyncTask);
|
|
339
349
|
}
|
|
350
|
+
|
|
340
351
|
/**
|
|
341
352
|
* The main saga for fetching models. Must be initialized with an EndpointMappings object that can be fetched
|
|
342
353
|
* and an API root to prepend to any partial URLs specified in the EndpointMappings object. A logger should normally be provided
|
|
@@ -369,12 +380,17 @@ function* modelFetchRecurring(action) {
|
|
|
369
380
|
* @param tokenAccessFunctionParam function that returns an optional OAuth token
|
|
370
381
|
* @param errorHandlerParam A function that is called when any fetch exceptions occur
|
|
371
382
|
*/
|
|
372
|
-
function
|
|
373
|
-
|
|
383
|
+
export default function modelFetchSaga(endpointMappingsParam, apiRootParam) {
|
|
384
|
+
let tokenAccessFunctionParam = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : defaultTokenAccessFunction;
|
|
385
|
+
let errorHandlerParam = arguments.length > 3 && arguments[3] !== undefined ? arguments[3] : defaultErrorHandler;
|
|
386
|
+
return function* () {
|
|
387
|
+
setApiRoot(apiRootParam);
|
|
374
388
|
setEndpointMappings(endpointMappingsParam);
|
|
375
389
|
errorHandler = errorHandlerParam;
|
|
376
390
|
tokenAccessFunction = tokenAccessFunctionParam;
|
|
377
|
-
logger =
|
|
378
|
-
yield
|
|
379
|
-
yield
|
|
391
|
+
logger = getLogger();
|
|
392
|
+
yield takeEvery(MODEL_FETCH_REQUEST_ACTION_TYPE.FETCH_REQUEST, modelFetch);
|
|
393
|
+
yield takeEvery(MODEL_FETCH_REQUEST_ACTION_TYPE.PERIODIC_FETCH_REQUEST, modelFetchRecurring);
|
|
394
|
+
}();
|
|
380
395
|
}
|
|
396
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["isNil","isPlainObject","merge","moment","call","cancel","cancelled","delay","fork","put","take","takeEvery","networkOfflineErrorData","unknownErrorData","getLastActivityDate","sendFetch","setApiRoot","windowService","HTTP_STATUS_CODE","constructErrorFromCaughtError","isFetchErrorData","prepareFetch","getLogger","MODEL_FETCH_ERROR_ACTION_TYPE","MODEL_FETCH_REQUEST_ACTION_TYPE","MODEL_FETCH_RESULT_ACTION_TYPE","MODEL_FETCH_START_ACTION_TYPE","MODEL_REMOVE_KEY_ACTION_TYPE","PERIODIC_MODEL_FETCH_TERMINATION_ACTION_TYPE","TRY_LIMIT","matchesTerminationAction","incomingAction","dataRequestAction","type","TERMINATE","taskId","takeMatchesTerminationAction","defaultTokenAccessFunction","defaultErrorHandler","getChildCollectionKeys","endpointMapping","Object","keys","reduce","out","key","value","prototype","hasOwnProperty","em","_config","isCollection","push","convertCollections","prepareFetchResult","data","endpointConfig","fetchConfig","isCollectionItemCreate","isCollectionItemFetch","childCollectionKeys","hasChildCollections","length","method","convertDataCollectionItem","convertDataToCollection","collectionKeys","includes","childEndpointMapping","id","isUserIdle","lastActivityDateString","lastActivityDateUtc","utc","nowUtc","idleMinutesThreshold","diff","endpointMappings","tokenAccessFunction","errorHandler","logger","setEndpointMappings","endpointMappingsParam","modelFetch","modelFetchRequestAction","undefined","error","modelPath","isCollectionItemDelete","tryLimit","noRetry","tryCount","didFail","lastFetchResult","lastError","lastErrorData","fetchResultUndefinedErrorMessage","fetchResultNotOkErrorMessage","noStore","TRANSIENT_FETCH_START","FETCH_START","guid","fetchResult","oauthToken","modelName","access_token","headers","Authorization","Error","ok","resultActionType","TRANSIENT_FETCH_RESULT_RECEIVED","FETCH_RESULT_RECEIVED","modelPathLevels","split","pop","join","replaceValue","caughtError","baseErrorMessage","isNetworkOnline","getIsOnline","errorData","message","path","title","status","detail","TRY_FETCH_FAILED","BAD_REQUEST","INTERNAL_SERVER_ERROR","REQUEST_TIMEOUT","TRANSIENT_FETCH_FAILED","FETCH_FAILED","UNAUTHORIZED","modelFetchLoop","action","hasFetched","period","SUCCEEDED","modelFetchRecurring","bgSyncTask","modelFetchSaga","apiRootParam","tokenAccessFunctionParam","arguments","errorHandlerParam","FETCH_REQUEST","PERIODIC_FETCH_REQUEST"],"sources":["../../../src/redux/sagas/modelFetchSaga.ts"],"sourcesContent":["import { SagaIterator, Task } from '@redux-saga/core'\nimport { isNil, isPlainObject, merge } from 'lodash'\nimport moment from 'moment-timezone'\nimport { AnyAction } from 'redux'\nimport { call, cancel, cancelled, delay, fork, put, take, takeEvery } from 'redux-saga/effects'\nimport { networkOfflineErrorData, unknownErrorData } from '../../constants/fetchErrorData'\nimport { getLastActivityDate } from '../../services/dateService'\nimport { sendFetch, setApiRoot } from '../../services/fetchService'\nimport { windowService } from '../../services/windowService'\nimport {\n\tEndpointMapping,\n\tEndpointMappings,\n\tErrorHandler,\n\tFetchErrorData,\n\tFetchResult,\n\tHTTP_STATUS_CODE,\n\tModel,\n\tModelCollection,\n\tOAuthTokenOrNull,\n\tTokenAccessFunction\n} from '../../types'\nimport { constructErrorFromCaughtError } from '../../utils/error'\nimport { isFetchErrorData, prepareFetch, PrepareFetchResult } from '../../utils/fetch'\nimport { getLogger, Logger } from '../../utils/logger'\nimport {\n\tMODEL_FETCH_ERROR_ACTION_TYPE,\n\tMODEL_FETCH_REQUEST_ACTION_TYPE,\n\tMODEL_FETCH_RESULT_ACTION_TYPE,\n\tMODEL_FETCH_START_ACTION_TYPE,\n\tMODEL_REMOVE_KEY_ACTION_TYPE,\n\tModelFetchErrorAction,\n\tModelFetchRequestAction,\n\tModelFetchResultAction,\n\tModelFetchStartAction,\n\tModelRemoveKeyAction,\n\tPERIODIC_MODEL_FETCH_TERMINATION_ACTION_TYPE,\n\tPeriodicModelFetchRequestAction,\n\tPeriodicModelFetchTerminationAction\n} from '../actions'\n\n/** The total number of tries for `fetchData`, including the initial request. */\nexport const TRY_LIMIT = 5\n\n//#region Helpers\n\nexport const matchesTerminationAction = (\n\tincomingAction: AnyAction,\n\tdataRequestAction: PeriodicModelFetchRequestAction\n) =>\n\tincomingAction.type === PERIODIC_MODEL_FETCH_TERMINATION_ACTION_TYPE.TERMINATE &&\n\tincomingAction.taskId === dataRequestAction.taskId\n\nexport const takeMatchesTerminationAction =\n\t(dataRequestAction: PeriodicModelFetchRequestAction) => (incomingAction: AnyAction) =>\n\t\tmatchesTerminationAction(incomingAction, dataRequestAction)\n\n/* istanbul ignore next */\n/* eslint-disable-next-line require-yield */\nexport const defaultTokenAccessFunction: TokenAccessFunction = function* () {\n\treturn null\n}\n\n/* istanbul ignore next */\nexport const defaultErrorHandler: ErrorHandler = () => {\n\treturn\n}\n\ninterface GenericModel extends Model, Record<string, any> {}\n\nconst getChildCollectionKeys = (endpointMapping: EndpointMapping) =>\n\tObject.keys(endpointMapping).reduce((out: string[], key) => {\n\t\tconst value = endpointMapping[key]\n\t\tif (!value || !Object.prototype.hasOwnProperty.call(value, '_config')) {\n\t\t\treturn out\n\t\t}\n\t\tconst em = value as EndpointMapping\n\t\tif (!!em._config && !!em._config.isCollection) {\n\t\t\tout.push(key)\n\t\t}\n\t\treturn out\n\t}, [])\n\nconst convertCollections = (endpointMapping: EndpointMapping, prepareFetchResult: PrepareFetchResult, data: any) => {\n\tconst { endpointConfig, fetchConfig, isCollectionItemCreate, isCollectionItemFetch } = prepareFetchResult\n\tconst childCollectionKeys = getChildCollectionKeys(endpointMapping)\n\tconst hasChildCollections = childCollectionKeys.length > 0\n\t// no collections, return as-is\n\tif (!endpointConfig.isCollection && !hasChildCollections) {\n\t\treturn data\n\t}\n\t// delete returns no data\n\tif (fetchConfig.method === 'DELETE') {\n\t\treturn {}\n\t}\n\t// data is a collection item, convert any child collections in item response\n\tif (isCollectionItemFetch || isCollectionItemCreate) {\n\t\treturn convertDataCollectionItem(data, childCollectionKeys, endpointMapping)\n\t}\n\t// data is a collection, convert collection response, and any child collections in each item\n\treturn endpointConfig.isCollection\n\t\t? convertDataToCollection(data, childCollectionKeys, endpointMapping)\n\t\t: convertDataCollectionItem(data, childCollectionKeys, endpointMapping)\n}\n\nconst convertDataCollectionItem = (data: any, collectionKeys: string[], endpointMapping: EndpointMapping) => {\n\tif (isNil(data)) {\n\t\treturn data\n\t}\n\treturn Object.keys(data).reduce((out: GenericModel, key) => {\n\t\tlet value = data[key]\n\t\tif (collectionKeys.includes(key)) {\n\t\t\tconst childEndpointMapping = endpointMapping[key] as EndpointMapping\n\t\t\tconst childCollectionKeys = getChildCollectionKeys(childEndpointMapping)\n\t\t\tvalue = convertDataToCollection(value, childCollectionKeys, childEndpointMapping)\n\t\t}\n\t\tout[key] = value\n\t\treturn out\n\t}, {})\n}\n\nconst convertDataToCollection = (data: any, collectionKeys: string[], endpointMapping: EndpointMapping) => {\n\tif (isNil(data)) {\n\t\treturn data\n\t}\n\treturn Object.keys(data).reduce((out: ModelCollection<any>, key) => {\n\t\tconst value = convertDataCollectionItem(data[key], collectionKeys, endpointMapping)\n\t\t/* istanbul ignore else */\n\t\tif (value && value.id) {\n\t\t\tout[value.id] = value\n\t\t}\n\t\treturn out\n\t}, {})\n}\n\n/**\n * Using the `lastActivityDate` set in local storage by \"studiokit-caliper-js\",\n * determine if the user has been idle, e.g. has had no activity in the last 15 minutes.\n */\nfunction isUserIdle() {\n\tconst lastActivityDateString = getLastActivityDate()\n\tif (!lastActivityDateString) {\n\t\treturn false\n\t}\n\n\tconst lastActivityDateUtc = moment.utc(lastActivityDateString)\n\tconst nowUtc = moment.utc()\n\tconst idleMinutesThreshold = 1000 * 60 * 15 // 15 minutes\n\n\treturn nowUtc.diff(lastActivityDateUtc) >= idleMinutesThreshold\n}\n\n//#endregion Helpers\n\n//#region Local Variables\n\nlet endpointMappings: EndpointMappings\nlet tokenAccessFunction: TokenAccessFunction\nlet errorHandler: ErrorHandler\nlet logger: Logger\n\n//#endregion Local Variables\n\n/** Set the shared EndpointMappings variable, exposed for testability */\nexport function setEndpointMappings(endpointMappingsParam: EndpointMappings) {\n\tendpointMappings = endpointMappingsParam\n}\n\n/**\n * Construct a request based on the provided dataRequestAction, make a request with a configurable retry,\n * and handle errors, logging and dispatching all steps.\n *\n * @param modelFetchRequestAction A model fetch request action with the request configuration\n */\nexport function* modelFetch(\n\tmodelFetchRequestAction: ModelFetchRequestAction | PeriodicModelFetchRequestAction\n): SagaIterator {\n\tlet prepareFetchResult: PrepareFetchResult | undefined = undefined\n\ttry {\n\t\tprepareFetchResult = prepareFetch(modelFetchRequestAction, endpointMappings)\n\t} catch (error) {\n\t\t// Note: `prepareFetch` already ensures that `error` is an actual `Error`\n\t\tlogger.error(error)\n\t\terrorHandler(error as Error, modelFetchRequestAction)\n\t\treturn\n\t}\n\n\t// get prepared variables\n\tconst { modelPath, fetchConfig, endpointMapping, isCollectionItemCreate, isCollectionItemDelete } =\n\t\tprepareFetchResult\n\n\t// Configure retry\n\tconst tryLimit: number = modelFetchRequestAction.noRetry ? 1 : TRY_LIMIT\n\tlet tryCount = 0\n\tlet didFail = false\n\tlet lastFetchResult: FetchResult | undefined\n\tlet lastError: Error | undefined\n\tlet lastErrorData: FetchErrorData | undefined\n\n\tconst fetchResultUndefinedErrorMessage = \"'fetchResult' is undefined\"\n\tconst fetchResultNotOkErrorMessage = \"'fetchResult.ok' is false\"\n\n\t// Run retry loop\n\tdo {\n\t\tdidFail = false\n\t\ttryCount++\n\n\t\t// dispatch that the fetch request started, which updates `_metadata` and `guid`, if provided, at the target redux `modelPath`\n\t\tyield put<ModelFetchStartAction>({\n\t\t\ttype: modelFetchRequestAction.noStore\n\t\t\t\t? MODEL_FETCH_START_ACTION_TYPE.TRANSIENT_FETCH_START\n\t\t\t\t: MODEL_FETCH_START_ACTION_TYPE.FETCH_START,\n\t\t\tmodelPath,\n\t\t\tguid: modelFetchRequestAction.guid\n\t\t})\n\n\t\tlet fetchResult: FetchResult | undefined = undefined\n\t\ttry {\n\t\t\t// get the OAuth Token, if any, and call `fetch`\n\t\t\tconst oauthToken: OAuthTokenOrNull = yield call(tokenAccessFunction, modelFetchRequestAction.modelName)\n\t\t\tif (oauthToken?.access_token) {\n\t\t\t\tfetchConfig.headers = merge({}, fetchConfig.headers, {\n\t\t\t\t\tAuthorization: `Bearer ${oauthToken.access_token}`\n\t\t\t\t})\n\t\t\t}\n\t\t\tfetchResult = yield call(sendFetch, fetchConfig)\n\t\t\t// store `fetchResult` for reference after the do-while loop\n\t\t\tlastFetchResult = fetchResult\n\n\t\t\t// throw on unsuccessful result to short-circuit and trigger catch+retry\n\t\t\tif (!fetchResult) throw new Error(fetchResultUndefinedErrorMessage)\n\t\t\tif (!fetchResult.ok) throw new Error(fetchResultNotOkErrorMessage)\n\n\t\t\t// handle success\n\t\t\tconst resultActionType = modelFetchRequestAction.noStore\n\t\t\t\t? MODEL_FETCH_RESULT_ACTION_TYPE.TRANSIENT_FETCH_RESULT_RECEIVED\n\t\t\t\t: MODEL_FETCH_RESULT_ACTION_TYPE.FETCH_RESULT_RECEIVED\n\n\t\t\tconst data = convertCollections(endpointMapping, prepareFetchResult, fetchResult.data)\n\t\t\t// attach guid to result data if it is an object\n\t\t\tif (modelFetchRequestAction.guid && isPlainObject(data)) {\n\t\t\t\tdata.guid = modelFetchRequestAction.guid\n\t\t\t}\n\n\t\t\t// POST new collection item\n\t\t\tif (isCollectionItemCreate) {\n\t\t\t\tconst modelPathLevels = modelPath.split('.')\n\t\t\t\t// remove guid\n\t\t\t\tmodelPathLevels.pop()\n\t\t\t\t// add by new result's id\n\t\t\t\tyield put<ModelFetchResultAction>({\n\t\t\t\t\ttype: resultActionType,\n\t\t\t\t\tmodelPath: `${modelPathLevels.join('.')}.${data.id}`,\n\t\t\t\t\tguid: modelFetchRequestAction.guid,\n\t\t\t\t\treplaceValue: modelFetchRequestAction.replaceValue,\n\t\t\t\t\tdata\n\t\t\t\t})\n\t\t\t\t// remove temp item under guid key\n\t\t\t\tyield put<ModelRemoveKeyAction>({\n\t\t\t\t\ttype: MODEL_REMOVE_KEY_ACTION_TYPE,\n\t\t\t\t\tmodelPath: modelPath\n\t\t\t\t})\n\t\t\t}\n\t\t\t// DELETE collection item\n\t\t\telse if (isCollectionItemDelete) {\n\t\t\t\tyield put<ModelRemoveKeyAction>({\n\t\t\t\t\ttype: MODEL_REMOVE_KEY_ACTION_TYPE,\n\t\t\t\t\tmodelPath: modelPath\n\t\t\t\t})\n\t\t\t} else {\n\t\t\t\tyield put<ModelFetchResultAction>({\n\t\t\t\t\ttype: resultActionType,\n\t\t\t\t\tmodelPath: modelPath,\n\t\t\t\t\tguid: modelFetchRequestAction.guid,\n\t\t\t\t\treplaceValue: modelFetchRequestAction.replaceValue,\n\t\t\t\t\tdata\n\t\t\t\t})\n\t\t\t}\n\t\t} catch (caughtError: unknown) {\n\t\t\t// construct `Error` using `caughtError`\n\t\t\tconst baseErrorMessage = `'modelFetch' failed for '${modelPath}'`\n\t\t\tconst error = constructErrorFromCaughtError(caughtError, 'ModelFetchError', baseErrorMessage)\n\t\t\t// create a default `errorData` object\n\t\t\tconst isNetworkOnline = windowService.getIsOnline()\n\t\t\tlet errorData: FetchErrorData = isNetworkOnline ? unknownErrorData : networkOfflineErrorData\n\t\t\t// update `errorData` from the `fetchResult.data`\n\t\t\t// for when the fetch got a result, but it was an unsuccessful status\n\t\t\tif (fetchResult?.data && isFetchErrorData(fetchResult.data)) {\n\t\t\t\terrorData = fetchResult.data\n\t\t\t\t// update the error's message to include info from the fetch result\n\t\t\t\terror.message = `${baseErrorMessage}.\\n${fetchConfig.method || 'GET'} ${fetchConfig.path} failed.\\n${\n\t\t\t\t\terrorData.title\n\t\t\t\t} (${errorData.status})`\n\t\t\t\tif (errorData.detail) {\n\t\t\t\t\terror.message += `: ${errorData.detail}`\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// track `errorData` and `error` for after the while-loop\n\t\t\tlastErrorData = errorData\n\t\t\tlastError = error\n\n\t\t\t// dispatch that this try failed\n\t\t\tyield put<ModelFetchErrorAction>({\n\t\t\t\ttype: MODEL_FETCH_ERROR_ACTION_TYPE.TRY_FETCH_FAILED,\n\t\t\t\tmodelPath: modelPath,\n\t\t\t\tguid: modelFetchRequestAction.guid,\n\t\t\t\terrorData\n\t\t\t})\n\n\t\t\t// log to the console\n\t\t\tlogger.error(error)\n\n\t\t\tdidFail = true\n\n\t\t\t// Do not retry if the response is between 400-500 (except 408: request timeout)\n\t\t\tif (\n\t\t\t\terrorData.status >= HTTP_STATUS_CODE.BAD_REQUEST &&\n\t\t\t\terrorData.status < HTTP_STATUS_CODE.INTERNAL_SERVER_ERROR &&\n\t\t\t\terrorData.status !== HTTP_STATUS_CODE.REQUEST_TIMEOUT\n\t\t\t) {\n\t\t\t\ttryCount = tryLimit\n\t\t\t}\n\n\t\t\t// if there are tries remaining, perform an exponential backoff\n\t\t\tif (tryCount < tryLimit) {\n\t\t\t\tyield delay(2 ** (tryCount - 1) * 100) // 100, 200, 400, 800\n\t\t\t}\n\t\t}\n\t} while (tryCount < tryLimit && didFail)\n\n\t// if fetch failed completely after all retries\n\tif (tryCount === tryLimit && didFail) {\n\t\t// these will always exist\n\t\tconst errorData = lastErrorData as FetchErrorData\n\t\tconst error = lastError as Error\n\n\t\t// dispatch that the fetch failed, which updates `_metadata` (and `guid`, if provided) at the target redux `modelPath`\n\t\tyield put<ModelFetchErrorAction>({\n\t\t\ttype: modelFetchRequestAction.noStore\n\t\t\t\t? MODEL_FETCH_ERROR_ACTION_TYPE.TRANSIENT_FETCH_FAILED\n\t\t\t\t: MODEL_FETCH_ERROR_ACTION_TYPE.FETCH_FAILED,\n\t\t\tmodelPath: modelPath,\n\t\t\tguid: modelFetchRequestAction.guid,\n\t\t\terrorData\n\t\t})\n\n\t\t// Send error to error handler, except for 401s\n\t\tif (errorData.status !== HTTP_STATUS_CODE.UNAUTHORIZED) {\n\t\t\terrorHandler(error, modelFetchRequestAction, fetchConfig, lastFetchResult, errorData)\n\t\t}\n\t}\n}\n\n/**\n * The loop saga that makes the request every {action.period} milliseconds until cancelled\n *\n * @param action An action with the request configuration\n */\nexport function* modelFetchLoop(action: PeriodicModelFetchRequestAction): SagaIterator {\n\ttry {\n\t\tlet hasFetched = false\n\t\tdo {\n\t\t\t// call the fetch if this is the first loop iteration,\n\t\t\t// or if the user is not idle\n\t\t\tif (!hasFetched || !isUserIdle()) {\n\t\t\t\tyield call(modelFetch, action)\n\t\t\t\thasFetched = true\n\t\t\t}\n\t\t\tyield delay(action.period)\n\t\t} while (true)\n\t} catch (caughtError: unknown) {\n\t\tconst error = constructErrorFromCaughtError(caughtError, 'ModelFetchLoopError', 'modelFetchLoop failed')\n\t\tlogger.error(error)\n\t\t// catch and log any unexpected errors not caught inside `fetchData`\n\t\terrorHandler(error, action)\n\t} finally {\n\t\tif (yield cancelled()) {\n\t\t\tyield put<PeriodicModelFetchTerminationAction>({\n\t\t\t\ttype: PERIODIC_MODEL_FETCH_TERMINATION_ACTION_TYPE.SUCCEEDED,\n\t\t\t\ttaskId: action.taskId\n\t\t\t})\n\t\t}\n\t}\n}\n\n/**\n * Call the fetchData saga every {action.period} milliseconds. This saga requires the 'period' and 'taskId' properties\n * on the action parameter.\n *\n * @param action An action with the request configuration\n */\nexport function* modelFetchRecurring(action: PeriodicModelFetchRequestAction): SagaIterator {\n\tif (!action.period) {\n\t\tthrow new Error(\"'period' is required\")\n\t}\n\tif (!action.taskId) {\n\t\tthrow new Error(\"'taskId' is required\")\n\t}\n\n\tconst bgSyncTask: Task = yield fork(modelFetchLoop, action)\n\tyield take(takeMatchesTerminationAction(action))\n\tyield cancel(bgSyncTask)\n}\n\n/**\n * The main saga for fetching models. Must be initialized with an EndpointMappings object that can be fetched\n * and an API root to prepend to any partial URLs specified in the EndpointMappings object. A logger should normally be provided\n * as well.\n *\n * `EndpointMappings` object require a form as follows (with optional nested models):\n * ```\n * {\n * \tfryModel: {\n * \t\tpath: '/api/Foo'\n * \t},\n * \tgroupOfModels: {\n * \t\tleelaModel: {\n * \t\t\tpath: '/api/Bar'\n * \t\t},\n * \t\tbenderModel: {\n * \t\t\tpath: '/api/Baz'\n * \t\t}\n * \t}\n * }\n * ```\n * `EndpointMapping` objects are referenced in the actions.DATA_REQUESTED action by path, i.e.\n * `{ type: actions.DATA_REQUESTED, { modelName: 'fryModel' } }`\n * -- or --\n * `{ type: actions.DATA_REQUESTED, { modelName: 'groupOfModels.leelaModel' } }`\n *\n * @export\n * @param endpointMappingsParam A mapping of API endpoints available in the application\n * @param apiRootParam A url to which partial URLs are appended (i.e.) 'https://myapp.com'\n * @param tokenAccessFunctionParam function that returns an optional OAuth token\n * @param errorHandlerParam A function that is called when any fetch exceptions occur\n */\nexport default function* modelFetchSaga(\n\tendpointMappingsParam: EndpointMappings,\n\tapiRootParam?: string,\n\ttokenAccessFunctionParam: TokenAccessFunction | undefined = defaultTokenAccessFunction,\n\terrorHandlerParam: ErrorHandler | undefined = defaultErrorHandler\n): SagaIterator {\n\tsetApiRoot(apiRootParam)\n\tsetEndpointMappings(endpointMappingsParam)\n\terrorHandler = errorHandlerParam\n\ttokenAccessFunction = tokenAccessFunctionParam\n\tlogger = getLogger()\n\n\tyield takeEvery(MODEL_FETCH_REQUEST_ACTION_TYPE.FETCH_REQUEST, modelFetch)\n\tyield takeEvery(MODEL_FETCH_REQUEST_ACTION_TYPE.PERIODIC_FETCH_REQUEST, modelFetchRecurring)\n}\n"],"mappings":"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","ignoreList":[]}
|