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.
Files changed (258) hide show
  1. package/lib/components/ActionList.js +170 -34
  2. package/lib/components/AlertDialog.js +133 -9
  3. package/lib/components/AlertWithIcon.js +92 -25
  4. package/lib/components/ConnectedModal.js +43 -13
  5. package/lib/components/Dropdowns/GroupsDropdown.js +69 -44
  6. package/lib/components/Dropdowns/ManagedNavDropdown.js +100 -67
  7. package/lib/components/Dropdowns/UserDropdown.js +111 -21
  8. package/lib/components/Dropdowns/index.js +27 -9
  9. package/lib/components/EntityOwnerList.js +52 -18
  10. package/lib/components/Error.js +106 -11
  11. package/lib/components/ErrorBoundary.js +134 -35
  12. package/lib/components/ErrorMessage.js +44 -9
  13. package/lib/components/Forms/DateField.js +61 -42
  14. package/lib/components/Forms/TimeField.js +81 -42
  15. package/lib/components/Forms/index.js +27 -4
  16. package/lib/components/Groups/CreateEditCopySaveButtons.js +114 -11
  17. package/lib/components/Groups/ExternalGroups/Attach.js +212 -148
  18. package/lib/components/Groups/ExternalGroups/Table.js +181 -45
  19. package/lib/components/Groups/GroupCreateOrEditCommonProps.js +5 -1
  20. package/lib/components/Groups/RosterSyncInfo.js +147 -20
  21. package/lib/components/HOC/AccessibleAppComponent.js +98 -73
  22. package/lib/components/HOC/ActivityRequiredComponent.js +92 -49
  23. package/lib/components/HOC/AsyncComponent.js +54 -39
  24. package/lib/components/HOC/AuthenticatedComponent.js +58 -38
  25. package/lib/components/HOC/CollectionComponent.js +170 -110
  26. package/lib/components/HOC/CollectionFirstItemComponent.js +51 -38
  27. package/lib/components/HOC/CollectionItemComponent.js +168 -106
  28. package/lib/components/HOC/ConnectedModalComponent.js +109 -80
  29. package/lib/components/HOC/DataDependentComponent.js +29 -21
  30. package/lib/components/HOC/EntityComponent.js +71 -57
  31. package/lib/components/HOC/FullscreenModalComponent.js +163 -123
  32. package/lib/components/HOC/GroupActivityRequiredComponent.js +45 -31
  33. package/lib/components/HOC/GuidComponent.js +29 -22
  34. package/lib/components/HOC/ModelContextDependencyVerifyComponent.js +41 -31
  35. package/lib/components/HOC/ModelErrorRedirectComponent.js +51 -47
  36. package/lib/components/HOC/SearchPersistorComponent.js +240 -166
  37. package/lib/components/HOC/UnauthenticatedComponent.js +37 -25
  38. package/lib/components/HOC/UserComponent.js +12 -7
  39. package/lib/components/Icons/IconAlphaList.js +33 -5
  40. package/lib/components/Icons/IconExternalUser.js +33 -5
  41. package/lib/components/Icons/IconImpersonation.js +33 -5
  42. package/lib/components/Icons/IconStopImpersonating.js +33 -5
  43. package/lib/components/Icons/IconTable.js +35 -7
  44. package/lib/components/Icons/IconTableDeleteCol.js +33 -5
  45. package/lib/components/Icons/IconTableDeleteRow.js +33 -5
  46. package/lib/components/Icons/IconTableInsertCol.js +33 -5
  47. package/lib/components/Icons/IconTableInsertRow.js +33 -5
  48. package/lib/components/Impersonation/Button.js +77 -13
  49. package/lib/components/Impersonation/Link.js +77 -13
  50. package/lib/components/Impersonation/UserDetail.js +66 -9
  51. package/lib/components/Loading.js +26 -4
  52. package/lib/components/LockDownBrowser/Check.js +194 -49
  53. package/lib/components/LockDownBrowser/ExitButton.js +26 -9
  54. package/lib/components/LockDownBrowser/Launch.js +70 -62
  55. package/lib/components/Lti/Confirm.js +152 -11
  56. package/lib/components/Lti/CreateNonLtiGroupAlertDialog.js +170 -33
  57. package/lib/components/Lti/Launch.js +105 -24
  58. package/lib/components/Lti/LaunchGroup.js +85 -13
  59. package/lib/components/ManageTable.js +309 -87
  60. package/lib/components/ManageTableNoDataComponent.js +42 -4
  61. package/lib/components/NewVersionAlert.js +82 -46
  62. package/lib/components/NotFound.js +86 -10
  63. package/lib/components/Notifications.js +185 -126
  64. package/lib/components/PaginationNextButton.js +33 -6
  65. package/lib/components/PaginationPreviousButton.js +33 -6
  66. package/lib/components/Quill/CustomToolbar.js +432 -218
  67. package/lib/components/Quill/Formats/Image.js +73 -63
  68. package/lib/components/Quill/Formats/List.js +45 -45
  69. package/lib/components/Quill/Formats/Video.js +28 -24
  70. package/lib/components/Quill/ImageDropModule.js +147 -117
  71. package/lib/components/Quill/ImageWarning.js +47 -9
  72. package/lib/components/Quill/ImageWithAltTextModal.js +425 -86
  73. package/lib/components/Quill/Specs/CustomImageSpec.js +42 -34
  74. package/lib/components/Quill/Specs/CustomVideoSpec.js +34 -28
  75. package/lib/components/Quill/TableModule/Blots/BaseTableBlot.js +98 -98
  76. package/lib/components/Quill/TableModule/Blots/TableBlot.js +52 -47
  77. package/lib/components/Quill/TableModule/Blots/TableBodyBlot.js +53 -48
  78. package/lib/components/Quill/TableModule/Blots/TableCellBlot.js +224 -221
  79. package/lib/components/Quill/TableModule/Blots/TableContainer.js +80 -83
  80. package/lib/components/Quill/TableModule/Blots/TableRowBlot.js +75 -70
  81. package/lib/components/Quill/TableModule/constants.js +45 -41
  82. package/lib/components/Quill/TableModule/index.js +362 -301
  83. package/lib/components/Quill/TableModule/utils.js +42 -38
  84. package/lib/components/Quill/accessibilityFix.js +234 -232
  85. package/lib/components/Quill/index.js +34 -28
  86. package/lib/components/RefreshIndicator/Bordered.js +47 -6
  87. package/lib/components/RefreshIndicator/Inline.js +47 -8
  88. package/lib/components/RefreshIndicator/index.js +263 -59
  89. package/lib/components/SearchControls.js +216 -11
  90. package/lib/components/SentryRoute.js +11 -6
  91. package/lib/components/Tables/RoleFilter.js +69 -32
  92. package/lib/components/Tables/TextFilter.js +62 -13
  93. package/lib/components/UserRoles/Add.js +199 -96
  94. package/lib/components/UserRoles/Context.js +11 -7
  95. package/lib/components/UserRoles/RoleCell.js +181 -72
  96. package/lib/components/UserRoles/Select.js +157 -17
  97. package/lib/components/UserRoles/Table.js +221 -80
  98. package/lib/components/UserRoles/index.js +534 -384
  99. package/lib/config/eslint/index.js +32 -28
  100. package/lib/config/eslint/lib/order.js +26 -27
  101. package/lib/config/eslint/lib/prettier.js +20 -18
  102. package/lib/config/eslint/lib/typescript.js +93 -112
  103. package/lib/config/eslint/react.js +24 -14
  104. package/lib/constants/baseActivity.js +30 -26
  105. package/lib/constants/baseRole.js +14 -10
  106. package/lib/constants/configuration.js +33 -29
  107. package/lib/constants/externalProviderType.js +10 -6
  108. package/lib/constants/fetchErrorData.js +15 -11
  109. package/lib/constants/index.js +137 -14
  110. package/lib/constants/lockDownBrowser.js +28 -24
  111. package/lib/constants/mockData.js +382 -297
  112. package/lib/constants/modelStatus.js +15 -11
  113. package/lib/constants/notificationType.js +12 -8
  114. package/lib/constants/operatingSystem.js +12 -8
  115. package/lib/constants/shard.js +11 -7
  116. package/lib/constants/table.js +21 -21
  117. package/lib/constants/tier.js +12 -8
  118. package/lib/constants/userRole.js +15 -5
  119. package/lib/endpointMappings.js +197 -181
  120. package/lib/hooks/useCollection.js +82 -62
  121. package/lib/hooks/useCollectionConfiguration.js +228 -83
  122. package/lib/hooks/useCollectionItem.js +154 -54
  123. package/lib/hooks/useGuid.js +20 -8
  124. package/lib/hooks/usePrevious.js +19 -13
  125. package/lib/index.js +157 -25
  126. package/lib/redux/actionCreator.js +50 -28
  127. package/lib/redux/actions/AuthAction.js +44 -31
  128. package/lib/redux/actions/ModalAction.js +10 -6
  129. package/lib/redux/actions/ModelAction.js +77 -39
  130. package/lib/redux/actions/NotificationAction.js +10 -6
  131. package/lib/redux/actions/SearchAction.js +9 -5
  132. package/lib/redux/actions/index.js +60 -7
  133. package/lib/redux/configureReducers.js +60 -49
  134. package/lib/redux/configureStore.js +83 -87
  135. package/lib/redux/helpers.js +6 -2
  136. package/lib/redux/reducers/authReducer.js +50 -43
  137. package/lib/redux/reducers/index.js +41 -13
  138. package/lib/redux/reducers/modalsReducer.js +47 -29
  139. package/lib/redux/reducers/modelsReducer.js +178 -173
  140. package/lib/redux/reducers/notificationsReducer.js +24 -18
  141. package/lib/redux/reducers/searchReducer.js +25 -19
  142. package/lib/redux/sagas/appInsightsSaga.js +22 -18
  143. package/lib/redux/sagas/authSaga.js +253 -218
  144. package/lib/redux/sagas/caliperSaga.js +159 -143
  145. package/lib/redux/sagas/clockOffsetSaga.js +34 -31
  146. package/lib/redux/sagas/configurationSaga.js +11 -7
  147. package/lib/redux/sagas/downtimeApiErrorSaga.js +20 -17
  148. package/lib/redux/sagas/errorSaga.js +27 -21
  149. package/lib/redux/sagas/googleAnalyticsSaga.js +28 -24
  150. package/lib/redux/sagas/identityProviderSaga.js +22 -18
  151. package/lib/redux/sagas/initialDataLoadSaga.js +37 -28
  152. package/lib/redux/sagas/lockDownBrowserErrorSaga.js +29 -20
  153. package/lib/redux/sagas/modelFetchSaga.js +355 -322
  154. package/lib/redux/sagas/noStoreSaga.js +61 -48
  155. package/lib/redux/sagas/postLoginDataSaga.js +45 -34
  156. package/lib/redux/sagas/postLoginRedirectSaga.js +27 -27
  157. package/lib/redux/sagas/rootSaga.js +82 -57
  158. package/lib/redux/sagas/sentrySaga.js +29 -25
  159. package/lib/redux/sagas/userIdSaga.js +16 -12
  160. package/lib/services/codeProviderService.js +25 -19
  161. package/lib/services/dateService.js +12 -7
  162. package/lib/services/documentService.js +17 -12
  163. package/lib/services/fetchService.js +129 -112
  164. package/lib/services/persistenceService.js +33 -29
  165. package/lib/services/ticketProviderService.js +29 -23
  166. package/lib/services/tokenPersistenceService.js +12 -8
  167. package/lib/services/windowService.js +18 -14
  168. package/lib/startup.js +132 -114
  169. package/lib/types/AppConfiguration.js +5 -1
  170. package/lib/types/Artifact.js +11 -7
  171. package/lib/types/BaseReduxState.js +5 -1
  172. package/lib/types/Client.js +5 -1
  173. package/lib/types/Collection.js +5 -1
  174. package/lib/types/Configuration.js +5 -1
  175. package/lib/types/DeepLinkingResponseRequest.js +5 -1
  176. package/lib/types/DeletableModel.js +5 -1
  177. package/lib/types/Event.js +5 -1
  178. package/lib/types/ExternalGroup.js +5 -1
  179. package/lib/types/ExternalProvider.js +5 -1
  180. package/lib/types/ExternalTerm.js +5 -1
  181. package/lib/types/Group.js +5 -1
  182. package/lib/types/IdentityProvider.js +5 -1
  183. package/lib/types/LtiLaunch.js +5 -1
  184. package/lib/types/NameOnlyEntity.js +5 -1
  185. package/lib/types/Notification.js +5 -1
  186. package/lib/types/OptionalRecord.js +5 -1
  187. package/lib/types/OwnerSchedule.js +5 -1
  188. package/lib/types/PropertyOfType.js +5 -1
  189. package/lib/types/Quill.js +5 -1
  190. package/lib/types/RoleDescription.js +5 -1
  191. package/lib/types/Search.js +5 -1
  192. package/lib/types/SimpleLocation.js +5 -1
  193. package/lib/types/UniTime.js +5 -1
  194. package/lib/types/User.js +5 -1
  195. package/lib/types/UserRole.js +5 -1
  196. package/lib/types/auth/AuthState.js +5 -1
  197. package/lib/types/auth/CasV1LoginRequestBody.js +5 -1
  198. package/lib/types/auth/ClientCredentials.js +5 -1
  199. package/lib/types/auth/CodeProviderService.js +5 -1
  200. package/lib/types/auth/LocalLoginRequestBody.js +5 -1
  201. package/lib/types/auth/TicketProviderService.js +5 -1
  202. package/lib/types/auth/TokenPersistenceService.js +5 -1
  203. package/lib/types/auth/index.js +82 -9
  204. package/lib/types/externals.d.js +2 -0
  205. package/lib/types/index.js +313 -30
  206. package/lib/types/net/EndpointConfig.js +5 -1
  207. package/lib/types/net/EndpointMapping.js +5 -1
  208. package/lib/types/net/EndpointMappings.js +5 -1
  209. package/lib/types/net/ErrorHandler.js +5 -1
  210. package/lib/types/net/FetchConfig.js +5 -1
  211. package/lib/types/net/FetchErrorData.js +10 -6
  212. package/lib/types/net/FetchResult.js +5 -1
  213. package/lib/types/net/HTTPMethod.js +5 -1
  214. package/lib/types/net/HTTPStatusCode.js +16 -12
  215. package/lib/types/net/Metadata.js +5 -1
  216. package/lib/types/net/Model.js +5 -1
  217. package/lib/types/net/ModelCollection.js +5 -1
  218. package/lib/types/net/ModelsState.js +5 -1
  219. package/lib/types/net/OAuthToken.js +5 -1
  220. package/lib/types/net/OAuthTokenOrNull.js +5 -1
  221. package/lib/types/net/TokenAccessFunction.js +5 -1
  222. package/lib/types/net/index.js +181 -18
  223. package/lib/utils/baseActivity.js +133 -123
  224. package/lib/utils/baseRole.js +37 -33
  225. package/lib/utils/collection.js +425 -298
  226. package/lib/utils/cookies.js +22 -19
  227. package/lib/utils/date.js +303 -279
  228. package/lib/utils/dom.js +176 -165
  229. package/lib/utils/domainIdentifier.js +9 -5
  230. package/lib/utils/entityUserRole.js +6 -2
  231. package/lib/utils/error.js +17 -15
  232. package/lib/utils/events.js +40 -31
  233. package/lib/utils/externalGroup.js +22 -18
  234. package/lib/utils/externalProviders.js +8 -4
  235. package/lib/utils/externalTerms.js +10 -3
  236. package/lib/utils/fetch.js +179 -180
  237. package/lib/utils/group.js +18 -7
  238. package/lib/utils/groupDates.js +37 -33
  239. package/lib/utils/groupRoles.js +25 -21
  240. package/lib/utils/lockDownBrowser.js +15 -11
  241. package/lib/utils/logger.js +26 -22
  242. package/lib/utils/lti.js +9 -4
  243. package/lib/utils/model.js +36 -41
  244. package/lib/utils/number.js +21 -18
  245. package/lib/utils/promise.js +28 -21
  246. package/lib/utils/quill.js +65 -62
  247. package/lib/utils/route.js +58 -55
  248. package/lib/utils/search.js +76 -80
  249. package/lib/utils/shard.js +37 -37
  250. package/lib/utils/sort.js +50 -42
  251. package/lib/utils/string.js +13 -8
  252. package/lib/utils/table.js +38 -33
  253. package/lib/utils/timezone.js +10 -6
  254. package/lib/utils/url.js +142 -142
  255. package/lib/utils/user.js +58 -55
  256. package/lib/utils/userAgent.js +10 -10
  257. package/lib/utils/userRole.js +57 -49
  258. package/package.json +17 -3
