studiokit-scaffolding-js 7.0.12-next.1.3 → 7.0.12-next.2.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (258) hide show
  1. package/lib/components/ActionList.js +164 -37
  2. package/lib/components/AlertDialog.js +128 -12
  3. package/lib/components/AlertWithIcon.js +88 -29
  4. package/lib/components/ConnectedModal.js +35 -12
  5. package/lib/components/Dropdowns/GroupsDropdown.js +63 -45
  6. package/lib/components/Dropdowns/ManagedNavDropdown.js +92 -67
  7. package/lib/components/Dropdowns/UserDropdown.js +105 -24
  8. package/lib/components/Dropdowns/index.js +4 -10
  9. package/lib/components/EntityOwnerList.js +47 -21
  10. package/lib/components/Error.js +101 -12
  11. package/lib/components/ErrorBoundary.js +127 -38
  12. package/lib/components/ErrorMessage.js +39 -12
  13. package/lib/components/Forms/DateField.js +56 -45
  14. package/lib/components/Forms/TimeField.js +76 -45
  15. package/lib/components/Forms/index.js +3 -5
  16. package/lib/components/Groups/CreateEditCopySaveButtons.js +109 -14
  17. package/lib/components/Groups/ExternalGroups/Attach.js +206 -151
  18. package/lib/components/Groups/ExternalGroups/Table.js +176 -48
  19. package/lib/components/Groups/GroupCreateOrEditCommonProps.js +2 -2
  20. package/lib/components/Groups/RosterSyncInfo.js +142 -23
  21. package/lib/components/HOC/AccessibleAppComponent.js +88 -72
  22. package/lib/components/HOC/ActivityRequiredComponent.js +68 -33
  23. package/lib/components/HOC/AsyncComponent.js +49 -41
  24. package/lib/components/HOC/AuthenticatedComponent.js +55 -44
  25. package/lib/components/HOC/CollectionComponent.js +154 -104
  26. package/lib/components/HOC/CollectionFirstItemComponent.js +45 -40
  27. package/lib/components/HOC/CollectionItemComponent.js +152 -100
  28. package/lib/components/HOC/ConnectedModalComponent.js +87 -69
  29. package/lib/components/HOC/DataDependentComponent.js +26 -27
  30. package/lib/components/HOC/EntityComponent.js +57 -53
  31. package/lib/components/HOC/FullscreenModalComponent.js +139 -108
  32. package/lib/components/HOC/GroupActivityRequiredComponent.js +27 -20
  33. package/lib/components/HOC/GuidComponent.js +20 -20
  34. package/lib/components/HOC/ModelContextDependencyVerifyComponent.js +32 -29
  35. package/lib/components/HOC/ModelErrorRedirectComponent.js +37 -39
  36. package/lib/components/HOC/SearchPersistorComponent.js +237 -173
  37. package/lib/components/HOC/UnauthenticatedComponent.js +32 -30
  38. package/lib/components/HOC/UserComponent.js +6 -8
  39. package/lib/components/Icons/IconAlphaList.js +28 -8
  40. package/lib/components/Icons/IconExternalUser.js +28 -8
  41. package/lib/components/Icons/IconImpersonation.js +28 -8
  42. package/lib/components/Icons/IconStopImpersonating.js +28 -8
  43. package/lib/components/Icons/IconTable.js +29 -9
  44. package/lib/components/Icons/IconTableDeleteCol.js +28 -8
  45. package/lib/components/Icons/IconTableDeleteRow.js +28 -8
  46. package/lib/components/Icons/IconTableInsertCol.js +28 -8
  47. package/lib/components/Icons/IconTableInsertRow.js +28 -8
  48. package/lib/components/Impersonation/Button.js +71 -16
  49. package/lib/components/Impersonation/Link.js +72 -16
  50. package/lib/components/Impersonation/UserDetail.js +60 -11
  51. package/lib/components/Loading.js +23 -8
  52. package/lib/components/LockDownBrowser/Check.js +188 -51
  53. package/lib/components/LockDownBrowser/ExitButton.js +22 -13
  54. package/lib/components/LockDownBrowser/Launch.js +64 -64
  55. package/lib/components/Lti/Confirm.js +147 -14
  56. package/lib/components/Lti/CreateNonLtiGroupAlertDialog.js +165 -36
  57. package/lib/components/Lti/Launch.js +99 -25
  58. package/lib/components/Lti/LaunchGroup.js +81 -16
  59. package/lib/components/ManageTable.js +304 -90
  60. package/lib/components/ManageTableNoDataComponent.js +38 -7
  61. package/lib/components/NewVersionAlert.js +76 -49
  62. package/lib/components/NotFound.js +81 -11
  63. package/lib/components/Notifications.js +179 -129
  64. package/lib/components/PaginationNextButton.js +28 -9
  65. package/lib/components/PaginationPreviousButton.js +28 -9
  66. package/lib/components/Quill/CustomToolbar.js +427 -222
  67. package/lib/components/Quill/Formats/Image.js +67 -67
  68. package/lib/components/Quill/Formats/List.js +38 -47
  69. package/lib/components/Quill/Formats/Video.js +23 -26
  70. package/lib/components/Quill/ImageDropModule.js +136 -114
  71. package/lib/components/Quill/ImageWarning.js +41 -12
  72. package/lib/components/Quill/ImageWithAltTextModal.js +420 -89
  73. package/lib/components/Quill/Specs/CustomImageSpec.js +32 -31
  74. package/lib/components/Quill/Specs/CustomVideoSpec.js +22 -23
  75. package/lib/components/Quill/TableModule/Blots/BaseTableBlot.js +89 -97
  76. package/lib/components/Quill/TableModule/Blots/TableBlot.js +47 -50
  77. package/lib/components/Quill/TableModule/Blots/TableBodyBlot.js +48 -51
  78. package/lib/components/Quill/TableModule/Blots/TableCellBlot.js +219 -224
  79. package/lib/components/Quill/TableModule/Blots/TableContainer.js +75 -86
  80. package/lib/components/Quill/TableModule/Blots/TableRowBlot.js +70 -73
  81. package/lib/components/Quill/TableModule/constants.js +40 -42
  82. package/lib/components/Quill/TableModule/index.js +357 -305
  83. package/lib/components/Quill/TableModule/utils.js +39 -48
  84. package/lib/components/Quill/accessibilityFix.js +219 -223
  85. package/lib/components/Quill/index.js +30 -33
  86. package/lib/components/RefreshIndicator/Bordered.js +44 -10
  87. package/lib/components/RefreshIndicator/Inline.js +43 -12
  88. package/lib/components/RefreshIndicator/index.js +257 -62
  89. package/lib/components/SearchControls.js +211 -14
  90. package/lib/components/SentryRoute.js +5 -7
  91. package/lib/components/Tables/RoleFilter.js +66 -38
  92. package/lib/components/Tables/TextFilter.js +58 -18
  93. package/lib/components/UserRoles/Add.js +193 -99
  94. package/lib/components/UserRoles/Context.js +3 -6
  95. package/lib/components/UserRoles/RoleCell.js +176 -75
  96. package/lib/components/UserRoles/Select.js +151 -20
  97. package/lib/components/UserRoles/Table.js +215 -82
  98. package/lib/components/UserRoles/index.js +526 -386
  99. package/lib/config/eslint/index.js +26 -29
  100. package/lib/config/eslint/lib/order.js +21 -28
  101. package/lib/config/eslint/lib/prettier.js +15 -19
  102. package/lib/config/eslint/lib/typescript.js +87 -113
  103. package/lib/config/eslint/react.js +18 -15
  104. package/lib/constants/baseActivity.js +26 -28
  105. package/lib/constants/baseRole.js +10 -12
  106. package/lib/constants/configuration.js +43 -55
  107. package/lib/constants/externalProviderType.js +6 -8
  108. package/lib/constants/fetchErrorData.js +10 -12
  109. package/lib/constants/index.js +13 -15
  110. package/lib/constants/lockDownBrowser.js +23 -25
  111. package/lib/constants/mockData.js +370 -300
  112. package/lib/constants/modelStatus.js +11 -13
  113. package/lib/constants/notificationType.js +8 -10
  114. package/lib/constants/operatingSystem.js +8 -10
  115. package/lib/constants/shard.js +7 -9
  116. package/lib/constants/table.js +18 -22
  117. package/lib/constants/tier.js +8 -10
  118. package/lib/constants/userRole.js +11 -8
  119. package/lib/endpointMappings.js +191 -182
  120. package/lib/hooks/useCollection.js +79 -65
  121. package/lib/hooks/useCollectionConfiguration.js +220 -80
  122. package/lib/hooks/useCollectionItem.js +151 -57
  123. package/lib/hooks/useGuid.js +16 -9
  124. package/lib/hooks/usePrevious.js +14 -13
  125. package/lib/index.js +11 -26
  126. package/lib/redux/actionCreator.js +44 -35
  127. package/lib/redux/actions/AuthAction.js +45 -32
  128. package/lib/redux/actions/ModalAction.js +6 -8
  129. package/lib/redux/actions/ModelAction.js +95 -43
  130. package/lib/redux/actions/NotificationAction.js +6 -8
  131. package/lib/redux/actions/SearchAction.js +5 -7
  132. package/lib/redux/actions/index.js +6 -8
  133. package/lib/redux/configureReducers.js +48 -46
  134. package/lib/redux/configureStore.js +77 -91
  135. package/lib/redux/helpers.js +2 -5
  136. package/lib/redux/reducers/authReducer.js +44 -43
  137. package/lib/redux/reducers/index.js +7 -14
  138. package/lib/redux/reducers/modalsReducer.js +43 -31
  139. package/lib/redux/reducers/modelsReducer.js +131 -137
  140. package/lib/redux/reducers/notificationsReducer.js +20 -20
  141. package/lib/redux/reducers/searchReducer.js +13 -13
  142. package/lib/redux/sagas/appInsightsSaga.js +19 -21
  143. package/lib/redux/sagas/authSaga.js +248 -234
  144. package/lib/redux/sagas/caliperSaga.js +142 -131
  145. package/lib/redux/sagas/clockOffsetSaga.js +29 -32
  146. package/lib/redux/sagas/configurationSaga.js +8 -10
  147. package/lib/redux/sagas/downtimeApiErrorSaga.js +16 -19
  148. package/lib/redux/sagas/errorSaga.js +23 -24
  149. package/lib/redux/sagas/googleAnalyticsSaga.js +24 -27
  150. package/lib/redux/sagas/identityProviderSaga.js +19 -21
  151. package/lib/redux/sagas/initialDataLoadSaga.js +34 -31
  152. package/lib/redux/sagas/lockDownBrowserErrorSaga.js +25 -22
  153. package/lib/redux/sagas/modelFetchSaga.js +302 -286
  154. package/lib/redux/sagas/noStoreSaga.js +60 -61
  155. package/lib/redux/sagas/postLoginDataSaga.js +37 -32
  156. package/lib/redux/sagas/postLoginRedirectSaga.js +22 -27
  157. package/lib/redux/sagas/rootSaga.js +77 -60
  158. package/lib/redux/sagas/sentrySaga.js +25 -28
  159. package/lib/redux/sagas/userIdSaga.js +13 -15
  160. package/lib/services/codeProviderService.js +21 -21
  161. package/lib/services/dateService.js +6 -8
  162. package/lib/services/documentService.js +10 -11
  163. package/lib/services/fetchService.js +103 -95
  164. package/lib/services/persistenceService.js +27 -30
  165. package/lib/services/ticketProviderService.js +25 -25
  166. package/lib/services/tokenPersistenceService.js +8 -10
  167. package/lib/services/windowService.js +14 -16
  168. package/lib/startup.js +110 -101
  169. package/lib/types/AppConfiguration.js +2 -2
  170. package/lib/types/Artifact.js +7 -9
  171. package/lib/types/BaseReduxState.js +2 -2
  172. package/lib/types/Client.js +2 -2
  173. package/lib/types/Collection.js +2 -2
  174. package/lib/types/Configuration.js +2 -2
  175. package/lib/types/DeepLinkingResponseRequest.js +2 -2
  176. package/lib/types/DeletableModel.js +2 -2
  177. package/lib/types/Event.js +2 -2
  178. package/lib/types/ExternalGroup.js +2 -2
  179. package/lib/types/ExternalProvider.js +2 -2
  180. package/lib/types/ExternalTerm.js +2 -2
  181. package/lib/types/Group.js +2 -2
  182. package/lib/types/IdentityProvider.js +2 -2
  183. package/lib/types/LtiLaunch.js +2 -2
  184. package/lib/types/NameOnlyEntity.js +2 -2
  185. package/lib/types/Notification.js +2 -2
  186. package/lib/types/OptionalRecord.js +2 -2
  187. package/lib/types/OwnerSchedule.js +2 -2
  188. package/lib/types/PropertyOfType.js +2 -2
  189. package/lib/types/Quill.js +2 -2
  190. package/lib/types/RoleDescription.js +2 -2
  191. package/lib/types/Search.js +2 -2
  192. package/lib/types/SimpleLocation.js +2 -2
  193. package/lib/types/UniTime.js +2 -2
  194. package/lib/types/User.js +2 -2
  195. package/lib/types/UserRole.js +2 -2
  196. package/lib/types/auth/AuthState.js +2 -2
  197. package/lib/types/auth/CasV1LoginRequestBody.js +2 -2
  198. package/lib/types/auth/ClientCredentials.js +2 -2
  199. package/lib/types/auth/CodeProviderService.js +2 -2
  200. package/lib/types/auth/LocalLoginRequestBody.js +2 -2
  201. package/lib/types/auth/TicketProviderService.js +2 -2
  202. package/lib/types/auth/TokenPersistenceService.js +2 -2
  203. package/lib/types/auth/index.js +8 -10
  204. package/lib/types/externals.d.js +2 -0
  205. package/lib/types/index.js +29 -31
  206. package/lib/types/net/EndpointConfig.js +2 -2
  207. package/lib/types/net/EndpointMapping.js +2 -2
  208. package/lib/types/net/EndpointMappings.js +2 -2
  209. package/lib/types/net/ErrorHandler.js +2 -2
  210. package/lib/types/net/FetchConfig.js +2 -2
  211. package/lib/types/net/FetchErrorData.js +6 -8
  212. package/lib/types/net/FetchResult.js +2 -2
  213. package/lib/types/net/HTTPMethod.js +2 -2
  214. package/lib/types/net/HTTPStatusCode.js +12 -14
  215. package/lib/types/net/Metadata.js +2 -2
  216. package/lib/types/net/Model.js +2 -2
  217. package/lib/types/net/ModelCollection.js +2 -2
  218. package/lib/types/net/ModelsState.js +2 -2
  219. package/lib/types/net/OAuthToken.js +2 -2
  220. package/lib/types/net/OAuthTokenOrNull.js +2 -2
  221. package/lib/types/net/TokenAccessFunction.js +2 -2
  222. package/lib/types/net/index.js +17 -19
  223. package/lib/utils/baseActivity.js +83 -85
  224. package/lib/utils/baseRole.js +32 -36
  225. package/lib/utils/collection.js +403 -297
  226. package/lib/utils/cookies.js +19 -23
  227. package/lib/utils/date.js +188 -205
  228. package/lib/utils/dom.js +130 -131
  229. package/lib/utils/domainIdentifier.js +4 -8
  230. package/lib/utils/entityUserRole.js +2 -5
  231. package/lib/utils/error.js +14 -19
  232. package/lib/utils/events.js +32 -31
  233. package/lib/utils/externalGroup.js +20 -25
  234. package/lib/utils/externalProviders.js +4 -7
  235. package/lib/utils/externalTerms.js +6 -6
  236. package/lib/utils/fetch.js +168 -176
  237. package/lib/utils/group.js +14 -11
  238. package/lib/utils/groupDates.js +38 -46
  239. package/lib/utils/groupRoles.js +23 -32
  240. package/lib/utils/lockDownBrowser.js +12 -15
  241. package/lib/utils/logger.js +23 -28
  242. package/lib/utils/lti.js +4 -7
  243. package/lib/utils/model.js +28 -43
  244. package/lib/utils/number.js +9 -13
  245. package/lib/utils/promise.js +23 -26
  246. package/lib/utils/quill.js +55 -60
  247. package/lib/utils/route.js +52 -60
  248. package/lib/utils/search.js +72 -87
  249. package/lib/utils/shard.js +33 -42
  250. package/lib/utils/sort.js +47 -50
  251. package/lib/utils/string.js +10 -12
  252. package/lib/utils/table.js +29 -33
  253. package/lib/utils/timezone.js +7 -12
  254. package/lib/utils/url.js +130 -144
  255. package/lib/utils/user.js +54 -64
  256. package/lib/utils/userAgent.js +7 -14
  257. package/lib/utils/userRole.js +36 -39
  258. package/package.json +17 -3
