Skip to content

Commit

Permalink
[TSVB] Visualize runtime fields
Browse files Browse the repository at this point in the history
  • Loading branch information
alexwizp committed Apr 9, 2021
1 parent dfaf3ac commit 663636c
Show file tree
Hide file tree
Showing 61 changed files with 485 additions and 411 deletions.
5 changes: 1 addition & 4 deletions src/plugins/vis_type_timeseries/common/calculate_label.ts
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@ import { includes, startsWith } from 'lodash';
import { i18n } from '@kbn/i18n';
import { lookup } from './agg_lookup';
import { MetricsItemsSchema, SanitizedFieldType } from './types';
import { extractFieldLabel } from './fields_utils';

const paths = [
'cumulative_sum',
Expand All @@ -26,10 +27,6 @@ const paths = [
'positive_only',
];

export const extractFieldLabel = (fields: SanitizedFieldType[], name: string) => {
return fields.find((f) => f.name === name)?.label ?? name;
};

export const calculateLabel = (
metric: MetricsItemsSchema,
metrics: MetricsItemsSchema[] = [],
Expand Down
1 change: 1 addition & 0 deletions src/plugins/vis_type_timeseries/common/constants.ts
Original file line number Diff line number Diff line change
Expand Up @@ -13,3 +13,4 @@ export const ROUTES = {
VIS_DATA: '/api/metrics/vis/data',
FIELDS: '/api/metrics/fields',
};
export const USE_KIBANA_INDEXES_KEY = 'use_kibana_indexes';
13 changes: 1 addition & 12 deletions src/plugins/vis_type_timeseries/common/fields_utils.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@
*/

import { toSanitizedFieldType } from './fields_utils';
import type { FieldSpec, RuntimeField } from '../../data/common';
import type { FieldSpec } from '../../data/common';

describe('fields_utils', () => {
describe('toSanitizedFieldType', () => {
Expand All @@ -34,17 +34,6 @@ describe('fields_utils', () => {
`);
});

test('should filter runtime fields', async () => {
const fields: FieldSpec[] = [
{
...mockedField,
runtimeField: {} as RuntimeField,
},
];

expect(toSanitizedFieldType(fields)).toMatchInlineSnapshot(`Array []`);
});

test('should filter non-aggregatable fields', async () => {
const fields: FieldSpec[] = [
{
Expand Down
55 changes: 46 additions & 9 deletions src/plugins/vis_type_timeseries/common/fields_utils.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,17 +6,55 @@
* Side Public License, v 1.
*/

import { i18n } from '@kbn/i18n';
import { FieldSpec } from '../../data/common';
import { isNestedField } from '../../data/common';
import { SanitizedFieldType } from './types';
import { FetchedIndexPattern, SanitizedFieldType } from './types';

export const toSanitizedFieldType = (fields: FieldSpec[]) => {
return fields
.filter(
(field) =>
// Make sure to only include mapped fields, e.g. no index pattern runtime fields
!field.runtimeField && field.aggregatable && !isNestedField(field)
)
export class FieldNotFoundError extends Error {
constructor(name: string) {
super(
i18n.translate('visTypeTimeseries.fields.fieldNotFound', {
defaultMessage: `Field "{field}" not found`,
values: { field: name },
})
);
}

public get name() {
return this.constructor.name;
}

public get body() {
return this.message;
}
}

export const extractFieldLabel = (fields: SanitizedFieldType[], name: string) => {
if (fields.length && name) {
const field = fields.find((f) => f.name === name);

if (field) {
return field.label || field.name;
}

throw new FieldNotFoundError(name);
}
return name;
};

export function validateField(name: string, index: FetchedIndexPattern) {
if (name && index.indexPattern) {
const field = index.indexPattern.fields.find((f) => f.name === name);
if (!field) {
throw new FieldNotFoundError(name);
}
}
}

export const toSanitizedFieldType = (fields: FieldSpec[]) =>
fields
.filter((field) => field.aggregatable && !isNestedField(field))
.map(
(field) =>
({
Expand All @@ -25,4 +63,3 @@ export const toSanitizedFieldType = (fields: FieldSpec[]) => {
type: field.type,
} as SanitizedFieldType)
);
};
Original file line number Diff line number Diff line change
Expand Up @@ -81,15 +81,17 @@ describe('fetchIndexPattern', () => {
});

describe('text-based index', () => {
test('should return the Kibana index if it exists', async () => {
test('should return the Kibana index if it exists (fetchKibabaIndexForStringIndexes is true)', async () => {
mockedIndices = [
{
id: 'indexId',
title: 'indexTitle',
},
] as IndexPattern[];

const value = await fetchIndexPattern('indexTitle', indexPatternsService);
const value = await fetchIndexPattern('indexTitle', indexPatternsService, {
fetchKibabaIndexForStringIndexes: true,
});

expect(value).toMatchInlineSnapshot(`
Object {
Expand All @@ -102,8 +104,10 @@ describe('fetchIndexPattern', () => {
`);
});

test('should return only indexPatternString if Kibana index does not exist', async () => {
const value = await fetchIndexPattern('indexTitle', indexPatternsService);
test('should return only indexPatternString if Kibana index does not exist (fetchKibabaIndexForStringIndexes is true)', async () => {
const value = await fetchIndexPattern('indexTitle', indexPatternsService, {
fetchKibabaIndexForStringIndexes: true,
});

expect(value).toMatchInlineSnapshot(`
Object {
Expand Down
18 changes: 13 additions & 5 deletions src/plugins/vis_type_timeseries/common/index_patterns_utils.ts
Original file line number Diff line number Diff line change
Expand Up @@ -52,7 +52,12 @@ export const extractIndexPatternValues = (

export const fetchIndexPattern = async (
indexPatternValue: IndexPatternValue | undefined,
indexPatternsService: Pick<IndexPatternsService, 'getDefault' | 'get' | 'find'>
indexPatternsService: Pick<IndexPatternsService, 'getDefault' | 'get' | 'find'>,
options: {
fetchKibabaIndexForStringIndexes: boolean;
} = {
fetchKibabaIndexForStringIndexes: false,
}
): Promise<FetchedIndexPattern> => {
let indexPattern: FetchedIndexPattern['indexPattern'];
let indexPatternString: string = '';
Expand All @@ -61,13 +66,16 @@ export const fetchIndexPattern = async (
indexPattern = await indexPatternsService.getDefault();
} else {
if (isStringTypeIndexPattern(indexPatternValue)) {
indexPattern = (await indexPatternsService.find(indexPatternValue)).find(
(index) => index.title === indexPatternValue
);

if (options.fetchKibabaIndexForStringIndexes) {
indexPattern = (await indexPatternsService.find(indexPatternValue)).find(
(index) => index.title === indexPatternValue
);
}
if (!indexPattern) {
indexPatternString = indexPatternValue;
}

indexPatternString = indexPatternValue;
} else if (indexPatternValue.id) {
indexPattern = await indexPatternsService.get(indexPatternValue.id);
}
Expand Down
4 changes: 3 additions & 1 deletion src/plugins/vis_type_timeseries/common/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,7 @@ interface TableData {
export type SeriesData = {
type: Exclude<PANEL_TYPES, PANEL_TYPES.TABLE>;
uiRestrictions: TimeseriesUIRestrictions;
error?: string;
} & {
[key: string]: PanelSeries;
};
Expand All @@ -56,7 +57,7 @@ interface PanelSeries {
};
id: string;
series: PanelData[];
error?: unknown;
error?: string;
}

export interface PanelData {
Expand All @@ -66,6 +67,7 @@ export interface PanelData {
seriesId: string;
splitByLabel: string;
isSplitByTerms: boolean;
error?: string;
}

export const isVisTableData = (data: TimeseriesVisData): data is TableData =>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -5,19 +5,26 @@
* in compliance with, at your election, the Elastic License 2.0 or the Server
* Side Public License, v 1.
*/

import React from 'react';
import { i18n } from '@kbn/i18n';
import { EuiComboBox, EuiComboBoxProps, EuiComboBoxOptionOption } from '@elastic/eui';
import { METRIC_TYPES } from '../../../../common/metric_types';
import React, { ReactNode, useContext } from 'react';
import {
EuiComboBox,
EuiComboBoxProps,
EuiComboBoxOptionOption,
EuiFormRow,
htmlIdGenerator,
} from '@elastic/eui';
import { getIndexPatternKey } from '../../../../common/index_patterns_utils';
import type { SanitizedFieldType, IndexPatternValue } from '../../../../common/types';
import type { TimeseriesUIRestrictions } from '../../../../common/ui_restrictions';

// @ts-ignore
import { isFieldEnabled } from '../../lib/check_ui_restrictions';
import { PanelModelContext } from '../../contexts/panel_model_context';
import { USE_KIBANA_INDEXES_KEY } from '../../../../common/constants';

interface FieldSelectProps {
label: string | ReactNode;
type: string;
fields: Record<string, SanitizedFieldType[]>;
indexPattern: IndexPatternValue;
Expand Down Expand Up @@ -45,6 +52,7 @@ const sortByLabel = (a: EuiComboBoxOptionOption<string>, b: EuiComboBoxOptionOpt
};

export function FieldSelect({
label,
type,
fields,
indexPattern = '',
Expand All @@ -56,11 +64,10 @@ export function FieldSelect({
uiRestrictions,
'data-test-subj': dataTestSubj = 'metricsIndexPatternFieldsSelect',
}: FieldSelectProps) {
if (type === METRIC_TYPES.COUNT) {
return null;
}
const panelModel = useContext(PanelModelContext);
const htmlId = htmlIdGenerator();

const selectedOptions: Array<EuiComboBoxOptionOption<string>> = [];
let selectedOptions: Array<EuiComboBoxOptionOption<string>> = [];
let newPlaceholder = placeholder;
const fieldsSelector = getIndexPatternKey(indexPattern);

Expand Down Expand Up @@ -112,19 +119,43 @@ export function FieldSelect({
}
});

if (value && !selectedOptions.length) {
onChange([]);
let isInvalid;

if (Boolean(panelModel?.[USE_KIBANA_INDEXES_KEY])) {
isInvalid = Boolean(value && fields[fieldsSelector] && !selectedOptions.length);

if (value && !selectedOptions.length) {
selectedOptions = [{ label: value!, id: 'INVALID_FIELD' }];
}
} else {
if (value && !selectedOptions.length) {
onChange([]);
}
}

return (
<EuiComboBox
data-test-subj={dataTestSubj}
placeholder={newPlaceholder}
isDisabled={disabled}
options={groupedOptions}
selectedOptions={selectedOptions}
onChange={onChange}
singleSelection={{ asPlainText: true }}
/>
<EuiFormRow
id={htmlId('timeField')}
label={label}
isInvalid={isInvalid}
error={i18n.translate('visTypeTimeseries.fieldSelect.fieldIsNotValid', {
defaultMessage:
'The "{fieldParameter}" field is not valid for use with the current index. Please select a new field.',
values: {
fieldParameter: value,
},
})}
>
<EuiComboBox
data-test-subj={dataTestSubj}
placeholder={newPlaceholder}
isDisabled={disabled}
options={groupedOptions}
selectedOptions={selectedOptions}
onChange={onChange}
singleSelection={{ asPlainText: true }}
isInvalid={isInvalid}
/>
</EuiFormRow>
);
}
Original file line number Diff line number Diff line change
Expand Up @@ -153,24 +153,20 @@ export const FilterRatioAgg = (props) => {

{model.metric_agg !== 'count' ? (
<EuiFlexItem>
<EuiFormRow
id={htmlId('aggField')}
<FieldSelect
label={
<FormattedMessage
id="visTypeTimeseries.filterRatio.fieldLabel"
defaultMessage="Field"
/>
}
>
<FieldSelect
fields={fields}
type={model.metric_agg}
restrict={getSupportedFieldsByMetricType(model.metric_agg)}
indexPattern={indexPattern}
value={model.field}
onChange={handleSelectChange('field')}
/>
</EuiFormRow>
fields={fields}
type={model.metric_agg}
restrict={getSupportedFieldsByMetricType(model.metric_agg)}
indexPattern={indexPattern}
value={model.field}
onChange={handleSelectChange('field')}
/>
</EuiFlexItem>
) : null}
</EuiFlexGroup>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -78,24 +78,20 @@ export function PercentileAgg(props) {
/>
</EuiFlexItem>
<EuiFlexItem>
<EuiFormRow
id={htmlId('field')}
<FieldSelect
label={
<FormattedMessage
id="visTypeTimeseries.percentile.fieldLabel"
defaultMessage="Field"
/>
}
>
<FieldSelect
fields={fields}
type={model.type}
restrict={RESTRICT_FIELDS}
indexPattern={indexPattern}
value={model.field}
onChange={handleSelectChange('field')}
/>
</EuiFormRow>
fields={fields}
type={model.type}
restrict={RESTRICT_FIELDS}
indexPattern={indexPattern}
value={model.field}
onChange={handleSelectChange('field')}
/>
</EuiFlexItem>
<EuiFlexItem>
<EuiFormRow
Expand Down
Loading

0 comments on commit 663636c

Please sign in to comment.