You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
	
	
		
			202 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			TypeScript
		
	
			
		
		
	
	
			202 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			TypeScript
		
	
| import _ from 'lodash';
 | |
| import { createOrUpdateItem, getItemById, hasSyncedInitialConfigurationItem } from '../data/data';
 | |
| import { ConversationTypeEnum } from '../models/conversation';
 | |
| import {
 | |
|   joinOpenGroupV2WithUIEvents,
 | |
|   parseOpenGroupV2,
 | |
| } from '../session/apis/open_group_api/opengroupV2/JoinOpenGroupV2';
 | |
| import { getOpenGroupV2ConversationId } from '../session/apis/open_group_api/utils/OpenGroupUtils';
 | |
| import { SignalService } from '../protobuf';
 | |
| import { getConversationController } from '../session/conversations';
 | |
| import { UserUtils } from '../session/utils';
 | |
| import { toHex } from '../session/utils/String';
 | |
| import { configurationMessageReceived, trigger } from '../shims/events';
 | |
| import { BlockedNumberController } from '../util';
 | |
| import { removeFromCache } from './cache';
 | |
| import { handleNewClosedGroup } from './closedGroups';
 | |
| import { updateProfileOneAtATime } from './dataMessage';
 | |
| import { EnvelopePlus } from './types';
 | |
| import { ConversationInteraction } from '../interactions';
 | |
| 
 | |