@@ -1,380 +1,413 @@
1
1
  "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.defaultErrorHandler = exports.defaultTokenAccessFunction = exports.takeMatchesTerminationAction = exports.matchesTerminationAction = exports.TRY_LIMIT = void 0;
4
- exports.setEndpointMappings = setEndpointMappings;
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.default = modelFetchSaga;
9
- const tslib_1 = require("tslib");
10
- const lodash_1 = require("lodash");
11
- const moment_timezone_1 = tslib_1.__importDefault(require("moment-timezone"));
12
- const effects_1 = require("redux-saga/effects");
13
- const fetchErrorData_1 = require("../../constants/fetchErrorData");
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");
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
- const matchesTerminationAction = (incomingAction, dataRequestAction) => incomingAction.type === actions_1.PERIODIC_MODEL_FETCH_TERMINATION_ACTION_TYPE.TERMINATE &&
26
- incomingAction.taskId === dataRequestAction.taskId;
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 = (dataRequestAction) => (incomingAction) => (0, exports.matchesTerminationAction)(incomingAction, dataRequestAction);
29
- exports.takeMatchesTerminationAction = takeMatchesTerminationAction;
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
- return null;
40
+ return null;
34
41
  };
35
- exports.defaultTokenAccessFunction = defaultTokenAccessFunction;
42
+
36
43
  /* istanbul ignore next */
