grafana/public/app/plugins/datasource/elasticsearch/specs/datasource.test.ts

393 lines
10 KiB
TypeScript
Raw Normal View History

import angular from 'angular';
import * as dateMath from 'app/core/utils/datemath';
2017-12-20 05:33:33 -06:00
import _ from 'lodash';
import moment from 'moment';
import { ElasticDatasource } from '../datasource';
2018-07-04 03:43:36 -05:00
describe('ElasticDatasource', function(this: any) {
const backendSrv = {
2018-07-03 03:47:50 -05:00
datasourceRequest: jest.fn(),
2018-07-03 03:12:07 -05:00
};
const $rootScope = {
2018-07-03 03:12:07 -05:00
$on: jest.fn(),
appEvent: jest.fn(),
};
const templateSrv = {
2019-01-09 00:46:20 -06:00
replace: jest.fn(text => {
if (text.startsWith('$')) {
2019-01-09 00:46:20 -06:00
return `resolvedVariable`;
} else {
return text;
}
}),
2018-07-04 03:43:36 -05:00
getAdhocFilters: jest.fn(() => []),
};
2018-07-03 03:47:50 -05:00
const timeSrv = {
2018-07-04 03:43:36 -05:00
time: { from: 'now-1h', to: 'now' },
2018-07-04 04:23:12 -05:00
timeRange: jest.fn(() => {
2018-07-04 03:43:36 -05:00
return {
from: dateMath.parse(this.time.from, false),
to: dateMath.parse(this.time.to, true),
};
}),
setTime: jest.fn(time => {
this.time = time;
}),
};
2018-07-03 03:12:07 -05:00
const ctx = {
2018-07-03 03:12:07 -05:00
$rootScope,
backendSrv,
} as any;
function createDatasource(instanceSettings) {
instanceSettings.jsonData = instanceSettings.jsonData || {};
2018-07-03 03:47:50 -05:00
ctx.ds = new ElasticDatasource(instanceSettings, {}, backendSrv, templateSrv, timeSrv);
}
describe('When testing datasource with index pattern', () => {
beforeEach(() => {
createDatasource({
2017-12-20 05:33:33 -06:00
url: 'http://es.com',
index: '[asd-]YYYY.MM.DD',
jsonData: { interval: 'Daily', esVersion: '2' },
});
});
it('should translate index pattern to current day', () => {
let requestOptions;
2018-07-03 03:47:50 -05:00
ctx.backendSrv.datasourceRequest = jest.fn(options => {
requestOptions = options;
2018-07-03 03:12:07 -05:00
return Promise.resolve({ data: {} });
});
ctx.ds.testDatasource();
const today = moment.utc().format('YYYY.MM.DD');
2018-07-03 03:47:50 -05:00
expect(requestOptions.url).toBe('http://es.com/asd-' + today + '/_mapping');
});
});
describe('When issuing metric query with interval pattern', () => {
let requestOptions, parts, header, query, result;
beforeEach(async () => {
createDatasource({
2017-12-20 05:33:33 -06:00
url: 'http://es.com',
index: '[asd-]YYYY.MM.DD',
jsonData: { interval: 'Daily', esVersion: '2' },
});
2018-07-03 03:47:50 -05:00
ctx.backendSrv.datasourceRequest = jest.fn(options => {
requestOptions = options;
return Promise.resolve({
data: {
responses: [
{
aggregations: {
'1': {
buckets: [
{
doc_count: 10,
key: 1000,
},
],
},
},
},
],
},
});
2018-07-03 03:12:07 -05:00
});
2019-01-09 00:46:20 -06:00
query = {
range: {
from: moment.utc([2015, 4, 30, 10]),
2017-12-20 05:33:33 -06:00
to: moment.utc([2015, 5, 1, 10]),
},
targets: [
{
alias: '$varAlias',
bucketAggs: [{ type: 'date_histogram', field: '@timestamp', id: '1' }],
metrics: [{ type: 'count', id: '1' }],
2017-12-20 05:33:33 -06:00
query: 'escape\\:test',
},
],
2019-01-09 00:46:20 -06:00
};
result = await ctx.ds.query(query);
2017-12-20 05:33:33 -06:00
parts = requestOptions.data.split('\n');
header = angular.fromJson(parts[0]);
});
it('should translate index pattern to current day', () => {
2018-07-03 03:12:07 -05:00
expect(header.index).toEqual(['asd-2015.05.30', 'asd-2015.05.31', 'asd-2015.06.01']);
});
it('should not resolve the variable in the original alias field in the query', () => {
expect(query.targets[0].alias).toEqual('$varAlias');
});
it('should resolve the alias variable for the alias/target in the result', () => {
expect(result.data[0].target).toEqual('resolvedVariable');
2019-01-09 00:46:20 -06:00
});
it('should json escape lucene query', () => {
const body = angular.fromJson(parts[1]);
2018-07-03 03:12:07 -05:00
expect(body.query.bool.filter[1].query_string.query).toBe('escape\\:test');
});
});
describe('When issuing document query', () => {
let requestOptions, parts, header;
beforeEach(() => {
createDatasource({
2017-12-20 05:33:33 -06:00
url: 'http://es.com',
index: 'test',
jsonData: { esVersion: '2' },
});
2018-07-03 03:47:50 -05:00
ctx.backendSrv.datasourceRequest = jest.fn(options => {
requestOptions = options;
2018-07-03 03:12:07 -05:00
return Promise.resolve({ data: { responses: [] } });
});
ctx.ds.query({
range: {
from: moment([2015, 4, 30, 10]),
2017-12-20 05:33:33 -06:00
to: moment([2015, 5, 1, 10]),
},
targets: [
{
bucketAggs: [],
2017-12-20 05:33:33 -06:00
metrics: [{ type: 'raw_document' }],
query: 'test',
},
],
});
2017-12-20 05:33:33 -06:00
parts = requestOptions.data.split('\n');
header = angular.fromJson(parts[0]);
});
it('should set search type to query_then_fetch', () => {
2018-07-03 03:12:07 -05:00
expect(header.search_type).toEqual('query_then_fetch');
});
it('should set size', () => {
const body = angular.fromJson(parts[1]);
2018-07-03 03:12:07 -05:00
expect(body.size).toBe(500);
});
});
describe('When getting fields', () => {
2018-07-03 03:47:50 -05:00
beforeEach(() => {
2017-12-20 05:33:33 -06:00
createDatasource({ url: 'http://es.com', index: 'metricbeat' });
2018-07-03 03:47:50 -05:00
ctx.backendSrv.datasourceRequest = jest.fn(options => {
2018-07-03 03:12:07 -05:00
return Promise.resolve({
data: {
metricbeat: {
mappings: {
metricsets: {
_all: {},
properties: {
2017-12-20 05:33:33 -06:00
'@timestamp': { type: 'date' },
beat: {
properties: {
name: {
2017-12-20 05:33:33 -06:00
fields: { raw: { type: 'keyword' } },
type: 'string',
},
2017-12-20 05:33:33 -06:00
hostname: { type: 'string' },
},
},
system: {
properties: {
cpu: {
properties: {
2017-12-20 05:33:33 -06:00
system: { type: 'float' },
user: { type: 'float' },
},
},
process: {
properties: {
cpu: {
properties: {
2017-12-20 05:33:33 -06:00
total: { type: 'float' },
},
},
2017-12-20 05:33:33 -06:00
name: { type: 'string' },
},
},
},
},
},
},
},
},
},
});
2018-07-03 03:47:50 -05:00
});
});
it('should return nested fields', () => {
ctx.ds
.getFields({
2017-12-20 05:33:33 -06:00
find: 'fields',
query: '*',
})
.then(fieldObjects => {
const fields = _.map(fieldObjects, 'text');
2018-07-03 03:12:07 -05:00
expect(fields).toEqual([
2017-12-20 05:33:33 -06:00
'@timestamp',
'beat.name.raw',
'beat.name',
'beat.hostname',
'system.cpu.system',
'system.cpu.user',
'system.process.cpu.total',
'system.process.name',
]);
});
});
it('should return fields related to query type', () => {
ctx.ds
.getFields({
2017-12-20 05:33:33 -06:00
find: 'fields',
query: '*',
type: 'number',
})
.then(fieldObjects => {
const fields = _.map(fieldObjects, 'text');
2018-07-03 03:12:07 -05:00
expect(fields).toEqual(['system.cpu.system', 'system.cpu.user', 'system.process.cpu.total']);
});
ctx.ds
.getFields({
2017-12-20 05:33:33 -06:00
find: 'fields',
query: '*',
type: 'date',
})
.then(fieldObjects => {
const fields = _.map(fieldObjects, 'text');
2018-07-03 03:12:07 -05:00
expect(fields).toEqual(['@timestamp']);
});
});
});
describe('When issuing aggregation query on es5.x', () => {
let requestOptions, parts, header;
2016-06-19 17:40:16 -05:00
beforeEach(() => {
createDatasource({
2017-12-20 05:33:33 -06:00
url: 'http://es.com',
index: 'test',
jsonData: { esVersion: '5' },
});
2016-06-19 17:40:16 -05:00
2018-07-03 03:47:50 -05:00
ctx.backendSrv.datasourceRequest = jest.fn(options => {
2016-06-19 17:40:16 -05:00
requestOptions = options;
2018-07-03 03:47:50 -05:00
return Promise.resolve({ data: { responses: [] } });
});
2016-06-19 17:40:16 -05:00
ctx.ds.query({
range: {
from: moment([2015, 4, 30, 10]),
2017-12-20 05:33:33 -06:00
to: moment([2015, 5, 1, 10]),
},
targets: [
{
bucketAggs: [{ type: 'date_histogram', field: '@timestamp', id: '2' }],
2017-12-20 05:33:33 -06:00
metrics: [{ type: 'count' }],
query: 'test',
},
],
2016-06-19 17:40:16 -05:00
});
2017-12-20 05:33:33 -06:00
parts = requestOptions.data.split('\n');
2016-06-19 17:40:16 -05:00
header = angular.fromJson(parts[0]);
});
it('should not set search type to count', () => {
2018-07-03 03:12:07 -05:00
expect(header.search_type).not.toEqual('count');
2016-06-19 17:40:16 -05:00
});
it('should set size to 0', () => {
const body = angular.fromJson(parts[1]);
2018-07-03 03:12:07 -05:00
expect(body.size).toBe(0);
2016-06-19 17:40:16 -05:00
});
});
describe('When issuing metricFind query on es5.x', () => {
let requestOptions, parts, header, body, results;
2016-06-19 17:40:16 -05:00
2018-07-03 03:47:50 -05:00
beforeEach(() => {
createDatasource({
2017-12-20 05:33:33 -06:00
url: 'http://es.com',
index: 'test',
jsonData: { esVersion: '5' },
});
2016-06-19 17:40:16 -05:00
2018-07-03 03:47:50 -05:00
ctx.backendSrv.datasourceRequest = jest.fn(options => {
2016-06-19 17:40:16 -05:00
requestOptions = options;
2018-07-03 03:47:50 -05:00
return Promise.resolve({
data: {
responses: [
{
aggregations: {
2017-12-20 05:33:33 -06:00
'1': {
buckets: [
2017-12-20 05:33:33 -06:00
{ doc_count: 1, key: 'test' },
{
doc_count: 2,
2017-12-20 05:33:33 -06:00
key: 'test2',
key_as_string: 'test2_as_string',
},
],
},
},
},
],
},
2016-06-19 17:40:16 -05:00
});
2018-07-03 03:47:50 -05:00
});
2016-06-19 17:40:16 -05:00
ctx.ds.metricFindQuery('{"find": "terms", "field": "test"}').then(res => {
results = res;
});
2017-12-20 05:33:33 -06:00
parts = requestOptions.data.split('\n');
2016-06-19 17:40:16 -05:00
header = angular.fromJson(parts[0]);
body = angular.fromJson(parts[1]);
2016-06-19 17:40:16 -05:00
});
2018-07-03 03:47:50 -05:00
it('should get results', () => {
2018-07-03 03:12:07 -05:00
expect(results.length).toEqual(2);
});
2018-07-03 03:47:50 -05:00
it('should use key or key_as_string', () => {
2018-07-03 03:12:07 -05:00
expect(results[0].text).toEqual('test');
expect(results[1].text).toEqual('test2_as_string');
});
2018-07-03 03:47:50 -05:00
it('should not set search type to count', () => {
2018-07-03 03:12:07 -05:00
expect(header.search_type).not.toEqual('count');
2016-06-19 17:40:16 -05:00
});
2018-07-03 03:47:50 -05:00
it('should set size to 0', () => {
2018-07-03 03:12:07 -05:00
expect(body.size).toBe(0);
2016-06-19 17:40:16 -05:00
});
2018-07-03 03:47:50 -05:00
it('should not set terms aggregation size to 0', () => {
2018-07-03 03:12:07 -05:00
expect(body['aggs']['1']['terms'].size).not.toBe(0);
});
2016-06-19 17:40:16 -05:00
});
});