You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

173 lines
6.2 KiB

import { of, throwError } from 'rxjs';
import { DataSourceInstanceSettings, toUtc } from '@grafana/data';
import CloudMonitoringDataSource from '../datasource';
import { TemplateSrv } from 'app/features/templating/template_srv';
import { CloudMonitoringOptions } from '../types';
import { backendSrv } from 'app/core/services/backend_srv'; // will use the version in __mocks__
import { TimeSrv } from 'app/features/dashboard/services/TimeSrv';
import { CustomVariableModel } from '../../../../features/variables/types';
import { initialCustomVariableModelState } from '../../../../features/variables/custom/reducer';
import { createFetchResponse } from 'test/helpers/createFetchResponse';
jest.mock('@grafana/runtime', () => ({
...((jest.requireActual('@grafana/runtime') as unknown) as object),
getBackendSrv: () => backendSrv,
}));
type Args = { response?: any; throws?: boolean; templateSrv?: TemplateSrv };
function getTestcontext({ response = {}, throws = false, templateSrv = new TemplateSrv() }: Args = {}) {
jest.clearAllMocks();
const instanceSettings = ({
jsonData: {
defaultProject: 'testproject',
},
} as unknown) as DataSourceInstanceSettings<CloudMonitoringOptions>;
const timeSrv = {} as TimeSrv;
const fetchMock = jest.spyOn(backendSrv, 'fetch');
throws
? fetchMock.mockImplementation(() => throwError(response))
: fetchMock.mockImplementation(() => of(createFetchResponse(response)));
const ds = new CloudMonitoringDataSource(instanceSettings, templateSrv, timeSrv);
return { ds };
}
describe('CloudMonitoringDataSource', () => {
describe('When performing query', () => {
describe('and no time series data is returned', () => {
it('should return a list of datapoints', async () => {
const options = {
range: {
from: toUtc('2017-08-22T20:00:00Z'),
to: toUtc('2017-08-22T23:59:00Z'),
},
rangeRaw: {
from: 'now-4h',
to: 'now',
},
targets: [
{
refId: 'A',
},
],
};
const response: any = {
results: {
A: {
refId: 'A',
meta: {
rawQuery: 'arawquerystring',
},
series: null,
tables: null,
},
},
};
const { ds } = getTestcontext({ response });
await expect(ds.query(options as any)).toEmitValuesWith((received) => {
const results = received[0];
expect(results.data.length).toBe(0);
});
});
});
});
describe('when interpolating a template variable for the filter', () => {
describe('and is single value variable', () => {
it('should replace the variable with the value', () => {
const templateSrv = initTemplateSrv('filtervalue1');
const { ds } = getTestcontext({ templateSrv });
const interpolated = ds.interpolateFilters(['resource.label.zone', '=~', '${test}'], {});
expect(interpolated.length).toBe(3);
expect(interpolated[2]).toBe('filtervalue1');
});
});
describe('and is single value variable for the label part', () => {
it('should replace the variable with the value and not with regex formatting', () => {
const templateSrv = initTemplateSrv('resource.label.zone');
const { ds } = getTestcontext({ templateSrv });
const interpolated = ds.interpolateFilters(['${test}', '=~', 'europe-north-1a'], {});
expect(interpolated.length).toBe(3);
expect(interpolated[0]).toBe('resource.label.zone');
});
});
describe('and is multi value variable', () => {
it('should replace the variable with a regex expression', () => {
const templateSrv = initTemplateSrv(['filtervalue1', 'filtervalue2'], true);
const { ds } = getTestcontext({ templateSrv });
const interpolated = ds.interpolateFilters(['resource.label.zone', '=~', '[[test]]'], {});
expect(interpolated[2]).toBe('(filtervalue1|filtervalue2)');
});
it('should not escape a regex', () => {
const templateSrv = initTemplateSrv('/[a-Z]*.html', true);
const { ds } = getTestcontext({ templateSrv });
const interpolated = ds.interpolateFilters(['resource.label.zone', '=~', '[[test]]'], {});
expect(interpolated[2]).toBe('/[a-Z]*.html');
});
it('should not escape an array of regexes but join them as a regex', () => {
const templateSrv = initTemplateSrv(['/[a-Z]*.html', '/foo.html'], true);
const { ds } = getTestcontext({ templateSrv });
const interpolated = ds.interpolateFilters(['resource.label.zone', '=~', '[[test]]'], {});
expect(interpolated[2]).toBe('(/[a-Z]*.html|/foo.html)');
});
});
});
describe('when interpolating a template variable for group bys', () => {
describe('and is single value variable', () => {
it('should replace the variable with the value', () => {
const templateSrv = initTemplateSrv('groupby1');
const { ds } = getTestcontext({ templateSrv });
const interpolated = ds.interpolateGroupBys(['[[test]]'], {});
expect(interpolated.length).toBe(1);
expect(interpolated[0]).toBe('groupby1');
});
});
describe('and is multi value variable', () => {
it('should replace the variable with an array of group bys', () => {
const templateSrv = initTemplateSrv(['groupby1', 'groupby2'], true);
const { ds } = getTestcontext({ templateSrv });
const interpolated = ds.interpolateGroupBys(['[[test]]'], {});
expect(interpolated.length).toBe(2);
expect(interpolated[0]).toBe('groupby1');
expect(interpolated[1]).toBe('groupby2');
});
});
});
});
function initTemplateSrv(values: any, multi = false) {
const templateSrv = new TemplateSrv();
const test: CustomVariableModel = {
...initialCustomVariableModelState,
id: 'test',
name: 'test',
current: { value: values, text: Array.isArray(values) ? values.toString() : values, selected: true },
options: [{ value: values, text: Array.isArray(values) ? values.toString() : values, selected: false }],
multi,
};
templateSrv.init([test]);
return templateSrv;
}