Skip to content

Commit 2d2e0a3

Browse files
fix: ensure data is sent to analytics on preuninstall
CLI's analytics process is a detached one, so during `preuninstall` command, CLI just sends information to the detached process what should be tracked and finishes CLI's execution. After that `npm` starts removing the CLI package and its `node_modules`. At this point the analytics process may still work and trying to send data, but as it relies on components in node_modules, which npm currently deletes, it fails to send information to Google Analytics. To ensure correct data is send to Analytics, ensure tracking is finished before ending the preuninstall command.
1 parent 7f18de6 commit 2d2e0a3

File tree

2 files changed

+10
-1
lines changed

2 files changed

+10
-1
lines changed

lib/common/commands/preuninstall.ts

+1
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,7 @@ export class PreUninstallCommand implements ICommand {
2727
}
2828

2929
this.$fs.deleteFile(path.join(this.$settingsService.getProfileDir(), "KillSwitches", "cli"));
30+
await this.$analyticsService.finishTracking();
3031
}
3132

3233
private async handleIntentionalUninstall(): Promise<void> {

lib/common/test/unit-tests/preuninstall.ts

+9-1
Original file line numberDiff line numberDiff line change
@@ -30,7 +30,8 @@ describe("preuninstall", () => {
3030
});
3131

3232
testInjector.register("analyticsService", {
33-
trackEventActionInGoogleAnalytics: async (data: IEventActionData): Promise<void> => undefined
33+
trackEventActionInGoogleAnalytics: async (data: IEventActionData): Promise<void> => undefined,
34+
finishTracking: async(): Promise<void> => undefined
3435
});
3536

3637
testInjector.registerCommand("dev-preuninstall", PreUninstallCommand);
@@ -83,15 +84,22 @@ describe("preuninstall", () => {
8384
trackedData.push(data);
8485
};
8586

87+
let isFinishTrackingCalled = false;
88+
analyticsService.finishTracking = async (): Promise<void> => {
89+
isFinishTrackingCalled = true;
90+
};
91+
8692
const preUninstallCommand: ICommand = testInjector.resolveCommand("dev-preuninstall");
8793
for (const testCase of testData) {
8894
helpers.isInteractive = () => testCase.isInteractive;
8995
helpers.doesCurrentNpmCommandMatch = () => testCase.isIntentionalUninstall;
96+
isFinishTrackingCalled = false;
9097
await preUninstallCommand.execute([]);
9198
assert.deepEqual(trackedData, [{
9299
action: "Uninstall CLI",
93100
additionalData: testCase.expecteEventLabelData
94101
}]);
102+
assert.isTrue(isFinishTrackingCalled, "At the end of the command, finishTracking must be called");
95103
trackedData = [];
96104
}
97105
});

0 commit comments

Comments
 (0)