@@ -1,342 +1,353 @@
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;
5
- exports.modelFetch = modelFetch;
6
- exports.modelFetchLoop = modelFetchLoop;
7
- 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");
1
+ import { isNil, isPlainObject, merge } from 'lodash';
2
+ import moment from 'moment-timezone';
3
+ import { call, cancel, cancelled, delay, fork, put, take, takeEvery } from 'redux-saga/effects';
4
+ import { networkOfflineErrorData, unknownErrorData } from '../../constants/fetchErrorData';
5
+ import { getLastActivityDate } from '../../services/dateService';
6
+ import { sendFetch, setApiRoot } from '../../services/fetchService';
7
+ import { windowService } from '../../services/windowService';
8
+ import { HTTP_STATUS_CODE } from '../../types';
9
+ import { constructErrorFromCaughtError } from '../../utils/error';
10
+ import { isFetchErrorData, prepareFetch } from '../../utils/fetch';
11
+ import { getLogger } from '../../utils/logger';
12
+ import { MODEL_FETCH_ERROR_ACTION_TYPE, MODEL_FETCH_REQUEST_ACTION_TYPE, MODEL_FETCH_RESULT_ACTION_TYPE, MODEL_FETCH_START_ACTION_TYPE, MODEL_REMOVE_KEY_ACTION_TYPE, PERIODIC_MODEL_FETCH_TERMINATION_ACTION_TYPE } from '../actions';
13
+
22
14
  /** The total number of tries for `fetchData`, including the initial request. */
23
- exports.TRY_LIMIT = 5;
15
+ export const TRY_LIMIT = 5;
16
+
24
17
  //#region Helpers
25
- const matchesTerminationAction = (incomingAction, dataRequestAction) => incomingAction.type === actions_1.PERIODIC_MODEL_FETCH_TERMINATION_ACTION_TYPE.TERMINATE &&
26
- incomingAction.taskId === dataRequestAction.taskId;
27
- exports.matchesTerminationAction = matchesTerminationAction;
28
- const takeMatchesTerminationAction = (dataRequestAction) => (incomingAction) => (0, exports.matchesTerminationAction)(incomingAction, dataRequestAction);
29
- exports.takeMatchesTerminationAction = takeMatchesTerminationAction;
18
+
19
+ export const matchesTerminationAction = (incomingAction, dataRequestAction) => incomingAction.type === PERIODIC_MODEL_FETCH_TERMINATION_ACTION_TYPE.TERMINATE && incomingAction.taskId === dataRequestAction.taskId;
20
+ export const takeMatchesTerminationAction = dataRequestAction => incomingAction => matchesTerminationAction(incomingAction, dataRequestAction);
21
+
30
22
  /* istanbul ignore next */
31
23
  /* eslint-disable-next-line require-yield */
32
- const defaultTokenAccessFunction = function* () {
33
- return null;
24
+ export const defaultTokenAccessFunction = function* () {
25
+ return null;
34
26
  };
35
- exports.defaultTokenAccessFunction = defaultTokenAccessFunction;
27
+
36
28
  /* istanbul ignore next */
37
- const defaultErrorHandler = () => {
38
- return;
29
+ export const defaultErrorHandler = () => {
30
+ return;
39
31
  };
40
- 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
- }
32
+ const getChildCollectionKeys = endpointMapping => Object.keys(endpointMapping).reduce((out, key) => {
33
+ const value = endpointMapping[key];
34
+ if (!value || !Object.prototype.hasOwnProperty.call(value, '_config')) {
50
35
  return out;
36
+ }
37
+ const em = value;
38
+ if (!!em._config && !!em._config.isCollection) {
39
+ out.push(key);
40
+ }
41
+ return out;
51
42
  }, []);
52
43
  const convertCollections = (endpointMapping, prepareFetchResult, data) => {
53
- 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);
44
+ const {
45
+ endpointConfig,
46
+ fetchConfig,
47
+ isCollectionItemCreate,
48
+ isCollectionItemFetch
49
+ } = prepareFetchResult;
50
+ const childCollectionKeys = getChildCollectionKeys(endpointMapping);
51
+ const hasChildCollections = childCollectionKeys.length > 0;
52
+ // no collections, return as-is
53
+ if (!endpointConfig.isCollection && !hasChildCollections) {
54
+ return data;
55
+ }
56
+ // delete returns no data
57
+ if (fetchConfig.method === 'DELETE') {
58
+ return {};
59
+ }
60
+ // data is a collection item, convert any child collections in item response
61
+ if (isCollectionItemFetch || isCollectionItemCreate) {
62
+ return convertDataCollectionItem(data, childCollectionKeys, endpointMapping);
63
+ }
64
+ // data is a collection, convert collection response, and any child collections in each item
65
+ return endpointConfig.isCollection ? convertDataToCollection(data, childCollectionKeys, endpointMapping) : convertDataCollectionItem(data, childCollectionKeys, endpointMapping);
72
66
  };
