studiokit-scaffolding-js 7.0.12 → 7.1.0

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 (390) hide show
  1. package/LICENSE +21 -21
  2. package/README.md +137 -137
  3. package/lib/components/ActionList.d.ts +2 -2
  4. package/lib/components/ActionList.js +166 -87
  5. package/lib/components/AlertDialog.d.ts +3 -3
  6. package/lib/components/AlertDialog.js +133 -18
  7. package/lib/components/AlertWithIcon.js +93 -52
  8. package/lib/components/ConnectedModal.d.ts +1 -2
  9. package/lib/components/ConnectedModal.js +43 -37
  10. package/lib/components/Dropdowns/GroupsDropdown.d.ts +4 -4
  11. package/lib/components/Dropdowns/GroupsDropdown.js +69 -82
  12. package/lib/components/Dropdowns/ManagedNavDropdown.d.ts +2 -2
  13. package/lib/components/Dropdowns/ManagedNavDropdown.js +99 -111
  14. package/lib/components/Dropdowns/UserDropdown.js +109 -70
  15. package/lib/components/Dropdowns/index.js +27 -11
  16. package/lib/components/EntityOwnerList.d.ts +3 -3
  17. package/lib/components/EntityOwnerList.js +52 -60
  18. package/lib/components/Error.js +107 -27
  19. package/lib/components/ErrorBoundary.js +133 -94
  20. package/lib/components/ErrorMessage.d.ts +2 -2
  21. package/lib/components/ErrorMessage.js +44 -14
  22. package/lib/components/Forms/DateField.d.ts +2 -2
  23. package/lib/components/Forms/DateField.js +63 -95
  24. package/lib/components/Forms/TimeField.d.ts +2 -2
  25. package/lib/components/Forms/TimeField.js +84 -108
  26. package/lib/components/Forms/index.js +27 -13
  27. package/lib/components/Groups/CreateEditCopySaveButtons.d.ts +2 -2
  28. package/lib/components/Groups/CreateEditCopySaveButtons.js +102 -14
  29. package/lib/components/Groups/ExternalGroups/Attach.d.ts +6 -7
  30. package/lib/components/Groups/ExternalGroups/Attach.js +213 -188
  31. package/lib/components/Groups/ExternalGroups/Table.js +183 -76
  32. package/lib/components/Groups/GroupCreateOrEditCommonProps.js +5 -1
  33. package/lib/components/Groups/RosterSyncInfo.js +140 -49
  34. package/lib/components/HOC/AccessibleAppComponent.d.ts +2 -2
  35. package/lib/components/HOC/AccessibleAppComponent.js +101 -124
  36. package/lib/components/HOC/ActivityRequiredComponent.d.ts +14 -14
  37. package/lib/components/HOC/ActivityRequiredComponent.js +93 -107
  38. package/lib/components/HOC/AsyncComponent.d.ts +9 -9
  39. package/lib/components/HOC/AsyncComponent.js +53 -129
  40. package/lib/components/HOC/AuthenticatedComponent.d.ts +4 -4
  41. package/lib/components/HOC/AuthenticatedComponent.js +59 -96
  42. package/lib/components/HOC/CollectionComponent.d.ts +12 -12
  43. package/lib/components/HOC/CollectionComponent.js +171 -171
  44. package/lib/components/HOC/CollectionFirstItemComponent.d.ts +13 -13
  45. package/lib/components/HOC/CollectionFirstItemComponent.js +52 -94
  46. package/lib/components/HOC/CollectionItemComponent.d.ts +9 -9
  47. package/lib/components/HOC/CollectionItemComponent.js +169 -168
  48. package/lib/components/HOC/ConnectedModalComponent.d.ts +9 -9
  49. package/lib/components/HOC/ConnectedModalComponent.js +110 -140
  50. package/lib/components/HOC/DataDependentComponent.d.ts +3 -55
  51. package/lib/components/HOC/DataDependentComponent.js +30 -79
  52. package/lib/components/HOC/EntityComponent.d.ts +13 -13
  53. package/lib/components/HOC/EntityComponent.js +74 -112
  54. package/lib/components/HOC/FullscreenModalComponent.d.ts +9 -9
  55. package/lib/components/HOC/FullscreenModalComponent.js +165 -185
  56. package/lib/components/HOC/GroupActivityRequiredComponent.d.ts +2 -2
  57. package/lib/components/HOC/GroupActivityRequiredComponent.js +48 -47
  58. package/lib/components/HOC/GuidComponent.d.ts +1 -1
  59. package/lib/components/HOC/GuidComponent.js +29 -68
  60. package/lib/components/HOC/ModelContextDependencyVerifyComponent.d.ts +28 -29
  61. package/lib/components/HOC/ModelContextDependencyVerifyComponent.js +42 -65
  62. package/lib/components/HOC/ModelErrorRedirectComponent.d.ts +10 -10
  63. package/lib/components/HOC/ModelErrorRedirectComponent.js +51 -94
  64. package/lib/components/HOC/SearchPersistorComponent.d.ts +4 -4
  65. package/lib/components/HOC/SearchPersistorComponent.js +243 -216
  66. package/lib/components/HOC/UnauthenticatedComponent.d.ts +12 -12
  67. package/lib/components/HOC/UnauthenticatedComponent.js +37 -82
  68. package/lib/components/HOC/UserComponent.d.ts +1 -1
  69. package/lib/components/HOC/UserComponent.js +12 -11
  70. package/lib/components/Icons/IconAlphaList.js +34 -20
  71. package/lib/components/Icons/IconExternalUser.js +34 -20
  72. package/lib/components/Icons/IconImpersonation.js +34 -20
  73. package/lib/components/Icons/IconStopImpersonating.js +34 -20
  74. package/lib/components/Icons/IconTable.js +35 -21
  75. package/lib/components/Icons/IconTableDeleteCol.js +34 -20
  76. package/lib/components/Icons/IconTableDeleteRow.js +34 -20
  77. package/lib/components/Icons/IconTableInsertCol.js +34 -20
  78. package/lib/components/Icons/IconTableInsertRow.js +34 -20
  79. package/lib/components/Impersonation/Button.d.ts +2 -2
  80. package/lib/components/Impersonation/Button.js +76 -56
  81. package/lib/components/Impersonation/Link.d.ts +2 -2
  82. package/lib/components/Impersonation/Link.js +76 -53
  83. package/lib/components/Impersonation/UserDetail.css +22 -22
  84. package/lib/components/Impersonation/UserDetail.d.ts +2 -2
  85. package/lib/components/Impersonation/UserDetail.js +66 -17
  86. package/lib/components/Loading.js +27 -21
  87. package/lib/components/LockDownBrowser/Check.js +194 -107
  88. package/lib/components/LockDownBrowser/ExitButton.js +27 -14
  89. package/lib/components/LockDownBrowser/Launch.js +70 -85
  90. package/lib/components/Lti/Confirm.js +152 -33
  91. package/lib/components/Lti/CreateNonLtiGroupAlertDialog.d.ts +4 -2
  92. package/lib/components/Lti/CreateNonLtiGroupAlertDialog.js +170 -78
  93. package/lib/components/Lti/Launch.js +105 -49
  94. package/lib/components/Lti/LaunchGroup.js +85 -39
  95. package/lib/components/ManageTable.d.ts +1 -2
  96. package/lib/components/ManageTable.js +309 -104
  97. package/lib/components/ManageTableNoDataComponent.js +42 -22
  98. package/lib/components/NewVersionAlert.d.ts +2 -2
  99. package/lib/components/NewVersionAlert.js +82 -91
  100. package/lib/components/NotFound.js +87 -28
  101. package/lib/components/Notifications.d.ts +4 -4
  102. package/lib/components/Notifications.js +187 -169
  103. package/lib/components/PaginationNextButton.js +33 -21
  104. package/lib/components/PaginationPreviousButton.js +33 -21
  105. package/lib/components/Quill/CustomToolbar.d.ts +2 -2
  106. package/lib/components/Quill/CustomToolbar.js +432 -257
  107. package/lib/components/Quill/Formats/Image.d.ts +10 -25
  108. package/lib/components/Quill/Formats/Image.js +80 -86
  109. package/lib/components/Quill/Formats/List.d.ts +6 -3
  110. package/lib/components/Quill/Formats/List.js +52 -74
  111. package/lib/components/Quill/Formats/Video.d.ts +2 -1
  112. package/lib/components/Quill/Formats/Video.js +31 -45
  113. package/lib/components/Quill/ImageDropModule.d.ts +1 -1
  114. package/lib/components/Quill/ImageDropModule.js +149 -122
  115. package/lib/components/Quill/ImageWarning.d.ts +2 -2
  116. package/lib/components/Quill/ImageWarning.js +45 -32
  117. package/lib/components/Quill/ImageWithAltTextModal.d.ts +0 -1
  118. package/lib/components/Quill/ImageWithAltTextModal.js +425 -141
  119. package/lib/components/Quill/Specs/CustomImageSpec.d.ts +1 -1
  120. package/lib/components/Quill/Specs/CustomImageSpec.js +43 -51
  121. package/lib/components/Quill/Specs/CustomVideoSpec.d.ts +1 -1
  122. package/lib/components/Quill/Specs/CustomVideoSpec.js +35 -44
  123. package/lib/components/Quill/TableModule/Blots/BaseTableBlot.d.ts +2 -1
  124. package/lib/components/Quill/TableModule/Blots/BaseTableBlot.js +105 -121
  125. package/lib/components/Quill/TableModule/Blots/TableBlot.js +53 -66
  126. package/lib/components/Quill/TableModule/Blots/TableBodyBlot.js +54 -67
  127. package/lib/components/Quill/TableModule/Blots/TableCellBlot.d.ts +1 -1
  128. package/lib/components/Quill/TableModule/Blots/TableCellBlot.js +225 -241
  129. package/lib/components/Quill/TableModule/Blots/TableContainer.js +82 -99
  130. package/lib/components/Quill/TableModule/Blots/TableRowBlot.js +76 -87
  131. package/lib/components/Quill/TableModule/constants.js +45 -41
  132. package/lib/components/Quill/TableModule/index.css +171 -171
  133. package/lib/components/Quill/TableModule/index.d.ts +14 -5
  134. package/lib/components/Quill/TableModule/index.js +373 -313
  135. package/lib/components/Quill/TableModule/utils.js +45 -45
  136. package/lib/components/Quill/accessibilityFix.d.ts +0 -1
  137. package/lib/components/Quill/accessibilityFix.js +235 -251
  138. package/lib/components/Quill/index.js +36 -32
  139. package/lib/components/RefreshIndicator/Bordered.js +48 -32
  140. package/lib/components/RefreshIndicator/Inline.js +48 -33
  141. package/lib/components/RefreshIndicator/index.d.ts +2 -2
  142. package/lib/components/RefreshIndicator/index.js +263 -113
  143. package/lib/components/SearchControls.js +216 -31
  144. package/lib/components/SentryRoute.js +10 -23
  145. package/lib/components/Tables/RoleFilter.d.ts +3 -4
  146. package/lib/components/Tables/RoleFilter.js +71 -46
  147. package/lib/components/Tables/TextFilter.d.ts +2 -3
  148. package/lib/components/Tables/TextFilter.js +62 -20
  149. package/lib/components/UserRoles/Add.d.ts +3 -3
  150. package/lib/components/UserRoles/Add.js +199 -161
  151. package/lib/components/UserRoles/Context.js +11 -9
  152. package/lib/components/UserRoles/RoleCell.js +180 -100
  153. package/lib/components/UserRoles/Select.js +157 -51
  154. package/lib/components/UserRoles/Table.js +221 -102
  155. package/lib/components/UserRoles/index.d.ts +6 -6
  156. package/lib/components/UserRoles/index.js +537 -466
  157. package/lib/config/eslint/index.d.ts +2 -0
  158. package/lib/config/eslint/index.js +36 -0
  159. package/lib/config/eslint/lib/order.d.ts +3 -0
  160. package/lib/config/eslint/lib/order.js +30 -0
  161. package/lib/config/eslint/lib/prettier.d.ts +3 -0
  162. package/lib/config/eslint/lib/prettier.js +23 -0
  163. package/lib/config/eslint/lib/typescript.d.ts +2 -0
  164. package/lib/config/eslint/lib/typescript.js +98 -0
  165. package/lib/config/eslint/react.d.ts +2 -0
  166. package/lib/config/eslint/react.js +30 -0
  167. package/lib/constants/baseActivity.js +30 -26
  168. package/lib/constants/baseRole.js +14 -10
  169. package/lib/constants/configuration.d.ts +1 -1
  170. package/lib/constants/configuration.js +43 -39
  171. package/lib/constants/externalProviderType.js +10 -6
  172. package/lib/constants/fetchErrorData.js +15 -11
  173. package/lib/constants/index.js +137 -23
  174. package/lib/constants/lockDownBrowser.js +28 -24
  175. package/lib/constants/mockData.d.ts +4 -6
  176. package/lib/constants/mockData.js +385 -326
  177. package/lib/constants/modelStatus.js +15 -11
  178. package/lib/constants/notificationType.js +12 -8
  179. package/lib/constants/operatingSystem.js +12 -8
  180. package/lib/constants/shard.js +11 -7
  181. package/lib/constants/table.js +21 -21
  182. package/lib/constants/tier.js +12 -8
  183. package/lib/constants/userRole.d.ts +1 -2
  184. package/lib/constants/userRole.js +15 -17
  185. package/lib/css/base/_base.css +98 -98
  186. package/lib/css/base/_typography.css +130 -130
  187. package/lib/css/components/_alert.css +86 -86
  188. package/lib/css/components/_bootstrap-grid.css +28 -28
  189. package/lib/css/components/_buttons.css +397 -397
  190. package/lib/css/components/_forms.css +101 -101
  191. package/lib/css/components/_menu.css +56 -56
  192. package/lib/css/components/_modals.css +46 -46
  193. package/lib/css/components/_quill.css +315 -315
  194. package/lib/css/components/_tables.css +497 -497
  195. package/lib/css/components/_tags.css +12 -12
  196. package/lib/css/index-with-variables.css +15 -15
  197. package/lib/css/index.css +14 -14
  198. package/lib/css/utils/_border.css +463 -463
  199. package/lib/css/utils/_color.css +317 -317
  200. package/lib/css/utils/_display.css +305 -312
  201. package/lib/css/utils/_general.css +48 -48
  202. package/lib/css/utils/_icon.css +16 -16
  203. package/lib/css/utils/_text.css +25 -24
  204. package/lib/css/utils/_width.css +60 -60
  205. package/lib/css/variables.css +84 -84
  206. package/lib/endpointMappings.js +197 -193
  207. package/lib/hooks/useCollection.js +83 -65
  208. package/lib/hooks/useCollectionConfiguration.d.ts +0 -1
  209. package/lib/hooks/useCollectionConfiguration.js +185 -88
  210. package/lib/hooks/useCollectionItem.js +155 -58
  211. package/lib/hooks/useEventCalback.d.ts +14 -0
  212. package/lib/hooks/useEventCalback.js +51 -0
  213. package/lib/hooks/useGuid.js +21 -10
  214. package/lib/hooks/usePrevious.d.ts +24 -1
  215. package/lib/hooks/usePrevious.js +84 -14
  216. package/lib/index.js +156 -36
  217. package/lib/redux/actionCreator.d.ts +6 -6
  218. package/lib/redux/actionCreator.js +51 -40
  219. package/lib/redux/actions/AuthAction.js +44 -31
  220. package/lib/redux/actions/ModalAction.d.ts +1 -1
  221. package/lib/redux/actions/ModalAction.js +10 -6
  222. package/lib/redux/actions/ModelAction.d.ts +1 -1
  223. package/lib/redux/actions/ModelAction.js +77 -49
  224. package/lib/redux/actions/NotificationAction.js +10 -6
  225. package/lib/redux/actions/SearchAction.js +9 -5
  226. package/lib/redux/actions/index.js +60 -16
  227. package/lib/redux/configureReducers.d.ts +2 -5
  228. package/lib/redux/configureReducers.js +67 -78
  229. package/lib/redux/configureStore.d.ts +2 -2
  230. package/lib/redux/configureStore.js +83 -115
  231. package/lib/redux/helpers.js +7 -2
  232. package/lib/redux/reducers/authReducer.js +52 -46
  233. package/lib/redux/reducers/index.js +41 -15
  234. package/lib/redux/reducers/modalsReducer.js +49 -52
  235. package/lib/redux/reducers/modelsReducer.d.ts +2 -2
  236. package/lib/redux/reducers/modelsReducer.js +183 -182
  237. package/lib/redux/reducers/notificationsReducer.js +26 -21
  238. package/lib/redux/reducers/searchReducer.js +26 -20
  239. package/lib/redux/sagas/appInsightsSaga.js +24 -62
  240. package/lib/redux/sagas/authSaga.js +287 -414
  241. package/lib/redux/sagas/caliperSaga.js +169 -263
  242. package/lib/redux/sagas/clockOffsetSaga.js +36 -77
  243. package/lib/redux/sagas/configurationSaga.js +13 -46
  244. package/lib/redux/sagas/downtimeApiErrorSaga.d.ts +0 -1
  245. package/lib/redux/sagas/downtimeApiErrorSaga.js +23 -64
  246. package/lib/redux/sagas/errorSaga.d.ts +0 -1
  247. package/lib/redux/sagas/errorSaga.js +29 -63
  248. package/lib/redux/sagas/googleAnalyticsSaga.js +29 -69
  249. package/lib/redux/sagas/identityProviderSaga.js +24 -69
  250. package/lib/redux/sagas/initialDataLoadSaga.js +38 -29
  251. package/lib/redux/sagas/lockDownBrowserErrorSaga.d.ts +0 -1
  252. package/lib/redux/sagas/lockDownBrowserErrorSaga.js +32 -81
  253. package/lib/redux/sagas/modelFetchSaga.js +379 -459
  254. package/lib/redux/sagas/noStoreSaga.d.ts +1 -1
  255. package/lib/redux/sagas/noStoreSaga.js +68 -101
  256. package/lib/redux/sagas/postLoginDataSaga.js +47 -79
  257. package/lib/redux/sagas/postLoginRedirectSaga.js +30 -71
  258. package/lib/redux/sagas/rootSaga.js +85 -137
  259. package/lib/redux/sagas/sentrySaga.js +29 -85
  260. package/lib/redux/sagas/userIdSaga.js +18 -54
  261. package/lib/services/codeProviderService.js +25 -29
  262. package/lib/services/dateService.js +14 -12
  263. package/lib/services/documentService.js +18 -13
  264. package/lib/services/fetchService.js +134 -168
  265. package/lib/services/persistenceService.d.ts +4 -4
  266. package/lib/services/persistenceService.js +45 -43
  267. package/lib/services/ticketProviderService.js +29 -33
  268. package/lib/services/tokenPersistenceService.js +13 -9
  269. package/lib/services/windowService.js +21 -17
  270. package/lib/startup.d.ts +1 -1
  271. package/lib/startup.js +132 -135
  272. package/lib/types/AppConfiguration.js +5 -1
  273. package/lib/types/Artifact.js +11 -7
  274. package/lib/types/BaseReduxState.js +5 -1
  275. package/lib/types/Client.js +5 -1
  276. package/lib/types/Collection.d.ts +3 -3
  277. package/lib/types/Collection.js +5 -1
  278. package/lib/types/Configuration.js +5 -1
  279. package/lib/types/DeepLinkingResponseRequest.js +5 -1
  280. package/lib/types/DeletableModel.js +5 -1
  281. package/lib/types/Event.js +5 -1
  282. package/lib/types/ExternalGroup.js +5 -1
  283. package/lib/types/ExternalProvider.js +5 -1
  284. package/lib/types/ExternalTerm.js +5 -1
  285. package/lib/types/Group.js +5 -1
  286. package/lib/types/IdentityProvider.js +5 -1
  287. package/lib/types/LtiLaunch.js +5 -1
  288. package/lib/types/NameOnlyEntity.js +5 -1
  289. package/lib/types/Notification.d.ts +2 -2
  290. package/lib/types/Notification.js +5 -1
  291. package/lib/types/OptionalRecord.d.ts +1 -1
  292. package/lib/types/OptionalRecord.js +5 -1
  293. package/lib/types/OwnerSchedule.d.ts +3 -1
  294. package/lib/types/OwnerSchedule.js +5 -1
  295. package/lib/types/PropertyOfType.d.ts +3 -3
  296. package/lib/types/PropertyOfType.js +5 -1
  297. package/lib/types/Quill.js +5 -1
  298. package/lib/types/RoleDescription.d.ts +0 -1
  299. package/lib/types/RoleDescription.js +5 -1
  300. package/lib/types/Search.js +5 -1
  301. package/lib/types/SimpleLocation.js +5 -1
  302. package/lib/types/UniTime.js +5 -1
  303. package/lib/types/User.js +5 -1
  304. package/lib/types/UserRole.js +5 -1
  305. package/lib/types/auth/AuthState.js +5 -1
  306. package/lib/types/auth/CasV1LoginRequestBody.js +5 -1
  307. package/lib/types/auth/ClientCredentials.js +5 -1
  308. package/lib/types/auth/CodeProviderService.js +5 -1
  309. package/lib/types/auth/LocalLoginRequestBody.js +5 -1
  310. package/lib/types/auth/TicketProviderService.js +5 -1
  311. package/lib/types/auth/TokenPersistenceService.js +5 -1
  312. package/lib/types/auth/index.js +82 -18
  313. package/lib/types/externals.d.js +2 -0
  314. package/lib/types/index.js +313 -39
  315. package/lib/types/net/EndpointConfig.js +5 -1
  316. package/lib/types/net/EndpointMapping.js +5 -1
  317. package/lib/types/net/EndpointMappings.js +5 -1
  318. package/lib/types/net/ErrorHandler.d.ts +1 -1
  319. package/lib/types/net/ErrorHandler.js +5 -1
  320. package/lib/types/net/FetchConfig.d.ts +1 -1
  321. package/lib/types/net/FetchConfig.js +5 -1
  322. package/lib/types/net/FetchErrorData.js +10 -6
  323. package/lib/types/net/FetchResult.d.ts +2 -2
  324. package/lib/types/net/FetchResult.js +5 -1
  325. package/lib/types/net/HTTPMethod.d.ts +1 -1
  326. package/lib/types/net/HTTPMethod.js +5 -1
  327. package/lib/types/net/HTTPStatusCode.js +16 -12
  328. package/lib/types/net/Metadata.js +5 -1
  329. package/lib/types/net/Model.js +5 -1
  330. package/lib/types/net/ModelCollection.js +5 -1
  331. package/lib/types/net/ModelsState.js +5 -1
  332. package/lib/types/net/OAuthToken.js +5 -1
  333. package/lib/types/net/OAuthTokenOrNull.d.ts +1 -1
  334. package/lib/types/net/OAuthTokenOrNull.js +5 -1
  335. package/lib/types/net/TokenAccessFunction.d.ts +1 -1
  336. package/lib/types/net/TokenAccessFunction.js +5 -1
  337. package/lib/types/net/index.js +181 -27
  338. package/lib/utils/baseActivity.d.ts +2 -2
  339. package/lib/utils/baseActivity.js +138 -136
  340. package/lib/utils/baseRole.js +37 -33
  341. package/lib/utils/collection.js +432 -310
  342. package/lib/utils/cookies.js +22 -37
  343. package/lib/utils/date.js +320 -320
  344. package/lib/utils/dom.js +186 -174
  345. package/lib/utils/domainIdentifier.js +9 -9
  346. package/lib/utils/entityUserRole.js +6 -2
  347. package/lib/utils/error.d.ts +1 -1
  348. package/lib/utils/error.js +18 -16
  349. package/lib/utils/events.js +40 -33
  350. package/lib/utils/externalGroup.js +25 -21
  351. package/lib/utils/externalProviders.js +8 -4
  352. package/lib/utils/externalTerms.js +9 -6
  353. package/lib/utils/fetch.d.ts +1 -1
  354. package/lib/utils/fetch.js +188 -185
  355. package/lib/utils/group.js +17 -9
  356. package/lib/utils/groupDates.d.ts +1 -1
  357. package/lib/utils/groupDates.js +43 -39
  358. package/lib/utils/groupRoles.js +28 -28
  359. package/lib/utils/lockDownBrowser.js +16 -12
  360. package/lib/utils/logger.js +30 -25
  361. package/lib/utils/lti.js +10 -5
  362. package/lib/utils/model.d.ts +2 -2
  363. package/lib/utils/model.js +39 -48
  364. package/lib/utils/number.js +23 -26
  365. package/lib/utils/promise.d.ts +6 -1
  366. package/lib/utils/promise.js +34 -15
  367. package/lib/utils/quill.js +66 -77
  368. package/lib/utils/route.d.ts +2 -2
  369. package/lib/utils/route.js +61 -60
  370. package/lib/utils/search.d.ts +1 -1
  371. package/lib/utils/search.js +78 -85
  372. package/lib/utils/shard.js +38 -41
  373. package/lib/utils/sort.d.ts +3 -3
  374. package/lib/utils/sort.js +62 -58
  375. package/lib/utils/string.js +14 -10
  376. package/lib/utils/table.d.ts +3 -3
  377. package/lib/utils/table.js +39 -36
  378. package/lib/utils/timezone.js +12 -26
  379. package/lib/utils/url.d.ts +2 -1
  380. package/lib/utils/url.js +146 -200
  381. package/lib/utils/user.js +59 -61
  382. package/lib/utils/userAgent.js +11 -11
  383. package/lib/utils/userRole.d.ts +3 -3
  384. package/lib/utils/userRole.js +58 -56
  385. package/package.json +224 -207
  386. package/lib/config/eslint/index.cjs +0 -21
  387. package/lib/config/eslint/lib/order.cjs +0 -22
  388. package/lib/config/eslint/lib/prettier.cjs +0 -18
  389. package/lib/config/eslint/lib/typescript.cjs +0 -86
  390. package/lib/config/eslint/react.cjs +0 -9
