GTMSessionFetcherService.m 47 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383
  1. /* Copyright 2014 Google Inc. All rights reserved.
  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. */
  15. #if !defined(__has_feature) || !__has_feature(objc_arc)
  16. #error "This file requires ARC support."
  17. #endif
  18. #import "GTMSessionFetcherService.h"
  19. NSString *const kGTMSessionFetcherServiceSessionBecameInvalidNotification =
  20. @"kGTMSessionFetcherServiceSessionBecameInvalidNotification";
  21. NSString *const kGTMSessionFetcherServiceSessionKey = @"kGTMSessionFetcherServiceSessionKey";
  22. #if !GTMSESSION_BUILD_COMBINED_SOURCES
  23. @interface GTMSessionFetcher (ServiceMethods)
  24. - (BOOL)beginFetchMayDelay:(BOOL)mayDelay
  25. mayAuthorize:(BOOL)mayAuthorize
  26. mayDecorate:(BOOL)mayDecorate;
  27. @end
  28. #endif // !GTMSESSION_BUILD_COMBINED_SOURCES
  29. @interface GTMSessionFetcherService ()
  30. @property(atomic, strong, readwrite) NSDictionary *delayedFetchersByHost;
  31. @property(atomic, strong, readwrite) NSDictionary *runningFetchersByHost;
  32. // Ordered collection of id<GTMFetcherDecoratorProtocol>, held weakly.
  33. @property(atomic, strong, readonly) NSPointerArray *decoratorsPointerArray;
  34. @end
  35. // Since NSURLSession doesn't support a separate delegate per task (!), instances of this
  36. // class serve as a session delegate trampoline.
  37. //
  38. // This class maps a session's tasks to fetchers, and resends delegate messages to the task's
  39. // fetcher.
  40. @interface GTMSessionFetcherSessionDelegateDispatcher : NSObject <NSURLSessionDelegate>
  41. // The session for the tasks in this dispatcher's task-to-fetcher map.
  42. @property(atomic) NSURLSession *session;
  43. // The timer interval for invalidating a session that has no active tasks.
  44. @property(atomic) NSTimeInterval discardInterval;
  45. // The current discard timer.
  46. @property(atomic, readonly) NSTimer *discardTimer;
  47. - (instancetype)initWithParentService:(GTMSessionFetcherService *)parentService
  48. sessionDiscardInterval:(NSTimeInterval)discardInterval;
  49. - (void)setFetcher:(GTMSessionFetcher *)fetcher forTask:(NSURLSessionTask *)task;
  50. - (void)removeFetcher:(GTMSessionFetcher *)fetcher;
  51. // Before using a session, tells the delegate dispatcher to stop the discard timer.
  52. - (void)startSessionUsage;
  53. // When abandoning a delegate dispatcher, we want to avoid the session retaining
  54. // the delegate after tasks complete.
  55. - (void)abandon;
  56. @end
  57. @implementation GTMSessionFetcherService {
  58. NSMutableDictionary *_delayedFetchersByHost;
  59. NSMutableDictionary *_runningFetchersByHost;
  60. NSUInteger _maxRunningFetchersPerHost;
  61. // When this ivar is nil, the service will not reuse sessions.
  62. GTMSessionFetcherSessionDelegateDispatcher *_delegateDispatcher;
  63. // Fetchers will wait on this if another fetcher is creating the shared NSURLSession.
  64. dispatch_semaphore_t _sessionCreationSemaphore;
  65. dispatch_queue_t _callbackQueue;
  66. NSOperationQueue *_delegateQueue;
  67. NSHTTPCookieStorage *_cookieStorage;
  68. NSString *_userAgent;
  69. NSTimeInterval _timeout;
  70. NSURLCredential *_credential; // Username & password.
  71. NSURLCredential *_proxyCredential; // Credential supplied to proxy servers.
  72. NSInteger _cookieStorageMethod;
  73. id<GTMFetcherAuthorizationProtocol> _authorizer;
  74. // For waitForCompletionOfAllFetchersWithTimeout: we need to wait on stopped fetchers since
  75. // they've not yet finished invoking their queued callbacks. This array is nil except when
  76. // waiting on fetchers.
  77. NSMutableArray *_stoppedFetchersToWaitFor;
  78. // For fetchers that enqueued their callbacks before stopAllFetchers was called on the service,
  79. // set a barrier so the callbacks know to bail out.
  80. NSDate *_stoppedAllFetchersDate;
  81. }
  82. // Clang-format likes to cram all @synthesize items onto the fewest lines, rather than one-per.
  83. // clang-format off
  84. @synthesize maxRunningFetchersPerHost = _maxRunningFetchersPerHost,
  85. configuration = _configuration,
  86. configurationBlock = _configurationBlock,
  87. cookieStorage = _cookieStorage,
  88. userAgent = _userAgent,
  89. challengeBlock = _challengeBlock,
  90. credential = _credential,
  91. proxyCredential = _proxyCredential,
  92. allowedInsecureSchemes = _allowedInsecureSchemes,
  93. allowLocalhostRequest = _allowLocalhostRequest,
  94. allowInvalidServerCertificates = _allowInvalidServerCertificates,
  95. retryEnabled = _retryEnabled,
  96. retryBlock = _retryBlock,
  97. maxRetryInterval = _maxRetryInterval,
  98. minRetryInterval = _minRetryInterval,
  99. metricsCollectionBlock = _metricsCollectionBlock,
  100. properties = _properties,
  101. unusedSessionTimeout = _unusedSessionTimeout,
  102. decoratorsPointerArray = _decoratorsPointerArray,
  103. testBlock = _testBlock;
  104. // clang-format on
  105. #if GTM_BACKGROUND_TASK_FETCHING
  106. @synthesize skipBackgroundTask = _skipBackgroundTask;
  107. #endif
  108. - (instancetype)init {
  109. self = [super init];
  110. if (self) {
  111. _delayedFetchersByHost = [[NSMutableDictionary alloc] init];
  112. _runningFetchersByHost = [[NSMutableDictionary alloc] init];
  113. _maxRunningFetchersPerHost = 10;
  114. _cookieStorageMethod = -1;
  115. _unusedSessionTimeout = 60.0;
  116. _delegateDispatcher = [[GTMSessionFetcherSessionDelegateDispatcher alloc]
  117. initWithParentService:self
  118. sessionDiscardInterval:_unusedSessionTimeout];
  119. _callbackQueue = dispatch_get_main_queue();
  120. _delegateQueue = [[NSOperationQueue alloc] init];
  121. _delegateQueue.maxConcurrentOperationCount = 1;
  122. _delegateQueue.name = @"com.google.GTMSessionFetcher.NSURLSessionDelegateQueue";
  123. _sessionCreationSemaphore = dispatch_semaphore_create(1);
  124. // Starting with the SDKs for OS X 10.11/iOS 9, the service has a default useragent.
  125. // Apps can remove this and get the default system "CFNetwork" useragent by setting the
  126. // fetcher service's userAgent property to nil.
  127. _userAgent = GTMFetcherStandardUserAgentString(nil);
  128. }
  129. return self;
  130. }
  131. - (void)dealloc {
  132. [self detachAuthorizer];
  133. [_delegateDispatcher abandon];
  134. }
  135. #pragma mark Generate a new fetcher
  136. // Clients may override this method. Clients should not override any other library methods.
  137. - (id)fetcherWithRequest:(NSURLRequest *)request fetcherClass:(Class)fetcherClass {
  138. GTMSessionFetcher *fetcher = [[fetcherClass alloc] initWithRequest:request
  139. configuration:self.configuration];
  140. fetcher.callbackQueue = self.callbackQueue;
  141. fetcher.sessionDelegateQueue = self.sessionDelegateQueue;
  142. fetcher.challengeBlock = self.challengeBlock;
  143. fetcher.credential = self.credential;
  144. fetcher.proxyCredential = self.proxyCredential;
  145. fetcher.authorizer = self.authorizer;
  146. fetcher.cookieStorage = self.cookieStorage;
  147. fetcher.allowedInsecureSchemes = self.allowedInsecureSchemes;
  148. fetcher.allowLocalhostRequest = self.allowLocalhostRequest;
  149. fetcher.allowInvalidServerCertificates = self.allowInvalidServerCertificates;
  150. fetcher.configurationBlock = self.configurationBlock;
  151. fetcher.retryEnabled = self.retryEnabled;
  152. fetcher.retryBlock = self.retryBlock;
  153. fetcher.maxRetryInterval = self.maxRetryInterval;
  154. fetcher.minRetryInterval = self.minRetryInterval;
  155. if (@available(iOS 10.0, *)) {
  156. fetcher.metricsCollectionBlock = self.metricsCollectionBlock;
  157. }
  158. fetcher.properties = self.properties;
  159. fetcher.service = self;
  160. #pragma clang diagnostic push
  161. #pragma clang diagnostic ignored "-Wdeprecated-declarations"
  162. if (self.cookieStorageMethod >= 0) {
  163. [fetcher setCookieStorageMethod:self.cookieStorageMethod];
  164. }
  165. #pragma clang diagnostic pop
  166. #if GTM_BACKGROUND_TASK_FETCHING
  167. fetcher.skipBackgroundTask = self.skipBackgroundTask;
  168. #endif
  169. NSString *userAgent = self.userAgent;
  170. if (userAgent.length > 0 && [request valueForHTTPHeaderField:@"User-Agent"] == nil) {
  171. [fetcher setRequestValue:userAgent forHTTPHeaderField:@"User-Agent"];
  172. }
  173. fetcher.testBlock = self.testBlock;
  174. return fetcher;
  175. }
  176. - (GTMSessionFetcher *)fetcherWithRequest:(NSURLRequest *)request {
  177. return [self fetcherWithRequest:request fetcherClass:[GTMSessionFetcher class]];
  178. }
  179. - (GTMSessionFetcher *)fetcherWithURL:(NSURL *)requestURL {
  180. return [self fetcherWithRequest:[NSURLRequest requestWithURL:requestURL]];
  181. }
  182. - (GTMSessionFetcher *)fetcherWithURLString:(NSString *)requestURLString {
  183. NSURL *url = [NSURL URLWithString:requestURLString];
  184. return [self fetcherWithURL:url];
  185. }
  186. - (void)addDecorator:(id<GTMFetcherDecoratorProtocol>)decorator {
  187. @synchronized(self) {
  188. if (!_decoratorsPointerArray) {
  189. _decoratorsPointerArray = [NSPointerArray weakObjectsPointerArray];
  190. }
  191. [_decoratorsPointerArray addPointer:(__bridge void *)decorator];
  192. }
  193. }
  194. - (nullable NSArray<id<GTMFetcherDecoratorProtocol>> *)decorators {
  195. @synchronized(self) {
  196. return _decoratorsPointerArray.allObjects;
  197. }
  198. }
  199. - (void)removeDecorator:(id<GTMFetcherDecoratorProtocol>)decorator {
  200. @synchronized(self) {
  201. NSUInteger i = 0;
  202. for (id<GTMFetcherDecoratorProtocol> decoratorCandidate in _decoratorsPointerArray) {
  203. if (decoratorCandidate == decorator) {
  204. break;
  205. }
  206. ++i;
  207. }
  208. GTMSESSION_ASSERT_DEBUG(i < _decoratorsPointerArray.count,
  209. @"decorator %@ must be passed to -addDecorator: before removing",
  210. decorator);
  211. if (i < _decoratorsPointerArray.count) {
  212. [_decoratorsPointerArray removePointerAtIndex:i];
  213. }
  214. }
  215. }
  216. // Returns a session for the fetcher's host, or nil.
  217. - (NSURLSession *)session {
  218. @synchronized(self) {
  219. GTMSessionMonitorSynchronized(self);
  220. NSURLSession *session = _delegateDispatcher.session;
  221. return session;
  222. }
  223. }
  224. // Returns a session for the fetcher's host, or nil. For shared sessions, this
  225. // waits on a semaphore, blocking other fetchers while the caller creates the
  226. // session if needed.
  227. - (NSURLSession *)sessionForFetcherCreation {
  228. @synchronized(self) {
  229. GTMSessionMonitorSynchronized(self);
  230. if (!_delegateDispatcher) {
  231. // This fetcher is creating a non-shared session, so skip the semaphore usage.
  232. return nil;
  233. }
  234. }
  235. // Wait if another fetcher is currently creating a session; avoid waiting
  236. // inside the @synchronized block, as that can deadlock.
  237. dispatch_semaphore_wait(_sessionCreationSemaphore, DISPATCH_TIME_FOREVER);
  238. @synchronized(self) {
  239. GTMSessionMonitorSynchronized(self);
  240. // Before getting the NSURLSession for task creation, it is
  241. // important to invalidate and nil out the session discard timer; otherwise
  242. // the session can be invalidated between when it is returned to the
  243. // fetcher, and when the fetcher attempts to create its NSURLSessionTask.
  244. [_delegateDispatcher startSessionUsage];
  245. NSURLSession *session = _delegateDispatcher.session;
  246. if (session) {
  247. // The calling fetcher will receive a preexisting session, so
  248. // we can allow other fetchers to create a session.
  249. dispatch_semaphore_signal(_sessionCreationSemaphore);
  250. } else {
  251. // No existing session was obtained, so the calling fetcher will create the session;
  252. // it *must* invoke fetcherDidCreateSession: to signal the dispatcher's semaphore after
  253. // the session has been created (or fails to be created) to avoid a hang.
  254. }
  255. return session;
  256. }
  257. }
  258. - (id<NSURLSessionDelegate>)sessionDelegate {
  259. @synchronized(self) {
  260. GTMSessionMonitorSynchronized(self);
  261. return _delegateDispatcher;
  262. }
  263. }
  264. #pragma mark Queue Management
  265. - (void)addRunningFetcher:(GTMSessionFetcher *)fetcher forHost:(NSString *)host {
  266. // Add to the array of running fetchers for this host, creating the array if needed.
  267. NSMutableArray *runningForHost = [_runningFetchersByHost objectForKey:host];
  268. if (runningForHost == nil) {
  269. runningForHost = [NSMutableArray arrayWithObject:fetcher];
  270. [_runningFetchersByHost setObject:runningForHost forKey:host];
  271. } else {
  272. [runningForHost addObject:fetcher];
  273. }
  274. }
  275. - (void)addDelayedFetcher:(GTMSessionFetcher *)fetcher forHost:(NSString *)host {
  276. // Add to the array of delayed fetchers for this host, creating the array if needed.
  277. NSMutableArray *delayedForHost = [_delayedFetchersByHost objectForKey:host];
  278. if (delayedForHost == nil) {
  279. delayedForHost = [NSMutableArray arrayWithObject:fetcher];
  280. [_delayedFetchersByHost setObject:delayedForHost forKey:host];
  281. } else {
  282. [delayedForHost addObject:fetcher];
  283. }
  284. }
  285. - (BOOL)isDelayingFetcher:(GTMSessionFetcher *)fetcher {
  286. @synchronized(self) {
  287. GTMSessionMonitorSynchronized(self);
  288. NSString *host = fetcher.request.URL.host;
  289. if (host == nil) {
  290. return NO;
  291. }
  292. NSArray *delayedForHost = [_delayedFetchersByHost objectForKey:host];
  293. NSUInteger idx = [delayedForHost indexOfObjectIdenticalTo:fetcher];
  294. BOOL isDelayed = (delayedForHost != nil) && (idx != NSNotFound);
  295. return isDelayed;
  296. }
  297. }
  298. - (BOOL)fetcherShouldBeginFetching:(GTMSessionFetcher *)fetcher {
  299. // Entry point from the fetcher
  300. NSURL *requestURL = fetcher.request.URL;
  301. NSString *host = requestURL.host;
  302. // Addresses "file:///path" case where localhost is the implicit host.
  303. if (host.length == 0 && [requestURL isFileURL]) {
  304. host = @"localhost";
  305. }
  306. if (host.length == 0) {
  307. // Data URIs legitimately have no host, reject other hostless URLs.
  308. GTMSESSION_ASSERT_DEBUG([[requestURL scheme] isEqual:@"data"], @"%@ lacks host", fetcher);
  309. return YES;
  310. }
  311. BOOL shouldBeginResult;
  312. @synchronized(self) {
  313. GTMSessionMonitorSynchronized(self);
  314. NSMutableArray *runningForHost = [_runningFetchersByHost objectForKey:host];
  315. if (runningForHost != nil && [runningForHost indexOfObjectIdenticalTo:fetcher] != NSNotFound) {
  316. GTMSESSION_ASSERT_DEBUG(NO, @"%@ was already running", fetcher);
  317. return YES;
  318. }
  319. BOOL shouldRunNow = (fetcher.usingBackgroundSession || _maxRunningFetchersPerHost == 0 ||
  320. _maxRunningFetchersPerHost >
  321. [[self class] numberOfNonBackgroundSessionFetchers:runningForHost]);
  322. if (shouldRunNow) {
  323. [self addRunningFetcher:fetcher forHost:host];
  324. shouldBeginResult = YES;
  325. } else {
  326. [self addDelayedFetcher:fetcher forHost:host];
  327. shouldBeginResult = NO;
  328. }
  329. } // @synchronized(self)
  330. // We'll save the host that serves as the key for this fetcher's array
  331. // to avoid any chance of the underlying request changing, stranding
  332. // the fetcher in the wrong array
  333. fetcher.serviceHost = host;
  334. return shouldBeginResult;
  335. }
  336. - (void)startFetcher:(GTMSessionFetcher *)fetcher {
  337. [fetcher beginFetchMayDelay:NO mayAuthorize:YES mayDecorate:YES];
  338. }
  339. // Internal utility. Returns a fetcher's delegate if it's a dispatcher, or nil if the fetcher
  340. // is its own delegate (possibly via proxy) and has no dispatcher.
  341. - (GTMSessionFetcherSessionDelegateDispatcher *)delegateDispatcherForFetcher:
  342. (GTMSessionFetcher *)fetcher {
  343. GTMSessionCheckNotSynchronized(self);
  344. NSURLSession *fetcherSession = fetcher.session;
  345. if (fetcherSession) {
  346. id<NSURLSessionDelegate> fetcherDelegate = fetcherSession.delegate;
  347. // If the delegate is non-nil and claims to be a GTMSessionFetcher, there is no dispatcher;
  348. // assume the fetcher is the delegate or has been proxied (some third-party frameworks
  349. // are known to swizzle NSURLSession to proxy its delegate).
  350. BOOL hasDispatcher =
  351. (fetcherDelegate != nil && ![fetcherDelegate isKindOfClass:[GTMSessionFetcher class]]);
  352. if (hasDispatcher) {
  353. GTMSESSION_ASSERT_DEBUG(
  354. [fetcherDelegate isKindOfClass:[GTMSessionFetcherSessionDelegateDispatcher class]],
  355. @"Fetcher delegate class: %@", [fetcherDelegate class]);
  356. return (GTMSessionFetcherSessionDelegateDispatcher *)fetcherDelegate;
  357. }
  358. }
  359. return nil;
  360. }
  361. - (void)fetcherDidCreateSession:(GTMSessionFetcher *)fetcher {
  362. if (fetcher.canShareSession) {
  363. NSURLSession *fetcherSession = fetcher.session;
  364. GTMSESSION_ASSERT_DEBUG(fetcherSession != nil, @"Fetcher missing its session: %@", fetcher);
  365. GTMSessionFetcherSessionDelegateDispatcher *delegateDispatcher =
  366. [self delegateDispatcherForFetcher:fetcher];
  367. if (delegateDispatcher) {
  368. GTMSESSION_ASSERT_DEBUG(delegateDispatcher.session == nil,
  369. @"Fetcher made an extra session: %@", fetcher);
  370. // Save this fetcher's session.
  371. delegateDispatcher.session = fetcherSession;
  372. // Allow other fetchers to request this session now.
  373. dispatch_semaphore_signal(_sessionCreationSemaphore);
  374. }
  375. }
  376. }
  377. - (void)fetcherDidBeginFetching:(GTMSessionFetcher *)fetcher {
  378. // If this fetcher has a separate delegate with a shared session, then
  379. // this fetcher should be added to the delegate's map of tasks to fetchers.
  380. GTMSessionFetcherSessionDelegateDispatcher *delegateDispatcher =
  381. [self delegateDispatcherForFetcher:fetcher];
  382. if (delegateDispatcher) {
  383. GTMSESSION_ASSERT_DEBUG(fetcher.canShareSession, @"Inappropriate shared session: %@", fetcher);
  384. // There should already be a session, from this or a previous fetcher.
  385. //
  386. // Sanity check that the fetcher's session is the delegate's shared session.
  387. NSURLSession *sharedSession = delegateDispatcher.session;
  388. NSURLSession *fetcherSession = fetcher.session;
  389. GTMSESSION_ASSERT_DEBUG(sharedSession != nil, @"Missing delegate session: %@", fetcher);
  390. GTMSESSION_ASSERT_DEBUG(fetcherSession == sharedSession,
  391. @"Inconsistent session: %@ %@ (shared: %@)", fetcher, fetcherSession,
  392. sharedSession);
  393. if (sharedSession != nil && fetcherSession == sharedSession) {
  394. NSURLSessionTask *task = fetcher.sessionTask;
  395. GTMSESSION_ASSERT_DEBUG(task != nil, @"Missing session task: %@", fetcher);
  396. if (task) {
  397. [delegateDispatcher setFetcher:fetcher forTask:task];
  398. }
  399. }
  400. }
  401. }
  402. - (void)stopFetcher:(GTMSessionFetcher *)fetcher {
  403. [fetcher stopFetching];
  404. }
  405. - (void)fetcherDidStop:(GTMSessionFetcher *)fetcher {
  406. // Entry point from the fetcher
  407. NSString *host = fetcher.serviceHost;
  408. if (!host) {
  409. // fetcher has been stopped previously
  410. return;
  411. }
  412. // This removeFetcher: invocation is a fallback; typically, fetchers are removed from the task
  413. // map when the task completes.
  414. GTMSessionFetcherSessionDelegateDispatcher *delegateDispatcher =
  415. [self delegateDispatcherForFetcher:fetcher];
  416. [delegateDispatcher removeFetcher:fetcher];
  417. NSMutableArray *fetchersToStart;
  418. @synchronized(self) {
  419. GTMSessionMonitorSynchronized(self);
  420. // If a test is waiting for all fetchers to stop, it needs to wait for this one
  421. // to invoke its callbacks on the callback queue.
  422. [_stoppedFetchersToWaitFor addObject:fetcher];
  423. NSMutableArray *runningForHost = [_runningFetchersByHost objectForKey:host];
  424. [runningForHost removeObject:fetcher];
  425. NSMutableArray *delayedForHost = [_delayedFetchersByHost objectForKey:host];
  426. [delayedForHost removeObject:fetcher];
  427. while (delayedForHost.count > 0 &&
  428. [[self class] numberOfNonBackgroundSessionFetchers:runningForHost] <
  429. _maxRunningFetchersPerHost) {
  430. // Start another delayed fetcher running, scanning for the minimum
  431. // priority value, defaulting to FIFO for equal priorities
  432. GTMSessionFetcher *nextFetcher = nil;
  433. for (GTMSessionFetcher *delayedFetcher in delayedForHost) {
  434. if (nextFetcher == nil || delayedFetcher.servicePriority < nextFetcher.servicePriority) {
  435. nextFetcher = delayedFetcher;
  436. }
  437. }
  438. if (nextFetcher) {
  439. [self addRunningFetcher:nextFetcher forHost:host];
  440. runningForHost = [_runningFetchersByHost objectForKey:host];
  441. [delayedForHost removeObjectIdenticalTo:nextFetcher];
  442. if (!fetchersToStart) {
  443. fetchersToStart = [NSMutableArray array];
  444. }
  445. [fetchersToStart addObject:nextFetcher];
  446. }
  447. }
  448. if (runningForHost.count == 0) {
  449. // None left; remove the empty array
  450. [_runningFetchersByHost removeObjectForKey:host];
  451. }
  452. if (delayedForHost.count == 0) {
  453. [_delayedFetchersByHost removeObjectForKey:host];
  454. }
  455. } // @synchronized(self)
  456. // Start fetchers outside of the synchronized block to avoid a deadlock.
  457. for (GTMSessionFetcher *nextFetcher in fetchersToStart) {
  458. [self startFetcher:nextFetcher];
  459. }
  460. // The fetcher is no longer in the running or the delayed array,
  461. // so remove its host and thread properties
  462. fetcher.serviceHost = nil;
  463. }
  464. - (NSUInteger)numberOfFetchers {
  465. NSUInteger running = [self numberOfRunningFetchers];
  466. NSUInteger delayed = [self numberOfDelayedFetchers];
  467. return running + delayed;
  468. }
  469. - (NSUInteger)numberOfRunningFetchers {
  470. @synchronized(self) {
  471. GTMSessionMonitorSynchronized(self);
  472. NSUInteger sum = 0;
  473. for (NSString *host in _runningFetchersByHost) {
  474. NSArray *fetchers = [_runningFetchersByHost objectForKey:host];
  475. sum += fetchers.count;
  476. }
  477. return sum;
  478. }
  479. }
  480. - (NSUInteger)numberOfDelayedFetchers {
  481. @synchronized(self) {
  482. GTMSessionMonitorSynchronized(self);
  483. NSUInteger sum = 0;
  484. for (NSString *host in _delayedFetchersByHost) {
  485. NSArray *fetchers = [_delayedFetchersByHost objectForKey:host];
  486. sum += fetchers.count;
  487. }
  488. return sum;
  489. }
  490. }
  491. - (NSArray *)issuedFetchers {
  492. @synchronized(self) {
  493. GTMSessionMonitorSynchronized(self);
  494. NSMutableArray *allFetchers = [NSMutableArray array];
  495. void (^accumulateFetchers)(id, id, BOOL *) =
  496. ^(NSString *host, NSArray *fetchersForHost, BOOL *stop) {
  497. [allFetchers addObjectsFromArray:fetchersForHost];
  498. };
  499. [_runningFetchersByHost enumerateKeysAndObjectsUsingBlock:accumulateFetchers];
  500. [_delayedFetchersByHost enumerateKeysAndObjectsUsingBlock:accumulateFetchers];
  501. GTMSESSION_ASSERT_DEBUG(allFetchers.count == [NSSet setWithArray:allFetchers].count,
  502. @"Fetcher appears multiple times\n running: %@\n delayed: %@",
  503. _runningFetchersByHost, _delayedFetchersByHost);
  504. return allFetchers.count > 0 ? allFetchers : nil;
  505. }
  506. }
  507. - (NSArray *)issuedFetchersWithRequestURL:(NSURL *)requestURL {
  508. NSString *host = requestURL.host;
  509. if (host.length == 0) return nil;
  510. NSURL *targetURL = [requestURL absoluteURL];
  511. NSArray *allFetchers = [self issuedFetchers];
  512. NSIndexSet *indexes = [allFetchers
  513. indexesOfObjectsPassingTest:^BOOL(GTMSessionFetcher *fetcher, NSUInteger idx, BOOL *stop) {
  514. NSURL *fetcherURL = [fetcher.request.URL absoluteURL];
  515. return [fetcherURL isEqual:targetURL];
  516. }];
  517. NSArray *result = nil;
  518. if (indexes.count > 0) {
  519. result = [allFetchers objectsAtIndexes:indexes];
  520. }
  521. return result;
  522. }
  523. - (void)stopAllFetchers {
  524. NSArray *delayedFetchersByHost;
  525. NSArray *runningFetchersByHost;
  526. @synchronized(self) {
  527. GTMSessionMonitorSynchronized(self);
  528. // Set the time barrier so fetchers know not to call back even if
  529. // the stop calls below occur after the fetchers naturally
  530. // stopped and so were removed from _runningFetchersByHost,
  531. // but while the callbacks were already enqueued before stopAllFetchers
  532. // was invoked.
  533. _stoppedAllFetchersDate = [[NSDate alloc] init];
  534. // Remove fetchers from the delayed list to avoid fetcherDidStop: from
  535. // starting more fetchers running as a side effect of stopping one
  536. delayedFetchersByHost = _delayedFetchersByHost.allValues;
  537. [_delayedFetchersByHost removeAllObjects];
  538. runningFetchersByHost = _runningFetchersByHost.allValues;
  539. [_runningFetchersByHost removeAllObjects];
  540. }
  541. for (NSArray *delayedForHost in delayedFetchersByHost) {
  542. for (GTMSessionFetcher *fetcher in delayedForHost) {
  543. [self stopFetcher:fetcher];
  544. }
  545. }
  546. for (NSArray *runningForHost in runningFetchersByHost) {
  547. for (GTMSessionFetcher *fetcher in runningForHost) {
  548. [self stopFetcher:fetcher];
  549. }
  550. }
  551. }
  552. - (NSDate *)stoppedAllFetchersDate {
  553. @synchronized(self) {
  554. GTMSessionMonitorSynchronized(self);
  555. return _stoppedAllFetchersDate;
  556. }
  557. }
  558. #pragma mark Accessors
  559. - (BOOL)reuseSession {
  560. @synchronized(self) {
  561. GTMSessionMonitorSynchronized(self);
  562. return _delegateDispatcher != nil;
  563. }
  564. }
  565. - (void)setReuseSession:(BOOL)shouldReuse {
  566. @synchronized(self) {
  567. GTMSessionMonitorSynchronized(self);
  568. BOOL wasReusing = (_delegateDispatcher != nil);
  569. if (shouldReuse != wasReusing) {
  570. [self abandonDispatcher];
  571. if (shouldReuse) {
  572. _delegateDispatcher = [[GTMSessionFetcherSessionDelegateDispatcher alloc]
  573. initWithParentService:self
  574. sessionDiscardInterval:_unusedSessionTimeout];
  575. } else {
  576. _delegateDispatcher = nil;
  577. }
  578. }
  579. }
  580. }
  581. - (void)resetSession {
  582. GTMSessionCheckNotSynchronized(self);
  583. dispatch_semaphore_wait(_sessionCreationSemaphore, DISPATCH_TIME_FOREVER);
  584. @synchronized(self) {
  585. GTMSessionMonitorSynchronized(self);
  586. [self resetSessionInternal];
  587. }
  588. dispatch_semaphore_signal(_sessionCreationSemaphore);
  589. }
  590. - (void)resetSessionInternal {
  591. GTMSessionCheckSynchronized(self);
  592. // The old dispatchers may be retained as delegates of any ongoing sessions by those sessions.
  593. if (_delegateDispatcher) {
  594. [self abandonDispatcher];
  595. _delegateDispatcher = [[GTMSessionFetcherSessionDelegateDispatcher alloc]
  596. initWithParentService:self
  597. sessionDiscardInterval:_unusedSessionTimeout];
  598. }
  599. }
  600. - (void)resetSessionForDispatcherDiscardTimer:(NSTimer *)timer {
  601. GTMSessionCheckNotSynchronized(self);
  602. dispatch_semaphore_wait(_sessionCreationSemaphore, DISPATCH_TIME_FOREVER);
  603. @synchronized(self) {
  604. GTMSessionMonitorSynchronized(self);
  605. if (_delegateDispatcher.discardTimer == timer) {
  606. // If the delegate dispatcher's current discardTimer is the same object as the timer
  607. // that fired, no fetcher has recently attempted to start using the session by calling
  608. // startSessionUsage, which invalidates and nils out the timer.
  609. [self resetSessionInternal];
  610. } else {
  611. // A fetcher has invalidated the timer between its triggering and now, potentially
  612. // meaning a fetcher has requested access to the NSURLSession, and may be in the process
  613. // of starting a new task. The dispatcher should not be abandoned, as this can lead
  614. // to a race condition between calling -finishTasksAndInvalidate on the NSURLSession
  615. // and the fetcher attempting to create a new task.
  616. }
  617. }
  618. dispatch_semaphore_signal(_sessionCreationSemaphore);
  619. }
  620. - (NSTimeInterval)unusedSessionTimeout {
  621. @synchronized(self) {
  622. GTMSessionMonitorSynchronized(self);
  623. return _unusedSessionTimeout;
  624. }
  625. }
  626. - (void)setUnusedSessionTimeout:(NSTimeInterval)timeout {
  627. @synchronized(self) {
  628. GTMSessionMonitorSynchronized(self);
  629. _unusedSessionTimeout = timeout;
  630. _delegateDispatcher.discardInterval = timeout;
  631. }
  632. }
  633. // This method should be called inside of @synchronized(self)
  634. - (void)abandonDispatcher {
  635. GTMSessionCheckSynchronized(self);
  636. [_delegateDispatcher abandon];
  637. }
  638. - (NSDictionary *)runningFetchersByHost {
  639. @synchronized(self) {
  640. GTMSessionMonitorSynchronized(self);
  641. return [_runningFetchersByHost copy];
  642. }
  643. }
  644. - (void)setRunningFetchersByHost:(NSDictionary *)dict {
  645. @synchronized(self) {
  646. GTMSessionMonitorSynchronized(self);
  647. _runningFetchersByHost = [dict mutableCopy];
  648. }
  649. }
  650. - (NSDictionary *)delayedFetchersByHost {
  651. @synchronized(self) {
  652. GTMSessionMonitorSynchronized(self);
  653. return [_delayedFetchersByHost copy];
  654. }
  655. }
  656. - (void)setDelayedFetchersByHost:(NSDictionary *)dict {
  657. @synchronized(self) {
  658. GTMSessionMonitorSynchronized(self);
  659. _delayedFetchersByHost = [dict mutableCopy];
  660. }
  661. }
  662. - (id<GTMFetcherAuthorizationProtocol>)authorizer {
  663. @synchronized(self) {
  664. GTMSessionMonitorSynchronized(self);
  665. return _authorizer;
  666. }
  667. }
  668. - (void)setAuthorizer:(id<GTMFetcherAuthorizationProtocol>)obj {
  669. @synchronized(self) {
  670. GTMSessionMonitorSynchronized(self);
  671. if (obj != _authorizer) {
  672. [self detachAuthorizer];
  673. }
  674. _authorizer = obj;
  675. }
  676. // Use the fetcher service for the authorization fetches if the auth
  677. // object supports fetcher services
  678. if ([obj respondsToSelector:@selector(setFetcherService:)]) {
  679. [obj setFetcherService:self];
  680. }
  681. }
  682. // This should be called inside a @synchronized(self) block except during dealloc.
  683. - (void)detachAuthorizer {
  684. // This method is called by the fetcher service's dealloc and setAuthorizer:
  685. // methods; do not override.
  686. //
  687. // The fetcher service retains the authorizer, and the authorizer has a
  688. // weak pointer to the fetcher service (a non-zeroing pointer for
  689. // compatibility with iOS 4 and Mac OS X 10.5/10.6.)
  690. //
  691. // When this fetcher service no longer uses the authorizer, we want to remove
  692. // the authorizer's dependence on the fetcher service. Authorizers can still
  693. // function without a fetcher service.
  694. if ([_authorizer respondsToSelector:@selector(fetcherService)]) {
  695. id authFetcherService = [_authorizer fetcherService];
  696. if (authFetcherService == self) {
  697. [_authorizer setFetcherService:nil];
  698. }
  699. }
  700. }
  701. - (nonnull dispatch_queue_t)callbackQueue {
  702. @synchronized(self) {
  703. GTMSessionMonitorSynchronized(self);
  704. return _callbackQueue;
  705. } // @synchronized(self)
  706. }
  707. - (void)setCallbackQueue:(dispatch_queue_t)queue {
  708. @synchronized(self) {
  709. GTMSessionMonitorSynchronized(self);
  710. _callbackQueue = queue ?: dispatch_get_main_queue();
  711. } // @synchronized(self)
  712. }
  713. - (NSOperationQueue *)sessionDelegateQueue {
  714. @synchronized(self) {
  715. GTMSessionMonitorSynchronized(self);
  716. return _delegateQueue;
  717. } // @synchronized(self)
  718. }
  719. - (void)setSessionDelegateQueue:(NSOperationQueue *)queue {
  720. @synchronized(self) {
  721. GTMSessionMonitorSynchronized(self);
  722. _delegateQueue = queue ?: [NSOperationQueue mainQueue];
  723. } // @synchronized(self)
  724. }
  725. - (NSOperationQueue *)delegateQueue {
  726. // Provided for compatibility with the old fetcher service. The gtm-oauth2 code respects
  727. // any custom delegate queue for calling the app.
  728. return nil;
  729. }
  730. + (NSUInteger)numberOfNonBackgroundSessionFetchers:(NSArray *)fetchers {
  731. NSUInteger sum = 0;
  732. for (GTMSessionFetcher *fetcher in fetchers) {
  733. if (!fetcher.usingBackgroundSession) {
  734. ++sum;
  735. }
  736. }
  737. return sum;
  738. }
  739. @end
  740. @implementation GTMSessionFetcherService (TestingSupport)
  741. + (instancetype)mockFetcherServiceWithFakedData:(NSData *)fakedDataOrNil
  742. fakedError:(NSError *)fakedErrorOrNil {
  743. #if !GTM_DISABLE_FETCHER_TEST_BLOCK
  744. NSURL *url = [NSURL URLWithString:@"http://example.invalid"];
  745. NSHTTPURLResponse *fakedResponse =
  746. [[NSHTTPURLResponse alloc] initWithURL:url
  747. statusCode:(fakedErrorOrNil ? 500 : 200)HTTPVersion:@"HTTP/1.1"
  748. headerFields:nil];
  749. return [self mockFetcherServiceWithFakedData:fakedDataOrNil
  750. fakedResponse:fakedResponse
  751. fakedError:fakedErrorOrNil];
  752. #else
  753. GTMSESSION_ASSERT_DEBUG(0, @"Test blocks disabled");
  754. return nil;
  755. #endif // GTM_DISABLE_FETCHER_TEST_BLOCK
  756. }
  757. + (instancetype)mockFetcherServiceWithFakedData:(NSData *)fakedDataOrNil
  758. fakedResponse:(NSHTTPURLResponse *)fakedResponse
  759. fakedError:(NSError *)fakedErrorOrNil {
  760. #if !GTM_DISABLE_FETCHER_TEST_BLOCK
  761. GTMSessionFetcherService *service = [[self alloc] init];
  762. service.allowedInsecureSchemes = @[ @"http" ];
  763. service.testBlock =
  764. ^(GTMSessionFetcher *fetcherToTest, GTMSessionFetcherTestResponse testResponse) {
  765. testResponse(fakedResponse, fakedDataOrNil, fakedErrorOrNil);
  766. };
  767. return service;
  768. #else
  769. GTMSESSION_ASSERT_DEBUG(0, @"Test blocks disabled");
  770. return nil;
  771. #endif // GTM_DISABLE_FETCHER_TEST_BLOCK
  772. }
  773. #pragma mark Synchronous Wait for Unit Testing
  774. - (BOOL)waitForCompletionOfAllFetchersWithTimeout:(NSTimeInterval)timeoutInSeconds {
  775. NSDate *giveUpDate = [NSDate dateWithTimeIntervalSinceNow:timeoutInSeconds];
  776. _stoppedFetchersToWaitFor = [NSMutableArray array];
  777. BOOL shouldSpinRunLoop = [NSThread isMainThread];
  778. const NSTimeInterval kSpinInterval = 0.001;
  779. BOOL didTimeOut = NO;
  780. while (([self numberOfFetchers] > 0 || _stoppedFetchersToWaitFor.count > 0)) {
  781. didTimeOut = [giveUpDate timeIntervalSinceNow] < 0;
  782. if (didTimeOut) break;
  783. GTMSessionFetcher *stoppedFetcher = _stoppedFetchersToWaitFor.firstObject;
  784. if (stoppedFetcher) {
  785. [_stoppedFetchersToWaitFor removeObject:stoppedFetcher];
  786. [stoppedFetcher waitForCompletionWithTimeout:10.0 * kSpinInterval];
  787. }
  788. if (shouldSpinRunLoop) {
  789. NSDate *stopDate = [NSDate dateWithTimeIntervalSinceNow:kSpinInterval];
  790. [[NSRunLoop currentRunLoop] runUntilDate:stopDate];
  791. } else {
  792. [NSThread sleepForTimeInterval:kSpinInterval];
  793. }
  794. }
  795. _stoppedFetchersToWaitFor = nil;
  796. return !didTimeOut;
  797. }
  798. @end
  799. @implementation GTMSessionFetcherService (BackwardsCompatibilityOnly)
  800. - (NSInteger)cookieStorageMethod {
  801. @synchronized(self) {
  802. GTMSessionMonitorSynchronized(self);
  803. return _cookieStorageMethod;
  804. }
  805. }
  806. - (void)setCookieStorageMethod:(NSInteger)cookieStorageMethod {
  807. @synchronized(self) {
  808. GTMSessionMonitorSynchronized(self);
  809. _cookieStorageMethod = cookieStorageMethod;
  810. }
  811. }
  812. @end
  813. @implementation GTMSessionFetcherSessionDelegateDispatcher {
  814. __weak GTMSessionFetcherService *_parentService;
  815. NSURLSession *_session;
  816. // The task map maps NSURLSessionTasks to GTMSessionFetchers
  817. NSMutableDictionary *_taskToFetcherMap;
  818. // The discard timer will invalidate sessions after the session's last task completes.
  819. NSTimer *_discardTimer;
  820. NSTimeInterval _discardInterval;
  821. }
  822. @synthesize discardInterval = _discardInterval, session = _session;
  823. - (instancetype)init {
  824. [self doesNotRecognizeSelector:_cmd];
  825. return nil;
  826. }
  827. - (instancetype)initWithParentService:(GTMSessionFetcherService *)parentService
  828. sessionDiscardInterval:(NSTimeInterval)discardInterval {
  829. self = [super init];
  830. if (self) {
  831. _discardInterval = discardInterval;
  832. _parentService = parentService;
  833. }
  834. return self;
  835. }
  836. - (NSString *)description {
  837. return
  838. [NSString stringWithFormat:@"%@ %p %@ %@", [self class], self, _session ?: @"<no session>",
  839. _taskToFetcherMap.count > 0 ? _taskToFetcherMap : @"<no tasks>"];
  840. }
  841. - (NSTimer *)discardTimer {
  842. GTMSessionCheckNotSynchronized(self);
  843. @synchronized(self) {
  844. return _discardTimer;
  845. }
  846. }
  847. // This method should be called inside of a @synchronized(self) block.
  848. - (void)startDiscardTimer {
  849. GTMSessionCheckSynchronized(self);
  850. [_discardTimer invalidate];
  851. _discardTimer = nil;
  852. if (_discardInterval > 0) {
  853. _discardTimer = [NSTimer timerWithTimeInterval:_discardInterval
  854. target:self
  855. selector:@selector(discardTimerFired:)
  856. userInfo:nil
  857. repeats:NO];
  858. [_discardTimer setTolerance:(_discardInterval / 10)];
  859. [[NSRunLoop mainRunLoop] addTimer:_discardTimer forMode:NSRunLoopCommonModes];
  860. }
  861. }
  862. // This method should be called inside of a @synchronized(self) block.
  863. - (void)destroyDiscardTimer {
  864. GTMSessionCheckSynchronized(self);
  865. [_discardTimer invalidate];
  866. _discardTimer = nil;
  867. }
  868. - (void)discardTimerFired:(NSTimer *)timer {
  869. GTMSessionFetcherService *service;
  870. @synchronized(self) {
  871. GTMSessionMonitorSynchronized(self);
  872. NSUInteger numberOfTasks = _taskToFetcherMap.count;
  873. if (numberOfTasks == 0) {
  874. service = _parentService;
  875. }
  876. }
  877. // Inform the service that the discard timer has fired, and should check whether the
  878. // service can abandon us. -resetSession cannot be called directly, as there is a
  879. // race condition that must be guarded against with the NSURLSession being returned
  880. // from sessionForFetcherCreation outside other locks. The service can take steps
  881. // to prevent resetting the session if that has occurred.
  882. //
  883. // The service must be called from outside the @synchronized block.
  884. [service resetSessionForDispatcherDiscardTimer:timer];
  885. }
  886. - (void)abandon {
  887. @synchronized(self) {
  888. GTMSessionMonitorSynchronized(self);
  889. [self destroySessionAndTimer];
  890. }
  891. }
  892. - (void)startSessionUsage {
  893. @synchronized(self) {
  894. GTMSessionMonitorSynchronized(self);
  895. [self destroyDiscardTimer];
  896. }
  897. }
  898. // This method should be called inside of a @synchronized(self) block.
  899. - (void)destroySessionAndTimer {
  900. GTMSessionCheckSynchronized(self);
  901. [self destroyDiscardTimer];
  902. // Break any retain cycle from the session holding the delegate.
  903. [_session finishTasksAndInvalidate];
  904. // Immediately clear the session so no new task may be issued with it.
  905. //
  906. // The _taskToFetcherMap needs to stay valid until the outstanding tasks finish.
  907. _session = nil;
  908. }
  909. - (void)setFetcher:(GTMSessionFetcher *)fetcher forTask:(NSURLSessionTask *)task {
  910. GTMSESSION_ASSERT_DEBUG(fetcher != nil, @"missing fetcher");
  911. @synchronized(self) {
  912. GTMSessionMonitorSynchronized(self);
  913. if (_taskToFetcherMap == nil) {
  914. _taskToFetcherMap = [[NSMutableDictionary alloc] init];
  915. }
  916. if (fetcher) {
  917. [_taskToFetcherMap setObject:fetcher forKey:task];
  918. [self destroyDiscardTimer];
  919. }
  920. }
  921. }
  922. - (void)removeFetcher:(GTMSessionFetcher *)fetcher {
  923. @synchronized(self) {
  924. GTMSessionMonitorSynchronized(self);
  925. // Typically, a fetcher should be removed when its task invokes
  926. // URLSession:task:didCompleteWithError:.
  927. //
  928. // When fetching with a testBlock, though, the task completed delegate
  929. // method may not be invoked, requiring cleanup here.
  930. NSArray *tasks = [_taskToFetcherMap allKeysForObject:fetcher];
  931. GTMSESSION_ASSERT_DEBUG(tasks.count <= 1, @"fetcher task not unmapped: %@", tasks);
  932. [_taskToFetcherMap removeObjectsForKeys:tasks];
  933. if (_taskToFetcherMap.count == 0) {
  934. [self startDiscardTimer];
  935. }
  936. }
  937. }
  938. // This helper method provides synchronized access to the task map for the delegate
  939. // methods below.
  940. - (id)fetcherForTask:(NSURLSessionTask *)task {
  941. @synchronized(self) {
  942. GTMSessionMonitorSynchronized(self);
  943. return [_taskToFetcherMap objectForKey:task];
  944. }
  945. }
  946. - (void)removeTaskFromMap:(NSURLSessionTask *)task {
  947. @synchronized(self) {
  948. GTMSessionMonitorSynchronized(self);
  949. [_taskToFetcherMap removeObjectForKey:task];
  950. }
  951. }
  952. - (void)setSession:(NSURLSession *)session {
  953. @synchronized(self) {
  954. GTMSessionMonitorSynchronized(self);
  955. _session = session;
  956. }
  957. }
  958. - (NSURLSession *)session {
  959. @synchronized(self) {
  960. GTMSessionMonitorSynchronized(self);
  961. return _session;
  962. }
  963. }
  964. - (NSTimeInterval)discardInterval {
  965. @synchronized(self) {
  966. GTMSessionMonitorSynchronized(self);
  967. return _discardInterval;
  968. }
  969. }
  970. - (void)setDiscardInterval:(NSTimeInterval)interval {
  971. @synchronized(self) {
  972. GTMSessionMonitorSynchronized(self);
  973. _discardInterval = interval;
  974. }
  975. }
  976. // NSURLSessionDelegate protocol methods.
  977. // - (void)URLSessionDidFinishEventsForBackgroundURLSession:(NSURLSession *)session;
  978. //
  979. // TODO(seh): How do we route this to an appropriate fetcher?
  980. - (void)URLSession:(NSURLSession *)session didBecomeInvalidWithError:(NSError *)error {
  981. GTM_LOG_SESSION_DELEGATE(@"%@ %p URLSession:%@ didBecomeInvalidWithError:%@", [self class], self,
  982. session, error);
  983. NSDictionary *localTaskToFetcherMap;
  984. @synchronized(self) {
  985. GTMSessionMonitorSynchronized(self);
  986. _session = nil;
  987. localTaskToFetcherMap = [_taskToFetcherMap copy];
  988. }
  989. // Any "suspended" tasks may not have received callbacks from NSURLSession when the session
  990. // completes; we'll call them now.
  991. [localTaskToFetcherMap enumerateKeysAndObjectsUsingBlock:^(
  992. NSURLSessionTask *task, GTMSessionFetcher *fetcher, BOOL *stop) {
  993. if (fetcher.session == session) {
  994. // Our delegate method URLSession:task:didCompleteWithError: will rely on
  995. // _taskToFetcherMap so that should still contain this fetcher.
  996. NSError *canceledError = [NSError errorWithDomain:NSURLErrorDomain
  997. code:NSURLErrorCancelled
  998. userInfo:nil];
  999. [self URLSession:session task:task didCompleteWithError:canceledError];
  1000. } else {
  1001. GTMSESSION_ASSERT_DEBUG(0, @"Unexpected session in fetcher: %@ has %@ (expected %@)", fetcher,
  1002. fetcher.session, session);
  1003. }
  1004. }];
  1005. // Our tests rely on this notification to know the session discard timer fired.
  1006. NSDictionary *userInfo = @{kGTMSessionFetcherServiceSessionKey : session};
  1007. NSNotificationCenter *nc = [NSNotificationCenter defaultCenter];
  1008. [nc postNotificationName:kGTMSessionFetcherServiceSessionBecameInvalidNotification
  1009. object:_parentService
  1010. userInfo:userInfo];
  1011. }
  1012. #pragma mark - NSURLSessionTaskDelegate
  1013. // NSURLSessionTaskDelegate protocol methods.
  1014. //
  1015. // We won't test here if the fetcher responds to these since we only want this
  1016. // class to implement the same delegate methods the fetcher does (so NSURLSession's
  1017. // tests for respondsToSelector: will have the same result whether the session
  1018. // delegate is the fetcher or this dispatcher.)
  1019. - (void)URLSession:(NSURLSession *)session
  1020. task:(NSURLSessionTask *)task
  1021. willPerformHTTPRedirection:(NSHTTPURLResponse *)response
  1022. newRequest:(NSURLRequest *)request
  1023. completionHandler:(void (^)(NSURLRequest *))completionHandler {
  1024. id<NSURLSessionTaskDelegate> fetcher = [self fetcherForTask:task];
  1025. [fetcher URLSession:session
  1026. task:task
  1027. willPerformHTTPRedirection:response
  1028. newRequest:request
  1029. completionHandler:completionHandler];
  1030. }
  1031. - (void)URLSession:(NSURLSession *)session
  1032. task:(NSURLSessionTask *)task
  1033. didReceiveChallenge:(NSURLAuthenticationChallenge *)challenge
  1034. completionHandler:(void (^)(NSURLSessionAuthChallengeDisposition, NSURLCredential *))handler {
  1035. id<NSURLSessionTaskDelegate> fetcher = [self fetcherForTask:task];
  1036. [fetcher URLSession:session task:task didReceiveChallenge:challenge completionHandler:handler];
  1037. }
  1038. - (void)URLSession:(NSURLSession *)session
  1039. task:(NSURLSessionTask *)task
  1040. needNewBodyStream:(void (^)(NSInputStream *bodyStream))handler {
  1041. id<NSURLSessionTaskDelegate> fetcher = [self fetcherForTask:task];
  1042. [fetcher URLSession:session task:task needNewBodyStream:handler];
  1043. }
  1044. - (void)URLSession:(NSURLSession *)session
  1045. task:(NSURLSessionTask *)task
  1046. didSendBodyData:(int64_t)bytesSent
  1047. totalBytesSent:(int64_t)totalBytesSent
  1048. totalBytesExpectedToSend:(int64_t)totalBytesExpectedToSend {
  1049. id<NSURLSessionTaskDelegate> fetcher = [self fetcherForTask:task];
  1050. [fetcher URLSession:session
  1051. task:task
  1052. didSendBodyData:bytesSent
  1053. totalBytesSent:totalBytesSent
  1054. totalBytesExpectedToSend:totalBytesExpectedToSend];
  1055. }
  1056. - (void)URLSession:(NSURLSession *)session
  1057. task:(NSURLSessionTask *)task
  1058. didCompleteWithError:(NSError *)error {
  1059. id<NSURLSessionTaskDelegate> fetcher = [self fetcherForTask:task];
  1060. // This is the usual way tasks are removed from the task map.
  1061. [self removeTaskFromMap:task];
  1062. [fetcher URLSession:session task:task didCompleteWithError:error];
  1063. }
  1064. - (void)URLSession:(NSURLSession *)session
  1065. task:(NSURLSessionTask *)task
  1066. didFinishCollectingMetrics:(NSURLSessionTaskMetrics *)metrics
  1067. API_AVAILABLE(ios(10.0), macosx(10.12), tvos(10.0), watchos(6.0)) {
  1068. id<NSURLSessionTaskDelegate> fetcher = [self fetcherForTask:task];
  1069. [fetcher URLSession:session task:task didFinishCollectingMetrics:metrics];
  1070. }
  1071. // NSURLSessionDataDelegate protocol methods.
  1072. - (void)URLSession:(NSURLSession *)session
  1073. dataTask:(NSURLSessionDataTask *)dataTask
  1074. didReceiveResponse:(NSURLResponse *)response
  1075. completionHandler:(void (^)(NSURLSessionResponseDisposition))handler {
  1076. id<NSURLSessionDataDelegate> fetcher = [self fetcherForTask:dataTask];
  1077. [fetcher URLSession:session
  1078. dataTask:dataTask
  1079. didReceiveResponse:response
  1080. completionHandler:handler];
  1081. }
  1082. - (void)URLSession:(NSURLSession *)session
  1083. dataTask:(NSURLSessionDataTask *)dataTask
  1084. didBecomeDownloadTask:(NSURLSessionDownloadTask *)downloadTask {
  1085. id<NSURLSessionDataDelegate> fetcher = [self fetcherForTask:dataTask];
  1086. GTMSESSION_ASSERT_DEBUG(fetcher != nil, @"Missing fetcher for %@", dataTask);
  1087. [self removeTaskFromMap:dataTask];
  1088. if (fetcher) {
  1089. GTMSESSION_ASSERT_DEBUG([fetcher isKindOfClass:[GTMSessionFetcher class]],
  1090. @"Expecting GTMSessionFetcher");
  1091. [self setFetcher:(GTMSessionFetcher *)fetcher forTask:downloadTask];
  1092. }
  1093. [fetcher URLSession:session dataTask:dataTask didBecomeDownloadTask:downloadTask];
  1094. }
  1095. - (void)URLSession:(NSURLSession *)session
  1096. dataTask:(NSURLSessionDataTask *)dataTask
  1097. didReceiveData:(NSData *)data {
  1098. id<NSURLSessionDataDelegate> fetcher = [self fetcherForTask:dataTask];
  1099. [fetcher URLSession:session dataTask:dataTask didReceiveData:data];
  1100. }
  1101. - (void)URLSession:(NSURLSession *)session
  1102. dataTask:(NSURLSessionDataTask *)dataTask
  1103. willCacheResponse:(NSCachedURLResponse *)proposedResponse
  1104. completionHandler:(void (^)(NSCachedURLResponse *))handler {
  1105. id<NSURLSessionDataDelegate> fetcher = [self fetcherForTask:dataTask];
  1106. [fetcher URLSession:session
  1107. dataTask:dataTask
  1108. willCacheResponse:proposedResponse
  1109. completionHandler:handler];
  1110. }
  1111. // NSURLSessionDownloadDelegate protocol methods.
  1112. - (void)URLSession:(NSURLSession *)session
  1113. downloadTask:(NSURLSessionDownloadTask *)downloadTask
  1114. didFinishDownloadingToURL:(NSURL *)location {
  1115. id<NSURLSessionDownloadDelegate> fetcher = [self fetcherForTask:downloadTask];
  1116. [fetcher URLSession:session downloadTask:downloadTask didFinishDownloadingToURL:location];
  1117. }
  1118. - (void)URLSession:(NSURLSession *)session
  1119. downloadTask:(NSURLSessionDownloadTask *)downloadTask
  1120. didWriteData:(int64_t)bytesWritten
  1121. totalBytesWritten:(int64_t)totalWritten
  1122. totalBytesExpectedToWrite:(int64_t)totalExpected {
  1123. id<NSURLSessionDownloadDelegate> fetcher = [self fetcherForTask:downloadTask];
  1124. [fetcher URLSession:session
  1125. downloadTask:downloadTask
  1126. didWriteData:bytesWritten
  1127. totalBytesWritten:totalWritten
  1128. totalBytesExpectedToWrite:totalExpected];
  1129. }
  1130. - (void)URLSession:(NSURLSession *)session
  1131. downloadTask:(NSURLSessionDownloadTask *)downloadTask
  1132. didResumeAtOffset:(int64_t)fileOffset
  1133. expectedTotalBytes:(int64_t)expectedTotalBytes {
  1134. id<NSURLSessionDownloadDelegate> fetcher = [self fetcherForTask:downloadTask];
  1135. [fetcher URLSession:session
  1136. downloadTask:downloadTask
  1137. didResumeAtOffset:fileOffset
  1138. expectedTotalBytes:expectedTotalBytes];
  1139. }
  1140. @end