|  |  |  | @ -42,6 +42,13 @@ | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | NS_ASSUME_NONNULL_BEGIN | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | void AssertIsOnSendingQueue() | 
		
	
		
			
				|  |  |  |  | { | 
		
	
		
			
				|  |  |  |  |     if (SYSTEM_VERSION_GREATER_THAN_OR_EQUAL_TO(10, 0)) { | 
		
	
		
			
				|  |  |  |  |         dispatch_assert_queue([OWSDispatch sendingQueue]); | 
		
	
		
			
				|  |  |  |  |     } // else, skip assert as it's a development convenience. | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | /** | 
		
	
		
			
				|  |  |  |  |  * OWSSendMessageOperation encapsulates all the work associated with sending a message, e.g. uploading attachments, | 
		
	
		
			
				|  |  |  |  |  * getting proper keys, and retrying upon failure. | 
		
	
	
		
			
				
					|  |  |  | @ -724,6 +731,7 @@ NSString *const OWSMessageSenderRateLimitedException = @"RateLimitedException"; | 
		
	
		
			
				|  |  |  |  |             failure:(RetryableFailureHandler)failureHandler | 
		
	
		
			
				|  |  |  |  | { | 
		
	
		
			
				|  |  |  |  |     DDLogDebug(@"%@ sending message to service: %@", self.tag, message.debugDescription); | 
		
	
		
			
				|  |  |  |  |     AssertIsOnSendingQueue(); | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |     if ([TSPreKeyManager isAppLockedDueToPreKeyUpdateFailures]) { | 
		
	
		
			
				|  |  |  |  |         OWSAnalyticsError(@"Message send failed due to prekey update failures"); | 
		
	
	
		
			
				
					|  |  |  | @ -857,8 +865,7 @@ NSString *const OWSMessageSenderRateLimitedException = @"RateLimitedException"; | 
		
	
		
			
				|  |  |  |  |                         return failureHandler(error, YES); | 
		
	
		
			
				|  |  |  |  |                     } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |                     [self handleMismatchedDevices:serializedResponse recipient:recipient]; | 
		
	
		
			
				|  |  |  |  |                     retrySend(); | 
		
	
		
			
				|  |  |  |  |                     [self handleMismatchedDevices:serializedResponse recipient:recipient completion:retrySend]; | 
		
	
		
			
				|  |  |  |  |                     break; | 
		
	
		
			
				|  |  |  |  |                 } | 
		
	
		
			
				|  |  |  |  |                 case 410: { | 
		
	
	
		
			
				
					|  |  |  | @ -871,8 +878,9 @@ NSString *const OWSMessageSenderRateLimitedException = @"RateLimitedException"; | 
		
	
		
			
				|  |  |  |  |                         return failureHandler(error, YES); | 
		
	
		
			
				|  |  |  |  |                     } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |                     [self handleStaleDevicesWithResponse:responseData recipientId:recipient.uniqueId]; | 
		
	
		
			
				|  |  |  |  |                     retrySend(); | 
		
	
		
			
				|  |  |  |  |                     [self handleStaleDevicesWithResponse:responseData | 
		
	
		
			
				|  |  |  |  |                                              recipientId:recipient.uniqueId | 
		
	
		
			
				|  |  |  |  |                                               completion:retrySend]; | 
		
	
		
			
				|  |  |  |  |                     break; | 
		
	
		
			
				|  |  |  |  |                 } | 
		
	
		
			
				|  |  |  |  |                 default: | 
		
	
	
		
			
				
					|  |  |  | @ -882,11 +890,14 @@ NSString *const OWSMessageSenderRateLimitedException = @"RateLimitedException"; | 
		
	
		
			
				|  |  |  |  |         }]; | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | - (void)handleMismatchedDevices:(NSDictionary *)dictionary recipient:(SignalRecipient *)recipient | 
		
	
		
			
				|  |  |  |  | - (void)handleMismatchedDevices:(NSDictionary *)dictionary | 
		
	
		
			
				|  |  |  |  |                       recipient:(SignalRecipient *)recipient | 
		
	
		
			
				|  |  |  |  |                      completion:(void (^)())completionHandler | 
		
	
		
			
				|  |  |  |  | { | 
		
	
		
			
				|  |  |  |  |     NSArray *extraDevices = [dictionary objectForKey:@"extraDevices"]; | 
		
	
		
			
				|  |  |  |  |     NSArray *missingDevices = [dictionary objectForKey:@"missingDevices"]; | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |     dispatch_async([OWSDispatch sessionStoreQueue], ^{ | 
		
	
		
			
				|  |  |  |  |         if (extraDevices && extraDevices.count > 0) { | 
		
	
		
			
				|  |  |  |  |             for (NSNumber *extraDeviceId in extraDevices) { | 
		
	
		
			
				|  |  |  |  |                 [self.storageManager deleteSessionForContact:recipient.uniqueId deviceId:extraDeviceId.intValue]; | 
		
	
	
		
			
				
					|  |  |  | @ -900,6 +911,8 @@ NSString *const OWSMessageSenderRateLimitedException = @"RateLimitedException"; | 
		
	
		
			
				|  |  |  |  |         } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |         [recipient save]; | 
		
	
		
			
				|  |  |  |  |         completionHandler(); | 
		
	
		
			
				|  |  |  |  |     }); | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | - (void)handleMessageSentLocally:(TSOutgoingMessage *)message | 
		
	
	
		
			
				
					|  |  |  | @ -1173,7 +1186,9 @@ NSString *const OWSMessageSenderRateLimitedException = @"RateLimitedException"; | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | - (void)handleStaleDevicesWithResponse:(NSData *)responseData recipientId:(NSString *)identifier | 
		
	
		
			
				|  |  |  |  | - (void)handleStaleDevicesWithResponse:(NSData *)responseData | 
		
	
		
			
				|  |  |  |  |                            recipientId:(NSString *)identifier | 
		
	
		
			
				|  |  |  |  |                             completion:(void (^)())completionHandler | 
		
	
		
			
				|  |  |  |  | { | 
		
	
		
			
				|  |  |  |  |     dispatch_async([OWSDispatch sendingQueue], ^{ | 
		
	
		
			
				|  |  |  |  |         NSDictionary *serialization = [NSJSONSerialization JSONObjectWithData:responseData options:0 error:nil]; | 
		
	
	
		
			
				
					|  |  |  | @ -1183,10 +1198,13 @@ NSString *const OWSMessageSenderRateLimitedException = @"RateLimitedException"; | 
		
	
		
			
				|  |  |  |  |             return; | 
		
	
		
			
				|  |  |  |  |         } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |         dispatch_async([OWSDispatch sessionStoreQueue], ^{ | 
		
	
		
			
				|  |  |  |  |             for (NSUInteger i = 0; i < [devices count]; i++) { | 
		
	
		
			
				|  |  |  |  |                 int deviceNumber = [devices[i] intValue]; | 
		
	
		
			
				|  |  |  |  |                 [[TSStorageManager sharedManager] deleteSessionForContact:identifier deviceId:deviceNumber]; | 
		
	
		
			
				|  |  |  |  |             } | 
		
	
		
			
				|  |  |  |  |             completionHandler(); | 
		
	
		
			
				|  |  |  |  |         }); | 
		
	
		
			
				|  |  |  |  |     }); | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
	
		
			
				
					|  |  |  | 
 |