Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[Data Table] Expensive queries are causing unnecessary load and delays on Elasticsearch #98903

Merged
merged 13 commits into from
Aug 30, 2021
Merged
Original file line number Diff line number Diff line change
Expand Up @@ -42,14 +42,18 @@ const mockVisualizations = {

describe('vis_type_table getStats', () => {
const mockSoClient = ({
find: jest.fn().mockResolvedValue(mockVisualizations),
createPointInTimeFinder: jest.fn().mockResolvedValue({
find: function* asyncGenerator() {
yield mockVisualizations;
},
}),
} as unknown) as SavedObjectsClientContract;

test('Returns stats from saved objects for table vis only', async () => {
const result = await getStats(mockSoClient);
expect(mockSoClient.find).toHaveBeenCalledWith({
expect(mockSoClient.createPointInTimeFinder).toHaveBeenCalledWith({
type: 'visualization',
perPage: 10000,
perPage: 1000,
});
expect(result).toEqual({
total: 4,
Expand Down
58 changes: 40 additions & 18 deletions src/plugins/vis_type_table/server/usage_collector/get_stats.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,12 +6,16 @@
* Side Public License, v 1.
*/

import { ISavedObjectsRepository, SavedObjectsClientContract } from 'kibana/server';
import {
SavedVisState,
VisualizationSavedObjectAttributes,
} from 'src/plugins/visualizations/common';
import { TableVisParams, VIS_TYPE_TABLE } from '../../common';
ISavedObjectsRepository,
SavedObjectsClientContract,
SavedObjectsFindResult,
} from 'kibana/server';
import { SavedVisState } from 'src/plugins/visualizations/common';
import { VIS_TYPE_TABLE } from '../../common';

// elasticsearch index.max_result_window default value
const ES_MAX_RESULT_WINDOW_DEFAULT_VALUE = 1000;
alexwizp marked this conversation as resolved.
Show resolved Hide resolved

export interface VisTypeTableUsage {
/**
Expand Down Expand Up @@ -44,17 +48,13 @@ export interface VisTypeTableUsage {
export async function getStats(
soClient: SavedObjectsClientContract | ISavedObjectsRepository
): Promise<VisTypeTableUsage | undefined> {
const visualizations = await soClient.find<VisualizationSavedObjectAttributes>({
const finder = await soClient.createPointInTimeFinder({
type: 'visualization',
perPage: 10000,
perPage: ES_MAX_RESULT_WINDOW_DEFAULT_VALUE,
});

const tableVisualizations = visualizations.saved_objects
.map<SavedVisState<TableVisParams>>(({ attributes }) => JSON.parse(attributes.visState))
.filter(({ type }) => type === VIS_TYPE_TABLE);

const defaultStats = {
total: tableVisualizations.length,
const stats: VisTypeTableUsage = {
total: 0,
total_split: 0,
split_columns: {
total: 0,
Expand All @@ -66,20 +66,42 @@ export async function getStats(
},
};

return tableVisualizations.reduce((acc, { aggs, params }) => {
const doTelemetry = ({ aggs, params }: SavedVisState) => {
stats.total += 1;

const hasSplitAgg = aggs.find((agg) => agg.schema === 'split');

if (hasSplitAgg) {
acc.total_split += 1;
stats.total_split += 1;

const isSplitRow = params.row;
const isSplitEnabled = hasSplitAgg.enabled;

const container = isSplitRow ? acc.split_rows : acc.split_columns;
const container = isSplitRow ? stats.split_rows : stats.split_columns;
container.total += 1;
container.enabled = isSplitEnabled ? container.enabled + 1 : container.enabled;
}
};

for await (const response of finder.find()) {
(response.saved_objects || []).forEach(({ attributes }: SavedObjectsFindResult<any>) => {
if (attributes?.visState) {
try {
const visState: SavedVisState = JSON.parse(attributes.visState);

if (visState.type === VIS_TYPE_TABLE) {
doTelemetry(visState);
}
} catch {
// nothing to be here, "so" not valid
}
}
});

if (!response.saved_objects.length) {
await finder.close();
}
}

return acc;
}, defaultStats);
return stats;
}