Session.swift 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499
  1. //
  2. // SessionManager.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. open class Session {
  26. public static let `default` = Session()
  27. public let delegate: SessionDelegate
  28. public let rootQueue: DispatchQueue
  29. public let requestQueue: DispatchQueue
  30. public let serializationQueue: DispatchQueue
  31. public let adapter: RequestAdapter?
  32. public let retrier: RequestRetrier?
  33. public let serverTrustManager: ServerTrustManager?
  34. public let session: URLSession
  35. public let eventMonitor: CompositeEventMonitor
  36. public let defaultEventMonitors: [EventMonitor] = [AlamofireNotifications()]
  37. var requestTaskMap = RequestTaskMap()
  38. public let startRequestsImmediately: Bool
  39. public init(startRequestsImmediately: Bool = true,
  40. session: URLSession,
  41. delegate: SessionDelegate,
  42. rootQueue: DispatchQueue,
  43. requestQueue: DispatchQueue? = nil,
  44. serializationQueue: DispatchQueue? = nil,
  45. adapter: RequestAdapter? = nil,
  46. serverTrustManager: ServerTrustManager? = nil,
  47. retrier: RequestRetrier? = nil,
  48. eventMonitors: [EventMonitor] = []) {
  49. precondition(session.delegate === delegate,
  50. "SessionManager(session:) initializer must be passed the delegate that has been assigned to the URLSession as the SessionDataProvider.")
  51. precondition(session.delegateQueue.underlyingQueue === rootQueue,
  52. "SessionManager(session:) intializer must be passed the DispatchQueue used as the delegateQueue's underlyingQueue as rootQueue.")
  53. self.startRequestsImmediately = startRequestsImmediately
  54. self.session = session
  55. self.delegate = delegate
  56. self.rootQueue = rootQueue
  57. self.requestQueue = requestQueue ?? DispatchQueue(label: "\(rootQueue.label).requestQueue", target: rootQueue)
  58. self.serializationQueue = serializationQueue ?? DispatchQueue(label: "\(rootQueue.label).serializationQueue", target: rootQueue)
  59. self.adapter = adapter
  60. self.retrier = retrier
  61. self.serverTrustManager = serverTrustManager
  62. eventMonitor = CompositeEventMonitor(monitors: defaultEventMonitors + eventMonitors)
  63. delegate.eventMonitor = eventMonitor
  64. delegate.stateProvider = self
  65. }
  66. public convenience init(startRequestsImmediately: Bool = true,
  67. configuration: URLSessionConfiguration = .alamofireDefault,
  68. delegate: SessionDelegate = SessionDelegate(),
  69. rootQueue: DispatchQueue = DispatchQueue(label: "org.alamofire.sessionManager.rootQueue"),
  70. requestQueue: DispatchQueue? = nil,
  71. serializationQueue: DispatchQueue? = nil,
  72. adapter: RequestAdapter? = nil,
  73. serverTrustManager: ServerTrustManager? = nil,
  74. retrier: RequestRetrier? = nil,
  75. eventMonitors: [EventMonitor] = []) {
  76. let delegateQueue = OperationQueue(maxConcurrentOperationCount: 1, underlyingQueue: rootQueue, name: "org.alamofire.sessionManager.sessionDelegateQueue")
  77. let session = URLSession(configuration: configuration, delegate: delegate, delegateQueue: delegateQueue)
  78. self.init(startRequestsImmediately: startRequestsImmediately,
  79. session: session,
  80. delegate: delegate,
  81. rootQueue: rootQueue,
  82. requestQueue: requestQueue,
  83. serializationQueue: serializationQueue,
  84. adapter: adapter,
  85. serverTrustManager: serverTrustManager,
  86. retrier: retrier,
  87. eventMonitors: eventMonitors)
  88. }
  89. deinit {
  90. session.invalidateAndCancel()
  91. }
  92. // MARK: - Request
  93. struct RequestConvertible: URLRequestConvertible {
  94. let url: URLConvertible
  95. let method: HTTPMethod
  96. let parameters: Parameters?
  97. let encoding: ParameterEncoding
  98. let headers: HTTPHeaders?
  99. func asURLRequest() throws -> URLRequest {
  100. let request = try URLRequest(url: url, method: method, headers: headers)
  101. return try encoding.encode(request, with: parameters)
  102. }
  103. }
  104. open func request(_ url: URLConvertible,
  105. method: HTTPMethod = .get,
  106. parameters: Parameters? = nil,
  107. encoding: ParameterEncoding = URLEncoding.default,
  108. headers: HTTPHeaders? = nil) -> DataRequest {
  109. let convertible = RequestConvertible(url: url,
  110. method: method,
  111. parameters: parameters,
  112. encoding: encoding,
  113. headers: headers)
  114. return request(convertible)
  115. }
  116. open func request(_ convertible: URLRequestConvertible) -> DataRequest {
  117. let request = DataRequest(convertible: convertible,
  118. underlyingQueue: rootQueue,
  119. serializationQueue: serializationQueue,
  120. eventMonitor: eventMonitor,
  121. delegate: self)
  122. perform(request)
  123. return request
  124. }
  125. // MARK: - Download
  126. open func download(_ convertible: URLConvertible,
  127. method: HTTPMethod = .get,
  128. parameters: Parameters? = nil,
  129. encoding: ParameterEncoding = URLEncoding.default,
  130. headers: HTTPHeaders? = nil,
  131. to destination: DownloadRequest.Destination? = nil) -> DownloadRequest {
  132. let convertible = RequestConvertible(url: convertible,
  133. method: method,
  134. parameters: parameters,
  135. encoding: encoding,
  136. headers: headers)
  137. return download(convertible, to: destination)
  138. }
  139. open func download(_ convertible: URLRequestConvertible,
  140. to destination: DownloadRequest.Destination? = nil) -> DownloadRequest {
  141. let request = DownloadRequest(downloadable: .request(convertible),
  142. underlyingQueue: rootQueue,
  143. serializationQueue: serializationQueue,
  144. eventMonitor: eventMonitor,
  145. delegate: self,
  146. destination: destination)
  147. perform(request)
  148. return request
  149. }
  150. open func download(resumingWith data: Data,
  151. to destination: DownloadRequest.Destination? = nil) -> DownloadRequest {
  152. let request = DownloadRequest(downloadable: .resumeData(data),
  153. underlyingQueue: rootQueue,
  154. serializationQueue: serializationQueue,
  155. eventMonitor: eventMonitor,
  156. delegate: self,
  157. destination: destination)
  158. perform(request)
  159. return request
  160. }
  161. // MARK: - Upload
  162. struct ParameterlessRequestConvertible: URLRequestConvertible {
  163. let url: URLConvertible
  164. let method: HTTPMethod
  165. let headers: HTTPHeaders?
  166. func asURLRequest() throws -> URLRequest {
  167. return try URLRequest(url: url, method: method, headers: headers)
  168. }
  169. }
  170. struct Upload: UploadConvertible {
  171. let request: URLRequestConvertible
  172. let uploadable: UploadableConvertible
  173. func createUploadable() throws -> UploadRequest.Uploadable {
  174. return try uploadable.createUploadable()
  175. }
  176. func asURLRequest() throws -> URLRequest {
  177. return try request.asURLRequest()
  178. }
  179. }
  180. open func upload(_ data: Data,
  181. to convertible: URLConvertible,
  182. method: HTTPMethod = .post,
  183. headers: HTTPHeaders? = nil) -> UploadRequest {
  184. let convertible = ParameterlessRequestConvertible(url: convertible, method: method, headers: headers)
  185. return upload(data, with: convertible)
  186. }
  187. open func upload(_ data: Data, with convertible: URLRequestConvertible) -> UploadRequest {
  188. return upload(.data(data), with: convertible)
  189. }
  190. open func upload(_ fileURL: URL,
  191. to convertible: URLConvertible,
  192. method: HTTPMethod = .post,
  193. headers: HTTPHeaders? = nil) -> UploadRequest {
  194. let convertible = ParameterlessRequestConvertible(url: convertible, method: method, headers: headers)
  195. return upload(fileURL, with: convertible)
  196. }
  197. open func upload(_ fileURL: URL, with convertible: URLRequestConvertible) -> UploadRequest {
  198. return upload(.file(fileURL, shouldRemove: false), with: convertible)
  199. }
  200. open func upload(_ stream: InputStream,
  201. to convertible: URLConvertible,
  202. method: HTTPMethod = .post,
  203. headers: HTTPHeaders? = nil) -> UploadRequest {
  204. let convertible = ParameterlessRequestConvertible(url: convertible, method: method, headers: headers)
  205. return upload(stream, with: convertible)
  206. }
  207. open func upload(_ stream: InputStream, with convertible: URLRequestConvertible) -> UploadRequest {
  208. return upload(.stream(stream), with: convertible)
  209. }
  210. open func upload(multipartFormData: @escaping (MultipartFormData) -> Void,
  211. usingThreshold encodingMemoryThreshold: UInt64 = MultipartUpload.encodingMemoryThreshold,
  212. fileManager: FileManager = .default,
  213. to url: URLConvertible,
  214. method: HTTPMethod = .post,
  215. headers: HTTPHeaders? = nil) -> UploadRequest {
  216. let convertible = ParameterlessRequestConvertible(url: url, method: method, headers: headers)
  217. return upload(multipartFormData: multipartFormData, usingThreshold: encodingMemoryThreshold, with: convertible)
  218. }
  219. open func upload(multipartFormData: @escaping (MultipartFormData) -> Void,
  220. usingThreshold encodingMemoryThreshold: UInt64 = MultipartUpload.encodingMemoryThreshold,
  221. fileManager: FileManager = .default,
  222. with request: URLRequestConvertible) -> UploadRequest {
  223. let multipartUpload = MultipartUpload(isInBackgroundSession: (session.configuration.identifier != nil),
  224. encodingMemoryThreshold: encodingMemoryThreshold,
  225. request: request,
  226. fileManager: fileManager,
  227. multipartBuilder: multipartFormData)
  228. return upload(multipartUpload)
  229. }
  230. // MARK: - Internal API
  231. // MARK: Uploadable
  232. func upload(_ uploadable: UploadRequest.Uploadable, with convertible: URLRequestConvertible) -> UploadRequest {
  233. let uploadable = Upload(request: convertible, uploadable: uploadable)
  234. return upload(uploadable)
  235. }
  236. func upload(_ upload: UploadConvertible) -> UploadRequest {
  237. let request = UploadRequest(convertible: upload,
  238. underlyingQueue: rootQueue,
  239. serializationQueue: serializationQueue,
  240. eventMonitor: eventMonitor,
  241. delegate: self)
  242. perform(request)
  243. return request
  244. }
  245. // MARK: Perform
  246. func perform(_ request: Request) {
  247. switch request {
  248. case let r as DataRequest: perform(r)
  249. case let r as UploadRequest: perform(r)
  250. case let r as DownloadRequest: perform(r)
  251. default: fatalError("Attempted to perform unsupported Request subclass: \(type(of: request))")
  252. }
  253. }
  254. func perform(_ request: DataRequest) {
  255. requestQueue.async {
  256. guard !request.isCancelled else { return }
  257. self.performSetupOperations(for: request, convertible: request.convertible)
  258. }
  259. }
  260. func perform(_ request: UploadRequest) {
  261. requestQueue.async {
  262. guard !request.isCancelled else { return }
  263. do {
  264. let uploadable = try request.upload.createUploadable()
  265. self.rootQueue.async { request.didCreateUploadable(uploadable) }
  266. self.performSetupOperations(for: request, convertible: request.convertible)
  267. } catch {
  268. self.rootQueue.async { request.didFailToCreateUploadable(with: error) }
  269. }
  270. }
  271. }
  272. func perform(_ request: DownloadRequest) {
  273. requestQueue.async {
  274. switch request.downloadable {
  275. case let .request(convertible):
  276. self.performSetupOperations(for: request, convertible: convertible)
  277. case let .resumeData(resumeData):
  278. self.rootQueue.async { self.didReceiveResumeData(resumeData, for: request) }
  279. }
  280. }
  281. }
  282. func performSetupOperations(for request: Request, convertible: URLRequestConvertible) {
  283. do {
  284. let initialRequest = try convertible.asURLRequest()
  285. rootQueue.async { request.didCreateURLRequest(initialRequest) }
  286. guard !request.isCancelled else { return }
  287. if let adapter = adapter {
  288. adapter.adapt(initialRequest) { (result) in
  289. do {
  290. let adaptedRequest = try result.unwrap()
  291. self.rootQueue.async {
  292. request.didAdaptInitialRequest(initialRequest, to: adaptedRequest)
  293. self.didCreateURLRequest(adaptedRequest, for: request)
  294. }
  295. } catch {
  296. self.rootQueue.async { request.didFailToAdaptURLRequest(initialRequest, withError: error) }
  297. }
  298. }
  299. } else {
  300. rootQueue.async { self.didCreateURLRequest(initialRequest, for: request) }
  301. }
  302. } catch {
  303. rootQueue.async { request.didFailToCreateURLRequest(with: error) }
  304. }
  305. }
  306. // MARK: - Task Handling
  307. func didCreateURLRequest(_ urlRequest: URLRequest, for request: Request) {
  308. guard !request.isCancelled else { return }
  309. let task = request.task(for: urlRequest, using: session)
  310. requestTaskMap[request] = task
  311. request.didCreateTask(task)
  312. resumeOrSuspendTask(task, ifNecessaryForRequest: request)
  313. }
  314. func didReceiveResumeData(_ data: Data, for request: DownloadRequest) {
  315. guard !request.isCancelled else { return }
  316. let task = request.task(forResumeData: data, using: session)
  317. requestTaskMap[request] = task
  318. request.didCreateTask(task)
  319. resumeOrSuspendTask(task, ifNecessaryForRequest: request)
  320. }
  321. func resumeOrSuspendTask(_ task: URLSessionTask, ifNecessaryForRequest request: Request) {
  322. if startRequestsImmediately || request.isResumed {
  323. task.resume()
  324. request.didResume()
  325. }
  326. if request.isSuspended {
  327. task.suspend()
  328. request.didSuspend()
  329. }
  330. }
  331. }
  332. // MARK: - RequestDelegate
  333. extension Session: RequestDelegate {
  334. public var sessionConfiguration: URLSessionConfiguration {
  335. return session.configuration
  336. }
  337. public func willRetryRequest(_ request: Request) -> Bool {
  338. return (retrier != nil)
  339. }
  340. public func retryRequest(_ request: Request, ifNecessaryWithError error: Error) {
  341. guard let retrier = retrier else { return }
  342. retrier.should(self, retry: request, with: error) { (shouldRetry, retryInterval) in
  343. guard !request.isCancelled else { return }
  344. self.rootQueue.async {
  345. guard !request.isCancelled else { return }
  346. guard shouldRetry else { request.finish(); return }
  347. self.rootQueue.after(retryInterval) {
  348. guard !request.isCancelled else { return }
  349. request.requestIsRetrying()
  350. self.perform(request)
  351. }
  352. }
  353. }
  354. }
  355. public func cancelRequest(_ request: Request) {
  356. rootQueue.async {
  357. guard let task = self.requestTaskMap[request] else {
  358. request.didCancel()
  359. request.finish()
  360. return
  361. }
  362. request.didCancel()
  363. task.cancel()
  364. }
  365. }
  366. public func cancelDownloadRequest(_ request: DownloadRequest, byProducingResumeData: @escaping (Data?) -> Void) {
  367. rootQueue.async {
  368. guard let downloadTask = self.requestTaskMap[request] as? URLSessionDownloadTask else {
  369. request.didCancel()
  370. request.finish()
  371. return
  372. }
  373. downloadTask.cancel { (data) in
  374. self.rootQueue.async {
  375. byProducingResumeData(data)
  376. request.didCancel()
  377. }
  378. }
  379. }
  380. }
  381. public func suspendRequest(_ request: Request) {
  382. rootQueue.async {
  383. guard !request.isCancelled, let task = self.requestTaskMap[request] else { return }
  384. task.suspend()
  385. request.didSuspend()
  386. }
  387. }
  388. public func resumeRequest(_ request: Request) {
  389. rootQueue.async {
  390. guard !request.isCancelled, let task = self.requestTaskMap[request] else { return }
  391. task.resume()
  392. request.didResume()
  393. }
  394. }
  395. }
  396. // MARK: - SessionDelegateDelegate
  397. extension Session: SessionStateProvider {
  398. public func request(for task: URLSessionTask) -> Request? {
  399. return requestTaskMap[task]
  400. }
  401. public func didCompleteTask(_ task: URLSessionTask) {
  402. requestTaskMap[task] = nil
  403. }
  404. public func credential(for task: URLSessionTask, protectionSpace: URLProtectionSpace) -> URLCredential? {
  405. return requestTaskMap[task]?.credential ?? session.configuration.urlCredentialStorage?.defaultCredential(for: protectionSpace)
  406. }
  407. }