2022-04-22 14:33:13 +01:00
|
|
|
import { isNumber, set, unset, get, cloneDeep } from 'lodash';
|
2023-01-02 02:27:48 -08:00
|
|
|
import { useMemo, useRef } from 'react';
|
|
|
|
|
import usePrevious from 'react-use/lib/usePrevious';
|
2022-04-22 14:33:13 +01:00
|
|
|
|
2023-03-17 06:51:06 -07:00
|
|
|
import { VariableFormatID } from '@grafana/schema';
|
|
|
|
|
|
2023-01-02 02:27:48 -08:00
|
|
|
import { compareArrayValues, compareDataFrameStructures, guessFieldTypeForField } from '../dataframe';
|
|
|
|
|
import { PanelPlugin } from '../panel/PanelPlugin';
|
|
|
|
|
import { GrafanaTheme2 } from '../themes';
|
2022-04-22 14:33:13 +01:00
|
|
|
import { asHexString } from '../themes/colorManipulator';
|
|
|
|
|
import { fieldMatchers, reduceField, ReducerID } from '../transformations';
|
2019-12-23 06:22:54 +01:00
|
|
|
import {
|
2020-09-04 11:21:24 +02:00
|
|
|
ApplyFieldOverrideOptions,
|
2023-03-28 19:22:34 +02:00
|
|
|
DataContextScopedVar,
|
2019-12-23 06:22:54 +01:00
|
|
|
DataFrame,
|
2022-04-20 10:15:36 +01:00
|
|
|
DataLink,
|
2022-07-28 01:58:42 -05:00
|
|
|
DecimalCount,
|
2021-06-02 11:01:23 -07:00
|
|
|
DisplayProcessor,
|
|
|
|
|
DisplayValue,
|
2020-09-04 11:21:24 +02:00
|
|
|
DynamicConfigValue,
|
2019-12-23 06:22:54 +01:00
|
|
|
Field,
|
2020-10-09 09:34:57 +02:00
|
|
|
FieldColorModeId,
|
2020-09-04 11:21:24 +02:00
|
|
|
FieldConfig,
|
2020-04-07 15:02:08 +02:00
|
|
|
FieldConfigPropertyItem,
|
2023-01-02 02:27:48 -08:00
|
|
|
FieldConfigSource,
|
2020-09-04 11:21:24 +02:00
|
|
|
FieldOverrideContext,
|
|
|
|
|
FieldType,
|
|
|
|
|
InterpolateFunction,
|
|
|
|
|
LinkModel,
|
2020-11-29 09:22:16 -08:00
|
|
|
NumericRange,
|
2023-01-02 02:27:48 -08:00
|
|
|
PanelData,
|
2020-09-04 11:21:24 +02:00
|
|
|
ScopedVars,
|
2020-04-27 15:28:06 +02:00
|
|
|
TimeZone,
|
2020-09-04 11:21:24 +02:00
|
|
|
ValueLinkConfig,
|
2019-12-23 06:22:54 +01:00
|
|
|
} from '../types';
|
2019-12-12 14:55:30 -08:00
|
|
|
import { FieldMatcher } from '../types/transformations';
|
2023-03-28 19:22:34 +02:00
|
|
|
import { locationUtil } from '../utils';
|
2022-04-22 14:33:13 +01:00
|
|
|
import { mapInternalLinkToExplore } from '../utils/dataLinks';
|
|
|
|
|
|
|
|
|
|
import { FieldConfigOptionsRegistry } from './FieldConfigOptionsRegistry';
|
|
|
|
|
import { getDisplayProcessor, getRawDisplayProcessor } from './displayProcessor';
|
|
|
|
|
import { standardFieldConfigEditorRegistry } from './standardFieldConfigEditorRegistry';
|
2019-12-12 14:55:30 -08:00
|
|
|
|
|
|
|
|
interface OverrideProps {
|
|
|
|
|
match: FieldMatcher;
|
|
|
|
|
properties: DynamicConfigValue[];
|
|
|
|
|
}
|
|
|
|
|
|
2020-11-29 09:22:16 -08:00
|
|
|
export function findNumericFieldMinMax(data: DataFrame[]): NumericRange {
|
2020-11-13 10:28:03 +01:00
|
|
|
let min: number | null = null;
|
|
|
|
|
let max: number | null = null;
|
2019-12-12 14:55:30 -08:00
|
|
|
|
|
|
|
|
const reducers = [ReducerID.min, ReducerID.max];
|
2020-05-04 13:37:43 +02:00
|
|
|
|
2019-12-12 14:55:30 -08:00
|
|
|
for (const frame of data) {
|
|
|
|
|
for (const field of frame.fields) {
|
|
|
|
|
if (field.type === FieldType.number) {
|
|
|
|
|
const stats = reduceField({ field, reducers });
|
2020-10-28 08:04:30 +01:00
|
|
|
const statsMin = stats[ReducerID.min];
|
|
|
|
|
const statsMax = stats[ReducerID.max];
|
|
|
|
|
|
2020-11-13 10:28:03 +01:00
|
|
|
if (min === null || statsMin < min) {
|
2020-10-28 08:04:30 +01:00
|
|
|
min = statsMin;
|
|
|
|
|
}
|
|
|
|
|
|
2020-11-13 10:28:03 +01:00
|
|
|
if (max === null || statsMax > max) {
|
2020-10-28 08:04:30 +01:00
|
|
|
max = statsMax;
|
2019-12-12 14:55:30 -08:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-11-29 09:22:16 -08:00
|
|
|
return { min, max, delta: (max ?? 0) - (min ?? 0) };
|
2019-12-12 14:55:30 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 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;
|
|
|
|
|
|
2020-10-09 09:34:57 +02:00
|
|
|
let seriesIndex = 0;
|
2020-11-29 09:22:16 -08:00
|
|
|
let globalRange: NumericRange | undefined = undefined;
|
2019-12-12 14:55:30 -08:00
|
|
|
|
|
|
|
|
// 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,
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-06-19 07:36:59 +02:00
|
|
|
return options.data.map((originalFrame, index) => {
|
2020-09-16 20:07:14 +02:00
|
|
|
// Need to define this new frame here as it's passed to the getLinkSupplier function inside the fields loop
|
2021-06-19 07:36:59 +02:00
|
|
|
const newFrame: DataFrame = { ...originalFrame };
|
|
|
|
|
// Copy fields
|
|
|
|
|
newFrame.fields = newFrame.fields.map((field) => {
|
|
|
|
|
return {
|
|
|
|
|
...field,
|
|
|
|
|
config: cloneDeep(field.config),
|
|
|
|
|
state: {
|
|
|
|
|
...field.state,
|
|
|
|
|
},
|
|
|
|
|
};
|
|
|
|
|
});
|
2020-09-16 20:07:14 +02:00
|
|
|
|
2021-06-19 07:36:59 +02:00
|
|
|
for (const field of newFrame.fields) {
|
|
|
|
|
const config = field.config;
|
Field: getFieldTitle as field / series display identity and use it in all field name matchers & field / series name displays (#24024)
* common title handling
* show labels
* update comment
* Update changelog for v7.0.0-beta1 (#24007)
Co-Authored-By: Marcus Efraimsson <marcus.efraimsson@gmail.com>
Co-Authored-By: Andrej Ocenas <mr.ocenas@gmail.com>
Co-Authored-By: Hugo Häggmark <hugo.haggmark@grafana.com>
Co-authored-by: Tobias Skarhed <1438972+tskarhed@users.noreply.github.com>
* verify-repo-update: Fix Dockerfile.deb (#24030)
Signed-off-by: Arve Knudsen <arve.knudsen@gmail.com>
* CircleCI: Upgrade build pipeline tool (#24021)
* CircleCI: Upgrade build pipeline tool
* Devenv: ignore enterprise (#24037)
* Add header icon to Add data source page (#24033)
* latest.json: Update testing version (#24038)
Signed-off-by: Arve Knudsen <arve.knudsen@gmail.com>
* Fix login page redirected from password reset (#24032)
* Storybook: Rewrite stories to CSF (#23989)
* ColorPicker to CSF format
* Convert stories to CSF
* Do not export ClipboardButton
* Update ConfirmButton
* Remove unused imports
* Fix feedback
* changelog enterprise 7.0.0-beta1 (#24039)
* CircleCI: Bump grafana/build-container revision (#24043)
Signed-off-by: Arve Knudsen <arve.knudsen@gmail.com>
* Changelog: Updates changelog with more feature details (#24040)
* Changelog: Updates changelog with more feature details
* spell fix
* spell fix
* Updates
* Readme update
* Updates
* Select: fixes so component loses focus on selecting value or pressing outside of input. (#24008)
* changed the value container to a class component to get it to work with focus (maybe something with context?).
* added e2e tests to verify that the select focus is working as it should.
* fixed according to feedback.
* updated snapshot.
* Devenv: add remote renderer to grafana (#24050)
* NewPanelEditor: minor UI twekas (#24042)
* Forward ref for tabs, use html props
* Inspect: add inspect label to drawer title
* Add tooltips to sidebar pane tabs, copy changes
* Remove unused import
* Place tooltips over tabs
* Inspector: dont show transformations select if there is only one data frame
* Review
* Changelog: Add a breaking change (#24051)
Signed-off-by: Arve Knudsen <arve.knudsen@gmail.com>
* CircleCI: Unpin grafana/docs-base (#24054)
Signed-off-by: Arve Knudsen <arve.knudsen@gmail.com>
* Search: close overlay on Esc press (#24003)
* Search: Close on Esc
* Search: Increase bottom padding for the last item in section
* Search: Move closing search to keybindingsSrv
* Search: Fix folder view
* Search: Do not move folders if already in folder
* Docs: Adds deprecation notice to changelog and docs for scripted dashboards (#24060)
* Update CHANGELOG.md (#24047)
Fix typo
Co-authored-by: Daniel Lee <dan.limerick@gmail.com>
* Documentation: Alternative Team Sync Wording (#23960)
* Alternative wording for team sync docs
Signed-off-by: Joe Elliott <number101010@gmail.com>
* Update docs/sources/auth/team-sync.md
Co-Authored-By: Diana Payton <52059945+oddlittlebird@users.noreply.github.com>
Co-authored-by: Diana Payton <52059945+oddlittlebird@users.noreply.github.com>
* Fix misspell issues (#23905)
* Fix misspell issues
See,
$ golangci-lint run --timeout 10m --disable-all -E misspell ./...
Signed-off-by: Mario Trangoni <mjtrangoni@gmail.com>
* Fix codespell issues
See,
$ codespell -S './.git*' -L 'uint,thru,pres,unknwon,serie,referer,uptodate,durationm'
Signed-off-by: Mario Trangoni <mjtrangoni@gmail.com>
* ci please?
* non-empty commit - ci?
* Trigger build
Co-authored-by: bergquist <carl.bergquist@gmail.com>
Co-authored-by: Kyle Brandt <kyle@grafana.com>
* fix compile error
* better series display
* better display
* now with prometheus and loki
* a few more tests
* Improvements and tests
* thinking
* More advanced and smart default title generation
* Another fix
* Progress but dam this will be hard
* Reverting the time series Value field name change
* revert revert going in circles
* add a field state object
* Use state title when converting back to legacy format
* Improved the join (series to columsn) transformer
* Got tests running again
* Rewrite of seriesToColums that simplifies and fixing tests
* Fixed the tricky problem of multiple time field when not used in join
* Prometheus: Restoring prometheus formatting
* Graphite: Disable Grafana's series naming
* fixed imports
* Fixed tests and made rename transform change title instead
* Fixing more tests
* fix more tests
* fixed import issue
* Fixed more circular dependencies
* Renamed to getFieldTitle
* More rename
* Review feedback
* Fix for showing field title in calculate field transformer
* fieldOverride: Make it clear that state title after applying defaults & overrides
* Fixed ts issue
* Update packages/grafana-ui/src/components/TransformersUI/OrganizeFieldsTransformerEditor.tsx
Co-authored-by: Marcus Efraimsson <marcus.efraimsson@gmail.com>
Co-authored-by: Andrej Ocenas <mr.ocenas@gmail.com>
Co-authored-by: Hugo Häggmark <hugo.haggmark@grafana.com>
Co-authored-by: Tobias Skarhed <1438972+tskarhed@users.noreply.github.com>
Co-authored-by: Arve Knudsen <arve.knudsen@gmail.com>
Co-authored-by: Leonard Gram <leo@xlson.com>
Co-authored-by: Ivana Huckova <30407135+ivanahuckova@users.noreply.github.com>
Co-authored-by: Alexander Zobnin <alexanderzobnin@gmail.com>
Co-authored-by: Torkel Ödegaard <torkel@grafana.com>
Co-authored-by: Marcus Andersson <marcus.andersson@grafana.com>
Co-authored-by: Dominik Prokop <dominik.prokop@grafana.com>
Co-authored-by: Alex Khomenko <Clarity-89@users.noreply.github.com>
Co-authored-by: Richard Hartmann <RichiH@users.noreply.github.com>
Co-authored-by: Daniel Lee <dan.limerick@gmail.com>
Co-authored-by: Joe Elliott <joe.elliott@grafana.com>
Co-authored-by: Diana Payton <52059945+oddlittlebird@users.noreply.github.com>
Co-authored-by: Mario Trangoni <mario@mariotrangoni.de>
Co-authored-by: bergquist <carl.bergquist@gmail.com>
Co-authored-by: Kyle Brandt <kyle@grafana.com>
2020-05-07 01:42:03 -07:00
|
|
|
|
2021-06-19 07:36:59 +02:00
|
|
|
field.state!.scopedVars = {
|
2023-04-05 11:10:33 +02:00
|
|
|
__dataContext: {
|
|
|
|
|
value: {
|
|
|
|
|
data: options.data!,
|
|
|
|
|
frame: newFrame,
|
|
|
|
|
frameIndex: index,
|
|
|
|
|
field: field,
|
|
|
|
|
},
|
2021-06-19 07:36:59 +02:00
|
|
|
},
|
Field: getFieldTitle as field / series display identity and use it in all field name matchers & field / series name displays (#24024)
* common title handling
* show labels
* update comment
* Update changelog for v7.0.0-beta1 (#24007)
Co-Authored-By: Marcus Efraimsson <marcus.efraimsson@gmail.com>
Co-Authored-By: Andrej Ocenas <mr.ocenas@gmail.com>
Co-Authored-By: Hugo Häggmark <hugo.haggmark@grafana.com>
Co-authored-by: Tobias Skarhed <1438972+tskarhed@users.noreply.github.com>
* verify-repo-update: Fix Dockerfile.deb (#24030)
Signed-off-by: Arve Knudsen <arve.knudsen@gmail.com>
* CircleCI: Upgrade build pipeline tool (#24021)
* CircleCI: Upgrade build pipeline tool
* Devenv: ignore enterprise (#24037)
* Add header icon to Add data source page (#24033)
* latest.json: Update testing version (#24038)
Signed-off-by: Arve Knudsen <arve.knudsen@gmail.com>
* Fix login page redirected from password reset (#24032)
* Storybook: Rewrite stories to CSF (#23989)
* ColorPicker to CSF format
* Convert stories to CSF
* Do not export ClipboardButton
* Update ConfirmButton
* Remove unused imports
* Fix feedback
* changelog enterprise 7.0.0-beta1 (#24039)
* CircleCI: Bump grafana/build-container revision (#24043)
Signed-off-by: Arve Knudsen <arve.knudsen@gmail.com>
* Changelog: Updates changelog with more feature details (#24040)
* Changelog: Updates changelog with more feature details
* spell fix
* spell fix
* Updates
* Readme update
* Updates
* Select: fixes so component loses focus on selecting value or pressing outside of input. (#24008)
* changed the value container to a class component to get it to work with focus (maybe something with context?).
* added e2e tests to verify that the select focus is working as it should.
* fixed according to feedback.
* updated snapshot.
* Devenv: add remote renderer to grafana (#24050)
* NewPanelEditor: minor UI twekas (#24042)
* Forward ref for tabs, use html props
* Inspect: add inspect label to drawer title
* Add tooltips to sidebar pane tabs, copy changes
* Remove unused import
* Place tooltips over tabs
* Inspector: dont show transformations select if there is only one data frame
* Review
* Changelog: Add a breaking change (#24051)
Signed-off-by: Arve Knudsen <arve.knudsen@gmail.com>
* CircleCI: Unpin grafana/docs-base (#24054)
Signed-off-by: Arve Knudsen <arve.knudsen@gmail.com>
* Search: close overlay on Esc press (#24003)
* Search: Close on Esc
* Search: Increase bottom padding for the last item in section
* Search: Move closing search to keybindingsSrv
* Search: Fix folder view
* Search: Do not move folders if already in folder
* Docs: Adds deprecation notice to changelog and docs for scripted dashboards (#24060)
* Update CHANGELOG.md (#24047)
Fix typo
Co-authored-by: Daniel Lee <dan.limerick@gmail.com>
* Documentation: Alternative Team Sync Wording (#23960)
* Alternative wording for team sync docs
Signed-off-by: Joe Elliott <number101010@gmail.com>
* Update docs/sources/auth/team-sync.md
Co-Authored-By: Diana Payton <52059945+oddlittlebird@users.noreply.github.com>
Co-authored-by: Diana Payton <52059945+oddlittlebird@users.noreply.github.com>
* Fix misspell issues (#23905)
* Fix misspell issues
See,
$ golangci-lint run --timeout 10m --disable-all -E misspell ./...
Signed-off-by: Mario Trangoni <mjtrangoni@gmail.com>
* Fix codespell issues
See,
$ codespell -S './.git*' -L 'uint,thru,pres,unknwon,serie,referer,uptodate,durationm'
Signed-off-by: Mario Trangoni <mjtrangoni@gmail.com>
* ci please?
* non-empty commit - ci?
* Trigger build
Co-authored-by: bergquist <carl.bergquist@gmail.com>
Co-authored-by: Kyle Brandt <kyle@grafana.com>
* fix compile error
* better series display
* better display
* now with prometheus and loki
* a few more tests
* Improvements and tests
* thinking
* More advanced and smart default title generation
* Another fix
* Progress but dam this will be hard
* Reverting the time series Value field name change
* revert revert going in circles
* add a field state object
* Use state title when converting back to legacy format
* Improved the join (series to columsn) transformer
* Got tests running again
* Rewrite of seriesToColums that simplifies and fixing tests
* Fixed the tricky problem of multiple time field when not used in join
* Prometheus: Restoring prometheus formatting
* Graphite: Disable Grafana's series naming
* fixed imports
* Fixed tests and made rename transform change title instead
* Fixing more tests
* fix more tests
* fixed import issue
* Fixed more circular dependencies
* Renamed to getFieldTitle
* More rename
* Review feedback
* Fix for showing field title in calculate field transformer
* fieldOverride: Make it clear that state title after applying defaults & overrides
* Fixed ts issue
* Update packages/grafana-ui/src/components/TransformersUI/OrganizeFieldsTransformerEditor.tsx
Co-authored-by: Marcus Efraimsson <marcus.efraimsson@gmail.com>
Co-authored-by: Andrej Ocenas <mr.ocenas@gmail.com>
Co-authored-by: Hugo Häggmark <hugo.haggmark@grafana.com>
Co-authored-by: Tobias Skarhed <1438972+tskarhed@users.noreply.github.com>
Co-authored-by: Arve Knudsen <arve.knudsen@gmail.com>
Co-authored-by: Leonard Gram <leo@xlson.com>
Co-authored-by: Ivana Huckova <30407135+ivanahuckova@users.noreply.github.com>
Co-authored-by: Alexander Zobnin <alexanderzobnin@gmail.com>
Co-authored-by: Torkel Ödegaard <torkel@grafana.com>
Co-authored-by: Marcus Andersson <marcus.andersson@grafana.com>
Co-authored-by: Dominik Prokop <dominik.prokop@grafana.com>
Co-authored-by: Alex Khomenko <Clarity-89@users.noreply.github.com>
Co-authored-by: Richard Hartmann <RichiH@users.noreply.github.com>
Co-authored-by: Daniel Lee <dan.limerick@gmail.com>
Co-authored-by: Joe Elliott <joe.elliott@grafana.com>
Co-authored-by: Diana Payton <52059945+oddlittlebird@users.noreply.github.com>
Co-authored-by: Mario Trangoni <mario@mariotrangoni.de>
Co-authored-by: bergquist <carl.bergquist@gmail.com>
Co-authored-by: Kyle Brandt <kyle@grafana.com>
2020-05-07 01:42:03 -07:00
|
|
|
};
|
2019-12-12 14:55:30 -08:00
|
|
|
|
2020-02-13 21:37:24 +01:00
|
|
|
const context = {
|
2021-06-19 07:36:59 +02:00
|
|
|
field: field,
|
2020-02-13 21:37:24 +01:00
|
|
|
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);
|
2021-06-19 07:36:59 +02:00
|
|
|
|
2019-12-12 14:55:30 -08:00
|
|
|
// Find any matching rules and then override
|
|
|
|
|
for (const rule of override) {
|
2021-06-19 07:36:59 +02:00
|
|
|
if (rule.match(field, newFrame, options.data!)) {
|
2019-12-12 14:55:30 -08:00
|
|
|
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;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-12-12 14:55:30 -08:00
|
|
|
// Set the Min/Max value automatically
|
2020-11-29 09:22:16 -08:00
|
|
|
let range: NumericRange | undefined = undefined;
|
|
|
|
|
if (field.type === FieldType.number) {
|
|
|
|
|
if (!globalRange && (!isNumber(config.min) || !isNumber(config.max))) {
|
|
|
|
|
globalRange = findNumericFieldMinMax(options.data!);
|
2019-12-12 14:55:30 -08:00
|
|
|
}
|
2020-11-29 09:22:16 -08:00
|
|
|
const min = config.min ?? globalRange!.min;
|
|
|
|
|
const max = config.max ?? globalRange!.max;
|
|
|
|
|
range = { min, max, delta: max! - min! };
|
2019-12-12 14:55:30 -08:00
|
|
|
}
|
|
|
|
|
|
2021-08-04 13:02:27 -05:00
|
|
|
field.state!.seriesIndex = seriesIndex;
|
|
|
|
|
field.state!.range = range;
|
|
|
|
|
field.type = type;
|
|
|
|
|
|
2020-10-09 09:34:57 +02:00
|
|
|
// Some color modes needs series index to assign field color so we count
|
|
|
|
|
// up series index here but ignore time fields
|
|
|
|
|
if (field.type !== FieldType.time) {
|
|
|
|
|
seriesIndex++;
|
|
|
|
|
}
|
|
|
|
|
|
2019-12-28 17:32:58 -08:00
|
|
|
// and set the display processor using it
|
2021-06-19 07:36:59 +02:00
|
|
|
field.display = getDisplayProcessor({
|
|
|
|
|
field: field,
|
2020-01-15 12:02:52 -08:00
|
|
|
theme: options.theme,
|
|
|
|
|
timeZone: options.timeZone,
|
|
|
|
|
});
|
2020-04-20 07:37:38 +02:00
|
|
|
|
2021-06-02 11:01:23 -07:00
|
|
|
// Wrap the display with a cache to avoid double calls
|
2021-06-19 07:36:59 +02:00
|
|
|
if (field.config.unit !== 'dateTimeFromNow') {
|
|
|
|
|
field.display = cachingDisplayProcessor(field.display, 2500);
|
2021-06-02 11:01:23 -07:00
|
|
|
}
|
|
|
|
|
|
2020-04-20 07:37:38 +02:00
|
|
|
// Attach data links supplier
|
2021-06-19 07:36:59 +02:00
|
|
|
field.getLinks = getLinksSupplier(
|
2021-04-29 12:44:06 +02:00
|
|
|
newFrame,
|
2021-06-19 07:36:59 +02:00
|
|
|
field,
|
|
|
|
|
field.state!.scopedVars,
|
2021-04-29 12:44:06 +02:00
|
|
|
context.replaceVariables,
|
|
|
|
|
options.timeZone
|
|
|
|
|
);
|
2021-06-19 07:36:59 +02:00
|
|
|
}
|
2020-04-20 07:37:38 +02:00
|
|
|
|
2020-09-16 20:07:14 +02:00
|
|
|
return newFrame;
|
2019-12-12 14:55:30 -08:00
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2023-03-20 14:17:51 +01:00
|
|
|
// this is a significant optimization for streaming, where we currently re-process all values in the buffer on ech update
|
2021-07-23 20:50:47 -05:00
|
|
|
// via field.display(value). this can potentially be removed once we...
|
|
|
|
|
// 1. process data packets incrementally and/if cache the results in the streaming datafame (maybe by buffer index)
|
|
|
|
|
// 2. have the ability to selectively get display color or text (but not always both, which are each quite expensive)
|
2021-12-09 00:04:58 +09:00
|
|
|
// 3. sufficently optimize text formatting and threshold color determinitation
|
2021-06-02 11:01:23 -07:00
|
|
|
function cachingDisplayProcessor(disp: DisplayProcessor, maxCacheSize = 2500): DisplayProcessor {
|
2022-07-28 01:58:42 -05:00
|
|
|
type dispCache = Map<any, DisplayValue>;
|
|
|
|
|
// decimals -> cache mapping, -1 is unspecified decimals
|
|
|
|
|
const caches = new Map<number, dispCache>();
|
|
|
|
|
|
|
|
|
|
// pre-init caches for up to 15 decimals
|
|
|
|
|
for (let i = -1; i <= 15; i++) {
|
|
|
|
|
caches.set(i, new Map());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return (value: any, decimals?: DecimalCount) => {
|
|
|
|
|
let cache = caches.get(decimals ?? -1)!;
|
2021-06-02 11:01:23 -07:00
|
|
|
|
|
|
|
|
let v = cache.get(value);
|
2021-07-23 20:50:47 -05:00
|
|
|
|
2021-06-02 11:01:23 -07:00
|
|
|
if (!v) {
|
|
|
|
|
// Don't grow too big
|
|
|
|
|
if (cache.size === maxCacheSize) {
|
|
|
|
|
cache.clear();
|
|
|
|
|
}
|
|
|
|
|
|
2022-07-28 01:58:42 -05:00
|
|
|
v = disp(value, decimals);
|
2021-07-23 20:50:47 -05:00
|
|
|
|
|
|
|
|
// convert to hex6 or hex8 so downstream we can cheaply test for alpha (and set new alpha)
|
|
|
|
|
// via a simple length check (in colorManipulator) rather using slow parsing via tinycolor
|
2021-07-26 15:54:13 -07:00
|
|
|
if (v.color) {
|
2021-07-26 11:11:53 -07:00
|
|
|
v.color = asHexString(v.color);
|
2021-07-23 20:50:47 -05:00
|
|
|
}
|
|
|
|
|
|
2021-06-02 11:01:23 -07:00
|
|
|
cache.set(value, v);
|
|
|
|
|
}
|
2021-07-23 20:50:47 -05:00
|
|
|
|
2021-06-02 11:01:23 -07:00
|
|
|
return v;
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
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);
|
2021-06-19 07:36:59 +02:00
|
|
|
|
2020-06-09 17:43:29 +02:00
|
|
|
if (!item) {
|
2020-02-13 21:37:24 +01:00
|
|
|
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);
|
|
|
|
|
}
|
|
|
|
|
|
2021-06-19 07:36:59 +02:00
|
|
|
function processFieldConfigValue(
|
2020-02-13 21:37:24 +01:00
|
|
|
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
|
2021-06-19 07:36:59 +02: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
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-06-19 07:36:59 +02:00
|
|
|
}
|
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 (!config.color) {
|
|
|
|
|
if (thresholds) {
|
|
|
|
|
config.color = {
|
2020-10-09 09:34:57 +02:00
|
|
|
mode: FieldColorModeId.Thresholds,
|
2019-12-28 17:32:58 -08:00
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
// No Color settings
|
|
|
|
|
} else if (!config.color.mode) {
|
|
|
|
|
// Without a mode, skip color altogether
|
|
|
|
|
delete config.color;
|
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
|
|
|
|
2022-02-02 12:02:32 +00:00
|
|
|
export const getLinksSupplier =
|
|
|
|
|
(
|
|
|
|
|
frame: DataFrame,
|
|
|
|
|
field: Field,
|
|
|
|
|
fieldScopedVars: ScopedVars,
|
|
|
|
|
replaceVariables: InterpolateFunction,
|
|
|
|
|
timeZone?: TimeZone
|
|
|
|
|
) =>
|
|
|
|
|
(config: ValueLinkConfig): Array<LinkModel<Field>> => {
|
|
|
|
|
if (!field.config.links || field.config.links.length === 0) {
|
|
|
|
|
return [];
|
|
|
|
|
}
|
|
|
|
|
|
2022-04-20 10:15:36 +01:00
|
|
|
return field.config.links.map((link: DataLink) => {
|
2023-04-05 11:10:33 +02:00
|
|
|
const dataContext: DataContextScopedVar = getFieldDataContextClone(frame, field, fieldScopedVars);
|
|
|
|
|
const dataLinkScopedVars = {
|
|
|
|
|
...fieldScopedVars,
|
|
|
|
|
__dataContext: dataContext,
|
|
|
|
|
};
|
2022-02-02 12:02:32 +00:00
|
|
|
|
|
|
|
|
// We are not displaying reduction result
|
|
|
|
|
if (config.valueRowIndex !== undefined && !isNaN(config.valueRowIndex)) {
|
2023-03-28 19:22:34 +02:00
|
|
|
dataContext.value.rowIndex = config.valueRowIndex;
|
2022-02-02 12:02:32 +00:00
|
|
|
} else {
|
2023-03-28 19:22:34 +02:00
|
|
|
dataContext.value.calculatedValue = config.calculatedValue;
|
2022-02-02 12:02:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (link.onClick) {
|
|
|
|
|
return {
|
|
|
|
|
href: link.url,
|
2023-04-05 11:10:33 +02:00
|
|
|
title: replaceVariables(link.title || '', dataLinkScopedVars),
|
2022-02-02 12:02:32 +00:00
|
|
|
target: link.targetBlank ? '_blank' : undefined,
|
|
|
|
|
onClick: (evt, origin) => {
|
|
|
|
|
link.onClick!({
|
|
|
|
|
origin: origin ?? field,
|
|
|
|
|
e: evt,
|
2023-04-05 11:10:33 +02:00
|
|
|
replaceVariables: (v) => replaceVariables(v, dataLinkScopedVars),
|
2022-02-02 12:02:32 +00:00
|
|
|
});
|
|
|
|
|
},
|
|
|
|
|
origin: field,
|
2020-04-20 07:37:38 +02:00
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
2022-02-02 12:02:32 +00:00
|
|
|
if (link.internal) {
|
|
|
|
|
// For internal links at the moment only destination is Explore.
|
|
|
|
|
return mapInternalLinkToExplore({
|
|
|
|
|
link,
|
|
|
|
|
internalLink: link.internal,
|
2023-04-05 11:10:33 +02:00
|
|
|
scopedVars: dataLinkScopedVars,
|
2022-02-02 12:02:32 +00:00
|
|
|
field,
|
2023-03-14 09:12:46 -06:00
|
|
|
range: link.internal.range ?? ({} as any),
|
2022-02-02 12:02:32 +00:00
|
|
|
replaceVariables,
|
|
|
|
|
});
|
|
|
|
|
}
|
2022-12-21 05:09:31 -08:00
|
|
|
let href = link.onBuildUrl
|
|
|
|
|
? link.onBuildUrl({
|
|
|
|
|
origin: field,
|
|
|
|
|
replaceVariables,
|
|
|
|
|
})
|
|
|
|
|
: link.url;
|
|
|
|
|
|
|
|
|
|
if (href) {
|
2023-03-17 06:51:06 -07:00
|
|
|
href = locationUtil.assureBaseUrl(href.replace(/\n/g, ''));
|
2023-04-05 11:10:33 +02:00
|
|
|
href = replaceVariables(href, dataLinkScopedVars, VariableFormatID.PercentEncode);
|
2022-12-21 05:09:31 -08:00
|
|
|
href = locationUtil.processUrl(href);
|
|
|
|
|
}
|
2022-02-02 12:02:32 +00:00
|
|
|
|
|
|
|
|
const info: LinkModel<Field> = {
|
|
|
|
|
href,
|
2023-04-05 11:10:33 +02:00
|
|
|
title: replaceVariables(link.title || '', dataLinkScopedVars),
|
2021-07-13 09:43:39 -07:00
|
|
|
target: link.targetBlank ? '_blank' : undefined,
|
|
|
|
|
origin: field,
|
|
|
|
|
};
|
2022-02-02 12:02:32 +00:00
|
|
|
return info;
|
|
|
|
|
});
|
|
|
|
|
};
|
2020-09-04 11:21:24 +02:00
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Return a copy of the DataFrame with raw data
|
|
|
|
|
*/
|
|
|
|
|
export function applyRawFieldOverrides(data: DataFrame[]): DataFrame[] {
|
|
|
|
|
if (!data || data.length === 0) {
|
|
|
|
|
return [];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const newData = [...data];
|
|
|
|
|
const processor = getRawDisplayProcessor();
|
|
|
|
|
|
|
|
|
|
for (let frameIndex = 0; frameIndex < newData.length; frameIndex++) {
|
|
|
|
|
const newFrame = { ...newData[frameIndex] };
|
|
|
|
|
const newFields = [...newFrame.fields];
|
|
|
|
|
|
|
|
|
|
for (let fieldIndex = 0; fieldIndex < newFields.length; fieldIndex++) {
|
|
|
|
|
newFields[fieldIndex] = {
|
|
|
|
|
...newFields[fieldIndex],
|
|
|
|
|
display: processor,
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
newData[frameIndex] = {
|
|
|
|
|
...newFrame,
|
|
|
|
|
fields: newFields,
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return newData;
|
|
|
|
|
}
|
2023-01-02 02:27:48 -08:00
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @internal
|
|
|
|
|
*/
|
|
|
|
|
export function useFieldOverrides(
|
|
|
|
|
plugin: PanelPlugin | undefined,
|
|
|
|
|
fieldConfig: FieldConfigSource | undefined,
|
|
|
|
|
data: PanelData | undefined,
|
|
|
|
|
timeZone: string,
|
|
|
|
|
theme: GrafanaTheme2,
|
|
|
|
|
replace: InterpolateFunction
|
|
|
|
|
): PanelData | undefined {
|
|
|
|
|
const fieldConfigRegistry = plugin?.fieldConfigRegistry;
|
|
|
|
|
const structureRev = useRef(0);
|
|
|
|
|
const prevSeries = usePrevious(data?.series);
|
|
|
|
|
|
|
|
|
|
return useMemo(() => {
|
|
|
|
|
if (!fieldConfigRegistry || !fieldConfig || !data) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const series = data?.series;
|
|
|
|
|
|
|
|
|
|
if (
|
|
|
|
|
data.structureRev == null &&
|
|
|
|
|
series &&
|
|
|
|
|
prevSeries &&
|
|
|
|
|
!compareArrayValues(series, prevSeries, compareDataFrameStructures)
|
|
|
|
|
) {
|
|
|
|
|
structureRev.current++;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
structureRev: structureRev.current,
|
|
|
|
|
...data,
|
|
|
|
|
series: applyFieldOverrides({
|
|
|
|
|
data: series,
|
|
|
|
|
fieldConfig,
|
|
|
|
|
fieldConfigRegistry,
|
|
|
|
|
replaceVariables: replace,
|
|
|
|
|
theme,
|
|
|
|
|
timeZone,
|
|
|
|
|
}),
|
|
|
|
|
};
|
|
|
|
|
}, [fieldConfigRegistry, fieldConfig, data, prevSeries, timeZone, theme, replace]);
|
|
|
|
|
}
|
2023-04-05 11:10:33 +02:00
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Clones the existing dataContext or creates a new one
|
|
|
|
|
*/
|
|
|
|
|
function getFieldDataContextClone(frame: DataFrame, field: Field, fieldScopedVars: ScopedVars) {
|
|
|
|
|
if (fieldScopedVars?.__dataContext) {
|
|
|
|
|
return {
|
|
|
|
|
value: {
|
|
|
|
|
...fieldScopedVars.__dataContext.value,
|
|
|
|
|
},
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return { value: { frame, field, data: [frame] } };
|
|
|
|
|
}
|