/* 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 { MatrixEvent } from "matrix-js-sdk/src"; 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"; // 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; 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 { private static internalInstance = new AutoRageshakeStore(); private constructor() { super(defaultDispatcher, { reportedSessionIds: new Set(), 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) { // we don't actually do anything here } protected async onReady() { if (!SettingsStore.getValue("automaticDecryptionErrorReporting")) return; if (this.matrixClient) { this.matrixClient.on('Event.decrypted', this.onDecryptionAttempt); this.matrixClient.on('toDeviceEvent', this.onDeviceMessage); this.matrixClient.on('sync', this.onSyncStateChange); } } protected async onNotReady() { if (this.matrixClient) { this.matrixClient.removeListener('toDeviceEvent', this.onDeviceMessage); this.matrixClient.removeListener('Event.decrypted', this.onDecryptionAttempt); } } private async onDecryptionAttempt(ev: MatrixEvent): Promise { 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 { 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), }, }); } } } window.mxAutoRageshakeStore = AutoRageshakeStore.instance;