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,274 +1,309 @@
1
1
  "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.getOAuthToken = exports.takeMatchesModelFetchFailed = exports.matchesModelFetchFailed = exports.takeMatchesModelFetchReceived = exports.matchesModelFetchReceived = void 0;
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.casTicketLoginFlow = casTicketLoginFlow;
7
+ exports.casV1LoginFlow = casV1LoginFlow;
8
+ exports.credentialsLoginFlow = credentialsLoginFlow;
9
+ exports.default = authSaga;
10
+ exports.getOAuthToken = void 0;
4
11
  exports.getTokenFromCode = getTokenFromCode;
5
12
  exports.getTokenFromRefreshToken = getTokenFromRefreshToken;
6
- exports.performTokenRefresh = performTokenRefresh;
7
- exports.loginFlow = loginFlow;
8
- exports.credentialsLoginFlow = credentialsLoginFlow;
9
- exports.casV1LoginFlow = casV1LoginFlow;
10
- exports.localLoginFlow = localLoginFlow;
11
- exports.casTicketLoginFlow = casTicketLoginFlow;
12
13
  exports.handleAuthFailure = handleAuthFailure;
13
- exports.default = authSaga;
14
- const effects_1 = require("redux-saga/effects");
15
- const codeProviderService_1 = require("../../services/codeProviderService");
16
- const ticketProviderService_1 = require("../../services/ticketProviderService");
17
- const tokenPersistenceService_1 = require("../../services/tokenPersistenceService");
18
- const types_1 = require("../../types");
19
- const logger_1 = require("../../utils/logger");
20
- const actions_1 = require("../actions");
14
+ exports.localLoginFlow = localLoginFlow;
15
+ exports.loginFlow = loginFlow;
16
+ exports.matchesModelFetchReceived = exports.matchesModelFetchFailed = void 0;
17
+ exports.performTokenRefresh = performTokenRefresh;
18
+ exports.takeMatchesModelFetchReceived = exports.takeMatchesModelFetchFailed = void 0;
19
+ var _effects = require("redux-saga/effects");
20
+ var _codeProviderService = require("../../services/codeProviderService");
21
+ var _ticketProviderService = require("../../services/ticketProviderService");
22
+ var _tokenPersistenceService = require("../../services/tokenPersistenceService");
23
+ var _types = require("../../types");
24
+ var _logger = require("../../utils/logger");
25
+ var _actions = require("../actions");
21
26
  //#region Helpers
22
- const matchesModelFetchReceived = (action, modelName) => (0, actions_1.isTransientModelFetchResultAction)(action) && action.modelPath === modelName;
27
+
28
+ const matchesModelFetchReceived = (action, modelName) => (0, _actions.isTransientModelFetchResultAction)(action) && action.modelPath === modelName;
23
29
  exports.matchesModelFetchReceived = matchesModelFetchReceived;
24
- const takeMatchesModelFetchReceived = (modelName) => (incomingAction) => (0, exports.matchesModelFetchReceived)(incomingAction, modelName);
30
+ const takeMatchesModelFetchReceived = modelName => incomingAction => matchesModelFetchReceived(incomingAction, modelName);
25
31
  exports.takeMatchesModelFetchReceived = takeMatchesModelFetchReceived;
26
- const matchesModelFetchFailed = (action, modelName) => (0, actions_1.isTransientModelFetchErrorAction)(action) && action.modelPath === modelName;
32
+ const matchesModelFetchFailed = (action, modelName) => (0, _actions.isTransientModelFetchErrorAction)(action) && action.modelPath === modelName;
27
33
  exports.matchesModelFetchFailed = matchesModelFetchFailed;
28
- const takeMatchesModelFetchFailed = (modelName) => (incomingAction) => (0, exports.matchesModelFetchFailed)(incomingAction, modelName);
29
- exports.takeMatchesModelFetchFailed = takeMatchesModelFetchFailed;
34
+ const takeMatchesModelFetchFailed = modelName => incomingAction => matchesModelFetchFailed(incomingAction, modelName);
35
+
30
36
  //#endregion Helpers
37
+
31
38
  //#region Local Variables
39
+ exports.takeMatchesModelFetchFailed = takeMatchesModelFetchFailed;
32
40
  let clientCredentials;
33
41
  let oauthToken = null;
34
42
  let tokenPersistenceService;
35
43
  let refreshLock;
36
44
  let logger;
45
+
37
46
  //#endregion Local Variables
47
+
38
48
  function* getTokenFromCode(code) {
39
- const getTokenModelName = 'getToken';
40
- // Manually creating form-url-encoded body here because NOTHING else uses this content-type
41
- // but the OAuth spec requires it
42
- const formBody = [
43
- 'grant_type=authorization_code',
44
- `client_id=${clientCredentials.client_id}`,
45
- `client_secret=${clientCredentials.client_secret}`,
46
- `code=${encodeURIComponent(code)}`
47
- ];
48
- const formBodyString = formBody.join('&');
49
- yield (0, effects_1.put)({
50
- type: actions_1.MODEL_FETCH_REQUEST_ACTION_TYPE.FETCH_REQUEST,
51
- modelName: getTokenModelName,
52
- body: formBodyString,
53
- noStore: true
54
- });
55
- const { fetchResultAction, fetchErrorAction } = yield (0, effects_1.race)({
56
- fetchResultAction: (0, effects_1.take)((0, exports.takeMatchesModelFetchReceived)(getTokenModelName)),
57
- fetchErrorAction: (0, effects_1.take)((0, exports.takeMatchesModelFetchFailed)(getTokenModelName))
58
- });
59
- if (fetchErrorAction || !(fetchResultAction === null || fetchResultAction === void 0 ? void 0 : fetchResultAction.data)) {
60
- return null;
61
- }
62
- return fetchResultAction.data;
49
+ const getTokenModelName = 'getToken';
50
+ // Manually creating form-url-encoded body here because NOTHING else uses this content-type
51
+ // but the OAuth spec requires it
52
+ const formBody = ['grant_type=authorization_code', `client_id=${clientCredentials.client_id}`, `client_secret=${clientCredentials.client_secret}`, `code=${encodeURIComponent(code)}`];
53
+ const formBodyString = formBody.join('&');
54
+ yield (0, _effects.put)({
55
+ type: _actions.MODEL_FETCH_REQUEST_ACTION_TYPE.FETCH_REQUEST,
56
+ modelName: getTokenModelName,
57
+ body: formBodyString,
58
+ noStore: true
59
+ });
60
+ const {
61
+ fetchResultAction,
62
+ fetchErrorAction
63
+ } = yield (0, _effects.race)({
64
+ fetchResultAction: (0, _effects.take)(takeMatchesModelFetchReceived(getTokenModelName)),
65
+ fetchErrorAction: (0, _effects.take)(takeMatchesModelFetchFailed(getTokenModelName))
66
+ });
67
+ if (fetchErrorAction || !fetchResultAction?.data) {
68
+ return null;
69
+ }
70
+ return fetchResultAction.data;
63
71
  }
