172 lines
6.8 KiB
TypeScript
172 lines
6.8 KiB
TypeScript
/*
|
|
Copyright 2022 The Matrix.org Foundation C.I.C.
|
|
|
|
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.
|
|
*/
|
|
|
|
import { ClientEvent, MatrixEvent, MatrixEventEvent } from "matrix-js-sdk/src/matrix";
|
|
import { sleep } from "matrix-js-sdk/src/utils";
|
|
import { ISyncStateData, SyncState } from "matrix-js-sdk/src/sync";
|
|
|
|
import SdkConfig from '../SdkConfig';
|
|
import sendBugReport from '../rageshake/submit-rageshake';
|
|
import defaultDispatcher from '../dispatcher/dispatcher';
|
|
import { AsyncStoreWithClient } from './AsyncStoreWithClient';
|
|
import { ActionPayload } from '../dispatcher/payloads';
|
|
import SettingsStore from "../settings/SettingsStore";
|
|
import { Action } from "../dispatcher/actions";
|
|
|
|
// Minimum interval of 1 minute between reports
|
|
const RAGESHAKE_INTERVAL = 60000;
|
|
// Before rageshaking, wait 5 seconds and see if the message has successfully decrypted
|
|
const GRACE_PERIOD = 5000;
|
|
// Event type for to-device messages requesting sender auto-rageshakes
|
|
const AUTO_RS_REQUEST = "im.vector.auto_rs_request";
|
|
|
|
interface IState {
|
|
reportedSessionIds: Set<string>;
|
|
lastRageshakeTime: number;
|
|
initialSyncCompleted: boolean;
|
|
}
|
|
|
|
/**
|
|
* Watches for decryption errors to auto-report if the relevant lab is
|
|
* enabled, and keeps track of session IDs that have already been
|
|
* reported.
|
|
*/
|
|
export default class AutoRageshakeStore extends AsyncStoreWithClient<IState> {
|
|
private static internalInstance = new AutoRageshakeStore();
|
|
|
|
private constructor() {
|
|
super(defaultDispatcher, {
|
|
reportedSessionIds: new Set<string>(),
|
|
lastRageshakeTime: 0,
|
|
initialSyncCompleted: false,
|
|
});
|
|
this.onDecryptionAttempt = this.onDecryptionAttempt.bind(this);
|
|
this.onDeviceMessage = this.onDeviceMessage.bind(this);
|
|
this.onSyncStateChange = this.onSyncStateChange.bind(this);
|
|
}
|
|
|
|
public static get instance(): AutoRageshakeStore {
|
|
return AutoRageshakeStore.internalInstance;
|
|
}
|
|
|
|
protected async onAction(payload: ActionPayload) {
|
|
switch (payload.action) {
|
|
case Action.ReportKeyBackupNotEnabled:
|
|
this.onReportKeyBackupNotEnabled();
|
|
}
|
|
}
|
|
|
|
protected async onReady() {
|
|
if (!SettingsStore.getValue("automaticDecryptionErrorReporting")) return;
|
|
|
|
if (this.matrixClient) {
|
|
this.matrixClient.on(MatrixEventEvent.Decrypted, this.onDecryptionAttempt);
|
|
this.matrixClient.on(ClientEvent.ToDeviceEvent, this.onDeviceMessage);
|
|
this.matrixClient.on(ClientEvent.Sync, this.onSyncStateChange);
|
|
}
|
|
}
|
|
|
|
protected async onNotReady() {
|
|
if (this.matrixClient) {
|
|
this.matrixClient.removeListener(ClientEvent.ToDeviceEvent, this.onDeviceMessage);
|
|
this.matrixClient.removeListener(MatrixEventEvent.Decrypted, this.onDecryptionAttempt);
|
|
this.matrixClient.removeListener(ClientEvent.Sync, this.onSyncStateChange);
|
|
}
|
|
}
|
|
|
|
private async onDecryptionAttempt(ev: MatrixEvent): Promise<void> {
|
|
if (!this.state.initialSyncCompleted) { return; }
|
|
|
|
const wireContent = ev.getWireContent();
|
|
const sessionId = wireContent.session_id;
|
|
if (ev.isDecryptionFailure() && !this.state.reportedSessionIds.has(sessionId)) {
|
|
await sleep(GRACE_PERIOD);
|
|
if (!ev.isDecryptionFailure()) { return; }
|
|
|
|
const newReportedSessionIds = new Set(this.state.reportedSessionIds);
|
|
await this.updateState({ reportedSessionIds: newReportedSessionIds.add(sessionId) });
|
|
|
|
const now = new Date().getTime();
|
|
if (now - this.state.lastRageshakeTime < RAGESHAKE_INTERVAL) { return; }
|
|
|
|
await this.updateState({ lastRageshakeTime: now });
|
|
|
|
const eventInfo = {
|
|
"event_id": ev.getId(),
|
|
"room_id": ev.getRoomId(),
|
|
"session_id": sessionId,
|
|
"device_id": wireContent.device_id,
|
|
"user_id": ev.getSender(),
|
|
"sender_key": wireContent.sender_key,
|
|
};
|
|
|
|
const rageshakeURL = await sendBugReport(SdkConfig.get().bug_report_endpoint_url, {
|
|
userText: "Auto-reporting decryption error (recipient)",
|
|
sendLogs: true,
|
|
labels: ["Z-UISI", "web", "uisi-recipient"],
|
|
customApp: SdkConfig.get().uisi_autorageshake_app,
|
|
customFields: { "auto_uisi": JSON.stringify(eventInfo) },
|
|
});
|
|
|
|
const messageContent = {
|
|
...eventInfo,
|
|
"recipient_rageshake": rageshakeURL,
|
|
};
|
|
this.matrixClient.sendToDevice(
|
|
AUTO_RS_REQUEST,
|
|
{ [messageContent.user_id]: { [messageContent.device_id]: messageContent } },
|
|
);
|
|
}
|
|
}
|
|
|
|
private async onSyncStateChange(_state: SyncState, _prevState: SyncState, data: ISyncStateData) {
|
|
if (!this.state.initialSyncCompleted) {
|
|
await this.updateState({ initialSyncCompleted: !!data.nextSyncToken });
|
|
}
|
|
}
|
|
|
|
private async onDeviceMessage(ev: MatrixEvent): Promise<void> {
|
|
if (ev.getType() !== AUTO_RS_REQUEST) return;
|
|
const messageContent = ev.getContent();
|
|
const recipientRageshake = messageContent["recipient_rageshake"] || "";
|
|
const now = new Date().getTime();
|
|
if (now - this.state.lastRageshakeTime > RAGESHAKE_INTERVAL) {
|
|
await this.updateState({ lastRageshakeTime: now });
|
|
await sendBugReport(SdkConfig.get().bug_report_endpoint_url, {
|
|
userText: `Auto-reporting decryption error (sender)\nRecipient rageshake: ${recipientRageshake}`,
|
|
sendLogs: true,
|
|
labels: ["Z-UISI", "web", "uisi-sender"],
|
|
customApp: SdkConfig.get().uisi_autorageshake_app,
|
|
customFields: {
|
|
"recipient_rageshake": recipientRageshake,
|
|
"auto_uisi": JSON.stringify(messageContent),
|
|
},
|
|
});
|
|
}
|
|
}
|
|
|
|
private async onReportKeyBackupNotEnabled(): Promise<void> {
|
|
if (!SettingsStore.getValue("automaticKeyBackNotEnabledReporting")) return;
|
|
|
|
await sendBugReport(SdkConfig.get().bug_report_endpoint_url, {
|
|
userText: `Auto-reporting key backup not enabled`,
|
|
sendLogs: true,
|
|
labels: ["web", Action.ReportKeyBackupNotEnabled],
|
|
});
|
|
}
|
|
}
|
|
|
|
window.mxAutoRageshakeStore = AutoRageshakeStore.instance;
|