mirror of https://github.com/prometheus/prometheus
Move Settings context data into Redux settings slice
Signed-off-by: Julius Volz <julius.volz@gmail.com>mantine-ui-uplot
parent
edf31da311
commit
8fae131733
|
@ -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
|
||||||
|
|
|
@ -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],
|
||||||
|
|
|
@ -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 (
|
||||||
|
|
|
@ -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>
|
||||||
);
|
);
|
||||||
|
|
|
@ -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>
|
||||||
|
|
|
@ -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);
|
||||||
|
|
|
@ -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) {
|
||||||
|
|
|
@ -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);
|
||||||
|
|
|
@ -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}>
|
||||||
|
|
|
@ -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,
|
|
||||||
});
|
|
|
@ -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;
|
||||||
|
|
Loading…
Reference in New Issue