diff --git a/docs/development/core/server/kibana-plugin-server.elasticsearchclientconfig.md b/docs/development/core/server/kibana-plugin-server.elasticsearchclientconfig.md index cd42a84f621a9..39e6ac428292b 100644 --- a/docs/development/core/server/kibana-plugin-server.elasticsearchclientconfig.md +++ b/docs/development/core/server/kibana-plugin-server.elasticsearchclientconfig.md @@ -4,6 +4,7 @@ ## ElasticsearchClientConfig type + Signature: ```typescript diff --git a/docs/development/core/server/kibana-plugin-server.elasticsearchservicesetup.createclient.md b/docs/development/core/server/kibana-plugin-server.elasticsearchservicesetup.createclient.md index 8b1ce883c4a14..c29d3fbbf69ab 100644 --- a/docs/development/core/server/kibana-plugin-server.elasticsearchservicesetup.createclient.md +++ b/docs/development/core/server/kibana-plugin-server.elasticsearchservicesetup.createclient.md @@ -4,8 +4,20 @@ ## ElasticsearchServiceSetup.createClient property +Create application specific Elasticsearch cluster API client with customized config. + Signature: ```typescript -readonly createClient: (type: string, config: ElasticsearchClientConfig) => ClusterClient; +readonly createClient: (type: string, clientConfig?: Partial) => ClusterClient; ``` + +## Example + + +```js +const client = elasticsearch.createCluster('my-app-name', config); +const data = await client.callAsInternalUser(); + +``` + diff --git a/docs/development/core/server/kibana-plugin-server.elasticsearchservicesetup.md b/docs/development/core/server/kibana-plugin-server.elasticsearchservicesetup.md index 3197869d6d663..a295022971a14 100644 --- a/docs/development/core/server/kibana-plugin-server.elasticsearchservicesetup.md +++ b/docs/development/core/server/kibana-plugin-server.elasticsearchservicesetup.md @@ -16,7 +16,7 @@ export interface ElasticsearchServiceSetup | Property | Type | Description | | --- | --- | --- | | [adminClient$](./kibana-plugin-server.elasticsearchservicesetup.adminclient$.md) | Observable<ClusterClient> | | -| [createClient](./kibana-plugin-server.elasticsearchservicesetup.createclient.md) | (type: string, config: ElasticsearchClientConfig) => ClusterClient | | +| [createClient](./kibana-plugin-server.elasticsearchservicesetup.createclient.md) | (type: string, clientConfig?: Partial<ElasticsearchClientConfig>) => ClusterClient | Create application specific Elasticsearch cluster API client with customized config. | | [dataClient$](./kibana-plugin-server.elasticsearchservicesetup.dataclient$.md) | Observable<ClusterClient> | | | [legacy](./kibana-plugin-server.elasticsearchservicesetup.legacy.md) | {
readonly config$: Observable<ElasticsearchConfig>;
} | | diff --git a/src/core/server/elasticsearch/elasticsearch_client_config.ts b/src/core/server/elasticsearch/elasticsearch_client_config.ts index 9fe56bf8c7e58..dcc09f711abbe 100644 --- a/src/core/server/elasticsearch/elasticsearch_client_config.ts +++ b/src/core/server/elasticsearch/elasticsearch_client_config.ts @@ -28,7 +28,7 @@ import { Logger } from '../logging'; import { ElasticsearchConfig } from './elasticsearch_config'; /** - * @internalremarks Config that consumers can pass to the Elasticsearch JS client is complex and includes + * @privateRemarks Config that consumers can pass to the Elasticsearch JS client is complex and includes * not only entries from standard `elasticsearch.*` yaml config, but also some Elasticsearch JS * client specific options like `keepAlive` or `plugins` (that eventually will be deprecated). * diff --git a/src/core/server/elasticsearch/elasticsearch_service.test.ts b/src/core/server/elasticsearch/elasticsearch_service.test.ts index a0f7180129382..e84ddb2baa30c 100644 --- a/src/core/server/elasticsearch/elasticsearch_service.test.ts +++ b/src/core/server/elasticsearch/elasticsearch_service.test.ts @@ -39,8 +39,12 @@ const deps = { configService.atPath.mockReturnValue( new BehaviorSubject({ hosts: ['http://1.2.3.4'], - healthCheck: {}, - ssl: {}, + healthCheck: { + delay: 2000, + }, + ssl: { + verificationMode: 'none', + }, } as any) ); @@ -57,7 +61,7 @@ beforeEach(() => { afterEach(() => jest.clearAllMocks()); describe('#setup', () => { - test('returns legacy Elasticsearch config as a part of the contract', async () => { + it('returns legacy Elasticsearch config as a part of the contract', async () => { const setupContract = await elasticsearchService.setup(deps); await expect(setupContract.legacy.config$.pipe(first()).toPromise()).resolves.toBeInstanceOf( @@ -65,7 +69,7 @@ describe('#setup', () => { ); }); - test('returns data and admin client observables as a part of the contract', async () => { + it('returns data and admin client observables as a part of the contract', async () => { const mockAdminClusterClientInstance = { close: jest.fn() }; const mockDataClusterClientInstance = { close: jest.fn() }; MockClusterClient.mockImplementationOnce( @@ -103,27 +107,83 @@ describe('#setup', () => { expect(mockDataClusterClientInstance.close).not.toHaveBeenCalled(); }); - test('returns `createClient` as a part of the contract', async () => { - const setupContract = await elasticsearchService.setup(deps); - - const mockClusterClientInstance = { close: jest.fn() }; - MockClusterClient.mockImplementation(() => mockClusterClientInstance); - - const mockConfig = { logQueries: true }; - const clusterClient = setupContract.createClient('some-custom-type', mockConfig as any); - - expect(clusterClient).toBe(mockClusterClientInstance); - - expect(MockClusterClient).toHaveBeenCalledWith( - mockConfig, - expect.objectContaining({ context: ['elasticsearch', 'some-custom-type'] }), - expect.any(Function) - ); + describe('#createClient', () => { + it('allows to specify config properties', async () => { + const setupContract = await elasticsearchService.setup(deps); + + const mockClusterClientInstance = { close: jest.fn() }; + MockClusterClient.mockImplementation(() => mockClusterClientInstance); + + const customConfig = { logQueries: true }; + const clusterClient = setupContract.createClient('some-custom-type', customConfig); + + expect(clusterClient).toBe(mockClusterClientInstance); + + expect(MockClusterClient).toHaveBeenCalledWith( + expect.objectContaining(customConfig), + expect.objectContaining({ context: ['elasticsearch', 'some-custom-type'] }), + expect.any(Function) + ); + }); + + it('falls back to elasticsearch default config values if property not specified', async () => { + const setupContract = await elasticsearchService.setup(deps); + // reset all mocks called during setup phase + MockClusterClient.mockClear(); + + const customConfig = { + hosts: ['http://8.8.8.8'], + logQueries: true, + ssl: { certificate: 'certificate-value' }, + }; + setupContract.createClient('some-custom-type', customConfig); + + const config = MockClusterClient.mock.calls[0][0]; + expect(config).toMatchInlineSnapshot(` +Object { + "healthCheckDelay": 2000, + "hosts": Array [ + "http://8.8.8.8", + ], + "logQueries": true, + "requestHeadersWhitelist": Array [ + undefined, + ], + "ssl": Object { + "certificate": "certificate-value", + "verificationMode": "none", + }, +} +`); + }); + it('falls back to elasticsearch config if custom config not passed', async () => { + const setupContract = await elasticsearchService.setup(deps); + // reset all mocks called during setup phase + MockClusterClient.mockClear(); + + setupContract.createClient('another-type'); + + const config = MockClusterClient.mock.calls[0][0]; + expect(config).toMatchInlineSnapshot(` +Object { + "healthCheckDelay": 2000, + "hosts": Array [ + "http://1.2.3.4", + ], + "requestHeadersWhitelist": Array [ + undefined, + ], + "ssl": Object { + "verificationMode": "none", + }, +} +`); + }); }); }); describe('#stop', () => { - test('stops both admin and data clients', async () => { + it('stops both admin and data clients', async () => { const mockAdminClusterClientInstance = { close: jest.fn() }; const mockDataClusterClientInstance = { close: jest.fn() }; MockClusterClient.mockImplementationOnce( diff --git a/src/core/server/elasticsearch/elasticsearch_service.ts b/src/core/server/elasticsearch/elasticsearch_service.ts index 4e90ff2e2baec..38a0d19b1ae3f 100644 --- a/src/core/server/elasticsearch/elasticsearch_service.ts +++ b/src/core/server/elasticsearch/elasticsearch_service.ts @@ -18,7 +18,8 @@ */ import { ConnectableObservable, Observable, Subscription } from 'rxjs'; -import { filter, map, publishReplay, switchMap } from 'rxjs/operators'; +import { filter, first, map, publishReplay, switchMap } from 'rxjs/operators'; +import { merge } from 'lodash'; import { CoreService } from '../../types'; import { CoreContext } from '../core_context'; import { Logger } from '../logging'; @@ -44,8 +45,27 @@ export interface ElasticsearchServiceSetup { readonly legacy: { readonly config$: Observable; }; - - readonly createClient: (type: string, config: ElasticsearchClientConfig) => ClusterClient; + /** + * Create application specific Elasticsearch cluster API client with customized config. + * + * @param type Unique identifier of the client + * @param clientConfig A config consists of Elasticsearch JS client options and + * valid sub-set of Elasticsearch service config. + * We fill all the missing properties in the `clientConfig` using the default + * Elasticsearch config so that we don't depend on default values set and + * controlled by underlying Elasticsearch JS client. + * We don't run validation against passed config expect it to be valid. + * + * @example + * ```js + * const client = elasticsearch.createCluster('my-app-name', config); + * const data = await client.callAsInternalUser(); + * ``` + */ + readonly createClient: ( + type: string, + clientConfig?: Partial + ) => ClusterClient; readonly adminClient$: Observable; readonly dataClient$: Observable; } @@ -101,14 +121,17 @@ export class ElasticsearchService implements CoreService clients.config)) }, adminClient$: clients$.pipe(map(clients => clients.adminClient)), dataClient$: clients$.pipe(map(clients => clients.dataClient)), - createClient: (type: string, clientConfig: ElasticsearchClientConfig) => { - return this.createClusterClient(type, clientConfig, deps.http.auth.getAuthHeaders); + createClient: (type: string, clientConfig: Partial = {}) => { + const finalConfig = merge({}, config, clientConfig); + return this.createClusterClient(type, finalConfig, deps.http.auth.getAuthHeaders); }, }; } diff --git a/src/core/server/server.api.md b/src/core/server/server.api.md index 6b1fe4622caed..c0464107210e6 100644 --- a/src/core/server/server.api.md +++ b/src/core/server/server.api.md @@ -143,8 +143,7 @@ export class ElasticsearchErrorHelpers { export interface ElasticsearchServiceSetup { // (undocumented) readonly adminClient$: Observable; - // (undocumented) - readonly createClient: (type: string, config: ElasticsearchClientConfig) => ClusterClient; + readonly createClient: (type: string, clientConfig?: Partial) => ClusterClient; // (undocumented) readonly dataClient$: Observable; // (undocumented) diff --git a/src/legacy/core_plugins/elasticsearch/index.js b/src/legacy/core_plugins/elasticsearch/index.js index 8dc9c77e79d99..51dac9e3a4682 100644 --- a/src/legacy/core_plugins/elasticsearch/index.js +++ b/src/legacy/core_plugins/elasticsearch/index.js @@ -77,13 +77,7 @@ export default function (kibana) { throw new Error(`cluster '${name}' already exists`); } - // We fill all the missing properties in the `clientConfig` using the default - // Elasticsearch config so that we don't depend on default values set and - // controlled by underlying Elasticsearch JS client. - const cluster = new Cluster(server.newPlatform.setup.core.elasticsearch.createClient(name, { - ...esConfig, - ...clientConfig, - })); + const cluster = new Cluster(server.newPlatform.setup.core.elasticsearch.createClient(name, clientConfig)); clusters.set(name, cluster);