Skip to content

Commit

Permalink
Bug 1647994 - ensure the sync ping always records the hashed UID if k…
Browse files Browse the repository at this point in the history
…nown. r=rfkelly

Differential Revision: https://phabricator.services.mozilla.com/D80816
  • Loading branch information
mhammond committed Jun 24, 2020
1 parent 87d28d6 commit 362bde7
Show file tree
Hide file tree
Showing 2 changed files with 59 additions and 23 deletions.
70 changes: 47 additions & 23 deletions services/sync/modules/telemetry.js
Original file line number Diff line number Diff line change
Expand Up @@ -340,13 +340,14 @@ class EngineRecord {
}
}

class TelemetryRecord {
// The record of a single "sync" - typically many of these are submitted in
// a single ping (ie, as a 'syncs' array)
class SyncRecord {
constructor(allowedEngines, why) {
this.allowedEngines = allowedEngines;
// Our failure reason. This property only exists in the generated ping if an
// error actually occurred.
this.failureReason = undefined;
this.uid = "";
this.syncNodeType = null;
this.when = Date.now();
this.startTime = tryGetMonotonicTimestamp();
Expand Down Expand Up @@ -393,7 +394,6 @@ class TelemetryRecord {
this.failureReason = SyncTelemetry.transformError(error);
}

this.uid = fxAccounts.telemetry.getSanitizedUID() || EMPTY_UID;
this.syncNodeType = Weave.Service.identity.telemetryNodeType;

// Check for engine statuses. -- We do this now, and not in engine.finished
Expand Down Expand Up @@ -551,6 +551,8 @@ function cleanErrorMessage(error) {
return error;
}

// The entire "sync ping" - it includes all the syncs, events etc recorded in
// the ping.
class SyncTelemetryImpl {
constructor(allowedEngines) {
log.manageLevelFromPref("services.sync.log.logger.telemetry");
Expand All @@ -571,6 +573,7 @@ class SyncTelemetryImpl {
this.lastSubmissionTime = Telemetry.msSinceProcessStart();
this.lastUID = EMPTY_UID;
this.lastSyncNodeType = null;
this.currentSyncNodeType = null;
// Note that the sessionStartDate is somewhat arbitrary - the telemetry
// modules themselves just use `new Date()`. This means that our startDate
// isn't going to be the same as the sessionStartDate in the main pings,
Expand Down Expand Up @@ -752,6 +755,8 @@ class SyncTelemetryImpl {
);
TelemetryController.submitExternalPing("sync", record, {
usePingSender: true,
}).catch(err => {
log.error("failed to submit ping", err);
});
return true;
}
Expand All @@ -778,7 +783,7 @@ class SyncTelemetryImpl {
// Just discard the old record, consistent with our handling of engines, above.
this.current = null;
}
this.current = new TelemetryRecord(this.allowedEngines, why);
this.current = new SyncRecord(this.allowedEngines, why);
}

// We need to ensure that the telemetry `deletion-request` ping always contains the user's
Expand Down Expand Up @@ -834,32 +839,56 @@ class SyncTelemetryImpl {
return true;
}

shouldSubmitForDataChange() {
let newID = this.current.uid;
_shouldSubmitForDataChange() {
let newID = fxAccounts.telemetry.getSanitizedUID() || EMPTY_UID;
let oldID = this.lastUID;
if (
newID != EMPTY_UID &&
oldID != EMPTY_UID &&
// Both are "real" uids, so we care if they've changed.
newID != oldID
) {
log.trace(
`shouldSubmitForDataChange - uid from '${oldID}' -> '${newID}'`
);
return true;
}
// We've gone from knowing one of the ids to not knowing it (which we
// ignore) or we've gone from not knowing it to knowing it (which is fine),
// Now check the node type because a change there also means we should
// submit.
if (
this.current.syncNodeType &&
this.lastSyncNodeType &&
this.current.syncNodeType != this.lastSyncNodeType
this.currentSyncNodeType != this.lastSyncNodeType
) {
log.trace(
`shouldSubmitForDataChange - nodeType from '${this.lastSyncNodeType}' -> '${this.currentSyncNodeType}'`
);
return true;
}
// We don't need to submit.
log.trace("shouldSubmitForDataChange - no need to submit");
return false;
}

maybeSubmitForDataChange() {
if (this._shouldSubmitForDataChange()) {
log.info(
"Early submission of sync telemetry due to changed IDs/NodeType"
);
this.finish("idchange"); // this actually submits.
this.lastSubmissionTime = Telemetry.msSinceProcessStart();
}

// Only update the last UIDs if we actually know them.
let current_uid = fxAccounts.telemetry.getSanitizedUID();
if (current_uid) {
this.lastUID = current_uid;
}
if (this.currentSyncNodeType) {
this.lastSyncNodeType = this.currentSyncNodeType;
}
}

maybeSubmitForInterval() {
// We want to submit the ping every `this.submissionInterval` but only when
// there's no current sync in progress, otherwise we may end up submitting
Expand All @@ -880,26 +909,19 @@ class SyncTelemetryImpl {
return;
}
this.current.finished(error);
if (this.payloads.length) {
if (this.shouldSubmitForDataChange()) {
log.info("Early submission of sync telemetry due to changed IDs");
this.finish("idchange");
this.lastSubmissionTime = Telemetry.msSinceProcessStart();
}
}
// Only update the last UIDs if we actually know them.
if (this.current.uid !== EMPTY_UID) {
this.lastUID = this.current.uid;
}
if (this.current.syncNodeType) {
this.lastSyncNodeType = this.current.syncNodeType;
}
this.currentSyncNodeType = this.current.syncNodeType;
// We check for "data change" before appending the current sync to payloads,
// as it is the current sync which has the data with the new data, and thus
// must go in the *next* submission.
this.maybeSubmitForDataChange();
if (this.payloads.length < this.maxPayloadCount) {
this.payloads.push(this.current.toJSON());
} else {
++this.discarded;
}
this.current = null;
// If we are submitting due to timing, it's desirable that the most recent
// sync is included, so we check after appending `this.current`.
this.maybeSubmitForInterval();
}

Expand All @@ -910,6 +932,8 @@ class SyncTelemetryImpl {
}

_recordEvent(eventDetails) {
this.maybeSubmitForDataChange();

if (this.events.length >= this.maxEventsCount) {
log.warn("discarding event - already queued our maximum", eventDetails);
return;
Expand Down
12 changes: 12 additions & 0 deletions services/sync/tests/unit/test_telemetry.js
Original file line number Diff line number Diff line change
Expand Up @@ -1290,6 +1290,8 @@ add_task(async function test_node_type_change() {
// Default to submitting each hour - we should still submit on node change.
let telem = get_sync_test_telemetry();
telem.submissionInterval = 60 * 60 * 1000;
// reset the node type from previous test or our first sync will submit.
telem.lastSyncNodeType = null;
// do 2 syncs with the same node type.
await Service.sync();
await Service.sync();
Expand All @@ -1308,6 +1310,16 @@ add_task(async function test_node_type_change() {
await promiseStopServer(server);
});

add_task(async function test_ids() {
let telem = get_sync_test_telemetry();
Assert.ok(!telem._shouldSubmitForDataChange());
fxAccounts.telemetry._setHashedUID("new_uid");
Assert.ok(telem._shouldSubmitForDataChange());
telem.maybeSubmitForDataChange();
// now it's been submitted the new uid is current.
Assert.ok(!telem._shouldSubmitForDataChange());
});

add_task(async function test_deletion_request_ping() {
async function assertRecordedSyncDeviceID(expected) {
// The scalar gets updated asynchronously, so wait a tick before checking.
Expand Down

0 comments on commit 362bde7

Please sign in to comment.