Skip to content

Commit

Permalink
Merge branch 'master' into np-vega
Browse files Browse the repository at this point in the history
  • Loading branch information
elasticmachine authored Apr 23, 2020
2 parents e323eb1 + ebf5503 commit 67a9287
Show file tree
Hide file tree
Showing 57 changed files with 504 additions and 258 deletions.
3 changes: 3 additions & 0 deletions .github/paths-labeller.yml
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,9 @@
- "Feature:ExpressionLanguage":
- "src/plugins/expressions/**/*.*"
- "src/plugins/bfetch/**/*.*"
- "Team:apm"
- "x-pack/plugins/apm/**/*.*"
- "x-pack/legacy/plugins/apm/**/*.*"
- "Team:uptime":
- "x-pack/plugins/uptime/**/*.*"
- "x-pack/legacy/plugins/uptime/**/*.*"
2 changes: 1 addition & 1 deletion .i18nrc.json
Original file line number Diff line number Diff line change
Expand Up @@ -49,7 +49,7 @@
"visTypeMarkdown": "src/plugins/vis_type_markdown",
"visTypeMetric": "src/plugins/vis_type_metric",
"visTypeTable": "src/plugins/vis_type_table",
"visTypeTagCloud": "src/legacy/core_plugins/vis_type_tagcloud",
"visTypeTagCloud": "src/plugins/vis_type_tagcloud",
"visTypeTimeseries": ["src/legacy/core_plugins/vis_type_timeseries", "src/plugins/vis_type_timeseries"],
"visTypeVega": "src/plugins/vis_type_vega",
"visTypeVislib": "src/legacy/core_plugins/vis_type_vislib",
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -21,14 +21,16 @@ import expect from '@kbn/expect';
import _ from 'lodash';
import d3 from 'd3';

import { TagCloud } from '../tag_cloud';
import { fromNode, delay } from 'bluebird';
import { ImageComparator } from 'test_utils/image_comparator';
import simpleloadPng from './simpleload.png';

// Replace with mock when converting to jest tests
// eslint-disable-next-line @kbn/eslint/no-restricted-paths
import { seedColors } from '../../../../../../plugins/charts/public/services/colors/seed_colors';
// Will be replaced with new path when tests are moved
// eslint-disable-next-line @kbn/eslint/no-restricted-paths
import { TagCloud } from '../../../../../../plugins/vis_type_tagcloud/public/components/tag_cloud';

