grafana/public/app/features/explore/LogsVolumePanel.tsx
Sven Grossmann 0bd326d846
LogsVolume: Make log volume work with chunking (#63181)
* aggregate logs volume on streaming

* enable log volume queries to be split

* add streaming indicator

* added tests

* also chunk logs queries

* change extraInfo back to undefined
2023-02-10 14:43:41 +01:00

148 lines
4.5 KiB
TypeScript

import { css } from '@emotion/css';
import React from 'react';
import {
AbsoluteTimeRange,
DataQueryResponse,
GrafanaTheme2,
LoadingState,
SplitOpen,
TimeZone,
EventBus,
LogsVolumeType,
} from '@grafana/data';
import { Button, Collapse, Icon, InlineField, Tooltip, TooltipDisplayMode, useStyles2, useTheme2 } from '@grafana/ui';
import { ExploreGraph } from './Graph/ExploreGraph';
import { SupplementaryResultError } from './SupplementaryResultError';
type Props = {
logsVolumeData: DataQueryResponse | undefined;
absoluteRange: AbsoluteTimeRange;
logLinesBasedData: DataQueryResponse | undefined;
logLinesBasedDataVisibleRange: AbsoluteTimeRange | undefined;
timeZone: TimeZone;
splitOpen: SplitOpen;
width: number;
onUpdateTimeRange: (timeRange: AbsoluteTimeRange) => void;
onLoadLogsVolume: () => void;
onHiddenSeriesChanged: (hiddenSeries: string[]) => void;
eventBus: EventBus;
};
export function LogsVolumePanel(props: Props) {
const { width, timeZone, splitOpen, onUpdateTimeRange, onLoadLogsVolume, onHiddenSeriesChanged } = props;
const theme = useTheme2();
const styles = useStyles2(getStyles);
const spacing = parseInt(theme.spacing(2).slice(0, -2), 10);
const height = 150;
if (props.logsVolumeData === undefined) {
return null;
}
const logsVolumeData = props.logsVolumeData;
const range = logsVolumeData.data[0]?.meta?.custom?.absoluteRange || props.absoluteRange;
if (logsVolumeData.error !== undefined) {
return <SupplementaryResultError error={logsVolumeData.error} title="Failed to load log volume for this query" />;
}
let LogsVolumePanelContent;
if (logsVolumeData?.state === LoadingState.Loading) {
LogsVolumePanelContent = <span>Log volume is loading...</span>;
} else if (logsVolumeData?.data) {
if (logsVolumeData.data.length > 0) {
LogsVolumePanelContent = (
<ExploreGraph
graphStyle="lines"
loadingState={logsVolumeData.state ?? LoadingState.Done}
data={logsVolumeData.data}
height={height}
width={width - spacing * 2}
absoluteRange={range}
onChangeTime={onUpdateTimeRange}
timeZone={timeZone}
splitOpenFn={splitOpen}
tooltipDisplayMode={TooltipDisplayMode.Multi}
onHiddenSeriesChanged={onHiddenSeriesChanged}
anchorToZero
eventBus={props.eventBus}
/>
);
} else {
LogsVolumePanelContent = <span>No volume data.</span>;
}
}
let extraInfo;
if (logsVolumeData.data[0]?.meta?.custom?.logsVolumeType !== LogsVolumeType.Limited) {
const zoomRatio = logsLevelZoomRatio(logsVolumeData, range);
if (zoomRatio !== undefined && zoomRatio < 1) {
extraInfo = (
<InlineField label="Reload log volume" transparent>
<Button size="xs" icon="sync" variant="secondary" onClick={onLoadLogsVolume} id="reload-volume" />
</InlineField>
);
}
} else {
extraInfo = (
<div className={styles.oldInfoText}>
This datasource does not support full-range histograms. The graph is based on the logs seen in the response.
</div>
);
}
if (logsVolumeData.state === LoadingState.Streaming) {
extraInfo = (
<>
{extraInfo}
<Tooltip content="Streaming">
<Icon name="circle-mono" size="md" className={styles.streaming} data-testid="logs-volume-streaming" />
</Tooltip>
</>
);
}
return (
<Collapse label="" isOpen={true}>
<div style={{ height }} className={styles.contentContainer}>
{LogsVolumePanelContent}
</div>
<div className={styles.extraInfoContainer}>{extraInfo}</div>
</Collapse>
);
}
const getStyles = (theme: GrafanaTheme2) => {
return {
extraInfoContainer: css`
display: flex;
justify-content: end;
position: absolute;
right: 5px;
top: 5px;
`,
contentContainer: css`
display: flex;
align-items: center;
justify-content: center;
`,
oldInfoText: css`
font-size: ${theme.typography.size.sm};
color: ${theme.colors.text.secondary};
`,
streaming: css`
color: ${theme.colors.success.text};
`,
};
};
function logsLevelZoomRatio(
logsVolumeData: DataQueryResponse | undefined,
selectedTimeRange: AbsoluteTimeRange
): number | undefined {
const dataRange = logsVolumeData && logsVolumeData.data[0] && logsVolumeData.data[0].meta?.custom?.absoluteRange;
return dataRange ? (selectedTimeRange.from - selectedTimeRange.to) / (dataRange.from - dataRange.to) : undefined;
}