SessionManager.swift 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351
  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 SessionManager {
  26. open static let `default` = SessionManager()
  27. let delegate: SessionDelegate
  28. let rootQueue: DispatchQueue
  29. let requestQueue: DispatchQueue
  30. open let adapter: RequestAdapter?
  31. open let retrier: RequestRetrier?
  32. open let serverTrustManager: ServerTrustManager?
  33. open let session: URLSession
  34. open let eventMonitor: CompositeEventMonitor
  35. open let defaultEventMonitors: [EventMonitor] = [] // TODO: Create notification event monitor, make default
  36. public init(session: URLSession,
  37. delegate: SessionDelegate,
  38. rootQueue: DispatchQueue,
  39. requestQueue: DispatchQueue? = nil,
  40. adapter: RequestAdapter? = nil,
  41. serverTrustManager: ServerTrustManager? = nil,
  42. retrier: RequestRetrier? = nil,
  43. eventMonitors: [EventMonitor] = []) {
  44. precondition(session.delegate === delegate,
  45. "URLSession SessionManager initializer must pass the SessionDelegate that has been assigned to the URLSession as its delegate.")
  46. precondition(session.delegateQueue.underlyingQueue === rootQueue,
  47. "URLSession SessionManager intializer must pass the DispatchQueue used as the delegateQueue's underlyingQueue as the rootQueue.")
  48. self.session = session
  49. self.delegate = delegate
  50. self.rootQueue = rootQueue
  51. self.requestQueue = requestQueue ?? DispatchQueue(label: "\(rootQueue.label).requestQueue", target: rootQueue)
  52. self.adapter = adapter
  53. self.retrier = retrier
  54. self.serverTrustManager = serverTrustManager
  55. eventMonitor = CompositeEventMonitor(monitors: defaultEventMonitors + eventMonitors)
  56. delegate.didCreateSessionManager(self, withEventMonitor: eventMonitor)
  57. }
  58. public convenience init(configuration: URLSessionConfiguration = .default,
  59. delegate: SessionDelegate = SessionDelegate(),
  60. rootQueue: DispatchQueue = DispatchQueue(label: "org.alamofire.sessionManager.rootQueue"),
  61. requestQueue: DispatchQueue? = nil,
  62. adapter: RequestAdapter? = nil,
  63. serverTrustManager: ServerTrustManager? = nil,
  64. retrier: RequestRetrier? = nil,
  65. eventMonitors: [EventMonitor] = []) {
  66. let delegateQueue = OperationQueue(maxConcurrentOperationCount: 1, underlyingQueue: rootQueue, name: "org.alamofire.sessionManager.sessionDelegateQueue")
  67. let session = URLSession(configuration: configuration, delegate: delegate, delegateQueue: delegateQueue)
  68. self.init(session: session,
  69. delegate: delegate,
  70. rootQueue: rootQueue,
  71. requestQueue: requestQueue,
  72. adapter: adapter,
  73. serverTrustManager: serverTrustManager,
  74. retrier: retrier,
  75. eventMonitors: eventMonitors)
  76. }
  77. deinit {
  78. session.invalidateAndCancel()
  79. }
  80. // MARK: - Request
  81. struct RequestConvertible: URLRequestConvertible {
  82. let url: URLConvertible
  83. let method: HTTPMethod
  84. let parameters: Parameters?
  85. let encoding: ParameterEncoding
  86. let headers: HTTPHeaders?
  87. func asURLRequest() throws -> URLRequest {
  88. let request = try URLRequest(url: url, method: method, headers: headers)
  89. return try encoding.encode(request, with: parameters)
  90. }
  91. }
  92. // TODO: Serialization Queue support?
  93. open func request(_ url: URLConvertible,
  94. method: HTTPMethod = .get,
  95. parameters: Parameters? = nil,
  96. encoding: ParameterEncoding = URLEncoding.default,
  97. headers: HTTPHeaders? = nil) -> DataRequest {
  98. let convertible = RequestConvertible(url: url,
  99. method: method,
  100. parameters: parameters,
  101. encoding: encoding,
  102. headers: headers)
  103. return request(convertible)
  104. }
  105. open func request(_ convertible: URLRequestConvertible) -> DataRequest {
  106. let request = DataRequest(convertible: convertible,
  107. underlyingQueue: rootQueue,
  108. eventMonitor: eventMonitor,
  109. delegate: delegate)
  110. perform(request)
  111. return request
  112. }
  113. // MARK: - Download
  114. open func download(_ convertible: URLConvertible,
  115. method: HTTPMethod = .get,
  116. parameters: Parameters? = nil,
  117. encoding: ParameterEncoding = URLEncoding.default,
  118. headers: HTTPHeaders? = nil,
  119. to destination: DownloadRequest.Destination? = nil) -> DownloadRequest {
  120. let convertible = RequestConvertible(url: convertible,
  121. method: method,
  122. parameters: parameters,
  123. encoding: encoding,
  124. headers: headers)
  125. return download(convertible, to: destination)
  126. }
  127. open func download(_ convertible: URLRequestConvertible,
  128. to destination: DownloadRequest.Destination? = nil) -> DownloadRequest {
  129. let request = DownloadRequest(downloadable: .request(convertible),
  130. underlyingQueue: rootQueue,
  131. eventMonitor: eventMonitor,
  132. delegate: delegate,
  133. destination: destination)
  134. perform(request)
  135. return request
  136. }
  137. open func download(resumingWith data: Data,
  138. to destination: DownloadRequest.Destination? = nil) -> DownloadRequest {
  139. let request = DownloadRequest(downloadable: .resumeData(data),
  140. underlyingQueue: rootQueue,
  141. eventMonitor: eventMonitor,
  142. delegate: delegate,
  143. destination: destination)
  144. perform(request)
  145. return request
  146. }
  147. // MARK: - Upload
  148. struct ParameterlessRequestConvertible: URLRequestConvertible {
  149. let url: URLConvertible
  150. let method: HTTPMethod
  151. let headers: HTTPHeaders?
  152. func asURLRequest() throws -> URLRequest {
  153. return try URLRequest(url: url, method: method, headers: headers)
  154. }
  155. }
  156. struct Upload: UploadConvertible {
  157. let request: URLRequestConvertible
  158. let uploadable: UploadableConvertible
  159. func createUploadable() throws -> UploadRequest.Uploadable {
  160. return try uploadable.createUploadable()
  161. }
  162. func asURLRequest() throws -> URLRequest {
  163. return try request.asURLRequest()
  164. }
  165. }
  166. open func upload(_ data: Data,
  167. to convertible: URLConvertible,
  168. method: HTTPMethod = .post,
  169. headers: HTTPHeaders? = nil) -> UploadRequest {
  170. let convertible = ParameterlessRequestConvertible(url: convertible, method: method, headers: headers)
  171. return upload(data, with: convertible)
  172. }
  173. open func upload(_ data: Data, with convertible: URLRequestConvertible) -> UploadRequest {
  174. return upload(.data(data), with: convertible)
  175. }
  176. open func upload(_ fileURL: URL,
  177. to convertible: URLConvertible,
  178. method: HTTPMethod = .post,
  179. headers: HTTPHeaders? = nil) -> UploadRequest {
  180. let convertible = ParameterlessRequestConvertible(url: convertible, method: method, headers: headers)
  181. return upload(fileURL, with: convertible)
  182. }
  183. open func upload(_ fileURL: URL, with convertible: URLRequestConvertible) -> UploadRequest {
  184. return upload(.file(fileURL, shouldRemove: false), with: convertible)
  185. }
  186. open func upload(_ stream: InputStream,
  187. to convertible: URLConvertible,
  188. method: HTTPMethod = .post,
  189. headers: HTTPHeaders? = nil) -> UploadRequest {
  190. let convertible = ParameterlessRequestConvertible(url: convertible, method: method, headers: headers)
  191. return upload(stream, with: convertible)
  192. }
  193. open func upload(_ stream: InputStream, with convertible: URLRequestConvertible) -> UploadRequest {
  194. return upload(.stream(stream), with: convertible)
  195. }
  196. open func upload(multipartFormData: @escaping (MultipartFormData) -> Void,
  197. usingThreshold encodingMemoryThreshold: UInt64 = MultipartUpload.multipartFormDataEncodingMemoryThreshold,
  198. to url: URLConvertible,
  199. method: HTTPMethod = .post,
  200. headers: HTTPHeaders? = nil) -> UploadRequest {
  201. let convertible = ParameterlessRequestConvertible(url: url, method: method, headers: headers)
  202. return upload(multipartFormData: multipartFormData, usingThreshold: encodingMemoryThreshold, with: convertible)
  203. }
  204. open func upload(multipartFormData: @escaping (MultipartFormData) -> Void,
  205. usingThreshold encodingMemoryThreshold: UInt64 = MultipartUpload.multipartFormDataEncodingMemoryThreshold,
  206. with request: URLRequestConvertible) -> UploadRequest {
  207. let multipartUpload = MultipartUpload(isInBackgroundSession: (session.configuration.identifier != nil),
  208. encodingMemoryThreshold: encodingMemoryThreshold,
  209. request: request,
  210. multipartBuilder: multipartFormData)
  211. return upload(multipartUpload)
  212. }
  213. // MARK: - Internal API
  214. // MARK: Uploadable
  215. func upload(_ uploadable: UploadRequest.Uploadable, with convertible: URLRequestConvertible) -> UploadRequest {
  216. let uploadable = Upload(request: convertible, uploadable: uploadable)
  217. return upload(uploadable)
  218. }
  219. func upload(_ upload: UploadConvertible) -> UploadRequest {
  220. let request = UploadRequest(convertible: upload,
  221. underlyingQueue: rootQueue,
  222. eventMonitor: eventMonitor,
  223. delegate: delegate)
  224. perform(request)
  225. return request
  226. }
  227. // MARK: Downloadable
  228. // func download
  229. // MARK: Perform
  230. func perform(_ request: Request) {
  231. switch request {
  232. case let r as DataRequest: perform(r)
  233. case let r as UploadRequest: perform(r)
  234. case let r as DownloadRequest: perform(r)
  235. default: fatalError("Attempted to perform unsupported Request subclass: \(type(of: request))")
  236. }
  237. }
  238. func perform(_ request: DataRequest) {
  239. requestQueue.async {
  240. guard !request.isCancelled else { return }
  241. self.performSetupOperations(for: request, convertible: request.convertible)
  242. }
  243. }
  244. func perform(_ request: UploadRequest) {
  245. requestQueue.async {
  246. guard !request.isCancelled else { return }
  247. do {
  248. let uploadable = try request.upload.createUploadable()
  249. self.rootQueue.async { request.didCreateUploadable(uploadable) }
  250. self.performSetupOperations(for: request, convertible: request.convertible)
  251. } catch {
  252. self.rootQueue.async { request.didFailToCreateUploadable(with: error) }
  253. }
  254. }
  255. }
  256. func perform(_ request: DownloadRequest) {
  257. requestQueue.async {
  258. switch request.downloadable {
  259. case let .request(convertible):
  260. self.performSetupOperations(for: request, convertible: convertible)
  261. case let .resumeData(resumeData):
  262. self.rootQueue.async { self.delegate.didReceiveResumeData(resumeData, for: request) }
  263. }
  264. }
  265. }
  266. func performSetupOperations(for request: Request, convertible: URLRequestConvertible) {
  267. do {
  268. let initialRequest = try convertible.asURLRequest()
  269. self.rootQueue.async { request.didCreateURLRequest(initialRequest) }
  270. guard !request.isCancelled else { return }
  271. if let adapter = adapter {
  272. do {
  273. let adaptedRequest = try adapter.adapt(initialRequest)
  274. self.rootQueue.async {
  275. request.didAdaptInitialRequest(initialRequest, to: adaptedRequest)
  276. self.delegate.didCreateURLRequest(adaptedRequest, for: request)
  277. }
  278. } catch {
  279. self.rootQueue.async { request.didFailToAdaptURLRequest(initialRequest, withError: error) }
  280. }
  281. } else {
  282. self.rootQueue.async { self.delegate.didCreateURLRequest(initialRequest, for: request) }
  283. }
  284. } catch {
  285. self.rootQueue.async { request.didFailToCreateURLRequest(with: error) }
  286. }
  287. }
  288. }