Session.swift 26 KB

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