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.
		
		
		
		
		
			
		
			
				
	
	
		
			391 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			Objective-C
		
	
			
		
		
	
	
			391 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			Objective-C
		
	
| //
 | |
| //  Copyright (c) 2019 Open Whisper Systems. All rights reserved.
 | |
| //
 | |
| 
 | |
| #import "OWSSyncManager.h"
 | |
| #import "Environment.h"
 | |
| #import "OWSContactsManager.h"
 | |
| #import "OWSPreferences.h"
 | |
| #import "OWSProfileManager.h"
 | |
| #import "OWSReadReceiptManager.h"
 | |
| #import <PromiseKit/AnyPromise.h>
 | |
| #import <SignalServiceKit/AppReadiness.h>
 | |
| #import <SignalServiceKit/DataSource.h>
 | |
| #import <SignalServiceKit/MIMETypeUtil.h>
 | |
| #import <SignalServiceKit/OWSMessageSender.h>
 | |
| #import <SignalServiceKit/OWSPrimaryStorage.h>
 | |
| #import <SignalServiceKit/OWSSyncConfigurationMessage.h>
 | |
| #import <SignalServiceKit/OWSSyncContactsMessage.h>
 | |
| #import <SignalServiceKit/OWSSyncGroupsMessage.h>
 | |
| #import <SignalServiceKit/LKSyncOpenGroupsMessage.h>
 | |
| #import <SignalServiceKit/SSKEnvironment.h>
 | |
| #import <SignalServiceKit/SignalAccount.h>
 | |
| #import <SignalServiceKit/SignalServiceKit-Swift.h>
 | |
| #import <SignalServiceKit/TSAccountManager.h>
 | |
| #import <SignalServiceKit/YapDatabaseConnection+OWS.h>
 | |
| #import <SignalServiceKit/TSContactThread.h>
 | |
| #import <SignalServiceKit/TSGroupThread.h>
 | |
| 
 | |
| NS_ASSUME_NONNULL_BEGIN
 | |
| 
 | |
| NSString *const kSyncManagerCollection = @"kTSStorageManagerOWSSyncManagerCollection";
 | |
| NSString *const kSyncManagerLastContactSyncKey = @"kTSStorageManagerOWSSyncManagerLastMessageKey";
 | |
| 
 | |
| @interface OWSSyncManager ()
 | |
| 
 | |
| @property (nonatomic, readonly) dispatch_queue_t serialQueue;
 | |
| 
 | |
| @property (nonatomic) BOOL isRequestInFlight;
 | |
| 
 | |
| @end
 | |
| 
 | |
| @implementation OWSSyncManager
 | |
| 
 | |
| + (instancetype)shared {
 | |
|     OWSAssertDebug(SSKEnvironment.shared.syncManager);
 | |
| 
 | |
|     return SSKEnvironment.shared.syncManager;
 | |
| }
 | |
| 
 | |
| - (instancetype)initDefault {
 | |
|     self = [super init];
 | |
| 
 | |
|     if (!self) {
 | |
|         return self;
 | |
|     }
 | |
| 
 | |
|     OWSSingletonAssert();
 | |
| 
 | |
|     [[NSNotificationCenter defaultCenter] addObserver:self
 | |
|                                              selector:@selector(signalAccountsDidChange:)
 | |
|                                                  name:OWSContactsManagerSignalAccountsDidChangeNotification
 | |
|                                                object:nil];
 | |
|     [[NSNotificationCenter defaultCenter] addObserver:self
 | |
|                                              selector:@selector(profileKeyDidChange:)
 | |
|                                                  name:kNSNotificationName_ProfileKeyDidChange
 | |
|                                                object:nil];
 | |
| 
 | |
|     return self;
 | |
| }
 | |
| 
 | |
| - (void)dealloc {
 | |
|     [[NSNotificationCenter defaultCenter] removeObserver:self];
 | |
| }
 | |
| 
 | |
| #pragma mark - Dependencies
 | |
| 
 | |
