You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 

147 lines
5.4 KiB

/*
* Copyright 2017 Google
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#import "FIRMessagingSyncMessageManager.h"
#import "FIRMessagingConstants.h"
#import "FIRMessagingDefines.h"
#import "FIRMessagingLogger.h"
#import "FIRMessagingPersistentSyncMessage.h"
#import "FIRMessagingRmqManager.h"
#import "FIRMessagingUtilities.h"
static const int64_t kDefaultSyncMessageTTL = 4 * 7 * 24 * 60 * 60; // 4 weeks
// 4 MB of free space is required to persist Sync messages
static const uint64_t kMinFreeDiskSpaceInMB = 1;
@interface FIRMessagingSyncMessageManager()
@property(nonatomic, readwrite, strong) FIRMessagingRmqManager *rmqManager;
@end
@implementation FIRMessagingSyncMessageManager
- (instancetype)init {
FIRMessagingInvalidateInitializer();
}
- (instancetype)initWithRmqManager:(FIRMessagingRmqManager *)rmqManager {
_FIRMessagingDevAssert(rmqManager, @"Invalid nil rmq manager while initalizing sync message manager");
self = [super init];
if (self) {
_rmqManager = rmqManager;
}
return self;
}
- (void)removeExpiredSyncMessages {
NSError *error;
int deleteCount = [self.rmqManager deleteExpiredOrFinishedSyncMessages:&error];
if (error) {
FIRMessagingLoggerError(kFIRMessagingMessageCodeSyncMessageManager000,
@"Error while deleting expired sync messages %@", error);
} else if (deleteCount > 0) {
FIRMessagingLoggerDebug(kFIRMessagingMessageCodeSyncMessageManager001,
@"Successfully deleted %d sync messages from store", deleteCount);
}
}
- (BOOL)didReceiveAPNSSyncMessage:(NSDictionary *)message {
return [self didReceiveSyncMessage:message viaAPNS:YES viaMCS:NO];
}
- (BOOL)didReceiveMCSSyncMessage:(NSDictionary *)message {
return [self didReceiveSyncMessage:message viaAPNS:NO viaMCS:YES];
}
- (BOOL)didReceiveSyncMessage:(NSDictionary *)message
viaAPNS:(BOOL)viaAPNS
viaMCS:(BOOL)viaMCS {
NSString *rmqID = message[kFIRMessagingMessageIDKey];
_FIRMessagingDevAssert([rmqID length], @"Invalid nil rmqID for message");
if (![rmqID length]) {
FIRMessagingLoggerError(kFIRMessagingMessageCodeSyncMessageManager002,
@"Invalid nil rmqID for sync message.");
return NO;
}
FIRMessagingPersistentSyncMessage *persistentMessage =
[self.rmqManager querySyncMessageWithRmqID:rmqID];
NSError *error;
if (!persistentMessage) {
// Do not persist the new message if we don't have enough disk space
uint64_t freeDiskSpace = FIRMessagingGetFreeDiskSpaceInMB();
if (freeDiskSpace < kMinFreeDiskSpaceInMB) {
return NO;
}
int64_t expirationTime = [[self class] expirationTimeForSyncMessage:message];
if (![self.rmqManager saveSyncMessageWithRmqID:rmqID
expirationTime:expirationTime
apnsReceived:viaAPNS
mcsReceived:viaMCS
error:&error]) {
FIRMessagingLoggerError(kFIRMessagingMessageCodeSyncMessageManager003,
@"Failed to save sync message with rmqID %@", rmqID);
} else {
FIRMessagingLoggerInfo(kFIRMessagingMessageCodeSyncMessageManager004,
@"Added sync message to cache: %@", rmqID);
}
return NO;
}
if (viaAPNS && !persistentMessage.apnsReceived) {
persistentMessage.apnsReceived = YES;
if (![self.rmqManager updateSyncMessageViaAPNSWithRmqID:rmqID error:&error]) {
FIRMessagingLoggerError(kFIRMessagingMessageCodeSyncMessageManager005,
@"Failed to update APNS state for sync message %@", rmqID);
}
} else if (viaMCS && !persistentMessage.mcsReceived) {
persistentMessage.mcsReceived = YES;
if (![self.rmqManager updateSyncMessageViaMCSWithRmqID:rmqID error:&error]) {
FIRMessagingLoggerError(kFIRMessagingMessageCodeSyncMessageManager006,
@"Failed to update MCS state for sync message %@", rmqID);
}
}
// Received message via both ways we can safely delete it.
if (persistentMessage.apnsReceived && persistentMessage.mcsReceived) {
if (![self.rmqManager deleteSyncMessageWithRmqID:rmqID]) {
FIRMessagingLoggerError(kFIRMessagingMessageCodeSyncMessageManager007,
@"Failed to delete sync message %@", rmqID);
} else {
FIRMessagingLoggerInfo(kFIRMessagingMessageCodeSyncMessageManager008,
@"Successfully deleted sync message from cache %@", rmqID);
}
}
// Already received this message either via MCS or APNS.
return YES;
}
+ (int64_t)expirationTimeForSyncMessage:(NSDictionary *)message {
int64_t ttl = kDefaultSyncMessageTTL;
if (message[kFIRMessagingMessageSyncMessageTTLKey]) {
ttl = [message[kFIRMessagingMessageSyncMessageTTLKey] longLongValue];
}
int64_t currentTime = FIRMessagingCurrentTimestampInSeconds();
return currentTime + ttl;
}
@end