From 897aa9415d1bd269b3ed455e4f0601e2cda2d727 Mon Sep 17 00:00:00 2001 From: Tim Sullivan <tsullivan@users.noreply.github.com> Date: Mon, 10 Dec 2018 13:34:18 -0700 Subject: [PATCH] [Telemetry] Pull local Kibana usage stats (#26496) (#26913) * add kibana stats * fix tests * format the stats for telemetry * fix the os/platform stats * add version to locally-source kibana telemetry stats * use callWithInternalUser * better get_kibana module unit test verification * separate handleKibanaStats * variable rename * fix comment * fix functional test * keep the return object literal from handleLocalStats * validate the payload fields * add warning log if no kibana stats returned --- .../local/__tests__/get_local_stats.js | 162 +++++--- .../telemetry/local/__tests__/get_xpack.js | 6 +- .../server/lib/telemetry/local/get_kibana.js | 49 +++ .../lib/telemetry/local/get_local_stats.js | 26 +- .../server/lib/telemetry/local/get_xpack.js | 7 +- .../apis/xpack_main/telemetry/index.js | 1 + .../xpack_main/telemetry/telemetry_local.js | 377 ++++++++++++++++++ 7 files changed, 551 insertions(+), 77 deletions(-) create mode 100644 x-pack/plugins/xpack_main/server/lib/telemetry/local/get_kibana.js create mode 100644 x-pack/test/api_integration/apis/xpack_main/telemetry/telemetry_local.js diff --git a/x-pack/plugins/xpack_main/server/lib/telemetry/local/__tests__/get_local_stats.js b/x-pack/plugins/xpack_main/server/lib/telemetry/local/__tests__/get_local_stats.js index 9b638785f5dd9..eca758d1a2d8a 100644 --- a/x-pack/plugins/xpack_main/server/lib/telemetry/local/__tests__/get_local_stats.js +++ b/x-pack/plugins/xpack_main/server/lib/telemetry/local/__tests__/get_local_stats.js @@ -18,18 +18,24 @@ import { handleLocalStats, } from '../get_local_stats'; +const getMockServer = (getCluster = sinon.stub(), kibanaUsage = {}) => ({ + log(tags, message) { + console.log({ tags, message }); + }, + usage: { collectorSet: { bulkFetch: () => kibanaUsage, toObject: data => data } }, + plugins: { + xpack_main: { status: { plugin: { kbnServer: { version: '8675309-snapshot' } } } }, + elasticsearch: { getCluster }, + }, +}); + function mockGetLocalStats(callCluster, clusterInfo, clusterStats, license, usage) { mockGetClusterInfo(callCluster, clusterInfo); mockGetClusterStats(callCluster, clusterStats); mockGetXPack(callCluster, license, usage); } -function dropTimestamp(localStats) { - return omit(localStats, 'timestamp'); -} - describe('get_local_stats', () => { - const clusterUuid = 'abc123'; const clusterName = 'my-cool-cluster'; const version = '2.3.4'; @@ -47,102 +53,127 @@ describe('get_local_stats', () => { nodes: { yup: 'abc' }, random: 123 }; - const license = { - fancy: 'license' - }; - const usage = { - also: 'fancy' + const license = { fancy: 'license' }; + const xpack = { also: 'fancy' }; + const kibana = { + kibana: { + great: 'googlymoogly', + versions: [{ version: '8675309', count: 1 }] + }, + kibana_stats: { + os: { + platform: 'rocky', + platformRelease: 'iv', + } + }, + sun: { chances: 5 }, + clouds: { chances: 95 }, + rain: { chances: 2 }, + snow: { chances: 0 }, }; - const xpack = { - license, - stack_stats: { - xpack: usage - } - }; - const localStats = { + + const combinedStatsResult = { collection: 'local', cluster_uuid: clusterUuid, cluster_name: clusterName, + license: { + fancy: 'license' + }, version, cluster_stats: omit(clusterStats, '_nodes', 'cluster_name'), - ...xpack + stack_stats: { + kibana: { + great: 'googlymoogly', + count: 1, + indices: 1, + os: { + platforms: [{ platform: 'rocky', count: 1 }], + platformReleases: [{ platformRelease: 'iv', count: 1 }] + }, + versions: [{ version: '8675309', count: 1 }], + plugins: { + sun: { chances: 5 }, + clouds: { chances: 95 }, + rain: { chances: 2 }, + snow: { chances: 0 }, + } + }, + xpack: { also: 'fancy' }, + } }; - const noXpackLocalStats = omit(localStats, 'license', 'stack_stats'); describe('handleLocalStats', () => { - - it('returns expected object without xpack data', () => { - expect(dropTimestamp(handleLocalStats(clusterInfo, clusterStats))).to.eql(noXpackLocalStats); - expect(dropTimestamp(handleLocalStats(clusterInfo, clusterStats, { }))).to.eql(noXpackLocalStats); + it('returns expected object without xpack and kibana data', () => { + const result = handleLocalStats(getMockServer(), clusterInfo, clusterStats); + expect(result.cluster_uuid).to.eql(combinedStatsResult.cluster_uuid); + expect(result.cluster_name).to.eql(combinedStatsResult.cluster_name); + expect(result.cluster_stats).to.eql(combinedStatsResult.cluster_stats); + expect(result.version).to.be('2.3.4'); + expect(result.collection).to.be('local'); + expect(result.license).to.be(undefined); + expect(result.stack_stats).to.eql({ kibana: undefined, xpack: undefined }); }); - it('returns expected object with xpack data', () => { - expect(dropTimestamp(handleLocalStats(clusterInfo, clusterStats, xpack))).to.eql(localStats); + it('returns expected object with xpack', () => { + const result = handleLocalStats(getMockServer(), clusterInfo, clusterStats, license, xpack); + const { stack_stats: stack, ...cluster } = result; + expect(cluster.collection).to.be(combinedStatsResult.collection); + expect(cluster.cluster_uuid).to.be(combinedStatsResult.cluster_uuid); + expect(cluster.cluster_name).to.be(combinedStatsResult.cluster_name); + expect(stack.kibana).to.be(undefined); // not mocked for this test + + expect(cluster.version).to.eql(combinedStatsResult.version); + expect(cluster.cluster_stats).to.eql(combinedStatsResult.cluster_stats); + expect(cluster.license).to.eql(combinedStatsResult.license); + expect(stack.xpack).to.eql(combinedStatsResult.stack_stats.xpack); }); - }); describe('getLocalStatsWithCaller', () => { - it('returns expected object without xpack data when X-Pack fails to respond', async () => { const callClusterUsageFailed = sinon.stub(); - const callClusterLicenseFailed = sinon.stub(); - const callClusterBothFailed = sinon.stub(); mockGetLocalStats( callClusterUsageFailed, Promise.resolve(clusterInfo), Promise.resolve(clusterStats), - Promise.resolve(license), Promise.reject('usage failed') + Promise.resolve(license), + Promise.reject('usage failed') ); - mockGetLocalStats( - callClusterLicenseFailed, - Promise.resolve(clusterInfo), - Promise.resolve(clusterStats), - Promise.reject('license failed'), Promise.resolve(usage) - ); + const result = await getLocalStatsWithCaller(getMockServer(), callClusterUsageFailed); + expect(result.cluster_uuid).to.eql(combinedStatsResult.cluster_uuid); + expect(result.cluster_name).to.eql(combinedStatsResult.cluster_name); + expect(result.cluster_stats).to.eql(combinedStatsResult.cluster_stats); + expect(result.version).to.be('2.3.4'); + expect(result.collection).to.be('local'); - mockGetLocalStats( - callClusterBothFailed, - Promise.resolve(clusterInfo), - Promise.resolve(clusterStats), - Promise.reject('license failed'), Promise.reject('usage failed') - ); - - expect(dropTimestamp(await getLocalStatsWithCaller(callClusterUsageFailed))).to.eql(noXpackLocalStats); - expect(dropTimestamp(await getLocalStatsWithCaller(callClusterLicenseFailed))).to.eql(noXpackLocalStats); - expect(dropTimestamp(await getLocalStatsWithCaller(callClusterBothFailed))).to.eql(noXpackLocalStats); + // license and xpack usage info come from the same cluster call + expect(result.license).to.be(undefined); + expect(result.stack_stats.xpack).to.be(undefined); }); - it('returns expected object with xpack data', async () => { + it('returns expected object with xpack and kibana data', async () => { const callCluster = sinon.stub(); mockGetLocalStats( callCluster, Promise.resolve(clusterInfo), Promise.resolve(clusterStats), - Promise.resolve(license), Promise.resolve(usage) + Promise.resolve(license), + Promise.resolve(xpack) ); - expect(dropTimestamp(await getLocalStatsWithCaller(callCluster))).to.eql(localStats); + const result = await getLocalStatsWithCaller(getMockServer(callCluster, kibana), callCluster); + expect(result.stack_stats.xpack).to.eql(combinedStatsResult.stack_stats.xpack); + expect(result.stack_stats.kibana).to.eql(combinedStatsResult.stack_stats.kibana); }); - }); describe('getLocalStats', () => { - it('uses callWithInternalUser from data cluster', async () => { const getCluster = sinon.stub(); - const req = { - server: { - plugins: { - elasticsearch: { - getCluster - } - } - } - }; + const req = { server: getMockServer(getCluster) }; const callWithInternalUser = sinon.stub(); getCluster.withArgs('data').returns({ callWithInternalUser }); @@ -151,12 +182,15 @@ describe('get_local_stats', () => { callWithInternalUser, Promise.resolve(clusterInfo), Promise.resolve(clusterStats), - Promise.resolve(license), Promise.resolve(usage) + Promise.resolve(license), + Promise.resolve(xpack) ); - expect(dropTimestamp(await getLocalStats(req))).to.eql(localStats); + const result = await getLocalStats(req); + expect(result.cluster_uuid).to.eql(combinedStatsResult.cluster_uuid); + expect(result.cluster_name).to.eql(combinedStatsResult.cluster_name); + expect(result.version).to.eql(combinedStatsResult.version); + expect(result.cluster_stats).to.eql(combinedStatsResult.cluster_stats); }); - }); - }); diff --git a/x-pack/plugins/xpack_main/server/lib/telemetry/local/__tests__/get_xpack.js b/x-pack/plugins/xpack_main/server/lib/telemetry/local/__tests__/get_xpack.js index eee2a15fe85ac..4112f91cb7238 100644 --- a/x-pack/plugins/xpack_main/server/lib/telemetry/local/__tests__/get_xpack.js +++ b/x-pack/plugins/xpack_main/server/lib/telemetry/local/__tests__/get_xpack.js @@ -93,15 +93,15 @@ describe('get_xpack', () => { it('returns the formatted response object', async () => { const license = { fancy: 'license' }; - const usage = { also: 'fancy' }; + const xpack = { also: 'fancy' }; const callCluster = sinon.stub(); - mockGetXPack(callCluster, Promise.resolve(license), Promise.resolve(usage)); + mockGetXPack(callCluster, Promise.resolve(license), Promise.resolve(xpack)); const data = await getXPack(callCluster); - expect(data).to.eql({ license, stack_stats: { xpack: usage } }); + expect(data).to.eql({ license, xpack }); }); it('returns empty object upon license failure', async () => { diff --git a/x-pack/plugins/xpack_main/server/lib/telemetry/local/get_kibana.js b/x-pack/plugins/xpack_main/server/lib/telemetry/local/get_kibana.js new file mode 100644 index 0000000000000..50e68e2b74d68 --- /dev/null +++ b/x-pack/plugins/xpack_main/server/lib/telemetry/local/get_kibana.js @@ -0,0 +1,49 @@ +/* + * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one + * or more contributor license agreements. Licensed under the Elastic License; + * you may not use this file except in compliance with the Elastic License. + */ + +import { get, omit } from 'lodash'; + +export function handleKibanaStats(server, response) { + if (!response) { + server.log(['warning', 'telemetry', 'local-stats'], 'No Kibana stats returned from usage collectors'); + return; + } + + const { kibana, kibana_stats: stats, ...plugins } = response; + + const platform = get(stats, 'os.platform', 'unknown'); + const platformRelease = get(stats, 'os.platformRelease', 'unknown'); + + let version; + const { kbnServer } = get(server, 'plugins.xpack_main.status.plugin'); + if (kbnServer) { + version = kbnServer.version.replace(/-snapshot/i, ''); + } + + // combine core stats (os types, saved objects) with plugin usage stats + // organize the object into the same format as monitoring-enabled telemetry + return { + ...omit(kibana, 'index'), // discard index + count: 1, + indices: 1, + os: { + platforms: [{ platform, count: 1 }], + platformReleases: [{ platformRelease, count: 1 }], + }, + versions: [{ version, count: 1 }], + plugins, + }; +} + +/* + * Check user privileges for read access to monitoring + * Pass callWithInternalUser to bulkFetchUsage + */ +export async function getKibana(server, callWithInternalUser) { + const { collectorSet } = server.usage; + const usage = await collectorSet.bulkFetch(callWithInternalUser); + return collectorSet.toObject(usage); +} diff --git a/x-pack/plugins/xpack_main/server/lib/telemetry/local/get_local_stats.js b/x-pack/plugins/xpack_main/server/lib/telemetry/local/get_local_stats.js index 2cb796943989e..3f040c9fc8922 100644 --- a/x-pack/plugins/xpack_main/server/lib/telemetry/local/get_local_stats.js +++ b/x-pack/plugins/xpack_main/server/lib/telemetry/local/get_local_stats.js @@ -8,6 +8,7 @@ import { get, omit } from 'lodash'; import { getClusterInfo } from './get_cluster_info'; import { getClusterStats } from './get_cluster_stats'; import { getXPack } from './get_xpack'; +import { getKibana, handleKibanaStats } from './get_kibana'; /** * Handle the separate local calls by combining them into a single object response that looks like the @@ -18,7 +19,7 @@ import { getXPack } from './get_xpack'; * @param {Object} xpack License and X-Pack details * @return {Object} A combined object containing the different responses. */ -export function handleLocalStats(clusterInfo, clusterStats, xpack) { +export function handleLocalStats(server, clusterInfo, clusterStats, license, xpack, kibana) { return { timestamp: (new Date()).toISOString(), cluster_uuid: get(clusterInfo, 'cluster_uuid'), @@ -26,7 +27,11 @@ export function handleLocalStats(clusterInfo, clusterStats, xpack) { version: get(clusterInfo, 'version.number'), cluster_stats: omit(clusterStats, '_nodes', 'cluster_name'), collection: 'local', - ...xpack + license, + stack_stats: { + kibana: handleKibanaStats(server, kibana), + xpack, + } }; } @@ -37,13 +42,16 @@ export function handleLocalStats(clusterInfo, clusterStats, xpack) { * @param {function} callCluster The callWithInternalUser handler (exposed for testing) * @return {Promise} The object containing the current Elasticsearch cluster's telemetry. */ -export function getLocalStatsWithCaller(callCluster) { +export function getLocalStatsWithCaller(server, callCluster) { return Promise.all([ getClusterInfo(callCluster), // cluster info getClusterStats(callCluster), // cluster stats (not to be confused with cluster _state_) - getXPack(callCluster), // license, stack_stats - ]) - .then(([clusterInfo, clusterStats, xpack]) => handleLocalStats(clusterInfo, clusterStats, xpack)); + getXPack(callCluster), // { license, xpack } + getKibana(server, callCluster) + ]).then(([clusterInfo, clusterStats, { license, xpack }, kibana]) => { + return handleLocalStats(server, clusterInfo, clusterStats, license, xpack, kibana); + } + ); } /** @@ -53,7 +61,7 @@ export function getLocalStatsWithCaller(callCluster) { * @return {Promise} The cluster object containing telemetry. */ export function getLocalStats(req) { - const { callWithInternalUser } = req.server.plugins.elasticsearch.getCluster('data'); - - return getLocalStatsWithCaller(callWithInternalUser); + const { server } = req; + const { callWithInternalUser } = server.plugins.elasticsearch.getCluster('data'); + return getLocalStatsWithCaller(server, callWithInternalUser); } diff --git a/x-pack/plugins/xpack_main/server/lib/telemetry/local/get_xpack.js b/x-pack/plugins/xpack_main/server/lib/telemetry/local/get_xpack.js index 9c71c6a2b7f29..3ad1a15b45719 100644 --- a/x-pack/plugins/xpack_main/server/lib/telemetry/local/get_xpack.js +++ b/x-pack/plugins/xpack_main/server/lib/telemetry/local/get_xpack.js @@ -75,7 +75,12 @@ export function getXPack(callCluster) { getXPackLicense(callCluster), getXPackUsage(callCluster), ]) - .then(([license, usage]) => handleXPack(license, usage)) + .then(([license, xpack]) => { + return { + license, + xpack, + }; + }) .catch(() => { return {}; }); } diff --git a/x-pack/test/api_integration/apis/xpack_main/telemetry/index.js b/x-pack/test/api_integration/apis/xpack_main/telemetry/index.js index 3f47fdd99bf91..d941cda9e3fae 100644 --- a/x-pack/test/api_integration/apis/xpack_main/telemetry/index.js +++ b/x-pack/test/api_integration/apis/xpack_main/telemetry/index.js @@ -7,5 +7,6 @@ export default function ({ loadTestFile }) { describe('Telemetry', () => { loadTestFile(require.resolve('./telemetry')); + loadTestFile(require.resolve('./telemetry_local')); }); } diff --git a/x-pack/test/api_integration/apis/xpack_main/telemetry/telemetry_local.js b/x-pack/test/api_integration/apis/xpack_main/telemetry/telemetry_local.js new file mode 100644 index 0000000000000..6bdbfdb527a6e --- /dev/null +++ b/x-pack/test/api_integration/apis/xpack_main/telemetry/telemetry_local.js @@ -0,0 +1,377 @@ +/* + * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one + * or more contributor license agreements. Licensed under the Elastic License; + * you may not use this file except in compliance with the Elastic License. + */ + +import expect from 'expect.js'; + +/* + * Create a single-level array with strings for all the paths to values in the + * source object + */ +const flatKeys = (source, path = [], results = []) => { + Object.keys(source).forEach(key => { + if (typeof source[key] === 'object' && source[key] != null) { + results = flatKeys(source[key], [...path, key], results); + } else { + results = [].concat(results, [...path, key].join('.')); + } + }); + return results; +}; + +const disableCollection = { + "persistent": + { + xpack: { + monitoring: { + collection: { + enabled: false + } + } + } + } +}; + +export default function ({ getService }) { + const supertest = getService('supertest'); + const esSupertest = getService('esSupertest'); + + describe('/api/telemetry/v1/clusters/_stats with monitoring disabled', () => { + before('', async () => { + await esSupertest.put('/_cluster/settings').send(disableCollection).expect(200); + }); + + it('should pull local stats and validate fields', async () => { + const timeRange = { + min: '2018-07-23T22:07:00Z', + max: '2018-07-23T22:13:00Z' + }; + + const { body } = await supertest + .post('/api/telemetry/v1/clusters/_stats') + .set('kbn-xsrf', 'xxx') + .send({ timeRange }) + .expect(200); + + const stats = body[0]; + const actual = flatKeys(stats).sort(); + const expected = [ + 'cluster_name', + 'cluster_stats.cluster_uuid', + 'cluster_stats.indices.completion.size_in_bytes', + 'cluster_stats.indices.count', + 'cluster_stats.indices.docs.count', + 'cluster_stats.indices.docs.deleted', + 'cluster_stats.indices.fielddata.evictions', + 'cluster_stats.indices.fielddata.memory_size_in_bytes', + 'cluster_stats.indices.query_cache.cache_count', + 'cluster_stats.indices.query_cache.cache_size', + 'cluster_stats.indices.query_cache.evictions', + 'cluster_stats.indices.query_cache.hit_count', + 'cluster_stats.indices.query_cache.memory_size_in_bytes', + 'cluster_stats.indices.query_cache.miss_count', + 'cluster_stats.indices.query_cache.total_count', + 'cluster_stats.indices.segments.count', + 'cluster_stats.indices.segments.doc_values_memory_in_bytes', + 'cluster_stats.indices.segments.fixed_bit_set_memory_in_bytes', + 'cluster_stats.indices.segments.index_writer_memory_in_bytes', + 'cluster_stats.indices.segments.max_unsafe_auto_id_timestamp', + 'cluster_stats.indices.segments.memory_in_bytes', + 'cluster_stats.indices.segments.norms_memory_in_bytes', + 'cluster_stats.indices.segments.points_memory_in_bytes', + 'cluster_stats.indices.segments.stored_fields_memory_in_bytes', + 'cluster_stats.indices.segments.term_vectors_memory_in_bytes', + 'cluster_stats.indices.segments.terms_memory_in_bytes', + 'cluster_stats.indices.segments.version_map_memory_in_bytes', + 'cluster_stats.indices.shards.index.primaries.avg', + 'cluster_stats.indices.shards.index.primaries.max', + 'cluster_stats.indices.shards.index.primaries.min', + 'cluster_stats.indices.shards.index.replication.avg', + 'cluster_stats.indices.shards.index.replication.max', + 'cluster_stats.indices.shards.index.replication.min', + 'cluster_stats.indices.shards.index.shards.avg', + 'cluster_stats.indices.shards.index.shards.max', + 'cluster_stats.indices.shards.index.shards.min', + 'cluster_stats.indices.shards.primaries', + 'cluster_stats.indices.shards.replication', + 'cluster_stats.indices.shards.total', + 'cluster_stats.indices.store.size_in_bytes', + 'cluster_stats.nodes.count.coordinating_only', + 'cluster_stats.nodes.count.data', + 'cluster_stats.nodes.count.ingest', + 'cluster_stats.nodes.count.master', + 'cluster_stats.nodes.count.total', + 'cluster_stats.nodes.fs.available_in_bytes', + 'cluster_stats.nodes.fs.free_in_bytes', + 'cluster_stats.nodes.fs.total_in_bytes', + 'cluster_stats.nodes.jvm.max_uptime_in_millis', + 'cluster_stats.nodes.jvm.mem.heap_max_in_bytes', + 'cluster_stats.nodes.jvm.mem.heap_used_in_bytes', + 'cluster_stats.nodes.jvm.threads', + 'cluster_stats.nodes.jvm.versions.0.count', + 'cluster_stats.nodes.jvm.versions.0.version', + 'cluster_stats.nodes.jvm.versions.0.vm_name', + 'cluster_stats.nodes.jvm.versions.0.vm_vendor', + 'cluster_stats.nodes.jvm.versions.0.vm_version', + 'cluster_stats.nodes.network_types.http_types.security4', + 'cluster_stats.nodes.network_types.transport_types.security4', + 'cluster_stats.nodes.os.allocated_processors', + 'cluster_stats.nodes.os.available_processors', + 'cluster_stats.nodes.os.mem.free_in_bytes', + 'cluster_stats.nodes.os.mem.free_percent', + 'cluster_stats.nodes.os.mem.total_in_bytes', + 'cluster_stats.nodes.os.mem.used_in_bytes', + 'cluster_stats.nodes.os.mem.used_percent', + 'cluster_stats.nodes.os.names.0.count', + 'cluster_stats.nodes.os.names.0.name', + 'cluster_stats.nodes.os.pretty_names.0.count', + 'cluster_stats.nodes.os.pretty_names.0.pretty_name', + 'cluster_stats.nodes.process.cpu.percent', + 'cluster_stats.nodes.process.open_file_descriptors.avg', + 'cluster_stats.nodes.process.open_file_descriptors.max', + 'cluster_stats.nodes.process.open_file_descriptors.min', + 'cluster_stats.nodes.versions.0', + 'cluster_stats.status', + 'cluster_stats.timestamp', + 'cluster_uuid', + 'collection', + 'license.expiry_date', + 'license.expiry_date_in_millis', + 'license.issue_date', + 'license.issue_date_in_millis', + 'license.issued_to', + 'license.issuer', + 'license.max_nodes', + 'license.start_date_in_millis', + 'license.status', + 'license.type', + 'license.uid', + 'stack_stats.kibana.count', + 'stack_stats.kibana.dashboard.total', + 'stack_stats.kibana.graph_workspace.total', + 'stack_stats.kibana.index_pattern.total', + 'stack_stats.kibana.indices', + 'stack_stats.kibana.os.platformReleases.0.count', + 'stack_stats.kibana.os.platformReleases.0.platformRelease', + 'stack_stats.kibana.os.platforms.0.count', + 'stack_stats.kibana.os.platforms.0.platform', + 'stack_stats.kibana.plugins.apm.has_any_services', + 'stack_stats.kibana.plugins.infraops.last_24_hours.hits.infraops_docker', + 'stack_stats.kibana.plugins.infraops.last_24_hours.hits.infraops_hosts', + 'stack_stats.kibana.plugins.infraops.last_24_hours.hits.infraops_kubernetes', + 'stack_stats.kibana.plugins.infraops.last_24_hours.hits.logs', + 'stack_stats.kibana.plugins.kql.defaultQueryLanguage', + 'stack_stats.kibana.plugins.kql.optInCount', + 'stack_stats.kibana.plugins.kql.optOutCount', + 'stack_stats.kibana.plugins.reporting.PNG.available', + 'stack_stats.kibana.plugins.reporting.PNG.total', + 'stack_stats.kibana.plugins.reporting._all', + 'stack_stats.kibana.plugins.reporting.available', + 'stack_stats.kibana.plugins.reporting.browser_type', + 'stack_stats.kibana.plugins.reporting.csv.available', + 'stack_stats.kibana.plugins.reporting.csv.total', + 'stack_stats.kibana.plugins.reporting.enabled', + 'stack_stats.kibana.plugins.reporting.last7Days.PNG.available', + 'stack_stats.kibana.plugins.reporting.last7Days.PNG.total', + 'stack_stats.kibana.plugins.reporting.last7Days._all', + 'stack_stats.kibana.plugins.reporting.last7Days.csv.available', + 'stack_stats.kibana.plugins.reporting.last7Days.csv.total', + 'stack_stats.kibana.plugins.reporting.last7Days.printable_pdf.app.dashboard', + 'stack_stats.kibana.plugins.reporting.last7Days.printable_pdf.app.visualization', + 'stack_stats.kibana.plugins.reporting.last7Days.printable_pdf.available', + 'stack_stats.kibana.plugins.reporting.last7Days.printable_pdf.layout.preserve_layout', + 'stack_stats.kibana.plugins.reporting.last7Days.printable_pdf.layout.print', + 'stack_stats.kibana.plugins.reporting.last7Days.printable_pdf.total', + 'stack_stats.kibana.plugins.reporting.lastDay.PNG.available', + 'stack_stats.kibana.plugins.reporting.lastDay.PNG.total', + 'stack_stats.kibana.plugins.reporting.lastDay._all', + 'stack_stats.kibana.plugins.reporting.lastDay.csv.available', + 'stack_stats.kibana.plugins.reporting.lastDay.csv.total', + 'stack_stats.kibana.plugins.reporting.lastDay.printable_pdf.app.dashboard', + 'stack_stats.kibana.plugins.reporting.lastDay.printable_pdf.app.visualization', + 'stack_stats.kibana.plugins.reporting.lastDay.printable_pdf.available', + 'stack_stats.kibana.plugins.reporting.lastDay.printable_pdf.layout.preserve_layout', + 'stack_stats.kibana.plugins.reporting.lastDay.printable_pdf.layout.print', + 'stack_stats.kibana.plugins.reporting.lastDay.printable_pdf.total', + 'stack_stats.kibana.plugins.reporting.printable_pdf.app.dashboard', + 'stack_stats.kibana.plugins.reporting.printable_pdf.app.visualization', + 'stack_stats.kibana.plugins.reporting.printable_pdf.available', + 'stack_stats.kibana.plugins.reporting.printable_pdf.layout.preserve_layout', + 'stack_stats.kibana.plugins.reporting.printable_pdf.layout.print', + 'stack_stats.kibana.plugins.reporting.printable_pdf.total', + 'stack_stats.kibana.plugins.rollups.index_patterns.total', + 'stack_stats.kibana.plugins.rollups.saved_searches.total', + 'stack_stats.kibana.plugins.rollups.visualizations.saved_searches.total', + 'stack_stats.kibana.plugins.rollups.visualizations.total', + 'stack_stats.kibana.plugins.spaces.available', + 'stack_stats.kibana.plugins.spaces.count', + 'stack_stats.kibana.plugins.spaces.enabled', + 'stack_stats.kibana.search.total', + 'stack_stats.kibana.timelion_sheet.total', + 'stack_stats.kibana.versions.0.count', + 'stack_stats.kibana.versions.0.version', + 'stack_stats.kibana.visualization.total', + 'stack_stats.xpack.graph.available', + 'stack_stats.xpack.graph.enabled', + 'stack_stats.xpack.logstash.available', + 'stack_stats.xpack.logstash.enabled', + 'stack_stats.xpack.ml.available', + 'stack_stats.xpack.ml.datafeeds._all.count', + 'stack_stats.xpack.ml.enabled', + 'stack_stats.xpack.ml.jobs._all.count', + 'stack_stats.xpack.ml.jobs._all.detectors.avg', + 'stack_stats.xpack.ml.jobs._all.detectors.max', + 'stack_stats.xpack.ml.jobs._all.detectors.min', + 'stack_stats.xpack.ml.jobs._all.detectors.total', + 'stack_stats.xpack.ml.jobs._all.forecasts.forecasted_jobs', + 'stack_stats.xpack.ml.jobs._all.forecasts.total', + 'stack_stats.xpack.ml.jobs._all.model_size.avg', + 'stack_stats.xpack.ml.jobs._all.model_size.max', + 'stack_stats.xpack.ml.jobs._all.model_size.min', + 'stack_stats.xpack.ml.jobs._all.model_size.total', + 'stack_stats.xpack.ml.node_count', + 'stack_stats.xpack.monitoring.available', + 'stack_stats.xpack.monitoring.collection_enabled', + 'stack_stats.xpack.monitoring.enabled', + 'stack_stats.xpack.monitoring.enabled_exporters.local', + 'stack_stats.xpack.rollup.available', + 'stack_stats.xpack.rollup.enabled', + 'stack_stats.xpack.security.anonymous.enabled', + 'stack_stats.xpack.security.audit.enabled', + 'stack_stats.xpack.security.audit.outputs.0', + 'stack_stats.xpack.security.available', + 'stack_stats.xpack.security.enabled', + 'stack_stats.xpack.security.ipfilter.http', + 'stack_stats.xpack.security.ipfilter.transport', + 'stack_stats.xpack.security.realms.active_directory.available', + 'stack_stats.xpack.security.realms.active_directory.enabled', + 'stack_stats.xpack.security.realms.file.available', + 'stack_stats.xpack.security.realms.file.cache.0.size', + 'stack_stats.xpack.security.realms.file.enabled', + 'stack_stats.xpack.security.realms.file.name.0', + 'stack_stats.xpack.security.realms.file.order.0', + 'stack_stats.xpack.security.realms.file.size.0', + 'stack_stats.xpack.security.realms.kerberos.available', + 'stack_stats.xpack.security.realms.kerberos.enabled', + 'stack_stats.xpack.security.realms.ldap.available', + 'stack_stats.xpack.security.realms.ldap.enabled', + 'stack_stats.xpack.security.realms.native.available', + 'stack_stats.xpack.security.realms.native.cache.0.size', + 'stack_stats.xpack.security.realms.native.enabled', + 'stack_stats.xpack.security.realms.native.name.0', + 'stack_stats.xpack.security.realms.native.order.0', + 'stack_stats.xpack.security.realms.native.size.0', + 'stack_stats.xpack.security.realms.pki.available', + 'stack_stats.xpack.security.realms.pki.enabled', + 'stack_stats.xpack.security.realms.saml.available', + 'stack_stats.xpack.security.realms.saml.enabled', + 'stack_stats.xpack.security.role_mapping.native.enabled', + 'stack_stats.xpack.security.role_mapping.native.size', + 'stack_stats.xpack.security.roles.file.dls', + 'stack_stats.xpack.security.roles.file.fls', + 'stack_stats.xpack.security.roles.file.size', + 'stack_stats.xpack.security.roles.native.dls', + 'stack_stats.xpack.security.roles.native.fls', + 'stack_stats.xpack.security.roles.native.size', + 'stack_stats.xpack.security.ssl.http.enabled', + 'stack_stats.xpack.security.ssl.transport.enabled', + 'stack_stats.xpack.sql.available', + 'stack_stats.xpack.sql.enabled', + 'stack_stats.xpack.sql.features.command', + 'stack_stats.xpack.sql.features.groupby', + 'stack_stats.xpack.sql.features.having', + 'stack_stats.xpack.sql.features.join', + 'stack_stats.xpack.sql.features.limit', + 'stack_stats.xpack.sql.features.local', + 'stack_stats.xpack.sql.features.orderby', + 'stack_stats.xpack.sql.features.subselect', + 'stack_stats.xpack.sql.features.where', + 'stack_stats.xpack.sql.queries._all.failed', + 'stack_stats.xpack.sql.queries._all.paging', + 'stack_stats.xpack.sql.queries._all.total', + 'stack_stats.xpack.sql.queries.canvas.failed', + 'stack_stats.xpack.sql.queries.canvas.paging', + 'stack_stats.xpack.sql.queries.canvas.total', + 'stack_stats.xpack.sql.queries.cli.failed', + 'stack_stats.xpack.sql.queries.cli.paging', + 'stack_stats.xpack.sql.queries.cli.total', + 'stack_stats.xpack.sql.queries.jdbc.failed', + 'stack_stats.xpack.sql.queries.jdbc.paging', + 'stack_stats.xpack.sql.queries.jdbc.total', + 'stack_stats.xpack.sql.queries.odbc.failed', + 'stack_stats.xpack.sql.queries.odbc.paging', + 'stack_stats.xpack.sql.queries.odbc.total', + 'stack_stats.xpack.sql.queries.rest.failed', + 'stack_stats.xpack.sql.queries.rest.paging', + 'stack_stats.xpack.sql.queries.rest.total', + 'stack_stats.xpack.sql.queries.translate.count', + 'stack_stats.xpack.watcher.available', + 'stack_stats.xpack.watcher.count.active', + 'stack_stats.xpack.watcher.count.total', + 'stack_stats.xpack.watcher.enabled', + 'stack_stats.xpack.watcher.execution.actions._all.total', + 'stack_stats.xpack.watcher.execution.actions._all.total_time_in_ms', + 'stack_stats.xpack.watcher.watch.input._all.active', + 'stack_stats.xpack.watcher.watch.input._all.total', + 'stack_stats.xpack.watcher.watch.trigger._all.active', + 'stack_stats.xpack.watcher.watch.trigger._all.total', + 'timestamp', + 'version', + ]; + + expect(actual).to.eql(expected); + }); + + it('should pull local stats and validate data types', async () => { + const timeRange = { + min: '2018-07-23T22:07:00Z', + max: '2018-07-23T22:13:00Z' + }; + + const { body } = await supertest + .post('/api/telemetry/v1/clusters/_stats') + .set('kbn-xsrf', 'xxx') + .send({ timeRange }) + .expect(200); + + expect(body.length).to.be(1); + const stats = body[0]; + + expect(stats.collection).to.be('local'); + expect(stats.license.issuer).to.be('elasticsearch'); + expect(stats.license.status).to.be('active'); + + expect(stats.stack_stats.kibana.count).to.be(1); + expect(stats.stack_stats.kibana.indices).to.be(1); + + expect(stats.stack_stats.kibana.dashboard.total).to.be.a('number'); + expect(stats.stack_stats.kibana.graph_workspace.total).to.be.a('number'); + expect(stats.stack_stats.kibana.index_pattern.total).to.be.a('number'); + expect(stats.stack_stats.kibana.search.total).to.be.a('number'); + expect(stats.stack_stats.kibana.timelion_sheet.total).to.be.a('number'); + expect(stats.stack_stats.kibana.visualization.total).to.be.a('number'); + + expect(stats.stack_stats.kibana.plugins.apm.services_per_agent).to.be.an('object'); + expect(stats.stack_stats.kibana.plugins.infraops.last_24_hours).to.be.an('object'); + expect(stats.stack_stats.kibana.plugins.kql.defaultQueryLanguage).to.be.a('string'); + expect(stats.stack_stats.kibana.plugins.reporting.enabled).to.be(true); + expect(stats.stack_stats.kibana.plugins.rollups.index_patterns).to.be.an('object'); + expect(stats.stack_stats.kibana.plugins.spaces.available).to.be(true); + + expect(stats.stack_stats.kibana.os.platforms[0].platform).to.be.a('string'); + expect(stats.stack_stats.kibana.os.platforms[0].count).to.be(1); + expect(stats.stack_stats.kibana.os.platformReleases[0].platformRelease).to.be.a('string'); + expect(stats.stack_stats.kibana.os.platformReleases[0].count).to.be(1); + + expect(stats.stack_stats.xpack.graph).to.be.an('object'); + expect(stats.stack_stats.xpack.ilm).to.be.an('object'); + expect(stats.stack_stats.xpack.logstash).to.be.an('object'); + expect(stats.stack_stats.xpack.ml).to.be.an('object'); + expect(stats.stack_stats.xpack.monitoring).to.be.an('object'); + expect(stats.stack_stats.xpack.rollup).to.be.an('object'); + }); + }); +} +