FIRMessagingRmqManagerTest.m 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404
  1. /*
  2. * Copyright 2017 Google
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License");
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. */
  16. #import <XCTest/XCTest.h>
  17. #import <OCMock/OCMock.h>
  18. #import "FIRTestsAssertionHandler.h"
  19. #import "XCTestCase+FIRMessagingRmqManagerTests.h"
  20. #import "Firebase/Messaging/FIRMessagingPersistentSyncMessage.h"
  21. #import "Firebase/Messaging/FIRMessagingRmqManager.h"
  22. #import "Firebase/Messaging/FIRMessagingUtilities.h"
  23. #import "Firebase/Messaging/Protos/GtalkCore.pbobjc.h"
  24. static NSString *const kRmqDatabaseName = @"rmq-test-db";
  25. static NSString *const kRmqDataMessageCategory = @"com.google.gcm-rmq-test";
  26. static const NSTimeInterval kAsyncTestTimout = 0.5;
  27. @interface FIRMessagingRmqManager (ExposedForTest)
  28. - (void)removeDatabase;
  29. - (dispatch_queue_t)databaseOperationQueue;
  30. @end
  31. @interface FIRMessagingRmqManagerTest : XCTestCase
  32. @property(nonatomic, readwrite, strong) FIRMessagingRmqManager *rmqManager;
  33. @property(nonatomic, strong) id assertionHandlerMock;
  34. @property(nonatomic, strong) FIRTestsAssertionHandler *testAssertionHandler;
  35. @end
  36. @implementation FIRMessagingRmqManagerTest
  37. - (void)setUp {
  38. [super setUp];
  39. self.testAssertionHandler = [[FIRTestsAssertionHandler alloc] init];
  40. self.assertionHandlerMock = OCMClassMock([NSAssertionHandler class]);
  41. OCMStub([self.assertionHandlerMock currentHandler]).andReturn(self.testAssertionHandler);
  42. // Make sure we start off with a clean state each time
  43. _rmqManager = [[FIRMessagingRmqManager alloc] initWithDatabaseName:kRmqDatabaseName];
  44. }
  45. - (void)tearDown {
  46. [self.rmqManager removeDatabase];
  47. [self waitForDrainDatabaseQueueForRmqManager:self.rmqManager];
  48. [self.assertionHandlerMock stopMocking];
  49. self.assertionHandlerMock = nil;
  50. self.testAssertionHandler = nil;
  51. [super tearDown];
  52. }
  53. /**
  54. * Add s2d messages with different RMQ-ID's to the RMQ. Fetch the messages
  55. * and verify that all messages were successfully saved.
  56. */
  57. - (void)testSavingS2dMessages {
  58. NSArray *messageIDs = @[ @"message1", @"message2", @"123456" ];
  59. for (NSString *messageID in messageIDs) {
  60. [self.rmqManager saveS2dMessageWithRmqId:messageID];
  61. }
  62. NSArray *rmqMessages = [self.rmqManager unackedS2dRmqIds];
  63. XCTAssertEqual(messageIDs.count, rmqMessages.count);
  64. for (NSString *messageID in rmqMessages) {
  65. XCTAssertTrue([messageIDs containsObject:messageID]);
  66. }
  67. }
  68. /**
  69. * Add s2d messages with different RMQ-ID's to the RMQ. Delete some of the
  70. * messages stored, assuming we received a server ACK for them. The remaining
  71. * messages should be fetched successfully.
  72. */
  73. - (void)testDeletingS2dMessages {
  74. NSArray *addMessages = @[ @"message1", @"message2", @"message3", @"message4" ];
  75. for (NSString *messageID in addMessages) {
  76. [self.rmqManager saveS2dMessageWithRmqId:messageID];
  77. }
  78. NSArray *removeMessages = @[ addMessages[1], addMessages[3] ];
  79. [self.rmqManager removeS2dIds:removeMessages];
  80. NSArray *remainingMessages = [self.rmqManager unackedS2dRmqIds];
  81. XCTAssertEqual(2, remainingMessages.count);
  82. XCTAssertTrue([remainingMessages containsObject:addMessages[0]]);
  83. XCTAssertTrue([remainingMessages containsObject:addMessages[2]]);
  84. }
  85. /**
  86. * Test deleting a s2d message that is not in the persistent store. This shouldn't
  87. * crash or alter the valid contents of the RMQ store.
  88. */
  89. - (void)testDeletingInvalidS2dMessage {
  90. NSString *validMessageID = @"validMessage123";
  91. [self.rmqManager saveS2dMessageWithRmqId:validMessageID];
  92. NSString *invalidMessageID = @"invalidMessage123";
  93. [self.rmqManager removeS2dIds:@[ invalidMessageID ]];
  94. NSArray *remainingMessages = [self.rmqManager unackedS2dRmqIds];
  95. XCTAssertEqual(1, remainingMessages.count);
  96. XCTAssertEqualObjects(validMessageID, remainingMessages[0]);
  97. }
  98. /**
  99. * Test that outgoing RMQ messages are correctly saved
  100. */
  101. - (void)testOutgoingRmqWithValidMessages {
  102. XCTestExpectation *expectation =
  103. [self expectationWithDescription:@"Scan outgoing messages is complete"];
  104. NSString *from = @"rmq-test";
  105. [self.rmqManager loadRmqId];
  106. GtalkDataMessageStanza *message1 = [self dataMessageWithMessageID:@"message1" from:from data:nil];
  107. // should successfully save the message to RMQ
  108. [self.rmqManager saveRmqMessage:message1
  109. withCompletionHandler:^(BOOL success) {
  110. XCTAssertTrue(success);
  111. GtalkDataMessageStanza *message2 = [self dataMessageWithMessageID:@"message2"
  112. from:from
  113. data:nil];
  114. // should successfully save the second message to RMQ
  115. [self.rmqManager saveRmqMessage:message2
  116. withCompletionHandler:^(BOOL success) {
  117. XCTAssertTrue(success);
  118. // message1 should have RMQ-ID = 2, message2 = 3
  119. [self.rmqManager scanWithRmqMessageHandler:^(NSDictionary *messages) {
  120. XCTAssertTrue([[messages allKeys] containsObject:@(2)]);
  121. XCTAssertTrue([[messages allKeys] containsObject:@(3)]);
  122. [expectation fulfill];
  123. }];
  124. }];
  125. }];
  126. [self waitForExpectationsWithTimeout:kAsyncTestTimout handler:nil];
  127. }
  128. /**
  129. * Test that an outgoing message with different properties is correctly saved to the RMQ.
  130. */
  131. - (void)testOutgoingDataMessageIsCorrectlySaved {
  132. XCTestExpectation *expectation = [self expectationWithDescription:@"Messages are saved"];
  133. NSString *from = @"rmq-test";
  134. NSString *messageID = @"message123";
  135. NSString *to = @"to-senderID-123";
  136. int32_t ttl = 2400;
  137. NSString *registrationToken = @"registration-token";
  138. NSDictionary *data = @{
  139. @"hello" : @"world",
  140. @"count" : @"2",
  141. };
  142. [self.rmqManager loadRmqId];
  143. GtalkDataMessageStanza *message = [self dataMessageWithMessageID:messageID from:from data:data];
  144. [message setTo:to];
  145. [message setTtl:ttl];
  146. [message setRegId:registrationToken];
  147. // should successfully save the message to RMQ
  148. [self.rmqManager saveRmqMessage:message
  149. withCompletionHandler:^(BOOL success) {
  150. XCTAssertTrue(success);
  151. [self.rmqManager scanWithRmqMessageHandler:^(NSDictionary *messages) {
  152. for (NSString *rmqID in messages) {
  153. GtalkDataMessageStanza *stanza = (GtalkDataMessageStanza *)messages[rmqID];
  154. XCTAssertEqualObjects(from, stanza.from);
  155. XCTAssertEqualObjects(messageID, stanza.id_p);
  156. XCTAssertEqualObjects(to, stanza.to);
  157. XCTAssertEqualObjects(registrationToken, stanza.regId);
  158. XCTAssertEqual(ttl, stanza.ttl);
  159. NSMutableDictionary *d = [NSMutableDictionary dictionary];
  160. for (GtalkAppData *appData in stanza.appDataArray) {
  161. d[appData.key] = appData.value;
  162. }
  163. XCTAssertTrue([data isEqualToDictionary:d]);
  164. }
  165. [expectation fulfill];
  166. }];
  167. }];
  168. [self waitForExpectationsWithTimeout:kAsyncTestTimout handler:nil];
  169. }
  170. /**
  171. * Test D2S messages being deleted from RMQ.
  172. */
  173. - (void)testDeletingD2SMessagesFromRMQ {
  174. XCTestExpectation *expectation = [self expectationWithDescription:@"Messages are deleted"];
  175. NSString *message1 = @"message123";
  176. NSString *ackedMessage = @"message234";
  177. NSString *from = @"from-rmq-test";
  178. GtalkDataMessageStanza *stanza1 = [self dataMessageWithMessageID:message1 from:from data:nil];
  179. GtalkDataMessageStanza *stanza2 = [self dataMessageWithMessageID:ackedMessage from:from data:nil];
  180. [self.rmqManager
  181. saveRmqMessage:stanza1
  182. withCompletionHandler:^(BOOL success) {
  183. XCTAssertTrue(success);
  184. [self.rmqManager
  185. saveRmqMessage:stanza2
  186. withCompletionHandler:^(BOOL success) {
  187. XCTAssertTrue(success);
  188. __block int64_t ackedMessageRmqID = -1;
  189. [self.rmqManager scanWithRmqMessageHandler:^(NSDictionary *messages) {
  190. for (NSString *rmqID in messages) {
  191. GtalkDataMessageStanza *stanza = (GtalkDataMessageStanza *)messages[rmqID];
  192. if ([stanza.id_p isEqualToString:ackedMessage]) {
  193. ackedMessageRmqID = rmqID.intValue;
  194. // should be a valid RMQ ID
  195. XCTAssertTrue(ackedMessageRmqID > 0);
  196. // delete the acked message
  197. NSString *rmqIDString = [NSString stringWithFormat:@"%lld", ackedMessageRmqID];
  198. [self.rmqManager removeRmqMessagesWithRmqIds:@[ rmqIDString ]];
  199. // should only have one message in the d2s RMQ
  200. [self.rmqManager scanWithRmqMessageHandler:^(NSDictionary *messages) {
  201. for (NSString *rmqID in messages) {
  202. GtalkDataMessageStanza *stanza2 = (GtalkDataMessageStanza *)messages[rmqID];
  203. // the acked message was queued later so should have
  204. // rmqID = ackedMessageRMQID - 1
  205. XCTAssertEqual(ackedMessageRmqID - 1, rmqID.intValue);
  206. XCTAssertEqual(message1, stanza2.id_p);
  207. }
  208. [expectation fulfill];
  209. }];
  210. }
  211. }
  212. }];
  213. }];
  214. }];
  215. [self waitForExpectationsWithTimeout:kAsyncTestTimout handler:nil];
  216. }
  217. /**
  218. * Test saving a sync message to SYNC_RMQ.
  219. */
  220. - (void)testSavingSyncMessage {
  221. NSString *rmqID = @"fake-rmq-id-1";
  222. int64_t expirationTime = FIRMessagingCurrentTimestampInSeconds() + 1;
  223. [self.rmqManager saveSyncMessageWithRmqID:rmqID
  224. expirationTime:expirationTime
  225. apnsReceived:YES
  226. mcsReceived:NO];
  227. FIRMessagingPersistentSyncMessage *persistentMessage =
  228. [self.rmqManager querySyncMessageWithRmqID:rmqID];
  229. XCTAssertEqual(persistentMessage.expirationTime, expirationTime);
  230. XCTAssertTrue(persistentMessage.apnsReceived);
  231. XCTAssertFalse(persistentMessage.mcsReceived);
  232. }
  233. /**
  234. * Test updating a sync message initially received via MCS, now being received via APNS.
  235. */
  236. - (void)testUpdateMessageReceivedViaAPNS {
  237. NSString *rmqID = @"fake-rmq-id-1";
  238. int64_t expirationTime = FIRMessagingCurrentTimestampInSeconds() + 1;
  239. [self.rmqManager saveSyncMessageWithRmqID:rmqID
  240. expirationTime:expirationTime
  241. apnsReceived:NO
  242. mcsReceived:YES];
  243. // Message was now received via APNS
  244. [self.rmqManager updateSyncMessageViaAPNSWithRmqID:rmqID];
  245. FIRMessagingPersistentSyncMessage *persistentMessage =
  246. [self.rmqManager querySyncMessageWithRmqID:rmqID];
  247. XCTAssertTrue(persistentMessage.apnsReceived);
  248. XCTAssertTrue(persistentMessage.mcsReceived);
  249. }
  250. /**
  251. * Test updating a sync message initially received via APNS, now being received via MCS.
  252. */
  253. - (void)testUpdateMessageReceivedViaMCS {
  254. NSString *rmqID = @"fake-rmq-id-1";
  255. int64_t expirationTime = FIRMessagingCurrentTimestampInSeconds() + 1;
  256. [self.rmqManager saveSyncMessageWithRmqID:rmqID
  257. expirationTime:expirationTime
  258. apnsReceived:YES
  259. mcsReceived:NO];
  260. // Message was now received via APNS
  261. [self.rmqManager updateSyncMessageViaMCSWithRmqID:rmqID];
  262. FIRMessagingPersistentSyncMessage *persistentMessage =
  263. [self.rmqManager querySyncMessageWithRmqID:rmqID];
  264. XCTAssertTrue(persistentMessage.apnsReceived);
  265. XCTAssertTrue(persistentMessage.mcsReceived);
  266. }
  267. /**
  268. * Test deleting sync messages from SYNC_RMQ.
  269. */
  270. - (void)testDeleteSyncMessage {
  271. NSString *rmqID = @"fake-rmq-id-1";
  272. int64_t expirationTime = FIRMessagingCurrentTimestampInSeconds() + 1;
  273. [self.rmqManager saveSyncMessageWithRmqID:rmqID
  274. expirationTime:expirationTime
  275. apnsReceived:YES
  276. mcsReceived:NO];
  277. XCTAssertNotNil([self.rmqManager querySyncMessageWithRmqID:rmqID]);
  278. // should successfully delete the message
  279. [self.rmqManager deleteSyncMessageWithRmqID:rmqID];
  280. XCTAssertNil([self.rmqManager querySyncMessageWithRmqID:rmqID]);
  281. }
  282. - (void)testInitWhenDatabaseIsBrokenThenDatabaseIsDeleted {
  283. NSString *databaseName = @"invalid-database-file";
  284. NSString *databasePath = [self createBrokenDatabaseWithName:databaseName];
  285. XCTAssert([[NSFileManager defaultManager] fileExistsAtPath:databasePath]);
  286. // Expect for at least one assertion.
  287. XCTestExpectation *assertionFailureExpectation =
  288. [self expectationWithDescription:@"assertionFailureExpectation"];
  289. assertionFailureExpectation.assertForOverFulfill = NO;
  290. // The flag FIR_MESSAGING_ASSERTIONS_BLOCKED can be set by blaze when running tests from google3.
  291. #ifndef FIR_MESSAGING_ASSERTIONS_BLOCKED
  292. [self.testAssertionHandler
  293. setMethodFailureHandlerForClass:[FIRMessagingRmqManager class]
  294. handler:^(id object, NSString *fileName, NSInteger lineNumber) {
  295. [assertionFailureExpectation fulfill];
  296. }];
  297. #else
  298. // If FIR_MESSAGING_ASSERTIONS_BLOCKED is defined, then no assertion handlers will be called,
  299. // so don't wait for it.
  300. dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(0.5 * NSEC_PER_SEC)),
  301. dispatch_get_main_queue(), ^{
  302. [assertionFailureExpectation fulfill];
  303. });
  304. #endif // FIR_MESSAGING_ASSERTIONS_BLOCKED
  305. // Create `FIRMessagingRmqManager` instance with a broken database.
  306. FIRMessagingRmqManager *manager =
  307. [[FIRMessagingRmqManager alloc] initWithDatabaseName:databaseName];
  308. [self waitForExpectations:@[ assertionFailureExpectation ] timeout:0.5];
  309. [self waitForDrainDatabaseQueueForRmqManager:manager];
  310. // Check that the file was deleted.
  311. XCTAssertFalse([[NSFileManager defaultManager] fileExistsAtPath:databasePath]);
  312. }
  313. #pragma mark - Private Helpers
  314. - (GtalkDataMessageStanza *)dataMessageWithMessageID:(NSString *)messageID
  315. from:(NSString *)from
  316. data:(NSDictionary *)data {
  317. GtalkDataMessageStanza *stanza = [[GtalkDataMessageStanza alloc] init];
  318. [stanza setId_p:messageID];
  319. [stanza setFrom:from];
  320. [stanza setCategory:kRmqDataMessageCategory];
  321. for (NSString *key in data) {
  322. NSString *val = data[key];
  323. GtalkAppData *appData = [[GtalkAppData alloc] init];
  324. [appData setKey:key];
  325. [appData setValue:val];
  326. [[stanza appDataArray] addObject:appData];
  327. }
  328. return stanza;
  329. }
  330. - (NSString *)createBrokenDatabaseWithName:(NSString *)name {
  331. NSString *databasePath = [FIRMessagingRmqManager pathForDatabaseWithName:name];
  332. NSMutableArray *pathComponents = [[databasePath pathComponents] mutableCopy];
  333. [pathComponents removeLastObject];
  334. NSString *directoryPath = [NSString pathWithComponents:pathComponents];
  335. // Create directory if doesn't exist.
  336. [[NSFileManager defaultManager] createDirectoryAtPath:directoryPath
  337. withIntermediateDirectories:YES
  338. attributes:nil
  339. error:NULL];
  340. // Remove the file if exists.
  341. [[NSFileManager defaultManager] removeItemAtPath:databasePath error:NULL];
  342. NSData *brokenDBFileContent = [@"not a valid DB" dataUsingEncoding:NSUTF8StringEncoding];
  343. [brokenDBFileContent writeToFile:databasePath atomically:YES];
  344. XCTAssertEqualObjects([NSData dataWithContentsOfFile:databasePath], brokenDBFileContent);
  345. return databasePath;
  346. }
  347. @end