@@ -1,436 +1,309 @@
1
1
  "use strict";
2
- var __generator = (this && this.__generator) || function (thisArg, body) {
3
- var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g;
4
- return g = { next: verb(0), "throw": verb(1), "return": verb(2) }, typeof Symbol === "function" && (g[Symbol.iterator] = function() { return this; }), g;
5
- function verb(n) { return function (v) { return step([n, v]); }; }
6
- function step(op) {
7
- if (f) throw new TypeError("Generator is already executing.");
8
- while (_) try {
9
- if (f = 1, y && (t = op[0] & 2 ? y["return"] : op[0] ? y["throw"] || ((t = y["return"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;
10
- if (y = 0, t) op = [op[0] & 2, t.value];
11
- switch (op[0]) {
12
- case 0: case 1: t = op; break;
13
- case 4: _.label++; return { value: op[1], done: false };
14
- case 5: _.label++; y = op[1]; op = [0]; continue;
15
- case 7: op = _.ops.pop(); _.trys.pop(); continue;
16
- default:
17
- if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }
18
- if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }
19
- if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }
20
- if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }
21
- if (t[2]) _.ops.pop();
22
- _.trys.pop(); continue;
23
- }
24
- op = body.call(thisArg, _);
25
- } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }
26
- if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };
27
- }
28
- };
29
- Object.defineProperty(exports, "__esModule", { value: true });
30
- exports.getOAuthToken = exports.handleAuthFailure = exports.casTicketLoginFlow = exports.localLoginFlow = exports.casV1LoginFlow = exports.credentialsLoginFlow = exports.loginFlow = exports.performTokenRefresh = exports.getTokenFromRefreshToken = exports.getTokenFromCode = exports.takeMatchesModelFetchFailed = exports.matchesModelFetchFailed = exports.takeMatchesModelFetchReceived = exports.matchesModelFetchReceived = void 0;
31
- var effects_1 = require("redux-saga/effects");
32
- var codeProviderService_1 = require("../../services/codeProviderService");
33
- var ticketProviderService_1 = require("../../services/ticketProviderService");
34
- var tokenPersistenceService_1 = require("../../services/tokenPersistenceService");
35
- var types_1 = require("../../types");
36
- var logger_1 = require("../../utils/logger");
37
- var actions_1 = require("../actions");
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;
11
+ exports.getTokenFromCode = getTokenFromCode;
12
+ exports.getTokenFromRefreshToken = getTokenFromRefreshToken;
13
+ exports.handleAuthFailure = handleAuthFailure;
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");
38
26
  //#region Helpers
