2018-06-15 14:33:07 +02:00
|
|
|
/*
|
2021-06-22 18:23:13 +02:00
|
|
|
Copyright 2018 - 2021 The Matrix.org Foundation C.I.C.
|
2018-06-15 14:33:07 +02:00
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2024-05-16 02:25:58 +02:00
|
|
|
import { CryptoEvent, HttpApiEvent, MatrixClient, MatrixEventEvent, MatrixEvent } from "matrix-js-sdk/src/matrix";
|
2022-06-07 21:08:36 +02:00
|
|
|
import { Error as ErrorEvent } from "@matrix-org/analytics-events/types/typescript/Error";
|
2024-04-17 14:36:01 +02:00
|
|
|
import { DecryptionFailureCode } from "matrix-js-sdk/src/crypto-api";
|
2022-01-19 20:31:43 +01:00
|
|
|
|
2022-12-12 12:24:14 +01:00
|
|
|
import { PosthogAnalytics } from "./PosthogAnalytics";
|
2021-06-22 18:23:13 +02:00
|
|
|
|
2018-07-05 14:54:44 +02:00
|
|
|
export class DecryptionFailure {
|
2024-05-16 02:25:58 +02:00
|
|
|
/**
|
|
|
|
* The time between our initial failure to decrypt and our successful
|
|
|
|
* decryption (if we managed to decrypt).
|
|
|
|
*/
|
|
|
|
public timeToDecryptMillis?: number;
|
2021-06-22 18:23:13 +02:00
|
|
|
|
2024-01-02 19:56:39 +01:00
|
|
|
public constructor(
|
|
|
|
public readonly failedEventId: string,
|
2024-04-17 14:36:01 +02:00
|
|
|
public readonly errorCode: DecryptionFailureCode,
|
2024-05-16 02:25:58 +02:00
|
|
|
/**
|
|
|
|
* The time that we failed to decrypt the event. If we failed to decrypt
|
|
|
|
* multiple times, this will be the time of the first failure.
|
|
|
|
*/
|
|
|
|
public readonly ts: number,
|
|
|
|
/**
|
|
|
|
* Is the sender on a different server from us?
|
|
|
|
*/
|
|
|
|
public readonly isFederated: boolean | undefined,
|
|
|
|
/**
|
|
|
|
* Was the failed event ever visible to the user?
|
|
|
|
*/
|
|
|
|
public wasVisibleToUser: boolean,
|
|
|
|
/**
|
|
|
|
* Has the user verified their own cross-signing identity, as of the most
|
|
|
|
* recent decryption attempt for this event?
|
|
|
|
*/
|
|
|
|
public userTrustsOwnIdentity: boolean | undefined,
|
|
|
|
) {}
|
2018-06-15 14:33:07 +02:00
|
|
|
}
|
|
|
|
|
2024-04-17 14:36:01 +02:00
|
|
|
type ErrorCode = ErrorEvent["name"];
|
2024-05-16 02:25:58 +02:00
|
|
|
/** Properties associated with decryption errors, for classifying the error. */
|
|
|
|
export type ErrorProperties = Omit<ErrorEvent, "eventName" | "domain" | "name" | "context">;
|
|
|
|
type TrackingFn = (trackedErrCode: ErrorCode, rawError: string, properties: ErrorProperties) => void;
|
2024-04-17 14:36:01 +02:00
|
|
|
export type ErrCodeMapFn = (errcode: DecryptionFailureCode) => ErrorCode;
|
2021-06-22 18:23:13 +02:00
|
|
|
|
2018-07-05 14:54:44 +02:00
|
|
|
export class DecryptionFailureTracker {
|
2022-12-12 12:24:14 +01:00
|
|
|
private static internalInstance = new DecryptionFailureTracker(
|
2024-05-16 02:25:58 +02:00
|
|
|
(errorCode, rawError, properties) => {
|
|
|
|
const event: ErrorEvent = {
|
|
|
|
eventName: "Error",
|
|
|
|
domain: "E2EE",
|
|
|
|
name: errorCode,
|
|
|
|
context: `mxc_crypto_error_type_${rawError}`,
|
|
|
|
...properties,
|
|
|
|
};
|
|
|
|
PosthogAnalytics.instance.trackEvent<ErrorEvent>(event);
|
2022-12-12 12:24:14 +01:00
|
|
|
},
|
|
|
|
(errorCode) => {
|
|
|
|
// Map JS-SDK error codes to tracker codes for aggregation
|
|
|
|
switch (errorCode) {
|
2024-04-17 14:36:01 +02:00
|
|
|
case DecryptionFailureCode.MEGOLM_UNKNOWN_INBOUND_SESSION_ID:
|
2022-12-12 12:24:14 +01:00
|
|
|
return "OlmKeysNotSentError";
|
2024-04-17 14:36:01 +02:00
|
|
|
case DecryptionFailureCode.OLM_UNKNOWN_MESSAGE_INDEX:
|
2022-12-12 12:24:14 +01:00
|
|
|
return "OlmIndexError";
|
2024-04-17 14:36:01 +02:00
|
|
|
case DecryptionFailureCode.HISTORICAL_MESSAGE_NO_KEY_BACKUP:
|
|
|
|
case DecryptionFailureCode.HISTORICAL_MESSAGE_BACKUP_UNCONFIGURED:
|
|
|
|
case DecryptionFailureCode.HISTORICAL_MESSAGE_WORKING_BACKUP:
|
|
|
|
return "HistoricalMessage";
|
2024-04-29 19:18:57 +02:00
|
|
|
case DecryptionFailureCode.HISTORICAL_MESSAGE_USER_NOT_JOINED:
|
|
|
|
return "ExpectedDueToMembership";
|
2022-12-12 12:24:14 +01:00
|
|
|
default:
|
|
|
|
return "UnknownError";
|
|
|
|
}
|
|
|
|
},
|
|
|
|
);
|
2022-01-19 20:31:43 +01:00
|
|
|
|
2024-05-16 02:25:58 +02:00
|
|
|
/** Map of event IDs to `DecryptionFailure` items.
|
|
|
|
*
|
|
|
|
* Every `CHECK_INTERVAL_MS`, this map is checked for failures that happened >
|
|
|
|
* `MAXIMUM_LATE_DECRYPTION_PERIOD` ago (considered undecryptable), or
|
|
|
|
* decryptions that took > `GRACE_PERIOD_MS` (considered late decryptions).
|
|
|
|
* These are accumulated in `failuresToReport`.
|
|
|
|
*/
|
2022-01-19 20:31:43 +01:00
|
|
|
public failures: Map<string, DecryptionFailure> = new Map();
|
|
|
|
|
2024-05-16 02:25:58 +02:00
|
|
|
/** Set of event IDs that have been visible to the user.
|
|
|
|
*
|
|
|
|
* This will only contain events that are not already in `failures` or in
|
|
|
|
* `trackedEvents`.
|
|
|
|
*/
|
2022-01-19 20:31:43 +01:00
|
|
|
public visibleEvents: Set<string> = new Set();
|
|
|
|
|
2024-05-16 02:25:58 +02:00
|
|
|
/** The failures that will be reported at the next tracking interval. These are
|
|
|
|
* events that we have decided are undecryptable due to exceeding the
|
|
|
|
* `MAXIMUM_LATE_DECRYPTION_PERIOD`, or that we decrypted but we consider as late
|
|
|
|
* decryptions. */
|
|
|
|
public failuresToReport: Set<DecryptionFailure> = new Set();
|
2018-06-15 14:33:07 +02:00
|
|
|
|
2024-05-16 02:25:58 +02:00
|
|
|
/** Event IDs of failures that were tracked previously */
|
2022-01-19 20:31:43 +01:00
|
|
|
public trackedEvents: Set<string> = new Set();
|
2018-06-15 15:48:20 +02:00
|
|
|
|
2024-05-16 02:25:58 +02:00
|
|
|
/** Set to an interval ID when `start` is called */
|
2023-02-03 16:27:47 +01:00
|
|
|
public checkInterval: number | null = null;
|
|
|
|
public trackInterval: number | null = null;
|
2018-06-15 18:58:43 +02:00
|
|
|
|
2024-05-16 02:25:58 +02:00
|
|
|
/** Spread the load on `Analytics` by tracking at a low frequency, `TRACK_INTERVAL_MS`. */
|
2022-12-16 13:29:59 +01:00
|
|
|
public static TRACK_INTERVAL_MS = 60000;
|
2018-06-15 14:33:07 +02:00
|
|
|
|
2024-05-16 02:25:58 +02:00
|
|
|
/** Call `checkFailures` every `CHECK_INTERVAL_MS`. */
|
2024-02-29 16:29:59 +01:00
|
|
|
public static CHECK_INTERVAL_MS = 40000;
|
2018-06-15 14:33:07 +02:00
|
|
|
|
2024-05-16 02:25:58 +02:00
|
|
|
/** If the event is successfully decrypted in less than 4s, we don't report. */
|
|
|
|
public static GRACE_PERIOD_MS = 4000;
|
|
|
|
|
|
|
|
/** Maximum time for an event to be decrypted to be considered a late
|
|
|
|
* decryption. If it takes longer, we consider it undecryptable. */
|
|
|
|
public static MAXIMUM_LATE_DECRYPTION_PERIOD = 60000;
|
|
|
|
|
|
|
|
/** Properties that will be added to all reported events (mainly reporting
|
|
|
|
* information about the Matrix client). */
|
|
|
|
private baseProperties?: ErrorProperties = {};
|
|
|
|
|
|
|
|
/** The user's domain (homeserver name). */
|
|
|
|
private userDomain?: string;
|
|
|
|
|
|
|
|
/** Whether the user has verified their own cross-signing keys. */
|
|
|
|
private userTrustsOwnIdentity: boolean | undefined = undefined;
|
|
|
|
|
|
|
|
/** Whether we are currently checking our own verification status. */
|
|
|
|
private checkingVerificationStatus: boolean = false;
|
|
|
|
|
|
|
|
/** Whether we should retry checking our own verification status after we're
|
|
|
|
* done our current check. i.e. we got notified that our keys changed while
|
|
|
|
* we were already checking, so the result could be out of date. */
|
|
|
|
private retryVerificationStatus: boolean = false;
|
2018-06-15 14:33:07 +02:00
|
|
|
|
2018-07-05 14:54:44 +02:00
|
|
|
/**
|
|
|
|
* Create a new DecryptionFailureTracker.
|
|
|
|
*
|
2024-05-16 02:25:58 +02:00
|
|
|
* Call `start(client)` to start the tracker. The tracker will listen for
|
|
|
|
* decryption events on the client and track decryption failures, and will
|
|
|
|
* automatically stop tracking when the client logs out.
|
2018-07-05 14:54:44 +02:00
|
|
|
*
|
|
|
|
* @param {function} fn The tracking function, which will be called when failures
|
2024-05-16 02:25:58 +02:00
|
|
|
* are tracked. The function should have a signature `(trackedErrorCode, rawError, properties) => {...}`,
|
|
|
|
* where `errorCode` matches the output of `errorCodeMapFn`, `rawError` is the original
|
|
|
|
* error (that is, the input to `errorCodeMapFn`), and `properties` is a map of the
|
|
|
|
* error properties for classifying the error.
|
2024-04-17 14:36:01 +02:00
|
|
|
*
|
|
|
|
* @param {function} errorCodeMapFn The function used to map decryption failure reason codes to the
|
|
|
|
* `trackedErrorCode`.
|
2018-07-05 14:54:44 +02:00
|
|
|
*/
|
2024-01-02 19:56:39 +01:00
|
|
|
private constructor(
|
|
|
|
private readonly fn: TrackingFn,
|
|
|
|
private readonly errorCodeMapFn: ErrCodeMapFn,
|
|
|
|
) {
|
2022-12-12 12:24:14 +01:00
|
|
|
if (!fn || typeof fn !== "function") {
|
|
|
|
throw new Error("DecryptionFailureTracker requires tracking function");
|
2018-06-15 14:33:07 +02:00
|
|
|
}
|
|
|
|
|
2022-12-12 12:24:14 +01:00
|
|
|
if (typeof errorCodeMapFn !== "function") {
|
|
|
|
throw new Error("DecryptionFailureTracker second constructor argument should be a function");
|
2018-07-05 14:54:44 +02:00
|
|
|
}
|
2018-06-15 14:33:07 +02:00
|
|
|
}
|
|
|
|
|
2022-01-19 20:31:43 +01:00
|
|
|
public static get instance(): DecryptionFailureTracker {
|
|
|
|
return DecryptionFailureTracker.internalInstance;
|
|
|
|
}
|
|
|
|
|
|
|
|
// loadTrackedEvents() {
|
|
|
|
// this.trackedEvents = new Set(JSON.parse(localStorage.getItem('mx-decryption-failure-event-ids')) || []);
|
2018-06-15 18:08:11 +02:00
|
|
|
// }
|
2018-06-15 16:26:53 +02:00
|
|
|
|
2022-01-19 20:31:43 +01:00
|
|
|
// saveTrackedEvents() {
|
|
|
|
// localStorage.setItem('mx-decryption-failure-event-ids', JSON.stringify([...this.trackedEvents]));
|
2018-06-15 18:08:11 +02:00
|
|
|
// }
|
2018-06-15 16:26:53 +02:00
|
|
|
|
2024-05-16 02:25:58 +02:00
|
|
|
/** Callback for when an event is decrypted.
|
|
|
|
*
|
|
|
|
* This function is called by our `MatrixEventEvent.Decrypted` event
|
|
|
|
* handler after a decryption attempt on an event, whether the decryption
|
|
|
|
* is successful or not.
|
|
|
|
*
|
|
|
|
* @param matrixEvent the event that was decrypted
|
|
|
|
*
|
|
|
|
* @param nowTs the current timestamp
|
|
|
|
*/
|
|
|
|
private eventDecrypted(e: MatrixEvent, nowTs: number): void {
|
2024-04-17 14:36:01 +02:00
|
|
|
// for now we only track megolm decryption failures
|
2022-06-30 09:55:05 +02:00
|
|
|
if (e.getWireContent().algorithm != "m.megolm.v1.aes-sha2") {
|
|
|
|
return;
|
|
|
|
}
|
2024-04-17 14:36:01 +02:00
|
|
|
const errCode = e.decryptionFailureReason;
|
2024-05-16 02:25:58 +02:00
|
|
|
if (errCode === null) {
|
2018-06-15 14:33:07 +02:00
|
|
|
// Could be an event in the failures, remove it
|
2024-05-16 02:25:58 +02:00
|
|
|
this.removeDecryptionFailuresForEvent(e, nowTs);
|
|
|
|
return;
|
2018-06-15 14:33:07 +02:00
|
|
|
}
|
2024-05-16 02:25:58 +02:00
|
|
|
|
|
|
|
const eventId = e.getId()!;
|
|
|
|
|
|
|
|
// if we already have a record of this event, use the previously-recorded timestamp
|
|
|
|
const failure = this.failures.get(eventId);
|
|
|
|
const ts = failure ? failure.ts : nowTs;
|
|
|
|
|
|
|
|
const sender = e.getSender();
|
|
|
|
const senderDomain = sender?.replace(/^.*?:/, "");
|
|
|
|
let isFederated: boolean | undefined;
|
|
|
|
if (this.userDomain !== undefined && senderDomain !== undefined) {
|
|
|
|
isFederated = this.userDomain !== senderDomain;
|
|
|
|
}
|
|
|
|
const wasVisibleToUser = this.visibleEvents.has(eventId);
|
|
|
|
this.addDecryptionFailure(
|
|
|
|
new DecryptionFailure(eventId, errCode, ts, isFederated, wasVisibleToUser, this.userTrustsOwnIdentity),
|
|
|
|
);
|
2018-06-15 14:33:07 +02:00
|
|
|
}
|
|
|
|
|
2022-01-19 20:31:43 +01:00
|
|
|
public addVisibleEvent(e: MatrixEvent): void {
|
2023-02-16 18:21:44 +01:00
|
|
|
const eventId = e.getId()!;
|
2022-01-19 20:31:43 +01:00
|
|
|
|
2024-05-16 02:25:58 +02:00
|
|
|
// if it's already reported, we don't need to do anything
|
2022-12-12 12:24:14 +01:00
|
|
|
if (this.trackedEvents.has(eventId)) {
|
|
|
|
return;
|
|
|
|
}
|
2022-01-19 20:31:43 +01:00
|
|
|
|
2024-05-16 02:25:58 +02:00
|
|
|
// if we've already marked the event as a failure, mark it as visible
|
|
|
|
// in the failure object
|
|
|
|
const failure = this.failures.get(eventId);
|
|
|
|
if (failure) {
|
|
|
|
failure.wasVisibleToUser = true;
|
2022-01-19 20:31:43 +01:00
|
|
|
}
|
2024-05-16 02:25:58 +02:00
|
|
|
|
|
|
|
this.visibleEvents.add(eventId);
|
2022-01-19 20:31:43 +01:00
|
|
|
}
|
|
|
|
|
2021-06-22 18:23:13 +02:00
|
|
|
public addDecryptionFailure(failure: DecryptionFailure): void {
|
2022-01-19 20:31:43 +01:00
|
|
|
const eventId = failure.failedEventId;
|
|
|
|
|
2022-12-12 12:24:14 +01:00
|
|
|
if (this.trackedEvents.has(eventId)) {
|
|
|
|
return;
|
|
|
|
}
|
2022-01-19 20:31:43 +01:00
|
|
|
|
|
|
|
this.failures.set(eventId, failure);
|
2018-06-15 14:33:07 +02:00
|
|
|
}
|
|
|
|
|
2024-05-16 02:25:58 +02:00
|
|
|
public removeDecryptionFailuresForEvent(e: MatrixEvent, nowTs: number): void {
|
2023-02-16 18:21:44 +01:00
|
|
|
const eventId = e.getId()!;
|
2024-05-16 02:25:58 +02:00
|
|
|
const failure = this.failures.get(eventId);
|
|
|
|
if (failure) {
|
|
|
|
this.failures.delete(eventId);
|
|
|
|
|
|
|
|
const timeToDecryptMillis = nowTs - failure.ts;
|
|
|
|
if (timeToDecryptMillis < DecryptionFailureTracker.GRACE_PERIOD_MS) {
|
|
|
|
// the event decrypted on time, so we don't need to report it
|
|
|
|
return;
|
|
|
|
} else if (timeToDecryptMillis <= DecryptionFailureTracker.MAXIMUM_LATE_DECRYPTION_PERIOD) {
|
|
|
|
// The event is a late decryption, so store the time it took.
|
|
|
|
// If the time to decrypt is longer than
|
|
|
|
// MAXIMUM_LATE_DECRYPTION_PERIOD, we consider the event as
|
|
|
|
// undecryptable, and leave timeToDecryptMillis undefined
|
|
|
|
failure.timeToDecryptMillis = timeToDecryptMillis;
|
|
|
|
}
|
|
|
|
this.failuresToReport.add(failure);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private async handleKeysChanged(client: MatrixClient): Promise<void> {
|
|
|
|
if (this.checkingVerificationStatus) {
|
|
|
|
// Flag that we'll need to do another check once the current check completes.
|
|
|
|
this.retryVerificationStatus = true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.checkingVerificationStatus = true;
|
|
|
|
try {
|
|
|
|
do {
|
|
|
|
this.retryVerificationStatus = false;
|
|
|
|
this.userTrustsOwnIdentity = (
|
|
|
|
await client.getCrypto()!.getUserVerificationStatus(client.getUserId()!)
|
|
|
|
).isCrossSigningVerified();
|
|
|
|
} while (this.retryVerificationStatus);
|
|
|
|
} finally {
|
|
|
|
this.checkingVerificationStatus = false;
|
|
|
|
}
|
2018-06-15 14:33:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Start checking for and tracking failures.
|
|
|
|
*/
|
2024-05-16 02:25:58 +02:00
|
|
|
public start(client: MatrixClient): void {
|
|
|
|
this.calculateClientProperties(client);
|
|
|
|
this.registerHandlers(client);
|
2022-11-30 12:32:56 +01:00
|
|
|
this.checkInterval = window.setInterval(
|
2018-06-15 14:33:07 +02:00
|
|
|
() => this.checkFailures(Date.now()),
|
|
|
|
DecryptionFailureTracker.CHECK_INTERVAL_MS,
|
|
|
|
);
|
|
|
|
|
2022-12-12 12:24:14 +01:00
|
|
|
this.trackInterval = window.setInterval(() => this.trackFailures(), DecryptionFailureTracker.TRACK_INTERVAL_MS);
|
2018-06-15 18:58:43 +02:00
|
|
|
}
|
2018-06-15 14:33:07 +02:00
|
|
|
|
2024-05-16 02:25:58 +02:00
|
|
|
private async calculateClientProperties(client: MatrixClient): Promise<void> {
|
|
|
|
const baseProperties: ErrorProperties = {};
|
|
|
|
this.baseProperties = baseProperties;
|
|
|
|
|
|
|
|
this.userDomain = client.getDomain() ?? undefined;
|
|
|
|
if (this.userDomain === "matrix.org") {
|
|
|
|
baseProperties.isMatrixDotOrg = true;
|
|
|
|
} else if (this.userDomain !== undefined) {
|
|
|
|
baseProperties.isMatrixDotOrg = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
const crypto = client.getCrypto();
|
|
|
|
if (crypto) {
|
|
|
|
const version = crypto.getVersion();
|
|
|
|
if (version.startsWith("Rust SDK")) {
|
|
|
|
baseProperties.cryptoSDK = "Rust";
|
|
|
|
} else {
|
|
|
|
baseProperties.cryptoSDK = "Legacy";
|
|
|
|
}
|
|
|
|
this.userTrustsOwnIdentity = (
|
|
|
|
await crypto.getUserVerificationStatus(client.getUserId()!)
|
|
|
|
).isCrossSigningVerified();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private registerHandlers(client: MatrixClient): void {
|
|
|
|
// After the client attempts to decrypt an event, we examine it to see
|
|
|
|
// if it needs to be reported.
|
|
|
|
const decryptedHandler = (e: MatrixEvent): void => this.eventDecrypted(e, Date.now());
|
|
|
|
// When our keys change, we check if the cross-signing keys are now trusted.
|
|
|
|
const keysChangedHandler = (): void => {
|
|
|
|
this.handleKeysChanged(client).catch((e) => {
|
|
|
|
console.log("Error handling KeysChanged event", e);
|
|
|
|
});
|
|
|
|
};
|
|
|
|
// When logging out, remove our handlers and destroy state
|
|
|
|
const loggedOutHandler = (): void => {
|
|
|
|
client.removeListener(MatrixEventEvent.Decrypted, decryptedHandler);
|
|
|
|
client.removeListener(CryptoEvent.KeysChanged, keysChangedHandler);
|
|
|
|
client.removeListener(HttpApiEvent.SessionLoggedOut, loggedOutHandler);
|
|
|
|
this.stop();
|
|
|
|
};
|
|
|
|
|
|
|
|
client.on(MatrixEventEvent.Decrypted, decryptedHandler);
|
|
|
|
client.on(CryptoEvent.KeysChanged, keysChangedHandler);
|
|
|
|
client.on(HttpApiEvent.SessionLoggedOut, loggedOutHandler);
|
|
|
|
}
|
|
|
|
|
2018-06-15 18:58:43 +02:00
|
|
|
/**
|
|
|
|
* Clear state and stop checking for and tracking failures.
|
|
|
|
*/
|
2024-05-16 02:25:58 +02:00
|
|
|
private stop(): void {
|
2023-02-16 18:21:44 +01:00
|
|
|
if (this.checkInterval) clearInterval(this.checkInterval);
|
|
|
|
if (this.trackInterval) clearInterval(this.trackInterval);
|
2018-06-15 14:33:07 +02:00
|
|
|
|
2024-05-16 02:25:58 +02:00
|
|
|
this.userTrustsOwnIdentity = undefined;
|
2022-01-19 20:31:43 +01:00
|
|
|
this.failures = new Map();
|
|
|
|
this.visibleEvents = new Set();
|
2024-05-16 02:25:58 +02:00
|
|
|
this.failuresToReport = new Set();
|
2018-06-15 14:33:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2024-05-16 02:25:58 +02:00
|
|
|
* Mark failures as undecryptable or late. Only mark one failure per event ID.
|
|
|
|
*
|
2018-06-15 14:33:07 +02:00
|
|
|
* @param {number} nowTs the timestamp that represents the time now.
|
|
|
|
*/
|
2021-06-22 18:23:13 +02:00
|
|
|
public checkFailures(nowTs: number): void {
|
2022-01-19 20:31:43 +01:00
|
|
|
const failuresNotReady: Map<string, DecryptionFailure> = new Map();
|
2024-05-16 02:25:58 +02:00
|
|
|
for (const [eventId, failure] of this.failures) {
|
|
|
|
if (
|
|
|
|
failure.timeToDecryptMillis !== undefined ||
|
|
|
|
nowTs > failure.ts + DecryptionFailureTracker.MAXIMUM_LATE_DECRYPTION_PERIOD
|
|
|
|
) {
|
|
|
|
// we report failures under two conditions:
|
|
|
|
// - if `timeToDecryptMillis` is set, we successfully decrypted
|
|
|
|
// the event, but we got the key late. We report it so that we
|
|
|
|
// have the late decrytion stats.
|
|
|
|
// - we haven't decrypted yet and it's past the time for it to be
|
|
|
|
// considered a "late" decryption, so we count it as
|
|
|
|
// undecryptable.
|
|
|
|
this.addFailure(eventId, failure);
|
2018-06-15 15:45:11 +02:00
|
|
|
} else {
|
2024-05-16 02:25:58 +02:00
|
|
|
// the event isn't old enough, so we still need to keep track of it
|
2022-01-19 20:31:43 +01:00
|
|
|
failuresNotReady.set(eventId, failure);
|
2018-06-15 15:45:11 +02:00
|
|
|
}
|
|
|
|
}
|
2024-05-16 02:25:58 +02:00
|
|
|
this.failures = failuresNotReady;
|
2018-06-15 15:48:20 +02:00
|
|
|
|
2018-06-15 18:08:11 +02:00
|
|
|
// Commented out for now for expediency, we need to consider unbound nature of storing
|
|
|
|
// this in localStorage
|
2022-01-19 20:31:43 +01:00
|
|
|
// this.saveTrackedEvents();
|
2018-07-05 14:54:44 +02:00
|
|
|
}
|
|
|
|
|
2024-05-16 02:25:58 +02:00
|
|
|
private addFailure(eventId: string, failure: DecryptionFailure): void {
|
|
|
|
this.failuresToReport.add(failure);
|
|
|
|
this.trackedEvents.add(eventId);
|
|
|
|
// once we've added it to trackedEvents, we won't check
|
|
|
|
// visibleEvents for it any more
|
|
|
|
this.visibleEvents.delete(eventId);
|
2018-06-15 14:33:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2018-06-28 17:06:12 +02:00
|
|
|
* If there are failures that should be tracked, call the given trackDecryptionFailure
|
2024-05-16 02:25:58 +02:00
|
|
|
* function with the failures that should be tracked.
|
2018-06-15 14:33:07 +02:00
|
|
|
*/
|
2021-06-22 18:23:13 +02:00
|
|
|
public trackFailures(): void {
|
2024-05-16 02:25:58 +02:00
|
|
|
for (const failure of this.failuresToReport) {
|
|
|
|
const errorCode = failure.errorCode;
|
|
|
|
const trackedErrorCode = this.errorCodeMapFn(errorCode);
|
|
|
|
const properties: ErrorProperties = {
|
|
|
|
timeToDecryptMillis: failure.timeToDecryptMillis ?? -1,
|
|
|
|
wasVisibleToUser: failure.wasVisibleToUser,
|
|
|
|
};
|
|
|
|
if (failure.isFederated !== undefined) {
|
|
|
|
properties.isFederated = failure.isFederated;
|
|
|
|
}
|
|
|
|
if (failure.userTrustsOwnIdentity !== undefined) {
|
|
|
|
properties.userTrustsOwnIdentity = failure.userTrustsOwnIdentity;
|
|
|
|
}
|
|
|
|
if (this.baseProperties) {
|
|
|
|
Object.assign(properties, this.baseProperties);
|
2018-07-05 14:54:44 +02:00
|
|
|
}
|
2024-05-16 02:25:58 +02:00
|
|
|
this.fn(trackedErrorCode, errorCode, properties);
|
2018-06-15 14:33:07 +02:00
|
|
|
}
|
2024-05-16 02:25:58 +02:00
|
|
|
this.failuresToReport = new Set();
|
2018-06-15 14:33:07 +02:00
|
|
|
}
|
|
|
|
}
|