mirror of
https://github.com/grafana/grafana.git
synced 2025-02-12 08:35:43 -06:00
453 lines
19 KiB
TypeScript
453 lines
19 KiB
TypeScript
import { reduxTester } from '../../../../../test/core/redux/reduxTester';
|
|
import { getRootReducer } from '../../state/helpers';
|
|
import { TemplatingState } from '../../state/reducers';
|
|
import { initialVariableModelState, QueryVariableModel, VariableRefresh, VariableSort } from '../../types';
|
|
import {
|
|
hideOptions,
|
|
moveOptionsHighlight,
|
|
showOptions,
|
|
toggleOption,
|
|
toggleTag,
|
|
updateOptionsAndFilter,
|
|
updateSearchQuery,
|
|
} from './reducer';
|
|
import {
|
|
commitChangesToVariable,
|
|
filterOrSearchOptions,
|
|
navigateOptions,
|
|
toggleAndFetchTag,
|
|
toggleOptionByHighlight,
|
|
} from './actions';
|
|
import { NavigationKey } from '../types';
|
|
import { toVariablePayload } from '../../state/types';
|
|
import { addVariable, changeVariableProp, setCurrentVariableValue } from '../../state/sharedReducer';
|
|
import { variableAdapters } from '../../adapters';
|
|
import { createQueryVariableAdapter } from '../../query/adapter';
|
|
import { updateLocation } from 'app/core/actions';
|
|
|
|
const datasource = {
|
|
metricFindQuery: jest.fn(() => Promise.resolve([])),
|
|
};
|
|
|
|
jest.mock('@grafana/runtime', () => {
|
|
const original = jest.requireActual('@grafana/runtime');
|
|
|
|
return {
|
|
...original,
|
|
getDataSourceSrv: jest.fn(() => ({
|
|
get: () => datasource,
|
|
})),
|
|
};
|
|
});
|
|
|
|
describe('options picker actions', () => {
|
|
variableAdapters.setInit(() => [createQueryVariableAdapter()]);
|
|
|
|
describe('when navigateOptions is dispatched with navigation key cancel', () => {
|
|
it('then correct actions are dispatched', async () => {
|
|
const variable = createMultiVariable({
|
|
options: [createOption('A', 'A', true)],
|
|
current: createOption(['A'], ['A'], true),
|
|
});
|
|
|
|
const clearOthers = false;
|
|
const key = NavigationKey.cancel;
|
|
|
|
const tester = await reduxTester<{ templating: TemplatingState }>()
|
|
.givenRootReducer(getRootReducer())
|
|
.whenActionIsDispatched(addVariable(toVariablePayload(variable, { global: false, index: 0, model: variable })))
|
|
.whenActionIsDispatched(showOptions(variable))
|
|
.whenAsyncActionIsDispatched(navigateOptions(key, clearOthers), true);
|
|
|
|
const option = {
|
|
...createOption(['A']),
|
|
selected: true,
|
|
value: ['A'],
|
|
tags: [] as any[],
|
|
};
|
|
|
|
tester.thenDispatchedActionsShouldEqual(
|
|
setCurrentVariableValue(toVariablePayload(variable, { option })),
|
|
changeVariableProp(toVariablePayload(variable, { propName: 'queryValue', propValue: '' })),
|
|
hideOptions()
|
|
);
|
|
});
|
|
});
|
|
|
|
describe('when navigateOptions is dispatched with navigation key select without clearOthers', () => {
|
|
it('then correct actions are dispatched', async () => {
|
|
const option = createOption('A', 'A', true);
|
|
const variable = createMultiVariable({
|
|
options: [option],
|
|
current: createOption(['A'], ['A'], true),
|
|
includeAll: false,
|
|
});
|
|
|
|
const clearOthers = false;
|
|
const key = NavigationKey.select;
|
|
|
|
const tester = await reduxTester<{ templating: TemplatingState }>()
|
|
.givenRootReducer(getRootReducer())
|
|
.whenActionIsDispatched(addVariable(toVariablePayload(variable, { global: false, index: 0, model: variable })))
|
|
.whenActionIsDispatched(showOptions(variable))
|
|
.whenActionIsDispatched(navigateOptions(NavigationKey.moveDown, false))
|
|
.whenAsyncActionIsDispatched(navigateOptions(key, clearOthers), true);
|
|
|
|
tester.thenDispatchedActionsShouldEqual(toggleOption({ option, forceSelect: false, clearOthers }));
|
|
});
|
|
});
|
|
|
|
describe('when navigateOptions is dispatched with navigation key select with clearOthers', () => {
|
|
it('then correct actions are dispatched', async () => {
|
|
const option = createOption('A', 'A', true);
|
|
const variable = createMultiVariable({
|
|
options: [option],
|
|
current: createOption(['A'], ['A'], true),
|
|
includeAll: false,
|
|
});
|
|
|
|
const clearOthers = true;
|
|
const key = NavigationKey.select;
|
|
|
|
const tester = await reduxTester<{ templating: TemplatingState }>()
|
|
.givenRootReducer(getRootReducer())
|
|
.whenActionIsDispatched(addVariable(toVariablePayload(variable, { global: false, index: 0, model: variable })))
|
|
.whenActionIsDispatched(showOptions(variable))
|
|
.whenActionIsDispatched(navigateOptions(NavigationKey.moveDown, clearOthers))
|
|
.whenAsyncActionIsDispatched(navigateOptions(key, clearOthers), true);
|
|
|
|
tester.thenDispatchedActionsShouldEqual(toggleOption({ option, forceSelect: false, clearOthers }));
|
|
});
|
|
});
|
|
|
|
describe('when navigateOptions is dispatched with navigation key select after highlighting the third option', () => {
|
|
it('then correct actions are dispatched', async () => {
|
|
const options = [createOption('A'), createOption('B'), createOption('C')];
|
|
const variable = createMultiVariable({ options, current: createOption(['A'], ['A'], true), includeAll: false });
|
|
|
|
const clearOthers = true;
|
|
const key = NavigationKey.select;
|
|
|
|
const tester = await reduxTester<{ templating: TemplatingState }>()
|
|
.givenRootReducer(getRootReducer())
|
|
.whenActionIsDispatched(addVariable(toVariablePayload(variable, { global: false, index: 0, model: variable })))
|
|
.whenActionIsDispatched(showOptions(variable))
|
|
.whenActionIsDispatched(navigateOptions(NavigationKey.moveDown, clearOthers))
|
|
.whenActionIsDispatched(navigateOptions(NavigationKey.moveDown, clearOthers))
|
|
.whenActionIsDispatched(navigateOptions(NavigationKey.moveDown, clearOthers))
|
|
.whenAsyncActionIsDispatched(navigateOptions(key, clearOthers), true);
|
|
|
|
tester.thenDispatchedActionsShouldEqual(toggleOption({ option: options[2], forceSelect: false, clearOthers }));
|
|
});
|
|
});
|
|
|
|
describe('when navigateOptions is dispatched with navigation key select after highlighting the second option', () => {
|
|
it('then correct actions are dispatched', async () => {
|
|
const options = [createOption('A'), createOption('B'), createOption('C')];
|
|
const variable = createMultiVariable({ options, current: createOption(['A'], ['A'], true), includeAll: false });
|
|
|
|
const clearOthers = true;
|
|
const key = NavigationKey.select;
|
|
|
|
const tester = await reduxTester<{ templating: TemplatingState }>()
|
|
.givenRootReducer(getRootReducer())
|
|
.whenActionIsDispatched(addVariable(toVariablePayload(variable, { global: false, index: 0, model: variable })))
|
|
.whenActionIsDispatched(showOptions(variable))
|
|
.whenActionIsDispatched(navigateOptions(NavigationKey.moveDown, clearOthers))
|
|
.whenActionIsDispatched(navigateOptions(NavigationKey.moveDown, clearOthers))
|
|
.whenActionIsDispatched(navigateOptions(NavigationKey.moveDown, clearOthers))
|
|
.whenActionIsDispatched(navigateOptions(NavigationKey.moveUp, clearOthers))
|
|
.whenAsyncActionIsDispatched(navigateOptions(key, clearOthers), true);
|
|
|
|
tester.thenDispatchedActionsShouldEqual(toggleOption({ option: options[1], forceSelect: false, clearOthers }));
|
|
});
|
|
});
|
|
|
|
describe('when navigateOptions is dispatched with navigation key selectAndClose after highlighting the second option', () => {
|
|
it('then correct actions are dispatched', async () => {
|
|
const options = [createOption('A'), createOption('B'), createOption('C')];
|
|
const variable = createMultiVariable({ options, current: createOption(['A'], ['A'], true), includeAll: false });
|
|
|
|
const clearOthers = false;
|
|
const key = NavigationKey.selectAndClose;
|
|
|
|
const tester = await reduxTester<{ templating: TemplatingState }>()
|
|
.givenRootReducer(getRootReducer())
|
|
.whenActionIsDispatched(addVariable(toVariablePayload(variable, { global: false, index: 0, model: variable })))
|
|
.whenActionIsDispatched(showOptions(variable))
|
|
.whenActionIsDispatched(navigateOptions(NavigationKey.moveDown, clearOthers))
|
|
.whenActionIsDispatched(navigateOptions(NavigationKey.moveDown, clearOthers))
|
|
.whenActionIsDispatched(navigateOptions(NavigationKey.moveDown, clearOthers))
|
|
.whenActionIsDispatched(navigateOptions(NavigationKey.moveUp, clearOthers))
|
|
.whenAsyncActionIsDispatched(navigateOptions(key, clearOthers), true);
|
|
|
|
const option = {
|
|
...createOption(['B']),
|
|
selected: true,
|
|
value: ['B'],
|
|
tags: [] as any[],
|
|
};
|
|
|
|
tester.thenDispatchedActionsShouldEqual(
|
|
toggleOption({ option: options[1], forceSelect: true, clearOthers }),
|
|
setCurrentVariableValue(toVariablePayload(variable, { option })),
|
|
changeVariableProp(toVariablePayload(variable, { propName: 'queryValue', propValue: '' })),
|
|
hideOptions(),
|
|
setCurrentVariableValue(toVariablePayload(variable, { option })),
|
|
updateLocation({ query: { 'var-Constant': ['B'] } })
|
|
);
|
|
});
|
|
});
|
|
|
|
describe('when filterOrSearchOptions is dispatched with simple filter', () => {
|
|
it('then correct actions are dispatched', async () => {
|
|
const options = [createOption('A'), createOption('B'), createOption('C')];
|
|
const variable = createMultiVariable({ options, current: createOption(['A'], ['A'], true), includeAll: false });
|
|
const filter = 'A';
|
|
|
|
const tester = await reduxTester<{ templating: TemplatingState }>()
|
|
.givenRootReducer(getRootReducer())
|
|
.whenActionIsDispatched(addVariable(toVariablePayload(variable, { global: false, index: 0, model: variable })))
|
|
.whenActionIsDispatched(showOptions(variable))
|
|
.whenAsyncActionIsDispatched(filterOrSearchOptions(filter), true);
|
|
|
|
tester.thenDispatchedActionsShouldEqual(updateSearchQuery(filter), updateOptionsAndFilter(variable.options));
|
|
});
|
|
});
|
|
|
|
describe('when commitChangesToVariable is dispatched with no changes', () => {
|
|
it('then correct actions are dispatched', async () => {
|
|
const options = [createOption('A', 'A', true), createOption('B'), createOption('C')];
|
|
const variable = createMultiVariable({ options, current: createOption(['A'], ['A'], true), includeAll: false });
|
|
|
|
const tester = await reduxTester<{ templating: TemplatingState }>()
|
|
.givenRootReducer(getRootReducer())
|
|
.whenActionIsDispatched(addVariable(toVariablePayload(variable, { global: false, index: 0, model: variable })))
|
|
.whenActionIsDispatched(showOptions(variable))
|
|
.whenAsyncActionIsDispatched(commitChangesToVariable(), true);
|
|
|
|
const option = {
|
|
...createOption(['A']),
|
|
selected: true,
|
|
value: ['A'] as any[],
|
|
tags: [] as any[],
|
|
};
|
|
|
|
tester.thenDispatchedActionsShouldEqual(
|
|
setCurrentVariableValue(toVariablePayload(variable, { option })),
|
|
changeVariableProp(toVariablePayload(variable, { propName: 'queryValue', propValue: '' })),
|
|
hideOptions()
|
|
);
|
|
});
|
|
});
|
|
|
|
describe('when commitChangesToVariable is dispatched with changes', () => {
|
|
it('then correct actions are dispatched', async () => {
|
|
const options = [createOption('A', 'A', true), createOption('B'), createOption('C')];
|
|
const variable = createMultiVariable({ options, current: createOption(['A'], ['A'], true), includeAll: false });
|
|
const clearOthers = false;
|
|
|
|
const tester = await reduxTester<{ templating: TemplatingState }>()
|
|
.givenRootReducer(getRootReducer())
|
|
.whenActionIsDispatched(addVariable(toVariablePayload(variable, { global: false, index: 0, model: variable })))
|
|
.whenActionIsDispatched(showOptions(variable))
|
|
.whenActionIsDispatched(navigateOptions(NavigationKey.moveDown, clearOthers))
|
|
.whenActionIsDispatched(toggleOptionByHighlight(clearOthers))
|
|
.whenAsyncActionIsDispatched(commitChangesToVariable(), true);
|
|
|
|
const option = {
|
|
...createOption([]),
|
|
selected: true,
|
|
value: [],
|
|
tags: [] as any[],
|
|
};
|
|
|
|
tester.thenDispatchedActionsShouldEqual(
|
|
setCurrentVariableValue(toVariablePayload(variable, { option })),
|
|
changeVariableProp(toVariablePayload(variable, { propName: 'queryValue', propValue: '' })),
|
|
hideOptions(),
|
|
setCurrentVariableValue(toVariablePayload(variable, { option })),
|
|
updateLocation({ query: { 'var-Constant': [] } })
|
|
);
|
|
});
|
|
});
|
|
|
|
describe('when commitChangesToVariable is dispatched with changes and list of options is filtered', () => {
|
|
it('then correct actions are dispatched', async () => {
|
|
const options = [createOption('A', 'A', true), createOption('B'), createOption('C')];
|
|
const variable = createMultiVariable({ options, current: createOption(['A'], ['A'], true), includeAll: false });
|
|
const clearOthers = false;
|
|
|
|
const tester = await reduxTester<{ templating: TemplatingState }>()
|
|
.givenRootReducer(getRootReducer())
|
|
.whenActionIsDispatched(addVariable(toVariablePayload(variable, { global: false, index: 0, model: variable })))
|
|
.whenActionIsDispatched(showOptions(variable))
|
|
.whenActionIsDispatched(navigateOptions(NavigationKey.moveDown, clearOthers))
|
|
.whenActionIsDispatched(toggleOptionByHighlight(clearOthers))
|
|
.whenActionIsDispatched(filterOrSearchOptions('C'))
|
|
.whenAsyncActionIsDispatched(commitChangesToVariable(), true);
|
|
|
|
const option = {
|
|
...createOption([]),
|
|
selected: true,
|
|
value: [],
|
|
tags: [] as any[],
|
|
};
|
|
|
|
tester.thenDispatchedActionsShouldEqual(
|
|
setCurrentVariableValue(toVariablePayload(variable, { option })),
|
|
changeVariableProp(toVariablePayload(variable, { propName: 'queryValue', propValue: 'C' })),
|
|
hideOptions(),
|
|
setCurrentVariableValue(toVariablePayload(variable, { option })),
|
|
updateLocation({ query: { 'var-Constant': [] } })
|
|
);
|
|
});
|
|
});
|
|
|
|
describe('when toggleOptionByHighlight is dispatched with changes', () => {
|
|
it('then correct actions are dispatched', async () => {
|
|
const options = [createOption('A'), createOption('B'), createOption('C')];
|
|
const variable = createMultiVariable({ options, current: createOption(['A'], ['A'], true), includeAll: false });
|
|
const clearOthers = false;
|
|
|
|
const tester = await reduxTester<{ templating: TemplatingState }>()
|
|
.givenRootReducer(getRootReducer())
|
|
.whenActionIsDispatched(addVariable(toVariablePayload(variable, { global: false, index: 0, model: variable })))
|
|
.whenActionIsDispatched(showOptions(variable))
|
|
.whenActionIsDispatched(navigateOptions(NavigationKey.moveDown, clearOthers))
|
|
.whenActionIsDispatched(toggleOptionByHighlight(clearOthers), true);
|
|
|
|
const option = createOption('A');
|
|
|
|
tester.thenDispatchedActionsShouldEqual(toggleOption({ option, forceSelect: false, clearOthers }));
|
|
});
|
|
});
|
|
|
|
describe('when toggleOptionByHighlight is dispatched with changes selected from a filtered options list', () => {
|
|
it('then correct actions are dispatched', async () => {
|
|
const options = [createOption('A'), createOption('B'), createOption('BC'), createOption('BD')];
|
|
const variable = createMultiVariable({ options, current: createOption(['A'], ['A'], true), includeAll: false });
|
|
const clearOthers = false;
|
|
|
|
const tester = await reduxTester<{ templating: TemplatingState }>()
|
|
.givenRootReducer(getRootReducer())
|
|
.whenActionIsDispatched(addVariable(toVariablePayload(variable, { global: false, index: 0, model: variable })))
|
|
.whenActionIsDispatched(showOptions(variable))
|
|
.whenActionIsDispatched(navigateOptions(NavigationKey.moveDown, clearOthers))
|
|
.whenActionIsDispatched(toggleOptionByHighlight(clearOthers), true)
|
|
.whenActionIsDispatched(filterOrSearchOptions('B'))
|
|
.whenActionIsDispatched(navigateOptions(NavigationKey.moveDown, clearOthers))
|
|
.whenActionIsDispatched(navigateOptions(NavigationKey.moveDown, clearOthers))
|
|
.whenActionIsDispatched(toggleOptionByHighlight(clearOthers));
|
|
|
|
const optionA = createOption('A');
|
|
const optionBC = createOption('BD');
|
|
|
|
tester.thenDispatchedActionsShouldEqual(
|
|
toggleOption({ option: optionA, forceSelect: false, clearOthers }),
|
|
updateSearchQuery('B'),
|
|
updateOptionsAndFilter(variable.options),
|
|
moveOptionsHighlight(1),
|
|
moveOptionsHighlight(1),
|
|
toggleOption({ option: optionBC, forceSelect: false, clearOthers })
|
|
);
|
|
});
|
|
});
|
|
|
|
describe('when toggleAndFetchTag is dispatched with values', () => {
|
|
it('then correct actions are dispatched', async () => {
|
|
const options = [createOption('A'), createOption('B'), createOption('C')];
|
|
const tag = createTag('tag', []);
|
|
const variable = createMultiVariable({
|
|
options,
|
|
current: createOption(['A'], ['A'], true),
|
|
includeAll: false,
|
|
tags: [tag],
|
|
});
|
|
|
|
const tester = await reduxTester<{ templating: TemplatingState }>()
|
|
.givenRootReducer(getRootReducer())
|
|
.whenActionIsDispatched(addVariable(toVariablePayload(variable, { global: false, index: 0, model: variable })))
|
|
.whenActionIsDispatched(showOptions(variable))
|
|
.whenAsyncActionIsDispatched(toggleAndFetchTag(tag), true);
|
|
|
|
tester.thenDispatchedActionsShouldEqual(toggleTag(tag));
|
|
});
|
|
});
|
|
|
|
describe('when toggleAndFetchTag is dispatched without values', () => {
|
|
it('then correct actions are dispatched', async () => {
|
|
const options = [createOption('A'), createOption('B'), createOption('C')];
|
|
const tag = createTag('tag');
|
|
const values = [createMetric('b')];
|
|
const variable = createMultiVariable({
|
|
options,
|
|
current: createOption(['A'], ['A'], true),
|
|
includeAll: false,
|
|
tags: [tag],
|
|
});
|
|
|
|
datasource.metricFindQuery.mockReset();
|
|
// @ts-ignore strict null error TS2345: Argument of type '() => Promise<{ value: string; text: string; }[]>' is not assignable to parameter of type '() => Promise<never[]>'
|
|
datasource.metricFindQuery.mockImplementation(() => Promise.resolve(values));
|
|
|
|
const tester = await reduxTester<{ templating: TemplatingState }>()
|
|
.givenRootReducer(getRootReducer())
|
|
.whenActionIsDispatched(addVariable(toVariablePayload(variable, { global: false, index: 0, model: variable })))
|
|
.whenActionIsDispatched(showOptions(variable))
|
|
.whenAsyncActionIsDispatched(toggleAndFetchTag(tag), true);
|
|
|
|
tester.thenDispatchedActionsShouldEqual(toggleTag({ ...tag, values: ['b'] }));
|
|
});
|
|
});
|
|
});
|
|
|
|
function createMultiVariable(extend?: Partial<QueryVariableModel>): QueryVariableModel {
|
|
return {
|
|
...initialVariableModelState,
|
|
type: 'query',
|
|
id: '0',
|
|
index: 0,
|
|
current: createOption([]),
|
|
options: [],
|
|
query: 'options-query',
|
|
name: 'Constant',
|
|
datasource: 'datasource',
|
|
definition: '',
|
|
sort: VariableSort.alphabeticalAsc,
|
|
tags: [],
|
|
tagsQuery: 'tags-query',
|
|
tagValuesQuery: '',
|
|
useTags: true,
|
|
refresh: VariableRefresh.onDashboardLoad,
|
|
regex: '',
|
|
multi: true,
|
|
includeAll: true,
|
|
...(extend ?? {}),
|
|
};
|
|
}
|
|
|
|
function createOption(text: string | string[], value?: string | string[], selected?: boolean) {
|
|
const metric = createMetric(text);
|
|
return {
|
|
...metric,
|
|
value: value ?? metric.value,
|
|
selected: selected ?? false,
|
|
};
|
|
}
|
|
|
|
function createMetric(value: string | string[]) {
|
|
return {
|
|
value: value,
|
|
text: value,
|
|
};
|
|
}
|
|
|
|
function createTag(name: string, values?: any[]) {
|
|
return {
|
|
selected: false,
|
|
text: name,
|
|
values,
|
|
};
|
|
}
|