mirror of
https://github.com/ckaczor/azuredatastudio.git
synced 2026-01-31 09:35:39 -05:00
move code from parts to contrib (#8319)
This commit is contained in:
@@ -0,0 +1,179 @@
|
||||
/*---------------------------------------------------------------------------------------------
|
||||
* Copyright (c) Microsoft Corporation. All rights reserved.
|
||||
* Licensed under the Source EULA. See License.txt in the project root for license information.
|
||||
*--------------------------------------------------------------------------------------------*/
|
||||
|
||||
import { IAdsTelemetryService } from 'sql/platform/telemetry/common/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';
|
||||
import { TelemetryView } from 'sql/platform/telemetry/common/telemetryKeys';
|
||||
|
||||
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'].some(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);
|
||||
Reference in New Issue
Block a user