-
Notifications
You must be signed in to change notification settings - Fork 26
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
STCOR-926 validate that cookies and storage are available (#1581)
Detect the presence of localStorage, sessionStorage, and cookies early early early in the stripes-init process and show an error message if any are unavailable. This prevents a white screen of death if, say, session storage is unavailable but we call it anyway, resulting in an untrapped exception (Here's looking at you, OIDCRedirect). Refs STCOR-926
- Loading branch information
Showing
7 changed files
with
192 additions
and
9 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,36 @@ | ||
import { isStorageEnabled } from './App'; | ||
|
||
const storageMock = () => ({ | ||
getItem: () => { | ||
throw new Error(); | ||
}, | ||
}); | ||
|
||
describe('isStorageEnabled', () => { | ||
afterEach(() => { | ||
jest.restoreAllMocks(); | ||
}); | ||
|
||
it('returns true when all storage options are enabled', () => { | ||
expect(isStorageEnabled()).toBeTrue; | ||
}); | ||
|
||
describe('returns false when any storage option is disabled', () => { | ||
it('handles local storage', () => { | ||
Object.defineProperty(window, 'localStorage', { value: storageMock }); | ||
const isEnabled = isStorageEnabled(); | ||
expect(isEnabled).toBeFalse; | ||
}); | ||
it('handles session storage', () => { | ||
Object.defineProperty(window, 'sessionStorage', { value: storageMock }); | ||
const isEnabled = isStorageEnabled(); | ||
expect(isEnabled).toBeFalse; | ||
}); | ||
|
||
it('handles cookies', () => { | ||
jest.spyOn(navigator, 'cookieEnabled', 'get').mockReturnValue(false); | ||
const isEnabled = isStorageEnabled(); | ||
expect(isEnabled).toBeFalse; | ||
}); | ||
}); | ||
}); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,32 @@ | ||
@import "@folio/stripes-components/lib/variables.css"; | ||
|
||
.wrapper { | ||
display: flex; | ||
justify-content: center; | ||
min-height: 100vh; | ||
} | ||
|
||
.container { | ||
width: 100%; | ||
max-width: 940px; | ||
min-height: 330px; | ||
margin: 12vh 2rem 0; | ||
} | ||
|
||
@media (--medium-up) { | ||
.container { | ||
min-height: initial; | ||
} | ||
} | ||
|
||
@media (--large-up) { | ||
.header { | ||
font-size: var(--font-size-xx-large); | ||
} | ||
} | ||
|
||
@media (height <= 440px) { | ||
.container { | ||
min-height: 330px; | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,47 @@ | ||
import { branding } from 'stripes-config'; | ||
|
||
import { | ||
Row, | ||
Col, | ||
Headline, | ||
} from '@folio/stripes-components'; | ||
|
||
import OrganizationLogo from '../OrganizationLogo'; | ||
import styles from './AppConfigError.css'; | ||
|
||
/** | ||
* AppConfigError | ||
* Show an error message. This component is rendered by App, before anything | ||
* else, when it detects that local storage, session storage, or cookies are | ||
* unavailable. This happens _before_ Root has been initialized, i.e. before | ||
* an IntlProvider is available, hence the hard-coded, English-only message. | ||
* | ||
* @returns English-only error message | ||
*/ | ||
const AppConfigError = () => { | ||
return ( | ||
<main> | ||
<div className={styles.wrapper} style={branding?.style?.login ?? {}}> | ||
<div className={styles.container}> | ||
<Row center="xs"> | ||
<Col xs={6}> | ||
<OrganizationLogo /> | ||
</Col> | ||
</Row> | ||
<Row center="xs"> | ||
<Col xs={6}> | ||
<Headline | ||
size="xx-large" | ||
tag="h1" | ||
> | ||
FOLIO requires cookies, sessionStorage, and localStorage. Please enable these features and try again. | ||
</Headline> | ||
</Col> | ||
</Row> | ||
</div> | ||
</div> | ||
</main> | ||
); | ||
}; | ||
|
||
export default AppConfigError; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,12 @@ | ||
import { render, screen } from '@folio/jest-config-stripes/testing-library/react'; | ||
import AppConfigError from './AppConfigError'; | ||
|
||
jest.mock('../OrganizationLogo', () => () => 'OrganizationLogo'); | ||
describe('AppConfigError', () => { | ||
it('displays a warning message', async () => { | ||
render(<AppConfigError />); | ||
|
||
expect(screen.getByText(/cookies/i)).toBeInTheDocument(); | ||
expect(screen.getByText(/storage/i)).toBeInTheDocument(); | ||
}); | ||
}); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
export { default } from './AppConfigError'; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters