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