FIRMessagingRmqManagerTest.m 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398
  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 = [self expectationWithDescription:@"Scan outgoing messages is complete"];
  103. NSString *from = @"rmq-test";
  104. [self.rmqManager loadRmqId];
  105. GtalkDataMessageStanza *message1 = [self dataMessageWithMessageID:@"message1"
  106. from:from
  107. data:nil];
  108. // should successfully save the message to RMQ
  109. [self.rmqManager saveRmqMessage:message1 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 withCompletionHandler:^(BOOL success) {
  116. XCTAssertTrue(success);
  117. // message1 should have RMQ-ID = 2, message2 = 3
  118. [self.rmqManager scanWithRmqMessageHandler:^(NSDictionary *messages) {
  119. XCTAssertTrue([[messages allKeys] containsObject:@(2)]);
  120. XCTAssertTrue([[messages allKeys] containsObject:@(3)]);
  121. [expectation fulfill];
  122. }];
  123. }];
  124. }];
  125. [self waitForExpectationsWithTimeout:kAsyncTestTimout handler:nil];
  126. }
  127. /**
  128. * Test that an outgoing message with different properties is correctly saved to the RMQ.
  129. */
  130. - (void)testOutgoingDataMessageIsCorrectlySaved {
  131. XCTestExpectation *expectation = [self expectationWithDescription:@"Messages are saved"];
  132. NSString *from = @"rmq-test";
  133. NSString *messageID = @"message123";
  134. NSString *to = @"to-senderID-123";
  135. int32_t ttl = 2400;
  136. NSString *registrationToken = @"registration-token";
  137. NSDictionary *data = @{
  138. @"hello" : @"world",
  139. @"count" : @"2",
  140. };
  141. [self.rmqManager loadRmqId];
  142. GtalkDataMessageStanza *message = [self dataMessageWithMessageID:messageID
  143. from:from
  144. data:data];
  145. [message setTo:to];
  146. [message setTtl:ttl];
  147. [message setRegId:registrationToken];
  148. // should successfully save the message to RMQ
  149. [self.rmqManager saveRmqMessage:message 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
  180. from:from
  181. data:nil];
  182. [self.rmqManager saveRmqMessage:stanza1 withCompletionHandler:^(BOOL success) {
  183. XCTAssertTrue(success);
  184. [self.rmqManager saveRmqMessage:stanza2 withCompletionHandler:^(BOOL success) {
  185. XCTAssertTrue(success);
  186. __block int64_t ackedMessageRmqID = -1;
  187. [self.rmqManager scanWithRmqMessageHandler:^(NSDictionary *messages) {
  188. for (NSString *rmqID in messages) {
  189. GtalkDataMessageStanza *stanza = (GtalkDataMessageStanza *)messages[rmqID];
  190. if ([stanza.id_p isEqualToString:ackedMessage]) {
  191. ackedMessageRmqID = rmqID.intValue;
  192. // should be a valid RMQ ID
  193. XCTAssertTrue(ackedMessageRmqID > 0);
  194. // delete the acked message
  195. NSString *rmqIDString = [NSString stringWithFormat:@"%lld", ackedMessageRmqID];
  196. [self.rmqManager removeRmqMessagesWithRmqIds:@[rmqIDString]];
  197. // should only have one message in the d2s RMQ
  198. [self.rmqManager scanWithRmqMessageHandler:^(NSDictionary *messages) {
  199. for (NSString *rmqID in messages) {
  200. GtalkDataMessageStanza *stanza2 = (GtalkDataMessageStanza *)messages[rmqID];
  201. // the acked message was queued later so should have
  202. // rmqID = ackedMessageRMQID - 1
  203. XCTAssertEqual(ackedMessageRmqID - 1, rmqID.intValue);
  204. XCTAssertEqual(message1, stanza2.id_p);
  205. }
  206. [expectation fulfill];
  207. }];
  208. }
  209. }
  210. }];
  211. }];
  212. }];
  213. [self waitForExpectationsWithTimeout:kAsyncTestTimout handler:nil];
  214. }
  215. /**
  216. * Test saving a sync message to SYNC_RMQ.
  217. */
  218. - (void)testSavingSyncMessage {
  219. NSString *rmqID = @"fake-rmq-id-1";
  220. int64_t expirationTime = FIRMessagingCurrentTimestampInSeconds() + 1;
  221. [self.rmqManager saveSyncMessageWithRmqID:rmqID
  222. expirationTime:expirationTime
  223. apnsReceived:YES
  224. mcsReceived:NO];
  225. FIRMessagingPersistentSyncMessage *persistentMessage = [self.rmqManager querySyncMessageWithRmqID:rmqID];
  226. XCTAssertEqual(persistentMessage.expirationTime, expirationTime);
  227. XCTAssertTrue(persistentMessage.apnsReceived);
  228. XCTAssertFalse(persistentMessage.mcsReceived);
  229. }
  230. /**
  231. * Test updating a sync message initially received via MCS, now being received via APNS.
  232. */
  233. - (void)testUpdateMessageReceivedViaAPNS {
  234. NSString *rmqID = @"fake-rmq-id-1";
  235. int64_t expirationTime = FIRMessagingCurrentTimestampInSeconds() + 1;
  236. [self.rmqManager saveSyncMessageWithRmqID:rmqID
  237. expirationTime:expirationTime
  238. apnsReceived:NO
  239. mcsReceived:YES];
  240. // Message was now received via APNS
  241. [self.rmqManager updateSyncMessageViaAPNSWithRmqID:rmqID];
  242. FIRMessagingPersistentSyncMessage *persistentMessage = [self.rmqManager querySyncMessageWithRmqID:rmqID];
  243. XCTAssertTrue(persistentMessage.apnsReceived);
  244. XCTAssertTrue(persistentMessage.mcsReceived);
  245. }
  246. /**
  247. * Test updating a sync message initially received via APNS, now being received via MCS.
  248. */
  249. - (void)testUpdateMessageReceivedViaMCS {
  250. NSString *rmqID = @"fake-rmq-id-1";
  251. int64_t expirationTime = FIRMessagingCurrentTimestampInSeconds() + 1;
  252. [self.rmqManager saveSyncMessageWithRmqID:rmqID
  253. expirationTime:expirationTime
  254. apnsReceived:YES
  255. mcsReceived:NO];
  256. // Message was now received via APNS
  257. [self.rmqManager updateSyncMessageViaMCSWithRmqID:rmqID];
  258. FIRMessagingPersistentSyncMessage *persistentMessage = [self.rmqManager querySyncMessageWithRmqID:rmqID];
  259. XCTAssertTrue(persistentMessage.apnsReceived);
  260. XCTAssertTrue(persistentMessage.mcsReceived);
  261. }
  262. /**
  263. * Test deleting sync messages from SYNC_RMQ.
  264. */
  265. - (void)testDeleteSyncMessage {
  266. NSString *rmqID = @"fake-rmq-id-1";
  267. int64_t expirationTime = FIRMessagingCurrentTimestampInSeconds() + 1;
  268. [self.rmqManager saveSyncMessageWithRmqID:rmqID
  269. expirationTime:expirationTime
  270. apnsReceived:YES
  271. mcsReceived:NO];
  272. XCTAssertNotNil([self.rmqManager querySyncMessageWithRmqID:rmqID]);
  273. // should successfully delete the message
  274. [self.rmqManager deleteSyncMessageWithRmqID:rmqID];
  275. XCTAssertNil([self.rmqManager querySyncMessageWithRmqID:rmqID]);
  276. }
  277. - (void)testInitWhenDatabaseIsBrokenThenDatabaseIsDeleted {
  278. NSString *databaseName = @"invalid-database-file";
  279. NSString *databasePath = [self createBrokenDatabaseWithName:databaseName];
  280. XCTAssert([[NSFileManager defaultManager] fileExistsAtPath:databasePath]);
  281. // Expect for at least one assertion.
  282. XCTestExpectation *assertionFailureExpectation = [self expectationWithDescription:@"assertionFailureExpectation"];
  283. assertionFailureExpectation.assertForOverFulfill = NO;
  284. // The flag FIR_MESSAGING_ASSERTIONS_BLOCKED can be set by blaze when running tests from google3.
  285. #ifndef FIR_MESSAGING_ASSERTIONS_BLOCKED
  286. [self.testAssertionHandler setMethodFailureHandlerForClass:[FIRMessagingRmqManager class]
  287. handler:^(id object, NSString *fileName, NSInteger lineNumber) {
  288. [assertionFailureExpectation fulfill];
  289. }];
  290. #else
  291. // If FIR_MESSAGING_ASSERTIONS_BLOCKED is defined, then no assertion handlers will be called,
  292. // so don't wait for it.
  293. dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(0.5 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
  294. [assertionFailureExpectation fulfill];
  295. });
  296. #endif // FIR_MESSAGING_ASSERTIONS_BLOCKED
  297. // Create `FIRMessagingRmqManager` instance with a broken database.
  298. FIRMessagingRmqManager *manager = [[FIRMessagingRmqManager alloc] initWithDatabaseName:databaseName];
  299. [self waitForExpectations:@[ assertionFailureExpectation ] timeout:0.5];
  300. [self waitForDrainDatabaseQueueForRmqManager:manager];
  301. // Check that the file was deleted.
  302. XCTAssertFalse([[NSFileManager defaultManager] fileExistsAtPath:databasePath]);
  303. }
  304. #pragma mark - Private Helpers
  305. - (GtalkDataMessageStanza *)dataMessageWithMessageID:(NSString *)messageID
  306. from:(NSString *)from
  307. data:(NSDictionary *)data {
  308. GtalkDataMessageStanza *stanza = [[GtalkDataMessageStanza alloc] init];
  309. [stanza setId_p:messageID];
  310. [stanza setFrom:from];
  311. [stanza setCategory:kRmqDataMessageCategory];
  312. for (NSString *key in data) {
  313. NSString *val = data[key];
  314. GtalkAppData *appData = [[GtalkAppData alloc] init];
  315. [appData setKey:key];
  316. [appData setValue:val];
  317. [[stanza appDataArray] addObject:appData];
  318. }
  319. return stanza;
  320. }
  321. - (NSString *)createBrokenDatabaseWithName:(NSString *)name {
  322. NSString *databasePath = [FIRMessagingRmqManager pathForDatabaseWithName:name];
  323. NSMutableArray *pathComponents = [[databasePath pathComponents] mutableCopy];
  324. [pathComponents removeLastObject];
  325. NSString *directoryPath = [NSString pathWithComponents:pathComponents];
  326. // Create directory if doesn't exist.
  327. [[NSFileManager defaultManager] createDirectoryAtPath:directoryPath withIntermediateDirectories:YES attributes:nil error:NULL];
  328. // Remove the file if exists.
  329. [[NSFileManager defaultManager] removeItemAtPath:databasePath error:NULL];
  330. NSData *brokenDBFileContent = [@"not a valid DB" dataUsingEncoding:NSUTF8StringEncoding];
  331. [brokenDBFileContent writeToFile:databasePath atomically:YES];
  332. XCTAssertEqualObjects([NSData dataWithContentsOfFile:databasePath], brokenDBFileContent);
  333. return databasePath;
  334. }
  335. @end