From 8abd3a64084904e6f7691a678da7e41c9087916f Mon Sep 17 00:00:00 2001 From: Hugh Nimmo-Smith Date: Thu, 19 Sep 2024 09:50:54 +0100 Subject: [PATCH] Use new crypto api to send to-device messages from widgets --- src/stores/widgets/StopGapWidgetDriver.ts | 49 ++++++++++------------- 1 file changed, 22 insertions(+), 27 deletions(-) diff --git a/src/stores/widgets/StopGapWidgetDriver.ts b/src/stores/widgets/StopGapWidgetDriver.ts index 88e79d97f2..916fca5b5f 100644 --- a/src/stores/widgets/StopGapWidgetDriver.ts +++ b/src/stores/widgets/StopGapWidgetDriver.ts @@ -422,33 +422,28 @@ export class StopGapWidgetDriver extends WidgetDriver { const client = MatrixClientPeg.safeGet(); if (encrypted) { - const deviceInfoMap = await client.crypto!.deviceList.downloadKeys(Object.keys(contentMap), false); - - await Promise.all( - Object.entries(contentMap).flatMap(([userId, userContentMap]) => - Object.entries(userContentMap).map(async ([deviceId, content]): Promise => { - const devices = deviceInfoMap.get(userId); - if (!devices) return; - - if (deviceId === "*") { - // Send the message to all devices we have keys for - await client.encryptAndSendToDevices( - Array.from(devices.values()).map((deviceInfo) => ({ - userId, - deviceInfo, - })), - content, - ); - } else if (devices.has(deviceId)) { - // Send the message to a specific device - await client.encryptAndSendToDevices( - [{ userId, deviceInfo: devices.get(deviceId)! }], - content, - ); - } - }), - ), - ); + const crypto = client.getCrypto(); + if (!crypto) throw new Error("E2EE not enabled"); + + // attempt to re-batch these up into a single request + const invertedContentMap: { [content: string]: { userId: string; deviceId: string }[] } = {}; + + Object.entries(contentMap).forEach(([userId, userContentMap]) => + Object.entries(userContentMap).forEach(([deviceId, content]) => { + const stringifiedContent = JSON.stringify(content); + invertedContentMap[stringifiedContent] = invertedContentMap[stringifiedContent] || []; + + invertedContentMap[stringifiedContent].push({ userId, deviceId }); + }), + ), + + await Promise.all(Object.entries(invertedContentMap).map( + async ([stringifiedContent, recipients]) => { + const batch = await crypto.encryptToDeviceMessages(eventType, recipients, JSON.parse(stringifiedContent)); + + await client.queueToDevice(batch); + }, + )); } else { await client.queueToDevice({ eventType,