Skip to content

Commit

Permalink
STCOR-907 cautiously evaluate localforage data (#1561)
Browse files Browse the repository at this point in the history
Be thorough when evaluating data from localforage to determine whether a
session exists by checking for `user.id`, `tenant`, and
`isAuthenticated` values rather than accepting the existence of a
(possibly empty, or mostly empty) object as proof of life. Without these
more thorough checks, a sparsely-populated value may be passed to
`validateUser()`, causing it to throw when evaluating `user.id`, and
misleadingly dispatching a server-down message.

It is likely that a rogue RTR process is responsible for writing this
garbled/sparse session data. We need to research that and resolve that
problem too, but at least we have a handle on this from the other end.

Refs STCOR-907
  • Loading branch information
zburke authored Nov 22, 2024
1 parent aed9571 commit 494bfcd
Show file tree
Hide file tree
Showing 5 changed files with 207 additions and 32 deletions.
4 changes: 4 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,5 +1,9 @@
# Change history for stripes-core

## 10.1.4 IN PROGRESS

* Cautiously evaluate localforage data when restoring a session. Refs STCOR-907.

## [10.1.3](https://github.com/folio-org/stripes-core/tree/v10.1.3) (2024-11-19)
[Full Changelog](https://github.com/folio-org/stripes-core/compare/v10.1.2...v10.1.3)

Expand Down
6 changes: 5 additions & 1 deletion src/components/Root/Root.js
Original file line number Diff line number Diff line change
Expand Up @@ -81,7 +81,11 @@ class Root extends Component {

componentDidMount() {
const { okapi, store, locale, defaultTranslations } = this.props;
if (this.withOkapi) checkOkapiSession(okapi.url, store, okapi.tenant);
if (this.withOkapi) {
// checkOkapiSession is async and will cause a re-render after a delay
// when an API call's .then() handler updates the store.
checkOkapiSession(okapi.url, store, okapi.tenant);
}
// TODO: remove this after we load locale and translations at start from a public endpoint
loadTranslations(store, locale, defaultTranslations);
}
Expand Down
12 changes: 12 additions & 0 deletions src/components/Root/token-util.test.js
Original file line number Diff line number Diff line change
@@ -1,3 +1,5 @@
import localforage from 'localforage';

Check warning on line 1 in src/components/Root/token-util.test.js

View workflow job for this annotation

GitHub Actions / build-npm

'localforage' is defined but never used. Allowed unused vars must match /React/u

import { RTRError, UnexpectedResourceError } from './Errors';
import {
isFolioApiRequest,
Expand All @@ -12,6 +14,16 @@ import {
} from './token-util';
import { RTR_SUCCESS_EVENT } from './Events';

jest.mock('localforage', () => ({
getItem: jest.fn(() => Promise.resolve({
user: { id: 'robert' },
tenant: 'manhattan',
isAuthenticated: 'i am',
})),
setItem: jest.fn((k, v) => Promise.resolve(v)),
removeItem: jest.fn(() => Promise.resolve()),
}));

describe('isFolioApiRequest', () => {
it('accepts requests whose origin matches okapi\'s', () => {
const oUrl = 'https://millicent-sounds-kinda-like-malificent.edu';
Expand Down
54 changes: 37 additions & 17 deletions src/loginServices.js
Original file line number Diff line number Diff line change
Expand Up @@ -77,18 +77,18 @@ const SESSION_NAME = 'okapiSess';
* simple wrapper around access to values stored in localforage
* to insulate RTR functions from that API.
*
* @returns {object}
* @returns Promise.resolve({object})
*/
export const getOkapiSession = async () => {
return localforage.getItem(SESSION_NAME);
};

/**
* getTokenSess
* getTokenExpiry
* simple wrapper around access to values stored in localforage
* to insulate RTR functions from that API.
*
* @returns {object} shaped like { atExpires, rtExpires }; each is a millisecond timestamp
* @returns Promise.resolve({object}) shaped like { atExpires, rtExpires }; each is a millisecond timestamp
*/
export const getTokenExpiry = async () => {
const sess = await getOkapiSession();
Expand All @@ -102,11 +102,15 @@ export const getTokenExpiry = async () => {
* session with updated token expiration data.
*
* @param {object} shaped like { atExpires, rtExpires }; each is a millisecond timestamp
* @returns {object} updated session object
* @returns Promise.resolve({object}) updated session object
*/
export const setTokenExpiry = async (te) => {
const sess = await getOkapiSession();
return localforage.setItem(SESSION_NAME, { ...sess, tokenExpiration: te });
if (sess) {
return localforage.setItem(SESSION_NAME, { ...sess, tokenExpiration: te });
}

return null;
};


Expand Down Expand Up @@ -709,18 +713,23 @@ export function validateUser(okapiUrl, store, tenant, session) {

/**
* checkOkapiSession
* 1. Pull the session from local storage; if non-empty validate it, dispatching load-resources actions.
* 2. Check if SSO (SAML) is enabled, dispatching check-sso actions
* 3. dispatch set-okapi-ready.
* 1 Pull the session from local storage; if non-empty validate it,
* dispatching load-resources actions to populate the store.
* 2 Check if SSO (SAML) is enabled, dispatching check-sso actions
* 3 dispatch set-okapi-ready.
*
* @param {string} okapiUrl
* @param {redux store} store
* @param {string} tenant
*/
export function checkOkapiSession(okapiUrl, store, tenant) {
getOkapiSession()
return getOkapiSession()
.then((sess) => {
return sess !== null ? validateUser(okapiUrl, store, tenant, sess) : null;
if (sess?.user?.id && sess?.tenant && sess?.isAuthenticated) {
return validateUser(okapiUrl, store, tenant, sess);
}

return null;
})
.then(() => {
return getSSOEnabled(okapiUrl, store, tenant);
Expand Down Expand Up @@ -822,27 +831,38 @@ export function requestSSOLogin(okapiUrl, tenant) {
export function updateUser(store, data) {
return getOkapiSession()
.then((sess) => {
sess.user = { ...sess.user, ...data };
return localforage.setItem(SESSION_NAME, sess);
if (sess?.user?.id && sess?.tenant && sess?.isAuthenticated) {
sess.user = { ...sess.user, ...data };
return localforage.setItem(SESSION_NAME, sess);
}

throw Error('The user could not be updated because an existing session could not be found');
})
.then(() => {
store.dispatch(updateCurrentUser(data));
})
.catch((e) => {
console.error(e); // eslint-disable-line no-console
});
}

/**
* updateTenant
* 1. prepare user info for requested tenant
* 2. update okapi session
* 2. update okapi session with user and tenant data
* @param {object} okapi
* @param {string} tenant
*
* @returns {Promise}
*/
export async function updateTenant(okapi, tenant) {
const okapiSess = await getOkapiSession();
const userWithPermsResponse = await fetchUserWithPerms(okapi.url, tenant, okapi.token);
const userWithPerms = await userWithPermsResponse.json();
const sess = await getOkapiSession();
if (sess?.user?.id && sess?.tenant && sess?.isAuthenticated) {
const userWithPermsResponse = await fetchUserWithPerms(okapi.url, tenant, okapi.token);
const userWithPerms = await userWithPermsResponse.json();

await localforage.setItem(SESSION_NAME, { ...okapiSess, tenant, ...spreadUserWithPerms(userWithPerms) });
await localforage.setItem(SESSION_NAME, { ...sess, tenant, ...spreadUserWithPerms(userWithPerms) });
} else {
console.error('The tenant could not be updated because an existing session could not be found'); // eslint-disable-line no-console
}
}
163 changes: 149 additions & 14 deletions src/loginServices.test.js
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
import localforage from 'localforage';

import {
checkOkapiSession,
createOkapiSession,
getOkapiSession,
getTokenExpiry,
Expand Down Expand Up @@ -342,17 +343,41 @@ describe('validateUser', () => {
});

describe('updateUser', () => {
it('dispatches updateCurrentUser', async () => {
it('dispatches updateCurrentUser when a session exists', async () => {
const s = {
user: { id: 'robert' },
tenant: 'manhattan',
isAuthenticated: 'i am',
};
localforage.getItem = () => Promise.resolve(s);

const store = {
dispatch: jest.fn(),
};
const data = { thunder: 'chicken' };
await updateUser(store, data);
expect(store.dispatch).toHaveBeenCalledWith(updateCurrentUser(data));
});

it('does nothing when session is not found', async () => {
localforage.getItem = () => Promise.resolve(null);

const store = {
dispatch: jest.fn(),
};
const data = { thunder: 'chicken' };
await updateUser(store, data);
expect(store.dispatch).not.toHaveBeenCalled();
});
});

describe('updateTenant', () => {
const s = {
user: { id: 'robert' },
tenant: 'manhattan',
isAuthenticated: 'i am',
};

const okapi = {
currentPerms: {},
};
Expand All @@ -371,16 +396,30 @@ describe('updateTenant', () => {
localforage.setItem.mockClear();
});

it('should set tenant and updated user in session', async () => {
it('updates tenant and user when session exists', async () => {
localforage.getItem = () => Promise.resolve(s);
localforage.setItem = jest.fn(() => Promise.resolve(s));

mockFetchSuccess(data);
await updateTenant(okapi, tenant);
mockFetchCleanUp();

expect(localforage.setItem).toHaveBeenCalledWith('okapiSess', {
...s,
...spreadUserWithPerms(data),
tenant,
});
});

it('does nothing when session is not found', async () => {
localforage.getItem = () => Promise.resolve(null);

mockFetchSuccess(data);
await updateTenant(okapi, tenant);
mockFetchCleanUp();

expect(localforage.setItem).not.toHaveBeenCalled();
});
});

describe('localforage session wrapper', () => {
Expand Down Expand Up @@ -410,22 +449,118 @@ describe('localforage session wrapper', () => {
});
});

it('setTokenExpiry set', async () => {
describe('setTokenExpiry', () => {
it('saves data when a session exists', async () => {
const o = {
margo: 'timmins',
margot: 'margot with a t looks better',
also: 'i thought we were talking about margot robbie?',
tokenExpiration: 'time out of mind',
};
localforage.getItem = () => Promise.resolve(o);
localforage.setItem = (k, v) => Promise.resolve(v);

const te = {
trinity: 'cowboy junkies',
sweet: 'james',
};

const s = await setTokenExpiry(te);
expect(s).toMatchObject({ ...o, tokenExpiration: te });
});

it('does nothing when session does not exist', async () => {
const o = null;
localforage.getItem = () => Promise.resolve(o);
localforage.setItem = (k, v) => Promise.resolve(v);

const te = {
now: 'i am become death, destroyer of sessions',
};

const s = await setTokenExpiry(te);
expect(s).toBeNull();
});
});
});

describe('checkOkapiSession', () => {
it('dispatches setOkapiReady', async () => {
const o = {
margo: 'timmins',
margot: 'margot with a t looks better',
also: 'i thought we were talking about margot robbie?',
tokenExpiration: 'time out of mind',
user: { id: 'id' },
tenant: 'tenant',
isAuthenticated: true,
};
localforage.getItem = jest.fn(() => Promise.resolve(o));
const store = {
dispatch: jest.fn(),
getState: () => ({
okapi: {
currentPerms: [],
}
}),
};
localforage.getItem = () => Promise.resolve(o);
localforage.setItem = (k, v) => Promise.resolve(v);

const te = {
trinity: 'cowboy junkies',
sweet: 'james',
const data = { data: 'd' };

mockFetchSuccess(data);

await checkOkapiSession('url', store, o.tenant);
expect(store.dispatch).toHaveBeenCalledWith(setOkapiReady());

mockFetchCleanUp();
});

it('when getOkapiSession returns full session data, validates it', async () => {
const o = {
user: { id: 'id' },
tenant: 'tenant',
isAuthenticated: true,
};
localforage.getItem = jest.fn(() => Promise.resolve(o));
const store = {
dispatch: jest.fn(),
getState: () => ({
okapi: {
currentPerms: [],
}
}),
};

const data = { data: 'd' };

mockFetchSuccess(data);

await checkOkapiSession('url', store, 'tenant');

expect(store.dispatch).toHaveBeenCalledWith(setAuthError(null));
expect(store.dispatch).toHaveBeenCalledWith(setOkapiReady());

mockFetchCleanUp();
});

it('when getOkapiSession returns sparse session data, ignores it', async () => {
const o = { monkey: 'bagel' };
localforage.getItem = jest.fn(() => Promise.resolve(o));
const store = {
dispatch: jest.fn(),
getState: () => ({
okapi: {
currentPerms: [],
}
}),
};

const s = await setTokenExpiry(te);
expect(s).toMatchObject({ ...o, tokenExpiration: te });
const data = { data: 'd' };

mockFetchSuccess(data);

await checkOkapiSession('url', store, o.tenant);

expect(store.dispatch).not.toHaveBeenCalledWith(setAuthError(null));
expect(store.dispatch).not.toHaveBeenCalledWith(setLoginData(data));
expect(store.dispatch).toHaveBeenCalledWith(setOkapiReady(null));

mockFetchCleanUp();
});
});

0 comments on commit 494bfcd

Please sign in to comment.