Session.swift 68 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232
  1. //
  2. // Session.swift
  3. //
  4. // Copyright (c) 2014-2018 Alamofire Software Foundation (http://alamofire.org/)
  5. //
  6. // Permission is hereby granted, free of charge, to any person obtaining a copy
  7. // of this software and associated documentation files (the "Software"), to deal
  8. // in the Software without restriction, including without limitation the rights
  9. // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  10. // copies of the Software, and to permit persons to whom the Software is
  11. // furnished to do so, subject to the following conditions:
  12. //
  13. // The above copyright notice and this permission notice shall be included in
  14. // all copies or substantial portions of the Software.
  15. //
  16. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  17. // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  18. // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  19. // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  20. // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  21. // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
  22. // THE SOFTWARE.
  23. //
  24. import Foundation
  25. /// `Session` creates and manages Alamofire's `Request` types during their lifetimes. It also provides common
  26. /// functionality for all `Request`s, including queuing, interception, trust management, redirect handling, and response
  27. /// cache handling.
  28. open class Session {
  29. /// Shared singleton instance used by all `AF.request` APIs. Cannot be modified.
  30. public static let `default` = Session()
  31. /// Underlying `URLSession` used to create `URLSessionTasks` for this instance, and for which this instance's
  32. /// `delegate` handles `URLSessionDelegate` callbacks.
  33. public let session: URLSession
  34. /// Instance's `SessionDelegate`, which handles the `URLSessionDelegate` methods and `Request` interaction.
  35. public let delegate: SessionDelegate
  36. /// Root `DispatchQueue` for all internal callbacks and state update. **MUST** be a serial queue.
  37. public let rootQueue: DispatchQueue
  38. /// Value determining whether this instance automatically calls `resume()` on all created `Request`s.
  39. public let startRequestsImmediately: Bool
  40. /// `DispatchQueue` on which `URLRequest`s are created asynchronously. By default this queue uses `rootQueue` as its
  41. /// `target`, but a separate queue can be used if request creation is determined to be a bottleneck. Always profile
  42. /// and test before introducing an additional queue.
  43. public let requestQueue: DispatchQueue
  44. /// `DispatchQueue` passed to all `Request`s on which they perform their response serialization. By default this
  45. /// queue uses `rootQueue` as its `target` but a separate queue can be used if response serialization is determined
  46. /// to be a bottleneck. Always profile and test before introducing an additional queue.
  47. public let serializationQueue: DispatchQueue
  48. /// `RequestInterceptor` used for all `Request` created by the instance. `RequestInterceptor`s can also be set on a
  49. /// per-`Request` basis, in which case the `Request`'s interceptor takes precedence over this value.
  50. public let interceptor: RequestInterceptor?
  51. /// `ServerTrustManager` instance used to evaluate all trust challenges and provide certificate and key pinning.
  52. public let serverTrustManager: ServerTrustManager?
  53. /// `RedirectHandler` instance used to provide customization for request redirection.
  54. public let redirectHandler: RedirectHandler?
  55. /// `CachedResponseHandler` instance used to provide customization of cached response handling.
  56. public let cachedResponseHandler: CachedResponseHandler?
  57. /// `CompositeEventMonitor` used to compose Alamofire's `defaultEventMonitors` and any passed `EventMonitor`s.
  58. public let eventMonitor: CompositeEventMonitor
  59. /// `EventMonitor`s included in all instances. `[AlamofireNotifications()]` by default.
  60. public let defaultEventMonitors: [EventMonitor] = [AlamofireNotifications()]
  61. /// Internal map between `Request`s and any `URLSessionTasks` that may be in flight for them.
  62. var requestTaskMap = RequestTaskMap()
  63. /// `Set` of currently active `Request`s.
  64. var activeRequests: Set<Request> = []
  65. /// Completion events awaiting `URLSessionTaskMetrics`.
  66. var waitingCompletions: [URLSessionTask: () -> Void] = [:]
  67. /// Creates a `Session` from a `URLSession` and other parameters.
  68. ///
  69. /// - Note: When passing a `URLSession`, you must create the `URLSession` with a specific `delegateQueue` value and
  70. /// pass the `delegateQueue`'s `underlyingQueue` as the `rootQueue` parameter of this initializer.
  71. ///
  72. /// - Parameters:
  73. /// - session: Underlying `URLSession` for this instance.
  74. /// - delegate: `SessionDelegate` that handles `session`'s delegate callbacks as well as `Request`
  75. /// interaction.
  76. /// - rootQueue: Root `DispatchQueue` for all internal callbacks and state updates. **MUST** be a
  77. /// serial queue.
  78. /// - startRequestsImmediately: Determines whether this instance will automatically start all `Request`s. `true`
  79. /// by default. If set to `false`, all `Request`s created must have `.resume()` called.
  80. /// on them for them to start.
  81. /// - requestQueue: `DispatchQueue` on which to perform `URLRequest` creation. By default this queue
  82. /// will use the `rootQueue` as its `target`. A separate queue can be used if it's
  83. /// determined request creation is a bottleneck, but that should only be done after
  84. /// careful testing and profiling. `nil` by default.
  85. /// - serializationQueue: `DispatchQueue` on which to perform all response serialization. By default this
  86. /// queue will use the `rootQueue` as its `target`. A separate queue can be used if
  87. /// it's determined response serialization is a bottleneck, but that should only be
  88. /// done after careful testing and profiling. `nil` by default.
  89. /// - interceptor: `RequestInterceptor` to be used for all `Request`s created by this instance. `nil`
  90. /// by default.
  91. /// - serverTrustManager: `ServerTrustManager` to be used for all trust evaluations by this instance. `nil`
  92. /// by default.
  93. /// - redirectHandler: `RedirectHandler` to be used by all `Request`s created by this instance. `nil` by
  94. /// default.
  95. /// - cachedResponseHandler: `CachedResponseHandler` to be used by all `Request`s created by this instance.
  96. /// `nil` by default.
  97. /// - eventMonitors: Additional `EventMonitor`s used by the instance. Alamofire always adds a
  98. /// `AlamofireNotifications` `EventMonitor` to the array passed here. `[]` by default.
  99. public init(session: URLSession,
  100. delegate: SessionDelegate,
  101. rootQueue: DispatchQueue,
  102. startRequestsImmediately: Bool = true,
  103. requestQueue: DispatchQueue? = nil,
  104. serializationQueue: DispatchQueue? = nil,
  105. interceptor: RequestInterceptor? = nil,
  106. serverTrustManager: ServerTrustManager? = nil,
  107. redirectHandler: RedirectHandler? = nil,
  108. cachedResponseHandler: CachedResponseHandler? = nil,
  109. eventMonitors: [EventMonitor] = []) {
  110. precondition(session.configuration.identifier == nil,
  111. "Alamofire does not support background URLSessionConfigurations.")
  112. precondition(session.delegateQueue.underlyingQueue === rootQueue,
  113. "Session(session:) initializer must be passed the DispatchQueue used as the delegateQueue's underlyingQueue as rootQueue.")
  114. self.session = session
  115. self.delegate = delegate
  116. self.rootQueue = rootQueue
  117. self.startRequestsImmediately = startRequestsImmediately
  118. self.requestQueue = requestQueue ?? DispatchQueue(label: "\(rootQueue.label).requestQueue", target: rootQueue)
  119. self.serializationQueue = serializationQueue ?? DispatchQueue(label: "\(rootQueue.label).serializationQueue", target: rootQueue)
  120. self.interceptor = interceptor
  121. self.serverTrustManager = serverTrustManager
  122. self.redirectHandler = redirectHandler
  123. self.cachedResponseHandler = cachedResponseHandler
  124. eventMonitor = CompositeEventMonitor(monitors: defaultEventMonitors + eventMonitors)
  125. delegate.eventMonitor = eventMonitor
  126. delegate.stateProvider = self
  127. }
  128. /// Creates a `Session` from a `URLSessionConfiguration`.
  129. ///
  130. /// - Note: This initializer lets Alamofire handle the creation of the underlying `URLSession` and its
  131. /// `delegateQueue`, and is the recommended initializer for most uses.
  132. ///
  133. /// - Parameters:
  134. /// - configuration: `URLSessionConfiguration` to be used to create the underlying `URLSession`. Changes
  135. /// to this value after being passed to this initializer will have no effect.
  136. /// `URLSessionConfiguration.af.default` by default.
  137. /// - delegate: `SessionDelegate` that handles `session`'s delegate callbacks as well as `Request`
  138. /// interaction. `SessionDelegate()` by default.
  139. /// - rootQueue: Root `DispatchQueue` for all internal callbacks and state updates. **MUST** be a
  140. /// serial queue. `DispatchQueue(label: "org.alamofire.session.rootQueue")` by default.
  141. /// - startRequestsImmediately: Determines whether this instance will automatically start all `Request`s. `true`
  142. /// by default. If set to `false`, all `Request`s created must have `.resume()` called.
  143. /// on them for them to start.
  144. /// - requestQueue: `DispatchQueue` on which to perform `URLRequest` creation. By default this queue
  145. /// will use the `rootQueue` as its `target`. A separate queue can be used if it's
  146. /// determined request creation is a bottleneck, but that should only be done after
  147. /// careful testing and profiling. `nil` by default.
  148. /// - serializationQueue: `DispatchQueue` on which to perform all response serialization. By default this
  149. /// queue will use the `rootQueue` as its `target`. A separate queue can be used if
  150. /// it's determined response serialization is a bottleneck, but that should only be
  151. /// done after careful testing and profiling. `nil` by default.
  152. /// - interceptor: `RequestInterceptor` to be used for all `Request`s created by this instance. `nil`
  153. /// by default.
  154. /// - serverTrustManager: `ServerTrustManager` to be used for all trust evaluations by this instance. `nil`
  155. /// by default.
  156. /// - redirectHandler: `RedirectHandler` to be used by all `Request`s created by this instance. `nil` by
  157. /// default.
  158. /// - cachedResponseHandler: `CachedResponseHandler` to be used by all `Request`s created by this instance.
  159. /// `nil` by default.
  160. /// - eventMonitors: Additional `EventMonitor`s used by the instance. Alamofire always adds a
  161. /// `AlamofireNotifications` `EventMonitor` to the array passed here. `[]` by default.
  162. public convenience init(configuration: URLSessionConfiguration = URLSessionConfiguration.af.default,
  163. delegate: SessionDelegate = SessionDelegate(),
  164. rootQueue: DispatchQueue = DispatchQueue(label: "org.alamofire.session.rootQueue"),
  165. startRequestsImmediately: Bool = true,
  166. requestQueue: DispatchQueue? = nil,
  167. serializationQueue: DispatchQueue? = nil,
  168. interceptor: RequestInterceptor? = nil,
  169. serverTrustManager: ServerTrustManager? = nil,
  170. redirectHandler: RedirectHandler? = nil,
  171. cachedResponseHandler: CachedResponseHandler? = nil,
  172. eventMonitors: [EventMonitor] = []) {
  173. precondition(configuration.identifier == nil, "Alamofire does not support background URLSessionConfigurations.")
  174. let delegateQueue = OperationQueue(maxConcurrentOperationCount: 1, underlyingQueue: rootQueue, name: "org.alamofire.session.sessionDelegateQueue")
  175. let session = URLSession(configuration: configuration, delegate: delegate, delegateQueue: delegateQueue)
  176. self.init(session: session,
  177. delegate: delegate,
  178. rootQueue: rootQueue,
  179. startRequestsImmediately: startRequestsImmediately,
  180. requestQueue: requestQueue,
  181. serializationQueue: serializationQueue,
  182. interceptor: interceptor,
  183. serverTrustManager: serverTrustManager,
  184. redirectHandler: redirectHandler,
  185. cachedResponseHandler: cachedResponseHandler,
  186. eventMonitors: eventMonitors)
  187. }
  188. deinit {
  189. finishRequestsForDeinit()
  190. session.invalidateAndCancel()
  191. }
  192. // MARK: - Cancellation
  193. /// Cancel all active `Request`s, optionally calling a completion handler when complete.
  194. ///
  195. /// - Note: This is an asynchronous operation and does not block the creation of future `Request`s. Cancelled
  196. /// `Request`s may not cancel immediately due internal work, and may not cancel at all if they are close to
  197. /// completion when cancelled.
  198. ///
  199. /// - Parameters:
  200. /// - queue: `DispatchQueue` on which the completion handler is run. `.main` by default.
  201. /// - completion: Closure to be called when all `Request`s have been cancelled.
  202. public func cancelAllRequests(completingOnQueue queue: DispatchQueue = .main, completion: (() -> Void)? = nil) {
  203. rootQueue.async {
  204. self.activeRequests.forEach { $0.cancel() }
  205. queue.async { completion?() }
  206. }
  207. }
  208. // MARK: - DataRequest
  209. /// Closure which provides a `URLRequest` for mutation.
  210. public typealias RequestModifier = (inout URLRequest) throws -> Void
  211. struct RequestConvertible: URLRequestConvertible {
  212. let url: URLConvertible
  213. let method: HTTPMethod
  214. let parameters: Parameters?
  215. let encoding: ParameterEncoding
  216. let headers: HTTPHeaders?
  217. let requestModifier: RequestModifier?
  218. func asURLRequest() throws -> URLRequest {
  219. var request = try URLRequest(url: url, method: method, headers: headers)
  220. try requestModifier?(&request)
  221. return try encoding.encode(request, with: parameters)
  222. }
  223. }
  224. /// Creates a `DataRequest` from a `URLRequest` created using the passed components and a `RequestInterceptor`.
  225. ///
  226. /// - Parameters:
  227. /// - convertible: `URLConvertible` value to be used as the `URLRequest`'s `URL`.
  228. /// - method: `HTTPMethod` for the `URLRequest`. `.get` by default.
  229. /// - parameters: `Parameters` (a.k.a. `[String: Any]`) value to be encoded into the `URLRequest`. `nil` by
  230. /// default.
  231. /// - encoding: `ParameterEncoding` to be used to encode the `parameters` value into the `URLRequest`.
  232. /// `URLEncoding.default` by default.
  233. /// - headers: `HTTPHeaders` value to be added to the `URLRequest`. `nil` by default.
  234. /// - interceptor: `RequestInterceptor` value to be used by the returned `DataRequest`. `nil` by default.
  235. /// - requestModifier: `RequestModifier` which will be applied to the `URLRequest` created from the provided
  236. /// parameters. `nil` by default.
  237. ///
  238. /// - Returns: The created `DataRequest`.
  239. open func request(_ convertible: URLConvertible,
  240. method: HTTPMethod = .get,
  241. parameters: Parameters? = nil,
  242. encoding: ParameterEncoding = URLEncoding.default,
  243. headers: HTTPHeaders? = nil,
  244. interceptor: RequestInterceptor? = nil,
  245. requestModifier: RequestModifier? = nil) -> DataRequest {
  246. let convertible = RequestConvertible(url: convertible,
  247. method: method,
  248. parameters: parameters,
  249. encoding: encoding,
  250. headers: headers,
  251. requestModifier: requestModifier)
  252. return request(convertible, interceptor: interceptor)
  253. }
  254. struct RequestEncodableConvertible<Parameters: Encodable>: URLRequestConvertible {
  255. let url: URLConvertible
  256. let method: HTTPMethod
  257. let parameters: Parameters?
  258. let encoder: ParameterEncoder
  259. let headers: HTTPHeaders?
  260. let requestModifier: RequestModifier?
  261. func asURLRequest() throws -> URLRequest {
  262. var request = try URLRequest(url: url, method: method, headers: headers)
  263. try requestModifier?(&request)
  264. return try parameters.map { try encoder.encode($0, into: request) } ?? request
  265. }
  266. }
  267. /// Creates a `DataRequest` from a `URLRequest` created using the passed components, `Encodable` parameters, and a
  268. /// `RequestInterceptor`.
  269. ///
  270. /// - Parameters:
  271. /// - convertible: `URLConvertible` value to be used as the `URLRequest`'s `URL`.
  272. /// - method: `HTTPMethod` for the `URLRequest`. `.get` by default.
  273. /// - parameters: `Encodable` value to be encoded into the `URLRequest`. `nil` by default.
  274. /// - encoder: `ParameterEncoder` to be used to encode the `parameters` value into the `URLRequest`.
  275. /// `URLEncodedFormParameterEncoder.default` by default.
  276. /// - headers: `HTTPHeaders` value to be added to the `URLRequest`. `nil` by default.
  277. /// - interceptor: `RequestInterceptor` value to be used by the returned `DataRequest`. `nil` by default.
  278. ///
  279. /// - Returns: The created `DataRequest`.
  280. open func request<Parameters: Encodable>(_ convertible: URLConvertible,
  281. method: HTTPMethod = .get,
  282. parameters: Parameters? = nil,
  283. encoder: ParameterEncoder = URLEncodedFormParameterEncoder.default,
  284. headers: HTTPHeaders? = nil,
  285. interceptor: RequestInterceptor? = nil,
  286. requestModifier: RequestModifier? = nil) -> DataRequest {
  287. let convertible = RequestEncodableConvertible(url: convertible,
  288. method: method,
  289. parameters: parameters,
  290. encoder: encoder,
  291. headers: headers,
  292. requestModifier: requestModifier)
  293. return request(convertible, interceptor: interceptor)
  294. }
  295. /// Creates a `DataRequest` from a `URLRequestConvertible` value and a `RequestInterceptor`.
  296. ///
  297. /// - Parameters:
  298. /// - convertible: `URLRequestConvertible` value to be used to create the `URLRequest`.
  299. /// - interceptor: `RequestInterceptor` value to be used by the returned `DataRequest`. `nil` by default.
  300. ///
  301. /// - Returns: The created `DataRequest`.
  302. open func request(_ convertible: URLRequestConvertible, interceptor: RequestInterceptor? = nil) -> DataRequest {
  303. let request = DataRequest(convertible: convertible,
  304. underlyingQueue: rootQueue,
  305. serializationQueue: serializationQueue,
  306. eventMonitor: eventMonitor,
  307. interceptor: interceptor,
  308. delegate: self)
  309. perform(request)
  310. return request
  311. }
  312. // MARK: - DataStreamRequest
  313. /// Creates a `DataStreamRequest` from the passed components, `Encodable` parameters, and `RequestInterceptor`.
  314. ///
  315. /// - Parameters:
  316. /// - convertible: `URLConvertible` value to be used as the `URLRequest`'s `URL`.
  317. /// - method: `HTTPMethod` for the `URLRequest`. `.get` by default.
  318. /// - parameters: `Encodable` value to be encoded into the `URLRequest`. `nil` by default.
  319. /// - encoder: `ParameterEncoder` to be used to encode the `parameters` value into the
  320. /// `URLRequest`.
  321. /// `URLEncodedFormParameterEncoder.default` by default.
  322. /// - headers: `HTTPHeaders` value to be added to the `URLRequest`. `nil` by default.
  323. /// - automaticallyCancelOnStreamError: `Bool` indicating whether the instance should be canceled when an `Error`
  324. /// is thrown while serializing stream `Data`. `false` by default.
  325. /// - interceptor: `RequestInterceptor` value to be used by the returned `DataRequest`. `nil`
  326. /// by default.
  327. /// - requestModifier: `RequestModifier` which will be applied to the `URLRequest` created from
  328. /// the provided parameters. `nil` by default.
  329. ///
  330. /// - Returns: The created `DataStream` request.
  331. open func streamRequest<Parameters: Encodable>(_ convertible: URLConvertible,
  332. method: HTTPMethod = .get,
  333. parameters: Parameters? = nil,
  334. encoder: ParameterEncoder = URLEncodedFormParameterEncoder.default,
  335. headers: HTTPHeaders? = nil,
  336. automaticallyCancelOnStreamError: Bool = false,
  337. interceptor: RequestInterceptor? = nil,
  338. requestModifier: RequestModifier? = nil) -> DataStreamRequest {
  339. let convertible = RequestEncodableConvertible(url: convertible,
  340. method: method,
  341. parameters: parameters,
  342. encoder: encoder,
  343. headers: headers,
  344. requestModifier: requestModifier)
  345. return streamRequest(convertible,
  346. automaticallyCancelOnStreamError: automaticallyCancelOnStreamError,
  347. interceptor: interceptor)
  348. }
  349. /// Creates a `DataStreamRequest` from the passed components and `RequestInterceptor`.
  350. ///
  351. /// - Parameters:
  352. /// - convertible: `URLConvertible` value to be used as the `URLRequest`'s `URL`.
  353. /// - method: `HTTPMethod` for the `URLRequest`. `.get` by default.
  354. /// - headers: `HTTPHeaders` value to be added to the `URLRequest`. `nil` by default.
  355. /// - automaticallyCancelOnStreamError: `Bool` indicating whether the instance should be canceled when an `Error`
  356. /// is thrown while serializing stream `Data`. `false` by default.
  357. /// - interceptor: `RequestInterceptor` value to be used by the returned `DataRequest`. `nil`
  358. /// by default.
  359. /// - requestModifier: `RequestModifier` which will be applied to the `URLRequest` created from
  360. /// the provided parameters. `nil` by default.
  361. ///
  362. /// - Returns: The created `DataStream` request.
  363. open func streamRequest(_ convertible: URLConvertible,
  364. method: HTTPMethod = .get,
  365. headers: HTTPHeaders? = nil,
  366. automaticallyCancelOnStreamError: Bool = false,
  367. interceptor: RequestInterceptor? = nil,
  368. requestModifier: RequestModifier? = nil) -> DataStreamRequest {
  369. let convertible = RequestEncodableConvertible(url: convertible,
  370. method: method,
  371. parameters: Optional<Empty>.none,
  372. encoder: URLEncodedFormParameterEncoder.default,
  373. headers: headers,
  374. requestModifier: requestModifier)
  375. return streamRequest(convertible,
  376. automaticallyCancelOnStreamError: automaticallyCancelOnStreamError,
  377. interceptor: interceptor)
  378. }
  379. /// Creates a `DataStreamRequest` from the passed `URLRequestConvertible` value and `RequestInterceptor`.
  380. ///
  381. /// - Parameters:
  382. /// - convertible: `URLRequestConvertible` value to be used to create the `URLRequest`.
  383. /// - automaticallyCancelOnStreamError: `Bool` indicating whether the instance should be canceled when an `Error`
  384. /// is thrown while serializing stream `Data`. `false` by default.
  385. /// - interceptor: `RequestInterceptor` value to be used by the returned `DataRequest`. `nil`
  386. /// by default.
  387. ///
  388. /// - Returns: The created `DataStreamRequest`.
  389. open func streamRequest(_ convertible: URLRequestConvertible,
  390. automaticallyCancelOnStreamError: Bool = false,
  391. interceptor: RequestInterceptor? = nil) -> DataStreamRequest {
  392. let request = DataStreamRequest(convertible: convertible,
  393. automaticallyCancelOnStreamError: automaticallyCancelOnStreamError,
  394. underlyingQueue: rootQueue,
  395. serializationQueue: serializationQueue,
  396. eventMonitor: eventMonitor,
  397. interceptor: interceptor,
  398. delegate: self)
  399. perform(request)
  400. return request
  401. }
  402. // MARK: - DownloadRequest
  403. /// Creates a `DownloadRequest` using a `URLRequest` created using the passed components, `RequestInterceptor`, and
  404. /// `Destination`.
  405. ///
  406. /// - Parameters:
  407. /// - convertible: `URLConvertible` value to be used as the `URLRequest`'s `URL`.
  408. /// - method: `HTTPMethod` for the `URLRequest`. `.get` by default.
  409. /// - parameters: `Parameters` (a.k.a. `[String: Any]`) value to be encoded into the `URLRequest`. `nil` by
  410. /// default.
  411. /// - encoding: `ParameterEncoding` to be used to encode the `parameters` value into the `URLRequest`.
  412. /// Defaults to `URLEncoding.default`.
  413. /// - headers: `HTTPHeaders` value to be added to the `URLRequest`. `nil` by default.
  414. /// - interceptor: `RequestInterceptor` value to be used by the returned `DataRequest`. `nil` by default.
  415. /// - requestModifier: `RequestModifier` which will be applied to the `URLRequest` created from the provided
  416. /// parameters. `nil` by default.
  417. /// - destination: `DownloadRequest.Destination` closure used to determine how and where the downloaded file
  418. /// should be moved. `nil` by default.
  419. ///
  420. /// - Returns: The created `DownloadRequest`.
  421. open func download(_ convertible: URLConvertible,
  422. method: HTTPMethod = .get,
  423. parameters: Parameters? = nil,
  424. encoding: ParameterEncoding = URLEncoding.default,
  425. headers: HTTPHeaders? = nil,
  426. interceptor: RequestInterceptor? = nil,
  427. requestModifier: RequestModifier? = nil,
  428. to destination: DownloadRequest.Destination? = nil) -> DownloadRequest {
  429. let convertible = RequestConvertible(url: convertible,
  430. method: method,
  431. parameters: parameters,
  432. encoding: encoding,
  433. headers: headers,
  434. requestModifier: requestModifier)
  435. return download(convertible, interceptor: interceptor, to: destination)
  436. }
  437. /// Creates a `DownloadRequest` from a `URLRequest` created using the passed components, `Encodable` parameters, and
  438. /// a `RequestInterceptor`.
  439. ///
  440. /// - Parameters:
  441. /// - convertible: `URLConvertible` value to be used as the `URLRequest`'s `URL`.
  442. /// - method: `HTTPMethod` for the `URLRequest`. `.get` by default.
  443. /// - parameters: Value conforming to `Encodable` to be encoded into the `URLRequest`. `nil` by default.
  444. /// - encoder: `ParameterEncoder` to be used to encode the `parameters` value into the `URLRequest`.
  445. /// Defaults to `URLEncodedFormParameterEncoder.default`.
  446. /// - headers: `HTTPHeaders` value to be added to the `URLRequest`. `nil` by default.
  447. /// - interceptor: `RequestInterceptor` value to be used by the returned `DataRequest`. `nil` by default.
  448. /// - requestModifier: `RequestModifier` which will be applied to the `URLRequest` created from the provided
  449. /// parameters. `nil` by default.
  450. /// - destination: `DownloadRequest.Destination` closure used to determine how and where the downloaded file
  451. /// should be moved. `nil` by default.
  452. ///
  453. /// - Returns: The created `DownloadRequest`.
  454. open func download<Parameters: Encodable>(_ convertible: URLConvertible,
  455. method: HTTPMethod = .get,
  456. parameters: Parameters? = nil,
  457. encoder: ParameterEncoder = URLEncodedFormParameterEncoder.default,
  458. headers: HTTPHeaders? = nil,
  459. interceptor: RequestInterceptor? = nil,
  460. requestModifier: RequestModifier? = nil,
  461. to destination: DownloadRequest.Destination? = nil) -> DownloadRequest {
  462. let convertible = RequestEncodableConvertible(url: convertible,
  463. method: method,
  464. parameters: parameters,
  465. encoder: encoder,
  466. headers: headers,
  467. requestModifier: requestModifier)
  468. return download(convertible, interceptor: interceptor, to: destination)
  469. }
  470. /// Creates a `DownloadRequest` from a `URLRequestConvertible` value, a `RequestInterceptor`, and a `Destination`.
  471. ///
  472. /// - Parameters:
  473. /// - convertible: `URLRequestConvertible` value to be used to create the `URLRequest`.
  474. /// - interceptor: `RequestInterceptor` value to be used by the returned `DataRequest`. `nil` by default.
  475. /// - destination: `DownloadRequest.Destination` closure used to determine how and where the downloaded file
  476. /// should be moved. `nil` by default.
  477. ///
  478. /// - Returns: The created `DownloadRequest`.
  479. open func download(_ convertible: URLRequestConvertible,
  480. interceptor: RequestInterceptor? = nil,
  481. to destination: DownloadRequest.Destination? = nil) -> DownloadRequest {
  482. let request = DownloadRequest(downloadable: .request(convertible),
  483. underlyingQueue: rootQueue,
  484. serializationQueue: serializationQueue,
  485. eventMonitor: eventMonitor,
  486. interceptor: interceptor,
  487. delegate: self,
  488. destination: destination ?? DownloadRequest.defaultDestination)
  489. perform(request)
  490. return request
  491. }
  492. /// Creates a `DownloadRequest` from the `resumeData` produced from a previously cancelled `DownloadRequest`, as
  493. /// well as a `RequestInterceptor`, and a `Destination`.
  494. ///
  495. /// - Note: If `destination` is not specified, the download will be moved to a temporary location determined by
  496. /// Alamofire. The file will not be deleted until the system purges the temporary files.
  497. ///
  498. /// - Note: On some versions of all Apple platforms (iOS 10 - 10.2, macOS 10.12 - 10.12.2, tvOS 10 - 10.1, watchOS 3 - 3.1.1),
  499. /// `resumeData` is broken on background URL session configurations. There's an underlying bug in the `resumeData`
  500. /// generation logic where the data is written incorrectly and will always fail to resume the download. For more
  501. /// information about the bug and possible workarounds, please refer to the [this Stack Overflow post](http://stackoverflow.com/a/39347461/1342462).
  502. ///
  503. /// - Parameters:
  504. /// - data: The resume data from a previously cancelled `DownloadRequest` or `URLSessionDownloadTask`.
  505. /// - interceptor: `RequestInterceptor` value to be used by the returned `DataRequest`. `nil` by default.
  506. /// - destination: `DownloadRequest.Destination` closure used to determine how and where the downloaded file
  507. /// should be moved. `nil` by default.
  508. ///
  509. /// - Returns: The created `DownloadRequest`.
  510. open func download(resumingWith data: Data,
  511. interceptor: RequestInterceptor? = nil,
  512. to destination: DownloadRequest.Destination? = nil) -> DownloadRequest {
  513. let request = DownloadRequest(downloadable: .resumeData(data),
  514. underlyingQueue: rootQueue,
  515. serializationQueue: serializationQueue,
  516. eventMonitor: eventMonitor,
  517. interceptor: interceptor,
  518. delegate: self,
  519. destination: destination ?? DownloadRequest.defaultDestination)
  520. perform(request)
  521. return request
  522. }
  523. // MARK: - UploadRequest
  524. struct ParameterlessRequestConvertible: URLRequestConvertible {
  525. let url: URLConvertible
  526. let method: HTTPMethod
  527. let headers: HTTPHeaders?
  528. let requestModifier: RequestModifier?
  529. func asURLRequest() throws -> URLRequest {
  530. var request = try URLRequest(url: url, method: method, headers: headers)
  531. try requestModifier?(&request)
  532. return request
  533. }
  534. }
  535. struct Upload: UploadConvertible {
  536. let request: URLRequestConvertible
  537. let uploadable: UploadableConvertible
  538. func createUploadable() throws -> UploadRequest.Uploadable {
  539. try uploadable.createUploadable()
  540. }
  541. func asURLRequest() throws -> URLRequest {
  542. try request.asURLRequest()
  543. }
  544. }
  545. // MARK: Data
  546. /// Creates an `UploadRequest` for the given `Data`, `URLRequest` components, and `RequestInterceptor`.
  547. ///
  548. /// - Parameters:
  549. /// - data: The `Data` to upload.
  550. /// - convertible: `URLConvertible` value to be used as the `URLRequest`'s `URL`.
  551. /// - method: `HTTPMethod` for the `URLRequest`. `.post` by default.
  552. /// - headers: `HTTPHeaders` value to be added to the `URLRequest`. `nil` by default.
  553. /// - interceptor: `RequestInterceptor` value to be used by the returned `DataRequest`. `nil` by default.
  554. /// - fileManager: `FileManager` instance to be used by the returned `UploadRequest`. `.default` instance by
  555. /// default.
  556. /// - requestModifier: `RequestModifier` which will be applied to the `URLRequest` created from the provided
  557. /// parameters. `nil` by default.
  558. ///
  559. /// - Returns: The created `UploadRequest`.
  560. open func upload(_ data: Data,
  561. to convertible: URLConvertible,
  562. method: HTTPMethod = .post,
  563. headers: HTTPHeaders? = nil,
  564. interceptor: RequestInterceptor? = nil,
  565. fileManager: FileManager = .default,
  566. requestModifier: RequestModifier? = nil) -> UploadRequest {
  567. let convertible = ParameterlessRequestConvertible(url: convertible,
  568. method: method,
  569. headers: headers,
  570. requestModifier: requestModifier)
  571. return upload(data, with: convertible, interceptor: interceptor, fileManager: fileManager)
  572. }
  573. /// Creates an `UploadRequest` for the given `Data` using the `URLRequestConvertible` value and `RequestInterceptor`.
  574. ///
  575. /// - Parameters:
  576. /// - data: The `Data` to upload.
  577. /// - convertible: `URLRequestConvertible` value to be used to create the `URLRequest`.
  578. /// - interceptor: `RequestInterceptor` value to be used by the returned `DataRequest`. `nil` by default.
  579. /// - fileManager: `FileManager` instance to be used by the returned `UploadRequest`. `.default` instance by
  580. /// default.
  581. ///
  582. /// - Returns: The created `UploadRequest`.
  583. open func upload(_ data: Data,
  584. with convertible: URLRequestConvertible,
  585. interceptor: RequestInterceptor? = nil,
  586. fileManager: FileManager = .default) -> UploadRequest {
  587. upload(.data(data), with: convertible, interceptor: interceptor, fileManager: fileManager)
  588. }
  589. // MARK: File
  590. /// Creates an `UploadRequest` for the file at the given file `URL`, using a `URLRequest` from the provided
  591. /// components and `RequestInterceptor`.
  592. ///
  593. /// - Parameters:
  594. /// - fileURL: The `URL` of the file to upload.
  595. /// - convertible: `URLConvertible` value to be used as the `URLRequest`'s `URL`.
  596. /// - method: `HTTPMethod` for the `URLRequest`. `.post` by default.
  597. /// - headers: `HTTPHeaders` value to be added to the `URLRequest`. `nil` by default.
  598. /// - interceptor: `RequestInterceptor` value to be used by the returned `UploadRequest`. `nil` by default.
  599. /// - fileManager: `FileManager` instance to be used by the returned `UploadRequest`. `.default` instance by
  600. /// default.
  601. /// - requestModifier: `RequestModifier` which will be applied to the `URLRequest` created from the provided
  602. /// parameters. `nil` by default.
  603. ///
  604. /// - Returns: The created `UploadRequest`.
  605. open func upload(_ fileURL: URL,
  606. to convertible: URLConvertible,
  607. method: HTTPMethod = .post,
  608. headers: HTTPHeaders? = nil,
  609. interceptor: RequestInterceptor? = nil,
  610. fileManager: FileManager = .default,
  611. requestModifier: RequestModifier? = nil) -> UploadRequest {
  612. let convertible = ParameterlessRequestConvertible(url: convertible,
  613. method: method,
  614. headers: headers,
  615. requestModifier: requestModifier)
  616. return upload(fileURL, with: convertible, interceptor: interceptor, fileManager: fileManager)
  617. }
  618. /// Creates an `UploadRequest` for the file at the given file `URL` using the `URLRequestConvertible` value and
  619. /// `RequestInterceptor`.
  620. ///
  621. /// - Parameters:
  622. /// - fileURL: The `URL` of the file to upload.
  623. /// - convertible: `URLRequestConvertible` value to be used to create the `URLRequest`.
  624. /// - interceptor: `RequestInterceptor` value to be used by the returned `DataRequest`. `nil` by default.
  625. /// - fileManager: `FileManager` instance to be used by the returned `UploadRequest`. `.default` instance by
  626. /// default.
  627. ///
  628. /// - Returns: The created `UploadRequest`.
  629. open func upload(_ fileURL: URL,
  630. with convertible: URLRequestConvertible,
  631. interceptor: RequestInterceptor? = nil,
  632. fileManager: FileManager = .default) -> UploadRequest {
  633. upload(.file(fileURL, shouldRemove: false), with: convertible, interceptor: interceptor, fileManager: fileManager)
  634. }
  635. // MARK: InputStream
  636. /// Creates an `UploadRequest` from the `InputStream` provided using a `URLRequest` from the provided components and
  637. /// `RequestInterceptor`.
  638. ///
  639. /// - Parameters:
  640. /// - stream: The `InputStream` that provides the data to upload.
  641. /// - convertible: `URLConvertible` value to be used as the `URLRequest`'s `URL`.
  642. /// - method: `HTTPMethod` for the `URLRequest`. `.post` by default.
  643. /// - headers: `HTTPHeaders` value to be added to the `URLRequest`. `nil` by default.
  644. /// - interceptor: `RequestInterceptor` value to be used by the returned `DataRequest`. `nil` by default.
  645. /// - fileManager: `FileManager` instance to be used by the returned `UploadRequest`. `.default` instance by
  646. /// default.
  647. /// - requestModifier: `RequestModifier` which will be applied to the `URLRequest` created from the provided
  648. /// parameters. `nil` by default.
  649. ///
  650. /// - Returns: The created `UploadRequest`.
  651. open func upload(_ stream: InputStream,
  652. to convertible: URLConvertible,
  653. method: HTTPMethod = .post,
  654. headers: HTTPHeaders? = nil,
  655. interceptor: RequestInterceptor? = nil,
  656. fileManager: FileManager = .default,
  657. requestModifier: RequestModifier? = nil) -> UploadRequest {
  658. let convertible = ParameterlessRequestConvertible(url: convertible,
  659. method: method,
  660. headers: headers,
  661. requestModifier: requestModifier)
  662. return upload(stream, with: convertible, interceptor: interceptor, fileManager: fileManager)
  663. }
  664. /// Creates an `UploadRequest` from the provided `InputStream` using the `URLRequestConvertible` value and
  665. /// `RequestInterceptor`.
  666. ///
  667. /// - Parameters:
  668. /// - stream: The `InputStream` that provides the data to upload.
  669. /// - convertible: `URLRequestConvertible` value to be used to create the `URLRequest`.
  670. /// - interceptor: `RequestInterceptor` value to be used by the returned `DataRequest`. `nil` by default.
  671. /// - fileManager: `FileManager` instance to be used by the returned `UploadRequest`. `.default` instance by
  672. /// default.
  673. ///
  674. /// - Returns: The created `UploadRequest`.
  675. open func upload(_ stream: InputStream,
  676. with convertible: URLRequestConvertible,
  677. interceptor: RequestInterceptor? = nil,
  678. fileManager: FileManager = .default) -> UploadRequest {
  679. upload(.stream(stream), with: convertible, interceptor: interceptor, fileManager: fileManager)
  680. }
  681. // MARK: MultipartFormData
  682. /// Creates an `UploadRequest` for the multipart form data built using a closure and sent using the provided
  683. /// `URLRequest` components and `RequestInterceptor`.
  684. ///
  685. /// It is important to understand the memory implications of uploading `MultipartFormData`. If the cumulative
  686. /// payload is small, encoding the data in-memory and directly uploading to a server is the by far the most
  687. /// efficient approach. However, if the payload is too large, encoding the data in-memory could cause your app to
  688. /// be terminated. Larger payloads must first be written to disk using input and output streams to keep the memory
  689. /// footprint low, then the data can be uploaded as a stream from the resulting file. Streaming from disk MUST be
  690. /// used for larger payloads such as video content.
  691. ///
  692. /// The `encodingMemoryThreshold` parameter allows Alamofire to automatically determine whether to encode in-memory
  693. /// or stream from disk. If the content length of the `MultipartFormData` is below the `encodingMemoryThreshold`,
  694. /// encoding takes place in-memory. If the content length exceeds the threshold, the data is streamed to disk
  695. /// during the encoding process. Then the result is uploaded as data or as a stream depending on which encoding
  696. /// technique was used.
  697. ///
  698. /// - Parameters:
  699. /// - multipartFormData: `MultipartFormData` building closure.
  700. /// - convertible: `URLConvertible` value to be used as the `URLRequest`'s `URL`.
  701. /// - encodingMemoryThreshold: Byte threshold used to determine whether the form data is encoded into memory or
  702. /// onto disk before being uploaded. `MultipartFormData.encodingMemoryThreshold` by
  703. /// default.
  704. /// - method: `HTTPMethod` for the `URLRequest`. `.post` by default.
  705. /// - headers: `HTTPHeaders` value to be added to the `URLRequest`. `nil` by default.
  706. /// - interceptor: `RequestInterceptor` value to be used by the returned `DataRequest`. `nil` by default.
  707. /// - fileManager: `FileManager` to be used if the form data exceeds the memory threshold and is
  708. /// written to disk before being uploaded. `.default` instance by default.
  709. /// - requestModifier: `RequestModifier` which will be applied to the `URLRequest` created from the
  710. /// provided parameters. `nil` by default.
  711. ///
  712. /// - Returns: The created `UploadRequest`.
  713. open func upload(multipartFormData: @escaping (MultipartFormData) -> Void,
  714. to url: URLConvertible,
  715. usingThreshold encodingMemoryThreshold: UInt64 = MultipartFormData.encodingMemoryThreshold,
  716. method: HTTPMethod = .post,
  717. headers: HTTPHeaders? = nil,
  718. interceptor: RequestInterceptor? = nil,
  719. fileManager: FileManager = .default,
  720. requestModifier: RequestModifier? = nil) -> UploadRequest {
  721. let convertible = ParameterlessRequestConvertible(url: url,
  722. method: method,
  723. headers: headers,
  724. requestModifier: requestModifier)
  725. let formData = MultipartFormData(fileManager: fileManager)
  726. multipartFormData(formData)
  727. return upload(multipartFormData: formData,
  728. with: convertible,
  729. usingThreshold: encodingMemoryThreshold,
  730. interceptor: interceptor,
  731. fileManager: fileManager)
  732. }
  733. /// Creates an `UploadRequest` using a `MultipartFormData` building closure, the provided `URLRequestConvertible`
  734. /// value, and a `RequestInterceptor`.
  735. ///
  736. /// It is important to understand the memory implications of uploading `MultipartFormData`. If the cumulative
  737. /// payload is small, encoding the data in-memory and directly uploading to a server is the by far the most
  738. /// efficient approach. However, if the payload is too large, encoding the data in-memory could cause your app to
  739. /// be terminated. Larger payloads must first be written to disk using input and output streams to keep the memory
  740. /// footprint low, then the data can be uploaded as a stream from the resulting file. Streaming from disk MUST be
  741. /// used for larger payloads such as video content.
  742. ///
  743. /// The `encodingMemoryThreshold` parameter allows Alamofire to automatically determine whether to encode in-memory
  744. /// or stream from disk. If the content length of the `MultipartFormData` is below the `encodingMemoryThreshold`,
  745. /// encoding takes place in-memory. If the content length exceeds the threshold, the data is streamed to disk
  746. /// during the encoding process. Then the result is uploaded as data or as a stream depending on which encoding
  747. /// technique was used.
  748. ///
  749. /// - Parameters:
  750. /// - multipartFormData: `MultipartFormData` building closure.
  751. /// - request: `URLRequestConvertible` value to be used to create the `URLRequest`.
  752. /// - encodingMemoryThreshold: Byte threshold used to determine whether the form data is encoded into memory or
  753. /// onto disk before being uploaded. `MultipartFormData.encodingMemoryThreshold` by
  754. /// default.
  755. /// - interceptor: `RequestInterceptor` value to be used by the returned `DataRequest`. `nil` by default.
  756. /// - fileManager: `FileManager` to be used if the form data exceeds the memory threshold and is
  757. /// written to disk before being uploaded. `.default` instance by default.
  758. ///
  759. /// - Returns: The created `UploadRequest`.
  760. open func upload(multipartFormData: @escaping (MultipartFormData) -> Void,
  761. with request: URLRequestConvertible,
  762. usingThreshold encodingMemoryThreshold: UInt64 = MultipartFormData.encodingMemoryThreshold,
  763. interceptor: RequestInterceptor? = nil,
  764. fileManager: FileManager = .default) -> UploadRequest {
  765. let formData = MultipartFormData(fileManager: fileManager)
  766. multipartFormData(formData)
  767. return upload(multipartFormData: formData,
  768. with: request,
  769. usingThreshold: encodingMemoryThreshold,
  770. interceptor: interceptor,
  771. fileManager: fileManager)
  772. }
  773. /// Creates an `UploadRequest` for the prebuilt `MultipartFormData` value using the provided `URLRequest` components
  774. /// and `RequestInterceptor`.
  775. ///
  776. /// It is important to understand the memory implications of uploading `MultipartFormData`. If the cumulative
  777. /// payload is small, encoding the data in-memory and directly uploading to a server is the by far the most
  778. /// efficient approach. However, if the payload is too large, encoding the data in-memory could cause your app to
  779. /// be terminated. Larger payloads must first be written to disk using input and output streams to keep the memory
  780. /// footprint low, then the data can be uploaded as a stream from the resulting file. Streaming from disk MUST be
  781. /// used for larger payloads such as video content.
  782. ///
  783. /// The `encodingMemoryThreshold` parameter allows Alamofire to automatically determine whether to encode in-memory
  784. /// or stream from disk. If the content length of the `MultipartFormData` is below the `encodingMemoryThreshold`,
  785. /// encoding takes place in-memory. If the content length exceeds the threshold, the data is streamed to disk
  786. /// during the encoding process. Then the result is uploaded as data or as a stream depending on which encoding
  787. /// technique was used.
  788. ///
  789. /// - Parameters:
  790. /// - multipartFormData: `MultipartFormData` instance to upload.
  791. /// - url: `URLConvertible` value to be used as the `URLRequest`'s `URL`.
  792. /// - encodingMemoryThreshold: Byte threshold used to determine whether the form data is encoded into memory or
  793. /// onto disk before being uploaded. `MultipartFormData.encodingMemoryThreshold` by
  794. /// default.
  795. /// - method: `HTTPMethod` for the `URLRequest`. `.post` by default.
  796. /// - headers: `HTTPHeaders` value to be added to the `URLRequest`. `nil` by default.
  797. /// - interceptor: `RequestInterceptor` value to be used by the returned `DataRequest`. `nil` by default.
  798. /// - fileManager: `FileManager` to be used if the form data exceeds the memory threshold and is
  799. /// written to disk before being uploaded. `.default` instance by default.
  800. /// - requestModifier: `RequestModifier` which will be applied to the `URLRequest` created from the
  801. /// provided parameters. `nil` by default.
  802. ///
  803. /// - Returns: The created `UploadRequest`.
  804. open func upload(multipartFormData: MultipartFormData,
  805. to url: URLConvertible,
  806. usingThreshold encodingMemoryThreshold: UInt64 = MultipartFormData.encodingMemoryThreshold,
  807. method: HTTPMethod = .post,
  808. headers: HTTPHeaders? = nil,
  809. interceptor: RequestInterceptor? = nil,
  810. fileManager: FileManager = .default,
  811. requestModifier: RequestModifier? = nil) -> UploadRequest {
  812. let convertible = ParameterlessRequestConvertible(url: url,
  813. method: method,
  814. headers: headers,
  815. requestModifier: requestModifier)
  816. let multipartUpload = MultipartUpload(isInBackgroundSession: session.configuration.identifier != nil,
  817. encodingMemoryThreshold: encodingMemoryThreshold,
  818. request: convertible,
  819. multipartFormData: multipartFormData)
  820. return upload(multipartUpload, interceptor: interceptor, fileManager: fileManager)
  821. }
  822. /// Creates an `UploadRequest` for the prebuilt `MultipartFormData` value using the providing `URLRequestConvertible`
  823. /// value and `RequestInterceptor`.
  824. ///
  825. /// It is important to understand the memory implications of uploading `MultipartFormData`. If the cumulative
  826. /// payload is small, encoding the data in-memory and directly uploading to a server is the by far the most
  827. /// efficient approach. However, if the payload is too large, encoding the data in-memory could cause your app to
  828. /// be terminated. Larger payloads must first be written to disk using input and output streams to keep the memory
  829. /// footprint low, then the data can be uploaded as a stream from the resulting file. Streaming from disk MUST be
  830. /// used for larger payloads such as video content.
  831. ///
  832. /// The `encodingMemoryThreshold` parameter allows Alamofire to automatically determine whether to encode in-memory
  833. /// or stream from disk. If the content length of the `MultipartFormData` is below the `encodingMemoryThreshold`,
  834. /// encoding takes place in-memory. If the content length exceeds the threshold, the data is streamed to disk
  835. /// during the encoding process. Then the result is uploaded as data or as a stream depending on which encoding
  836. /// technique was used.
  837. ///
  838. /// - Parameters:
  839. /// - multipartFormData: `MultipartFormData` instance to upload.
  840. /// - request: `URLRequestConvertible` value to be used to create the `URLRequest`.
  841. /// - encodingMemoryThreshold: Byte threshold used to determine whether the form data is encoded into memory or
  842. /// onto disk before being uploaded. `MultipartFormData.encodingMemoryThreshold` by
  843. /// default.
  844. /// - interceptor: `RequestInterceptor` value to be used by the returned `DataRequest`. `nil` by default.
  845. /// - fileManager: `FileManager` instance to be used by the returned `UploadRequest`. `.default` instance by
  846. /// default.
  847. ///
  848. /// - Returns: The created `UploadRequest`.
  849. open func upload(multipartFormData: MultipartFormData,
  850. with request: URLRequestConvertible,
  851. usingThreshold encodingMemoryThreshold: UInt64 = MultipartFormData.encodingMemoryThreshold,
  852. interceptor: RequestInterceptor? = nil,
  853. fileManager: FileManager = .default) -> UploadRequest {
  854. let multipartUpload = MultipartUpload(isInBackgroundSession: session.configuration.identifier != nil,
  855. encodingMemoryThreshold: encodingMemoryThreshold,
  856. request: request,
  857. multipartFormData: multipartFormData)
  858. return upload(multipartUpload, interceptor: interceptor, fileManager: fileManager)
  859. }
  860. // MARK: - Internal API
  861. // MARK: Uploadable
  862. func upload(_ uploadable: UploadRequest.Uploadable,
  863. with convertible: URLRequestConvertible,
  864. interceptor: RequestInterceptor?,
  865. fileManager: FileManager) -> UploadRequest {
  866. let uploadable = Upload(request: convertible, uploadable: uploadable)
  867. return upload(uploadable, interceptor: interceptor, fileManager: fileManager)
  868. }
  869. func upload(_ upload: UploadConvertible, interceptor: RequestInterceptor?, fileManager: FileManager) -> UploadRequest {
  870. let request = UploadRequest(convertible: upload,
  871. underlyingQueue: rootQueue,
  872. serializationQueue: serializationQueue,
  873. eventMonitor: eventMonitor,
  874. interceptor: interceptor,
  875. fileManager: fileManager,
  876. delegate: self)
  877. perform(request)
  878. return request
  879. }
  880. // MARK: Perform
  881. /// Starts performing the provided `Request`.
  882. ///
  883. /// - Parameter request: The `Request` to perform.
  884. func perform(_ request: Request) {
  885. rootQueue.async {
  886. guard !request.isCancelled else { return }
  887. self.activeRequests.insert(request)
  888. self.requestQueue.async {
  889. // Leaf types must come first, otherwise they will cast as their superclass.
  890. switch request {
  891. case let r as UploadRequest: self.performUploadRequest(r) // UploadRequest must come before DataRequest due to subtype relationship.
  892. case let r as DataRequest: self.performDataRequest(r)
  893. case let r as DownloadRequest: self.performDownloadRequest(r)
  894. case let r as DataStreamRequest: self.performDataStreamRequest(r)
  895. default: fatalError("Attempted to perform unsupported Request subclass: \(type(of: request))")
  896. }
  897. }
  898. }
  899. }
  900. func performDataRequest(_ request: DataRequest) {
  901. dispatchPrecondition(condition: .onQueue(requestQueue))
  902. performSetupOperations(for: request, convertible: request.convertible)
  903. }
  904. func performDataStreamRequest(_ request: DataStreamRequest) {
  905. dispatchPrecondition(condition: .onQueue(requestQueue))
  906. performSetupOperations(for: request, convertible: request.convertible)
  907. }
  908. func performUploadRequest(_ request: UploadRequest) {
  909. dispatchPrecondition(condition: .onQueue(requestQueue))
  910. do {
  911. let uploadable = try request.upload.createUploadable()
  912. rootQueue.async { request.didCreateUploadable(uploadable) }
  913. performSetupOperations(for: request, convertible: request.convertible)
  914. } catch {
  915. rootQueue.async { request.didFailToCreateUploadable(with: error.asAFError(or: .createUploadableFailed(error: error))) }
  916. }
  917. }
  918. func performDownloadRequest(_ request: DownloadRequest) {
  919. dispatchPrecondition(condition: .onQueue(requestQueue))
  920. switch request.downloadable {
  921. case let .request(convertible):
  922. performSetupOperations(for: request, convertible: convertible)
  923. case let .resumeData(resumeData):
  924. rootQueue.async { self.didReceiveResumeData(resumeData, for: request) }
  925. }
  926. }
  927. func performSetupOperations(for request: Request, convertible: URLRequestConvertible) {
  928. dispatchPrecondition(condition: .onQueue(requestQueue))
  929. let initialRequest: URLRequest
  930. do {
  931. initialRequest = try convertible.asURLRequest()
  932. try initialRequest.validate()
  933. } catch {
  934. rootQueue.async { request.didFailToCreateURLRequest(with: error.asAFError(or: .createURLRequestFailed(error: error))) }
  935. return
  936. }
  937. rootQueue.async { request.didCreateInitialURLRequest(initialRequest) }
  938. guard !request.isCancelled else { return }
  939. guard let adapter = adapter(for: request) else {
  940. rootQueue.async { self.didCreateURLRequest(initialRequest, for: request) }
  941. return
  942. }
  943. adapter.adapt(initialRequest, for: self) { result in
  944. do {
  945. let adaptedRequest = try result.get()
  946. try adaptedRequest.validate()
  947. self.rootQueue.async {
  948. request.didAdaptInitialRequest(initialRequest, to: adaptedRequest)
  949. self.didCreateURLRequest(adaptedRequest, for: request)
  950. }
  951. } catch {
  952. self.rootQueue.async { request.didFailToAdaptURLRequest(initialRequest, withError: .requestAdaptationFailed(error: error)) }
  953. }
  954. }
  955. }
  956. // MARK: - Task Handling
  957. func didCreateURLRequest(_ urlRequest: URLRequest, for request: Request) {
  958. dispatchPrecondition(condition: .onQueue(rootQueue))
  959. request.didCreateURLRequest(urlRequest)
  960. guard !request.isCancelled else { return }
  961. let task = request.task(for: urlRequest, using: session)
  962. requestTaskMap[request] = task
  963. request.didCreateTask(task)
  964. updateStatesForTask(task, request: request)
  965. }
  966. func didReceiveResumeData(_ data: Data, for request: DownloadRequest) {
  967. dispatchPrecondition(condition: .onQueue(rootQueue))
  968. guard !request.isCancelled else { return }
  969. let task = request.task(forResumeData: data, using: session)
  970. requestTaskMap[request] = task
  971. request.didCreateTask(task)
  972. updateStatesForTask(task, request: request)
  973. }
  974. func updateStatesForTask(_ task: URLSessionTask, request: Request) {
  975. dispatchPrecondition(condition: .onQueue(rootQueue))
  976. request.withState { state in
  977. switch state {
  978. case .initialized, .finished:
  979. // Do nothing.
  980. break
  981. case .resumed:
  982. task.resume()
  983. rootQueue.async { request.didResumeTask(task) }
  984. case .suspended:
  985. task.suspend()
  986. rootQueue.async { request.didSuspendTask(task) }
  987. case .cancelled:
  988. // Resume to ensure metrics are gathered.
  989. task.resume()
  990. task.cancel()
  991. rootQueue.async { request.didCancelTask(task) }
  992. }
  993. }
  994. }
  995. // MARK: - Adapters and Retriers
  996. func adapter(for request: Request) -> RequestAdapter? {
  997. if let requestInterceptor = request.interceptor, let sessionInterceptor = interceptor {
  998. return Interceptor(adapters: [requestInterceptor, sessionInterceptor])
  999. } else {
  1000. return request.interceptor ?? interceptor
  1001. }
  1002. }
  1003. func retrier(for request: Request) -> RequestRetrier? {
  1004. if let requestInterceptor = request.interceptor, let sessionInterceptor = interceptor {
  1005. return Interceptor(retriers: [requestInterceptor, sessionInterceptor])
  1006. } else {
  1007. return request.interceptor ?? interceptor
  1008. }
  1009. }
  1010. // MARK: - Invalidation
  1011. func finishRequestsForDeinit() {
  1012. requestTaskMap.requests.forEach { request in
  1013. rootQueue.async {
  1014. request.finish(error: AFError.sessionDeinitialized)
  1015. }
  1016. }
  1017. }
  1018. }
  1019. // MARK: - RequestDelegate
  1020. extension Session: RequestDelegate {
  1021. public var sessionConfiguration: URLSessionConfiguration {
  1022. session.configuration
  1023. }
  1024. public var startImmediately: Bool { startRequestsImmediately }
  1025. public func cleanup(after request: Request) {
  1026. activeRequests.remove(request)
  1027. }
  1028. public func retryResult(for request: Request, dueTo error: AFError, completion: @escaping (RetryResult) -> Void) {
  1029. guard let retrier = retrier(for: request) else {
  1030. rootQueue.async { completion(.doNotRetry) }
  1031. return
  1032. }
  1033. retrier.retry(request, for: self, dueTo: error) { retryResult in
  1034. self.rootQueue.async {
  1035. guard let retryResultError = retryResult.error else { completion(retryResult); return }
  1036. let retryError = AFError.requestRetryFailed(retryError: retryResultError, originalError: error)
  1037. completion(.doNotRetryWithError(retryError))
  1038. }
  1039. }
  1040. }
  1041. public func retryRequest(_ request: Request, withDelay timeDelay: TimeInterval?) {
  1042. rootQueue.async {
  1043. let retry: () -> Void = {
  1044. guard !request.isCancelled else { return }
  1045. request.prepareForRetry()
  1046. self.perform(request)
  1047. }
  1048. if let retryDelay = timeDelay {
  1049. self.rootQueue.after(retryDelay) { retry() }
  1050. } else {
  1051. retry()
  1052. }
  1053. }
  1054. }
  1055. }
  1056. // MARK: - SessionStateProvider
  1057. extension Session: SessionStateProvider {
  1058. func request(for task: URLSessionTask) -> Request? {
  1059. dispatchPrecondition(condition: .onQueue(rootQueue))
  1060. return requestTaskMap[task]
  1061. }
  1062. func didGatherMetricsForTask(_ task: URLSessionTask) {
  1063. dispatchPrecondition(condition: .onQueue(rootQueue))
  1064. let didDisassociate = requestTaskMap.disassociateIfNecessaryAfterGatheringMetricsForTask(task)
  1065. if didDisassociate {
  1066. waitingCompletions[task]?()
  1067. waitingCompletions[task] = nil
  1068. }
  1069. }
  1070. func didCompleteTask(_ task: URLSessionTask, completion: @escaping () -> Void) {
  1071. dispatchPrecondition(condition: .onQueue(rootQueue))
  1072. let didDisassociate = requestTaskMap.disassociateIfNecessaryAfterCompletingTask(task)
  1073. if didDisassociate {
  1074. completion()
  1075. } else {
  1076. waitingCompletions[task] = completion
  1077. }
  1078. }
  1079. func credential(for task: URLSessionTask, in protectionSpace: URLProtectionSpace) -> URLCredential? {
  1080. dispatchPrecondition(condition: .onQueue(rootQueue))
  1081. return requestTaskMap[task]?.credential ??
  1082. session.configuration.urlCredentialStorage?.defaultCredential(for: protectionSpace)
  1083. }
  1084. func cancelRequestsForSessionInvalidation(with error: Error?) {
  1085. dispatchPrecondition(condition: .onQueue(rootQueue))
  1086. requestTaskMap.requests.forEach { $0.finish(error: AFError.sessionInvalidated(error: error)) }
  1087. }
  1088. }