73
67
  const convertDataCollectionItem = (data, collectionKeys, endpointMapping) => {
74
- if ((0, lodash_1.isNil)(data)) {
75
- return data;
68
+ if (isNil(data)) {
69
+ return data;
70
+ }
71
+ return Object.keys(data).reduce((out, key) => {
72
+ let value = data[key];
73
+ if (collectionKeys.includes(key)) {
74
+ const childEndpointMapping = endpointMapping[key];
75
+ const childCollectionKeys = getChildCollectionKeys(childEndpointMapping);
76
+ value = convertDataToCollection(value, childCollectionKeys, childEndpointMapping);
76
77
  }
77
- 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
- }, {});
78
+ out[key] = value;
79
+ return out;
80
+ }, {});
87
81
  };
88
82
  const convertDataToCollection = (data, collectionKeys, endpointMapping) => {
89
- if ((0, lodash_1.isNil)(data)) {
90
- return data;
83
+ if (isNil(data)) {
84
+ return data;
85
+ }
86
+ return Object.keys(data).reduce((out, key) => {
87
+ const value = convertDataCollectionItem(data[key], collectionKeys, endpointMapping);
88
+ /* istanbul ignore else */
89
+ if (value && value.id) {
90
+ out[value.id] = value;
91
91
  }
92
- return 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
- }, {});
92
+ return out;
93
+ }, {});
100
94
  };
95
+
101
96
  /**
102
97
  * Using the `lastActivityDate` set in local storage by "studiokit-caliper-js",
103
98
  * determine if the user has been idle, e.g. has had no activity in the last 15 minutes.
104
99
  */
105
100
  function isUserIdle() {
106
- 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;
101
+ const lastActivityDateString = getLastActivityDate();
102
+ if (!lastActivityDateString) {
103
+ return false;
104
+ }
105
+ const lastActivityDateUtc = moment.utc(lastActivityDateString);
106
+ const nowUtc = moment.utc();
107
+ const idleMinutesThreshold = 1000 * 60 * 15; // 15 minutes
108
+
109
+ return nowUtc.diff(lastActivityDateUtc) >= idleMinutesThreshold;
114
110
  }
111
+
115
112
  //#endregion Helpers
113
+
116
114
  //#region Local Variables
115
+
117
116
  let endpointMappings;
118
117
  let tokenAccessFunction;
119
118
  let errorHandler;
120
119
  let logger;
120
+
121
121
  //#endregion Local Variables
122
+
122
123
  /** Set the shared EndpointMappings variable, exposed for testability */
