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 views, getExistingIndices method to server side #168522

Merged
merged 12 commits into from
Oct 16, 2023
6 changes: 6 additions & 0 deletions src/plugins/data_views/common/constants.ts
Original file line number Diff line number Diff line change
Expand Up @@ -57,3 +57,9 @@ export const PLUGIN_NAME = 'DataViews';
* @public
*/
export const FIELDS_FOR_WILDCARD_PATH = '/internal/data_views/_fields_for_wildcard';

/**
* Existing indices path
* @public
*/
export const EXISTING_INDICES_PATH = '/internal/data_views/_existing_indices';
32 changes: 0 additions & 32 deletions src/plugins/data_views/common/data_views/data_views.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -662,36 +662,4 @@ describe('IndexPatterns', () => {
expect(apiClient.getFieldsForWildcard.mock.calls[0][0].allowNoIndex).toBe(true);
});
});

describe('getExistingIndices', () => {
test('getExistingIndices returns the valid matched indices', async () => {
apiClient.getFieldsForWildcard = jest
.fn()
.mockResolvedValueOnce({ fields: ['length'] })
.mockResolvedValue({ fields: [] });
const patternList = await indexPatterns.getExistingIndices(['packetbeat-*', 'filebeat-*']);
expect(apiClient.getFieldsForWildcard).toBeCalledTimes(2);
expect(patternList.length).toBe(1);
});

test('getExistingIndices checks the positive pattern if provided with a negative pattern', async () => {
const mockFn = jest.fn().mockResolvedValue({ fields: ['length'] });
apiClient.getFieldsForWildcard = mockFn;
const patternList = await indexPatterns.getExistingIndices(['-filebeat-*', 'filebeat-*']);
expect(mockFn.mock.calls[0][0].pattern).toEqual('filebeat-*');
expect(mockFn.mock.calls[1][0].pattern).toEqual('filebeat-*');
expect(patternList).toEqual(['-filebeat-*', 'filebeat-*']);
});

test('getExistingIndices handles an error', async () => {
apiClient.getFieldsForWildcard = jest
.fn()
.mockImplementationOnce(async () => {
throw new DataViewMissingIndices('Catch me if you can!');
})
.mockImplementation(() => Promise.resolve({ fields: ['length'] }));
const patternList = await indexPatterns.getExistingIndices(['packetbeat-*', 'filebeat-*']);
expect(patternList).toEqual(['filebeat-*']);
});
});
});
43 changes: 0 additions & 43 deletions src/plugins/data_views/common/data_views/data_views.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,12 +7,10 @@
*/

import { i18n } from '@kbn/i18n';
import { defer, from } from 'rxjs';
import type { PublicMethodsOf } from '@kbn/utility-types';
import { castEsToKbnFieldTypeName } from '@kbn/field-types';
import { FieldFormatsStartCommon, FORMATS_UI_SETTINGS } from '@kbn/field-formats-plugin/common';
import { v4 as uuidv4 } from 'uuid';
import { rateLimitingForkJoin } from './utils';
import { PersistenceAPI } from '../types';