39
- var matchesModelFetchReceived = function (action, modelName) {
40
- return actions_1.isTransientModelFetchResultAction(action) && action.modelPath === modelName;
41
- };
27
+
28
+ const matchesModelFetchReceived = (action, modelName) => (0, _actions.isTransientModelFetchResultAction)(action) && action.modelPath === modelName;
42
29
  exports.matchesModelFetchReceived = matchesModelFetchReceived;
43
- var takeMatchesModelFetchReceived = function (modelName) { return function (incomingAction) {
44
- return exports.matchesModelFetchReceived(incomingAction, modelName);
45
- }; };
30
+ const takeMatchesModelFetchReceived = modelName => incomingAction => matchesModelFetchReceived(incomingAction, modelName);
46
31
  exports.takeMatchesModelFetchReceived = takeMatchesModelFetchReceived;
47
- var matchesModelFetchFailed = function (action, modelName) {
48
- return actions_1.isTransientModelFetchErrorAction(action) && action.modelPath === modelName;
49
- };
32
+ const matchesModelFetchFailed = (action, modelName) => (0, _actions.isTransientModelFetchErrorAction)(action) && action.modelPath === modelName;
50
33
  exports.matchesModelFetchFailed = matchesModelFetchFailed;
51
- var takeMatchesModelFetchFailed = function (modelName) { return function (incomingAction) {
52
- return exports.matchesModelFetchFailed(incomingAction, modelName);
53
- }; };
54
- exports.takeMatchesModelFetchFailed = takeMatchesModelFetchFailed;
34
+ const takeMatchesModelFetchFailed = modelName => incomingAction => matchesModelFetchFailed(incomingAction, modelName);
35
+
55
36
  //#endregion Helpers
37
+
56
38
  //#region Local Variables
57
- var clientCredentials;
58
- var oauthToken = null;
59
- var tokenPersistenceService;
60
- var refreshLock;
61
- var logger;
39
+ exports.takeMatchesModelFetchFailed = takeMatchesModelFetchFailed;
40
+ let clientCredentials;
41
+ let oauthToken = null;
42
+ let tokenPersistenceService;
43
+ let refreshLock;
44
+ let logger;
45
+
62
46
  //#endregion Local Variables
63
- function getTokenFromCode(code) {
64
- var getTokenModelName, formBody, formBodyString, _a, fetchResultAction, fetchErrorAction;
65
- return __generator(this, function (_b) {
66
- switch (_b.label) {
67
- case 0:
68
- getTokenModelName = 'getToken';
69
- formBody = [
70
- 'grant_type=authorization_code',
71
- "client_id=" + clientCredentials.client_id,
72
- "client_secret=" + clientCredentials.client_secret,
73
- "code=" + encodeURIComponent(code)
74
- ];
75
- formBodyString = formBody.join('&');
76
- return [4 /*yield*/, 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
- case 1:
83
- _b.sent();
84
- return [4 /*yield*/, effects_1.race({
85
- fetchResultAction: effects_1.take(exports.takeMatchesModelFetchReceived(getTokenModelName)),
86
- fetchErrorAction: effects_1.take(exports.takeMatchesModelFetchFailed(getTokenModelName))
87
- })];
88
- case 2:
89
- _a = _b.sent(), fetchResultAction = _a.fetchResultAction, fetchErrorAction = _a.fetchErrorAction;
90
- if (fetchErrorAction || !(fetchResultAction === null || fetchResultAction === void 0 ? void 0 : fetchResultAction.data)) {
91
- return [2 /*return*/, null];
92
- }
93
- return [2 /*return*/, fetchResultAction.data];
94
- }
95
- });
47
+
48
+ function* getTokenFromCode(code) {
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;
96
71
  }
97
- exports.getTokenFromCode = getTokenFromCode;
98
- function getTokenFromRefreshToken(oauthTokenParam) {
99
- var getTokenModelName, formBody, formBodyString, _a, fetchResultAction, fetchErrorAction;
100
- var _b;
101
- return __generator(this, function (_c) {
102
- switch (_c.label) {
103
- case 0:
104
- getTokenModelName = 'getToken';
105
- formBody = [
106
- 'grant_type=refresh_token',
107
- "client_id=" + clientCredentials.client_id,
108
- "client_secret=" + clientCredentials.client_secret,
109
- "refresh_token=" + encodeURIComponent(oauthTokenParam.refresh_token)
110
- ];
111
- formBodyString = formBody.join('&');
112
- return [4 /*yield*/, effects_1.put({
113
- type: actions_1.MODEL_FETCH_REQUEST_ACTION_TYPE.FETCH_REQUEST,
114
- modelName: getTokenModelName,
115
- body: formBodyString,
116
- noStore: true
117
- })];
118
- case 1:
119
- _c.sent();
120
- return [4 /*yield*/, effects_1.race({
121
- fetchResultAction: effects_1.take(exports.takeMatchesModelFetchReceived(getTokenModelName)),
122
- fetchErrorAction: effects_1.take(exports.takeMatchesModelFetchFailed(getTokenModelName))
123
- })
124
- // any error response
125
- ];
126
- case 2:
127
- _a = _c.sent(), fetchResultAction = _a.fetchResultAction, fetchErrorAction = _a.fetchErrorAction;
128
- // any error response
129
- if (fetchErrorAction) {
130
- // ignore server errors
131
- if (((_b = fetchErrorAction.errorData) === null || _b === void 0 ? void 0 : _b.status) &&
132
- fetchErrorAction.errorData.status >= types_1.HTTP_STATUS_CODE.INTERNAL_SERVER_ERROR) {
133
- return [2 /*return*/, oauthTokenParam];
134
- }
135
- return [2 /*return*/, null];
136
- }
137
- // for some reason the response had no body
138
- if (!(fetchResultAction === null || fetchResultAction === void 0 ? void 0 : fetchResultAction.data)) {
139
- return [2 /*return*/, null];
140
- }
141
- return [2 /*return*/, fetchResultAction.data];
142
- }
143
- });
72
+ function* getTokenFromRefreshToken(oauthTokenParam) {
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;
96
+ }
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;
144
104
  }
145
- exports.getTokenFromRefreshToken = getTokenFromRefreshToken;
146
- function performTokenRefresh() {
147
- var originalAccessToken;
148
- return __generator(this, function (_a) {
149
- switch (_a.label) {
150
- case 0:
151
- if (!(refreshLock || !oauthToken)) return [3 /*break*/, 2];
152
- // already refreshing. wait for the current refresh to succeed or fail.
153
- return [4 /*yield*/, effects_1.race({
154
- refreshSuccess: effects_1.take(actions_1.AUTH_TOKEN_SUCCESS_ACTION_TYPE.TOKEN_REFRESH_SUCCEEDED),
155
- refreshFailed: effects_1.take(actions_1.AUTH_ACTION_TYPE.TOKEN_REFRESH_FAILED)
156
- })];
157
- case 1:
158
- // already refreshing. wait for the current refresh to succeed or fail.
159
- _a.sent();
160
- return [2 /*return*/];
161
- case 2:
162
- logger.debug('Refreshing OAuth token');
163
- refreshLock = true;
164
- originalAccessToken = oauthToken.access_token;
165
- return [4 /*yield*/, effects_1.call(getTokenFromRefreshToken, oauthToken)];
166
- case 3:
167
- oauthToken = _a.sent();
168
- if (!(!!oauthToken && oauthToken.access_token !== originalAccessToken)) return [3 /*break*/, 6];
169
- logger.debug('OAuth token refreshed');
170
- return [4 /*yield*/, effects_1.call(tokenPersistenceService.persistToken, oauthToken)];
171
- case 4:
172
- _a.sent();
173
- return [4 /*yield*/, effects_1.put({ type: actions_1.AUTH_TOKEN_SUCCESS_ACTION_TYPE.TOKEN_REFRESH_SUCCEEDED, oauthToken: oauthToken })];
174
- case 5:
175
- _a.sent();
176
- return [3 /*break*/, 8];
177
- case 6:
178
- if (!(oauthToken === null)) return [3 /*break*/, 8];
179
- logger.debug('OAuth token failed to refresh');
180
- // This should never happen outside of the token having been revoked on the server side
181
- return [4 /*yield*/, effects_1.all({
182
- refreshFailed: effects_1.put({ type: actions_1.AUTH_ACTION_TYPE.TOKEN_REFRESH_FAILED }),
183
- logOut: effects_1.put({ type: actions_1.AUTH_ACTION_TYPE.LOG_OUT_REQUESTED })
184
- })];
185
- case 7:
186
- // This should never happen outside of the token having been revoked on the server side
187
- _a.sent();
188
- _a.label = 8;
189
- case 8:
190
- refreshLock = false;
191
- return [2 /*return*/];
192
- }
105
+ function* performTokenRefresh() {
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)
193
111
  });
194
- }
195
- exports.performTokenRefresh = performTokenRefresh;
196
- function loginFlow(modelFetchRequestAction) {
197
- var _a, fetchResultAction, fetchErrorAction, code;
198
- var _b;
199
- return __generator(this, function (_c) {
200
- switch (_c.label) {
201
- case 0: return [4 /*yield*/, effects_1.put(modelFetchRequestAction)];
202
- case 1:
203
- _c.sent();
204
- return [4 /*yield*/, effects_1.race({
205
- fetchResultAction: effects_1.take(exports.takeMatchesModelFetchReceived(modelFetchRequestAction.modelName)),
206
- fetchErrorAction: effects_1.take(exports.takeMatchesModelFetchFailed(modelFetchRequestAction.modelName))
207
- })];
208
- case 2:
209
- _a = _c.sent(), fetchResultAction = _a.fetchResultAction, fetchErrorAction = _a.fetchErrorAction;
210
- if (fetchErrorAction) {
211
- return [2 /*return*/, null];
212
- }
213
- code = (_b = fetchResultAction === null || fetchResultAction === void 0 ? void 0 : fetchResultAction.data) === null || _b === void 0 ? void 0 : _b.code;
214
- if (!code) {
215
- return [2 /*return*/, null];
216
- }
217
- return [4 /*yield*/, effects_1.call(getTokenFromCode, code)];
218
- case 3: return [2 /*return*/, _c.sent()];
219
- }
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
220
128
  });
221
- }
222
- exports.loginFlow = loginFlow;
223
- function credentialsLoginFlow(action, modelName) {
224
- var modelFetchRequestAction;
225
- return __generator(this, function (_a) {
226
- switch (_a.label) {
227
- case 0:
228
- modelFetchRequestAction = {
229
- // set required defaults
230
- type: actions_1.MODEL_FETCH_REQUEST_ACTION_TYPE.FETCH_REQUEST,
231
- modelName: modelName,
232
- noStore: true,
233
- // force no retry
234
- noRetry: true,
235
- // pass thru body
236
- body: action.body
237
- };
238
- return [4 /*yield*/, effects_1.call(loginFlow, modelFetchRequestAction)];
239
- case 1: return [2 /*return*/, _a.sent()];
240
- }
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
+ })
241
139
  });
140
+ }
141
+ refreshLock = false;
242
142
  }
