|  |  |  | // | 
					
						
							|  |  |  | //  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; | 
					
						
							|  |  |  |         }         | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ([LKSyncMessagesProtocol shouldSkipConfigurationSyncMessage]) { return; } | 
					
						
							|  |  |  |         [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 | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     [LKSyncMessagesProtocol syncContactWithHexEncodedPublicKey:hexEncodedPubKey in:transaction]; | 
					
						
							|  |  |  |     return [AnyPromise promiseWithValue:@1]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | - (AnyPromise *)syncAllContacts | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return [LKSyncMessagesProtocol syncAllContacts]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | - (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 | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return [LKSyncMessagesProtocol syncAllClosedGroups]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | - (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 | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return [LKSyncMessagesProtocol syncAllOpenGroups]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | NS_ASSUME_NONNULL_END |