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.
		
		
		
		
		
			
		
			
				
	
	
		
			337 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Objective-C
		
	
			
		
		
	
	
			337 lines
		
	
	
		
			11 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 <SessionServiceKit/AppReadiness.h>
 | 
						|
#import <SessionServiceKit/DataSource.h>
 | 
						|
#import <SessionServiceKit/MIMETypeUtil.h>
 | 
						|
#import <SessionServiceKit/OWSMessageSender.h>
 | 
						|
#import <SessionServiceKit/OWSPrimaryStorage.h>
 | 
						|
#import <SessionServiceKit/OWSSyncConfigurationMessage.h>
 | 
						|
#import <SessionServiceKit/OWSSyncContactsMessage.h>
 | 
						|
#import <SessionServiceKit/OWSSyncGroupsMessage.h>
 | 
						|
#import <SessionServiceKit/LKSyncOpenGroupsMessage.h>
 | 
						|
#import <SessionServiceKit/SSKEnvironment.h>
 | 
						|
#import <SessionServiceKit/SignalAccount.h>
 | 
						|
#import <SessionServiceKit/SessionServiceKit-Swift.h>
 | 
						|
#import <SessionServiceKit/TSAccountManager.h>
 | 
						|
#import <SessionServiceKit/YapDatabaseConnection+OWS.h>
 | 
						|
#import <SessionServiceKit/TSContactThread.h>
 | 
						|
#import <SessionServiceKit/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;
 | 
						|
        }        
 | 
						|
 | 
						|
        [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];
 | 
						|
 | 
						|
    [LKStorage writeSyncWithBlock:^(YapDatabaseReadWriteTransaction *_Nonnull transaction) {
 | 
						|
        [self.messageSenderJobQueue addMessage:syncConfigurationMessage transaction:transaction];
 | 
						|
    } error:nil];
 | 
						|
}
 | 
						|
 | 
						|
#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
 | 
						|
{
 | 
						|
    return [LKSyncMessagesProtocol syncContactWithPublicKey:hexEncodedPubKey];
 | 
						|
}
 | 
						|
 | 
						|
- (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
 | 
						|
{
 | 
						|
    if (thread.usesSharedSenderKeys) {
 | 
						|
        __block AnyPromise *promise;
 | 
						|
        [LKStorage writeSyncWithBlock:^(YapDatabaseReadWriteTransaction *transaction) {
 | 
						|
            promise = [LKSyncMessagesProtocol syncClosedGroup:thread transaction:transaction];
 | 
						|
        } error:nil];
 | 
						|
        return promise;
 | 
						|
    } else {
 | 
						|
        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
 |