feat(auth): render UI only for authenticated users
This commit is contained in:
parent
6f1fd87e90
commit
22c2a4da69
|
@ -3,7 +3,7 @@ import { BrowserRouter, Route, Routes } from 'react-router-dom';
|
||||||
import { ThemeProvider } from '@mui/material';
|
import { ThemeProvider } from '@mui/material';
|
||||||
import { Root } from '../pages/root';
|
import { Root } from '../pages/root';
|
||||||
import { Search } from '../pages/search';
|
import { Search } from '../pages/search';
|
||||||
import { Header } from '../features/auth';
|
import { Header } from '../features/auth/header';
|
||||||
import theme from '../features/theme';
|
import theme from '../features/theme';
|
||||||
|
|
||||||
import './normalize.css';
|
import './normalize.css';
|
||||||
|
|
|
@ -4,14 +4,14 @@ import { Button, IconButton } from '@mui/material';
|
||||||
import SettingsIcon from '@mui/icons-material/Settings';
|
import SettingsIcon from '@mui/icons-material/Settings';
|
||||||
import './header.scss';
|
import './header.scss';
|
||||||
import { useNavigate } from 'react-router-dom';
|
import { useNavigate } from 'react-router-dom';
|
||||||
import useKeycloakAuth from './keycloak';
|
import { useKeycloakAuth } from '../keycloak';
|
||||||
|
|
||||||
const name = cn('Header');
|
const name = cn('Header');
|
||||||
|
|
||||||
export const Header: React.FC = () => {
|
export const Header: React.FC = () => {
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
|
|
||||||
const {authenticated, keycloak} = useKeycloakAuth();
|
const { authenticated, keycloak } = useKeycloakAuth();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<header className={name()}>
|
<header className={name()}>
|
||||||
|
@ -36,7 +36,7 @@ export const Header: React.FC = () => {
|
||||||
Logout
|
Logout
|
||||||
</Button>
|
</Button>
|
||||||
<IconButton
|
<IconButton
|
||||||
sx={{alignSelf: 'center', ml: 2}}
|
sx={{ alignSelf: 'center', ml: 2 }}
|
||||||
onClick={() => document.location = `${process.env.KEYCLOAK_URL}/realms/${process.env.KEYCLOAK_REALM}/account/`}
|
onClick={() => document.location = `${process.env.KEYCLOAK_URL}/realms/${process.env.KEYCLOAK_REALM}/account/`}
|
||||||
>
|
>
|
||||||
<SettingsIcon />
|
<SettingsIcon />
|
||||||
|
@ -56,7 +56,7 @@ export const Header: React.FC = () => {
|
||||||
>
|
>
|
||||||
Registration
|
Registration
|
||||||
</Button>
|
</Button>
|
||||||
</div>}
|
</div>}
|
||||||
</header>
|
</header>
|
||||||
);
|
);
|
||||||
}
|
}
|
|
@ -0,0 +1 @@
|
||||||
|
export * from './header';
|
|
@ -1,39 +1,44 @@
|
||||||
import Keycloak from 'keycloak-js';
|
import Keycloak from 'keycloak-js';
|
||||||
import { useEffect, useState } from 'react';
|
import { useEffect, useState } from 'react';
|
||||||
|
|
||||||
const kc = new Keycloak({
|
const keycloakConfig = {
|
||||||
url: process.env.KEYCLOAK_URL as string,
|
url: process.env.KEYCLOAK_URL as string,
|
||||||
realm: process.env.KEYCLOAK_REALM as string,
|
realm: process.env.KEYCLOAK_REALM as string,
|
||||||
clientId: process.env.KEYCLOAK_CLIENT as string,
|
clientId: process.env.KEYCLOAK_CLIENT as string,
|
||||||
});
|
|
||||||
|
|
||||||
const initKeycloak = () => {
|
|
||||||
return kc.init({
|
|
||||||
onLoad: 'check-sso',
|
|
||||||
silentCheckSsoRedirectUri: window.location.origin + "/silent-check-sso.html",
|
|
||||||
checkLoginIframe: false,
|
|
||||||
pkceMethod: 'S256',
|
|
||||||
});
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export default function useKeycloakAuth() {
|
const kc = new Keycloak(keycloakConfig);
|
||||||
|
|
||||||
|
let keycloakInitialized = false; // Prevent multiple initializations
|
||||||
|
|
||||||
|
const initKeycloak = async () => {
|
||||||
|
if (!keycloakInitialized) {
|
||||||
|
keycloakInitialized = true;
|
||||||
|
return kc.init({
|
||||||
|
onLoad: 'check-sso',
|
||||||
|
silentCheckSsoRedirectUri: window.location.origin + "/silent-check-sso.html",
|
||||||
|
checkLoginIframe: false,
|
||||||
|
pkceMethod: 'S256',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return kc.authenticated ?? false;
|
||||||
|
};
|
||||||
|
|
||||||
|
export function useKeycloakAuth() {
|
||||||
const [authenticated, setAuthenticated] = useState<boolean | null>(null);
|
const [authenticated, setAuthenticated] = useState<boolean | null>(null);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
initKeycloak().then(auth => {
|
initKeycloak().then(auth => setAuthenticated(auth));
|
||||||
setAuthenticated(auth);
|
|
||||||
});
|
|
||||||
|
|
||||||
// Handle token expiration
|
// Ensure event listeners are only set once
|
||||||
if (kc.onTokenExpired) {
|
if (!kc.onTokenExpired) {
|
||||||
kc.onTokenExpired = () => kc.updateToken().then(() => setAuthenticated(true));
|
kc.onTokenExpired = () => kc.updateToken().then(() => setAuthenticated(true));
|
||||||
}
|
}
|
||||||
|
|
||||||
// Listen for authentication events
|
|
||||||
kc.onAuthSuccess = () => setAuthenticated(true);
|
kc.onAuthSuccess = () => setAuthenticated(true);
|
||||||
kc.onAuthError = () => setAuthenticated(false);
|
kc.onAuthError = () => setAuthenticated(false);
|
||||||
kc.onAuthLogout = () => setAuthenticated(false);
|
kc.onAuthLogout = () => setAuthenticated(false);
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
return { authenticated, keycloak: kc };
|
return { authenticated, keycloak: kc };
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1 @@
|
||||||
|
export * from './withAuth';
|
|
@ -0,0 +1,10 @@
|
||||||
|
@use "../../../shared/styles/colors.scss" as *;
|
||||||
|
|
||||||
|
.NotAuthenticated {
|
||||||
|
font-family: Roboto, Helvetica, Arial, sans-serif;
|
||||||
|
font-weight: 400;
|
||||||
|
font-size: 2rem;
|
||||||
|
text-align: center;
|
||||||
|
margin-top: 5rem;
|
||||||
|
color: $text-secondary;
|
||||||
|
}
|
|
@ -0,0 +1,17 @@
|
||||||
|
import { cn } from '@bem-react/classname';
|
||||||
|
import { useKeycloakAuth } from '../keycloak';
|
||||||
|
import React from 'react';
|
||||||
|
|
||||||
|
import './withAuth.scss';
|
||||||
|
|
||||||
|
const name = cn('NotAuthenticated');
|
||||||
|
|
||||||
|
export function withAuth(Cmp: React.FC) {
|
||||||
|
return () => {
|
||||||
|
const { authenticated } = useKeycloakAuth();
|
||||||
|
|
||||||
|
return authenticated ? <Cmp /> : <div className={name()}>
|
||||||
|
Not authenticated
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,8 +1,9 @@
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { CreateRecord } from '../features/createRecord';
|
import { CreateRecord } from '../features/createRecord';
|
||||||
import { Page } from '../shared/components/Page';
|
import { Page } from '../shared/components/Page';
|
||||||
|
import { withAuth } from '../features/auth/withAuth';
|
||||||
|
|
||||||
export const Root: React.FC = () => {
|
export const Root: React.FC = withAuth(() => {
|
||||||
return (
|
return (
|
||||||
<div>
|
<div>
|
||||||
<Page>
|
<Page>
|
||||||
|
@ -10,4 +11,4 @@ export const Root: React.FC = () => {
|
||||||
</Page>
|
</Page>
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
}
|
});
|
||||||
|
|
|
@ -1,9 +1,10 @@
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
|
import { withAuth } from '../features/auth/withAuth';
|
||||||
|
|
||||||
export const Search: React.FC = () => {
|
export const Search: React.FC = withAuth(() => {
|
||||||
return (
|
return (
|
||||||
<div>
|
<div>
|
||||||
Search
|
Search
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
}
|
});
|
||||||
|
|
Loading…
Reference in New Issue