243
- exports.credentialsLoginFlow = credentialsLoginFlow;
244
- function casV1LoginFlow(action) {
245
- return __generator(this, function (_a) {
246
- switch (_a.label) {
247
- case 0: return [4 /*yield*/, effects_1.call(credentialsLoginFlow, action, 'codeFromCasV1')];
248
- case 1: return [2 /*return*/, _a.sent()];
249
- }
250
- });
143
+ function* loginFlow(modelFetchRequestAction) {
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);
251
160
  }
252
- exports.casV1LoginFlow = casV1LoginFlow;
253
- function localLoginFlow(action) {
254
- return __generator(this, function (_a) {
255
- switch (_a.label) {
256
- case 0: return [4 /*yield*/, effects_1.call(credentialsLoginFlow, action, 'codeFromLocalCredentials')];
257
- case 1: return [2 /*return*/, _a.sent()];
258
- }
259
- });
161
+ function* credentialsLoginFlow(action, modelName) {
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);
260
173
  }
261
- exports.localLoginFlow = localLoginFlow;
262
- function casTicketLoginFlow(ticket, service) {
263
- var modelFetchRequestAction;
264
- return __generator(this, function (_a) {
265
- switch (_a.label) {
266
- case 0:
267
- modelFetchRequestAction = {
268
- type: actions_1.MODEL_FETCH_REQUEST_ACTION_TYPE.FETCH_REQUEST,
269
- modelName: 'codeFromCasTicket',
270
- noStore: true,
271
- queryParams: {
272
- ticket: ticket,
273
- service: service
274
- }
275
- };
276
- return [4 /*yield*/, effects_1.call(loginFlow, modelFetchRequestAction)];
277
- case 1: return [2 /*return*/, _a.sent()];
278
- }
279
- });
174
+ function* casV1LoginFlow(action) {
175
+ return yield (0, _effects.call)(credentialsLoginFlow, action, 'codeFromCasV1');
280
176
  }
281
- exports.casTicketLoginFlow = casTicketLoginFlow;
282
- function handleAuthFailure(action) {
283
- return __generator(this, function (_a) {
284
- switch (_a.label) {
285
- case 0:
286
- if (!(oauthToken && action.errorData && action.errorData.status === types_1.HTTP_STATUS_CODE.UNAUTHORIZED)) return [3 /*break*/, 2];
287
- logger.debug('token expired - refreshing');
288
- return [4 /*yield*/, effects_1.call(performTokenRefresh)];
289
- case 1:
290
- _a.sent();
291
- _a.label = 2;
292
- case 2: return [2 /*return*/];
293
- }
294
- });
177
+ function* localLoginFlow(action) {
178
+ return yield (0, _effects.call)(credentialsLoginFlow, action, 'codeFromLocalCredentials');
295
179
  }
296
- exports.handleAuthFailure = handleAuthFailure;
297
- var getOAuthToken = function (modelName) {
298
- var thirtySecondsFromNow;
299
- return __generator(this, function (_a) {
300
- switch (_a.label) {
301
- case 0:
302
- // Don't try to refresh the token if we're already in a request to refresh the token
303
- if (modelName === 'getToken') {
304
- return [2 /*return*/, null];
305
- }
306
- if (!(oauthToken && oauthToken['.expires'])) return [3 /*break*/, 2];
307
- thirtySecondsFromNow = new Date();
308
- thirtySecondsFromNow.setSeconds(thirtySecondsFromNow.getSeconds() + 30);
309
- if (!(new Date(oauthToken['.expires']) < thirtySecondsFromNow)) return [3 /*break*/, 2];
310
- // start a token refresh and wait for the success action in case another refresh is currently happening
311
- return [4 /*yield*/, effects_1.call(performTokenRefresh)];
312
- case 1:
313
- // start a token refresh and wait for the success action in case another refresh is currently happening
314
- _a.sent();
315
- return [2 /*return*/, oauthToken];
316
- case 2: return [2 /*return*/, oauthToken];
317
- }
318
- });
180
+ function* casTicketLoginFlow(ticket, service) {
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);
191
+ }
192
+ function* handleAuthFailure(action) {
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
+ }
199
+ }
200
+ const getOAuthToken = function* (modelName) {
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;
212
+ }
213
+ }
214
+ return oauthToken;
319
215
  };
