mirror of https://github.com/oxen-io/session-ios
				
				
				
			
			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.
		
		
		
		
		
			
		
			
				
	
	
		
			214 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Swift
		
	
			
		
		
	
	
			214 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Swift
		
	
| // Copyright © 2022 Rangeproof Pty Ltd. All rights reserved.
 | |
| 
 | |
| import Foundation
 | |
| import GRDB
 | |
| import Sodium
 | |
| import SessionUtilitiesKit
 | |
| 
 | |
| extension MessageReceiver {
 | |
|     internal static func handleConfigurationMessage(_ db: Database, message: ConfigurationMessage) throws {
 | |
|         guard !Features.useSharedUtilForUserConfig else {
 | |
|             // TODO: Show warning prompt for X days
 | |
|             return
 | |
|         }
 | |
|         
 | |
|         let userPublicKey = getUserHexEncodedPublicKey(db)
 | |
|         
 | |
|         guard message.sender == userPublicKey else { return }
 | |
|         
 | |
|         SNLog("Configuration message received.")
 | |
|         
 | |
|         // Note: `message.sentTimestamp` is in ms (convert to TimeInterval before converting to
 | |
|         // seconds to maintain the accuracy)
 | |
|         let isInitialSync: Bool = (!UserDefaults.standard[.hasSyncedInitialConfiguration])
 | |
|         let messageSentTimestamp: TimeInterval = TimeInterval((message.sentTimestamp ?? 0) / 1000)
 | |
|         let lastConfigTimestamp: TimeInterval = UserDefaults.standard[.lastConfigurationSync]
 | |
|             .defaulting(to: Date(timeIntervalSince1970: 0))
 | |
|             .timeIntervalSince1970
 | |
|         
 | |
|         // Handle user profile changes
 | |
|         try ProfileManager.updateProfileIfNeeded(
 | |
|             db,
 | |
|             publicKey: userPublicKey,
 | |
|             name: message.displayName,
 | |
|             avatarUpdate: {
 | |
|                 guard
 | |
|                     let profilePictureUrl: String = message.profilePictureUrl,
 | |
|                     let profileKey: Data = message.profileKey
 | |
|                 else { return .none }
 | |
|                 
 | |
|                 return .updateTo(
 | |
|                     url: profilePictureUrl,
 | |
|                     key: profileKey,
 | |
|                     fileName: nil
 | |
|                 )
 | |
|             }(),
 | |
|             sentTimestamp: messageSentTimestamp,
 | |
|             calledFromConfigHandling: true
 | |
|         )
 | |
|         
 | |
|         // Create a contact for the current user if needed (also force-approve the current user
 | |
|         // in case the account got into a weird state or restored directly from a migration)
 | |
|         let userContact: Contact = Contact.fetchOrCreate(db, id: userPublicKey)
 | |
|         
 | |
|         if !userContact.isTrusted || !userContact.isApproved || !userContact.didApproveMe {
 | |
|             try userContact.save(db)
 | |
|             try Contact
 | |
|                 .filter(id: userPublicKey)
 | |
|                 .updateAll( // Handling a config update so don't use `updateAllAndConfig`
 | |
|                     db,
 | |
|                     Contact.Columns.isTrusted.set(to: true),
 | |
|                     Contact.Columns.isApproved.set(to: true),
 | |
|                     Contact.Columns.didApproveMe.set(to: true)
 | |
|                 )
 | |
|         }
 | |
|         
 | |
|         if isInitialSync || messageSentTimestamp > lastConfigTimestamp {
 | |
|             if isInitialSync {
 | |
|                 UserDefaults.standard[.hasSyncedInitialConfiguration] = true
 | |
|                 NotificationCenter.default.post(name: .initialConfigurationMessageReceived, object: nil)
 | |
|             }
 | |
|             
 | |
|             UserDefaults.standard[.lastConfigurationSync] = Date(timeIntervalSince1970: messageSentTimestamp)
 | |
|             
 | |
|             // Contacts
 | |
|             try message.contacts.forEach { contactInfo in
 | |
|                 guard let sessionId: String = contactInfo.publicKey else { return }
 | |
|                 
 | |
|                 // If the contact is a blinded contact then only add them if they haven't already been
 | |
|                 // unblinded
 | |
|                 if SessionId.Prefix(from: sessionId) == .blinded {
 | |
|                     let hasUnblindedContact: Bool = BlindedIdLookup
 | |
|                         .filter(BlindedIdLookup.Columns.blindedId == sessionId)
 | |
|                         .filter(BlindedIdLookup.Columns.sessionId != nil)
 | |
|                         .isNotEmpty(db)
 | |
|                     
 | |
|                     if hasUnblindedContact {
 | |
|                         return
 | |
|                     }
 | |
|                 }
 | |
|                 
 | |
|                 // Note: We only update the contact and profile records if the data has actually changed
 | |
|                 // in order to avoid triggering UI updates for every thread on the home screen
 | |
|                 let contact: Contact = Contact.fetchOrCreate(db, id: sessionId)
 | |
|                 let profile: Profile = Profile.fetchOrCreate(db, id: sessionId)
 | |
|                 
 | |
|                 if
 | |
|                     profile.name != contactInfo.displayName ||
 | |
|                     profile.profilePictureUrl != contactInfo.profilePictureUrl ||
 | |
|                     profile.profileEncryptionKey != contactInfo.profileKey
 | |
|                 {
 | |
|                     try profile.save(db)
 | |
|                     try Profile
 | |
|                         .filter(id: sessionId)
 | |
|                         .updateAll( // Handling a config update so don't use `updateAllAndConfig`
 | |
|                             db,
 | |
|                             [
 | |
|                                 Profile.Columns.name.set(to: contactInfo.displayName),
 | |
|                                 (contactInfo.profilePictureUrl == nil ? nil :
 | |
|                                     Profile.Columns.profilePictureUrl.set(to: contactInfo.profilePictureUrl)
 | |
|                                 ),
 | |
|                                 (contactInfo.profileKey == nil ? nil :
 | |
|                                     Profile.Columns.profileEncryptionKey.set(to: contactInfo.profileKey)
 | |
|                                 )
 | |
|                             ].compactMap { $0 }
 | |
|                         )
 | |
|                 }
 | |
|                 
 | |
|                 /// We only update these values if the proto actually has values for them (this is to prevent an
 | |
|                 /// edge case where an old client could override the values with default values since they aren't included)
 | |
|                 ///
 | |
|                 /// **Note:** Since message requests have no reverse, we should only handle setting `isApproved`
 | |
|                 /// and `didApproveMe` to `true`. This may prevent some weird edge cases where a config message
 | |
|                 /// swapping `isApproved` and `didApproveMe` to `false`
 | |
|                 if
 | |
|                     (contactInfo.hasIsApproved && (contact.isApproved != contactInfo.isApproved)) ||
 | |
|                     (contactInfo.hasIsBlocked && (contact.isBlocked != contactInfo.isBlocked)) ||
 | |
|                     (contactInfo.hasDidApproveMe && (contact.didApproveMe != contactInfo.didApproveMe))
 | |
|                 {
 | |
|                     try contact.save(db)
 | |
|                     try Contact
 | |
|                         .filter(id: sessionId)
 | |
|                         .updateAll( // Handling a config update so don't use `updateAllAndConfig`
 | |
|                             db,
 | |
|                             [
 | |
|                                 (!contactInfo.hasIsApproved || !contactInfo.isApproved ? nil :
 | |
|                                     Contact.Columns.isApproved.set(to: true)
 | |
|                                 ),
 | |
|                                 (!contactInfo.hasIsBlocked ? nil :
 | |
|                                     Contact.Columns.isBlocked.set(to: contactInfo.isBlocked)
 | |
|                                 ),
 | |
|                                 (!contactInfo.hasDidApproveMe || !contactInfo.didApproveMe ? nil :
 | |
|                                     Contact.Columns.didApproveMe.set(to: contactInfo.didApproveMe)
 | |
|                                 )
 | |
|                             ].compactMap { $0 }
 | |
|                         )
 | |
|                 }
 | |
|                 
 | |
|                 // If the contact is blocked
 | |
|                 if contactInfo.hasIsBlocked && contactInfo.isBlocked {
 | |
|                     // If this message changed them to the blocked state and there is an existing thread
 | |
|                     // associated with them that is a message request thread then delete it (assume
 | |
|                     // that the current user had deleted that message request)
 | |
|                     if
 | |
|                         contactInfo.isBlocked != contact.isBlocked, // 'contact.isBlocked' will be the old value
 | |
|                         let thread: SessionThread = try? SessionThread.fetchOne(db, id: sessionId),
 | |
|                         thread.isMessageRequest(db)
 | |
|                     {
 | |
|                         _ = try thread.delete(db)
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|             
 | |
|             // Closed groups
 | |
|             //
 | |
|             // Note: Only want to add these for initial sync to avoid re-adding closed groups the user
 | |
|             // intentionally left (any closed groups joined since the first processed sync message should
 | |
|             // get added via the 'handleNewClosedGroup' method anyway as they will have come through in the
 | |
|             // past two weeks)
 | |
|             if isInitialSync {
 | |
|                 let existingClosedGroupsIds: [String] = (try? SessionThread
 | |
|                     .filter(SessionThread.Columns.variant == SessionThread.Variant.legacyGroup)
 | |
|                     .fetchAll(db))
 | |
|                     .defaulting(to: [])
 | |
|                     .map { $0.id }
 | |
|                 
 | |
|                 try message.closedGroups.forEach { closedGroup in
 | |
|                     guard !existingClosedGroupsIds.contains(closedGroup.publicKey) else { return }
 | |
|                     
 | |
|                     let keyPair: Box.KeyPair = Box.KeyPair(
 | |
|                         publicKey: closedGroup.encryptionKeyPublicKey.bytes,
 | |
|                         secretKey: closedGroup.encryptionKeySecretKey.bytes
 | |
|                     )
 | |
|                     
 | |
|                     try MessageReceiver.handleNewClosedGroup(
 | |
|                         db,
 | |
|                         groupPublicKey: closedGroup.publicKey,
 | |
|                         name: closedGroup.name,
 | |
|                         encryptionKeyPair: keyPair,
 | |
|                         members: [String](closedGroup.members),
 | |
|                         admins: [String](closedGroup.admins),
 | |
|                         expirationTimer: closedGroup.expirationTimer,
 | |
|                         messageSentTimestamp: message.sentTimestamp!
 | |
|                     )
 | |
|                 }
 | |
|             }
 | |
|             
 | |
|             // Open groups
 | |
|             for openGroupURL in message.openGroups {
 | |
|                 if let (room, server, publicKey) = OpenGroupManager.parseOpenGroup(from: openGroupURL) {
 | |
|                     OpenGroupManager.shared
 | |
|                         .add(
 | |
|                             db,
 | |
|                             roomToken: room,
 | |
|                             server: server,
 | |
|                             publicKey: publicKey,
 | |
|                             calledFromConfigHandling: true
 | |
|                         )
 | |
|                         .sinkUntilComplete()
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 |