| - (OWSContactsManager *)contactsManager {
 | |
|     OWSAssertDebug(Environment.shared.contactsManager);
 | |
| 
 | |
|     return Environment.shared.contactsManager;
 | |
| }
 | |
| 
 | |
| - (OWSIdentityManager *)identityManager {
 | |
|     OWSAssertDebug(SSKEnvironment.shared.identityManager);
 | |
| 
 | |
|     return SSKEnvironment.shared.identityManager;
 | |
| }
 | |
| 
 | |
| - (OWSMessageSender *)messageSender {
 | |
|     OWSAssertDebug(SSKEnvironment.shared.messageSender);
 | |
| 
 | |
|     return SSKEnvironment.shared.messageSender;
 | |
| }
 | |
| 
 | |
| - (SSKMessageSenderJobQueue *)messageSenderJobQueue
 | |
| {
 | |
|     OWSAssertDebug(SSKEnvironment.shared.messageSenderJobQueue);
 | |
| 
 | |
|     return SSKEnvironment.shared.messageSenderJobQueue;
 | |
| }
 | |
| 
 | |
| - (OWSProfileManager *)profileManager {
 | |
|     OWSAssertDebug(SSKEnvironment.shared.profileManager);
 | |
| 
 | |
|     return SSKEnvironment.shared.profileManager;
 | |
| }
 | |
| 
 | |
| - (TSAccountManager *)tsAccountManager
 | |
| {
 | |
|     return TSAccountManager.sharedInstance;
 | |
| }
 | |
| 
 | |
| - (id<OWSTypingIndicators>)typingIndicators
 | |
| {
 | |
|     return SSKEnvironment.shared.typingIndicators;
 | |
| }
 | |
| 
 | |
| #pragma mark - Notifications
 | |
| 
 | |
| - (void)signalAccountsDidChange:(id)notification {
 | |
|     OWSAssertIsOnMainThread();
 | |
| 
 | |
|     [self sendSyncContactsMessageIfPossible];
 | |
| }
 | |
| 
 | |
| - (void)profileKeyDidChange:(id)notification {
 | |
|     OWSAssertIsOnMainThread();
 | |
| 
 | |
|     [self sendSyncContactsMessageIfPossible];
 | |
| }
 | |
| 
 | |
| #pragma mark -
 | |
| 
 | |
| - (YapDatabaseConnection *)editingDatabaseConnection
 | |
| {
 | |
|     return OWSPrimaryStorage.sharedManager.dbReadWriteConnection;
 | |
| }
 | |
| 
 | |
| - (YapDatabaseConnection *)readDatabaseConnection
 | |
| {
 | |
|     return OWSPrimaryStorage.sharedManager.dbReadConnection;
 | |
| }
 | |
| 
 | |
| #pragma mark - Methods
 | |
| 
 | |
