2019-12-23 06:22:54 +01:00
|
|
|
import {
|
|
|
|
|
DynamicConfigValue,
|
|
|
|
|
FieldConfig,
|
|
|
|
|
DataFrame,
|
|
|
|
|
Field,
|
|
|
|
|
FieldType,
|
2019-12-28 17:32:58 -08:00
|
|
|
ThresholdsMode,
|
|
|
|
|
FieldColorMode,
|
|
|
|
|
ColorScheme,
|
2020-02-13 21:37:24 +01:00
|
|
|
FieldOverrideContext,
|
|
|
|
|
ScopedVars,
|
2020-03-16 14:26:03 +01:00
|
|
|
ApplyFieldOverrideOptions,
|
2020-04-07 15:02:08 +02:00
|
|
|
FieldConfigPropertyItem,
|
2020-04-20 07:37:38 +02:00
|
|
|
LinkModel,
|
|
|
|
|
InterpolateFunction,
|
|
|
|
|
ValueLinkConfig,
|
|
|
|
|
GrafanaTheme,
|
2019-12-23 06:22:54 +01:00
|
|
|
} from '../types';
|
2019-12-12 14:55:30 -08:00
|
|
|
import { fieldMatchers, ReducerID, reduceField } from '../transformations';
|
|
|
|
|
import { FieldMatcher } from '../types/transformations';
|
|
|
|
|
import isNumber from 'lodash/isNumber';
|
2020-04-07 15:02:08 +02:00
|
|
|
import set from 'lodash/set';
|
|
|
|
|
import unset from 'lodash/unset';
|
|
|
|
|
import get from 'lodash/get';
|
2019-12-12 14:55:30 -08:00
|
|
|
import { getDisplayProcessor } from './displayProcessor';
|
2020-04-20 07:37:38 +02:00
|
|
|
import { getTimeField, guessFieldTypeForField } from '../dataframe';
|
2020-02-13 21:37:24 +01:00
|
|
|
import { standardFieldConfigEditorRegistry } from './standardFieldConfigEditorRegistry';
|
2020-04-06 16:24:41 +02:00
|
|
|
import { FieldConfigOptionsRegistry } from './FieldConfigOptionsRegistry';
|
2020-04-20 07:37:38 +02:00
|
|
|
import { DataLinkBuiltInVars, locationUtil } from '../utils';
|
|
|
|
|
import { formattedValueToString } from '../valueFormats';
|
|
|
|
|
import { getFieldDisplayValuesProxy } from './getFieldDisplayValuesProxy';
|
2019-12-12 14:55:30 -08:00
|
|
|
|
|
|
|
|
interface OverrideProps {
|
|
|
|
|
match: FieldMatcher;
|
|
|
|
|
properties: DynamicConfigValue[];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
interface GlobalMinMax {
|
|
|
|
|
min: number;
|
|
|
|
|
max: number;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export function findNumericFieldMinMax(data: DataFrame[]): GlobalMinMax {
|
|
|
|
|
let min = Number.MAX_VALUE;
|
|
|
|
|
let max = Number.MIN_VALUE;
|
|
|
|
|
|
|
|
|
|
const reducers = [ReducerID.min, ReducerID.max];
|
|
|
|
|
for (const frame of data) {
|
|
|
|
|
for (const field of frame.fields) {
|
|
|
|
|
if (field.type === FieldType.number) {
|
|
|
|
|
const stats = reduceField({ field, reducers });
|
|
|
|
|
if (stats[ReducerID.min] < min) {
|
|
|
|
|
min = stats[ReducerID.min];
|
|
|
|
|
}
|
|
|
|
|
if (stats[ReducerID.max] > max) {
|
|
|
|
|
max = stats[ReducerID.max];
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return { min, max };
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Return a copy of the DataFrame with all rules applied
|
|
|
|
|
*/
|
2019-12-23 06:22:54 +01:00
|
|
|
export function applyFieldOverrides(options: ApplyFieldOverrideOptions): DataFrame[] {
|
2019-12-13 08:36:49 -08:00
|
|
|
if (!options.data) {
|
|
|
|
|
return [];
|
|
|
|
|
}
|
2019-12-23 06:22:54 +01:00
|
|
|
|
2020-04-06 16:24:41 +02:00
|
|
|
const source = options.fieldConfig;
|
2019-12-12 14:55:30 -08:00
|
|
|
if (!source) {
|
2019-12-13 08:36:49 -08:00
|
|
|
return options.data;
|
2019-12-12 14:55:30 -08:00
|
|
|
}
|
2019-12-23 06:22:54 +01:00
|
|
|
|
2020-04-06 16:24:41 +02:00
|
|
|
const fieldConfigRegistry = options.fieldConfigRegistry ?? standardFieldConfigEditorRegistry;
|
|
|
|
|
|
2019-12-12 14:55:30 -08:00
|
|
|
let range: GlobalMinMax | undefined = undefined;
|
|
|
|
|
|
|
|
|
|
// Prepare the Matchers
|
|
|
|
|
const override: OverrideProps[] = [];
|
|
|
|
|
if (source.overrides) {
|
|
|
|
|
for (const rule of source.overrides) {
|
|
|
|
|
const info = fieldMatchers.get(rule.matcher.id);
|
|
|
|
|
if (info) {
|
|
|
|
|
override.push({
|
2019-12-23 06:22:54 +01:00
|
|
|
match: info.get(rule.matcher.options),
|
2019-12-12 14:55:30 -08:00
|
|
|
properties: rule.properties,
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-12-13 08:36:49 -08:00
|
|
|
return options.data.map((frame, index) => {
|
2019-12-12 14:55:30 -08:00
|
|
|
let name = frame.name;
|
|
|
|
|
if (!name) {
|
|
|
|
|
name = `Series[${index}]`;
|
|
|
|
|
}
|
2020-02-16 14:56:24 +01:00
|
|
|
|
|
|
|
|
const scopedVars: ScopedVars = {
|
|
|
|
|
__series: { text: 'Series', value: { name } },
|
|
|
|
|
};
|
2019-12-12 14:55:30 -08:00
|
|
|
|
2020-02-13 21:37:24 +01:00
|
|
|
const fields: Field[] = frame.fields.map((field, fieldIndex) => {
|
2019-12-12 14:55:30 -08:00
|
|
|
// Config is mutable within this scope
|
2020-02-13 21:37:24 +01:00
|
|
|
let fieldName = field.name;
|
|
|
|
|
if (!fieldName) {
|
|
|
|
|
fieldName = `Field[${fieldIndex}]`;
|
2019-12-12 14:55:30 -08:00
|
|
|
}
|
2020-04-20 07:37:38 +02:00
|
|
|
const fieldScopedVars = { ...scopedVars };
|
|
|
|
|
fieldScopedVars['__field'] = { text: 'Field', value: { name: fieldName } };
|
2019-12-12 14:55:30 -08:00
|
|
|
|
2020-04-20 07:37:38 +02:00
|
|
|
const config: FieldConfig = { ...field.config, scopedVars: fieldScopedVars } || {};
|
2020-02-13 21:37:24 +01:00
|
|
|
const context = {
|
|
|
|
|
field,
|
|
|
|
|
data: options.data!,
|
|
|
|
|
dataFrameIndex: index,
|
|
|
|
|
replaceVariables: options.replaceVariables,
|
2020-04-06 16:24:41 +02:00
|
|
|
fieldConfigRegistry: fieldConfigRegistry,
|
2020-02-13 21:37:24 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// Anything in the field config that's not set by the datasource
|
|
|
|
|
// will be filled in by panel's field configuration
|
|
|
|
|
setFieldConfigDefaults(config, source.defaults, context);
|
|
|
|
|
|
2019-12-12 14:55:30 -08:00
|
|
|
// Find any matching rules and then override
|
|
|
|
|
for (const rule of override) {
|
|
|
|
|
if (rule.match(field)) {
|
|
|
|
|
for (const prop of rule.properties) {
|
2020-02-13 21:37:24 +01:00
|
|
|
// config.scopedVars is set already here
|
|
|
|
|
setDynamicConfigValue(config, prop, context);
|
2019-12-12 14:55:30 -08:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-12-28 17:32:58 -08:00
|
|
|
// Try harder to set a real value that is not 'other'
|
|
|
|
|
let type = field.type;
|
|
|
|
|
if (!type || type === FieldType.other) {
|
|
|
|
|
const t = guessFieldTypeForField(field);
|
|
|
|
|
if (t) {
|
|
|
|
|
type = t;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Some units have an implied range
|
|
|
|
|
if (config.unit === 'percent') {
|
|
|
|
|
if (!isNumber(config.min)) {
|
|
|
|
|
config.min = 0;
|
|
|
|
|
}
|
|
|
|
|
if (!isNumber(config.max)) {
|
|
|
|
|
config.max = 100;
|
|
|
|
|
}
|
|
|
|
|
} else if (config.unit === 'percentunit') {
|
|
|
|
|
if (!isNumber(config.min)) {
|
|
|
|
|
config.min = 0;
|
|
|
|
|
}
|
|
|
|
|
if (!isNumber(config.max)) {
|
|
|
|
|
config.max = 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-12-12 14:55:30 -08:00
|
|
|
// Set the Min/Max value automatically
|
2019-12-13 08:36:49 -08:00
|
|
|
if (options.autoMinMax && field.type === FieldType.number) {
|
2019-12-12 14:55:30 -08:00
|
|
|
if (!isNumber(config.min) || !isNumber(config.max)) {
|
|
|
|
|
if (!range) {
|
2019-12-13 08:36:49 -08:00
|
|
|
range = findNumericFieldMinMax(options.data!); // Global value
|
2019-12-12 14:55:30 -08:00
|
|
|
}
|
|
|
|
|
if (!isNumber(config.min)) {
|
|
|
|
|
config.min = range.min;
|
|
|
|
|
}
|
|
|
|
|
if (!isNumber(config.max)) {
|
|
|
|
|
config.max = range.max;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-12-28 17:32:58 -08:00
|
|
|
// Overwrite the configs
|
|
|
|
|
const f: Field = {
|
2019-12-12 14:55:30 -08:00
|
|
|
...field,
|
|
|
|
|
config,
|
2019-12-28 17:32:58 -08:00
|
|
|
type,
|
2019-12-12 14:55:30 -08:00
|
|
|
};
|
2020-02-16 14:56:24 +01:00
|
|
|
|
2019-12-28 17:32:58 -08:00
|
|
|
// and set the display processor using it
|
2020-01-15 12:02:52 -08:00
|
|
|
f.display = getDisplayProcessor({
|
|
|
|
|
field: f,
|
|
|
|
|
theme: options.theme,
|
|
|
|
|
timeZone: options.timeZone,
|
|
|
|
|
});
|
2020-04-20 07:37:38 +02:00
|
|
|
|
|
|
|
|
// Attach data links supplier
|
|
|
|
|
f.getLinks = getLinksSupplier(frame, f, fieldScopedVars, context.replaceVariables, {
|
|
|
|
|
theme: options.theme,
|
|
|
|
|
});
|
|
|
|
|
|
2019-12-28 17:32:58 -08:00
|
|
|
return f;
|
2019-12-12 14:55:30 -08:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
...frame,
|
|
|
|
|
fields,
|
|
|
|
|
name,
|
|
|
|
|
};
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-13 21:37:24 +01:00
|
|
|
export interface FieldOverrideEnv extends FieldOverrideContext {
|
2020-04-06 16:24:41 +02:00
|
|
|
fieldConfigRegistry: FieldConfigOptionsRegistry;
|
2019-12-12 14:55:30 -08:00
|
|
|
}
|
|
|
|
|
|
2020-04-07 15:02:08 +02:00
|
|
|
export function setDynamicConfigValue(config: FieldConfig, value: DynamicConfigValue, context: FieldOverrideEnv) {
|
2020-04-06 16:24:41 +02:00
|
|
|
const reg = context.fieldConfigRegistry;
|
|
|
|
|
const item = reg.getIfExists(value.id);
|
2020-02-13 21:37:24 +01:00
|
|
|
if (!item || !item.shouldApply(context.field!)) {
|
|
|
|
|
return;
|
2019-12-12 14:55:30 -08:00
|
|
|
}
|
|
|
|
|
|
2020-02-13 21:37:24 +01:00
|
|
|
const val = item.process(value.value, context, item.settings);
|
|
|
|
|
|
|
|
|
|
const remove = val === undefined || val === null;
|
|
|
|
|
|
|
|
|
|
if (remove) {
|
2020-04-07 15:02:08 +02:00
|
|
|
if (item.isCustom && config.custom) {
|
|
|
|
|
unset(config.custom, item.path);
|
2020-02-13 21:37:24 +01:00
|
|
|
} else {
|
2020-04-07 15:02:08 +02:00
|
|
|
unset(config, item.path);
|
2019-12-12 14:55:30 -08:00
|
|
|
}
|
2020-02-13 21:37:24 +01:00
|
|
|
} else {
|
2020-04-07 15:02:08 +02:00
|
|
|
if (item.isCustom) {
|
2020-02-13 21:37:24 +01:00
|
|
|
if (!config.custom) {
|
|
|
|
|
config.custom = {};
|
|
|
|
|
}
|
2020-04-07 15:02:08 +02:00
|
|
|
set(config.custom, item.path, val);
|
2020-02-13 21:37:24 +01:00
|
|
|
} else {
|
2020-04-07 15:02:08 +02:00
|
|
|
set(config, item.path, val);
|
2019-12-12 14:55:30 -08:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-13 21:37:24 +01:00
|
|
|
// config -> from DS
|
|
|
|
|
// defaults -> from Panel config
|
|
|
|
|
export function setFieldConfigDefaults(config: FieldConfig, defaults: FieldConfig, context: FieldOverrideEnv) {
|
2020-04-07 15:02:08 +02:00
|
|
|
for (const fieldConfigProperty of context.fieldConfigRegistry.list()) {
|
|
|
|
|
if (fieldConfigProperty.isCustom && !config.custom) {
|
|
|
|
|
config.custom = {};
|
2019-12-12 14:55:30 -08:00
|
|
|
}
|
2020-04-07 15:02:08 +02:00
|
|
|
processFieldConfigValue(
|
|
|
|
|
fieldConfigProperty.isCustom ? config.custom : config,
|
|
|
|
|
fieldConfigProperty.isCustom ? defaults.custom : defaults,
|
|
|
|
|
fieldConfigProperty,
|
|
|
|
|
context
|
|
|
|
|
);
|
2019-12-12 14:55:30 -08:00
|
|
|
}
|
2020-04-06 16:24:41 +02:00
|
|
|
|
2019-12-28 17:32:58 -08:00
|
|
|
validateFieldConfig(config);
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-13 21:37:24 +01:00
|
|
|
const processFieldConfigValue = (
|
|
|
|
|
destination: Record<string, any>, // it's mutable
|
|
|
|
|
source: Record<string, any>,
|
2020-04-07 15:02:08 +02:00
|
|
|
fieldConfigProperty: FieldConfigPropertyItem,
|
2020-04-06 16:24:41 +02:00
|
|
|
context: FieldOverrideEnv
|
2020-02-13 21:37:24 +01:00
|
|
|
) => {
|
2020-04-07 15:02:08 +02:00
|
|
|
const currentConfig = get(destination, fieldConfigProperty.path);
|
|
|
|
|
|
2020-02-13 21:37:24 +01:00
|
|
|
if (currentConfig === null || currentConfig === undefined) {
|
2020-04-07 15:02:08 +02:00
|
|
|
const item = context.fieldConfigRegistry.getIfExists(fieldConfigProperty.id);
|
2020-03-09 15:09:32 +01:00
|
|
|
if (!item) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-13 21:37:24 +01:00
|
|
|
if (item && item.shouldApply(context.field!)) {
|
2020-04-07 15:02:08 +02:00
|
|
|
const val = item.process(get(source, item.path), context, item.settings);
|
2020-02-13 21:37:24 +01:00
|
|
|
if (val !== undefined && val !== null) {
|
2020-04-07 15:02:08 +02:00
|
|
|
set(destination, item.path, val);
|
2020-02-13 21:37:24 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2019-12-28 17:32:58 -08:00
|
|
|
/**
|
|
|
|
|
* This checks that all options on FieldConfig make sense. It mutates any value that needs
|
|
|
|
|
* fixed. In particular this makes sure that the first threshold value is -Infinity (not valid in JSON)
|
|
|
|
|
*/
|
|
|
|
|
export function validateFieldConfig(config: FieldConfig) {
|
|
|
|
|
const { thresholds } = config;
|
|
|
|
|
if (thresholds) {
|
|
|
|
|
if (!thresholds.mode) {
|
|
|
|
|
thresholds.mode = ThresholdsMode.Absolute;
|
|
|
|
|
}
|
|
|
|
|
if (!thresholds.steps) {
|
|
|
|
|
thresholds.steps = [];
|
|
|
|
|
} else if (thresholds.steps.length) {
|
|
|
|
|
// First value is always -Infinity
|
|
|
|
|
// JSON saves it as null
|
|
|
|
|
thresholds.steps[0].value = -Infinity;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!config.color) {
|
|
|
|
|
if (thresholds) {
|
|
|
|
|
config.color = {
|
|
|
|
|
mode: FieldColorMode.Thresholds,
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
// No Color settings
|
|
|
|
|
} else if (!config.color.mode) {
|
|
|
|
|
// Without a mode, skip color altogether
|
|
|
|
|
delete config.color;
|
|
|
|
|
} else {
|
|
|
|
|
const { color } = config;
|
|
|
|
|
if (color.mode === FieldColorMode.Scheme) {
|
|
|
|
|
if (!color.schemeName) {
|
|
|
|
|
color.schemeName = ColorScheme.BrBG;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
delete color.schemeName;
|
|
|
|
|
}
|
2019-12-12 14:55:30 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Verify that max > min (swap if necessary)
|
|
|
|
|
if (config.hasOwnProperty('min') && config.hasOwnProperty('max') && config.min! > config.max!) {
|
|
|
|
|
const tmp = config.max;
|
|
|
|
|
config.max = config.min;
|
|
|
|
|
config.min = tmp;
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-04-20 07:37:38 +02:00
|
|
|
|
|
|
|
|
const getLinksSupplier = (
|
|
|
|
|
frame: DataFrame,
|
|
|
|
|
field: Field,
|
|
|
|
|
fieldScopedVars: ScopedVars,
|
|
|
|
|
replaceVariables: InterpolateFunction,
|
|
|
|
|
options: {
|
|
|
|
|
theme: GrafanaTheme;
|
|
|
|
|
}
|
|
|
|
|
) => (config: ValueLinkConfig): Array<LinkModel<Field>> => {
|
|
|
|
|
if (!field.config.links || field.config.links.length === 0) {
|
|
|
|
|
return [];
|
|
|
|
|
}
|
|
|
|
|
const timeRangeUrl = locationUtil.getTimeRangeUrlParams();
|
|
|
|
|
const { timeField } = getTimeField(frame);
|
|
|
|
|
|
|
|
|
|
return field.config.links.map(link => {
|
|
|
|
|
let href = link.url;
|
|
|
|
|
let dataFrameVars = {};
|
|
|
|
|
let valueVars = {};
|
|
|
|
|
|
|
|
|
|
const info: LinkModel<Field> = {
|
|
|
|
|
href: locationUtil.assureBaseUrl(href.replace(/\n/g, '')),
|
|
|
|
|
title: replaceVariables(link.title || ''),
|
|
|
|
|
target: link.targetBlank ? '_blank' : '_self',
|
|
|
|
|
origin: field,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const variablesQuery = locationUtil.getVariablesUrlParams();
|
|
|
|
|
|
|
|
|
|
// We are not displaying reduction result
|
|
|
|
|
if (config.valueRowIndex !== undefined && !isNaN(config.valueRowIndex)) {
|
|
|
|
|
const fieldsProxy = getFieldDisplayValuesProxy(frame, config.valueRowIndex, options);
|
|
|
|
|
valueVars = {
|
|
|
|
|
raw: field.values.get(config.valueRowIndex),
|
|
|
|
|
numeric: fieldsProxy[field.name].numeric,
|
|
|
|
|
text: fieldsProxy[field.name].text,
|
|
|
|
|
time: timeField ? timeField.values.get(config.valueRowIndex) : undefined,
|
|
|
|
|
};
|
|
|
|
|
dataFrameVars = {
|
|
|
|
|
__data: {
|
|
|
|
|
value: {
|
|
|
|
|
name: frame.name,
|
|
|
|
|
refId: frame.refId,
|
|
|
|
|
fields: fieldsProxy,
|
|
|
|
|
},
|
|
|
|
|
text: 'Data',
|
|
|
|
|
},
|
|
|
|
|
};
|
|
|
|
|
} else {
|
|
|
|
|
if (config.calculatedValue) {
|
|
|
|
|
valueVars = {
|
|
|
|
|
raw: config.calculatedValue.numeric,
|
|
|
|
|
numeric: config.calculatedValue.numeric,
|
|
|
|
|
text: formattedValueToString(config.calculatedValue),
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
info.href = replaceVariables(info.href, {
|
|
|
|
|
...fieldScopedVars,
|
|
|
|
|
__value: {
|
|
|
|
|
text: 'Value',
|
|
|
|
|
value: valueVars,
|
|
|
|
|
},
|
|
|
|
|
...dataFrameVars,
|
|
|
|
|
[DataLinkBuiltInVars.keepTime]: {
|
|
|
|
|
text: timeRangeUrl,
|
|
|
|
|
value: timeRangeUrl,
|
|
|
|
|
},
|
|
|
|
|
[DataLinkBuiltInVars.includeVars]: {
|
|
|
|
|
text: variablesQuery,
|
|
|
|
|
value: variablesQuery,
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
info.href = locationUtil.processUrl(info.href);
|
|
|
|
|
|
|
|
|
|
return info;
|
|
|
|
|
});
|
|
|
|
|
};
|