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.
		
		
		
		
		
			
		
			
				
	
	
		
			192 lines
		
	
	
		
			7.8 KiB
		
	
	
	
		
			Swift
		
	
			
		
		
	
	
			192 lines
		
	
	
		
			7.8 KiB
		
	
	
	
		
			Swift
		
	
| //
 | |
| //  Copyright (c) 2018 Open Whisper Systems. All rights reserved.
 | |
| //
 | |
| 
 | |
| import Foundation
 | |
| import PromiseKit
 | |
| import SignalServiceKit
 | |
| 
 | |
| @objc
 | |
| public class ProfileFetcherJob: NSObject {
 | |
| 
 | |
|     let TAG = "[ProfileFetcherJob]"
 | |
| 
 | |
|     let networkManager: TSNetworkManager
 | |
|     let storageManager: TSStorageManager
 | |
| 
 | |
|     // This property is only accessed on the main queue.
 | |
|     static var fetchDateMap = [String: Date]()
 | |
| 
 | |
|     let ignoreThrottling: Bool
 | |
| 
 | |
|     @objc
 | |
|     public class func run(thread: TSThread, networkManager: TSNetworkManager) {
 | |
|         ProfileFetcherJob(networkManager: networkManager).run(recipientIds: thread.recipientIdentifiers)
 | |
|     }
 | |
| 
 | |
|     @objc
 | |
|     public class func run(recipientId: String, networkManager: TSNetworkManager, ignoreThrottling: Bool) {
 | |
|         ProfileFetcherJob(networkManager: networkManager, ignoreThrottling:ignoreThrottling).run(recipientIds: [recipientId])
 | |
|     }
 | |
| 
 | |
|     public init(networkManager: TSNetworkManager, ignoreThrottling: Bool = false) {
 | |
|         self.networkManager = networkManager
 | |
|         self.storageManager = TSStorageManager.shared()
 | |
|         self.ignoreThrottling = ignoreThrottling
 | |
|     }
 | |
| 
 | |
|     public func run(recipientIds: [String]) {
 | |
|         AssertIsOnMainThread()
 | |
| 
 | |
|         if (!CurrentAppContext().isMainApp) {
 | |
|             // Only refresh profiles in the MainApp to decrease the chance of missed SN notifications
 | |
|             // in the AppExtension for our users who choose not to verify contacts.
 | |
|             owsFail("Should only fetch profiles in the main app")
 | |
|             return
 | |
|         }
 | |
| 
 | |
|         DispatchQueue.main.async {
 | |
|             for recipientId in recipientIds {
 | |
|                 self.updateProfile(recipientId: recipientId)
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     enum ProfileFetcherJobError: Error {
 | |
|         case throttled(lastTimeInterval: TimeInterval),
 | |
|              unknownNetworkError
 | |
|     }
 | |
| 
 | |
|     public func updateProfile(recipientId: String, remainingRetries: Int = 3) {
 | |
|         self.getProfile(recipientId: recipientId).then { profile in
 | |
|             self.updateProfile(signalServiceProfile: profile)
 | |
|         }.catch { error in
 | |
|             switch error {
 | |
|             case ProfileFetcherJobError.throttled(let lastTimeInterval):
 | |
|                 Logger.info("\(self.TAG) skipping updateProfile: \(recipientId), lastTimeInterval: \(lastTimeInterval)")
 | |
|             case let error as SignalServiceProfile.ValidationError:
 | |
|                 Logger.warn("\(self.TAG) skipping updateProfile retry. Invalid profile for: \(recipientId) error: \(error)")
 | |
|             default:
 | |
|                 if remainingRetries > 0 {
 | |
|                     self.updateProfile(recipientId: recipientId, remainingRetries: remainingRetries - 1)
 | |
|                 } else {
 | |
|                     Logger.error("\(self.TAG) in \(#function) failed to get profile with error: \(error)")
 | |
|                 }
 | |
|             }
 | |
|         }.retainUntilComplete()
 | |
|     }
 | |
| 
 | |
|     public func getProfile(recipientId: String) -> Promise<SignalServiceProfile> {
 | |
|         AssertIsOnMainThread()
 | |
|         if !ignoreThrottling {
 | |
|             if let lastDate = ProfileFetcherJob.fetchDateMap[recipientId] {
 | |
|                 let lastTimeInterval = fabs(lastDate.timeIntervalSinceNow)
 | |
|                 // Don't check a profile more often than every N seconds.
 | |
|                 //
 | |
|                 // Throttle less in debug to make it easier to test problems
 | |
|                 // with our fetching logic.
 | |
|                 let kGetProfileMaxFrequencySeconds = _isDebugAssertConfiguration() ? 60 : 60.0 * 5.0
 | |
|                 guard lastTimeInterval > kGetProfileMaxFrequencySeconds else {
 | |
|                     return Promise(error: ProfileFetcherJobError.throttled(lastTimeInterval: lastTimeInterval))
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|         ProfileFetcherJob.fetchDateMap[recipientId] = Date()
 | |
| 
 | |
|         Logger.error("\(self.TAG) getProfile: \(recipientId)")
 | |
| 
 | |
|         let request = OWSGetProfileRequest(recipientId: recipientId)
 | |
| 
 | |
|         let (promise, fulfill, reject) = Promise<SignalServiceProfile>.pending()
 | |
| 
 | |
|         self.networkManager.makeRequest(
 | |
|             request,
 | |
|             success: { (_: URLSessionDataTask?, responseObject: Any?) -> Void in
 | |
|                 do {
 | |
|                     let profile = try SignalServiceProfile(recipientId: recipientId, rawResponse: responseObject)
 | |
|                     fulfill(profile)
 | |
|                 } catch {
 | |
|                     reject(error)
 | |
|                 }
 | |
|         },
 | |
|             failure: { (_: URLSessionDataTask?, error: Error?) in
 | |
| 
 | |
|                 if let error = error {
 | |
|                     reject(error)
 | |
|                 }
 | |
| 
 | |
|                 reject(ProfileFetcherJobError.unknownNetworkError)
 | |
|         })
 | |
| 
 | |
|         return promise
 | |
|     }
 | |
| 
 | |
|     private func updateProfile(signalServiceProfile: SignalServiceProfile) {
 | |
|         verifyIdentityUpToDateAsync(recipientId: signalServiceProfile.recipientId, latestIdentityKey: signalServiceProfile.identityKey)
 | |
| 
 | |
|         OWSProfileManager.shared().updateProfile(forRecipientId: signalServiceProfile.recipientId,
 | |
|                                                  profileNameEncrypted: signalServiceProfile.profileNameEncrypted,
 | |
|                                                  avatarUrlPath: signalServiceProfile.avatarUrlPath)
 | |
|     }
 | |
| 
 | |
|     private func verifyIdentityUpToDateAsync(recipientId: String, latestIdentityKey: Data) {
 | |
|         storageManager.newDatabaseConnection().asyncReadWrite { (transaction) in
 | |
|             if OWSIdentityManager.shared().saveRemoteIdentity(latestIdentityKey, recipientId: recipientId, protocolContext: transaction) {
 | |
|                 Logger.info("\(self.TAG) updated identity key with fetched profile for recipient: \(recipientId)")
 | |
|                 self.storageManager.archiveAllSessions(forContact: recipientId, protocolContext: transaction)
 | |
|             } else {
 | |
|                 // no change in identity.
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| @objc
 | |
| public class SignalServiceProfile: NSObject {
 | |
|     let TAG = "[SignalServiceProfile]"
 | |
| 
 | |
|     public enum ValidationError: Error {
 | |
|         case invalid(description: String)
 | |
|         case invalidIdentityKey(description: String)
 | |
|         case invalidProfileName(description: String)
 | |
|     }
 | |
| 
 | |
|     public let recipientId: String
 | |
|     public let identityKey: Data
 | |
|     public let profileNameEncrypted: Data?
 | |
|     public let avatarUrlPath: String?
 | |
| 
 | |
|     init(recipientId: String, rawResponse: Any?) throws {
 | |
|         self.recipientId = recipientId
 | |
| 
 | |
|         guard let responseDict = rawResponse as? [String: Any?] else {
 | |
|             throw ValidationError.invalid(description: "\(TAG) unexpected type: \(String(describing: rawResponse))")
 | |
|         }
 | |
| 
 | |
|         guard let identityKeyString = responseDict["identityKey"] as? String else {
 | |
|             throw ValidationError.invalidIdentityKey(description: "\(TAG) missing identity key: \(String(describing: rawResponse))")
 | |
|         }
 | |
|         guard let identityKeyWithType = Data(base64Encoded: identityKeyString) else {
 | |
|             throw ValidationError.invalidIdentityKey(description: "\(TAG) unable to parse identity key: \(identityKeyString)")
 | |
|         }
 | |
|         let kIdentityKeyLength = 33
 | |
|         guard identityKeyWithType.count == kIdentityKeyLength else {
 | |
|             throw ValidationError.invalidIdentityKey(description: "\(TAG) malformed key \(identityKeyString) with decoded length: \(identityKeyWithType.count)")
 | |
|         }
 | |
| 
 | |
|         if let profileNameString = responseDict["name"] as? String {
 | |
|             guard let data = Data(base64Encoded: profileNameString) else {
 | |
|                 throw ValidationError.invalidProfileName(description: "\(TAG) unable to parse profile name: \(profileNameString)")
 | |
|             }
 | |
|             self.profileNameEncrypted = data
 | |
|         } else {
 | |
|             self.profileNameEncrypted = nil
 | |
|         }
 | |
| 
 | |
|         self.avatarUrlPath = responseDict["avatar"] as? String
 | |
| 
 | |
|         // `removeKeyType` is an objc category method only on NSData, so temporarily cast.
 | |
|         self.identityKey = (identityKeyWithType as NSData).removeKeyType() as Data
 | |
|     }
 | |
| }
 |