import { createDataViewCache } from '.';
Expand Down Expand Up @@ -240,12 +238,6 @@ export interface DataViewsServicePublicMethods {
* @param options - options for getting fields
*/
getFieldsForWildcard: (options: GetFieldsOptions) => Promise<FieldSpec[]>;
/**
* Get existing index pattern list by providing string array index pattern list.
* @param indices - index pattern list
* @returns index pattern list of index patterns that match indices
*/
getExistingIndices: (indices: string[]) => Promise<string[]>;
/**
* Get list of data view ids.
* @param refresh - clear cache and fetch from server
Expand Down Expand Up @@ -524,41 +516,6 @@ export class DataViewsService {
return fields;
};

/**
* Get existing index pattern list by providing string array index pattern list.
* @param indices index pattern list
* @returns index pattern list
*/
getExistingIndices = async (indices: string[]): Promise<string[]> => {
const indicesObs = indices.map((pattern) => {
// when checking a negative pattern, check if the positive pattern exists
const indexToQuery = pattern.trim().startsWith('-')
? pattern.trim().substring(1)
: pattern.trim();
return defer(() =>
from(
this.getFieldsForWildcard({
// check one field to keep request fast/small
fields: ['_id'],
// true so no errors thrown in browser
allowNoIndex: true,
pattern: indexToQuery,
})
)
);
});

return new Promise<boolean[]>((resolve) => {
rateLimitingForkJoin(indicesObs, 3, []).subscribe((value) => {
resolve(value.map((v) => v.length > 0));
});
})
.then((allPatterns: boolean[]) =>
indices.filter((pattern, i, self) => self.indexOf(pattern) === i && allPatterns[i])
)
.catch(() => indices);
};

/**
* Get field list by providing an index patttern (or spec).
* @param options options for getting field list
Expand Down
5 changes: 5 additions & 0 deletions src/plugins/data_views/common/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -324,6 +324,11 @@ export interface FieldsForWildcardResponse {
indices: string[];
}

/**
* Existing Indices response
*/
export type ExistingIndicesResponse = string[];

export interface IDataViewsApiClient {
getFieldsForWildcard: (options: GetFieldsOptions) => Promise<FieldsForWildcardResponse>;
hasUserDataView: () => Promise<boolean>;
Expand Down
19 changes: 19 additions & 0 deletions src/plugins/data_views/public/data_views_service_public.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,11 +6,15 @@
* Side Public License, v 1.
*/

import { HttpStart } from '@kbn/core/public';
import { DataViewsService, MatchedItem } from '.';

import { DataViewsServiceDeps } from '../common/data_views/data_views';
import { HasDataService } from '../common';

import { ExistingIndicesResponse } from '../common/types';
import { EXISTING_INDICES_PATH } from '../common/constants';

/**
* Data Views public service dependencies
* @public
Expand All @@ -32,6 +36,7 @@ export interface DataViewsServicePublicDeps extends DataViewsServiceDeps {

getRollupsEnabled: () => boolean;
scriptedFieldsEnabled: boolean;
http: HttpStart;
}

/**
Expand All @@ -48,6 +53,7 @@ export class DataViewsServicePublic extends DataViewsService {
}) => Promise<MatchedItem[]>;
public hasData: HasDataService;
private rollupsEnabled: boolean = false;
private readonly http: HttpStart;
public readonly scriptedFieldsEnabled: boolean;

/**
Expand All @@ -62,9 +68,22 @@ export class DataViewsServicePublic extends DataViewsService {
this.getIndices = deps.getIndices;
this.rollupsEnabled = deps.getRollupsEnabled();
this.scriptedFieldsEnabled = deps.scriptedFieldsEnabled;
this.http = deps.http;
}

getRollupsEnabled() {
return this.rollupsEnabled;
}

/**
* Get existing index pattern list by providing string array index pattern list.
* @param indices - index pattern list
* @returns index pattern list of index patterns that match indices
*/
async getExistingIndices(indices: string[]): Promise<ExistingIndicesResponse> {
return this.http.get<ExistingIndicesResponse>(EXISTING_INDICES_PATH, {
query: { indices },
version: '1',
});
}
}
1 change: 1 addition & 0 deletions src/plugins/data_views/public/plugin.ts
Original file line number Diff line number Diff line change
Expand Up @@ -88,6 +88,7 @@ export class DataViewsPublicPlugin
savedObjectsClient: new ContentMagementWrapper(contentManagement.client),
apiClient: new DataViewsApiClient(http),
fieldFormats,
http,
onNotification: (toastInputFields, key) => {
onNotifDebounced(key)(toastInputFields);
},
Expand Down
6 changes: 6 additions & 0 deletions src/plugins/data_views/public/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -123,6 +123,12 @@ export interface DataViewsServicePublic extends DataViewsServicePublicMethods {
}) => Promise<MatchedItem[]>;
getRollupsEnabled: () => boolean;
scriptedFieldsEnabled: boolean;
/**
* Get existing index pattern list by providing string array index pattern list.
* @param indices - index pattern list
* @returns index pattern list of index patterns that match indices
*/
getExistingIndices: (indices: string[]) => Promise<string[]>;
}

export type DataViewsContract = DataViewsServicePublic;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@
import * as estypes from '@elastic/elasticsearch/lib/api/typesWithBodyKey';
import { IndexPatternsFetcher } from '.';
import { elasticsearchServiceMock } from '@kbn/core/server/mocks';
import { DataViewMissingIndices } from '../../common';

