Skip to content

Commit

Permalink
Add app context and revamps App.tsx
Browse files Browse the repository at this point in the history
Signed-off-by: lucferbux <lferrnan@redhat.com>
  • Loading branch information
lucferbux committed Aug 22, 2024
1 parent 82ec37a commit 1ef0d15
Show file tree
Hide file tree
Showing 14 changed files with 476 additions and 42 deletions.
36 changes: 23 additions & 13 deletions clients/ui/frontend/package-lock.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

1 change: 1 addition & 0 deletions clients/ui/frontend/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -86,6 +86,7 @@
"@patternfly/react-core": "6.0.0-alpha.102",
"@patternfly/react-icons": "6.0.0-alpha.35",
"@patternfly/react-styles": "6.0.0-alpha.35",
"lodash-es": "^4.17.21",
"npm-run-all": "^4.1.5",
"react": "^18",
"react-dom": "^18"
Expand Down
100 changes: 86 additions & 14 deletions clients/ui/frontend/src/app/App.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -3,22 +3,88 @@ import '@patternfly/react-core/dist/styles/base.css';
import AppRoutes from './AppRoutes';
import './app.css';
import {
Alert,
Bullseye,
Button,
Flex,
Masthead,
MastheadContent,
MastheadToggle,
Page,
PageSection,
PageToggleButton,
Title
} from '@patternfly/react-core';
import NavSidebar from './NavSidebar';
import { BarsIcon } from '@patternfly/react-icons';
Spinner,
Stack,
StackItem,
Title,
} from "@patternfly/react-core";
import NavSidebar from "./NavSidebar";
import { BarsIcon } from "@patternfly/react-icons";
import { AppContext } from "./AppContext";
import { useSettings } from "./useSettings";

