|
|
|
//
|
|
|
|
// Copyright (c) 2018 Open Whisper Systems. All rights reserved.
|
|
|
|
//
|
|
|
|
|
|
|
|
import Foundation
|
|
|
|
import PromiseKit
|
|
|
|
import SignalServiceKit
|
|
|
|
|
|
|
|
@objc
|
|
|
|
public class ProfileFetcherJob: NSObject {
|
|
|
|
|
|
|
|
let networkManager: TSNetworkManager
|
|
|
|
let socketManager: TSSocketManager
|
|
|
|
let primaryStorage: OWSPrimaryStorage
|
|
|
|
|
|
|
|
// This property is only accessed on the main queue.
|
|
|
|
static var fetchDateMap = [String: Date]()
|
|
|
|
|
|
|
|
let ignoreThrottling: Bool
|
|
|
|
|
|
|
|
var backgroundTask: OWSBackgroundTask?
|
|
|
|
|
|
|
|
@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.socketManager = TSSocketManager.shared()
|
|
|
|
self.primaryStorage = OWSPrimaryStorage.shared()
|
|
|
|
self.ignoreThrottling = ignoreThrottling
|
|
|
|
}
|
|
|
|
|
|
|
|
public func run(recipientIds: [String]) {
|
|
|
|
AssertIsOnMainThread()
|
|
|
|
|
|
|
|
backgroundTask = OWSBackgroundTask(label: "\(#function)", completionBlock: { [weak self] status in
|
|
|
|
AssertIsOnMainThread()
|
|
|
|
|
|
|
|
guard status == .expired else {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
guard let _ = self else {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
Logger.error("background task time ran out before profile fetch completed.")
|
|
|
|
})
|
|
|
|
|
|
|
|
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("skipping updateProfile: \(recipientId), lastTimeInterval: \(lastTimeInterval)")
|
|
|
|
case let error as SignalServiceProfile.ValidationError:
|
|
|
|
Logger.warn("skipping updateProfile retry. Invalid profile for: \(recipientId) error: \(error)")
|
|
|
|
default:
|
|
|
|
if remainingRetries > 0 {
|
|
|
|
self.updateProfile(recipientId: recipientId, remainingRetries: remainingRetries - 1)
|
|
|
|
} else {
|
|
|
|
Logger.error("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("getProfile: \(recipientId)")
|
|
|
|
|
|
|
|
let request = OWSRequestFactory.getProfileRequest(withRecipientId: recipientId)
|
|
|
|
|
|
|
|
let (promise, fulfill, reject) = Promise<SignalServiceProfile>.pending()
|
|
|
|
|
|
|
|
if TSSocketManager.canMakeRequests() {
|
|
|
|
self.socketManager.make(request,
|
|
|
|
success: { (responseObject: Any?) -> Void in
|
|
|
|
do {
|
|
|
|
let profile = try SignalServiceProfile(recipientId: recipientId, rawResponse: responseObject)
|
|
|
|
fulfill(profile)
|
|
|
|
} catch {
|
|
|
|
reject(error)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
failure: { (_: NSInteger, _:Data?, error: Error) in
|
|
|
|
reject(error)
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
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) {
|
|
|
|
primaryStorage.newDatabaseConnection().asyncReadWrite { (transaction) in
|
|
|
|
if OWSIdentityManager.shared().saveRemoteIdentity(latestIdentityKey, recipientId: recipientId, protocolContext: transaction) {
|
|
|
|
Logger.info("updated identity key with fetched profile for recipient: \(recipientId)")
|
|
|
|
self.primaryStorage.archiveAllSessions(forContact: recipientId, protocolContext: transaction)
|
|
|
|
} else {
|
|
|
|
// no change in identity.
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@objc
|
|
|
|
public class SignalServiceProfile: NSObject {
|
|
|
|
|
|
|
|
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: "\(SignalServiceProfile.logTag()) unexpected type: \(String(describing: rawResponse))")
|
|
|
|
}
|
|
|
|
|
|
|
|
guard let identityKeyString = responseDict["identityKey"] as? String else {
|
|
|
|
throw ValidationError.invalidIdentityKey(description: "\(SignalServiceProfile.logTag()) missing identity key: \(String(describing: rawResponse))")
|
|
|
|
}
|
|
|
|
guard let identityKeyWithType = Data(base64Encoded: identityKeyString) else {
|
|
|
|
throw ValidationError.invalidIdentityKey(description: "\(SignalServiceProfile.logTag()) unable to parse identity key: \(identityKeyString)")
|
|
|
|
}
|
|
|
|
let kIdentityKeyLength = 33
|
|
|
|
guard identityKeyWithType.count == kIdentityKeyLength else {
|
|
|
|
throw ValidationError.invalidIdentityKey(description: "\(SignalServiceProfile.logTag()) 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: "\(SignalServiceProfile.logTag()) 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
|
|
|
|
}
|
|
|
|
}
|