Skip to content

Commit 7f18de6

Browse files
feat: enable to possibility to ensure analytics tracking is finished
Introudce finishTracking method in `analyticsService` which will ensure all pending information is send to Google Analytics. This is required for some rare cases in which we want to be sure all the information has been sent.
1 parent f7eaafa commit 7f18de6

File tree

4 files changed

+60
-1
lines changed

4 files changed

+60
-1
lines changed

lib/common/declarations.d.ts

+7
Original file line numberDiff line numberDiff line change
@@ -207,6 +207,12 @@ declare const enum TrackingTypes {
207207
* Defines that the broker process should get and track the data from preview app to Google Analytics
208208
*/
209209
PreviewAppData = "PreviewAppData",
210+
211+
/**
212+
* Defines that the broker process should send all the pending information to Analytics.
213+
* After that the process should send information it has finished tracking and die gracefully.
214+
*/
215+
FinishTracking = "FinishTracking",
210216
}
211217

212218
/**
@@ -688,6 +694,7 @@ interface IAnalyticsService {
688694
setStatus(settingName: string, enabled: boolean): Promise<void>;
689695
getStatusMessage(settingName: string, jsonFormat: boolean, readableSettingName: string): Promise<string>;
690696
isEnabled(settingName: string): Promise<boolean>;
697+
finishTracking(): Promise<void>;
691698

692699
/**
693700
* Tracks the answer of question if user allows to be tracked.

lib/detached-processes/detached-process-enums.d.ts

+6-1
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,12 @@ declare const enum DetachedProcessMessages {
55
/**
66
* The detached process is initialized and is ready to receive information for tracking.
77
*/
8-
ProcessReadyToReceive = "ProcessReadyToReceive"
8+
ProcessReadyToReceive = "ProcessReadyToReceive",
9+
10+
/**
11+
* The detached process finished its tasks and will now exit.
12+
*/
13+
ProcessFinishedTasks = "ProcessFinishedTasks"
914
}
1015

1116
/**

lib/services/analytics/analytics-broker-process.ts

+19
Original file line numberDiff line numberDiff line change
@@ -29,6 +29,10 @@ const finishTracking = async (data?: ITrackingInformation) => {
2929
analyticsLoggingService.logData({ message: `analytics-broker-process finish tracking started` });
3030
await trackingQueue;
3131
analyticsLoggingService.logData({ message: `analytics-broker-process tracking finished` });
32+
33+
};
34+
35+
const killCurrentProcessGracefully = () => {
3236
$injector.dispose();
3337
process.exit();
3438
};
@@ -68,12 +72,27 @@ process.on("message", async (data: ITrackingInformation) => {
6872
return;
6973
}
7074

75+
if (data.type === TrackingTypes.FinishTracking) {
76+
await finishTracking();
77+
78+
if (process.connected) {
79+
analyticsLoggingService.logData({ message: `analytics-broker-process will send ${DetachedProcessMessages.ProcessFinishedTasks} message` });
80+
process.send(DetachedProcessMessages.ProcessFinishedTasks, () => {
81+
analyticsLoggingService.logData({ message: `analytics-broker-process sent ${DetachedProcessMessages.ProcessFinishedTasks} message and will exit gracefully now` });
82+
killCurrentProcessGracefully();
83+
});
84+
}
85+
86+
return;
87+
}
88+
7189
await sendDataForTracking(data);
7290
});
7391

7492
process.on("disconnect", async () => {
7593
analyticsLoggingService.logData({ message: "analytics-broker-process received process.disconnect event" });
7694
await finishTracking();
95+
killCurrentProcessGracefully();
7796
});
7897

7998
analyticsLoggingService.logData({ message: `analytics-broker-process will send ${DetachedProcessMessages.ProcessReadyToReceive} message` });

lib/services/analytics/analytics-service.ts

+28
Original file line numberDiff line numberDiff line change
@@ -146,6 +146,34 @@ export class AnalyticsService implements IAnalyticsService, IDisposable {
146146
await this.trackInGoogleAnalytics(eventActionData);
147147
}
148148

149+
public async finishTracking(): Promise<void> {
150+
return new Promise((resolve, reject) => {
151+
if (this.brokerProcess && this.brokerProcess.connected) {
152+
let timer: NodeJS.Timer;
153+
154+
const handler = (data: string) => {
155+
if (data === DetachedProcessMessages.ProcessFinishedTasks) {
156+
this.brokerProcess.removeListener("message", handler);
157+
clearTimeout(timer);
158+
resolve();
159+
}
160+
};
161+
162+
timer = setTimeout(() => {
163+
this.brokerProcess.removeListener("message", handler);
164+
resolve();
165+
}, 3000);
166+
167+
this.brokerProcess.on("message", handler);
168+
169+
const msg = { type: TrackingTypes.FinishTracking };
170+
this.brokerProcess.send(msg, (err: Error) => this.$logger.trace(`Error while sending ${JSON.stringify(msg)}`));
171+
} else {
172+
resolve();
173+
}
174+
});
175+
}
176+
149177
private forcefullyTrackInGoogleAnalytics(gaSettings: IGoogleAnalyticsData): Promise<void> {
150178
gaSettings.customDimensions = gaSettings.customDimensions || {};
151179
gaSettings.customDimensions[GoogleAnalyticsCustomDimensions.client] = this.$options.analyticsClient || (isInteractive() ? AnalyticsClients.Cli : AnalyticsClients.Unknown);

0 commit comments

Comments
 (0)