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.
		
		
		
		
		
			
		
			
				
	
	
		
			218 lines
		
	
	
		
			7.1 KiB
		
	
	
	
		
			Objective-C
		
	
			
		
		
	
	
			218 lines
		
	
	
		
			7.1 KiB
		
	
	
	
		
			Objective-C
		
	
| //
 | |
| //  Copyright (c) 2018 Open Whisper Systems. All rights reserved.
 | |
| //
 | |
| 
 | |
| #import "SignalRecipient.h"
 | |
| #import "ProfileManagerProtocol.h"
 | |
| #import "SSKEnvironment.h"
 | |
| #import "TSAccountManager.h"
 | |
| #import <YapDatabase/YapDatabaseConnection.h>
 | |
| 
 | |
| NS_ASSUME_NONNULL_BEGIN
 | |
| 
 | |
| @interface SignalRecipient ()
 | |
| 
 | |
| @property (nonatomic) NSOrderedSet *devices;
 | |
| 
 | |
| @end
 | |
| 
 | |
| #pragma mark -
 | |
| 
 | |
| @implementation SignalRecipient
 | |
| 
 | |
| #pragma mark - Dependencies
 | |
| 
 | |
| - (id<ProfileManagerProtocol>)profileManager
 | |
| {
 | |
|     return SSKEnvironment.shared.profileManager;
 | |
| }
 | |
| 
 | |
| - (TSAccountManager *)tsAccountManager
 | |
| {
 | |
|     return SSKEnvironment.shared.tsAccountManager;
 | |
| }
 | |
| 
 | |
| #pragma mark -
 | |
| 
 | |
| + (instancetype)getOrBuildUnsavedRecipientForRecipientId:(NSString *)recipientId
 | |
|                                              transaction:(YapDatabaseReadTransaction *)transaction
 | |
| {
 | |
|     SignalRecipient *_Nullable recipient =
 | |
|         [self registeredRecipientForRecipientId:recipientId mustHaveDevices:NO transaction:transaction];
 | |
|     if (!recipient) {
 | |
|         recipient = [[self alloc] initWithTextSecureIdentifier:recipientId];
 | |
|     }
 | |
|     return recipient;
 | |
| }
 | |
| 
 | |
| - (instancetype)initWithTextSecureIdentifier:(NSString *)textSecureIdentifier
 | |
| {
 | |
|     self = [super initWithUniqueId:textSecureIdentifier];
 | |
|     if (!self) {
 | |
|         return self;
 | |
|     }
 | |
|    
 | |
|     _devices = [NSOrderedSet orderedSetWithObject:@(1)];
 | |
| 
 | |
|     return self;
 | |
| }
 | |
| 
 | |
| - (nullable instancetype)initWithCoder:(NSCoder *)coder
 | |
