mirror of
https://github.com/grafana/grafana.git
synced 2025-02-13 00:55:47 -06:00
* Add GraphView component * Add service map panel * Add more metadata visuals * Add context menu on click * Add context menu for services * Fix service map in dashboard * Add field proxy in explore linkSupplier * Refactor the link creation * Remove test file * Fix scale change when view is panned * Fix node centering * Don't show context menu if no links * Fix service map containers * Add collapsible around the service map * Fix stats computation * Remove debug log * Fix time stats * Allow string timestamp * Make panning bounded * Add zooming by mouse wheel * Clean up the colors * Fix stats for single trace graph * Don't show debug config * Add more complex layout * Update layout with better fixing of the root nodes * Code cleanup * Change how we pass in link creation function and some more cleanup * Refactor the panel section into separate render methods * Make the edge hover more readable * Move stats computation to data source * Put edge labels to front * Simplify layout for better multi graph layout * Update for dark theme * Move function to utils * Visual improvements * Improve context menu detail * Allow custom details * Rename to NodeGraph * Remove unused dependencies * Use named color palette and add some fallbacks for missing data * Add test data scenario * Rename plugin * Switch scroll zoom direction to align with google maps * Do some perf optimisations and rise the node limit * Update alert styling * Rename function * Add tests * Add more tests * Change data frame column mapping to use column names * Fix test * Fix type errors * Don't show context menu without links * Add beta status to panel * Fix tests * Changed function to standard methods * Fix typing * Clean up yarn.lock * Add some UI improvements - better styling of the zoom buttons - disable buttons when max reached * Fix panel references after rename * Add panel icon
122 lines
3.6 KiB
TypeScript
122 lines
3.6 KiB
TypeScript
import { useCallback } from 'react';
|
|
import {
|
|
Field,
|
|
LinkModel,
|
|
TimeRange,
|
|
mapInternalLinkToExplore,
|
|
InterpolateFunction,
|
|
ScopedVars,
|
|
DataFrame,
|
|
getFieldDisplayValuesProxy,
|
|
} from '@grafana/data';
|
|
import { getLinkSrv } from '../../panel/panellinks/link_srv';
|
|
import { config, getTemplateSrv } from '@grafana/runtime';
|
|
import { splitOpen } from '../state/main';
|
|
|
|
/**
|
|
* Get links from the field of a dataframe and in addition check if there is associated
|
|
* metadata with datasource in which case we will add onClick to open the link in new split window. This assumes
|
|
* that we just supply datasource name and field value and Explore split window will know how to render that
|
|
* appropriately. This is for example used for transition from log with traceId to trace datasource to show that
|
|
* trace.
|
|
*/
|
|
export const getFieldLinksForExplore = (options: {
|
|
field: Field;
|
|
rowIndex: number;
|
|
splitOpenFn?: typeof splitOpen;
|
|
range: TimeRange;
|
|
vars?: ScopedVars;
|
|
dataFrame?: DataFrame;
|
|
}): Array<LinkModel<Field>> => {
|
|
const { field, vars, splitOpenFn, range, rowIndex, dataFrame } = options;
|
|
const scopedVars: any = { ...(vars || {}) };
|
|
scopedVars['__value'] = {
|
|
value: {
|
|
raw: field.values.get(rowIndex),
|
|
},
|
|
text: 'Raw value',
|
|
};
|
|
|
|
// If we have a dataFrame we can allow referencing other columns and their values in the interpolation.
|
|
if (dataFrame) {
|
|
scopedVars['__data'] = {
|
|
value: {
|
|
name: dataFrame.name,
|
|
refId: dataFrame.refId,
|
|
fields: getFieldDisplayValuesProxy(dataFrame, rowIndex, {
|
|
theme: config.theme,
|
|
}),
|
|
},
|
|
text: 'Data',
|
|
};
|
|
}
|
|
|
|
return field.config.links
|
|
? field.config.links.map(link => {
|
|
if (!link.internal) {
|
|
const replace: InterpolateFunction = (value, vars) =>
|
|
getTemplateSrv().replace(value, { ...vars, ...scopedVars });
|
|
|
|
const linkModel = getLinkSrv().getDataLinkUIModel(link, replace, field);
|
|
if (!linkModel.title) {
|
|
linkModel.title = getTitleFromHref(linkModel.href);
|
|
}
|
|
return linkModel;
|
|
} else {
|
|
return mapInternalLinkToExplore({
|
|
link,
|
|
internalLink: link.internal,
|
|
scopedVars: scopedVars,
|
|
range,
|
|
field,
|
|
onClickFn: splitOpenFn,
|
|
replaceVariables: getTemplateSrv().replace.bind(getTemplateSrv()),
|
|
});
|
|
}
|
|
})
|
|
: [];
|
|
};
|
|
|
|
function getTitleFromHref(href: string): string {
|
|
// The URL constructor needs the url to have protocol
|
|
if (href.indexOf('://') < 0) {
|
|
// Doesn't really matter what protocol we use.
|
|
href = `http://${href}`;
|
|
}
|
|
let title;
|
|
try {
|
|
const parsedUrl = new URL(href);
|
|
title = parsedUrl.hostname;
|
|
} catch (_e) {
|
|
// Should be good enough fallback, user probably did not input valid url.
|
|
title = href;
|
|
}
|
|
return title;
|
|
}
|
|
|
|
/**
|
|
* Hook that returns a function that can be used to retrieve all the links for a row. This returns all the links from
|
|
* all the fields so is useful for visualisation where the whole row is represented as single clickable item like a
|
|
* service map.
|
|
*/
|
|
export function useLinks(range: TimeRange, splitOpenFn?: typeof splitOpen) {
|
|
return useCallback(
|
|
(dataFrame: DataFrame, rowIndex: number) => {
|
|
return dataFrame.fields.flatMap(f => {
|
|
if (f.config?.links && f.config?.links.length) {
|
|
return getFieldLinksForExplore({
|
|
field: f,
|
|
rowIndex: rowIndex,
|
|
range,
|
|
dataFrame,
|
|
splitOpenFn,
|
|
});
|
|
} else {
|
|
return [];
|
|
}
|
|
});
|
|
},
|
|
[range, splitOpenFn]
|
|
);
|
|
}
|