const App: React.FC = () => {
const {
configSettings,
userSettings,
loaded: configLoaded,
loadError: configError,
} = useSettings();

const contextValue = React.useMemo(
() =>
configSettings && userSettings
? {
config: configSettings!,
user: userSettings!,
}
: null,
[configSettings, userSettings]
);

// We lack the critical data to startup the app
if (configError) {
// There was an error fetching critical data
return (
<Page>
<PageSection>
<Stack hasGutter>
<StackItem>
<Alert variant="danger" isInline title="General loading error">
<p>
{configError.message ||
"Unknown error occurred during startup."}
</p>
<p>Logging out and logging back in may solve the issue.</p>
</Alert>
</StackItem>
<StackItem>
<Button
variant="secondary"
onClick={() => {
// TODO: logout
}}
>
Logout
</Button>
</StackItem>
</Stack>
</PageSection>
</Page>
);
}

// Waiting on the API to finish
const loading = !configLoaded || !userSettings || !configSettings || !contextValue;

const masthead = (
<Masthead>
<MastheadToggle>
<PageToggleButton id="page-nav-toggle" variant="plain" aria-label="Dashboard navigation">
<PageToggleButton
id="page-nav-toggle"
variant="plain"
aria-label="Dashboard navigation"
>
<BarsIcon />
</PageToggleButton>
</MastheadToggle>
Expand All @@ -33,15 +99,21 @@ const App: React.FC = () => {
</Masthead>
);

return (
<Page
mainContainerId='primary-app-container'
masthead={masthead}
isManagedSidebar
sidebar={<NavSidebar />}
>
<AppRoutes />
</Page>
return loading ? (
<Bullseye>
<Spinner />
</Bullseye>
) : (
<AppContext.Provider value={contextValue}>
<Page
mainContainerId="primary-app-container"
masthead={masthead}
isManagedSidebar
sidebar={<NavSidebar />}
>
<AppRoutes />
</Page>
</AppContext.Provider>
);
};

Expand Down
13 changes: 13 additions & 0 deletions clients/ui/frontend/src/app/AppContext.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
import * as React from 'react';
import { UserSettings, ConfigSettings } from '~/types';


type AppContextProps = {
config: ConfigSettings;
user: UserSettings;
};

// eslint-disable-next-line @typescript-eslint/consistent-type-assertions
export const AppContext = React.createContext({} as AppContextProps);

export const useAppContext = (): AppContextProps => React.useContext(AppContext);
85 changes: 85 additions & 0 deletions clients/ui/frontend/src/app/useSettings.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,85 @@
import * as React from "react";
import useTimeBasedRefresh from "./useTimeBasedRefresh";
import { ConfigSettings, PlatformType, UserSettings } from "../types";
import { POLL_INTERVAL } from "~/utilities/const";
import { useDeepCompareMemoize } from "../utilities/useDeepCompareMemoize";

export const useSettings = (): {
configSettings: ConfigSettings | null;
userSettings: UserSettings | null;
loaded: boolean;
loadError: Error | undefined;
} => {
const [loaded, setLoaded] = React.useState(false);
const [loadError, setLoadError] = React.useState<Error>();
const [config, setConfig] = React.useState<ConfigSettings | null>(null);
const [user, setUser] = React.useState<UserSettings | null>(null);
const setRefreshMarker = useTimeBasedRefresh();

React.useEffect(() => {
let watchHandle: ReturnType<typeof setTimeout>;
let cancelled = false;
const watchConfig = () => {
Promise.all([fetchConfig(), fetchUser()])
.then(([config, user]) => {
if (cancelled) {
return;
}
setConfig(config);
setUser(user);
setLoaded(true);
setLoadError(undefined);
})
.catch((e) => {
if (e?.message?.includes("Error getting Oauth Info for user")) {
// NOTE: this endpoint only requests oauth because of the security layer, this is not an ironclad use-case
// Something went wrong on the server with the Oauth, let us just log them out and refresh for them
/* eslint-disable-next-line no-console */
console.error(
"Something went wrong with the oauth token, please log out...",
e.message,
e
);
setRefreshMarker(new Date());
return;
}
setLoadError(e);
});
watchHandle = setTimeout(watchConfig, POLL_INTERVAL);
};
watchConfig();

return () => {
cancelled = true;
clearTimeout(watchHandle);
};
}, [setRefreshMarker]);

const retConfig = useDeepCompareMemoize<ConfigSettings | null>(config);
const retUser = useDeepCompareMemoize<UserSettings | null>(user);

return { configSettings: retConfig, userSettings: retUser, loaded, loadError };
};

// Mock a settings config call
// TODO: replace with thea actual call once we have the endpoint
export const fetchConfig = async (): Promise<ConfigSettings> => {
return {
platform: PlatformType.KUBEFLOW,
common: {
featureFlags: {
modelRegistry: true,
},
},
};
};

// Mock a settings user call
// TODO: replace with thea actual call once we have the endpoint
export const fetchUser = async (): Promise<UserSettings> => {
return {
username: "admin",
isAdmin: true,
isAllowed: true,
};
};
48 changes: 48 additions & 0 deletions clients/ui/frontend/src/app/useTimeBasedRefresh.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,48 @@
import * as React from 'react';
import { useBrowserStorage } from '~/components/browserStorage';


export type SetTime = (refreshDateMarker: Date) => void;

const useTimeBasedRefresh = (): SetTime => {
const KEY_NAME = 'kf.dashboard.last.auto.refresh';
const [lastRefreshTimestamp, setLastRefreshTimestamp] = useBrowserStorage(
KEY_NAME,
'0',
false,
true,
);
const ref = React.useRef<{
lastRefreshTimestamp: string;
setLastRefreshTimestamp: (newValue: string) => void;
}>({ lastRefreshTimestamp, setLastRefreshTimestamp });
ref.current = { lastRefreshTimestamp, setLastRefreshTimestamp };

return React.useCallback<SetTime>((refreshDateMarker) => {
// Intentionally avoid referential changes. We want the value at call time.
// Recomputing the ref is not needed and will impact usage in hooks if it does.
const lastDate = new Date(ref.current.lastRefreshTimestamp);
const setNewDateString = ref.current.setLastRefreshTimestamp;

/* eslint-disable no-console */
// Print into the console in case we are not refreshing or the browser has preserve log enabled
console.warn('Attempting to re-trigger an auto refresh');
console.log('Last refresh was on:', lastDate);
console.log('Refreshing requested after:', refreshDateMarker);

lastDate.setHours(lastDate.getHours() + 1);
if (lastDate < refreshDateMarker) {
setNewDateString(refreshDateMarker.toString());
console.log('Logging out and refreshing');
// TODO: Replace with actual logout function
//logout().then(() => window.location.reload());
} else {
console.error(
`We should have refreshed but it appears the last time we auto-refreshed was less than an hour ago. '${KEY_NAME}' session storage setting can be cleared for this to refresh again within the hour from the last refresh.`,
);
}
/* eslint-enable no-console */
}, []);
};

export default useTimeBasedRefresh;
Empty file.
Loading

0 comments on commit 1ef0d15

Please sign in to comment.