const rollupResponse = {
foo: {
Expand All @@ -34,17 +35,16 @@ describe('Index Pattern Fetcher - server', () => {
beforeEach(() => {
jest.clearAllMocks();
esClient = elasticsearchServiceMock.createElasticsearchClient();
esClient.fieldCaps.mockResponse(response as unknown as estypes.FieldCapsResponse);
indexPatterns = new IndexPatternsFetcher(esClient, false, true);
});
it('calls fieldcaps once', async () => {
esClient.fieldCaps.mockResponse(response as unknown as estypes.FieldCapsResponse);
indexPatterns = new IndexPatternsFetcher(esClient, true, true);
await indexPatterns.getFieldsForWildcard({ pattern: patternList });
expect(esClient.fieldCaps).toHaveBeenCalledTimes(1);
});

it('calls rollup api when given rollup data view', async () => {
esClient.fieldCaps.mockResponse(response as unknown as estypes.FieldCapsResponse);
esClient.rollup.getRollupIndexCaps.mockResponse(
rollupResponse as unknown as estypes.RollupGetRollupIndexCapsResponse
);
Expand All @@ -58,7 +58,6 @@ describe('Index Pattern Fetcher - server', () => {
});

it("doesn't call rollup api when given rollup data view and rollups are disabled", async () => {
esClient.fieldCaps.mockResponse(response as unknown as estypes.FieldCapsResponse);
esClient.rollup.getRollupIndexCaps.mockResponse(
rollupResponse as unknown as estypes.RollupGetRollupIndexCapsResponse
);
Expand All @@ -70,4 +69,39 @@ describe('Index Pattern Fetcher - server', () => {
});
expect(esClient.rollup.getRollupIndexCaps).toHaveBeenCalledTimes(0);
});

describe('getExistingIndices', () => {
test('getExistingIndices returns the valid matched indices', async () => {
indexPatterns = new IndexPatternsFetcher(esClient, true, true);
indexPatterns.getFieldsForWildcard = jest
.fn()
.mockResolvedValueOnce({ indices: ['length'] })
.mockResolvedValue({ indices: [] });
const result = await indexPatterns.getExistingIndices(['packetbeat-*', 'filebeat-*']);
expect(indexPatterns.getFieldsForWildcard).toBeCalledTimes(2);
expect(result.length).toBe(1);
});

test('getExistingIndices checks the positive pattern if provided with a negative pattern', async () => {
indexPatterns = new IndexPatternsFetcher(esClient, true, true);
const mockFn = jest.fn().mockResolvedValue({ indices: ['length'] });
indexPatterns.getFieldsForWildcard = mockFn;
const result = await indexPatterns.getExistingIndices(['-filebeat-*', 'filebeat-*']);
expect(mockFn.mock.calls[0][0].pattern).toEqual('filebeat-*');
expect(mockFn.mock.calls[1][0].pattern).toEqual('filebeat-*');
expect(result).toEqual(['-filebeat-*', 'filebeat-*']);
});

test('getExistingIndices handles an error', async () => {
indexPatterns = new IndexPatternsFetcher(esClient, true, true);
indexPatterns.getFieldsForWildcard = jest
.fn()
.mockImplementationOnce(async () => {
throw new DataViewMissingIndices('Catch me if you can!');
})
.mockImplementation(() => Promise.resolve({ indices: ['length'] }));
const result = await indexPatterns.getExistingIndices(['packetbeat-*', 'filebeat-*']);
expect(result).toEqual(['filebeat-*']);
});
});
});
35 changes: 35 additions & 0 deletions src/plugins/data_views/server/fetcher/index_patterns_fetcher.ts
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,8 @@
import * as estypes from '@elastic/elasticsearch/lib/api/typesWithBodyKey';
import { ElasticsearchClient } from '@kbn/core/server';
import { keyBy } from 'lodash';
import { defer, from } from 'rxjs';
import { rateLimitingForkJoin } from '../../common/data_views/utils';
import type { QueryDslQueryContainer } from '../../common/types';

import {
Expand Down Expand Up @@ -129,4 +131,37 @@ export class IndexPatternsFetcher {
}
return fieldCapsResponse;
}

/**
* Get existing index pattern list by providing string array index pattern list.
* @param indices - index pattern list
* @returns index pattern list of index patterns that match indices
*/
async getExistingIndices(indices: string[]): Promise<string[]> {
const indicesObs = indices.map((pattern) => {
// when checking a negative pattern, check if the positive pattern exists
const indexToQuery = pattern.trim().startsWith('-')
? pattern.trim().substring(1)
: pattern.trim();
return defer(() =>
from(
this.getFieldsForWildcard({
// check one field to keep request fast/small
fields: ['_id'],
pattern: indexToQuery,
})
)
);
});

return new Promise<boolean[]>((resolve) => {
rateLimitingForkJoin(indicesObs, 3, { fields: [], indices: [] }).subscribe((value) => {
resolve(value.map((v) => v.indices.length > 0));
});
})
.then((allPatterns: boolean[]) =>
indices.filter((pattern, i, self) => self.indexOf(pattern) === i && allPatterns[i])
)
.catch(() => indices);
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,53 @@
/*
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
* or more contributor license agreements. Licensed under the Elastic License
* 2.0 and the Server Side Public License, v 1; you may not use this file except
* in compliance with, at your election, the Elastic License 2.0 or the Server
* Side Public License, v 1.
*/

import { parseIndices } from './existing_indices';

describe('parseIndices', () => {
it('should return an array of indices when given an array of indices', () => {
const indices = ['index1', 'index2', 'index3'];
const result = parseIndices(indices);
expect(result).toEqual(indices);
});

it('should return an array of indices when given a JSON-stringified array of indices', () => {
const indices = '["index1", "index2", "index3"]';
const result = parseIndices(indices);
expect(result).toEqual(['index1', 'index2', 'index3']);
});

it('should return an array with a single index when given a single index name', () => {
const index = 'index1';
const result = parseIndices(index);
expect(result).toEqual(['index1']);
});

it('should throw an error when given an invalid JSON-stringified array', () => {
const indices = '["index1", "index2"';
expect(() => {
parseIndices(indices);
}).toThrowError(
'indices should be an array of index aliases, a JSON-stringified array of index aliases, or a single index alias'
);
});

it('should throw an error when given a string with a comma but not JSON-stringified', () => {
const indices = 'index1,index2';
expect(() => {
parseIndices(indices);
}).toThrowError(
'indices should be an array of index aliases, a JSON-stringified array of index aliases, or a single index alias'
);
});

it('should return an empty array when given an empty array', () => {
const indices: string[] = [];
const result = parseIndices(indices);
expect(result).toEqual([]);
});
});
Loading
Loading