2024-07-21 14:55:08 -07:00
|
|
|
import { FC, useEffect, useId, useState } from "react";
|
2024-07-17 12:09:30 -07:00
|
|
|
import { Alert, Skeleton, Box, LoadingOverlay } from "@mantine/core";
|
2024-03-08 08:44:21 -08:00
|
|
|
import { IconAlertTriangle, IconInfoCircle } from "@tabler/icons-react";
|
2024-08-28 05:58:10 -07:00
|
|
|
import {
|
|
|
|
InstantQueryResult,
|
|
|
|
RangeQueryResult,
|
|
|
|
} from "../../api/responseTypes/query";
|
2024-08-16 11:41:18 -07:00
|
|
|
import { SuccessAPIResponse, useAPIQuery } from "../../api/api";
|
2024-07-19 03:13:14 -07:00
|
|
|
import classes from "./Graph.module.css";
|
2024-07-21 14:55:08 -07:00
|
|
|
import {
|
|
|
|
GraphDisplayMode,
|
|
|
|
GraphResolution,
|
|
|
|
getEffectiveResolution,
|
|
|
|
} from "../../state/queryPageSlice";
|
2024-07-19 03:13:14 -07:00
|
|
|
import "uplot/dist/uPlot.min.css";
|
|
|
|
import "./uplot.css";
|
|
|
|
import { useElementSize } from "@mantine/hooks";
|
2024-07-21 14:55:08 -07:00
|
|
|
import UPlotChart, { UPlotChartRange } from "./UPlotChart";
|
2024-03-08 08:44:21 -08:00
|
|
|
|
|
|
|
export interface GraphProps {
|
|
|
|
expr: string;
|
|
|
|
endTime: number | null;
|
|
|
|
range: number;
|
2024-07-21 14:55:08 -07:00
|
|
|
resolution: GraphResolution;
|
2024-03-08 08:44:21 -08:00
|
|
|
showExemplars: boolean;
|
|
|
|
displayMode: GraphDisplayMode;
|
|
|
|
retriggerIdx: number;
|
2024-07-21 09:26:11 -07:00
|
|
|
onSelectRange: (start: number, end: number) => void;
|
2024-03-08 08:44:21 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
const Graph: FC<GraphProps> = ({
|
|
|
|
expr,
|
|
|
|
endTime,
|
|
|
|
range,
|
|
|
|
resolution,
|
2024-07-21 09:26:11 -07:00
|
|
|
showExemplars,
|
2024-07-17 12:09:30 -07:00
|
|
|
displayMode,
|
2024-03-08 08:44:21 -08:00
|
|
|
retriggerIdx,
|
2024-07-21 09:26:11 -07:00
|
|
|
onSelectRange,
|
2024-03-08 08:44:21 -08:00
|
|
|
}) => {
|
2024-07-21 09:26:11 -07:00
|
|
|
const { ref, width } = useElementSize();
|
|
|
|
const [rerender, setRerender] = useState(true);
|
|
|
|
|
|
|
|
const effectiveEndTime = (endTime !== null ? endTime : Date.now()) / 1000;
|
|
|
|
const startTime = effectiveEndTime - range / 1000;
|
2024-07-21 14:55:08 -07:00
|
|
|
const effectiveResolution = getEffectiveResolution(resolution, range) / 1000;
|
2024-07-19 03:13:14 -07:00
|
|
|
|
2024-03-08 08:44:21 -08:00
|
|
|
const { data, error, isFetching, isLoading, refetch } =
|
2024-08-28 05:58:10 -07:00
|
|
|
useAPIQuery<RangeQueryResult>({
|
2024-03-08 08:44:21 -08:00
|
|
|
key: useId(),
|
|
|
|
path: "/query_range",
|
|
|
|
params: {
|
|
|
|
query: expr,
|
2024-07-21 09:26:11 -07:00
|
|
|
step: effectiveResolution.toString(),
|
|
|
|
start: startTime.toString(),
|
|
|
|
end: effectiveEndTime.toString(),
|
2024-03-08 08:44:21 -08:00
|
|
|
},
|
|
|
|
enabled: expr !== "",
|
|
|
|
});
|
|
|
|
|
2024-08-16 11:41:18 -07:00
|
|
|
// Bundle the chart data and the displayed range together. This has two purposes:
|
|
|
|
// 1. If we update them separately, we cause unnecessary rerenders of the uPlot chart itself.
|
|
|
|
// 2. We want to keep displaying the old range in the chart while a query for a new range
|
|
|
|
// is still in progress.
|
|
|
|
const [dataAndRange, setDataAndRange] = useState<{
|
2024-08-28 05:58:10 -07:00
|
|
|
data: SuccessAPIResponse<RangeQueryResult>;
|
2024-08-16 11:41:18 -07:00
|
|
|
range: UPlotChartRange;
|
|
|
|
} | null>(null);
|
2024-07-21 09:26:11 -07:00
|
|
|
|
|
|
|
useEffect(() => {
|
2024-08-16 11:41:18 -07:00
|
|
|
if (data !== undefined) {
|
|
|
|
setDataAndRange({
|
|
|
|
data: data,
|
|
|
|
range: {
|
|
|
|
startTime: startTime,
|
|
|
|
endTime: effectiveEndTime,
|
|
|
|
resolution: effectiveResolution,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
// We actually want to update the displayed range only once the new data is there,
|
|
|
|
// so we don't want to include any of the range-related parameters in the dependencies.
|
2024-07-21 14:55:08 -07:00
|
|
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
2024-07-21 09:26:11 -07:00
|
|
|
}, [data]);
|
|
|
|
|
2024-08-16 11:41:18 -07:00
|
|
|
// Re-execute the query when the user presses Enter (or hits the Execute button).
|
2024-03-08 08:44:21 -08:00
|
|
|
useEffect(() => {
|
|
|
|
expr !== "" && refetch();
|
|
|
|
}, [retriggerIdx, refetch, expr, endTime, range, resolution]);
|
|
|
|
|
2024-08-16 11:41:18 -07:00
|
|
|
// The useElementSize hook above only gets a valid size on the second render, so this
|
|
|
|
// is a workaround to make the component render twice after mount.
|
2024-07-21 09:26:11 -07:00
|
|
|
useEffect(() => {
|
2024-08-16 11:41:18 -07:00
|
|
|
if (dataAndRange !== null && rerender) {
|
2024-07-21 09:26:11 -07:00
|
|
|
setRerender(false);
|
|
|
|
}
|
2024-08-16 11:41:18 -07:00
|
|
|
}, [dataAndRange, rerender, setRerender]);
|
2024-07-21 09:26:11 -07:00
|
|
|
|
2024-08-16 11:41:18 -07:00
|
|
|
// TODO: Share all the loading/error/empty data notices with the DataTable?
|
2024-03-08 08:44:21 -08:00
|
|
|
|
|
|
|
// Show a skeleton only on the first load, not on subsequent ones.
|
|
|
|
if (isLoading) {
|
|
|
|
return (
|
|
|
|
<Box>
|
|
|
|
{Array.from(Array(5), (_, i) => (
|
|
|
|
<Skeleton key={i} height={30} mb={15} />
|
|
|
|
))}
|
|
|
|
</Box>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (error) {
|
|
|
|
return (
|
|
|
|
<Alert
|
|
|
|
color="red"
|
|
|
|
title="Error executing query"
|
|
|
|
icon={<IconAlertTriangle size={14} />}
|
|
|
|
>
|
|
|
|
{error.message}
|
|
|
|
</Alert>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2024-08-16 11:41:18 -07:00
|
|
|
if (dataAndRange === null) {
|
2024-03-08 08:44:21 -08:00
|
|
|
return <Alert variant="transparent">No data queried yet</Alert>;
|
|
|
|
}
|
|
|
|
|
2024-08-28 05:58:10 -07:00
|
|
|
const { result } = dataAndRange.data.data;
|
2024-07-17 12:09:30 -07:00
|
|
|
|
2024-03-08 08:44:21 -08:00
|
|
|
if (result.length === 0) {
|
|
|
|
return (
|
|
|
|
<Alert title="Empty query result" icon={<IconInfoCircle size={14} />}>
|
|
|
|
This query returned no data.
|
|
|
|
</Alert>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
2024-07-19 03:13:14 -07:00
|
|
|
<Box pos="relative" ref={ref} className={classes.chartWrapper}>
|
2024-03-08 08:44:21 -08:00
|
|
|
<LoadingOverlay
|
|
|
|
visible={isFetching}
|
|
|
|
zIndex={1000}
|
2024-07-21 09:26:11 -07:00
|
|
|
h={570}
|
|
|
|
overlayProps={{ radius: "sm", blur: 0.5 }}
|
2024-07-22 16:31:06 -07:00
|
|
|
loaderProps={{ type: "dots", color: "gray.6" }}
|
2024-07-21 09:26:11 -07:00
|
|
|
// loaderProps={{
|
|
|
|
// children: <Skeleton m={0} w="100%" h="100%" />,
|
|
|
|
// }}
|
|
|
|
// styles={{ loader: { width: "100%", height: "100%" } }}
|
2024-03-08 08:44:21 -08:00
|
|
|
/>
|
2024-07-21 09:26:11 -07:00
|
|
|
<UPlotChart
|
2024-08-16 11:41:18 -07:00
|
|
|
data={dataAndRange.data.data.result}
|
|
|
|
range={dataAndRange.range}
|
2024-07-21 09:26:11 -07:00
|
|
|
width={width}
|
|
|
|
showExemplars={showExemplars}
|
|
|
|
displayMode={displayMode}
|
|
|
|
onSelectRange={onSelectRange}
|
2024-07-17 12:09:30 -07:00
|
|
|
/>
|
2024-03-08 08:44:21 -08:00
|
|
|
</Box>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
export default Graph;
|