ResponseSerialization.swift 66 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282
  1. //
  2. // ResponseSerialization.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. // MARK: Protocols
  26. /// The type to which all data response serializers must conform in order to serialize a response.
  27. public protocol DataResponseSerializerProtocol {
  28. /// The type of serialized object to be created.
  29. associatedtype SerializedObject
  30. /// Serialize the response `Data` into the provided type..
  31. ///
  32. /// - Parameters:
  33. /// - request: `URLRequest` which was used to perform the request, if any.
  34. /// - response: `HTTPURLResponse` received from the server, if any.
  35. /// - data: `Data` returned from the server, if any.
  36. /// - error: `Error` produced by Alamofire or the underlying `URLSession` during the request.
  37. ///
  38. /// - Returns: The `SerializedObject`.
  39. /// - Throws: Any `Error` produced during serialization.
  40. func serialize(request: URLRequest?, response: HTTPURLResponse?, data: Data?, error: Error?) throws -> SerializedObject
  41. }
  42. /// The type to which all download response serializers must conform in order to serialize a response.
  43. public protocol DownloadResponseSerializerProtocol {
  44. /// The type of serialized object to be created.
  45. associatedtype SerializedObject
  46. /// Serialize the downloaded response `Data` from disk into the provided type..
  47. ///
  48. /// - Parameters:
  49. /// - request: `URLRequest` which was used to perform the request, if any.
  50. /// - response: `HTTPURLResponse` received from the server, if any.
  51. /// - fileURL: File `URL` to which the response data was downloaded.
  52. /// - error: `Error` produced by Alamofire or the underlying `URLSession` during the request.
  53. ///
  54. /// - Returns: The `SerializedObject`.
  55. /// - Throws: Any `Error` produced during serialization.
  56. func serializeDownload(request: URLRequest?, response: HTTPURLResponse?, fileURL: URL?, error: Error?) throws -> SerializedObject
  57. }
  58. /// A serializer that can handle both data and download responses.
  59. public protocol ResponseSerializer: DataResponseSerializerProtocol & DownloadResponseSerializerProtocol {
  60. /// `DataPreprocessor` used to prepare incoming `Data` for serialization.
  61. var dataPreprocessor: DataPreprocessor { get }
  62. /// `HTTPMethod`s for which empty response bodies are considered appropriate.
  63. var emptyRequestMethods: Set<HTTPMethod> { get }
  64. /// HTTP response codes for which empty response bodies are considered appropriate.
  65. var emptyResponseCodes: Set<Int> { get }
  66. }
  67. /// Type used to preprocess `Data` before it handled by a serializer.
  68. public protocol DataPreprocessor {
  69. /// Process `Data` before it's handled by a serializer.
  70. /// - Parameter data: The raw `Data` to process.
  71. func preprocess(_ data: Data) throws -> Data
  72. }
  73. /// `DataPreprocessor` that returns passed `Data` without any transform.
  74. public struct PassthroughPreprocessor: DataPreprocessor {
  75. public init() {}
  76. public func preprocess(_ data: Data) throws -> Data { data }
  77. }
  78. /// `DataPreprocessor` that trims Google's typical `)]}',\n` XSSI JSON header.
  79. public struct GoogleXSSIPreprocessor: DataPreprocessor {
  80. public init() {}
  81. public func preprocess(_ data: Data) throws -> Data {
  82. (data.prefix(6) == Data(")]}',\n".utf8)) ? data.dropFirst(6) : data
  83. }
  84. }
  85. #if swift(>=5.5)
  86. extension DataPreprocessor where Self == PassthroughPreprocessor {
  87. /// Provides a `PassthroughPreprocessor` instance.
  88. public static var passthrough: PassthroughPreprocessor { PassthroughPreprocessor() }
  89. }
  90. extension DataPreprocessor where Self == GoogleXSSIPreprocessor {
  91. /// Provides a `GoogleXSSIPreprocessor` instance.
  92. public static var googleXSSI: GoogleXSSIPreprocessor { GoogleXSSIPreprocessor() }
  93. }
  94. #endif
  95. extension ResponseSerializer {
  96. /// Default `DataPreprocessor`. `PassthroughPreprocessor` by default.
  97. public static var defaultDataPreprocessor: DataPreprocessor { PassthroughPreprocessor() }
  98. /// Default `HTTPMethod`s for which empty response bodies are considered appropriate. `[.head]` by default.
  99. public static var defaultEmptyRequestMethods: Set<HTTPMethod> { [.head] }
  100. /// HTTP response codes for which empty response bodies are considered appropriate. `[204, 205]` by default.
  101. public static var defaultEmptyResponseCodes: Set<Int> { [204, 205] }
  102. public var dataPreprocessor: DataPreprocessor { Self.defaultDataPreprocessor }
  103. public var emptyRequestMethods: Set<HTTPMethod> { Self.defaultEmptyRequestMethods }
  104. public var emptyResponseCodes: Set<Int> { Self.defaultEmptyResponseCodes }
  105. /// Determines whether the `request` allows empty response bodies, if `request` exists.
  106. ///
  107. /// - Parameter request: `URLRequest` to evaluate.
  108. ///
  109. /// - Returns: `Bool` representing the outcome of the evaluation, or `nil` if `request` was `nil`.
  110. public func requestAllowsEmptyResponseData(_ request: URLRequest?) -> Bool? {
  111. request.flatMap(\.httpMethod)
  112. .flatMap(HTTPMethod.init)
  113. .map { emptyRequestMethods.contains($0) }
  114. }
  115. /// Determines whether the `response` allows empty response bodies, if `response` exists`.
  116. ///
  117. /// - Parameter response: `HTTPURLResponse` to evaluate.
  118. ///
  119. /// - Returns: `Bool` representing the outcome of the evaluation, or `nil` if `response` was `nil`.
  120. public func responseAllowsEmptyResponseData(_ response: HTTPURLResponse?) -> Bool? {
  121. response.map(\.statusCode)
  122. .map { emptyResponseCodes.contains($0) }
  123. }
  124. /// Determines whether `request` and `response` allow empty response bodies.
  125. ///
  126. /// - Parameters:
  127. /// - request: `URLRequest` to evaluate.
  128. /// - response: `HTTPURLResponse` to evaluate.
  129. ///
  130. /// - Returns: `true` if `request` or `response` allow empty bodies, `false` otherwise.
  131. public func emptyResponseAllowed(forRequest request: URLRequest?, response: HTTPURLResponse?) -> Bool {
  132. (requestAllowsEmptyResponseData(request) == true) || (responseAllowsEmptyResponseData(response) == true)
  133. }
  134. }
  135. /// By default, any serializer declared to conform to both types will get file serialization for free, as it just feeds
  136. /// the data read from disk into the data response serializer.
  137. extension DownloadResponseSerializerProtocol where Self: DataResponseSerializerProtocol {
  138. public func serializeDownload(request: URLRequest?, response: HTTPURLResponse?, fileURL: URL?, error: Error?) throws -> Self.SerializedObject {
  139. guard error == nil else { throw error! }
  140. guard let fileURL = fileURL else {
  141. throw AFError.responseSerializationFailed(reason: .inputFileNil)
  142. }
  143. let data: Data
  144. do {
  145. data = try Data(contentsOf: fileURL)
  146. } catch {
  147. throw AFError.responseSerializationFailed(reason: .inputFileReadFailed(at: fileURL))
  148. }
  149. do {
  150. return try serialize(request: request, response: response, data: data, error: error)
  151. } catch {
  152. throw error
  153. }
  154. }
  155. }
  156. // MARK: - Default
  157. extension DataRequest {
  158. /// Adds a handler to be called once the request has finished.
  159. ///
  160. /// - Parameters:
  161. /// - queue: The queue on which the completion handler is dispatched. `.main` by default.
  162. /// - completionHandler: The code to be executed once the request has finished.
  163. ///
  164. /// - Returns: The request.
  165. @discardableResult
  166. public func response(queue: DispatchQueue = .main, completionHandler: @escaping (AFDataResponse<Data?>) -> Void) -> Self {
  167. appendResponseSerializer {
  168. // Start work that should be on the serialization queue.
  169. let result = AFResult<Data?>(value: self.data, error: self.error)
  170. // End work that should be on the serialization queue.
  171. self.underlyingQueue.async {
  172. let response = DataResponse(request: self.request,
  173. response: self.response,
  174. data: self.data,
  175. metrics: self.metrics,
  176. serializationDuration: 0,
  177. result: result)
  178. self.eventMonitor?.request(self, didParseResponse: response)
  179. self.responseSerializerDidComplete { queue.async { completionHandler(response) } }
  180. }
  181. }
  182. return self
  183. }
  184. private func _response<Serializer: DataResponseSerializerProtocol>(queue: DispatchQueue = .main,
  185. responseSerializer: Serializer,
  186. completionHandler: @escaping (AFDataResponse<Serializer.SerializedObject>) -> Void)
  187. -> Self {
  188. appendResponseSerializer {
  189. // Start work that should be on the serialization queue.
  190. let start = ProcessInfo.processInfo.systemUptime
  191. let result: AFResult<Serializer.SerializedObject> = Result {
  192. try responseSerializer.serialize(request: self.request,
  193. response: self.response,
  194. data: self.data,
  195. error: self.error)
  196. }.mapError { error in
  197. error.asAFError(or: .responseSerializationFailed(reason: .customSerializationFailed(error: error)))
  198. }
  199. let end = ProcessInfo.processInfo.systemUptime
  200. // End work that should be on the serialization queue.
  201. self.underlyingQueue.async {
  202. let response = DataResponse(request: self.request,
  203. response: self.response,
  204. data: self.data,
  205. metrics: self.metrics,
  206. serializationDuration: end - start,
  207. result: result)
  208. self.eventMonitor?.request(self, didParseResponse: response)
  209. guard !self.isCancelled, let serializerError = result.failure, let delegate = self.delegate else {
  210. self.responseSerializerDidComplete { queue.async { completionHandler(response) } }
  211. return
  212. }
  213. delegate.retryResult(for: self, dueTo: serializerError) { retryResult in
  214. var didComplete: (() -> Void)?
  215. defer {
  216. if let didComplete = didComplete {
  217. self.responseSerializerDidComplete { queue.async { didComplete() } }
  218. }
  219. }
  220. switch retryResult {
  221. case .doNotRetry:
  222. didComplete = { completionHandler(response) }
  223. case let .doNotRetryWithError(retryError):
  224. let result: AFResult<Serializer.SerializedObject> = .failure(retryError.asAFError(orFailWith: "Received retryError was not already AFError"))
  225. let response = DataResponse(request: self.request,
  226. response: self.response,
  227. data: self.data,
  228. metrics: self.metrics,
  229. serializationDuration: end - start,
  230. result: result)
  231. didComplete = { completionHandler(response) }
  232. case .retry, .retryWithDelay:
  233. delegate.retryRequest(self, withDelay: retryResult.delay)
  234. }
  235. }
  236. }
  237. }
  238. return self
  239. }
  240. /// Adds a handler to be called once the request has finished.
  241. ///
  242. /// - Parameters:
  243. /// - queue: The queue on which the completion handler is dispatched. `.main` by default
  244. /// - responseSerializer: The response serializer responsible for serializing the request, response, and data.
  245. /// - completionHandler: The code to be executed once the request has finished.
  246. ///
  247. /// - Returns: The request.
  248. @discardableResult
  249. public func response<Serializer: DataResponseSerializerProtocol>(queue: DispatchQueue = .main,
  250. responseSerializer: Serializer,
  251. completionHandler: @escaping (AFDataResponse<Serializer.SerializedObject>) -> Void)
  252. -> Self {
  253. _response(queue: queue, responseSerializer: responseSerializer, completionHandler: completionHandler)
  254. }
  255. /// Adds a handler to be called once the request has finished.
  256. ///
  257. /// - Parameters:
  258. /// - queue: The queue on which the completion handler is dispatched. `.main` by default
  259. /// - responseSerializer: The response serializer responsible for serializing the request, response, and data.
  260. /// - completionHandler: The code to be executed once the request has finished.
  261. ///
  262. /// - Returns: The request.
  263. @discardableResult
  264. public func response<Serializer: ResponseSerializer>(queue: DispatchQueue = .main,
  265. responseSerializer: Serializer,
  266. completionHandler: @escaping (AFDataResponse<Serializer.SerializedObject>) -> Void)
  267. -> Self {
  268. _response(queue: queue, responseSerializer: responseSerializer, completionHandler: completionHandler)
  269. }
  270. }
  271. extension DownloadRequest {
  272. /// Adds a handler to be called once the request has finished.
  273. ///
  274. /// - Parameters:
  275. /// - queue: The queue on which the completion handler is dispatched. `.main` by default.
  276. /// - completionHandler: The code to be executed once the request has finished.
  277. ///
  278. /// - Returns: The request.
  279. @discardableResult
  280. public func response(queue: DispatchQueue = .main,
  281. completionHandler: @escaping (AFDownloadResponse<URL?>) -> Void)
  282. -> Self {
  283. appendResponseSerializer {
  284. // Start work that should be on the serialization queue.
  285. let result = AFResult<URL?>(value: self.fileURL, error: self.error)
  286. // End work that should be on the serialization queue.
  287. self.underlyingQueue.async {
  288. let response = DownloadResponse(request: self.request,
  289. response: self.response,
  290. fileURL: self.fileURL,
  291. resumeData: self.resumeData,
  292. metrics: self.metrics,
  293. serializationDuration: 0,
  294. result: result)
  295. self.eventMonitor?.request(self, didParseResponse: response)
  296. self.responseSerializerDidComplete { queue.async { completionHandler(response) } }
  297. }
  298. }
  299. return self
  300. }
  301. private func _response<Serializer: DownloadResponseSerializerProtocol>(queue: DispatchQueue = .main,
  302. responseSerializer: Serializer,
  303. completionHandler: @escaping (AFDownloadResponse<Serializer.SerializedObject>) -> Void)
  304. -> Self {
  305. appendResponseSerializer {
  306. // Start work that should be on the serialization queue.
  307. let start = ProcessInfo.processInfo.systemUptime
  308. let result: AFResult<Serializer.SerializedObject> = Result {
  309. try responseSerializer.serializeDownload(request: self.request,
  310. response: self.response,
  311. fileURL: self.fileURL,
  312. error: self.error)
  313. }.mapError { error in
  314. error.asAFError(or: .responseSerializationFailed(reason: .customSerializationFailed(error: error)))
  315. }
  316. let end = ProcessInfo.processInfo.systemUptime
  317. // End work that should be on the serialization queue.
  318. self.underlyingQueue.async {
  319. let response = DownloadResponse(request: self.request,
  320. response: self.response,
  321. fileURL: self.fileURL,
  322. resumeData: self.resumeData,
  323. metrics: self.metrics,
  324. serializationDuration: end - start,
  325. result: result)
  326. self.eventMonitor?.request(self, didParseResponse: response)
  327. guard let serializerError = result.failure, let delegate = self.delegate else {
  328. self.responseSerializerDidComplete { queue.async { completionHandler(response) } }
  329. return
  330. }
  331. delegate.retryResult(for: self, dueTo: serializerError) { retryResult in
  332. var didComplete: (() -> Void)?
  333. defer {
  334. if let didComplete = didComplete {
  335. self.responseSerializerDidComplete { queue.async { didComplete() } }
  336. }
  337. }
  338. switch retryResult {
  339. case .doNotRetry:
  340. didComplete = { completionHandler(response) }
  341. case let .doNotRetryWithError(retryError):
  342. let result: AFResult<Serializer.SerializedObject> = .failure(retryError.asAFError(orFailWith: "Received retryError was not already AFError"))
  343. let response = DownloadResponse(request: self.request,
  344. response: self.response,
  345. fileURL: self.fileURL,
  346. resumeData: self.resumeData,
  347. metrics: self.metrics,
  348. serializationDuration: end - start,
  349. result: result)
  350. didComplete = { completionHandler(response) }
  351. case .retry, .retryWithDelay:
  352. delegate.retryRequest(self, withDelay: retryResult.delay)
  353. }
  354. }
  355. }
  356. }
  357. return self
  358. }
  359. /// Adds a handler to be called once the request has finished.
  360. ///
  361. /// - Parameters:
  362. /// - queue: The queue on which the completion handler is dispatched. `.main` by default.
  363. /// - responseSerializer: The response serializer responsible for serializing the request, response, and data
  364. /// contained in the destination `URL`.
  365. /// - completionHandler: The code to be executed once the request has finished.
  366. ///
  367. /// - Returns: The request.
  368. @discardableResult
  369. public func response<Serializer: DownloadResponseSerializerProtocol>(queue: DispatchQueue = .main,
  370. responseSerializer: Serializer,
  371. completionHandler: @escaping (AFDownloadResponse<Serializer.SerializedObject>) -> Void)
  372. -> Self {
  373. _response(queue: queue, responseSerializer: responseSerializer, completionHandler: completionHandler)
  374. }
  375. /// Adds a handler to be called once the request has finished.
  376. ///
  377. /// - Parameters:
  378. /// - queue: The queue on which the completion handler is dispatched. `.main` by default.
  379. /// - responseSerializer: The response serializer responsible for serializing the request, response, and data
  380. /// contained in the destination `URL`.
  381. /// - completionHandler: The code to be executed once the request has finished.
  382. ///
  383. /// - Returns: The request.
  384. @discardableResult
  385. public func response<Serializer: ResponseSerializer>(queue: DispatchQueue = .main,
  386. responseSerializer: Serializer,
  387. completionHandler: @escaping (AFDownloadResponse<Serializer.SerializedObject>) -> Void)
  388. -> Self {
  389. _response(queue: queue, responseSerializer: responseSerializer, completionHandler: completionHandler)
  390. }
  391. }
  392. // MARK: - URL
  393. /// A `DownloadResponseSerializerProtocol` that performs only `Error` checking and ensures that a downloaded `fileURL`
  394. /// is present.
  395. public struct URLResponseSerializer: DownloadResponseSerializerProtocol {
  396. /// Creates an instance.
  397. public init() {}
  398. public func serializeDownload(request: URLRequest?,
  399. response: HTTPURLResponse?,
  400. fileURL: URL?,
  401. error: Error?) throws -> URL {
  402. guard error == nil else { throw error! }
  403. guard let url = fileURL else {
  404. throw AFError.responseSerializationFailed(reason: .inputFileNil)
  405. }
  406. return url
  407. }
  408. }
  409. #if swift(>=5.5)
  410. extension DownloadResponseSerializerProtocol where Self == URLResponseSerializer {
  411. /// Provides a `URLResponseSerializer` instance.
  412. public static var url: URLResponseSerializer { URLResponseSerializer() }
  413. }
  414. #endif
  415. extension DownloadRequest {
  416. /// Adds a handler using a `URLResponseSerializer` to be called once the request is finished.
  417. ///
  418. /// - Parameters:
  419. /// - queue: The queue on which the completion handler is called. `.main` by default.
  420. /// - completionHandler: A closure to be executed once the request has finished.
  421. ///
  422. /// - Returns: The request.
  423. @discardableResult
  424. public func responseURL(queue: DispatchQueue = .main,
  425. completionHandler: @escaping (AFDownloadResponse<URL>) -> Void) -> Self {
  426. response(queue: queue, responseSerializer: URLResponseSerializer(), completionHandler: completionHandler)
  427. }
  428. }
  429. // MARK: - Data
  430. /// A `ResponseSerializer` that performs minimal response checking and returns any response `Data` as-is. By default, a
  431. /// request returning `nil` or no data is considered an error. However, if the request has an `HTTPMethod` or the
  432. /// response has an HTTP status code valid for empty responses, then an empty `Data` value is returned.
  433. public final class DataResponseSerializer: ResponseSerializer {
  434. public let dataPreprocessor: DataPreprocessor
  435. public let emptyResponseCodes: Set<Int>
  436. public let emptyRequestMethods: Set<HTTPMethod>
  437. /// Creates a `DataResponseSerializer` using the provided parameters.
  438. ///
  439. /// - Parameters:
  440. /// - dataPreprocessor: `DataPreprocessor` used to prepare the received `Data` for serialization.
  441. /// - emptyResponseCodes: The HTTP response codes for which empty responses are allowed. `[204, 205]` by default.
  442. /// - emptyRequestMethods: The HTTP request methods for which empty responses are allowed. `[.head]` by default.
  443. public init(dataPreprocessor: DataPreprocessor = DataResponseSerializer.defaultDataPreprocessor,
  444. emptyResponseCodes: Set<Int> = DataResponseSerializer.defaultEmptyResponseCodes,
  445. emptyRequestMethods: Set<HTTPMethod> = DataResponseSerializer.defaultEmptyRequestMethods) {
  446. self.dataPreprocessor = dataPreprocessor
  447. self.emptyResponseCodes = emptyResponseCodes
  448. self.emptyRequestMethods = emptyRequestMethods
  449. }
  450. public func serialize(request: URLRequest?, response: HTTPURLResponse?, data: Data?, error: Error?) throws -> Data {
  451. guard error == nil else { throw error! }
  452. guard var data = data, !data.isEmpty else {
  453. guard emptyResponseAllowed(forRequest: request, response: response) else {
  454. throw AFError.responseSerializationFailed(reason: .inputDataNilOrZeroLength)
  455. }
  456. return Data()
  457. }
  458. data = try dataPreprocessor.preprocess(data)
  459. return data
  460. }
  461. }
  462. #if swift(>=5.5)
  463. extension ResponseSerializer where Self == DataResponseSerializer {
  464. /// Provides a default `DataResponseSerializer` instance.
  465. public static var data: DataResponseSerializer { DataResponseSerializer() }
  466. /// Creates a `DataResponseSerializer` using the provided parameters.
  467. ///
  468. /// - Parameters:
  469. /// - dataPreprocessor: `DataPreprocessor` used to prepare the received `Data` for serialization.
  470. /// - emptyResponseCodes: The HTTP response codes for which empty responses are allowed. `[204, 205]` by default.
  471. /// - emptyRequestMethods: The HTTP request methods for which empty responses are allowed. `[.head]` by default.
  472. ///
  473. /// - Returns: The `DataResponseSerializer`.
  474. public static func data(dataPreprocessor: DataPreprocessor = DataResponseSerializer.defaultDataPreprocessor,
  475. emptyResponseCodes: Set<Int> = DataResponseSerializer.defaultEmptyResponseCodes,
  476. emptyRequestMethods: Set<HTTPMethod> = DataResponseSerializer.defaultEmptyRequestMethods) -> DataResponseSerializer {
  477. DataResponseSerializer(dataPreprocessor: dataPreprocessor,
  478. emptyResponseCodes: emptyResponseCodes,
  479. emptyRequestMethods: emptyRequestMethods)
  480. }
  481. }
  482. #endif
  483. extension DataRequest {
  484. /// Adds a handler using a `DataResponseSerializer` to be called once the request has finished.
  485. ///
  486. /// - Parameters:
  487. /// - queue: The queue on which the completion handler is called. `.main` by default.
  488. /// - dataPreprocessor: `DataPreprocessor` which processes the received `Data` before calling the
  489. /// `completionHandler`. `PassthroughPreprocessor()` by default.
  490. /// - emptyResponseCodes: HTTP status codes for which empty responses are always valid. `[204, 205]` by default.
  491. /// - emptyRequestMethods: `HTTPMethod`s for which empty responses are always valid. `[.head]` by default.
  492. /// - completionHandler: A closure to be executed once the request has finished.
  493. ///
  494. /// - Returns: The request.
  495. @discardableResult
  496. public func responseData(queue: DispatchQueue = .main,
  497. dataPreprocessor: DataPreprocessor = DataResponseSerializer.defaultDataPreprocessor,
  498. emptyResponseCodes: Set<Int> = DataResponseSerializer.defaultEmptyResponseCodes,
  499. emptyRequestMethods: Set<HTTPMethod> = DataResponseSerializer.defaultEmptyRequestMethods,
  500. completionHandler: @escaping (AFDataResponse<Data>) -> Void) -> Self {
  501. response(queue: queue,
  502. responseSerializer: DataResponseSerializer(dataPreprocessor: dataPreprocessor,
  503. emptyResponseCodes: emptyResponseCodes,
  504. emptyRequestMethods: emptyRequestMethods),
  505. completionHandler: completionHandler)
  506. }
  507. }
  508. extension DownloadRequest {
  509. /// Adds a handler using a `DataResponseSerializer` to be called once the request has finished.
  510. ///
  511. /// - Parameters:
  512. /// - queue: The queue on which the completion handler is called. `.main` by default.
  513. /// - dataPreprocessor: `DataPreprocessor` which processes the received `Data` before calling the
  514. /// `completionHandler`. `PassthroughPreprocessor()` by default.
  515. /// - emptyResponseCodes: HTTP status codes for which empty responses are always valid. `[204, 205]` by default.
  516. /// - emptyRequestMethods: `HTTPMethod`s for which empty responses are always valid. `[.head]` by default.
  517. /// - completionHandler: A closure to be executed once the request has finished.
  518. ///
  519. /// - Returns: The request.
  520. @discardableResult
  521. public func responseData(queue: DispatchQueue = .main,
  522. dataPreprocessor: DataPreprocessor = DataResponseSerializer.defaultDataPreprocessor,
  523. emptyResponseCodes: Set<Int> = DataResponseSerializer.defaultEmptyResponseCodes,
  524. emptyRequestMethods: Set<HTTPMethod> = DataResponseSerializer.defaultEmptyRequestMethods,
  525. completionHandler: @escaping (AFDownloadResponse<Data>) -> Void) -> Self {
  526. response(queue: queue,
  527. responseSerializer: DataResponseSerializer(dataPreprocessor: dataPreprocessor,
  528. emptyResponseCodes: emptyResponseCodes,
  529. emptyRequestMethods: emptyRequestMethods),
  530. completionHandler: completionHandler)
  531. }
  532. }
  533. // MARK: - String
  534. /// A `ResponseSerializer` that decodes the response data as a `String`. By default, a request returning `nil` or no
  535. /// data is considered an error. However, if the request has an `HTTPMethod` or the response has an HTTP status code
  536. /// valid for empty responses, then an empty `String` is returned.
  537. public final class StringResponseSerializer: ResponseSerializer {
  538. public let dataPreprocessor: DataPreprocessor
  539. /// Optional string encoding used to validate the response.
  540. public let encoding: String.Encoding?
  541. public let emptyResponseCodes: Set<Int>
  542. public let emptyRequestMethods: Set<HTTPMethod>
  543. /// Creates an instance with the provided values.
  544. ///
  545. /// - Parameters:
  546. /// - dataPreprocessor: `DataPreprocessor` used to prepare the received `Data` for serialization.
  547. /// - encoding: A string encoding. Defaults to `nil`, in which case the encoding will be determined
  548. /// from the server response, falling back to the default HTTP character set, `ISO-8859-1`.
  549. /// - emptyResponseCodes: The HTTP response codes for which empty responses are allowed. `[204, 205]` by default.
  550. /// - emptyRequestMethods: The HTTP request methods for which empty responses are allowed. `[.head]` by default.
  551. public init(dataPreprocessor: DataPreprocessor = StringResponseSerializer.defaultDataPreprocessor,
  552. encoding: String.Encoding? = nil,
  553. emptyResponseCodes: Set<Int> = StringResponseSerializer.defaultEmptyResponseCodes,
  554. emptyRequestMethods: Set<HTTPMethod> = StringResponseSerializer.defaultEmptyRequestMethods) {
  555. self.dataPreprocessor = dataPreprocessor
  556. self.encoding = encoding
  557. self.emptyResponseCodes = emptyResponseCodes
  558. self.emptyRequestMethods = emptyRequestMethods
  559. }
  560. public func serialize(request: URLRequest?, response: HTTPURLResponse?, data: Data?, error: Error?) throws -> String {
  561. guard error == nil else { throw error! }
  562. guard var data = data, !data.isEmpty else {
  563. guard emptyResponseAllowed(forRequest: request, response: response) else {
  564. throw AFError.responseSerializationFailed(reason: .inputDataNilOrZeroLength)
  565. }
  566. return ""
  567. }
  568. data = try dataPreprocessor.preprocess(data)
  569. var convertedEncoding = encoding
  570. if let encodingName = response?.textEncodingName, convertedEncoding == nil {
  571. convertedEncoding = String.Encoding(ianaCharsetName: encodingName)
  572. }
  573. let actualEncoding = convertedEncoding ?? .isoLatin1
  574. guard let string = String(data: data, encoding: actualEncoding) else {
  575. throw AFError.responseSerializationFailed(reason: .stringSerializationFailed(encoding: actualEncoding))
  576. }
  577. return string
  578. }
  579. }
  580. #if swift(>=5.5)
  581. extension ResponseSerializer where Self == StringResponseSerializer {
  582. /// Provides a default `StringResponseSerializer` instance.
  583. public static var string: StringResponseSerializer { StringResponseSerializer() }
  584. /// Creates a `StringResponseSerializer` with the provided values.
  585. ///
  586. /// - Parameters:
  587. /// - dataPreprocessor: `DataPreprocessor` used to prepare the received `Data` for serialization.
  588. /// - encoding: A string encoding. Defaults to `nil`, in which case the encoding will be determined
  589. /// from the server response, falling back to the default HTTP character set, `ISO-8859-1`.
  590. /// - emptyResponseCodes: The HTTP response codes for which empty responses are allowed. `[204, 205]` by default.
  591. /// - emptyRequestMethods: The HTTP request methods for which empty responses are allowed. `[.head]` by default.
  592. ///
  593. /// - Returns: The `StringResponseSerializer`.
  594. public static func string(dataPreprocessor: DataPreprocessor = StringResponseSerializer.defaultDataPreprocessor,
  595. encoding: String.Encoding? = nil,
  596. emptyResponseCodes: Set<Int> = StringResponseSerializer.defaultEmptyResponseCodes,
  597. emptyRequestMethods: Set<HTTPMethod> = StringResponseSerializer.defaultEmptyRequestMethods) -> StringResponseSerializer {
  598. StringResponseSerializer(dataPreprocessor: dataPreprocessor,
  599. encoding: encoding,
  600. emptyResponseCodes: emptyResponseCodes,
  601. emptyRequestMethods: emptyRequestMethods)
  602. }
  603. }
  604. #endif
  605. extension DataRequest {
  606. /// Adds a handler using a `StringResponseSerializer` to be called once the request has finished.
  607. ///
  608. /// - Parameters:
  609. /// - queue: The queue on which the completion handler is dispatched. `.main` by default.
  610. /// - dataPreprocessor: `DataPreprocessor` which processes the received `Data` before calling the
  611. /// `completionHandler`. `PassthroughPreprocessor()` by default.
  612. /// - encoding: The string encoding. Defaults to `nil`, in which case the encoding will be determined
  613. /// from the server response, falling back to the default HTTP character set, `ISO-8859-1`.
  614. /// - emptyResponseCodes: HTTP status codes for which empty responses are always valid. `[204, 205]` by default.
  615. /// - emptyRequestMethods: `HTTPMethod`s for which empty responses are always valid. `[.head]` by default.
  616. /// - completionHandler: A closure to be executed once the request has finished.
  617. ///
  618. /// - Returns: The request.
  619. @discardableResult
  620. public func responseString(queue: DispatchQueue = .main,
  621. dataPreprocessor: DataPreprocessor = StringResponseSerializer.defaultDataPreprocessor,
  622. encoding: String.Encoding? = nil,
  623. emptyResponseCodes: Set<Int> = StringResponseSerializer.defaultEmptyResponseCodes,
  624. emptyRequestMethods: Set<HTTPMethod> = StringResponseSerializer.defaultEmptyRequestMethods,
  625. completionHandler: @escaping (AFDataResponse<String>) -> Void) -> Self {
  626. response(queue: queue,
  627. responseSerializer: StringResponseSerializer(dataPreprocessor: dataPreprocessor,
  628. encoding: encoding,
  629. emptyResponseCodes: emptyResponseCodes,
  630. emptyRequestMethods: emptyRequestMethods),
  631. completionHandler: completionHandler)
  632. }
  633. }
  634. extension DownloadRequest {
  635. /// Adds a handler using a `StringResponseSerializer` to be called once the request has finished.
  636. ///
  637. /// - Parameters:
  638. /// - queue: The queue on which the completion handler is dispatched. `.main` by default.
  639. /// - dataPreprocessor: `DataPreprocessor` which processes the received `Data` before calling the
  640. /// `completionHandler`. `PassthroughPreprocessor()` by default.
  641. /// - encoding: The string encoding. Defaults to `nil`, in which case the encoding will be determined
  642. /// from the server response, falling back to the default HTTP character set, `ISO-8859-1`.
  643. /// - emptyResponseCodes: HTTP status codes for which empty responses are always valid. `[204, 205]` by default.
  644. /// - emptyRequestMethods: `HTTPMethod`s for which empty responses are always valid. `[.head]` by default.
  645. /// - completionHandler: A closure to be executed once the request has finished.
  646. ///
  647. /// - Returns: The request.
  648. @discardableResult
  649. public func responseString(queue: DispatchQueue = .main,
  650. dataPreprocessor: DataPreprocessor = StringResponseSerializer.defaultDataPreprocessor,
  651. encoding: String.Encoding? = nil,
  652. emptyResponseCodes: Set<Int> = StringResponseSerializer.defaultEmptyResponseCodes,
  653. emptyRequestMethods: Set<HTTPMethod> = StringResponseSerializer.defaultEmptyRequestMethods,
  654. completionHandler: @escaping (AFDownloadResponse<String>) -> Void) -> Self {
  655. response(queue: queue,
  656. responseSerializer: StringResponseSerializer(dataPreprocessor: dataPreprocessor,
  657. encoding: encoding,
  658. emptyResponseCodes: emptyResponseCodes,
  659. emptyRequestMethods: emptyRequestMethods),
  660. completionHandler: completionHandler)
  661. }
  662. }
  663. // MARK: - JSON
  664. /// A `ResponseSerializer` that decodes the response data using `JSONSerialization`. By default, a request returning
  665. /// `nil` or no data is considered an error. However, if the request has an `HTTPMethod` or the response has an
  666. /// HTTP status code valid for empty responses, then an `NSNull` value is returned.
  667. @available(*, deprecated, message: "JSONResponseSerializer deprecated and will be removed in Alamofire 6. Use DecodableResponseSerializer instead.")
  668. public final class JSONResponseSerializer: ResponseSerializer {
  669. public let dataPreprocessor: DataPreprocessor
  670. public let emptyResponseCodes: Set<Int>
  671. public let emptyRequestMethods: Set<HTTPMethod>
  672. /// `JSONSerialization.ReadingOptions` used when serializing a response.
  673. public let options: JSONSerialization.ReadingOptions
  674. /// Creates an instance with the provided values.
  675. ///
  676. /// - Parameters:
  677. /// - dataPreprocessor: `DataPreprocessor` used to prepare the received `Data` for serialization.
  678. /// - emptyResponseCodes: The HTTP response codes for which empty responses are allowed. `[204, 205]` by default.
  679. /// - emptyRequestMethods: The HTTP request methods for which empty responses are allowed. `[.head]` by default.
  680. /// - options: The options to use. `.allowFragments` by default.
  681. public init(dataPreprocessor: DataPreprocessor = JSONResponseSerializer.defaultDataPreprocessor,
  682. emptyResponseCodes: Set<Int> = JSONResponseSerializer.defaultEmptyResponseCodes,
  683. emptyRequestMethods: Set<HTTPMethod> = JSONResponseSerializer.defaultEmptyRequestMethods,
  684. options: JSONSerialization.ReadingOptions = .allowFragments) {
  685. self.dataPreprocessor = dataPreprocessor
  686. self.emptyResponseCodes = emptyResponseCodes
  687. self.emptyRequestMethods = emptyRequestMethods
  688. self.options = options
  689. }
  690. public func serialize(request: URLRequest?, response: HTTPURLResponse?, data: Data?, error: Error?) throws -> Any {
  691. guard error == nil else { throw error! }
  692. guard var data = data, !data.isEmpty else {
  693. guard emptyResponseAllowed(forRequest: request, response: response) else {
  694. throw AFError.responseSerializationFailed(reason: .inputDataNilOrZeroLength)
  695. }
  696. return NSNull()
  697. }
  698. data = try dataPreprocessor.preprocess(data)
  699. do {
  700. return try JSONSerialization.jsonObject(with: data, options: options)
  701. } catch {
  702. throw AFError.responseSerializationFailed(reason: .jsonSerializationFailed(error: error))
  703. }
  704. }
  705. }
  706. extension DataRequest {
  707. /// Adds a handler using a `JSONResponseSerializer` to be called once the request has finished.
  708. ///
  709. /// - Parameters:
  710. /// - queue: The queue on which the completion handler is dispatched. `.main` by default.
  711. /// - dataPreprocessor: `DataPreprocessor` which processes the received `Data` before calling the
  712. /// `completionHandler`. `PassthroughPreprocessor()` by default.
  713. /// - emptyResponseCodes: HTTP status codes for which empty responses are always valid. `[204, 205]` by default.
  714. /// - emptyRequestMethods: `HTTPMethod`s for which empty responses are always valid. `[.head]` by default.
  715. /// - options: `JSONSerialization.ReadingOptions` used when parsing the response. `.allowFragments`
  716. /// by default.
  717. /// - completionHandler: A closure to be executed once the request has finished.
  718. ///
  719. /// - Returns: The request.
  720. @available(*, deprecated, message: "responseJSON deprecated and will be removed in Alamofire 6. Use responseDecodable instead.")
  721. @discardableResult
  722. public func responseJSON(queue: DispatchQueue = .main,
  723. dataPreprocessor: DataPreprocessor = JSONResponseSerializer.defaultDataPreprocessor,
  724. emptyResponseCodes: Set<Int> = JSONResponseSerializer.defaultEmptyResponseCodes,
  725. emptyRequestMethods: Set<HTTPMethod> = JSONResponseSerializer.defaultEmptyRequestMethods,
  726. options: JSONSerialization.ReadingOptions = .allowFragments,
  727. completionHandler: @escaping (AFDataResponse<Any>) -> Void) -> Self {
  728. response(queue: queue,
  729. responseSerializer: JSONResponseSerializer(dataPreprocessor: dataPreprocessor,
  730. emptyResponseCodes: emptyResponseCodes,
  731. emptyRequestMethods: emptyRequestMethods,
  732. options: options),
  733. completionHandler: completionHandler)
  734. }
  735. }
  736. extension DownloadRequest {
  737. /// Adds a handler using a `JSONResponseSerializer` to be called once the request has finished.
  738. ///
  739. /// - Parameters:
  740. /// - queue: The queue on which the completion handler is dispatched. `.main` by default.
  741. /// - dataPreprocessor: `DataPreprocessor` which processes the received `Data` before calling the
  742. /// `completionHandler`. `PassthroughPreprocessor()` by default.
  743. /// - emptyResponseCodes: HTTP status codes for which empty responses are always valid. `[204, 205]` by default.
  744. /// - emptyRequestMethods: `HTTPMethod`s for which empty responses are always valid. `[.head]` by default.
  745. /// - options: `JSONSerialization.ReadingOptions` used when parsing the response. `.allowFragments`
  746. /// by default.
  747. /// - completionHandler: A closure to be executed once the request has finished.
  748. ///
  749. /// - Returns: The request.
  750. @available(*, deprecated, message: "responseJSON deprecated and will be removed in Alamofire 6. Use responseDecodable instead.")
  751. @discardableResult
  752. public func responseJSON(queue: DispatchQueue = .main,
  753. dataPreprocessor: DataPreprocessor = JSONResponseSerializer.defaultDataPreprocessor,
  754. emptyResponseCodes: Set<Int> = JSONResponseSerializer.defaultEmptyResponseCodes,
  755. emptyRequestMethods: Set<HTTPMethod> = JSONResponseSerializer.defaultEmptyRequestMethods,
  756. options: JSONSerialization.ReadingOptions = .allowFragments,
  757. completionHandler: @escaping (AFDownloadResponse<Any>) -> Void) -> Self {
  758. response(queue: queue,
  759. responseSerializer: JSONResponseSerializer(dataPreprocessor: dataPreprocessor,
  760. emptyResponseCodes: emptyResponseCodes,
  761. emptyRequestMethods: emptyRequestMethods,
  762. options: options),
  763. completionHandler: completionHandler)
  764. }
  765. }
  766. // MARK: - Empty
  767. /// Protocol representing an empty response. Use `T.emptyValue()` to get an instance.
  768. public protocol EmptyResponse {
  769. /// Empty value for the conforming type.
  770. ///
  771. /// - Returns: Value of `Self` to use for empty values.
  772. static func emptyValue() -> Self
  773. }
  774. /// Type representing an empty value. Use `Empty.value` to get the static instance.
  775. public struct Empty: Codable {
  776. /// Static `Empty` instance used for all `Empty` responses.
  777. public static let value = Empty()
  778. }
  779. extension Empty: EmptyResponse {
  780. public static func emptyValue() -> Empty {
  781. value
  782. }
  783. }
  784. // MARK: - DataDecoder Protocol
  785. /// Any type which can decode `Data` into a `Decodable` type.
  786. public protocol DataDecoder {
  787. /// Decode `Data` into the provided type.
  788. ///
  789. /// - Parameters:
  790. /// - type: The `Type` to be decoded.
  791. /// - data: The `Data` to be decoded.
  792. ///
  793. /// - Returns: The decoded value of type `D`.
  794. /// - Throws: Any error that occurs during decode.
  795. func decode<D: Decodable>(_ type: D.Type, from data: Data) throws -> D
  796. }
  797. /// `JSONDecoder` automatically conforms to `DataDecoder`.
  798. extension JSONDecoder: DataDecoder {}
  799. /// `PropertyListDecoder` automatically conforms to `DataDecoder`.
  800. extension PropertyListDecoder: DataDecoder {}
  801. // MARK: - Decodable
  802. /// A `ResponseSerializer` that decodes the response data as a generic value using any type that conforms to
  803. /// `DataDecoder`. By default, this is an instance of `JSONDecoder`. Additionally, a request returning `nil` or no data
  804. /// is considered an error. However, if the request has an `HTTPMethod` or the response has an HTTP status code valid
  805. /// for empty responses then an empty value will be returned. If the decoded type conforms to `EmptyResponse`, the
  806. /// type's `emptyValue()` will be returned. If the decoded type is `Empty`, the `.value` instance is returned. If the
  807. /// decoded type *does not* conform to `EmptyResponse` and isn't `Empty`, an error will be produced.
  808. public final class DecodableResponseSerializer<T: Decodable>: ResponseSerializer {
  809. public let dataPreprocessor: DataPreprocessor
  810. /// The `DataDecoder` instance used to decode responses.
  811. public let decoder: DataDecoder
  812. public let emptyResponseCodes: Set<Int>
  813. public let emptyRequestMethods: Set<HTTPMethod>
  814. /// Creates an instance using the values provided.
  815. ///
  816. /// - Parameters:
  817. /// - dataPreprocessor: `DataPreprocessor` used to prepare the received `Data` for serialization.
  818. /// - decoder: The `DataDecoder`. `JSONDecoder()` by default.
  819. /// - emptyResponseCodes: The HTTP response codes for which empty responses are allowed. `[204, 205]` by default.
  820. /// - emptyRequestMethods: The HTTP request methods for which empty responses are allowed. `[.head]` by default.
  821. public init(dataPreprocessor: DataPreprocessor = DecodableResponseSerializer.defaultDataPreprocessor,
  822. decoder: DataDecoder = JSONDecoder(),
  823. emptyResponseCodes: Set<Int> = DecodableResponseSerializer.defaultEmptyResponseCodes,
  824. emptyRequestMethods: Set<HTTPMethod> = DecodableResponseSerializer.defaultEmptyRequestMethods) {
  825. self.dataPreprocessor = dataPreprocessor
  826. self.decoder = decoder
  827. self.emptyResponseCodes = emptyResponseCodes
  828. self.emptyRequestMethods = emptyRequestMethods
  829. }
  830. public func serialize(request: URLRequest?, response: HTTPURLResponse?, data: Data?, error: Error?) throws -> T {
  831. guard error == nil else { throw error! }
  832. guard var data = data, !data.isEmpty else {
  833. guard emptyResponseAllowed(forRequest: request, response: response) else {
  834. throw AFError.responseSerializationFailed(reason: .inputDataNilOrZeroLength)
  835. }
  836. guard let emptyResponseType = T.self as? EmptyResponse.Type, let emptyValue = emptyResponseType.emptyValue() as? T else {
  837. throw AFError.responseSerializationFailed(reason: .invalidEmptyResponse(type: "\(T.self)"))
  838. }
  839. return emptyValue
  840. }
  841. data = try dataPreprocessor.preprocess(data)
  842. do {
  843. return try decoder.decode(T.self, from: data)
  844. } catch {
  845. throw AFError.responseSerializationFailed(reason: .decodingFailed(error: error))
  846. }
  847. }
  848. }
  849. #if swift(>=5.5)
  850. extension ResponseSerializer {
  851. /// Creates a `DecodableResponseSerializer` using the values provided.
  852. ///
  853. /// - Parameters:
  854. /// - type: `Decodable` type to decode from response data.
  855. /// - dataPreprocessor: `DataPreprocessor` used to prepare the received `Data` for serialization.
  856. /// - decoder: The `DataDecoder`. `JSONDecoder()` by default.
  857. /// - emptyResponseCodes: The HTTP response codes for which empty responses are allowed. `[204, 205]` by default.
  858. /// - emptyRequestMethods: The HTTP request methods for which empty responses are allowed. `[.head]` by default.
  859. ///
  860. /// - Returns: The `DecodableResponseSerializer`.
  861. public static func decodable<T: Decodable>(of type: T.Type,
  862. dataPreprocessor: DataPreprocessor = DecodableResponseSerializer<T>.defaultDataPreprocessor,
  863. decoder: DataDecoder = JSONDecoder(),
  864. emptyResponseCodes: Set<Int> = DecodableResponseSerializer<T>.defaultEmptyResponseCodes,
  865. emptyRequestMethods: Set<HTTPMethod> = DecodableResponseSerializer<T>.defaultEmptyRequestMethods) -> DecodableResponseSerializer<T> where Self == DecodableResponseSerializer<T> {
  866. DecodableResponseSerializer<T>(dataPreprocessor: dataPreprocessor,
  867. decoder: decoder,
  868. emptyResponseCodes: emptyResponseCodes,
  869. emptyRequestMethods: emptyRequestMethods)
  870. }
  871. }
  872. #endif
  873. extension DataRequest {
  874. /// Adds a handler using a `DecodableResponseSerializer` to be called once the request has finished.
  875. ///
  876. /// - Parameters:
  877. /// - type: `Decodable` type to decode from response data.
  878. /// - queue: The queue on which the completion handler is dispatched. `.main` by default.
  879. /// - dataPreprocessor: `DataPreprocessor` which processes the received `Data` before calling the
  880. /// `completionHandler`. `PassthroughPreprocessor()` by default.
  881. /// - decoder: `DataDecoder` to use to decode the response. `JSONDecoder()` by default.
  882. /// - emptyResponseCodes: HTTP status codes for which empty responses are always valid. `[204, 205]` by default.
  883. /// - emptyRequestMethods: `HTTPMethod`s for which empty responses are always valid. `[.head]` by default.
  884. /// - completionHandler: A closure to be executed once the request has finished.
  885. ///
  886. /// - Returns: The request.
  887. @discardableResult
  888. public func responseDecodable<T: Decodable>(of type: T.Type = T.self,
  889. queue: DispatchQueue = .main,
  890. dataPreprocessor: DataPreprocessor = DecodableResponseSerializer<T>.defaultDataPreprocessor,
  891. decoder: DataDecoder = JSONDecoder(),
  892. emptyResponseCodes: Set<Int> = DecodableResponseSerializer<T>.defaultEmptyResponseCodes,
  893. emptyRequestMethods: Set<HTTPMethod> = DecodableResponseSerializer<T>.defaultEmptyRequestMethods,
  894. completionHandler: @escaping (AFDataResponse<T>) -> Void) -> Self {
  895. response(queue: queue,
  896. responseSerializer: DecodableResponseSerializer(dataPreprocessor: dataPreprocessor,
  897. decoder: decoder,
  898. emptyResponseCodes: emptyResponseCodes,
  899. emptyRequestMethods: emptyRequestMethods),
  900. completionHandler: completionHandler)
  901. }
  902. }
  903. extension DownloadRequest {
  904. /// Adds a handler using a `DecodableResponseSerializer` to be called once the request has finished.
  905. ///
  906. /// - Parameters:
  907. /// - type: `Decodable` type to decode from response data.
  908. /// - queue: The queue on which the completion handler is dispatched. `.main` by default.
  909. /// - dataPreprocessor: `DataPreprocessor` which processes the received `Data` before calling the
  910. /// `completionHandler`. `PassthroughPreprocessor()` by default.
  911. /// - decoder: `DataDecoder` to use to decode the response. `JSONDecoder()` by default.
  912. /// - emptyResponseCodes: HTTP status codes for which empty responses are always valid. `[204, 205]` by default.
  913. /// - emptyRequestMethods: `HTTPMethod`s for which empty responses are always valid. `[.head]` by default.
  914. /// - completionHandler: A closure to be executed once the request has finished.
  915. ///
  916. /// - Returns: The request.
  917. @discardableResult
  918. public func responseDecodable<T: Decodable>(of type: T.Type = T.self,
  919. queue: DispatchQueue = .main,
  920. dataPreprocessor: DataPreprocessor = DecodableResponseSerializer<T>.defaultDataPreprocessor,
  921. decoder: DataDecoder = JSONDecoder(),
  922. emptyResponseCodes: Set<Int> = DecodableResponseSerializer<T>.defaultEmptyResponseCodes,
  923. emptyRequestMethods: Set<HTTPMethod> = DecodableResponseSerializer<T>.defaultEmptyRequestMethods,
  924. completionHandler: @escaping (AFDownloadResponse<T>) -> Void) -> Self {
  925. response(queue: queue,
  926. responseSerializer: DecodableResponseSerializer(dataPreprocessor: dataPreprocessor,
  927. decoder: decoder,
  928. emptyResponseCodes: emptyResponseCodes,
  929. emptyRequestMethods: emptyRequestMethods),
  930. completionHandler: completionHandler)
  931. }
  932. }
  933. // MARK: - DataStreamRequest
  934. /// A type which can serialize incoming `Data`.
  935. public protocol DataStreamSerializer {
  936. /// Type produced from the serialized `Data`.
  937. associatedtype SerializedObject
  938. /// Serializes incoming `Data` into a `SerializedObject` value.
  939. ///
  940. /// - Parameter data: `Data` to be serialized.
  941. ///
  942. /// - Throws: Any error produced during serialization.
  943. func serialize(_ data: Data) throws -> SerializedObject
  944. }
  945. /// `DataStreamSerializer` which uses the provided `DataPreprocessor` and `DataDecoder` to serialize the incoming `Data`.
  946. public struct DecodableStreamSerializer<T: Decodable>: DataStreamSerializer {
  947. /// `DataDecoder` used to decode incoming `Data`.
  948. public let decoder: DataDecoder
  949. /// `DataPreprocessor` incoming `Data` is passed through before being passed to the `DataDecoder`.
  950. public let dataPreprocessor: DataPreprocessor
  951. /// Creates an instance with the provided `DataDecoder` and `DataPreprocessor`.
  952. /// - Parameters:
  953. /// - decoder: ` DataDecoder` used to decode incoming `Data`. `JSONDecoder()` by default.
  954. /// - dataPreprocessor: `DataPreprocessor` used to process incoming `Data` before it's passed through the
  955. /// `decoder`. `PassthroughPreprocessor()` by default.
  956. public init(decoder: DataDecoder = JSONDecoder(), dataPreprocessor: DataPreprocessor = PassthroughPreprocessor()) {
  957. self.decoder = decoder
  958. self.dataPreprocessor = dataPreprocessor
  959. }
  960. public func serialize(_ data: Data) throws -> T {
  961. let processedData = try dataPreprocessor.preprocess(data)
  962. do {
  963. return try decoder.decode(T.self, from: processedData)
  964. } catch {
  965. throw AFError.responseSerializationFailed(reason: .decodingFailed(error: error))
  966. }
  967. }
  968. }
  969. /// `DataStreamSerializer` which performs no serialization on incoming `Data`.
  970. public struct PassthroughStreamSerializer: DataStreamSerializer {
  971. /// Creates an instance.
  972. public init() {}
  973. public func serialize(_ data: Data) throws -> Data { data }
  974. }
  975. /// `DataStreamSerializer` which serializes incoming stream `Data` into `UTF8`-decoded `String` values.
  976. public struct StringStreamSerializer: DataStreamSerializer {
  977. /// Creates an instance.
  978. public init() {}
  979. public func serialize(_ data: Data) throws -> String {
  980. String(decoding: data, as: UTF8.self)
  981. }
  982. }
  983. #if swift(>=5.5)
  984. extension DataStreamSerializer {
  985. /// Creates a `DecodableStreamSerializer` instance with the provided `DataDecoder` and `DataPreprocessor`.
  986. ///
  987. /// - Parameters:
  988. /// - type: `Decodable` type to decode from stream data.
  989. /// - decoder: ` DataDecoder` used to decode incoming `Data`. `JSONDecoder()` by default.
  990. /// - dataPreprocessor: `DataPreprocessor` used to process incoming `Data` before it's passed through the
  991. /// `decoder`. `PassthroughPreprocessor()` by default.
  992. public static func decodable<T: Decodable>(of type: T.Type,
  993. decoder: DataDecoder = JSONDecoder(),
  994. dataPreprocessor: DataPreprocessor = PassthroughPreprocessor()) -> Self where Self == DecodableStreamSerializer<T> {
  995. DecodableStreamSerializer<T>(decoder: decoder, dataPreprocessor: dataPreprocessor)
  996. }
  997. }
  998. extension DataStreamSerializer where Self == PassthroughStreamSerializer {
  999. /// Provides a `PassthroughStreamSerializer` instance.
  1000. public static var passthrough: PassthroughStreamSerializer { PassthroughStreamSerializer() }
  1001. }
  1002. extension DataStreamSerializer where Self == StringStreamSerializer {
  1003. /// Provides a `StringStreamSerializer` instance.
  1004. public static var string: StringStreamSerializer { StringStreamSerializer() }
  1005. }
  1006. #endif
  1007. extension DataStreamRequest {
  1008. /// Adds a `StreamHandler` which performs no parsing on incoming `Data`.
  1009. ///
  1010. /// - Parameters:
  1011. /// - queue: `DispatchQueue` on which to perform `StreamHandler` closure.
  1012. /// - stream: `StreamHandler` closure called as `Data` is received. May be called multiple times.
  1013. ///
  1014. /// - Returns: The `DataStreamRequest`.
  1015. @discardableResult
  1016. public func responseStream(on queue: DispatchQueue = .main, stream: @escaping Handler<Data, Never>) -> Self {
  1017. let parser = { [unowned self] (data: Data) in
  1018. queue.async {
  1019. self.capturingError {
  1020. try stream(.init(event: .stream(.success(data)), token: .init(self)))
  1021. }
  1022. self.updateAndCompleteIfPossible()
  1023. }
  1024. }
  1025. $streamMutableState.write { $0.streams.append(parser) }
  1026. appendStreamCompletion(on: queue, stream: stream)
  1027. return self
  1028. }
  1029. /// Adds a `StreamHandler` which uses the provided `DataStreamSerializer` to process incoming `Data`.
  1030. ///
  1031. /// - Parameters:
  1032. /// - serializer: `DataStreamSerializer` used to process incoming `Data`. Its work is done on the `serializationQueue`.
  1033. /// - queue: `DispatchQueue` on which to perform `StreamHandler` closure.
  1034. /// - stream: `StreamHandler` closure called as `Data` is received. May be called multiple times.
  1035. ///
  1036. /// - Returns: The `DataStreamRequest`.
  1037. @discardableResult
  1038. public func responseStream<Serializer: DataStreamSerializer>(using serializer: Serializer,
  1039. on queue: DispatchQueue = .main,
  1040. stream: @escaping Handler<Serializer.SerializedObject, AFError>) -> Self {
  1041. let parser = { [unowned self] (data: Data) in
  1042. self.serializationQueue.async {
  1043. // Start work on serialization queue.
  1044. let result = Result { try serializer.serialize(data) }
  1045. .mapError { $0.asAFError(or: .responseSerializationFailed(reason: .customSerializationFailed(error: $0))) }
  1046. // End work on serialization queue.
  1047. self.underlyingQueue.async {
  1048. self.eventMonitor?.request(self, didParseStream: result)
  1049. if result.isFailure, self.automaticallyCancelOnStreamError {
  1050. self.cancel()
  1051. }
  1052. queue.async {
  1053. self.capturingError {
  1054. try stream(.init(event: .stream(result), token: .init(self)))
  1055. }
  1056. self.updateAndCompleteIfPossible()
  1057. }
  1058. }
  1059. }
  1060. }
  1061. $streamMutableState.write { $0.streams.append(parser) }
  1062. appendStreamCompletion(on: queue, stream: stream)
  1063. return self
  1064. }
  1065. /// Adds a `StreamHandler` which parses incoming `Data` as a UTF8 `String`.
  1066. ///
  1067. /// - Parameters:
  1068. /// - queue: `DispatchQueue` on which to perform `StreamHandler` closure.
  1069. /// - stream: `StreamHandler` closure called as `Data` is received. May be called multiple times.
  1070. ///
  1071. /// - Returns: The `DataStreamRequest`.
  1072. @discardableResult
  1073. public func responseStreamString(on queue: DispatchQueue = .main,
  1074. stream: @escaping Handler<String, Never>) -> Self {
  1075. let parser = { [unowned self] (data: Data) in
  1076. self.serializationQueue.async {
  1077. // Start work on serialization queue.
  1078. let string = String(decoding: data, as: UTF8.self)
  1079. // End work on serialization queue.
  1080. self.underlyingQueue.async {
  1081. self.eventMonitor?.request(self, didParseStream: .success(string))
  1082. queue.async {
  1083. self.capturingError {
  1084. try stream(.init(event: .stream(.success(string)), token: .init(self)))
  1085. }
  1086. self.updateAndCompleteIfPossible()
  1087. }
  1088. }
  1089. }
  1090. }
  1091. $streamMutableState.write { $0.streams.append(parser) }
  1092. appendStreamCompletion(on: queue, stream: stream)
  1093. return self
  1094. }
  1095. private func updateAndCompleteIfPossible() {
  1096. $streamMutableState.write { state in
  1097. state.numberOfExecutingStreams -= 1
  1098. guard state.numberOfExecutingStreams == 0, !state.enqueuedCompletionEvents.isEmpty else { return }
  1099. let completionEvents = state.enqueuedCompletionEvents
  1100. self.underlyingQueue.async { completionEvents.forEach { $0() } }
  1101. state.enqueuedCompletionEvents.removeAll()
  1102. }
  1103. }
  1104. /// Adds a `StreamHandler` which parses incoming `Data` using the provided `DataDecoder`.
  1105. ///
  1106. /// - Parameters:
  1107. /// - type: `Decodable` type to parse incoming `Data` into.
  1108. /// - queue: `DispatchQueue` on which to perform `StreamHandler` closure.
  1109. /// - decoder: `DataDecoder` used to decode the incoming `Data`.
  1110. /// - preprocessor: `DataPreprocessor` used to process the incoming `Data` before it's passed to the `decoder`.
  1111. /// - stream: `StreamHandler` closure called as `Data` is received. May be called multiple times.
  1112. ///
  1113. /// - Returns: The `DataStreamRequest`.
  1114. @discardableResult
  1115. public func responseStreamDecodable<T: Decodable>(of type: T.Type = T.self,
  1116. on queue: DispatchQueue = .main,
  1117. using decoder: DataDecoder = JSONDecoder(),
  1118. preprocessor: DataPreprocessor = PassthroughPreprocessor(),
  1119. stream: @escaping Handler<T, AFError>) -> Self {
  1120. responseStream(using: DecodableStreamSerializer<T>(decoder: decoder, dataPreprocessor: preprocessor),
  1121. stream: stream)
  1122. }
  1123. }