studiokit-scaffolding-js 7.0.12-next.1.4 → 7.0.12-next.2.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/lib/components/ActionList.js +170 -34
- package/lib/components/AlertDialog.js +133 -9
- package/lib/components/AlertWithIcon.js +92 -25
- package/lib/components/ConnectedModal.js +43 -13
- package/lib/components/Dropdowns/GroupsDropdown.js +69 -44
- package/lib/components/Dropdowns/ManagedNavDropdown.js +100 -67
- package/lib/components/Dropdowns/UserDropdown.js +111 -21
- package/lib/components/Dropdowns/index.js +27 -9
- package/lib/components/EntityOwnerList.js +52 -18
- package/lib/components/Error.js +106 -11
- package/lib/components/ErrorBoundary.js +134 -35
- package/lib/components/ErrorMessage.js +44 -9
- package/lib/components/Forms/DateField.js +61 -42
- package/lib/components/Forms/TimeField.js +81 -42
- package/lib/components/Forms/index.js +27 -4
- package/lib/components/Groups/CreateEditCopySaveButtons.js +114 -11
- package/lib/components/Groups/ExternalGroups/Attach.js +212 -148
- package/lib/components/Groups/ExternalGroups/Table.js +181 -45
- package/lib/components/Groups/GroupCreateOrEditCommonProps.js +5 -1
- package/lib/components/Groups/RosterSyncInfo.js +147 -20
- package/lib/components/HOC/AccessibleAppComponent.js +98 -73
- package/lib/components/HOC/ActivityRequiredComponent.js +92 -49
- package/lib/components/HOC/AsyncComponent.js +54 -39
- package/lib/components/HOC/AuthenticatedComponent.js +58 -38
- package/lib/components/HOC/CollectionComponent.js +170 -110
- package/lib/components/HOC/CollectionFirstItemComponent.js +51 -38
- package/lib/components/HOC/CollectionItemComponent.js +168 -106
- package/lib/components/HOC/ConnectedModalComponent.js +109 -80
- package/lib/components/HOC/DataDependentComponent.js +29 -21
- package/lib/components/HOC/EntityComponent.js +71 -57
- package/lib/components/HOC/FullscreenModalComponent.js +163 -123
- package/lib/components/HOC/GroupActivityRequiredComponent.js +45 -31
- package/lib/components/HOC/GuidComponent.js +29 -22
- package/lib/components/HOC/ModelContextDependencyVerifyComponent.js +41 -31
- package/lib/components/HOC/ModelErrorRedirectComponent.js +51 -47
- package/lib/components/HOC/SearchPersistorComponent.js +240 -166
- package/lib/components/HOC/UnauthenticatedComponent.js +37 -25
- package/lib/components/HOC/UserComponent.js +12 -7
- package/lib/components/Icons/IconAlphaList.js +33 -5
- package/lib/components/Icons/IconExternalUser.js +33 -5
- package/lib/components/Icons/IconImpersonation.js +33 -5
- package/lib/components/Icons/IconStopImpersonating.js +33 -5
- package/lib/components/Icons/IconTable.js +35 -7
- package/lib/components/Icons/IconTableDeleteCol.js +33 -5
- package/lib/components/Icons/IconTableDeleteRow.js +33 -5
- package/lib/components/Icons/IconTableInsertCol.js +33 -5
- package/lib/components/Icons/IconTableInsertRow.js +33 -5
- package/lib/components/Impersonation/Button.js +77 -13
- package/lib/components/Impersonation/Link.js +77 -13
- package/lib/components/Impersonation/UserDetail.js +66 -9
- package/lib/components/Loading.js +26 -4
- package/lib/components/LockDownBrowser/Check.js +194 -49
- package/lib/components/LockDownBrowser/ExitButton.js +26 -9
- package/lib/components/LockDownBrowser/Launch.js +70 -62
- package/lib/components/Lti/Confirm.js +152 -11
- package/lib/components/Lti/CreateNonLtiGroupAlertDialog.js +170 -33
- package/lib/components/Lti/Launch.js +105 -24
- package/lib/components/Lti/LaunchGroup.js +85 -13
- package/lib/components/ManageTable.js +309 -87
- package/lib/components/ManageTableNoDataComponent.js +42 -4
- package/lib/components/NewVersionAlert.js +82 -46
- package/lib/components/NotFound.js +86 -10
- package/lib/components/Notifications.js +185 -126
- package/lib/components/PaginationNextButton.js +33 -6
- package/lib/components/PaginationPreviousButton.js +33 -6
- package/lib/components/Quill/CustomToolbar.js +432 -218
- package/lib/components/Quill/Formats/Image.js +73 -63
- package/lib/components/Quill/Formats/List.js +45 -45
- package/lib/components/Quill/Formats/Video.js +28 -24
- package/lib/components/Quill/ImageDropModule.js +147 -117
- package/lib/components/Quill/ImageWarning.js +47 -9
- package/lib/components/Quill/ImageWithAltTextModal.js +425 -86
- package/lib/components/Quill/Specs/CustomImageSpec.js +42 -34
- package/lib/components/Quill/Specs/CustomVideoSpec.js +34 -28
- package/lib/components/Quill/TableModule/Blots/BaseTableBlot.js +98 -98
- package/lib/components/Quill/TableModule/Blots/TableBlot.js +52 -47
- package/lib/components/Quill/TableModule/Blots/TableBodyBlot.js +53 -48
- package/lib/components/Quill/TableModule/Blots/TableCellBlot.js +224 -221
- package/lib/components/Quill/TableModule/Blots/TableContainer.js +80 -83
- package/lib/components/Quill/TableModule/Blots/TableRowBlot.js +75 -70
- package/lib/components/Quill/TableModule/constants.js +45 -41
- package/lib/components/Quill/TableModule/index.js +362 -301
- package/lib/components/Quill/TableModule/utils.js +42 -38
- package/lib/components/Quill/accessibilityFix.js +234 -232
- package/lib/components/Quill/index.js +34 -28
- package/lib/components/RefreshIndicator/Bordered.js +47 -6
- package/lib/components/RefreshIndicator/Inline.js +47 -8
- package/lib/components/RefreshIndicator/index.js +263 -59
- package/lib/components/SearchControls.js +216 -11
- package/lib/components/SentryRoute.js +11 -6
- package/lib/components/Tables/RoleFilter.js +69 -32
- package/lib/components/Tables/TextFilter.js +62 -13
- package/lib/components/UserRoles/Add.js +199 -96
- package/lib/components/UserRoles/Context.js +11 -7
- package/lib/components/UserRoles/RoleCell.js +181 -72
- package/lib/components/UserRoles/Select.js +157 -17
- package/lib/components/UserRoles/Table.js +221 -80
- package/lib/components/UserRoles/index.js +534 -384
- package/lib/config/eslint/index.js +32 -28
- package/lib/config/eslint/lib/order.js +26 -27
- package/lib/config/eslint/lib/prettier.js +20 -18
- package/lib/config/eslint/lib/typescript.js +93 -112
- package/lib/config/eslint/react.js +24 -14
- package/lib/constants/baseActivity.js +30 -26
- package/lib/constants/baseRole.js +14 -10
- package/lib/constants/configuration.js +33 -29
- package/lib/constants/externalProviderType.js +10 -6
- package/lib/constants/fetchErrorData.js +15 -11
- package/lib/constants/index.js +137 -14
- package/lib/constants/lockDownBrowser.js +28 -24
- package/lib/constants/mockData.js +382 -297
- package/lib/constants/modelStatus.js +15 -11
- package/lib/constants/notificationType.js +12 -8
- package/lib/constants/operatingSystem.js +12 -8
- package/lib/constants/shard.js +11 -7
- package/lib/constants/table.js +21 -21
- package/lib/constants/tier.js +12 -8
- package/lib/constants/userRole.js +15 -5
- package/lib/endpointMappings.js +197 -181
- package/lib/hooks/useCollection.js +82 -62
- package/lib/hooks/useCollectionConfiguration.js +228 -83
- package/lib/hooks/useCollectionItem.js +154 -54
- package/lib/hooks/useGuid.js +20 -8
- package/lib/hooks/usePrevious.js +19 -13
- package/lib/index.js +157 -25
- package/lib/redux/actionCreator.js +50 -28
- package/lib/redux/actions/AuthAction.js +44 -31
- package/lib/redux/actions/ModalAction.js +10 -6
- package/lib/redux/actions/ModelAction.js +77 -39
- package/lib/redux/actions/NotificationAction.js +10 -6
- package/lib/redux/actions/SearchAction.js +9 -5
- package/lib/redux/actions/index.js +60 -7
- package/lib/redux/configureReducers.js +60 -49
- package/lib/redux/configureStore.js +83 -87
- package/lib/redux/helpers.js +6 -2
- package/lib/redux/reducers/authReducer.js +50 -43
- package/lib/redux/reducers/index.js +41 -13
- package/lib/redux/reducers/modalsReducer.js +47 -29
- package/lib/redux/reducers/modelsReducer.js +178 -173
- package/lib/redux/reducers/notificationsReducer.js +24 -18
- package/lib/redux/reducers/searchReducer.js +25 -19
- package/lib/redux/sagas/appInsightsSaga.js +22 -18
- package/lib/redux/sagas/authSaga.js +253 -218
- package/lib/redux/sagas/caliperSaga.js +159 -143
- package/lib/redux/sagas/clockOffsetSaga.js +34 -31
- package/lib/redux/sagas/configurationSaga.js +11 -7
- package/lib/redux/sagas/downtimeApiErrorSaga.js +20 -17
- package/lib/redux/sagas/errorSaga.js +27 -21
- package/lib/redux/sagas/googleAnalyticsSaga.js +28 -24
- package/lib/redux/sagas/identityProviderSaga.js +22 -18
- package/lib/redux/sagas/initialDataLoadSaga.js +37 -28
- package/lib/redux/sagas/lockDownBrowserErrorSaga.js +29 -20
- package/lib/redux/sagas/modelFetchSaga.js +355 -322
- package/lib/redux/sagas/noStoreSaga.js +61 -48
- package/lib/redux/sagas/postLoginDataSaga.js +45 -34
- package/lib/redux/sagas/postLoginRedirectSaga.js +27 -27
- package/lib/redux/sagas/rootSaga.js +82 -57
- package/lib/redux/sagas/sentrySaga.js +29 -25
- package/lib/redux/sagas/userIdSaga.js +16 -12
- package/lib/services/codeProviderService.js +25 -19
- package/lib/services/dateService.js +12 -7
- package/lib/services/documentService.js +17 -12
- package/lib/services/fetchService.js +129 -112
- package/lib/services/persistenceService.js +33 -29
- package/lib/services/ticketProviderService.js +29 -23
- package/lib/services/tokenPersistenceService.js +12 -8
- package/lib/services/windowService.js +18 -14
- package/lib/startup.js +132 -114
- package/lib/types/AppConfiguration.js +5 -1
- package/lib/types/Artifact.js +11 -7
- package/lib/types/BaseReduxState.js +5 -1
- package/lib/types/Client.js +5 -1
- package/lib/types/Collection.js +5 -1
- package/lib/types/Configuration.js +5 -1
- package/lib/types/DeepLinkingResponseRequest.js +5 -1
- package/lib/types/DeletableModel.js +5 -1
- package/lib/types/Event.js +5 -1
- package/lib/types/ExternalGroup.js +5 -1
- package/lib/types/ExternalProvider.js +5 -1
- package/lib/types/ExternalTerm.js +5 -1
- package/lib/types/Group.js +5 -1
- package/lib/types/IdentityProvider.js +5 -1
- package/lib/types/LtiLaunch.js +5 -1
- package/lib/types/NameOnlyEntity.js +5 -1
- package/lib/types/Notification.js +5 -1
- package/lib/types/OptionalRecord.js +5 -1
- package/lib/types/OwnerSchedule.js +5 -1
- package/lib/types/PropertyOfType.js +5 -1
- package/lib/types/Quill.js +5 -1
- package/lib/types/RoleDescription.js +5 -1
- package/lib/types/Search.js +5 -1
- package/lib/types/SimpleLocation.js +5 -1
- package/lib/types/UniTime.js +5 -1
- package/lib/types/User.js +5 -1
- package/lib/types/UserRole.js +5 -1
- package/lib/types/auth/AuthState.js +5 -1
- package/lib/types/auth/CasV1LoginRequestBody.js +5 -1
- package/lib/types/auth/ClientCredentials.js +5 -1
- package/lib/types/auth/CodeProviderService.js +5 -1
- package/lib/types/auth/LocalLoginRequestBody.js +5 -1
- package/lib/types/auth/TicketProviderService.js +5 -1
- package/lib/types/auth/TokenPersistenceService.js +5 -1
- package/lib/types/auth/index.js +82 -9
- package/lib/types/externals.d.js +2 -0
- package/lib/types/index.js +313 -30
- package/lib/types/net/EndpointConfig.js +5 -1
- package/lib/types/net/EndpointMapping.js +5 -1
- package/lib/types/net/EndpointMappings.js +5 -1
- package/lib/types/net/ErrorHandler.js +5 -1
- package/lib/types/net/FetchConfig.js +5 -1
- package/lib/types/net/FetchErrorData.js +10 -6
- package/lib/types/net/FetchResult.js +5 -1
- package/lib/types/net/HTTPMethod.js +5 -1
- package/lib/types/net/HTTPStatusCode.js +16 -12
- package/lib/types/net/Metadata.js +5 -1
- package/lib/types/net/Model.js +5 -1
- package/lib/types/net/ModelCollection.js +5 -1
- package/lib/types/net/ModelsState.js +5 -1
- package/lib/types/net/OAuthToken.js +5 -1
- package/lib/types/net/OAuthTokenOrNull.js +5 -1
- package/lib/types/net/TokenAccessFunction.js +5 -1
- package/lib/types/net/index.js +181 -18
- package/lib/utils/baseActivity.js +133 -123
- package/lib/utils/baseRole.js +37 -33
- package/lib/utils/collection.js +425 -298
- package/lib/utils/cookies.js +22 -19
- package/lib/utils/date.js +303 -279
- package/lib/utils/dom.js +176 -165
- package/lib/utils/domainIdentifier.js +9 -5
- package/lib/utils/entityUserRole.js +6 -2
- package/lib/utils/error.js +17 -15
- package/lib/utils/events.js +40 -31
- package/lib/utils/externalGroup.js +22 -18
- package/lib/utils/externalProviders.js +8 -4
- package/lib/utils/externalTerms.js +10 -3
- package/lib/utils/fetch.js +179 -180
- package/lib/utils/group.js +18 -7
- package/lib/utils/groupDates.js +37 -33
- package/lib/utils/groupRoles.js +25 -21
- package/lib/utils/lockDownBrowser.js +15 -11
- package/lib/utils/logger.js +26 -22
- package/lib/utils/lti.js +9 -4
- package/lib/utils/model.js +36 -41
- package/lib/utils/number.js +21 -18
- package/lib/utils/promise.js +28 -21
- package/lib/utils/quill.js +65 -62
- package/lib/utils/route.js +58 -55
- package/lib/utils/search.js +76 -80
- package/lib/utils/shard.js +37 -37
- package/lib/utils/sort.js +50 -42
- package/lib/utils/string.js +13 -8
- package/lib/utils/table.js +38 -33
- package/lib/utils/timezone.js +10 -6
- package/lib/utils/url.js +142 -142
- package/lib/utils/user.js +58 -55
- package/lib/utils/userAgent.js +10 -10
- package/lib/utils/userRole.js +57 -49
- package/package.json +17 -3
|
@@ -1,274 +1,309 @@
|
|
|
1
1
|
"use strict";
|
|
2
|
-
|
|
3
|
-
|
|
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.
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
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
|
-
|
|
27
|
+
|
|
28
|
+
const matchesModelFetchReceived = (action, modelName) => (0, _actions.isTransientModelFetchResultAction)(action) && action.modelPath === modelName;
|
|
23
29
|
exports.matchesModelFetchReceived = matchesModelFetchReceived;
|
|
24
|
-
const takeMatchesModelFetchReceived =
|
|
30
|
+
const takeMatchesModelFetchReceived = modelName => incomingAction => matchesModelFetchReceived(incomingAction, modelName);
|
|
25
31
|
exports.takeMatchesModelFetchReceived = takeMatchesModelFetchReceived;
|
|
26
|
-
const matchesModelFetchFailed = (action, modelName) => (0,
|
|
32
|
+
const matchesModelFetchFailed = (action, modelName) => (0, _actions.isTransientModelFetchErrorAction)(action) && action.modelPath === modelName;
|
|
27
33
|
exports.matchesModelFetchFailed = matchesModelFetchFailed;
|
|
28
|
-
const takeMatchesModelFetchFailed =
|
|
29
|
-
|
|
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
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
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
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
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
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
return
|
|
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
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
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
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
return
|
|
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
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
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
|
-
|
|
175
|
+
return yield (0, _effects.call)(credentialsLoginFlow, action, 'codeFromCasV1');
|
|
164
176
|
}
|
|
165
177
|
function* localLoginFlow(action) {
|
|
166
|
-
|
|
178
|
+
return yield (0, _effects.call)(credentialsLoginFlow, action, 'codeFromLocalCredentials');
|
|
167
179
|
}
|
|
168
180
|
function* casTicketLoginFlow(ticket, service) {
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
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
|
-
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
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
|
-
|
|
190
|
-
|
|
191
|
-
|
|
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
|
-
|
|
194
|
-
|
|
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
|
|
206
|
-
|
|
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,
|
|
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
|
-
|
|
214
|
-
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
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
|
-
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
|
|
228
|
-
|
|
229
|
-
|
|
230
|
-
|
|
231
|
-
|
|
232
|
-
|
|
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,
|
|
236
|
-
|
|
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
|
-
|
|
239
|
-
|
|
240
|
-
|
|
241
|
-
|
|
242
|
-
|
|
243
|
-
|
|
244
|
-
|
|
245
|
-
|
|
246
|
-
|
|
247
|
-
|
|
248
|
-
|
|
249
|
-
|
|
250
|
-
|
|
251
|
-
if (
|
|
252
|
-
|
|
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
|
-
|
|
269
|
-
|
|
270
|
-
|
|
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
|
-
|
|
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,{"version":3,"names":["_effects","require","_codeProviderService","_ticketProviderService","_tokenPersistenceService","_types","_logger","_actions","matchesModelFetchReceived","action","modelName","isTransientModelFetchResultAction","modelPath","exports","takeMatchesModelFetchReceived","incomingAction","matchesModelFetchFailed","isTransientModelFetchErrorAction","takeMatchesModelFetchFailed","clientCredentials","oauthToken","tokenPersistenceService","refreshLock","logger","getTokenFromCode","code","getTokenModelName","formBody","client_id","client_secret","encodeURIComponent","formBodyString","join","put","type","MODEL_FETCH_REQUEST_ACTION_TYPE","FETCH_REQUEST","body","noStore","fetchResultAction","fetchErrorAction","race","take","data","getTokenFromRefreshToken","oauthTokenParam","refresh_token","errorData","status","HTTP_STATUS_CODE","INTERNAL_SERVER_ERROR","performTokenRefresh","refreshSuccess","AUTH_TOKEN_SUCCESS_ACTION_TYPE","TOKEN_REFRESH_SUCCEEDED","refreshFailed","AUTH_ACTION_TYPE","TOKEN_REFRESH_FAILED","debug","originalAccessToken","access_token","call","persistToken","all","logOut","LOG_OUT_REQUESTED","loginFlow","modelFetchRequestAction","credentialsLoginFlow","noRetry","casV1LoginFlow","localLoginFlow","casTicketLoginFlow","ticket","service","queryParams","handleAuthFailure","UNAUTHORIZED","getOAuthToken","thirtySecondsFromNow","Date","setSeconds","getSeconds","authSaga","clientCredentialsParam","tokenPersistenceServiceParam","arguments","length","undefined","defaultTokenPersistenceService","ticketProviderService","defaultTicketProviderService","codeProviderService","defaultCodeProviderService","getLogger","getPersistedToken","casTicket","getTicket","removeTicket","getAppServiceName","getCode","clearUserData","MODEL_REMOVE_KEY_ACTION_TYPE","clearPersistentToken","removeCode","AUTH_TOKEN_ACTION_TYPE","AUTH_INITIALIZED","takeEvery","MODEL_FETCH_ERROR_ACTION_TYPE","TRY_FETCH_FAILED","casV1Action","localLoginAction","AUTH_CAS_V1_LOGIN_REQUEST_ACTION_TYPE","CAS_V1_LOGIN_REQUEST","AUTH_LOCAL_LOGIN_REQUEST_ACTION_TYPE","LOCAL_LOGIN_REQUEST","LOGIN_REQUESTED","loginSuccess","GET_TOKEN_SUCCEEDED","getUserInfo","LOGIN_FAILED"],"sources":["../../../src/redux/sagas/authSaga.ts"],"sourcesContent":["import { SagaIterator } from '@redux-saga/core'\r\nimport { AnyAction } from 'redux'\r\nimport { all, call, put, race, take, takeEvery } from 'redux-saga/effects'\r\nimport { codeProviderService as defaultCodeProviderService } from '../../services/codeProviderService'\r\nimport { ticketProviderService as defaultTicketProviderService } from '../../services/ticketProviderService'\r\nimport { tokenPersistenceService as defaultTokenPersistenceService } from '../../services/tokenPersistenceService'\r\nimport {\r\n\tClientCredentials,\r\n\tCodeProviderService,\r\n\tHTTP_STATUS_CODE,\r\n\tOAuthToken,\r\n\tOAuthTokenOrNull,\r\n\tTicketProviderService,\r\n\tTokenAccessFunction,\r\n\tTokenPersistenceService\r\n} from '../../types'\r\nimport { getLogger, Logger } from '../../utils/logger'\r\nimport {\r\n\tAUTH_ACTION_TYPE,\r\n\tAUTH_CAS_V1_LOGIN_REQUEST_ACTION_TYPE,\r\n\tAUTH_LOCAL_LOGIN_REQUEST_ACTION_TYPE,\r\n\tAUTH_TOKEN_ACTION_TYPE,\r\n\tAUTH_TOKEN_SUCCESS_ACTION_TYPE,\r\n\tAuthAction,\r\n\tAuthCasV1LoginRequestAction,\r\n\tAuthLocalLoginRequestAction,\r\n\tAuthTokenAction,\r\n\tAuthTokenSuccessAction,\r\n\tisTransientModelFetchErrorAction,\r\n\tisTransientModelFetchResultAction,\r\n\tMODEL_FETCH_ERROR_ACTION_TYPE,\r\n\tMODEL_FETCH_REQUEST_ACTION_TYPE,\r\n\tMODEL_REMOVE_KEY_ACTION_TYPE,\r\n\tModelFetchErrorAction,\r\n\tModelFetchRequestAction,\r\n\tModelFetchResultAction,\r\n\tModelRemoveKeyAction\r\n} from '../actions'\r\n\r\n//#region Helpers\r\n\r\nexport const matchesModelFetchReceived = (action: AnyAction, modelName: string) =>\r\n\tisTransientModelFetchResultAction(action) && action.modelPath === modelName\r\n\r\nexport const takeMatchesModelFetchReceived = (modelName: string) => (incomingAction: AnyAction) =>\r\n\tmatchesModelFetchReceived(incomingAction, modelName)\r\n\r\nexport const matchesModelFetchFailed = (action: AnyAction, modelName: string) =>\r\n\tisTransientModelFetchErrorAction(action) && action.modelPath === modelName\r\n\r\nexport const takeMatchesModelFetchFailed = (modelName: string) => (incomingAction: AnyAction) =>\r\n\tmatchesModelFetchFailed(incomingAction, modelName)\r\n\r\n//#endregion Helpers\r\n\r\n//#region Local Variables\r\n\r\nlet clientCredentials: ClientCredentials\r\nlet oauthToken: OAuthTokenOrNull = null\r\nlet tokenPersistenceService: TokenPersistenceService\r\nlet refreshLock: boolean\r\nlet logger: Logger\r\n\r\n//#endregion Local Variables\r\n\r\nexport function* getTokenFromCode(code: string): SagaIterator {\r\n\tconst getTokenModelName = 'getToken'\r\n\t// Manually creating form-url-encoded body here because NOTHING else uses this content-type\r\n\t// but the OAuth spec requires it\r\n\tconst formBody = [\r\n\t\t'grant_type=authorization_code',\r\n\t\t`client_id=${clientCredentials.client_id}`,\r\n\t\t`client_secret=${clientCredentials.client_secret}`,\r\n\t\t`code=${encodeURIComponent(code)}`\r\n\t]\r\n\tconst formBodyString = formBody.join('&')\r\n\tyield put<ModelFetchRequestAction>({\r\n\t\ttype: MODEL_FETCH_REQUEST_ACTION_TYPE.FETCH_REQUEST,\r\n\t\tmodelName: getTokenModelName,\r\n\t\tbody: formBodyString,\r\n\t\tnoStore: true\r\n\t})\r\n\tconst {\r\n\t\tfetchResultAction,\r\n\t\tfetchErrorAction\r\n\t}: { fetchResultAction?: ModelFetchResultAction; fetchErrorAction?: ModelFetchErrorAction } = yield race({\r\n\t\tfetchResultAction: take(takeMatchesModelFetchReceived(getTokenModelName)),\r\n\t\tfetchErrorAction: take(takeMatchesModelFetchFailed(getTokenModelName))\r\n\t})\r\n\tif (fetchErrorAction || !fetchResultAction?.data) {\r\n\t\treturn null\r\n\t}\r\n\treturn fetchResultAction.data\r\n}\r\n\r\nexport function* getTokenFromRefreshToken(oauthTokenParam: OAuthToken): SagaIterator {\r\n\tconst getTokenModelName = 'getToken'\r\n\t// Manually creating form-url-encoded body here because NOTHING else uses this content-type\r\n\t// but the OAuth spec requires it\r\n\tconst formBody = [\r\n\t\t'grant_type=refresh_token',\r\n\t\t`client_id=${clientCredentials.client_id}`,\r\n\t\t`client_secret=${clientCredentials.client_secret}`,\r\n\t\t`refresh_token=${encodeURIComponent(oauthTokenParam.refresh_token)}`\r\n\t]\r\n\tconst formBodyString = formBody.join('&')\r\n\tyield put<ModelFetchRequestAction>({\r\n\t\ttype: MODEL_FETCH_REQUEST_ACTION_TYPE.FETCH_REQUEST,\r\n\t\tmodelName: getTokenModelName,\r\n\t\tbody: formBodyString,\r\n\t\tnoStore: true\r\n\t})\r\n\tconst {\r\n\t\tfetchResultAction,\r\n\t\tfetchErrorAction\r\n\t}: { fetchResultAction?: ModelFetchResultAction; fetchErrorAction?: ModelFetchErrorAction } = yield race({\r\n\t\tfetchResultAction: take(takeMatchesModelFetchReceived(getTokenModelName)),\r\n\t\tfetchErrorAction: take(takeMatchesModelFetchFailed(getTokenModelName))\r\n\t})\r\n\t// any error response\r\n\tif (fetchErrorAction) {\r\n\t\t// ignore server errors\r\n\t\tif (\r\n\t\t\tfetchErrorAction.errorData?.status &&\r\n\t\t\tfetchErrorAction.errorData.status >= HTTP_STATUS_CODE.INTERNAL_SERVER_ERROR\r\n\t\t) {\r\n\t\t\treturn oauthTokenParam\r\n\t\t}\r\n\t\treturn null\r\n\t}\r\n\t// for some reason the response had no body\r\n\tif (!fetchResultAction?.data) {\r\n\t\treturn null\r\n\t}\r\n\treturn fetchResultAction.data\r\n}\r\n\r\nexport function* performTokenRefresh(): SagaIterator {\r\n\tif (refreshLock || !oauthToken) {\r\n\t\t// already refreshing. wait for the current refresh to succeed or fail.\r\n\t\tyield race({\r\n\t\t\trefreshSuccess: take(AUTH_TOKEN_SUCCESS_ACTION_TYPE.TOKEN_REFRESH_SUCCEEDED),\r\n\t\t\trefreshFailed: take(AUTH_ACTION_TYPE.TOKEN_REFRESH_FAILED)\r\n\t\t})\r\n\t\treturn\r\n\t}\r\n\tlogger.debug('Refreshing OAuth token')\r\n\trefreshLock = true\r\n\t// oauthToken will be set to:\r\n\t// 1. new token (success)\r\n\t// 2. same token (failed from timeout or server error)\r\n\t// 3. null (fail)\r\n\tconst originalAccessToken = oauthToken.access_token\r\n\toauthToken = yield call(getTokenFromRefreshToken, oauthToken)\r\n\tif (!!oauthToken && oauthToken.access_token !== originalAccessToken) {\r\n\t\tlogger.debug('OAuth token refreshed')\r\n\t\tyield call(tokenPersistenceService.persistToken, oauthToken)\r\n\t\tyield put<AuthTokenSuccessAction>({ type: AUTH_TOKEN_SUCCESS_ACTION_TYPE.TOKEN_REFRESH_SUCCEEDED, oauthToken })\r\n\t} else if (oauthToken === null) {\r\n\t\tlogger.debug('OAuth token failed to refresh')\r\n\t\t// This should never happen outside of the token having been revoked on the server side\r\n\t\tyield all({\r\n\t\t\trefreshFailed: put<AuthAction>({ type: AUTH_ACTION_TYPE.TOKEN_REFRESH_FAILED }),\r\n\t\t\tlogOut: put<AuthAction>({ type: AUTH_ACTION_TYPE.LOG_OUT_REQUESTED })\r\n\t\t})\r\n\t}\r\n\trefreshLock = false\r\n}\r\n\r\nexport function* loginFlow(modelFetchRequestAction: ModelFetchRequestAction): SagaIterator {\r\n\tyield put(modelFetchRequestAction)\r\n\tconst {\r\n\t\tfetchResultAction,\r\n\t\tfetchErrorAction\r\n\t}: { fetchResultAction?: ModelFetchResultAction; fetchErrorAction?: ModelFetchErrorAction } = yield race({\r\n\t\tfetchResultAction: take(takeMatchesModelFetchReceived(modelFetchRequestAction.modelName)),\r\n\t\tfetchErrorAction: take(takeMatchesModelFetchFailed(modelFetchRequestAction.modelName))\r\n\t})\r\n\tif (fetchErrorAction) {\r\n\t\treturn null\r\n\t}\r\n\tconst code: string | undefined = fetchResultAction?.data?.code\r\n\tif (!code) {\r\n\t\treturn null\r\n\t}\r\n\treturn yield call(getTokenFromCode, code)\r\n}\r\n\r\nexport function* credentialsLoginFlow(\r\n\taction: AuthCasV1LoginRequestAction | AuthLocalLoginRequestAction,\r\n\tmodelName: string\r\n): SagaIterator {\r\n\tconst modelFetchRequestAction: ModelFetchRequestAction = {\r\n\t\t// set required defaults\r\n\t\ttype: MODEL_FETCH_REQUEST_ACTION_TYPE.FETCH_REQUEST,\r\n\t\tmodelName,\r\n\t\tnoStore: true,\r\n\t\t// force no retry\r\n\t\tnoRetry: true,\r\n\t\t// pass thru body\r\n\t\tbody: action.body\r\n\t}\r\n\treturn yield call(loginFlow, modelFetchRequestAction)\r\n}\r\n\r\nexport function* casV1LoginFlow(action: AuthCasV1LoginRequestAction): SagaIterator {\r\n\treturn yield call(credentialsLoginFlow, action, 'codeFromCasV1')\r\n}\r\n\r\nexport function* localLoginFlow(action: AuthLocalLoginRequestAction): SagaIterator {\r\n\treturn yield call(credentialsLoginFlow, action, 'codeFromLocalCredentials')\r\n}\r\n\r\nexport function* casTicketLoginFlow(ticket: string, service: string): SagaIterator {\r\n\tconst modelFetchRequestAction: ModelFetchRequestAction = {\r\n\t\ttype: MODEL_FETCH_REQUEST_ACTION_TYPE.FETCH_REQUEST,\r\n\t\tmodelName: 'codeFromCasTicket',\r\n\t\tnoStore: true,\r\n\t\tqueryParams: {\r\n\t\t\tticket,\r\n\t\t\tservice\r\n\t\t}\r\n\t}\r\n\treturn yield call(loginFlow, modelFetchRequestAction)\r\n}\r\n\r\nexport function* handleAuthFailure(action: ModelFetchErrorAction): SagaIterator {\r\n\t// This should be unlikely since we normally have a refresh token loop happening\r\n\t// but if the app is backgrounded, the loop might not be caught up yet\r\n\tif (oauthToken && action.errorData && action.errorData.status === HTTP_STATUS_CODE.UNAUTHORIZED) {\r\n\t\tlogger.debug('token expired - refreshing')\r\n\t\tyield call(performTokenRefresh)\r\n\t}\r\n}\r\n\r\nexport const getOAuthToken: TokenAccessFunction = function* (modelName: string) {\r\n\t// Don't try to refresh the token if we're already in a request to refresh the token\r\n\tif (modelName === 'getToken') {\r\n\t\treturn null\r\n\t}\r\n\tif (oauthToken && oauthToken['.expires']) {\r\n\t\tconst thirtySecondsFromNow = new Date()\r\n\t\tthirtySecondsFromNow.setSeconds(thirtySecondsFromNow.getSeconds() + 30)\r\n\t\tif (new Date(oauthToken['.expires']) < thirtySecondsFromNow) {\r\n\t\t\t// start a token refresh and wait for the success action in case another refresh is currently happening\r\n\t\t\tyield call(performTokenRefresh)\r\n\t\t\treturn oauthToken\r\n\t\t}\r\n\t}\r\n\treturn oauthToken\r\n}\r\n\r\nexport default function* authSaga(\r\n\tclientCredentialsParam: ClientCredentials,\r\n\ttokenPersistenceServiceParam: TokenPersistenceService = defaultTokenPersistenceService,\r\n\tticketProviderService: TicketProviderService = defaultTicketProviderService,\r\n\tcodeProviderService: CodeProviderService = defaultCodeProviderService\r\n): SagaIterator {\r\n\tlogger = getLogger()\r\n\r\n\tclientCredentials = clientCredentialsParam\r\n\ttokenPersistenceService = tokenPersistenceServiceParam\r\n\r\n\t// Try to get persisted token (normally in AsyncStorage or LocalStorage)\r\n\toauthToken = yield call(tokenPersistenceService.getPersistedToken)\r\n\r\n\t// If no token, try to get CAS ticket (normally in the URL), use it to get a token\r\n\tif (!oauthToken) {\r\n\t\tconst casTicket = ticketProviderService.getTicket()\r\n\t\tticketProviderService.removeTicket()\r\n\t\tconst service = ticketProviderService.getAppServiceName()\r\n\t\tif (casTicket && service) {\r\n\t\t\toauthToken = yield call(casTicketLoginFlow, casTicket, service)\r\n\t\t}\r\n\t}\r\n\r\n\t// If OAuth Code exists (normally in the URL), use it to get a token\r\n\t// e.g. LTI, Shibboleth, Facebook, Google\r\n\tconst code = codeProviderService.getCode()\r\n\tif (code) {\r\n\t\t// Log out the current user if a new token is about to be generated from the OAuth Code\r\n\t\tif (oauthToken) {\r\n\t\t\tyield all({\r\n\t\t\t\tclearUserData: put<ModelRemoveKeyAction>({ type: MODEL_REMOVE_KEY_ACTION_TYPE, modelPath: 'user' }),\r\n\t\t\t\tclearPersistentToken: call(tokenPersistenceService.persistToken, null)\r\n\t\t\t})\r\n\t\t\toauthToken = null\r\n\t\t}\r\n\t\toauthToken = yield call(getTokenFromCode, code)\r\n\t}\r\n\tcodeProviderService.removeCode()\r\n\r\n\tyield put<AuthTokenAction>({ type: AUTH_TOKEN_ACTION_TYPE.AUTH_INITIALIZED, oauthToken })\r\n\r\n\tyield takeEvery(MODEL_FETCH_ERROR_ACTION_TYPE.TRY_FETCH_FAILED, handleAuthFailure)\r\n\r\n\tdo {\r\n\t\tif (!oauthToken) {\r\n\t\t\tconst {\r\n\t\t\t\tcasV1Action,\r\n\t\t\t\tlocalLoginAction\r\n\t\t\t}: {\r\n\t\t\t\tcasV1Action?: AuthCasV1LoginRequestAction\r\n\t\t\t\tlocalLoginAction?: AuthLocalLoginRequestAction\r\n\t\t\t} = yield race({\r\n\t\t\t\tcasV1Action: take(AUTH_CAS_V1_LOGIN_REQUEST_ACTION_TYPE.CAS_V1_LOGIN_REQUEST),\r\n\t\t\t\tlocalLoginAction: take(AUTH_LOCAL_LOGIN_REQUEST_ACTION_TYPE.LOCAL_LOGIN_REQUEST)\r\n\t\t\t})\r\n\r\n\t\t\tyield put<AuthAction>({ type: AUTH_ACTION_TYPE.LOGIN_REQUESTED })\r\n\t\t\tif (casV1Action) {\r\n\t\t\t\toauthToken = yield call(casV1LoginFlow, casV1Action)\r\n\t\t\t} else if (localLoginAction) {\r\n\t\t\t\toauthToken = yield call(localLoginFlow, localLoginAction)\r\n\t\t\t}\r\n\t\t}\r\n\r\n\t\tif (oauthToken) {\r\n\t\t\tyield call(tokenPersistenceService.persistToken, oauthToken)\r\n\t\t\tyield all({\r\n\t\t\t\tloginSuccess: put<AuthTokenSuccessAction>({\r\n\t\t\t\t\ttype: AUTH_TOKEN_SUCCESS_ACTION_TYPE.GET_TOKEN_SUCCEEDED,\r\n\t\t\t\t\toauthToken\r\n\t\t\t\t}),\r\n\t\t\t\tgetUserInfo: put<ModelFetchRequestAction>({\r\n\t\t\t\t\ttype: MODEL_FETCH_REQUEST_ACTION_TYPE.FETCH_REQUEST,\r\n\t\t\t\t\tmodelName: 'user.userInfo'\r\n\t\t\t\t}),\r\n\t\t\t\tlogOut: take(AUTH_ACTION_TYPE.LOG_OUT_REQUESTED)\r\n\t\t\t})\r\n\t\t} else {\r\n\t\t\tyield put<AuthAction>({ type: AUTH_ACTION_TYPE.LOGIN_FAILED })\r\n\t\t}\r\n\r\n\t\tyield all({\r\n\t\t\tclearUserData: put<ModelRemoveKeyAction>({ type: MODEL_REMOVE_KEY_ACTION_TYPE, modelPath: 'user' }),\r\n\t\t\tclearPersistentToken: call(tokenPersistenceService.persistToken, null)\r\n\t\t})\r\n\t\toauthToken = null\r\n\t} while (true)\r\n}\r\n"],"mappings":"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","ignoreList":[]}
|