64
72
  function* getTokenFromRefreshToken(oauthTokenParam) {
65
- var _a;
66
- const getTokenModelName = 'getToken';
67
- // Manually creating form-url-encoded body here because NOTHING else uses this content-type
68
- // but the OAuth spec requires it
69
- const formBody = [
70
- 'grant_type=refresh_token',
71
- `client_id=${clientCredentials.client_id}`,
72
- `client_secret=${clientCredentials.client_secret}`,
73
- `refresh_token=${encodeURIComponent(oauthTokenParam.refresh_token)}`
74
- ];
75
- const formBodyString = formBody.join('&');
76
- yield (0, effects_1.put)({
77
- type: actions_1.MODEL_FETCH_REQUEST_ACTION_TYPE.FETCH_REQUEST,
78
- modelName: getTokenModelName,
79
- body: formBodyString,
80
- noStore: true
81
- });
82
- const { fetchResultAction, fetchErrorAction } = yield (0, effects_1.race)({
83
- fetchResultAction: (0, effects_1.take)((0, exports.takeMatchesModelFetchReceived)(getTokenModelName)),
84
- fetchErrorAction: (0, effects_1.take)((0, exports.takeMatchesModelFetchFailed)(getTokenModelName))
85
- });
86
- // any error response
87
- if (fetchErrorAction) {
88
- // ignore server errors
89
- if (((_a = fetchErrorAction.errorData) === null || _a === void 0 ? void 0 : _a.status) &&
90
- fetchErrorAction.errorData.status >= types_1.HTTP_STATUS_CODE.INTERNAL_SERVER_ERROR) {
91
- return oauthTokenParam;
92
- }
93
- return null;
73
+ const getTokenModelName = 'getToken';
74
+ // Manually creating form-url-encoded body here because NOTHING else uses this content-type
75
+ // but the OAuth spec requires it
76
+ const formBody = ['grant_type=refresh_token', `client_id=${clientCredentials.client_id}`, `client_secret=${clientCredentials.client_secret}`, `refresh_token=${encodeURIComponent(oauthTokenParam.refresh_token)}`];
77
+ const formBodyString = formBody.join('&');
78
+ yield (0, _effects.put)({
79
+ type: _actions.MODEL_FETCH_REQUEST_ACTION_TYPE.FETCH_REQUEST,
80
+ modelName: getTokenModelName,
81
+ body: formBodyString,
82
+ noStore: true
83
+ });
84
+ const {
85
+ fetchResultAction,
86
+ fetchErrorAction
87
+ } = yield (0, _effects.race)({
88
+ fetchResultAction: (0, _effects.take)(takeMatchesModelFetchReceived(getTokenModelName)),
89
+ fetchErrorAction: (0, _effects.take)(takeMatchesModelFetchFailed(getTokenModelName))
90
+ });
91
+ // any error response
92
+ if (fetchErrorAction) {
93
+ // ignore server errors
94
+ if (fetchErrorAction.errorData?.status && fetchErrorAction.errorData.status >= _types.HTTP_STATUS_CODE.INTERNAL_SERVER_ERROR) {
95
+ return oauthTokenParam;
94
96
  }
95
- // for some reason the response had no body
96
- if (!(fetchResultAction === null || fetchResultAction === void 0 ? void 0 : fetchResultAction.data)) {
97
- return null;
98
- }
99
- return fetchResultAction.data;
97
+ return null;
98
+ }
99
+ // for some reason the response had no body
100
+ if (!fetchResultAction?.data) {
101
+ return null;
102
+ }
103
+ return fetchResultAction.data;
100
104
  }
101
105
  function* performTokenRefresh() {
102
- if (refreshLock || !oauthToken) {
103
- // already refreshing. wait for the current refresh to succeed or fail.
104
- yield (0, effects_1.race)({
105
- refreshSuccess: (0, effects_1.take)(actions_1.AUTH_TOKEN_SUCCESS_ACTION_TYPE.TOKEN_REFRESH_SUCCEEDED),
106
- refreshFailed: (0, effects_1.take)(actions_1.AUTH_ACTION_TYPE.TOKEN_REFRESH_FAILED)
107
- });
108
- return;
109
- }
110
- logger.debug('Refreshing OAuth token');
111
- refreshLock = true;
112
- // oauthToken will be set to:
113
- // 1. new token (success)
114
- // 2. same token (failed from timeout or server error)
115
- // 3. null (fail)
116
- const originalAccessToken = oauthToken.access_token;
117
- oauthToken = yield (0, effects_1.call)(getTokenFromRefreshToken, oauthToken);
118
- if (!!oauthToken && oauthToken.access_token !== originalAccessToken) {
119
- logger.debug('OAuth token refreshed');
120
- yield (0, effects_1.call)(tokenPersistenceService.persistToken, oauthToken);
121
- yield (0, effects_1.put)({ type: actions_1.AUTH_TOKEN_SUCCESS_ACTION_TYPE.TOKEN_REFRESH_SUCCEEDED, oauthToken });
122
- }
123
- else if (oauthToken === null) {
124
- logger.debug('OAuth token failed to refresh');
125
- // This should never happen outside of the token having been revoked on the server side
126
- yield (0, effects_1.all)({
127
- refreshFailed: (0, effects_1.put)({ type: actions_1.AUTH_ACTION_TYPE.TOKEN_REFRESH_FAILED }),
128
- logOut: (0, effects_1.put)({ type: actions_1.AUTH_ACTION_TYPE.LOG_OUT_REQUESTED })
129
- });
130
- }
131
- refreshLock = false;
106
+ if (refreshLock || !oauthToken) {
107
+ // already refreshing. wait for the current refresh to succeed or fail.
108
+ yield (0, _effects.race)({
109
+ refreshSuccess: (0, _effects.take)(_actions.AUTH_TOKEN_SUCCESS_ACTION_TYPE.TOKEN_REFRESH_SUCCEEDED),
110
+ refreshFailed: (0, _effects.take)(_actions.AUTH_ACTION_TYPE.TOKEN_REFRESH_FAILED)
111
+ });
112
+ return;
113
+ }
114
+ logger.debug('Refreshing OAuth token');
115
+ refreshLock = true;
116
+ // oauthToken will be set to:
117
+ // 1. new token (success)
118
+ // 2. same token (failed from timeout or server error)
119
+ // 3. null (fail)
120
+ const originalAccessToken = oauthToken.access_token;
121
+ oauthToken = yield (0, _effects.call)(getTokenFromRefreshToken, oauthToken);
122
+ if (!!oauthToken && oauthToken.access_token !== originalAccessToken) {
123
+ logger.debug('OAuth token refreshed');
124
+ yield (0, _effects.call)(tokenPersistenceService.persistToken, oauthToken);
125
+ yield (0, _effects.put)({
126
+ type: _actions.AUTH_TOKEN_SUCCESS_ACTION_TYPE.TOKEN_REFRESH_SUCCEEDED,
127
+ oauthToken
128
+ });
129
+ } else if (oauthToken === null) {
130
+ logger.debug('OAuth token failed to refresh');
131
+ // This should never happen outside of the token having been revoked on the server side
132
+ yield (0, _effects.all)({
133
+ refreshFailed: (0, _effects.put)({
134
+ type: _actions.AUTH_ACTION_TYPE.TOKEN_REFRESH_FAILED
135
+ }),
136
+ logOut: (0, _effects.put)({
137
+ type: _actions.AUTH_ACTION_TYPE.LOG_OUT_REQUESTED
138
+ })
139
+ });
140
+ }
141
+ refreshLock = false;
132
142
  }
133
143
  function* loginFlow(modelFetchRequestAction) {
134
- var _a;
135
- yield (0, effects_1.put)(modelFetchRequestAction);
136
- const { fetchResultAction, fetchErrorAction } = yield (0, effects_1.race)({
137
- fetchResultAction: (0, effects_1.take)((0, exports.takeMatchesModelFetchReceived)(modelFetchRequestAction.modelName)),
138
- fetchErrorAction: (0, effects_1.take)((0, exports.takeMatchesModelFetchFailed)(modelFetchRequestAction.modelName))
139
- });
140
- if (fetchErrorAction) {
141
- return null;
142
- }
143
- const code = (_a = fetchResultAction === null || fetchResultAction === void 0 ? void 0 : fetchResultAction.data) === null || _a === void 0 ? void 0 : _a.code;
144
- if (!code) {
145
- return null;
146
- }
147
- return yield (0, effects_1.call)(getTokenFromCode, code);
144
+ yield (0, _effects.put)(modelFetchRequestAction);
145
+ const {
146
+ fetchResultAction,
147
+ fetchErrorAction
148
+ } = yield (0, _effects.race)({
149
+ fetchResultAction: (0, _effects.take)(takeMatchesModelFetchReceived(modelFetchRequestAction.modelName)),
150
+ fetchErrorAction: (0, _effects.take)(takeMatchesModelFetchFailed(modelFetchRequestAction.modelName))
151
+ });
152
+ if (fetchErrorAction) {
153
+ return null;
154
+ }
155
+ const code = fetchResultAction?.data?.code;
156
+ if (!code) {
157
+ return null;
158
+ }
159
+ return yield (0, _effects.call)(getTokenFromCode, code);
148
160
  }
149
161
  function* credentialsLoginFlow(action, modelName) {
150
- const modelFetchRequestAction = {
151
- // set required defaults
152
- type: actions_1.MODEL_FETCH_REQUEST_ACTION_TYPE.FETCH_REQUEST,
153
- modelName,
154
- noStore: true,
155
- // force no retry
156
- noRetry: true,
157
- // pass thru body
158
- body: action.body
159
- };
160
- return yield (0, effects_1.call)(loginFlow, modelFetchRequestAction);
162
+ const modelFetchRequestAction = {
163
+ // set required defaults
164
+ type: _actions.MODEL_FETCH_REQUEST_ACTION_TYPE.FETCH_REQUEST,
165
+ modelName,
166
+ noStore: true,
167
+ // force no retry
168
+ noRetry: true,
169
+ // pass thru body
170
+ body: action.body
171
+ };
172
+ return yield (0, _effects.call)(loginFlow, modelFetchRequestAction);
161
173
  }
162
174
  function* casV1LoginFlow(action) {
163
- return yield (0, effects_1.call)(credentialsLoginFlow, action, 'codeFromCasV1');
175
+ return yield (0, _effects.call)(credentialsLoginFlow, action, 'codeFromCasV1');
164
176
  }
165
177
  function* localLoginFlow(action) {
166
- return yield (0, effects_1.call)(credentialsLoginFlow, action, 'codeFromLocalCredentials');
178
+ return yield (0, _effects.call)(credentialsLoginFlow, action, 'codeFromLocalCredentials');
167
179
  }
168
180
  function* casTicketLoginFlow(ticket, service) {
169
- const modelFetchRequestAction = {
170
- type: actions_1.MODEL_FETCH_REQUEST_ACTION_TYPE.FETCH_REQUEST,
171
- modelName: 'codeFromCasTicket',
172
- noStore: true,
173
- queryParams: {
174
- ticket,
175
- service
176
- }
177
- };
178
- return yield (0, effects_1.call)(loginFlow, modelFetchRequestAction);
181
+ const modelFetchRequestAction = {
182
+ type: _actions.MODEL_FETCH_REQUEST_ACTION_TYPE.FETCH_REQUEST,
183
+ modelName: 'codeFromCasTicket',
184
+ noStore: true,
185
+ queryParams: {
186
+ ticket,
187
+ service
188
+ }
189
+ };
190
+ return yield (0, _effects.call)(loginFlow, modelFetchRequestAction);
179
191
  }
180
192
  function* handleAuthFailure(action) {
181
- // This should be unlikely since we normally have a refresh token loop happening
182
- // but if the app is backgrounded, the loop might not be caught up yet
183
- if (oauthToken && action.errorData && action.errorData.status === types_1.HTTP_STATUS_CODE.UNAUTHORIZED) {
184
- logger.debug('token expired - refreshing');
185
- yield (0, effects_1.call)(performTokenRefresh);
186
- }
193
+ // This should be unlikely since we normally have a refresh token loop happening
194
+ // but if the app is backgrounded, the loop might not be caught up yet
195
+ if (oauthToken && action.errorData && action.errorData.status === _types.HTTP_STATUS_CODE.UNAUTHORIZED) {
196
+ logger.debug('token expired - refreshing');
197
+ yield (0, _effects.call)(performTokenRefresh);
198
+ }
187
199
  }
188
200
  const getOAuthToken = function* (modelName) {
189
- // Don't try to refresh the token if we're already in a request to refresh the token
190
- if (modelName === 'getToken') {
191
- return null;
201
+ // Don't try to refresh the token if we're already in a request to refresh the token
202
+ if (modelName === 'getToken') {
203
+ return null;
204
+ }
205
+ if (oauthToken && oauthToken['.expires']) {
206
+ const thirtySecondsFromNow = new Date();
207
+ thirtySecondsFromNow.setSeconds(thirtySecondsFromNow.getSeconds() + 30);
208
+ if (new Date(oauthToken['.expires']) < thirtySecondsFromNow) {
209
+ // start a token refresh and wait for the success action in case another refresh is currently happening
210
+ yield (0, _effects.call)(performTokenRefresh);
211
+ return oauthToken;
192
212
  }
193
- if (oauthToken && oauthToken['.expires']) {
194
- const thirtySecondsFromNow = new Date();
195
- thirtySecondsFromNow.setSeconds(thirtySecondsFromNow.getSeconds() + 30);
196
- if (new Date(oauthToken['.expires']) < thirtySecondsFromNow) {
197
- // start a token refresh and wait for the success action in case another refresh is currently happening
198
- yield (0, effects_1.call)(performTokenRefresh);
199
- return oauthToken;
200
- }
201
- }
202
- return oauthToken;
213
+ }
214
+ return oauthToken;
203
215
  };
204
216
  exports.getOAuthToken = getOAuthToken;
205
- function* authSaga(clientCredentialsParam, tokenPersistenceServiceParam = tokenPersistenceService_1.tokenPersistenceService, ticketProviderService = ticketProviderService_1.ticketProviderService, codeProviderService = codeProviderService_1.codeProviderService) {
206
- logger = (0, logger_1.getLogger)();
217
+ function authSaga(clientCredentialsParam) {
218
+ let tokenPersistenceServiceParam = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : _tokenPersistenceService.tokenPersistenceService;
219
+ let ticketProviderService = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : _ticketProviderService.ticketProviderService;
220
+ let codeProviderService = arguments.length > 3 && arguments[3] !== undefined ? arguments[3] : _codeProviderService.codeProviderService;
221
+ return function* () {
222
+ logger = (0, _logger.getLogger)();
207
223
  clientCredentials = clientCredentialsParam;
208
224
  tokenPersistenceService = tokenPersistenceServiceParam;
225
+
209
226
  // Try to get persisted token (normally in AsyncStorage or LocalStorage)
210
- oauthToken = yield (0, effects_1.call)(tokenPersistenceService.getPersistedToken);
227
+ oauthToken = yield (0, _effects.call)(tokenPersistenceService.getPersistedToken);
228
+
211
229
  // If no token, try to get CAS ticket (normally in the URL), use it to get a token
212
230
  if (!oauthToken) {
213
- const casTicket = ticketProviderService.getTicket();
214
- ticketProviderService.removeTicket();
215
- const service = ticketProviderService.getAppServiceName();
216
- if (casTicket && service) {
217
- oauthToken = yield (0, effects_1.call)(casTicketLoginFlow, casTicket, service);
218
- }
231
+ const casTicket = ticketProviderService.getTicket();
232
+ ticketProviderService.removeTicket();
233
+ const service = ticketProviderService.getAppServiceName();
234
+ if (casTicket && service) {
235
+ oauthToken = yield (0, _effects.call)(casTicketLoginFlow, casTicket, service);
236
+ }
219
237
  }
238
+
220
239
  // If OAuth Code exists (normally in the URL), use it to get a token
221
240
  // e.g. LTI, Shibboleth, Facebook, Google
222
241
  const code = codeProviderService.getCode();
223
242
  if (code) {
224
- // Log out the current user if a new token is about to be generated from the OAuth Code
225
- if (oauthToken) {
226
- yield (0, effects_1.all)({
227
- clearUserData: (0, effects_1.put)({ type: actions_1.MODEL_REMOVE_KEY_ACTION_TYPE, modelPath: 'user' }),
228
- clearPersistentToken: (0, effects_1.call)(tokenPersistenceService.persistToken, null)
229
- });
230
- oauthToken = null;
231
- }
232
- oauthToken = yield (0, effects_1.call)(getTokenFromCode, code);
243
+ // Log out the current user if a new token is about to be generated from the OAuth Code
244
+ if (oauthToken) {
245
+ yield (0, _effects.all)({
246
+ clearUserData: (0, _effects.put)({
247
+ type: _actions.MODEL_REMOVE_KEY_ACTION_TYPE,
248
+ modelPath: 'user'
249
+ }),
250
+ clearPersistentToken: (0, _effects.call)(tokenPersistenceService.persistToken, null)
251
+ });
252
+ oauthToken = null;
253
+ }
254
+ oauthToken = yield (0, _effects.call)(getTokenFromCode, code);
233
255
  }
234
256
  codeProviderService.removeCode();
235
- yield (0, effects_1.put)({ type: actions_1.AUTH_TOKEN_ACTION_TYPE.AUTH_INITIALIZED, oauthToken });
236
- yield (0, effects_1.takeEvery)(actions_1.MODEL_FETCH_ERROR_ACTION_TYPE.TRY_FETCH_FAILED, handleAuthFailure);
257
+ yield (0, _effects.put)({
258
+ type: _actions.AUTH_TOKEN_ACTION_TYPE.AUTH_INITIALIZED,
259
+ oauthToken
260
+ });
261
+ yield (0, _effects.takeEvery)(_actions.MODEL_FETCH_ERROR_ACTION_TYPE.TRY_FETCH_FAILED, handleAuthFailure);
237
262
  do {
238
- if (!oauthToken) {
239
- const { casV1Action, localLoginAction } = yield (0, effects_1.race)({
240
- casV1Action: (0, effects_1.take)(actions_1.AUTH_CAS_V1_LOGIN_REQUEST_ACTION_TYPE.CAS_V1_LOGIN_REQUEST),
241
- localLoginAction: (0, effects_1.take)(actions_1.AUTH_LOCAL_LOGIN_REQUEST_ACTION_TYPE.LOCAL_LOGIN_REQUEST)
242
- });
243
- yield (0, effects_1.put)({ type: actions_1.AUTH_ACTION_TYPE.LOGIN_REQUESTED });
244
- if (casV1Action) {
245
- oauthToken = yield (0, effects_1.call)(casV1LoginFlow, casV1Action);
246
- }
247
- else if (localLoginAction) {
248
- oauthToken = yield (0, effects_1.call)(localLoginFlow, localLoginAction);
249
- }
250
- }
251
- if (oauthToken) {
252
- yield (0, effects_1.call)(tokenPersistenceService.persistToken, oauthToken);
253
- yield (0, effects_1.all)({
254
- loginSuccess: (0, effects_1.put)({
255
- type: actions_1.AUTH_TOKEN_SUCCESS_ACTION_TYPE.GET_TOKEN_SUCCEEDED,
256
- oauthToken
257
- }),
258
- getUserInfo: (0, effects_1.put)({
259
- type: actions_1.MODEL_FETCH_REQUEST_ACTION_TYPE.FETCH_REQUEST,
260
- modelName: 'user.userInfo'
261
- }),
262
- logOut: (0, effects_1.take)(actions_1.AUTH_ACTION_TYPE.LOG_OUT_REQUESTED)
263
- });
264
- }
265
- else {
266
- yield (0, effects_1.put)({ type: actions_1.AUTH_ACTION_TYPE.LOGIN_FAILED });
263
+ if (!oauthToken) {
264
+ const {
265
+ casV1Action,
266
+ localLoginAction
267
+ } = yield (0, _effects.race)({
268
+ casV1Action: (0, _effects.take)(_actions.AUTH_CAS_V1_LOGIN_REQUEST_ACTION_TYPE.CAS_V1_LOGIN_REQUEST),
269
+ localLoginAction: (0, _effects.take)(_actions.AUTH_LOCAL_LOGIN_REQUEST_ACTION_TYPE.LOCAL_LOGIN_REQUEST)
270
+ });
271
+ yield (0, _effects.put)({
272
+ type: _actions.AUTH_ACTION_TYPE.LOGIN_REQUESTED
273
+ });
274
+ if (casV1Action) {
275
+ oauthToken = yield (0, _effects.call)(casV1LoginFlow, casV1Action);
276
+ } else if (localLoginAction) {
277
+ oauthToken = yield (0, _effects.call)(localLoginFlow, localLoginAction);
267
278
  }
268
- yield (0, effects_1.all)({
269
- clearUserData: (0, effects_1.put)({ type: actions_1.MODEL_REMOVE_KEY_ACTION_TYPE, modelPath: 'user' }),
270
- clearPersistentToken: (0, effects_1.call)(tokenPersistenceService.persistToken, null)
279
+ }
280
+ if (oauthToken) {
281
+ yield (0, _effects.call)(tokenPersistenceService.persistToken, oauthToken);
282
+ yield (0, _effects.all)({
283
+ loginSuccess: (0, _effects.put)({
284
+ type: _actions.AUTH_TOKEN_SUCCESS_ACTION_TYPE.GET_TOKEN_SUCCEEDED,
285
+ oauthToken
286
+ }),
287
+ getUserInfo: (0, _effects.put)({
288
+ type: _actions.MODEL_FETCH_REQUEST_ACTION_TYPE.FETCH_REQUEST,
289
+ modelName: 'user.userInfo'
290
+ }),
291
+ logOut: (0, _effects.take)(_actions.AUTH_ACTION_TYPE.LOG_OUT_REQUESTED)
271
292
  });
272
- oauthToken = null;
293
+ } else {
294
+ yield (0, _effects.put)({
295
+ type: _actions.AUTH_ACTION_TYPE.LOGIN_FAILED
296
+ });
297
+ }
298
+ yield (0, _effects.all)({
299
+ clearUserData: (0, _effects.put)({
300
+ type: _actions.MODEL_REMOVE_KEY_ACTION_TYPE,
301
+ modelPath: 'user'
302
+ }),
303
+ clearPersistentToken: (0, _effects.call)(tokenPersistenceService.persistToken, null)
304
+ });
305
+ oauthToken = null;
273
306
  } while (true);
307
+ }();
274
308
  }
309
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJfZWZmZWN0cyIsInJlcXVpcmUiLCJfY29kZVByb3ZpZGVyU2VydmljZSIsIl90aWNrZXRQcm92aWRlclNlcnZpY2UiLCJfdG9rZW5QZXJzaXN0ZW5jZVNlcnZpY2UiLCJfdHlwZXMiLCJfbG9nZ2VyIiwiX2FjdGlvbnMiLCJtYXRjaGVzTW9kZWxGZXRjaFJlY2VpdmVkIiwiYWN0aW9uIiwibW9kZWxOYW1lIiwiaXNUcmFuc2llbnRNb2RlbEZldGNoUmVzdWx0QWN0aW9uIiwibW9kZWxQYXRoIiwiZXhwb3J0cyIsInRha2VNYXRjaGVzTW9kZWxGZXRjaFJlY2VpdmVkIiwiaW5jb21pbmdBY3Rpb24iLCJtYXRjaGVzTW9kZWxGZXRjaEZhaWxlZCIsImlzVHJhbnNpZW50TW9kZWxGZXRjaEVycm9yQWN0aW9uIiwidGFrZU1hdGNoZXNNb2RlbEZldGNoRmFpbGVkIiwiY2xpZW50Q3JlZGVudGlhbHMiLCJvYXV0aFRva2VuIiwidG9rZW5QZXJzaXN0ZW5jZVNlcnZpY2UiLCJyZWZyZXNoTG9jayIsImxvZ2dlciIsImdldFRva2VuRnJvbUNvZGUiLCJjb2RlIiwiZ2V0VG9rZW5Nb2RlbE5hbWUiLCJmb3JtQm9keSIsImNsaWVudF9pZCIsImNsaWVudF9zZWNyZXQiLCJlbmNvZGVVUklDb21wb25lbnQiLCJmb3JtQm9keVN0cmluZyIsImpvaW4iLCJwdXQiLCJ0eXBlIiwiTU9ERUxfRkVUQ0hfUkVRVUVTVF9BQ1RJT05fVFlQRSIsIkZFVENIX1JFUVVFU1QiLCJib2R5Iiwibm9TdG9yZSIsImZldGNoUmVzdWx0QWN0aW9uIiwiZmV0Y2hFcnJvckFjdGlvbiIsInJhY2UiLCJ0YWtlIiwiZGF0YSIsImdldFRva2VuRnJvbVJlZnJlc2hUb2tlbiIsIm9hdXRoVG9rZW5QYXJhbSIsInJlZnJlc2hfdG9rZW4iLCJlcnJvckRhdGEiLCJzdGF0dXMiLCJIVFRQX1NUQVRVU19DT0RFIiwiSU5URVJOQUxfU0VSVkVSX0VSUk9SIiwicGVyZm9ybVRva2VuUmVmcmVzaCIsInJlZnJlc2hTdWNjZXNzIiwiQVVUSF9UT0tFTl9TVUNDRVNTX0FDVElPTl9UWVBFIiwiVE9LRU5fUkVGUkVTSF9TVUNDRUVERUQiLCJyZWZyZXNoRmFpbGVkIiwiQVVUSF9BQ1RJT05fVFlQRSIsIlRPS0VOX1JFRlJFU0hfRkFJTEVEIiwiZGVidWciLCJvcmlnaW5hbEFjY2Vzc1Rva2VuIiwiYWNjZXNzX3Rva2VuIiwiY2FsbCIsInBlcnNpc3RUb2tlbiIsImFsbCIsImxvZ091dCIsIkxPR19PVVRfUkVRVUVTVEVEIiwibG9naW5GbG93IiwibW9kZWxGZXRjaFJlcXVlc3RBY3Rpb24iLCJjcmVkZW50aWFsc0xvZ2luRmxvdyIsIm5vUmV0cnkiLCJjYXNWMUxvZ2luRmxvdyIsImxvY2FsTG9naW5GbG93IiwiY2FzVGlja2V0TG9naW5GbG93IiwidGlja2V0Iiwic2VydmljZSIsInF1ZXJ5UGFyYW1zIiwiaGFuZGxlQXV0aEZhaWx1cmUiLCJVTkFVVEhPUklaRUQiLCJnZXRPQXV0aFRva2VuIiwidGhpcnR5U2Vjb25kc0Zyb21Ob3ciLCJEYXRlIiwic2V0U2Vjb25kcyIsImdldFNlY29uZHMiLCJhdXRoU2FnYSIsImNsaWVudENyZWRlbnRpYWxzUGFyYW0iLCJ0b2tlblBlcnNpc3RlbmNlU2VydmljZVBhcmFtIiwiYXJndW1lbnRzIiwibGVuZ3RoIiwidW5kZWZpbmVkIiwiZGVmYXVsdFRva2VuUGVyc2lzdGVuY2VTZXJ2aWNlIiwidGlja2V0UHJvdmlkZXJTZXJ2aWNlIiwiZGVmYXVsdFRpY2tldFByb3ZpZGVyU2VydmljZSIsImNvZGVQcm92aWRlclNlcnZpY2UiLCJkZWZhdWx0Q29kZVByb3ZpZGVyU2VydmljZSIsImdldExvZ2dlciIsImdldFBlcnNpc3RlZFRva2VuIiwiY2FzVGlja2V0IiwiZ2V0VGlja2V0IiwicmVtb3ZlVGlja2V0IiwiZ2V0QXBwU2VydmljZU5hbWUiLCJnZXRDb2RlIiwiY2xlYXJVc2VyRGF0YSIsIk1PREVMX1JFTU9WRV9LRVlfQUNUSU9OX1RZUEUiLCJjbGVhclBlcnNpc3RlbnRUb2tlbiIsInJlbW92ZUNvZGUiLCJBVVRIX1RPS0VOX0FDVElPTl9UWVBFIiwiQVVUSF9JTklUSUFMSVpFRCIsInRha2VFdmVyeSIsIk1PREVMX0ZFVENIX0VSUk9SX0FDVElPTl9UWVBFIiwiVFJZX0ZFVENIX0ZBSUxFRCIsImNhc1YxQWN0aW9uIiwibG9jYWxMb2dpbkFjdGlvbiIsIkFVVEhfQ0FTX1YxX0xPR0lOX1JFUVVFU1RfQUNUSU9OX1RZUEUiLCJDQVNfVjFfTE9HSU5fUkVRVUVTVCIsIkFVVEhfTE9DQUxfTE9HSU5fUkVRVUVTVF9BQ1RJT05fVFlQRSIsIkxPQ0FMX0xPR0lOX1JFUVVFU1QiLCJMT0dJTl9SRVFVRVNURUQiLCJsb2dpblN1Y2Nlc3MiLCJHRVRfVE9LRU5fU1VDQ0VFREVEIiwiZ2V0VXNlckluZm8iLCJMT0dJTl9GQUlMRUQiXSwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvcmVkdXgvc2FnYXMvYXV0aFNhZ2EudHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgU2FnYUl0ZXJhdG9yIH0gZnJvbSAnQHJlZHV4LXNhZ2EvY29yZSdcclxuaW1wb3J0IHsgQW55QWN0aW9uIH0gZnJvbSAncmVkdXgnXHJcbmltcG9ydCB7IGFsbCwgY2FsbCwgcHV0LCByYWNlLCB0YWtlLCB0YWtlRXZlcnkgfSBmcm9tICdyZWR1eC1zYWdhL2VmZmVjdHMnXHJcbmltcG9ydCB7IGNvZGVQcm92aWRlclNlcnZpY2UgYXMgZGVmYXVsdENvZGVQcm92aWRlclNlcnZpY2UgfSBmcm9tICcuLi8uLi9zZXJ2aWNlcy9jb2RlUHJvdmlkZXJTZXJ2aWNlJ1xyXG5pbXBvcnQgeyB0aWNrZXRQcm92aWRlclNlcnZpY2UgYXMgZGVmYXVsdFRpY2tldFByb3ZpZGVyU2VydmljZSB9IGZyb20gJy4uLy4uL3NlcnZpY2VzL3RpY2tldFByb3ZpZGVyU2VydmljZSdcclxuaW1wb3J0IHsgdG9rZW5QZXJzaXN0ZW5jZVNlcnZpY2UgYXMgZGVmYXVsdFRva2VuUGVyc2lzdGVuY2VTZXJ2aWNlIH0gZnJvbSAnLi4vLi4vc2VydmljZXMvdG9rZW5QZXJzaXN0ZW5jZVNlcnZpY2UnXHJcbmltcG9ydCB7XHJcblx0Q2xpZW50Q3JlZGVudGlhbHMsXHJcblx0Q29kZVByb3ZpZGVyU2VydmljZSxcclxuXHRIVFRQX1NUQVRVU19DT0RFLFxyXG5cdE9BdXRoVG9rZW4sXHJcblx0T0F1dGhUb2tlbk9yTnVsbCxcclxuXHRUaWNrZXRQcm92aWRlclNlcnZpY2UsXHJcblx0VG9rZW5BY2Nlc3NGdW5jdGlvbixcclxuXHRUb2tlblBlcnNpc3RlbmNlU2VydmljZVxyXG59IGZyb20gJy4uLy4uL3R5cGVzJ1xyXG5pbXBvcnQgeyBnZXRMb2dnZXIsIExvZ2dlciB9IGZyb20gJy4uLy4uL3V0aWxzL2xvZ2dlcidcclxuaW1wb3J0IHtcclxuXHRBVVRIX0FDVElPTl9UWVBFLFxyXG5cdEFVVEhfQ0FTX1YxX0xPR0lOX1JFUVVFU1RfQUNUSU9OX1RZUEUsXHJcblx0QVVUSF9MT0NBTF9MT0dJTl9SRVFVRVNUX0FDVElPTl9UWVBFLFxyXG5cdEFVVEhfVE9LRU5fQUNUSU9OX1RZUEUsXHJcblx0QVVUSF9UT0tFTl9TVUNDRVNTX0FDVElPTl9UWVBFLFxyXG5cdEF1dGhBY3Rpb24sXHJcblx0QXV0aENhc1YxTG9naW5SZXF1ZXN0QWN0aW9uLFxyXG5cdEF1dGhMb2NhbExvZ2luUmVxdWVzdEFjdGlvbixcclxuXHRBdXRoVG9rZW5BY3Rpb24sXHJcblx0QXV0aFRva2VuU3VjY2Vzc0FjdGlvbixcclxuXHRpc1RyYW5zaWVudE1vZGVsRmV0Y2hFcnJvckFjdGlvbixcclxuXHRpc1RyYW5zaWVudE1vZGVsRmV0Y2hSZXN1bHRBY3Rpb24sXHJcblx0TU9ERUxfRkVUQ0hfRVJST1JfQUNUSU9OX1RZUEUsXHJcblx0TU9ERUxfRkVUQ0hfUkVRVUVTVF9BQ1RJT05fVFlQRSxcclxuXHRNT0RFTF9SRU1PVkVfS0VZX0FDVElPTl9UWVBFLFxyXG5cdE1vZGVsRmV0Y2hFcnJvckFjdGlvbixcclxuXHRNb2RlbEZldGNoUmVxdWVzdEFjdGlvbixcclxuXHRNb2RlbEZldGNoUmVzdWx0QWN0aW9uLFxyXG5cdE1vZGVsUmVtb3ZlS2V5QWN0aW9uXHJcbn0gZnJvbSAnLi4vYWN0aW9ucydcclxuXHJcbi8vI3JlZ2lvbiBIZWxwZXJzXHJcblxyXG5leHBvcnQgY29uc3QgbWF0Y2hlc01vZGVsRmV0Y2hSZWNlaXZlZCA9IChhY3Rpb246IEFueUFjdGlvbiwgbW9kZWxOYW1lOiBzdHJpbmcpID0+XHJcblx0aXNUcmFuc2llbnRNb2RlbEZldGNoUmVzdWx0QWN0aW9uKGFjdGlvbikgJiYgYWN0aW9uLm1vZGVsUGF0aCA9PT0gbW9kZWxOYW1lXHJcblxyXG5leHBvcnQgY29uc3QgdGFrZU1hdGNoZXNNb2RlbEZldGNoUmVjZWl2ZWQgPSAobW9kZWxOYW1lOiBzdHJpbmcpID0+IChpbmNvbWluZ0FjdGlvbjogQW55QWN0aW9uKSA9PlxyXG5cdG1hdGNoZXNNb2RlbEZldGNoUmVjZWl2ZWQoaW5jb21pbmdBY3Rpb24sIG1vZGVsTmFtZSlcclxuXHJcbmV4cG9ydCBjb25zdCBtYXRjaGVzTW9kZWxGZXRjaEZhaWxlZCA9IChhY3Rpb246IEFueUFjdGlvbiwgbW9kZWxOYW1lOiBzdHJpbmcpID0+XHJcblx0aXNUcmFuc2llbnRNb2RlbEZldGNoRXJyb3JBY3Rpb24oYWN0aW9uKSAmJiBhY3Rpb24ubW9kZWxQYXRoID09PSBtb2RlbE5hbWVcclxuXHJcbmV4cG9ydCBjb25zdCB0YWtlTWF0Y2hlc01vZGVsRmV0Y2hGYWlsZWQgPSAobW9kZWxOYW1lOiBzdHJpbmcpID0+IChpbmNvbWluZ0FjdGlvbjogQW55QWN0aW9uKSA9PlxyXG5cdG1hdGNoZXNNb2RlbEZldGNoRmFpbGVkKGluY29taW5nQWN0aW9uLCBtb2RlbE5hbWUpXHJcblxyXG4vLyNlbmRyZWdpb24gSGVscGVyc1xyXG5cclxuLy8jcmVnaW9uIExvY2FsIFZhcmlhYmxlc1xyXG5cclxubGV0IGNsaWVudENyZWRlbnRpYWxzOiBDbGllbnRDcmVkZW50aWFsc1xyXG5sZXQgb2F1dGhUb2tlbjogT0F1dGhUb2tlbk9yTnVsbCA9IG51bGxcclxubGV0IHRva2VuUGVyc2lzdGVuY2VTZXJ2aWNlOiBUb2tlblBlcnNpc3RlbmNlU2VydmljZVxyXG5sZXQgcmVmcmVzaExvY2s6IGJvb2xlYW5cclxubGV0IGxvZ2dlcjogTG9nZ2VyXHJcblxyXG4vLyNlbmRyZWdpb24gTG9jYWwgVmFyaWFibGVzXHJcblxyXG5leHBvcnQgZnVuY3Rpb24qIGdldFRva2VuRnJvbUNvZGUoY29kZTogc3RyaW5nKTogU2FnYUl0ZXJhdG9yIHtcclxuXHRjb25zdCBnZXRUb2tlbk1vZGVsTmFtZSA9ICdnZXRUb2tlbidcclxuXHQvLyBNYW51YWxseSBjcmVhdGluZyBmb3JtLXVybC1lbmNvZGVkIGJvZHkgaGVyZSBiZWNhdXNlIE5PVEhJTkcgZWxzZSB1c2VzIHRoaXMgY29udGVudC10eXBlXHJcblx0Ly8gYnV0IHRoZSBPQXV0aCBzcGVjIHJlcXVpcmVzIGl0XHJcblx0Y29uc3QgZm9ybUJvZHkgPSBbXHJcblx0XHQnZ3JhbnRfdHlwZT1hdXRob3JpemF0aW9uX2NvZGUnLFxyXG5cdFx0YGNsaWVudF9pZD0ke2NsaWVudENyZWRlbnRpYWxzLmNsaWVudF9pZH1gLFxyXG5cdFx0YGNsaWVudF9zZWNyZXQ9JHtjbGllbnRDcmVkZW50aWFscy5jbGllbnRfc2VjcmV0fWAsXHJcblx0XHRgY29kZT0ke2VuY29kZVVSSUNvbXBvbmVudChjb2RlKX1gXHJcblx0XVxyXG5cdGNvbnN0IGZvcm1Cb2R5U3RyaW5nID0gZm9ybUJvZHkuam9pbignJicpXHJcblx0eWllbGQgcHV0PE1vZGVsRmV0Y2hSZXF1ZXN0QWN0aW9uPih7XHJcblx0XHR0eXBlOiBNT0RFTF9GRVRDSF9SRVFVRVNUX0FDVElPTl9UWVBFLkZFVENIX1JFUVVFU1QsXHJcblx0XHRtb2RlbE5hbWU6IGdldFRva2VuTW9kZWxOYW1lLFxyXG5cdFx0Ym9keTogZm9ybUJvZHlTdHJpbmcsXHJcblx0XHRub1N0b3JlOiB0cnVlXHJcblx0fSlcclxuXHRjb25zdCB7XHJcblx0XHRmZXRjaFJlc3VsdEFjdGlvbixcclxuXHRcdGZldGNoRXJyb3JBY3Rpb25cclxuXHR9OiB7IGZldGNoUmVzdWx0QWN0aW9uPzogTW9kZWxGZXRjaFJlc3VsdEFjdGlvbjsgZmV0Y2hFcnJvckFjdGlvbj86IE1vZGVsRmV0Y2hFcnJvckFjdGlvbiB9ID0geWllbGQgcmFjZSh7XHJcblx0XHRmZXRjaFJlc3VsdEFjdGlvbjogdGFrZSh0YWtlTWF0Y2hlc01vZGVsRmV0Y2hSZWNlaXZlZChnZXRUb2tlbk1vZGVsTmFtZSkpLFxyXG5cdFx0ZmV0Y2hFcnJvckFjdGlvbjogdGFrZSh0YWtlTWF0Y2hlc01vZGVsRmV0Y2hGYWlsZWQoZ2V0VG9rZW5Nb2RlbE5hbWUpKVxyXG5cdH0pXHJcblx0aWYgKGZldGNoRXJyb3JBY3Rpb24gfHwgIWZldGNoUmVzdWx0QWN0aW9uPy5kYXRhKSB7XHJcblx0XHRyZXR1cm4gbnVsbFxyXG5cdH1cclxuXHRyZXR1cm4gZmV0Y2hSZXN1bHRBY3Rpb24uZGF0YVxyXG59XHJcblxyXG5leHBvcnQgZnVuY3Rpb24qIGdldFRva2VuRnJvbVJlZnJlc2hUb2tlbihvYXV0aFRva2VuUGFyYW06IE9BdXRoVG9rZW4pOiBTYWdhSXRlcmF0b3Ige1xyXG5cdGNvbnN0IGdldFRva2VuTW9kZWxOYW1lID0gJ2dldFRva2VuJ1xyXG5cdC8vIE1hbnVhbGx5IGNyZWF0aW5nIGZvcm0tdXJsLWVuY29kZWQgYm9keSBoZXJlIGJlY2F1c2UgTk9USElORyBlbHNlIHVzZXMgdGhpcyBjb250ZW50LXR5cGVcclxuXHQvLyBidXQgdGhlIE9BdXRoIHNwZWMgcmVxdWlyZXMgaXRcclxuXHRjb25zdCBmb3JtQm9keSA9IFtcclxuXHRcdCdncmFudF90eXBlPXJlZnJlc2hfdG9rZW4nLFxyXG5cdFx0YGNsaWVudF9pZD0ke2NsaWVudENyZWRlbnRpYWxzLmNsaWVudF9pZH1gLFxyXG5cdFx0YGNsaWVudF9zZWNyZXQ9JHtjbGllbnRDcmVkZW50aWFscy5jbGllbnRfc2VjcmV0fWAsXHJcblx0XHRgcmVmcmVzaF90b2tlbj0ke2VuY29kZVVSSUNvbXBvbmVudChvYXV0aFRva2VuUGFyYW0ucmVmcmVzaF90b2tlbil9YFxyXG5cdF1cclxuXHRjb25zdCBmb3JtQm9keVN0cmluZyA9IGZvcm1Cb2R5LmpvaW4oJyYnKVxyXG5cdHlpZWxkIHB1dDxNb2RlbEZldGNoUmVxdWVzdEFjdGlvbj4oe1xyXG5cdFx0dHlwZTogTU9ERUxfRkVUQ0hfUkVRVUVTVF9BQ1RJT05fVFlQRS5GRVRDSF9SRVFVRVNULFxyXG5cdFx0bW9kZWxOYW1lOiBnZXRUb2tlbk1vZGVsTmFtZSxcclxuXHRcdGJvZHk6IGZvcm1Cb2R5U3RyaW5nLFxyXG5cdFx0bm9TdG9yZTogdHJ1ZVxyXG5cdH0pXHJcblx0Y29uc3Qge1xyXG5cdFx0ZmV0Y2hSZXN1bHRBY3Rpb24sXHJcblx0XHRmZXRjaEVycm9yQWN0aW9uXHJcblx0fTogeyBmZXRjaFJlc3VsdEFjdGlvbj86IE1vZGVsRmV0Y2hSZXN1bHRBY3Rpb247IGZldGNoRXJyb3JBY3Rpb24/OiBNb2RlbEZldGNoRXJyb3JBY3Rpb24gfSA9IHlpZWxkIHJhY2Uoe1xyXG5cdFx0ZmV0Y2hSZXN1bHRBY3Rpb246IHRha2UodGFrZU1hdGNoZXNNb2RlbEZldGNoUmVjZWl2ZWQoZ2V0VG9rZW5Nb2RlbE5hbWUpKSxcclxuXHRcdGZldGNoRXJyb3JBY3Rpb246IHRha2UodGFrZU1hdGNoZXNNb2RlbEZldGNoRmFpbGVkKGdldFRva2VuTW9kZWxOYW1lKSlcclxuXHR9KVxyXG5cdC8vIGFueSBlcnJvciByZXNwb25zZVxyXG5cdGlmIChmZXRjaEVycm9yQWN0aW9uKSB7XHJcblx0XHQvLyBpZ25vcmUgc2VydmVyIGVycm9yc1xyXG5cdFx0aWYgKFxyXG5cdFx0XHRmZXRjaEVycm9yQWN0aW9uLmVycm9yRGF0YT8uc3RhdHVzICYmXHJcblx0XHRcdGZldGNoRXJyb3JBY3Rpb24uZXJyb3JEYXRhLnN0YXR1cyA+PSBIVFRQX1NUQVRVU19DT0RFLklOVEVSTkFMX1NFUlZFUl9FUlJPUlxyXG5cdFx0KSB7XHJcblx0XHRcdHJldHVybiBvYXV0aFRva2VuUGFyYW1cclxuXHRcdH1cclxuXHRcdHJldHVybiBudWxsXHJcblx0fVxyXG5cdC8vIGZvciBzb21lIHJlYXNvbiB0aGUgcmVzcG9uc2UgaGFkIG5vIGJvZHlcclxuXHRpZiAoIWZldGNoUmVzdWx0QWN0aW9uPy5kYXRhKSB7XHJcblx0XHRyZXR1cm4gbnVsbFxyXG5cdH1cclxuXHRyZXR1cm4gZmV0Y2hSZXN1bHRBY3Rpb24uZGF0YVxyXG59XHJcblxyXG5leHBvcnQgZnVuY3Rpb24qIHBlcmZvcm1Ub2tlblJlZnJlc2goKTogU2FnYUl0ZXJhdG9yIHtcclxuXHRpZiAocmVmcmVzaExvY2sgfHwgIW9hdXRoVG9rZW4pIHtcclxuXHRcdC8vIGFscmVhZHkgcmVmcmVzaGluZy4gd2FpdCBmb3IgdGhlIGN1cnJlbnQgcmVmcmVzaCB0byBzdWNjZWVkIG9yIGZhaWwuXHJcblx0XHR5aWVsZCByYWNlKHtcclxuXHRcdFx0cmVmcmVzaFN1Y2Nlc3M6IHRha2UoQVVUSF9UT0tFTl9TVUNDRVNTX0FDVElPTl9UWVBFLlRPS0VOX1JFRlJFU0hfU1VDQ0VFREVEKSxcclxuXHRcdFx0cmVmcmVzaEZhaWxlZDogdGFrZShBVVRIX0FDVElPTl9UWVBFLlRPS0VOX1JFRlJFU0hfRkFJTEVEKVxyXG5cdFx0fSlcclxuXHRcdHJldHVyblxyXG5cdH1cclxuXHRsb2dnZXIuZGVidWcoJ1JlZnJlc2hpbmcgT0F1dGggdG9rZW4nKVxyXG5cdHJlZnJlc2hMb2NrID0gdHJ1ZVxyXG5cdC8vIG9hdXRoVG9rZW4gd2lsbCBiZSBzZXQgdG86XHJcblx0Ly8gMS4gbmV3IHRva2VuIChzdWNjZXNzKVxyXG5cdC8vIDIuIHNhbWUgdG9rZW4gKGZhaWxlZCBmcm9tIHRpbWVvdXQgb3Igc2VydmVyIGVycm9yKVxyXG5cdC8vIDMuIG51bGwgKGZhaWwpXHJcblx0Y29uc3Qgb3JpZ2luYWxBY2Nlc3NUb2tlbiA9IG9hdXRoVG9rZW4uYWNjZXNzX3Rva2VuXHJcblx0b2F1dGhUb2tlbiA9IHlpZWxkIGNhbGwoZ2V0VG9rZW5Gcm9tUmVmcmVzaFRva2VuLCBvYXV0aFRva2VuKVxyXG5cdGlmICghIW9hdXRoVG9rZW4gJiYgb2F1dGhUb2tlbi5hY2Nlc3NfdG9rZW4gIT09IG9yaWdpbmFsQWNjZXNzVG9rZW4pIHtcclxuXHRcdGxvZ2dlci5kZWJ1ZygnT0F1dGggdG9rZW4gcmVmcmVzaGVkJylcclxuXHRcdHlpZWxkIGNhbGwodG9rZW5QZXJzaXN0ZW5jZVNlcnZpY2UucGVyc2lzdFRva2VuLCBvYXV0aFRva2VuKVxyXG5cdFx0eWllbGQgcHV0PEF1dGhUb2tlblN1Y2Nlc3NBY3Rpb24+KHsgdHlwZTogQVVUSF9UT0tFTl9TVUNDRVNTX0FDVElPTl9UWVBFLlRPS0VOX1JFRlJFU0hfU1VDQ0VFREVELCBvYXV0aFRva2VuIH0pXHJcblx0fSBlbHNlIGlmIChvYXV0aFRva2VuID09PSBudWxsKSB7XHJcblx0XHRsb2dnZXIuZGVidWcoJ09BdXRoIHRva2VuIGZhaWxlZCB0byByZWZyZXNoJylcclxuXHRcdC8vIFRoaXMgc2hvdWxkIG5ldmVyIGhhcHBlbiBvdXRzaWRlIG9mIHRoZSB0b2tlbiBoYXZpbmcgYmVlbiByZXZva2VkIG9uIHRoZSBzZXJ2ZXIgc2lkZVxyXG5cdFx0eWllbGQgYWxsKHtcclxuXHRcdFx0cmVmcmVzaEZhaWxlZDogcHV0PEF1dGhBY3Rpb24+KHsgdHlwZTogQVVUSF9BQ1RJT05fVFlQRS5UT0tFTl9SRUZSRVNIX0ZBSUxFRCB9KSxcclxuXHRcdFx0bG9nT3V0OiBwdXQ8QXV0aEFjdGlvbj4oeyB0eXBlOiBBVVRIX0FDVElPTl9UWVBFLkxPR19PVVRfUkVRVUVTVEVEIH0pXHJcblx0XHR9KVxyXG5cdH1cclxuXHRyZWZyZXNoTG9jayA9IGZhbHNlXHJcbn1cclxuXHJcbmV4cG9ydCBmdW5jdGlvbiogbG9naW5GbG93KG1vZGVsRmV0Y2hSZXF1ZXN0QWN0aW9uOiBNb2RlbEZldGNoUmVxdWVzdEFjdGlvbik6IFNhZ2FJdGVyYXRvciB7XHJcblx0eWllbGQgcHV0KG1vZGVsRmV0Y2hSZXF1ZXN0QWN0aW9uKVxyXG5cdGNvbnN0IHtcclxuXHRcdGZldGNoUmVzdWx0QWN0aW9uLFxyXG5cdFx0ZmV0Y2hFcnJvckFjdGlvblxyXG5cdH06IHsgZmV0Y2hSZXN1bHRBY3Rpb24/OiBNb2RlbEZldGNoUmVzdWx0QWN0aW9uOyBmZXRjaEVycm9yQWN0aW9uPzogTW9kZWxGZXRjaEVycm9yQWN0aW9uIH0gPSB5aWVsZCByYWNlKHtcclxuXHRcdGZldGNoUmVzdWx0QWN0aW9uOiB0YWtlKHRha2VNYXRjaGVzTW9kZWxGZXRjaFJlY2VpdmVkKG1vZGVsRmV0Y2hSZXF1ZXN0QWN0aW9uLm1vZGVsTmFtZSkpLFxyXG5cdFx0ZmV0Y2hFcnJvckFjdGlvbjogdGFrZSh0YWtlTWF0Y2hlc01vZGVsRmV0Y2hGYWlsZWQobW9kZWxGZXRjaFJlcXVlc3RBY3Rpb24ubW9kZWxOYW1lKSlcclxuXHR9KVxyXG5cdGlmIChmZXRjaEVycm9yQWN0aW9uKSB7XHJcblx0XHRyZXR1cm4gbnVsbFxyXG5cdH1cclxuXHRjb25zdCBjb2RlOiBzdHJpbmcgfCB1bmRlZmluZWQgPSBmZXRjaFJlc3VsdEFjdGlvbj8uZGF0YT8uY29kZVxyXG5cdGlmICghY29kZSkge1xyXG5cdFx0cmV0dXJuIG51bGxcclxuXHR9XHJcblx0cmV0dXJuIHlpZWxkIGNhbGwoZ2V0VG9rZW5Gcm9tQ29kZSwgY29kZSlcclxufVxyXG5cclxuZXhwb3J0IGZ1bmN0aW9uKiBjcmVkZW50aWFsc0xvZ2luRmxvdyhcclxuXHRhY3Rpb246IEF1dGhDYXNWMUxvZ2luUmVxdWVzdEFjdGlvbiB8IEF1dGhMb2NhbExvZ2luUmVxdWVzdEFjdGlvbixcclxuXHRtb2RlbE5hbWU6IHN0cmluZ1xyXG4pOiBTYWdhSXRlcmF0b3Ige1xyXG5cdGNvbnN0IG1vZGVsRmV0Y2hSZXF1ZXN0QWN0aW9uOiBNb2RlbEZldGNoUmVxdWVzdEFjdGlvbiA9IHtcclxuXHRcdC8vIHNldCByZXF1aXJlZCBkZWZhdWx0c1xyXG5cdFx0dHlwZTogTU9ERUxfRkVUQ0hfUkVRVUVTVF9BQ1RJT05fVFlQRS5GRVRDSF9SRVFVRVNULFxyXG5cdFx0bW9kZWxOYW1lLFxyXG5cdFx0bm9TdG9yZTogdHJ1ZSxcclxuXHRcdC8vIGZvcmNlIG5vIHJldHJ5XHJcblx0XHRub1JldHJ5OiB0cnVlLFxyXG5cdFx0Ly8gcGFzcyB0aHJ1IGJvZHlcclxuXHRcdGJvZHk6IGFjdGlvbi5ib2R5XHJcblx0fVxyXG5cdHJldHVybiB5aWVsZCBjYWxsKGxvZ2luRmxvdywgbW9kZWxGZXRjaFJlcXVlc3RBY3Rpb24pXHJcbn1cclxuXHJcbmV4cG9ydCBmdW5jdGlvbiogY2FzVjFMb2dpbkZsb3coYWN0aW9uOiBBdXRoQ2FzVjFMb2dpblJlcXVlc3RBY3Rpb24pOiBTYWdhSXRlcmF0b3Ige1xyXG5cdHJldHVybiB5aWVsZCBjYWxsKGNyZWRlbnRpYWxzTG9naW5GbG93LCBhY3Rpb24sICdjb2RlRnJvbUNhc1YxJylcclxufVxyXG5cclxuZXhwb3J0IGZ1bmN0aW9uKiBsb2NhbExvZ2luRmxvdyhhY3Rpb246IEF1dGhMb2NhbExvZ2luUmVxdWVzdEFjdGlvbik6IFNhZ2FJdGVyYXRvciB7XHJcblx0cmV0dXJuIHlpZWxkIGNhbGwoY3JlZGVudGlhbHNMb2dpbkZsb3csIGFjdGlvbiwgJ2NvZGVGcm9tTG9jYWxDcmVkZW50aWFscycpXHJcbn1cclxuXHJcbmV4cG9ydCBmdW5jdGlvbiogY2FzVGlja2V0TG9naW5GbG93KHRpY2tldDogc3RyaW5nLCBzZXJ2aWNlOiBzdHJpbmcpOiBTYWdhSXRlcmF0b3Ige1xyXG5cdGNvbnN0IG1vZGVsRmV0Y2hSZXF1ZXN0QWN0aW9uOiBNb2RlbEZldGNoUmVxdWVzdEFjdGlvbiA9IHtcclxuXHRcdHR5cGU6IE1PREVMX0ZFVENIX1JFUVVFU1RfQUNUSU9OX1RZUEUuRkVUQ0hfUkVRVUVTVCxcclxuXHRcdG1vZGVsTmFtZTogJ2NvZGVGcm9tQ2FzVGlja2V0JyxcclxuXHRcdG5vU3RvcmU6IHRydWUsXHJcblx0XHRxdWVyeVBhcmFtczoge1xyXG5cdFx0XHR0aWNrZXQsXHJcblx0XHRcdHNlcnZpY2VcclxuXHRcdH1cclxuXHR9XHJcblx0cmV0dXJuIHlpZWxkIGNhbGwobG9naW5GbG93LCBtb2RlbEZldGNoUmVxdWVzdEFjdGlvbilcclxufVxyXG5cclxuZXhwb3J0IGZ1bmN0aW9uKiBoYW5kbGVBdXRoRmFpbHVyZShhY3Rpb246IE1vZGVsRmV0Y2hFcnJvckFjdGlvbik6IFNhZ2FJdGVyYXRvciB7XHJcblx0Ly8gVGhpcyBzaG91bGQgYmUgdW5saWtlbHkgc2luY2Ugd2Ugbm9ybWFsbHkgaGF2ZSBhIHJlZnJlc2ggdG9rZW4gbG9vcCBoYXBwZW5pbmdcclxuXHQvLyBidXQgaWYgdGhlIGFwcCBpcyBiYWNrZ3JvdW5kZWQsIHRoZSBsb29wIG1pZ2h0IG5vdCBiZSBjYXVnaHQgdXAgeWV0XHJcblx0aWYgKG9hdXRoVG9rZW4gJiYgYWN0aW9uLmVycm9yRGF0YSAmJiBhY3Rpb24uZXJyb3JEYXRhLnN0YXR1cyA9PT0gSFRUUF9TVEFUVVNfQ09ERS5VTkFVVEhPUklaRUQpIHtcclxuXHRcdGxvZ2dlci5kZWJ1ZygndG9rZW4gZXhwaXJlZCAtIHJlZnJlc2hpbmcnKVxyXG5cdFx0eWllbGQgY2FsbChwZXJmb3JtVG9rZW5SZWZyZXNoKVxyXG5cdH1cclxufVxyXG5cclxuZXhwb3J0IGNvbnN0IGdldE9BdXRoVG9rZW46IFRva2VuQWNjZXNzRnVuY3Rpb24gPSBmdW5jdGlvbiogKG1vZGVsTmFtZTogc3RyaW5nKSB7XHJcblx0Ly8gRG9uJ3QgdHJ5IHRvIHJlZnJlc2ggdGhlIHRva2VuIGlmIHdlJ3JlIGFscmVhZHkgaW4gYSByZXF1ZXN0IHRvIHJlZnJlc2ggdGhlIHRva2VuXHJcblx0aWYgKG1vZGVsTmFtZSA9PT0gJ2dldFRva2VuJykge1xyXG5cdFx0cmV0dXJuIG51bGxcclxuXHR9XHJcblx0aWYgKG9hdXRoVG9rZW4gJiYgb2F1dGhUb2tlblsnLmV4cGlyZXMnXSkge1xyXG5cdFx0Y29uc3QgdGhpcnR5U2Vjb25kc0Zyb21Ob3cgPSBuZXcgRGF0ZSgpXHJcblx0XHR0aGlydHlTZWNvbmRzRnJvbU5vdy5zZXRTZWNvbmRzKHRoaXJ0eVNlY29uZHNGcm9tTm93LmdldFNlY29uZHMoKSArIDMwKVxyXG5cdFx0aWYgKG5ldyBEYXRlKG9hdXRoVG9rZW5bJy5leHBpcmVzJ10pIDwgdGhpcnR5U2Vjb25kc0Zyb21Ob3cpIHtcclxuXHRcdFx0Ly8gc3RhcnQgYSB0b2tlbiByZWZyZXNoIGFuZCB3YWl0IGZvciB0aGUgc3VjY2VzcyBhY3Rpb24gaW4gY2FzZSBhbm90aGVyIHJlZnJlc2ggaXMgY3VycmVudGx5IGhhcHBlbmluZ1xyXG5cdFx0XHR5aWVsZCBjYWxsKHBlcmZvcm1Ub2tlblJlZnJlc2gpXHJcblx0XHRcdHJldHVybiBvYXV0aFRva2VuXHJcblx0XHR9XHJcblx0fVxyXG5cdHJldHVybiBvYXV0aFRva2VuXHJcbn1cclxuXHJcbmV4cG9ydCBkZWZhdWx0IGZ1bmN0aW9uKiBhdXRoU2FnYShcclxuXHRjbGllbnRDcmVkZW50aWFsc1BhcmFtOiBDbGllbnRDcmVkZW50aWFscyxcclxuXHR0b2tlblBlcnNpc3RlbmNlU2VydmljZVBhcmFtOiBUb2tlblBlcnNpc3RlbmNlU2VydmljZSA9IGRlZmF1bHRUb2tlblBlcnNpc3RlbmNlU2VydmljZSxcclxuXHR0aWNrZXRQcm92aWRlclNlcnZpY2U6IFRpY2tldFByb3ZpZGVyU2VydmljZSA9IGRlZmF1bHRUaWNrZXRQcm92aWRlclNlcnZpY2UsXHJcblx0Y29kZVByb3ZpZGVyU2VydmljZTogQ29kZVByb3ZpZGVyU2VydmljZSA9IGRlZmF1bHRDb2RlUHJvdmlkZXJTZXJ2aWNlXHJcbik6IFNhZ2FJdGVyYXRvciB7XHJcblx0bG9nZ2VyID0gZ2V0TG9nZ2VyKClcclxuXHJcblx0Y2xpZW50Q3JlZGVudGlhbHMgPSBjbGllbnRDcmVkZW50aWFsc1BhcmFtXHJcblx0dG9rZW5QZXJzaXN0ZW5jZVNlcnZpY2UgPSB0b2tlblBlcnNpc3RlbmNlU2VydmljZVBhcmFtXHJcblxyXG5cdC8vIFRyeSB0byBnZXQgcGVyc2lzdGVkIHRva2VuIChub3JtYWxseSBpbiBBc3luY1N0b3JhZ2Ugb3IgTG9jYWxTdG9yYWdlKVxyXG5cdG9hdXRoVG9rZW4gPSB5aWVsZCBjYWxsKHRva2VuUGVyc2lzdGVuY2VTZXJ2aWNlLmdldFBlcnNpc3RlZFRva2VuKVxyXG5cclxuXHQvLyBJZiBubyB0b2tlbiwgdHJ5IHRvIGdldCBDQVMgdGlja2V0IChub3JtYWxseSBpbiB0aGUgVVJMKSwgdXNlIGl0IHRvIGdldCBhIHRva2VuXHJcblx0aWYgKCFvYXV0aFRva2VuKSB7XHJcblx0XHRjb25zdCBjYXNUaWNrZXQgPSB0aWNrZXRQcm92aWRlclNlcnZpY2UuZ2V0VGlja2V0KClcclxuXHRcdHRpY2tldFByb3ZpZGVyU2VydmljZS5yZW1vdmVUaWNrZXQoKVxyXG5cdFx0Y29uc3Qgc2VydmljZSA9IHRpY2tldFByb3ZpZGVyU2VydmljZS5nZXRBcHBTZXJ2aWNlTmFtZSgpXHJcblx0XHRpZiAoY2FzVGlja2V0ICYmIHNlcnZpY2UpIHtcclxuXHRcdFx0b2F1dGhUb2tlbiA9IHlpZWxkIGNhbGwoY2FzVGlja2V0TG9naW5GbG93LCBjYXNUaWNrZXQsIHNlcnZpY2UpXHJcblx0XHR9XHJcblx0fVxyXG5cclxuXHQvLyBJZiBPQXV0aCBDb2RlIGV4aXN0cyAobm9ybWFsbHkgaW4gdGhlIFVSTCksIHVzZSBpdCB0byBnZXQgYSB0b2tlblxyXG5cdC8vIGUuZy4gTFRJLCBTaGliYm9sZXRoLCBGYWNlYm9vaywgR29vZ2xlXHJcblx0Y29uc3QgY29kZSA9IGNvZGVQcm92aWRlclNlcnZpY2UuZ2V0Q29kZSgpXHJcblx0aWYgKGNvZGUpIHtcclxuXHRcdC8vIExvZyBvdXQgdGhlIGN1cnJlbnQgdXNlciBpZiBhIG5ldyB0b2tlbiBpcyBhYm91dCB0byBiZSBnZW5lcmF0ZWQgZnJvbSB0aGUgT0F1dGggQ29kZVxyXG5cdFx0aWYgKG9hdXRoVG9rZW4pIHtcclxuXHRcdFx0eWllbGQgYWxsKHtcclxuXHRcdFx0XHRjbGVhclVzZXJEYXRhOiBwdXQ8TW9kZWxSZW1vdmVLZXlBY3Rpb24+KHsgdHlwZTogTU9ERUxfUkVNT1ZFX0tFWV9BQ1RJT05fVFlQRSwgbW9kZWxQYXRoOiAndXNlcicgfSksXHJcblx0XHRcdFx0Y2xlYXJQZXJzaXN0ZW50VG9rZW46IGNhbGwodG9rZW5QZXJzaXN0ZW5jZVNlcnZpY2UucGVyc2lzdFRva2VuLCBudWxsKVxyXG5cdFx0XHR9KVxyXG5cdFx0XHRvYXV0aFRva2VuID0gbnVsbFxyXG5cdFx0fVxyXG5cdFx0b2F1dGhUb2tlbiA9IHlpZWxkIGNhbGwoZ2V0VG9rZW5Gcm9tQ29kZSwgY29kZSlcclxuXHR9XHJcblx0Y29kZVByb3ZpZGVyU2VydmljZS5yZW1vdmVDb2RlKClcclxuXHJcblx0eWllbGQgcHV0PEF1dGhUb2tlbkFjdGlvbj4oeyB0eXBlOiBBVVRIX1RPS0VOX0FDVElPTl9UWVBFLkFVVEhfSU5JVElBTElaRUQsIG9hdXRoVG9rZW4gfSlcclxuXHJcblx0eWllbGQgdGFrZUV2ZXJ5KE1PREVMX0ZFVENIX0VSUk9SX0FDVElPTl9UWVBFLlRSWV9GRVRDSF9GQUlMRUQsIGhhbmRsZUF1dGhGYWlsdXJlKVxyXG5cclxuXHRkbyB7XHJcblx0XHRpZiAoIW9hdXRoVG9rZW4pIHtcclxuXHRcdFx0Y29uc3Qge1xyXG5cdFx0XHRcdGNhc1YxQWN0aW9uLFxyXG5cdFx0XHRcdGxvY2FsTG9naW5BY3Rpb25cclxuXHRcdFx0fToge1xyXG5cdFx0XHRcdGNhc1YxQWN0aW9uPzogQXV0aENhc1YxTG9naW5SZXF1ZXN0QWN0aW9uXHJcblx0XHRcdFx0bG9jYWxMb2dpbkFjdGlvbj86IEF1dGhMb2NhbExvZ2luUmVxdWVzdEFjdGlvblxyXG5cdFx0XHR9ID0geWllbGQgcmFjZSh7XHJcblx0XHRcdFx0Y2FzVjFBY3Rpb246IHRha2UoQVVUSF9DQVNfVjFfTE9HSU5fUkVRVUVTVF9BQ1RJT05fVFlQRS5DQVNfVjFfTE9HSU5fUkVRVUVTVCksXHJcblx0XHRcdFx0bG9jYWxMb2dpbkFjdGlvbjogdGFrZShBVVRIX0xPQ0FMX0xPR0lOX1JFUVVFU1RfQUNUSU9OX1RZUEUuTE9DQUxfTE9HSU5fUkVRVUVTVClcclxuXHRcdFx0fSlcclxuXHJcblx0XHRcdHlpZWxkIHB1dDxBdXRoQWN0aW9uPih7IHR5cGU6IEFVVEhfQUNUSU9OX1RZUEUuTE9HSU5fUkVRVUVTVEVEIH0pXHJcblx0XHRcdGlmIChjYXNWMUFjdGlvbikge1xyXG5cdFx0XHRcdG9hdXRoVG9rZW4gPSB5aWVsZCBjYWxsKGNhc1YxTG9naW5GbG93LCBjYXNWMUFjdGlvbilcclxuXHRcdFx0fSBlbHNlIGlmIChsb2NhbExvZ2luQWN0aW9uKSB7XHJcblx0XHRcdFx0b2F1dGhUb2tlbiA9IHlpZWxkIGNhbGwobG9jYWxMb2dpbkZsb3csIGxvY2FsTG9naW5BY3Rpb24pXHJcblx0XHRcdH1cclxuXHRcdH1cclxuXHJcblx0XHRpZiAob2F1dGhUb2tlbikge1xyXG5cdFx0XHR5aWVsZCBjYWxsKHRva2VuUGVyc2lzdGVuY2VTZXJ2aWNlLnBlcnNpc3RUb2tlbiwgb2F1dGhUb2tlbilcclxuXHRcdFx0eWllbGQgYWxsKHtcclxuXHRcdFx0XHRsb2dpblN1Y2Nlc3M6IHB1dDxBdXRoVG9rZW5TdWNjZXNzQWN0aW9uPih7XHJcblx0XHRcdFx0XHR0eXBlOiBBVVRIX1RPS0VOX1NVQ0NFU1NfQUNUSU9OX1RZUEUuR0VUX1RPS0VOX1NVQ0NFRURFRCxcclxuXHRcdFx0XHRcdG9hdXRoVG9rZW5cclxuXHRcdFx0XHR9KSxcclxuXHRcdFx0XHRnZXRVc2VySW5mbzogcHV0PE1vZGVsRmV0Y2hSZXF1ZXN0QWN0aW9uPih7XHJcblx0XHRcdFx0XHR0eXBlOiBNT0RFTF9GRVRDSF9SRVFVRVNUX0FDVElPTl9UWVBFLkZFVENIX1JFUVVFU1QsXHJcblx0XHRcdFx0XHRtb2RlbE5hbWU6ICd1c2VyLnVzZXJJbmZvJ1xyXG5cdFx0XHRcdH0pLFxyXG5cdFx0XHRcdGxvZ091dDogdGFrZShBVVRIX0FDVElPTl9UWVBFLkxPR19PVVRfUkVRVUVTVEVEKVxyXG5cdFx0XHR9KVxyXG5cdFx0fSBlbHNlIHtcclxuXHRcdFx0eWllbGQgcHV0PEF1dGhBY3Rpb24+KHsgdHlwZTogQVVUSF9BQ1RJT05fVFlQRS5MT0dJTl9GQUlMRUQgfSlcclxuXHRcdH1cclxuXHJcblx0XHR5aWVsZCBhbGwoe1xyXG5cdFx0XHRjbGVhclVzZXJEYXRhOiBwdXQ8TW9kZWxSZW1vdmVLZXlBY3Rpb24+KHsgdHlwZTogTU9ERUxfUkVNT1ZFX0tFWV9BQ1RJT05fVFlQRSwgbW9kZWxQYXRoOiAndXNlcicgfSksXHJcblx0XHRcdGNsZWFyUGVyc2lzdGVudFRva2VuOiBjYWxsKHRva2VuUGVyc2lzdGVuY2VTZXJ2aWNlLnBlcnNpc3RUb2tlbiwgbnVsbClcclxuXHRcdH0pXHJcblx0XHRvYXV0aFRva2VuID0gbnVsbFxyXG5cdH0gd2hpbGUgKHRydWUpXHJcbn1cclxuIl0sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFFQSxJQUFBQSxRQUFBLEdBQUFDLE9BQUE7QUFDQSxJQUFBQyxvQkFBQSxHQUFBRCxPQUFBO0FBQ0EsSUFBQUUsc0JBQUEsR0FBQUYsT0FBQTtBQUNBLElBQUFHLHdCQUFBLEdBQUFILE9BQUE7QUFDQSxJQUFBSSxNQUFBLEdBQUFKLE9BQUE7QUFVQSxJQUFBSyxPQUFBLEdBQUFMLE9BQUE7QUFDQSxJQUFBTSxRQUFBLEdBQUFOLE9BQUE7QUFzQkE7O0FBRU8sTUFBTU8seUJBQXlCLEdBQUdBLENBQUNDLE1BQWlCLEVBQUVDLFNBQWlCLEtBQzdFLElBQUFDLDBDQUFpQyxFQUFDRixNQUFNLENBQUMsSUFBSUEsTUFBTSxDQUFDRyxTQUFTLEtBQUtGLFNBQVM7QUFBQUcsT0FBQSxDQUFBTCx5QkFBQSxHQUFBQSx5QkFBQTtBQUVyRSxNQUFNTSw2QkFBNkIsR0FBSUosU0FBaUIsSUFBTUssY0FBeUIsSUFDN0ZQLHlCQUF5QixDQUFDTyxjQUFjLEVBQUVMLFNBQVMsQ0FBQztBQUFBRyxPQUFBLENBQUFDLDZCQUFBLEdBQUFBLDZCQUFBO0FBRTlDLE1BQU1FLHVCQUF1QixHQUFHQSxDQUFDUCxNQUFpQixFQUFFQyxTQUFpQixLQUMzRSxJQUFB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