| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276 |
- /*
- * Copyright 2018 Google
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- #import "GDTCORLibrary/Private/GDTCORUploadCoordinator.h"
- #import <GoogleDataTransport/GDTCORAssert.h>
- #import <GoogleDataTransport/GDTCORClock.h>
- #import <GoogleDataTransport/GDTCORConsoleLogger.h>
- #import <GoogleDataTransport/GDTCORReachability.h>
- #import "GDTCORLibrary/Private/GDTCORRegistrar_Private.h"
- #import "GDTCORLibrary/Private/GDTCORStorage.h"
- @implementation GDTCORUploadCoordinator
- + (instancetype)sharedInstance {
- static GDTCORUploadCoordinator *sharedUploader;
- static dispatch_once_t onceToken;
- dispatch_once(&onceToken, ^{
- sharedUploader = [[GDTCORUploadCoordinator alloc] init];
- [sharedUploader startTimer];
- });
- return sharedUploader;
- }
- - (instancetype)init {
- self = [super init];
- if (self) {
- _coordinationQueue =
- dispatch_queue_create("com.google.GDTCORUploadCoordinator", DISPATCH_QUEUE_SERIAL);
- _registrar = [GDTCORRegistrar sharedInstance];
- _timerInterval = 30 * NSEC_PER_SEC;
- _timerLeeway = 5 * NSEC_PER_SEC;
- _targetToInFlightPackages = [[NSMutableDictionary alloc] init];
- }
- return self;
- }
- - (void)forceUploadForTarget:(GDTCORTarget)target {
- dispatch_async(_coordinationQueue, ^{
- GDTCORLogDebug("Forcing an upload of target %ld", (long)target);
- GDTCORUploadConditions conditions = [self uploadConditions];
- conditions |= GDTCORUploadConditionHighPriority;
- [self uploadTargets:@[ @(target) ] conditions:conditions];
- });
- }
- #pragma mark - Property overrides
- // GDTCORStorage and GDTCORUploadCoordinator +sharedInstance methods call each other, so this breaks
- // the loop.
- - (GDTCORStorage *)storage {
- if (!_storage) {
- _storage = [GDTCORStorage sharedInstance];
- }
- return _storage;
- }
- #pragma mark - Private helper methods
- /** Starts a timer that checks whether or not events can be uploaded at regular intervals. It will
- * check the next-upload clocks of all targets to determine if an upload attempt can be made.
- */
- - (void)startTimer {
- dispatch_sync(_coordinationQueue, ^{
- self->_timer =
- dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, 0, 0, self->_coordinationQueue);
- dispatch_source_set_timer(self->_timer, DISPATCH_TIME_NOW, self->_timerInterval,
- self->_timerLeeway);
- dispatch_source_set_event_handler(self->_timer, ^{
- if (![[GDTCORApplication sharedApplication] isRunningInBackground]) {
- GDTCORUploadConditions conditions = [self uploadConditions];
- GDTCORLogDebug("%@", @"Upload timer fired");
- [self uploadTargets:[self.registrar.targetToUploader allKeys] conditions:conditions];
- }
- });
- GDTCORLogDebug("%@", @"Upload timer started");
- dispatch_resume(self->_timer);
- });
- }
- /** Stops the currently running timer. */
- - (void)stopTimer {
- if (_timer) {
- dispatch_source_cancel(_timer);
- }
- }
- /** Triggers the uploader implementations for the given targets to upload.
- *
- * @param targets An array of targets to trigger.
- * @param conditions The set of upload conditions.
- */
- - (void)uploadTargets:(NSArray<NSNumber *> *)targets conditions:(GDTCORUploadConditions)conditions {
- dispatch_async(_coordinationQueue, ^{
- if ((conditions & GDTCORUploadConditionNoNetwork) == GDTCORUploadConditionNoNetwork) {
- return;
- }
- for (NSNumber *target in targets) {
- // Don't trigger uploads for targets that have an in-flight package already.
- if (self->_targetToInFlightPackages[target]) {
- GDTCORLogDebug("Target %@ will not upload, there's an upload in flight", target);
- continue;
- }
- // Ask the uploader if they can upload and do so, if it can.
- id<GDTCORUploader> uploader = self.registrar.targetToUploader[target];
- if ([uploader readyToUploadTarget:target.intValue conditions:conditions]) {
- id<GDTCORPrioritizer> prioritizer = self.registrar.targetToPrioritizer[target];
- GDTCORUploadPackage *package = [prioritizer uploadPackageWithTarget:target.intValue
- conditions:conditions];
- if (package.events.count) {
- self->_targetToInFlightPackages[target] = package;
- GDTCORLogDebug("Package of %ld events is being handed over to an uploader",
- (long)package.events.count);
- [uploader uploadPackage:package];
- } else {
- [package completeDelivery];
- }
- }
- GDTCORLogDebug("Target %@ is not ready to upload", target);
- }
- });
- }
- /** Returns the current upload conditions after making determinations about the network connection.
- *
- * @return The current upload conditions.
- */
- - (GDTCORUploadConditions)uploadConditions {
- #if TARGET_OS_WATCH
- return GDTCORUploadConditionNoNetwork;
- #else
- SCNetworkReachabilityFlags currentFlags = [GDTCORReachability currentFlags];
- BOOL reachable =
- (currentFlags & kSCNetworkReachabilityFlagsReachable) == kSCNetworkReachabilityFlagsReachable;
- BOOL connectionRequired = (currentFlags & kSCNetworkReachabilityFlagsConnectionRequired) ==
- kSCNetworkReachabilityFlagsConnectionRequired;
- BOOL networkConnected = reachable && !connectionRequired;
- if (!networkConnected) {
- return GDTCORUploadConditionNoNetwork;
- }
- BOOL isWWAN = GDTCORReachabilityFlagsContainWWAN(currentFlags);
- if (isWWAN) {
- return GDTCORUploadConditionMobileData;
- } else {
- return GDTCORUploadConditionWifiData;
- }
- #endif
- }
- #pragma mark - NSSecureCoding support
- /** The NSKeyedCoder key for the targetToInFlightPackages property. */
- static NSString *const ktargetToInFlightPackagesKey =
- @"GDTCORUploadCoordinatortargetToInFlightPackages";
- + (BOOL)supportsSecureCoding {
- return YES;
- }
- - (instancetype)initWithCoder:(NSCoder *)aDecoder {
- GDTCORUploadCoordinator *sharedCoordinator = [GDTCORUploadCoordinator sharedInstance];
- dispatch_sync(sharedCoordinator->_coordinationQueue, ^{
- @try {
- NSSet *classes =
- [NSSet setWithObjects:[NSMutableDictionary class], [GDTCORUploadPackage class], nil];
- sharedCoordinator->_targetToInFlightPackages =
- [aDecoder decodeObjectOfClasses:classes forKey:ktargetToInFlightPackagesKey];
- } @catch (NSException *exception) {
- sharedCoordinator->_targetToInFlightPackages = [NSMutableDictionary dictionary];
- }
- });
- return sharedCoordinator;
- }
- - (void)encodeWithCoder:(NSCoder *)aCoder {
- dispatch_sync(_coordinationQueue, ^{
- // All packages that have been given to uploaders need to be tracked so that their expiration
- // timers can be called.
- if (self->_targetToInFlightPackages.count > 0) {
- [aCoder encodeObject:self->_targetToInFlightPackages forKey:ktargetToInFlightPackagesKey];
- }
- });
- }
- #pragma mark - GDTCORLifecycleProtocol
- - (void)appWillForeground:(GDTCORApplication *)app {
- // Not entirely thread-safe, but it should be fine.
- [self startTimer];
- }
- - (void)appWillBackground:(GDTCORApplication *)app {
- // Should be thread-safe. If it ends up not being, put this in a dispatch_sync.
- [self stopTimer];
- }
- - (void)appWillTerminate:(GDTCORApplication *)application {
- dispatch_sync(_coordinationQueue, ^{
- [self stopTimer];
- });
- }
- #pragma mark - GDTCORUploadPackageProtocol
- - (void)packageDelivered:(GDTCORUploadPackage *)package successful:(BOOL)successful {
- if (!_coordinationQueue) {
- return;
- }
- dispatch_async(_coordinationQueue, ^{
- NSNumber *targetNumber = @(package.target);
- NSMutableDictionary<NSNumber *, GDTCORUploadPackage *> *targetToInFlightPackages =
- self->_targetToInFlightPackages;
- GDTCORRegistrar *registrar = self->_registrar;
- if (targetToInFlightPackages) {
- [targetToInFlightPackages removeObjectForKey:targetNumber];
- }
- NSSet<GDTCOREvent *> *packageEvents = [package.events copy];
- if (registrar) {
- id<GDTCORPrioritizer> prioritizer = registrar.targetToPrioritizer[targetNumber];
- if (!prioritizer) {
- GDTCORLogError(GDTCORMCEPrioritizerError,
- @"A prioritizer should be registered for this target: %@", targetNumber);
- }
- if ([prioritizer respondsToSelector:@selector(packageDelivered:successful:)]) {
- [prioritizer packageDelivered:[package copy] successful:successful];
- }
- }
- if (successful && packageEvents.count) {
- [self.storage removeEvents:packageEvents];
- }
- });
- }
- - (void)packageExpired:(GDTCORUploadPackage *)package {
- if (!_coordinationQueue) {
- return;
- }
- dispatch_async(_coordinationQueue, ^{
- NSNumber *targetNumber = @(package.target);
- NSMutableDictionary<NSNumber *, GDTCORUploadPackage *> *targetToInFlightPackages =
- self->_targetToInFlightPackages;
- GDTCORRegistrar *registrar = self->_registrar;
- if (targetToInFlightPackages) {
- [targetToInFlightPackages removeObjectForKey:targetNumber];
- }
- if (registrar) {
- id<GDTCORPrioritizer> prioritizer = registrar.targetToPrioritizer[targetNumber];
- id<GDTCORUploader> uploader = registrar.targetToUploader[targetNumber];
- if ([prioritizer respondsToSelector:@selector(packageExpired:)]) {
- [prioritizer packageExpired:package];
- }
- if ([uploader respondsToSelector:@selector(packageExpired:)]) {
- [uploader packageExpired:package];
- }
- }
- });
- }
- @end
|