123
- function setEndpointMappings(endpointMappingsParam) {
124
- endpointMappings = endpointMappingsParam;
124
+ export function setEndpointMappings(endpointMappingsParam) {
125
+ endpointMappings = endpointMappingsParam;
125
126
  }
127
+
126
128
  /**
127
129
  * Construct a request based on the provided dataRequestAction, make a request with a configurable retry,
128
130
  * and handle errors, logging and dispatching all steps.
129
131
  *
130
132
  * @param modelFetchRequestAction A model fetch request action with the request configuration
131
133
  */
132
- function* modelFetch(modelFetchRequestAction) {
133
- let prepareFetchResult = undefined;
134
+ export function* modelFetch(modelFetchRequestAction) {
135
+ let prepareFetchResult = undefined;
136
+ try {
137
+ prepareFetchResult = prepareFetch(modelFetchRequestAction, endpointMappings);
138
+ } catch (error) {
139
+ // Note: `prepareFetch` already ensures that `error` is an actual `Error`
140
+ logger.error(error);
141
+ errorHandler(error, modelFetchRequestAction);
142
+ return;
143
+ }
144
+
145
+ // get prepared variables
146
+ const {
147
+ modelPath,
148
+ fetchConfig,
149
+ endpointMapping,
150
+ isCollectionItemCreate,
151
+ isCollectionItemDelete
152
+ } = prepareFetchResult;
153
+
154
+ // Configure retry
155
+ const tryLimit = modelFetchRequestAction.noRetry ? 1 : TRY_LIMIT;
156
+ let tryCount = 0;
157
+ let didFail = false;
158
+ let lastFetchResult;
159
+ let lastError;
160
+ let lastErrorData;
161
+ const fetchResultUndefinedErrorMessage = "'fetchResult' is undefined";
162
+ const fetchResultNotOkErrorMessage = "'fetchResult.ok' is false";
163
+
164
+ // Run retry loop
165
+ do {
166
+ didFail = false;
167
+ tryCount++;
168
+
169
+ // dispatch that the fetch request started, which updates `_metadata` and `guid`, if provided, at the target redux `modelPath`
170
+ yield put({
171
+ type: modelFetchRequestAction.noStore ? MODEL_FETCH_START_ACTION_TYPE.TRANSIENT_FETCH_START : MODEL_FETCH_START_ACTION_TYPE.FETCH_START,
172
+ modelPath,
173
+ guid: modelFetchRequestAction.guid
174
+ });
175
+ let fetchResult = undefined;
134
176
  try {
135
- 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
177
+ // get the OAuth Token, if any, and call `fetch`
178
+ const oauthToken = yield call(tokenAccessFunction, modelFetchRequestAction.modelName);
179
+ if (oauthToken?.access_token) {
180
+ fetchConfig.headers = merge({}, fetchConfig.headers, {
181
+ Authorization: `Bearer ${oauthToken.access_token}`
165
182
  });
166
- 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
183
+ }
184
+ fetchResult = yield call(sendFetch, fetchConfig);
185
+ // store `fetchResult` for reference after the do-while loop
186
+ lastFetchResult = fetchResult;
187
+
188
+ // throw on unsuccessful result to short-circuit and trigger catch+retry
189
+ if (!fetchResult) throw new Error(fetchResultUndefinedErrorMessage);
190
+ if (!fetchResult.ok) throw new Error(fetchResultNotOkErrorMessage);
191
+
192
+ // handle success
193
+ const resultActionType = modelFetchRequestAction.noStore ? MODEL_FETCH_RESULT_ACTION_TYPE.TRANSIENT_FETCH_RESULT_RECEIVED : MODEL_FETCH_RESULT_ACTION_TYPE.FETCH_RESULT_RECEIVED;
194
+ const data = convertCollections(endpointMapping, prepareFetchResult, fetchResult.data);
195
+ // attach guid to result data if it is an object
196
+ if (modelFetchRequestAction.guid && isPlainObject(data)) {
197
+ data.guid = modelFetchRequestAction.guid;
198
+ }
199
+
200
+ // POST new collection item
201
+ if (isCollectionItemCreate) {
202
+ const modelPathLevels = modelPath.split('.');
203
+ // remove guid
204
+ modelPathLevels.pop();
205
+ // add by new result's id
206
+ yield put({
207
+ type: resultActionType,
208
+ modelPath: `${modelPathLevels.join('.')}.${data.id}`,
209
+ guid: modelFetchRequestAction.guid,
210
+ replaceValue: modelFetchRequestAction.replaceValue,
211
+ data
212
+ });
213
+ // remove temp item under guid key
214
+ yield put({
215
+ type: MODEL_REMOVE_KEY_ACTION_TYPE,
216
+ modelPath: modelPath
283
217
  });
284
- // 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);
218
+ }
219
+ // DELETE collection item
220
+ else if (isCollectionItemDelete) {
221
+ yield put({
222
+ type: MODEL_REMOVE_KEY_ACTION_TYPE,
223
+ modelPath: modelPath
224
+ });
225
+ } else {
226
+ yield put({
227
+ type: resultActionType,
228
+ modelPath: modelPath,
229
+ guid: modelFetchRequestAction.guid,
230
+ replaceValue: modelFetchRequestAction.replaceValue,
231
+ data
232
+ });
233
+ }
234
+ } catch (caughtError) {
235
+ // construct `Error` using `caughtError`
236
+ const baseErrorMessage = `'modelFetch' failed for '${modelPath}'`;
237
+ const error = constructErrorFromCaughtError(caughtError, 'ModelFetchError', baseErrorMessage);
238
+ // create a default `errorData` object
239
+ const isNetworkOnline = windowService.getIsOnline();
240
+ let errorData = isNetworkOnline ? unknownErrorData : networkOfflineErrorData;
241
+ // update `errorData` from the `fetchResult.data`
242
+ // for when the fetch got a result, but it was an unsuccessful status
243
+ if (fetchResult?.data && isFetchErrorData(fetchResult.data)) {
244
+ errorData = fetchResult.data;
245
+ // update the error's message to include info from the fetch result
246
+ error.message = `${baseErrorMessage}.\n${fetchConfig.method || 'GET'} ${fetchConfig.path} failed.\n${errorData.title} (${errorData.status})`;
247
+ if (errorData.detail) {
248
+ error.message += `: ${errorData.detail}`;
287
249
  }
250
+ }
251
+
252
+ // track `errorData` and `error` for after the while-loop
253
+ lastErrorData = errorData;
254
+ lastError = error;
255
+
256
+ // dispatch that this try failed
257
+ yield put({
258
+ type: MODEL_FETCH_ERROR_ACTION_TYPE.TRY_FETCH_FAILED,
259
+ modelPath: modelPath,
260
+ guid: modelFetchRequestAction.guid,
261
+ errorData
262
+ });
263
+
264
+ // log to the console
265
+ logger.error(error);
266
+ didFail = true;
267
+
268
+ // Do not retry if the response is between 400-500 (except 408: request timeout)
269
+ if (errorData.status >= HTTP_STATUS_CODE.BAD_REQUEST && errorData.status < HTTP_STATUS_CODE.INTERNAL_SERVER_ERROR && errorData.status !== HTTP_STATUS_CODE.REQUEST_TIMEOUT) {
270
+ tryCount = tryLimit;
271
+ }
272
+
273
+ // if there are tries remaining, perform an exponential backoff
274
+ if (tryCount < tryLimit) {
275
+ yield delay(2 ** (tryCount - 1) * 100); // 100, 200, 400, 800
276
+ }
277
+ }
278
+ } while (tryCount < tryLimit && didFail);
279
+
280
+ // if fetch failed completely after all retries
281
+ if (tryCount === tryLimit && didFail) {
282
+ // these will always exist
283
+ const errorData = lastErrorData;
284
+ const error = lastError;
285
+
286
+ // dispatch that the fetch failed, which updates `_metadata` (and `guid`, if provided) at the target redux `modelPath`
287
+ yield put({
288
+ type: modelFetchRequestAction.noStore ? MODEL_FETCH_ERROR_ACTION_TYPE.TRANSIENT_FETCH_FAILED : MODEL_FETCH_ERROR_ACTION_TYPE.FETCH_FAILED,
289
+ modelPath: modelPath,
290
+ guid: modelFetchRequestAction.guid,
291
+ errorData
292
+ });
293
+
294
+ // Send error to error handler, except for 401s
295
+ if (errorData.status !== HTTP_STATUS_CODE.UNAUTHORIZED) {
296
+ errorHandler(error, modelFetchRequestAction, fetchConfig, lastFetchResult, errorData);
288
297
  }
298
+ }
289
299
  }
300
+
290
301
  /**
291
302
  * The loop saga that makes the request every {action.period} milliseconds until cancelled
292
303
  *
293
304
  * @param action An action with the request configuration
294
305
  */
295
- function* modelFetchLoop(action) {
296
- 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
- }
306
+ export function* modelFetchLoop(action) {
307
+ try {
308
+ let hasFetched = false;
309
+ do {
310
+ // call the fetch if this is the first loop iteration,
311
+ // or if the user is not idle
312
+ if (!hasFetched || !isUserIdle()) {
313
+ yield call(modelFetch, action);
314
+ hasFetched = true;
315
+ }
316
+ yield delay(action.period);
317
+ } while (true);
318
+ } catch (caughtError) {
319
+ const error = constructErrorFromCaughtError(caughtError, 'ModelFetchLoopError', 'modelFetchLoop failed');
320
+ logger.error(error);
321
+ // catch and log any unexpected errors not caught inside `fetchData`
322
+ errorHandler(error, action);
323
+ } finally {
324
+ if (yield cancelled()) {
325
+ yield put({
326
+ type: PERIODIC_MODEL_FETCH_TERMINATION_ACTION_TYPE.SUCCEEDED,
327
+ taskId: action.taskId
328
+ });
321
329
  }
330
+ }
322
331
  }
332
+
323
333
  /**
324
334
  * Call the fetchData saga every {action.period} milliseconds. This saga requires the 'period' and 'taskId' properties
325
335
  * on the action parameter.
326
336
  *
327
337
  * @param action An action with the request configuration
328
338
  */
329
- function* modelFetchRecurring(action) {
330
- 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);
339
+ export function* modelFetchRecurring(action) {
340
+ if (!action.period) {
341
+ throw new Error("'period' is required");
342
+ }
343
+ if (!action.taskId) {
344
+ throw new Error("'taskId' is required");
345
+ }
346
+ const bgSyncTask = yield fork(modelFetchLoop, action);
347
+ yield take(takeMatchesTerminationAction(action));
348
+ yield cancel(bgSyncTask);
339
349
  }
350
+
340
351
  /**
341
352
  * The main saga for fetching models. Must be initialized with an EndpointMappings object that can be fetched
342
353
  * and an API root to prepend to any partial URLs specified in the EndpointMappings object. A logger should normally be provided
@@ -369,12 +380,17 @@ function* modelFetchRecurring(action) {
369
380
  * @param tokenAccessFunctionParam function that returns an optional OAuth token
370
381
  * @param errorHandlerParam A function that is called when any fetch exceptions occur
371
382
  */
372
- function* modelFetchSaga(endpointMappingsParam, apiRootParam, tokenAccessFunctionParam = exports.defaultTokenAccessFunction, errorHandlerParam = exports.defaultErrorHandler) {
373
- (0, fetchService_1.setApiRoot)(apiRootParam);
383
+ export default function modelFetchSaga(endpointMappingsParam, apiRootParam) {
384
+ let tokenAccessFunctionParam = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : defaultTokenAccessFunction;
385
+ let errorHandlerParam = arguments.length > 3 && arguments[3] !== undefined ? arguments[3] : defaultErrorHandler;
386
+ return function* () {
387
+ setApiRoot(apiRootParam);
374
388
  setEndpointMappings(endpointMappingsParam);
375
389
  errorHandler = errorHandlerParam;
376
390
  tokenAccessFunction = tokenAccessFunctionParam;
377
- logger = (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);
391
+ logger = getLogger();
392
+ yield takeEvery(MODEL_FETCH_REQUEST_ACTION_TYPE.FETCH_REQUEST, modelFetch);
393
+ yield takeEvery(MODEL_FETCH_REQUEST_ACTION_TYPE.PERIODIC_FETCH_REQUEST, modelFetchRecurring);
394
+ }();
380
395
  }
396
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["isNil","isPlainObject","merge","moment","call","cancel","cancelled","delay","fork","put","take","takeEvery","networkOfflineErrorData","unknownErrorData","getLastActivityDate","sendFetch","setApiRoot","windowService","HTTP_STATUS_CODE","constructErrorFromCaughtError","isFetchErrorData","prepareFetch","getLogger","MODEL_FETCH_ERROR_ACTION_TYPE","MODEL_FETCH_REQUEST_ACTION_TYPE","MODEL_FETCH_RESULT_ACTION_TYPE","MODEL_FETCH_START_ACTION_TYPE","MODEL_REMOVE_KEY_ACTION_TYPE","PERIODIC_MODEL_FETCH_TERMINATION_ACTION_TYPE","TRY_LIMIT","matchesTerminationAction","incomingAction","dataRequestAction","type","TERMINATE","taskId","takeMatchesTerminationAction","defaultTokenAccessFunction","defaultErrorHandler","getChildCollectionKeys","endpointMapping","Object","keys","reduce","out","key","value","prototype","hasOwnProperty","em","_config","isCollection","push","convertCollections","prepareFetchResult","data","endpointConfig","fetchConfig","isCollectionItemCreate","isCollectionItemFetch","childCollectionKeys","hasChildCollections","length","method","convertDataCollectionItem","convertDataToCollection","collectionKeys","includes","childEndpointMapping","id","isUserIdle","lastActivityDateString","lastActivityDateUtc","utc","nowUtc","idleMinutesThreshold","diff","endpointMappings","tokenAccessFunction","errorHandler","logger","setEndpointMappings","endpointMappingsParam","modelFetch","modelFetchRequestAction","undefined","error","modelPath","isCollectionItemDelete","tryLimit","noRetry","tryCount","didFail","lastFetchResult","lastError","lastErrorData","fetchResultUndefinedErrorMessage","fetchResultNotOkErrorMessage","noStore","TRANSIENT_FETCH_START","FETCH_START","guid","fetchResult","oauthToken","modelName","access_token","headers","Authorization","Error","ok","resultActionType","TRANSIENT_FETCH_RESULT_RECEIVED","FETCH_RESULT_RECEIVED","modelPathLevels","split","pop","join","replaceValue","caughtError","baseErrorMessage","isNetworkOnline","getIsOnline","errorData","message","path","title","status","detail","TRY_FETCH_FAILED","BAD_REQUEST","INTERNAL_SERVER_ERROR","REQUEST_TIMEOUT","TRANSIENT_FETCH_FAILED","FETCH_FAILED","UNAUTHORIZED","modelFetchLoop","action","hasFetched","period","SUCCEEDED","modelFetchRecurring","bgSyncTask","modelFetchSaga","apiRootParam","tokenAccessFunctionParam","arguments","errorHandlerParam","FETCH_REQUEST","PERIODIC_FETCH_REQUEST"],"sources":["../../../src/redux/sagas/modelFetchSaga.ts"],"sourcesContent":["import { SagaIterator, Task } from '@redux-saga/core'\nimport { isNil, isPlainObject, merge } from 'lodash'\nimport moment from 'moment-timezone'\nimport { AnyAction } from 'redux'\nimport { call, cancel, cancelled, delay, fork, put, take, takeEvery } from 'redux-saga/effects'\nimport { networkOfflineErrorData, unknownErrorData } from '../../constants/fetchErrorData'\nimport { getLastActivityDate } from '../../services/dateService'\nimport { sendFetch, setApiRoot } from '../../services/fetchService'\nimport { windowService } from '../../services/windowService'\nimport {\n\tEndpointMapping,\n\tEndpointMappings,\n\tErrorHandler,\n\tFetchErrorData,\n\tFetchResult,\n\tHTTP_STATUS_CODE,\n\tModel,\n\tModelCollection,\n\tOAuthTokenOrNull,\n\tTokenAccessFunction\n} from '../../types'\nimport { constructErrorFromCaughtError } from '../../utils/error'\nimport { isFetchErrorData, prepareFetch, PrepareFetchResult } from '../../utils/fetch'\nimport { getLogger, Logger } from '../../utils/logger'\nimport {\n\tMODEL_FETCH_ERROR_ACTION_TYPE,\n\tMODEL_FETCH_REQUEST_ACTION_TYPE,\n\tMODEL_FETCH_RESULT_ACTION_TYPE,\n\tMODEL_FETCH_START_ACTION_TYPE,\n\tMODEL_REMOVE_KEY_ACTION_TYPE,\n\tModelFetchErrorAction,\n\tModelFetchRequestAction,\n\tModelFetchResultAction,\n\tModelFetchStartAction,\n\tModelRemoveKeyAction,\n\tPERIODIC_MODEL_FETCH_TERMINATION_ACTION_TYPE,\n\tPeriodicModelFetchRequestAction,\n\tPeriodicModelFetchTerminationAction\n} from '../actions'\n\n/** The total number of tries for `fetchData`, including the initial request. */\nexport const TRY_LIMIT = 5\n\n//#region Helpers\n\nexport const matchesTerminationAction = (\n\tincomingAction: AnyAction,\n\tdataRequestAction: PeriodicModelFetchRequestAction\n) =>\n\tincomingAction.type === PERIODIC_MODEL_FETCH_TERMINATION_ACTION_TYPE.TERMINATE &&\n\tincomingAction.taskId === dataRequestAction.taskId\n\nexport const takeMatchesTerminationAction =\n\t(dataRequestAction: PeriodicModelFetchRequestAction) => (incomingAction: AnyAction) =>\n\t\tmatchesTerminationAction(incomingAction, dataRequestAction)\n\n/* istanbul ignore next */\n/* eslint-disable-next-line require-yield */\nexport const defaultTokenAccessFunction: TokenAccessFunction = function* () {\n\treturn null\n}\n\n/* istanbul ignore next */\nexport const defaultErrorHandler: ErrorHandler = () => {\n\treturn\n}\n\ninterface GenericModel extends Model, Record<string, any> {}\n\nconst getChildCollectionKeys = (endpointMapping: EndpointMapping) =>\n\tObject.keys(endpointMapping).reduce((out: string[], key) => {\n\t\tconst value = endpointMapping[key]\n\t\tif (!value || !Object.prototype.hasOwnProperty.call(value, '_config')) {\n\t\t\treturn out\n\t\t}\n\t\tconst em = value as EndpointMapping\n\t\tif (!!em._config && !!em._config.isCollection) {\n\t\t\tout.push(key)\n\t\t}\n\t\treturn out\n\t}, [])\n\nconst convertCollections = (endpointMapping: EndpointMapping, prepareFetchResult: PrepareFetchResult, data: any) => {\n\tconst { endpointConfig, fetchConfig, isCollectionItemCreate, isCollectionItemFetch } = prepareFetchResult\n\tconst childCollectionKeys = getChildCollectionKeys(endpointMapping)\n\tconst hasChildCollections = childCollectionKeys.length > 0\n\t// no collections, return as-is\n\tif (!endpointConfig.isCollection && !hasChildCollections) {\n\t\treturn data\n\t}\n\t// delete returns no data\n\tif (fetchConfig.method === 'DELETE') {\n\t\treturn {}\n\t}\n\t// data is a collection item, convert any child collections in item response\n\tif (isCollectionItemFetch || isCollectionItemCreate) {\n\t\treturn convertDataCollectionItem(data, childCollectionKeys, endpointMapping)\n\t}\n\t// data is a collection, convert collection response, and any child collections in each item\n\treturn endpointConfig.isCollection\n\t\t? convertDataToCollection(data, childCollectionKeys, endpointMapping)\n\t\t: convertDataCollectionItem(data, childCollectionKeys, endpointMapping)\n}\n\nconst convertDataCollectionItem = (data: any, collectionKeys: string[], endpointMapping: EndpointMapping) => {\n\tif (isNil(data)) {\n\t\treturn data\n\t}\n\treturn Object.keys(data).reduce((out: GenericModel, key) => {\n\t\tlet value = data[key]\n\t\tif (collectionKeys.includes(key)) {\n\t\t\tconst childEndpointMapping = endpointMapping[key] as EndpointMapping\n\t\t\tconst childCollectionKeys = getChildCollectionKeys(childEndpointMapping)\n\t\t\tvalue = convertDataToCollection(value, childCollectionKeys, childEndpointMapping)\n\t\t}\n\t\tout[key] = value\n\t\treturn out\n\t}, {})\n}\n\nconst convertDataToCollection = (data: any, collectionKeys: string[], endpointMapping: EndpointMapping) => {\n\tif (isNil(data)) {\n\t\treturn data\n\t}\n\treturn Object.keys(data).reduce((out: ModelCollection<any>, key) => {\n\t\tconst value = convertDataCollectionItem(data[key], collectionKeys, endpointMapping)\n\t\t/* istanbul ignore else */\n\t\tif (value && value.id) {\n\t\t\tout[value.id] = value\n\t\t}\n\t\treturn out\n\t}, {})\n}\n\n/**\n * Using the `lastActivityDate` set in local storage by \"studiokit-caliper-js\",\n * determine if the user has been idle, e.g. has had no activity in the last 15 minutes.\n */\nfunction isUserIdle() {\n\tconst lastActivityDateString = getLastActivityDate()\n\tif (!lastActivityDateString) {\n\t\treturn false\n\t}\n\n\tconst lastActivityDateUtc = moment.utc(lastActivityDateString)\n\tconst nowUtc = moment.utc()\n\tconst idleMinutesThreshold = 1000 * 60 * 15 // 15 minutes\n\n\treturn nowUtc.diff(lastActivityDateUtc) >= idleMinutesThreshold\n}\n\n//#endregion Helpers\n\n//#region Local Variables\n\nlet endpointMappings: EndpointMappings\nlet tokenAccessFunction: TokenAccessFunction\nlet errorHandler: ErrorHandler\nlet logger: Logger\n\n//#endregion Local Variables\n\n/** Set the shared EndpointMappings variable, exposed for testability */\nexport function setEndpointMappings(endpointMappingsParam: EndpointMappings) {\n\tendpointMappings = endpointMappingsParam\n}\n\n/**\n * Construct a request based on the provided dataRequestAction, make a request with a configurable retry,\n * and handle errors, logging and dispatching all steps.\n *\n * @param modelFetchRequestAction A model fetch request action with the request configuration\n */\nexport function* modelFetch(\n\tmodelFetchRequestAction: ModelFetchRequestAction | PeriodicModelFetchRequestAction\n): SagaIterator {\n\tlet prepareFetchResult: PrepareFetchResult | undefined = undefined\n\ttry {\n\t\tprepareFetchResult = prepareFetch(modelFetchRequestAction, endpointMappings)\n\t} catch (error) {\n\t\t// Note: `prepareFetch` already ensures that `error` is an actual `Error`\n\t\tlogger.error(error)\n\t\terrorHandler(error as Error, modelFetchRequestAction)\n\t\treturn\n\t}\n\n\t// get prepared variables\n\tconst { modelPath, fetchConfig, endpointMapping, isCollectionItemCreate, isCollectionItemDelete } =\n\t\tprepareFetchResult\n\n\t// Configure retry\n\tconst tryLimit: number = modelFetchRequestAction.noRetry ? 1 : TRY_LIMIT\n\tlet tryCount = 0\n\tlet didFail = false\n\tlet lastFetchResult: FetchResult | undefined\n\tlet lastError: Error | undefined\n\tlet lastErrorData: FetchErrorData | undefined\n\n\tconst fetchResultUndefinedErrorMessage = \"'fetchResult' is undefined\"\n\tconst fetchResultNotOkErrorMessage = \"'fetchResult.ok' is false\"\n\n\t// Run retry loop\n\tdo {\n\t\tdidFail = false\n\t\ttryCount++\n\n\t\t// dispatch that the fetch request started, which updates `_metadata` and `guid`, if provided, at the target redux `modelPath`\n\t\tyield put<ModelFetchStartAction>({\n\t\t\ttype: modelFetchRequestAction.noStore\n\t\t\t\t? MODEL_FETCH_START_ACTION_TYPE.TRANSIENT_FETCH_START\n\t\t\t\t: MODEL_FETCH_START_ACTION_TYPE.FETCH_START,\n\t\t\tmodelPath,\n\t\t\tguid: modelFetchRequestAction.guid\n\t\t})\n\n\t\tlet fetchResult: FetchResult | undefined = undefined\n\t\ttry {\n\t\t\t// get the OAuth Token, if any, and call `fetch`\n\t\t\tconst oauthToken: OAuthTokenOrNull = yield call(tokenAccessFunction, modelFetchRequestAction.modelName)\n\t\t\tif (oauthToken?.access_token) {\n\t\t\t\tfetchConfig.headers = merge({}, fetchConfig.headers, {\n\t\t\t\t\tAuthorization: `Bearer ${oauthToken.access_token}`\n\t\t\t\t})\n\t\t\t}\n\t\t\tfetchResult = yield call(sendFetch, fetchConfig)\n\t\t\t// store `fetchResult` for reference after the do-while loop\n\t\t\tlastFetchResult = fetchResult\n\n\t\t\t// throw on unsuccessful result to short-circuit and trigger catch+retry\n\t\t\tif (!fetchResult) throw new Error(fetchResultUndefinedErrorMessage)\n\t\t\tif (!fetchResult.ok) throw new Error(fetchResultNotOkErrorMessage)\n\n\t\t\t// handle success\n\t\t\tconst resultActionType = modelFetchRequestAction.noStore\n\t\t\t\t? MODEL_FETCH_RESULT_ACTION_TYPE.TRANSIENT_FETCH_RESULT_RECEIVED\n\t\t\t\t: MODEL_FETCH_RESULT_ACTION_TYPE.FETCH_RESULT_RECEIVED\n\n\t\t\tconst data = convertCollections(endpointMapping, prepareFetchResult, fetchResult.data)\n\t\t\t// attach guid to result data if it is an object\n\t\t\tif (modelFetchRequestAction.guid && isPlainObject(data)) {\n\t\t\t\tdata.guid = modelFetchRequestAction.guid\n\t\t\t}\n\n\t\t\t// POST new collection item\n\t\t\tif (isCollectionItemCreate) {\n\t\t\t\tconst modelPathLevels = modelPath.split('.')\n\t\t\t\t// remove guid\n\t\t\t\tmodelPathLevels.pop()\n\t\t\t\t// add by new result's id\n\t\t\t\tyield put<ModelFetchResultAction>({\n\t\t\t\t\ttype: resultActionType,\n\t\t\t\t\tmodelPath: `${modelPathLevels.join('.')}.${data.id}`,\n\t\t\t\t\tguid: modelFetchRequestAction.guid,\n\t\t\t\t\treplaceValue: modelFetchRequestAction.replaceValue,\n\t\t\t\t\tdata\n\t\t\t\t})\n\t\t\t\t// remove temp item under guid key\n\t\t\t\tyield put<ModelRemoveKeyAction>({\n\t\t\t\t\ttype: MODEL_REMOVE_KEY_ACTION_TYPE,\n\t\t\t\t\tmodelPath: modelPath\n\t\t\t\t})\n\t\t\t}\n\t\t\t// DELETE collection item\n\t\t\telse if (isCollectionItemDelete) {\n\t\t\t\tyield put<ModelRemoveKeyAction>({\n\t\t\t\t\ttype: MODEL_REMOVE_KEY_ACTION_TYPE,\n\t\t\t\t\tmodelPath: modelPath\n\t\t\t\t})\n\t\t\t} else {\n\t\t\t\tyield put<ModelFetchResultAction>({\n\t\t\t\t\ttype: resultActionType,\n\t\t\t\t\tmodelPath: modelPath,\n\t\t\t\t\tguid: modelFetchRequestAction.guid,\n\t\t\t\t\treplaceValue: modelFetchRequestAction.replaceValue,\n\t\t\t\t\tdata\n\t\t\t\t})\n\t\t\t}\n\t\t} catch (caughtError: unknown) {\n\t\t\t// construct `Error` using `caughtError`\n\t\t\tconst baseErrorMessage = `'modelFetch' failed for '${modelPath}'`\n\t\t\tconst error = constructErrorFromCaughtError(caughtError, 'ModelFetchError', baseErrorMessage)\n\t\t\t// create a default `errorData` object\n\t\t\tconst isNetworkOnline = windowService.getIsOnline()\n\t\t\tlet errorData: FetchErrorData = isNetworkOnline ? unknownErrorData : networkOfflineErrorData\n\t\t\t// update `errorData` from the `fetchResult.data`\n\t\t\t// for when the fetch got a result, but it was an unsuccessful status\n\t\t\tif (fetchResult?.data && isFetchErrorData(fetchResult.data)) {\n\t\t\t\terrorData = fetchResult.data\n\t\t\t\t// update the error's message to include info from the fetch result\n\t\t\t\terror.message = `${baseErrorMessage}.\\n${fetchConfig.method || 'GET'} ${fetchConfig.path} failed.\\n${\n\t\t\t\t\terrorData.title\n\t\t\t\t} (${errorData.status})`\n\t\t\t\tif (errorData.detail) {\n\t\t\t\t\terror.message += `: ${errorData.detail}`\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// track `errorData` and `error` for after the while-loop\n\t\t\tlastErrorData = errorData\n\t\t\tlastError = error\n\n\t\t\t// dispatch that this try failed\n\t\t\tyield put<ModelFetchErrorAction>({\n\t\t\t\ttype: MODEL_FETCH_ERROR_ACTION_TYPE.TRY_FETCH_FAILED,\n\t\t\t\tmodelPath: modelPath,\n\t\t\t\tguid: modelFetchRequestAction.guid,\n\t\t\t\terrorData\n\t\t\t})\n\n\t\t\t// log to the console\n\t\t\tlogger.error(error)\n\n\t\t\tdidFail = true\n\n\t\t\t// Do not retry if the response is between 400-500 (except 408: request timeout)\n\t\t\tif (\n\t\t\t\terrorData.status >= HTTP_STATUS_CODE.BAD_REQUEST &&\n\t\t\t\terrorData.status < HTTP_STATUS_CODE.INTERNAL_SERVER_ERROR &&\n\t\t\t\terrorData.status !== HTTP_STATUS_CODE.REQUEST_TIMEOUT\n\t\t\t) {\n\t\t\t\ttryCount = tryLimit\n\t\t\t}\n\n\t\t\t// if there are tries remaining, perform an exponential backoff\n\t\t\tif (tryCount < tryLimit) {\n\t\t\t\tyield delay(2 ** (tryCount - 1) * 100) // 100, 200, 400, 800\n\t\t\t}\n\t\t}\n\t} while (tryCount < tryLimit && didFail)\n\n\t// if fetch failed completely after all retries\n\tif (tryCount === tryLimit && didFail) {\n\t\t// these will always exist\n\t\tconst errorData = lastErrorData as FetchErrorData\n\t\tconst error = lastError as Error\n\n\t\t// dispatch that the fetch failed, which updates `_metadata` (and `guid`, if provided) at the target redux `modelPath`\n\t\tyield put<ModelFetchErrorAction>({\n\t\t\ttype: modelFetchRequestAction.noStore\n\t\t\t\t? MODEL_FETCH_ERROR_ACTION_TYPE.TRANSIENT_FETCH_FAILED\n\t\t\t\t: MODEL_FETCH_ERROR_ACTION_TYPE.FETCH_FAILED,\n\t\t\tmodelPath: modelPath,\n\t\t\tguid: modelFetchRequestAction.guid,\n\t\t\terrorData\n\t\t})\n\n\t\t// Send error to error handler, except for 401s\n\t\tif (errorData.status !== HTTP_STATUS_CODE.UNAUTHORIZED) {\n\t\t\terrorHandler(error, modelFetchRequestAction, fetchConfig, lastFetchResult, errorData)\n\t\t}\n\t}\n}\n\n/**\n * The loop saga that makes the request every {action.period} milliseconds until cancelled\n *\n * @param action An action with the request configuration\n */\nexport function* modelFetchLoop(action: PeriodicModelFetchRequestAction): SagaIterator {\n\ttry {\n\t\tlet hasFetched = false\n\t\tdo {\n\t\t\t// call the fetch if this is the first loop iteration,\n\t\t\t// or if the user is not idle\n\t\t\tif (!hasFetched || !isUserIdle()) {\n\t\t\t\tyield call(modelFetch, action)\n\t\t\t\thasFetched = true\n\t\t\t}\n\t\t\tyield delay(action.period)\n\t\t} while (true)\n\t} catch (caughtError: unknown) {\n\t\tconst error = constructErrorFromCaughtError(caughtError, 'ModelFetchLoopError', 'modelFetchLoop failed')\n\t\tlogger.error(error)\n\t\t// catch and log any unexpected errors not caught inside `fetchData`\n\t\terrorHandler(error, action)\n\t} finally {\n\t\tif (yield cancelled()) {\n\t\t\tyield put<PeriodicModelFetchTerminationAction>({\n\t\t\t\ttype: PERIODIC_MODEL_FETCH_TERMINATION_ACTION_TYPE.SUCCEEDED,\n\t\t\t\ttaskId: action.taskId\n\t\t\t})\n\t\t}\n\t}\n}\n\n/**\n * Call the fetchData saga every {action.period} milliseconds. This saga requires the 'period' and 'taskId' properties\n * on the action parameter.\n *\n * @param action An action with the request configuration\n */\nexport function* modelFetchRecurring(action: PeriodicModelFetchRequestAction): SagaIterator {\n\tif (!action.period) {\n\t\tthrow new Error(\"'period' is required\")\n\t}\n\tif (!action.taskId) {\n\t\tthrow new Error(\"'taskId' is required\")\n\t}\n\n\tconst bgSyncTask: Task = yield fork(modelFetchLoop, action)\n\tyield take(takeMatchesTerminationAction(action))\n\tyield cancel(bgSyncTask)\n}\n\n/**\n * The main saga for fetching models. Must be initialized with an EndpointMappings object that can be fetched\n * and an API root to prepend to any partial URLs specified in the EndpointMappings object. A logger should normally be provided\n * as well.\n *\n * `EndpointMappings` object require a form as follows (with optional nested models):\n * ```\n * {\n * \tfryModel: {\n * \t\tpath: '/api/Foo'\n * \t},\n * \tgroupOfModels: {\n * \t\tleelaModel: {\n * \t\t\tpath: '/api/Bar'\n * \t\t},\n * \t\tbenderModel: {\n * \t\t\tpath: '/api/Baz'\n * \t\t}\n * \t}\n * }\n * ```\n * `EndpointMapping` objects are referenced in the actions.DATA_REQUESTED action by path, i.e.\n * `{ type: actions.DATA_REQUESTED, { modelName: 'fryModel' } }`\n * -- or --\n * `{ type: actions.DATA_REQUESTED, { modelName: 'groupOfModels.leelaModel' } }`\n *\n * @export\n * @param endpointMappingsParam A mapping of API endpoints available in the application\n * @param apiRootParam A url to which partial URLs are appended (i.e.) 'https://myapp.com'\n * @param tokenAccessFunctionParam function that returns an optional OAuth token\n * @param errorHandlerParam A function that is called when any fetch exceptions occur\n */\nexport default function* modelFetchSaga(\n\tendpointMappingsParam: EndpointMappings,\n\tapiRootParam?: string,\n\ttokenAccessFunctionParam: TokenAccessFunction | undefined = defaultTokenAccessFunction,\n\terrorHandlerParam: ErrorHandler | undefined = defaultErrorHandler\n): SagaIterator {\n\tsetApiRoot(apiRootParam)\n\tsetEndpointMappings(endpointMappingsParam)\n\terrorHandler = errorHandlerParam\n\ttokenAccessFunction = tokenAccessFunctionParam\n\tlogger = getLogger()\n\n\tyield takeEvery(MODEL_FETCH_REQUEST_ACTION_TYPE.FETCH_REQUEST, modelFetch)\n\tyield takeEvery(MODEL_FETCH_REQUEST_ACTION_TYPE.PERIODIC_FETCH_REQUEST, modelFetchRecurring)\n}\n"],"mappings":"AACA,SAASA,KAAK,EAAEC,aAAa,EAAEC,KAAK,QAAQ,QAAQ;AACpD,OAAOC,MAAM,MAAM,iBAAiB;AAEpC,SAASC,IAAI,EAAEC,MAAM,EAAEC,SAAS,EAAEC,KAAK,EAAEC,IAAI,EAAEC,GAAG,EAAEC,IAAI,EAAEC,SAAS,QAAQ,oBAAoB;AAC/F,SAASC,uBAAuB,EAAEC,gBAAgB,QAAQ,gCAAgC;AAC1F,SAASC,mBAAmB,QAAQ,4BAA4B;AAChE,SAASC,SAAS,EAAEC,UAAU,QAAQ,6BAA6B;AACnE,SAASC,aAAa,QAAQ,8BAA8B;AAC5D,SAMCC,gBAAgB,QAKV,aAAa;AACpB,SAASC,6BAA6B,QAAQ,mBAAmB;AACjE,SAASC,gBAAgB,EAAEC,YAAY,QAA4B,mBAAmB;AACtF,SAASC,SAAS,QAAgB,oBAAoB;AACtD,SACCC,6BAA6B,EAC7BC,+BAA+B,EAC/BC,8BAA8B,EAC9BC,6BAA6B,EAC7BC,4BAA4B,EAM5BC,4CAA4C,QAGtC,YAAY;;AAEnB;AACA,OAAO,MAAMC,SAAS,GAAG,CAAC;;AAE1B;;AAEA,OAAO,MAAMC,wBAAwB,GAAGA,CACvCC,cAAyB,EACzBC,iBAAkD,KAElDD,cAAc,CAACE,IAAI,KAAKL,4CAA4C,CAACM,SAAS,IAC9EH,cAAc,CAACI,MAAM,KAAKH,iBAAiB,CAACG,MAAM;AAEnD,OAAO,MAAMC,4BAA4B,GACvCJ,iBAAkD,IAAMD,cAAyB,IACjFD,wBAAwB,CAACC,cAAc,EAAEC,iBAAiB,CAAC;;AAE7D;AACA;AACA,OAAO,MAAMK,0BAA+C,GAAG,UAAAA,CAAA,EAAa;EAC3E,OAAO,IAAI;AACZ,CAAC;;AAED;AACA,OAAO,MAAMC,mBAAiC,GAAGA,CAAA,KAAM;EACtD;AACD,CAAC;AAID,MAAMC,sBAAsB,GAAIC,eAAgC,IAC/DC,MAAM,CAACC,IAAI,CAACF,eAAe,CAAC,CAACG,MAAM,CAAC,CAACC,GAAa,EAAEC,GAAG,KAAK;EAC3D,MAAMC,KAAK,GAAGN,eAAe,CAACK,GAAG,CAAC;EAClC,IAAI,CAACC,KAAK,IAAI,CAACL,MAAM,CAACM,SAAS,CAACC,cAAc,CAAC5C,IAAI,CAAC0C,KAAK,EAAE,SAAS,CAAC,EAAE;IACtE,OAAOF,GAAG;EACX;EACA,MAAMK,EAAE,GAAGH,KAAwB;EACnC,IAAI,CAAC,CAACG,EAAE,CAACC,OAAO,IAAI,CAAC,CAACD,EAAE,CAACC,OAAO,CAACC,YAAY,EAAE;IAC9CP,GAAG,CAACQ,IAAI,CAACP,GAAG,CAAC;EACd;EACA,OAAOD,GAAG;AACX,CAAC,EAAE,EAAE,CAAC;AAEP,MAAMS,kBAAkB,GAAGA,CAACb,eAAgC,EAAEc,kBAAsC,EAAEC,IAAS,KAAK;EACnH,MAAM;IAAEC,cAAc;IAAEC,WAAW;IAAEC,sBAAsB;IAAEC;EAAsB,CAAC,GAAGL,kBAAkB;EACzG,MAAMM,mBAAmB,GAAGrB,sBAAsB,CAACC,eAAe,CAAC;EACnE,MAAMqB,mBAAmB,GAAGD,mBAAmB,CAACE,MAAM,GAAG,CAAC;EAC1D;EACA,IAAI,CAACN,cAAc,CAACL,YAAY,IAAI,CAACU,mBAAmB,EAAE;IACzD,OAAON,IAAI;EACZ;EACA;EACA,IAAIE,WAAW,CAACM,MAAM,KAAK,QAAQ,EAAE;IACpC,OAAO,CAAC,CAAC;EACV;EACA;EACA,IAAIJ,qBAAqB,IAAID,sBAAsB,EAAE;IACpD,OAAOM,yBAAyB,CAACT,IAAI,EAAEK,mBAAmB,EAAEpB,eAAe,CAAC;EAC7E;EACA;EACA,OAAOgB,cAAc,CAACL,YAAY,GAC/Bc,uBAAuB,CAACV,IAAI,EAAEK,mBAAmB,EAAEpB,eAAe,CAAC,GACnEwB,yBAAyB,CAACT,IAAI,EAAEK,mBAAmB,EAAEpB,eAAe,CAAC;AACzE,CAAC;AAED,MAAMwB,yBAAyB,GAAGA,CAACT,IAAS,EAAEW,cAAwB,EAAE1B,eAAgC,KAAK;EAC5G,IAAIxC,KAAK,CAACuD,IAAI,CAAC,EAAE;IAChB,OAAOA,IAAI;EACZ;EACA,OAAOd,MAAM,CAACC,IAAI,CAACa,IAAI,CAAC,CAACZ,MAAM,CAAC,CAACC,GAAiB,EAAEC,GAAG,KAAK;IAC3D,IAAIC,KAAK,GAAGS,IAAI,CAACV,GAAG,CAAC;IACrB,IAAIqB,cAAc,CAACC,QAAQ,CAACtB,GAAG,CAAC,EAAE;MACjC,MAAMuB,oBAAoB,GAAG5B,eAAe,CAACK,GAAG,CAAoB;MACpE,MAAMe,mBAAmB,GAAGrB,sBAAsB,CAAC6B,oBAAoB,CAAC;MACxEtB,KAAK,GAAGmB,uBAAuB,CAACnB,KAAK,EAAEc,mBAAmB,EAAEQ,oBAAoB,CAAC;IAClF;IACAxB,GAAG,CAACC,GAAG,CAAC,GAAGC,KAAK;IAChB,OAAOF,GAAG;EACX,CAAC,EAAE,CAAC,CAAC,CAAC;AACP,CAAC;AAED,MAAMqB,uBAAuB,GAAGA,CAACV,IAAS,EAAEW,cAAwB,EAAE1B,eAAgC,KAAK;EAC1G,IAAIxC,KAAK,CAACuD,IAAI,CAAC,EAAE;IAChB,OAAOA,IAAI;EACZ;EACA,OAAOd,MAAM,CAACC,IAAI,CAACa,IAAI,CAAC,CAACZ,MAAM,CAAC,CAACC,GAAyB,EAAEC,GAAG,KAAK;IACnE,MAAMC,KAAK,GAAGkB,yBAAyB,CAACT,IAAI,CAACV,GAAG,CAAC,EAAEqB,cAAc,EAAE1B,eAAe,CAAC;IACnF;IACA,IAAIM,KAAK,IAAIA,KAAK,CAACuB,EAAE,EAAE;MACtBzB,GAAG,CAACE,KAAK,CAACuB,EAAE,CAAC,GAAGvB,KAAK;IACtB;IACA,OAAOF,GAAG;EACX,CAAC,EAAE,CAAC,CAAC,CAAC;AACP,CAAC;;AAED;AACA;AACA;AACA;AACA,SAAS0B,UAAUA,CAAA,EAAG;EACrB,MAAMC,sBAAsB,GAAGzD,mBAAmB,CAAC,CAAC;EACpD,IAAI,CAACyD,sBAAsB,EAAE;IAC5B,OAAO,KAAK;EACb;EAEA,MAAMC,mBAAmB,GAAGrE,MAAM,CAACsE,GAAG,CAACF,sBAAsB,CAAC;EAC9D,MAAMG,MAAM,GAAGvE,MAAM,CAACsE,GAAG,CAAC,CAAC;EAC3B,MAAME,oBAAoB,GAAG,IAAI,GAAG,EAAE,GAAG,EAAE,EAAC;;EAE5C,OAAOD,MAAM,CAACE,IAAI,CAACJ,mBAAmB,CAAC,IAAIG,oBAAoB;AAChE;;AAEA;;AAEA;;AAEA,IAAIE,gBAAkC;AACtC,IAAIC,mBAAwC;AAC5C,IAAIC,YAA0B;AAC9B,IAAIC,MAAc;;AAElB;;AAEA;AACA,OAAO,SAASC,mBAAmBA,CAACC,qBAAuC,EAAE;EAC5EL,gBAAgB,GAAGK,qBAAqB;AACzC;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,OAAO,UAAUC,UAAUA,CAC1BC,uBAAkF,EACnE;EACf,IAAI9B,kBAAkD,GAAG+B,SAAS;EAClE,IAAI;IACH/B,kBAAkB,GAAGjC,YAAY,CAAC+D,uBAAuB,EAAEP,gBAAgB,CAAC;EAC7E,CAAC,CAAC,OAAOS,KAAK,EAAE;IACf;IACAN,MAAM,CAACM,KAAK,CAACA,KAAK,CAAC;IACnBP,YAAY,CAACO,KAAK,EAAWF,uBAAuB,CAAC;IACrD;EACD;;EAEA;EACA,MAAM;IAAEG,SAAS;IAAE9B,WAAW;IAAEjB,eAAe;IAAEkB,sBAAsB;IAAE8B;EAAuB,CAAC,GAChGlC,kBAAkB;;EAEnB;EACA,MAAMmC,QAAgB,GAAGL,uBAAuB,CAACM,OAAO,GAAG,CAAC,GAAG7D,SAAS;EACxE,IAAI8D,QAAQ,GAAG,CAAC;EAChB,IAAIC,OAAO,GAAG,KAAK;EACnB,IAAIC,eAAwC;EAC5C,IAAIC,SAA4B;EAChC,IAAIC,aAAyC;EAE7C,MAAMC,gCAAgC,GAAG,4BAA4B;EACrE,MAAMC,4BAA4B,GAAG,2BAA2B;;EAEhE;EACA,GAAG;IACFL,OAAO,GAAG,KAAK;IACfD,QAAQ,EAAE;;IAEV;IACA,MAAMlF,GAAG,CAAwB;MAChCwB,IAAI,EAAEmD,uBAAuB,CAACc,OAAO,GAClCxE,6BAA6B,CAACyE,qBAAqB,GACnDzE,6BAA6B,CAAC0E,WAAW;MAC5Cb,SAAS;MACTc,IAAI,EAAEjB,uBAAuB,CAACiB;IAC/B,CAAC,CAAC;IAEF,IAAIC,WAAoC,GAAGjB,SAAS;IACpD,IAAI;MACH;MACA,MAAMkB,UAA4B,GAAG,MAAMnG,IAAI,CAAC0E,mBAAmB,EAAEM,uBAAuB,CAACoB,SAAS,CAAC;MACvG,IAAID,UAAU,EAAEE,YAAY,EAAE;QAC7BhD,WAAW,CAACiD,OAAO,GAAGxG,KAAK,CAAC,CAAC,CAAC,EAAEuD,WAAW,CAACiD,OAAO,EAAE;UACpDC,aAAa,EAAE,UAAUJ,UAAU,CAACE,YAAY;QACjD,CAAC,CAAC;MACH;MACAH,WAAW,GAAG,MAAMlG,IAAI,CAACW,SAAS,EAAE0C,WAAW,CAAC;MAChD;MACAoC,eAAe,GAAGS,WAAW;;MAE7B;MACA,IAAI,CAACA,WAAW,EAAE,MAAM,IAAIM,KAAK,CAACZ,gCAAgC,CAAC;MACnE,IAAI,CAACM,WAAW,CAACO,EAAE,EAAE,MAAM,IAAID,KAAK,CAACX,4BAA4B,CAAC;;MAElE;MACA,MAAMa,gBAAgB,GAAG1B,uBAAuB,CAACc,OAAO,GACrDzE,8BAA8B,CAACsF,+BAA+B,GAC9DtF,8BAA8B,CAACuF,qBAAqB;MAEvD,MAAMzD,IAAI,GAAGF,kBAAkB,CAACb,eAAe,EAAEc,kBAAkB,EAAEgD,WAAW,CAAC/C,IAAI,CAAC;MACtF;MACA,IAAI6B,uBAAuB,CAACiB,IAAI,IAAIpG,aAAa,CAACsD,IAAI,CAAC,EAAE;QACxDA,IAAI,CAAC8C,IAAI,GAAGjB,uBAAuB,CAACiB,IAAI;MACzC;;MAEA;MACA,IAAI3C,sBAAsB,EAAE;QAC3B,MAAMuD,eAAe,GAAG1B,SAAS,CAAC2B,KAAK,CAAC,GAAG,CAAC;QAC5C;QACAD,eAAe,CAACE,GAAG,CAAC,CAAC;QACrB;QACA,MAAM1G,GAAG,CAAyB;UACjCwB,IAAI,EAAE6E,gBAAgB;UACtBvB,SAAS,EAAE,GAAG0B,eAAe,CAACG,IAAI,CAAC,GAAG,CAAC,IAAI7D,IAAI,CAACc,EAAE,EAAE;UACpDgC,IAAI,EAAEjB,uBAAuB,CAACiB,IAAI;UAClCgB,YAAY,EAAEjC,uBAAuB,CAACiC,YAAY;UAClD9D;QACD,CAAC,CAAC;QACF;QACA,MAAM9C,GAAG,CAAuB;UAC/BwB,IAAI,EAAEN,4BAA4B;UAClC4D,SAAS,EAAEA;QACZ,CAAC,CAAC;MACH;MACA;MAAA,KACK,IAAIC,sBAAsB,EAAE;QAChC,MAAM/E,GAAG,CAAuB;UAC/BwB,IAAI,EAAEN,4BAA4B;UAClC4D,SAAS,EAAEA;QACZ,CAAC,CAAC;MACH,CAAC,MAAM;QACN,MAAM9E,GAAG,CAAyB;UACjCwB,IAAI,EAAE6E,gBAAgB;UACtBvB,SAAS,EAAEA,SAAS;UACpBc,IAAI,EAAEjB,uBAAuB,CAACiB,IAAI;UAClCgB,YAAY,EAAEjC,uBAAuB,CAACiC,YAAY;UAClD9D;QACD,CAAC,CAAC;MACH;IACD,CAAC,CAAC,OAAO+D,WAAoB,EAAE;MAC9B;MACA,MAAMC,gBAAgB,GAAG,4BAA4BhC,SAAS,GAAG;MACjE,MAAMD,KAAK,GAAGnE,6BAA6B,CAACmG,WAAW,EAAE,iBAAiB,EAAEC,gBAAgB,CAAC;MAC7F;MACA,MAAMC,eAAe,GAAGvG,aAAa,CAACwG,WAAW,CAAC,CAAC;MACnD,IAAIC,SAAyB,GAAGF,eAAe,GAAG3G,gBAAgB,GAAGD,uBAAuB;MAC5F;MACA;MACA,IAAI0F,WAAW,EAAE/C,IAAI,IAAInC,gBAAgB,CAACkF,WAAW,CAAC/C,IAAI,CAAC,EAAE;QAC5DmE,SAAS,GAAGpB,WAAW,CAAC/C,IAAI;QAC5B;QACA+B,KAAK,CAACqC,OAAO,GAAG,GAAGJ,gBAAgB,MAAM9D,WAAW,CAACM,MAAM,IAAI,KAAK,IAAIN,WAAW,CAACmE,IAAI,aACvFF,SAAS,CAACG,KAAK,KACXH,SAAS,CAACI,MAAM,GAAG;QACxB,IAAIJ,SAAS,CAACK,MAAM,EAAE;UACrBzC,KAAK,CAACqC,OAAO,IAAI,KAAKD,SAAS,CAACK,MAAM,EAAE;QACzC;MACD;;MAEA;MACAhC,aAAa,GAAG2B,SAAS;MACzB5B,SAAS,GAAGR,KAAK;;MAEjB;MACA,MAAM7E,GAAG,CAAwB;QAChCwB,IAAI,EAAEV,6BAA6B,CAACyG,gBAAgB;QACpDzC,SAAS,EAAEA,SAAS;QACpBc,IAAI,EAAEjB,uBAAuB,CAACiB,IAAI;QAClCqB;MACD,CAAC,CAAC;;MAEF;MACA1C,MAAM,CAACM,KAAK,CAACA,KAAK,CAAC;MAEnBM,OAAO,GAAG,IAAI;;MAEd;MACA,IACC8B,SAAS,CAACI,MAAM,IAAI5G,gBAAgB,CAAC+G,WAAW,IAChDP,SAAS,CAACI,MAAM,GAAG5G,gBAAgB,CAACgH,qBAAqB,IACzDR,SAAS,CAACI,MAAM,KAAK5G,gBAAgB,CAACiH,eAAe,EACpD;QACDxC,QAAQ,GAAGF,QAAQ;MACpB;;MAEA;MACA,IAAIE,QAAQ,GAAGF,QAAQ,EAAE;QACxB,MAAMlF,KAAK,CAAC,CAAC,KAAKoF,QAAQ,GAAG,CAAC,CAAC,GAAG,GAAG,CAAC,EAAC;MACxC;IACD;EACD,CAAC,QAAQA,QAAQ,GAAGF,QAAQ,IAAIG,OAAO;;EAEvC;EACA,IAAID,QAAQ,KAAKF,QAAQ,IAAIG,OAAO,EAAE;IACrC;IACA,MAAM8B,SAAS,GAAG3B,aAA+B;IACjD,MAAMT,KAAK,GAAGQ,SAAkB;;IAEhC;IACA,MAAMrF,GAAG,CAAwB;MAChCwB,IAAI,EAAEmD,uBAAuB,CAACc,OAAO,GAClC3E,6BAA6B,CAAC6G,sBAAsB,GACpD7G,6BAA6B,CAAC8G,YAAY;MAC7C9C,SAAS,EAAEA,SAAS;MACpBc,IAAI,EAAEjB,uBAAuB,CAACiB,IAAI;MAClCqB;IACD,CAAC,CAAC;;IAEF;IACA,IAAIA,SAAS,CAACI,MAAM,KAAK5G,gBAAgB,CAACoH,YAAY,EAAE;MACvDvD,YAAY,CAACO,KAAK,EAAEF,uBAAuB,EAAE3B,WAAW,EAAEoC,eAAe,EAAE6B,SAAS,CAAC;IACtF;EACD;AACD;;AAEA;AACA;AACA;AACA;AACA;AACA,OAAO,UAAUa,cAAcA,CAACC,MAAuC,EAAgB;EACtF,IAAI;IACH,IAAIC,UAAU,GAAG,KAAK;IACtB,GAAG;MACF;MACA;MACA,IAAI,CAACA,UAAU,IAAI,CAACnE,UAAU,CAAC,CAAC,EAAE;QACjC,MAAMlE,IAAI,CAAC+E,UAAU,EAAEqD,MAAM,CAAC;QAC9BC,UAAU,GAAG,IAAI;MAClB;MACA,MAAMlI,KAAK,CAACiI,MAAM,CAACE,MAAM,CAAC;IAC3B,CAAC,QAAQ,IAAI;EACd,CAAC,CAAC,OAAOpB,WAAoB,EAAE;IAC9B,MAAMhC,KAAK,GAAGnE,6BAA6B,CAACmG,WAAW,EAAE,qBAAqB,EAAE,uBAAuB,CAAC;IACxGtC,MAAM,CAACM,KAAK,CAACA,KAAK,CAAC;IACnB;IACAP,YAAY,CAACO,KAAK,EAAEkD,MAAM,CAAC;EAC5B,CAAC,SAAS;IACT,IAAI,MAAMlI,SAAS,CAAC,CAAC,EAAE;MACtB,MAAMG,GAAG,CAAsC;QAC9CwB,IAAI,EAAEL,4CAA4C,CAAC+G,SAAS;QAC5DxG,MAAM,EAAEqG,MAAM,CAACrG;MAChB,CAAC,CAAC;IACH;EACD;AACD;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,OAAO,UAAUyG,mBAAmBA,CAACJ,MAAuC,EAAgB;EAC3F,IAAI,CAACA,MAAM,CAACE,MAAM,EAAE;IACnB,MAAM,IAAI9B,KAAK,CAAC,sBAAsB,CAAC;EACxC;EACA,IAAI,CAAC4B,MAAM,CAACrG,MAAM,EAAE;IACnB,MAAM,IAAIyE,KAAK,CAAC,sBAAsB,CAAC;EACxC;EAEA,MAAMiC,UAAgB,GAAG,MAAMrI,IAAI,CAAC+H,cAAc,EAAEC,MAAM,CAAC;EAC3D,MAAM9H,IAAI,CAAC0B,4BAA4B,CAACoG,MAAM,CAAC,CAAC;EAChD,MAAMnI,MAAM,CAACwI,UAAU,CAAC;AACzB;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,SAAUC,cAAcA,CACtC5D,qBAAuC,EACvC6D,YAAqB;EAAA,IACrBC,wBAAyD,GAAAC,SAAA,CAAAnF,MAAA,QAAAmF,SAAA,QAAA5D,SAAA,GAAA4D,SAAA,MAAG5G,0BAA0B;EAAA,IACtF6G,iBAA2C,GAAAD,SAAA,CAAAnF,MAAA,QAAAmF,SAAA,QAAA5D,SAAA,GAAA4D,SAAA,MAAG3G,mBAAmB;EAAA,oBAClD;IACftB,UAAU,CAAC+H,YAAY,CAAC;IACxB9D,mBAAmB,CAACC,qBAAqB,CAAC;IAC1CH,YAAY,GAAGmE,iBAAiB;IAChCpE,mBAAmB,GAAGkE,wBAAwB;IAC9ChE,MAAM,GAAG1D,SAAS,CAAC,CAAC;IAEpB,MAAMX,SAAS,CAACa,+BAA+B,CAAC2H,aAAa,EAAEhE,UAAU,CAAC;IAC1E,MAAMxE,SAAS,CAACa,+BAA+B,CAAC4H,sBAAsB,EAAER,mBAAmB,CAAC;EAC7F,CAAC;AAAA","ignoreList":[]}