mirror of
https://github.com/grafana/grafana.git
synced 2024-12-30 10:47:30 -06:00
Tempo: Fix streaming query restart after Grafana server reboot (#77614)
* Fix streaming query restart after Grafana server reboot * TraceQL Search filter name improvements * Add flag to enable streaming in tempo docker block --------- Co-authored-by: Andrej Ocenas <mr.ocenas@gmail.com>
This commit is contained in:
parent
25779bb6e5
commit
8aa5d470ca
@ -52,3 +52,5 @@ storage:
|
|||||||
|
|
||||||
overrides:
|
overrides:
|
||||||
metrics_generator_processors: [local-blocks, service-graphs, span-metrics]
|
metrics_generator_processors: [local-blocks, service-graphs, span-metrics]
|
||||||
|
|
||||||
|
stream_over_http_enabled: true
|
||||||
|
@ -342,9 +342,9 @@ export const SpanFilters = memo((props: SpanFilterProps) => {
|
|||||||
</InlineFieldRow>
|
</InlineFieldRow>
|
||||||
<InlineFieldRow>
|
<InlineFieldRow>
|
||||||
<InlineField
|
<InlineField
|
||||||
label="Span Duration"
|
label="Duration"
|
||||||
labelWidth={16}
|
labelWidth={16}
|
||||||
tooltip="Filter by span duration. Accepted units are ns, us, ms, s, m, h"
|
tooltip="Filter by duration. Accepted units are ns, us, ms, s, m, h"
|
||||||
>
|
>
|
||||||
<HorizontalGroup spacing="xs" align="flex-start">
|
<HorizontalGroup spacing="xs" align="flex-start">
|
||||||
<Select
|
<Select
|
||||||
|
@ -150,7 +150,7 @@ const TraceQLSearch = ({ datasource, query, onChange }: Props) => {
|
|||||||
/>
|
/>
|
||||||
</InlineSearchField>
|
</InlineSearchField>
|
||||||
<InlineSearchField
|
<InlineSearchField
|
||||||
label={'Duration'}
|
label={'Span Duration'}
|
||||||
tooltip="The span duration, i.e. end - start time of the span. Accepted units are ns, ms, s, m, h"
|
tooltip="The span duration, i.e. end - start time of the span. Accepted units are ns, ms, s, m, h"
|
||||||
>
|
>
|
||||||
<HorizontalGroup spacing={'sm'}>
|
<HorizontalGroup spacing={'sm'}>
|
||||||
|
@ -41,6 +41,10 @@ export const filterTitle = (f: TraceqlFilter) => {
|
|||||||
if (f.tag === 'name') {
|
if (f.tag === 'name') {
|
||||||
return 'Span Name';
|
return 'Span Name';
|
||||||
}
|
}
|
||||||
|
// Special case for the resource service name
|
||||||
|
if (f.tag === 'service.name' && f.scope === TraceqlSearchScope.Resource) {
|
||||||
|
return 'Service Name';
|
||||||
|
}
|
||||||
return startCase(filterScopedTag(f));
|
return startCase(filterScopedTag(f));
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
import { capitalize } from 'lodash';
|
import { capitalize } from 'lodash';
|
||||||
import { map, Observable, defer, mergeMap } from 'rxjs';
|
import { map, Observable, takeWhile } from 'rxjs';
|
||||||
import { v4 as uuidv4 } from 'uuid';
|
import { v4 as uuidv4 } from 'uuid';
|
||||||
|
|
||||||
import {
|
import {
|
||||||
@ -20,7 +20,7 @@ import { SearchStreamingState } from './dataquery.gen';
|
|||||||
import { DEFAULT_SPSS, TempoDatasource } from './datasource';
|
import { DEFAULT_SPSS, TempoDatasource } from './datasource';
|
||||||
import { formatTraceQLResponse } from './resultTransformer';
|
import { formatTraceQLResponse } from './resultTransformer';
|
||||||
import { SearchMetrics, TempoJsonData, TempoQuery } from './types';
|
import { SearchMetrics, TempoJsonData, TempoQuery } from './types';
|
||||||
export async function getLiveStreamKey(): Promise<string> {
|
function getLiveStreamKey(): string {
|
||||||
return uuidv4();
|
return uuidv4();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -34,59 +34,66 @@ export function doTempoChannelStream(
|
|||||||
|
|
||||||
let frames: DataFrame[] | undefined = undefined;
|
let frames: DataFrame[] | undefined = undefined;
|
||||||
let state: LoadingState = LoadingState.NotStarted;
|
let state: LoadingState = LoadingState.NotStarted;
|
||||||
|
const requestTime = performance.now();
|
||||||
|
|
||||||
return defer(() => getLiveStreamKey()).pipe(
|
return getGrafanaLiveSrv()
|
||||||
mergeMap((key) => {
|
.getStream<MutableDataFrame>({
|
||||||
const requestTime = performance.now();
|
scope: LiveChannelScope.DataSource,
|
||||||
return getGrafanaLiveSrv()
|
namespace: ds.uid,
|
||||||
.getStream<MutableDataFrame>({
|
path: `search/${getLiveStreamKey()}`,
|
||||||
scope: LiveChannelScope.DataSource,
|
data: {
|
||||||
namespace: ds.uid,
|
...query,
|
||||||
path: `search/${key}`,
|
SpansPerSpanSet: query.spss ?? DEFAULT_SPSS,
|
||||||
data: {
|
timeRange: {
|
||||||
...query,
|
from: range.from.toISOString(),
|
||||||
SpansPerSpanSet: query.spss ?? DEFAULT_SPSS,
|
to: range.to.toISOString(),
|
||||||
timeRange: {
|
},
|
||||||
from: range.from.toISOString(),
|
},
|
||||||
to: range.to.toISOString(),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
})
|
|
||||||
.pipe(
|
|
||||||
map((evt) => {
|
|
||||||
if ('message' in evt && evt?.message) {
|
|
||||||
const currentTime = performance.now();
|
|
||||||
const elapsedTime = currentTime - requestTime;
|
|
||||||
// Schema should be [traces, metrics, state, error]
|
|
||||||
const traces = evt.message.data.values[0][0];
|
|
||||||
const metrics = evt.message.data.values[1][0];
|
|
||||||
const frameState: SearchStreamingState = evt.message.data.values[2][0];
|
|
||||||
const error = evt.message.data.values[3][0];
|
|
||||||
|
|
||||||
switch (frameState) {
|
|
||||||
case SearchStreamingState.Done:
|
|
||||||
state = LoadingState.Done;
|
|
||||||
break;
|
|
||||||
case SearchStreamingState.Streaming:
|
|
||||||
state = LoadingState.Streaming;
|
|
||||||
break;
|
|
||||||
case SearchStreamingState.Error:
|
|
||||||
throw new Error(error);
|
|
||||||
}
|
|
||||||
|
|
||||||
frames = [
|
|
||||||
metricsDataFrame(metrics, frameState, elapsedTime),
|
|
||||||
...formatTraceQLResponse(traces, instanceSettings, query.tableType),
|
|
||||||
];
|
|
||||||
}
|
|
||||||
return {
|
|
||||||
data: frames || [],
|
|
||||||
state,
|
|
||||||
};
|
|
||||||
})
|
|
||||||
);
|
|
||||||
})
|
})
|
||||||
);
|
.pipe(
|
||||||
|
takeWhile((evt) => {
|
||||||
|
if ('message' in evt && evt?.message) {
|
||||||
|
const frameState: SearchStreamingState = evt.message.data.values[2][0];
|
||||||
|
if (frameState === SearchStreamingState.Done || frameState === SearchStreamingState.Error) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}, true)
|
||||||
|
)
|
||||||
|
.pipe(
|
||||||
|
map((evt) => {
|
||||||
|
if ('message' in evt && evt?.message) {
|
||||||
|
const currentTime = performance.now();
|
||||||
|
const elapsedTime = currentTime - requestTime;
|
||||||
|
// Schema should be [traces, metrics, state, error]
|
||||||
|
const traces = evt.message.data.values[0][0];
|
||||||
|
const metrics = evt.message.data.values[1][0];
|
||||||
|
const frameState: SearchStreamingState = evt.message.data.values[2][0];
|
||||||
|
const error = evt.message.data.values[3][0];
|
||||||
|
|
||||||
|
switch (frameState) {
|
||||||
|
case SearchStreamingState.Done:
|
||||||
|
state = LoadingState.Done;
|
||||||
|
break;
|
||||||
|
case SearchStreamingState.Streaming:
|
||||||
|
state = LoadingState.Streaming;
|
||||||
|
break;
|
||||||
|
case SearchStreamingState.Error:
|
||||||
|
throw new Error(error);
|
||||||
|
}
|
||||||
|
|
||||||
|
frames = [
|
||||||
|
metricsDataFrame(metrics, frameState, elapsedTime),
|
||||||
|
...formatTraceQLResponse(traces, instanceSettings, query.tableType),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
return {
|
||||||
|
data: frames || [],
|
||||||
|
state,
|
||||||
|
};
|
||||||
|
})
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
function metricsDataFrame(metrics: SearchMetrics, state: SearchStreamingState, elapsedTime: number) {
|
function metricsDataFrame(metrics: SearchMetrics, state: SearchStreamingState, elapsedTime: number) {
|
||||||
|
Loading…
Reference in New Issue
Block a user