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