| {
 | |
|     self = [super initWithCoder:coder];
 | |
|     if (!self) {
 | |
|         return self;
 | |
|     }
 | |
| 
 | |
|     if (_devices == nil) {
 | |
|         _devices = [NSOrderedSet new];
 | |
|     }
 | |
| 
 | |
|     // Since we use device count to determine whether a user is registered or not,
 | |
|     // ensure the local user always has at least *this* device.
 | |
|     if (![_devices containsObject:@(1)]) {
 | |
|         if ([self.uniqueId isEqualToString:self.tsAccountManager.localNumber]) {
 | |
|             [self addDevices:[NSSet setWithObject:@(1)]];
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return self;
 | |
| }
 | |
| 
 | |
| + (nullable instancetype)registeredRecipientForRecipientId:(NSString *)recipientId
 | |
|                                            mustHaveDevices:(BOOL)mustHaveDevices
 | |
|                                                transaction:(YapDatabaseReadTransaction *)transaction
 | |
| {
 | |
|     SignalRecipient *_Nullable signalRecipient = [self fetchObjectWithUniqueID:recipientId transaction:transaction];
 | |
|     if (mustHaveDevices && signalRecipient.devices.count < 1) {
 | |
|         return nil;
 | |
|     }
 | |
|     return signalRecipient;
 | |
| }
 | |
| 
 | |
| - (void)addDevices:(NSSet *)devices
 | |
| {
 | |
|     NSMutableOrderedSet *updatedDevices = [self.devices mutableCopy];
 | |
|     [updatedDevices unionSet:devices];
 | |
|     self.devices = [updatedDevices copy];
 | |
| }
 | |
| 
 | |
| - (void)removeDevices:(NSSet *)devices
 | |
| {
 | |
|     NSMutableOrderedSet *updatedDevices = [self.devices mutableCopy];
 | |
|     [updatedDevices minusSet:devices];
 | |
|     self.devices = [updatedDevices copy];
 | |
| }
 | |
| 
 | |
| - (void)updateRegisteredRecipientWithDevicesToAdd:(nullable NSArray *)devicesToAdd
 | |
|                                   devicesToRemove:(nullable NSArray *)devicesToRemove
 | |
|                                       transaction:(YapDatabaseReadWriteTransaction *)transaction {
 | |
|     // Add before we remove, since removeDevicesFromRecipient:...
 | |
|     // can markRecipientAsUnregistered:... if the recipient has
 | |
|     // no devices left.
 | |
|     if (devicesToAdd.count > 0) {
 | |
|         [self addDevicesToRegisteredRecipient:[NSSet setWithArray:devicesToAdd] transaction:transaction];
 | |
|     }
 | |
|     if (devicesToRemove.count > 0) {
 | |
|         [self removeDevicesFromRecipient:[NSSet setWithArray:devicesToRemove] transaction:transaction];
 | |
|     }
 | |
| }
 | |
| 
 | |
| - (void)addDevicesToRegisteredRecipient:(NSSet *)devices transaction:(YapDatabaseReadWriteTransaction *)transaction
 | |
| {
 | |
|     [self reloadWithTransaction:transaction];
 | |
|     [self addDevices:devices];
 | |
|     [self saveWithTransaction_internal:transaction];
 | |
| }
 | |
| 
 | |
| - (void)removeDevicesFromRecipient:(NSSet *)devices transaction:(YapDatabaseReadWriteTransaction *)transaction
 | |
| {
 | |
|     [self reloadWithTransaction:transaction ignoreMissing:YES];
 | |
|     [self removeDevices:devices];
 | |
|     [self saveWithTransaction_internal:transaction];
 | |
| }
 | |
| 
 | |
| - (NSString *)recipientId
 | |
| {
 | |
|     return self.uniqueId;
 | |
| }
 | |
| 
 | |
| - (NSComparisonResult)compare:(SignalRecipient *)other
 | |
| {
 | |
|     return [self.recipientId compare:other.recipientId];
 | |
| }
 | |
| 
 | |
| - (void)removeWithTransaction:(YapDatabaseReadWriteTransaction *)transaction
 | |
| {
 | |
|     // We need to distinguish between "users we know to be unregistered" and
 | |
|     // "users whose registration status is unknown".  The former correspond to
 | |
|     // instances of SignalRecipient with no devices.  The latter do not
 | |
|     // correspond to an instance of SignalRecipient in the database (although
 | |
|     // they may correspond to an "unsaved" instance of SignalRecipient built
 | |
|     // by getOrBuildUnsavedRecipientForRecipientId.
 | |
| 
 | |
|     [super removeWithTransaction:transaction];
 | |
| }
 | |
| 
 | |
| - (void)saveWithTransaction:(YapDatabaseReadWriteTransaction *)transaction
 | |
| {
 | |
|     // We only want to mutate the persisted SignalRecipients in the database
 | |
|     // using other methods of this class, e.g. markRecipientAsRegistered...
 | |
|     // to create, addDevices and removeDevices to mutate.  We're trying to
 | |
|     // be strict about using persisted SignalRecipients as a cache to
 | |
|     // reflect "last known registration status".  Forcing our codebase to
 | |
|     // use those methods helps ensure that we update the cache deliberately.
 | |
| 
 | |
|     [self saveWithTransaction_internal:transaction];
 | |
| }
 | |
| 
 | |
| - (void)saveWithTransaction_internal:(YapDatabaseReadWriteTransaction *)transaction
 | |
| {
 | |
|     [super saveWithTransaction:transaction];
 | |
| }
 | |
| 
 | |
| + (BOOL)isRegisteredRecipient:(NSString *)recipientId transaction:(YapDatabaseReadTransaction *)transaction
 | |
| {
 | |
|     return nil != [self registeredRecipientForRecipientId:recipientId mustHaveDevices:YES transaction:transaction];
 | |
| }
 | |
| 
 | |
| + (SignalRecipient *)markRecipientAsRegisteredAndGet:(NSString *)recipientId
 | |
|                                          transaction:(YapDatabaseReadWriteTransaction *)transaction
 | |
| {
 | |
|     SignalRecipient *_Nullable instance =
 | |
|         [self registeredRecipientForRecipientId:recipientId mustHaveDevices:YES transaction:transaction];
 | |
| 
 | |
|     if (!instance) {
 | |
| 
 | |
|         instance = [[self alloc] initWithTextSecureIdentifier:recipientId];
 | |
|         [instance saveWithTransaction_internal:transaction];
 | |
|     }
 | |
|     return instance;
 | |
| }
 | |
| 
 | |
| + (void)markRecipientAsRegistered:(NSString *)recipientId
 | |
|                          deviceId:(UInt32)deviceId
 | |
|                       transaction:(YapDatabaseReadWriteTransaction *)transaction
 | |
| {
 | |
|     SignalRecipient *recipient = [self markRecipientAsRegisteredAndGet:recipientId transaction:transaction];
 | |
|     if (![recipient.devices containsObject:@(deviceId)]) {
 | |
| 
 | |
|         [recipient addDevices:[NSSet setWithObject:@(deviceId)]];
 | |
|         [recipient saveWithTransaction_internal:transaction];
 | |
|     }
 | |
| }
 | |
| 
 | |
| + (void)markRecipientAsUnregistered:(NSString *)recipientId transaction:(YapDatabaseReadWriteTransaction *)transaction
 | |
| {
 | |
|     SignalRecipient *instance = [self getOrBuildUnsavedRecipientForRecipientId:recipientId
 | |
|                                                                    transaction:transaction];
 | |
|     if (instance.devices.count > 0) {
 | |
|         [instance removeDevices:instance.devices.set];
 | |
|     }
 | |
|     [instance saveWithTransaction_internal:transaction];
 | |
| }
 | |
| 
 | |
| @end
 | |
| 
 | |
| NS_ASSUME_NONNULL_END
 |