Fetch profiles from profile manager. Update profile manager with profile fetch results.

// FREEBIE
pull/1/head
Matthew Chen 8 years ago
parent 6ec756de44
commit 9c0f94f1c0

@ -59,6 +59,11 @@ extern NSString *const kNSNotificationName_OtherUsersProfileDidChange;
- (void)refreshProfileForRecipientId:(NSString *)recipientId; - (void)refreshProfileForRecipientId:(NSString *)recipientId;
- (void)updateProfileForRecipientId:(NSString *)recipientId
profileNameEncrypted:(NSData *_Nullable)profileNameEncrypted
avatarUrlEncrypted:(NSData *_Nullable)avatarUrlEncrypted
avatarDigestEncrypted:(NSData *_Nullable)avatarDigestEncrypted;
@end @end
NS_ASSUME_NONNULL_END NS_ASSUME_NONNULL_END

@ -4,6 +4,7 @@
#import "OWSProfileManager.h" #import "OWSProfileManager.h"
#import "Environment.h" #import "Environment.h"
#import "Signal-Swift.h"
#import <SignalServiceKit/NSData+hexString.h> #import <SignalServiceKit/NSData+hexString.h>
#import <SignalServiceKit/NSDate+OWS.h> #import <SignalServiceKit/NSDate+OWS.h>
#import <SignalServiceKit/OWSMessageSender.h> #import <SignalServiceKit/OWSMessageSender.h>
@ -27,7 +28,7 @@ NS_ASSUME_NONNULL_BEGIN
// These properties may be accessed only from the main thread. // These properties may be accessed only from the main thread.
@property (nonatomic, nullable) NSString *profileName; @property (nonatomic, nullable) NSString *profileName;
@property (nonatomic, nullable) NSString *avatarUrl; @property (nonatomic, nullable) NSString *avatarUrl;
@property (nonatomic, nullable) NSString *avatarDigest; @property (nonatomic, nullable) NSData *avatarDigest;
// This filename is relative to OWSProfileManager.profileAvatarsDirPath. // This filename is relative to OWSProfileManager.profileAvatarsDirPath.
@property (nonatomic, nullable) NSString *avatarFileName; @property (nonatomic, nullable) NSString *avatarFileName;
@ -68,8 +69,7 @@ NS_ASSUME_NONNULL_BEGIN
{ {
return ([other isKindOfClass:[UserProfile class]] && [self.recipientId isEqualToString:other.recipientId] && return ([other isKindOfClass:[UserProfile class]] && [self.recipientId isEqualToString:other.recipientId] &&
[self.profileName isEqualToString:other.profileName] && [self.avatarUrl isEqualToString:other.avatarUrl] && [self.profileName isEqualToString:other.profileName] && [self.avatarUrl isEqualToString:other.avatarUrl] &&
[self.avatarDigest isEqualToString:other.avatarDigest] && [self.avatarDigest isEqual:other.avatarDigest] && [self.avatarFileName isEqualToString:other.avatarFileName]);
[self.avatarFileName isEqualToString:other.avatarFileName]);
} }
- (NSUInteger)hash - (NSUInteger)hash
@ -97,6 +97,7 @@ static const NSInteger kProfileKeyLength = 16;
@property (nonatomic, readonly) OWSMessageSender *messageSender; @property (nonatomic, readonly) OWSMessageSender *messageSender;
@property (nonatomic, readonly) YapDatabaseConnection *dbConnection; @property (nonatomic, readonly) YapDatabaseConnection *dbConnection;
@property (nonatomic, readonly) TSNetworkManager *networkManager;
@property (atomic, nullable) UserProfile *localUserProfile; @property (atomic, nullable) UserProfile *localUserProfile;
// This property should only be mutated on the main thread, // This property should only be mutated on the main thread,
@ -131,12 +132,14 @@ static const NSInteger kProfileKeyLength = 16;
{ {
TSStorageManager *storageManager = [TSStorageManager sharedManager]; TSStorageManager *storageManager = [TSStorageManager sharedManager];
OWSMessageSender *messageSender = [Environment getCurrent].messageSender; OWSMessageSender *messageSender = [Environment getCurrent].messageSender;
TSNetworkManager *networkManager = [Environment getCurrent].networkManager;
return [self initWithStorageManager:storageManager messageSender:messageSender]; return [self initWithStorageManager:storageManager messageSender:messageSender networkManager:networkManager];
} }
- (instancetype)initWithStorageManager:(TSStorageManager *)storageManager - (instancetype)initWithStorageManager:(TSStorageManager *)storageManager
messageSender:(OWSMessageSender *)messageSender messageSender:(OWSMessageSender *)messageSender
networkManager:(TSNetworkManager *)networkManager
{ {
self = [super init]; self = [super init];
@ -147,9 +150,12 @@ static const NSInteger kProfileKeyLength = 16;
OWSAssert([NSThread isMainThread]); OWSAssert([NSThread isMainThread]);
OWSAssert(storageManager); OWSAssert(storageManager);
OWSAssert(messageSender); OWSAssert(messageSender);
OWSAssert(messageSender);
_messageSender = messageSender; _messageSender = messageSender;
_dbConnection = storageManager.newDatabaseConnection; _dbConnection = storageManager.newDatabaseConnection;
_networkManager = networkManager;
_userProfileWhitelistCache = [NSMutableDictionary new]; _userProfileWhitelistCache = [NSMutableDictionary new];
_groupProfileWhitelistCache = [NSMutableDictionary new]; _groupProfileWhitelistCache = [NSMutableDictionary new];
_otherUsersProfileAvatarImageCache = [NSCache new]; _otherUsersProfileAvatarImageCache = [NSCache new];
@ -217,6 +223,16 @@ static const NSInteger kProfileKeyLength = 16;
[self.dbConnection readWriteWithBlock:^(YapDatabaseReadWriteTransaction *transaction) { [self.dbConnection readWriteWithBlock:^(YapDatabaseReadWriteTransaction *transaction) {
[userProfile saveWithTransaction:transaction]; [userProfile saveWithTransaction:transaction];
}]; }];
if (userProfile == self.localUserProfile) {
[[NSNotificationCenter defaultCenter] postNotificationName:kNSNotificationName_LocalProfileDidChange
object:nil
userInfo:nil];
} else {
[[NSNotificationCenter defaultCenter] postNotificationName:kNSNotificationName_OtherUsersProfileDidChange
object:nil
userInfo:nil];
}
} }
#pragma mark - Local Profile Key #pragma mark - Local Profile Key
@ -271,8 +287,8 @@ static const NSInteger kProfileKeyLength = 16;
// //
// * Try to update the service. // * Try to update the service.
// * Update client state on success. // * Update client state on success.
void (^tryToUpdateService)(NSString *_Nullable, NSString *_Nullable, NSString *_Nullable) = ^( void (^tryToUpdateService)(NSString *_Nullable, NSData *_Nullable, NSString *_Nullable) = ^(
NSString *_Nullable avatarUrl, NSString *_Nullable avatarDigest, NSString *_Nullable avatarFileName) { NSString *_Nullable avatarUrl, NSData *_Nullable avatarDigest, NSString *_Nullable avatarFileName) {
[self updateProfileOnService:profileName [self updateProfileOnService:profileName
avatarUrl:avatarUrl avatarUrl:avatarUrl
avatarDigest:avatarDigest avatarDigest:avatarDigest
@ -291,10 +307,6 @@ static const NSInteger kProfileKeyLength = 16;
self.localCachedAvatarImage = avatarImage; self.localCachedAvatarImage = avatarImage;
successBlock(); successBlock();
[[NSNotificationCenter defaultCenter] postNotificationName:kNSNotificationName_LocalProfileDidChange
object:nil
userInfo:nil];
}); });
} }
failure:^{ failure:^{
@ -325,7 +337,7 @@ static const NSInteger kProfileKeyLength = 16;
success:^(NSData *data, NSString *fileName) { success:^(NSData *data, NSString *fileName) {
[self uploadAvatarToService:data [self uploadAvatarToService:data
fileName:fileName fileName:fileName
success:^(NSString *avatarUrl, NSString *avatarDigest) { success:^(NSString *avatarUrl, NSData *avatarDigest) {
tryToUpdateService(avatarUrl, avatarDigest, fileName); tryToUpdateService(avatarUrl, avatarDigest, fileName);
} }
failure:^{ failure:^{
@ -372,7 +384,7 @@ static const NSInteger kProfileKeyLength = 16;
// TODO: The exact API & encryption scheme for avatars is not yet settled. // TODO: The exact API & encryption scheme for avatars is not yet settled.
- (void)uploadAvatarToService:(NSData *)data - (void)uploadAvatarToService:(NSData *)data
fileName:(NSString *)fileName fileName:(NSString *)fileName
success:(void (^)(NSString *avatarUrl, NSString *avatarDigest))successBlock success:(void (^)(NSString *avatarUrl, NSData *avatarDigest))successBlock
failure:(void (^)())failureBlock failure:(void (^)())failureBlock
{ {
OWSAssert(data.length > 0); OWSAssert(data.length > 0);
@ -383,7 +395,7 @@ static const NSInteger kProfileKeyLength = 16;
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{ dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
// TODO: // TODO:
NSString *avatarUrl = @"avatarUrl"; NSString *avatarUrl = @"avatarUrl";
NSString *avatarDigest = @"avatarDigest"; NSData *avatarDigest = [@"avatarDigest" dataUsingEncoding:NSUTF8StringEncoding];
if (YES) { if (YES) {
successBlock(avatarUrl, avatarDigest); successBlock(avatarUrl, avatarDigest);
return; return;
@ -395,7 +407,7 @@ static const NSInteger kProfileKeyLength = 16;
// TODO: The exact API & encryption scheme for profiles is not yet settled. // TODO: The exact API & encryption scheme for profiles is not yet settled.
- (void)updateProfileOnService:(nullable NSString *)localProfileName - (void)updateProfileOnService:(nullable NSString *)localProfileName
avatarUrl:(nullable NSString *)avatarUrl avatarUrl:(nullable NSString *)avatarUrl
avatarDigest:(nullable NSString *)avatarDigest avatarDigest:(nullable NSData *)avatarDigest
success:(void (^)())successBlock success:(void (^)())successBlock
failure:(void (^)())failureBlock failure:(void (^)())failureBlock
{ {
@ -403,6 +415,12 @@ static const NSInteger kProfileKeyLength = 16;
OWSAssert(failureBlock); OWSAssert(failureBlock);
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{ dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
// TODO: Do we need to use NSDataBase64EncodingOptions?
NSString *_Nullable localProfileNameEncrypted =
[[self encryptProfileString:localProfileName] base64EncodedString];
NSString *_Nullable avatarUrlEncrypted = [[self encryptProfileString:avatarUrl] base64EncodedString];
NSString *_Nullable avatarDigestEncrypted = [[self encryptProfileData:avatarDigest] base64EncodedString];
// TODO: // TODO:
if (YES) { if (YES) {
successBlock(); successBlock();
@ -534,6 +552,8 @@ static const NSInteger kProfileKeyLength = 16;
userProfile.profileKey = profileKey; userProfile.profileKey = profileKey;
[self saveUserProfile:userProfile]; [self saveUserProfile:userProfile];
[self refreshProfileForRecipientId:recipientId ignoreThrottling:YES];
}); });
} }
@ -575,18 +595,40 @@ static const NSInteger kProfileKeyLength = 16;
if (image) { if (image) {
[self.otherUsersProfileAvatarImageCache setObject:image forKey:recipientId]; [self.otherUsersProfileAvatarImageCache setObject:image forKey:recipientId];
} }
} else if (userProfile.avatarUrl) {
[self downloadProfileAvatarWithUrl:userProfile.avatarUrl recipientId:recipientId];
} }
return image; return image;
} }
- (void)downloadProfileAvatarWithUrl:(NSString *)avatarUrl recipientId:(NSString *)recipientId
{
OWSAssert(avatarUrl.length > 0);
OWSAssert(recipientId.length > 0);
// TODO:
}
- (void)refreshProfileForRecipientId:(NSString *)recipientId - (void)refreshProfileForRecipientId:(NSString *)recipientId
{
[self refreshProfileForRecipientId:recipientId ignoreThrottling:NO];
}
- (void)refreshProfileForRecipientId:(NSString *)recipientId ignoreThrottling:(BOOL)ignoreThrottling
{ {
OWSAssert([NSThread isMainThread]); OWSAssert([NSThread isMainThread]);
OWSAssert(recipientId.length > 0); OWSAssert(recipientId.length > 0);
UserProfile *userProfile = [self getOrCreateUserProfileForRecipientId:recipientId]; UserProfile *userProfile = [self getOrCreateUserProfileForRecipientId:recipientId];
if (!userProfile.profileKey) {
// There's no point in fetching the profile for a user
// if we don't have their profile key; we won't be able
// to decrypt it.
return;
}
// Throttle and debounce the updates. // Throttle and debounce the updates.
const NSTimeInterval kMaxRefreshFrequency = 5 * kMinuteInterval; const NSTimeInterval kMaxRefreshFrequency = 5 * kMinuteInterval;
if (userProfile.lastUpdateDate && fabs([userProfile.lastUpdateDate timeIntervalSinceNow]) < kMaxRefreshFrequency) { if (userProfile.lastUpdateDate && fabs([userProfile.lastUpdateDate timeIntervalSinceNow]) < kMaxRefreshFrequency) {
@ -598,7 +640,153 @@ static const NSInteger kProfileKeyLength = 16;
[self saveUserProfile:userProfile]; [self saveUserProfile:userProfile];
// TODO: Actually update the profile. [ProfileFetcherJob runWithRecipientId:recipientId
networkManager:self.networkManager
ignoreThrottling:ignoreThrottling];
}
- (void)updateProfileForRecipientId:(NSString *)recipientId
profileNameEncrypted:(NSData *_Nullable)profileNameEncrypted
avatarUrlEncrypted:(NSData *_Nullable)avatarUrlEncrypted
avatarDigestEncrypted:(NSData *_Nullable)avatarDigestEncrypted
{
OWSAssert(recipientId.length > 0);
UserProfile *userProfile = [self getOrCreateUserProfileForRecipientId:recipientId];
if (!userProfile.profileKey) {
return;
}
NSString *_Nullable profileName =
[self decryptProfileString:profileNameEncrypted profileKey:userProfile.profileKey];
NSString *_Nullable avatarUrl = [self decryptProfileString:avatarUrlEncrypted profileKey:userProfile.profileKey];
NSData *_Nullable avatarDigest = [self decryptProfileData:avatarDigestEncrypted profileKey:userProfile.profileKey];
if (!avatarUrl || !avatarDigest) {
// If either avatar url or digest is missing, skip both.
avatarUrl = nil;
avatarDigest = nil;
}
BOOL isProfileNameSame = [self isNullableStringEqual:userProfile.profileName toString:profileName];
BOOL isAvatarSame = ([self isNullableStringEqual:userProfile.avatarUrl toString:avatarUrl] &&
[self isNullableDataEqual:userProfile.avatarDigest toData:avatarDigest]);
if (isProfileNameSame && isAvatarSame) {
return;
}
dispatch_async(dispatch_get_main_queue(), ^{
userProfile.profileName = profileName;
userProfile.avatarUrl = avatarUrl;
userProfile.avatarDigest = avatarDigest;
if (!isAvatarSame) {
// Evacuate avatar image cache.
[self.otherUsersProfileAvatarImageCache removeObjectForKey:recipientId];
}
if (avatarUrl) {
[self downloadProfileAvatarWithUrl:avatarUrl recipientId:recipientId];
}
[self saveUserProfile:userProfile];
});
}
- (BOOL)isNullableDataEqual:(NSData *_Nullable)left toData:(NSData *_Nullable)right
{
if (left == nil && right == nil) {
return YES;
} else if (left == nil || right == nil) {
return YES;
} else {
return [left isEqual:right];
}
}
- (BOOL)isNullableStringEqual:(NSString *_Nullable)left toString:(NSString *_Nullable)right
{
if (left == nil && right == nil) {
return YES;
} else if (left == nil || right == nil) {
return YES;
} else {
return [left isEqualToString:right];
}
}
#pragma mark - Profile Encryption
+ (NSData *_Nullable)decryptProfileData:(NSData *_Nullable)encryptedData profileKey:(NSData *)profileKey
{
OWSAssert(profileKey.length == kProfileKeyLength);
if (!encryptedData) {
return nil;
}
// TODO: Decrypt.
return nil;
}
+ (NSString *_Nullable)decryptProfileString:(NSData *_Nullable)encryptedData profileKey:(NSData *)profileKey
{
OWSAssert(profileKey.length == kProfileKeyLength);
NSData *_Nullable decryptedData = [self decryptProfileData:encryptedData profileKey:profileKey];
if (decryptedData) {
return [[NSString alloc] initWithData:decryptedData encoding:NSUTF8StringEncoding];
} else {
return nil;
}
}
+ (NSData *_Nullable)encryptProfileData:(NSData *_Nullable)data profileKey:(NSData *)profileKey
{
OWSAssert(profileKey.length == kProfileKeyLength);
if (!data) {
return nil;
}
// TODO: Encrypt.
return nil;
}
+ (NSData *_Nullable)encryptProfileString:(NSString *_Nullable)value profileKey:(NSData *)profileKey
{
OWSAssert(profileKey.length == kProfileKeyLength);
if (value) {
NSData *_Nullable data = [value dataUsingEncoding:NSUTF8StringEncoding];
if (data) {
NSData *_Nullable encryptedData = [self encryptProfileData:data profileKey:profileKey];
return encryptedData;
}
}
return nil;
}
- (NSData *_Nullable)decryptProfileData:(NSData *_Nullable)encryptedData profileKey:(NSData *)profileKey
{
return [OWSProfileManager decryptProfileData:encryptedData profileKey:profileKey];
}
- (NSString *_Nullable)decryptProfileString:(NSData *_Nullable)encryptedData profileKey:(NSData *)profileKey
{
return [OWSProfileManager decryptProfileString:encryptedData profileKey:profileKey];
}
- (NSData *_Nullable)encryptProfileData:(NSData *_Nullable)data
{
return [OWSProfileManager encryptProfileData:data profileKey:self.localProfileKey];
}
- (NSData *_Nullable)encryptProfileString:(NSString *_Nullable)value
{
return [OWSProfileManager encryptProfileString:value profileKey:self.localProfileKey];
} }
#pragma mark - Avatar Disk Cache #pragma mark - Avatar Disk Cache

@ -16,17 +16,20 @@ class ProfileFetcherJob: NSObject {
// This property is only accessed on the main queue. // This property is only accessed on the main queue.
static var fetchDateMap = [String: Date]() static var fetchDateMap = [String: Date]()
let ignoreThrottling: Bool
public class func run(thread: TSThread, networkManager: TSNetworkManager) { public class func run(thread: TSThread, networkManager: TSNetworkManager) {
ProfileFetcherJob(networkManager: networkManager).run(recipientIds: thread.recipientIdentifiers) ProfileFetcherJob(networkManager: networkManager).run(recipientIds: thread.recipientIdentifiers)
} }
public class func run(recipientId: String, networkManager: TSNetworkManager) { public class func run(recipientId: String, networkManager: TSNetworkManager, ignoreThrottling: Bool) {
ProfileFetcherJob(networkManager: networkManager).run(recipientIds: [recipientId]) ProfileFetcherJob(networkManager: networkManager, ignoreThrottling:ignoreThrottling).run(recipientIds: [recipientId])
} }
init(networkManager: TSNetworkManager) { init(networkManager: TSNetworkManager, ignoreThrottling: Bool = false) {
self.networkManager = networkManager self.networkManager = networkManager
self.storageManager = TSStorageManager.shared() self.storageManager = TSStorageManager.shared()
self.ignoreThrottling = ignoreThrottling
} }
public func run(recipientIds: [String]) { public func run(recipientIds: [String]) {
@ -65,15 +68,17 @@ class ProfileFetcherJob: NSObject {
public func getProfile(recipientId: String) -> Promise<SignalServiceProfile> { public func getProfile(recipientId: String) -> Promise<SignalServiceProfile> {
AssertIsOnMainThread() AssertIsOnMainThread()
if let lastDate = ProfileFetcherJob.fetchDateMap[recipientId] { if !ignoreThrottling {
let lastTimeInterval = fabs(lastDate.timeIntervalSinceNow) if let lastDate = ProfileFetcherJob.fetchDateMap[recipientId] {
// Don't check a profile more often than every N minutes. let lastTimeInterval = fabs(lastDate.timeIntervalSinceNow)
// // Don't check a profile more often than every N minutes.
// Only throttle profile fetch in production builds in order to //
// facilitate debugging. // Only throttle profile fetch in production builds in order to
let kGetProfileMaxFrequencySeconds = _isDebugAssertConfiguration() ? 0 : 60.0 * 5.0 // facilitate debugging.
guard lastTimeInterval > kGetProfileMaxFrequencySeconds else { let kGetProfileMaxFrequencySeconds = _isDebugAssertConfiguration() ? 0 : 60.0 * 5.0
return Promise(error: ProfileFetcherJobError.throttled(lastTimeInterval: lastTimeInterval)) guard lastTimeInterval > kGetProfileMaxFrequencySeconds else {
return Promise(error: ProfileFetcherJobError.throttled(lastTimeInterval: lastTimeInterval))
}
} }
} }
ProfileFetcherJob.fetchDateMap[recipientId] = Date() ProfileFetcherJob.fetchDateMap[recipientId] = Date()
@ -109,7 +114,10 @@ class ProfileFetcherJob: NSObject {
private func updateProfile(signalServiceProfile: SignalServiceProfile) { private func updateProfile(signalServiceProfile: SignalServiceProfile) {
verifyIdentityUpToDateAsync(recipientId: signalServiceProfile.recipientId, latestIdentityKey: signalServiceProfile.identityKey) verifyIdentityUpToDateAsync(recipientId: signalServiceProfile.recipientId, latestIdentityKey: signalServiceProfile.identityKey)
// Eventually we'll want to do more things with new SignalServiceProfile fields here. OWSProfileManager.shared().updateProfile(forRecipientId : signalServiceProfile.recipientId,
profileNameEncrypted : signalServiceProfile.profileNameEncrypted,
avatarUrlEncrypted : signalServiceProfile.avatarUrlEncrypted,
avatarDigestEncrypted : signalServiceProfile.avatarDigestEncrypted)
} }
private func verifyIdentityUpToDateAsync(recipientId: String, latestIdentityKey: Data) { private func verifyIdentityUpToDateAsync(recipientId: String, latestIdentityKey: Data) {
@ -130,14 +138,22 @@ struct SignalServiceProfile {
enum ValidationError: Error { enum ValidationError: Error {
case invalid(description: String) case invalid(description: String)
case invalidIdentityKey(description: String) case invalidIdentityKey(description: String)
case invalidProfileName(description: String)
case invalidAvatarUrl(description: String)
case invalidAvatarDigest(description: String)
} }
public let recipientId: String public let recipientId: String
public let identityKey: Data public let identityKey: Data
public let profileNameEncrypted: Data?
public let avatarUrlEncrypted: Data?
public let avatarDigestEncrypted: Data?
init(recipientId: String, rawResponse: Any?) throws { init(recipientId: String, rawResponse: Any?) throws {
self.recipientId = recipientId self.recipientId = recipientId
Logger.info("rawResponse: \(rawResponse)")
guard let responseDict = rawResponse as? [String: Any?] else { guard let responseDict = rawResponse as? [String: Any?] else {
throw ValidationError.invalid(description: "\(TAG) unexpected type: \(String(describing: rawResponse))") throw ValidationError.invalid(description: "\(TAG) unexpected type: \(String(describing: rawResponse))")
} }
@ -145,17 +161,42 @@ struct SignalServiceProfile {
guard let identityKeyString = responseDict["identityKey"] as? String else { guard let identityKeyString = responseDict["identityKey"] as? String else {
throw ValidationError.invalidIdentityKey(description: "\(TAG) missing identity key: \(String(describing: rawResponse))") throw ValidationError.invalidIdentityKey(description: "\(TAG) missing identity key: \(String(describing: rawResponse))")
} }
guard let identityKeyWithType = Data(base64Encoded: identityKeyString) else { guard let identityKeyWithType = Data(base64Encoded: identityKeyString) else {
throw ValidationError.invalidIdentityKey(description: "\(TAG) unable to parse identity key: \(identityKeyString)") throw ValidationError.invalidIdentityKey(description: "\(TAG) unable to parse identity key: \(identityKeyString)")
} }
let kIdentityKeyLength = 33 let kIdentityKeyLength = 33
guard identityKeyWithType.count == kIdentityKeyLength else { guard identityKeyWithType.count == kIdentityKeyLength else {
throw ValidationError.invalidIdentityKey(description: "\(TAG) malformed key \(identityKeyString) with decoded length: \(identityKeyWithType.count)") throw ValidationError.invalidIdentityKey(description: "\(TAG) malformed key \(identityKeyString) with decoded length: \(identityKeyWithType.count)")
} }
var profileNameEncrypted: Data? = nil
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)")
}
profileNameEncrypted = data
}
var avatarUrlEncrypted: Data? = nil
if let avatarUrlString = responseDict["avatar"] as? String {
guard let data = Data(base64Encoded: avatarUrlString) else {
throw ValidationError.invalidAvatarUrl(description: "\(TAG) unable to parse avatar URL: \(avatarUrlString)")
}
avatarUrlEncrypted = data
}
var avatarDigestEncrypted: Data? = nil
if let avatarDigestString = responseDict["avatarDigest"] as? String {
guard let data = Data(base64Encoded: avatarDigestString) else {
throw ValidationError.invalidAvatarDigest(description: "\(TAG) unable to parse avatar digest: \(avatarDigestString)")
}
avatarDigestEncrypted = data
}
// `removeKeyType` is an objc category method only on NSData, so temporarily cast. // `removeKeyType` is an objc category method only on NSData, so temporarily cast.
self.identityKey = (identityKeyWithType as NSData).removeKeyType() as Data self.identityKey = (identityKeyWithType as NSData).removeKeyType() as Data
self.profileNameEncrypted = profileNameEncrypted
self.avatarUrlEncrypted = avatarUrlEncrypted
self.avatarDigestEncrypted = avatarDigestEncrypted
} }
} }

@ -18,6 +18,7 @@
#import "OWSDatabaseMigration.h" #import "OWSDatabaseMigration.h"
#import "OWSLogger.h" #import "OWSLogger.h"
#import "OWSMessageEditing.h" #import "OWSMessageEditing.h"
#import "OWSProfileManager.h"
#import "OWSProgressView.h" #import "OWSProgressView.h"
#import "OWSViewController.h" #import "OWSViewController.h"
#import "OWSWebRTCDataProtos.pb.h" #import "OWSWebRTCDataProtos.pb.h"

Loading…
Cancel
Save