GDTCORUploadCoordinator.m 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266
  1. /*
  2. * Copyright 2018 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 "GDTCORLibrary/Private/GDTCORUploadCoordinator.h"
  17. #import <GoogleDataTransport/GDTCORAssert.h>
  18. #import <GoogleDataTransport/GDTCORClock.h>
  19. #import <GoogleDataTransport/GDTCORConsoleLogger.h>
  20. #import "GDTCORLibrary/Private/GDTCORReachability.h"
  21. #import "GDTCORLibrary/Private/GDTCORRegistrar_Private.h"
  22. #import "GDTCORLibrary/Private/GDTCORStorage.h"
  23. @implementation GDTCORUploadCoordinator
  24. + (instancetype)sharedInstance {
  25. static GDTCORUploadCoordinator *sharedUploader;
  26. static dispatch_once_t onceToken;
  27. dispatch_once(&onceToken, ^{
  28. sharedUploader = [[GDTCORUploadCoordinator alloc] init];
  29. [sharedUploader startTimer];
  30. });
  31. return sharedUploader;
  32. }
  33. - (instancetype)init {
  34. self = [super init];
  35. if (self) {
  36. _coordinationQueue =
  37. dispatch_queue_create("com.google.GDTCORUploadCoordinator", DISPATCH_QUEUE_SERIAL);
  38. _registrar = [GDTCORRegistrar sharedInstance];
  39. _timerInterval = 30 * NSEC_PER_SEC;
  40. _timerLeeway = 5 * NSEC_PER_SEC;
  41. _targetToInFlightPackages = [[NSMutableDictionary alloc] init];
  42. }
  43. return self;
  44. }
  45. - (void)forceUploadForTarget:(GDTCORTarget)target {
  46. dispatch_async(_coordinationQueue, ^{
  47. GDTCORUploadConditions conditions = [self uploadConditions];
  48. conditions |= GDTCORUploadConditionHighPriority;
  49. [self uploadTargets:@[ @(target) ] conditions:conditions];
  50. });
  51. }
  52. #pragma mark - Property overrides
  53. // GDTCORStorage and GDTCORUploadCoordinator +sharedInstance methods call each other, so this breaks
  54. // the loop.
  55. - (GDTCORStorage *)storage {
  56. if (!_storage) {
  57. _storage = [GDTCORStorage sharedInstance];
  58. }
  59. return _storage;
  60. }
  61. #pragma mark - Private helper methods
  62. /** Starts a timer that checks whether or not events can be uploaded at regular intervals. It will
  63. * check the next-upload clocks of all targets to determine if an upload attempt can be made.
  64. */
  65. - (void)startTimer {
  66. dispatch_sync(_coordinationQueue, ^{
  67. self->_timer =
  68. dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, 0, 0, self->_coordinationQueue);
  69. dispatch_source_set_timer(self->_timer, DISPATCH_TIME_NOW, self->_timerInterval,
  70. self->_timerLeeway);
  71. dispatch_source_set_event_handler(self->_timer, ^{
  72. if (![[GDTCORApplication sharedApplication] isRunningInBackground]) {
  73. GDTCORUploadConditions conditions = [self uploadConditions];
  74. [self uploadTargets:[self.registrar.targetToUploader allKeys] conditions:conditions];
  75. }
  76. });
  77. dispatch_resume(self->_timer);
  78. });
  79. }
  80. /** Stops the currently running timer. */
  81. - (void)stopTimer {
  82. if (_timer) {
  83. dispatch_source_cancel(_timer);
  84. }
  85. }
  86. /** Triggers the uploader implementations for the given targets to upload.
  87. *
  88. * @param targets An array of targets to trigger.
  89. * @param conditions The set of upload conditions.
  90. */
  91. - (void)uploadTargets:(NSArray<NSNumber *> *)targets conditions:(GDTCORUploadConditions)conditions {
  92. dispatch_async(_coordinationQueue, ^{
  93. if ((conditions & GDTCORUploadConditionNoNetwork) == GDTCORUploadConditionNoNetwork) {
  94. return;
  95. }
  96. for (NSNumber *target in targets) {
  97. // Don't trigger uploads for targets that have an in-flight package already.
  98. if (self->_targetToInFlightPackages[target]) {
  99. continue;
  100. }
  101. // Ask the uploader if they can upload and do so, if it can.
  102. id<GDTCORUploader> uploader = self.registrar.targetToUploader[target];
  103. if ([uploader readyToUploadWithConditions:conditions]) {
  104. id<GDTCORPrioritizer> prioritizer = self.registrar.targetToPrioritizer[target];
  105. GDTCORUploadPackage *package = [prioritizer uploadPackageWithConditions:conditions];
  106. if (package.events.count) {
  107. self->_targetToInFlightPackages[target] = package;
  108. [uploader uploadPackage:package];
  109. } else {
  110. [package completeDelivery];
  111. }
  112. }
  113. }
  114. });
  115. }
  116. /** Returns the current upload conditions after making determinations about the network connection.
  117. *
  118. * @return The current upload conditions.
  119. */
  120. - (GDTCORUploadConditions)uploadConditions {
  121. #if TARGET_OS_WATCH
  122. return GDTCORUploadConditionNoNetwork;
  123. #else
  124. SCNetworkReachabilityFlags currentFlags = [GDTCORReachability currentFlags];
  125. BOOL reachable =
  126. (currentFlags & kSCNetworkReachabilityFlagsReachable) == kSCNetworkReachabilityFlagsReachable;
  127. BOOL connectionRequired = (currentFlags & kSCNetworkReachabilityFlagsConnectionRequired) ==
  128. kSCNetworkReachabilityFlagsConnectionRequired;
  129. BOOL networkConnected = reachable && !connectionRequired;
  130. if (!networkConnected) {
  131. return GDTCORUploadConditionNoNetwork;
  132. }
  133. BOOL isWWAN = GDTCORReachabilityFlagsContainWWAN(currentFlags);
  134. if (isWWAN) {
  135. return GDTCORUploadConditionMobileData;
  136. } else {
  137. return GDTCORUploadConditionWifiData;
  138. }
  139. #endif
  140. }
  141. #pragma mark - NSSecureCoding support
  142. /** The NSKeyedCoder key for the targetToInFlightPackages property. */
  143. static NSString *const ktargetToInFlightPackagesKey =
  144. @"GDTCORUploadCoordinatortargetToInFlightPackages";
  145. + (BOOL)supportsSecureCoding {
  146. return YES;
  147. }
  148. - (instancetype)initWithCoder:(NSCoder *)aDecoder {
  149. GDTCORUploadCoordinator *sharedCoordinator = [GDTCORUploadCoordinator sharedInstance];
  150. dispatch_sync(sharedCoordinator->_coordinationQueue, ^{
  151. @try {
  152. sharedCoordinator->_targetToInFlightPackages =
  153. [aDecoder decodeObjectOfClass:[NSMutableDictionary class]
  154. forKey:ktargetToInFlightPackagesKey];
  155. } @catch (NSException *exception) {
  156. sharedCoordinator->_targetToInFlightPackages = [NSMutableDictionary dictionary];
  157. }
  158. });
  159. return sharedCoordinator;
  160. }
  161. - (void)encodeWithCoder:(NSCoder *)aCoder {
  162. dispatch_sync(_coordinationQueue, ^{
  163. // All packages that have been given to uploaders need to be tracked so that their expiration
  164. // timers can be called.
  165. if (self->_targetToInFlightPackages.count > 0) {
  166. [aCoder encodeObject:self->_targetToInFlightPackages forKey:ktargetToInFlightPackagesKey];
  167. }
  168. });
  169. }
  170. #pragma mark - GDTCORLifecycleProtocol
  171. - (void)appWillForeground:(GDTCORApplication *)app {
  172. // Not entirely thread-safe, but it should be fine.
  173. [self startTimer];
  174. }
  175. - (void)appWillBackground:(GDTCORApplication *)app {
  176. // Should be thread-safe. If it ends up not being, put this in a dispatch_sync.
  177. [self stopTimer];
  178. }
  179. - (void)appWillTerminate:(GDTCORApplication *)application {
  180. dispatch_sync(_coordinationQueue, ^{
  181. [self stopTimer];
  182. });
  183. }
  184. #pragma mark - GDTCORUploadPackageProtocol
  185. - (void)packageDelivered:(GDTCORUploadPackage *)package successful:(BOOL)successful {
  186. if (!_coordinationQueue) {
  187. return;
  188. }
  189. dispatch_async(_coordinationQueue, ^{
  190. NSNumber *targetNumber = @(package.target);
  191. NSMutableDictionary<NSNumber *, GDTCORUploadPackage *> *targetToInFlightPackages =
  192. self->_targetToInFlightPackages;
  193. GDTCORRegistrar *registrar = self->_registrar;
  194. if (targetToInFlightPackages) {
  195. [targetToInFlightPackages removeObjectForKey:targetNumber];
  196. }
  197. if (registrar) {
  198. id<GDTCORPrioritizer> prioritizer = registrar.targetToPrioritizer[targetNumber];
  199. if (!prioritizer) {
  200. GDTCORLogError(GDTCORMCEPrioritizerError,
  201. @"A prioritizer should be registered for this target: %@", targetNumber);
  202. }
  203. if ([prioritizer respondsToSelector:@selector(packageDelivered:successful:)]) {
  204. [prioritizer packageDelivered:package successful:successful];
  205. }
  206. }
  207. if (package.events != nil) {
  208. [self.storage removeEvents:package.events];
  209. }
  210. });
  211. }
  212. - (void)packageExpired:(GDTCORUploadPackage *)package {
  213. if (!_coordinationQueue) {
  214. return;
  215. }
  216. dispatch_async(_coordinationQueue, ^{
  217. NSNumber *targetNumber = @(package.target);
  218. NSMutableDictionary<NSNumber *, GDTCORUploadPackage *> *targetToInFlightPackages =
  219. self->_targetToInFlightPackages;
  220. GDTCORRegistrar *registrar = self->_registrar;
  221. if (targetToInFlightPackages) {
  222. [targetToInFlightPackages removeObjectForKey:targetNumber];
  223. }
  224. if (registrar) {
  225. id<GDTCORPrioritizer> prioritizer = registrar.targetToPrioritizer[targetNumber];
  226. id<GDTCORUploader> uploader = registrar.targetToUploader[targetNumber];
  227. if ([prioritizer respondsToSelector:@selector(packageExpired:)]) {
  228. [prioritizer packageExpired:package];
  229. }
  230. if ([uploader respondsToSelector:@selector(packageExpired:)]) {
  231. [uploader packageExpired:package];
  232. }
  233. }
  234. });
  235. }
  236. @end