mirror of
https://github.com/ckaczor/azuredatastudio.git
synced 2026-02-03 17:23:42 -05:00
Inital platform relayering (#6385)
* moving test files and inital refactoring * relayer extension host code * fix imports * make insights work * relayer dashboard * relayer notebooks * moveing more code around * formatting * accept angular as browser * fix serializer * add missing files * remove declarations from extensions * fix build errors * more relayering * change urls to relative to help code relayering * remove layering to prep for merge * fix hygiene errors * fix hygiene errors * fix tests
This commit is contained in:
@@ -1,179 +0,0 @@
|
||||
/*---------------------------------------------------------------------------------------------
|
||||
* Copyright (c) Microsoft Corporation. All rights reserved.
|
||||
* Licensed under the Source EULA. See License.txt in the project root for license information.
|
||||
*--------------------------------------------------------------------------------------------*/
|
||||
|
||||
import { TelemetryView } from 'sql/platform/telemetry/telemetryKeys';
|
||||
import { IAdsTelemetryService } from 'sql/platform/telemetry/telemetry';
|
||||
import { Registry } from 'vs/platform/registry/common/platform';
|
||||
import { Extensions as WorkbenchExtensions, IWorkbenchContributionsRegistry, IWorkbenchContribution } from 'vs/workbench/common/contributions';
|
||||
import { LifecyclePhase } from 'vs/platform/lifecycle/common/lifecycle';
|
||||
import { Disposable } from 'vs/base/common/lifecycle';
|
||||
import { IStorageService, StorageScope } from 'vs/platform/storage/common/storage';
|
||||
import { ICommandService, ICommandEvent } from 'vs/platform/commands/common/commands';
|
||||
|
||||
export class SqlTelemetryContribution extends Disposable implements IWorkbenchContribution {
|
||||
|
||||
constructor(
|
||||
@IAdsTelemetryService private telemetryService: IAdsTelemetryService,
|
||||
@IStorageService storageService: IStorageService,
|
||||
@ICommandService commandService: ICommandService
|
||||
) {
|
||||
super();
|
||||
|
||||
this._register(
|
||||
commandService.onWillExecuteCommand(
|
||||
(e: ICommandEvent) => {
|
||||
// Filter out high-frequency events
|
||||
if (!['type', 'cursorUp', 'cursorDown', 'cursorRight', 'cursorLeft', 'deleteLeft', 'deleteRight'].find(id => id === e.commandId)) {
|
||||
telemetryService.sendActionEvent(TelemetryView.Shell, 'adsCommandExecuted', e.commandId);
|
||||
}
|
||||
}));
|
||||
const dailyLastUseDate: number = Date.parse(storageService.get('telemetry.dailyLastUseDate', StorageScope.GLOBAL, '0'));
|
||||
const weeklyLastUseDate: number = Date.parse(storageService.get('telemetry.weeklyLastUseDate', StorageScope.GLOBAL, '0'));
|
||||
const monthlyLastUseDate: number = Date.parse(storageService.get('telemetry.monthlyLastUseDate', StorageScope.GLOBAL, '0'));
|
||||
const firstTimeUser: boolean = dailyLastUseDate === Date.parse('0');
|
||||
|
||||
let todayString: string = new Date().toUTCString();
|
||||
|
||||
// daily user event
|
||||
if (this.didDayChange(dailyLastUseDate)) {
|
||||
// daily first use
|
||||
telemetryService.sendTelemetryEvent('telemetry.dailyFirstUse', { dailyFirstUse: 'true' });
|
||||
storageService.store('telemetry.dailyLastUseDate', todayString, StorageScope.GLOBAL);
|
||||
}
|
||||
|
||||
// weekly user event
|
||||
if (this.didWeekChange(weeklyLastUseDate)) {
|
||||
// weekly first use
|
||||
telemetryService.sendTelemetryEvent('telemetry.weeklyFirstUse', { weeklyFirstUse: 'true' });
|
||||
storageService.store('telemetry.weeklyLastUseDate', todayString, StorageScope.GLOBAL);
|
||||
}
|
||||
|
||||
|
||||
/* send monthly uses once the user launches on a day that's in a month
|
||||
after the last time we sent a monthly usage count */
|
||||
const monthlyUseCount: number = storageService.getNumber('telemetry.monthlyUseCount', StorageScope.GLOBAL, 0);
|
||||
if (this.didMonthChange(monthlyLastUseDate)) {
|
||||
telemetryService.sendTelemetryEvent('telemetry.monthlyUse', { monthlyFirstUse: 'true' });
|
||||
// the month changed, so send the user usage type event based on monthly count for last month
|
||||
// and reset the count for this month
|
||||
let lastMonthDate = new Date(monthlyLastUseDate);
|
||||
this.sendUsageEvent(monthlyUseCount, lastMonthDate);
|
||||
|
||||
const wasActiveLastMonth: boolean = storageService.getBoolean('telemetry.wasActiveLastMonth', StorageScope.GLOBAL, false);
|
||||
|
||||
if (firstTimeUser) {
|
||||
// new user
|
||||
this.sendGrowthTypeEvent(UserGrowthType.NewUser, lastMonthDate);
|
||||
}
|
||||
|
||||
// continuing or returning user
|
||||
this.sendGrowthTypeEvent(wasActiveLastMonth ? UserGrowthType.ContinuingUser : UserGrowthType.ReturningUser, lastMonthDate);
|
||||
|
||||
// set wasActiveUserLastMonth
|
||||
storageService.store('telemetry.wasActiveLastMonth', true, StorageScope.GLOBAL);
|
||||
|
||||
// reset the monthly count for the new month
|
||||
storageService.store('telemetry.monthlyUseCount', 1, StorageScope.GLOBAL);
|
||||
storageService.store('telemetry.monthlyLastUseDate', todayString, StorageScope.GLOBAL);
|
||||
} else {
|
||||
// if it's the same month, increment the monthly use count
|
||||
storageService.store('telemetry.monthlyUseCount', monthlyUseCount + 1, StorageScope.GLOBAL);
|
||||
}
|
||||
}
|
||||
|
||||
private didDayChange(lastUseDateNumber: number): boolean {
|
||||
let nowDateNumber: number = Date.parse(new Date().toUTCString());
|
||||
if (this.diffInDays(nowDateNumber, lastUseDateNumber) >= 1) {
|
||||
return true;
|
||||
} else {
|
||||
let nowDate = new Date(nowDateNumber);
|
||||
let lastUseDate = new Date(lastUseDateNumber);
|
||||
return nowDate.getUTCDay() !== lastUseDate.getUTCDay();
|
||||
}
|
||||
}
|
||||
|
||||
private didWeekChange(lastUseDateNumber: number): boolean {
|
||||
let nowDateNumber: number = Date.parse(new Date().toUTCString());
|
||||
if (this.diffInDays(nowDateNumber, lastUseDateNumber) >= 7) {
|
||||
return true;
|
||||
} else {
|
||||
let nowDate = new Date(nowDateNumber);
|
||||
let lastUseDate = new Date(lastUseDateNumber);
|
||||
return nowDate.getUTCDay() < lastUseDate.getUTCDay();
|
||||
}
|
||||
}
|
||||
|
||||
private didMonthChange(lastUseDateNumber: number): boolean {
|
||||
let nowDateNumber: number = Date.parse(new Date().toUTCString());
|
||||
if (this.diffInDays(nowDateNumber, lastUseDateNumber) >= 30) {
|
||||
return true;
|
||||
} else {
|
||||
let nowDate = new Date(nowDateNumber);
|
||||
let lastUseDate = new Date(lastUseDateNumber);
|
||||
return nowDate.getUTCMonth() !== lastUseDate.getUTCMonth();
|
||||
}
|
||||
}
|
||||
|
||||
private diffInDays(nowDate: number, lastUseDate: number): number {
|
||||
return (nowDate - lastUseDate) / (3600 * 1000 * 24);
|
||||
}
|
||||
|
||||
// Usage Metrics
|
||||
private sendUsageEvent(monthlyUseCount: number, lastMonthDate: Date): void {
|
||||
let userUsageType: UserUsageType | undefined;
|
||||
if (monthlyUseCount === 1) {
|
||||
userUsageType = UserUsageType.TireKicker;
|
||||
} else if (monthlyUseCount >= 2 && monthlyUseCount <= 11) {
|
||||
userUsageType = UserUsageType.Occasional;
|
||||
} else if (monthlyUseCount >= 12 && monthlyUseCount <= 20) {
|
||||
userUsageType = UserUsageType.Engaged;
|
||||
} else if (monthlyUseCount > 20) {
|
||||
userUsageType = UserUsageType.Dedicated;
|
||||
}
|
||||
if (userUsageType) {
|
||||
this.telemetryService.sendTelemetryEvent('telemetry.userUsage',
|
||||
{ userType: userUsageType.toString(), monthlyUseCount: monthlyUseCount.toString(), month: lastMonthDate.getMonth().toString(), year: lastMonthDate.getFullYear().toString() });
|
||||
}
|
||||
}
|
||||
|
||||
// Growth Metrics
|
||||
private sendGrowthTypeEvent(growthType: UserGrowthType, lastMonthDate: Date): void {
|
||||
this.telemetryService.sendTelemetryEvent('telemetry.userGrowthType', {
|
||||
userGrowthType: growthType.toString(), month: lastMonthDate.getMonth().toString(), year: lastMonthDate.getFullYear().toString()
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Growth Metrics
|
||||
* Active here means opened app atleast 1 time in a month
|
||||
*/
|
||||
export enum UserGrowthType {
|
||||
// first time opening app
|
||||
NewUser = 1,
|
||||
// was active before, wasn't active last month, but is active this month
|
||||
ReturningUser = 2,
|
||||
// was active last month and this month
|
||||
ContinuingUser = 3
|
||||
}
|
||||
|
||||
/**
|
||||
* Usage Metrics
|
||||
* TireKicker = 1 day/month
|
||||
* Occasional = 2-11 days/month
|
||||
* Engaged = 12-20 days/month
|
||||
* Dedicated = 20+ days/month
|
||||
*/
|
||||
export enum UserUsageType {
|
||||
/* 1 day per month */
|
||||
TireKicker = 1,
|
||||
/* 2-11 days per month */
|
||||
Occasional = 2,
|
||||
/* 12-20 days per month */
|
||||
Engaged = 3,
|
||||
/* 20+ days per month */
|
||||
Dedicated = 4
|
||||
}
|
||||
Registry.as<IWorkbenchContributionsRegistry>(WorkbenchExtensions.Workbench).registerWorkbenchContribution(SqlTelemetryContribution, LifecyclePhase.Starting);
|
||||
@@ -0,0 +1,107 @@
|
||||
/*---------------------------------------------------------------------------------------------
|
||||
* Copyright (c) Microsoft Corporation. All rights reserved.
|
||||
* Licensed under the Source EULA. See License.txt in the project root for license information.
|
||||
*--------------------------------------------------------------------------------------------*/
|
||||
|
||||
import * as TelemetryUtils from 'sql/platform/telemetry/common/telemetryUtilities';
|
||||
import { ITelemetryService } from 'vs/platform/telemetry/common/telemetry';
|
||||
import * as TypeMoq from 'typemoq';
|
||||
import * as assert from 'assert';
|
||||
import { NullLogService } from 'vs/platform/log/common/log';
|
||||
import { SimpleTelemetryService } from 'vs/workbench/browser/web.simpleservices';
|
||||
|
||||
suite('SQL Telemetry Utilities tests', () => {
|
||||
let telemetryService: TypeMoq.Mock<ITelemetryService>;
|
||||
let none: void;
|
||||
let providerName: string = 'provider name';
|
||||
let telemetryKey: string = 'tel key';
|
||||
|
||||
let connectionProfile = {
|
||||
connectionName: '',
|
||||
databaseName: '',
|
||||
serverName: '',
|
||||
authenticationType: '',
|
||||
getOptionsKey: () => '',
|
||||
matches: undefined,
|
||||
groupFullName: '',
|
||||
groupId: '',
|
||||
id: '',
|
||||
options: {},
|
||||
password: '',
|
||||
providerName: providerName,
|
||||
savePassword: true,
|
||||
saveProfile: true,
|
||||
userName: ''
|
||||
};
|
||||
|
||||
setup(() => {
|
||||
telemetryService = TypeMoq.Mock.ofType(SimpleTelemetryService, TypeMoq.MockBehavior.Strict);
|
||||
telemetryService.setup(x => x.publicLog(TypeMoq.It.isAny(), TypeMoq.It.isAny())).returns(x => Promise.resolve(none));
|
||||
});
|
||||
|
||||
test('addTelemetry should add provider id using the connection', (done) => {
|
||||
let data: TelemetryUtils.IConnectionTelemetryData = {
|
||||
};
|
||||
const logService = new NullLogService();
|
||||
TelemetryUtils.addTelemetry(telemetryService.object, logService, telemetryKey, data, connectionProfile).then(() => {
|
||||
telemetryService.verify(x => x.publicLog(TypeMoq.It.is(a => a === telemetryKey), TypeMoq.It.is(b => b.provider === providerName)), TypeMoq.Times.once());
|
||||
done();
|
||||
}).catch(err => {
|
||||
assert.fail(err);
|
||||
done(err);
|
||||
});
|
||||
});
|
||||
|
||||
test('addTelemetry should pass the telemetry data to telemetry service', (done) => {
|
||||
let data: TelemetryUtils.IConnectionTelemetryData = {
|
||||
target: 'target',
|
||||
from: 'from'
|
||||
};
|
||||
data.test1 = '1';
|
||||
|
||||
const logService = new NullLogService();
|
||||
TelemetryUtils.addTelemetry(telemetryService.object, logService, telemetryKey, data, connectionProfile).then(() => {
|
||||
telemetryService.verify(x => x.publicLog(
|
||||
TypeMoq.It.is(a => a === telemetryKey),
|
||||
TypeMoq.It.is(b => b.provider === providerName
|
||||
&& b.from === data.from
|
||||
&& b.target === data.target
|
||||
&& b.test1 === data.test1
|
||||
&& b.connection === undefined)), TypeMoq.Times.once());
|
||||
done();
|
||||
}).catch(err => {
|
||||
assert.fail(err);
|
||||
done(err);
|
||||
});
|
||||
});
|
||||
|
||||
test('addTelemetry should not crash not given data', (done) => {
|
||||
|
||||
const logService = new NullLogService();
|
||||
TelemetryUtils.addTelemetry(telemetryService.object, logService, telemetryKey).then(() => {
|
||||
telemetryService.verify(x => x.publicLog(
|
||||
TypeMoq.It.is(a => a === telemetryKey),
|
||||
TypeMoq.It.is(b => b !== undefined)), TypeMoq.Times.once());
|
||||
done();
|
||||
}).catch(err => {
|
||||
assert.fail(err);
|
||||
done(err);
|
||||
});
|
||||
});
|
||||
|
||||
test('addTelemetry should try to get the provider name from data first', (done) => {
|
||||
let data: TelemetryUtils.IConnectionTelemetryData = {
|
||||
connection: connectionProfile
|
||||
};
|
||||
data.provider = providerName + '1';
|
||||
|
||||
const logService = new NullLogService();
|
||||
TelemetryUtils.addTelemetry(telemetryService.object, logService, telemetryKey, data, connectionProfile).then(() => {
|
||||
telemetryService.verify(x => x.publicLog(TypeMoq.It.is(a => a === telemetryKey), TypeMoq.It.is(b => b.provider === data.provider)), TypeMoq.Times.once());
|
||||
done();
|
||||
}).catch(err => {
|
||||
assert.fail(err);
|
||||
done(err);
|
||||
});
|
||||
});
|
||||
});
|
||||
Reference in New Issue
Block a user