FIRMessagingRmqManagerTest.m 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326
  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 "Firebase/Messaging/FIRMessagingPersistentSyncMessage.h"
  18. #import "Firebase/Messaging/FIRMessagingRmqManager.h"
  19. #import "Firebase/Messaging/FIRMessagingUtilities.h"
  20. #import "Firebase/Messaging/Protos/GtalkCore.pbobjc.h"
  21. static NSString *const kRmqDatabaseName = @"rmq-test-db";
  22. static NSString *const kRmqDataMessageCategory = @"com.google.gcm-rmq-test";
  23. static const NSTimeInterval kAsyncTestTimout = 0.5;
  24. @interface FIRMessagingRmqManager (ExposedForTest)
  25. - (void)removeDatabase;
  26. @end
  27. @interface FIRMessagingRmqManagerTest : XCTestCase
  28. @property(nonatomic, readwrite, strong) FIRMessagingRmqManager *rmqManager;
  29. @end
  30. @implementation FIRMessagingRmqManagerTest
  31. - (void)setUp {
  32. [super setUp];
  33. // Make sure we start off with a clean state each time
  34. _rmqManager = [[FIRMessagingRmqManager alloc] initWithDatabaseName:kRmqDatabaseName];
  35. }
  36. - (void)tearDown {
  37. [self.rmqManager removeDatabase];
  38. [super tearDown];
  39. }
  40. /**
  41. * Add s2d messages with different RMQ-ID's to the RMQ. Fetch the messages
  42. * and verify that all messages were successfully saved.
  43. */
  44. - (void)testSavingS2dMessages {
  45. NSArray *messageIDs = @[ @"message1", @"message2", @"123456" ];
  46. for (NSString *messageID in messageIDs) {
  47. [self.rmqManager saveS2dMessageWithRmqId:messageID];
  48. }
  49. NSArray *rmqMessages = [self.rmqManager unackedS2dRmqIds];
  50. XCTAssertEqual(messageIDs.count, rmqMessages.count);
  51. for (NSString *messageID in rmqMessages) {
  52. XCTAssertTrue([messageIDs containsObject:messageID]);
  53. }
  54. }
  55. /**
  56. * Add s2d messages with different RMQ-ID's to the RMQ. Delete some of the
  57. * messages stored, assuming we received a server ACK for them. The remaining
  58. * messages should be fetched successfully.
  59. */
  60. - (void)testDeletingS2dMessages {
  61. NSArray *addMessages = @[ @"message1", @"message2", @"message3", @"message4"];
  62. for (NSString *messageID in addMessages) {
  63. [self.rmqManager saveS2dMessageWithRmqId:messageID];
  64. }
  65. NSArray *removeMessages = @[ addMessages[1], addMessages[3] ];
  66. [self.rmqManager removeS2dIds:removeMessages];
  67. NSArray *remainingMessages = [self.rmqManager unackedS2dRmqIds];
  68. XCTAssertEqual(2, remainingMessages.count);
  69. XCTAssertTrue([remainingMessages containsObject:addMessages[0]]);
  70. XCTAssertTrue([remainingMessages containsObject:addMessages[2]]);
  71. }
  72. /**
  73. * Test deleting a s2d message that is not in the persistent store. This shouldn't
  74. * crash or alter the valid contents of the RMQ store.
  75. */
  76. - (void)testDeletingInvalidS2dMessage {
  77. NSString *validMessageID = @"validMessage123";
  78. [self.rmqManager saveS2dMessageWithRmqId:validMessageID];
  79. NSString *invalidMessageID = @"invalidMessage123";
  80. [self.rmqManager removeS2dIds:@[invalidMessageID]];
  81. NSArray *remainingMessages = [self.rmqManager unackedS2dRmqIds];
  82. XCTAssertEqual(1, remainingMessages.count);
  83. XCTAssertEqualObjects(validMessageID, remainingMessages[0]);
  84. }
  85. /**
  86. * Test that outgoing RMQ messages are correctly saved
  87. */
  88. - (void)testOutgoingRmqWithValidMessages {
  89. XCTestExpectation *expectation = [self expectationWithDescription:@"Scan outgoing messages is complete"];
  90. NSString *from = @"rmq-test";
  91. [self.rmqManager loadRmqId];
  92. GtalkDataMessageStanza *message1 = [self dataMessageWithMessageID:@"message1"
  93. from:from
  94. data:nil];
  95. // should successfully save the message to RMQ
  96. [self.rmqManager saveRmqMessage:message1 withCompletionHandler:^(BOOL success) {
  97. XCTAssertTrue(success);
  98. GtalkDataMessageStanza *message2 = [self dataMessageWithMessageID:@"message2"
  99. from:from
  100. data:nil];
  101. // should successfully save the second message to RMQ
  102. [self.rmqManager saveRmqMessage:message2 withCompletionHandler:^(BOOL success) {
  103. XCTAssertTrue(success);
  104. // message1 should have RMQ-ID = 2, message2 = 3
  105. [self.rmqManager scanWithRmqMessageHandler:^(NSDictionary *messages) {
  106. XCTAssertTrue([[messages allKeys] containsObject:@(2)]);
  107. XCTAssertTrue([[messages allKeys] containsObject:@(3)]);
  108. [expectation fulfill];
  109. }];
  110. }];
  111. }];
  112. [self waitForExpectationsWithTimeout:kAsyncTestTimout handler:nil];
  113. }
  114. /**
  115. * Test that an outgoing message with different properties is correctly saved to the RMQ.
  116. */
  117. - (void)testOutgoingDataMessageIsCorrectlySaved {
  118. XCTestExpectation *expectation = [self expectationWithDescription:@"Messages are saved"];
  119. NSString *from = @"rmq-test";
  120. NSString *messageID = @"message123";
  121. NSString *to = @"to-senderID-123";
  122. int32_t ttl = 2400;
  123. NSString *registrationToken = @"registration-token";
  124. NSDictionary *data = @{
  125. @"hello" : @"world",
  126. @"count" : @"2",
  127. };
  128. [self.rmqManager loadRmqId];
  129. GtalkDataMessageStanza *message = [self dataMessageWithMessageID:messageID
  130. from:from
  131. data:data];
  132. [message setTo:to];
  133. [message setTtl:ttl];
  134. [message setRegId:registrationToken];
  135. // should successfully save the message to RMQ
  136. [self.rmqManager saveRmqMessage:message withCompletionHandler:^(BOOL success) {
  137. XCTAssertTrue(success);
  138. [self.rmqManager scanWithRmqMessageHandler:^(NSDictionary *messages) {
  139. for (NSString *rmqID in messages) {
  140. GtalkDataMessageStanza *stanza = (GtalkDataMessageStanza *)messages[rmqID];
  141. XCTAssertEqualObjects(from, stanza.from);
  142. XCTAssertEqualObjects(messageID, stanza.id_p);
  143. XCTAssertEqualObjects(to, stanza.to);
  144. XCTAssertEqualObjects(registrationToken, stanza.regId);
  145. XCTAssertEqual(ttl, stanza.ttl);
  146. NSMutableDictionary *d = [NSMutableDictionary dictionary];
  147. for (GtalkAppData *appData in stanza.appDataArray) {
  148. d[appData.key] = appData.value;
  149. }
  150. XCTAssertTrue([data isEqualToDictionary:d]);
  151. }
  152. [expectation fulfill];
  153. }];
  154. }];
  155. [self waitForExpectationsWithTimeout:kAsyncTestTimout handler:nil];
  156. }
  157. /**
  158. * Test D2S messages being deleted from RMQ.
  159. */
  160. - (void)testDeletingD2SMessagesFromRMQ {
  161. XCTestExpectation *expectation = [self expectationWithDescription:@"Messages are deleted"];
  162. NSString *message1 = @"message123";
  163. NSString *ackedMessage = @"message234";
  164. NSString *from = @"from-rmq-test";
  165. GtalkDataMessageStanza *stanza1 = [self dataMessageWithMessageID:message1 from:from data:nil];
  166. GtalkDataMessageStanza *stanza2 = [self dataMessageWithMessageID:ackedMessage
  167. from:from
  168. data:nil];
  169. [self.rmqManager saveRmqMessage:stanza1 withCompletionHandler:^(BOOL success) {
  170. XCTAssertTrue(success);
  171. [self.rmqManager saveRmqMessage:stanza2 withCompletionHandler:^(BOOL success) {
  172. XCTAssertTrue(success);
  173. __block int64_t ackedMessageRmqID = -1;
  174. [self.rmqManager scanWithRmqMessageHandler:^(NSDictionary *messages) {
  175. for (NSString *rmqID in messages) {
  176. GtalkDataMessageStanza *stanza = (GtalkDataMessageStanza *)messages[rmqID];
  177. if ([stanza.id_p isEqualToString:ackedMessage]) {
  178. ackedMessageRmqID = rmqID.intValue;
  179. // should be a valid RMQ ID
  180. XCTAssertTrue(ackedMessageRmqID > 0);
  181. // delete the acked message
  182. NSString *rmqIDString = [NSString stringWithFormat:@"%lld", ackedMessageRmqID];
  183. [self.rmqManager removeRmqMessagesWithRmqIds:@[rmqIDString]];
  184. // should only have one message in the d2s RMQ
  185. [self.rmqManager scanWithRmqMessageHandler:^(NSDictionary *messages) {
  186. for (NSString *rmqID in messages) {
  187. GtalkDataMessageStanza *stanza2 = (GtalkDataMessageStanza *)messages[rmqID];
  188. // the acked message was queued later so should have
  189. // rmqID = ackedMessageRMQID - 1
  190. XCTAssertEqual(ackedMessageRmqID - 1, rmqID.intValue);
  191. XCTAssertEqual(message1, stanza2.id_p);
  192. }
  193. [expectation fulfill];
  194. }];
  195. }
  196. }
  197. }];
  198. }];
  199. }];
  200. [self waitForExpectationsWithTimeout:kAsyncTestTimout handler:nil];
  201. }
  202. /**
  203. * Test saving a sync message to SYNC_RMQ.
  204. */
  205. - (void)testSavingSyncMessage {
  206. NSString *rmqID = @"fake-rmq-id-1";
  207. int64_t expirationTime = FIRMessagingCurrentTimestampInSeconds() + 1;
  208. [self.rmqManager saveSyncMessageWithRmqID:rmqID
  209. expirationTime:expirationTime
  210. apnsReceived:YES
  211. mcsReceived:NO];
  212. FIRMessagingPersistentSyncMessage *persistentMessage = [self.rmqManager querySyncMessageWithRmqID:rmqID];
  213. XCTAssertEqual(persistentMessage.expirationTime, expirationTime);
  214. XCTAssertTrue(persistentMessage.apnsReceived);
  215. XCTAssertFalse(persistentMessage.mcsReceived);
  216. }
  217. /**
  218. * Test updating a sync message initially received via MCS, now being received via APNS.
  219. */
  220. - (void)testUpdateMessageReceivedViaAPNS {
  221. NSString *rmqID = @"fake-rmq-id-1";
  222. int64_t expirationTime = FIRMessagingCurrentTimestampInSeconds() + 1;
  223. [self.rmqManager saveSyncMessageWithRmqID:rmqID
  224. expirationTime:expirationTime
  225. apnsReceived:NO
  226. mcsReceived:YES];
  227. // Message was now received via APNS
  228. [self.rmqManager updateSyncMessageViaAPNSWithRmqID:rmqID];
  229. FIRMessagingPersistentSyncMessage *persistentMessage = [self.rmqManager querySyncMessageWithRmqID:rmqID];
  230. XCTAssertTrue(persistentMessage.apnsReceived);
  231. XCTAssertTrue(persistentMessage.mcsReceived);
  232. }
  233. /**
  234. * Test updating a sync message initially received via APNS, now being received via MCS.
  235. */
  236. - (void)testUpdateMessageReceivedViaMCS {
  237. NSString *rmqID = @"fake-rmq-id-1";
  238. int64_t expirationTime = FIRMessagingCurrentTimestampInSeconds() + 1;
  239. [self.rmqManager saveSyncMessageWithRmqID:rmqID
  240. expirationTime:expirationTime
  241. apnsReceived:YES
  242. mcsReceived:NO];
  243. // Message was now received via APNS
  244. [self.rmqManager updateSyncMessageViaMCSWithRmqID:rmqID];
  245. FIRMessagingPersistentSyncMessage *persistentMessage = [self.rmqManager querySyncMessageWithRmqID:rmqID];
  246. XCTAssertTrue(persistentMessage.apnsReceived);
  247. XCTAssertTrue(persistentMessage.mcsReceived);
  248. }
  249. /**
  250. * Test deleting sync messages from SYNC_RMQ.
  251. */
  252. - (void)testDeleteSyncMessage {
  253. NSString *rmqID = @"fake-rmq-id-1";
  254. int64_t expirationTime = FIRMessagingCurrentTimestampInSeconds() + 1;
  255. [self.rmqManager saveSyncMessageWithRmqID:rmqID
  256. expirationTime:expirationTime
  257. apnsReceived:YES
  258. mcsReceived:NO];
  259. XCTAssertNotNil([self.rmqManager querySyncMessageWithRmqID:rmqID]);
  260. // should successfully delete the message
  261. [self.rmqManager deleteSyncMessageWithRmqID:rmqID];
  262. XCTAssertNil([self.rmqManager querySyncMessageWithRmqID:rmqID]);
  263. }
  264. #pragma mark - Private Helpers
  265. - (GtalkDataMessageStanza *)dataMessageWithMessageID:(NSString *)messageID
  266. from:(NSString *)from
  267. data:(NSDictionary *)data {
  268. GtalkDataMessageStanza *stanza = [[GtalkDataMessageStanza alloc] init];
  269. [stanza setId_p:messageID];
  270. [stanza setFrom:from];
  271. [stanza setCategory:kRmqDataMessageCategory];
  272. for (NSString *key in data) {
  273. NSString *val = data[key];
  274. GtalkAppData *appData = [[GtalkAppData alloc] init];
  275. [appData setKey:key];
  276. [appData setValue:val];
  277. [[stanza appDataArray] addObject:appData];
  278. }
  279. return stanza;
  280. }
  281. @end