Session.swift 22 KB

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