2024-07-30 01:48:18 -07:00
|
|
|
import {
|
|
|
|
FC,
|
|
|
|
ReactNode,
|
|
|
|
useEffect,
|
|
|
|
useId,
|
|
|
|
useLayoutEffect,
|
|
|
|
useState,
|
|
|
|
} from "react";
|
2024-07-09 13:51:37 -07:00
|
|
|
import {
|
|
|
|
Table,
|
|
|
|
Alert,
|
|
|
|
Skeleton,
|
|
|
|
Box,
|
|
|
|
LoadingOverlay,
|
|
|
|
SegmentedControl,
|
|
|
|
ScrollArea,
|
2024-07-10 04:16:15 -07:00
|
|
|
Group,
|
|
|
|
Stack,
|
2024-07-17 02:31:32 -07:00
|
|
|
Text,
|
2024-07-30 01:48:18 -07:00
|
|
|
Anchor,
|
2024-07-09 13:51:37 -07:00
|
|
|
} from "@mantine/core";
|
2024-03-07 04:16:54 -08:00
|
|
|
import { IconAlertTriangle, IconInfoCircle } from "@tabler/icons-react";
|
|
|
|
import {
|
|
|
|
InstantQueryResult,
|
|
|
|
InstantSample,
|
|
|
|
RangeSamples,
|
|
|
|
} from "../../api/responseTypes/query";
|
|
|
|
import SeriesName from "./SeriesName";
|
|
|
|
import { useAPIQuery } from "../../api/api";
|
2024-03-07 07:58:51 -08:00
|
|
|
import classes from "./DataTable.module.css";
|
2024-04-03 05:45:35 -07:00
|
|
|
import dayjs from "dayjs";
|
|
|
|
import timezone from "dayjs/plugin/timezone";
|
|
|
|
import { formatTimestamp } from "../../lib/formatTime";
|
2024-07-09 13:51:37 -07:00
|
|
|
import HistogramChart from "./HistogramChart";
|
|
|
|
import { Histogram } from "../../types/types";
|
|
|
|
import { bucketRangeString } from "./HistogramHelpers";
|
2024-07-15 13:19:47 -07:00
|
|
|
import { useSettings } from "../../state/settingsSlice";
|
2024-07-31 07:19:38 -07:00
|
|
|
import { useAppDispatch, useAppSelector } from "../../state/hooks";
|
|
|
|
import { setVisualizer } from "../../state/queryPageSlice";
|
|
|
|
import TimeInput from "./TimeInput";
|
2024-04-03 05:45:35 -07:00
|
|
|
dayjs.extend(timezone);
|
2024-03-07 04:16:54 -08:00
|
|
|
|
|
|
|
const maxFormattableSeries = 1000;
|
2024-07-31 07:18:23 -07:00
|
|
|
const maxDisplayableSeries = 10000;
|
2024-03-07 04:16:54 -08:00
|
|
|
|
|
|
|
const limitSeries = <S extends InstantSample | RangeSamples>(
|
2024-07-30 01:48:18 -07:00
|
|
|
series: S[],
|
|
|
|
limit: boolean
|
2024-03-07 04:16:54 -08:00
|
|
|
): S[] => {
|
2024-07-30 01:48:18 -07:00
|
|
|
if (limit && series.length > maxDisplayableSeries) {
|
2024-03-07 04:16:54 -08:00
|
|
|
return series.slice(0, maxDisplayableSeries);
|
|
|
|
}
|
|
|
|
return series;
|
|
|
|
};
|
|
|
|
|
2024-03-08 08:43:38 -08:00
|
|
|
export interface DataTableProps {
|
2024-07-31 07:19:38 -07:00
|
|
|
panelIdx: number;
|
2024-03-07 04:16:54 -08:00
|
|
|
retriggerIdx: number;
|
|
|
|
}
|
|
|
|
|
2024-07-31 07:19:38 -07:00
|
|
|
const DataTable: FC<DataTableProps> = ({ panelIdx, retriggerIdx }) => {
|
2024-07-09 13:51:37 -07:00
|
|
|
const [scale, setScale] = useState<string>("exponential");
|
2024-07-30 01:48:18 -07:00
|
|
|
const [limitResults, setLimitResults] = useState<boolean>(true);
|
2024-07-31 07:19:38 -07:00
|
|
|
const [responseTime, setResponseTime] = useState<number>(0);
|
|
|
|
|
|
|
|
const { expr, visualizer } = useAppSelector(
|
|
|
|
(state) => state.queryPage.panels[panelIdx]
|
|
|
|
);
|
|
|
|
const dispatch = useAppDispatch();
|
|
|
|
|
|
|
|
const { endTime, range } = visualizer;
|
2024-07-09 13:51:37 -07:00
|
|
|
|
2024-03-07 04:16:54 -08:00
|
|
|
const { data, error, isFetching, isLoading, refetch } =
|
|
|
|
useAPIQuery<InstantQueryResult>({
|
|
|
|
key: useId(),
|
|
|
|
path: "/query",
|
|
|
|
params: {
|
|
|
|
query: expr,
|
2024-07-31 07:19:38 -07:00
|
|
|
time: `${(endTime !== null ? endTime : Date.now()) / 1000}`,
|
2024-03-07 04:16:54 -08:00
|
|
|
},
|
|
|
|
enabled: expr !== "",
|
2024-07-31 07:19:38 -07:00
|
|
|
recordResponseTime: setResponseTime,
|
2024-03-07 04:16:54 -08:00
|
|
|
});
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
expr !== "" && refetch();
|
2024-07-31 07:19:38 -07:00
|
|
|
}, [retriggerIdx, refetch, expr, endTime]);
|
2024-03-07 04:16:54 -08:00
|
|
|
|
2024-07-30 01:48:18 -07:00
|
|
|
useLayoutEffect(() => {
|
|
|
|
setLimitResults(true);
|
|
|
|
}, [data, isFetching]);
|
|
|
|
|
2024-07-15 13:19:47 -07:00
|
|
|
const { useLocalTime } = useSettings();
|
2024-04-03 05:45:35 -07:00
|
|
|
|
2024-03-07 04:16:54 -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>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (data === undefined) {
|
|
|
|
return <Alert variant="transparent">No data queried yet</Alert>;
|
|
|
|
}
|
|
|
|
|
|
|
|
const { result, resultType } = data.data;
|
|
|
|
|
|
|
|
if (result.length === 0) {
|
|
|
|
return (
|
2024-03-08 08:43:38 -08:00
|
|
|
<Alert title="Empty query result" icon={<IconInfoCircle size={14} />}>
|
2024-03-07 04:16:54 -08:00
|
|
|
This query returned no data.
|
|
|
|
</Alert>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
const doFormat = result.length <= maxFormattableSeries;
|
|
|
|
|
|
|
|
return (
|
2024-07-31 07:19:38 -07:00
|
|
|
<Stack gap="lg" mt="sm">
|
|
|
|
{isLoading ? (
|
|
|
|
<Box>
|
|
|
|
{Array.from(Array(5), (_, i) => (
|
|
|
|
<Skeleton key={i} height={30} mb={15} />
|
|
|
|
))}
|
|
|
|
</Box>
|
|
|
|
) : data === undefined ? (
|
|
|
|
<Alert variant="transparent">No data queried yet</Alert>
|
|
|
|
) : result.length === 0 ? (
|
|
|
|
<Alert title="Empty query result" icon={<IconInfoCircle size={14} />}>
|
|
|
|
This query returned no data.
|
2024-07-30 01:48:18 -07:00
|
|
|
</Alert>
|
2024-07-31 07:19:38 -07:00
|
|
|
) : (
|
|
|
|
<>
|
|
|
|
<Group justify="space-between">
|
|
|
|
<TimeInput
|
|
|
|
time={endTime}
|
|
|
|
range={range}
|
|
|
|
description="Evaluation time"
|
|
|
|
onChangeTime={(time) =>
|
|
|
|
dispatch(
|
|
|
|
setVisualizer({
|
|
|
|
idx: panelIdx,
|
|
|
|
visualizer: { ...visualizer, endTime: time },
|
|
|
|
})
|
|
|
|
)
|
|
|
|
}
|
|
|
|
/>
|
|
|
|
<Text size="xs" c="gray">
|
|
|
|
Load time: {responseTime}ms   Result series: {result.length}
|
|
|
|
</Text>
|
|
|
|
</Group>
|
|
|
|
|
|
|
|
{limitResults &&
|
|
|
|
["vector", "matrix"].includes(resultType) &&
|
|
|
|
result.length > maxDisplayableSeries && (
|
2024-07-30 01:48:18 -07:00
|
|
|
<Alert
|
|
|
|
color="red"
|
|
|
|
icon={<IconAlertTriangle size={14} />}
|
2024-07-31 07:19:38 -07:00
|
|
|
title="Showing limited results"
|
2024-07-30 01:48:18 -07:00
|
|
|
>
|
2024-07-31 07:19:38 -07:00
|
|
|
Fetched {data.data.result.length} metrics, only displaying first{" "}
|
|
|
|
{maxDisplayableSeries} for performance reasons.
|
|
|
|
<Anchor ml="md" fz="1em" onClick={() => setLimitResults(false)}>
|
|
|
|
Show all results
|
|
|
|
</Anchor>
|
2024-07-30 01:48:18 -07:00
|
|
|
</Alert>
|
|
|
|
)}
|
2024-07-31 07:19:38 -07:00
|
|
|
|
|
|
|
{!doFormat && (
|
|
|
|
<Alert
|
|
|
|
title="Formatting turned off"
|
|
|
|
icon={<IconInfoCircle size={14} />}
|
|
|
|
>
|
|
|
|
Showing more than {maxFormattableSeries} series, turning off label
|
|
|
|
formatting for performance reasons.
|
|
|
|
</Alert>
|
|
|
|
)}
|
|
|
|
|
|
|
|
<Box pos="relative" className={classes.tableWrapper}>
|
|
|
|
<LoadingOverlay
|
|
|
|
visible={isFetching}
|
|
|
|
zIndex={1000}
|
|
|
|
overlayProps={{ radius: "sm", blur: 1 }}
|
|
|
|
loaderProps={{
|
|
|
|
children: <Skeleton m={0} w="100%" h="100%" />,
|
|
|
|
}}
|
|
|
|
styles={{ loader: { width: "100%", height: "100%" } }}
|
|
|
|
/>
|
|
|
|
|
|
|
|
<Table fz="xs">
|
|
|
|
<Table.Tbody>
|
|
|
|
{resultType === "vector" ? (
|
|
|
|
limitSeries<InstantSample>(result, limitResults).map(
|
|
|
|
(s, idx) => (
|
|
|
|
<Table.Tr key={idx}>
|
|
|
|
<Table.Td>
|
|
|
|
<SeriesName labels={s.metric} format={doFormat} />
|
|
|
|
</Table.Td>
|
|
|
|
<Table.Td className={classes.numberCell}>
|
|
|
|
{s.value && s.value[1]}
|
|
|
|
{s.histogram && (
|
|
|
|
<Stack>
|
|
|
|
<HistogramChart
|
|
|
|
histogram={s.histogram[1]}
|
|
|
|
index={idx}
|
|
|
|
scale={scale}
|
|
|
|
/>
|
|
|
|
<Group
|
|
|
|
justify="space-between"
|
|
|
|
align="center"
|
|
|
|
p={10}
|
|
|
|
>
|
|
|
|
<Group align="center" gap="1rem">
|
|
|
|
<span>
|
|
|
|
<strong>Count:</strong>{" "}
|
|
|
|
{s.histogram[1].count}
|
|
|
|
</span>
|
|
|
|
<span>
|
|
|
|
<strong>Sum:</strong> {s.histogram[1].sum}
|
|
|
|
</span>
|
|
|
|
</Group>
|
|
|
|
<Group align="center" gap="1rem">
|
|
|
|
<span>x-axis scale:</span>
|
|
|
|
<SegmentedControl
|
|
|
|
size={"xs"}
|
|
|
|
value={scale}
|
|
|
|
onChange={setScale}
|
|
|
|
data={["exponential", "linear"]}
|
|
|
|
/>
|
|
|
|
</Group>
|
|
|
|
</Group>
|
|
|
|
{histogramTable(s.histogram[1])}
|
|
|
|
</Stack>
|
|
|
|
)}
|
|
|
|
</Table.Td>
|
|
|
|
</Table.Tr>
|
|
|
|
)
|
|
|
|
)
|
|
|
|
) : resultType === "matrix" ? (
|
|
|
|
limitSeries<RangeSamples>(result, limitResults).map(
|
|
|
|
(s, idx) => (
|
|
|
|
<Table.Tr key={idx}>
|
|
|
|
<Table.Td>
|
|
|
|
<SeriesName labels={s.metric} format={doFormat} />
|
|
|
|
</Table.Td>
|
|
|
|
<Table.Td className={classes.numberCell}>
|
|
|
|
{s.values &&
|
|
|
|
s.values.map((v, idx) => (
|
|
|
|
<div key={idx}>
|
|
|
|
{v[1]}{" "}
|
|
|
|
<Text
|
|
|
|
span
|
|
|
|
c="gray.7"
|
|
|
|
size="1em"
|
|
|
|
title={formatTimestamp(v[0], useLocalTime)}
|
|
|
|
>
|
|
|
|
@ {v[0]}
|
|
|
|
</Text>
|
|
|
|
</div>
|
|
|
|
))}
|
|
|
|
</Table.Td>
|
|
|
|
</Table.Tr>
|
|
|
|
)
|
|
|
|
)
|
|
|
|
) : resultType === "scalar" ? (
|
|
|
|
<Table.Tr>
|
|
|
|
<Table.Td>Scalar value</Table.Td>
|
|
|
|
<Table.Td className={classes.numberCell}>
|
|
|
|
{result[1]}
|
|
|
|
</Table.Td>
|
|
|
|
</Table.Tr>
|
|
|
|
) : resultType === "string" ? (
|
|
|
|
<Table.Tr>
|
|
|
|
<Table.Td>String value</Table.Td>
|
|
|
|
<Table.Td>{result[1]}</Table.Td>
|
|
|
|
</Table.Tr>
|
|
|
|
) : (
|
|
|
|
<Alert
|
|
|
|
color="red"
|
|
|
|
title="Invalid query response"
|
|
|
|
icon={<IconAlertTriangle size={14} />}
|
|
|
|
>
|
|
|
|
Invalid result value type
|
|
|
|
</Alert>
|
|
|
|
)}
|
|
|
|
</Table.Tbody>
|
|
|
|
</Table>
|
|
|
|
</Box>
|
|
|
|
</>
|
|
|
|
)}
|
|
|
|
</Stack>
|
2024-03-07 04:16:54 -08:00
|
|
|
);
|
|
|
|
};
|
|
|
|
|
2024-07-09 13:51:37 -07:00
|
|
|
const histogramTable = (h: Histogram): ReactNode => (
|
2024-07-10 04:16:15 -07:00
|
|
|
<Table withTableBorder fz="xs">
|
2024-07-09 13:51:37 -07:00
|
|
|
<Table.Tbody
|
|
|
|
style={{
|
|
|
|
display: "flex",
|
|
|
|
flexDirection: "column",
|
|
|
|
justifyContent: "space-between",
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<Table.Tr
|
|
|
|
style={{
|
|
|
|
display: "flex",
|
|
|
|
flexDirection: "row",
|
|
|
|
justifyContent: "space-between",
|
|
|
|
}}
|
|
|
|
>
|
2024-07-10 04:16:15 -07:00
|
|
|
<Table.Th>Bucket range</Table.Th>
|
2024-07-09 13:51:37 -07:00
|
|
|
<Table.Th>Count</Table.Th>
|
|
|
|
</Table.Tr>
|
2024-07-10 04:16:15 -07:00
|
|
|
<ScrollArea w={"100%"} h={265}>
|
2024-07-09 13:51:37 -07:00
|
|
|
{h.buckets?.map((b, i) => (
|
|
|
|
<Table.Tr key={i}>
|
|
|
|
<Table.Td style={{ textAlign: "left" }}>
|
|
|
|
{bucketRangeString(b)}
|
|
|
|
</Table.Td>
|
|
|
|
<Table.Td>{b[3]}</Table.Td>
|
|
|
|
</Table.Tr>
|
|
|
|
))}
|
|
|
|
</ScrollArea>
|
|
|
|
</Table.Tbody>
|
|
|
|
</Table>
|
|
|
|
);
|
|
|
|
|
2024-03-07 04:16:54 -08:00
|
|
|
export default DataTable;
|