| - (void)sendSyncContactsMessageIfNecessary {
 | |
|     OWSAssertIsOnMainThread();
 | |
| 
 | |
|     if (!self.serialQueue) {
 | |
|         _serialQueue = dispatch_queue_create("org.whispersystems.contacts.syncing", DISPATCH_QUEUE_SERIAL);
 | |
|     }
 | |
| 
 | |
|     dispatch_async(self.serialQueue, ^{
 | |
|         if (self.isRequestInFlight) {
 | |
|             // De-bounce.  It's okay if we ignore some new changes;
 | |
|             // `sendSyncContactsMessageIfPossible` is called fairly
 | |
|             // often so we'll sync soon.
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         OWSSyncContactsMessage *syncContactsMessage =
 | |
|             [[OWSSyncContactsMessage alloc] initWithSignalAccounts:self.contactsManager.signalAccounts
 | |
|                                                    identityManager:self.identityManager
 | |
|                                                     profileManager:self.profileManager];
 | |
| 
 | |
|         __block NSData *_Nullable messageData;
 | |
|         __block NSData *_Nullable lastMessageData;
 | |
|         [self.readDatabaseConnection readWithBlock:^(YapDatabaseReadTransaction *transaction) {
 | |
|             messageData = [syncContactsMessage buildPlainTextAttachmentDataWithTransaction:transaction];
 | |
|             lastMessageData = [transaction objectForKey:kSyncManagerLastContactSyncKey
 | |
|                                            inCollection:kSyncManagerCollection];
 | |
|         }];
 | |
| 
 | |
|         if (!messageData) {
 | |
|             OWSFailDebug(@"Failed to serialize contacts sync message.");
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         if (lastMessageData && [lastMessageData isEqual:messageData]) {
 | |
|             // Ignore redundant contacts sync message.
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         self.isRequestInFlight = YES;
 | |
| 
 | |
|         // DURABLE CLEANUP - we could replace the custom durability logic in this class
 | |
|         // with a durable JobQueue.
 | |
|         DataSource *dataSource = [DataSourceValue dataSourceWithSyncMessageData:messageData];
 | |
|         [self.messageSender sendTemporaryAttachment:dataSource
 | |
|             contentType:OWSMimeTypeApplicationOctetStream
 | |
|             inMessage:syncContactsMessage
 | |
|             success:^{
 | |
|                 OWSLogInfo(@"Successfully sent contacts sync message.");
 | |
| 
 | |
|                 [self.editingDatabaseConnection setObject:messageData
 | |
|                                                    forKey:kSyncManagerLastContactSyncKey
 | |
|                                              inCollection:kSyncManagerCollection];
 | |
| 
 | |
|                 dispatch_async(self.serialQueue, ^{
 | |
|                     self.isRequestInFlight = NO;
 | |
|                 });
 | |
|             }
 | |
|             failure:^(NSError *error) {
 | |
|                 OWSLogError(@"Failed to send contacts sync message with error: %@", error);
 | |
| 
 | |
|                 dispatch_async(self.serialQueue, ^{
 | |
|                     self.isRequestInFlight = NO;
 | |
|                 });
 | |
|             }];
 | |
|     });
 | |
| }
 | |
| 
 | |
| - (void)sendSyncContactsMessageIfPossible {
 | |
|     OWSAssertIsOnMainThread();
 | |
| 
 | |
|     if (!self.contactsManager.isSetup) {
 | |
|         // Don't bother if the contacts manager hasn't finished setup.
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if ([TSAccountManager sharedInstance].isRegisteredAndReady) {
 | |
|         [self sendSyncContactsMessageIfNecessary];
 | |
|     }
 | |
| }
 | |
| 
 | |
| - (void)sendConfigurationSyncMessage {
 | |
|     [AppReadiness runNowOrWhenAppDidBecomeReady:^{
 | |
|         if (!self.tsAccountManager.isRegisteredAndReady) {
 | |
|             return;
 | |
|         }        
 | |
|         
 | |
|         NSUserDefaults *userDefaults = NSUserDefaults.standardUserDefaults;
 | |
|         BOOL hasLaunchedOnce = [userDefaults boolForKey:@"hasLaunchedOnce"];
 | |
|         if (hasLaunchedOnce) { // FIXME: Quick and dirty workaround to not do this on initial launch
 | |
|             [self sendConfigurationSyncMessage_AppReady];
 | |
|         }
 | |
|     }];
 | |
| }
 | |
| 
 | |
| - (void)sendConfigurationSyncMessage_AppReady {
 | |
|     DDLogInfo(@"");
 | |
| 
 | |
|     if (![TSAccountManager sharedInstance].isRegisteredAndReady) {
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     BOOL areReadReceiptsEnabled = SSKEnvironment.shared.readReceiptManager.areReadReceiptsEnabled;
 | |
|     BOOL showUnidentifiedDeliveryIndicators = Environment.shared.preferences.shouldShowUnidentifiedDeliveryIndicators;
 | |
|     BOOL showTypingIndicators = self.typingIndicators.areTypingIndicatorsEnabled;
 | |
|     BOOL sendLinkPreviews = SSKPreferences.areLinkPreviewsEnabled;
 | |
| 
 | |
|     OWSSyncConfigurationMessage *syncConfigurationMessage =
 | |
|         [[OWSSyncConfigurationMessage alloc] initWithReadReceiptsEnabled:areReadReceiptsEnabled
 | |
|                                       showUnidentifiedDeliveryIndicators:showUnidentifiedDeliveryIndicators
 | |
|                                                     showTypingIndicators:showTypingIndicators
 | |
|                                                         sendLinkPreviews:sendLinkPreviews];
 | |
| 
 | |
|     [self.editingDatabaseConnection readWriteWithBlock:^(YapDatabaseReadWriteTransaction *_Nonnull transaction) {
 | |
|         [self.messageSenderJobQueue addMessage:syncConfigurationMessage transaction:transaction];
 | |
|     }];
 | |
| }
 | |
| 
 | |
| #pragma mark - Local Sync
 | |
| 
 | |
| - (AnyPromise *)syncLocalContact
 | |
| {
 | |
|     NSString *localNumber = self.tsAccountManager.localNumber;
 | |
|     SignalAccount *signalAccount = [[SignalAccount alloc] initWithRecipientId:localNumber];
 | |
|     signalAccount.contact = [Contact new];
 | |
| 
 | |
|     return [self syncContactsForSignalAccounts:@[ signalAccount ]];
 | |
| }
 | |
| 
 | |
| - (AnyPromise *)syncContact:(NSString *)hexEncodedPubKey transaction:(YapDatabaseReadTransaction *)transaction
 | |
| {
 | |
|     TSContactThread *thread = [TSContactThread getThreadWithContactId:hexEncodedPubKey transaction:transaction];
 | |
|     if (thread != nil && thread.isContactFriend) {
 | |
|         return [self syncContactsForSignalAccounts:@[[[SignalAccount alloc] initWithRecipientId:hexEncodedPubKey]]];
 | |
|     }
 | |
|     return [AnyPromise promiseWithValue:@1];
 | |
| }
 | |
| 
 | |
| - (AnyPromise *)syncAllContacts
 | |
| {
 | |
|     NSMutableArray<SignalAccount *> *friends = @[].mutableCopy;
 | |
|     NSMutableArray<AnyPromise *> *promises = @[].mutableCopy;
 | |
|     [TSContactThread enumerateCollectionObjectsUsingBlock:^(TSContactThread *thread, BOOL *stop) {
 | |
|         NSString *hexEncodedPublicKey = thread.contactIdentifier;
 | |
|         if (hexEncodedPublicKey != nil && thread.isContactFriend && thread.shouldThreadBeVisible && !thread.isForceHidden) {
 | |
|             [friends addObject:[[SignalAccount alloc] initWithRecipientId:hexEncodedPublicKey]];
 | |
|         }
 | |
|     }];
 | |
|     [friends addObject:[[SignalAccount alloc] initWithRecipientId:self.tsAccountManager.localNumber]];
 | |
|     NSMutableArray<SignalAccount *> *signalAccounts = @[].mutableCopy;
 | |
|     for (SignalAccount *contact in friends) {
 | |
|         [signalAccounts addObject:contact];
 | |
|         if (signalAccounts.count >= 3) {
 | |
|             [promises addObject:[self syncContactsForSignalAccounts:[signalAccounts copy]]];
 | |
|             [signalAccounts removeAllObjects];
 | |
|         }
 | |
|     }
 | |
|     if (signalAccounts.count > 0) {
 | |
|         [promises addObject:[self syncContactsForSignalAccounts:signalAccounts]];
 | |
|     }
 | |
|     AnyPromise *promise = PMKJoin(promises);
 | |
|     [promise retainUntilComplete];
 | |
|     return promise;
 | |
| }
 | |
| 
 | |
| - (AnyPromise *)syncContactsForSignalAccounts:(NSArray<SignalAccount *> *)signalAccounts
 | |
| {
 | |
|     OWSSyncContactsMessage *syncContactsMessage = [[OWSSyncContactsMessage alloc] initWithSignalAccounts:signalAccounts identityManager:self.identityManager profileManager:self.profileManager];
 | |
|     AnyPromise *promise = [AnyPromise promiseWithResolverBlock:^(PMKResolver resolve) {
 | |
|         [self.messageSender sendMessage:syncContactsMessage
 | |
|             success:^{
 | |
|                 OWSLogInfo(@"Successfully sent contacts sync message.");
 | |
|                 resolve(@(1));
 | |
|             }
 | |
|             failure:^(NSError *error) {
 | |
|                 OWSLogError(@"Failed to send contacts sync message with error: %@.", error);
 | |
|                 resolve(error);
 | |
|             }];
 | |
|     }];
 | |
|     [promise retainUntilComplete];
 | |
|     return promise;
 | |
| }
 | |
| 
 | |
| - (AnyPromise *)syncAllGroups
 | |
| {
 | |
|     NSMutableArray<TSGroupThread *> *groupThreads = @[].mutableCopy;
 | |
|     NSMutableArray<AnyPromise *> *promises = @[].mutableCopy;
 | |
|     [TSGroupThread enumerateCollectionObjectsUsingBlock:^(id obj, BOOL *stop) {
 | |
|         if (![obj isKindOfClass:[TSGroupThread class]]) {
 | |
|             if (![obj isKindOfClass:[TSContactThread class]]) { // FIXME: Isn't this redundant?
 | |
|                 OWSLogWarn(@"Ignoring non-group thread in thread collection: %@.", obj);
 | |
|             }
 | |
|             return;
 | |
|         }
 | |
|         TSGroupThread *thread = (TSGroupThread *)obj;
 | |
|         if (thread.groupModel.groupType == closedGroup && thread.shouldThreadBeVisible && !thread.isForceHidden) {
 | |
|             [groupThreads addObject:thread];
 | |
|         }
 | |
|     }];
 | |
|     for (TSGroupThread *groupThread in groupThreads) {
 | |
|         [promises addObject:[self syncGroupForThread:groupThread]];
 | |
|     }
 | |
|     AnyPromise *promise = PMKJoin(promises);
 | |
|     [promise retainUntilComplete];
 | |
|     return promise;
 | |
| }
 | |
| 
 | |
| - (AnyPromise *)syncGroupForThread:(TSGroupThread *)thread
 | |
| {
 | |
|     OWSSyncGroupsMessage *syncGroupsMessage = [[OWSSyncGroupsMessage alloc] initWithGroupThread:thread];
 | |
|     AnyPromise *promise = [AnyPromise promiseWithResolverBlock:^(PMKResolver resolve) {
 | |
|         [self.messageSender sendMessage:syncGroupsMessage
 | |
|             success:^{
 | |
|                 OWSLogInfo(@"Successfully sent group sync message.");
 | |
|                 resolve(@(1));
 | |
|             }
 | |
|             failure:^(NSError *error) {
 | |
|                 OWSLogError(@"Failed to send group sync message due to error: %@.", error);
 | |
|                 resolve(error);
 | |
|             }];
 | |
|     }];
 | |
|     [promise retainUntilComplete];
 | |
|     return promise;
 | |
| }
 | |
| 
 | |
| - (AnyPromise *)syncAllOpenGroups
 | |
| {
 | |
|     LKSyncOpenGroupsMessage *syncOpenGroupsMessage = [[LKSyncOpenGroupsMessage alloc] init];
 | |
|     AnyPromise *promise = [AnyPromise promiseWithResolverBlock:^(PMKResolver resolve) {
 | |
|         [self.messageSender sendMessage:syncOpenGroupsMessage
 | |
|             success:^{
 | |
|                 OWSLogInfo(@"Successfully sent open group sync message.");
 | |
|                 resolve(@(1));
 | |
|             }
 | |
|             failure:^(NSError *error) {
 | |
|                 OWSLogError(@"Failed to send open group sync message due to error: %@.", error);
 | |
|                 resolve(error);
 | |
|             }];
 | |
|     }];
 | |
|     [promise retainUntilComplete];
 | |
|     return promise;
 | |
| }
 | |
| 
 | |
| @end
 | |
| 
 | |
| NS_ASSUME_NONNULL_END
 |