// Copyright 2012 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#import "ios/chrome/browser/download/model/installation_notifier.h"
#import "ios/chrome/browser/download/model/installation_notifier+Testing.h"
#import <UIKit/UIKit.h>
#import <stdint.h>
#import "base/ios/block_types.h"
#import "base/task/current_thread.h"
#import "ios/web/public/test/web_task_environment.h"
#import "net/base/backoff_entry.h"
#import "testing/platform_test.h"
#import "third_party/ocmock/OCMock/OCMock.h"
@interface FakeDispatcher : NSObject <DispatcherProtocol>
- (int64_t)lastDelayInNSec;
@end
@implementation FakeDispatcher {
int _dispatchCount;
int64_t _lastDelayInNSec;
NSMutableDictionary* _blocks;
}
- (instancetype)init {
if ((self = [super init]))
_blocks = [[NSMutableDictionary alloc] init];
return self;
}
#pragma mark - Testing methods
- (void)executeAfter:(int)dispatchCount block:(ProceduralBlock)block {
[_blocks setObject:[block copy]
forKey:[NSNumber numberWithInt:dispatchCount]];
}
- (int64_t)lastDelayInNSec {
return _lastDelayInNSec;
}
#pragma mark - DispatcherProtocol
- (void)dispatchAfter:(int64_t)delayInNSec withBlock:(dispatch_block_t)block {
_lastDelayInNSec = delayInNSec;
void (^blockToCallForThisIteration)(void) =
[_blocks objectForKey:[NSNumber numberWithInt:_dispatchCount]];
if (blockToCallForThisIteration)
blockToCallForThisIteration();
_dispatchCount++;
block();
}
@end
@interface MockNotificationReceiver : NSObject
@end
@implementation MockNotificationReceiver {
int _notificationCount;
}
- (int)notificationCount {
return _notificationCount;
}
- (void)receivedNotification {
_notificationCount++;
}
@end
namespace {
class InstallationNotifierTest : public PlatformTest {
protected:
void SetUp() override {
installationNotifier_ = [InstallationNotifier sharedInstance];
FakeDispatcher* dispatcher = [[FakeDispatcher alloc] init];
dispatcher_ = dispatcher;
notificationReceiver1_ = ([[MockNotificationReceiver alloc] init]);
notificationReceiver2_ = ([[MockNotificationReceiver alloc] init]);
application_ = OCMClassMock([UIApplication class]);
OCMStub([application_ sharedApplication]).andReturn(application_);
[installationNotifier_ setDispatcher:dispatcher_];
}
~InstallationNotifierTest() override {
[installationNotifier_ resetDispatcher];
[application_ stopMocking];
}
void VerifyDelay(int pollingIteration) {
double delayInMSec = [dispatcher_ lastDelayInNSec] / NSEC_PER_MSEC;
double initialDelayInMSec =
[installationNotifier_ backOffPolicy]->initial_delay_ms;
double multiplyFactor =
[installationNotifier_ backOffPolicy]->multiply_factor;
double expectedDelayInMSec =
initialDelayInMSec * pow(multiplyFactor, pollingIteration);
double jitter = [installationNotifier_ backOffPolicy]->jitter_factor;
EXPECT_NEAR(delayInMSec, expectedDelayInMSec,
50 + jitter * expectedDelayInMSec);
}
web::WebTaskEnvironment task_environment_;
__weak InstallationNotifier* installationNotifier_;
__weak FakeDispatcher* dispatcher_;
MockNotificationReceiver* notificationReceiver1_;
MockNotificationReceiver* notificationReceiver2_;
id application_;
};
TEST_F(InstallationNotifierTest, RegisterWithAppAlreadyInstalled) {
OCMStub([application_ canOpenURL:[OCMArg any]]).andReturn(YES);
[installationNotifier_
registerForInstallationNotifications:notificationReceiver1_
withSelector:@selector(receivedNotification)
forScheme:@"foo-scheme"];
EXPECT_EQ(1, [notificationReceiver1_ notificationCount]);
[installationNotifier_
registerForInstallationNotifications:notificationReceiver1_
withSelector:@selector(receivedNotification)
forScheme:@"foo-scheme"];
EXPECT_EQ(2, [notificationReceiver1_ notificationCount]);
}
TEST_F(InstallationNotifierTest, RegisterWithAppInstalledAfterSomeTime) {
[dispatcher_
executeAfter:10
block:^{
OCMStub([application_ canOpenURL:[OCMArg any]]).andReturn(YES);
}];
[installationNotifier_
registerForInstallationNotifications:notificationReceiver1_
withSelector:@selector(receivedNotification)
forScheme:@"foo-scheme"];
EXPECT_EQ(1, [notificationReceiver1_ notificationCount]);
}
TEST_F(InstallationNotifierTest, RegisterForTwoInstallations) {
[dispatcher_
executeAfter:10
block:^{
OCMStub([application_ canOpenURL:[OCMArg any]]).andReturn(YES);
}];
[installationNotifier_
registerForInstallationNotifications:notificationReceiver1_
withSelector:@selector(receivedNotification)
forScheme:@"foo-scheme"
startPolling:NO];
[installationNotifier_
registerForInstallationNotifications:notificationReceiver2_
withSelector:@selector(receivedNotification)
forScheme:@"foo-scheme"
startPolling:NO];
[installationNotifier_
registerForInstallationNotifications:notificationReceiver2_
withSelector:@selector(receivedNotification)
forScheme:@"bar-scheme"
startPolling:NO];
[installationNotifier_ dispatchInstallationNotifierBlock];
EXPECT_EQ(1, [notificationReceiver1_ notificationCount]);
EXPECT_EQ(2, [notificationReceiver2_ notificationCount]);
}
TEST_F(InstallationNotifierTest, RegisterAndThenUnregister) {
OCMStub([application_ canOpenURL:[OCMArg any]]).andReturn(NO);
[dispatcher_ executeAfter:10
block:^{
[installationNotifier_
unregisterForNotifications:notificationReceiver1_];
}];
[installationNotifier_
registerForInstallationNotifications:notificationReceiver1_
withSelector:@selector(receivedNotification)
forScheme:@"foo-scheme"];
EXPECT_EQ(0, [notificationReceiver1_ notificationCount]);
}
TEST_F(InstallationNotifierTest, TestExponentialBackoff) {
OCMStub([application_ canOpenURL:[OCMArg any]]).andReturn(NO);
// Making sure that delay is multiplied by `multiplyFactor` every time.
[dispatcher_ executeAfter:0
block:^{
VerifyDelay(0);
}];
[dispatcher_ executeAfter:1
block:^{
VerifyDelay(1);
}];
[dispatcher_ executeAfter:2
block:^{
VerifyDelay(2);
}];
// Registering for the installation of another application and making sure
// that the delay is reset to the initial delay.
[dispatcher_
executeAfter:3
block:^{
VerifyDelay(3);
[installationNotifier_
registerForInstallationNotifications:notificationReceiver1_
withSelector:@selector
(receivedNotification)
forScheme:@"bar-scheme"
startPolling:NO];
}];
[dispatcher_ executeAfter:4
block:^{
VerifyDelay(0);
}];
[dispatcher_ executeAfter:5
block:^{
VerifyDelay(1);
[installationNotifier_
unregisterForNotifications:notificationReceiver1_];
}];
[installationNotifier_
registerForInstallationNotifications:notificationReceiver1_
withSelector:@selector(receivedNotification)
forScheme:@"foo-scheme"];
}
TEST_F(InstallationNotifierTest, TestThatEmptySchemeDoesntCrashChrome) {
[installationNotifier_
registerForInstallationNotifications:notificationReceiver1_
withSelector:@selector(receivedNotification)
forScheme:nil];
[installationNotifier_ unregisterForNotifications:notificationReceiver1_];
}
} // namespace