mirror of
https://github.com/grafana/grafana.git
synced 2025-09-25 08:45:13 +08:00

* WIP: Search jaeger traces * Add more customizable query params * Fix failing test * Fix e2e test * Add tags input field * Minor changes * Fix tests * Add docs * Make sure linking is working * Add tests to datasource.ts * Apply suggestions from code review Co-authored-by: achatterjee-grafana <70489351+achatterjee-grafana@users.noreply.github.com> * Update Jaeger doc * UI updates * Address review comments * Add new screenshots to docs * Update placeholder text for tags Co-authored-by: achatterjee-grafana <70489351+achatterjee-grafana@users.noreply.github.com>
110 lines
3.4 KiB
TypeScript
110 lines
3.4 KiB
TypeScript
import { DataFrame, DataSourceInstanceSettings, FieldType, MutableDataFrame, TraceSpanRow } from '@grafana/data';
|
|
import { transformTraceData } from '@jaegertracing/jaeger-ui-components';
|
|
import { Span, TraceProcess, TraceResponse } from './types';
|
|
|
|
export function createTraceFrame(data: TraceResponse): DataFrame {
|
|
const spans = data.spans.map((s) => toSpanRow(s, data.processes));
|
|
|
|
const frame = new MutableDataFrame({
|
|
fields: [
|
|
{ name: 'traceID', type: FieldType.string },
|
|
{ name: 'spanID', type: FieldType.string },
|
|
{ name: 'parentSpanID', type: FieldType.string },
|
|
{ name: 'operationName', type: FieldType.string },
|
|
{ name: 'serviceName', type: FieldType.string },
|
|
{ name: 'serviceTags', type: FieldType.other },
|
|
{ name: 'startTime', type: FieldType.number },
|
|
{ name: 'duration', type: FieldType.number },
|
|
{ name: 'logs', type: FieldType.other },
|
|
{ name: 'tags', type: FieldType.other },
|
|
{ name: 'warnings', type: FieldType.other },
|
|
{ name: 'stackTraces', type: FieldType.other },
|
|
],
|
|
meta: {
|
|
preferredVisualisationType: 'trace',
|
|
},
|
|
});
|
|
|
|
for (const span of spans) {
|
|
frame.add(span);
|
|
}
|
|
|
|
return frame;
|
|
}
|
|
|
|
function toSpanRow(span: Span, processes: Record<string, TraceProcess>): TraceSpanRow {
|
|
return {
|
|
spanID: span.spanID,
|
|
traceID: span.traceID,
|
|
parentSpanID: span.references?.find((r) => r.refType === 'CHILD_OF')?.spanID,
|
|
operationName: span.operationName,
|
|
// from micro to millis
|
|
startTime: span.startTime / 1000,
|
|
duration: span.duration / 1000,
|
|
logs: span.logs.map((l) => ({
|
|
...l,
|
|
timestamp: l.timestamp / 1000,
|
|
})),
|
|
tags: span.tags,
|
|
warnings: span.warnings ?? undefined,
|
|
stackTraces: span.stackTraces,
|
|
serviceName: processes[span.processID].serviceName,
|
|
serviceTags: processes[span.processID].tags,
|
|
};
|
|
}
|
|
|
|
export function createTableFrame(data: TraceResponse[], instanceSettings: DataSourceInstanceSettings): DataFrame {
|
|
const frame = new MutableDataFrame({
|
|
fields: [
|
|
{
|
|
name: 'traceID',
|
|
type: FieldType.string,
|
|
config: {
|
|
displayNameFromDS: 'Trace ID',
|
|
links: [
|
|
{
|
|
title: 'Trace: ${__value.raw}',
|
|
url: '',
|
|
internal: {
|
|
datasourceUid: instanceSettings.uid,
|
|
datasourceName: instanceSettings.name,
|
|
query: {
|
|
query: '${__value.raw}',
|
|
},
|
|
},
|
|
},
|
|
],
|
|
},
|
|
},
|
|
{ name: 'traceName', type: FieldType.string, config: { displayNameFromDS: 'Trace name' } },
|
|
{ name: 'startTime', type: FieldType.time, config: { displayNameFromDS: 'Start time' } },
|
|
{ name: 'duration', type: FieldType.number, config: { displayNameFromDS: 'Duration', unit: 'µs' } },
|
|
],
|
|
meta: {
|
|
preferredVisualisationType: 'table',
|
|
},
|
|
});
|
|
// Show the most recent traces
|
|
const traceData = data.map(transformToTraceData).sort((a, b) => b?.startTime! - a?.startTime!);
|
|
|
|
for (const trace of traceData) {
|
|
frame.add(trace);
|
|
}
|
|
|
|
return frame;
|
|
}
|
|
|
|
function transformToTraceData(data: TraceResponse) {
|
|
const traceData = transformTraceData(data);
|
|
if (!traceData) {
|
|
return;
|
|
}
|
|
|
|
return {
|
|
traceID: traceData.traceID,
|
|
startTime: traceData.startTime / 1000,
|
|
duration: traceData.duration,
|
|
traceName: traceData.traceName,
|
|
};
|
|
}
|