Elasticsearch: Add query's refId to each series returned by a query (#27614)

This commit is contained in:
Giordano Ricci 2020-09-24 18:06:19 +01:00 committed by GitHub
parent 7feaddea8f
commit d4f785d20a
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 266 additions and 9 deletions

View File

@ -32,7 +32,7 @@ export class ElasticBucketAggCtrl {
); );
$scope.init = () => { $scope.init = () => {
$scope.agg = bucketAggs[$scope.index]; $scope.agg = bucketAggs[$scope.index] || {};
$scope.validateModel(); $scope.validateModel();
}; };

View File

@ -30,7 +30,7 @@ export class ElasticResponse {
switch (metric.type) { switch (metric.type) {
case 'count': { case 'count': {
newSeries = { datapoints: [], metric: 'count', props: props }; newSeries = { datapoints: [], metric: 'count', props: props, refId: target.refId };
for (i = 0; i < esAgg.buckets.length; i++) { for (i = 0; i < esAgg.buckets.length; i++) {
bucket = esAgg.buckets[i]; bucket = esAgg.buckets[i];
value = bucket.doc_count; value = bucket.doc_count;
@ -53,6 +53,7 @@ export class ElasticResponse {
metric: 'p' + percentileName, metric: 'p' + percentileName,
props: props, props: props,
field: metric.field, field: metric.field,
refId: target.refId,
}; };
for (i = 0; i < esAgg.buckets.length; i++) { for (i = 0; i < esAgg.buckets.length; i++) {
@ -76,6 +77,7 @@ export class ElasticResponse {
metric: statName, metric: statName,
props: props, props: props,
field: metric.field, field: metric.field,
refId: target.refId,
}; };
for (i = 0; i < esAgg.buckets.length; i++) { for (i = 0; i < esAgg.buckets.length; i++) {
@ -101,6 +103,7 @@ export class ElasticResponse {
field: metric.field, field: metric.field,
metricId: metric.id, metricId: metric.id,
props: props, props: props,
refId: target.refId,
}; };
for (i = 0; i < esAgg.buckets.length; i++) { for (i = 0; i < esAgg.buckets.length; i++) {
bucket = esAgg.buckets[i]; bucket = esAgg.buckets[i];
@ -200,7 +203,7 @@ export class ElasticResponse {
// This is quite complex // This is quite complex
// need to recurse down the nested buckets to build series // need to recurse down the nested buckets to build series
processBuckets(aggs: any, target: any, seriesList: any, table: any, props: any, depth: any) { processBuckets(aggs: any, target: any, seriesList: any, table: TableModel, props: any, depth: any) {
let bucket, aggDef: any, esAgg, aggId; let bucket, aggDef: any, esAgg, aggId;
const maxDepth = target.bucketAggs.length - 1; const maxDepth = target.bucketAggs.length - 1;
@ -324,12 +327,13 @@ export class ElasticResponse {
} }
} }
processHits(hits: { total: { value: any }; hits: any[] }, seriesList: any[]) { processHits(hits: { total: { value: any }; hits: any[] }, seriesList: any[], target: any) {
const hitsTotal = typeof hits.total === 'number' ? hits.total : hits.total.value; // <- Works with Elasticsearch 7.0+ const hitsTotal = typeof hits.total === 'number' ? hits.total : hits.total.value; // <- Works with Elasticsearch 7.0+
const series: any = { const series: any = {
target: 'docs', target: target.refId,
type: 'docs', type: 'docs',
refId: target.refId,
datapoints: [], datapoints: [],
total: hitsTotal, total: hitsTotal,
filterable: true, filterable: true,
@ -438,6 +442,8 @@ export class ElasticResponse {
if (isLogsRequest) { if (isLogsRequest) {
series = addPreferredVisualisationType(series, 'logs'); series = addPreferredVisualisationType(series, 'logs');
} }
const target = this.targets[n];
series.refId = target.refId;
dataFrame.push(series); dataFrame.push(series);
} }
} }
@ -453,7 +459,9 @@ export class ElasticResponse {
this.nameSeries(tmpSeriesList, target); this.nameSeries(tmpSeriesList, target);
if (table.rows.length > 0) { if (table.rows.length > 0) {
dataFrame.push(toDataFrame(table)); const series = toDataFrame(table);
series.refId = target.refId;
dataFrame.push(series);
} }
for (let y = 0; y < tmpSeriesList.length; y++) { for (let y = 0; y < tmpSeriesList.length; y++) {
@ -464,6 +472,7 @@ export class ElasticResponse {
series = addPreferredVisualisationType(series, 'graph'); series = addPreferredVisualisationType(series, 'graph');
} }
series.refId = target.refId;
dataFrame.push(series); dataFrame.push(series);
} }
} }
@ -477,19 +486,21 @@ export class ElasticResponse {
for (let i = 0; i < this.response.responses.length; i++) { for (let i = 0; i < this.response.responses.length; i++) {
const response = this.response.responses[i]; const response = this.response.responses[i];
const target = this.targets[i];
if (response.error) { if (response.error) {
throw this.getErrorFromElasticResponse(this.response, response.error); throw this.getErrorFromElasticResponse(this.response, response.error);
} }
if (response.hits && response.hits.hits.length > 0) { if (response.hits && response.hits.hits.length > 0) {
this.processHits(response.hits, seriesList); this.processHits(response.hits, seriesList, target);
} }
if (response.aggregations) { if (response.aggregations) {
const aggregations = response.aggregations; const aggregations = response.aggregations;
const target = this.targets[i];
const tmpSeriesList: any[] = []; const tmpSeriesList: any[] = [];
const table = new TableModel(); const table = new TableModel();
table.refId = target.refId;
this.processBuckets(aggregations, target, tmpSeriesList, table, {}, 0); this.processBuckets(aggregations, target, tmpSeriesList, table, {}, 0);
this.trimDatapoints(tmpSeriesList, target); this.trimDatapoints(tmpSeriesList, target);

View File

@ -3,10 +3,256 @@ import { ElasticResponse } from '../elastic_response';
import flatten from 'app/core/utils/flatten'; import flatten from 'app/core/utils/flatten';
describe('ElasticResponse', () => { describe('ElasticResponse', () => {
let targets; let targets: any;
let response: any; let response: any;
let result: any; let result: any;
describe('refId matching', () => {
// We default to the old table structure to ensure backward compatibility,
// therefore we only process responses as DataFrames when there's at least one
// raw_data (new) query type.
// We should test if refId gets populated wether there's such type of query or not
const countQuery = {
target: {
refId: 'COUNT_GROUPBY_DATE_HISTOGRAM',
metrics: [{ type: 'count', id: 'c_1' }],
bucketAggs: [{ type: 'date_histogram', field: '@timestamp', id: 'c_2' }],
},
response: {
aggregations: {
c_2: {
buckets: [
{
doc_count: 10,
key: 1000,
},
],
},
},
},
};
const countGroupByHistogramQuery = {
target: {
refId: 'COUNT_GROUPBY_HISTOGRAM',
metrics: [{ type: 'count', id: 'h_3' }],
bucketAggs: [{ type: 'histogram', field: 'bytes', id: 'h_4' }],
},
response: {
aggregations: {
h_4: {
buckets: [{ doc_count: 1, key: 1000 }],
},
},
},
};
const rawDocumentQuery = {
target: {
refId: 'RAW_DOC',
metrics: [{ type: 'raw_document', id: 'r_5' }],
bucketAggs: [],
},
response: {
hits: {
total: 2,
hits: [
{
_id: '5',
_type: 'type',
_index: 'index',
_source: { sourceProp: 'asd' },
fields: { fieldProp: 'field' },
},
{
_source: { sourceProp: 'asd2' },
fields: { fieldProp: 'field2' },
},
],
},
},
};
const percentilesQuery = {
target: {
refId: 'PERCENTILE',
metrics: [{ type: 'percentiles', settings: { percents: [75, 90] }, id: 'p_1' }],
bucketAggs: [{ type: 'date_histogram', field: '@timestamp', id: 'p_3' }],
},
response: {
aggregations: {
p_3: {
buckets: [
{
p_1: { values: { '75': 3.3, '90': 5.5 } },
doc_count: 10,
key: 1000,
},
{
p_1: { values: { '75': 2.3, '90': 4.5 } },
doc_count: 15,
key: 2000,
},
],
},
},
},
};
const extendedStatsQuery = {
target: {
refId: 'EXTENDEDSTATS',
metrics: [
{
type: 'extended_stats',
meta: { max: true, std_deviation_bounds_upper: true },
id: 'e_1',
},
],
bucketAggs: [
{ type: 'terms', field: 'host', id: 'e_3' },
{ type: 'date_histogram', id: 'e_4' },
],
},
response: {
aggregations: {
e_3: {
buckets: [
{
key: 'server1',
e_4: {
buckets: [
{
e_1: {
max: 10.2,
min: 5.5,
std_deviation_bounds: { upper: 3, lower: -2 },
},
doc_count: 10,
key: 1000,
},
],
},
},
{
key: 'server2',
e_4: {
buckets: [
{
e_1: {
max: 10.2,
min: 5.5,
std_deviation_bounds: { upper: 3, lower: -2 },
},
doc_count: 10,
key: 1000,
},
],
},
},
],
},
},
},
};
const commonTargets = [
{ ...countQuery.target },
{ ...countGroupByHistogramQuery.target },
{ ...rawDocumentQuery.target },
{ ...percentilesQuery.target },
{ ...extendedStatsQuery.target },
];
const commonResponses = [
{ ...countQuery.response },
{ ...countGroupByHistogramQuery.response },
{ ...rawDocumentQuery.response },
{ ...percentilesQuery.response },
{ ...extendedStatsQuery.response },
];
describe('When processing responses as DataFrames (raw_data query present)', () => {
beforeEach(() => {
targets = [
...commonTargets,
// Raw Data Query
{
refId: 'D',
metrics: [{ type: 'raw_data', id: '6' }],
bucketAggs: [],
},
];
response = {
responses: [
...commonResponses,
// Raw Data Query
{
hits: {
total: {
relation: 'eq',
value: 1,
},
hits: [
{
_id: '6',
_type: '_doc',
_index: 'index',
_source: { sourceProp: 'asd' },
},
],
},
},
],
};
result = new ElasticResponse(targets, response).getTimeSeries();
});
it('should add the correct refId to each returned series', () => {
expect(result.data[0].refId).toBe(countQuery.target.refId);
expect(result.data[1].refId).toBe(countGroupByHistogramQuery.target.refId);
expect(result.data[2].refId).toBe(rawDocumentQuery.target.refId);
expect(result.data[3].refId).toBe(percentilesQuery.target.refId);
expect(result.data[4].refId).toBe(percentilesQuery.target.refId);
expect(result.data[5].refId).toBe(extendedStatsQuery.target.refId);
// Raw Data query
expect(result.data[result.data.length - 1].refId).toBe('D');
});
});
describe('When NOT processing responses as DataFrames (raw_data query NOT present)', () => {
beforeEach(() => {
targets = [...commonTargets];
response = {
responses: [...commonResponses],
};
result = new ElasticResponse(targets, response).getTimeSeries();
});
it('should add the correct refId to each returned series', () => {
expect(result.data[0].refId).toBe(countQuery.target.refId);
expect(result.data[1].refId).toBe(countGroupByHistogramQuery.target.refId);
expect(result.data[2].refId).toBe(rawDocumentQuery.target.refId);
expect(result.data[3].refId).toBe(percentilesQuery.target.refId);
expect(result.data[4].refId).toBe(percentilesQuery.target.refId);
expect(result.data[5].refId).toBe(extendedStatsQuery.target.refId);
});
});
});
describe('simple query and count', () => { describe('simple query and count', () => {
beforeEach(() => { beforeEach(() => {
targets = [ targets = [