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.
		
		
		
		
		
			
		
			
	
	
		
			161 lines
		
	
	
		
			6.5 KiB
		
	
	
	
		
			Swift
		
	
		
		
			
		
	
	
			161 lines
		
	
	
		
			6.5 KiB
		
	
	
	
		
			Swift
		
	
| 
											3 years ago
										 | // Copyright © 2023 Rangeproof Pty Ltd. All rights reserved. | ||
| 
											3 years ago
										 | 
 | ||
|  | import Foundation | ||
|  | import GRDB | ||
|  | import SessionUtil | ||
|  | import SessionUtilitiesKit | ||
|  | 
 | ||
|  | internal extension SessionUtil { | ||
| 
											3 years ago
										 |     static let columnsRelatedToUserProfile: [Profile.Columns] = [ | ||
|  |         Profile.Columns.name, | ||
|  |         Profile.Columns.profilePictureUrl, | ||
|  |         Profile.Columns.profileEncryptionKey | ||
|  |     ] | ||
|  |      | ||
| 
											3 years ago
										 |     // MARK: - Incoming Changes | ||
|  |      | ||
| 
											3 years ago
										 |     static func handleUserProfileUpdate( | ||
|  |         _ db: Database, | ||
| 
											3 years ago
										 |         in conf: UnsafeMutablePointer<config_object>?, | ||
|  |         mergeNeedsDump: Bool, | ||
| 
											3 years ago
										 |         latestConfigUpdateSentTimestamp: TimeInterval | ||
| 
											3 years ago
										 |     ) throws { | ||
| 
											3 years ago
										 |         typealias ProfileData = (profileName: String, profilePictureUrl: String?, profilePictureKey: Data?) | ||
|  |          | ||
| 
											3 years ago
										 |         guard mergeNeedsDump else { return } | ||
|  |         guard conf != nil else { throw SessionUtilError.nilConfigObject } | ||
| 
											3 years ago
										 |          | ||
| 
											3 years ago
										 |         // A profile must have a name so if this is null then it's invalid and can be ignored | ||
|  |         guard let profileNamePtr: UnsafePointer<CChar> = user_profile_get_name(conf) else { return } | ||
| 
											3 years ago
										 |          | ||
| 
											3 years ago
										 |         let userPublicKey: String = getUserHexEncodedPublicKey(db) | ||
|  |         let profileName: String = String(cString: profileNamePtr) | ||
|  |         let profilePic: user_profile_pic = user_profile_get_pic(conf) | ||
|  |         let profilePictureUrl: String? = String(libSessionVal: profilePic.url, nullIfEmpty: true) | ||
| 
											3 years ago
										 |          | ||
| 
											3 years ago
										 |         // Handle user profile changes | ||
|  |         try ProfileManager.updateProfileIfNeeded( | ||
| 
											3 years ago
										 |             db, | ||
|  |             publicKey: userPublicKey, | ||
| 
											3 years ago
										 |             name: profileName, | ||
| 
											3 years ago
										 |             avatarUpdate: { | ||
| 
											3 years ago
										 |                 guard let profilePictureUrl: String = profilePictureUrl else { return .remove } | ||
| 
											3 years ago
										 |                  | ||
|  |                 return .updateTo( | ||
|  |                     url: profilePictureUrl, | ||
| 
											3 years ago
										 |                     key: Data( | ||
| 
											3 years ago
										 |                         libSessionVal: profilePic.key, | ||
| 
											3 years ago
										 |                         count: ProfileManager.avatarAES256KeyByteLength | ||
|  |                     ), | ||
| 
											3 years ago
										 |                     fileName: nil | ||
|  |                 ) | ||
|  |             }(), | ||
|  |             sentTimestamp: latestConfigUpdateSentTimestamp, | ||
|  |             calledFromConfigHandling: true | ||
| 
											3 years ago
										 |         ) | ||
| 
											3 years ago
										 |          | ||
| 
											3 years ago
										 |         // Update the 'Note to Self' visibility and priority | ||
|  |         let threadInfo: PriorityVisibilityInfo? = try? SessionThread | ||
|  |             .filter(id: userPublicKey) | ||
|  |             .select(.id, .variant, .pinnedPriority, .shouldBeVisible) | ||
|  |             .asRequest(of: PriorityVisibilityInfo.self) | ||
|  |             .fetchOne(db) | ||
|  |         let targetPriority: Int32 = user_profile_get_nts_priority(conf) | ||
|  |          | ||
|  |         // Create the 'Note to Self' thread if it doesn't exist | ||
|  |         if let threadInfo: PriorityVisibilityInfo = threadInfo { | ||
|  |             let threadChanges: [ConfigColumnAssignment] = [ | ||
| 
											3 years ago
										 |                 ((threadInfo.shouldBeVisible == SessionUtil.shouldBeVisible(priority: targetPriority)) ? nil : | ||
|  |                     SessionThread.Columns.shouldBeVisible.set(to: SessionUtil.shouldBeVisible(priority: targetPriority)) | ||
| 
											3 years ago
										 |                 ), | ||
|  |                 (threadInfo.pinnedPriority == targetPriority ? nil : | ||
|  |                     SessionThread.Columns.pinnedPriority.set(to: targetPriority) | ||
|  |                 ) | ||
|  |             ].compactMap { $0 } | ||
|  |              | ||
|  |             if !threadChanges.isEmpty { | ||
|  |                 try SessionThread | ||
|  |                     .filter(id: userPublicKey) | ||
|  |                     .updateAll( // Handling a config update so don't use `updateAllAndConfig` | ||
|  |                         db, | ||
|  |                         threadChanges | ||
|  |                     ) | ||
|  |             } | ||
|  |         } | ||
|  |         else { | ||
|  |             try SessionThread | ||
|  |                 .fetchOrCreate( | ||
|  |                     db, | ||
|  |                     id: userPublicKey, | ||
|  |                     variant: .contact, | ||
| 
											3 years ago
										 |                     shouldBeVisible: SessionUtil.shouldBeVisible(priority: targetPriority) | ||
| 
											3 years ago
										 |                 ) | ||
|  |              | ||
|  |             try SessionThread | ||
|  |                 .filter(id: userPublicKey) | ||
|  |                 .updateAll( // Handling a config update so don't use `updateAllAndConfig` | ||
|  |                     db, | ||
|  |                     SessionThread.Columns.pinnedPriority.set(to: targetPriority) | ||
|  |                 ) | ||
| 
											3 years ago
										 |              | ||
|  |             // If the 'Note to Self' conversation is hidden then we should trigger the proper | ||
|  |             // `deleteOrLeave` behaviour (for 'Note to Self' this will leave the conversation | ||
|  |             // but remove the associated interactions) | ||
| 
											3 years ago
										 |             if !SessionUtil.shouldBeVisible(priority: targetPriority) { | ||
| 
											3 years ago
										 |                 try SessionThread | ||
|  |                     .deleteOrLeave( | ||
|  |                         db, | ||
|  |                         threadId: userPublicKey, | ||
|  |                         threadVariant: .contact, | ||
| 
											3 years ago
										 |                         groupLeaveType: .forced, | ||
| 
											3 years ago
										 |                         calledFromConfigHandling: true | ||
|  |                     ) | ||
|  |             } | ||
| 
											3 years ago
										 |         } | ||
|  |          | ||
| 
											3 years ago
										 |         // 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),    // Always trust the current user | ||
|  |                     Contact.Columns.isApproved.set(to: true), | ||
|  |                     Contact.Columns.didApproveMe.set(to: true) | ||
|  |                 ) | ||
|  |         } | ||
| 
											3 years ago
										 |     } | ||