| async function handleOurProfileUpdate(
 | |
|   sentAt: number | Long,
 | |
|   configMessage: SignalService.ConfigurationMessage,
 | |
|   ourPubkey: string
 | |
| ) {
 | |
|   const latestProfileUpdateTimestamp = UserUtils.getLastProfileUpdateTimestamp();
 | |
|   if (!latestProfileUpdateTimestamp || sentAt > latestProfileUpdateTimestamp) {
 | |
|     window?.log?.info(
 | |
|       `Handling our profileUdpate ourLastUpdate:${latestProfileUpdateTimestamp}, envelope sent at: ${sentAt}`
 | |
|     );
 | |
|     const { profileKey, profilePicture, displayName } = configMessage;
 | |
| 
 | |
|     const ourConversation = getConversationController().get(ourPubkey);
 | |
|     if (!ourConversation) {
 | |
|       window?.log?.error('We need a convo with ourself at all times');
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     const lokiProfile = {
 | |
|       displayName,
 | |
|       profilePicture,
 | |
|     };
 | |
|     await updateProfileOneAtATime(ourConversation, lokiProfile, profileKey);
 | |
|     UserUtils.setLastProfileUpdateTimestamp(_.toNumber(sentAt));
 | |
|     // do not trigger a signin by linking if the display name is empty
 | |
|     if (displayName) {
 | |
|       trigger(configurationMessageReceived, displayName);
 | |
|     } else {
 | |
|       window?.log?.warn('Got a configuration message but the display name is empty');
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| async function handleGroupsAndContactsFromConfigMessage(
 | |
|   envelope: EnvelopePlus,
 | |
|   configMessage: SignalService.ConfigurationMessage
 | |
| ) {
 | |
|   const lastConfigUpdate = await getItemById(hasSyncedInitialConfigurationItem);
 | |
|   const lastConfigTimestamp = lastConfigUpdate?.timestamp;
 | |
|   const isNewerConfig = lastConfigTimestamp && lastConfigTimestamp < _.toNumber(envelope.timestamp);
 | |
| 
 | |
|   if (isNewerConfig) {
 | |
|     window?.log?.info(
 | |
|       'Dropping configuration groups change as we already handled one... Only handling contacts '
 | |
|     );
 | |
|     if (isNewerConfig) {
 | |
|       if (configMessage.contacts?.length) {
 | |
|         await Promise.all(
 | |
|           configMessage.contacts.map(async c => handleContactReceived(c, envelope))
 | |
|         );
 | |
|       }
 | |
|       return;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   await createOrUpdateItem({
 | |
|     id: 'hasSyncedInitialConfigurationItem',
 | |
|     value: true,
 | |
|     timestamp: _.toNumber(envelope.timestamp),
 | |
|   });
 | |
| 
 | |
|   const numberClosedGroup = configMessage.closedGroups?.length || 0;
 | |
| 
 | |
|   window?.log?.info(
 | |
|     `Received ${numberClosedGroup} closed group on configuration. Creating them... `
 | |
|   );
 | |
| 
 | |
|   await Promise.all(
 | |
|     configMessage.closedGroups.map(async c => {
 | |
|       const groupUpdate = new SignalService.DataMessage.ClosedGroupControlMessage({
 | |
|         type: SignalService.DataMessage.ClosedGroupControlMessage.Type.NEW,
 | |
|         encryptionKeyPair: c.encryptionKeyPair,
 | |
|         name: c.name,
 | |
|         admins: c.admins,
 | |
|         members: c.members,
 | |
|         publicKey: c.publicKey,
 | |
|       });
 | |
|       try {
 | |
|         await handleNewClosedGroup(envelope, groupUpdate);
 | |
|       } catch (e) {
 | |
|         window?.log?.warn('failed to handle  a new closed group from configuration message');
 | |
|       }
 | |
|     })
 | |
|   );
 | |
| 
 | |
|   const numberOpenGroup = configMessage.openGroups?.length || 0;
 | |
| 
 | |
|   // Trigger a join for all open groups we are not already in.
 | |
|   // Currently, if you left an open group but kept the conversation, you won't rejoin it here.
 | |
|   for (let i = 0; i < numberOpenGroup; i++) {
 | |
|     const currentOpenGroupUrl = configMessage.openGroups[i];
 | |
|     const parsedRoom = parseOpenGroupV2(currentOpenGroupUrl);
 | |
|     if (!parsedRoom) {
 | |
|       continue;
 | |
|     }
 | |
|     const roomConvoId = getOpenGroupV2ConversationId(parsedRoom.serverUrl, parsedRoom.roomId);
 | |
|     if (!getConversationController().get(roomConvoId)) {
 | |
|       window?.log?.info(
 | |
|         `triggering join of public chat '${currentOpenGroupUrl}' from ConfigurationMessage`
 | |
|       );
 | |
|       void joinOpenGroupV2WithUIEvents(currentOpenGroupUrl, false, true);
 | |
|     }
 | |
|   }
 | |
|   if (configMessage.contacts?.length && isNewerConfig) {
 | |
|     await Promise.all(configMessage.contacts.map(async c => handleContactReceived(c, envelope)));
 | |
|   }
 | |
| }
 | |
| 
 | |
| const handleContactReceived = async (
 | |
|   contactReceived: SignalService.ConfigurationMessage.IContact,
 | |
|   envelope: EnvelopePlus
 | |
| ) => {
 | |
|   try {
 | |
|     if (!contactReceived.publicKey?.length) {
 | |
|       return;
 | |
|     }
 | |
|     const contactConvo = await getConversationController().getOrCreateAndWait(
 | |
|       toHex(contactReceived.publicKey),
 | |
|       ConversationTypeEnum.PRIVATE
 | |
|     );
 | |
|     const profile = {
 | |
|       displayName: contactReceived.name,
 | |
|       profilePictre: contactReceived.profilePicture,
 | |
|     };
 | |
| 
 | |
|     const existingActiveAt = contactConvo.get('active_at');
 | |
|     if (!existingActiveAt || existingActiveAt === 0) {
 | |
|       contactConvo.set('active_at', _.toNumber(envelope.timestamp));
 | |
|     }
 | |
| 
 | |
|     // checking for existence of field on protobuf
 | |
|     if (contactReceived.isApproved === true) {
 | |
|       if (!contactConvo.isApproved()) {
 | |
|         await contactConvo.setIsApproved(Boolean(contactReceived.isApproved));
 | |
|         await contactConvo.addOutgoingApprovalMessage(_.toNumber(envelope.timestamp));
 | |
|       }
 | |
| 
 | |
|       if (contactReceived.didApproveMe === true) {
 | |
|         // checking for existence of field on message
 | |
|         await contactConvo.setDidApproveMe(Boolean(contactReceived.didApproveMe));
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     // only set for explicit true/false values incase outdated sender doesn't have the fields
 | |
|     if (contactReceived.isBlocked === true) {
 | |
|       if (contactConvo.isIncomingRequest()) {
 | |
|         // handling case where restored device's declined message requests were getting restored
 | |
|         await ConversationInteraction.deleteAllMessagesByConvoIdNoConfirmation(contactConvo.id);
 | |
|       }
 | |
|       await BlockedNumberController.block(contactConvo.id);
 | |
|     } else if (contactReceived.isBlocked === false) {
 | |
|       await BlockedNumberController.unblock(contactConvo.id);
 | |
|     }
 | |
| 
 | |
|     void updateProfileOneAtATime(contactConvo, profile, contactReceived.profileKey);
 | |
|   } catch (e) {
 | |
|     window?.log?.warn('failed to handle  a new closed group from configuration message');
 | |
|   }
 | |
| };
 | |
| 
 | |
| export async function handleConfigurationMessage(
 | |
|   envelope: EnvelopePlus,
 | |
|   configurationMessage: SignalService.ConfigurationMessage
 | |
| ): Promise<void> {
 | |
|   window?.log?.info('Handling configuration message');
 | |
|   const ourPubkey = UserUtils.getOurPubKeyStrFromCache();
 | |
|   if (!ourPubkey) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if (envelope.source !== ourPubkey) {
 | |
|     window?.log?.info('Dropping configuration change from someone else than us.');
 | |
|     return removeFromCache(envelope);
 | |
|   }
 | |
| 
 | |
|   await handleOurProfileUpdate(envelope.timestamp, configurationMessage, ourPubkey);
 | |
| 
 | |
|   await handleGroupsAndContactsFromConfigMessage(envelope, configurationMessage);
 | |
| 
 | |
|   await removeFromCache(envelope);
 | |
| }
 |