44
+ exports.defaultTokenAccessFunction = defaultTokenAccessFunction;
37
45
  const defaultErrorHandler = () => {
38
- return;
46
+ return;
39
47
  };
40
48
  exports.defaultErrorHandler = defaultErrorHandler;
41
- const getChildCollectionKeys = (endpointMapping) => Object.keys(endpointMapping).reduce((out, key) => {
42
- const value = endpointMapping[key];
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
- }
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
- const { endpointConfig, fetchConfig, isCollectionItemCreate, isCollectionItemFetch } = prepareFetchResult;
54
- const childCollectionKeys = getChildCollectionKeys(endpointMapping);
55
- const hasChildCollections = childCollectionKeys.length > 0;
56
- // no collections, return as-is
57
- if (!endpointConfig.isCollection && !hasChildCollections) {
58
- return data;
59
- }
60
- // delete returns no data
61
- if (fetchConfig.method === 'DELETE') {
62
- return {};
63
- }
64
- // data is a collection item, convert any child collections in item response
65
- if (isCollectionItemFetch || isCollectionItemCreate) {
66
- return convertDataCollectionItem(data, childCollectionKeys, endpointMapping);
67
- }
68
- // data is a collection, convert collection response, and any child collections in each item
69
- return endpointConfig.isCollection
70
- ? convertDataToCollection(data, childCollectionKeys, endpointMapping)
71
- : convertDataCollectionItem(data, childCollectionKeys, endpointMapping);
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
- if ((0, lodash_1.isNil)(data)) {
75
- return data;
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
- return Object.keys(data).reduce((out, key) => {
78
- let value = data[key];
79
- if (collectionKeys.includes(key)) {
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
- if ((0, lodash_1.isNil)(data)) {
90
- return data;
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 Object.keys(data).reduce((out, key) => {
93
- const value = convertDataCollectionItem(data[key], collectionKeys, endpointMapping);
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
- * Using the `lastActivityDate` set in local storage by "studiokit-caliper-js",
103
- * determine if the user has been idle, e.g. has had no activity in the last 15 minutes.
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
- const lastActivityDateString = (0, dateService_1.getLastActivityDate)();
107
- if (!lastActivityDateString) {
108
- return false;
109
- }
110
- const lastActivityDateUtc = moment_timezone_1.default.utc(lastActivityDateString);
111
- const nowUtc = moment_timezone_1.default.utc();
112
- const idleMinutesThreshold = 1000 * 60 * 15; // 15 minutes
113
- return nowUtc.diff(lastActivityDateUtc) >= idleMinutesThreshold;
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
- endpointMappings = endpointMappingsParam;
142
+ endpointMappings = endpointMappingsParam;
125
143
  }
126
- /**
127
- * Construct a request based on the provided dataRequestAction, make a request with a configurable retry,
128
- * and handle errors, logging and dispatching all steps.
129
- *
130
- * @param modelFetchRequestAction A model fetch request action with the request configuration
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
- let prepareFetchResult = undefined;
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
- prepareFetchResult = (0, fetch_1.prepareFetch)(modelFetchRequestAction, endpointMappings);
136
- }
137
- catch (error) {
138
- // Note: `prepareFetch` already ensures that `error` is an actual `Error`
139
- logger.error(error);
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
- let fetchResult = undefined;
167
- try {
168
- // get the OAuth Token, if any, and call `fetch`
169
- const oauthToken = yield (0, effects_1.call)(tokenAccessFunction, modelFetchRequestAction.modelName);
170
- if (oauthToken === null || oauthToken === void 0 ? void 0 : oauthToken.access_token) {
171
- fetchConfig.headers = (0, lodash_1.merge)({}, fetchConfig.headers, {
172
- Authorization: `Bearer ${oauthToken.access_token}`
173
- });
174
- }
175
- fetchResult = yield (0, effects_1.call)(fetchService_1.sendFetch, fetchConfig);
176
- // store `fetchResult` for reference after the do-while loop
177
- lastFetchResult = fetchResult;
178
- // throw on unsuccessful result to short-circuit and trigger catch+retry
179
- if (!fetchResult)
180
- throw new Error(fetchResultUndefinedErrorMessage);
181
- if (!fetchResult.ok)
182
- throw new Error(fetchResultNotOkErrorMessage);
183
- // handle success
184
- const resultActionType = modelFetchRequestAction.noStore
185
- ? actions_1.MODEL_FETCH_RESULT_ACTION_TYPE.TRANSIENT_FETCH_RESULT_RECEIVED
186
- : actions_1.MODEL_FETCH_RESULT_ACTION_TYPE.FETCH_RESULT_RECEIVED;
187
- const data = convertCollections(endpointMapping, prepareFetchResult, fetchResult.data);
188
- // attach guid to result data if it is an object
189
- if (modelFetchRequestAction.guid && (0, lodash_1.isPlainObject)(data)) {
190
- data.guid = modelFetchRequestAction.guid;
191
- }
192
- // POST new collection item
193
- if (isCollectionItemCreate) {
194
- const modelPathLevels = modelPath.split('.');
195
- // remove guid
196
- modelPathLevels.pop();
197
- // add by new result's id
198
- yield (0, effects_1.put)({
199
- type: resultActionType,
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
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
- // Send error to error handler, except for 401s
285
- if (errorData.status !== types_1.HTTP_STATUS_CODE.UNAUTHORIZED) {
286
- errorHandler(error, modelFetchRequestAction, fetchConfig, lastFetchResult, errorData);
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
- * The loop saga that makes the request every {action.period} milliseconds until cancelled
292
- *
293
- * @param action An action with the request configuration
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
- try {
297
- let hasFetched = false;
298
- do {
299
- // call the fetch if this is the first loop iteration,
300
- // or if the user is not idle
301
- if (!hasFetched || !isUserIdle()) {
302
- yield (0, effects_1.call)(modelFetch, action);
303
- hasFetched = true;
304
- }
305
- yield (0, effects_1.delay)(action.period);
306
- } while (true);
307
- }
308
- catch (caughtError) {
309
- const error = (0, error_1.constructErrorFromCaughtError)(caughtError, 'ModelFetchLoopError', 'modelFetchLoop failed');
310
- logger.error(error);
311
- // catch and log any unexpected errors not caught inside `fetchData`
312
- errorHandler(error, action);
313
- }
314
- finally {
315
- if (yield (0, effects_1.cancelled)()) {
316
- yield (0, effects_1.put)({
317
- type: actions_1.PERIODIC_MODEL_FETCH_TERMINATION_ACTION_TYPE.SUCCEEDED,
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
- * Call the fetchData saga every {action.period} milliseconds. This saga requires the 'period' and 'taskId' properties
325
- * on the action parameter.
326
- *
327
- * @param action An action with the request configuration
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
- if (!action.period) {
331
- throw new Error("'period' is required");
332
- }
333
- if (!action.taskId) {
334
- throw new Error("'taskId' is required");
335
- }
336
- const bgSyncTask = yield (0, effects_1.fork)(modelFetchLoop, action);
337
- yield (0, effects_1.take)((0, exports.takeMatchesTerminationAction)(action));
338
- yield (0, effects_1.cancel)(bgSyncTask);
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
- * The main saga for fetching models. Must be initialized with an EndpointMappings object that can be fetched
342
- * and an API root to prepend to any partial URLs specified in the EndpointMappings object. A logger should normally be provided
343
- * as well.
344
- *
345
- * `EndpointMappings` object require a form as follows (with optional nested models):
346
- * ```
347
- * {
348
- * fryModel: {
349
- * path: '/api/Foo'
350
- * },
351
- * groupOfModels: {
352
- * leelaModel: {
353
- * path: '/api/Bar'
354
- * },
355
- * benderModel: {
356
- * path: '/api/Baz'
357
- * }
358
- * }
359
- * }
360
- * ```
361
- * `EndpointMapping` objects are referenced in the actions.DATA_REQUESTED action by path, i.e.
362
- * `{ type: actions.DATA_REQUESTED, { modelName: 'fryModel' } }`
363
- * -- or --
364
- * `{ type: actions.DATA_REQUESTED, { modelName: 'groupOfModels.leelaModel' } }`
365
- *
366
- * @export
367
- * @param endpointMappingsParam A mapping of API endpoints available in the application
368
- * @param apiRootParam A url to which partial URLs are appended (i.e.) 'https://myapp.com'
369
- * @param tokenAccessFunctionParam function that returns an optional OAuth token
370
- * @param errorHandlerParam A function that is called when any fetch exceptions occur
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* modelFetchSaga(endpointMappingsParam, apiRootParam, tokenAccessFunctionParam = exports.defaultTokenAccessFunction, errorHandlerParam = exports.defaultErrorHandler) {
373
- (0, fetchService_1.setApiRoot)(apiRootParam);
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, logger_1.getLogger)();
378
- yield (0, effects_1.takeEvery)(actions_1.MODEL_FETCH_REQUEST_ACTION_TYPE.FETCH_REQUEST, modelFetch);
379
- yield (0, effects_1.takeEvery)(actions_1.MODEL_FETCH_REQUEST_ACTION_TYPE.PERIODIC_FETCH_REQUEST, modelFetchRecurring);
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":[]}