|  |      | ||
| 
											3 years ago
										 |     // MARK: - Outgoing Changes | ||
|  |      | ||
|  |     static func update( | ||
| 
											3 years ago
										 |         profile: Profile, | ||
| 
											3 years ago
										 |         in conf: UnsafeMutablePointer<config_object>? | ||
| 
											3 years ago
										 |     ) throws { | ||
| 
											3 years ago
										 |         guard conf != nil else { throw SessionUtilError.nilConfigObject } | ||
|  |          | ||
|  |         // Update the name | ||
| 
											3 years ago
										 |         var updatedName: [CChar] = profile.name.cArray.nullTerminated() | ||
| 
											3 years ago
										 |         user_profile_set_name(conf, &updatedName) | ||
|  |          | ||
|  |         // Either assign the updated profile pic, or sent a blank profile pic (to remove the current one) | ||
|  |         var profilePic: user_profile_pic = user_profile_pic() | ||
|  |         profilePic.url = profile.profilePictureUrl.toLibSession() | ||
|  |         profilePic.key = profile.profileEncryptionKey.toLibSession() | ||
|  |         user_profile_set_pic(conf, profilePic) | ||
|  |     } | ||
|  |      | ||
| 
											3 years ago
										 |     static func updateNoteToSelf( | ||
| 
											3 years ago
										 |         priority: Int32, | ||
| 
											3 years ago
										 |         in conf: UnsafeMutablePointer<config_object>? | ||
| 
											3 years ago
										 |     ) throws { | ||
| 
											3 years ago
										 |         guard conf != nil else { throw SessionUtilError.nilConfigObject } | ||
| 
											3 years ago
										 |          | ||
| 
											3 years ago
										 |         user_profile_set_nts_priority(conf, priority) | ||
| 
											3 years ago
										 |     } | ||
|  | } |