// // DownloadTests.swift // // Copyright (c) 2014-2018 Alamofire Software Foundation (http://alamofire.org/) // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to deal // in the Software without restriction, including without limitation the rights // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell // copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in // all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN // THE SOFTWARE. // import Alamofire import Foundation import XCTest final class DownloadInitializationTests: BaseTestCase { func testDownloadClassMethodWithMethodURLAndDestination() { // Given let endpoint = Endpoint.get let expectation = self.expectation(description: "download should complete") // When let request = AF.download(endpoint).response { _ in expectation.fulfill() } waitForExpectations(timeout: timeout) // Then XCTAssertNotNil(request.request) XCTAssertEqual(request.request?.httpMethod, "GET") XCTAssertEqual(request.request?.url, endpoint.url) XCTAssertNotNil(request.response) } func testDownloadClassMethodWithMethodURLHeadersAndDestination() { // Given let endpoint = Endpoint.get let headers: HTTPHeaders = ["Authorization": "123456"] let expectation = self.expectation(description: "download should complete") // When let request = AF.download(endpoint, headers: headers).response { _ in expectation.fulfill() } waitForExpectations(timeout: timeout) // Then XCTAssertNotNil(request.request) XCTAssertEqual(request.request?.httpMethod, "GET") XCTAssertEqual(request.request?.url, endpoint.url) XCTAssertEqual(request.request?.headers["Authorization"], "123456") XCTAssertNotNil(request.response) } } // MARK: - final class DownloadResponseTests: BaseTestCase { private var randomCachesFileURL: URL { testDirectoryURL.appendingPathComponent("\(UUID().uuidString).json") } func testDownloadRequest() { // Given let fileURL = randomCachesFileURL let numberOfLines = 10 let endpoint = Endpoint.stream(numberOfLines) let destination: DownloadRequest.Destination = { _, _ in (fileURL, []) } let expectation = self.expectation(description: "Download request should download data to file: \(endpoint.url.absoluteString)") var response: DownloadResponse? // When AF.download(endpoint, to: destination) .response { resp in response = resp expectation.fulfill() } waitForExpectations(timeout: timeout) // Then XCTAssertNotNil(response?.request) XCTAssertNotNil(response?.response) XCTAssertNotNil(response?.fileURL) XCTAssertNil(response?.resumeData) XCTAssertNil(response?.error) if let destinationURL = response?.fileURL { XCTAssertTrue(FileManager.default.fileExists(atPath: destinationURL.path)) if let data = try? Data(contentsOf: destinationURL) { XCTAssertGreaterThan(data.count, 0) } else { XCTFail("data should exist for contents of destinationURL") } } } func testDownloadRequestResponseURLProducesURL() throws { // Given let expectation = self.expectation(description: "Download request should download data") var response: DownloadResponse? // When AF.download(.get) .responseURL { resp in response = resp expectation.fulfill() } waitForExpectations(timeout: timeout, handler: nil) // Then XCTAssertNotNil(response?.request) XCTAssertNotNil(response?.response) XCTAssertNotNil(response?.fileURL) XCTAssertNil(response?.resumeData) XCTAssertNil(response?.error) let url = try XCTUnwrap(response?.value) XCTAssertTrue(FileManager.default.fileExists(atPath: url.path)) } func testCancelledDownloadRequest() { // Given let fileURL = randomCachesFileURL let numberOfLines = 10 let destination: DownloadRequest.Destination = { _, _ in (fileURL, []) } let expectation = self.expectation(description: "Cancelled download request should not download data to file") var response: DownloadResponse? // When AF.download(.stream(numberOfLines), to: destination) .response { resp in response = resp expectation.fulfill() } .cancel() waitForExpectations(timeout: timeout) // Then XCTAssertNil(response?.response) XCTAssertNil(response?.fileURL) XCTAssertNotNil(response?.error) XCTAssertEqual(response?.error?.isExplicitlyCancelledError, true) } func testDownloadRequestWithProgress() { // Given let randomBytes = 1 * 25 * 1024 let endpoint = Endpoint.bytes(randomBytes) let expectation = self.expectation(description: "Bytes download progress should be reported: \(endpoint.url)") var progressValues: [Double] = [] var response: DownloadResponse? // When AF.download(endpoint) .downloadProgress { progress in progressValues.append(progress.fractionCompleted) } .response { resp in response = resp expectation.fulfill() } waitForExpectations(timeout: timeout) // Then XCTAssertNotNil(response?.request) XCTAssertNotNil(response?.response) XCTAssertNotNil(response?.fileURL) XCTAssertNil(response?.resumeData) XCTAssertNil(response?.error) var previousProgress: Double = progressValues.first ?? 0.0 for progress in progressValues { XCTAssertGreaterThanOrEqual(progress, previousProgress) previousProgress = progress } if let lastProgressValue = progressValues.last { XCTAssertEqual(lastProgressValue, 1.0) } else { XCTFail("last item in progressValues should not be nil") } } func testDownloadRequestWithParameters() { // Given let fileURL = randomCachesFileURL let parameters = ["foo": "bar"] let destination: DownloadRequest.Destination = { _, _ in (fileURL, []) } let expectation = self.expectation(description: "Download request should download data to file") var response: DownloadResponse? // When AF.download(Endpoint.get, parameters: parameters, to: destination) .response { resp in response = resp expectation.fulfill() } waitForExpectations(timeout: timeout) // Then XCTAssertNotNil(response?.request) XCTAssertNotNil(response?.response) XCTAssertNotNil(response?.fileURL) XCTAssertNil(response?.resumeData) XCTAssertNil(response?.error) if let data = try? Data(contentsOf: fileURL), let jsonObject = try? JSONSerialization.jsonObject(with: data, options: []), let json = jsonObject as? [String: Any], let args = json["args"] as? [String: String] { XCTAssertEqual(args["foo"], "bar") } else { XCTFail("args parameter in JSON should not be nil") } } func testDownloadRequestWithHeaders() { // Given let fileURL = randomCachesFileURL let endpoint = Endpoint.get let headers: HTTPHeaders = ["Authorization": "123456"] let destination: DownloadRequest.Destination = { _, _ in (fileURL, []) } let expectation = self.expectation(description: "Download request should download data to file: \(endpoint.url)") var response: DownloadResponse? // When AF.download(endpoint, headers: headers, to: destination) .response { resp in response = resp expectation.fulfill() } waitForExpectations(timeout: timeout) // Then XCTAssertNotNil(response?.request) XCTAssertNotNil(response?.response) XCTAssertNotNil(response?.fileURL) XCTAssertNil(response?.resumeData) XCTAssertNil(response?.error) if let data = try? Data(contentsOf: fileURL), let jsonObject = try? JSONSerialization.jsonObject(with: data, options: []), let json = jsonObject as? [String: Any], let headers = json["headers"] as? [String: String] { XCTAssertEqual(headers["Authorization"], "123456") } else { XCTFail("headers parameter in JSON should not be nil") } } func testThatDownloadingFileAndMovingToDirectoryThatDoesNotExistThrowsError() { // Given let fileURL = testDirectoryURL.appendingPathComponent("some/random/folder/test_output.json") let expectation = self.expectation(description: "Download request should download data but fail to move file") var response: DownloadResponse? // When AF.download(.get, to: { _, _ in (fileURL, []) }) .response { resp in response = resp expectation.fulfill() } waitForExpectations(timeout: timeout) // Then XCTAssertNotNil(response?.request) XCTAssertNotNil(response?.response) XCTAssertNil(response?.fileURL) XCTAssertNil(response?.resumeData) XCTAssertNotNil(response?.error) XCTAssertEqual((response?.error?.underlyingError as? CocoaError)?.code, .fileNoSuchFile) } func testThatDownloadOptionsCanCreateIntermediateDirectoriesPriorToMovingFile() { // Given let fileURL = testDirectoryURL.appendingPathComponent("some/random/folder/test_output.json") let expectation = self.expectation(description: "Download request should download data to file: \(fileURL)") var response: DownloadResponse? // When AF.download(.get, to: { _, _ in (fileURL, [.createIntermediateDirectories]) }) .response { resp in response = resp expectation.fulfill() } waitForExpectations(timeout: timeout) // Then XCTAssertNotNil(response?.request) XCTAssertNotNil(response?.response) XCTAssertNotNil(response?.fileURL) XCTAssertNil(response?.resumeData) XCTAssertNil(response?.error) } func testThatDownloadingFileAndMovingToDestinationThatIsOccupiedThrowsError() throws { // Given let directoryURL = testDirectoryURL.appendingPathComponent("some/random/folder") let directoryCreated = FileManager.createDirectory(at: directoryURL) let fileURL = directoryURL.appendingPathComponent("test_output.json") try "random_data".write(to: fileURL, atomically: true, encoding: .utf8) let expectation = self.expectation(description: "Download should complete but fail to move file") var response: DownloadResponse? // When AF.download(.get, to: { _, _ in (fileURL, []) }) .response { resp in response = resp expectation.fulfill() } waitForExpectations(timeout: timeout) // Then XCTAssertTrue(directoryCreated) XCTAssertNotNil(response?.request) XCTAssertNotNil(response?.response) XCTAssertNil(response?.fileURL) XCTAssertNil(response?.resumeData) XCTAssertNotNil(response?.error) XCTAssertEqual((response?.error?.underlyingError as? CocoaError)?.code, .fileWriteFileExists) } func testThatDownloadOptionsCanRemovePreviousFilePriorToMovingFile() { // Given let directoryURL = testDirectoryURL.appendingPathComponent("some/random/folder") let directoryCreated = FileManager.createDirectory(at: directoryURL) let fileURL = directoryURL.appendingPathComponent("test_output.json") let expectation = self.expectation(description: "Download should complete and move file to URL: \(fileURL)") var response: DownloadResponse? // When AF.download(.get, to: { _, _ in (fileURL, [.removePreviousFile, .createIntermediateDirectories]) }) .response { resp in response = resp expectation.fulfill() } waitForExpectations(timeout: timeout) // Then XCTAssertTrue(directoryCreated) XCTAssertNotNil(response?.request) XCTAssertNotNil(response?.response) XCTAssertNotNil(response?.fileURL) XCTAssertNil(response?.resumeData) XCTAssertNil(response?.error) } } // MARK: - final class DownloadRequestEventsTestCase: BaseTestCase { func testThatDownloadRequestTriggersAllAppropriateLifetimeEvents() { // Given let eventMonitor = ClosureEventMonitor() let session = Session(eventMonitors: [eventMonitor]) let taskDidFinishCollecting = expectation(description: "taskDidFinishCollecting should fire") let didCreateInitialURLRequest = expectation(description: "didCreateInitialURLRequest should fire") let didCreateURLRequest = expectation(description: "didCreateURLRequest should fire") let didCreateTask = expectation(description: "didCreateTask should fire") let didGatherMetrics = expectation(description: "didGatherMetrics should fire") let didComplete = expectation(description: "didComplete should fire") let didWriteData = expectation(description: "didWriteData should fire") let didFinishDownloading = expectation(description: "didFinishDownloading should fire") let didFinishWithResult = expectation(description: "didFinishWithResult should fire") let didCreate = expectation(description: "didCreate should fire") let didFinish = expectation(description: "didFinish should fire") let didResume = expectation(description: "didResume should fire") let didResumeTask = expectation(description: "didResumeTask should fire") let didParseResponse = expectation(description: "didParseResponse should fire") let responseHandler = expectation(description: "responseHandler should fire") var wroteData = false eventMonitor.taskDidFinishCollectingMetrics = { _, _, _ in taskDidFinishCollecting.fulfill() } eventMonitor.requestDidCreateInitialURLRequest = { _, _ in didCreateInitialURLRequest.fulfill() } eventMonitor.requestDidCreateURLRequest = { _, _ in didCreateURLRequest.fulfill() } eventMonitor.requestDidCreateTask = { _, _ in didCreateTask.fulfill() } eventMonitor.requestDidGatherMetrics = { _, _ in didGatherMetrics.fulfill() } eventMonitor.requestDidCompleteTaskWithError = { _, _, _ in didComplete.fulfill() } eventMonitor.downloadTaskDidWriteData = { _, _, _, _, _ in guard !wroteData else { return } wroteData = true didWriteData.fulfill() } eventMonitor.downloadTaskDidFinishDownloadingToURL = { _, _, _ in didFinishDownloading.fulfill() } eventMonitor.requestDidFinishDownloadingUsingTaskWithResult = { _, _, _ in didFinishWithResult.fulfill() } eventMonitor.requestDidCreateDestinationURL = { _, _ in didCreate.fulfill() } eventMonitor.requestDidFinish = { _ in didFinish.fulfill() } eventMonitor.requestDidResume = { _ in didResume.fulfill() } eventMonitor.requestDidResumeTask = { _, _ in didResumeTask.fulfill() } eventMonitor.requestDidParseDownloadResponse = { _, _ in didParseResponse.fulfill() } // When let request = session.download(.get).response { _ in responseHandler.fulfill() } waitForExpectations(timeout: timeout) // Then XCTAssertEqual(request.state, .finished) } func testThatCancelledDownloadRequestTriggersAllAppropriateLifetimeEvents() { // Given let eventMonitor = ClosureEventMonitor() let session = Session(startRequestsImmediately: false, eventMonitors: [eventMonitor]) let taskDidFinishCollecting = expectation(description: "taskDidFinishCollecting should fire") let didCreateInitialURLRequest = expectation(description: "didCreateInitialURLRequest should fire") let didCreateURLRequest = expectation(description: "didCreateURLRequest should fire") let didCreateTask = expectation(description: "didCreateTask should fire") let didGatherMetrics = expectation(description: "didGatherMetrics should fire") let didComplete = expectation(description: "didComplete should fire") let didFinish = expectation(description: "didFinish should fire") let didResume = expectation(description: "didResume should fire") let didResumeTask = expectation(description: "didResumeTask should fire") let didParseResponse = expectation(description: "didParseResponse should fire") let didCancel = expectation(description: "didCancel should fire") let didCancelTask = expectation(description: "didCancelTask should fire") let responseHandler = expectation(description: "responseHandler should fire") eventMonitor.taskDidFinishCollectingMetrics = { _, _, _ in taskDidFinishCollecting.fulfill() } eventMonitor.requestDidCreateInitialURLRequest = { _, _ in didCreateInitialURLRequest.fulfill() } eventMonitor.requestDidCreateURLRequest = { _, _ in didCreateURLRequest.fulfill() } eventMonitor.requestDidCreateTask = { _, _ in didCreateTask.fulfill() } eventMonitor.requestDidGatherMetrics = { _, _ in didGatherMetrics.fulfill() } eventMonitor.requestDidCompleteTaskWithError = { _, _, _ in didComplete.fulfill() } eventMonitor.requestDidFinish = { _ in didFinish.fulfill() } eventMonitor.requestDidResume = { _ in didResume.fulfill() } eventMonitor.requestDidParseDownloadResponse = { _, _ in didParseResponse.fulfill() } eventMonitor.requestDidCancel = { _ in didCancel.fulfill() } eventMonitor.requestDidCancelTask = { _, _ in didCancelTask.fulfill() } // When let request = session.download(.delay(5)).response { _ in responseHandler.fulfill() } eventMonitor.requestDidResumeTask = { [unowned request] _, _ in request.cancel() didResumeTask.fulfill() } request.resume() waitForExpectations(timeout: timeout) // Then XCTAssertEqual(request.state, .cancelled) } } // MARK: - final class DownloadResumeDataTestCase: BaseTestCase { func testThatCancelledDownloadRequestDoesNotProduceResumeData() { // Given let expectation = self.expectation(description: "Download should be cancelled") var cancelled = false var response: DownloadResponse? // When let download = AF.download(.download()) download.downloadProgress { [unowned download] progress in guard !cancelled else { return } if progress.fractionCompleted > 0.1 { download.cancel() cancelled = true } } download.response { resp in response = resp expectation.fulfill() } waitForExpectations(timeout: timeout) // Then XCTAssertNotNil(response?.request) XCTAssertNotNil(response?.response) XCTAssertNil(response?.fileURL) XCTAssertNotNil(response?.error) XCTAssertNil(response?.resumeData) XCTAssertNil(download.resumeData) } func testThatDownloadRequestProducesResumeDataOnError() { // Given let expectation = self.expectation(description: "download complete") var response: DownloadResponse? // When let download = AF.download(.download(produceError: true)) download.response { resp in response = resp expectation.fulfill() } waitForExpectations(timeout: timeout) // Then XCTAssertNotNil(response?.request) XCTAssertNotNil(response?.response) XCTAssertNil(response?.fileURL) XCTAssertNotNil(response?.error) XCTAssertNotNil(response?.resumeData) XCTAssertNotNil(download.resumeData) #if !(os(Linux) || os(Windows)) XCTAssertNotNil(download.error?.downloadResumeData) XCTAssertEqual(download.error?.downloadResumeData, response?.resumeData) #endif XCTAssertEqual(response?.resumeData, download.resumeData) } func testThatCancelledDownloadResponseDataMatchesResumeData() { // Given let expectation = self.expectation(description: "Download should be cancelled") var cancelled = false var response: DownloadResponse? // When let download = AF.download(.download()) download.downloadProgress { [unowned download] progress in guard !cancelled else { return } if progress.fractionCompleted > 0.1 { download.cancel(producingResumeData: true) cancelled = true } } download.response { resp in response = resp expectation.fulfill() } waitForExpectations(timeout: timeout) // Then XCTAssertNotNil(response?.request) XCTAssertNotNil(response?.response) XCTAssertNil(response?.fileURL) XCTAssertNotNil(response?.error) XCTAssertNotNil(response?.resumeData) XCTAssertNotNil(download.resumeData) XCTAssertEqual(response?.resumeData, download.resumeData) } func testThatCancelledDownloadResumeDataIsAvailableWithDecodableResponseSerializer() { // Given let expectation = self.expectation(description: "Download should be cancelled") var cancelled = false var response: DownloadResponse? // When let download = AF.download(.download()) download.downloadProgress { [unowned download] progress in guard !cancelled else { return } if progress.fractionCompleted > 0.1 { download.cancel(producingResumeData: true) cancelled = true } } download.responseDecodable(of: TestResponse.self) { resp in response = resp expectation.fulfill() } waitForExpectations(timeout: timeout) // Then XCTAssertNotNil(response?.request) XCTAssertNotNil(response?.response) XCTAssertNil(response?.fileURL) XCTAssertEqual(response?.result.isFailure, true) XCTAssertNotNil(response?.result.failure) XCTAssertNotNil(response?.resumeData) XCTAssertNotNil(download.resumeData) XCTAssertEqual(response?.resumeData, download.resumeData) } func testThatCancelledDownloadCanBeResumedWithResumeData() { // Given let expectation1 = expectation(description: "Download should be cancelled") var cancelled = false var response1: DownloadResponse? // When let download = AF.download(.download()) download.downloadProgress { [unowned download] progress in guard !cancelled else { return } if progress.fractionCompleted > 0.1 { download.cancel(producingResumeData: true) cancelled = true } } download.responseData { resp in response1 = resp expectation1.fulfill() } waitForExpectations(timeout: timeout) guard let resumeData = download.resumeData else { XCTFail("resumeData should not be nil") return } let expectation2 = expectation(description: "Download should complete") var progressValues: [Double] = [] var response2: DownloadResponse? AF.download(resumingWith: resumeData) .downloadProgress { progress in progressValues.append(progress.fractionCompleted) } .responseData { resp in response2 = resp expectation2.fulfill() } waitForExpectations(timeout: timeout) // Then XCTAssertNotNil(response1?.request) XCTAssertNotNil(response1?.response) XCTAssertNil(response1?.fileURL) XCTAssertEqual(response1?.result.isFailure, true) XCTAssertNotNil(response1?.result.failure) XCTAssertNotNil(response2?.response) XCTAssertNotNil(response2?.fileURL) XCTAssertEqual(response2?.result.isSuccess, true) XCTAssertNil(response2?.result.failure) progressValues.forEach { XCTAssertGreaterThanOrEqual($0, 0.1) } } func testThatCancelledDownloadProducesMatchingResumeData() { // Given let expectation = self.expectation(description: "Download should be cancelled") var cancelled = false var receivedResumeData: Data? var response: DownloadResponse? // When let download = AF.download(.download()) download.downloadProgress { [unowned download] progress in guard !cancelled else { return } if progress.fractionCompleted > 0.1 { download.cancel { receivedResumeData = $0 } cancelled = true } } download.response { resp in response = resp expectation.fulfill() } waitForExpectations(timeout: timeout) // Then XCTAssertNotNil(response?.request) XCTAssertNotNil(response?.response) XCTAssertNil(response?.fileURL) XCTAssertNotNil(response?.error) XCTAssertNotNil(response?.resumeData) XCTAssertNotNil(download.resumeData) XCTAssertEqual(response?.resumeData, download.resumeData) XCTAssertEqual(response?.resumeData, receivedResumeData) XCTAssertEqual(download.resumeData, receivedResumeData) } } // MARK: - final class DownloadResponseMapTestCase: BaseTestCase { func testThatMapTransformsSuccessValue() { // Given let expectation = self.expectation(description: "request should succeed") var response: DownloadResponse? // When AF.download(.get, parameters: ["foo": "bar"]).responseDecodable(of: TestResponse.self) { resp in response = resp.map { response in response.args?["foo"] ?? "invalid" } expectation.fulfill() } waitForExpectations(timeout: timeout) // Then XCTAssertNotNil(response?.request) XCTAssertNotNil(response?.response) XCTAssertNotNil(response?.fileURL) XCTAssertNil(response?.resumeData) XCTAssertNil(response?.error) XCTAssertEqual(response?.result.success, "bar") XCTAssertNotNil(response?.metrics) } func testThatMapPreservesFailureError() { // Given let urlString = String.invalidURL let expectation = self.expectation(description: "request should fail with invalid URL") var response: DownloadResponse? // When AF.download(urlString, parameters: ["foo": "bar"]).responseDecodable(of: TestResponse.self) { resp in response = resp.map { _ in "ignored" } expectation.fulfill() } waitForExpectations(timeout: timeout) // Then XCTAssertNotNil(response?.request) XCTAssertNil(response?.response) XCTAssertNil(response?.fileURL) XCTAssertNil(response?.resumeData) XCTAssertNotNil(response?.error) XCTAssertEqual(response?.result.isFailure, true) XCTAssertNotNil(response?.metrics) } } // MARK: - final class DownloadResponseTryMapTestCase: BaseTestCase { func testThatTryMapTransformsSuccessValue() { // Given let expectation = self.expectation(description: "request should succeed") var response: DownloadResponse? // When AF.download(.get, parameters: ["foo": "bar"]).responseDecodable(of: TestResponse.self) { resp in response = resp.tryMap { response in response.args?["foo"] ?? "invalid" } expectation.fulfill() } waitForExpectations(timeout: timeout) // Then XCTAssertNotNil(response?.request) XCTAssertNotNil(response?.response) XCTAssertNotNil(response?.fileURL) XCTAssertNil(response?.resumeData) XCTAssertNil(response?.error) XCTAssertEqual(response?.result.success, "bar") XCTAssertNotNil(response?.metrics) } func testThatTryMapCatchesTransformationError() { // Given struct TransformError: Error {} let expectation = self.expectation(description: "request should succeed") var response: DownloadResponse? // When AF.download(.get, parameters: ["foo": "bar"]).responseDecodable(of: TestResponse.self) { resp in response = resp.tryMap { _ in throw TransformError() } expectation.fulfill() } waitForExpectations(timeout: timeout) // Then XCTAssertNotNil(response?.request) XCTAssertNotNil(response?.response) XCTAssertNotNil(response?.fileURL) XCTAssertNil(response?.resumeData) if let error = response?.result.failure { XCTAssertTrue(error is TransformError) } else { XCTFail("flatMap should catch the transformation error") } XCTAssertNotNil(response?.metrics) } func testThatTryMapPreservesFailureError() { // Given let urlString = String.invalidURL let expectation = self.expectation(description: "request should fail with 404") var response: DownloadResponse? // When AF.download(urlString, parameters: ["foo": "bar"]).responseDecodable(of: TestResponse.self) { resp in response = resp.tryMap { _ in "ignored" } expectation.fulfill() } waitForExpectations(timeout: timeout) // Then XCTAssertNotNil(response?.request) XCTAssertNil(response?.response) XCTAssertNil(response?.fileURL) XCTAssertNil(response?.resumeData) XCTAssertNotNil(response?.error) XCTAssertEqual(response?.result.isFailure, true) XCTAssertNotNil(response?.metrics) } } final class DownloadResponseMapErrorTestCase: BaseTestCase { func testThatMapErrorTransformsFailureValue() { // Given let urlString = String.invalidURL let expectation = self.expectation(description: "request should not succeed") var response: DownloadResponse? // When AF.download(urlString).responseDecodable(of: TestResponse.self) { resp in response = resp.mapError { error in TestError.error(error: error) } expectation.fulfill() } waitForExpectations(timeout: timeout) // Then XCTAssertNotNil(response?.request) XCTAssertNil(response?.response) XCTAssertNil(response?.fileURL) XCTAssertNil(response?.resumeData) XCTAssertNotNil(response?.error) XCTAssertEqual(response?.result.isFailure, true) guard let error = response?.error, case .error = error else { XCTFail(); return } XCTAssertNotNil(response?.metrics) } func testThatMapErrorPreservesSuccessValue() { // Given let expectation = self.expectation(description: "request should succeed") var response: DownloadResponse? // When AF.download(.get).responseData { resp in response = resp.mapError { TestError.error(error: $0) } expectation.fulfill() } waitForExpectations(timeout: timeout) // Then XCTAssertNotNil(response?.request) XCTAssertNotNil(response?.response) XCTAssertNotNil(response?.fileURL) XCTAssertNil(response?.resumeData) XCTAssertEqual(response?.result.isSuccess, true) XCTAssertNotNil(response?.metrics) } } // MARK: - final class DownloadResponseTryMapErrorTestCase: BaseTestCase { func testThatTryMapErrorPreservesSuccessValue() { // Given let expectation = self.expectation(description: "request should succeed") var response: DownloadResponse? // When AF.download(.get).responseData { resp in response = resp.tryMapError { TestError.error(error: $0) } expectation.fulfill() } waitForExpectations(timeout: timeout) // Then XCTAssertNotNil(response?.request) XCTAssertNotNil(response?.response) XCTAssertNotNil(response?.fileURL) XCTAssertNil(response?.resumeData) XCTAssertNil(response?.error) XCTAssertEqual(response?.result.isSuccess, true) XCTAssertNotNil(response?.metrics) } func testThatTryMapErrorCatchesTransformationError() { // Given let urlString = String.invalidURL let expectation = self.expectation(description: "request should fail") var response: DownloadResponse? // When AF.download(urlString).responseData { resp in response = resp.tryMapError { _ in try TransformationError.error.alwaysFails() } expectation.fulfill() } waitForExpectations(timeout: timeout) // Then XCTAssertNotNil(response?.request) XCTAssertNil(response?.response) XCTAssertNil(response?.fileURL) XCTAssertNil(response?.resumeData) XCTAssertNotNil(response?.error) XCTAssertEqual(response?.result.isFailure, true) if let error = response?.result.failure { XCTAssertTrue(error is TransformationError) } else { XCTFail("flatMapError should catch the transformation error") } XCTAssertNotNil(response?.metrics) } func testThatTryMapErrorTransformsError() { // Given let urlString = String.invalidURL let expectation = self.expectation(description: "request should fail") var response: DownloadResponse? // When AF.download(urlString).responseData { resp in response = resp.tryMapError { TestError.error(error: $0) } expectation.fulfill() } waitForExpectations(timeout: timeout) // Then XCTAssertNotNil(response?.request) XCTAssertNil(response?.response) XCTAssertNil(response?.fileURL) XCTAssertNil(response?.resumeData) XCTAssertNotNil(response?.error) XCTAssertEqual(response?.result.isFailure, true) guard let error = response?.error, case TestError.error = error else { XCTFail(); return } XCTAssertNotNil(response?.metrics) } }