2
0

Session.swift 25 KB

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