describe('tag cloud tests', function() {
const minValue = 1;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,6 @@
import expect from '@kbn/expect';
import ngMock from 'ng_mock';
import { ImageComparator } from 'test_utils/image_comparator';
import { createTagCloudVisualization } from '../tag_cloud_visualization';
import basicdrawPng from './basicdraw.png';
import afterresizePng from './afterresize.png';
import afterparamChange from './afterparamchange.png';
Expand All @@ -32,7 +31,14 @@ import { ExprVis } from '../../../../../../plugins/visualizations/public/express
import { seedColors } from '../../../../../../plugins/charts/public/services/colors/seed_colors';
// eslint-disable-next-line @kbn/eslint/no-restricted-paths
import { BaseVisType } from '../../../../../../plugins/visualizations/public/vis_types/base_vis_type';
import { createTagCloudVisTypeDefinition } from '../../tag_cloud_type';
// Will be replaced with new path when tests are moved
// eslint-disable-next-line @kbn/eslint/no-restricted-paths
import { createTagCloudVisTypeDefinition } from '../../../../../../plugins/vis_type_tagcloud/public/tag_cloud_type';
// eslint-disable-next-line @kbn/eslint/no-restricted-paths
import { createTagCloudVisualization } from '../../../../../../plugins/vis_type_tagcloud/public/components/tag_cloud_visualization';
import { npStart } from 'ui/new_platform';
// eslint-disable-next-line @kbn/eslint/no-restricted-paths
import { setFormatService } from '../../../../../../plugins/vis_type_tagcloud/public/services';

const THRESHOLD = 0.65;
const PIXEL_DIFF = 64;
Expand Down Expand Up @@ -66,6 +72,8 @@ describe('TagCloudVisualizationTest', function() {
},
});

before(() => setFormatService(npStart.plugins.data.fieldFormats));

beforeEach(ngMock.module('kibana'));

describe('TagCloudVisualization - basics', function() {
Expand Down
44 changes: 0 additions & 44 deletions src/legacy/core_plugins/vis_type_tagcloud/index.ts

This file was deleted.

4 changes: 0 additions & 4 deletions src/legacy/core_plugins/vis_type_tagcloud/package.json

This file was deleted.

2 changes: 1 addition & 1 deletion src/plugins/home/kibana.json
Original file line number Diff line number Diff line change
Expand Up @@ -4,5 +4,5 @@
"server": true,
"ui": true,
"requiredPlugins": ["data", "kibanaLegacy"],
"optionalPlugins": ["usage_collection", "telemetry"]
"optionalPlugins": ["usageCollection", "telemetry"]
}
4 changes: 2 additions & 2 deletions src/plugins/home/server/plugin.ts
Original file line number Diff line number Diff line change
Expand Up @@ -29,7 +29,7 @@ import { UsageCollectionSetup } from '../../usage_collection/server';
import { sampleDataTelemetry } from './saved_objects';

interface HomeServerPluginSetupDependencies {
usage_collection?: UsageCollectionSetup;
usageCollection?: UsageCollectionSetup;
}

export class HomeServerPlugin implements Plugin<HomeServerPluginSetup, HomeServerPluginStart> {
Expand All @@ -41,7 +41,7 @@ export class HomeServerPlugin implements Plugin<HomeServerPluginSetup, HomeServe
core.savedObjects.registerType(sampleDataTelemetry);
return {
tutorials: { ...this.tutorialsRegistry.setup(core) },
sampleData: { ...this.sampleDataRegistry.setup(core, plugins.usage_collection) },
sampleData: { ...this.sampleDataRegistry.setup(core, plugins.usageCollection) },
};
}

Expand Down
23 changes: 20 additions & 3 deletions src/plugins/telemetry/public/mocks.ts
Original file line number Diff line number Diff line change
Expand Up @@ -25,11 +25,20 @@ import { httpServiceMock } from '../../../core/public/http/http_service.mock';
import { notificationServiceMock } from '../../../core/public/notifications/notifications_service.mock';
import { TelemetryService } from './services/telemetry_service';
import { TelemetryNotifications } from './services/telemetry_notifications/telemetry_notifications';
import { TelemetryPluginStart } from './plugin';
import { TelemetryPluginStart, TelemetryPluginConfig } from './plugin';

// The following is to be able to access private methods
/* eslint-disable dot-notation */

export interface TelemetryServiceMockOptions {
reportOptInStatusChange?: boolean;
config?: Partial<TelemetryPluginConfig>;
}

export function mockTelemetryService({
reportOptInStatusChange,
}: { reportOptInStatusChange?: boolean } = {}) {
config: configOverride = {},
}: TelemetryServiceMockOptions = {}) {
const config = {
enabled: true,
url: 'http://localhost',
Expand All @@ -39,14 +48,22 @@ export function mockTelemetryService({
banner: true,
allowChangingOptInStatus: true,
telemetryNotifyUserAboutOptInDefault: true,
...configOverride,
};

return new TelemetryService({
const telemetryService = new TelemetryService({
config,
http: httpServiceMock.createStartContract(),
notifications: notificationServiceMock.createStartContract(),
reportOptInStatusChange,
});

const originalReportOptInStatus = telemetryService['reportOptInStatus'];
telemetryService['reportOptInStatus'] = jest.fn().mockImplementation(optInPayload => {
return originalReportOptInStatus(optInPayload); // Actually calling the original method
});

return telemetryService;
}

export function mockTelemetryNotifications({
Expand Down
103 changes: 86 additions & 17 deletions src/plugins/telemetry/public/services/telemetry_service.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -67,17 +67,31 @@ describe('TelemetryService', () => {
});

describe('setOptIn', () => {
it('does not call the api if canChangeOptInStatus==false', async () => {
const telemetryService = mockTelemetryService({
reportOptInStatusChange: false,
config: { allowChangingOptInStatus: false },
});
expect(await telemetryService.setOptIn(true)).toBe(false);

expect(telemetryService['http'].post).toBeCalledTimes(0);
});

it('calls api if canChangeOptInStatus', async () => {
const telemetryService = mockTelemetryService({ reportOptInStatusChange: false });
telemetryService.getCanChangeOptInStatus = jest.fn().mockReturnValue(true);
const telemetryService = mockTelemetryService({
reportOptInStatusChange: false,
config: { allowChangingOptInStatus: true },
});
await telemetryService.setOptIn(true);

expect(telemetryService['http'].post).toBeCalledTimes(1);
});

it('sends enabled true if optedIn: true', async () => {
const telemetryService = mockTelemetryService({ reportOptInStatusChange: false });
telemetryService.getCanChangeOptInStatus = jest.fn().mockReturnValue(true);
const telemetryService = mockTelemetryService({
reportOptInStatusChange: false,
config: { allowChangingOptInStatus: true },
});
const optedIn = true;
await telemetryService.setOptIn(optedIn);

Expand All @@ -87,8 +101,10 @@ describe('TelemetryService', () => {
});

it('sends enabled false if optedIn: false', async () => {
const telemetryService = mockTelemetryService({ reportOptInStatusChange: false });
telemetryService.getCanChangeOptInStatus = jest.fn().mockReturnValue(true);
const telemetryService = mockTelemetryService({
reportOptInStatusChange: false,
config: { allowChangingOptInStatus: true },
});
const optedIn = false;
await telemetryService.setOptIn(optedIn);

Expand All @@ -98,29 +114,32 @@ describe('TelemetryService', () => {
});

it('does not call reportOptInStatus if reportOptInStatusChange is false', async () => {
const telemetryService = mockTelemetryService({ reportOptInStatusChange: false });
telemetryService.getCanChangeOptInStatus = jest.fn().mockReturnValue(true);
telemetryService['reportOptInStatus'] = jest.fn();
const telemetryService = mockTelemetryService({
reportOptInStatusChange: false,
config: { allowChangingOptInStatus: true },
});
await telemetryService.setOptIn(true);

expect(telemetryService['reportOptInStatus']).toBeCalledTimes(0);
expect(telemetryService['http'].post).toBeCalledTimes(1);
});

it('calls reportOptInStatus if reportOptInStatusChange is true', async () => {
const telemetryService = mockTelemetryService({ reportOptInStatusChange: true });
telemetryService.getCanChangeOptInStatus = jest.fn().mockReturnValue(true);
telemetryService['reportOptInStatus'] = jest.fn();
const telemetryService = mockTelemetryService({
reportOptInStatusChange: true,
config: { allowChangingOptInStatus: true },
});
await telemetryService.setOptIn(true);

expect(telemetryService['reportOptInStatus']).toBeCalledTimes(1);
expect(telemetryService['http'].post).toBeCalledTimes(1);
});

it('adds an error toast on api error', async () => {
const telemetryService = mockTelemetryService({ reportOptInStatusChange: false });
telemetryService.getCanChangeOptInStatus = jest.fn().mockReturnValue(true);
telemetryService['reportOptInStatus'] = jest.fn();
const telemetryService = mockTelemetryService({
reportOptInStatusChange: false,
config: { allowChangingOptInStatus: true },
});
telemetryService['http'].post = jest.fn().mockImplementation((url: string) => {
if (url === '/api/telemetry/v2/optIn') {
throw Error('failed to update opt in.');
Expand All @@ -133,9 +152,13 @@ describe('TelemetryService', () => {
expect(telemetryService['notifications'].toasts.addError).toBeCalledTimes(1);
});

// This one should not happen because the entire method is fully caught but hey! :)
it('adds an error toast on reportOptInStatus error', async () => {
const telemetryService = mockTelemetryService({ reportOptInStatusChange: true });
telemetryService.getCanChangeOptInStatus = jest.fn().mockReturnValue(true);
const telemetryService = mockTelemetryService({
reportOptInStatusChange: true,
config: { allowChangingOptInStatus: true },
});

telemetryService['reportOptInStatus'] = jest.fn().mockImplementation(() => {
throw Error('failed to report OptIn Status.');
});
Expand All @@ -146,4 +169,50 @@ describe('TelemetryService', () => {
expect(telemetryService['notifications'].toasts.addError).toBeCalledTimes(1);
});
});

describe('getTelemetryUrl', () => {
it('should return the config.url parameter', async () => {
const url = 'http://test.com';
const telemetryService = mockTelemetryService({
config: { url },
});

expect(telemetryService.getTelemetryUrl()).toBe(url);
});
});

describe('setUserHasSeenNotice', () => {
it('should hit the API and change the config', async () => {
const telemetryService = mockTelemetryService({
config: { telemetryNotifyUserAboutOptInDefault: undefined },
});

expect(telemetryService.userHasSeenOptedInNotice).toBe(undefined);
expect(telemetryService.getUserHasSeenOptedInNotice()).toBe(false);
await telemetryService.setUserHasSeenNotice();
expect(telemetryService['http'].put).toBeCalledTimes(1);
expect(telemetryService.userHasSeenOptedInNotice).toBe(true);
expect(telemetryService.getUserHasSeenOptedInNotice()).toBe(true);
});

it('should show a toast notification if the request fail', async () => {
const telemetryService = mockTelemetryService({
config: { telemetryNotifyUserAboutOptInDefault: undefined },
});

telemetryService['http'].put = jest.fn().mockImplementation((url: string) => {
if (url === '/api/telemetry/v2/userHasSeenNotice') {
throw Error('failed to update opt in.');
}
});

expect(telemetryService.userHasSeenOptedInNotice).toBe(undefined);
expect(telemetryService.getUserHasSeenOptedInNotice()).toBe(false);
await telemetryService.setUserHasSeenNotice();
expect(telemetryService['http'].put).toBeCalledTimes(1);
expect(telemetryService['notifications'].toasts.addError).toBeCalledTimes(1);
expect(telemetryService.userHasSeenOptedInNotice).toBe(false);
expect(telemetryService.getUserHasSeenOptedInNotice()).toBe(false);
});
});
});
16 changes: 12 additions & 4 deletions src/plugins/telemetry/public/services/telemetry_service.ts
Original file line number Diff line number Diff line change
Expand Up @@ -122,11 +122,15 @@ export class TelemetryService {
}

try {
await this.http.post('/api/telemetry/v2/optIn', {
// Report the option to the Kibana server to store the settings.
// It returns the encrypted update to send to the telemetry cluster [{cluster_uuid, opt_in_status}]
const optInPayload = await this.http.post<string[]>('/api/telemetry/v2/optIn', {
body: JSON.stringify({ enabled: optedIn }),
});
if (this.reportOptInStatusChange) {
await this.reportOptInStatus(optedIn);
// Use the response to report about the change to the remote telemetry cluster.
// If it's opt-out, this will be the last communication to the remote service.
await this.reportOptInStatus(optInPayload);
}
this.isOptedIn = optedIn;
} catch (err) {
Expand Down Expand Up @@ -162,7 +166,11 @@ export class TelemetryService {
}
};

private reportOptInStatus = async (OptInStatus: boolean): Promise<void> => {
/**
* Pushes the encrypted payload [{cluster_uuid, opt_in_status}] to the remote telemetry service
* @param optInPayload [{cluster_uuid, opt_in_status}] encrypted by the server into an array of strings
*/
private reportOptInStatus = async (optInPayload: string[]): Promise<void> => {
const telemetryOptInStatusUrl = this.getOptInStatusUrl();

try {
Expand All @@ -171,7 +179,7 @@ export class TelemetryService {
headers: {
'Content-Type': 'application/json',
},
body: JSON.stringify({ enabled: OptInStatus }),
body: JSON.stringify(optInPayload),
});
} catch (err) {
// Sending the ping is best-effort. Telemetry tries to send the ping once and discards it immediately if sending fails.
Expand Down
Loading

0 comments on commit 67a9287

Please sign in to comment.