Move Settings context data into Redux settings slice

Signed-off-by: Julius Volz <julius.volz@gmail.com>
mantine-ui-uplot
Julius Volz 2024-07-15 22:19:47 +02:00
parent edf31da311
commit 8fae131733
11 changed files with 48 additions and 60 deletions

View File

@ -55,13 +55,12 @@ import TSDBStatusPage from "./pages/TSDBStatusPage";
import FlagsPage from "./pages/FlagsPage"; import FlagsPage from "./pages/FlagsPage";
import ConfigPage from "./pages/ConfigPage"; import ConfigPage from "./pages/ConfigPage";
import AgentPage from "./pages/AgentPage"; import AgentPage from "./pages/AgentPage";
import { Suspense, useContext } from "react"; import { Suspense } from "react";
import ErrorBoundary from "./components/ErrorBoundary"; import ErrorBoundary from "./components/ErrorBoundary";
import { ThemeSelector } from "./components/ThemeSelector"; import { ThemeSelector } from "./components/ThemeSelector";
import { SettingsContext } from "./settings";
import { Notifications } from "@mantine/notifications"; import { Notifications } from "@mantine/notifications";
import { useAppDispatch } from "./state/hooks"; import { useAppDispatch } from "./state/hooks";
import { updateSettings } from "./state/settingsSlice"; import { updateSettings, useSettings } from "./state/settingsSlice";
import SettingsMenu from "./components/SettingsMenu"; import SettingsMenu from "./components/SettingsMenu";
const queryClient = new QueryClient(); const queryClient = new QueryClient();
@ -182,12 +181,13 @@ const navLinkXPadding = "md";
function App() { function App() {
const [scroll, scrollTo] = useWindowScroll(); const [scroll, scrollTo] = useWindowScroll();
const [opened, { toggle }] = useDisclosure(); const [opened, { toggle }] = useDisclosure();
const { agentMode } = useContext(SettingsContext);
const pathPrefix = getPathPrefix(window.location.pathname); const pathPrefix = getPathPrefix(window.location.pathname);
const dispatch = useAppDispatch(); const dispatch = useAppDispatch();
dispatch(updateSettings({ pathPrefix })); dispatch(updateSettings({ pathPrefix }));
const { agentMode } = useSettings();
const navLinks = ( const navLinks = (
<> <>
{mainNavPages {mainNavPages

View File

@ -1,5 +1,5 @@
import { useQuery, useSuspenseQuery } from "@tanstack/react-query"; import { useQuery, useSuspenseQuery } from "@tanstack/react-query";
import { useAppSelector } from "../state/hooks"; import { useSettings } from "../state/settingsSlice";
export const API_PATH = "api/v1"; export const API_PATH = "api/v1";
@ -92,7 +92,7 @@ export const useAPIQuery = <T>({
params, params,
enabled, enabled,
}: QueryOptions) => { }: QueryOptions) => {
const pathPrefix = useAppSelector((state) => state.settings.pathPrefix); const { pathPrefix } = useSettings();
return useQuery<SuccessAPIResponse<T>>({ return useQuery<SuccessAPIResponse<T>>({
queryKey: key ? [key] : [path, params], queryKey: key ? [key] : [path, params],
@ -105,7 +105,7 @@ export const useAPIQuery = <T>({
}; };
export const useSuspenseAPIQuery = <T>({ key, path, params }: QueryOptions) => { export const useSuspenseAPIQuery = <T>({ key, path, params }: QueryOptions) => {
const pathPrefix = useAppSelector((state) => state.settings.pathPrefix); const { pathPrefix } = useSettings();
return useSuspenseQuery<SuccessAPIResponse<T>>({ return useSuspenseQuery<SuccessAPIResponse<T>>({
queryKey: key ? [key] : [path, params], queryKey: key ? [key] : [path, params],

View File

@ -1,8 +1,8 @@
import { Popover, ActionIcon, Fieldset, Checkbox, Stack } from "@mantine/core"; import { Popover, ActionIcon, Fieldset, Checkbox, Stack } from "@mantine/core";
import { IconSettings } from "@tabler/icons-react"; import { IconSettings } from "@tabler/icons-react";
import { FC } from "react"; import { FC } from "react";
import { useAppDispatch, useAppSelector } from "../state/hooks"; import { useAppDispatch } from "../state/hooks";
import { updateSettings } from "../state/settingsSlice"; import { updateSettings, useSettings } from "../state/settingsSlice";
const SettingsMenu: FC = () => { const SettingsMenu: FC = () => {
const { const {
@ -12,7 +12,7 @@ const SettingsMenu: FC = () => {
enableSyntaxHighlighting, enableSyntaxHighlighting,
enableLinter, enableLinter,
showAnnotations, showAnnotations,
} = useAppSelector((state) => state.settings); } = useSettings();
const dispatch = useAppDispatch(); const dispatch = useAppDispatch();
return ( return (

View File

@ -1,33 +1,14 @@
import React from "react"; import React from "react";
import ReactDOM from "react-dom/client"; import ReactDOM from "react-dom/client";
import App from "./App.tsx"; import App from "./App.tsx";
import { Settings, SettingsContext } from "./settings.ts";
import store from "./state/store.ts"; import store from "./state/store.ts";
import { Provider } from "react-redux"; import { Provider } from "react-redux";
import "./fonts/codicon.ttf"; import "./fonts/codicon.ttf";
// Declared/defined in public/index.html, value replaced by Prometheus when serving bundle.
declare const GLOBAL_CONSOLES_LINK: string;
declare const GLOBAL_AGENT_MODE: string;
declare const GLOBAL_READY: string;
const settings: Settings = {
consolesLink:
GLOBAL_CONSOLES_LINK === "CONSOLES_LINK_PLACEHOLDER" ||
GLOBAL_CONSOLES_LINK === "" ||
GLOBAL_CONSOLES_LINK === null
? null
: GLOBAL_CONSOLES_LINK,
agentMode: GLOBAL_AGENT_MODE === "true",
ready: GLOBAL_READY === "true",
};
ReactDOM.createRoot(document.getElementById("root")!).render( ReactDOM.createRoot(document.getElementById("root")!).render(
<React.StrictMode> <React.StrictMode>
<SettingsContext.Provider value={settings}> <Provider store={store}>
<Provider store={store}> <App />
<App /> </Provider>
</Provider>
</SettingsContext.Provider>
</React.StrictMode> </React.StrictMode>
); );

View File

@ -22,6 +22,7 @@ import { useAppDispatch, useAppSelector } from "../state/hooks";
import { IconInfoCircle, IconSearch } from "@tabler/icons-react"; import { IconInfoCircle, IconSearch } from "@tabler/icons-react";
import { LabelBadges } from "../components/LabelBadges"; import { LabelBadges } from "../components/LabelBadges";
import { updateAlertFilters } from "../state/alertsPageSlice"; import { updateAlertFilters } from "../state/alertsPageSlice";
import { useSettings } from "../state/settingsSlice";
export default function AlertsPage() { export default function AlertsPage() {
const { data } = useSuspenseAPIQuery<AlertingRulesResult>({ const { data } = useSuspenseAPIQuery<AlertingRulesResult>({
@ -32,9 +33,7 @@ export default function AlertsPage() {
}); });
const dispatch = useAppDispatch(); const dispatch = useAppDispatch();
const showAnnotations = useAppSelector( const { showAnnotations } = useSettings();
(state) => state.settings.showAnnotations
);
const filters = useAppSelector((state) => state.alertsPage.filters); const filters = useAppSelector((state) => state.alertsPage.filters);
const ruleStatsCount = { const ruleStatsCount = {
@ -56,8 +55,8 @@ export default function AlertsPage() {
o === "inactive" o === "inactive"
? badgeClasses.healthOk ? badgeClasses.healthOk
: o === "pending" : o === "pending"
? badgeClasses.healthWarn ? badgeClasses.healthWarn
: badgeClasses.healthErr : badgeClasses.healthErr
} }
placeholder="Filter by alert state" placeholder="Filter by alert state"
values={filters.state} values={filters.state}
@ -121,8 +120,8 @@ export default function AlertsPage() {
numFiring > 0 numFiring > 0
? "5px solid var(--mantine-color-red-4)" ? "5px solid var(--mantine-color-red-4)"
: numPending > 0 : numPending > 0
? "5px solid var(--mantine-color-orange-5)" ? "5px solid var(--mantine-color-orange-5)"
: "5px solid var(--mantine-color-green-4)", : "5px solid var(--mantine-color-green-4)",
}} }}
> >
<Accordion.Control> <Accordion.Control>

View File

@ -1,8 +1,8 @@
import { Stack, Card, Group, Table, Text } from "@mantine/core"; import { Stack, Card, Group, Table, Text } from "@mantine/core";
import { useSuspenseAPIQuery } from "../api/api"; import { useSuspenseAPIQuery } from "../api/api";
import { TSDBStatusResult } from "../api/responseTypes/tsdbStatus"; import { TSDBStatusResult } from "../api/responseTypes/tsdbStatus";
import { useAppSelector } from "../state/hooks";
import { formatTimestamp } from "../lib/formatTime"; import { formatTimestamp } from "../lib/formatTime";
import { useSettings } from "../state/settingsSlice";
export default function TSDBStatusPage() { export default function TSDBStatusPage() {
const { const {
@ -17,7 +17,7 @@ export default function TSDBStatusPage() {
}, },
} = useSuspenseAPIQuery<TSDBStatusResult>({ path: `/status/tsdb` }); } = useSuspenseAPIQuery<TSDBStatusResult>({ path: `/status/tsdb` });
const useLocalTime = useAppSelector((state) => state.settings.useLocalTime); const { useLocalTime } = useSettings();
const unixToTime = (unix: number): string => { const unixToTime = (unix: number): string => {
const formatted = formatTimestamp(unix, useLocalTime); const formatted = formatTimestamp(unix, useLocalTime);

View File

@ -21,11 +21,11 @@ import { useAPIQuery } from "../../api/api";
import classes from "./DataTable.module.css"; import classes from "./DataTable.module.css";
import dayjs from "dayjs"; import dayjs from "dayjs";
import timezone from "dayjs/plugin/timezone"; import timezone from "dayjs/plugin/timezone";
import { useAppSelector } from "../../state/hooks";
import { formatTimestamp } from "../../lib/formatTime"; import { formatTimestamp } from "../../lib/formatTime";
import HistogramChart from "./HistogramChart"; import HistogramChart from "./HistogramChart";
import { Histogram } from "../../types/types"; import { Histogram } from "../../types/types";
import { bucketRangeString } from "./HistogramHelpers"; import { bucketRangeString } from "./HistogramHelpers";
import { useSettings } from "../../state/settingsSlice";
dayjs.extend(timezone); dayjs.extend(timezone);
const maxFormattableSeries = 1000; const maxFormattableSeries = 1000;
@ -64,7 +64,7 @@ const DataTable: FC<DataTableProps> = ({ expr, evalTime, retriggerIdx }) => {
expr !== "" && refetch(); expr !== "" && refetch();
}, [retriggerIdx, refetch, expr, evalTime]); }, [retriggerIdx, refetch, expr, evalTime]);
const useLocalTime = useAppSelector((state) => state.settings.useLocalTime); const { useLocalTime } = useSettings();
// Show a skeleton only on the first load, not on subsequent ones. // Show a skeleton only on the first load, not on subsequent ones.
if (isLoading) { if (isLoading) {

View File

@ -61,7 +61,7 @@ import {
} from "@tabler/icons-react"; } from "@tabler/icons-react";
import { useAPIQuery } from "../../api/api"; import { useAPIQuery } from "../../api/api";
import { notifications } from "@mantine/notifications"; import { notifications } from "@mantine/notifications";
import { useAppSelector } from "../../state/hooks"; import { useSettings } from "../../state/settingsSlice";
const promqlExtension = new PromQLExtension(); const promqlExtension = new PromQLExtension();
@ -126,7 +126,7 @@ const ExpressionInput: FC<ExpressionInputProps> = ({
enableAutocomplete, enableAutocomplete,
enableSyntaxHighlighting, enableSyntaxHighlighting,
enableLinter, enableLinter,
} = useAppSelector((state) => state.settings); } = useSettings();
const [expr, setExpr] = useState(initialExpr); const [expr, setExpr] = useState(initialExpr);
useEffect(() => { useEffect(() => {
setExpr(initialExpr); setExpr(initialExpr);

View File

@ -2,7 +2,7 @@ import { Group, ActionIcon, CloseButton } from "@mantine/core";
import { DatesProvider, DateTimePicker } from "@mantine/dates"; import { DatesProvider, DateTimePicker } from "@mantine/dates";
import { IconChevronLeft, IconChevronRight } from "@tabler/icons-react"; import { IconChevronLeft, IconChevronRight } from "@tabler/icons-react";
import { FC } from "react"; import { FC } from "react";
import { useAppSelector } from "../../state/hooks"; import { useSettings } from "../../state/settingsSlice";
interface TimeInputProps { interface TimeInputProps {
time: number | null; // Timestamp in milliseconds. time: number | null; // Timestamp in milliseconds.
@ -20,7 +20,7 @@ const TimeInput: FC<TimeInputProps> = ({
onChangeTime, onChangeTime,
}) => { }) => {
const baseTime = () => (time !== null ? time : Date.now().valueOf()); const baseTime = () => (time !== null ? time : Date.now().valueOf());
const useLocalTime = useAppSelector((state) => state.settings.useLocalTime); const { useLocalTime } = useSettings();
return ( return (
<Group gap={5}> <Group gap={5}>

View File

@ -1,13 +0,0 @@
import { createContext } from "react";
export interface Settings {
consolesLink: string | null;
agentMode: boolean;
ready: boolean;
}
export const SettingsContext = createContext<Settings>({
consolesLink: null,
agentMode: false,
ready: false,
});

View File

@ -1,6 +1,10 @@
import { PayloadAction, createSlice } from "@reduxjs/toolkit"; import { PayloadAction, createSlice } from "@reduxjs/toolkit";
import { useAppSelector } from "./hooks";
interface Settings { interface Settings {
consolesLink: string | null;
agentMode: boolean;
ready: boolean;
pathPrefix: string; pathPrefix: string;
useLocalTime: boolean; useLocalTime: boolean;
enableQueryHistory: boolean; enableQueryHistory: boolean;
@ -10,7 +14,20 @@ interface Settings {
showAnnotations: boolean; showAnnotations: boolean;
} }
// Declared/defined in public/index.html, value replaced by Prometheus when serving bundle.
declare const GLOBAL_CONSOLES_LINK: string;
declare const GLOBAL_AGENT_MODE: string;
declare const GLOBAL_READY: string;
const initialState: Settings = { const initialState: Settings = {
consolesLink:
GLOBAL_CONSOLES_LINK === "CONSOLES_LINK_PLACEHOLDER" ||
GLOBAL_CONSOLES_LINK === "" ||
GLOBAL_CONSOLES_LINK === null
? null
: GLOBAL_CONSOLES_LINK,
agentMode: GLOBAL_AGENT_MODE === "true",
ready: GLOBAL_READY === "true",
pathPrefix: "", pathPrefix: "",
useLocalTime: false, useLocalTime: false,
enableQueryHistory: false, enableQueryHistory: false,
@ -32,4 +49,8 @@ export const settingsSlice = createSlice({
export const { updateSettings } = settingsSlice.actions; export const { updateSettings } = settingsSlice.actions;
export const useSettings = () => {
return useAppSelector((state) => state.settings);
};
export default settingsSlice.reducer; export default settingsSlice.reducer;