ImageDownloader.swift 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435
  1. //
  2. // ImageDownloader.swift
  3. // Kingfisher
  4. //
  5. // Created by Wei Wang on 15/4/6.
  6. //
  7. // Copyright (c) 2019 Wei Wang <onevcat@gmail.com>
  8. //
  9. // Permission is hereby granted, free of charge, to any person obtaining a copy
  10. // of this software and associated documentation files (the "Software"), to deal
  11. // in the Software without restriction, including without limitation the rights
  12. // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  13. // copies of the Software, and to permit persons to whom the Software is
  14. // furnished to do so, subject to the following conditions:
  15. //
  16. // The above copyright notice and this permission notice shall be included in
  17. // all copies or substantial portions of the Software.
  18. //
  19. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  20. // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  21. // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  22. // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  23. // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  24. // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
  25. // THE SOFTWARE.
  26. #if os(macOS)
  27. import AppKit
  28. #else
  29. import UIKit
  30. #endif
  31. typealias DownloadResult = Result<ImageLoadingResult, KingfisherError>
  32. /// Represents a success result of an image downloading progress.
  33. public struct ImageLoadingResult {
  34. /// The downloaded image.
  35. public let image: KFCrossPlatformImage
  36. /// Original URL of the image request.
  37. public let url: URL?
  38. /// The raw data received from downloader.
  39. public let originalData: Data
  40. }
  41. /// Represents a task of an image downloading process.
  42. public struct DownloadTask {
  43. /// The `SessionDataTask` object bounded to this download task. Multiple `DownloadTask`s could refer
  44. /// to a same `sessionTask`. This is an optimization in Kingfisher to prevent multiple downloading task
  45. /// for the same URL resource at the same time.
  46. ///
  47. /// When you `cancel` a `DownloadTask`, this `SessionDataTask` and its cancel token will be pass through.
  48. /// You can use them to identify the cancelled task.
  49. public let sessionTask: SessionDataTask
  50. /// The cancel token which is used to cancel the task. This is only for identify the task when it is cancelled.
  51. /// To cancel a `DownloadTask`, use `cancel` instead.
  52. public let cancelToken: SessionDataTask.CancelToken
  53. /// Cancel this task if it is running. It will do nothing if this task is not running.
  54. ///
  55. /// - Note:
  56. /// In Kingfisher, there is an optimization to prevent starting another download task if the target URL is being
  57. /// downloading. However, even when internally no new session task created, a `DownloadTask` will be still created
  58. /// and returned when you call related methods, but it will share the session downloading task with a previous task.
  59. /// In this case, if multiple `DownloadTask`s share a single session download task, cancelling a `DownloadTask`
  60. /// does not affect other `DownloadTask`s.
  61. ///
  62. /// If you need to cancel all `DownloadTask`s of a url, use `ImageDownloader.cancel(url:)`. If you need to cancel
  63. /// all downloading tasks of an `ImageDownloader`, use `ImageDownloader.cancelAll()`.
  64. public func cancel() {
  65. sessionTask.cancel(token: cancelToken)
  66. }
  67. }
  68. extension DownloadTask {
  69. enum WrappedTask {
  70. case download(DownloadTask)
  71. case dataProviding
  72. func cancel() {
  73. switch self {
  74. case .download(let task): task.cancel()
  75. case .dataProviding: break
  76. }
  77. }
  78. var value: DownloadTask? {
  79. switch self {
  80. case .download(let task): return task
  81. case .dataProviding: return nil
  82. }
  83. }
  84. }
  85. }
  86. /// Represents a downloading manager for requesting the image with a URL from server.
  87. open class ImageDownloader {
  88. // MARK: Singleton
  89. /// The default downloader.
  90. public static let `default` = ImageDownloader(name: "default")
  91. // MARK: Public Properties
  92. /// The duration before the downloading is timeout. Default is 15 seconds.
  93. open var downloadTimeout: TimeInterval = 15.0
  94. /// A set of trusted hosts when receiving server trust challenges. A challenge with host name contained in this
  95. /// set will be ignored. You can use this set to specify the self-signed site. It only will be used if you don't
  96. /// specify the `authenticationChallengeResponder`.
  97. ///
  98. /// If `authenticationChallengeResponder` is set, this property will be ignored and the implementation of
  99. /// `authenticationChallengeResponder` will be used instead.
  100. open var trustedHosts: Set<String>?
  101. /// Use this to set supply a configuration for the downloader. By default,
  102. /// NSURLSessionConfiguration.ephemeralSessionConfiguration() will be used.
  103. ///
  104. /// You could change the configuration before a downloading task starts.
  105. /// A configuration without persistent storage for caches is requested for downloader working correctly.
  106. open var sessionConfiguration = URLSessionConfiguration.ephemeral {
  107. didSet {
  108. session.invalidateAndCancel()
  109. session = URLSession(configuration: sessionConfiguration, delegate: sessionDelegate, delegateQueue: nil)
  110. }
  111. }
  112. /// Whether the download requests should use pipeline or not. Default is false.
  113. open var requestsUsePipelining = false
  114. /// Delegate of this `ImageDownloader` object. See `ImageDownloaderDelegate` protocol for more.
  115. open weak var delegate: ImageDownloaderDelegate?
  116. /// A responder for authentication challenge.
  117. /// Downloader will forward the received authentication challenge for the downloading session to this responder.
  118. open weak var authenticationChallengeResponder: AuthenticationChallengeResponsable?
  119. private let name: String
  120. private let sessionDelegate: SessionDelegate
  121. private var session: URLSession
  122. // MARK: Initializers
  123. /// Creates a downloader with name.
  124. ///
  125. /// - Parameter name: The name for the downloader. It should not be empty.
  126. public init(name: String) {
  127. if name.isEmpty {
  128. fatalError("[Kingfisher] You should specify a name for the downloader. "
  129. + "A downloader with empty name is not permitted.")
  130. }
  131. self.name = name
  132. sessionDelegate = SessionDelegate()
  133. session = URLSession(
  134. configuration: sessionConfiguration,
  135. delegate: sessionDelegate,
  136. delegateQueue: nil)
  137. authenticationChallengeResponder = self
  138. setupSessionHandler()
  139. }
  140. deinit { session.invalidateAndCancel() }
  141. private func setupSessionHandler() {
  142. sessionDelegate.onReceiveSessionChallenge.delegate(on: self) { (self, invoke) in
  143. self.authenticationChallengeResponder?.downloader(self, didReceive: invoke.1, completionHandler: invoke.2)
  144. }
  145. sessionDelegate.onReceiveSessionTaskChallenge.delegate(on: self) { (self, invoke) in
  146. self.authenticationChallengeResponder?.downloader(
  147. self, task: invoke.1, didReceive: invoke.2, completionHandler: invoke.3)
  148. }
  149. sessionDelegate.onValidStatusCode.delegate(on: self) { (self, code) in
  150. return (self.delegate ?? self).isValidStatusCode(code, for: self)
  151. }
  152. sessionDelegate.onDownloadingFinished.delegate(on: self) { (self, value) in
  153. let (url, result) = value
  154. do {
  155. let value = try result.get()
  156. self.delegate?.imageDownloader(self, didFinishDownloadingImageForURL: url, with: value, error: nil)
  157. } catch {
  158. self.delegate?.imageDownloader(self, didFinishDownloadingImageForURL: url, with: nil, error: error)
  159. }
  160. }
  161. sessionDelegate.onDidDownloadData.delegate(on: self) { (self, task) in
  162. guard let url = task.originalURL else {
  163. return task.mutableData
  164. }
  165. return (self.delegate ?? self).imageDownloader(self, didDownload: task.mutableData, for: url)
  166. }
  167. }
  168. // Wraps `completionHandler` to `onCompleted` respectively.
  169. private func createCompletionCallBack(_ completionHandler: ((DownloadResult) -> Void)?) -> Delegate<DownloadResult, Void>? {
  170. return completionHandler.map { block -> Delegate<DownloadResult, Void> in
  171. let delegate = Delegate<Result<ImageLoadingResult, KingfisherError>, Void>()
  172. delegate.delegate(on: self) { (self, callback) in
  173. block(callback)
  174. }
  175. return delegate
  176. }
  177. }
  178. private func createTaskCallback(
  179. _ completionHandler: ((DownloadResult) -> Void)?,
  180. options: KingfisherParsedOptionsInfo
  181. ) -> SessionDataTask.TaskCallback
  182. {
  183. return SessionDataTask.TaskCallback(
  184. onCompleted: createCompletionCallBack(completionHandler),
  185. options: options
  186. )
  187. }
  188. private func addDownloadTask(
  189. context: DownloadingContext,
  190. callback: SessionDataTask.TaskCallback
  191. ) -> DownloadTask
  192. {
  193. // Ready to start download. Add it to session task manager (`sessionHandler`)
  194. let downloadTask: DownloadTask
  195. if let existingTask = sessionDelegate.task(for: context.url) {
  196. downloadTask = sessionDelegate.append(existingTask, url: context.url, callback: callback)
  197. } else {
  198. let sessionDataTask = session.dataTask(with: context.request)
  199. sessionDataTask.priority = context.options.downloadPriority
  200. downloadTask = sessionDelegate.add(sessionDataTask, url: context.url, callback: callback)
  201. }
  202. return downloadTask
  203. }
  204. private func startDownloadTask(
  205. context: DownloadingContext,
  206. callback: SessionDataTask.TaskCallback
  207. ) -> DownloadTask
  208. {
  209. let downloadTask = addDownloadTask(context: context, callback: callback)
  210. let sessionTask = downloadTask.sessionTask
  211. guard !sessionTask.started else {
  212. return downloadTask
  213. }
  214. sessionTask.onTaskDone.delegate(on: self) { (self, done) in
  215. // Underlying downloading finishes.
  216. // result: Result<(Data, URLResponse?)>, callbacks: [TaskCallback]
  217. let (result, callbacks) = done
  218. // Before processing the downloaded data.
  219. do {
  220. let value = try result.get()
  221. self.delegate?.imageDownloader(
  222. self,
  223. didFinishDownloadingImageForURL: context.url,
  224. with: value.1,
  225. error: nil
  226. )
  227. } catch {
  228. self.delegate?.imageDownloader(
  229. self,
  230. didFinishDownloadingImageForURL: context.url,
  231. with: nil,
  232. error: error
  233. )
  234. }
  235. switch result {
  236. // Download finished. Now process the data to an image.
  237. case .success(let (data, response)):
  238. let processor = ImageDataProcessor(
  239. data: data, callbacks: callbacks, processingQueue: context.options.processingQueue)
  240. processor.onImageProcessed.delegate(on: self) { (self, result) in
  241. // `onImageProcessed` will be called for `callbacks.count` times, with each
  242. // `SessionDataTask.TaskCallback` as the input parameter.
  243. // result: Result<Image>, callback: SessionDataTask.TaskCallback
  244. let (result, callback) = result
  245. if let image = try? result.get() {
  246. self.delegate?.imageDownloader(self, didDownload: image, for: context.url, with: response)
  247. }
  248. let imageResult = result.map { ImageLoadingResult(image: $0, url: context.url, originalData: data) }
  249. let queue = callback.options.callbackQueue
  250. queue.execute { callback.onCompleted?.call(imageResult) }
  251. }
  252. processor.process()
  253. case .failure(let error):
  254. callbacks.forEach { callback in
  255. let queue = callback.options.callbackQueue
  256. queue.execute { callback.onCompleted?.call(.failure(error)) }
  257. }
  258. }
  259. }
  260. delegate?.imageDownloader(self, willDownloadImageForURL: context.url, with: context.request)
  261. sessionTask.resume()
  262. return downloadTask
  263. }
  264. // MARK: Dowloading Task
  265. /// Downloads an image with a URL and option. Invoked internally by Kingfisher. Subclasses must invoke super.
  266. ///
  267. /// - Parameters:
  268. /// - url: Target URL.
  269. /// - options: The options could control download behavior. See `KingfisherOptionsInfo`.
  270. /// - completionHandler: Called when the download progress finishes. This block will be called in the queue
  271. /// defined in `.callbackQueue` in `options` parameter.
  272. /// - Returns: A downloading task. You could call `cancel` on it to stop the download task.
  273. @discardableResult
  274. open func downloadImage(
  275. with url: URL,
  276. options: KingfisherParsedOptionsInfo,
  277. completionHandler: ((Result<ImageLoadingResult, KingfisherError>) -> Void)? = nil) -> DownloadTask?
  278. {
  279. // Creates default request.
  280. var request = URLRequest(url: url, cachePolicy: .reloadIgnoringLocalCacheData, timeoutInterval: downloadTimeout)
  281. request.httpShouldUsePipelining = requestsUsePipelining
  282. if let requestModifier = options.requestModifier {
  283. // Modifies request before sending.
  284. guard let r = requestModifier.modified(for: request) else {
  285. options.callbackQueue.execute {
  286. completionHandler?(.failure(KingfisherError.requestError(reason: .emptyRequest)))
  287. }
  288. return nil
  289. }
  290. request = r
  291. }
  292. // There is a possibility that request modifier changed the url to `nil` or empty.
  293. // In this case, throw an error.
  294. guard let url = request.url, !url.absoluteString.isEmpty else {
  295. options.callbackQueue.execute {
  296. completionHandler?(.failure(KingfisherError.requestError(reason: .invalidURL(request: request))))
  297. }
  298. return nil
  299. }
  300. let downloadTask = startDownloadTask(
  301. context: DownloadingContext(url: url, request: request, options: options),
  302. callback: createTaskCallback(completionHandler, options: options)
  303. )
  304. return downloadTask
  305. }
  306. /// Downloads an image with a URL and option.
  307. ///
  308. /// - Parameters:
  309. /// - url: Target URL.
  310. /// - options: The options could control download behavior. See `KingfisherOptionsInfo`.
  311. /// - progressBlock: Called when the download progress updated. This block will be always be called in main queue.
  312. /// - completionHandler: Called when the download progress finishes. This block will be called in the queue
  313. /// defined in `.callbackQueue` in `options` parameter.
  314. /// - Returns: A downloading task. You could call `cancel` on it to stop the download task.
  315. @discardableResult
  316. open func downloadImage(
  317. with url: URL,
  318. options: KingfisherOptionsInfo? = nil,
  319. progressBlock: DownloadProgressBlock? = nil,
  320. completionHandler: ((Result<ImageLoadingResult, KingfisherError>) -> Void)? = nil) -> DownloadTask?
  321. {
  322. var info = KingfisherParsedOptionsInfo(options)
  323. if let block = progressBlock {
  324. info.onDataReceived = (info.onDataReceived ?? []) + [ImageLoadingProgressSideEffect(block)]
  325. }
  326. return downloadImage(
  327. with: url,
  328. options: info,
  329. completionHandler: completionHandler)
  330. }
  331. /// Downloads an image with a URL and option.
  332. ///
  333. /// - Parameters:
  334. /// - url: Target URL.
  335. /// - options: The options could control download behavior. See `KingfisherOptionsInfo`.
  336. /// - completionHandler: Called when the download progress finishes. This block will be called in the queue
  337. /// defined in `.callbackQueue` in `options` parameter.
  338. /// - Returns: A downloading task. You could call `cancel` on it to stop the download task.
  339. @discardableResult
  340. open func downloadImage(
  341. with url: URL,
  342. options: KingfisherOptionsInfo? = nil,
  343. completionHandler: ((Result<ImageLoadingResult, KingfisherError>) -> Void)? = nil) -> DownloadTask?
  344. {
  345. downloadImage(
  346. with: url,
  347. options: KingfisherParsedOptionsInfo(options),
  348. completionHandler: completionHandler
  349. )
  350. }
  351. }
  352. // MARK: Cancelling Task
  353. extension ImageDownloader {
  354. /// Cancel all downloading tasks for this `ImageDownloader`. It will trigger the completion handlers
  355. /// for all not-yet-finished downloading tasks.
  356. ///
  357. /// If you need to only cancel a certain task, call `cancel()` on the `DownloadTask`
  358. /// returned by the downloading methods. If you need to cancel all `DownloadTask`s of a certain url,
  359. /// use `ImageDownloader.cancel(url:)`.
  360. public func cancelAll() {
  361. sessionDelegate.cancelAll()
  362. }
  363. /// Cancel all downloading tasks for a given URL. It will trigger the completion handlers for
  364. /// all not-yet-finished downloading tasks for the URL.
  365. ///
  366. /// - Parameter url: The URL which you want to cancel downloading.
  367. public func cancel(url: URL) {
  368. sessionDelegate.cancel(url: url)
  369. }
  370. }
  371. // Use the default implementation from extension of `AuthenticationChallengeResponsable`.
  372. extension ImageDownloader: AuthenticationChallengeResponsable {}
  373. // Use the default implementation from extension of `ImageDownloaderDelegate`.
  374. extension ImageDownloader: ImageDownloaderDelegate {}
  375. extension ImageDownloader {
  376. struct DownloadingContext {
  377. let url: URL
  378. let request: URLRequest
  379. let options: KingfisherParsedOptionsInfo
  380. }
  381. }