From 332cd3005213a1293ee486c0e1893ee5f55774bf Mon Sep 17 00:00:00 2001 From: Beaudan Brown Date: Thu, 7 Nov 2019 11:30:29 +1100 Subject: [PATCH] Send contact sync message when becoming friends with contact and auto accept/send friend requests when receiving contact sync --- js/background.js | 8 ++++++++ js/models/conversations.js | 11 ++++++++--- libloki/api.js | 12 ++++++------ libtextsecure/sendmessage.js | 28 ++++++++++++++++++++++++++++ 4 files changed, 50 insertions(+), 9 deletions(-) diff --git a/js/background.js b/js/background.js index 0a356c577..617aade94 100644 --- a/js/background.js +++ b/js/background.js @@ -1239,6 +1239,14 @@ await conversation.setSecondaryStatus(true); } + if (conversation.isFriendRequestStatusNone()) { + // Will be replaced with automatic friend request + libloki.api.sendBackgroundMessage(conversation.id); + } else { + // Accept any pending friend requests if there are any + conversation.onAcceptFriendRequest({ fromContactSync: true }); + } + if (details.profileKey) { const profileKey = window.Signal.Crypto.arrayBufferToBase64( details.profileKey diff --git a/js/models/conversations.js b/js/models/conversations.js index 6740514fd..1f562ab23 100644 --- a/js/models/conversations.js +++ b/js/models/conversations.js @@ -774,7 +774,8 @@ }); } }, - async setFriendRequestStatus(newStatus) { + async setFriendRequestStatus(newStatus, options = {}) { + const { fromContactSync } = options; // Ensure that the new status is a valid FriendStatusEnum value if (!(newStatus in Object.values(FriendRequestStatusEnum))) { return; @@ -791,6 +792,10 @@ Conversation: Whisper.Conversation, }); await this.updateTextInputState(); + if (!fromContactSync && newStatus === FriendRequestStatusEnum.friends) { + // Sync contact + this.wrapSend(textsecure.messaging.sendContactSyncMessage(this)); + } } }, async respondToAllFriendRequests(options) { @@ -837,12 +842,12 @@ await window.libloki.storage.removeContactPreKeyBundle(this.id); }, // We have accepted an incoming friend request - async onAcceptFriendRequest() { + async onAcceptFriendRequest(options = {}) { if (this.unlockTimer) { clearTimeout(this.unlockTimer); } if (this.hasReceivedFriendRequest()) { - this.setFriendRequestStatus(FriendRequestStatusEnum.friends); + this.setFriendRequestStatus(FriendRequestStatusEnum.friends, options); await this.respondToAllFriendRequests({ response: 'accepted', direction: 'incoming', diff --git a/libloki/api.js b/libloki/api.js index a41e3c34d..5d5949d0b 100644 --- a/libloki/api.js +++ b/libloki/api.js @@ -114,11 +114,7 @@ result.reset(); return result; } - async function createContactSyncProtoMessage() { - const conversations = await window.Signal.Data.getConversationsWithFriendStatus( - window.friends.friendRequestStatusEnum.friends, - { ConversationCollection: Whisper.ConversationCollection } - ); + async function createContactSyncProtoMessage(conversations) { // Extract required contacts information out of conversations const rawContacts = conversations.map(conversation => { const profile = conversation.getLokiProfile(); @@ -186,7 +182,11 @@ profile, }); // Attach contact list - const syncMessage = await createContactSyncProtoMessage(); + const conversations = await window.Signal.Data.getConversationsWithFriendStatus( + window.friends.friendRequestStatusEnum.friends, + { ConversationCollection: Whisper.ConversationCollection } + ); + const syncMessage = await createContactSyncProtoMessage(conversations); const content = new textsecure.protobuf.Content({ pairingAuthorisation, dataMessage, diff --git a/libtextsecure/sendmessage.js b/libtextsecure/sendmessage.js index 21e234967..31f354cee 100644 --- a/libtextsecure/sendmessage.js +++ b/libtextsecure/sendmessage.js @@ -563,6 +563,33 @@ MessageSender.prototype = { return Promise.resolve(); }, + async sendContactSyncMessage(contactConversation) { + const primaryDeviceKey = window.storage.get('primaryDevicePubKey'); + const allOurDevices = (await libloki.storage.getAllDevicePubKeysForPrimaryPubKey( + primaryDeviceKey + )) + // Don't send to ourselves + .filter(pubKey => pubKey !== textsecure.storage.user.getNumber()); + if (allOurDevices.length === 0) { + return Promise.resolve(); + } + + const syncMessage = await libloki.api.createContactSyncProtoMessage([ + contactConversation, + ]); + const contentMessage = new textsecure.protobuf.Content(); + contentMessage.syncMessage = syncMessage; + + const silent = true; + return this.sendIndividualProto( + primaryDeviceKey, + contentMessage, + Date.now(), + silent, + {} // options + ); + }, + sendRequestContactSyncMessage(options) { const myNumber = textsecure.storage.user.getNumber(); const myDevice = textsecure.storage.user.getDeviceId(); @@ -1160,6 +1187,7 @@ textsecure.MessageSender = function MessageSenderWrapper(username, password) { this.sendRequestContactSyncMessage = sender.sendRequestContactSyncMessage.bind( sender ); + this.sendContactSyncMessage = sender.sendContactSyncMessage.bind(sender); this.sendRequestConfigurationSyncMessage = sender.sendRequestConfigurationSyncMessage.bind( sender );