FIRExperimentControllerTest.m 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447
  1. // Copyright 2019 Google
  2. //
  3. // Licensed under the Apache License, Version 2.0 (the "License");
  4. // you may not use this file except in compliance with the License.
  5. // You may obtain a copy of the License at
  6. //
  7. // http://www.apache.org/licenses/LICENSE-2.0
  8. //
  9. // Unless required by applicable law or agreed to in writing, software
  10. // distributed under the License is distributed on an "AS IS" BASIS,
  11. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. // See the License for the specific language governing permissions and
  13. // limitations under the License.
  14. #import <XCTest/XCTest.h>
  15. #import "OCMock.h"
  16. #import "FirebaseABTesting/Sources/ABTConditionalUserPropertyController.h"
  17. #import "FirebaseABTesting/Sources/ABTConstants.h"
  18. #import "FirebaseABTesting/Sources/Private/ABTExperimentPayload.h"
  19. #import "FirebaseABTesting/Sources/Public/FirebaseABTesting/FIRExperimentController.h"
  20. #import "FirebaseABTesting/Sources/Public/FirebaseABTesting/FIRLifecycleEvents.h"
  21. #import "FirebaseABTesting/Tests/Unit/ABTFakeFIRAConditionalUserPropertyController.h"
  22. #import "FirebaseABTesting/Tests/Unit/ABTTestUniversalConstants.h"
  23. #import "FirebaseABTesting/Tests/Unit/Utilities/ABTTestUtilities.h"
  24. #import "FirebaseCore/Sources/Private/FirebaseCoreInternal.h"
  25. #import "Interop/Analytics/Public/FIRAnalyticsInterop.h"
  26. extern ABTExperimentPayload *ABTDeserializeExperimentPayload(NSData *payload);
  27. extern NSArray<ABTExperimentPayload *> *ABTExperimentsToSetFromPayloads(
  28. NSArray<NSData *> *payloads,
  29. NSArray<NSDictionary<NSString *, NSString *> *> *experiments,
  30. id<FIRAnalyticsInterop> _Nullable analytics);
  31. extern NSArray *ABTExperimentsToClearFromPayloads(
  32. NSArray<NSData *> *payloads,
  33. NSArray<NSDictionary<NSString *, NSString *> *> *experiments,
  34. id<FIRAnalyticsInterop> _Nullable analytics);
  35. @interface FIRExperimentController (ExposedForTest)
  36. - (void)
  37. updateExperimentConditionalUserPropertiesWithServiceOrigin:(NSString *)origin
  38. events:(FIRLifecycleEvents *)events
  39. policy:
  40. (ABTExperimentPayloadExperimentOverflowPolicy)
  41. policy
  42. lastStartTime:(NSTimeInterval)lastStartTime
  43. payloads:(NSArray<NSData *> *)payloads
  44. completionHandler:
  45. (nullable void (^)(NSError *_Nullable error))
  46. completionHandler;
  47. /// Surface internal initializer to avoid singleton usage during tests.
  48. - (instancetype)initWithAnalytics:(nullable id<FIRAnalyticsInterop>)analytics;
  49. @end
  50. @interface ABTConditionalUserPropertyController (ExposedForTest)
  51. - (void)maxNumberOfExperimentsOfOrigin:(NSString *)origin
  52. completionHandler:(void (^)(int32_t))completionHandler;
  53. - (int32_t)maxNumberOfExperimentsOfOrigin:(NSString *)origin;
  54. - (id)createExperimentFromOrigin:(NSString *)origin
  55. payload:(ABTExperimentPayload *)payload
  56. events:(FIRLifecycleEvents *)events;
  57. - (ABTExperimentPayloadExperimentOverflowPolicy)
  58. overflowPolicyWithPayload:(ABTExperimentPayload *)payload
  59. originalPolicy:(ABTExperimentPayloadExperimentOverflowPolicy)originalPolicy;
  60. @end
  61. @interface FIRExperimentControllerTest : XCTestCase {
  62. FIRExperimentController *_experimentController;
  63. ABTFakeFIRAConditionalUserPropertyController *_fakeController;
  64. id _mockCUPController;
  65. }
  66. @end
  67. @implementation FIRExperimentControllerTest
  68. - (void)setUp {
  69. [super setUp];
  70. _fakeController = [ABTFakeFIRAConditionalUserPropertyController sharedInstance];
  71. id<FIRAnalyticsInterop> fakeAnalytics =
  72. [[FakeAnalytics alloc] initWithFakeController:_fakeController];
  73. _experimentController = [[FIRExperimentController alloc] initWithAnalytics:fakeAnalytics];
  74. ABTConditionalUserPropertyController *controller =
  75. [ABTConditionalUserPropertyController sharedInstanceWithAnalytics:fakeAnalytics];
  76. _mockCUPController = OCMPartialMock(controller);
  77. OCMStub([_mockCUPController maxNumberOfExperimentsOfOrigin:[OCMArg any]]).andReturn(3);
  78. }
  79. - (void)tearDown {
  80. [_fakeController resetExperiments];
  81. [_mockCUPController stopMocking];
  82. [super tearDown];
  83. }
  84. - (void)testDeserializeInvalidPayload {
  85. FIRExperimentController *controller = _experimentController;
  86. XCTAssertNotNil(controller);
  87. NSString *sampleString = @"sample_invalid_payload";
  88. NSData *invalidData = [sampleString dataUsingEncoding:NSUTF8StringEncoding];
  89. XCTAssertNil(ABTDeserializeExperimentPayload(invalidData));
  90. }
  91. - (void)testLifecycleEvents {
  92. FIRLifecycleEvents *events = [[FIRLifecycleEvents alloc] init];
  93. XCTAssertEqualObjects(FIRSetExperimentEventName, events.setExperimentEventName);
  94. XCTAssertEqualObjects(FIRActivateExperimentEventName, events.activateExperimentEventName);
  95. XCTAssertEqualObjects(FIRTimeoutExperimentEventName, events.timeoutExperimentEventName);
  96. XCTAssertEqualObjects(FIRExpireExperimentEventName, events.expireExperimentEventName);
  97. XCTAssertEqualObjects(FIRClearExperimentEventName, events.clearExperimentEventName);
  98. // Should be able to override event name values.
  99. events.setExperimentEventName = @"_new_set_experiment";
  100. XCTAssertEqualObjects(events.setExperimentEventName, @"_new_set_experiment");
  101. events.setExperimentEventName = @"name_without_prefix";
  102. XCTAssertEqualObjects(FIRSetExperimentEventName, events.setExperimentEventName);
  103. events.activateExperimentEventName = @"_new_activate_experiment";
  104. XCTAssertEqualObjects(events.activateExperimentEventName, @"_new_activate_experiment");
  105. events.activateExperimentEventName = @"";
  106. XCTAssertEqualObjects(FIRActivateExperimentEventName, events.activateExperimentEventName);
  107. events.timeoutExperimentEventName = @"__";
  108. XCTAssertEqualObjects(events.timeoutExperimentEventName, @"__");
  109. events.timeoutExperimentEventName = @"name_with_";
  110. XCTAssertEqualObjects(FIRTimeoutExperimentEventName, events.timeoutExperimentEventName);
  111. events.expireExperimentEventName = @"_";
  112. XCTAssertEqualObjects(events.expireExperimentEventName, @"_");
  113. #pragma clang diagnostic push
  114. #pragma clang diagnostic ignored "-Wnonnull"
  115. events.expireExperimentEventName = nil;
  116. #pragma clang diagnostic pop
  117. XCTAssertEqualObjects(FIRExpireExperimentEventName, events.expireExperimentEventName);
  118. events.clearExperimentEventName = @"_new_set_experiment";
  119. XCTAssertEqualObjects(events.clearExperimentEventName, @"_new_set_experiment");
  120. events.clearExperimentEventName = @"";
  121. XCTAssertEqualObjects(FIRClearExperimentEventName, events.clearExperimentEventName);
  122. }
  123. - (void)testSetExperimentWithBadPayload {
  124. [[_mockCUPController reject]
  125. setExperimentWithOrigin:[OCMArg any]
  126. payload:[OCMArg any]
  127. events:[OCMArg any]
  128. policy:ABTExperimentPayloadExperimentOverflowPolicyDiscardOldest];
  129. NSString *sampleString = @"sample_invalid_payload";
  130. NSData *invalidData = [sampleString dataUsingEncoding:NSUTF8StringEncoding];
  131. XCTAssertNil(ABTDeserializeExperimentPayload(invalidData));
  132. }
  133. - (void)testUpdateExperiments {
  134. NSDate *now = [NSDate date];
  135. NSData *payload2Data =
  136. [ABTTestUtilities payloadJSONDataFromFile:@"TestABTPayload2"
  137. modifiedStartTime:[now dateByAddingTimeInterval:1500]];
  138. NSData *payload3Data =
  139. [ABTTestUtilities payloadJSONDataFromFile:@"TestABTPayload3"
  140. modifiedStartTime:[now dateByAddingTimeInterval:900]];
  141. NSData *payload4Data =
  142. [ABTTestUtilities payloadJSONDataFromFile:@"TestABTPayload4"
  143. modifiedStartTime:[now dateByAddingTimeInterval:-900]];
  144. __block BOOL completionHandlerCalled = NO;
  145. FIRLifecycleEvents *events = [[FIRLifecycleEvents alloc] init];
  146. NSArray *payloads = @[ payload2Data, payload3Data, payload4Data ];
  147. [_experimentController
  148. updateExperimentConditionalUserPropertiesWithServiceOrigin:gABTTestOrigin
  149. events:events
  150. policy:
  151. ABTExperimentPayloadExperimentOverflowPolicyDiscardOldest // NOLINT
  152. lastStartTime:[now timeIntervalSince1970]
  153. payloads:payloads
  154. completionHandler:^(NSError *_Nullable error) {
  155. completionHandlerCalled = YES;
  156. }];
  157. XCTAssertEqual([_mockCUPController experimentsWithOrigin:gABTTestOrigin].count, 2);
  158. XCTAssertTrue(completionHandlerCalled);
  159. // Second time update exp_1 no longer exist, should be cleared from experiments.
  160. payloads = @[ payload3Data, payload4Data ];
  161. [_experimentController
  162. updateExperimentConditionalUserPropertiesWithServiceOrigin:gABTTestOrigin
  163. events:events
  164. policy:
  165. ABTExperimentPayloadExperimentOverflowPolicyDiscardOldest // NOLINT
  166. lastStartTime:[now timeIntervalSince1970]
  167. payloads:payloads
  168. completionHandler:nil];
  169. XCTAssertEqual([_mockCUPController experimentsWithOrigin:gABTTestOrigin].count, 1);
  170. }
  171. - (void)testLatestExperimentStartTimestamps {
  172. // Mock incoming payloads
  173. NSMutableArray<NSData *> *payloads = [[NSMutableArray alloc] init];
  174. NSDate *now = [NSDate date];
  175. NSTimeInterval nowInterval = [now timeIntervalSince1970];
  176. NSData *payload2Data = [ABTTestUtilities payloadJSONDataFromFile:@"TestABTPayload2"
  177. modifiedStartTime:now];
  178. [payloads addObject:payload2Data];
  179. NSData *payload3Data =
  180. [ABTTestUtilities payloadJSONDataFromFile:@"TestABTPayload3"
  181. modifiedStartTime:[now dateByAddingTimeInterval:500]];
  182. [payloads addObject:payload3Data];
  183. NSString *sampleString = @"sample_invalid_payload";
  184. NSData *invalidPayload = [sampleString dataUsingEncoding:NSUTF8StringEncoding];
  185. [payloads addObject:invalidPayload];
  186. XCTAssertEqualWithAccuracy(
  187. [now timeIntervalSince1970] + 500,
  188. [_experimentController latestExperimentStartTimestampBetweenTimestamp:nowInterval + 200
  189. andPayloads:payloads],
  190. 1);
  191. XCTAssertEqualWithAccuracy(
  192. [now timeIntervalSince1970] + 1000,
  193. [_experimentController latestExperimentStartTimestampBetweenTimestamp:nowInterval + 1000
  194. andPayloads:payloads],
  195. 1);
  196. XCTAssertEqualWithAccuracy(
  197. [now timeIntervalSince1970] + 500,
  198. [_experimentController latestExperimentStartTimestampBetweenTimestamp:nowInterval - 10000
  199. andPayloads:payloads],
  200. 1);
  201. }
  202. - (void)testExperimentsToSetFromPayloads {
  203. // Mock conditional user property objects in experiments.
  204. NSMutableArray *currentExperiments = [[NSMutableArray alloc] init];
  205. NSDictionary<NSString *, NSString *> *CUP1 = @{@"name" : @"exp_1", @"value" : @"v1"};
  206. [currentExperiments addObject:CUP1];
  207. NSDictionary<NSString *, NSString *> *CUP2 = @{@"name" : @"exp_2", @"value" : @"v200"};
  208. [currentExperiments addObject:CUP2];
  209. NSMutableArray<NSData *> *payloads = [[NSMutableArray alloc] init];
  210. NSData *payload1Data = [ABTTestUtilities payloadJSONDataFromFile:@"TestABTPayload1"
  211. modifiedStartTime:nil];
  212. [payloads addObject:payload1Data];
  213. NSData *payload2Data = [ABTTestUtilities payloadJSONDataFromFile:@"TestABTPayload2"
  214. modifiedStartTime:nil];
  215. [payloads addObject:payload2Data];
  216. NSString *sampleString = @"sample_invalid_payload";
  217. NSData *invalidPayload = [sampleString dataUsingEncoding:NSUTF8StringEncoding];
  218. [payloads addObject:invalidPayload];
  219. NSArray<ABTExperimentPayload *> *experimentsToSet =
  220. ABTExperimentsToSetFromPayloads(payloads, currentExperiments, nil);
  221. XCTAssertEqual(experimentsToSet.count, 1);
  222. ABTExperimentPayload *payloadToAdd = experimentsToSet.firstObject;
  223. XCTAssertEqualObjects(payloadToAdd.experimentId, @"exp_1");
  224. XCTAssertEqualObjects(payloadToAdd.variantId, @"var_1");
  225. }
  226. - (void)testExperimentsToClearFromPayloads {
  227. // Mock conditional user property objects in experiments.
  228. NSMutableArray *currentExperiments = [[NSMutableArray alloc] init];
  229. NSDictionary<NSString *, NSString *> *CUP1 = @{@"name" : @"exp_1", @"value" : @"v1"};
  230. [currentExperiments addObject:CUP1];
  231. NSDictionary<NSString *, NSString *> *CUP2 = @{@"name" : @"exp_2", @"value" : @"v2"};
  232. [currentExperiments addObject:CUP2];
  233. NSMutableArray<NSData *> *payloads = [[NSMutableArray alloc] init];
  234. NSData *payload1Data = [ABTTestUtilities payloadJSONDataFromFile:@"TestABTPayload4"
  235. modifiedStartTime:nil];
  236. [payloads addObject:payload1Data];
  237. NSData *payload2Data = [ABTTestUtilities payloadJSONDataFromFile:@"TestABTPayload5"
  238. modifiedStartTime:nil];
  239. [payloads addObject:payload2Data];
  240. NSString *sampleString = @"sample_invalid_payload";
  241. NSData *invalidPayload = [sampleString dataUsingEncoding:NSUTF8StringEncoding];
  242. [payloads addObject:invalidPayload];
  243. NSArray<NSDictionary<NSString *, NSString *> *> *experimentsToClear =
  244. ABTExperimentsToClearFromPayloads(payloads, currentExperiments, nil);
  245. XCTAssertEqual(experimentsToClear.count, 1);
  246. NSDictionary<NSString *, NSString *> *experimentToRemove = experimentsToClear.firstObject;
  247. XCTAssertEqualObjects(experimentToRemove[@"name"], @"exp_1");
  248. XCTAssertEqualObjects(experimentToRemove[@"value"], @"v1");
  249. }
  250. - (void)testInvalidExperiments {
  251. [[_mockCUPController reject]
  252. setExperimentWithOrigin:[OCMArg any]
  253. payload:[OCMArg any]
  254. events:[OCMArg any]
  255. policy:ABTExperimentPayloadExperimentOverflowPolicyDiscardOldest];
  256. [[_mockCUPController reject]
  257. setExperimentWithOrigin:[OCMArg any]
  258. payload:[OCMArg any]
  259. events:[OCMArg any]
  260. policy:ABTExperimentPayloadExperimentOverflowPolicyDiscardOldest];
  261. OCMStub([_mockCUPController experimentsWithOrigin:gABTTestOrigin]).andReturn(nil);
  262. NSMutableArray<NSData *> *payloads = [[NSMutableArray alloc] init];
  263. __block BOOL completionHandlerWithErrorCalled = NO;
  264. FIRLifecycleEvents *events = [[FIRLifecycleEvents alloc] init];
  265. [_experimentController
  266. updateExperimentConditionalUserPropertiesWithServiceOrigin:gABTTestOrigin
  267. events:events
  268. policy:
  269. ABTExperimentPayloadExperimentOverflowPolicyDiscardOldest // NOLINT
  270. lastStartTime:-1
  271. payloads:payloads
  272. completionHandler:^(NSError *_Nullable error) {
  273. if (error &&
  274. error.code ==
  275. kABTInternalErrorFailedToFetchConditionalUserProperties) {
  276. completionHandlerWithErrorCalled = YES;
  277. }
  278. }];
  279. // Verify completion handler is still called.
  280. XCTAssertTrue(completionHandlerWithErrorCalled);
  281. }
  282. - (void)testValidateRunningExperimentsWithEmptyArray {
  283. NSDate *now = [NSDate date];
  284. NSData *payload2Data =
  285. [ABTTestUtilities payloadJSONDataFromFile:@"TestABTPayload2"
  286. modifiedStartTime:[now dateByAddingTimeInterval:1500]];
  287. NSData *payload3Data =
  288. [ABTTestUtilities payloadJSONDataFromFile:@"TestABTPayload3"
  289. modifiedStartTime:[now dateByAddingTimeInterval:900]];
  290. FIRLifecycleEvents *events = [[FIRLifecycleEvents alloc] init];
  291. NSArray *payloads = @[ payload2Data, payload3Data ];
  292. [_experimentController
  293. updateExperimentConditionalUserPropertiesWithServiceOrigin:gABTTestOrigin
  294. events:events
  295. policy:
  296. ABTExperimentPayloadExperimentOverflowPolicyDiscardOldest // NOLINT
  297. lastStartTime:[now timeIntervalSince1970]
  298. payloads:payloads
  299. completionHandler:nil];
  300. XCTAssertEqual([_mockCUPController experimentsWithOrigin:gABTTestOrigin].count, 2);
  301. [_experimentController validateRunningExperimentsForServiceOrigin:gABTTestOrigin
  302. runningExperimentPayloads:[NSArray array]];
  303. // Expect all experiments have been cleared.
  304. XCTAssertEqual([_mockCUPController experimentsWithOrigin:gABTTestOrigin].count, 0);
  305. }
  306. - (void)testValidateRunningExperimentsClearingOne {
  307. NSDate *now = [NSDate date];
  308. NSData *payload2Data =
  309. [ABTTestUtilities payloadJSONDataFromFile:@"TestABTPayload2"
  310. modifiedStartTime:[now dateByAddingTimeInterval:1500]];
  311. NSData *payload3Data =
  312. [ABTTestUtilities payloadJSONDataFromFile:@"TestABTPayload3"
  313. modifiedStartTime:[now dateByAddingTimeInterval:900]];
  314. FIRLifecycleEvents *events = [[FIRLifecycleEvents alloc] init];
  315. NSArray *payloads = @[ payload2Data, payload3Data ];
  316. [_experimentController
  317. updateExperimentConditionalUserPropertiesWithServiceOrigin:gABTTestOrigin
  318. events:events
  319. policy:
  320. ABTExperimentPayloadExperimentOverflowPolicyDiscardOldest // NOLINT
  321. lastStartTime:[now timeIntervalSince1970]
  322. payloads:payloads
  323. completionHandler:nil];
  324. XCTAssertEqual([_mockCUPController experimentsWithOrigin:gABTTestOrigin].count, 2);
  325. ABTExperimentPayload *validatingPayload2 =
  326. [ABTTestUtilities payloadFromTestFilename:@"TestABTPayload2"];
  327. [_experimentController validateRunningExperimentsForServiceOrigin:gABTTestOrigin
  328. runningExperimentPayloads:@[ validatingPayload2 ]];
  329. // Expect no experiments have been cleared.
  330. XCTAssertEqual([_mockCUPController experimentsWithOrigin:gABTTestOrigin].count, 1);
  331. }
  332. - (void)testValidateRunningExperimentsKeepingAll {
  333. NSDate *now = [NSDate date];
  334. NSData *payload2Data =
  335. [ABTTestUtilities payloadJSONDataFromFile:@"TestABTPayload2"
  336. modifiedStartTime:[now dateByAddingTimeInterval:1500]];
  337. NSData *payload3Data =
  338. [ABTTestUtilities payloadJSONDataFromFile:@"TestABTPayload3"
  339. modifiedStartTime:[now dateByAddingTimeInterval:900]];
  340. FIRLifecycleEvents *events = [[FIRLifecycleEvents alloc] init];
  341. NSArray *payloads = @[ payload2Data, payload3Data ];
  342. [_experimentController
  343. updateExperimentConditionalUserPropertiesWithServiceOrigin:gABTTestOrigin
  344. events:events
  345. policy:
  346. ABTExperimentPayloadExperimentOverflowPolicyDiscardOldest // NOLINT
  347. lastStartTime:[now timeIntervalSince1970]
  348. payloads:payloads
  349. completionHandler:nil];
  350. XCTAssertEqual([_mockCUPController experimentsWithOrigin:gABTTestOrigin].count, 2);
  351. ABTExperimentPayload *validatingPayload2 =
  352. [ABTTestUtilities payloadFromTestFilename:@"TestABTPayload2"];
  353. ABTExperimentPayload *validatingPayload3 =
  354. [ABTTestUtilities payloadFromTestFilename:@"TestABTPayload3"];
  355. [_experimentController
  356. validateRunningExperimentsForServiceOrigin:gABTTestOrigin
  357. runningExperimentPayloads:@[ validatingPayload2, validatingPayload3 ]];
  358. // Expect no experiments have been cleared.
  359. XCTAssertEqual([_mockCUPController experimentsWithOrigin:gABTTestOrigin].count, 2);
  360. }
  361. - (void)testActivateExperiment {
  362. ABTExperimentPayload *activeExperiment =
  363. [ABTTestUtilities payloadFromTestFilename:@"TestABTPayload1"];
  364. [_experimentController activateExperiment:activeExperiment forServiceOrigin:gABTTestOrigin];
  365. NSArray *experiments = [_mockCUPController experimentsWithOrigin:gABTTestOrigin];
  366. NSDictionary *userPropertyForExperiment = [experiments firstObject];
  367. // Verify that the triggerEventName is cleared, making this experiment active.
  368. XCTAssertNil([userPropertyForExperiment valueForKeyPath:@"triggerEventName"]);
  369. }
  370. @end