2018-06-15 12:33:07 +00:00
|
|
|
/*
|
|
|
|
Copyright 2018 New Vector Ltd
|
|
|
|
|
|
|
|
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 00:25:58 +00:00
|
|
|
import { mocked, Mocked } from "jest-mock";
|
|
|
|
import { CryptoEvent, HttpApiEvent, MatrixEvent, MatrixEventEvent } from "matrix-js-sdk/src/matrix";
|
2024-03-25 19:57:29 +00:00
|
|
|
import { decryptExistingEvent, mkDecryptionFailureMatrixEvent } from "matrix-js-sdk/src/testing";
|
2024-05-16 00:25:58 +00:00
|
|
|
import { CryptoApi, DecryptionFailureCode, UserVerificationStatus } from "matrix-js-sdk/src/crypto-api";
|
|
|
|
import { sleep } from "matrix-js-sdk/src/utils";
|
2018-06-15 12:33:07 +00:00
|
|
|
|
2024-05-16 00:25:58 +00:00
|
|
|
import { DecryptionFailureTracker, ErrorProperties } from "../src/DecryptionFailureTracker";
|
|
|
|
import { stubClient } from "./test-utils";
|
2021-10-22 22:23:32 +00:00
|
|
|
|
2024-05-16 00:25:58 +00:00
|
|
|
async function createFailedDecryptionEvent(opts: { sender?: string; code?: DecryptionFailureCode } = {}) {
|
2024-03-22 15:48:29 +00:00
|
|
|
return await mkDecryptionFailureMatrixEvent({
|
|
|
|
roomId: "!room:id",
|
2024-05-16 00:25:58 +00:00
|
|
|
sender: opts.sender ?? "@alice:example.com",
|
|
|
|
code: opts.code ?? DecryptionFailureCode.UNKNOWN_ERROR,
|
2024-03-22 15:48:29 +00:00
|
|
|
msg: ":(",
|
2018-06-15 12:33:07 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2024-05-16 00:25:58 +00:00
|
|
|
// wrap tracker.eventDecrypted so that we don't need to have so many `ts-ignore`s
|
|
|
|
function eventDecrypted(tracker: DecryptionFailureTracker, e: MatrixEvent, nowTs: number): void {
|
|
|
|
// @ts-ignore access to private member
|
|
|
|
return tracker.eventDecrypted(e, nowTs);
|
|
|
|
}
|
|
|
|
|
2018-07-05 13:55:29 +00:00
|
|
|
describe("DecryptionFailureTracker", function () {
|
2024-03-22 15:48:29 +00:00
|
|
|
it("tracks a failed decryption for a visible event", async function () {
|
|
|
|
const failedDecryptionEvent = await createFailedDecryptionEvent();
|
2018-06-28 14:03:47 +00:00
|
|
|
|
|
|
|
let count = 0;
|
2024-03-25 19:57:29 +00:00
|
|
|
// @ts-ignore access to private constructor
|
2021-12-06 10:43:42 +00:00
|
|
|
const tracker = new DecryptionFailureTracker(
|
2024-05-16 00:25:58 +00:00
|
|
|
() => count++,
|
2021-12-06 10:43:42 +00:00
|
|
|
() => "UnknownError",
|
|
|
|
);
|
2018-06-15 12:33:07 +00:00
|
|
|
|
2022-01-19 19:31:43 +00:00
|
|
|
tracker.addVisibleEvent(failedDecryptionEvent);
|
2024-05-16 00:25:58 +00:00
|
|
|
eventDecrypted(tracker, failedDecryptionEvent, Date.now());
|
2018-06-15 12:33:07 +00:00
|
|
|
|
|
|
|
// Pretend "now" is Infinity
|
|
|
|
tracker.checkFailures(Infinity);
|
|
|
|
|
2018-07-05 12:54:44 +00:00
|
|
|
// Immediately track the newest failures
|
|
|
|
tracker.trackFailures();
|
2018-06-15 12:33:07 +00:00
|
|
|
|
2024-03-25 19:57:29 +00:00
|
|
|
// should track a failure for an event that failed decryption
|
|
|
|
expect(count).not.toBe(0);
|
2018-06-15 12:33:07 +00:00
|
|
|
});
|
|
|
|
|
2024-03-22 15:48:29 +00:00
|
|
|
it("tracks a failed decryption with expected raw error for a visible event", async function () {
|
2024-05-16 00:25:58 +00:00
|
|
|
const failedDecryptionEvent = await createFailedDecryptionEvent({
|
|
|
|
code: DecryptionFailureCode.OLM_UNKNOWN_MESSAGE_INDEX,
|
|
|
|
});
|
2022-05-05 03:46:03 +00:00
|
|
|
|
|
|
|
let count = 0;
|
|
|
|
let reportedRawCode = "";
|
2024-03-25 19:57:29 +00:00
|
|
|
// @ts-ignore access to private constructor
|
2022-05-05 03:46:03 +00:00
|
|
|
const tracker = new DecryptionFailureTracker(
|
2024-05-16 00:25:58 +00:00
|
|
|
(_errCode: string, rawCode: string) => {
|
|
|
|
count++;
|
2022-05-05 03:46:03 +00:00
|
|
|
reportedRawCode = rawCode;
|
|
|
|
},
|
|
|
|
() => "UnknownError",
|
|
|
|
);
|
|
|
|
|
|
|
|
tracker.addVisibleEvent(failedDecryptionEvent);
|
2024-05-16 00:25:58 +00:00
|
|
|
eventDecrypted(tracker, failedDecryptionEvent, Date.now());
|
2022-05-05 03:46:03 +00:00
|
|
|
|
|
|
|
// Pretend "now" is Infinity
|
|
|
|
tracker.checkFailures(Infinity);
|
|
|
|
|
|
|
|
// Immediately track the newest failures
|
|
|
|
tracker.trackFailures();
|
|
|
|
|
2024-03-25 19:57:29 +00:00
|
|
|
// should track a failure for an event that failed decryption
|
|
|
|
expect(count).not.toBe(0);
|
|
|
|
|
|
|
|
// Should add the rawCode to the event context
|
2024-04-17 12:36:01 +00:00
|
|
|
expect(reportedRawCode).toBe("OLM_UNKNOWN_MESSAGE_INDEX");
|
2022-05-05 03:46:03 +00:00
|
|
|
});
|
|
|
|
|
2024-03-22 15:48:29 +00:00
|
|
|
it("tracks a failed decryption for an event that becomes visible later", async function () {
|
|
|
|
const failedDecryptionEvent = await createFailedDecryptionEvent();
|
2022-01-19 19:31:43 +00:00
|
|
|
|
|
|
|
let count = 0;
|
2024-03-25 19:57:29 +00:00
|
|
|
// @ts-ignore access to private constructor
|
2022-01-19 19:31:43 +00:00
|
|
|
const tracker = new DecryptionFailureTracker(
|
2024-05-16 00:25:58 +00:00
|
|
|
() => count++,
|
2022-01-19 19:31:43 +00:00
|
|
|
() => "UnknownError",
|
|
|
|
);
|
|
|
|
|
2024-05-16 00:25:58 +00:00
|
|
|
eventDecrypted(tracker, failedDecryptionEvent, Date.now());
|
2022-01-19 19:31:43 +00:00
|
|
|
tracker.addVisibleEvent(failedDecryptionEvent);
|
|
|
|
|
|
|
|
// Pretend "now" is Infinity
|
|
|
|
tracker.checkFailures(Infinity);
|
|
|
|
|
|
|
|
// Immediately track the newest failures
|
|
|
|
tracker.trackFailures();
|
|
|
|
|
2024-03-25 19:57:29 +00:00
|
|
|
// should track a failure for an event that failed decryption
|
|
|
|
expect(count).not.toBe(0);
|
2022-01-19 19:31:43 +00:00
|
|
|
});
|
|
|
|
|
2024-05-16 00:25:58 +00:00
|
|
|
it("tracks visible vs. not visible events", async () => {
|
|
|
|
const propertiesByErrorCode: Record<string, ErrorProperties> = {};
|
2024-03-25 19:57:29 +00:00
|
|
|
// @ts-ignore access to private constructor
|
2022-01-19 19:31:43 +00:00
|
|
|
const tracker = new DecryptionFailureTracker(
|
2024-05-16 00:25:58 +00:00
|
|
|
(errorCode: string, rawError: string, properties: ErrorProperties) => {
|
|
|
|
propertiesByErrorCode[errorCode] = properties;
|
|
|
|
},
|
|
|
|
(error: string) => error,
|
2022-01-19 19:31:43 +00:00
|
|
|
);
|
|
|
|
|
2024-05-16 00:25:58 +00:00
|
|
|
// use three different errors so that we can distinguish the reports
|
|
|
|
const error1 = DecryptionFailureCode.MEGOLM_UNKNOWN_INBOUND_SESSION_ID;
|
|
|
|
const error2 = DecryptionFailureCode.MEGOLM_BAD_ROOM;
|
|
|
|
const error3 = DecryptionFailureCode.MEGOLM_MISSING_FIELDS;
|
|
|
|
|
|
|
|
// event that will be marked as visible before it's marked as undecryptable
|
|
|
|
const markedVisibleFirst = await createFailedDecryptionEvent({ code: error1 });
|
|
|
|
// event that will be marked as undecryptable before it's marked as visible
|
|
|
|
const markedUndecryptableFirst = await createFailedDecryptionEvent({ code: error2 });
|
|
|
|
// event that is never marked as visible
|
|
|
|
const neverVisible = await createFailedDecryptionEvent({ code: error3 });
|
|
|
|
|
|
|
|
tracker.addVisibleEvent(markedVisibleFirst);
|
|
|
|
|
|
|
|
const now = Date.now();
|
|
|
|
eventDecrypted(tracker, markedVisibleFirst, now);
|
|
|
|
eventDecrypted(tracker, markedUndecryptableFirst, now);
|
|
|
|
eventDecrypted(tracker, neverVisible, now);
|
|
|
|
|
|
|
|
tracker.addVisibleEvent(markedUndecryptableFirst);
|
2022-01-19 19:31:43 +00:00
|
|
|
|
|
|
|
// Pretend "now" is Infinity
|
|
|
|
tracker.checkFailures(Infinity);
|
|
|
|
|
|
|
|
tracker.trackFailures();
|
|
|
|
|
2024-05-16 00:25:58 +00:00
|
|
|
expect(propertiesByErrorCode[error1].wasVisibleToUser).toBe(true);
|
|
|
|
expect(propertiesByErrorCode[error2].wasVisibleToUser).toBe(true);
|
|
|
|
expect(propertiesByErrorCode[error3].wasVisibleToUser).toBe(false);
|
2022-01-19 19:31:43 +00:00
|
|
|
});
|
|
|
|
|
2024-03-22 15:48:29 +00:00
|
|
|
it("does not track a failed decryption where the event is subsequently successfully decrypted", async () => {
|
|
|
|
const decryptedEvent = await createFailedDecryptionEvent();
|
2024-03-25 19:57:29 +00:00
|
|
|
// @ts-ignore access to private constructor
|
2018-06-28 14:03:47 +00:00
|
|
|
const tracker = new DecryptionFailureTracker(
|
2024-05-16 00:25:58 +00:00
|
|
|
() => {
|
2024-03-25 19:57:29 +00:00
|
|
|
// should not track an event that has since been decrypted correctly
|
|
|
|
expect(true).toBe(false);
|
2021-12-06 10:43:42 +00:00
|
|
|
},
|
|
|
|
() => "UnknownError",
|
|
|
|
);
|
2018-06-15 12:33:07 +00:00
|
|
|
|
2022-01-19 19:31:43 +00:00
|
|
|
tracker.addVisibleEvent(decryptedEvent);
|
2024-05-16 00:25:58 +00:00
|
|
|
eventDecrypted(tracker, decryptedEvent, Date.now());
|
2018-06-15 12:33:07 +00:00
|
|
|
|
2024-03-25 19:57:29 +00:00
|
|
|
// Indicate successful decryption.
|
|
|
|
await decryptExistingEvent(decryptedEvent, {
|
|
|
|
plainType: "m.room.message",
|
|
|
|
plainContent: { body: "success" },
|
|
|
|
});
|
2024-05-16 00:25:58 +00:00
|
|
|
eventDecrypted(tracker, decryptedEvent, Date.now());
|
2018-06-15 12:33:07 +00:00
|
|
|
|
|
|
|
// Pretend "now" is Infinity
|
|
|
|
tracker.checkFailures(Infinity);
|
|
|
|
|
2018-07-05 12:54:44 +00:00
|
|
|
// Immediately track the newest failures
|
|
|
|
tracker.trackFailures();
|
2018-06-15 12:33:07 +00:00
|
|
|
});
|
|
|
|
|
2022-01-19 19:31:43 +00:00
|
|
|
it(
|
|
|
|
"does not track a failed decryption where the event is subsequently successfully decrypted " +
|
|
|
|
"and later becomes visible",
|
2024-03-22 15:48:29 +00:00
|
|
|
async () => {
|
|
|
|
const decryptedEvent = await createFailedDecryptionEvent();
|
2024-03-25 19:57:29 +00:00
|
|
|
// @ts-ignore access to private constructor
|
2022-01-19 19:31:43 +00:00
|
|
|
const tracker = new DecryptionFailureTracker(
|
2024-05-16 00:25:58 +00:00
|
|
|
() => {
|
2024-03-25 19:57:29 +00:00
|
|
|
// should not track an event that has since been decrypted correctly
|
|
|
|
expect(true).toBe(false);
|
2022-01-19 19:31:43 +00:00
|
|
|
},
|
|
|
|
() => "UnknownError",
|
|
|
|
);
|
2022-12-12 11:24:14 +00:00
|
|
|
|
2024-05-16 00:25:58 +00:00
|
|
|
eventDecrypted(tracker, decryptedEvent, Date.now());
|
2022-12-12 11:24:14 +00:00
|
|
|
|
2024-03-25 19:57:29 +00:00
|
|
|
// Indicate successful decryption.
|
|
|
|
await decryptExistingEvent(decryptedEvent, {
|
|
|
|
plainType: "m.room.message",
|
|
|
|
plainContent: { body: "success" },
|
|
|
|
});
|
2024-05-16 00:25:58 +00:00
|
|
|
eventDecrypted(tracker, decryptedEvent, Date.now());
|
2022-12-12 11:24:14 +00:00
|
|
|
|
2022-01-19 19:31:43 +00:00
|
|
|
tracker.addVisibleEvent(decryptedEvent);
|
2022-12-12 11:24:14 +00:00
|
|
|
|
2022-01-19 19:31:43 +00:00
|
|
|
// Pretend "now" is Infinity
|
|
|
|
tracker.checkFailures(Infinity);
|
2022-12-12 11:24:14 +00:00
|
|
|
|
2022-01-19 19:31:43 +00:00
|
|
|
// Immediately track the newest failures
|
|
|
|
tracker.trackFailures();
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
2024-03-22 15:48:29 +00:00
|
|
|
it("only tracks a single failure per event, despite multiple failed decryptions for multiple events", async () => {
|
|
|
|
const decryptedEvent = await createFailedDecryptionEvent();
|
|
|
|
const decryptedEvent2 = await createFailedDecryptionEvent();
|
2018-06-15 12:33:07 +00:00
|
|
|
|
|
|
|
let count = 0;
|
2024-03-25 19:57:29 +00:00
|
|
|
// @ts-ignore access to private constructor
|
2021-12-06 10:43:42 +00:00
|
|
|
const tracker = new DecryptionFailureTracker(
|
2024-05-16 00:25:58 +00:00
|
|
|
() => count++,
|
2021-12-06 10:43:42 +00:00
|
|
|
() => "UnknownError",
|
|
|
|
);
|
2018-06-15 12:33:07 +00:00
|
|
|
|
2022-01-19 19:31:43 +00:00
|
|
|
tracker.addVisibleEvent(decryptedEvent);
|
|
|
|
|
2018-06-15 12:33:07 +00:00
|
|
|
// Arbitrary number of failed decryptions for both events
|
2024-05-16 00:25:58 +00:00
|
|
|
const now = Date.now();
|
|
|
|
eventDecrypted(tracker, decryptedEvent, now);
|
|
|
|
eventDecrypted(tracker, decryptedEvent, now);
|
|
|
|
eventDecrypted(tracker, decryptedEvent, now);
|
|
|
|
eventDecrypted(tracker, decryptedEvent, now);
|
|
|
|
eventDecrypted(tracker, decryptedEvent, now);
|
|
|
|
eventDecrypted(tracker, decryptedEvent2, now);
|
|
|
|
eventDecrypted(tracker, decryptedEvent2, now);
|
2022-01-19 19:31:43 +00:00
|
|
|
tracker.addVisibleEvent(decryptedEvent2);
|
2024-05-16 00:25:58 +00:00
|
|
|
eventDecrypted(tracker, decryptedEvent2, now);
|
2018-06-15 12:33:07 +00:00
|
|
|
|
|
|
|
// Pretend "now" is Infinity
|
|
|
|
tracker.checkFailures(Infinity);
|
|
|
|
|
2018-07-05 12:54:44 +00:00
|
|
|
// Simulated polling of `trackFailures`, an arbitrary number ( > 2 ) times
|
|
|
|
tracker.trackFailures();
|
|
|
|
tracker.trackFailures();
|
|
|
|
tracker.trackFailures();
|
|
|
|
tracker.trackFailures();
|
2018-06-15 12:33:07 +00:00
|
|
|
|
2024-03-25 19:57:29 +00:00
|
|
|
// should only track a single failure per event
|
|
|
|
expect(count).toBe(2);
|
2018-06-15 12:33:07 +00:00
|
|
|
});
|
|
|
|
|
2024-03-22 15:48:29 +00:00
|
|
|
it("should not track a failure for an event that was tracked previously", async () => {
|
|
|
|
const decryptedEvent = await createFailedDecryptionEvent();
|
2018-06-15 13:48:20 +00:00
|
|
|
|
2018-06-28 14:03:47 +00:00
|
|
|
let count = 0;
|
2024-03-25 19:57:29 +00:00
|
|
|
// @ts-ignore access to private constructor
|
2021-12-06 10:43:42 +00:00
|
|
|
const tracker = new DecryptionFailureTracker(
|
2024-05-16 00:25:58 +00:00
|
|
|
() => count++,
|
2021-12-06 10:43:42 +00:00
|
|
|
() => "UnknownError",
|
|
|
|
);
|
2018-06-15 13:48:20 +00:00
|
|
|
|
2022-01-19 19:31:43 +00:00
|
|
|
tracker.addVisibleEvent(decryptedEvent);
|
|
|
|
|
2018-06-15 13:48:20 +00:00
|
|
|
// Indicate decryption
|
2024-05-16 00:25:58 +00:00
|
|
|
eventDecrypted(tracker, decryptedEvent, Date.now());
|
2018-06-15 13:48:20 +00:00
|
|
|
|
|
|
|
// Pretend "now" is Infinity
|
|
|
|
tracker.checkFailures(Infinity);
|
|
|
|
|
2018-07-05 12:54:44 +00:00
|
|
|
tracker.trackFailures();
|
2018-06-15 13:48:20 +00:00
|
|
|
|
|
|
|
// Indicate a second decryption, after having tracked the failure
|
2024-05-16 00:25:58 +00:00
|
|
|
eventDecrypted(tracker, decryptedEvent, Date.now());
|
2018-06-15 13:48:20 +00:00
|
|
|
|
2018-07-05 12:54:44 +00:00
|
|
|
tracker.trackFailures();
|
2018-06-15 13:48:20 +00:00
|
|
|
|
2024-03-25 19:57:29 +00:00
|
|
|
// should only track a single failure per event
|
|
|
|
expect(count).toBe(1);
|
2018-06-15 13:48:20 +00:00
|
|
|
});
|
2018-06-15 14:26:53 +00:00
|
|
|
|
2024-03-22 15:48:29 +00:00
|
|
|
it.skip("should not track a failure for an event that was tracked in a previous session", async () => {
|
2018-06-15 14:26:53 +00:00
|
|
|
// This test uses localStorage, clear it beforehand
|
|
|
|
localStorage.clear();
|
|
|
|
|
2024-03-22 15:48:29 +00:00
|
|
|
const decryptedEvent = await createFailedDecryptionEvent();
|
2018-06-15 14:26:53 +00:00
|
|
|
|
2018-06-28 14:03:47 +00:00
|
|
|
let count = 0;
|
2024-03-25 19:57:29 +00:00
|
|
|
// @ts-ignore access to private constructor
|
2021-12-06 10:43:42 +00:00
|
|
|
const tracker = new DecryptionFailureTracker(
|
2024-05-16 00:25:58 +00:00
|
|
|
() => count++,
|
2021-12-06 10:43:42 +00:00
|
|
|
() => "UnknownError",
|
|
|
|
);
|
2018-06-15 14:26:53 +00:00
|
|
|
|
2022-01-19 19:31:43 +00:00
|
|
|
tracker.addVisibleEvent(decryptedEvent);
|
|
|
|
|
2018-06-15 14:26:53 +00:00
|
|
|
// Indicate decryption
|
2024-05-16 00:25:58 +00:00
|
|
|
eventDecrypted(tracker, decryptedEvent, Date.now());
|
2018-06-15 14:26:53 +00:00
|
|
|
|
|
|
|
// Pretend "now" is Infinity
|
|
|
|
// NB: This saves to localStorage specific to DFT
|
|
|
|
tracker.checkFailures(Infinity);
|
|
|
|
|
2018-07-05 12:54:44 +00:00
|
|
|
tracker.trackFailures();
|
2018-06-15 14:26:53 +00:00
|
|
|
|
|
|
|
// Simulate the browser refreshing by destroying tracker and creating a new tracker
|
2024-03-25 19:57:29 +00:00
|
|
|
// @ts-ignore access to private constructor
|
2021-12-06 10:43:42 +00:00
|
|
|
const secondTracker = new DecryptionFailureTracker(
|
2024-03-25 19:57:29 +00:00
|
|
|
(total: number) => (count += total),
|
2021-12-06 10:43:42 +00:00
|
|
|
() => "UnknownError",
|
|
|
|
);
|
2018-06-15 16:08:11 +00:00
|
|
|
|
2022-01-19 19:31:43 +00:00
|
|
|
secondTracker.addVisibleEvent(decryptedEvent);
|
|
|
|
|
|
|
|
//secondTracker.loadTrackedEvents();
|
2018-06-15 16:08:11 +00:00
|
|
|
|
2024-05-16 00:25:58 +00:00
|
|
|
eventDecrypted(secondTracker, decryptedEvent, Date.now());
|
2018-06-15 14:26:53 +00:00
|
|
|
secondTracker.checkFailures(Infinity);
|
2018-07-05 12:54:44 +00:00
|
|
|
secondTracker.trackFailures();
|
2018-06-15 14:26:53 +00:00
|
|
|
|
2024-03-25 19:57:29 +00:00
|
|
|
// should only track a single failure per event
|
|
|
|
expect(count).toBe(1);
|
2018-06-15 14:26:53 +00:00
|
|
|
});
|
2018-07-05 12:54:44 +00:00
|
|
|
|
2024-03-22 15:48:29 +00:00
|
|
|
it("should count different error codes separately for multiple failures with different error codes", async () => {
|
2024-03-25 19:57:29 +00:00
|
|
|
const counts: Record<string, number> = {};
|
|
|
|
|
|
|
|
// @ts-ignore access to private constructor
|
2018-07-05 12:54:44 +00:00
|
|
|
const tracker = new DecryptionFailureTracker(
|
2024-05-16 00:25:58 +00:00
|
|
|
(errorCode: string) => (counts[errorCode] = (counts[errorCode] || 0) + 1),
|
2024-04-17 12:36:01 +00:00
|
|
|
(error: DecryptionFailureCode) =>
|
|
|
|
error === DecryptionFailureCode.UNKNOWN_ERROR ? "UnknownError" : "OlmKeysNotSentError",
|
2018-07-05 12:54:44 +00:00
|
|
|
);
|
|
|
|
|
2024-05-16 00:25:58 +00:00
|
|
|
const decryptedEvent1 = await createFailedDecryptionEvent({
|
|
|
|
code: DecryptionFailureCode.UNKNOWN_ERROR,
|
|
|
|
});
|
|
|
|
const decryptedEvent2 = await createFailedDecryptionEvent({
|
|
|
|
code: DecryptionFailureCode.MEGOLM_UNKNOWN_INBOUND_SESSION_ID,
|
|
|
|
});
|
|
|
|
const decryptedEvent3 = await createFailedDecryptionEvent({
|
|
|
|
code: DecryptionFailureCode.MEGOLM_UNKNOWN_INBOUND_SESSION_ID,
|
|
|
|
});
|
2022-01-19 19:31:43 +00:00
|
|
|
|
|
|
|
tracker.addVisibleEvent(decryptedEvent1);
|
|
|
|
tracker.addVisibleEvent(decryptedEvent2);
|
|
|
|
tracker.addVisibleEvent(decryptedEvent3);
|
|
|
|
|
2024-05-16 00:25:58 +00:00
|
|
|
// One failure of UNKNOWN_ERROR, and effectively two for MEGOLM_UNKNOWN_INBOUND_SESSION_ID
|
|
|
|
const now = Date.now();
|
|
|
|
eventDecrypted(tracker, decryptedEvent1, now);
|
|
|
|
eventDecrypted(tracker, decryptedEvent2, now);
|
|
|
|
eventDecrypted(tracker, decryptedEvent2, now);
|
|
|
|
eventDecrypted(tracker, decryptedEvent3, now);
|
2018-07-05 12:54:44 +00:00
|
|
|
|
|
|
|
// Pretend "now" is Infinity
|
|
|
|
tracker.checkFailures(Infinity);
|
|
|
|
|
|
|
|
tracker.trackFailures();
|
|
|
|
|
2022-01-19 19:31:43 +00:00
|
|
|
//expect(counts['UnknownError']).toBe(1, 'should track one UnknownError');
|
2024-03-25 19:57:29 +00:00
|
|
|
expect(counts["OlmKeysNotSentError"]).toBe(2);
|
2018-07-05 12:54:44 +00:00
|
|
|
});
|
|
|
|
|
2024-03-22 15:48:29 +00:00
|
|
|
it("should aggregate error codes correctly", async () => {
|
2024-03-25 19:57:29 +00:00
|
|
|
const counts: Record<string, number> = {};
|
|
|
|
|
|
|
|
// @ts-ignore access to private constructor
|
2018-07-05 12:54:44 +00:00
|
|
|
const tracker = new DecryptionFailureTracker(
|
2024-05-16 00:25:58 +00:00
|
|
|
(errorCode: string) => (counts[errorCode] = (counts[errorCode] || 0) + 1),
|
2024-03-25 19:57:29 +00:00
|
|
|
(_errorCode: string) => "OlmUnspecifiedError",
|
2018-07-05 12:54:44 +00:00
|
|
|
);
|
|
|
|
|
2024-05-16 00:25:58 +00:00
|
|
|
const decryptedEvent1 = await createFailedDecryptionEvent({
|
|
|
|
code: DecryptionFailureCode.MEGOLM_UNKNOWN_INBOUND_SESSION_ID,
|
|
|
|
});
|
|
|
|
const decryptedEvent2 = await createFailedDecryptionEvent({
|
|
|
|
code: DecryptionFailureCode.OLM_UNKNOWN_MESSAGE_INDEX,
|
|
|
|
});
|
|
|
|
const decryptedEvent3 = await createFailedDecryptionEvent({
|
|
|
|
code: DecryptionFailureCode.UNKNOWN_ERROR,
|
|
|
|
});
|
2022-01-19 19:31:43 +00:00
|
|
|
|
|
|
|
tracker.addVisibleEvent(decryptedEvent1);
|
|
|
|
tracker.addVisibleEvent(decryptedEvent2);
|
|
|
|
tracker.addVisibleEvent(decryptedEvent3);
|
|
|
|
|
2024-05-16 00:25:58 +00:00
|
|
|
const now = Date.now();
|
|
|
|
eventDecrypted(tracker, decryptedEvent1, now);
|
|
|
|
eventDecrypted(tracker, decryptedEvent2, now);
|
|
|
|
eventDecrypted(tracker, decryptedEvent3, now);
|
2018-07-05 12:54:44 +00:00
|
|
|
|
|
|
|
// Pretend "now" is Infinity
|
|
|
|
tracker.checkFailures(Infinity);
|
|
|
|
|
|
|
|
tracker.trackFailures();
|
|
|
|
|
2024-03-25 19:57:29 +00:00
|
|
|
expect(counts["OlmUnspecifiedError"]).toBe(3);
|
2018-07-05 12:54:44 +00:00
|
|
|
});
|
2022-01-19 19:31:43 +00:00
|
|
|
|
2024-03-22 15:48:29 +00:00
|
|
|
it("should remap error codes correctly", async () => {
|
2024-03-25 19:57:29 +00:00
|
|
|
const counts: Record<string, number> = {};
|
|
|
|
|
|
|
|
// @ts-ignore access to private constructor
|
2022-01-19 19:31:43 +00:00
|
|
|
const tracker = new DecryptionFailureTracker(
|
2024-05-16 00:25:58 +00:00
|
|
|
(errorCode: string) => (counts[errorCode] = (counts[errorCode] || 0) + 1),
|
2024-03-25 19:57:29 +00:00
|
|
|
(errorCode: string) => Array.from(errorCode).reverse().join(""),
|
2022-01-19 19:31:43 +00:00
|
|
|
);
|
|
|
|
|
2024-05-16 00:25:58 +00:00
|
|
|
const decryptedEvent = await createFailedDecryptionEvent({
|
|
|
|
code: DecryptionFailureCode.OLM_UNKNOWN_MESSAGE_INDEX,
|
|
|
|
});
|
2022-01-19 19:31:43 +00:00
|
|
|
tracker.addVisibleEvent(decryptedEvent);
|
2024-05-16 00:25:58 +00:00
|
|
|
eventDecrypted(tracker, decryptedEvent, Date.now());
|
2022-01-19 19:31:43 +00:00
|
|
|
|
|
|
|
// Pretend "now" is Infinity
|
|
|
|
tracker.checkFailures(Infinity);
|
|
|
|
|
|
|
|
tracker.trackFailures();
|
|
|
|
|
2024-03-25 19:57:29 +00:00
|
|
|
// should track remapped error code
|
2024-04-17 12:36:01 +00:00
|
|
|
expect(counts["XEDNI_EGASSEM_NWONKNU_MLO"]).toBe(1);
|
2022-01-19 19:31:43 +00:00
|
|
|
});
|
2024-04-29 17:18:57 +00:00
|
|
|
|
|
|
|
it("default error code mapper maps error codes correctly", async () => {
|
|
|
|
const errorCodes: string[] = [];
|
|
|
|
|
|
|
|
// @ts-ignore access to private constructor
|
|
|
|
const tracker = new DecryptionFailureTracker(
|
2024-05-16 00:25:58 +00:00
|
|
|
(errorCode: string) => {
|
2024-04-29 17:18:57 +00:00
|
|
|
errorCodes.push(errorCode);
|
|
|
|
},
|
|
|
|
// @ts-ignore access to private member
|
|
|
|
DecryptionFailureTracker.instance.errorCodeMapFn,
|
|
|
|
);
|
|
|
|
|
2024-05-16 00:25:58 +00:00
|
|
|
const now = Date.now();
|
|
|
|
|
|
|
|
const event1 = await createFailedDecryptionEvent({
|
|
|
|
code: DecryptionFailureCode.MEGOLM_UNKNOWN_INBOUND_SESSION_ID,
|
|
|
|
});
|
2024-04-29 17:18:57 +00:00
|
|
|
tracker.addVisibleEvent(event1);
|
2024-05-16 00:25:58 +00:00
|
|
|
eventDecrypted(tracker, event1, now);
|
2024-04-29 17:18:57 +00:00
|
|
|
|
2024-05-16 00:25:58 +00:00
|
|
|
const event2 = await createFailedDecryptionEvent({
|
|
|
|
code: DecryptionFailureCode.OLM_UNKNOWN_MESSAGE_INDEX,
|
|
|
|
});
|
2024-04-29 17:18:57 +00:00
|
|
|
tracker.addVisibleEvent(event2);
|
2024-05-16 00:25:58 +00:00
|
|
|
eventDecrypted(tracker, event2, now);
|
2024-04-29 17:18:57 +00:00
|
|
|
|
2024-05-16 00:25:58 +00:00
|
|
|
const event3 = await createFailedDecryptionEvent({
|
|
|
|
code: DecryptionFailureCode.HISTORICAL_MESSAGE_NO_KEY_BACKUP,
|
|
|
|
});
|
2024-04-29 17:18:57 +00:00
|
|
|
tracker.addVisibleEvent(event3);
|
2024-05-16 00:25:58 +00:00
|
|
|
eventDecrypted(tracker, event3, now);
|
2024-04-29 17:18:57 +00:00
|
|
|
|
2024-05-16 00:25:58 +00:00
|
|
|
const event4 = await createFailedDecryptionEvent({
|
|
|
|
code: DecryptionFailureCode.HISTORICAL_MESSAGE_BACKUP_UNCONFIGURED,
|
|
|
|
});
|
2024-04-29 17:18:57 +00:00
|
|
|
tracker.addVisibleEvent(event4);
|
2024-05-16 00:25:58 +00:00
|
|
|
eventDecrypted(tracker, event4, now);
|
2024-04-29 17:18:57 +00:00
|
|
|
|
2024-05-16 00:25:58 +00:00
|
|
|
const event5 = await createFailedDecryptionEvent({
|
|
|
|
code: DecryptionFailureCode.HISTORICAL_MESSAGE_WORKING_BACKUP,
|
|
|
|
});
|
2024-04-29 17:18:57 +00:00
|
|
|
tracker.addVisibleEvent(event5);
|
2024-05-16 00:25:58 +00:00
|
|
|
eventDecrypted(tracker, event5, now);
|
2024-04-29 17:18:57 +00:00
|
|
|
|
2024-05-16 00:25:58 +00:00
|
|
|
const event6 = await createFailedDecryptionEvent({
|
|
|
|
code: DecryptionFailureCode.HISTORICAL_MESSAGE_USER_NOT_JOINED,
|
|
|
|
});
|
2024-04-29 17:18:57 +00:00
|
|
|
tracker.addVisibleEvent(event6);
|
2024-05-16 00:25:58 +00:00
|
|
|
eventDecrypted(tracker, event6, now);
|
2024-04-29 17:18:57 +00:00
|
|
|
|
2024-05-16 00:25:58 +00:00
|
|
|
const event7 = await createFailedDecryptionEvent({
|
|
|
|
code: DecryptionFailureCode.UNKNOWN_ERROR,
|
|
|
|
});
|
2024-04-29 17:18:57 +00:00
|
|
|
tracker.addVisibleEvent(event7);
|
2024-05-16 00:25:58 +00:00
|
|
|
eventDecrypted(tracker, event7, now);
|
2024-04-29 17:18:57 +00:00
|
|
|
|
|
|
|
// Pretend "now" is Infinity
|
|
|
|
tracker.checkFailures(Infinity);
|
|
|
|
|
|
|
|
tracker.trackFailures();
|
|
|
|
|
|
|
|
expect(errorCodes).toEqual([
|
|
|
|
"OlmKeysNotSentError",
|
|
|
|
"OlmIndexError",
|
|
|
|
"HistoricalMessage",
|
|
|
|
"HistoricalMessage",
|
|
|
|
"HistoricalMessage",
|
|
|
|
"ExpectedDueToMembership",
|
|
|
|
"UnknownError",
|
|
|
|
]);
|
|
|
|
});
|
2024-05-16 00:25:58 +00:00
|
|
|
|
|
|
|
it("tracks late decryptions vs. undecryptable", async () => {
|
|
|
|
const propertiesByErrorCode: Record<string, ErrorProperties> = {};
|
|
|
|
// @ts-ignore access to private constructor
|
|
|
|
const tracker = new DecryptionFailureTracker(
|
|
|
|
(errorCode: string, rawError: string, properties: ErrorProperties) => {
|
|
|
|
propertiesByErrorCode[errorCode] = properties;
|
|
|
|
},
|
|
|
|
(error: string) => error,
|
|
|
|
);
|
|
|
|
|
|
|
|
// use three different errors so that we can distinguish the reports
|
|
|
|
const error1 = DecryptionFailureCode.MEGOLM_UNKNOWN_INBOUND_SESSION_ID;
|
|
|
|
const error2 = DecryptionFailureCode.MEGOLM_BAD_ROOM;
|
|
|
|
const error3 = DecryptionFailureCode.MEGOLM_MISSING_FIELDS;
|
|
|
|
|
|
|
|
// event that will be slow to decrypt
|
|
|
|
const lateDecryption = await createFailedDecryptionEvent({ code: error1 });
|
|
|
|
// event that will be so slow to decrypt, it gets counted as undecryptable
|
|
|
|
const veryLateDecryption = await createFailedDecryptionEvent({ code: error2 });
|
|
|
|
// event that never gets decrypted
|
|
|
|
const neverDecrypted = await createFailedDecryptionEvent({ code: error3 });
|
|
|
|
|
|
|
|
tracker.addVisibleEvent(lateDecryption);
|
|
|
|
tracker.addVisibleEvent(veryLateDecryption);
|
|
|
|
tracker.addVisibleEvent(neverDecrypted);
|
|
|
|
|
|
|
|
const now = Date.now();
|
|
|
|
eventDecrypted(tracker, lateDecryption, now);
|
|
|
|
eventDecrypted(tracker, veryLateDecryption, now);
|
|
|
|
eventDecrypted(tracker, neverDecrypted, now);
|
|
|
|
|
|
|
|
await decryptExistingEvent(lateDecryption, {
|
|
|
|
plainType: "m.room.message",
|
|
|
|
plainContent: { body: "success" },
|
|
|
|
});
|
|
|
|
await decryptExistingEvent(veryLateDecryption, {
|
|
|
|
plainType: "m.room.message",
|
|
|
|
plainContent: { body: "success" },
|
|
|
|
});
|
|
|
|
eventDecrypted(tracker, lateDecryption, now + 40000);
|
|
|
|
eventDecrypted(tracker, veryLateDecryption, now + 100000);
|
|
|
|
|
|
|
|
// Pretend "now" is Infinity
|
|
|
|
tracker.checkFailures(Infinity);
|
|
|
|
|
|
|
|
tracker.trackFailures();
|
|
|
|
|
|
|
|
expect(propertiesByErrorCode[error1].timeToDecryptMillis).toEqual(40000);
|
|
|
|
expect(propertiesByErrorCode[error2].timeToDecryptMillis).toEqual(-1);
|
|
|
|
expect(propertiesByErrorCode[error3].timeToDecryptMillis).toEqual(-1);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("listens for client events", async () => {
|
|
|
|
// Test that the decryption failure tracker registers the right event
|
|
|
|
// handlers on start, and unregisters them when the client logs out.
|
|
|
|
const client = mocked(stubClient());
|
|
|
|
const mockCrypto = {
|
|
|
|
getVersion: jest.fn().mockReturnValue("Rust SDK 0.7.0 (61b175b), Vodozemac 0.5.1"),
|
|
|
|
getUserVerificationStatus: jest.fn().mockResolvedValue(new UserVerificationStatus(false, false, false)),
|
|
|
|
} as unknown as Mocked<CryptoApi>;
|
|
|
|
client.getCrypto.mockReturnValue(mockCrypto);
|
|
|
|
|
|
|
|
let errorCount: number = 0;
|
|
|
|
// @ts-ignore access to private constructor
|
|
|
|
const tracker = new DecryptionFailureTracker(
|
|
|
|
(errorCode: string, rawError: string, properties: ErrorProperties) => {
|
|
|
|
errorCount++;
|
|
|
|
},
|
|
|
|
(error: string) => error,
|
|
|
|
);
|
|
|
|
|
|
|
|
// Calling .start will start some intervals. This test shouldn't run
|
|
|
|
// long enough for the timers to fire, but we'll use fake timers just
|
|
|
|
// to be safe.
|
|
|
|
jest.useFakeTimers();
|
|
|
|
tracker.start(client);
|
|
|
|
|
|
|
|
// If the client fails to decrypt, it should get tracked
|
|
|
|
const failedDecryption = await createFailedDecryptionEvent();
|
|
|
|
client.emit(MatrixEventEvent.Decrypted, failedDecryption);
|
|
|
|
|
|
|
|
tracker.checkFailures(Infinity);
|
|
|
|
tracker.trackFailures();
|
|
|
|
|
|
|
|
expect(errorCount).toEqual(1);
|
|
|
|
|
|
|
|
client.emit(HttpApiEvent.SessionLoggedOut, {} as any);
|
|
|
|
|
|
|
|
// After the client has logged out, we shouldn't be listening to events
|
|
|
|
// any more, so even if the client emits an event regarding a failed
|
|
|
|
// decryption, we won't track it.
|
|
|
|
const anotherFailedDecryption = await createFailedDecryptionEvent();
|
|
|
|
client.emit(MatrixEventEvent.Decrypted, anotherFailedDecryption);
|
|
|
|
|
|
|
|
// Pretend "now" is Infinity
|
|
|
|
tracker.checkFailures(Infinity);
|
|
|
|
tracker.trackFailures();
|
|
|
|
|
|
|
|
expect(errorCount).toEqual(1);
|
|
|
|
|
|
|
|
jest.useRealTimers();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("tracks client information", async () => {
|
|
|
|
const client = mocked(stubClient());
|
|
|
|
const mockCrypto = {
|
|
|
|
getVersion: jest.fn().mockReturnValue("Rust SDK 0.7.0 (61b175b), Vodozemac 0.5.1"),
|
|
|
|
getUserVerificationStatus: jest.fn().mockResolvedValue(new UserVerificationStatus(false, false, false)),
|
|
|
|
} as unknown as Mocked<CryptoApi>;
|
|
|
|
client.getCrypto.mockReturnValue(mockCrypto);
|
|
|
|
|
|
|
|
const propertiesByErrorCode: Record<string, ErrorProperties> = {};
|
|
|
|
// @ts-ignore access to private constructor
|
|
|
|
const tracker = new DecryptionFailureTracker(
|
|
|
|
(errorCode: string, rawError: string, properties: ErrorProperties) => {
|
|
|
|
propertiesByErrorCode[errorCode] = properties;
|
|
|
|
},
|
|
|
|
(error: string) => error,
|
|
|
|
);
|
|
|
|
|
|
|
|
// @ts-ignore access to private method
|
|
|
|
await tracker.calculateClientProperties(client);
|
|
|
|
// @ts-ignore access to private method
|
|
|
|
await tracker.registerHandlers(client);
|
|
|
|
|
|
|
|
// use three different errors so that we can distinguish the reports
|
|
|
|
const error1 = DecryptionFailureCode.MEGOLM_UNKNOWN_INBOUND_SESSION_ID;
|
|
|
|
const error2 = DecryptionFailureCode.MEGOLM_BAD_ROOM;
|
|
|
|
const error3 = DecryptionFailureCode.MEGOLM_MISSING_FIELDS;
|
|
|
|
|
|
|
|
// event from a federated user (@alice:example.com)
|
|
|
|
const federatedDecryption = await createFailedDecryptionEvent({
|
|
|
|
code: error1,
|
|
|
|
});
|
|
|
|
// event from a local user
|
|
|
|
const localDecryption = await createFailedDecryptionEvent({
|
|
|
|
sender: "@bob:matrix.org",
|
|
|
|
code: error2,
|
|
|
|
});
|
|
|
|
|
|
|
|
tracker.addVisibleEvent(federatedDecryption);
|
|
|
|
tracker.addVisibleEvent(localDecryption);
|
|
|
|
|
|
|
|
const now = Date.now();
|
|
|
|
eventDecrypted(tracker, federatedDecryption, now);
|
|
|
|
|
|
|
|
mockCrypto.getUserVerificationStatus.mockResolvedValue(new UserVerificationStatus(true, true, false));
|
|
|
|
client.emit(CryptoEvent.KeysChanged, {});
|
|
|
|
await sleep(100);
|
|
|
|
eventDecrypted(tracker, localDecryption, now);
|
|
|
|
|
|
|
|
// Pretend "now" is Infinity
|
|
|
|
tracker.checkFailures(Infinity);
|
|
|
|
|
|
|
|
tracker.trackFailures();
|
|
|
|
|
|
|
|
expect(propertiesByErrorCode[error1].isMatrixDotOrg).toBe(true);
|
|
|
|
expect(propertiesByErrorCode[error1].cryptoSDK).toEqual("Rust");
|
|
|
|
|
|
|
|
expect(propertiesByErrorCode[error1].isFederated).toBe(true);
|
|
|
|
expect(propertiesByErrorCode[error1].userTrustsOwnIdentity).toEqual(false);
|
|
|
|
expect(propertiesByErrorCode[error2].isFederated).toBe(false);
|
|
|
|
expect(propertiesByErrorCode[error2].userTrustsOwnIdentity).toEqual(true);
|
|
|
|
|
|
|
|
// change client params, and make sure the reports the right values
|
|
|
|
client.getDomain.mockReturnValue("example.com");
|
|
|
|
mockCrypto.getVersion.mockReturnValue("Olm 0.0.0");
|
|
|
|
// @ts-ignore access to private method
|
|
|
|
await tracker.calculateClientProperties(client);
|
|
|
|
|
|
|
|
const anotherFailure = await createFailedDecryptionEvent({
|
|
|
|
code: error3,
|
|
|
|
});
|
|
|
|
tracker.addVisibleEvent(anotherFailure);
|
|
|
|
eventDecrypted(tracker, anotherFailure, now);
|
|
|
|
tracker.checkFailures(Infinity);
|
|
|
|
tracker.trackFailures();
|
|
|
|
expect(propertiesByErrorCode[error3].isMatrixDotOrg).toBe(false);
|
|
|
|
expect(propertiesByErrorCode[error3].cryptoSDK).toEqual("Legacy");
|
|
|
|
});
|
|
|
|
|
|
|
|
it("keeps the original timestamp after repeated decryption failures", async () => {
|
|
|
|
const failedDecryptionEvent = await createFailedDecryptionEvent();
|
|
|
|
|
|
|
|
let failure: ErrorProperties | undefined;
|
|
|
|
// @ts-ignore access to private constructor
|
|
|
|
const tracker = new DecryptionFailureTracker(
|
|
|
|
(errorCode: string, rawError: string, properties: ErrorProperties) => {
|
|
|
|
failure = properties;
|
|
|
|
},
|
|
|
|
() => "UnknownError",
|
|
|
|
);
|
|
|
|
|
|
|
|
tracker.addVisibleEvent(failedDecryptionEvent);
|
|
|
|
|
|
|
|
const now = Date.now();
|
|
|
|
eventDecrypted(tracker, failedDecryptionEvent, now);
|
|
|
|
eventDecrypted(tracker, failedDecryptionEvent, now + 20000);
|
|
|
|
await decryptExistingEvent(failedDecryptionEvent, {
|
|
|
|
plainType: "m.room.message",
|
|
|
|
plainContent: { body: "success" },
|
|
|
|
});
|
|
|
|
eventDecrypted(tracker, failedDecryptionEvent, now + 50000);
|
|
|
|
|
|
|
|
// Pretend "now" is Infinity
|
|
|
|
tracker.checkFailures(Infinity);
|
|
|
|
tracker.trackFailures();
|
|
|
|
|
|
|
|
// the time to decrypt should be relative to the first time we failed
|
|
|
|
// to decrypt, not the second
|
|
|
|
expect(failure?.timeToDecryptMillis).toEqual(50000);
|
|
|
|
});
|
2018-06-15 12:33:07 +00:00
|
|
|
});
|