320
216
  exports.getOAuthToken = getOAuthToken;
321
- function authSaga(clientCredentialsParam, tokenPersistenceServiceParam, ticketProviderService, codeProviderService) {
322
- var casTicket, service, code, _a, casV1Action, localLoginAction;
323
- if (tokenPersistenceServiceParam === void 0) { tokenPersistenceServiceParam = tokenPersistenceService_1.tokenPersistenceService; }
324
- if (ticketProviderService === void 0) { ticketProviderService = ticketProviderService_1.ticketProviderService; }
325
- if (codeProviderService === void 0) { codeProviderService = codeProviderService_1.codeProviderService; }
326
- return __generator(this, function (_b) {
327
- switch (_b.label) {
328
- case 0:
329
- logger = logger_1.getLogger();
330
- /* istanbul ignore if */
331
- if (!clientCredentialsParam) {
332
- throw new Error("'clientCredentialsParam' is required for authSaga");
333
- }
334
- clientCredentials = clientCredentialsParam;
335
- tokenPersistenceService = tokenPersistenceServiceParam;
336
- return [4 /*yield*/, effects_1.call(tokenPersistenceService.getPersistedToken)
337
- // If no token, try to get CAS ticket (normally in the URL), use it to get a token
338
- ];
339
- case 1:
340
- // Try to get persisted token (normally in AsyncStorage or LocalStorage)
341
- oauthToken = _b.sent();
342
- if (!!oauthToken) return [3 /*break*/, 3];
343
- casTicket = ticketProviderService.getTicket();
344
- ticketProviderService.removeTicket();
345
- service = ticketProviderService.getAppServiceName();
346
- if (!(casTicket && service)) return [3 /*break*/, 3];
347
- return [4 /*yield*/, effects_1.call(casTicketLoginFlow, casTicket, service)];
348
- case 2:
349
- oauthToken = _b.sent();
350
- _b.label = 3;
351
- case 3:
352
- code = codeProviderService.getCode();
353
- if (!code) return [3 /*break*/, 7];
354
- if (!oauthToken) return [3 /*break*/, 5];
355
- return [4 /*yield*/, effects_1.all({
356
- clearUserData: effects_1.put({ type: actions_1.MODEL_REMOVE_KEY_ACTION_TYPE, modelPath: 'user' }),
357
- clearPersistentToken: effects_1.call(tokenPersistenceService.persistToken, null)
358
- })];
359
- case 4:
360
- _b.sent();
361
- oauthToken = null;
362
- _b.label = 5;
363
- case 5: return [4 /*yield*/, effects_1.call(getTokenFromCode, code)];
364
- case 6:
365
- oauthToken = _b.sent();
366
- _b.label = 7;
367
- case 7:
368
- codeProviderService.removeCode();
369
- return [4 /*yield*/, effects_1.put({ type: actions_1.AUTH_TOKEN_ACTION_TYPE.AUTH_INITIALIZED, oauthToken: oauthToken })];
370
- case 8:
371
- _b.sent();
372
- return [4 /*yield*/, effects_1.takeEvery(actions_1.MODEL_FETCH_ERROR_ACTION_TYPE.TRY_FETCH_FAILED, handleAuthFailure)];
373
- case 9:
374
- _b.sent();
375
- _b.label = 10;
376
- case 10:
377
- if (!!oauthToken) return [3 /*break*/, 16];
378
- return [4 /*yield*/, effects_1.race({
379
- casV1Action: effects_1.take(actions_1.AUTH_CAS_V1_LOGIN_REQUEST_ACTION_TYPE.CAS_V1_LOGIN_REQUEST),
380
- localLoginAction: effects_1.take(actions_1.AUTH_LOCAL_LOGIN_REQUEST_ACTION_TYPE.LOCAL_LOGIN_REQUEST)
381
- })];
382
- case 11:
383
- _a = _b.sent(), casV1Action = _a.casV1Action, localLoginAction = _a.localLoginAction;
384
- return [4 /*yield*/, effects_1.put({ type: actions_1.AUTH_ACTION_TYPE.LOGIN_REQUESTED })];
385
- case 12:
386
- _b.sent();
387
- if (!casV1Action) return [3 /*break*/, 14];
388
- return [4 /*yield*/, effects_1.call(casV1LoginFlow, casV1Action)];
389
- case 13:
390
- oauthToken = _b.sent();
391
- return [3 /*break*/, 16];
392
- case 14:
393
- if (!localLoginAction) return [3 /*break*/, 16];
394
- return [4 /*yield*/, effects_1.call(localLoginFlow, localLoginAction)];
395
- case 15:
396
- oauthToken = _b.sent();
397
- _b.label = 16;
398
- case 16:
399
- if (!oauthToken) return [3 /*break*/, 19];
400
- return [4 /*yield*/, effects_1.call(tokenPersistenceService.persistToken, oauthToken)];
401
- case 17:
402
- _b.sent();
403
- return [4 /*yield*/, effects_1.all({
404
- loginSuccess: effects_1.put({
405
- type: actions_1.AUTH_TOKEN_SUCCESS_ACTION_TYPE.GET_TOKEN_SUCCEEDED,
406
- oauthToken: oauthToken
407
- }),
408
- getUserInfo: effects_1.put({
409
- type: actions_1.MODEL_FETCH_REQUEST_ACTION_TYPE.FETCH_REQUEST,
410
- modelName: 'user.userInfo'
411
- }),
412
- logOut: effects_1.take(actions_1.AUTH_ACTION_TYPE.LOG_OUT_REQUESTED)
413
- })];
414
- case 18:
415
- _b.sent();
416
- return [3 /*break*/, 21];
417
- case 19: return [4 /*yield*/, effects_1.put({ type: actions_1.AUTH_ACTION_TYPE.LOGIN_FAILED })];
418
- case 20:
419
- _b.sent();
420
- _b.label = 21;
421
- case 21: return [4 /*yield*/, effects_1.all({
422
- clearUserData: effects_1.put({ type: actions_1.MODEL_REMOVE_KEY_ACTION_TYPE, modelPath: 'user' }),
423
- clearPersistentToken: effects_1.call(tokenPersistenceService.persistToken, null)
424
- })];
425
- case 22:
426
- _b.sent();
427
- oauthToken = null;
428
- _b.label = 23;
429
- case 23:
430
- if (true) return [3 /*break*/, 10];
431
- _b.label = 24;
432
- case 24: return [2 /*return*/];
433
- }
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)();
223
+ clientCredentials = clientCredentialsParam;
224
+ tokenPersistenceService = tokenPersistenceServiceParam;
225
+
226
+ // Try to get persisted token (normally in AsyncStorage or LocalStorage)
227
+ oauthToken = yield (0, _effects.call)(tokenPersistenceService.getPersistedToken);
228
+
229
+ // If no token, try to get CAS ticket (normally in the URL), use it to get a token
230
+ if (!oauthToken) {
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
+ }
237
+ }
238
+
239
+ // If OAuth Code exists (normally in the URL), use it to get a token
240
+ // e.g. LTI, Shibboleth, Facebook, Google
241
+ const code = codeProviderService.getCode();
242
+ if (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);
255
+ }
256
+ codeProviderService.removeCode();
257
+ yield (0, _effects.put)({
258
+ type: _actions.AUTH_TOKEN_ACTION_TYPE.AUTH_INITIALIZED,
259
+ oauthToken
434
260
  });
261
+ yield (0, _effects.takeEvery)(_actions.MODEL_FETCH_ERROR_ACTION_TYPE.TRY_FETCH_FAILED, handleAuthFailure);
262
+ do {
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);
278
+ }
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)
292
+ });
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;
306
+ } while (true);
307
+ }();
435
308
  }
436
- exports.default = authSaga;
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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