GTMSessionFetcher.h 62 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386
  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. // GTMSessionFetcher is a wrapper around NSURLSession for http operations.
  16. //
  17. // What does this offer on top of of NSURLSession?
  18. //
  19. // - Block-style callbacks for useful functionality like progress rather
  20. // than delegate methods.
  21. // - Out-of-process uploads and downloads using NSURLSession, including
  22. // management of fetches after relaunch.
  23. // - Integration with GTMAppAuth for invisible management and refresh of
  24. // authorization tokens.
  25. // - Pretty-printed http logging.
  26. // - Cookies handling that does not interfere with or get interfered with
  27. // by WebKit cookies or on Mac by Safari and other apps.
  28. // - Credentials handling for the http operation.
  29. // - Rate-limiting and cookie grouping when fetchers are created with
  30. // GTMSessionFetcherService.
  31. //
  32. // If the bodyData or bodyFileURL property is set, then a POST request is assumed.
  33. //
  34. // Each fetcher is assumed to be for a one-shot fetch request; don't reuse the object
  35. // for a second fetch.
  36. //
  37. // The fetcher will be self-retained as long as a connection is pending.
  38. //
  39. // To keep user activity private, URLs must have an https scheme (unless the property
  40. // allowedInsecureSchemes is set to permit the scheme.)
  41. //
  42. // Callbacks will be released when the fetch completes or is stopped, so there is no need
  43. // to use weak self references in the callback blocks.
  44. //
  45. // Sample usage:
  46. //
  47. // _fetcherService = [[GTMSessionFetcherService alloc] init];
  48. //
  49. // GTMSessionFetcher *myFetcher = [_fetcherService fetcherWithURLString:myURLString];
  50. // myFetcher.retryEnabled = YES;
  51. // myFetcher.comment = @"First profile image";
  52. //
  53. // // Optionally specify a file URL or NSData for the request body to upload.
  54. // myFetcher.bodyData = [postString dataUsingEncoding:NSUTF8StringEncoding];
  55. //
  56. // [myFetcher beginFetchWithCompletionHandler:^(NSData *data, NSError *error) {
  57. // if (error != nil) {
  58. // // Server status code or network error.
  59. // //
  60. // // If the domain is kGTMSessionFetcherStatusDomain then the error code
  61. // // is a failure status from the server.
  62. // } else {
  63. // // Fetch succeeded.
  64. // }
  65. // }];
  66. //
  67. // There is also a beginFetch call that takes a pointer and selector for the completion handler;
  68. // a pointer and selector is a better style when the callback is a substantial, separate method.
  69. //
  70. // NOTE: Fetches may retrieve data from the server even though the server
  71. // returned an error, so the criteria for success is a non-nil error.
  72. // The completion handler is called when the server status is >= 300 with an NSError
  73. // having domain kGTMSessionFetcherStatusDomain and code set to the server status.
  74. //
  75. // Status codes are at <http://www.w3.org/Protocols/rfc2616/rfc2616-sec10.html>
  76. //
  77. //
  78. // Background session support:
  79. //
  80. // Out-of-process uploads and downloads may be created by setting the fetcher's
  81. // useBackgroundSession property. Data to be uploaded should be provided via
  82. // the uploadFileURL property; the download destination should be specified with
  83. // the destinationFileURL. NOTE: Background upload files should be in a location
  84. // that will be valid even after the device is restarted, so the file should not
  85. // be uploaded from a system temporary or cache directory.
  86. //
  87. // Background session transfers are slower, and should typically be used only
  88. // for very large downloads or uploads (hundreds of megabytes).
  89. //
  90. // When background sessions are used in iOS apps, the application delegate must
  91. // pass through the parameters from UIApplicationDelegate's
  92. // application:handleEventsForBackgroundURLSession:completionHandler: to the
  93. // fetcher class.
  94. //
  95. // When the application has been relaunched, it may also create a new fetcher
  96. // instance to handle completion of the transfers.
  97. //
  98. // - (void)application:(UIApplication *)application
  99. // handleEventsForBackgroundURLSession:(NSString *)identifier
  100. // completionHandler:(void (^)())completionHandler {
  101. // // Application was re-launched on completing an out-of-process download.
  102. //
  103. // // Pass the URLSession info related to this re-launch to the fetcher class.
  104. // [GTMSessionFetcher application:application
  105. // handleEventsForBackgroundURLSession:identifier
  106. // completionHandler:completionHandler];
  107. //
  108. // // Get a fetcher related to this re-launch and re-hook up a completionHandler to it.
  109. // GTMSessionFetcher *fetcher = [GTMSessionFetcher fetcherWithSessionIdentifier:identifier];
  110. // NSURL *destinationFileURL = fetcher.destinationFileURL;
  111. // fetcher.completionHandler = ^(NSData *data, NSError *error) {
  112. // [self downloadCompletedToFile:destinationFileURL error:error];
  113. // };
  114. // }
  115. //
  116. //
  117. // Threading and queue support:
  118. //
  119. // Networking always happens on a background thread; there is no advantage to
  120. // changing thread or queue to create or start a fetcher.
  121. //
  122. // Callbacks are run on the main thread; alternatively, the app may set the
  123. // fetcher's callbackQueue to a dispatch queue.
  124. //
  125. // Once the fetcher's beginFetch method has been called, the fetcher's methods and
  126. // properties may be accessed from any thread.
  127. //
  128. // Downloading to disk:
  129. //
  130. // To have downloaded data saved directly to disk, specify a file URL for the
  131. // destinationFileURL property.
  132. //
  133. // HTTP methods and headers:
  134. //
  135. // Alternative HTTP methods, like PUT, and custom headers can be specified by
  136. // creating the fetcher with an appropriate NSMutableURLRequest.
  137. //
  138. // Custom headers can also be provided per-request via an instance of `GTMFetcherDecoratorProtocol`
  139. // passed to `-[GTMSessionFetcherService addDecorator:]`.
  140. //
  141. // Caching:
  142. //
  143. // The fetcher avoids caching. That is best for API requests, but may hurt
  144. // repeat fetches of static data. Apps may enable a persistent disk cache by
  145. // customizing the config:
  146. //
  147. // fetcher.configurationBlock = ^(GTMSessionFetcher *configFetcher,
  148. // NSURLSessionConfiguration *config) {
  149. // config.URLCache = [NSURLCache sharedURLCache];
  150. // };
  151. //
  152. // Or use the standard system config to share cookie storage with web views
  153. // and to enable disk caching:
  154. //
  155. // fetcher.configuration = [NSURLSessionConfiguration defaultSessionConfiguration];
  156. //
  157. //
  158. // Cookies:
  159. //
  160. // There are three supported mechanisms for remembering cookies between fetches.
  161. //
  162. // By default, a standalone GTMSessionFetcher uses a mutable array held
  163. // statically to track cookies for all instantiated fetchers. This avoids
  164. // cookies being set by servers for the application from interfering with
  165. // Safari and WebKit cookie settings, and vice versa.
  166. // The fetcher cookies are lost when the application quits.
  167. //
  168. // To rely instead on WebKit's global NSHTTPCookieStorage, set the fetcher's
  169. // cookieStorage property:
  170. // myFetcher.cookieStorage = [NSHTTPCookieStorage sharedHTTPCookieStorage];
  171. //
  172. // To share cookies with other apps, use the method introduced in iOS 9/OS X 10.11:
  173. // myFetcher.cookieStorage =
  174. // [NSHTTPCookieStorage sharedCookieStorageForGroupContainerIdentifier:kMyCompanyContainedID];
  175. //
  176. // To ignore existing cookies and only have cookies related to the single fetch
  177. // be applied, make a temporary cookie storage object:
  178. // myFetcher.cookieStorage = [[GTMSessionCookieStorage alloc] init];
  179. //
  180. // Note: cookies set while following redirects will be sent to the server, as
  181. // the redirects are followed by the fetcher.
  182. //
  183. // To completely disable cookies, adjust the session configuration appropriately
  184. // in the fetcher or fetcher service:
  185. // fetcher.configurationBlock = ^(GTMSessionFetcher *configFetcher,
  186. // NSURLSessionConfiguration *config) {
  187. // config.HTTPCookieAcceptPolicy = NSHTTPCookieAcceptPolicyNever;
  188. // config.HTTPShouldSetCookies = NO;
  189. // };
  190. //
  191. // If the fetcher is created from a GTMSessionFetcherService object
  192. // then the cookie storage mechanism is set to use the cookie storage in the
  193. // service object rather than the static storage. Disabling cookies in the
  194. // session configuration set on a service object will disable cookies for all
  195. // fetchers created from that GTMSessionFetcherService object, since the session
  196. // configuration is propagated to the fetcher.
  197. //
  198. //
  199. // Monitoring data transfers.
  200. //
  201. // The fetcher supports a variety of properties for progress monitoring
  202. // progress with callback blocks.
  203. // GTMSessionFetcherSendProgressBlock sendProgressBlock
  204. // GTMSessionFetcherReceivedProgressBlock receivedProgressBlock
  205. // GTMSessionFetcherDownloadProgressBlock downloadProgressBlock
  206. //
  207. // If supplied by the server, the anticipated total download size is available
  208. // as [[myFetcher response] expectedContentLength] (and may be -1 for unknown
  209. // download sizes.)
  210. //
  211. //
  212. // Automatic retrying of fetches
  213. //
  214. // The fetcher can optionally create a timer and reattempt certain kinds of
  215. // fetch failures (status codes 408, request timeout; 502, gateway failure;
  216. // 503, service unavailable; 504, gateway timeout; networking errors
  217. // NSURLErrorTimedOut and NSURLErrorNetworkConnectionLost.) The user may
  218. // set a retry selector to customize the type of errors which will be retried.
  219. //
  220. // Retries are done in an exponential-backoff fashion (that is, after 1 second,
  221. // 2, 4, 8, and so on.)
  222. //
  223. // Enabling automatic retries looks like this:
  224. // myFetcher.retryEnabled = YES;
  225. //
  226. // With retries enabled, the completion callbacks are called only
  227. // when no more retries will be attempted. Calling the fetcher's stopFetching
  228. // method will terminate the retry timer, without the finished or failure
  229. // selectors being invoked.
  230. //
  231. // Optionally, the client may set the maximum retry interval:
  232. // myFetcher.maxRetryInterval = 60.0; // in seconds; default is 60 seconds
  233. // // for downloads, 600 for uploads
  234. //
  235. // Servers should never send a 400 or 500 status for errors that are retryable
  236. // by clients, as those values indicate permanent failures. In nearly all
  237. // cases, the default standard retry behavior is correct for clients, and no
  238. // custom client retry behavior is needed or appropriate. Servers that send
  239. // non-retryable status codes and expect the client to retry the request are
  240. // faulty.
  241. //
  242. // Still, the client may provide a block to determine if a status code or other
  243. // error should be retried. The block returns YES to set the retry timer or NO
  244. // to fail without additional fetch attempts.
  245. //
  246. // The retry method may return the |suggestedWillRetry| argument to get the
  247. // default retry behavior. Server status codes are present in the
  248. // error argument, and have the domain kGTMSessionFetcherStatusDomain. The
  249. // user's method may look something like this:
  250. //
  251. // myFetcher.retryBlock = ^(BOOL suggestedWillRetry, NSError *error,
  252. // GTMSessionFetcherRetryResponse response) {
  253. // // Perhaps examine error.domain and error.code, or fetcher.retryCount
  254. // //
  255. // // Respond with YES to start the retry timer, NO to proceed to the failure
  256. // // callback, or suggestedWillRetry to get default behavior for the
  257. // // current error domain and code values.
  258. // response(suggestedWillRetry);
  259. // };
  260. #import <Foundation/Foundation.h>
  261. #if TARGET_OS_IPHONE
  262. #import <UIKit/UIKit.h>
  263. #endif
  264. #if TARGET_OS_WATCH
  265. #import <WatchKit/WatchKit.h>
  266. #endif
  267. // By default it is stripped from non DEBUG builds. Developers can override
  268. // this in their project settings.
  269. #ifndef STRIP_GTM_FETCH_LOGGING
  270. #if !DEBUG
  271. #define STRIP_GTM_FETCH_LOGGING 1
  272. #else
  273. #define STRIP_GTM_FETCH_LOGGING 0
  274. #endif
  275. #endif
  276. // Logs in debug builds.
  277. #ifndef GTMSESSION_LOG_DEBUG
  278. #if DEBUG
  279. #define GTMSESSION_LOG_DEBUG(...) NSLog(__VA_ARGS__)
  280. #else
  281. #define GTMSESSION_LOG_DEBUG(...) \
  282. do { \
  283. } while (0)
  284. #endif
  285. #endif
  286. // Asserts in debug builds (or logs in debug builds if GTMSESSION_ASSERT_AS_LOG
  287. // or NS_BLOCK_ASSERTIONS are defined.)
  288. #ifndef GTMSESSION_ASSERT_DEBUG
  289. #if DEBUG && !defined(NS_BLOCK_ASSERTIONS) && !GTMSESSION_ASSERT_AS_LOG
  290. #undef GTMSESSION_ASSERT_AS_LOG
  291. #define GTMSESSION_ASSERT_AS_LOG 1
  292. #endif
  293. #if DEBUG && !GTMSESSION_ASSERT_AS_LOG
  294. #define GTMSESSION_ASSERT_DEBUG(...) NSAssert(__VA_ARGS__)
  295. #elif DEBUG
  296. #define GTMSESSION_ASSERT_DEBUG(pred, ...) \
  297. if (!(pred)) { \
  298. NSLog(__VA_ARGS__); \
  299. }
  300. #else
  301. #define GTMSESSION_ASSERT_DEBUG(pred, ...) \
  302. do { \
  303. } while (0)
  304. #endif
  305. #endif
  306. // Asserts in debug builds, logs in release builds (or logs in debug builds if
  307. // GTMSESSION_ASSERT_AS_LOG is defined.)
  308. #ifndef GTMSESSION_ASSERT_DEBUG_OR_LOG
  309. #if DEBUG && !GTMSESSION_ASSERT_AS_LOG
  310. #define GTMSESSION_ASSERT_DEBUG_OR_LOG(...) NSAssert(__VA_ARGS__)
  311. #else
  312. #define GTMSESSION_ASSERT_DEBUG_OR_LOG(pred, ...) \
  313. if (!(pred)) { \
  314. NSLog(__VA_ARGS__); \
  315. }
  316. #endif
  317. #endif
  318. // Macro useful for examining messages from NSURLSession during debugging.
  319. #if 0
  320. #define GTM_LOG_SESSION_DELEGATE(...) GTMSESSION_LOG_DEBUG(__VA_ARGS__)
  321. #else
  322. #define GTM_LOG_SESSION_DELEGATE(...)
  323. #endif
  324. // These will be removed in the near future, folks should move off of them.
  325. #ifndef GTM_NULLABLE
  326. #if __has_feature(nullability) // Available starting in Xcode 6.3
  327. #define GTM_NULLABLE_TYPE __nullable
  328. #define GTM_NONNULL_TYPE __nonnull
  329. #define GTM_NULLABLE nullable
  330. #define GTM_NONNULL_DECL nonnull // GTM_NONNULL is used by GTMDefines.h
  331. #define GTM_NULL_RESETTABLE null_resettable
  332. #define GTM_ASSUME_NONNULL_BEGIN NS_ASSUME_NONNULL_BEGIN
  333. #define GTM_ASSUME_NONNULL_END NS_ASSUME_NONNULL_END
  334. #else
  335. #define GTM_NULLABLE_TYPE
  336. #define GTM_NONNULL_TYPE
  337. #define GTM_NULLABLE
  338. #define GTM_NONNULL_DECL
  339. #define GTM_NULL_RESETTABLE
  340. #define GTM_ASSUME_NONNULL_BEGIN
  341. #define GTM_ASSUME_NONNULL_END
  342. #endif // __has_feature(nullability)
  343. #endif // GTM_NULLABLE
  344. #ifndef GTM_DECLARE_GENERICS
  345. #if __has_feature(objc_generics)
  346. #define GTM_DECLARE_GENERICS 1
  347. #else
  348. #define GTM_DECLARE_GENERICS 0
  349. #endif
  350. #endif
  351. #ifndef GTM_NSArrayOf
  352. #if GTM_DECLARE_GENERICS
  353. #define GTM_NSArrayOf(value) NSArray<value>
  354. #define GTM_NSDictionaryOf(key, value) NSDictionary<key, value>
  355. #else
  356. #define GTM_NSArrayOf(value) NSArray
  357. #define GTM_NSDictionaryOf(key, value) NSDictionary
  358. #endif // __has_feature(objc_generics)
  359. #endif // GTM_NSArrayOf
  360. // For iOS, the fetcher can declare itself a background task to allow fetches
  361. // to finish when the app leaves the foreground.
  362. //
  363. // (This is unrelated to providing a background configuration, which allows
  364. // out-of-process uploads and downloads.)
  365. //
  366. // To disallow use of background tasks during fetches, the target should define
  367. // GTM_BACKGROUND_TASK_FETCHING to 0, or alternatively may set the
  368. // skipBackgroundTask property to YES.
  369. #if !defined(GTM_BACKGROUND_TASK_FETCHING) && \
  370. (TARGET_OS_IOS || TARGET_OS_TV || TARGET_OS_MACCATALYST)
  371. #define GTM_BACKGROUND_TASK_FETCHING 1
  372. #endif
  373. #ifdef __cplusplus
  374. extern "C" {
  375. #endif
  376. #if !defined(GTMBridgeFetcher)
  377. // The bridge macros are deprecated, and should be replaced; GTMHTTPFetcher is no longer
  378. // supported and all code should switch to use GTMSessionFetcher types directly.
  379. #define GTMBridgeFetcher GTMSessionFetcher
  380. #define GTMBridgeFetcherService GTMSessionFetcherService
  381. #define GTMBridgeFetcherServiceProtocol GTMSessionFetcherServiceProtocol
  382. #define GTMBridgeAssertValidSelector GTMSessionFetcherAssertValidSelector
  383. #define GTMBridgeCookieStorage GTMSessionCookieStorage
  384. #define GTMBridgeCleanedUserAgentString GTMFetcherCleanedUserAgentString
  385. #define GTMBridgeSystemVersionString GTMFetcherSystemVersionString
  386. #define GTMBridgeApplicationIdentifier GTMFetcherApplicationIdentifier
  387. #define kGTMBridgeFetcherStatusDomain kGTMSessionFetcherStatusDomain
  388. #define kGTMBridgeFetcherStatusBadRequest GTMSessionFetcherStatusBadRequest
  389. #endif
  390. // When creating background sessions to perform out-of-process uploads and
  391. // downloads, on app launch any background sessions must be reconnected in
  392. // order to receive events that occurred while the app was not running.
  393. //
  394. // The fetcher will automatically attempt to recreate the sessions on app
  395. // start, but doing so reads from NSUserDefaults. This may have launch-time
  396. // performance impacts.
  397. //
  398. // To avoid launch performance impacts, on iPhone/iPad with iOS 13+ the
  399. // GTMSessionFetcher class will register for the app launch notification and
  400. // perform the reconnect then.
  401. //
  402. // Apps targeting Mac or older iOS SDKs can opt into the new behavior by defining
  403. // GTMSESSION_RECONNECT_BACKGROUND_SESSIONS_ON_LAUNCH=1.
  404. //
  405. // Apps targeting new SDKs can force the old behavior by defining
  406. // GTMSESSION_RECONNECT_BACKGROUND_SESSIONS_ON_LAUNCH = 0.
  407. #ifndef GTMSESSION_RECONNECT_BACKGROUND_SESSIONS_ON_LAUNCH
  408. // Default to the on-launch behavior for iOS 13+.
  409. #if TARGET_OS_IOS && defined(__IPHONE_13_0) && __IPHONE_OS_VERSION_MAX_ALLOWED >= __IPHONE_13_0
  410. #define GTMSESSION_RECONNECT_BACKGROUND_SESSIONS_ON_LAUNCH 1
  411. #else
  412. #define GTMSESSION_RECONNECT_BACKGROUND_SESSIONS_ON_LAUNCH 0
  413. #endif
  414. #endif
  415. NS_ASSUME_NONNULL_BEGIN
  416. // Notifications
  417. //
  418. // Fetch started and stopped, and fetch retry delay started and stopped.
  419. extern NSString *const kGTMSessionFetcherStartedNotification;
  420. extern NSString *const kGTMSessionFetcherStoppedNotification;
  421. extern NSString *const kGTMSessionFetcherRetryDelayStartedNotification;
  422. extern NSString *const kGTMSessionFetcherRetryDelayStoppedNotification;
  423. // Completion handler notification. This is intended for use by code capturing
  424. // and replaying fetch requests and results for testing. For fetches where
  425. // destinationFileURL or accumulateDataBlock is set for the fetcher, the data
  426. // will be nil for successful fetches.
  427. //
  428. // This notification is posted on the main thread.
  429. extern NSString *const kGTMSessionFetcherCompletionInvokedNotification;
  430. extern NSString *const kGTMSessionFetcherCompletionDataKey;
  431. extern NSString *const kGTMSessionFetcherCompletionErrorKey;
  432. // Constants for NSErrors created by the fetcher (excluding server status errors,
  433. // and error objects originating in the OS.)
  434. extern NSString *const kGTMSessionFetcherErrorDomain;
  435. // The fetcher turns server error status values (3XX, 4XX, 5XX) into NSErrors
  436. // with domain kGTMSessionFetcherStatusDomain.
  437. //
  438. // Any server response body data accompanying the status error is added to the
  439. // userInfo dictionary with key kGTMSessionFetcherStatusDataKey.
  440. extern NSString *const kGTMSessionFetcherStatusDomain;
  441. extern NSString *const kGTMSessionFetcherStatusDataKey;
  442. extern NSString *const kGTMSessionFetcherStatusDataContentTypeKey;
  443. // When a fetch fails with an error, these keys are included in the error userInfo
  444. // dictionary if retries were attempted.
  445. extern NSString *const kGTMSessionFetcherNumberOfRetriesDoneKey;
  446. extern NSString *const kGTMSessionFetcherElapsedIntervalWithRetriesKey;
  447. // Background session support requires access to NSUserDefaults.
  448. // If [NSUserDefaults standardUserDefaults] doesn't yield the correct NSUserDefaults for your usage,
  449. // ie for an App Extension, then implement this class/method to return the correct NSUserDefaults.
  450. // https://developer.apple.com/library/ios/documentation/General/Conceptual/ExtensibilityPG/ExtensionScenarios.html#//apple_ref/doc/uid/TP40014214-CH21-SW6
  451. @interface GTMSessionFetcherUserDefaultsFactory : NSObject
  452. + (NSUserDefaults *)fetcherUserDefaults;
  453. @end
  454. #ifdef __cplusplus
  455. }
  456. #endif
  457. typedef NS_ENUM(NSInteger, GTMSessionFetcherError) {
  458. GTMSessionFetcherErrorDownloadFailed = -1,
  459. GTMSessionFetcherErrorUploadChunkUnavailable = -2,
  460. GTMSessionFetcherErrorBackgroundExpiration = -3,
  461. GTMSessionFetcherErrorBackgroundFetchFailed = -4,
  462. GTMSessionFetcherErrorInsecureRequest = -5,
  463. GTMSessionFetcherErrorTaskCreationFailed = -6,
  464. };
  465. typedef NS_ENUM(NSInteger, GTMSessionFetcherStatus) {
  466. // Standard http status codes.
  467. GTMSessionFetcherStatusNotModified = 304,
  468. GTMSessionFetcherStatusBadRequest = 400,
  469. GTMSessionFetcherStatusUnauthorized = 401,
  470. GTMSessionFetcherStatusForbidden = 403,
  471. GTMSessionFetcherStatusPreconditionFailed = 412
  472. };
  473. #ifdef __cplusplus
  474. extern "C" {
  475. #endif
  476. @class GTMSessionCookieStorage;
  477. @class GTMSessionFetcher;
  478. // The configuration block is for modifying the NSURLSessionConfiguration only.
  479. // DO NOT change any fetcher properties in the configuration block.
  480. typedef void (^GTMSessionFetcherConfigurationBlock)(GTMSessionFetcher *fetcher,
  481. NSURLSessionConfiguration *configuration);
  482. typedef void (^GTMSessionFetcherSystemCompletionHandler)(void);
  483. typedef void (^GTMSessionFetcherCompletionHandler)(NSData *_Nullable data,
  484. NSError *_Nullable error);
  485. typedef void (^GTMSessionFetcherBodyStreamProviderResponse)(NSInputStream *bodyStream);
  486. typedef void (^GTMSessionFetcherBodyStreamProvider)(
  487. GTMSessionFetcherBodyStreamProviderResponse response);
  488. typedef void (^GTMSessionFetcherDidReceiveResponseDispositionBlock)(
  489. NSURLSessionResponseDisposition disposition);
  490. typedef void (^GTMSessionFetcherDidReceiveResponseBlock)(
  491. NSURLResponse *response, GTMSessionFetcherDidReceiveResponseDispositionBlock dispositionBlock);
  492. typedef void (^GTMSessionFetcherChallengeDispositionBlock)(
  493. NSURLSessionAuthChallengeDisposition disposition, NSURLCredential *_Nullable credential);
  494. typedef void (^GTMSessionFetcherChallengeBlock)(
  495. GTMSessionFetcher *fetcher, NSURLAuthenticationChallenge *challenge,
  496. GTMSessionFetcherChallengeDispositionBlock dispositionBlock);
  497. typedef void (^GTMSessionFetcherWillRedirectResponse)(NSURLRequest *_Nullable redirectedRequest);
  498. typedef void (^GTMSessionFetcherWillRedirectBlock)(NSHTTPURLResponse *redirectResponse,
  499. NSURLRequest *redirectRequest,
  500. GTMSessionFetcherWillRedirectResponse response);
  501. typedef void (^GTMSessionFetcherAccumulateDataBlock)(NSData *_Nullable buffer);
  502. typedef void (^GTMSessionFetcherSimulateByteTransferBlock)(NSData *_Nullable buffer,
  503. int64_t bytesWritten,
  504. int64_t totalBytesWritten,
  505. int64_t totalBytesExpectedToWrite);
  506. typedef void (^GTMSessionFetcherReceivedProgressBlock)(int64_t bytesWritten,
  507. int64_t totalBytesWritten);
  508. typedef void (^GTMSessionFetcherDownloadProgressBlock)(int64_t bytesWritten,
  509. int64_t totalBytesWritten,
  510. int64_t totalBytesExpectedToWrite);
  511. typedef void (^GTMSessionFetcherSendProgressBlock)(int64_t bytesSent, int64_t totalBytesSent,
  512. int64_t totalBytesExpectedToSend);
  513. typedef void (^GTMSessionFetcherWillCacheURLResponseResponse)(
  514. NSCachedURLResponse *_Nullable cachedResponse);
  515. typedef void (^GTMSessionFetcherWillCacheURLResponseBlock)(
  516. NSCachedURLResponse *proposedResponse,
  517. GTMSessionFetcherWillCacheURLResponseResponse responseBlock);
  518. typedef void (^GTMSessionFetcherRetryResponse)(BOOL shouldRetry);
  519. typedef void (^GTMSessionFetcherRetryBlock)(BOOL suggestedWillRetry, NSError *_Nullable error,
  520. GTMSessionFetcherRetryResponse response);
  521. API_AVAILABLE(ios(10.0), macosx(10.12), tvos(10.0), watchos(6.0))
  522. typedef void (^GTMSessionFetcherMetricsCollectionBlock)(NSURLSessionTaskMetrics *metrics);
  523. typedef void (^GTMSessionFetcherTestResponse)(NSHTTPURLResponse *_Nullable response,
  524. NSData *_Nullable data, NSError *_Nullable error);
  525. typedef void (^GTMSessionFetcherTestBlock)(GTMSessionFetcher *fetcherToTest,
  526. GTMSessionFetcherTestResponse testResponse);
  527. void GTMSessionFetcherAssertValidSelector(id _Nullable obj, SEL _Nullable sel, ...);
  528. // Utility functions for applications self-identifying to servers via a
  529. // user-agent header
  530. // The "standard" user agent includes the application identifier, taken from the bundle,
  531. // followed by a space and the system version string. Pass nil to use +mainBundle as the source
  532. // of the bundle identifier.
  533. //
  534. // Applications may use this as a starting point for their own user agent strings, perhaps
  535. // with additional sections appended. Use GTMFetcherCleanedUserAgentString() below to
  536. // clean up any string being added to the user agent.
  537. NSString *GTMFetcherStandardUserAgentString(NSBundle *_Nullable bundle);
  538. // Make a generic name and version for the current application, like
  539. // com.example.MyApp/1.2.3 relying on the bundle identifier and the
  540. // CFBundleShortVersionString or CFBundleVersion.
  541. //
  542. // The bundle ID may be overridden as the base identifier string by
  543. // adding to the bundle's Info.plist a "GTMUserAgentID" key.
  544. //
  545. // The application version may be overridden by adding to the bundle's
  546. // Info.plist a "GTMUserAgentVersion" key.
  547. //
  548. // If no bundle ID or override is available, the process name preceded
  549. // by "proc_" is used.
  550. NSString *GTMFetcherApplicationIdentifier(NSBundle *_Nullable bundle);
  551. // Make an identifier like "MacOSX/10.7.1" or "iPod_Touch/4.1 hw/iPod1_1"
  552. NSString *GTMFetcherSystemVersionString(void);
  553. // Make a parseable user-agent identifier from the given string, replacing whitespace
  554. // and commas with underscores, and removing other characters that may interfere
  555. // with parsing of the full user-agent string.
  556. //
  557. // For example, @"[My App]" would become @"My_App"
  558. NSString *GTMFetcherCleanedUserAgentString(NSString *str);
  559. // Grab the data from an input stream. Since streams cannot be assumed to be rewindable,
  560. // this may be destructive; the caller can try to rewind the stream (by setting the
  561. // NSStreamFileCurrentOffsetKey property) or can just use the NSData to make a new
  562. // NSInputStream. This function is intended to facilitate testing rather than be used in
  563. // production.
  564. //
  565. // This function operates synchronously on the current thread. Depending on how the
  566. // input stream is implemented, it may be appropriate to dispatch to a different
  567. // queue before calling this function.
  568. //
  569. // Failure is indicated by a returned data value of nil.
  570. NSData *_Nullable GTMDataFromInputStream(NSInputStream *inputStream, NSError **outError);
  571. #ifdef __cplusplus
  572. } // extern "C"
  573. #endif
  574. // Completion handler passed to -[GTMFetcherDecoratorProtocol fetcherWillStart:completionHandler:].
  575. typedef void (^GTMFetcherDecoratorFetcherWillStartCompletionHandler)(NSURLRequest *_Nullable,
  576. NSError *_Nullable);
  577. // Allows intercepting a request and optionally modifying it before the request (or a retry)
  578. // is sent. See `-[GTMSessionFetcherService addDecorator:]` and `-[GTMSessionFetcherService
  579. // removeDecorator:]`.
  580. //
  581. // Decorator methods must be thread-safe, as they might be invoked on any queue.
  582. @protocol GTMFetcherDecoratorProtocol <NSObject>
  583. // Invoked just before a fetcher's request starts.
  584. //
  585. // After the decorator's work is complete, the decorator must invoke `handler(request, error)`
  586. // either synchronously or asynchronously (on any queue).
  587. //
  588. // If no changes are to be made, pass `nil` for both `request` and `error`.
  589. //
  590. // Otherwise, if `error` is non-nil, then the fetcher is stopped with the given error, and any
  591. // further decorators' `-fetcherWillStart:completionHandler:` methods are not invoked.
  592. //
  593. // Otherwise, the decorator may use `[fetcher.request mutableCopy]`, make changes to the mutable
  594. // copy of the request, and pass the result to the handler via the `request` parameter.
  595. //
  596. // To distinguish the initial fetch from retries, the decorator can look at `fetcher.retryCount`.
  597. //
  598. // This method must not block the caller (e.g., performing synchronous I/O). Perform any blocking
  599. // work or I/O on a different queue, then invoke `handler` with the results after the blocking work
  600. // completes.
  601. - (void)fetcherWillStart:(GTMSessionFetcher *)fetcher
  602. completionHandler:(GTMFetcherDecoratorFetcherWillStartCompletionHandler)handler;
  603. // Invoked just after a fetcher's request finishes (either on success or on failure).
  604. //
  605. // After the decorator's work is complete, the decorator must invoke `handler()` either
  606. // synchronously or asynchronously (on any queue).
  607. //
  608. // To access the result of the fetch, the decorator can look at `fetcher.response`.
  609. //
  610. // This method must not block the caller (e.g., performing synchronous I/O). Perform any blocking
  611. // work or I/O on a different queue, then invoke `handler` with the results after the blocking work
  612. // completes.
  613. - (void)fetcherDidFinish:(GTMSessionFetcher *)fetcher
  614. withData:(nullable NSData *)data
  615. error:(nullable NSError *)error
  616. completionHandler:(void (^)(void))handler;
  617. @end
  618. // This protocol allows abstract references to the fetcher service, primarily for
  619. // fetchers (which may be compiled without the fetcher service class present.)
  620. //
  621. // Apps should not need to use this protocol.
  622. @protocol GTMSessionFetcherServiceProtocol <NSObject>
  623. // This protocol allows us to call into the service without requiring
  624. // GTMSessionFetcherService sources in this project
  625. @property(atomic, strong) dispatch_queue_t callbackQueue;
  626. - (BOOL)fetcherShouldBeginFetching:(GTMSessionFetcher *)fetcher;
  627. - (void)fetcherDidCreateSession:(GTMSessionFetcher *)fetcher;
  628. - (void)fetcherDidBeginFetching:(GTMSessionFetcher *)fetcher;
  629. - (void)fetcherDidStop:(GTMSessionFetcher *)fetcher;
  630. - (GTMSessionFetcher *)fetcherWithRequest:(NSURLRequest *)request;
  631. - (BOOL)isDelayingFetcher:(GTMSessionFetcher *)fetcher;
  632. @property(atomic, assign) BOOL reuseSession;
  633. - (nullable NSURLSession *)session;
  634. - (nullable NSURLSession *)sessionForFetcherCreation;
  635. - (nullable id<NSURLSessionDelegate>)sessionDelegate;
  636. - (nullable NSDate *)stoppedAllFetchersDate;
  637. @property(atomic, readonly, strong, nullable) NSOperationQueue *delegateQueue;
  638. @optional
  639. // This property is optional, for now, to enable releasing the feature without breaking existing
  640. // code that fakes the service but doesn't implement this.
  641. @property(atomic, readonly, strong, nullable) NSArray<id<GTMFetcherDecoratorProtocol>> *decorators;
  642. @end // @protocol GTMSessionFetcherServiceProtocol
  643. #ifndef GTM_FETCHER_AUTHORIZATION_PROTOCOL
  644. #define GTM_FETCHER_AUTHORIZATION_PROTOCOL 1
  645. @protocol GTMFetcherAuthorizationProtocol <NSObject>
  646. @required
  647. // This protocol allows us to call the authorizer without requiring its sources
  648. // in this project.
  649. - (void)authorizeRequest:(nullable NSMutableURLRequest *)request
  650. delegate:(id)delegate
  651. didFinishSelector:(SEL)sel;
  652. - (void)stopAuthorization;
  653. - (void)stopAuthorizationForRequest:(NSURLRequest *)request;
  654. - (BOOL)isAuthorizingRequest:(NSURLRequest *)request;
  655. - (BOOL)isAuthorizedRequest:(NSURLRequest *)request;
  656. @property(atomic, strong, readonly, nullable) NSString *userEmail;
  657. @optional
  658. // Indicate if authorization may be attempted. Even if this succeeds,
  659. // authorization may fail if the user's permissions have been revoked.
  660. @property(atomic, readonly) BOOL canAuthorize;
  661. // For development only, allow authorization of non-SSL requests, allowing
  662. // transmission of the bearer token unencrypted.
  663. @property(atomic, assign) BOOL shouldAuthorizeAllRequests;
  664. - (void)authorizeRequest:(nullable NSMutableURLRequest *)request
  665. completionHandler:(void (^)(NSError *_Nullable error))handler;
  666. @property(atomic, weak, nullable) id<GTMSessionFetcherServiceProtocol> fetcherService;
  667. - (BOOL)primeForRefresh;
  668. @end
  669. #endif // GTM_FETCHER_AUTHORIZATION_PROTOCOL
  670. #if GTM_BACKGROUND_TASK_FETCHING
  671. // A protocol for an alternative target for messages from GTMSessionFetcher to UIApplication.
  672. // Set the target using +[GTMSessionFetcher setSubstituteUIApplication:]
  673. @protocol GTMUIApplicationProtocol <NSObject>
  674. - (UIBackgroundTaskIdentifier)beginBackgroundTaskWithName:(nullable NSString *)taskName
  675. expirationHandler:(void (^__nullable)(void))handler;
  676. - (void)endBackgroundTask:(UIBackgroundTaskIdentifier)identifier;
  677. @end
  678. #endif
  679. #pragma mark -
  680. // GTMSessionFetcher objects are used for async retrieval of an http get or post
  681. //
  682. // See additional comments at the beginning of this file
  683. @interface GTMSessionFetcher : NSObject <NSURLSessionDelegate>
  684. // Create a fetcher
  685. //
  686. // fetcherWithRequest will return an autoreleased fetcher, but if
  687. // the connection is successfully created, the connection should retain the
  688. // fetcher for the life of the connection as well. So the caller doesn't have
  689. // to retain the fetcher explicitly unless they want to be able to cancel it.
  690. + (instancetype)fetcherWithRequest:(nullable NSURLRequest *)request;
  691. // Convenience methods that make a request, like +fetcherWithRequest
  692. + (instancetype)fetcherWithURL:(NSURL *)requestURL;
  693. + (instancetype)fetcherWithURLString:(NSString *)requestURLString;
  694. // Methods for creating fetchers to continue previous fetches.
  695. + (instancetype)fetcherWithDownloadResumeData:(NSData *)resumeData;
  696. + (nullable instancetype)fetcherWithSessionIdentifier:(NSString *)sessionIdentifier;
  697. // Returns an array of currently active fetchers for background sessions,
  698. // both restarted and newly created ones.
  699. + (NSArray<GTMSessionFetcher *> *)fetchersForBackgroundSessions;
  700. // Designated initializer.
  701. //
  702. // Applications should create fetchers with a "fetcherWith..." method on a fetcher
  703. // service or a class method, not with this initializer.
  704. //
  705. // The configuration should typically be nil. Applications needing to customize
  706. // the configuration may do so by setting the configurationBlock property.
  707. - (instancetype)initWithRequest:(nullable NSURLRequest *)request
  708. configuration:(nullable NSURLSessionConfiguration *)configuration;
  709. // The fetcher's request. This may not be set after beginFetch has been invoked. The request
  710. // may change due to redirects.
  711. @property(atomic, strong, nullable) NSURLRequest *request;
  712. // Set a header field value on the request. Header field value changes will not
  713. // affect a fetch after the fetch has begun.
  714. - (void)setRequestValue:(nullable NSString *)value forHTTPHeaderField:(NSString *)field;
  715. // Data used for resuming a download task.
  716. @property(atomic, readonly, nullable) NSData *downloadResumeData;
  717. // The configuration; this must be set before the fetch begins. If no configuration is
  718. // set or inherited from the fetcher service, then the fetcher uses an ephemeral config.
  719. //
  720. // NOTE: This property should typically be nil. Applications needing to customize
  721. // the configuration should do so by setting the configurationBlock property.
  722. // That allows the fetcher to pick an appropriate base configuration, with the
  723. // application setting only the configuration properties it needs to customize.
  724. @property(atomic, strong, nullable) NSURLSessionConfiguration *configuration;
  725. // A block the client may use to customize the configuration used to create the session.
  726. //
  727. // This is called synchronously, either on the thread that begins the fetch or, during a retry,
  728. // on the main thread. The configuration block may be called repeatedly if multiple fetchers are
  729. // created.
  730. //
  731. // The configuration block is for modifying the NSURLSessionConfiguration only.
  732. // DO NOT change any fetcher properties in the configuration block. Fetcher properties
  733. // may be set in the fetcher service prior to fetcher creation, or on the fetcher prior
  734. // to invoking beginFetch.
  735. @property(atomic, copy, nullable) GTMSessionFetcherConfigurationBlock configurationBlock;
  736. // A session is created as needed by the fetcher. A fetcher service object
  737. // may maintain sessions for multiple fetches to the same host.
  738. @property(atomic, strong, nullable) NSURLSession *session;
  739. // The task in flight.
  740. @property(atomic, readonly, nullable) NSURLSessionTask *sessionTask;
  741. // The background session identifier.
  742. @property(atomic, readonly, nullable) NSString *sessionIdentifier;
  743. // Indicates a fetcher created to finish a background session task.
  744. @property(atomic, readonly) BOOL wasCreatedFromBackgroundSession;
  745. // Indicates the client has committed to reconnecting this background session when
  746. // the app restarts. If this value is YES, the session fetcher will not automatically
  747. // call beginFetchWithCompletionHandler: on the restored fetcher on app start, and
  748. // the session will not handle system events until the client explicitly does.
  749. @property(atomic, assign) BOOL clientWillReconnectBackgroundSession;
  750. // Additional user-supplied data to encode into the session identifier. Since session identifier
  751. // length limits are unspecified, this should be kept small. Key names beginning with an underscore
  752. // are reserved for use by the fetcher.
  753. @property(atomic, strong, nullable) NSDictionary<NSString *, NSString *> *sessionUserInfo;
  754. // The human-readable description to be assigned to the task.
  755. @property(atomic, copy, nullable) NSString *taskDescription;
  756. // The priority assigned to the task, if any. Use NSURLSessionTaskPriorityLow,
  757. // NSURLSessionTaskPriorityDefault, or NSURLSessionTaskPriorityHigh.
  758. @property(atomic, assign) float taskPriority;
  759. // The fetcher encodes information used to resume a session in the session identifier.
  760. // This method, intended for internal use returns the encoded information. The sessionUserInfo
  761. // dictionary is stored as identifier metadata.
  762. - (nullable NSDictionary<NSString *, NSString *> *)sessionIdentifierMetadata;
  763. #if TARGET_OS_IPHONE && !TARGET_OS_WATCH
  764. // The app should pass to this method the completion handler passed in the app delegate method
  765. // application:handleEventsForBackgroundURLSession:completionHandler:
  766. + (void)application:(UIApplication *)application
  767. handleEventsForBackgroundURLSession:(NSString *)identifier
  768. completionHandler:(GTMSessionFetcherSystemCompletionHandler)completionHandler;
  769. #endif
  770. // Indicate that a newly created session should be a background session.
  771. // A new session identifier will be created by the fetcher.
  772. //
  773. // Warning: The only thing background sessions are for is rare download
  774. // of huge, batched files of data. And even just for those, there's a lot
  775. // of pain and hackery needed to get transfers to actually happen reliably
  776. // with background sessions.
  777. //
  778. // Don't try to upload or download in many background sessions, since the system
  779. // will impose an exponentially increasing time penalty to prevent the app from
  780. // getting too much background execution time.
  781. //
  782. // References:
  783. //
  784. // "Moving to Fewer, Larger Transfers"
  785. // https://forums.developer.apple.com/thread/14853
  786. //
  787. // "NSURLSession’s Resume Rate Limiter"
  788. // https://forums.developer.apple.com/thread/14854
  789. //
  790. // "Background Session Task state persistence"
  791. // https://forums.developer.apple.com/thread/11554
  792. //
  793. @property(atomic, assign) BOOL useBackgroundSession;
  794. // Indicates if the fetcher was started using a background session.
  795. @property(atomic, readonly, getter=isUsingBackgroundSession) BOOL usingBackgroundSession;
  796. // Indicates if uploads should use an upload task. This is always set for file or stream-provider
  797. // bodies, but may be set explicitly for NSData bodies.
  798. @property(atomic, assign) BOOL useUploadTask;
  799. // Indicates that the fetcher is using a session that may be shared with other fetchers.
  800. @property(atomic, readonly) BOOL canShareSession;
  801. // By default, the fetcher allows only secure (https) schemes unless this
  802. // property is set, or the GTM_ALLOW_INSECURE_REQUESTS build flag is set.
  803. //
  804. // For example, during debugging when fetching from a development server that lacks SSL support,
  805. // this may be set to @[ @"http" ], or when the fetcher is used to retrieve local files,
  806. // this may be set to @[ @"file" ].
  807. //
  808. // This should be left as nil for release builds to avoid creating the opportunity for
  809. // leaking private user behavior and data. If a server is providing insecure URLs
  810. // for fetching by the client app, report the problem as server security & privacy bug.
  811. //
  812. // For builds with the iOS 9/OS X 10.11 and later SDKs, this property is required only when
  813. // the app specifies NSAppTransportSecurity/NSAllowsArbitraryLoads in the main bundle's Info.plist.
  814. @property(atomic, copy, nullable) NSArray<NSString *> *allowedInsecureSchemes;
  815. // By default, the fetcher prohibits localhost requests unless this property is set,
  816. // or the GTM_ALLOW_INSECURE_REQUESTS build flag is set.
  817. //
  818. // For localhost requests, the URL scheme is not checked when this property is set.
  819. //
  820. // For builds with the iOS 9/OS X 10.11 and later SDKs, this property is required only when
  821. // the app specifies NSAppTransportSecurity/NSAllowsArbitraryLoads in the main bundle's Info.plist.
  822. @property(atomic, assign) BOOL allowLocalhostRequest;
  823. // By default, the fetcher requires valid server certs. This may be bypassed
  824. // temporarily for development against a test server with an invalid cert.
  825. @property(atomic, assign) BOOL allowInvalidServerCertificates;
  826. // Cookie storage object for this fetcher. If nil, the fetcher will use a static cookie
  827. // storage instance shared among fetchers. If this fetcher was created by a fetcher service
  828. // object, it will be set to use the service object's cookie storage. See Cookies section above for
  829. // the full discussion.
  830. //
  831. // Because as of Jan 2014 standalone instances of NSHTTPCookieStorage do not actually
  832. // store any cookies (Radar 15735276) we use our own subclass, GTMSessionCookieStorage,
  833. // to hold cookies in memory.
  834. @property(atomic, strong, nullable) NSHTTPCookieStorage *cookieStorage;
  835. // Setting the credential is optional; it is used if the connection receives
  836. // an authentication challenge.
  837. @property(atomic, strong, nullable) NSURLCredential *credential;
  838. // Setting the proxy credential is optional; it is used if the connection
  839. // receives an authentication challenge from a proxy.
  840. @property(atomic, strong, nullable) NSURLCredential *proxyCredential;
  841. // If body data, body file URL, or body stream provider is not set, then a GET request
  842. // method is assumed.
  843. @property(atomic, strong, nullable) NSData *bodyData;
  844. // File to use as the request body. This forces use of an upload task.
  845. @property(atomic, strong, nullable) NSURL *bodyFileURL;
  846. // Length of body to send, expected or actual.
  847. @property(atomic, readonly) int64_t bodyLength;
  848. // The body stream provider may be called repeatedly to provide a body.
  849. // Setting a body stream provider forces use of an upload task.
  850. @property(atomic, copy, nullable) GTMSessionFetcherBodyStreamProvider bodyStreamProvider;
  851. // Object to add authorization to the request, if needed.
  852. //
  853. // This may not be changed once beginFetch has been invoked.
  854. @property(atomic, strong, nullable) id<GTMFetcherAuthorizationProtocol> authorizer;
  855. // The service object that created and monitors this fetcher, if any.
  856. @property(atomic, strong) id<GTMSessionFetcherServiceProtocol> service;
  857. // The host, if any, used to classify this fetcher in the fetcher service.
  858. @property(atomic, copy, nullable) NSString *serviceHost;
  859. // The priority, if any, used for starting fetchers in the fetcher service.
  860. //
  861. // Lower values are higher priority; the default is 0, and values may
  862. // be negative or positive. This priority affects only the start order of
  863. // fetchers that are being delayed by a fetcher service when the running fetchers
  864. // exceeds the service's maxRunningFetchersPerHost. A priority of NSIntegerMin will
  865. // exempt this fetcher from delay.
  866. @property(atomic, assign) NSInteger servicePriority;
  867. // The delegate's optional didReceiveResponse block may be used to inspect or alter
  868. // the session task response.
  869. //
  870. // This is called on the callback queue.
  871. @property(atomic, copy, nullable) GTMSessionFetcherDidReceiveResponseBlock didReceiveResponseBlock;
  872. // The delegate's optional challenge block may be used to inspect or alter
  873. // the session task challenge.
  874. //
  875. // If this block is not set, the fetcher's default behavior for the NSURLSessionTask
  876. // didReceiveChallenge: delegate method is to use the fetcher's respondToChallenge: method
  877. // which relies on the fetcher's credential and proxyCredential properties.
  878. //
  879. // Warning: This may be called repeatedly if the challenge fails. Check
  880. // challenge.previousFailureCount to identify repeated invocations.
  881. //
  882. // This is called on the callback queue.
  883. @property(atomic, copy, nullable) GTMSessionFetcherChallengeBlock challengeBlock;
  884. // The delegate's optional willRedirect block may be used to inspect or alter
  885. // the redirection.
  886. //
  887. // This is called on the callback queue.
  888. @property(atomic, copy, nullable) GTMSessionFetcherWillRedirectBlock willRedirectBlock;
  889. // The optional send progress block reports body bytes uploaded.
  890. //
  891. // This is called on the callback queue.
  892. @property(atomic, copy, nullable) GTMSessionFetcherSendProgressBlock sendProgressBlock;
  893. // The optional accumulate block may be set by clients wishing to accumulate data
  894. // themselves rather than let the fetcher append each buffer to an NSData.
  895. //
  896. // When this is called with nil data (such as on redirect) the client
  897. // should empty its accumulation buffer.
  898. //
  899. // This is called on the callback queue.
  900. @property(atomic, copy, nullable) GTMSessionFetcherAccumulateDataBlock accumulateDataBlock;
  901. // The optional received progress block may be used to monitor data
  902. // received from a data task.
  903. //
  904. // This is called on the callback queue.
  905. @property(atomic, copy, nullable) GTMSessionFetcherReceivedProgressBlock receivedProgressBlock;
  906. // The delegate's optional downloadProgress block may be used to monitor download
  907. // progress in writing to disk.
  908. //
  909. // This is called on the callback queue.
  910. @property(atomic, copy, nullable) GTMSessionFetcherDownloadProgressBlock downloadProgressBlock;
  911. // The delegate's optional willCacheURLResponse block may be used to alter the cached
  912. // NSURLResponse. The user may prevent caching by passing nil to the block's response.
  913. //
  914. // This is called on the callback queue.
  915. @property(atomic, copy, nullable)
  916. GTMSessionFetcherWillCacheURLResponseBlock willCacheURLResponseBlock;
  917. // Enable retrying; see comments at the top of this file. Setting
  918. // retryEnabled=YES resets the min and max retry intervals.
  919. @property(atomic, assign, getter=isRetryEnabled) BOOL retryEnabled;
  920. // Retry block is optional for retries.
  921. //
  922. // If present, this block should call the response block with YES to cause a retry or NO to end the
  923. // fetch.
  924. // See comments at the top of this file.
  925. @property(atomic, copy, nullable) GTMSessionFetcherRetryBlock retryBlock;
  926. // The optional block for collecting the metrics of the present session.
  927. //
  928. // This is called on the callback queue.
  929. @property(atomic, copy, nullable)
  930. GTMSessionFetcherMetricsCollectionBlock metricsCollectionBlock API_AVAILABLE(
  931. ios(10.0), macosx(10.12), tvos(10.0), watchos(6.0));
  932. // Retry intervals must be strictly less than maxRetryInterval, else
  933. // they will be limited to maxRetryInterval and no further retries will
  934. // be attempted. Setting maxRetryInterval to 0.0 will reset it to the
  935. // default value, 60 seconds for downloads and 600 seconds for uploads.
  936. @property(atomic, assign) NSTimeInterval maxRetryInterval;
  937. // Starting retry interval. Setting minRetryInterval to 0.0 will reset it
  938. // to a random value between 1.0 and 2.0 seconds. Clients should normally not
  939. // set this except for unit testing.
  940. @property(atomic, assign) NSTimeInterval minRetryInterval;
  941. // Multiplier used to increase the interval between retries, typically 2.0.
  942. // Clients should not need to set this.
  943. @property(atomic, assign) double retryFactor;
  944. // Number of retries attempted.
  945. @property(atomic, readonly) NSUInteger retryCount;
  946. // Interval delay to precede next retry.
  947. @property(atomic, readonly) NSTimeInterval nextRetryInterval;
  948. #if GTM_BACKGROUND_TASK_FETCHING
  949. // Skip use of a UIBackgroundTask, thus requiring fetches to complete when the app is in the
  950. // foreground.
  951. //
  952. // Targets should define GTM_BACKGROUND_TASK_FETCHING to 0 to avoid use of a UIBackgroundTask
  953. // on iOS to allow fetches to complete in the background. This property is available when
  954. // it's not practical to set the preprocessor define.
  955. @property(atomic, assign) BOOL skipBackgroundTask;
  956. #endif // GTM_BACKGROUND_TASK_FETCHING
  957. // Begin fetching the request
  958. //
  959. // The delegate may optionally implement the callback or pass nil for the selector or handler.
  960. //
  961. // The delegate and all callback blocks are retained between the beginFetch call until after the
  962. // finish callback, or until the fetch is stopped.
  963. //
  964. // An error is passed to the callback for server statuses 300 or
  965. // higher, with the status stored as the error object's code.
  966. //
  967. // finishedSEL has a signature like:
  968. // - (void)fetcher:(GTMSessionFetcher *)fetcher
  969. // finishedWithData:(NSData *)data
  970. // error:(NSError *)error;
  971. //
  972. // If the application has specified a destinationFileURL or an accumulateDataBlock
  973. // for the fetcher, the data parameter passed to the callback will be nil.
  974. - (void)beginFetchWithDelegate:(nullable id)delegate didFinishSelector:(nullable SEL)finishedSEL;
  975. - (void)beginFetchWithCompletionHandler:(nullable GTMSessionFetcherCompletionHandler)handler;
  976. // Returns YES if this fetcher is in the process of fetching a URL.
  977. @property(atomic, readonly, getter=isFetching) BOOL fetching;
  978. // Cancel the fetch of the request that's currently in progress. The completion handler
  979. // will not be called.
  980. - (void)stopFetching;
  981. // A block to be called when the fetch completes.
  982. @property(atomic, copy, nullable) GTMSessionFetcherCompletionHandler completionHandler;
  983. // A block to be called if download resume data becomes available.
  984. @property(atomic, strong, nullable) void (^resumeDataBlock)(NSData *);
  985. // Return the status code from the server response.
  986. @property(atomic, readonly) NSInteger statusCode;
  987. // Return the http headers from the response.
  988. @property(atomic, strong, readonly, nullable) NSDictionary<NSString *, NSString *> *responseHeaders;
  989. // The response, once it's been received.
  990. @property(atomic, strong, readonly, nullable) NSURLResponse *response;
  991. // Bytes downloaded so far.
  992. @property(atomic, readonly) int64_t downloadedLength;
  993. // Buffer of currently-downloaded data, if available.
  994. @property(atomic, readonly, strong, nullable) NSData *downloadedData;
  995. // Local path to which the downloaded file will be moved.
  996. //
  997. // If a file already exists at the path, it will be overwritten.
  998. // Will create the enclosing folders if they are not present.
  999. @property(atomic, strong, nullable) NSURL *destinationFileURL;
  1000. // The time this fetcher originally began fetching. This is useful as a time
  1001. // barrier for ignoring irrelevant fetch notifications or callbacks.
  1002. @property(atomic, strong, readonly, nullable) NSDate *initialBeginFetchDate;
  1003. // userData is retained solely for the convenience of the client.
  1004. @property(atomic, strong, nullable) id userData;
  1005. // Stored property values are retained solely for the convenience of the client.
  1006. @property(atomic, copy, nullable) NSDictionary<NSString *, id> *properties;
  1007. - (void)setProperty:(nullable id)obj
  1008. forKey:(NSString *)key; // Pass nil for obj to remove the property.
  1009. - (nullable id)propertyForKey:(NSString *)key;
  1010. - (void)addPropertiesFromDictionary:(NSDictionary<NSString *, id> *)dict;
  1011. // Comments are useful for logging, so are strongly recommended for each fetcher.
  1012. @property(atomic, copy, nullable) NSString *comment;
  1013. - (void)setCommentWithFormat:(NSString *)format, ... NS_FORMAT_FUNCTION(1, 2);
  1014. // Log of request and response, if logging is enabled
  1015. @property(atomic, copy, nullable) NSString *log;
  1016. // Callbacks are run on this queue. If none is supplied, the main queue is used.
  1017. //
  1018. // CAUTION: This block MUST be a serial queue. Setting a concurrent queue can result in callbacks
  1019. // being dispatched concurrently, leading events to appear out-of-order.
  1020. @property(atomic, strong, null_resettable) dispatch_queue_t callbackQueue;
  1021. // The queue used internally by the session to invoke its delegate methods in the fetcher.
  1022. //
  1023. // Application callbacks are always called by the fetcher on the callbackQueue above,
  1024. // not on this queue. Apps should generally not change this queue.
  1025. //
  1026. // The default delegate queue is the main queue.
  1027. //
  1028. // This value is ignored after the session has been created, so this
  1029. // property should be set in the fetcher service rather in the fetcher as it applies
  1030. // to a shared session.
  1031. @property(atomic, strong, null_resettable) NSOperationQueue *sessionDelegateQueue;
  1032. // DEPRECATED: Callers should use XCTestExpectation instead.
  1033. //
  1034. // Spin the run loop or sleep the thread, discarding events, until the fetch has completed.
  1035. //
  1036. // This is only for use in testing or in tools without a user interface.
  1037. //
  1038. // Note: Synchronous fetches should never be used by shipping apps; they are
  1039. // sufficient reason for rejection from the app store.
  1040. //
  1041. // Returns NO if timed out.
  1042. - (BOOL)waitForCompletionWithTimeout:(NSTimeInterval)timeoutInSeconds
  1043. __deprecated_msg("Use XCTestExpectation instead");
  1044. // Test block is optional for testing.
  1045. //
  1046. // If present, this block will cause the fetcher to skip starting the session, and instead
  1047. // use the test block response values when calling the completion handler and delegate code.
  1048. //
  1049. // Test code can set this on the fetcher or on the fetcher service. For testing libraries
  1050. // that use a fetcher without exposing either the fetcher or the fetcher service, the global
  1051. // method setGlobalTestBlock: will set the block for all fetchers that do not have a test
  1052. // block set.
  1053. //
  1054. // The test code can pass nil for all response parameters to indicate that the fetch
  1055. // should proceed.
  1056. //
  1057. // Applications can exclude test block support by setting GTM_DISABLE_FETCHER_TEST_BLOCK.
  1058. @property(atomic, copy, nullable) GTMSessionFetcherTestBlock testBlock;
  1059. + (void)setGlobalTestBlock:(nullable GTMSessionFetcherTestBlock)block;
  1060. // When using the testBlock, |testBlockAccumulateDataChunkCount| is the desired number of chunks to
  1061. // divide the response data into if the client has streaming enabled. The data will be divided up to
  1062. // |testBlockAccumulateDataChunkCount| chunks; however, the exact amount may vary depending on the
  1063. // size of the response data (e.g. a 1-byte response can only be divided into one chunk).
  1064. @property(atomic, readwrite) NSUInteger testBlockAccumulateDataChunkCount;
  1065. #if GTM_BACKGROUND_TASK_FETCHING
  1066. // For testing or to override UIApplication invocations, apps may specify an alternative
  1067. // target for messages to UIApplication.
  1068. + (void)setSubstituteUIApplication:(nullable id<GTMUIApplicationProtocol>)substituteUIApplication;
  1069. + (nullable id<GTMUIApplicationProtocol>)substituteUIApplication;
  1070. #endif // GTM_BACKGROUND_TASK_FETCHING
  1071. // Exposed for testing.
  1072. + (GTMSessionCookieStorage *)staticCookieStorage;
  1073. + (BOOL)appAllowsInsecureRequests;
  1074. #if STRIP_GTM_FETCH_LOGGING
  1075. // If logging is stripped, provide a stub for the main method
  1076. // for controlling logging.
  1077. + (void)setLoggingEnabled:(BOOL)flag;
  1078. + (BOOL)isLoggingEnabled;
  1079. #else
  1080. // These methods let an application log specific body text, such as the text description of a binary
  1081. // request or response. The application should set the fetcher to defer response body logging until
  1082. // the response has been received and the log response body has been set by the app. For example:
  1083. //
  1084. // fetcher.logRequestBody = [binaryObject stringDescription];
  1085. // fetcher.deferResponseBodyLogging = YES;
  1086. // [fetcher beginFetchWithCompletionHandler:^(NSData *data, NSError *error) {
  1087. // if (error == nil) {
  1088. // fetcher.logResponseBody = [[[MyThing alloc] initWithData:data] stringDescription];
  1089. // }
  1090. // fetcher.deferResponseBodyLogging = NO;
  1091. // }];
  1092. @property(atomic, copy, nullable) NSString *logRequestBody;
  1093. @property(atomic, assign) BOOL deferResponseBodyLogging;
  1094. @property(atomic, copy, nullable) NSString *logResponseBody;
  1095. // Internal logging support.
  1096. @property(atomic, readonly) NSData *loggedStreamData;
  1097. @property(atomic, assign) BOOL hasLoggedError;
  1098. @property(atomic, strong, nullable) NSURL *redirectedFromURL;
  1099. - (void)appendLoggedStreamData:(NSData *)dataToAdd;
  1100. - (void)clearLoggedStreamData;
  1101. #endif // STRIP_GTM_FETCH_LOGGING
  1102. @end
  1103. @interface GTMSessionFetcher (BackwardsCompatibilityOnly)
  1104. // Clients using GTMSessionFetcher should set the cookie storage explicitly themselves;
  1105. // this method is deprecated and will be removed soon.
  1106. - (void)setCookieStorageMethod:(NSInteger)method
  1107. __deprecated_msg("Create an NSHTTPCookieStorage and set .cookieStorage directly.");
  1108. @end
  1109. // Until we can just instantiate NSHTTPCookieStorage for local use, we'll
  1110. // implement all the public methods ourselves. This stores cookies only in
  1111. // memory. Additional methods are provided for testing.
  1112. //
  1113. // iOS 9/OS X 10.11 added +[NSHTTPCookieStorage sharedCookieStorageForGroupContainerIdentifier:]
  1114. // which may also be used to create cookie storage.
  1115. @interface GTMSessionCookieStorage : NSHTTPCookieStorage
  1116. // Add the array off cookies to the storage, replacing duplicates.
  1117. // Also removes expired cookies from the storage.
  1118. - (void)setCookies:(nullable NSArray<NSHTTPCookie *> *)cookies;
  1119. - (void)removeAllCookies;
  1120. @end
  1121. // Macros to monitor synchronization blocks in debug builds.
  1122. // These report problems using GTMSessionCheckDebug.
  1123. //
  1124. // GTMSessionMonitorSynchronized Start monitoring a top-level-only
  1125. // @sync scope.
  1126. // GTMSessionMonitorRecursiveSynchronized Start monitoring a top-level or
  1127. // recursive @sync scope.
  1128. // GTMSessionCheckSynchronized Verify that the current execution
  1129. // is inside a @sync scope.
  1130. // GTMSessionCheckNotSynchronized Verify that the current execution
  1131. // is not inside a @sync scope.
  1132. //
  1133. // Example usage:
  1134. //
  1135. // - (void)myExternalMethod {
  1136. // @synchronized(self) {
  1137. // GTMSessionMonitorSynchronized(self)
  1138. //
  1139. // - (void)myInternalMethod {
  1140. // GTMSessionCheckSynchronized(self);
  1141. //
  1142. // - (void)callMyCallbacks {
  1143. // GTMSessionCheckNotSynchronized(self);
  1144. //
  1145. // GTMSessionCheckNotSynchronized is available for verifying the code isn't
  1146. // in a deadlockable @sync state when posting notifications and invoking
  1147. // callbacks. Don't use GTMSessionCheckNotSynchronized immediately before a
  1148. // @sync scope; the normal recursiveness check of GTMSessionMonitorSynchronized
  1149. // can catch those.
  1150. #ifdef __OBJC__
  1151. // If asserts are entirely no-ops, the synchronization monitor is just a bunch
  1152. // of counting code that doesn't report exceptional circumstances in any way.
  1153. // Only build the synchronization monitor code if NS_BLOCK_ASSERTIONS is not
  1154. // defined or asserts are being logged instead.
  1155. #if DEBUG && (!defined(NS_BLOCK_ASSERTIONS) || GTMSESSION_ASSERT_AS_LOG)
  1156. #define __GTMSessionMonitorSynchronizedVariableInner(varname, counter) varname##counter
  1157. #define __GTMSessionMonitorSynchronizedVariable(varname, counter) \
  1158. __GTMSessionMonitorSynchronizedVariableInner(varname, counter)
  1159. #define GTMSessionMonitorSynchronized(obj) \
  1160. NS_VALID_UNTIL_END_OF_SCOPE id __GTMSessionMonitorSynchronizedVariable(__monitor, __COUNTER__) = \
  1161. [[GTMSessionSyncMonitorInternal alloc] initWithSynchronizationObject:obj \
  1162. allowRecursive:NO \
  1163. functionName:__func__]
  1164. #define GTMSessionMonitorRecursiveSynchronized(obj) \
  1165. NS_VALID_UNTIL_END_OF_SCOPE id __GTMSessionMonitorSynchronizedVariable(__monitor, __COUNTER__) = \
  1166. [[GTMSessionSyncMonitorInternal alloc] initWithSynchronizationObject:obj \
  1167. allowRecursive:YES \
  1168. functionName:__func__]
  1169. #define GTMSessionCheckSynchronized(obj) \
  1170. { \
  1171. GTMSESSION_ASSERT_DEBUG( \
  1172. [GTMSessionSyncMonitorInternal functionsHoldingSynchronizationOnObject:obj], \
  1173. @"GTMSessionCheckSynchronized(" #obj ") failed: not sync'd" \
  1174. @" on " #obj " in %s. Call stack:\n%@", \
  1175. __func__, [NSThread callStackSymbols]); \
  1176. }
  1177. #define GTMSessionCheckNotSynchronized(obj) \
  1178. { \
  1179. GTMSESSION_ASSERT_DEBUG( \
  1180. ![GTMSessionSyncMonitorInternal functionsHoldingSynchronizationOnObject:obj], \
  1181. @"GTMSessionCheckNotSynchronized(" #obj ") failed: was sync'd" \
  1182. @" on " #obj " in %s by %@. Call stack:\n%@", \
  1183. __func__, [GTMSessionSyncMonitorInternal functionsHoldingSynchronizationOnObject:obj], \
  1184. [NSThread callStackSymbols]); \
  1185. }
  1186. // GTMSessionSyncMonitorInternal is a private class that keeps track of the
  1187. // beginning and end of synchronized scopes.
  1188. //
  1189. // This class should not be used directly, but only via the
  1190. // GTMSessionMonitorSynchronized macro.
  1191. @interface GTMSessionSyncMonitorInternal : NSObject
  1192. - (instancetype)initWithSynchronizationObject:(id)object
  1193. allowRecursive:(BOOL)allowRecursive
  1194. functionName:(const char *)functionName;
  1195. // Return the names of the functions that hold sync on the object, or nil if none.
  1196. + (nullable NSArray *)functionsHoldingSynchronizationOnObject:(id)object;
  1197. @end
  1198. #else
  1199. #define GTMSessionMonitorSynchronized(obj) \
  1200. do { \
  1201. } while (0)
  1202. #define GTMSessionMonitorRecursiveSynchronized(obj) \
  1203. do { \
  1204. } while (0)
  1205. #define GTMSessionCheckSynchronized(obj) \
  1206. do { \
  1207. } while (0)
  1208. #define GTMSessionCheckNotSynchronized(obj) \
  1209. do { \
  1210. } while (0)
  1211. #endif // !DEBUG
  1212. #endif // __OBJC__
  1213. NS_ASSUME_NONNULL_END